C#利用HOOK注册全局热键

C#每个人都应该看到使用HOOK注册全局热键的效果。例如,即使在后台运行,QQ的截图功能仍然可以获得键盘输入。 应该注意的是,当C#程序失去焦点时,键盘事件(如KeyDown)无法获得消息。使用全局HOOK技术,即使程序失去焦点或者背景仍然在运行,用户的键盘输入也可以获得。

为了方便并清楚地看到HOCK获取和拦截消息的能力,这里我使用了两个标签,分别显示来自HOCK的消息和来自button3的键盘消息。当HOOK被按下不截取消息时,标签1和标签2的内容是相同的。当HOOK被按下截取消息时,按钮的KeyDown事件将不会从用户那里接收到按键信息。

C#代码
KeyboardHookLib.cs键盘HOOK管理类:

/// <summary>
/// 键盘Hook管理类
/// </summary>
public class KeyboardHookLib
{
private const int WH_KEYBOARD_LL = 13; //键盘
//键盘处理事件委托 ,当捕获键盘输入时调用定义该委托的方法.
private delegate int HookHandle(int nCode, int wParam, IntPtr lParam);
//客户端键盘处理事件
public delegate void ProcessKeyHandle(HookStruct param, out bool handle);
//接收SetWindowsHookEx返回值
private static int _hHookValue = 0;
//勾子程序处理事件
private HookHandle _KeyBoardHookProcedure;
//Hook结构
[StructLayout(LayoutKind.Sequential)]
public class HookStruct
{
public int vkCode;
public int scanCode;
public int flags;
public int time;
public int dwExtraInfo;
}
//设置HOOK
[DllImport("user32.dll")]
private static extern int SetWindowsHookEx(int idHook, HookHandle lpfn, IntPtr hInstance, int threadId);
//取消HOOK
[DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
private static extern bool UnhookWindowsHookEx(int idHook);
//调用下一个HOOK
[DllImport("user32.dll")]
private static extern int CallNextHookEx(int idHook, int nCode, int wParam, IntPtr lParam);
//获取当前线程ID
[DllImport("kernel32.dll")]
private static extern int GetCurrentThreadId();
//Gets the main module for the associated process.
[DllImport("kernel32.dll")]
private static extern IntPtr GetModuleHandle(string name);
private IntPtr _hookWindowPtr = IntPtr.Zero;
//构造器
public KeyboardHookLib() { }
//外部调用的键盘处理事件
private static ProcessKeyHandle _clientMethod = null;
/// <summary>
/// 安装HOOK
/// </summary>
/// <param name="hookProcess">外部调用的键盘处理事件</param>
public void InstallHook(ProcessKeyHandle clientMethod)
{
_clientMethod = clientMethod;
// 安装键盘HOOK
if (_hHookValue == 0)
{
_KeyBoardHookProcedure = new HookHandle(OnHookProc);
_hookWindowPtr = GetModuleHandle(Process.GetCurrentProcess().MainModule.ModuleName);
_hHookValue = SetWindowsHookEx(
WH_KEYBOARD_LL,
_KeyBoardHookProcedure,
_hookWindowPtr,
0);
//如果设置HOOK失败. 
if (_hHookValue == 0) UninstallHook();
}
}
//取消HOOK事件 
public void UninstallHook()
{
if (_hHookValue != 0)
{
bool ret = UnhookWindowsHookEx(_hHookValue);
if (ret) _hHookValue = 0;
}
}
//HOOK事件内部调用,调用_clientMethod方法转发到客户端应用。
private static int OnHookProc(int nCode, int wParam, IntPtr lParam)
{
if (nCode >= 0)
{
//转换结构
HookStruct hookStruct = (HookStruct)Marshal.PtrToStructure(lParam, typeof(HookStruct));
if (_clientMethod != null)
{
bool handle = false;
//调用客户提供的事件处理程序。
_clientMethod(hookStruct, out handle);
if (handle) return 1; //1:表示拦截键盘,return 退出
}
}
return CallNextHookEx(_hHookValue, nCode, wParam, lParam);
}
}

在Form1_Load事件中注册安装HOOK或者在按钮点击事件中注册安装HOOK。
_keyboardHook = new KeyboardHookLib();
_keyboardHook.InstallHook(this.OnKeyPress);
注意声明全局变量
KeyboardHookLib _keyboardHook;
HOOK处理方法
 /// <summary>
/// 客户端键盘捕捉事件.
/// </summary>
/// <param name="hookStruct">由Hook程序发送的按键信息</param>
/// <param name="handle">是否拦截</param>
public void OnKeyPress(KeyboardHookLib.HookStruct hookStruct, out bool handle)
{
//是否拦截这个键
handle = false;
Keys key = (Keys)hookStruct.vkCode;
if (key.ToString() == "F8")
{
//拦截F8
handle = true;
}
else if (key.ToString() == "F12")
{
//拦截F12
handle = true;
}
label1.Text = "HOOK按下了:" + key;
}
在按钮3的KeyDown事件中写下获取按键代码:
label2.Text = "按下了:" + e.KeyCode;

在关闭窗口事件Form1_FormClosed或者按钮点击事件中卸载HOOK
 if (_keyboardHook != null) _keyboardHook.UninstallHook();

整个HOOK全局热键和KeyDown效果程序完成。至于HOOK全局热键能做什么,这取决于每个人如何使用它。心理阴郁的人用它作为键盘记录器,秘密记录别人的输入,截取各种密码隐私等。 
哈哈哈,不要试了,杀柔首先是HOOK,当然没有注入功能,一般不会被杀,最后警告大家不要伤害别人的电脑哟~ 

转载请注明出处:C#利用HOOK注册全局热键
本文链接:https://www.cmezy.com/blog/chook
 

评论0

请先

没有账号? 忘记密码?