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类的一半的情况下,查看此行为是否足够好。