1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099 |
- /*
- Pulldown menu code
- Copyright (C) 1994-2025
- Free Software Foundation, Inc.
- Written by:
- Andrew Borodin <aborodin@vmail.ru>, 2012-2022
- 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/key.h" /* key macros */
- #include "lib/strutil.h"
- #include "lib/widget.h"
- #include "lib/event.h" /* mc_event_raise() */
- /*** global variables ****************************************************************************/
- const global_keymap_t *menu_map = NULL;
- /*** 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;
- 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 current; /* pointer to current menu entry */
- char *help_node;
- };
- /*** forward declarations (file scope functions) *************************************************/
- /*** 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 (const WMenuBar *menubar, unsigned int idx, int color)
- {
- const WRect *w = &CONST_WIDGET (menubar)->rect;
- const menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- 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_gotoyx (menubar, 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_gotoyx (menubar, y, x + menu->max_entry_len + 3);
- tty_print_alt_char (ACS_RTEE, FALSE);
- }
- else
- {
- int yt, xt;
- /* menu text */
- tty_setcolor (color);
- widget_gotoyx (menubar, 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_gotoyx (menubar, y, x + menu->max_hotkey_len + 3);
- tty_print_string (entry->shortcut);
- }
- /* move cursor to the start of entry text */
- widget_gotoyx (menubar, y, x + 1);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_draw_drop (const WMenuBar *menubar)
- {
- const WRect *w = &CONST_WIDGET (menubar)->rect;
- const menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- 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;
- if (mc_global.tty.shadows)
- tty_draw_box_shadow (w->y + 1, w->x + column, count + 2, menu->max_entry_len + 5,
- SHADOW_COLOR);
- 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->current ? MENU_SELECTED_COLOR : MENU_ENTRY_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_set_color (const WMenuBar *menubar, gboolean current, gboolean hotkey)
- {
- if (!widget_get_state (CONST_WIDGET (menubar), WST_FOCUSED))
- 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 (const WMenuBar *menubar)
- {
- const WRect *w = &CONST_WIDGET (menubar)->rect;
- GList *i;
- /* First draw the complete menubar */
- tty_setcolor (widget_get_state (WIDGET (menubar), WST_FOCUSED) ? 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->current == (gsize) g_list_position (menubar->menu, i));
- menubar_set_color (menubar, is_selected, FALSE);
- widget_gotoyx (menubar, 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_gotoyx (menubar, 0,
- MENU (g_list_nth_data (menubar->menu, menubar->current))->start_x);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_remove (WMenuBar *menubar)
- {
- Widget *g;
- 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. */
- g = WIDGET (WIDGET (menubar)->owner);
- GROUP (g)->current = widget_find (g, widget_find_by_id (g, menubar->previous_widget));
- menubar->is_dropped = FALSE;
- do_refresh ();
- menubar->is_dropped = TRUE;
- /* restore current widget */
- GROUP (g)->current = widget_find (g, WIDGET (menubar));
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_left (WMenuBar *menubar)
- {
- menubar_remove (menubar);
- if (menubar->current == 0)
- menubar->current = g_list_length (menubar->menu) - 1;
- else
- menubar->current--;
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_right (WMenuBar *menubar)
- {
- menubar_remove (menubar);
- menubar->current = (menubar->current + 1) % g_list_length (menubar->menu);
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_finish (WMenuBar *menubar)
- {
- Widget *w = WIDGET (menubar);
- menubar->is_dropped = FALSE;
- w->rect.lines = 1;
- widget_want_hotkey (w, FALSE);
- widget_set_options (w, WOP_SELECTABLE, FALSE);
- if (!mc_global.keybar_visible)
- widget_hide (w);
- else
- {
- /* Move the menubar to the bottom so that widgets displayed on top of
- * an "invisible" menubar get the first chance to respond to mouse events. */
- widget_set_bottom (w);
- }
- /* background must be bottom */
- if (DIALOG (w->owner)->bg != NULL)
- widget_set_bottom (WIDGET (DIALOG (w->owner)->bg));
- group_select_widget_by_id (w->owner, menubar->previous_widget);
- do_refresh ();
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_drop (WMenuBar *menubar, unsigned int selected)
- {
- menubar->is_dropped = TRUE;
- menubar->current = selected;
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_execute (WMenuBar *menubar)
- {
- const menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- const menu_entry_t *entry = MENUENTRY (g_list_nth_data (menu->entries, menu->current));
- if ((entry != NULL) && (entry->command != CK_IgnoreKey))
- {
- Widget *w = WIDGET (menubar);
- mc_global.widget.is_right = (menubar->current != 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->current));
- const unsigned int len = g_list_length (menu->entries);
- menu_entry_t *entry;
- menubar_paint_idx (menubar, menu->current, MENU_ENTRY_COLOR);
- do
- {
- menu->current = (menu->current + 1) % len;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->current));
- }
- while ((entry == NULL) || (entry->command == CK_IgnoreKey));
- menubar_paint_idx (menubar, menu->current, MENU_SELECTED_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_up (WMenuBar *menubar)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- const unsigned int len = g_list_length (menu->entries);
- menu_entry_t *entry;
- menubar_paint_idx (menubar, menu->current, MENU_ENTRY_COLOR);
- do
- {
- if (menu->current == 0)
- menu->current = len - 1;
- else
- menu->current--;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->current));
- }
- while ((entry == NULL) || (entry->command == CK_IgnoreKey));
- menubar_paint_idx (menubar, menu->current, MENU_SELECTED_COLOR);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_first (WMenuBar *menubar)
- {
- if (menubar->is_dropped)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- if (menu->current == 0)
- return;
- menubar_paint_idx (menubar, menu->current, MENU_ENTRY_COLOR);
- menu->current = 0;
- while (TRUE)
- {
- menu_entry_t *entry;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->current));
- if ((entry == NULL) || (entry->command == CK_IgnoreKey))
- menu->current++;
- else
- break;
- }
- menubar_paint_idx (menubar, menu->current, MENU_SELECTED_COLOR);
- }
- else
- {
- menubar->current = 0;
- menubar_draw (menubar);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_last (WMenuBar *menubar)
- {
- if (menubar->is_dropped)
- {
- menu_t *menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- const unsigned int len = g_list_length (menu->entries);
- menu_entry_t *entry;
- if (menu->current == len - 1)
- return;
- menubar_paint_idx (menubar, menu->current, MENU_ENTRY_COLOR);
- menu->current = len;
- do
- {
- menu->current--;
- entry = MENUENTRY (g_list_nth_data (menu->entries, menu->current));
- }
- while ((entry == NULL) || (entry->command == CK_IgnoreKey));
- menubar_paint_idx (menubar, menu->current, MENU_SELECTED_COLOR);
- }
- else
- {
- menubar->current = g_list_length (menubar->menu) - 1;
- menubar_draw (menubar);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static cb_ret_t
- menubar_try_drop_menu (WMenuBar *menubar, int hotkey)
- {
- GList *i;
- for (i = menubar->menu; i != NULL; i = g_list_next (i))
- {
- menu_t *menu = MENU (i->data);
- if (menu->text.hotkey != NULL && hotkey == g_ascii_tolower (menu->text.hotkey[0]))
- {
- menubar_drop (menubar, g_list_position (menubar->menu, i));
- return MSG_HANDLED;
- }
- }
- return MSG_NOT_HANDLED;
- }
- /* --------------------------------------------------------------------------------------------- */
- static cb_ret_t
- menubar_try_exec_menu (WMenuBar *menubar, int hotkey)
- {
- menu_t *menu;
- GList *i;
- menu = g_list_nth_data (menubar->menu, menubar->current);
- for (i = menu->entries; i != NULL; i = g_list_next (i))
- {
- const menu_entry_t *entry = MENUENTRY (i->data);
- if (entry != NULL && entry->text.hotkey != NULL
- && hotkey == g_ascii_tolower (entry->text.hotkey[0]))
- {
- menu->current = g_list_position (menu->entries, i);
- menubar_execute (menubar);
- return MSG_HANDLED;
- }
- }
- return MSG_NOT_HANDLED;
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_help (const WMenuBar *menubar)
- {
- ev_help_t event_data;
- event_data.filename = NULL;
- if (menubar->is_dropped)
- event_data.node = MENU (g_list_nth_data (menubar->menu, menubar->current))->help_node;
- else
- event_data.node = "[Menu Bar]";
- mc_event_raise (MCEVENT_GROUP_CORE, "help", &event_data);
- menubar_draw (menubar);
- }
- /* --------------------------------------------------------------------------------------------- */
- static cb_ret_t
- menubar_execute_cmd (WMenuBar *menubar, long command)
- {
- cb_ret_t ret = MSG_HANDLED;
- switch (command)
- {
- case CK_Help:
- menubar_help (menubar);
- break;
- case CK_Left:
- menubar_left (menubar);
- break;
- case CK_Right:
- menubar_right (menubar);
- break;
- case CK_Up:
- if (menubar->is_dropped)
- menubar_up (menubar);
- break;
- case CK_Down:
- if (menubar->is_dropped)
- menubar_down (menubar);
- else
- menubar_drop (menubar, menubar->current);
- break;
- case CK_Home:
- menubar_first (menubar);
- break;
- case CK_End:
- menubar_last (menubar);
- break;
- case CK_Enter:
- if (menubar->is_dropped)
- menubar_execute (menubar);
- else
- menubar_drop (menubar, menubar->current);
- break;
- case CK_Quit:
- menubar_finish (menubar);
- break;
- default:
- ret = MSG_NOT_HANDLED;
- break;
- }
- return ret;
- }
- /* --------------------------------------------------------------------------------------------- */
- static int
- menubar_handle_key (WMenuBar *menubar, int key)
- {
- long cmd;
- cb_ret_t ret = MSG_NOT_HANDLED;
- cmd = widget_lookup_key (WIDGET (menubar), key);
- if (cmd != CK_IgnoreKey)
- ret = menubar_execute_cmd (menubar, cmd);
- if (ret != MSG_HANDLED)
- {
- if (menubar->is_dropped)
- ret = menubar_try_exec_menu (menubar, key);
- else
- ret = menubar_try_drop_menu (menubar, key);
- }
- return ret;
- }
- /* --------------------------------------------------------------------------------------------- */
- static gboolean
- menubar_refresh (WMenuBar *menubar)
- {
- Widget *w = WIDGET (menubar);
- if (!widget_get_state (w, WST_FOCUSED))
- return FALSE;
- /* Trick to get all the mouse events */
- w->rect.lines = LINES;
- /* Trick to get all of the hotkeys */
- widget_want_hotkey (w, TRUE);
- return TRUE;
- }
- /* --------------------------------------------------------------------------------------------- */
- static inline void
- menubar_free_menu (WMenuBar *menubar)
- {
- g_clear_list (&menubar->menu, (GDestroyNotify) menu_free);
- }
- /* --------------------------------------------------------------------------------------------- */
- 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_refresh (menubar))
- {
- menubar_draw (menubar);
- return MSG_HANDLED;
- }
- return MSG_NOT_HANDLED;
- case MSG_UNFOCUS:
- return widget_get_state (w, WST_FOCUSED) ? MSG_NOT_HANDLED : MSG_HANDLED;
- /* We don't want the buttonbar to activate while using the menubar */
- case MSG_HOTKEY:
- case MSG_KEY:
- if (widget_get_state (w, WST_FOCUSED))
- {
- 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_DRAW:
- if (widget_get_state (w, WST_VISIBLE) || menubar_refresh (menubar))
- menubar_draw (menubar);
- return MSG_HANDLED;
- case MSG_RESIZE:
- /* try show menu after screen resize */
- widget_default_callback (w, NULL, MSG_RESIZE, 0, data);
- menubar_refresh (menubar);
- return MSG_HANDLED;
- case MSG_DESTROY:
- menubar_free_menu (menubar);
- return MSG_HANDLED;
- default:
- return widget_default_callback (w, sender, msg, parm, data);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static unsigned int
- menubar_get_menu_by_x_coord (const WMenuBar *menubar, int x)
- {
- unsigned int i;
- GList *menu;
- for (i = 0, menu = menubar->menu;
- menu != NULL && x >= MENU (menu->data)->start_x; i++, menu = g_list_next (menu))
- ;
- /* Don't set the invalid value -1 */
- if (i != 0)
- i--;
- return i;
- }
- /* --------------------------------------------------------------------------------------------- */
- static gboolean
- menubar_mouse_on_menu (const WMenuBar *menubar, int y, int x)
- {
- const WRect *w = &CONST_WIDGET (menubar)->rect;
- menu_t *menu;
- int left_x, right_x, bottom_y;
- if (!menubar->is_dropped)
- return FALSE;
- menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- 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) + 2; /* skip bar and top frame */
- return (x >= left_x && x < right_x && y > 1 && y < bottom_y);
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_change_selected_item (WMenuBar *menubar, int y)
- {
- menu_t *menu;
- menu_entry_t *entry;
- y -= 2; /* skip bar and top frame */
- menu = MENU (g_list_nth_data (menubar->menu, menubar->current));
- entry = MENUENTRY (g_list_nth_data (menu->entries, y));
- if (entry != NULL && entry->command != CK_IgnoreKey)
- {
- menubar_paint_idx (menubar, menu->current, MENU_ENTRY_COLOR);
- menu->current = y;
- menubar_paint_idx (menubar, menu->current, MENU_SELECTED_COLOR);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- static void
- menubar_mouse_callback (Widget *w, mouse_msg_t msg, mouse_event_t *event)
- {
- static gboolean was_drag = FALSE;
- WMenuBar *menubar = MENUBAR (w);
- gboolean mouse_on_drop;
- mouse_on_drop = menubar_mouse_on_menu (menubar, event->y, event->x);
- switch (msg)
- {
- case MSG_MOUSE_DOWN:
- was_drag = FALSE;
- if (event->y == 0)
- {
- /* events on menubar */
- unsigned int selected;
- selected = menubar_get_menu_by_x_coord (menubar, event->x);
- menubar_activate (menubar, TRUE, selected);
- menubar_remove (menubar); /* if already shown */
- menubar_drop (menubar, selected);
- }
- else if (mouse_on_drop)
- menubar_change_selected_item (menubar, event->y);
- else
- {
- /* mouse click outside menubar or dropdown -- close menu */
- menubar_finish (menubar);
- /*
- * @FIXME.
- *
- * Unless we clear the 'capture' flag, we'll receive MSG_MOUSE_DRAG
- * events belonging to this click (in case the user drags the mouse,
- * of course).
- *
- * For the time being, we mark this with FIXME as this flag should
- * preferably be regarded as "implementation detail" and not be
- * touched by us. We should think of some other way of communicating
- * this to the system.
- */
- w->mouse.capture = FALSE;
- }
- break;
- case MSG_MOUSE_UP:
- if (was_drag && mouse_on_drop)
- menubar_execute (menubar);
- was_drag = FALSE;
- break;
- case MSG_MOUSE_CLICK:
- was_drag = FALSE;
- if ((event->buttons & GPM_B_MIDDLE) != 0 && event->y > 0 && menubar->is_dropped)
- {
- /* middle click -- everywhere */
- menubar_execute (menubar);
- }
- else if (mouse_on_drop)
- menubar_execute (menubar);
- else if (event->y > 0)
- /* releasing the mouse button outside the menu -- close menu */
- menubar_finish (menubar);
- break;
- case MSG_MOUSE_DRAG:
- if (event->y == 0)
- {
- menubar_remove (menubar);
- menubar_drop (menubar, menubar_get_menu_by_x_coord (menubar, event->x));
- }
- else if (mouse_on_drop)
- menubar_change_selected_item (menubar, event->y);
- was_drag = TRUE;
- break;
- case MSG_MOUSE_SCROLL_UP:
- case MSG_MOUSE_SCROLL_DOWN:
- was_drag = FALSE;
- if (widget_get_state (w, WST_FOCUSED))
- {
- if (event->y == 0)
- {
- /* menubar: left/right */
- if (msg == MSG_MOUSE_SCROLL_UP)
- menubar_left (menubar);
- else
- menubar_right (menubar);
- }
- else if (mouse_on_drop)
- {
- /* drop-down menu: up/down */
- if (msg == MSG_MOUSE_SCROLL_UP)
- menubar_up (menubar);
- else
- menubar_down (menubar);
- }
- }
- break;
- default:
- was_drag = FALSE;
- break;
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- /*** public functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- menu_entry_t *
- menu_entry_new (const char *name, long command)
- {
- menu_entry_t *entry;
- entry = g_new (menu_entry_t, 1);
- entry->first_letter = ' ';
- entry->text = hotkey_new (name);
- entry->command = command;
- entry->shortcut = NULL;
- return entry;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menu_entry_free (menu_entry_t *entry)
- {
- if (entry != NULL)
- {
- hotkey_free (entry->text);
- g_free (entry->shortcut);
- g_free (entry);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- menu_t *
- menu_new (const char *name, GList *entries, const char *help_node)
- {
- menu_t *menu;
- menu = g_new (menu_t, 1);
- menu->start_x = 0;
- menu->text = hotkey_new (name);
- menu->entries = entries;
- menu->max_entry_len = 1;
- menu->max_hotkey_len = 0;
- menu->current = 0;
- menu->help_node = g_strdup (help_node);
- return menu;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menu_set_name (menu_t *menu, const char *name)
- {
- hotkey_free (menu->text);
- menu->text = hotkey_new (name);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menu_free (menu_t *menu)
- {
- hotkey_free (menu->text);
- g_list_free_full (menu->entries, (GDestroyNotify) menu_entry_free);
- g_free (menu->help_node);
- g_free (menu);
- }
- /* --------------------------------------------------------------------------------------------- */
- WMenuBar *
- menubar_new (GList *menu)
- {
- WRect r = { 0, 0, 1, COLS };
- WMenuBar *menubar;
- Widget *w;
- menubar = g_new0 (WMenuBar, 1);
- w = WIDGET (menubar);
- widget_init (w, &r, menubar_callback, menubar_mouse_callback);
- w->pos_flags = WPOS_KEEP_HORZ | WPOS_KEEP_TOP;
- w->options |= WOP_TOP_SELECT;
- w->keymap = menu_map;
- menubar_set_menu (menubar, menu);
- return menubar;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menubar_set_menu (WMenuBar *menubar, GList *menu)
- {
- /* delete previous menu */
- menubar_free_menu (menubar);
- /* add new menu */
- menubar->is_dropped = FALSE;
- menubar->menu = menu;
- menubar->current = 0;
- menubar_arrange (menubar);
- widget_set_state (WIDGET (menubar), WST_FOCUSED, FALSE);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- menubar_add_menu (WMenuBar *menubar, menu_t *menu)
- {
- if (menu != NULL)
- {
- menu_arrange (menu, DIALOG (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)->rect.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 *
- menubar_find (const WDialog *h)
- {
- return MENUBAR (widget_find_by_type (CONST_WIDGET (h), menubar_callback));
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Activate menu bar.
- *
- * @param menubar menu bar object
- * @param dropped whether dropdown menus should be drooped or not
- * @which number of active dropdown menu
- */
- void
- menubar_activate (WMenuBar *menubar, gboolean dropped, int which)
- {
- Widget *w = WIDGET (menubar);
- widget_show (w);
- if (!widget_get_state (w, WST_FOCUSED))
- {
- widget_set_options (w, WOP_SELECTABLE, TRUE);
- menubar->is_dropped = dropped;
- if (which >= 0)
- menubar->current = (guint) which;
- menubar->previous_widget = group_get_current_widget_id (w->owner);
- /* Bring it to the top so it receives all mouse events before any other widget.
- * See also comment in menubar_finish(). */
- widget_select (w);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
|