完美PNG半透明窗体解决方案
当年Vista系统刚出来的时候,最吸引人的莫过于半透明磨砂的窗体界面了,迷倒了多少人。这个界面技术随即引发了编程界的一阵骚动,很多人都在问:如何实现这一界面效果?当然,在Vista下倒是很简单,系统本身支持,所以几乎不需要写一句代码,但是当时还是XP的天下,于是大家就可以研究在XP下如何实现这一效果。
最先实现的应该是桌面天气秀,还有笨笨钟,后来鱼鱼软件的鱼鱼桌面秀也成功在XP下模仿了Vista的侧边栏,的确,让人很激动,但是他们保密,问也问不到究竟是用了什么技术,记得当年大富翁论坛(http://www.delphibbs.com 比较著名的Delphi论坛)上还为此进行过讨论,最后有个ID叫小雨哥的人提供了一个方法(当然是不是他原创我不得而知,小雨哥目前在盛大网络,见过一面,无限膜拜中),下面是Delphi的实现代码:
01.var02.pt1, pt2 : TPoint;03.sz : TSize;04.bf : TBlendFunction;05.begin06. 07.bitmap:=tgpbitmap.Create(PNGFile);//这个PNGFile是具体的PNG图片路径08.pt1 := Point(left,top); //窗口做上角的坐标09.pt2 := Point(0, 0); //这个就不用说了,一看见(0,0)就应该明白了10.sz.cx := bitmap.GetWidth; //尺寸不要超过图像大小,不然窗口就什么都没有了,连个影子都没有11.sz.cy := bitmap.GetHeight; //同上12.bf.BlendOp := AC_SRC_OVER; //这些死记就行了13.bf.BlendFlags := 0; //同上14.if (nTran<0) or (nTran>255) then nTran:=255;15.bf.SourceConstantAlpha := nTran; //同上16.bf.AlphaFormat := AC_SRC_ALPHA; //同上17.DeleteObject(bmp); //前面就是在这里犯的错误,不然占用的内存会无限增大18.bitmap.GetHBITMAP(0,bmp); // HBITMAP是windows标准位图格式,支持透明,这里是从tgpbitmap 转化成 HBITMAP19.DeleteDC(DC);20.DC := CreateCompatibleDC(Canvas.Handle); 21.old_bmp := SelectObject(DC, bmp); 22.UpdateLayeredWindow(Handle, Canvas.Handle, @pt1, @sz, DC, @pt2,0, @bf,ULW_ALPHA);//调用UpdateLayeredWindow实现23.end;这个方法其实是生成一个PNG的窗体,我们知道,PNG图片是具有Alpha属性的,所以,如果PNG是半透明磨砂装的,那么生成的窗体也就是半透明磨砂装的,注意,上面的代码需要使用GDIPlus类uses gdipapi, gdipobj;
我们这里不讨论这段代码,这段代码是别人写的,看似非常完美,比如日期查询器的系统初始化界面就可以由上面的代码生成:

但是这段代码有个致命的问题,你可以试试在Form上面放一些控件,比如button,edit等,再次编译你会神奇的发现,所有的控件都不显示,这是怎么回事呢?通过查阅MSDN,我们发现问题出在UpdateLayeredWindow函数上。
MSDN中,关于该函数的Remarks中有这样一段说明:
The UpdateLayeredWindow function maintains the window's appearance on the screen. The windows underneath a layered window do not need to be repainted when they are uncovered due to a call to UpdateLayeredWindow, because the system will automatically repaint them. This permits seamless animation of the layered window.
大意是说,使用这个函数以后,下一层窗体不会再重新绘制,也就是说,窗体不会响应Onpaint事件来重绘所有控件,导致控件无法看见,但是实际上控件是存在的,你可以在响应位置上点击一下button,你会发现button依然会响应点击事件,但就是看不见。
这就头疼了,如果不能使用控件,或者说控件看不见,光一个窗体再好看有什么用呢?其实微软貌似是用这个函数做无缝连接动画用的,MSDN说的很清楚嘛:This permits seamless animation of the layered window.
嗯,好吧,既然这个方法不行,那就换一个吧,于是有人想到了使用2个窗体来解决。
2个窗体怎么解决呢?其实也很简单,一个窗体作为半透明的PNG放在后面,一个窗体作为放置控件的窗体放在前面,然后只要2个窗体同步移动就可以了,就拿日期查询器来说吧,登陆窗体:

这个窗体上下都有半透明的边框,上面也有控件显示,也许你不好理解,如果我分解一下:


怎么样?这样你就发现了,其实是2个窗体,后面一个背景窗体,前面一个Border:=none的控件窗体,然后两个窗体同步移动即可,这样我们就伪造了一个半透明的窗体。事实上,很多软件也是这么做的,包括有些带阴影的窗体也是同样原理。
至于同步移动,也很简单,处理下OnMove消息就可以了:
01.function WndNewProc(Wnd: HWND; uMsg: UINT; wPar: WPARAM; lPar: LPARAM): LRESULT; stdcall;02.var Rect: TRect;03.begin04.Result := 0;05.case uMsg of06.WM_LBUTTONDOWN: SendMessage(Wnd, WM_SYSCOMMAND, SC_MOVE+2, 0);07.else08.begin09.if ((uMsg = WM_MOVING) or (uMsg = WM_MOVE)) and GetWindowRect(Wnd, Rect) then10.SetWindowPos(ComponentForm.Handle, 0, Rect.Left, Rect.Top, 0, 0, SWP_NOSIZE);11.Result := DefWindowProc(Wnd, uMsg, wPar, lPar);12.end;13.end;14.end;我们假设放置控件的窗体名字叫ComponentForm,当我们鼠标左键按下并移动背景窗体的时候,控件窗体跟着同步移动即可。
当然,有时候我们可以直接用代码生成背景窗体,这样的话可以减少一些程序体积,生成背景窗体我们可以用一个函数叫CreateWindowEx,注意这里要用带Ex的,表示会有附加参数,我们只要把这个函数的第一个参数设置为WS_EX_LAYERED就可以了,他表示一个额外的层属性。
两个方法都可以,我们既可以直接用2个窗体,也可以用CreateWindowEx函数来生成背景窗体,效果是一样的,看个人喜好。
完美PNG半透明窗体解决方案的更多相关文章
- (一)C# Windows Mobile 半透明窗体
Windows Mobile,个人心中臻至完美的系统. 不忍自己对WM的钻研成果消逝,故留作纪念. 系列开篇,便是一个曾令自己困扰很久的问题:如何实现半透明窗体. 如果了解Win32编程,其实很简单. ...
- 翻译「C++ Rvalue References Explained」C++右值引用详解 Part8:Perfect Forwarding(完美转发):解决方案
本文为第八部分,目录请参阅概述部分:http://www.cnblogs.com/harrywong/p/cpp-rvalue-references-explained-introduction.ht ...
- delphi 半透明窗体类
{******************************************************************************* 半透明窗体控件 版本:1.0 功能说明 ...
- (二十二)c#Winform自定义控件-半透明窗体
前提 入行已经7,8年了,一直想做一套漂亮点的自定义控件,于是就有了本系列文章. 开源地址:https://gitee.com/kwwwvagaa/net_winform_custom_control ...
- nim_duilib(14)之xml配置半透明窗体控件不透明
before starting note 截至目前,我只能用xml写一些简单的布局和设置控件属性,循序渐进吧. 正在学习nim_duilib的xml的一些属性. xml配置半透明 GTAV中就有很多控 ...
- Delphi 半透明窗体,窗体以及控件透明度
很简单了 现在,适用所有控件和窗体: delphi设置窗口透明 form1.AlphaBlend :=true; //透明form1.AlphaBlendValue :=180; //透明度form1 ...
- 打造完美的xml技术解决方案(dom4j/xstream)
转: XML 技术是随着 Java 的发展而发展起来的.在 XML 出现之前对于简单的数据格式通常是存储在 ini 配置文件等文本文件中,复杂的格式则采用自定义的文件格式,因此对于每种文件格式都要有专 ...
- QT绘制半透明窗体(改写paintEvent,超级简单)
在派生类中重载QDialog的void paintEvent(QPaintEvent *)事件,在这个函数中加入以下代码 QPainter painter(this); QLinearGradi ...
- VB 半透明窗体
Private Declare Function GetWindowLong Lib "user32" Alias "GetWindowLongA" (ByVa ...
随机推荐
- java ee eclipse 配置 ssh框架
mvnDebug tomcat:run 这条命令主要用来远程测试,它会监听远程测试用的8000端口,在eclipse里打开远程测试后,它就会跑起来了,设断点,调试,一切都是这么简单. 0.如果是mav ...
- hibernate一对多关系映射(自身关联)
示例:一个类别(Category)下面有多个子类别,多个子类别属于同一个父类别. public class Category { private Integer id; private String ...
- uvalive4513
https://vjudge.net/problem/UVALive-4513 终于做出来了......... 各种sb错误,最后对拍出来了,还没改对..................... 快半天 ...
- mapreduce 读写lzo文件
1.读lzo文件 需要添加以下代码,并导入lzo相关的jar包 job.setInputFormatClass(LzoTextInputFormat.class); 2.写lzo文件 lzo格式默认 ...
- oracle表空间查询维护命令大全之二(undo表空间)
--undo表空间汇总 --查看全部的表空间名字 select name from v$tablespace; --创建新的UNDO表空间,并设置自己主动扩展參数; create undo table ...
- 往另外1个ListView中添加当前选中的项目
//往另外1个ListView中添加当前选中的项目 function AddSelItems(listview1:TListView;ListView2:TListView):Boolean; ...
- [转] 考验你的JavaScript底细
http://sentsin.com/ 尽管今日的JavaScript已经突飞猛进,但JS的许多特性仍然保留,以下题目并不是有意设坑,许多地方将验证你的JS底细,如果错了一半,请别告诉我你从事前端. ...
- 洛谷比赛 堕落的Joe
/*暴力50*/ #include<iostream> #include<cstdio> #include<cstring> #define maxn 100010 ...
- lucene 抛出的异常(分享)
1) too many boolean clauses异常 例如: String keyword=".......";//(keyword的长度太长) Query indexQue ...
- GitHub Desktop安装异常解决
为了更好的共同学习,共同进步,哥们推荐我使用GitHub记录自己每天的学习记录,当下很火的提供一个分布式的版本控制系统(Git)服务的网站,GitHub提供GitHub Desktop桌面程序方便协同 ...