首页 > 代码库 > mfc 按钮自绘

mfc 按钮自绘

MFC  按钮自绘

:songyanwu

            如果你是大神就没必要看这个文章了!


说明

源码下载:mfc 按钮自绘


先说说自己的一些想法:我就想把按钮封装成一个类,每次在使用的时候会很方便,当然在自己的类中去重载也可以!


此文章可借鉴学习:MFC基础,MFC自绘控件学习总结. (我也主要研究了自绘控件的子类化方法  ),看完前面推荐的文章,你似乎有何种感觉呢?

 

先实际操作吧;原理在后面介绍:


1 新建一个对画框 应用程序

2 新添加一个CMyButton继承CButton

3 为你自己添加的类 添加消息WM_LBUTTONDOWN,WM_LBUTTONUP消息,并重写DrawItem虚函数(函数从绘调用),

4 在你添加的类中 添加Bool  变量 m_BtnDown来判断按钮的状态

5 添加消息代码如下

[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. <span style="font-size:14px;">void CMyButton::OnLButtonDown(UINT nFlags, CPoint point)  
  2. {  
  3.     // TODO: 在此添加消息处理程序代码和/或调用默认值  
  4.     m_BtnDown = true;  
  5.     Invalidate();  
  6.     CButton::OnLButtonDown(nFlags, point);  
  7. }</span>  


[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. <span style="font-size:14px;">void CMyButton::OnLButtonUp(UINT nFlags, CPoint point)  
  2. {  
  3.     // TODO: 在此添加消息处理程序代码和/或调用默认值  
  4.     m_BtnDown=false;  
  5.   
  6.     Invalidate();  
  7.   
  8.     CButton::OnLButtonUp(nFlags, point);  
  9. }</span>  

注:Invalidate(BOOL bErase=TRUE)函数的作用是使整个窗口的客户区无效。客户区无效就意味着需要重绘。例如,如果一个被其它窗口遮住的窗口变成了前台窗口,那么原来被遮住的部分就是无效的,需要重绘。这时Windows会在应用程序的消息队列中放置WM_PAINT消息。MFC为窗口类提供了WM_PAINT的消息处理函数OnPaintOnPaint负责重绘窗口。视图类有一些例外,在视图类的OnPaint函数中调用了OnDraw函数,实际的重绘工作由OnDraw来完成。参数bEraseTRUE时,重绘区域内的背景将被擦除,否则,背景将保持不变。

             UpdateWindow的区别是UpdateWindow是使窗口立即重绘。Invalidate只是发送一个WM_PAINT消息,WM_PAINT消息的优先级很低,需要等到消息队列中其他消息处理完成后才能被处理,所以窗口不会立即刷新。

             窗口重绘时会重新生成按钮,按钮生成时会查看其风格,如果为OwnerDraw风格,则会调用DrawItem函数。关于OwnerDraw风格的设置,后边会讲。


6 重载DramItem

 

[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. <span style="font-size:14px;">void CMyButton::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct/*lpDrawItemStruct*/)  
  2. {  
  3.   
  4.     // TODO:  添加您的代码以绘制指定项  
  5.     //通过DRAWITEMSTRUCT结构获取句柄,并且类型转换,这个FromHandle很常用   
  6.   
  7.     CDC* pDC=CDC::FromHandle(lpDrawItemStruct->hDC);    
  8.   
  9.     //下面两行来获取这个结构中的一个字段   
  10.   
  11.     UINT state=lpDrawItemStruct->itemState;//按钮的状态   
  12.   
  13.     CRect rect=lpDrawItemStruct->rcItem;//重绘的范围,也就是按钮   
  14.   
  15.     //获取按钮上显示的文字  
  16.   
  17.     CString str;    
  18.   
  19.     GetWindowText(str);    
  20.   
  21.     if (state & ODS_SELECTED)//如果按钮被选中   
  22.   
  23.     {           
  24.   
  25.         //绘制一个控件,并制定控件的类型和状态  
  26.   
  27.         pDC->DrawFrameControl(&rect,DFC_BUTTON,DFCS_BUTTONPUSH | DFCS_PUSHED);    
  28.   
  29.     }    
  30.   
  31.     else    
  32.   
  33.     {    
  34.   
  35.         pDC->DrawFrameControl(&rect,DFC_BUTTON,DFCS_BUTTONPUSH);    
  36.   
  37.     }     
  38.   
  39.     if (m_BtnDown)    
  40.   
  41.     {       //填充矩形  
  42.   
  43.         pDC->FillSolidRect(rect,RGB(255,255,0));    
  44.   
  45.     }    
  46.   
  47.     else    
  48.   
  49.     {    
  50.   
  51.         pDC->FillSolidRect(rect,RGB(0,0,255));    
  52.   
  53.     }     
  54.   
  55.     if (!str.IsEmpty())    
  56.   
  57.     {        
  58.   
  59.         //以下是当按钮按下时要稍微移动按钮上文字的位置   
  60.   
  61.         //使用当前字体计算一行文字的宽度和长度以确定尺寸  
  62.   
  63.         CSize Extent=pDC->GetTextExtent(str);    
  64.   
  65.         CPoint pt=CPoint(rect.CenterPoint().x-Extent.cx/2,rect.CenterPoint().y-Extent.cy/2);    
  66.   
  67.         //如果被选中  
  68.   
  69.         if (state & ODS_SELECTED)    
  70.   
  71.         {    
  72.   
  73.             //向一个点中添加x,y坐标  
  74.   
  75.             pt.Offset(1,1);    
  76.   
  77.         }    
  78.   
  79.         //设置背景样式,值有两个:  
  80.   
  81.         //OPAQUE   在文字,画刷,铅笔绘画之前,背景由当前背景颜色填充。这时默认的背景模式。  
  82.   
  83.         //TRANSPARENT   在绘制前背景不变  
  84.   
  85.         int nMode=pDC->SetBkMode(TRANSPARENT);    
  86.   
  87.         if (state & ODS_DISABLED)    
  88.   
  89.         {    
  90.   
  91.             pDC->DrawState(pt,Extent,str,DSS_DISABLED,TRUE,0,(HBRUSH)NULL);    
  92.   
  93.         }    
  94.   
  95.         else    
  96.   
  97.         {    
  98.   
  99.             pDC->TextOut(pt.x,pt.y,str);    
  100.   
  101.         }    
  102.   
  103.         pDC->SetBkMode(nMode);    
  104.   
  105.     }    
  106.   
  107.   
  108. }  
  109. </span>  

 

注:

1. DrawItem函数用于重绘单个控件。其参数为指向DRAWITEMSTRUCT的一个长指针。

DRAWITEMSTRUCT为需要自绘的控件或者菜单项提供了必要的信息。在需要绘制的控件或者菜单项对应的WM_DRAWITEM消息函数中得到一个指向该结构的指针。

2

[cpp] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. <span style="font-size:14px;"> typedef struct tagDRAWITEMSTRUCT {  
  2.   
  3.          UINT CtlType;  
  4.   
  5.          UINT CtlID;  
  6.   
  7.            UINT itemID;  
  8.   
  9.          UINT itemAction;  
  10.   
  11.          UINT itemState;  
  12.   
  13.          HWND hwndItem;  
  14.   
  15.          HDC hDC;  
  16.   
  17.          RECT rcItem;  
  18.   
  19.          DWORD itemData;  
  20.   
  21. } DRAWITEMSTRUCT;  
  22.   
  23. </span><span style="font-size:14px;">  
  24. </span>  




CtlType:指定参数类型。例:ODT_BUTTONODT_COMBOBOXODT_LISTBOX等。

CtlIDcombo box, list box, 或者 button的控件ID。菜单项不需要此参数

itemID:菜单项的ID,也可以表示列表框或者组合框中某项的索引值。对于一个空的列表框或组合框此值为-1.这时应用程序只绘制焦点矩形(该矩形的坐标有rcItem成员给出)。虽然此时控件没有需要显示的项,但绘制焦点矩形还是很有必要的。因为这样能够提示用户该控件是否具有输入焦点。当然也可设置itemAction的值,使得无需绘制焦点。

itemAction:指定绘制行为,可取多值。ODA_DRAWENTIRE:当整个控件都需要被绘制时,设置该值。ODA_FOCUS:控件在获得或是去焦点时被绘制。此时应该检查itemState成员,以确定控件是否具有输入焦点。ODA_SELECT:控件在选中状态改变时被重绘。此时应检查itemState成员,以确定控件是否被选中。

itemState:指定当前绘制操作完成后,所绘项的可见状态。

ODS_CHECKED:菜单项被选中。该值只对菜单项有用。

             ODS_COMBOBOXEDIT:在自绘组合框控件中只绘制选择区域。

             ODS_DEFAULT:默认值。

             ODS_DISABLED:控件被禁止,则设置该值。

             ODS_FOCUS:控件需要输入焦点,则设置该值。

             ODS_GRAYED:控件被灰色显示。该值只在绘制菜单时使用。

             ODS_HOTLIGHTWindows XP: 如果鼠标指针位于控件之上,则设置该值,这时控件会显示高亮颜色。

       ODS_INACTIVE Windows XP: 表示没有激活的菜单项。

             ODS_NOACCELWindows XP: 控件是否有快速键盘。

             ODS_NOFOCUSRECTWindows XP,不绘制捕获焦点的效果。

             ODS_SELECTED:选中的菜单项。

hwndItem:指定了组合框、列表框和按钮等自绘控件的窗口句柄;如果自绘的对象是菜单项,则表示包含该菜单项的菜单句柄。

hDC:指定了绘制操作所使用的设备环境。

rcItem:指定了将被绘制的矩形区域。这个矩形区域就是上面hDC的作用范围。系统会自动裁剪组合框、列表框或按钮等控件的自绘制区域以外的部分。也就是说rcItem中的坐标点(00)指的就是控件的左上角。但是系统不裁剪菜单项,所以在绘制菜单项的时候,必须先通过一定的换算得到该菜单项的位置,以保证绘制操作在我们希望的区域中进行.

3.       DC(设备上下文):设备上下文是一种包含有关某个设备(如显示器或打印机)的绘制属性信息的 Windows 数据结构。所有绘制调用都通过设备上下文对象进行,这些对象封装了用于绘制线条、形状和文本的 Windows API设备上下文允许在 Windows 中进行与设备无关的绘制。设备上下文可用于绘制到屏幕、打印机或者图元文件

hDCDC的一个DC的句柄。CDC是所有DC的基类。

4.       FromHandleHANDLE h)从网上查找的解释是,先进行查找,如果h与某个CWnd(窗口)对象相关联,就返回此CWnd对象的指针;如果h没有与任何CWnd对象关联,则new一个新的CWnd对象返回。

7.      自定义类与标准控件的关联

方法一:设置按钮的属性为Owner Drawn.

方法二:在程序中,重载PreSubClassWindow。此函数在CWnd::CreateDDX_Contorl之后调用,即在窗口子类化创建后和窗口显示前调用。在此函数中添加ModifyStyle(0,BS_OWNERDRAW);此函数用于改变控件的类型。

8.      创建子类化的自绘按钮

   方法一、在创建按钮时可以给这个按钮绑定一个CMyButton类对象,但是这个时候在classwizard中还不能看到这个类:  可以删掉当前目录下的clw文件,然后重新打开程序,按下ctrl+w,然后一般选择add all,点击ok,这个时候再添加类对象,就会发现这个类了

方法二、上述方法不适合动态添加,下面用程序的方法动态添加

在窗口中拖入一个Button控件。在Dlg的头文件中加入成员函数CMyButton m_MyButton(注意要加入头文件)。然后在此窗口的OnInitDialog函数中调用m_MyButton.SubClassDlgItem(IDC_BUTTON1,this).此函数可以动态的子类化一个由此窗口创建的控件并将此控件关联至此CWnd.

如果你已经有了一个窗口的指针,或者你的工作在一个CView或者其他CWnd的派生类中且里面的控件被动态创建,或者你不想用上边的函数,你可以使用下述方法。

CWnd* pWnd = GetDlgItem(IDC_BUTTON1); 
m_btnMyButton.SubclassWindow(pWnd->GetSafeHwnd());

mfc 按钮自绘