在《GetWindowDC无法在标题栏输出文字原因分析和酷炫窗口代码实现》一文中,介绍了使用GetWindowDC无法在标题栏输出文字原因分析,同时为了得到标题栏的高度,我做了一个计算。本来用系统方法也是可以的,只是在那篇文章中不增加额外的负担,自己计算也显得更加自由。不过,我却没有在那篇文章中仔细介绍计算标题栏高度的方法,只有代码,我相信稍微有点基础,花点时间就可以明白的。
要详细了解窗口组成结构和计算标题栏高度,请阅读文章《windows窗口界面组成结构图解》,文中有详细的说明。然后介绍系统提供的快捷方法。自己计算标题栏的高度,在看了前面提到的文章后,可以看实现代码。代码如下:
RECT rect,rectClient;
GetWindowRect(hwnd,&rect);//-得到窗口矩形大小
//-rect.bottom :窗口的高度,rect.right : 窗口的宽度
GetClientRect(hwnd,&rectClient);//-得到客户区矩形大小
//-rectClient.bottom :客户区的高度,rectClient.right : 客户区的宽度
rect.bottom = rect.bottom-rectClient.bottom-(rect.right-rectClient.right)/2;
可以看到,这样计算的方式确实非常麻烦,所以系统给我们提供了快速获取标题栏高度的方法。快速获取系统信息的函数为GetSystemMetrics,传入SM_CYCAPTION即可返回标题栏的高度。代码如下:
rect.bottom = GetSystemMetrics(SM_CYCAPTION);
然而这样得到的标题栏并不包括上边框,如果要从窗口顶部绘制到客户区之间的所有区域,就要加上上边框的高度,上边框的高度一般和左右边框粗细一样,所以可以用窗口宽度减去客户区宽度,然后除以二得到的值当做上边框宽度。所以,从窗口顶部到客户区开始之间的区域高度,是标题栏高度+上边框高度,代码如下:rect.bottom = GetSystemMetrics(SM_CYCAPTION)+(rect.right-rectClient.right)/2;
如此一来,其实如果要得到窗口顶部到客户区开始的高度,反而麻烦了。但是如果只是得到标题栏的高度的话,用GetSystemMetrics还是挺方便的。你可以看到,通过将获取的标题栏的高度加上我们计算的上边框的高度,就完全正确了。效果如下:
标题栏和上边框高度正确计算后绘制标题栏成功
我们绘制的标题栏的位置才可以拖动,客户区的位置不能拖动。如果不加上上边框,就导致我们绘制的标题栏外的客户区紧挨着标题栏的地方可以拖动,这只是我们的绘制错误导致的,少绘制了一部分的标题栏。如果你看不懂这一段话,请仔细研究文中开头提到的文章的程序。本文使用的程序所有代码如下:
#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);//-计算得到的和系统得到的一致的,然而能拖到窗口的还包括顶部边框
rect.bottom = GetSystemMetrics(SM_CYCAPTION)+(rect.right-rectClient.right)/2;
FillRect(hdc,&rect,(HBRUSH)GetStockObject(GRAY_BRUSH));
// - 在标题栏上输出文字
rect.top+=7;// - 调节输出的文字的起始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_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;
}