/* 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 "module-formats.h" #include "signals.h" #include "misc.h" #include "settings.h" #include "special-vars.h" #include "levels.h" #include "servers.h" #include "completion.h" #include "command-history.h" #include "keyboard.h" #include "printtext.h" #include "term.h" #include "gui-entry.h" #include "gui-windows.h" #include "utf8.h" #include #include /* 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 GTimeVal 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_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', '~' }; static void sig_input(void); void input_listen_init(int handle) { readtag = g_input_add_poll(handle, G_PRIORITY_HIGH, G_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 == '/') { 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()); } 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 (arr[i-1] == '\n' && 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 (arr[i] == '\n') 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 (arr[i] == '\n') { if (!last_lf && i+1 != buf->len && isblank(arr[i+1])) { last_lf_pos = dest; *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 (arr[i] == '\r' || arr[i] == '\n') { 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 (arr[i] == '\r' || arr[i] == '\n') { 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(int send) { 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) paste_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 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) { GTimeVal 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; } g_get_current_time(&now); 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_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) { if (settings_get_bool("bell_beeps")) { signal_emit("beep", 0); } 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) { paste_was_bracketed_mode = paste_bracketed_mode; /* number of lines after splitting extra-long messages */ int 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(TRUE); 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 && last_char != '\n' && last_char != '\r') { /* 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 */ paste_timeout(NULL); paste_bracketed_mode = FALSE; } static void paste_bracketed_middle() { 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 == 11 || key == 3) paste_flush(key == 11); 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) { paste_bracketed_mode = TRUE; } time_t get_idle_time(void) { return last_keypress.tv_sec; } 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(active_entry, line); gui_entry_set_pos(active_entry, 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(active_entry, line); gui_entry_set_pos(active_entry, 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_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); } 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; g_get_current_time(&last_keypress); 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); 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); /* 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;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); key_bind("paste_start", "Bracketed paste start", "meta2-200~", "paste_start", (SIGNAL_FUNC) key_paste_start); /* 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); /* 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", "", NULL, 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); /* 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("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("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); }