MaskBlt 的替代品





5.00/5 (2投票s)
2001年1月25日

132854

3467
现在您可以在 Windows 9X 上使用 MaskBlt
引言
当我开发一个基于位图的 CAD 系统时,我遇到了问题,因为一些 API 在 Windows 9X 上不受支持,例如 MaskBlt
。 我可以使用其他方法来解决这个问题,但我希望实现这个 API,以便在任何 Windows 9X 平台上使用 MaskBlt
。
所以这就是我所做的。
它几乎与绘制透明位图相同,但我添加了其他光栅操作,以便此函数与 MaskBlt
具有相同的功能。
示例显示,第一行的结果来自 WIN32 API 中的 MaskBlt。下一行的结果来自 MyMaskBlt。第三个是用于测试应用程序的掩码位图。
以下步骤描述了代码的工作原理。
- 创建一个用于掩码位图的 HDC。
- 创建一个 HDC/位图来保存掩码后的背景位图。
- 使用 3 个 BitBlt 将位图复制到掩码后的背景位图。
在此步骤中,我使用
BACK_ROP3(dwRop)
从 ROP4 获取 ROP3。并使用DSTERASE
来掩码位图。 - 创建一个 HDC/位图来保存掩码后的前景位图。
在此步骤中,我使用
FORE_ROP3(dwRop)
从 ROP4 获取 ROP3。此宏从 ROP4 中提取源 ROP3。然后将SRCAND
应用于掩码位图。 - 使用
SRCPAINT
合并从步骤 3 和 4 创建的两个位图,并将此位图复制到最终目标 HDC( hdcDest )。 - 清理。
#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。