Windows界面编程第五篇 静态控件背景透明化

 上一篇《Windows界面编程第三篇 异形窗体 普通版》和《Windows界面编程第四篇异形窗体 高富帅版》介绍了异形窗口(异形窗体)的创建,并总结出了异形窗口的“三要素”:

1.WS_EX_LAYERED属性

2.指定透明色

3.以位图为窗口背景

    本篇文章将主要介绍Windows编程中如何实现静态控件背景的透明化,这将进一步的美化界面。下面先看一张没有做静态控件背景透明化的对话框程序的运行画面,对话框的彩色图片背景可以参考《Windows界面编程第一篇位图背景与位图画刷》。

    可以看出静态控件的灰度背景在对话框的彩色图片背景中显得很不谐调,因此有必要对其美化一下。

    在第一篇《Windows界面编程第一篇位图背景与位图画刷》中介绍了通过WM_CTLCOLORDLG消息来设置对话框的背景,而在Windows系统中还有类似于WM_CTLCOLORDLG消息的还有WM_CTLCOLORBTNWM_CTLCOLOREDITWM_CTLCOLORLISTBOXWM_CTLCOLORSCROLLBARWM_CTLCOLORSTATIC这五种来分别管理按钮,编辑框,列表框,滚动条,静态框。因此我们首先在WM_CTLCOLORSTATIC消息中返回一个空画刷看看能不能达到背景透明化的要求。另外对于文字区域的背景透明可以通过SetBkMode来设置。其函数原型如下:

intSetBkMode(

   HDC hdc,     // handle to DC

   int iBkMode   // background mode

);

这个函数的第二个参数设置为TRANSPARENT时就可以将文字区域的前景设置成透明。

下面给出完整的源代码(下载地址:http://download.csdn.net/download/morewindows/4966826):

  1. // 静态控件背景透明化WM_CTLCOLORSTATIC中返回空画刷 
  2. //By MoreWindows-(http://blog.csdn.net/MoreWindows) 
  3. #include <windows.h> 
  4. #include "resource.h" 
  5.  
  6. constchar szDlgTitle[] = "静态控件背景透明化MoreWindows-(http://blog.csdn.net/MoreWindows)"
  7.  
  8. // 对话框消息处理函数 
  9. BOOL CALLBACK DlgProc(HWND hDlg,UINT message, WPARAM wParam,LPARAM lParam); 
  10.            
  11. int APIENTRY WinMain(HINSTANCE hInstance, 
  12.                      HINSTANCE hPrevInstance, 
  13.                     LPSTR     lpCmdLine, 
  14.                      int       nCmdShow) 
  15.     DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG1), NULL, DlgProc); 
  16.     return 0; 
  17.  
  18.  
  19. BOOL CALLBACK DlgProc(HWND hDlg,UINT message, WPARAM wParam,LPARAM lParam) 
  20.     RECT       rcDialog; 
  21.     HBITMAP    hBitmap; 
  22.     static BITMAP s_bm; 
  23.     staticHDC    s_hdcMem; 
  24.  
  25.     switch (message) 
  26.     { 
  27.     case WM_INITDIALOG: 
  28.         // 设置对话框标题 
  29.         SetWindowText(hDlg, szDlgTitle); 
  30.         // 设置对话框大小可调节 
  31.         SetWindowLong(hDlg, GWL_STYLE, GetWindowLong(hDlg, GWL_STYLE) | WS_SIZEBOX); 
  32.  
  33.         // 加载背影图片 
  34.         hBitmap = (HBITMAP)LoadImage(NULL,"006.bmp", IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION); 
  35.         if (hBitmap == NULL) 
  36.         { 
  37.             MessageBox(hDlg, "LoadImage failed", "Error", MB_ICONERROR); 
  38.             exit(0); 
  39.         }        
  40.         else 
  41.         { 
  42.             // 将背影图片放入HDC - s_hdcMem 
  43.             HDC        hdc; 
  44.             hdc = GetDC(hDlg); 
  45.             s_hdcMem = CreateCompatibleDC(hdc); 
  46.             SelectObject(s_hdcMem, hBitmap);     
  47.             ReleaseDC(hDlg, hdc); 
  48.  
  49.             // 得到位图信息 
  50.             GetObject(hBitmap,sizeof(s_bm), &s_bm); 
  51.         } 
  52.  
  53.         return 0; 
  54.  
  55.     case WM_COMMAND: 
  56.         switch (LOWORD(wParam)) 
  57.         { 
  58.         case IDCANCEL: 
  59.             DeleteDC(s_hdcMem); 
  60.             EndDialog(hDlg, LOWORD(wParam)); 
  61.             return TRUE; 
  62.         } 
  63.         break
  64.  
  65.  
  66.     case WM_SIZE: 
  67.         InvalidateRect(hDlg, NULL, TRUE); 
  68.         return TRUE; 
  69.  
  70.     case WM_CTLCOLORSTATIC: 
  71.         SetBkMode((HDC)wParam, TRANSPARENT); 
  72.         return (BOOL)((HBRUSH)GetStockObject(NULL_BRUSH)); 
  73.  
  74.     case WM_CTLCOLORDLG: 
  75.         GetClientRect(hDlg, &rcDialog); 
  76.         //通过SetStretchBltMode的设置能使StretchBlt在缩放图像更加清晰 
  77.         SetStretchBltMode((HDC)wParam, COLORONCOLOR); 
  78.         StretchBlt((HDC)wParam, 0, 0, rcDialog.right, rcDialog.bottom, s_hdcMem, 0, 0, s_bm.bmWidth, s_bm.bmHeight, SRCCOPY);    
  79.         return (BOOL)((HBRUSH)GetStockObject(NULL_BRUSH)); 
  80.     } 
  81.     return FALSE; 

这份代码也是在《Windows界面编程第一篇位图背景与位图画刷》文章的代码上增加了

    case WM_CTLCOLORSTATIC:

            SetBkMode((HDC)wParam,TRANSPARENT);

             return (BOOL)((HBRUSH)GetStockObject(NULL_BRUSH));

来达到静态控件背景透明化的效果的,程序运行效果如下:

由图可以看出,虽然Static Text控件的是达到了背景透明化的要求,但是Group Box控件的描述文字的显示却显得很不美观。

要解决这一问题,可以试下位图画刷,我们在WM_CTLCOLORSTATIC消息中像WM_CTLCOLORDLG消息一样也返回一个位图画刷来试试。

  1. // 静态控件背景透明化- 在WM_CTLCOLORDLG返回窗口背景的位图画刷 
  2. //By MoreWindows-(http://blog.csdn.net/MoreWindows) 
  3. #include <windows.h> 
  4. #include "resource.h" 
  5.  
  6. constchar szDlgTitle[] = "静态控件背景透明化MoreWindows-(http://blog.csdn.net/MoreWindows)"
  7.  
  8.  
  9. // 对话框消息处理函数 
  10. BOOL CALLBACK DlgProc(HWND hDlg,UINT message, WPARAM wParam,LPARAM lParam); 
  11.  
  12. int APIENTRY WinMain(HINSTANCE hInstance, 
  13.                     HINSTANCE hPrevInstance, 
  14.                      LPSTR     lpCmdLine, 
  15.                     int       nCmdShow) 
  16.     DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG1), NULL, DlgProc); 
  17.     return 0; 
  18.  
  19.  
  20. BOOL CALLBACK DlgProc(HWND hDlg,UINT message, WPARAM wParam,LPARAM lParam) 
  21.     staticHBRUSH    s_hBitmapBrush; //位图画刷 
  22.  
  23.     switch (message) 
  24.     { 
  25.     case WM_INITDIALOG: 
  26.         // 设置对话框标题 
  27.         SetWindowText(hDlg, szDlgTitle); 
  28.         // 设置对话框大小可调节 
  29.         SetWindowLong(hDlg, GWL_STYLE, GetWindowLong(hDlg, GWL_STYLE) | WS_SIZEBOX); 
  30.  
  31.         // 加载背影图片 
  32.         HBITMAP hBitmap; 
  33.         hBitmap = (HBITMAP)LoadImage(NULL,"006.bmp", IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION); 
  34.         if (hBitmap == NULL) 
  35.         { 
  36.             MessageBox(hDlg,"LoadImage failed", "Error", MB_ICONERROR); 
  37.             exit(0); 
  38.         }        
  39.  
  40.         // 创建位图画刷 
  41.         s_hBitmapBrush = CreatePatternBrush(hBitmap); 
  42.         return 0; 
  43.  
  44.  
  45.     case WM_COMMAND: 
  46.         switch (LOWORD(wParam)) 
  47.         { 
  48.         case IDCANCEL: 
  49.             DeleteObject(s_hBitmapBrush); 
  50.             EndDialog(hDlg, LOWORD(wParam)); 
  51.             return TRUE; 
  52.         } 
  53.         break
  54.  
  55.     case WM_CTLCOLORSTATIC: 
  56.         SetBkMode((HDC)wParam, TRANSPARENT); 
  57.  
  58.     case WM_CTLCOLORDLG: 
  59.         return (BOOL)s_hBitmapBrush; 
  60.     } 
  61.     return FALSE; 

与上一个程序一样,这个程序也只是在在《Windows界面编程第一篇位图背景与位图画刷》文章的代码上增加了

       case WM_CTLCOLORSTATIC:

             SetBkMode((HDC)wParam,TRANSPARENT);

来设置静态控件背景透明化,程序运行效果如下:

由图可以看出,静态控件的透明化还是非常方便的,只要在WM_CTLCOLORSTATIC消息中完成二个步骤即可:

1.通过SetBkMode((HDC)wParam,TRANSPARENT);来设置文字区域背景透明。

2.返回空画刷或与父窗口相同的画刷。

原文地址:https://www.cnblogs.com/jinsedemaitian/p/5589194.html