1 /* Menu support for GNU Emacs on the for Mac OS.
2 Copyright (C) 2000, 2001, 2002 Free Software Foundation, Inc.
4 This file is part of GNU Emacs.
6 GNU Emacs is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2, or (at your option)
11 GNU Emacs is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with GNU Emacs; see the file COPYING. If not, write to
18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 Boston, MA 02111-1307, USA. */
21 /* Contributed by Andrew Choi (akochoi@mac.com). */
28 #include "termhooks.h"
33 #include "blockinput.h"
45 /* Macros max and min defined in lisp.h conflict with those in
46 precompiled header Carbon.h. */
50 #include <Carbon/Carbon.h>
52 #define Z (current_buffer->text->z)
54 #define free unexec_free
56 #define malloc unexec_malloc
58 #define realloc unexec_realloc
60 #define min(a, b) ((a) < (b) ? (a) : (b))
62 #define max(a, b) ((a) > (b) ? (a) : (b))
64 #define init_process emacs_init_process
65 #else /* not MAC_OSX */
68 #include <QuickDraw.h>
69 #include <ToolUtils.h>
74 #if defined (__MRC__) || (__MSL__ >= 0x6000)
75 #include <ControlDefinitions.h>
77 #endif /* not MAC_OSX */
79 /* This may include sys/types.h, and that somehow loses
80 if this is not done before the other system files. */
83 /* Load sys/types.h if not already loaded.
84 In some systems loading it twice is suicidal. */
86 #include <sys/types.h>
89 #include "dispextern.h"
91 #define POPUP_SUBMENU_ID 235
92 #define MIN_MENU_ID 256
93 #define MIN_SUBMENU_ID 1
95 #define DIALOG_WINDOW_RESOURCE 130
97 #define HAVE_DIALOGS 1
99 #undef HAVE_MULTILINGUAL_MENU
100 #undef HAVE_DIALOGS /* TODO: Implement native dialogs. */
102 /******************************************************************/
103 /* Definitions copied from lwlib.h */
105 typedef void * XtPointer
;
114 /* This structure is based on the one in ../lwlib/lwlib.h, modified
116 typedef struct _widget_value
120 /* value (meaning depend on widget type) */
122 /* keyboard equivalent. no implications for XtTranslations */
124 /* Help string or nil if none.
125 GC finds this string through the frame's menu_bar_vector
126 or through menu_items. */
128 /* true if enabled */
130 /* true if selected */
132 /* The type of a button. */
133 enum button_type button_type
;
134 /* true if menu title */
137 /* true if was edited (maintained by get_value) */
139 /* true if has changed (maintained by lw library) */
141 /* true if this widget itself has changed,
142 but not counting the other widgets found in the `next' field. */
143 change_type this_one_change
;
145 /* Contents of the sub-widgets, also selected slot for checkbox */
146 struct _widget_value
* contents
;
147 /* data passed to callback */
149 /* next one in the list */
150 struct _widget_value
* next
;
152 /* slot for the toolkit dependent part. Always initialize to NULL. */
154 /* tell us if we should free the toolkit data slot when freeing the
155 widget_value itself. */
156 Boolean free_toolkit_data
;
158 /* we resource the widget_value structures; this points to the next
159 one on the free list if this one has been deallocated.
161 struct _widget_value
*free_list
;
165 /* Assumed by other routines to zero area returned. */
166 #define malloc_widget_value() (void *)memset (xmalloc (sizeof (widget_value)),\
167 0, (sizeof (widget_value)))
168 #define free_widget_value(wv) xfree (wv)
170 /******************************************************************/
177 Lisp_Object Vmenu_updating_frame
;
179 Lisp_Object Qdebug_on_next_call
;
181 extern Lisp_Object Qmenu_bar
;
183 extern Lisp_Object QCtoggle
, QCradio
;
185 extern Lisp_Object Voverriding_local_map
;
186 extern Lisp_Object Voverriding_local_map_menu_flag
;
188 extern Lisp_Object Qoverriding_local_map
, Qoverriding_terminal_local_map
;
190 extern Lisp_Object Qmenu_bar_update_hook
;
192 void set_frame_menubar ();
194 static void push_menu_item
P_ ((Lisp_Object
, Lisp_Object
, Lisp_Object
,
195 Lisp_Object
, Lisp_Object
, Lisp_Object
,
196 Lisp_Object
, Lisp_Object
));
198 static Lisp_Object
mac_dialog_show ();
200 static Lisp_Object
mac_menu_show ();
202 static void keymap_panes ();
203 static void single_keymap_panes ();
204 static void single_menu_item ();
205 static void list_of_panes ();
206 static void list_of_items ();
208 static void fill_submenu (MenuHandle
, widget_value
*, int);
209 static void fill_menubar (widget_value
*);
212 /* This holds a Lisp vector that holds the results of decoding
213 the keymaps or alist-of-alists that specify a menu.
215 It describes the panes and items within the panes.
217 Each pane is described by 3 elements in the vector:
218 t, the pane name, the pane's prefix key.
219 Then follow the pane's items, with 5 elements per item:
220 the item string, the enable flag, the item's value,
221 the definition, and the equivalent keyboard key's description string.
223 In some cases, multiple levels of menus may be described.
224 A single vector slot containing nil indicates the start of a submenu.
225 A single vector slot containing lambda indicates the end of a submenu.
226 The submenu follows a menu item which is the way to reach the submenu.
228 A single vector slot containing quote indicates that the
229 following items should appear on the right of a dialog box.
231 Using a Lisp vector to hold this information while we decode it
232 takes care of protecting all the data from GC. */
234 #define MENU_ITEMS_PANE_NAME 1
235 #define MENU_ITEMS_PANE_PREFIX 2
236 #define MENU_ITEMS_PANE_LENGTH 3
240 MENU_ITEMS_ITEM_NAME
= 0,
241 MENU_ITEMS_ITEM_ENABLE
,
242 MENU_ITEMS_ITEM_VALUE
,
243 MENU_ITEMS_ITEM_EQUIV_KEY
,
244 MENU_ITEMS_ITEM_DEFINITION
,
245 MENU_ITEMS_ITEM_TYPE
,
246 MENU_ITEMS_ITEM_SELECTED
,
247 MENU_ITEMS_ITEM_HELP
,
248 MENU_ITEMS_ITEM_LENGTH
251 static Lisp_Object menu_items
;
253 /* Number of slots currently allocated in menu_items. */
254 static int menu_items_allocated
;
256 /* This is the index in menu_items of the first empty slot. */
257 static int menu_items_used
;
259 /* The number of panes currently recorded in menu_items,
260 excluding those within submenus. */
261 static int menu_items_n_panes
;
263 /* Current depth within submenus. */
264 static int menu_items_submenu_depth
;
266 /* Flag which when set indicates a dialog or menu has been posted by
267 Xt on behalf of one of the widget sets. */
268 static int popup_activated_flag
;
270 static int next_menubar_widget_id
;
272 /* This is set nonzero after the user activates the menu bar, and set
273 to zero again after the menu bars are redisplayed by prepare_menu_bar.
274 While it is nonzero, all calls to set_frame_menubar go deep.
276 I don't understand why this is needed, but it does seem to be
277 needed on Motif, according to Marcus Daniels <marcus@sysc.pdx.edu>. */
279 int pending_menu_activation
;
281 /* Initialize the menu_items structure if we haven't already done so.
282 Also mark it as currently empty. */
287 if (NILP (menu_items
))
289 menu_items_allocated
= 60;
290 menu_items
= Fmake_vector (make_number (menu_items_allocated
), Qnil
);
294 menu_items_n_panes
= 0;
295 menu_items_submenu_depth
= 0;
298 /* Call at the end of generating the data in menu_items.
299 This fills in the number of items in the last pane. */
306 /* Call when finished using the data for the current menu
310 discard_menu_items ()
312 /* Free the structure if it is especially large.
313 Otherwise, hold on to it, to save time. */
314 if (menu_items_allocated
> 200)
317 menu_items_allocated
= 0;
321 /* Make the menu_items vector twice as large. */
327 int old_size
= menu_items_allocated
;
330 menu_items_allocated
*= 2;
331 menu_items
= Fmake_vector (make_number (menu_items_allocated
), Qnil
);
332 bcopy (XVECTOR (old
)->contents
, XVECTOR (menu_items
)->contents
,
333 old_size
* sizeof (Lisp_Object
));
336 /* Begin a submenu. */
339 push_submenu_start ()
341 if (menu_items_used
+ 1 > menu_items_allocated
)
344 XVECTOR (menu_items
)->contents
[menu_items_used
++] = Qnil
;
345 menu_items_submenu_depth
++;
353 if (menu_items_used
+ 1 > menu_items_allocated
)
356 XVECTOR (menu_items
)->contents
[menu_items_used
++] = Qlambda
;
357 menu_items_submenu_depth
--;
360 /* Indicate boundary between left and right. */
363 push_left_right_boundary ()
365 if (menu_items_used
+ 1 > menu_items_allocated
)
368 XVECTOR (menu_items
)->contents
[menu_items_used
++] = Qquote
;
371 /* Start a new menu pane in menu_items.
372 NAME is the pane name. PREFIX_VEC is a prefix key for this pane. */
375 push_menu_pane (name
, prefix_vec
)
376 Lisp_Object name
, prefix_vec
;
378 if (menu_items_used
+ MENU_ITEMS_PANE_LENGTH
> menu_items_allocated
)
381 if (menu_items_submenu_depth
== 0)
382 menu_items_n_panes
++;
383 XVECTOR (menu_items
)->contents
[menu_items_used
++] = Qt
;
384 XVECTOR (menu_items
)->contents
[menu_items_used
++] = name
;
385 XVECTOR (menu_items
)->contents
[menu_items_used
++] = prefix_vec
;
388 /* Push one menu item into the current pane. NAME is the string to
389 display. ENABLE if non-nil means this item can be selected. KEY
390 is the key generated by choosing this item, or nil if this item
391 doesn't really have a definition. DEF is the definition of this
392 item. EQUIV is the textual description of the keyboard equivalent
393 for this item (or nil if none). TYPE is the type of this menu
394 item, one of nil, `toggle' or `radio'. */
397 push_menu_item (name
, enable
, key
, def
, equiv
, type
, selected
, help
)
398 Lisp_Object name
, enable
, key
, def
, equiv
, type
, selected
, help
;
400 if (menu_items_used
+ MENU_ITEMS_ITEM_LENGTH
> menu_items_allocated
)
403 XVECTOR (menu_items
)->contents
[menu_items_used
++] = name
;
404 XVECTOR (menu_items
)->contents
[menu_items_used
++] = enable
;
405 XVECTOR (menu_items
)->contents
[menu_items_used
++] = key
;
406 XVECTOR (menu_items
)->contents
[menu_items_used
++] = equiv
;
407 XVECTOR (menu_items
)->contents
[menu_items_used
++] = def
;
408 XVECTOR (menu_items
)->contents
[menu_items_used
++] = type
;
409 XVECTOR (menu_items
)->contents
[menu_items_used
++] = selected
;
410 XVECTOR (menu_items
)->contents
[menu_items_used
++] = help
;
413 /* Look through KEYMAPS, a vector of keymaps that is NMAPS long,
414 and generate menu panes for them in menu_items.
415 If NOTREAL is nonzero,
416 don't bother really computing whether an item is enabled. */
419 keymap_panes (keymaps
, nmaps
, notreal
)
420 Lisp_Object
*keymaps
;
428 /* Loop over the given keymaps, making a pane for each map.
429 But don't make a pane that is empty--ignore that map instead.
430 P is the number of panes we have made so far. */
431 for (mapno
= 0; mapno
< nmaps
; mapno
++)
432 single_keymap_panes (keymaps
[mapno
],
433 Fkeymap_prompt (keymaps
[mapno
]), Qnil
, notreal
, 10);
435 finish_menu_items ();
438 /* This is a recursive subroutine of keymap_panes.
439 It handles one keymap, KEYMAP.
440 The other arguments are passed along
441 or point to local variables of the previous function.
442 If NOTREAL is nonzero, only check for equivalent key bindings, don't
443 evaluate expressions in menu items and don't make any menu.
445 If we encounter submenus deeper than MAXDEPTH levels, ignore them. */
448 single_keymap_panes (keymap
, pane_name
, prefix
, notreal
, maxdepth
)
450 Lisp_Object pane_name
;
455 Lisp_Object pending_maps
= Qnil
;
456 Lisp_Object tail
, item
;
457 struct gcpro gcpro1
, gcpro2
;
462 push_menu_pane (pane_name
, prefix
);
464 for (tail
= keymap
; CONSP (tail
); tail
= XCDR (tail
))
466 GCPRO2 (keymap
, pending_maps
);
467 /* Look at each key binding, and if it is a menu item add it
471 single_menu_item (XCAR (item
), XCDR (item
),
472 &pending_maps
, notreal
, maxdepth
);
473 else if (VECTORP (item
))
475 /* Loop over the char values represented in the vector. */
476 int len
= XVECTOR (item
)->size
;
478 for (c
= 0; c
< len
; c
++)
480 Lisp_Object character
;
481 XSETFASTINT (character
, c
);
482 single_menu_item (character
, XVECTOR (item
)->contents
[c
],
483 &pending_maps
, notreal
, maxdepth
);
489 /* Process now any submenus which want to be panes at this level. */
490 while (!NILP (pending_maps
))
492 Lisp_Object elt
, eltcdr
, string
;
493 elt
= Fcar (pending_maps
);
495 string
= XCAR (eltcdr
);
496 /* We no longer discard the @ from the beginning of the string here.
497 Instead, we do this in mac_menu_show. */
498 single_keymap_panes (Fcar (elt
), string
,
499 XCDR (eltcdr
), notreal
, maxdepth
- 1);
500 pending_maps
= Fcdr (pending_maps
);
504 /* This is a subroutine of single_keymap_panes that handles one
506 KEY is a key in a keymap and ITEM is its binding.
507 PENDING_MAPS_PTR points to a list of keymaps waiting to be made into
509 If NOTREAL is nonzero, only check for equivalent key bindings, don't
510 evaluate expressions in menu items and don't make any menu.
511 If we encounter submenus deeper than MAXDEPTH levels, ignore them. */
514 single_menu_item (key
, item
, pending_maps_ptr
, notreal
, maxdepth
)
515 Lisp_Object key
, item
;
516 Lisp_Object
*pending_maps_ptr
;
517 int maxdepth
, notreal
;
519 Lisp_Object map
, item_string
, enabled
;
520 struct gcpro gcpro1
, gcpro2
;
523 /* Parse the menu item and leave the result in item_properties. */
525 res
= parse_menu_item (item
, notreal
, 0);
528 return; /* Not a menu item. */
530 map
= XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_MAP
];
534 /* We don't want to make a menu, just traverse the keymaps to
535 precompute equivalent key bindings. */
537 single_keymap_panes (map
, Qnil
, key
, 1, maxdepth
- 1);
541 enabled
= XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_ENABLE
];
542 item_string
= XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_NAME
];
544 if (!NILP (map
) && SREF (item_string
, 0) == '@')
547 /* An enabled separate pane. Remember this to handle it later. */
548 *pending_maps_ptr
= Fcons (Fcons (map
, Fcons (item_string
, key
)),
553 push_menu_item (item_string
, enabled
, key
,
554 XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_DEF
],
555 XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_KEYEQ
],
556 XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_TYPE
],
557 XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_SELECTED
],
558 XVECTOR (item_properties
)->contents
[ITEM_PROPERTY_HELP
]);
560 /* Display a submenu using the toolkit. */
561 if (! (NILP (map
) || NILP (enabled
)))
563 push_submenu_start ();
564 single_keymap_panes (map
, Qnil
, key
, 0, maxdepth
- 1);
569 /* Push all the panes and items of a menu described by the
570 alist-of-alists MENU.
571 This handles old-fashioned calls to x-popup-menu. */
581 for (tail
= menu
; !NILP (tail
); tail
= Fcdr (tail
))
583 Lisp_Object elt
, pane_name
, pane_data
;
585 pane_name
= Fcar (elt
);
586 CHECK_STRING (pane_name
);
587 push_menu_pane (pane_name
, Qnil
);
588 pane_data
= Fcdr (elt
);
589 CHECK_CONS (pane_data
);
590 list_of_items (pane_data
);
593 finish_menu_items ();
596 /* Push the items in a single pane defined by the alist PANE. */
602 Lisp_Object tail
, item
, item1
;
604 for (tail
= pane
; !NILP (tail
); tail
= Fcdr (tail
))
608 push_menu_item (item
, Qnil
, Qnil
, Qt
, Qnil
, Qnil
, Qnil
, Qnil
);
609 else if (NILP (item
))
610 push_left_right_boundary ();
615 CHECK_STRING (item1
);
616 push_menu_item (item1
, Qt
, Fcdr (item
), Qt
, Qnil
, Qnil
, Qnil
, Qnil
);
621 DEFUN ("x-popup-menu", Fx_popup_menu
, Sx_popup_menu
, 2, 2, 0,
622 doc
: /* Pop up a deck-of-cards menu and return user's selection.
623 POSITION is a position specification. This is either a mouse button
624 event or a list ((XOFFSET YOFFSET) WINDOW) where XOFFSET and YOFFSET
625 are positions in pixels from the top left corner of WINDOW's frame
626 \(WINDOW may be a frame object instead of a window). This controls the
627 position of the center of the first line in the first pane of the
628 menu, not the top left of the menu as a whole. If POSITION is t, it
629 means to use the current mouse position.
631 MENU is a specifier for a menu. For the simplest case, MENU is a keymap.
632 The menu items come from key bindings that have a menu string as well as
633 a definition; actually, the \"definition\" in such a key binding looks like
634 \(STRING . REAL-DEFINITION). To give the menu a title, put a string into
635 the keymap as a top-level element.
637 If REAL-DEFINITION is nil, that puts a nonselectable string in the menu.
638 Otherwise, REAL-DEFINITION should be a valid key binding definition.
640 You can also use a list of keymaps as MENU. Then each keymap makes a
641 separate pane. When MENU is a keymap or a list of keymaps, the return
642 value is a list of events.
644 Alternatively, you can specify a menu of multiple panes with a list of
645 the form (TITLE PANE1 PANE2...), where each pane is a list of
646 form (TITLE ITEM1 ITEM2...).
647 Each ITEM is normally a cons cell (STRING . VALUE); but a string can
648 appear as an item--that makes a nonselectable line in the menu.
649 With this form of menu, the return value is VALUE from the chosen item.
651 If POSITION is nil, don't display the menu at all, just precalculate the
652 cached information about equivalent key sequences. */)
654 Lisp_Object position
, menu
;
656 Lisp_Object keymap
, tem
;
657 int xpos
= 0, ypos
= 0;
660 Lisp_Object selection
;
662 Lisp_Object x
, y
, window
;
668 if (! NILP (position
))
672 /* Decode the first argument: find the window and the coordinates. */
673 if (EQ (position
, Qt
)
674 || (CONSP (position
) && (EQ (XCAR (position
), Qmenu_bar
)
675 || EQ (XCAR (position
), Qtool_bar
))))
677 /* Use the mouse's current position. */
678 FRAME_PTR new_f
= SELECTED_FRAME ();
679 Lisp_Object bar_window
;
680 enum scroll_bar_part part
;
683 if (mouse_position_hook
)
684 (*mouse_position_hook
) (&new_f
, 1, &bar_window
,
685 &part
, &x
, &y
, &time
);
687 XSETFRAME (window
, new_f
);
690 window
= selected_window
;
697 tem
= Fcar (position
);
700 window
= Fcar (Fcdr (position
));
702 y
= Fcar (Fcdr (tem
));
707 tem
= Fcar (Fcdr (position
)); /* EVENT_START (position) */
708 window
= Fcar (tem
); /* POSN_WINDOW (tem) */
709 tem
= Fcar (Fcdr (Fcdr (tem
))); /* POSN_WINDOW_POSN (tem) */
718 /* Decode where to put the menu. */
726 else if (WINDOWP (window
))
728 CHECK_LIVE_WINDOW (window
);
729 f
= XFRAME (WINDOW_FRAME (XWINDOW (window
)));
731 xpos
= (FONT_WIDTH (FRAME_FONT (f
))
732 * XFASTINT (XWINDOW (window
)->left
));
733 ypos
= (FRAME_LINE_HEIGHT (f
)
734 * XFASTINT (XWINDOW (window
)->top
));
737 /* ??? Not really clean; should be CHECK_WINDOW_OR_FRAME,
738 but I don't want to make one now. */
739 CHECK_WINDOW (window
);
744 XSETFRAME (Vmenu_updating_frame
, f
);
746 Vmenu_updating_frame
= Qnil
;
747 #endif /* HAVE_MENUS */
752 /* Decode the menu items from what was specified. */
754 keymap
= get_keymap (menu
, 0, 0);
757 /* We were given a keymap. Extract menu info from the keymap. */
760 /* Extract the detailed info to make one pane. */
761 keymap_panes (&menu
, 1, NILP (position
));
763 /* Search for a string appearing directly as an element of the keymap.
764 That string is the title of the menu. */
765 prompt
= Fkeymap_prompt (keymap
);
766 if (NILP (title
) && !NILP (prompt
))
769 /* Make that be the pane title of the first pane. */
770 if (!NILP (prompt
) && menu_items_n_panes
>= 0)
771 XVECTOR (menu_items
)->contents
[MENU_ITEMS_PANE_NAME
] = prompt
;
775 else if (CONSP (menu
) && KEYMAPP (XCAR (menu
)))
777 /* We were given a list of keymaps. */
778 int nmaps
= XFASTINT (Flength (menu
));
780 = (Lisp_Object
*) alloca (nmaps
* sizeof (Lisp_Object
));
785 /* The first keymap that has a prompt string
786 supplies the menu title. */
787 for (tem
= menu
, i
= 0; CONSP (tem
); tem
= Fcdr (tem
))
791 maps
[i
++] = keymap
= get_keymap (Fcar (tem
), 1, 0);
793 prompt
= Fkeymap_prompt (keymap
);
794 if (NILP (title
) && !NILP (prompt
))
798 /* Extract the detailed info to make one pane. */
799 keymap_panes (maps
, nmaps
, NILP (position
));
801 /* Make the title be the pane title of the first pane. */
802 if (!NILP (title
) && menu_items_n_panes
>= 0)
803 XVECTOR (menu_items
)->contents
[MENU_ITEMS_PANE_NAME
] = title
;
809 /* We were given an old-fashioned menu. */
811 CHECK_STRING (title
);
813 list_of_panes (Fcdr (menu
));
820 discard_menu_items ();
826 /* Display them in a menu. */
829 selection
= mac_menu_show (f
, xpos
, ypos
, for_click
,
830 keymaps
, title
, &error_name
);
833 discard_menu_items ();
836 #endif /* HAVE_MENUS */
838 if (error_name
) error (error_name
);
844 DEFUN ("x-popup-dialog", Fx_popup_dialog
, Sx_popup_dialog
, 2, 2, 0,
845 doc
: /* Pop up a dialog box and return user's selection.
846 POSITION specifies which frame to use.
847 This is normally a mouse button event or a window or frame.
848 If POSITION is t, it means to use the frame the mouse is on.
849 The dialog box appears in the middle of the specified frame.
851 CONTENTS specifies the alternatives to display in the dialog box.
852 It is a list of the form (TITLE ITEM1 ITEM2...).
853 Each ITEM is a cons cell (STRING . VALUE).
854 The return value is VALUE from the chosen item.
856 An ITEM may also be just a string--that makes a nonselectable item.
857 An ITEM may also be nil--that means to put all preceding items
858 on the left of the dialog box and all following items on the right.
859 \(By default, approximately half appear on each side.) */)
861 Lisp_Object position
, contents
;
868 /* Decode the first argument: find the window or frame to use. */
869 if (EQ (position
, Qt
)
870 || (CONSP (position
) && (EQ (XCAR (position
), Qmenu_bar
)
871 || EQ (XCAR (position
), Qtool_bar
))))
873 #if 0 /* Using the frame the mouse is on may not be right. */
874 /* Use the mouse's current position. */
875 FRAME_PTR new_f
= SELECTED_FRAME ();
876 Lisp_Object bar_window
;
877 enum scroll_bar_part part
;
881 (*mouse_position_hook
) (&new_f
, 1, &bar_window
, &part
, &x
, &y
, &time
);
884 XSETFRAME (window
, new_f
);
886 window
= selected_window
;
888 window
= selected_window
;
890 else if (CONSP (position
))
893 tem
= Fcar (position
);
895 window
= Fcar (Fcdr (position
));
898 tem
= Fcar (Fcdr (position
)); /* EVENT_START (position) */
899 window
= Fcar (tem
); /* POSN_WINDOW (tem) */
902 else if (WINDOWP (position
) || FRAMEP (position
))
907 /* Decode where to put the menu. */
911 else if (WINDOWP (window
))
913 CHECK_LIVE_WINDOW (window
);
914 f
= XFRAME (WINDOW_FRAME (XWINDOW (window
)));
917 /* ??? Not really clean; should be CHECK_WINDOW_OR_FRAME,
918 but I don't want to make one now. */
919 CHECK_WINDOW (window
);
922 /* Display a menu with these alternatives
923 in the middle of frame F. */
925 Lisp_Object x
, y
, frame
, newpos
;
926 XSETFRAME (frame
, f
);
927 XSETINT (x
, x_pixel_width (f
) / 2);
928 XSETINT (y
, x_pixel_height (f
) / 2);
929 newpos
= Fcons (Fcons (x
, Fcons (y
, Qnil
)), Fcons (frame
, Qnil
));
931 return Fx_popup_menu (newpos
,
932 Fcons (Fcar (contents
), Fcons (contents
, Qnil
)));
934 #else /* HAVE_DIALOGS */
938 Lisp_Object selection
;
940 /* Decode the dialog items from what was specified. */
941 title
= Fcar (contents
);
942 CHECK_STRING (title
);
944 list_of_panes (Fcons (contents
, Qnil
));
946 /* Display them in a dialog box. */
948 selection
= mac_dialog_show (f
, 0, title
, &error_name
);
951 discard_menu_items ();
953 if (error_name
) error (error_name
);
956 #endif /* HAVE_DIALOGS */
959 /* Activate the menu bar of frame F.
960 This is called from keyboard.c when it gets the
961 MENU_BAR_ACTIVATE_EVENT out of the Emacs event queue.
963 To activate the menu bar, we signal to the input thread that it can
964 return from the WM_INITMENU message, allowing the normal Windows
965 processing of the menus.
967 But first we recompute the menu bar contents (the whole tree).
969 This way we can safely execute Lisp code. */
972 x_activate_menubar (f
)
976 extern Point saved_menu_event_location
;
978 set_frame_menubar (f
, 0, 1);
981 menu_choice
= MenuSelect (saved_menu_event_location
);
982 do_menu_choice (menu_choice
);
987 /* This callback is called from the menu bar pulldown menu
988 when the user makes a selection.
989 Figure out what the user chose
990 and put the appropriate events into the keyboard buffer. */
993 menubar_selection_callback (FRAME_PTR f
, int client_data
)
995 Lisp_Object prefix
, entry
;
997 Lisp_Object
*subprefix_stack
;
998 int submenu_depth
= 0;
1004 subprefix_stack
= (Lisp_Object
*) alloca (f
->menu_bar_items_used
* sizeof (Lisp_Object
));
1005 vector
= f
->menu_bar_vector
;
1008 while (i
< f
->menu_bar_items_used
)
1010 if (EQ (XVECTOR (vector
)->contents
[i
], Qnil
))
1012 subprefix_stack
[submenu_depth
++] = prefix
;
1016 else if (EQ (XVECTOR (vector
)->contents
[i
], Qlambda
))
1018 prefix
= subprefix_stack
[--submenu_depth
];
1021 else if (EQ (XVECTOR (vector
)->contents
[i
], Qt
))
1023 prefix
= XVECTOR (vector
)->contents
[i
+ MENU_ITEMS_PANE_PREFIX
];
1024 i
+= MENU_ITEMS_PANE_LENGTH
;
1028 entry
= XVECTOR (vector
)->contents
[i
+ MENU_ITEMS_ITEM_VALUE
];
1029 /* The EMACS_INT cast avoids a warning. There's no problem
1030 as long as pointers have enough bits to hold small integers. */
1031 if ((int) (EMACS_INT
) client_data
== i
)
1034 struct input_event buf
;
1037 XSETFRAME (frame
, f
);
1038 buf
.kind
= MENU_BAR_EVENT
;
1039 buf
.frame_or_window
= frame
;
1041 kbd_buffer_store_event (&buf
);
1043 for (j
= 0; j
< submenu_depth
; j
++)
1044 if (!NILP (subprefix_stack
[j
]))
1046 buf
.kind
= MENU_BAR_EVENT
;
1047 buf
.frame_or_window
= frame
;
1048 buf
.arg
= subprefix_stack
[j
];
1049 kbd_buffer_store_event (&buf
);
1054 buf
.kind
= MENU_BAR_EVENT
;
1055 buf
.frame_or_window
= frame
;
1057 kbd_buffer_store_event (&buf
);
1060 buf
.kind
= MENU_BAR_EVENT
;
1061 buf
.frame_or_window
= frame
;
1063 kbd_buffer_store_event (&buf
);
1065 f
->output_data
.mac
->menu_command_in_progress
= 0;
1066 f
->output_data
.mac
->menubar_active
= 0;
1069 i
+= MENU_ITEMS_ITEM_LENGTH
;
1072 f
->output_data
.mac
->menu_command_in_progress
= 0;
1073 f
->output_data
.mac
->menubar_active
= 0;
1076 /* Allocate a widget_value, blocking input. */
1079 xmalloc_widget_value ()
1081 widget_value
*value
;
1084 value
= malloc_widget_value ();
1090 /* This recursively calls free_widget_value on the tree of widgets.
1091 It must free all data that was malloc'ed for these widget_values.
1092 In Emacs, many slots are pointers into the data of Lisp_Strings, and
1093 must be left alone. */
1096 free_menubar_widget_value_tree (wv
)
1101 wv
->name
= wv
->value
= wv
->key
= (char *) 0xDEADBEEF;
1103 if (wv
->contents
&& (wv
->contents
!= (widget_value
*)1))
1105 free_menubar_widget_value_tree (wv
->contents
);
1106 wv
->contents
= (widget_value
*) 0xDEADBEEF;
1110 free_menubar_widget_value_tree (wv
->next
);
1111 wv
->next
= (widget_value
*) 0xDEADBEEF;
1114 free_widget_value (wv
);
1118 /* Return a tree of widget_value structures for a menu bar item
1119 whose event type is ITEM_KEY (with string ITEM_NAME)
1120 and whose contents come from the list of keymaps MAPS. */
1122 static widget_value
*
1123 single_submenu (item_key
, item_name
, maps
)
1124 Lisp_Object item_key
, item_name
, maps
;
1126 widget_value
*wv
, *prev_wv
, *save_wv
, *first_wv
;
1128 int submenu_depth
= 0;
1131 Lisp_Object
*mapvec
;
1132 widget_value
**submenu_stack
;
1133 int previous_items
= menu_items_used
;
1134 int top_level_items
= 0;
1136 length
= Flength (maps
);
1137 len
= XINT (length
);
1139 /* Convert the list MAPS into a vector MAPVEC. */
1140 mapvec
= (Lisp_Object
*) alloca (len
* sizeof (Lisp_Object
));
1141 for (i
= 0; i
< len
; i
++)
1143 mapvec
[i
] = Fcar (maps
);
1147 menu_items_n_panes
= 0;
1149 /* Loop over the given keymaps, making a pane for each map.
1150 But don't make a pane that is empty--ignore that map instead. */
1151 for (i
= 0; i
< len
; i
++)
1153 if (SYMBOLP (mapvec
[i
])
1154 || (CONSP (mapvec
[i
]) && !KEYMAPP (mapvec
[i
])))
1156 /* Here we have a command at top level in the menu bar
1157 as opposed to a submenu. */
1158 top_level_items
= 1;
1159 push_menu_pane (Qnil
, Qnil
);
1160 push_menu_item (item_name
, Qt
, item_key
, mapvec
[i
],
1161 Qnil
, Qnil
, Qnil
, Qnil
);
1164 single_keymap_panes (mapvec
[i
], item_name
, item_key
, 0, 10);
1167 /* Create a tree of widget_value objects
1168 representing the panes and their items. */
1171 = (widget_value
**) alloca (menu_items_used
* sizeof (widget_value
*));
1172 wv
= xmalloc_widget_value ();
1176 wv
->button_type
= BUTTON_TYPE_NONE
;
1182 /* Loop over all panes and items made during this call
1183 and construct a tree of widget_value objects.
1184 Ignore the panes and items made by previous calls to
1185 single_submenu, even though those are also in menu_items. */
1187 while (i
< menu_items_used
)
1189 if (EQ (XVECTOR (menu_items
)->contents
[i
], Qnil
))
1191 submenu_stack
[submenu_depth
++] = save_wv
;
1196 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qlambda
))
1199 save_wv
= submenu_stack
[--submenu_depth
];
1202 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qt
)
1203 && submenu_depth
!= 0)
1204 i
+= MENU_ITEMS_PANE_LENGTH
;
1205 /* Ignore a nil in the item list.
1206 It's meaningful only for dialog boxes. */
1207 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qquote
))
1209 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qt
))
1211 /* Create a new pane. */
1212 Lisp_Object pane_name
, prefix
;
1215 pane_name
= XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_PANE_NAME
];
1216 prefix
= XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_PANE_PREFIX
];
1218 #ifndef HAVE_MULTILINGUAL_MENU
1219 if (STRINGP (pane_name
) && STRING_MULTIBYTE (pane_name
))
1221 pane_name
= ENCODE_SYSTEM (pane_name
);
1222 AREF (menu_items
, i
+ MENU_ITEMS_PANE_NAME
) = pane_name
;
1225 pane_string
= (NILP (pane_name
)
1226 ? "" : (char *) SDATA (pane_name
));
1227 /* If there is just one top-level pane, put all its items directly
1228 under the top-level menu. */
1229 if (menu_items_n_panes
== 1)
1232 /* If the pane has a meaningful name,
1233 make the pane a top-level menu item
1234 with its items as a submenu beneath it. */
1235 if (strcmp (pane_string
, ""))
1237 wv
= xmalloc_widget_value ();
1241 first_wv
->contents
= wv
;
1242 wv
->name
= pane_string
;
1243 /* Ignore the @ that means "separate pane".
1244 This is a kludge, but this isn't worth more time. */
1245 if (!NILP (prefix
) && wv
->name
[0] == '@')
1249 wv
->button_type
= BUTTON_TYPE_NONE
;
1254 i
+= MENU_ITEMS_PANE_LENGTH
;
1258 /* Create a new item within current pane. */
1259 Lisp_Object item_name
, enable
, descrip
, def
, type
, selected
;
1262 item_name
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_NAME
);
1263 enable
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_ENABLE
);
1264 descrip
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_EQUIV_KEY
);
1265 def
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_DEFINITION
);
1266 type
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_TYPE
);
1267 selected
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_SELECTED
);
1268 help
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_HELP
);
1270 #ifndef HAVE_MULTILINGUAL_MENU
1271 if (STRING_MULTIBYTE (item_name
))
1273 item_name
= ENCODE_SYSTEM (item_name
);
1274 AREF (menu_items
, i
+ MENU_ITEMS_ITEM_NAME
) = item_name
;
1277 if (STRINGP (descrip
) && STRING_MULTIBYTE (descrip
))
1279 descrip
= ENCODE_SYSTEM (descrip
);
1280 AREF (menu_items
, i
+ MENU_ITEMS_ITEM_EQUIV_KEY
) = descrip
;
1282 #endif /* not HAVE_MULTILINGUAL_MENU */
1284 wv
= xmalloc_widget_value ();
1288 save_wv
->contents
= wv
;
1290 wv
->name
= (char *) SDATA (item_name
);
1291 if (!NILP (descrip
))
1292 wv
->key
= (char *) SDATA (descrip
);
1294 /* The EMACS_INT cast avoids a warning. There's no problem
1295 as long as pointers have enough bits to hold small integers. */
1296 wv
->call_data
= (!NILP (def
) ? (void *) (EMACS_INT
) i
: 0);
1297 wv
->enabled
= !NILP (enable
);
1300 wv
->button_type
= BUTTON_TYPE_NONE
;
1301 else if (EQ (type
, QCradio
))
1302 wv
->button_type
= BUTTON_TYPE_RADIO
;
1303 else if (EQ (type
, QCtoggle
))
1304 wv
->button_type
= BUTTON_TYPE_TOGGLE
;
1308 wv
->selected
= !NILP (selected
);
1309 if (!STRINGP (help
))
1316 i
+= MENU_ITEMS_ITEM_LENGTH
;
1320 /* If we have just one "menu item"
1321 that was originally a button, return it by itself. */
1322 if (top_level_items
&& first_wv
->contents
&& first_wv
->contents
->next
== 0)
1324 wv
= first_wv
->contents
;
1325 free_widget_value (first_wv
);
1332 /* Set the contents of the menubar widgets of frame F.
1333 The argument FIRST_TIME is currently ignored;
1334 it is set the first time this is called, from initialize_frame_menubar. */
1337 set_frame_menubar (f
, first_time
, deep_p
)
1342 int menubar_widget
= f
->output_data
.mac
->menubar_widget
;
1344 widget_value
*wv
, *first_wv
, *prev_wv
= 0;
1347 /* We must not change the menubar when actually in use. */
1348 if (f
->output_data
.mac
->menubar_active
)
1351 XSETFRAME (Vmenu_updating_frame
, f
);
1353 if (! menubar_widget
)
1355 else if (pending_menu_activation
&& !deep_p
)
1358 wv
= xmalloc_widget_value ();
1359 wv
->name
= "menubar";
1362 wv
->button_type
= BUTTON_TYPE_NONE
;
1368 /* Make a widget-value tree representing the entire menu trees. */
1370 struct buffer
*prev
= current_buffer
;
1372 int specpdl_count
= SPECPDL_INDEX ();
1373 int previous_menu_items_used
= f
->menu_bar_items_used
;
1374 Lisp_Object
*previous_items
1375 = (Lisp_Object
*) alloca (previous_menu_items_used
1376 * sizeof (Lisp_Object
));
1378 /* If we are making a new widget, its contents are empty,
1379 do always reinitialize them. */
1380 if (! menubar_widget
)
1381 previous_menu_items_used
= 0;
1383 buffer
= XWINDOW (FRAME_SELECTED_WINDOW (f
))->buffer
;
1384 specbind (Qinhibit_quit
, Qt
);
1385 /* Don't let the debugger step into this code
1386 because it is not reentrant. */
1387 specbind (Qdebug_on_next_call
, Qnil
);
1389 record_unwind_protect (Fset_match_data
, Fmatch_data (Qnil
, Qnil
));
1390 if (NILP (Voverriding_local_map_menu_flag
))
1392 specbind (Qoverriding_terminal_local_map
, Qnil
);
1393 specbind (Qoverriding_local_map
, Qnil
);
1396 set_buffer_internal_1 (XBUFFER (buffer
));
1398 /* Run the Lucid hook. */
1399 safe_run_hooks (Qactivate_menubar_hook
);
1400 /* If it has changed current-menubar from previous value,
1401 really recompute the menubar from the value. */
1402 if (! NILP (Vlucid_menu_bar_dirty_flag
))
1403 call0 (Qrecompute_lucid_menubar
);
1404 safe_run_hooks (Qmenu_bar_update_hook
);
1405 FRAME_MENU_BAR_ITEMS (f
) = menu_bar_items (FRAME_MENU_BAR_ITEMS (f
));
1407 items
= FRAME_MENU_BAR_ITEMS (f
);
1409 inhibit_garbage_collection ();
1411 /* Save the frame's previous menu bar contents data. */
1412 if (previous_menu_items_used
)
1413 bcopy (XVECTOR (f
->menu_bar_vector
)->contents
, previous_items
,
1414 previous_menu_items_used
* sizeof (Lisp_Object
));
1416 /* Fill in the current menu bar contents. */
1417 menu_items
= f
->menu_bar_vector
;
1418 menu_items_allocated
= VECTORP (menu_items
) ? ASIZE (menu_items
) : 0;
1420 for (i
= 0; i
< XVECTOR (items
)->size
; i
+= 4)
1422 Lisp_Object key
, string
, maps
;
1424 key
= XVECTOR (items
)->contents
[i
];
1425 string
= XVECTOR (items
)->contents
[i
+ 1];
1426 maps
= XVECTOR (items
)->contents
[i
+ 2];
1430 wv
= single_submenu (key
, string
, maps
);
1434 first_wv
->contents
= wv
;
1435 /* Don't set wv->name here; GC during the loop might relocate it. */
1437 wv
->button_type
= BUTTON_TYPE_NONE
;
1441 finish_menu_items ();
1443 set_buffer_internal_1 (prev
);
1444 unbind_to (specpdl_count
, Qnil
);
1446 /* If there has been no change in the Lisp-level contents
1447 of the menu bar, skip redisplaying it. Just exit. */
1449 for (i
= 0; i
< previous_menu_items_used
; i
++)
1450 if (menu_items_used
== i
1451 || (!Fequal (previous_items
[i
], XVECTOR (menu_items
)->contents
[i
])))
1453 if (i
== menu_items_used
&& i
== previous_menu_items_used
&& i
!= 0)
1455 free_menubar_widget_value_tree (first_wv
);
1461 /* Now GC cannot happen during the lifetime of the widget_value,
1462 so it's safe to store data from a Lisp_String, as long as
1463 local copies are made when the actual menu is created.
1464 Windows takes care of this for normal string items, but
1465 not for owner-drawn items or additional item-info. */
1466 wv
= first_wv
->contents
;
1467 for (i
= 0; i
< XVECTOR (items
)->size
; i
+= 4)
1470 string
= XVECTOR (items
)->contents
[i
+ 1];
1473 wv
->name
= (char *) SDATA (string
);
1477 f
->menu_bar_vector
= menu_items
;
1478 f
->menu_bar_items_used
= menu_items_used
;
1483 /* Make a widget-value tree containing
1484 just the top level menu bar strings. */
1486 items
= FRAME_MENU_BAR_ITEMS (f
);
1487 for (i
= 0; i
< XVECTOR (items
)->size
; i
+= 4)
1491 string
= XVECTOR (items
)->contents
[i
+ 1];
1495 wv
= xmalloc_widget_value ();
1496 wv
->name
= (char *) SDATA (string
);
1499 wv
->button_type
= BUTTON_TYPE_NONE
;
1501 /* This prevents lwlib from assuming this
1502 menu item is really supposed to be empty. */
1503 /* The EMACS_INT cast avoids a warning.
1504 This value just has to be different from small integers. */
1505 wv
->call_data
= (void *) (EMACS_INT
) (-1);
1510 first_wv
->contents
= wv
;
1514 /* Forget what we thought we knew about what is in the
1515 detailed contents of the menu bar menus.
1516 Changing the top level always destroys the contents. */
1517 f
->menu_bar_items_used
= 0;
1520 /* Create or update the menu bar widget. */
1524 /* Non-null value to indicate menubar has already been "created". */
1525 f
->output_data
.mac
->menubar_widget
= 1;
1528 int i
= MIN_MENU_ID
;
1529 MenuHandle menu
= GetMenuHandle (i
);
1530 while (menu
!= NULL
)
1534 menu
= GetMenuHandle (++i
);
1538 menu
= GetMenuHandle (i
);
1539 while (menu
!= NULL
)
1543 menu
= GetMenuHandle (++i
);
1547 fill_menubar (first_wv
->contents
);
1551 free_menubar_widget_value_tree (first_wv
);
1556 /* Called from Fx_create_frame to create the initial menubar of a frame
1557 before it is mapped, so that the window is mapped with the menubar already
1558 there instead of us tacking it on later and thrashing the window after it
1562 initialize_frame_menubar (f
)
1565 /* This function is called before the first chance to redisplay
1566 the frame. It has to be, so the frame will have the right size. */
1567 FRAME_MENU_BAR_ITEMS (f
) = menu_bar_items (FRAME_MENU_BAR_ITEMS (f
));
1568 set_frame_menubar (f
, 1, 1);
1571 /* Get rid of the menu bar of frame F, and free its storage.
1572 This is used when deleting a frame, and when turning off the menu bar. */
1575 free_frame_menubar (f
)
1578 f
->output_data
.mac
->menubar_widget
= NULL
;
1582 /* mac_menu_show actually displays a menu using the panes and items in
1583 menu_items and returns the value selected from it; we assume input
1584 is blocked by the caller. */
1586 /* F is the frame the menu is for.
1587 X and Y are the frame-relative specified position,
1588 relative to the inside upper left corner of the frame F.
1589 FOR_CLICK is nonzero if this menu was invoked for a mouse click.
1590 KEYMAPS is 1 if this menu was specified with keymaps;
1591 in that case, we return a list containing the chosen item's value
1592 and perhaps also the pane's prefix.
1593 TITLE is the specified menu title.
1594 ERROR is a place to store an error message string in case of failure.
1595 (We return nil on failure, but the value doesn't actually matter.) */
1598 mac_menu_show (f
, x
, y
, for_click
, keymaps
, title
, error
)
1608 int menu_item_selection
;
1611 widget_value
*wv
, *save_wv
= 0, *first_wv
= 0, *prev_wv
= 0;
1612 widget_value
**submenu_stack
1613 = (widget_value
**) alloca (menu_items_used
* sizeof (widget_value
*));
1614 Lisp_Object
*subprefix_stack
1615 = (Lisp_Object
*) alloca (menu_items_used
* sizeof (Lisp_Object
));
1616 int submenu_depth
= 0;
1621 if (menu_items_used
<= MENU_ITEMS_PANE_LENGTH
)
1623 *error
= "Empty menu";
1627 /* Create a tree of widget_value objects
1628 representing the panes and their items. */
1629 wv
= xmalloc_widget_value ();
1633 wv
->button_type
= BUTTON_TYPE_NONE
;
1638 /* Loop over all panes and items, filling in the tree. */
1640 while (i
< menu_items_used
)
1642 if (EQ (XVECTOR (menu_items
)->contents
[i
], Qnil
))
1644 submenu_stack
[submenu_depth
++] = save_wv
;
1650 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qlambda
))
1653 save_wv
= submenu_stack
[--submenu_depth
];
1657 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qt
)
1658 && submenu_depth
!= 0)
1659 i
+= MENU_ITEMS_PANE_LENGTH
;
1660 /* Ignore a nil in the item list.
1661 It's meaningful only for dialog boxes. */
1662 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qquote
))
1664 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qt
))
1666 /* Create a new pane. */
1667 Lisp_Object pane_name
, prefix
;
1669 pane_name
= AREF (menu_items
, i
+ MENU_ITEMS_PANE_NAME
);
1670 prefix
= AREF (menu_items
, i
+ MENU_ITEMS_PANE_PREFIX
);
1671 #ifndef HAVE_MULTILINGUAL_MENU
1672 if (STRINGP (pane_name
) && STRING_MULTIBYTE (pane_name
))
1674 pane_name
= ENCODE_SYSTEM (pane_name
);
1675 AREF (menu_items
, i
+ MENU_ITEMS_PANE_NAME
) = pane_name
;
1678 pane_string
= (NILP (pane_name
)
1679 ? "" : (char *) SDATA (pane_name
));
1680 /* If there is just one top-level pane, put all its items directly
1681 under the top-level menu. */
1682 if (menu_items_n_panes
== 1)
1685 /* If the pane has a meaningful name,
1686 make the pane a top-level menu item
1687 with its items as a submenu beneath it. */
1688 if (!keymaps
&& strcmp (pane_string
, ""))
1690 wv
= xmalloc_widget_value ();
1694 first_wv
->contents
= wv
;
1695 wv
->name
= pane_string
;
1696 if (keymaps
&& !NILP (prefix
))
1700 wv
->button_type
= BUTTON_TYPE_NONE
;
1705 else if (first_pane
)
1711 i
+= MENU_ITEMS_PANE_LENGTH
;
1715 /* Create a new item within current pane. */
1716 Lisp_Object item_name
, enable
, descrip
, def
, type
, selected
, help
;
1718 item_name
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_NAME
);
1719 enable
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_ENABLE
);
1720 descrip
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_EQUIV_KEY
);
1721 def
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_DEFINITION
);
1722 type
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_TYPE
);
1723 selected
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_SELECTED
);
1724 help
= AREF (menu_items
, i
+ MENU_ITEMS_ITEM_HELP
);
1726 #ifndef HAVE_MULTILINGUAL_MENU
1727 if (STRINGP (item_name
) && STRING_MULTIBYTE (item_name
))
1729 item_name
= ENCODE_SYSTEM (item_name
);
1730 AREF (menu_items
, i
+ MENU_ITEMS_ITEM_NAME
) = item_name
;
1732 if (STRINGP (descrip
) && STRING_MULTIBYTE (descrip
))
1734 descrip
= ENCODE_SYSTEM (descrip
);
1735 AREF (menu_items
, i
+ MENU_ITEMS_ITEM_EQUIV_KEY
) = descrip
;
1737 #endif /* not HAVE_MULTILINGUAL_MENU */
1739 wv
= xmalloc_widget_value ();
1743 save_wv
->contents
= wv
;
1744 wv
->name
= (char *) SDATA (item_name
);
1745 if (!NILP (descrip
))
1746 wv
->key
= (char *) SDATA (descrip
);
1748 /* Use the contents index as call_data, since we are
1749 restricted to 16-bits. */
1750 wv
->call_data
= !NILP (def
) ? (void *) (EMACS_INT
) i
: 0;
1751 wv
->enabled
= !NILP (enable
);
1754 wv
->button_type
= BUTTON_TYPE_NONE
;
1755 else if (EQ (type
, QCtoggle
))
1756 wv
->button_type
= BUTTON_TYPE_TOGGLE
;
1757 else if (EQ (type
, QCradio
))
1758 wv
->button_type
= BUTTON_TYPE_RADIO
;
1762 wv
->selected
= !NILP (selected
);
1763 if (!STRINGP (help
))
1770 i
+= MENU_ITEMS_ITEM_LENGTH
;
1774 /* Deal with the title, if it is non-nil. */
1777 widget_value
*wv_title
= xmalloc_widget_value ();
1778 widget_value
*wv_sep
= xmalloc_widget_value ();
1780 /* Maybe replace this separator with a bitmap or owner-draw item
1781 so that it looks better. Having two separators looks odd. */
1782 wv_sep
->name
= "--";
1783 wv_sep
->next
= first_wv
->contents
;
1784 wv_sep
->help
= Qnil
;
1786 #ifndef HAVE_MULTILINGUAL_MENU
1787 if (STRING_MULTIBYTE (title
))
1788 title
= ENCODE_SYSTEM (title
);
1790 wv_title
->name
= (char *) SDATA (title
);
1791 wv_title
->enabled
= TRUE
;
1792 wv_title
->title
= TRUE
;
1793 wv_title
->button_type
= BUTTON_TYPE_NONE
;
1794 wv_title
->help
= Qnil
;
1795 wv_title
->next
= wv_sep
;
1796 first_wv
->contents
= wv_title
;
1799 /* Actually create the menu. */
1800 menu
= NewMenu (POPUP_SUBMENU_ID
, "\p");
1801 fill_submenu (menu
, first_wv
->contents
, 0);
1803 /* Adjust coordinates to be root-window-relative. */
1807 #if TARGET_API_MAC_CARBON
1808 SetPort (GetWindowPort (FRAME_MAC_WINDOW (f
)));
1810 SetPort (FRAME_MAC_WINDOW (f
));
1813 LocalToGlobal (&pos
);
1815 /* No selection has been chosen yet. */
1816 menu_item_selection
= 0;
1818 InsertMenu (menu
, -1);
1820 /* Display the menu. */
1821 menu_item_selection
= LoWord (PopUpMenuSelect (menu
, pos
.v
, pos
.h
, 0));
1823 DeleteMenu (POPUP_SUBMENU_ID
);
1826 /* Clean up extraneous mouse events which might have been generated
1828 discard_mouse_events ();
1831 /* Free the widget_value objects we used to specify the
1833 free_menubar_widget_value_tree (first_wv
);
1837 /* Find the selected item, and its pane, to return
1838 the proper value. */
1839 if (menu_item_selection
!= 0)
1841 Lisp_Object prefix
, entry
;
1844 prefix
= entry
= Qnil
;
1846 while (i
< menu_items_used
)
1848 if (EQ (XVECTOR (menu_items
)->contents
[i
], Qnil
))
1850 subprefix_stack
[submenu_depth
++] = prefix
;
1854 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qlambda
))
1856 prefix
= subprefix_stack
[--submenu_depth
];
1859 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qt
))
1862 = XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_PANE_PREFIX
];
1863 i
+= MENU_ITEMS_PANE_LENGTH
;
1866 /* Ignore a nil in the item list.
1867 It's meaningful only for dialog boxes. */
1868 else if (EQ (XVECTOR (menu_items
)->contents
[i
], Qquote
))
1873 = XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_ITEM_VALUE
];
1874 if (menu_item_selection
== j
)
1880 entry
= Fcons (entry
, Qnil
);
1882 entry
= Fcons (prefix
, entry
);
1883 for (j
= submenu_depth
- 1; j
>= 0; j
--)
1884 if (!NILP (subprefix_stack
[j
]))
1885 entry
= Fcons (subprefix_stack
[j
], entry
);
1889 i
+= MENU_ITEMS_ITEM_LENGTH
;
1900 /* Construct native Mac OS menubar based on widget_value tree. */
1903 mac_dialog (widget_value
*wv
)
1907 char **button_labels
;
1914 WindowPtr window_ptr
;
1917 EventRecord event_record
;
1919 int control_part_code
;
1922 dialog_name
= wv
->name
;
1923 nb_buttons
= dialog_name
[1] - '0';
1924 left_count
= nb_buttons
- (dialog_name
[4] - '0');
1925 button_labels
= (char **) alloca (sizeof (char *) * nb_buttons
);
1926 ref_cons
= (UInt32
*) alloca (sizeof (UInt32
) * nb_buttons
);
1929 prompt
= (char *) alloca (strlen (wv
->value
) + 1);
1930 strcpy (prompt
, wv
->value
);
1934 for (i
= 0; i
< nb_buttons
; i
++)
1936 button_labels
[i
] = wv
->value
;
1937 button_labels
[i
] = (char *) alloca (strlen (wv
->value
) + 1);
1938 strcpy (button_labels
[i
], wv
->value
);
1939 c2pstr (button_labels
[i
]);
1940 ref_cons
[i
] = (UInt32
) wv
->call_data
;
1944 window_ptr
= GetNewCWindow (DIALOG_WINDOW_RESOURCE
, NULL
, (WindowPtr
) -1);
1946 #if TARGET_API_MAC_CARBON
1947 SetPort (GetWindowPort (window_ptr
));
1949 SetPort (window_ptr
);
1953 /* Left and right margins in the dialog are 13 pixels each.*/
1955 /* Calculate width of dialog box: 8 pixels on each side of the text
1956 label in each button, 12 pixels between buttons. */
1957 for (i
= 0; i
< nb_buttons
; i
++)
1958 dialog_width
+= StringWidth (button_labels
[i
]) + 16 + 12;
1960 if (left_count
!= 0 && nb_buttons
- left_count
!= 0)
1963 dialog_width
= max (dialog_width
, StringWidth (prompt
) + 26);
1965 SizeWindow (window_ptr
, dialog_width
, 78, 0);
1966 ShowWindow (window_ptr
);
1968 #if TARGET_API_MAC_CARBON
1969 SetPort (GetWindowPort (window_ptr
));
1971 SetPort (window_ptr
);
1977 DrawString (prompt
);
1980 for (i
= 0; i
< nb_buttons
; i
++)
1982 int button_width
= StringWidth (button_labels
[i
]) + 16;
1983 SetRect (&rect
, left
, 45, left
+ button_width
, 65);
1984 ch
= NewControl (window_ptr
, &rect
, button_labels
[i
], 1, 0, 0, 0,
1985 kControlPushButtonProc
, ref_cons
[i
]);
1986 left
+= button_width
+ 12;
1987 if (i
== left_count
- 1)
1994 if (WaitNextEvent (mDownMask
, &event_record
, 10, NULL
))
1995 if (event_record
.what
== mouseDown
)
1997 part_code
= FindWindow (event_record
.where
, &window_ptr
);
1998 if (part_code
== inContent
)
2000 mouse
= event_record
.where
;
2001 GlobalToLocal (&mouse
);
2002 control_part_code
= FindControl (mouse
, window_ptr
, &ch
);
2003 if (control_part_code
== kControlButtonPart
)
2004 if (TrackControl (ch
, mouse
, NULL
))
2005 i
= GetControlReference (ch
);
2010 DisposeWindow (window_ptr
);
2015 static char * button_names
[] = {
2016 "button1", "button2", "button3", "button4", "button5",
2017 "button6", "button7", "button8", "button9", "button10" };
2020 mac_dialog_show (f
, keymaps
, title
, error
)
2026 int i
, nb_buttons
=0;
2027 char dialog_name
[6];
2028 int menu_item_selection
;
2030 widget_value
*wv
, *first_wv
= 0, *prev_wv
= 0;
2032 /* Number of elements seen so far, before boundary. */
2034 /* 1 means we've seen the boundary between left-hand elts and right-hand. */
2035 int boundary_seen
= 0;
2039 if (menu_items_n_panes
> 1)
2041 *error
= "Multiple panes in dialog box";
2045 /* Create a tree of widget_value objects
2046 representing the text label and buttons. */
2048 Lisp_Object pane_name
, prefix
;
2050 pane_name
= XVECTOR (menu_items
)->contents
[MENU_ITEMS_PANE_NAME
];
2051 prefix
= XVECTOR (menu_items
)->contents
[MENU_ITEMS_PANE_PREFIX
];
2052 pane_string
= (NILP (pane_name
)
2053 ? "" : (char *) SDATA (pane_name
));
2054 prev_wv
= xmalloc_widget_value ();
2055 prev_wv
->value
= pane_string
;
2056 if (keymaps
&& !NILP (prefix
))
2058 prev_wv
->enabled
= 1;
2059 prev_wv
->name
= "message";
2060 prev_wv
->help
= Qnil
;
2063 /* Loop over all panes and items, filling in the tree. */
2064 i
= MENU_ITEMS_PANE_LENGTH
;
2065 while (i
< menu_items_used
)
2068 /* Create a new item within current pane. */
2069 Lisp_Object item_name
, enable
, descrip
, help
;
2071 item_name
= XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_ITEM_NAME
];
2072 enable
= XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_ITEM_ENABLE
];
2074 = XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_ITEM_EQUIV_KEY
];
2075 help
= XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_ITEM_HELP
];
2077 if (NILP (item_name
))
2079 free_menubar_widget_value_tree (first_wv
);
2080 *error
= "Submenu in dialog items";
2083 if (EQ (item_name
, Qquote
))
2085 /* This is the boundary between left-side elts
2086 and right-side elts. Stop incrementing right_count. */
2091 if (nb_buttons
>= 9)
2093 free_menubar_widget_value_tree (first_wv
);
2094 *error
= "Too many dialog items";
2098 wv
= xmalloc_widget_value ();
2100 wv
->name
= (char *) button_names
[nb_buttons
];
2101 if (!NILP (descrip
))
2102 wv
->key
= (char *) SDATA (descrip
);
2103 wv
->value
= (char *) SDATA (item_name
);
2104 wv
->call_data
= (void *) i
;
2105 /* menu item is identified by its index in menu_items table */
2106 wv
->enabled
= !NILP (enable
);
2110 if (! boundary_seen
)
2114 i
+= MENU_ITEMS_ITEM_LENGTH
;
2117 /* If the boundary was not specified,
2118 by default put half on the left and half on the right. */
2119 if (! boundary_seen
)
2120 left_count
= nb_buttons
- nb_buttons
/ 2;
2122 wv
= xmalloc_widget_value ();
2123 wv
->name
= dialog_name
;
2126 /* Dialog boxes use a really stupid name encoding
2127 which specifies how many buttons to use
2128 and how many buttons are on the right.
2129 The Q means something also. */
2130 dialog_name
[0] = 'Q';
2131 dialog_name
[1] = '0' + nb_buttons
;
2132 dialog_name
[2] = 'B';
2133 dialog_name
[3] = 'R';
2134 /* Number of buttons to put on the right. */
2135 dialog_name
[4] = '0' + nb_buttons
- left_count
;
2137 wv
->contents
= first_wv
;
2141 /* Actually create the dialog. */
2143 menu_item_selection
= mac_dialog (first_wv
);
2145 menu_item_selection
= 0;
2148 /* Free the widget_value objects we used to specify the contents. */
2149 free_menubar_widget_value_tree (first_wv
);
2151 /* Find the selected item, and its pane, to return the proper
2153 if (menu_item_selection
!= 0)
2159 while (i
< menu_items_used
)
2163 if (EQ (XVECTOR (menu_items
)->contents
[i
], Qt
))
2166 = XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_PANE_PREFIX
];
2167 i
+= MENU_ITEMS_PANE_LENGTH
;
2172 = XVECTOR (menu_items
)->contents
[i
+ MENU_ITEMS_ITEM_VALUE
];
2173 if (menu_item_selection
== i
)
2177 entry
= Fcons (entry
, Qnil
);
2179 entry
= Fcons (prefix
, entry
);
2183 i
+= MENU_ITEMS_ITEM_LENGTH
;
2190 #endif /* HAVE_DIALOGS */
2193 /* Is this item a separator? */
2195 name_is_separator (name
)
2200 /* Check if name string consists of only dashes ('-'). */
2201 while (*name
== '-') name
++;
2202 /* Separators can also be of the form "--:TripleSuperMegaEtched"
2203 or "--deep-shadow". We don't implement them yet, se we just treat
2204 them like normal separators. */
2205 return (*name
== '\0' || start
+ 2 == name
);
2209 add_menu_item (MenuHandle menu
, widget_value
*wv
, int submenu
, int indent
,
2215 if (name_is_separator (wv
->name
))
2216 AppendMenu (menu
, "\p-");
2219 AppendMenu (menu
, "\pX");
2221 #if TARGET_API_MAC_CARBON
2222 pos
= CountMenuItems (menu
);
2224 pos
= CountMItems (menu
);
2227 strcpy (item_name
, "");
2228 for (i
= 0; i
< indent
; i
++)
2229 strcat (item_name
, " ");
2230 strcat (item_name
, wv
->name
);
2231 if (wv
->key
!= NULL
)
2233 strcat (item_name
, " ");
2234 strcat (item_name
, wv
->key
);
2237 SetMenuItemText (menu
, pos
, item_name
);
2239 if (wv
->enabled
&& !force_disable
)
2240 #if TARGET_API_MAC_CARBON
2241 EnableMenuItem (menu
, pos
);
2243 EnableItem (menu
, pos
);
2246 #if TARGET_API_MAC_CARBON
2247 DisableMenuItem (menu
, pos
);
2249 DisableItem (menu
, pos
);
2252 /* Draw radio buttons and tickboxes. */
2254 if (wv
->selected
&& (wv
->button_type
== BUTTON_TYPE_TOGGLE
||
2255 wv
->button_type
== BUTTON_TYPE_RADIO
))
2256 SetItemMark (menu
, pos
, checkMark
);
2258 SetItemMark (menu
, pos
, noMark
);
2262 SetMenuItemRefCon (menu
, pos
, (UInt32
) wv
->call_data
);
2264 if (submenu
!= NULL
)
2265 SetMenuItemHierarchicalID (menu
, pos
, submenu
);
2268 static int submenu_id
;
2270 /* Construct native Mac OS menubar based on widget_value tree. */
2273 fill_submenu (MenuHandle menu
, widget_value
*wv
, int indent
)
2275 for ( ; wv
!= NULL
; wv
= wv
->next
)
2278 add_menu_item (menu
, wv
, NULL
, indent
, 1);
2280 fill_submenu (menu
, wv
->contents
, indent
+ 1);
2283 add_menu_item (menu
, wv
, NULL
, indent
, 0);
2287 /* Construct native Mac OS menu based on widget_value tree. */
2290 fill_menu (MenuHandle menu
, widget_value
*wv
)
2292 for ( ; wv
!= NULL
; wv
= wv
->next
)
2295 MenuHandle submenu
= NewMenu (submenu_id
, "\pX");
2296 fill_submenu (submenu
, wv
->contents
, 0);
2297 InsertMenu (submenu
, -1);
2298 add_menu_item (menu
, wv
, submenu_id
, 0, 0);
2302 add_menu_item (menu
, wv
, NULL
, 0, 0);
2305 /* Construct native Mac OS menubar based on widget_value tree. */
2308 fill_menubar (widget_value
*wv
)
2312 submenu_id
= MIN_SUBMENU_ID
;
2314 for (id
= MIN_MENU_ID
; wv
!= NULL
; wv
= wv
->next
, id
++)
2319 strcpy (title
, wv
->name
);
2321 menu
= NewMenu (id
, title
);
2324 fill_menu (menu
, wv
->contents
);
2326 InsertMenu (menu
, 0);
2330 #endif /* HAVE_MENUS */
2336 staticpro (&menu_items
);
2339 Qdebug_on_next_call
= intern ("debug-on-next-call");
2340 staticpro (&Qdebug_on_next_call
);
2342 DEFVAR_LISP ("menu-updating-frame", &Vmenu_updating_frame
,
2343 doc
: /* Frame for which we are updating a menu.
2344 The enable predicate for a menu command should check this variable. */);
2345 Vmenu_updating_frame
= Qnil
;
2347 defsubr (&Sx_popup_menu
);
2349 defsubr (&Sx_popup_dialog
);