为什么关闭最后一个子窗口会使其父窗口最小化?

时间:2012-05-25 15:53:31

标签: c# wpf

我有以下简单的wpf应用程序:

的App.xaml:

<Application x:Class="TestWpf2.App"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
</Application>

App.xaml.cs:

public partial class App : Application
{
    protected override void OnStartup(StartupEventArgs e)
    {
        var parentWindow = new Window();
        parentWindow.Show();

        var childWindow1 = new Window { Owner = parentWindow };
        childWindow1.Show();

        var childWindow2 = new Window { Owner = parentWindow };
        childWindow2.Show();
    }
}

该应用程序导致3个窗口出现在屏幕上。如果运行应用程序并关闭两个子窗口,父窗口将最小化到任务栏。如果您注释掉childWindow2.show(),运行应用程序,并关闭单个子窗口,则父窗口不会最小化到任务栏。

我可以添加以下代码来解决此问题:

childWindow1.Closing += delegate(object sender, CancelEventArgs ex)
{
    (sender as Window).Owner = null;
};

但我不想使用这样的黑客,我想了解为什么会出现这个问题。

为什么会这样?

3 个答案:

答案 0 :(得分:17)

这是WPF未记录的功能(Bug)

7年多以前,已向Microsoft报告此错误。

Modal dialog on top of non-modal window sends main window to back.

  

WPF团队最近审核了这个问题,但不会   解决这个问题,因为此时团队正在关注这些问题   影响最多的WPF开发人员。

让我们稍微看看这个未记录的功能的行为。

它没有最小化,只是在一个窗口后面(在调试模式下,在visual studio窗口后面)

证明的步骤:

  • 运行此应用程序。

  • 最小化所有其他窗口(例如:visual studio和所有其他窗口)。只在屏幕上保留这三个窗口。

  • 关闭这两个孩子,父母仍在Normal

测试此代码以获得进一步证明: StateChange不会发射。

        protected override void OnStartup(StartupEventArgs e)
        {
            var parentWindow = new Window() { Title = "Parent" };
            parentWindow.StateChanged += (sender, ep)=>
            {
                var state = ((Window)sender).WindowState;
            };
            parentWindow.Show();

            var childWindow1 = new Window { Owner = parentWindow, Title = "Child1" };
            childWindow1.Show();

            var childWindow2 = new Window { Owner = parentWindow, Title = "Child2" };
            childWindow2.Show();
        }

成为TopMost会阻止这种情况发生

如果父窗口是TopMost,则不会发生这种情况。但在许多情况下,这可能不是一种选择。

parentWindow.Topmost = true;

解决方法

激活child2关闭时的父级。

childWindow2.Closed += (a, b) => { parentWindow.Activate(); };

答案 1 :(得分:3)

这是因为在使用所有者设置显示子窗口后,您无法使用父窗口。只是在孩子在屏幕上时尝试访问父窗口,它不会让你看到我的意思。

如果您未指定所有者,则不会发生此行为。

答案 2 :(得分:1)

我从来没有弄清楚导致这种情况发生的原因!最后,我编写了在关闭任何子窗口后调用的代码:

// App inherits from Application, and has a Window property called MainWindow
// and a List<Window> property called OpenWindows.
if (!App.OpenWindows.Any())
    App.ParentWindow.Activate();

因此,如果最后一个子窗口关闭(使App.OpenWindows.Any()为false),则会激活父窗口。这不会导致闪烁(主窗口最小化,然后最大化,例如。)

这不是最好的解决方案。我不会关闭这个问题,希望有人可以解释为什么WPF有这个功能!