Is there any way to detect a mouseclick outside a

2020-02-05 12:18发布

I'm creating a custom dropdown box, and I want to register when the mouse is clicked outside the dropdown box, in order to hide it. Is it possible to detect a click outside a control? or should I make some mechanism on the containing form and check for mouseclick when any dropdownbox is open?

user control

5条回答
\"骚年 ilove
2楼-- · 2020-02-05 12:41

Technically, you'll need to p/invoke SetCapture() in order to receive click events that happen outside of your control.

But in your case, handling the Leave event, as @Martin suggests, should be sufficient.

EDIT: While looking for an usage example for SetCapture(), I came across the Control.Capture property, of which I was not aware. Using that property means you won't have to p/invoke anything, which is always a good thing in my book.

So, you'll have to set Capture to true when showing the dropdown, then determine if the mouse pointer lies inside the control in your click event handler and, if it doesn't, set Capture to false and close the dropdown.

查看更多
啃猪蹄的小仙女
3楼-- · 2020-02-05 12:41

You are probably looking for the leave event:

http://msdn.microsoft.com/en-us/library/system.windows.forms.control.leave.aspx

Leave occurs when the input focus leaves the control.

查看更多
淡お忘
4楼-- · 2020-02-05 12:47

I've done this myself, and this is how I did it.

When the drop down is opened, register a click event on the control's parent form:

this.Form.Click += new EventHandler(CloseDropDown);

But this only takes you half the way. You probably want your drop down to close also when the current window gets deactivated. The most reliable way of detecting this has for me been through a timer that checks which window is currently active:

[System.Runtime.InteropServices.DllImport("user32.dll")]
static extern IntPtr GetForegroundWindow();

and

var timer = new Timer();
timer.Interval = 100;
timer.Tick += (sender, args) =>
{
    IntPtr f = GetForegroundWindow();
    if (this.Form == null || f != this.Form.Handle)
    {
        CloseDropDown();
    }
};

You should of course only let the timer run when the drop down is visible. Also, there's probably a few other events on the parent form you'd want to register when the drop down is opened:

this.Form.LocationChanged += new EventHandler(CloseDropDown);
this.Form.SizeChanged += new EventHandler(CloseDropDown);

Just don't forget to unregister all these events in the CloseDropDown method :)

EDIT:

I forgot, you should also register the Leave event on you control to see if another control gets activated/clicked:

this.Leave += new EventHandler(CloseDropDown);

I think I've got it now, this should cover all bases. Let me know if I'm missing something.

查看更多
一夜七次
5楼-- · 2020-02-05 13:00

So I finally understand that you only want it to close when the user clicks outside of it. In that case, the Leave event should work just fine... For some reason, I got the impression you wanted it to close whenever they moved the mouse outside of your custom dropdown. The Leave event is raised whenever your control loses the focus, and if the user clicks on something else, it will certainly lose focus as the thing they clicked on gains the focus.

The documentation also says that this event cascades up and down the control chain as necessary:

The Enter and Leave events are hierarchical and will cascade up and down the parent chain until the appropriate control is reached. For example, assume you have a Form with two GroupBox controls, and each GroupBox control has one TextBox control. When the caret is moved from one TextBox to the other, the Leave event is raised for the TextBox and GroupBox, and the Enter event is raised for the other GroupBox and TextBox.

Overriding your UserControl's OnLeave method is the best way to handle this:

protected override void OnLeave(EventArgs e)
{
   // Call the base class
   base.OnLeave(e);

   // When this control loses the focus, close it
   this.Hide();
}

And then for testing purposes, I created a form that shows the drop-down UserControl on command:

public partial class Form1 : Form
{
   private UserControl1 customDropDown;

   public Form1()
   {
      InitializeComponent();

      // Create the user control
      customDropDown = new UserControl1();

      // Add it to the form's Controls collection
      Controls.Add(customDropDown);
      customDropDown.Hide();
   }

   private void button1_Click(object sender, EventArgs e)
   {         
      // Display the user control
      customDropDown.Show();
      customDropDown.BringToFront();   // display in front of other controls
      customDropDown.Select();         // make sure it gets the focus
   }
}

Everything works perfectly with the above code, except for one thing: if the user clicks on a blank area of the form, the UserControl doesn't close. Hmm, why not? Well, because the form itself doesn't want the focus. Only controls can get the focus, and we didn't click on a control. And because nothing else stole the focus, the Leave event never got raised, meaning that the UserControl didn't know it was supposed to close itself.

If you need the UserControl to close itself when the user clicks on a blank area in the form, you need some special case handling for that. Since you say that you're only concerned about clicks, you can just handle the Click event for the form, and set the focus to a different control:

protected override void OnClick(EventArgs e)
{
   // Call the base class
   base.OnClick(e);

   // See if our custom drop-down is visible
   if (customDropDown.Visible)
   {
      // Set the focus to a different control on the form,
      // which will force the drop-down to close
      this.SelectNextControl(customDropDown, true, true, true, true);
   }
}

Yes, this last part feels like a hack. The better solution, as others have mentioned, is to use the SetCapture function to instruct Windows to capture the mouse over your UserControl's window. The control's Capture property provides an even simpler way to do the same thing.

查看更多
手持菜刀,她持情操
6楼-- · 2020-02-05 13:00

Handle the Form's MouseDown event, or override the Form's OnMouseDown method:

enter code here

And then:

protected override void OnMouseDown(MouseEventArgs e)
{

    if (!theListBox.Bounds.Contains(e.Location)) 
    {
        theListBox.Visible = false;
    }
}

The Contains method old System.Drawing.Rectangle can be used to indicate if a point is contained inside a rectangle. The Bounds property of a Control is the outer Rectangle defined by the edges of the Control. The Location property of the MouseEventArgs is the Point relative to the Control which received the MouseDown event. The Bounds property of a Control in a Form is relative to the Form.

查看更多
登录 后发表回答