mirror of
https://github.com/tmux/tmux.git
synced 2025-01-22 05:37:07 +00:00
Merge branch 'obsd-master'
This commit is contained in:
commit
2db9a18362
@ -30,8 +30,8 @@ const struct cmd_entry cmd_choose_tree_entry = {
|
|||||||
.name = "choose-tree",
|
.name = "choose-tree",
|
||||||
.alias = NULL,
|
.alias = NULL,
|
||||||
|
|
||||||
.args = { "F:Gf:NO:st:wZ", 0, 1 },
|
.args = { "F:Gf:NO:rst:wZ", 0, 1 },
|
||||||
.usage = "[-GNsw] [-F format] [-f filter] [-O sort-order] "
|
.usage = "[-GNrswZ] [-F format] [-f filter] [-O sort-order] "
|
||||||
CMD_TARGET_PANE_USAGE " [template]",
|
CMD_TARGET_PANE_USAGE " [template]",
|
||||||
|
|
||||||
.target = { 't', CMD_FIND_PANE, 0 },
|
.target = { 't', CMD_FIND_PANE, 0 },
|
||||||
@ -44,8 +44,8 @@ const struct cmd_entry cmd_choose_client_entry = {
|
|||||||
.name = "choose-client",
|
.name = "choose-client",
|
||||||
.alias = NULL,
|
.alias = NULL,
|
||||||
|
|
||||||
.args = { "F:f:NO:t:Z", 0, 1 },
|
.args = { "F:f:NO:rt:Z", 0, 1 },
|
||||||
.usage = "[-N] [-F format] [-f filter] [-O sort-order] "
|
.usage = "[-NrZ] [-F format] [-f filter] [-O sort-order] "
|
||||||
CMD_TARGET_PANE_USAGE " [template]",
|
CMD_TARGET_PANE_USAGE " [template]",
|
||||||
|
|
||||||
.target = { 't', CMD_FIND_PANE, 0 },
|
.target = { 't', CMD_FIND_PANE, 0 },
|
||||||
@ -58,8 +58,8 @@ const struct cmd_entry cmd_choose_buffer_entry = {
|
|||||||
.name = "choose-buffer",
|
.name = "choose-buffer",
|
||||||
.alias = NULL,
|
.alias = NULL,
|
||||||
|
|
||||||
.args = { "F:f:NO:t:Z", 0, 1 },
|
.args = { "F:f:NO:rt:Z", 0, 1 },
|
||||||
.usage = "[-N] [-F format] [-f filter] [-O sort-order] "
|
.usage = "[-NrZ] [-F format] [-f filter] [-O sort-order] "
|
||||||
CMD_TARGET_PANE_USAGE " [template]",
|
CMD_TARGET_PANE_USAGE " [template]",
|
||||||
|
|
||||||
.target = { 't', CMD_FIND_PANE, 0 },
|
.target = { 't', CMD_FIND_PANE, 0 },
|
||||||
|
25
mode-tree.c
25
mode-tree.c
@ -39,7 +39,7 @@ struct mode_tree_data {
|
|||||||
|
|
||||||
const char **sort_list;
|
const char **sort_list;
|
||||||
u_int sort_size;
|
u_int sort_size;
|
||||||
u_int sort_type;
|
struct mode_tree_sort_criteria sort_crit;
|
||||||
|
|
||||||
mode_tree_build_cb buildcb;
|
mode_tree_build_cb buildcb;
|
||||||
mode_tree_draw_cb drawcb;
|
mode_tree_draw_cb drawcb;
|
||||||
@ -334,7 +334,6 @@ mode_tree_start(struct window_pane *wp, struct args *args,
|
|||||||
|
|
||||||
mtd->sort_list = sort_list;
|
mtd->sort_list = sort_list;
|
||||||
mtd->sort_size = sort_size;
|
mtd->sort_size = sort_size;
|
||||||
mtd->sort_type = 0;
|
|
||||||
|
|
||||||
mtd->preview = !args_has(args, 'N');
|
mtd->preview = !args_has(args, 'N');
|
||||||
|
|
||||||
@ -342,9 +341,10 @@ mode_tree_start(struct window_pane *wp, struct args *args,
|
|||||||
if (sort != NULL) {
|
if (sort != NULL) {
|
||||||
for (i = 0; i < sort_size; i++) {
|
for (i = 0; i < sort_size; i++) {
|
||||||
if (strcasecmp(sort, sort_list[i]) == 0)
|
if (strcasecmp(sort, sort_list[i]) == 0)
|
||||||
mtd->sort_type = i;
|
mtd->sort_crit.field = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
mtd->sort_crit.reversed = args_has(args, 'r');
|
||||||
|
|
||||||
if (args_has(args, 'f'))
|
if (args_has(args, 'f'))
|
||||||
mtd->filter = xstrdup(args_get(args, 'f'));
|
mtd->filter = xstrdup(args_get(args, 'f'));
|
||||||
@ -392,10 +392,10 @@ mode_tree_build(struct mode_tree_data *mtd)
|
|||||||
TAILQ_CONCAT(&mtd->saved, &mtd->children, entry);
|
TAILQ_CONCAT(&mtd->saved, &mtd->children, entry);
|
||||||
TAILQ_INIT(&mtd->children);
|
TAILQ_INIT(&mtd->children);
|
||||||
|
|
||||||
mtd->buildcb(mtd->modedata, mtd->sort_type, &tag, mtd->filter);
|
mtd->buildcb(mtd->modedata, &mtd->sort_crit, &tag, mtd->filter);
|
||||||
mtd->no_matches = TAILQ_EMPTY(&mtd->children);
|
mtd->no_matches = TAILQ_EMPTY(&mtd->children);
|
||||||
if (mtd->no_matches)
|
if (mtd->no_matches)
|
||||||
mtd->buildcb(mtd->modedata, mtd->sort_type, &tag, NULL);
|
mtd->buildcb(mtd->modedata, &mtd->sort_crit, &tag, NULL);
|
||||||
|
|
||||||
mode_tree_free_items(&mtd->saved);
|
mode_tree_free_items(&mtd->saved);
|
||||||
TAILQ_INIT(&mtd->saved);
|
TAILQ_INIT(&mtd->saved);
|
||||||
@ -634,8 +634,9 @@ mode_tree_draw(struct mode_tree_data *mtd)
|
|||||||
screen_write_cursormove(&ctx, 0, h, 0);
|
screen_write_cursormove(&ctx, 0, h, 0);
|
||||||
screen_write_box(&ctx, w, sy - h);
|
screen_write_box(&ctx, w, sy - h);
|
||||||
|
|
||||||
xasprintf(&text, " %s (sort: %s)", mti->name,
|
xasprintf(&text, " %s (sort: %s%s)", mti->name,
|
||||||
mtd->sort_list[mtd->sort_type]);
|
mtd->sort_list[mtd->sort_crit.field],
|
||||||
|
mtd->sort_crit.reversed ? ", reversed" : "");
|
||||||
if (w - 2 >= strlen(text)) {
|
if (w - 2 >= strlen(text)) {
|
||||||
screen_write_cursormove(&ctx, 1, h, 0);
|
screen_write_cursormove(&ctx, 1, h, 0);
|
||||||
screen_write_puts(&ctx, &gc0, "%s", text);
|
screen_write_puts(&ctx, &gc0, "%s", text);
|
||||||
@ -993,9 +994,13 @@ mode_tree_key(struct mode_tree_data *mtd, struct client *c, key_code *key,
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'O':
|
case 'O':
|
||||||
mtd->sort_type++;
|
mtd->sort_crit.field++;
|
||||||
if (mtd->sort_type == mtd->sort_size)
|
if (mtd->sort_crit.field == mtd->sort_size)
|
||||||
mtd->sort_type = 0;
|
mtd->sort_crit.field = 0;
|
||||||
|
mode_tree_build(mtd);
|
||||||
|
break;
|
||||||
|
case 'r':
|
||||||
|
mtd->sort_crit.reversed = !mtd->sort_crit.reversed;
|
||||||
mode_tree_build(mtd);
|
mode_tree_build(mtd);
|
||||||
break;
|
break;
|
||||||
case KEYC_LEFT:
|
case KEYC_LEFT:
|
||||||
|
27
tmux.1
27
tmux.1
@ -1697,7 +1697,7 @@ the end of the visible pane.
|
|||||||
The default is to capture only the visible contents of the pane.
|
The default is to capture only the visible contents of the pane.
|
||||||
.It Xo
|
.It Xo
|
||||||
.Ic choose-client
|
.Ic choose-client
|
||||||
.Op Fl NZ
|
.Op Fl NrZ
|
||||||
.Op Fl F Ar format
|
.Op Fl F Ar format
|
||||||
.Op Fl f Ar filter
|
.Op Fl f Ar filter
|
||||||
.Op Fl O Ar sort-order
|
.Op Fl O Ar sort-order
|
||||||
@ -1726,7 +1726,8 @@ The following keys may be used in client mode:
|
|||||||
.It Li "z" Ta "Suspend selected client"
|
.It Li "z" Ta "Suspend selected client"
|
||||||
.It Li "Z" Ta "Suspend tagged clients"
|
.It Li "Z" Ta "Suspend tagged clients"
|
||||||
.It Li "f" Ta "Enter a format to filter items"
|
.It Li "f" Ta "Enter a format to filter items"
|
||||||
.It Li "O" Ta "Change sort order"
|
.It Li "O" Ta "Change sort field"
|
||||||
|
.It Li "r" Ta "Reverse sort order"
|
||||||
.It Li "v" Ta "Toggle preview"
|
.It Li "v" Ta "Toggle preview"
|
||||||
.It Li "q" Ta "Exit mode"
|
.It Li "q" Ta "Exit mode"
|
||||||
.El
|
.El
|
||||||
@ -1741,12 +1742,14 @@ If
|
|||||||
is not given, "detach-client -t '%%'" is used.
|
is not given, "detach-client -t '%%'" is used.
|
||||||
.Pp
|
.Pp
|
||||||
.Fl O
|
.Fl O
|
||||||
specifies the initial sort order: one of
|
specifies the initial sort field: one of
|
||||||
.Ql name ,
|
.Ql name ,
|
||||||
.Ql size ,
|
.Ql size ,
|
||||||
.Ql creation ,
|
.Ql creation ,
|
||||||
or
|
or
|
||||||
.Ql activity .
|
.Ql activity .
|
||||||
|
.Fl r
|
||||||
|
reverses the sort order.
|
||||||
.Fl f
|
.Fl f
|
||||||
specifies an initial filter: the filter is a format - if it evaluates to zero,
|
specifies an initial filter: the filter is a format - if it evaluates to zero,
|
||||||
the item in the list is not shown, otherwise it is shown.
|
the item in the list is not shown, otherwise it is shown.
|
||||||
@ -1758,7 +1761,7 @@ starts without the preview.
|
|||||||
This command works only if at least one client is attached.
|
This command works only if at least one client is attached.
|
||||||
.It Xo
|
.It Xo
|
||||||
.Ic choose-tree
|
.Ic choose-tree
|
||||||
.Op Fl GNswZ
|
.Op Fl GNrswZ
|
||||||
.Op Fl F Ar format
|
.Op Fl F Ar format
|
||||||
.Op Fl f Ar filter
|
.Op Fl f Ar filter
|
||||||
.Op Fl O Ar sort-order
|
.Op Fl O Ar sort-order
|
||||||
@ -1790,7 +1793,8 @@ The following keys may be used in tree mode:
|
|||||||
.It Li "C-t" Ta "Tag all items"
|
.It Li "C-t" Ta "Tag all items"
|
||||||
.It Li "\&:" Ta "Run a command for each tagged item"
|
.It Li "\&:" Ta "Run a command for each tagged item"
|
||||||
.It Li "f" Ta "Enter a format to filter items"
|
.It Li "f" Ta "Enter a format to filter items"
|
||||||
.It Li "O" Ta "Change sort order"
|
.It Li "O" Ta "Change sort field"
|
||||||
|
.It Li "r" Ta "Reverse sort order"
|
||||||
.It Li "v" Ta "Toggle preview"
|
.It Li "v" Ta "Toggle preview"
|
||||||
.It Li "q" Ta "Exit mode"
|
.It Li "q" Ta "Exit mode"
|
||||||
.El
|
.El
|
||||||
@ -1805,11 +1809,13 @@ If
|
|||||||
is not given, "switch-client -t '%%'" is used.
|
is not given, "switch-client -t '%%'" is used.
|
||||||
.Pp
|
.Pp
|
||||||
.Fl O
|
.Fl O
|
||||||
specifies the initial sort order: one of
|
specifies the initial sort field: one of
|
||||||
.Ql index ,
|
.Ql index ,
|
||||||
.Ql name ,
|
.Ql name ,
|
||||||
or
|
or
|
||||||
.Ql time .
|
.Ql time .
|
||||||
|
.Fl r
|
||||||
|
reverses the sort order.
|
||||||
.Fl f
|
.Fl f
|
||||||
specifies an initial filter: the filter is a format - if it evaluates to zero,
|
specifies an initial filter: the filter is a format - if it evaluates to zero,
|
||||||
the item in the list is not shown, otherwise it is shown.
|
the item in the list is not shown, otherwise it is shown.
|
||||||
@ -4867,7 +4873,7 @@ The buffer commands are as follows:
|
|||||||
.Bl -tag -width Ds
|
.Bl -tag -width Ds
|
||||||
.It Xo
|
.It Xo
|
||||||
.Ic choose-buffer
|
.Ic choose-buffer
|
||||||
.Op Fl NZ
|
.Op Fl NZr
|
||||||
.Op Fl F Ar format
|
.Op Fl F Ar format
|
||||||
.Op Fl f Ar filter
|
.Op Fl f Ar filter
|
||||||
.Op Fl O Ar sort-order
|
.Op Fl O Ar sort-order
|
||||||
@ -4894,7 +4900,8 @@ The following keys may be used in buffer mode:
|
|||||||
.It Li "d" Ta "Delete selected buffer"
|
.It Li "d" Ta "Delete selected buffer"
|
||||||
.It Li "D" Ta "Delete tagged buffers"
|
.It Li "D" Ta "Delete tagged buffers"
|
||||||
.It Li "f" Ta "Enter a format to filter items"
|
.It Li "f" Ta "Enter a format to filter items"
|
||||||
.It Li "O" Ta "Change sort order"
|
.It Li "O" Ta "Change sort field"
|
||||||
|
.It Li "r" Ta "Reverse sort order"
|
||||||
.It Li "v" Ta "Toggle preview"
|
.It Li "v" Ta "Toggle preview"
|
||||||
.It Li "q" Ta "Exit mode"
|
.It Li "q" Ta "Exit mode"
|
||||||
.El
|
.El
|
||||||
@ -4909,11 +4916,13 @@ If
|
|||||||
is not given, "paste-buffer -b '%%'" is used.
|
is not given, "paste-buffer -b '%%'" is used.
|
||||||
.Pp
|
.Pp
|
||||||
.Fl O
|
.Fl O
|
||||||
specifies the initial sort order: one of
|
specifies the initial sort field: one of
|
||||||
.Ql time ,
|
.Ql time ,
|
||||||
.Ql name
|
.Ql name
|
||||||
or
|
or
|
||||||
.Ql size .
|
.Ql size .
|
||||||
|
.Fl r
|
||||||
|
reverses the sort order.
|
||||||
.Fl f
|
.Fl f
|
||||||
specifies an initial filter: the filter is a format - if it evaluates to zero,
|
specifies an initial filter: the filter is a format - if it evaluates to zero,
|
||||||
the item in the list is not shown, otherwise it is shown.
|
the item in the list is not shown, otherwise it is shown.
|
||||||
|
9
tmux.h
9
tmux.h
@ -1684,6 +1684,12 @@ struct spawn_context {
|
|||||||
#define SPAWN_EMPTY 0x40
|
#define SPAWN_EMPTY 0x40
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* Mode tree sort order. */
|
||||||
|
struct mode_tree_sort_criteria {
|
||||||
|
u_int field;
|
||||||
|
int reversed;
|
||||||
|
};
|
||||||
|
|
||||||
/* tmux.c */
|
/* tmux.c */
|
||||||
extern struct options *global_options;
|
extern struct options *global_options;
|
||||||
extern struct options *global_s_options;
|
extern struct options *global_s_options;
|
||||||
@ -2475,7 +2481,8 @@ u_int layout_set_next(struct window *);
|
|||||||
u_int layout_set_previous(struct window *);
|
u_int layout_set_previous(struct window *);
|
||||||
|
|
||||||
/* mode-tree.c */
|
/* mode-tree.c */
|
||||||
typedef void (*mode_tree_build_cb)(void *, u_int, uint64_t *, const char *);
|
typedef void (*mode_tree_build_cb)(void *, struct mode_tree_sort_criteria *,
|
||||||
|
uint64_t *, const char *);
|
||||||
typedef void (*mode_tree_draw_cb)(void *, void *, struct screen_write_ctx *,
|
typedef void (*mode_tree_draw_cb)(void *, void *, struct screen_write_ctx *,
|
||||||
u_int, u_int);
|
u_int, u_int);
|
||||||
typedef int (*mode_tree_search_cb)(void *, void *, const char *);
|
typedef int (*mode_tree_search_cb)(void *, void *, const char *);
|
||||||
|
@ -74,6 +74,7 @@ static const char *window_buffer_sort_list[] = {
|
|||||||
"name",
|
"name",
|
||||||
"size"
|
"size"
|
||||||
};
|
};
|
||||||
|
static struct mode_tree_sort_criteria *window_buffer_sort;
|
||||||
|
|
||||||
struct window_buffer_itemdata {
|
struct window_buffer_itemdata {
|
||||||
const char *name;
|
const char *name;
|
||||||
@ -112,43 +113,29 @@ window_buffer_free_item(struct window_buffer_itemdata *item)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
window_buffer_cmp_name(const void *a0, const void *b0)
|
window_buffer_cmp(const void *a0, const void *b0)
|
||||||
{
|
{
|
||||||
const struct window_buffer_itemdata *const *a = a0;
|
const struct window_buffer_itemdata *const *a = a0;
|
||||||
const struct window_buffer_itemdata *const *b = b0;
|
const struct window_buffer_itemdata *const *b = b0;
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
return (strcmp((*a)->name, (*b)->name));
|
if (window_buffer_sort->field == WINDOW_BUFFER_BY_TIME)
|
||||||
}
|
result = (*b)->order - (*a)->order;
|
||||||
|
else if (window_buffer_sort->field == WINDOW_BUFFER_BY_SIZE)
|
||||||
|
result = (*b)->size - (*a)->size;
|
||||||
|
|
||||||
static int
|
/* Use WINDOW_BUFFER_BY_NAME as default order and tie breaker. */
|
||||||
window_buffer_cmp_time(const void *a0, const void *b0)
|
if (result == 0)
|
||||||
{
|
result = strcmp((*a)->name, (*b)->name);
|
||||||
const struct window_buffer_itemdata *const *a = a0;
|
|
||||||
const struct window_buffer_itemdata *const *b = b0;
|
|
||||||
|
|
||||||
if ((*a)->order > (*b)->order)
|
if (window_buffer_sort->reversed)
|
||||||
return (-1);
|
result = -result;
|
||||||
if ((*a)->order < (*b)->order)
|
return (result);
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->name, (*b)->name));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
window_buffer_cmp_size(const void *a0, const void *b0)
|
|
||||||
{
|
|
||||||
const struct window_buffer_itemdata *const *a = a0;
|
|
||||||
const struct window_buffer_itemdata *const *b = b0;
|
|
||||||
|
|
||||||
if ((*a)->size > (*b)->size)
|
|
||||||
return (-1);
|
|
||||||
if ((*a)->size < (*b)->size)
|
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->name, (*b)->name));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
window_buffer_build(void *modedata, u_int sort_type, __unused uint64_t *tag,
|
window_buffer_build(void *modedata, struct mode_tree_sort_criteria *sort_crit,
|
||||||
const char *filter)
|
__unused uint64_t *tag, const char *filter)
|
||||||
{
|
{
|
||||||
struct window_buffer_modedata *data = modedata;
|
struct window_buffer_modedata *data = modedata;
|
||||||
struct window_buffer_itemdata *item;
|
struct window_buffer_itemdata *item;
|
||||||
@ -174,20 +161,9 @@ window_buffer_build(void *modedata, u_int sort_type, __unused uint64_t *tag,
|
|||||||
item->order = paste_buffer_order(pb);
|
item->order = paste_buffer_order(pb);
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (sort_type) {
|
window_buffer_sort = sort_crit;
|
||||||
case WINDOW_BUFFER_BY_NAME:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
||||||
window_buffer_cmp_name);
|
window_buffer_cmp);
|
||||||
break;
|
|
||||||
case WINDOW_BUFFER_BY_TIME:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
|
||||||
window_buffer_cmp_time);
|
|
||||||
break;
|
|
||||||
case WINDOW_BUFFER_BY_SIZE:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
|
||||||
window_buffer_cmp_size);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cmd_find_valid_state(&data->fs)) {
|
if (cmd_find_valid_state(&data->fs)) {
|
||||||
s = data->fs.s;
|
s = data->fs.s;
|
||||||
|
@ -75,6 +75,7 @@ static const char *window_client_sort_list[] = {
|
|||||||
"creation",
|
"creation",
|
||||||
"activity"
|
"activity"
|
||||||
};
|
};
|
||||||
|
static struct mode_tree_sort_criteria *window_client_sort;
|
||||||
|
|
||||||
struct window_client_itemdata {
|
struct window_client_itemdata {
|
||||||
struct client *c;
|
struct client *c;
|
||||||
@ -110,60 +111,48 @@ window_client_free_item(struct window_client_itemdata *item)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
window_client_cmp_name(const void *a0, const void *b0)
|
window_client_cmp(const void *a0, const void *b0)
|
||||||
{
|
{
|
||||||
const struct window_client_itemdata *const *a = a0;
|
const struct window_client_itemdata *const *a = a0;
|
||||||
const struct window_client_itemdata *const *b = b0;
|
const struct window_client_itemdata *const *b = b0;
|
||||||
|
const struct window_client_itemdata *itema = *a;
|
||||||
|
const struct window_client_itemdata *itemb = *b;
|
||||||
|
struct client *ca = itema->c;
|
||||||
|
struct client *cb = itemb->c;
|
||||||
|
int result = 0;
|
||||||
|
|
||||||
return (strcmp((*a)->c->name, (*b)->c->name));
|
switch (window_client_sort->field) {
|
||||||
|
case WINDOW_CLIENT_BY_SIZE:
|
||||||
|
result = ca->tty.sx - cb->tty.sx;
|
||||||
|
if (result == 0)
|
||||||
|
result = ca->tty.sy - cb->tty.sy;
|
||||||
|
break;
|
||||||
|
case WINDOW_CLIENT_BY_CREATION_TIME:
|
||||||
|
if (timercmp(&ca->creation_time, &cb->creation_time, >))
|
||||||
|
result = -1;
|
||||||
|
else if (timercmp(&ca->creation_time, &cb->creation_time, <))
|
||||||
|
result = 1;
|
||||||
|
break;
|
||||||
|
case WINDOW_CLIENT_BY_ACTIVITY_TIME:
|
||||||
|
if (timercmp(&ca->activity_time, &cb->activity_time, >))
|
||||||
|
result = -1;
|
||||||
|
else if (timercmp(&ca->activity_time, &cb->activity_time, <))
|
||||||
|
result = 1;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
/* Use WINDOW_CLIENT_BY_NAME as default order and tie breaker. */
|
||||||
window_client_cmp_size(const void *a0, const void *b0)
|
if (result == 0)
|
||||||
{
|
result = strcmp(ca->name, cb->name);
|
||||||
const struct window_client_itemdata *const *a = a0;
|
|
||||||
const struct window_client_itemdata *const *b = b0;
|
|
||||||
|
|
||||||
if ((*a)->c->tty.sx < (*b)->c->tty.sx)
|
if (window_client_sort->reversed)
|
||||||
return (-1);
|
result = -result;
|
||||||
if ((*a)->c->tty.sx > (*b)->c->tty.sx)
|
return (result);
|
||||||
return (1);
|
|
||||||
if ((*a)->c->tty.sy < (*b)->c->tty.sy)
|
|
||||||
return (-1);
|
|
||||||
if ((*a)->c->tty.sy > (*b)->c->tty.sy)
|
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->c->name, (*b)->c->name));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
window_client_cmp_creation_time(const void *a0, const void *b0)
|
|
||||||
{
|
|
||||||
const struct window_client_itemdata *const *a = a0;
|
|
||||||
const struct window_client_itemdata *const *b = b0;
|
|
||||||
|
|
||||||
if (timercmp(&(*a)->c->creation_time, &(*b)->c->creation_time, >))
|
|
||||||
return (-1);
|
|
||||||
if (timercmp(&(*a)->c->creation_time, &(*b)->c->creation_time, <))
|
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->c->name, (*b)->c->name));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
window_client_cmp_activity_time(const void *a0, const void *b0)
|
|
||||||
{
|
|
||||||
const struct window_client_itemdata *const *a = a0;
|
|
||||||
const struct window_client_itemdata *const *b = b0;
|
|
||||||
|
|
||||||
if (timercmp(&(*a)->c->activity_time, &(*b)->c->activity_time, >))
|
|
||||||
return (-1);
|
|
||||||
if (timercmp(&(*a)->c->activity_time, &(*b)->c->activity_time, <))
|
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->c->name, (*b)->c->name));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
window_client_build(void *modedata, u_int sort_type, __unused uint64_t *tag,
|
window_client_build(void *modedata, struct mode_tree_sort_criteria *sort_crit,
|
||||||
const char *filter)
|
__unused uint64_t *tag, const char *filter)
|
||||||
{
|
{
|
||||||
struct window_client_modedata *data = modedata;
|
struct window_client_modedata *data = modedata;
|
||||||
struct window_client_itemdata *item;
|
struct window_client_itemdata *item;
|
||||||
@ -187,24 +176,9 @@ window_client_build(void *modedata, u_int sort_type, __unused uint64_t *tag,
|
|||||||
c->references++;
|
c->references++;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (sort_type) {
|
window_client_sort = sort_crit;
|
||||||
case WINDOW_CLIENT_BY_NAME:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
||||||
window_client_cmp_name);
|
window_client_cmp);
|
||||||
break;
|
|
||||||
case WINDOW_CLIENT_BY_SIZE:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
|
||||||
window_client_cmp_size);
|
|
||||||
break;
|
|
||||||
case WINDOW_CLIENT_BY_CREATION_TIME:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
|
||||||
window_client_cmp_creation_time);
|
|
||||||
break;
|
|
||||||
case WINDOW_CLIENT_BY_ACTIVITY_TIME:
|
|
||||||
qsort(data->item_list, data->item_size, sizeof *data->item_list,
|
|
||||||
window_client_cmp_activity_time);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < data->item_size; i++) {
|
for (i = 0; i < data->item_size; i++) {
|
||||||
item = data->item_list[i];
|
item = data->item_list[i];
|
||||||
|
158
window-tree.c
158
window-tree.c
@ -89,6 +89,7 @@ static const char *window_tree_sort_list[] = {
|
|||||||
"name",
|
"name",
|
||||||
"time"
|
"time"
|
||||||
};
|
};
|
||||||
|
static struct mode_tree_sort_criteria *window_tree_sort;
|
||||||
|
|
||||||
enum window_tree_type {
|
enum window_tree_type {
|
||||||
WINDOW_TREE_NONE,
|
WINDOW_TREE_NONE,
|
||||||
@ -184,62 +185,92 @@ window_tree_free_item(struct window_tree_itemdata *item)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
window_tree_cmp_session_name(const void *a0, const void *b0)
|
window_tree_cmp_session(const void *a0, const void *b0)
|
||||||
{
|
{
|
||||||
const struct session *const *a = a0;
|
const struct session *const *a = a0;
|
||||||
const struct session *const *b = b0;
|
const struct session *const *b = b0;
|
||||||
|
const struct session *sa = *a;
|
||||||
|
const struct session *sb = *b;
|
||||||
|
int result;
|
||||||
|
|
||||||
return (strcmp((*a)->name, (*b)->name));
|
switch (window_tree_sort->field) {
|
||||||
|
case WINDOW_TREE_BY_INDEX:
|
||||||
|
result = sa->id - sb->id;
|
||||||
|
break;
|
||||||
|
case WINDOW_TREE_BY_TIME:
|
||||||
|
if (timercmp(&sa->activity_time, &sb->activity_time, >)) {
|
||||||
|
result = -1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (timercmp(&sa->activity_time, &sb->activity_time, <)) {
|
||||||
|
result = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* FALLTHROUGH */
|
||||||
|
case WINDOW_TREE_BY_NAME:
|
||||||
|
result = strcmp(sa->name, sb->name);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window_tree_sort->reversed)
|
||||||
|
result = -result;
|
||||||
|
return (result);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
window_tree_cmp_session_time(const void *a0, const void *b0)
|
window_tree_cmp_window(const void *a0, const void *b0)
|
||||||
{
|
|
||||||
const struct session *const *a = a0;
|
|
||||||
const struct session *const *b = b0;
|
|
||||||
|
|
||||||
if (timercmp(&(*a)->activity_time, &(*b)->activity_time, >))
|
|
||||||
return (-1);
|
|
||||||
if (timercmp(&(*a)->activity_time, &(*b)->activity_time, <))
|
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->name, (*b)->name));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
window_tree_cmp_window_name(const void *a0, const void *b0)
|
|
||||||
{
|
{
|
||||||
const struct winlink *const *a = a0;
|
const struct winlink *const *a = a0;
|
||||||
const struct winlink *const *b = b0;
|
const struct winlink *const *b = b0;
|
||||||
|
const struct winlink *wla = *a;
|
||||||
|
const struct winlink *wlb = *b;
|
||||||
|
struct window *wa = wla->window;
|
||||||
|
struct window *wb = wlb->window;
|
||||||
|
int result;
|
||||||
|
|
||||||
return (strcmp((*a)->window->name, (*b)->window->name));
|
switch (window_tree_sort->field) {
|
||||||
|
case WINDOW_TREE_BY_INDEX:
|
||||||
|
result = wla->idx - wlb->idx;
|
||||||
|
break;
|
||||||
|
case WINDOW_TREE_BY_TIME:
|
||||||
|
if (timercmp(&wa->activity_time, &wb->activity_time, >)) {
|
||||||
|
result = -1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (timercmp(&wa->activity_time, &wb->activity_time, <)) {
|
||||||
|
result = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* FALLTHROUGH */
|
||||||
|
case WINDOW_TREE_BY_NAME:
|
||||||
|
result = strcmp(wa->name, wb->name);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (window_tree_sort->reversed)
|
||||||
|
result = -result;
|
||||||
|
return (result);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
window_tree_cmp_window_time(const void *a0, const void *b0)
|
window_tree_cmp_pane(const void *a0, const void *b0)
|
||||||
{
|
|
||||||
const struct winlink *const *a = a0;
|
|
||||||
const struct winlink *const *b = b0;
|
|
||||||
|
|
||||||
if (timercmp(&(*a)->window->activity_time,
|
|
||||||
&(*b)->window->activity_time, >))
|
|
||||||
return (-1);
|
|
||||||
if (timercmp(&(*a)->window->activity_time,
|
|
||||||
&(*b)->window->activity_time, <))
|
|
||||||
return (1);
|
|
||||||
return (strcmp((*a)->window->name, (*b)->window->name));
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
window_tree_cmp_pane_time(const void *a0, const void *b0)
|
|
||||||
{
|
{
|
||||||
const struct window_pane *const *a = a0;
|
const struct window_pane *const *a = a0;
|
||||||
const struct window_pane *const *b = b0;
|
const struct window_pane *const *b = b0;
|
||||||
|
int result;
|
||||||
|
|
||||||
if ((*a)->active_point < (*b)->active_point)
|
if (window_tree_sort->field == WINDOW_TREE_BY_TIME)
|
||||||
return (-1);
|
result = (*a)->active_point - (*b)->active_point;
|
||||||
if ((*a)->active_point > (*b)->active_point)
|
else {
|
||||||
return (1);
|
/*
|
||||||
return (0);
|
* Panes don't have names, so use number order for any other
|
||||||
|
* sort field.
|
||||||
|
*/
|
||||||
|
result = (*a)->id - (*b)->id;
|
||||||
|
}
|
||||||
|
if (window_tree_sort->reversed)
|
||||||
|
result *= -1;
|
||||||
|
return (result);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -285,8 +316,9 @@ window_tree_filter_pane(struct session *s, struct winlink *wl,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
window_tree_build_window(struct session *s, struct winlink *wl, void* modedata,
|
window_tree_build_window(struct session *s, struct winlink *wl,
|
||||||
u_int sort_type, struct mode_tree_item *parent, const char *filter)
|
void* modedata, struct mode_tree_sort_criteria *sort_crit,
|
||||||
|
struct mode_tree_item *parent, const char *filter)
|
||||||
{
|
{
|
||||||
struct window_tree_modedata *data = modedata;
|
struct window_tree_modedata *data = modedata;
|
||||||
struct window_tree_itemdata *item;
|
struct window_tree_itemdata *item;
|
||||||
@ -335,16 +367,8 @@ window_tree_build_window(struct session *s, struct winlink *wl, void* modedata,
|
|||||||
if (n == 0)
|
if (n == 0)
|
||||||
goto empty;
|
goto empty;
|
||||||
|
|
||||||
switch (sort_type) {
|
window_tree_sort = sort_crit;
|
||||||
case WINDOW_TREE_BY_INDEX:
|
qsort(l, n, sizeof *l, window_tree_cmp_pane);
|
||||||
break;
|
|
||||||
case WINDOW_TREE_BY_NAME:
|
|
||||||
/* Panes don't have names, so leave in number order. */
|
|
||||||
break;
|
|
||||||
case WINDOW_TREE_BY_TIME:
|
|
||||||
qsort(l, n, sizeof *l, window_tree_cmp_pane_time);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < n; i++)
|
for (i = 0; i < n; i++)
|
||||||
window_tree_build_pane(s, wl, l[i], modedata, mti);
|
window_tree_build_pane(s, wl, l[i], modedata, mti);
|
||||||
@ -360,7 +384,7 @@ empty:
|
|||||||
|
|
||||||
static void
|
static void
|
||||||
window_tree_build_session(struct session *s, void* modedata,
|
window_tree_build_session(struct session *s, void* modedata,
|
||||||
u_int sort_type, const char *filter)
|
struct mode_tree_sort_criteria *sort_crit, const char *filter)
|
||||||
{
|
{
|
||||||
struct window_tree_modedata *data = modedata;
|
struct window_tree_modedata *data = modedata;
|
||||||
struct window_tree_itemdata *item;
|
struct window_tree_itemdata *item;
|
||||||
@ -392,20 +416,12 @@ window_tree_build_session(struct session *s, void* modedata,
|
|||||||
l = xreallocarray(l, n + 1, sizeof *l);
|
l = xreallocarray(l, n + 1, sizeof *l);
|
||||||
l[n++] = wl;
|
l[n++] = wl;
|
||||||
}
|
}
|
||||||
switch (sort_type) {
|
window_tree_sort = sort_crit;
|
||||||
case WINDOW_TREE_BY_INDEX:
|
qsort(l, n, sizeof *l, window_tree_cmp_window);
|
||||||
break;
|
|
||||||
case WINDOW_TREE_BY_NAME:
|
|
||||||
qsort(l, n, sizeof *l, window_tree_cmp_window_name);
|
|
||||||
break;
|
|
||||||
case WINDOW_TREE_BY_TIME:
|
|
||||||
qsort(l, n, sizeof *l, window_tree_cmp_window_time);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
empty = 0;
|
empty = 0;
|
||||||
for (i = 0; i < n; i++) {
|
for (i = 0; i < n; i++) {
|
||||||
if (!window_tree_build_window(s, l[i], modedata, sort_type, mti,
|
if (!window_tree_build_window(s, l[i], modedata, sort_crit, mti,
|
||||||
filter))
|
filter))
|
||||||
empty++;
|
empty++;
|
||||||
}
|
}
|
||||||
@ -418,8 +434,8 @@ window_tree_build_session(struct session *s, void* modedata,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
window_tree_build(void *modedata, u_int sort_type, uint64_t *tag,
|
window_tree_build(void *modedata, struct mode_tree_sort_criteria *sort_crit,
|
||||||
const char *filter)
|
uint64_t *tag, const char *filter)
|
||||||
{
|
{
|
||||||
struct window_tree_modedata *data = modedata;
|
struct window_tree_modedata *data = modedata;
|
||||||
struct session *s, **l;
|
struct session *s, **l;
|
||||||
@ -446,19 +462,11 @@ window_tree_build(void *modedata, u_int sort_type, uint64_t *tag,
|
|||||||
l = xreallocarray(l, n + 1, sizeof *l);
|
l = xreallocarray(l, n + 1, sizeof *l);
|
||||||
l[n++] = s;
|
l[n++] = s;
|
||||||
}
|
}
|
||||||
switch (sort_type) {
|
window_tree_sort = sort_crit;
|
||||||
case WINDOW_TREE_BY_INDEX:
|
qsort(l, n, sizeof *l, window_tree_cmp_session);
|
||||||
break;
|
|
||||||
case WINDOW_TREE_BY_NAME:
|
|
||||||
qsort(l, n, sizeof *l, window_tree_cmp_session_name);
|
|
||||||
break;
|
|
||||||
case WINDOW_TREE_BY_TIME:
|
|
||||||
qsort(l, n, sizeof *l, window_tree_cmp_session_time);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (i = 0; i < n; i++)
|
for (i = 0; i < n; i++)
|
||||||
window_tree_build_session(l[i], modedata, sort_type, filter);
|
window_tree_build_session(l[i], modedata, sort_crit, filter);
|
||||||
free(l);
|
free(l);
|
||||||
|
|
||||||
switch (data->type) {
|
switch (data->type) {
|
||||||
|
Loading…
Reference in New Issue
Block a user