/* * Keyboard related functions * * Copyright 1993 Bob Amstadt * Copyright 1996 Albrecht Kleine * Copyright 1997 David Faure * Copyright 1998 Morten Welinder * Copyright 1998 Ulrich Weigand * Copyright 1999 Ove Kåven * Copyright 2011, 2012, 2013 Ken Thomases for CodeWeavers Inc. * Copyright 2013 Alexandre Julliard * Copyright 2015 Josh DuBois for CodeWeavers Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #define NONAMELESSUNION #define NONAMELESSSTRUCT #include "config.h" #include "android.h" #include "wine/unicode.h" #include "wine/server.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(keyboard); WINE_DECLARE_DEBUG_CHANNEL(key); static const UINT keycode_to_vkey[] = { 0, /* AKEYCODE_UNKNOWN */ 0, /* AKEYCODE_SOFT_LEFT */ 0, /* AKEYCODE_SOFT_RIGHT */ 0, /* AKEYCODE_HOME */ 0, /* AKEYCODE_BACK */ 0, /* AKEYCODE_CALL */ 0, /* AKEYCODE_ENDCALL */ '0', /* AKEYCODE_0 */ '1', /* AKEYCODE_1 */ '2', /* AKEYCODE_2 */ '3', /* AKEYCODE_3 */ '4', /* AKEYCODE_4 */ '5', /* AKEYCODE_5 */ '6', /* AKEYCODE_6 */ '7', /* AKEYCODE_7 */ '8', /* AKEYCODE_8 */ '9', /* AKEYCODE_9 */ 0, /* AKEYCODE_STAR */ 0, /* AKEYCODE_POUND */ VK_UP, /* AKEYCODE_DPAD_UP */ VK_DOWN, /* AKEYCODE_DPAD_DOWN */ VK_LEFT, /* AKEYCODE_DPAD_LEFT */ VK_RIGHT, /* AKEYCODE_DPAD_RIGHT */ 0, /* AKEYCODE_DPAD_CENTER */ 0, /* AKEYCODE_VOLUME_UP */ 0, /* AKEYCODE_VOLUME_DOWN */ 0, /* AKEYCODE_POWER */ 0, /* AKEYCODE_CAMERA */ 0, /* AKEYCODE_CLEAR */ 'A', /* AKEYCODE_A */ 'B', /* AKEYCODE_B */ 'C', /* AKEYCODE_C */ 'D', /* AKEYCODE_D */ 'E', /* AKEYCODE_E */ 'F', /* AKEYCODE_F */ 'G', /* AKEYCODE_G */ 'H', /* AKEYCODE_H */ 'I', /* AKEYCODE_I */ 'J', /* AKEYCODE_J */ 'K', /* AKEYCODE_K */ 'L', /* AKEYCODE_L */ 'M', /* AKEYCODE_M */ 'N', /* AKEYCODE_N */ 'O', /* AKEYCODE_O */ 'P', /* AKEYCODE_P */ 'Q', /* AKEYCODE_Q */ 'R', /* AKEYCODE_R */ 'S', /* AKEYCODE_S */ 'T', /* AKEYCODE_T */ 'U', /* AKEYCODE_U */ 'V', /* AKEYCODE_V */ 'W', /* AKEYCODE_W */ 'X', /* AKEYCODE_X */ 'Y', /* AKEYCODE_Y */ 'Z', /* AKEYCODE_Z */ VK_OEM_COMMA, /* AKEYCODE_COMMA */ VK_OEM_PERIOD, /* AKEYCODE_PERIOD */ VK_LMENU, /* AKEYCODE_ALT_LEFT */ VK_RMENU, /* AKEYCODE_ALT_RIGHT */ VK_LSHIFT, /* AKEYCODE_SHIFT_LEFT */ VK_RSHIFT, /* AKEYCODE_SHIFT_RIGHT */ VK_TAB, /* AKEYCODE_TAB */ VK_SPACE, /* AKEYCODE_SPACE */ 0, /* AKEYCODE_SYM */ 0, /* AKEYCODE_EXPLORER */ 0, /* AKEYCODE_ENVELOPE */ VK_RETURN, /* AKEYCODE_ENTER */ VK_BACK, /* AKEYCODE_DEL */ VK_OEM_3, /* AKEYCODE_GRAVE */ VK_OEM_MINUS, /* AKEYCODE_MINUS */ VK_OEM_PLUS, /* AKEYCODE_EQUALS */ VK_OEM_4, /* AKEYCODE_LEFT_BRACKET */ VK_OEM_6, /* AKEYCODE_RIGHT_BRACKET */ VK_OEM_5, /* AKEYCODE_BACKSLASH */ VK_OEM_1, /* AKEYCODE_SEMICOLON */ VK_OEM_7, /* AKEYCODE_APOSTROPHE */ VK_OEM_2, /* AKEYCODE_SLASH */ 0, /* AKEYCODE_AT */ 0, /* AKEYCODE_NUM */ 0, /* AKEYCODE_HEADSETHOOK */ 0, /* AKEYCODE_FOCUS */ 0, /* AKEYCODE_PLUS */ 0, /* AKEYCODE_MENU */ 0, /* AKEYCODE_NOTIFICATION */ 0, /* AKEYCODE_SEARCH */ VK_MEDIA_PLAY_PAUSE, /* AKEYCODE_MEDIA_PLAY_PAUSE */ VK_MEDIA_STOP, /* AKEYCODE_MEDIA_STOP */ VK_MEDIA_NEXT_TRACK, /* AKEYCODE_MEDIA_NEXT */ VK_MEDIA_PREV_TRACK, /* AKEYCODE_MEDIA_PREVIOUS */ 0, /* AKEYCODE_MEDIA_REWIND */ 0, /* AKEYCODE_MEDIA_FAST_FORWARD */ 0, /* AKEYCODE_MUTE */ VK_PRIOR, /* AKEYCODE_PAGE_UP */ VK_NEXT, /* AKEYCODE_PAGE_DOWN */ 0, /* AKEYCODE_PICTSYMBOLS */ 0, /* AKEYCODE_SWITCH_CHARSET */ 0, /* AKEYCODE_BUTTON_A */ 0, /* AKEYCODE_BUTTON_B */ 0, /* AKEYCODE_BUTTON_C */ 0, /* AKEYCODE_BUTTON_X */ 0, /* AKEYCODE_BUTTON_Y */ 0, /* AKEYCODE_BUTTON_Z */ 0, /* AKEYCODE_BUTTON_L1 */ 0, /* AKEYCODE_BUTTON_R1 */ 0, /* AKEYCODE_BUTTON_L2 */ 0, /* AKEYCODE_BUTTON_R2 */ 0, /* AKEYCODE_BUTTON_THUMBL */ 0, /* AKEYCODE_BUTTON_THUMBR */ 0, /* AKEYCODE_BUTTON_START */ 0, /* AKEYCODE_BUTTON_SELECT */ 0, /* AKEYCODE_BUTTON_MODE */ VK_ESCAPE, /* AKEYCODE_ESCAPE */ VK_DELETE, /* AKEYCODE_FORWARD_DEL */ VK_LCONTROL, /* AKEYCODE_CTRL_LEFT */ VK_RCONTROL, /* AKEYCODE_CTRL_RIGHT */ VK_CAPITAL, /* AKEYCODE_CAPS_LOCK */ VK_SCROLL, /* AKEYCODE_SCROLL_LOCK */ VK_LWIN, /* AKEYCODE_META_LEFT */ VK_RWIN, /* AKEYCODE_META_RIGHT */ 0, /* AKEYCODE_FUNCTION */ 0, /* AKEYCODE_SYSRQ */ 0, /* AKEYCODE_BREAK */ VK_HOME, /* AKEYCODE_MOVE_HOME */ VK_END, /* AKEYCODE_MOVE_END */ VK_INSERT, /* AKEYCODE_INSERT */ 0, /* AKEYCODE_FORWARD */ 0, /* AKEYCODE_MEDIA_PLAY */ 0, /* AKEYCODE_MEDIA_PAUSE */ 0, /* AKEYCODE_MEDIA_CLOSE */ 0, /* AKEYCODE_MEDIA_EJECT */ 0, /* AKEYCODE_MEDIA_RECORD */ VK_F1, /* AKEYCODE_F1 */ VK_F2, /* AKEYCODE_F2 */ VK_F3, /* AKEYCODE_F3 */ VK_F4, /* AKEYCODE_F4 */ VK_F5, /* AKEYCODE_F5 */ VK_F6, /* AKEYCODE_F6 */ VK_F7, /* AKEYCODE_F7 */ VK_F8, /* AKEYCODE_F8 */ VK_F9, /* AKEYCODE_F9 */ VK_F10, /* AKEYCODE_F10 */ VK_F11, /* AKEYCODE_F11 */ VK_F12, /* AKEYCODE_F12 */ VK_NUMLOCK, /* AKEYCODE_NUM_LOCK */ VK_NUMPAD0, /* AKEYCODE_NUMPAD_0 */ VK_NUMPAD1, /* AKEYCODE_NUMPAD_1 */ VK_NUMPAD2, /* AKEYCODE_NUMPAD_2 */ VK_NUMPAD3, /* AKEYCODE_NUMPAD_3 */ VK_NUMPAD4, /* AKEYCODE_NUMPAD_4 */ VK_NUMPAD5, /* AKEYCODE_NUMPAD_5 */ VK_NUMPAD6, /* AKEYCODE_NUMPAD_6 */ VK_NUMPAD7, /* AKEYCODE_NUMPAD_7 */ VK_NUMPAD8, /* AKEYCODE_NUMPAD_8 */ VK_NUMPAD9, /* AKEYCODE_NUMPAD_9 */ VK_DIVIDE, /* AKEYCODE_NUMPAD_DIVIDE */ VK_MULTIPLY, /* AKEYCODE_NUMPAD_MULTIPLY */ VK_SUBTRACT, /* AKEYCODE_NUMPAD_SUBTRACT */ VK_ADD, /* AKEYCODE_NUMPAD_ADD */ VK_DECIMAL, /* AKEYCODE_NUMPAD_DOT */ 0, /* AKEYCODE_NUMPAD_COMMA */ 0, /* AKEYCODE_NUMPAD_ENTER */ 0, /* AKEYCODE_NUMPAD_EQUALS */ 0, /* AKEYCODE_NUMPAD_LEFT_PAREN */ 0, /* AKEYCODE_NUMPAD_RIGHT_PAREN */ 0, /* AKEYCODE_VOLUME_MUTE */ 0, /* AKEYCODE_INFO */ 0, /* AKEYCODE_CHANNEL_UP */ 0, /* AKEYCODE_CHANNEL_DOWN */ 0, /* AKEYCODE_ZOOM_IN */ 0, /* AKEYCODE_ZOOM_OUT */ 0, /* AKEYCODE_TV */ 0, /* AKEYCODE_WINDOW */ 0, /* AKEYCODE_GUIDE */ 0, /* AKEYCODE_DVR */ 0, /* AKEYCODE_BOOKMARK */ 0, /* AKEYCODE_CAPTIONS */ 0, /* AKEYCODE_SETTINGS */ 0, /* AKEYCODE_TV_POWER */ 0, /* AKEYCODE_TV_INPUT */ 0, /* AKEYCODE_STB_POWER */ 0, /* AKEYCODE_STB_INPUT */ 0, /* AKEYCODE_AVR_POWER */ 0, /* AKEYCODE_AVR_INPUT */ 0, /* AKEYCODE_PROG_RED */ 0, /* AKEYCODE_PROG_GREEN */ 0, /* AKEYCODE_PROG_YELLOW */ 0, /* AKEYCODE_PROG_BLUE */ 0, /* AKEYCODE_APP_SWITCH */ 0, /* AKEYCODE_BUTTON_1 */ 0, /* AKEYCODE_BUTTON_2 */ 0, /* AKEYCODE_BUTTON_3 */ 0, /* AKEYCODE_BUTTON_4 */ 0, /* AKEYCODE_BUTTON_5 */ 0, /* AKEYCODE_BUTTON_6 */ 0, /* AKEYCODE_BUTTON_7 */ 0, /* AKEYCODE_BUTTON_8 */ 0, /* AKEYCODE_BUTTON_9 */ 0, /* AKEYCODE_BUTTON_10 */ 0, /* AKEYCODE_BUTTON_11 */ 0, /* AKEYCODE_BUTTON_12 */ 0, /* AKEYCODE_BUTTON_13 */ 0, /* AKEYCODE_BUTTON_14 */ 0, /* AKEYCODE_BUTTON_15 */ 0, /* AKEYCODE_BUTTON_16 */ 0, /* AKEYCODE_LANGUAGE_SWITCH */ 0, /* AKEYCODE_MANNER_MODE */ 0, /* AKEYCODE_3D_MODE */ 0, /* AKEYCODE_CONTACTS */ 0, /* AKEYCODE_CALENDAR */ 0, /* AKEYCODE_MUSIC */ 0, /* AKEYCODE_CALCULATOR */ 0, /* AKEYCODE_ZENKAKU_HANKAKU */ 0, /* AKEYCODE_EISU */ 0, /* AKEYCODE_MUHENKAN */ 0, /* AKEYCODE_HENKAN */ 0, /* AKEYCODE_KATAKANA_HIRAGANA */ 0, /* AKEYCODE_YEN */ 0, /* AKEYCODE_RO */ VK_KANA, /* AKEYCODE_KANA */ 0, /* AKEYCODE_ASSIST */ }; static const WORD vkey_to_scancode[] = { 0, /* 0x00 undefined */ 0, /* VK_LBUTTON */ 0, /* VK_RBUTTON */ 0, /* VK_CANCEL */ 0, /* VK_MBUTTON */ 0, /* VK_XBUTTON1 */ 0, /* VK_XBUTTON2 */ 0, /* 0x07 undefined */ 0x0e, /* VK_BACK */ 0x0f, /* VK_TAB */ 0, /* 0x0a undefined */ 0, /* 0x0b undefined */ 0, /* VK_CLEAR */ 0x1c, /* VK_RETURN */ 0, /* 0x0e undefined */ 0, /* 0x0f undefined */ 0x2a, /* VK_SHIFT */ 0x1d, /* VK_CONTROL */ 0x38, /* VK_MENU */ 0, /* VK_PAUSE */ 0x3a, /* VK_CAPITAL */ 0, /* VK_KANA */ 0, /* 0x16 undefined */ 0, /* VK_JUNJA */ 0, /* VK_FINAL */ 0, /* VK_HANJA */ 0, /* 0x1a undefined */ 0x01, /* VK_ESCAPE */ 0, /* VK_CONVERT */ 0, /* VK_NONCONVERT */ 0, /* VK_ACCEPT */ 0, /* VK_MODECHANGE */ 0x39, /* VK_SPACE */ 0x149, /* VK_PRIOR */ 0x151, /* VK_NEXT */ 0x14f, /* VK_END */ 0x147, /* VK_HOME */ 0x14b, /* VK_LEFT */ 0x148, /* VK_UP */ 0x14d, /* VK_RIGHT */ 0x150, /* VK_DOWN */ 0, /* VK_SELECT */ 0, /* VK_PRINT */ 0, /* VK_EXECUTE */ 0, /* VK_SNAPSHOT */ 0x152, /* VK_INSERT */ 0x153, /* VK_DELETE */ 0, /* VK_HELP */ 0x0b, /* VK_0 */ 0x02, /* VK_1 */ 0x03, /* VK_2 */ 0x04, /* VK_3 */ 0x05, /* VK_4 */ 0x06, /* VK_5 */ 0x07, /* VK_6 */ 0x08, /* VK_7 */ 0x09, /* VK_8 */ 0x0a, /* VK_9 */ 0, /* 0x3a undefined */ 0, /* 0x3b undefined */ 0, /* 0x3c undefined */ 0, /* 0x3d undefined */ 0, /* 0x3e undefined */ 0, /* 0x3f undefined */ 0, /* 0x40 undefined */ 0x1e, /* VK_A */ 0x30, /* VK_B */ 0x2e, /* VK_C */ 0x20, /* VK_D */ 0x12, /* VK_E */ 0x21, /* VK_F */ 0x22, /* VK_G */ 0x23, /* VK_H */ 0x17, /* VK_I */ 0x24, /* VK_J */ 0x25, /* VK_K */ 0x26, /* VK_L */ 0x32, /* VK_M */ 0x31, /* VK_N */ 0x18, /* VK_O */ 0x19, /* VK_P */ 0x10, /* VK_Q */ 0x13, /* VK_R */ 0x1f, /* VK_S */ 0x14, /* VK_T */ 0x16, /* VK_U */ 0x2f, /* VK_V */ 0x11, /* VK_W */ 0x2d, /* VK_X */ 0x15, /* VK_Y */ 0x2c, /* VK_Z */ 0, /* VK_LWIN */ 0, /* VK_RWIN */ 0, /* VK_APPS */ 0, /* 0x5e undefined */ 0, /* VK_SLEEP */ 0x52, /* VK_NUMPAD0 */ 0x4f, /* VK_NUMPAD1 */ 0x50, /* VK_NUMPAD2 */ 0x51, /* VK_NUMPAD3 */ 0x4b, /* VK_NUMPAD4 */ 0x4c, /* VK_NUMPAD5 */ 0x4d, /* VK_NUMPAD6 */ 0x47, /* VK_NUMPAD7 */ 0x48, /* VK_NUMPAD8 */ 0x49, /* VK_NUMPAD9 */ 0x37, /* VK_MULTIPLY */ 0x4e, /* VK_ADD */ 0x7e, /* VK_SEPARATOR */ 0x4a, /* VK_SUBTRACT */ 0x53, /* VK_DECIMAL */ 0135, /* VK_DIVIDE */ 0x3b, /* VK_F1 */ 0x3c, /* VK_F2 */ 0x3d, /* VK_F3 */ 0x3e, /* VK_F4 */ 0x3f, /* VK_F5 */ 0x40, /* VK_F6 */ 0x41, /* VK_F7 */ 0x42, /* VK_F8 */ 0x43, /* VK_F9 */ 0x44, /* VK_F10 */ 0x57, /* VK_F11 */ 0x58, /* VK_F12 */ 0x64, /* VK_F13 */ 0x65, /* VK_F14 */ 0x66, /* VK_F15 */ 0x67, /* VK_F16 */ 0x68, /* VK_F17 */ 0x69, /* VK_F18 */ 0x6a, /* VK_F19 */ 0x6b, /* VK_F20 */ 0, /* VK_F21 */ 0, /* VK_F22 */ 0, /* VK_F23 */ 0, /* VK_F24 */ 0, /* 0x88 undefined */ 0, /* 0x89 undefined */ 0, /* 0x8a undefined */ 0, /* 0x8b undefined */ 0, /* 0x8c undefined */ 0, /* 0x8d undefined */ 0, /* 0x8e undefined */ 0, /* 0x8f undefined */ 0, /* VK_NUMLOCK */ 0, /* VK_SCROLL */ 0x10d, /* VK_OEM_NEC_EQUAL */ 0, /* VK_OEM_FJ_JISHO */ 0, /* VK_OEM_FJ_MASSHOU */ 0, /* VK_OEM_FJ_TOUROKU */ 0, /* VK_OEM_FJ_LOYA */ 0, /* VK_OEM_FJ_ROYA */ 0, /* 0x97 undefined */ 0, /* 0x98 undefined */ 0, /* 0x99 undefined */ 0, /* 0x9a undefined */ 0, /* 0x9b undefined */ 0, /* 0x9c undefined */ 0, /* 0x9d undefined */ 0, /* 0x9e undefined */ 0, /* 0x9f undefined */ 0x2a, /* VK_LSHIFT */ 0x36, /* VK_RSHIFT */ 0x1d, /* VK_LCONTROL */ 0x11d, /* VK_RCONTROL */ 0x38, /* VK_LMENU */ 0x138, /* VK_RMENU */ 0, /* VK_BROWSER_BACK */ 0, /* VK_BROWSER_FORWARD */ 0, /* VK_BROWSER_REFRESH */ 0, /* VK_BROWSER_STOP */ 0, /* VK_BROWSER_SEARCH */ 0, /* VK_BROWSER_FAVORITES */ 0, /* VK_BROWSER_HOME */ 0x100, /* VK_VOLUME_MUTE */ 0x100, /* VK_VOLUME_DOWN */ 0x100, /* VK_VOLUME_UP */ 0, /* VK_MEDIA_NEXT_TRACK */ 0, /* VK_MEDIA_PREV_TRACK */ 0, /* VK_MEDIA_STOP */ 0, /* VK_MEDIA_PLAY_PAUSE */ 0, /* VK_LAUNCH_MAIL */ 0, /* VK_LAUNCH_MEDIA_SELECT */ 0, /* VK_LAUNCH_APP1 */ 0, /* VK_LAUNCH_APP2 */ 0, /* 0xb8 undefined */ 0, /* 0xb9 undefined */ 0x27, /* VK_OEM_1 */ 0x0d, /* VK_OEM_PLUS */ 0x33, /* VK_OEM_COMMA */ 0x0c, /* VK_OEM_MINUS */ 0x34, /* VK_OEM_PERIOD */ 0x35, /* VK_OEM_2 */ 0x29, /* VK_OEM_3 */ 0, /* 0xc1 undefined */ 0, /* 0xc2 undefined */ 0, /* 0xc3 undefined */ 0, /* 0xc4 undefined */ 0, /* 0xc5 undefined */ 0, /* 0xc6 undefined */ 0, /* 0xc7 undefined */ 0, /* 0xc8 undefined */ 0, /* 0xc9 undefined */ 0, /* 0xca undefined */ 0, /* 0xcb undefined */ 0, /* 0xcc undefined */ 0, /* 0xcd undefined */ 0, /* 0xce undefined */ 0, /* 0xcf undefined */ 0, /* 0xd0 undefined */ 0, /* 0xd1 undefined */ 0, /* 0xd2 undefined */ 0, /* 0xd3 undefined */ 0, /* 0xd4 undefined */ 0, /* 0xd5 undefined */ 0, /* 0xd6 undefined */ 0, /* 0xd7 undefined */ 0, /* 0xd8 undefined */ 0, /* 0xd9 undefined */ 0, /* 0xda undefined */ 0x1a, /* VK_OEM_4 */ 0x2b, /* VK_OEM_5 */ 0x1b, /* VK_OEM_6 */ 0x28, /* VK_OEM_7 */ 0, /* VK_OEM_8 */ 0, /* 0xe0 undefined */ 0, /* VK_OEM_AX */ 0x56, /* VK_OEM_102 */ 0, /* VK_ICO_HELP */ 0, /* VK_ICO_00 */ 0, /* VK_PROCESSKEY */ 0, /* VK_ICO_CLEAR */ 0, /* VK_PACKET */ 0, /* 0xe8 undefined */ 0x71, /* VK_OEM_RESET */ 0, /* VK_OEM_JUMP */ 0, /* VK_OEM_PA1 */ 0, /* VK_OEM_PA2 */ 0, /* VK_OEM_PA3 */ 0, /* VK_OEM_WSCTRL */ 0, /* VK_OEM_CUSEL */ 0, /* VK_OEM_ATTN */ 0, /* VK_OEM_FINISH */ 0, /* VK_OEM_COPY */ 0, /* VK_OEM_AUTO */ 0, /* VK_OEM_ENLW */ 0, /* VK_OEM_BACKTAB */ 0, /* VK_ATTN */ 0, /* VK_CRSEL */ 0, /* VK_EXSEL */ 0, /* VK_EREOF */ 0, /* VK_PLAY */ 0, /* VK_ZOOM */ 0, /* VK_NONAME */ 0, /* VK_PA1 */ 0x59, /* VK_OEM_CLEAR */ 0, /* 0xff undefined */ }; static const struct { DWORD vkey; const char *name; } vkey_names[] = { { VK_ADD, "Num +" }, { VK_BACK, "Backspace" }, { VK_CAPITAL, "Caps Lock" }, { VK_CONTROL, "Ctrl" }, { VK_DECIMAL, "Num Del" }, { VK_DELETE | 0x100, "Delete" }, { VK_DIVIDE | 0x100, "Num /" }, { VK_DOWN | 0x100, "Down" }, { VK_END | 0x100, "End" }, { VK_ESCAPE, "Esc" }, { VK_F1, "F1" }, { VK_F2, "F2" }, { VK_F3, "F3" }, { VK_F4, "F4" }, { VK_F5, "F5" }, { VK_F6, "F6" }, { VK_F7, "F7" }, { VK_F8, "F8" }, { VK_F9, "F9" }, { VK_F10, "F10" }, { VK_F11, "F11" }, { VK_F12, "F12" }, { VK_F13, "F13" }, { VK_F14, "F14" }, { VK_F15, "F15" }, { VK_F16, "F16" }, { VK_F17, "F17" }, { VK_F18, "F18" }, { VK_F19, "F19" }, { VK_F20, "F20" }, { VK_F21, "F21" }, { VK_F22, "F22" }, { VK_F23, "F23" }, { VK_F24, "F24" }, { VK_HELP | 0x100, "Help" }, { VK_HOME | 0x100, "Home" }, { VK_INSERT | 0x100, "Insert" }, { VK_LCONTROL, "Ctrl" }, { VK_LEFT | 0x100, "Left" }, { VK_LMENU, "Alt" }, { VK_LSHIFT, "Shift" }, { VK_LWIN | 0x100, "Win" }, { VK_MENU, "Alt" }, { VK_MULTIPLY, "Num *" }, { VK_NEXT | 0x100, "Page Down" }, { VK_NUMLOCK | 0x100, "Num Lock" }, { VK_NUMPAD0, "Num 0" }, { VK_NUMPAD1, "Num 1" }, { VK_NUMPAD2, "Num 2" }, { VK_NUMPAD3, "Num 3" }, { VK_NUMPAD4, "Num 4" }, { VK_NUMPAD5, "Num 5" }, { VK_NUMPAD6, "Num 6" }, { VK_NUMPAD7, "Num 7" }, { VK_NUMPAD8, "Num 8" }, { VK_NUMPAD9, "Num 9" }, { VK_OEM_CLEAR, "Num Clear" }, { VK_OEM_NEC_EQUAL | 0x100, "Num =" }, { VK_PRIOR | 0x100, "Page Up" }, { VK_RCONTROL | 0x100, "Right Ctrl" }, { VK_RETURN, "Return" }, { VK_RETURN | 0x100, "Num Enter" }, { VK_RIGHT | 0x100, "Right" }, { VK_RMENU | 0x100, "Right Alt" }, { VK_RSHIFT, "Right Shift" }, { VK_RWIN | 0x100, "Right Win" }, { VK_SEPARATOR, "Num ," }, { VK_SHIFT, "Shift" }, { VK_SPACE, "Space" }, { VK_SUBTRACT, "Num -" }, { VK_TAB, "Tab" }, { VK_UP | 0x100, "Up" }, { VK_VOLUME_DOWN | 0x100, "Volume Down" }, { VK_VOLUME_MUTE | 0x100, "Mute" }, { VK_VOLUME_UP | 0x100, "Volume Up" }, { VK_OEM_MINUS, "-" }, { VK_OEM_PLUS, "=" }, { VK_OEM_1, ";" }, { VK_OEM_2, "/" }, { VK_OEM_3, "`" }, { VK_OEM_4, "[" }, { VK_OEM_5, "\\" }, { VK_OEM_6, "]" }, { VK_OEM_7, "'" }, { VK_OEM_COMMA, "," }, { VK_OEM_PERIOD, "." }, }; static const SHORT char_vkey_map[] = { 0x332, 0x241, 0x242, 0x003, 0x244, 0x245, 0x246, 0x247, 0x008, 0x009, 0x20d, 0x24b, 0x24c, 0x00d, 0x24e, 0x24f, 0x250, 0x251, 0x252, 0x253, 0x254, 0x255, 0x256, 0x257, 0x258, 0x259, 0x25a, 0x01b, 0x2dc, 0x2dd, 0x336, 0x3bd, 0x020, 0x131, 0x1de, 0x133, 0x134, 0x135, 0x137, 0x0de, 0x139, 0x130, 0x138, 0x1bb, 0x0bc, 0x0bd, 0x0be, 0x0bf, 0x030, 0x031, 0x032, 0x033, 0x034, 0x035, 0x036, 0x037, 0x038, 0x039, 0x1ba, 0x0ba, 0x1bc, 0x0bb, 0x1be, 0x1bf, 0x132, 0x141, 0x142, 0x143, 0x144, 0x145, 0x146, 0x147, 0x148, 0x149, 0x14a, 0x14b, 0x14c, 0x14d, 0x14e, 0x14f, 0x150, 0x151, 0x152, 0x153, 0x154, 0x155, 0x156, 0x157, 0x158, 0x159, 0x15a, 0x0db, 0x0dc, 0x0dd, 0x136, 0x1bd, 0x0c0, 0x041, 0x042, 0x043, 0x044, 0x045, 0x046, 0x047, 0x048, 0x049, 0x04a, 0x04b, 0x04c, 0x04d, 0x04e, 0x04f, 0x050, 0x051, 0x052, 0x053, 0x054, 0x055, 0x056, 0x057, 0x058, 0x059, 0x05a, 0x1db, 0x1dc, 0x1dd, 0x1c0, 0x208 }; static UINT scancode_to_vkey( UINT scan ) { UINT j; for (j = 0; j < ARRAY_SIZE( vkey_to_scancode ); j++) if (vkey_to_scancode[j] == scan) return j; return 0; } static const char* vkey_to_name( UINT vkey ) { UINT j; for (j = 0; j < ARRAY_SIZE( vkey_names ); j++) if (vkey_names[j].vkey == vkey) return vkey_names[j].name; return NULL; } static BOOL get_async_key_state( BYTE state[256] ) { BOOL ret; SERVER_START_REQ( get_key_state ) { req->tid = 0; req->key = -1; wine_server_set_reply( req, state, 256 ); ret = !wine_server_call( req ); } SERVER_END_REQ; return ret; } static void send_keyboard_input( HWND hwnd, WORD vkey, WORD scan, DWORD flags ) { INPUT input; input.type = INPUT_KEYBOARD; input.u.ki.wVk = vkey; input.u.ki.wScan = scan; input.u.ki.dwFlags = flags; input.u.ki.time = 0; input.u.ki.dwExtraInfo = 0; __wine_send_input( hwnd, &input ); } /*********************************************************************** * update_keyboard_lock_state */ void update_keyboard_lock_state( WORD vkey, UINT state ) { BYTE keystate[256]; if (!get_async_key_state( keystate )) return; if (!(keystate[VK_CAPITAL] & 0x01) != !(state & AMETA_CAPS_LOCK_ON) && vkey != VK_CAPITAL) { TRACE( "adjusting CapsLock state (%02x)\n", keystate[VK_CAPITAL] ); send_keyboard_input( 0, VK_CAPITAL, 0x3a, 0 ); send_keyboard_input( 0, VK_CAPITAL, 0x3a, KEYEVENTF_KEYUP ); } if (!(keystate[VK_NUMLOCK] & 0x01) != !(state & AMETA_NUM_LOCK_ON) && (vkey & 0xff) != VK_NUMLOCK) { TRACE( "adjusting NumLock state (%02x)\n", keystate[VK_NUMLOCK] ); send_keyboard_input( 0, VK_NUMLOCK, 0x45, KEYEVENTF_EXTENDEDKEY ); send_keyboard_input( 0, VK_NUMLOCK, 0x45, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP ); } if (!(keystate[VK_SCROLL] & 0x01) != !(state & AMETA_SCROLL_LOCK_ON) && vkey != VK_SCROLL) { TRACE( "adjusting ScrollLock state (%02x)\n", keystate[VK_SCROLL] ); send_keyboard_input( 0, VK_SCROLL, 0x46, 0 ); send_keyboard_input( 0, VK_SCROLL, 0x46, KEYEVENTF_KEYUP ); } } /*********************************************************************** * keyboard_event * * JNI callback, runs in the context of the Java thread. */ jboolean keyboard_event( JNIEnv *env, jobject obj, jint win, jint action, jint keycode, jint state ) { union event_data data; if ((unsigned int)keycode >= ARRAY_SIZE( keycode_to_vkey ) || !keycode_to_vkey[keycode]) { p__android_log_print( ANDROID_LOG_WARN, "wine", "keyboard_event: win %x code %u unmapped key, ignoring", win, keycode ); return JNI_FALSE; } data.type = KEYBOARD_EVENT; data.kbd.hwnd = LongToHandle( win ); data.kbd.lock_state = state; data.kbd.input.type = INPUT_KEYBOARD; data.kbd.input.u.ki.wVk = keycode_to_vkey[keycode]; data.kbd.input.u.ki.wScan = vkey_to_scancode[data.kbd.input.u.ki.wVk]; data.kbd.input.u.ki.time = 0; data.kbd.input.u.ki.dwExtraInfo = 0; data.kbd.input.u.ki.dwFlags = (data.kbd.input.u.ki.wScan & 0x100) ? KEYEVENTF_EXTENDEDKEY : 0; if (action == AKEY_EVENT_ACTION_UP) data.kbd.input.u.ki.dwFlags |= KEYEVENTF_KEYUP; p__android_log_print( ANDROID_LOG_INFO, "wine", "keyboard_event: win %x code %u vkey %x scan %x meta %x", win, keycode, data.kbd.input.u.ki.wVk, data.kbd.input.u.ki.wScan, state ); send_event( &data ); return JNI_TRUE; } /*********************************************************************** * ANDROID_ToUnicodeEx */ INT CDECL ANDROID_ToUnicodeEx( UINT virt, UINT scan, const BYTE *state, LPWSTR buf, int size, UINT flags, HKL hkl ) { WCHAR buffer[2]; BOOL shift = state[VK_SHIFT] & 0x80; BOOL ctrl = state[VK_CONTROL] & 0x80; BOOL numlock = state[VK_NUMLOCK] & 0x01; buffer[0] = buffer[1] = 0; if (scan & 0x8000) return 0; /* key up */ /* FIXME: hardcoded layout */ if (!ctrl) { switch (virt) { case VK_BACK: buffer[0] = '\b'; break; case VK_OEM_1: buffer[0] = shift ? ':' : ';'; break; case VK_OEM_2: buffer[0] = shift ? '?' : '/'; break; case VK_OEM_3: buffer[0] = shift ? '~' : '`'; break; case VK_OEM_4: buffer[0] = shift ? '{' : '['; break; case VK_OEM_5: buffer[0] = shift ? '|' : '\\'; break; case VK_OEM_6: buffer[0] = shift ? '}' : ']'; break; case VK_OEM_7: buffer[0] = shift ? '"' : '\''; break; case VK_OEM_COMMA: buffer[0] = shift ? '<' : ','; break; case VK_OEM_MINUS: buffer[0] = shift ? '_' : '-'; break; case VK_OEM_PERIOD: buffer[0] = shift ? '>' : '.'; break; case VK_OEM_PLUS: buffer[0] = shift ? '+' : '='; break; case VK_RETURN: buffer[0] = '\r'; break; case VK_SPACE: buffer[0] = ' '; break; case VK_TAB: buffer[0] = '\t'; break; case VK_MULTIPLY: buffer[0] = '*'; break; case VK_ADD: buffer[0] = '+'; break; case VK_SUBTRACT: buffer[0] = '-'; break; case VK_DIVIDE: buffer[0] = '/'; break; default: if (virt >= '0' && virt <= '9') { buffer[0] = shift ? ")!@#$%^&*("[virt - '0'] : virt; break; } if (virt >= 'A' && virt <= 'Z') { buffer[0] = shift || (state[VK_CAPITAL] & 0x01) ? virt : virt + 'a' - 'A'; break; } if (virt >= VK_NUMPAD0 && virt <= VK_NUMPAD9 && numlock && !shift) { buffer[0] = '0' + virt - VK_NUMPAD0; break; } if (virt == VK_DECIMAL && numlock && !shift) { buffer[0] = '.'; break; } break; } } else /* Control codes */ { if (virt >= 'A' && virt <= 'Z') buffer[0] = virt - 'A' + 1; else { switch (virt) { case VK_OEM_4: buffer[0] = 0x1b; break; case VK_OEM_5: buffer[0] = 0x1c; break; case VK_OEM_6: buffer[0] = 0x1d; break; case VK_SUBTRACT: buffer[0] = 0x1e; break; } } } lstrcpynW( buf, buffer, size ); TRACE( "returning %d / %s\n", strlenW( buffer ), debugstr_wn(buf, strlenW( buffer ))); return strlenW( buffer ); } /*********************************************************************** * ANDROID_GetKeyNameText */ INT CDECL ANDROID_GetKeyNameText( LONG lparam, LPWSTR buffer, INT size ) { int scancode, vkey, len; const char *name; char key[2]; scancode = (lparam >> 16) & 0x1FF; vkey = scancode_to_vkey( scancode ); if (lparam & (1 << 25)) { /* Caller doesn't care about distinctions between left and right keys. */ switch (vkey) { case VK_LSHIFT: case VK_RSHIFT: vkey = VK_SHIFT; break; case VK_LCONTROL: case VK_RCONTROL: vkey = VK_CONTROL; break; case VK_LMENU: case VK_RMENU: vkey = VK_MENU; break; } } if (scancode & 0x100) vkey |= 0x100; if ((vkey >= 0x30 && vkey <= 0x39) || (vkey >= 0x41 && vkey <= 0x5a)) { key[0] = vkey; if (vkey >= 0x41) key[0] += 0x20; key[1] = 0; name = key; } else { name = vkey_to_name( vkey ); } len = MultiByteToWideChar( CP_UTF8, 0, name, -1, buffer, size ); if (len) len--; if (!len) { static const WCHAR format[] = {'K','e','y',' ','0','x','%','0','2','x',0}; snprintfW( buffer, size, format, vkey ); len = strlenW( buffer ); } TRACE( "lparam 0x%08x -> %s\n", lparam, debugstr_w( buffer )); return len; } /*********************************************************************** * ANDROID_MapVirtualKeyEx */ UINT CDECL ANDROID_MapVirtualKeyEx( UINT code, UINT maptype, HKL hkl ) { UINT ret = 0; const char *s; TRACE_(key)( "code=0x%x, maptype=%d, hkl %p\n", code, maptype, hkl ); switch (maptype) { case MAPVK_VK_TO_VSC_EX: case MAPVK_VK_TO_VSC: /* vkey to scancode */ switch (code) { case VK_SHIFT: code = VK_LSHIFT; break; case VK_CONTROL: code = VK_LCONTROL; break; case VK_MENU: code = VK_LMENU; break; } if (code < ARRAY_SIZE( vkey_to_scancode )) ret = vkey_to_scancode[code]; /* set scan code prefix */ if (maptype == MAPVK_VK_TO_VSC_EX && (code == VK_RCONTROL || code == VK_RMENU)) ret |= 0xe000; break; case MAPVK_VSC_TO_VK: case MAPVK_VSC_TO_VK_EX: /* scancode to vkey */ ret = scancode_to_vkey( code ); if (maptype == MAPVK_VSC_TO_VK) switch (ret) { case VK_LSHIFT: case VK_RSHIFT: ret = VK_SHIFT; break; case VK_LCONTROL: case VK_RCONTROL: ret = VK_CONTROL; break; case VK_LMENU: case VK_RMENU: ret = VK_MENU; break; } break; case MAPVK_VK_TO_CHAR: s = vkey_to_name( code ); if (s && (strlen( s ) == 1)) ret = s[0]; else ret = 0; break; default: FIXME( "Unknown maptype %d\n", maptype ); break; } TRACE_(key)( "returning 0x%04x\n", ret ); return ret; } /*********************************************************************** * ANDROID_GetKeyboardLayout */ HKL CDECL ANDROID_GetKeyboardLayout( DWORD thread_id ) { ULONG_PTR layout = GetUserDefaultLCID(); LANGID langid; static int once; langid = PRIMARYLANGID(LANGIDFROMLCID( layout )); if (langid == LANG_CHINESE || langid == LANG_JAPANESE || langid == LANG_KOREAN) layout = MAKELONG( layout, 0xe001 ); /* IME */ else layout |= layout << 16; if (!once++) FIXME( "returning %lx\n", layout ); return (HKL)layout; } /*********************************************************************** * ANDROID_VkKeyScanEx */ SHORT CDECL ANDROID_VkKeyScanEx( WCHAR ch, HKL hkl ) { SHORT ret = -1; if (ch < ARRAY_SIZE( char_vkey_map )) ret = char_vkey_map[ch]; TRACE_(key)( "ch %04x hkl %p -> %04x\n", ch, hkl, ret ); return ret; }