而实际上,它确实有用哦。将GetWindowDC获得的窗口DC在标题上输出文字,一个是在WM_PAINT客户区重绘消息中输出文字和填充背景颜色,一个是在非客户区绘制消息WM_NCPAINT中处理,两个的效果基本一样,只是在填充背景时,获取的矩形区域一个是客户区的区域,一个是窗口的矩形区域。下面是两个效果:
而我们在直接想用GetWindowDC获得的窗口DC在窗口标题上输出文字,就是直接在WM_PAINT消息或者其他鼠标单击消息等处理,而却没有达到想要的效果。实际上,可能是书中忽略了一点,或者是你在看书时忽略了一点。不过我比较偏向于书上没有说清楚。
事实上,GetWindowDC确实可以在窗口的标题栏中输出文字,这里不是问题的关键。我也相信你一定会用这个函数。而问题的关键就在于,忽略了一个消息,那就是非客户区绘制消息WM_NCPAINT。这个消息会让系统绘制一个默认的窗口标题和窗口外观,而我们只是在其他地方设置了这个标题,然后很快又没非客户区消息的默认处理给覆盖掉了。所以始终都看不到效果。
那么如果干掉烦人的非客户区重绘消息的处理呢?在《Win32消息处理后必须返回0,窗口过程中必须有默认消息处理函数》中,我们做了介绍,即处理过消息就返回0,表示我们已经处理过了。而我们通常都没有处理WM_NCPAINT消息,也就是默认让这个消息给默认处理函数处理了,这样自然就看到的始终都是默认的标题文字和样式。
所以,要看到自己画的标题栏,可以自己处理这个WM_NCPAINT消息,然后简单的返回0,从而拦截了默认处理,但是你不做任何处理,就可以在其他地方设置了。这就是上面两个图显示的在客户区处理的结果。看上去还挺漂亮的哦,是不是有点小激动呢?
所以,这个WM_NCPAINT消息才是问题的关键,只要你知道这一点了,分分钟就可以达到你心中的目标了。
当然,虽然在客户区中是可以绘制非客户区,但是,这不是推荐的做法,有点越级的意思。因为WM_NCPAINT消息本来就是来绘制非客户区的,所以,这个工作就交给在WM_NCPAINT消息来处理咯,只不过你确实可以这样过一把瘾。还有,提示下,鼠标键盘消息处理中也可以实现哦。
不过需要注意一点,GetWindowDC获得的DC绘制的内容只能显示在非客户区,所以,要是显示到了客户区的位置的内容,就看不见了哦。这个与你在哪写代码实现无关,至于DC有关哦。
我们只需要将输出的文字的其实y坐标调大即可看到这个效果,如下图所示:
以下是完整的代码:
#include "windows.h"
#include <tchar.h>
// - 项目是Unicode字符集
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
HDC hdc;
PAINTSTRUCT ps;
RECT rect,rectTitle,rectClient;
static TCHAR str[]=_T("C++技术网http://www.cjjjs.com");
static int i=0;
static TCHAR ch;
switch (message)
{
case WM_PAINT:
BeginPaint(hwnd,&ps);// - 不能去掉,不然会闪烁
hdc = GetWindowDC(hwnd);
// - 填充整个窗口非客户区
GetWindowRect(hwnd,&rect);
FillRect(hdc,&rect,(HBRUSH)GetStockObject(LTGRAY_BRUSH));
// - 填充窗口的标题栏
GetClientRect(hwnd,&rectClient);
rect.bottom = rect.bottom-rectClient.bottom-(rect.right-rectClient.right)/2;
FillRect(hdc,&rect,(HBRUSH)GetStockObject(GRAY_BRUSH));
// - 在标题栏上输出文字
rect.top+=20;// - 调节输出的文字的起始y坐标
rect.left+=40;
SetBkMode(hdc,TRANSPARENT);// - 去掉文字背景
SetTextColor(hdc,RGB(255,255,0));// - 设置文字颜色
DrawText(hdc,_T("C++技术网www.cjjjs.com 提醒您,这里是窗口标题区"),-1,&rect,DT_SINGLELINE);
ReleaseDC(hwnd,hdc);
EndPaint(hwnd,&ps);
return 0;
case WM_NCPAINT:
//hdc = GetWindowDC(hwnd);
//GetWindowRect(hwnd,&rect);
//FillRect(hdc,&rect,(HBRUSH)GetStockObject(LTGRAY_BRUSH));
//GetClientRect(hwnd,&rectClient);
//rect.bottom = rect.bottom-rectClient.bottom-(rect.right-rectClient.right)/2;
//FillRect(hdc,&rect,(HBRUSH)GetStockObject(GRAY_BRUSH));
//rect.top+=20;
//rect.left+=40;
//SetBkMode(hdc,TRANSPARENT);
//SetTextColor(hdc,RGB(255,255,0));
//DrawText(hdc,_T("C++技术网www.cjjjs.com 提醒您,这里是窗口标题区"),-1,&rect,DT_SINGLELINE);
//ReleaseDC(hwnd,hdc);
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
default:
return DefWindowProc(hwnd, message, wParam, lParam);
}
}
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrev,LPSTR lpCmd,int iShow)
{
TCHAR ClassName[] = _T("MyClass");
TCHAR title1[] = _T("原始标题");
WNDCLASS wndClass;
wndClass.cbClsExtra=0;
wndClass.cbWndExtra=0;
wndClass.hbrBackground= (HBRUSH)GetStockObject(GRAY_BRUSH);
wndClass.hCursor=LoadCursor(NULL,IDC_ARROW);
wndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION);
wndClass.hInstance = hInstance;
wndClass.lpfnWndProc = WinProc;
wndClass.lpszClassName = ClassName;
wndClass.lpszMenuName=NULL;
wndClass.style=CS_HREDRAW|CS_VREDRAW;
if(!RegisterClass(&wndClass))
{
return 0;
}
HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,0,0,440,400,NULL,NULL,hInstance,NULL);
ShowWindow(hwnd,SW_SHOWNORMAL);
MSG msg;
while (GetMessage(&msg,NULL,0,0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return msg.wParam;
}
代码的WM_PAINT和WM_NCPAINT一次只执行一个哦。相信通过本文的分析,之前的这个疑问都清楚了吧。如果还有问题,欢迎提出来。