且构网

分享程序员开发的那些事...
且构网 - 分享程序员编程开发的那些事

VC++界面编程之--使用分层窗口实现界面皮肤

更新时间:2022-09-10 15:44:37

VC++界面编程之--使用分层窗口实现界面皮肤

使用分层界面来实现界面皮肤的好处是:可以保证图片边缘处理不失真,且能用于异形窗口上,如一些不规则的窗口,你很难用SetWindowRgn来达到理想效果。

在很多情况下,界面的漂亮与否,取决于PS的制作及创意,而界面编程所需要做的就是将图片完整无缺的展示给用户。


分层窗口的好处就是:用一层窗口来描绘窗口背景,以确保背景图片不会失真,而另一层窗口用来放置控件,除控件之外的区域全是透明的。为什么要这么麻烦呢?因为我们将要用到UpdateLayeredWindow来实现窗口背景描绘,一旦执行这个函数后,窗口的控件将会被覆盖,不会再显示。所以我们要再加一层透明窗口,来放置控件,以确保控件不会受到UpdateLayeredWindow的影响。

这样一来,窗口绘图将不会影响到控件,并且可以达到窗口透明,控件不透明的效果。当然这个只是个别的窗口实现效果需求,这里不做重点讨论。

这里我用了两个窗口类来完成此项工作,CThemedLayerWnd和CThemedAlphaWnd。CThemedLayerWnd负责截取正常窗口的创建消息,并在正常窗口创建之前,创建CThemedAlphaWnd;CThemedAlphaWnd主要负责窗口的描绘,也就是使用UpdateLayeredWindow来让Windows自己描绘窗口皮肤。

正常窗口只需要继承模板类CThemedLayerWnd,并将CThemedLayerWnd加入到自己的窗口消息链中即可:

  1. class CMainDlg :    public CThemedLayerWnd<CMainDlg>,  
  2.                     public CUpdateUI<CMainDlg>,  
  3.                     public CMessageFilter,  
  4.                     public CIdleHandler  
  5. {  
  6.     BEGIN_MSG_MAP(CMainDlg)  
  7.         CHAIN_MSG_MAP(CThemedLayerWnd<CMainDlg>)  
  8.         REFLECT_NOTIFICATIONS()  
  9.     END_MSG_MAP()  
  10. };  

在正常窗口执行Create函数时,提前创建CThemedAlphaWnd,这样一来,CThemedAlphaWnd就是CThemedLayerWnd的父窗口,CThemedLayerWnd理所当然成为子窗口,并自动将窗口位置置于CThemedAlphaWnd之上,这也正是我们想要的效果。

  1. // Create normal dialog and alpha window.  
  2. HWND Create(const HWND wndParent, const int nImageID)  
  3. {  
  4.     // Create alpha window before normal dialog created.  
  5.     // Do not use WS_CHILD style, we need separated alpha window.  
  6.     m_wndAlpha = new CThemedAlphaWnd;  
  7.     m_wndAlpha->Create(wndParent, CRect(0, 0, 100, 100), NULL, WS_VISIBLE);  
  8.     ATLASSERT(m_wndAlpha->m_hWnd != NULL);  
  9.     SetBackgroundImage(nImageID);  
  10.   
  11.     // Create normal dialog.  
  12.     HWND hwnd = CDialogImpl<T>::Create(m_wndAlpha->m_hWnd);  
  13.   
  14.     // Pass normal window handle.  
  15.     m_wndAlpha->SetNormalWnd(m_hWnd);  
  16.   
  17.     // Set normal dialog style.  
  18.     UpdateWindowStyle();  
  19.     return hwnd;  
  20. }  

这里还有个关键的问题,窗口上面是有控件的,如果控件要获取父窗口的背景画刷该怎么办?而CThemedLayerWnd是透明的,你是没办法获取到透明背景的。所以需要截取CThemedLayerWnd的WM_PAINT消息,当控件想要通过WM_PAINT来获取窗口背景时,将窗口皮肤图片传递给控件。

  1. // Only for control get background image.  
  2. void DoPaint(CDCHandle dc)  
  3. {  
  4.     if (m_bmpBack.m_hBitmap != NULL)  
  5.     {  
  6.         HBITMAP hBmp = dc.SelectBitmap(m_bmpBack.m_hBitmap);  
  7.         ATLASSERT(hBmp != NULL);  
  8.     }  
  9. }  

分层窗口带来的不便之处就是:你对窗口的操作都是介于两个窗口之间的,所以你需要在此上面多花费一些功夫,比如移动窗口,实际是移动两个窗口。再比如正常窗口调用CenterWindow,实际是让CThemedLayerWnd来移动窗口,你不得不重写CenterWindow来先移动CThemedAlphaWnd然后再来移动CThemedLayerWnd。

凡事皆有取舍,一切都根据实际需求做出取舍。正如分层窗口一样,丧失了一些正常窗口的普通函数操作能力,但换来了界面皮肤的完美显示。


本文代码免费下载链接:http://download.csdn.net/detail/renstarone/6361907

 

 

from:http://blog.csdn.net/renstarone/article/details/12371309