66867ffb7de0924074087efaf629eacbe806e4ab
[silc.git] / apps / irssi / src / fe-text / gui-readline.c
1 /*
2  gui-readline.c : irssi
3
4     Copyright (C) 1999 Timo Sirainen
5
6     This program is free software; you can redistribute it and/or modify
7     it under the terms of the GNU General Public License as published by
8     the Free Software Foundation; either version 2 of the License, or
9     (at your option) any later version.
10
11     This program is distributed in the hope that it will be useful,
12     but WITHOUT ANY WARRANTY; without even the implied warranty of
13     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14     GNU General Public License for more details.
15
16     You should have received a copy of the GNU General Public License
17     along with this program; if not, write to the Free Software
18     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19 */
20
21 #include "module.h"
22 #include "signals.h"
23 #include "misc.h"
24 #include "settings.h"
25 #include "special-vars.h"
26 #include "servers.h"
27
28 #include "completion.h"
29 #include "command-history.h"
30 #include "keyboard.h"
31 #include "translation.h"
32
33 #include "term.h"
34 #include "gui-entry.h"
35 #include "gui-windows.h"
36 #include "utf8.h"
37
38 #include <signal.h>
39
40 typedef void (*ENTRY_REDIRECT_KEY_FUNC) (int key, void *data, SERVER_REC *server, WI_ITEM_REC *item);
41 typedef void (*ENTRY_REDIRECT_ENTRY_FUNC) (const char *line, void *data, SERVER_REC *server, WI_ITEM_REC *item);
42
43 typedef struct {
44         SIGNAL_FUNC func;
45         int flags;
46         void *data;
47 } ENTRY_REDIRECT_REC;
48
49 static KEYBOARD_REC *keyboard;
50 static ENTRY_REDIRECT_REC *redir;
51 static int escape_next_key;
52
53 static int readtag;
54 static time_t idle_time;
55
56 static void sig_input(void);
57
58 void input_listen_init(int handle)
59 {
60         GIOChannel *stdin_channel;
61
62         stdin_channel = g_io_channel_unix_new(handle);
63         readtag = g_input_add_full(stdin_channel,
64                                    G_PRIORITY_HIGH, G_INPUT_READ,
65                                    (GInputFunction) sig_input, NULL);
66         g_io_channel_unref(stdin_channel);
67 }
68
69 void input_listen_deinit(void)
70 {
71         g_source_remove(readtag);
72         readtag = -1;
73 }
74
75 static void handle_key_redirect(int key)
76 {
77         ENTRY_REDIRECT_KEY_FUNC func;
78         void *data;
79
80         func = (ENTRY_REDIRECT_KEY_FUNC) redir->func;
81         data = redir->data;
82         g_free_and_null(redir);
83
84         gui_entry_set_prompt(active_entry, "");
85
86         if (func != NULL)
87                 func(key, data, active_win->active_server, active_win->active);
88 }
89
90 static void handle_entry_redirect(const char *line)
91 {
92         ENTRY_REDIRECT_ENTRY_FUNC func;
93         void *data;
94
95         gui_entry_set_hidden(active_entry, FALSE);
96
97         func = (ENTRY_REDIRECT_ENTRY_FUNC) redir->func;
98         data = redir->data;
99         g_free_and_null(redir);
100
101         gui_entry_set_prompt(active_entry, "");
102
103         if (func != NULL) {
104                 func(line, data, active_win->active_server,
105                      active_win->active);
106         }
107 }
108
109 static int get_scroll_count(void)
110 {
111         const char *str;
112         double count;
113
114         str = settings_get_str("scroll_page_count");
115         count = atof(str + (*str == '/'));
116         if (count <= 0)
117                 count = 1;
118         else if (count < 1)
119                 count = 1.0/count;
120
121         if (*str == '/') {
122                 count = (active_mainwin->height-active_mainwin->statusbar_lines)/count;
123         }
124         return (int)count;
125 }
126
127 static void window_prev_page(void)
128 {
129         gui_window_scroll(active_win, -get_scroll_count());
130 }
131
132 static void window_next_page(void)
133 {
134         gui_window_scroll(active_win, get_scroll_count());
135 }
136
137 static void sig_gui_key_pressed(gpointer keyp)
138 {
139         unichar key;
140         char str[20];
141
142         key = GPOINTER_TO_INT(keyp);
143
144         if (redir != NULL && redir->flags & ENTRY_REDIRECT_FLAG_HOTKEY) {
145                 handle_key_redirect(key);
146                 return;
147         }
148
149         idle_time = time(NULL);
150
151         if (key < 32) {
152                 /* control key */
153                 str[0] = '^';
154                 str[1] = (char)key+'@';
155                 str[2] = '\0';
156         } else if (key == 127) {
157                 str[0] = '^';
158                 str[1] = '?';
159                 str[2] = '\0';
160         } else if (!active_entry->utf8) {
161                 str[0] = (char)key;
162                 str[1] = '\0';
163         } else {
164                 /* need to convert to utf8 */
165                 str[utf16_char_to_utf8(key, str)] = '\0';
166         }
167
168         if (strcmp(str, "^") == 0) {
169                 /* change it as ^^ */
170                 str[1] = '^';
171                 str[2] = '\0';
172         }
173
174         if (escape_next_key || !key_pressed(keyboard, str)) {
175                 /* key wasn't used for anything, print it */
176                 escape_next_key = FALSE;
177                 gui_entry_insert_char(active_entry, key);
178         }
179 }
180
181 static void key_send_line(void)
182 {
183         HISTORY_REC *history;
184         char *str, *add_history;
185
186         str = gui_entry_get_text(active_entry);
187
188         /* we can't use gui_entry_get_text() later, since the entry might
189            have been destroyed after we get back */
190         add_history = *str == '\0' ? NULL : g_strdup(str);
191         history = command_history_current(active_win);
192
193         translate_output(str);
194
195         if (redir == NULL) {
196                 signal_emit("send command", 3, str,
197                             active_win->active_server,
198                             active_win->active);
199         } else {
200                 if (redir->flags & ENTRY_REDIRECT_FLAG_HIDDEN)
201                         g_free_and_null(add_history);
202                 handle_entry_redirect(str);
203         }
204
205         if (add_history != NULL) {
206                 history = command_history_find(history);
207                 if (history != NULL)
208                         command_history_add(history, add_history);
209                 g_free(add_history);
210         }
211
212         if (active_entry != NULL)
213                 gui_entry_set_text(active_entry, "");
214         command_history_clear_pos(active_win);
215
216         g_free(str);
217 }
218
219 static void key_combo(void)
220 {
221 }
222
223 static void key_backward_history(void)
224 {
225         const char *text;
226         char *line;
227
228         line = gui_entry_get_text(active_entry);
229         text = command_history_prev(active_win, line);
230         gui_entry_set_text(active_entry, text);
231         g_free(line);
232 }
233
234 static void key_forward_history(void)
235 {
236         const char *text;
237         char *line;
238
239         line = gui_entry_get_text(active_entry);
240         text = command_history_next(active_win, line);
241         gui_entry_set_text(active_entry, text);
242         g_free(line);
243 }
244
245 static void key_beginning_of_line(void)
246 {
247         gui_entry_set_pos(active_entry, 0);
248 }
249
250 static void key_end_of_line(void)
251 {
252         gui_entry_set_pos(active_entry, active_entry->text_len);
253 }
254
255 static void key_backward_character(void)
256 {
257         gui_entry_move_pos(active_entry, -1);
258 }
259
260 static void key_forward_character(void)
261 {
262         gui_entry_move_pos(active_entry, 1);
263 }
264
265 static void key_backward_word(void)
266 {
267         gui_entry_move_words(active_entry, -1, FALSE);
268 }
269
270 static void key_forward_word(void)
271 {
272         gui_entry_move_words(active_entry, 1, FALSE);
273 }
274
275 static void key_backward_to_space(void)
276 {
277         gui_entry_move_words(active_entry, -1, TRUE);
278 }
279
280 static void key_forward_to_space(void)
281 {
282         gui_entry_move_words(active_entry, 1, TRUE);
283 }
284
285 static void key_erase_line(void)
286 {
287         gui_entry_set_pos(active_entry, active_entry->text_len);
288         gui_entry_erase(active_entry, active_entry->text_len, TRUE);
289 }
290
291 static void key_erase_to_beg_of_line(void)
292 {
293         int pos;
294
295         pos = gui_entry_get_pos(active_entry);
296         gui_entry_erase(active_entry, pos, TRUE);
297 }
298
299 static void key_erase_to_end_of_line(void)
300 {
301         int pos;
302
303         pos = gui_entry_get_pos(active_entry);
304         gui_entry_set_pos(active_entry, active_entry->text_len);
305         gui_entry_erase(active_entry, active_entry->text_len - pos, TRUE);
306 }
307
308 static void key_yank_from_cutbuffer(void)
309 {
310         char *cutbuffer;
311
312         cutbuffer = gui_entry_get_cutbuffer(active_entry);
313         if (cutbuffer != NULL) {
314                 gui_entry_insert_text(active_entry, cutbuffer);
315                 g_free(cutbuffer);
316         }
317 }
318
319 static void key_transpose_characters(void)
320 {
321         gui_entry_transpose_chars(active_entry);
322 }
323
324 static void key_delete_character(void)
325 {
326         if (gui_entry_get_pos(active_entry) < active_entry->text_len) {
327                 gui_entry_move_pos(active_entry, 1);
328                 gui_entry_erase(active_entry, 1, FALSE);
329         }
330 }
331
332 static void key_backspace(void)
333 {
334         gui_entry_erase(active_entry, 1, FALSE);
335 }
336
337 static void key_delete_previous_word(void)
338 {
339         gui_entry_erase_word(active_entry, FALSE);
340 }
341
342 static void key_delete_next_word(void)
343 {
344         gui_entry_erase_next_word(active_entry, FALSE);
345 }
346
347 static void key_delete_to_previous_space(void)
348 {
349         gui_entry_erase_word(active_entry, TRUE);
350 }
351
352 static void key_delete_to_next_space(void)
353 {
354         gui_entry_erase_next_word(active_entry, TRUE);
355 }
356
357 static void sig_input(void)
358 {
359         unichar buffer[128];
360         int ret, i;
361
362         if (!active_entry) {
363                 /* no active entry yet - wait until we have it */
364                 return;
365         }
366
367         ret = term_gets(buffer, sizeof(buffer)/sizeof(buffer[0]));
368         if (ret == -1) {
369                 /* lost terminal */
370                 if (!term_detached)
371                         signal_emit("command quit", 1, "Lost terminal");
372         } else {
373                 for (i = 0; i < ret; i++) {
374                         signal_emit("gui key pressed", 1,
375                                     GINT_TO_POINTER(buffer[i]));
376                 }
377         }
378 }
379
380 time_t get_idle_time(void)
381 {
382         return idle_time;
383 }
384
385 static void key_scroll_backward(void)
386 {
387         window_prev_page();
388 }
389
390 static void key_scroll_forward(void)
391 {
392         window_next_page();
393 }
394
395 static void key_scroll_start(void)
396 {
397         signal_emit("command scrollback home", 3, NULL, active_win->active_server, active_win->active);
398 }
399
400 static void key_scroll_end(void)
401 {
402         signal_emit("command scrollback end", 3, NULL, active_win->active_server, active_win->active);
403 }
404
405 static void key_change_window(const char *data)
406 {
407         signal_emit("command window goto", 3, data, active_win->active_server, active_win->active);
408 }
409
410 static void key_completion(int erase)
411 {
412         char *text, *line;
413         int pos;
414
415         pos = gui_entry_get_pos(active_entry);
416
417         text = gui_entry_get_text(active_entry);
418         line = word_complete(active_win, text, &pos, erase);
419         g_free(text);
420
421         if (line != NULL) {
422                 gui_entry_set_text(active_entry, line);
423                 gui_entry_set_pos(active_entry, pos);
424                 g_free(line);
425         }
426 }
427
428 static void key_word_completion(void)
429 {
430         key_completion(FALSE);
431 }
432
433 static void key_erase_completion(void)
434 {
435         key_completion(TRUE);
436 }
437
438 static void key_check_replaces(void)
439 {
440         char *text, *line;
441         int pos;
442
443         pos = gui_entry_get_pos(active_entry);
444
445         text = gui_entry_get_text(active_entry);
446         line = auto_word_complete(text, &pos);
447         g_free(text);
448
449         if (line != NULL) {
450                 gui_entry_set_text(active_entry, line);
451                 gui_entry_set_pos(active_entry, pos);
452                 g_free(line);
453         }
454 }
455
456 static void key_previous_window(void)
457 {
458         signal_emit("command window previous", 3, "", active_win->active_server, active_win->active);
459 }
460
461 static void key_next_window(void)
462 {
463         signal_emit("command window next", 3, "", active_win->active_server, active_win->active);
464 }
465
466 static void key_left_window(void)
467 {
468         signal_emit("command window left", 3, "", active_win->active_server, active_win->active);
469 }
470
471 static void key_right_window(void)
472 {
473         signal_emit("command window right", 3, "", active_win->active_server, active_win->active);
474 }
475
476 static void key_upper_window(void)
477 {
478         signal_emit("command window up", 3, "", active_win->active_server, active_win->active);
479 }
480
481 static void key_lower_window(void)
482 {
483         signal_emit("command window down", 3, "", active_win->active_server, active_win->active);
484 }
485
486 static void key_active_window(void)
487 {
488         signal_emit("command window goto", 3, "active", active_win->active_server, active_win->active);
489 }
490
491 static SERVER_REC *get_prev_server(SERVER_REC *current)
492 {
493         int pos;
494
495         if (current == NULL) {
496                 return servers != NULL ? g_slist_last(servers)->data :
497                         lookup_servers != NULL ?
498                         g_slist_last(lookup_servers)->data : NULL;
499         }
500
501         /* connect2 -> connect1 -> server2 -> server1 -> connect2 -> .. */
502
503         pos = g_slist_index(servers, current);
504         if (pos != -1) {
505                 if (pos > 0)
506                         return g_slist_nth(servers, pos-1)->data;
507                 if (lookup_servers != NULL)
508                         return g_slist_last(lookup_servers)->data;
509                 return g_slist_last(servers)->data;
510         }
511
512         pos = g_slist_index(lookup_servers, current);
513         g_assert(pos >= 0);
514
515         if (pos > 0)
516                 return g_slist_nth(lookup_servers, pos-1)->data;
517         if (servers != NULL)
518                 return g_slist_last(servers)->data;
519         return g_slist_last(lookup_servers)->data;
520 }
521
522 static SERVER_REC *get_next_server(SERVER_REC *current)
523 {
524         GSList *pos;
525
526         if (current == NULL) {
527                 return servers != NULL ? servers->data :
528                         lookup_servers != NULL ? lookup_servers->data : NULL;
529         }
530
531         /* server1 -> server2 -> connect1 -> connect2 -> server1 -> .. */
532
533         pos = g_slist_find(servers, current);
534         if (pos != NULL) {
535                 if (pos->next != NULL)
536                         return pos->next->data;
537                 if (lookup_servers != NULL)
538                         return lookup_servers->data;
539                 return servers->data;
540         }
541
542         pos = g_slist_find(lookup_servers, current);
543         g_assert(pos != NULL);
544
545         if (pos->next != NULL)
546                 return pos->next->data;
547         if (servers != NULL)
548                 return servers->data;
549         return lookup_servers->data;
550 }
551
552 static void key_previous_window_item(void)
553 {
554         SERVER_REC *server;
555
556         if (active_win->items != NULL) {
557                 signal_emit("command window item prev", 3, "",
558                             active_win->active_server, active_win->active);
559         } else if (servers != NULL || lookup_servers != NULL) {
560                 /* change server */
561                 server = active_win->active_server;
562                 if (server == NULL)
563                         server = active_win->connect_server;
564                 server = get_prev_server(server);
565                 signal_emit("command window server", 3, server->tag,
566                             active_win->active_server, active_win->active);
567         }
568 }
569
570 static void key_next_window_item(void)
571 {
572         SERVER_REC *server;
573
574         if (active_win->items != NULL) {
575                 signal_emit("command window item next", 3, "",
576                             active_win->active_server, active_win->active);
577         } else if (servers != NULL || lookup_servers != NULL) {
578                 /* change server */
579                 server = active_win->active_server;
580                 if (server == NULL)
581                         server = active_win->connect_server;
582                 server = get_next_server(server);
583                 signal_emit("command window server", 3, server->tag,
584                             active_win->active_server, active_win->active);
585         }
586 }
587
588 static void key_escape(void)
589 {
590         escape_next_key = TRUE;
591 }
592
593 static void key_insert_text(const char *data)
594 {
595         char *str;
596
597         str = parse_special_string(data, active_win->active_server,
598                                    active_win->active, "", NULL, 0);
599         gui_entry_insert_text(active_entry, str);
600         g_free(str);
601 }
602
603 static void key_sig_stop(void)
604 {
605         term_stop();
606 }
607
608 static void sig_window_auto_changed(void)
609 {
610         char *text;
611
612         if (active_entry == NULL)
613                 return;
614
615         text = gui_entry_get_text(active_entry);
616         command_history_next(active_win, text);
617         gui_entry_set_text(active_entry, "");
618         g_free(text);
619 }
620
621 static void sig_gui_entry_redirect(SIGNAL_FUNC func, const char *entry,
622                                    void *flags, void *data)
623 {
624         redir = g_new0(ENTRY_REDIRECT_REC, 1);
625         redir->func = func;
626         redir->flags = GPOINTER_TO_INT(flags);
627         redir->data = data;
628
629         if (redir->flags & ENTRY_REDIRECT_FLAG_HIDDEN)
630                 gui_entry_set_hidden(active_entry, TRUE);
631         gui_entry_set_prompt(active_entry, entry);
632 }
633
634 void gui_readline_init(void)
635 {
636         static char changekeys[] = "1234567890qwertyuio";
637         char *key, data[MAX_INT_STRLEN];
638         int n;
639
640         escape_next_key = FALSE;
641         redir = NULL;
642         idle_time = time(NULL);
643         input_listen_init(STDIN_FILENO);
644
645         settings_add_str("history", "scroll_page_count", "/2");
646
647         keyboard = keyboard_create(NULL);
648         key_configure_freeze();
649
650         key_bind("key", NULL, " ", "space", (SIGNAL_FUNC) key_combo);
651         key_bind("key", NULL, "^M", "return", (SIGNAL_FUNC) key_combo);
652         key_bind("key", NULL, "^J", "return", (SIGNAL_FUNC) key_combo);
653         key_bind("key", NULL, "^H", "backspace", (SIGNAL_FUNC) key_combo);
654         key_bind("key", NULL, "^?", "backspace", (SIGNAL_FUNC) key_combo);
655         key_bind("key", NULL, "^I", "tab", (SIGNAL_FUNC) key_combo);
656
657         /* meta */
658         key_bind("key", NULL, "^[", "meta", (SIGNAL_FUNC) key_combo);
659         key_bind("key", NULL, "meta-[", "meta2", (SIGNAL_FUNC) key_combo);
660         key_bind("key", NULL, "meta-O", "meta2", (SIGNAL_FUNC) key_combo);
661         key_bind("key", NULL, "meta-[O", "meta2", (SIGNAL_FUNC) key_combo);
662
663         /* arrow keys */
664         key_bind("key", NULL, "meta2-A", "up", (SIGNAL_FUNC) key_combo);
665         key_bind("key", NULL, "meta2-B", "down", (SIGNAL_FUNC) key_combo);
666         key_bind("key", NULL, "meta2-C", "right", (SIGNAL_FUNC) key_combo);
667         key_bind("key", NULL, "meta2-D", "left", (SIGNAL_FUNC) key_combo);
668
669         key_bind("key", NULL, "meta2-1~", "home", (SIGNAL_FUNC) key_combo);
670         key_bind("key", NULL, "meta2-7~", "home", (SIGNAL_FUNC) key_combo);
671         key_bind("key", NULL, "meta2-H", "home", (SIGNAL_FUNC) key_combo);
672
673         key_bind("key", NULL, "meta2-4~", "end", (SIGNAL_FUNC) key_combo);
674         key_bind("key", NULL, "meta2-8~", "end", (SIGNAL_FUNC) key_combo);
675         key_bind("key", NULL, "meta2-F", "end", (SIGNAL_FUNC) key_combo);
676
677         key_bind("key", NULL, "meta2-5~", "prior", (SIGNAL_FUNC) key_combo);
678         key_bind("key", NULL, "meta2-I", "prior", (SIGNAL_FUNC) key_combo);
679         key_bind("key", NULL, "meta2-6~", "next", (SIGNAL_FUNC) key_combo);
680         key_bind("key", NULL, "meta2-G", "next", (SIGNAL_FUNC) key_combo);
681
682         key_bind("key", NULL, "meta2-2~", "insert", (SIGNAL_FUNC) key_combo);
683         key_bind("key", NULL, "meta2-3~", "delete", (SIGNAL_FUNC) key_combo);
684
685         key_bind("key", NULL, "meta2-d", "cleft", (SIGNAL_FUNC) key_combo);
686         key_bind("key", NULL, "meta2-c", "cright", (SIGNAL_FUNC) key_combo);
687         key_bind("key", NULL, "meta2-5D", "cleft", (SIGNAL_FUNC) key_combo);
688         key_bind("key", NULL, "meta2-5C", "cright", (SIGNAL_FUNC) key_combo);
689
690         /* cursor movement */
691         key_bind("backward_character", "", "left", NULL, (SIGNAL_FUNC) key_backward_character);
692         key_bind("forward_character", "", "right", NULL, (SIGNAL_FUNC) key_forward_character);
693         key_bind("backward_word", "", "cleft", NULL, (SIGNAL_FUNC) key_backward_word);
694         key_bind("backward_word", NULL, "meta-b", NULL, (SIGNAL_FUNC) key_backward_word);
695         key_bind("forward_word", "", "cright", NULL, (SIGNAL_FUNC) key_forward_word);
696         key_bind("forward_word", NULL, "meta-f", NULL, (SIGNAL_FUNC) key_forward_word);
697         key_bind("backward_to_space", "", NULL, NULL, (SIGNAL_FUNC) key_backward_to_space);
698         key_bind("forward_to_space", "", NULL, NULL, (SIGNAL_FUNC) key_forward_to_space);
699         key_bind("beginning_of_line", "", "home", NULL, (SIGNAL_FUNC) key_beginning_of_line);
700         key_bind("beginning_of_line", NULL, "^A", NULL, (SIGNAL_FUNC) key_beginning_of_line);
701         key_bind("end_of_line", "", "end", NULL, (SIGNAL_FUNC) key_end_of_line);
702         key_bind("end_of_line", NULL, "^E", NULL, (SIGNAL_FUNC) key_end_of_line);
703
704         /* history */
705         key_bind("backward_history", "", "up", NULL, (SIGNAL_FUNC) key_backward_history);
706         key_bind("forward_history", "", "down", NULL, (SIGNAL_FUNC) key_forward_history);
707
708         /* line editing */
709         key_bind("backspace", "", "backspace", NULL, (SIGNAL_FUNC) key_backspace);
710         key_bind("delete_character", "", "delete", NULL, (SIGNAL_FUNC) key_delete_character);
711         key_bind("delete_character", NULL, "^D", NULL, (SIGNAL_FUNC) key_delete_character);
712         key_bind("delete_next_word", "meta-d", NULL, NULL, (SIGNAL_FUNC) key_delete_next_word);
713         key_bind("delete_previous_word", "meta-backspace", NULL, NULL, (SIGNAL_FUNC) key_delete_previous_word);
714         key_bind("delete_to_previous_space", "", "^W", NULL, (SIGNAL_FUNC) key_delete_to_previous_space);
715         key_bind("delete_to_next_space", "", "", NULL, (SIGNAL_FUNC) key_delete_to_next_space);
716         key_bind("erase_line", "", "^U", NULL, (SIGNAL_FUNC) key_erase_line);
717         key_bind("erase_to_beg_of_line", "", NULL, NULL, (SIGNAL_FUNC) key_erase_to_beg_of_line);
718         key_bind("erase_to_end_of_line", "", "^K", NULL, (SIGNAL_FUNC) key_erase_to_end_of_line);
719         key_bind("yank_from_cutbuffer", "", "^Y", NULL, (SIGNAL_FUNC) key_yank_from_cutbuffer);
720         key_bind("transpose_characters", "Swap current and previous character", "^T", NULL, (SIGNAL_FUNC) key_transpose_characters);
721
722         /* line transmitting */
723         key_bind("send_line", "Execute the input line", "return", NULL, (SIGNAL_FUNC) key_send_line);
724         key_bind("word_completion", "", "tab", NULL, (SIGNAL_FUNC) key_word_completion);
725         key_bind("erase_completion", "", "meta-k", NULL, (SIGNAL_FUNC) key_erase_completion);
726         key_bind("check_replaces", "Check word replaces", NULL, NULL, (SIGNAL_FUNC) key_check_replaces);
727
728         /* window managing */
729         key_bind("previous_window", "Previous window", "^P", NULL, (SIGNAL_FUNC) key_previous_window);
730         key_bind("next_window", "Next window", "^N", NULL, (SIGNAL_FUNC) key_next_window);
731         key_bind("upper_window", "Upper window", "meta-up", NULL, (SIGNAL_FUNC) key_upper_window);
732         key_bind("lower_window", "Lower window", "meta-down", NULL, (SIGNAL_FUNC) key_lower_window);
733         key_bind("left_window", "Window in left", "meta-left", NULL, (SIGNAL_FUNC) key_left_window);
734         key_bind("right_window", "Window in right", "meta-right", NULL, (SIGNAL_FUNC) key_right_window);
735         key_bind("active_window", "Go to next window with the highest activity", "meta-a", NULL, (SIGNAL_FUNC) key_active_window);
736         key_bind("next_window_item", "Next channel/query", "^X", NULL, (SIGNAL_FUNC) key_next_window_item);
737         key_bind("previous_window_item", "Previous channel/query", NULL, NULL, (SIGNAL_FUNC) key_previous_window_item);
738
739         key_bind("refresh_screen", "Redraw screen", "^L", NULL, (SIGNAL_FUNC) irssi_redraw);
740         key_bind("scroll_backward", "Previous page", "prior", NULL, (SIGNAL_FUNC) key_scroll_backward);
741         key_bind("scroll_backward", NULL, "meta-p", NULL, (SIGNAL_FUNC) key_scroll_backward);
742         key_bind("scroll_forward", "Next page", "next", NULL, (SIGNAL_FUNC) key_scroll_forward);
743         key_bind("scroll_forward", NULL, "meta-n", NULL, (SIGNAL_FUNC) key_scroll_forward);
744         key_bind("scroll_start", "Beginning of the window", "", NULL, (SIGNAL_FUNC) key_scroll_start);
745         key_bind("scroll_end", "End of the window", "", NULL, (SIGNAL_FUNC) key_scroll_end);
746
747         /* inserting special input characters to line.. */
748         key_bind("escape_char", "Escape the next keypress", NULL, NULL, (SIGNAL_FUNC) key_escape);
749         key_bind("insert_text", "Append text to line", NULL, NULL, (SIGNAL_FUNC) key_insert_text);
750
751         /* autoreplaces */
752         key_bind("multi", NULL, "return", "check_replaces;send_line", NULL);
753         key_bind("multi", NULL, "space", "check_replaces;insert_text  ", NULL);
754
755         /* moving between windows */
756         for (n = 0; changekeys[n] != '\0'; n++) {
757                 key = g_strdup_printf("meta-%c", changekeys[n]);
758                 ltoa(data, n+1);
759                 key_bind("change_window", "Change window", key, data, (SIGNAL_FUNC) key_change_window);
760                 g_free(key);
761         }
762
763         /* misc */
764         key_bind("stop_irc", "Send SIGSTOP to client", "^Z", NULL, (SIGNAL_FUNC) key_sig_stop);
765
766         key_configure_thaw();
767
768         signal_add("window changed automatic", (SIGNAL_FUNC) sig_window_auto_changed);
769         signal_add("gui entry redirect", (SIGNAL_FUNC) sig_gui_entry_redirect);
770         signal_add("gui key pressed", (SIGNAL_FUNC) sig_gui_key_pressed);
771 }
772
773 void gui_readline_deinit(void)
774 {
775         input_listen_deinit();
776
777         key_configure_freeze();
778
779         key_unbind("backward_character", (SIGNAL_FUNC) key_backward_character);
780         key_unbind("forward_character", (SIGNAL_FUNC) key_forward_character);
781         key_unbind("backward_word", (SIGNAL_FUNC) key_backward_word);
782         key_unbind("forward_word", (SIGNAL_FUNC) key_forward_word);
783         key_unbind("backward_to_space", (SIGNAL_FUNC) key_backward_to_space);
784         key_unbind("forward_to_space", (SIGNAL_FUNC) key_forward_to_space);
785         key_unbind("beginning_of_line", (SIGNAL_FUNC) key_beginning_of_line);
786         key_unbind("end_of_line", (SIGNAL_FUNC) key_end_of_line);
787
788         key_unbind("backward_history", (SIGNAL_FUNC) key_backward_history);
789         key_unbind("forward_history", (SIGNAL_FUNC) key_forward_history);
790
791         key_unbind("backspace", (SIGNAL_FUNC) key_backspace);
792         key_unbind("delete_character", (SIGNAL_FUNC) key_delete_character);
793         key_unbind("delete_next_word", (SIGNAL_FUNC) key_delete_next_word);
794         key_unbind("delete_previous_word", (SIGNAL_FUNC) key_delete_previous_word);
795         key_unbind("delete_to_next_space", (SIGNAL_FUNC) key_delete_to_next_space);
796         key_unbind("delete_to_previous_space", (SIGNAL_FUNC) key_delete_to_previous_space);
797         key_unbind("erase_line", (SIGNAL_FUNC) key_erase_line);
798         key_unbind("erase_to_beg_of_line", (SIGNAL_FUNC) key_erase_to_beg_of_line);
799         key_unbind("erase_to_end_of_line", (SIGNAL_FUNC) key_erase_to_end_of_line);
800         key_unbind("yank_from_cutbuffer", (SIGNAL_FUNC) key_yank_from_cutbuffer);
801         key_unbind("transpose_characters", (SIGNAL_FUNC) key_transpose_characters);
802
803         key_unbind("send_line", (SIGNAL_FUNC) key_send_line);
804         key_unbind("word_completion", (SIGNAL_FUNC) key_word_completion);
805         key_unbind("erase_completion", (SIGNAL_FUNC) key_erase_completion);
806         key_unbind("check_replaces", (SIGNAL_FUNC) key_check_replaces);
807
808         key_unbind("previous_window", (SIGNAL_FUNC) key_previous_window);
809         key_unbind("next_window", (SIGNAL_FUNC) key_next_window);
810         key_unbind("upper_window", (SIGNAL_FUNC) key_upper_window);
811         key_unbind("lower_window", (SIGNAL_FUNC) key_lower_window);
812         key_unbind("left_window", (SIGNAL_FUNC) key_left_window);
813         key_unbind("right_window", (SIGNAL_FUNC) key_right_window);
814         key_unbind("active_window", (SIGNAL_FUNC) key_active_window);
815         key_unbind("next_window_item", (SIGNAL_FUNC) key_next_window_item);
816         key_unbind("previous_window_item", (SIGNAL_FUNC) key_previous_window_item);
817
818         key_unbind("refresh_screen", (SIGNAL_FUNC) irssi_redraw);
819         key_unbind("scroll_backward", (SIGNAL_FUNC) key_scroll_backward);
820         key_unbind("scroll_forward", (SIGNAL_FUNC) key_scroll_forward);
821         key_unbind("scroll_start", (SIGNAL_FUNC) key_scroll_start);
822         key_unbind("scroll_end", (SIGNAL_FUNC) key_scroll_end);
823
824         key_unbind("escape_char", (SIGNAL_FUNC) key_escape);
825         key_unbind("insert_text", (SIGNAL_FUNC) key_insert_text);
826         key_unbind("change_window", (SIGNAL_FUNC) key_change_window);
827         key_unbind("stop_irc", (SIGNAL_FUNC) key_sig_stop);
828         keyboard_destroy(keyboard);
829
830         key_configure_thaw();
831
832         signal_remove("window changed automatic", (SIGNAL_FUNC) sig_window_auto_changed);
833         signal_remove("gui entry redirect", (SIGNAL_FUNC) sig_gui_entry_redirect);
834         signal_remove("gui key pressed", (SIGNAL_FUNC) sig_gui_key_pressed);
835 }