forked from aniani/vim
patch 8.2.0385: menu functionality insufficiently tested
Problem: Menu functionality insufficiently tested. Solution: Add tests. Add menu_info(). (Yegappan Lakshmanan, closes #5760)
This commit is contained in:
@@ -2601,6 +2601,7 @@ matchstr({expr}, {pat} [, {start} [, {count}]])
|
|||||||
matchstrpos({expr}, {pat} [, {start} [, {count}]])
|
matchstrpos({expr}, {pat} [, {start} [, {count}]])
|
||||||
List {count}'th match of {pat} in {expr}
|
List {count}'th match of {pat} in {expr}
|
||||||
max({expr}) Number maximum value of items in {expr}
|
max({expr}) Number maximum value of items in {expr}
|
||||||
|
menu_info({name} [, {mode}]) Dict get menu item information
|
||||||
min({expr}) Number minimum value of items in {expr}
|
min({expr}) Number minimum value of items in {expr}
|
||||||
mkdir({name} [, {path} [, {prot}]])
|
mkdir({name} [, {path} [, {prot}]])
|
||||||
Number create directory {name}
|
Number create directory {name}
|
||||||
@@ -7124,6 +7125,7 @@ matchstrpos({expr}, {pat} [, {start} [, {count}]]) *matchstrpos()*
|
|||||||
Can also be used as a |method|: >
|
Can also be used as a |method|: >
|
||||||
GetText()->matchstrpos('word')
|
GetText()->matchstrpos('word')
|
||||||
<
|
<
|
||||||
|
|
||||||
*max()*
|
*max()*
|
||||||
max({expr}) Return the maximum value of all items in {expr}.
|
max({expr}) Return the maximum value of all items in {expr}.
|
||||||
{expr} can be a List or a Dictionary. For a Dictionary,
|
{expr} can be a List or a Dictionary. For a Dictionary,
|
||||||
@@ -7135,6 +7137,66 @@ max({expr}) Return the maximum value of all items in {expr}.
|
|||||||
Can also be used as a |method|: >
|
Can also be used as a |method|: >
|
||||||
mylist->max()
|
mylist->max()
|
||||||
|
|
||||||
|
|
||||||
|
menu_info({name} [, {mode}]) *menu_info()*
|
||||||
|
Return information about the specified menu {name} in
|
||||||
|
mode {mode}. The menu name should be specified without the
|
||||||
|
shortcut character ('&').
|
||||||
|
|
||||||
|
{mode} can be one of these strings:
|
||||||
|
"n" Normal
|
||||||
|
"v" Visual (including Select)
|
||||||
|
"o" Operator-pending
|
||||||
|
"i" Insert
|
||||||
|
"c" Cmd-line
|
||||||
|
"s" Select
|
||||||
|
"x" Visual
|
||||||
|
"t" Terminal-Job
|
||||||
|
"" Normal, Visual and Operator-pending
|
||||||
|
"!" Insert and Cmd-line
|
||||||
|
When {mode} is omitted, the modes for "" are used.
|
||||||
|
|
||||||
|
Returns a |Dictionary| containing the following items:
|
||||||
|
accel menu item accelerator text |menu-text|
|
||||||
|
display display name (name without '&')
|
||||||
|
enabled v:true if this menu item is enabled
|
||||||
|
Refer to |:menu-enable|
|
||||||
|
icon name of the icon file (for toolbar)
|
||||||
|
|toolbar-icon|
|
||||||
|
iconidx index of a built-in icon
|
||||||
|
modes modes for which the menu is defined. In
|
||||||
|
addition to the modes mentioned above, these
|
||||||
|
characters will be used:
|
||||||
|
" " Normal, Visual and Operator-pending
|
||||||
|
name menu item name.
|
||||||
|
noremenu v:true if the {rhs} of the menu item is not
|
||||||
|
remappable else v:false.
|
||||||
|
priority menu order priority |menu-priority|
|
||||||
|
rhs right-hand-side of the menu item. The returned
|
||||||
|
string has special characters translated like
|
||||||
|
in the output of the ":menu" command listing.
|
||||||
|
When the {rhs} of a menu item is empty, then
|
||||||
|
"<Nop>" is returned.
|
||||||
|
script v:true if script-local remapping of {rhs} is
|
||||||
|
allowed else v:false. See |:menu-script|.
|
||||||
|
shortcut shortcut key (character after '&' in
|
||||||
|
the menu name) |menu-shortcut|
|
||||||
|
silent v:true if the menu item is created
|
||||||
|
with <silent> argument |:menu-silent|
|
||||||
|
submenus |List| containing the names of
|
||||||
|
all the submenus. Present only if the menu
|
||||||
|
item has submenus.
|
||||||
|
|
||||||
|
Returns an empty dictionary if the menu item is not found.
|
||||||
|
|
||||||
|
Examples: >
|
||||||
|
:echo maparg('Edit.Cut')
|
||||||
|
:echo maparg('File.Save', 'n')
|
||||||
|
<
|
||||||
|
Can also be used as a |method|: >
|
||||||
|
GetMenuName()->maparg('v')
|
||||||
|
|
||||||
|
|
||||||
< *min()*
|
< *min()*
|
||||||
min({expr}) Return the minimum value of all items in {expr}.
|
min({expr}) Return the minimum value of all items in {expr}.
|
||||||
{expr} can be a List or a Dictionary. For a Dictionary,
|
{expr} can be a List or a Dictionary. For a Dictionary,
|
||||||
|
@@ -578,9 +578,11 @@ tooltips for menus. See |terminal-typing|.
|
|||||||
|
|
||||||
Special characters in a menu name:
|
Special characters in a menu name:
|
||||||
|
|
||||||
|
*menu-shortcut*
|
||||||
& The next character is the shortcut key. Make sure each
|
& The next character is the shortcut key. Make sure each
|
||||||
shortcut key is only used once in a (sub)menu. If you want to
|
shortcut key is only used once in a (sub)menu. If you want to
|
||||||
insert a literal "&" in the menu name use "&&".
|
insert a literal "&" in the menu name use "&&".
|
||||||
|
*menu-text*
|
||||||
<Tab> Separates the menu name from right-aligned text. This can be
|
<Tab> Separates the menu name from right-aligned text. This can be
|
||||||
used to show the equivalent typed command. The text "<Tab>"
|
used to show the equivalent typed command. The text "<Tab>"
|
||||||
can be used here for convenience. If you are using a real
|
can be used here for convenience. If you are using a real
|
||||||
@@ -954,7 +956,7 @@ item for the keyword under the cursor. The register "z" is used. >
|
|||||||
mappings, or put these lines in your gvimrc; "<C-R>" is CTRL-R, "<CR>" is
|
mappings, or put these lines in your gvimrc; "<C-R>" is CTRL-R, "<CR>" is
|
||||||
the <CR> key. |<>|)
|
the <CR> key. |<>|)
|
||||||
|
|
||||||
|
*tooltips* *menu-tips*
|
||||||
5.8 Tooltips & Menu tips
|
5.8 Tooltips & Menu tips
|
||||||
|
|
||||||
See section |42.4| in the user manual.
|
See section |42.4| in the user manual.
|
||||||
|
@@ -942,10 +942,11 @@ Window size and position: *window-size-functions*
|
|||||||
winsaveview() get view of current window
|
winsaveview() get view of current window
|
||||||
winrestview() restore saved view of current window
|
winrestview() restore saved view of current window
|
||||||
|
|
||||||
Mappings: *mapping-functions*
|
Mappings and Menus: *mapping-functions*
|
||||||
hasmapto() check if a mapping exists
|
hasmapto() check if a mapping exists
|
||||||
mapcheck() check if a matching mapping exists
|
mapcheck() check if a matching mapping exists
|
||||||
maparg() get rhs of a mapping
|
maparg() get rhs of a mapping
|
||||||
|
menu_info() get information about a menu item
|
||||||
wildmenumode() check if the wildmode is active
|
wildmenumode() check if the wildmode is active
|
||||||
|
|
||||||
Testing: *test-functions*
|
Testing: *test-functions*
|
||||||
|
@@ -646,6 +646,7 @@ static funcentry_T global_functions[] =
|
|||||||
{"matchstr", 2, 4, FEARG_1, ret_string, f_matchstr},
|
{"matchstr", 2, 4, FEARG_1, ret_string, f_matchstr},
|
||||||
{"matchstrpos", 2, 4, FEARG_1, ret_list_any, f_matchstrpos},
|
{"matchstrpos", 2, 4, FEARG_1, ret_list_any, f_matchstrpos},
|
||||||
{"max", 1, 1, FEARG_1, ret_any, f_max},
|
{"max", 1, 1, FEARG_1, ret_any, f_max},
|
||||||
|
{"menu_info", 1, 2, FEARG_1, ret_dict_any, f_menu_info},
|
||||||
{"min", 1, 1, FEARG_1, ret_any, f_min},
|
{"min", 1, 1, FEARG_1, ret_any, f_min},
|
||||||
{"mkdir", 1, 3, FEARG_1, ret_number, f_mkdir},
|
{"mkdir", 1, 3, FEARG_1, ret_number, f_mkdir},
|
||||||
{"mode", 0, 1, FEARG_1, ret_string, f_mode},
|
{"mode", 0, 1, FEARG_1, ret_string, f_mode},
|
||||||
@@ -2469,7 +2470,17 @@ f_feedkeys(typval_T *argvars, typval_T *rettv UNUSED)
|
|||||||
if (lowlevel)
|
if (lowlevel)
|
||||||
{
|
{
|
||||||
#ifdef USE_INPUT_BUF
|
#ifdef USE_INPUT_BUF
|
||||||
add_to_input_buf(keys, (int)STRLEN(keys));
|
int idx;
|
||||||
|
int len = (int)STRLEN(keys);
|
||||||
|
|
||||||
|
for (idx = 0; idx < len; ++idx)
|
||||||
|
{
|
||||||
|
// if a CTRL-C was typed, set got_int
|
||||||
|
if (keys[idx] == 3 && ctrl_c_interrupts)
|
||||||
|
got_int = TRUE;
|
||||||
|
else
|
||||||
|
add_to_input_buf(keys + idx, 1);
|
||||||
|
}
|
||||||
#else
|
#else
|
||||||
emsg(_("E980: lowlevel input not supported"));
|
emsg(_("E980: lowlevel input not supported"));
|
||||||
#endif
|
#endif
|
||||||
|
269
src/menu.c
269
src/menu.c
@@ -1684,6 +1684,49 @@ get_menu_cmd_modes(
|
|||||||
return modes;
|
return modes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return the string representation of the menu modes. Does the opposite
|
||||||
|
* of get_menu_cmd_modes().
|
||||||
|
*/
|
||||||
|
static char_u *
|
||||||
|
get_menu_mode_str(int modes)
|
||||||
|
{
|
||||||
|
if ((modes & (MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE |
|
||||||
|
MENU_VISUAL_MODE | MENU_SELECT_MODE | MENU_OP_PENDING_MODE))
|
||||||
|
== (MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE |
|
||||||
|
MENU_VISUAL_MODE | MENU_SELECT_MODE | MENU_OP_PENDING_MODE))
|
||||||
|
return (char_u *)"a";
|
||||||
|
if ((modes & (MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE |
|
||||||
|
MENU_OP_PENDING_MODE))
|
||||||
|
== (MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE |
|
||||||
|
MENU_OP_PENDING_MODE))
|
||||||
|
return (char_u *)" ";
|
||||||
|
if ((modes & (MENU_INSERT_MODE | MENU_CMDLINE_MODE))
|
||||||
|
== (MENU_INSERT_MODE | MENU_CMDLINE_MODE))
|
||||||
|
return (char_u *)"!";
|
||||||
|
if ((modes & (MENU_VISUAL_MODE | MENU_SELECT_MODE))
|
||||||
|
== (MENU_VISUAL_MODE | MENU_SELECT_MODE))
|
||||||
|
return (char_u *)"v";
|
||||||
|
if (modes & MENU_VISUAL_MODE)
|
||||||
|
return (char_u *)"x";
|
||||||
|
if (modes & MENU_SELECT_MODE)
|
||||||
|
return (char_u *)"s";
|
||||||
|
if (modes & MENU_OP_PENDING_MODE)
|
||||||
|
return (char_u *)"o";
|
||||||
|
if (modes & MENU_INSERT_MODE)
|
||||||
|
return (char_u *)"i";
|
||||||
|
if (modes & MENU_TERMINAL_MODE)
|
||||||
|
return (char_u *)"tl";
|
||||||
|
if (modes & MENU_CMDLINE_MODE)
|
||||||
|
return (char_u *)"c";
|
||||||
|
if (modes & MENU_NORMAL_MODE)
|
||||||
|
return (char_u *)"n";
|
||||||
|
if (modes & MENU_TIP_MODE)
|
||||||
|
return (char_u *)"t";
|
||||||
|
|
||||||
|
return (char_u *)"";
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Modify a menu name starting with "PopUp" to include the mode character.
|
* Modify a menu name starting with "PopUp" to include the mode character.
|
||||||
* Returns the name in allocated memory (NULL for failure).
|
* Returns the name in allocated memory (NULL for failure).
|
||||||
@@ -2393,40 +2436,21 @@ execute_menu(exarg_T *eap, vimmenu_T *menu, int mode_idx)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Given a menu descriptor, e.g. "File.New", find it in the menu hierarchy and
|
* Lookup a menu by the descriptor name e.g. "File.New"
|
||||||
* execute it.
|
* Returns NULL if the menu is not found
|
||||||
*/
|
*/
|
||||||
void
|
static vimmenu_T *
|
||||||
ex_emenu(exarg_T *eap)
|
menu_getbyname(char_u *name_arg)
|
||||||
{
|
{
|
||||||
vimmenu_T *menu;
|
|
||||||
char_u *name;
|
char_u *name;
|
||||||
char_u *saved_name;
|
char_u *saved_name;
|
||||||
char_u *arg = eap->arg;
|
vimmenu_T *menu;
|
||||||
char_u *p;
|
char_u *p;
|
||||||
int gave_emsg = FALSE;
|
int gave_emsg = FALSE;
|
||||||
int mode_idx = -1;
|
|
||||||
|
|
||||||
if (arg[0] && VIM_ISWHITE(arg[1]))
|
saved_name = vim_strsave(name_arg);
|
||||||
{
|
|
||||||
switch (arg[0])
|
|
||||||
{
|
|
||||||
case 'n': mode_idx = MENU_INDEX_NORMAL; break;
|
|
||||||
case 'v': mode_idx = MENU_INDEX_VISUAL; break;
|
|
||||||
case 's': mode_idx = MENU_INDEX_SELECT; break;
|
|
||||||
case 'o': mode_idx = MENU_INDEX_OP_PENDING; break;
|
|
||||||
case 't': mode_idx = MENU_INDEX_TERMINAL; break;
|
|
||||||
case 'i': mode_idx = MENU_INDEX_INSERT; break;
|
|
||||||
case 'c': mode_idx = MENU_INDEX_CMDLINE; break;
|
|
||||||
default: semsg(_(e_invarg2), arg);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
arg = skipwhite(arg + 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
saved_name = vim_strsave(arg);
|
|
||||||
if (saved_name == NULL)
|
if (saved_name == NULL)
|
||||||
return;
|
return NULL;
|
||||||
|
|
||||||
menu = *get_root_menu(saved_name);
|
menu = *get_root_menu(saved_name);
|
||||||
name = saved_name;
|
name = saved_name;
|
||||||
@@ -2463,10 +2487,45 @@ ex_emenu(exarg_T *eap)
|
|||||||
if (menu == NULL)
|
if (menu == NULL)
|
||||||
{
|
{
|
||||||
if (!gave_emsg)
|
if (!gave_emsg)
|
||||||
semsg(_("E334: Menu not found: %s"), arg);
|
semsg(_("E334: Menu not found: %s"), name_arg);
|
||||||
return;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return menu;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Given a menu descriptor, e.g. "File.New", find it in the menu hierarchy and
|
||||||
|
* execute it.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
ex_emenu(exarg_T *eap)
|
||||||
|
{
|
||||||
|
vimmenu_T *menu;
|
||||||
|
char_u *arg = eap->arg;
|
||||||
|
int mode_idx = -1;
|
||||||
|
|
||||||
|
if (arg[0] && VIM_ISWHITE(arg[1]))
|
||||||
|
{
|
||||||
|
switch (arg[0])
|
||||||
|
{
|
||||||
|
case 'n': mode_idx = MENU_INDEX_NORMAL; break;
|
||||||
|
case 'v': mode_idx = MENU_INDEX_VISUAL; break;
|
||||||
|
case 's': mode_idx = MENU_INDEX_SELECT; break;
|
||||||
|
case 'o': mode_idx = MENU_INDEX_OP_PENDING; break;
|
||||||
|
case 't': mode_idx = MENU_INDEX_TERMINAL; break;
|
||||||
|
case 'i': mode_idx = MENU_INDEX_INSERT; break;
|
||||||
|
case 'c': mode_idx = MENU_INDEX_CMDLINE; break;
|
||||||
|
default: semsg(_(e_invarg2), arg);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
arg = skipwhite(arg + 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
menu = menu_getbyname(arg);
|
||||||
|
if (menu == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
// Found the menu, so execute.
|
// Found the menu, so execute.
|
||||||
execute_menu(eap, menu, mode_idx);
|
execute_menu(eap, menu, mode_idx);
|
||||||
}
|
}
|
||||||
@@ -2773,4 +2832,158 @@ menu_translate_tab_and_shift(char_u *arg_start)
|
|||||||
return arg;
|
return arg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get the information about a menu item in mode 'which'
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
menuitem_getinfo(vimmenu_T *menu, int modes, dict_T *dict)
|
||||||
|
{
|
||||||
|
int status;
|
||||||
|
|
||||||
|
if (menu_is_tearoff(menu->dname)) // skip tearoff menu item
|
||||||
|
return OK;
|
||||||
|
|
||||||
|
status = dict_add_string(dict, "name", menu->name);
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_string(dict, "display", menu->dname);
|
||||||
|
if (status == OK && menu->actext != NULL)
|
||||||
|
status = dict_add_string(dict, "accel", menu->actext);
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_number(dict, "priority", menu->priority);
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_string(dict, "modes",
|
||||||
|
get_menu_mode_str(menu->modes));
|
||||||
|
#ifdef FEAT_TOOLBAR
|
||||||
|
if (status == OK && menu->iconfile != NULL)
|
||||||
|
status = dict_add_string(dict, "icon", menu->iconfile);
|
||||||
|
if (status == OK && menu->iconidx >= 0)
|
||||||
|
status = dict_add_number(dict, "iconidx", menu->iconidx);
|
||||||
|
#endif
|
||||||
|
if (status == OK)
|
||||||
|
{
|
||||||
|
char_u buf[NUMBUFLEN];
|
||||||
|
|
||||||
|
if (has_mbyte)
|
||||||
|
buf[utf_char2bytes(menu->mnemonic, buf)] = NUL;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
buf[0] = (char_u)menu->mnemonic;
|
||||||
|
buf[1] = NUL;
|
||||||
|
}
|
||||||
|
status = dict_add_string(dict, "shortcut", buf);
|
||||||
|
}
|
||||||
|
if (status == OK && menu->children == NULL)
|
||||||
|
{
|
||||||
|
int bit;
|
||||||
|
|
||||||
|
// Get the first mode in which the menu is available
|
||||||
|
for (bit = 0; (bit < MENU_MODES) && !((1 << bit) & modes); bit++)
|
||||||
|
;
|
||||||
|
if (menu->strings[bit] != NULL)
|
||||||
|
status = dict_add_string(dict, "rhs",
|
||||||
|
*menu->strings[bit] == NUL ?
|
||||||
|
vim_strsave((char_u *)"<Nop>") :
|
||||||
|
str2special_save(menu->strings[bit], FALSE));
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_bool(dict, "noremenu",
|
||||||
|
menu->noremap[bit] == REMAP_NONE);
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_bool(dict, "script",
|
||||||
|
menu->noremap[bit] == REMAP_SCRIPT);
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_bool(dict, "silent", menu->silent[bit]);
|
||||||
|
if (status == OK)
|
||||||
|
status = dict_add_bool(dict, "enabled",
|
||||||
|
((menu->enabled & (1 << bit)) != 0));
|
||||||
|
}
|
||||||
|
// If there are submenus, add all the submenu display names
|
||||||
|
if (status == OK && menu->children != NULL)
|
||||||
|
{
|
||||||
|
list_T *l = list_alloc();
|
||||||
|
vimmenu_T *child;
|
||||||
|
|
||||||
|
if (l == NULL)
|
||||||
|
return FAIL;
|
||||||
|
|
||||||
|
dict_add_list(dict, "submenus", l);
|
||||||
|
child = menu->children;
|
||||||
|
while (child)
|
||||||
|
{
|
||||||
|
if (!menu_is_tearoff(child->dname)) // skip tearoff menu
|
||||||
|
list_append_string(l, child->dname, -1);
|
||||||
|
child = child->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* "menu_info()" function
|
||||||
|
* Return information about a menu (including all the child menus)
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
f_menu_info(typval_T *argvars, typval_T *rettv)
|
||||||
|
{
|
||||||
|
char_u *menu_name;
|
||||||
|
char_u *which;
|
||||||
|
int modes;
|
||||||
|
char_u *saved_name;
|
||||||
|
char_u *name;
|
||||||
|
vimmenu_T *menu;
|
||||||
|
dict_T *retdict;
|
||||||
|
|
||||||
|
if (rettv_dict_alloc(rettv) != OK)
|
||||||
|
return;
|
||||||
|
retdict = rettv->vval.v_dict;
|
||||||
|
|
||||||
|
menu_name = tv_get_string_chk(&argvars[0]);
|
||||||
|
if (menu_name == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
// menu mode
|
||||||
|
if (argvars[1].v_type != VAR_UNKNOWN)
|
||||||
|
which = tv_get_string_chk(&argvars[1]);
|
||||||
|
else
|
||||||
|
which = (char_u *)""; // Default is modes for "menu"
|
||||||
|
if (which == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
modes = get_menu_cmd_modes(which, *which == '!', NULL, NULL);
|
||||||
|
|
||||||
|
// Locate the specified menu or menu item
|
||||||
|
menu = *get_root_menu(menu_name);
|
||||||
|
saved_name = vim_strsave(menu_name);
|
||||||
|
if (saved_name == NULL)
|
||||||
|
return;
|
||||||
|
if (*saved_name != NUL)
|
||||||
|
{
|
||||||
|
char_u *p;
|
||||||
|
|
||||||
|
name = saved_name;
|
||||||
|
while (*name)
|
||||||
|
{
|
||||||
|
// Find in the menu hierarchy
|
||||||
|
p = menu_name_skip(name);
|
||||||
|
while (menu != NULL)
|
||||||
|
{
|
||||||
|
if (menu_name_equal(name, menu))
|
||||||
|
break;
|
||||||
|
menu = menu->next;
|
||||||
|
}
|
||||||
|
if (menu == NULL || *p == NUL)
|
||||||
|
break;
|
||||||
|
menu = menu->children;
|
||||||
|
name = p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vim_free(saved_name);
|
||||||
|
|
||||||
|
if (menu == NULL) // specified menu not found
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (menu->modes & modes)
|
||||||
|
menuitem_getinfo(menu, modes, retdict);
|
||||||
|
}
|
||||||
|
|
||||||
#endif // FEAT_MENU
|
#endif // FEAT_MENU
|
||||||
|
@@ -23,4 +23,5 @@ void ex_emenu(exarg_T *eap);
|
|||||||
void winbar_click(win_T *wp, int col);
|
void winbar_click(win_T *wp, int col);
|
||||||
vimmenu_T *gui_find_menu(char_u *path_name);
|
vimmenu_T *gui_find_menu(char_u *path_name);
|
||||||
void ex_menutranslate(exarg_T *eap);
|
void ex_menutranslate(exarg_T *eap);
|
||||||
|
void f_menu_info(typval_T *argvars, typval_T *rettv);
|
||||||
/* vim: set ft=c : */
|
/* vim: set ft=c : */
|
||||||
|
@@ -89,6 +89,35 @@ func Test_menu_commands()
|
|||||||
unlet g:did_menu
|
unlet g:did_menu
|
||||||
endfun
|
endfun
|
||||||
|
|
||||||
|
" Test various menu related errors
|
||||||
|
func Test_menu_errors()
|
||||||
|
menu Test.Foo :version<CR>
|
||||||
|
|
||||||
|
" Error cases
|
||||||
|
call assert_fails('menu .Test.Foo :ls<CR>', 'E475:')
|
||||||
|
call assert_fails('menu Test. :ls<CR>', 'E330:')
|
||||||
|
call assert_fails('menu Foo. :ls<CR>', 'E331:')
|
||||||
|
call assert_fails('unmenu Test.Foo abc', 'E488:')
|
||||||
|
call assert_fails('menu <Tab>:ls :ls<CR>', 'E792:')
|
||||||
|
call assert_fails('menu Test.<Tab>:ls :ls<CR>', 'E792:')
|
||||||
|
call assert_fails('menu Test.Foo.Bar :ls<CR>', 'E327:')
|
||||||
|
call assert_fails('menu Test.-Sep-.Baz :ls<CR>', 'E332:')
|
||||||
|
call assert_fails('menu Foo.Bar.--.Baz :ls<CR>', 'E332:')
|
||||||
|
call assert_fails('menu disable Test.Foo.Bar', 'E327:')
|
||||||
|
call assert_fails('menu disable T.Foo', 'E329:')
|
||||||
|
call assert_fails('unmenu Test.Foo.Bar', 'E327:')
|
||||||
|
call assert_fails('cunmenu Test.Foo', 'E328:')
|
||||||
|
call assert_fails('unmenu Test.Bar', 'E329:')
|
||||||
|
call assert_fails('menu Test.Foo.Bar', 'E327:')
|
||||||
|
call assert_fails('cmenu Test.Foo', 'E328:')
|
||||||
|
call assert_fails('emenu x Test.Foo', 'E475:')
|
||||||
|
call assert_fails('emenu Test.Foo.Bar', 'E334:')
|
||||||
|
call assert_fails('menutranslate Test', 'E474:')
|
||||||
|
|
||||||
|
silent! unmenu Foo
|
||||||
|
unmenu Test
|
||||||
|
endfunc
|
||||||
|
|
||||||
" Test for menu item completion in command line
|
" Test for menu item completion in command line
|
||||||
func Test_menu_expand()
|
func Test_menu_expand()
|
||||||
" Create the menu itmes for test
|
" Create the menu itmes for test
|
||||||
@@ -119,8 +148,336 @@ func Test_menu_expand()
|
|||||||
\ "\<C-A>\<C-B>\"\<CR>", 'xt')
|
\ "\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
call assert_equal('"emenu Buffers. Xmenu.', @:)
|
call assert_equal('"emenu Buffers. Xmenu.', @:)
|
||||||
|
|
||||||
|
" Test for expanding only submenus
|
||||||
|
call feedkeys(":popup Xmenu.\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
|
call assert_equal('"popup Xmenu.A1 A2 A3 A4', @:)
|
||||||
|
|
||||||
|
" Test for expanding menus after enable/disable
|
||||||
|
call feedkeys(":menu enable Xmenu.\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
|
call assert_equal('"menu enable Xmenu.A1. A2. A3. A4.', @:)
|
||||||
|
call feedkeys(":menu disable Xmenu.\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
|
call assert_equal('"menu disable Xmenu.A1. A2. A3. A4.', @:)
|
||||||
|
|
||||||
|
" Test for expanding non-existing menu path
|
||||||
|
call feedkeys(":menu xyz.\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
|
call assert_equal('"menu xyz.', @:)
|
||||||
|
call feedkeys(":menu Xmenu.A1.A1B1.xyz.\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
|
call assert_equal('"menu Xmenu.A1.A1B1.xyz.', @:)
|
||||||
|
|
||||||
set wildmenu&
|
set wildmenu&
|
||||||
unmenu Xmenu
|
unmenu Xmenu
|
||||||
|
|
||||||
|
" Test for expanding popup menus with some hidden items
|
||||||
|
menu Xmenu.foo.A1 a1
|
||||||
|
menu Xmenu.]bar bar
|
||||||
|
menu Xmenu.]baz.B1 b1
|
||||||
|
menu Xmenu.-sep- :
|
||||||
|
call feedkeys(":popup Xmenu.\<C-A>\<C-B>\"\<CR>", 'xt')
|
||||||
|
call assert_equal('"popup Xmenu.foo', @:)
|
||||||
|
unmenu Xmenu
|
||||||
|
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for the menu_info() function
|
||||||
|
func Test_menu_info()
|
||||||
|
" Define menus with various attributes
|
||||||
|
10nnoremenu 10.10 T&est.F&oo :echo 'foo'<CR>
|
||||||
|
10nmenu <silent> 10.20 T&est.B&ar<Tab>:bar :echo 'bar'<CR>
|
||||||
|
10nmenu <script> 10.30.5 T&est.Ba&z.Qu&x :echo 'qux'<CR>
|
||||||
|
|
||||||
|
let d = #{name: "B&ar\t:bar", display: 'Bar', modes: 'n', shortcut: 'a',
|
||||||
|
\ accel: ':bar', priority: 20, enabled: v:true, silent: v:true,
|
||||||
|
\ noremenu: v:false, script: v:false, rhs: ":echo 'bar'<CR>"}
|
||||||
|
call assert_equal(d, menu_info('Test.Bar'))
|
||||||
|
|
||||||
|
let d = #{name: 'Ba&z', display: 'Baz', modes: 'n', shortcut: 'z',
|
||||||
|
\ priority: 30, submenus: ['Qux']}
|
||||||
|
call assert_equal(d, menu_info('Test.Baz'))
|
||||||
|
|
||||||
|
let d = #{name: 'T&est', display: 'Test', modes: 'n', shortcut: 'e',
|
||||||
|
\ priority: 10, submenus: ['Foo', 'Bar', 'Baz']}
|
||||||
|
call assert_equal(d, menu_info('Test'))
|
||||||
|
call assert_equal({}, menu_info('Test.Dummy'))
|
||||||
|
call assert_equal({}, menu_info('Dummy'))
|
||||||
|
|
||||||
|
nmenu disable Test.Foo
|
||||||
|
call assert_equal(v:false, menu_info('Test.Foo').enabled)
|
||||||
|
nmenu enable Test.Foo
|
||||||
|
call assert_equal(v:true, menu_info('Test.Foo').enabled)
|
||||||
|
|
||||||
|
call assert_equal(menu_info('Test.Foo'), menu_info('Test.Foo', ''))
|
||||||
|
nmenu Test.abc <Nop>
|
||||||
|
call assert_equal('<Nop>', menu_info('Test.abc').rhs)
|
||||||
|
call assert_fails('call menu_info([])', 'E730:')
|
||||||
|
nunmenu Test
|
||||||
|
|
||||||
|
" Test for defining menus in different modes
|
||||||
|
menu Test.menu :menu<CR>
|
||||||
|
menu! Test.menu! :menu!<CR>
|
||||||
|
amenu Test.amenu :amenu<CR>
|
||||||
|
nmenu Test.nmenu :nmenu<CR>
|
||||||
|
omenu Test.omenu :omenu<CR>
|
||||||
|
vmenu Test.vmenu :vmenu<CR>
|
||||||
|
xmenu Test.xmenu :xmenu<CR>
|
||||||
|
smenu Test.smenu :smenu<CR>
|
||||||
|
imenu <silent> <script> Test.imenu :imenu<CR>
|
||||||
|
cmenu Test.cmenu :cmenu<CR>
|
||||||
|
tlmenu Test.tlmenu :tlmenu<CR>
|
||||||
|
tmenu Test.nmenu Normal mode menu
|
||||||
|
tmenu Test.omenu Op-pending mode menu
|
||||||
|
noremenu Test.noremenu :noremenu<CR>
|
||||||
|
noremenu! Test.noremenu! :noremenu!<CR>
|
||||||
|
anoremenu Test.anoremenu :anoremenu<CR>
|
||||||
|
nnoremenu Test.nnoremenu :nnoremenu<CR>
|
||||||
|
onoremenu Test.onoremenu :onoremenu<CR>
|
||||||
|
vnoremenu Test.vnoremenu :vnoremenu<CR>
|
||||||
|
xnoremenu Test.xnoremenu :xnoremenu<CR>
|
||||||
|
snoremenu Test.snoremenu :snoremenu<CR>
|
||||||
|
inoremenu <silent> Test.inoremenu :inoremenu<CR>
|
||||||
|
cnoremenu Test.cnoremenu :cnoremenu<CR>
|
||||||
|
tlnoremenu Test.tlnoremenu :tlnoremenu<CR>
|
||||||
|
call assert_equal(#{name: 'menu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'menu', modes: ' ', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ":menu<CR>", noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.menu'))
|
||||||
|
call assert_equal(#{name: 'menu!', priority: 500, shortcut: '',
|
||||||
|
\ display: 'menu!', modes: '!', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ":menu!<CR>", noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.menu!', '!'))
|
||||||
|
call assert_equal(#{name: 'amenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'amenu', modes: 'a', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ":amenu<CR>", noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.amenu', 'a'))
|
||||||
|
call assert_equal(#{name: 'nmenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'nmenu', modes: 'n', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':nmenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.nmenu', 'n'))
|
||||||
|
call assert_equal(#{name: 'omenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'omenu', modes: 'o', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':omenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.omenu', 'o'))
|
||||||
|
call assert_equal(#{name: 'vmenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'vmenu', modes: 'v', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':vmenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.vmenu', 'v'))
|
||||||
|
call assert_equal(#{name: 'xmenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'xmenu', modes: 'x', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':xmenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.xmenu', 'x'))
|
||||||
|
call assert_equal(#{name: 'smenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'smenu', modes: 's', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':smenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.smenu', 's'))
|
||||||
|
call assert_equal(#{name: 'imenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'imenu', modes: 'i', enabled: v:true, silent: v:true,
|
||||||
|
\ rhs: ':imenu<CR>', noremenu: v:false, script: v:true},
|
||||||
|
\ menu_info('Test.imenu', 'i'))
|
||||||
|
call assert_equal(#{ name: 'cmenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'cmenu', modes: 'c', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':cmenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.cmenu', 'c'))
|
||||||
|
call assert_equal(#{name: 'tlmenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'tlmenu', modes: 'tl', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':tlmenu<CR>', noremenu: v:false, script: v:false},
|
||||||
|
\ menu_info('Test.tlmenu', 'tl'))
|
||||||
|
call assert_equal(#{name: 'noremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'noremenu', modes: ' ', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ":noremenu<CR>", noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.noremenu'))
|
||||||
|
call assert_equal(#{name: 'noremenu!', priority: 500, shortcut: '',
|
||||||
|
\ display: 'noremenu!', modes: '!', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ":noremenu!<CR>", noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.noremenu!', '!'))
|
||||||
|
call assert_equal(#{name: 'anoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'anoremenu', modes: 'a', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ":anoremenu<CR>", noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.anoremenu', 'a'))
|
||||||
|
call assert_equal(#{name: 'nnoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'nnoremenu', modes: 'n', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':nnoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.nnoremenu', 'n'))
|
||||||
|
call assert_equal(#{name: 'onoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'onoremenu', modes: 'o', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':onoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.onoremenu', 'o'))
|
||||||
|
call assert_equal(#{name: 'vnoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'vnoremenu', modes: 'v', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':vnoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.vnoremenu', 'v'))
|
||||||
|
call assert_equal(#{name: 'xnoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'xnoremenu', modes: 'x', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':xnoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.xnoremenu', 'x'))
|
||||||
|
call assert_equal(#{name: 'snoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'snoremenu', modes: 's', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':snoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.snoremenu', 's'))
|
||||||
|
call assert_equal(#{name: 'inoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'inoremenu', modes: 'i', enabled: v:true, silent: v:true,
|
||||||
|
\ rhs: ':inoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.inoremenu', 'i'))
|
||||||
|
call assert_equal(#{ name: 'cnoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'cnoremenu', modes: 'c', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':cnoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.cnoremenu', 'c'))
|
||||||
|
call assert_equal(#{name: 'tlnoremenu', priority: 500, shortcut: '',
|
||||||
|
\ display: 'tlnoremenu', modes: 'tl', enabled: v:true, silent: v:false,
|
||||||
|
\ rhs: ':tlnoremenu<CR>', noremenu: v:true, script: v:false},
|
||||||
|
\ menu_info('Test.tlnoremenu', 'tl'))
|
||||||
|
aunmenu Test
|
||||||
|
tlunmenu Test
|
||||||
|
call assert_equal({}, menu_info('Test'))
|
||||||
|
call assert_equal({}, menu_info('Test', '!'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'a'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'n'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'o'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'v'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'x'))
|
||||||
|
call assert_equal({}, menu_info('Test', 's'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'i'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'c'))
|
||||||
|
call assert_equal({}, menu_info('Test', 't'))
|
||||||
|
call assert_equal({}, menu_info('Test', 'tl'))
|
||||||
|
|
||||||
|
amenu Test.amenu :amenu<CR>
|
||||||
|
call assert_equal(':amenu<CR>', menu_info('Test.amenu', '').rhs)
|
||||||
|
call assert_equal('<C-\><C-O>:amenu<CR>', menu_info('Test.amenu', '!').rhs)
|
||||||
|
call assert_equal(':amenu<CR>', menu_info('Test.amenu', 'n').rhs)
|
||||||
|
call assert_equal('<C-C>:amenu<CR><C-\><C-G>',
|
||||||
|
\ menu_info('Test.amenu', 'o').rhs)
|
||||||
|
call assert_equal('<C-C>:amenu<CR><C-\><C-G>',
|
||||||
|
\ menu_info('Test.amenu', 'v').rhs)
|
||||||
|
call assert_equal('<C-C>:amenu<CR><C-\><C-G>',
|
||||||
|
\ menu_info('Test.amenu', 'x').rhs)
|
||||||
|
call assert_equal('<C-C>:amenu<CR><C-\><C-G>',
|
||||||
|
\ menu_info('Test.amenu', 's').rhs)
|
||||||
|
call assert_equal('<C-\><C-O>:amenu<CR>', menu_info('Test.amenu', 'i').rhs)
|
||||||
|
call assert_equal('<C-C>:amenu<CR><C-\><C-G>',
|
||||||
|
\ menu_info('Test.amenu', 'c').rhs)
|
||||||
|
aunmenu Test.amenu
|
||||||
|
|
||||||
|
" Test for hidden menus
|
||||||
|
menu ]Test.menu :menu<CR>
|
||||||
|
call assert_equal(#{name: ']Test', display: ']Test', priority: 500,
|
||||||
|
\ shortcut: '', modes: ' ', submenus: ['menu']},
|
||||||
|
\ menu_info(']Test'))
|
||||||
|
unmenu ]Test
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for <special> keyword in a menu with 'cpo' containing '<'
|
||||||
|
func Test_menu_special()
|
||||||
|
new
|
||||||
|
set cpo+=<
|
||||||
|
nmenu Test.Sign am<Tab>n<Esc>
|
||||||
|
call feedkeys(":emenu n Test.Sign\<CR>", 'x')
|
||||||
|
call assert_equal("m<Tab>n<Esc>", getline(1))
|
||||||
|
nunmenu Test.Sign
|
||||||
|
nmenu <special> Test.Sign am<Tab>n<Esc>
|
||||||
|
call setline(1, '')
|
||||||
|
call feedkeys(":emenu n Test.Sign\<CR>", 'x')
|
||||||
|
call assert_equal("m\tn", getline(1))
|
||||||
|
set cpo-=<
|
||||||
|
close!
|
||||||
|
nunmenu Test.Sign
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for "icon=filname" in a toolbar
|
||||||
|
func Test_menu_icon()
|
||||||
|
CheckFeature toolbar
|
||||||
|
nmenu icon=myicon.xpm Toolbar.Foo :echo "Foo"<CR>
|
||||||
|
call assert_equal('myicon.xpm', "Toolbar.Foo"->menu_info().icon)
|
||||||
|
nunmenu Toolbar.Foo
|
||||||
|
|
||||||
|
" Test for using the builtin icon
|
||||||
|
amenu ToolBar.BuiltIn22 :echo "BuiltIn22"<CR>
|
||||||
|
call assert_equal(#{name: 'BuiltIn22', display: 'BuiltIn22',
|
||||||
|
\ enabled: v:true, shortcut: '', modes: 'a', script: v:false,
|
||||||
|
\ iconidx: 22, priority: 500, silent: v:false,
|
||||||
|
\ rhs: ':echo "BuiltIn22"<CR>', noremenu: v:false},
|
||||||
|
\ menu_info("ToolBar.BuiltIn22"))
|
||||||
|
aunmenu ToolBar.BuiltIn22
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for ":emenu" command in different modes
|
||||||
|
func Test_emenu_cmd()
|
||||||
|
new
|
||||||
|
xmenu Test.foo rx
|
||||||
|
call setline(1, ['aaaa', 'bbbb'])
|
||||||
|
normal ggVj
|
||||||
|
%emenu Test.foo
|
||||||
|
call assert_equal(['xxxx', 'xxxx'], getline(1, 2))
|
||||||
|
call setline(1, ['aaaa', 'bbbb'])
|
||||||
|
exe "normal ggVj\<Esc>"
|
||||||
|
%emenu Test.foo
|
||||||
|
call assert_equal(['xxxx', 'xxxx'], getline(1, 2))
|
||||||
|
call setline(1, ['aaaa', 'bbbb'])
|
||||||
|
exe "normal ggV\<Esc>"
|
||||||
|
2emenu Test.foo
|
||||||
|
call assert_equal(['aaaa', 'xxxx'], getline(1, 2))
|
||||||
|
xunmenu Test.foo
|
||||||
|
close!
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for PopUp menus
|
||||||
|
func Test_popup_menu()
|
||||||
|
20menu PopUp.foo :echo 'foo'<CR>
|
||||||
|
20menu PopUp.bar :echo 'bar'<CR>
|
||||||
|
call assert_equal(#{name: 'PopUp', display: 'PopUp', priority: 20,
|
||||||
|
\ shortcut: '', modes: ' ', submenus: ['foo', 'bar']},
|
||||||
|
\ menu_info('PopUp'))
|
||||||
|
menu disable PopUp.bar
|
||||||
|
call assert_equal(v:true, "PopUp.foo"->menu_info().enabled)
|
||||||
|
call assert_equal(v:false, "PopUp.bar"->menu_info().enabled)
|
||||||
|
menu enable PopUp.bar
|
||||||
|
call assert_equal(v:true, "PopUp.bar"->menu_info().enabled)
|
||||||
|
unmenu PopUp
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for listing the menus using the :menu command
|
||||||
|
func Test_show_menus()
|
||||||
|
" In the GUI, tear-off menu items are present in the output below
|
||||||
|
" So skip this test
|
||||||
|
CheckNotGui
|
||||||
|
aunmenu *
|
||||||
|
call assert_equal(['--- Menus ---'], split(execute('menu'), "\n"))
|
||||||
|
nmenu <script> 200.10 Test.nmenu1 :nmenu1<CR>
|
||||||
|
nmenu 200.20 Test.nmenu2 :nmenu2<CR>
|
||||||
|
nnoremenu 200.30 Test.nmenu3 :nmenu3<CR>
|
||||||
|
nmenu 200.40 Test.nmenu4 :nmenu4<CR>
|
||||||
|
nmenu 200.50 disable Test.nmenu4
|
||||||
|
let exp =<< trim [TEXT]
|
||||||
|
--- Menus ---
|
||||||
|
200 Test
|
||||||
|
10 nmenu1
|
||||||
|
n& :nmenu1<CR>
|
||||||
|
20 nmenu2
|
||||||
|
n :nmenu2<CR>
|
||||||
|
30 nmenu3
|
||||||
|
n* :nmenu3<CR>
|
||||||
|
40 nmenu4
|
||||||
|
n - :nmenu4<CR>
|
||||||
|
[TEXT]
|
||||||
|
call assert_equal(exp, split(execute('nmenu'), "\n"))
|
||||||
|
nunmenu Test
|
||||||
|
endfunc
|
||||||
|
|
||||||
|
" Test for menu tips
|
||||||
|
func Test_tmenu()
|
||||||
|
tunmenu *
|
||||||
|
call assert_equal(['--- Menus ---'], split(execute('tmenu'), "\n"))
|
||||||
|
tmenu Test.nmenu1 nmenu1
|
||||||
|
tmenu Test.nmenu2.sub1 nmenu2.sub1
|
||||||
|
let exp =<< trim [TEXT]
|
||||||
|
--- Menus ---
|
||||||
|
500 Test
|
||||||
|
500 nmenu1
|
||||||
|
t - nmenu1
|
||||||
|
500 nmenu2
|
||||||
|
500 sub1
|
||||||
|
t - nmenu2.sub1
|
||||||
|
[TEXT]
|
||||||
|
call assert_equal(exp, split(execute('tmenu'), "\n"))
|
||||||
|
tunmenu Test
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
" vim: shiftwidth=2 sts=2 expandtab
|
" vim: shiftwidth=2 sts=2 expandtab
|
||||||
|
@@ -851,6 +851,12 @@ func Test_popup_command()
|
|||||||
CheckScreendump
|
CheckScreendump
|
||||||
CheckFeature menu
|
CheckFeature menu
|
||||||
|
|
||||||
|
menu Test.Foo Foo
|
||||||
|
call assert_fails('popup Test.Foo', 'E336:')
|
||||||
|
call assert_fails('popup Test.Foo.X', 'E327:')
|
||||||
|
call assert_fails('popup Foo', 'E337:')
|
||||||
|
unmenu Test.Foo
|
||||||
|
|
||||||
let lines =<< trim END
|
let lines =<< trim END
|
||||||
one two three four five
|
one two three four five
|
||||||
and one two Xthree four five
|
and one two Xthree four five
|
||||||
|
@@ -979,6 +979,39 @@ func Test_term_mouse_middle_click_in_cmdline_to_paste()
|
|||||||
call test_override('no_query_mouse', 0)
|
call test_override('no_query_mouse', 0)
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
" Test for displaying the popup menu using the right mouse click
|
||||||
|
func Test_mouse_popup_menu()
|
||||||
|
CheckFeature menu
|
||||||
|
new
|
||||||
|
call setline(1, 'popup menu test')
|
||||||
|
let save_mouse = &mouse
|
||||||
|
let save_term = &term
|
||||||
|
let save_ttymouse = &ttymouse
|
||||||
|
let save_mousemodel = &mousemodel
|
||||||
|
call test_override('no_query_mouse', 1)
|
||||||
|
set mouse=a term=xterm mousemodel=popup
|
||||||
|
|
||||||
|
menu PopUp.foo :let g:menustr = 'foo'<CR>
|
||||||
|
menu PopUp.bar :let g:menustr = 'bar'<CR>
|
||||||
|
menu PopUp.baz :let g:menustr = 'baz'<CR>
|
||||||
|
|
||||||
|
for ttymouse_val in s:ttymouse_values
|
||||||
|
exe 'set ttymouse=' .. ttymouse_val
|
||||||
|
let g:menustr = ''
|
||||||
|
call feedkeys(MouseRightClickCode(1, 4)
|
||||||
|
\ .. MouseRightReleaseCode(1, 4) .. "\<Down>\<Down>\<CR>", "x")
|
||||||
|
call assert_equal('bar', g:menustr)
|
||||||
|
endfor
|
||||||
|
|
||||||
|
unmenu PopUp
|
||||||
|
let &mouse = save_mouse
|
||||||
|
let &term = save_term
|
||||||
|
let &ttymouse = save_ttymouse
|
||||||
|
let &mousemodel = save_mousemodel
|
||||||
|
call test_override('no_query_mouse', 0)
|
||||||
|
close!
|
||||||
|
endfunc
|
||||||
|
|
||||||
" This only checks if the sequence is recognized.
|
" This only checks if the sequence is recognized.
|
||||||
func Test_term_rgb_response()
|
func Test_term_rgb_response()
|
||||||
set t_RF=x
|
set t_RF=x
|
||||||
@@ -1501,3 +1534,5 @@ func Test_cmdline_literal()
|
|||||||
|
|
||||||
set timeoutlen&
|
set timeoutlen&
|
||||||
endfunc
|
endfunc
|
||||||
|
|
||||||
|
" vim: shiftwidth=2 sts=2 expandtab
|
||||||
|
@@ -738,6 +738,8 @@ static char *(features[]) =
|
|||||||
|
|
||||||
static int included_patches[] =
|
static int included_patches[] =
|
||||||
{ /* Add new patch number below this line */
|
{ /* Add new patch number below this line */
|
||||||
|
/**/
|
||||||
|
385,
|
||||||
/**/
|
/**/
|
||||||
384,
|
384,
|
||||||
/**/
|
/**/
|
||||||
|
Reference in New Issue
Block a user