/* * This work is licensed under the terms of the GNU GPL, version 2 or * (at your option) any later version. See the COPYING file in the * top-level directory. * * The win32 keyboard hooking code was imported from project spice-gtk. */ #include "qemu/osdep.h" #include "sysemu/sysemu.h" #include "ui/win32-kbd-hook.h" static Notifier win32_unhook_notifier; static HHOOK win32_keyboard_hook; static HWND win32_window; static DWORD win32_grab; static LRESULT CALLBACK keyboard_hook_cb(int code, WPARAM wparam, LPARAM lparam) { if (win32_window && code == HC_ACTION && win32_window == GetFocus()) { KBDLLHOOKSTRUCT *hooked = (KBDLLHOOKSTRUCT *)lparam; if (wparam != WM_KEYUP) { DWORD dwmsg = (hooked->flags << 24) | ((hooked->scanCode & 0xff) << 16) | 1; switch (hooked->vkCode) { case VK_CAPITAL: /* fall through */ case VK_SCROLL: /* fall through */ case VK_NUMLOCK: /* fall through */ case VK_LSHIFT: /* fall through */ case VK_RSHIFT: /* fall through */ case VK_RCONTROL: /* fall through */ case VK_LMENU: /* fall through */ case VK_RMENU: break; case VK_LCONTROL: /* * When pressing AltGr, an extra VK_LCONTROL with a special * scancode with bit 9 set is sent. Let's ignore the extra * VK_LCONTROL, as that will make AltGr misbehave. */ if (hooked->scanCode & 0x200) { return 1; } break; default: if (win32_grab) { SendMessage(win32_window, wparam, hooked->vkCode, dwmsg); return 1; } break; } } else { switch (hooked->vkCode) { case VK_LCONTROL: if (hooked->scanCode & 0x200) { return 1; } break; } } } return CallNextHookEx(NULL, code, wparam, lparam); } static void keyboard_hook_unhook(Notifier *n, void *data) { UnhookWindowsHookEx(win32_keyboard_hook); win32_keyboard_hook = NULL; } void win32_kbd_set_window(void *hwnd) { if (hwnd && !win32_keyboard_hook) { /* note: the installing thread must have a message loop */ win32_keyboard_hook = SetWindowsHookEx(WH_KEYBOARD_LL, keyboard_hook_cb, GetModuleHandle(NULL), 0); if (win32_keyboard_hook) { win32_unhook_notifier.notify = keyboard_hook_unhook; qemu_add_exit_notifier(&win32_unhook_notifier); } } win32_window = hwnd; } void win32_kbd_set_grab(bool grab) { win32_grab = grab; }