区域需要先创建好才能使用。区域是一个GDI对象,与矩形不一样,所以才需要创建。这个就和画笔、画刷一样的,所以区域也是GDI对象,也就不难理解了。
创建区域的说明,请参见文章《GDI中多个多边形区域的创建函数的参数详细分析》,这篇文章有相关文章的引用。
创建区域不成问题了。那么区域如何变成剪切区域呢?this is a question!前面说了,区域就是一个GDI对象。画笔也是GDU对象。画笔要使用,需要选进DC。同样,区域要使用,也是要选进DC咯。当区域选进DC后,区域就成了剪切区域。
所以,使用SelectObject函数即可让区域变成剪切区域。当然,SelectClipRgn函数也是可以完成这个效果。SelectObject和SelectClipRgn使用方法都是差不多的。只不过SelectObject更加通用,它还可以选择画刷画笔等。而SelectClipRgn只是用来选择剪切区域的。
默认的剪切区域是客户区范围,如果你设置了一个小剪切区域,绘图完毕后,想要恢复原来的剪切区域。那么你需要再次设置一次剪切区域哦。只不过这次用客户区范围来设置剪切区域而已。
我们先设置一个小的剪切区域输出字符串,然后恢复为客户区范围的剪切区域,再输出完整的字符串。你会发现前面输出的字符串并不完整哦。下面是效果图:
【小剪切区域输出的字符串不完整】
下面是完整的代码:#include "windows.h"
#include <tchar.h>
TCHAR txt[100]=_T("C++技术网http://www.cjjjs.com");
// - 项目是Unicode字符集
LRESULT CALLBACK WinProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{
HDC hdc;
PAINTSTRUCT ps;
switch (message)
{
case WM_PAINT:
{
hdc = BeginPaint(hwnd,&ps);
//设置矩形(0,0,100,100)范围为剪切区域,无法显示完整字符串
HRGN hRgn = CreateRectRgn(0,0,100,100);
SelectObject(hdc,hRgn);//剪切区开始生效
TextOut(hdc,50,50,txt,lstrlen(txt));
DeleteObject(hRgn);//删除区域
//恢复剪切区域为客户区,则可以完成显示字符串
RECT rect;
GetClientRect(hwnd,&rect);//获取客户区矩形
HRGN hRgnClient = CreateRectRgnIndirect(&rect);
SelectClipRgn(hdc,hRgnClient);//剪切区开始生效
TextOut(hdc,50,70,txt,lstrlen(txt));
DeleteObject(hRgnClient);//删除区域
EndPaint(hwnd,&ps);
}
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
default:
break;//跳出到默认处理
}
return DefWindowProc(hwnd, message, wParam, lParam);
}
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrev,LPSTR lpCmd,int iShow)
{
TCHAR ClassName[] = _T("MyClass");
TCHAR title1[] = _T("C++技术网http://www.cjjjs.com");
WNDCLASS wndClass;
wndClass.cbClsExtra=0;
wndClass.cbWndExtra=0;
wndClass.hbrBackground= (HBRUSH)GetStockObject(WHITE_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|CS_DBLCLKS;
if(!RegisterClass(&wndClass))return 0;
HWND hwnd = CreateWindow(ClassName,title1,WS_OVERLAPPEDWINDOW,10,100,550,400,NULL,NULL,hInstance,NULL);
ShowWindow(hwnd,SW_SHOWNORMAL);
MSG msg;
while (GetMessage(&msg,NULL,0,0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return msg.wParam;
}