如何使用按钮单击切换 ContextMenuStrip 显示,但也允许它正常关闭(菜单项单击、失去焦点等)

本文关键字:单击 菜单项 常关闭 许它正 失去 焦点 按钮 何使用 ContextMenuStrip 显示 | 更新日期: 2023-09-27 18:20:04

我创建了一个由标签和ContextMenuStrip组成的简单UserControl。我让它的功能像一个组合框,但我显示的不是下拉菜单,而是上下文菜单条。

我让它工作,但有一些我无法弄清楚的棘手问题。

我正在尝试使标签ComboButton的工作方式与ComboBox相同。单击按钮,将显示下拉列表。再次单击该按钮,它将缩回。问题是,上下文菜单在任何鼠标单击时都会消失。因此,当我第二次单击按钮以收回菜单时,菜单首先消失,然后触发单击事件,再次显示菜单。

我仍然希望当用户选择菜单项并且当他们像普通上下文菜单一样单击表单上的任意位置时,菜单会消失。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Drawing;
using System.Data;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Drawing.Drawing2D;
using System.Windows.Forms.VisualStyles;
using System.Diagnostics;
namespace Controls
{
    public partial class CMenu : UserControl
    {
        ButtonState _buttonState = ButtonState.Normal;
        public CMenu()
        {
            InitializeComponent();
        }
        private void lblSelect_Paint(object sender, PaintEventArgs e)
        {
            base.OnPaint(e);
            ControlPaint.DrawComboButton(e.Graphics, getLabelRect(), _buttonState);
        }
        private bool IsDropdownHit(MouseEventArgs e)
        {
            Rectangle cursor = new Rectangle(e.X, e.Y, 1, 1);
            if (e.Button == MouseButtons.Left && cursor.IntersectsWith(getLabelRect()))
            {
                return true;
            }
            return false;
        }
        private void lblSelect_MouseUp(object sender, MouseEventArgs e)
        {
            if (!IsDropdownHit(e))
                return;
            if (!cmsItems.Visible)
                lblSelect.ContextMenuStrip = cmsItems;
                cmsItems.Width = lblSelect.Width;
                cmsItems.Show(lblSelect, 0, lblSelect.Height);
        }
        private Rectangle getLabelRect()
        {
            return new Rectangle(lblSelect.Width - 20, 0, 20, lblSelect.Height);
        }
    }
}

如何使用按钮单击切换 ContextMenuStrip 显示,但也允许它正常关闭(菜单项单击、失去焦点等)

所以,我稍微重写了你的鼠标向上:

private void lblSelect_MouseUp(object sender, MouseEventArgs e)
    {
        if (IsDropdownHit(e) && cmsItems.Tag == null)
        {
            cmsItems.Width = lblSelect.Width;
            cmsItems.Show(lblSelect, 0, lblSelect.Height);
            cmsItems.Tag = "Shown";
        }
        else
        {
            cmsItems.Hide();
            cmsItems.Tag = null;
        }
    }

现在,它将关闭。但是 - 如果您不使用按钮关闭,下次必须单击两次才能打开它。

此"错误"的解决方法:

void cmsItems_Closed(object sender, ToolStripDropDownClosedEventArgs e)
    {
        Point c = PointToClient(Cursor.Position);
        if (!IsDropdownHit(new MouseEventArgs(MouseButtons.Left, 1, c.X - lblSelect.Location.X, c.Y - lblSelect.Location.Y, 0)))
            cmsItems.Tag = null;
    }

根据您设计窗体的方式,您可能需要调整 MouseEventArgs-Coordinates,才能成功确定下拉列表命中数。我刚刚将 CMenu 控件添加到一个空窗体中。

对于解决方法,请不要忘记将处理程序添加到 CMenu 的构造函数中.cs:

cmsItems.Closed += cmsItems_Closed;