Событие мыши (C #)
Я не могу получить событие Mouse Wheel в основной форме.
В качестве демонстрации я придумал простой пример:
public partial class Form1 : Form { public Form1() { InitializeComponent(); this.panel1.MouseWheel += new MouseEventHandler(panel1_MouseWheel); this.panel1.MouseMove += new MouseEventHandler(panel1_MouseWheel); Form2 f2 = new Form2(); f2.Show(this); } private void panel1_MouseWheel(object sender, MouseEventArgs e) { if(e.Delta != 0) Console.Out.WriteLine(e.Delta); } } public partial class Form2 : Form { public Form2() { InitializeComponent(); this.MouseMove += new MouseEventHandler(Form2_MouseMove); this.MouseWheel += new MouseEventHandler(Form2_MouseMove); } private void Form2_MouseMove(object sender, MouseEventArgs e) { if(e.Delta != 0) Console.Out.WriteLine(e.Delta); } }
Я получаю событие колеса мыши в Form2, но не Form1 никаких идей?
- Импорт Excel в DataTable Быстро
- Каков самый быстрый способ объединить два xml-файла в один
- Связывание со статическим classом
- Разница в поведении при использовании dynamic_cast со ссылкой и указателями
- используя \ в строке как литерал вместо escape
Ура,
Джеймс
- Можно ли установить компилятор C # без Visual Studio?
- Ошибка CookieContainer?
- GCC и предварительно скомпилированные заголовки
- WinRT и региональные настройки. Правильный способ форматирования дат и чисел на основе региональных настроек пользователя?
- Если строки .NET неизменны в .NET, то почему Подстановка принимает O (n) время?
- Как убить stream мгновенно в C #?
- Глобально конвертировать UTC DateTimes в пользовательские локальные DateTimes
- Что такое идиома копирования и свопинга?
Я подозреваю, что OP хочет получить события прокрутки, когда только мышь нависает над панелью, даже если панель не имеет фокуса.
Здесь объясняется способ достижения такого поведения:
http://social.msdn.microsoft.com/forums/en-US/winforms/thread/eb922ed2-1036-41ca-bd15-49daed7b637c/
и здесь:
http://social.msdn.microsoft.com/forums/en-US/winforms/thread/6bfb9287-986d-4c60-bbcc-23486e239384/
Один из fragmentов кода, взятых из связанного форума:
using System; using System.ComponentModel; using System.Drawing; using System.Windows.Forms; using System.Runtime.InteropServices; namespace WindowsApplication1 { public partial class Form1 : Form, IMessageFilter { public Form1() { InitializeComponent(); Application.AddMessageFilter(this); } public bool PreFilterMessage(ref Message m) { if (m.Msg == 0x20a) { // WM_MOUSEWHEEL, find the control at screen position m.LParam Point pos = new Point(m.LParam.ToInt32() & 0xffff, m.LParam.ToInt32() >> 16); IntPtr hWnd = WindowFromPoint(pos); if (hWnd != IntPtr.Zero && hWnd != m.HWnd && Control.FromHandle(hWnd) != null) { SendMessage(hWnd, m.Msg, m.WParam, m.LParam); return true; } } return false; } // P/Invoke declarations [DllImport("user32.dll")] private static extern IntPtr WindowFromPoint(Point pt); [DllImport("user32.dll")] private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, IntPtr lp); } }
Этот код будет в основном перехватывать все события wm_mousewheel и перенаправлять их на элемент управления, который в данный момент находится в курсе мыши. Для получения колесных событий панели больше не нужно фокусироваться.
Ваша проблема возникает из-за того, что form1 имеет фокус, а не панель1. … что означает, что это события формы1, которые будут запущены, а не события панели1.
Я восстановил ваш сценарий со следующими изменениями в конструкторе в Form1 и подтвердил, что он запускает событие колеса прокрутки.
public Form1() { InitializeComponent(); /* --- Old code that don't work --- this.panel1.MouseWheel += new MouseEventHandler(panel1_MouseWheel); this.panel1.MouseMove += new MouseEventHandler(panel1_MouseWheel); */ this.MouseWheel += new MouseEventHandler(panel1_MouseWheel); this.MouseMove += new MouseEventHandler(panel1_MouseWheel); Form2 f2 = new Form2(); f2.Show(this); } }
Добавьте еще одно событие панели MouseEnter
и в его функции обратного вызова получите фокус ввода:
void MouseEnterEvent() { this.Panel.Focus(); }
Благодаря ответу @nitrogenycs я написал простой общий class, чтобы легко решить проблему:
using System; using System.Windows.Forms; using System.Runtime.InteropServices; using System.Drawing; namespace MyNamespace { public class MouseWheelManagedForm : Form, IMessageFilter { private bool managed; public MouseWheelManagedForm () : this (true) { } public MouseWheelManagedForm (bool start) { managed = false; if (start) ManagedMouseWheelStart(); } protected override void Dispose (bool disposing) { if (disposing) ManagedMouseWheelStop(); base.Dispose(disposing); } /************************************ * IMessageFilter implementation * *********************************/ private const int WM_MOUSEWHEEL = 0x20a; // P/Invoke declarations [DllImport("user32.dll")] private static extern IntPtr WindowFromPoint (Point pt); [DllImport("user32.dll")] private static extern IntPtr SendMessage (IntPtr hWnd, int msg, IntPtr wp, IntPtr lp); private bool IsChild (Control ctrl) { Control loopCtrl = ctrl; while (loopCtrl != null && loopCtrl != this) loopCtrl = loopCtrl.Parent; return (loopCtrl == this); } public bool PreFilterMessage (ref Message m) { if (m.Msg == WM_MOUSEWHEEL) { //Ensure the message was sent to a child of the current form if (IsChild(Control.FromHandle(m.HWnd))) { // Find the control at screen position m.LParam Point pos = new Point(m.LParam.ToInt32() & 0xffff, m.LParam.ToInt32() >> 16); //Ensure control under the mouse is valid and is not the target control //otherwise we'd be trap in a loop. IntPtr hWnd = WindowFromPoint(pos); if (hWnd != IntPtr.Zero && hWnd != m.HWnd && Control.FromHandle(hWnd) != null) { SendMessage(hWnd, m.Msg, m.WParam, m.LParam); return true; } } } return false; } /**************************************** * MouseWheelManagedForm specific methods * **************************************/ public void ManagedMouseWheelStart () { if (!managed) { managed = true; Application.AddMessageFilter(this); } } public void ManagedMouseWheelStop () { if (managed) { managed = false; Application.RemoveMessageFilter(this); } } } }
Оттуда вам нужно только наследовать свою форму из этого classа вместо формы для каждой формы, в которой вам нужно управлять MouseWheel:
using System; using System.Collections.Generic; using System.Data; using System.Linq; using System.Windows.Forms; namespace MyApp { public partial class MyForm : MyNamespace.MouseWheelManagedForm { public MyForm () { InitializeComponent(); } } }
Надеюсь, это поможет кому-то другому (чем мне).
Может быть, это сработает для вас?
public partial class Form1 : Form { public Form1() { InitializeComponent(); Form2 f2 = new Form2(); f2.MouseWheel += new MouseEventHandler(panel1_MouseWheel); f2.MouseMove += new MouseEventHandler(panel1_MouseWheel); f2.Show(this); } private void panel1_MouseWheel(object sender, MouseEventArgs e) { if(e.Delta != 0) Console.Out.WriteLine(e.Delta); } }
Панель не может иметь фокус, только элемент, расположенный внутри панели, может иметь фокус. Панель получит только событие MouseWheel, если что-то помещено внутри него, и эта вещь имеет фокус. Простое наведение на панель и перемещение колеса мыши отправят событие в форму, а не на панель.
В этом разница между двумя вашими примерами.
Я думаю, вы немного неправильно поняли мою проблему. В моей основной форме с приведенным ниже кодом я не получаю событие MouseWheel:
public Form1() { InitializeComponent(); this.panel1.MouseWheel += new MouseEventHandler(panel1_MouseWheel); this.panel1.MouseMove += new MouseEventHandler(panel1_MouseWheel); }
Но я получаю событие:
public Form1() { InitializeComponent(); this.MouseWheel += new MouseEventHandler(panel1_MouseWheel); }
Надеюсь, что это станет яснее без путаницы Form2. Я просто пытаюсь получить события MouseWheel на панели в моей основной форме.
Ура,
Джеймс
this.MouseWheel += pictureBox1_MouseWheel; //tanımlama void pictureBox1_MouseWheel(object sender, MouseEventArgs e) { if (Convert.ToString(e.Delta) == "120") { //yukarı } else if (Convert.ToString(e.Delta) == "-120") { //aşağı } }