本文共 3474 字,大约阅读时间需要 11 分钟。
1:第一次尝试
设置 IDC_BUTTON3按钮风格的bitmap为true
在OnInitialDilog中:
CButton* cbpTest = NULL;
HINSTANCE hInstance=AfxGetResourceHandle();
HBITMAP m_hBmp4=(HBITMAP)LoadImage(hInstance, MAKEINTRESOURCE(IDB_BITMAP1), IMAGE_BITMAP, 0, 0, 0); cbpTest = (CButton*)GetDlgItem(IDC_BUTTON3);::PostMessage(cbpTest->GetSafeHwnd(), BM_SETIMAGE, IMAGE_BITMAP, (LPARAM)m_hBmp4);
覆盖掉了文字,并且也有闪屏
2:第二次尝试:https://wenku.baidu.com/view/95f2be7a1711cc7931b716e2.html
2.1建立基于对话框的MFC,并且将 IDC_BUTTON3按钮风格的ower-draw打钩:
2.2 声明CBitamapButton m_btnBMP;
2.3 在对话框类CtestDlg的构造函数中m_btnBMP.loadbitmaps(IDB_BITMAP1,IDB_BITMAP2,IDB_BITMAP3,IDB_BITMAP4);
2.4 然后在OnInitialDilog中:m_btnBMP.SubClassDlgItem(IDC_BUTTON3,this);m_btnBMP.SizeToContent();
缺点:覆盖掉了文字,并且也有闪屏
由于MFC提供的按钮外观不是很漂亮(其实是很丑),所以使用按钮的时候往往要自己重新绘制,最多的就是添加自己的漂亮图片。
1.CButton中提供了SetBitmap函数,可以给按钮设置位图,使用方法如下: CButton button; button.Create(_T("My button"), WS_CHILD|WS_VISIBLE|BS_PUSHBUTTON, CRect(10,10,100,30),this, 1); CBitmap bmp; bmp.LoadBitmap(IDB_BMP); button.SetBitmap(bmp); 优点:使用方法简单 缺点:位图单一,不能显示区分按钮各种状态,有闪屏情况 2.使用MS做好的CBitmapButton类: CBitmapButton是MS封装好的位图按钮类,从CButton类继承而来,提供两种使用方法,但使用前要创建一个CButton对象与其关联,另外,CBitmapButton类必须具有BS_OWNERDRAW属性: <1> 使用LoadBitmaps方法: CBitmapButton bmpButton; bmpButton.Create(NULL, WS_CHILD | WS_VISIBLE | BS_OWNERDRAW, CRect(10, 10, 100, 100), this, BTN_ID); bmpButton.LoadBitmaps(IDB_UP, IDB_DOWN, IDB_FOCUS, IDB_DISABLE);<2>使用AutoLoad方法:
创建一个按钮资源ID为IDC_MYBUTTON并且具有BS_OWNERDRAW属性,设定Caption为:MYBUTTON,插入四张资源位图ID为"MYBUTTONU","MYBUTTOND","MYBUTTONF","MYBUTTONX",分别表示按钮的”弹起“,”按下“,”焦点“,”禁用“状态。最后调用AutoLoad函数就可以自动加载了: CBitmapButton bmpButton; bmpButton.AutoLoad(IDC_MYBUTTON, this);以上两种用法
优点:使用简单,很好的用位图表示了按钮的各种状态,丰富了按钮的外观 缺点:有闪屏情况MS真让人受伤啊,提供了位图按钮的使用,但就是不能用,都有闪屏。所以只有自己重新绘制了:
3.添加自定义类,从CButton继承,在类向导中添加虚函数DrawItem,然后重画:
//从lpDrawItemStruct获取控件的相关信息 CRect rect = lpDrawItemStruct->rcItem; CDC *pDC=CDC::FromHandle(lpDrawItemStruct->hDC); int nSaveDC=pDC->SaveDC(); UINT state = lpDrawItemStruct->itemState; if(state & ODS_SELECTED) { // 画按钮按下状态的时候需要显示的位图 DrawBmpState(pDC, m_bmpResDown); } else { // 画按钮其他状态的时候需要显示的位图 DrawBmpState(pDC, m_bmpResUp); } // 恢复DC pDC->RestoreDC(nSaveDC);注:DrawBmpState用来画按钮的各种状态,采用双缓冲技术,在PC上测试运行正常,但是在手持设备运行有闪屏情况。闪屏发生在按其他按钮的时候,由于这个按钮要丢失焦点,状态转换太快,处理器速度跟不上的原因。(这只是我的个人猜测,高人知道的请赐教)。或者是其他的原因。总之闪屏就像一个甩不掉的魔鬼一样如影随形,最后只能自己重画了。
4.在OnPaint中彻底重画,定制自己的位图按钮:
添加CBmpButton类继承自CButton: 添加BOOL m_bDown;初始化为FALSE,用来标识按钮的按下状态,当按钮按下是置为TRUE,否则为FALSE,主要是为了避免当鼠标在其他地方按下后,然后在按钮上释放造成的按钮重绘。 添加WM_PAINT消息重绘: CPaintDC dc(this); // device context for painting if(m_bDown) { DrawBmpState(&dc, m_bmpResDown); } else { DrawBmpState(&dc, m_bmpResUp); } DrawBmpState函数用来画按钮各个状态的位图,采用双缓冲技术避免了按钮重绘时的闪屏,这里只列出了两种:弹起和按下: VOID CBmpButton::DrawBmpState(CDC *pDC, HBITMAP hBitmap) { CRect rect; //GetWindowRect(&rect); GetClientRect(&rect); CDC MemDC,mdc; mdc.CreateCompatibleDC(NULL); MemDC.CreateCompatibleDC(NULL); CBitmap MemBitmap; MemBitmap.CreateCompatibleBitmap(pDC,rect.Width(),rect.Height()); MemDC.SelectObject(&MemBitmap); MemDC.FillSolidRect(0,0,rect.Width(),rect.Height(),RGB(0,0,0)); //b1.LoadBitmap(bmpID); //pb1 = CBitmap::FromHandle(hBitmap); mdc.SelectObject(hBitmap); MemDC.BitBlt(0,0,rect.Width(),rect.Height(),&mdc,0,0,SRCCOPY); pDC->BitBlt(0,0,rect.Width(),rect.Height(),&MemDC,0,0,SRCCOPY); //绘图完成后的清理 MemBitmap.DeleteObject(); MemDC.DeleteDC(); } 添加WM_LBUTTONDOWN事件: m_bDown = TRUE; InvalidateRect(NULL); 添加WM_LBUTTONUP事件: if(m_bDown) { m_bDown = FALSE; InvalidateRect(NULL); } 这样就避免了重绘按钮的闪屏,希望闪屏以后不要再来!转载地址:http://gimws.baihongyu.com/