123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463 |
- /** \file widget-common.h
- * \brief Header: shared stuff of widgets
- */
- #ifndef MC__WIDGET_COMMON_H
- #define MC__WIDGET_COMMON_H
- #include "lib/keybind.h" /* global_keymap_t */
- #include "lib/tty/mouse.h"
- #include "lib/widget/mouse.h" /* mouse_msg_t, mouse_event_t */
- /*** typedefs(not structures) and defined constants **********************************************/
- #define WIDGET(x) ((Widget *) (x))
- #define CONST_WIDGET(x) ((const Widget *) (x))
- #define widget_gotoyx(w, _y, _x) \
- tty_gotoyx (CONST_WIDGET (w)->rect.y + (_y), CONST_WIDGET (w)->rect.x + (_x))
- /* Sets/clear the specified flag in the options field */
- #define widget_want_cursor(w, i) widget_set_options (w, WOP_WANT_CURSOR, i)
- #define widget_want_hotkey(w, i) widget_set_options (w, WOP_WANT_HOTKEY, i)
- #define widget_want_tab(w, i) widget_set_options (w, WOP_WANT_TAB, i)
- #define widget_idle(w, i) widget_set_state (w, WST_IDLE, i)
- #define widget_disable(w, i) widget_set_state (w, WST_DISABLED, i)
- /*** enums ***************************************************************************************/
- /* Widget messages */
- typedef enum
- {
- MSG_INIT = 0, /* Initialize widget */
- MSG_FOCUS, /* Draw widget in focused state or widget has got focus */
- MSG_UNFOCUS, /* Draw widget in unfocused state or widget has been unfocused */
- MSG_CHANGED_FOCUS, /* Notification to owner about focus state change */
- MSG_ENABLE, /* Change state to enabled */
- MSG_DISABLE, /* Change state to disabled */
- MSG_DRAW, /* Draw widget on screen */
- MSG_KEY, /* Sent to widgets on key press */
- MSG_HOTKEY, /* Sent to widget to catch preprocess key */
- MSG_HOTKEY_HANDLED, /* A widget has got the hotkey */
- MSG_UNHANDLED_KEY, /* Key that no widget handled */
- MSG_POST_KEY, /* The key has been handled */
- MSG_ACTION, /* Send to widget to handle command */
- MSG_NOTIFY, /* Typically sent to dialog to inform it of state-change
- * of listboxes, check- and radiobuttons. */
- MSG_CURSOR, /* Sent to widget to position the cursor */
- MSG_IDLE, /* The idle state is active */
- MSG_RESIZE, /* Screen size has changed */
- MSG_VALIDATE, /* Dialog is to be closed */
- MSG_END, /* Shut down dialog */
- MSG_DESTROY /* Sent to widget at destruction time */
- } widget_msg_t;
- /* Widgets are expected to answer to the following messages:
- MSG_FOCUS: MSG_HANDLED if the accept the focus, MSG_NOT_HANDLED if they do not.
- MSG_UNFOCUS: MSG_HANDLED if they accept to release the focus, MSG_NOT_HANDLED if they don't.
- MSG_KEY: MSG_HANDLED if they actually used the key, MSG_NOT_HANDLED if not.
- MSG_HOTKEY: MSG_HANDLED if they actually used the key, MSG_NOT_HANDLED if not.
- */
- typedef enum
- {
- MSG_NOT_HANDLED = 0,
- MSG_HANDLED = 1
- } cb_ret_t;
- /* Widget options */
- typedef enum
- {
- WOP_DEFAULT = (0 << 0),
- WOP_WANT_HOTKEY = (1 << 0),
- WOP_WANT_CURSOR = (1 << 1),
- WOP_WANT_TAB = (1 << 2), /* Should the tab key be sent to the dialog? */
- WOP_IS_INPUT = (1 << 3),
- WOP_SELECTABLE = (1 << 4),
- WOP_TOP_SELECT = (1 << 5)
- } widget_options_t;
- /* Widget state */
- typedef enum
- {
- WST_DEFAULT = (0 << 0),
- WST_VISIBLE = (1 << 0), /* Widget is visible */
- WST_DISABLED = (1 << 1), /* Widget cannot be selected */
- WST_IDLE = (1 << 2),
- WST_MODAL = (1 << 3), /* Widget (dialog) is modal */
- WST_FOCUSED = (1 << 4),
- WST_CONSTRUCT = (1 << 15), /* Widget has been constructed but not run yet */
- WST_ACTIVE = (1 << 16), /* Dialog is visible and active */
- WST_SUSPENDED = (1 << 17), /* Dialog is suspended */
- WST_CLOSED = (1 << 18) /* Dialog is closed */
- } widget_state_t;
- /* Flags for widget repositioning on dialog resize */
- typedef enum
- {
- WPOS_FULLSCREEN = (1 << 0), /* widget occupies the whole screen */
- WPOS_CENTER_HORZ = (1 << 1), /* center widget in horizontal */
- WPOS_CENTER_VERT = (1 << 2), /* center widget in vertical */
- WPOS_CENTER = WPOS_CENTER_HORZ | WPOS_CENTER_VERT, /* center widget */
- WPOS_TRYUP = (1 << 3), /* try to move two lines up the widget */
- WPOS_KEEP_LEFT = (1 << 4), /* keep widget distance to left border of dialog */
- WPOS_KEEP_RIGHT = (1 << 5), /* keep widget distance to right border of dialog */
- WPOS_KEEP_TOP = (1 << 6), /* keep widget distance to top border of dialog */
- WPOS_KEEP_BOTTOM = (1 << 7), /* keep widget distance to bottom border of dialog */
- WPOS_KEEP_HORZ = WPOS_KEEP_LEFT | WPOS_KEEP_RIGHT,
- WPOS_KEEP_VERT = WPOS_KEEP_TOP | WPOS_KEEP_BOTTOM,
- WPOS_KEEP_ALL = WPOS_KEEP_HORZ | WPOS_KEEP_VERT,
- WPOS_KEEP_DEFAULT = WPOS_KEEP_LEFT | WPOS_KEEP_TOP
- } widget_pos_flags_t;
- /* NOTES:
- * If WPOS_FULLSCREEN is set then all other position flags are ignored.
- * If WPOS_CENTER_HORZ flag is used, other horizontal flags (WPOS_KEEP_LEFT, WPOS_KEEP_RIGHT,
- * and WPOS_KEEP_HORZ) are ignored.
- * If WPOS_CENTER_VERT flag is used, other horizontal flags (WPOS_KEEP_TOP, WPOS_KEEP_BOTTOM,
- * and WPOS_KEEP_VERT) are ignored.
- */
- /*** structures declarations (and typedefs of structures)*****************************************/
- /* Widget callback */
- typedef cb_ret_t (*widget_cb_fn) (Widget *widget, Widget *sender, widget_msg_t msg, int parm,
- void *data);
- /* Widget mouse callback */
- typedef void (*widget_mouse_cb_fn) (Widget *w, mouse_msg_t msg, mouse_event_t *event);
- /* translate mouse event and process it */
- typedef int (*widget_mouse_handle_fn) (Widget *w, Gpm_Event *event);
- /* Every Widget must have this as its first element */
- struct Widget
- {
- WRect rect; /* position and size */
- /* ATTENTION! For groups, don't change @rect members directly to avoid
- incorrect reposion and resize of group members. */
- widget_pos_flags_t pos_flags; /* repositioning flags */
- widget_options_t options;
- widget_state_t state;
- unsigned long id; /* uniq widget ID */
- widget_cb_fn callback;
- widget_mouse_cb_fn mouse_callback;
- WGroup *owner;
- /* Key-related fields */
- const global_keymap_t *keymap; /* main keymap */
- const global_keymap_t *ext_keymap; /* extended keymap */
- gboolean ext_mode; /* use keymap or ext_keymap */
- /* Mouse-related fields. */
- widget_mouse_handle_fn mouse_handler;
- struct
- {
- /* Public members: */
- gboolean
- forced_capture; /* Overrides the 'capture' member. Set explicitly by the programmer. */
- /* Implementation details: */
- gboolean capture; /* Whether the widget "owns" the mouse. */
- mouse_msg_t last_msg; /* The previous event type processed. */
- int last_buttons_down;
- } mouse;
- void (*make_global) (Widget *w, const WRect *delta);
- void (*make_local) (Widget *w, const WRect *delta);
- GList *(*find) (const Widget *w, const Widget *what);
- Widget *(*find_by_type) (const Widget *w, widget_cb_fn cb);
- Widget *(*find_by_id) (const Widget *w, unsigned long id);
- /* *INDENT-OFF* */
- cb_ret_t (*set_state) (Widget *w, widget_state_t state, gboolean enable);
- /* *INDENT-ON* */
- void (*destroy) (Widget *w);
- const int *(*get_colors) (const Widget *w);
- };
- /* structure for label (caption) with hotkey, if original text does not contain
- * hotkey, only start is valid and is equal to original text
- * hotkey is defined as char*, but mc support only singlebyte hotkey
- */
- typedef struct hotkey_t
- {
- char *start; /* never NULL */
- char *hotkey; /* can be NULL */
- char *end; /* can be NULL */
- } hotkey_t;
- /*** global variables defined in .c file *********************************************************/
- /*** declarations of public functions ************************************************************/
- /* create hotkey from text */
- hotkey_t hotkey_new (const char *text);
- /* release hotkey, free all mebers of hotkey_t */
- void hotkey_free (const hotkey_t hotkey);
- /* return width on terminal of hotkey */
- int hotkey_width (const hotkey_t hotkey);
- /* compare two hotkeys */
- gboolean hotkey_equal (const hotkey_t hotkey1, const hotkey_t hotkey2);
- /* draw hotkey of widget */
- void hotkey_draw (const Widget *w, const hotkey_t hotkey, gboolean focused);
- /* get text of hotkey */
- char *hotkey_get_text (const hotkey_t hotkey);
- /* widget initialization */
- void widget_init (Widget *w, const WRect *r, widget_cb_fn callback,
- widget_mouse_cb_fn mouse_callback);
- /* Default callback for widgets */
- cb_ret_t widget_default_callback (Widget *w, Widget *sender, widget_msg_t msg, int parm,
- void *data);
- void widget_set_options (Widget *w, widget_options_t options, gboolean enable);
- void widget_adjust_position (widget_pos_flags_t pos_flags, WRect *r);
- void widget_set_size (Widget *w, int y, int x, int lines, int cols);
- void widget_set_size_rect (Widget *w, WRect *r);
- /* select color for widget in dependence of state */
- void widget_selectcolor (const Widget *w, gboolean focused, gboolean hotkey);
- cb_ret_t widget_draw (Widget *w);
- void widget_erase (Widget *w);
- void widget_set_visibility (Widget *w, gboolean make_visible);
- gboolean widget_is_active (const void *w);
- void widget_replace (Widget *old, Widget *new);
- gboolean widget_is_focusable (const Widget *w);
- void widget_select (Widget *w);
- void widget_set_bottom (Widget *w);
- long widget_lookup_key (Widget *w, int key);
- void widget_default_make_global (Widget *w, const WRect *delta);
- void widget_default_make_local (Widget *w, const WRect *delta);
- GList *widget_default_find (const Widget *w, const Widget *what);
- Widget *widget_default_find_by_type (const Widget *w, widget_cb_fn cb);
- Widget *widget_default_find_by_id (const Widget *w, unsigned long id);
- cb_ret_t widget_default_set_state (Widget *w, widget_state_t state, gboolean enable);
- void widget_default_destroy (Widget *w);
- /* get mouse pointer location within widget */
- Gpm_Event mouse_get_local (const Gpm_Event *global, const Widget *w);
- gboolean mouse_global_in_widget (const Gpm_Event *event, const Widget *w);
- /* --------------------------------------------------------------------------------------------- */
- /*** inline functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- static inline cb_ret_t
- send_message (void *w, void *sender, widget_msg_t msg, int parm, void *data)
- {
- cb_ret_t ret = MSG_NOT_HANDLED;
- #if 1
- if (w != NULL) /* This must be always true, but... */
- #endif
- ret = WIDGET (w)->callback (WIDGET (w), WIDGET (sender), msg, parm, data);
- return ret;
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Check whether one or several option flags are set or not.
- * @param w widget
- * @param options widget option flags
- *
- * @return TRUE if all requested option flags are set, FALSE otherwise.
- */
- static inline gboolean
- widget_get_options (const Widget *w, widget_options_t options)
- {
- return ((w->options & options) == options);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Check whether one or several state flags are set or not.
- * @param w widget
- * @param state widget state flags
- *
- * @return TRUE if all requested state flags are set, FALSE otherwise.
- */
- static inline gboolean
- widget_get_state (const Widget *w, widget_state_t state)
- {
- return ((w->state & state) == state);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Convert widget coordinates from local (relative to owner) to global (relative to screen).
- *
- * @param w widget
- */
- static inline void
- widget_make_global (Widget *w)
- {
- w->make_global (w, NULL);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Convert widget coordinates from global (relative to screen) to local (relative to owner).
- *
- * @param w widget
- */
- static inline void
- widget_make_local (Widget *w)
- {
- w->make_local (w, NULL);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Find widget.
- *
- * @param w widget
- * @param what widget to find
- *
- * @return result of @w->find()
- */
- static inline GList *
- widget_find (const Widget *w, const Widget *what)
- {
- return w->find (w, what);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Find widget by widget type using widget callback.
- *
- * @param w widget
- * @param cb widget callback
- *
- * @return result of @w->find_by_type()
- */
- static inline Widget *
- widget_find_by_type (const Widget *w, widget_cb_fn cb)
- {
- return w->find_by_type (w, cb);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Find widget by widget ID.
- *
- * @param w widget
- * @param id widget ID
- *
- * @return result of @w->find_by_id()
- */
- static inline Widget *
- widget_find_by_id (const Widget *w, unsigned long id)
- {
- return w->find_by_id (w, id);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * 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.
- */
- static inline cb_ret_t
- widget_set_state (Widget *w, widget_state_t state, gboolean enable)
- {
- return w->set_state (w, state, enable);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Destroy widget.
- *
- * @param w widget
- */
- static inline void
- widget_destroy (Widget *w)
- {
- w->destroy (w);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Get color colors of widget.
- *
- * @param w widget
- * @return color colors
- */
- static inline const int *
- widget_get_colors (const Widget *w)
- {
- return w->get_colors (w);
- }
- /* --------------------------------------------------------------------------------------------- */
- /**
- * Update cursor position in the specified widget.
- *
- * @param w widget
- *
- * @return TRUE if cursor was updated successfully, FALSE otherwise
- */
- static inline gboolean
- widget_update_cursor (Widget *w)
- {
- return (send_message (w, NULL, MSG_CURSOR, 0, NULL) == MSG_HANDLED);
- }
- /* --------------------------------------------------------------------------------------------- */
- static inline void
- widget_show (Widget *w)
- {
- widget_set_visibility (w, TRUE);
- }
- /* --------------------------------------------------------------------------------------------- */
- static inline void
- widget_hide (Widget *w)
- {
- widget_set_visibility (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.
- */
- static inline gboolean
- widget_overlapped (const Widget *a, const Widget *b)
- {
- return rects_are_overlapped (&a->rect, &b->rect);
- }
- /* --------------------------------------------------------------------------------------------- */
- #endif /* MC__WIDGET_COMMON_H */
|