1

我创建了我自己的类似 ComboBox 的控件,其中下拉部分包含一棵树。我见过那些使用普通 ComboBox 并覆盖 WndProc 的解决方案,但是尽管有很多代码,但总是有一些奇怪的行为。所以我决定让它变得简单:只是一个带有 ToolStripDropDown/ToolStripControlHost 的标签,当鼠标放在标签上时它会打开。缺少的 ComboBox 三角形不会受到伤害。

一切都很完美,除了一件小事:就像股票组合框一样,当我再次单击标签时,我希望隐藏下拉菜单。但是当我点击它时,下拉菜单会隐藏一秒钟,只是为了再次出现。如果我在标签外部单击,下拉菜单就会隐藏,就像它应该的那样。

public class RepoNodeComboBox: Label
{
    RepoTreeView repoTreeView;
    ToolStripControlHost treeViewHost;
    ToolStripDropDown dropDown;
    bool isDropDownOpen;

    public int DropDownHeight;

    public RepoNodeComboBox()
    {
        repoTreeView = new RepoTreeView();
        repoTreeView.BorderStyle = BorderStyle.None;
        repoTreeView.LabelEdit = false;

        treeViewHost = new ToolStripControlHost(repoTreeView);
        treeViewHost.Margin = Padding.Empty;
        treeViewHost.Padding = Padding.Empty;
        treeViewHost.AutoSize = false;

        dropDown = new ToolStripDropDown();
        dropDown.CanOverflow = true;
        dropDown.AutoClose = true;
        dropDown.DropShadowEnabled = true;
        dropDown.Items.Add(treeViewHost);
        dropDown.Closing += dropDownClosing;

        TextAlign = ContentAlignment.MiddleLeft;
        BackColor = SystemColors.Window;
        BorderStyle = BorderStyle.FixedSingle;

        DropDownHeight = 400;
    }

    protected override void Dispose(bool disposing)
    {
        if (disposing) 
        {
            if (dropDown != null) 
            {
                dropDown.Dispose();
                dropDown = null;
            }
        }
        base.Dispose(disposing);
    }

    // when mouse goes down on the label, this is executed first            
    void dropDownClosing(object sender, System.ComponentModel.CancelEventArgs e)
    {
        // just to test if I can get more out of it than with dropdown.Visible
        isDropDownOpen = false; 
    }

    // this is subsidiary to the Closing event of the dropdown
    protected override void OnMouseDown(MouseEventArgs e)
    {
        if (dropDown != null) 
        {
            if (isDropDownOpen)
            {
                dropDown.Hide();
            }
            else
            {
                repoTreeView.Size = new Size(Width, DropDownHeight);
                treeViewHost.Width = Width;
                treeViewHost.Height = DropDownHeight;
                dropDown.Show(this, 0, Height);
                isDropDownOpen = true;
            }
        }

        base.OnMouseDown(e);
    }
}

据我所见(断点),下拉列表首先捕获 MOUSEDOWN 事件以便自行关闭。只有在那之后,我的标签才通过 MOUSEDOWN 事件,并且由于它看到下拉菜单已关闭,它认为标签已像第一次一样被点击 - 并再次打开下拉菜单。

因此,标签似乎没有机会知道 MOUSEDOWN 是否是关闭下拉项的结果。我可以每隔一个事件打开它,但这不需要其他关闭事件发生。

有什么方法可以确保即使我点击标签,打开的下拉项目也会关闭?

4

1 回答 1

1

尝试在浮动主机关闭时检查鼠标位置:

void dropDownClosing(object sender, CancelEventArgs e) {
  isDropDownOpen = this.ClientRectangle.Contains(this.PointToClient(Control.MousePosition));
}

在 MouseDown 代码中,确保在 true 块中将 isDropDownOpen 设置为 false:

protected override void OnMouseDown(MouseEventArgs e) {
  if (dropDown != null) {
    if (isDropDownOpen) {
      isDropDownOpen = false;
      dropDown.Hide();
    } else {
      isDropDownOpen = true;
      repoTreeView.Size = new Size(Width, DropDownHeight);
      treeViewHost.Width = Width;
      treeViewHost.Height = DropDownHeight;
      dropDown.Show(this, 0, Height);
    }
  }
  base.OnMouseDown(e);
}
于 2018-09-18T19:23:09.137 回答