1997-01-20 00:38:22 +00:00
|
|
|
|
/* Menu support for GNU Emacs on the Microsoft W32 API.
|
2007-01-14 03:38:22 +00:00
|
|
|
|
Copyright (C) 1986, 1988, 1993, 1994, 1996, 1998, 1999, 2001, 2002,
|
2008-01-08 04:37:27 +00:00
|
|
|
|
2003, 2004, 2005, 2006, 2007, 2008
|
|
|
|
|
Free Software Foundation, Inc.
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
This file is part of GNU Emacs.
|
|
|
|
|
|
2008-05-14 07:50:26 +00:00
|
|
|
|
GNU Emacs is free software: you can redistribute it and/or modify
|
1995-11-07 07:52:28 +00:00
|
|
|
|
it under the terms of the GNU General Public License as published by
|
2008-05-14 07:50:26 +00:00
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
(at your option) any later version.
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
GNU Emacs is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2008-05-14 07:50:26 +00:00
|
|
|
|
along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
#include <config.h>
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
2007-05-15 23:02:24 +00:00
|
|
|
|
#include <signal.h>
|
1995-11-07 07:52:28 +00:00
|
|
|
|
#include <stdio.h>
|
2007-06-16 18:13:15 +00:00
|
|
|
|
#include <mbstring.h>
|
2007-05-15 23:02:24 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
#include "lisp.h"
|
2000-08-22 23:37:05 +00:00
|
|
|
|
#include "keyboard.h"
|
2001-10-08 11:12:23 +00:00
|
|
|
|
#include "keymap.h"
|
1995-11-07 07:52:28 +00:00
|
|
|
|
#include "frame.h"
|
2007-05-15 23:02:24 +00:00
|
|
|
|
#include "termhooks.h"
|
1995-11-07 07:52:28 +00:00
|
|
|
|
#include "window.h"
|
|
|
|
|
#include "blockinput.h"
|
1996-05-21 18:09:43 +00:00
|
|
|
|
#include "buffer.h"
|
2000-05-30 21:52:06 +00:00
|
|
|
|
#include "charset.h"
|
2007-06-13 21:05:52 +00:00
|
|
|
|
#include "character.h"
|
2000-05-30 21:52:06 +00:00
|
|
|
|
#include "coding.h"
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
/* This may include sys/types.h, and that somehow loses
|
|
|
|
|
if this is not done before the other system files. */
|
|
|
|
|
#include "w32term.h"
|
|
|
|
|
|
|
|
|
|
/* Load sys/types.h if not already loaded.
|
|
|
|
|
In some systems loading it twice is suicidal. */
|
|
|
|
|
#ifndef makedev
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include "dispextern.h"
|
|
|
|
|
|
2000-11-28 20:28:02 +00:00
|
|
|
|
#undef HAVE_DIALOGS /* TODO: Implement native dialogs. */
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/******************************************************************/
|
|
|
|
|
/* Definitions copied from lwlib.h */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
typedef void * XtPointer;
|
|
|
|
|
typedef char Boolean;
|
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
enum button_type
|
|
|
|
|
{
|
|
|
|
|
BUTTON_TYPE_NONE,
|
|
|
|
|
BUTTON_TYPE_TOGGLE,
|
|
|
|
|
BUTTON_TYPE_RADIO
|
|
|
|
|
};
|
1998-04-17 05:09:51 +00:00
|
|
|
|
|
2001-12-09 16:48:03 +00:00
|
|
|
|
/* This structure is based on the one in ../lwlib/lwlib.h, modified
|
|
|
|
|
for Windows. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
typedef struct _widget_value
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* name of widget */
|
2004-09-02 23:26:16 +00:00
|
|
|
|
Lisp_Object lname;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
char* name;
|
|
|
|
|
/* value (meaning depend on widget type) */
|
|
|
|
|
char* value;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
/* keyboard equivalent. no implications for XtTranslations */
|
2004-09-02 23:26:16 +00:00
|
|
|
|
Lisp_Object lkey;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
char* key;
|
2001-12-09 16:48:03 +00:00
|
|
|
|
/* Help string or nil if none.
|
|
|
|
|
GC finds this string through the frame's menu_bar_vector
|
|
|
|
|
or through menu_items. */
|
|
|
|
|
Lisp_Object help;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* true if enabled */
|
|
|
|
|
Boolean enabled;
|
|
|
|
|
/* true if selected */
|
|
|
|
|
Boolean selected;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
/* The type of a button. */
|
|
|
|
|
enum button_type button_type;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* true if menu title */
|
|
|
|
|
Boolean title;
|
|
|
|
|
#if 0
|
|
|
|
|
/* true if was edited (maintained by get_value) */
|
|
|
|
|
Boolean edited;
|
|
|
|
|
/* true if has changed (maintained by lw library) */
|
|
|
|
|
change_type change;
|
|
|
|
|
/* true if this widget itself has changed,
|
|
|
|
|
but not counting the other widgets found in the `next' field. */
|
|
|
|
|
change_type this_one_change;
|
|
|
|
|
#endif
|
|
|
|
|
/* Contents of the sub-widgets, also selected slot for checkbox */
|
|
|
|
|
struct _widget_value* contents;
|
|
|
|
|
/* data passed to callback */
|
|
|
|
|
XtPointer call_data;
|
|
|
|
|
/* next one in the list */
|
|
|
|
|
struct _widget_value* next;
|
|
|
|
|
#if 0
|
|
|
|
|
/* slot for the toolkit dependent part. Always initialize to NULL. */
|
|
|
|
|
void* toolkit_data;
|
|
|
|
|
/* tell us if we should free the toolkit data slot when freeing the
|
|
|
|
|
widget_value itself. */
|
|
|
|
|
Boolean free_toolkit_data;
|
|
|
|
|
|
|
|
|
|
/* we resource the widget_value structures; this points to the next
|
|
|
|
|
one on the free list if this one has been deallocated.
|
|
|
|
|
*/
|
|
|
|
|
struct _widget_value *free_list;
|
|
|
|
|
#endif
|
|
|
|
|
} widget_value;
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
/* Local memory management */
|
|
|
|
|
#define local_heap (GetProcessHeap ())
|
|
|
|
|
#define local_alloc(n) (HeapAlloc (local_heap, HEAP_ZERO_MEMORY, (n)))
|
|
|
|
|
#define local_free(p) (HeapFree (local_heap, 0, ((LPVOID) (p))))
|
|
|
|
|
|
|
|
|
|
#define malloc_widget_value() ((widget_value *) local_alloc (sizeof (widget_value)))
|
|
|
|
|
#define free_widget_value(wv) (local_free ((wv)))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/******************************************************************/
|
|
|
|
|
|
|
|
|
|
#ifndef TRUE
|
|
|
|
|
#define TRUE 1
|
|
|
|
|
#define FALSE 0
|
|
|
|
|
#endif /* no TRUE */
|
|
|
|
|
|
2007-02-22 22:49:41 +00:00
|
|
|
|
HMENU current_popup_menu;
|
2001-12-01 11:16:54 +00:00
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
void syms_of_w32menu ();
|
2002-12-18 06:16:28 +00:00
|
|
|
|
void globals_of_w32menu ();
|
2002-11-17 22:35:26 +00:00
|
|
|
|
|
|
|
|
|
typedef BOOL (WINAPI * GetMenuItemInfoA_Proc) (
|
|
|
|
|
IN HMENU,
|
|
|
|
|
IN UINT,
|
|
|
|
|
IN BOOL,
|
2004-09-02 23:26:16 +00:00
|
|
|
|
IN OUT LPMENUITEMINFOA);
|
2002-11-17 22:35:26 +00:00
|
|
|
|
typedef BOOL (WINAPI * SetMenuItemInfoA_Proc) (
|
|
|
|
|
IN HMENU,
|
|
|
|
|
IN UINT,
|
|
|
|
|
IN BOOL,
|
2004-09-02 23:26:16 +00:00
|
|
|
|
IN LPCMENUITEMINFOA);
|
2002-11-17 22:35:26 +00:00
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
GetMenuItemInfoA_Proc get_menu_item_info = NULL;
|
|
|
|
|
SetMenuItemInfoA_Proc set_menu_item_info = NULL;
|
|
|
|
|
AppendMenuW_Proc unicode_append_menu = NULL;
|
2001-12-01 11:16:54 +00:00
|
|
|
|
|
1996-05-21 18:09:43 +00:00
|
|
|
|
Lisp_Object Qdebug_on_next_call;
|
|
|
|
|
|
2006-09-10 05:52:01 +00:00
|
|
|
|
extern Lisp_Object Vmenu_updating_frame;
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
extern Lisp_Object Qmenu_bar;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
extern Lisp_Object QCtoggle, QCradio;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1996-05-21 18:09:43 +00:00
|
|
|
|
extern Lisp_Object Voverriding_local_map;
|
|
|
|
|
extern Lisp_Object Voverriding_local_map_menu_flag;
|
|
|
|
|
|
|
|
|
|
extern Lisp_Object Qoverriding_local_map, Qoverriding_terminal_local_map;
|
|
|
|
|
|
|
|
|
|
extern Lisp_Object Qmenu_bar_update_hook;
|
|
|
|
|
|
2008-04-10 15:20:07 +00:00
|
|
|
|
void set_frame_menubar P_ ((FRAME_PTR, int, int));
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
static void push_menu_item P_ ((Lisp_Object, Lisp_Object, Lisp_Object,
|
|
|
|
|
Lisp_Object, Lisp_Object, Lisp_Object,
|
|
|
|
|
Lisp_Object, Lisp_Object));
|
2001-11-28 20:31:43 +00:00
|
|
|
|
#ifdef HAVE_DIALOGS
|
2008-04-10 15:20:07 +00:00
|
|
|
|
static Lisp_Object w32_dialog_show P_ ((FRAME_PTR, int, Lisp_Object, char**));
|
|
|
|
|
#else
|
|
|
|
|
static int is_simple_dialog P_ ((Lisp_Object));
|
|
|
|
|
static Lisp_Object simple_dialog_show P_ ((FRAME_PTR, Lisp_Object, Lisp_Object));
|
2001-11-28 20:31:43 +00:00
|
|
|
|
#endif
|
2008-04-10 15:20:07 +00:00
|
|
|
|
static Lisp_Object w32_menu_show P_ ((FRAME_PTR, int, int, int, int,
|
|
|
|
|
Lisp_Object, char **));
|
|
|
|
|
|
|
|
|
|
static void keymap_panes P_ ((Lisp_Object *, int, int));
|
|
|
|
|
static void single_keymap_panes P_ ((Lisp_Object, Lisp_Object, Lisp_Object,
|
|
|
|
|
int, int));
|
|
|
|
|
static void single_menu_item P_ ((Lisp_Object, Lisp_Object,
|
|
|
|
|
Lisp_Object *, int, int));
|
|
|
|
|
static void list_of_panes P_ ((Lisp_Object));
|
|
|
|
|
static void list_of_items P_ ((Lisp_Object));
|
|
|
|
|
void w32_free_menu_strings P_((HWND));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* This holds a Lisp vector that holds the results of decoding
|
|
|
|
|
the keymaps or alist-of-alists that specify a menu.
|
|
|
|
|
|
|
|
|
|
It describes the panes and items within the panes.
|
|
|
|
|
|
|
|
|
|
Each pane is described by 3 elements in the vector:
|
|
|
|
|
t, the pane name, the pane's prefix key.
|
|
|
|
|
Then follow the pane's items, with 5 elements per item:
|
|
|
|
|
the item string, the enable flag, the item's value,
|
|
|
|
|
the definition, and the equivalent keyboard key's description string.
|
|
|
|
|
|
|
|
|
|
In some cases, multiple levels of menus may be described.
|
|
|
|
|
A single vector slot containing nil indicates the start of a submenu.
|
|
|
|
|
A single vector slot containing lambda indicates the end of a submenu.
|
|
|
|
|
The submenu follows a menu item which is the way to reach the submenu.
|
|
|
|
|
|
|
|
|
|
A single vector slot containing quote indicates that the
|
|
|
|
|
following items should appear on the right of a dialog box.
|
|
|
|
|
|
|
|
|
|
Using a Lisp vector to hold this information while we decode it
|
|
|
|
|
takes care of protecting all the data from GC. */
|
|
|
|
|
|
|
|
|
|
#define MENU_ITEMS_PANE_NAME 1
|
|
|
|
|
#define MENU_ITEMS_PANE_PREFIX 2
|
|
|
|
|
#define MENU_ITEMS_PANE_LENGTH 3
|
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
enum menu_item_idx
|
|
|
|
|
{
|
|
|
|
|
MENU_ITEMS_ITEM_NAME = 0,
|
|
|
|
|
MENU_ITEMS_ITEM_ENABLE,
|
|
|
|
|
MENU_ITEMS_ITEM_VALUE,
|
|
|
|
|
MENU_ITEMS_ITEM_EQUIV_KEY,
|
|
|
|
|
MENU_ITEMS_ITEM_DEFINITION,
|
|
|
|
|
MENU_ITEMS_ITEM_TYPE,
|
|
|
|
|
MENU_ITEMS_ITEM_SELECTED,
|
|
|
|
|
MENU_ITEMS_ITEM_HELP,
|
|
|
|
|
MENU_ITEMS_ITEM_LENGTH
|
|
|
|
|
};
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
static Lisp_Object menu_items;
|
|
|
|
|
|
|
|
|
|
/* Number of slots currently allocated in menu_items. */
|
|
|
|
|
static int menu_items_allocated;
|
|
|
|
|
|
|
|
|
|
/* This is the index in menu_items of the first empty slot. */
|
|
|
|
|
static int menu_items_used;
|
|
|
|
|
|
|
|
|
|
/* The number of panes currently recorded in menu_items,
|
|
|
|
|
excluding those within submenus. */
|
|
|
|
|
static int menu_items_n_panes;
|
|
|
|
|
|
|
|
|
|
/* Current depth within submenus. */
|
|
|
|
|
static int menu_items_submenu_depth;
|
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
static int next_menubar_widget_id;
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* This is set nonzero after the user activates the menu bar, and set
|
|
|
|
|
to zero again after the menu bars are redisplayed by prepare_menu_bar.
|
|
|
|
|
While it is nonzero, all calls to set_frame_menubar go deep.
|
|
|
|
|
|
|
|
|
|
I don't understand why this is needed, but it does seem to be
|
|
|
|
|
needed on Motif, according to Marcus Daniels <marcus@sysc.pdx.edu>. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int pending_menu_activation;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return the frame whose ->output_data.w32->menubar_widget equals
|
2000-01-23 03:22:21 +00:00
|
|
|
|
ID, or 0 if none. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
static struct frame *
|
2000-01-23 03:22:21 +00:00
|
|
|
|
menubar_id_to_frame (id)
|
|
|
|
|
HMENU id;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, frame;
|
|
|
|
|
FRAME_PTR f;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
2007-06-29 03:37:33 +00:00
|
|
|
|
for (tail = Vframe_list; CONSP (tail); tail = XCDR (tail))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
1999-09-12 07:05:34 +00:00
|
|
|
|
frame = XCAR (tail);
|
2007-06-29 03:37:33 +00:00
|
|
|
|
if (!FRAMEP (frame))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
continue;
|
|
|
|
|
f = XFRAME (frame);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
if (!FRAME_WINDOW_P (f))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
continue;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
if (f->output_data.w32->menubar_widget == id)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Initialize the menu_items structure if we haven't already done so.
|
|
|
|
|
Also mark it as currently empty. */
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
init_menu_items ()
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (menu_items))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
menu_items_allocated = 60;
|
|
|
|
|
menu_items = Fmake_vector (make_number (menu_items_allocated), Qnil);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
menu_items_used = 0;
|
|
|
|
|
menu_items_n_panes = 0;
|
|
|
|
|
menu_items_submenu_depth = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Call at the end of generating the data in menu_items.
|
|
|
|
|
This fills in the number of items in the last pane. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
finish_menu_items ()
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
|
|
|
|
/* Call when finished using the data for the current menu
|
|
|
|
|
in menu_items. */
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
discard_menu_items ()
|
1997-09-03 00:53:34 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Free the structure if it is especially large.
|
|
|
|
|
Otherwise, hold on to it, to save time. */
|
|
|
|
|
if (menu_items_allocated > 200)
|
1997-09-03 00:53:34 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
menu_items = Qnil;
|
|
|
|
|
menu_items_allocated = 0;
|
1997-09-03 00:53:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Make the menu_items vector twice as large. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
grow_menu_items ()
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
menu_items_allocated *= 2;
|
2007-10-13 12:11:14 +00:00
|
|
|
|
menu_items = larger_vector (menu_items, menu_items_allocated, Qnil);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Begin a submenu. */
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
push_submenu_start ()
|
|
|
|
|
{
|
|
|
|
|
if (menu_items_used + 1 > menu_items_allocated)
|
|
|
|
|
grow_menu_items ();
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
2008-02-11 11:28:35 +00:00
|
|
|
|
ASET (menu_items, menu_items_used, Qnil);
|
|
|
|
|
menu_items_used++;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
menu_items_submenu_depth++;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* End a submenu. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
push_submenu_end ()
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menu_items_used + 1 > menu_items_allocated)
|
|
|
|
|
grow_menu_items ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
2008-02-11 11:28:35 +00:00
|
|
|
|
ASET (menu_items, menu_items_used, Qlambda);
|
|
|
|
|
menu_items_used++;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
menu_items_submenu_depth--;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Indicate boundary between left and right. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
push_left_right_boundary ()
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menu_items_used + 1 > menu_items_allocated)
|
|
|
|
|
grow_menu_items ();
|
|
|
|
|
|
2008-02-11 11:28:35 +00:00
|
|
|
|
ASET (menu_items, menu_items_used, Qquote);
|
|
|
|
|
menu_items_used++;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2001-11-18 22:01:13 +00:00
|
|
|
|
/* Start a new menu pane in menu_items.
|
1998-04-23 23:43:02 +00:00
|
|
|
|
NAME is the pane name. PREFIX_VEC is a prefix key for this pane. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
push_menu_pane (name, prefix_vec)
|
|
|
|
|
Lisp_Object name, prefix_vec;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menu_items_used + MENU_ITEMS_PANE_LENGTH > menu_items_allocated)
|
|
|
|
|
grow_menu_items ();
|
|
|
|
|
|
|
|
|
|
if (menu_items_submenu_depth == 0)
|
|
|
|
|
menu_items_n_panes++;
|
2008-02-11 11:28:35 +00:00
|
|
|
|
ASET (menu_items, menu_items_used, Qt); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, name); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, prefix_vec); menu_items_used++;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
/* Push one menu item into the current pane. NAME is the string to
|
|
|
|
|
display. ENABLE if non-nil means this item can be selected. KEY
|
|
|
|
|
is the key generated by choosing this item, or nil if this item
|
|
|
|
|
doesn't really have a definition. DEF is the definition of this
|
|
|
|
|
item. EQUIV is the textual description of the keyboard equivalent
|
|
|
|
|
for this item (or nil if none). TYPE is the type of this menu
|
|
|
|
|
item, one of nil, `toggle' or `radio'. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
static void
|
2000-01-23 03:22:21 +00:00
|
|
|
|
push_menu_item (name, enable, key, def, equiv, type, selected, help)
|
|
|
|
|
Lisp_Object name, enable, key, def, equiv, type, selected, help;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
if (menu_items_used + MENU_ITEMS_ITEM_LENGTH > menu_items_allocated)
|
|
|
|
|
grow_menu_items ();
|
|
|
|
|
|
2008-02-11 11:28:35 +00:00
|
|
|
|
ASET (menu_items, menu_items_used, name); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, enable); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, key); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, equiv); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, def); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, type); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, selected); menu_items_used++;
|
|
|
|
|
ASET (menu_items, menu_items_used, help); menu_items_used++;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Look through KEYMAPS, a vector of keymaps that is NMAPS long,
|
|
|
|
|
and generate menu panes for them in menu_items.
|
|
|
|
|
If NOTREAL is nonzero,
|
|
|
|
|
don't bother really computing whether an item is enabled. */
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
keymap_panes (keymaps, nmaps, notreal)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object *keymaps;
|
|
|
|
|
int nmaps;
|
|
|
|
|
int notreal;
|
|
|
|
|
{
|
|
|
|
|
int mapno;
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
init_menu_items ();
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Loop over the given keymaps, making a pane for each map.
|
|
|
|
|
But don't make a pane that is empty--ignore that map instead.
|
|
|
|
|
P is the number of panes we have made so far. */
|
|
|
|
|
for (mapno = 0; mapno < nmaps; mapno++)
|
2000-08-20 21:14:08 +00:00
|
|
|
|
single_keymap_panes (keymaps[mapno],
|
2001-10-08 11:12:23 +00:00
|
|
|
|
Fkeymap_prompt (keymaps[mapno]), Qnil, notreal, 10);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
finish_menu_items ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This is a recursive subroutine of keymap_panes.
|
|
|
|
|
It handles one keymap, KEYMAP.
|
|
|
|
|
The other arguments are passed along
|
|
|
|
|
or point to local variables of the previous function.
|
1998-04-23 23:43:02 +00:00
|
|
|
|
If NOTREAL is nonzero, only check for equivalent key bindings, don't
|
|
|
|
|
evaluate expressions in menu items and don't make any menu.
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
If we encounter submenus deeper than MAXDEPTH levels, ignore them. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
single_keymap_panes (keymap, pane_name, prefix, notreal, maxdepth)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object keymap;
|
|
|
|
|
Lisp_Object pane_name;
|
|
|
|
|
Lisp_Object prefix;
|
|
|
|
|
int notreal;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int maxdepth;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object pending_maps = Qnil;
|
|
|
|
|
Lisp_Object tail, item;
|
|
|
|
|
struct gcpro gcpro1, gcpro2;
|
|
|
|
|
|
|
|
|
|
if (maxdepth <= 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
push_menu_pane (pane_name, prefix);
|
|
|
|
|
|
1999-09-12 07:05:34 +00:00
|
|
|
|
for (tail = keymap; CONSP (tail); tail = XCDR (tail))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
GCPRO2 (keymap, pending_maps);
|
|
|
|
|
/* Look at each key binding, and if it is a menu item add it
|
|
|
|
|
to this menu. */
|
1999-09-12 07:05:34 +00:00
|
|
|
|
item = XCAR (tail);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (CONSP (item))
|
1999-09-12 07:05:34 +00:00
|
|
|
|
single_menu_item (XCAR (item), XCDR (item),
|
2000-03-01 20:49:16 +00:00
|
|
|
|
&pending_maps, notreal, maxdepth);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
else if (VECTORP (item))
|
|
|
|
|
{
|
|
|
|
|
/* Loop over the char values represented in the vector. */
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
int len = ASIZE (item);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
int c;
|
|
|
|
|
for (c = 0; c < len; c++)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object character;
|
|
|
|
|
XSETFASTINT (character, c);
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
single_menu_item (character, AREF (item, c),
|
2000-03-01 20:49:16 +00:00
|
|
|
|
&pending_maps, notreal, maxdepth);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
UNGCPRO;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Process now any submenus which want to be panes at this level. */
|
|
|
|
|
while (!NILP (pending_maps))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object elt, eltcdr, string;
|
|
|
|
|
elt = Fcar (pending_maps);
|
1999-09-12 07:05:34 +00:00
|
|
|
|
eltcdr = XCDR (elt);
|
|
|
|
|
string = XCAR (eltcdr);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* We no longer discard the @ from the beginning of the string here.
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Instead, we do this in w32_menu_show. */
|
|
|
|
|
single_keymap_panes (Fcar (elt), string,
|
1999-09-12 07:05:34 +00:00
|
|
|
|
XCDR (eltcdr), notreal, maxdepth - 1);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
pending_maps = Fcdr (pending_maps);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This is a subroutine of single_keymap_panes that handles one
|
|
|
|
|
keymap entry.
|
2003-02-04 14:56:31 +00:00
|
|
|
|
KEY is a key in a keymap and ITEM is its binding.
|
1998-04-23 23:43:02 +00:00
|
|
|
|
PENDING_MAPS_PTR points to a list of keymaps waiting to be made into
|
|
|
|
|
separate panes.
|
|
|
|
|
If NOTREAL is nonzero, only check for equivalent key bindings, don't
|
|
|
|
|
evaluate expressions in menu items and don't make any menu.
|
2000-03-01 20:49:16 +00:00
|
|
|
|
If we encounter submenus deeper than MAXDEPTH levels, ignore them. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
static void
|
2000-03-01 20:49:16 +00:00
|
|
|
|
single_menu_item (key, item, pending_maps_ptr, notreal, maxdepth)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object key, item;
|
|
|
|
|
Lisp_Object *pending_maps_ptr;
|
|
|
|
|
int maxdepth, notreal;
|
|
|
|
|
{
|
2000-01-23 03:22:21 +00:00
|
|
|
|
Lisp_Object map, item_string, enabled;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
struct gcpro gcpro1, gcpro2;
|
|
|
|
|
int res;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Parse the menu item and leave the result in item_properties. */
|
|
|
|
|
GCPRO2 (key, item);
|
|
|
|
|
res = parse_menu_item (item, notreal, 0);
|
|
|
|
|
UNGCPRO;
|
|
|
|
|
if (!res)
|
|
|
|
|
return; /* Not a menu item. */
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
map = AREF (item_properties, ITEM_PROPERTY_MAP);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (notreal)
|
|
|
|
|
{
|
|
|
|
|
/* We don't want to make a menu, just traverse the keymaps to
|
|
|
|
|
precompute equivalent key bindings. */
|
|
|
|
|
if (!NILP (map))
|
|
|
|
|
single_keymap_panes (map, Qnil, key, 1, maxdepth - 1);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
enabled = AREF (item_properties, ITEM_PROPERTY_ENABLE);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
item_string = AREF (item_properties, ITEM_PROPERTY_NAME);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2002-07-15 00:01:34 +00:00
|
|
|
|
if (!NILP (map) && SREF (item_string, 0) == '@')
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
if (!NILP (enabled))
|
|
|
|
|
/* An enabled separate pane. Remember this to handle it later. */
|
|
|
|
|
*pending_maps_ptr = Fcons (Fcons (map, Fcons (item_string, key)),
|
|
|
|
|
*pending_maps_ptr);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
push_menu_item (item_string, enabled, key,
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
AREF (item_properties, ITEM_PROPERTY_DEF),
|
|
|
|
|
AREF (item_properties, ITEM_PROPERTY_KEYEQ),
|
|
|
|
|
AREF (item_properties, ITEM_PROPERTY_TYPE),
|
|
|
|
|
AREF (item_properties, ITEM_PROPERTY_SELECTED),
|
|
|
|
|
AREF (item_properties, ITEM_PROPERTY_HELP));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* Display a submenu using the toolkit. */
|
|
|
|
|
if (! (NILP (map) || NILP (enabled)))
|
|
|
|
|
{
|
|
|
|
|
push_submenu_start ();
|
|
|
|
|
single_keymap_panes (map, Qnil, key, 0, maxdepth - 1);
|
|
|
|
|
push_submenu_end ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1996-01-05 10:03:17 +00:00
|
|
|
|
/* Push all the panes and items of a menu described by the
|
1995-11-07 07:52:28 +00:00
|
|
|
|
alist-of-alists MENU.
|
|
|
|
|
This handles old-fashioned calls to x-popup-menu. */
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
list_of_panes (menu)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object menu;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail;
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
init_menu_items ();
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
* xselect.c (x_own_selection, x_handle_selection_clear)
(x_clear_frame_selections):
* w32menu.c (list_of_panes, list_of_items):
* w32fns.c (w32_color_map_lookup, Fx_create_frame, Fx_display_list):
* textprop.c (validate_plist, interval_has_all_properties)
(interval_has_some_properties, interval_has_some_properties_list)
(add_properties, text_property_list):
* process.c (Fget_buffer_process, list_processes_1, status_notify):
* minibuf.c (Fassoc_string):
* macselect.c (x_own_selection, x_clear_frame_selections)
(Fx_disown_selection_internal):
* keymap.c (Fcommand_remapping, where_is_internal, describe_map_tree):
Use CONSP rather than !NILP and XC[AD]R rather than Fc[ad]r.
2007-10-17 23:43:52 +00:00
|
|
|
|
for (tail = menu; CONSP (tail); tail = XCDR (tail))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object elt, pane_name, pane_data;
|
* xselect.c (x_own_selection, x_handle_selection_clear)
(x_clear_frame_selections):
* w32menu.c (list_of_panes, list_of_items):
* w32fns.c (w32_color_map_lookup, Fx_create_frame, Fx_display_list):
* textprop.c (validate_plist, interval_has_all_properties)
(interval_has_some_properties, interval_has_some_properties_list)
(add_properties, text_property_list):
* process.c (Fget_buffer_process, list_processes_1, status_notify):
* minibuf.c (Fassoc_string):
* macselect.c (x_own_selection, x_clear_frame_selections)
(Fx_disown_selection_internal):
* keymap.c (Fcommand_remapping, where_is_internal, describe_map_tree):
Use CONSP rather than !NILP and XC[AD]R rather than Fc[ad]r.
2007-10-17 23:43:52 +00:00
|
|
|
|
elt = XCAR (tail);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
pane_name = Fcar (elt);
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_STRING (pane_name);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
push_menu_pane (pane_name, Qnil);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
pane_data = Fcdr (elt);
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_CONS (pane_data);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
list_of_items (pane_data);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
finish_menu_items ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Push the items in a single pane defined by the alist PANE. */
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static void
|
|
|
|
|
list_of_items (pane)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object pane;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, item, item1;
|
|
|
|
|
|
* xselect.c (x_own_selection, x_handle_selection_clear)
(x_clear_frame_selections):
* w32menu.c (list_of_panes, list_of_items):
* w32fns.c (w32_color_map_lookup, Fx_create_frame, Fx_display_list):
* textprop.c (validate_plist, interval_has_all_properties)
(interval_has_some_properties, interval_has_some_properties_list)
(add_properties, text_property_list):
* process.c (Fget_buffer_process, list_processes_1, status_notify):
* minibuf.c (Fassoc_string):
* macselect.c (x_own_selection, x_clear_frame_selections)
(Fx_disown_selection_internal):
* keymap.c (Fcommand_remapping, where_is_internal, describe_map_tree):
Use CONSP rather than !NILP and XC[AD]R rather than Fc[ad]r.
2007-10-17 23:43:52 +00:00
|
|
|
|
for (tail = pane; CONSP (tail); tail = XCDR (tail))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
* xselect.c (x_own_selection, x_handle_selection_clear)
(x_clear_frame_selections):
* w32menu.c (list_of_panes, list_of_items):
* w32fns.c (w32_color_map_lookup, Fx_create_frame, Fx_display_list):
* textprop.c (validate_plist, interval_has_all_properties)
(interval_has_some_properties, interval_has_some_properties_list)
(add_properties, text_property_list):
* process.c (Fget_buffer_process, list_processes_1, status_notify):
* minibuf.c (Fassoc_string):
* macselect.c (x_own_selection, x_clear_frame_selections)
(Fx_disown_selection_internal):
* keymap.c (Fcommand_remapping, where_is_internal, describe_map_tree):
Use CONSP rather than !NILP and XC[AD]R rather than Fc[ad]r.
2007-10-17 23:43:52 +00:00
|
|
|
|
item = XCAR (tail);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (STRINGP (item))
|
2000-01-23 03:22:21 +00:00
|
|
|
|
push_menu_item (item, Qnil, Qnil, Qt, Qnil, Qnil, Qnil, Qnil);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
else if (NILP (item))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
push_left_right_boundary ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_CONS (item);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
item1 = Fcar (item);
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_STRING (item1);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
push_menu_item (item1, Qt, Fcdr (item), Qt, Qnil, Qnil, Qnil, Qnil);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
DEFUN ("x-popup-menu", Fx_popup_menu, Sx_popup_menu, 2, 2, 0,
|
2001-11-12 23:57:09 +00:00
|
|
|
|
doc: /* Pop up a deck-of-cards menu and return user's selection.
|
|
|
|
|
POSITION is a position specification. This is either a mouse button
|
|
|
|
|
event or a list ((XOFFSET YOFFSET) WINDOW) where XOFFSET and YOFFSET
|
|
|
|
|
are positions in pixels from the top left corner of WINDOW's frame
|
|
|
|
|
\(WINDOW may be a frame object instead of a window). This controls the
|
|
|
|
|
position of the center of the first line in the first pane of the
|
|
|
|
|
menu, not the top left of the menu as a whole. If POSITION is t, it
|
|
|
|
|
means to use the current mouse position.
|
|
|
|
|
|
|
|
|
|
MENU is a specifier for a menu. For the simplest case, MENU is a keymap.
|
|
|
|
|
The menu items come from key bindings that have a menu string as well as
|
|
|
|
|
a definition; actually, the \"definition\" in such a key binding looks like
|
|
|
|
|
\(STRING . REAL-DEFINITION). To give the menu a title, put a string into
|
|
|
|
|
the keymap as a top-level element.
|
|
|
|
|
|
|
|
|
|
If REAL-DEFINITION is nil, that puts a nonselectable string in the menu.
|
|
|
|
|
Otherwise, REAL-DEFINITION should be a valid key binding definition.
|
|
|
|
|
|
|
|
|
|
You can also use a list of keymaps as MENU. Then each keymap makes a
|
|
|
|
|
separate pane. When MENU is a keymap or a list of keymaps, the return
|
|
|
|
|
value is a list of events.
|
|
|
|
|
|
|
|
|
|
Alternatively, you can specify a menu of multiple panes with a list of
|
|
|
|
|
the form (TITLE PANE1 PANE2...), where each pane is a list of
|
|
|
|
|
form (TITLE ITEM1 ITEM2...).
|
|
|
|
|
Each ITEM is normally a cons cell (STRING . VALUE); but a string can
|
|
|
|
|
appear as an item--that makes a nonselectable line in the menu.
|
|
|
|
|
With this form of menu, the return value is VALUE from the chosen item.
|
|
|
|
|
|
|
|
|
|
If POSITION is nil, don't display the menu at all, just precalculate the
|
|
|
|
|
cached information about equivalent key sequences. */)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
(position, menu)
|
|
|
|
|
Lisp_Object position, menu;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object keymap, tem;
|
2001-10-30 03:38:24 +00:00
|
|
|
|
int xpos = 0, ypos = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object title;
|
|
|
|
|
char *error_name;
|
|
|
|
|
Lisp_Object selection;
|
2001-10-30 03:38:24 +00:00
|
|
|
|
FRAME_PTR f = NULL;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object x, y, window;
|
|
|
|
|
int keymaps = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int for_click = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
struct gcpro gcpro1;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
#ifdef HAVE_MENUS
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (! NILP (position))
|
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
check_w32 ();
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Decode the first argument: find the window and the coordinates. */
|
1997-01-13 01:41:59 +00:00
|
|
|
|
if (EQ (position, Qt)
|
2000-10-19 19:25:53 +00:00
|
|
|
|
|| (CONSP (position) && (EQ (XCAR (position), Qmenu_bar)
|
|
|
|
|
|| EQ (XCAR (position), Qtool_bar))))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
/* Use the mouse's current position. */
|
2000-01-23 03:22:21 +00:00
|
|
|
|
FRAME_PTR new_f = SELECTED_FRAME ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object bar_window;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
enum scroll_bar_part part;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
unsigned long time;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2007-05-15 23:02:24 +00:00
|
|
|
|
if (FRAME_TERMINAL (new_f)->mouse_position_hook)
|
|
|
|
|
(*FRAME_TERMINAL (new_f)->mouse_position_hook) (&new_f, 1, &bar_window,
|
1998-04-23 23:43:02 +00:00
|
|
|
|
&part, &x, &y, &time);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (new_f != 0)
|
|
|
|
|
XSETFRAME (window, new_f);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
window = selected_window;
|
|
|
|
|
XSETFASTINT (x, 0);
|
|
|
|
|
XSETFASTINT (y, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tem = Fcar (position);
|
|
|
|
|
if (CONSP (tem))
|
|
|
|
|
{
|
|
|
|
|
window = Fcar (Fcdr (position));
|
|
|
|
|
x = Fcar (tem);
|
|
|
|
|
y = Fcar (Fcdr (tem));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
for_click = 1;
|
|
|
|
|
tem = Fcar (Fcdr (position)); /* EVENT_START (position) */
|
|
|
|
|
window = Fcar (tem); /* POSN_WINDOW (tem) */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
tem = Fcar (Fcdr (Fcdr (tem))); /* POSN_WINDOW_POSN (tem) */
|
|
|
|
|
x = Fcar (tem);
|
|
|
|
|
y = Fcdr (tem);
|
|
|
|
|
}
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_NUMBER (x);
|
|
|
|
|
CHECK_NUMBER (y);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
/* Decode where to put the menu. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (FRAMEP (window))
|
|
|
|
|
{
|
|
|
|
|
f = XFRAME (window);
|
|
|
|
|
xpos = 0;
|
|
|
|
|
ypos = 0;
|
|
|
|
|
}
|
|
|
|
|
else if (WINDOWP (window))
|
|
|
|
|
{
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_LIVE_WINDOW (window);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
f = XFRAME (WINDOW_FRAME (XWINDOW (window)));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2003-05-24 22:07:03 +00:00
|
|
|
|
xpos = WINDOW_LEFT_EDGE_X (XWINDOW (window));
|
|
|
|
|
ypos = WINDOW_TOP_EDGE_Y (XWINDOW (window));
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
/* ??? Not really clean; should be CHECK_WINDOW_OR_FRAME,
|
|
|
|
|
but I don't want to make one now. */
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_WINDOW (window);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
xpos += XINT (x);
|
|
|
|
|
ypos += XINT (y);
|
1998-04-17 05:09:51 +00:00
|
|
|
|
|
|
|
|
|
XSETFRAME (Vmenu_updating_frame, f);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
2005-12-20 21:38:27 +00:00
|
|
|
|
else
|
|
|
|
|
Vmenu_updating_frame = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#endif /* HAVE_MENUS */
|
1998-04-17 05:09:51 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
title = Qnil;
|
|
|
|
|
GCPRO1 (title);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* Decode the menu items from what was specified. */
|
|
|
|
|
|
2000-10-27 22:20:19 +00:00
|
|
|
|
keymap = get_keymap (menu, 0, 0);
|
|
|
|
|
if (CONSP (keymap))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
/* We were given a keymap. Extract menu info from the keymap. */
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
|
|
|
|
|
/* Extract the detailed info to make one pane. */
|
|
|
|
|
keymap_panes (&menu, 1, NILP (position));
|
|
|
|
|
|
|
|
|
|
/* Search for a string appearing directly as an element of the keymap.
|
|
|
|
|
That string is the title of the menu. */
|
2001-10-08 11:12:23 +00:00
|
|
|
|
prompt = Fkeymap_prompt (keymap);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (title) && !NILP (prompt))
|
|
|
|
|
title = prompt;
|
|
|
|
|
|
|
|
|
|
/* Make that be the pane title of the first pane. */
|
|
|
|
|
if (!NILP (prompt) && menu_items_n_panes >= 0)
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, MENU_ITEMS_PANE_NAME, prompt);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
keymaps = 1;
|
|
|
|
|
}
|
2000-10-27 22:20:19 +00:00
|
|
|
|
else if (CONSP (menu) && KEYMAPP (XCAR (menu)))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
/* We were given a list of keymaps. */
|
|
|
|
|
int nmaps = XFASTINT (Flength (menu));
|
|
|
|
|
Lisp_Object *maps
|
|
|
|
|
= (Lisp_Object *) alloca (nmaps * sizeof (Lisp_Object));
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
title = Qnil;
|
|
|
|
|
|
|
|
|
|
/* The first keymap that has a prompt string
|
|
|
|
|
supplies the menu title. */
|
|
|
|
|
for (tem = menu, i = 0; CONSP (tem); tem = Fcdr (tem))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
|
2000-10-27 22:20:19 +00:00
|
|
|
|
maps[i++] = keymap = get_keymap (Fcar (tem), 1, 0);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2001-10-08 11:12:23 +00:00
|
|
|
|
prompt = Fkeymap_prompt (keymap);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (title) && !NILP (prompt))
|
|
|
|
|
title = prompt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Extract the detailed info to make one pane. */
|
|
|
|
|
keymap_panes (maps, nmaps, NILP (position));
|
|
|
|
|
|
|
|
|
|
/* Make the title be the pane title of the first pane. */
|
|
|
|
|
if (!NILP (title) && menu_items_n_panes >= 0)
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, MENU_ITEMS_PANE_NAME, title);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
keymaps = 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* We were given an old-fashioned menu. */
|
|
|
|
|
title = Fcar (menu);
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_STRING (title);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
list_of_panes (Fcdr (menu));
|
|
|
|
|
|
|
|
|
|
keymaps = 0;
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (NILP (position))
|
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
discard_menu_items ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
UNGCPRO;
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
#ifdef HAVE_MENUS
|
2006-03-12 16:29:25 +00:00
|
|
|
|
/* If resources from a previous popup menu still exist, does nothing
|
|
|
|
|
until the `menu_free_timer' has freed them (see w32fns.c). This
|
|
|
|
|
can occur if you press ESC or click outside a menu without selecting
|
|
|
|
|
a menu item.
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
*/
|
|
|
|
|
if (current_popup_menu)
|
|
|
|
|
{
|
|
|
|
|
discard_menu_items ();
|
|
|
|
|
UNGCPRO;
|
|
|
|
|
return Qnil;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Display them in a menu. */
|
|
|
|
|
BLOCK_INPUT;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
selection = w32_menu_show (f, xpos, ypos, for_click,
|
|
|
|
|
keymaps, title, &error_name);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
UNBLOCK_INPUT;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
discard_menu_items ();
|
2006-03-12 16:29:25 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
#endif /* HAVE_MENUS */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
UNGCPRO;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (error_name) error (error_name);
|
|
|
|
|
return selection;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#ifdef HAVE_MENUS
|
|
|
|
|
|
2005-05-24 08:45:35 +00:00
|
|
|
|
DEFUN ("x-popup-dialog", Fx_popup_dialog, Sx_popup_dialog, 2, 3, 0,
|
2001-11-12 23:57:09 +00:00
|
|
|
|
doc: /* Pop up a dialog box and return user's selection.
|
|
|
|
|
POSITION specifies which frame to use.
|
|
|
|
|
This is normally a mouse button event or a window or frame.
|
|
|
|
|
If POSITION is t, it means to use the frame the mouse is on.
|
|
|
|
|
The dialog box appears in the middle of the specified frame.
|
|
|
|
|
|
|
|
|
|
CONTENTS specifies the alternatives to display in the dialog box.
|
|
|
|
|
It is a list of the form (TITLE ITEM1 ITEM2...).
|
|
|
|
|
Each ITEM is a cons cell (STRING . VALUE).
|
|
|
|
|
The return value is VALUE from the chosen item.
|
|
|
|
|
|
|
|
|
|
An ITEM may also be just a string--that makes a nonselectable item.
|
|
|
|
|
An ITEM may also be nil--that means to put all preceding items
|
|
|
|
|
on the left of the dialog box and all following items on the right.
|
2005-05-24 08:38:58 +00:00
|
|
|
|
\(By default, approximately half appear on each side.)
|
|
|
|
|
|
|
|
|
|
If HEADER is non-nil, the frame title for the box is "Information",
|
|
|
|
|
otherwise it is "Question". */)
|
|
|
|
|
(position, contents, header)
|
|
|
|
|
Lisp_Object position, contents, header;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
2001-10-30 03:38:24 +00:00
|
|
|
|
FRAME_PTR f = NULL;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
Lisp_Object window;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
check_w32 ();
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Decode the first argument: find the window or frame to use. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (EQ (position, Qt)
|
2000-10-19 19:25:53 +00:00
|
|
|
|
|| (CONSP (position) && (EQ (XCAR (position), Qmenu_bar)
|
|
|
|
|
|| EQ (XCAR (position), Qtool_bar))))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#if 0 /* Using the frame the mouse is on may not be right. */
|
|
|
|
|
/* Use the mouse's current position. */
|
2000-01-23 03:22:21 +00:00
|
|
|
|
FRAME_PTR new_f = SELECTED_FRAME ();
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object bar_window;
|
2000-08-22 23:37:05 +00:00
|
|
|
|
enum scroll_bar_part part;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
unsigned long time;
|
|
|
|
|
Lisp_Object x, y;
|
|
|
|
|
|
|
|
|
|
(*mouse_position_hook) (&new_f, 1, &bar_window, &part, &x, &y, &time);
|
|
|
|
|
|
|
|
|
|
if (new_f != 0)
|
|
|
|
|
XSETFRAME (window, new_f);
|
|
|
|
|
else
|
1995-11-07 07:52:28 +00:00
|
|
|
|
window = selected_window;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#endif
|
|
|
|
|
window = selected_window;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
else if (CONSP (position))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tem;
|
|
|
|
|
tem = Fcar (position);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (CONSP (tem))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
window = Fcar (Fcdr (position));
|
|
|
|
|
else
|
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
tem = Fcar (Fcdr (position)); /* EVENT_START (position) */
|
|
|
|
|
window = Fcar (tem); /* POSN_WINDOW (tem) */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (WINDOWP (position) || FRAMEP (position))
|
|
|
|
|
window = position;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
else
|
|
|
|
|
window = Qnil;
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Decode where to put the menu. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (FRAMEP (window))
|
|
|
|
|
f = XFRAME (window);
|
|
|
|
|
else if (WINDOWP (window))
|
|
|
|
|
{
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_LIVE_WINDOW (window);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
f = XFRAME (WINDOW_FRAME (XWINDOW (window)));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
/* ??? Not really clean; should be CHECK_WINDOW_OR_FRAME,
|
|
|
|
|
but I don't want to make one now. */
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_WINDOW (window);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
#ifndef HAVE_DIALOGS
|
2008-04-10 15:20:07 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
2008-04-10 15:20:07 +00:00
|
|
|
|
/* Handle simple Yes/No choices as MessageBox popups. */
|
|
|
|
|
if (is_simple_dialog (contents))
|
|
|
|
|
return simple_dialog_show (f, contents, header);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Display a menu with these alternatives
|
|
|
|
|
in the middle of frame F. */
|
|
|
|
|
Lisp_Object x, y, frame, newpos;
|
|
|
|
|
XSETFRAME (frame, f);
|
|
|
|
|
XSETINT (x, x_pixel_width (f) / 2);
|
|
|
|
|
XSETINT (y, x_pixel_height (f) / 2);
|
|
|
|
|
newpos = Fcons (Fcons (x, Fcons (y, Qnil)), Fcons (frame, Qnil));
|
|
|
|
|
return Fx_popup_menu (newpos,
|
|
|
|
|
Fcons (Fcar (contents), Fcons (contents, Qnil)));
|
|
|
|
|
}
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
2000-01-23 03:22:21 +00:00
|
|
|
|
#else /* HAVE_DIALOGS */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object title;
|
|
|
|
|
char *error_name;
|
|
|
|
|
Lisp_Object selection;
|
|
|
|
|
|
|
|
|
|
/* Decode the dialog items from what was specified. */
|
|
|
|
|
title = Fcar (contents);
|
2001-11-02 20:46:55 +00:00
|
|
|
|
CHECK_STRING (title);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
list_of_panes (Fcons (contents, Qnil));
|
|
|
|
|
|
|
|
|
|
/* Display them in a dialog box. */
|
|
|
|
|
BLOCK_INPUT;
|
2005-05-24 08:38:58 +00:00
|
|
|
|
selection = w32_dialog_show (f, 0, title, header, &error_name);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
UNBLOCK_INPUT;
|
|
|
|
|
|
|
|
|
|
discard_menu_items ();
|
|
|
|
|
|
|
|
|
|
if (error_name) error (error_name);
|
|
|
|
|
return selection;
|
|
|
|
|
}
|
2000-01-23 03:22:21 +00:00
|
|
|
|
#endif /* HAVE_DIALOGS */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-09-03 00:53:34 +00:00
|
|
|
|
/* Activate the menu bar of frame F.
|
|
|
|
|
This is called from keyboard.c when it gets the
|
2002-06-13 14:59:38 +00:00
|
|
|
|
MENU_BAR_ACTIVATE_EVENT out of the Emacs event queue.
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
|
|
|
|
To activate the menu bar, we signal to the input thread that it can
|
|
|
|
|
return from the WM_INITMENU message, allowing the normal Windows
|
|
|
|
|
processing of the menus.
|
|
|
|
|
|
|
|
|
|
But first we recompute the menu bar contents (the whole tree).
|
|
|
|
|
|
|
|
|
|
This way we can safely execute Lisp code. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
void
|
1997-09-03 00:53:34 +00:00
|
|
|
|
x_activate_menubar (f)
|
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
{
|
|
|
|
|
set_frame_menubar (f, 0, 1);
|
|
|
|
|
|
|
|
|
|
/* Lock out further menubar changes while active. */
|
|
|
|
|
f->output_data.w32->menubar_active = 1;
|
|
|
|
|
|
|
|
|
|
/* Signal input thread to return from WM_INITMENU. */
|
|
|
|
|
complete_deferred_msg (FRAME_W32_WINDOW (f), WM_INITMENU, 0);
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* This callback is called from the menu bar pulldown menu
|
|
|
|
|
when the user makes a selection.
|
|
|
|
|
Figure out what the user chose
|
|
|
|
|
and put the appropriate events into the keyboard buffer. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
menubar_selection_callback (FRAME_PTR f, void * client_data)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object prefix, entry;
|
|
|
|
|
Lisp_Object vector;
|
|
|
|
|
Lisp_Object *subprefix_stack;
|
|
|
|
|
int submenu_depth = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
int i;
|
1996-05-21 18:09:43 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (!f)
|
1997-09-03 00:53:34 +00:00
|
|
|
|
return;
|
2001-07-11 18:31:41 +00:00
|
|
|
|
entry = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
subprefix_stack = (Lisp_Object *) alloca (f->menu_bar_items_used * sizeof (Lisp_Object));
|
|
|
|
|
vector = f->menu_bar_vector;
|
|
|
|
|
prefix = Qnil;
|
|
|
|
|
i = 0;
|
|
|
|
|
while (i < f->menu_bar_items_used)
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
if (EQ (AREF (vector, i), Qnil))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
subprefix_stack[submenu_depth++] = prefix;
|
|
|
|
|
prefix = entry;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (vector, i), Qlambda))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
prefix = subprefix_stack[--submenu_depth];
|
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (vector, i), Qt))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
prefix = AREF (vector, i + MENU_ITEMS_PANE_PREFIX);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
entry = AREF (vector, i + MENU_ITEMS_ITEM_VALUE);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* The EMACS_INT cast avoids a warning. There's no problem
|
|
|
|
|
as long as pointers have enough bits to hold small integers. */
|
|
|
|
|
if ((int) (EMACS_INT) client_data == i)
|
|
|
|
|
{
|
|
|
|
|
int j;
|
|
|
|
|
struct input_event buf;
|
|
|
|
|
Lisp_Object frame;
|
2003-06-15 21:05:34 +00:00
|
|
|
|
EVENT_INIT (buf);
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
XSETFRAME (frame, f);
|
2000-07-13 14:07:57 +00:00
|
|
|
|
buf.kind = MENU_BAR_EVENT;
|
|
|
|
|
buf.frame_or_window = frame;
|
|
|
|
|
buf.arg = frame;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
kbd_buffer_store_event (&buf);
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
for (j = 0; j < submenu_depth; j++)
|
|
|
|
|
if (!NILP (subprefix_stack[j]))
|
|
|
|
|
{
|
2000-07-13 14:07:57 +00:00
|
|
|
|
buf.kind = MENU_BAR_EVENT;
|
|
|
|
|
buf.frame_or_window = frame;
|
|
|
|
|
buf.arg = subprefix_stack[j];
|
1998-04-23 23:43:02 +00:00
|
|
|
|
kbd_buffer_store_event (&buf);
|
|
|
|
|
}
|
1996-05-21 18:09:43 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (!NILP (prefix))
|
|
|
|
|
{
|
2000-07-13 14:07:57 +00:00
|
|
|
|
buf.kind = MENU_BAR_EVENT;
|
|
|
|
|
buf.frame_or_window = frame;
|
|
|
|
|
buf.arg = prefix;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
kbd_buffer_store_event (&buf);
|
|
|
|
|
}
|
1996-05-21 18:09:43 +00:00
|
|
|
|
|
2000-07-13 14:07:57 +00:00
|
|
|
|
buf.kind = MENU_BAR_EVENT;
|
|
|
|
|
buf.frame_or_window = frame;
|
|
|
|
|
buf.arg = entry;
|
2002-02-23 00:01:34 +00:00
|
|
|
|
/* Free memory used by owner-drawn and help-echo strings. */
|
|
|
|
|
w32_free_menu_strings (FRAME_W32_WINDOW (f));
|
2007-02-22 22:49:41 +00:00
|
|
|
|
kbd_buffer_store_event (&buf);
|
|
|
|
|
|
2002-02-23 00:01:34 +00:00
|
|
|
|
f->output_data.w32->menubar_active = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
i += MENU_ITEMS_ITEM_LENGTH;
|
|
|
|
|
}
|
1997-09-03 00:53:34 +00:00
|
|
|
|
}
|
2002-02-23 00:01:34 +00:00
|
|
|
|
/* Free memory used by owner-drawn and help-echo strings. */
|
|
|
|
|
w32_free_menu_strings (FRAME_W32_WINDOW (f));
|
|
|
|
|
f->output_data.w32->menubar_active = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Allocate a widget_value, blocking input. */
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
widget_value *
|
|
|
|
|
xmalloc_widget_value ()
|
|
|
|
|
{
|
|
|
|
|
widget_value *value;
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1996-05-21 18:09:43 +00:00
|
|
|
|
BLOCK_INPUT;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
value = malloc_widget_value ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
UNBLOCK_INPUT;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
return value;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* This recursively calls free_widget_value on the tree of widgets.
|
|
|
|
|
It must free all data that was malloc'ed for these widget_values.
|
|
|
|
|
In Emacs, many slots are pointers into the data of Lisp_Strings, and
|
|
|
|
|
must be left alone. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
free_menubar_widget_value_tree (wv)
|
|
|
|
|
widget_value *wv;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (! wv) return;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv->name = wv->value = wv->key = (char *) 0xDEADBEEF;
|
|
|
|
|
|
|
|
|
|
if (wv->contents && (wv->contents != (widget_value*)1))
|
|
|
|
|
{
|
|
|
|
|
free_menubar_widget_value_tree (wv->contents);
|
|
|
|
|
wv->contents = (widget_value *) 0xDEADBEEF;
|
|
|
|
|
}
|
|
|
|
|
if (wv->next)
|
|
|
|
|
{
|
|
|
|
|
free_menubar_widget_value_tree (wv->next);
|
|
|
|
|
wv->next = (widget_value *) 0xDEADBEEF;
|
|
|
|
|
}
|
|
|
|
|
BLOCK_INPUT;
|
|
|
|
|
free_widget_value (wv);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
UNBLOCK_INPUT;
|
|
|
|
|
}
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
/* Set up data i menu_items for a menu bar item
|
1998-04-23 23:43:02 +00:00
|
|
|
|
whose event type is ITEM_KEY (with string ITEM_NAME)
|
|
|
|
|
and whose contents come from the list of keymaps MAPS. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
static int
|
|
|
|
|
parse_single_submenu (item_key, item_name, maps)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object item_key, item_name, maps;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object length;
|
|
|
|
|
int len;
|
|
|
|
|
Lisp_Object *mapvec;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
int i;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int top_level_items = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
length = Flength (maps);
|
|
|
|
|
len = XINT (length);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Convert the list MAPS into a vector MAPVEC. */
|
|
|
|
|
mapvec = (Lisp_Object *) alloca (len * sizeof (Lisp_Object));
|
|
|
|
|
for (i = 0; i < len; i++)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
mapvec[i] = Fcar (maps);
|
|
|
|
|
maps = Fcdr (maps);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Loop over the given keymaps, making a pane for each map.
|
|
|
|
|
But don't make a pane that is empty--ignore that map instead. */
|
|
|
|
|
for (i = 0; i < len; i++)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (SYMBOLP (mapvec[i])
|
2000-10-27 22:20:19 +00:00
|
|
|
|
|| (CONSP (mapvec[i]) && !KEYMAPP (mapvec[i])))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
/* Here we have a command at top level in the menu bar
|
|
|
|
|
as opposed to a submenu. */
|
|
|
|
|
top_level_items = 1;
|
|
|
|
|
push_menu_pane (Qnil, Qnil);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
push_menu_item (item_name, Qt, item_key, mapvec[i],
|
|
|
|
|
Qnil, Qnil, Qnil, Qnil);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2003-01-20 20:18:13 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object prompt;
|
|
|
|
|
prompt = Fkeymap_prompt (mapvec[i]);
|
|
|
|
|
single_keymap_panes (mapvec[i],
|
|
|
|
|
!NILP (prompt) ? prompt : item_name,
|
|
|
|
|
item_key, 0, 10);
|
|
|
|
|
}
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
return top_level_items;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
|
|
|
|
|
/* Create a tree of widget_value objects
|
|
|
|
|
representing the panes and items
|
|
|
|
|
in menu_items starting at index START, up to index END. */
|
|
|
|
|
|
|
|
|
|
static widget_value *
|
|
|
|
|
digest_single_submenu (start, end, top_level_items)
|
2003-01-20 20:18:13 +00:00
|
|
|
|
int start, end, top_level_items;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
{
|
|
|
|
|
widget_value *wv, *prev_wv, *save_wv, *first_wv;
|
|
|
|
|
int i;
|
|
|
|
|
int submenu_depth = 0;
|
|
|
|
|
widget_value **submenu_stack;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
submenu_stack
|
|
|
|
|
= (widget_value **) alloca (menu_items_used * sizeof (widget_value *));
|
|
|
|
|
wv = xmalloc_widget_value ();
|
|
|
|
|
wv->name = "menu";
|
|
|
|
|
wv->value = 0;
|
|
|
|
|
wv->enabled = 1;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
first_wv = wv;
|
|
|
|
|
save_wv = 0;
|
|
|
|
|
prev_wv = 0;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
2003-01-20 20:18:13 +00:00
|
|
|
|
/* Loop over all panes and items made by the preceding call
|
|
|
|
|
to parse_single_submenu and construct a tree of widget_value objects.
|
|
|
|
|
Ignore the panes and items used by previous calls to
|
|
|
|
|
digest_single_submenu, even though those are also in menu_items. */
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
i = start;
|
|
|
|
|
while (i < end)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
if (EQ (AREF (menu_items, i), Qnil))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
submenu_stack[submenu_depth++] = save_wv;
|
|
|
|
|
save_wv = prev_wv;
|
|
|
|
|
prev_wv = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qlambda))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv = save_wv;
|
|
|
|
|
save_wv = submenu_stack[--submenu_depth];
|
1995-11-07 07:52:28 +00:00
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qt)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
&& submenu_depth != 0)
|
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Ignore a nil in the item list.
|
|
|
|
|
It's meaningful only for dialog boxes. */
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qquote))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
i += 1;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qt))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
/* Create a new pane. */
|
|
|
|
|
Lisp_Object pane_name, prefix;
|
|
|
|
|
char *pane_string;
|
2001-10-21 10:54:17 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
pane_name = AREF (menu_items, i + MENU_ITEMS_PANE_NAME);
|
|
|
|
|
prefix = AREF (menu_items, i + MENU_ITEMS_PANE_PREFIX);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (STRINGP (pane_name))
|
2001-10-21 10:54:17 +00:00
|
|
|
|
{
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
/* Encode as UTF-8 for now. */
|
|
|
|
|
pane_name = ENCODE_UTF_8 (pane_name);
|
|
|
|
|
else if (STRING_MULTIBYTE (pane_name))
|
|
|
|
|
pane_name = ENCODE_SYSTEM (pane_name);
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, i + MENU_ITEMS_PANE_NAME, pane_name);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
}
|
2004-09-02 23:26:16 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
pane_string = (NILP (pane_name)
|
2002-07-15 00:01:34 +00:00
|
|
|
|
? "" : (char *) SDATA (pane_name));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* If there is just one top-level pane, put all its items directly
|
|
|
|
|
under the top-level menu. */
|
|
|
|
|
if (menu_items_n_panes == 1)
|
|
|
|
|
pane_string = "";
|
|
|
|
|
|
|
|
|
|
/* If the pane has a meaningful name,
|
|
|
|
|
make the pane a top-level menu item
|
|
|
|
|
with its items as a submenu beneath it. */
|
|
|
|
|
if (strcmp (pane_string, ""))
|
|
|
|
|
{
|
|
|
|
|
wv = xmalloc_widget_value ();
|
|
|
|
|
if (save_wv)
|
|
|
|
|
save_wv->next = wv;
|
|
|
|
|
else
|
|
|
|
|
first_wv->contents = wv;
|
2004-09-02 23:26:16 +00:00
|
|
|
|
wv->lname = pane_name;
|
|
|
|
|
/* Set value to 1 so update_submenu_strings can handle '@' */
|
|
|
|
|
wv->value = (char *) 1;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv->enabled = 1;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
|
|
|
|
save_wv = wv;
|
|
|
|
|
prev_wv = 0;
|
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Create a new item within current pane. */
|
2000-01-23 03:22:21 +00:00
|
|
|
|
Lisp_Object item_name, enable, descrip, def, type, selected;
|
|
|
|
|
Lisp_Object help;
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
2001-10-21 10:54:17 +00:00
|
|
|
|
item_name = AREF (menu_items, i + MENU_ITEMS_ITEM_NAME);
|
|
|
|
|
enable = AREF (menu_items, i + MENU_ITEMS_ITEM_ENABLE);
|
|
|
|
|
descrip = AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY);
|
|
|
|
|
def = AREF (menu_items, i + MENU_ITEMS_ITEM_DEFINITION);
|
|
|
|
|
type = AREF (menu_items, i + MENU_ITEMS_ITEM_TYPE);
|
|
|
|
|
selected = AREF (menu_items, i + MENU_ITEMS_ITEM_SELECTED);
|
|
|
|
|
help = AREF (menu_items, i + MENU_ITEMS_ITEM_HELP);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (STRINGP (item_name))
|
2001-10-21 10:54:17 +00:00
|
|
|
|
{
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
item_name = ENCODE_UTF_8 (item_name);
|
|
|
|
|
else if (STRING_MULTIBYTE (item_name))
|
|
|
|
|
item_name = ENCODE_SYSTEM (item_name);
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, i + MENU_ITEMS_ITEM_NAME, item_name);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
}
|
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
if (STRINGP (descrip) && STRING_MULTIBYTE (descrip))
|
2001-10-21 10:54:17 +00:00
|
|
|
|
{
|
|
|
|
|
descrip = ENCODE_SYSTEM (descrip);
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY, descrip);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
wv = xmalloc_widget_value ();
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (prev_wv)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv->next = wv;
|
|
|
|
|
else
|
|
|
|
|
save_wv->contents = wv;
|
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
wv->lname = item_name;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (!NILP (descrip))
|
2004-09-02 23:26:16 +00:00
|
|
|
|
wv->lkey = descrip;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv->value = 0;
|
|
|
|
|
/* The EMACS_INT cast avoids a warning. There's no problem
|
|
|
|
|
as long as pointers have enough bits to hold small integers. */
|
|
|
|
|
wv->call_data = (!NILP (def) ? (void *) (EMACS_INT) i : 0);
|
|
|
|
|
wv->enabled = !NILP (enable);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
|
|
|
|
if (NILP (type))
|
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
|
|
|
|
else if (EQ (type, QCradio))
|
|
|
|
|
wv->button_type = BUTTON_TYPE_RADIO;
|
|
|
|
|
else if (EQ (type, QCtoggle))
|
|
|
|
|
wv->button_type = BUTTON_TYPE_TOGGLE;
|
|
|
|
|
else
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
wv->selected = !NILP (selected);
|
2001-12-09 16:48:03 +00:00
|
|
|
|
if (!STRINGP (help))
|
|
|
|
|
help = Qnil;
|
|
|
|
|
|
|
|
|
|
wv->help = help;
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv = wv;
|
|
|
|
|
|
|
|
|
|
i += MENU_ITEMS_ITEM_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* If we have just one "menu item"
|
|
|
|
|
that was originally a button, return it by itself. */
|
|
|
|
|
if (top_level_items && first_wv->contents && first_wv->contents->next == 0)
|
|
|
|
|
{
|
|
|
|
|
wv = first_wv->contents;
|
|
|
|
|
free_widget_value (first_wv);
|
|
|
|
|
return wv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return first_wv;
|
|
|
|
|
}
|
2004-09-02 23:26:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Walk through the widget_value tree starting at FIRST_WV and update
|
|
|
|
|
the char * pointers from the corresponding lisp values.
|
|
|
|
|
We do this after building the whole tree, since GC may happen while the
|
|
|
|
|
tree is constructed, and small strings are relocated. So we must wait
|
|
|
|
|
until no GC can happen before storing pointers into lisp values. */
|
|
|
|
|
static void
|
|
|
|
|
update_submenu_strings (first_wv)
|
|
|
|
|
widget_value *first_wv;
|
|
|
|
|
{
|
|
|
|
|
widget_value *wv;
|
|
|
|
|
|
|
|
|
|
for (wv = first_wv; wv; wv = wv->next)
|
|
|
|
|
{
|
|
|
|
|
if (wv->lname && ! NILP (wv->lname))
|
|
|
|
|
{
|
|
|
|
|
wv->name = SDATA (wv->lname);
|
|
|
|
|
|
|
|
|
|
/* Ignore the @ that means "separate pane".
|
|
|
|
|
This is a kludge, but this isn't worth more time. */
|
|
|
|
|
if (wv->value == (char *)1)
|
|
|
|
|
{
|
|
|
|
|
if (wv->name[0] == '@')
|
|
|
|
|
wv->name++;
|
|
|
|
|
wv->value = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (wv->lkey && ! NILP (wv->lkey))
|
|
|
|
|
wv->key = SDATA (wv->lkey);
|
|
|
|
|
|
|
|
|
|
if (wv->contents)
|
|
|
|
|
update_submenu_strings (wv->contents);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* Set the contents of the menubar widgets of frame F.
|
|
|
|
|
The argument FIRST_TIME is currently ignored;
|
|
|
|
|
it is set the first time this is called, from initialize_frame_menubar. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
set_frame_menubar (f, first_time, deep_p)
|
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
int first_time;
|
|
|
|
|
int deep_p;
|
|
|
|
|
{
|
|
|
|
|
HMENU menubar_widget = f->output_data.w32->menubar_widget;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
Lisp_Object items;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
widget_value *wv, *first_wv, *prev_wv = 0;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
int i, last_i;
|
|
|
|
|
int *submenu_start, *submenu_end;
|
2003-01-20 20:18:13 +00:00
|
|
|
|
int *submenu_top_level_items, *submenu_n_panes;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* We must not change the menubar when actually in use. */
|
|
|
|
|
if (f->output_data.w32->menubar_active)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
XSETFRAME (Vmenu_updating_frame, f);
|
|
|
|
|
|
|
|
|
|
if (! menubar_widget)
|
|
|
|
|
deep_p = 1;
|
|
|
|
|
else if (pending_menu_activation && !deep_p)
|
|
|
|
|
deep_p = 1;
|
|
|
|
|
|
|
|
|
|
if (deep_p)
|
|
|
|
|
{
|
|
|
|
|
/* Make a widget-value tree representing the entire menu trees. */
|
|
|
|
|
|
|
|
|
|
struct buffer *prev = current_buffer;
|
|
|
|
|
Lisp_Object buffer;
|
2002-07-11 14:18:02 +00:00
|
|
|
|
int specpdl_count = SPECPDL_INDEX ();
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int previous_menu_items_used = f->menu_bar_items_used;
|
|
|
|
|
Lisp_Object *previous_items
|
|
|
|
|
= (Lisp_Object *) alloca (previous_menu_items_used
|
|
|
|
|
* sizeof (Lisp_Object));
|
|
|
|
|
|
|
|
|
|
/* If we are making a new widget, its contents are empty,
|
|
|
|
|
do always reinitialize them. */
|
|
|
|
|
if (! menubar_widget)
|
|
|
|
|
previous_menu_items_used = 0;
|
|
|
|
|
|
|
|
|
|
buffer = XWINDOW (FRAME_SELECTED_WINDOW (f))->buffer;
|
|
|
|
|
specbind (Qinhibit_quit, Qt);
|
|
|
|
|
/* Don't let the debugger step into this code
|
|
|
|
|
because it is not reentrant. */
|
|
|
|
|
specbind (Qdebug_on_next_call, Qnil);
|
|
|
|
|
|
2005-06-08 22:33:36 +00:00
|
|
|
|
record_unwind_save_match_data ();
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (Voverriding_local_map_menu_flag))
|
|
|
|
|
{
|
|
|
|
|
specbind (Qoverriding_terminal_local_map, Qnil);
|
|
|
|
|
specbind (Qoverriding_local_map, Qnil);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set_buffer_internal_1 (XBUFFER (buffer));
|
|
|
|
|
|
|
|
|
|
/* Run the Lucid hook. */
|
2001-02-23 10:15:27 +00:00
|
|
|
|
safe_run_hooks (Qactivate_menubar_hook);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* If it has changed current-menubar from previous value,
|
|
|
|
|
really recompute the menubar from the value. */
|
|
|
|
|
if (! NILP (Vlucid_menu_bar_dirty_flag))
|
|
|
|
|
call0 (Qrecompute_lucid_menubar);
|
|
|
|
|
safe_run_hooks (Qmenu_bar_update_hook);
|
|
|
|
|
FRAME_MENU_BAR_ITEMS (f) = menu_bar_items (FRAME_MENU_BAR_ITEMS (f));
|
|
|
|
|
|
|
|
|
|
items = FRAME_MENU_BAR_ITEMS (f);
|
|
|
|
|
|
|
|
|
|
/* Save the frame's previous menu bar contents data. */
|
2001-09-30 21:55:04 +00:00
|
|
|
|
if (previous_menu_items_used)
|
|
|
|
|
bcopy (XVECTOR (f->menu_bar_vector)->contents, previous_items,
|
|
|
|
|
previous_menu_items_used * sizeof (Lisp_Object));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
/* Fill in menu_items with the current menu bar contents.
|
|
|
|
|
This can evaluate Lisp code. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
menu_items = f->menu_bar_vector;
|
2001-09-30 21:55:04 +00:00
|
|
|
|
menu_items_allocated = VECTORP (menu_items) ? ASIZE (menu_items) : 0;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
submenu_start = (int *) alloca (XVECTOR (items)->size * sizeof (int *));
|
|
|
|
|
submenu_end = (int *) alloca (XVECTOR (items)->size * sizeof (int *));
|
2003-01-20 20:18:13 +00:00
|
|
|
|
submenu_n_panes = (int *) alloca (XVECTOR (items)->size * sizeof (int));
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
submenu_top_level_items
|
|
|
|
|
= (int *) alloca (XVECTOR (items)->size * sizeof (int *));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
init_menu_items ();
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
for (i = 0; i < ASIZE (items); i += 4)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object key, string, maps;
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
last_i = i;
|
|
|
|
|
|
|
|
|
|
key = AREF (items, i);
|
|
|
|
|
string = AREF (items, i + 1);
|
|
|
|
|
maps = AREF (items, i + 2);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (string))
|
|
|
|
|
break;
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
submenu_start[i] = menu_items_used;
|
|
|
|
|
|
|
|
|
|
menu_items_n_panes = 0;
|
|
|
|
|
submenu_top_level_items[i]
|
|
|
|
|
= parse_single_submenu (key, string, maps);
|
2003-01-20 20:18:13 +00:00
|
|
|
|
submenu_n_panes[i] = menu_items_n_panes;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
|
|
|
|
|
submenu_end[i] = menu_items_used;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
finish_menu_items ();
|
|
|
|
|
|
|
|
|
|
/* Convert menu_items into widget_value trees
|
|
|
|
|
to display the menu. This cannot evaluate Lisp code. */
|
|
|
|
|
|
|
|
|
|
wv = xmalloc_widget_value ();
|
|
|
|
|
wv->name = "menubar";
|
|
|
|
|
wv->value = 0;
|
|
|
|
|
wv->enabled = 1;
|
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
|
|
|
|
wv->help = Qnil;
|
|
|
|
|
first_wv = wv;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < last_i; i += 4)
|
|
|
|
|
{
|
2003-01-20 20:18:13 +00:00
|
|
|
|
menu_items_n_panes = submenu_n_panes[i];
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
wv = digest_single_submenu (submenu_start[i], submenu_end[i],
|
|
|
|
|
submenu_top_level_items[i]);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (prev_wv)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv->next = wv;
|
|
|
|
|
else
|
|
|
|
|
first_wv->contents = wv;
|
|
|
|
|
/* Don't set wv->name here; GC during the loop might relocate it. */
|
|
|
|
|
wv->enabled = 1;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv = wv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set_buffer_internal_1 (prev);
|
|
|
|
|
unbind_to (specpdl_count, Qnil);
|
|
|
|
|
|
|
|
|
|
/* If there has been no change in the Lisp-level contents
|
|
|
|
|
of the menu bar, skip redisplaying it. Just exit. */
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < previous_menu_items_used; i++)
|
|
|
|
|
if (menu_items_used == i
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
|| (!EQ (previous_items[i], AREF (menu_items, i))))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
break;
|
|
|
|
|
if (i == menu_items_used && i == previous_menu_items_used && i != 0)
|
|
|
|
|
{
|
|
|
|
|
free_menubar_widget_value_tree (first_wv);
|
|
|
|
|
menu_items = Qnil;
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Now GC cannot happen during the lifetime of the widget_value,
|
2001-12-01 11:16:54 +00:00
|
|
|
|
so it's safe to store data from a Lisp_String, as long as
|
|
|
|
|
local copies are made when the actual menu is created.
|
|
|
|
|
Windows takes care of this for normal string items, but
|
|
|
|
|
not for owner-drawn items or additional item-info. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv = first_wv->contents;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
for (i = 0; i < ASIZE (items); i += 4)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object string;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
string = AREF (items, i + 1);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (string))
|
|
|
|
|
break;
|
2002-07-15 00:01:34 +00:00
|
|
|
|
wv->name = (char *) SDATA (string);
|
2004-09-02 23:26:16 +00:00
|
|
|
|
update_submenu_strings (wv->contents);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv = wv->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f->menu_bar_vector = menu_items;
|
|
|
|
|
f->menu_bar_items_used = menu_items_used;
|
|
|
|
|
menu_items = Qnil;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Make a widget-value tree containing
|
2000-01-23 03:22:21 +00:00
|
|
|
|
just the top level menu bar strings. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
wv = xmalloc_widget_value ();
|
|
|
|
|
wv->name = "menubar";
|
|
|
|
|
wv->value = 0;
|
|
|
|
|
wv->enabled = 1;
|
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
|
|
|
|
wv->help = Qnil;
|
|
|
|
|
first_wv = wv;
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
items = FRAME_MENU_BAR_ITEMS (f);
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
for (i = 0; i < ASIZE (items); i += 4)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object string;
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
string = AREF (items, i + 1);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (NILP (string))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
wv = xmalloc_widget_value ();
|
2002-07-15 00:01:34 +00:00
|
|
|
|
wv->name = (char *) SDATA (string);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv->value = 0;
|
|
|
|
|
wv->enabled = 1;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* This prevents lwlib from assuming this
|
|
|
|
|
menu item is really supposed to be empty. */
|
|
|
|
|
/* The EMACS_INT cast avoids a warning.
|
|
|
|
|
This value just has to be different from small integers. */
|
|
|
|
|
wv->call_data = (void *) (EMACS_INT) (-1);
|
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (prev_wv)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv->next = wv;
|
|
|
|
|
else
|
|
|
|
|
first_wv->contents = wv;
|
|
|
|
|
prev_wv = wv;
|
|
|
|
|
}
|
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
/* Forget what we thought we knew about what is in the
|
|
|
|
|
detailed contents of the menu bar menus.
|
|
|
|
|
Changing the top level always destroys the contents. */
|
|
|
|
|
f->menu_bar_items_used = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Create or update the menu bar widget. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
BLOCK_INPUT;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menubar_widget)
|
|
|
|
|
{
|
|
|
|
|
/* Empty current menubar, rather than creating a fresh one. */
|
|
|
|
|
while (DeleteMenu (menubar_widget, 0, MF_BYPOSITION))
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
menubar_widget = CreateMenu ();
|
|
|
|
|
}
|
|
|
|
|
fill_in_menu (menubar_widget, first_wv->contents);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
free_menubar_widget_value_tree (first_wv);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
HMENU old_widget = f->output_data.w32->menubar_widget;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
f->output_data.w32->menubar_widget = menubar_widget;
|
|
|
|
|
SetMenu (FRAME_W32_WINDOW (f), f->output_data.w32->menubar_widget);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
/* Causes flicker when menu bar is updated
|
1998-04-23 23:43:02 +00:00
|
|
|
|
DrawMenuBar (FRAME_W32_WINDOW (f)); */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Force the window size to be recomputed so that the frame's text
|
|
|
|
|
area remains the same, if menubar has just been created. */
|
|
|
|
|
if (old_widget == NULL)
|
2003-05-24 22:07:03 +00:00
|
|
|
|
x_set_window_size (f, 0, FRAME_COLS (f), FRAME_LINES (f));
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
UNBLOCK_INPUT;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Called from Fx_create_frame to create the initial menubar of a frame
|
|
|
|
|
before it is mapped, so that the window is mapped with the menubar already
|
|
|
|
|
there instead of us tacking it on later and thrashing the window after it
|
|
|
|
|
is visible. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
initialize_frame_menubar (f)
|
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
{
|
|
|
|
|
/* This function is called before the first chance to redisplay
|
|
|
|
|
the frame. It has to be, so the frame will have the right size. */
|
|
|
|
|
FRAME_MENU_BAR_ITEMS (f) = menu_bar_items (FRAME_MENU_BAR_ITEMS (f));
|
|
|
|
|
set_frame_menubar (f, 1, 1);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Get rid of the menu bar of frame F, and free its storage.
|
|
|
|
|
This is used when deleting a frame, and when turning off the menu bar. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
free_frame_menubar (f)
|
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
{
|
|
|
|
|
BLOCK_INPUT;
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
HMENU old = GetMenu (FRAME_W32_WINDOW (f));
|
|
|
|
|
SetMenu (FRAME_W32_WINDOW (f), NULL);
|
|
|
|
|
f->output_data.w32->menubar_widget = NULL;
|
|
|
|
|
DestroyMenu (old);
|
|
|
|
|
}
|
2001-10-21 10:54:17 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
UNBLOCK_INPUT;
|
|
|
|
|
}
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* w32_menu_show actually displays a menu using the panes and items in
|
|
|
|
|
menu_items and returns the value selected from it; we assume input
|
|
|
|
|
is blocked by the caller. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
/* F is the frame the menu is for.
|
|
|
|
|
X and Y are the frame-relative specified position,
|
|
|
|
|
relative to the inside upper left corner of the frame F.
|
1998-04-23 23:43:02 +00:00
|
|
|
|
FOR_CLICK is nonzero if this menu was invoked for a mouse click.
|
1995-11-07 07:52:28 +00:00
|
|
|
|
KEYMAPS is 1 if this menu was specified with keymaps;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
in that case, we return a list containing the chosen item's value
|
|
|
|
|
and perhaps also the pane's prefix.
|
1995-11-07 07:52:28 +00:00
|
|
|
|
TITLE is the specified menu title.
|
|
|
|
|
ERROR is a place to store an error message string in case of failure.
|
|
|
|
|
(We return nil on failure, but the value doesn't actually matter.) */
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static Lisp_Object
|
|
|
|
|
w32_menu_show (f, x, y, for_click, keymaps, title, error)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
int x;
|
|
|
|
|
int y;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int for_click;
|
|
|
|
|
int keymaps;
|
|
|
|
|
Lisp_Object title;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
char **error;
|
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int i;
|
|
|
|
|
int menu_item_selection;
|
|
|
|
|
HMENU menu;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
POINT pos;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
widget_value *wv, *save_wv = 0, *first_wv = 0, *prev_wv = 0;
|
|
|
|
|
widget_value **submenu_stack
|
|
|
|
|
= (widget_value **) alloca (menu_items_used * sizeof (widget_value *));
|
|
|
|
|
Lisp_Object *subprefix_stack
|
|
|
|
|
= (Lisp_Object *) alloca (menu_items_used * sizeof (Lisp_Object));
|
|
|
|
|
int submenu_depth = 0;
|
|
|
|
|
int first_pane;
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
*error = NULL;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
if (menu_items_used <= MENU_ITEMS_PANE_LENGTH)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
*error = "Empty menu";
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* Create a tree of widget_value objects
|
|
|
|
|
representing the panes and their items. */
|
|
|
|
|
wv = xmalloc_widget_value ();
|
|
|
|
|
wv->name = "menu";
|
|
|
|
|
wv->value = 0;
|
|
|
|
|
wv->enabled = 1;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
2002-01-08 19:33:24 +00:00
|
|
|
|
wv->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
first_wv = wv;
|
|
|
|
|
first_pane = 1;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Loop over all panes and items, filling in the tree. */
|
|
|
|
|
i = 0;
|
|
|
|
|
while (i < menu_items_used)
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
if (EQ (AREF (menu_items, i), Qnil))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
submenu_stack[submenu_depth++] = save_wv;
|
|
|
|
|
save_wv = prev_wv;
|
|
|
|
|
prev_wv = 0;
|
|
|
|
|
first_pane = 1;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qlambda))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
prev_wv = save_wv;
|
|
|
|
|
save_wv = submenu_stack[--submenu_depth];
|
|
|
|
|
first_pane = 0;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qt)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
&& submenu_depth != 0)
|
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
/* Ignore a nil in the item list.
|
|
|
|
|
It's meaningful only for dialog boxes. */
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qquote))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
i += 1;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qt))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
/* Create a new pane. */
|
|
|
|
|
Lisp_Object pane_name, prefix;
|
|
|
|
|
char *pane_string;
|
2001-10-21 10:54:17 +00:00
|
|
|
|
pane_name = AREF (menu_items, i + MENU_ITEMS_PANE_NAME);
|
|
|
|
|
prefix = AREF (menu_items, i + MENU_ITEMS_PANE_PREFIX);
|
2004-09-02 23:26:16 +00:00
|
|
|
|
|
|
|
|
|
if (STRINGP (pane_name))
|
2001-10-21 10:54:17 +00:00
|
|
|
|
{
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
pane_name = ENCODE_UTF_8 (pane_name);
|
|
|
|
|
else if (STRING_MULTIBYTE (pane_name))
|
|
|
|
|
pane_name = ENCODE_SYSTEM (pane_name);
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, i + MENU_ITEMS_PANE_NAME, pane_name);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
}
|
2004-09-02 23:26:16 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
pane_string = (NILP (pane_name)
|
2002-07-15 00:01:34 +00:00
|
|
|
|
? "" : (char *) SDATA (pane_name));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* If there is just one top-level pane, put all its items directly
|
|
|
|
|
under the top-level menu. */
|
|
|
|
|
if (menu_items_n_panes == 1)
|
|
|
|
|
pane_string = "";
|
|
|
|
|
|
|
|
|
|
/* If the pane has a meaningful name,
|
|
|
|
|
make the pane a top-level menu item
|
|
|
|
|
with its items as a submenu beneath it. */
|
|
|
|
|
if (!keymaps && strcmp (pane_string, ""))
|
|
|
|
|
{
|
|
|
|
|
wv = xmalloc_widget_value ();
|
|
|
|
|
if (save_wv)
|
|
|
|
|
save_wv->next = wv;
|
|
|
|
|
else
|
|
|
|
|
first_wv->contents = wv;
|
|
|
|
|
wv->name = pane_string;
|
|
|
|
|
if (keymaps && !NILP (prefix))
|
|
|
|
|
wv->name++;
|
|
|
|
|
wv->value = 0;
|
|
|
|
|
wv->enabled = 1;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
save_wv = wv;
|
|
|
|
|
prev_wv = 0;
|
|
|
|
|
}
|
|
|
|
|
else if (first_pane)
|
|
|
|
|
{
|
|
|
|
|
save_wv = wv;
|
|
|
|
|
prev_wv = 0;
|
|
|
|
|
}
|
|
|
|
|
first_pane = 0;
|
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Create a new item within current pane. */
|
2000-02-27 21:38:00 +00:00
|
|
|
|
Lisp_Object item_name, enable, descrip, def, type, selected, help;
|
|
|
|
|
|
2001-10-21 10:54:17 +00:00
|
|
|
|
item_name = AREF (menu_items, i + MENU_ITEMS_ITEM_NAME);
|
|
|
|
|
enable = AREF (menu_items, i + MENU_ITEMS_ITEM_ENABLE);
|
|
|
|
|
descrip = AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY);
|
|
|
|
|
def = AREF (menu_items, i + MENU_ITEMS_ITEM_DEFINITION);
|
|
|
|
|
type = AREF (menu_items, i + MENU_ITEMS_ITEM_TYPE);
|
|
|
|
|
selected = AREF (menu_items, i + MENU_ITEMS_ITEM_SELECTED);
|
|
|
|
|
help = AREF (menu_items, i + MENU_ITEMS_ITEM_HELP);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (STRINGP (item_name))
|
2001-10-21 10:54:17 +00:00
|
|
|
|
{
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
item_name = ENCODE_UTF_8 (item_name);
|
|
|
|
|
else if (STRING_MULTIBYTE (item_name))
|
|
|
|
|
item_name = ENCODE_SYSTEM (item_name);
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, i + MENU_ITEMS_ITEM_NAME, item_name);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
}
|
2004-09-02 23:26:16 +00:00
|
|
|
|
|
|
|
|
|
if (STRINGP (descrip) && STRING_MULTIBYTE (descrip))
|
2001-10-21 10:54:17 +00:00
|
|
|
|
{
|
|
|
|
|
descrip = ENCODE_SYSTEM (descrip);
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
ASET (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY, descrip);
|
2001-10-21 10:54:17 +00:00
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
wv = xmalloc_widget_value ();
|
2003-02-04 14:56:31 +00:00
|
|
|
|
if (prev_wv)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv->next = wv;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
else
|
1998-04-23 23:43:02 +00:00
|
|
|
|
save_wv->contents = wv;
|
2002-07-15 00:01:34 +00:00
|
|
|
|
wv->name = (char *) SDATA (item_name);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (!NILP (descrip))
|
2002-07-15 00:01:34 +00:00
|
|
|
|
wv->key = (char *) SDATA (descrip);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv->value = 0;
|
|
|
|
|
/* Use the contents index as call_data, since we are
|
2001-11-18 22:01:13 +00:00
|
|
|
|
restricted to 16-bits. */
|
1998-07-11 21:52:17 +00:00
|
|
|
|
wv->call_data = !NILP (def) ? (void *) (EMACS_INT) i : 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv->enabled = !NILP (enable);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
|
|
|
|
if (NILP (type))
|
|
|
|
|
wv->button_type = BUTTON_TYPE_NONE;
|
|
|
|
|
else if (EQ (type, QCtoggle))
|
|
|
|
|
wv->button_type = BUTTON_TYPE_TOGGLE;
|
|
|
|
|
else if (EQ (type, QCradio))
|
|
|
|
|
wv->button_type = BUTTON_TYPE_RADIO;
|
|
|
|
|
else
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
wv->selected = !NILP (selected);
|
2001-12-09 16:48:03 +00:00
|
|
|
|
if (!STRINGP (help))
|
|
|
|
|
help = Qnil;
|
|
|
|
|
|
|
|
|
|
wv->help = help;
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv = wv;
|
|
|
|
|
|
|
|
|
|
i += MENU_ITEMS_ITEM_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Deal with the title, if it is non-nil. */
|
|
|
|
|
if (!NILP (title))
|
|
|
|
|
{
|
|
|
|
|
widget_value *wv_title = xmalloc_widget_value ();
|
|
|
|
|
widget_value *wv_sep = xmalloc_widget_value ();
|
|
|
|
|
|
|
|
|
|
/* Maybe replace this separator with a bitmap or owner-draw item
|
|
|
|
|
so that it looks better. Having two separators looks odd. */
|
|
|
|
|
wv_sep->name = "--";
|
|
|
|
|
wv_sep->next = first_wv->contents;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv_sep->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
title = ENCODE_UTF_8 (title);
|
|
|
|
|
else if (STRING_MULTIBYTE (title))
|
2000-05-30 21:52:06 +00:00
|
|
|
|
title = ENCODE_SYSTEM (title);
|
2004-09-02 23:26:16 +00:00
|
|
|
|
|
2002-07-15 00:01:34 +00:00
|
|
|
|
wv_title->name = (char *) SDATA (title);
|
2000-10-27 22:20:19 +00:00
|
|
|
|
wv_title->enabled = TRUE;
|
|
|
|
|
wv_title->title = TRUE;
|
2000-01-23 03:22:21 +00:00
|
|
|
|
wv_title->button_type = BUTTON_TYPE_NONE;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv_title->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
wv_title->next = wv_sep;
|
|
|
|
|
first_wv->contents = wv_title;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Actually create the menu. */
|
2001-12-01 11:16:54 +00:00
|
|
|
|
current_popup_menu = menu = CreatePopupMenu ();
|
1998-04-23 23:43:02 +00:00
|
|
|
|
fill_in_menu (menu, first_wv->contents);
|
2000-01-23 03:22:21 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
/* Adjust coordinates to be root-window-relative. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
pos.x = x;
|
|
|
|
|
pos.y = y;
|
1996-11-19 04:59:23 +00:00
|
|
|
|
ClientToScreen (FRAME_W32_WINDOW (f), &pos);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
/* No selection has been chosen yet. */
|
|
|
|
|
menu_item_selection = 0;
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Display the menu. */
|
2003-02-04 14:56:31 +00:00
|
|
|
|
menu_item_selection = SendMessage (FRAME_W32_WINDOW (f),
|
1998-04-23 23:43:02 +00:00
|
|
|
|
WM_EMACS_TRACKPOPUPMENU,
|
|
|
|
|
(WPARAM)menu, (LPARAM)&pos);
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
|
|
|
|
/* Clean up extraneous mouse events which might have been generated
|
|
|
|
|
during the call. */
|
|
|
|
|
discard_mouse_events ();
|
|
|
|
|
|
2000-07-15 11:52:21 +00:00
|
|
|
|
/* Free the widget_value objects we used to specify the contents. */
|
|
|
|
|
free_menubar_widget_value_tree (first_wv);
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
DestroyMenu (menu);
|
|
|
|
|
|
2007-02-22 23:15:08 +00:00
|
|
|
|
/* Free the owner-drawn and help-echo menu strings. */
|
|
|
|
|
w32_free_menu_strings (FRAME_W32_WINDOW (f));
|
2007-02-22 23:47:43 +00:00
|
|
|
|
f->output_data.w32->menubar_active = 0;
|
2007-02-22 23:15:08 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Find the selected item, and its pane, to return
|
|
|
|
|
the proper value. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menu_item_selection != 0)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
1998-04-23 23:43:02 +00:00
|
|
|
|
Lisp_Object prefix, entry;
|
|
|
|
|
|
2001-07-11 18:31:41 +00:00
|
|
|
|
prefix = entry = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
i = 0;
|
|
|
|
|
while (i < menu_items_used)
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
if (EQ (AREF (menu_items, i), Qnil))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
subprefix_stack[submenu_depth++] = prefix;
|
|
|
|
|
prefix = entry;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qlambda))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
|
|
|
|
prefix = subprefix_stack[--submenu_depth];
|
|
|
|
|
i++;
|
|
|
|
|
}
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qt))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
prefix = AREF (menu_items, i + MENU_ITEMS_PANE_PREFIX);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
/* Ignore a nil in the item list.
|
|
|
|
|
It's meaningful only for dialog boxes. */
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
else if (EQ (AREF (menu_items, i), Qquote))
|
1998-04-23 23:43:02 +00:00
|
|
|
|
i += 1;
|
|
|
|
|
else
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
entry = AREF (menu_items, i + MENU_ITEMS_ITEM_VALUE);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menu_item_selection == i)
|
|
|
|
|
{
|
|
|
|
|
if (keymaps != 0)
|
|
|
|
|
{
|
|
|
|
|
int j;
|
|
|
|
|
|
|
|
|
|
entry = Fcons (entry, Qnil);
|
|
|
|
|
if (!NILP (prefix))
|
|
|
|
|
entry = Fcons (prefix, entry);
|
|
|
|
|
for (j = submenu_depth - 1; j >= 0; j--)
|
|
|
|
|
if (!NILP (subprefix_stack[j]))
|
|
|
|
|
entry = Fcons (subprefix_stack[j], entry);
|
|
|
|
|
}
|
|
|
|
|
return entry;
|
|
|
|
|
}
|
|
|
|
|
i += MENU_ITEMS_ITEM_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
}
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
2006-08-03 22:08:09 +00:00
|
|
|
|
else if (!for_click)
|
|
|
|
|
/* Make "Cancel" equivalent to C-g. */
|
|
|
|
|
Fsignal (Qquit, Qnil);
|
1997-09-03 00:53:34 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
2001-12-01 11:16:54 +00:00
|
|
|
|
#ifdef HAVE_DIALOGS
|
2008-04-10 15:20:07 +00:00
|
|
|
|
/* TODO: On Windows, there are two ways of defining a dialog.
|
|
|
|
|
|
|
|
|
|
1. Create a predefined dialog resource and include it in nt/emacs.rc.
|
|
|
|
|
Using this method, we could then set the titles and make unneeded
|
|
|
|
|
buttons invisible before displaying the dialog. Everything would
|
|
|
|
|
be a fixed size though, so there is a risk that text does not
|
|
|
|
|
fit on a button.
|
|
|
|
|
2. Create the dialog template in memory on the fly. This allows us
|
|
|
|
|
to size the dialog and buttons dynamically, probably giving more
|
|
|
|
|
natural looking results for dialogs with few buttons, and eliminating
|
|
|
|
|
the problem of text overflowing the buttons. But the API for this is
|
|
|
|
|
quite complex - structures have to be allocated in particular ways,
|
|
|
|
|
text content is tacked onto the end of structures in variable length
|
|
|
|
|
arrays with further structures tacked on after these, there are
|
|
|
|
|
certain alignment requirements for all this, and we have to
|
|
|
|
|
measure all the text and convert to "dialog coordinates" to figure
|
|
|
|
|
out how big to make everything.
|
|
|
|
|
|
|
|
|
|
For now, we'll just stick with menus for dialogs that are more
|
|
|
|
|
complicated than simple yes/no type questions for which we can use
|
|
|
|
|
the MessageBox function.
|
|
|
|
|
*/
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static char * button_names [] = {
|
1995-11-07 07:52:28 +00:00
|
|
|
|
"button1", "button2", "button3", "button4", "button5",
|
1998-04-23 23:43:02 +00:00
|
|
|
|
"button6", "button7", "button8", "button9", "button10" };
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
static Lisp_Object
|
2005-05-24 08:38:58 +00:00
|
|
|
|
w32_dialog_show (f, keymaps, title, header, error)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
int keymaps;
|
2005-05-24 08:38:58 +00:00
|
|
|
|
Lisp_Object title, header;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
char **error;
|
|
|
|
|
{
|
|
|
|
|
int i, nb_buttons=0;
|
|
|
|
|
char dialog_name[6];
|
1998-04-23 23:43:02 +00:00
|
|
|
|
int menu_item_selection;
|
|
|
|
|
|
2001-07-11 18:31:41 +00:00
|
|
|
|
widget_value *wv, *first_wv = 0, *prev_wv = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Number of elements seen so far, before boundary. */
|
|
|
|
|
int left_count = 0;
|
|
|
|
|
/* 1 means we've seen the boundary between left-hand elts and right-hand. */
|
|
|
|
|
int boundary_seen = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
*error = NULL;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (menu_items_n_panes > 1)
|
|
|
|
|
{
|
|
|
|
|
*error = "Multiple panes in dialog box";
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Create a tree of widget_value objects
|
|
|
|
|
representing the text label and buttons. */
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object pane_name, prefix;
|
|
|
|
|
char *pane_string;
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
pane_name = AREF (menu_items, MENU_ITEMS_PANE_NAME);
|
|
|
|
|
prefix = AREF (menu_items, MENU_ITEMS_PANE_PREFIX);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
pane_string = (NILP (pane_name)
|
2003-02-04 14:56:31 +00:00
|
|
|
|
? "" : (char *) SDATA (pane_name));
|
1998-04-23 23:43:02 +00:00
|
|
|
|
prev_wv = xmalloc_widget_value ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
prev_wv->value = pane_string;
|
|
|
|
|
if (keymaps && !NILP (prefix))
|
|
|
|
|
prev_wv->name++;
|
|
|
|
|
prev_wv->enabled = 1;
|
|
|
|
|
prev_wv->name = "message";
|
2002-01-07 00:26:57 +00:00
|
|
|
|
prev_wv->help = Qnil;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
first_wv = prev_wv;
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Loop over all panes and items, filling in the tree. */
|
|
|
|
|
i = MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
while (i < menu_items_used)
|
|
|
|
|
{
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Create a new item within current pane. */
|
2000-02-27 21:38:00 +00:00
|
|
|
|
Lisp_Object item_name, enable, descrip, help;
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
item_name = AREF (menu_items, i + MENU_ITEMS_ITEM_NAME);
|
|
|
|
|
enable = AREF (menu_items, i + MENU_ITEMS_ITEM_ENABLE);
|
|
|
|
|
descrip = AREF (menu_items, i + MENU_ITEMS_ITEM_EQUIV_KEY);
|
|
|
|
|
help = AREF (menu_items, i + MENU_ITEMS_ITEM_HELP);
|
2003-02-04 14:56:31 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (NILP (item_name))
|
|
|
|
|
{
|
|
|
|
|
free_menubar_widget_value_tree (first_wv);
|
|
|
|
|
*error = "Submenu in dialog items";
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
if (EQ (item_name, Qquote))
|
|
|
|
|
{
|
|
|
|
|
/* This is the boundary between left-side elts
|
|
|
|
|
and right-side elts. Stop incrementing right_count. */
|
|
|
|
|
boundary_seen = 1;
|
|
|
|
|
i++;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (nb_buttons >= 9)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
free_menubar_widget_value_tree (first_wv);
|
|
|
|
|
*error = "Too many dialog items";
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
wv = xmalloc_widget_value ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
prev_wv->next = wv;
|
|
|
|
|
wv->name = (char *) button_names[nb_buttons];
|
|
|
|
|
if (!NILP (descrip))
|
2002-07-15 00:01:34 +00:00
|
|
|
|
wv->key = (char *) SDATA (descrip);
|
|
|
|
|
wv->value = (char *) SDATA (item_name);
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
wv->call_data = (void *) &AREF (menu_items, i);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
wv->enabled = !NILP (enable);
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv->help = Qnil;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
prev_wv = wv;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
if (! boundary_seen)
|
|
|
|
|
left_count++;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
nb_buttons++;
|
|
|
|
|
i += MENU_ITEMS_ITEM_LENGTH;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* If the boundary was not specified,
|
|
|
|
|
by default put half on the left and half on the right. */
|
|
|
|
|
if (! boundary_seen)
|
|
|
|
|
left_count = nb_buttons - nb_buttons / 2;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
wv = xmalloc_widget_value ();
|
1995-11-07 07:52:28 +00:00
|
|
|
|
wv->name = dialog_name;
|
2002-01-07 00:26:57 +00:00
|
|
|
|
wv->help = Qnil;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
2005-05-24 08:38:58 +00:00
|
|
|
|
/* Frame title: 'Q' = Question, 'I' = Information.
|
|
|
|
|
Can also have 'E' = Error if, one day, we want
|
|
|
|
|
a popup for errors. */
|
|
|
|
|
if (NILP(header))
|
|
|
|
|
dialog_name[0] = 'Q';
|
|
|
|
|
else
|
|
|
|
|
dialog_name[0] = 'I';
|
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Dialog boxes use a really stupid name encoding
|
|
|
|
|
which specifies how many buttons to use
|
2005-05-24 08:38:58 +00:00
|
|
|
|
and how many buttons are on the right. */
|
1995-11-07 07:52:28 +00:00
|
|
|
|
dialog_name[1] = '0' + nb_buttons;
|
|
|
|
|
dialog_name[2] = 'B';
|
|
|
|
|
dialog_name[3] = 'R';
|
|
|
|
|
/* Number of buttons to put on the right. */
|
|
|
|
|
dialog_name[4] = '0' + nb_buttons - left_count;
|
|
|
|
|
dialog_name[5] = 0;
|
|
|
|
|
wv->contents = first_wv;
|
|
|
|
|
first_wv = wv;
|
|
|
|
|
}
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Actually create the dialog. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
dialog_id = widget_id_tick++;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
menu = lw_create_widget (first_wv->name, "dialog", dialog_id, first_wv,
|
1996-11-19 04:59:23 +00:00
|
|
|
|
f->output_data.w32->widget, 1, 0,
|
1995-11-07 07:52:28 +00:00
|
|
|
|
dialog_selection_callback, 0);
|
2000-10-27 22:20:19 +00:00
|
|
|
|
lw_modify_all_widgets (dialog_id, first_wv->contents, TRUE);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Free the widget_value objects we used to specify the contents. */
|
|
|
|
|
free_menubar_widget_value_tree (first_wv);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* No selection has been chosen yet. */
|
|
|
|
|
menu_item_selection = 0;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Display the menu. */
|
|
|
|
|
lw_pop_up_all_widgets (dialog_id);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
/* Process events that apply to the menu. */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
popup_get_selection ((XEvent *) 0, FRAME_X_DISPLAY_INFO (f), dialog_id);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
2003-02-04 14:56:31 +00:00
|
|
|
|
lw_destroy_all_widgets (dialog_id);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
|
|
|
|
/* Find the selected item, and its pane, to return
|
|
|
|
|
the proper value. */
|
|
|
|
|
if (menu_item_selection != 0)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object prefix;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
prefix = Qnil;
|
|
|
|
|
i = 0;
|
|
|
|
|
while (i < menu_items_used)
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object entry;
|
|
|
|
|
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
if (EQ (AREF (menu_items, i), Qt))
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
prefix = AREF (menu_items, i + MENU_ITEMS_PANE_PREFIX);
|
1995-11-07 07:52:28 +00:00
|
|
|
|
i += MENU_ITEMS_PANE_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
entry = AREF (menu_items, i + MENU_ITEMS_ITEM_VALUE);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
if (menu_item_selection == i)
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
|
|
|
|
if (keymaps != 0)
|
|
|
|
|
{
|
|
|
|
|
entry = Fcons (entry, Qnil);
|
|
|
|
|
if (!NILP (prefix))
|
|
|
|
|
entry = Fcons (prefix, entry);
|
|
|
|
|
}
|
|
|
|
|
return entry;
|
|
|
|
|
}
|
|
|
|
|
i += MENU_ITEMS_ITEM_LENGTH;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-08-03 22:08:09 +00:00
|
|
|
|
else
|
|
|
|
|
/* Make "Cancel" equivalent to C-g. */
|
|
|
|
|
Fsignal (Qquit, Qnil);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
2008-04-10 15:20:07 +00:00
|
|
|
|
#else /* !HAVE_DIALOGS */
|
|
|
|
|
|
|
|
|
|
/* Currently we only handle Yes No dialogs (y-or-n-p and yes-or-no-p) as
|
|
|
|
|
simple dialogs. We could handle a few more, but I'm not aware of
|
|
|
|
|
anywhere in Emacs that uses the other specific dialog choices that
|
|
|
|
|
MessageBox provides. */
|
|
|
|
|
|
|
|
|
|
static int is_simple_dialog (contents)
|
|
|
|
|
Lisp_Object contents;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object options = XCDR (contents);
|
|
|
|
|
Lisp_Object name, yes, no, other;
|
|
|
|
|
|
|
|
|
|
yes = build_string ("Yes");
|
|
|
|
|
no = build_string ("No");
|
|
|
|
|
|
|
|
|
|
if (!CONSP (options))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
name = XCAR (XCAR (options));
|
|
|
|
|
if (!CONSP (options))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (!NILP (Fstring_equal (name, yes)))
|
|
|
|
|
other = no;
|
|
|
|
|
else if (!NILP (Fstring_equal (name, no)))
|
|
|
|
|
other = yes;
|
|
|
|
|
else
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
options = XCDR (options);
|
|
|
|
|
if (!CONSP (options))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
name = XCAR (XCAR (options));
|
|
|
|
|
if (NILP (Fstring_equal (name, other)))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Check there are no more options. */
|
|
|
|
|
options = XCDR (options);
|
|
|
|
|
return !(CONSP (options));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static Lisp_Object simple_dialog_show (f, contents, header)
|
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
Lisp_Object contents, header;
|
|
|
|
|
{
|
|
|
|
|
int answer;
|
|
|
|
|
UINT type;
|
|
|
|
|
char *text, *title;
|
|
|
|
|
Lisp_Object lispy_answer = Qnil, temp = XCAR (contents);
|
|
|
|
|
|
|
|
|
|
if (STRINGP (temp))
|
|
|
|
|
text = SDATA (temp);
|
|
|
|
|
else
|
|
|
|
|
text = "";
|
|
|
|
|
|
|
|
|
|
if (NILP (header))
|
|
|
|
|
{
|
|
|
|
|
title = "Question";
|
|
|
|
|
type = MB_ICONQUESTION;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
title = "Information";
|
|
|
|
|
type = MB_ICONINFORMATION;
|
|
|
|
|
}
|
|
|
|
|
type |= MB_YESNO;
|
|
|
|
|
|
|
|
|
|
/* Since we only handle Yes/No dialogs, and we already checked
|
|
|
|
|
is_simple_dialog, we don't need to worry about checking contents
|
|
|
|
|
to see what type of dialog to use. */
|
|
|
|
|
answer = MessageBox (FRAME_W32_WINDOW (f), text, title, type);
|
|
|
|
|
|
|
|
|
|
if (answer == IDYES)
|
|
|
|
|
lispy_answer = build_string ("Yes");
|
|
|
|
|
else if (answer == IDNO)
|
|
|
|
|
lispy_answer = build_string ("No");
|
|
|
|
|
else
|
|
|
|
|
Fsignal (Qquit, Qnil);
|
|
|
|
|
|
|
|
|
|
for (temp = XCDR (contents); CONSP (temp); temp = XCDR (temp))
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object item, name, value;
|
|
|
|
|
item = XCAR (temp);
|
|
|
|
|
if (CONSP (item))
|
|
|
|
|
{
|
|
|
|
|
name = XCAR (item);
|
|
|
|
|
value = XCDR (item);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
name = item;
|
|
|
|
|
value = Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!NILP (Fstring_equal (name, lispy_answer)))
|
|
|
|
|
{
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Fsignal (Qquit, Qnil);
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
#endif /* !HAVE_DIALOGS */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Is this item a separator? */
|
|
|
|
|
static int
|
|
|
|
|
name_is_separator (name)
|
|
|
|
|
char *name;
|
|
|
|
|
{
|
2001-07-11 18:31:41 +00:00
|
|
|
|
char *start = name;
|
|
|
|
|
|
|
|
|
|
/* Check if name string consists of only dashes ('-'). */
|
1998-04-23 23:43:02 +00:00
|
|
|
|
while (*name == '-') name++;
|
2001-07-11 18:31:41 +00:00
|
|
|
|
/* Separators can also be of the form "--:TripleSuperMegaEtched"
|
|
|
|
|
or "--deep-shadow". We don't implement them yet, se we just treat
|
|
|
|
|
them like normal separators. */
|
|
|
|
|
return (*name == '\0' || start + 2 == name);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Indicate boundary between left and right. */
|
|
|
|
|
static int
|
|
|
|
|
add_left_right_boundary (HMENU menu)
|
|
|
|
|
{
|
|
|
|
|
return AppendMenu (menu, MF_MENUBARBREAK, 0, NULL);
|
|
|
|
|
}
|
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
/* UTF8: 0xxxxxxx, 110xxxxx 10xxxxxx, 1110xxxx, 10xxxxxx, 10xxxxxx */
|
|
|
|
|
static void
|
|
|
|
|
utf8to16 (unsigned char * src, int len, WCHAR * dest)
|
|
|
|
|
{
|
|
|
|
|
while (len > 0)
|
|
|
|
|
{
|
|
|
|
|
int utf16;
|
|
|
|
|
if (*src < 0x80)
|
|
|
|
|
{
|
|
|
|
|
*dest = (WCHAR) *src;
|
|
|
|
|
dest++; src++; len--;
|
|
|
|
|
}
|
|
|
|
|
/* Since we might get >3 byte sequences which we don't handle, ignore the extra parts. */
|
|
|
|
|
else if (*src < 0xC0)
|
|
|
|
|
{
|
|
|
|
|
src++; len--;
|
|
|
|
|
}
|
|
|
|
|
/* 2 char UTF-8 sequence. */
|
|
|
|
|
else if (*src < 0xE0)
|
|
|
|
|
{
|
|
|
|
|
*dest = (WCHAR) (((*src & 0x1f) << 6)
|
|
|
|
|
| (*(src + 1) & 0x3f));
|
|
|
|
|
src += 2; len -= 2; dest++;
|
|
|
|
|
}
|
|
|
|
|
else if (*src < 0xF0)
|
|
|
|
|
{
|
|
|
|
|
*dest = (WCHAR) (((*src & 0x0f) << 12)
|
|
|
|
|
| ((*(src + 1) & 0x3f) << 6)
|
|
|
|
|
| (*(src + 2) & 0x3f));
|
|
|
|
|
src += 3; len -= 3; dest++;
|
|
|
|
|
}
|
|
|
|
|
else /* Not encodable. Insert Unicode Substitution char. */
|
|
|
|
|
{
|
|
|
|
|
*dest = (WCHAR) 0xfffd;
|
|
|
|
|
src++; len--; dest++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*dest = 0;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
static int
|
|
|
|
|
add_menu_item (HMENU menu, widget_value *wv, HMENU item)
|
|
|
|
|
{
|
|
|
|
|
UINT fuFlags;
|
2007-06-16 18:13:15 +00:00
|
|
|
|
char *out_string, *p, *q;
|
2000-03-01 20:49:16 +00:00
|
|
|
|
int return_value;
|
2007-06-16 18:13:15 +00:00
|
|
|
|
size_t nlen, orig_len;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
|
|
|
|
|
if (name_is_separator (wv->name))
|
2000-11-28 20:28:02 +00:00
|
|
|
|
{
|
|
|
|
|
fuFlags = MF_SEPARATOR;
|
|
|
|
|
out_string = NULL;
|
|
|
|
|
}
|
2003-02-04 14:56:31 +00:00
|
|
|
|
else
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
1998-07-13 19:54:15 +00:00
|
|
|
|
if (wv->enabled)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
fuFlags = MF_STRING;
|
|
|
|
|
else
|
|
|
|
|
fuFlags = MF_STRING | MF_GRAYED;
|
|
|
|
|
|
|
|
|
|
if (wv->key != NULL)
|
|
|
|
|
{
|
|
|
|
|
out_string = alloca (strlen (wv->name) + strlen (wv->key) + 2);
|
|
|
|
|
strcpy (out_string, wv->name);
|
|
|
|
|
strcat (out_string, "\t");
|
|
|
|
|
strcat (out_string, wv->key);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
out_string = wv->name;
|
|
|
|
|
|
2007-06-16 18:13:15 +00:00
|
|
|
|
/* Quote any special characters within the menu item's text and
|
|
|
|
|
key binding. */
|
|
|
|
|
nlen = orig_len = strlen (out_string);
|
2007-06-17 22:11:16 +00:00
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
{
|
|
|
|
|
/* With UTF-8, & cannot be part of a multibyte character. */
|
|
|
|
|
for (p = out_string; *p; p++)
|
|
|
|
|
{
|
|
|
|
|
if (*p == '&')
|
|
|
|
|
nlen++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* If encoded with the system codepage, use multibyte string
|
|
|
|
|
functions in case of multibyte characters that contain '&'. */
|
|
|
|
|
for (p = out_string; *p; p = _mbsinc (p))
|
|
|
|
|
{
|
|
|
|
|
if (_mbsnextc (p) == '&')
|
|
|
|
|
nlen++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2007-06-16 18:13:15 +00:00
|
|
|
|
if (nlen > orig_len)
|
2007-06-17 22:11:16 +00:00
|
|
|
|
{
|
|
|
|
|
p = out_string;
|
|
|
|
|
out_string = alloca (nlen + 1);
|
|
|
|
|
q = out_string;
|
|
|
|
|
while (*p)
|
|
|
|
|
{
|
|
|
|
|
if (unicode_append_menu)
|
|
|
|
|
{
|
|
|
|
|
if (*p == '&')
|
|
|
|
|
*q++ = *p;
|
|
|
|
|
*q++ = *p++;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (_mbsnextc (p) == '&')
|
|
|
|
|
{
|
|
|
|
|
_mbsncpy (q, p, 1);
|
|
|
|
|
q = _mbsinc (q);
|
|
|
|
|
}
|
|
|
|
|
_mbsncpy (q, p, 1);
|
|
|
|
|
p = _mbsinc (p);
|
|
|
|
|
q = _mbsinc (q);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*q = '\0';
|
|
|
|
|
}
|
2007-06-16 18:13:15 +00:00
|
|
|
|
|
2001-12-09 16:48:03 +00:00
|
|
|
|
if (item != NULL)
|
|
|
|
|
fuFlags = MF_POPUP;
|
|
|
|
|
else if (wv->title || wv->call_data == 0)
|
1998-04-23 23:43:02 +00:00
|
|
|
|
{
|
2001-12-01 11:16:54 +00:00
|
|
|
|
/* Only use MF_OWNERDRAW if GetMenuItemInfo is usable, since
|
|
|
|
|
we can't deallocate the memory otherwise. */
|
|
|
|
|
if (get_menu_item_info)
|
|
|
|
|
{
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
out_string = (char *) local_alloc (strlen (wv->name) + 1);
|
|
|
|
|
strcpy (out_string, wv->name);
|
2001-12-09 16:48:03 +00:00
|
|
|
|
#ifdef MENU_DEBUG
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
DebPrint ("Menu: allocing %ld for owner-draw", out_string);
|
2001-12-09 16:48:03 +00:00
|
|
|
|
#endif
|
2001-12-01 11:16:54 +00:00
|
|
|
|
fuFlags = MF_OWNERDRAW | MF_DISABLED;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
fuFlags = MF_DISABLED;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
2001-11-28 20:31:43 +00:00
|
|
|
|
|
2000-01-23 03:22:21 +00:00
|
|
|
|
/* Draw radio buttons and tickboxes. */
|
2001-03-08 21:21:44 +00:00
|
|
|
|
else if (wv->selected && (wv->button_type == BUTTON_TYPE_TOGGLE ||
|
2001-11-15 22:53:54 +00:00
|
|
|
|
wv->button_type == BUTTON_TYPE_RADIO))
|
2001-03-08 21:21:44 +00:00
|
|
|
|
fuFlags |= MF_CHECKED;
|
2000-03-01 20:49:16 +00:00
|
|
|
|
else
|
2001-03-08 21:21:44 +00:00
|
|
|
|
fuFlags |= MF_UNCHECKED;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
2001-03-08 21:21:44 +00:00
|
|
|
|
|
2004-09-02 23:26:16 +00:00
|
|
|
|
if (unicode_append_menu && out_string)
|
|
|
|
|
{
|
|
|
|
|
/* Convert out_string from UTF-8 to UTF-16-LE. */
|
|
|
|
|
int utf8_len = strlen (out_string);
|
|
|
|
|
WCHAR * utf16_string;
|
|
|
|
|
if (fuFlags & MF_OWNERDRAW)
|
|
|
|
|
utf16_string = local_alloc ((utf8_len + 1) * sizeof (WCHAR));
|
|
|
|
|
else
|
|
|
|
|
utf16_string = alloca ((utf8_len + 1) * sizeof (WCHAR));
|
|
|
|
|
|
|
|
|
|
utf8to16 (out_string, utf8_len, utf16_string);
|
|
|
|
|
return_value = unicode_append_menu (menu, fuFlags,
|
|
|
|
|
item != NULL ? (UINT) item
|
|
|
|
|
: (UINT) wv->call_data,
|
|
|
|
|
utf16_string);
|
2005-08-31 15:22:15 +00:00
|
|
|
|
if (!return_value)
|
|
|
|
|
{
|
|
|
|
|
/* On W9x/ME, unicode menus are not supported, though AppendMenuW
|
|
|
|
|
apparently does exist at least in some cases and appears to be
|
|
|
|
|
stubbed out to do nothing. out_string is UTF-8, but since
|
|
|
|
|
our standard menus are in English and this is only going to
|
|
|
|
|
happen the first time a menu is used, the encoding is
|
|
|
|
|
of minor importance compared with menus not working at all. */
|
|
|
|
|
return_value =
|
|
|
|
|
AppendMenu (menu, fuFlags,
|
|
|
|
|
item != NULL ? (UINT) item: (UINT) wv->call_data,
|
|
|
|
|
out_string);
|
|
|
|
|
/* Don't use unicode menus in future. */
|
|
|
|
|
unicode_append_menu = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (unicode_append_menu && (fuFlags & MF_OWNERDRAW))
|
2004-09-02 23:26:16 +00:00
|
|
|
|
local_free (out_string);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return_value =
|
|
|
|
|
AppendMenu (menu,
|
|
|
|
|
fuFlags,
|
|
|
|
|
item != NULL ? (UINT) item : (UINT) wv->call_data,
|
|
|
|
|
out_string );
|
|
|
|
|
}
|
2000-03-01 20:49:16 +00:00
|
|
|
|
|
|
|
|
|
/* This must be done after the menu item is created. */
|
2001-11-15 22:53:54 +00:00
|
|
|
|
if (!wv->title && wv->call_data != 0)
|
2001-03-08 21:21:44 +00:00
|
|
|
|
{
|
|
|
|
|
if (set_menu_item_info)
|
|
|
|
|
{
|
|
|
|
|
MENUITEMINFO info;
|
|
|
|
|
bzero (&info, sizeof (info));
|
|
|
|
|
info.cbSize = sizeof (info);
|
|
|
|
|
info.fMask = MIIM_DATA;
|
|
|
|
|
|
2001-12-09 16:48:03 +00:00
|
|
|
|
/* Set help string for menu item. Leave it as a Lisp_Object
|
|
|
|
|
until it is ready to be displayed, since GC can happen while
|
|
|
|
|
menus are active. */
|
2004-05-18 19:53:08 +00:00
|
|
|
|
if (!NILP (wv->help))
|
|
|
|
|
#ifdef USE_LISP_UNION_TYPE
|
|
|
|
|
info.dwItemData = (DWORD) (wv->help).i;
|
|
|
|
|
#else
|
|
|
|
|
info.dwItemData = (DWORD) (wv->help);
|
|
|
|
|
#endif
|
2001-03-08 21:21:44 +00:00
|
|
|
|
if (wv->button_type == BUTTON_TYPE_RADIO)
|
|
|
|
|
{
|
|
|
|
|
/* CheckMenuRadioItem allows us to differentiate TOGGLE and
|
|
|
|
|
RADIO items, but is not available on NT 3.51 and earlier. */
|
|
|
|
|
info.fMask |= MIIM_TYPE | MIIM_STATE;
|
|
|
|
|
info.fType = MFT_RADIOCHECK | MFT_STRING;
|
|
|
|
|
info.dwTypeData = out_string;
|
|
|
|
|
info.fState = wv->selected ? MFS_CHECKED : MFS_UNCHECKED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set_menu_item_info (menu,
|
|
|
|
|
item != NULL ? (UINT) item : (UINT) wv->call_data,
|
|
|
|
|
FALSE, &info);
|
|
|
|
|
}
|
|
|
|
|
}
|
2000-03-01 20:49:16 +00:00
|
|
|
|
return return_value;
|
1998-04-23 23:43:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Construct native Windows menu(bar) based on widget_value tree. */
|
2000-08-22 23:37:05 +00:00
|
|
|
|
int
|
1998-04-23 23:43:02 +00:00
|
|
|
|
fill_in_menu (HMENU menu, widget_value *wv)
|
|
|
|
|
{
|
|
|
|
|
int items_added = 0;
|
1995-11-07 07:52:28 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
for ( ; wv != NULL; wv = wv->next)
|
|
|
|
|
{
|
|
|
|
|
if (wv->contents)
|
|
|
|
|
{
|
|
|
|
|
HMENU sub_menu = CreatePopupMenu ();
|
|
|
|
|
|
|
|
|
|
if (sub_menu == NULL)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (!fill_in_menu (sub_menu, wv->contents) ||
|
|
|
|
|
!add_menu_item (menu, wv, sub_menu))
|
|
|
|
|
{
|
|
|
|
|
DestroyMenu (sub_menu);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (!add_menu_item (menu, wv, NULL))
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-22 23:15:30 +00:00
|
|
|
|
/* Display help string for currently pointed to menu item. Not
|
|
|
|
|
supported on NT 3.51 and earlier, as GetMenuItemInfo is not
|
|
|
|
|
available. */
|
|
|
|
|
void
|
2001-11-28 22:25:51 +00:00
|
|
|
|
w32_menu_display_help (HWND owner, HMENU menu, UINT item, UINT flags)
|
2000-03-22 23:15:30 +00:00
|
|
|
|
{
|
|
|
|
|
if (get_menu_item_info)
|
|
|
|
|
{
|
2001-11-28 20:31:43 +00:00
|
|
|
|
struct frame *f = x_window_to_frame (&one_w32_display_info, owner);
|
|
|
|
|
Lisp_Object frame, help;
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
2005-07-25 22:17:02 +00:00
|
|
|
|
/* No help echo on owner-draw menu items, or when the keyboard is used
|
|
|
|
|
to navigate the menus, since tooltips are distracting if they pop
|
|
|
|
|
up elsewhere. */
|
|
|
|
|
if (flags & MF_OWNERDRAW || flags & MF_POPUP
|
|
|
|
|
|| !(flags & MF_MOUSESELECT))
|
2001-12-01 11:16:54 +00:00
|
|
|
|
help = Qnil;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
MENUITEMINFO info;
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
2001-12-01 11:16:54 +00:00
|
|
|
|
bzero (&info, sizeof (info));
|
|
|
|
|
info.cbSize = sizeof (info);
|
|
|
|
|
info.fMask = MIIM_DATA;
|
|
|
|
|
get_menu_item_info (menu, item, FALSE, &info);
|
|
|
|
|
|
2004-05-18 19:53:08 +00:00
|
|
|
|
#ifdef USE_LISP_UNION_TYPE
|
|
|
|
|
help = info.dwItemData ? (Lisp_Object) ((EMACS_INT) info.dwItemData)
|
|
|
|
|
: Qnil;
|
|
|
|
|
#else
|
2001-12-09 16:48:03 +00:00
|
|
|
|
help = info.dwItemData ? (Lisp_Object) info.dwItemData : Qnil;
|
2004-05-18 19:53:08 +00:00
|
|
|
|
#endif
|
2001-12-01 11:16:54 +00:00
|
|
|
|
}
|
2000-08-20 21:14:08 +00:00
|
|
|
|
|
2001-11-28 20:31:43 +00:00
|
|
|
|
/* Store the help echo in the keyboard buffer as the X toolkit
|
|
|
|
|
version does, rather than directly showing it. This seems to
|
|
|
|
|
solve the GC problems that were present when we based the
|
|
|
|
|
Windows code on the non-toolkit version. */
|
|
|
|
|
if (f)
|
|
|
|
|
{
|
|
|
|
|
XSETFRAME (frame, f);
|
|
|
|
|
kbd_buffer_store_help_event (frame, help);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
/* X version has a loop through frames here, which doesn't
|
|
|
|
|
appear to do anything, unless it has some side effect. */
|
|
|
|
|
show_help_echo (help, Qnil, Qnil, Qnil, 1);
|
2000-03-22 23:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-09 16:48:03 +00:00
|
|
|
|
/* Free memory used by owner-drawn strings. */
|
2001-12-01 11:16:54 +00:00
|
|
|
|
static void
|
|
|
|
|
w32_free_submenu_strings (menu)
|
|
|
|
|
HMENU menu;
|
|
|
|
|
{
|
|
|
|
|
int i, num = GetMenuItemCount (menu);
|
|
|
|
|
for (i = 0; i < num; i++)
|
|
|
|
|
{
|
|
|
|
|
MENUITEMINFO info;
|
2001-12-10 22:52:51 +00:00
|
|
|
|
bzero (&info, sizeof (info));
|
2001-12-01 11:16:54 +00:00
|
|
|
|
info.cbSize = sizeof (info);
|
2001-12-09 16:48:03 +00:00
|
|
|
|
info.fMask = MIIM_DATA | MIIM_TYPE | MIIM_SUBMENU;
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
2001-12-01 11:16:54 +00:00
|
|
|
|
get_menu_item_info (menu, i, TRUE, &info);
|
|
|
|
|
|
2001-12-09 16:48:03 +00:00
|
|
|
|
/* Owner-drawn names are held in dwItemData. */
|
|
|
|
|
if ((info.fType & MF_OWNERDRAW) && info.dwItemData)
|
|
|
|
|
{
|
|
|
|
|
#ifdef MENU_DEBUG
|
|
|
|
|
DebPrint ("Menu: freeing %ld for owner-draw", info.dwItemData);
|
|
|
|
|
#endif
|
(local_heap, local_alloc, local_free): New macros.
(malloc_widget_value, free_widget_value)
(w32_free_submenu_strings): Use them.
(push_submenu_start, push_submenu_end, push_left_right_boundary)
(push_menu_pane, push_menu_item, single_keymap_panes)
(single_menu_item, Fx_popup_menu, menubar_selection_callback)
(single_submenu, set_frame_menubar)
(w32_menu_show, w32_dialog_show): Use AREF, ASET, ASIZE.
(Fx_popup_menu): Don't show pop up menu until preceding one is
actually cleaned up. Moved UNGCPRO outside #ifdef HAVE_MENUS block.
Changes adapted from xmenu.c
(set_frame_menubar): First parse all submenus,
then make widget_value trees from them.
Don't allocate any widget_value objects
until we are done with the parsing.
(parse_single_submenu): New function.
(digest_single_submenu): New function.
(single_submenu): Function deleted, replaced by those two.
2002-08-05 16:33:44 +00:00
|
|
|
|
local_free (info.dwItemData);
|
2001-12-09 16:48:03 +00:00
|
|
|
|
}
|
2001-12-01 11:16:54 +00:00
|
|
|
|
|
|
|
|
|
/* Recurse down submenus. */
|
|
|
|
|
if (info.hSubMenu)
|
|
|
|
|
w32_free_submenu_strings (info.hSubMenu);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
w32_free_menu_strings (hwnd)
|
|
|
|
|
HWND hwnd;
|
|
|
|
|
{
|
|
|
|
|
HMENU menu = current_popup_menu;
|
|
|
|
|
|
|
|
|
|
if (get_menu_item_info)
|
|
|
|
|
{
|
|
|
|
|
/* If there is no popup menu active, free the strings from the frame's
|
|
|
|
|
menubar. */
|
|
|
|
|
if (!menu)
|
|
|
|
|
menu = GetMenu (hwnd);
|
|
|
|
|
|
|
|
|
|
if (menu)
|
|
|
|
|
w32_free_submenu_strings (menu);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
current_popup_menu = NULL;
|
|
|
|
|
}
|
2000-03-22 23:15:30 +00:00
|
|
|
|
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#endif /* HAVE_MENUS */
|
2001-12-01 11:16:54 +00:00
|
|
|
|
|
2006-11-06 16:47:33 +00:00
|
|
|
|
/* The following is used by delayed window autoselection. */
|
|
|
|
|
|
|
|
|
|
DEFUN ("menu-or-popup-active-p", Fmenu_or_popup_active_p, Smenu_or_popup_active_p, 0, 0, 0,
|
|
|
|
|
doc: /* Return t if a menu or popup dialog is active on selected frame. */)
|
|
|
|
|
()
|
|
|
|
|
{
|
|
|
|
|
#ifdef HAVE_MENUS
|
|
|
|
|
FRAME_PTR f;
|
|
|
|
|
f = SELECTED_FRAME ();
|
|
|
|
|
return (f->output_data.w32->menubar_active > 0) ? Qt : Qnil;
|
|
|
|
|
#else
|
|
|
|
|
return Qnil;
|
|
|
|
|
#endif /* HAVE_MENUS */
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-17 22:35:26 +00:00
|
|
|
|
void syms_of_w32menu ()
|
1995-11-07 07:52:28 +00:00
|
|
|
|
{
|
2007-06-11 22:07:18 +00:00
|
|
|
|
globals_of_w32menu ();
|
1998-04-23 23:43:02 +00:00
|
|
|
|
staticpro (&menu_items);
|
|
|
|
|
menu_items = Qnil;
|
|
|
|
|
|
2001-12-01 11:16:54 +00:00
|
|
|
|
current_popup_menu = NULL;
|
|
|
|
|
|
2007-06-11 22:07:18 +00:00
|
|
|
|
DEFSYM (Qdebug_on_next_call, "debug-on-next-call");
|
1996-05-21 18:09:43 +00:00
|
|
|
|
|
1995-11-07 07:52:28 +00:00
|
|
|
|
defsubr (&Sx_popup_menu);
|
2006-11-05 12:17:26 +00:00
|
|
|
|
defsubr (&Smenu_or_popup_active_p);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#ifdef HAVE_MENUS
|
1995-11-07 07:52:28 +00:00
|
|
|
|
defsubr (&Sx_popup_dialog);
|
1998-04-23 23:43:02 +00:00
|
|
|
|
#endif
|
1995-11-07 07:52:28 +00:00
|
|
|
|
}
|
2002-12-18 06:16:28 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
globals_of_w32menu is used to initialize those global variables that
|
|
|
|
|
must always be initialized on startup even when the global variable
|
|
|
|
|
initialized is non zero (see the function main in emacs.c).
|
|
|
|
|
globals_of_w32menu is called from syms_of_w32menu when the global
|
|
|
|
|
variable initialized is 0 and directly from main when initialized
|
|
|
|
|
is non zero.
|
|
|
|
|
*/
|
|
|
|
|
void globals_of_w32menu ()
|
|
|
|
|
{
|
|
|
|
|
/* See if Get/SetMenuItemInfo functions are available. */
|
|
|
|
|
HMODULE user32 = GetModuleHandle ("user32.dll");
|
|
|
|
|
get_menu_item_info = (GetMenuItemInfoA_Proc) GetProcAddress (user32, "GetMenuItemInfoA");
|
|
|
|
|
set_menu_item_info = (SetMenuItemInfoA_Proc) GetProcAddress (user32, "SetMenuItemInfoA");
|
2004-09-02 23:26:16 +00:00
|
|
|
|
unicode_append_menu = (AppendMenuW_Proc) GetProcAddress (user32, "AppendMenuW");
|
2002-12-18 06:16:28 +00:00
|
|
|
|
}
|
2003-09-01 15:45:59 +00:00
|
|
|
|
|
|
|
|
|
/* arch-tag: 0eaed431-bb4e-4aac-a527-95a1b4f1fed0
|
|
|
|
|
(do not change this comment) */
|