WPF中不規矩窗體與WindowsFormsHost控件兼容成績的處理辦法。本站提示廣大學習愛好者:(WPF中不規矩窗體與WindowsFormsHost控件兼容成績的處理辦法)文章只能為提供參考,不一定能成為您想要的結果。以下是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,把最外層的那層邊框被去失落了。那末須要甚麼呢,思緒是有了,對吧,那末就行為吧,谷歌 和百度一通,發明還真有很多例子,c++的例子最周全,可以參考一下。那末就整頓了一下須要這些函數:
SetWindowLong 設置值window的款式
GetWindowLong 獲得window的款式
SetWindowRgn 設置window的任務區
CreateRoundRectRgn 創立帶有圓角的區域
SetLayeredWindowAttributes 設置條理窗體,停止通明度的設置
直接百度,百科有對他們的具體說明,不外給出的是C++的說明,那末須要你對C++的器械停止轉化成C#的器械,有關C#若何挪用C++的DLL文件,百度和谷歌中有你想要的謎底,我就補若干了,不外要留意類型的轉化和字符
集的轉化。
上面我把我轉化好的函數給年夜家貼下去,以飨讀者。
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);
}
至此成績就全體處理了,願望本文所述對年夜家的WPF法式設計有所贊助。