# HG changeset patch # User Bram Moolenaar # Date 1658549707 -7200 # Node ID 339fe29686908ebf51aa1b2b9eddc22017578794 # Parent 6d6fa84d617c2f6c344f1d7e7ef498167616f4cb patch 9.0.0058: Win32: cannot test low level events Commit: https://github.com/vim/vim/commit/81a3ff97e2012bdafc3ece796289f2e11e2754f3 Author: Yegappan Lakshmanan Date: Sat Jul 23 05:04:16 2022 +0100 patch 9.0.0058: Win32: cannot test low level events Problem: Win32: cannot test low level events. Solution: Add "sendevent" to test_gui_event(). (Yegappan Lakshmanan, closes #10679) diff --git a/runtime/doc/testing.txt b/runtime/doc/testing.txt --- a/runtime/doc/testing.txt +++ b/runtime/doc/testing.txt @@ -94,6 +94,7 @@ test_gui_event({event}, {args}) "findrepl" search and replace text. "mouse" mouse button click event. "scrollbar" move or drag the scrollbar. + "sendevent" send a low-level GUI event. "tabline" select a tab page by mouse click. "tabmenu" select a tabline menu entry. @@ -177,6 +178,15 @@ test_gui_event({event}, {args}) dragging: 1 to drag the scrollbar and 0 to click in the scrollbar. + "sendevent": + Send a low-level GUI event (e.g. key-up or down). + Currently only supported on MS-Windows. + The supported items in {args} are: + event: The supported string values are: + keyup generate a keyup event + keydown generate a keydown event + keycode: Keycode to use for a keyup or a keydown event. + "tabline": Inject a mouse click event on the tabline to select a tabpage. The supported items in {args} are: diff --git a/src/errors.h b/src/errors.h --- a/src/errors.h +++ b/src/errors.h @@ -3303,4 +3303,6 @@ EXTERN char e_could_not_check_for_pendin #ifdef FEAT_EVAL EXTERN char e_substitute_nesting_too_deep[] INIT(= N_("E1290: substitute nesting too deep")); +EXTERN char e_invalid_argument_nr[] + INIT(= N_("E1291: Invalid argument: %ld")); #endif diff --git a/src/gui_w32.c b/src/gui_w32.c --- a/src/gui_w32.c +++ b/src/gui_w32.c @@ -8541,3 +8541,42 @@ netbeans_draw_multisign_indicator(int ro SetPixel(s_hdc, x+2, y, gui.currFgColor); } #endif + +#if defined(FEAT_EVAL) || defined(PROTO) + int +test_gui_w32_sendevent(dict_T *args) +{ + char_u *event; + INPUT inputs[1]; + + event = dict_get_string(args, "event", TRUE); + if (event == NULL) + return FALSE; + + ZeroMemory(inputs, sizeof(inputs)); + + if (STRICMP(event, "keydown") == 0 || STRICMP(event, "keyup") == 0) + { + WORD vkCode; + + vkCode = dict_get_number_def(args, "keycode", 0); + if (vkCode <= 0 || vkCode >= 0xFF) + { + semsg(_(e_invalid_argument_nr), (long)vkCode); + return FALSE; + } + + inputs[0].type = INPUT_KEYBOARD; + inputs[0].ki.wVk = vkCode; + if (STRICMP(event, "keyup") == 0) + inputs[0].ki.dwFlags = KEYEVENTF_KEYUP; + SendInput(ARRAYSIZE(inputs), inputs, sizeof(INPUT)); + } + else + semsg(_(e_invalid_argument_str), event); + + vim_free(event); + + return TRUE; +} +#endif diff --git a/src/proto/gui_w32.pro b/src/proto/gui_w32.pro --- a/src/proto/gui_w32.pro +++ b/src/proto/gui_w32.pro @@ -96,4 +96,5 @@ void gui_mch_post_balloon(BalloonEval *b BalloonEval *gui_mch_create_beval_area(void *target, char_u *mesg, void (*mesgCB)(BalloonEval *, int), void *clientData); void gui_mch_destroy_beval_area(BalloonEval *beval); void netbeans_draw_multisign_indicator(int row); +int test_gui_w32_sendevent(dict_T *args); /* vim: set ft=c : */ diff --git a/src/testdir/test_gui.vim b/src/testdir/test_gui.vim --- a/src/testdir/test_gui.vim +++ b/src/testdir/test_gui.vim @@ -1606,4 +1606,88 @@ func Test_gui_dialog_file() call delete('Xlines') endfunc +" Test for sending low level key presses +func SendKeys(keylist) + for k in a:keylist + call test_gui_event("sendevent", #{event: "keydown", keycode: k}) + endfor + for k in reverse(a:keylist) + call test_gui_event("sendevent", #{event: "keyup", keycode: k}) + endfor +endfunc + +func Test_gui_lowlevel_keyevent() + CheckMSWindows + new + + " Test for to keys + for kc in range(65, 90) + call SendKeys([0x11, kc]) + let ch = getcharstr() + call assert_equal(nr2char(kc - 64), ch) + endfor + + " Test for the various Ctrl and Shift key combinations. + let keytests = [ + \ [[0x10, 0x21], "\", 2], + \ [[0x11, 0x21], "\", 4], + \ [[0x10, 0x22], "\", 2], + \ [[0x11, 0x22], "\", 4], + \ [[0x10, 0x23], "\", 0], + \ [[0x11, 0x23], "\", 0], + \ [[0x10, 0x24], "\", 0], + \ [[0x11, 0x24], "\", 0], + \ [[0x10, 0x25], "\", 0], + \ [[0x11, 0x25], "\", 0], + \ [[0x10, 0x26], "\", 0], + \ [[0x11, 0x26], "\", 4], + \ [[0x10, 0x27], "\", 0], + \ [[0x11, 0x27], "\", 0], + \ [[0x10, 0x28], "\", 0], + \ [[0x11, 0x28], "\", 4], + \ [[0x11, 0x30], "\", 4], + \ [[0x11, 0x31], "\", 4], + \ [[0x11, 0x32], "\", 4], + \ [[0x11, 0x33], "\", 4], + \ [[0x11, 0x34], "\", 4], + \ [[0x11, 0x35], "\", 4], + \ [[0x11, 0x36], "\", 0], + \ [[0x11, 0x37], "\", 4], + \ [[0x11, 0x38], "\", 4], + \ [[0x11, 0x39], "\", 4], + \ [[0x11, 0x60], "\", 4], + \ [[0x11, 0x61], "\", 4], + \ [[0x11, 0x62], "\", 4], + \ [[0x11, 0x63], "\", 4], + \ [[0x11, 0x64], "\", 4], + \ [[0x11, 0x65], "\", 4], + \ [[0x11, 0x66], "\", 4], + \ [[0x11, 0x67], "\", 4], + \ [[0x11, 0x68], "\", 4], + \ [[0x11, 0x69], "\", 4], + \ [[0x11, 0x6A], "\", 4], + \ [[0x11, 0x6B], "\", 4], + \ [[0x11, 0x6D], "\", 4], + \ [[0x11, 0x70], "\", 4], + \ [[0x11, 0x71], "\", 4], + \ [[0x11, 0x72], "\", 4], + \ [[0x11, 0x73], "\", 4], + \ [[0x11, 0x74], "\", 4], + \ [[0x11, 0x75], "\", 4], + \ [[0x11, 0x76], "\", 4], + \ [[0x11, 0x77], "\", 4], + \ [[0x11, 0x78], "\", 4], + \ ] + + for [kcodes, kstr, kmod] in keytests + call SendKeys(kcodes) + let ch = getcharstr() + let mod = getcharmod() + call assert_equal(kstr, ch, $"key = {kstr}") + call assert_equal(kmod, mod) + endfor + + bw! +endfunc + " vim: shiftwidth=2 sts=2 expandtab diff --git a/src/testing.c b/src/testing.c --- a/src/testing.c +++ b/src/testing.c @@ -1500,6 +1500,12 @@ f_test_gui_event(typval_T *argvars UNUSE rettv->v_type = VAR_BOOL; rettv->vval.v_number = FALSE; + if (sandbox != 0) + { + emsg(_(e_not_allowed_in_sandbox)); + return; + } + if (check_for_string_arg(argvars, 0) == FAIL || check_for_dict_arg(argvars, 1) == FAIL || argvars[1].vval.v_dict == NULL) @@ -1520,6 +1526,10 @@ f_test_gui_event(typval_T *argvars UNUSE rettv->vval.v_number = test_gui_tabline_event(argvars[1].vval.v_dict); else if (STRCMP(event, "tabmenu") == 0) rettv->vval.v_number = test_gui_tabmenu_event(argvars[1].vval.v_dict); +# ifdef FEAT_GUI_MSWIN + else if (STRCMP(event, "sendevent") == 0) + rettv->vval.v_number = test_gui_w32_sendevent(argvars[1].vval.v_dict); +# endif else { semsg(_(e_invalid_argument_str), event); diff --git a/src/version.c b/src/version.c --- a/src/version.c +++ b/src/version.c @@ -736,6 +736,8 @@ static char *(features[]) = static int included_patches[] = { /* Add new patch number below this line */ /**/ + 58, +/**/ 57, /**/ 56,