Abstract binding functions to bindings.h
Patch status: superseded
Patch by Tony Crisci
Long description:
Adds the following new functions: * `initialize_bindings` * `configure_binding` * `run_binding` Abstracts these existing functions with minor changes: * Renames `get_binding` to `get_keyboard_binding` and switches the parameters `keycode` and `release`. * Renames `grab_keys`to `grab_bound_keys`. * Renames `translate_keysyms` to `translate_binding_keysyms`. * Renames `switch_mode` to `switch_binding_mode`. * Renames `check_for_duplicate_bindings` to `validate_bindings` Additionally makes the modes structure and list of modes private to bindings.c, and makes `modifiers_from_str` public.
To apply this patch, use:
curl http://cr.i3wm.org/patch/382/raw.patch | git am
b/include/all.h
44 |
@@ -79,6 +79,7 @@ |
45 |
#include "scratchpad.h" |
46 |
#include "commands.h" |
47 |
#include "commands_parser.h" |
48 |
+#include "bindings.h" |
49 |
#include "config_directives.h" |
50 |
#include "config_parser.h" |
51 |
#include "fake_outputs.h" |
b/include/bindings.h
57 |
@@ -0,0 +1,74 @@ |
58 |
+/* |
59 |
+ * vim:ts=4:sw=4:expandtab |
60 |
+ * |
61 |
+ * i3 - an improved dynamic tiling window manager |
62 |
+ * © 2009-2014 Michael Stapelberg and contributors (see also: LICENSE) |
63 |
+ * |
64 |
+ * bindings.h: Functions for configuring, finding, and running bindings. |
65 |
+ * |
66 |
+ */ |
67 |
+#pragma once |
68 |
+ |
69 |
+/** |
70 |
+ * The name of the default mode. |
71 |
+ * |
72 |
+ */ |
73 |
+#define DEFAULT_BINDING_MODE "default" |
74 |
+ |
75 |
+/** |
76 |
+ * Clears any existing modes and bindings and initializes the data structures. |
77 |
+ * To be called before parsing or reloading the config. |
78 |
+ * |
79 |
+ */ |
80 |
+void initialize_bindings(void); |
81 |
+ |
82 |
+/** |
83 |
+ * Adds a binding from config parameters given as strings and returns a |
84 |
+ * pointer to the binding structure. Returns NULL if the input code could not |
85 |
+ * be parsed. |
86 |
+ * |
87 |
+ */ |
88 |
+Binding *configure_binding(const char *bindtype, const char *modifiers, const char *input_code, const char *release, const char *command, const char *mode); |
89 |
+ |
90 |
+/** |
91 |
+ * Gets the keyboard binding with the specified modifiers and keycode. Returns |
92 |
+ * NULL if no such binding exists. |
93 |
+ * |
94 |
+ */ |
95 |
+Binding *get_keyboard_binding(uint16_t modifiers, uint32_t keycode, bool release); |
96 |
+ |
97 |
+/** |
98 |
+ * Runs the specified binding and returns the `CommandResult`. |
99 |
+ * |
100 |
+ */ |
101 |
+struct CommandResult *run_binding(Binding *binding); |
102 |
+ |
103 |
+ |
104 |
+/** |
105 |
+ * Translates keysymbols to keycodes for all bindings which use keysyms. |
106 |
+ * |
107 |
+ */ |
108 |
+void translate_binding_keysyms(void); |
109 |
+ |
110 |
+/** |
111 |
+ * Grabs the keys for which bindings exist. The server will send us keypress |
112 |
+ * events for those keycodes. |
113 |
+ * |
114 |
+ */ |
115 |
+void grab_bound_keys(xcb_connection_t *conn, bool bind_mode_switch); |
116 |
+ |
117 |
+/** |
118 |
+ * Changes the set of currently active bindings to those that were configured |
119 |
+ * in the specified mode if the mode exists. |
120 |
+ * |
121 |
+ */ |
122 |
+void switch_binding_mode(const char *new_mode); |
123 |
+ |
124 |
+/** |
125 |
+ * Checks for duplicate bindings (the same keycode, keysym, or button is |
126 |
+ * configured more than once). If a duplicate binding is found, a message is |
127 |
+ * printed to stderr and the has_errors variable is set to true, which will |
128 |
+ * start i3-nagbar. |
129 |
+ * |
130 |
+ */ |
131 |
+void validate_bindings(struct context *context); |
b/include/config.h
136 |
@@ -21,7 +21,6 @@ typedef struct Config Config; |
137 |
typedef struct Barconfig Barconfig; |
138 |
extern char *current_configpath; |
139 |
extern Config config; |
140 |
-extern SLIST_HEAD(modes_head, Mode) modes; |
141 |
extern TAILQ_HEAD(barconfig_head, Barconfig) barconfigs; |
142 |
|
143 |
/** |
144 |
@@ -70,19 +69,6 @@ struct Variable { |
145 |
}; |
146 |
|
147 |
/** |
148 |
- * The configuration file can contain multiple sets of bindings. Apart from the |
149 |
- * default set (name == "default"), you can specify other sets and change the |
150 |
- * currently active set of bindings by using the "mode <name>" command. |
151 |
- * |
152 |
- */ |
153 |
-struct Mode { |
154 |
- char *name; |
155 |
- struct bindings_head *bindings; |
156 |
- |
157 |
- SLIST_ENTRY(Mode) modes; |
158 |
-}; |
159 |
- |
160 |
-/** |
161 |
* Holds part of the configuration (the part which is not already in dedicated |
162 |
* structures in include/data.h). |
163 |
* |
164 |
@@ -340,13 +326,6 @@ void switch_mode(const char *new_mode); |
165 |
*/void update_barconfig(); |
166 |
|
167 |
/** |
168 |
- * Returns a pointer to the Binding with the specified modifiers and keycode |
169 |
- * or NULL if no such binding exists. |
170 |
- * |
171 |
- */ |
172 |
-Binding *get_binding(uint16_t modifiers, bool key_release, xcb_keycode_t keycode); |
173 |
- |
174 |
-/** |
175 |
* Kills the configerror i3-nagbar process, if any. |
176 |
* |
177 |
* Called when reloading/restarting. |
b/include/config_directives.h
182 |
@@ -11,6 +11,12 @@ |
183 |
|
184 |
#include "config_parser.h" |
185 |
|
186 |
+/** |
187 |
+ * A utility function to convert a string of modifiers to the corresponding bit |
188 |
+ * mask. |
189 |
+ */ |
190 |
+uint32_t modifiers_from_str(const char *str); |
191 |
+ |
192 |
/** The beginning of the prototype for every cfg_ function. */ |
193 |
#define I3_CFG Match *current_match, struct ConfigResult *result |
194 |
|
b/src/bindings.c
200 |
@@ -0,0 +1,364 @@ |
201 |
+/* |
202 |
+ * vim:ts=4:sw=4:expandtab |
203 |
+ * |
204 |
+ * i3 - an improved dynamic tiling window manager |
205 |
+ * © 2009-2014 Michael Stapelberg and contributors (see also: LICENSE) |
206 |
+ * |
207 |
+ * binding.c: Functions for configuring, finding and, running bindings. |
208 |
+ */ |
209 |
+#include "all.h" |
210 |
+ |
211 |
+/* We need Xlib for XStringToKeysym */ |
212 |
+#include <X11/Xlib.h> |
213 |
+ |
214 |
+/* |
215 |
+ * The configuration file can contain multiple sets of bindings. Apart from the |
216 |
+ * default set, you can specify other sets and change the currently active set |
217 |
+ * of bindings by using the "mode <name>" command. |
218 |
+ * |
219 |
+ */ |
220 |
+static SLIST_HEAD(modes_head, Mode) modes; |
221 |
+ |
222 |
+struct Mode { |
223 |
+ char *name; |
224 |
+ struct bindings_head *bindings; |
225 |
+ |
226 |
+ SLIST_ENTRY(Mode) modes; |
227 |
+}; |
228 |
+ |
229 |
+/* |
230 |
+ * Returns the mode specified by `name` or creates a new mode and adds it to |
231 |
+ * the list of modes. |
232 |
+ */ |
233 |
+static struct Mode *mode_from_name(const char *name) { |
234 |
+ struct Mode *mode; |
235 |
+ SLIST_FOREACH(mode, &modes, modes) { |
236 |
+ if (strcmp(mode->name, name) == 0) |
237 |
+ break; |
238 |
+ } |
239 |
+ |
240 |
+ if (mode == NULL) { |
241 |
+ mode = scalloc(sizeof(struct Mode)); |
242 |
+ mode->name = sstrdup(name); |
243 |
+ mode->bindings = scalloc(sizeof(struct bindings_head)); |
244 |
+ TAILQ_INIT(mode->bindings); |
245 |
+ SLIST_INSERT_HEAD(&modes, mode, modes); |
246 |
+ } |
247 |
+ |
248 |
+ return mode; |
249 |
+} |
250 |
+ |
251 |
+/* |
252 |
+ * Destroys all bindings and all modes. |
253 |
+ */ |
254 |
+static void clear_bindings(void) { |
255 |
+ struct Mode *mode; |
256 |
+ Binding *bind; |
257 |
+ while (!SLIST_EMPTY(&modes)) { |
258 |
+ mode = SLIST_FIRST(&modes); |
259 |
+ FREE(mode->name); |
260 |
+ |
261 |
+ /* Clear the old binding list */ |
262 |
+ bindings = mode->bindings; |
263 |
+ while (!TAILQ_EMPTY(bindings)) { |
264 |
+ bind = TAILQ_FIRST(bindings); |
265 |
+ TAILQ_REMOVE(bindings, bind, bindings); |
266 |
+ FREE(bind->translated_to); |
267 |
+ FREE(bind->command); |
268 |
+ FREE(bind); |
269 |
+ } |
270 |
+ FREE(bindings); |
271 |
+ SLIST_REMOVE(&modes, mode, Mode, modes); |
272 |
+ } |
273 |
+} |
274 |
+ |
275 |
+/* |
276 |
+ * Clears any existing modes and bindings and initializes the data structures. |
277 |
+ * To be called before parsing or reloading the config. |
278 |
+ * |
279 |
+ */ |
280 |
+void initialize_bindings() { |
281 |
+ if (bindings != NULL) |
282 |
+ clear_bindings(); |
283 |
+ |
284 |
+ SLIST_INIT(&modes); |
285 |
+ struct Mode *default_mode = mode_from_name(DEFAULT_BINDING_MODE); |
286 |
+ bindings = default_mode->bindings; |
287 |
+} |
288 |
+ |
289 |
+/* |
290 |
+ * Adds a binding from config parameters given as strings and returns a |
291 |
+ * pointer to the binding structure. Returns NULL if the input code could not |
292 |
+ * be parsed. |
293 |
+ * |
294 |
+ */ |
295 |
+Binding *configure_binding(const char *bindtype, const char *modifiers, const char *input_code, const char *release, const char *command, const char *modename) { |
296 |
+ Binding *new_binding = scalloc(sizeof(Binding)); |
297 |
+ DLOG("bindtype %s, modifiers %s, input code %s, release %s\n", bindtype, modifiers, input_code, release); |
298 |
+ new_binding->release = (release != NULL ? B_UPON_KEYRELEASE : B_UPON_KEYPRESS); |
299 |
+ if (strcmp(bindtype, "bindsym") == 0) { |
300 |
+ new_binding->symbol = sstrdup(input_code); |
301 |
+ } else { |
302 |
+ // TODO: strtol with proper error handling |
303 |
+ new_binding->keycode = atoi(input_code); |
304 |
+ if (new_binding->keycode == 0) { |
305 |
+ ELOG("Could not parse \"%s\" as an input code, ignoring this binding.\n", input_code); |
306 |
+ return NULL; |
307 |
+ } |
308 |
+ } |
309 |
+ new_binding->mods = modifiers_from_str(modifiers); |
310 |
+ new_binding->command = sstrdup(command); |
311 |
+ new_binding->input_type = B_KEYBOARD; |
312 |
+ |
313 |
+ struct Mode *mode = mode_from_name(modename); |
314 |
+ TAILQ_INSERT_TAIL(mode->bindings, new_binding, bindings); |
315 |
+ |
316 |
+ return new_binding; |
317 |
+} |
318 |
+ |
319 |
+/* |
320 |
+ * Gets the keyboard binding with the specified modifiers and keycode. Returns |
321 |
+ * NULL if no such binding exists. |
322 |
+ * |
323 |
+ */ |
324 |
+Binding *get_keyboard_binding(uint16_t modifiers, uint32_t keycode, bool key_release) { |
325 |
+ Binding *bind; |
326 |
+ |
327 |
+ if (!key_release) { |
328 |
+ /* On a KeyPress event, we first reset all |
329 |
+ * B_UPON_KEYRELEASE_IGNORE_MODS bindings back to B_UPON_KEYRELEASE */ |
330 |
+ TAILQ_FOREACH(bind, bindings, bindings) { |
331 |
+ if (bind->release == B_UPON_KEYRELEASE_IGNORE_MODS) |
332 |
+ bind->release = B_UPON_KEYRELEASE; |
333 |
+ } |
334 |
+ } |
335 |
+ |
336 |
+ TAILQ_FOREACH(bind, bindings, bindings) { |
337 |
+ /* First compare the modifiers (unless this is a |
338 |
+ * B_UPON_KEYRELEASE_IGNORE_MODS binding and this is a KeyRelease |
339 |
+ * event) */ |
340 |
+ if (bind->mods != modifiers && |
341 |
+ (bind->release != B_UPON_KEYRELEASE_IGNORE_MODS || |
342 |
+ !key_release)) |
343 |
+ continue; |
344 |
+ |
345 |
+ /* If a symbol was specified by the user, we need to look in |
346 |
+ * the array of translated keycodes for the event’s keycode */ |
347 |
+ if (bind->symbol != NULL) { |
348 |
+ if (memmem(bind->translated_to, |
349 |
+ bind->number_keycodes * sizeof(xcb_keycode_t), |
350 |
+ &keycode, sizeof(xcb_keycode_t)) == NULL) |
351 |
+ continue; |
352 |
+ } else { |
353 |
+ /* This case is easier: The user specified a keycode */ |
354 |
+ if (bind->keycode != keycode) |
355 |
+ continue; |
356 |
+ } |
357 |
+ |
358 |
+ /* If this keybinding is a KeyRelease binding, it matches the key which |
359 |
+ * the user pressed. We therefore mark it as |
360 |
+ * B_UPON_KEYRELEASE_IGNORE_MODS for later, so that the user can |
361 |
+ * release the modifiers before the actual key and the KeyRelease will |
362 |
+ * still be matched. */ |
363 |
+ if (bind->release == B_UPON_KEYRELEASE && !key_release) |
364 |
+ bind->release = B_UPON_KEYRELEASE_IGNORE_MODS; |
365 |
+ |
366 |
+ /* Check if the binding is for a KeyPress or a KeyRelease event */ |
367 |
+ if ((bind->release == B_UPON_KEYPRESS && key_release) || |
368 |
+ (bind->release >= B_UPON_KEYRELEASE && !key_release)) |
369 |
+ continue; |
370 |
+ |
371 |
+ break; |
372 |
+ } |
373 |
+ |
374 |
+ return (bind == TAILQ_END(bindings) ? NULL : bind); |
375 |
+} |
376 |
+ |
377 |
+/* |
378 |
+ * Runs the specified binding and returns the `CommandResult`. |
379 |
+ * |
380 |
+ */ |
381 |
+struct CommandResult *run_binding(Binding *binding) { |
382 |
+ char *command_copy = sstrdup(binding->command); |
383 |
+ struct CommandResult *command_output = parse_command(command_copy); |
384 |
+ free(command_copy); |
385 |
+ |
386 |
+ if (command_output->needs_tree_render) |
387 |
+ tree_render(); |
388 |
+ |
389 |
+ return command_output; |
390 |
+} |
391 |
+ |
392 |
+ |
393 |
+static void grab_keycode_for_binding(xcb_connection_t *conn, Binding *bind, uint32_t keycode) { |
394 |
+ DLOG("Grabbing %d with modifiers %d (with mod_mask_lock %d)\n", keycode, bind->mods, bind->mods | XCB_MOD_MASK_LOCK); |
395 |
+ /* Grab the key in all combinations */ |
396 |
+ #define GRAB_KEY(modifier) \ |
397 |
+ do { \ |
398 |
+ xcb_grab_key(conn, 0, root, modifier, keycode, \ |
399 |
+ XCB_GRAB_MODE_SYNC, XCB_GRAB_MODE_ASYNC); \ |
400 |
+ } while (0) |
401 |
+ int mods = bind->mods; |
402 |
+ if ((bind->mods & BIND_MODE_SWITCH) != 0) { |
403 |
+ mods &= ~BIND_MODE_SWITCH; |
404 |
+ if (mods == 0) |
405 |
+ mods = XCB_MOD_MASK_ANY; |
406 |
+ } |
407 |
+ GRAB_KEY(mods); |
408 |
+ GRAB_KEY(mods | xcb_numlock_mask); |
409 |
+ GRAB_KEY(mods | XCB_MOD_MASK_LOCK); |
410 |
+ GRAB_KEY(mods | xcb_numlock_mask | XCB_MOD_MASK_LOCK); |
411 |
+} |
412 |
+ |
413 |
+/* |
414 |
+ * Translates keysymbols to keycodes for all bindings which use keysyms. |
415 |
+ * |
416 |
+ */ |
417 |
+void translate_binding_keysyms(void) { |
418 |
+ Binding *bind; |
419 |
+ xcb_keysym_t keysym; |
420 |
+ int col; |
421 |
+ xcb_keycode_t i, |
422 |
+ min_keycode = xcb_get_setup(conn)->min_keycode, |
423 |
+ max_keycode = xcb_get_setup(conn)->max_keycode; |
424 |
+ |
425 |
+ TAILQ_FOREACH(bind, bindings, bindings) { |
426 |
+ if (bind->keycode > 0) |
427 |
+ continue; |
428 |
+ |
429 |
+ /* We need to translate the symbol to a keycode */ |
430 |
+ keysym = XStringToKeysym(bind->symbol); |
431 |
+ if (keysym == NoSymbol) { |
432 |
+ ELOG("Could not translate string to key symbol: \"%s\"\n", |
433 |
+ bind->symbol); |
434 |
+ continue; |
435 |
+ } |
436 |
+ |
437 |
+ /* Base column we use for looking up key symbols. We always consider |
438 |
+ * the base column and the corresponding shift column, so without |
439 |
+ * mode_switch, we look in 0 and 1, with mode_switch we look in 2 and |
440 |
+ * 3. */ |
441 |
+ col = (bind->mods & BIND_MODE_SWITCH ? 2 : 0); |
442 |
+ |
443 |
+ FREE(bind->translated_to); |
444 |
+ bind->number_keycodes = 0; |
445 |
+ |
446 |
+ for (i = min_keycode; i && i <= max_keycode; i++) { |
447 |
+ if ((xcb_key_symbols_get_keysym(keysyms, i, col) != keysym) && |
448 |
+ (xcb_key_symbols_get_keysym(keysyms, i, col+1) != keysym)) |
449 |
+ continue; |
450 |
+ bind->number_keycodes++; |
451 |
+ bind->translated_to = srealloc(bind->translated_to, |
452 |
+ (sizeof(xcb_keycode_t) * |
453 |
+ bind->number_keycodes)); |
454 |
+ bind->translated_to[bind->number_keycodes-1] = i; |
455 |
+ } |
456 |
+ |
457 |
+ DLOG("Translated symbol \"%s\" to %d keycode\n", bind->symbol, |
458 |
+ bind->number_keycodes); |
459 |
+ } |
460 |
+} |
461 |
+ |
462 |
+/* |
463 |
+ * Grabs the keys for which bindings exist. The server will send us keypress |
464 |
+ * events for those keycodes. |
465 |
+ * |
466 |
+ */ |
467 |
+void grab_bound_keys(xcb_connection_t *conn, bool bind_mode_switch) { |
468 |
+ Binding *bind; |
469 |
+ TAILQ_FOREACH(bind, bindings, bindings) { |
470 |
+ if ((bind_mode_switch && (bind->mods & BIND_MODE_SWITCH) == 0) || |
471 |
+ (!bind_mode_switch && (bind->mods & BIND_MODE_SWITCH) != 0)) |
472 |
+ continue; |
473 |
+ |
474 |
+ /* The easy case: the user specified a keycode directly. */ |
475 |
+ if (bind->keycode > 0) { |
476 |
+ grab_keycode_for_binding(conn, bind, bind->keycode); |
477 |
+ continue; |
478 |
+ } |
479 |
+ |
480 |
+ xcb_keycode_t *walk = bind->translated_to; |
481 |
+ for (uint32_t i = 0; i < bind->number_keycodes; i++) |
482 |
+ grab_keycode_for_binding(conn, bind, *walk++); |
483 |
+ } |
484 |
+} |
485 |
+ |
486 |
+/* |
487 |
+ * Changes the set of currently active bindings to those that were configured |
488 |
+ * in the specified mode if the mode exists. |
489 |
+ * |
490 |
+ */ |
491 |
+void switch_binding_mode(const char *new_mode) { |
492 |
+ struct Mode *mode; |
493 |
+ |
494 |
+ LOG("Switching to mode %s\n", new_mode); |
495 |
+ |
496 |
+ SLIST_FOREACH(mode, &modes, modes) { |
497 |
+ if (strcasecmp(mode->name, new_mode) != 0) |
498 |
+ continue; |
499 |
+ |
500 |
+ ungrab_all_keys(conn); |
501 |
+ bindings = mode->bindings; |
502 |
+ translate_binding_keysyms(); |
503 |
+ grab_bound_keys(conn, false); |
504 |
+ |
505 |
+ char *event_msg; |
506 |
+ sasprintf(&event_msg, "{\"change\":\"%s\"}", mode->name); |
507 |
+ |
508 |
+ ipc_send_event("mode", I3_IPC_EVENT_MODE, event_msg); |
509 |
+ FREE(event_msg); |
510 |
+ |
511 |
+ return; |
512 |
+ } |
513 |
+ |
514 |
+ ELOG("ERROR: Mode not found\n"); |
515 |
+} |
516 |
+ |
517 |
+/* |
518 |
+ * Checks for duplicate bindings (the same keycode, keysym, or button is |
519 |
+ * configured more than once). If a duplicate binding is found, a message is |
520 |
+ * printed to stderr and the has_errors variable is set to true, which will |
521 |
+ * start i3-nagbar. |
522 |
+ * |
523 |
+ */ |
524 |
+void validate_bindings(struct context *context) { |
525 |
+ Binding *bind, *current; |
526 |
+ TAILQ_FOREACH(current, bindings, bindings) { |
527 |
+ TAILQ_FOREACH(bind, bindings, bindings) { |
528 |
+ /* Abort when we reach the current keybinding, only check the |
529 |
+ * bindings before */ |
530 |
+ if (bind == current) |
531 |
+ break; |
532 |
+ |
533 |
+ /* Check if one is using keysym while the other is using bindsym. |
534 |
+ * If so, skip. */ |
535 |
+ /* XXX: It should be checked at a later place (when translating the |
536 |
+ * keysym to keycodes) if there are any duplicates */ |
537 |
+ if ((bind->symbol == NULL && current->symbol != NULL) || |
538 |
+ (bind->symbol != NULL && current->symbol == NULL)) |
539 |
+ continue; |
540 |
+ |
541 |
+ /* If bind is NULL, current has to be NULL, too (see above). |
542 |
+ * If the keycodes differ, it can't be a duplicate. */ |
543 |
+ if (bind->symbol != NULL && |
544 |
+ strcasecmp(bind->symbol, current->symbol) != 0) |
545 |
+ continue; |
546 |
+ |
547 |
+ /* Check if the keycodes or modifiers are different. If so, they |
548 |
+ * can't be duplicate */ |
549 |
+ if (bind->keycode != current->keycode || |
550 |
+ bind->mods != current->mods || |
551 |
+ bind->release != current->release) |
552 |
+ continue; |
553 |
+ |
554 |
+ context->has_errors = true; |
555 |
+ if (current->keycode != 0) { |
556 |
+ ELOG("Duplicate keybinding in config file:\n modmask %d with keycode %d, command \"%s\"\n", |
557 |
+ current->mods, current->keycode, current->command); |
558 |
+ } else { |
559 |
+ ELOG("Duplicate keybinding in config file:\n modmask %d with keysym %s, command \"%s\"\n", |
560 |
+ current->mods, current->symbol, current->command); |
561 |
+ } |
562 |
+ } |
563 |
+ } |
564 |
+} |
b/src/commands.c
569 |
@@ -1057,7 +1057,7 @@ void cmd_unmark(I3_CMD, char *mark) { |
570 |
*/ |
571 |
void cmd_mode(I3_CMD, char *mode) { |
572 |
DLOG("mode=%s\n", mode); |
573 |
- switch_mode(mode); |
574 |
+ switch_binding_mode(mode); |
575 |
|
576 |
// XXX: default reply for now, make this a better reply |
577 |
ysuccess(true); |
b/src/config.c
582 |
@@ -17,7 +17,6 @@ |
583 |
|
584 |
char *current_configpath = NULL; |
585 |
Config config; |
586 |
-struct modes_head modes; |
587 |
struct barconfig_head barconfigs = TAILQ_HEAD_INITIALIZER(barconfigs); |
588 |
|
589 |
/** |
590 |
@@ -30,186 +29,6 @@ void ungrab_all_keys(xcb_connection_t *conn) { |
591 |
xcb_ungrab_key(conn, XCB_GRAB_ANY, root, XCB_BUTTON_MASK_ANY); |
592 |
} |
593 |
|
594 |
-static void grab_keycode_for_binding(xcb_connection_t *conn, Binding *bind, uint32_t keycode) { |
595 |
- DLOG("Grabbing %d with modifiers %d (with mod_mask_lock %d)\n", keycode, bind->mods, bind->mods | XCB_MOD_MASK_LOCK); |
596 |
- /* Grab the key in all combinations */ |
597 |
- #define GRAB_KEY(modifier) \ |
598 |
- do { \ |
599 |
- xcb_grab_key(conn, 0, root, modifier, keycode, \ |
600 |
- XCB_GRAB_MODE_SYNC, XCB_GRAB_MODE_ASYNC); \ |
601 |
- } while (0) |
602 |
- int mods = bind->mods; |
603 |
- if ((bind->mods & BIND_MODE_SWITCH) != 0) { |
604 |
- mods &= ~BIND_MODE_SWITCH; |
605 |
- if (mods == 0) |
606 |
- mods = XCB_MOD_MASK_ANY; |
607 |
- } |
608 |
- GRAB_KEY(mods); |
609 |
- GRAB_KEY(mods | xcb_numlock_mask); |
610 |
- GRAB_KEY(mods | XCB_MOD_MASK_LOCK); |
611 |
- GRAB_KEY(mods | xcb_numlock_mask | XCB_MOD_MASK_LOCK); |
612 |
-} |
613 |
- |
614 |
-/* |
615 |
- * Returns a pointer to the Binding with the specified modifiers and keycode |
616 |
- * or NULL if no such binding exists. |
617 |
- * |
618 |
- */ |
619 |
-Binding *get_binding(uint16_t modifiers, bool key_release, xcb_keycode_t keycode) { |
620 |
- Binding *bind; |
621 |
- |
622 |
- if (!key_release) { |
623 |
- /* On a KeyPress event, we first reset all |
624 |
- * B_UPON_KEYRELEASE_IGNORE_MODS bindings back to B_UPON_KEYRELEASE */ |
625 |
- TAILQ_FOREACH(bind, bindings, bindings) { |
626 |
- if (bind->release == B_UPON_KEYRELEASE_IGNORE_MODS) |
627 |
- bind->release = B_UPON_KEYRELEASE; |
628 |
- } |
629 |
- } |
630 |
- |
631 |
- TAILQ_FOREACH(bind, bindings, bindings) { |
632 |
- /* First compare the modifiers (unless this is a |
633 |
- * B_UPON_KEYRELEASE_IGNORE_MODS binding and this is a KeyRelease |
634 |
- * event) */ |
635 |
- if (bind->mods != modifiers && |
636 |
- (bind->release != B_UPON_KEYRELEASE_IGNORE_MODS || |
637 |
- !key_release)) |
638 |
- continue; |
639 |
- |
640 |
- /* If a symbol was specified by the user, we need to look in |
641 |
- * the array of translated keycodes for the event’s keycode */ |
642 |
- if (bind->symbol != NULL) { |
643 |
- if (memmem(bind->translated_to, |
644 |
- bind->number_keycodes * sizeof(xcb_keycode_t), |
645 |
- &keycode, sizeof(xcb_keycode_t)) == NULL) |
646 |
- continue; |
647 |
- } else { |
648 |
- /* This case is easier: The user specified a keycode */ |
649 |
- if (bind->keycode != keycode) |
650 |
- continue; |
651 |
- } |
652 |
- |
653 |
- /* If this keybinding is a KeyRelease binding, it matches the key which |
654 |
- * the user pressed. We therefore mark it as |
655 |
- * B_UPON_KEYRELEASE_IGNORE_MODS for later, so that the user can |
656 |
- * release the modifiers before the actual key and the KeyRelease will |
657 |
- * still be matched. */ |
658 |
- if (bind->release == B_UPON_KEYRELEASE && !key_release) |
659 |
- bind->release = B_UPON_KEYRELEASE_IGNORE_MODS; |
660 |
- |
661 |
- /* Check if the binding is for a KeyPress or a KeyRelease event */ |
662 |
- if ((bind->release == B_UPON_KEYPRESS && key_release) || |
663 |
- (bind->release >= B_UPON_KEYRELEASE && !key_release)) |
664 |
- continue; |
665 |
- |
666 |
- break; |
667 |
- } |
668 |
- |
669 |
- return (bind == TAILQ_END(bindings) ? NULL : bind); |
670 |
-} |
671 |
- |
672 |
-/* |
673 |
- * Translates keysymbols to keycodes for all bindings which use keysyms. |
674 |
- * |
675 |
- */ |
676 |
-void translate_keysyms(void) { |
677 |
- Binding *bind; |
678 |
- xcb_keysym_t keysym; |
679 |
- int col; |
680 |
- xcb_keycode_t i, |
681 |
- min_keycode = xcb_get_setup(conn)->min_keycode, |
682 |
- max_keycode = xcb_get_setup(conn)->max_keycode; |
683 |
- |
684 |
- TAILQ_FOREACH(bind, bindings, bindings) { |
685 |
- if (bind->keycode > 0) |
686 |
- continue; |
687 |
- |
688 |
- /* We need to translate the symbol to a keycode */ |
689 |
- keysym = XStringToKeysym(bind->symbol); |
690 |
- if (keysym == NoSymbol) { |
691 |
- ELOG("Could not translate string to key symbol: \"%s\"\n", |
692 |
- bind->symbol); |
693 |
- continue; |
694 |
- } |
695 |
- |
696 |
- /* Base column we use for looking up key symbols. We always consider |
697 |
- * the base column and the corresponding shift column, so without |
698 |
- * mode_switch, we look in 0 and 1, with mode_switch we look in 2 and |
699 |
- * 3. */ |
700 |
- col = (bind->mods & BIND_MODE_SWITCH ? 2 : 0); |
701 |
- |
702 |
- FREE(bind->translated_to); |
703 |
- bind->number_keycodes = 0; |
704 |
- |
705 |
- for (i = min_keycode; i && i <= max_keycode; i++) { |
706 |
- if ((xcb_key_symbols_get_keysym(keysyms, i, col) != keysym) && |
707 |
- (xcb_key_symbols_get_keysym(keysyms, i, col+1) != keysym)) |
708 |
- continue; |
709 |
- bind->number_keycodes++; |
710 |
- bind->translated_to = srealloc(bind->translated_to, |
711 |
- (sizeof(xcb_keycode_t) * |
712 |
- bind->number_keycodes)); |
713 |
- bind->translated_to[bind->number_keycodes-1] = i; |
714 |
- } |
715 |
- |
716 |
- DLOG("Translated symbol \"%s\" to %d keycode\n", bind->symbol, |
717 |
- bind->number_keycodes); |
718 |
- } |
719 |
-} |
720 |
- |
721 |
-/* |
722 |
- * Grab the bound keys (tell X to send us keypress events for those keycodes) |
723 |
- * |
724 |
- */ |
725 |
-void grab_all_keys(xcb_connection_t *conn, bool bind_mode_switch) { |
726 |
- Binding *bind; |
727 |
- TAILQ_FOREACH(bind, bindings, bindings) { |
728 |
- if ((bind_mode_switch && (bind->mods & BIND_MODE_SWITCH) == 0) || |
729 |
- (!bind_mode_switch && (bind->mods & BIND_MODE_SWITCH) != 0)) |
730 |
- continue; |
731 |
- |
732 |
- /* The easy case: the user specified a keycode directly. */ |
733 |
- if (bind->keycode > 0) { |
734 |
- grab_keycode_for_binding(conn, bind, bind->keycode); |
735 |
- continue; |
736 |
- } |
737 |
- |
738 |
- xcb_keycode_t *walk = bind->translated_to; |
739 |
- for (uint32_t i = 0; i < bind->number_keycodes; i++) |
740 |
- grab_keycode_for_binding(conn, bind, *walk++); |
741 |
- } |
742 |
-} |
743 |
- |
744 |
-/* |
745 |
- * Switches the key bindings to the given mode, if the mode exists |
746 |
- * |
747 |
- */ |
748 |
-void switch_mode(const char *new_mode) { |
749 |
- struct Mode *mode; |
750 |
- |
751 |
- LOG("Switching to mode %s\n", new_mode); |
752 |
- |
753 |
- SLIST_FOREACH(mode, &modes, modes) { |
754 |
- if (strcasecmp(mode->name, new_mode) != 0) |
755 |
- continue; |
756 |
- |
757 |
- ungrab_all_keys(conn); |
758 |
- bindings = mode->bindings; |
759 |
- translate_keysyms(); |
760 |
- grab_all_keys(conn, false); |
761 |
- |
762 |
- char *event_msg; |
763 |
- sasprintf(&event_msg, "{\"change\":\"%s\"}", mode->name); |
764 |
- |
765 |
- ipc_send_event("mode", I3_IPC_EVENT_MODE, event_msg); |
766 |
- FREE(event_msg); |
767 |
- |
768 |
- return; |
769 |
- } |
770 |
- |
771 |
- ELOG("ERROR: Mode not found\n"); |
772 |
-} |
773 |
- |
774 |
/* |
775 |
* Sends the current bar configuration as an event to all barconfig_update listeners. |
776 |
* This update mechnism currently only includes the hidden_state and the mode in the config. |
777 |
@@ -344,25 +163,6 @@ void load_configuration(xcb_connection_t *conn, const char *override_configpath, |
778 |
/* First ungrab the keys */ |
779 |
ungrab_all_keys(conn); |
780 |
|
781 |
- struct Mode *mode; |
782 |
- Binding *bind; |
783 |
- while (!SLIST_EMPTY(&modes)) { |
784 |
- mode = SLIST_FIRST(&modes); |
785 |
- FREE(mode->name); |
786 |
- |
787 |
- /* Clear the old binding list */ |
788 |
- bindings = mode->bindings; |
789 |
- while (!TAILQ_EMPTY(bindings)) { |
790 |
- bind = TAILQ_FIRST(bindings); |
791 |
- TAILQ_REMOVE(bindings, bind, bindings); |
792 |
- FREE(bind->translated_to); |
793 |
- FREE(bind->command); |
794 |
- FREE(bind); |
795 |
- } |
796 |
- FREE(bindings); |
797 |
- SLIST_REMOVE(&modes, mode, Mode, modes); |
798 |
- } |
799 |
- |
800 |
struct Assignment *assign; |
801 |
while (!TAILQ_EMPTY(&assignments)) { |
802 |
assign = TAILQ_FIRST(&assignments); |
803 |
@@ -424,15 +224,7 @@ void load_configuration(xcb_connection_t *conn, const char *override_configpath, |
804 |
free_font(); |
805 |
} |
806 |
|
807 |
- SLIST_INIT(&modes); |
808 |
- |
809 |
- struct Mode *default_mode = scalloc(sizeof(struct Mode)); |
810 |
- default_mode->name = sstrdup("default"); |
811 |
- default_mode->bindings = scalloc(sizeof(struct bindings_head)); |
812 |
- TAILQ_INIT(default_mode->bindings); |
813 |
- SLIST_INSERT_HEAD(&modes, default_mode, modes); |
814 |
- |
815 |
- bindings = default_mode->bindings; |
816 |
+ initialize_bindings(); |
817 |
|
818 |
#define REQUIRED_OPTION(name) \ |
819 |
if (config.name == NULL) \ |
820 |
@@ -477,8 +269,8 @@ void load_configuration(xcb_connection_t *conn, const char *override_configpath, |
821 |
parse_configuration(override_configpath); |
822 |
|
823 |
if (reload) { |
824 |
- translate_keysyms(); |
825 |
- grab_all_keys(conn, false); |
826 |
+ translate_binding_keysyms(); |
827 |
+ grab_bound_keys(conn, false); |
828 |
} |
829 |
|
830 |
if (config.font.type == FONT_TYPE_NONE) { |
b/src/config_directives.c
835 |
@@ -130,7 +130,11 @@ static bool eval_boolstr(const char *str) { |
836 |
strcasecmp(str, "active") == 0); |
837 |
} |
838 |
|
839 |
-static uint32_t modifiers_from_str(const char *str) { |
840 |
+/* |
841 |
+ * A utility function to convert a string of modifiers to the corresponding bit |
842 |
+ * mask. |
843 |
+ */ |
844 |
+uint32_t modifiers_from_str(const char *str) { |
845 |
/* It might be better to use strtok() here, but the simpler strstr() should |
846 |
* do for now. */ |
847 |
uint32_t result = 0; |
848 |
@@ -167,24 +171,8 @@ CFGFUN(font, const char *font) { |
849 |
font_pattern = sstrdup(font); |
850 |
} |
851 |
|
852 |
-// TODO: refactor with mode_binding |
853 |
CFGFUN(binding, const char *bindtype, const char *modifiers, const char *key, const char *release, const char *command) { |
854 |
- Binding *new_binding = scalloc(sizeof(Binding)); |
855 |
- DLOG("bindtype %s, modifiers %s, key %s, release %s\n", bindtype, modifiers, key, release); |
856 |
- new_binding->release = (release != NULL ? B_UPON_KEYRELEASE : B_UPON_KEYPRESS); |
857 |
- if (strcmp(bindtype, "bindsym") == 0) { |
858 |
- new_binding->symbol = sstrdup(key); |
859 |
- } else { |
860 |
- // TODO: strtol with proper error handling |
861 |
- new_binding->keycode = atoi(key); |
862 |
- if (new_binding->keycode == 0) { |
863 |
- ELOG("Could not parse \"%s\" as a keycode, ignoring this binding.\n", key); |
864 |
- return; |
865 |
- } |
866 |
- } |
867 |
- new_binding->mods = modifiers_from_str(modifiers); |
868 |
- new_binding->command = sstrdup(command); |
869 |
- TAILQ_INSERT_TAIL(bindings, new_binding, bindings); |
870 |
+ configure_binding(bindtype, modifiers, key, release, command, DEFAULT_BINDING_MODE); |
871 |
} |
872 |
|
873 |
|
874 |
@@ -192,39 +180,20 @@ CFGFUN(binding, const char *bindtype, const char *modifiers, const char *key, co |
875 |
* Mode handling |
876 |
******************************************************************************/ |
877 |
|
878 |
-static struct bindings_head *current_bindings; |
879 |
+static char *current_mode; |
880 |
|
881 |
CFGFUN(mode_binding, const char *bindtype, const char *modifiers, const char *key, const char *release, const char *command) { |
882 |
- Binding *new_binding = scalloc(sizeof(Binding)); |
883 |
- DLOG("bindtype %s, modifiers %s, key %s, release %s\n", bindtype, modifiers, key, release); |
884 |
- new_binding->release = (release != NULL ? B_UPON_KEYRELEASE : B_UPON_KEYPRESS); |
885 |
- if (strcmp(bindtype, "bindsym") == 0) { |
886 |
- new_binding->symbol = sstrdup(key); |
887 |
- } else { |
888 |
- // TODO: strtol with proper error handling |
889 |
- new_binding->keycode = atoi(key); |
890 |
- if (new_binding->keycode == 0) { |
891 |
- ELOG("Could not parse \"%s\" as a keycode, ignoring this binding.\n", key); |
892 |
- return; |
893 |
- } |
894 |
- } |
895 |
- new_binding->mods = modifiers_from_str(modifiers); |
896 |
- new_binding->command = sstrdup(command); |
897 |
- TAILQ_INSERT_TAIL(current_bindings, new_binding, bindings); |
898 |
+ configure_binding(bindtype, modifiers, key, release, command, current_mode); |
899 |
} |
900 |
|
901 |
CFGFUN(enter_mode, const char *modename) { |
902 |
- if (strcasecmp(modename, "default") == 0) { |
903 |
- ELOG("You cannot use the name \"default\" for your mode\n"); |
904 |
+ if (strcasecmp(modename,DEFAULT_BINDING_MODE) == 0) { |
905 |
+ ELOG("You cannot use the name %s for your mode\n", DEFAULT_BINDING_MODE); |
906 |
exit(1); |
907 |
} |
908 |
DLOG("\t now in mode %s\n", modename); |
909 |
- struct Mode *mode = scalloc(sizeof(struct Mode)); |
910 |
- mode->name = sstrdup(modename); |
911 |
- mode->bindings = scalloc(sizeof(struct bindings_head)); |
912 |
- TAILQ_INIT(mode->bindings); |
913 |
- current_bindings = mode->bindings; |
914 |
- SLIST_INSERT_HEAD(&modes, mode, modes); |
915 |
+ FREE(current_mode); |
916 |
+ current_mode = sstrdup(modename); |
917 |
} |
918 |
|
919 |
CFGFUN(exec, const char *exectype, const char *no_startup_id, const char *command) { |
b/src/config_parser.c
924 |
@@ -844,55 +844,6 @@ static char *migrate_config(char *input, off_t size) { |
925 |
} |
926 |
|
927 |
/* |
928 |
- * Checks for duplicate key bindings (the same keycode or keysym is configured |
929 |
- * more than once). If a duplicate binding is found, a message is printed to |
930 |
- * stderr and the has_errors variable is set to true, which will start |
931 |
- * i3-nagbar. |
932 |
- * |
933 |
- */ |
934 |
-static void check_for_duplicate_bindings(struct context *context) { |
935 |
- Binding *bind, *current; |
936 |
- TAILQ_FOREACH(current, bindings, bindings) { |
937 |
- TAILQ_FOREACH(bind, bindings, bindings) { |
938 |
- /* Abort when we reach the current keybinding, only check the |
939 |
- * bindings before */ |
940 |
- if (bind == current) |
941 |
- break; |
942 |
- |
943 |
- /* Check if one is using keysym while the other is using bindsym. |
944 |
- * If so, skip. */ |
945 |
- /* XXX: It should be checked at a later place (when translating the |
946 |
- * keysym to keycodes) if there are any duplicates */ |
947 |
- if ((bind->symbol == NULL && current->symbol != NULL) || |
948 |
- (bind->symbol != NULL && current->symbol == NULL)) |
949 |
- continue; |
950 |
- |
951 |
- /* If bind is NULL, current has to be NULL, too (see above). |
952 |
- * If the keycodes differ, it can't be a duplicate. */ |
953 |
- if (bind->symbol != NULL && |
954 |
- strcasecmp(bind->symbol, current->symbol) != 0) |
955 |
- continue; |
956 |
- |
957 |
- /* Check if the keycodes or modifiers are different. If so, they |
958 |
- * can't be duplicate */ |
959 |
- if (bind->keycode != current->keycode || |
960 |
- bind->mods != current->mods || |
961 |
- bind->release != current->release) |
962 |
- continue; |
963 |
- |
964 |
- context->has_errors = true; |
965 |
- if (current->keycode != 0) { |
966 |
- ELOG("Duplicate keybinding in config file:\n modmask %d with keycode %d, command \"%s\"\n", |
967 |
- current->mods, current->keycode, current->command); |
968 |
- } else { |
969 |
- ELOG("Duplicate keybinding in config file:\n modmask %d with keysym %s, command \"%s\"\n", |
970 |
- current->mods, current->symbol, current->command); |
971 |
- } |
972 |
- } |
973 |
- } |
974 |
-} |
975 |
- |
976 |
-/* |
977 |
* Parses the given file by first replacing the variables, then calling |
978 |
* parse_config and possibly launching i3-nagbar. |
979 |
* |
980 |
@@ -1056,7 +1007,7 @@ void parse_file(const char *f) { |
981 |
struct ConfigResult *config_output = parse_config(new, context); |
982 |
yajl_gen_free(config_output->json_gen); |
983 |
|
984 |
- check_for_duplicate_bindings(context); |
985 |
+ validate_bindings(context); |
986 |
|
987 |
if (context->has_errors || context->has_warnings) { |
988 |
ELOG("FYI: You are using i3 version " I3_VERSION "\n"); |
b/src/handlers.c
993 |
@@ -259,8 +259,8 @@ static void handle_mapping_notify(xcb_mapping_notify_event_t *event) { |
994 |
xcb_numlock_mask = aio_get_mod_mask_for(XCB_NUM_LOCK, keysyms); |
995 |
|
996 |
ungrab_all_keys(conn); |
997 |
- translate_keysyms(); |
998 |
- grab_all_keys(conn, false); |
999 |
+ translate_binding_keysyms(); |
1000 |
+ grab_bound_keys(conn, false); |
1001 |
|
1002 |
return; |
1003 |
} |
b/src/key_press.c
1008 |
@@ -84,7 +84,7 @@ void handle_key_press(xcb_key_press_event_t *event) { |
1009 |
DLOG("(checked mode_switch, state %d)\n", state_filtered); |
1010 |
|
1011 |
/* Find the binding */ |
1012 |
- Binding *bind = get_binding(state_filtered, key_release, event->detail); |
1013 |
+ Binding *bind = get_keyboard_binding(state_filtered, event->detail, key_release); |
1014 |
|
1015 |
/* No match? Then the user has Mode_switch enabled but does not have a |
1016 |
* specific keybinding. Fall back to the default keybindings (without |
1017 |
@@ -93,7 +93,7 @@ void handle_key_press(xcb_key_press_event_t *event) { |
1018 |
if (bind == NULL) { |
1019 |
state_filtered &= ~(BIND_MODE_SWITCH); |
1020 |
DLOG("no match, new state_filtered = %d\n", state_filtered); |
1021 |
- if ((bind = get_binding(state_filtered, key_release, event->detail)) == NULL) { |
1022 |
+ if ((bind = get_keyboard_binding(state_filtered, event->detail, key_release)) == NULL) { |
1023 |
/* This is not a real error since we can have release and |
1024 |
* non-release keybindings. On a KeyPress event for which there is |
1025 |
* only a !release-binding, but no release-binding, the |
1026 |
@@ -105,12 +105,7 @@ void handle_key_press(xcb_key_press_event_t *event) { |
1027 |
} |
1028 |
} |
1029 |
|
1030 |
- char *command_copy = sstrdup(bind->command); |
1031 |
- struct CommandResult *command_output = parse_command(command_copy); |
1032 |
- free(command_copy); |
1033 |
- |
1034 |
- if (command_output->needs_tree_render) |
1035 |
- tree_render(); |
1036 |
+ struct CommandResult *command_output = run_binding(bind); |
1037 |
|
1038 |
/* We parse the JSON reply to figure out whether there was an error |
1039 |
* ("success" being false in on of the returned dictionaries). */ |
b/src/main.c
1044 |
@@ -206,13 +206,13 @@ static void xkb_got_event(EV_P_ struct ev_io *w, int revents) { |
1045 |
|
1046 |
if (ev.state.group == XkbGroup2Index) { |
1047 |
DLOG("Mode_switch enabled\n"); |
1048 |
- grab_all_keys(conn, true); |
1049 |
+ grab_bound_keys(conn, true); |
1050 |
} |
1051 |
|
1052 |
if (ev.state.group == XkbGroup1Index) { |
1053 |
DLOG("Mode_switch disabled\n"); |
1054 |
ungrab_all_keys(conn); |
1055 |
- grab_all_keys(conn, false); |
1056 |
+ grab_bound_keys(conn, false); |
1057 |
} |
1058 |
} |
1059 |
|
1060 |
@@ -227,8 +227,8 @@ static void xkb_got_event(EV_P_ struct ev_io *w, int revents) { |
1061 |
|
1062 |
ungrab_all_keys(conn); |
1063 |
DLOG("Re-grabbing...\n"); |
1064 |
- translate_keysyms(); |
1065 |
- grab_all_keys(conn, (xkb_current_group == XkbGroup2Index)); |
1066 |
+ translate_binding_keysyms(); |
1067 |
+ grab_bound_keys(conn, (xkb_current_group == XkbGroup2Index)); |
1068 |
DLOG("Done\n"); |
1069 |
} |
1070 |
|
1071 |
@@ -663,8 +663,8 @@ int main(int argc, char *argv[]) { |
1072 |
|
1073 |
xcb_numlock_mask = aio_get_mod_mask_for(XCB_NUM_LOCK, keysyms); |
1074 |
|
1075 |
- translate_keysyms(); |
1076 |
- grab_all_keys(conn, false); |
1077 |
+ translate_binding_keysyms(); |
1078 |
+ grab_bound_keys(conn, false); |
1079 |
|
1080 |
bool needs_tree_init = true; |
1081 |
if (layout_path) { |
b/src/workspace.c
1086 |
@@ -116,7 +116,9 @@ Con *create_workspace_on_output(Output *output, Con *content) { |
1087 |
Con *ws = con_new(NULL, NULL); |
1088 |
ws->type = CT_WORKSPACE; |
1089 |
|
1090 |
- /* try the configured workspace bindings first to find a free name */ |
1091 |
+ /* try the configured workspace bindings first to find a free name. |
1092 |
+ * TODO: try to figure out a way to do this without directly using the list |
1093 |
+ * of bindings.*/ |
1094 |
Binding *bind; |
1095 |
TAILQ_FOREACH(bind, bindings, bindings) { |
1096 |
DLOG("binding with command %s\n", bind->command); |