跟我一起玩Win32开发(22):抓取屏幕
关于如何拷贝屏幕并保存,这里已经有现成的例子,我也不必去Copy人家了,我一向不喜欢Copy。这里有一个完整的例子,可以看看。
http://msdn.microsoft.com/EN-US/library/windows/desktop/dd183402(v=vs.85).aspx
把屏幕的内容复制到窗口的客户区域中,通常会用BitBlt函数,函数的功能是把一块颜色数据从一个DC复制到另一个DC,这个我也不知道怎么翻译才能通俗一点。这样说吧,就是从源设备上下文的图形表面截取一个矩形区域并且复制到另一个设备上下文的区域中。就像我们要做一个截屏工具一样,把屏幕的一部分内容复制到窗口上。
下面呢,我用另一个函数来进行拷贝——StretchBlt函数,这个函数与BitBlt差不多,不过,它有一点,就是可以根据目标的区域对源图像进行拉伸。
注意看代码。
- {
- // 屏幕的DC
- HDC hdcScreen = GetDC(NULL);
- // 本窗口的DC
- HDC hdcWindow = GetDC(hWnd);
- // 屏幕的宽度
- int scrWidth = GetSystemMetrics(SM_CXSCREEN);
- // 屏幕的高度
- int scrHeight = GetSystemMetrics(SM_CYSCREEN);
- // 窗口的客户区域
- RECT rectClient;
- GetClientRect(hWnd, &rectClient);
- // 使用StretchBlt进行复制
- StretchBlt(hdcWindow,0,0,rectClient.right,rectClient.bottom,hdcScreen,0,0,scrWidth,scrHeight,SRCCOPY);
- // 释放DC
- ReleaseDC(NULL, hdcScreen);
- ReleaseDC(hWnd, hdcWindow);
- }
现在,我们要明确,我们是要把屏幕上的东西往窗口区域复制,所以我们想到,必须先有两个DC,一个是屏幕的DC,另一个就是窗口的DC。DC通过GetDC函数可以获取,将参数设置为NULL,也就是获得主屏幕的DC句柄了,NULL可以认为获取桌面的DC。
获取到DC后,我们还必须知道源区域的宽度和高度,以及目标窗口区域的宽度和高度。
源区域是屏幕,所以我们只要知道了当前屏幕的高度和宽度就可以了,GetSystemMetrics(SM_CXSCREEN)返回当前屏幕的宽度,GetSystemMetrics(SM_CYSCREEN)获得当前屏幕的高度。
而窗口的区域大小呢?我们不妨先得到窗口客户区域的矩形大小,用GetClientRect函数填充一个RECT结构体,这个结构体的right成员就是窗口客户区域的宽度,bottom成员就是窗口客户区域的高度了。
好了,有了这些基本参数,后面的事情就好办了。
BOOL StretchBlt( _In_ HDC hdcDest, _In_ int nXOriginDest, _In_ int nYOriginDest, _In_ int nWidthDest, _In_ int nHeightDest, _In_ HDC hdcSrc, _In_ int nXOriginSrc, _In_ int nYOriginSrc, _In_ int nWidthSrc, _In_ int nHeightSrc, _In_ DWORD dwRop);
以上是StretchBlt函数的声明,带“Dest”字样的都与目标区域有关,带“Src”字样的都与源区域有关,至于什么含义,看参数名字就知道了,你懂的。
最后一个参数,是一个标志,就是告诉函数用什么形式去复制,我们这里使用SRCCOPY就是按源来的数据复制,不作修改。这个参数可以在Wingdi.h找到,说明在MSDN文档。
现在,你可以看看它的效果。
再如,如果我把StretchBlt的最后一个参数改为NOTSRCCOPY,这就使得源区域与当前区域进行取反,也就是我们常说的“反色”。再看看。
StretchBlt(hdcWindow,0,0,rectClient.right,rectClient.bottom,hdcScreen,0,0,scrWidth,scrHeight,NOTSRCCOPY);
但是你会发现,当进行多次操作,窗口上显示的有一部分变成了正色,为什么呢?因为在你第二次截屏时,窗口中显示了上一次的截屏的内容,是反色的,而对反色再进行反色,不就变成了正色了吗?负负得正,所以才会看到像下图所示的效果。