今天,有个同事问我,怎样在C#中使用全局钩子?以前写的全局钩子都是用unmanaged C或C++写个DLL来实现,可大家都知道,C#是基于.Net Framework的,是managed,怎么实现全局钩子呢?于是开始到网上搜索,好不容易找到一篇,318804 - HOW TO: Set a Windows Hook in Visual C# .NET,里面详细的说明了如何使用鼠标钩子捕获鼠标的移动等,可是,它只能在Application里起作用,出了Application就没用了,就是说它还是没有实现全局钩子,而且文章结尾处说:“Global Hooks are not supported in the .NET Framework...”,这可怎么办呢?

  别担心,办法总是有的,经过一番摸索以后,发现WH_KEYBORAD_LL和WH_MOUSE_LL这两个low-level的hook可以被安装成全局的,这就好办了,我们不妨用这两个low-level的hook替换掉WH_KEYBORAD和WH_MOUSE,于是开始测试。结果成功了,在C#里实现了全局钩子。

  我们来看一下主要代码段。

  首先倒入所需要的windows函数,主要有三个,SetWindowsHookEX用来安装钩子,UnhookWindowsHookEX用来卸载钩子以及CallNextHookEX用来将hook信息传递到链表中下一个hook处理过程。

  1. [DllImport("user32.dll", CharSet = CharSet.Auto,
  2. CallingConvention = CallingConvention.StdCall, SetLastError = true)]
  3. private static extern int SetWindowsHookEx(
  4. int idHook,
  5. HookProc lpfn,
  6. IntPtr hMod,
  7. int dwThreadId);
  8. [DllImport("user32.dll", CharSet = CharSet.Auto,
  9. CallingConvention = CallingConvention.StdCall, SetLastError = true)]
  10. private static extern int UnhookWindowsHookEx(int idHook);
  11. [DllImport("user32.dll", CharSet = CharSet.Auto,
  12. CallingConvention = CallingConvention.StdCall)]
  13. private static extern int CallNextHookEx(
  14. int idHook,
  15. int nCode,
  16. int wParam,
  17. IntPtr lParam);
  18.   下面是有关这两个low-level hook在Winuser.h中的定义:
  19. /// <summary>
  20. /// Windows NT/2000/XP: Installs a hook procedure that monitors low-level mouse input events.
  21. /// </summary>
  22. private const int WH_MOUSE_LL       = 14;
  23. /// <summary>
  24. /// Windows NT/2000/XP: Installs a hook procedure that monitors low-level keyboard input events.
  25. /// </summary>
  26. private const int WH_KEYBOARD_LL    = 13;
  27.   在安装全局钩子的时候,我们就要做替换了,将WH_MOUSE和WH_KEYBORAD分别换成WH_MOUSE_LL和WH_KEYBORAD_LL:
  28. //install hook
  29. hMouseHook = SetWindowsHookEx(
  30. WH_MOUSE_LL, //原来是WH_MOUSE
  31. MouseHookProcedure,
  32. Marshal.GetHINSTANCE(
  33. Assembly.GetExecutingAssembly().GetModules()[0]),
  34. 0);
  35. //install hook
  36. hKeyboardHook = SetWindowsHookEx(
  37. WH_KEYBOARD_LL, //原来是WH_KEYBORAD
  38. KeyboardHookProcedure,
  39. Marshal.GetHINSTANCE(
  40. Assembly.GetExecutingAssembly().GetModules()[0]),
  41. 0);
  42.   这样替换了之后,我们就可以实现全局钩子了,而且,不需要写DLL。看一下程序运行情况:
  43.   下面是关于鼠标和键盘的两个Callback函数:
  44. private int MouseHookProc(int nCode, int wParam, IntPtr lParam)
  45. {
  46. // if ok and someone listens to our events
  47. if ((nCode >= 0) && (OnMouseActivity != null))
  48. {
  49. //Marshall the data from callback.
  50. MouseLLHookStruct mouseHookStruct = (MouseLLHookStruct)Marshal.PtrToStructure(lParam, typeof(MouseLLHookStruct));
  51. //detect button clicked
  52. MouseButtons button = MouseButtons.None;
  53. short mouseDelta = 0;
  54. switch (wParam)
  55. {
  56. case WM_LBUTTONDOWN:
  57. //case WM_LBUTTONUP:
  58. //case WM_LBUTTONDBLCLK:
  59. button = MouseButtons.Left;
  60. break;
  61. case WM_RBUTTONDOWN:
  62. //case WM_RBUTTONUP:
  63. //case WM_RBUTTONDBLCLK:
  64. button = MouseButtons.Right;
  65. break;
  66. case WM_MOUSEWHEEL:
  67. //If the message is WM_MOUSEWHEEL, the high-order word of mouseData member is the wheel delta.
  68. //One wheel click is defined as WHEEL_DELTA, which is 120.
  69. //(value >> 16) & 0xffff; retrieves the high-order word from the given 32-bit value
  70. mouseDelta = (short)((mouseHookStruct.mouseData >> 16) & 0xffff);
  71. //TODO: X BUTTONS (I havent them so was unable to test)
  72. //If the message is WM_XBUTTONDOWN, WM_XBUTTONUP, WM_XBUTTONDBLCLK, WM_NCXBUTTONDOWN, WM_NCXBUTTONUP,
  73. //or WM_NCXBUTTONDBLCLK, the high-order word specifies which X button was pressed or released,
  74. //and the low-order word is reserved. This value can be one or more of the following values.
  75. //Otherwise, mouseData is not used.
  76. break;
  77. }
  78. //double clicks
  79. int clickCount = 0;
  80. if (button != MouseButtons.None)
  81. if (wParam == WM_LBUTTONDBLCLK || wParam == WM_RBUTTONDBLCLK) clickCount = 2;
  82. else clickCount = 1;
  83. //generate event
  84. MouseEventArgs e = new MouseEventArgs(
  85. button,
  86. clickCount,
  87. mouseHookStruct.pt.x,
  88. mouseHookStruct.pt.y,
  89. mouseDelta);
  90. //raise it
  91. OnMouseActivity(this, e);
  92. }
  93. //call next hook
  94. return CallNextHookEx(hMouseHook, nCode, wParam, lParam);
  95. }
  96. private int KeyboardHookProc(int nCode, Int32 wParam, IntPtr lParam)
  97. {
  98. //indicates if any of underlaing events set e.Handled flag
  99. bool handled = false;
  100. //it was ok and someone listens to events
  101. if ((nCode >= 0) && (KeyDown != null || KeyUp != null || KeyPress != null))
  102. {
  103. //read structure KeyboardHookStruct at lParam
  104. KeyboardHookStruct MyKeyboardHookStruct = (KeyboardHookStruct)Marshal.PtrToStructure(lParam, typeof(KeyboardHookStruct));
  105. //raise KeyDown
  106. if (KeyDown != null && (wParam == WM_KEYDOWN || wParam == WM_SYSKEYDOWN))
  107. {
  108. Keys keyData = (Keys)MyKeyboardHookStruct.vkCode;
  109. KeyEventArgs e = new KeyEventArgs(keyData);
  110. KeyDown(this, e);
  111. handled = handled || e.Handled;
  112. }
  113. // raise KeyPress
  114. if (KeyPress != null && wParam == WM_KEYDOWN)
  115. {
  116. bool isDownShift = ((GetKeyState(VK_SHIFT) & 0x80) == 0x80 ? true : false);
  117. bool isDownCapslock = (GetKeyState(VK_CAPITAL) != 0 ? true : false);
  118. byte[] keyState = new byte[256];
  119. GetKeyboardState(keyState);
  120. byte[] inBuffer = new byte[2];
  121. if (ToAscii(MyKeyboardHookStruct.vkCode,
  122. MyKeyboardHookStruct.scanCode,
  123. keyState,
  124. inBuffer,
  125. MyKeyboardHookStruct.flags) == 1)
  126. {
  127. char key = (char)inBuffer[0];
  128. if ((isDownCapslock ^ isDownShift) && Char.IsLetter(key)) key = Char.ToUpper(key);
  129. KeyPressEventArgs e = new KeyPressEventArgs(key);
  130. KeyPress(this, e);
  131. handled = handled || e.Handled;
  132. }
  133. }
  134. // raise KeyUp
  135. if (KeyUp != null && (wParam == WM_KEYUP || wParam == WM_SYSKEYUP))
  136. {
  137. Keys keyData = (Keys)MyKeyboardHookStruct.vkCode;
  138. KeyEventArgs e = new KeyEventArgs(keyData);
  139. KeyUp(this, e);
  140. handled = handled || e.Handled;
  141. }
  142. }
  143. //if event handled in application do not handoff to other listeners
  144. if (handled)
  145. return 1;
  146. else
  147. return CallNextHookEx(hKeyboardHook, nCode, wParam, lParam);
  148. }

在C#中使用全局鼠标、键盘Hook的更多相关文章

  1. C#全局鼠标键盘Hook

    原文出自:http://www.cnblogs.com/iEgrhn/archive/2008/02/17/1071392.html using System; using System.Collec ...

  2. 如何在C#中使用全局鼠标、键盘Hook

    今天,有个同事问我,怎样在C#中使用全局钩子?以前写的全局钩子都是用unmanaged C或C++写个DLL来实现,可大家都知道,C#是基于.Net Framework的,是managed,怎么实现全 ...

  3. 全局鼠标钩子:WH_MOUSE_LL, 在【 win 10 上网本】上因为太卡,运行中丢失全局鼠标钩子

    一台几年前买的上网本,让我安装了一个 win 10,然后用来测试程序的时候, 发现 使用 SetWindowsHookEx(WH_MOUSE_LL, mouseHook, GetModuleHandl ...

  4. 将CodedUI Test 放到控制台程序中,模拟鼠标键盘操作

    CodedUI Test是微软的自动化测试工具,在VS中非常好用.可以用来模拟鼠标点击,键盘输入.但执行的时候必须要用mstest调用,无法传入参数(当然可以写入config文件中,但每次修改十分麻烦 ...

  5. [No0000AC]全局鼠标键盘模拟器

    之前网上下载的一位前辈写的工具,名叫:Dragon键盘鼠标模拟器,网址http://www.esc0.com/. 本软件能够录制键盘鼠标操作,并能按要求回放,对于重复的键盘鼠标操作,可以代替人去做,操 ...

  6. Python——pyHook监听鼠标键盘事件

    pyHook包为Windows中的全局鼠标和键盘事件提供回调. 底层C库报告的信息包括事件的时间,事件发生的窗口名称,事件的值,任何键盘修饰符等. 而正常工作需要pythoncom等操作系统的API的 ...

  7. hook 鼠标键盘消息实例分析

    1.木马控制及通信方法包含:双管道,port重用.反弹技术.Hook技术,今天重点引用介绍一下hook的使用方法,hook信息后能够将结果发送到hacker邮箱等.实现攻击的目的. 转自:http:/ ...

  8. C#鼠标键盘钩子

    using System;using System.Collections.Generic; using System.Reflection; using System.Runtime.Interop ...

  9. 键盘Hook【Delphi版】

    原文:https://www.cnblogs.com/edisonfeng/archive/2012/05/18/2507858.html 一.钩子的基本概念 a) Hook作用:监视windows消 ...

随机推荐

  1. selenium python (十三)对于分页的处理

    #!/usr/bin/python# -*- coding: utf-8 -*-__author__ = 'zuoanvip' #对于web上分页的功能,一般做如下操作:    #获取总页数    # ...

  2. LoadRunner中常见参数和变量

    1.参数和字符串变量的交换 ①lr_save_string(“hello world”,“param”)         将hello world 保存在参数 param中 ②lr_eval_stri ...

  3. jQuery中的bind绑定事件与文本框改变事件的临时解决方法

    暂时没有想到什么好的解决办法,我现在加了个浏览器判断非ie的话就注册blur事件,这样有个问题就是blur实在别的控件活动焦点的时候,txtStation控件注册的方法是为了填充它紧挨着的一个下拉列表 ...

  4. NGUI-制作位图字体以及图文混排

    制作字体过程 首先得下载一个位图制作工具Bitmap font generator,可以点击这里下载 1.新建txt文件,输入字体里面包含的文字 2.保存为utf-8格式:点击文件另存为,选择编码格式 ...

  5. 三年程序学习之二:(对web初认识)

    接着上一篇讲,之后第二天我就来公司上班了,主要是前端,CSS+DIV,table,网站维护之类的,这样的日子过了将近3个星期,一直没什么进展,自己也学不到什么技术,不过我觉得CSS+DIV我算是基础的 ...

  6. 装饰模式decorator

    C++设计模式——装饰模式 前言 在实际开发时,你有没有碰到过这种问题:开发一个类,封装了一个对象的核心操作,而这些操作就是客户使用该类时都会去调用的操作:而有一些非核心的操作,可能会使用,也可能不会 ...

  7. Container View Controller

    有时候,我们的Controler中包含有另一个controler view的view时,可以使用这种方式. https://developer.apple.com/library/ios/featur ...

  8. flex中文说明手册

    http://help.adobe.com/zh_CN/Flex/4.0/UsingFlashBuilder/WS6f97d7caa66ef6eb1e63e3d11b6c4d0d21-7f07.htm ...

  9. cocos2d-js屏幕任何位置点击开始的实现

    ctor:function () { this._super(); if ('mouse' in cc.sys.capabilities) cc.eventManager.addListener({ ...

  10. SharePoint咨询师之路:设计之前的那些事四:负载均衡 - web服务器

     提示:本系列只是一个学习笔记系列,大部分内容都可以从微软官方网站找到,本人只是按照自己的学习路径来学习和呈现这些知识.有些内容是自己的经验和积累,如果有不当之处,请指正. 容量管理 规模 体系结构 ...