mirror of
https://github.com/tmux/tmux.git
synced 2025-01-12 03:08:46 +00:00
646995384d
been a limitation for a long time. There are two new options, window-size and default-size, and a new command, resize-window. The force-width and force-height options and the session_width and session_height formats have been removed. The new window-size option tells tmux how to work out the size of windows: largest means it picks the size of the largest session, smallest the smallest session (similar to the old behaviour) and manual means that it does not automatically resize windows. The default is currently largest but this may change. aggressive-resize modifies the choice of session for largest and smallest as it did before. If a window is in a session attached to a client that is too small, only part of the window is shown. tmux attempts to keep the cursor visible, so the part of the window displayed is changed as the cursor moves (with a small delay, to try and avoid excess redrawing when applications redraw status lines or similar that are not currently visible). The offset of the visible portion of the window is shown in status-right. Drawing windows which are larger than the client is not as efficient as those which fit, particularly when the cursor moves, so it is recommended to avoid using this on slow machines or networks (set window-size to smallest or manual). The resize-window command can be used to resize a window manually. If it is used, the window-size option is automatically set to manual for the window (undo this with "setw -u window-size"). resize-window works in a similar way to resize-pane (-U -D -L -R -x -y flags) but also has -a and -A flags. -a sets the window to the size of the smallest client (what it would be if window-size was smallest) and -A the largest. For the same behaviour as force-width or force-height, use resize-window -x or -y, and "setw -u window-size" to revert to automatic sizing.. If the global window-size option is set to manual, the default-size option is used for new windows. If -x or -y is used with new-session, that sets the default-size option for the new session. The maximum size of a window is 10000x10000. But expect applications to complain and much higher memory use if making a window excessively big. The minimum size is the size required for the current layout including borders. The refresh-client command can be used to pan around a window, -U -D -L -R moves up, down, left or right and -c returns to automatic cursor tracking. The position is reset when the current window is changed.
1060 lines
26 KiB
C
1060 lines
26 KiB
C
/* $OpenBSD$ */
|
|
|
|
/*
|
|
* Copyright (c) 2009 Nicholas Marriott <nicholas.marriott@gmail.com>
|
|
* Copyright (c) 2016 Stephen Kent <smkent@smkent.net>
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
|
|
* IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "tmux.h"
|
|
|
|
/*
|
|
* The window layout is a tree of cells each of which can be one of: a
|
|
* left-right container for a list of cells, a top-bottom container for a list
|
|
* of cells, or a container for a window pane.
|
|
*
|
|
* Each window has a pointer to the root of its layout tree (containing its
|
|
* panes), every pane has a pointer back to the cell containing it, and each
|
|
* cell a pointer to its parent cell.
|
|
*/
|
|
|
|
static u_int layout_resize_check(struct window *, struct layout_cell *,
|
|
enum layout_type);
|
|
static int layout_resize_pane_grow(struct window *, struct layout_cell *,
|
|
enum layout_type, int, int);
|
|
static int layout_resize_pane_shrink(struct window *, struct layout_cell *,
|
|
enum layout_type, int);
|
|
static int layout_need_status(struct layout_cell *, int);
|
|
static u_int layout_new_pane_size(struct window *, u_int,
|
|
struct layout_cell *, enum layout_type, u_int, u_int,
|
|
u_int);
|
|
static int layout_set_size_check(struct window *, struct layout_cell *,
|
|
enum layout_type, int);
|
|
static void layout_resize_child_cells(struct window *,
|
|
struct layout_cell *);
|
|
|
|
struct layout_cell *
|
|
layout_create_cell(struct layout_cell *lcparent)
|
|
{
|
|
struct layout_cell *lc;
|
|
|
|
lc = xmalloc(sizeof *lc);
|
|
lc->type = LAYOUT_WINDOWPANE;
|
|
lc->parent = lcparent;
|
|
|
|
TAILQ_INIT(&lc->cells);
|
|
|
|
lc->sx = UINT_MAX;
|
|
lc->sy = UINT_MAX;
|
|
|
|
lc->xoff = UINT_MAX;
|
|
lc->yoff = UINT_MAX;
|
|
|
|
lc->wp = NULL;
|
|
|
|
return (lc);
|
|
}
|
|
|
|
void
|
|
layout_free_cell(struct layout_cell *lc)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
|
|
switch (lc->type) {
|
|
case LAYOUT_LEFTRIGHT:
|
|
case LAYOUT_TOPBOTTOM:
|
|
while (!TAILQ_EMPTY(&lc->cells)) {
|
|
lcchild = TAILQ_FIRST(&lc->cells);
|
|
TAILQ_REMOVE(&lc->cells, lcchild, entry);
|
|
layout_free_cell(lcchild);
|
|
}
|
|
break;
|
|
case LAYOUT_WINDOWPANE:
|
|
if (lc->wp != NULL)
|
|
lc->wp->layout_cell = NULL;
|
|
break;
|
|
}
|
|
|
|
free(lc);
|
|
}
|
|
|
|
void
|
|
layout_print_cell(struct layout_cell *lc, const char *hdr, u_int n)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
const char *type;
|
|
|
|
switch (lc->type) {
|
|
case LAYOUT_LEFTRIGHT:
|
|
type = "LEFTRIGHT";
|
|
break;
|
|
case LAYOUT_TOPBOTTOM:
|
|
type = "TOPBOTTOM";
|
|
break;
|
|
case LAYOUT_WINDOWPANE:
|
|
type = "WINDOWPANE";
|
|
break;
|
|
default:
|
|
type = "UNKNOWN";
|
|
break;
|
|
}
|
|
log_debug("%s:%*s%p type %s [parent %p] wp=%p [%u,%u %ux%u]", hdr, n,
|
|
" ", lc, type, lc->parent, lc->wp, lc->xoff, lc->yoff, lc->sx,
|
|
lc->sy);
|
|
switch (lc->type) {
|
|
case LAYOUT_LEFTRIGHT:
|
|
case LAYOUT_TOPBOTTOM:
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry)
|
|
layout_print_cell(lcchild, hdr, n + 1);
|
|
break;
|
|
case LAYOUT_WINDOWPANE:
|
|
break;
|
|
}
|
|
}
|
|
|
|
struct layout_cell *
|
|
layout_search_by_border(struct layout_cell *lc, u_int x, u_int y)
|
|
{
|
|
struct layout_cell *lcchild, *last = NULL;
|
|
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
if (x >= lcchild->xoff && x < lcchild->xoff + lcchild->sx &&
|
|
y >= lcchild->yoff && y < lcchild->yoff + lcchild->sy) {
|
|
/* Inside the cell - recurse. */
|
|
return (layout_search_by_border(lcchild, x, y));
|
|
}
|
|
|
|
if (last == NULL) {
|
|
last = lcchild;
|
|
continue;
|
|
}
|
|
|
|
switch (lc->type) {
|
|
case LAYOUT_LEFTRIGHT:
|
|
if (x < lcchild->xoff && x >= last->xoff + last->sx)
|
|
return (last);
|
|
break;
|
|
case LAYOUT_TOPBOTTOM:
|
|
if (y < lcchild->yoff && y >= last->yoff + last->sy)
|
|
return (last);
|
|
break;
|
|
case LAYOUT_WINDOWPANE:
|
|
break;
|
|
}
|
|
|
|
last = lcchild;
|
|
}
|
|
|
|
return (NULL);
|
|
}
|
|
|
|
void
|
|
layout_set_size(struct layout_cell *lc, u_int sx, u_int sy, u_int xoff,
|
|
u_int yoff)
|
|
{
|
|
lc->sx = sx;
|
|
lc->sy = sy;
|
|
|
|
lc->xoff = xoff;
|
|
lc->yoff = yoff;
|
|
}
|
|
|
|
void
|
|
layout_make_leaf(struct layout_cell *lc, struct window_pane *wp)
|
|
{
|
|
lc->type = LAYOUT_WINDOWPANE;
|
|
|
|
TAILQ_INIT(&lc->cells);
|
|
|
|
wp->layout_cell = lc;
|
|
lc->wp = wp;
|
|
}
|
|
|
|
void
|
|
layout_make_node(struct layout_cell *lc, enum layout_type type)
|
|
{
|
|
if (type == LAYOUT_WINDOWPANE)
|
|
fatalx("bad layout type");
|
|
lc->type = type;
|
|
|
|
TAILQ_INIT(&lc->cells);
|
|
|
|
if (lc->wp != NULL)
|
|
lc->wp->layout_cell = NULL;
|
|
lc->wp = NULL;
|
|
}
|
|
|
|
/* Fix cell offsets based on their sizes. */
|
|
void
|
|
layout_fix_offsets(struct layout_cell *lc)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
u_int xoff, yoff;
|
|
|
|
if (lc->type == LAYOUT_LEFTRIGHT) {
|
|
xoff = lc->xoff;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
lcchild->xoff = xoff;
|
|
lcchild->yoff = lc->yoff;
|
|
if (lcchild->type != LAYOUT_WINDOWPANE)
|
|
layout_fix_offsets(lcchild);
|
|
xoff += lcchild->sx + 1;
|
|
}
|
|
} else {
|
|
yoff = lc->yoff;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
lcchild->xoff = lc->xoff;
|
|
lcchild->yoff = yoff;
|
|
if (lcchild->type != LAYOUT_WINDOWPANE)
|
|
layout_fix_offsets(lcchild);
|
|
yoff += lcchild->sy + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Returns 1 if we need to reserve space for the pane status line. This is the
|
|
* case for the most upper panes only.
|
|
*/
|
|
static int
|
|
layout_need_status(struct layout_cell *lc, int at_top)
|
|
{
|
|
struct layout_cell *first_lc;
|
|
|
|
if (lc->parent != NULL) {
|
|
if (lc->parent->type == LAYOUT_LEFTRIGHT)
|
|
return (layout_need_status(lc->parent, at_top));
|
|
|
|
if (at_top)
|
|
first_lc = TAILQ_FIRST(&lc->parent->cells);
|
|
else
|
|
first_lc = TAILQ_LAST(&lc->parent->cells,layout_cells);
|
|
if (lc == first_lc)
|
|
return (layout_need_status(lc->parent, at_top));
|
|
return (0);
|
|
}
|
|
return (1);
|
|
}
|
|
|
|
/* Update pane offsets and sizes based on their cells. */
|
|
void
|
|
layout_fix_panes(struct window *w)
|
|
{
|
|
struct window_pane *wp;
|
|
struct layout_cell *lc;
|
|
int shift, status;
|
|
|
|
status = options_get_number(w->options, "pane-border-status");
|
|
TAILQ_FOREACH(wp, &w->panes, entry) {
|
|
if ((lc = wp->layout_cell) == NULL)
|
|
continue;
|
|
|
|
if (status != 0)
|
|
shift = layout_need_status(lc, status == 1);
|
|
else
|
|
shift = 0;
|
|
|
|
wp->xoff = lc->xoff;
|
|
wp->yoff = lc->yoff;
|
|
|
|
if (shift && status == 1)
|
|
wp->yoff += 1;
|
|
|
|
window_pane_resize(wp, lc->sx, lc->sy - shift);
|
|
}
|
|
}
|
|
|
|
/* Count the number of available cells in a layout. */
|
|
u_int
|
|
layout_count_cells(struct layout_cell *lc)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
u_int count;
|
|
|
|
switch (lc->type) {
|
|
case LAYOUT_WINDOWPANE:
|
|
return (1);
|
|
case LAYOUT_LEFTRIGHT:
|
|
case LAYOUT_TOPBOTTOM:
|
|
count = 0;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry)
|
|
count += layout_count_cells(lcchild);
|
|
return (count);
|
|
default:
|
|
fatalx("bad layout type");
|
|
}
|
|
}
|
|
|
|
/* Calculate how much size is available to be removed from a cell. */
|
|
static u_int
|
|
layout_resize_check(struct window *w, struct layout_cell *lc,
|
|
enum layout_type type)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
u_int available, minimum;
|
|
int status;
|
|
|
|
status = options_get_number(w->options, "pane-border-status");
|
|
if (lc->type == LAYOUT_WINDOWPANE) {
|
|
/* Space available in this cell only. */
|
|
minimum = PANE_MINIMUM;
|
|
if (type == LAYOUT_LEFTRIGHT)
|
|
available = lc->sx;
|
|
else {
|
|
available = lc->sy;
|
|
if (status != 0)
|
|
minimum += layout_need_status(lc, status == 1);
|
|
}
|
|
if (available > minimum)
|
|
available -= minimum;
|
|
else
|
|
available = 0;
|
|
} else if (lc->type == type) {
|
|
/* Same type: total of available space in all child cells. */
|
|
available = 0;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry)
|
|
available += layout_resize_check(w, lcchild, type);
|
|
} else {
|
|
/* Different type: minimum of available space in child cells. */
|
|
minimum = UINT_MAX;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
available = layout_resize_check(w, lcchild, type);
|
|
if (available < minimum)
|
|
minimum = available;
|
|
}
|
|
available = minimum;
|
|
}
|
|
|
|
return (available);
|
|
}
|
|
|
|
/*
|
|
* Adjust cell size evenly, including altering its children. This function
|
|
* expects the change to have already been bounded to the space available.
|
|
*/
|
|
void
|
|
layout_resize_adjust(struct window *w, struct layout_cell *lc,
|
|
enum layout_type type, int change)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
|
|
/* Adjust the cell size. */
|
|
if (type == LAYOUT_LEFTRIGHT)
|
|
lc->sx += change;
|
|
else
|
|
lc->sy += change;
|
|
|
|
/* If this is a leaf cell, that is all that is necessary. */
|
|
if (type == LAYOUT_WINDOWPANE)
|
|
return;
|
|
|
|
/* Child cell runs in a different direction. */
|
|
if (lc->type != type) {
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry)
|
|
layout_resize_adjust(w, lcchild, type, change);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Child cell runs in the same direction. Adjust each child equally
|
|
* until no further change is possible.
|
|
*/
|
|
while (change != 0) {
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
if (change == 0)
|
|
break;
|
|
if (change > 0) {
|
|
layout_resize_adjust(w, lcchild, type, 1);
|
|
change--;
|
|
continue;
|
|
}
|
|
if (layout_resize_check(w, lcchild, type) > 0) {
|
|
layout_resize_adjust(w, lcchild, type, -1);
|
|
change++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Destroy a cell and redistribute the space. */
|
|
void
|
|
layout_destroy_cell(struct window *w, struct layout_cell *lc,
|
|
struct layout_cell **lcroot)
|
|
{
|
|
struct layout_cell *lcother, *lcparent;
|
|
|
|
/*
|
|
* If no parent, this is the last pane so window close is imminent and
|
|
* there is no need to resize anything.
|
|
*/
|
|
lcparent = lc->parent;
|
|
if (lcparent == NULL) {
|
|
layout_free_cell(lc);
|
|
*lcroot = NULL;
|
|
return;
|
|
}
|
|
|
|
/* Merge the space into the previous or next cell. */
|
|
if (lc == TAILQ_FIRST(&lcparent->cells))
|
|
lcother = TAILQ_NEXT(lc, entry);
|
|
else
|
|
lcother = TAILQ_PREV(lc, layout_cells, entry);
|
|
if (lcparent->type == LAYOUT_LEFTRIGHT)
|
|
layout_resize_adjust(w, lcother, lcparent->type, lc->sx + 1);
|
|
else
|
|
layout_resize_adjust(w, lcother, lcparent->type, lc->sy + 1);
|
|
|
|
/* Remove this from the parent's list. */
|
|
TAILQ_REMOVE(&lcparent->cells, lc, entry);
|
|
layout_free_cell(lc);
|
|
|
|
/*
|
|
* If the parent now has one cell, remove the parent from the tree and
|
|
* replace it by that cell.
|
|
*/
|
|
lc = TAILQ_FIRST(&lcparent->cells);
|
|
if (TAILQ_NEXT(lc, entry) == NULL) {
|
|
TAILQ_REMOVE(&lcparent->cells, lc, entry);
|
|
|
|
lc->parent = lcparent->parent;
|
|
if (lc->parent == NULL) {
|
|
lc->xoff = 0; lc->yoff = 0;
|
|
*lcroot = lc;
|
|
} else
|
|
TAILQ_REPLACE(&lc->parent->cells, lcparent, lc, entry);
|
|
|
|
layout_free_cell(lcparent);
|
|
}
|
|
}
|
|
|
|
void
|
|
layout_init(struct window *w, struct window_pane *wp)
|
|
{
|
|
struct layout_cell *lc;
|
|
|
|
lc = w->layout_root = layout_create_cell(NULL);
|
|
layout_set_size(lc, w->sx, w->sy, 0, 0);
|
|
layout_make_leaf(lc, wp);
|
|
layout_fix_panes(w);
|
|
}
|
|
|
|
void
|
|
layout_free(struct window *w)
|
|
{
|
|
layout_free_cell(w->layout_root);
|
|
}
|
|
|
|
/* Resize the entire layout after window resize. */
|
|
void
|
|
layout_resize(struct window *w, u_int sx, u_int sy)
|
|
{
|
|
struct layout_cell *lc = w->layout_root;
|
|
int xlimit, ylimit, xchange, ychange;
|
|
|
|
/*
|
|
* Adjust horizontally. Do not attempt to reduce the layout lower than
|
|
* the minimum (more than the amount returned by layout_resize_check).
|
|
*
|
|
* This can mean that the window size is smaller than the total layout
|
|
* size: redrawing this is handled at a higher level, but it does leave
|
|
* a problem with growing the window size here: if the current size is
|
|
* < the minimum, growing proportionately by adding to each pane is
|
|
* wrong as it would keep the layout size larger than the window size.
|
|
* Instead, spread the difference between the minimum and the new size
|
|
* out proportionately - this should leave the layout fitting the new
|
|
* window size.
|
|
*/
|
|
xchange = sx - w->sx;
|
|
xlimit = layout_resize_check(w, lc, LAYOUT_LEFTRIGHT);
|
|
if (xchange < 0 && xchange < -xlimit)
|
|
xchange = -xlimit;
|
|
if (xlimit == 0) {
|
|
if (sx <= lc->sx) /* lc->sx is minimum possible */
|
|
xchange = 0;
|
|
else
|
|
xchange = sx - lc->sx;
|
|
}
|
|
if (xchange != 0)
|
|
layout_resize_adjust(w, lc, LAYOUT_LEFTRIGHT, xchange);
|
|
|
|
/* Adjust vertically in a similar fashion. */
|
|
ychange = sy - w->sy;
|
|
ylimit = layout_resize_check(w, lc, LAYOUT_TOPBOTTOM);
|
|
if (ychange < 0 && ychange < -ylimit)
|
|
ychange = -ylimit;
|
|
if (ylimit == 0) {
|
|
if (sy <= lc->sy) /* lc->sy is minimum possible */
|
|
ychange = 0;
|
|
else
|
|
ychange = sy - lc->sy;
|
|
}
|
|
if (ychange != 0)
|
|
layout_resize_adjust(w, lc, LAYOUT_TOPBOTTOM, ychange);
|
|
|
|
/* Fix cell offsets. */
|
|
layout_fix_offsets(lc);
|
|
layout_fix_panes(w);
|
|
}
|
|
|
|
/* Resize a pane to an absolute size. */
|
|
void
|
|
layout_resize_pane_to(struct window_pane *wp, enum layout_type type,
|
|
u_int new_size)
|
|
{
|
|
struct layout_cell *lc, *lcparent;
|
|
int change, size;
|
|
|
|
lc = wp->layout_cell;
|
|
|
|
/* Find next parent of the same type. */
|
|
lcparent = lc->parent;
|
|
while (lcparent != NULL && lcparent->type != type) {
|
|
lc = lcparent;
|
|
lcparent = lc->parent;
|
|
}
|
|
if (lcparent == NULL)
|
|
return;
|
|
|
|
/* Work out the size adjustment. */
|
|
if (type == LAYOUT_LEFTRIGHT)
|
|
size = lc->sx;
|
|
else
|
|
size = lc->sy;
|
|
if (lc == TAILQ_LAST(&lcparent->cells, layout_cells))
|
|
change = size - new_size;
|
|
else
|
|
change = new_size - size;
|
|
|
|
/* Resize the pane. */
|
|
layout_resize_pane(wp, type, change, 1);
|
|
}
|
|
|
|
void
|
|
layout_resize_layout(struct window *w, struct layout_cell *lc,
|
|
enum layout_type type, int change, int opposite)
|
|
{
|
|
int needed, size;
|
|
|
|
/* Grow or shrink the cell. */
|
|
needed = change;
|
|
while (needed != 0) {
|
|
if (change > 0) {
|
|
size = layout_resize_pane_grow(w, lc, type, needed,
|
|
opposite);
|
|
needed -= size;
|
|
} else {
|
|
size = layout_resize_pane_shrink(w, lc, type, needed);
|
|
needed += size;
|
|
}
|
|
|
|
if (size == 0) /* no more change possible */
|
|
break;
|
|
}
|
|
|
|
/* Fix cell offsets. */
|
|
layout_fix_offsets(w->layout_root);
|
|
layout_fix_panes(w);
|
|
notify_window("window-layout-changed", w);
|
|
}
|
|
|
|
/* Resize a single pane within the layout. */
|
|
void
|
|
layout_resize_pane(struct window_pane *wp, enum layout_type type, int change,
|
|
int opposite)
|
|
{
|
|
struct layout_cell *lc, *lcparent;
|
|
|
|
lc = wp->layout_cell;
|
|
|
|
/* Find next parent of the same type. */
|
|
lcparent = lc->parent;
|
|
while (lcparent != NULL && lcparent->type != type) {
|
|
lc = lcparent;
|
|
lcparent = lc->parent;
|
|
}
|
|
if (lcparent == NULL)
|
|
return;
|
|
|
|
/* If this is the last cell, move back one. */
|
|
if (lc == TAILQ_LAST(&lcparent->cells, layout_cells))
|
|
lc = TAILQ_PREV(lc, layout_cells, entry);
|
|
|
|
layout_resize_layout(wp->window, lc, type, change, opposite);
|
|
}
|
|
|
|
/* Helper function to grow pane. */
|
|
static int
|
|
layout_resize_pane_grow(struct window *w, struct layout_cell *lc,
|
|
enum layout_type type, int needed, int opposite)
|
|
{
|
|
struct layout_cell *lcadd, *lcremove;
|
|
u_int size = 0;
|
|
|
|
/* Growing. Always add to the current cell. */
|
|
lcadd = lc;
|
|
|
|
/* Look towards the tail for a suitable cell for reduction. */
|
|
lcremove = TAILQ_NEXT(lc, entry);
|
|
while (lcremove != NULL) {
|
|
size = layout_resize_check(w, lcremove, type);
|
|
if (size > 0)
|
|
break;
|
|
lcremove = TAILQ_NEXT(lcremove, entry);
|
|
}
|
|
|
|
/* If none found, look towards the head. */
|
|
if (opposite && lcremove == NULL) {
|
|
lcremove = TAILQ_PREV(lc, layout_cells, entry);
|
|
while (lcremove != NULL) {
|
|
size = layout_resize_check(w, lcremove, type);
|
|
if (size > 0)
|
|
break;
|
|
lcremove = TAILQ_PREV(lcremove, layout_cells, entry);
|
|
}
|
|
}
|
|
if (lcremove == NULL)
|
|
return (0);
|
|
|
|
/* Change the cells. */
|
|
if (size > (u_int) needed)
|
|
size = needed;
|
|
layout_resize_adjust(w, lcadd, type, size);
|
|
layout_resize_adjust(w, lcremove, type, -size);
|
|
return (size);
|
|
}
|
|
|
|
/* Helper function to shrink pane. */
|
|
static int
|
|
layout_resize_pane_shrink(struct window *w, struct layout_cell *lc,
|
|
enum layout_type type, int needed)
|
|
{
|
|
struct layout_cell *lcadd, *lcremove;
|
|
u_int size;
|
|
|
|
/* Shrinking. Find cell to remove from by walking towards head. */
|
|
lcremove = lc;
|
|
do {
|
|
size = layout_resize_check(w, lcremove, type);
|
|
if (size != 0)
|
|
break;
|
|
lcremove = TAILQ_PREV(lcremove, layout_cells, entry);
|
|
} while (lcremove != NULL);
|
|
if (lcremove == NULL)
|
|
return (0);
|
|
|
|
/* And add onto the next cell (from the original cell). */
|
|
lcadd = TAILQ_NEXT(lc, entry);
|
|
if (lcadd == NULL)
|
|
return (0);
|
|
|
|
/* Change the cells. */
|
|
if (size > (u_int) -needed)
|
|
size = -needed;
|
|
layout_resize_adjust(w, lcadd, type, size);
|
|
layout_resize_adjust(w, lcremove, type, -size);
|
|
return (size);
|
|
}
|
|
|
|
/* Assign window pane to newly split cell. */
|
|
void
|
|
layout_assign_pane(struct layout_cell *lc, struct window_pane *wp)
|
|
{
|
|
layout_make_leaf(lc, wp);
|
|
layout_fix_panes(wp->window);
|
|
}
|
|
|
|
/* Calculate the new pane size for resized parent. */
|
|
static u_int
|
|
layout_new_pane_size(struct window *w, u_int previous, struct layout_cell *lc,
|
|
enum layout_type type, u_int size, u_int count_left, u_int size_left)
|
|
{
|
|
u_int new_size, min, max, available;
|
|
|
|
/* If this is the last cell, it can take all of the remaining size. */
|
|
if (count_left == 1)
|
|
return (size_left);
|
|
|
|
/* How much is available in this parent? */
|
|
available = layout_resize_check(w, lc, type);
|
|
|
|
/*
|
|
* Work out the minimum size of this cell and the new size
|
|
* proportionate to the previous size.
|
|
*/
|
|
min = (PANE_MINIMUM + 1) * (count_left - 1);
|
|
if (type == LAYOUT_LEFTRIGHT) {
|
|
if (lc->sx - available > min)
|
|
min = lc->sx - available;
|
|
new_size = (lc->sx * size) / previous;
|
|
} else {
|
|
if (lc->sy - available > min)
|
|
min = lc->sy - available;
|
|
new_size = (lc->sy * size) / previous;
|
|
}
|
|
|
|
/* Check against the maximum and minimum size. */
|
|
max = size_left - min;
|
|
if (new_size > max)
|
|
new_size = max;
|
|
if (new_size < PANE_MINIMUM)
|
|
new_size = PANE_MINIMUM;
|
|
return (new_size);
|
|
}
|
|
|
|
/* Check if the cell and all its children can be resized to a specific size. */
|
|
static int
|
|
layout_set_size_check(struct window *w, struct layout_cell *lc,
|
|
enum layout_type type, int size)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
u_int new_size, available, previous, count, idx;
|
|
|
|
/* Cells with no children must just be bigger than minimum. */
|
|
if (lc->type == LAYOUT_WINDOWPANE)
|
|
return (size >= PANE_MINIMUM);
|
|
available = size;
|
|
|
|
/* Count number of children. */
|
|
count = 0;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry)
|
|
count++;
|
|
|
|
/* Check new size will work for each child. */
|
|
if (lc->type == type) {
|
|
if (type == LAYOUT_LEFTRIGHT)
|
|
previous = lc->sx;
|
|
else
|
|
previous = lc->sy;
|
|
|
|
idx = 0;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
new_size = layout_new_pane_size(w, previous, lcchild,
|
|
type, size, count - idx, available);
|
|
if (new_size > available)
|
|
return (0);
|
|
|
|
available -= (new_size + 1);
|
|
if (!layout_set_size_check(w, lcchild, type, new_size))
|
|
return (0);
|
|
|
|
idx++;
|
|
}
|
|
} else {
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
if (lcchild->type == LAYOUT_WINDOWPANE)
|
|
continue;
|
|
if (!layout_set_size_check(w, lcchild, type, size))
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
return (1);
|
|
}
|
|
|
|
/* Resize all child cells to fit within the current cell. */
|
|
static void
|
|
layout_resize_child_cells(struct window *w, struct layout_cell *lc)
|
|
{
|
|
struct layout_cell *lcchild;
|
|
u_int previous, available, count, idx;
|
|
|
|
if (lc->type == LAYOUT_WINDOWPANE)
|
|
return;
|
|
|
|
/* What is the current size used? */
|
|
count = 0;
|
|
previous = 0;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
count++;
|
|
if (lc->type == LAYOUT_LEFTRIGHT)
|
|
previous += lcchild->sx;
|
|
else if (lc->type == LAYOUT_TOPBOTTOM)
|
|
previous += lcchild->sy;
|
|
}
|
|
previous += (count - 1);
|
|
|
|
/* And how much is available? */
|
|
available = 0;
|
|
if (lc->type == LAYOUT_LEFTRIGHT)
|
|
available = lc->sx;
|
|
else if (lc->type == LAYOUT_TOPBOTTOM)
|
|
available = lc->sy;
|
|
|
|
/* Resize children into the new size. */
|
|
idx = 0;
|
|
TAILQ_FOREACH(lcchild, &lc->cells, entry) {
|
|
if (lc->type == LAYOUT_TOPBOTTOM) {
|
|
lcchild->sx = lc->sx;
|
|
lcchild->xoff = lc->xoff;
|
|
} else {
|
|
lcchild->sx = layout_new_pane_size(w, previous, lcchild,
|
|
lc->type, lc->sx, count - idx, available);
|
|
available -= (lcchild->sx + 1);
|
|
}
|
|
if (lc->type == LAYOUT_LEFTRIGHT)
|
|
lcchild->sy = lc->sy;
|
|
else {
|
|
lcchild->sy = layout_new_pane_size(w, previous, lcchild,
|
|
lc->type, lc->sy, count - idx, available);
|
|
available -= (lcchild->sy + 1);
|
|
}
|
|
layout_resize_child_cells(w, lcchild);
|
|
idx++;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Split a pane into two. size is a hint, or -1 for default half/half
|
|
* split. This must be followed by layout_assign_pane before much else happens!
|
|
*/
|
|
struct layout_cell *
|
|
layout_split_pane(struct window_pane *wp, enum layout_type type, int size,
|
|
int insert_before, int full_size)
|
|
{
|
|
struct layout_cell *lc, *lcparent, *lcnew, *lc1, *lc2;
|
|
u_int sx, sy, xoff, yoff, size1, size2;
|
|
u_int new_size, saved_size, resize_first = 0;
|
|
|
|
/*
|
|
* If full_size is specified, add a new cell at the top of the window
|
|
* layout. Otherwise, split the cell for the current pane.
|
|
*/
|
|
if (full_size)
|
|
lc = wp->window->layout_root;
|
|
else
|
|
lc = wp->layout_cell;
|
|
|
|
/* Copy the old cell size. */
|
|
sx = lc->sx;
|
|
sy = lc->sy;
|
|
xoff = lc->xoff;
|
|
yoff = lc->yoff;
|
|
|
|
/* Check there is enough space for the two new panes. */
|
|
switch (type) {
|
|
case LAYOUT_LEFTRIGHT:
|
|
if (sx < PANE_MINIMUM * 2 + 1)
|
|
return (NULL);
|
|
break;
|
|
case LAYOUT_TOPBOTTOM:
|
|
if (sy < PANE_MINIMUM * 2 + 1)
|
|
return (NULL);
|
|
break;
|
|
default:
|
|
fatalx("bad layout type");
|
|
}
|
|
|
|
/*
|
|
* Calculate new cell sizes. size is the target size or -1 for middle
|
|
* split, size1 is the size of the top/left and size2 the bottom/right.
|
|
*/
|
|
if (type == LAYOUT_LEFTRIGHT)
|
|
saved_size = sx;
|
|
else
|
|
saved_size = sy;
|
|
if (size < 0)
|
|
size2 = ((saved_size + 1) / 2) - 1;
|
|
else if (insert_before)
|
|
size2 = saved_size - size - 1;
|
|
else
|
|
size2 = size;
|
|
if (size2 < PANE_MINIMUM)
|
|
size2 = PANE_MINIMUM;
|
|
else if (size2 > saved_size - 2)
|
|
size2 = saved_size - 2;
|
|
size1 = saved_size - 1 - size2;
|
|
|
|
/* Which size are we using? */
|
|
if (insert_before)
|
|
new_size = size2;
|
|
else
|
|
new_size = size1;
|
|
|
|
/* Confirm there is enough space for full size pane. */
|
|
if (full_size && !layout_set_size_check(wp->window, lc, type, new_size))
|
|
return (NULL);
|
|
|
|
if (lc->parent != NULL && lc->parent->type == type) {
|
|
/*
|
|
* If the parent exists and is of the same type as the split,
|
|
* create a new cell and insert it after this one.
|
|
*/
|
|
lcparent = lc->parent;
|
|
lcnew = layout_create_cell(lcparent);
|
|
if (insert_before)
|
|
TAILQ_INSERT_BEFORE(lc, lcnew, entry);
|
|
else
|
|
TAILQ_INSERT_AFTER(&lcparent->cells, lc, lcnew, entry);
|
|
} else if (full_size && lc->parent == NULL && lc->type == type) {
|
|
/*
|
|
* If the new full size pane is the same type as the root
|
|
* split, insert the new pane under the existing root cell
|
|
* instead of creating a new root cell. The existing layout
|
|
* must be resized before inserting the new cell.
|
|
*/
|
|
if (lc->type == LAYOUT_LEFTRIGHT) {
|
|
lc->sx = new_size;
|
|
layout_resize_child_cells(wp->window, lc);
|
|
lc->sx = saved_size;
|
|
} else if (lc->type == LAYOUT_TOPBOTTOM) {
|
|
lc->sy = new_size;
|
|
layout_resize_child_cells(wp->window, lc);
|
|
lc->sy = saved_size;
|
|
}
|
|
resize_first = 1;
|
|
|
|
/* Create the new cell. */
|
|
lcnew = layout_create_cell(lc);
|
|
size = saved_size - 1 - new_size;
|
|
if (lc->type == LAYOUT_LEFTRIGHT)
|
|
layout_set_size(lcnew, size, sy, 0, 0);
|
|
else if (lc->type == LAYOUT_TOPBOTTOM)
|
|
layout_set_size(lcnew, sx, size, 0, 0);
|
|
if (insert_before)
|
|
TAILQ_INSERT_HEAD(&lc->cells, lcnew, entry);
|
|
else
|
|
TAILQ_INSERT_TAIL(&lc->cells, lcnew, entry);
|
|
} else {
|
|
/*
|
|
* Otherwise create a new parent and insert it.
|
|
*/
|
|
|
|
/* Create and insert the replacement parent. */
|
|
lcparent = layout_create_cell(lc->parent);
|
|
layout_make_node(lcparent, type);
|
|
layout_set_size(lcparent, sx, sy, xoff, yoff);
|
|
if (lc->parent == NULL)
|
|
wp->window->layout_root = lcparent;
|
|
else
|
|
TAILQ_REPLACE(&lc->parent->cells, lc, lcparent, entry);
|
|
|
|
/* Insert the old cell. */
|
|
lc->parent = lcparent;
|
|
TAILQ_INSERT_HEAD(&lcparent->cells, lc, entry);
|
|
|
|
/* Create the new child cell. */
|
|
lcnew = layout_create_cell(lcparent);
|
|
if (insert_before)
|
|
TAILQ_INSERT_HEAD(&lcparent->cells, lcnew, entry);
|
|
else
|
|
TAILQ_INSERT_TAIL(&lcparent->cells, lcnew, entry);
|
|
}
|
|
if (insert_before) {
|
|
lc1 = lcnew;
|
|
lc2 = lc;
|
|
} else {
|
|
lc1 = lc;
|
|
lc2 = lcnew;
|
|
}
|
|
|
|
/*
|
|
* Set new cell sizes. size1 is the size of the top/left and size2 the
|
|
* bottom/right.
|
|
*/
|
|
if (!resize_first && type == LAYOUT_LEFTRIGHT) {
|
|
layout_set_size(lc1, size1, sy, xoff, yoff);
|
|
layout_set_size(lc2, size2, sy, xoff + lc1->sx + 1, yoff);
|
|
} else if (!resize_first && type == LAYOUT_TOPBOTTOM) {
|
|
layout_set_size(lc1, sx, size1, xoff, yoff);
|
|
layout_set_size(lc2, sx, size2, xoff, yoff + lc1->sy + 1);
|
|
}
|
|
if (full_size) {
|
|
if (!resize_first)
|
|
layout_resize_child_cells(wp->window, lc);
|
|
layout_fix_offsets(wp->window->layout_root);
|
|
} else
|
|
layout_make_leaf(lc, wp);
|
|
|
|
return (lcnew);
|
|
}
|
|
|
|
/* Destroy the cell associated with a pane. */
|
|
void
|
|
layout_close_pane(struct window_pane *wp)
|
|
{
|
|
struct window *w = wp->window;
|
|
|
|
/* Remove the cell. */
|
|
layout_destroy_cell(w, wp->layout_cell, &w->layout_root);
|
|
|
|
/* Fix pane offsets and sizes. */
|
|
if (w->layout_root != NULL) {
|
|
layout_fix_offsets(w->layout_root);
|
|
layout_fix_panes(w);
|
|
}
|
|
notify_window("window-layout-changed", w);
|
|
}
|
|
|
|
int
|
|
layout_spread_cell(struct window *w, struct layout_cell *parent)
|
|
{
|
|
struct layout_cell *lc;
|
|
u_int number, each, size;
|
|
int change, changed;
|
|
|
|
number = 0;
|
|
TAILQ_FOREACH (lc, &parent->cells, entry)
|
|
number++;
|
|
if (number <= 1)
|
|
return (0);
|
|
|
|
if (parent->type == LAYOUT_LEFTRIGHT)
|
|
size = parent->sx;
|
|
else if (parent->type == LAYOUT_TOPBOTTOM)
|
|
size = parent->sy;
|
|
else
|
|
return (0);
|
|
each = (size - (number - 1)) / number;
|
|
|
|
changed = 0;
|
|
TAILQ_FOREACH (lc, &parent->cells, entry) {
|
|
if (TAILQ_NEXT(lc, entry) == NULL)
|
|
each = size - ((each + 1) * (number - 1));
|
|
change = 0;
|
|
if (parent->type == LAYOUT_LEFTRIGHT) {
|
|
change = each - (int)lc->sx;
|
|
layout_resize_adjust(w, lc, LAYOUT_LEFTRIGHT, change);
|
|
} else if (parent->type == LAYOUT_TOPBOTTOM) {
|
|
change = each - (int)lc->sy;
|
|
layout_resize_adjust(w, lc, LAYOUT_TOPBOTTOM, change);
|
|
}
|
|
if (change != 0)
|
|
changed = 1;
|
|
}
|
|
return (changed);
|
|
}
|
|
|
|
void
|
|
layout_spread_out(struct window_pane *wp)
|
|
{
|
|
struct layout_cell *parent;
|
|
struct window *w = wp->window;
|
|
|
|
parent = wp->layout_cell->parent;
|
|
if (parent == NULL)
|
|
return;
|
|
|
|
do {
|
|
if (layout_spread_cell(w, parent)) {
|
|
layout_fix_offsets(parent);
|
|
layout_fix_panes(w);
|
|
break;
|
|
}
|
|
} while ((parent = parent->parent) != NULL);
|
|
}
|