利用掩码位图实现透明显示

来源:互联网 发布:英语单词数据库 mysql 编辑:程序博客网 时间:2024/05/16 10:20

位图的透明显示一般有两种机制,即透明色机制和掩码位图机制。透明色机制是把位图中的某种颜色设为透明,掩码位图机制是利用一个掩码位图来确定要透明的部分。本文只介绍掩码位图机制。

首先来看一个函数:

void DrawMaskBmp(CDC *pDC,int nX,int nY,CBitmap &bitmap,CBitmap &maskBitmap)
{
  CDC bitmapDC;
  CBitmap *pOldBmp1;

  BITMAP bmp;
  bitmap.GetBitmap(&bmp);
  bitmapDC.CreateCompatibleDC(pDC);
  pOldBmp1 = bitmapDC.SelectObject(&bitmap); 

  CDC maskDC;
  CBitmap *pOldBmp2;

  maskDC.CreateCompatibleDC(pDC);
  pOldBmp2 = maskDC.SelectObject(&maskBitmap);

  CDC bufDC;
  CBitmap bufBitmap,*pOldBmp3;
  bufBitmap.CreateCompatibleBitmap(pDC,bmp.bmWidth,bmp.bmHeight);
  bufDC.CreateCompatibleDC(pDC);
  pOldBmp3 = bufDC.SelectObject(&bufBitmap);
  bufDC.BitBlt(0,0,bmp.bmWidth,bmp.bmHeight,&bitmapDC,0,0,SRCCOPY);
  bufDC.BitBlt(0,0,bmp.bmWidth,bmp.bmHeight,&maskDC,0,0,SRCINVERT);

  bitmapDC.SelectObject(pOldBmp1);
  bitmapDC.DeleteDC();

  pDC->BitBlt(nX,nY,bmp.bmWidth,bmp.bmHeight,&maskDC,0,0,SRCAND);
  pDC->BitBlt(nX,nY,bmp.bmWidth,bmp.bmHeight,&bufDC,0,0,SRCPAINT);

  maskDC.SelectObject(pOldBmp2);
  maskDC.DeleteDC();
  bufDC.SelectObject(pOldBmp3);
  bufDC.DeleteDC();
  bufBitmap.DeleteObject();
}

该函数实现了把位图bitmap透明显示到pDC的nX、nY位置,maskBitmap是掩码位图。它的调用方法如下:

void CMyDialog::MyDraw()
 {
   CBitmap bitmap,maskBitmap;
   bitmap.LoadBitmap(IDB_BITMAP);
   maskBitmap.LoadBitmap(IDB_BITMAP_MASK);
   CClientDC cdc(this);
   DrawMaskBmp(&cdc,85,250,bitmap,maskBitmap);

   bitmap.DeleteObject();
   maskBitmap.DeleteObject();
 }

从函数DrawMaskBmp可以看到,要实现位图的透明显示,要经过如下几个步骤:

1、创建一个要显示位图的掩码位图;

要显示的位图:,掩码位图:。掩码位图是一个单色位图,它的黑色部分就是位图显示时要保留的部分,白色部分就是要透明的部分。


 2、创建一个和该位图大小一样的内存位图和DC(bufBitmap、bufDC),把该位图用SRCCOPY(拷贝)方式显示到这个DC中;


 3、把掩码位图用SRCINVERT(XOR)方式叠加到该内存DC中;

原位图和掩码位图做了XOR操作后,内存DC中的位图如下所示:,即白色XOR白色=黑色(白色的RGB都是255,1^1=0),黑色XOR任何颜色=原颜色(0^0=0,0^1=1)。


 4、把掩码位图用SRCAND(AND)方式叠加到pDC中;

 and  ,即白色and任何颜色=原颜色(1&1=1,1&0=0),黑色and任何颜色=黑色(0&0=0,0&1=0)。


 5、把bufDC用SRCPAINT(or)方式叠加到pDC中。

 or  = ,即黑色or任何颜色=原颜色(0|1=1,0|0=0)。

 

当然了,上文只是为了说明利用掩码位图进行透明显示的原理,实际应用中用MaskBlt函数更简单,它的原型如下:

BOOL MaskBlt ( 
  int
 x
  int
 y
  int
 nWidth
  int
 nHeight
  CDC* 
pSrcDC
  int
 xSrc
  int
 ySrc
  CBitmap& 
maskBitmap
  int
 xMask
  int
 yMask
  DWORD
 dwRop );

针对上述情况,pDC->MaskBlt(nX,nY,bmp.bmWidth,bmp.bmHeight,&bitmapDC,0,0,maskBitmap,0,0,0xccaa0000);就可以了。

 

补充说明: 0xccaa0000的由来

MaskBlt的最后一个参数是一个四元光栅操作码,可用MAKEROP4(fore,back)由两个三元光栅操作码生成,当maskBitmap中对应的位为1时,采用fore光栅操作码,否则采用back光栅操作码。此处fore为0x00AA0000(和目标相同),back为0x00CC0000(和源相同),即maskBitmap中黑色(0)对应部分采用pSrcDC中的图像,白色(1)对应部分不改变。三元光栅操作码请参考MSDN中的"Ternary Raster Operations"。



转帖:http://blog.csdn.net/iamshuke/article/details/5749897