xref: /minix3/external/bsd/tmux/dist/cmd-find-window.c (revision 0a6a1f1d05b60e214de2f05a7310ddd1f0e590e7)
1*0a6a1f1dSLionel Sambuc /* Id */
2eda6f593SDavid van Moolenbroek 
3eda6f593SDavid van Moolenbroek /*
4eda6f593SDavid van Moolenbroek  * Copyright (c) 2009 Nicholas Marriott <nicm@users.sourceforge.net>
5eda6f593SDavid van Moolenbroek  *
6eda6f593SDavid van Moolenbroek  * Permission to use, copy, modify, and distribute this software for any
7eda6f593SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
8eda6f593SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
9eda6f593SDavid van Moolenbroek  *
10eda6f593SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11eda6f593SDavid van Moolenbroek  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12eda6f593SDavid van Moolenbroek  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13eda6f593SDavid van Moolenbroek  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14eda6f593SDavid van Moolenbroek  * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
15eda6f593SDavid van Moolenbroek  * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
16eda6f593SDavid van Moolenbroek  * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17eda6f593SDavid van Moolenbroek  */
18eda6f593SDavid van Moolenbroek 
19eda6f593SDavid van Moolenbroek #include <sys/types.h>
20eda6f593SDavid van Moolenbroek 
21eda6f593SDavid van Moolenbroek #include <fnmatch.h>
22*0a6a1f1dSLionel Sambuc #include <stdlib.h>
23eda6f593SDavid van Moolenbroek #include <string.h>
24eda6f593SDavid van Moolenbroek 
25eda6f593SDavid van Moolenbroek #include "tmux.h"
26eda6f593SDavid van Moolenbroek 
27eda6f593SDavid van Moolenbroek /*
28eda6f593SDavid van Moolenbroek  * Find window containing text.
29eda6f593SDavid van Moolenbroek  */
30eda6f593SDavid van Moolenbroek 
31*0a6a1f1dSLionel Sambuc enum cmd_retval	 cmd_find_window_exec(struct cmd *, struct cmd_q *);
32eda6f593SDavid van Moolenbroek 
33*0a6a1f1dSLionel Sambuc void	cmd_find_window_callback(struct window_choose_data *);
34*0a6a1f1dSLionel Sambuc 
35*0a6a1f1dSLionel Sambuc /* Flags for determining matching behavior. */
36*0a6a1f1dSLionel Sambuc #define CMD_FIND_WINDOW_BY_TITLE   0x1
37*0a6a1f1dSLionel Sambuc #define CMD_FIND_WINDOW_BY_CONTENT 0x2
38*0a6a1f1dSLionel Sambuc #define CMD_FIND_WINDOW_BY_NAME    0x4
39*0a6a1f1dSLionel Sambuc 
40*0a6a1f1dSLionel Sambuc #define CMD_FIND_WINDOW_ALL		\
41*0a6a1f1dSLionel Sambuc 	(CMD_FIND_WINDOW_BY_TITLE |	\
42*0a6a1f1dSLionel Sambuc 	 CMD_FIND_WINDOW_BY_CONTENT |	\
43*0a6a1f1dSLionel Sambuc 	 CMD_FIND_WINDOW_BY_NAME)
44eda6f593SDavid van Moolenbroek 
45eda6f593SDavid van Moolenbroek const struct cmd_entry cmd_find_window_entry = {
46eda6f593SDavid van Moolenbroek 	"find-window", "findw",
47*0a6a1f1dSLionel Sambuc 	"F:CNt:T", 1, 4,
48*0a6a1f1dSLionel Sambuc 	"[-CNT] [-F format] " CMD_TARGET_WINDOW_USAGE " match-string",
49eda6f593SDavid van Moolenbroek 	0,
50eda6f593SDavid van Moolenbroek 	NULL,
51eda6f593SDavid van Moolenbroek 	cmd_find_window_exec
52eda6f593SDavid van Moolenbroek };
53eda6f593SDavid van Moolenbroek 
54eda6f593SDavid van Moolenbroek struct cmd_find_window_data {
55*0a6a1f1dSLionel Sambuc 	struct winlink	*wl;
56*0a6a1f1dSLionel Sambuc 	char		*list_ctx;
57*0a6a1f1dSLionel Sambuc 	u_int		 pane_id;
58eda6f593SDavid van Moolenbroek };
59*0a6a1f1dSLionel Sambuc ARRAY_DECL(cmd_find_window_data_list, struct cmd_find_window_data);
60eda6f593SDavid van Moolenbroek 
61*0a6a1f1dSLionel Sambuc u_int	cmd_find_window_match_flags(struct args *);
62*0a6a1f1dSLionel Sambuc void	cmd_find_window_match(struct cmd_find_window_data_list *, int,
63*0a6a1f1dSLionel Sambuc 	    struct winlink *, const char *, const char *);
64*0a6a1f1dSLionel Sambuc 
65*0a6a1f1dSLionel Sambuc u_int
cmd_find_window_match_flags(struct args * args)66*0a6a1f1dSLionel Sambuc cmd_find_window_match_flags(struct args *args)
67eda6f593SDavid van Moolenbroek {
68*0a6a1f1dSLionel Sambuc 	u_int	match_flags = 0;
69eda6f593SDavid van Moolenbroek 
70*0a6a1f1dSLionel Sambuc 	/* Turn on flags based on the options. */
71*0a6a1f1dSLionel Sambuc 	if (args_has(args, 'T'))
72*0a6a1f1dSLionel Sambuc 		match_flags |= CMD_FIND_WINDOW_BY_TITLE;
73*0a6a1f1dSLionel Sambuc 	if (args_has(args, 'C'))
74*0a6a1f1dSLionel Sambuc 		match_flags |= CMD_FIND_WINDOW_BY_CONTENT;
75*0a6a1f1dSLionel Sambuc 	if (args_has(args, 'N'))
76*0a6a1f1dSLionel Sambuc 		match_flags |= CMD_FIND_WINDOW_BY_NAME;
77*0a6a1f1dSLionel Sambuc 
78*0a6a1f1dSLionel Sambuc 	/* If none of the flags were set, default to matching anything. */
79*0a6a1f1dSLionel Sambuc 	if (match_flags == 0)
80*0a6a1f1dSLionel Sambuc 		match_flags = CMD_FIND_WINDOW_ALL;
81*0a6a1f1dSLionel Sambuc 
82*0a6a1f1dSLionel Sambuc 	return (match_flags);
83eda6f593SDavid van Moolenbroek }
84eda6f593SDavid van Moolenbroek 
85*0a6a1f1dSLionel Sambuc void
cmd_find_window_match(struct cmd_find_window_data_list * find_list,int match_flags,struct winlink * wl,const char * str,const char * searchstr)86*0a6a1f1dSLionel Sambuc cmd_find_window_match(struct cmd_find_window_data_list *find_list,
87*0a6a1f1dSLionel Sambuc     int match_flags, struct winlink *wl, const char *str, const char *searchstr)
88*0a6a1f1dSLionel Sambuc {
89*0a6a1f1dSLionel Sambuc 	struct cmd_find_window_data	 find_data;
90*0a6a1f1dSLionel Sambuc 	struct window_pane		*wp;
91*0a6a1f1dSLionel Sambuc 	u_int				 i, line;
92*0a6a1f1dSLionel Sambuc 	char				*sres;
93eda6f593SDavid van Moolenbroek 
94*0a6a1f1dSLionel Sambuc 	memset(&find_data, 0, sizeof find_data);
95eda6f593SDavid van Moolenbroek 
96eda6f593SDavid van Moolenbroek 	i = 0;
97*0a6a1f1dSLionel Sambuc 	TAILQ_FOREACH(wp, &wl->window->panes, entry) {
98eda6f593SDavid van Moolenbroek 		i++;
99eda6f593SDavid van Moolenbroek 
100*0a6a1f1dSLionel Sambuc 		if ((match_flags & CMD_FIND_WINDOW_BY_NAME) &&
101*0a6a1f1dSLionel Sambuc 		    fnmatch(searchstr, wl->window->name, 0) == 0) {
102*0a6a1f1dSLionel Sambuc 			find_data.list_ctx = xstrdup("");
103*0a6a1f1dSLionel Sambuc 			break;
104*0a6a1f1dSLionel Sambuc 		}
105eda6f593SDavid van Moolenbroek 
106*0a6a1f1dSLionel Sambuc 		if ((match_flags & CMD_FIND_WINDOW_BY_TITLE) &&
107*0a6a1f1dSLionel Sambuc 		    fnmatch(searchstr, wp->base.title, 0) == 0) {
108*0a6a1f1dSLionel Sambuc 			xasprintf(&find_data.list_ctx,
109*0a6a1f1dSLionel Sambuc 			    "pane %u title: \"%s\"", i - 1, wp->base.title);
110*0a6a1f1dSLionel Sambuc 			break;
111*0a6a1f1dSLionel Sambuc 		}
112*0a6a1f1dSLionel Sambuc 
113*0a6a1f1dSLionel Sambuc 		if (match_flags & CMD_FIND_WINDOW_BY_CONTENT &&
114*0a6a1f1dSLionel Sambuc 		    (sres = window_pane_search(wp, str, &line)) != NULL) {
115*0a6a1f1dSLionel Sambuc 			xasprintf(&find_data.list_ctx,
116*0a6a1f1dSLionel Sambuc 			    "pane %u line %u: \"%s\"", i - 1, line + 1, sres);
117*0a6a1f1dSLionel Sambuc 			free(sres);
118*0a6a1f1dSLionel Sambuc 			break;
119*0a6a1f1dSLionel Sambuc 		}
120*0a6a1f1dSLionel Sambuc 	}
121*0a6a1f1dSLionel Sambuc 	if (find_data.list_ctx != NULL) {
122*0a6a1f1dSLionel Sambuc 		find_data.wl = wl;
123*0a6a1f1dSLionel Sambuc 		find_data.pane_id = i - 1;
124*0a6a1f1dSLionel Sambuc 		ARRAY_ADD(find_list, find_data);
125eda6f593SDavid van Moolenbroek 	}
126eda6f593SDavid van Moolenbroek }
127eda6f593SDavid van Moolenbroek 
128*0a6a1f1dSLionel Sambuc enum cmd_retval
cmd_find_window_exec(struct cmd * self,struct cmd_q * cmdq)129*0a6a1f1dSLionel Sambuc cmd_find_window_exec(struct cmd *self, struct cmd_q *cmdq)
130*0a6a1f1dSLionel Sambuc {
131*0a6a1f1dSLionel Sambuc 	struct args			*args = self->args;
132*0a6a1f1dSLionel Sambuc 	struct client			*c;
133*0a6a1f1dSLionel Sambuc 	struct window_choose_data	*cdata;
134*0a6a1f1dSLionel Sambuc 	struct session			*s;
135*0a6a1f1dSLionel Sambuc 	struct winlink			*wl, *wm;
136*0a6a1f1dSLionel Sambuc 	struct cmd_find_window_data_list find_list;
137*0a6a1f1dSLionel Sambuc 	char				*str, *searchstr;
138*0a6a1f1dSLionel Sambuc 	const char			*template;
139*0a6a1f1dSLionel Sambuc 	u_int				 i, match_flags;
140eda6f593SDavid van Moolenbroek 
141*0a6a1f1dSLionel Sambuc 	if ((c = cmd_current_client(cmdq)) == NULL) {
142*0a6a1f1dSLionel Sambuc 		cmdq_error(cmdq, "no client available");
143*0a6a1f1dSLionel Sambuc 		return (CMD_RETURN_ERROR);
144*0a6a1f1dSLionel Sambuc 	}
145*0a6a1f1dSLionel Sambuc 	s = c->session;
146*0a6a1f1dSLionel Sambuc 
147*0a6a1f1dSLionel Sambuc 	if ((wl = cmd_find_window(cmdq, args_get(args, 't'), NULL)) == NULL)
148*0a6a1f1dSLionel Sambuc 		return (CMD_RETURN_ERROR);
149*0a6a1f1dSLionel Sambuc 
150*0a6a1f1dSLionel Sambuc 	if ((template = args_get(args, 'F')) == NULL)
151*0a6a1f1dSLionel Sambuc 		template = FIND_WINDOW_TEMPLATE;
152*0a6a1f1dSLionel Sambuc 
153*0a6a1f1dSLionel Sambuc 	match_flags = cmd_find_window_match_flags(args);
154*0a6a1f1dSLionel Sambuc 	str = args->argv[0];
155*0a6a1f1dSLionel Sambuc 
156*0a6a1f1dSLionel Sambuc 	ARRAY_INIT(&find_list);
157*0a6a1f1dSLionel Sambuc 
158*0a6a1f1dSLionel Sambuc 	xasprintf(&searchstr, "*%s*", str);
159*0a6a1f1dSLionel Sambuc 	RB_FOREACH(wm, winlinks, &s->windows)
160*0a6a1f1dSLionel Sambuc 	    cmd_find_window_match (&find_list, match_flags, wm, str, searchstr);
161*0a6a1f1dSLionel Sambuc 	free(searchstr);
162*0a6a1f1dSLionel Sambuc 
163*0a6a1f1dSLionel Sambuc 	if (ARRAY_LENGTH(&find_list) == 0) {
164*0a6a1f1dSLionel Sambuc 		cmdq_error(cmdq, "no windows matching: %s", str);
165*0a6a1f1dSLionel Sambuc 		ARRAY_FREE(&find_list);
166*0a6a1f1dSLionel Sambuc 		return (CMD_RETURN_ERROR);
167eda6f593SDavid van Moolenbroek 	}
168eda6f593SDavid van Moolenbroek 
169*0a6a1f1dSLionel Sambuc 	if (ARRAY_LENGTH(&find_list) == 1) {
170*0a6a1f1dSLionel Sambuc 		if (session_select(s, ARRAY_FIRST(&find_list).wl->idx) == 0)
171eda6f593SDavid van Moolenbroek 			server_redraw_session(s);
172eda6f593SDavid van Moolenbroek 		recalculate_sizes();
173eda6f593SDavid van Moolenbroek 		goto out;
174eda6f593SDavid van Moolenbroek 	}
175eda6f593SDavid van Moolenbroek 
176eda6f593SDavid van Moolenbroek 	if (window_pane_set_mode(wl->window->active, &window_choose_mode) != 0)
177eda6f593SDavid van Moolenbroek 		goto out;
178eda6f593SDavid van Moolenbroek 
179*0a6a1f1dSLionel Sambuc 	for (i = 0; i < ARRAY_LENGTH(&find_list); i++) {
180*0a6a1f1dSLionel Sambuc 		wm = ARRAY_ITEM(&find_list, i).wl;
181eda6f593SDavid van Moolenbroek 
182*0a6a1f1dSLionel Sambuc 		cdata = window_choose_data_create(TREE_OTHER, c, c->session);
183*0a6a1f1dSLionel Sambuc 		cdata->idx = wm->idx;
184*0a6a1f1dSLionel Sambuc 		cdata->wl = wm;
185*0a6a1f1dSLionel Sambuc 
186*0a6a1f1dSLionel Sambuc 		cdata->ft_template = xstrdup(template);
187*0a6a1f1dSLionel Sambuc 		cdata->pane_id = ARRAY_ITEM(&find_list, i).pane_id;
188*0a6a1f1dSLionel Sambuc 
189*0a6a1f1dSLionel Sambuc 		format_add(cdata->ft, "line", "%u", i);
190*0a6a1f1dSLionel Sambuc 		format_add(cdata->ft, "window_find_matches", "%s",
191*0a6a1f1dSLionel Sambuc 		    ARRAY_ITEM(&find_list, i).list_ctx);
192*0a6a1f1dSLionel Sambuc 		format_session(cdata->ft, s);
193*0a6a1f1dSLionel Sambuc 		format_winlink(cdata->ft, s, wm);
194*0a6a1f1dSLionel Sambuc 		format_window_pane(cdata->ft, wm->window->active);
195*0a6a1f1dSLionel Sambuc 
196*0a6a1f1dSLionel Sambuc 		window_choose_add(wl->window->active, cdata);
197eda6f593SDavid van Moolenbroek 	}
198eda6f593SDavid van Moolenbroek 
199*0a6a1f1dSLionel Sambuc 	window_choose_ready(wl->window->active, 0, cmd_find_window_callback);
200eda6f593SDavid van Moolenbroek 
201eda6f593SDavid van Moolenbroek out:
202*0a6a1f1dSLionel Sambuc 	ARRAY_FREE(&find_list);
203*0a6a1f1dSLionel Sambuc 	return (CMD_RETURN_NORMAL);
204eda6f593SDavid van Moolenbroek }
205eda6f593SDavid van Moolenbroek 
206eda6f593SDavid van Moolenbroek void
cmd_find_window_callback(struct window_choose_data * cdata)207*0a6a1f1dSLionel Sambuc cmd_find_window_callback(struct window_choose_data *cdata)
208eda6f593SDavid van Moolenbroek {
209*0a6a1f1dSLionel Sambuc 	struct session		*s;
210*0a6a1f1dSLionel Sambuc 	struct window_pane	*wp;
211eda6f593SDavid van Moolenbroek 
212*0a6a1f1dSLionel Sambuc 	if (cdata == NULL)
213eda6f593SDavid van Moolenbroek 		return;
214*0a6a1f1dSLionel Sambuc 
215*0a6a1f1dSLionel Sambuc 	s = cdata->start_session;
216eda6f593SDavid van Moolenbroek 	if (!session_alive(s))
217eda6f593SDavid van Moolenbroek 		return;
218eda6f593SDavid van Moolenbroek 
219*0a6a1f1dSLionel Sambuc 	wp = window_pane_at_index(cdata->wl->window, cdata->pane_id);
220*0a6a1f1dSLionel Sambuc 	if (wp != NULL && window_pane_visible(wp))
221*0a6a1f1dSLionel Sambuc 		window_set_active_pane(cdata->wl->window, wp);
222*0a6a1f1dSLionel Sambuc 
223*0a6a1f1dSLionel Sambuc 	if (session_select(s, cdata->idx) == 0) {
224eda6f593SDavid van Moolenbroek 		server_redraw_session(s);
225eda6f593SDavid van Moolenbroek 		recalculate_sizes();
226eda6f593SDavid van Moolenbroek 	}
227eda6f593SDavid van Moolenbroek }
228