history.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. /*
  2. Widgets for the Midnight Commander
  3. Copyright (C) 1994, 1995, 1996, 1998, 1999, 2000, 2001, 2002, 2003,
  4. 2004, 2005, 2006, 2007, 2009, 2010, 2011, 2012, 2013
  5. The Free Software Foundation, Inc.
  6. Authors:
  7. Radek Doulik, 1994, 1995
  8. Miguel de Icaza, 1994, 1995
  9. Jakub Jelinek, 1995
  10. Andrej Borsenkow, 1996
  11. Norbert Warmuth, 1997
  12. Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2011, 2012, 2013
  13. This file is part of the Midnight Commander.
  14. The Midnight Commander is free software: you can redistribute it
  15. and/or modify it under the terms of the GNU General Public License as
  16. published by the Free Software Foundation, either version 3 of the License,
  17. or (at your option) any later version.
  18. The Midnight Commander is distributed in the hope that it will be useful,
  19. but WITHOUT ANY WARRANTY; without even the implied warranty of
  20. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  21. GNU General Public License for more details.
  22. You should have received a copy of the GNU General Public License
  23. along with this program. If not, see <http://www.gnu.org/licenses/>.
  24. */
  25. /** \file history.c
  26. * \brief Source: save, load and show history
  27. */
  28. #include <config.h>
  29. #include <errno.h>
  30. #include <stdlib.h>
  31. #include <stdio.h>
  32. #include <string.h>
  33. #include <sys/types.h>
  34. #include <sys/stat.h>
  35. #include <fcntl.h>
  36. #include "lib/global.h"
  37. #include "lib/tty/tty.h" /* LINES, COLS */
  38. #include "lib/mcconfig.h" /* for history loading and saving */
  39. #include "lib/fileloc.h"
  40. #include "lib/strutil.h"
  41. #include "lib/util.h" /* list_append_unique */
  42. #include "lib/widget.h"
  43. /*** global variables ****************************************************************************/
  44. /* how much history items are used */
  45. int num_history_items_recorded = 60;
  46. /*** file scope macro definitions ****************************************************************/
  47. /*** file scope type declarations ****************************************************************/
  48. typedef struct
  49. {
  50. Widget *widget;
  51. size_t count;
  52. size_t maxlen;
  53. } history_dlg_data;
  54. /*** file scope variables ************************************************************************/
  55. /*** file scope functions ************************************************************************/
  56. static cb_ret_t
  57. history_dlg_reposition (WDialog * dlg_head)
  58. {
  59. history_dlg_data *data;
  60. int x = 0, y, he, wi;
  61. /* guard checks */
  62. if ((dlg_head == NULL) || (dlg_head->data == NULL))
  63. return MSG_NOT_HANDLED;
  64. data = (history_dlg_data *) dlg_head->data;
  65. y = data->widget->y;
  66. he = data->count + 2;
  67. if (he <= y || y > (LINES - 6))
  68. {
  69. he = min (he, y - 1);
  70. y -= he;
  71. }
  72. else
  73. {
  74. y++;
  75. he = min (he, LINES - y);
  76. }
  77. if (data->widget->x > 2)
  78. x = data->widget->x - 2;
  79. wi = data->maxlen + 4;
  80. if ((wi + x) > COLS)
  81. {
  82. wi = min (wi, COLS);
  83. x = COLS - wi;
  84. }
  85. dlg_set_position (dlg_head, y, x, y + he, x + wi);
  86. return MSG_HANDLED;
  87. }
  88. /* --------------------------------------------------------------------------------------------- */
  89. static cb_ret_t
  90. history_dlg_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
  91. {
  92. switch (msg)
  93. {
  94. case MSG_RESIZE:
  95. return history_dlg_reposition (DIALOG (w));
  96. case MSG_ACTION:
  97. return dlg_broadcast_msg_to (DIALOG (w), msg, FALSE, 0);
  98. default:
  99. return dlg_default_callback (w, sender, msg, parm, data);
  100. }
  101. }
  102. /* --------------------------------------------------------------------------------------------- */
  103. /*** public functions ****************************************************************************/
  104. /* --------------------------------------------------------------------------------------------- */
  105. /**
  106. * Load the history from the ${XDG_CACHE_HOME}/mc/history file.
  107. * It is called with the widgets history name and returns the GList list.
  108. */
  109. GList *
  110. history_get (const char *input_name)
  111. {
  112. GList *hist = NULL;
  113. char *profile;
  114. mc_config_t *cfg;
  115. if (num_history_items_recorded == 0) /* this is how to disable */
  116. return NULL;
  117. if ((input_name == NULL) || (*input_name == '\0'))
  118. return NULL;
  119. profile = mc_config_get_full_path (MC_HISTORY_FILE);
  120. cfg = mc_config_init (profile, TRUE);
  121. hist = history_load (cfg, input_name);
  122. mc_config_deinit (cfg);
  123. g_free (profile);
  124. return hist;
  125. }
  126. /* --------------------------------------------------------------------------------------------- */
  127. /**
  128. * Load history form the mc_config
  129. */
  130. GList *
  131. history_load (struct mc_config_t * cfg, const char *name)
  132. {
  133. size_t i;
  134. GList *hist = NULL;
  135. char **keys;
  136. size_t keys_num = 0;
  137. GIConv conv = INVALID_CONV;
  138. GString *buffer;
  139. if (name == NULL || *name == '\0')
  140. return NULL;
  141. /* get number of keys */
  142. keys = mc_config_get_keys (cfg, name, &keys_num);
  143. g_strfreev (keys);
  144. /* create charset conversion handler to convert strings
  145. from utf-8 to system codepage */
  146. if (!mc_global.utf8_display)
  147. conv = str_crt_conv_from ("UTF-8");
  148. buffer = g_string_sized_new (64);
  149. for (i = 0; i < keys_num; i++)
  150. {
  151. char key[BUF_TINY];
  152. char *this_entry;
  153. g_snprintf (key, sizeof (key), "%lu", (unsigned long) i);
  154. this_entry = mc_config_get_string_raw (cfg, name, key, "");
  155. if (this_entry == NULL)
  156. continue;
  157. if (conv == INVALID_CONV)
  158. hist = list_append_unique (hist, this_entry);
  159. else
  160. {
  161. g_string_set_size (buffer, 0);
  162. if (str_convert (conv, this_entry, buffer) == ESTR_FAILURE)
  163. hist = list_append_unique (hist, this_entry);
  164. else
  165. {
  166. hist = list_append_unique (hist, g_strndup (buffer->str, buffer->len));
  167. g_free (this_entry);
  168. }
  169. }
  170. }
  171. g_string_free (buffer, TRUE);
  172. if (conv != INVALID_CONV)
  173. str_close_conv (conv);
  174. /* return pointer to the last entry in the list */
  175. return g_list_last (hist);
  176. }
  177. /* --------------------------------------------------------------------------------------------- */
  178. /**
  179. * Save history to the mc_config, but don't save config to file
  180. */
  181. void
  182. history_save (struct mc_config_t *cfg, const char *name, GList * h)
  183. {
  184. GIConv conv = INVALID_CONV;
  185. GString *buffer;
  186. int i;
  187. if (name == NULL || *name == '\0' || h == NULL)
  188. return;
  189. /* go to end of list */
  190. h = g_list_last (h);
  191. /* go back 60 places */
  192. for (i = 0; (i < num_history_items_recorded - 1) && (h->prev != NULL); i++)
  193. h = g_list_previous (h);
  194. if (name != NULL)
  195. mc_config_del_group (cfg, name);
  196. /* create charset conversion handler to convert strings
  197. from system codepage to UTF-8 */
  198. if (!mc_global.utf8_display)
  199. conv = str_crt_conv_to ("UTF-8");
  200. buffer = g_string_sized_new (64);
  201. /* dump history into profile */
  202. for (i = 0; h != NULL; h = g_list_next (h))
  203. {
  204. char key[BUF_TINY];
  205. char *text = (char *) h->data;
  206. /* We shouldn't have null entries, but let's be sure */
  207. if (text == NULL)
  208. continue;
  209. g_snprintf (key, sizeof (key), "%d", i++);
  210. if (conv == INVALID_CONV)
  211. mc_config_set_string_raw (cfg, name, key, text);
  212. else
  213. {
  214. g_string_set_size (buffer, 0);
  215. if (str_convert (conv, text, buffer) == ESTR_FAILURE)
  216. mc_config_set_string_raw (cfg, name, key, text);
  217. else
  218. mc_config_set_string_raw (cfg, name, key, buffer->str);
  219. }
  220. }
  221. g_string_free (buffer, TRUE);
  222. if (conv != INVALID_CONV)
  223. str_close_conv (conv);
  224. }
  225. /* --------------------------------------------------------------------------------------------- */
  226. char *
  227. history_show (GList ** history, Widget * widget, int current)
  228. {
  229. GList *z, *hlist = NULL, *hi;
  230. size_t maxlen, count = 0;
  231. char *r = NULL;
  232. WDialog *query_dlg;
  233. WListbox *query_list;
  234. history_dlg_data hist_data;
  235. if (*history == NULL)
  236. return NULL;
  237. maxlen = str_term_width1 (_("History")) + 2;
  238. for (z = *history; z != NULL; z = g_list_previous (z))
  239. {
  240. WLEntry *entry;
  241. size_t i;
  242. i = str_term_width1 ((char *) z->data);
  243. maxlen = max (maxlen, i);
  244. count++;
  245. entry = g_new0 (WLEntry, 1);
  246. /* history is being reverted here */
  247. entry->text = g_strdup ((char *) z->data);
  248. hlist = g_list_prepend (hlist, entry);
  249. }
  250. hist_data.widget = widget;
  251. hist_data.count = count;
  252. hist_data.maxlen = maxlen;
  253. query_dlg =
  254. dlg_create (TRUE, 0, 0, 4, 4, dialog_colors, history_dlg_callback, NULL,
  255. "[History-query]", _("History"), DLG_COMPACT);
  256. query_dlg->data = &hist_data;
  257. query_list = listbox_new (1, 1, 2, 2, TRUE, NULL);
  258. /* this call makes list stick to all sides of dialog, effectively make
  259. it be resized with dialog */
  260. add_widget_autopos (query_dlg, query_list, WPOS_KEEP_ALL, NULL);
  261. /* to avoid diplicating of (calculating sizes in two places)
  262. code, call dlg_hist_callback function here, to set dialog and
  263. controls positions.
  264. The main idea - create 4x4 dialog and add 2x2 list in
  265. center of it, and let dialog function resize it to needed
  266. size. */
  267. send_message (query_dlg, NULL, MSG_RESIZE, 0, NULL);
  268. if (WIDGET (query_dlg)->y < widget->y)
  269. {
  270. /* draw list entries from bottom upto top */
  271. listbox_set_list (query_list, hlist);
  272. if (current < 0 || (size_t) current >= count)
  273. listbox_select_last (query_list);
  274. else
  275. listbox_select_entry (query_list, count - 1 - (size_t) current);
  276. }
  277. else
  278. {
  279. /* draw list entries from top downto bottom */
  280. /* revert history direction */
  281. hlist = g_list_reverse (hlist);
  282. listbox_set_list (query_list, hlist);
  283. if (current > 0)
  284. listbox_select_entry (query_list, current);
  285. }
  286. if (!mc_global.tty.slow_terminal)
  287. {
  288. WScrollBar *scrollbar;
  289. scrollbar = scrollbar_new (WIDGET (query_list), SCROLLBAR_VERTICAL);
  290. scrollbar_set_current (scrollbar, &query_list->pos);
  291. scrollbar_set_total (scrollbar, &query_list->count);
  292. scrollbar_set_first_displayed (scrollbar, &query_list->top);
  293. add_widget (query_dlg, scrollbar);
  294. }
  295. if (dlg_run (query_dlg) != B_CANCEL)
  296. {
  297. char *q;
  298. listbox_get_current (query_list, &q, NULL);
  299. r = g_strdup (q);
  300. }
  301. /* get modified history from dialog */
  302. z = NULL;
  303. for (hi = query_list->list; hi != NULL; hi = g_list_next (hi))
  304. {
  305. WLEntry *entry = LENTRY (hi->data);
  306. /* history is being reverted here again */
  307. z = g_list_prepend (z, entry->text);
  308. entry->text = NULL;
  309. }
  310. /* restore history direction */
  311. if (WIDGET (query_dlg)->y < widget->y)
  312. z = g_list_reverse (z);
  313. dlg_destroy (query_dlg);
  314. g_list_free_full (*history, g_free);
  315. *history = g_list_last (z);
  316. return r;
  317. }
  318. /* --------------------------------------------------------------------------------------------- */