summaryrefslogtreecommitdiff
path: root/usr.bin/tmux/input.c
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/tmux/input.c
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/tmux/input.c')
-rw-r--r--usr.bin/tmux/input.c76
1 files changed, 73 insertions, 3 deletions
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;