在Winform和WPF中注冊全局快捷鍵實(shí)現(xiàn)思路及代碼
更新時間:2013年02月28日 11:05:44 作者:
如果注冊快捷鍵,RegisterHotKey中的fsModifiers參數(shù)為0,即None選項(xiàng),一些安全軟件會警報,可能因?yàn)檫@樣就可以全局監(jiān)聽鍵盤而造成安全問題,感興趣的你可以參考下本文
快捷鍵輔助類
復(fù)制代碼 代碼如下:
class HotKey
{
/// <summary>
/// 如果函數(shù)執(zhí)行成功,返回值不為0。
/// 如果函數(shù)執(zhí)行失敗,返回值為0。要得到擴(kuò)展錯誤信息,調(diào)用GetLastError。.NET方法:Marshal.GetLastWin32Error()
/// </summary>
/// <param name="hWnd">要定義熱鍵的窗口的句柄</param>
/// <param name="id">定義熱鍵ID(不能與其它ID重復(fù)) </param>
/// <param name="fsModifiers">標(biāo)識熱鍵是否在按Alt、Ctrl、Shift、Windows等鍵時才會生效</param>
/// <param name="vk">定義熱鍵的內(nèi)容,WinForm中可以使用Keys枚舉轉(zhuǎn)換,
/// WPF中Key枚舉是不正確的,應(yīng)該使用System.Windows.Forms.Keys枚舉,或者自定義正確的枚舉或int常量</param>
/// <returns></returns>
[DllImport("user32.dll", SetLastError = true)]
public static extern bool RegisterHotKey(
IntPtr hWnd,
int id,
KeyModifiers fsModifiers,
int vk
);
/// <summary>
/// 取消注冊熱鍵
/// </summary>
/// <param name="hWnd">要取消熱鍵的窗口的句柄</param>
/// <param name="id">要取消熱鍵的ID</param>
/// <returns></returns>
[DllImport("user32.dll", SetLastError = true)]
public static extern bool UnregisterHotKey(
IntPtr hWnd,
int id
);
/// <summary>
/// 向全局原子表添加一個字符串,并返回這個字符串的唯一標(biāo)識符,成功則返回值為新創(chuàng)建的原子ID,失敗返回0
/// </summary>
/// <param name="lpString"></param>
/// <returns></returns>
[DllImport("kernel32", SetLastError = true)]
public static extern short GlobalAddAtom(string lpString);
[DllImport("kernel32", SetLastError = true)]
public static extern short GlobalDeleteAtom(short nAtom);
/// <summary>
/// 定義了輔助鍵的名稱(將數(shù)字轉(zhuǎn)變?yōu)樽址员阌谟洃?,也可去除此枚舉而直接使用數(shù)值)
/// </summary>
[Flags()]
public enum KeyModifiers
{
None = 0,
Alt = 1,
Ctrl = 2,
Shift = 4,
WindowsKey = 8
}
/// <summary>
/// 熱鍵的對應(yīng)的消息ID
/// </summary>
public const int WM_HOTKEY = 0x312;
}
WinForm方法:
在中在onload中注冊事件,然后重載WndProc,對返回的消息進(jìn)行處理
復(fù)制代碼 代碼如下:
int alts,altd;
protected override void OnLoad(EventArgs e)
{
alts= HotKey.GlobalAddAtom("Alt-S");
altd= HotKey.GlobalAddAtom("Alt-D");
HotKey.RegisterHotKey(this.Handle, alts, HotKey.KeyModifiers.Alt, (int)Keys.S);
HotKey.RegisterHotKey(this.Handle, altd, HotKey.KeyModifiers.Alt, (int)Keys.D);
}
protected override void WndProc(ref Message m)// 監(jiān)視Windows消息
{
switch (m.Msg)
{
case HotKey.WM_HOTKEY:
ProcessHotkey(m);//按下熱鍵時調(diào)用ProcessHotkey()函數(shù)
break;
}
base.WndProc(ref m); //將系統(tǒng)消息傳遞自父類的WndProc
}
private void ProcessHotkey(Message m) //按下設(shè)定的鍵時調(diào)用該函數(shù)
{
IntPtr id = m.WParam;//IntPtr用于表示指針或句柄的平臺特定類型
int sid=id.ToInt32();
if(sid==alts)
{
MessageBox.Show("按下Alt+S");
}
else if(sid==altd)
{
MessageBox.Show("按下Alt+D");
}
}
也可以在Application使用AddMessageFilter添加處理來代替重載WndProc,可以使form自身實(shí)現(xiàn)IMessageFilter接口,
注冊方法
復(fù)制代碼 代碼如下:
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
var form = new Form1();
Application.AddMessageFilter(form);
Application.Run(form);
}
實(shí)現(xiàn)接口
復(fù)制代碼 代碼如下:
bool IMessageFilter.PreFilterMessage(ref Message m)
{
const int WM_HOTKEY = 0x0312;//如果m.Msg的值為0x0312那么表示用戶按下了熱鍵
switch (m.Msg)
{
case WM_HOTKEY:
ProcessHotkey(m);//按下熱鍵時調(diào)用ProcessHotkey()函數(shù)
break;
}
//如果篩選消息并禁止消息被調(diào)度,則為 true;如果允許消息繼續(xù)到達(dá)下一個篩選器或控件,則為 false
return false ;
}
如果同時在AddMessageFilter和WndProc處理消息,順序是先IMessageFilter,然后調(diào)用WndProc
WPF方法:
WPF中的注冊方法和Winform一樣,只是WPF的Key枚舉的值不能正確監(jiān)聽,要引用System.Windows.Forms.Keys或者自己定義才可以正確注冊,窗口句柄也需要借助WindowInteropHelper來得到,處理函數(shù)的加入也和Winform不同,需要HwndSource來添加處理函數(shù)。
注冊及處理方法
復(fù)制代碼 代碼如下:
int alts, altd;
private void Window_Loaded(object sender, RoutedEventArgs e)
{
HwndSource hWndSource;
WindowInteropHelper wih = new WindowInteropHelper(this);
hWndSource = HwndSource.FromHwnd(wih.Handle);
//添加處理程序
hWndSource.AddHook(MainWindowProc);
alts = HotKey.GlobalAddAtom("Alt-S");
altd = HotKey.GlobalAddAtom("Alt-D");
HotKey.RegisterHotKey(wih.Handle, alts, HotKey.KeyModifiers.Alt, (int)System.Windows.Forms.Keys.S);
HotKey.RegisterHotKey(wih.Handle, altd, HotKey.KeyModifiers.Alt, (int)System.Windows.Forms.Keys.D);
}
private IntPtr MainWindowProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
{
switch (msg)
{
case HotKey.WM_HOTKEY:
{
int sid = wParam.ToInt32();
if (sid == alts)
{
MessageBox.Show("按下Alt+S");
}
else if (sid == altd)
{
MessageBox.Show("按下Alt+D");
}
handled = true;
break;
}
}
return IntPtr.Zero;
}
注意:
如果注冊快捷鍵,RegisterHotKey中的fsModifiers參數(shù)為0,即None選項(xiàng),一些安全軟件會警報,可能因?yàn)檫@樣就可以全局監(jiān)聽鍵盤而造成安全問題
代碼下載
相關(guān)文章
C# CancellationToken和CancellationTokenSource的用法詳解
做了.net core之后,發(fā)現(xiàn)CancellationToken用的越來越平凡了。這也難怪,原來.net framework使用異步的不是很多,而.net core首推異步編程,到處可以看到Task的影子,而CancellationToken正好是異步Task的一個控制器,所以花點(diǎn)時間做個筆記2021-06-06
C# 表達(dá)式目錄樹Expression的實(shí)現(xiàn)
本文主要介紹了C# 表達(dá)式目錄樹Expression的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09
解析c#在未出現(xiàn)異常情況下查看當(dāng)前調(diào)用堆棧的解決方法
本篇文章是對c#在未出現(xiàn)異常情況下查看當(dāng)前調(diào)用堆棧的解決方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
c#方法中調(diào)用參數(shù)的值傳遞方式和引用傳遞方式以及ref與out的區(qū)別深入解析
以下是對c#方法中調(diào)用參數(shù)的值傳遞方式和引用傳遞方式,以及ref與out的區(qū)進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下2013-07-07
C#學(xué)習(xí)基礎(chǔ)概念二十五問續(xù)2
C#學(xué)習(xí)基礎(chǔ)概念二十五問續(xù)2...2007-04-04

