上一篇讲到将WPF的窗口转为WinForm窗口需要注意的问题,这里给出了另一种解决方案,闲话不说,请看代码:
?
//============================================================================== // File Name : WpfModalDialogFixer.cs // // Copyright (C) 2010 GrapeCity Inc. All rights reserved. // // Distributable under LGPL license. // //============================================================================== // <fileinformation> // <summary> // This file defines the class WpfModalDialogFixer for solve the problem as below: // When showing a modal dialog which ShowTaskBar is false, first deactive the application the activate it again. // The modal dialog would be at behind of MainWindow. // </summary> // <author name="WinkingZhang" mail="Winking.Zhang@GrapeCity.com"/> // <seealso ref=""/> // <remarks> // // </remarks> // </fileinformation> // <history> // <record date="10/21/2010 6:07:04 PM" author="WinkingZhang" revision="1.00.000"> // Create the file and define the class. // </record> // </history> using System; using System.Threading; using System.Windows; using System.Windows.Interop; using System.Runtime.InteropServices; namespace GrapeCity.Windows { /// <summary> /// Represents the <see cref="WpfModalDialogFixer"/> class. /// </summary> public class WpfModalDialogFixer { [ThreadStatic] private static WpfModalDialogFixer _fixer; private static readonly object _rootSync = new object (); static WpfModalDialogFixer() { ComponentDispatcher.EnterThreadModal += new EventHandler(OnComponentDispatcherEnterThreadModal); ComponentDispatcher.LeaveThreadModal += new EventHandler(OnComponentDispatcherLeaveThreadModal); } private WpfModalDialogFixer() { } static void OnComponentDispatcherLeaveThreadModal( object sender, EventArgs e) { if (WpfModalDialogFixer.Current.Enable) { HwndSource src = HwndSource.FromVisual(Application.Current.MainWindow) as HwndSource; if (src != null ) { src.RemoveHook( new HwndSourceHook(OnComponentDispatcherThreadPreprocessMessage)); } } } static IntPtr OnComponentDispatcherThreadPreprocessMessage(IntPtr hwnd, int message, IntPtr wparam, IntPtr lparam, ref bool handled) { // Need take care the message: WM_SETFOCUS, and if now in Modal dialog mode. if (message == 0x7 && ComponentDispatcher.IsThreadModal) { bool actived = false ; foreach (Window w in Application.Current.Windows) { HwndSource src = HwndSource.FromVisual(w) as HwndSource; if (src != null /*&& src.Handle != hwnd*/ ) { if (IsWindowEnabled(src.Handle)) { w.Activate(); actived = true ; break ; } } } if (!actived) // in this case, caused by MessageBox.Show(...) { //TODO: how to handle? } handled = true ; // set handled to prevent default implement. } return IntPtr.Zero; } static void OnComponentDispatcherEnterThreadModal( object sender, EventArgs e) { if (WpfModalDialogFixer.Current.Enable) { HwndSource src = HwndSource.FromVisual(Application.Current.MainWindow) as HwndSource; if (src != null ) { src.AddHook( new HwndSourceHook(OnComponentDispatcherThreadPreprocessMessage)); } } } [DllImport( "user32.dll" )] [ return : MarshalAs(UnmanagedType.Bool)] private static extern bool IsWindowEnabled(IntPtr hwnd); public static WpfModalDialogFixer Current { get { if (_fixer == null ) { lock (_rootSync) { if (_fixer == null ) { _fixer = new WpfModalDialogFixer(); } } } return _fixer; } } public bool Enable { get ; set ; } } } |
点击查看更多内容
为 TA 点赞
评论
共同学习,写下你的评论
评论加载中...
作者其他优质文章
正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦