windows程序设计第14章的一个程序,下面是对该程序的分析:
位图显示的话一般都是矩形,比如之前写的一篇关于显示位图的博文
显示位图
那么如果我们想椭圆显示呢?如图:
哦,可以这样,将选好的图片复制到画图程序中,用白色的画笔”擦除”椭圆外的部分,然后设置客户区颜色为白色,再在客户区显示位图,如果将客户区颜色改变呢?显然这样做是不行的,为此,有一种非常通用的技术来解决此类问题,该技术包括”位掩码”位图和一些光栅操作。
位掩码是一种单色位图,尺寸与要显示的位图尺寸相同,因为是单色,所以1是白色,0是黑色,那么如何做呢?慢慢一步步来
hInstance = ((LPCREATESTRUCT)lParam)->hInstance;
//加载位图资源
hBitmapImag = LoadBitmap(hInstance, MAKEINTRESOURCE(IDB_BITMAP1));
//获取位图宽度,高度
GetObject(hBitmapImag, sizeof(BITMAP), &bitmap);
cxBitmap = bitmap.bmWidth;
cyBitmap = bitmap.bmHeight;
//选进内存DC
hdcMemImag = CreateCompatibleDC(NULL);
SelectObject(hdcMemImag, hBitmapImag);
//创建位掩码位图
//因为是单色位图,所以1为白色,0为黑色
hBitmapMask = CreateBitmap(cxBitmap, cyBitmap, 1, 1, NULL);
hdcMemMask = CreateCompatibleDC(NULL);
SelectObject(hdcMemMask, hBitmapMask);
//这里是hdcMemMask中位掩码位图的整个矩形颜色为黑色
SelectObject(hdcMemMask, GetStockObject(BLACK_BRUSH));
Rectangle(hdcMemMask, 0, 0, cxBitmap, cyBitmap);
//然后使得其椭圆区域为白色
SelectObject(hdcMemMask, GetStockObject(WHITE_BRUSH));
Ellipse(hdcMemMask, 0, 0, cxBitmap, cyBitmap);
//将hdcMemImag中的位图与位掩码位图合并,源位图与目标位图像素进行光栅与运算
BitBlt(hdcMemImag, 0, 0, cxBitmap, cyBitmap,
hdcMemMask, 0, 0, SRCAND);
DeleteDC(hdcMemImag);
DeleteDC(hdcMemMask);
return 0;
现在来看一下经过WM_CREATE消息里处理后的位图怎么样
位掩码怎么样呢?
嗯,然后怎么做呢?
看一下这条语句:
BitBlt(hdc, x, y, cxBitmap, cyBitmap, hdcMemMask, 0, 0, 0x220326);
后面的ox220326是神马东西???查看msdn BitBlt的用法,一些基本常见的三元(源位图,画刷,目标位图 )光栅操作都用宏定义了,但还有很多没定义,
函数不懂,查madn……
然后msdn是这么解释的:
00220326 DSna Dest = (NOT Src) AND Dest
也就是说,目标位图像素值和源位图相应像素取反后的值再做与运算,
所以,现在客户区应该是这样
到了这里,是不是想到了,对,现在椭圆内饰黑色的,即像素值0,再调用一次BitBle就可以了,WM_CREATE里处理后的位图椭圆内像素值与0(客户区中椭圆)做或运算,结果是位图像素值,椭圆外的像素值(为0)与客户区相应区域做位运算,结果是客户区像素值,正好
所以,再调用一次:
BitBlt(hdc, x, y, cxBitmap, cyBitmap, hdcMemImag, 0, 0, SRCPAINT);
SRCPAINT:
Combines the colors of the source and destination rectangles
by using the Boolean OR operator.
再次看下结果,实现也不是那么难,对吧