xref: /openbsd-src/usr.bin/tmux/cmd-bind-key.c (revision 1e1971d6873b29337fd9e50dd5d4bebd02e87f1b)
1 /* $OpenBSD: cmd-bind-key.c,v 1.45 2021/08/25 09:18:08 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  * Bind a key to a command.
28  */
29 
30 static enum args_parse_type	cmd_bind_key_args_parse(struct args *, u_int,
31 				    char **);
32 static enum cmd_retval		cmd_bind_key_exec(struct cmd *,
33 				    struct cmdq_item *);
34 
35 const struct cmd_entry cmd_bind_key_entry = {
36 	.name = "bind-key",
37 	.alias = "bind",
38 
39 	.args = { "nrN:T:", 1, -1, cmd_bind_key_args_parse },
40 	.usage = "[-nr] [-T key-table] [-N note] key "
41 	         "[command [arguments]]",
42 
43 	.flags = CMD_AFTERHOOK,
44 	.exec = cmd_bind_key_exec
45 };
46 
47 static enum args_parse_type
48 cmd_bind_key_args_parse(__unused struct args *args, __unused u_int idx,
49     __unused char **cause)
50 {
51 	return (ARGS_PARSE_COMMANDS_OR_STRING);
52 }
53 
54 static enum cmd_retval
55 cmd_bind_key_exec(struct cmd *self, struct cmdq_item *item)
56 {
57 	struct args		 *args = cmd_get_args(self);
58 	key_code		  key;
59 	const char		 *tablename, *note = args_get(args, 'N');
60 	struct cmd_parse_result	 *pr;
61 	char			**argv;
62 	int			  argc, repeat;
63 	struct args_value	 *value;
64 	u_int			  count = args_count(args);
65 
66 	key = key_string_lookup_string(args_string(args, 0));
67 	if (key == KEYC_NONE || key == KEYC_UNKNOWN) {
68 		cmdq_error(item, "unknown key: %s", args_string(args, 0));
69 		return (CMD_RETURN_ERROR);
70 	}
71 
72 	if (args_has(args, 'T'))
73 		tablename = args_get(args, 'T');
74 	else if (args_has(args, 'n'))
75 		tablename = "root";
76 	else
77 		tablename = "prefix";
78 	repeat = args_has(args, 'r');
79 
80 	if (count == 1) {
81 		key_bindings_add(tablename, key, note, repeat, NULL);
82 		return (CMD_RETURN_NORMAL);
83 	}
84 
85 	value = args_value(args, 1);
86 	if (count == 2 && value->type == ARGS_COMMANDS) {
87 		key_bindings_add(tablename, key, note, repeat, value->cmdlist);
88 		value->cmdlist->references++;
89 		return (CMD_RETURN_NORMAL);
90 	}
91 
92 	if (count == 2)
93 		pr = cmd_parse_from_string(args_string(args, 1), NULL);
94 	else {
95 		args_vector(args, &argc, &argv);
96 		pr = cmd_parse_from_arguments(argc - 1, argv + 1, NULL);
97 		cmd_free_argv(argc, argv);
98 	}
99 	switch (pr->status) {
100 	case CMD_PARSE_ERROR:
101 		cmdq_error(item, "%s", pr->error);
102 		free(pr->error);
103 		return (CMD_RETURN_ERROR);
104 	case CMD_PARSE_SUCCESS:
105 		break;
106 	}
107 	key_bindings_add(tablename, key, note, repeat, pr->cmdlist);
108 	return (CMD_RETURN_NORMAL);
109 }
110