123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946 |
- /*
- Pulldown menu code
- Copyright (C) 1994-2015
- Free Software Foundation, Inc.
- Written by:
- Andrew Borodin <aborodin@vmail.ru>, 2012, 2013
- This file is part of the Midnight Commander.
- The Midnight Commander is free software: you can redistribute it
- and/or modify it under the terms of the GNU General Public License as
- published by the Free Software Foundation, either version 3 of the License,
- or (at your option) any later version.
- The Midnight Commander 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
- along with this program. If not, see <http://www.gnu.org/licenses/>.
- */
- /** \file menu.c
- * \brief Source: pulldown menu code
- */
- #include <config.h>
- #include <ctype.h>
- #include <stdarg.h>
- #include <string.h>
- #include <sys/types.h>
- #include "lib/global.h"
- #include "lib/tty/tty.h"
- #include "lib/skin.h"
- #include "lib/tty/mouse.h"
- #include "lib/tty/key.h" /* key macros */
- #include "lib/strutil.h"
- #include "lib/widget.h"
- #include "lib/event.h" /* mc_event_dispatch() */
- /*** global variables ****************************************************************************/
- /*** file scope macro definitions ****************************************************************/
- #define MENUENTRY(x) ((menu_entry_t *)(x))
- #define MENU(x) ((menu_t *)(x))
- /*** file scope type declarations ****************************************************************/
- struct menu_entry_t
- {
- unsigned char first_letter;
- hotkey_t text;
- unsigned long command;
- char *shortcut;
- };
- struct menu_t
- {
- int start_x; /* position relative to menubar start */
- hotkey_t text;
- GList *entries;
- size_t max_entry_len; /* cached max length of entry texts (text + shortcut) */
- size_t max_hotkey_len; /* cached max length of shortcuts */
- unsigned int selected; /* pointer to current menu entry */
- char *help_node;
- };
- /*** file scope variables ************************************************************************/
- /*** file scope functions ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- static void
- menu_arrange (menu_t * menu, dlg_shortcut_str get_shortcut)
- {
- if (menu != NULL)
- {
- GList *i;
- size_t max_shortcut_len = 0;
- menu->max_entry_len = 1;
- menu->max_hotkey_len = 1;
- for (i = menu->entries; i != NULL; i = g_list_next (i))
- {
- menu_entry_t *entry = MENUENTRY (i->data);
- if (entry != NULL)
- {
- size_t len;
- len = (size_t) hotkey_width (entry->text);
- menu->max_hotkey_len = max (menu->max_hotkey_len, len);
- if (get_shortcut != NULL)
- entry->shortcut = get_shortcut (entry->command);
- if (entry->shortcut != NULL)
- {
- len = (size_t) str_term_width1 (entry->shortcut);
- max_shortcut_len = max (max_shortcut_len, len);
- }
- }
- }
- menu->max_entry_len = menu->max_hotkey_len + max_shortcut_len;
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_paint_idx (WMenuBar * menubar, unsigned int idx, int color)
- {
- Widget *w = WIDGET (menubar);
- const menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- const menu_entry_t *entry = MENUENTRY (g_list_nth_data (menu->entries, idx));
- const int y = 2 + idx;
- int x = menu->start_x;
- if (x + menu->max_entry_len + 4 > (gsize) w->cols)
- x = w->cols - menu->max_entry_len - 4;
- if (entry == NULL)
- {
- /* menu separator */
- tty_setcolor (MENU_ENTRY_COLOR);
- widget_move (w, y, x - 1);
- tty_print_alt_char (ACS_LTEE, FALSE);
- tty_draw_hline (w->y + y, w->x + x, ACS_HLINE, menu->max_entry_len + 3);
- widget_move (w, y, x + menu->max_entry_len + 3);
- tty_print_alt_char (ACS_RTEE, FALSE);
- }
- else
- {
- int yt, xt;
- /* menu text */
- tty_setcolor (color);
- widget_move (w, y, x);
- tty_print_char ((unsigned char) entry->first_letter);
- tty_getyx (&yt, &xt);
- tty_draw_hline (yt, xt, ' ', menu->max_entry_len + 2); /* clear line */
- tty_print_string (entry->text.start);
- if (entry->text.hotkey != NULL)
- {
- tty_setcolor (color == MENU_SELECTED_COLOR ? MENU_HOTSEL_COLOR : MENU_HOT_COLOR);
- tty_print_string (entry->text.hotkey);
- tty_setcolor (color);
- }
- if (entry->text.end != NULL)
- tty_print_string (entry->text.end);
- if (entry->shortcut != NULL)
- {
- widget_move (w, y, x + menu->max_hotkey_len + 3);
- tty_print_string (entry->shortcut);
- }
- /* move cursor to the start of entry text */
- widget_move (w, y, x + 1);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_draw_drop (WMenuBar * menubar)
- {
- Widget *w = WIDGET (menubar);
- const menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- const unsigned int count = g_list_length (menu->entries);
- int column = menu->start_x - 1;
- unsigned int i;
- if (column + menu->max_entry_len + 5 > (gsize) w->cols)
- column = w->cols - menu->max_entry_len - 5;
- tty_setcolor (MENU_ENTRY_COLOR);
- tty_draw_box (w->y + 1, w->x + column, count + 2, menu->max_entry_len + 5, FALSE);
- for (i = 0; i < count; i++)
- menubar_paint_idx (menubar, i,
- i == menu->selected ? MENU_SELECTED_COLOR : MENU_ENTRY_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_set_color (WMenuBar * menubar, gboolean current, gboolean hotkey)
- {
- if (!menubar->is_active)
- tty_setcolor (MENU_INACTIVE_COLOR);
- else if (current)
- tty_setcolor (hotkey ? MENU_HOTSEL_COLOR : MENU_SELECTED_COLOR);
- else
- tty_setcolor (hotkey ? MENU_HOT_COLOR : MENU_ENTRY_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_draw (WMenuBar * menubar)
- {
- Widget *w = WIDGET (menubar);
- GList *i;
- /* First draw the complete menubar */
- tty_setcolor (menubar->is_active ? MENU_ENTRY_COLOR : MENU_INACTIVE_COLOR);
- tty_draw_hline (w->y, w->x, ' ', w->cols);
- /* Now each one of the entries */
- for (i = menubar->menu; i != NULL; i = g_list_next (i))
- {
- menu_t *menu = MENU (i->data);
- gboolean is_selected = (menubar->selected == (gsize) g_list_position (menubar->menu, i));
- menubar_set_color (menubar, is_selected, FALSE);
- widget_move (w, 0, menu->start_x);
- tty_print_char (' ');
- tty_print_string (menu->text.start);
- if (menu->text.hotkey != NULL)
- {
- menubar_set_color (menubar, is_selected, TRUE);
- tty_print_string (menu->text.hotkey);
- menubar_set_color (menubar, is_selected, FALSE);
- }
- if (menu->text.end != NULL)
- tty_print_string (menu->text.end);
- tty_print_char (' ');
- }
- if (menubar->is_dropped)
- menubar_draw_drop (menubar);
- else
- widget_move (w, 0, MENU (g_list_nth_data (menubar->menu, menubar->selected))->start_x);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_remove (WMenuBar * menubar)
- {
- WDialog *h;
- if (!menubar->is_dropped)
- return;
- /* HACK: before refresh the dialog, change the current widget to keep the order
- of overlapped widgets. This is useful in multi-window editor.
- In general, menubar should be a special object, not an ordinary widget
- in the current dialog. */
- h = WIDGET (menubar)->owner;
- h->current = g_list_find (h->widgets, dlg_find_by_id (h, menubar->previous_widget));
- menubar->is_dropped = FALSE;
- do_refresh ();
- menubar->is_dropped = TRUE;
- /* restore current widget */
- h->current = g_list_find (h->widgets, menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_left (WMenuBar * menubar)
- {
- menubar_remove (menubar);
- if (menubar->selected == 0)
- menubar->selected = g_list_length (menubar->menu) - 1;
- else
- menubar->selected--;
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_right (WMenuBar * menubar)
- {
- menubar_remove (menubar);
- menubar->selected = (menubar->selected + 1) % g_list_length (menubar->menu);
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_finish (WMenuBar * menubar)
- {
- Widget *w = WIDGET (menubar);
- menubar->is_dropped = FALSE;
- menubar->is_active = FALSE;
- w->lines = 1;
- widget_want_hotkey (w, 0);
- dlg_select_by_id (w->owner, menubar->previous_widget);
- do_refresh ();
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_drop (WMenuBar * menubar, unsigned int selected)
- {
- menubar->is_dropped = TRUE;
- menubar->selected = selected;
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_execute (WMenuBar * menubar)
- {
- const menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- const menu_entry_t *entry = MENUENTRY (g_list_nth_data (menu->entries, menu->selected));
- if ((entry != NULL) && (entry->command != CK_IgnoreKey))
- {
- Widget *w = WIDGET (menubar);
- mc_global.widget.is_right = (menubar->selected != 0);
- menubar_finish (menubar);
- send_message (w->owner, w, MSG_ACTION, entry->command, NULL);
- do_refresh ();
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_down (WMenuBar * menubar)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- const unsigned int len = g_list_length (menu->entries);
- menu_entry_t *entry;
- menubar_paint_idx (menubar, menu->selected, MENU_ENTRY_COLOR);
- do
- {
- menu->selected = (menu->selected + 1) % len;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->selected));
- }
- while ((entry == NULL) || (entry->command == CK_IgnoreKey));
- menubar_paint_idx (menubar, menu->selected, MENU_SELECTED_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_up (WMenuBar * menubar)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- const unsigned int len = g_list_length (menu->entries);
- menu_entry_t *entry;
- menubar_paint_idx (menubar, menu->selected, MENU_ENTRY_COLOR);
- do
- {
- if (menu->selected == 0)
- menu->selected = len - 1;
- else
- menu->selected--;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->selected));
- }
- while ((entry == NULL) || (entry->command == CK_IgnoreKey));
- menubar_paint_idx (menubar, menu->selected, MENU_SELECTED_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_first (WMenuBar * menubar)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- if (menu->selected == 0)
- return;
- menubar_paint_idx (menubar, menu->selected, MENU_ENTRY_COLOR);
- menu->selected = 0;
- while (TRUE)
- {
- menu_entry_t *entry;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->selected));
- if ((entry == NULL) || (entry->command == CK_IgnoreKey))
- menu->selected++;
- else
- break;
- }
- menubar_paint_idx (menubar, menu->selected, MENU_SELECTED_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_last (WMenuBar * menubar)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- const unsigned int len = g_list_length (menu->entries);
- menu_entry_t *entry;
- if (menu->selected == len - 1)
- return;
- menubar_paint_idx (menubar, menu->selected, MENU_ENTRY_COLOR);
- menu->selected = len;
- do
- {
- menu->selected--;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->selected));
- }
- while ((entry == NULL) || (entry->command == CK_IgnoreKey));
- menubar_paint_idx (menubar, menu->selected, MENU_SELECTED_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static int
- menubar_handle_key (WMenuBar * menubar, int key)
- {
- /* Lowercase */
- if (isascii (key))
- key = g_ascii_tolower (key);
- if (is_abort_char (key))
- {
- menubar_finish (menubar);
- return 1;
- }
- /* menubar help or menubar navigation */
- switch (key)
- {
- case KEY_F (1):
- {
- ev_help_t event_data = { NULL, NULL };
- if (menubar->is_dropped)
- event_data.node =
- MENU (g_list_nth_data (menubar->menu, menubar->selected))->help_node;
- else
- event_data.node = "[Menu Bar]";
- mc_event_dispatch (MCEVENT_GROUP_CORE, "help", &event_data, NULL, NULL);
- menubar_draw (menubar);
- return 1;
- }
- case KEY_LEFT:
- case XCTRL ('b'):
- menubar_left (menubar);
- return 1;
- case KEY_RIGHT:
- case XCTRL ('f'):
- menubar_right (menubar);
- return 1;
- }
- if (!menubar->is_dropped)
- {
- GList *i;
- /* drop menu by hotkey */
- for (i = menubar->menu; i != NULL; i = g_list_next (i))
- {
- menu_t *menu = MENU (i->data);
- if ((menu->text.hotkey != NULL) && (key == g_ascii_tolower (menu->text.hotkey[0])))
- {
- menubar_drop (menubar, g_list_position (menubar->menu, i));
- return 1;
- }
- }
- /* drop menu by Enter or Dowwn key */
- if (key == KEY_ENTER || key == XCTRL ('n') || key == KEY_DOWN || key == '\n')
- menubar_drop (menubar, menubar->selected);
- return 1;
- }
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- GList *i;
- /* execute menu command by hotkey */
- for (i = menu->entries; i != NULL; i = g_list_next (i))
- {
- const menu_entry_t *entry = MENUENTRY (i->data);
- if ((entry != NULL) && (entry->command != CK_IgnoreKey)
- && (entry->text.hotkey != NULL) && (key == g_ascii_tolower (entry->text.hotkey[0])))
- {
- menu->selected = g_list_position (menu->entries, i);
- menubar_execute (menubar);
- return 1;
- }
- }
- /* menu execute by Enter or menu navigation */
- switch (key)
- {
- case KEY_ENTER:
- case '\n':
- menubar_execute (menubar);
- return 1;
- case KEY_HOME:
- case ALT ('<'):
- menubar_first (menubar);
- break;
- case KEY_END:
- case ALT ('>'):
- menubar_last (menubar);
- break;
- case KEY_DOWN:
- case XCTRL ('n'):
- menubar_down (menubar);
- break;
- case KEY_UP:
- case XCTRL ('p'):
- menubar_up (menubar);
- break;
- }
- }
- return 0;
- }
- /* --------------------------------------------------------------------------------------------- */
- static cb_ret_t
- menubar_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
- {
- WMenuBar *menubar = MENUBAR (w);
- switch (msg)
- {
- /* We do not want the focus unless we have been activated */
- case MSG_FOCUS:
- if (!menubar->is_active)
- return MSG_NOT_HANDLED;
- /* Trick to get all the mouse events */
- w->lines = LINES;
- /* Trick to get all of the hotkeys */
- widget_want_hotkey (w, 1);
- menubar_draw (menubar);
- return MSG_HANDLED;
- /* We don't want the buttonbar to activate while using the menubar */
- case MSG_HOTKEY:
- case MSG_KEY:
- if (menubar->is_active)
- {
- menubar_handle_key (menubar, parm);
- return MSG_HANDLED;
- }
- return MSG_NOT_HANDLED;
- case MSG_CURSOR:
- /* Put the cursor in a suitable place */
- return MSG_NOT_HANDLED;
- case MSG_UNFOCUS:
- return menubar->is_active ? MSG_NOT_HANDLED : MSG_HANDLED;
- case MSG_DRAW:
- if (menubar->is_visible)
- {
- menubar_draw (menubar);
- return MSG_HANDLED;
- }
- /* fall through */
- case MSG_RESIZE:
- /* try show menu after screen resize */
- send_message (w, sender, MSG_FOCUS, 0, data);
- return MSG_HANDLED;
- case MSG_DESTROY:
- menubar_set_menu (menubar, NULL);
- return MSG_HANDLED;
- default:
- return widget_default_callback (w, sender, msg, parm, data);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static int
- menubar_event (Gpm_Event * event, void *data)
- {
- WMenuBar *menubar = MENUBAR (data);
- Widget *w = WIDGET (data);
- gboolean was_active = TRUE;
- int left_x, right_x, bottom_y;
- menu_t *menu;
- Gpm_Event local;
- if (!mouse_global_in_widget (event, w))
- return MOU_UNHANDLED;
- /* ignore unsupported events */
- if ((event->type & (GPM_UP | GPM_DOWN | GPM_DRAG)) == 0)
- return MOU_NORMAL;
- /* ignore wheel events if menu is inactive */
- if (!menubar->is_active && ((event->buttons & (GPM_B_MIDDLE | GPM_B_UP | GPM_B_DOWN)) != 0))
- return MOU_NORMAL;
- local = mouse_get_local (event, w);
- if (local.y == 1 && (local.type & GPM_UP) != 0)
- return MOU_NORMAL;
- if (!menubar->is_dropped)
- {
- if (local.y > 1)
- {
- /* mouse click below menubar -- close menu and send focus to widget under mouse */
- menubar_finish (menubar);
- return MOU_UNHANDLED;
- }
- menubar->previous_widget = dlg_get_current_widget_id (w->owner);
- menubar->is_active = TRUE;
- menubar->is_dropped = TRUE;
- was_active = FALSE;
- }
- /* Mouse operations on the menubar */
- if (local.y == 1 || !was_active)
- {
- /* wheel events on menubar */
- if ((local.buttons & GPM_B_UP) != 0)
- menubar_left (menubar);
- else if ((local.buttons & GPM_B_DOWN) != 0)
- menubar_right (menubar);
- else
- {
- const unsigned int len = g_list_length (menubar->menu);
- unsigned int new_selection = 0;
- while ((new_selection < len)
- && (local.x > MENU (g_list_nth_data (menubar->menu, new_selection))->start_x))
- new_selection++;
- if (new_selection != 0) /* Don't set the invalid value -1 */
- new_selection--;
- if (!was_active)
- {
- menubar->selected = new_selection;
- dlg_select_widget (menubar);
- }
- else
- {
- menubar_remove (menubar);
- menubar->selected = new_selection;
- }
- menubar_draw (menubar);
- }
- return MOU_NORMAL;
- }
- if (!menubar->is_dropped || (local.y < 2))
- return MOU_NORMAL;
- /* middle click -- everywhere */
- if (((local.buttons & GPM_B_MIDDLE) != 0) && ((local.type & GPM_DOWN) != 0))
- {
- menubar_execute (menubar);
- return MOU_NORMAL;
- }
- /* the mouse operation is on the menus or it is not */
- menu = MENU (g_list_nth_data (menubar->menu, menubar->selected));
- left_x = menu->start_x;
- right_x = left_x + menu->max_entry_len + 3;
- if (right_x > w->cols)
- {
- left_x = w->cols - menu->max_entry_len - 3;
- right_x = w->cols;
- }
- bottom_y = g_list_length (menu->entries) + 3;
- if ((local.x >= left_x) && (local.x <= right_x) && (local.y <= bottom_y))
- {
- int pos = local.y - 3;
- const menu_entry_t *entry = MENUENTRY (g_list_nth_data (menu->entries, pos));
- /* mouse wheel */
- if ((local.buttons & GPM_B_UP) != 0 && (local.type & GPM_DOWN) != 0)
- {
- menubar_up (menubar);
- return MOU_NORMAL;
- }
- if ((local.buttons & GPM_B_DOWN) != 0 && (local.type & GPM_DOWN) != 0)
- {
- menubar_down (menubar);
- return MOU_NORMAL;
- }
- /* ignore events above and below dropped down menu */
- if ((pos < 0) || (pos >= bottom_y - 3))
- return MOU_NORMAL;
- if ((entry != NULL) && (entry->command != CK_IgnoreKey))
- {
- menubar_paint_idx (menubar, menu->selected, MENU_ENTRY_COLOR);
- menu->selected = pos;
- menubar_paint_idx (menubar, menu->selected, MENU_SELECTED_COLOR);
- if ((event->type & GPM_UP) != 0)
- menubar_execute (menubar);
- }
- }
- else if (((local.type & GPM_DOWN) != 0) && ((local.buttons & (GPM_B_UP | GPM_B_DOWN)) == 0))
- {
- /* use click not wheel to close menu */
- menubar_finish (menubar);
- }
- return MOU_NORMAL;
- }
- /* --------------------------------------------------------------------------------------------- */
- /*** public functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- menu_entry_t *
- menu_entry_create (const char *name, unsigned long command)
- {
- menu_entry_t *entry;
- entry = g_new (menu_entry_t, 1);
- entry->first_letter = ' ';
- entry->text = parse_hotkey (name);
- entry->command = command;
- entry->shortcut = NULL;
- return entry;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menu_entry_free (menu_entry_t * entry)
- {
- if (entry != NULL)
- {
- release_hotkey (entry->text);
- g_free (entry->shortcut);
- g_free (entry);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- menu_t *
- create_menu (const char *name, GList * entries, const char *help_node)
- {
- menu_t *menu;
- menu = g_new (menu_t, 1);
- menu->start_x = 0;
- menu->text = parse_hotkey (name);
- menu->entries = entries;
- menu->max_entry_len = 1;
- menu->max_hotkey_len = 0;
- menu->selected = 0;
- menu->help_node = g_strdup (help_node);
- return menu;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menu_set_name (menu_t * menu, const char *name)
- {
- release_hotkey (menu->text);
- menu->text = parse_hotkey (name);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- destroy_menu (menu_t * menu)
- {
- release_hotkey (menu->text);
- g_list_free_full (menu->entries, (GDestroyNotify) menu_entry_free);
- g_free (menu->help_node);
- g_free (menu);
- }
- /* --------------------------------------------------------------------------------------------- */
- WMenuBar *
- menubar_new (int y, int x, int cols, GList * menu, gboolean visible)
- {
- WMenuBar *menubar;
- Widget *w;
- menubar = g_new0 (WMenuBar, 1);
- w = WIDGET (menubar);
- widget_init (w, y, x, 1, cols, menubar_callback, menubar_event);
- menubar->is_visible = visible;
- widget_want_cursor (w, FALSE);
- menubar_set_menu (menubar, menu);
- return menubar;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menubar_set_menu (WMenuBar * menubar, GList * menu)
- {
- /* delete previous menu */
- if (menubar->menu != NULL)
- g_list_free_full (menubar->menu, (GDestroyNotify) destroy_menu);
- /* add new menu */
- menubar->is_active = FALSE;
- menubar->is_dropped = FALSE;
- menubar->menu = menu;
- menubar->selected = 0;
- menubar_arrange (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menubar_add_menu (WMenuBar * menubar, menu_t * menu)
- {
- if (menu != NULL)
- {
- menu_arrange (menu, WIDGET (menubar)->owner->get_shortcut);
- menubar->menu = g_list_append (menubar->menu, menu);
- }
- menubar_arrange (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Properly space menubar items. Should be called when menubar is created
- * and also when widget width is changed (i.e. upon xterm resize).
- */
- void
- menubar_arrange (WMenuBar * menubar)
- {
- int start_x = 1;
- GList *i;
- int gap;
- if (menubar->menu == NULL)
- return;
- gap = WIDGET (menubar)->cols - 2;
- /* First, calculate gap between items... */
- for (i = menubar->menu; i != NULL; i = g_list_next (i))
- {
- menu_t *menu = MENU (i->data);
- /* preserve length here, to be used below */
- menu->start_x = hotkey_width (menu->text) + 2;
- gap -= menu->start_x;
- }
- if (g_list_next (menubar->menu) == NULL)
- gap = 1;
- else
- gap /= (g_list_length (menubar->menu) - 1);
- if (gap <= 0)
- {
- /* We are out of luck - window is too narrow... */
- gap = 1;
- }
- else if (gap >= 3)
- gap = 3;
- /* ...and now fix start positions of menubar items */
- for (i = menubar->menu; i != NULL; i = g_list_next (i))
- {
- menu_t *menu = MENU (i->data);
- int len = menu->start_x;
- menu->start_x = start_x;
- start_x += len + gap;
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- /** Find MenuBar widget in the dialog */
- WMenuBar *
- find_menubar (const WDialog * h)
- {
- return MENUBAR (find_widget_type (h, menubar_callback));
- }
- /* --------------------------------------------------------------------------------------------- */
|