1

我的应用程序顶部有一个菜单。
菜单中的第一项(文件)具有通常的新建打开...保存另存为...以及打开最近。最后一个是 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);
    }
}

主要部分工作正常 -fn_clicked当我单击文件时调用该方法,它将执行它应该执行的操作。

MouseDown处理程序中,我可以从列表中删除文件,如下所示:

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);

我没有删除文件,而是在正确的位置获得了上下文菜单,但是带有文件列表的原始菜单已经消失。

如何在右键单击菜单项时显示上下文菜单,而不会使主菜单消失。

4

1 回答 1

1

要强制ToolStripDropDown保持打开状态,您可以将其AutoClose属性设置为false
然后,您可以在生成右键单击的位置显示 ContextMenuStrip。

订阅MouseUp需要 ContextMenuStrip 的任何 ToolStripMenuItem 的事件,如果e.Button == MouseButtons.Right测试结果为肯定,则阻止作为所选 ToolStripMenuItem所有者的 ToolStripDropDown。
对所有 ToolStripMenuItem使用相同的MouseUp事件处理程序。

当 ContextMenuStrip 关闭时,将AutoClose属性设置回true.

当然,您需要对层次结构中的所有 ToolStripDropDown 组件执行此操作,因为您可能希望为 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# 版本中不可用,请显式地内联声明变量

于 2021-06-01T16:51:01.190 回答