blob: 432fbe972765a4cd05d4e26b8c31b5d629b6fa90 [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 Moolenaara4d158b2022-08-14 14:17:45 +0100302 redraw_later(UPD_NOT_VALID);
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200303
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 }
Luuk van Baal5ed39172022-09-13 11:55:10 +0100439 curwin->w_prev_height = curwin->w_height;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200440 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000441
442theend:
Bram Moolenaar071d4272004-06-13 20:20:40 +0000443 ;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000444}
445
446/*
447 * Add the menu with the given name to the menu hierarchy
448 */
449 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100450add_menu_path(
451 char_u *menu_path,
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100452 vimmenu_T *menuarg, // passes modes, iconfile, iconidx,
453 // icon_builtin, silent[0], noremap[0]
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100454 int *pri_tab,
455 char_u *call_data
Bram Moolenaar4f974752019-02-17 17:44:42 +0100456#ifdef FEAT_GUI_MSWIN
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100457 , int addtearoff // may add tearoff item
Bram Moolenaar071d4272004-06-13 20:20:40 +0000458#endif
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100459 )
Bram Moolenaar071d4272004-06-13 20:20:40 +0000460{
461 char_u *path_name;
462 int modes = menuarg->modes;
463 vimmenu_T **menup;
464 vimmenu_T *menu = NULL;
465 vimmenu_T *parent;
466 vimmenu_T **lower_pri;
467 char_u *p;
468 char_u *name;
469 char_u *dname;
470 char_u *next_name;
471 int i;
472 int c;
Bram Moolenaar7871a502010-05-14 21:19:23 +0200473 int d;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000474#ifdef FEAT_GUI
475 int idx;
476 int new_idx;
477#endif
478 int pri_idx = 0;
479 int old_modes = 0;
480 int amenu;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200481#ifdef FEAT_MULTI_LANG
482 char_u *en_name;
483 char_u *map_to = NULL;
484#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200485 vimmenu_T **root_menu_ptr;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000486
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100487 // Make a copy so we can stuff around with it, since it could be const
Bram Moolenaar071d4272004-06-13 20:20:40 +0000488 path_name = vim_strsave(menu_path);
489 if (path_name == NULL)
490 return FAIL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200491 root_menu_ptr = get_root_menu(menu_path);
492 menup = root_menu_ptr;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000493 parent = NULL;
494 name = path_name;
495 while (*name)
496 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100497 // Get name of this element in the menu hierarchy, and the simplified
498 // name (without mnemonic and accelerator text).
Bram Moolenaar071d4272004-06-13 20:20:40 +0000499 next_name = menu_name_skip(name);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200500#ifdef FEAT_MULTI_LANG
Bram Moolenaar442b4222010-05-24 21:34:22 +0200501 map_to = menutrans_lookup(name, (int)STRLEN(name));
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200502 if (map_to != NULL)
503 {
504 en_name = name;
505 name = map_to;
506 }
507 else
508 en_name = NULL;
509#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000510 dname = menu_text(name, NULL, NULL);
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000511 if (dname == NULL)
512 goto erret;
513 if (*dname == NUL)
514 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100515 // Only a mnemonic or accelerator is not valid.
Bram Moolenaar677658a2022-01-05 16:09:06 +0000516 emsg(_(e_empty_menu_name));
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000517 goto erret;
518 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000519
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100520 // See if it's already there
Bram Moolenaar071d4272004-06-13 20:20:40 +0000521 lower_pri = menup;
522#ifdef FEAT_GUI
523 idx = 0;
524 new_idx = 0;
525#endif
526 menu = *menup;
527 while (menu != NULL)
528 {
529 if (menu_name_equal(name, menu) || menu_name_equal(dname, menu))
530 {
531 if (*next_name == NUL && menu->children != NULL)
532 {
533 if (!sys_menu)
Bram Moolenaard88be5b2022-01-04 19:57:55 +0000534 emsg(_(e_menu_path_must_not_lead_to_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000535 goto erret;
536 }
537 if (*next_name != NUL && menu->children == NULL
Bram Moolenaar4f974752019-02-17 17:44:42 +0100538#ifdef FEAT_GUI_MSWIN
Bram Moolenaar071d4272004-06-13 20:20:40 +0000539 && addtearoff
540#endif
541 )
542 {
543 if (!sys_menu)
Bram Moolenaareaaac012022-01-02 17:00:40 +0000544 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000545 goto erret;
546 }
547 break;
548 }
549 menup = &menu->next;
550
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100551 // Count menus, to find where this one needs to be inserted.
552 // Ignore menus that are not in the menubar (PopUp and Toolbar)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000553 if (parent != NULL || menu_is_menubar(menu->name))
554 {
555#ifdef FEAT_GUI
556 ++idx;
557#endif
558 if (menu->priority <= pri_tab[pri_idx])
559 {
560 lower_pri = menup;
561#ifdef FEAT_GUI
562 new_idx = idx;
563#endif
564 }
565 }
566 menu = menu->next;
567 }
568
569 if (menu == NULL)
570 {
571 if (*next_name == NUL && parent == NULL)
572 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000573 emsg(_(e_must_not_add_menu_items_directly_to_menu_bar));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000574 goto erret;
575 }
576
577 if (menu_is_separator(dname) && *next_name != NUL)
578 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000579 emsg(_(e_separator_cannot_be_part_of_menu_path));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000580 goto erret;
581 }
582
Dominique Pelleaf4a61a2021-12-27 17:21:41 +0000583 // Not already there, so let's add it
Bram Moolenaare809a4e2019-07-04 17:35:05 +0200584 menu = ALLOC_CLEAR_ONE(vimmenu_T);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000585 if (menu == NULL)
586 goto erret;
587
588 menu->modes = modes;
589 menu->enabled = MENU_ALL_MODES;
590 menu->name = vim_strsave(name);
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100591 // separate mnemonic and accelerator text from actual menu name
Bram Moolenaar071d4272004-06-13 20:20:40 +0000592 menu->dname = menu_text(name, &menu->mnemonic, &menu->actext);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +0200593#ifdef FEAT_MULTI_LANG
594 if (en_name != NULL)
595 {
596 menu->en_name = vim_strsave(en_name);
597 menu->en_dname = menu_text(en_name, NULL, NULL);
598 }
599 else
600 {
601 menu->en_name = NULL;
602 menu->en_dname = NULL;
603 }
604#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000605 menu->priority = pri_tab[pri_idx];
606 menu->parent = parent;
607#ifdef FEAT_GUI_MOTIF
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100608 menu->sensitive = TRUE; // the default
Bram Moolenaar071d4272004-06-13 20:20:40 +0000609#endif
610#ifdef FEAT_BEVAL_TIP
611 menu->tip = NULL;
612#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000613 /*
614 * Add after menu that has lower priority.
615 */
616 menu->next = *lower_pri;
617 *lower_pri = menu;
618
619 old_modes = 0;
620
621#ifdef FEAT_TOOLBAR
622 menu->iconidx = menuarg->iconidx;
623 menu->icon_builtin = menuarg->icon_builtin;
624 if (*next_name == NUL && menuarg->iconfile != NULL)
625 menu->iconfile = vim_strsave(menuarg->iconfile);
626#endif
Bram Moolenaar4f974752019-02-17 17:44:42 +0100627#if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100628 // the tearoff item must be present in the modes of each item.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000629 if (parent != NULL && menu_is_tearoff(parent->children->dname))
630 parent->children->modes |= modes;
631#endif
632 }
633 else
634 {
635 old_modes = menu->modes;
636
637 /*
638 * If this menu option was previously only available in other
639 * modes, then make sure it's available for this one now
640 * Also enable a menu when it's created or changed.
641 */
Bram Moolenaar4f974752019-02-17 17:44:42 +0100642#ifdef FEAT_GUI_MSWIN
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100643 // If adding a tearbar (addtearoff == FALSE) don't update modes
Bram Moolenaar071d4272004-06-13 20:20:40 +0000644 if (addtearoff)
645#endif
646 {
647 menu->modes |= modes;
648 menu->enabled |= modes;
649 }
650 }
651
652#ifdef FEAT_GUI
653 /*
654 * Add the menu item when it's used in one of the modes, but not when
655 * only a tooltip is defined.
656 */
657 if ((old_modes & MENU_ALL_MODES) == 0
658 && (menu->modes & MENU_ALL_MODES) != 0)
659 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100660 if (gui.in_use) // Otherwise it will be added when GUI starts
Bram Moolenaar071d4272004-06-13 20:20:40 +0000661 {
662 if (*next_name == NUL)
663 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100664 // Real menu item, not sub-menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000665 gui_mch_add_menu_item(menu, new_idx);
666
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100667 // Want to update menus now even if mode not changed
Bram Moolenaar071d4272004-06-13 20:20:40 +0000668 force_menu_update = TRUE;
669 }
670 else
671 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100672 // Sub-menu (not at end of path yet)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000673 gui_mch_add_menu(menu, new_idx);
674 }
675 }
676
K.Takata54119102022-02-03 13:33:03 +0000677# if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100678 // When adding a new submenu, may add a tearoff item
Bram Moolenaar071d4272004-06-13 20:20:40 +0000679 if ( addtearoff
680 && *next_name
681 && vim_strchr(p_go, GO_TEAROFF) != NULL
Bram Moolenaar310c32e2019-11-29 23:15:25 +0100682 && menu_is_menubar(name)
683# ifdef VIMDLL
684 && (gui.in_use || gui.starting)
685# endif
686 )
Bram Moolenaar071d4272004-06-13 20:20:40 +0000687 {
688 char_u *tearpath;
689
690 /*
691 * The pointers next_name & path_name refer to a string with
692 * \'s and ^V's stripped out. But menu_path is a "raw"
693 * string, so we must correct for special characters.
694 */
Bram Moolenaar964b3742019-05-24 18:54:09 +0200695 tearpath = alloc(STRLEN(menu_path) + TEAR_LEN + 2);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000696 if (tearpath != NULL)
697 {
698 char_u *s;
699 int idx;
700
701 STRCPY(tearpath, menu_path);
702 idx = (int)(next_name - path_name - 1);
Bram Moolenaar91acfff2017-03-12 19:22:36 +0100703 for (s = tearpath; *s && s < tearpath + idx; MB_PTR_ADV(s))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000704 {
705 if ((*s == '\\' || *s == Ctrl_V) && s[1])
706 {
707 ++idx;
708 ++s;
709 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000710 }
711 tearpath[idx] = NUL;
712 gui_add_tearoff(tearpath, pri_tab, pri_idx);
713 vim_free(tearpath);
714 }
715 }
716# endif
717 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100718#endif // FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +0000719
720 menup = &menu->children;
721 parent = menu;
722 name = next_name;
Bram Moolenaard23a8232018-02-10 18:45:26 +0100723 VIM_CLEAR(dname);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000724 if (pri_tab[pri_idx + 1] != -1)
725 ++pri_idx;
726 }
727 vim_free(path_name);
728
729 /*
730 * Only add system menu items which have not been defined yet.
731 * First check if this was an ":amenu".
732 */
733 amenu = ((modes & (MENU_NORMAL_MODE | MENU_INSERT_MODE)) ==
734 (MENU_NORMAL_MODE | MENU_INSERT_MODE));
735 if (sys_menu)
736 modes &= ~old_modes;
737
738 if (menu != NULL && modes)
739 {
740#ifdef FEAT_GUI
741 menu->cb = gui_menu_cb;
742#endif
743 p = (call_data == NULL) ? NULL : vim_strsave(call_data);
744
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100745 // loop over all modes, may add more than one
Bram Moolenaar071d4272004-06-13 20:20:40 +0000746 for (i = 0; i < MENU_MODES; ++i)
747 {
748 if (modes & (1 << i))
749 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100750 // free any old menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000751 free_menu_string(menu, i);
752
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100753 // For "amenu", may insert an extra character.
754 // Don't do this if adding a tearbar (addtearoff == FALSE).
755 // Don't do this for "<Nop>".
Bram Moolenaar071d4272004-06-13 20:20:40 +0000756 c = 0;
Bram Moolenaar7871a502010-05-14 21:19:23 +0200757 d = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000758 if (amenu && call_data != NULL && *call_data != NUL
Bram Moolenaar4f974752019-02-17 17:44:42 +0100759#ifdef FEAT_GUI_MSWIN
Bram Moolenaar071d4272004-06-13 20:20:40 +0000760 && addtearoff
761#endif
762 )
763 {
764 switch (1 << i)
765 {
766 case MENU_VISUAL_MODE:
Bram Moolenaarb3656ed2006-03-20 21:59:49 +0000767 case MENU_SELECT_MODE:
Bram Moolenaar071d4272004-06-13 20:20:40 +0000768 case MENU_OP_PENDING_MODE:
769 case MENU_CMDLINE_MODE:
770 c = Ctrl_C;
771 break;
772 case MENU_INSERT_MODE:
Bram Moolenaar7871a502010-05-14 21:19:23 +0200773 c = Ctrl_BSL;
774 d = Ctrl_O;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000775 break;
776 }
777 }
778
Bram Moolenaar7871a502010-05-14 21:19:23 +0200779 if (c != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000780 {
Bram Moolenaar964b3742019-05-24 18:54:09 +0200781 menu->strings[i] = alloc(STRLEN(call_data) + 5);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000782 if (menu->strings[i] != NULL)
783 {
784 menu->strings[i][0] = c;
Bram Moolenaar7871a502010-05-14 21:19:23 +0200785 if (d == 0)
786 STRCPY(menu->strings[i] + 1, call_data);
787 else
788 {
789 menu->strings[i][1] = d;
790 STRCPY(menu->strings[i] + 2, call_data);
791 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000792 if (c == Ctrl_C)
793 {
Bram Moolenaara93fa7e2006-04-17 22:14:47 +0000794 int len = (int)STRLEN(menu->strings[i]);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000795
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100796 // Append CTRL-\ CTRL-G to obey 'insertmode'.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000797 menu->strings[i][len] = Ctrl_BSL;
798 menu->strings[i][len + 1] = Ctrl_G;
799 menu->strings[i][len + 2] = NUL;
800 }
801 }
802 }
803 else
804 menu->strings[i] = p;
805 menu->noremap[i] = menuarg->noremap[0];
806 menu->silent[i] = menuarg->silent[0];
807 }
808 }
Bram Moolenaar4f974752019-02-17 17:44:42 +0100809#if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN) \
Bram Moolenaarc3719bd2017-11-18 22:13:31 +0100810 && (defined(FEAT_BEVAL_GUI) || defined(FEAT_GUI_GTK))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100811 // Need to update the menu tip.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000812 if (modes & MENU_TIP_MODE)
813 gui_mch_menu_set_tip(menu);
814#endif
815 }
816 return OK;
817
818erret:
819 vim_free(path_name);
820 vim_free(dname);
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000821
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100822 // Delete any empty submenu we added before discovering the error. Repeat
823 // for higher levels.
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000824 while (parent != NULL && parent->children == NULL)
825 {
826 if (parent->parent == NULL)
Bram Moolenaar1b9645d2017-09-17 23:03:31 +0200827 menup = root_menu_ptr;
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000828 else
829 menup = &parent->parent->children;
830 for ( ; *menup != NULL && *menup != parent; menup = &((*menup)->next))
831 ;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100832 if (*menup == NULL) // safety check
Bram Moolenaar18a0b122006-08-16 13:55:16 +0000833 break;
834 parent = parent->parent;
835 free_menu(menup);
836 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000837 return FAIL;
838}
839
840/*
841 * Set the (sub)menu with the given name to enabled or disabled.
842 * Called recursively.
843 */
844 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100845menu_nable_recurse(
846 vimmenu_T *menu,
847 char_u *name,
848 int modes,
849 int enable)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000850{
851 char_u *p;
852
853 if (menu == NULL)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100854 return OK; // Got to bottom of hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000855
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100856 // Get name of this element in the menu hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000857 p = menu_name_skip(name);
858
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100859 // Find the menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000860 while (menu != NULL)
861 {
862 if (*name == NUL || *name == '*' || menu_name_equal(name, menu))
863 {
864 if (*p != NUL)
865 {
866 if (menu->children == NULL)
867 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000868 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000869 return FAIL;
870 }
871 if (menu_nable_recurse(menu->children, p, modes, enable)
872 == FAIL)
873 return FAIL;
874 }
875 else
876 if (enable)
877 menu->enabled |= modes;
878 else
879 menu->enabled &= ~modes;
880
881 /*
882 * When name is empty, we are doing all menu items for the given
883 * modes, so keep looping, otherwise we are just doing the named
884 * menu item (which has been found) so break here.
885 */
886 if (*name != NUL && *name != '*')
887 break;
888 }
889 menu = menu->next;
890 }
891 if (*name != NUL && *name != '*' && menu == NULL)
892 {
Bram Moolenaareaaac012022-01-02 17:00:40 +0000893 semsg(_(e_no_menu_str), name);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000894 return FAIL;
895 }
896
897#ifdef FEAT_GUI
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100898 // Want to update menus now even if mode not changed
Bram Moolenaar071d4272004-06-13 20:20:40 +0000899 force_menu_update = TRUE;
900#endif
901
902 return OK;
903}
904
905/*
906 * Remove the (sub)menu with the given name from the menu hierarchy
907 * Called recursively.
908 */
909 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +0100910remove_menu(
911 vimmenu_T **menup,
912 char_u *name,
913 int modes,
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100914 int silent) // don't give error messages
Bram Moolenaar071d4272004-06-13 20:20:40 +0000915{
916 vimmenu_T *menu;
917 vimmenu_T *child;
918 char_u *p;
919
920 if (*menup == NULL)
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100921 return OK; // Got to bottom of hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000922
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100923 // Get name of this element in the menu hierarchy
Bram Moolenaar071d4272004-06-13 20:20:40 +0000924 p = menu_name_skip(name);
925
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100926 // Find the menu
Bram Moolenaar071d4272004-06-13 20:20:40 +0000927 while ((menu = *menup) != NULL)
928 {
929 if (*name == NUL || menu_name_equal(name, menu))
930 {
931 if (*p != NUL && menu->children == NULL)
932 {
933 if (!silent)
Bram Moolenaareaaac012022-01-02 17:00:40 +0000934 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000935 return FAIL;
936 }
937 if ((menu->modes & modes) != 0x0)
938 {
Bram Moolenaar4f974752019-02-17 17:44:42 +0100939#if defined(FEAT_GUI_MSWIN) & defined(FEAT_TEAROFF)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000940 /*
941 * If we are removing all entries for this menu,MENU_ALL_MODES,
942 * Then kill any tearoff before we start
943 */
944 if (*p == NUL && modes == MENU_ALL_MODES)
945 {
946 if (IsWindow(menu->tearoff_handle))
947 DestroyWindow(menu->tearoff_handle);
948 }
949#endif
950 if (remove_menu(&menu->children, p, modes, silent) == FAIL)
951 return FAIL;
952 }
953 else if (*name != NUL)
954 {
955 if (!silent)
Bram Moolenaar3a846e62022-01-01 16:21:00 +0000956 emsg(_(e_menu_only_exists_in_another_mode));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000957 return FAIL;
958 }
959
960 /*
961 * When name is empty, we are removing all menu items for the given
962 * modes, so keep looping, otherwise we are just removing the named
963 * menu item (which has been found) so break here.
964 */
965 if (*name != NUL)
966 break;
967
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100968 // Remove the menu item for the given mode[s]. If the menu item
969 // is no longer valid in ANY mode, delete it
Bram Moolenaar071d4272004-06-13 20:20:40 +0000970 menu->modes &= ~modes;
971 if (modes & MENU_TIP_MODE)
972 free_menu_string(menu, MENU_INDEX_TIP);
973 if ((menu->modes & MENU_ALL_MODES) == 0)
974 free_menu(menup);
975 else
976 menup = &menu->next;
977 }
978 else
979 menup = &menu->next;
980 }
981 if (*name != NUL)
982 {
983 if (menu == NULL)
984 {
985 if (!silent)
Bram Moolenaareaaac012022-01-02 17:00:40 +0000986 semsg(_(e_no_menu_str), name);
Bram Moolenaar071d4272004-06-13 20:20:40 +0000987 return FAIL;
988 }
989
990
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100991 // Recalculate modes for menu based on the new updated children
Bram Moolenaar071d4272004-06-13 20:20:40 +0000992 menu->modes &= ~modes;
Bram Moolenaar4f974752019-02-17 17:44:42 +0100993#if defined(FEAT_GUI_MSWIN) & defined(FEAT_TEAROFF)
Dominique Pelleaf4a61a2021-12-27 17:21:41 +0000994 if ((s_tearoffs) && (menu->children != NULL)) // there's a tear bar.
Bram Moolenaar4ba37b52019-12-04 21:57:43 +0100995 child = menu->children->next; // don't count tearoff bar
Bram Moolenaar071d4272004-06-13 20:20:40 +0000996 else
997#endif
998 child = menu->children;
999 for ( ; child != NULL; child = child->next)
1000 menu->modes |= child->modes;
1001 if (modes & MENU_TIP_MODE)
1002 {
1003 free_menu_string(menu, MENU_INDEX_TIP);
Bram Moolenaar4f974752019-02-17 17:44:42 +01001004#if defined(FEAT_TOOLBAR) && !defined(FEAT_GUI_MSWIN) \
Bram Moolenaarc3719bd2017-11-18 22:13:31 +01001005 && (defined(FEAT_BEVAL_GUI) || defined(FEAT_GUI_GTK))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001006 // Need to update the menu tip.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001007 if (gui.in_use)
1008 gui_mch_menu_set_tip(menu);
1009#endif
1010 }
1011 if ((menu->modes & MENU_ALL_MODES) == 0)
1012 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001013 // The menu item is no longer valid in ANY mode, so delete it
Bram Moolenaar4f974752019-02-17 17:44:42 +01001014#if defined(FEAT_GUI_MSWIN) & defined(FEAT_TEAROFF)
Dominique Pelleaf4a61a2021-12-27 17:21:41 +00001015 if (s_tearoffs && menu->children != NULL) // there's a tear bar.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001016 free_menu(&menu->children);
1017#endif
1018 *menup = menu;
1019 free_menu(menup);
1020 }
1021 }
1022
1023 return OK;
1024}
1025
1026/*
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001027 * Remove the WinBar menu from window "wp".
1028 */
1029 void
1030remove_winbar(win_T *wp)
1031{
1032 remove_menu(&wp->w_winbar, (char_u *)"", MENU_ALL_MODES, TRUE);
1033 vim_free(wp->w_winbar_items);
1034}
1035
1036/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00001037 * Free the given menu structure and remove it from the linked list.
1038 */
1039 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001040free_menu(vimmenu_T **menup)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001041{
1042 int i;
1043 vimmenu_T *menu;
1044
1045 menu = *menup;
1046
1047#ifdef FEAT_GUI
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001048 // Free machine specific menu structures (only when already created)
1049 // Also may rebuild a tearoff'ed menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001050 if (gui.in_use)
1051 gui_mch_destroy_menu(menu);
1052#endif
1053
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001054 // Don't change *menup until after calling gui_mch_destroy_menu(). The
1055 // MacOS code needs the original structure to properly delete the menu.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001056 *menup = menu->next;
1057 vim_free(menu->name);
1058 vim_free(menu->dname);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001059#ifdef FEAT_MULTI_LANG
1060 vim_free(menu->en_name);
1061 vim_free(menu->en_dname);
1062#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001063 vim_free(menu->actext);
1064#ifdef FEAT_TOOLBAR
1065 vim_free(menu->iconfile);
Bram Moolenaarbee0c5b2005-02-07 22:03:36 +00001066# ifdef FEAT_GUI_MOTIF
1067 vim_free(menu->xpm_fname);
1068# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001069#endif
1070 for (i = 0; i < MENU_MODES; i++)
1071 free_menu_string(menu, i);
1072 vim_free(menu);
1073
1074#ifdef FEAT_GUI
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001075 // Want to update menus now even if mode not changed
Bram Moolenaar071d4272004-06-13 20:20:40 +00001076 force_menu_update = TRUE;
1077#endif
1078}
1079
1080/*
1081 * Free the menu->string with the given index.
1082 */
1083 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001084free_menu_string(vimmenu_T *menu, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001085{
1086 int count = 0;
1087 int i;
1088
1089 for (i = 0; i < MENU_MODES; i++)
1090 if (menu->strings[i] == menu->strings[idx])
1091 count++;
1092 if (count == 1)
1093 vim_free(menu->strings[idx]);
1094 menu->strings[idx] = NULL;
1095}
1096
1097/*
1098 * Show the mapping associated with a menu item or hierarchy in a sub-menu.
1099 */
1100 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001101show_menus(char_u *path_name, int modes)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001102{
1103 char_u *p;
1104 char_u *name;
1105 vimmenu_T *menu;
1106 vimmenu_T *parent = NULL;
1107
Bram Moolenaar071d4272004-06-13 20:20:40 +00001108 name = path_name = vim_strsave(path_name);
1109 if (path_name == NULL)
1110 return FAIL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001111 menu = *get_root_menu(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001112
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001113 // First, find the (sub)menu with the given name
Bram Moolenaar071d4272004-06-13 20:20:40 +00001114 while (*name)
1115 {
1116 p = menu_name_skip(name);
1117 while (menu != NULL)
1118 {
1119 if (menu_name_equal(name, menu))
1120 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001121 // Found menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001122 if (*p != NUL && menu->children == NULL)
1123 {
Bram Moolenaareaaac012022-01-02 17:00:40 +00001124 emsg(_(e_part_of_menu_item_path_is_not_sub_menu));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001125 vim_free(path_name);
1126 return FAIL;
1127 }
1128 else if ((menu->modes & modes) == 0x0)
1129 {
Bram Moolenaar3a846e62022-01-01 16:21:00 +00001130 emsg(_(e_menu_only_exists_in_another_mode));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001131 vim_free(path_name);
1132 return FAIL;
1133 }
1134 break;
1135 }
1136 menu = menu->next;
1137 }
1138 if (menu == NULL)
1139 {
Bram Moolenaareaaac012022-01-02 17:00:40 +00001140 semsg(_(e_no_menu_str), name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001141 vim_free(path_name);
1142 return FAIL;
1143 }
1144 name = p;
1145 parent = menu;
1146 menu = menu->children;
1147 }
Bram Moolenaaracbd4422008-08-17 21:44:45 +00001148 vim_free(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001149
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001150 // Now we have found the matching menu, and we list the mappings
1151 // Highlight title
Bram Moolenaar32526b32019-01-19 17:43:09 +01001152 msg_puts_title(_("\n--- Menus ---"));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001153
1154 show_menus_recursive(parent, modes, 0);
1155 return OK;
1156}
1157
1158/*
1159 * Recursively show the mappings associated with the menus under the given one
1160 */
1161 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001162show_menus_recursive(vimmenu_T *menu, int modes, int depth)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001163{
1164 int i;
1165 int bit;
1166
1167 if (menu != NULL && (menu->modes & modes) == 0x0)
1168 return;
1169
1170 if (menu != NULL)
1171 {
1172 msg_putchar('\n');
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001173 if (got_int) // "q" hit for "--more--"
Bram Moolenaar071d4272004-06-13 20:20:40 +00001174 return;
1175 for (i = 0; i < depth; i++)
Bram Moolenaar32526b32019-01-19 17:43:09 +01001176 msg_puts(" ");
Bram Moolenaar071d4272004-06-13 20:20:40 +00001177 if (menu->priority)
1178 {
1179 msg_outnum((long)menu->priority);
Bram Moolenaar32526b32019-01-19 17:43:09 +01001180 msg_puts(" ");
Bram Moolenaar071d4272004-06-13 20:20:40 +00001181 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001182 // Same highlighting as for directories!?
Bram Moolenaar8820b482017-03-16 17:23:31 +01001183 msg_outtrans_attr(menu->name, HL_ATTR(HLF_D));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001184 }
1185
1186 if (menu != NULL && menu->children == NULL)
1187 {
1188 for (bit = 0; bit < MENU_MODES; bit++)
1189 if ((menu->modes & modes & (1 << bit)) != 0)
1190 {
1191 msg_putchar('\n');
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001192 if (got_int) // "q" hit for "--more--"
Bram Moolenaar071d4272004-06-13 20:20:40 +00001193 return;
1194 for (i = 0; i < depth + 2; i++)
Bram Moolenaar32526b32019-01-19 17:43:09 +01001195 msg_puts(" ");
1196 msg_puts(menu_mode_chars[bit]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001197 if (menu->noremap[bit] == REMAP_NONE)
1198 msg_putchar('*');
1199 else if (menu->noremap[bit] == REMAP_SCRIPT)
1200 msg_putchar('&');
1201 else
1202 msg_putchar(' ');
1203 if (menu->silent[bit])
1204 msg_putchar('s');
1205 else
1206 msg_putchar(' ');
1207 if ((menu->modes & menu->enabled & (1 << bit)) == 0)
1208 msg_putchar('-');
1209 else
1210 msg_putchar(' ');
Bram Moolenaar32526b32019-01-19 17:43:09 +01001211 msg_puts(" ");
Bram Moolenaar071d4272004-06-13 20:20:40 +00001212 if (*menu->strings[bit] == NUL)
Bram Moolenaar32526b32019-01-19 17:43:09 +01001213 msg_puts_attr("<Nop>", HL_ATTR(HLF_8));
Bram Moolenaar071d4272004-06-13 20:20:40 +00001214 else
Bram Moolenaar725310d2019-04-24 23:08:23 +02001215 msg_outtrans_special(menu->strings[bit], FALSE, 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001216 }
1217 }
1218 else
1219 {
1220 if (menu == NULL)
1221 {
1222 menu = root_menu;
1223 depth--;
1224 }
1225 else
1226 menu = menu->children;
1227
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001228 // recursively show all children. Skip PopUp[nvoci].
Bram Moolenaar071d4272004-06-13 20:20:40 +00001229 for (; menu != NULL && !got_int; menu = menu->next)
1230 if (!menu_is_hidden(menu->dname))
1231 show_menus_recursive(menu, modes, depth + 1);
1232 }
1233}
1234
Bram Moolenaar071d4272004-06-13 20:20:40 +00001235/*
1236 * Used when expanding menu names.
1237 */
1238static vimmenu_T *expand_menu = NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001239static vimmenu_T *expand_menu_alt = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001240static int expand_modes = 0x0;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001241static int expand_emenu; // TRUE for ":emenu" command
Bram Moolenaar071d4272004-06-13 20:20:40 +00001242
1243/*
1244 * Work out what to complete when doing command line completion of menu names.
1245 */
1246 char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001247set_context_in_menu_cmd(
1248 expand_T *xp,
1249 char_u *cmd,
1250 char_u *arg,
1251 int forceit)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001252{
1253 char_u *after_dot;
1254 char_u *p;
1255 char_u *path_name = NULL;
1256 char_u *name;
1257 int unmenu;
1258 vimmenu_T *menu;
1259 int expand_menus;
1260
1261 xp->xp_context = EXPAND_UNSUCCESSFUL;
1262
1263
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001264 // Check for priority numbers, enable and disable
Bram Moolenaar071d4272004-06-13 20:20:40 +00001265 for (p = arg; *p; ++p)
1266 if (!VIM_ISDIGIT(*p) && *p != '.')
1267 break;
1268
Bram Moolenaar1c465442017-03-12 20:10:05 +01001269 if (!VIM_ISWHITE(*p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001270 {
1271 if (STRNCMP(arg, "enable", 6) == 0
Bram Moolenaar1c465442017-03-12 20:10:05 +01001272 && (arg[6] == NUL || VIM_ISWHITE(arg[6])))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001273 p = arg + 6;
1274 else if (STRNCMP(arg, "disable", 7) == 0
Bram Moolenaar1c465442017-03-12 20:10:05 +01001275 && (arg[7] == NUL || VIM_ISWHITE(arg[7])))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001276 p = arg + 7;
1277 else
1278 p = arg;
1279 }
1280
Bram Moolenaar1c465442017-03-12 20:10:05 +01001281 while (*p != NUL && VIM_ISWHITE(*p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001282 ++p;
1283
1284 arg = after_dot = p;
1285
Bram Moolenaar1c465442017-03-12 20:10:05 +01001286 for (; *p && !VIM_ISWHITE(*p); ++p)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001287 {
1288 if ((*p == '\\' || *p == Ctrl_V) && p[1] != NUL)
1289 p++;
1290 else if (*p == '.')
1291 after_dot = p + 1;
1292 }
1293
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001294 // ":tearoff" and ":popup" only use menus, not entries
Bram Moolenaar071d4272004-06-13 20:20:40 +00001295 expand_menus = !((*cmd == 't' && cmd[1] == 'e') || *cmd == 'p');
1296 expand_emenu = (*cmd == 'e');
Bram Moolenaar1c465442017-03-12 20:10:05 +01001297 if (expand_menus && VIM_ISWHITE(*p))
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001298 return NULL; // TODO: check for next command?
1299 if (*p == NUL) // Complete the menu name
Bram Moolenaar071d4272004-06-13 20:20:40 +00001300 {
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001301 int try_alt_menu = TRUE;
1302
Bram Moolenaar071d4272004-06-13 20:20:40 +00001303 /*
1304 * With :unmenu, you only want to match menus for the appropriate mode.
1305 * With :menu though you might want to add a menu with the same name as
1306 * one in another mode, so match menus from other modes too.
1307 */
1308 expand_modes = get_menu_cmd_modes(cmd, forceit, NULL, &unmenu);
1309 if (!unmenu)
1310 expand_modes = MENU_ALL_MODES;
1311
1312 menu = root_menu;
1313 if (after_dot != arg)
1314 {
Bram Moolenaar964b3742019-05-24 18:54:09 +02001315 path_name = alloc(after_dot - arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001316 if (path_name == NULL)
1317 return NULL;
Bram Moolenaarce0842a2005-07-18 21:58:11 +00001318 vim_strncpy(path_name, arg, after_dot - arg - 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001319 }
1320 name = path_name;
1321 while (name != NULL && *name)
1322 {
1323 p = menu_name_skip(name);
1324 while (menu != NULL)
1325 {
1326 if (menu_name_equal(name, menu))
1327 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001328 // Found menu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001329 if ((*p != NUL && menu->children == NULL)
1330 || ((menu->modes & expand_modes) == 0x0))
1331 {
1332 /*
1333 * Menu path continues, but we have reached a leaf.
1334 * Or menu exists only in another mode.
1335 */
1336 vim_free(path_name);
1337 return NULL;
1338 }
1339 break;
1340 }
1341 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001342 if (menu == NULL && try_alt_menu)
1343 {
1344 menu = curwin->w_winbar;
1345 try_alt_menu = FALSE;
1346 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001347 }
1348 if (menu == NULL)
1349 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001350 // No menu found with the name we were looking for
Bram Moolenaar071d4272004-06-13 20:20:40 +00001351 vim_free(path_name);
1352 return NULL;
1353 }
1354 name = p;
1355 menu = menu->children;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001356 try_alt_menu = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001357 }
Bram Moolenaareb3593b2006-04-22 22:33:57 +00001358 vim_free(path_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001359
1360 xp->xp_context = expand_menus ? EXPAND_MENUNAMES : EXPAND_MENUS;
1361 xp->xp_pattern = after_dot;
1362 expand_menu = menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001363 if (expand_menu == root_menu)
1364 expand_menu_alt = curwin->w_winbar;
1365 else
1366 expand_menu_alt = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001367 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001368 else // We're in the mapping part
Bram Moolenaar071d4272004-06-13 20:20:40 +00001369 xp->xp_context = EXPAND_NOTHING;
1370 return NULL;
1371}
1372
1373/*
1374 * Function given to ExpandGeneric() to obtain the list of (sub)menus (not
1375 * entries).
1376 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001377 char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001378get_menu_name(expand_T *xp UNUSED, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001379{
1380 static vimmenu_T *menu = NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001381 static int did_alt_menu = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001382 char_u *str;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001383#ifdef FEAT_MULTI_LANG
1384 static int should_advance = FALSE;
1385#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001386
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001387 if (idx == 0) // first call: start at first item
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001388 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001389 menu = expand_menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001390 did_alt_menu = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001391#ifdef FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001392 should_advance = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001393#endif
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001394 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001395
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001396 // Skip PopUp[nvoci].
Bram Moolenaar071d4272004-06-13 20:20:40 +00001397 while (menu != NULL && (menu_is_hidden(menu->dname)
1398 || menu_is_separator(menu->dname)
1399 || menu_is_tearoff(menu->dname)
1400 || menu->children == NULL))
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001401 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001402 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001403 if (menu == NULL && !did_alt_menu)
1404 {
1405 menu = expand_menu_alt;
1406 did_alt_menu = TRUE;
1407 }
1408 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001409
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001410 if (menu == NULL) // at end of linked list
Bram Moolenaar071d4272004-06-13 20:20:40 +00001411 return NULL;
1412
1413 if (menu->modes & expand_modes)
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001414#ifdef FEAT_MULTI_LANG
1415 if (should_advance)
1416 str = menu->en_dname;
1417 else
1418 {
1419#endif
1420 str = menu->dname;
1421#ifdef FEAT_MULTI_LANG
1422 if (menu->en_dname == NULL)
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001423 should_advance = TRUE;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001424 }
1425#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001426 else
1427 str = (char_u *)"";
1428
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001429#ifdef FEAT_MULTI_LANG
1430 if (should_advance)
1431#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001432 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001433 // Advance to next menu entry.
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001434 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001435 if (menu == NULL && !did_alt_menu)
1436 {
1437 menu = expand_menu_alt;
1438 did_alt_menu = TRUE;
1439 }
1440 }
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001441
1442#ifdef FEAT_MULTI_LANG
1443 should_advance = !should_advance;
1444#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001445
1446 return str;
1447}
1448
1449/*
1450 * Function given to ExpandGeneric() to obtain the list of menus and menu
1451 * entries.
1452 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001453 char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001454get_menu_names(expand_T *xp UNUSED, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001455{
1456 static vimmenu_T *menu = NULL;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001457 static int did_alt_menu = FALSE;
Bram Moolenaaref9d6aa2011-04-11 16:56:35 +02001458#define TBUFFER_LEN 256
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001459 static char_u tbuffer[TBUFFER_LEN]; //hack
Bram Moolenaar071d4272004-06-13 20:20:40 +00001460 char_u *str;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001461#ifdef FEAT_MULTI_LANG
1462 static int should_advance = FALSE;
1463#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001464
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001465 if (idx == 0) // first call: start at first item
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001466 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001467 menu = expand_menu;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001468 did_alt_menu = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001469#ifdef FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001470 should_advance = FALSE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001471#endif
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001472 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001473
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001474 // Skip Browse-style entries, popup menus and separators.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001475 while (menu != NULL
1476 && ( menu_is_hidden(menu->dname)
1477 || (expand_emenu && menu_is_separator(menu->dname))
1478 || menu_is_tearoff(menu->dname)
1479#ifndef FEAT_BROWSE
1480 || menu->dname[STRLEN(menu->dname) - 1] == '.'
1481#endif
1482 ))
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001483 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00001484 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001485 if (menu == NULL && !did_alt_menu)
1486 {
1487 menu = expand_menu_alt;
1488 did_alt_menu = TRUE;
1489 }
1490 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001491
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001492 if (menu == NULL) // at end of linked list
Bram Moolenaar071d4272004-06-13 20:20:40 +00001493 return NULL;
1494
1495 if (menu->modes & expand_modes)
1496 {
1497 if (menu->children != NULL)
1498 {
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001499#ifdef FEAT_MULTI_LANG
1500 if (should_advance)
Bram Moolenaaref9d6aa2011-04-11 16:56:35 +02001501 vim_strncpy(tbuffer, menu->en_dname, TBUFFER_LEN - 2);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001502 else
1503 {
1504#endif
Bram Moolenaaref9d6aa2011-04-11 16:56:35 +02001505 vim_strncpy(tbuffer, menu->dname, TBUFFER_LEN - 2);
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001506#ifdef FEAT_MULTI_LANG
1507 if (menu->en_dname == NULL)
1508 should_advance = TRUE;
1509 }
1510#endif
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001511 // hack on menu separators: use a 'magic' char for the separator
1512 // so that '.' in names gets escaped properly
Bram Moolenaar071d4272004-06-13 20:20:40 +00001513 STRCAT(tbuffer, "\001");
1514 str = tbuffer;
1515 }
1516 else
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001517#ifdef FEAT_MULTI_LANG
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001518 {
1519 if (should_advance)
1520 str = menu->en_dname;
1521 else
1522 {
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001523#endif
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001524 str = menu->dname;
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001525#ifdef FEAT_MULTI_LANG
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001526 if (menu->en_dname == NULL)
1527 should_advance = TRUE;
1528 }
1529 }
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001530#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001531 }
1532 else
1533 str = (char_u *)"";
1534
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001535#ifdef FEAT_MULTI_LANG
1536 if (should_advance)
1537#endif
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001538 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001539 // Advance to next menu entry.
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001540 menu = menu->next;
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02001541 if (menu == NULL && !did_alt_menu)
1542 {
1543 menu = expand_menu_alt;
1544 did_alt_menu = TRUE;
1545 }
1546 }
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001547
1548#ifdef FEAT_MULTI_LANG
1549 should_advance = !should_advance;
1550#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001551
1552 return str;
1553}
Bram Moolenaar071d4272004-06-13 20:20:40 +00001554
1555/*
1556 * Skip over this element of the menu path and return the start of the next
1557 * element. Any \ and ^Vs are removed from the current element.
Bram Moolenaar342337a2005-07-21 21:11:17 +00001558 * "name" may be modified.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001559 */
Bram Moolenaar5843f5f2019-08-20 20:13:45 +02001560 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001561menu_name_skip(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001562{
1563 char_u *p;
1564
Bram Moolenaar91acfff2017-03-12 19:22:36 +01001565 for (p = name; *p && *p != '.'; MB_PTR_ADV(p))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001566 {
1567 if (*p == '\\' || *p == Ctrl_V)
1568 {
Bram Moolenaar8c8de832008-06-24 22:58:06 +00001569 STRMOVE(p, p + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001570 if (*p == NUL)
1571 break;
1572 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00001573 }
1574 if (*p)
1575 *p++ = NUL;
1576 return p;
1577}
1578
1579/*
1580 * Return TRUE when "name" matches with menu "menu". The name is compared in
1581 * two ways: raw menu name and menu name without '&'. ignore part after a TAB.
1582 */
1583 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001584menu_name_equal(char_u *name, vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001585{
Bram Moolenaar41375642010-05-16 12:49:27 +02001586#ifdef FEAT_MULTI_LANG
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001587 if (menu->en_name != NULL
Bram Moolenaard91f7042011-01-04 17:49:32 +01001588 && (menu_namecmp(name, menu->en_name)
1589 || menu_namecmp(name, menu->en_dname)))
Bram Moolenaarcc448b32010-07-14 16:52:17 +02001590 return TRUE;
Bram Moolenaar41375642010-05-16 12:49:27 +02001591#endif
Bram Moolenaar70b11cd2010-05-14 22:24:40 +02001592 return menu_namecmp(name, menu->name) || menu_namecmp(name, menu->dname);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001593}
1594
1595 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001596menu_namecmp(char_u *name, char_u *mname)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001597{
1598 int i;
1599
1600 for (i = 0; name[i] != NUL && name[i] != TAB; ++i)
1601 if (name[i] != mname[i])
1602 break;
1603 return ((name[i] == NUL || name[i] == TAB)
1604 && (mname[i] == NUL || mname[i] == TAB));
1605}
1606
1607/*
1608 * Return the modes specified by the given menu command (eg :menu! returns
1609 * MENU_CMDLINE_MODE | MENU_INSERT_MODE).
1610 * If "noremap" is not NULL, then the flag it points to is set according to
1611 * whether the command is a "nore" command.
1612 * If "unmenu" is not NULL, then the flag it points to is set according to
1613 * whether the command is an "unmenu" command.
1614 */
1615 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001616get_menu_cmd_modes(
1617 char_u *cmd,
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001618 int forceit, // Was there a "!" after the command?
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001619 int *noremap,
1620 int *unmenu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001621{
1622 int modes;
1623
1624 switch (*cmd++)
1625 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001626 case 'v': // vmenu, vunmenu, vnoremenu
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001627 modes = MENU_VISUAL_MODE | MENU_SELECT_MODE;
1628 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001629 case 'x': // xmenu, xunmenu, xnoremenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001630 modes = MENU_VISUAL_MODE;
1631 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001632 case 's': // smenu, sunmenu, snoremenu
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001633 modes = MENU_SELECT_MODE;
1634 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001635 case 'o': // omenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001636 modes = MENU_OP_PENDING_MODE;
1637 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001638 case 'i': // imenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001639 modes = MENU_INSERT_MODE;
1640 break;
1641 case 't':
Bram Moolenaar6ed545e2022-05-09 20:09:23 +01001642 if (*cmd == 'l') // tlmenu, tlunmenu, tlnoremenu
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001643 {
1644 modes = MENU_TERMINAL_MODE;
1645 ++cmd;
1646 break;
1647 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001648 modes = MENU_TIP_MODE; // tmenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001649 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001650 case 'c': // cmenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001651 modes = MENU_CMDLINE_MODE;
1652 break;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001653 case 'a': // amenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001654 modes = MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001655 | MENU_VISUAL_MODE | MENU_SELECT_MODE
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00001656 | MENU_OP_PENDING_MODE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001657 break;
1658 case 'n':
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001659 if (*cmd != 'o') // nmenu, not noremenu
Bram Moolenaar071d4272004-06-13 20:20:40 +00001660 {
1661 modes = MENU_NORMAL_MODE;
1662 break;
1663 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001664 // FALLTHROUGH
Bram Moolenaar071d4272004-06-13 20:20:40 +00001665 default:
1666 --cmd;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001667 if (forceit) // menu!!
Bram Moolenaar071d4272004-06-13 20:20:40 +00001668 modes = MENU_INSERT_MODE | MENU_CMDLINE_MODE;
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001669 else // menu
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001670 modes = MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE
Bram Moolenaar071d4272004-06-13 20:20:40 +00001671 | MENU_OP_PENDING_MODE;
1672 }
1673
1674 if (noremap != NULL)
1675 *noremap = (*cmd == 'n' ? REMAP_NONE : REMAP_YES);
1676 if (unmenu != NULL)
1677 *unmenu = (*cmd == 'u');
1678 return modes;
1679}
1680
1681/*
Bram Moolenaar0eabd4d2020-03-15 16:13:53 +01001682 * Return the string representation of the menu modes. Does the opposite
1683 * of get_menu_cmd_modes().
1684 */
1685 static char_u *
1686get_menu_mode_str(int modes)
1687{
1688 if ((modes & (MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE |
1689 MENU_VISUAL_MODE | MENU_SELECT_MODE | MENU_OP_PENDING_MODE))
1690 == (MENU_INSERT_MODE | MENU_CMDLINE_MODE | MENU_NORMAL_MODE |
1691 MENU_VISUAL_MODE | MENU_SELECT_MODE | MENU_OP_PENDING_MODE))
1692 return (char_u *)"a";
1693 if ((modes & (MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE |
1694 MENU_OP_PENDING_MODE))
1695 == (MENU_NORMAL_MODE | MENU_VISUAL_MODE | MENU_SELECT_MODE |
1696 MENU_OP_PENDING_MODE))
1697 return (char_u *)" ";
1698 if ((modes & (MENU_INSERT_MODE | MENU_CMDLINE_MODE))
1699 == (MENU_INSERT_MODE | MENU_CMDLINE_MODE))
1700 return (char_u *)"!";
1701 if ((modes & (MENU_VISUAL_MODE | MENU_SELECT_MODE))
1702 == (MENU_VISUAL_MODE | MENU_SELECT_MODE))
1703 return (char_u *)"v";
1704 if (modes & MENU_VISUAL_MODE)
1705 return (char_u *)"x";
1706 if (modes & MENU_SELECT_MODE)
1707 return (char_u *)"s";
1708 if (modes & MENU_OP_PENDING_MODE)
1709 return (char_u *)"o";
1710 if (modes & MENU_INSERT_MODE)
1711 return (char_u *)"i";
1712 if (modes & MENU_TERMINAL_MODE)
1713 return (char_u *)"tl";
1714 if (modes & MENU_CMDLINE_MODE)
1715 return (char_u *)"c";
1716 if (modes & MENU_NORMAL_MODE)
1717 return (char_u *)"n";
1718 if (modes & MENU_TIP_MODE)
1719 return (char_u *)"t";
1720
1721 return (char_u *)"";
1722}
1723
1724/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00001725 * Modify a menu name starting with "PopUp" to include the mode character.
1726 * Returns the name in allocated memory (NULL for failure).
1727 */
1728 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001729popup_mode_name(char_u *name, int idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001730{
1731 char_u *p;
1732 int len = (int)STRLEN(name);
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001733 char *mode_chars = menu_mode_chars[idx];
1734 int mode_chars_len = (int)strlen(mode_chars);
1735 int i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001736
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001737 p = vim_strnsave(name, len + mode_chars_len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001738 if (p != NULL)
1739 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001740 mch_memmove(p + 5 + mode_chars_len, p + 5, (size_t)(len - 4));
1741 for (i = 0; i < mode_chars_len; ++i)
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001742 p[5 + i] = menu_mode_chars[idx][i];
Bram Moolenaar071d4272004-06-13 20:20:40 +00001743 }
1744 return p;
1745}
1746
1747#if defined(FEAT_GUI) || defined(PROTO)
1748/*
1749 * Return the index into the menu->strings or menu->noremap arrays for the
1750 * current state. Returns MENU_INDEX_INVALID if there is no mapping for the
1751 * given menu in the current mode.
1752 */
1753 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001754get_menu_index(vimmenu_T *menu, int state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001755{
1756 int idx;
1757
Bram Moolenaar24959102022-05-07 20:01:16 +01001758 if ((state & MODE_INSERT))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001759 idx = MENU_INDEX_INSERT;
Bram Moolenaar24959102022-05-07 20:01:16 +01001760 else if (state & MODE_CMDLINE)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001761 idx = MENU_INDEX_CMDLINE;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001762#ifdef FEAT_TERMINAL
1763 else if (term_use_loop())
1764 idx = MENU_INDEX_TERMINAL;
1765#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001766 else if (VIsual_active)
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001767 {
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00001768 if (VIsual_select)
1769 idx = MENU_INDEX_SELECT;
1770 else
1771 idx = MENU_INDEX_VISUAL;
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001772 }
Bram Moolenaar24959102022-05-07 20:01:16 +01001773 else if (state == MODE_HITRETURN || state == MODE_ASKMORE)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001774 idx = MENU_INDEX_CMDLINE;
1775 else if (finish_op)
1776 idx = MENU_INDEX_OP_PENDING;
Bram Moolenaar24959102022-05-07 20:01:16 +01001777 else if ((state & MODE_NORMAL))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001778 idx = MENU_INDEX_NORMAL;
1779 else
1780 idx = MENU_INDEX_INVALID;
1781
1782 if (idx != MENU_INDEX_INVALID && menu->strings[idx] == NULL)
1783 idx = MENU_INDEX_INVALID;
1784 return idx;
1785}
1786#endif
1787
1788/*
1789 * Duplicate the menu item text and then process to see if a mnemonic key
1790 * and/or accelerator text has been identified.
1791 * Returns a pointer to allocated memory, or NULL for failure.
1792 * If mnemonic != NULL, *mnemonic is set to the character after the first '&'.
1793 * If actext != NULL, *actext is set to the text after the first TAB.
1794 */
1795 static char_u *
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001796menu_text(char_u *str, int *mnemonic, char_u **actext)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001797{
1798 char_u *p;
1799 char_u *text;
1800
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001801 // Locate accelerator text, after the first TAB
Bram Moolenaar071d4272004-06-13 20:20:40 +00001802 p = vim_strchr(str, TAB);
1803 if (p != NULL)
1804 {
1805 if (actext != NULL)
1806 *actext = vim_strsave(p + 1);
Bram Moolenaar71ccd032020-06-12 22:59:11 +02001807 text = vim_strnsave(str, p - str);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001808 }
1809 else
1810 text = vim_strsave(str);
1811
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001812 // Find mnemonic characters "&a" and reduce "&&" to "&".
Bram Moolenaar071d4272004-06-13 20:20:40 +00001813 for (p = text; p != NULL; )
1814 {
1815 p = vim_strchr(p, '&');
1816 if (p != NULL)
1817 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001818 if (p[1] == NUL) // trailing "&"
Bram Moolenaar071d4272004-06-13 20:20:40 +00001819 break;
1820 if (mnemonic != NULL && p[1] != '&')
1821#if !defined(__MVS__) || defined(MOTIF390_MNEMONIC_FIXED)
1822 *mnemonic = p[1];
1823#else
1824 {
1825 /*
1826 * Well there is a bug in the Motif libraries on OS390 Unix.
1827 * The mnemonic keys needs to be converted to ASCII values
1828 * first.
1829 * This behavior has been seen in 2.8 and 2.9.
1830 */
1831 char c = p[1];
1832 __etoa_l(&c, 1);
1833 *mnemonic = c;
1834 }
1835#endif
Bram Moolenaar8c8de832008-06-24 22:58:06 +00001836 STRMOVE(p, p + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001837 p = p + 1;
1838 }
1839 }
1840 return text;
1841}
1842
1843/*
1844 * Return TRUE if "name" can be a menu in the MenuBar.
1845 */
1846 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001847menu_is_menubar(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001848{
1849 return (!menu_is_popup(name)
1850 && !menu_is_toolbar(name)
Bram Moolenaar378daf82017-09-23 23:58:28 +02001851 && !menu_is_winbar(name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001852 && *name != MNU_HIDDEN_CHAR);
1853}
1854
1855/*
1856 * Return TRUE if "name" is a popup menu name.
1857 */
1858 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001859menu_is_popup(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001860{
1861 return (STRNCMP(name, "PopUp", 5) == 0);
1862}
1863
1864#if (defined(FEAT_GUI_MOTIF) && (XmVersion <= 1002)) || defined(PROTO)
1865/*
1866 * Return TRUE if "name" is part of a popup menu.
1867 */
1868 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001869menu_is_child_of_popup(vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001870{
1871 while (menu->parent != NULL)
1872 menu = menu->parent;
1873 return menu_is_popup(menu->name);
1874}
1875#endif
1876
1877/*
1878 * Return TRUE if "name" is a toolbar menu name.
1879 */
1880 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001881menu_is_toolbar(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001882{
1883 return (STRNCMP(name, "ToolBar", 7) == 0);
1884}
1885
1886/*
1887 * Return TRUE if the name is a menu separator identifier: Starts and ends
1888 * with '-'
1889 */
1890 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001891menu_is_separator(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001892{
1893 return (name[0] == '-' && name[STRLEN(name) - 1] == '-');
1894}
1895
1896/*
1897 * Return TRUE if the menu is hidden: Starts with ']'
1898 */
1899 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001900menu_is_hidden(char_u *name)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001901{
1902 return (name[0] == ']') || (menu_is_popup(name) && name[5] != NUL);
1903}
1904
Bram Moolenaar071d4272004-06-13 20:20:40 +00001905/*
1906 * Return TRUE if the menu is the tearoff menu.
1907 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001908 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001909menu_is_tearoff(char_u *name UNUSED)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001910{
1911#ifdef FEAT_GUI
1912 return (STRCMP(name, TEAR_STRING) == 0);
1913#else
1914 return FALSE;
1915#endif
1916}
Bram Moolenaar071d4272004-06-13 20:20:40 +00001917
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001918#if defined(FEAT_GUI) || defined(FEAT_TERM_POPUP_MENU) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001919
1920 static int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01001921get_menu_mode(void)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001922{
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001923#ifdef FEAT_TERMINAL
1924 if (term_use_loop())
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001925 return MENU_INDEX_TERMINAL;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001926#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001927 if (VIsual_active)
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001928 {
Bram Moolenaarc9b4b052006-04-30 18:54:39 +00001929 if (VIsual_select)
1930 return MENU_INDEX_SELECT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001931 return MENU_INDEX_VISUAL;
Bram Moolenaarb3656ed2006-03-20 21:59:49 +00001932 }
Bram Moolenaar24959102022-05-07 20:01:16 +01001933 if (State & MODE_INSERT)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001934 return MENU_INDEX_INSERT;
Bram Moolenaar24959102022-05-07 20:01:16 +01001935 if ((State & MODE_CMDLINE) || State == MODE_ASKMORE
1936 || State == MODE_HITRETURN)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001937 return MENU_INDEX_CMDLINE;
1938 if (finish_op)
1939 return MENU_INDEX_OP_PENDING;
Bram Moolenaar24959102022-05-07 20:01:16 +01001940 if (State & MODE_NORMAL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001941 return MENU_INDEX_NORMAL;
Bram Moolenaar24959102022-05-07 20:01:16 +01001942 if (State & MODE_LANGMAP) // must be a "r" command, like Insert mode
Bram Moolenaar071d4272004-06-13 20:20:40 +00001943 return MENU_INDEX_INSERT;
1944 return MENU_INDEX_INVALID;
1945}
1946
Bram Moolenaar29a2c082018-03-05 21:06:23 +01001947 int
1948get_menu_mode_flag(void)
1949{
1950 int mode = get_menu_mode();
1951
1952 if (mode == MENU_INDEX_INVALID)
1953 return 0;
1954 return 1 << mode;
1955}
1956
Bram Moolenaar071d4272004-06-13 20:20:40 +00001957/*
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001958 * Display the Special "PopUp" menu as a pop-up at the current mouse
1959 * position. The "PopUpn" menu is for Normal mode, "PopUpi" for Insert mode,
1960 * etc.
1961 */
1962 void
1963show_popupmenu(void)
1964{
1965 vimmenu_T *menu;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001966 int menu_mode;
1967 char* mode;
1968 int mode_len;
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001969
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001970 menu_mode = get_menu_mode();
1971 if (menu_mode == MENU_INDEX_INVALID)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001972 return;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001973 mode = menu_mode_chars[menu_mode];
1974 mode_len = (int)strlen(mode);
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001975
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001976 apply_autocmds(EVENT_MENUPOPUP, (char_u*)mode, NULL, FALSE, curbuf);
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001977
Bram Moolenaar00d253e2020-04-06 22:13:01 +02001978 FOR_ALL_MENUS(menu)
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02001979 if (STRNCMP("PopUp", menu->name, 5) == 0 && STRNCMP(menu->name + 5, mode, mode_len) == 0)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001980 break;
1981
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001982 // Only show a popup when it is defined and has entries
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001983 if (menu != NULL && menu->children != NULL)
1984 {
1985# if defined(FEAT_GUI)
1986 if (gui.in_use)
1987 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01001988 // Update the menus now, in case the MenuPopup autocommand did
1989 // anything.
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01001990 gui_update_menus(0);
1991 gui_mch_show_popupmenu(menu);
1992 }
1993# endif
1994# if defined(FEAT_GUI) && defined(FEAT_TERM_POPUP_MENU)
1995 else
1996# endif
1997# if defined(FEAT_TERM_POPUP_MENU)
1998 pum_show_popupmenu(menu);
1999# endif
2000 }
2001}
2002#endif
2003
2004#if defined(FEAT_GUI) || defined(PROTO)
2005
2006/*
Bram Moolenaar968bbbe2006-08-16 19:41:08 +00002007 * Check that a pointer appears in the menu tree. Used to protect from using
2008 * a menu that was deleted after it was selected but before the event was
2009 * handled.
2010 * Return OK or FAIL. Used recursively.
2011 */
2012 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002013check_menu_pointer(vimmenu_T *root, vimmenu_T *menu_to_check)
Bram Moolenaar968bbbe2006-08-16 19:41:08 +00002014{
2015 vimmenu_T *p;
2016
2017 for (p = root; p != NULL; p = p->next)
2018 if (p == menu_to_check
2019 || (p->children != NULL
2020 && check_menu_pointer(p->children, menu_to_check) == OK))
2021 return OK;
2022 return FAIL;
2023}
2024
2025/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00002026 * After we have started the GUI, then we can create any menus that have been
2027 * defined. This is done once here. add_menu_path() may have already been
2028 * called to define these menus, and may be called again. This function calls
2029 * itself recursively. Should be called at the top level with:
Bram Moolenaara06ecab2016-07-16 14:47:36 +02002030 * gui_create_initial_menus(root_menu);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002031 */
2032 void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002033gui_create_initial_menus(vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002034{
2035 int idx = 0;
2036
2037 while (menu != NULL)
2038 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002039 // Don't add a menu when only a tip was defined.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002040 if (menu->modes & MENU_ALL_MODES)
2041 {
2042 if (menu->children != NULL)
2043 {
2044 gui_mch_add_menu(menu, idx);
2045 gui_create_initial_menus(menu->children);
2046 }
2047 else
2048 gui_mch_add_menu_item(menu, idx);
2049 }
2050 menu = menu->next;
2051 ++idx;
2052 }
2053}
2054
2055/*
2056 * Used recursively by gui_update_menus (see below)
2057 */
2058 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002059gui_update_menus_recurse(vimmenu_T *menu, int mode)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002060{
2061 int grey;
2062
2063 while (menu)
2064 {
2065 if ((menu->modes & menu->enabled & mode)
Bram Moolenaar4f974752019-02-17 17:44:42 +01002066# if defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002067 || menu_is_tearoff(menu->dname)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002068# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002069 )
2070 grey = FALSE;
2071 else
2072 grey = TRUE;
Bram Moolenaar0b962e52022-04-03 18:02:37 +01002073
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002074 // Never hide a toplevel menu, it may make the menubar resize or
2075 // disappear. Same problem for ToolBar items.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002076 if (vim_strchr(p_go, GO_GREY) != NULL || menu->parent == NULL
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002077# ifdef FEAT_TOOLBAR
Bram Moolenaar071d4272004-06-13 20:20:40 +00002078 || menu_is_toolbar(menu->parent->name)
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002079# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002080 )
2081 gui_mch_menu_grey(menu, grey);
2082 else
2083 gui_mch_menu_hidden(menu, grey);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002084 gui_update_menus_recurse(menu->children, mode);
2085 menu = menu->next;
2086 }
2087}
2088
2089/*
2090 * Make sure only the valid menu items appear for this mode. If
2091 * force_menu_update is not TRUE, then we only do this if the mode has changed
2092 * since last time. If "modes" is not 0, then we use these modes instead.
2093 */
2094 void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002095gui_update_menus(int modes)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002096{
2097 static int prev_mode = -1;
2098 int mode = 0;
2099
2100 if (modes != 0x0)
2101 mode = modes;
2102 else
Bram Moolenaar29a2c082018-03-05 21:06:23 +01002103 mode = get_menu_mode_flag();
Bram Moolenaar071d4272004-06-13 20:20:40 +00002104
2105 if (force_menu_update || mode != prev_mode)
2106 {
2107 gui_update_menus_recurse(root_menu, mode);
2108 gui_mch_draw_menubar();
2109 prev_mode = mode;
2110 force_menu_update = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002111 }
2112}
2113
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002114# if defined(FEAT_GUI_MSWIN) || defined(FEAT_GUI_MOTIF) \
Bram Moolenaar241a8aa2005-12-06 20:04:44 +00002115 || defined(FEAT_GUI_GTK) || defined(FEAT_GUI_PHOTON) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002116/*
2117 * Check if a key is used as a mnemonic for a toplevel menu.
2118 * Case of the key is ignored.
2119 */
2120 int
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002121gui_is_menu_shortcut(int key)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002122{
2123 vimmenu_T *menu;
2124
2125 if (key < 256)
2126 key = TOLOWER_LOC(key);
Bram Moolenaar00d253e2020-04-06 22:13:01 +02002127 FOR_ALL_MENUS(menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002128 if (menu->mnemonic == key
2129 || (menu->mnemonic < 256 && TOLOWER_LOC(menu->mnemonic) == key))
2130 return TRUE;
2131 return FALSE;
2132}
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002133# endif
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002134#endif // FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00002135
Bram Moolenaar4f974752019-02-17 17:44:42 +01002136#if (defined(FEAT_GUI_MSWIN) && defined(FEAT_TEAROFF)) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002137
2138/*
2139 * Deal with tearoff items that are added like a menu item.
2140 * Currently only for Win32 GUI. Others may follow later.
2141 */
2142
2143 void
2144gui_mch_toggle_tearoffs(int enable)
2145{
2146 int pri_tab[MENUDEPTH + 1];
2147 int i;
2148
2149 if (enable)
2150 {
2151 for (i = 0; i < MENUDEPTH; ++i)
2152 pri_tab[i] = 500;
2153 pri_tab[MENUDEPTH] = -1;
2154 gui_create_tearoffs_recurse(root_menu, (char_u *)"", pri_tab, 0);
2155 }
2156 else
2157 gui_destroy_tearoffs_recurse(root_menu);
2158 s_tearoffs = enable;
2159}
2160
2161/*
2162 * Recursively add tearoff items
2163 */
2164 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002165gui_create_tearoffs_recurse(
2166 vimmenu_T *menu,
2167 const char_u *pname,
2168 int *pri_tab,
2169 int pri_idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002170{
2171 char_u *newpname = NULL;
2172 int len;
2173 char_u *s;
2174 char_u *d;
2175
2176 if (pri_tab[pri_idx + 1] != -1)
2177 ++pri_idx;
2178 while (menu != NULL)
2179 {
2180 if (menu->children != NULL && menu_is_menubar(menu->name))
2181 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002182 // Add the menu name to the menu path. Insert a backslash before
2183 // dots (it's used to separate menu names).
Bram Moolenaar071d4272004-06-13 20:20:40 +00002184 len = (int)STRLEN(pname) + (int)STRLEN(menu->name);
2185 for (s = menu->name; *s; ++s)
2186 if (*s == '.' || *s == '\\')
2187 ++len;
2188 newpname = alloc(len + TEAR_LEN + 2);
2189 if (newpname != NULL)
2190 {
2191 STRCPY(newpname, pname);
2192 d = newpname + STRLEN(newpname);
2193 for (s = menu->name; *s; ++s)
2194 {
2195 if (*s == '.' || *s == '\\')
2196 *d++ = '\\';
2197 *d++ = *s;
2198 }
2199 *d = NUL;
2200
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002201 // check if tearoff already exists
Bram Moolenaar071d4272004-06-13 20:20:40 +00002202 if (STRCMP(menu->children->name, TEAR_STRING) != 0)
2203 {
2204 gui_add_tearoff(newpname, pri_tab, pri_idx - 1);
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002205 *d = NUL; // remove TEAR_STRING
Bram Moolenaar071d4272004-06-13 20:20:40 +00002206 }
2207
2208 STRCAT(newpname, ".");
2209 gui_create_tearoffs_recurse(menu->children, newpname,
2210 pri_tab, pri_idx);
2211 vim_free(newpname);
2212 }
2213 }
2214 menu = menu->next;
2215 }
2216}
2217
2218/*
2219 * Add tear-off menu item for a submenu.
2220 * "tearpath" is the menu path, and must have room to add TEAR_STRING.
2221 */
2222 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002223gui_add_tearoff(char_u *tearpath, int *pri_tab, int pri_idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002224{
2225 char_u *tbuf;
2226 int t;
2227 vimmenu_T menuarg;
2228
2229 tbuf = alloc(5 + (unsigned int)STRLEN(tearpath));
2230 if (tbuf != NULL)
2231 {
2232 tbuf[0] = K_SPECIAL;
2233 tbuf[1] = K_SECOND(K_TEAROFF);
2234 tbuf[2] = K_THIRD(K_TEAROFF);
2235 STRCPY(tbuf + 3, tearpath);
2236 STRCAT(tbuf + 3, "\r");
2237
2238 STRCAT(tearpath, ".");
2239 STRCAT(tearpath, TEAR_STRING);
2240
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002241 // Priority of tear-off is always 1
Bram Moolenaar071d4272004-06-13 20:20:40 +00002242 t = pri_tab[pri_idx + 1];
2243 pri_tab[pri_idx + 1] = 1;
2244
2245#ifdef FEAT_TOOLBAR
2246 menuarg.iconfile = NULL;
2247 menuarg.iconidx = -1;
2248 menuarg.icon_builtin = FALSE;
2249#endif
2250 menuarg.noremap[0] = REMAP_NONE;
2251 menuarg.silent[0] = TRUE;
2252
2253 menuarg.modes = MENU_ALL_MODES;
2254 add_menu_path(tearpath, &menuarg, pri_tab, tbuf, FALSE);
2255
2256 menuarg.modes = MENU_TIP_MODE;
2257 add_menu_path(tearpath, &menuarg, pri_tab,
2258 (char_u *)_("Tear off this menu"), FALSE);
2259
2260 pri_tab[pri_idx + 1] = t;
2261 vim_free(tbuf);
2262 }
2263}
2264
2265/*
2266 * Recursively destroy tearoff items
2267 */
2268 static void
Bram Moolenaar52ea13d2016-01-30 18:51:09 +01002269gui_destroy_tearoffs_recurse(vimmenu_T *menu)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002270{
2271 while (menu)
2272 {
2273 if (menu->children)
2274 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002275 // check if tearoff exists
Bram Moolenaar071d4272004-06-13 20:20:40 +00002276 if (STRCMP(menu->children->name, TEAR_STRING) == 0)
2277 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002278 // Disconnect the item and free the memory
Bram Moolenaar071d4272004-06-13 20:20:40 +00002279 free_menu(&menu->children);
2280 }
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002281 if (menu->children != NULL) // if not the last one
Bram Moolenaar071d4272004-06-13 20:20:40 +00002282 gui_destroy_tearoffs_recurse(menu->children);
2283 }
2284 menu = menu->next;
2285 }
2286}
2287
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002288#endif // FEAT_GUI_MSWIN && FEAT_TEAROFF
Bram Moolenaar071d4272004-06-13 20:20:40 +00002289
2290/*
Bram Moolenaar1b9645d2017-09-17 23:03:31 +02002291 * Execute "menu". Use by ":emenu" and the window toolbar.
2292 * "eap" is NULL for the window toolbar.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002293 * "mode_idx" specifies a MENU_INDEX_ value, use -1 to depend on the current
2294 * state.
Bram Moolenaar071d4272004-06-13 20:20:40 +00002295 */
Bram Moolenaaraef8c3d2018-03-03 18:59:16 +01002296 void
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002297execute_menu(exarg_T *eap, vimmenu_T *menu, int mode_idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002298{
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002299 int idx = mode_idx;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002300
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002301 if (idx < 0)
2302 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002303 // Use the Insert mode entry when returning to Insert mode.
Bram Moolenaar9b8d6222020-12-28 18:26:00 +01002304 if (restart_edit && !current_sctx.sc_sid)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002305 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002306 idx = MENU_INDEX_INSERT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002307 }
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002308#ifdef FEAT_TERMINAL
2309 else if (term_use_loop())
Bram Moolenaar071d4272004-06-13 20:20:40 +00002310 {
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002311 idx = MENU_INDEX_TERMINAL;
2312 }
2313#endif
2314 else if (VIsual_active)
2315 {
2316 idx = MENU_INDEX_VISUAL;
2317 }
2318 else if (eap != NULL && eap->addr_count)
2319 {
2320 pos_T tpos;
2321
2322 idx = MENU_INDEX_VISUAL;
2323
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002324 // GEDDES: This is not perfect - but it is a
2325 // quick way of detecting whether we are doing this from a
2326 // selection - see if the range matches up with the visual
2327 // select start and end.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002328 if ((curbuf->b_visual.vi_start.lnum == eap->line1)
2329 && (curbuf->b_visual.vi_end.lnum) == eap->line2)
2330 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002331 // Set it up for visual mode - equivalent to gv.
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002332 VIsual_mode = curbuf->b_visual.vi_mode;
2333 tpos = curbuf->b_visual.vi_end;
2334 curwin->w_cursor = curbuf->b_visual.vi_start;
2335 curwin->w_curswant = curbuf->b_visual.vi_curswant;
2336 }
2337 else
2338 {
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002339 // Set it up for line-wise visual mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002340 VIsual_mode = 'V';
2341 curwin->w_cursor.lnum = eap->line1;
2342 curwin->w_cursor.col = 1;
2343 tpos.lnum = eap->line2;
2344 tpos.col = MAXCOL;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002345 tpos.coladd = 0;
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002346 }
2347
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002348 // Activate visual mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002349 VIsual_active = TRUE;
2350 VIsual_reselect = TRUE;
2351 check_cursor();
2352 VIsual = curwin->w_cursor;
2353 curwin->w_cursor = tpos;
2354
2355 check_cursor();
2356
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002357 // Adjust the cursor to make sure it is in the correct pos
2358 // for exclusive mode
Bram Moolenaar4c5d8152018-10-19 22:36:53 +02002359 if (*p_sel == 'e' && gchar_cursor() != NUL)
2360 ++curwin->w_cursor.col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002361 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002362 }
Bram Moolenaara21a6a92017-09-23 16:33:50 +02002363
Bram Moolenaar4ba37b52019-12-04 21:57:43 +01002364 // For the WinBar menu always use the Normal mode menu.
zeertzjq79ae1522022-07-01 12:13:15 +01002365 if (idx == MENU_INDEX_INVALID || eap == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002366 idx = MENU_INDEX_NORMAL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002367
zeertzjq79ae1522022-07-01 12:13:15 +01002368 if (menu->strings[idx] != NULL && (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(
zeertzjqcdc83932022-09-12 13:38:41 +01002894 menu->strings[bit], FALSE, FALSE)));
Bram Moolenaar292b90d2020-03-18 15:23:16 +01002895 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