Abstract binding configuration to bindings.[ch]
Patch status: needinfo
Patch by Tony Crisci
Long description:
Create files bindings.[ch] to contain functions for configuring, finding, and running bindings. Use the new function `configure_binding` for binding configuration. This function adds a binding from config parameters. Export the function `modifiers_from_str` from config_directives.h. This change is made in preparation for the new bindmouse functionality.
To apply this patch, use:
curl http://cr.i3wm.org/patch/386/raw.patch | git am
b/include/all.h
28 |
@@ -79,6 +79,7 @@ |
29 |
#include "scratchpad.h" |
30 |
#include "commands.h" |
31 |
#include "commands_parser.h" |
32 |
+#include "bindings.h" |
33 |
#include "config_directives.h" |
34 |
#include "config_parser.h" |
35 |
#include "fake_outputs.h" |
b/include/bindings.h
41 |
@@ -0,0 +1,24 @@ |
42 |
+/* |
43 |
+ * vim:ts=4:sw=4:expandtab |
44 |
+ * |
45 |
+ * i3 - an improved dynamic tiling window manager |
46 |
+ * © 2009-2014 Michael Stapelberg and contributors (see also: LICENSE) |
47 |
+ * |
48 |
+ * bindings.h: Functions for configuring, finding, and running bindings. |
49 |
+ * |
50 |
+ */ |
51 |
+#pragma once |
52 |
+ |
53 |
+/** |
54 |
+ * The name of the default mode. |
55 |
+ * |
56 |
+ */ |
57 |
+#define DEFAULT_BINDING_MODE "default" |
58 |
+ |
59 |
+/** |
60 |
+ * Adds a binding from config parameters given as strings and returns a |
61 |
+ * pointer to the binding structure. Returns NULL if the input code could not |
62 |
+ * be parsed. |
63 |
+ * |
64 |
+ */ |
65 |
+Binding *configure_binding(const char *bindtype, const char *modifiers, const char *input_code, const char *release, const char *command, const char *mode); |
b/include/config_directives.h
70 |
@@ -11,6 +11,12 @@ |
71 |
|
72 |
#include "config_parser.h" |
73 |
|
74 |
+/** |
75 |
+ * A utility function to convert a string of modifiers to the corresponding bit |
76 |
+ * mask. |
77 |
+ */ |
78 |
+uint32_t modifiers_from_str(const char *str); |
79 |
+ |
80 |
/** The beginning of the prototype for every cfg_ function. */ |
81 |
#define I3_CFG Match *current_match, struct ConfigResult *result |
82 |
|
b/src/bindings.c
88 |
@@ -0,0 +1,62 @@ |
89 |
+/* |
90 |
+ * vim:ts=4:sw=4:expandtab |
91 |
+ * |
92 |
+ * i3 - an improved dynamic tiling window manager |
93 |
+ * © 2009-2014 Michael Stapelberg and contributors (see also: LICENSE) |
94 |
+ * |
95 |
+ * bindings.c: Functions for configuring, finding and, running bindings. |
96 |
+ */ |
97 |
+#include "all.h" |
98 |
+ |
99 |
+/* |
100 |
+ * Returns the mode specified by `name` or creates a new mode and adds it to |
101 |
+ * the list of modes. |
102 |
+ * |
103 |
+ */ |
104 |
+static struct Mode *mode_from_name(const char *name) { |
105 |
+ struct Mode *mode; |
106 |
+ SLIST_FOREACH(mode, &modes, modes) { |
107 |
+ if (strcmp(mode->name, name) == 0) |
108 |
+ break; |
109 |
+ } |
110 |
+ |
111 |
+ if (mode == NULL) { |
112 |
+ mode = scalloc(sizeof(struct Mode)); |
113 |
+ mode->name = sstrdup(name); |
114 |
+ mode->bindings = scalloc(sizeof(struct bindings_head)); |
115 |
+ TAILQ_INIT(mode->bindings); |
116 |
+ SLIST_INSERT_HEAD(&modes, mode, modes); |
117 |
+ } |
118 |
+ |
119 |
+ return mode; |
120 |
+} |
121 |
+ |
122 |
+/* |
123 |
+ * Adds a binding from config parameters given as strings and returns a |
124 |
+ * pointer to the binding structure. Returns NULL if the input code could not |
125 |
+ * be parsed. |
126 |
+ * |
127 |
+ */ |
128 |
+Binding *configure_binding(const char *bindtype, const char *modifiers, const char *input_code, const char *release, const char *command, const char *modename) { |
129 |
+ Binding *new_binding = scalloc(sizeof(Binding)); |
130 |
+ DLOG("bindtype %s, modifiers %s, input code %s, release %s\n", bindtype, modifiers, input_code, release); |
131 |
+ new_binding->release = (release != NULL ? B_UPON_KEYRELEASE : B_UPON_KEYPRESS); |
132 |
+ if (strcmp(bindtype, "bindsym") == 0) { |
133 |
+ new_binding->symbol = sstrdup(input_code); |
134 |
+ } else { |
135 |
+ // TODO: strtol with proper error handling |
136 |
+ new_binding->keycode = atoi(input_code); |
137 |
+ if (new_binding->keycode == 0) { |
138 |
+ ELOG("Could not parse \"%s\" as an input code, ignoring this binding.\n", input_code); |
139 |
+ return NULL; |
140 |
+ } |
141 |
+ } |
142 |
+ new_binding->mods = modifiers_from_str(modifiers); |
143 |
+ new_binding->command = sstrdup(command); |
144 |
+ new_binding->input_type = B_KEYBOARD; |
145 |
+ |
146 |
+ struct Mode *mode = mode_from_name(modename); |
147 |
+ TAILQ_INSERT_TAIL(mode->bindings, new_binding, bindings); |
148 |
+ |
149 |
+ return new_binding; |
150 |
+} |
b/src/config_directives.c
155 |
@@ -130,7 +130,11 @@ static bool eval_boolstr(const char *str) { |
156 |
strcasecmp(str, "active") == 0); |
157 |
} |
158 |
|
159 |
-static uint32_t modifiers_from_str(const char *str) { |
160 |
+/* |
161 |
+ * A utility function to convert a string of modifiers to the corresponding bit |
162 |
+ * mask. |
163 |
+ */ |
164 |
+uint32_t modifiers_from_str(const char *str) { |
165 |
/* It might be better to use strtok() here, but the simpler strstr() should |
166 |
* do for now. */ |
167 |
uint32_t result = 0; |
168 |
@@ -167,24 +171,8 @@ CFGFUN(font, const char *font) { |
169 |
font_pattern = sstrdup(font); |
170 |
} |
171 |
|
172 |
-// TODO: refactor with mode_binding |
173 |
CFGFUN(binding, const char *bindtype, const char *modifiers, const char *key, const char *release, const char *command) { |
174 |
- Binding *new_binding = scalloc(sizeof(Binding)); |
175 |
- DLOG("bindtype %s, modifiers %s, key %s, release %s\n", bindtype, modifiers, key, release); |
176 |
- new_binding->release = (release != NULL ? B_UPON_KEYRELEASE : B_UPON_KEYPRESS); |
177 |
- if (strcmp(bindtype, "bindsym") == 0) { |
178 |
- new_binding->symbol = sstrdup(key); |
179 |
- } else { |
180 |
- // TODO: strtol with proper error handling |
181 |
- new_binding->keycode = atoi(key); |
182 |
- if (new_binding->keycode == 0) { |
183 |
- ELOG("Could not parse \"%s\" as a keycode, ignoring this binding.\n", key); |
184 |
- return; |
185 |
- } |
186 |
- } |
187 |
- new_binding->mods = modifiers_from_str(modifiers); |
188 |
- new_binding->command = sstrdup(command); |
189 |
- TAILQ_INSERT_TAIL(bindings, new_binding, bindings); |
190 |
+ configure_binding(bindtype, modifiers, key, release, command, DEFAULT_BINDING_MODE); |
191 |
} |
192 |
|
193 |
|
194 |
@@ -192,39 +180,20 @@ CFGFUN(binding, const char *bindtype, const char *modifiers, const char *key, co |
195 |
* Mode handling |
196 |
******************************************************************************/ |
197 |
|
198 |
-static struct bindings_head *current_bindings; |
199 |
+static char *current_mode; |
200 |
|
201 |
CFGFUN(mode_binding, const char *bindtype, const char *modifiers, const char *key, const char *release, const char *command) { |
202 |
- Binding *new_binding = scalloc(sizeof(Binding)); |
203 |
- DLOG("bindtype %s, modifiers %s, key %s, release %s\n", bindtype, modifiers, key, release); |
204 |
- new_binding->release = (release != NULL ? B_UPON_KEYRELEASE : B_UPON_KEYPRESS); |
205 |
- if (strcmp(bindtype, "bindsym") == 0) { |
206 |
- new_binding->symbol = sstrdup(key); |
207 |
- } else { |
208 |
- // TODO: strtol with proper error handling |
209 |
- new_binding->keycode = atoi(key); |
210 |
- if (new_binding->keycode == 0) { |
211 |
- ELOG("Could not parse \"%s\" as a keycode, ignoring this binding.\n", key); |
212 |
- return; |
213 |
- } |
214 |
- } |
215 |
- new_binding->mods = modifiers_from_str(modifiers); |
216 |
- new_binding->command = sstrdup(command); |
217 |
- TAILQ_INSERT_TAIL(current_bindings, new_binding, bindings); |
218 |
+ configure_binding(bindtype, modifiers, key, release, command, current_mode); |
219 |
} |
220 |
|
221 |
CFGFUN(enter_mode, const char *modename) { |
222 |
- if (strcasecmp(modename, "default") == 0) { |
223 |
- ELOG("You cannot use the name \"default\" for your mode\n"); |
224 |
+ if (strcasecmp(modename, DEFAULT_BINDING_MODE) == 0) { |
225 |
+ ELOG("You cannot use the name %s for your mode\n", DEFAULT_BINDING_MODE); |
226 |
exit(1); |
227 |
} |
228 |
DLOG("\t now in mode %s\n", modename); |
229 |
- struct Mode *mode = scalloc(sizeof(struct Mode)); |
230 |
- mode->name = sstrdup(modename); |
231 |
- mode->bindings = scalloc(sizeof(struct bindings_head)); |
232 |
- TAILQ_INIT(mode->bindings); |
233 |
- current_bindings = mode->bindings; |
234 |
- SLIST_INSERT_HEAD(&modes, mode, modes); |
235 |
+ FREE(current_mode); |
236 |
+ current_mode = sstrdup(modename); |
237 |
} |
238 |
|
239 |
CFGFUN(exec, const char *exectype, const char *no_startup_id, const char *command) { |