Application.OpenForms.Count = 0总是

时间:2010-09-20 12:40:58

标签: c# .net winforms

我有这种情况。 Application.OpenForms不会返回正确的结果。即Application.OpenForms.Count = 0总是..

获取表单的目的是获取表单的所有者,以便我可以将所有者作为MessageBox.Show()函数的参数传递。

2 个答案:

答案 0 :(得分:38)

Windows窗体中存在一个错误,它使表单从Application.OpenForms集合中消失。在创建窗口后<_ em>分配ShowInTaskbar,FormBorderStyle,ControlBox,Min / MaximizedBox,RightToLeftLayout,HelpButton,Opacity,TransparencyKey,ShowIcon或MdiParent属性时,会发生这种情况。这些属性的特殊之处在于它们在本机CreateWindowEx()调用中被指定为样式标志。此示例表单演示了错误:

public partial class Form1 : Form {
    public Form1() {
        InitializeComponent();
        button1.Click += button1_Click;
    }
    private void button1_Click(object sender, EventArgs e) {
        Console.WriteLine(Application.OpenForms.Count);
        this.ShowInTaskbar = !this.ShowInTaskbar;
        Console.WriteLine(Application.OpenForms.Count);
    }
}

Windows窗体必须再次调用CreateWindowEx()以使更改的属性生效,并传递不同的样式标志。首先破坏原始窗口的副作用超出了非常明显的闪烁,其中之一是Application类失去了对窗体的跟踪,因为它看到窗口消失了。有了它在创建新窗口时不会将其添加回来的错误。通过仅在构造函数中设置属性来避免错误,在调用CreateWindowEx()之前运行的代码,而不是在任何事件处理程序中。

通常,由于此错误,请避免依赖OpenForms。通过其构造函数为需要显示消息框的类提供对表单实例的引用。 MessageBox通常正确地计算父窗口btw,它将选择活动窗口,并且99%的时间都是正确的。如果需要它从工作线程调用BeginInvoke(),请确保在构造函数中复制SynchronizationContext.Current并稍后调用其Post()方法。确保您的库也可以与其他GUI类库一起使用。

答案 1 :(得分:0)

当我使用ShowInTaskBar = true时,我遇到了问题。我通过使用Windows API而不是.Net属性解决了这个问题。 Application.OpenForms保持不变。

我不知道它是否可以作为一般解决方法使用SetWindowLong来更改属性,但它适用于ShowInTaskBar = true。

 public static class ShowInTaskBar {

    [DllImport("User32.dll")]
    private static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);
    [DllImport("User32.dll")]
    private static extern int GetWindowLong(IntPtr hWnd, int nIndex);

    [DllImport("user32.dll")]
    private static extern bool ShowWindow(IntPtr hWnd, int nCmdShow);

    private const int SW_HIDE = 0x00;
    private const int SW_SHOW = 0x05;

    private const int WS_EX_APPWINDOW = 0x40000;
    private const int GWL_EXSTYLE = -0x14;

    public static void ShowWindowInTaskbar(IntPtr pMainWindow) {
        SetWindowLong(pMainWindow, GWL_EXSTYLE, GetWindowLong(pMainWindow, GWL_EXSTYLE) | WS_EX_APPWINDOW);

        ShowWindow(pMainWindow, SW_HIDE);
        ShowWindow(pMainWindow, SW_SHOW);
    }

    public static void HideWindowFromTaskbar(IntPtr pMainWindow) {
        SetWindowLong(pMainWindow, GWL_EXSTYLE, GetWindowLong(pMainWindow, GWL_EXSTYLE) & ~WS_EX_APPWINDOW);

        ShowWindow(pMainWindow, SW_HIDE);
        ShowWindow(pMainWindow, SW_SHOW);
    }
}