home · contact · privacy
4231da492531cf8a832f5e5bc817a8427e724682
[plomrogue] / src / client / keybindings.c
1 /* src/client/keybindings.c */
2
3 #include "keybindings.h"
4 #include <ncurses.h> /* keycode defines, cbreak(), halfdelay(), getch() */
5 #include <stddef.h> /* NULL */
6 #include <stdint.h> /* uint8_t, uint16_t, uint32_t */
7 #include <stdio.h> /* FILE, sprintf() */
8 #include "../common/try_malloc.h" /* try_malloc() */
9 #include "windows.h" /* draw_all_wins() */
10 #include "world.h" /* global world */
11 struct Command;
12
13
14 /* Return pointer to global keybindings or to keybindings for wingeometry config
15  * (c = "g") or winkeys config (c = "k") or active window's keybindings ("w").
16  */
17 static struct KeyBindingDB * char_selected_kb_db(char c);
18
19 /* If "keycode_given" equals "keycode_match", copy "keyname_match" to "keyname"
20  * and return 1; otherwise return 0.
21  */
22 static uint8_t try_keycode(uint16_t keycode_given, char * keyname,
23                            uint16_t keycode_match, char * keyname_match);
24
25
26
27 static struct KeyBindingDB * char_selected_kb_db(char c)
28 {
29     struct KeyBindingDB * kbdb;
30     kbdb = &world.kb_global;
31     if      ('g' == c)
32     {
33         kbdb = &world.kb_wingeom;
34     }
35     else if ('k' == c)
36     {
37         kbdb = &world.kb_winkeys;
38     }
39     else if ('w' == c)
40     {
41         struct Win * w = get_win_by_id(world.winDB.active);
42         kbdb = &w->kb;
43     }
44     return kbdb;
45 }
46
47
48
49 static uint8_t try_keycode(uint16_t keycode_given, char * keyname,
50                            uint16_t keycode_match, char * keyname_match)
51 {
52     if (keycode_given == keycode_match)
53     {
54         sprintf(keyname, "%s", keyname_match);
55         return 1;
56     }
57     return 0;
58 }
59
60
61
62 extern struct Command * get_command_to_keycode(struct KeyBindingDB * kbdb,
63                                                uint16_t keycode)
64 {
65     uint8_t n_kb;
66     for (n_kb = 0; n_kb < kbdb->n_of_kbs; n_kb++)
67     {
68         if (keycode == kbdb->kbs[n_kb].keycode)
69         {
70             return kbdb->kbs[n_kb].command;
71         }
72     }
73     return NULL;
74 }
75
76
77
78 extern char * get_keyname_to_keycode(uint16_t keycode)
79 {
80     char * f_name = "get_name_to_keycode()";
81     char * keyname = try_malloc(10, f_name);        /* max keyname length + 1 */
82     if (32 < keycode && keycode < 127)
83     {
84         sprintf(keyname, "%c", keycode);
85     }
86     else if (keycode >= KEY_F0 && keycode <= KEY_F(63))
87     {
88         uint16_t f = keycode - KEY_F0;
89         sprintf(keyname, "F%d", f);
90     }
91     else if (   try_keycode(keycode, keyname, 9, "TAB")
92              || try_keycode(keycode, keyname, 10, "RETURN")
93              || try_keycode(keycode, keyname, 27, "ESCAPE")
94              || try_keycode(keycode, keyname, 32, "SPACE")
95              || try_keycode(keycode, keyname, KEY_UP, "UP")
96              || try_keycode(keycode, keyname, KEY_DOWN, "DOWN")
97              || try_keycode(keycode, keyname, KEY_LEFT, "LEFT")
98              || try_keycode(keycode, keyname, KEY_RIGHT, "RIGHT")
99              || try_keycode(keycode, keyname, KEY_HOME, "HOME")
100              || try_keycode(keycode, keyname, KEY_BACKSPACE, "BACKSPACE")
101              || try_keycode(keycode, keyname, KEY_DC, "DELETE")
102              || try_keycode(keycode, keyname, KEY_IC, "INSERT")
103              || try_keycode(keycode, keyname, KEY_NPAGE, "NEXT PAGE")
104              || try_keycode(keycode, keyname, KEY_PPAGE, "PREV PAGE")
105              || try_keycode(keycode, keyname, KEY_END, "END"))
106     {
107         ;
108     }
109     else
110     {
111         sprintf(keyname, "(unknown)");
112     }
113     return keyname;
114 }
115
116
117
118 extern void mod_selected_keyb(char kb_c)
119 {
120     struct KeyBindingDB * kbdb = char_selected_kb_db(kb_c);
121     kbdb->edit = 1;
122     draw_all_wins();
123     cbreak();
124     int keycode = getch();
125     halfdelay(world.halfdelay);
126     if (keycode < 1000)
127     {
128         kbdb->kbs[kbdb->select].keycode = keycode;
129     }
130     kbdb->edit = 0;
131 }
132
133
134
135 extern void move_keyb_selection(char kb_c, char dir)
136 {
137     struct KeyBindingDB * kbdb = char_selected_kb_db(kb_c);
138     if      ('u' == dir && kbdb->select > 0)
139     {
140         kbdb->select--;
141     }
142     else if ('d' == dir && kbdb->select < kbdb->n_of_kbs - 1)
143     {
144         kbdb->select++;
145     }
146 }