鼠标键盘钩子捕获(初版)

兴趣使然,想找到C#鼠标键盘钩子代码在网上查找资料发现C#鼠标钩子的资料不太完善,在实际操作过程中发现有些效果也不太理想,

所以自己进行了修改完善,当然,学识有限,还有很多地方需要完善,

欢迎指正!

这个项目不能直接启动,需要使用Ctrl+F5非调试启动或者直接启动bin文件下编译程序才不出错

:这个只是学习参考,可别动坏脑筋

  1 using System;
  2 
  3 using System.Runtime.InteropServices;
  4 using System.Windows.Forms;
  5 using System.Reflection;
  6 
  7 namespace autoInput.Support
  8 {
  9     /// <summary>
 10     /// 键盘钩子
 11     /// [以下代码来自某网友,并由“Murphy丶悦”改造完善]
 12     /// [仅供参考学习,请勿进行非法操作]
 13     /// </summary>
 14     public class KeyboardHook
 15     {
 16         public event KeyEventHandler KeyDownEvent;
 17         public event KeyPressEventHandler KeyPressEvent;
 18         public event KeyEventHandler KeyUpEvent;
 19         public event MouseEventHandler MouseDownDouble;
 20         public event MouseEventHandler MouseDown;
 21         public event MouseEventHandler MouseDownRight;
 22         public event MouseEventHandler Mousemov;
 23         public delegate int HookProc(int nCode, Int32 wParam, IntPtr lParam);
 24         static int hKeyboardHook = 0; //声明键盘钩子处理的初始值
 25         static int hMouseHook = 0;//申明鼠标钩子处理的初始值
 26         //值在Microsoft SDK的Winuser.h里查询
 27         public const int WH_KEYBOARD_LL = 13;   //线程键盘钩子监听鼠标消息设为2,全局键盘监听鼠标消息设为13
 28         HookProc KeyboardHookProcedure; //声明KeyboardHookProcedure作为HookProc类型
 29         HookProc MouseHookProceduremz; //声明MouseHookProcedure作为HookProc类型
 30 
 31         //键盘结构
 32         [StructLayout(LayoutKind.Sequential)]
 33         public class KeyboardHookStruct
 34         {
 35             public int vkCode;  //定一个虚拟键码。该代码必须有一个价值的范围1至254
 36             public int scanCode; // 指定的硬件扫描码的关键
 37             public int flags;  // 键标志
 38             public int time; // 指定的时间戳记的这个讯息
 39             public int dwExtraInfo; // 指定额外信息相关的信息
 40         }
 41         //使用此功能,安装了一个钩子
 42         [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
 43         public static extern int SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, int threadId);
 44 
 45 
 46         //调用此函数卸载钩子
 47         [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
 48         public static extern bool UnhookWindowsHookEx(int idHook);
 49 
 50 
 51         //使用此功能,通过信息钩子继续下一个钩子
 52         [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
 53         public static extern int CallNextHookEx(int idHook, int nCode, Int32 wParam, IntPtr lParam);
 54 
 55         // 取得当前线程编号(线程钩子需要用到)
 56         [DllImport("kernel32.dll")]
 57         static extern int GetCurrentThreadId();
 58 
 59         //使用WINDOWS API函数代替获取当前实例的函数,防止钩子失效
 60         [DllImport("kernel32.dll")]
 61         public static extern IntPtr GetModuleHandle(string name);
 62 
 63         //引入系统的双击时间
 64         [DllImport("user32.dll")]
 65         public static extern int GetDoubleClickTime();
 66 
 67         public void Start()
 68         {
 69             //安装鼠标钩子
 70             if (hMouseHook == 0)
 71             {
 72                 MouseHookProceduremz = new HookProc(MouseHookProcedure);
 73                 hMouseHook = SetWindowsHookEx(14, MouseHookProceduremz, Marshal.GetHINSTANCE(Assembly.GetExecutingAssembly().GetModules()[0]), 0);
 74                 if (hMouseHook == 0)
 75                 {
 76                     Stopmouse();
 77                     MessageBox.Show("安装鼠标钩子失败");
 78                 }
 79             }
 80             // 安装键盘钩子
 81             if (hKeyboardHook == 0)
 82             {
 83                 KeyboardHookProcedure = new HookProc(KeyboardHookProc);
 84                 hKeyboardHook = SetWindowsHookEx(WH_KEYBOARD_LL, KeyboardHookProcedure, GetModuleHandle(System.Diagnostics.Process.GetCurrentProcess().MainModule.ModuleName), 0);
 85                 // hKeyboardHook = SetWindowsHookEx(WH_KEYBOARD_LL,KeyboardHookProcedure, Marshal.GetHINSTANCE(Assembly.GetExecutingAssembly().GetModules()[0]), 0);
 86                 //键盘线程钩子
 87                 // SetWindowsHookEx( 2,KeyboardHookProcedure, IntPtr.Zero,GetCurrentThreadId());
 88                 //指定要监听的线程idGetCurrentThreadId(),
 89                 //键盘全局钩子,需要引用空间(using System.Reflection;)`
 90                 //关于SetWindowsHookEx (int idHook, HookProc lpfn, IntPtrhInstance, int threadId)函数将钩子加入到钩子链表中,说明一下四个参数:
 91                 //idHook 钩子类型,即确定钩子监听何种消息,上面的代码中设为2,即监听键盘消息并且是线程钩子,如果是全局钩子监听键盘消息应设为13,
 92                 //线程钩子监听鼠标消息设为7,全局钩子监听鼠标消息设为14。
 93                 //lpfn钩子子程的地址指针。如果dwThreadId参数为0 或是一个由别的进程创建的线程的标识,lpfn必须指向DLL中的钩子子程。 除此以外,
 94                 //lpfn可以指向当前进程的一段钩子子程代码。钩子函数的入口地址,当钩子钩到任何消息后便调用这个函数。hInstance应用程序实例的句柄。
 95                 //标识包含lpfn所指的子程的DLL。如果threadId 标识当前进程创建的一个线程,而且子
 96                 //程代码位于当前进程,hInstance必须为NULL。可以很简单的设定其为本应用程序的实例句柄。threadId 与安装的钩子子程相关联的线程的标识符
 97                 //如果为0,钩子子程与所有的线程关联,即为全局钩子
 98                 //如果SetWindowsHookEx失败
 99                 if (hKeyboardHook == 0)
100                 {
101                     Stop();
102                     MessageBox.Show("安装键盘钩子失败");
103                 }
104             }
105         }
106         private void Stopmouse()
107         {
108             bool retKeyboard = true;
109             if (hMouseHook != 0)
110             {
111                 retKeyboard = UnhookWindowsHookEx(hMouseHook);
112                 hMouseHook = 0;
113             }
114             if (!(retKeyboard)) MessageBox.Show("卸载鼠标钩子失败");
115         }
116         #region 鼠标钩子结构体
117         //声明一个Point的封送类型 
118         [StructLayout(LayoutKind.Sequential)]
119         public struct POINT
120         {
121             public int x;
122             public int y;
123         }
124         //声明鼠标钩子的封送结构类型 
125         [StructLayout(LayoutKind.Sequential)]
126         public struct MouseHookStruct
127         {
128             public POINT pt;
129             public int hWnd;
130             public int wHitTestCode;
131             public int dwExtraInfo;
132         }
133         //定义鼠标常数
134         private const int WM_MOUSEMOVE = 0x200;//鼠标移动 512(int)
135         private const int WM_LBUTTONDOWN = 0x201;//鼠标左键 513(int)
136         private const int WM_RBUTTONDOWN = 0x204;//鼠标右键 516(int)
137         private const int WM_MBUTTONDOWN = 0x207;//鼠标中健 519(int)
138         private const int WM_LBUTTONUP = 0x202;//左键弹起 514(int)
139         private const int WM_RBUTTONUP = 0x205;//右键弹起 517(int)
140         private const int WM_MBUTTONUP = 0x208;//中健弹起 520(int)
141         private const int WM_LBUTTONDBLCLK = 0x203;//双击左键 515(int)
142         private const int WM_RBUTTONDBLCLK = 0x206;//双击右键 518(int)
143         private const int WM_MBUTTONDBLCLK = 0x209;//双击中健 521(int)
144 
145         //对于鼠标双击的动作捕获,都是认为可以捕获WM_LBUTTONDBLCLK这个消息,通过该消息来设定是否为双击,
146         //但实际挂在钩子后,Windows并不会直接捕获到WM_LBUTTONDBLCLK这个消息,而是两次WM_LBUTTONDOWN消息,
147         //因此,如果要实现双击的捕获,可考虑计算两次点击的时间间隔,和系统的双击时间间隔进行比较,从而判定是否为双击。
148 
149         //在鼠标钩子中右键双击两次可以捕获到两次单击右键,
150         //但是在这个类拿到消息并进行鼠标常数更改为双击后在窗口中只能显示出单击一次
151         //另在各类资料中显示的中键鼠标常数全部无法捕获到
152         #endregion
153         //记录初始时间进行时间复位
154         private static int InitialTime = 0;//对单击两次后时间复位,防止无限判定双击事件
155         //上一次单击对应按键时间
156         private static int LeftClickTime = 0;//上一次单击左键时间
157         private static int RightClickTime = 0;//上一次单击右键时间
158         private static int MiddleClickTime = 0;//上一次单击中键时间
159         //上一次单击事件坐标
160         private int lastX = 0;//上一次鼠标X轴位置
161         private int lastY = 0;//上一次鼠标Y轴位置
162         public int i = 0;
163         private int MouseHookProcedure(int nCode, int wParam, IntPtr lParam)
164         {
165             int clickCount = 0;
166             if ((nCode >= 0) && (MouseDown != null || MouseDownDouble != null || Mousemov != null || MouseDownRight != null))
167             {
168                 MouseButtons button = MouseButtons.None;
169 
170                 MouseHookStruct MyMouseHookStruct = (MouseHookStruct)Marshal.PtrToStructure(lParam, typeof(MouseHookStruct));
171                 int X = MyMouseHookStruct.pt.x;//当前单击事件X轴坐标
172                 int Y = MyMouseHookStruct.pt.y;//当前单击事件Y轴坐标
173 
174                 //进行时间比对;手动对鼠标按键命令更改
175                 if (wParam == WM_LBUTTONDOWN)//对单击左键进行判定
176                     wParam = changeGorge(wParam, LeftClickTime, WM_LBUTTONDBLCLK, X, Y);
177                 if(wParam== WM_RBUTTONDOWN)//对单击右键进行判定
178                     wParam= changeGorge(wParam, RightClickTime, WM_RBUTTONDBLCLK, X, Y);
179                 if (wParam == WM_MBUTTONDOWN)//对单击中键进行判定
180                     wParam = changeGorge(wParam, MiddleClickTime, WM_MBUTTONDBLCLK, X, Y);
181 
182                 switch (wParam)
183                 {
184                     //左键按下
185                     case WM_LBUTTONDOWN:
186                         button = MouseButtons.Left;
187                         clickCount = 1;
188                         break;
189                     //左键抬起
190                     case WM_LBUTTONUP:
191                         button = MouseButtons.Left;
192                         clickCount = 1;
193                         break;
194                     //双击左键
195                     case WM_LBUTTONDBLCLK:
196                         button = MouseButtons.Left;
197                         clickCount = 2;
198                         break;
199                     //中键按下
200                     case WM_MBUTTONDOWN:
201                         button = MouseButtons.Middle;
202                         clickCount = 1;
203                         break;
204                     //中键抬起
205                     case WM_MBUTTONUP:
206                         button = MouseButtons.Middle;
207                         clickCount = 1;
208                         break;
209                     //双击中键
210                     case WM_MBUTTONDBLCLK:
211                         button = MouseButtons.Middle;
212                         clickCount = 2;
213                         break;
214                     //右键按下
215                     case WM_RBUTTONDOWN:
216                         button = MouseButtons.Right;
217                         clickCount = 1;
218                         break;
219                     //右键抬起
220                     case WM_RBUTTONUP:
221                         button = MouseButtons.Right;
222                         clickCount = 1;
223                         break;
224                     //双击右键
225                     case WM_RBUTTONDBLCLK:
226                         button = MouseButtons.Right;
227                         clickCount = 2;
228                         break;
229                     case WM_MOUSEMOVE:
230                         clickCount = 0;
231                         break;
232                 }
233 
234                 MouseEventArgs e = new MouseEventArgs(button, clickCount, MyMouseHookStruct.pt.x, MyMouseHookStruct.pt.y, 0);
235 
236                 //raise KeyDown;
237                 if (MouseDown != null && wParam == WM_LBUTTONDOWN)//单击事件
238                 {
239                     MouseDown(this, e);
240                 }
241                 if (this.MouseDownDouble != null && wParam == WM_LBUTTONDBLCLK)//双击事件
242                 {
243                     MouseDownDouble(this, e);
244                 }
245                 if (this.Mousemov != null && wParam == WM_LBUTTONDOWN && wParam == WM_MOUSEMOVE) //拖动事件
246                 {
247                     Mousemov(this, e);
248                 }
249                 if (this.MouseDownRight != null && wParam == WM_RBUTTONDOWN)//右击事件
250                 {
251                     MouseDownRight(this, e);
252                 }
253             }
254 
255             //如果返回1,则结束消息,这个消息到此为止,不再传递。
256             //如果返回0或调用CallNextHookEx函数则消息出了这个钩子继续往下传递,也就是传给消息真正的接受者 
257             return CallNextHookEx(hKeyboardHook, nCode, wParam, lParam);
258         }
259 
260         public void Stop()
261         {
262             bool retKeyboard = true;
263             if (hKeyboardHook != 0)
264             {
265                 retKeyboard = UnhookWindowsHookEx(hKeyboardHook);
266                 hKeyboardHook = 0;
267             }
268             if (!(retKeyboard)) MessageBox.Show("卸载键盘钩子失败");
269         }
270         //ToAscii职能的转换指定的虚拟键码和键盘状态的相应字符或字符
271         [DllImport("user32")]
272         public static extern int ToAscii(int uVirtKey, //[in] 指定虚拟关键代码进行翻译。
273                                          int uScanCode, // [in] 指定的硬件扫描码的关键须翻译成英文。高阶位的这个值设定的关键,如果是(不压)
274                                          byte[] lpbKeyState, // [in] 指针,以256字节数组,包含当前键盘的状态。每个元素(字节)的数组包含状态的一个关键。如果高阶位的字节是一套,关键是下跌(按下)。在低比特,如果设置表明,关键是对切换。在此功能,只有肘位的CAPS LOCK键是相关的。在切换状态的NUM个锁和滚动锁定键被忽略。
275                                          byte[] lpwTransKey, // [out] 指针的缓冲区收到翻译字符或字符。
276                                          int fuState); // [in] Specifieswhether a menu is active. This parameter must be 1 if a menu is active, or 0otherwise.
277 
278         //获取按键的状态
279         [DllImport("user32")]
280         public static extern int GetKeyboardState(byte[] pbKeyState);
281 
282 
283         [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
284         private static extern short GetKeyState(int vKey);
285 
286         private const int WM_KEYDOWN = 0x100;//KEYDOWN
287         private const int WM_KEYUP = 0x101;//KEYUP
288         private const int WM_SYSKEYDOWN = 0x104;//SYSKEYDOWN
289         private const int WM_SYSKEYUP = 0x105;//SYSKEYUP
290 
291         private int KeyboardHookProc(int nCode, Int32 wParam, IntPtr lParam)
292         {
293             // 侦听键盘事件
294             if ((nCode >= 0) && (KeyDownEvent != null || KeyUpEvent != null || KeyPressEvent != null))
295             {
296                 KeyboardHookStruct MyKeyboardHookStruct = (KeyboardHookStruct)Marshal.PtrToStructure(lParam, typeof(KeyboardHookStruct));
297                 // raise KeyDown
298                 if (KeyDownEvent != null && (wParam == WM_KEYDOWN || wParam == WM_SYSKEYDOWN))
299                 {
300                     Keys keyData = (Keys)MyKeyboardHookStruct.vkCode;
301                     KeyEventArgs e = new KeyEventArgs(keyData);
302                     KeyDownEvent(this, e);
303                 }
304 
305                 //键盘按下
306                 if (KeyPressEvent != null && wParam == WM_KEYDOWN)
307                 {
308                     byte[] keyState = new byte[256];
309                     GetKeyboardState(keyState);
310 
311                     byte[] inBuffer = new byte[2];
312                     if (ToAscii(MyKeyboardHookStruct.vkCode, MyKeyboardHookStruct.scanCode, keyState, inBuffer, MyKeyboardHookStruct.flags) == 1)
313                     {
314                         KeyPressEventArgs e = new KeyPressEventArgs((char)inBuffer[0]);
315                         KeyPressEvent(this, e);
316                     }
317                 }
318 
319                 // 键盘抬起
320                 if (KeyUpEvent != null && (wParam == WM_KEYUP || wParam == WM_SYSKEYUP))
321                 {
322                     Keys keyData = (Keys)MyKeyboardHookStruct.vkCode;
323                     KeyEventArgs e = new KeyEventArgs(keyData);
324                     KeyUpEvent(this, e);
325                 }
326 
327             }
328             //如果返回1,则结束消息,这个消息到此为止,不再传递。
329             //如果返回0或调用CallNextHookEx函数则消息出了这个钩子继续往下传递,也就是传给消息真正的接受者
330             return CallNextHookEx(hKeyboardHook, nCode, wParam, lParam);
331         }
332 
333         /// <summary>
334         /// 对单击事件进行判定是否为双击事件并进行修改
335         /// </summary>
336         /// <param name="wParams">串口内的单击事件消息</param>
337         /// <param name="lastClickTime">上一次单击事件事件</param>
338         /// <param name="doubleClick">正确双击事件消息</param>
339         /// <param name="X">上一次单击X轴坐标</param>
340         /// <param name="Y">上一次单击Y轴坐标</param>
341         /// <returns></returns>
342         public int changeGorge(int wParam,int lastClickTime,int doubleClick,int X,int Y)
343         {
344             int deltaMs = GetDoubleClickTime() + lastClickTime;
345             if (deltaMs >= dateTimeToInt(DateTime.Now) && lastX == X && lastY == Y)//在系统双击判定时间内同一坐标单击两次更改为双击事件
346             {
347                 wParam = doubleClick;
348                 if (lastClickTime == LeftClickTime)
349                     LeftClickTime = InitialTime;
350                 else if (lastClickTime == RightClickTime)
351                     RightClickTime = InitialTime;
352                 else if (lastClickTime == MiddleClickTime)
353                     MiddleClickTime = InitialTime;
354             }
355             else
356             {
357                 if (lastClickTime == LeftClickTime)
358                 {
359                     LeftClickTime = dateTimeToInt(DateTime.Now);
360                     RightClickTime = InitialTime;
361                     MiddleClickTime = InitialTime;
362                 }
363                 else if (lastClickTime == RightClickTime)
364                 {
365                     RightClickTime = dateTimeToInt(DateTime.Now);
366                     LeftClickTime = InitialTime;
367                     MiddleClickTime = InitialTime;
368                 }
369                 else if (lastClickTime == MiddleClickTime)
370                 {
371                     MiddleClickTime = dateTimeToInt(DateTime.Now);
372                     LeftClickTime = InitialTime;
373                     RightClickTime = InitialTime;
374                 }
375             }
376             lastX = X;
377             lastY = Y;
378             return wParam;
379         }
380         /// <summary>
381         /// 转换时间格式为HHmmssfff
382         /// </summary>
383         /// <param name="time">需转换的时间</param>
384         /// <returns></returns>
385         private int dateTimeToInt(DateTime time)
386         {
387             //转换时间格式为int类型并精确到毫秒
388             return Convert.ToInt32(time.ToString("HHmmssfff"));
389         }
390         ~KeyboardHook()
391         {
392             Stop();
393         }
394     }
395 }

下面是显示界面,使用的是WinFrom窗口,在窗口内用一个文本框和一个定时器

 1 using System;
 2 using System.Windows.Forms;
 3 
 4 using autoInput.Support;
 5 
 6 namespace autoInput
 7 {
 8     public partial class autoInput : Form
 9     {
10         public autoInput()
11         {
12             InitializeComponent();
13         }
14 
15         KeyboardHook k_hook;
16         private void autoInput_Load(object sender, EventArgs e)
17         {
18             k_hook = new KeyboardHook();
19             k_hook.KeyDownEvent += new KeyEventHandler(hook_KeyDown);//全局按键事件
20             k_hook.MouseDown += k_hook_MouseDown;//全局鼠标单击事件
21             k_hook.MouseDownDouble += k_hook_MouseDown;//双击事件
22             k_hook.Mousemov += k_hook_MouseDown;//拖动事件
23             k_hook.MouseDownRight += k_hook_MouseDown;//右击事件
24             k_hook.Start();//安装键盘钩子
25         }
26 
27         public string button = string.Empty;
28         public string keyboardRecordText = string.Empty;
29         //单击事件
30         void k_hook_MouseDown(object sender, MouseEventArgs e) 
31         {
32             if (keyboardRecordText != string.Empty && e.Button.ToString() != button)
33                 showText();
34             button = e.Button.ToString();
35             keyboardRecordText=string.Format("Button:{0} Clicks:{1} \tX:{2} Y:{3} \t{4} \n", e.Button, e.Clicks, e.X, e.Y, DateTime.Now.ToString("HH:mm:ss.fff"));
36 
37             if (e.Clicks == 2)
38             {
39                 showText();
40                 keyboardRecordText = string.Empty;
41 
42             }
43             else
44             {
45                 this.timer.Enabled = true;
46                 timer.Interval = KeyboardHook.GetDoubleClickTime();
47             }
48         }
49 
50         /// <summary>
51         /// 键盘事件
52         /// </summary>
53         /// <param name="sender"></param>
54         /// <param name="e"></param>
55         private void hook_KeyDown(object sender, KeyEventArgs e)
56         {
57             if (e.KeyValue == (int)Keys.Space)
58             {
59                 keyboardRecord.Text += " ";
60             }
61             else if (e.KeyValue == (int)Keys.Enter)
62             {
63                 keyboardRecord.Text += "\n";
64             }
65             else
66             {
67                 keyboardRecord.Text += e.KeyData.ToString();
68             }
69         }
70 
71         /// <summary>
72         /// 定时器
73         /// </summary>
74         /// <param name="sender"></param>
75         /// <param name="e"></param>
76         private void timer_Tick(object sender, EventArgs e)
77         {
78             showText();
79             this.timer.Enabled = false;
80         }
81         public void showText()
82         {
83             //keyboardRecord窗口程序内文本框的名称
84             keyboardRecord.Text += keyboardRecordText;
85             keyboardRecordText = string.Empty;
86         }
87     }
88 }

下面是我的项目截图

posted @ 2017-03-21 22:16  Murphy丶悦  阅读(2334)  评论(0编辑  收藏  举报