/* * gui-history.c - memorize commands or text for buffers (used by all GUI) * * Copyright (C) 2003-2023 Sébastien Helleu * Copyright (C) 2005 Emmanuel Bouthenot * * This file is part of WeeChat, the extensible chat client. * * WeeChat 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 3 of the License, or * (at your option) any later version. * * WeeChat 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 WeeChat. If not, see . */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include "../core/weechat.h" #include "../core/wee-config.h" #include "../core/wee-hashtable.h" #include "../core/wee-hdata.h" #include "../core/wee-hook.h" #include "../core/wee-infolist.h" #include "../core/wee-string.h" #include "../plugins/plugin.h" #include "gui-history.h" #include "gui-buffer.h" struct t_gui_history *gui_history = NULL; struct t_gui_history *last_gui_history = NULL; struct t_gui_history *gui_history_ptr = NULL; int num_gui_history = 0; /* * Adds a text/command to buffer's history. */ void gui_history_buffer_add (struct t_gui_buffer *buffer, const char *string) { struct t_gui_history *new_history, *ptr_history; if (!string) return; if (!buffer->history || (buffer->history && (strcmp (buffer->history->text, string) != 0))) { new_history = malloc (sizeof (*new_history)); if (new_history) { new_history->text = strdup (string); if (buffer->history) buffer->history->prev_history = new_history; else buffer->last_history = new_history; new_history->next_history = buffer->history; new_history->prev_history = NULL; buffer->history = new_history; buffer->num_history++; /* remove one command if necessary */ if ((CONFIG_INTEGER(config_history_max_commands) > 0) && (buffer->num_history > CONFIG_INTEGER(config_history_max_commands))) { ptr_history = buffer->last_history->prev_history; if (buffer->ptr_history == buffer->last_history) buffer->ptr_history = ptr_history; ((buffer->last_history)->prev_history)->next_history = NULL; if (buffer->last_history->text) free (buffer->last_history->text); free (buffer->last_history); buffer->last_history = ptr_history; buffer->num_history++; } } } } /* * Adds a text/command to global history. */ void gui_history_global_add (const char *string) { struct t_gui_history *new_history, *ptr_history; if (!string) return; if (!gui_history || (gui_history && (strcmp (gui_history->text, string) != 0))) { new_history = malloc (sizeof (*new_history)); if (new_history) { new_history->text = strdup (string); if (gui_history) gui_history->prev_history = new_history; else last_gui_history = new_history; new_history->next_history = gui_history; new_history->prev_history = NULL; gui_history = new_history; num_gui_history++; /* remove one command if necessary */ if ((CONFIG_INTEGER(config_history_max_commands) > 0) && (num_gui_history > CONFIG_INTEGER(config_history_max_commands))) { ptr_history = last_gui_history->prev_history; if (gui_history_ptr == last_gui_history) gui_history_ptr = ptr_history; (last_gui_history->prev_history)->next_history = NULL; if (last_gui_history->text) free (last_gui_history->text); free (last_gui_history); last_gui_history = ptr_history; num_gui_history--; } } } } /* * Adds a text/command to buffer's history + global history. */ void gui_history_add (struct t_gui_buffer *buffer, const char *string) { char *string2, str_buffer[128]; snprintf (str_buffer, sizeof (str_buffer), "0x%lx", (unsigned long)(buffer)); string2 = hook_modifier_exec (NULL, "history_add", str_buffer, string); /* * if message was NOT dropped by modifier, then we add it to buffer and * global history */ if (!string2 || string2[0]) { gui_history_buffer_add (buffer, (string2) ? string2 : string); gui_history_global_add ((string2) ? string2 : string); } if (string2) free (string2); } /* * Frees global history. */ void gui_history_global_free () { struct t_gui_history *ptr_history; while (gui_history) { ptr_history = gui_history->next_history; if (gui_history->text) free (gui_history->text); free (gui_history); gui_history = ptr_history; } gui_history = NULL; last_gui_history = NULL; gui_history_ptr = NULL; num_gui_history = 0; } /* * Frees history for a buffer. */ void gui_history_buffer_free (struct t_gui_buffer *buffer) { struct t_gui_history *ptr_history; if (!buffer) return; while (buffer->history) { ptr_history = buffer->history->next_history; if (buffer->history->text) free (buffer->history->text); free (buffer->history); buffer->history = ptr_history; } buffer->history = NULL; buffer->last_history = NULL; buffer->ptr_history = NULL; buffer->num_history = 0; } /* * Callback for updating history. */ int gui_history_hdata_history_update_cb (void *data, struct t_hdata *hdata, void *pointer, struct t_hashtable *hashtable) { struct t_gui_history *ptr_history; struct t_gui_buffer *ptr_buffer; const char *text, *buffer; unsigned long value; int rc; /* make C compiler happy */ (void) data; (void) hdata; rc = 0; text = hashtable_get (hashtable, "text"); if (!text) return rc; if (pointer) { /* update history */ ptr_history = (struct t_gui_history *)pointer; if (ptr_history->text) free (ptr_history->text); ptr_history->text = strdup (text); } else { /* create new entry in history */ ptr_buffer = NULL; if (hashtable_has_key (hashtable, "buffer")) { buffer = hashtable_get (hashtable, "buffer"); if (buffer) { rc = sscanf (buffer, "%lx", &value); if ((rc != EOF) && (rc != 0)) ptr_buffer = (struct t_gui_buffer *)value; } } if (ptr_buffer) gui_history_add (ptr_buffer, text); else gui_history_global_add (text); } return rc; } /* * Returns hdata for history. */ struct t_hdata * gui_history_hdata_history_cb (const void *pointer, void *data, const char *hdata_name) { struct t_hdata *hdata; /* make C compiler happy */ (void) pointer; (void) data; hdata = hdata_new (NULL, hdata_name, "prev_history", "next_history", 1, 1, &gui_history_hdata_history_update_cb, NULL); if (hdata) { HDATA_VAR(struct t_gui_history, text, STRING, 0, NULL, NULL); HDATA_VAR(struct t_gui_history, prev_history, POINTER, 0, NULL, hdata_name); HDATA_VAR(struct t_gui_history, next_history, POINTER, 0, NULL, hdata_name); HDATA_LIST(gui_history, WEECHAT_HDATA_LIST_CHECK_POINTERS); HDATA_LIST(last_gui_history, 0); } return hdata; } /* * Adds history in an infolist. * * Returns: * 1: OK * 0: error */ int gui_history_add_to_infolist (struct t_infolist *infolist, struct t_gui_history *history) { struct t_infolist_item *ptr_item; if (!infolist || !history) return 0; ptr_item = infolist_new_item (infolist); if (!ptr_item) return 0; if (!infolist_new_var_string (ptr_item, "text", history->text)) return 0; return 1; }