WPF中不规则窗体与WindowsFormsHost控件兼容问题的解决方法
本文实例讲述了WPF中不规则窗体与WindowsFormsHost控件兼容问题的解决方法。分享给大家供大家参考。具体方法如下:
这里首先说明一下,有关WPF中不规则窗体与WindowsFormsHost控件不兼容的问题,网上给出的很多解决方案不能满足所有的情况,是有特定条件的,比如有一篇《WPF中不规则窗体与WebBrowser控件的兼容问题解决办法》(感兴趣的朋友可以自己百度一下这篇文章)。该网友的解决办法也是别出心裁的,为什么这样说呢,他的webBrowser控件的是单独放在一个Form中,让这个Form与WPF中的一个Bord控件进行关联,进行同步移动,但是在移动的时候会出现闪烁,并且还会出现运动的白点,用户体验肯定不好。
OK,绕了一大圈,还是言归正传吧,为什么会出现该问题呢,是什么原因导致在WPF中设置了透明窗体之后,嵌入WinForm中的控件会显示不了呢。一开始我以为是没有正常加载,还要我有UISPY,通过这个软件,我捕获了一下当前运行的程序,发现我在WPF中内嵌的WinForm控件已经加载上了,只是没有看到而已罢了。很郁闷啊。
悲催的程序,头疼啊,是什么原因导致的呢,网上查资料,找到了http://msdn.microsoft.com/zh-cn/library/aa970688.aspx,让我了解了不少知识。由于项目要用到透明窗体还要制作圆角窗体,说以本来打算不改变WPF中对window的设置,即不改变WindowStyle=“None”和AllowTransparent=“True”这些设置,想在在WindowsFormsHost上做一些设置,发现这条路走不通。浪费了不少时间。
此路不通只有换思路了,那么把AllowTransparent=“false”,然后就可以显示,呵呵……当然还要修改啊,WPF的窗体多难看啊,外边有一个边框。怎么搞去啊,怎样办,这也是一个问题啊。想用WPF的特性,悲剧了,好像没有相关的方法啊。
OK,路还是有的,程序员就是来解决办法的,怎么办,只能调用Windows的API,把最外层的那层边框被去掉了。那么需要什么呢,思路是有了,对吧,那么就行动吧,google和百度一通,发现还真有不少例子,c++的例子最全面,可以参考一下。那么就整理了一下需要这些函数:
SetWindowLong 设置值window的样式
GetWindowLong 获取window的样式
SetWindowRgn 设置window的工作区
CreateRoundRectRgn 创建带有圆角的区域
SetLayeredWindowAttributes 设置层次窗体,进行透明度的设置
直接百度,百科有对他们的详细解释,不过给出的是C++的解释,那么需要你对C++的东西进行转化成C#的东西,有关C#如何调用C++的DLL文件,百度和google中有你想要的答案,我就补多少了,不过要注意类型的转化和字符
集的转化。
下面我把我转化好的函数给大家贴上来,以飨读者。
publicclassNativeMethods { ///<summary> ///带有外边框和标题的windows的样式 ///</summary> publicconstlongWS_CAPTION=0X00C0000L; //publicconstlongWS_BORDER=0X0080000L; ///<summary> ///window扩展样式分层显示 ///</summary> publicconstlongWS_EX_LAYERED=0x00080000L; ///<summary> ///带有alpha的样式 ///</summary> publicconstlongLWA_ALPHA=0x00000002L; ///<summary> ///颜色设置 ///</summary> publicconstlongLWA_COLORKEY=0x00000001L; ///<summary> ///window的基本样式 ///</summary> publicconstintGWL_STYLE=-16; ///<summary> ///window的扩展样式 ///</summary> publicconstintGWL_EXSTYLE=-20; ///<summary> ///设置窗体的样式 ///</summary> ///<paramname="handle">操作窗体的句柄</param> ///<paramname="oldStyle">进行设置窗体的样式类型.</param> ///<paramname="newStyle">新样式</param> [System.Runtime.InteropServices.DllImport("User32.dll")] publicstaticexternvoidSetWindowLong(IntPtrhandle,intoldStyle,longnewStyle); ///<summary> ///获取窗体指定的样式. ///</summary> ///<paramname="handle">操作窗体的句柄</param> ///<paramname="style">要进行返回的样式</param> ///<returns>当前window的样式</returns> [System.Runtime.InteropServices.DllImport("User32.dll")] publicstaticexternlongGetWindowLong(IntPtrhandle,intstyle); ///<summary> ///设置窗体的工作区域. ///</summary> ///<paramname="handle">操作窗体的句柄.</param> ///<paramname="handleRegion">操作窗体区域的句柄.</param> ///<paramname="regraw">ifsetto<c>true</c>[regraw].</param> ///<returns>返回值</returns> [System.Runtime.InteropServices.DllImport("User32.dll")] publicstaticexternintSetWindowRgn(IntPtrhandle,IntPtrhandleRegion,boolregraw); ///<summary> ///创建带有圆角的区域. ///</summary> ///<paramname="x1">左上角坐标的X值.</param> ///<paramname="y1">左上角坐标的Y值.</param> ///<paramname="x2">右下角坐标的X值.</param> ///<paramname="y2">右下角坐标的Y值.</param> ///<paramname="width">圆角椭圆的width.</param> ///<paramname="height">圆角椭圆的height.</param> ///<returns>hRgn的句柄</returns> [System.Runtime.InteropServices.DllImport("gdi32.dll")] publicstaticexternIntPtrCreateRoundRectRgn(intx1,inty1,intx2,inty2,intwidth,intheight); ///<summary> ///Setsthelayeredwindowattributes. ///</summary> ///<paramname="handle">要进行操作的窗口句柄</param> ///<paramname="colorKey">RGB的值</param> ///<paramname="alpha">Alpha的值,透明度</param> ///<paramname="flags">附带参数</param> ///<returns>trueorfalse</returns> [System.Runtime.InteropServices.DllImport("User32.dll")] publicstaticexternboolSetLayeredWindowAttributes(IntPtrhandle,ulongcolorKey,bytealpha,longflags); }
下面的问题就是如何进行操作了,首先在进行嵌入WinForm控件的WPF窗体中添加一个Load事件,在事件中添加如下代码:
//获取窗体句柄 IntPtrhwnd=newSystem.Windows.Interop.WindowInteropHelper(this).Handle; //获得窗体的样式 longoldstyle=NativeMethods.GetWindowLong(hwnd,NativeMethods.GWL_STYLE); //更改窗体的样式为无边框窗体 NativeMethods.SetWindowLong(hwnd,NativeMethods.GWL_STYLE,oldstyle&~NativeMethods.WS_CAPTION); //SetWindowLong(hwnd,GWL_EXSTYLE,oldstyle&~WS_EX_LAYERED); //1|2<<8|3<<16 r=1,g=2,b=3详见winuse.h文件 //设置窗体为透明窗体 NativeMethods.SetLayeredWindowAttributes(hwnd,1|2<<8|3<<16,0,NativeMethods.LWA_ALPHA); //创建圆角窗体 12这个值可以根据自身项目进行设置 NativeMethods.SetWindowRgn(hwnd,NativeMethods.CreateRoundRectRgn(0,0,Convert.ToInt32(this.ActualWidth),Convert.ToInt32(this.ActualHeight),12,12),true);
还有就是窗体大小改变之后还要重画圆角窗体,否则出现很不理想的显示效果,添加如下事件代码,解决窗体大小改变的时候,重画窗体的圆角区域:
///<summary> ///HandlestheSizeChangedeventoftheDesktopShellcontrol. ///</summary> ///<paramname="sender">Thesourceoftheevent.</param> ///<paramname="e">The<seecref="System.Windows.SizeChangedEventArgs"/>instancecontainingtheeventdata.</param> privatevoidDesktopShell_SizeChanged(objectsender,SizeChangedEventArgse) { //获取窗体句柄 IntPtrhwnd=newSystem.Windows.Interop.WindowInteropHelper(this).Handle; //创建圆角窗体 NativeMethods.SetWindowRgn(hwnd,NativeMethods.CreateRoundRectRgn(0,0,Convert.ToInt32(this.ActualWidth),Convert.ToInt32(this.ActualHeight),12,12),true); }
至此问题就全部解决了,希望本文所述对大家的WPF程序设计有所帮助。