2 /** \file widget-common.h
3 * \brief Header: shared stuff of widgets
6 #ifndef MC__WIDGET_COMMON_H
7 #define MC__WIDGET_COMMON_H
9 #include "lib/keybind.h" /* global_keymap_t */
10 #include "lib/tty/mouse.h"
11 #include "lib/widget/mouse.h" /* mouse_msg_t, mouse_event_t */
13 /*** typedefs(not structures) and defined constants **********************************************/
15 #define WIDGET(x) ((Widget *)(x))
16 #define CONST_WIDGET(x) ((const Widget *)(x))
18 #define widget_gotoyx(w, _y, _x) tty_gotoyx (CONST_WIDGET(w)->rect.y + (_y), CONST_WIDGET(w)->rect.x + (_x))
19 /* Sets/clear the specified flag in the options field */
20 #define widget_want_cursor(w,i) widget_set_options(w, WOP_WANT_CURSOR, i)
21 #define widget_want_hotkey(w,i) widget_set_options(w, WOP_WANT_HOTKEY, i)
22 #define widget_want_tab(w,i) widget_set_options(w, WOP_WANT_TAB, i)
23 #define widget_idle(w,i) widget_set_state(w, WST_IDLE, i)
24 #define widget_disable(w,i) widget_set_state(w, WST_DISABLED, i)
26 /*** enums ***************************************************************************************/
31 MSG_INIT
= 0, /* Initialize widget */
32 MSG_FOCUS
, /* Draw widget in focused state or widget has got focus */
33 MSG_UNFOCUS
, /* Draw widget in unfocused state or widget has been unfocused */
34 MSG_CHANGED_FOCUS
, /* Notification to owner about focus state change */
35 MSG_ENABLE
, /* Change state to enabled */
36 MSG_DISABLE
, /* Change state to disabled */
37 MSG_DRAW
, /* Draw widget on screen */
38 MSG_KEY
, /* Sent to widgets on key press */
39 MSG_HOTKEY
, /* Sent to widget to catch preprocess key */
40 MSG_HOTKEY_HANDLED
, /* A widget has got the hotkey */
41 MSG_UNHANDLED_KEY
, /* Key that no widget handled */
42 MSG_POST_KEY
, /* The key has been handled */
43 MSG_ACTION
, /* Send to widget to handle command */
44 MSG_NOTIFY
, /* Typically sent to dialog to inform it of state-change
45 * of listboxes, check- and radiobuttons. */
46 MSG_CURSOR
, /* Sent to widget to position the cursor */
47 MSG_IDLE
, /* The idle state is active */
48 MSG_RESIZE
, /* Screen size has changed */
49 MSG_VALIDATE
, /* Dialog is to be closed */
50 MSG_END
, /* Shut down dialog */
51 MSG_DESTROY
/* Sent to widget at destruction time */
54 /* Widgets are expected to answer to the following messages:
55 MSG_FOCUS: MSG_HANDLED if the accept the focus, MSG_NOT_HANDLED if they do not.
56 MSG_UNFOCUS: MSG_HANDLED if they accept to release the focus, MSG_NOT_HANDLED if they don't.
57 MSG_KEY: MSG_HANDLED if they actually used the key, MSG_NOT_HANDLED if not.
58 MSG_HOTKEY: MSG_HANDLED if they actually used the key, MSG_NOT_HANDLED if not.
70 WOP_DEFAULT
= (0 << 0),
71 WOP_WANT_HOTKEY
= (1 << 0),
72 WOP_WANT_CURSOR
= (1 << 1),
73 WOP_WANT_TAB
= (1 << 2), /* Should the tab key be sent to the dialog? */
74 WOP_IS_INPUT
= (1 << 3),
75 WOP_SELECTABLE
= (1 << 4),
76 WOP_TOP_SELECT
= (1 << 5)
82 WST_DEFAULT
= (0 << 0),
83 WST_VISIBLE
= (1 << 0), /* Widget is visible */
84 WST_DISABLED
= (1 << 1), /* Widget cannot be selected */
86 WST_MODAL
= (1 << 3), /* Widget (dialog) is modal */
87 WST_FOCUSED
= (1 << 4),
89 WST_CONSTRUCT
= (1 << 15), /* Widget has been constructed but not run yet */
90 WST_ACTIVE
= (1 << 16), /* Dialog is visible and active */
91 WST_SUSPENDED
= (1 << 17), /* Dialog is suspended */
92 WST_CLOSED
= (1 << 18) /* Dialog is closed */
95 /* Flags for widget repositioning on dialog resize */
98 WPOS_FULLSCREEN
= (1 << 0), /* widget occupies the whole screen */
99 WPOS_CENTER_HORZ
= (1 << 1), /* center widget in horizontal */
100 WPOS_CENTER_VERT
= (1 << 2), /* center widget in vertical */
101 WPOS_CENTER
= WPOS_CENTER_HORZ
| WPOS_CENTER_VERT
, /* center widget */
102 WPOS_TRYUP
= (1 << 3), /* try to move two lines up the widget */
103 WPOS_KEEP_LEFT
= (1 << 4), /* keep widget distance to left border of dialog */
104 WPOS_KEEP_RIGHT
= (1 << 5), /* keep widget distance to right border of dialog */
105 WPOS_KEEP_TOP
= (1 << 6), /* keep widget distance to top border of dialog */
106 WPOS_KEEP_BOTTOM
= (1 << 7), /* keep widget distance to bottom border of dialog */
107 WPOS_KEEP_HORZ
= WPOS_KEEP_LEFT
| WPOS_KEEP_RIGHT
,
108 WPOS_KEEP_VERT
= WPOS_KEEP_TOP
| WPOS_KEEP_BOTTOM
,
109 WPOS_KEEP_ALL
= WPOS_KEEP_HORZ
| WPOS_KEEP_VERT
,
110 WPOS_KEEP_DEFAULT
= WPOS_KEEP_LEFT
| WPOS_KEEP_TOP
111 } widget_pos_flags_t
;
113 * If WPOS_FULLSCREEN is set then all other position flags are ignored.
114 * If WPOS_CENTER_HORZ flag is used, other horizontal flags (WPOS_KEEP_LEFT, WPOS_KEEP_RIGHT,
115 * and WPOS_KEEP_HORZ) are ignored.
116 * If WPOS_CENTER_VERT flag is used, other horizontal flags (WPOS_KEEP_TOP, WPOS_KEEP_BOTTOM,
117 * and WPOS_KEEP_VERT) are ignored.
120 /*** structures declarations (and typedefs of structures)*****************************************/
122 /* Widget callback */
123 typedef cb_ret_t (*widget_cb_fn
) (Widget
* widget
, Widget
* sender
, widget_msg_t msg
, int parm
,
125 /* Widget mouse callback */
126 typedef void (*widget_mouse_cb_fn
) (Widget
* w
, mouse_msg_t msg
, mouse_event_t
* event
);
127 /* translate mouse event and process it */
128 typedef int (*widget_mouse_handle_fn
) (Widget
* w
, Gpm_Event
* event
);
130 /* Every Widget must have this as its first element */
133 WRect rect
; /* position and size */
134 /* ATTENTION! For groups, don't change @rect members directly to avoid
135 incorrect reposion and resize of group members. */
136 widget_pos_flags_t pos_flags
; /* repositioning flags */
137 widget_options_t options
;
138 widget_state_t state
;
139 unsigned long id
; /* uniq widget ID */
140 widget_cb_fn callback
;
141 widget_mouse_cb_fn mouse_callback
;
144 /* Key-related fields */
145 const global_keymap_t
*keymap
; /* main keymap */
146 const global_keymap_t
*ext_keymap
; /* extended keymap */
147 gboolean ext_mode
; /* use keymap or ext_keymap */
149 /* Mouse-related fields. */
150 widget_mouse_handle_fn mouse_handler
;
153 /* Public members: */
154 gboolean forced_capture
; /* Overrides the 'capture' member. Set explicitly by the programmer. */
156 /* Implementation details: */
157 gboolean capture
; /* Whether the widget "owns" the mouse. */
158 mouse_msg_t last_msg
; /* The previous event type processed. */
159 int last_buttons_down
;
162 void (*make_global
) (Widget
* w
, const WRect
* delta
);
163 void (*make_local
) (Widget
* w
, const WRect
* delta
);
165 GList
*(*find
) (const Widget
* w
, const Widget
* what
);
166 Widget
*(*find_by_type
) (const Widget
* w
, widget_cb_fn cb
);
167 Widget
*(*find_by_id
) (const Widget
* w
, unsigned long id
);
170 cb_ret_t (*set_state
) (Widget
* w
, widget_state_t state
, gboolean enable
);
172 void (*destroy
) (Widget
* w
);
174 const int *(*get_colors
) (const Widget
* w
);
177 /* structure for label (caption) with hotkey, if original text does not contain
178 * hotkey, only start is valid and is equal to original text
179 * hotkey is defined as char*, but mc support only singlebyte hotkey
181 typedef struct hotkey_t
183 char *start
; /* never NULL */
184 char *hotkey
; /* can be NULL */
185 char *end
; /* can be NULL */
188 /*** global variables defined in .c file *********************************************************/
190 /*** declarations of public functions ************************************************************/
192 /* create hotkey from text */
193 hotkey_t
hotkey_new (const char *text
);
194 /* release hotkey, free all mebers of hotkey_t */
195 void hotkey_free (const hotkey_t hotkey
);
196 /* return width on terminal of hotkey */
197 int hotkey_width (const hotkey_t hotkey
);
198 /* compare two hotkeys */
199 gboolean
hotkey_equal (const hotkey_t hotkey1
, const hotkey_t hotkey2
);
200 /* draw hotkey of widget */
201 void hotkey_draw (const Widget
* w
, const hotkey_t hotkey
, gboolean focused
);
202 /* get text of hotkey */
203 char *hotkey_get_text (const hotkey_t hotkey
);
205 /* widget initialization */
206 void widget_init (Widget
* w
, const WRect
* r
, widget_cb_fn callback
,
207 widget_mouse_cb_fn mouse_callback
);
208 /* Default callback for widgets */
209 cb_ret_t
widget_default_callback (Widget
* w
, Widget
* sender
, widget_msg_t msg
, int parm
,
211 void widget_set_options (Widget
* w
, widget_options_t options
, gboolean enable
);
212 void widget_adjust_position (widget_pos_flags_t pos_flags
, WRect
* r
);
213 void widget_set_size (Widget
* w
, int y
, int x
, int lines
, int cols
);
214 void widget_set_size_rect (Widget
* w
, WRect
* r
);
215 /* select color for widget in dependence of state */
216 void widget_selectcolor (const Widget
* w
, gboolean focused
, gboolean hotkey
);
217 cb_ret_t
widget_draw (Widget
* w
);
218 void widget_erase (Widget
* w
);
219 void widget_set_visibility (Widget
* w
, gboolean make_visible
);
220 gboolean
widget_is_active (const void *w
);
221 void widget_replace (Widget
* old
, Widget
* new);
222 gboolean
widget_is_focusable (const Widget
* w
);
223 void widget_select (Widget
* w
);
224 void widget_set_bottom (Widget
* w
);
226 long widget_lookup_key (Widget
* w
, int key
);
228 void widget_default_make_global (Widget
* w
, const WRect
* delta
);
229 void widget_default_make_local (Widget
* w
, const WRect
* delta
);
231 GList
*widget_default_find (const Widget
* w
, const Widget
* what
);
232 Widget
*widget_default_find_by_type (const Widget
* w
, widget_cb_fn cb
);
233 Widget
*widget_default_find_by_id (const Widget
* w
, unsigned long id
);
235 cb_ret_t
widget_default_set_state (Widget
* w
, widget_state_t state
, gboolean enable
);
237 void widget_default_destroy (Widget
* w
);
239 /* get mouse pointer location within widget */
240 Gpm_Event
mouse_get_local (const Gpm_Event
* global
, const Widget
* w
);
241 gboolean
mouse_global_in_widget (const Gpm_Event
* event
, const Widget
* w
);
243 /* --------------------------------------------------------------------------------------------- */
244 /*** inline functions ****************************************************************************/
245 /* --------------------------------------------------------------------------------------------- */
247 static inline cb_ret_t
248 send_message (void *w
, void *sender
, widget_msg_t msg
, int parm
, void *data
)
250 cb_ret_t ret
= MSG_NOT_HANDLED
;
253 if (w
!= NULL
) /* This must be always true, but... */
255 ret
= WIDGET (w
)->callback (WIDGET (w
), WIDGET (sender
), msg
, parm
, data
);
260 /* --------------------------------------------------------------------------------------------- */
262 * Check whether one or several option flags are set or not.
264 * @param options widget option flags
266 * @return TRUE if all requested option flags are set, FALSE otherwise.
269 static inline gboolean
270 widget_get_options (const Widget
*w
, widget_options_t options
)
272 return ((w
->options
& options
) == options
);
275 /* --------------------------------------------------------------------------------------------- */
278 * Check whether one or several state flags are set or not.
280 * @param state widget state flags
282 * @return TRUE if all requested state flags are set, FALSE otherwise.
285 static inline gboolean
286 widget_get_state (const Widget
*w
, widget_state_t state
)
288 return ((w
->state
& state
) == state
);
291 /* --------------------------------------------------------------------------------------------- */
294 * Convert widget coordinates from local (relative to owner) to global (relative to screen).
300 widget_make_global (Widget
*w
)
302 w
->make_global (w
, NULL
);
305 /* --------------------------------------------------------------------------------------------- */
308 * Convert widget coordinates from global (relative to screen) to local (relative to owner).
314 widget_make_local (Widget
*w
)
316 w
->make_local (w
, NULL
);
319 /* --------------------------------------------------------------------------------------------- */
325 * @param what widget to find
327 * @return result of @w->find()
330 static inline GList
*
331 widget_find (const Widget
*w
, const Widget
*what
)
333 return w
->find (w
, what
);
336 /* --------------------------------------------------------------------------------------------- */
339 * Find widget by widget type using widget callback.
342 * @param cb widget callback
344 * @return result of @w->find_by_type()
347 static inline Widget
*
348 widget_find_by_type (const Widget
*w
, widget_cb_fn cb
)
350 return w
->find_by_type (w
, cb
);
353 /* --------------------------------------------------------------------------------------------- */
355 * Find widget by widget ID.
358 * @param id widget ID
360 * @return result of @w->find_by_id()
363 static inline Widget
*
364 widget_find_by_id (const Widget
*w
, unsigned long id
)
366 return w
->find_by_id (w
, id
);
369 /* --------------------------------------------------------------------------------------------- */
371 * Modify state of widget.
374 * @param state widget state flag to modify
375 * @param enable specifies whether to turn the flag on (TRUE) or off (FALSE).
376 * Only one flag per call can be modified.
377 * @return MSG_HANDLED if set was handled successfully, MSG_NOT_HANDLED otherwise.
380 static inline cb_ret_t
381 widget_set_state (Widget
*w
, widget_state_t state
, gboolean enable
)
383 return w
->set_state (w
, state
, enable
);
386 /* --------------------------------------------------------------------------------------------- */
394 widget_destroy (Widget
*w
)
399 /* --------------------------------------------------------------------------------------------- */
402 * Get color colors of widget.
405 * @return color colors
407 static inline const int *
408 widget_get_colors (const Widget
*w
)
410 return w
->get_colors (w
);
413 /* --------------------------------------------------------------------------------------------- */
415 * Update cursor position in the specified widget.
419 * @return TRUE if cursor was updated successfully, FALSE otherwise
422 static inline gboolean
423 widget_update_cursor (Widget
*w
)
425 return (send_message (w
, NULL
, MSG_CURSOR
, 0, NULL
) == MSG_HANDLED
);
428 /* --------------------------------------------------------------------------------------------- */
431 widget_show (Widget
*w
)
433 widget_set_visibility (w
, TRUE
);
436 /* --------------------------------------------------------------------------------------------- */
439 widget_hide (Widget
*w
)
441 widget_set_visibility (w
, FALSE
);
445 /* --------------------------------------------------------------------------------------------- */
447 * Check whether two widgets are overlapped or not.
448 * @param a 1st widget
449 * @param b 2nd widget
451 * @return TRUE if widgets are overlapped, FALSE otherwise.
454 static inline gboolean
455 widget_overlapped (const Widget
*a
, const Widget
*b
)
457 return rects_are_overlapped (&a
->rect
, &b
->rect
);
460 /* --------------------------------------------------------------------------------------------- */
462 #endif /* MC__WIDGET_COMMON_H */