Annotation of embedaddon/tmux/cmd-set-option.c, revision 1.1

1.1     ! misho       1: /* $OpenBSD$ */
        !             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:  * Set an option.
        !            28:  */
        !            29: 
        !            30: static enum cmd_retval cmd_set_option_exec(struct cmd *, struct cmdq_item *);
        !            31: 
        !            32: static int     cmd_set_option_set(struct cmd *, struct cmdq_item *,
        !            33:                    struct options *, struct options_entry *, const char *);
        !            34: static int     cmd_set_option_flag(struct cmdq_item *,
        !            35:                    const struct options_table_entry *, struct options *,
        !            36:                    const char *);
        !            37: static int     cmd_set_option_choice(struct cmdq_item *,
        !            38:                    const struct options_table_entry *, struct options *,
        !            39:                    const char *);
        !            40: 
        !            41: const struct cmd_entry cmd_set_option_entry = {
        !            42:        .name = "set-option",
        !            43:        .alias = "set",
        !            44: 
        !            45:        .args = { "agoqst:uw", 1, 2 },
        !            46:        .usage = "[-agosquw] [-t target-window] option [value]",
        !            47: 
        !            48:        .tflag = CMD_WINDOW_CANFAIL,
        !            49: 
        !            50:        .flags = CMD_AFTERHOOK,
        !            51:        .exec = cmd_set_option_exec
        !            52: };
        !            53: 
        !            54: const struct cmd_entry cmd_set_window_option_entry = {
        !            55:        .name = "set-window-option",
        !            56:        .alias = "setw",
        !            57: 
        !            58:        .args = { "agoqt:u", 1, 2 },
        !            59:        .usage = "[-agoqu] " CMD_TARGET_WINDOW_USAGE " option [value]",
        !            60: 
        !            61:        .tflag = CMD_WINDOW_CANFAIL,
        !            62: 
        !            63:        .flags = CMD_AFTERHOOK,
        !            64:        .exec = cmd_set_option_exec
        !            65: };
        !            66: 
        !            67: static enum cmd_retval
        !            68: cmd_set_option_exec(struct cmd *self, struct cmdq_item *item)
        !            69: {
        !            70:        struct args                     *args = self->args;
        !            71:        int                              append = args_has(args, 'a');
        !            72:        struct cmd_find_state           *fs = &item->state.tflag;
        !            73:        struct session                  *s = fs->s;
        !            74:        struct winlink                  *wl = fs->wl;
        !            75:        struct window                   *w;
        !            76:        struct client                   *c;
        !            77:        enum options_table_scope         scope;
        !            78:        struct options                  *oo;
        !            79:        struct options_entry            *parent, *o;
        !            80:        const char                      *name, *value, *target;
        !            81:        int                              window, idx, already, error, ambiguous;
        !            82:        char                            *cause;
        !            83: 
        !            84:        /* Parse option name and index. */
        !            85:        name = options_match(args->argv[0], &idx, &ambiguous);
        !            86:        if (name == NULL) {
        !            87:                if (args_has(args, 'q'))
        !            88:                        return (CMD_RETURN_NORMAL);
        !            89:                if (ambiguous)
        !            90:                        cmdq_error(item, "ambiguous option: %s", args->argv[0]);
        !            91:                else
        !            92:                        cmdq_error(item, "invalid option: %s", args->argv[0]);
        !            93:                return (CMD_RETURN_ERROR);
        !            94:        }
        !            95:        if (args->argc < 2)
        !            96:                value = NULL;
        !            97:        else
        !            98:                value = args->argv[1];
        !            99: 
        !           100:        /*
        !           101:         * Figure out the scope: for user options it comes from the arguments,
        !           102:         * otherwise from the option name.
        !           103:         */
        !           104:        if (*name == '@') {
        !           105:                window = (self->entry == &cmd_set_window_option_entry);
        !           106:                scope = options_scope_from_flags(args, window, fs, &oo, &cause);
        !           107:        } else {
        !           108:                if (options_get_only(global_options, name) != NULL)
        !           109:                        scope = OPTIONS_TABLE_SERVER;
        !           110:                else if (options_get_only(global_s_options, name) != NULL)
        !           111:                        scope = OPTIONS_TABLE_SESSION;
        !           112:                else if (options_get_only(global_w_options, name) != NULL)
        !           113:                        scope = OPTIONS_TABLE_WINDOW;
        !           114:                else {
        !           115:                        scope = OPTIONS_TABLE_NONE;
        !           116:                        xasprintf(&cause, "unknown option: %s", args->argv[0]);
        !           117:                }
        !           118:        }
        !           119:        if (scope == OPTIONS_TABLE_NONE) {
        !           120:                if (args_has(args, 'q'))
        !           121:                        return (CMD_RETURN_NORMAL);
        !           122:                cmdq_error(item, "%s", cause);
        !           123:                free(cause);
        !           124:                return (CMD_RETURN_ERROR);
        !           125:        }
        !           126: 
        !           127:        /* Which table should this option go into? */
        !           128:        if (scope == OPTIONS_TABLE_SERVER)
        !           129:                oo = global_options;
        !           130:        else if (scope == OPTIONS_TABLE_SESSION) {
        !           131:                if (args_has(self->args, 'g'))
        !           132:                        oo = global_s_options;
        !           133:                else if (s == NULL) {
        !           134:                        target = args_get(args, 't');
        !           135:                        if (target != NULL)
        !           136:                                cmdq_error(item, "no such session: %s", target);
        !           137:                        else
        !           138:                                cmdq_error(item, "no current session");
        !           139:                        return (CMD_RETURN_ERROR);
        !           140:                } else
        !           141:                        oo = s->options;
        !           142:        } else if (scope == OPTIONS_TABLE_WINDOW) {
        !           143:                if (args_has(self->args, 'g'))
        !           144:                        oo = global_w_options;
        !           145:                else if (wl == NULL) {
        !           146:                        target = args_get(args, 't');
        !           147:                        if (target != NULL)
        !           148:                                cmdq_error(item, "no such window: %s", target);
        !           149:                        else
        !           150:                                cmdq_error(item, "no current window");
        !           151:                        return (CMD_RETURN_ERROR);
        !           152:                } else
        !           153:                        oo = wl->window->options;
        !           154:        }
        !           155:        o = options_get_only(oo, name);
        !           156:        parent = options_get(oo, name);
        !           157: 
        !           158:        /* Check that array options and indexes match up. */
        !           159:        if (idx != -1) {
        !           160:                if (*name == '@' || options_array_size(parent, NULL) == -1) {
        !           161:                        cmdq_error(item, "not an array: %s", args->argv[0]);
        !           162:                        return (CMD_RETURN_ERROR);
        !           163:                }
        !           164:        }
        !           165: 
        !           166:        /* With -o, check this option is not already set. */
        !           167:        if (!args_has(args, 'u') && args_has(args, 'o')) {
        !           168:                if (idx == -1)
        !           169:                        already = (o != NULL);
        !           170:                else {
        !           171:                        if (o == NULL)
        !           172:                                already = 0;
        !           173:                        else
        !           174:                                already = (options_array_get(o, idx) != NULL);
        !           175:                }
        !           176:                if (already) {
        !           177:                        if (args_has(args, 'q'))
        !           178:                                return (CMD_RETURN_NORMAL);
        !           179:                        cmdq_error(item, "already set: %s", args->argv[0]);
        !           180:                        return (CMD_RETURN_ERROR);
        !           181:                }
        !           182:        }
        !           183: 
        !           184:        /* Change the option. */
        !           185:        if (args_has(args, 'u')) {
        !           186:                if (o == NULL)
        !           187:                        return (CMD_RETURN_NORMAL);
        !           188:                if (idx == -1) {
        !           189:                        if (oo == global_options ||
        !           190:                            oo == global_s_options ||
        !           191:                            oo == global_w_options)
        !           192:                                options_default(oo, options_table_entry(o));
        !           193:                        else
        !           194:                                options_remove(o);
        !           195:                } else
        !           196:                        options_array_set(o, idx, NULL, 0);
        !           197:        } else if (*name == '@') {
        !           198:                if (value == NULL) {
        !           199:                        cmdq_error(item, "empty value");
        !           200:                        return (CMD_RETURN_ERROR);
        !           201:                }
        !           202:                options_set_string(oo, name, append, "%s", value);
        !           203:        } else if (idx == -1 && options_array_size(parent, NULL) == -1) {
        !           204:                error = cmd_set_option_set(self, item, oo, parent, value);
        !           205:                if (error != 0)
        !           206:                        return (CMD_RETURN_ERROR);
        !           207:        } else {
        !           208:                if (value == NULL) {
        !           209:                        cmdq_error(item, "empty value");
        !           210:                        return (CMD_RETURN_ERROR);
        !           211:                }
        !           212:                if (o == NULL)
        !           213:                        o = options_empty(oo, options_table_entry(parent));
        !           214:                if (idx == -1) {
        !           215:                        if (!append)
        !           216:                                options_array_clear(o);
        !           217:                        options_array_assign(o, value);
        !           218:                } else if (options_array_set(o, idx, value, append) != 0) {
        !           219:                        cmdq_error(item, "invalid index: %s", args->argv[0]);
        !           220:                        return (CMD_RETURN_ERROR);
        !           221:                }
        !           222:        }
        !           223: 
        !           224:        /* Update timers and so on for various options. */
        !           225:        if (strcmp(name, "automatic-rename") == 0) {
        !           226:                RB_FOREACH(w, windows, &windows) {
        !           227:                        if (w->active == NULL)
        !           228:                                continue;
        !           229:                        if (options_get_number(w->options, "automatic-rename"))
        !           230:                                w->active->flags |= PANE_CHANGED;
        !           231:                }
        !           232:        }
        !           233:        if (strcmp(name, "key-table") == 0) {
        !           234:                TAILQ_FOREACH(c, &clients, entry)
        !           235:                        server_client_set_key_table(c, NULL);
        !           236:        }
        !           237:        if (strcmp(name, "status") == 0 ||
        !           238:            strcmp(name, "status-interval") == 0)
        !           239:                status_timer_start_all();
        !           240:        if (strcmp(name, "monitor-silence") == 0)
        !           241:                alerts_reset_all();
        !           242:        if (strcmp(name, "window-style") == 0 ||
        !           243:            strcmp(name, "window-active-style") == 0) {
        !           244:                RB_FOREACH(w, windows, &windows)
        !           245:                        w->flags |= WINDOW_STYLECHANGED;
        !           246:        }
        !           247:        if (strcmp(name, "pane-border-status") == 0) {
        !           248:                RB_FOREACH(w, windows, &windows)
        !           249:                        layout_fix_panes(w, w->sx, w->sy);
        !           250:        }
        !           251:        RB_FOREACH (s, sessions, &sessions)
        !           252:                status_update_saved(s);
        !           253: 
        !           254:        /*
        !           255:         * Update sizes and redraw. May not always be necessary but do it
        !           256:         * anyway.
        !           257:         */
        !           258:        recalculate_sizes();
        !           259:        TAILQ_FOREACH(c, &clients, entry) {
        !           260:                if (c->session != NULL)
        !           261:                        server_redraw_client(c);
        !           262:        }
        !           263: 
        !           264:        return (CMD_RETURN_NORMAL);
        !           265: }
        !           266: 
        !           267: static int
        !           268: cmd_set_option_set(struct cmd *self, struct cmdq_item *item, struct options *oo,
        !           269:     struct options_entry *parent, const char *value)
        !           270: {
        !           271:        const struct options_table_entry        *oe;
        !           272:        struct args                             *args = self->args;
        !           273:        int                                      append = args_has(args, 'a');
        !           274:        struct options_entry                    *o;
        !           275:        long long                                number;
        !           276:        const char                              *errstr;
        !           277:        key_code                                 key;
        !           278: 
        !           279:        oe = options_table_entry(parent);
        !           280:        if (value == NULL &&
        !           281:            oe->type != OPTIONS_TABLE_FLAG &&
        !           282:            oe->type != OPTIONS_TABLE_CHOICE) {
        !           283:                cmdq_error(item, "empty value");
        !           284:                return (-1);
        !           285:        }
        !           286: 
        !           287:        switch (oe->type) {
        !           288:        case OPTIONS_TABLE_STRING:
        !           289:                options_set_string(oo, oe->name, append, "%s", value);
        !           290:                return (0);
        !           291:        case OPTIONS_TABLE_NUMBER:
        !           292:                number = strtonum(value, oe->minimum, oe->maximum, &errstr);
        !           293:                if (errstr != NULL) {
        !           294:                        cmdq_error(item, "value is %s: %s", errstr, value);
        !           295:                        return (-1);
        !           296:                }
        !           297:                options_set_number(oo, oe->name, number);
        !           298:                return (0);
        !           299:        case OPTIONS_TABLE_KEY:
        !           300:                key = key_string_lookup_string(value);
        !           301:                if (key == KEYC_UNKNOWN) {
        !           302:                        cmdq_error(item, "bad key: %s", value);
        !           303:                        return (-1);
        !           304:                }
        !           305:                options_set_number(oo, oe->name, key);
        !           306:                return (0);
        !           307:        case OPTIONS_TABLE_COLOUR:
        !           308:                if ((number = colour_fromstring(value)) == -1) {
        !           309:                        cmdq_error(item, "bad colour: %s", value);
        !           310:                        return (-1);
        !           311:                }
        !           312:                o = options_set_number(oo, oe->name, number);
        !           313:                options_style_update_new(oo, o);
        !           314:                return (0);
        !           315:        case OPTIONS_TABLE_ATTRIBUTES:
        !           316:                if ((number = attributes_fromstring(value)) == -1) {
        !           317:                        cmdq_error(item, "bad attributes: %s", value);
        !           318:                        return (-1);
        !           319:                }
        !           320:                o = options_set_number(oo, oe->name, number);
        !           321:                options_style_update_new(oo, o);
        !           322:                return (0);
        !           323:        case OPTIONS_TABLE_FLAG:
        !           324:                return (cmd_set_option_flag(item, oe, oo, value));
        !           325:        case OPTIONS_TABLE_CHOICE:
        !           326:                return (cmd_set_option_choice(item, oe, oo, value));
        !           327:        case OPTIONS_TABLE_STYLE:
        !           328:                o = options_set_style(oo, oe->name, append, value);
        !           329:                if (o == NULL) {
        !           330:                        cmdq_error(item, "bad style: %s", value);
        !           331:                        return (-1);
        !           332:                }
        !           333:                options_style_update_old(oo, o);
        !           334:                return (0);
        !           335:        case OPTIONS_TABLE_ARRAY:
        !           336:                break;
        !           337:        }
        !           338:        return (-1);
        !           339: }
        !           340: 
        !           341: static int
        !           342: cmd_set_option_flag(struct cmdq_item *item,
        !           343:     const struct options_table_entry *oe, struct options *oo,
        !           344:     const char *value)
        !           345: {
        !           346:        int     flag;
        !           347: 
        !           348:        if (value == NULL || *value == '\0')
        !           349:                flag = !options_get_number(oo, oe->name);
        !           350:        else if (strcmp(value, "1") == 0 ||
        !           351:            strcasecmp(value, "on") == 0 ||
        !           352:            strcasecmp(value, "yes") == 0)
        !           353:                flag = 1;
        !           354:        else if (strcmp(value, "0") == 0 ||
        !           355:            strcasecmp(value, "off") == 0 ||
        !           356:            strcasecmp(value, "no") == 0)
        !           357:                flag = 0;
        !           358:        else {
        !           359:                cmdq_error(item, "bad value: %s", value);
        !           360:                return (-1);
        !           361:        }
        !           362:        options_set_number(oo, oe->name, flag);
        !           363:        return (0);
        !           364: }
        !           365: 
        !           366: static int
        !           367: cmd_set_option_choice(struct cmdq_item *item,
        !           368:     const struct options_table_entry *oe, struct options *oo,
        !           369:     const char *value)
        !           370: {
        !           371:        const char      **cp;
        !           372:        int               n, choice = -1;
        !           373: 
        !           374:        if (value == NULL) {
        !           375:                choice = options_get_number(oo, oe->name);
        !           376:                if (choice < 2)
        !           377:                        choice = !choice;
        !           378:        } else {
        !           379:                n = 0;
        !           380:                for (cp = oe->choices; *cp != NULL; cp++) {
        !           381:                        if (strcmp(*cp, value) == 0)
        !           382:                                choice = n;
        !           383:                        n++;
        !           384:                }
        !           385:                if (choice == -1) {
        !           386:                        cmdq_error(item, "unknown value: %s", value);
        !           387:                        return (-1);
        !           388:                }
        !           389:        }
        !           390:        options_set_number(oo, oe->name, choice);
        !           391:        return (0);
        !           392: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>