summaryrefslogtreecommitdiffstats
path: root/src/testdir/test_mswin_event.vim
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/testdir/test_mswin_event.vim1006
1 files changed, 1006 insertions, 0 deletions
diff --git a/src/testdir/test_mswin_event.vim b/src/testdir/test_mswin_event.vim
new file mode 100644
index 0000000..e2add3b
--- /dev/null
+++ b/src/testdir/test_mswin_event.vim
@@ -0,0 +1,1006 @@
+" Test MS-Windows input event handling.
+" Most of this works the same in Windows GUI as well as Windows console.
+
+source check.vim
+CheckMSWindows
+source mouse.vim
+
+" Helper function for sending a grouped sequence of low level key presses
+" The modifer key(s) can be included as VK Key Codes in the sequence
+" Keydown events will be sent, to to the end of the group, then keyup events
+" will be sent in reverse order to release the keys.
+func SendKeyGroup(keygroup)
+ for k in a:keygroup
+ call test_mswin_event("key", {'event': "keydown", 'keycode': k})
+ endfor
+ for k in reverse(copy(a:keygroup))
+ call test_mswin_event("key", {'event': "keyup", 'keycode': k})
+ endfor
+endfunc
+
+" Send individual key press and release events.
+" the modifers for the key press can be specified in the modifiers arg.
+func SendKeyWithModifiers(key, modifiers)
+ let args = { }
+ let args.keycode = a:key
+ let args.modifiers = a:modifiers
+ let args.event = "keydown"
+ call test_mswin_event("key", args)
+ let args.event = "keyup"
+ call test_mswin_event("key", args)
+ unlet args
+endfunc
+
+" Send an individual key press, without modifiers.
+func SendKey(key)
+ call SendKeyWithModifiers(a:key, 0)
+endfunc
+
+" Send a string of individual key-press events, without modifiers.
+func SendKeyStr(keystring)
+ for k in a:keystring
+ call SendKey(k)
+ endfor
+endfunc
+
+" This tells Vim to execute the buffered keys as user commands,
+" ie. same as feekdeys with mode X would do.
+func ExecuteBufferedKeys()
+ if has('gui_running')
+ call feedkeys("\<Esc>", 'Lx!')
+ else
+ call test_mswin_event("key", {'execute': v:true})
+ endif
+endfunc
+
+" Refer to the following page for the virtual key codes:
+" https://docs.microsoft.com/en-us/windows/win32/inputdev/virtual-key-codes
+let s:VK = {
+ \ 'ENTER' : 0x0D,
+ \ 'SPACE' : 0x20,
+ \ 'SHIFT' : 0x10,
+ \ 'LSHIFT' : 0xA0,
+ \ 'RSHIFT' : 0xA1,
+ \ 'CONTROL' : 0x11,
+ \ 'LCONTROL' : 0xA2,
+ \ 'RCONTROL' : 0xA3,
+ \ 'MENU' : 0x12,
+ \ 'ALT' : 0x12,
+ \ 'LMENU' : 0xA4,
+ \ 'LALT' : 0xA4,
+ \ 'RMENU' : 0xA5,
+ \ 'RALT' : 0xA5,
+ \ 'OEM_1' : 0xBA,
+ \ 'OEM_2' : 0xBF,
+ \ 'OEM_3' : 0xC0,
+ \ 'OEM_4' : 0xDB,
+ \ 'OEM_5' : 0xDC,
+ \ 'OEM_6' : 0xDD,
+ \ 'OEM_7' : 0xDE,
+ \ 'OEM_PLUS' : 0xBB,
+ \ 'OEM_COMMA' : 0xBC,
+ \ 'OEM_MINUS' : 0xBD,
+ \ 'OEM_PERIOD' : 0xBE,
+ \ 'PRIOR' : 0x21,
+ \ 'NEXT' : 0x22,
+ \ 'END' : 0x23,
+ \ 'HOME' : 0x24,
+ \ 'LEFT' : 0x25,
+ \ 'UP' : 0x26,
+ \ 'RIGHT' : 0x27,
+ \ 'DOWN' : 0x28,
+ \ 'KEY_0' : 0x30,
+ \ 'KEY_1' : 0x31,
+ \ 'KEY_2' : 0x32,
+ \ 'KEY_3' : 0x33,
+ \ 'KEY_4' : 0x34,
+ \ 'KEY_5' : 0x35,
+ \ 'KEY_6' : 0x36,
+ \ 'KEY_7' : 0x37,
+ \ 'KEY_8' : 0x38,
+ \ 'KEY_9' : 0x39,
+ \ 'KEY_A' : 0x41,
+ \ 'KEY_B' : 0x42,
+ \ 'KEY_C' : 0x43,
+ \ 'KEY_D' : 0x44,
+ \ 'KEY_E' : 0x45,
+ \ 'KEY_F' : 0x46,
+ \ 'KEY_G' : 0x47,
+ \ 'KEY_H' : 0x48,
+ \ 'KEY_I' : 0x49,
+ \ 'KEY_J' : 0x4A,
+ \ 'KEY_K' : 0x4B,
+ \ 'KEY_L' : 0x4C,
+ \ 'KEY_M' : 0x4D,
+ \ 'KEY_N' : 0x4E,
+ \ 'KEY_O' : 0x4F,
+ \ 'KEY_P' : 0x50,
+ \ 'KEY_Q' : 0x51,
+ \ 'KEY_R' : 0x52,
+ \ 'KEY_S' : 0x53,
+ \ 'KEY_T' : 0x54,
+ \ 'KEY_U' : 0x55,
+ \ 'KEY_V' : 0x56,
+ \ 'KEY_W' : 0x57,
+ \ 'KEY_X' : 0x58,
+ \ 'KEY_Y' : 0x59,
+ \ 'KEY_Z' : 0x5A,
+ \ 'NUMPAD0' : 0x60,
+ \ 'NUMPAD1' : 0x61,
+ \ 'NUMPAD2' : 0x62,
+ \ 'NUMPAD3' : 0x63,
+ \ 'NUMPAD4' : 0x64,
+ \ 'NUMPAD5' : 0x65,
+ \ 'NUMPAD6' : 0x66,
+ \ 'NUMPAD7' : 0x67,
+ \ 'NUMPAD8' : 0x68,
+ \ 'NUMPAD9' : 0x69,
+ \ 'MULTIPLY' : 0x6A,
+ \ 'ADD' : 0x6B,
+ \ 'SUBTRACT' : 0x6D,
+ \ 'F1' : 0x70,
+ \ 'F2' : 0x71,
+ \ 'F3' : 0x72,
+ \ 'F4' : 0x73,
+ \ 'F5' : 0x74,
+ \ 'F6' : 0x75,
+ \ 'F7' : 0x76,
+ \ 'F8' : 0x77,
+ \ 'F9' : 0x78,
+ \ 'F10' : 0x79,
+ \ 'F11' : 0x7A,
+ \ 'F12' : 0x7B,
+ \ 'DELETE' : 0x2E,
+ \ 'BACK' : 0x08,
+ \ 'ESCAPE' : 0x1B
+ \ }
+
+ let s:MOD_MASK_SHIFT = 0x02
+ let s:MOD_MASK_CTRL = 0x04
+ let s:MOD_MASK_ALT = 0x08
+
+ let s:vim_key_modifiers = [
+ \ ["", 0, []],
+ \ ["S-", 2, [s:VK.LSHIFT]],
+ \ ["C-", 4, [s:VK.LCONTROL]],
+ \ ["C-S-", 6, [s:VK.LCONTROL, s:VK.LSHIFT]],
+ \ ["A-", 8, [s:VK.LMENU]],
+ \ ["A-S-", 10, [s:VK.LMENU, s:VK.LSHIFT]],
+ \ ["A-C-", 12, [s:VK.LMENU, s:VK.LCONTROL]],
+ \ ["A-C-S-", 14, [s:VK.LMENU, s:VK.LCONTROL, s:VK.LSHIFT]],
+ \]
+
+ " Assuming Standard US PC Keyboard layout
+ let s:test_ascii_key_chars = [
+ \ [[s:VK.SPACE], ' '],
+ \ [[s:VK.OEM_1], ';'],
+ \ [[s:VK.OEM_2], '/'],
+ \ [[s:VK.OEM_3], '`'],
+ \ [[s:VK.OEM_4], '['],
+ \ [[s:VK.OEM_5], '\'],
+ \ [[s:VK.OEM_6], ']'],
+ \ [[s:VK.OEM_7], ''''],
+ \ [[s:VK.OEM_PLUS], '='],
+ \ [[s:VK.OEM_COMMA], ','],
+ \ [[s:VK.OEM_MINUS], '-'],
+ \ [[s:VK.OEM_PERIOD], '.'],
+ \ [[s:VK.SHIFT, s:VK.OEM_1], ':'],
+ \ [[s:VK.SHIFT, s:VK.OEM_2], '?'],
+ \ [[s:VK.SHIFT, s:VK.OEM_3], '~'],
+ \ [[s:VK.SHIFT, s:VK.OEM_4], '{'],
+ \ [[s:VK.SHIFT, s:VK.OEM_5], '|'],
+ \ [[s:VK.SHIFT, s:VK.OEM_6], '}'],
+ \ [[s:VK.SHIFT, s:VK.OEM_7], '"'],
+ \ [[s:VK.SHIFT, s:VK.OEM_PLUS], '+'],
+ \ [[s:VK.SHIFT, s:VK.OEM_COMMA], '<'],
+ \ [[s:VK.SHIFT, s:VK.OEM_MINUS], '_'],
+ \ [[s:VK.SHIFT, s:VK.OEM_PERIOD], '>'],
+ \ [[s:VK.KEY_1], '1'],
+ \ [[s:VK.KEY_2], '2'],
+ \ [[s:VK.KEY_3], '3'],
+ \ [[s:VK.KEY_4], '4'],
+ \ [[s:VK.KEY_5], '5'],
+ \ [[s:VK.KEY_6], '6'],
+ \ [[s:VK.KEY_7], '7'],
+ \ [[s:VK.KEY_8], '8'],
+ \ [[s:VK.KEY_9], '9'],
+ \ [[s:VK.KEY_0], '0'],
+ \ [[s:VK.SHIFT, s:VK.KEY_1], '!'],
+ \ [[s:VK.SHIFT, s:VK.KEY_2], '@'],
+ \ [[s:VK.SHIFT, s:VK.KEY_3], '#'],
+ \ [[s:VK.SHIFT, s:VK.KEY_4], '$'],
+ \ [[s:VK.SHIFT, s:VK.KEY_5], '%'],
+ \ [[s:VK.SHIFT, s:VK.KEY_6], '^'],
+ \ [[s:VK.SHIFT, s:VK.KEY_7], '&'],
+ \ [[s:VK.SHIFT, s:VK.KEY_8], '*'],
+ \ [[s:VK.SHIFT, s:VK.KEY_9], '('],
+ \ [[s:VK.SHIFT, s:VK.KEY_0], ')'],
+ \ [[s:VK.KEY_A], 'a'],
+ \ [[s:VK.KEY_B], 'b'],
+ \ [[s:VK.KEY_C], 'c'],
+ \ [[s:VK.KEY_D], 'd'],
+ \ [[s:VK.KEY_E], 'e'],
+ \ [[s:VK.KEY_F], 'f'],
+ \ [[s:VK.KEY_G], 'g'],
+ \ [[s:VK.KEY_H], 'h'],
+ \ [[s:VK.KEY_I], 'i'],
+ \ [[s:VK.KEY_J], 'j'],
+ \ [[s:VK.KEY_K], 'k'],
+ \ [[s:VK.KEY_L], 'l'],
+ \ [[s:VK.KEY_M], 'm'],
+ \ [[s:VK.KEY_N], 'n'],
+ \ [[s:VK.KEY_O], 'o'],
+ \ [[s:VK.KEY_P], 'p'],
+ \ [[s:VK.KEY_Q], 'q'],
+ \ [[s:VK.KEY_R], 'r'],
+ \ [[s:VK.KEY_S], 's'],
+ \ [[s:VK.KEY_T], 't'],
+ \ [[s:VK.KEY_U], 'u'],
+ \ [[s:VK.KEY_V], 'v'],
+ \ [[s:VK.KEY_W], 'w'],
+ \ [[s:VK.KEY_X], 'x'],
+ \ [[s:VK.KEY_Y], 'y'],
+ \ [[s:VK.KEY_Z], 'z'],
+ \ [[s:VK.SHIFT, s:VK.KEY_A], 'A'],
+ \ [[s:VK.SHIFT, s:VK.KEY_B], 'B'],
+ \ [[s:VK.SHIFT, s:VK.KEY_C], 'C'],
+ \ [[s:VK.SHIFT, s:VK.KEY_D], 'D'],
+ \ [[s:VK.SHIFT, s:VK.KEY_E], 'E'],
+ \ [[s:VK.SHIFT, s:VK.KEY_F], 'F'],
+ \ [[s:VK.SHIFT, s:VK.KEY_G], 'G'],
+ \ [[s:VK.SHIFT, s:VK.KEY_H], 'H'],
+ \ [[s:VK.SHIFT, s:VK.KEY_I], 'I'],
+ \ [[s:VK.SHIFT, s:VK.KEY_J], 'J'],
+ \ [[s:VK.SHIFT, s:VK.KEY_K], 'K'],
+ \ [[s:VK.SHIFT, s:VK.KEY_L], 'L'],
+ \ [[s:VK.SHIFT, s:VK.KEY_M], 'M'],
+ \ [[s:VK.SHIFT, s:VK.KEY_N], 'N'],
+ \ [[s:VK.SHIFT, s:VK.KEY_O], 'O'],
+ \ [[s:VK.SHIFT, s:VK.KEY_P], 'P'],
+ \ [[s:VK.SHIFT, s:VK.KEY_Q], 'Q'],
+ \ [[s:VK.SHIFT, s:VK.KEY_R], 'R'],
+ \ [[s:VK.SHIFT, s:VK.KEY_S], 'S'],
+ \ [[s:VK.SHIFT, s:VK.KEY_T], 'T'],
+ \ [[s:VK.SHIFT, s:VK.KEY_U], 'U'],
+ \ [[s:VK.SHIFT, s:VK.KEY_V], 'V'],
+ \ [[s:VK.SHIFT, s:VK.KEY_W], 'W'],
+ \ [[s:VK.SHIFT, s:VK.KEY_X], 'X'],
+ \ [[s:VK.SHIFT, s:VK.KEY_Y], 'Y'],
+ \ [[s:VK.SHIFT, s:VK.KEY_Z], 'Z'],
+ \ [[s:VK.CONTROL, s:VK.KEY_A], 0x01],
+ \ [[s:VK.CONTROL, s:VK.KEY_B], 0x02],
+ \ [[s:VK.CONTROL, s:VK.KEY_C], 0x03],
+ \ [[s:VK.CONTROL, s:VK.KEY_D], 0x04],
+ \ [[s:VK.CONTROL, s:VK.KEY_E], 0x05],
+ \ [[s:VK.CONTROL, s:VK.KEY_F], 0x06],
+ \ [[s:VK.CONTROL, s:VK.KEY_G], 0x07],
+ \ [[s:VK.CONTROL, s:VK.KEY_H], 0x08],
+ \ [[s:VK.CONTROL, s:VK.KEY_I], 0x09],
+ \ [[s:VK.CONTROL, s:VK.KEY_J], 0x0A],
+ \ [[s:VK.CONTROL, s:VK.KEY_K], 0x0B],
+ \ [[s:VK.CONTROL, s:VK.KEY_L], 0x0C],
+ \ [[s:VK.CONTROL, s:VK.KEY_M], 0x0D],
+ \ [[s:VK.CONTROL, s:VK.KEY_N], 0x0E],
+ \ [[s:VK.CONTROL, s:VK.KEY_O], 0x0F],
+ \ [[s:VK.CONTROL, s:VK.KEY_P], 0x10],
+ \ [[s:VK.CONTROL, s:VK.KEY_Q], 0x11],
+ \ [[s:VK.CONTROL, s:VK.KEY_R], 0x12],
+ \ [[s:VK.CONTROL, s:VK.KEY_S], 0x13],
+ \ [[s:VK.CONTROL, s:VK.KEY_T], 0x14],
+ \ [[s:VK.CONTROL, s:VK.KEY_U], 0x15],
+ \ [[s:VK.CONTROL, s:VK.KEY_V], 0x16],
+ \ [[s:VK.CONTROL, s:VK.KEY_W], 0x17],
+ \ [[s:VK.CONTROL, s:VK.KEY_X], 0x18],
+ \ [[s:VK.CONTROL, s:VK.KEY_Y], 0x19],
+ \ [[s:VK.CONTROL, s:VK.KEY_Z], 0x1A],
+ \ [[s:VK.CONTROL, s:VK.OEM_4], 0x1B],
+ \ [[s:VK.CONTROL, s:VK.OEM_5], 0x1C],
+ \ [[s:VK.CONTROL, s:VK.OEM_6], 0x1D],
+ \ [[s:VK.CONTROL, s:VK.KEY_6], 0x1E],
+ \ [[s:VK.CONTROL, s:VK.OEM_MINUS], 0x1F],
+ \ ]
+
+let s:test_extra_key_chars = [
+ \ [[s:VK.ALT, s:VK.KEY_1], '±'],
+ \ [[s:VK.ALT, s:VK.KEY_2], '²'],
+ \ [[s:VK.ALT, s:VK.KEY_3], '³'],
+ \ [[s:VK.ALT, s:VK.KEY_4], '´'],
+ \ [[s:VK.ALT, s:VK.KEY_5], 'µ'],
+ \ [[s:VK.ALT, s:VK.KEY_6], '¶'],
+ \ [[s:VK.ALT, s:VK.KEY_7], '·'],
+ \ [[s:VK.ALT, s:VK.KEY_8], '¸'],
+ \ [[s:VK.ALT, s:VK.KEY_9], '¹'],
+ \ [[s:VK.ALT, s:VK.KEY_0], '°'],
+ \ [[s:VK.ALT, s:VK.KEY_A], 'á'],
+ \ [[s:VK.ALT, s:VK.KEY_B], 'â'],
+ \ [[s:VK.ALT, s:VK.KEY_C], 'ã'],
+ \ [[s:VK.ALT, s:VK.KEY_D], 'ä'],
+ \ [[s:VK.ALT, s:VK.KEY_E], 'å'],
+ \ [[s:VK.ALT, s:VK.KEY_F], 'æ'],
+ \ [[s:VK.ALT, s:VK.KEY_G], 'ç'],
+ \ [[s:VK.ALT, s:VK.KEY_H], 'è'],
+ \ [[s:VK.ALT, s:VK.KEY_I], 'é'],
+ \ [[s:VK.ALT, s:VK.KEY_J], 'ê'],
+ \ [[s:VK.ALT, s:VK.KEY_K], 'ë'],
+ \ [[s:VK.ALT, s:VK.KEY_L], 'ì'],
+ \ [[s:VK.ALT, s:VK.KEY_M], 'í'],
+ \ [[s:VK.ALT, s:VK.KEY_N], 'î'],
+ \ [[s:VK.ALT, s:VK.KEY_O], 'ï'],
+ \ [[s:VK.ALT, s:VK.KEY_P], 'ð'],
+ \ [[s:VK.ALT, s:VK.KEY_Q], 'ñ'],
+ \ [[s:VK.ALT, s:VK.KEY_R], 'ò'],
+ \ [[s:VK.ALT, s:VK.KEY_S], 'ó'],
+ \ [[s:VK.ALT, s:VK.KEY_T], 'ô'],
+ \ [[s:VK.ALT, s:VK.KEY_U], 'õ'],
+ \ [[s:VK.ALT, s:VK.KEY_V], 'ö'],
+ \ [[s:VK.ALT, s:VK.KEY_W], '÷'],
+ \ [[s:VK.ALT, s:VK.KEY_X], 'ø'],
+ \ [[s:VK.ALT, s:VK.KEY_Y], 'ù'],
+ \ [[s:VK.ALT, s:VK.KEY_Z], 'ú'],
+ \ ]
+
+func s:LoopTestKeyArray(arr)
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+
+ for [kcodes, kstr] in a:arr
+ " Send as a sequence of key presses.
+ call SendKeyGroup(kcodes)
+ let ch = getcharstr(0)
+ " need to deal a bit differently with the non-printable ascii chars < 0x20
+ if kstr < 0x20 && index([s:VK.CONTROL, s:VK.LCONTROL, s:VK.RCONTROL], kcodes[0]) >= 0
+ call assert_equal(nr2char(kstr), $"{ch}")
+ else
+ call assert_equal(kstr, $"{ch}")
+ endif
+ let mod_mask = getcharmod()
+ " the mod_mask is zero when no modifiers are used
+ " and when the virtual termcap maps the character
+ call assert_equal(0, mod_mask, $"key = {kstr}")
+
+ " Send as a single key press with a modifers mask.
+ let modifiers = 0
+ let key = kcodes[0]
+ for key in kcodes
+ if index([s:VK.SHIFT, s:VK.LSHIFT, s:VK.RSHIFT], key) >= 0
+ let modifiers = modifiers + s:MOD_MASK_SHIFT
+ endif
+ if index([s:VK.CONTROL, s:VK.LCONTROL, s:VK.RCONTROL], key) >= 0
+ let modifiers = modifiers + s:MOD_MASK_CTRL
+ endif
+ if index([s:VK.ALT, s:VK.LALT, s:VK.RALT], key) >= 0
+ let modifiers = modifiers + s:MOD_MASK_ALT
+ endif
+ endfor
+ call SendKeyWithModifiers(key, modifiers)
+ let ch = getcharstr(0)
+ " need to deal a bit differently with the non-printable ascii chars < 0x20
+ if kstr < 0x20 && index([s:VK.CONTROL, s:VK.LCONTROL, s:VK.RCONTROL], kcodes[0]) >= 0
+ call assert_equal(nr2char(kstr), $"{ch}")
+ else
+ call assert_equal(kstr, $"{ch}")
+ endif
+ let mod_mask = getcharmod()
+ " the mod_mask is zero when no modifiers are used
+ " and when the virtual termcap maps the character
+ call assert_equal(0, mod_mask, $"key = {kstr}")
+ endfor
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+
+endfunc
+
+" Test MS-Windows key events
+func Test_mswin_event_character_keys()
+ CheckMSWindows
+ new
+
+ call s:LoopTestKeyArray(s:test_ascii_key_chars)
+
+ if !has('gui_running')
+ call s:LoopTestKeyArray(s:test_extra_key_chars)
+ endif
+
+" Test keyboard codes for digits
+" (0x30 - 0x39) : VK_0 - VK_9 are the same as ASCII '0' - '9'
+ for kc in range(48, 57)
+ call SendKey(kc)
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc), ch)
+ call SendKeyWithModifiers(kc, 0)
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc), ch)
+ endfor
+
+" Test keyboard codes for Alt-0 to Alt-9
+" Expect +128 from the digit char codes
+ for modkey in [s:VK.ALT, s:VK.LALT, s:VK.RALT]
+ for kc in range(48, 57)
+ call SendKeyGroup([modkey, kc])
+ let ch = getchar(0)
+ call assert_equal(kc+128, ch)
+ call SendKeyWithModifiers(kc, s:MOD_MASK_ALT)
+ let ch = getchar(0)
+ call assert_equal(kc+128, ch)
+ endfor
+ endfor
+
+" Test for lowercase 'a' to 'z', VK codes 65(0x41) - 90(0x5A)
+" Note: VK_A-VK_Z virtual key codes coincide with uppercase ASCII codes A-Z.
+" eg VK_A is 65, and the ASCII character code for uppercase 'A' is also 65.
+" Caution: these are interpreted as lowercase when Shift is NOT pressed.
+" eg, sending VK_A (65) 'A' Key code without shift modifier, will produce ASCII
+" char 'a' (91) as the output. The ASCII codes for the lowercase letters are
+" numbered 32 higher than their uppercase versions.
+ for kc in range(65, 90)
+ call SendKey(kc)
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc + 32), ch)
+ call SendKeyWithModifiers(kc, 0)
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc + 32), ch)
+ endfor
+
+" Test for Uppercase 'A' - 'Z' keys
+" ie. with VK_SHIFT, expect the keycode = character code.
+ for modkey in [s:VK.SHIFT, s:VK.LSHIFT, s:VK.RSHIFT]
+ for kc in range(65, 90)
+ call SendKeyGroup([modkey, kc])
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc), ch)
+ call SendKeyWithModifiers(kc, s:MOD_MASK_SHIFT)
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc), ch)
+ endfor
+ endfor
+
+ " Test for <Ctrl-A> to <Ctrl-Z> keys
+ " Expect the unicode characters 0x01 to 0x1A
+ for modkey in [s:VK.CONTROL, s:VK.LCONTROL, s:VK.RCONTROL]
+ for kc in range(65, 90)
+ call SendKeyGroup([modkey, kc])
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc - 64), ch)
+ call SendKeyWithModifiers(kc, s:MOD_MASK_CTRL)
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(kc - 64), ch)
+ endfor
+ endfor
+
+ " Windows intercepts some of these keys in the GUI.
+ if !has("gui_running")
+ " Test for <Alt-A> to <Alt-Z> keys
+ " Expect the unicode characters 0xE1 to 0xFA
+ " ie. 160 higher than the lowercase equivalent
+ for modkey in [s:VK.ALT, s:VK.LALT, s:VK.RALT]
+ for kc in range(65, 90)
+ call SendKeyGroup([modkey, kc])
+ let ch = getchar(0)
+ call assert_equal(kc+160, ch)
+ call SendKeyWithModifiers(kc, s:MOD_MASK_ALT)
+ let ch = getchar(0)
+ call assert_equal(kc+160, ch)
+ endfor
+ endfor
+ endif
+
+endfun
+
+ " Test for Function Keys 'F1' to 'F12'
+ " VK codes 112(0x70) - 123(0x7B)
+ " Also with ALL permutatios of modifiers; Shift, Ctrl & Alt
+func Test_mswin_event_function_keys()
+
+ if has('gui_running')
+ let g:test_is_flaky = 1
+ endif
+
+ " NOTE: Windows intercepts these combinations in the GUI
+ let gui_nogo = ["A-F1", "A-F2", "A-F3", "A-F4", "A-S-F4", "A-C-S-F4",
+ \ "A-F5", "A-F6", "A-F7", "A-F8", "A-C-F8", "A-F9",
+ \ "A-F10", "A-F11" , "A-C-F11", "A-C-F12"]
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+
+ for [mod_str, vim_mod_mask, mod_keycodes] in s:vim_key_modifiers
+ for n in range(1, 12)
+ let expected_mod_mask = vim_mod_mask
+ let kstr = $"{mod_str}F{n}"
+ if !has('gui_running') || (has('gui_running') && n != 10
+ \ && index(gui_nogo, kstr) == -1)
+ let keycode = eval('"\<' .. kstr .. '>"')
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+ call SendKeyWithModifiers(111+n, vim_mod_mask)
+ let ch = getcharstr(0)
+ let mod_mask = getcharmod()
+ call assert_equal(keycode, $"{ch}", $"key = {kstr}")
+ " workaround for the virtual termcap maps changing the character
+ "instead of sending Shift
+ for mod_key in mod_keycodes
+ if index([s:VK.SHIFT, s:VK.LSHIFT, s:VK.RSHIFT], mod_key) >= 0
+ let expected_mod_mask -= s:MOD_MASK_SHIFT
+ break
+ endif
+ endfor
+ call assert_equal(expected_mod_mask, mod_mask, $"mod = {expected_mod_mask} for key = {kstr}")
+ endif
+ endfor
+ endfor
+endfunc
+
+func ExtractModifiers(mod_keycodes)
+ let has_shift = 0
+ let has_ctrl = 0
+ let has_alt = 0
+ for mod_key in a:mod_keycodes
+ if index([s:VK.SHIFT, s:VK.LSHIFT, s:VK.RSHIFT], mod_key) >= 0
+ let has_shift = 1
+ endif
+ if index([s:VK.CONTROL, s:VK.LCONTROL, s:VK.RCONTROL], mod_key) >= 0
+ let has_ctrl = 1
+ endif
+ if index([s:VK.MENU, s:VK.LMENU, s:VK.RMENU], mod_key) >= 0
+ let has_alt = 1
+ endif
+ endfor
+ return [has_shift, has_ctrl, has_alt]
+endfunc
+
+ " Test for Movement Keys;
+ " VK_PRIOR 0x21, VK_NEXT 0x22,
+ " VK_END 0x23, VK_HOME 0x24,
+ " VK_LEFT 0x25, VK_UP 0x26,
+ " VK_RIGHT 0x27, VK_DOWN 0x28
+ " With ALL permutations of modifiers; none, Shift, Ctrl & Alt
+func Test_mswin_event_movement_keys()
+
+ if has('gui_running')
+ let g:test_is_flaky = 1
+ endif
+
+ let movement_keys = [
+ \ [s:VK.PRIOR, "PageUp"],
+ \ [s:VK.NEXT, "PageDown"],
+ \ [s:VK.END, "End"],
+ \ [s:VK.HOME, "Home"],
+ \ [s:VK.LEFT, "Left"],
+ \ [s:VK.UP, "Up"],
+ \ [s:VK.RIGHT, "Right"],
+ \ [s:VK.DOWN, "Down"],
+ \ ]
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+
+ for [mod_str, vim_mod_mask, mod_keycodes] in s:vim_key_modifiers
+ for [kcode, kname] in movement_keys
+ let exp_mod_mask = vim_mod_mask
+ let kstr = $"{mod_str}{kname}"
+ let chstr_eval = eval('"\<' .. kstr .. '>"')
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+ execute 'call feedkeys("\<' .. kstr .. '>")'
+ let chstr_fk = getcharstr(0)
+ call assert_equal(chstr_eval, chstr_fk, $"feedkeys = <{kstr}>")
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+ call SendKey(kcode)
+ let chstr_alone = getcharstr(0)
+ let chstr_alone_end = chstr_alone[len(chstr_alone)-2:len(chstr_alone)-1]
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+ call SendKeyGroup(mod_keycodes + [kcode])
+ let chstr_mswin = getcharstr(0)
+ let chstr_mswin_end = chstr_mswin[len(chstr_mswin)-2:len(chstr_mswin)-1]
+ let mod_mask = getcharmod()
+
+ " The virtual termcap maps may** change the character and either;
+ " - remove the Shift modifier, or
+ " - remove the Ctrl modifier if the Shift modifier was not removed.
+ let [has_shift, has_ctrl, has_alt] = ExtractModifiers(mod_keycodes)
+ if chstr_alone_end != chstr_mswin_end
+ if has_shift != 0
+ let exp_mod_mask -= s:MOD_MASK_SHIFT
+ elseif has_ctrl != 0
+ let exp_mod_mask -= s:MOD_MASK_CTRL
+ endif
+ endif
+ " **Note: The appveyor Windows GUI test environments, from VS2017 on,
+ " consistently intercepts the Shift modifier WITHOUT changing the
+ " MOVEMENT character. This issue does not happen in any github actions
+ " CI Windows test environments. Attempted to reproduce this manually
+ " on Windows versions; 7, 8.1, 10, 11, Server 2019 and Server 2022, but
+ " the issue did not occur on any of those environments.
+ " Below is a workaround for the issue.
+ if has('gui_running') && has_shift != 0
+ if exp_mod_mask != mod_mask && chstr_eval != chstr_mswin
+ let kstr_sub = substitute(kstr, "S-", "", "")
+ let chstr_eval = eval('"\<' .. kstr_sub .. '>"')
+ if exp_mod_mask - s:MOD_MASK_SHIFT == mod_mask
+ let exp_mod_mask -= s:MOD_MASK_SHIFT
+ elseif has_ctrl != 0 && exp_mod_mask - s:MOD_MASK_CTRL == mod_mask
+ let exp_mod_mask -= s:MOD_MASK_CTRL
+ endif
+ endif
+ endif
+ call assert_equal(chstr_eval, chstr_mswin, $"key = {kstr}")
+ call assert_equal(exp_mod_mask, mod_mask, $"mod_mask for key = {kstr}")
+ endfor
+ endfor
+
+ bw!
+endfunc
+
+
+" Test for QWERTY Ctrl+- which should result in ^_
+" issue #10817
+func Test_QWERTY_Ctrl_minus()
+ CheckMSWindows
+ new
+
+ call SendKeyGroup([s:VK.CONTROL, s:VK.OEM_MINUS])
+ let ch = getcharstr(0)
+ call assert_equal(nr2char(0x1f),ch)
+
+ call SendKey(s:VK.KEY_I)
+ call SendKeyGroup([s:VK.CONTROL, s:VK.SUBTRACT])
+ call SendKey(s:VK.ESCAPE)
+ call ExecuteBufferedKeys()
+ call assert_equal('-', getline('$'))
+
+ %d _
+ imapclear
+ imap <C-_> BINGO
+ call SendKey(s:VK.KEY_I)
+ call SendKeyGroup([s:VK.CONTROL, s:VK.OEM_MINUS])
+ call SendKey(s:VK.ESCAPE)
+ call ExecuteBufferedKeys()
+ call assert_equal('BINGO', getline('$'))
+
+ %d _
+ imapclear
+ exec "imap \x1f BILBO"
+ call SendKey(s:VK.KEY_I)
+ call SendKeyGroup([s:VK.CONTROL, s:VK.OEM_MINUS])
+ call SendKey(s:VK.ESCAPE)
+ call ExecuteBufferedKeys()
+ call assert_equal('BILBO', getline('$'))
+
+ imapclear
+ bw!
+endfunc
+
+" Test MS-Windows mouse events
+func Test_mswin_event_mouse()
+ CheckMSWindows
+ new
+
+ set mousemodel=extend
+ call test_override('no_query_mouse', 1)
+ call WaitForResponses()
+
+ let msg = ''
+
+ call setline(1, ['one two three', 'four five six'])
+
+ " Test mouse movement
+ " by default, no mouse move events are generated
+ " this setting enables it to generate move events
+ set mousemev
+
+ if !has('gui_running')
+ " console version needs a button pressed,
+ " otherwise it ignores mouse movements.
+ call MouseLeftClick(2, 3)
+ endif
+ call MSWinMouseEvent(0x700, 8, 13, 0, 0, 0)
+ if has('gui_running')
+ call feedkeys("\<Esc>", 'Lx!')
+ endif
+ let pos = getmousepos()
+ call assert_equal(8, pos.screenrow)
+ call assert_equal(13, pos.screencol)
+
+ if !has('gui_running')
+ call MouseLeftClick(2, 3)
+ call MSWinMouseEvent(0x700, 6, 4, 1, 0, 0)
+ let pos = getmousepos()
+ call assert_equal(6, pos.screenrow)
+ call assert_equal(4, pos.screencol)
+ endif
+
+ " test cells vs pixels
+ if has('gui_running')
+ let args = { }
+ let args.row = 9
+ let args.col = 7
+ let args.move = 1
+ let args.cell = 1
+ call test_mswin_event("mouse", args)
+ call feedkeys("\<Esc>", 'Lx!')
+ let pos = getmousepos()
+ call assert_equal(9, pos.screenrow)
+ call assert_equal(7, pos.screencol)
+
+ let args.cell = 0
+ call test_mswin_event("mouse", args)
+ call feedkeys("\<Esc>", 'Lx!')
+ let pos = getmousepos()
+ call assert_equal(1, pos.screenrow)
+ call assert_equal(1, pos.screencol)
+
+ unlet args
+ endif
+
+ " finish testing mouse movement
+ set mousemev&
+
+ " place the cursor using left click and release in normal mode
+ call MouseLeftClick(2, 4)
+ call MouseLeftRelease(2, 4)
+ if has('gui_running')
+ call feedkeys("\<Esc>", 'Lx!')
+ endif
+ call assert_equal([0, 2, 4, 0], getpos('.'))
+
+ " select and yank a word
+ let @" = ''
+ call MouseLeftClick(1, 9)
+ let args = #{button: 0, row: 1, col: 9, multiclick: 1, modifiers: 0}
+ call test_mswin_event('mouse', args)
+ call MouseLeftRelease(1, 9)
+ call feedkeys("y", 'Lx!')
+ call assert_equal('three', @")
+
+ " create visual selection using right click
+ let @" = ''
+
+ call MouseLeftClick(2 ,6)
+ call MouseLeftRelease(2, 6)
+ call MouseRightClick(2, 13)
+ call MouseRightRelease(2, 13)
+ call feedkeys("y", 'Lx!')
+ call assert_equal('five six', @")
+
+ " paste using middle mouse button
+ let @* = 'abc '
+ call feedkeys('""', 'Lx!')
+ call MouseMiddleClick(1, 9)
+ call MouseMiddleRelease(1, 9)
+ if has('gui_running')
+ call feedkeys("\<Esc>", 'Lx!')
+ endif
+ call assert_equal(['one two abc three', 'four five six'], getline(1, '$'))
+
+ " test mouse scrolling (aka touchpad scrolling.)
+ %d _
+ set scrolloff=0
+ call setline(1, range(1, 100))
+
+ " Scroll Down
+ call MouseWheelDown(2, 1)
+ call MouseWheelDown(2, 1)
+ call MouseWheelDown(2, 1)
+ call feedkeys("H", 'Lx!')
+ call assert_equal(10, line('.'))
+
+ " Scroll Up
+ call MouseWheelUp(2, 1)
+ call MouseWheelUp(2, 1)
+ call feedkeys("H", 'Lx!')
+ call assert_equal(4, line('.'))
+
+ " Shift Scroll Down
+ call MouseShiftWheelDown(2, 1)
+ call feedkeys("H", 'Lx!')
+ " should scroll from where it is (4) + visible buffer height - cmdheight
+ let shift_scroll_height = line('w$') - line('w0') - &cmdheight
+ call assert_equal(4 + shift_scroll_height, line('.'))
+
+ " Shift Scroll Up
+ call MouseShiftWheelUp(2, 1)
+ call feedkeys("H", 'Lx!')
+ call assert_equal(4, line('.'))
+
+ if !has('gui_running')
+ " Shift Scroll Down (using MOD)
+ call MSWinMouseEvent(0x100, 2, 1, 0, 0, 0x04)
+ call feedkeys("H", 'Lx!')
+ " should scroll from where it is (4) + visible buffer height - cmdheight
+ let shift_scroll_height = line('w$') - line('w0') - &cmdheight
+ call assert_equal(4 + shift_scroll_height, line('.'))
+
+ " Shift Scroll Up (using MOD)
+ call MSWinMouseEvent(0x200, 2, 1, 0, 0, 0x04)
+ call feedkeys("H", 'Lx!')
+ call assert_equal(4, line('.'))
+ endif
+
+ set scrolloff&
+
+ %d _
+ set nowrap
+ " make the buffer 500 wide.
+ call setline(1, range(10)->join('')->repeat(50))
+ " Scroll Right
+ call MouseWheelRight(1, 5)
+ call MouseWheelRight(1, 10)
+ call MouseWheelRight(1, 15)
+ call feedkeys('g0', 'Lx!')
+ call assert_equal(19, col('.'))
+
+ " Scroll Left
+ call MouseWheelLeft(1, 15)
+ call MouseWheelLeft(1, 10)
+ call feedkeys('g0', 'Lx!')
+ call assert_equal(7, col('.'))
+
+ " Shift Scroll Right
+ call MouseShiftWheelRight(1, 10)
+ call feedkeys('g0', 'Lx!')
+ " should scroll from where it is (7) + window width
+ call assert_equal(7 + winwidth(0), col('.'))
+
+ " Shift Scroll Left
+ call MouseShiftWheelLeft(1, 50)
+ call feedkeys('g0', 'Lx!')
+ call assert_equal(7, col('.'))
+ set wrap&
+
+ %d _
+ call setline(1, repeat([repeat('a', 60)], 10))
+
+ " record various mouse events
+ let mouseEventNames = [
+ \ 'LeftMouse', 'LeftRelease', '2-LeftMouse', '3-LeftMouse',
+ \ 'S-LeftMouse', 'A-LeftMouse', 'C-LeftMouse', 'MiddleMouse',
+ \ 'MiddleRelease', '2-MiddleMouse', '3-MiddleMouse',
+ \ 'S-MiddleMouse', 'A-MiddleMouse', 'C-MiddleMouse',
+ \ 'RightMouse', 'RightRelease', '2-RightMouse',
+ \ '3-RightMouse', 'S-RightMouse', 'A-RightMouse', 'C-RightMouse',
+ \ ]
+ let mouseEventCodes = map(copy(mouseEventNames), "'<' .. v:val .. '>'")
+ let g:events = []
+ for e in mouseEventCodes
+ exe 'nnoremap ' .. e .. ' <Cmd>call add(g:events, "' ..
+ \ substitute(e, '[<>]', '', 'g') .. '")<CR>'
+ endfor
+
+ " Test various mouse buttons
+ "(0 - Left, 1 - Middle, 2 - Right,
+ " 0x300 - MOUSE_X1/FROM_LEFT_3RD_BUTTON,
+ " 0x400 - MOUSE_X2/FROM_LEFT_4TH_BUTTON)
+ for button in [0, 1, 2, 0x300, 0x400]
+ " Single click
+ let args = #{button: button, row: 2, col: 5, multiclick: 0, modifiers: 0}
+ call test_mswin_event('mouse', args)
+ let args.button = 3
+ call test_mswin_event('mouse', args)
+
+ " Double Click
+ let args.button = button
+ call test_mswin_event('mouse', args)
+ let args.multiclick = 1
+ call test_mswin_event('mouse', args)
+ let args.button = 3
+ let args.multiclick = 0
+ call test_mswin_event('mouse', args)
+
+ " Triple Click
+ let args.button = button
+ call test_mswin_event('mouse', args)
+ let args.multiclick = 1
+ call test_mswin_event('mouse', args)
+ call test_mswin_event('mouse', args)
+ let args.button = 3
+ let args.multiclick = 0
+ call test_mswin_event('mouse', args)
+
+ " Shift click
+ let args = #{button: button, row: 3, col: 7, multiclick: 0, modifiers: 4}
+ call test_mswin_event('mouse', args)
+ let args.button = 3
+ call test_mswin_event('mouse', args)
+
+ " Alt click
+ let args.button = button
+ let args.modifiers = 8
+ call test_mswin_event('mouse', args)
+ let args.button = 3
+ call test_mswin_event('mouse', args)
+
+ " Ctrl click
+ let args.button = button
+ let args.modifiers = 16
+ call test_mswin_event('mouse', args)
+ let args.button = 3
+ call test_mswin_event('mouse', args)
+
+ call feedkeys("\<Esc>", 'Lx!')
+ endfor
+
+ if has('gui_running')
+ call assert_equal(['LeftMouse', 'LeftRelease', 'LeftMouse',
+ \ '2-LeftMouse', 'LeftMouse', '2-LeftMouse', '3-LeftMouse',
+ \ 'S-LeftMouse', 'A-LeftMouse', 'C-LeftMouse', 'MiddleMouse',
+ \ 'MiddleRelease', 'MiddleMouse', '2-MiddleMouse', 'MiddleMouse',
+ \ '2-MiddleMouse', '3-MiddleMouse', 'S-MiddleMouse', 'A-MiddleMouse',
+ \ 'C-MiddleMouse', 'RightMouse', 'RightRelease', 'RightMouse',
+ \ '2-RightMouse', 'RightMouse', '2-RightMouse', '3-RightMouse',
+ \ 'S-RightMouse', 'A-RightMouse', 'C-RightMouse'],
+ \ g:events)
+ else
+ call assert_equal(['MiddleRelease', 'LeftMouse', '2-LeftMouse',
+ \ '3-LeftMouse', 'S-LeftMouse', 'MiddleMouse', '2-MiddleMouse',
+ \ '3-MiddleMouse', 'MiddleMouse', 'S-MiddleMouse', 'RightMouse',
+ \ '2-RightMouse', '3-RightMouse'],
+ \ g:events)
+ endif
+
+ for e in mouseEventCodes
+ exe 'nunmap ' .. e
+ endfor
+
+ bw!
+ call test_override('no_query_mouse', 0)
+ set mousemodel&
+endfunc
+
+
+" Test MS-Windows test_mswin_event error handling
+func Test_mswin_event_error_handling()
+
+ let args = #{button: 0xfff, row: 2, col: 4, move: 0, multiclick: 0, modifiers: 0}
+ if !has('gui_running')
+ call assert_fails("call test_mswin_event('mouse', args)",'E475:')
+ endif
+ let args = #{button: 0, row: 2, col: 4, move: 0, multiclick: 0, modifiers: 0}
+ call assert_fails("call test_mswin_event('a1b2c3', args)", 'E475:')
+ call assert_fails("call test_mswin_event(test_null_string(), {})", 'E475:')
+
+ call assert_fails("call test_mswin_event([], args)", 'E1174:')
+ call assert_fails("call test_mswin_event('abc', [])", 'E1206:')
+
+ call assert_false(test_mswin_event('mouse', test_null_dict()))
+ let args = #{row: 2, col: 4, multiclick: 0, modifiers: 0}
+ call assert_false(test_mswin_event('mouse', args))
+ let args = #{button: 0, col: 4, multiclick: 0, modifiers: 0}
+ call assert_false(test_mswin_event('mouse', args))
+ let args = #{button: 0, row: 2, multiclick: 0, modifiers: 0}
+ call assert_false(test_mswin_event('mouse', args))
+ let args = #{button: 0, row: 2, col: 4, modifiers: 0}
+ call assert_false(test_mswin_event('mouse', args))
+ let args = #{button: 0, row: 2, col: 4, multiclick: 0}
+ call assert_false(test_mswin_event('mouse', args))
+
+ call assert_false(test_mswin_event('key', test_null_dict()))
+ call assert_fails("call test_mswin_event('key', [])", 'E1206:')
+ call assert_fails("call test_mswin_event('key', {'event': 'keydown', 'keycode': 0x0})", 'E1291:')
+ call assert_fails("call test_mswin_event('key', {'event': 'keydown', 'keycode': [15]})", 'E745:')
+ call assert_fails("call test_mswin_event('key', {'event': 'keys', 'keycode': 0x41})", 'E475:')
+ call assert_fails("call test_mswin_event('key', {'keycode': 0x41})", 'E417:')
+ call assert_fails("call test_mswin_event('key', {'event': 'keydown'})", 'E1291:')
+
+ call assert_fails("sandbox call test_mswin_event('key', {'event': 'keydown', 'keycode': 61 })", 'E48:')
+
+ " flush out the typeahead buffer
+ while getchar(0)
+ endwhile
+endfunc
+
+
+" vim: shiftwidth=2 sts=2 expandtab