C# 在ToolStripMenuItem上添加右键单击上下文菜单,而不关闭菜单
我的应用程序顶部有一个菜单。C# 在ToolStripMenuItem上添加右键单击上下文菜单,而不关闭菜单,c#,winforms,menu,contextmenu,toolstripdropdown,C#,Winforms,Menu,Contextmenu,Toolstripdropdown,我的应用程序顶部有一个菜单。 菜单中的第一项(文件)具有通常的新建,打开…,保存,另存为…,以及打开最近的。最后一个是ToolStripDropDown,显示最近使用的文件名列表 我想在文件名中添加一个右键单击的上下文菜单(不是子菜单),该菜单将在上下文菜单中有一个项目,以从列表中删除右键单击的文件名 我将文件名加载到菜单中,如下所示: private void mnuFile_DropDownOpened(object sender, EventArgs e) { foreach (s
菜单中的第一项(文件)具有通常的新建,打开…,保存,另存为…,以及打开最近的。最后一个是ToolStripDropDown,显示最近使用的文件名列表 我想在文件名中添加一个右键单击的上下文菜单(不是子菜单),该菜单将在上下文菜单中有一个项目,以从列表中删除右键单击的文件名 我将文件名加载到菜单中,如下所示:
private void mnuFile_DropDownOpened(object sender, EventArgs e)
{
foreach (string fn in mru_files)
{
ToolStripMenuItem p = new ToolStripMenuItem(fn);
p.Click += fn_clicked;
p.MouseDown += fn_MouseDown;
openRecentToolStripMenuItem.DropDownItems.Add(p);
}
}
private void fn_MouseDown(object sender, MouseEventArgs e)
{
ToolStripMenuItem toolStripMenuItem = sender as ToolStripMenuItem;
if (toolStripMenuItem != null
&& e.Button == System.Windows.Forms.MouseButtons.Right
&& mru_files.Find(x => x == toolStripMenuItem.Text) != null)
{
mru_files.Remove(toolStripMenuItem.Text);
}
}
这项工作的主要部分很好——当我单击一个文件时,fn_clicked
方法被调用,它将执行它应该执行的操作
在MouseDown
处理程序中,我可以从列表中删除该文件,如下所示:
private void mnuFile_DropDownOpened(object sender, EventArgs e)
{
foreach (string fn in mru_files)
{
ToolStripMenuItem p = new ToolStripMenuItem(fn);
p.Click += fn_clicked;
p.MouseDown += fn_MouseDown;
openRecentToolStripMenuItem.DropDownItems.Add(p);
}
}
private void fn_MouseDown(object sender, MouseEventArgs e)
{
ToolStripMenuItem toolStripMenuItem = sender as ToolStripMenuItem;
if (toolStripMenuItem != null
&& e.Button == System.Windows.Forms.MouseButtons.Right
&& mru_files.Find(x => x == toolStripMenuItem.Text) != null)
{
mru_files.Remove(toolStripMenuItem.Text);
}
}
但这不显示菜单
如果我在表单中添加上下文菜单,并且
mnu_ctxMRUitem.Show(xyz, e.X, e.Y);
我没有删除该文件,而是将上下文菜单放在正确的位置,但包含文件列表的原始菜单已消失
如何在右键单击菜单项时显示上下文菜单,而不使主菜单消失。要强制a保持打开状态,可以将其属性设置为false
然后可以在生成右键单击的位置显示ContextMenuStrip 订阅任何需要ContextMenuItem的ToolStripMenuItem的
MouseUp
事件,如果e.Button==MouseButtons.Right
测试为阳性,则阻止所选ToolStripMenuItem的ToolStripDropDown。对于所有ToolStripMenuItems,使用相同的
MouseUp
事件处理程序
关闭ContextMenuStrip后,将AutoClose
属性设置回true
当然,您需要对层次结构中的所有ToolStripMenuItem下拉组件执行此操作,因为您可能希望将此功能激活到ToolStripMenuItem的子项。此操作由
SetMenutemsAutoClose()
方法执行
这里,激活ContextMenuStrip的ToolStripMenuItem保存到[ContextMenuStrip].Tag
属性中。请随意使用任何其他方法来存储此引用
private void anyToolStripMenuItem_MouseUp(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Right) {
var menuItem = sender as ToolStripMenuItem;
someContextMenuStrip.Tag = menuItem;
SetMenutemsAutoClose(menuItem, false);
someContextMenuStrip.Show(MousePosition);
someContextMenuStrip.Capture = true;
menuItem.BackColor = Color.FromArgb(42, SystemColors.MenuHighlight);
}
}
private async void SomeContextMenuStrip_Closed(object sender, ToolStripDropDownClosedEventArgs e)
{
var cms = sender as ContextMenuStrip;
if (cms.Tag != null && cms.Tag is ToolStripMenuItem menuItem) {
// Determine an action based on the ToolStripMenuItem
Console.WriteLine(menuItem.Name);
SetMenutemsAutoClose(menuItem, true);
menuItem.BackColor = Color.Transparent;
// Need to somewhat fight against the internal Timer
await Task.Delay(100);
if (menuItem.Owner != null) menuItem.Owner.Capture = true;
}
}
private void SetMenutemsAutoClose(ToolStripMenuItem menu, bool autoClose)
{
if (menu == null || menu.Owner == null) return;
while (menu.Owner is ToolStripDropDown dropDown) {
dropDown.AutoClose = autoClose;
menu = dropDown.OwnerItem as ToolStripMenuItem;
}
}
如果模式匹配在您的C#版本中不可用,请在线声明变量,当它显示上下文菜单时,明确地“someContextMenuStrip”的高亮颜色将丢失,因此很难确定右键单击了哪个元素。有没有办法保持突出显示(或聚焦)的颜色?我已经稍微改变了行为并为当前选择的菜单项着色。在不重写ContextMenuStrip类或ToolStripDropDown类的一半的情况下,查看此行为是否足够好。