/* Dialog box features module for the Midnight Commander Copyright (C) 1994, 1995 Radek Doulik, 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #ifndef MC_DLG_H #define MC_DLG_H #include "mouse.h" /* Color constants */ #define FOCUSC h->color[1] #define NORMALC h->color[0] #define HOT_NORMALC h->color[2] #define HOT_FOCUSC h->color[3] /* Common return values */ #define B_EXIT 0 #define B_CANCEL 1 #define B_ENTER 2 #define B_HELP 3 #define B_USER 100 /* Widget messages */ typedef enum { WIDGET_INIT, /* Initialize widget */ WIDGET_FOCUS, /* Draw widget in focused state */ WIDGET_UNFOCUS, /* Draw widget in unfocused state */ WIDGET_DRAW, /* Sent to widget to draw themselves */ WIDGET_KEY, /* Sent to widgets on key press */ WIDGET_HOTKEY, /* Sent to widget to catch preprocess key */ WIDGET_DESTROY, /* Sent to widget at destruction time */ WIDGET_CURSOR, /* Sent to widget to position the cursor */ WIDGET_IDLE, /* Send to widgets with options & W_WANT_IDLE*/ } widget_msg_t; typedef enum { MSG_NOT_HANDLED, MSG_HANDLED } cb_ret_t; /* Widgets are expected to answer to the following messages: WIDGET_FOCUS: 1 if the accept the focus, 0 if they do not. WIDGET_UNFOCUS: 1 if they accept to release the focus, 0 if they don't. WIDGET_KEY: 1 if they actually used the key, 0 if not. WIDGET_HOTKEY: 1 if they actually used the key, 0 if not. */ /* Dialog messages */ typedef enum { DLG_KEY, /* Key before sending to widget */ DLG_INIT, /* Initialize dialog */ DLG_END, /* Shut down dialog */ DLG_ACTION, /* State of check- and radioboxes has changed */ DLG_DRAW, /* Draw dialog on screen */ DLG_FOCUS, /* A widget has got focus */ DLG_UNFOCUS, /* A widget has been unfocused */ DLG_RESIZE, /* Window size has changed */ DLG_POST_KEY, /* The key has been handled */ DLG_IDLE, /* The idle state is active */ DLG_UNHANDLED_KEY, /* Key that no widget handled */ DLG_HOTKEY_HANDLED, /* A widget has got the hotkey */ DLG_VALIDATE /* Dialog is to be closed */ } dlg_msg_t; /* Dialog callback */ struct Dlg_head; typedef cb_ret_t (*dlg_cb_fn)(struct Dlg_head *h, dlg_msg_t msg, int parm); typedef struct Dlg_head { /* Set by the user */ int flags; /* User flags */ const char *help_ctx; /* Name of the help entry */ const int *color; /* Color set */ /*notconst*/ char *title; /* Title of the dialog */ /* Set and received by the user */ int ret_value; /* Result of run_dlg() */ /* Geometry */ int x, y; /* Position relative to screen origin */ int cols, lines; /* Width and height of the window */ /* Internal flags */ int running:1; /* The dialog is currently active */ int fullscreen:1; /* Parents dialogs don't need refresh */ int mouse_status; /* For the autorepeat status of the mouse */ /* Internal variables */ int count; /* Number of widgets */ struct Widget *current; /* Curently active widget */ dlg_cb_fn callback; struct Dlg_head *parent; /* Parent dialog */ } Dlg_head; typedef struct Widget Widget; /* Widget callback */ typedef cb_ret_t (*callback_fn) (Widget *widget, widget_msg_t msg, int parm); /* Every Widget must have this as it's first element */ struct Widget { int x, y; int cols, lines; int options; int dlg_id; /* Number of the widget, starting with 0 */ struct Widget *next; struct Widget *prev; callback_fn callback; /* The callback function */ mouse_h mouse; struct Dlg_head *parent; }; /* The options for the widgets */ #define W_WANT_HOTKEY 2 #define W_WANT_CURSOR 4 #define W_WANT_IDLE 8 #define W_IS_INPUT 16 /* draw box in window */ void draw_box (Dlg_head *h, int y, int x, int ys, int xs); /* doubled line if possible */ void draw_double_box (Dlg_head *h, int y, int x, int ys, int xs); /* Creates a dialog head */ Dlg_head *create_dlg (int y1, int x1, int lines, int cols, const int *color_set, dlg_cb_fn callback, const char *help_ctx, const char *title, int flags); /* The flags: */ #define DLG_REVERSE 32 /* Tab order is opposite to the add order */ #define DLG_WANT_TAB 16 /* Should the tab key be sent to the dialog? */ #define DLG_WANT_IDLE 8 /* Dialog wants idle events */ #define DLG_COMPACT 4 /* Suppress spaces around the frame */ #define DLG_TRYUP 2 /* Try to move two lines up the dialog */ #define DLG_CENTER 1 /* Center the dialog */ #define DLG_NONE 0 /* No options */ int add_widget (Dlg_head *dest, void *Widget); /* Runs dialog d */ int run_dlg (Dlg_head *d); void dlg_run_done (Dlg_head *h); void dlg_process_event (Dlg_head *h, int key, Gpm_Event *event); void init_dlg (Dlg_head *h); /* To activate/deactivate the idle message generation */ void set_idle_proc (Dlg_head *d, int enable); void dlg_redraw (Dlg_head *h); void destroy_dlg (Dlg_head *h); void widget_set_size (Widget *widget, int x1, int y1, int x2, int y2); void dlg_broadcast_msg (Dlg_head *h, widget_msg_t message, int reverse); void init_widget (Widget *w, int y, int x, int lines, int cols, callback_fn callback, mouse_h mouse_handler); /* Default callback for dialogs */ cb_ret_t default_dlg_callback (Dlg_head *h, dlg_msg_t msg, int parm); /* Default callback for widgets */ cb_ret_t default_proc (widget_msg_t msg, int parm); /* Default paint routine for dialogs */ void common_dialog_repaint (struct Dlg_head *h); #define widget_move(w, _y, _x) move(((Widget *)(w))->y + _y, \ ((Widget *)(w))->x + _x) #define dlg_move(h, _y, _x) move(((Dlg_head *)(h))->y + _y, \ ((Dlg_head *)(h))->x + _x) extern Dlg_head *current_dlg; /* A hook list for idle events */ extern Hook *idle_hook; static inline cb_ret_t send_message (Widget *w, widget_msg_t msg, int parm) { return (*(w->callback)) (w, msg, parm); } /* Return 1 if the widget is active, 0 otherwise */ static inline int dlg_widget_active (void *w) { Widget *w1 = (Widget *) w; return (w1->parent->current == w1); } void dlg_replace_widget (Dlg_head *h, Widget *old, Widget *new); int dlg_overlap (Widget *a, Widget *b); void widget_erase (Widget *); void dlg_erase (Dlg_head *h); void dlg_stop (Dlg_head *h); /* Widget selection */ void dlg_select_widget (Dlg_head *h, void *widget); void dlg_one_up (Dlg_head *h); void dlg_one_down (Dlg_head *h); int dlg_focus (Dlg_head *h); Widget *find_widget_type (Dlg_head *h, callback_fn callback); void dlg_select_by_id (Dlg_head *h, int id); /* Redraw all dialogs */ void do_refresh (void); /* Sets/clear the specified flag in the options field */ #define widget_option(w,f,i) \ w.options = ((i) ? (w.options | (f)) : (w.options & (~(f)))) #define widget_want_cursor(w,i) widget_option(w, W_WANT_CURSOR, i) #define widget_want_hotkey(w,i) widget_option(w, W_WANT_HOTKEY, i) /* Used in load_prompt() */ void update_cursor (Dlg_head *h); #endif /* MC_DLG_H */