我们最近升级到 VS 2012,同时升级到 .NET Framework 4.5。这引入了一个与 WinForms MenuStrip 控件相关的奇怪且令人讨厌的错误。同样的错误也出现在针对 .NET Framework 4.0 的应用程序中,因为 4.5 的安装程序显然也更新了 4.0 的部分内容。因此,一个完美工作的代码模式现在因为框架升级而被打破。
问题描述:
我有一个带有 MenuStrip 的 Form1。对于其中一个下拉项,事件处理程序打开另一个 Form2(不一定是子项,只是另一个 Form)。如果用户右键单击新的 Form2 或其子控件之一,这会触发 ContextMenuStrip 的 Show(),则原始 Form1 会再次弹出到前台。
这与 Form2 中的所有其他 UI 操作无关。可以调整大小、移动、最小化、最大化 Form2、在控件之间切换、输入文本等。 Form1 的 MenuStrip 似乎记得它导致 Form2 的打开并在第一次右键单击时获得焦点。
我正在尝试不同的方法,但到目前为止找不到解决方法。星座并不少见,可能会影响周围的许多 WinForms 应用程序。因此,我决定在这里发布它,因为可行的解决方案可能会引起普遍关注。如果有人知道解决方法或至少对我有一些线索,我会很高兴。
我能够在以下代码中提炼出它的精髓。它应该可以在任何安装了 .NET 4.5 的机器上重现,并且在分别针对 4.0 和 4.5 时发生 - 但不是在 3.5 或更低版本上。
using System;
using System.Windows.Forms;
namespace RightClickProblem {
static class Program {
[STAThread]
static void Main() {
// Construct a MenuStrip with one item "Menu" with one dropdown-item "Popup"
var mainMenu = new MenuStrip();
var mainItem = new ToolStripMenuItem("Menu");
mainItem.DropDownItems.Add(new ToolStripMenuItem("Popup", null, Popup));
mainMenu.Items.Add(mainItem);
// Create form with MenuStrip and Show
var form1 = new Form();
form1.Controls.Add(mainMenu);
Application.Run(form1);
}
private static void Popup(object sender, EventArgs e) {
// Create a form with a right click handler and show
var form2 = new Form();
var contextMenu = new ContextMenuStrip();
contextMenu.Items.Add("Just an item...");
form2.ContextMenuStrip = contextMenu;
form2.Show();
// Problem: contextMenu.Show() will give focus to form1
}
}
}
编辑:我花了一些时间逐步浏览 .NET Framework 源代码,发现根本原因很可能在 System.Windows.Forms.ToolStripManager 中。在那里,微软正在使用消息过滤器来跟踪窗口激活,这在某种程度上是为 MenuStrip 错误实现的。
与此同时,我还发现 Microsoft 已经在一个修补程序中解决了这个问题(参见http://support.microsoft.com/kb/2769674),并希望这将在未来的 .NET Framework 4.5 更新中找到方法。
不幸的是,很难强制将此修补程序应用于所有客户端计算机。因此,仍然非常感谢可行的解决方法。到目前为止,我自己一直无法找到切实可行的解决方案......
编辑#2:原始 KB 编号适用于 Win8,但在 KB 2756203 下有类似的适用于 Win7 和 Vista 的修补程序。如果有人可以使用它,请在此处找到下载的修补程序:http: //thehotfixshare.net/board/index.php ?autocom=downloads&showfile=15569。我们对其进行了测试,它确实解决了问题。如果我们很快就找不到解决方法,我们将使用修补程序。
编辑#3:对 spajce 提出的已接受解决方案的评论显然,在任何
ContextMenu
上调用 Show()将说服原始 MenuStrip 忘记它对焦点的要求。这可以通过某种方式完成,以便虚拟 ContextMenu 甚至不会显示在屏幕上。我找到了在任何 Form 的构造函数中插入以下代码段的最短且最容易实现的方法:
public partial class Form1 : Form {
public Form1() {
InitializeComponent();
using (var dummyMenu = new ContextMenuStrip()) {
dummyMenu.Items.Add(new ToolStripMenuItem());
dummyMenu.Show(Point.Empty);
}
}
}
因此,打开的每个表单都会清除 ToolStripMenu 系统的损坏状态。不妨将此代码放在像 FormHelper.FixToolStripState() 这样的静态方法中,或者将其放在模板 Form 的 OnCreateControl(...) 中并从中继承所有 Forms(无论如何我们幸运地做了什么)。