WPFを使用している場合、WPFはルーティングされたイベントシステムを使用してイベントをディスパッチするため、必要なイベントを簡単にキャッチできます。WinFormsでは、次の2つの方法のいずれかをお勧めします。
メッセージフィルタークラスを定義します。
public class KeyMessageFilter : IMessageFilter
{
private enum KeyMessages
{
WM_KEYFIRST = 0x100,
WM_KEYDOWN = 0x100,
WM_KEYUP = 0x101,
WM_CHAR = 0x102,
WM_SYSKEYDOWN = 0x0104,
WM_SYSKEYUP = 0x0105,
WM_SYSCHAR = 0x0106,
}
[DllImport("user32.dll")]
private static extern IntPtr GetParent(IntPtr hwnd);
// We check the events agains this control to only handle
// key event that happend inside this control.
Control _control;
public KeyMessageFilter()
{ }
public KeyMessageFilter(Control c)
{
_control = c;
}
public bool PreFilterMessage(ref Message m)
{
if (m.Msg == (int)KeyMessages.WM_KEYDOWN)
{
if (_control != null)
{
IntPtr hwnd = m.HWnd;
IntPtr handle = _control.Handle;
while (hwnd != IntPtr.Zero && handle != hwnd)
{
hwnd = GetParent(hwnd);
}
if (hwnd == IntPtr.Zero) // Didn't found the window. We are not interested in the event.
return false;
}
Keys key = (Keys)m.WParam;
switch (key)
{
case Keys.Left:
MessageBox.Show("Left");
return true;
case Keys.Right:
MessageBox.Show("Right");
return true;
}
}
return false;
}
}
つまり、Windowsフォームのすべてのメッセージが通過するクラスがあります。あなたはイベントであなたがやりたいことを何でもすることができます。メソッドがtrueを返す場合PreFilterMessage
、それはイベントがその応答制御にディスパッチされるべきではないことを意味します。
Keys
(列挙型の値は仮想キーコードとほぼ同じであることに注意してください)
これが機能する前に、アプリケーションのメッセージフィルターに追加する必要があります。
public partial class Form1 : Form
{
// We need an instance of the filter class
KeyMessageFilter filter;
public Form1()
{
InitializeComponent();
filter = new KeyMessageFilter(panel1);
// add the filter
Application.AddMessageFilter(filter);
}
protected override void OnFormClosed(FormClosedEventArgs e)
{
base.OnFormClosed(e);
// remove the filter
Application.RemoveMessageFilter(filter);
}
}
フィルタは、の存続期間中のみアクティブになりForm1
ます。
注意:これにより、あらゆる形式のイベントがキャッチされます。1つのフォームでのみ機能させる場合は、フォームをフィルタークラスに渡し、そのHandle
プロパティをm.HWnd
inと比較します。PreFilterMessage
これは、より高度で複雑な(そして低レベルの)アプローチです。そして、それはより多くのコードを必要とします。HookManager
プロセスを非常に簡単にするクラスを作成しました。クラスをgithubに公開し、それに関する記事を書きます。