我有这种情况。
Application.OpenForms不会返回正确的结果。即Application.OpenForms.Count = 0
总是..
获取表单的目的是获取表单的所有者,以便我可以将所有者作为MessageBox.Show()
函数的参数传递。
答案 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);
}
}