• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录

Woosa

合抱之木,生于毫末;九层之台,起于累土;千里之行,始于足下。
  • 博客园
  • 联系
  • 订阅
  • 管理

公告

View Post

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

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

1.WS_EX_LAYERED属性

2.指定透明色

3.以位图为窗口背景

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

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

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

intSetBkMode(

   HDC hdc,     // handle to DC

   int iBkMode   // background mode

);

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

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

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

const char szDlgTitle[] = "静态控件背景透明化MoreWindows-(http://blog.csdn.net/MoreWindows)";

// 对话框消息处理函数
BOOL CALLBACK DlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam);
          
int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
	DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG1), NULL, DlgProc);
	return 0;
}


BOOL CALLBACK DlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
	RECT       rcDialog;
	HBITMAP    hBitmap;
	static BITMAP s_bm;
	static HDC    s_hdcMem;

	switch (message)
	{
	case WM_INITDIALOG:
		// 设置对话框标题
		SetWindowText(hDlg, szDlgTitle);
		// 设置对话框大小可调节
		SetWindowLong(hDlg, GWL_STYLE, GetWindowLong(hDlg, GWL_STYLE) | WS_SIZEBOX);

		// 加载背影图片
		hBitmap = (HBITMAP)LoadImage(NULL, "006.bmp", IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION);
		if (hBitmap == NULL)
		{
			MessageBox(hDlg, "LoadImage failed", "Error", MB_ICONERROR);
			exit(0);
		}		
		else
		{
			// 将背影图片放入HDC - s_hdcMem
			HDC        hdc;
			hdc = GetDC(hDlg);
			s_hdcMem = CreateCompatibleDC(hdc);
			SelectObject(s_hdcMem, hBitmap);	
			ReleaseDC(hDlg, hdc);

			// 得到位图信息
			GetObject(hBitmap, sizeof(s_bm), &s_bm);
		}

		return 0;

	case WM_COMMAND:
		switch (LOWORD(wParam))
		{
		case IDCANCEL:
			DeleteDC(s_hdcMem);
			EndDialog(hDlg, LOWORD(wParam));
			return TRUE;
		}
		break;


	case WM_SIZE:
		InvalidateRect(hDlg, NULL, TRUE);
		return TRUE;

 	case WM_CTLCOLORSTATIC:
 		SetBkMode((HDC)wParam, TRANSPARENT);
 		return (BOOL)((HBRUSH)GetStockObject(NULL_BRUSH));

	case WM_CTLCOLORDLG:
		GetClientRect(hDlg, &rcDialog);
		//通过SetStretchBltMode的设置能使StretchBlt在缩放图像更加清晰
		SetStretchBltMode((HDC)wParam, COLORONCOLOR);
		StretchBlt((HDC)wParam, 0, 0, rcDialog.right, rcDialog.bottom, s_hdcMem, 0, 0, s_bm.bmWidth, s_bm.bmHeight, SRCCOPY);	
		return (BOOL)((HBRUSH)GetStockObject(NULL_BRUSH));
	}
	return FALSE;
}

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

    case WM_CTLCOLORSTATIC:

            SetBkMode((HDC)wParam,TRANSPARENT);

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

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

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

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

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

const char szDlgTitle[] = "静态控件背景透明化MoreWindows-(http://blog.csdn.net/MoreWindows)";


// 对话框消息处理函数
BOOL CALLBACK DlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam);

int APIENTRY WinMain(HINSTANCE hInstance,
					 HINSTANCE hPrevInstance,
					 LPSTR     lpCmdLine,
					 int       nCmdShow)
{
	DialogBox(hInstance, MAKEINTRESOURCE(IDD_DIALOG1), NULL, DlgProc);
	return 0;
}


BOOL CALLBACK DlgProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
	static HBRUSH    s_hBitmapBrush; //位图画刷

	switch (message)
	{
	case WM_INITDIALOG:
		// 设置对话框标题
		SetWindowText(hDlg, szDlgTitle);
		// 设置对话框大小可调节
		SetWindowLong(hDlg, GWL_STYLE, GetWindowLong(hDlg, GWL_STYLE) | WS_SIZEBOX);

		// 加载背影图片
		HBITMAP hBitmap;
		hBitmap = (HBITMAP)LoadImage(NULL, "006.bmp", IMAGE_BITMAP, 0, 0, LR_LOADFROMFILE | LR_CREATEDIBSECTION);
		if (hBitmap == NULL)
		{
			MessageBox(hDlg, "LoadImage failed", "Error", MB_ICONERROR);
			exit(0);
		}		

		// 创建位图画刷
		s_hBitmapBrush = CreatePatternBrush(hBitmap);
		return 0;


	case WM_COMMAND:
		switch (LOWORD(wParam))
		{
		case IDCANCEL:
			DeleteObject(s_hBitmapBrush);
			EndDialog(hDlg, LOWORD(wParam));
			return TRUE;
		}
		break;

	case WM_CTLCOLORSTATIC:
		SetBkMode((HDC)wParam, TRANSPARENT);

	case WM_CTLCOLORDLG:
		return (BOOL)s_hBitmapBrush;
	}
	return FALSE;
}

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

       case WM_CTLCOLORSTATIC:

             SetBkMode((HDC)wParam,TRANSPARENT);

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

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

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

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

posted on 2013-03-15 20:11  Woosa  阅读(245)  评论(0)    收藏  举报

刷新页面返回顶部
 
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3