/* Pulldown menu code. Copyright (C) 1994 Miguel de Icaza. This program 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 2 of the License, or (at your option) any later version. This program 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, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include "tty.h" #include #include #include #include #include "global.h" #include "menu.h" #include "dialog.h" #include "color.h" #include "main.h" #include "mouse.h" #include "win.h" #include "key.h" /* For mi_getch() */ /* "$Id$" */ extern int is_right; int menubar_visible = 1; /* This is the new default */ static void menu_scan_hotkey(Menu menu) { char* cp = strchr (menu->name, '&'); if (cp != NULL && cp[1] != '\0'){ strcpy (cp, cp+1); menu->hotkey = tolower(*cp); } else menu->hotkey = 0; } Menu create_menu (char *name, menu_entry *entries, int count) { Menu menu; menu = (Menu) g_malloc (sizeof (*menu)); menu->count = count; menu->max_entry_len = 0; menu->entries = entries; #ifdef ENABLE_NLS if (entries != (menu_entry*) NULL) { register menu_entry* mp; for (mp = entries; count--; mp++) { if (mp->text[0] == '\0') continue; mp->text = _(mp->text); } } #endif /* ENABLE_NLS */ menu->name = g_strdup ( _(name) ); menu_scan_hotkey(menu); menu->start_x = 0; return menu; } static void menubar_drop_compute (WMenu *menubar) { const Menu menu = menubar->menu [menubar->selected]; int max_entry_len = 0; int i; for (i = 0; i < menu->count; i++) max_entry_len = max (max_entry_len, strlen (menu->entries [i].text)); menubar->max_entry_len = max_entry_len = max (max_entry_len, 20); } static void menubar_paint_idx (WMenu *menubar, int idx, int color) { const Menu menu = menubar->menu [menubar->selected]; const int y = 2 + idx; int x = menubar-> menu[menubar->selected]->start_x; if (x + menubar->max_entry_len + 3 > menubar->widget.cols) x = menubar->widget.cols - menubar->max_entry_len - 3; widget_move (&menubar->widget, y, x); attrset (color); hline (' ', menubar->max_entry_len+2); if (!*menu->entries [idx].text){ attrset (SELECTED_COLOR); widget_move (&menubar->widget, y, x + 1); hline (slow_terminal ? ' ' : ACS_HLINE, menubar->max_entry_len); } else { unsigned char *text = menu->entries [idx].text; addch((unsigned char)menu->entries [idx].first_letter); for (text = menu->entries [idx].text; *text; text++) { if (*text == '&') { ++text; menu->entries [idx].hot_key = tolower(*text); attrset (color == MENU_SELECTED_COLOR ? MENU_HOTSEL_COLOR : MENU_HOT_COLOR); addch(*text); attrset(color); continue; } addch(*text); } } widget_move (&menubar->widget, y, x + 1); } static INLINE void menubar_draw_drop (WMenu *menubar) { const int count = (menubar->menu [menubar->selected])->count; int i; int sel = menubar->subsel; int column = menubar-> menu[menubar->selected]->start_x - 1; if (column + menubar->max_entry_len + 4 > menubar->widget.cols) column = menubar->widget.cols - menubar->max_entry_len - 4; attrset (SELECTED_COLOR); draw_box (menubar->widget.parent, menubar->widget.y+1, menubar->widget.x + column, count+2, menubar->max_entry_len + 4); column++; for (i = 0; i < count; i++){ if (i == sel) continue; menubar_paint_idx (menubar, i, MENU_ENTRY_COLOR); } menubar_paint_idx (menubar, sel, MENU_SELECTED_COLOR); } static void menubar_draw (WMenu *menubar) { const int items = menubar->items; int i; /* First draw the complete menubar */ attrset (SELECTED_COLOR); widget_move (&menubar->widget, 0, 0); /* ncurses bug: it should work with hline but it does not */ for (i = menubar->widget.cols; i; i--) addch (' '); attrset (SELECTED_COLOR); /* Now each one of the entries */ for (i = 0; i < items; i++){ if (menubar->active) attrset(i == menubar->selected?MENU_SELECTED_COLOR:SELECTED_COLOR); widget_move (&menubar->widget, 0, menubar->menu [i]->start_x); printw ("%s", menubar->menu [i]->name); } if (menubar->dropped) menubar_draw_drop (menubar); else widget_move (&menubar->widget, 0, menubar-> menu[menubar->selected]->start_x); } static INLINE void menubar_remove (WMenu *menubar) { menubar->subsel = 0; if (menubar->dropped){ menubar->dropped = 0; do_refresh (); menubar->dropped = 1; } } static void menubar_left (WMenu *menu) { menubar_remove (menu); menu->selected = (menu->selected - 1) % menu->items; if (menu->selected < 0) menu->selected = menu->items -1; menubar_drop_compute (menu); menubar_draw (menu); } static void menubar_right (WMenu *menu) { menubar_remove (menu); menu->selected = (menu->selected + 1) % menu->items; menubar_drop_compute (menu); menubar_draw (menu); } static void menubar_finish (WMenu *menubar) { menubar->dropped = 0; menubar->active = 0; menubar->widget.lines = 1; widget_want_hotkey (menubar->widget, 0); dlg_select_nth_widget (menubar->widget.parent, menubar->previous_selection); do_refresh (); } static void menubar_drop (WMenu *menubar, int selected) { menubar->dropped = 1; menubar->selected = selected; menubar->subsel = 0; menubar_drop_compute (menubar); menubar_draw (menubar); } static void menubar_execute (WMenu *menubar, int entry) { const Menu menu = menubar->menu [menubar->selected]; const callfn call_back = menu->entries [entry].call_back; is_right = menubar->selected != 0; /* This used to be the other way round, i.e. first callback and then menubar_finish. The new order (hack?) is needed to make change_panel () work which is used in quick_view_cmd () -- Norbert */ menubar_finish (menubar); (*call_back)(0); do_refresh (); } static void menubar_move (WMenu *menubar, int step) { const Menu menu = menubar->menu [menubar->selected]; menubar_paint_idx (menubar, menubar->subsel, MENU_ENTRY_COLOR); do { menubar->subsel += step; if (menubar->subsel < 0) menubar->subsel = menu->count - 1; menubar->subsel %= menu->count; } while (!menu->entries [menubar->subsel].call_back); menubar_paint_idx (menubar, menubar->subsel, MENU_SELECTED_COLOR); } static int menubar_handle_key (WMenu *menubar, int key) { int i; /* Lowercase */ if (key < 256 && isalpha (key)) /* Linux libc.so.5.x.x bug fix */ key = tolower (key); if (is_abort_char (key)){ menubar_finish (menubar); return 1; } if (key == KEY_LEFT || key == XCTRL('b')){ menubar_left (menubar); return 1; } else if (key == KEY_RIGHT || key == XCTRL ('f')){ menubar_right (menubar); return 1; } /* .ado: NT Alpha can not allow CTRL in Menubar */ #if defined(_OS_NT) if (!key) return 0; #endif if (!menubar->dropped){ const int items = menubar->items; for (i = 0; i < items; i++){ const Menu menu = menubar->menu [i]; if (menu->hotkey == key){ menubar_drop (menubar, i); return 1; } } if (key == KEY_ENTER || key == XCTRL ('n') || key == KEY_DOWN || key == '\n'){ menubar_drop (menubar, menubar->selected); return 1; } return 1; } else { const int selected = menubar->selected; const Menu menu = menubar->menu [selected]; const int items = menu->count; for (i = 0; i < items; i++){ if (!menu->entries [i].call_back) continue; if (key != menu->entries [i].hot_key) continue; menubar_execute (menubar, i); return 1; } if (key == KEY_ENTER || key == '\n'){ menubar_execute (menubar, menubar->subsel); return 1; } if (key == KEY_DOWN || key == XCTRL ('n')) menubar_move (menubar, 1); if (key == KEY_UP || key == XCTRL ('p')) menubar_move (menubar, -1); } return 0; } static int menubar_callback (Dlg_head *h, WMenu *menubar, int msg, int par) { switch (msg){ /* We do not want the focus unless we have been activated */ case WIDGET_FOCUS: if (menubar->active){ widget_want_cursor (menubar->widget, 1); /* Trick to get all the mouse events */ menubar->widget.lines = LINES; /* Trick to get all of the hotkeys */ widget_want_hotkey (menubar->widget, 1); menubar->subsel = 0; menubar_drop_compute (menubar); menubar_draw (menubar); return 1; } else return 0; /* We don't want the buttonbar to activate while using the menubar */ case WIDGET_HOTKEY: case WIDGET_KEY: if (menubar->active){ menubar_handle_key (menubar, par); return 1; } else return 0; case WIDGET_CURSOR: /* Put the cursor in a suitable place */ return 0; case WIDGET_UNFOCUS: if (menubar->active) return 0; else { widget_want_cursor (menubar->widget, 0); return 1; } case WIDGET_DRAW: if (menubar_visible) menubar_draw (menubar); } return default_proc (h, msg, par); } int menubar_event (Gpm_Event *event, WMenu *menubar) { int was_active; int new_selection; int left_x, right_x, bottom_y; if (!(event->type & (GPM_UP|GPM_DOWN|GPM_DRAG))) return MOU_NORMAL; if (!menubar->dropped){ menubar->previous_selection = dlg_item_number(menubar->widget.parent); menubar->active = 1; menubar->dropped = 1; was_active = 0; } else was_active = 1; /* Mouse operations on the menubar */ if (event->y == 1 || !was_active){ if (event->type & GPM_UP) return MOU_NORMAL; new_selection = 0; while (new_selection < menubar->items && event->x > menubar->menu[new_selection]->start_x ) new_selection++; if (new_selection) /* Don't set the invalid value -1 */ --new_selection; if (!was_active){ menubar->selected = new_selection; dlg_select_widget (menubar->widget.parent, menubar); menubar_drop_compute (menubar); menubar_draw (menubar); return MOU_NORMAL; } menubar_remove (menubar); menubar->selected = new_selection; menubar_drop_compute (menubar); menubar_draw (menubar); return MOU_NORMAL; } if (!menubar->dropped) return MOU_NORMAL; /* Ignore the events on anything below the third line */ if (event->y <= 2) return MOU_NORMAL; /* Else, the mouse operation is on the menus or it is not */ left_x = menubar->menu[menubar->selected]->start_x; right_x = left_x + menubar->max_entry_len + 4; if (right_x > menubar->widget.cols) { left_x = menubar->widget.cols - menubar->max_entry_len - 3; right_x = menubar->widget.cols - 1; } bottom_y = (menubar->menu [menubar->selected])->count + 3; if ((event->x > left_x) && (event->x < right_x) && (event->y < bottom_y)){ int pos = event->y - 3; if (!menubar->menu [menubar->selected]->entries [pos].call_back) return MOU_NORMAL; menubar_paint_idx (menubar, menubar->subsel, MENU_ENTRY_COLOR); menubar->subsel = pos; menubar_paint_idx (menubar, menubar->subsel, MENU_SELECTED_COLOR); if (event->type & GPM_UP) menubar_execute (menubar, pos); } else if (event->type & GPM_DOWN) menubar_finish (menubar); return MOU_NORMAL; } static void menubar_destroy (WMenu *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(WMenu* menubar) { register int i, start_x = 1; int items = menubar->items; #ifndef RESIZABLE_MENUBAR int gap = 3; for (i = 0; i < items; i++) { int len = strlen(menubar->menu[i]->name); menubar->menu[i]->start_x = start_x; start_x += len + gap; } #else /* RESIZABLE_MENUBAR */ int gap = menubar->widget.cols - 2; /* First, calculate gap between items... */ for (i = 0; i < items; i++) { /* preserve length here, to be used below */ gap -= (menubar->menu[i]->start_x = strlen(menubar->menu[i]->name)); } gap /= (items - 1); if (gap <= 0) { /* We are out of luck - window is too narrow... */ gap = 1; } /* ...and now fix start positions of menubar items */ for (i = 0; i < items; i++) { int len = menubar->menu[i]->start_x; menubar->menu[i]->start_x = start_x; start_x += len + gap; } #endif /* RESIZABLE_MENUBAR */ } void destroy_menu (Menu menu) { g_free (menu->name); g_free (menu); } WMenu *menubar_new (int y, int x, int cols, Menu menu [], int items) { WMenu *menubar = g_new0 (WMenu, 1); /* FIXME: subsel used w/o being set */ init_widget (&menubar->widget, y, x, 1, cols, (callback_fn) menubar_callback, (destroy_fn) menubar_destroy, (mouse_h) menubar_event, NULL); menubar->menu = menu; menubar->active = 0; menubar->dropped = 0; menubar->items = items; menubar->selected = 0; widget_want_cursor (menubar->widget, 0); menubar_arrange(menubar); return menubar; }