Control With Scrollbar上的.NET C#MouseEnter侦听器

时间:2010-06-09 12:47:02

标签: c# .net winforms

只要鼠标位于特定控件上,我们就会显示某种形式。当鼠标离开控件时,我们会在超时后隐藏控件。这是标准的悬停行为。

但是,当一个控件(例如Treeview)有一个滚动条,并且鼠标是ON或滚动条时,事件不会触发......

如果我们可以获得对滚动条控件的引用,这将解决我们的问题,因为我们会将相同的侦听器事件添加到滚动条。但是,据我所知,滚动条无法访问...

我们如何解决这个问题?

2 个答案:

答案 0 :(得分:3)

滚动条位于树状视图的非客户区域中。当鼠标移动到那里时,它开始生成非客户端消息,如WM_NCMOUSEMOVE和WM_NCMOUSELEAVE。您必须对TreeView进行子类化并覆盖WndProc()以检测这些消息。

虽然这并没有真正解决你的问题,你仍然会遇到边缘情况。使用Timer的低技术方法始终有效:

    private Form frmPopup;

    private void treeView1_MouseEnter(object sender, EventArgs e) {
        timer1.Enabled = true;
        if (frmPopup == null) {
            frmPopup = new Form2();
            frmPopup.StartPosition = FormStartPosition.Manual;
            frmPopup.Location = PointToScreen(new Point(treeView1.Right + 20, treeView1.Top));
            frmPopup.FormClosed += (o, ea) => frmPopup = null;
            frmPopup.Show();
        }
    }

    private void timer1_Tick(object sender, EventArgs e) {
        Rectangle rc = treeView1.RectangleToScreen(new Rectangle(0, 0, treeView1.Width, treeView1.Height));
        if (!rc.Contains(Control.MousePosition)) {
            timer1.Enabled = false;
            if (frmPopup != null) frmPopup.Close();
        }
    }

答案 1 :(得分:2)

我认为有几种不同的方法可以做到这一点,但关键是你希望在行动上超时。我认为两种技术的结合可能有效:

将控件放在面板上,停靠以填充,然​​后使用面板的MouseEnter打开您的行为 - 这将包括控件的滚动条。您也可以使用面板的MouseLeave事件,但是您必须检查光标的位置以确保它没有移动到包含的控件中。这种方法最可靠,但快速移动鼠标可能会让人感到困惑。

如果将其与显示所显示/隐藏控件时启动的计时器结合使用,并定期检查光标位置。这将起作用,但是在隐藏控件之前的超时不一定是一致的(因为计时器在进入控件时启动)。你可以在控件中的mousemoves上停止/启动计时器,以减轻这种情况。

我整理了一个我在这里试过的不同方法的项目:http://lovethedot.s3.amazonaws.com/100609StackoverflowScrollbarQuestion.zip

通过对接你要在面板中跟踪的控件,它实际上将它包装起来,你将在跟踪控件的最边缘获得MouseEnter:

    private void panel1_MouseEnter(object sender, EventArgs e)
    {
        this.Text = "in";
    }

    private void panel1_MouseLeave(object sender, EventArgs e)
    {
        if (!new Rectangle(new Point(0, 0), panel1.Size).Contains(panel1.PointToClient(Control.MousePosition)))
            this.Text = "out";
    }

您正在跟踪进入控件周围的面板,并退出该面板,前提是光标不在被跟踪的控件内。

为了获得更好的“离开”体验,它结合了一个Timer来检查光标的位置:

        private void listBox3_MouseEnter(object sender, EventArgs e)
    {
        button1.Visible = true;
        visibleTimer.Stop();
        visibleTimer.Start();
    }

    void visibleTimer_Tick(object sender, EventArgs e)
    {
        if (!new Rectangle(new Point(0, 0), listBox3.Size).Contains(listBox3.PointToClient(Control.MousePosition)))
        {
            visibleTimer.Stop();
            button1.Visible = false;
        }
    }