关于CEdit控件的透明(重绘)

摘自:http://www.jcwcn.com/html/VC/10_19_51_12.htm

做一个透明的Edit控件的主要问题是字符的输出,在Edit里输出的刷新有几个时机,一个是在接收到键盘或鼠标消息的时候 ,还有就是在接收到WM_PAINT消息。刷新的时候也不是全部重画,所以想通过在继承的Edit类中处理WM_PAINT消息是行不通的。但是Edit控件自己总是知道怎么去刷新,因此只要给控件发消息,让其自己来刷新就可以了。通过使用spy++的得知需要刷新有几个时机,一个是按键的时候,内容变化,另一个是选择变化的时候,前者Edit控件会接收到GetCtlCode和KeyUp 消息,后者会接收到GetCtlCode和CaptureChange消息或KeyUp消息,因此在GetCtlCode里调用ReDrawWindow来强迫Edit刷新 整个控件。在ReDrawWindow中通过使用参数RDW_ERASE可以使控件重画背景,即调用OnEraseBkgnd(CDC* pDC),在该函数中重 画背景。比较特殊的情况是按住鼠标左键并来回拖动鼠标的时候,这时候选择要改变,接收的消息是MouseMove,为了正 确响应也要处理该消息,但是在每一个MouseMove中都刷新显示的开销太大,而且不可避免地有闪烁感,因此只有在鼠标左 键按下的时候才刷新显示。
大概的代码如下,主要是继承了一个CEdit的对象CTpEdit,使用的时候可以动态创建,或者采用SubClass的方法。我用的是后者。

[cpp] view plain copy

print?

  1. class CTESTDLG : public CDialog
  2. {
  3. ......
  4. //声明一个CTpEdit的成员变量
  5. private:
  6. CTpEdit m_tpedit;
  7. };
  8. //在OnInitDialog中Subclass对话框模板中的Edit控件
  9. BOOL CTESTDLG::OnInitDialog()
  10. {
  11. CDialog::OnInitDialog();
  12. m_tpedit.SubclassDlgItem(IDC_EDIT,this);
  13. return TRUE;
  14. }
  15. //在OnCtlColor中设置背景的透明,要改变Edit控件字体的颜色也在这里
  16. HBRUSH CTESTDLG::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
  17. {
  18. HBRUSH hbr = CDialog::OnCtlColor(pDC, pWnd, nCtlColor);
  19. if((nCtlColor == CTLCOLOR_EDIT) && (pWnd->GetDlgCtrlID()==IDC_EDIT))
  20. {
  21. pDC->SetBkMode(TRANSPARENT); //设置背景透明,这样,输出字符的时候就
  22. //是所谓的空心字,而不是有白的底色
  23. pDC->SetTextColor(RGB(255,0,0)); //改变字体的颜色
  24. return HBRUSH(GetStockObject(HOLLOW_BRUSH));
  25. }
  26. return hbr;
  27. }
  28. //CTpEdit对象
  29. #ifndef _CCOLOR_EDIT_H_
  30. #define _CCOLOR_EDIT_H_
  31. class CTpEdit : public CEdit
  32. {
  33. public:
  34. //m_mousedown用来记录鼠标左键是否按下
  35. BOOL m_mousedown;
  36. protected:
  37. //响应如下的消息
  38. //{{AFX_MSG(CTpEdit)
  39. afx_msg BOOL OnEraseBkgnd(CDC* pDC);
  40. afx_msg void OnMouseMove(UINT nFlags, CPoint point);
  41. afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
  42. afx_msg void OnLButtonUp(UINT nFlags, CPoint point);
  43. afx_msg UINT OnGetDlgCode();
  44. //}}AFX_MSG
  45. DECLARE_MESSAGE_MAP()
  46. };
  47. //{{AFX_INSERT_LOCATION}}
  48. #endif
  49. //CTpEdit的消息响应函数如下
  50. #include "stdafx.h"
  51. #include "ColorEdit.h"
  52. //画背景图
  53. BOOL CTpEdit::OnEraseBkgnd(CDC* pDC)
  54. {
  55. //得到Edit控件的外框,即背景区域
  56. RECT updatarect;
  57. GetClientRect(&updatarect);
  58. //画背景,我画的是一个黄色的矩形
  59. CBrush newBrush;
  60. newBrush.CreateSolidBrush(RGB(255,255,200));
  61. CBrush * oldBrush = pDC->SelectObject(&newBrush);
  62. pDC->Rectangle(&updatarect);
  63. pDC->SelectObject(oldBrush);
  64. return TRUE;
  65. }
  66. //强迫Edit控件擦除背景,重写字符
  67. UINT CTpEdit::OnGetDlgCode()
  68. { RedrawWindow(NULL, NULL,RDW_INVALIDATE | RDW_ERASE );
  69. return CEdit::OnGetDlgCode();
  70. }
  71. //记录鼠标左键是否按下
  72. void CTpEdit::OnLButtonDown(UINT nFlags, CPoint point)
  73. {
  74. m_mousedown = TRUE;
  75. SetCapture();
  76. CEdit::OnLButtonDown(nFlags, point);
  77. }
  78. void CTpEdit::OnLButtonUp(UINT nFlags, CPoint point)
  79. {
  80. if(m_mousedown)
  81. ReleaseCapture();
  82. m_mousedown = FALSE;
  83. CEdit::OnLButtonUp(nFlags, point);
  84. }
  85. //如果左键按下并且拖动鼠标就要刷新显示
  86. void CTpEdit::OnMouseMove(UINT nFlags, CPoint point)
  87. {
  88. if(m_mousedown)
  89. RedrawWindow(NULL, NULL,RDW_INVALIDATE | RDW_ERASE );
  90. CEdit::OnMouseMove(nFlags, point);
  91. }
  92. //初始化成员变量
  93. CTpEdit::CTpEdit()
  94. {
  95. m_mousedown=FALSE;
  96. }

http://blog.csdn.net/witch_soya/article/details/6889970

时间: 2025-01-01 21:13:25

关于CEdit控件的透明(重绘)的相关文章

窗口使用半透明图片,控件不透明的方法

最近项目软件需要制作半透明窗口,但控件不能透明.窗口和控件都是自绘的.网上大部分例子都是直接设置窗口透明度实现,这样只能让整个窗口透明,达不到我们想要的效果.经研究发现,需要使用父子两个窗口配合制作,父窗口不能在OnPaint那里自绘,而是直接GetDC得到句柄自绘背景,子窗口负责自绘控件. 关键代码:父窗口 OnInitDialog() { m_dlg.Create(m_hWnd); m_dlg.ShowWindow(SW_SHOW); m_dlg.MoveWindow(&rct); HDC

MFC 加入背景图片并让控件背景透明

/*加入背景图片*/ BOOL CTOOLDlg::OnEraseBkgnd(CDC* pDC) { // TODO: 在此加入消息处理程序代码和/或调用默认值 CDialog::OnEraseBkgnd(pDC); HBITMAP   m_hBitmap; HDC           m_hBkDC; m_hBitmap   =   ::LoadBitmap(::GetModuleHandle(NULL),MAKEINTRESOURCE(IDB_BITMAP2)); m_hBkDC     =

Android给TextView和EditText等控件设置透明背景、圆角边框

第一种方法:在drawable文件夹下新建一个文件设置背景样式 代码: 在drawable文件夹下面新建textviewborder.xml <?xml version="1.0" encoding="UTF-8"?> <shape xmlns:android="http://schemas.android.com/apk/res/android"> <solid android:color="#80858

实现控件的透明背景

很多情况下,我们需要控件的背景是透明的,就是要求直接看到控件父窗口的背景颜色.背景位图,比如标签控件.单选Radio控件.复选Check控件,通常都要求在父窗口的背景上进行绘制.然而要求控件的画布透明,这个技术在GDI的文档中没有看到Microsoft作任何说明,当然还是有别的办法. 其一:如果程序支持桌面主题服务的话,则可调用主题服务的API来实现背景.我们先看看这个API: HRESULT DrawThemeParentBackground(HWND hwnd, HDC hdc, RECT 

iOS 设置视图半透明而子控件不透明

转载:http://www.jianshu.com/p/abe815018b2a 让一个控件半透明,通常我们第一个想到的方法就是调整控件的alpha值.当然如果你的视图上如果有其他子控件就会发现这样的方法是可能达不到你的预期,否则你应该看不到这篇文章了. 对视图直接设置alpha属性的值会导致其子控件也变得半透明,而通常我们的需求是:背景半透明而其子控件不透明. 先说解决办法: 1.用一张半透明的图片做背景. 这个方法当然可以达到要求,但是从编程初期前辈就教导我说在工程中尽量少加入资源,能不用图

VC窗体透明而控件不透明以及Static文本背景透明方法

出自http://my.oschina.net/ypimgt/blog/60951 优点:    1.Dialog 窗体完全透明.     2. 窗体上的控件不透明. DC 绘制的图形不透明.     3. 拖动窗体上用 DC 绘制的图形可以移动窗体. 缺点:     1. 窗体设置透明使用是掩码颜色,所以在窗体上用 DC 绘图的过程当中如果采用了和透明掩码颜色相同的颜色将不会显示出来.     2. 如果将 Border 属性设置成了 NONE ,那么想通过响应 WM_NCHITEST 消息来

MFC输入框CEdit控件十六进制转换

做MFC串口发送的时候,一般使用CEdit控件作为发送输入框,一般是发送输入框关联一个CString的变量,通过updatedata更新,大部分情况下我们需要使用十六进制发送,但是CString默认是字符串格式,比如输入框输入11 22 33 44 55,实际上是内容"11 22 33 44 AA",我们希望将它转换为0x11,0x22,0x33,0x44,0xaa. 那么如何实现了,通过下列的2个函数实现 1 //将一个字符串作为十六进制串转化为一个字节数组,字节间可用空格分隔, 2

C#控件背景透明的几种解决方案

已经很少做winform程序了,最新参与了一个小项目,遇到了控件背景透明的功能要求,特在此总结一下,供有需要的同行参考. 0.背景透明的概念和分类 背景透明是啥意思呢,就是背景透明.哈哈,废话了.其实你想过没有,要求不一样,实现的难道和技术手段也不一样. 最基本的就是不显示控件自己的背景,那,那显示谁的背景? 背景透明后,透过去显示谁,也就是说后面的谁不透明,这是问题的关键,换句话说,透明控件的parent是一个还是多个. 1.简单的背景透明,parent是一个的情况 a.自然透明,如图labe

MFC对话框的CEdit控件回车换Tab键,并获得焦点全选中文本

对话框上有五个CEdit和两个Button控件 //重写PreTranslateMessage函数 BOOL CAddDlg::PreTranslateMessage(MSG* pMsg) { // TODO: 在此添加专用代码和/或调用基类 if (pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_RETURN) { CWnd pwnd = GetFocus(); if (pwnd != NULL) { //判断是否是按钮控件