Benjamin

静以修身,俭以养德,非澹薄无以明志,非宁静无以致远。
随笔 - 195, 文章 - 0, 评论 - 197, 引用 - 0
数据加载中……

关于VC图片透明处理的补充

前一阵子我的博客中有篇图片透明处理的文章,下面对它做个补充。这里也可以用在EVC和VC的各个版本上
1、透明:包括关键色透明和alpha透明,关键色透明就是选择一种颜色作为透明色,一般选取背景色。  
  alpha透明就是和背景融合,一般范围是0~255,数值越小背景越明显,0表示完全透明,255表示不透明  
  公式  
  dstred   =   srcred   *   (alpha   /   255.0)   +   dstred   *   (1.0   -   alpha   /   255.0);  
  dstred   =   srcred   *   (alpha   /   255.0)   +   dstred   *   (1.0   -   alpha   /   255.0);  
  dstred   =   srcred   *   (alpha   /   255.0)   +   dstred   *   (1.0   -   alpha   /   255.0);  
   
  2、窗口透明  
  2.1、如果系统是2000以上可以设置窗口的透明属性如下:  
   
  //至少需要Windows   2000系统  
  ModifyStyleEx(0,WS_EX_LAYERED);   //修改Window的风格  
  //如果不设置属性,窗口就看不到  
  //SetLayeredWindowAttributes(GetSysColor(COLOR_BTNFACE),0,LWA_COLORKEY);   //透明色是按钮背景颜色   这个就可以实现不规则窗口  
  SetLayeredWindowAttributes(0,0,LWA_ALPHA);   //半透明  
  //SetLayeredWindowAttributes(GetSysColor(COLOR_BTNFACE),128,LWA_COLORKEY   |   LWA_ALPHA);   //两个混合  
   
  注意要设置_WIN32_WINNT的值   0x500  
把上面的代码加入窗体初始化代码中就可以了。
.2、  
  如果是2000以下的系统,可以设置窗口的Region来达到关键色透明效果  
  void   MyWnd::SetRegion(COLORREF   clrTrans)  
  {  
  m_bmp.LoadBitmap(IDB_BITMAP1);   //读取位图资源  
  BITMAP   bm;  
  CSize   sz;  
  m_bmp.GetBitmap(&bm);  
  sz.cx   =   bm.bmWidth;  
  sz.cy   =   bm.bmHeight;  
  unsigned   char   *pBuffer   =   new   unsigned   char[sz.cx   *   sz.cy   *   (bm.bmBitsPixel   >>   3)];  
  m_bmp.GetBitmapBits(sz.cx   *   sz.cy   *   (bm.bmBitsPixel   >>   3),pBuffer);  
  unsigned   char   *pOriBuffer   =   pBuffer;  
   
  CRgn   rgn,rgn2;  
  rgn.CreateRectRgn(0,0,sz.cx,sz.cy);  
  rgn2.CreateRectRgn(0,0,1,1);  
   
  int   off   =   (bm.bmBitsPixel   >>   3);  
  for(int   y   =   0;   y   <   sz.cy;   y++)  
  {  
  for(int   x   =   0;   x   <   sz.cx;   x++)  
  {  
  if(((*pBuffer)   ==   255)   &&   ((*(pBuffer   +   1))   ==   0)   &&   ((*(pBuffer   +   2))   ==   255))  
  {  
  rgn2.SetRectRgn(x,y,x   +   1,y   +   1);  
  rgn.CombineRgn(&rgn,&rgn2,RGN_DIFF);  
  }  
  pBuffer   +=   off;  
  }  
  }  
   
  delete   []pOriBuffer;  
  SetWindowRgn((HRGN)rgn.GetSafeHandle(),TRUE);  
  }  
3、AlphaBlend   Alpha透明  
   
  //绘制可以透明的矩形  
  void   DrawAlphaRect(CDC   *pDC,CRect&   r,COLORREF   clr,unsigned   char   alpha)  
  {  
  CDC   memdc;  
  memdc.CreateCompatibleDC(pDC);  
  CBitmap   bmp,*pOldBitmap;  
  bmp.CreateCompatibleBitmap(pDC,r.Width(),r.Height());  
  pOldBitmap   =   memdc.SelectObject(&bmp);  
  memdc.FillSolidRect(0,0,r.Width(),r.Height(),clr);  
  BLENDFUNCTION   bf;  
  bf.BlendOp   =   AC_SRC_OVER;  
  bf.BlendFlags   =   0;  
  bf.SourceConstantAlpha   =   alpha;  
  bf.AlphaFormat   =   0;  
  pDC->AlphaBlend(r.left,r.top,r.Width(),r.Height(),&memdc,0,0,r.Width(),r.Height(),bf);  
  memdc.SelectObject(pOldBitmap);  
  }  
   
  //绘制可以透明的位图  
  void   DrawAlphaBitmap(CDC   *pDC,CRect&   r,CBitmap&   bmp,unsigned   char   alpha)  
  {  
  CDC   memdc;  
  memdc.CreateCompatibleDC(pDC);  
  CBitmap   *pOldBitmap;  
  CSize   sz   =   bmp.GetBitmapDimension();  
  BITMAP   bm;  
  bmp.GetBitmap(&bm);  
  sz.SetSize(bm.bmWidth,bm.bmHeight);  
  pOldBitmap   =   memdc.SelectObject(&bmp);  
  BLENDFUNCTION   bf;  
  bf.BlendOp   =   AC_SRC_OVER;  
  bf.BlendFlags   =   0;  
  bf.SourceConstantAlpha   =   alpha;  
  bf.AlphaFormat   =   0;  
  pDC->AlphaBlend(r.left,r.top,r.Width(),r.Height(),&memdc,0,0,sz.cx,sz.cy,bf);  
  memdc.SelectObject(pOldBitmap);  
  }  
   
  这样调用  
  COLORREF   clr   =   RGB(141,173,223);  
  DrawAlphaRect(pDC,CRect(10,50,500,400),clr,192);  
   
  DrawAlphaBitmap(pDC,CRect(10,10,300,200),bmp,128);  

posted on 2009-05-04 21:43 Benjamin 阅读(3886) 评论(2)  编辑 收藏 引用 所属分类: VC

评论

# re: 关于VC图片透明处理的补充  回复  更多评论   

/绘制可以透明的位图 这个函数放在那里调用啊?CDC *pDC就是窗口吗?
2015-03-20 13:13 | 张银平

# re: 关于VC图片透明处理的补充  回复  更多评论   

加载图片的时候调用
2015-04-17 08:44 | Benjamin

只有注册用户登录后才能发表评论。
【推荐】超50万行VC++源码: 大型组态工控、电力仿真CAD与GIS源码库
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理