private static void ForceUnsunscribeFromGlobalKeyboardEvents() { if (s_KeyboardHookHandle != 0) { //uninstall hook int result = HookManager.UnhookWindowsHookEx(s_KeyboardHookHandle); //reset invalid handle s_KeyboardHookHandle = 0; //Free up for GC s_KeyboardDelegate = null; //if failed and exception must be thrown if (result == 0) { //Returns the error code returned by the last unmanaged function called using platform invoke that has the DllImportAttribute.SetLastError flag set. int errorCode = Marshal.GetLastWin32Error(); //Initializes and throws a new instance of the Win32Exception class with the specified error. throw new Win32Exception(errorCode); } } }
/// <summary> /// A callback function which will be called every Time a mouse activity detected. /// </summary> /// <param name="nCode"> /// [in] Specifies whether the hook procedure must process the message. /// If nCode is HC_ACTION, the hook procedure must process the message. /// If nCode is less than zero, the hook procedure must pass the message to the /// CallNextHookEx function without further processing and must return the /// value returned by CallNextHookEx. /// </param> /// <param name="wParam"> /// [in] Specifies whether the message was sent by the current thread. /// If the message was sent by the current thread, it is nonzero; otherwise, it is zero. /// </param> /// <param name="lParam"> /// [in] Pointer to a CWPSTRUCT structure that contains details about the message. /// </param> /// <returns> /// If nCode is less than zero, the hook procedure must return the value returned by CallNextHookEx. /// If nCode is greater than or equal to zero, it is highly recommended that you call CallNextHookEx /// and return the value it returns; otherwise, other applications that have installed WH_CALLWNDPROC /// hooks will not receive hook notifications and may behave incorrectly as a result. If the hook /// procedure does not call CallNextHookEx, the return value should be zero. /// </returns> private static int MouseHookProc(int nCode, int wParam, IntPtr lParam) { if (nCode >= 0) { //Marshall the data from callback. HookManager.MouseLLHookStruct mouseHookStruct = (HookManager.MouseLLHookStruct)Marshal.PtrToStructure(lParam, typeof(HookManager.MouseLLHookStruct)); //detect button clicked MouseButtons button = MouseButtons.None; short mouseDelta = 0; int clickCount = 0; bool mouseDown = false; bool mouseUp = false; switch (wParam) { case HookManager.WM_LBUTTONDOWN: mouseDown = true; button = MouseButtons.Left; clickCount = 1; break; case HookManager.WM_LBUTTONUP: mouseUp = true; button = MouseButtons.Left; clickCount = 1; break; case HookManager.WM_LBUTTONDBLCLK: button = MouseButtons.Left; clickCount = 2; break; case HookManager.WM_RBUTTONDOWN: mouseDown = true; button = MouseButtons.Right; clickCount = 1; break; case HookManager.WM_RBUTTONUP: mouseUp = true; button = MouseButtons.Right; clickCount = 1; break; case HookManager.WM_RBUTTONDBLCLK: button = MouseButtons.Right; clickCount = 2; break; case HookManager.WM_MOUSEWHEEL: //If the message is WM_MOUSEWHEEL, the high-order word of MouseData member is the wheel delta. //One wheel click is defined as WHEEL_DELTA, which is 120. //(value >> 16) & 0xffff; retrieves the high-order word from the given 32-bit value mouseDelta = (short)((mouseHookStruct.MouseData >> 16) & 0xffff); //TODO: X BUTTONS (I havent them so was unable to test) //If the message is WM_XBUTTONDOWN, WM_XBUTTONUP, WM_XBUTTONDBLCLK, WM_NCXBUTTONDOWN, WM_NCXBUTTONUP, //or WM_NCXBUTTONDBLCLK, the high-order word specifies which X button was pressed or released, //and the low-order word is reserved. This value can be one or more of the following values. //Otherwise, MouseData is not used. break; } //generate event MouseEventExtArgs e = new MouseEventExtArgs( button, clickCount, mouseHookStruct.Point.X, mouseHookStruct.Point.Y, mouseDelta); //Mouse up if (HookManager.s_MouseUp != null && mouseUp) { HookManager.s_MouseUp.Invoke(null, e); } //Mouse down if (HookManager.s_MouseDown != null && mouseDown) { HookManager.s_MouseDown.Invoke(null, e); } //If someone listens to click and a click is heppened if (HookManager.s_MouseClick != null && clickCount > 0) { HookManager.s_MouseClick.Invoke(null, e); } //If someone listens to click and a click is heppened if (HookManager.s_MouseClickExt != null && clickCount > 0) { HookManager.s_MouseClickExt.Invoke(null, e); } //If someone listens to double click and a click is heppened if (HookManager.s_MouseDoubleClick != null && clickCount == 2) { HookManager.s_MouseDoubleClick.Invoke(null, e); } //Wheel was moved if (HookManager.s_MouseWheel != null && mouseDelta != 0) { HookManager.s_MouseWheel.Invoke(null, e); } //If someone listens to move and there was a change in coordinates raise move event if ((HookManager.s_MouseMove != null || HookManager.s_MouseMoveExt != null) && (m_OldX != mouseHookStruct.Point.X || m_OldY != mouseHookStruct.Point.Y)) { m_OldX = mouseHookStruct.Point.X; m_OldY = mouseHookStruct.Point.Y; if (HookManager.s_MouseMove != null) { HookManager.s_MouseMove.Invoke(null, e); } if (HookManager.s_MouseMoveExt != null) { HookManager.s_MouseMoveExt.Invoke(null, e); } } if (e.Handled) { return(-1); } } //call next hook return(HookManager.CallNextHookEx(s_MouseHookHandle, nCode, wParam, lParam)); }
/// <summary> /// A callback function which will be called every Time a keyboard activity detected. /// </summary> /// <param name="nCode"> /// [in] Specifies whether the hook procedure must process the message. /// If nCode is HC_ACTION, the hook procedure must process the message. /// If nCode is less than zero, the hook procedure must pass the message to the /// CallNextHookEx function without further processing and must return the /// value returned by CallNextHookEx. /// </param> /// <param name="wParam"> /// [in] Specifies whether the message was sent by the current thread. /// If the message was sent by the current thread, it is nonzero; otherwise, it is zero. /// </param> /// <param name="lParam"> /// [in] Pointer to a CWPSTRUCT structure that contains details about the message. /// </param> /// <returns> /// If nCode is less than zero, the hook procedure must return the value returned by CallNextHookEx. /// If nCode is greater than or equal to zero, it is highly recommended that you call CallNextHookEx /// and return the value it returns; otherwise, other applications that have installed WH_CALLWNDPROC /// hooks will not receive hook notifications and may behave incorrectly as a result. If the hook /// procedure does not call CallNextHookEx, the return value should be zero. /// </returns> private static int KeyboardHookProc(int nCode, Int32 wParam, IntPtr lParam) { //indicates if any of underlaing events set e.Handled flag bool handled = false; if (nCode >= 0) { //read structure KeyboardHookStruct at lParam HookManager.KeyboardHookStruct MyKeyboardHookStruct = (HookManager.KeyboardHookStruct)Marshal.PtrToStructure(lParam, typeof(HookManager.KeyboardHookStruct)); //raise KeyDown if (HookManager.s_KeyDown != null && (wParam == HookManager.WM_KEYDOWN || wParam == HookManager.WM_SYSKEYDOWN)) { Keys keyData = (Keys)MyKeyboardHookStruct.VirtualKeyCode; KeyEventArgs e = new KeyEventArgs(keyData); HookManager.s_KeyDown.Invoke(null, e); handled = e.Handled; } // raise KeyPress if (HookManager.s_KeyPress != null && wParam == HookManager.WM_KEYDOWN) { bool isDownShift = ((HookManager.GetKeyState(HookManager.VK_SHIFT) & 0x80) == 0x80 ? true : false); bool isDownCapslock = (HookManager.GetKeyState(HookManager.VK_CAPITAL) != 0 ? true : false); byte[] keyState = new byte[256]; HookManager.GetKeyboardState(keyState); byte[] inBuffer = new byte[2]; if (HookManager.ToAscii(MyKeyboardHookStruct.VirtualKeyCode, MyKeyboardHookStruct.ScanCode, keyState, inBuffer, MyKeyboardHookStruct.Flags) == 1) { char key = (char)inBuffer[0]; if ((isDownCapslock ^ isDownShift) && Char.IsLetter(key)) { key = Char.ToUpper(key); } KeyPressEventArgs e = new KeyPressEventArgs(key); HookManager.s_KeyPress.Invoke(null, e); handled = handled || e.Handled; } } // raise KeyUp if (HookManager.s_KeyUp != null && (wParam == HookManager.WM_KEYUP || wParam == HookManager.WM_SYSKEYUP)) { Keys keyData = (Keys)MyKeyboardHookStruct.VirtualKeyCode; KeyEventArgs e = new KeyEventArgs(keyData); HookManager.s_KeyUp.Invoke(null, e); handled = handled || e.Handled; } } //if event handled in application do not handoff to other listeners if (handled) { return(-1); } //forward to other application return(HookManager.CallNextHookEx(s_KeyboardHookHandle, nCode, wParam, lParam)); }