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中有你想要的答案,我就补多少了,不过要注意类型的转化和字符

集的转化。

下面我把我转化好的函数给大家贴上来,以飨读者。

public class NativeMethods

{

/// <summary>

/// 带有外边框和标题的windows的样式

/// </summary>

public const long WS_CAPTION = 0X00C0000L;

// public const long WS_BORDER = 0X0080000L;

/// <summary>

/// window 扩展样式 分层显示

/// </summary>

public const long WS_EX_LAYERED = 0x00080000L;

/// <summary>

/// 带有alpha的样式

/// </summary>

public const long LWA_ALPHA = 0x00000002L;

/// <summary>

/// 颜色设置

/// </summary>

public const long LWA_COLORKEY = 0x00000001L;

/// <summary>

/// window的基本样式

/// </summary>

public const int GWL_STYLE = -16;

/// <summary>

/// window的扩展样式

/// </summary>

public const int GWL_EXSTYLE = -20;

/// <summary>

/// 设置窗体的样式

/// </summary>

/// <param name="handle">操作窗体的句柄</param>

/// <param name="oldStyle">进行设置窗体的样式类型.</param>

/// <param name="newStyle">新样式</param>

[System.Runtime.InteropServices.DllImport("User32.dll")]

public static extern void SetWindowLong(IntPtr handle, int oldStyle, long newStyle);

/// <summary>

/// 获取窗体指定的样式.

/// </summary>

/// <param name="handle">操作窗体的句柄</param>

/// <param name="style">要进行返回的样式</param>

/// <returns>当前window的样式</returns>

[System.Runtime.InteropServices.DllImport("User32.dll")]

public static extern long GetWindowLong(IntPtr handle, int style);

/// <summary>

/// 设置窗体的工作区域.

/// </summary>

/// <param name="handle">操作窗体的句柄.</param>

/// <param name="handleRegion">操作窗体区域的句柄.</param>

/// <param name="regraw">if set to <c>true</c> [regraw].</param>

/// <returns>返回值</returns>

[System.Runtime.InteropServices.DllImport("User32.dll")]

public static extern int SetWindowRgn(IntPtr handle, IntPtr handleRegion, bool regraw);

/// <summary>

/// 创建带有圆角的区域.

/// </summary>

/// <param name="x1">左上角坐标的X值.</param>

/// <param name="y1">左上角坐标的Y值.</param>

/// <param name="x2">右下角坐标的X值.</param>

/// <param name="y2">右下角坐标的Y值.</param>

/// <param name="width">圆角椭圆的width.</param>

/// <param name="height">圆角椭圆的height.</param>

/// <returns>hRgn的句柄</returns>

[System.Runtime.InteropServices.DllImport("gdi32.dll")]

public static extern IntPtr CreateRoundRectRgn(int x1, int y1, int x2, int y2, int width, int height);

/// <summary>

/// Sets the layered window attributes.

/// </summary>

/// <param name="handle">要进行操作的窗口句柄</param>

/// <param name="colorKey">RGB的值</param>

/// <param name="alpha">Alpha的值,透明度</param>

/// <param name="flags">附带参数</param>

/// <returns>true or false</returns>

[System.Runtime.InteropServices.DllImport("User32.dll")]

public static extern bool SetLayeredWindowAttributes(IntPtr handle, ulong colorKey, byte alpha, long flags);

}

下面的问题就是如何进行操作了,首先在进行嵌入WinForm控件的WPF窗体中添加一个Load事件,在事件中添加如下代码:

// 获取窗体句柄

IntPtr hwnd = new System.Windows.Interop.WindowInteropHelper(this).Handle;

// 获得窗体的 样式

long oldstyle = 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>

/// Handles the SizeChanged event of the DesktopShell control.

/// </summary>

/// <param name="sender">The source of the event.</param>

/// <param name="e">The <see cref="System.Windows.SizeChangedEventArgs"/> instance containing the event data.</param>

private void DesktopShell_SizeChanged(object sender, SizeChangedEventArgs e)

{

// 获取窗体句柄

IntPtr hwnd = new System.Windows.Interop.WindowInteropHelper(this).Handle;

// 创建圆角窗体

NativeMethods.SetWindowRgn(hwnd,NativeMethods.CreateRoundRectRgn(0, 0, Convert.ToInt32(this.ActualWidth), Convert.ToInt32(this.ActualHeight), 12, 12), true);

}

至此问题就全部解决了,如果有朋友有疑问,可以留言,能帮助到你,我很荣幸。

时间: 2024-10-04 08:41:52

WPF中不规则窗体与WindowsFormsHost控件的兼容问题完美解决方案的相关文章

wpf PNG图形窗体 使用webbrowser控件

wpf ,PNG图形半透明窗体 ,使用webbrowser控件 附件:http://files.cnblogs.com/xe2011/WpfApplication1_webbrowser_transparent.rar MainWindow1.XAML <Window x:Name="MainWindow1" x:Class="WpfApplication1.MainWindow" xmlns="http://schemas.microsoft.co

WPF布局之让你的控件随着窗口等比放大缩小,适应多分辨率满屏填充应用

一直以来,我们设计windows应用程序,都是将控件的尺寸定好,无论窗体大小怎么变,都不会改变,这样的设计对于一般的应用程序来说是没有问题的,但是对于一些比较特殊的应用,比如有背景图片的,需要铺面整个屏幕,由于存在多种不同的分辨率,所以会出现布局混乱的情况.今天我们来看看WPF中如何让我们的控件也随着分辨率放大缩小.下面来写一个例子看看效果吧~  一.普通布局中的问题 这里我们写一个简单的页面,新建WPF项目,在MainWindow里面添加按钮,如下图: 这个页面很简单,只有三个按钮,我们想的是

WPF设计の不规则窗体

原文:WPF设计の不规则窗体 我们在工作中,经常会需要画一些不规则的窗体,现在总结如下. 一.利用VisualBrush实现.这依赖于VisualBrush的特性,任何控件可以作为画刷,而画刷又可以作为背景. 此种方法可以用于实现一些文字窗体等.(注意设置窗体的透明属性) 例如: <Window x:Class="IconFontTest.Window1" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/prese

TFORMer窗体打印条形码控件

TFORMer-能提供专业的打印方案-能用于大多数应用程序中,如:报告,标签,邮件,商业窗体等等.TFORMer能满足您的标签应用程序, 条形码,和其他报表以及窗体打印的需要. 产品特征: 通过运行时模块(ActiveX and DLL), 软件开发可以整合强大的TFORMer打印引擎到到他们的应用软件中. 产品具体特征如下所示: 可通过预定义的标签模板立即应用于工业和办公等环境. 立即打印-动态数据自动录入(独立于数据库). 免费整合设计器软件 (无需程序技巧也可完成设计). 漂亮的用户界面支

WPF自定义控件与样式(10)-进度控件ProcessBar自定义样

一.前言 申明:WPF自定义控件与样式是一个系列文章,前后是有些关联的,但大多是按照由简到繁的顺序逐步发布的等,若有不明白的地方可以参考本系列前面的文章,文末附有部分文章链接. 本文主要内容: ProcessBar自定义标准样式: ProcessBar自定义环形进度样式: 二.ProcessBar标准样式 效果图: ProcessBar的样式非常简单: <!--ProgressBar Style--> <Style TargetType="ProgressBar" x

[C# 学习]窗体间调用控件

一.方法1: 假如有两个窗体,Form_A和Form_B,每个窗体里都有一个按键,Button_A和Button_B,要实现单击Button_A显示窗体B,那么窗体A中Buttom_A的单击事件的程序应该是: private void button_A_Click(object sender, EventArgs e) { Form_B f = new Form_B(); f.Show(); } 如果希望单击窗体B中的按键Button_B,实现改变窗体A的背景色,那么你需要做: 1. Form_

如何让窗体大小随着控件的大小变化而变化

form的autosize=true,formborderstyle=fixedsingle tablelayoutpanel及其里面的容器的dock=fill,autosize=true label中的文字比较贴着窗体边缘时可以通过padding属性来调整 因定宽度则可以使用tablelayoutpanel中的列固定 atablelayoutpanel中的行也可以固定,而不固定行直接按百分比来设置 如何让窗体大小随着控件的大小变化而变化

VC窗体透明而控件不透明以及Static文本背景透明方法

出自http://my.oschina.net/ypimgt/blog/60951 优点:    1.Dialog 窗体完全透明.     2. 窗体上的控件不透明. DC 绘制的图形不透明.     3. 拖动窗体上用 DC 绘制的图形可以移动窗体. 缺点:     1. 窗体设置透明使用是掩码颜色,所以在窗体上用 DC 绘图的过程当中如果采用了和透明掩码颜色相同的颜色将不会显示出来.     2. 如果将 Border 属性设置成了 NONE ,那么想通过响应 WM_NCHITEST 消息来

[译]- 6-1 排列窗体上的控件(Laying Out Widgets on a Form)

 排列窗体上的控件(Laying Out Widgets on a Form) 中英文对照:form(窗体),layout(布局或者排列,意思是进行窗体上控件的排列的过程,如大小位置等) absolute positioning(绝对位置定位),manual layout(手工布局), layout managers(布局管理器) Qt中有三种方式对窗体上的控件进行布局管理:绝对位置定位(absolute positioning),手工布局(manual layout),布局管理器(layout