[BACK]Return to cmd-send-keys.c CVS log [TXT][DIR] Up to [local] / src / usr.bin / tmux

Annotation of src/usr.bin/tmux/cmd-send-keys.c, Revision 1.40

1.40    ! nicm        1: /* $OpenBSD: cmd-send-keys.c,v 1.39 2017/04/22 10:22:39 nicm Exp $ */
1.1       nicm        2:
                      3: /*
1.28      nicm        4:  * Copyright (c) 2008 Nicholas Marriott <nicholas.marriott@gmail.com>
1.1       nicm        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>
1.10      nicm       22: #include <string.h>
1.1       nicm       23:
                     24: #include "tmux.h"
                     25:
                     26: /*
                     27:  * Send keys to client.
                     28:  */
                     29:
1.33      nicm       30: static enum cmd_retval cmd_send_keys_exec(struct cmd *, struct cmdq_item *);
1.1       nicm       31:
                     32: const struct cmd_entry cmd_send_keys_entry = {
1.25      nicm       33:        .name = "send-keys",
                     34:        .alias = "send",
                     35:
1.30      nicm       36:        .args = { "lXRMN:t:", 0, -1 },
                     37:        .usage = "[-lXRM] [-N repeat-count] " CMD_TARGET_PANE_USAGE " key ...",
1.25      nicm       38:
1.39      nicm       39:        .target = { 't', CMD_FIND_PANE, 0 },
1.26      nicm       40:
1.31      nicm       41:        .flags = CMD_AFTERHOOK,
1.25      nicm       42:        .exec = cmd_send_keys_exec
1.1       nicm       43: };
                     44:
1.14      nicm       45: const struct cmd_entry cmd_send_prefix_entry = {
1.25      nicm       46:        .name = "send-prefix",
                     47:        .alias = NULL,
                     48:
                     49:        .args = { "2t:", 0, 0 },
                     50:        .usage = "[-2] " CMD_TARGET_PANE_USAGE,
                     51:
1.39      nicm       52:        .target = { 't', CMD_FIND_PANE, 0 },
1.26      nicm       53:
1.31      nicm       54:        .flags = CMD_AFTERHOOK,
1.25      nicm       55:        .exec = cmd_send_keys_exec
1.14      nicm       56: };
                     57:
1.40    ! nicm       58: static void
        !            59: cmd_send_keys_inject(struct client *c, struct cmdq_item *item, key_code key)
        !            60: {
        !            61:        struct window_pane      *wp = item->target.wp;
        !            62:        struct session          *s = item->target.s;
        !            63:        struct key_table        *table;
        !            64:        struct key_binding      *bd, bd_find;
        !            65:
        !            66:        if (wp->mode == NULL || wp->mode->key_table == NULL) {
        !            67:                window_pane_key(wp, NULL, s, key, NULL);
        !            68:                return;
        !            69:        }
        !            70:        table = key_bindings_get_table(wp->mode->key_table(wp), 1);
        !            71:
        !            72:        bd_find.key = (key & ~KEYC_XTERM);
        !            73:        bd = RB_FIND(key_bindings, &table->key_bindings, &bd_find);
        !            74:        if (bd != NULL) {
        !            75:                table->references++;
        !            76:                key_bindings_dispatch(bd, c, NULL, &item->target);
        !            77:                key_bindings_unref_table(table);
        !            78:        }
        !            79: }
        !            80:
1.29      nicm       81: static enum cmd_retval
1.33      nicm       82: cmd_send_keys_exec(struct cmd *self, struct cmdq_item *item)
1.1       nicm       83: {
1.9       nicm       84:        struct args             *args = self->args;
1.39      nicm       85:        struct client           *c = cmd_find_client(item, NULL, 1);
                     86:        struct window_pane      *wp = item->target.wp;
                     87:        struct session          *s = item->target.s;
1.38      nicm       88:        struct mouse_event      *m = &item->shared->mouse;
1.35      nicm       89:        struct utf8_data        *ud, *uc;
                     90:        wchar_t                  wc;
1.23      nicm       91:        int                      i, literal;
1.22      nicm       92:        key_code                 key;
1.34      nicm       93:        u_int                    np = 1;
1.30      nicm       94:        char                    *cause = NULL;
                     95:
                     96:        if (args_has(args, 'N')) {
                     97:                np = args_strtonum(args, 'N', 1, UINT_MAX, &cause);
                     98:                if (cause != NULL) {
1.34      nicm       99:                        cmdq_error(item, "repeat count %s", cause);
1.30      nicm      100:                        free(cause);
                    101:                        return (CMD_RETURN_ERROR);
                    102:                }
1.36      nicm      103:                if (args_has(args, 'X') || args->argc == 0)
                    104:                        wp->modeprefix = np;
1.30      nicm      105:        }
                    106:
                    107:        if (args_has(args, 'X')) {
                    108:                if (wp->mode == NULL || wp->mode->command == NULL) {
1.33      nicm      109:                        cmdq_error(item, "not in a mode");
1.30      nicm      110:                        return (CMD_RETURN_ERROR);
                    111:                }
                    112:                if (!m->valid)
                    113:                        wp->mode->command(wp, c, s, args, NULL);
                    114:                else
                    115:                        wp->mode->command(wp, c, s, args, m);
                    116:                return (CMD_RETURN_NORMAL);
                    117:        }
                    118:
1.19      nicm      119:        if (args_has(args, 'M')) {
                    120:                wp = cmd_mouse_pane(m, &s, NULL);
                    121:                if (wp == NULL) {
1.33      nicm      122:                        cmdq_error(item, "no mouse target");
1.19      nicm      123:                        return (CMD_RETURN_ERROR);
                    124:                }
                    125:                window_pane_key(wp, NULL, s, m->key, m);
                    126:                return (CMD_RETURN_NORMAL);
                    127:        }
1.14      nicm      128:
                    129:        if (self->entry == &cmd_send_prefix_entry) {
                    130:                if (args_has(args, '2'))
1.21      nicm      131:                        key = options_get_number(s->options, "prefix2");
1.14      nicm      132:                else
1.21      nicm      133:                        key = options_get_number(s->options, "prefix");
1.40    ! nicm      134:                cmd_send_keys_inject(c, item, key);
1.14      nicm      135:                return (CMD_RETURN_NORMAL);
                    136:        }
1.10      nicm      137:
1.37      nicm      138:        if (args_has(args, 'R')) {
                    139:                window_pane_reset_palette(wp);
1.27      nicm      140:                input_reset(wp, 1);
1.37      nicm      141:        }
1.1       nicm      142:
1.34      nicm      143:        for (; np != 0; np--) {
                    144:                for (i = 0; i < args->argc; i++) {
                    145:                        literal = args_has(args, 'l');
                    146:                        if (!literal) {
                    147:                                key = key_string_lookup_string(args->argv[i]);
                    148:                                if (key != KEYC_NONE && key != KEYC_UNKNOWN)
1.40    ! nicm      149:                                        cmd_send_keys_inject(c, item, key);
1.34      nicm      150:                                else
                    151:                                        literal = 1;
                    152:                        }
                    153:                        if (literal) {
1.35      nicm      154:                                ud = utf8_fromcstr(args->argv[i]);
                    155:                                for (uc = ud; uc->size != 0; uc++) {
1.37      nicm      156:                                        if (utf8_combine(uc, &wc) != UTF8_DONE)
                    157:                                                continue;
1.40    ! nicm      158:                                        cmd_send_keys_inject(c, item, wc);
1.35      nicm      159:                                }
                    160:                                free(ud);
1.34      nicm      161:                        }
1.9       nicm      162:                }
1.34      nicm      163:
1.9       nicm      164:        }
1.1       nicm      165:
1.12      nicm      166:        return (CMD_RETURN_NORMAL);
1.1       nicm      167: }