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


图一、叠加在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

用Visual C++设计“精灵”窗体的更多相关文章

  1. 游戏UI框架设计(三) : 窗体的层级管理

    游戏UI框架设计(三) ---窗体的层级管理 UI框架中UI窗体的"层级管理",最核心的问题是如何进行窗体的显示管理.窗体(预设)的显示我们前面定义了三种类型: 普通.隐藏其他.反 ...

  2. 【转载】Visual Studio中WinForm窗体程序如何切换.NET Framework版本

    在C#语言的WinForm窗体程序中,有时候我们需要切换WinForm窗体程序项目的.NET Framework版本号,例如从.NET Framework 4.5版本切换到.NET Framework ...

  3. pyqt5对用qt designer设计的窗体实现弹出子窗口的示例

    pyqt5对用qt designer设计的窗体实现弹出子窗口的示例 脚本专栏 python 1. 用qt designer编写主窗体,窗体类型是MainWindow,空白窗口上一个按钮.并转换成mai ...

  4. WPF设计の自定义窗体

    效果图如下: 实现思路: 1.继承Window类 2.为自定义的CustomWindow类设计窗体样式(使用Blend很方便!) 3.为窗体增加最大最小化和关闭按钮,并实现鼠标拖拽改变窗体大小(使用D ...

  5. WPF设计の不规则窗体

    我们在工作中,经常会需要画一些不规则的窗体,现在总结如下. 一.利用VisualBrush实现.这依赖于VisualBrush的特性,任何控件可以作为画刷,而画刷又可以作为背景. 此种方法可以用于实现 ...

  6. visual studio 设计第一个WinForm小程序

    WinForm小程序之消息框 首先打开visual studio 软件,然后[文件]-[新建]-[项目]-[Visual C#]-[Windows],选择Windows窗体应用程序,根据自己的需要修改 ...

  7. Visual Studio 2015 改变窗体图标 & 任意位置打开窗体 & 禁止鼠标改动窗体大小

    1.改变窗体图标 先把图标放到项目文件夹中,然后点击窗体属性的ICON添加即可. 参考:https://www.cnblogs.com/yangxuli/p/8075484.html?tdsource ...

  8. delphi如何设计不规则窗体

    制作多边形窗体的关键在于设定多边形的区域,并根据这个指定的区域改变窗体的形状.Windows的CreatePolygonRgn和SetWindowRgn函数可以解决这两个难点.利用以下代码即可将窗体设 ...

  9. 解析大型.NET ERP系统核心组件 查询设计器 报表设计器 窗体设计器 工作流设计器 任务计划设计器

    企业管理软件包含一些公共的组件,这些基础的组件在每个新项目立项阶段就必须考虑.核心的稳定不变功能,方便系统开发与维护,也为系统二次开发提供了诸多便利.比如通用权限管理系统,通用附件管理,通用查询等组件 ...

随机推荐

  1. idea15破解

    注册方法:   注册码可以沿用14的,只是在 注册时选择 License server ,填 http://idea.lanyus.com ,然后点击 OK 14的话,网上可以找到一个,根据你的用户名 ...

  2. C# 如何从List集合当中取出子集合

    今天项目要求随机从数据库中随机取出若干条数据,放到首页.那么要如何随机取出这个子集合呢?本人向到的方法如下: 1.假设数据量很少,如我数据库中只有10条数据,而我要求随机取出8条.对于这种低数据量,大 ...

  3. Java学习之基本数据类型

    基本类型,或者叫做内置类型,是JAVA中不同于类的特殊类型.它们是我们编程中使用最频繁的类型.java是一种强类型语言,第一次申明变量必须说明数据类型,第一次变量赋值称为变量的初始化. 1. Java ...

  4. Labview学习之程序Web发布

    Labview学习之程序Web发布 1. LabVIEW Web服务器     在LabVIEW开发环境中,自身带了一个已连接好的Web服务器.LabVIEW Web服务器除了与其他Web服务器一样能 ...

  5. Eclipse+EGit的配置注意点, 以及解决Github多个本地仓库之间的冲突

    问题描述 不同本地仓库(e.g. Repo1, Repo2)之间同时修改一个文件时, 出现文件无法merge的情况. 具体表现为, 冲突(红色双向实心箭头)一直存在, 点pull没反应, 点push报 ...

  6. Oracle 查询时间在当天的数据

    要实现这个功能需要用到trunc这个函数对时间的操作 select trunc(sysdate) from dual --2014-12-27 今天的日期为2014-12-27 select trun ...

  7. hdu 4034 Graph floyd

    题目链接 给出一个有向图各个点之间的最短距离, 求出这个有向图最少有几条边, 如果无法构成图, 输出impossible. folyd跑一遍, 如果dp[i][j] == dp[i][k]+dp[k] ...

  8. 关于scanf("%c",&ch)直接跳过的问题

    有时候scanf("%c",&ch)本应该阻塞等待用户输入一个char型数据的,但为什么会跳过呢? 例:在该程序段中,  int year;    printf(" ...

  9. Snort

    https://www.snort.org/ http://blog.csdn.net/htttw/article/details/7521053 http://www.ibm.com/develop ...

  10. Microsoft Azure 负载平衡服务

     Microsoft Azure 为在其中托管的虚拟机(IaaS) 和云服务(PaaS) 提供负载平衡服务.负载平衡支持应用程序伸缩,并且提供应用程序故障恢复以及其他优势. 可以通过以下方式访问负 ...