xref: /openbsd/usr.bin/tmux/cmd-list-keys.c (revision 1e415b51)
1 /* $OpenBSD: cmd-list-keys.c,v 1.67 2023/01/17 10:40:51 nicm Exp $ */
2 
3 /*
4  * Copyright (c) 2007 Nicholas Marriott <nicholas.marriott@gmail.com>
5  *
6  * Permission to use, copy, modify, and distribute this software for any
7  * purpose with or without fee is hereby granted, provided that the above
8  * copyright notice and this permission notice appear in all copies.
9  *
10  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14  * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
15  * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
16  * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17  */
18 
19 #include <sys/types.h>
20 
21 #include <stdlib.h>
22 #include <string.h>
23 
24 #include "tmux.h"
25 
26 /*
27  * List key bindings.
28  */
29 
30 static enum cmd_retval	cmd_list_keys_exec(struct cmd *, struct cmdq_item *);
31 
32 static enum cmd_retval	cmd_list_keys_commands(struct cmd *,
33 			    struct cmdq_item *);
34 
35 const struct cmd_entry cmd_list_keys_entry = {
36 	.name = "list-keys",
37 	.alias = "lsk",
38 
39 	.args = { "1aNP:T:", 0, 1, NULL },
40 	.usage = "[-1aN] [-P prefix-string] [-T key-table] [key]",
41 
42 	.flags = CMD_STARTSERVER|CMD_AFTERHOOK,
43 	.exec = cmd_list_keys_exec
44 };
45 
46 const struct cmd_entry cmd_list_commands_entry = {
47 	.name = "list-commands",
48 	.alias = "lscm",
49 
50 	.args = { "F:", 0, 1, NULL },
51 	.usage = "[-F format] [command]",
52 
53 	.flags = CMD_STARTSERVER|CMD_AFTERHOOK,
54 	.exec = cmd_list_keys_exec
55 };
56 
57 static u_int
cmd_list_keys_get_width(const char * tablename,key_code only)58 cmd_list_keys_get_width(const char *tablename, key_code only)
59 {
60 	struct key_table	*table;
61 	struct key_binding	*bd;
62 	u_int			 width, keywidth = 0;
63 
64 	table = key_bindings_get_table(tablename, 0);
65 	if (table == NULL)
66 		return (0);
67 	bd = key_bindings_first(table);
68 	while (bd != NULL) {
69 		if ((only != KEYC_UNKNOWN && bd->key != only) ||
70 		    KEYC_IS_MOUSE(bd->key) ||
71 		    bd->note == NULL ||
72 		    *bd->note == '\0') {
73 			bd = key_bindings_next(table, bd);
74 			continue;
75 		}
76 		width = utf8_cstrwidth(key_string_lookup_key(bd->key, 0));
77 		if (width > keywidth)
78 			keywidth = width;
79 
80 		bd = key_bindings_next(table, bd);
81 	}
82 	return (keywidth);
83 }
84 
85 static int
cmd_list_keys_print_notes(struct cmdq_item * item,struct args * args,const char * tablename,u_int keywidth,key_code only,const char * prefix)86 cmd_list_keys_print_notes(struct cmdq_item *item, struct args *args,
87     const char *tablename, u_int keywidth, key_code only, const char *prefix)
88 {
89 	struct client		*tc = cmdq_get_target_client(item);
90 	struct key_table	*table;
91 	struct key_binding	*bd;
92 	const char		*key;
93 	char			*tmp, *note;
94 	int	                 found = 0;
95 
96 	table = key_bindings_get_table(tablename, 0);
97 	if (table == NULL)
98 		return (0);
99 	bd = key_bindings_first(table);
100 	while (bd != NULL) {
101 		if ((only != KEYC_UNKNOWN && bd->key != only) ||
102 		    KEYC_IS_MOUSE(bd->key) ||
103 		    ((bd->note == NULL || *bd->note == '\0') &&
104 		    !args_has(args, 'a'))) {
105 			bd = key_bindings_next(table, bd);
106 			continue;
107 		}
108 		found = 1;
109 		key = key_string_lookup_key(bd->key, 0);
110 
111 		if (bd->note == NULL || *bd->note == '\0')
112 			note = cmd_list_print(bd->cmdlist, 1);
113 		else
114 			note = xstrdup(bd->note);
115 		tmp = utf8_padcstr(key, keywidth + 1);
116 		if (args_has(args, '1') && tc != NULL) {
117 			status_message_set(tc, -1, 1, 0, "%s%s%s", prefix, tmp,
118 			    note);
119 		} else
120 			cmdq_print(item, "%s%s%s", prefix, tmp, note);
121 		free(tmp);
122 		free(note);
123 
124 		if (args_has(args, '1'))
125 			break;
126 		bd = key_bindings_next(table, bd);
127 	}
128 	return (found);
129 }
130 
131 static char *
cmd_list_keys_get_prefix(struct args * args,key_code * prefix)132 cmd_list_keys_get_prefix(struct args *args, key_code *prefix)
133 {
134 	char	*s;
135 
136 	*prefix = options_get_number(global_s_options, "prefix");
137 	if (!args_has(args, 'P')) {
138 		if (*prefix != KEYC_NONE)
139 			xasprintf(&s, "%s ", key_string_lookup_key(*prefix, 0));
140 		else
141 			s = xstrdup("");
142 	} else
143 		s = xstrdup(args_get(args, 'P'));
144 	return (s);
145 }
146 
147 static enum cmd_retval
cmd_list_keys_exec(struct cmd * self,struct cmdq_item * item)148 cmd_list_keys_exec(struct cmd *self, struct cmdq_item *item)
149 {
150 	struct args		*args = cmd_get_args(self);
151 	struct client		*tc = cmdq_get_target_client(item);
152 	struct key_table	*table;
153 	struct key_binding	*bd;
154 	const char		*tablename, *r, *keystr;
155 	char			*key, *cp, *tmp, *start, *empty;
156 	key_code		 prefix, only = KEYC_UNKNOWN;
157 	int			 repeat, width, tablewidth, keywidth, found = 0;
158 	size_t			 tmpsize, tmpused, cplen;
159 
160 	if (cmd_get_entry(self) == &cmd_list_commands_entry)
161 		return (cmd_list_keys_commands(self, item));
162 
163 	if ((keystr = args_string(args, 0)) != NULL) {
164 		only = key_string_lookup_string(keystr);
165 		if (only == KEYC_UNKNOWN) {
166 			cmdq_error(item, "invalid key: %s", keystr);
167 			return (CMD_RETURN_ERROR);
168 		}
169 		only &= (KEYC_MASK_KEY|KEYC_MASK_MODIFIERS);
170 	}
171 
172 	tablename = args_get(args, 'T');
173 	if (tablename != NULL && key_bindings_get_table(tablename, 0) == NULL) {
174 		cmdq_error(item, "table %s doesn't exist", tablename);
175 		return (CMD_RETURN_ERROR);
176 	}
177 
178 	if (args_has(args, 'N')) {
179 		if (tablename == NULL) {
180 			start = cmd_list_keys_get_prefix(args, &prefix);
181 			keywidth = cmd_list_keys_get_width("root", only);
182 			if (prefix != KEYC_NONE) {
183 				width = cmd_list_keys_get_width("prefix", only);
184 				if (width == 0)
185 					prefix = KEYC_NONE;
186 				else if (width > keywidth)
187 					keywidth = width;
188 			}
189 			empty = utf8_padcstr("", utf8_cstrwidth(start));
190 
191 			found = cmd_list_keys_print_notes(item, args, "root",
192 			    keywidth, only, empty);
193 			if (prefix != KEYC_NONE) {
194 				if (cmd_list_keys_print_notes(item, args,
195 				    "prefix", keywidth, only, start))
196 					found = 1;
197 			}
198 			free(empty);
199 		} else {
200 			if (args_has(args, 'P'))
201 				start = xstrdup(args_get(args, 'P'));
202 			else
203 				start = xstrdup("");
204 			keywidth = cmd_list_keys_get_width(tablename, only);
205 			found = cmd_list_keys_print_notes(item, args, tablename,
206 			    keywidth, only, start);
207 
208 		}
209 		free(start);
210 		goto out;
211 	}
212 
213 	repeat = 0;
214 	tablewidth = keywidth = 0;
215 	table = key_bindings_first_table();
216 	while (table != NULL) {
217 		if (tablename != NULL && strcmp(table->name, tablename) != 0) {
218 			table = key_bindings_next_table(table);
219 			continue;
220 		}
221 		bd = key_bindings_first(table);
222 		while (bd != NULL) {
223 			if (only != KEYC_UNKNOWN && bd->key != only) {
224 				bd = key_bindings_next(table, bd);
225 				continue;
226 			}
227 			key = args_escape(key_string_lookup_key(bd->key, 0));
228 
229 			if (bd->flags & KEY_BINDING_REPEAT)
230 				repeat = 1;
231 
232 			width = utf8_cstrwidth(table->name);
233 			if (width > tablewidth)
234 				tablewidth = width;
235 			width = utf8_cstrwidth(key);
236 			if (width > keywidth)
237 				keywidth = width;
238 
239 			free(key);
240 			bd = key_bindings_next(table, bd);
241 		}
242 		table = key_bindings_next_table(table);
243 	}
244 
245 	tmpsize = 256;
246 	tmp = xmalloc(tmpsize);
247 
248 	table = key_bindings_first_table();
249 	while (table != NULL) {
250 		if (tablename != NULL && strcmp(table->name, tablename) != 0) {
251 			table = key_bindings_next_table(table);
252 			continue;
253 		}
254 		bd = key_bindings_first(table);
255 		while (bd != NULL) {
256 			if (only != KEYC_UNKNOWN && bd->key != only) {
257 				bd = key_bindings_next(table, bd);
258 				continue;
259 			}
260 			found = 1;
261 			key = args_escape(key_string_lookup_key(bd->key, 0));
262 
263 			if (!repeat)
264 				r = "";
265 			else if (bd->flags & KEY_BINDING_REPEAT)
266 				r = "-r ";
267 			else
268 				r = "   ";
269 			tmpused = xsnprintf(tmp, tmpsize, "%s-T ", r);
270 
271 			cp = utf8_padcstr(table->name, tablewidth);
272 			cplen = strlen(cp) + 1;
273 			while (tmpused + cplen + 1 >= tmpsize) {
274 				tmpsize *= 2;
275 				tmp = xrealloc(tmp, tmpsize);
276 			}
277 			strlcat(tmp, cp, tmpsize);
278 			tmpused = strlcat(tmp, " ", tmpsize);
279 			free(cp);
280 
281 			cp = utf8_padcstr(key, keywidth);
282 			cplen = strlen(cp) + 1;
283 			while (tmpused + cplen + 1 >= tmpsize) {
284 				tmpsize *= 2;
285 				tmp = xrealloc(tmp, tmpsize);
286 			}
287 			strlcat(tmp, cp, tmpsize);
288 			tmpused = strlcat(tmp, " ", tmpsize);
289 			free(cp);
290 
291 			cp = cmd_list_print(bd->cmdlist, 1);
292 			cplen = strlen(cp);
293 			while (tmpused + cplen + 1 >= tmpsize) {
294 				tmpsize *= 2;
295 				tmp = xrealloc(tmp, tmpsize);
296 			}
297 			strlcat(tmp, cp, tmpsize);
298 			free(cp);
299 
300 			if (args_has(args, '1') && tc != NULL) {
301 				status_message_set(tc, -1, 1, 0, "bind-key %s",
302 				    tmp);
303 			} else
304 				cmdq_print(item, "bind-key %s", tmp);
305 			free(key);
306 
307 			if (args_has(args, '1'))
308 				break;
309 			bd = key_bindings_next(table, bd);
310 		}
311 		table = key_bindings_next_table(table);
312 	}
313 
314 	free(tmp);
315 
316 out:
317 	if (only != KEYC_UNKNOWN && !found) {
318 		cmdq_error(item, "unknown key: %s", args_string(args, 0));
319 		return (CMD_RETURN_ERROR);
320 	}
321 	return (CMD_RETURN_NORMAL);
322 }
323 
324 static enum cmd_retval
cmd_list_keys_commands(struct cmd * self,struct cmdq_item * item)325 cmd_list_keys_commands(struct cmd *self, struct cmdq_item *item)
326 {
327 	struct args		 *args = cmd_get_args(self);
328 	const struct cmd_entry	**entryp;
329 	const struct cmd_entry	 *entry;
330 	struct format_tree	 *ft;
331 	const char		 *template, *s, *command;
332 	char			 *line;
333 
334 	if ((template = args_get(args, 'F')) == NULL) {
335 		template = "#{command_list_name}"
336 		    "#{?command_list_alias, (#{command_list_alias}),} "
337 		    "#{command_list_usage}";
338 	}
339 
340 	ft = format_create(cmdq_get_client(item), item, FORMAT_NONE, 0);
341 	format_defaults(ft, NULL, NULL, NULL, NULL);
342 
343 	command = args_string(args, 0);
344 	for (entryp = cmd_table; *entryp != NULL; entryp++) {
345 		entry = *entryp;
346 		if (command != NULL &&
347 		    (strcmp(entry->name, command) != 0 &&
348 		    (entry->alias == NULL ||
349 		    strcmp(entry->alias, command) != 0)))
350 		    continue;
351 
352 		format_add(ft, "command_list_name", "%s", entry->name);
353 		if (entry->alias != NULL)
354 			s = entry->alias;
355 		else
356 			s = "";
357 		format_add(ft, "command_list_alias", "%s", s);
358 		if (entry->usage != NULL)
359 			s = entry->usage;
360 		else
361 			s = "";
362 		format_add(ft, "command_list_usage", "%s", s);
363 
364 		line = format_expand(ft, template);
365 		if (*line != '\0')
366 			cmdq_print(item, "%s", line);
367 		free(line);
368 	}
369 
370 	format_free(ft);
371 	return (CMD_RETURN_NORMAL);
372 }
373