summaryrefslogtreecommitdiff
path: root/usr.bin
diff options
context:
space:
mode:
authorNicholas Marriott <nicm@cvs.openbsd.org>2009-07-13 10:43:53 +0000
committerNicholas Marriott <nicm@cvs.openbsd.org>2009-07-13 10:43:53 +0000
commit3b86c193d95e633ff0e65671513fc1a762e8d569 (patch)
treea06d7b73ad58380a65dd71108691bc68d02c7776 /usr.bin
parent2c70086ac6c34217b330a1710ab3417cee0adb7e (diff)
Support "alternate screen" mode (terminfo smcup/rmcup) typically used by full
screen interactive programs to preserve the screen contents. When activated, it saves a copy of the visible grid and disables scrolling into and resizing out of the history; when deactivated the visible data is restored and the history reenabled.
Diffstat (limited to 'usr.bin')
-rw-r--r--usr.bin/tmux/grid-view.c4
-rw-r--r--usr.bin/tmux/grid.c48
-rw-r--r--usr.bin/tmux/input.c76
-rw-r--r--usr.bin/tmux/screen.c28
-rw-r--r--usr.bin/tmux/tmux.h12
-rw-r--r--usr.bin/tmux/window.c6
6 files changed, 159 insertions, 15 deletions
diff --git a/usr.bin/tmux/grid-view.c b/usr.bin/tmux/grid-view.c
index 76017828c50..70aee107187 100644
--- a/usr.bin/tmux/grid-view.c
+++ b/usr.bin/tmux/grid-view.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: grid-view.c,v 1.5 2009/07/09 07:58:14 nicm Exp $ */
+/* $OpenBSD: grid-view.c,v 1.6 2009/07/13 10:43:52 nicm Exp $ */
/*
* Copyright (c) 2008 Nicholas Marriott <nicm@users.sourceforge.net>
@@ -92,7 +92,7 @@ grid_view_scroll_region_up(struct grid *gd, u_int rupper, u_int rlower)
{
GRID_DEBUG(gd, "rupper=%u, rlower=%u", rupper, rlower);
- if (rupper == 0 && rlower == gd->sy - 1) {
+ if (gd->flags & GRID_HISTORY && rupper == 0 && rlower == gd->sy - 1) {
grid_scroll_line(gd);
return;
}
diff --git a/usr.bin/tmux/grid.c b/usr.bin/tmux/grid.c
index 50f085256dd..0e14c780829 100644
--- a/usr.bin/tmux/grid.c
+++ b/usr.bin/tmux/grid.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: grid.c,v 1.6 2009/07/11 20:11:18 nicm Exp $ */
+/* $OpenBSD: grid.c,v 1.7 2009/07/13 10:43:52 nicm Exp $ */
/*
* Copyright (c) 2008 Nicholas Marriott <nicm@users.sourceforge.net>
@@ -95,6 +95,8 @@ grid_create(u_int sx, u_int sy, u_int hlimit)
gd->sx = sx;
gd->sy = sy;
+ gd->flags = GRID_HISTORY;
+
gd->hsize = 0;
gd->hlimit = hlimit;
@@ -517,3 +519,47 @@ grid_string_cells(struct grid *gd, u_int px, u_int py, u_int nx)
buf[off] = '\0';
return (buf);
}
+
+/*
+ * Duplicate a set of lines between two grids. If there aren't enough lines in
+ * either source or destination, the number of lines is limited to the number
+ * available.
+ */
+void
+grid_duplicate_lines(
+ struct grid *dst, u_int dy, struct grid *src, u_int sy, u_int ny)
+{
+ u_int yy;
+
+ GRID_DEBUG(src, "dy=%u, sy=%u, ny=%u", dy, sy, ny);
+
+ if (dy + ny > dst->hsize + dst->sy)
+ ny = dst->hsize + dst->sy - dy;
+ if (sy + ny > src->hsize + src->sy)
+ ny = src->hsize + src->sy - sy;
+ grid_clear_lines(dst, dy, ny);
+
+ for (yy = 0; yy < ny; yy++) {
+ dst->size[dy] = src->size[sy];
+ if (src->size[sy] == 0)
+ dst->data[dy] = NULL;
+ else {
+ dst->data[dy] = xcalloc(
+ src->size[sy], sizeof **dst->data);
+ memcpy(dst->data[dy], src->data[sy],
+ src->size[sy] * (sizeof **dst->data));
+ }
+
+ dst->usize[dy] = src->usize[sy];
+ if (src->usize[sy] == 0)
+ dst->udata[dy] = NULL;
+ else {
+ dst->udata[sy] = xcalloc(
+ src->usize[sy], sizeof **dst->udata);
+ memcpy(dst->udata[dy], src->udata[sy],
+ src->usize[sy] * (sizeof **dst->udata));
+ }
+
+ sy++; dy++;
+ }
+}
diff --git a/usr.bin/tmux/input.c b/usr.bin/tmux/input.c
index cef36d76bca..71f9721fc57 100644
--- a/usr.bin/tmux/input.c
+++ b/usr.bin/tmux/input.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: input.c,v 1.8 2009/06/04 21:02:21 nicm Exp $ */
+/* $OpenBSD: input.c,v 1.9 2009/07/13 10:43:52 nicm Exp $ */
/*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@@ -1151,7 +1151,10 @@ input_handle_sequence_el(struct input_ctx *ictx)
void
input_handle_sequence_sm(struct input_ctx *ictx)
{
- uint16_t n;
+ struct window_pane *wp = ictx->wp;
+ struct screen *s = &wp->base;
+ u_int sx, sy;
+ uint16_t n;
if (ARRAY_LENGTH(&ictx->args) > 1)
return;
@@ -1172,6 +1175,29 @@ input_handle_sequence_sm(struct input_ctx *ictx)
screen_write_mousemode(&ictx->ctx, 1);
log_debug("mouse on");
break;
+ case 1049:
+ if (wp->saved_grid != NULL)
+ break;
+ sx = screen_size_x(s);
+ sy = screen_size_y(s);
+
+ /*
+ * Enter alternative screen mode. A copy of the visible
+ * screen is saved and the history is not updated
+ */
+
+ wp->saved_grid = grid_create(sx, sy, 0);
+ grid_duplicate_lines(
+ wp->saved_grid, 0, s->grid, screen_hsize(s), sy);
+ wp->saved_cx = s->cx;
+ wp->saved_cy = s->cy;
+
+ grid_view_clear(s->grid, 0, 0, sx, sy);
+
+ wp->base.grid->flags &= ~GRID_HISTORY;
+
+ wp->flags |= PANE_REDRAW;
+ break;
default:
log_debug("unknown SM [%hhu]: %u", ictx->private, n);
break;
@@ -1195,7 +1221,10 @@ input_handle_sequence_sm(struct input_ctx *ictx)
void
input_handle_sequence_rm(struct input_ctx *ictx)
{
- uint16_t n;
+ struct window_pane *wp = ictx->wp;
+ struct screen *s = &wp->base;
+ u_int sx, sy;
+ uint16_t n;
if (ARRAY_LENGTH(&ictx->args) > 1)
return;
@@ -1216,6 +1245,47 @@ input_handle_sequence_rm(struct input_ctx *ictx)
screen_write_mousemode(&ictx->ctx, 0);
log_debug("mouse off");
break;
+ case 1049:
+ if (wp->saved_grid == NULL)
+ break;
+ sx = screen_size_x(s);
+ sy = screen_size_y(s);
+
+ /*
+ * Exit alternative screen mode and restore the copied
+ * grid.
+ */
+
+ /*
+ * If the current size is bigger, temporarily resize
+ * to the old size before copying back.
+ */
+ if (sy > wp->saved_grid->sy)
+ screen_resize(s, sx, wp->saved_grid->sy);
+
+ /* Restore the grid and cursor position. */
+ grid_duplicate_lines(
+ s->grid, screen_hsize(s), wp->saved_grid, 0, sy);
+ s->cx = wp->saved_cx;
+ if (s->cx > screen_size_x(s) - 1)
+ s->cx = screen_size_x(s) - 1;
+ s->cy = wp->saved_cy;
+ if (s->cy > screen_size_y(s) - 1)
+ s->cy = screen_size_y(s) - 1;
+
+ /*
+ * Turn history back on (so resize can use it) and then
+ * resize back to the current size.
+ */
+ wp->base.grid->flags |= GRID_HISTORY;
+ if (sy > wp->saved_grid->sy)
+ screen_resize(s, sx, sy);
+
+ grid_destroy(wp->saved_grid);
+ wp->saved_grid = NULL;
+
+ wp->flags |= PANE_REDRAW;
+ break;
default:
log_debug("unknown RM [%hhu]: %u", ictx->private, n);
break;
diff --git a/usr.bin/tmux/screen.c b/usr.bin/tmux/screen.c
index a3077ae774e..af09daf2b92 100644
--- a/usr.bin/tmux/screen.c
+++ b/usr.bin/tmux/screen.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: screen.c,v 1.7 2009/07/09 22:48:20 nicm Exp $ */
+/* $OpenBSD: screen.c,v 1.8 2009/07/13 10:43:52 nicm Exp $ */
/*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@@ -174,10 +174,20 @@ screen_resize_y(struct screen *s, u_int sy)
needed -= available;
/*
- * Now just increase the history size to take over the lines
- * which are left. XXX Should apply history limit?
+ * Now just increase the history size, if possible, to take
+ * over the lines which are left. If history is off, delete
+ * lines from the top.
+ *
+ * XXX Should apply history limit?
*/
- gd->hsize += needed;
+ available = s->cy;
+ if (gd->flags & GRID_HISTORY)
+ gd->hsize += needed;
+ else if (available > 0) {
+ if (available > needed)
+ available = needed;
+ grid_view_delete_lines(gd, 0, available);
+ }
s->cy -= needed;
}
@@ -191,14 +201,18 @@ screen_resize_y(struct screen *s, u_int sy)
if (sy > oldy) {
needed = sy - oldy;
- /* Try to pull as much as possible out of the history. */
+ /*
+ * Try to pull as much as possible out of the history, if is
+ * is enabled.
+ */
available = gd->hsize;
- if (available > 0) {
+ if (gd->flags & GRID_HISTORY && available > 0) {
if (available > needed)
available = needed;
gd->hsize -= available;
s->cy += available;
- }
+ } else
+ available = 0;
needed -= available;
/* Then fill the rest in with blanks. */
diff --git a/usr.bin/tmux/tmux.h b/usr.bin/tmux/tmux.h
index fea65a71754..6264ace2c87 100644
--- a/usr.bin/tmux/tmux.h
+++ b/usr.bin/tmux/tmux.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: tmux.h,v 1.24 2009/07/12 23:46:49 nicm Exp $ */
+/* $OpenBSD: tmux.h,v 1.25 2009/07/13 10:43:52 nicm Exp $ */
/*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@@ -443,6 +443,9 @@ struct grid_utf8 {
/* Entire grid of cells. */
struct grid {
+ int flags;
+#define GRID_HISTORY 0x1 /* scroll lines into history */
+
u_int sx;
u_int sy;
@@ -614,6 +617,11 @@ struct window_pane {
struct screen *screen;
struct screen base;
+ /* Saved in alternative screen mode. */
+ u_int saved_cx;
+ u_int saved_cy;
+ struct grid *saved_grid;
+
const struct window_mode *mode;
void *modedata;
@@ -1328,6 +1336,8 @@ void grid_clear_lines(struct grid *, u_int, u_int);
void grid_move_lines(struct grid *, u_int, u_int, u_int);
void grid_move_cells(struct grid *, u_int, u_int, u_int, u_int);
char *grid_string_cells(struct grid *, u_int, u_int, u_int);
+void grid_duplicate_lines(
+ struct grid *, u_int, struct grid *, u_int, u_int);
/* grid-view.c */
const struct grid_cell *grid_view_peek_cell(struct grid *, u_int, u_int);
diff --git a/usr.bin/tmux/window.c b/usr.bin/tmux/window.c
index 616e5032f87..0a0325d91b9 100644
--- a/usr.bin/tmux/window.c
+++ b/usr.bin/tmux/window.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: window.c,v 1.8 2009/07/08 05:26:45 nicm Exp $ */
+/* $OpenBSD: window.c,v 1.9 2009/07/13 10:43:52 nicm Exp $ */
/*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@@ -407,6 +407,8 @@ window_pane_create(struct window *w, u_int sx, u_int sy, u_int hlimit)
wp->sx = sx;
wp->sy = sy;
+ wp->saved_grid = NULL;
+
screen_init(&wp->base, sx, sy, hlimit);
wp->screen = &wp->base;
@@ -425,6 +427,8 @@ window_pane_destroy(struct window_pane *wp)
window_pane_reset_mode(wp);
screen_free(&wp->base);
+ if (wp->saved_grid != NULL)
+ grid_destroy(wp->saved_grid);
buffer_destroy(wp->in);
buffer_destroy(wp->out);