65.9K
CodeProject 正在变化。 阅读更多。
Home

MaskBlt 的替代品

starIconstarIconstarIconstarIconstarIcon

5.00/5 (2投票s)

2001年1月25日

viewsIcon

132854

downloadIcon

3467

现在您可以在 Windows 9X 上使用 MaskBlt

MaskTest Image

引言

当我开发一个基于位图的 CAD 系统时,我遇到了问题,因为一些 API 在 Windows 9X 上不受支持,例如 MaskBlt。 我可以使用其他方法来解决这个问题,但我希望实现这个 API,以便在任何 Windows 9X 平台上使用 MaskBlt

所以这就是我所做的。

它几乎与绘制透明位图相同,但我添加了其他光栅操作,以便此函数与 MaskBlt 具有相同的功能。

示例显示,第一行的结果来自 WIN32 API 中的 MaskBlt。下一行的结果来自 MyMaskBlt。第三个是用于测试应用程序的掩码位图。

以下步骤描述了代码的工作原理。

  1. 创建一个用于掩码位图的 HDC。

  2. 创建一个 HDC/位图来保存掩码后的背景位图。

  3. 使用 3 个 BitBlt 将位图复制到掩码后的背景位图。

    在此步骤中,我使用 BACK_ROP3(dwRop) 从 ROP4 获取 ROP3。并使用 DSTERASE 来掩码位图。

  4. 创建一个 HDC/位图来保存掩码后的前景位图。

    在此步骤中,我使用 FORE_ROP3(dwRop) 从 ROP4 获取 ROP3。此宏从 ROP4 中提取源 ROP3。然后将 SRCAND 应用于掩码位图。

  5. 使用 SRCPAINT 合并从步骤 3 和 4 创建的两个位图,并将此位图复制到最终目标 HDC( hdcDest )。
  6. 清理。

#define    FORE_ROP3(ROP4)        (0x00FFFFFF&(ROP4))
#define    BACK_ROP3(ROP4)        (ROP3FromIndex(SwapROP3_SrcDst(BYTE((ROP4)>>24))))
#define DSTCOPY 0x00AA0029
#define DSTERASE 0x00220326 // dest = dest & (~src) : DSna

BOOL WINAPI MyMaskBlt(HDC hdcDest, int nXDest, int nYDest, int nWidth, int nHeight,
    HDC hdcSrc, int nXSrc, int nYSrc,
    HBITMAP hbmMask, int xMask, int yMask,
    DWORD dwRop
)
{
    if ( hbmMask == NULL )
        return BitBlt(hdcDest, nXDest, nYDest, nWidth, nHeight, hdcSrc, 
                      nXSrc, nYSrc, FORE_ROP3(dwRop));

    // 1. make mask bitmap's dc
    HDC hDCMask = ::CreateCompatibleDC(hdcDest);
    HBITMAP hOldMaskBitmap = (HBITMAP)::SelectObject(hDCMask, hbmMask);
    ASSERT ( hOldMaskBitmap );

    // 2. make masked Background bitmap

    // 2.1 make bitmap
    HDC hDC1 = ::CreateCompatibleDC(hdcDest);
    ASSERT ( hDC1 );
    HBITMAP hBitmap2 = ::CreateCompatibleBitmap(hdcDest, nWidth, nHeight);
    HBITMAP hOldBitmap2 = (HBITMAP)::SelectObject(hDC1, hBitmap2);
    ASSERT ( hOldBitmap2 );

    // 2.2 draw dest bitmap and mask
    DWORD dwRop3 = BACK_ROP3(dwRop);
    ::BitBlt(hDC1, 0, 0, nWidth, nHeight, hdcSrc, nXSrc, nYSrc, SRCCOPY);
    ::BitBlt(hDC1, 0, 0, nWidth, nHeight, hdcDest, nXDest, nYDest, dwRop3);
    ::BitBlt(hDC1, 0, 0, nWidth, nHeight, hDCMask, xMask, yMask, DSTERASE);

    // 3. make masked Foreground bitmap

    // 3.1 make bitmap
    HDC hDC2 = ::CreateCompatibleDC(hdcDest);
    ASSERT ( hDC2 );
    HBITMAP hBitmap3 = ::CreateCompatibleBitmap(hdcDest, nWidth, nHeight);
    HBITMAP hOldBitmap3 = (HBITMAP)::SelectObject(hDC2, hBitmap3);
    ASSERT ( hOldBitmap3 );

    // 3.2 draw src bitmap and mask
    dwRop3 = FORE_ROP3(dwRop);
    ::BitBlt(hDC2, 0, 0, nWidth, nHeight, hdcDest, nXDest, nYDest, SRCCOPY);
    ::BitBlt(hDC2, 0, 0, nWidth, nHeight, hdcSrc, nXSrc, nYSrc, dwRop3);
    ::BitBlt(hDC2, 0, 0, nWidth, nHeight, hDCMask, xMask, yMask, SRCAND);

    // 4. combine two bitmap and copy it to hdcDest
    ::BitBlt(hDC1, 0, 0, nWidth, nHeight, hDC2, 0, 0, SRCPAINT);
    ::BitBlt(hdcDest, nXDest, nYDest, nWidth, nHeight, hDC1, 0, 0, SRCCOPY);

    // 5. restore all object
    ::SelectObject(hDCMask, hOldMaskBitmap);
    ::SelectObject(hDC1, hOldBitmap2);
    ::SelectObject(hDC2, hOldBitmap3);

    // 6. delete all temp object
    DeleteObject(hBitmap2);
    DeleteObject(hBitmap3);

    DeleteDC(hDC1);
    DeleteDC(hDC2);
    DeleteDC(hDCMask);

    return TRUE;
}

提交代码后,我才意识到其中存在一个错误。Lonnie Cumberland 让我知道这个错误,我修复了这个错误。实际上,在修复之前它与 MaskBlt 不兼容。:)

示例中使用的某些图像来自 wxwindows 项目。

感谢 Lonnie。

© . All rights reserved.