123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561 |
- /*
- Widgets for the Midnight Commander
- Copyright (C) 1994-2017
- Free Software Foundation, Inc.
- Authors:
- Radek Doulik, 1994, 1995
- Miguel de Icaza, 1994, 1995
- Jakub Jelinek, 1995
- Andrej Borsenkow, 1996
- Norbert Warmuth, 1997
- Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2011, 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 widget-common.c
- * \brief Source: shared stuff of widgets
- */
- #include <config.h>
- #include <stdlib.h>
- #include <string.h>
- #include "lib/global.h"
- #include "lib/tty/tty.h"
- #include "lib/tty/color.h"
- #include "lib/skin.h"
- #include "lib/strutil.h"
- #include "lib/widget.h"
- /*** global variables ****************************************************************************/
- /*** file scope macro definitions ****************************************************************/
- /*** file scope type declarations ****************************************************************/
- /*** file scope variables ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- /*** file scope functions ************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- static void
- widget_do_focus (Widget * w, gboolean enable)
- {
- if (w != NULL && widget_get_state (WIDGET (w->owner), WST_FOCUSED))
- widget_set_state (w, WST_FOCUSED, enable);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Focus specified widget in it's owner.
- *
- * @param w widget to be focused.
- */
- static void
- widget_focus (Widget * w)
- {
- WDialog *h = DIALOG (w->owner);
- if (h == NULL)
- return;
- if (WIDGET (h->current->data) != w)
- {
- widget_do_focus (WIDGET (h->current->data), FALSE);
- /* Test if focus lost was allowed and focus has really been loose */
- if (h->current == NULL || !widget_get_state (WIDGET (h->current->data), WST_FOCUSED))
- {
- widget_do_focus (w, TRUE);
- h->current = dlg_find (h, w);
- }
- }
- else if (!widget_get_state (w, WST_FOCUSED))
- widget_do_focus (w, TRUE);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Put widget on top or bottom of Z-order.
- */
- static void
- widget_reorder (GList * l, gboolean set_top)
- {
- WDialog *h = WIDGET (l->data)->owner;
- h->widgets = g_list_remove_link (h->widgets, l);
- if (set_top)
- h->widgets = g_list_concat (h->widgets, l);
- else
- h->widgets = g_list_concat (l, h->widgets);
- }
- /* --------------------------------------------------------------------------------------------- */
- /*** public functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- struct hotkey_t
- parse_hotkey (const char *text)
- {
- hotkey_t result;
- const char *cp, *p;
- if (text == NULL)
- text = "";
- /* search for '&', that is not on the of text */
- cp = strchr (text, '&');
- if (cp != NULL && cp[1] != '\0')
- {
- result.start = g_strndup (text, cp - text);
- /* skip '&' */
- cp++;
- p = str_cget_next_char (cp);
- result.hotkey = g_strndup (cp, p - cp);
- cp = p;
- result.end = g_strdup (cp);
- }
- else
- {
- result.start = g_strdup (text);
- result.hotkey = NULL;
- result.end = NULL;
- }
- return result;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- release_hotkey (const hotkey_t hotkey)
- {
- g_free (hotkey.start);
- g_free (hotkey.hotkey);
- g_free (hotkey.end);
- }
- /* --------------------------------------------------------------------------------------------- */
- int
- hotkey_width (const hotkey_t hotkey)
- {
- int result;
- result = str_term_width1 (hotkey.start);
- result += (hotkey.hotkey != NULL) ? str_term_width1 (hotkey.hotkey) : 0;
- result += (hotkey.end != NULL) ? str_term_width1 (hotkey.end) : 0;
- return result;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- hotkey_draw (Widget * w, const hotkey_t hotkey, gboolean focused)
- {
- widget_selectcolor (w, focused, FALSE);
- tty_print_string (hotkey.start);
- if (hotkey.hotkey != NULL)
- {
- widget_selectcolor (w, focused, TRUE);
- tty_print_string (hotkey.hotkey);
- widget_selectcolor (w, focused, FALSE);
- }
- if (hotkey.end != NULL)
- tty_print_string (hotkey.end);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- widget_init (Widget * w, int y, int x, int lines, int cols,
- widget_cb_fn callback, widget_mouse_cb_fn mouse_callback)
- {
- w->x = x;
- w->y = y;
- w->cols = cols;
- w->lines = lines;
- w->pos_flags = WPOS_KEEP_DEFAULT;
- w->callback = callback;
- w->mouse_callback = mouse_callback;
- w->owner = NULL;
- w->mouse.forced_capture = FALSE;
- w->mouse.capture = FALSE;
- w->mouse.last_msg = MSG_MOUSE_NONE;
- w->mouse.last_buttons_down = 0;
- w->options = WOP_DEFAULT;
- w->state = WST_DEFAULT;
- }
- /* --------------------------------------------------------------------------------------------- */
- /* Default callback for widgets */
- cb_ret_t
- widget_default_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
- {
- (void) w;
- (void) sender;
- (void) parm;
- (void) data;
- switch (msg)
- {
- case MSG_INIT:
- case MSG_FOCUS:
- case MSG_UNFOCUS:
- case MSG_ENABLE:
- case MSG_DISABLE:
- case MSG_DRAW:
- case MSG_DESTROY:
- case MSG_CURSOR:
- case MSG_IDLE:
- return MSG_HANDLED;
- default:
- return MSG_NOT_HANDLED;
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Apply new options to widget.
- *
- * @param w widget
- * @param options widget option flags to modify. Several flags per call can be modified.
- * @param enable TRUE if specified options should be added, FALSE if options should be removed
- */
- void
- widget_set_options (Widget * w, widget_options_t options, gboolean enable)
- {
- if (enable)
- w->options |= options;
- else
- w->options &= ~options;
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Modify state of widget.
- *
- * @param w widget
- * @param state widget state flag to modify
- * @param enable specifies whether to turn the flag on (TRUE) or off (FALSE).
- * Only one flag per call can be modified.
- * @return MSG_HANDLED if set was handled successfully, MSG_NOT_HANDLED otherwise.
- */
- cb_ret_t
- widget_set_state (Widget * w, widget_state_t state, gboolean enable)
- {
- gboolean ret = MSG_HANDLED;
- if (enable)
- w->state |= state;
- else
- w->state &= ~state;
- if (enable)
- {
- /* exclusive bits */
- if ((state & WST_CONSTRUCT) != 0)
- w->state &= ~(WST_ACTIVE | WST_SUSPENDED | WST_CLOSED);
- else if ((state & WST_ACTIVE) != 0)
- w->state &= ~(WST_CONSTRUCT | WST_SUSPENDED | WST_CLOSED);
- else if ((state & WST_SUSPENDED) != 0)
- w->state &= ~(WST_CONSTRUCT | WST_ACTIVE | WST_CLOSED);
- else if ((state & WST_CLOSED) != 0)
- w->state &= ~(WST_CONSTRUCT | WST_ACTIVE | WST_SUSPENDED);
- }
- if (w->owner == NULL)
- return MSG_NOT_HANDLED;
- switch (state)
- {
- case WST_DISABLED:
- ret = send_message (w, NULL, enable ? MSG_DISABLE : MSG_ENABLE, 0, NULL);
- if (ret == MSG_HANDLED && widget_get_state (WIDGET (w->owner), WST_ACTIVE))
- ret = send_message (w, NULL, MSG_DRAW, 0, NULL);
- break;
- case WST_FOCUSED:
- {
- widget_msg_t msg;
- msg = enable ? MSG_FOCUS : MSG_UNFOCUS;
- ret = send_message (w, NULL, msg, 0, NULL);
- if (ret == MSG_HANDLED && widget_get_state (WIDGET (w->owner), WST_ACTIVE))
- {
- send_message (w, NULL, MSG_DRAW, 0, NULL);
- /* Notify owner that focus was moved from one widget to another */
- send_message (w->owner, w, MSG_CHANGED_FOCUS, 0, NULL);
- }
- }
- break;
- default:
- break;
- }
- return ret;
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- widget_set_size (Widget * widget, int y, int x, int lines, int cols)
- {
- widget->x = x;
- widget->y = y;
- widget->cols = cols;
- widget->lines = lines;
- send_message (widget, NULL, MSG_RESIZE, 0, NULL);
- if (widget->owner != NULL && widget_get_state (WIDGET (widget->owner), WST_ACTIVE))
- send_message (widget, NULL, MSG_DRAW, 0, NULL);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- widget_selectcolor (Widget * w, gboolean focused, gboolean hotkey)
- {
- WDialog *h = w->owner;
- int color;
- if (widget_get_state (w, WST_DISABLED))
- color = DISABLED_COLOR;
- else if (hotkey)
- {
- if (focused)
- color = h->color[DLG_COLOR_HOT_FOCUS];
- else
- color = h->color[DLG_COLOR_HOT_NORMAL];
- }
- else
- {
- if (focused)
- color = h->color[DLG_COLOR_FOCUS];
- else
- color = h->color[DLG_COLOR_NORMAL];
- }
- tty_setcolor (color);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- widget_erase (Widget * w)
- {
- if (w != NULL)
- tty_fill_region (w->y, w->x, w->lines, w->cols, ' ');
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Check whether widget is active or not.
- * @param w the widget
- *
- * @return TRUE if the widget is active, FALSE otherwise
- */
- gboolean
- widget_is_active (const void *w)
- {
- return (w == CONST_WIDGET (w)->owner->current->data);
- }
- /* --------------------------------------------------------------------------------------------- */
- void
- widget_redraw (Widget * w)
- {
- if (w != NULL)
- {
- WDialog *h = w->owner;
- if (h != NULL && widget_get_state (WIDGET (h), WST_ACTIVE))
- w->callback (w, NULL, MSG_DRAW, 0, NULL);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Replace widget in the dialog.
- *
- * @param old_w old widget that need to be replaced
- * @param new_w new widget that will replace @old_w
- */
- void
- widget_replace (Widget * old_w, Widget * new_w)
- {
- WDialog *h = old_w->owner;
- gboolean should_focus = FALSE;
- GList *holder;
- if (h->widgets == NULL)
- return;
- if (h->current == NULL)
- h->current = h->widgets;
- /* locate widget position in the list */
- if (old_w == h->current->data)
- holder = h->current;
- else
- holder = g_list_find (h->widgets, old_w);
- /* if old widget is focused, we should focus the new one... */
- if (widget_get_state (old_w, WST_FOCUSED))
- should_focus = TRUE;
- /* ...but if new widget isn't selectable, we cannot focus it */
- if (!widget_get_options (new_w, WOP_SELECTABLE))
- should_focus = FALSE;
- /* if new widget isn't selectable, select other widget before replace */
- if (!should_focus)
- {
- GList *l;
- for (l = dlg_get_widget_next_of (holder);
- !widget_get_options (WIDGET (l->data), WOP_SELECTABLE)
- && !widget_get_state (WIDGET (l->data), WST_DISABLED); l = dlg_get_widget_next_of (l))
- ;
- widget_select (WIDGET (l->data));
- }
- /* replace widget */
- new_w->owner = h;
- new_w->id = old_w->id;
- holder->data = new_w;
- send_message (old_w, NULL, MSG_DESTROY, 0, NULL);
- send_message (new_w, NULL, MSG_INIT, 0, NULL);
- if (should_focus)
- widget_select (new_w);
- else
- widget_redraw (new_w);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Select specified widget in it's owner.
- *
- * Note: this function (and widget_focus(), which it calls) is a no-op
- * if the widget is already selected.
- *
- * @param w widget to be selected
- */
- void
- widget_select (Widget * w)
- {
- WDialog *h;
- if (!widget_get_options (w, WOP_SELECTABLE))
- return;
- h = w->owner;
- if (h != NULL)
- {
- if (widget_get_options (w, WOP_TOP_SELECT))
- {
- GList *l;
- l = dlg_find (h, w);
- widget_reorder (l, TRUE);
- }
- widget_focus (w);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Set widget at bottom of widget list.
- */
- void
- widget_set_bottom (Widget * w)
- {
- widget_reorder (dlg_find (w->owner, w), FALSE);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Check whether two widgets are overlapped or not.
- * @param a 1st widget
- * @param b 2nd widget
- *
- * @return TRUE if widgets are overlapped, FALSE otherwise.
- */
- gboolean
- widget_overlapped (const Widget * a, const Widget * b)
- {
- return !((b->x >= a->x + a->cols)
- || (a->x >= b->x + b->cols) || (b->y >= a->y + a->lines) || (a->y >= b->y + b->lines));
- }
- /* --------------------------------------------------------------------------------------------- */
- /* get mouse pointer location within widget */
- Gpm_Event
- mouse_get_local (const Gpm_Event * global, const Widget * w)
- {
- Gpm_Event local;
- local.buttons = global->buttons;
- #ifdef HAVE_LIBGPM
- local.clicks = 0;
- local.margin = 0;
- local.modifiers = 0;
- local.vc = 0;
- #endif
- local.x = global->x - w->x;
- local.y = global->y - w->y;
- local.type = global->type;
- return local;
- }
- /* --------------------------------------------------------------------------------------------- */
- gboolean
- mouse_global_in_widget (const Gpm_Event * event, const Widget * w)
- {
- return (event->x > w->x) && (event->y > w->y) && (event->x <= w->x + w->cols)
- && (event->y <= w->y + w->lines);
- }
- /* --------------------------------------------------------------------------------------------- */
|