dialog.c 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341
  1. /*
  2. Dialog box features module for the Midnight Commander
  3. Copyright (C) 1994, 1995, 1998, 1999, 2000, 2001, 2002, 2003, 2004,
  4. 2005, 2007, 2009, 2010, 2011, 2013
  5. The Free Software Foundation, Inc.
  6. This file is part of the Midnight Commander.
  7. The Midnight Commander is free software: you can redistribute it
  8. and/or modify it under the terms of the GNU General Public License as
  9. published by the Free Software Foundation, either version 3 of the License,
  10. or (at your option) any later version.
  11. The Midnight Commander is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. GNU General Public License for more details.
  15. You should have received a copy of the GNU General Public License
  16. along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. /** \file dialog.c
  19. * \brief Source: dialog box features module
  20. */
  21. #include <config.h>
  22. #include <ctype.h>
  23. #include <errno.h>
  24. #include <stdlib.h>
  25. #include <stdio.h>
  26. #include <string.h>
  27. #include <sys/types.h>
  28. #include <sys/stat.h>
  29. #include <fcntl.h> /* open() */
  30. #include "lib/global.h"
  31. #include "lib/tty/tty.h"
  32. #include "lib/skin.h"
  33. #include "lib/tty/key.h"
  34. #include "lib/strutil.h"
  35. #include "lib/widget.h"
  36. #include "lib/fileloc.h" /* MC_HISTORY_FILE */
  37. #include "lib/event.h" /* mc_event_raise() */
  38. /*** global variables ****************************************************************************/
  39. /* Color styles for normal and error dialogs */
  40. dlg_colors_t dialog_colors;
  41. dlg_colors_t alarm_colors;
  42. /* Primitive way to check if the the current dialog is our dialog */
  43. /* This is needed by async routines like load_prompt */
  44. GList *top_dlg = NULL;
  45. /* A hook list for idle events */
  46. hook_t *idle_hook = NULL;
  47. /* If set then dialogs just clean the screen when refreshing, else */
  48. /* they do a complete refresh, refreshing all the parts of the program */
  49. int fast_refresh = 0;
  50. /* left click outside of dialog closes it */
  51. int mouse_close_dialog = 0;
  52. const global_keymap_t *dialog_map = NULL;
  53. /*** file scope macro definitions ****************************************************************/
  54. /*** file scope type declarations ****************************************************************/
  55. /** What to do if the requested widget doesn't take focus */
  56. typedef enum
  57. {
  58. SELECT_NEXT, /* go the the next widget */
  59. SELECT_PREV, /* go the the previous widget */
  60. SELECT_EXACT /* use current widget */
  61. } select_dir_t;
  62. /*** file scope variables ************************************************************************/
  63. /*** file scope functions ************************************************************************/
  64. static GList *
  65. dlg_widget_next (WDialog * h, GList * l)
  66. {
  67. GList *next;
  68. next = g_list_next (l);
  69. if (next == NULL)
  70. next = h->widgets;
  71. return next;
  72. }
  73. /* --------------------------------------------------------------------------------------------- */
  74. static GList *
  75. dlg_widget_prev (WDialog * h, GList * l)
  76. {
  77. GList *prev;
  78. prev = g_list_previous (l);
  79. if (prev == NULL)
  80. prev = g_list_last (h->widgets);
  81. return prev;
  82. }
  83. /* --------------------------------------------------------------------------------------------- */
  84. /**
  85. * Read histories from the ${XDG_CACHE_HOME}/mc/history file
  86. */
  87. static void
  88. dlg_read_history (WDialog * h)
  89. {
  90. char *profile;
  91. ev_history_load_save_t event_data;
  92. if (num_history_items_recorded == 0) /* this is how to disable */
  93. return;
  94. profile = mc_config_get_full_path (MC_HISTORY_FILE);
  95. event_data.cfg = mc_config_init (profile, TRUE);
  96. event_data.receiver = NULL;
  97. /* create all histories in dialog */
  98. mc_event_raise (h->event_group, MCEVENT_HISTORY_LOAD, &event_data);
  99. mc_config_deinit (event_data.cfg);
  100. g_free (profile);
  101. }
  102. /* --------------------------------------------------------------------------------------------- */
  103. static gboolean
  104. dlg_unfocus (WDialog * h)
  105. {
  106. /* we can unfocus disabled widget */
  107. if ((h->current != NULL) && (h->state == DLG_CONSTRUCT || h->state == DLG_ACTIVE))
  108. {
  109. Widget *current = WIDGET (h->current->data);
  110. if (send_message (current, NULL, MSG_UNFOCUS, 0, NULL) == MSG_HANDLED)
  111. {
  112. send_message (h, current, MSG_UNFOCUS, 0, NULL);
  113. return TRUE;
  114. }
  115. }
  116. return FALSE;
  117. }
  118. /* --------------------------------------------------------------------------------------------- */
  119. static int
  120. dlg_find_widget_callback (const void *a, const void *b)
  121. {
  122. const Widget *w = WIDGET (a);
  123. widget_cb_fn f = (widget_cb_fn) b;
  124. return (w->callback == f) ? 0 : 1;
  125. }
  126. /* --------------------------------------------------------------------------------------------- */
  127. /**
  128. * Try to select another widget. If forward is set, follow tab order.
  129. * Otherwise go to the previous widget.
  130. */
  131. static void
  132. do_select_widget (WDialog * h, GList * w, select_dir_t dir)
  133. {
  134. Widget *w0 = WIDGET (h->current->data);
  135. if (!dlg_unfocus (h))
  136. return;
  137. h->current = w;
  138. do
  139. {
  140. if (dlg_focus (h))
  141. break;
  142. switch (dir)
  143. {
  144. case SELECT_EXACT:
  145. h->current = g_list_find (h->widgets, w0);
  146. if (dlg_focus (h))
  147. return;
  148. /* try find another widget that can take focus */
  149. dir = SELECT_NEXT;
  150. /* fallthrough */
  151. case SELECT_NEXT:
  152. h->current = dlg_widget_next (h, h->current);
  153. break;
  154. case SELECT_PREV:
  155. h->current = dlg_widget_prev (h, h->current);
  156. break;
  157. }
  158. }
  159. while (h->current != w /* && (WIDGET (h->current->data)->options & W_DISABLED) == 0 */ );
  160. if (widget_overlapped (w0, WIDGET (h->current->data)))
  161. {
  162. send_message (h->current->data, NULL, MSG_DRAW, 0, NULL);
  163. send_message (h->current->data, NULL, MSG_FOCUS, 0, NULL);
  164. }
  165. }
  166. /* --------------------------------------------------------------------------------------------- */
  167. static void
  168. refresh_cmd (void)
  169. {
  170. #ifdef HAVE_SLANG
  171. tty_touch_screen ();
  172. mc_refresh ();
  173. #else
  174. /* Use this if the refreshes fail */
  175. clr_scr ();
  176. repaint_screen ();
  177. #endif /* HAVE_SLANG */
  178. }
  179. /* --------------------------------------------------------------------------------------------- */
  180. static cb_ret_t
  181. dlg_execute_cmd (WDialog * h, unsigned long command)
  182. {
  183. cb_ret_t ret = MSG_HANDLED;
  184. switch (command)
  185. {
  186. case CK_Ok:
  187. h->ret_value = B_ENTER;
  188. dlg_stop (h);
  189. break;
  190. case CK_Cancel:
  191. h->ret_value = B_CANCEL;
  192. dlg_stop (h);
  193. break;
  194. case CK_Up:
  195. case CK_Left:
  196. dlg_one_up (h);
  197. break;
  198. case CK_Down:
  199. case CK_Right:
  200. dlg_one_down (h);
  201. break;
  202. case CK_Help:
  203. {
  204. ev_help_t event_data = { NULL, h->help_ctx };
  205. mc_event_raise (MCEVENT_GROUP_CORE, "help", &event_data);
  206. }
  207. break;
  208. case CK_Suspend:
  209. mc_event_raise (MCEVENT_GROUP_CORE, "suspend", NULL);
  210. refresh_cmd ();
  211. break;
  212. case CK_Refresh:
  213. refresh_cmd ();
  214. break;
  215. case CK_ScreenList:
  216. if (!h->modal)
  217. dialog_switch_list ();
  218. else
  219. ret = MSG_NOT_HANDLED;
  220. break;
  221. case CK_ScreenNext:
  222. if (!h->modal)
  223. dialog_switch_next ();
  224. else
  225. ret = MSG_NOT_HANDLED;
  226. break;
  227. case CK_ScreenPrev:
  228. if (!h->modal)
  229. dialog_switch_prev ();
  230. else
  231. ret = MSG_NOT_HANDLED;
  232. break;
  233. default:
  234. ret = MSG_NOT_HANDLED;
  235. }
  236. return ret;
  237. }
  238. /* --------------------------------------------------------------------------------------------- */
  239. static cb_ret_t
  240. dlg_handle_key (WDialog * h, int d_key)
  241. {
  242. unsigned long command;
  243. command = keybind_lookup_keymap_command (dialog_map, d_key);
  244. if (command == CK_IgnoreKey)
  245. return MSG_NOT_HANDLED;
  246. if (send_message (h, NULL, MSG_ACTION, command, NULL) == MSG_HANDLED
  247. || dlg_execute_cmd (h, command) == MSG_HANDLED)
  248. return MSG_HANDLED;
  249. return MSG_NOT_HANDLED;
  250. }
  251. /* --------------------------------------------------------------------------------------------- */
  252. static int
  253. dlg_mouse_event (WDialog * h, Gpm_Event * event)
  254. {
  255. Widget *wh = WIDGET (h);
  256. GList *p, *first;
  257. /* close the dialog by mouse left click out of dialog area */
  258. if (mouse_close_dialog && !h->fullscreen && ((event->buttons & GPM_B_LEFT) != 0)
  259. && ((event->type & GPM_DOWN) != 0) && !mouse_global_in_widget (event, wh))
  260. {
  261. h->ret_value = B_CANCEL;
  262. dlg_stop (h);
  263. return MOU_NORMAL;
  264. }
  265. if (wh->mouse != NULL)
  266. {
  267. int mou;
  268. mou = wh->mouse (event, wh);
  269. if (mou != MOU_UNHANDLED)
  270. return mou;
  271. }
  272. first = h->current;
  273. p = first;
  274. do
  275. {
  276. Widget *w = WIDGET (p->data);
  277. p = dlg_widget_prev (h, p);
  278. if ((w->options & W_DISABLED) == 0 && w->mouse != NULL)
  279. {
  280. /* put global cursor position to the widget */
  281. int ret;
  282. ret = w->mouse (event, w);
  283. if (ret != MOU_UNHANDLED)
  284. return ret;
  285. }
  286. }
  287. while (p != first);
  288. return MOU_UNHANDLED;
  289. }
  290. /* --------------------------------------------------------------------------------------------- */
  291. static cb_ret_t
  292. dlg_try_hotkey (WDialog * h, int d_key)
  293. {
  294. GList *hot_cur;
  295. Widget *current;
  296. cb_ret_t handled;
  297. int c;
  298. if (h->widgets == NULL)
  299. return MSG_NOT_HANDLED;
  300. if (h->current == NULL)
  301. h->current = h->widgets;
  302. /*
  303. * Explanation: we don't send letter hotkeys to other widgets if
  304. * the currently selected widget is an input line
  305. */
  306. current = WIDGET (h->current->data);
  307. if ((current->options & W_DISABLED) != 0)
  308. return MSG_NOT_HANDLED;
  309. if (current->options & W_IS_INPUT)
  310. {
  311. /* skip ascii control characters, anything else can valid character in
  312. * some encoding */
  313. if (d_key >= 32 && d_key < 256)
  314. return MSG_NOT_HANDLED;
  315. }
  316. /* If it's an alt key, send the message */
  317. c = d_key & ~ALT (0);
  318. if (d_key & ALT (0) && g_ascii_isalpha (c))
  319. d_key = g_ascii_tolower (c);
  320. handled = MSG_NOT_HANDLED;
  321. if ((current->options & W_WANT_HOTKEY) != 0)
  322. handled = send_message (current, NULL, MSG_HOTKEY, d_key, NULL);
  323. /* If not used, send hotkey to other widgets */
  324. if (handled == MSG_HANDLED)
  325. return MSG_HANDLED;
  326. hot_cur = dlg_widget_next (h, h->current);
  327. /* send it to all widgets */
  328. while (h->current != hot_cur && handled == MSG_NOT_HANDLED)
  329. {
  330. current = WIDGET (hot_cur->data);
  331. if ((current->options & W_WANT_HOTKEY) != 0 && (current->options & W_DISABLED) == 0)
  332. handled = send_message (current, NULL, MSG_HOTKEY, d_key, NULL);
  333. if (handled == MSG_NOT_HANDLED)
  334. hot_cur = dlg_widget_next (h, hot_cur);
  335. }
  336. if (handled == MSG_HANDLED)
  337. do_select_widget (h, hot_cur, SELECT_EXACT);
  338. return handled;
  339. }
  340. /* --------------------------------------------------------------------------------------------- */
  341. static void
  342. dlg_key_event (WDialog * h, int d_key)
  343. {
  344. cb_ret_t handled;
  345. if (h->widgets == NULL)
  346. return;
  347. if (h->current == NULL)
  348. h->current = h->widgets;
  349. /* TAB used to cycle */
  350. if ((h->flags & DLG_WANT_TAB) == 0)
  351. {
  352. if (d_key == '\t')
  353. {
  354. dlg_one_down (h);
  355. return;
  356. }
  357. else if ((d_key & ~(KEY_M_SHIFT | KEY_M_CTRL)) == '\t')
  358. {
  359. dlg_one_up (h);
  360. return;
  361. }
  362. }
  363. /* first can dlg_callback handle the key */
  364. handled = send_message (h, NULL, MSG_KEY, d_key, NULL);
  365. /* next try the hotkey */
  366. if (handled == MSG_NOT_HANDLED)
  367. handled = dlg_try_hotkey (h, d_key);
  368. if (handled == MSG_HANDLED)
  369. send_message (h, NULL, MSG_HOTKEY_HANDLED, 0, NULL);
  370. else
  371. /* not used - then try widget_callback */
  372. handled = send_message (h->current->data, NULL, MSG_KEY, d_key, NULL);
  373. /* not used- try to use the unhandled case */
  374. if (handled == MSG_NOT_HANDLED)
  375. handled = send_message (h, NULL, MSG_UNHANDLED_KEY, d_key, NULL);
  376. if (handled == MSG_NOT_HANDLED)
  377. handled = dlg_handle_key (h, d_key);
  378. (void) handled;
  379. send_message (h, NULL, MSG_POST_KEY, d_key, NULL);
  380. }
  381. /* --------------------------------------------------------------------------------------------- */
  382. static void
  383. frontend_dlg_run (WDialog * h)
  384. {
  385. Gpm_Event event;
  386. event.x = -1;
  387. /* close opened editors, viewers, etc */
  388. if (!h->modal && mc_global.midnight_shutdown)
  389. {
  390. send_message (h, NULL, MSG_VALIDATE, 0, NULL);
  391. return;
  392. }
  393. while (h->state == DLG_ACTIVE)
  394. {
  395. int d_key;
  396. if (mc_global.tty.winch_flag != 0)
  397. dialog_change_screen_size ();
  398. if (is_idle ())
  399. {
  400. if (idle_hook)
  401. execute_hooks (idle_hook);
  402. while ((WIDGET (h)->options & W_WANT_IDLE) != 0 && is_idle ())
  403. send_message (h, NULL, MSG_IDLE, 0, NULL);
  404. /* Allow terminating the dialog from the idle handler */
  405. if (h->state != DLG_ACTIVE)
  406. break;
  407. }
  408. update_cursor (h);
  409. /* Clear interrupt flag */
  410. tty_got_interrupt ();
  411. d_key = tty_get_event (&event, h->mouse_status == MOU_REPEAT, TRUE);
  412. dlg_process_event (h, d_key, &event);
  413. if (h->state == DLG_CLOSED)
  414. send_message (h, NULL, MSG_VALIDATE, 0, NULL);
  415. }
  416. }
  417. /* --------------------------------------------------------------------------------------------- */
  418. static int
  419. dlg_find_widget_by_id (gconstpointer a, gconstpointer b)
  420. {
  421. Widget *w = WIDGET (a);
  422. unsigned long id = GPOINTER_TO_UINT (b);
  423. return w->id == id ? 0 : 1;
  424. }
  425. /* --------------------------------------------------------------------------------------------- */
  426. /*** public functions ****************************************************************************/
  427. /* --------------------------------------------------------------------------------------------- */
  428. /** Clean the dialog area, draw the frame and the title */
  429. void
  430. dlg_default_repaint (WDialog * h)
  431. {
  432. Widget *wh = WIDGET (h);
  433. int space;
  434. if (h->state != DLG_ACTIVE)
  435. return;
  436. space = (h->flags & DLG_COMPACT) ? 0 : 1;
  437. tty_setcolor (h->color[DLG_COLOR_NORMAL]);
  438. dlg_erase (h);
  439. tty_draw_box (wh->y + space, wh->x + space, wh->lines - 2 * space, wh->cols - 2 * space, FALSE);
  440. if (h->title != NULL)
  441. {
  442. tty_setcolor (h->color[DLG_COLOR_TITLE]);
  443. widget_move (h, space, (wh->cols - str_term_width1 (h->title)) / 2);
  444. tty_print_string (h->title);
  445. }
  446. }
  447. /* --------------------------------------------------------------------------------------------- */
  448. /** this function allows to set dialog position */
  449. void
  450. dlg_set_position (WDialog * h, int y1, int x1, int y2, int x2)
  451. {
  452. Widget *wh = WIDGET (h);
  453. /* save old positions, will be used to reposition childs */
  454. int ox, oy, oc, ol;
  455. int shift_x, shift_y, scale_x, scale_y;
  456. /* save old positions, will be used to reposition childs */
  457. ox = wh->x;
  458. oy = wh->y;
  459. oc = wh->cols;
  460. ol = wh->lines;
  461. wh->x = x1;
  462. wh->y = y1;
  463. wh->lines = y2 - y1;
  464. wh->cols = x2 - x1;
  465. /* dialog is empty */
  466. if (h->widgets == NULL)
  467. return;
  468. if (h->current == NULL)
  469. h->current = h->widgets;
  470. /* values by which controls should be moved */
  471. shift_x = wh->x - ox;
  472. shift_y = wh->y - oy;
  473. scale_x = wh->cols - oc;
  474. scale_y = wh->lines - ol;
  475. if ((shift_x != 0) || (shift_y != 0) || (scale_x != 0) || (scale_y != 0))
  476. {
  477. GList *w;
  478. for (w = h->widgets; w != NULL; w = g_list_next (w))
  479. {
  480. /* there are, mainly, 2 generally possible
  481. situations:
  482. 1. control sticks to one side - it
  483. should be moved
  484. 2. control sticks to two sides of
  485. one direction - it should be sized */
  486. Widget *c = WIDGET (w->data);
  487. int x = c->x;
  488. int y = c->y;
  489. int cols = c->cols;
  490. int lines = c->lines;
  491. if ((c->pos_flags & WPOS_CENTER_HORZ) != 0)
  492. x = wh->x + (wh->cols - c->cols) / 2;
  493. else if ((c->pos_flags & WPOS_KEEP_LEFT) != 0 && (c->pos_flags & WPOS_KEEP_RIGHT) != 0)
  494. {
  495. x += shift_x;
  496. cols += scale_x;
  497. }
  498. else if ((c->pos_flags & WPOS_KEEP_LEFT) != 0)
  499. x += shift_x;
  500. else if ((c->pos_flags & WPOS_KEEP_RIGHT) != 0)
  501. x += shift_x + scale_x;
  502. if ((c->pos_flags & WPOS_CENTER_VERT) != 0)
  503. y = wh->y + (wh->lines - c->lines) / 2;
  504. else if ((c->pos_flags & WPOS_KEEP_TOP) != 0 && (c->pos_flags & WPOS_KEEP_BOTTOM) != 0)
  505. {
  506. y += shift_y;
  507. lines += scale_y;
  508. }
  509. else if ((c->pos_flags & WPOS_KEEP_TOP) != 0)
  510. y += shift_y;
  511. else if ((c->pos_flags & WPOS_KEEP_BOTTOM) != 0)
  512. y += shift_y + scale_y;
  513. widget_set_size (c, y, x, lines, cols);
  514. }
  515. }
  516. }
  517. /* --------------------------------------------------------------------------------------------- */
  518. /** this function sets only size, leaving positioning to automatic methods */
  519. void
  520. dlg_set_size (WDialog * h, int lines, int cols)
  521. {
  522. int x = WIDGET (h)->x;
  523. int y = WIDGET (h)->y;
  524. if (h->flags & DLG_CENTER)
  525. {
  526. y = (LINES - lines) / 2;
  527. x = (COLS - cols) / 2;
  528. }
  529. if ((h->flags & DLG_TRYUP) && (y > 3))
  530. y -= 2;
  531. dlg_set_position (h, y, x, y + lines, x + cols);
  532. }
  533. /* --------------------------------------------------------------------------------------------- */
  534. /** Default dialog callback */
  535. cb_ret_t
  536. dlg_default_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
  537. {
  538. WDialog *h = DIALOG (w);
  539. (void) sender;
  540. (void) parm;
  541. (void) data;
  542. switch (msg)
  543. {
  544. case MSG_DRAW:
  545. if (h->color != NULL)
  546. {
  547. dlg_default_repaint (h);
  548. return MSG_HANDLED;
  549. }
  550. return MSG_NOT_HANDLED;
  551. case MSG_IDLE:
  552. dlg_broadcast_msg_to (h, MSG_IDLE, FALSE, W_WANT_IDLE);
  553. return MSG_HANDLED;
  554. case MSG_RESIZE:
  555. /* this is default resizing mechanism */
  556. /* the main idea of this code is to resize dialog
  557. according to flags (if any of flags require automatic
  558. resizing, like DLG_CENTER, end after that reposition
  559. controls in dialog according to flags of widget) */
  560. dlg_set_size (h, w->lines, w->cols);
  561. return MSG_HANDLED;
  562. default:
  563. break;
  564. }
  565. return MSG_NOT_HANDLED;
  566. }
  567. /* --------------------------------------------------------------------------------------------- */
  568. WDialog *
  569. dlg_create (gboolean modal, int y1, int x1, int lines, int cols,
  570. const int *colors, widget_cb_fn callback, mouse_h mouse_handler,
  571. const char *help_ctx, const char *title, dlg_flags_t flags)
  572. {
  573. WDialog *new_d;
  574. Widget *w;
  575. new_d = g_new0 (WDialog, 1);
  576. w = WIDGET (new_d);
  577. widget_init (w, y1, x1, lines, cols, (callback != NULL) ? callback : dlg_default_callback,
  578. mouse_handler);
  579. widget_want_cursor (w, FALSE);
  580. new_d->state = DLG_CONSTRUCT;
  581. new_d->modal = modal;
  582. if (colors != NULL)
  583. memmove (new_d->color, colors, sizeof (dlg_colors_t));
  584. new_d->help_ctx = help_ctx;
  585. new_d->flags = flags;
  586. new_d->data = NULL;
  587. dlg_set_size (new_d, lines, cols);
  588. new_d->fullscreen = (w->x == 0 && w->y == 0 && w->cols == COLS && w->lines == LINES);
  589. new_d->mouse_status = MOU_UNHANDLED;
  590. /* Strip existing spaces, add one space before and after the title */
  591. if (title != NULL && *title != '\0')
  592. {
  593. char *t;
  594. t = g_strstrip (g_strdup (title));
  595. if (*t != '\0')
  596. new_d->title = g_strdup_printf (" %s ", t);
  597. g_free (t);
  598. }
  599. /* unique name of event group for this dialog */
  600. new_d->event_group = g_strdup_printf ("%s_%p", MCEVENT_GROUP_DIALOG, (void *) new_d);
  601. return new_d;
  602. }
  603. /* --------------------------------------------------------------------------------------------- */
  604. void
  605. dlg_set_default_colors (void)
  606. {
  607. dialog_colors[DLG_COLOR_NORMAL] = COLOR_NORMAL;
  608. dialog_colors[DLG_COLOR_FOCUS] = COLOR_FOCUS;
  609. dialog_colors[DLG_COLOR_HOT_NORMAL] = COLOR_HOT_NORMAL;
  610. dialog_colors[DLG_COLOR_HOT_FOCUS] = COLOR_HOT_FOCUS;
  611. dialog_colors[DLG_COLOR_TITLE] = COLOR_TITLE;
  612. alarm_colors[DLG_COLOR_NORMAL] = ERROR_COLOR;
  613. alarm_colors[DLG_COLOR_FOCUS] = ERROR_FOCUS;
  614. alarm_colors[DLG_COLOR_HOT_NORMAL] = ERROR_HOT_NORMAL;
  615. alarm_colors[DLG_COLOR_HOT_FOCUS] = ERROR_HOT_FOCUS;
  616. alarm_colors[DLG_COLOR_TITLE] = ERROR_TITLE;
  617. }
  618. /* --------------------------------------------------------------------------------------------- */
  619. void
  620. dlg_erase (WDialog * h)
  621. {
  622. if ((h != NULL) && (h->state == DLG_ACTIVE))
  623. {
  624. Widget *wh = WIDGET (h);
  625. tty_fill_region (wh->y, wh->x, wh->lines, wh->cols, ' ');
  626. }
  627. }
  628. /* --------------------------------------------------------------------------------------------- */
  629. /**
  630. * Insert widget to dialog before requested widget. Make the widget current. Return widget ID.
  631. */
  632. unsigned long
  633. add_widget_autopos (WDialog * h, void *w, widget_pos_flags_t pos_flags, const void *before)
  634. {
  635. Widget *wh = WIDGET (h);
  636. Widget *widget;
  637. /* Don't accept 0 widgets */
  638. if (w == NULL)
  639. abort ();
  640. widget = WIDGET (w);
  641. if ((pos_flags & WPOS_CENTER_HORZ) != 0)
  642. widget->x = (wh->cols - widget->cols) / 2;
  643. widget->x += wh->x;
  644. if ((pos_flags & WPOS_CENTER_VERT) != 0)
  645. widget->y = (wh->lines - widget->lines) / 2;
  646. widget->y += wh->y;
  647. widget->owner = h;
  648. widget->pos_flags = pos_flags;
  649. widget->id = h->widget_id++;
  650. if (h->widgets == NULL || before == NULL)
  651. {
  652. h->widgets = g_list_append (h->widgets, widget);
  653. h->current = g_list_last (h->widgets);
  654. }
  655. else
  656. {
  657. GList *b;
  658. b = g_list_find (h->widgets, before);
  659. /* don't accept widget not from dialog. This shouldn't happen */
  660. if (b == NULL)
  661. abort ();
  662. b = g_list_next (b);
  663. h->widgets = g_list_insert_before (h->widgets, b, widget);
  664. if (b != NULL)
  665. h->current = g_list_previous (b);
  666. else
  667. h->current = g_list_last (h->widgets);
  668. }
  669. /* widget has been added in runtime */
  670. if (h->state == DLG_ACTIVE)
  671. {
  672. send_message (widget, NULL, MSG_INIT, 0, NULL);
  673. send_message (widget, NULL, MSG_DRAW, 0, NULL);
  674. send_message (widget, NULL, MSG_FOCUS, 0, NULL);
  675. }
  676. return widget->id;
  677. }
  678. /* --------------------------------------------------------------------------------------------- */
  679. /** wrapper to simply add lefttop positioned controls */
  680. unsigned long
  681. add_widget (WDialog * h, void *w)
  682. {
  683. return add_widget_autopos (h, w, WPOS_KEEP_DEFAULT,
  684. h->current != NULL ? h->current->data : NULL);
  685. }
  686. /* --------------------------------------------------------------------------------------------- */
  687. unsigned long
  688. add_widget_before (WDialog * h, void *w, void *before)
  689. {
  690. return add_widget_autopos (h, w, WPOS_KEEP_DEFAULT, before);
  691. }
  692. /* --------------------------------------------------------------------------------------------- */
  693. /** delete widget from dialog */
  694. void
  695. del_widget (void *w)
  696. {
  697. WDialog *h;
  698. GList *d;
  699. /* Don't accept NULL widget. This shouldn't happen */
  700. if (w == NULL)
  701. abort ();
  702. h = WIDGET (w)->owner;
  703. d = g_list_find (h->widgets, w);
  704. if (d == h->current)
  705. h->current = dlg_widget_next (h, d);
  706. h->widgets = g_list_remove_link (h->widgets, d);
  707. send_message (d->data, NULL, MSG_DESTROY, 0, NULL);
  708. g_free (d->data);
  709. g_list_free_1 (d);
  710. /* widget has been deleted in runtime */
  711. if (h->state == DLG_ACTIVE)
  712. {
  713. dlg_redraw (h);
  714. dlg_focus (h);
  715. }
  716. }
  717. /* --------------------------------------------------------------------------------------------- */
  718. void
  719. do_refresh (void)
  720. {
  721. GList *d = top_dlg;
  722. if (fast_refresh)
  723. {
  724. if ((d != NULL) && (d->data != NULL))
  725. dlg_redraw (DIALOG (d->data));
  726. }
  727. else
  728. {
  729. /* Search first fullscreen dialog */
  730. for (; d != NULL; d = g_list_next (d))
  731. if (d->data != NULL && DIALOG (d->data)->fullscreen)
  732. break;
  733. /* back to top dialog */
  734. for (; d != NULL; d = g_list_previous (d))
  735. if (d->data != NULL)
  736. dlg_redraw (DIALOG (d->data));
  737. }
  738. }
  739. /* --------------------------------------------------------------------------------------------- */
  740. /** broadcast a message to all the widgets in a dialog */
  741. void
  742. dlg_broadcast_msg (WDialog * h, widget_msg_t msg)
  743. {
  744. dlg_broadcast_msg_to (h, msg, FALSE, 0);
  745. }
  746. /* --------------------------------------------------------------------------------------------- */
  747. gboolean
  748. dlg_focus (WDialog * h)
  749. {
  750. /* cannot focus disabled widget */
  751. if ((h->current != NULL) && (h->state == DLG_CONSTRUCT || h->state == DLG_ACTIVE))
  752. {
  753. Widget *current = WIDGET (h->current->data);
  754. if (((current->options & W_DISABLED) == 0)
  755. && (send_message (current, NULL, MSG_FOCUS, 0, NULL) == MSG_HANDLED))
  756. {
  757. send_message (h, current, MSG_FOCUS, 0, NULL);
  758. return TRUE;
  759. }
  760. }
  761. return FALSE;
  762. }
  763. /* --------------------------------------------------------------------------------------------- */
  764. /** Find the widget with the given callback in the dialog h */
  765. Widget *
  766. find_widget_type (const WDialog * h, widget_cb_fn callback)
  767. {
  768. GList *w;
  769. w = g_list_find_custom (h->widgets, callback, dlg_find_widget_callback);
  770. return (w == NULL) ? NULL : WIDGET (w->data);
  771. }
  772. /* --------------------------------------------------------------------------------------------- */
  773. /** Find the widget with the given id */
  774. Widget *
  775. dlg_find_by_id (const WDialog * h, unsigned long id)
  776. {
  777. GList *w;
  778. w = g_list_find_custom (h->widgets, GUINT_TO_POINTER (id), dlg_find_widget_by_id);
  779. return w != NULL ? WIDGET (w->data) : NULL;
  780. }
  781. /* --------------------------------------------------------------------------------------------- */
  782. /** Find the widget with the given id in the dialog h and select it */
  783. void
  784. dlg_select_by_id (const WDialog * h, unsigned long id)
  785. {
  786. Widget *w;
  787. w = dlg_find_by_id (h, id);
  788. if (w != NULL)
  789. dlg_select_widget (w);
  790. }
  791. /* --------------------------------------------------------------------------------------------- */
  792. /**
  793. * Try to select widget in the dialog.
  794. */
  795. void
  796. dlg_select_widget (void *w)
  797. {
  798. Widget *widget = WIDGET (w);
  799. WDialog *h = widget->owner;
  800. do_select_widget (h, g_list_find (h->widgets, widget), SELECT_EXACT);
  801. }
  802. /* --------------------------------------------------------------------------------------------- */
  803. /**
  804. * Set widget at top of widget list and make it current.
  805. */
  806. void
  807. dlg_set_top_widget (void *w)
  808. {
  809. Widget *widget = WIDGET (w);
  810. WDialog *h = widget->owner;
  811. GList *l;
  812. l = g_list_find (h->widgets, w);
  813. if (l == NULL)
  814. abort (); /* widget is not in dialog, this should not happen */
  815. /* unfocus prevoius widget and focus current one before widget reordering */
  816. if (h->state == DLG_ACTIVE)
  817. do_select_widget (h, l, SELECT_EXACT);
  818. /* widget reordering */
  819. h->widgets = g_list_remove_link (h->widgets, l);
  820. h->widgets = g_list_concat (h->widgets, l);
  821. h->current = l;
  822. }
  823. /* --------------------------------------------------------------------------------------------- */
  824. /** Try to select previous widget in the tab order */
  825. void
  826. dlg_one_up (WDialog * h)
  827. {
  828. if (h->widgets != NULL)
  829. do_select_widget (h, dlg_widget_prev (h, h->current), SELECT_PREV);
  830. }
  831. /* --------------------------------------------------------------------------------------------- */
  832. /** Try to select next widget in the tab order */
  833. void
  834. dlg_one_down (WDialog * h)
  835. {
  836. if (h->widgets != NULL)
  837. do_select_widget (h, dlg_widget_next (h, h->current), SELECT_NEXT);
  838. }
  839. /* --------------------------------------------------------------------------------------------- */
  840. void
  841. update_cursor (WDialog * h)
  842. {
  843. GList *p = h->current;
  844. if ((p != NULL) && (h->state == DLG_ACTIVE))
  845. {
  846. Widget *w;
  847. w = WIDGET (p->data);
  848. if (((w->options & W_DISABLED) == 0) && ((w->options & W_WANT_CURSOR) != 0))
  849. send_message (w, NULL, MSG_CURSOR, 0, NULL);
  850. else
  851. do
  852. {
  853. p = dlg_widget_next (h, p);
  854. if (p == h->current)
  855. break;
  856. w = WIDGET (p->data);
  857. if (((w->options & W_DISABLED) == 0) && ((w->options & W_WANT_CURSOR) != 0))
  858. if (send_message (w, NULL, MSG_CURSOR, 0, NULL) == MSG_HANDLED)
  859. break;
  860. }
  861. while (TRUE);
  862. }
  863. }
  864. /* --------------------------------------------------------------------------------------------- */
  865. /**
  866. * Redraw the widgets in reverse order, leaving the current widget
  867. * as the last one
  868. */
  869. void
  870. dlg_redraw (WDialog * h)
  871. {
  872. if (h->state != DLG_ACTIVE)
  873. return;
  874. if (h->winch_pending)
  875. {
  876. h->winch_pending = FALSE;
  877. send_message (h, NULL, MSG_RESIZE, 0, NULL);
  878. }
  879. send_message (h, NULL, MSG_DRAW, 0, NULL);
  880. dlg_broadcast_msg (h, MSG_DRAW);
  881. update_cursor (h);
  882. }
  883. /* --------------------------------------------------------------------------------------------- */
  884. void
  885. dlg_stop (WDialog * h)
  886. {
  887. h->state = DLG_CLOSED;
  888. }
  889. /* --------------------------------------------------------------------------------------------- */
  890. /** Init the process */
  891. void
  892. dlg_init (WDialog * h)
  893. {
  894. if (top_dlg != NULL && DIALOG (top_dlg->data)->modal)
  895. h->modal = TRUE;
  896. /* add dialog to the stack */
  897. top_dlg = g_list_prepend (top_dlg, h);
  898. /* Initialize dialog manager and widgets */
  899. if (h->state == DLG_CONSTRUCT)
  900. {
  901. if (!h->modal)
  902. dialog_switch_add (h);
  903. send_message (h, NULL, MSG_INIT, 0, NULL);
  904. dlg_broadcast_msg (h, MSG_INIT);
  905. dlg_read_history (h);
  906. }
  907. h->state = DLG_ACTIVE;
  908. /* first send MSG_DRAW to dialog itself and all widgets... */
  909. dlg_redraw (h);
  910. /* ...then send MSG_FOCUS to select the first widget that can take focus */
  911. while (h->current != NULL && !dlg_focus (h))
  912. h->current = dlg_widget_next (h, h->current);
  913. h->ret_value = 0;
  914. }
  915. /* --------------------------------------------------------------------------------------------- */
  916. void
  917. dlg_process_event (WDialog * h, int key, Gpm_Event * event)
  918. {
  919. if (key == EV_NONE)
  920. {
  921. if (tty_got_interrupt ())
  922. if (send_message (h, NULL, MSG_ACTION, CK_Cancel, NULL) != MSG_HANDLED)
  923. dlg_execute_cmd (h, CK_Cancel);
  924. return;
  925. }
  926. if (key == EV_MOUSE)
  927. h->mouse_status = dlg_mouse_event (h, event);
  928. else
  929. dlg_key_event (h, key);
  930. }
  931. /* --------------------------------------------------------------------------------------------- */
  932. /** Shutdown the dlg_run */
  933. void
  934. dlg_run_done (WDialog * h)
  935. {
  936. top_dlg = g_list_remove (top_dlg, h);
  937. if (h->state == DLG_CLOSED)
  938. {
  939. send_message (h, h->current->data, MSG_END, 0, NULL);
  940. if (!h->modal)
  941. dialog_switch_remove (h);
  942. }
  943. }
  944. /* --------------------------------------------------------------------------------------------- */
  945. /**
  946. * Standard run dialog routine
  947. * We have to keep this routine small so that we can duplicate it's
  948. * behavior on complex routines like the file routines, this way,
  949. * they can call the dlg_process_event without rewriting all the code
  950. */
  951. int
  952. dlg_run (WDialog * h)
  953. {
  954. dlg_init (h);
  955. frontend_dlg_run (h);
  956. dlg_run_done (h);
  957. return h->ret_value;
  958. }
  959. /* --------------------------------------------------------------------------------------------- */
  960. void
  961. dlg_destroy (WDialog * h)
  962. {
  963. /* if some widgets have history, save all history at one moment here */
  964. dlg_save_history (h);
  965. dlg_broadcast_msg (h, MSG_DESTROY);
  966. g_list_free_full (h->widgets, g_free);
  967. mc_event_group_del (h->event_group);
  968. g_free (h->event_group);
  969. g_free (h->title);
  970. g_free (h);
  971. do_refresh ();
  972. }
  973. /* --------------------------------------------------------------------------------------------- */
  974. /**
  975. * Write history to the ${XDG_CACHE_HOME}/mc/history file
  976. */
  977. void
  978. dlg_save_history (WDialog * h)
  979. {
  980. char *profile;
  981. int i;
  982. if (num_history_items_recorded == 0) /* this is how to disable */
  983. return;
  984. profile = mc_config_get_full_path (MC_HISTORY_FILE);
  985. i = open (profile, O_CREAT | O_EXCL, S_IRUSR | S_IWUSR);
  986. if (i != -1)
  987. close (i);
  988. /* Make sure the history is only readable by the user */
  989. if (chmod (profile, S_IRUSR | S_IWUSR) != -1 || errno == ENOENT)
  990. {
  991. ev_history_load_save_t event_data;
  992. event_data.cfg = mc_config_init (profile, FALSE);
  993. event_data.receiver = NULL;
  994. /* get all histories in dialog */
  995. mc_event_raise (h->event_group, MCEVENT_HISTORY_SAVE, &event_data);
  996. mc_config_save_file (event_data.cfg, NULL);
  997. mc_config_deinit (event_data.cfg);
  998. }
  999. g_free (profile);
  1000. }
  1001. /* --------------------------------------------------------------------------------------------- */
  1002. char *
  1003. dlg_get_title (const WDialog * h, size_t len)
  1004. {
  1005. char *t;
  1006. if (h == NULL)
  1007. abort ();
  1008. if (h->get_title != NULL)
  1009. t = h->get_title (h, len);
  1010. else
  1011. t = g_strdup ("");
  1012. return t;
  1013. }
  1014. /* --------------------------------------------------------------------------------------------- */
  1015. /**
  1016. * broadcast a message to all the widgets in a dialog that have
  1017. * the options set to flags. If flags is zero, the message is sent
  1018. * to all widgets.
  1019. */
  1020. cb_ret_t
  1021. dlg_broadcast_msg_to (WDialog * h, widget_msg_t msg, gboolean reverse, int flags)
  1022. {
  1023. GList *p, *first;
  1024. cb_ret_t ret = MSG_NOT_HANDLED;
  1025. if (h->widgets == NULL)
  1026. return ret;
  1027. if (h->current == NULL)
  1028. h->current = h->widgets;
  1029. if (reverse)
  1030. p = dlg_widget_prev (h, h->current);
  1031. else
  1032. p = dlg_widget_next (h, h->current);
  1033. first = p;
  1034. do
  1035. {
  1036. Widget *w = WIDGET (p->data);
  1037. if (reverse)
  1038. p = dlg_widget_prev (h, p);
  1039. else
  1040. p = dlg_widget_next (h, p);
  1041. if ((flags == 0) || ((flags & w->options) != 0))
  1042. {
  1043. cb_ret_t widget_ret;
  1044. widget_ret = send_message (w, NULL, msg, 0, NULL);
  1045. if (ret == MSG_NOT_HANDLED && widget_ret != MSG_NOT_HANDLED)
  1046. ret = widget_ret;
  1047. }
  1048. }
  1049. while (first != p);
  1050. return ret;
  1051. }
  1052. /* --------------------------------------------------------------------------------------------- */