blob: a5886742267736d8462fb03d4ef16a8ab4c6004d [file] [log] [blame]
Bram Moolenaaredf3f972016-08-29 22:49:24 +02001/* vi:set ts=8 sts=4 sw=4 noet:
Bram Moolenaar071d4272004-06-13 20:20:40 +00002 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 * GUI/Motif support by Robert Webb
5 *
6 * Do ":help uganda" in Vim to read copying and usage conditions.
7 * Do ":help credits" in Vim to see a list of people who contributed.
8 * See README.txt for an overview of the Vim source code.
9 */
10
11/*
12 * Code for menus. Used for the GUI and 'wildmenu'.
13 */
14
15#include "vim.h"
16
17#if defined(FEAT_MENU) || defined(PROTO)
18
Bram Moolenaar4ba37b52019-12-04 21:57:43 +010019#define MENUDEPTH 10 // maximum depth of menus
Bram Moolenaar071d4272004-06-13 20:20:40 +000020
Bram Moolenaar4f974752019-02-17 17:44:42 +010021#ifdef FEAT_GUI_MSWIN
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010022static int add_menu_path(char_u *, vimmenu_T *, int *, char_u *, int);
Bram Moolenaar071d4272004-06-13 20:20:40 +000023#else
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010024static int add_menu_path(char_u *, vimmenu_T *, int *, char_u *);
Bram Moolenaar071d4272004-06-13 20:20:40 +000025#endif
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010026static int menu_nable_recurse(vimmenu_T *menu, char_u *name, int modes, int enable);
27static int remove_menu(vimmenu_T **, char_u *, int, int silent);
28static void free_menu(vimmenu_T **menup);
29static void free_menu_string(vimmenu_T *, int);
30static int show_menus(char_u *, int);
31static void show_menus_recursive(vimmenu_T *, int, int);
Bram Moolenaar5843f5f2019-08-20 20:13:45 +020032static char_u *menu_name_skip(char_u *name);
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010033static int menu_name_equal(char_u *name, vimmenu_T *menu);
34static int menu_namecmp(char_u *name, char_u *mname);
35static int get_menu_cmd_modes(char_u *, int, int *, int *);
36static char_u *popup_mode_name(char_u *name, int idx);
37static char_u *menu_text(char_u *text, int *mnemonic, char_u **actext);
Bram Moolenaar071d4272004-06-13 20:20:40 +000038
Bram Moolenaar4f974752019-02-17 17:44:42 +010039#if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010040static void gui_create_tearoffs_recurse(vimmenu_T *menu, const char_u *pname, int *pri_tab, int pri_idx);
41static void gui_add_tearoff(char_u *tearpath, int *pri_tab, int pri_idx);
42static void gui_destroy_tearoffs_recurse(vimmenu_T *menu);
Bram Moolenaar071d4272004-06-13 20:20:40 +000043static int s_tearoffs = FALSE;
44#endif
45
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010046static int menu_is_hidden(char_u *name);
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010047static int menu_is_tearoff(char_u *name);
Bram Moolenaar071d4272004-06-13 20:20:40 +000048
49#if defined(FEAT_MULTI_LANG) || defined(FEAT_TOOLBAR)
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010050static char_u *menu_skip_part(char_u *p);
Bram Moolenaar071d4272004-06-13 20:20:40 +000051#endif
52#ifdef FEAT_MULTI_LANG
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010053static char_u *menutrans_lookup(char_u *name, int len);
54static void menu_unescape_name(char_u *p);
Bram Moolenaar071d4272004-06-13 20:20:40 +000055#endif
56
Bram Moolenaar92b8b2d2016-01-29 22:36:45 +010057static char_u *menu_translate_tab_and_shift(char_u *arg_start);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +020058
Bram Moolenaar4ba37b52019-12-04 21:57:43 +010059// The character for each menu mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +020060static char *menu_mode_chars[] = {"n", "v", "s", "o", "i", "c", "tl", "t"};
Bram Moolenaar071d4272004-06-13 20:20:40 +000061
Bram Moolenaar071d4272004-06-13 20:20:40 +000062#ifdef FEAT_TOOLBAR
63static const char *toolbar_names[] =
64{
65 /* 0 */ "New", "Open", "Save", "Undo", "Redo",
66 /* 5 */ "Cut", "Copy", "Paste", "Print", "Help",
67 /* 10 */ "Find", "SaveAll", "SaveSesn", "NewSesn", "LoadSesn",
68 /* 15 */ "RunScript", "Replace", "WinClose", "WinMax", "WinMin",
69 /* 20 */ "WinSplit", "Shell", "FindPrev", "FindNext", "FindHelp",
70 /* 25 */ "Make", "TagJump", "RunCtags", "WinVSplit", "WinMaxWidth",
71 /* 30 */ "WinMinWidth", "Exit"
72};
K.Takataeeec2542021-06-02 13:28:16 +020073# define TOOLBAR_NAME_COUNT ARRAY_LENGTH(toolbar_names)
Bram Moolenaar071d4272004-06-13 20:20:40 +000074#endif
75
76/*
Bram Moolenaar1b9645d2017-09-17 23:03:31 +020077 * Return TRUE if "name" is a window toolbar menu name.
78 */
79 static int
80menu_is_winbar(char_u *name)
81{
Bram Moolenaar378daf82017-09-23 23:58:28 +020082 return (STRNCMP(name, "WinBar", 6) == 0);
Bram Moolenaar1b9645d2017-09-17 23:03:31 +020083}
84
85 int
86winbar_height(win_T *wp)
87{
88 if (wp->w_winbar != NULL && wp->w_winbar->children != NULL)
89 return 1;
90 return 0;
91}
92
93 static vimmenu_T **
94get_root_menu(char_u *name)
95{
96 if (menu_is_winbar(name))
97 return &curwin->w_winbar;
98 return &root_menu;
99}
100
101/*
Bram Moolenaar071d4272004-06-13 20:20:40 +0000102 * Do the :menu command and relatives.
103 */
104 void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100105ex_menu(
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100106 exarg_T *eap) // Ex command arguments
Bram Moolenaar071d4272004-06-13 20:20:40 +0000107{
108 char_u *menu_path;
109 int modes;
110 char_u *map_to;
111 int noremap;
112 int silent = FALSE;
Bram Moolenaar8b2d9c42006-05-03 21:28:47 +0000113 int special = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000114 int unmenu;
115 char_u *map_buf;
116 char_u *arg;
117 char_u *p;
118 int i;
Bram Moolenaar241a8aa2005-12-06 20:04:44 +0000119#if defined(FEAT_GUI) && !defined(FEAT_GUI_GTK)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000120 int old_menu_height;
Bram Moolenaar4f974752019-02-17 17:44:42 +0100121# if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000122 int old_toolbar_height;
123# endif
124#endif
125 int pri_tab[MENUDEPTH + 1];
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100126 int enable = MAYBE; // TRUE for "menu enable", FALSE for "menu
127 // disable
Bram Moolenaar071d4272004-06-13 20:20:40 +0000128#ifdef FEAT_TOOLBAR
129 char_u *icon = NULL;
130#endif
131 vimmenu_T menuarg;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200132 vimmenu_T **root_menu_ptr;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000133
134 modes = get_menu_cmd_modes(eap->cmd, eap->forceit, &noremap, &unmenu);
135 arg = eap->arg;
136
137 for (;;)
138 {
139 if (STRNCMP(arg, "<script>", 8) == 0)
140 {
141 noremap = REMAP_SCRIPT;
142 arg = skipwhite(arg + 8);
143 continue;
144 }
145 if (STRNCMP(arg, "<silent>", 8) == 0)
146 {
147 silent = TRUE;
148 arg = skipwhite(arg + 8);
149 continue;
150 }
Bram Moolenaar8b2d9c42006-05-03 21:28:47 +0000151 if (STRNCMP(arg, "<special>", 9) == 0)
152 {
153 special = TRUE;
154 arg = skipwhite(arg + 9);
155 continue;
156 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000157 break;
158 }
159
160
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100161 // Locate an optional "icon=filename" argument.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000162 if (STRNCMP(arg, "icon=", 5) == 0)
163 {
164 arg += 5;
165#ifdef FEAT_TOOLBAR
166 icon = arg;
167#endif
168 while (*arg != NUL && *arg != ' ')
169 {
170 if (*arg == '\\')
Bram Moolenaar8c8de832008-06-24 22:58:06 +0000171 STRMOVE(arg, arg + 1);
Bram Moolenaar91acfff2017-03-12 19:22:36 +0100172 MB_PTR_ADV(arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000173 }
174 if (*arg != NUL)
175 {
176 *arg++ = NUL;
177 arg = skipwhite(arg);
178 }
179 }
180
181 /*
182 * Fill in the priority table.
183 */
184 for (p = arg; *p; ++p)
185 if (!VIM_ISDIGIT(*p) && *p != '.')
186 break;
Bram Moolenaar1c465442017-03-12 20:10:05 +0100187 if (VIM_ISWHITE(*p))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000188 {
Bram Moolenaar1c465442017-03-12 20:10:05 +0100189 for (i = 0; i < MENUDEPTH && !VIM_ISWHITE(*arg); ++i)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000190 {
191 pri_tab[i] = getdigits(&arg);
192 if (pri_tab[i] == 0)
193 pri_tab[i] = 500;
194 if (*arg == '.')
195 ++arg;
196 }
197 arg = skipwhite(arg);
198 }
199 else if (eap->addr_count && eap->line2 != 0)
200 {
201 pri_tab[0] = eap->line2;
202 i = 1;
203 }
204 else
205 i = 0;
206 while (i < MENUDEPTH)
207 pri_tab[i++] = 500;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100208 pri_tab[MENUDEPTH] = -1; // mark end of the table
Bram Moolenaar071d4272004-06-13 20:20:40 +0000209
210 /*
211 * Check for "disable" or "enable" argument.
212 */
Bram Moolenaar1c465442017-03-12 20:10:05 +0100213 if (STRNCMP(arg, "enable", 6) == 0 && VIM_ISWHITE(arg[6]))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000214 {
215 enable = TRUE;
216 arg = skipwhite(arg + 6);
217 }
Bram Moolenaar1c465442017-03-12 20:10:05 +0100218 else if (STRNCMP(arg, "disable", 7) == 0 && VIM_ISWHITE(arg[7]))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000219 {
220 enable = FALSE;
221 arg = skipwhite(arg + 7);
222 }
223
224 /*
225 * If there is no argument, display all menus.
226 */
227 if (*arg == NUL)
228 {
229 show_menus(arg, modes);
230 return;
231 }
232
233#ifdef FEAT_TOOLBAR
234 /*
235 * Need to get the toolbar icon index before doing the translation.
236 */
237 menuarg.iconidx = -1;
238 menuarg.icon_builtin = FALSE;
239 if (menu_is_toolbar(arg))
240 {
241 menu_path = menu_skip_part(arg);
242 if (*menu_path == '.')
243 {
244 p = menu_skip_part(++menu_path);
245 if (STRNCMP(menu_path, "BuiltIn", 7) == 0)
246 {
247 if (skipdigits(menu_path + 7) == p)
248 {
249 menuarg.iconidx = atoi((char *)menu_path + 7);
Bram Moolenaaraf0167f2009-05-16 15:31:32 +0000250 if (menuarg.iconidx >= (int)TOOLBAR_NAME_COUNT)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000251 menuarg.iconidx = -1;
252 else
253 menuarg.icon_builtin = TRUE;
254 }
255 }
256 else
257 {
Bram Moolenaaraf0167f2009-05-16 15:31:32 +0000258 for (i = 0; i < (int)TOOLBAR_NAME_COUNT; ++i)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000259 if (STRNCMP(toolbar_names[i], menu_path, p - menu_path)
260 == 0)
261 {
262 menuarg.iconidx = i;
263 break;
264 }
265 }
266 }
267 }
268#endif
269
Bram Moolenaar071d4272004-06-13 20:20:40 +0000270 menu_path = arg;
271 if (*menu_path == '.')
272 {
Bram Moolenaar436b5ad2021-12-31 22:49:24 +0000273 semsg(_(e_invalid_argument_str), menu_path);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000274 goto theend;
275 }
276
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200277 map_to = menu_translate_tab_and_shift(arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000278
279 /*
280 * If there is only a menu name, display menus with that name.
281 */
282 if (*map_to == NUL && !unmenu && enable == MAYBE)
283 {
284 show_menus(menu_path, modes);
285 goto theend;
286 }
287 else if (*map_to != NUL && (unmenu || enable != MAYBE))
288 {
Bram Moolenaar74409f62022-01-01 15:58:22 +0000289 semsg(_(e_trailing_characters_str), map_to);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000290 goto theend;
291 }
Bram Moolenaar241a8aa2005-12-06 20:04:44 +0000292#if defined(FEAT_GUI) && !(defined(FEAT_GUI_GTK) || defined(FEAT_GUI_PHOTON))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000293 old_menu_height = gui.menu_height;
Bram Moolenaar4f974752019-02-17 17:44:42 +0100294# if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000295 old_toolbar_height = gui.toolbar_height;
296# endif
297#endif
298
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200299 root_menu_ptr = get_root_menu(menu_path);
300 if (root_menu_ptr == &curwin->w_winbar)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100301 // Assume the window toolbar menu will change.
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200302 redraw_later(NOT_VALID);
303
Bram Moolenaar071d4272004-06-13 20:20:40 +0000304 if (enable != MAYBE)
305 {
306 /*
307 * Change sensitivity of the menu.
308 * For the PopUp menu, remove a menu for each mode separately.
309 * Careful: menu_nable_recurse() changes menu_path.
310 */
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100311 if (STRCMP(menu_path, "*") == 0) // meaning: do all menus
Bram Moolenaar071d4272004-06-13 20:20:40 +0000312 menu_path = (char_u *)"";
313
314 if (menu_is_popup(menu_path))
315 {
316 for (i = 0; i < MENU_INDEX_TIP; ++i)
317 if (modes & (1 << i))
318 {
319 p = popup_mode_name(menu_path, i);
320 if (p != NULL)
321 {
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200322 menu_nable_recurse(*root_menu_ptr, p, MENU_ALL_MODES,
Bram Moolenaar071d4272004-06-13 20:20:40 +0000323 enable);
324 vim_free(p);
325 }
326 }
327 }
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200328 menu_nable_recurse(*root_menu_ptr, menu_path, modes, enable);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000329 }
330 else if (unmenu)
331 {
332 /*
333 * Delete menu(s).
334 */
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100335 if (STRCMP(menu_path, "*") == 0) // meaning: remove all menus
Bram Moolenaar071d4272004-06-13 20:20:40 +0000336 menu_path = (char_u *)"";
337
338 /*
339 * For the PopUp menu, remove a menu for each mode separately.
340 */
341 if (menu_is_popup(menu_path))
342 {
343 for (i = 0; i < MENU_INDEX_TIP; ++i)
344 if (modes & (1 << i))
345 {
346 p = popup_mode_name(menu_path, i);
347 if (p != NULL)
348 {
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200349 remove_menu(root_menu_ptr, p, MENU_ALL_MODES, TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000350 vim_free(p);
351 }
352 }
353 }
354
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100355 // Careful: remove_menu() changes menu_path
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200356 remove_menu(root_menu_ptr, menu_path, modes, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000357 }
358 else
359 {
360 /*
361 * Add menu(s).
362 * Replace special key codes.
363 */
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100364 if (STRICMP(map_to, "<nop>") == 0) // "<Nop>" means nothing
Bram Moolenaar071d4272004-06-13 20:20:40 +0000365 {
366 map_to = (char_u *)"";
367 map_buf = NULL;
368 }
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000369 else if (modes & MENU_TIP_MODE)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100370 map_buf = NULL; // Menu tips are plain text.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000371 else
Bram Moolenaar459fd782019-10-13 16:43:39 +0200372 map_to = replace_termcodes(map_to, &map_buf,
373 REPTERM_DO_LT | (special ? REPTERM_SPECIAL : 0), NULL);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000374 menuarg.modes = modes;
375#ifdef FEAT_TOOLBAR
376 menuarg.iconfile = icon;
377#endif
378 menuarg.noremap[0] = noremap;
379 menuarg.silent[0] = silent;
380 add_menu_path(menu_path, &menuarg, pri_tab, map_to
Bram Moolenaar4f974752019-02-17 17:44:42 +0100381#ifdef FEAT_GUI_MSWIN
Bram Moolenaar071d4272004-06-13 20:20:40 +0000382 , TRUE
383#endif
384 );
385
386 /*
387 * For the PopUp menu, add a menu for each mode separately.
388 */
389 if (menu_is_popup(menu_path))
390 {
391 for (i = 0; i < MENU_INDEX_TIP; ++i)
392 if (modes & (1 << i))
393 {
394 p = popup_mode_name(menu_path, i);
395 if (p != NULL)
396 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100397 // Include all modes, to make ":amenu" work
Bram Moolenaar071d4272004-06-13 20:20:40 +0000398 menuarg.modes = modes;
399#ifdef FEAT_TOOLBAR
400 menuarg.iconfile = NULL;
401 menuarg.iconidx = -1;
402 menuarg.icon_builtin = FALSE;
403#endif
404 add_menu_path(p, &menuarg, pri_tab, map_to
Bram Moolenaar4f974752019-02-17 17:44:42 +0100405#ifdef FEAT_GUI_MSWIN
Bram Moolenaar071d4272004-06-13 20:20:40 +0000406 , TRUE
407#endif
408 );
409 vim_free(p);
410 }
411 }
412 }
413
414 vim_free(map_buf);
415 }
416
Bram Moolenaar241a8aa2005-12-06 20:04:44 +0000417#if defined(FEAT_GUI) && !(defined(FEAT_GUI_GTK))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100418 // If the menubar height changed, resize the window
Bram Moolenaar071d4272004-06-13 20:20:40 +0000419 if (gui.in_use
420 && (gui.menu_height != old_menu_height
Bram Moolenaar4f974752019-02-17 17:44:42 +0100421# if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000422 || gui.toolbar_height != old_toolbar_height
423# endif
424 ))
Bram Moolenaar04a9d452006-03-27 21:03:26 +0000425 gui_set_shellsize(FALSE, FALSE, RESIZE_VERT);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000426#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200427 if (root_menu_ptr == &curwin->w_winbar)
428 {
429 int h = winbar_height(curwin);
430
431 if (h != curwin->w_winbar_height)
432 {
433 if (h == 0)
434 ++curwin->w_height;
435 else if (curwin->w_height > 0)
436 --curwin->w_height;
437 curwin->w_winbar_height = h;
438 }
439 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000440
441theend:
Bram Moolenaar071d4272004-06-13 20:20:40 +0000442 ;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000443}
444
445/*
446 * Add the menu with the given name to the menu hierarchy
447 */
448 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100449add_menu_path(
450 char_u *menu_path,
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100451 vimmenu_T *menuarg, // passes modes, iconfile, iconidx,
452 // icon_builtin, silent[0], noremap[0]
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100453 int *pri_tab,
454 char_u *call_data
Bram Moolenaar4f974752019-02-17 17:44:42 +0100455#ifdef FEAT_GUI_MSWIN
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100456 , int addtearoff // may add tearoff item
Bram Moolenaar071d4272004-06-13 20:20:40 +0000457#endif
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100458 )
Bram Moolenaar071d4272004-06-13 20:20:40 +0000459{
460 char_u *path_name;
461 int modes = menuarg->modes;
462 vimmenu_T **menup;
463 vimmenu_T *menu = NULL;
464 vimmenu_T *parent;
465 vimmenu_T **lower_pri;
466 char_u *p;
467 char_u *name;
468 char_u *dname;
469 char_u *next_name;
470 int i;
471 int c;
Bram Moolenaar7871a502010-05-14 21:19:23 +0200472 int d;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000473#ifdef FEAT_GUI
474 int idx;
475 int new_idx;
476#endif
477 int pri_idx = 0;
478 int old_modes = 0;
479 int amenu;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200480#ifdef FEAT_MULTI_LANG
481 char_u *en_name;
482 char_u *map_to = NULL;
483#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200484 vimmenu_T **root_menu_ptr;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000485
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100486 // Make a copy so we can stuff around with it, since it could be const
Bram Moolenaar071d4272004-06-13 20:20:40 +0000487 path_name = vim_strsave(menu_path);
488 if (path_name == NULL)
489 return FAIL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200490 root_menu_ptr = get_root_menu(menu_path);
491 menup = root_menu_ptr;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000492 parent = NULL;
493 name = path_name;
494 while (*name)
495 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100496 // Get name of this element in the menu hierarchy, and the simplified
497 // name (without mnemonic and accelerator text).
Bram Moolenaar071d4272004-06-13 20:20:40 +0000498 next_name = menu_name_skip(name);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200499#ifdef FEAT_MULTI_LANG
Bram Moolenaar442b4222010-05-24 21:34:22 +0200500 map_to = menutrans_lookup(name, (int)STRLEN(name));
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200501 if (map_to != NULL)
502 {
503 en_name = name;
504 name = map_to;
505 }
506 else
507 en_name = NULL;
508#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000509 dname = menu_text(name, NULL, NULL);
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000510 if (dname == NULL)
511 goto erret;
512 if (*dname == NUL)
513 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100514 // Only a mnemonic or accelerator is not valid.
Bram Moolenaar677658a2022-01-05 16:09:06 +0000515 emsg(_(e_empty_menu_name));
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000516 goto erret;
517 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000518
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100519 // See if it's already there
Bram Moolenaar071d4272004-06-13 20:20:40 +0000520 lower_pri = menup;
521#ifdef FEAT_GUI
522 idx = 0;
523 new_idx = 0;
524#endif
525 menu = *menup;
526 while (menu != NULL)
527 {
528 if (menu_name_equal(name, menu) || menu_name_equal(dname, menu))
529 {
530 if (*next_name == NUL && menu->children != NULL)
531 {
532 if (!sys_menu)
Bram Moolenaard88be5b2022-01-04 19:57:55 +0000533 emsg(_(e_menu_path_must_not_lead_to_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000534 goto erret;
535 }
536 if (*next_name != NUL && menu->children == NULL
Bram Moolenaar4f974752019-02-17 17:44:42 +0100537#ifdef FEAT_GUI_MSWIN
Bram Moolenaar071d4272004-06-13 20:20:40 +0000538 && addtearoff
539#endif
540 )
541 {
542 if (!sys_menu)
Bram Moolenaareaaac012022-01-02 17:00:40 +0000543 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000544 goto erret;
545 }
546 break;
547 }
548 menup = &menu->next;
549
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100550 // Count menus, to find where this one needs to be inserted.
551 // Ignore menus that are not in the menubar (PopUp and Toolbar)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000552 if (parent != NULL || menu_is_menubar(menu->name))
553 {
554#ifdef FEAT_GUI
555 ++idx;
556#endif
557 if (menu->priority <= pri_tab[pri_idx])
558 {
559 lower_pri = menup;
560#ifdef FEAT_GUI
561 new_idx = idx;
562#endif
563 }
564 }
565 menu = menu->next;
566 }
567
568 if (menu == NULL)
569 {
570 if (*next_name == NUL && parent == NULL)
571 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000572 emsg(_(e_must_not_add_menu_items_directly_to_menu_bar));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000573 goto erret;
574 }
575
576 if (menu_is_separator(dname) && *next_name != NUL)
577 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000578 emsg(_(e_separator_cannot_be_part_of_menu_path));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000579 goto erret;
580 }
581
Dominique Pelleaf4a61a2021-12-27 17:21:41 +0000582 // Not already there, so let's add it
Bram Moolenaare809a4e2019-07-04 17:35:05 +0200583 menu = ALLOC_CLEAR_ONE(vimmenu_T);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000584 if (menu == NULL)
585 goto erret;
586
587 menu->modes = modes;
588 menu->enabled = MENU_ALL_MODES;
589 menu->name = vim_strsave(name);
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100590 // separate mnemonic and accelerator text from actual menu name
Bram Moolenaar071d4272004-06-13 20:20:40 +0000591 menu->dname = menu_text(name, &menu->mnemonic, &menu->actext);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200592#ifdef FEAT_MULTI_LANG
593 if (en_name != NULL)
594 {
595 menu->en_name = vim_strsave(en_name);
596 menu->en_dname = menu_text(en_name, NULL, NULL);
597 }
598 else
599 {
600 menu->en_name = NULL;
601 menu->en_dname = NULL;
602 }
603#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000604 menu->priority = pri_tab[pri_idx];
605 menu->parent = parent;
606#ifdef FEAT_GUI_MOTIF
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100607 menu->sensitive = TRUE; // the default
Bram Moolenaar071d4272004-06-13 20:20:40 +0000608#endif
609#ifdef FEAT_BEVAL_TIP
610 menu->tip = NULL;
611#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000612 /*
613 * Add after menu that has lower priority.
614 */
615 menu->next = *lower_pri;
616 *lower_pri = menu;
617
618 old_modes = 0;
619
620#ifdef FEAT_TOOLBAR
621 menu->iconidx = menuarg->iconidx;
622 menu->icon_builtin = menuarg->icon_builtin;
623 if (*next_name == NUL && menuarg->iconfile != NULL)
624 menu->iconfile = vim_strsave(menuarg->iconfile);
625#endif
Bram Moolenaar4f974752019-02-17 17:44:42 +0100626#if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100627 // the tearoff item must be present in the modes of each item.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000628 if (parent != NULL && menu_is_tearoff(parent->children->dname))
629 parent->children->modes |= modes;
630#endif
631 }
632 else
633 {
634 old_modes = menu->modes;
635
636 /*
637 * If this menu option was previously only available in other
638 * modes, then make sure it's available for this one now
639 * Also enable a menu when it's created or changed.
640 */
Bram Moolenaar4f974752019-02-17 17:44:42 +0100641#ifdef FEAT_GUI_MSWIN
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100642 // If adding a tearbar (addtearoff == FALSE) don't update modes
Bram Moolenaar071d4272004-06-13 20:20:40 +0000643 if (addtearoff)
644#endif
645 {
646 menu->modes |= modes;
647 menu->enabled |= modes;
648 }
649 }
650
651#ifdef FEAT_GUI
652 /*
653 * Add the menu item when it's used in one of the modes, but not when
654 * only a tooltip is defined.
655 */
656 if ((old_modes & MENU_ALL_MODES) == 0
657 && (menu->modes & MENU_ALL_MODES) != 0)
658 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100659 if (gui.in_use) // Otherwise it will be added when GUI starts
Bram Moolenaar071d4272004-06-13 20:20:40 +0000660 {
661 if (*next_name == NUL)
662 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100663 // Real menu item, not sub-menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000664 gui_mch_add_menu_item(menu, new_idx);
665
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100666 // Want to update menus now even if mode not changed
Bram Moolenaar071d4272004-06-13 20:20:40 +0000667 force_menu_update = TRUE;
668 }
669 else
670 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100671 // Sub-menu (not at end of path yet)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000672 gui_mch_add_menu(menu, new_idx);
673 }
674 }
675
K.Takata54119102022-02-03 13:33:03 +0000676# if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100677 // When adding a new submenu, may add a tearoff item
Bram Moolenaar071d4272004-06-13 20:20:40 +0000678 if ( addtearoff
679 && *next_name
680 && vim_strchr(p_go, GO_TEAROFF) != NULL
Bram Moolenaar310c32e2019-11-29 23:15:25 +0100681 && menu_is_menubar(name)
682# ifdef VIMDLL
683 && (gui.in_use || gui.starting)
684# endif
685 )
Bram Moolenaar071d4272004-06-13 20:20:40 +0000686 {
687 char_u *tearpath;
688
689 /*
690 * The pointers next_name & path_name refer to a string with
691 * \'s and ^V's stripped out. But menu_path is a "raw"
692 * string, so we must correct for special characters.
693 */
Bram Moolenaar964b3742019-05-24 18:54:09 +0200694 tearpath = alloc(STRLEN(menu_path) + TEAR_LEN + 2);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000695 if (tearpath != NULL)
696 {
697 char_u *s;
698 int idx;
699
700 STRCPY(tearpath, menu_path);
701 idx = (int)(next_name - path_name - 1);
Bram Moolenaar91acfff2017-03-12 19:22:36 +0100702 for (s = tearpath; *s && s < tearpath + idx; MB_PTR_ADV(s))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000703 {
704 if ((*s == '\\' || *s == Ctrl_V) && s[1])
705 {
706 ++idx;
707 ++s;
708 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000709 }
710 tearpath[idx] = NUL;
711 gui_add_tearoff(tearpath, pri_tab, pri_idx);
712 vim_free(tearpath);
713 }
714 }
715# endif
716 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100717#endif // FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +0000718
719 menup = &menu->children;
720 parent = menu;
721 name = next_name;
Bram Moolenaard23a8232018-02-10 18:45:26 +0100722 VIM_CLEAR(dname);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000723 if (pri_tab[pri_idx + 1] != -1)
724 ++pri_idx;
725 }
726 vim_free(path_name);
727
728 /*
729 * Only add system menu items which have not been defined yet.
730 * First check if this was an ":amenu".
731 */
732 amenu = ((modes & (MENU_NORMAL_MODE | MENU_INSERT_MODE)) ==
733 (MENU_NORMAL_MODE | MENU_INSERT_MODE));
734 if (sys_menu)
735 modes &= ~old_modes;
736
737 if (menu != NULL && modes)
738 {
739#ifdef FEAT_GUI
740 menu->cb = gui_menu_cb;
741#endif
742 p = (call_data == NULL) ? NULL : vim_strsave(call_data);
743
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100744 // loop over all modes, may add more than one
Bram Moolenaar071d4272004-06-13 20:20:40 +0000745 for (i = 0; i < MENU_MODES; ++i)
746 {
747 if (modes & (1 << i))
748 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100749 // free any old menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000750 free_menu_string(menu, i);
751
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100752 // For "amenu", may insert an extra character.
753 // Don't do this if adding a tearbar (addtearoff == FALSE).
754 // Don't do this for "<Nop>".
Bram Moolenaar071d4272004-06-13 20:20:40 +0000755 c = 0;
Bram Moolenaar7871a502010-05-14 21:19:23 +0200756 d = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000757 if (amenu && call_data != NULL && *call_data != NUL
Bram Moolenaar4f974752019-02-17 17:44:42 +0100758#ifdef FEAT_GUI_MSWIN
Bram Moolenaar071d4272004-06-13 20:20:40 +0000759 && addtearoff
760#endif
761 )
762 {
763 switch (1 << i)
764 {
765 case MENU_VISUAL_MODE:
Bram Moolenaarb3656ed2006-03-20 21:59:49 +0000766 case MENU_SELECT_MODE:
Bram Moolenaar071d4272004-06-13 20:20:40 +0000767 case MENU_OP_PENDING_MODE:
768 case MENU_CMDLINE_MODE:
769 c = Ctrl_C;
770 break;
771 case MENU_INSERT_MODE:
Bram Moolenaar7871a502010-05-14 21:19:23 +0200772 c = Ctrl_BSL;
773 d = Ctrl_O;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000774 break;
775 }
776 }
777
Bram Moolenaar7871a502010-05-14 21:19:23 +0200778 if (c != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000779 {
Bram Moolenaar964b3742019-05-24 18:54:09 +0200780 menu->strings[i] = alloc(STRLEN(call_data) + 5);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000781 if (menu->strings[i] != NULL)
782 {
783 menu->strings[i][0] = c;
Bram Moolenaar7871a502010-05-14 21:19:23 +0200784 if (d == 0)
785 STRCPY(menu->strings[i] + 1, call_data);
786 else
787 {
788 menu->strings[i][1] = d;
789 STRCPY(menu->strings[i] + 2, call_data);
790 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000791 if (c == Ctrl_C)
792 {
Bram Moolenaara93fa7e2006-04-17 22:14:47 +0000793 int len = (int)STRLEN(menu->strings[i]);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000794
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100795 // Append CTRL-\ CTRL-G to obey 'insertmode'.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000796 menu->strings[i][len] = Ctrl_BSL;
797 menu->strings[i][len + 1] = Ctrl_G;
798 menu->strings[i][len + 2] = NUL;
799 }
800 }
801 }
802 else
803 menu->strings[i] = p;
804 menu->noremap[i] = menuarg->noremap[0];
805 menu->silent[i] = menuarg->silent[0];
806 }
807 }
Bram Moolenaar4f974752019-02-17 17:44:42 +0100808#if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN) \
Bram Moolenaarc3719bd2017-11-18 22:13:31 +0100809 && (defined(FEAT_BEVAL_GUI) || defined(FEAT_GUI_GTK))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100810 // Need to update the menu tip.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000811 if (modes & MENU_TIP_MODE)
812 gui_mch_menu_set_tip(menu);
813#endif
814 }
815 return OK;
816
817erret:
818 vim_free(path_name);
819 vim_free(dname);
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000820
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100821 // Delete any empty submenu we added before discovering the error. Repeat
822 // for higher levels.
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000823 while (parent != NULL && parent->children == NULL)
824 {
825 if (parent->parent == NULL)
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200826 menup = root_menu_ptr;
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000827 else
828 menup = &parent->parent->children;
829 for ( ; *menup != NULL && *menup != parent; menup = &((*menup)->next))
830 ;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100831 if (*menup == NULL) // safety check
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000832 break;
833 parent = parent->parent;
834 free_menu(menup);
835 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000836 return FAIL;
837}
838
839/*
840 * Set the (sub)menu with the given name to enabled or disabled.
841 * Called recursively.
842 */
843 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100844menu_nable_recurse(
845 vimmenu_T *menu,
846 char_u *name,
847 int modes,
848 int enable)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000849{
850 char_u *p;
851
852 if (menu == NULL)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100853 return OK; // Got to bottom of hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000854
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100855 // Get name of this element in the menu hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000856 p = menu_name_skip(name);
857
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100858 // Find the menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000859 while (menu != NULL)
860 {
861 if (*name == NUL || *name == '*' || menu_name_equal(name, menu))
862 {
863 if (*p != NUL)
864 {
865 if (menu->children == NULL)
866 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000867 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000868 return FAIL;
869 }
870 if (menu_nable_recurse(menu->children, p, modes, enable)
871 == FAIL)
872 return FAIL;
873 }
874 else
875 if (enable)
876 menu->enabled |= modes;
877 else
878 menu->enabled &= ~modes;
879
880 /*
881 * When name is empty, we are doing all menu items for the given
882 * modes, so keep looping, otherwise we are just doing the named
883 * menu item (which has been found) so break here.
884 */
885 if (*name != NUL && *name != '*')
886 break;
887 }
888 menu = menu->next;
889 }
890 if (*name != NUL && *name != '*' && menu == NULL)
891 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000892 semsg(_(e_no_menu_str), name);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000893 return FAIL;
894 }
895
896#ifdef FEAT_GUI
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100897 // Want to update menus now even if mode not changed
Bram Moolenaar071d4272004-06-13 20:20:40 +0000898 force_menu_update = TRUE;
899#endif
900
901 return OK;
902}
903
904/*
905 * Remove the (sub)menu with the given name from the menu hierarchy
906 * Called recursively.
907 */
908 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100909remove_menu(
910 vimmenu_T **menup,
911 char_u *name,
912 int modes,
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100913 int silent) // don't give error messages
Bram Moolenaar071d4272004-06-13 20:20:40 +0000914{
915 vimmenu_T *menu;
916 vimmenu_T *child;
917 char_u *p;
918
919 if (*menup == NULL)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100920 return OK; // Got to bottom of hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000921
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100922 // Get name of this element in the menu hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000923 p = menu_name_skip(name);
924
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100925 // Find the menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000926 while ((menu = *menup) != NULL)
927 {
928 if (*name == NUL || menu_name_equal(name, menu))
929 {
930 if (*p != NUL && menu->children == NULL)
931 {
932 if (!silent)
Bram Moolenaareaaac012022-01-02 17:00:40 +0000933 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000934 return FAIL;
935 }
936 if ((menu->modes & modes) != 0x0)
937 {
Bram Moolenaar4f974752019-02-17 17:44:42 +0100938#if defined(FEAT_GUI_MSWIN) & defined(FEAT_TEAROFF)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000939 /*
940 * If we are removing all entries for this menu,MENU_ALL_MODES,
941 * Then kill any tearoff before we start
942 */
943 if (*p == NUL && modes == MENU_ALL_MODES)
944 {
945 if (IsWindow(menu->tearoff_handle))
946 DestroyWindow(menu->tearoff_handle);
947 }
948#endif
949 if (remove_menu(&menu->children, p, modes, silent) == FAIL)
950 return FAIL;
951 }
952 else if (*name != NUL)
953 {
954 if (!silent)
Bram Moolenaar3a846e62022-01-01 16:21:00 +0000955 emsg(_(e_menu_only_exists_in_another_mode));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000956 return FAIL;
957 }
958
959 /*
960 * When name is empty, we are removing all menu items for the given
961 * modes, so keep looping, otherwise we are just removing the named
962 * menu item (which has been found) so break here.
963 */
964 if (*name != NUL)
965 break;
966
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100967 // Remove the menu item for the given mode[s]. If the menu item
968 // is no longer valid in ANY mode, delete it
Bram Moolenaar071d4272004-06-13 20:20:40 +0000969 menu->modes &= ~modes;
970 if (modes & MENU_TIP_MODE)
971 free_menu_string(menu, MENU_INDEX_TIP);
972 if ((menu->modes & MENU_ALL_MODES) == 0)
973 free_menu(menup);
974 else
975 menup = &menu->next;
976 }
977 else
978 menup = &menu->next;
979 }
980 if (*name != NUL)
981 {
982 if (menu == NULL)
983 {
984 if (!silent)
Bram Moolenaareaaac012022-01-02 17:00:40 +0000985 semsg(_(e_no_menu_str), name);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000986 return FAIL;
987 }
988
989
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100990 // Recalculate modes for menu based on the new updated children
Bram Moolenaar071d4272004-06-13 20:20:40 +0000991 menu->modes &= ~modes;
Bram Moolenaar4f974752019-02-17 17:44:42 +0100992#if defined(FEAT_GUI_MSWIN) & defined(FEAT_TEAROFF)
Dominique Pelleaf4a61a2021-12-27 17:21:41 +0000993 if ((s_tearoffs) && (menu->children != NULL)) // there's a tear bar.
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100994 child = menu->children->next; // don't count tearoff bar
Bram Moolenaar071d4272004-06-13 20:20:40 +0000995 else
996#endif
997 child = menu->children;
998 for ( ; child != NULL; child = child->next)
999 menu->modes |= child->modes;
1000 if (modes & MENU_TIP_MODE)
1001 {
1002 free_menu_string(menu, MENU_INDEX_TIP);
Bram Moolenaar4f974752019-02-17 17:44:42 +01001003#if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN) \
Bram Moolenaarc3719bd2017-11-18 22:13:31 +01001004 && (defined(FEAT_BEVAL_GUI) || defined(FEAT_GUI_GTK))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001005 // Need to update the menu tip.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001006 if (gui.in_use)
1007 gui_mch_menu_set_tip(menu);
1008#endif
1009 }
1010 if ((menu->modes & MENU_ALL_MODES) == 0)
1011 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001012 // The menu item is no longer valid in ANY mode, so delete it
Bram Moolenaar4f974752019-02-17 17:44:42 +01001013#if defined(FEAT_GUI_MSWIN) & defined(FEAT_TEAROFF)
Dominique Pelleaf4a61a2021-12-27 17:21:41 +00001014 if (s_tearoffs && menu->children != NULL) // there's a tear bar.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001015 free_menu(&menu->children);
1016#endif
1017 *menup = menu;
1018 free_menu(menup);
1019 }
1020 }
1021
1022 return OK;
1023}
1024
1025/*
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001026 * Remove the WinBar menu from window "wp".
1027 */
1028 void
1029remove_winbar(win_T *wp)
1030{
1031 remove_menu(&wp->w_winbar, (char_u *)"", MENU_ALL_MODES, TRUE);
1032 vim_free(wp->w_winbar_items);
1033}
1034
1035/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00001036 * Free the given menu structure and remove it from the linked list.
1037 */
1038 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001039free_menu(vimmenu_T **menup)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001040{
1041 int i;
1042 vimmenu_T *menu;
1043
1044 menu = *menup;
1045
1046#ifdef FEAT_GUI
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001047 // Free machine specific menu structures (only when already created)
1048 // Also may rebuild a tearoff'ed menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001049 if (gui.in_use)
1050 gui_mch_destroy_menu(menu);
1051#endif
1052
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001053 // Don't change *menup until after calling gui_mch_destroy_menu(). The
1054 // MacOS code needs the original structure to properly delete the menu.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001055 *menup = menu->next;
1056 vim_free(menu->name);
1057 vim_free(menu->dname);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001058#ifdef FEAT_MULTI_LANG
1059 vim_free(menu->en_name);
1060 vim_free(menu->en_dname);
1061#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001062 vim_free(menu->actext);
1063#ifdef FEAT_TOOLBAR
1064 vim_free(menu->iconfile);
Bram Moolenaarbee0c5b2005-02-07 22:03:36 +00001065# ifdef FEAT_GUI_MOTIF
1066 vim_free(menu->xpm_fname);
1067# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001068#endif
1069 for (i = 0; i < MENU_MODES; i++)
1070 free_menu_string(menu, i);
1071 vim_free(menu);
1072
1073#ifdef FEAT_GUI
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001074 // Want to update menus now even if mode not changed
Bram Moolenaar071d4272004-06-13 20:20:40 +00001075 force_menu_update = TRUE;
1076#endif
1077}
1078
1079/*
1080 * Free the menu->string with the given index.
1081 */
1082 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001083free_menu_string(vimmenu_T *menu, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001084{
1085 int count = 0;
1086 int i;
1087
1088 for (i = 0; i < MENU_MODES; i++)
1089 if (menu->strings[i] == menu->strings[idx])
1090 count++;
1091 if (count == 1)
1092 vim_free(menu->strings[idx]);
1093 menu->strings[idx] = NULL;
1094}
1095
1096/*
1097 * Show the mapping associated with a menu item or hierarchy in a sub-menu.
1098 */
1099 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001100show_menus(char_u *path_name, int modes)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001101{
1102 char_u *p;
1103 char_u *name;
1104 vimmenu_T *menu;
1105 vimmenu_T *parent = NULL;
1106
Bram Moolenaar071d4272004-06-13 20:20:40 +00001107 name = path_name = vim_strsave(path_name);
1108 if (path_name == NULL)
1109 return FAIL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001110 menu = *get_root_menu(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001111
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001112 // First, find the (sub)menu with the given name
Bram Moolenaar071d4272004-06-13 20:20:40 +00001113 while (*name)
1114 {
1115 p = menu_name_skip(name);
1116 while (menu != NULL)
1117 {
1118 if (menu_name_equal(name, menu))
1119 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001120 // Found menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001121 if (*p != NUL && menu->children == NULL)
1122 {
Bram Moolenaareaaac012022-01-02 17:00:40 +00001123 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001124 vim_free(path_name);
1125 return FAIL;
1126 }
1127 else if ((menu->modes & modes) == 0x0)
1128 {
Bram Moolenaar3a846e62022-01-01 16:21:00 +00001129 emsg(_(e_menu_only_exists_in_another_mode));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001130 vim_free(path_name);
1131 return FAIL;
1132 }
1133 break;
1134 }
1135 menu = menu->next;
1136 }
1137 if (menu == NULL)
1138 {
Bram Moolenaareaaac012022-01-02 17:00:40 +00001139 semsg(_(e_no_menu_str), name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001140 vim_free(path_name);
1141 return FAIL;
1142 }
1143 name = p;
1144 parent = menu;
1145 menu = menu->children;
1146 }
Bram Moolenaaracbd4422008-08-17 21:44:45 +00001147 vim_free(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001148
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001149 // Now we have found the matching menu, and we list the mappings
1150 // Highlight title
Bram Moolenaar32526b32019-01-19 17:43:09 +01001151 msg_puts_title(_("\n--- Menus ---"));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001152
1153 show_menus_recursive(parent, modes, 0);
1154 return OK;
1155}
1156
1157/*
1158 * Recursively show the mappings associated with the menus under the given one
1159 */
1160 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001161show_menus_recursive(vimmenu_T *menu, int modes, int depth)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001162{
1163 int i;
1164 int bit;
1165
1166 if (menu != NULL && (menu->modes & modes) == 0x0)
1167 return;
1168
1169 if (menu != NULL)
1170 {
1171 msg_putchar('\n');
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001172 if (got_int) // "q" hit for "--more--"
Bram Moolenaar071d4272004-06-13 20:20:40 +00001173 return;
1174 for (i = 0; i < depth; i++)
Bram Moolenaar32526b32019-01-19 17:43:09 +01001175 msg_puts(" ");
Bram Moolenaar071d4272004-06-13 20:20:40 +00001176 if (menu->priority)
1177 {
1178 msg_outnum((long)menu->priority);
Bram Moolenaar32526b32019-01-19 17:43:09 +01001179 msg_puts(" ");
Bram Moolenaar071d4272004-06-13 20:20:40 +00001180 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001181 // Same highlighting as for directories!?
Bram Moolenaar8820b482017-03-16 17:23:31 +01001182 msg_outtrans_attr(menu->name, HL_ATTR(HLF_D));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001183 }
1184
1185 if (menu != NULL && menu->children == NULL)
1186 {
1187 for (bit = 0; bit < MENU_MODES; bit++)
1188 if ((menu->modes & modes & (1 << bit)) != 0)
1189 {
1190 msg_putchar('\n');
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001191 if (got_int) // "q" hit for "--more--"
Bram Moolenaar071d4272004-06-13 20:20:40 +00001192 return;
1193 for (i = 0; i < depth + 2; i++)
Bram Moolenaar32526b32019-01-19 17:43:09 +01001194 msg_puts(" ");
1195 msg_puts(menu_mode_chars[bit]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001196 if (menu->noremap[bit] == REMAP_NONE)
1197 msg_putchar('*');
1198 else if (menu->noremap[bit] == REMAP_SCRIPT)
1199 msg_putchar('&');
1200 else
1201 msg_putchar(' ');
1202 if (menu->silent[bit])
1203 msg_putchar('s');
1204 else
1205 msg_putchar(' ');
1206 if ((menu->modes & menu->enabled & (1 << bit)) == 0)
1207 msg_putchar('-');
1208 else
1209 msg_putchar(' ');
Bram Moolenaar32526b32019-01-19 17:43:09 +01001210 msg_puts(" ");
Bram Moolenaar071d4272004-06-13 20:20:40 +00001211 if (*menu->strings[bit] == NUL)
Bram Moolenaar32526b32019-01-19 17:43:09 +01001212 msg_puts_attr("<Nop>", HL_ATTR(HLF_8));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001213 else
Bram Moolenaar725310d2019-04-24 23:08:23 +02001214 msg_outtrans_special(menu->strings[bit], FALSE, 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001215 }
1216 }
1217 else
1218 {
1219 if (menu == NULL)
1220 {
1221 menu = root_menu;
1222 depth--;
1223 }
1224 else
1225 menu = menu->children;
1226
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001227 // recursively show all children. Skip PopUp[nvoci].
Bram Moolenaar071d4272004-06-13 20:20:40 +00001228 for (; menu != NULL && !got_int; menu = menu->next)
1229 if (!menu_is_hidden(menu->dname))
1230 show_menus_recursive(menu, modes, depth + 1);
1231 }
1232}
1233
Bram Moolenaar071d4272004-06-13 20:20:40 +00001234/*
1235 * Used when expanding menu names.
1236 */
1237static vimmenu_T *expand_menu = NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001238static vimmenu_T *expand_menu_alt = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001239static int expand_modes = 0x0;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001240static int expand_emenu; // TRUE for ":emenu" command
Bram Moolenaar071d4272004-06-13 20:20:40 +00001241
1242/*
1243 * Work out what to complete when doing command line completion of menu names.
1244 */
1245 char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001246set_context_in_menu_cmd(
1247 expand_T *xp,
1248 char_u *cmd,
1249 char_u *arg,
1250 int forceit)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001251{
1252 char_u *after_dot;
1253 char_u *p;
1254 char_u *path_name = NULL;
1255 char_u *name;
1256 int unmenu;
1257 vimmenu_T *menu;
1258 int expand_menus;
1259
1260 xp->xp_context = EXPAND_UNSUCCESSFUL;
1261
1262
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001263 // Check for priority numbers, enable and disable
Bram Moolenaar071d4272004-06-13 20:20:40 +00001264 for (p = arg; *p; ++p)
1265 if (!VIM_ISDIGIT(*p) && *p != '.')
1266 break;
1267
Bram Moolenaar1c465442017-03-12 20:10:05 +01001268 if (!VIM_ISWHITE(*p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001269 {
1270 if (STRNCMP(arg, "enable", 6) == 0
Bram Moolenaar1c465442017-03-12 20:10:05 +01001271 && (arg[6] == NUL || VIM_ISWHITE(arg[6])))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001272 p = arg + 6;
1273 else if (STRNCMP(arg, "disable", 7) == 0
Bram Moolenaar1c465442017-03-12 20:10:05 +01001274 && (arg[7] == NUL || VIM_ISWHITE(arg[7])))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001275 p = arg + 7;
1276 else
1277 p = arg;
1278 }
1279
Bram Moolenaar1c465442017-03-12 20:10:05 +01001280 while (*p != NUL && VIM_ISWHITE(*p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001281 ++p;
1282
1283 arg = after_dot = p;
1284
Bram Moolenaar1c465442017-03-12 20:10:05 +01001285 for (; *p && !VIM_ISWHITE(*p); ++p)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001286 {
1287 if ((*p == '\\' || *p == Ctrl_V) && p[1] != NUL)
1288 p++;
1289 else if (*p == '.')
1290 after_dot = p + 1;
1291 }
1292
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001293 // ":tearoff" and ":popup" only use menus, not entries
Bram Moolenaar071d4272004-06-13 20:20:40 +00001294 expand_menus = !((*cmd == 't' && cmd[1] == 'e') || *cmd == 'p');
1295 expand_emenu = (*cmd == 'e');
Bram Moolenaar1c465442017-03-12 20:10:05 +01001296 if (expand_menus && VIM_ISWHITE(*p))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001297 return NULL; // TODO: check for next command?
1298 if (*p == NUL) // Complete the menu name
Bram Moolenaar071d4272004-06-13 20:20:40 +00001299 {
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001300 int try_alt_menu = TRUE;
1301
Bram Moolenaar071d4272004-06-13 20:20:40 +00001302 /*
1303 * With :unmenu, you only want to match menus for the appropriate mode.
1304 * With :menu though you might want to add a menu with the same name as
1305 * one in another mode, so match menus from other modes too.
1306 */
1307 expand_modes = get_menu_cmd_modes(cmd, forceit, NULL, &unmenu);
1308 if (!unmenu)
1309 expand_modes = MENU_ALL_MODES;
1310
1311 menu = root_menu;
1312 if (after_dot != arg)
1313 {
Bram Moolenaar964b3742019-05-24 18:54:09 +02001314 path_name = alloc(after_dot - arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001315 if (path_name == NULL)
1316 return NULL;
Bram Moolenaarce0842a2005-07-18 21:58:11 +00001317 vim_strncpy(path_name, arg, after_dot - arg - 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001318 }
1319 name = path_name;
1320 while (name != NULL && *name)
1321 {
1322 p = menu_name_skip(name);
1323 while (menu != NULL)
1324 {
1325 if (menu_name_equal(name, menu))
1326 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001327 // Found menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001328 if ((*p != NUL && menu->children == NULL)
1329 || ((menu->modes & expand_modes) == 0x0))
1330 {
1331 /*
1332 * Menu path continues, but we have reached a leaf.
1333 * Or menu exists only in another mode.
1334 */
1335 vim_free(path_name);
1336 return NULL;
1337 }
1338 break;
1339 }
1340 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001341 if (menu == NULL && try_alt_menu)
1342 {
1343 menu = curwin->w_winbar;
1344 try_alt_menu = FALSE;
1345 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001346 }
1347 if (menu == NULL)
1348 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001349 // No menu found with the name we were looking for
Bram Moolenaar071d4272004-06-13 20:20:40 +00001350 vim_free(path_name);
1351 return NULL;
1352 }
1353 name = p;
1354 menu = menu->children;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001355 try_alt_menu = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001356 }
Bram Moolenaareb3593b2006-04-22 22:33:57 +00001357 vim_free(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001358
1359 xp->xp_context = expand_menus ? EXPAND_MENUNAMES : EXPAND_MENUS;
1360 xp->xp_pattern = after_dot;
1361 expand_menu = menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001362 if (expand_menu == root_menu)
1363 expand_menu_alt = curwin->w_winbar;
1364 else
1365 expand_menu_alt = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001366 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001367 else // We're in the mapping part
Bram Moolenaar071d4272004-06-13 20:20:40 +00001368 xp->xp_context = EXPAND_NOTHING;
1369 return NULL;
1370}
1371
1372/*
1373 * Function given to ExpandGeneric() to obtain the list of (sub)menus (not
1374 * entries).
1375 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001376 char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001377get_menu_name(expand_T *xp UNUSED, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001378{
1379 static vimmenu_T *menu = NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001380 static int did_alt_menu = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001381 char_u *str;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001382#ifdef FEAT_MULTI_LANG
1383 static int should_advance = FALSE;
1384#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001385
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001386 if (idx == 0) // first call: start at first item
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001387 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001388 menu = expand_menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001389 did_alt_menu = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001390#ifdef FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001391 should_advance = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001392#endif
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001393 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001394
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001395 // Skip PopUp[nvoci].
Bram Moolenaar071d4272004-06-13 20:20:40 +00001396 while (menu != NULL && (menu_is_hidden(menu->dname)
1397 || menu_is_separator(menu->dname)
1398 || menu_is_tearoff(menu->dname)
1399 || menu->children == NULL))
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001400 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001401 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001402 if (menu == NULL && !did_alt_menu)
1403 {
1404 menu = expand_menu_alt;
1405 did_alt_menu = TRUE;
1406 }
1407 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001408
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001409 if (menu == NULL) // at end of linked list
Bram Moolenaar071d4272004-06-13 20:20:40 +00001410 return NULL;
1411
1412 if (menu->modes & expand_modes)
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001413#ifdef FEAT_MULTI_LANG
1414 if (should_advance)
1415 str = menu->en_dname;
1416 else
1417 {
1418#endif
1419 str = menu->dname;
1420#ifdef FEAT_MULTI_LANG
1421 if (menu->en_dname == NULL)
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001422 should_advance = TRUE;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001423 }
1424#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001425 else
1426 str = (char_u *)"";
1427
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001428#ifdef FEAT_MULTI_LANG
1429 if (should_advance)
1430#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001431 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001432 // Advance to next menu entry.
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001433 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001434 if (menu == NULL && !did_alt_menu)
1435 {
1436 menu = expand_menu_alt;
1437 did_alt_menu = TRUE;
1438 }
1439 }
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001440
1441#ifdef FEAT_MULTI_LANG
1442 should_advance = !should_advance;
1443#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001444
1445 return str;
1446}
1447
1448/*
1449 * Function given to ExpandGeneric() to obtain the list of menus and menu
1450 * entries.
1451 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001452 char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001453get_menu_names(expand_T *xp UNUSED, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001454{
1455 static vimmenu_T *menu = NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001456 static int did_alt_menu = FALSE;
Bram Moolenaaref9d6aa2011-04-11 16:56:35 +02001457#define TBUFFER_LEN 256
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001458 static char_u tbuffer[TBUFFER_LEN]; //hack
Bram Moolenaar071d4272004-06-13 20:20:40 +00001459 char_u *str;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001460#ifdef FEAT_MULTI_LANG
1461 static int should_advance = FALSE;
1462#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001463
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001464 if (idx == 0) // first call: start at first item
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001465 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001466 menu = expand_menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001467 did_alt_menu = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001468#ifdef FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001469 should_advance = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001470#endif
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001471 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001472
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001473 // Skip Browse-style entries, popup menus and separators.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001474 while (menu != NULL
1475 && ( menu_is_hidden(menu->dname)
1476 || (expand_emenu && menu_is_separator(menu->dname))
1477 || menu_is_tearoff(menu->dname)
1478#ifndef FEAT_BROWSE
1479 || menu->dname[STRLEN(menu->dname) - 1] == '.'
1480#endif
1481 ))
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001482 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001483 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001484 if (menu == NULL && !did_alt_menu)
1485 {
1486 menu = expand_menu_alt;
1487 did_alt_menu = TRUE;
1488 }
1489 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001490
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001491 if (menu == NULL) // at end of linked list
Bram Moolenaar071d4272004-06-13 20:20:40 +00001492 return NULL;
1493
1494 if (menu->modes & expand_modes)
1495 {
1496 if (menu->children != NULL)
1497 {
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001498#ifdef FEAT_MULTI_LANG
1499 if (should_advance)
Bram Moolenaaref9d6aa2011-04-11 16:56:35 +02001500 vim_strncpy(tbuffer, menu->en_dname, TBUFFER_LEN - 2);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001501 else
1502 {
1503#endif
Bram Moolenaaref9d6aa2011-04-11 16:56:35 +02001504 vim_strncpy(tbuffer, menu->dname, TBUFFER_LEN - 2);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001505#ifdef FEAT_MULTI_LANG
1506 if (menu->en_dname == NULL)
1507 should_advance = TRUE;
1508 }
1509#endif
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001510 // hack on menu separators: use a 'magic' char for the separator
1511 // so that '.' in names gets escaped properly
Bram Moolenaar071d4272004-06-13 20:20:40 +00001512 STRCAT(tbuffer, "\001");
1513 str = tbuffer;
1514 }
1515 else
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001516#ifdef FEAT_MULTI_LANG
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001517 {
1518 if (should_advance)
1519 str = menu->en_dname;
1520 else
1521 {
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001522#endif
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001523 str = menu->dname;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001524#ifdef FEAT_MULTI_LANG
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001525 if (menu->en_dname == NULL)
1526 should_advance = TRUE;
1527 }
1528 }
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001529#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001530 }
1531 else
1532 str = (char_u *)"";
1533
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001534#ifdef FEAT_MULTI_LANG
1535 if (should_advance)
1536#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001537 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001538 // Advance to next menu entry.
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001539 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001540 if (menu == NULL && !did_alt_menu)
1541 {
1542 menu = expand_menu_alt;
1543 did_alt_menu = TRUE;
1544 }
1545 }
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001546
1547#ifdef FEAT_MULTI_LANG
1548 should_advance = !should_advance;
1549#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001550
1551 return str;
1552}
Bram Moolenaar071d4272004-06-13 20:20:40 +00001553
1554/*
1555 * Skip over this element of the menu path and return the start of the next
1556 * element. Any \ and ^Vs are removed from the current element.
Bram Moolenaar342337a2005-07-21 21:11:17 +00001557 * "name" may be modified.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001558 */
Bram Moolenaar5843f5f2019-08-20 20:13:45 +02001559 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001560menu_name_skip(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001561{
1562 char_u *p;
1563
Bram Moolenaar91acfff2017-03-12 19:22:36 +01001564 for (p = name; *p && *p != '.'; MB_PTR_ADV(p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001565 {
1566 if (*p == '\\' || *p == Ctrl_V)
1567 {
Bram Moolenaar8c8de832008-06-24 22:58:06 +00001568 STRMOVE(p, p + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001569 if (*p == NUL)
1570 break;
1571 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001572 }
1573 if (*p)
1574 *p++ = NUL;
1575 return p;
1576}
1577
1578/*
1579 * Return TRUE when "name" matches with menu "menu". The name is compared in
1580 * two ways: raw menu name and menu name without '&'. ignore part after a TAB.
1581 */
1582 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001583menu_name_equal(char_u *name, vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001584{
Bram Moolenaar41375642010-05-16 12:49:27 +02001585#ifdef FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001586 if (menu->en_name != NULL
Bram Moolenaard91f7042011-01-04 17:49:32 +01001587 && (menu_namecmp(name, menu->en_name)
1588 || menu_namecmp(name, menu->en_dname)))
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001589 return TRUE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001590#endif
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001591 return menu_namecmp(name, menu->name) || menu_namecmp(name, menu->dname);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001592}
1593
1594 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001595menu_namecmp(char_u *name, char_u *mname)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001596{
1597 int i;
1598
1599 for (i = 0; name[i] != NUL && name[i] != TAB; ++i)
1600 if (name[i] != mname[i])
1601 break;
1602 return ((name[i] == NUL || name[i] == TAB)
1603 && (mname[i] == NUL || mname[i] == TAB));
1604}
1605
1606/*
1607 * Return the modes specified by the given menu command (eg :menu! returns
1608 * MENU_CMDLINE_MODE | MENU_INSERT_MODE).
1609 * If "noremap" is not NULL, then the flag it points to is set according to
1610 * whether the command is a "nore" command.
1611 * If "unmenu" is not NULL, then the flag it points to is set according to
1612 * whether the command is an "unmenu" command.
1613 */
1614 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001615get_menu_cmd_modes(
1616 char_u *cmd,
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001617 int forceit, // Was there a "!" after the command?
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001618 int *noremap,
1619 int *unmenu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001620{
1621 int modes;
1622
1623 switch (*cmd++)
1624 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001625 case 'v': // vmenu, vunmenu, vnoremenu
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001626 modes = MENU_VISUAL_MODE | MENU_SELECT_MODE;
1627 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001628 case 'x': // xmenu, xunmenu, xnoremenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001629 modes = MENU_VISUAL_MODE;
1630 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001631 case 's': // smenu, sunmenu, snoremenu
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001632 modes = MENU_SELECT_MODE;
1633 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001634 case 'o': // omenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001635 modes = MENU_OP_PENDING_MODE;
1636 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001637 case 'i': // imenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001638 modes = MENU_INSERT_MODE;
1639 break;
1640 case 't':
Bram Moolenaar6ed545e2022-05-09 20:09:23 +01001641 if (*cmd == 'l') // tlmenu, tlunmenu, tlnoremenu
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001642 {
1643 modes = MENU_TERMINAL_MODE;
1644 ++cmd;
1645 break;
1646 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001647 modes = MENU_TIP_MODE; // tmenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001648 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001649 case 'c': // cmenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001650 modes = MENU_CMDLINE_MODE;
1651 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001652 case 'a': // amenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001653 modes = MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001654 | MENU_VISUAL_MODE | MENU_SELECT_MODE
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00001655 | MENU_OP_PENDING_MODE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001656 break;
1657 case 'n':
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001658 if (*cmd != 'o') // nmenu, not noremenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001659 {
1660 modes = MENU_NORMAL_MODE;
1661 break;
1662 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001663 // FALLTHROUGH
Bram Moolenaar071d4272004-06-13 20:20:40 +00001664 default:
1665 --cmd;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001666 if (forceit) // menu!!
Bram Moolenaar071d4272004-06-13 20:20:40 +00001667 modes = MENU_INSERT_MODE | MENU_CMDLINE_MODE;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001668 else // menu
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001669 modes = MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE
Bram Moolenaar071d4272004-06-13 20:20:40 +00001670 | MENU_OP_PENDING_MODE;
1671 }
1672
1673 if (noremap != NULL)
1674 *noremap = (*cmd == 'n' ? REMAP_NONE : REMAP_YES);
1675 if (unmenu != NULL)
1676 *unmenu = (*cmd == 'u');
1677 return modes;
1678}
1679
1680/*
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01001681 * Return the string representation of the menu modes. Does the opposite
1682 * of get_menu_cmd_modes().
1683 */
1684 static char_u *
1685get_menu_mode_str(int modes)
1686{
1687 if ((modes & (MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE |
1688 MENU_VISUAL_MODE | MENU_SELECT_MODE | MENU_OP_PENDING_MODE))
1689 == (MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE |
1690 MENU_VISUAL_MODE | MENU_SELECT_MODE | MENU_OP_PENDING_MODE))
1691 return (char_u *)"a";
1692 if ((modes & (MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE |
1693 MENU_OP_PENDING_MODE))
1694 == (MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE |
1695 MENU_OP_PENDING_MODE))
1696 return (char_u *)" ";
1697 if ((modes & (MENU_INSERT_MODE | MENU_CMDLINE_MODE))
1698 == (MENU_INSERT_MODE | MENU_CMDLINE_MODE))
1699 return (char_u *)"!";
1700 if ((modes & (MENU_VISUAL_MODE | MENU_SELECT_MODE))
1701 == (MENU_VISUAL_MODE | MENU_SELECT_MODE))
1702 return (char_u *)"v";
1703 if (modes & MENU_VISUAL_MODE)
1704 return (char_u *)"x";
1705 if (modes & MENU_SELECT_MODE)
1706 return (char_u *)"s";
1707 if (modes & MENU_OP_PENDING_MODE)
1708 return (char_u *)"o";
1709 if (modes & MENU_INSERT_MODE)
1710 return (char_u *)"i";
1711 if (modes & MENU_TERMINAL_MODE)
1712 return (char_u *)"tl";
1713 if (modes & MENU_CMDLINE_MODE)
1714 return (char_u *)"c";
1715 if (modes & MENU_NORMAL_MODE)
1716 return (char_u *)"n";
1717 if (modes & MENU_TIP_MODE)
1718 return (char_u *)"t";
1719
1720 return (char_u *)"";
1721}
1722
1723/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00001724 * Modify a menu name starting with "PopUp" to include the mode character.
1725 * Returns the name in allocated memory (NULL for failure).
1726 */
1727 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001728popup_mode_name(char_u *name, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001729{
1730 char_u *p;
1731 int len = (int)STRLEN(name);
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001732 char *mode_chars = menu_mode_chars[idx];
1733 int mode_chars_len = (int)strlen(mode_chars);
1734 int i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001735
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001736 p = vim_strnsave(name, len + mode_chars_len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001737 if (p != NULL)
1738 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001739 mch_memmove(p + 5 + mode_chars_len, p + 5, (size_t)(len - 4));
1740 for (i = 0; i < mode_chars_len; ++i)
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001741 p[5 + i] = menu_mode_chars[idx][i];
Bram Moolenaar071d4272004-06-13 20:20:40 +00001742 }
1743 return p;
1744}
1745
1746#if defined(FEAT_GUI) || defined(PROTO)
1747/*
1748 * Return the index into the menu->strings or menu->noremap arrays for the
1749 * current state. Returns MENU_INDEX_INVALID if there is no mapping for the
1750 * given menu in the current mode.
1751 */
1752 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001753get_menu_index(vimmenu_T *menu, int state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001754{
1755 int idx;
1756
Bram Moolenaar24959102022-05-07 20:01:16 +01001757 if ((state & MODE_INSERT))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001758 idx = MENU_INDEX_INSERT;
Bram Moolenaar24959102022-05-07 20:01:16 +01001759 else if (state & MODE_CMDLINE)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001760 idx = MENU_INDEX_CMDLINE;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001761#ifdef FEAT_TERMINAL
1762 else if (term_use_loop())
1763 idx = MENU_INDEX_TERMINAL;
1764#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001765 else if (VIsual_active)
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001766 {
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00001767 if (VIsual_select)
1768 idx = MENU_INDEX_SELECT;
1769 else
1770 idx = MENU_INDEX_VISUAL;
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001771 }
Bram Moolenaar24959102022-05-07 20:01:16 +01001772 else if (state == MODE_HITRETURN || state == MODE_ASKMORE)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001773 idx = MENU_INDEX_CMDLINE;
1774 else if (finish_op)
1775 idx = MENU_INDEX_OP_PENDING;
Bram Moolenaar24959102022-05-07 20:01:16 +01001776 else if ((state & MODE_NORMAL))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001777 idx = MENU_INDEX_NORMAL;
1778 else
1779 idx = MENU_INDEX_INVALID;
1780
1781 if (idx != MENU_INDEX_INVALID && menu->strings[idx] == NULL)
1782 idx = MENU_INDEX_INVALID;
1783 return idx;
1784}
1785#endif
1786
1787/*
1788 * Duplicate the menu item text and then process to see if a mnemonic key
1789 * and/or accelerator text has been identified.
1790 * Returns a pointer to allocated memory, or NULL for failure.
1791 * If mnemonic != NULL, *mnemonic is set to the character after the first '&'.
1792 * If actext != NULL, *actext is set to the text after the first TAB.
1793 */
1794 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001795menu_text(char_u *str, int *mnemonic, char_u **actext)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001796{
1797 char_u *p;
1798 char_u *text;
1799
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001800 // Locate accelerator text, after the first TAB
Bram Moolenaar071d4272004-06-13 20:20:40 +00001801 p = vim_strchr(str, TAB);
1802 if (p != NULL)
1803 {
1804 if (actext != NULL)
1805 *actext = vim_strsave(p + 1);
Bram Moolenaar71ccd032020-06-12 22:59:11 +02001806 text = vim_strnsave(str, p - str);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001807 }
1808 else
1809 text = vim_strsave(str);
1810
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001811 // Find mnemonic characters "&a" and reduce "&&" to "&".
Bram Moolenaar071d4272004-06-13 20:20:40 +00001812 for (p = text; p != NULL; )
1813 {
1814 p = vim_strchr(p, '&');
1815 if (p != NULL)
1816 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001817 if (p[1] == NUL) // trailing "&"
Bram Moolenaar071d4272004-06-13 20:20:40 +00001818 break;
1819 if (mnemonic != NULL && p[1] != '&')
1820#if !defined(__MVS__) || defined(MOTIF390_MNEMONIC_FIXED)
1821 *mnemonic = p[1];
1822#else
1823 {
1824 /*
1825 * Well there is a bug in the Motif libraries on OS390 Unix.
1826 * The mnemonic keys needs to be converted to ASCII values
1827 * first.
1828 * This behavior has been seen in 2.8 and 2.9.
1829 */
1830 char c = p[1];
1831 __etoa_l(&c, 1);
1832 *mnemonic = c;
1833 }
1834#endif
Bram Moolenaar8c8de832008-06-24 22:58:06 +00001835 STRMOVE(p, p + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001836 p = p + 1;
1837 }
1838 }
1839 return text;
1840}
1841
1842/*
1843 * Return TRUE if "name" can be a menu in the MenuBar.
1844 */
1845 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001846menu_is_menubar(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001847{
1848 return (!menu_is_popup(name)
1849 && !menu_is_toolbar(name)
Bram Moolenaar378daf82017-09-23 23:58:28 +02001850 && !menu_is_winbar(name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001851 && *name != MNU_HIDDEN_CHAR);
1852}
1853
1854/*
1855 * Return TRUE if "name" is a popup menu name.
1856 */
1857 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001858menu_is_popup(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001859{
1860 return (STRNCMP(name, "PopUp", 5) == 0);
1861}
1862
1863#if (defined(FEAT_GUI_MOTIF) && (XmVersion <= 1002)) || defined(PROTO)
1864/*
1865 * Return TRUE if "name" is part of a popup menu.
1866 */
1867 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001868menu_is_child_of_popup(vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001869{
1870 while (menu->parent != NULL)
1871 menu = menu->parent;
1872 return menu_is_popup(menu->name);
1873}
1874#endif
1875
1876/*
1877 * Return TRUE if "name" is a toolbar menu name.
1878 */
1879 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001880menu_is_toolbar(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001881{
1882 return (STRNCMP(name, "ToolBar", 7) == 0);
1883}
1884
1885/*
1886 * Return TRUE if the name is a menu separator identifier: Starts and ends
1887 * with '-'
1888 */
1889 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001890menu_is_separator(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001891{
1892 return (name[0] == '-' && name[STRLEN(name) - 1] == '-');
1893}
1894
1895/*
1896 * Return TRUE if the menu is hidden: Starts with ']'
1897 */
1898 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001899menu_is_hidden(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001900{
1901 return (name[0] == ']') || (menu_is_popup(name) && name[5] != NUL);
1902}
1903
Bram Moolenaar071d4272004-06-13 20:20:40 +00001904/*
1905 * Return TRUE if the menu is the tearoff menu.
1906 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001907 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001908menu_is_tearoff(char_u *name UNUSED)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001909{
1910#ifdef FEAT_GUI
1911 return (STRCMP(name, TEAR_STRING) == 0);
1912#else
1913 return FALSE;
1914#endif
1915}
Bram Moolenaar071d4272004-06-13 20:20:40 +00001916
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001917#if defined(FEAT_GUI) || defined(FEAT_TERM_POPUP_MENU) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001918
1919 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001920get_menu_mode(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001921{
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001922#ifdef FEAT_TERMINAL
1923 if (term_use_loop())
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001924 return MENU_INDEX_TERMINAL;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001925#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001926 if (VIsual_active)
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001927 {
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00001928 if (VIsual_select)
1929 return MENU_INDEX_SELECT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001930 return MENU_INDEX_VISUAL;
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001931 }
Bram Moolenaar24959102022-05-07 20:01:16 +01001932 if (State & MODE_INSERT)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001933 return MENU_INDEX_INSERT;
Bram Moolenaar24959102022-05-07 20:01:16 +01001934 if ((State & MODE_CMDLINE) || State == MODE_ASKMORE
1935 || State == MODE_HITRETURN)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001936 return MENU_INDEX_CMDLINE;
1937 if (finish_op)
1938 return MENU_INDEX_OP_PENDING;
Bram Moolenaar24959102022-05-07 20:01:16 +01001939 if (State & MODE_NORMAL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001940 return MENU_INDEX_NORMAL;
Bram Moolenaar24959102022-05-07 20:01:16 +01001941 if (State & MODE_LANGMAP) // must be a "r" command, like Insert mode
Bram Moolenaar071d4272004-06-13 20:20:40 +00001942 return MENU_INDEX_INSERT;
1943 return MENU_INDEX_INVALID;
1944}
1945
Bram Moolenaar29a2c082018-03-05 21:06:23 +01001946 int
1947get_menu_mode_flag(void)
1948{
1949 int mode = get_menu_mode();
1950
1951 if (mode == MENU_INDEX_INVALID)
1952 return 0;
1953 return 1 << mode;
1954}
1955
Bram Moolenaar071d4272004-06-13 20:20:40 +00001956/*
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001957 * Display the Special "PopUp" menu as a pop-up at the current mouse
1958 * position. The "PopUpn" menu is for Normal mode, "PopUpi" for Insert mode,
1959 * etc.
1960 */
1961 void
1962show_popupmenu(void)
1963{
1964 vimmenu_T *menu;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001965 int menu_mode;
1966 char* mode;
1967 int mode_len;
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001968
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001969 menu_mode = get_menu_mode();
1970 if (menu_mode == MENU_INDEX_INVALID)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001971 return;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001972 mode = menu_mode_chars[menu_mode];
1973 mode_len = (int)strlen(mode);
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001974
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001975 apply_autocmds(EVENT_MENUPOPUP, (char_u*)mode, NULL, FALSE, curbuf);
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001976
Bram Moolenaar00d253e2020-04-06 22:13:01 +02001977 FOR_ALL_MENUS(menu)
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001978 if (STRNCMP("PopUp", menu->name, 5) == 0 && STRNCMP(menu->name + 5, mode, mode_len) == 0)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001979 break;
1980
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001981 // Only show a popup when it is defined and has entries
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001982 if (menu != NULL && menu->children != NULL)
1983 {
1984# if defined(FEAT_GUI)
1985 if (gui.in_use)
1986 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001987 // Update the menus now, in case the MenuPopup autocommand did
1988 // anything.
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001989 gui_update_menus(0);
1990 gui_mch_show_popupmenu(menu);
1991 }
1992# endif
1993# if defined(FEAT_GUI) && defined(FEAT_TERM_POPUP_MENU)
1994 else
1995# endif
1996# if defined(FEAT_TERM_POPUP_MENU)
1997 pum_show_popupmenu(menu);
1998# endif
1999 }
2000}
2001#endif
2002
2003#if defined(FEAT_GUI) || defined(PROTO)
2004
2005/*
Bram Moolenaar968bbbe2006-08-16 19:41:08 +00002006 * Check that a pointer appears in the menu tree. Used to protect from using
2007 * a menu that was deleted after it was selected but before the event was
2008 * handled.
2009 * Return OK or FAIL. Used recursively.
2010 */
2011 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002012check_menu_pointer(vimmenu_T *root, vimmenu_T *menu_to_check)
Bram Moolenaar968bbbe2006-08-16 19:41:08 +00002013{
2014 vimmenu_T *p;
2015
2016 for (p = root; p != NULL; p = p->next)
2017 if (p == menu_to_check
2018 || (p->children != NULL
2019 && check_menu_pointer(p->children, menu_to_check) == OK))
2020 return OK;
2021 return FAIL;
2022}
2023
2024/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00002025 * After we have started the GUI, then we can create any menus that have been
2026 * defined. This is done once here. add_menu_path() may have already been
2027 * called to define these menus, and may be called again. This function calls
2028 * itself recursively. Should be called at the top level with:
Bram Moolenaara06ecab2016-07-16 14:47:36 +02002029 * gui_create_initial_menus(root_menu);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002030 */
2031 void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002032gui_create_initial_menus(vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002033{
2034 int idx = 0;
2035
2036 while (menu != NULL)
2037 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002038 // Don't add a menu when only a tip was defined.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002039 if (menu->modes & MENU_ALL_MODES)
2040 {
2041 if (menu->children != NULL)
2042 {
2043 gui_mch_add_menu(menu, idx);
2044 gui_create_initial_menus(menu->children);
2045 }
2046 else
2047 gui_mch_add_menu_item(menu, idx);
2048 }
2049 menu = menu->next;
2050 ++idx;
2051 }
2052}
2053
2054/*
2055 * Used recursively by gui_update_menus (see below)
2056 */
2057 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002058gui_update_menus_recurse(vimmenu_T *menu, int mode)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002059{
2060 int grey;
2061
2062 while (menu)
2063 {
2064 if ((menu->modes & menu->enabled & mode)
Bram Moolenaar4f974752019-02-17 17:44:42 +01002065# if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002066 || menu_is_tearoff(menu->dname)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002067# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002068 )
2069 grey = FALSE;
2070 else
2071 grey = TRUE;
Bram Moolenaar0b962e52022-04-03 18:02:37 +01002072
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002073 // Never hide a toplevel menu, it may make the menubar resize or
2074 // disappear. Same problem for ToolBar items.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002075 if (vim_strchr(p_go, GO_GREY) != NULL || menu->parent == NULL
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002076# ifdef FEAT_TOOLBAR
Bram Moolenaar071d4272004-06-13 20:20:40 +00002077 || menu_is_toolbar(menu->parent->name)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002078# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002079 )
2080 gui_mch_menu_grey(menu, grey);
2081 else
2082 gui_mch_menu_hidden(menu, grey);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002083 gui_update_menus_recurse(menu->children, mode);
2084 menu = menu->next;
2085 }
2086}
2087
2088/*
2089 * Make sure only the valid menu items appear for this mode. If
2090 * force_menu_update is not TRUE, then we only do this if the mode has changed
2091 * since last time. If "modes" is not 0, then we use these modes instead.
2092 */
2093 void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002094gui_update_menus(int modes)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002095{
2096 static int prev_mode = -1;
2097 int mode = 0;
2098
2099 if (modes != 0x0)
2100 mode = modes;
2101 else
Bram Moolenaar29a2c082018-03-05 21:06:23 +01002102 mode = get_menu_mode_flag();
Bram Moolenaar071d4272004-06-13 20:20:40 +00002103
2104 if (force_menu_update || mode != prev_mode)
2105 {
2106 gui_update_menus_recurse(root_menu, mode);
2107 gui_mch_draw_menubar();
2108 prev_mode = mode;
2109 force_menu_update = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002110 }
2111}
2112
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002113# if defined(FEAT_GUI_MSWIN) || defined(FEAT_GUI_MOTIF) \
Bram Moolenaar241a8aa2005-12-06 20:04:44 +00002114 || defined(FEAT_GUI_GTK) || defined(FEAT_GUI_PHOTON) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002115/*
2116 * Check if a key is used as a mnemonic for a toplevel menu.
2117 * Case of the key is ignored.
2118 */
2119 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002120gui_is_menu_shortcut(int key)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002121{
2122 vimmenu_T *menu;
2123
2124 if (key < 256)
2125 key = TOLOWER_LOC(key);
Bram Moolenaar00d253e2020-04-06 22:13:01 +02002126 FOR_ALL_MENUS(menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002127 if (menu->mnemonic == key
2128 || (menu->mnemonic < 256 && TOLOWER_LOC(menu->mnemonic) == key))
2129 return TRUE;
2130 return FALSE;
2131}
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002132# endif
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002133#endif // FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00002134
Bram Moolenaar4f974752019-02-17 17:44:42 +01002135#if (defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002136
2137/*
2138 * Deal with tearoff items that are added like a menu item.
2139 * Currently only for Win32 GUI. Others may follow later.
2140 */
2141
2142 void
2143gui_mch_toggle_tearoffs(int enable)
2144{
2145 int pri_tab[MENUDEPTH + 1];
2146 int i;
2147
2148 if (enable)
2149 {
2150 for (i = 0; i < MENUDEPTH; ++i)
2151 pri_tab[i] = 500;
2152 pri_tab[MENUDEPTH] = -1;
2153 gui_create_tearoffs_recurse(root_menu, (char_u *)"", pri_tab, 0);
2154 }
2155 else
2156 gui_destroy_tearoffs_recurse(root_menu);
2157 s_tearoffs = enable;
2158}
2159
2160/*
2161 * Recursively add tearoff items
2162 */
2163 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002164gui_create_tearoffs_recurse(
2165 vimmenu_T *menu,
2166 const char_u *pname,
2167 int *pri_tab,
2168 int pri_idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002169{
2170 char_u *newpname = NULL;
2171 int len;
2172 char_u *s;
2173 char_u *d;
2174
2175 if (pri_tab[pri_idx + 1] != -1)
2176 ++pri_idx;
2177 while (menu != NULL)
2178 {
2179 if (menu->children != NULL && menu_is_menubar(menu->name))
2180 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002181 // Add the menu name to the menu path. Insert a backslash before
2182 // dots (it's used to separate menu names).
Bram Moolenaar071d4272004-06-13 20:20:40 +00002183 len = (int)STRLEN(pname) + (int)STRLEN(menu->name);
2184 for (s = menu->name; *s; ++s)
2185 if (*s == '.' || *s == '\\')
2186 ++len;
2187 newpname = alloc(len + TEAR_LEN + 2);
2188 if (newpname != NULL)
2189 {
2190 STRCPY(newpname, pname);
2191 d = newpname + STRLEN(newpname);
2192 for (s = menu->name; *s; ++s)
2193 {
2194 if (*s == '.' || *s == '\\')
2195 *d++ = '\\';
2196 *d++ = *s;
2197 }
2198 *d = NUL;
2199
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002200 // check if tearoff already exists
Bram Moolenaar071d4272004-06-13 20:20:40 +00002201 if (STRCMP(menu->children->name, TEAR_STRING) != 0)
2202 {
2203 gui_add_tearoff(newpname, pri_tab, pri_idx - 1);
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002204 *d = NUL; // remove TEAR_STRING
Bram Moolenaar071d4272004-06-13 20:20:40 +00002205 }
2206
2207 STRCAT(newpname, ".");
2208 gui_create_tearoffs_recurse(menu->children, newpname,
2209 pri_tab, pri_idx);
2210 vim_free(newpname);
2211 }
2212 }
2213 menu = menu->next;
2214 }
2215}
2216
2217/*
2218 * Add tear-off menu item for a submenu.
2219 * "tearpath" is the menu path, and must have room to add TEAR_STRING.
2220 */
2221 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002222gui_add_tearoff(char_u *tearpath, int *pri_tab, int pri_idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002223{
2224 char_u *tbuf;
2225 int t;
2226 vimmenu_T menuarg;
2227
2228 tbuf = alloc(5 + (unsigned int)STRLEN(tearpath));
2229 if (tbuf != NULL)
2230 {
2231 tbuf[0] = K_SPECIAL;
2232 tbuf[1] = K_SECOND(K_TEAROFF);
2233 tbuf[2] = K_THIRD(K_TEAROFF);
2234 STRCPY(tbuf + 3, tearpath);
2235 STRCAT(tbuf + 3, "\r");
2236
2237 STRCAT(tearpath, ".");
2238 STRCAT(tearpath, TEAR_STRING);
2239
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002240 // Priority of tear-off is always 1
Bram Moolenaar071d4272004-06-13 20:20:40 +00002241 t = pri_tab[pri_idx + 1];
2242 pri_tab[pri_idx + 1] = 1;
2243
2244#ifdef FEAT_TOOLBAR
2245 menuarg.iconfile = NULL;
2246 menuarg.iconidx = -1;
2247 menuarg.icon_builtin = FALSE;
2248#endif
2249 menuarg.noremap[0] = REMAP_NONE;
2250 menuarg.silent[0] = TRUE;
2251
2252 menuarg.modes = MENU_ALL_MODES;
2253 add_menu_path(tearpath, &menuarg, pri_tab, tbuf, FALSE);
2254
2255 menuarg.modes = MENU_TIP_MODE;
2256 add_menu_path(tearpath, &menuarg, pri_tab,
2257 (char_u *)_("Tear off this menu"), FALSE);
2258
2259 pri_tab[pri_idx + 1] = t;
2260 vim_free(tbuf);
2261 }
2262}
2263
2264/*
2265 * Recursively destroy tearoff items
2266 */
2267 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002268gui_destroy_tearoffs_recurse(vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002269{
2270 while (menu)
2271 {
2272 if (menu->children)
2273 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002274 // check if tearoff exists
Bram Moolenaar071d4272004-06-13 20:20:40 +00002275 if (STRCMP(menu->children->name, TEAR_STRING) == 0)
2276 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002277 // Disconnect the item and free the memory
Bram Moolenaar071d4272004-06-13 20:20:40 +00002278 free_menu(&menu->children);
2279 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002280 if (menu->children != NULL) // if not the last one
Bram Moolenaar071d4272004-06-13 20:20:40 +00002281 gui_destroy_tearoffs_recurse(menu->children);
2282 }
2283 menu = menu->next;
2284 }
2285}
2286
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002287#endif // FEAT_GUI_MSWIN && FEAT_TEAROFF
Bram Moolenaar071d4272004-06-13 20:20:40 +00002288
2289/*
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002290 * Execute "menu". Use by ":emenu" and the window toolbar.
2291 * "eap" is NULL for the window toolbar.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002292 * "mode_idx" specifies a MENU_INDEX_ value, use -1 to depend on the current
2293 * state.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002294 */
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002295 void
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002296execute_menu(exarg_T *eap, vimmenu_T *menu, int mode_idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002297{
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002298 int idx = mode_idx;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002299
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002300 if (idx < 0)
2301 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002302 // Use the Insert mode entry when returning to Insert mode.
Bram Moolenaar9b8d6222020-12-28 18:26:00 +01002303 if (restart_edit && !current_sctx.sc_sid)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002304 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002305 idx = MENU_INDEX_INSERT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002306 }
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002307#ifdef FEAT_TERMINAL
2308 else if (term_use_loop())
Bram Moolenaar071d4272004-06-13 20:20:40 +00002309 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002310 idx = MENU_INDEX_TERMINAL;
2311 }
2312#endif
2313 else if (VIsual_active)
2314 {
2315 idx = MENU_INDEX_VISUAL;
2316 }
2317 else if (eap != NULL && eap->addr_count)
2318 {
2319 pos_T tpos;
2320
2321 idx = MENU_INDEX_VISUAL;
2322
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002323 // GEDDES: This is not perfect - but it is a
2324 // quick way of detecting whether we are doing this from a
2325 // selection - see if the range matches up with the visual
2326 // select start and end.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002327 if ((curbuf->b_visual.vi_start.lnum == eap->line1)
2328 && (curbuf->b_visual.vi_end.lnum) == eap->line2)
2329 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002330 // Set it up for visual mode - equivalent to gv.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002331 VIsual_mode = curbuf->b_visual.vi_mode;
2332 tpos = curbuf->b_visual.vi_end;
2333 curwin->w_cursor = curbuf->b_visual.vi_start;
2334 curwin->w_curswant = curbuf->b_visual.vi_curswant;
2335 }
2336 else
2337 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002338 // Set it up for line-wise visual mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002339 VIsual_mode = 'V';
2340 curwin->w_cursor.lnum = eap->line1;
2341 curwin->w_cursor.col = 1;
2342 tpos.lnum = eap->line2;
2343 tpos.col = MAXCOL;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002344 tpos.coladd = 0;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002345 }
2346
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002347 // Activate visual mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002348 VIsual_active = TRUE;
2349 VIsual_reselect = TRUE;
2350 check_cursor();
2351 VIsual = curwin->w_cursor;
2352 curwin->w_cursor = tpos;
2353
2354 check_cursor();
2355
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002356 // Adjust the cursor to make sure it is in the correct pos
2357 // for exclusive mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002358 if (*p_sel == 'e' && gchar_cursor() != NUL)
2359 ++curwin->w_cursor.col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002360 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002361 }
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002362
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002363 // For the WinBar menu always use the Normal mode menu.
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002364 if (idx == -1 || eap == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002365 idx = MENU_INDEX_NORMAL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002366
Bram Moolenaarce793532019-05-05 14:19:20 +02002367 if (idx != MENU_INDEX_INVALID && menu->strings[idx] != NULL
2368 && (menu->modes & (1 << idx)))
Bram Moolenaar071d4272004-06-13 20:20:40 +00002369 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002370 // When executing a script or function execute the commands right now.
2371 // Also for the window toolbar.
2372 // Otherwise put them in the typeahead buffer.
Bram Moolenaar9b8d6222020-12-28 18:26:00 +01002373 if (eap == NULL || current_sctx.sc_sid != 0)
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002374 {
2375 save_state_T save_state;
2376
2377 ++ex_normal_busy;
2378 if (save_current_state(&save_state))
2379 exec_normal_cmd(menu->strings[idx], menu->noremap[idx],
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00002380 menu->silent[idx]);
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002381 restore_current_state(&save_state);
2382 --ex_normal_busy;
2383 }
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00002384 else
2385 ins_typebuf(menu->strings[idx], menu->noremap[idx], 0,
Bram Moolenaar071d4272004-06-13 20:20:40 +00002386 TRUE, menu->silent[idx]);
2387 }
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002388 else if (eap != NULL)
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002389 {
2390 char_u *mode;
2391
2392 switch (idx)
2393 {
2394 case MENU_INDEX_VISUAL:
2395 mode = (char_u *)"Visual";
2396 break;
2397 case MENU_INDEX_SELECT:
2398 mode = (char_u *)"Select";
2399 break;
2400 case MENU_INDEX_OP_PENDING:
2401 mode = (char_u *)"Op-pending";
2402 break;
2403 case MENU_INDEX_TERMINAL:
2404 mode = (char_u *)"Terminal";
2405 break;
2406 case MENU_INDEX_INSERT:
2407 mode = (char_u *)"Insert";
2408 break;
2409 case MENU_INDEX_CMDLINE:
2410 mode = (char_u *)"Cmdline";
2411 break;
2412 // case MENU_INDEX_TIP: cannot happen
2413 default:
2414 mode = (char_u *)"Normal";
2415 }
Bram Moolenaareaaac012022-01-02 17:00:40 +00002416 semsg(_(e_menu_not_defined_for_str_mode), mode);
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002417 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002418}
2419
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002420/*
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002421 * Lookup a menu by the descriptor name e.g. "File.New"
2422 * Returns NULL if the menu is not found
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002423 */
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002424 static vimmenu_T *
2425menu_getbyname(char_u *name_arg)
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002426{
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002427 char_u *name;
2428 char_u *saved_name;
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002429 vimmenu_T *menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002430 char_u *p;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002431 int gave_emsg = FALSE;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002432
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002433 saved_name = vim_strsave(name_arg);
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002434 if (saved_name == NULL)
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002435 return NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002436
2437 menu = *get_root_menu(saved_name);
2438 name = saved_name;
2439 while (*name)
2440 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002441 // Find in the menu hierarchy
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002442 p = menu_name_skip(name);
2443
2444 while (menu != NULL)
2445 {
2446 if (menu_name_equal(name, menu))
2447 {
2448 if (*p == NUL && menu->children != NULL)
2449 {
Bram Moolenaareaaac012022-01-02 17:00:40 +00002450 emsg(_(e_menu_path_must_lead_to_menu_item));
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002451 gave_emsg = TRUE;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002452 menu = NULL;
2453 }
2454 else if (*p != NUL && menu->children == NULL)
2455 {
Bram Moolenaareaaac012022-01-02 17:00:40 +00002456 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002457 menu = NULL;
2458 }
2459 break;
2460 }
2461 menu = menu->next;
2462 }
2463 if (menu == NULL || *p == NUL)
2464 break;
2465 menu = menu->children;
2466 name = p;
2467 }
2468 vim_free(saved_name);
2469 if (menu == NULL)
2470 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002471 if (!gave_emsg)
Bram Moolenaareaaac012022-01-02 17:00:40 +00002472 semsg(_(e_menu_not_found_str), name_arg);
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002473 return NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002474 }
2475
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002476 return menu;
2477}
2478
2479/*
2480 * Given a menu descriptor, e.g. "File.New", find it in the menu hierarchy and
2481 * execute it.
2482 */
2483 void
2484ex_emenu(exarg_T *eap)
2485{
2486 vimmenu_T *menu;
2487 char_u *arg = eap->arg;
2488 int mode_idx = -1;
2489
2490 if (arg[0] && VIM_ISWHITE(arg[1]))
2491 {
2492 switch (arg[0])
2493 {
2494 case 'n': mode_idx = MENU_INDEX_NORMAL; break;
2495 case 'v': mode_idx = MENU_INDEX_VISUAL; break;
2496 case 's': mode_idx = MENU_INDEX_SELECT; break;
2497 case 'o': mode_idx = MENU_INDEX_OP_PENDING; break;
2498 case 't': mode_idx = MENU_INDEX_TERMINAL; break;
2499 case 'i': mode_idx = MENU_INDEX_INSERT; break;
2500 case 'c': mode_idx = MENU_INDEX_CMDLINE; break;
Bram Moolenaar436b5ad2021-12-31 22:49:24 +00002501 default: semsg(_(e_invalid_argument_str), arg);
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002502 return;
2503 }
2504 arg = skipwhite(arg + 2);
2505 }
2506
2507 menu = menu_getbyname(arg);
2508 if (menu == NULL)
2509 return;
2510
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002511 // Found the menu, so execute.
2512 execute_menu(eap, menu, mode_idx);
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002513}
2514
2515/*
2516 * Handle a click in the window toolbar of "wp" at column "col".
2517 */
2518 void
2519winbar_click(win_T *wp, int col)
2520{
2521 int idx;
2522
2523 if (wp->w_winbar_items == NULL)
2524 return;
2525 for (idx = 0; wp->w_winbar_items[idx].wb_menu != NULL; ++idx)
2526 {
2527 winbar_item_T *item = &wp->w_winbar_items[idx];
2528
2529 if (col >= item->wb_startcol && col <= item->wb_endcol)
2530 {
Bram Moolenaard2fad672019-05-04 16:55:25 +02002531 win_T *save_curwin = NULL;
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002532 pos_T save_visual = VIsual;
2533 int save_visual_active = VIsual_active;
2534 int save_visual_select = VIsual_select;
2535 int save_visual_reselect = VIsual_reselect;
2536 int save_visual_mode = VIsual_mode;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002537
2538 if (wp != curwin)
2539 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002540 // Clicking in the window toolbar of a not-current window.
2541 // Make that window the current one and save Visual mode.
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002542 save_curwin = curwin;
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002543 VIsual_active = FALSE;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002544 curwin = wp;
2545 curbuf = curwin->w_buffer;
2546 check_cursor();
2547 }
2548
Bram Moolenaard2fad672019-05-04 16:55:25 +02002549 // Note: the command might close the current window.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002550 execute_menu(NULL, item->wb_menu, -1);
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002551
Bram Moolenaard2fad672019-05-04 16:55:25 +02002552 if (save_curwin != NULL && win_valid(save_curwin))
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002553 {
2554 curwin = save_curwin;
2555 curbuf = curwin->w_buffer;
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002556 VIsual = save_visual;
2557 VIsual_active = save_visual_active;
2558 VIsual_select = save_visual_select;
2559 VIsual_reselect = save_visual_reselect;
2560 VIsual_mode = save_visual_mode;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002561 }
Bram Moolenaard2fad672019-05-04 16:55:25 +02002562 if (!win_valid(wp))
2563 break;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002564 }
2565 }
2566}
2567
2568#if defined(FEAT_GUI_MSWIN) || defined(FEAT_GUI_GTK) \
Bram Moolenaarb3f74062020-02-26 16:16:53 +01002569 || defined(FEAT_TERM_POPUP_MENU) || defined(FEAT_GUI_HAIKU) \
Bram Moolenaar071d4272004-06-13 20:20:40 +00002570 || defined(FEAT_BEVAL_TIP) || defined(PROTO)
2571/*
2572 * Given a menu descriptor, e.g. "File.New", find it in the menu hierarchy.
2573 */
2574 vimmenu_T *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002575gui_find_menu(char_u *path_name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002576{
2577 vimmenu_T *menu = NULL;
2578 char_u *name;
2579 char_u *saved_name;
2580 char_u *p;
2581
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002582 menu = *get_root_menu(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002583
2584 saved_name = vim_strsave(path_name);
2585 if (saved_name == NULL)
2586 return NULL;
2587
2588 name = saved_name;
2589 while (*name)
2590 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002591 // find the end of one dot-separated name and put a NUL at the dot
Bram Moolenaar071d4272004-06-13 20:20:40 +00002592 p = menu_name_skip(name);
2593
2594 while (menu != NULL)
2595 {
Bram Moolenaard91f7042011-01-04 17:49:32 +01002596 if (menu_name_equal(name, menu))
Bram Moolenaar071d4272004-06-13 20:20:40 +00002597 {
2598 if (menu->children == NULL)
2599 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002600 // found a menu item instead of a sub-menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00002601 if (*p == NUL)
Bram Moolenaareaaac012022-01-02 17:00:40 +00002602 emsg(_(e_menu_path_must_lead_to_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +00002603 else
Bram Moolenaareaaac012022-01-02 17:00:40 +00002604 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +00002605 menu = NULL;
2606 goto theend;
2607 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002608 if (*p == NUL) // found a full match
Bram Moolenaar071d4272004-06-13 20:20:40 +00002609 goto theend;
2610 break;
2611 }
2612 menu = menu->next;
2613 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002614 if (menu == NULL) // didn't find it
Bram Moolenaar071d4272004-06-13 20:20:40 +00002615 break;
2616
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002617 // Found a match, search the sub-menu.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002618 menu = menu->children;
2619 name = p;
2620 }
2621
2622 if (menu == NULL)
Bram Moolenaareaaac012022-01-02 17:00:40 +00002623 emsg(_(e_menu_not_found_check_menu_names));
Bram Moolenaar071d4272004-06-13 20:20:40 +00002624theend:
2625 vim_free(saved_name);
2626 return menu;
2627}
2628#endif
2629
2630#ifdef FEAT_MULTI_LANG
2631/*
2632 * Translation of menu names. Just a simple lookup table.
2633 */
2634
2635typedef struct
2636{
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002637 char_u *from; // English name
2638 char_u *from_noamp; // same, without '&'
2639 char_u *to; // translated name
Bram Moolenaar071d4272004-06-13 20:20:40 +00002640} menutrans_T;
2641
2642static garray_T menutrans_ga = {0, 0, 0, 0, NULL};
2643#endif
2644
2645/*
2646 * ":menutrans".
2647 * This function is also defined without the +multi_lang feature, in which
2648 * case the commands are ignored.
2649 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002650 void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002651ex_menutranslate(exarg_T *eap UNUSED)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002652{
2653#ifdef FEAT_MULTI_LANG
2654 char_u *arg = eap->arg;
2655 menutrans_T *tp;
2656 int i;
2657 char_u *from, *from_noamp, *to;
2658
2659 if (menutrans_ga.ga_itemsize == 0)
Bram Moolenaar04935fb2022-01-08 16:19:22 +00002660 ga_init2(&menutrans_ga, sizeof(menutrans_T), 5);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002661
2662 /*
2663 * ":menutrans clear": clear all translations.
2664 */
Bram Moolenaar1966c242020-04-20 22:42:32 +02002665 if (STRNCMP(arg, "clear", 5) == 0 && ends_excmd2(arg, skipwhite(arg + 5)))
Bram Moolenaar071d4272004-06-13 20:20:40 +00002666 {
2667 tp = (menutrans_T *)menutrans_ga.ga_data;
2668 for (i = 0; i < menutrans_ga.ga_len; ++i)
2669 {
2670 vim_free(tp[i].from);
2671 vim_free(tp[i].from_noamp);
2672 vim_free(tp[i].to);
2673 }
2674 ga_clear(&menutrans_ga);
2675# ifdef FEAT_EVAL
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002676 // Delete all "menutrans_" global variables.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002677 del_menutrans_vars();
2678# endif
2679 }
2680 else
2681 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002682 // ":menutrans from to": add translation
Bram Moolenaar071d4272004-06-13 20:20:40 +00002683 from = arg;
2684 arg = menu_skip_part(arg);
2685 to = skipwhite(arg);
2686 *arg = NUL;
2687 arg = menu_skip_part(to);
Bram Moolenaar1966c242020-04-20 22:42:32 +02002688 if (arg == to || ends_excmd2(eap->arg, from)
2689 || ends_excmd2(eap->arg, to)
2690 || !ends_excmd2(eap->arg, skipwhite(arg)))
Bram Moolenaar436b5ad2021-12-31 22:49:24 +00002691 emsg(_(e_invalid_argument));
Bram Moolenaar071d4272004-06-13 20:20:40 +00002692 else
2693 {
2694 if (ga_grow(&menutrans_ga, 1) == OK)
2695 {
2696 tp = (menutrans_T *)menutrans_ga.ga_data;
2697 from = vim_strsave(from);
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00002698 if (from != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002699 {
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00002700 from_noamp = menu_text(from, NULL, NULL);
Bram Moolenaar71ccd032020-06-12 22:59:11 +02002701 to = vim_strnsave(to, arg - to);
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00002702 if (from_noamp != NULL && to != NULL)
2703 {
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002704 menu_translate_tab_and_shift(from);
2705 menu_translate_tab_and_shift(to);
2706 menu_unescape_name(from);
2707 menu_unescape_name(to);
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00002708 tp[menutrans_ga.ga_len].from = from;
2709 tp[menutrans_ga.ga_len].from_noamp = from_noamp;
2710 tp[menutrans_ga.ga_len].to = to;
2711 ++menutrans_ga.ga_len;
2712 }
2713 else
2714 {
2715 vim_free(from);
2716 vim_free(from_noamp);
2717 vim_free(to);
2718 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002719 }
2720 }
2721 }
2722 }
2723#endif
2724}
2725
2726#if defined(FEAT_MULTI_LANG) || defined(FEAT_TOOLBAR)
2727/*
2728 * Find the character just after one part of a menu name.
2729 */
2730 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002731menu_skip_part(char_u *p)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002732{
Bram Moolenaar1c465442017-03-12 20:10:05 +01002733 while (*p != NUL && *p != '.' && !VIM_ISWHITE(*p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00002734 {
2735 if ((*p == '\\' || *p == Ctrl_V) && p[1] != NUL)
2736 ++p;
2737 ++p;
2738 }
2739 return p;
2740}
2741#endif
2742
2743#ifdef FEAT_MULTI_LANG
2744/*
2745 * Lookup part of a menu name in the translations.
2746 * Return a pointer to the translation or NULL if not found.
2747 */
2748 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002749menutrans_lookup(char_u *name, int len)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002750{
2751 menutrans_T *tp = (menutrans_T *)menutrans_ga.ga_data;
2752 int i;
2753 char_u *dname;
2754
2755 for (i = 0; i < menutrans_ga.ga_len; ++i)
Bram Moolenaar11dd8c12017-03-04 20:41:34 +01002756 if (STRNICMP(name, tp[i].from, len) == 0 && tp[i].from[len] == NUL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002757 return tp[i].to;
2758
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002759 // Now try again while ignoring '&' characters.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002760 i = name[len];
2761 name[len] = NUL;
2762 dname = menu_text(name, NULL, NULL);
2763 name[len] = i;
2764 if (dname != NULL)
2765 {
2766 for (i = 0; i < menutrans_ga.ga_len; ++i)
Bram Moolenaar11dd8c12017-03-04 20:41:34 +01002767 if (STRICMP(dname, tp[i].from_noamp) == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002768 {
2769 vim_free(dname);
2770 return tp[i].to;
2771 }
2772 vim_free(dname);
2773 }
2774
2775 return NULL;
2776}
Bram Moolenaar071d4272004-06-13 20:20:40 +00002777
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002778/*
2779 * Unescape the name in the translate dictionary table.
2780 */
2781 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002782menu_unescape_name(char_u *name)
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002783{
2784 char_u *p;
2785
Bram Moolenaar91acfff2017-03-12 19:22:36 +01002786 for (p = name; *p && *p != '.'; MB_PTR_ADV(p))
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002787 if (*p == '\\')
2788 STRMOVE(p, p + 1);
2789}
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002790#endif // FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002791
2792/*
2793 * Isolate the menu name.
2794 * Skip the menu name, and translate <Tab> into a real TAB.
2795 */
2796 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002797menu_translate_tab_and_shift(char_u *arg_start)
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002798{
2799 char_u *arg = arg_start;
2800
Bram Moolenaar1c465442017-03-12 20:10:05 +01002801 while (*arg && !VIM_ISWHITE(*arg))
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02002802 {
2803 if ((*arg == '\\' || *arg == Ctrl_V) && arg[1] != NUL)
2804 arg++;
2805 else if (STRNICMP(arg, "<TAB>", 5) == 0)
2806 {
2807 *arg = TAB;
2808 STRMOVE(arg + 1, arg + 5);
2809 }
2810 arg++;
2811 }
2812 if (*arg != NUL)
2813 *arg++ = NUL;
2814 arg = skipwhite(arg);
2815
2816 return arg;
2817}
2818
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002819/*
2820 * Get the information about a menu item in mode 'which'
2821 */
2822 static int
Yegappan Lakshmanan51491ad2021-09-30 19:00:00 +01002823menuitem_getinfo(char_u *menu_name, vimmenu_T *menu, int modes, dict_T *dict)
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002824{
2825 int status;
Yegappan Lakshmanan51491ad2021-09-30 19:00:00 +01002826 list_T *l;
2827
2828 if (*menu_name == NUL)
2829 {
2830 // Return all the top-level menus
2831 vimmenu_T *topmenu;
2832
2833 l = list_alloc();
2834 if (l == NULL)
2835 return FAIL;
2836
2837 dict_add_list(dict, "submenus", l);
2838 // get all the children. Skip PopUp[nvoci].
2839 for (topmenu = menu; topmenu != NULL; topmenu = topmenu->next)
2840 if (!menu_is_hidden(topmenu->dname))
2841 list_append_string(l, topmenu->dname, -1);
2842 return OK;
2843 }
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002844
2845 if (menu_is_tearoff(menu->dname)) // skip tearoff menu item
2846 return OK;
2847
2848 status = dict_add_string(dict, "name", menu->name);
2849 if (status == OK)
2850 status = dict_add_string(dict, "display", menu->dname);
2851 if (status == OK && menu->actext != NULL)
2852 status = dict_add_string(dict, "accel", menu->actext);
2853 if (status == OK)
2854 status = dict_add_number(dict, "priority", menu->priority);
2855 if (status == OK)
2856 status = dict_add_string(dict, "modes",
2857 get_menu_mode_str(menu->modes));
2858#ifdef FEAT_TOOLBAR
2859 if (status == OK && menu->iconfile != NULL)
2860 status = dict_add_string(dict, "icon", menu->iconfile);
2861 if (status == OK && menu->iconidx >= 0)
2862 status = dict_add_number(dict, "iconidx", menu->iconidx);
2863#endif
2864 if (status == OK)
2865 {
2866 char_u buf[NUMBUFLEN];
2867
2868 if (has_mbyte)
2869 buf[utf_char2bytes(menu->mnemonic, buf)] = NUL;
2870 else
2871 {
2872 buf[0] = (char_u)menu->mnemonic;
2873 buf[1] = NUL;
2874 }
2875 status = dict_add_string(dict, "shortcut", buf);
2876 }
2877 if (status == OK && menu->children == NULL)
2878 {
2879 int bit;
2880
2881 // Get the first mode in which the menu is available
Bram Moolenaar56cb3372020-03-16 20:04:41 +01002882 for (bit = 0; bit < MENU_MODES && !((1 << bit) & modes); bit++)
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002883 ;
Bram Moolenaar56cb3372020-03-16 20:04:41 +01002884 if (bit < MENU_MODES) // just in case, avoid Coverity warning
2885 {
2886 if (menu->strings[bit] != NULL)
Bram Moolenaar292b90d2020-03-18 15:23:16 +01002887 {
2888 char_u *tofree = NULL;
2889
Bram Moolenaar56cb3372020-03-16 20:04:41 +01002890 status = dict_add_string(dict, "rhs",
2891 *menu->strings[bit] == NUL
Bram Moolenaar292b90d2020-03-18 15:23:16 +01002892 ? (char_u *)"<Nop>"
2893 : (tofree = str2special_save(
2894 menu->strings[bit], FALSE)));
2895 vim_free(tofree);
2896 }
Bram Moolenaar56cb3372020-03-16 20:04:41 +01002897 if (status == OK)
2898 status = dict_add_bool(dict, "noremenu",
2899 menu->noremap[bit] == REMAP_NONE);
2900 if (status == OK)
2901 status = dict_add_bool(dict, "script",
2902 menu->noremap[bit] == REMAP_SCRIPT);
2903 if (status == OK)
2904 status = dict_add_bool(dict, "silent", menu->silent[bit]);
2905 if (status == OK)
2906 status = dict_add_bool(dict, "enabled",
2907 ((menu->enabled & (1 << bit)) != 0));
2908 }
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002909 }
Bram Moolenaar56cb3372020-03-16 20:04:41 +01002910
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002911 // If there are submenus, add all the submenu display names
2912 if (status == OK && menu->children != NULL)
2913 {
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002914 vimmenu_T *child;
2915
Yegappan Lakshmanan51491ad2021-09-30 19:00:00 +01002916 l = list_alloc();
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002917 if (l == NULL)
2918 return FAIL;
2919
2920 dict_add_list(dict, "submenus", l);
2921 child = menu->children;
2922 while (child)
2923 {
2924 if (!menu_is_tearoff(child->dname)) // skip tearoff menu
2925 list_append_string(l, child->dname, -1);
2926 child = child->next;
2927 }
2928 }
2929
2930 return status;
2931}
2932
2933/*
2934 * "menu_info()" function
2935 * Return information about a menu (including all the child menus)
2936 */
2937 void
2938f_menu_info(typval_T *argvars, typval_T *rettv)
2939{
2940 char_u *menu_name;
2941 char_u *which;
2942 int modes;
2943 char_u *saved_name;
2944 char_u *name;
2945 vimmenu_T *menu;
2946 dict_T *retdict;
2947
Bram Moolenaar93a10962022-06-16 11:42:09 +01002948 if (rettv_dict_alloc(rettv) == FAIL)
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002949 return;
2950 retdict = rettv->vval.v_dict;
2951
Yegappan Lakshmanan4490ec42021-07-27 22:00:44 +02002952 if (in_vim9script()
2953 && (check_for_string_arg(argvars, 0) == FAIL
2954 || check_for_opt_string_arg(argvars, 1) == FAIL))
2955 return;
2956
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01002957 menu_name = tv_get_string_chk(&argvars[0]);
2958 if (menu_name == NULL)
2959 return;
2960
2961 // menu mode
2962 if (argvars[1].v_type != VAR_UNKNOWN)
2963 which = tv_get_string_chk(&argvars[1]);
2964 else
2965 which = (char_u *)""; // Default is modes for "menu"
2966 if (which == NULL)
2967 return;
2968
2969 modes = get_menu_cmd_modes(which, *which == '!', NULL, NULL);
2970
2971 // Locate the specified menu or menu item
2972 menu = *get_root_menu(menu_name);
2973 saved_name = vim_strsave(menu_name);
2974 if (saved_name == NULL)
2975 return;
2976 if (*saved_name != NUL)
2977 {
2978 char_u *p;
2979
2980 name = saved_name;
2981 while (*name)
2982 {
2983 // Find in the menu hierarchy
2984 p = menu_name_skip(name);
2985 while (menu != NULL)
2986 {
2987 if (menu_name_equal(name, menu))
2988 break;
2989 menu = menu->next;
2990 }
2991 if (menu == NULL || *p == NUL)
2992 break;
2993 menu = menu->children;
2994 name = p;
2995 }
2996 }
2997 vim_free(saved_name);
2998
2999 if (menu == NULL) // specified menu not found
3000 return;
3001
3002 if (menu->modes & modes)
Yegappan Lakshmanan51491ad2021-09-30 19:00:00 +01003003 menuitem_getinfo(menu_name, menu, modes, retdict);
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01003004}
3005
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01003006#endif // FEAT_MENU