用Visual C++设计“精灵”窗体

随着Microsoft凭借Windows在操作系统上取得的巨大成绩,Windows用户界面也日益成为业界标准。统一的界面给广大用户对应用软件的学习与使用带来了很大方便。但每天都面对同一副面孔,日久天长难免会产生一些厌倦,想开发一些"离经叛道"的应用程序,如果能够一改Windows千篇一律的"标准"界面,一定会给用户带来一种清新的感觉。标准Windows应用程序窗口一般为带有标题栏的浅灰色矩形外观,因而"异形"对话框/窗口也主要是颜色与外形上动手脚。本实例实现了一个"精灵"窗口,程序编译运行后的界面效果如图一所示:

用Visual C++设计“精灵”窗体
图一、叠加在Visual C++开发工具上的透明"精灵"窗体

  一、实现方法

  一般情况下,实现异型窗体主要是作两方面的工作,一是改变背景颜色,二是改变窗口外形。改变窗口背景颜色是最简单的改变Windows应用程序外观的方法,根据Windows创建与管理机理,一般有两种方法。一种是处理WM_CTLCOLOR消息,需要重画窗口或对话(或对话的子控件)时,Windows向应用程序发送消息WM_CTLCOLOR,应用程序处理WM_CTLCOLOR消息并返回一个用来绘画窗体背景的刷子句柄。另外一种是响应Windows的WM_ERASEBKGND消息,Windows向窗口发送一个WM_ERASEBKGND消息通知该窗口擦除背景,可以使用 VC++的ClassWizard重载该消息的缺省处理程序来擦除背景(实际是用刷子画),并返回TRUE以防止Windows擦除窗口。对于改变窗体的外形,可以通过使用新的SDK函数SetWindowRgn(),将绘画和鼠标消息限定在窗口的一个指定的区域,因此实际上是使窗口成为指定的不规则形状(区域形状)。"区域"是Windows GDI中一种强有力的机制,区域是设备上的一块空间,可以是任意形状,复杂的区域可以由各个小区域组合而成。Windows内含的区域创建函数有 CreateRectRgn()、CreatePolyRgn()、CreatePolygonRgn()、CreateRoundRectRgn()和 CreateEllipticRgn(),再通过CombineRgn()来组合区域,即可得到复杂形状的区域,获得复杂形状的窗口外形。

  通过上面的方法虽然可以得到"异形"窗口,但感觉颜色单调,外形也不够"COOL",能否获得更酷的"异形"窗口呢?回答是肯定的。下面就介绍利用位图和蒙板创建"异形"窗口的方法。本实例实现的"精灵"效果就是通过这种方法实现的。

  利用位图创建异形窗体的原理是根据象素的颜色来进行"扣像"处理,对所有非指定颜色象素区域进行区域组合。利用这一技术,实际上就是实现对话框/窗口的位图背景,并且对指定的颜色区域进行透明处理。下面就以透明位图为背景的窗体为例来说明:

  首先用绘图软件如PhotoShop绘制编辑一幅拟做程序背景用的图片以及该图片相应的掩模图片,用BMP格式保存,下一步是用Visual C++的资源编辑器引入该背景图片和掩模图片文件,设置其ID为IDB_SHOW和IDB_MASK。需要说明的是,虽然Visual C++集成开发环境的资源编辑器只能编辑不超过16色的位图,但完全我们可以以真彩色方式存储,不必理会Visual C++的警告。

  有了上述的工作,剩下的核心工作就是根据背景位图和掩模位图来确定最终显示的位图区域,也就是说,"扣除"的区域将以透明效果显示。下面的代码实现了这一功能:

/////////////////////////////////////////////////////////////////////////////
// 获得窗体矩形 
CRect rectWnd;
this->GetWindowRect(rectWnd);
// 读取"掩模"位图资源
CBitmap myBitmap,*pOldBitmap;
myBitmap.LoadBitmap(nMaskId);
// 创建"内存一致"设备
CDC memDC;
memDC.CreateCompatibleDC(pDC);
// 选择绘图设备
pOldBitmap = memDC.SelectObject(&myBitmap);
// 创建窗体的初始区域
CRgn rgnWnd,rgnTemp;
rgnWnd.CreateRectRgn(0,0,rectWnd.Width(),rectWnd.Height());
int nWidth,nHeight;
COLORREF color; 
//下面的两层循环为检查背景位图象素颜色,进行透明区域处理;
//当象素颜色为指定的透明值时,即将该点从区域中剪裁掉。
for (nWidth = 0;nWidth <= rectWnd.Width()-1;nWidth++)
{
 for (nHeight = 0;nHeight <= rectWnd.Height();nHeight++)
 {
  color = memDC.GetPixel(nWidth,nHeight);
  // 当象素是白色时,去掉该点
  if (color == RGB(255,255,255))
  {
   //象素颜色为指定的透明色,创建透明"微区域"
   rgnTemp.CreateRectRgn(nWidth,nHeight,nWidth+1,nHeight+1);
   //"扣像",从完整的区域中"扣除"透明的"微区域"
   rgnWnd.CombineRgn(&rgnWnd,&rgnTemp,RGN_XOR);
   //删除刚创建的透明"微区域",释放系统资源
   rgnTemp.DeleteObject(); 
  }
 }
}
memDC.SelectObject(pOldBitmap);
SetWindowRgn((HRGN)rgnWnd,TRUE); //用最终设定窗口的显示区域为指定区域

  为了最终显示透明效果的窗体,还需要重置系统默认的背景擦除操作,即添加WM_ERASEBKGND消息处理过程,在其中实现背景位图的显示功能,这一步可以借助ClassWizard来实现。

  最后需要注意的是,为了使异形窗体应用程序能够正常地脱动和退出,需要处理窗体的WM_NCHITTEST、WM_CHAR等消息,这部分内容本书中其它实例已经作过介绍,这里就不再赘述了,感性趣的读者朋友可以参考相关实例。
 二、编程步骤

  1、 启动Visual C++6.0,生成一个Win32应用程序,将该程序命名为"transparent";

  2、 使用Class Wizard在应用程序中添加CMyWnd类,其基类选择CWnd;

  3、 向应用程序的项目中添加背景位图和掩模位图,其ID分别设置为IDB_SHOW、IDB_MASK;

  4、 添加代码,编译运行程序。

  三、程序代码

//////////////////////////////////////////////////////////////// MyWnd.h : header file
#if !defined(AFX_MYWND_H__A761190E_CDF2_4A56_8848_AEE5AC7AD160__INCLUDED_)
#define AFX_MYWND_H__A761190E_CDF2_4A56_8848_AEE5AC7AD160__INCLUDED_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
// CMyWnd window 
class CMyWnd : public CWnd
{
 // Construction
 public:
  CMyWnd();
  // Attributes
 public:
  // Operations
 public:
  // Overrides
  // ClassWizard generated virtual function overrides
  //{{AFX_VIRTUAL(CMyWnd)
  //}}AFX_VIRTUAL
  // Implementation
 public:
  void Initialize(LPCTSTR lpszName,CRect &rectWnd,UINT nMaskId,UINT nShowId);
  void Display(CDC *pDC,UINT nMaskId);
  virtual ~CMyWnd();
  // Generated message map functions
 protected:
  //{{AFX_MSG(CMyWnd)
   afx_msg BOOL OnEraseBkgnd(CDC* pDC);
   afx_msg UINT OnNcHitTest(CPoint point);
   afx_msg void OnChar(UINT nChar, UINT nRepCnt, UINT nFlags);
  //}}AFX_MSG
  DECLARE_MESSAGE_MAP()
 private:
  UINT m_nBitmapId;
};
//{{AFX_INSERT_LOCATION}}
// Microsoft Visual C++ will insert additional declarations immediately before the previous line.
#endif // !defined(AFX_MYWND_H__A761190E_CDF2_4A56_8848_AEE5AC7AD160__INCLUDED_)

////////////////////////////////////////////////////////// MyWnd.cpp : implementation file
#include "stdafx.h"
#include "transparent.h"
#include "MyWnd.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
// CMyWnd
CMyWnd::CMyWnd()
{}

CMyWnd::~CMyWnd()
{}

BEGIN_MESSAGE_MAP(CMyWnd, CWnd)
 //{{AFX_MSG_MAP(CMyWnd)
  ON_WM_ERASEBKGND()
  ON_WM_NCHITTEST()
  ON_WM_CHAR()
 //}}AFX_MSG_MAP
END_MESSAGE_MAP()

BOOL CMyWnd::OnEraseBkgnd(CDC* pDC) 
{
 // TODO: Add your message handler code here and/or call default
 CRect rectWnd;
 this->GetWindowRect(&rectWnd);
 CDC memDC;
 CBitmap myBitmap,*pOldBitmap;
 myBitmap.LoadBitmap(m_nBitmapId);
 memDC.CreateCompatibleDC(pDC);
 pOldBitmap = memDC.SelectObject(&myBitmap);
 pDC->BitBlt(0,0,rectWnd.Width(),rectWnd.Height(), &memDC,0,0,SRCCOPY);
 // 将设备还原
 memDC.SelectObject(pOldBitmap );
 return TRUE;
}

UINT CMyWnd::OnNcHitTest(CPoint point) 
{
 // TODO: Add your message handler code here and/or call default
 UINT nHitTest = CWnd :: OnNcHitTest(point) ; 
 return (nHitTest == HTCLIENT) ? HTCAPTION : nHitTest ; 
}

void CMyWnd::Display(CDC *pDC, UINT nMaskId)
{
 // 获得窗体矩形 
 CRect rectWnd;
 this->GetWindowRect(rectWnd);
 // 读取位图资源
 CBitmap myBitmap,*pOldBitmap;
 myBitmap.LoadBitmap(nMaskId);
 // 创建"内存一致"设备
 CDC memDC;
 memDC.CreateCompatibleDC(pDC);
 //选择绘图设备
 pOldBitmap = memDC.SelectObject(&myBitmap);
 // 创建窗体的初始区域
 CRgn rgnWnd,rgnTemp;
 rgnWnd.CreateRectRgn(0,0,rectWnd.Width(),rectWnd.Height());
 int nWidth,nHeight;
 COLORREF color;
 for (nWidth = 0;nWidth <= rectWnd.Width()-1;nWidth++)
 {
  for (nHeight = 0;nHeight <= rectWnd.Height();nHeight++)
  {
   color = memDC.GetPixel(nWidth,nHeight);

   // 当象素是白色时,去掉该点
   if (color == RGB(255,255,255))
   {
    rgnTemp.CreateRectRgn(nWidth,nHeight,nWidth+1,nHeight+1);
    rgnWnd.CombineRgn(&rgnWnd,&rgnTemp,RGN_XOR);
    rgnTemp.DeleteObject(); 
   }
  }
 }
 memDC.SelectObject(pOldBitmap);
 SetWindowRgn((HRGN)rgnWnd,TRUE);
}

void CMyWnd::Initialize(LPCTSTR lpszName, CRect &rectWnd, UINT nMaskId, UINT nShowId)
{
 this->CreateEx(0,AfxRegisterWndClass(0),lpszName,WS_POPUP
| WS_SYSMENU,rectWnd,NULL,NULL,NULL);
 this->Display(GetWindowDC(),nMaskId);
 m_nBitmapId = nShowId;
}

void CMyWnd::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags) 
{
 // TODO: Add your message handler code here and/or call default
 if(nChar==VK_ESCAPE)
 {
  PostMessage(WM_QUIT);
 }
 CWnd::OnChar(nChar, nRepCnt, nFlags);
}
////////////////////////////////////////////////////////////
BOOL CTransparentApp::InitInstance()
{
 #ifdef _AFXDLL
  Enable3dControls(); // Call this when using MFC in a shared DLL
 #else
  Enable3dControlsStatic(); // Call this when linking to MFC statically
 #endif
 m_pMainWnd = &m_wndMain;// m_wndMain是CmyWnd类成员变量;
 CRect rectWnd(500, 300, 580, 380);
 m_wndMain.Initialize(_T("loveghost"),rectWnd,IDB_MASK,IDB_SHOW);
 m_wndMain.ShowWindow(SW_SHOW); // 窗体创建完毕,显示之;
 return TRUE; // 返回非零值表示要继续处理消息
}

  四、小结

  这种异形窗口的创建方法适应于所有的基于CWnd类的派生窗口,采用这一方法,可以创建出任何形状的"异形"窗体。

from:http://www.cnblogs.com/MVC2/archive/2009/08/16/1547577.html

上一篇:[Java]重载,重写以及继承,多态的区别


下一篇:物联网推动eSIM市场需求 用户签约管理平台将成核心