|
|
随着Microsoft凭借Windows在操作系统上取得的巨大成绩,Windows用户界面也日益成为业界标准。统一的界面给广大用户对应用软件的学习与使用带来了很大方便。但每天都面对同一副面孔,日久天长难免会产生一些厌倦,想开发一些"离经叛道"的应用程序,如果能够一改Windows千篇一律的"标准"界面,一定会给用户带来一种清新的感觉。标准Windows应用程序窗口一般为带有标题栏的浅灰色矩形外观,因而"异形"对话框/窗口也主要是颜色与外形上动手脚。本实例实现了一个"精灵"窗口,程序编译运行后的界面效果如图一所示:( O7 P6 G* l$ Y; g c( M
9 w/ q5 @" e% [* ^' o3 I% M" \3 c% E7 A j$ m
图一、叠加在Visual C++开发工具上的透明"精灵"窗体
: \) K/ v5 C, g o5 A
( D: ^5 r! d+ \" y+ ]9 r 一、实现方法
: n; N2 E* u3 v2 Y$ Y9 _! H' i7 Y) E: K7 v2 ]' f' l1 b5 e
一般情况下,实现异型窗体主要是作两方面的工作,一是改变背景颜色,二是改变窗口外形。改变窗口背景颜色是最简单的改变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()来组合区域,即可得到复杂形状的区域,获得复杂形状的窗口外形。
K/ j4 L" {# m7 a+ Y1 @7 A8 Y# w+ Y
" H4 F& P1 q0 B5 p, [ 通过上面的方法虽然可以得到"异形"窗口,但感觉颜色单调,外形也不够"COOL",能否获得更酷的"异形"窗口呢?回答是肯定的。下面就介绍利用位图和蒙板创建"异形"窗口的方法。本实例实现的"精灵"效果就是通过这种方法实现的。5 _& A5 h; r4 C7 x. _
1 \7 h( k8 W3 w6 ?4 x) e
利用位图创建异形窗体的原理是根据象素的颜色来进行"扣像"处理,对所有非指定颜色象素区域进行区域组合。利用这一技术,实际上就是实现对话框/窗口的位图背景,并且对指定的颜色区域进行透明处理。下面就以透明位图为背景的窗体为例来说明:" A( N }3 U* {9 z
9 e. X, ?% V- M: @/ @. }4 ` 首先用绘图软件如PhotoShop绘制编辑一幅拟做程序背景用的图片以及该图片相应的掩模图片,用BMP格式保存,下一步是用Visual C++的资源编辑器引入该背景图片和掩模图片文件,设置其ID为IDB_SHOW和IDB_MASK。需要说明的是,虽然Visual C++集成开发环境的资源编辑器只能编辑不超过16色的位图,但完全我们可以以真彩色方式存储,不必理会Visual C++的警告。
0 A+ w( a! u* Q7 O
R5 w& R H- q. ]4 F6 { 有了上述的工作,剩下的核心工作就是根据背景位图和掩模位图来确定最终显示的位图区域,也就是说,"扣除"的区域将以透明效果显示。下面的代码实现了这一功能:
. L: s7 g' y- F3 W# r, m& c
+ f2 ^8 }8 r# t, ^. B" q! E( L: P/////////////////////////////////////////////////////////////////////////////
. q! O% p4 _2 m" q0 r* ^: g// 获得窗体矩形 0 W2 U8 [. ?& Z1 x1 ~" a
CRect rectWnd;% A3 K: N. w+ P# c0 S8 Y
this->GetWindowRect(rectWnd);
? M; o: a7 S+ I: F4 q) x' |/ ^// 读取"掩模"位图资源1 g6 m; F4 `, \+ `) {
CBitmap myBitmap,*pOldBitmap;; M- c" K/ T6 }1 A% ^7 Z
myBitmap.LoadBitmap(nMaskId);9 p D1 x$ [! n" V/ H
// 创建"内存一致"设备
' K) V, L# Y. w& L, g uCDC memDC;: C8 r* M9 ]0 o( }# n
memDC.CreateCompatibleDC(pDC);
- D; u4 Y( ]# L7 Z' H8 W// 选择绘图设备
) r/ O1 E8 ~2 H. y! i1 `" WpOldBitmap = memDC.SelectObject(&myBitmap);1 Y; s7 w/ I% }8 A* ]0 v, y
// 创建窗体的初始区域% j' H4 u1 A E6 L
CRgn rgnWnd,rgnTemp;
% E( D o& N+ jrgnWnd.CreateRectRgn(0,0,rectWnd.Width(),rectWnd.Height());7 V6 D% N& B6 n8 b \% `& }. Q% d6 @
int nWidth,nHeight;
! b# C# }1 j5 m. i/ }COLORREF color; 5 | t& \6 D0 P; }
//下面的两层循环为检查背景位图象素颜色,进行透明区域处理;7 N& T1 L- G/ b+ q1 v; J$ a8 |
//当象素颜色为指定的透明值时,即将该点从区域中剪裁掉。; e" Q( U6 g$ p- Y+ h
for (nWidth = 0;nWidth <= rectWnd.Width()-1;nWidth++), s8 H7 b, e1 w2 X2 T+ N- k
{8 S+ W2 E7 n' X5 _
for (nHeight = 0;nHeight <= rectWnd.Height();nHeight++)
# E: h% X( W& N5 [ {- N5 n, E# O+ l4 f, q6 x5 e
color = memDC.GetPixel(nWidth,nHeight);
# M( `! ]# @0 h- b$ j* f // 当象素是白色时,去掉该点
0 ~. _* @+ y% Q6 f if (color == RGB(255,255,255))* v+ B; r3 E# E! {
{; v8 l, \! P0 A$ S7 O: ]0 G D
//象素颜色为指定的透明色,创建透明"微区域"
9 U, f8 O- ^4 O1 L' S. Z' u; X2 \ rgnTemp.CreateRectRgn(nWidth,nHeight,nWidth+1,nHeight+1);
' X5 N" F, V6 ]! W* ]) B //"扣像",从完整的区域中"扣除"透明的"微区域"
6 |/ I. r; y/ M4 Y1 O( y1 @ rgnWnd.CombineRgn(&rgnWnd,&rgnTemp,RGN_XOR);& D, j f. r% G, f ]/ Y2 W
//删除刚创建的透明"微区域",释放系统资源+ S' C& M' j9 E4 Q
rgnTemp.DeleteObject();
6 J* a% R; s7 o9 k' i5 B2 }7 C }
) g& H, q' V- ^2 |) s. I }
) H$ p5 T6 C3 S}* B& R5 @) ?' u& H6 i) j( w2 _
memDC.SelectObject(pOldBitmap);
" q* `+ h, t/ \9 Z. [SetWindowRgn((HRGN)rgnWnd,TRUE); //用最终设定窗口的显示区域为指定区域 & h9 z4 g% p3 z+ T2 O" ]
' @# j8 y, H0 A1 D5 v1 l+ ` 为了最终显示透明效果的窗体,还需要重置系统默认的背景擦除操作,即添加WM_ERASEBKGND消息处理过程,在其中实现背景位图的显示功能,这一步可以借助ClassWizard来实现。. A8 T" R: d9 }$ d3 Y E
! c; e7 B0 a' [
最后需要注意的是,为了使异形窗体应用程序能够正常地脱动和退出,需要处理窗体的WM_NCHITTEST、WM_CHAR等消息,这部分内容本书中其它实例已经作过介绍,这里就不再赘述了,感性趣的读者朋友可以参考相关实例 |
|