Annotation of src/usr.bin/tmux/cmd-new-window.c, Revision 1.68
1.68 ! nicm 1: /* $OpenBSD: cmd-new-window.c,v 1.67 2017/03/08 13:36:12 nicm Exp $ */
1.1 nicm 2:
3: /*
1.59 nicm 4: * Copyright (c) 2007 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:
1.35 nicm 21: #include <errno.h>
22: #include <fcntl.h>
1.1 nicm 23: #include <stdlib.h>
1.35 nicm 24: #include <string.h>
25: #include <unistd.h>
1.1 nicm 26:
27: #include "tmux.h"
28:
29: /*
30: * Create a new window.
31: */
1.43 nicm 32:
33: #define NEW_WINDOW_TEMPLATE "#{session_name}:#{window_index}.#{pane_index}"
1.1 nicm 34:
1.64 nicm 35: static enum cmd_retval cmd_new_window_exec(struct cmd *, struct cmdq_item *);
1.1 nicm 36:
37: const struct cmd_entry cmd_new_window_entry = {
1.57 nicm 38: .name = "new-window",
39: .alias = "neww",
40:
41: .args = { "ac:dF:kn:Pt:", 0, -1 },
42: .usage = "[-adkP] [-c start-directory] [-F format] [-n window-name] "
43: CMD_TARGET_WINDOW_USAGE " [command]",
44:
1.58 nicm 45: .tflag = CMD_WINDOW_INDEX,
46:
47: .flags = 0,
1.57 nicm 48: .exec = cmd_new_window_exec
1.1 nicm 49: };
50:
1.60 nicm 51: static enum cmd_retval
1.64 nicm 52: cmd_new_window_exec(struct cmd *self, struct cmdq_item *item)
1.1 nicm 53: {
1.21 nicm 54: struct args *args = self->args;
1.64 nicm 55: struct session *s = item->state.tflag.s;
56: struct winlink *wl = item->state.tflag.wl;
57: struct client *c = item->state.c;
58: int idx = item->state.tflag.idx;
1.50 nicm 59: const char *cmd, *path, *template, *cwd, *to_free;
1.40 nicm 60: char **argv, *cause, *cp;
1.55 nicm 61: int argc, detached;
1.39 nicm 62: struct environ_entry *envent;
1.62 nicm 63: struct cmd_find_state fs;
1.17 nicm 64:
65: if (args_has(args, 'a')) {
1.46 nicm 66: if ((idx = winlink_shuffle_up(s, wl)) == -1) {
1.64 nicm 67: cmdq_error(item, "no free window indexes");
1.26 nicm 68: return (CMD_RETURN_ERROR);
1.13 nicm 69: }
70: }
1.17 nicm 71: detached = args_has(args, 'd');
1.1 nicm 72:
1.40 nicm 73: if (args->argc == 0) {
1.48 nicm 74: cmd = options_get_string(s->options, "default-command");
1.40 nicm 75: if (cmd != NULL && *cmd != '\0') {
76: argc = 1;
1.41 nicm 77: argv = (char **)&cmd;
1.40 nicm 78: } else {
79: argc = 0;
80: argv = NULL;
81: }
82: } else {
83: argc = args->argc;
84: argv = args->argv;
85: }
1.35 nicm 86:
1.39 nicm 87: path = NULL;
1.64 nicm 88: if (item->client != NULL && item->client->session == NULL)
89: envent = environ_find(item->client->environ, "PATH");
1.39 nicm 90: else
1.49 nicm 91: envent = environ_find(s->environ, "PATH");
1.39 nicm 92: if (envent != NULL)
93: path = envent->value;
94:
1.50 nicm 95: to_free = NULL;
1.35 nicm 96: if (args_has(args, 'c')) {
1.67 nicm 97: cwd = args_get(args, 'c');
98: to_free = cwd = format_single(item, cwd, c, s, NULL, NULL);
1.64 nicm 99: } else if (item->client != NULL && item->client->session == NULL)
100: cwd = item->client->cwd;
1.35 nicm 101: else
102: cwd = s->cwd;
1.38 nicm 103:
104: wl = NULL;
105: if (idx != -1)
106: wl = winlink_find_by_index(&s->windows, idx);
107: if (wl != NULL && args_has(args, 'k')) {
108: /*
109: * Can't use session_detach as it will destroy session if this
110: * makes it empty.
111: */
1.65 nicm 112: notify_session_window("window-unlinked", s, wl->window);
1.38 nicm 113: wl->flags &= ~WINLINK_ALERTFLAGS;
114: winlink_stack_remove(&s->lastw, wl);
115: winlink_remove(&s->windows, wl);
116:
117: /* Force select/redraw if current. */
118: if (wl == s->curw) {
119: detached = 0;
120: s->curw = NULL;
121: }
122: }
1.1 nicm 123:
1.6 nicm 124: if (idx == -1)
1.48 nicm 125: idx = -1 - options_get_number(s->options, "base-index");
1.40 nicm 126: wl = session_new(s, args_get(args, 'n'), argc, argv, path, cwd, idx,
127: &cause);
1.1 nicm 128: if (wl == NULL) {
1.64 nicm 129: cmdq_error(item, "create window failed: %s", cause);
1.25 nicm 130: free(cause);
1.35 nicm 131: goto error;
1.1 nicm 132: }
1.17 nicm 133: if (!detached) {
1.1 nicm 134: session_select(s, wl->idx);
1.8 nicm 135: server_redraw_session_group(s);
1.11 nicm 136: } else
1.8 nicm 137: server_status_session_group(s);
1.1 nicm 138:
1.21 nicm 139: if (args_has(args, 'P')) {
1.24 nicm 140: if ((template = args_get(args, 'F')) == NULL)
1.27 nicm 141: template = NEW_WINDOW_TEMPLATE;
1.67 nicm 142: cp = format_single(item, template, c, s, wl, NULL);
1.64 nicm 143: cmdq_print(item, "%s", cp);
1.25 nicm 144: free(cp);
1.21 nicm 145: }
1.61 nicm 146:
1.50 nicm 147: if (to_free != NULL)
148: free((void *)to_free);
1.62 nicm 149:
1.68 ! nicm 150: cmd_find_from_winlink(&fs, wl);
1.64 nicm 151: hooks_insert(s->hooks, item, &fs, "after-new-window");
1.63 nicm 152:
1.26 nicm 153: return (CMD_RETURN_NORMAL);
1.35 nicm 154:
155: error:
1.50 nicm 156: if (to_free != NULL)
157: free((void *)to_free);
1.35 nicm 158: return (CMD_RETURN_ERROR);
1.1 nicm 159: }