views:

1213

answers:

3

I need to detect when the user moves the mouse over the Form and all it's child controls and also when it leaves the Form. I tried the MouseEnter and MouseLeave events of the Form, I tried the WM_MOUSEMOVE/WM_MOUSELAVE and WM_NCMOUSEMOVE/WM_NCMOUSELEAVE windows messages but none seem to work as I want...

Most of my Form is occupied by child controls of many sorts, there's not much client area visible. This means that if I move the mouse very quickly the mouse movement will not be detected, although the mouse is inside the Form.

For instance, I have a TextBox that is docked at the bottom and between the desktop and the TextBox, there's only a very small border. If I quickly move the mouse from the bottom into the TextBox, the mouse movement won't be detected, but the mouse is inside the TextBox, therefor, inside the Form.

How can I achieve what I need?

A: 

well, that's probably not very clean, but you could probably use a global system hook to detect mouse movement (see here), and check whether it is inside your form with this code :

if (this.ClientRectangle.Contains(mousePosition))
{
    // do something
}
Thomas Levesque
Global hook is too much for this request. And repeatly check, if mouse in form area is not good solution. First - it taky performance, Second - form must not be rectangle.
TcKs
+1  A: 

You can hook the main message loop and preprocess/postprocess any (WM_MOUSEMOVE) message what you want.

public class Form1 : Form {
 private MouseMoveMessageFilter mouseMessageFilter;
 protected override void OnLoad( EventArgs e ) {
  base.OnLoad( e );

  this.mouseMessageFilter = new MouseMoveMessageFilter();
  this.mouseMessageFilter.TargetForm = this;
  Application.AddMessageFilter( this.mouseMessageFilter );
 }

 protected override void OnClosed( EventArgs e ) {
  base.OnClosed( e );

  Application.RemoveMessageFilter( this.mouseMessageFilter );
 }

 class MouseMoveMessageFilter : IMessageFilter {
  public Form TargetForm { get; set; }

  public bool PreFilterMessage( ref Message m ) {
   int numMsg = m.Msg;
   if ( numMsg == 0x0200 /*WM_MOUSEMOVE*/) {
    this.TargetForm.Text = string.Format( "X:{0}, Y:{1}", Control.MousePosition.X, Control.MousePosition.Y );
   }

   return false;
  }

 }
}
TcKs
That only makes it work the other way... I mean, now it detects when the mouse is over the Form's child controls, but no the form itself. I want to detect the WHOLE thing. I also need to detect when the mouse entered the Form and when it left, not just that it's moving inside.
Nazgulled
Nazgulled
Did you even saw my request TcKS? Maybe you do this time... Could you please edit your post and elaborate a bit on the whole IMessageFilter thing and how it works?
Nazgulled
I'm not sure, what exactly you want. The IMessageFilter is very common interface and can be used for almost anything related with windows messaging system.On the internet are tons of samples and tutorials. I don't have no additional knowledge about it, than can be found in the tutorials.
TcKs
A: 

How about this: In your form's OnLoad, recursively go through all of the child controls (and their children) and hook up the MouseEnter event.

Then whenever the mouse enters any descendant, the event handler will be called. Similarly, you could hook up MouseMove and/or MouseLeave events.

protected override void OnLoad()
{
   HookupMouseEnterEvents(this);
}

private void HookupMouseEnterEvents(Control control)
{
   foreach (Control childControl in control.Controls)
   {
      childControl.MouseEnter += new MouseEventHandler(mouseEnter);

      // Recurse on this child to get all of its descendents.
      HookupMouseEnterEvents(childControl);
   }
}
NascarEd
I would call that a "workaround" and not the solution, so no...
Nazgulled
It will not work when the controls are changing during form's life.
TcKs