GetWindowDC无法在标题栏输出文字原因分析和酷炫窗口代码实现

4105 人浏览 | 时间: 2015-12-06 16:06:06 | 作者: codexia 会员文章,禁止转载

    我们在看win32编程的书时,都看到过,获取窗口DC后,就可以在窗口的标题上写字了,而不只是客户区。我想你也一定会很激动,然而,在你多次尝试之后,会发现,并不会成功。很可能你就会唾骂书上写的不对,甚至会得出一个结论,这个有个卵用,假的。
    而实际上,它确实有用哦。将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一次只执行一个哦。
    相信通过本文的分析,之前的这个疑问都清楚了吧。如果还有问题,欢迎提出来。
当前文章为会员文章,请前往[用户中心]开通会员后继续阅读。

相关阅读