(single_submenu): Initialize wv->help to Qnil.
[bpt/emacs.git] / src / xmenu.c
CommitLineData
dcfdbac7 1/* X Communication module for terminals which understand the X protocol.
950eaee7
GM
2 Copyright (C) 1986, 88, 93, 94, 96, 99, 2000, 2001
3 Free Software Foundation, Inc.
dcfdbac7
JB
4
5This file is part of GNU Emacs.
6
7GNU Emacs is free software; you can redistribute it and/or modify
8it under the terms of the GNU General Public License as published by
774910eb 9the Free Software Foundation; either version 2, or (at your option)
dcfdbac7
JB
10any later version.
11
12GNU Emacs is distributed in the hope that it will be useful,
13but WITHOUT ANY WARRANTY; without even the implied warranty of
14MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15GNU General Public License for more details.
16
17You should have received a copy of the GNU General Public License
18along with GNU Emacs; see the file COPYING. If not, write to
3b7ad313
EN
19the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
20Boston, MA 02111-1307, USA. */
dcfdbac7 21
c9e60620 22/* X pop-up deck-of-cards menu facility for GNU Emacs.
dcfdbac7
JB
23 *
24 * Written by Jon Arnold and Roman Budzianowski
25 * Mods and rewrite by Robert Krawitz
26 *
27 */
28
18686d47
RS
29/* Modified by Fred Pierresteguy on December 93
30 to make the popup menus and menubar use the Xt. */
31
78589e07
RS
32/* Rewritten for clarity and GC protection by rms in Feb 94. */
33
68c45bf0
PE
34#include <config.h>
35
dcfdbac7
JB
36/* On 4.3 this loses if it comes after xterm.h. */
37#include <signal.h>
565620a5
RS
38
39#include <stdio.h>
7ee72033 40
dcfdbac7 41#include "lisp.h"
18686d47 42#include "termhooks.h"
02067692 43#include "keyboard.h"
5ee707b8 44#include "keymap.h"
7708e9bd 45#include "frame.h"
dcfdbac7 46#include "window.h"
9ac0d9e0 47#include "blockinput.h"
88766961 48#include "buffer.h"
cc45fb40
SM
49#include "charset.h"
50#include "coding.h"
dcfdbac7 51
eeee3112
RS
52#ifdef MSDOS
53#include "msdos.h"
54#endif
55
87485d6f 56#ifdef HAVE_X_WINDOWS
dcfdbac7
JB
57/* This may include sys/types.h, and that somehow loses
58 if this is not done before the other system files. */
59#include "xterm.h"
87485d6f 60#endif
dcfdbac7
JB
61
62/* Load sys/types.h if not already loaded.
63 In some systems loading it twice is suicidal. */
64#ifndef makedev
65#include <sys/types.h>
66#endif
67
68#include "dispextern.h"
69
87485d6f 70#ifdef HAVE_X_WINDOWS
703dc2a8 71#undef HAVE_MULTILINGUAL_MENU
18686d47 72#ifdef USE_X_TOOLKIT
991786dd 73#include "widget.h"
18686d47
RS
74#include <X11/Xlib.h>
75#include <X11/IntrinsicP.h>
76#include <X11/CoreP.h>
77#include <X11/StringDefs.h>
60f312e2 78#include <X11/Shell.h>
5ee80bd3 79#ifdef USE_LUCID
1d1c1567 80#include <X11/Xaw/Paned.h>
5ee80bd3 81#endif /* USE_LUCID */
18686d47 82#include "../lwlib/lwlib.h"
a352a815
RS
83#else /* not USE_X_TOOLKIT */
84#include "../oldXMenu/XMenu.h"
85#endif /* not USE_X_TOOLKIT */
86#endif /* HAVE_X_WINDOWS */
18686d47 87
dcfdbac7
JB
88#ifndef TRUE
89#define TRUE 1
90#define FALSE 0
78589e07 91#endif /* no TRUE */
dcfdbac7 92
bfc524bc
RS
93Lisp_Object Vmenu_updating_frame;
94
0314aacb
RS
95Lisp_Object Qdebug_on_next_call;
96
18686d47 97extern Lisp_Object Qmenu_bar;
92280f67 98extern Lisp_Object Qmouse_click, Qevent_kind;
78589e07 99
de57a39c 100extern Lisp_Object QCtoggle, QCradio;
8fbc986d 101
88766961
RS
102extern Lisp_Object Voverriding_local_map;
103extern Lisp_Object Voverriding_local_map_menu_flag;
104
105extern Lisp_Object Qoverriding_local_map, Qoverriding_terminal_local_map;
106
107extern Lisp_Object Qmenu_bar_update_hook;
108
18686d47 109#ifdef USE_X_TOOLKIT
88766961 110extern void set_frame_menubar ();
78589e07
RS
111extern void process_expose_from_menu ();
112extern XtAppContext Xt_app_con;
113
165e1749 114static Lisp_Object xdialog_show ();
4dedbfe0 115void popup_get_selection ();
3427a3db 116
c9e60620 117/* Define HAVE_BOXES if menus can handle radio and toggle buttons. */
3427a3db
GM
118
119#define HAVE_BOXES 1
120#endif
121
122static void push_menu_item P_ ((Lisp_Object, Lisp_Object, Lisp_Object,
123 Lisp_Object, Lisp_Object, Lisp_Object,
850df50b 124 Lisp_Object, Lisp_Object));
f61a541b
GM
125static int update_frame_menubar P_ ((struct frame *));
126static Lisp_Object xmenu_show P_ ((struct frame *, int, int, int, int,
127 Lisp_Object, char **));
128static void keymap_panes P_ ((Lisp_Object *, int, int));
129static void single_keymap_panes P_ ((Lisp_Object, Lisp_Object, Lisp_Object,
130 int, int));
131static void single_menu_item P_ ((Lisp_Object, Lisp_Object, Lisp_Object *,
132 int, int, int *));
133static void list_of_panes P_ ((Lisp_Object));
134static void list_of_items P_ ((Lisp_Object));
78589e07
RS
135\f
136/* This holds a Lisp vector that holds the results of decoding
137 the keymaps or alist-of-alists that specify a menu.
dcfdbac7 138
78589e07 139 It describes the panes and items within the panes.
dcfdbac7 140
78589e07
RS
141 Each pane is described by 3 elements in the vector:
142 t, the pane name, the pane's prefix key.
a352a815 143 Then follow the pane's items, with 5 elements per item:
78589e07 144 the item string, the enable flag, the item's value,
a352a815 145 the definition, and the equivalent keyboard key's description string.
dcfdbac7 146
101bb4a5
RS
147 In some cases, multiple levels of menus may be described.
148 A single vector slot containing nil indicates the start of a submenu.
149 A single vector slot containing lambda indicates the end of a submenu.
150 The submenu follows a menu item which is the way to reach the submenu.
151
fcaa7665
RS
152 A single vector slot containing quote indicates that the
153 following items should appear on the right of a dialog box.
154
78589e07
RS
155 Using a Lisp vector to hold this information while we decode it
156 takes care of protecting all the data from GC. */
dcfdbac7 157
78589e07
RS
158#define MENU_ITEMS_PANE_NAME 1
159#define MENU_ITEMS_PANE_PREFIX 2
160#define MENU_ITEMS_PANE_LENGTH 3
088831f6 161
850df50b
GM
162enum menu_item_idx
163{
164 MENU_ITEMS_ITEM_NAME = 0,
165 MENU_ITEMS_ITEM_ENABLE,
166 MENU_ITEMS_ITEM_VALUE,
167 MENU_ITEMS_ITEM_EQUIV_KEY,
168 MENU_ITEMS_ITEM_DEFINITION,
169 MENU_ITEMS_ITEM_TYPE,
170 MENU_ITEMS_ITEM_SELECTED,
171 MENU_ITEMS_ITEM_HELP,
172 MENU_ITEMS_ITEM_LENGTH
173};
7da99777 174
78589e07 175static Lisp_Object menu_items;
18686d47 176
78589e07
RS
177/* Number of slots currently allocated in menu_items. */
178static int menu_items_allocated;
18686d47 179
78589e07
RS
180/* This is the index in menu_items of the first empty slot. */
181static int menu_items_used;
18686d47 182
101bb4a5
RS
183/* The number of panes currently recorded in menu_items,
184 excluding those within submenus. */
78589e07 185static int menu_items_n_panes;
18686d47 186
101bb4a5
RS
187/* Current depth within submenus. */
188static int menu_items_submenu_depth;
189
4dedbfe0 190/* Flag which when set indicates a dialog or menu has been posted by
c98fcf4b 191 Xt on behalf of one of the widget sets. */
7555d825 192int popup_activated_flag;
4dedbfe0 193
88766961 194static int next_menubar_widget_id;
745c34fb 195
a9be6839
RS
196/* This is set nonzero after the user activates the menu bar, and set
197 to zero again after the menu bars are redisplayed by prepare_menu_bar.
198 While it is nonzero, all calls to set_frame_menubar go deep.
199
200 I don't understand why this is needed, but it does seem to be
201 needed on Motif, according to Marcus Daniels <marcus@sysc.pdx.edu>. */
202
203int pending_menu_activation;
bd3a4da2 204\f
88766961 205#ifdef USE_X_TOOLKIT
bd3a4da2 206
7556890b 207/* Return the frame whose ->output_data.x->id equals ID, or 0 if none. */
bd3a4da2 208
88766961
RS
209static struct frame *
210menubar_id_to_frame (id)
211 LWLIB_ID id;
212{
213 Lisp_Object tail, frame;
214 FRAME_PTR f;
bd3a4da2 215
8e713be6 216 for (tail = Vframe_list; GC_CONSP (tail); tail = XCDR (tail))
bd3a4da2 217 {
8e713be6 218 frame = XCAR (tail);
88766961
RS
219 if (!GC_FRAMEP (frame))
220 continue;
221 f = XFRAME (frame);
2d764c78 222 if (!FRAME_WINDOW_P (f))
88766961 223 continue;
7556890b 224 if (f->output_data.x->id == id)
88766961 225 return f;
bd3a4da2 226 }
88766961 227 return 0;
bd3a4da2 228}
88766961
RS
229
230#endif
4dedbfe0 231\f
78589e07
RS
232/* Initialize the menu_items structure if we haven't already done so.
233 Also mark it as currently empty. */
234
235static void
236init_menu_items ()
237{
238 if (NILP (menu_items))
239 {
240 menu_items_allocated = 60;
241 menu_items = Fmake_vector (make_number (menu_items_allocated), Qnil);
18686d47
RS
242 }
243
78589e07
RS
244 menu_items_used = 0;
245 menu_items_n_panes = 0;
101bb4a5 246 menu_items_submenu_depth = 0;
78589e07 247}
18686d47 248
0b1f4572 249/* Call at the end of generating the data in menu_items. */
1658603c 250
78589e07
RS
251static void
252finish_menu_items ()
253{
254}
1658603c 255
78589e07
RS
256/* Call when finished using the data for the current menu
257 in menu_items. */
1658603c 258
78589e07
RS
259static void
260discard_menu_items ()
261{
262 /* Free the structure if it is especially large.
263 Otherwise, hold on to it, to save time. */
264 if (menu_items_allocated > 200)
265 {
266 menu_items = Qnil;
267 menu_items_allocated = 0;
268 }
269}
1658603c 270
101bb4a5
RS
271/* Make the menu_items vector twice as large. */
272
273static void
274grow_menu_items ()
275{
276 Lisp_Object old;
277 int old_size = menu_items_allocated;
278 old = menu_items;
279
280 menu_items_allocated *= 2;
281 menu_items = Fmake_vector (make_number (menu_items_allocated), Qnil);
282 bcopy (XVECTOR (old)->contents, XVECTOR (menu_items)->contents,
283 old_size * sizeof (Lisp_Object));
284}
285
286/* Begin a submenu. */
287
288static void
289push_submenu_start ()
290{
291 if (menu_items_used + 1 > menu_items_allocated)
292 grow_menu_items ();
293
294 XVECTOR (menu_items)->contents[menu_items_used++] = Qnil;
295 menu_items_submenu_depth++;
296}
297
298/* End a submenu. */
299
300static void
301push_submenu_end ()
302{
303 if (menu_items_used + 1 > menu_items_allocated)
304 grow_menu_items ();
305
306 XVECTOR (menu_items)->contents[menu_items_used++] = Qlambda;
307 menu_items_submenu_depth--;
308}
309
fcaa7665
RS
310/* Indicate boundary between left and right. */
311
312static void
313push_left_right_boundary ()
314{
315 if (menu_items_used + 1 > menu_items_allocated)
316 grow_menu_items ();
317
318 XVECTOR (menu_items)->contents[menu_items_used++] = Qquote;
319}
320
c9e60620 321/* Start a new menu pane in menu_items.
78589e07 322 NAME is the pane name. PREFIX_VEC is a prefix key for this pane. */
1658603c 323
78589e07
RS
324static void
325push_menu_pane (name, prefix_vec)
326 Lisp_Object name, prefix_vec;
327{
328 if (menu_items_used + MENU_ITEMS_PANE_LENGTH > menu_items_allocated)
101bb4a5 329 grow_menu_items ();
dcfdbac7 330
101bb4a5
RS
331 if (menu_items_submenu_depth == 0)
332 menu_items_n_panes++;
78589e07
RS
333 XVECTOR (menu_items)->contents[menu_items_used++] = Qt;
334 XVECTOR (menu_items)->contents[menu_items_used++] = name;
335 XVECTOR (menu_items)->contents[menu_items_used++] = prefix_vec;
336}
dcfdbac7 337
3427a3db
GM
338/* Push one menu item into the current pane. NAME is the string to
339 display. ENABLE if non-nil means this item can be selected. KEY
340 is the key generated by choosing this item, or nil if this item
341 doesn't really have a definition. DEF is the definition of this
342 item. EQUIV is the textual description of the keyboard equivalent
343 for this item (or nil if none). TYPE is the type of this menu
344 item, one of nil, `toggle' or `radio'. */
18686d47 345
78589e07 346static void
850df50b
GM
347push_menu_item (name, enable, key, def, equiv, type, selected, help)
348 Lisp_Object name, enable, key, def, equiv, type, selected, help;
78589e07
RS
349{
350 if (menu_items_used + MENU_ITEMS_ITEM_LENGTH > menu_items_allocated)
101bb4a5 351 grow_menu_items ();
088831f6 352
78589e07
RS
353 XVECTOR (menu_items)->contents[menu_items_used++] = name;
354 XVECTOR (menu_items)->contents[menu_items_used++] = enable;
355 XVECTOR (menu_items)->contents[menu_items_used++] = key;
356 XVECTOR (menu_items)->contents[menu_items_used++] = equiv;
a352a815 357 XVECTOR (menu_items)->contents[menu_items_used++] = def;
3427a3db
GM
358 XVECTOR (menu_items)->contents[menu_items_used++] = type;
359 XVECTOR (menu_items)->contents[menu_items_used++] = selected;
850df50b 360 XVECTOR (menu_items)->contents[menu_items_used++] = help;
78589e07
RS
361}
362\f
78589e07 363/* Look through KEYMAPS, a vector of keymaps that is NMAPS long,
101bb4a5
RS
364 and generate menu panes for them in menu_items.
365 If NOTREAL is nonzero,
366 don't bother really computing whether an item is enabled. */
18686d47 367
78589e07 368static void
101bb4a5 369keymap_panes (keymaps, nmaps, notreal)
78589e07
RS
370 Lisp_Object *keymaps;
371 int nmaps;
101bb4a5 372 int notreal;
18686d47 373{
78589e07 374 int mapno;
18686d47 375
78589e07 376 init_menu_items ();
18686d47 377
78589e07
RS
378 /* Loop over the given keymaps, making a pane for each map.
379 But don't make a pane that is empty--ignore that map instead.
380 P is the number of panes we have made so far. */
381 for (mapno = 0; mapno < nmaps; mapno++)
488205c0 382 single_keymap_panes (keymaps[mapno],
5ee707b8 383 Fkeymap_prompt (keymaps[mapno]), Qnil, notreal, 10);
78589e07
RS
384
385 finish_menu_items ();
386}
387
388/* This is a recursive subroutine of keymap_panes.
389 It handles one keymap, KEYMAP.
390 The other arguments are passed along
101bb4a5 391 or point to local variables of the previous function.
de57a39c
RS
392 If NOTREAL is nonzero, only check for equivalent key bindings, don't
393 evaluate expressions in menu items and don't make any menu.
98381e42
RS
394
395 If we encounter submenus deeper than MAXDEPTH levels, ignore them. */
78589e07
RS
396
397static void
98381e42 398single_keymap_panes (keymap, pane_name, prefix, notreal, maxdepth)
78589e07
RS
399 Lisp_Object keymap;
400 Lisp_Object pane_name;
401 Lisp_Object prefix;
101bb4a5 402 int notreal;
98381e42 403 int maxdepth;
78589e07 404{
de57a39c
RS
405 Lisp_Object pending_maps = Qnil;
406 Lisp_Object tail, item;
407 struct gcpro gcpro1, gcpro2;
408 int notbuttons = 0;
78589e07 409
98381e42
RS
410 if (maxdepth <= 0)
411 return;
412
78589e07
RS
413 push_menu_pane (pane_name, prefix);
414
de57a39c
RS
415#ifndef HAVE_BOXES
416 /* Remember index for first item in this pane so we can go back and
417 add a prefix when (if) we see the first button. After that, notbuttons
418 is set to 0, to mark that we have seen a button and all non button
419 items need a prefix. */
420 notbuttons = menu_items_used;
421#endif
422
8e713be6 423 for (tail = keymap; CONSP (tail); tail = XCDR (tail))
18686d47 424 {
de57a39c
RS
425 GCPRO2 (keymap, pending_maps);
426 /* Look at each key binding, and if it is a menu item add it
427 to this menu. */
8e713be6 428 item = XCAR (tail);
b5bb2705 429 if (CONSP (item))
8e713be6 430 single_menu_item (XCAR (item), XCDR (item),
de57a39c 431 &pending_maps, notreal, maxdepth, &notbuttons);
b5bb2705 432 else if (VECTORP (item))
78589e07
RS
433 {
434 /* Loop over the char values represented in the vector. */
435 int len = XVECTOR (item)->size;
436 int c;
437 for (c = 0; c < len; c++)
438 {
439 Lisp_Object character;
33b43fa6 440 XSETFASTINT (character, c);
de57a39c
RS
441 single_menu_item (character, XVECTOR (item)->contents[c],
442 &pending_maps, notreal, maxdepth, &notbuttons);
78589e07 443 }
18686d47 444 }
de57a39c 445 UNGCPRO;
18686d47 446 }
78589e07
RS
447
448 /* Process now any submenus which want to be panes at this level. */
449 while (!NILP (pending_maps))
450 {
101bb4a5 451 Lisp_Object elt, eltcdr, string;
78589e07 452 elt = Fcar (pending_maps);
8e713be6
KR
453 eltcdr = XCDR (elt);
454 string = XCAR (eltcdr);
101bb4a5
RS
455 /* We no longer discard the @ from the beginning of the string here.
456 Instead, we do this in xmenu_show. */
457 single_keymap_panes (Fcar (elt), string,
8e713be6 458 XCDR (eltcdr), notreal, maxdepth - 1);
78589e07
RS
459 pending_maps = Fcdr (pending_maps);
460 }
18686d47 461}
78589e07 462\f
de57a39c
RS
463/* This is a subroutine of single_keymap_panes that handles one
464 keymap entry.
465 KEY is a key in a keymap and ITEM is its binding.
466 PENDING_MAPS_PTR points to a list of keymaps waiting to be made into
467 separate panes.
468 If NOTREAL is nonzero, only check for equivalent key bindings, don't
469 evaluate expressions in menu items and don't make any menu.
470 If we encounter submenus deeper than MAXDEPTH levels, ignore them.
471 NOTBUTTONS_PTR is only used when simulating toggle boxes and radio
472 buttons. It points to variable notbuttons in single_keymap_panes,
473 which keeps track of if we have seen a button in this menu or not. */
474
475static void
476single_menu_item (key, item, pending_maps_ptr, notreal, maxdepth,
477 notbuttons_ptr)
478 Lisp_Object key, item;
479 Lisp_Object *pending_maps_ptr;
480 int maxdepth, notreal;
481 int *notbuttons_ptr;
482{
faa935b6 483 Lisp_Object map, item_string, enabled;
de57a39c
RS
484 struct gcpro gcpro1, gcpro2;
485 int res;
486
487 /* Parse the menu item and leave the result in item_properties. */
488 GCPRO2 (key, item);
489 res = parse_menu_item (item, notreal, 0);
490 UNGCPRO;
491 if (!res)
492 return; /* Not a menu item. */
493
494 map = XVECTOR (item_properties)->contents[ITEM_PROPERTY_MAP];
495
496 if (notreal)
497 {
498 /* We don't want to make a menu, just traverse the keymaps to
499 precompute equivalent key bindings. */
500 if (!NILP (map))
501 single_keymap_panes (map, Qnil, key, 1, maxdepth - 1);
502 return;
503 }
504
505 enabled = XVECTOR (item_properties)->contents[ITEM_PROPERTY_ENABLE];
506 item_string = XVECTOR (item_properties)->contents[ITEM_PROPERTY_NAME];
507
508 if (!NILP (map) && XSTRING (item_string)->data[0] == '@')
509 {
510 if (!NILP (enabled))
511 /* An enabled separate pane. Remember this to handle it later. */
512 *pending_maps_ptr = Fcons (Fcons (map, Fcons (item_string, key)),
513 *pending_maps_ptr);
514 return;
515 }
516
517#ifndef HAVE_BOXES
518 /* Simulate radio buttons and toggle boxes by putting a prefix in
519 front of them. */
520 {
521 Lisp_Object prefix = Qnil;
522 Lisp_Object type = XVECTOR (item_properties)->contents[ITEM_PROPERTY_TYPE];
523 if (!NILP (type))
524 {
525 Lisp_Object selected
526 = XVECTOR (item_properties)->contents[ITEM_PROPERTY_SELECTED];
527
528 if (*notbuttons_ptr)
529 /* The first button. Line up previous items in this menu. */
530 {
531 int index = *notbuttons_ptr; /* Index for first item this menu. */
532 int submenu = 0;
533 Lisp_Object tem;
534 while (index < menu_items_used)
535 {
536 tem
537 = XVECTOR (menu_items)->contents[index + MENU_ITEMS_ITEM_NAME];
538 if (NILP (tem))
539 {
540 index++;
541 submenu++; /* Skip sub menu. */
542 }
543 else if (EQ (tem, Qlambda))
544 {
545 index++;
546 submenu--; /* End sub menu. */
547 }
548 else if (EQ (tem, Qt))
549 index += 3; /* Skip new pane marker. */
550 else if (EQ (tem, Qquote))
551 index++; /* Skip a left, right divider. */
552 else
553 {
554 if (!submenu && XSTRING (tem)->data[0] != '\0'
555 && XSTRING (tem)->data[0] != '-')
556 XVECTOR (menu_items)->contents[index + MENU_ITEMS_ITEM_NAME]
557 = concat2 (build_string (" "), tem);
558 index += MENU_ITEMS_ITEM_LENGTH;
559 }
560 }
561 *notbuttons_ptr = 0;
562 }
563
564 /* Calculate prefix, if any, for this item. */
565 if (EQ (type, QCtoggle))
566 prefix = build_string (NILP (selected) ? "[ ] " : "[X] ");
567 else if (EQ (type, QCradio))
568 prefix = build_string (NILP (selected) ? "( ) " : "(*) ");
569 }
570 /* Not a button. If we have earlier buttons, then we need a prefix. */
571 else if (!*notbuttons_ptr && XSTRING (item_string)->data[0] != '\0'
572 && XSTRING (item_string)->data[0] != '-')
573 prefix = build_string (" ");
574
575 if (!NILP (prefix))
576 item_string = concat2 (prefix, item_string);
577 }
578#endif /* not HAVE_BOXES */
579
580#ifndef USE_X_TOOLKIT
581 if (!NILP(map))
582 /* Indicate visually that this is a submenu. */
583 item_string = concat2 (item_string, build_string (" >"));
584#endif
585
586 push_menu_item (item_string, enabled, key,
587 XVECTOR (item_properties)->contents[ITEM_PROPERTY_DEF],
3427a3db
GM
588 XVECTOR (item_properties)->contents[ITEM_PROPERTY_KEYEQ],
589 XVECTOR (item_properties)->contents[ITEM_PROPERTY_TYPE],
850df50b
GM
590 XVECTOR (item_properties)->contents[ITEM_PROPERTY_SELECTED],
591 XVECTOR (item_properties)->contents[ITEM_PROPERTY_HELP]);
de57a39c
RS
592
593#ifdef USE_X_TOOLKIT
594 /* Display a submenu using the toolkit. */
595 if (! (NILP (map) || NILP (enabled)))
596 {
597 push_submenu_start ();
598 single_keymap_panes (map, Qnil, key, 0, maxdepth - 1);
599 push_submenu_end ();
600 }
601#endif
602}
603\f
8e6208c5 604/* Push all the panes and items of a menu described by the
78589e07
RS
605 alist-of-alists MENU.
606 This handles old-fashioned calls to x-popup-menu. */
18686d47 607
78589e07
RS
608static void
609list_of_panes (menu)
18686d47 610 Lisp_Object menu;
18686d47 611{
78589e07
RS
612 Lisp_Object tail;
613
614 init_menu_items ();
615
616 for (tail = menu; !NILP (tail); tail = Fcdr (tail))
617 {
618 Lisp_Object elt, pane_name, pane_data;
619 elt = Fcar (tail);
620 pane_name = Fcar (elt);
b7826503 621 CHECK_STRING (pane_name);
78589e07
RS
622 push_menu_pane (pane_name, Qnil);
623 pane_data = Fcdr (elt);
b7826503 624 CHECK_CONS (pane_data);
78589e07
RS
625 list_of_items (pane_data);
626 }
627
628 finish_menu_items ();
629}
630
631/* Push the items in a single pane defined by the alist PANE. */
632
633static void
634list_of_items (pane)
635 Lisp_Object pane;
636{
637 Lisp_Object tail, item, item1;
638
639 for (tail = pane; !NILP (tail); tail = Fcdr (tail))
640 {
641 item = Fcar (tail);
642 if (STRINGP (item))
850df50b 643 push_menu_item (item, Qnil, Qnil, Qt, Qnil, Qnil, Qnil, Qnil);
fcaa7665
RS
644 else if (NILP (item))
645 push_left_right_boundary ();
78589e07
RS
646 else
647 {
b7826503 648 CHECK_CONS (item);
78589e07 649 item1 = Fcar (item);
b7826503 650 CHECK_STRING (item1);
850df50b 651 push_menu_item (item1, Qt, Fcdr (item), Qt, Qnil, Qnil, Qnil, Qnil);
78589e07
RS
652 }
653 }
654}
655\f
540e52d1 656DEFUN ("x-popup-menu", Fx_popup_menu, Sx_popup_menu, 2, 2, 0,
f6e34773 657 doc: /* Pop up a deck-of-cards menu and return user's selection.
228299fa
GM
658POSITION is a position specification. This is either a mouse button event
659or a list ((XOFFSET YOFFSET) WINDOW)
660where XOFFSET and YOFFSET are positions in pixels from the top left
661corner of WINDOW's frame. (WINDOW may be a frame object instead of a window.)
662This controls the position of the center of the first line
663in the first pane of the menu, not the top left of the menu as a whole.
664If POSITION is t, it means to use the current mouse position.
665
666MENU is a specifier for a menu. For the simplest case, MENU is a keymap.
667The menu items come from key bindings that have a menu string as well as
668a definition; actually, the "definition" in such a key binding looks like
669\(STRING . REAL-DEFINITION). To give the menu a title, put a string into
670the keymap as a top-level element.
671
672If REAL-DEFINITION is nil, that puts a nonselectable string in the menu.
673Otherwise, REAL-DEFINITION should be a valid key binding definition.
674
675You can also use a list of keymaps as MENU.
676 Then each keymap makes a separate pane.
677When MENU is a keymap or a list of keymaps, the return value
678is a list of events.
679
680Alternatively, you can specify a menu of multiple panes
681 with a list of the form (TITLE PANE1 PANE2...),
682where each pane is a list of form (TITLE ITEM1 ITEM2...).
683Each ITEM is normally a cons cell (STRING . VALUE);
684but a string can appear as an item--that makes a nonselectable line
685in the menu.
686With this form of menu, the return value is VALUE from the chosen item.
687
688If POSITION is nil, don't display the menu at all, just precalculate the
7ee72033
MB
689cached information about equivalent key sequences. */)
690 (position, menu)
78589e07
RS
691 Lisp_Object position, menu;
692{
18686d47 693 Lisp_Object keymap, tem;
6bbd7a29 694 int xpos = 0, ypos = 0;
78589e07
RS
695 Lisp_Object title;
696 char *error_name;
697 Lisp_Object selection;
cc45fb40 698 FRAME_PTR f = NULL;
78589e07
RS
699 Lisp_Object x, y, window;
700 int keymaps = 0;
9685a93f 701 int for_click = 0;
78589e07
RS
702 struct gcpro gcpro1;
703
1e659e4c 704#ifdef HAVE_MENUS
78589e07
RS
705 if (! NILP (position))
706 {
101bb4a5
RS
707 check_x ();
708
78589e07 709 /* Decode the first argument: find the window and the coordinates. */
9572375b 710 if (EQ (position, Qt)
2b06561a
DL
711 || (CONSP (position) && (EQ (XCAR (position), Qmenu_bar)
712 || EQ (XCAR (position), Qtool_bar))))
78589e07
RS
713 {
714 /* Use the mouse's current position. */
b404828f 715 FRAME_PTR new_f = SELECTED_FRAME ();
78589e07 716 Lisp_Object bar_window;
dfcf069d 717 enum scroll_bar_part part;
78589e07
RS
718 unsigned long time;
719
b137c582 720 if (mouse_position_hook)
46b657e2
RS
721 (*mouse_position_hook) (&new_f, 1, &bar_window,
722 &part, &x, &y, &time);
5ca2ef64 723 if (new_f != 0)
a39f0477 724 XSETFRAME (window, new_f);
5ca2ef64
RS
725 else
726 {
727 window = selected_window;
33b43fa6
KH
728 XSETFASTINT (x, 0);
729 XSETFASTINT (y, 0);
5ca2ef64 730 }
78589e07
RS
731 }
732 else
733 {
734 tem = Fcar (position);
b5bb2705 735 if (CONSP (tem))
78589e07
RS
736 {
737 window = Fcar (Fcdr (position));
738 x = Fcar (tem);
739 y = Fcar (Fcdr (tem));
740 }
741 else
742 {
9685a93f 743 for_click = 1;
78589e07
RS
744 tem = Fcar (Fcdr (position)); /* EVENT_START (position) */
745 window = Fcar (tem); /* POSN_WINDOW (tem) */
746 tem = Fcar (Fcdr (Fcdr (tem))); /* POSN_WINDOW_POSN (tem) */
747 x = Fcar (tem);
748 y = Fcdr (tem);
78589e07
RS
749 }
750 }
751
b7826503
PJ
752 CHECK_NUMBER (x);
753 CHECK_NUMBER (y);
78589e07
RS
754
755 /* Decode where to put the menu. */
756
b5bb2705 757 if (FRAMEP (window))
78589e07
RS
758 {
759 f = XFRAME (window);
78589e07
RS
760 xpos = 0;
761 ypos = 0;
762 }
b5bb2705 763 else if (WINDOWP (window))
78589e07 764 {
b7826503 765 CHECK_LIVE_WINDOW (window);
78589e07
RS
766 f = XFRAME (WINDOW_FRAME (XWINDOW (window)));
767
cc45fb40 768 xpos = (FONT_WIDTH (FRAME_FONT (f))
99d3fac7 769 * XFASTINT (XWINDOW (window)->left));
cc45fb40 770 ypos = (FRAME_LINE_HEIGHT (f)
99d3fac7 771 * XFASTINT (XWINDOW (window)->top));
78589e07
RS
772 }
773 else
774 /* ??? Not really clean; should be CHECK_WINDOW_OR_FRAME,
775 but I don't want to make one now. */
b7826503 776 CHECK_WINDOW (window);
78589e07
RS
777
778 xpos += XINT (x);
779 ypos += XINT (y);
bfc524bc
RS
780
781 XSETFRAME (Vmenu_updating_frame, f);
78589e07 782 }
bfc524bc 783 Vmenu_updating_frame = Qnil;
1e659e4c 784#endif /* HAVE_MENUS */
78589e07
RS
785
786 title = Qnil;
787 GCPRO1 (title);
788
789 /* Decode the menu items from what was specified. */
18686d47 790
02067692
SM
791 keymap = get_keymap (menu, 0, 0);
792 if (CONSP (keymap))
18686d47
RS
793 {
794 /* We were given a keymap. Extract menu info from the keymap. */
795 Lisp_Object prompt;
18686d47 796
78589e07 797 /* Extract the detailed info to make one pane. */
101bb4a5 798 keymap_panes (&menu, 1, NILP (position));
78589e07 799
18686d47
RS
800 /* Search for a string appearing directly as an element of the keymap.
801 That string is the title of the menu. */
5ee707b8 802 prompt = Fkeymap_prompt (keymap);
4f9ad016
RS
803 if (NILP (title) && !NILP (prompt))
804 title = prompt;
18686d47 805
78589e07
RS
806 /* Make that be the pane title of the first pane. */
807 if (!NILP (prompt) && menu_items_n_panes >= 0)
808 XVECTOR (menu_items)->contents[MENU_ITEMS_PANE_NAME] = prompt;
809
810 keymaps = 1;
18686d47 811 }
02067692 812 else if (CONSP (menu) && KEYMAPP (XCAR (menu)))
18686d47
RS
813 {
814 /* We were given a list of keymaps. */
18686d47
RS
815 int nmaps = XFASTINT (Flength (menu));
816 Lisp_Object *maps
817 = (Lisp_Object *) alloca (nmaps * sizeof (Lisp_Object));
818 int i;
78589e07
RS
819
820 title = Qnil;
18686d47
RS
821
822 /* The first keymap that has a prompt string
823 supplies the menu title. */
b5bb2705 824 for (tem = menu, i = 0; CONSP (tem); tem = Fcdr (tem))
18686d47 825 {
78589e07
RS
826 Lisp_Object prompt;
827
02067692 828 maps[i++] = keymap = get_keymap (Fcar (tem), 1, 0);
18686d47 829
5ee707b8 830 prompt = Fkeymap_prompt (keymap);
78589e07
RS
831 if (NILP (title) && !NILP (prompt))
832 title = prompt;
18686d47
RS
833 }
834
835 /* Extract the detailed info to make one pane. */
101bb4a5 836 keymap_panes (maps, nmaps, NILP (position));
78589e07
RS
837
838 /* Make the title be the pane title of the first pane. */
839 if (!NILP (title) && menu_items_n_panes >= 0)
840 XVECTOR (menu_items)->contents[MENU_ITEMS_PANE_NAME] = title;
841
842 keymaps = 1;
18686d47
RS
843 }
844 else
845 {
846 /* We were given an old-fashioned menu. */
78589e07 847 title = Fcar (menu);
b7826503 848 CHECK_STRING (title);
18686d47 849
78589e07 850 list_of_panes (Fcdr (menu));
18686d47 851
78589e07
RS
852 keymaps = 0;
853 }
18686d47 854
78589e07 855 if (NILP (position))
18686d47 856 {
78589e07
RS
857 discard_menu_items ();
858 UNGCPRO;
859 return Qnil;
18686d47
RS
860 }
861
1e659e4c 862#ifdef HAVE_MENUS
78589e07
RS
863 /* Display them in a menu. */
864 BLOCK_INPUT;
18686d47 865
673a6211 866 selection = xmenu_show (f, xpos, ypos, for_click,
78589e07
RS
867 keymaps, title, &error_name);
868 UNBLOCK_INPUT;
18686d47 869
78589e07 870 discard_menu_items ();
18686d47 871
78589e07 872 UNGCPRO;
1e659e4c 873#endif /* HAVE_MENUS */
18686d47 874
78589e07
RS
875 if (error_name) error (error_name);
876 return selection;
18686d47 877}
165e1749 878
1e659e4c
RS
879#ifdef HAVE_MENUS
880
540e52d1 881DEFUN ("x-popup-dialog", Fx_popup_dialog, Sx_popup_dialog, 2, 2, 0,
f6e34773 882 doc: /* Pop up a dialog box and return user's selection.
228299fa
GM
883POSITION specifies which frame to use.
884This is normally a mouse button event or a window or frame.
885If POSITION is t, it means to use the frame the mouse is on.
886The dialog box appears in the middle of the specified frame.
887
888CONTENTS specifies the alternatives to display in the dialog box.
889It is a list of the form (TITLE ITEM1 ITEM2...).
890Each ITEM is a cons cell (STRING . VALUE).
891The return value is VALUE from the chosen item.
892
893An ITEM may also be just a string--that makes a nonselectable item.
894An ITEM may also be nil--that means to put all preceding items
895on the left of the dialog box and all following items on the right.
7ee72033
MB
896\(By default, approximately half appear on each side.) */)
897 (position, contents)
99fe880d 898 Lisp_Object position, contents;
165e1749 899{
cc45fb40 900 FRAME_PTR f = NULL;
99fe880d 901 Lisp_Object window;
165e1749
FP
902
903 check_x ();
904
99fe880d 905 /* Decode the first argument: find the window or frame to use. */
88d4f6ec 906 if (EQ (position, Qt)
2b06561a
DL
907 || (CONSP (position) && (EQ (XCAR (position), Qmenu_bar)
908 || EQ (XCAR (position), Qtool_bar))))
165e1749 909 {
b14db4d7 910#if 0 /* Using the frame the mouse is on may not be right. */
99fe880d 911 /* Use the mouse's current position. */
b404828f 912 FRAME_PTR new_f = SELECTED_FRAME ();
99fe880d 913 Lisp_Object bar_window;
cc45fb40 914 enum scroll_bar_part part;
99fe880d
RS
915 unsigned long time;
916 Lisp_Object x, y;
165e1749 917
46b657e2 918 (*mouse_position_hook) (&new_f, 1, &bar_window, &part, &x, &y, &time);
5ca2ef64 919
99fe880d 920 if (new_f != 0)
a39f0477 921 XSETFRAME (window, new_f);
99fe880d
RS
922 else
923 window = selected_window;
b14db4d7 924#endif
88d4f6ec 925 window = selected_window;
99fe880d
RS
926 }
927 else if (CONSP (position))
928 {
929 Lisp_Object tem;
930 tem = Fcar (position);
b5bb2705 931 if (CONSP (tem))
99fe880d 932 window = Fcar (Fcdr (position));
80670155
RS
933 else
934 {
99fe880d
RS
935 tem = Fcar (Fcdr (position)); /* EVENT_START (position) */
936 window = Fcar (tem); /* POSN_WINDOW (tem) */
165e1749 937 }
165e1749 938 }
99fe880d
RS
939 else if (WINDOWP (position) || FRAMEP (position))
940 window = position;
a9be6839
RS
941 else
942 window = Qnil;
165e1749 943
99fe880d 944 /* Decode where to put the menu. */
165e1749 945
b5bb2705 946 if (FRAMEP (window))
99fe880d 947 f = XFRAME (window);
b5bb2705 948 else if (WINDOWP (window))
165e1749 949 {
b7826503 950 CHECK_LIVE_WINDOW (window);
99fe880d 951 f = XFRAME (WINDOW_FRAME (XWINDOW (window)));
165e1749 952 }
99fe880d
RS
953 else
954 /* ??? Not really clean; should be CHECK_WINDOW_OR_FRAME,
955 but I don't want to make one now. */
b7826503 956 CHECK_WINDOW (window);
165e1749 957
99fe880d
RS
958#ifndef USE_X_TOOLKIT
959 /* Display a menu with these alternatives
960 in the middle of frame F. */
961 {
962 Lisp_Object x, y, frame, newpos;
a39f0477
KH
963 XSETFRAME (frame, f);
964 XSETINT (x, x_pixel_width (f) / 2);
965 XSETINT (y, x_pixel_height (f) / 2);
99fe880d
RS
966 newpos = Fcons (Fcons (x, Fcons (y, Qnil)), Fcons (frame, Qnil));
967
968 return Fx_popup_menu (newpos,
969 Fcons (Fcar (contents), Fcons (contents, Qnil)));
970 }
971#else
972 {
973 Lisp_Object title;
974 char *error_name;
975 Lisp_Object selection;
165e1749 976
99fe880d
RS
977 /* Decode the dialog items from what was specified. */
978 title = Fcar (contents);
b7826503 979 CHECK_STRING (title);
165e1749 980
99fe880d 981 list_of_panes (Fcons (contents, Qnil));
165e1749 982
99fe880d
RS
983 /* Display them in a dialog box. */
984 BLOCK_INPUT;
f8c2630d 985 selection = xdialog_show (f, 0, title, &error_name);
99fe880d 986 UNBLOCK_INPUT;
165e1749 987
99fe880d
RS
988 discard_menu_items ();
989
990 if (error_name) error (error_name);
991 return selection;
992 }
7464b131 993#endif
392d3f4b 994}
78589e07
RS
995\f
996#ifdef USE_X_TOOLKIT
18686d47 997
4dedbfe0 998/* Loop in Xt until the menu pulldown or dialog popup has been
f56e7ad2
RS
999 popped down (deactivated). This is used for x-popup-menu
1000 and x-popup-dialog; it is not used for the menu bar any more.
c98fcf4b 1001
2e2b8e22 1002 NOTE: All calls to popup_get_selection should be protected
c98fcf4b 1003 with BLOCK_INPUT, UNBLOCK_INPUT wrappers. */
aa669def 1004
4dedbfe0 1005void
2e2b8e22 1006popup_get_selection (initial_event, dpyinfo, id)
4dedbfe0 1007 XEvent *initial_event;
aa669def 1008 struct x_display_info *dpyinfo;
2e2b8e22 1009 LWLIB_ID id;
78589e07 1010{
4dedbfe0 1011 XEvent event;
78589e07 1012
aa669def
RS
1013 /* Define a queue to save up for later unreading
1014 all X events that don't pertain to the menu. */
1015 struct event_queue
1016 {
1017 XEvent event;
1018 struct event_queue *next;
1019 };
1020
1021 struct event_queue *queue = NULL;
1022 struct event_queue *queue_tmp;
1023
4dedbfe0
PR
1024 if (initial_event)
1025 event = *initial_event;
1026 else
1027 XtAppNextEvent (Xt_app_con, &event);
78589e07 1028
4dedbfe0 1029 while (1)
78589e07 1030 {
aa669def
RS
1031 /* Handle expose events for editor frames right away. */
1032 if (event.type == Expose)
1033 process_expose_from_menu (event);
10624005
KH
1034 /* Make sure we don't consider buttons grabbed after menu goes.
1035 And make sure to deactivate for any ButtonRelease,
1036 even if XtDispatchEvent doesn't do that. */
aa669def
RS
1037 else if (event.type == ButtonRelease
1038 && dpyinfo->display == event.xbutton.display)
10624005
KH
1039 {
1040 dpyinfo->grabbed &= ~(1 << event.xbutton.button);
1041 popup_activated_flag = 0;
c8b5aa3d
RS
1042#ifdef USE_MOTIF /* Pretending that the event came from a
1043 Btn1Down seems the only way to convince Motif to
1044 activate its callbacks; setting the XmNmenuPost
1045 isn't working. --marcus@sysc.pdx.edu. */
1046 event.xbutton.button = 1;
1047#endif
10624005 1048 }
2e2b8e22
KH
1049 /* If the user presses a key, deactivate the menu.
1050 The user is likely to do that if we get wedged. */
1051 else if (event.type == KeyPress
1052 && dpyinfo->display == event.xbutton.display)
1053 {
09155689
AS
1054 KeySym keysym = XLookupKeysym (&event.xkey, 0);
1055 if (!IsModifierKey (keysym))
1056 {
1057 popup_activated_flag = 0;
1058 break;
1059 }
2e2b8e22 1060 }
d31d42cc
RS
1061 /* Button presses outside the menu also pop it down. */
1062 else if (event.type == ButtonPress
1063 && event.xany.display == dpyinfo->display
1064 && x_any_window_to_frame (dpyinfo, event.xany.window))
1065 {
1066 popup_activated_flag = 0;
1067 break;
1068 }
aa669def
RS
1069
1070 /* Queue all events not for this popup,
10624005 1071 except for Expose, which we've already handled, and ButtonRelease.
9572375b
KH
1072 Note that the X window is associated with the frame if this
1073 is a menu bar popup, but not if it's a dialog box. So we use
1074 x_non_menubar_window_to_frame, not x_any_window_to_frame. */
aa669def 1075 if (event.type != Expose
10624005
KH
1076 && !(event.type == ButtonRelease
1077 && dpyinfo->display == event.xbutton.display)
aa669def 1078 && (event.xany.display != dpyinfo->display
9572375b 1079 || x_non_menubar_window_to_frame (dpyinfo, event.xany.window)))
aa669def 1080 {
3e40ba92
GM
1081 queue_tmp = (struct event_queue *) xmalloc (sizeof *queue_tmp);
1082 queue_tmp->event = event;
1083 queue_tmp->next = queue;
1084 queue = queue_tmp;
aa669def 1085 }
9572375b
KH
1086 else
1087 XtDispatchEvent (&event);
aa669def
RS
1088
1089 if (!popup_activated ())
4dedbfe0
PR
1090 break;
1091 XtAppNextEvent (Xt_app_con, &event);
78589e07 1092 }
aa669def
RS
1093
1094 /* Unread any events that we got but did not handle. */
1095 while (queue != NULL)
1096 {
1097 queue_tmp = queue;
1098 XPutBackEvent (queue_tmp->event.xany.display, &queue_tmp->event);
1099 queue = queue_tmp->next;
36cc2d38 1100 xfree ((char *)queue_tmp);
aa669def
RS
1101 /* Cause these events to get read as soon as we UNBLOCK_INPUT. */
1102 interrupt_input_pending = 1;
1103 }
78589e07
RS
1104}
1105
88766961
RS
1106/* Activate the menu bar of frame F.
1107 This is called from keyboard.c when it gets the
1108 menu_bar_activate_event out of the Emacs event queue.
1109
1110 To activate the menu bar, we use the X button-press event
ac78b144 1111 that was saved in saved_menu_event.
88766961
RS
1112 That makes the toolkit do its thing.
1113
1114 But first we recompute the menu bar contents (the whole tree).
1115
1116 The reason for saving the button event until here, instead of
1117 passing it to the toolkit right away, is that we can safely
1118 execute Lisp code. */
1119
dfcf069d 1120void
88766961
RS
1121x_activate_menubar (f)
1122 FRAME_PTR f;
1123{
ac78b144 1124 if (!f->output_data.x->saved_menu_event->type)
88766961
RS
1125 return;
1126
a9be6839 1127 set_frame_menubar (f, 0, 1);
88766961 1128 BLOCK_INPUT;
ac78b144 1129 XtDispatchEvent ((XEvent *) f->output_data.x->saved_menu_event);
88766961 1130 UNBLOCK_INPUT;
a9be6839 1131#ifdef USE_MOTIF
745c34fb
RS
1132 if (f->output_data.x->saved_menu_event->type == ButtonRelease)
1133 pending_menu_activation = 1;
a9be6839 1134#endif
745c34fb 1135
88766961 1136 /* Ignore this if we get it a second time. */
ac78b144 1137 f->output_data.x->saved_menu_event->type = 0;
88766961
RS
1138}
1139
c98fcf4b 1140/* Detect if a dialog or menu has been posted. */
aa669def 1141
4dedbfe0
PR
1142int
1143popup_activated ()
1144{
1145 return popup_activated_flag;
1146}
1147
4dedbfe0
PR
1148/* This callback is invoked when the user selects a menubar cascade
1149 pushbutton, but before the pulldown menu is posted. */
78589e07
RS
1150
1151static void
4dedbfe0 1152popup_activate_callback (widget, id, client_data)
78589e07
RS
1153 Widget widget;
1154 LWLIB_ID id;
1155 XtPointer client_data;
1156{
4dedbfe0 1157 popup_activated_flag = 1;
7555d825
GM
1158}
1159
1160/* This callback is invoked when a dialog or menu is finished being
1161 used and has been unposted. */
1162
1163static void
1164popup_deactivate_callback (widget, id, client_data)
1165 Widget widget;
1166 LWLIB_ID id;
1167 XtPointer client_data;
1168{
7555d825 1169 popup_activated_flag = 0;
78589e07
RS
1170}
1171
850df50b
GM
1172/* Lwlib callback called when menu items are highlighted/unhighlighted
1173 while moving the mouse over them. WIDGET is the menu bar or menu
1174 popup widget. ID is its LWLIB_ID. CALL_DATA contains a pointer to
1175 the widget_value structure for the menu item, or null in case of
1176 unhighlighting. */
1177
1178void
1179menu_highlight_callback (widget, id, call_data)
1180 Widget widget;
1181 LWLIB_ID id;
1182 void *call_data;
1183{
1184 widget_value *wv = (widget_value *) call_data;
1185 struct frame *f;
1186 Lisp_Object frame, help;
850df50b 1187
0b1f4572 1188 help = wv ? wv->help : Qnil;
9cd50434 1189
850df50b
GM
1190 /* Determine the frame for the help event. */
1191 f = menubar_id_to_frame (id);
1192 if (f)
9cd50434
GM
1193 {
1194 XSETFRAME (frame, f);
1195 kbd_buffer_store_help_event (frame, help);
1196 }
850df50b
GM
1197 else
1198 {
1199 /* WIDGET is the popup menu. It's parent is the frame's
1200 widget. See which frame that is. */
1201 Widget frame_widget = XtParent (widget);
1202 Lisp_Object tail;
1203
1204 for (tail = Vframe_list; GC_CONSP (tail); tail = XCDR (tail))
1205 {
1206 frame = XCAR (tail);
1207 if (GC_FRAMEP (frame)
1208 && (f = XFRAME (frame),
1209 FRAME_X_P (f) && f->output_data.x->widget == frame_widget))
1210 break;
1211 }
850df50b 1212
9cd50434
GM
1213 show_help_echo (help, Qnil, Qnil, Qnil, 1);
1214 }
850df50b
GM
1215}
1216
4dedbfe0
PR
1217/* This callback is called from the menu bar pulldown menu
1218 when the user makes a selection.
1219 Figure out what the user chose
1220 and put the appropriate events into the keyboard buffer. */
1221
78589e07 1222static void
4dedbfe0 1223menubar_selection_callback (widget, id, client_data)
78589e07
RS
1224 Widget widget;
1225 LWLIB_ID id;
1226 XtPointer client_data;
1227{
c63f6952 1228 Lisp_Object prefix, entry;
88766961 1229 FRAME_PTR f = menubar_id_to_frame (id);
4dedbfe0
PR
1230 Lisp_Object vector;
1231 Lisp_Object *subprefix_stack;
1232 int submenu_depth = 0;
1233 int i;
1234
1235 if (!f)
1236 return;
6bbd7a29 1237 entry = Qnil;
4dedbfe0
PR
1238 subprefix_stack = (Lisp_Object *) alloca (f->menu_bar_items_used * sizeof (Lisp_Object));
1239 vector = f->menu_bar_vector;
1240 prefix = Qnil;
1241 i = 0;
1242 while (i < f->menu_bar_items_used)
1243 {
4dedbfe0
PR
1244 if (EQ (XVECTOR (vector)->contents[i], Qnil))
1245 {
1246 subprefix_stack[submenu_depth++] = prefix;
1247 prefix = entry;
1248 i++;
1249 }
1250 else if (EQ (XVECTOR (vector)->contents[i], Qlambda))
1251 {
1252 prefix = subprefix_stack[--submenu_depth];
1253 i++;
1254 }
1255 else if (EQ (XVECTOR (vector)->contents[i], Qt))
1256 {
4cb35c39 1257 prefix = XVECTOR (vector)->contents[i + MENU_ITEMS_PANE_PREFIX];
4dedbfe0
PR
1258 i += MENU_ITEMS_PANE_LENGTH;
1259 }
1260 else
1261 {
4cb35c39 1262 entry = XVECTOR (vector)->contents[i + MENU_ITEMS_ITEM_VALUE];
01d5e892
RS
1263 /* The EMACS_INT cast avoids a warning. There's no problem
1264 as long as pointers have enough bits to hold small integers. */
1265 if ((int) (EMACS_INT) client_data == i)
4dedbfe0
PR
1266 {
1267 int j;
1268 struct input_event buf;
4cb35c39 1269 Lisp_Object frame;
4dedbfe0 1270
4cb35c39 1271 XSETFRAME (frame, f);
9979315d
GM
1272 buf.kind = MENU_BAR_EVENT;
1273 buf.frame_or_window = frame;
1274 buf.arg = frame;
4dedbfe0
PR
1275 kbd_buffer_store_event (&buf);
1276
1277 for (j = 0; j < submenu_depth; j++)
1278 if (!NILP (subprefix_stack[j]))
1279 {
9979315d
GM
1280 buf.kind = MENU_BAR_EVENT;
1281 buf.frame_or_window = frame;
1282 buf.arg = subprefix_stack[j];
4dedbfe0
PR
1283 kbd_buffer_store_event (&buf);
1284 }
1285
1286 if (!NILP (prefix))
1287 {
9979315d
GM
1288 buf.kind = MENU_BAR_EVENT;
1289 buf.frame_or_window = frame;
1290 buf.arg = prefix;
4dedbfe0
PR
1291 kbd_buffer_store_event (&buf);
1292 }
1293
9979315d
GM
1294 buf.kind = MENU_BAR_EVENT;
1295 buf.frame_or_window = frame;
1296 buf.arg = entry;
4dedbfe0
PR
1297 kbd_buffer_store_event (&buf);
1298
1299 return;
1300 }
1301 i += MENU_ITEMS_ITEM_LENGTH;
1302 }
1303 }
18686d47
RS
1304}
1305
f9655c60
RS
1306/* Allocate a widget_value, blocking input. */
1307
1308widget_value *
1309xmalloc_widget_value ()
1310{
1311 widget_value *value;
1312
1313 BLOCK_INPUT;
1314 value = malloc_widget_value ();
1315 UNBLOCK_INPUT;
1316
1317 return value;
1318}
4dedbfe0
PR
1319
1320/* This recursively calls free_widget_value on the tree of widgets.
18686d47 1321 It must free all data that was malloc'ed for these widget_values.
78589e07
RS
1322 In Emacs, many slots are pointers into the data of Lisp_Strings, and
1323 must be left alone. */
1324
18686d47
RS
1325void
1326free_menubar_widget_value_tree (wv)
1327 widget_value *wv;
1328{
1329 if (! wv) return;
18686d47
RS
1330
1331 wv->name = wv->value = wv->key = (char *) 0xDEADBEEF;
1332
1333 if (wv->contents && (wv->contents != (widget_value*)1))
1334 {
1335 free_menubar_widget_value_tree (wv->contents);
1336 wv->contents = (widget_value *) 0xDEADBEEF;
1337 }
1338 if (wv->next)
1339 {
1340 free_menubar_widget_value_tree (wv->next);
1341 wv->next = (widget_value *) 0xDEADBEEF;
1342 }
1343 BLOCK_INPUT;
1344 free_widget_value (wv);
1345 UNBLOCK_INPUT;
1346}
4dedbfe0
PR
1347\f
1348/* Return a tree of widget_value structures for a menu bar item
1349 whose event type is ITEM_KEY (with string ITEM_NAME)
1350 and whose contents come from the list of keymaps MAPS. */
1351
1352static widget_value *
1353single_submenu (item_key, item_name, maps)
1354 Lisp_Object item_key, item_name, maps;
1355{
1356 widget_value *wv, *prev_wv, *save_wv, *first_wv;
1357 int i;
1358 int submenu_depth = 0;
1359 Lisp_Object length;
1360 int len;
1361 Lisp_Object *mapvec;
1362 widget_value **submenu_stack;
4dedbfe0 1363 int previous_items = menu_items_used;
71dca3e3 1364 int top_level_items = 0;
4dedbfe0
PR
1365
1366 length = Flength (maps);
1367 len = XINT (length);
1368
1369 /* Convert the list MAPS into a vector MAPVEC. */
1370 mapvec = (Lisp_Object *) alloca (len * sizeof (Lisp_Object));
1371 for (i = 0; i < len; i++)
1372 {
1373 mapvec[i] = Fcar (maps);
1374 maps = Fcdr (maps);
1375 }
1376
1377 menu_items_n_panes = 0;
1378
1379 /* Loop over the given keymaps, making a pane for each map.
1380 But don't make a pane that is empty--ignore that map instead. */
1381 for (i = 0; i < len; i++)
71dca3e3 1382 {
846e8c10 1383 if (SYMBOLP (mapvec[i])
02067692 1384 || (CONSP (mapvec[i]) && !KEYMAPP (mapvec[i])))
71dca3e3 1385 {
846e8c10
RS
1386 /* Here we have a command at top level in the menu bar
1387 as opposed to a submenu. */
71dca3e3
RS
1388 top_level_items = 1;
1389 push_menu_pane (Qnil, Qnil);
850df50b
GM
1390 push_menu_item (item_name, Qt, item_key, mapvec[i],
1391 Qnil, Qnil, Qnil, Qnil);
71dca3e3
RS
1392 }
1393 else
98381e42 1394 single_keymap_panes (mapvec[i], item_name, item_key, 0, 10);
71dca3e3 1395 }
4dedbfe0
PR
1396
1397 /* Create a tree of widget_value objects
1398 representing the panes and their items. */
1399
1400 submenu_stack
1401 = (widget_value **) alloca (menu_items_used * sizeof (widget_value *));
f7fab165 1402 wv = xmalloc_widget_value ();
4dedbfe0
PR
1403 wv->name = "menu";
1404 wv->value = 0;
1405 wv->enabled = 1;
3427a3db 1406 wv->button_type = BUTTON_TYPE_NONE;
0b1e6b54 1407 wv->help = Qnil;
4dedbfe0
PR
1408 first_wv = wv;
1409 save_wv = 0;
71dca3e3 1410 prev_wv = 0;
4dedbfe0
PR
1411
1412 /* Loop over all panes and items made during this call
1413 and construct a tree of widget_value objects.
1414 Ignore the panes and items made by previous calls to
1415 single_submenu, even though those are also in menu_items. */
1416 i = previous_items;
1417 while (i < menu_items_used)
1418 {
1419 if (EQ (XVECTOR (menu_items)->contents[i], Qnil))
1420 {
1421 submenu_stack[submenu_depth++] = save_wv;
1422 save_wv = prev_wv;
1423 prev_wv = 0;
1424 i++;
1425 }
1426 else if (EQ (XVECTOR (menu_items)->contents[i], Qlambda))
1427 {
1428 prev_wv = save_wv;
1429 save_wv = submenu_stack[--submenu_depth];
1430 i++;
1431 }
1432 else if (EQ (XVECTOR (menu_items)->contents[i], Qt)
1433 && submenu_depth != 0)
1434 i += MENU_ITEMS_PANE_LENGTH;
1435 /* Ignore a nil in the item list.
1436 It's meaningful only for dialog boxes. */
1437 else if (EQ (XVECTOR (menu_items)->contents[i], Qquote))
1438 i += 1;
1439 else if (EQ (XVECTOR (menu_items)->contents[i], Qt))
1440 {
1441 /* Create a new pane. */
1442 Lisp_Object pane_name, prefix;
1443 char *pane_string;
4c329aa8 1444
4dedbfe0
PR
1445 pane_name = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_NAME];
1446 prefix = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_PREFIX];
4c329aa8 1447
703dc2a8
KH
1448#ifndef HAVE_MULTILINGUAL_MENU
1449 if (STRINGP (pane_name) && STRING_MULTIBYTE (pane_name))
4c329aa8 1450 {
cc45fb40 1451 pane_name = ENCODE_SYSTEM (pane_name);
4c329aa8
GM
1452 AREF (menu_items, i + MENU_ITEMS_PANE_NAME) = pane_name;
1453 }
703dc2a8 1454#endif
4dedbfe0
PR
1455 pane_string = (NILP (pane_name)
1456 ? "" : (char *) XSTRING (pane_name)->data);
1457 /* If there is just one top-level pane, put all its items directly
1458 under the top-level menu. */
1459 if (menu_items_n_panes == 1)
1460 pane_string = "";
1461
1462 /* If the pane has a meaningful name,
1463 make the pane a top-level menu item
1464 with its items as a submenu beneath it. */
1465 if (strcmp (pane_string, ""))
1466 {
f7fab165 1467 wv = xmalloc_widget_value ();
4dedbfe0
PR
1468 if (save_wv)
1469 save_wv->next = wv;
1470 else
1471 first_wv->contents = wv;
1472 wv->name = pane_string;
ffcb5a51
RS
1473 /* Ignore the @ that means "separate pane".
1474 This is a kludge, but this isn't worth more time. */
1475 if (!NILP (prefix) && wv->name[0] == '@')
4dedbfe0
PR
1476 wv->name++;
1477 wv->value = 0;
1478 wv->enabled = 1;
3427a3db 1479 wv->button_type = BUTTON_TYPE_NONE;
0b1e6b54 1480 wv->help = Qnil;
4dedbfe0
PR
1481 }
1482 save_wv = wv;
1483 prev_wv = 0;
1484 i += MENU_ITEMS_PANE_LENGTH;
1485 }
1486 else
1487 {
1488 /* Create a new item within current pane. */
3427a3db 1489 Lisp_Object item_name, enable, descrip, def, type, selected;
850df50b
GM
1490 Lisp_Object help;
1491
4c329aa8
GM
1492 item_name = AREF (menu_items, i + MENU_ITEMS_ITEM_NAME);
1493 enable = AREF (menu_items, i + MENU_ITEMS_ITEM_ENABLE);
1494 descrip = AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY);
1495 def = AREF (menu_items, i + MENU_ITEMS_ITEM_DEFINITION);
1496 type = AREF (menu_items, i + MENU_ITEMS_ITEM_TYPE);
1497 selected = AREF (menu_items, i + MENU_ITEMS_ITEM_SELECTED);
1498 help = AREF (menu_items, i + MENU_ITEMS_ITEM_HELP);
3427a3db 1499
703dc2a8 1500#ifndef HAVE_MULTILINGUAL_MENU
3427a3db 1501 if (STRING_MULTIBYTE (item_name))
4c329aa8 1502 {
cc45fb40 1503 item_name = ENCODE_SYSTEM (item_name);
4c329aa8
GM
1504 AREF (menu_items, i + MENU_ITEMS_ITEM_NAME) = item_name;
1505 }
1506
3427a3db 1507 if (STRINGP (descrip) && STRING_MULTIBYTE (descrip))
4c329aa8 1508 {
cc45fb40 1509 descrip = ENCODE_SYSTEM (descrip);
4c329aa8
GM
1510 AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY) = descrip;
1511 }
1512#endif /* not HAVE_MULTILINGUAL_MENU */
3427a3db 1513
f7fab165 1514 wv = xmalloc_widget_value ();
4dedbfe0
PR
1515 if (prev_wv)
1516 prev_wv->next = wv;
71dca3e3 1517 else
4dedbfe0 1518 save_wv->contents = wv;
71dca3e3 1519
4dedbfe0
PR
1520 wv->name = (char *) XSTRING (item_name)->data;
1521 if (!NILP (descrip))
1522 wv->key = (char *) XSTRING (descrip)->data;
1523 wv->value = 0;
01d5e892
RS
1524 /* The EMACS_INT cast avoids a warning. There's no problem
1525 as long as pointers have enough bits to hold small integers. */
1526 wv->call_data = (!NILP (def) ? (void *) (EMACS_INT) i : 0);
4dedbfe0 1527 wv->enabled = !NILP (enable);
3427a3db
GM
1528
1529 if (NILP (type))
1530 wv->button_type = BUTTON_TYPE_NONE;
1531 else if (EQ (type, QCradio))
1532 wv->button_type = BUTTON_TYPE_RADIO;
1533 else if (EQ (type, QCtoggle))
1534 wv->button_type = BUTTON_TYPE_TOGGLE;
1535 else
1536 abort ();
1537
1538 wv->selected = !NILP (selected);
0b1f4572
RS
1539 if (! STRINGP (help))
1540 help = Qnil;
1541
1542 wv->help = help;
cc45fb40 1543
4dedbfe0
PR
1544 prev_wv = wv;
1545
1546 i += MENU_ITEMS_ITEM_LENGTH;
1547 }
1548 }
1549
71dca3e3
RS
1550 /* If we have just one "menu item"
1551 that was originally a button, return it by itself. */
1552 if (top_level_items && first_wv->contents && first_wv->contents->next == 0)
1553 {
1554 wv = first_wv->contents;
1555 free_widget_value (first_wv);
1556 return wv;
1557 }
1558
4dedbfe0
PR
1559 return first_wv;
1560}
cffa74ea 1561
4dedbfe0 1562
f61a541b
GM
1563\f
1564/* Recompute all the widgets of frame F, when the menu bar has been
1565 changed. Value is non-zero if widgets were updated. */
1566
1567static int
6af6cbb5 1568update_frame_menubar (f)
18686d47
RS
1569 FRAME_PTR f;
1570{
7556890b 1571 struct x_output *x = f->output_data.x;
cffa74ea 1572 int columns, rows;
18686d47 1573
f61a541b
GM
1574 if (!x->menubar_widget || XtIsManaged (x->menubar_widget))
1575 return 0;
18686d47
RS
1576
1577 BLOCK_INPUT;
f61a541b
GM
1578 /* Save the size of the frame because the pane widget doesn't accept
1579 to resize itself. So force it. */
cffa74ea
FP
1580 columns = f->width;
1581 rows = f->height;
1582
f61a541b
GM
1583 /* Do the voodoo which means "I'm changing lots of things, don't try
1584 to refigure sizes until I'm done." */
4dedbfe0 1585 lw_refigure_widget (x->column_widget, False);
cffa74ea 1586
f61a541b
GM
1587 /* The order in which children are managed is the top to bottom
1588 order in which they are displayed in the paned window. First,
1589 remove the text-area widget. */
18686d47
RS
1590 XtUnmanageChild (x->edit_widget);
1591
f61a541b
GM
1592 /* Remove the menubar that is there now, and put up the menubar that
1593 should be there. */
1594 XtManageChild (x->menubar_widget);
1595 XtMapWidget (x->menubar_widget);
1596 XtVaSetValues (x->menubar_widget, XtNmappedWhenManaged, 1, NULL);
18686d47 1597
c98fcf4b 1598 /* Re-manage the text-area widget, and then thrash the sizes. */
18686d47 1599 XtManageChild (x->edit_widget);
4dedbfe0 1600 lw_refigure_widget (x->column_widget, True);
cffa74ea
FP
1601
1602 /* Force the pane widget to resize itself with the right values. */
1603 EmacsFrameSetCharSize (x->edit_widget, columns, rows);
18686d47 1604 UNBLOCK_INPUT;
f61a541b 1605 return 1;
18686d47
RS
1606}
1607
4bcdbab1
KH
1608/* Set the contents of the menubar widgets of frame F.
1609 The argument FIRST_TIME is currently ignored;
1610 it is set the first time this is called, from initialize_frame_menubar. */
1611
18686d47 1612void
88766961 1613set_frame_menubar (f, first_time, deep_p)
18686d47 1614 FRAME_PTR f;
706aa2f2 1615 int first_time;
88766961 1616 int deep_p;
18686d47 1617{
7556890b 1618 Widget menubar_widget = f->output_data.x->menubar_widget;
faa935b6 1619 Lisp_Object items;
4d19cb8e 1620 widget_value *wv, *first_wv, *prev_wv = 0;
5b3557df 1621 int i;
cc45fb40 1622
88766961 1623 LWLIB_ID id;
18686d47 1624
bfc524bc
RS
1625 XSETFRAME (Vmenu_updating_frame, f);
1626
7556890b
RS
1627 if (f->output_data.x->id == 0)
1628 f->output_data.x->id = next_menubar_widget_id++;
1629 id = f->output_data.x->id;
37a98547 1630
88766961
RS
1631 if (! menubar_widget)
1632 deep_p = 1;
745c34fb 1633 else if (pending_menu_activation && !deep_p)
a9be6839
RS
1634 deep_p = 1;
1635 /* Make the first call for any given frame always go deep. */
1636 else if (!f->output_data.x->saved_menu_event && !deep_p)
745c34fb
RS
1637 {
1638 deep_p = 1;
a9be6839
RS
1639 f->output_data.x->saved_menu_event = (XEvent*)xmalloc (sizeof (XEvent));
1640 f->output_data.x->saved_menu_event->type = 0;
745c34fb 1641 }
18686d47 1642
f7fab165 1643 wv = xmalloc_widget_value ();
18686d47
RS
1644 wv->name = "menubar";
1645 wv->value = 0;
1646 wv->enabled = 1;
3427a3db 1647 wv->button_type = BUTTON_TYPE_NONE;
4d19cb8e 1648 first_wv = wv;
8d8a3494 1649
88766961 1650 if (deep_p)
18686d47 1651 {
88766961
RS
1652 /* Make a widget-value tree representing the entire menu trees. */
1653
1654 struct buffer *prev = current_buffer;
1655 Lisp_Object buffer;
1656 int specpdl_count = specpdl_ptr - specpdl;
1657 int previous_menu_items_used = f->menu_bar_items_used;
1658 Lisp_Object *previous_items
1659 = (Lisp_Object *) alloca (previous_menu_items_used
1660 * sizeof (Lisp_Object));
1661
0b1cf399
RS
1662 /* If we are making a new widget, its contents are empty,
1663 do always reinitialize them. */
1664 if (! menubar_widget)
1665 previous_menu_items_used = 0;
1666
88766961
RS
1667 buffer = XWINDOW (FRAME_SELECTED_WINDOW (f))->buffer;
1668 specbind (Qinhibit_quit, Qt);
1669 /* Don't let the debugger step into this code
1670 because it is not reentrant. */
1671 specbind (Qdebug_on_next_call, Qnil);
1672
58226364 1673 record_unwind_protect (Fset_match_data, Fmatch_data (Qnil, Qnil));
88766961
RS
1674 if (NILP (Voverriding_local_map_menu_flag))
1675 {
1676 specbind (Qoverriding_terminal_local_map, Qnil);
1677 specbind (Qoverriding_local_map, Qnil);
1678 }
18686d47 1679
88766961 1680 set_buffer_internal_1 (XBUFFER (buffer));
18686d47 1681
88766961 1682 /* Run the Lucid hook. */
950eaee7
GM
1683 safe_run_hooks (Qactivate_menubar_hook);
1684
88766961
RS
1685 /* If it has changed current-menubar from previous value,
1686 really recompute the menubar from the value. */
1687 if (! NILP (Vlucid_menu_bar_dirty_flag))
1688 call0 (Qrecompute_lucid_menubar);
a57634d4 1689 safe_run_hooks (Qmenu_bar_update_hook);
88766961 1690 FRAME_MENU_BAR_ITEMS (f) = menu_bar_items (FRAME_MENU_BAR_ITEMS (f));
18686d47 1691
88766961 1692 items = FRAME_MENU_BAR_ITEMS (f);
8d8a3494 1693
88766961 1694 inhibit_garbage_collection ();
8d8a3494 1695
88766961 1696 /* Save the frame's previous menu bar contents data. */
86c04183
GM
1697 if (previous_menu_items_used)
1698 bcopy (XVECTOR (f->menu_bar_vector)->contents, previous_items,
1699 previous_menu_items_used * sizeof (Lisp_Object));
8d8a3494 1700
88766961
RS
1701 /* Fill in the current menu bar contents. */
1702 menu_items = f->menu_bar_vector;
86c04183 1703 menu_items_allocated = VECTORP (menu_items) ? ASIZE (menu_items) : 0;
88766961 1704 init_menu_items ();
f04366cb 1705 for (i = 0; i < XVECTOR (items)->size; i += 4)
88766961
RS
1706 {
1707 Lisp_Object key, string, maps;
1708
1709 key = XVECTOR (items)->contents[i];
1710 string = XVECTOR (items)->contents[i + 1];
1711 maps = XVECTOR (items)->contents[i + 2];
1712 if (NILP (string))
1713 break;
1714
1715 wv = single_submenu (key, string, maps);
1716 if (prev_wv)
1717 prev_wv->next = wv;
1718 else
1719 first_wv->contents = wv;
1720 /* Don't set wv->name here; GC during the loop might relocate it. */
1721 wv->enabled = 1;
3427a3db 1722 wv->button_type = BUTTON_TYPE_NONE;
88766961
RS
1723 prev_wv = wv;
1724 }
1725
1726 finish_menu_items ();
1727
1728 set_buffer_internal_1 (prev);
8d8a3494 1729 unbind_to (specpdl_count, Qnil);
8d8a3494 1730
88766961
RS
1731 /* If there has been no change in the Lisp-level contents
1732 of the menu bar, skip redisplaying it. Just exit. */
1733
1734 for (i = 0; i < previous_menu_items_used; i++)
1735 if (menu_items_used == i
99d3fac7 1736 || (!EQ (previous_items[i], XVECTOR (menu_items)->contents[i])))
88766961 1737 break;
62555c22 1738 if (i == menu_items_used && i == previous_menu_items_used && i != 0)
88766961
RS
1739 {
1740 free_menubar_widget_value_tree (first_wv);
1741 menu_items = Qnil;
1742
1743 return;
1744 }
1745
1746 /* Now GC cannot happen during the lifetime of the widget_value,
1747 so it's safe to store data from a Lisp_String. */
1748 wv = first_wv->contents;
f04366cb 1749 for (i = 0; i < XVECTOR (items)->size; i += 4)
88766961
RS
1750 {
1751 Lisp_Object string;
1752 string = XVECTOR (items)->contents[i + 1];
1753 if (NILP (string))
1754 break;
1755 wv->name = (char *) XSTRING (string)->data;
1756 wv = wv->next;
1757 }
1758
1759 f->menu_bar_vector = menu_items;
1760 f->menu_bar_items_used = menu_items_used;
1761 menu_items = Qnil;
4d19cb8e 1762 }
88766961
RS
1763 else
1764 {
1765 /* Make a widget-value tree containing
1766 just the top level menu bar strings. */
4d19cb8e 1767
88766961 1768 items = FRAME_MENU_BAR_ITEMS (f);
f04366cb 1769 for (i = 0; i < XVECTOR (items)->size; i += 4)
88766961
RS
1770 {
1771 Lisp_Object string;
1772
1773 string = XVECTOR (items)->contents[i + 1];
1774 if (NILP (string))
1775 break;
1776
f7fab165 1777 wv = xmalloc_widget_value ();
88766961
RS
1778 wv->name = (char *) XSTRING (string)->data;
1779 wv->value = 0;
1780 wv->enabled = 1;
3427a3db 1781 wv->button_type = BUTTON_TYPE_NONE;
fe8fa62f
RS
1782 /* This prevents lwlib from assuming this
1783 menu item is really supposed to be empty. */
1784 /* The EMACS_INT cast avoids a warning.
1785 This value just has to be different from small integers. */
1786 wv->call_data = (void *) (EMACS_INT) (-1);
88766961
RS
1787
1788 if (prev_wv)
1789 prev_wv->next = wv;
1790 else
1791 first_wv->contents = wv;
1792 prev_wv = wv;
1793 }
62555c22
RS
1794
1795 /* Forget what we thought we knew about what is in the
1796 detailed contents of the menu bar menus.
1797 Changing the top level always destroys the contents. */
1798 f->menu_bar_items_used = 0;
88766961 1799 }
4dedbfe0 1800
88766961 1801 /* Create or update the menu bar widget. */
aa669def
RS
1802
1803 BLOCK_INPUT;
1804
18686d47 1805 if (menubar_widget)
4dedbfe0
PR
1806 {
1807 /* Disable resizing (done for Motif!) */
7556890b 1808 lw_allow_resizing (f->output_data.x->widget, False);
4dedbfe0
PR
1809
1810 /* The third arg is DEEP_P, which says to consider the entire
1811 menu trees we supply, rather than just the menu bar item names. */
88766961 1812 lw_modify_all_widgets (id, first_wv, deep_p);
4dedbfe0 1813
c98fcf4b 1814 /* Re-enable the edit widget to resize. */
7556890b 1815 lw_allow_resizing (f->output_data.x->widget, True);
4dedbfe0 1816 }
18686d47
RS
1817 else
1818 {
88766961 1819 menubar_widget = lw_create_widget ("menubar", "menubar", id, first_wv,
7556890b 1820 f->output_data.x->column_widget,
4dedbfe0
PR
1821 0,
1822 popup_activate_callback,
1823 menubar_selection_callback,
850df50b
GM
1824 popup_deactivate_callback,
1825 menu_highlight_callback);
7556890b 1826 f->output_data.x->menubar_widget = menubar_widget;
18686d47 1827 }
1d1c1567
KH
1828
1829 {
1830 int menubar_size
7556890b
RS
1831 = (f->output_data.x->menubar_widget
1832 ? (f->output_data.x->menubar_widget->core.height
1833 + f->output_data.x->menubar_widget->core.border_width)
1d1c1567
KH
1834 : 0);
1835
f42aa681
RS
1836#if 0 /* Experimentally, we now get the right results
1837 for -geometry -0-0 without this. 24 Aug 96, rms. */
5ee80bd3 1838#ifdef USE_LUCID
1d1c1567
KH
1839 if (FRAME_EXTERNAL_MENU_BAR (f))
1840 {
1841 Dimension ibw = 0;
7556890b 1842 XtVaGetValues (f->output_data.x->column_widget,
1d1c1567
KH
1843 XtNinternalBorderWidth, &ibw, NULL);
1844 menubar_size += ibw;
1845 }
5ee80bd3 1846#endif /* USE_LUCID */
f42aa681 1847#endif /* 0 */
1d1c1567 1848
7556890b 1849 f->output_data.x->menubar_height = menubar_size;
1d1c1567 1850 }
18686d47
RS
1851
1852 free_menubar_widget_value_tree (first_wv);
364cd450 1853 update_frame_menubar (f);
18686d47
RS
1854
1855 UNBLOCK_INPUT;
1856}
85f487d1 1857
8e6208c5 1858/* Called from Fx_create_frame to create the initial menubar of a frame
4dedbfe0
PR
1859 before it is mapped, so that the window is mapped with the menubar already
1860 there instead of us tacking it on later and thrashing the window after it
1861 is visible. */
1862
1863void
1864initialize_frame_menubar (f)
1865 FRAME_PTR f;
1866{
1867 /* This function is called before the first chance to redisplay
1868 the frame. It has to be, so the frame will have the right size. */
1869 FRAME_MENU_BAR_ITEMS (f) = menu_bar_items (FRAME_MENU_BAR_ITEMS (f));
88766961 1870 set_frame_menubar (f, 1, 1);
4dedbfe0
PR
1871}
1872
aba25348 1873
4dedbfe0
PR
1874/* Get rid of the menu bar of frame F, and free its storage.
1875 This is used when deleting a frame, and when turning off the menu bar. */
1876
85f487d1
FP
1877void
1878free_frame_menubar (f)
1879 FRAME_PTR f;
1880{
1881 Widget menubar_widget;
85f487d1 1882
7556890b 1883 menubar_widget = f->output_data.x->menubar_widget;
a45bad2a
RS
1884
1885 f->output_data.x->menubar_height = 0;
85f487d1
FP
1886
1887 if (menubar_widget)
1888 {
aba25348
GM
1889#ifdef USE_MOTIF
1890 /* Removing the menu bar magically changes the shell widget's x
1891 and y position of (0, 0) which, when the menu bar is turned
1892 on again, leads to pull-down menuss appearing in strange
1893 positions near the upper-left corner of the display. This
1894 happens only with some window managers like twm and ctwm,
1895 but not with other like Motif's mwm or kwm, because the
1896 latter generate ConfigureNotify events when the menu bar
1897 is switched off, which fixes the shell position. */
1898 Position x0, y0, x1, y1;
1899#endif
1900
85f487d1 1901 BLOCK_INPUT;
aba25348
GM
1902
1903#ifdef USE_MOTIF
ae556422
GM
1904 if (f->output_data.x->widget)
1905 XtVaGetValues (f->output_data.x->widget, XtNx, &x0, XtNy, &y0, NULL);
aba25348
GM
1906#endif
1907
7556890b 1908 lw_destroy_all_widgets ((LWLIB_ID) f->output_data.x->id);
0c05dcd3 1909 f->output_data.x->menubar_widget = NULL;
aba25348
GM
1910
1911#ifdef USE_MOTIF
ae556422
GM
1912 if (f->output_data.x->widget)
1913 {
1914 XtVaGetValues (f->output_data.x->widget, XtNx, &x1, XtNy, &y1, NULL);
1915 if (x1 == 0 && y1 == 0)
1916 XtVaSetValues (f->output_data.x->widget, XtNx, x0, XtNy, y0, NULL);
1917 }
aba25348
GM
1918#endif
1919
85f487d1
FP
1920 UNBLOCK_INPUT;
1921 }
1922}
78589e07 1923
78589e07
RS
1924#endif /* USE_X_TOOLKIT */
1925\f
1926/* xmenu_show actually displays a menu using the panes and items in menu_items
1927 and returns the value selected from it.
1928 There are two versions of xmenu_show, one for Xt and one for Xlib.
1929 Both assume input is blocked by the caller. */
1930
1931/* F is the frame the menu is for.
1932 X and Y are the frame-relative specified position,
1933 relative to the inside upper left corner of the frame F.
c8b5aa3d 1934 FOR_CLICK is nonzero if this menu was invoked for a mouse click.
78589e07
RS
1935 KEYMAPS is 1 if this menu was specified with keymaps;
1936 in that case, we return a list containing the chosen item's value
1937 and perhaps also the pane's prefix.
1938 TITLE is the specified menu title.
1939 ERROR is a place to store an error message string in case of failure.
1940 (We return nil on failure, but the value doesn't actually matter.) */
18686d47
RS
1941
1942#ifdef USE_X_TOOLKIT
18686d47 1943
8ed87156 1944/* We need a unique id for each widget handled by the Lucid Widget
cc17e9bf
KH
1945 library.
1946
1947 For the main windows, and popup menus, we use this counter,
88766961 1948 which we increment each time after use. This starts from 1<<16.
cc17e9bf 1949
88766961
RS
1950 For menu bars, we use numbers starting at 0, counted in
1951 next_menubar_widget_id. */
8ed87156 1952LWLIB_ID widget_id_tick;
165e1749 1953
4dedbfe0 1954static Lisp_Object *volatile menu_item_selection;
4dedbfe0
PR
1955
1956static void
1957popup_selection_callback (widget, id, client_data)
1958 Widget widget;
1959 LWLIB_ID id;
1960 XtPointer client_data;
1961{
1962 menu_item_selection = (Lisp_Object *) client_data;
1963}
1964
78589e07 1965static Lisp_Object
673a6211 1966xmenu_show (f, x, y, for_click, keymaps, title, error)
18686d47 1967 FRAME_PTR f;
18686d47
RS
1968 int x;
1969 int y;
9685a93f 1970 int for_click;
78589e07
RS
1971 int keymaps;
1972 Lisp_Object title;
1973 char **error;
18686d47 1974{
78589e07 1975 int i;
cc17e9bf 1976 LWLIB_ID menu_id;
18686d47 1977 Widget menu;
ffcb5a51 1978 Arg av[2];
60f312e2 1979 int ac = 0;
78589e07 1980 widget_value *wv, *save_wv = 0, *first_wv = 0, *prev_wv = 0;
101bb4a5
RS
1981 widget_value **submenu_stack
1982 = (widget_value **) alloca (menu_items_used * sizeof (widget_value *));
1983 Lisp_Object *subprefix_stack
1984 = (Lisp_Object *) alloca (menu_items_used * sizeof (Lisp_Object));
1985 int submenu_depth = 0;
ffcb5a51 1986 XButtonPressedEvent dummy;
4e8d3549 1987
78c8278d
RS
1988 int first_pane;
1989
78589e07
RS
1990 *error = NULL;
1991
742f715d
KH
1992 if (menu_items_used <= MENU_ITEMS_PANE_LENGTH)
1993 {
1994 *error = "Empty menu";
1995 return Qnil;
1996 }
63c414df 1997
78589e07
RS
1998 /* Create a tree of widget_value objects
1999 representing the panes and their items. */
f7fab165 2000 wv = xmalloc_widget_value ();
78589e07
RS
2001 wv->name = "menu";
2002 wv->value = 0;
2003 wv->enabled = 1;
3427a3db 2004 wv->button_type = BUTTON_TYPE_NONE;
78589e07 2005 first_wv = wv;
78c8278d 2006 first_pane = 1;
78589e07
RS
2007
2008 /* Loop over all panes and items, filling in the tree. */
2009 i = 0;
2010 while (i < menu_items_used)
2011 {
101bb4a5
RS
2012 if (EQ (XVECTOR (menu_items)->contents[i], Qnil))
2013 {
2014 submenu_stack[submenu_depth++] = save_wv;
2015 save_wv = prev_wv;
2016 prev_wv = 0;
78c8278d 2017 first_pane = 1;
101bb4a5
RS
2018 i++;
2019 }
2020 else if (EQ (XVECTOR (menu_items)->contents[i], Qlambda))
2021 {
2022 prev_wv = save_wv;
2023 save_wv = submenu_stack[--submenu_depth];
78c8278d 2024 first_pane = 0;
101bb4a5
RS
2025 i++;
2026 }
2027 else if (EQ (XVECTOR (menu_items)->contents[i], Qt)
2028 && submenu_depth != 0)
2029 i += MENU_ITEMS_PANE_LENGTH;
fcaa7665
RS
2030 /* Ignore a nil in the item list.
2031 It's meaningful only for dialog boxes. */
2032 else if (EQ (XVECTOR (menu_items)->contents[i], Qquote))
2033 i += 1;
101bb4a5 2034 else if (EQ (XVECTOR (menu_items)->contents[i], Qt))
78589e07
RS
2035 {
2036 /* Create a new pane. */
2037 Lisp_Object pane_name, prefix;
2038 char *pane_string;
4c329aa8
GM
2039
2040 pane_name = AREF (menu_items, i + MENU_ITEMS_PANE_NAME);
2041 prefix = AREF (menu_items, i + MENU_ITEMS_PANE_PREFIX);
2042
703dc2a8 2043#ifndef HAVE_MULTILINGUAL_MENU
4c329aa8
GM
2044 if (STRINGP (pane_name) && STRING_MULTIBYTE (pane_name))
2045 {
cc45fb40 2046 pane_name = ENCODE_SYSTEM (pane_name);
4c329aa8
GM
2047 AREF (menu_items, i + MENU_ITEMS_PANE_NAME) = pane_name;
2048 }
703dc2a8 2049#endif
78589e07
RS
2050 pane_string = (NILP (pane_name)
2051 ? "" : (char *) XSTRING (pane_name)->data);
101bb4a5 2052 /* If there is just one top-level pane, put all its items directly
78589e07
RS
2053 under the top-level menu. */
2054 if (menu_items_n_panes == 1)
2055 pane_string = "";
2056
2057 /* If the pane has a meaningful name,
2058 make the pane a top-level menu item
2059 with its items as a submenu beneath it. */
78c8278d 2060 if (!keymaps && strcmp (pane_string, ""))
78589e07 2061 {
f7fab165 2062 wv = xmalloc_widget_value ();
78589e07
RS
2063 if (save_wv)
2064 save_wv->next = wv;
2065 else
2066 first_wv->contents = wv;
2067 wv->name = pane_string;
2068 if (keymaps && !NILP (prefix))
2069 wv->name++;
2070 wv->value = 0;
2071 wv->enabled = 1;
3427a3db 2072 wv->button_type = BUTTON_TYPE_NONE;
78c8278d
RS
2073 save_wv = wv;
2074 prev_wv = 0;
78589e07 2075 }
78c8278d
RS
2076 else if (first_pane)
2077 {
2078 save_wv = wv;
2079 prev_wv = 0;
2080 }
2081 first_pane = 0;
78589e07
RS
2082 i += MENU_ITEMS_PANE_LENGTH;
2083 }
2084 else
2085 {
2086 /* Create a new item within current pane. */
9cd50434 2087 Lisp_Object item_name, enable, descrip, def, type, selected, help;
4c329aa8
GM
2088 item_name = AREF (menu_items, i + MENU_ITEMS_ITEM_NAME);
2089 enable = AREF (menu_items, i + MENU_ITEMS_ITEM_ENABLE);
2090 descrip = AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY);
2091 def = AREF (menu_items, i + MENU_ITEMS_ITEM_DEFINITION);
2092 type = AREF (menu_items, i + MENU_ITEMS_ITEM_TYPE);
2093 selected = AREF (menu_items, i + MENU_ITEMS_ITEM_SELECTED);
2094 help = AREF (menu_items, i + MENU_ITEMS_ITEM_HELP);
3427a3db 2095
703dc2a8 2096#ifndef HAVE_MULTILINGUAL_MENU
3427a3db 2097 if (STRINGP (item_name) && STRING_MULTIBYTE (item_name))
4c329aa8 2098 {
cc45fb40 2099 item_name = ENCODE_SYSTEM (item_name);
4c329aa8
GM
2100 AREF (menu_items, i + MENU_ITEMS_ITEM_NAME) = item_name;
2101 }
2102
3427a3db 2103 if (STRINGP (descrip) && STRING_MULTIBYTE (descrip))
4c329aa8 2104 {
cc45fb40 2105 descrip = ENCODE_SYSTEM (descrip);
4c329aa8
GM
2106 AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY) = descrip;
2107 }
2108#endif /* not HAVE_MULTILINGUAL_MENU */
3427a3db 2109
f7fab165 2110 wv = xmalloc_widget_value ();
78589e07
RS
2111 if (prev_wv)
2112 prev_wv->next = wv;
2113 else
2114 save_wv->contents = wv;
4e8d3549 2115 wv->name = (char *) XSTRING (item_name)->data;
78589e07 2116 if (!NILP (descrip))
4e8d3549 2117 wv->key = (char *) XSTRING (descrip)->data;
78589e07 2118 wv->value = 0;
a352a815
RS
2119 /* If this item has a null value,
2120 make the call_data null so that it won't display a box
2121 when the mouse is on it. */
2122 wv->call_data
2123 = (!NILP (def) ? (void *) &XVECTOR (menu_items)->contents[i] : 0);
78589e07 2124 wv->enabled = !NILP (enable);
3427a3db
GM
2125
2126 if (NILP (type))
2127 wv->button_type = BUTTON_TYPE_NONE;
2128 else if (EQ (type, QCtoggle))
2129 wv->button_type = BUTTON_TYPE_TOGGLE;
2130 else if (EQ (type, QCradio))
2131 wv->button_type = BUTTON_TYPE_RADIO;
2132 else
2133 abort ();
2134
2135 wv->selected = !NILP (selected);
cc45fb40 2136
0b1f4572
RS
2137 if (! STRINGP (help))
2138 help = Qnil;
2139
2140 wv->help = help;
cc45fb40 2141
78589e07
RS
2142 prev_wv = wv;
2143
2144 i += MENU_ITEMS_ITEM_LENGTH;
2145 }
2146 }
2147
c98fcf4b 2148 /* Deal with the title, if it is non-nil. */
4dedbfe0
PR
2149 if (!NILP (title))
2150 {
f7fab165
RS
2151 widget_value *wv_title = xmalloc_widget_value ();
2152 widget_value *wv_sep1 = xmalloc_widget_value ();
2153 widget_value *wv_sep2 = xmalloc_widget_value ();
4dedbfe0
PR
2154
2155 wv_sep2->name = "--";
2156 wv_sep2->next = first_wv->contents;
2157
2158 wv_sep1->name = "--";
2159 wv_sep1->next = wv_sep2;
2160
703dc2a8
KH
2161#ifndef HAVE_MULTILINGUAL_MENU
2162 if (STRING_MULTIBYTE (title))
cc45fb40 2163 title = ENCODE_SYSTEM (title);
703dc2a8 2164#endif
4c329aa8 2165
4dedbfe0 2166 wv_title->name = (char *) XSTRING (title)->data;
cc45fb40 2167 wv_title->enabled = TRUE;
3427a3db 2168 wv_title->button_type = BUTTON_TYPE_NONE;
4dedbfe0
PR
2169 wv_title->next = wv_sep1;
2170 first_wv->contents = wv_title;
2171 }
2172
78589e07 2173 /* Actually create the menu. */
cc17e9bf 2174 menu_id = widget_id_tick++;
78589e07 2175 menu = lw_create_widget ("popup", first_wv->name, menu_id, first_wv,
7556890b 2176 f->output_data.x->widget, 1, 0,
4dedbfe0 2177 popup_selection_callback,
850df50b
GM
2178 popup_deactivate_callback,
2179 menu_highlight_callback);
60f312e2 2180
ffcb5a51
RS
2181 /* Adjust coordinates to relative to the outer (window manager) window. */
2182 {
2183 Window child;
2184 int win_x = 0, win_y = 0;
2185
2186 /* Find the position of the outside upper-left corner of
2187 the inner window, with respect to the outer window. */
2188 if (f->output_data.x->parent_desc != FRAME_X_DISPLAY_INFO (f)->root_window)
2189 {
2190 BLOCK_INPUT;
2191 XTranslateCoordinates (FRAME_X_DISPLAY (f),
2192
2193 /* From-window, to-window. */
2194 f->output_data.x->window_desc,
2195 f->output_data.x->parent_desc,
2196
2197 /* From-position, to-position. */
2198 0, 0, &win_x, &win_y,
2199
2200 /* Child of window. */
2201 &child);
2202 UNBLOCK_INPUT;
2203 x += win_x;
2204 y += win_y;
2205 }
2206 }
2207
2208 /* Adjust coordinates to be root-window-relative. */
2209 x += f->output_data.x->left_pos;
2210 y += f->output_data.x->top_pos;
2211
2212 dummy.type = ButtonPress;
2213 dummy.serial = 0;
2214 dummy.send_event = 0;
2215 dummy.display = FRAME_X_DISPLAY (f);
2216 dummy.time = CurrentTime;
ffcb5a51
RS
2217 dummy.root = FRAME_X_DISPLAY_INFO (f)->root_window;
2218 dummy.window = dummy.root;
2219 dummy.subwindow = dummy.root;
2220 dummy.x_root = x;
2221 dummy.y_root = y;
2222 dummy.x = x;
2223 dummy.y = y;
c8b5aa3d
RS
2224 dummy.state = (FRAME_X_DISPLAY_INFO (f)->grabbed >> 1) * Button1Mask;
2225 dummy.button = 0;
2226 for (i = 0; i < 5; i++)
2227 if (FRAME_X_DISPLAY_INFO (f)->grabbed & (1 << i))
2228 dummy.button = i;
ffcb5a51 2229
60f312e2
RS
2230 /* Don't allow any geometry request from the user. */
2231 XtSetArg (av[ac], XtNgeometry, 0); ac++;
2232 XtSetValues (menu, av, ac);
2233
78589e07
RS
2234 /* Free the widget_value objects we used to specify the contents. */
2235 free_menubar_widget_value_tree (first_wv);
2236
2237 /* No selection has been chosen yet. */
2238 menu_item_selection = 0;
2239
78589e07 2240 /* Display the menu. */
a52445bd 2241 lw_popup_menu (menu, (XEvent *) &dummy);
4dedbfe0 2242 popup_activated_flag = 1;
18686d47 2243
78589e07 2244 /* Process events that apply to the menu. */
2e2b8e22 2245 popup_get_selection ((XEvent *) 0, FRAME_X_DISPLAY_INFO (f), menu_id);
78589e07 2246
a9c90b7c
RS
2247 /* fp turned off the following statement and wrote a comment
2248 that it is unnecessary--that the menu has already disappeared.
21af8a68
KH
2249 Nowadays the menu disappears ok, all right, but
2250 we need to delete the widgets or multiple ones will pile up. */
78589e07 2251 lw_destroy_all_widgets (menu_id);
18686d47 2252
78589e07
RS
2253 /* Find the selected item, and its pane, to return
2254 the proper value. */
2255 if (menu_item_selection != 0)
2256 {
c63f6952 2257 Lisp_Object prefix, entry;
78589e07 2258
6bbd7a29 2259 prefix = entry = Qnil;
78589e07
RS
2260 i = 0;
2261 while (i < menu_items_used)
2262 {
101bb4a5
RS
2263 if (EQ (XVECTOR (menu_items)->contents[i], Qnil))
2264 {
2265 subprefix_stack[submenu_depth++] = prefix;
2266 prefix = entry;
2267 i++;
2268 }
2269 else if (EQ (XVECTOR (menu_items)->contents[i], Qlambda))
2270 {
2271 prefix = subprefix_stack[--submenu_depth];
2272 i++;
2273 }
2274 else if (EQ (XVECTOR (menu_items)->contents[i], Qt))
78589e07
RS
2275 {
2276 prefix
2277 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_PREFIX];
2278 i += MENU_ITEMS_PANE_LENGTH;
2279 }
d31d42cc
RS
2280 /* Ignore a nil in the item list.
2281 It's meaningful only for dialog boxes. */
2282 else if (EQ (XVECTOR (menu_items)->contents[i], Qquote))
2283 i += 1;
78589e07
RS
2284 else
2285 {
2286 entry
2287 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_VALUE];
2288 if (menu_item_selection == &XVECTOR (menu_items)->contents[i])
2289 {
2290 if (keymaps != 0)
2291 {
101bb4a5
RS
2292 int j;
2293
78589e07
RS
2294 entry = Fcons (entry, Qnil);
2295 if (!NILP (prefix))
2296 entry = Fcons (prefix, entry);
101bb4a5 2297 for (j = submenu_depth - 1; j >= 0; j--)
e48087b7 2298 if (!NILP (subprefix_stack[j]))
5964e450 2299 entry = Fcons (subprefix_stack[j], entry);
78589e07
RS
2300 }
2301 return entry;
2302 }
2303 i += MENU_ITEMS_ITEM_LENGTH;
2304 }
2305 }
2306 }
2307
2308 return Qnil;
18686d47 2309}
4dedbfe0
PR
2310\f
2311static void
2312dialog_selection_callback (widget, id, client_data)
2313 Widget widget;
2314 LWLIB_ID id;
2315 XtPointer client_data;
2316{
01d5e892
RS
2317 /* The EMACS_INT cast avoids a warning. There's no problem
2318 as long as pointers have enough bits to hold small integers. */
2319 if ((int) (EMACS_INT) client_data != -1)
4dedbfe0
PR
2320 menu_item_selection = (Lisp_Object *) client_data;
2321 BLOCK_INPUT;
2322 lw_destroy_all_widgets (id);
2323 UNBLOCK_INPUT;
9572375b 2324 popup_activated_flag = 0;
4dedbfe0 2325}
18686d47 2326
165e1749
FP
2327static char * button_names [] = {
2328 "button1", "button2", "button3", "button4", "button5",
2329 "button6", "button7", "button8", "button9", "button10" };
2330
2331static Lisp_Object
673a6211 2332xdialog_show (f, keymaps, title, error)
165e1749 2333 FRAME_PTR f;
165e1749
FP
2334 int keymaps;
2335 Lisp_Object title;
2336 char **error;
2337{
2338 int i, nb_buttons=0;
cc17e9bf 2339 LWLIB_ID dialog_id;
165e1749 2340 Widget menu;
80670155 2341 char dialog_name[6];
165e1749 2342
faa935b6 2343 widget_value *wv, *first_wv = 0, *prev_wv = 0;
165e1749 2344
fcaa7665
RS
2345 /* Number of elements seen so far, before boundary. */
2346 int left_count = 0;
2347 /* 1 means we've seen the boundary between left-hand elts and right-hand. */
2348 int boundary_seen = 0;
2349
165e1749
FP
2350 *error = NULL;
2351
80670155
RS
2352 if (menu_items_n_panes > 1)
2353 {
2354 *error = "Multiple panes in dialog box";
2355 return Qnil;
2356 }
2357
165e1749
FP
2358 /* Create a tree of widget_value objects
2359 representing the text label and buttons. */
2360 {
2361 Lisp_Object pane_name, prefix;
2362 char *pane_string;
2363 pane_name = XVECTOR (menu_items)->contents[MENU_ITEMS_PANE_NAME];
2364 prefix = XVECTOR (menu_items)->contents[MENU_ITEMS_PANE_PREFIX];
2365 pane_string = (NILP (pane_name)
2366 ? "" : (char *) XSTRING (pane_name)->data);
f7fab165 2367 prev_wv = xmalloc_widget_value ();
165e1749
FP
2368 prev_wv->value = pane_string;
2369 if (keymaps && !NILP (prefix))
2370 prev_wv->name++;
2371 prev_wv->enabled = 1;
2372 prev_wv->name = "message";
2373 first_wv = prev_wv;
2374
2375 /* Loop over all panes and items, filling in the tree. */
2376 i = MENU_ITEMS_PANE_LENGTH;
2377 while (i < menu_items_used)
2378 {
2379
2380 /* Create a new item within current pane. */
2381 Lisp_Object item_name, enable, descrip;
2382 item_name = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_NAME];
2383 enable = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_ENABLE];
2384 descrip
2385 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_EQUIV_KEY];
2386
80670155
RS
2387 if (NILP (item_name))
2388 {
2389 free_menubar_widget_value_tree (first_wv);
2390 *error = "Submenu in dialog items";
2391 return Qnil;
2392 }
fcaa7665
RS
2393 if (EQ (item_name, Qquote))
2394 {
2395 /* This is the boundary between left-side elts
2396 and right-side elts. Stop incrementing right_count. */
2397 boundary_seen = 1;
2398 i++;
2399 continue;
2400 }
86e71abf 2401 if (nb_buttons >= 9)
80670155
RS
2402 {
2403 free_menubar_widget_value_tree (first_wv);
2404 *error = "Too many dialog items";
2405 return Qnil;
2406 }
2407
f7fab165 2408 wv = xmalloc_widget_value ();
165e1749 2409 prev_wv->next = wv;
80670155 2410 wv->name = (char *) button_names[nb_buttons];
165e1749 2411 if (!NILP (descrip))
4e8d3549
RS
2412 wv->key = (char *) XSTRING (descrip)->data;
2413 wv->value = (char *) XSTRING (item_name)->data;
165e1749
FP
2414 wv->call_data = (void *) &XVECTOR (menu_items)->contents[i];
2415 wv->enabled = !NILP (enable);
2416 prev_wv = wv;
2417
fcaa7665
RS
2418 if (! boundary_seen)
2419 left_count++;
2420
165e1749
FP
2421 nb_buttons++;
2422 i += MENU_ITEMS_ITEM_LENGTH;
2423 }
2424
fcaa7665
RS
2425 /* If the boundary was not specified,
2426 by default put half on the left and half on the right. */
2427 if (! boundary_seen)
2428 left_count = nb_buttons - nb_buttons / 2;
2429
f7fab165 2430 wv = xmalloc_widget_value ();
80670155
RS
2431 wv->name = dialog_name;
2432
2433 /* Dialog boxes use a really stupid name encoding
2434 which specifies how many buttons to use
2435 and how many buttons are on the right.
2436 The Q means something also. */
2437 dialog_name[0] = 'Q';
2438 dialog_name[1] = '0' + nb_buttons;
2439 dialog_name[2] = 'B';
2440 dialog_name[3] = 'R';
fcaa7665
RS
2441 /* Number of buttons to put on the right. */
2442 dialog_name[4] = '0' + nb_buttons - left_count;
80670155 2443 dialog_name[5] = 0;
165e1749
FP
2444 wv->contents = first_wv;
2445 first_wv = wv;
165e1749
FP
2446 }
2447
2448 /* Actually create the dialog. */
cc17e9bf 2449 dialog_id = widget_id_tick++;
165e1749 2450 menu = lw_create_widget (first_wv->name, "dialog", dialog_id, first_wv,
7556890b 2451 f->output_data.x->widget, 1, 0,
850df50b 2452 dialog_selection_callback, 0, 0);
b5587215 2453 lw_modify_all_widgets (dialog_id, first_wv->contents, True);
165e1749
FP
2454 /* Free the widget_value objects we used to specify the contents. */
2455 free_menubar_widget_value_tree (first_wv);
2456
2457 /* No selection has been chosen yet. */
2458 menu_item_selection = 0;
2459
165e1749
FP
2460 /* Display the menu. */
2461 lw_pop_up_all_widgets (dialog_id);
aa669def 2462 popup_activated_flag = 1;
165e1749
FP
2463
2464 /* Process events that apply to the menu. */
2e2b8e22 2465 popup_get_selection ((XEvent *) 0, FRAME_X_DISPLAY_INFO (f), dialog_id);
165e1749 2466
21af8a68
KH
2467 lw_destroy_all_widgets (dialog_id);
2468
165e1749
FP
2469 /* Find the selected item, and its pane, to return
2470 the proper value. */
2471 if (menu_item_selection != 0)
2472 {
2473 Lisp_Object prefix;
2474
2475 prefix = Qnil;
2476 i = 0;
2477 while (i < menu_items_used)
2478 {
2479 Lisp_Object entry;
2480
2481 if (EQ (XVECTOR (menu_items)->contents[i], Qt))
2482 {
2483 prefix
2484 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_PREFIX];
2485 i += MENU_ITEMS_PANE_LENGTH;
2486 }
85996cfb
GM
2487 else if (EQ (XVECTOR (menu_items)->contents[i], Qquote))
2488 {
2489 /* This is the boundary between left-side elts and
2490 right-side elts. */
2491 ++i;
2492 }
165e1749
FP
2493 else
2494 {
2495 entry
2496 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_VALUE];
2497 if (menu_item_selection == &XVECTOR (menu_items)->contents[i])
2498 {
2499 if (keymaps != 0)
2500 {
2501 entry = Fcons (entry, Qnil);
2502 if (!NILP (prefix))
2503 entry = Fcons (prefix, entry);
2504 }
2505 return entry;
2506 }
2507 i += MENU_ITEMS_ITEM_LENGTH;
2508 }
2509 }
2510 }
2511
2512 return Qnil;
2513}
ba461919 2514
18686d47 2515#else /* not USE_X_TOOLKIT */
78589e07 2516
3e703b25
GM
2517/* The frame of the last activated non-toolkit menu bar.
2518 Used to generate menu help events. */
2519
2520static struct frame *menu_help_frame;
2521
2522
62145073
GM
2523/* Show help HELP_STRING, or clear help if HELP_STRING is null.
2524
2525 PANE is the pane number, and ITEM is the menu item number in
2526 the menu (currently not used).
2527
2528 This cannot be done with generating a HELP_EVENT because
2529 XMenuActivate contains a loop that doesn't let Emacs process
2530 keyboard events. */
3e703b25
GM
2531
2532static void
62145073 2533menu_help_callback (help_string, pane, item)
3e703b25 2534 char *help_string;
62145073 2535 int pane, item;
3e703b25 2536{
62145073
GM
2537 extern Lisp_Object Qmenu_item;
2538 Lisp_Object *first_item;
2539 Lisp_Object pane_name;
2540 Lisp_Object menu_object;
2541
2542 first_item = XVECTOR (menu_items)->contents;
2543 if (EQ (first_item[0], Qt))
2544 pane_name = first_item[MENU_ITEMS_PANE_NAME];
2545 else if (EQ (first_item[0], Qquote))
2546 /* This shouldn't happen, see xmenu_show. */
cc45fb40 2547 pane_name = empty_string;
62145073
GM
2548 else
2549 pane_name = first_item[MENU_ITEMS_ITEM_NAME];
2550
2551 /* (menu-item MENU-NAME PANE-NUMBER) */
2552 menu_object = Fcons (Qmenu_item,
2553 Fcons (pane_name,
2554 Fcons (make_number (pane), Qnil)));
ba461919 2555 show_help_echo (help_string ? build_string (help_string) : Qnil,
62145073 2556 Qnil, menu_object, make_number (item), 1);
3e703b25 2557}
62145073 2558
3e703b25 2559
78589e07 2560static Lisp_Object
673a6211 2561xmenu_show (f, x, y, for_click, keymaps, title, error)
78589e07
RS
2562 FRAME_PTR f;
2563 int x, y;
9685a93f
RS
2564 int for_click;
2565 int keymaps;
78589e07
RS
2566 Lisp_Object title;
2567 char **error;
dcfdbac7 2568{
78589e07
RS
2569 Window root;
2570 XMenu *menu;
2571 int pane, selidx, lpane, status;
2572 Lisp_Object entry, pane_prefix;
dcfdbac7
JB
2573 char *datap;
2574 int ulx, uly, width, height;
2575 int dispwidth, dispheight;
4e8d3549
RS
2576 int i, j;
2577 int maxwidth;
78589e07
RS
2578 int dummy_int;
2579 unsigned int dummy_uint;
088831f6 2580
07a675b7 2581 *error = 0;
78589e07
RS
2582 if (menu_items_n_panes == 0)
2583 return Qnil;
088831f6 2584
742f715d
KH
2585 if (menu_items_used <= MENU_ITEMS_PANE_LENGTH)
2586 {
2587 *error = "Empty menu";
2588 return Qnil;
2589 }
2590
78589e07 2591 /* Figure out which root window F is on. */
92280f67 2592 XGetGeometry (FRAME_X_DISPLAY (f), FRAME_X_WINDOW (f), &root,
78589e07
RS
2593 &dummy_int, &dummy_int, &dummy_uint, &dummy_uint,
2594 &dummy_uint, &dummy_uint);
18686d47 2595
78589e07 2596 /* Make the menu on that window. */
92280f67 2597 menu = XMenuCreate (FRAME_X_DISPLAY (f), root, "emacs");
78589e07 2598 if (menu == NULL)
dcfdbac7
JB
2599 {
2600 *error = "Can't create menu";
78589e07 2601 return Qnil;
dcfdbac7 2602 }
78589e07 2603
87485d6f 2604#ifdef HAVE_X_WINDOWS
78589e07 2605 /* Adjust coordinates to relative to the outer (window manager) window. */
78589e07
RS
2606 {
2607 Window child;
2608 int win_x = 0, win_y = 0;
2609
2610 /* Find the position of the outside upper-left corner of
2611 the inner window, with respect to the outer window. */
7556890b 2612 if (f->output_data.x->parent_desc != FRAME_X_DISPLAY_INFO (f)->root_window)
78589e07
RS
2613 {
2614 BLOCK_INPUT;
92280f67 2615 XTranslateCoordinates (FRAME_X_DISPLAY (f),
78589e07
RS
2616
2617 /* From-window, to-window. */
7556890b
RS
2618 f->output_data.x->window_desc,
2619 f->output_data.x->parent_desc,
78589e07
RS
2620
2621 /* From-position, to-position. */
2622 0, 0, &win_x, &win_y,
2623
2624 /* Child of window. */
2625 &child);
2626 UNBLOCK_INPUT;
2627 x += win_x;
2628 y += win_y;
2629 }
2630 }
87485d6f 2631#endif /* HAVE_X_WINDOWS */
78589e07
RS
2632
2633 /* Adjust coordinates to be root-window-relative. */
7556890b
RS
2634 x += f->output_data.x->left_pos;
2635 y += f->output_data.x->top_pos;
18686d47 2636
78589e07
RS
2637 /* Create all the necessary panes and their items. */
2638 i = 0;
2639 while (i < menu_items_used)
dcfdbac7 2640 {
78589e07 2641 if (EQ (XVECTOR (menu_items)->contents[i], Qt))
dcfdbac7 2642 {
78589e07
RS
2643 /* Create a new pane. */
2644 Lisp_Object pane_name, prefix;
2645 char *pane_string;
2646
2647 pane_name = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_NAME];
2648 prefix = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_PREFIX];
2649 pane_string = (NILP (pane_name)
2650 ? "" : (char *) XSTRING (pane_name)->data);
2651 if (keymaps && !NILP (prefix))
2652 pane_string++;
2653
92280f67 2654 lpane = XMenuAddPane (FRAME_X_DISPLAY (f), menu, pane_string, TRUE);
78589e07
RS
2655 if (lpane == XM_FAILURE)
2656 {
92280f67 2657 XMenuDestroy (FRAME_X_DISPLAY (f), menu);
78589e07
RS
2658 *error = "Can't create pane";
2659 return Qnil;
2660 }
2661 i += MENU_ITEMS_PANE_LENGTH;
4e8d3549
RS
2662
2663 /* Find the width of the widest item in this pane. */
2664 maxwidth = 0;
2665 j = i;
2666 while (j < menu_items_used)
2667 {
2668 Lisp_Object item;
2669 item = XVECTOR (menu_items)->contents[j];
2670 if (EQ (item, Qt))
2671 break;
2672 if (NILP (item))
2673 {
2674 j++;
2675 continue;
2676 }
fc932ac6 2677 width = STRING_BYTES (XSTRING (item));
4e8d3549
RS
2678 if (width > maxwidth)
2679 maxwidth = width;
2680
2681 j += MENU_ITEMS_ITEM_LENGTH;
2682 }
dcfdbac7 2683 }
fcaa7665
RS
2684 /* Ignore a nil in the item list.
2685 It's meaningful only for dialog boxes. */
2686 else if (EQ (XVECTOR (menu_items)->contents[i], Qquote))
2687 i += 1;
78589e07 2688 else
dcfdbac7 2689 {
78589e07 2690 /* Create a new item within current pane. */
3e703b25 2691 Lisp_Object item_name, enable, descrip, help;
4e8d3549 2692 unsigned char *item_data;
3e703b25 2693 char *help_string;
78589e07
RS
2694
2695 item_name = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_NAME];
2696 enable = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_ENABLE];
2697 descrip
2698 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_EQUIV_KEY];
3e703b25
GM
2699 help = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_HELP];
2700 help_string = STRINGP (help) ? XSTRING (help)->data : NULL;
2701
78589e07 2702 if (!NILP (descrip))
4e8d3549 2703 {
fc932ac6 2704 int gap = maxwidth - STRING_BYTES (XSTRING (item_name));
4e8d3549
RS
2705#ifdef C_ALLOCA
2706 Lisp_Object spacer;
2707 spacer = Fmake_string (make_number (gap), make_number (' '));
2708 item_name = concat2 (item_name, spacer);
2709 item_name = concat2 (item_name, descrip);
2710 item_data = XSTRING (item_name)->data;
2711#else
2712 /* if alloca is fast, use that to make the space,
2713 to reduce gc needs. */
2714 item_data
2715 = (unsigned char *) alloca (maxwidth
fc932ac6 2716 + STRING_BYTES (XSTRING (descrip)) + 1);
4e8d3549 2717 bcopy (XSTRING (item_name)->data, item_data,
fc932ac6 2718 STRING_BYTES (XSTRING (item_name)));
4e8d3549
RS
2719 for (j = XSTRING (item_name)->size; j < maxwidth; j++)
2720 item_data[j] = ' ';
2721 bcopy (XSTRING (descrip)->data, item_data + j,
fc932ac6
RS
2722 STRING_BYTES (XSTRING (descrip)));
2723 item_data[j + STRING_BYTES (XSTRING (descrip))] = 0;
4e8d3549
RS
2724#endif
2725 }
2726 else
2727 item_data = XSTRING (item_name)->data;
78589e07 2728
92280f67
RS
2729 if (XMenuAddSelection (FRAME_X_DISPLAY (f),
2730 menu, lpane, 0, item_data,
3e703b25 2731 !NILP (enable), help_string)
dcfdbac7
JB
2732 == XM_FAILURE)
2733 {
92280f67 2734 XMenuDestroy (FRAME_X_DISPLAY (f), menu);
dcfdbac7 2735 *error = "Can't add selection to menu";
78589e07 2736 return Qnil;
dcfdbac7 2737 }
78589e07 2738 i += MENU_ITEMS_ITEM_LENGTH;
dcfdbac7
JB
2739 }
2740 }
4e8d3549 2741
78589e07 2742 /* All set and ready to fly. */
92280f67 2743 XMenuRecompute (FRAME_X_DISPLAY (f), menu);
63685b9d
GM
2744 dispwidth = DisplayWidth (FRAME_X_DISPLAY (f), FRAME_X_SCREEN_NUMBER (f));
2745 dispheight = DisplayHeight (FRAME_X_DISPLAY (f), FRAME_X_SCREEN_NUMBER (f));
78589e07
RS
2746 x = min (x, dispwidth);
2747 y = min (y, dispheight);
2748 x = max (x, 1);
2749 y = max (y, 1);
92280f67 2750 XMenuLocate (FRAME_X_DISPLAY (f), menu, 0, 0, x, y,
dcfdbac7
JB
2751 &ulx, &uly, &width, &height);
2752 if (ulx+width > dispwidth)
2753 {
78589e07 2754 x -= (ulx + width) - dispwidth;
dcfdbac7
JB
2755 ulx = dispwidth - width;
2756 }
2757 if (uly+height > dispheight)
2758 {
78589e07 2759 y -= (uly + height) - dispheight;
dcfdbac7
JB
2760 uly = dispheight - height;
2761 }
78589e07
RS
2762 if (ulx < 0) x -= ulx;
2763 if (uly < 0) y -= uly;
121e4555
KH
2764
2765 XMenuSetAEQ (menu, TRUE);
78589e07
RS
2766 XMenuSetFreeze (menu, TRUE);
2767 pane = selidx = 0;
3e703b25
GM
2768
2769 /* Help display under X won't work because XMenuActivate contains
2770 a loop that doesn't give Emacs a chance to process it. */
2771 menu_help_frame = f;
92280f67 2772 status = XMenuActivate (FRAME_X_DISPLAY (f), menu, &pane, &selidx,
3e703b25
GM
2773 x, y, ButtonReleaseMask, &datap,
2774 menu_help_callback);
a352a815
RS
2775
2776
f1df80a8 2777#ifdef HAVE_X_WINDOWS
a352a815
RS
2778 /* Assume the mouse has moved out of the X window.
2779 If it has actually moved in, we will get an EnterNotify. */
29e460bd 2780 x_mouse_leave (FRAME_X_DISPLAY_INFO (f));
f1df80a8 2781#endif
a352a815 2782
dcfdbac7
JB
2783 switch (status)
2784 {
2785 case XM_SUCCESS:
2786#ifdef XDEBUG
2787 fprintf (stderr, "pane= %d line = %d\n", panes, selidx);
2788#endif
fa6d54d9 2789
78589e07
RS
2790 /* Find the item number SELIDX in pane number PANE. */
2791 i = 0;
2792 while (i < menu_items_used)
fa6d54d9 2793 {
78589e07 2794 if (EQ (XVECTOR (menu_items)->contents[i], Qt))
088831f6 2795 {
78589e07
RS
2796 if (pane == 0)
2797 pane_prefix
2798 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_PANE_PREFIX];
2799 pane--;
2800 i += MENU_ITEMS_PANE_LENGTH;
088831f6 2801 }
78589e07 2802 else
ab6ee1a0 2803 {
78589e07 2804 if (pane == -1)
ab6ee1a0 2805 {
78589e07 2806 if (selidx == 0)
ab6ee1a0 2807 {
78589e07
RS
2808 entry
2809 = XVECTOR (menu_items)->contents[i + MENU_ITEMS_ITEM_VALUE];
2810 if (keymaps != 0)
ab6ee1a0 2811 {
78589e07
RS
2812 entry = Fcons (entry, Qnil);
2813 if (!NILP (pane_prefix))
2814 entry = Fcons (pane_prefix, entry);
ab6ee1a0 2815 }
78589e07 2816 break;
ab6ee1a0 2817 }
78589e07 2818 selidx--;
ab6ee1a0 2819 }
78589e07 2820 i += MENU_ITEMS_ITEM_LENGTH;
ab6ee1a0
RS
2821 }
2822 }
78589e07 2823 break;
dcfdbac7 2824
78589e07 2825 case XM_FAILURE:
78589e07
RS
2826 *error = "Can't activate menu";
2827 case XM_IA_SELECT:
2828 case XM_NO_SELECT:
2829 entry = Qnil;
2830 break;
dcfdbac7 2831 }
92280f67 2832 XMenuDestroy (FRAME_X_DISPLAY (f), menu);
a5285df3 2833
87485d6f 2834#ifdef HAVE_X_WINDOWS
a5285df3
RS
2835 /* State that no mouse buttons are now held.
2836 (The oldXMenu code doesn't track this info for us.)
2837 That is not necessarily true, but the fiction leads to reasonable
2838 results, and it is a pain to ask which are actually held now. */
e9a79fb2 2839 FRAME_X_DISPLAY_INFO (f)->grabbed = 0;
87485d6f 2840#endif
a5285df3 2841
78589e07 2842 return entry;
dcfdbac7 2843}
4dedbfe0 2844
78589e07 2845#endif /* not USE_X_TOOLKIT */
1e659e4c
RS
2846
2847#endif /* HAVE_MENUS */
088831f6 2848\f
dfcf069d 2849void
78589e07 2850syms_of_xmenu ()
dcfdbac7 2851{
78589e07
RS
2852 staticpro (&menu_items);
2853 menu_items = Qnil;
dcfdbac7 2854
0314aacb
RS
2855 Qdebug_on_next_call = intern ("debug-on-next-call");
2856 staticpro (&Qdebug_on_next_call);
2857
7ee72033
MB
2858 DEFVAR_LISP ("menu-updating-frame", &Vmenu_updating_frame,
2859 doc: /* Frame for which we are updating a menu.
228299fa 2860The enable predicate for a menu command should check this variable. */);
bfc524bc
RS
2861 Vmenu_updating_frame = Qnil;
2862
8ed87156 2863#ifdef USE_X_TOOLKIT
4dedbfe0 2864 widget_id_tick = (1<<16);
88766961 2865 next_menubar_widget_id = 1;
8ed87156
RS
2866#endif
2867
78589e07 2868 defsubr (&Sx_popup_menu);
1e659e4c 2869#ifdef HAVE_MENUS
165e1749 2870 defsubr (&Sx_popup_dialog);
1e659e4c 2871#endif
dcfdbac7 2872}