Annotation of src/usr.bin/tmux/cmd-split-window.c, Revision 1.18
1.18 ! nicm 1: /* $OpenBSD: cmd-split-window.c,v 1.17 2010/01/07 19:47:10 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.11 nicm 21: #include <paths.h>
1.1 nicm 22: #include <stdlib.h>
23: #include <unistd.h>
24:
25: #include "tmux.h"
26:
27: /*
28: * Split a window (add a new pane).
29: */
30:
31: int cmd_split_window_parse(struct cmd *, int, char **, char **);
32: int cmd_split_window_exec(struct cmd *, struct cmd_ctx *);
33: void cmd_split_window_free(struct cmd *);
34: void cmd_split_window_init(struct cmd *, int);
35: size_t cmd_split_window_print(struct cmd *, char *, size_t);
36:
37: struct cmd_split_window_data {
38: char *target;
39: char *cmd;
40: int flag_detached;
1.5 nicm 41: int flag_horizontal;
1.1 nicm 42: int percentage;
1.5 nicm 43: int size;
1.1 nicm 44: };
45:
46: const struct cmd_entry cmd_split_window_entry = {
47: "split-window", "splitw",
1.17 nicm 48: "[-dhv] [-p percentage|-l size] [-t target-pane] [command]",
1.14 nicm 49: 0, "",
1.1 nicm 50: cmd_split_window_init,
51: cmd_split_window_parse,
52: cmd_split_window_exec,
53: cmd_split_window_free,
54: cmd_split_window_print
55: };
56:
57: void
1.5 nicm 58: cmd_split_window_init(struct cmd *self, int key)
1.1 nicm 59: {
60: struct cmd_split_window_data *data;
61:
62: self->data = data = xmalloc(sizeof *data);
63: data->target = NULL;
64: data->cmd = NULL;
65: data->flag_detached = 0;
1.5 nicm 66: data->flag_horizontal = 0;
1.1 nicm 67: data->percentage = -1;
1.5 nicm 68: data->size = -1;
69:
70: switch (key) {
71: case '%':
72: data->flag_horizontal = 1;
1.16 nicm 73: break;
1.5 nicm 74: case '"':
75: data->flag_horizontal = 0;
76: break;
77: }
1.1 nicm 78: }
79:
80: int
81: cmd_split_window_parse(struct cmd *self, int argc, char **argv, char **cause)
82: {
83: struct cmd_split_window_data *data;
1.5 nicm 84: int opt;
1.1 nicm 85: const char *errstr;
86:
1.13 nicm 87: self->entry->init(self, KEYC_NONE);
1.1 nicm 88: data = self->data;
89:
1.5 nicm 90: while ((opt = getopt(argc, argv, "dhl:p:t:v")) != -1) {
1.1 nicm 91: switch (opt) {
92: case 'd':
93: data->flag_detached = 1;
94: break;
1.5 nicm 95: case 'h':
96: data->flag_horizontal = 1;
97: break;
1.1 nicm 98: case 't':
99: if (data->target == NULL)
100: data->target = xstrdup(optarg);
101: break;
102: case 'l':
1.5 nicm 103: if (data->percentage != -1 || data->size != -1)
104: break;
105: data->size = strtonum(optarg, 1, INT_MAX, &errstr);
106: if (errstr != NULL) {
107: xasprintf(cause, "size %s", errstr);
108: goto error;
1.1 nicm 109: }
110: break;
111: case 'p':
1.5 nicm 112: if (data->size != -1 || data->percentage != -1)
113: break;
114: data->percentage = strtonum(optarg, 1, 100, &errstr);
115: if (errstr != NULL) {
116: xasprintf(cause, "percentage %s", errstr);
117: goto error;
1.1 nicm 118: }
119: break;
1.5 nicm 120: case 'v':
121: data->flag_horizontal = 0;
122: break;
1.1 nicm 123: default:
124: goto usage;
125: }
126: }
127: argc -= optind;
128: argv += optind;
129: if (argc != 0 && argc != 1)
130: goto usage;
131:
132: if (argc == 1)
133: data->cmd = xstrdup(argv[0]);
134:
135: return (0);
136:
137: usage:
138: xasprintf(cause, "usage: %s %s", self->entry->name, self->entry->usage);
139:
140: error:
141: self->entry->free(self);
142: return (-1);
143: }
144:
145: int
146: cmd_split_window_exec(struct cmd *self, struct cmd_ctx *ctx)
147: {
148: struct cmd_split_window_data *data = self->data;
149: struct session *s;
150: struct winlink *wl;
151: struct window *w;
1.17 nicm 152: struct window_pane *wp, *new_wp;
1.8 nicm 153: struct environ env;
1.1 nicm 154: char *cmd, *cwd, *cause;
1.11 nicm 155: const char *shell;
1.2 nicm 156: u_int hlimit;
1.5 nicm 157: int size;
158: enum layout_type type;
1.1 nicm 159:
1.17 nicm 160: if ((wl = cmd_find_pane(ctx, data->target, &s, &wp)) == NULL)
1.1 nicm 161: return (-1);
162: w = wl->window;
163:
1.8 nicm 164: environ_init(&env);
165: environ_copy(&global_environ, &env);
166: environ_copy(&s->environ, &env);
167: server_fill_environ(s, &env);
1.1 nicm 168:
169: cmd = data->cmd;
170: if (cmd == NULL)
171: cmd = options_get_string(&s->options, "default-command");
172: if (ctx->cmdclient == NULL || ctx->cmdclient->cwd == NULL)
1.3 nicm 173: cwd = options_get_string(&s->options, "default-path");
1.1 nicm 174: else
175: cwd = ctx->cmdclient->cwd;
176:
1.15 nicm 177: type = LAYOUT_TOPBOTTOM;
178: if (data->flag_horizontal)
179: type = LAYOUT_LEFTRIGHT;
180:
1.5 nicm 181: size = -1;
182: if (data->size != -1)
183: size = data->size;
1.15 nicm 184: else if (data->percentage != -1) {
185: if (type == LAYOUT_TOPBOTTOM)
1.18 ! nicm 186: size = (wp->sy * data->percentage) / 100;
1.15 nicm 187: else
1.18 ! nicm 188: size = (wp->sx * data->percentage) / 100;
1.15 nicm 189: }
1.5 nicm 190: hlimit = options_get_number(&s->options, "history-limit");
191:
1.11 nicm 192: shell = options_get_string(&s->options, "default-shell");
193: if (*shell == '\0' || areshell(shell))
194: shell = _PATH_BSHELL;
195:
1.17 nicm 196: new_wp = window_add_pane(w, hlimit);
197: if (window_pane_spawn(new_wp, cmd, shell, cwd, &env, s->tio, &cause) != 0)
1.5 nicm 198: goto error;
1.17 nicm 199: if (layout_split_pane(wp, type, size, new_wp) != 0) {
1.5 nicm 200: cause = xstrdup("pane too small");
201: goto error;
1.1 nicm 202: }
1.5 nicm 203:
1.1 nicm 204: server_redraw_window(w);
205:
206: if (!data->flag_detached) {
1.17 nicm 207: window_set_active_pane(w, new_wp);
1.1 nicm 208: session_select(s, wl->idx);
209: server_redraw_session(s);
210: } else
211: server_status_session(s);
212:
1.8 nicm 213: environ_free(&env);
1.1 nicm 214: return (0);
1.5 nicm 215:
216: error:
1.8 nicm 217: environ_free(&env);
1.17 nicm 218: if (new_wp != NULL)
219: window_remove_pane(w, new_wp);
1.5 nicm 220: ctx->error(ctx, "create pane failed: %s", cause);
221: xfree(cause);
222: return (-1);
1.1 nicm 223: }
224:
225: void
226: cmd_split_window_free(struct cmd *self)
227: {
228: struct cmd_split_window_data *data = self->data;
229:
230: if (data->target != NULL)
231: xfree(data->target);
232: if (data->cmd != NULL)
233: xfree(data->cmd);
234: xfree(data);
235: }
236:
237: size_t
238: cmd_split_window_print(struct cmd *self, char *buf, size_t len)
239: {
240: struct cmd_split_window_data *data = self->data;
241: size_t off = 0;
242:
243: off += xsnprintf(buf, len, "%s", self->entry->name);
244: if (data == NULL)
245: return (off);
246: if (off < len && data->flag_detached)
247: off += xsnprintf(buf + off, len - off, " -d");
1.5 nicm 248: if (off < len && data->flag_horizontal)
249: off += xsnprintf(buf + off, len - off, " -h");
1.10 nicm 250: if (off < len && data->size > 0)
251: off += xsnprintf(buf + off, len - off, " -l %d", data->size);
252: if (off < len && data->percentage > 0) {
253: off += xsnprintf(
254: buf + off, len - off, " -p %d", data->percentage);
255: }
1.1 nicm 256: if (off < len && data->target != NULL)
257: off += cmd_prarg(buf + off, len - off, " -t ", data->target);
258: if (off < len && data->cmd != NULL)
259: off += cmd_prarg(buf + off, len - off, " ", data->cmd);
260: return (off);
261: }