Annotation of src/usr.bin/tmux/cmd-split-window.c, Revision 1.59
1.59 ! nicm 1: /* $OpenBSD: cmd-split-window.c,v 1.58 2015/10/23 16:30:15 nicm Exp $ */
1.1 nicm 2:
3: /*
4: * Copyright (c) 2009 Nicholas Marriott <nicm@users.sourceforge.net>
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.46 nicm 21: #include <errno.h>
22: #include <fcntl.h>
1.11 nicm 23: #include <paths.h>
1.1 nicm 24: #include <stdlib.h>
1.46 nicm 25: #include <string.h>
1.1 nicm 26: #include <unistd.h>
27:
28: #include "tmux.h"
29:
30: /*
31: * Split a window (add a new pane).
32: */
1.53 nicm 33:
34: #define SPLIT_WINDOW_TEMPLATE "#{session_name}:#{window_index}.#{pane_index}"
1.1 nicm 35:
1.40 nicm 36: enum cmd_retval cmd_split_window_exec(struct cmd *, struct cmd_q *);
1.1 nicm 37:
38: const struct cmd_entry cmd_split_window_entry = {
39: "split-window", "splitw",
1.54 nicm 40: "bc:dF:l:hp:Pt:v", 0, -1,
41: "[-bdhvP] [-c start-directory] [-F format] [-p percentage|-l size] "
1.37 nicm 42: CMD_TARGET_PANE_USAGE " [command]",
1.22 nicm 43: 0,
44: cmd_split_window_exec
1.1 nicm 45: };
46:
1.35 nicm 47: enum cmd_retval
1.40 nicm 48: cmd_split_window_exec(struct cmd *self, struct cmd_q *cmdq)
1.1 nicm 49: {
1.22 nicm 50: struct args *args = self->args;
51: struct session *s;
52: struct winlink *wl;
53: struct window *w;
54: struct window_pane *wp, *new_wp = NULL;
55: struct environ env;
1.49 nicm 56: const char *cmd, *path, *shell, *template;
1.50 nicm 57: char **argv, *cause, *new_cause, *cp;
1.30 nicm 58: u_int hlimit;
1.50 nicm 59: int argc, size, percentage, cwd, fd = -1;
1.22 nicm 60: enum layout_type type;
61: struct layout_cell *lc;
1.30 nicm 62: struct format_tree *ft;
1.49 nicm 63: struct environ_entry *envent;
1.1 nicm 64:
1.40 nicm 65: if ((wl = cmd_find_pane(cmdq, args_get(args, 't'), &s, &wp)) == NULL)
1.35 nicm 66: return (CMD_RETURN_ERROR);
1.1 nicm 67: w = wl->window;
1.41 nicm 68: server_unzoom_window(w);
1.1 nicm 69:
1.8 nicm 70: environ_init(&env);
71: environ_copy(&global_environ, &env);
72: environ_copy(&s->environ, &env);
73: server_fill_environ(s, &env);
1.1 nicm 74:
1.50 nicm 75: if (args->argc == 0) {
1.59 ! nicm 76: cmd = options_get_string(s->options, "default-command");
1.50 nicm 77: if (cmd != NULL && *cmd != '\0') {
78: argc = 1;
1.51 nicm 79: argv = (char **)&cmd;
1.50 nicm 80: } else {
81: argc = 0;
82: argv = NULL;
83: }
84: } else {
85: argc = args->argc;
86: argv = args->argv;
87: }
1.46 nicm 88:
89: if (args_has(args, 'c')) {
90: ft = format_create();
1.55 nicm 91: format_defaults(ft, cmd_find_client(cmdq, NULL, 1), s, NULL,
92: NULL);
1.46 nicm 93: cp = format_expand(ft, args_get(args, 'c'));
94: format_free(ft);
95:
1.48 nicm 96: if (cp != NULL && *cp != '\0') {
97: fd = open(cp, O_RDONLY|O_DIRECTORY);
98: free(cp);
99: if (fd == -1) {
100: cmdq_error(cmdq, "bad working directory: %s",
101: strerror(errno));
102: return (CMD_RETURN_ERROR);
103: }
1.58 nicm 104: } else
1.48 nicm 105: free(cp);
1.46 nicm 106: cwd = fd;
1.47 nicm 107: } else if (cmdq->client != NULL && cmdq->client->session == NULL)
1.46 nicm 108: cwd = cmdq->client->cwd;
109: else
110: cwd = s->cwd;
1.1 nicm 111:
1.15 nicm 112: type = LAYOUT_TOPBOTTOM;
1.22 nicm 113: if (args_has(args, 'h'))
1.15 nicm 114: type = LAYOUT_LEFTRIGHT;
115:
1.5 nicm 116: size = -1;
1.23 nicm 117: if (args_has(args, 'l')) {
118: size = args_strtonum(args, 'l', 0, INT_MAX, &cause);
1.22 nicm 119: if (cause != NULL) {
1.24 nicm 120: xasprintf(&new_cause, "size %s", cause);
1.34 nicm 121: free(cause);
1.24 nicm 122: cause = new_cause;
123: goto error;
1.22 nicm 124: }
125: } else if (args_has(args, 'p')) {
126: percentage = args_strtonum(args, 'p', 0, INT_MAX, &cause);
127: if (cause != NULL) {
1.24 nicm 128: xasprintf(&new_cause, "percentage %s", cause);
1.34 nicm 129: free(cause);
1.24 nicm 130: cause = new_cause;
131: goto error;
1.22 nicm 132: }
1.15 nicm 133: if (type == LAYOUT_TOPBOTTOM)
1.22 nicm 134: size = (wp->sy * percentage) / 100;
1.15 nicm 135: else
1.22 nicm 136: size = (wp->sx * percentage) / 100;
1.15 nicm 137: }
1.59 ! nicm 138: hlimit = options_get_number(s->options, "history-limit");
1.5 nicm 139:
1.59 ! nicm 140: shell = options_get_string(s->options, "default-shell");
1.11 nicm 141: if (*shell == '\0' || areshell(shell))
142: shell = _PATH_BSHELL;
143:
1.54 nicm 144: lc = layout_split_pane(wp, type, size, args_has(args, 'b'));
145: if (lc == NULL) {
1.5 nicm 146: cause = xstrdup("pane too small");
147: goto error;
1.1 nicm 148: }
1.19 nicm 149: new_wp = window_add_pane(w, hlimit);
1.57 nicm 150: layout_assign_pane(lc, new_wp);
1.49 nicm 151:
152: path = NULL;
153: if (cmdq->client != NULL && cmdq->client->session == NULL)
154: envent = environ_find(&cmdq->client->environ, "PATH");
155: else
156: envent = environ_find(&s->environ, "PATH");
157: if (envent != NULL)
158: path = envent->value;
159:
1.50 nicm 160: if (window_pane_spawn(new_wp, argc, argv, path, shell, cwd, &env,
161: s->tio, &cause) != 0)
1.19 nicm 162: goto error;
1.5 nicm 163:
1.1 nicm 164: server_redraw_window(w);
165:
1.22 nicm 166: if (!args_has(args, 'd')) {
1.17 nicm 167: window_set_active_pane(w, new_wp);
1.1 nicm 168: session_select(s, wl->idx);
169: server_redraw_session(s);
170: } else
171: server_status_session(s);
172:
1.8 nicm 173: environ_free(&env);
1.21 nicm 174:
1.22 nicm 175: if (args_has(args, 'P')) {
1.33 nicm 176: if ((template = args_get(args, 'F')) == NULL)
1.36 nicm 177: template = SPLIT_WINDOW_TEMPLATE;
1.30 nicm 178:
179: ft = format_create();
1.55 nicm 180: format_defaults(ft, cmd_find_client(cmdq, NULL, 1), s, wl,
181: new_wp);
1.30 nicm 182:
183: cp = format_expand(ft, template);
1.40 nicm 184: cmdq_print(cmdq, "%s", cp);
1.34 nicm 185: free(cp);
1.30 nicm 186:
187: format_free(ft);
1.21 nicm 188: }
1.32 nicm 189: notify_window_layout_changed(w);
1.46 nicm 190:
191: if (fd != -1)
192: close(fd);
1.35 nicm 193: return (CMD_RETURN_NORMAL);
1.5 nicm 194:
195: error:
1.8 nicm 196: environ_free(&env);
1.57 nicm 197: if (new_wp != NULL) {
198: layout_close_pane(new_wp);
1.17 nicm 199: window_remove_pane(w, new_wp);
1.57 nicm 200: }
1.40 nicm 201: cmdq_error(cmdq, "create pane failed: %s", cause);
1.34 nicm 202: free(cause);
1.46 nicm 203: if (fd != -1)
204: close(fd);
1.35 nicm 205: return (CMD_RETURN_ERROR);
1.1 nicm 206: }