Add abstraction for running a binding
Patch status: merged
Patch by Tony Crisci
Long description:
Add run_binding function to bindings.h. > Runs the given binding and handles parse errors. Returns a > CommandResult for running the binding's command. Caller should render > tree if needs_tree_render is true. Free with command_result_free().
To apply this patch, use:
curl http://cr.i3wm.org/patch/562/raw.patch | git am
b/include/bindings.h
21 |
@@ -9,6 +9,8 @@ |
22 |
*/ |
23 |
#pragma once |
24 |
|
25 |
+extern pid_t command_error_nagbar_pid; |
26 |
+ |
27 |
/** |
28 |
* The name of the default mode. |
29 |
* |
30 |
@@ -57,3 +59,11 @@ void switch_mode(const char *new_mode); |
31 |
* |
32 |
*/ |
33 |
void check_for_duplicate_bindings(struct context *context); |
34 |
+ |
35 |
+/** |
36 |
+ * Runs the given binding and handles parse errors. Returns a CommandResult for |
37 |
+ * running the binding's command. Caller should render tree if |
38 |
+ * needs_tree_render is true. Free with command_result_free(). |
39 |
+ * |
40 |
+ */ |
41 |
+CommandResult *run_binding(Binding *bind); |
b/include/key_press.h
46 |
@@ -9,8 +9,6 @@ |
47 |
*/ |
48 |
#pragma once |
49 |
|
50 |
-extern pid_t command_error_nagbar_pid; |
51 |
- |
52 |
/** |
53 |
* There was a key press. We compare this key code with our bindings table and pass |
54 |
* the bound action to parse_command(). |
b/src/bindings.c
59 |
@@ -8,6 +8,8 @@ |
60 |
*/ |
61 |
#include "all.h" |
62 |
|
63 |
+pid_t command_error_nagbar_pid = -1; |
64 |
+ |
65 |
/* |
66 |
* The name of the default mode. |
67 |
* |
68 |
@@ -379,3 +381,42 @@ void check_for_duplicate_bindings(struct context *context) { |
69 |
} |
70 |
} |
71 |
} |
72 |
+ |
73 |
+/* |
74 |
+ * Runs the given binding and handles parse errors. Returns a CommandResult for |
75 |
+ * running the binding's command. Caller should render tree if |
76 |
+ * needs_tree_render is true. Free with command_result_free(). |
77 |
+ * |
78 |
+ */ |
79 |
+CommandResult *run_binding(Binding *bind) { |
80 |
+ char *command_copy = sstrdup(bind->command); |
81 |
+ CommandResult *result = parse_command(command_copy, NULL); |
82 |
+ free(command_copy); |
83 |
+ |
84 |
+ if (result->needs_tree_render) |
85 |
+ tree_render(); |
86 |
+ |
87 |
+ if (result->parse_error) { |
88 |
+ char *pageraction; |
89 |
+ sasprintf(&pageraction, "i3-sensible-pager \"%s\"\n", errorfilename); |
90 |
+ char *argv[] = { |
91 |
+ NULL, /* will be replaced by the executable path */ |
92 |
+ "-f", |
93 |
+ config.font.pattern, |
94 |
+ "-t", |
95 |
+ "error", |
96 |
+ "-m", |
97 |
+ "The configured command for this shortcut could not be run successfully.", |
98 |
+ "-b", |
99 |
+ "show errors", |
100 |
+ pageraction, |
101 |
+ NULL |
102 |
+ }; |
103 |
+ start_nagbar(&command_error_nagbar_pid, argv); |
104 |
+ free(pageraction); |
105 |
+ } |
106 |
+ |
107 |
+ /* TODO: emit event for running a binding */ |
108 |
+ |
109 |
+ return result; |
110 |
+} |
b/src/key_press.c
115 |
@@ -9,50 +9,8 @@ |
116 |
* key_press.c: key press handler |
117 |
* |
118 |
*/ |
119 |
-#include <sys/types.h> |
120 |
-#include <sys/stat.h> |
121 |
-#include <sys/wait.h> |
122 |
-#include <fcntl.h> |
123 |
#include "all.h" |
124 |
|
125 |
-static int current_nesting_level; |
126 |
-static bool parse_error_key; |
127 |
-static bool command_failed; |
128 |
- |
129 |
-pid_t command_error_nagbar_pid = -1; |
130 |
- |
131 |
-static int json_boolean(void *ctx, int boolval) { |
132 |
- DLOG("Got bool: %d, parse_error_key %d, nesting_level %d\n", boolval, parse_error_key, current_nesting_level); |
133 |
- |
134 |
- if (parse_error_key && current_nesting_level == 1 && boolval) |
135 |
- command_failed = true; |
136 |
- |
137 |
- return 1; |
138 |
-} |
139 |
- |
140 |
-static int json_map_key(void *ctx, const unsigned char *stringval, size_t stringlen) { |
141 |
- parse_error_key = (stringlen >= strlen("parse_error") && |
142 |
- strncmp((const char*)stringval, "parse_error", strlen("parse_error")) == 0); |
143 |
- return 1; |
144 |
-} |
145 |
- |
146 |
-static int json_start_map(void *ctx) { |
147 |
- current_nesting_level++; |
148 |
- return 1; |
149 |
-} |
150 |
- |
151 |
-static int json_end_map(void *ctx) { |
152 |
- current_nesting_level--; |
153 |
- return 1; |
154 |
-} |
155 |
- |
156 |
-static yajl_callbacks command_error_callbacks = { |
157 |
- .yajl_boolean = json_boolean, |
158 |
- .yajl_start_map = json_start_map, |
159 |
- .yajl_map_key = json_map_key, |
160 |
- .yajl_end_map = json_end_map, |
161 |
-}; |
162 |
- |
163 |
/* |
164 |
* There was a KeyPress or KeyRelease (both events have the same fields). We |
165 |
* compare this key code with our bindings table and pass the bound action to |
166 |
@@ -72,53 +30,10 @@ void handle_key_press(xcb_key_press_event_t *event) { |
167 |
if (bind == NULL) |
168 |
return; |
169 |
|
170 |
- yajl_gen gen = yajl_gen_alloc(NULL); |
171 |
- |
172 |
- char *command_copy = sstrdup(bind->command); |
173 |
- CommandResult *result = parse_command(command_copy, gen); |
174 |
- free(command_copy); |
175 |
+ CommandResult *result = run_binding(bind); |
176 |
|
177 |
if (result->needs_tree_render) |
178 |
tree_render(); |
179 |
|
180 |
command_result_free(result); |
181 |
- |
182 |
- /* We parse the JSON reply to figure out whether there was an error |
183 |
- * ("success" being false in on of the returned dictionaries). */ |
184 |
- const unsigned char *reply; |
185 |
- size_t length; |
186 |
- yajl_handle handle = yajl_alloc(&command_error_callbacks, NULL, NULL); |
187 |
- yajl_gen_get_buf(gen, &reply, &length); |
188 |
- |
189 |
- current_nesting_level = 0; |
190 |
- parse_error_key = false; |
191 |
- command_failed = false; |
192 |
- yajl_status state = yajl_parse(handle, reply, length); |
193 |
- if (state != yajl_status_ok) { |
194 |
- ELOG("Could not parse my own reply. That's weird. reply is %.*s\n", (int)length, reply); |
195 |
- } else { |
196 |
- if (command_failed) { |
197 |
- char *pageraction; |
198 |
- sasprintf(&pageraction, "i3-sensible-pager \"%s\"\n", errorfilename); |
199 |
- char *argv[] = { |
200 |
- NULL, /* will be replaced by the executable path */ |
201 |
- "-f", |
202 |
- config.font.pattern, |
203 |
- "-t", |
204 |
- "error", |
205 |
- "-m", |
206 |
- "The configured command for this shortcut could not be run successfully.", |
207 |
- "-b", |
208 |
- "show errors", |
209 |
- pageraction, |
210 |
- NULL |
211 |
- }; |
212 |
- start_nagbar(&command_error_nagbar_pid, argv); |
213 |
- free(pageraction); |
214 |
- } |
215 |
- } |
216 |
- |
217 |
- yajl_free(handle); |
218 |
- |
219 |
- yajl_gen_free(gen); |
220 |
} |