~$ 存档

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

屏幕抓取程序的意思是将整个屏幕图显示在应用程序的用户区中,等价于截图。

对桌面窗口的操作:
首先得知道桌面窗口的宽和高,获取宽和高需要利用窗口的设备句柄,而获取设备句柄需要知道窗口句柄,这一系列的连串关系如下:
窗口句柄——>设备句柄——>获取设备句柄属性(比如宽和高)。(书本采用的一种方法,但至少也把这种种被需要的关系复习了一下,必须承认,初学需要重复很多次才能形成熟练意识)

hwndDeskTop = GetDesktopWindow();// 获取桌面窗口的窗口句柄
hdcDeskTop = GetDC(hwndDeskTop);// 创建基本桌面窗口的设备描述表

这里借用两个函数,另外还有一种方法,直接创建一个桌面窗口设备的DC

hdcDeskTop = CreateDC("DISPLAY",NULL,NULL,NULL);//CreateDC()获得整个屏幕的DC,一步到位

有了句柄之后,就可以获得(或设置)这个DC的各种属性,比如,有多宽,多高等等....通过函数GetDeviceCaps()实现。摘抄函数原型如下:

int GetDeviceCaps(
    HDC hdc; //设备DC
    int nIndex; //返回的索引项
);

设备DC的信息很多,因此,nIndex的项尤其多。在这个例子只需要得到屏幕的宽和高。

cxScreen = GetDeviceCaps(hdcDeskTop, HORZRES);  // 获取桌面窗口的大小,也相当于矩形右下角的x,y坐标
cyScreen = GetDeviceCaps(hdcDeskTop, VERTRES);

 

接下来的这套流程比较繁杂,但是用最简易的方式描述,得到屏幕DC之后,再创建一个兼容DC(兼容屏幕设备),和一个兼容位图(兼容屏幕),将兼容位图放置在兼容DC中,准备用它来存放屏幕的位图。可以通俗的理解为是一个容器么?兼容DC容纳兼容位图。

hdcMem = CreateCompatibleDC(hdcDeskTop); // 创建与显示设备描述表兼容的内存设备描述表
hBitmap = CreateCompatibleBitmap( hdcDeskTop, cxScreen, cyScreen);
SelectObject(hdcMem, hBitmap);//并选入内存设备描述表

设备已经就绪,最后将屏幕的位图移动到兼容DC中的兼容位图上。

BitBlt(hdcMem, 0, 0, cxScreen, cyScreen,hdcDeskTop, 0, 0 ,SRCCOPY);

再看来BitBlt()函数,摘抄如下:

BOOL BitBlt(
HDC hDCDest,//目标设备描述表
int nXDest,//目标矩形左上角x坐标
int nYDest,//目标矩形左上角y坐标
int nWidth,//目标矩形宽
int nHeight,//目标矩形高
HDC hDCSrc,//源设备描述表
int nXDSrc,//源矩形x坐标(左上角)
int nYDSrc,//源矩形y坐标
DWORD dwRop //执行的光栅操作
);

其中最后一条DWORD dwRop光栅操作,让我联想起C语言的图形复制,就是在复制的过程中,如何与目标进行某种混合操作,比如让它反色(黑变白,白变黑),OR,异或操作等等,大概就是这个意思。

兼容内存设备描述表——>应用程序设备描述表

内存设备描述表的工作已经完成,接着,再把这个内存设备描述表中的位图复制到当前应用程序的用户区中,并让它显示。
即:兼容内存描述表——>应用程序设备描述表。

书本在这里考虑这样一个问题,让位图适当缩放来适应用户区的大小,于是采用了另一种复制函数StretchBlt。它的参数与BitBlt大部分相似,省略。
但是,在缩放之前,又设置了一个缩放模式,通过函数SetStretchBltMode实现,摘抄如下:

int SetStretchBltMode(
HDC hDC,
int iStretchMode
);
//iStretchMode的选项
BLACKONWHITE;//保留黑色像素,清除白色像素,一般用于单色位图中
COLORONCOLOR;//一般用于彩色位图中保留位图的颜色
WHITEONBLACK;//保留白色像素,清除黑色像素
HALFTONE;//对源图像进行复杂处理,慢但是得到高质量的图像

参数可以任意试验,明显HALFTONE参数截出的图清晰很多。
通过这一系列移来移去的复制操作完成了屏幕截图的工作,涉及一系列的位图函数,因此有必要认真总结一下。

/*---------------桌面截图程序研究版-----------------*/

#include <windows.h>
LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,PSTR szCmdLine, int iCmdShow)
{
     static TCHAR szAppName[] = TEXT ("HelloWin") ;
     HWND         hwnd ;                
     MSG          msg ;                
     WNDCLASS     wndclass ;          
     wndclass.style         = CS_HREDRAW | CS_VREDRAW ;
     wndclass.lpfnWndProc   = WndProc ;
     wndclass.cbClsExtra    = 0 ;
     wndclass.cbWndExtra    = 0 ;
     wndclass.hInstance     = hInstance ;
     wndclass.hIcon         = LoadIcon (NULL, IDI_APPLICATION) ;
     wndclass.hCursor       = LoadCursor (NULL, IDC_ARROW) ;
     wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;
     wndclass.lpszMenuName  = NULL ;
     wndclass.lpszClassName = szAppName ;

     if (!RegisterClass (&wndclass))
     {
            return 0 ;
     }
     hwnd = CreateWindow (szAppName,                  
                          TEXT ("鼠标消息"), 
                          WS_OVERLAPPEDWINDOW,        
                          CW_USEDEFAULT,              
                          CW_USEDEFAULT,              
                          CW_USEDEFAULT,             
                          CW_USEDEFAULT,              
                          NULL,                      
                          NULL,                       
                          hInstance,                  
                          NULL) ;                     
     
     ShowWindow (hwnd, iCmdShow) ;        
     UpdateWindow (hwnd) ;                 
     
     while (GetMessage (&msg, NULL, 0, 0))            
     {
          TranslateMessage (&msg) ;                 
          DispatchMessage (&msg) ;                   
     }
     return msg.wParam ;
}

LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
     static HDC hdc,hdcDeskTop,hdcMem;

     PAINTSTRUCT ps ;
     static int cxScreen,cyScreen,cxClient,cyClient;
     HBITMAP hBitmap;

     switch (message)
     {   
         
     case WM_CREATE:
         
    //hwndDeskTop = GetDesktopWindow();// 获取桌面窗口的窗口句柄    
    //hdcDeskTop = GetDC(hwndDeskTop);// 创建基本桌面窗口的设备描述表

    hdcDeskTop = CreateDC("DISPLAY",NULL,NULL,NULL);//CreateDC()获得整个屏幕的DC,一步到位
    hdcMem = CreateCompatibleDC(hdcDeskTop); // 创建兼容内存设备描述表        
  
    cxScreen = GetDeviceCaps(hdcDeskTop, HORZRES);  // 获取桌面窗口的大小
    cyScreen = GetDeviceCaps(hdcDeskTop, VERTRES);
     
    hBitmap = CreateCompatibleBitmap( hdcDeskTop, cxScreen, cyScreen);// 创建兼容位图                 
    SelectObject(hdcMem, hBitmap);//选入内存设备描述表
 

    //ShowWindow(hwnd, SW_HIDE);//先隐藏窗口,有这个必要吗?此时窗口未出现

    BitBlt(hdcMem, 0, 0, cxScreen, cyScreen,hdcDeskTop, 0, 0 ,SRCCOPY); // 将桌面位图的像素传送到兼容位图上                  
    //Sleep(1000);
    //ShowWindow(hwnd, SW_SHOW);//显示窗口
    DeleteDC(hdcDeskTop);

    return 0 ;

 

case WM_SIZE:
    cxClient = LOWORD(lParam);
    cyClient = HIWORD(lParam);
    return 0;
          
case WM_PAINT:

    hdc = BeginPaint (hwnd, &ps) ;
    // 将内存设备描述表的位图压缩显示到程序窗口内
    //SetStretchBltMode(hdc, COLORONCOLOR);//iStretchMode参数任意选择观察
    //SetStretchBltMode(hdc, HALFTONE);
    StretchBlt(hdc, 0, 0, cxClient, cyClient,  hdcMem, 0 ,0 , cxScreen, cyScreen, SRCCOPY);                  
    EndPaint (hwnd, &ps) ;
    return 0 ;

     case WM_DESTROY:
          PostQuitMessage (0) ;
          return 0 ;
     }
     return DefWindowProc (hwnd, message, wParam, lParam);
}

 

posted on 2016-04-19 04:03  LuoTian  阅读(1250)  评论(0编辑  收藏  举报