メインフォームでマウスホイールイベントを取得できません。
デモとして、簡単な例を思いつきました。
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では取得できません。
乾杯、
ジェームス
パネルがフォーカスを持っていなくても、マウスがパネル上をホバリングしているときに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/
リンクされたフォーラムから取られたコードスニペットの1つ:
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イベントをインターセプトし、マウスが現在ホバーしているコントロールにそれらをリダイレクトします。ホイールイベントを受け取るために、パネルにフォーカスを置く必要はもうありません。
問題は、panel1ではなくform1にフォーカスがあるという事実から発生します。 ...もちろん、panel1のイベントではなく、form1のイベントが起動されることを意味します。
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の回答のおかげで、この問題に簡単に対処するための簡単なジェネリッククラスを作成しました。
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);
}
}
}
}
そこから、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イベントを取得しません。
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イベントを取得しようとしています。
乾杯、
ジェームス
パネル自体にフォーカスを持つことはできません。パネル内に配置されたアイテムのみがフォーカスを持つことができます。パネルは、その中に何かが配置され、そのオブジェクトにフォーカスが置かれた場合にのみ、MouseWheelイベントを受け取ります。パネルにカーソルを合わせてマウスホイールを動かすだけで、パネルではなくフォームにイベントが送信されます。
これは、2つの例の違いです。