2

在标题中很难解释这一点,如果有人想更改它就可以了。

我有一种情况,在 WPF 中,我创建了一个对程序员透明的“隐藏”窗口。我的意思是这个窗口是在静态构造函数中创建的,隐藏并移动到屏幕之外,它的宽度和高度为 0。这是因为我正在使用这个窗口来执行一些互操作操作并允许所有的处理程序有人可能需要的 WndProcs 覆盖(有一个委托列表处理应该覆盖 WndProc 的方法)。

希望您理解我所说的(这并不容易),我的问题是,当我创建一个 WPF 项目并启动它时,如果我关闭主窗口(这不是对程序员透明地创建的窗口),我希望我的应用程序关闭。但是,使用我创建的代码不会发生这种情况,除非我使用 Application.Current.Shutdown();

有没有办法在不调用该方法的情况下解决这个问题?我想要一种透明的方式,其他程序员甚至不应该注意到(它是一个库,不应该以这种方式改变工作程序的行为)。

感谢您的任何建议,在这里您可以看到一些代码片段:

lib创建的窗口

public class InteropWindow : Window
{
    public HwndSource Source { get; protected set; }

    private static InteropWindow _Instance;

    static InteropWindow()
    {
        _WndProcs = new LinkedList<WndProcHandler>();
        _Instance = new InteropWindow();
    }

    private static WindowInteropHelper _InteropHelper;
    public static WindowInteropHelper InteropHelper
    {
        get
        {
            if (_InteropHelper == null)
            {
                _InteropHelper = new WindowInteropHelper(_Instance);
                _InteropHelper.EnsureHandle();
            }
            return _InteropHelper;
        }
    }

    public static IntPtr Handle { get { return InteropHelper.Handle; } }

    private InteropWindow()
    {
        Opacity = 0.0;
        //We have to "show" the window in order to obtain hwnd to process WndProc messages in WPF
        Top = -10;
        Left = -10;
        Width = 0;
        Height = 0;
        WindowStyle = WindowStyle.None;
        ShowInTaskbar = false;
        ShowActivated = false;
        Show();
        Hide();
    }

    private static LinkedList<WndProcHandler> _WndProcs;
    public static void AddWndProcHandler(WndProcHandler handler)
    {
        _WndProcs.AddLast(handler);
    }
    public static void RemoveWndProcHandler(WndProcHandler handler)
    {
        _WndProcs.Remove(handler);
    }

    private static IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
    {
        IntPtr result = IntPtr.Zero;
        foreach (WndProcHandler handler in _WndProcs)
        {
            IntPtr tmp = handler(hwnd, msg, wParam, lParam, ref handled);
            if (tmp != IntPtr.Zero)
            {
                if (result != IntPtr.Zero)
                    throw new InvalidOperationException(string.Format("result should be zero if tmp is non-zero:\nresult: {0}\ntmp: {1}", result.ToInt64().ToString(), tmp.ToInt64().ToString()));
                result = tmp;
            }
        }
        return result;
    }

    protected override void OnSourceInitialized(EventArgs e)
    {
        base.OnSourceInitialized(e);
        Source = PresentationSource.FromVisual(this) as HwndSource;
        Source.AddHook(WndProc);
        OnWindowInitialized(null, e);
    }

    protected override void OnClosed(EventArgs e)
    {
        base.OnClosed(e);
        if (Source != null)
            Source.RemoveHook(WndProc);
        OnWindowClosed(null, e);
    }

    private static void OnWindowInitialized(object sender, EventArgs e)
    {
        if (WindowInitialized != null) WindowInitialized(sender, e);
    }

    private static void OnWindowClosed(object sender, EventArgs e)
    {
        if (WindowClosed != null) WindowClosed(sender, e);
    }

    public static event EventHandler WindowInitialized;

    public static event EventHandler WindowClosed;
}

使用 wpf 创建的普通窗口(从项目创建的基本窗口)

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        ExClipboard.ClipboardUpdate += new RoutedEventHandler(ExClipboard_ClipboardUpdate);
        Closed += new EventHandler(MainWindow_Closed);
    }

    private void MainWindow_Closed(object sender, EventArgs e)
    {
        //InteropWindow.Dispose();
        App.Current.Shutdown(0);
    }
}

更新1:

为了回答你的答案,不,我想避免程序员使用我的库进行任何干预,所以理想的解决方案是在我的库中订阅一些 Application.Exit 事件并关闭我的窗口,显然我不能使用 Application。退出,因为应用程序由于我的窗口未关闭而未关闭

也许有一种方法可以计算属于应用程序的所有窗口?我也可以用它做点什么

4

3 回答 3

3

如果你有一个主窗口,你不能将Application.ShutdownMode设置为OnMainWindowClose吗?

默认值为 OnLastWindowClose,这很可能是您看到此行为的原因。

于 2011-07-16T23:09:04.903 回答
2

这是一个便宜的黑客,但我认为这可能会实现你所追求的......

在您的库中,您需要引用 xaml 依赖项(PresentationCore、PresentationFramework、System.Xaml 和 WindowsBase)

在你的库的静态构造函数中,你可以添加类似的东西

Application.Current.MainWindow.Closed += new EventHandler(MainWindow_Closed);

static void MainWindow_Closed(object sender, EventArgs e)
{
    Dispose();
}

dispose 在哪里关闭您的窗口 (_Instance.Close()) 并处理任何其他清理调用

于 2011-07-17T20:55:04.773 回答
1

从概念上讲,唯一想到的是事件消息通知服务,其中,第二个窗口正在侦听或等待消息关闭,第一个窗口发送消息使其关闭。这也需要使用 MVVM 模式。我不完全确定这一点,我也不确定这是否符合你不让其他程序员知道的想法。

这是一篇关于它的博客文章: 在 WPF MVVM 应用程序中发送通知

于 2011-07-16T21:27:41.003 回答