【问题标题】:Getting HBITMAP from layered window - incorrect data从分层窗口获取 HBITMAP - 不正确的数据
【发布时间】:2012-11-13 20:36:22
【问题描述】:

我创建了一个分层窗口(使用 WS_EX_LAYERED),大小约为 400X300 像素。
绘制窗口时(使用 UpdateLayeredWindow)一切正常。

问题是绘制后无法获取窗口的HBITMAP。 当试图通过窗口的 HDC 获取 HBITMAP 时,我得到一个空(黑色)位图,它是我整个桌面的大小(1920X1080 像素,400X300 像素)。

有人知道是否有可能获得分层窗口的 HDC\HBITMAP 吗?

代码示例

这是我如何绘制分层窗口的代码(同样,效果很好):

void MyLayeredWindow::DrawLayered() const
{
    RECT rcWindow;
    GetWindowRect(rcWindow);

    int nWidth = abs(rcWindow.right - rcWindow.left);
    int nHeight = abs(rcWindow.bottom - rcWindow.top);

    // Create 32Bit bitmap to apply transparency 
    // (have to set negative height because if not the (0,0) point would be the bottom left instead of top left)
    VOID *ppvBits = NULL;
    BITMAPINFO BitmapInfo = {0};
    BitmapInfo.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
    BitmapInfo.bmiHeader.biWidth = nWidth;
    BitmapInfo.bmiHeader.biHeight = -nHeight;
    BitmapInfo.bmiHeader.biPlanes = 1;
    BitmapInfo.bmiHeader.biBitCount = 32;
    BitmapInfo.bmiHeader.biCompression = BI_RGB;

    // Copy view buffer to a temp DC and bitmap
    HDC hDcTemp = ::CreateCompatibleDC(NULL);
    assert(hDcTemp);
    HBITMAP hBitmapTemp = ::CreateDIBSection(hDcTemp, &BitmapInfo, DIB_RGB_COLORS, &ppvBits, NULL, 0);
    assert(hBitmapTemp && hBitmapTemp!=(HBITMAP)ERROR_INVALID_PARAMETER)
    ::SelectObject(hDcTemp, hBitmapTemp);

    // Darwing the window's conent here
    // ....
    // ....

    // Call UpdateLayeredWindow
    BLENDFUNCTION blend = {0};
    blend.BlendOp = AC_SRC_OVER;
    blend.SourceConstantAlpha = 190;
    blend.AlphaFormat = AC_SRC_ALPHA;

    SIZE sizeWnd = {0};
    sizeWnd.cx = nWidth;
    sizeWnd.cy = nHeight;
    POINT ptPos = {0};
    ptPos.x = rcWindow.left;
    ptPos.y = rcWindow.top;
    POINT ptSrc = {0,0};

    ::UpdateLayeredWindow(m_hWnd, NULL, &ptPos, &sizeWnd, hDcTemp, &ptSrc, 0, &blend, ULW_ALPHA);

    if(hDcTemp)
        ::DeleteDC(hDcTemp);

    if(hBitmapTemp)
        ::DeleteObject(hBitmapTemp);
}

这是我如何捕获窗口的位图并将其保存到文件的代码: (注意:它适用于“普通”窗口,例如计算器)

bool MyLayeredWindow::SaveBitmapFile(__in const HWND& hWnd, __in const wstring& sFileName)
{
    HDC hDC = ::GetDC(hWnd);

    // get bitmap of DC
    HBITMAP hBmp = (HBITMAP)::GetCurrentObject( hDC, OBJ_BITMAP );

    // get info of bitmap
    BITMAPINFO stBmpInfo; 
    stBmpInfo.bmiHeader.biSize = sizeof( stBmpInfo.bmiHeader );
    stBmpInfo.bmiHeader.biBitCount = 0;
    GetDIBits( hDC, hBmp, 0, 0, NULL, &stBmpInfo, DIB_RGB_COLORS );

    // init info size
    ULONG iBmpInfoSize; 
    switch( stBmpInfo.bmiHeader.biBitCount )
    { 
    case 24:
        iBmpInfoSize = sizeof(BITMAPINFOHEADER); 
        break;
    case 16: 
    case 32:
        iBmpInfoSize = sizeof(BITMAPINFOHEADER)+sizeof(DWORD)*3; 
        break;
    default:
        iBmpInfoSize = sizeof(BITMAPINFOHEADER) + sizeof(RGBQUAD) * ( 1 << stBmpInfo.bmiHeader.biBitCount ); 
        break;
    } 

    // copy header
    PBITMAPINFO pstBmpInfo = NULL;
    if( iBmpInfoSize != sizeof(BITMAPINFOHEADER) ) 
    { 
        pstBmpInfo = (PBITMAPINFO)GlobalAlloc( GMEM_FIXED | GMEM_ZEROINIT, iBmpInfoSize );
        PBYTE pbtBmpInfoDest = (PBYTE)pstBmpInfo; 
        PBYTE pbtBmpInfoSrc = (PBYTE)&stBmpInfo; 
        ULONG iSizeTmp = sizeof( BITMAPINFOHEADER );

        while( iSizeTmp-- ) 
            *( ( pbtBmpInfoDest )++ ) = *( ( pbtBmpInfoSrc )++ ); 
    } 

    // create file
    HANDLE hFile = CreateFile( sFileName.c_str(), GENERIC_WRITE, 0 , NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_ARCHIVE, NULL );

    // init bmp file header
    BITMAPFILEHEADER stBmpFileHder = {0}; 
    stBmpFileHder.bfType = 0x4D42; // 'BM' 
    stBmpFileHder.bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + iBmpInfoSize + pstBmpInfo->bmiHeader.biSizeImage; 
    stBmpFileHder.bfReserved1 = 0;
    stBmpFileHder.bfReserved2 = 0; 
    stBmpFileHder.bfOffBits = sizeof(BITMAPFILEHEADER) + iBmpInfoSize; 

    // write header to file
    DWORD dRet;
    WriteFile( hFile, (LPCVOID)&stBmpFileHder, sizeof(BITMAPFILEHEADER), &dRet, NULL );

    // allocate size for rest of bmp (body)
    PBYTE pBits = (PBYTE)GlobalAlloc( GMEM_FIXED | GMEM_ZEROINIT, stBmpInfo.bmiHeader.biSizeImage );

    // get bmp bits
    HBITMAP hBmpOld; 
    HBITMAP hTmpBmp = CreateCompatibleBitmap( hDC, pstBmpInfo->bmiHeader.biWidth, pstBmpInfo->bmiHeader.biHeight );
    hBmpOld = (HBITMAP)SelectObject( hDC, hTmpBmp ); 
    GetDIBits( hDC, hBmp, 0, pstBmpInfo->bmiHeader.biHeight, (LPSTR)pBits, pstBmpInfo, DIB_RGB_COLORS );

    // write bmp info
    WriteFile( hFile, (LPCVOID)pstBmpInfo, iBmpInfoSize, &dRet, NULL );

    // write bmp bits
    WriteFile( hFile, (LPCVOID)pBits, pstBmpInfo->bmiHeader.biSizeImage, &dRet, NULL );

    // release handles and free memory
    SelectObject( hDC, hBmpOld ); 
    DeleteObject( hTmpBmp ); 
    CloseHandle( hFile ); 
    GlobalFree( pstBmpInfo ); 
    GlobalFree( pBits ); 
    ReleaseDC( hWnd, hDC );

    return true;
}

谢谢!

【问题讨论】:

  • 这种方法通常是错误的,只能在偶然的情况下起作用。窗口没有位图,您需要使用 BitBlt 或 PrintWindow。对分层窗口使用 CAPTURE_BLT 选项。 msdn.microsoft.com/en-us/library/windows/desktop/…
  • BitBlt 和 PrintWindow 似乎不适用于我的分层窗口。它返回一个空白(黑色\透明)DC。

标签: c++ winapi hbitmap gdi layered-windows


【解决方案1】:

由于没有得到更好的答案,我只是调用了一个“绘制”函数,将我的分层窗口绘制到一个临时 HDC 上。

意思是我不复制现有的位图,而是创建一个相同的位图,使用相同的绘图功能。

我仍然希望得到这个问题的更好答案。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-07-31
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-03-29
    相关资源
    最近更新 更多