Add formats to list sessions, windows or panes.

This commit is contained in:
nicm 2019-03-13 15:37:28 +00:00
parent 49f04a997a
commit 95ab1aaaec
2 changed files with 146 additions and 3 deletions

138
format.c
View File

@ -95,6 +95,9 @@ format_job_cmp(struct format_job *fj1, struct format_job *fj2)
#define FORMAT_QUOTE 0x8 #define FORMAT_QUOTE 0x8
#define FORMAT_LITERAL 0x10 #define FORMAT_LITERAL 0x10
#define FORMAT_EXPAND 0x20 #define FORMAT_EXPAND 0x20
#define FORMAT_SESSIONS 0x40
#define FORMAT_WINDOWS 0x80
#define FORMAT_PANES 0x100
/* Entry in format tree. */ /* Entry in format tree. */
struct format_entry { struct format_entry {
@ -113,6 +116,7 @@ struct format_tree {
struct window *w; struct window *w;
struct window_pane *wp; struct window_pane *wp;
struct cmdq_item *item;
struct client *client; struct client *client;
u_int tag; u_int tag;
int flags; int flags;
@ -673,6 +677,7 @@ format_create(struct client *c, struct cmdq_item *item, int tag, int flags)
ft->client = c; ft->client = c;
ft->client->references++; ft->client->references++;
} }
ft->item = item;
ft->tag = tag; ft->tag = tag;
ft->flags = flags; ft->flags = flags;
@ -1013,7 +1018,8 @@ format_build_modifiers(struct format_tree *ft, const char **s, u_int *count)
cp++; cp++;
/* Check single character modifiers with no arguments. */ /* Check single character modifiers with no arguments. */
if (strchr("lmCbdtqE", cp[0]) != NULL && format_is_end(cp[1])) { if (strchr("lmCbdtqESWP", cp[0]) != NULL &&
format_is_end(cp[1])) {
format_add_modifier(&list, count, cp, 1, NULL, 0); format_add_modifier(&list, count, cp, 1, NULL, 0);
cp++; cp++;
continue; continue;
@ -1127,6 +1133,111 @@ format_substitute(const char *source, const char *from, const char *to)
return (copy); return (copy);
} }
/* Loop over sessions. */
static char *
format_loop_sessions(struct format_tree *ft, const char *fmt)
{
struct cmdq_item *item = ft->item;
char *expanded, *value;
size_t valuelen;
struct session *s;
value = xcalloc(1, 1);
valuelen = 1;
RB_FOREACH(s, sessions, &sessions) {
expanded = format_single(item, fmt, ft->c, ft->s, NULL, NULL);
valuelen += strlen(expanded);
value = xrealloc(value, valuelen);
strlcat(value, expanded, valuelen);
free(expanded);
}
return (value);
}
/* Loop over windows. */
static char *
format_loop_windows(struct format_tree *ft, const char *fmt)
{
struct cmdq_item *item = ft->item;
char *all, *active, *use, *expanded, *value;
size_t valuelen;
struct winlink *wl;
if (ft->s == NULL)
return (NULL);
if (format_choose(ft, fmt, &all, &active, 0) != 0) {
all = xstrdup(fmt);
active = NULL;
}
value = xcalloc(1, 1);
valuelen = 1;
RB_FOREACH(wl, winlinks, &ft->s->windows) {
if (active != NULL && wl == ft->s->curw)
use = active;
else
use = all;
expanded = format_single(item, use, ft->c, ft->s, wl, NULL);
valuelen += strlen(expanded);
value = xrealloc(value, valuelen);
strlcat(value, expanded, valuelen);
free(expanded);
}
free(active);
free(all);
return (value);
}
/* Loop over panes. */
static char *
format_loop_panes(struct format_tree *ft, const char *fmt)
{
struct cmdq_item *item = ft->item;
char *all, *active, *use, *expanded, *value;
size_t valuelen;
struct window_pane *wp;
if (ft->w == NULL)
return (NULL);
if (format_choose(ft, fmt, &all, &active, 0) != 0) {
all = xstrdup(fmt);
active = NULL;
}
value = xcalloc(1, 1);
valuelen = 1;
TAILQ_FOREACH(wp, &ft->w->panes, entry) {
if (active != NULL && wp == ft->w->active)
use = active;
else
use = all;
expanded = format_single(item, use, ft->c, ft->s, ft->wl, wp);
valuelen += strlen(expanded);
value = xrealloc(value, valuelen);
strlcat(value, expanded, valuelen);
free(expanded);
}
free(active);
free(all);
return (value);
}
/* Replace a key. */ /* Replace a key. */
static int static int
format_replace(struct format_tree *ft, const char *key, size_t keylen, format_replace(struct format_tree *ft, const char *key, size_t keylen,
@ -1193,6 +1304,15 @@ format_replace(struct format_tree *ft, const char *key, size_t keylen,
case 'E': case 'E':
modifiers |= FORMAT_EXPAND; modifiers |= FORMAT_EXPAND;
break; break;
case 'S':
modifiers |= FORMAT_SESSIONS;
break;
case 'W':
modifiers |= FORMAT_WINDOWS;
break;
case 'P':
modifiers |= FORMAT_PANES;
break;
} }
} else if (fm->size == 2) { } else if (fm->size == 2) {
if (strcmp(fm->modifier, "||") == 0 || if (strcmp(fm->modifier, "||") == 0 ||
@ -1210,8 +1330,20 @@ format_replace(struct format_tree *ft, const char *key, size_t keylen,
goto done; goto done;
} }
/* Is this a comparison or a conditional? */ /* Is this a loop, comparison or condition? */
if (search != NULL) { if (modifiers & FORMAT_SESSIONS) {
value = format_loop_sessions(ft, copy);
if (value == NULL)
goto fail;
} else if (modifiers & FORMAT_WINDOWS) {
value = format_loop_windows(ft, copy);
if (value == NULL)
goto fail;
} else if (modifiers & FORMAT_PANES) {
value = format_loop_panes(ft, copy);
if (value == NULL)
goto fail;
} else if (search != NULL) {
/* Search in pane. */ /* Search in pane. */
if (wp == NULL) if (wp == NULL)
value = xstrdup("0"); value = xstrdup("0");

11
tmux.1
View File

@ -3719,6 +3719,17 @@ will expand the format twice, for example
is the result of expanding the content of the is the result of expanding the content of the
.Ic status-left .Ic status-left
option rather than the content itself. option rather than the content itself.
.Ql S: ,
.Ql W:
or
.Ql P:
will loop over each session, window or pane and insert the format once
for each. For windows and panes, two comma-separated formats may be
given, the second is used for the current window or active pane. For
example to get a list of windows formatted like the status line:
.Bd -literal -offset indent
#{W:#{E:window-status-format} ,#{E:window-status-current-format} }
.Ed
A prefix of the form A prefix of the form
.Ql s/foo/bar/: .Ql s/foo/bar/:
will substitute will substitute