MFC绘制不规则窗体的方法

所属分类: 软件编程 / C 语言 阅读数: 88
收藏 0 赞 0 分享

本文实例讲述了MFC 绘制不规则窗体的方法。分享给大家供大家参考。具体分析如下:

实现过程:

1、首先创建基于DLG的MFC应用程序,命名为:tryBGDlg,并将DLG的属性设置为:Title Bar :False ,其它设置不变

2、制作两幅图像,其中的一幅黑白图像,是根据播放器外观来制作的,其中白色区域是要保留的最终在桌面上显示的区域。将这两幅图像添加到工程中,第一个ID号设置为IDB_INTERFACE,第二个ID号设置为:IDB_MASK

3、在CtryBGDlg类中添加一个在函数:

函数说明:cBitmap是要传入的掩码位置变量,这里是指IDB_MASK创建的对象,TransColor是指要设为透明相素的RGB值

void CtryBGDlg::SetupRegion( CDC *pDC, CBitmap &cBitmap, COLORREF TransColor ) 
{ 
  CDC memDC; 
  memDC.CreateCompatibleDC(pDC); 
  CBitmap *pOldMemBmp=NULL; 
  pOldMemBmp=memDC.SelectObject(&cBitmap); 
  BITMAP bit; 
  cBitmap.GetBitmap (&bit); 
  CRgn crRgn, crRgnTmp; 
  crRgn.CreateRectRgn(0, 0, 0, 0);//创建一个空区域 
  int iX = 0;int iY = 0; 
  for (iY = 0; iY < bit.bmHeight; iY++) 
  { 
    do 
    { 
      //skip over transparent pixels at start of lines. 
      while (iX <= bit.bmWidth && memDC.GetPixel(iX, iY) == TransColor) 
        iX++; 
      //remember this pixel 
      int iLeftX = iX; 
      //now find first non transparent pixel 
      while (iX <= bit.bmWidth && memDC.GetPixel(iX, iY) != TransColor) 
        ++iX; 
      //create a temp region on this info 
      crRgnTmp.CreateRectRgn(iLeftX, iY, iX, iY+1); 
      //combine into main region. 
      crRgn.CombineRgn(&crRgn, &crRgnTmp, RGN_XOR); 
      //delete the temp region for next pass (otherwise you'll get an ASSERT) 
      crRgnTmp.DeleteObject(); 
    }while(iX < bit.bmWidth); 
    iX = 0; 
  } 
  SetWindowRgn(crRgn, TRUE); 
  iX = (GetSystemMetrics(SM_CXSCREEN))-700; 
  iY = (GetSystemMetrics(SM_CYSCREEN)) / 2 - (bit.bmHeight / 2); 
  SetWindowPos(&wndTop, iX, iY, bit.bmWidth, bit.bmHeight, NULL);  
  // Free resources. 
  memDC.SelectObject(pOldMemBmp);
  // Put the original bitmap back (prevents memory leaks) 
  memDC.DeleteDC(); 
  crRgn.DeleteObject(); 
}

4、在BOOL CtryBGDlg::OnInitDialog()函数中添加如下代码:

CBitmap bmp; 
bmp.LoadBitmapW(IDB_MASK); 
this->SetupRegion(this->GetWindowDC(),bmp,RGB(0,0,0));

5、添加对WM_ERASEBKGND消息响应,并在BOOL CtryBGDlg::OnEraseBkgnd(CDC* pDC)函数中添加如下代码

BOOL CtryBGDlg::OnEraseBkgnd(CDC* pDC) 
{ 
  // TODO: 在此添加消息处理程序代码和/或调用默认值 
  CRect rect; 
  this->GetWindowRect(&rect); 
  CDC memDC; 
  CBitmap bmp; 
  CBitmap *pOldBmp=NULL; 
  bmp.LoadBitmapW(IDB_INTERFACE); 
  memDC.CreateCompatibleDC(pDC); 
  pOldBmp=memDC.SelectObject(&bmp); 
  pDC->BitBlt(0,0,rect.Width(),rect.Height(),&memDC,0,0,SRCCOPY); 
  if(pOldBmp) 
  { 
    memDC.SelectObject(pOldBmp); 
  } 
  return true; 
// return CDialog::OnEraseBkgnd(pDC); 
}

到此就实现了不规则窗体的创建,创建后的视图如开头所示。

6、一般我们还要实现对窗体的托动操作,实现方法如下:
添加对WM_NCHITTEST消息的响应,并在生成的LRESULT CtryBGDlg::OnNcHitTest(CPoint point)函数中添加如下代码:

LRESULT CtryBGDlg::OnNcHitTest(CPoint point) 
{ 
  // TODO: 在此添加消息处理程序代码和/或调用默认值 
  CRect rc; 
  GetClientRect(&rc); 
  ClientToScreen(&rc); 
  return rc.PtInRect(point) ? HTCAPTION : CDialog::OnNcHitTest(point); 
// return CDialog::OnNcHitTest(point); 
}

至此就完全实现了,不规则窗体的创建和对窗体托动消息的响应部分。

下面将细致的讲解具体实现原理及部分的代码的解析:

总原理:这个程序的原理主要是先用IDB_MASK图像计算出要设定的窗体的轮廓,然后利用SetWindowRgn()函数来对其进行更改。最后在窗体重绘的时候响应WM_ERASEBKGND消息,将窗体背景图片IDB_INTERFACE贴到窗体上。
 
利用IDB_MASK图像计算窗体轮廓的原理:

计算窗体轮廓的代码主要靠SetupRegion()函数来实现,考虑到窗体的不规则,应采取掩模位图的方式来对其进行描述,对于本例,其白色区域为要保留的不规则窗体的轮廓区域。这段代码首先是用crRgn.CreateRectRgn(0, 0, 0, 0)创建一个空的区域,然后对IDB_MASK图像的像素信息进行一列一列的枚举,计算出每列中不设为透明的区域,然后跟crRgn合并,所以最后的crRgn就是所要设定的区域。

核心代码为:

CRgn crRgn, crRgnTmp; 
  //创建一个空区域 
  crRgn.CreateRectRgn(0, 0, 0, 0); 
  int iX = 0;int iY = 0; 
  for (iY = 0; iY < bit.bmHeight; iY++) 
  { 
    do 
    { 
      //skip over transparent pixels at start of lines. 
      //以一个相素列为单位,找到在这一个相素列中,第一个不是要设为透明相素的点iX。 
      //然后再找到以这个iX为起点的,在这个一个相素列中最后跟他临近的最后一个不是透明的点。 
      //然后将他们一起合并到crRgn中。 
      while (iX <= bit.bmWidth && memDC.GetPixel(iX, iY) == TransColor) 
        iX++;//在iY和iY+1这个相索列中,第一个不设为透明的点的X坐标 
      int iLeftX = iX;//保存这个点的坐标 
      while (iX <= bit.bmWidth && memDC.GetPixel(iX, iY) != TransColor) 
        ++iX;//这是找到在iX最临近的不透明的X坐标 
      crRgnTmp.CreateRectRgn(iLeftX, iY, iX, iY+1);//这四个点连在一起就是现在刚找到的不透明的区域 
      //合并区域 
      crRgn.CombineRgn(&crRgn, &crRgnTmp, RGN_OR); 
      //记得最终要手动删除crRgnTmp对象 
      crRgnTmp.DeleteObject(); 
    }while(iX < bit.bmWidth);
    //如果iX没有达到图片的末尾,说明还没有枚举完这一行,则在iY和iY+1这个行上,进行下一轮的//枚举 
    iX = 0; 
}

希望本文所述对大家的MFC程序设计有所帮助。

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多