diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-15 20:18:39 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-15 20:18:39 +0000 |
commit | fff5217f02d91268ce90c8c05665602c059faaef (patch) | |
tree | 2ba24d32dc96eafe7ed0a85269548e76796d849d /src/fe-text/gui-readline.c | |
parent | Initial commit. (diff) | |
download | irssi-fff5217f02d91268ce90c8c05665602c059faaef.tar.xz irssi-fff5217f02d91268ce90c8c05665602c059faaef.zip |
Adding upstream version 1.4.5.upstream/1.4.5upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r-- | src/fe-text/gui-readline.c | 1626 |
1 files changed, 1626 insertions, 0 deletions
diff --git a/src/fe-text/gui-readline.c b/src/fe-text/gui-readline.c new file mode 100644 index 0000000..a53e415 --- /dev/null +++ b/src/fe-text/gui-readline.c @@ -0,0 +1,1626 @@ +/* + gui-readline.c : irssi + + Copyright (C) 1999 Timo Sirainen + + 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. +*/ + +#include "module.h" +#include <irssi/src/fe-text/module-formats.h> +#include <irssi/src/core/signals.h> +#include <irssi/src/core/misc.h> +#include <irssi/src/core/settings.h> +#include <irssi/src/core/special-vars.h> +#include <irssi/src/core/levels.h> +#include <irssi/src/core/servers.h> + +#include <irssi/src/fe-common/core/completion.h> +#include <irssi/src/fe-common/core/command-history.h> +#include <irssi/src/fe-common/core/keyboard.h> +#include <irssi/src/fe-common/core/printtext.h> + +#include <irssi/src/fe-text/term.h> +#include <irssi/src/fe-text/gui-entry.h> +#include <irssi/src/fe-text/gui-windows.h> +#include <irssi/src/core/utf8.h> + +#include <string.h> +#include <signal.h> + +/* After LINE_SPLIT_LIMIT characters, the message will be split into multiple lines */ +#define LINE_SPLIT_LIMIT 400 + +typedef void (*ENTRY_REDIRECT_KEY_FUNC) (int key, void *data, SERVER_REC *server, WI_ITEM_REC *item); +typedef void (*ENTRY_REDIRECT_ENTRY_FUNC) (const char *line, void *data, SERVER_REC *server, WI_ITEM_REC *item); + +typedef struct { + SIGNAL_FUNC func; + int flags; + void *data; +} ENTRY_REDIRECT_REC; + +static KEYBOARD_REC *keyboard; +static ENTRY_REDIRECT_REC *redir; +static int escape_next_key; + +static int readtag; +static unichar prev_key; +static gint64 last_keypress; + +static int paste_detect_time, paste_verify_line_count; +static char *paste_entry; +static int paste_entry_pos; +static GArray *paste_buffer; +static GArray *paste_buffer_rest; + +static char *paste_old_prompt; +static int paste_prompt, paste_line_count; +static int paste_join_multiline; +static int paste_ignore_first_nl; +static int paste_timeout_id; +static int paste_use_bracketed_mode; +static int paste_bracketed_mode; +static int paste_was_bracketed_mode; +static int previous_yank_preceded; + +/* Terminal sequences that surround the input when the terminal has the + * bracketed paste mode active. Fror more details see + * https://cirw.in/blog/bracketed-paste */ +static const unichar bp_start[] = { 0x1b, '[', '2', '0', '0', '~' }; +static const unichar bp_end[] = { 0x1b, '[', '2', '0', '1', '~' }; + +#define BRACKETED_PASTE_TIMEOUT (5 * 1000) // ms + +#if GLIB_CHECK_VERSION(2, 62, 0) +/* nothing */ +#else +/* compatibility code for old GLib */ +GArray *g_array_copy(GArray *array) +{ + GArray *out; + guint elt_size; + + elt_size = g_array_get_element_size(array); + out = g_array_sized_new(FALSE, FALSE, elt_size, array->len); + memcpy(out->data, array->data, array->len * elt_size); + out->len = array->len; + + return out; +} +#endif + +static void sig_input(void); + +void input_listen_init(int handle) +{ + readtag = i_input_add_poll(handle, G_PRIORITY_HIGH, I_INPUT_READ, + (GInputFunction) sig_input, NULL); +} + +void input_listen_deinit(void) +{ + g_source_remove(readtag); + readtag = -1; +} + +static void handle_key_redirect(int key) +{ + ENTRY_REDIRECT_KEY_FUNC func; + void *data; + + func = (ENTRY_REDIRECT_KEY_FUNC) redir->func; + data = redir->data; + g_free_and_null(redir); + + gui_entry_set_prompt(active_entry, ""); + + if (func != NULL) + func(key, data, active_win->active_server, active_win->active); +} + +static void handle_entry_redirect(const char *line) +{ + ENTRY_REDIRECT_ENTRY_FUNC func; + void *data; + + gui_entry_set_hidden(active_entry, FALSE); + + func = (ENTRY_REDIRECT_ENTRY_FUNC) redir->func; + data = redir->data; + g_free_and_null(redir); + + gui_entry_set_prompt(active_entry, ""); + + if (func != NULL) { + func(line, data, active_win->active_server, + active_win->active); + } +} + +static int get_scroll_count(void) +{ + const char *str; + double count; + + str = settings_get_str("scroll_page_count"); + count = atof(str + (*str == '/')); + if (count == 0) + count = 1; + else if (count < 0) + count = active_mainwin->height-active_mainwin->statusbar_lines+count; + else if (count < 1) + count = 1.0/count; + + if (*str == '/' || *str == '.') { + count = (active_mainwin->height-active_mainwin->statusbar_lines)/count; + } + return (int)count; +} + +static void window_prev_page(void) +{ + gui_window_scroll(active_win, -get_scroll_count()); +} + +static void window_next_page(void) +{ + gui_window_scroll(active_win, get_scroll_count()); +} + +#define isnewline(x) ((x) == '\n' || (x) == '\r') +static void paste_buffer_join_lines(GArray *buf) +{ + unsigned int i, count, indent, line_len; + unichar *arr, *dest, *last_lf_pos; + int last_lf; + + /* first check if we actually want to join anything. This is assuming + that we only want to join lines if + + a) first line doesn't begin with whitespace + b) subsequent lines begin with same amount of whitespace + c) whenever there's no whitespace, goto a) + + For example: + + line 1 + line 2 + line 3 + line 4 + line 5 + line 6 + + -> + + line1 line2 line 3 + line4 + line5 line 6 + */ + if (buf->len == 0) + return; + + arr = (unichar *)buf->data; + + /* first line */ + if (isblank(arr[0])) + return; + + /* find the first beginning of indented line */ + for (i = 1; i < buf->len; i++) { + if (isnewline(arr[i-1]) && isblank(arr[i])) + break; + } + if (i == buf->len) + return; + + /* get how much indentation we have.. */ + for (indent = 0; i < buf->len; i++, indent++) { + if (!isblank(arr[i])) + break; + } + if (i == buf->len) + return; + + /* now, enforce these to all subsequent lines */ + count = indent; last_lf = TRUE; + for (; i < buf->len; i++) { + if (last_lf) { + if (isblank(arr[i])) + count++; + else { + last_lf = FALSE; + if (count != 0 && count != indent) + return; + count = 0; + } + } + if (isnewline(arr[i])) + last_lf = TRUE; + } + + /* all looks fine - now remove the whitespace, but don't let lines + get longer than LINE_SPLIT_LIMIT chars */ + dest = arr; last_lf = TRUE; last_lf_pos = NULL; line_len = 0; + for (i = 0; i < buf->len; i++) { + if (last_lf && isblank(arr[i])) { + /* whitespace, ignore */ + } else if (isnewline(arr[i])) { + if (!last_lf && i+1 != buf->len && + isblank(arr[i+1])) { + last_lf_pos = dest; + if (i != 0 && !isblank(arr[i-1])) + *dest++ = ' '; + } else { + *dest++ = '\n'; /* double-LF */ + line_len = 0; + last_lf_pos = NULL; + } + last_lf = TRUE; + } else { + last_lf = FALSE; + if (++line_len >= LINE_SPLIT_LIMIT && last_lf_pos != NULL) { + memmove(last_lf_pos+1, last_lf_pos, + (dest - last_lf_pos) * sizeof(unichar)); + *last_lf_pos = '\n'; last_lf_pos = NULL; + line_len = 0; + dest++; + } + *dest++ = arr[i]; + } + } + g_array_set_size(buf, dest - arr); +} + +static void paste_send_line(char *text) +{ + /* we need to get the current history every time because it might change between calls */ + command_history_add(command_history_current(active_win), text); + + signal_emit("send command", 3, text, active_win->active_server, active_win->active); +} + +static void paste_send(void) +{ + unichar *arr; + GString *str; + char out[10], *text; + unsigned int i; + + if (paste_join_multiline) + paste_buffer_join_lines(paste_buffer); + + arr = (unichar *) paste_buffer->data; + if (active_entry->text_len == 0) + i = 0; + else { + /* first line has to be kludged kind of to get pasting in the + middle of line right.. */ + for (i = 0; i < paste_buffer->len; i++) { + if (isnewline(arr[i])) { + i++; + break; + } + + gui_entry_insert_char(active_entry, arr[i]); + } + + text = gui_entry_get_text(active_entry); + paste_send_line(text); + g_free(text); + } + + /* rest of the lines */ + str = g_string_new(NULL); + for (; i < paste_buffer->len; i++) { + if (isnewline(arr[i])) { + paste_send_line(str->str); + g_string_truncate(str, 0); + } else if (active_entry->utf8) { + out[g_unichar_to_utf8(arr[i], out)] = '\0'; + g_string_append(str, out); + } else if (term_type == TERM_TYPE_BIG5) { + if (arr[i] > 0xff) + g_string_append_c(str, (arr[i] >> 8) & 0xff); + g_string_append_c(str, arr[i] & 0xff); + } else { + g_string_append_c(str, arr[i]); + } + } + + if (paste_was_bracketed_mode) { + /* the text before the bracket end should be sent along with the rest */ + paste_send_line(str->str); + gui_entry_set_text(active_entry, ""); + } else { + gui_entry_set_text(active_entry, str->str); + } + + g_string_free(str, TRUE); +} + +static void paste_flush(void (*send)(void)) +{ + if (paste_prompt) { + gui_entry_set_text(active_entry, paste_entry); + gui_entry_set_pos(active_entry, paste_entry_pos); + g_free_and_null(paste_entry); + } + + if (send != NULL) + send(); + g_array_set_size(paste_buffer, 0); + + /* re-add anything that may have been after the bracketed paste end */ + if (paste_buffer_rest->len > 0) { + g_array_append_vals(paste_buffer, paste_buffer_rest->data, paste_buffer_rest->len); + g_array_set_size(paste_buffer_rest, 0); + } + + gui_entry_set_prompt(active_entry, + paste_old_prompt == NULL ? "" : paste_old_prompt); + g_free(paste_old_prompt); paste_old_prompt = NULL; + paste_prompt = FALSE; + + paste_line_count = 0; + + gui_entry_redraw(active_entry); +} + +static void paste_print_line(const char *str) +{ + printformat_window(active_win, MSGLEVEL_CLIENTCRAP, TXT_PASTE_CONTENT, str); +} + +static void paste_print(void) +{ + GArray *garr; + unichar *arr; + GString *str; + char out[10]; + unsigned int i; + gboolean free_garr; + + if (paste_join_multiline) { + garr = g_array_copy(paste_buffer); + paste_buffer_join_lines(garr); + free_garr = TRUE; + } else { + garr = paste_buffer; + free_garr = FALSE; + } + + arr = &g_array_index(garr, unichar, 0); + + str = g_string_new(NULL); + for (i = 0; i < garr->len; i++) { + if (isnewline(arr[i])) { + paste_print_line(str->str); + g_string_truncate(str, 0); + } else if (active_entry->utf8) { + out[g_unichar_to_utf8(arr[i], out)] = '\0'; + g_string_append(str, out); + } else if (term_type == TERM_TYPE_BIG5) { + if (arr[i] > 0xff) + g_string_append_c(str, (arr[i] >> 8) & 0xff); + g_string_append_c(str, arr[i] & 0xff); + } else { + g_string_append_c(str, arr[i]); + } + } + + if (str->len) + paste_print_line(str->str); + + g_string_free(str, TRUE); + if (free_garr) + g_array_free(garr, TRUE); +} + +static void paste_event(const char *arg) +{ + GArray *garr; + unichar *arr; + GString *str; + char out[10]; + unsigned int i; + gboolean free_garr; + + if (paste_join_multiline) { + garr = g_array_copy(paste_buffer); + paste_buffer_join_lines(garr); + free_garr = TRUE; + } else { + garr = paste_buffer; + free_garr = FALSE; + } + + arr = &g_array_index(garr, unichar, 0); + str = g_string_new(NULL); + for (i = 0; i < garr->len; i++) { + if (isnewline(arr[i])) { + g_string_append_c(str, '\n'); + } else if (active_entry->utf8) { + out[g_unichar_to_utf8(arr[i], out)] = '\0'; + g_string_append(str, out); + } else if (term_type == TERM_TYPE_BIG5) { + if (arr[i] > 0xff) + g_string_append_c(str, (arr[i] >> 8) & 0xff); + g_string_append_c(str, arr[i] & 0xff); + } else { + g_string_append_c(str, arr[i]); + } + } + + if (signal_emit("paste event", 2, str->str, arg)) { + paste_flush(NULL); + } + + g_string_free(str, TRUE); + if (free_garr) + g_array_free(garr, TRUE); +} + +static void paste_insert_edit(void) +{ + unichar *arr; + unsigned int i; + + if (paste_join_multiline) + paste_buffer_join_lines(paste_buffer); + + arr = &g_array_index(paste_buffer, unichar, 0); + for (i = 0; i < paste_buffer->len; i++) { + if (isnewline(arr[i])) { + gui_entry_insert_char(active_entry, '\\'); + gui_entry_insert_char(active_entry, 'n'); + } else if (arr[i] == 9) { + gui_entry_insert_char(active_entry, '\\'); + gui_entry_insert_char(active_entry, 't'); + } else if (arr[i] == 27) { + gui_entry_insert_char(active_entry, '\\'); + gui_entry_insert_char(active_entry, 'e'); + } else if (arr[i] == '\\') { + gui_entry_insert_char(active_entry, '\\'); + gui_entry_insert_char(active_entry, '\\'); + } else { + gui_entry_insert_char(active_entry, arr[i]); + } + } +} + +static void insert_paste_prompt(void) +{ + char *str; + + /* The actual number of lines that will show up post-line-split */ + int actual_line_count = paste_line_count; + int split_lines = paste_buffer->len / LINE_SPLIT_LIMIT; + + /* in case this prompt is happening due to line-splitting, calculate the + number of lines obtained from this. The number isn't entirely accurate; + we just choose the greater of the two since the exact value isn't + important */ + if (split_lines > paste_verify_line_count && + split_lines > paste_line_count) { + actual_line_count = split_lines; + } + + paste_prompt = TRUE; + paste_old_prompt = g_strdup(active_entry->prompt); + printformat_window(active_win, MSGLEVEL_CLIENTNOTICE, + TXT_PASTE_WARNING, + actual_line_count, + active_win->active == NULL ? "window" : + active_win->active->visible_name); + + str = format_get_text(MODULE_NAME, active_win, NULL, NULL, + TXT_PASTE_PROMPT, 0, 0); + gui_entry_set_prompt(active_entry, str); + paste_entry = gui_entry_get_text(active_entry); + paste_entry_pos = gui_entry_get_pos(active_entry); + gui_entry_set_text(active_entry, ""); + g_free(str); +} + +static void sig_gui_key_pressed(gpointer keyp) +{ + gint64 now; + unichar key; + char str[20]; + int ret; + + key = GPOINTER_TO_INT(keyp); + + if (redir != NULL && redir->flags & ENTRY_REDIRECT_FLAG_HOTKEY) { + handle_key_redirect(key); + return; + } + + now = g_get_real_time(); + + if (key < 32) { + /* control key */ + str[0] = '^'; + str[1] = (char)key+'@'; + str[2] = '\0'; + } else if (key == 127) { + str[0] = '^'; + str[1] = '?'; + str[2] = '\0'; + } else if (!active_entry->utf8) { + if (key <= 0xff) { + str[0] = (char)key; + str[1] = '\0'; + } else { + str[0] = (char) (key >> 8); + str[1] = (char) (key & 0xff); + str[2] = '\0'; + } + } else { + /* need to convert to utf8 */ + str[g_unichar_to_utf8(key, str)] = '\0'; + } + + if (g_strcmp0(str, "^") == 0) { + /* change it as ^-, that is an invalid control char */ + str[1] = '-'; + str[2] = '\0'; + } + + if (escape_next_key) { + escape_next_key = FALSE; + gui_entry_insert_char(active_entry, key); + ret = 1; + } else { + previous_yank_preceded = active_entry->yank_preceded; + active_entry->yank_preceded = FALSE; + active_entry->previous_append_next_kill = active_entry->append_next_kill; + active_entry->append_next_kill = FALSE; + ret = key_pressed(keyboard, str); + if (ret < 0) { + /* key wasn't used for anything, print it */ + gui_entry_insert_char(active_entry, key); + } + if (ret == 0) { + /* combo not complete, ignore append_next_kill and yank_preceded */ + active_entry->append_next_kill = active_entry->previous_append_next_kill; + active_entry->yank_preceded = previous_yank_preceded; + } + } + + /* ret = 0 : some key create multiple characters - we're in the middle + of one. try to detect the keycombo as a single keypress rather than + multiple small onces to avoid incorrect paste detection. + + don't count repeated keys so paste detection won't go on when + you're holding some key down */ + if (ret != 0 && key != prev_key) { + last_keypress = now; + } + prev_key = key; +} + +static void key_send_line(void) +{ + HISTORY_REC *history; + char *str; + int add_history; + + str = gui_entry_get_text(active_entry); + + /* we can't use gui_entry_get_text() later, since the entry might + have been destroyed after we get back */ + add_history = *str != '\0'; + history = command_history_current(active_win); + + if (redir != NULL && redir->flags & ENTRY_REDIRECT_FLAG_HIDDEN) + add_history = 0; + + if (add_history && history != NULL) { + command_history_add(history, str); + } + + if (redir == NULL) { + signal_emit("send command", 3, str, + active_win->active_server, + active_win->active); + } else { + handle_entry_redirect(str); + } + + if (active_entry != NULL) + gui_entry_set_text(active_entry, ""); + command_history_clear_pos(active_win); + + g_free(str); +} + +static void key_combo(void) +{ +} + +static void key_backward_history(void) +{ + const char *text; + char *line; + + line = gui_entry_get_text(active_entry); + text = command_history_prev(active_win, line); + gui_entry_set_text(active_entry, text); + g_free(line); +} + +static void key_forward_history(void) +{ + const char *text; + char *line; + + line = gui_entry_get_text(active_entry); + text = command_history_next(active_win, line); + gui_entry_set_text(active_entry, text); + g_free(line); +} + +static void key_backward_global_history(void) +{ + const char *text; + char *line; + + line = gui_entry_get_text(active_entry); + text = command_global_history_prev(active_win, line); + gui_entry_set_text(active_entry, text); + g_free(line); +} + +static void key_forward_global_history(void) +{ + const char *text; + char *line; + + line = gui_entry_get_text(active_entry); + text = command_global_history_next(active_win, line); + gui_entry_set_text(active_entry, text); + g_free(line); +} + +static void key_erase_history_entry(void) +{ + const char *text; + char *line; + + line = gui_entry_get_text(active_entry); + text = command_history_delete_current(active_win, line); + gui_entry_set_text(active_entry, text); + g_free(line); +} + +static void key_beginning_of_line(void) +{ + gui_entry_set_pos(active_entry, 0); +} + +static void key_end_of_line(void) +{ + gui_entry_set_pos(active_entry, active_entry->text_len); +} + +static void key_backward_character(void) +{ + gui_entry_move_pos(active_entry, -1); +} + +static void key_forward_character(void) +{ + gui_entry_move_pos(active_entry, 1); +} + +static void key_backward_word(void) +{ + gui_entry_move_words(active_entry, -1, FALSE); +} + +static void key_forward_word(void) +{ + gui_entry_move_words(active_entry, 1, FALSE); +} + +static void key_backward_to_space(void) +{ + gui_entry_move_words(active_entry, -1, TRUE); +} + +static void key_forward_to_space(void) +{ + gui_entry_move_words(active_entry, 1, TRUE); +} + +static void key_erase_line(void) +{ + gui_entry_set_pos(active_entry, active_entry->text_len); + gui_entry_erase(active_entry, active_entry->text_len, CUTBUFFER_UPDATE_REPLACE); +} + +static void key_erase_to_beg_of_line(void) +{ + int pos; + + pos = gui_entry_get_pos(active_entry); + gui_entry_erase(active_entry, pos, CUTBUFFER_UPDATE_PREPEND); +} + +static void key_erase_to_end_of_line(void) +{ + int pos; + + pos = gui_entry_get_pos(active_entry); + gui_entry_set_pos(active_entry, active_entry->text_len); + gui_entry_erase(active_entry, active_entry->text_len - pos, CUTBUFFER_UPDATE_APPEND); +} + +static void key_yank_from_cutbuffer(void) +{ + char *cutbuffer; + + cutbuffer = gui_entry_get_cutbuffer(active_entry); + if (cutbuffer != NULL) { + gui_entry_insert_text(active_entry, cutbuffer); + active_entry->yank_preceded = TRUE; + g_free(cutbuffer); + } +} + +static void key_yank_next_cutbuffer(void) +{ + GUI_ENTRY_CUTBUFFER_REC *rec; + guint length = 0; + char *cutbuffer; + + if (!previous_yank_preceded) + return; + + if (active_entry->kill_ring == NULL) + return; + + rec = active_entry->kill_ring->data; + if (rec != NULL) length = rec->cutbuffer_len; + + cutbuffer = gui_entry_get_next_cutbuffer(active_entry); + if (cutbuffer != NULL) { + gui_entry_erase(active_entry, length, CUTBUFFER_UPDATE_NOOP); + gui_entry_insert_text(active_entry, cutbuffer); + active_entry->yank_preceded = TRUE; + g_free(cutbuffer); + } +} + +static void key_transpose_characters(void) +{ + gui_entry_transpose_chars(active_entry); +} + +static void key_transpose_words(void) +{ + gui_entry_transpose_words(active_entry); +} + +static void key_capitalize_word(void) +{ + gui_entry_capitalize_word(active_entry); +} + +static void key_downcase_word(void) +{ + gui_entry_downcase_word(active_entry); +} +static void key_upcase_word(void) +{ + gui_entry_upcase_word(active_entry); +} + +static void key_delete_character(void) +{ + if (gui_entry_get_pos(active_entry) < active_entry->text_len) { + gui_entry_erase_cell(active_entry); + } +} + +static void key_backspace(void) +{ + gui_entry_erase(active_entry, 1, CUTBUFFER_UPDATE_NOOP); +} + +static void key_delete_previous_word(void) +{ + gui_entry_erase_word(active_entry, FALSE, CUTBUFFER_UPDATE_PREPEND); +} + +static void key_delete_next_word(void) +{ + gui_entry_erase_next_word(active_entry, FALSE, CUTBUFFER_UPDATE_APPEND); +} + +static void key_delete_to_previous_space(void) +{ + gui_entry_erase_word(active_entry, TRUE, CUTBUFFER_UPDATE_PREPEND); +} + +static void key_delete_to_next_space(void) +{ + gui_entry_erase_next_word(active_entry, TRUE, CUTBUFFER_UPDATE_APPEND); +} + +static void key_append_next_kill(void) +{ + active_entry->append_next_kill = TRUE; +} + +static gboolean paste_timeout(gpointer data) +{ + int split_lines; + paste_was_bracketed_mode = paste_bracketed_mode; + + if (paste_ignore_first_nl && paste_line_count == 1) { + unichar last_char; + + last_char = g_array_index(paste_buffer, unichar, paste_buffer->len - 1); + + if (isnewline(last_char)) { + g_array_set_size(paste_buffer, paste_buffer->len - 1); + paste_line_count--; + } + } + + /* number of lines after splitting extra-long messages */ + split_lines = paste_buffer->len / LINE_SPLIT_LIMIT; + + /* Take into account the fact that a line may be split every LINE_SPLIT_LIMIT characters */ + if (paste_line_count == 0 && split_lines <= paste_verify_line_count) { + int i; + + for (i = 0; i < paste_buffer->len; i++) { + unichar key = g_array_index(paste_buffer, unichar, i); + signal_emit("gui key pressed", 1, GINT_TO_POINTER(key)); + } + g_array_set_size(paste_buffer, 0); + } else if (paste_verify_line_count > 0 && + (paste_line_count >= paste_verify_line_count || + split_lines > paste_verify_line_count) && + active_win->active != NULL) + insert_paste_prompt(); + else + paste_flush(paste_send); + paste_timeout_id = -1; + return FALSE; +} + +static void paste_bracketed_end(int i, gboolean rest) +{ + unichar last_char; + + /* if there's stuff after the end bracket, save it for later */ + if (rest) { + unichar *start = ((unichar *) paste_buffer->data) + i + G_N_ELEMENTS(bp_end); + int len = paste_buffer->len - i - G_N_ELEMENTS(bp_end); + + g_array_set_size(paste_buffer_rest, 0); + g_array_append_vals(paste_buffer_rest, start, len); + } + + /* remove the rest, including the trailing sequence chars */ + g_array_set_size(paste_buffer, i); + + last_char = g_array_index(paste_buffer, unichar, i - 1); + + if (paste_line_count > 0 && !isnewline(last_char)) { + /* there are newlines, but there's also stuff after the newline + * adjust line count to reflect this */ + paste_line_count++; + } + + /* decide what to do with the buffer */ + if (paste_timeout_id != -1) + g_source_remove(paste_timeout_id); + paste_timeout(NULL); + + paste_bracketed_mode = FALSE; +} + +static void paste_bracketed_middle(void) +{ + int i; + int marklen = G_N_ELEMENTS(bp_end); + int len = paste_buffer->len - marklen; + unichar *ptr = (unichar *) paste_buffer->data; + + if (len < 0) { + return; + } + + for (i = 0; i <= len; i++, ptr++) { + if (ptr[0] == bp_end[0] && memcmp(ptr, bp_end, sizeof(bp_end)) == 0) { + + /* if there are at least 6 bytes after the end, + * check for another start marker right afterwards */ + if (i <= (len - marklen) && + memcmp(ptr + marklen, bp_start, sizeof(bp_start)) == 0) { + + /* remove both markers*/ + g_array_remove_range(paste_buffer, i, marklen * 2); + len -= marklen * 2; + + /* go one step back */ + i--; + ptr--; + continue; + } + paste_bracketed_end(i, i != len); + break; + } + } +} + +static void sig_input(void) +{ + if (!active_entry) { + /* no active entry yet - wait until we have it */ + return; + } + + if (paste_prompt) { + GArray *buffer = g_array_new(FALSE, FALSE, sizeof(unichar)); + int line_count = 0; + unichar key; + term_gets(buffer, &line_count); + key = g_array_index(buffer, unichar, 0); + /* Either Ctrl-k or Ctrl-c is pressed */ + if (key < 32 && key != 13 /* CR */ && key != 10 /* LF */ && key != 27 /* Esc */) { + key_pressed(keyboard, "paste"); + signal_emit("gui key pressed", 1, GINT_TO_POINTER(key)); + } + g_array_free(buffer, TRUE); + } else { + term_gets(paste_buffer, &paste_line_count); + + /* use the bracketed paste mode to detect when the user pastes + * some text into the entry */ + if (paste_bracketed_mode) { + paste_bracketed_middle(); + + } else if (!paste_use_bracketed_mode && paste_detect_time > 0 && paste_buffer->len >= 3) { + if (paste_timeout_id != -1) + g_source_remove(paste_timeout_id); + paste_timeout_id = g_timeout_add(paste_detect_time, paste_timeout, NULL); + } else if (!paste_bracketed_mode) { + int i; + + for (i = 0; i < paste_buffer->len; i++) { + unichar key = g_array_index(paste_buffer, unichar, i); + signal_emit("gui key pressed", 1, GINT_TO_POINTER(key)); + + if (paste_bracketed_mode) { + /* just enabled by the signal, remove what was processed so far */ + g_array_remove_range(paste_buffer, 0, i + 1); + + /* handle single-line / small pastes here */ + paste_bracketed_middle(); + return; + } + } + g_array_set_size(paste_buffer, 0); + paste_line_count = 0; + } + } +} + +static void key_paste_start(void) +{ + if (paste_use_bracketed_mode) { + paste_bracketed_mode = TRUE; + if (paste_timeout_id != -1) + g_source_remove(paste_timeout_id); + paste_timeout_id = g_timeout_add(BRACKETED_PASTE_TIMEOUT, paste_timeout, NULL); + } +} + +static void key_paste_cancel(void) +{ + if (paste_prompt) { + paste_flush(NULL); + } +} + +static void key_paste_print(void) +{ + if (paste_prompt) { + paste_print(); + } +} + +static void key_paste_send(void) +{ + if (paste_prompt) { + paste_flush(paste_send); + } +} + +static void key_paste_edit(void) +{ + if (paste_prompt) { + paste_flush(paste_insert_edit); + } +} + +static void key_paste_event(const char *arg) +{ + if (paste_prompt) { + paste_event(arg); + } +} + +time_t get_idle_time(void) +{ + return last_keypress / G_TIME_SPAN_SECOND; +} + +static void key_scroll_backward(void) +{ + window_prev_page(); +} + +static void key_scroll_forward(void) +{ + window_next_page(); +} + +static void key_scroll_start(void) +{ + signal_emit("command scrollback home", 3, NULL, active_win->active_server, active_win->active); +} + +static void key_scroll_end(void) +{ + signal_emit("command scrollback end", 3, NULL, active_win->active_server, active_win->active); +} + +static void key_change_window(const char *data) +{ + signal_emit("command window goto", 3, data, active_win->active_server, active_win->active); +} + +static void key_completion(int erase, int backward) +{ + char *text, *line; + int pos; + + text = gui_entry_get_text_and_pos(active_entry, &pos); + line = word_complete(active_win, text, &pos, erase, backward); + g_free(text); + + if (line != NULL) { + gui_entry_set_text_and_pos_bytes(active_entry, line, pos); + g_free(line); + } +} + +static void key_word_completion_backward(void) +{ + key_completion(FALSE, TRUE); +} + +static void key_word_completion(void) +{ + key_completion(FALSE, FALSE); +} + +static void key_erase_completion(void) +{ + key_completion(TRUE, FALSE); +} + +static void key_check_replaces(void) +{ + char *text, *line; + int pos; + + text = gui_entry_get_text_and_pos(active_entry, &pos); + line = auto_word_complete(text, &pos); + g_free(text); + + if (line != NULL) { + gui_entry_set_text_and_pos_bytes(active_entry, line, pos); + g_free(line); + } +} + +static void key_previous_window(void) +{ + signal_emit("command window previous", 3, "", active_win->active_server, active_win->active); +} + +static void key_next_window(void) +{ + signal_emit("command window next", 3, "", active_win->active_server, active_win->active); +} + +static void key_left_window(void) +{ + signal_emit("command window left", 3, "", active_win->active_server, active_win->active); +} + +static void key_right_window(void) +{ + signal_emit("command window right", 3, "", active_win->active_server, active_win->active); +} + +static void key_upper_window(void) +{ + signal_emit("command window up", 3, "", active_win->active_server, active_win->active); +} + +static void key_lower_window(void) +{ + signal_emit("command window down", 3, "", active_win->active_server, active_win->active); +} + +static void key_active_window(void) +{ + signal_emit("command window goto", 3, "active", active_win->active_server, active_win->active); +} + +static SERVER_REC *get_prev_server(SERVER_REC *current) +{ + int pos; + + if (current == NULL) { + return servers != NULL ? g_slist_last(servers)->data : + lookup_servers != NULL ? + g_slist_last(lookup_servers)->data : NULL; + } + + /* connect2 -> connect1 -> server2 -> server1 -> connect2 -> .. */ + + pos = g_slist_index(servers, current); + if (pos != -1) { + if (pos > 0) + return g_slist_nth(servers, pos-1)->data; + if (lookup_servers != NULL) + return g_slist_last(lookup_servers)->data; + return g_slist_last(servers)->data; + } + + pos = g_slist_index(lookup_servers, current); + g_assert(pos >= 0); + + if (pos > 0) + return g_slist_nth(lookup_servers, pos-1)->data; + if (servers != NULL) + return g_slist_last(servers)->data; + return g_slist_last(lookup_servers)->data; +} + +static SERVER_REC *get_next_server(SERVER_REC *current) +{ + GSList *pos; + + if (current == NULL) { + return servers != NULL ? servers->data : + lookup_servers != NULL ? lookup_servers->data : NULL; + } + + /* server1 -> server2 -> connect1 -> connect2 -> server1 -> .. */ + + pos = g_slist_find(servers, current); + if (pos != NULL) { + if (pos->next != NULL) + return pos->next->data; + if (lookup_servers != NULL) + return lookup_servers->data; + return servers->data; + } + + pos = g_slist_find(lookup_servers, current); + g_assert(pos != NULL); + + if (pos->next != NULL) + return pos->next->data; + if (servers != NULL) + return servers->data; + return lookup_servers->data; +} + +static void key_previous_window_item(void) +{ + SERVER_REC *server; + + if (active_win->items != NULL) { + signal_emit("command window item prev", 3, "", + active_win->active_server, active_win->active); + } else if (servers != NULL || lookup_servers != NULL) { + /* change server */ + server = active_win->active_server; + if (server == NULL) + server = active_win->connect_server; + server = get_prev_server(server); + signal_emit("command window server", 3, server->tag, + active_win->active_server, active_win->active); + } +} + +static void key_next_window_item(void) +{ + SERVER_REC *server; + + if (active_win->items != NULL) { + signal_emit("command window item next", 3, "", + active_win->active_server, active_win->active); + } else if (servers != NULL || lookup_servers != NULL) { + /* change server */ + server = active_win->active_server; + if (server == NULL) + server = active_win->connect_server; + server = get_next_server(server); + signal_emit("command window server", 3, server->tag, + active_win->active_server, active_win->active); + } +} + +static void key_escape(void) +{ + escape_next_key = TRUE; +} + +static void key_insert_text(const char *data) +{ + char *str; + + str = parse_special_string(data, active_win->active_server, + active_win->active, "", NULL, 0); + gui_entry_insert_text(active_entry, str); + g_free(str); +} + +static void key_sig_stop(void) +{ + term_stop(); +} + +static void sig_window_auto_changed(void) +{ + char *text; + + if (active_entry == NULL) + return; + + text = gui_entry_get_text(active_entry); + command_history_next(active_win, text); + gui_entry_set_text(active_entry, ""); + g_free(text); +} + +static void sig_gui_entry_redirect(SIGNAL_FUNC func, const char *entry, + void *flags, void *data) +{ + redir = g_new0(ENTRY_REDIRECT_REC, 1); + redir->func = func; + redir->flags = GPOINTER_TO_INT(flags); + redir->data = data; + + if (redir->flags & ENTRY_REDIRECT_FLAG_HIDDEN) + gui_entry_set_hidden(active_entry, TRUE); + gui_entry_set_prompt(active_entry, entry); +} + +static void setup_changed(void) +{ + paste_detect_time = settings_get_time("paste_detect_time"); + + paste_verify_line_count = settings_get_int("paste_verify_line_count"); + paste_join_multiline = settings_get_bool("paste_join_multiline"); + paste_ignore_first_nl = settings_get_bool("paste_ignore_first_nl"); + paste_use_bracketed_mode = settings_get_bool("paste_use_bracketed_mode"); + + term_set_appkey_mode(settings_get_bool("term_appkey_mode")); + /* Enable the bracketed paste mode on demand */ + term_set_bracketed_paste_mode(paste_use_bracketed_mode); + if (!paste_use_bracketed_mode) + paste_bracketed_mode = FALSE; +} + +void gui_readline_init(void) +{ + static char changekeys[] = "1234567890qwertyuio"; + char *key, data[MAX_INT_STRLEN]; + int n; + + escape_next_key = FALSE; + redir = NULL; + paste_entry = NULL; + paste_entry_pos = 0; + paste_buffer = g_array_new(FALSE, FALSE, sizeof(unichar)); + paste_buffer_rest = g_array_new(FALSE, FALSE, sizeof(unichar)); + paste_old_prompt = NULL; + paste_timeout_id = -1; + paste_bracketed_mode = FALSE; + last_keypress = g_get_real_time(); + input_listen_init(STDIN_FILENO); + + settings_add_bool("lookandfeel", "term_appkey_mode", TRUE); + settings_add_str("history", "scroll_page_count", "/2"); + settings_add_time("misc", "paste_detect_time", "5msecs"); + settings_add_bool("misc", "paste_use_bracketed_mode", FALSE); + /* NOTE: function keys can generate at least 5 characters long + keycodes. this must be larger to allow them to work. */ + settings_add_int("misc", "paste_verify_line_count", 5); + settings_add_bool("misc", "paste_join_multiline", TRUE); + settings_add_bool("misc", "paste_ignore_first_nl", FALSE); + setup_changed(); + + keyboard = keyboard_create(NULL); + key_configure_freeze(); + + key_bind("key", NULL, " ", "space", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "^M", "return", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "^J", "return", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "^H", "backspace", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "^?", "backspace", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "^I", "tab", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-Z", "stab", (SIGNAL_FUNC) key_combo); + + /* meta */ + key_bind("key", NULL, "^[", "meta", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-[", "meta2", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-O", "meta2", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-[O", "meta2", (SIGNAL_FUNC) key_combo); + + /* arrow keys */ + key_bind("key", NULL, "meta2-A", "up", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-B", "down", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-C", "right", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-D", "left", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-1~", "home", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-7~", "home", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-H", "home", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-4~", "end", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-8~", "end", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-F", "end", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-5~", "prior", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-I", "prior", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-6~", "next", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-G", "next", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-2~", "insert", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-3~", "delete", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-d", "cleft", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-c", "cright", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-5D", "cleft", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-5C", "cright", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;5D", "cleft", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;5C", "cright", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;5A", "cup", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;5B", "cdown", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-1;3A", "mup", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;3B", "mdown", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;3D", "mleft", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;3C", "mright", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-up", "mup", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-down", "mdown", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-left", "mleft", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta-right", "mright", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-1;5~", "chome", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-7;5~", "chome", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-5H", "chome", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;5H", "chome", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta2-4;5~", "cend", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-8;5~", "cend", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-5F", "cend", (SIGNAL_FUNC) key_combo); + key_bind("key", NULL, "meta2-1;5F", "cend", (SIGNAL_FUNC) key_combo); + + key_bind("key", NULL, "meta-O-M", "return", (SIGNAL_FUNC) key_combo); + + /* clang-format off */ + key_bind("paste_start", "Bracketed paste start", "^[[200~", "paste_start", (SIGNAL_FUNC) key_paste_start); + key_bind("paste_cancel", "Cancel paste", "paste-^C", NULL, (SIGNAL_FUNC) key_paste_cancel); + key_bind("paste_print", "Print paste to screen", "paste-^P", NULL, (SIGNAL_FUNC) key_paste_print); + key_bind("paste_send", "Send paste to target", "paste-^K", NULL, (SIGNAL_FUNC) key_paste_send); + key_bind("paste_edit", "Insert paste to input line", "paste-^E", NULL, (SIGNAL_FUNC) key_paste_edit); + key_bind("paste_event", "Send paste to event", "paste-^U", NULL, (SIGNAL_FUNC) key_paste_event); + + /* cursor movement */ + key_bind("backward_character", "Move the cursor a character backward", "left", NULL, (SIGNAL_FUNC) key_backward_character); + key_bind("forward_character", "Move the cursor a character forward", "right", NULL, (SIGNAL_FUNC) key_forward_character); + key_bind("backward_word", "Move the cursor a word backward", "cleft", NULL, (SIGNAL_FUNC) key_backward_word); + key_bind("backward_word", NULL, "meta-b", NULL, (SIGNAL_FUNC) key_backward_word); + key_bind("forward_word", "Move the cursor a word forward", "cright", NULL, (SIGNAL_FUNC) key_forward_word); + key_bind("forward_word", NULL, "meta-f", NULL, (SIGNAL_FUNC) key_forward_word); + key_bind("backward_to_space", "Move the cursor backward to a space", NULL, NULL, (SIGNAL_FUNC) key_backward_to_space); + key_bind("forward_to_space", "Move the cursor forward to a space", NULL, NULL, (SIGNAL_FUNC) key_forward_to_space); + key_bind("beginning_of_line", "Move the cursor to the beginning of the line", "home", NULL, (SIGNAL_FUNC) key_beginning_of_line); + key_bind("beginning_of_line", NULL, "^A", NULL, (SIGNAL_FUNC) key_beginning_of_line); + key_bind("end_of_line", "Move the cursor to the end of the line", "end", NULL, (SIGNAL_FUNC) key_end_of_line); + key_bind("end_of_line", NULL, "^E", NULL, (SIGNAL_FUNC) key_end_of_line); + + /* history */ + key_bind("backward_history", "Go back one line in the history", "up", NULL, (SIGNAL_FUNC) key_backward_history); + key_bind("forward_history", "Go forward one line in the history", "down", NULL, (SIGNAL_FUNC) key_forward_history); + key_bind("backward_global_history", "Go back one line in the global history", "cup", NULL, (SIGNAL_FUNC) key_backward_global_history); + key_bind("forward_global_history", "Go forward one line in the global history", "cdown", NULL, (SIGNAL_FUNC) key_forward_global_history); + key_bind("erase_history_entry", "Erase the currently active entry from the history", NULL, NULL, (SIGNAL_FUNC) key_erase_history_entry); + + /* line editing */ + key_bind("backspace", "Delete the previous character", "backspace", NULL, (SIGNAL_FUNC) key_backspace); + key_bind("delete_character", "Delete the current character", "delete", NULL, (SIGNAL_FUNC) key_delete_character); + key_bind("delete_character", NULL, "^D", NULL, (SIGNAL_FUNC) key_delete_character); + key_bind("delete_next_word", "Delete the word after the cursor", "meta-d", NULL, (SIGNAL_FUNC) key_delete_next_word); + key_bind("delete_previous_word", "Delete the word before the cursor", "meta-backspace", NULL, (SIGNAL_FUNC) key_delete_previous_word); + key_bind("delete_to_previous_space", "Delete up to the previous space", "^W", NULL, (SIGNAL_FUNC) key_delete_to_previous_space); + key_bind("delete_to_next_space", "Delete up to the next space", "", NULL, (SIGNAL_FUNC) key_delete_to_next_space); + key_bind("erase_line", "Erase the whole input line", "^U", NULL, (SIGNAL_FUNC) key_erase_line); + key_bind("erase_to_beg_of_line", "Erase everything before the cursor", NULL, NULL, (SIGNAL_FUNC) key_erase_to_beg_of_line); + key_bind("erase_to_end_of_line", "Erase everything after the cursor", "^K", NULL, (SIGNAL_FUNC) key_erase_to_end_of_line); + key_bind("yank_from_cutbuffer", "\"Undelete\", paste the last deleted text", "^Y", NULL, (SIGNAL_FUNC) key_yank_from_cutbuffer); + key_bind("yank_next_cutbuffer", "Revert to the previous last deleted text", NULL, NULL, (SIGNAL_FUNC) key_yank_next_cutbuffer); + key_bind("append_next_kill", "Append next deletion", NULL, NULL, (SIGNAL_FUNC) key_append_next_kill); + key_bind("transpose_characters", "Swap current and previous character", "^T", NULL, (SIGNAL_FUNC) key_transpose_characters); + key_bind("transpose_words", "Swap current and previous word", NULL, NULL, (SIGNAL_FUNC) key_transpose_words); + key_bind("capitalize_word", "Capitalize the current word", NULL, NULL, (SIGNAL_FUNC) key_capitalize_word); + key_bind("downcase_word", "Downcase the current word", NULL, NULL, (SIGNAL_FUNC) key_downcase_word); + key_bind("upcase_word", "Upcase the current word", NULL, NULL, (SIGNAL_FUNC) key_upcase_word); + + /* line transmitting */ + key_bind("send_line", "Execute the input line", "return", NULL, (SIGNAL_FUNC) key_send_line); + key_bind("word_completion_backward", "Choose previous completion suggestion", "stab", NULL, (SIGNAL_FUNC) key_word_completion_backward); + key_bind("word_completion", "Complete the current word", "tab", NULL, (SIGNAL_FUNC) key_word_completion); + key_bind("erase_completion", "Remove the completion added by word_completion", "meta-k", NULL, (SIGNAL_FUNC) key_erase_completion); + key_bind("check_replaces", "Check word replaces", NULL, NULL, (SIGNAL_FUNC) key_check_replaces); + + /* window managing */ + key_bind("previous_window", "Go to the previous window", "^P", NULL, (SIGNAL_FUNC) key_previous_window); + key_bind("next_window", "Go to the next window", "^N", NULL, (SIGNAL_FUNC) key_next_window); + key_bind("upper_window", "Go to the split window above", "mup", NULL, (SIGNAL_FUNC) key_upper_window); + key_bind("lower_window", "Go to the split window below", "mdown", NULL, (SIGNAL_FUNC) key_lower_window); + key_bind("left_window", "Go to the previous window in the current split window", "mleft", NULL, (SIGNAL_FUNC) key_left_window); + key_bind("right_window", "Go to the next window in the current split window", "mright", NULL, (SIGNAL_FUNC) key_right_window); + key_bind("active_window", "Go to next window with the highest activity", "meta-a", NULL, (SIGNAL_FUNC) key_active_window); + key_bind("next_window_item", "Go to the next channel/query. In empty windows change to the next server", "^X", NULL, (SIGNAL_FUNC) key_next_window_item); + key_bind("previous_window_item", "Go to the previous channel/query. In empty windows change to the previous server", NULL, NULL, (SIGNAL_FUNC) key_previous_window_item); + + key_bind("refresh_screen", "Redraw screen", "^L", NULL, (SIGNAL_FUNC) irssi_redraw); + key_bind("scroll_backward", "Scroll to previous page", "prior", NULL, (SIGNAL_FUNC) key_scroll_backward); + key_bind("scroll_backward", NULL, "meta-p", NULL, (SIGNAL_FUNC) key_scroll_backward); + key_bind("scroll_forward", "Scroll to next page", "next", NULL, (SIGNAL_FUNC) key_scroll_forward); + key_bind("scroll_forward", NULL, "meta-n", NULL, (SIGNAL_FUNC) key_scroll_forward); + key_bind("scroll_start", "Scroll to the beginning of the window", "chome", NULL, (SIGNAL_FUNC) key_scroll_start); + key_bind("scroll_end", "Scroll to the end of the window", "cend", NULL, (SIGNAL_FUNC) key_scroll_end); + + /* inserting special input characters to line.. */ + key_bind("escape_char", "Insert the next character exactly as-is to input line", NULL, NULL, (SIGNAL_FUNC) key_escape); + key_bind("insert_text", "Append text to line", NULL, NULL, (SIGNAL_FUNC) key_insert_text); + /* clang-format on */ + + /* autoreplaces */ + key_bind("multi", NULL, "return", "check_replaces;send_line", NULL); + key_bind("multi", NULL, "space", "check_replaces;insert_text ", NULL); + + /* moving between windows */ + for (n = 0; changekeys[n] != '\0'; n++) { + key = g_strdup_printf("meta-%c", changekeys[n]); + ltoa(data, n+1); + key_bind("change_window", "Change window", key, data, (SIGNAL_FUNC) key_change_window); + g_free(key); + } + + /* misc */ + key_bind("stop_irc", "Send SIGSTOP to client", "^Z", NULL, (SIGNAL_FUNC) key_sig_stop); + + key_configure_thaw(); + + signal_add("window changed automatic", (SIGNAL_FUNC) sig_window_auto_changed); + signal_add("gui entry redirect", (SIGNAL_FUNC) sig_gui_entry_redirect); + signal_add("gui key pressed", (SIGNAL_FUNC) sig_gui_key_pressed); + signal_add("setup changed", (SIGNAL_FUNC) setup_changed); +} + +void gui_readline_deinit(void) +{ + input_listen_deinit(); + + key_configure_freeze(); + + key_unbind("paste_start", (SIGNAL_FUNC) key_paste_start); + key_unbind("paste_cancel", (SIGNAL_FUNC) key_paste_cancel); + key_unbind("paste_print", (SIGNAL_FUNC) key_paste_print); + key_unbind("paste_send", (SIGNAL_FUNC) key_paste_send); + key_unbind("paste_edit", (SIGNAL_FUNC) key_paste_edit); + key_unbind("paste_event", (SIGNAL_FUNC) key_paste_event); + + key_unbind("backward_character", (SIGNAL_FUNC) key_backward_character); + key_unbind("forward_character", (SIGNAL_FUNC) key_forward_character); + key_unbind("backward_word", (SIGNAL_FUNC) key_backward_word); + key_unbind("forward_word", (SIGNAL_FUNC) key_forward_word); + key_unbind("backward_to_space", (SIGNAL_FUNC) key_backward_to_space); + key_unbind("forward_to_space", (SIGNAL_FUNC) key_forward_to_space); + key_unbind("beginning_of_line", (SIGNAL_FUNC) key_beginning_of_line); + key_unbind("end_of_line", (SIGNAL_FUNC) key_end_of_line); + + key_unbind("backward_history", (SIGNAL_FUNC) key_backward_history); + key_unbind("forward_history", (SIGNAL_FUNC) key_forward_history); + key_unbind("backward_global_history", (SIGNAL_FUNC) key_backward_global_history); + key_unbind("forward_global_history", (SIGNAL_FUNC) key_forward_global_history); + key_unbind("erase_history_entry", (SIGNAL_FUNC) key_erase_history_entry); + + key_unbind("backspace", (SIGNAL_FUNC) key_backspace); + key_unbind("delete_character", (SIGNAL_FUNC) key_delete_character); + key_unbind("delete_next_word", (SIGNAL_FUNC) key_delete_next_word); + key_unbind("delete_previous_word", (SIGNAL_FUNC) key_delete_previous_word); + key_unbind("delete_to_next_space", (SIGNAL_FUNC) key_delete_to_next_space); + key_unbind("delete_to_previous_space", (SIGNAL_FUNC) key_delete_to_previous_space); + key_unbind("erase_line", (SIGNAL_FUNC) key_erase_line); + key_unbind("erase_to_beg_of_line", (SIGNAL_FUNC) key_erase_to_beg_of_line); + key_unbind("erase_to_end_of_line", (SIGNAL_FUNC) key_erase_to_end_of_line); + key_unbind("yank_from_cutbuffer", (SIGNAL_FUNC) key_yank_from_cutbuffer); + key_unbind("yank_next_cutbuffer", (SIGNAL_FUNC) key_yank_next_cutbuffer); + key_unbind("append_next_kill", (SIGNAL_FUNC) key_append_next_kill); + key_unbind("transpose_characters", (SIGNAL_FUNC) key_transpose_characters); + key_unbind("transpose_words", (SIGNAL_FUNC) key_transpose_words); + + key_unbind("capitalize_word", (SIGNAL_FUNC) key_capitalize_word); + key_unbind("downcase_word", (SIGNAL_FUNC) key_downcase_word); + key_unbind("upcase_word", (SIGNAL_FUNC) key_upcase_word); + + key_unbind("send_line", (SIGNAL_FUNC) key_send_line); + key_unbind("word_completion_backward", (SIGNAL_FUNC) key_word_completion_backward); + key_unbind("word_completion", (SIGNAL_FUNC) key_word_completion); + key_unbind("erase_completion", (SIGNAL_FUNC) key_erase_completion); + key_unbind("check_replaces", (SIGNAL_FUNC) key_check_replaces); + + key_unbind("previous_window", (SIGNAL_FUNC) key_previous_window); + key_unbind("next_window", (SIGNAL_FUNC) key_next_window); + key_unbind("upper_window", (SIGNAL_FUNC) key_upper_window); + key_unbind("lower_window", (SIGNAL_FUNC) key_lower_window); + key_unbind("left_window", (SIGNAL_FUNC) key_left_window); + key_unbind("right_window", (SIGNAL_FUNC) key_right_window); + key_unbind("active_window", (SIGNAL_FUNC) key_active_window); + key_unbind("next_window_item", (SIGNAL_FUNC) key_next_window_item); + key_unbind("previous_window_item", (SIGNAL_FUNC) key_previous_window_item); + + key_unbind("refresh_screen", (SIGNAL_FUNC) irssi_redraw); + key_unbind("scroll_backward", (SIGNAL_FUNC) key_scroll_backward); + key_unbind("scroll_forward", (SIGNAL_FUNC) key_scroll_forward); + key_unbind("scroll_start", (SIGNAL_FUNC) key_scroll_start); + key_unbind("scroll_end", (SIGNAL_FUNC) key_scroll_end); + + key_unbind("escape_char", (SIGNAL_FUNC) key_escape); + key_unbind("insert_text", (SIGNAL_FUNC) key_insert_text); + key_unbind("change_window", (SIGNAL_FUNC) key_change_window); + key_unbind("stop_irc", (SIGNAL_FUNC) key_sig_stop); + keyboard_destroy(keyboard); + g_array_free(paste_buffer, TRUE); + g_array_free(paste_buffer_rest, TRUE); + + key_configure_thaw(); + + signal_remove("window changed automatic", (SIGNAL_FUNC) sig_window_auto_changed); + signal_remove("gui entry redirect", (SIGNAL_FUNC) sig_gui_entry_redirect); + signal_remove("gui key pressed", (SIGNAL_FUNC) sig_gui_key_pressed); + signal_remove("setup changed", (SIGNAL_FUNC) setup_changed); +} |