123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395 |
- /* Widgets for the Midnight Commander
- Copyright (C) 1994, 1995, 1996, 1998, 1999, 2000, 2001, 2002, 2003,
- 2004, 2005, 2006, 2007, 2009, 2010 Free Software Foundation, Inc.
- Authors: 1994, 1995 Radek Doulik
- 1994, 1995 Miguel de Icaza
- 1995 Jakub Jelinek
- 1996 Andrej Borsenkow
- 1997 Norbert Warmuth
- 2009, 2010 Andrew Borodin
- 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.
- */
- /** \file history.c
- * \brief Source: save, load and show history
- */
- #include <config.h>
- #include <errno.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <fcntl.h>
- #include "lib/global.h"
- #include "lib/tty/tty.h" /* LINES, COLS */
- #include "lib/mcconfig.h" /* for history loading and saving */
- #include "lib/fileloc.h"
- #include "lib/strutil.h"
- #include "lib/util.h" /* list_append_unique */
- #include "lib/widget.h"
- /* TODO: these includes should be removed! */
- #include "src/setup.h" /* num_history_items_recorded */
- /*** global variables ****************************************************************************/
- int num_history_items_recorded = 60;
- /*** file scope macro definitions ****************************************************************/
- /*** file scope type declarations ****************************************************************/
- typedef struct
- {
- Widget *widget;
- size_t count;
- size_t maxlen;
- } history_dlg_data;
- /*** file scope variables ************************************************************************/
- /*** file scope functions ************************************************************************/
- static cb_ret_t
- history_dlg_reposition (Dlg_head * dlg_head)
- {
- history_dlg_data *data;
- int x = 0, y, he, wi;
- /* guard checks */
- if ((dlg_head == NULL) || (dlg_head->data == NULL))
- return MSG_NOT_HANDLED;
- data = (history_dlg_data *) dlg_head->data;
- y = data->widget->y;
- he = data->count + 2;
- if (he <= y || y > (LINES - 6))
- {
- he = min (he, y - 1);
- y -= he;
- }
- else
- {
- y++;
- he = min (he, LINES - y);
- }
- if (data->widget->x > 2)
- x = data->widget->x - 2;
- wi = data->maxlen + 4;
- if ((wi + x) > COLS)
- {
- wi = min (wi, COLS);
- x = COLS - wi;
- }
- dlg_set_position (dlg_head, y, x, y + he, x + wi);
- return MSG_HANDLED;
- }
- /* --------------------------------------------------------------------------------------------- */
- static cb_ret_t
- history_dlg_callback (Dlg_head * h, Widget * sender, dlg_msg_t msg, int parm, void *data)
- {
- switch (msg)
- {
- case DLG_RESIZE:
- return history_dlg_reposition (h);
- default:
- return default_dlg_callback (h, sender, msg, parm, data);
- }
- }
- /* --------------------------------------------------------------------------------------------- */
- /*** public functions ****************************************************************************/
- /* --------------------------------------------------------------------------------------------- */
- /*
- This loads the history of an input line to the widget. It is called
- with the widgets history name on creation of the widget, and returns
- the GList list.
- */
- GList *
- history_get (const char *input_name)
- {
- size_t i;
- GList *hist = NULL;
- char *profile;
- mc_config_t *cfg;
- char **keys;
- size_t keys_num = 0;
- GIConv conv = INVALID_CONV;
- GString *buffer;
- if (num_history_items_recorded == 0) /* this is how to disable */
- return NULL;
- if ((input_name == NULL) || (*input_name == '\0'))
- return NULL;
- profile = g_build_filename (home_dir, MC_USERCONF_DIR, MC_HISTORY_FILE, NULL);
- cfg = mc_config_init (profile);
- /* get number of keys */
- keys = mc_config_get_keys (cfg, input_name, &keys_num);
- g_strfreev (keys);
- /* create charset conversion handler to convert strings
- from utf-8 to system codepage */
- if (!utf8_display)
- conv = str_crt_conv_from ("UTF-8");
- buffer = g_string_sized_new (64);
- for (i = 0; i < keys_num; i++)
- {
- char key[BUF_TINY];
- char *this_entry;
- g_snprintf (key, sizeof (key), "%lu", (unsigned long) i);
- this_entry = mc_config_get_string_raw (cfg, input_name, key, "");
- if (this_entry == NULL)
- continue;
- if (conv == INVALID_CONV)
- hist = list_append_unique (hist, this_entry);
- else
- {
- g_string_set_size (buffer, 0);
- if (str_convert (conv, this_entry, buffer) == ESTR_FAILURE)
- hist = list_append_unique (hist, this_entry);
- else
- {
- hist = list_append_unique (hist, g_strdup (buffer->str));
- g_free (this_entry);
- }
- }
- }
- g_string_free (buffer, TRUE);
- if (conv != INVALID_CONV)
- str_close_conv (conv);
- mc_config_deinit (cfg);
- g_free (profile);
- /* return pointer to the last entry in the list */
- return g_list_last (hist);
- }
- /* --------------------------------------------------------------------------------------------- */
- /*
- This saves the history of an input line from the widget. It is called
- with the widgets history name. It stores histories in the file
- ~/.mc/history in using the profile code.
- */
- void
- history_put (const char *input_name, GList * h)
- {
- int i;
- char *profile;
- mc_config_t *cfg;
- GIConv conv = INVALID_CONV;
- GString *buffer;
- if (num_history_items_recorded == 0) /* this is how to disable */
- return;
- if ((input_name == NULL) || (*input_name == '\0'))
- return;
- if (h == NULL)
- return;
- profile = g_build_filename (home_dir, MC_USERCONF_DIR, MC_HISTORY_FILE, NULL);
- i = open (profile, O_CREAT | O_EXCL, S_IRUSR | S_IWUSR);
- if (i != -1)
- close (i);
- /* Make sure the history is only readable by the user */
- if (chmod (profile, S_IRUSR | S_IWUSR) == -1 && errno != ENOENT)
- {
- g_free (profile);
- return;
- }
- /* go to end of list */
- h = g_list_last (h);
- /* go back 60 places */
- for (i = 0; (i < num_history_items_recorded - 1) && (h->prev != NULL); i++)
- h = g_list_previous (h);
- cfg = mc_config_init (profile);
- if (input_name != NULL)
- mc_config_del_group (cfg, input_name);
- /* create charset conversion handler to convert strings
- from system codepage to UTF-8 */
- if (!utf8_display)
- conv = str_crt_conv_to ("UTF-8");
- buffer = g_string_sized_new (64);
- /* dump history into profile */
- for (i = 0; h != NULL; h = g_list_next (h))
- {
- char key[BUF_TINY];
- char *text = (char *) h->data;
- /* We shouldn't have null entries, but let's be sure */
- if (text == NULL)
- continue;
- g_snprintf (key, sizeof (key), "%d", i++);
- if (conv == INVALID_CONV)
- mc_config_set_string_raw (cfg, input_name, key, text);
- else
- {
- g_string_set_size (buffer, 0);
- if (str_convert (conv, text, buffer) == ESTR_FAILURE)
- mc_config_set_string_raw (cfg, input_name, key, text);
- else
- mc_config_set_string_raw (cfg, input_name, key, buffer->str);
- }
- }
- g_string_free (buffer, TRUE);
- if (conv != INVALID_CONV)
- str_close_conv (conv);
- mc_config_save_file (cfg, NULL);
- mc_config_deinit (cfg);
- g_free (profile);
- }
- /* --------------------------------------------------------------------------------------------- */
- char *
- history_show (GList ** history, Widget * widget)
- {
- GList *z, *hlist = NULL, *hi;
- size_t maxlen, i, count = 0;
- char *r = NULL;
- Dlg_head *query_dlg;
- WListbox *query_list;
- history_dlg_data hist_data;
- if (*history == NULL)
- return NULL;
- maxlen = str_term_width1 (_("History")) + 2;
- for (z = *history; z != NULL; z = g_list_previous (z))
- {
- WLEntry *entry;
- i = str_term_width1 ((char *) z->data);
- maxlen = max (maxlen, i);
- count++;
- entry = g_new0 (WLEntry, 1);
- /* history is being reverted here */
- entry->text = g_strdup ((char *) z->data);
- hlist = g_list_prepend (hlist, entry);
- }
- hist_data.widget = widget;
- hist_data.count = count;
- hist_data.maxlen = maxlen;
- query_dlg =
- create_dlg (TRUE, 0, 0, 4, 4, dialog_colors, history_dlg_callback,
- "[History-query]", _("History"), DLG_COMPACT);
- query_dlg->data = &hist_data;
- query_list = listbox_new (1, 1, 2, 2, TRUE, NULL);
- /* this call makes list stick to all sides of dialog, effectively make
- it be resized with dialog */
- add_widget_autopos (query_dlg, query_list, WPOS_KEEP_ALL);
- /* to avoid diplicating of (calculating sizes in two places)
- code, call dlg_hist_callback function here, to set dialog and
- controls positions.
- The main idea - create 4x4 dialog and add 2x2 list in
- center of it, and let dialog function resize it to needed
- size. */
- history_dlg_callback (query_dlg, NULL, DLG_RESIZE, 0, NULL);
- if (query_dlg->y < widget->y)
- {
- /* draw list entries from bottom upto top */
- listbox_set_list (query_list, hlist);
- listbox_select_last (query_list);
- }
- else
- {
- /* draw list entries from top downto bottom */
- /* revert history direction */
- hlist = g_list_reverse (hlist);
- listbox_set_list (query_list, hlist);
- }
- if (run_dlg (query_dlg) != B_CANCEL)
- {
- char *q;
- listbox_get_current (query_list, &q, NULL);
- if (q != NULL)
- r = g_strdup (q);
- }
- /* get modified history from dialog */
- z = NULL;
- for (hi = query_list->list; hi != NULL; hi = g_list_next (hi))
- {
- WLEntry *entry;
- entry = (WLEntry *) hi->data;
- /* history is being reverted here again */
- z = g_list_prepend (z, entry->text);
- entry->text = NULL;
- }
- /* restore history direction */
- if (query_dlg->y < widget->y)
- z = g_list_reverse (z);
- destroy_dlg (query_dlg);
- g_list_foreach (*history, (GFunc) g_free, NULL);
- g_list_free (*history);
- *history = g_list_last (z);
- return r;
- }
- /* --------------------------------------------------------------------------------------------- */
|