2011-10-15 14:56:47 +04:00
|
|
|
/*
|
|
|
|
Widgets for the Midnight Commander
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2020-01-18 19:54:37 +03:00
|
|
|
Copyright (C) 1994-2020
|
2014-02-12 10:33:10 +04:00
|
|
|
Free Software Foundation, Inc.
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
Authors:
|
|
|
|
Radek Doulik, 1994, 1995
|
|
|
|
Miguel de Icaza, 1994, 1995
|
|
|
|
Jakub Jelinek, 1995
|
|
|
|
Andrej Borsenkow, 1996
|
|
|
|
Norbert Warmuth, 1997
|
2013-06-24 10:47:03 +04:00
|
|
|
Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2011, 2012, 2013
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
This file is part of the Midnight Commander.
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
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,
|
2010-11-12 11:03:57 +03:00
|
|
|
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
|
2011-10-15 14:56:47 +04:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2010-11-12 11:03:57 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
/** \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 ************************************************************************/
|
|
|
|
|
2016-09-28 13:37:31 +03:00
|
|
|
/* maximum value of used widget ID */
|
|
|
|
static unsigned long widget_id = 0;
|
|
|
|
|
2016-07-16 21:00:28 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2010-11-12 11:03:57 +03:00
|
|
|
/*** file scope functions ************************************************************************/
|
2016-07-16 21:00:28 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2016-09-28 13:37:31 +03:00
|
|
|
/**
|
|
|
|
* Calc widget ID,
|
|
|
|
* Widget ID is uniq for each widget created during MC session (like PID in OS).
|
|
|
|
*
|
|
|
|
* @return widget ID.
|
|
|
|
*/
|
|
|
|
static unsigned long
|
|
|
|
widget_set_id (void)
|
|
|
|
{
|
|
|
|
unsigned long id;
|
|
|
|
|
|
|
|
id = widget_id++;
|
|
|
|
/* TODO IF NEEDED: if id is already used, find next free id. */
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2016-10-13 10:15:03 +03:00
|
|
|
static cb_ret_t
|
|
|
|
widget_default_resize (Widget * w, const WRect * r)
|
|
|
|
{
|
|
|
|
if (r == NULL)
|
|
|
|
return MSG_NOT_HANDLED;
|
|
|
|
|
|
|
|
w->y = r->y;
|
|
|
|
w->x = r->x;
|
|
|
|
w->lines = r->lines;
|
|
|
|
w->cols = r->cols;
|
|
|
|
|
|
|
|
return MSG_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2016-07-16 21:00:28 +03:00
|
|
|
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)
|
|
|
|
{
|
2016-09-27 13:53:17 +03:00
|
|
|
WGroup *g = w->owner;
|
2016-07-16 21:00:28 +03:00
|
|
|
|
2016-09-27 13:53:17 +03:00
|
|
|
if (g == NULL)
|
2016-07-16 21:00:28 +03:00
|
|
|
return;
|
|
|
|
|
2016-09-27 13:53:17 +03:00
|
|
|
if (WIDGET (g->current->data) != w)
|
2016-07-16 21:00:28 +03:00
|
|
|
{
|
2016-09-27 13:53:17 +03:00
|
|
|
widget_do_focus (WIDGET (g->current->data), FALSE);
|
2016-07-16 21:00:28 +03:00
|
|
|
/* Test if focus lost was allowed and focus has really been loose */
|
2016-09-27 13:53:17 +03:00
|
|
|
if (g->current == NULL || !widget_get_state (WIDGET (g->current->data), WST_FOCUSED))
|
2016-07-16 21:00:28 +03:00
|
|
|
{
|
|
|
|
widget_do_focus (w, TRUE);
|
2016-09-29 11:16:17 +03:00
|
|
|
g->current = widget_find (WIDGET (g), w);
|
2016-07-16 21:00:28 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
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)
|
|
|
|
{
|
2016-09-27 13:53:17 +03:00
|
|
|
WGroup *g = WIDGET (l->data)->owner;
|
2016-07-16 21:00:28 +03:00
|
|
|
|
2016-09-27 13:53:17 +03:00
|
|
|
g->widgets = g_list_remove_link (g->widgets, l);
|
2016-07-16 21:00:28 +03:00
|
|
|
if (set_top)
|
2016-09-27 13:53:17 +03:00
|
|
|
g->widgets = g_list_concat (g->widgets, l);
|
2016-07-16 21:00:28 +03:00
|
|
|
else
|
2016-09-27 13:53:17 +03:00
|
|
|
g->widgets = g_list_concat (l, g->widgets);
|
2016-07-16 21:00:28 +03:00
|
|
|
}
|
|
|
|
|
2019-12-14 12:53:44 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
hotkey_cmp (const char *s1, const char *s2)
|
|
|
|
{
|
|
|
|
gboolean n1, n2;
|
|
|
|
|
|
|
|
n1 = s1 != NULL;
|
|
|
|
n2 = s2 != NULL;
|
|
|
|
|
|
|
|
if (n1 != n2)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (n1 && n2 && strcmp (s1, s2) != 0)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2019-11-17 17:58:52 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2020-03-30 09:43:09 +03:00
|
|
|
static void
|
|
|
|
widget_default_mouse_callback (Widget * w, mouse_msg_t msg, mouse_event_t * event)
|
|
|
|
{
|
|
|
|
/* do nothing */
|
|
|
|
(void) w;
|
|
|
|
(void) msg;
|
|
|
|
(void) event;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2019-11-17 17:58:52 +03:00
|
|
|
static const int *
|
|
|
|
widget_default_get_colors (const Widget * w)
|
|
|
|
{
|
|
|
|
const Widget *owner = CONST_WIDGET (w->owner);
|
|
|
|
|
|
|
|
return (owner == NULL ? NULL : widget_get_colors (owner));
|
|
|
|
}
|
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/*** public functions ****************************************************************************/
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
struct hotkey_t
|
2019-12-14 11:50:04 +03:00
|
|
|
hotkey_new (const char *text)
|
2010-11-12 11:03:57 +03:00
|
|
|
{
|
|
|
|
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
|
2019-12-14 11:50:04 +03:00
|
|
|
hotkey_free (const hotkey_t hotkey)
|
2010-11-12 11:03:57 +03:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2019-12-14 12:53:44 +03:00
|
|
|
gboolean
|
|
|
|
hotkey_equal (const hotkey_t hotkey1, const hotkey_t hotkey2)
|
|
|
|
{
|
|
|
|
/* *INDENT-OFF* */
|
|
|
|
return (strcmp (hotkey1.start, hotkey2.start) == 0) &&
|
|
|
|
hotkey_cmp (hotkey1.hotkey, hotkey2.hotkey) &&
|
|
|
|
hotkey_cmp (hotkey1.end, hotkey2.end);
|
|
|
|
/* *INDENT-ON* */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
void
|
|
|
|
hotkey_draw (Widget * w, const hotkey_t hotkey, gboolean focused)
|
|
|
|
{
|
2019-12-14 16:56:18 +03:00
|
|
|
if (hotkey.start[0] != '\0')
|
|
|
|
{
|
|
|
|
widget_selectcolor (w, focused, FALSE);
|
|
|
|
tty_print_string (hotkey.start);
|
|
|
|
}
|
2010-11-12 11:03:57 +03:00
|
|
|
|
|
|
|
if (hotkey.hotkey != NULL)
|
|
|
|
{
|
|
|
|
widget_selectcolor (w, focused, TRUE);
|
|
|
|
tty_print_string (hotkey.hotkey);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hotkey.end != NULL)
|
2019-12-14 16:56:18 +03:00
|
|
|
{
|
|
|
|
widget_selectcolor (w, focused, FALSE);
|
2010-11-12 11:03:57 +03:00
|
|
|
tty_print_string (hotkey.end);
|
2019-12-14 16:56:18 +03:00
|
|
|
}
|
2010-11-12 11:03:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2019-12-14 16:32:26 +03:00
|
|
|
char *
|
|
|
|
hotkey_get_text (const hotkey_t hotkey)
|
|
|
|
{
|
|
|
|
GString *text;
|
|
|
|
|
|
|
|
text = g_string_new (hotkey.start);
|
|
|
|
|
|
|
|
if (hotkey.hotkey != NULL)
|
|
|
|
{
|
|
|
|
g_string_append_c (text, '&');
|
|
|
|
g_string_append (text, hotkey.hotkey);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hotkey.end != NULL)
|
|
|
|
g_string_append (text, hotkey.end);
|
|
|
|
|
|
|
|
return g_string_free (text, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
void
|
2013-06-24 11:40:53 +04:00
|
|
|
widget_init (Widget * w, int y, int x, int lines, int cols,
|
2016-03-06 09:44:16 +03:00
|
|
|
widget_cb_fn callback, widget_mouse_cb_fn mouse_callback)
|
2010-11-12 11:03:57 +03:00
|
|
|
{
|
2016-09-28 13:37:31 +03:00
|
|
|
w->id = widget_set_id ();
|
2010-11-12 11:03:57 +03:00
|
|
|
w->x = x;
|
|
|
|
w->y = y;
|
|
|
|
w->cols = cols;
|
|
|
|
w->lines = lines;
|
2013-08-12 13:51:01 +04:00
|
|
|
w->pos_flags = WPOS_KEEP_DEFAULT;
|
2010-11-12 11:03:57 +03:00
|
|
|
w->callback = callback;
|
2017-03-08 14:41:35 +03:00
|
|
|
|
|
|
|
w->keymap = NULL;
|
|
|
|
w->ext_keymap = NULL;
|
|
|
|
w->ext_mode = FALSE;
|
|
|
|
|
2020-03-30 09:43:09 +03:00
|
|
|
w->mouse_callback = mouse_callback != NULL ? mouse_callback : widget_default_mouse_callback;
|
2010-11-12 11:03:57 +03:00
|
|
|
w->owner = NULL;
|
2016-11-15 09:33:34 +03:00
|
|
|
w->mouse_handler = mouse_handle_event;
|
2016-03-06 09:44:16 +03:00
|
|
|
w->mouse.forced_capture = FALSE;
|
2016-03-23 19:17:05 +03:00
|
|
|
w->mouse.capture = FALSE;
|
|
|
|
w->mouse.last_msg = MSG_MOUSE_NONE;
|
2016-03-06 09:44:16 +03:00
|
|
|
w->mouse.last_buttons_down = 0;
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2016-05-08 19:51:45 +03:00
|
|
|
w->options = WOP_DEFAULT;
|
2019-12-19 16:47:05 +03:00
|
|
|
w->state = WST_CONSTRUCT;
|
2016-09-29 11:16:17 +03:00
|
|
|
|
|
|
|
w->find = widget_default_find;
|
|
|
|
w->find_by_type = widget_default_find_by_type;
|
|
|
|
w->find_by_id = widget_default_find_by_id;
|
2019-11-17 17:02:52 +03:00
|
|
|
|
|
|
|
w->set_state = widget_default_set_state;
|
2019-11-17 17:58:52 +03:00
|
|
|
w->get_colors = widget_default_get_colors;
|
2010-11-12 11:03:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2019-10-16 10:33:00 +03:00
|
|
|
void
|
|
|
|
widget_destroy (Widget * w)
|
|
|
|
{
|
|
|
|
send_message (w, NULL, MSG_DESTROY, 0, NULL);
|
|
|
|
g_free (w);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
/* Default callback for widgets */
|
|
|
|
cb_ret_t
|
2012-09-28 15:05:43 +04:00
|
|
|
widget_default_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
|
2010-11-12 11:03:57 +03:00
|
|
|
{
|
2012-06-26 11:52:21 +04:00
|
|
|
(void) sender;
|
2010-11-12 11:03:57 +03:00
|
|
|
(void) parm;
|
|
|
|
|
|
|
|
switch (msg)
|
|
|
|
{
|
2012-09-28 15:05:43 +04:00
|
|
|
case MSG_INIT:
|
|
|
|
case MSG_FOCUS:
|
|
|
|
case MSG_UNFOCUS:
|
2016-04-10 12:26:48 +03:00
|
|
|
case MSG_ENABLE:
|
|
|
|
case MSG_DISABLE:
|
2012-09-28 15:05:43 +04:00
|
|
|
case MSG_DRAW:
|
|
|
|
case MSG_DESTROY:
|
|
|
|
case MSG_CURSOR:
|
|
|
|
case MSG_IDLE:
|
2010-11-12 11:03:57 +03:00
|
|
|
return MSG_HANDLED;
|
|
|
|
|
2016-10-13 10:15:03 +03:00
|
|
|
case MSG_RESIZE:
|
|
|
|
return widget_default_resize (w, CONST_RECT (data));
|
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
default:
|
|
|
|
return MSG_NOT_HANDLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2012-09-09 16:36:50 +04:00
|
|
|
/**
|
2016-04-10 12:26:48 +03:00
|
|
|
* Apply new options to widget.
|
2012-09-09 16:36:50 +04:00
|
|
|
*
|
|
|
|
* @param w widget
|
2016-04-10 12:26:48 +03:00
|
|
|
* @param options widget option flags to modify. Several flags per call can be modified.
|
2012-09-09 16:36:50 +04:00
|
|
|
* @param enable TRUE if specified options should be added, FALSE if options should be removed
|
|
|
|
*/
|
|
|
|
void
|
2016-04-10 12:26:48 +03:00
|
|
|
widget_set_options (Widget * w, widget_options_t options, gboolean enable)
|
2012-09-09 16:36:50 +04:00
|
|
|
{
|
|
|
|
if (enable)
|
|
|
|
w->options |= options;
|
|
|
|
else
|
|
|
|
w->options &= ~options;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
2016-10-10 10:20:27 +03:00
|
|
|
void
|
|
|
|
widget_adjust_position (widget_pos_flags_t pos_flags, int *y, int *x, int *lines, int *cols)
|
|
|
|
{
|
|
|
|
if ((pos_flags & WPOS_FULLSCREEN) != 0)
|
|
|
|
{
|
|
|
|
*y = 0;
|
|
|
|
*x = 0;
|
|
|
|
*lines = LINES;
|
|
|
|
*cols = COLS;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ((pos_flags & WPOS_CENTER_HORZ) != 0)
|
|
|
|
*x = (COLS - *cols) / 2;
|
|
|
|
|
|
|
|
if ((pos_flags & WPOS_CENTER_VERT) != 0)
|
|
|
|
*y = (LINES - *lines) / 2;
|
|
|
|
|
|
|
|
if ((pos_flags & WPOS_TRYUP) != 0)
|
|
|
|
{
|
|
|
|
if (*y > 3)
|
|
|
|
*y -= 2;
|
|
|
|
else if (*y == 3)
|
|
|
|
*y = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2016-10-13 10:15:03 +03:00
|
|
|
/**
|
|
|
|
* Change widget position and size.
|
|
|
|
*
|
|
|
|
* @param w widget
|
|
|
|
* @param y y coordinate of top-left corner
|
|
|
|
* @param x x coordinate of top-left corner
|
|
|
|
* @param lines width
|
|
|
|
* @param cols height
|
|
|
|
*/
|
2016-10-10 10:20:27 +03:00
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
void
|
2016-10-13 10:15:03 +03:00
|
|
|
widget_set_size (Widget * w, int y, int x, int lines, int cols)
|
2010-11-12 11:03:57 +03:00
|
|
|
{
|
2016-10-13 10:15:03 +03:00
|
|
|
WRect r = { y, x, lines, cols };
|
|
|
|
|
|
|
|
send_message (w, NULL, MSG_RESIZE, 0, &r);
|
|
|
|
|
|
|
|
if (w->owner != NULL && widget_get_state (WIDGET (w->owner), WST_ACTIVE))
|
2016-10-30 12:40:20 +03:00
|
|
|
widget_draw (w);
|
2010-11-12 11:03:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
void
|
|
|
|
widget_selectcolor (Widget * w, gboolean focused, gboolean hotkey)
|
|
|
|
{
|
|
|
|
int color;
|
2019-11-17 17:58:52 +03:00
|
|
|
const int *colors;
|
|
|
|
|
|
|
|
colors = widget_get_colors (w);
|
2010-11-12 11:03:57 +03:00
|
|
|
|
2016-04-10 12:26:48 +03:00
|
|
|
if (widget_get_state (w, WST_DISABLED))
|
2010-11-12 11:03:57 +03:00
|
|
|
color = DISABLED_COLOR;
|
|
|
|
else if (hotkey)
|
2019-11-17 17:58:52 +03:00
|
|
|
color = colors[focused ? DLG_COLOR_HOT_FOCUS : DLG_COLOR_HOT_NORMAL];
|
2010-11-12 11:03:57 +03:00
|
|
|
else
|
2019-11-17 17:58:52 +03:00
|
|
|
color = colors[focused ? DLG_COLOR_FOCUS : DLG_COLOR_NORMAL];
|
2010-11-12 11:03:57 +03:00
|
|
|
|
|
|
|
tty_setcolor (color);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
void
|
|
|
|
widget_erase (Widget * w)
|
|
|
|
{
|
|
|
|
if (w != NULL)
|
|
|
|
tty_fill_region (w->y, w->x, w->lines, w->cols, ' ');
|
|
|
|
}
|
|
|
|
|
2013-06-24 10:47:03 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Check whether widget is active or not.
|
2016-09-30 10:50:19 +03:00
|
|
|
* Widget is active if it's current in the its owner and each owner in the chain is current too.
|
|
|
|
*
|
2013-06-24 10:47:03 +04:00
|
|
|
* @param w the widget
|
|
|
|
*
|
|
|
|
* @return TRUE if the widget is active, FALSE otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
widget_is_active (const void *w)
|
|
|
|
{
|
2016-09-30 10:50:19 +03:00
|
|
|
const WGroup *owner;
|
|
|
|
|
|
|
|
/* Is group top? */
|
|
|
|
if (w == top_dlg->data)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
owner = CONST_WIDGET (w)->owner;
|
|
|
|
|
|
|
|
/* Is widget in any group? */
|
|
|
|
if (owner == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (w != owner->current->data)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return widget_is_active (owner);
|
2013-06-24 10:47:03 +04:00
|
|
|
}
|
|
|
|
|
2010-11-12 11:03:57 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2011-03-21 16:43:56 +03:00
|
|
|
|
2016-10-30 12:40:20 +03:00
|
|
|
cb_ret_t
|
2019-10-18 15:44:57 +03:00
|
|
|
widget_draw (Widget * w)
|
2013-02-11 15:35:00 +04:00
|
|
|
{
|
2016-10-30 12:40:20 +03:00
|
|
|
cb_ret_t ret = MSG_NOT_HANDLED;
|
|
|
|
|
2013-02-11 15:35:00 +04:00
|
|
|
if (w != NULL)
|
|
|
|
{
|
2016-09-27 13:53:17 +03:00
|
|
|
WGroup *g = w->owner;
|
2013-02-11 15:35:00 +04:00
|
|
|
|
2016-09-27 13:53:17 +03:00
|
|
|
if (g != NULL && widget_get_state (WIDGET (g), WST_ACTIVE))
|
2016-10-30 12:40:20 +03:00
|
|
|
ret = w->callback (w, NULL, MSG_DRAW, 0, NULL);
|
2013-02-11 15:35:00 +04:00
|
|
|
}
|
2016-10-30 12:40:20 +03:00
|
|
|
|
|
|
|
return ret;
|
2013-02-11 15:35:00 +04:00
|
|
|
}
|
|
|
|
|
2013-06-24 10:57:01 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
{
|
2016-09-27 13:53:17 +03:00
|
|
|
WGroup *g = old_w->owner;
|
2013-06-24 10:57:01 +04:00
|
|
|
gboolean should_focus = FALSE;
|
2016-10-11 10:42:44 +03:00
|
|
|
GList *holder;
|
2013-06-24 10:57:01 +04:00
|
|
|
|
2016-09-27 13:53:17 +03:00
|
|
|
if (g->widgets == NULL)
|
2013-06-24 10:57:01 +04:00
|
|
|
return;
|
|
|
|
|
2016-09-27 13:53:17 +03:00
|
|
|
if (g->current == NULL)
|
|
|
|
g->current = g->widgets;
|
2013-06-24 10:57:01 +04:00
|
|
|
|
2016-10-11 10:42:44 +03:00
|
|
|
/* locate widget position in the list */
|
2016-09-27 13:53:17 +03:00
|
|
|
if (old_w == g->current->data)
|
|
|
|
holder = g->current;
|
2016-10-11 10:42:44 +03:00
|
|
|
else
|
2016-09-27 13:53:17 +03:00
|
|
|
holder = g_list_find (g->widgets, old_w);
|
2016-10-11 10:42:44 +03:00
|
|
|
|
|
|
|
/* if old widget is focused, we should focus the new one... */
|
|
|
|
if (widget_get_state (old_w, WST_FOCUSED))
|
2013-06-24 10:57:01 +04:00
|
|
|
should_focus = TRUE;
|
2016-10-11 10:42:44 +03:00
|
|
|
/* ...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;
|
|
|
|
|
2016-09-27 16:16:38 +03:00
|
|
|
for (l = group_get_widget_next_of (holder);
|
2016-10-11 10:42:44 +03:00
|
|
|
!widget_get_options (WIDGET (l->data), WOP_SELECTABLE)
|
2016-09-27 16:16:38 +03:00
|
|
|
&& !widget_get_state (WIDGET (l->data), WST_DISABLED);
|
|
|
|
l = group_get_widget_next_of (l))
|
2016-10-11 10:42:44 +03:00
|
|
|
;
|
|
|
|
|
|
|
|
widget_select (WIDGET (l->data));
|
|
|
|
}
|
2013-06-24 10:57:01 +04:00
|
|
|
|
2016-10-11 10:42:44 +03:00
|
|
|
/* replace widget */
|
2016-09-27 13:53:17 +03:00
|
|
|
new_w->owner = g;
|
2013-06-24 10:57:01 +04:00
|
|
|
new_w->id = old_w->id;
|
2016-10-11 10:42:44 +03:00
|
|
|
holder->data = new_w;
|
2013-06-24 10:57:01 +04:00
|
|
|
|
|
|
|
send_message (old_w, NULL, MSG_DESTROY, 0, NULL);
|
|
|
|
send_message (new_w, NULL, MSG_INIT, 0, NULL);
|
|
|
|
|
|
|
|
if (should_focus)
|
2016-08-09 09:35:27 +03:00
|
|
|
widget_select (new_w);
|
2016-10-11 10:42:44 +03:00
|
|
|
else
|
2019-10-18 15:44:57 +03:00
|
|
|
widget_draw (new_w);
|
2016-07-16 21:00:28 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Select specified widget in it's owner.
|
2017-02-26 00:16:04 +03:00
|
|
|
*
|
|
|
|
* Note: this function (and widget_focus(), which it calls) is a no-op
|
|
|
|
* if the widget is already selected.
|
2016-07-16 21:00:28 +03:00
|
|
|
*
|
|
|
|
* @param w widget to be selected
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
widget_select (Widget * w)
|
|
|
|
{
|
2016-09-29 11:16:17 +03:00
|
|
|
WGroup *g;
|
2016-07-16 21:00:28 +03:00
|
|
|
|
|
|
|
if (!widget_get_options (w, WOP_SELECTABLE))
|
|
|
|
return;
|
|
|
|
|
2016-09-29 11:16:17 +03:00
|
|
|
g = GROUP (w->owner);
|
|
|
|
if (g != NULL)
|
2016-07-16 21:00:28 +03:00
|
|
|
{
|
|
|
|
if (widget_get_options (w, WOP_TOP_SELECT))
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
|
2016-09-29 11:16:17 +03:00
|
|
|
l = widget_find (WIDGET (g), w);
|
2016-07-16 21:00:28 +03:00
|
|
|
widget_reorder (l, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
widget_focus (w);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Set widget at bottom of widget list.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
widget_set_bottom (Widget * w)
|
|
|
|
{
|
2016-09-29 11:16:17 +03:00
|
|
|
widget_reorder (widget_find (WIDGET (w->owner), w), FALSE);
|
2013-06-24 10:57:01 +04:00
|
|
|
}
|
|
|
|
|
2013-06-24 11:08:10 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* 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));
|
|
|
|
}
|
|
|
|
|
2017-03-08 14:41:35 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Look up key event of widget and translate it to command ID.
|
|
|
|
* @param w widget
|
|
|
|
* @param key key event
|
|
|
|
*
|
|
|
|
* @return command ID binded with @key.
|
|
|
|
*/
|
|
|
|
|
|
|
|
long
|
|
|
|
widget_lookup_key (Widget * w, int key)
|
|
|
|
{
|
|
|
|
if (w->ext_mode)
|
|
|
|
{
|
|
|
|
w->ext_mode = FALSE;
|
|
|
|
return keybind_lookup_keymap_command (w->ext_keymap, key);
|
|
|
|
}
|
|
|
|
|
|
|
|
return keybind_lookup_keymap_command (w->keymap, key);
|
|
|
|
}
|
|
|
|
|
2016-09-29 11:16:17 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Default callback function to find widget.
|
|
|
|
*
|
|
|
|
* @param w widget
|
|
|
|
* @param what widget to find
|
|
|
|
*
|
|
|
|
* @return holder of @what if widget is @what, NULL otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
GList *
|
|
|
|
widget_default_find (const Widget * w, const Widget * what)
|
|
|
|
{
|
|
|
|
return (w != what
|
|
|
|
|| w->owner == NULL) ? NULL : g_list_find (CONST_GROUP (w->owner)->widgets, what);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default callback function to find widget by widget type using widget callback.
|
|
|
|
*
|
|
|
|
* @param w widget
|
|
|
|
* @param cb widget callback
|
|
|
|
*
|
|
|
|
* @return @w if widget callback is @cb, NULL otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
Widget *
|
|
|
|
widget_default_find_by_type (const Widget * w, widget_cb_fn cb)
|
|
|
|
{
|
|
|
|
return (w->callback == cb ? WIDGET (w) : NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
/**
|
|
|
|
* Default callback function to find widget by widget ID.
|
|
|
|
*
|
|
|
|
* @param w widget
|
|
|
|
* @param id widget ID
|
|
|
|
*
|
|
|
|
* @return @w if widget id is equal to @id, NULL otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
Widget *
|
|
|
|
widget_default_find_by_id (const Widget * w, unsigned long id)
|
|
|
|
{
|
|
|
|
return (w->id == id ? WIDGET (w) : NULL);
|
|
|
|
}
|
|
|
|
|
2019-11-17 17:02:52 +03:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default callback function to 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_default_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 = widget_draw (w);
|
|
|
|
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))
|
|
|
|
{
|
|
|
|
widget_draw (w);
|
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2013-02-11 15:35:00 +04:00
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
2011-03-21 16:43:56 +03:00
|
|
|
/* get mouse pointer location within widget */
|
2013-06-24 10:47:03 +04:00
|
|
|
|
2011-03-21 16:43:56 +03:00
|
|
|
Gpm_Event
|
|
|
|
mouse_get_local (const Gpm_Event * global, const Widget * w)
|
|
|
|
{
|
|
|
|
Gpm_Event local;
|
|
|
|
|
|
|
|
local.buttons = global->buttons;
|
2014-07-22 13:35:51 +04:00
|
|
|
#ifdef HAVE_LIBGPM
|
2016-03-19 21:11:35 +03:00
|
|
|
local.clicks = 0;
|
|
|
|
local.margin = 0;
|
2014-04-20 17:24:33 +04:00
|
|
|
local.modifiers = 0;
|
2016-03-19 21:11:35 +03:00
|
|
|
local.vc = 0;
|
2014-07-22 13:35:51 +04:00
|
|
|
#endif
|
2011-03-21 16:43:56 +03:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|