Mass screen.c rename/tidy. Add a actual size (ysize) as distinct from display size (now dx,dy). Move functions which work on the displayed area into screen-display.c and tidy. Use macros consistently when accessing screen data (may want to move everything about again later!). This the first step on the road to scrollback.

This commit is contained in:
Nicholas Marriott 2007-11-20 21:42:29 +00:00
parent 0be6a3041f
commit ceab127fac
10 changed files with 617 additions and 555 deletions

View File

@ -1,4 +1,4 @@
# $Id: Makefile,v 1.43 2007-11-20 17:01:38 nicm Exp $ # $Id: Makefile,v 1.44 2007-11-20 21:42:28 nicm Exp $
.SUFFIXES: .c .o .y .h .SUFFIXES: .c .o .y .h
.PHONY: clean update-index.html upload-index.html .PHONY: clean update-index.html upload-index.html
@ -16,17 +16,17 @@ DEBUG=
META?= \002 # C-b META?= \002 # C-b
SRCS= tmux.c server.c server-msg.c server-fn.c buffer.c buffer-poll.c status.c \ SRCS= tmux.c server.c server-msg.c server-fn.c buffer.c buffer-poll.c status.c \
xmalloc.c xmalloc-debug.c input.c input-keys.c screen.c window.c \ xmalloc.c xmalloc-debug.c input.c input-keys.c screen.c screen-display.c \
session.c local.c log.c client.c client-msg.c client-fn.c key-string.c \ window.c session.c local.c log.c client.c client-msg.c client-fn.c \
key-bindings.c resize.c cmd.c cmd-new-session.c cmd-detach-client.c \ key-string.c key-bindings.c resize.c cmd.c cmd-new-session.c \
cmd-list-sessions.c cmd-new-window.c cmd-next-window.c cmd-bind-key.c \ cmd-detach-client.c cmd-list-sessions.c cmd-new-window.c cmd-bind-key.c \
cmd-unbind-key.c cmd-previous-window.c cmd-last-window.c cmd-list-keys.c \ cmd-unbind-key.c cmd-previous-window.c cmd-last-window.c cmd-list-keys.c \
cmd-set-option.c cmd-rename-window.c cmd-select-window.c \ cmd-set-option.c cmd-rename-window.c cmd-select-window.c \
cmd-list-windows.c cmd-attach-session.c cmd-send-prefix.c \ cmd-list-windows.c cmd-attach-session.c cmd-send-prefix.c \
cmd-refresh-client.c cmd-kill-window.c cmd-list-clients.c \ cmd-refresh-client.c cmd-kill-window.c cmd-list-clients.c \
cmd-has-session.c cmd-link-window.c cmd-unlink-window.c \ cmd-link-window.c cmd-unlink-window.c cmd-next-window.c \
cmd-swap-window.c cmd-rename-session.c cmd-kill-session.c \ cmd-swap-window.c cmd-rename-session.c cmd-kill-session.c \
cmd-switch-client.c cmd-switch-client.c cmd-has-session.c
CC?= cc CC?= cc
INCDIRS+= -I. -I- -I/usr/local/include INCDIRS+= -I. -I- -I/usr/local/include

View File

@ -1,4 +1,4 @@
/* $Id: cmd-list-windows.c,v 1.8 2007-11-16 21:12:31 nicm Exp $ */ /* $Id: cmd-list-windows.c,v 1.9 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -49,7 +49,7 @@ cmd_list_windows_exec(unused void *ptr, struct cmd_ctx *ctx)
w = wl->window; w = wl->window;
ctx->print(ctx, "%d: %s \"%s\" (%s) [%ux%u]", ctx->print(ctx, "%d: %s \"%s\" (%s) [%ux%u]",
wl->idx, w->name, w->screen.title, ttyname(w->fd), wl->idx, w->name, w->screen.title, ttyname(w->fd),
w->screen.sx, w->screen.sy); screen_size_x(&w->screen), screen_size_y(&w->screen));
} }
if (ctx->cmdclient != NULL) if (ctx->cmdclient != NULL)

130
input.c
View File

@ -1,4 +1,4 @@
/* $Id: input.c,v 1.32 2007-11-20 18:46:32 nicm Exp $ */ /* $Id: input.c,v 1.33 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -75,6 +75,13 @@ void input_handle_sequence_decstbm(struct input_ctx *);
void input_handle_sequence_sgr(struct input_ctx *); void input_handle_sequence_sgr(struct input_ctx *);
void input_handle_sequence_dsr(struct input_ctx *); void input_handle_sequence_dsr(struct input_ctx *);
#define input_limit(v, lower, upper) do { \
if (v < lower) \
v = lower; \
if (v > upper) \
v = upper; \
} while (0)
int int
input_new_argument(struct input_ctx *ictx) input_new_argument(struct input_ctx *ictx)
{ {
@ -365,18 +372,16 @@ input_handle_character(u_char ch, struct input_ctx *ictx)
log_debug2("-- ch %zu: %hhu (%c)", ictx->off, ch, ch); log_debug2("-- ch %zu: %hhu (%c)", ictx->off, ch, ch);
if (s->cx > s->sx || s->cy > s->sy - 1) if (s->cx == screen_size_x(s)) {
return;
if (s->cx == s->sx) {
input_store8(ictx->b, '\r'); input_store8(ictx->b, '\r');
input_store8(ictx->b, '\n'); input_store8(ictx->b, '\n');
s->cx = 0; s->cx = 0;
screen_cursor_down_scroll(s); screen_display_cursor_down(s);
} } else if (!screen_in_x(s, s->cx) || !screen_in_y(s, s->cy))
return;
screen_write_character(s, ch); screen_display_cursor_set(s, ch);
input_store8(ictx->b, ch); input_store8(ictx->b, ch);
s->cx++; s->cx++;
@ -393,7 +398,7 @@ input_handle_c0_control(u_char ch, struct input_ctx *ictx)
case '\0': /* NUL */ case '\0': /* NUL */
break; break;
case '\n': /* LF */ case '\n': /* LF */
screen_cursor_down_scroll(s); screen_display_cursor_down(s);
break; break;
case '\r': /* CR */ case '\r': /* CR */
s->cx = 0; s->cx = 0;
@ -407,9 +412,9 @@ input_handle_c0_control(u_char ch, struct input_ctx *ictx)
break; break;
case '\011': /* TAB */ case '\011': /* TAB */
s->cx = ((s->cx / 8) * 8) + 8; s->cx = ((s->cx / 8) * 8) + 8;
if (s->cx > s->sx) { if (s->cx > screen_last_x(s)) {
s->cx = 0; s->cx = 0;
screen_cursor_down_scroll(s); screen_display_cursor_down(s);
} }
input_store_two( input_store_two(
ictx->b, CODE_CURSORMOVE, s->cy + 1, s->cx + 1); ictx->b, CODE_CURSORMOVE, s->cy + 1, s->cx + 1);
@ -430,7 +435,7 @@ input_handle_c1_control(u_char ch, struct input_ctx *ictx)
switch (ch) { switch (ch) {
case 'M': /* RI */ case 'M': /* RI */
screen_cursor_up_scroll(s); screen_display_cursor_up(s);
input_store_zero(ictx->b, CODE_REVERSEINDEX); input_store_zero(ictx->b, CODE_REVERSEINDEX);
break; break;
default: default:
@ -521,7 +526,7 @@ input_handle_sequence(u_char ch, struct input_ctx *ictx)
log_debug2("-- sq %zu: %hhu (%c): %u [sx=%u, sy=%u, cx=%u, cy=%u]", log_debug2("-- sq %zu: %hhu (%c): %u [sx=%u, sy=%u, cx=%u, cy=%u]",
ictx->off, ch, ch, ARRAY_LENGTH(&ictx->args), ictx->off, ch, ch, ARRAY_LENGTH(&ictx->args),
s->sx, s->sy, s->cx, s->cy); screen_size_x(s), screen_size_y(s), s->cx, s->cy);
for (i = 0; i < ARRAY_LENGTH(&ictx->args); i++) { for (i = 0; i < ARRAY_LENGTH(&ictx->args); i++) {
iarg = &ARRAY_ITEM(&ictx->args, i); iarg = &ARRAY_ITEM(&ictx->args, i);
if (*iarg->data != '\0') if (*iarg->data != '\0')
@ -576,10 +581,9 @@ input_handle_sequence_cud(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sy - s->cy - 1) { if (n == 0)
log_debug3("cud: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_last_y(s) - s->cy);
s->cy += n; s->cy += n;
input_store_one(ictx->b, CODE_CURSORDOWN, n); input_store_one(ictx->b, CODE_CURSORDOWN, n);
@ -599,10 +603,9 @@ input_handle_sequence_cuf(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sx - s->cx - 1) { if (n == 0)
log_debug3("cuf: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_last_x(s) - s->cx);
s->cx += n; s->cx += n;
input_store_one(ictx->b, CODE_CURSORRIGHT, n); input_store_one(ictx->b, CODE_CURSORRIGHT, n);
@ -645,12 +648,11 @@ input_handle_sequence_dch(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sx - s->cx - 1) { if (n == 0)
log_debug3("dch: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_last_x(s) - s->cx);
screen_delete_characters(s, s->cx, s->cy, n); screen_display_delete_characters(s, s->cx, s->cy, n);
input_store_one(ictx->b, CODE_DELETECHARACTER, n); input_store_one(ictx->b, CODE_DELETECHARACTER, n);
} }
@ -668,15 +670,14 @@ input_handle_sequence_dl(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sy - s->cy - 1) { if (n == 0)
log_debug3("dl: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_last_y(s) - s->cy);
if (s->cy < s->rupper || s->cy > s->rlower) if (s->cy < s->rupper || s->cy > s->rlower)
screen_delete_lines(s, s->cy, n); screen_display_delete_lines(s, s->cy, n);
else else
screen_delete_lines_region(s, s->cy, n); screen_display_delete_lines_region(s, s->cy, n);
input_store_one(ictx->b, CODE_DELETELINE, n); input_store_one(ictx->b, CODE_DELETELINE, n);
} }
@ -694,12 +695,11 @@ input_handle_sequence_ich(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sx - s->cx - 1) { if (n == 0)
log_debug3("ich: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_last_x(s) - s->cx);
screen_insert_characters(s, s->cx, s->cy, n); screen_display_insert_characters(s, s->cx, s->cy, n);
input_store_one(ictx->b, CODE_INSERTCHARACTER, n); input_store_one(ictx->b, CODE_INSERTCHARACTER, n);
} }
@ -717,14 +717,14 @@ input_handle_sequence_il(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sy - s->cy - 1) { if (n == 0)
log_debug3("il: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_last_y(s) - s->cy);
if (s->cy < s->rupper || s->cy > s->rlower) if (s->cy < s->rupper || s->cy > s->rlower)
screen_insert_lines(s, s->cy, n); screen_display_insert_lines(s, s->cy, n);
else else
screen_insert_lines_region(s, s->cy, n); screen_display_insert_lines_region(s, s->cy, n);
input_store_one(ictx->b, CODE_INSERTLINE, n); input_store_one(ictx->b, CODE_INSERTLINE, n);
} }
@ -742,10 +742,9 @@ input_handle_sequence_vpa(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sy) { if (n == 0)
log_debug3("vpa: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_size_y(s));
s->cy = n - 1; s->cy = n - 1;
input_store_two(ictx->b, CODE_CURSORMOVE, n, s->cx + 1); input_store_two(ictx->b, CODE_CURSORMOVE, n, s->cx + 1);
@ -765,10 +764,9 @@ input_handle_sequence_hpa(struct input_ctx *ictx)
if (input_get_argument(ictx, 0, &n, 1) != 0) if (input_get_argument(ictx, 0, &n, 1) != 0)
return; return;
if (n == 0 || n > s->sx) { if (n == 0)
log_debug3("hpa: out of range: %hu", n);
return; return;
} input_limit(n, 1, screen_size_x(s));
s->cx = n - 1; s->cx = n - 1;
input_store_two(ictx->b, CODE_CURSORMOVE, s->cy + 1, n); input_store_two(ictx->b, CODE_CURSORMOVE, s->cy + 1, n);
@ -790,14 +788,8 @@ input_handle_sequence_cup(struct input_ctx *ictx)
if (input_get_argument(ictx, 1, &m, 1) != 0) if (input_get_argument(ictx, 1, &m, 1) != 0)
return; return;
if (n == 0) input_limit(n, 1, screen_size_y(s));
n = 1; input_limit(m, 1, screen_size_x(s));
if (n > s->sy)
n = s->sy;
if (m == 0)
m = 1;
if (m > s->sx)
m = s->sx;
s->cx = m - 1; s->cx = m - 1;
s->cy = n - 1; s->cy = n - 1;
@ -824,10 +816,11 @@ input_handle_sequence_ed(struct input_ctx *ictx)
switch (n) { switch (n) {
case 0: case 0:
screen_fill_end_of_screen( screen_display_fill_cursor_eos(
s, 0, s->cy, SCREEN_DEFDATA, s->attr, s->colr); s, SCREEN_DEFDATA, s->attr, s->colr);
input_store_zero(ictx->b, CODE_CLEARLINE); input_store_zero(ictx->b, CODE_CLEARLINE);
for (i = s->cy + 1; i < s->sy; i++) { for (i = s->cy + 1; i < screen_size_y(s); i++) {
input_store_two(ictx->b, CODE_CURSORMOVE, i + 1, 1); input_store_two(ictx->b, CODE_CURSORMOVE, i + 1, 1);
input_store_zero(ictx->b, CODE_CLEARLINE); input_store_zero(ictx->b, CODE_CLEARLINE);
} }
@ -835,8 +828,10 @@ input_handle_sequence_ed(struct input_ctx *ictx)
ictx->b, CODE_CURSORMOVE, s->cy + 1, s->cx + 1); ictx->b, CODE_CURSORMOVE, s->cy + 1, s->cx + 1);
break; break;
case 2: case 2:
screen_fill_screen(s, SCREEN_DEFDATA, s->attr, s->colr); screen_display_fill_lines(
for (i = 0; i < s->sy; i++) { s, 0, screen_size_y(s), SCREEN_DEFDATA, s->attr, s->colr);
for (i = 0; i < screen_size_y(s); i++) {
input_store_two(ictx->b, CODE_CURSORMOVE, i + 1, 1); input_store_two(ictx->b, CODE_CURSORMOVE, i + 1, 1);
input_store_zero(ictx->b, CODE_CLEARLINE); input_store_zero(ictx->b, CODE_CLEARLINE);
} }
@ -865,17 +860,18 @@ input_handle_sequence_el(struct input_ctx *ictx)
switch (n) { switch (n) {
case 0: case 0:
screen_fill_end_of_line( screen_display_fill_cursor_eol(
s, s->cx, s->cy, SCREEN_DEFDATA, s->attr, s->colr); s, SCREEN_DEFDATA, s->attr, s->colr);
input_store_zero(ictx->b, CODE_CLEARENDOFLINE); input_store_zero(ictx->b, CODE_CLEARENDOFLINE);
break; break;
case 1: case 1:
screen_fill_start_of_line( screen_display_fill_cursor_bol(
s, s->cx, s->cy, SCREEN_DEFDATA, s->attr, s->colr); s, SCREEN_DEFDATA, s->attr, s->colr);
input_store_zero(ictx->b, CODE_CLEARSTARTOFLINE); input_store_zero(ictx->b, CODE_CLEARSTARTOFLINE);
break; break;
case 2: case 2:
screen_fill_line(s, s->cy, SCREEN_DEFDATA, s->attr, s->colr); screen_display_fill_line(
s, s->cy, SCREEN_DEFDATA, s->attr, s->colr);
input_store_zero(ictx->b, CODE_CLEARLINE); input_store_zero(ictx->b, CODE_CLEARLINE);
break; break;
} }
@ -1008,17 +1004,11 @@ input_handle_sequence_decstbm(struct input_ctx *ictx)
/* XXX this will catch [0;0r and [;r etc too, is this right? */ /* XXX this will catch [0;0r and [;r etc too, is this right? */
if (n == 0 && m == 0) { if (n == 0 && m == 0) {
n = 1; n = 1;
m = s->sy; m = screen_size_y(s);
} }
if (n == 0) input_limit(n, 1, screen_size_y(s));
n = 1; input_limit(m, 1, screen_size_y(s));
if (n > s->sy)
n = s->sy;
if (m == 0)
m = 1;
if (m > s->sy)
m = s->sy;
if (n > m) { if (n > m) {
log_debug3("decstbm: out of range: %hu,%hu", n, m); log_debug3("decstbm: out of range: %hu,%hu", n, m);

View File

@ -1,4 +1,4 @@
/* $Id: key-bindings.c,v 1.16 2007-11-20 18:11:37 nicm Exp $ */ /* $Id: key-bindings.c,v 1.17 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -166,15 +166,18 @@ key_bindings_print(struct cmd_ctx *ctx, const char *fmt, ...)
va_list ap; va_list ap;
char *msg; char *msg;
size_t size; size_t size;
u_int i; u_int i, sx, sy;
sx = screen_size_x(s);
sy = screen_size_y(s);
buffer_ensure(c->out, sizeof hdr); buffer_ensure(c->out, sizeof hdr);
buffer_add(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr);
size = BUFFER_USED(c->out); size = BUFFER_USED(c->out);
if (line == 2 * s->sy || !(c->flags & CLIENT_HOLD)) { if (line == 2 * sy || !(c->flags & CLIENT_HOLD)) {
input_store_zero(c->out, CODE_CURSOROFF); input_store_zero(c->out, CODE_CURSOROFF);
for (i = 0; i < s->sy; i++) { for (i = 0; i < sy; i++) {
input_store_two(c->out, CODE_CURSORMOVE, i + 1, 1); input_store_two(c->out, CODE_CURSORMOVE, i + 1, 1);
input_store_zero(c->out, CODE_CLEARLINE); input_store_zero(c->out, CODE_CLEARLINE);
} }
@ -184,20 +187,19 @@ key_bindings_print(struct cmd_ctx *ctx, const char *fmt, ...)
line = 0; line = 0;
c->flags |= CLIENT_HOLD; c->flags |= CLIENT_HOLD;
} }
if (line >= s->sy) { if (line >= sy) {
input_store_two( input_store_two(c->out, CODE_CURSORMOVE, line - sy + 1, sx / 2);
c->out, CODE_CURSORMOVE, line - s->sy + 1, s->sx / 2);
} }
line++; line++;
va_start(ap, fmt); va_start(ap, fmt);
xvasprintf(&msg, fmt, ap); xvasprintf(&msg, fmt, ap);
va_end(ap); va_end(ap);
if (strlen(msg) > s->sx / 2) if (strlen(msg) > sx / 2)
msg[s->sx / 2] = '\0'; msg[sx / 2] = '\0';
buffer_write(c->out, msg, strlen(msg)); buffer_write(c->out, msg, strlen(msg));
if (line != s->sy && line != 2 * s->sy) { if (line != sy && line != 2 * sy) {
input_store8(c->out, '\r'); input_store8(c->out, '\r');
input_store8(c->out, '\n'); input_store8(c->out, '\n');
} }

View File

@ -1,4 +1,4 @@
/* $Id: resize.c,v 1.4 2007-10-19 10:21:35 nicm Exp $ */ /* $Id: resize.c,v 1.5 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -107,11 +107,12 @@ recalculate_sizes(void)
if (ssx == UINT_MAX || ssy == UINT_MAX) if (ssx == UINT_MAX || ssy == UINT_MAX)
continue; continue;
if (w->screen.sx == ssx && w->screen.sy == ssy) if (screen_size_x(&w->screen) == ssx &&
screen_size_y(&w->screen) == ssy)
continue; continue;
log_debug("window size %u,%u (was %u,%u)", log_debug("window size %u,%u (was %u,%u)", ssx, ssy,
ssx, ssy, w->screen.sx, w->screen.sy); screen_size_x(&w->screen), screen_size_y(&w->screen));
server_clear_window_cur(w); server_clear_window_cur(w);
window_resize(w, ssx, ssy); window_resize(w, ssx, ssy);

430
screen-display.c Normal file
View File

@ -0,0 +1,430 @@
/* $Id: screen-display.c,v 1.1 2007-11-20 21:42:29 nicm Exp $ */
/*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.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 <string.h>
#include "tmux.h"
/*
* Screen display modification functions. These alter the displayed portion
* of the screen.
*/
/* Create a region of lines. */
void
screen_display_make_lines(struct screen *s, u_int py, u_int ny)
{
if (ny == 0 || !screen_in_y(s, py) || !screen_in_y(s, py + ny - 1))
fatalx("bad value");
screen_make_lines(s, screen_y(s, py), ny);
}
/* Free a region of lines. */
void
screen_display_free_lines(struct screen *s, u_int py, u_int ny)
{
if (ny == 0 || !screen_in_y(s, py) || !screen_in_y(s, py + ny - 1))
fatalx("bad value");
screen_free_lines(s, screen_y(s, py), ny);
}
/* Move a set of lines. */
void
screen_display_move_lines(struct screen *s, u_int dy, u_int py, u_int ny)
{
if (ny == 0 || !screen_in_y(s, py) || !screen_in_y(s, py + ny - 1))
fatalx("bad value");
if (!screen_in_y(s, dy) || !screen_in_y(s, dy + ny - 1) || dy == py)
fatalx("bad value");
screen_move_lines(s, screen_y(s, dy), screen_y(s, py), ny);
}
/* Fill a set of lines. */
void
screen_display_fill_lines(
struct screen *s, u_int py, u_int ny, u_char data, u_char attr, u_char colr)
{
if (ny == 0 || !screen_in_y(s, py) || !screen_in_y(s, py + ny - 1))
fatalx("bad value");
screen_fill_lines(s, screen_y(s, py), ny, data, attr, colr);
}
/* Fill a set of cellss. */
void
screen_display_fill_cells(struct screen *s,
u_int px, u_int py, u_int nx, u_char data, u_char attr, u_char colr)
{
if (nx == 0 || !screen_in_x(s, px) || !screen_in_y(s, py))
fatalx("bad value");
screen_fill_cells(
s, screen_x(s, px), screen_y(s, py), nx, data, attr, colr);
}
/* Fill entire screen. */
void
screen_display_fill_screen(
struct screen *s, u_char data, u_char attr, u_char colr)
{
screen_display_fill_lines(s, 0, screen_size_y(s), data, attr, colr);
}
/* Fill end of screen from cursor. */
void
screen_display_fill_cursor_eos(
struct screen *s, u_char data, u_char attr, u_char colr)
{
screen_display_fill_cursor_eol(s, data, attr, colr);
if (s->cy != screen_last_y(s)) {
screen_display_fill_lines(
s, s->cy, screen_size_y(s) - s->cy, data, attr, colr);
}
}
/* Fill beginning of screen from cursor. */
void
screen_display_fill_cursor_bos(
struct screen *s, u_char data, u_char attr, u_char colr)
{
screen_display_fill_lines(s, 0, s->cy, data, attr, colr);
}
/* Fill a single line. */
void
screen_display_fill_line(
struct screen *s, u_int py, u_char data, u_char attr, u_char colr)
{
screen_display_fill_lines(s, py, 1, data, attr, colr);
}
/* Fill cursor to beginning of line. */
void
screen_display_fill_cursor_bol(
struct screen *s, u_char data, u_char attr, u_char colr)
{
screen_display_fill_cells(s, 0, s->cy, s->cx, data, attr, colr);
}
/* Fill cursor to end of line. */
void
screen_display_fill_cursor_eol(
struct screen *s, u_char data, u_char attr, u_char colr)
{
screen_display_fill_cells(
s, s->cx, s->cy, screen_size_x(s) - s->cx, data, attr, colr);
}
/* Set character at cursor. */
void
screen_display_cursor_set(struct screen *s, u_char ch)
{
u_int px, py;
px = screen_x(s, s->cx);
py = screen_y(s, s->cy);
s->grid_data[py][px] = ch;
s->grid_attr[py][px] = s->attr;
s->grid_colr[py][px] = s->colr;
}
/* Move cursor up and scroll if necessary. */
void
screen_display_cursor_up(struct screen *s)
{
if (s->cy == s->rupper)
screen_display_scroll_region_down(s);
else if (s->cy > 0)
s->cy--;
}
/* Move cursor down and scroll if necessary. */
void
screen_display_cursor_down(struct screen *s)
{
if (s->cy == s->rlower)
screen_display_scroll_region_up(s);
else if (s->cy < screen_last_y(s))
s->cy++;
}
/* Scroll region up. */
void
screen_display_scroll_region_up(struct screen *s)
{
/*
* Scroll scrolling region up:
* - delete rupper
* - move rupper + 1 to rlower to rupper
* - make new line at rlower
*
* Example: region is 12 to 24.
* rlower = 24, rupper = 12
* screen_free_lines(s, 12, 1);
* screen_move_lines(s, 12, 13, 12);
* screen_make_lines(s, 24, 1);
*/
screen_display_free_lines(s, s->rupper, 1);
if (s->rupper != s->rlower) {
screen_display_move_lines(s,
s->rupper, s->rupper + 1, s->rlower - s->rupper);
}
screen_display_make_lines(s, s->rlower, 1);
screen_display_fill_lines(
s, s->rlower, 1, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Scroll region down. */
void
screen_display_scroll_region_down(struct screen *s)
{
/*
* Scroll scrolling region down:
* - delete rlower
* - move rupper to rlower - 1 to rupper + 1
* - make new line at rupper
*
* Example: region is 12 to 24.
* rlower = 24, rupper = 12
* screen_free_lines(s, 24, 1);
* screen_move_lines(s, 13, 12, 12);
* screen_make_lines(s, 12, 1);
*/
screen_display_free_lines(s, s->rlower, 1);
if (s->rupper != s->rlower) {
screen_display_move_lines(s,
s->rupper + 1, s->rupper, s->rlower - s->rupper);
}
screen_display_make_lines(s, s->rupper, 1);
screen_display_fill_lines(
s, s->rupper, 1, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Insert lines. */
void
screen_display_insert_lines(struct screen *s, u_int py, u_int ny)
{
if (!screen_in_y(s, py))
fatalx("bad value");
if (ny == 0)
fatalx("bad value");
if (py + ny > screen_last_y(s))
ny = screen_last_y(s) - py;
/*
* Insert range of ny lines at py:
* - Free ny lines from end of screen.
* - Move from py to end of screen - ny to py + ny.
* - Create ny lines at py.
*
* Example: insert 2 lines at 4.
* sy = 10, py = 4, ny = 2
* screen_free_lines(s, 8, 2); - delete lines 8,9
* screen_move_lines(s, 6, 4, 4); - move 4,5,6,7 to 6,7,8,9
* screen_make_lines(s, 4, 2); - make lines 4,5
*/
screen_display_free_lines(s, screen_size_y(s) - ny, ny);
if (py != screen_last_y(s)) {
screen_display_move_lines(
s, py + ny, py, screen_size_y(s) - py - ny);
}
screen_display_make_lines(s, py, ny);
}
/* Insert lines in region. */
void
screen_display_insert_lines_region(struct screen *s, u_int py, u_int ny)
{
if (!screen_in_region(s, py))
fatalx("bad value");
if (ny == 0)
fatalx("bad value");
if (py + ny > s->rlower)
ny = s->rlower - py;
/*
* Insert range of ny lines at py:
* - Free ny lines from end of screen.
* - Move from py to end of screen - ny to py + ny.
* - Create ny lines at py.
*
* Example: insert 2 lines at 4.
* ryu = 11, ryl = 16, py = 13, ny = 2
* screen_free_lines(s, 15, 2); - delete lines 15,16
* screen_move_lines(s, 13, 15, 2);- move 13,14 to 15,16
* screen_make_lines(s, 13, 2); - make lines 13,14
*/
screen_display_free_lines(s, (s->rlower + 1) - ny, ny);
if (py != s->rlower) {
screen_display_move_lines(
s, py + ny, py, (s->rlower + 1) - py - ny);
}
screen_display_make_lines(s, py, ny);
}
/* Delete lines. */
void
screen_display_delete_lines(struct screen *s, u_int py, u_int ny)
{
if (!screen_in_y(s, py))
fatalx("bad value");
if (ny == 0)
fatalx("bad value");
if (py + ny > screen_last_y(s))
ny = screen_last_y(s) - py;
/*
* Delete range of ny lines at py:
* - Free ny lines at py.
* - Move from py + ny to end of screen to py.
* - Free and recreate last ny lines.
*
* Example: delete lines 3,4.
* sy = 10, py = 3, ny = 2
* screen_free_lines(s, 3, 2); - delete lines 3,4
* screen_move_lines(s, 3, 5, 5); - move 5,6,7,8,9 to 3
* screen_make_lines(s, 8, 2); - make lines 8,9
*/
screen_display_free_lines(s, py, ny);
if (py != screen_last_y(s)) {
screen_display_move_lines(
s, py, py + ny, screen_size_y(s) - py - ny);
}
screen_display_make_lines(s, screen_size_y(s) - ny, ny);
}
/* Delete lines inside scroll region. */
void
screen_display_delete_lines_region(struct screen *s, u_int py, u_int ny)
{
if (!screen_in_region(s, py))
fatalx("bad value");
if (ny == 0)
fatalx("bad value");
if (py + ny > s->rlower)
ny = s->rlower - py;
/*
* Delete range of ny lines at py:
* - Free ny lines at py.
* - Move from py + ny to end of region to py.
* - Free and recreate last ny lines.
*
* Example: delete lines 13,14.
* ryu = 11, ryl = 16, py = 13, ny = 2
* screen_free_lines(s, 13, 2); - delete lines 13,14
* screen_move_lines(s, 15, 16, 2);- move 15,16 to 13
* screen_make_lines(s, 15, 16); - make lines 15,16
*/
screen_display_free_lines(s, py, ny);
if (py != s->rlower) {
screen_display_move_lines(
s, py, py + ny, (s->rlower + 1) - py - ny);
}
screen_display_make_lines(s, (s->rlower + 1) - ny, ny);
}
/* Insert characters. */
void
screen_display_insert_characters(struct screen *s, u_int px, u_int py, u_int nx)
{
u_int mx;
px = screen_x(s, px);
py = screen_y(s, py);
if (!screen_in_x(s, px) || !screen_in_y(s, py))
fatalx("bad value");
if (px + nx > screen_last_x(s))
nx = screen_last_x(s) - px;
/*
* Inserting a range of nx at px.
*
* - Move sx - (px + nx) from px to px + nx.
* - Clear the range at px.
*/
if (px + nx != screen_last_x(s)) {
mx = screen_last_x(s) - (px + nx);
memmove(&s->grid_data[py][px + nx], &s->grid_data[py][px], mx);
memmove(&s->grid_attr[py][px + nx], &s->grid_attr[py][px], mx);
memmove(&s->grid_colr[py][px + nx], &s->grid_colr[py][px], mx);
}
memset(&s->grid_data[py][px], SCREEN_DEFDATA, nx);
memset(&s->grid_attr[py][px], SCREEN_DEFATTR, nx);
memset(&s->grid_colr[py][px], SCREEN_DEFCOLR, nx);
}
/* Delete characters. */
void
screen_display_delete_characters(struct screen *s, u_int px, u_int py, u_int nx)
{
u_int mx;
px = screen_x(s, px);
py = screen_y(s, py);
if (!screen_in_x(s, px) || !screen_in_y(s, py))
fatalx("bad value");
if (px + nx > screen_last_x(s))
nx = screen_last_x(s) - px;
/*
* Deleting the range from px to px + nx.
*
* - Move sx - (px + nx) from px + nx to px.
* - Clear the range from the last x - (rx - lx) to the last x.
*/
if (px + nx != screen_last_x(s)) {
mx = screen_last_x(s) - (px + nx);
memmove(&s->grid_data[py][px], &s->grid_data[py][px + nx], mx);
memmove(&s->grid_attr[py][px], &s->grid_attr[py][px + nx], mx);
memmove(&s->grid_colr[py][px], &s->grid_colr[py][px + nx], mx);
}
memset(&s->grid_data[py][screen_size_x(s) - nx], SCREEN_DEFDATA, nx);
memset(&s->grid_attr[py][screen_size_x(s) - nx], SCREEN_DEFATTR, nx);
memset(&s->grid_colr[py][screen_size_x(s) - nx], SCREEN_DEFCOLR, nx);
}

470
screen.c
View File

@ -1,4 +1,4 @@
/* $Id: screen.c,v 1.24 2007-11-20 18:46:32 nicm Exp $ */ /* $Id: screen.c,v 1.25 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -28,21 +28,6 @@
* XXX Much of this file sucks. * XXX Much of this file sucks.
*/ */
void screen_free_lines(struct screen *, u_int, u_int);
void screen_make_lines(struct screen *, u_int, u_int);
void screen_move_lines(struct screen *, u_int, u_int, u_int);
void screen_fill_lines(
struct screen *, u_int, u_int, u_char, u_char, u_char);
#define screen_last_y(s) ((s)->sy - 1)
#define screen_last_x(s) ((s)->sx - 1)
#define screen_range_y(lx, rx) (((rx) - (lx)) + 1)
#define screen_range_x(ux, lx) (((lx) - (ux)) + 1)
#define screen_offset_y(py, ny) ((py) + (ny) - 1)
#define screen_offset_x(px, nx) ((px) + (nx) - 1)
/* Colour to string. */ /* Colour to string. */
const char * const char *
screen_colourstring(u_char c) screen_colourstring(u_char c)
@ -97,15 +82,18 @@ screen_stringcolour(const char *s)
/* Create a new screen. */ /* Create a new screen. */
void void
screen_create(struct screen *s, u_int sx, u_int sy) screen_create(struct screen *s, u_int dx, u_int dy)
{ {
s->sx = sx; s->dx = dx;
s->sy = sy; s->dy = dy;
s->cx = 0; s->cx = 0;
s->cy = 0; s->cy = 0;
s->rupper = 0; s->rupper = 0;
s->rlower = screen_last_y(s); s->rlower = s->dy - 1;
s->ysize = dy;
s->ylimit = SHRT_MAX;
s->attr = SCREEN_DEFATTR; s->attr = SCREEN_DEFATTR;
s->colr = SCREEN_DEFCOLR; s->colr = SCREEN_DEFCOLR;
@ -113,11 +101,10 @@ screen_create(struct screen *s, u_int sx, u_int sy)
s->mode = MODE_CURSOR; s->mode = MODE_CURSOR;
*s->title = '\0'; *s->title = '\0';
s->grid_data = xmalloc(sy * (sizeof *s->grid_data)); s->grid_data = xmalloc(dy * (sizeof *s->grid_data));
s->grid_attr = xmalloc(sy * (sizeof *s->grid_attr)); s->grid_attr = xmalloc(dy * (sizeof *s->grid_attr));
s->grid_colr = xmalloc(sy * (sizeof *s->grid_colr)); s->grid_colr = xmalloc(dy * (sizeof *s->grid_colr));
screen_make_lines(s, 0, sy); screen_make_lines(s, 0, dy);
screen_fill_screen(s, SCREEN_DEFDATA, 0, SCREEN_DEFCOLR);
} }
/* Resize screen. */ /* Resize screen. */
@ -126,7 +113,7 @@ screen_resize(struct screen *s, u_int sx, u_int sy)
{ {
u_int i, ox, oy, ny; u_int i, ox, oy, ny;
if (sx == s->sx && sy == s->sy) if (sx == s->dx && sy == s->dy)
return; return;
if (sx < 1) if (sx < 1)
@ -134,16 +121,16 @@ screen_resize(struct screen *s, u_int sx, u_int sy)
if (sy < 1) if (sy < 1)
sy = 1; sy = 1;
ox = s->sx; ox = s->dx;
oy = s->sy; oy = s->dy;
log_debug("resizing screen (%u, %u) -> (%u, %u)", ox, oy, sx, sy); log_debug("resizing screen (%u, %u) -> (%u, %u)", ox, oy, sx, sy);
s->sx = sx; s->dx = sx;
s->sy = sy; s->dy = sy;
s->rupper = 0; s->rupper = 0;
s->rlower = screen_last_y(s); s->rlower = s->dy - 1;
if (sy < oy) { if (sy < oy) {
ny = oy - sy; ny = oy - sy;
@ -190,7 +177,7 @@ screen_resize(struct screen *s, u_int sx, u_int sy)
s->grid_data[i] = xmalloc(sx); s->grid_data[i] = xmalloc(sx);
s->grid_attr[i] = xmalloc(sx); s->grid_attr[i] = xmalloc(sx);
s->grid_colr[i] = xmalloc(sx); s->grid_colr[i] = xmalloc(sx);
screen_fill_line(s, i, screen_display_fill_line(s, i,
SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR); SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
} }
sy = oy; sy = oy;
@ -203,7 +190,7 @@ screen_resize(struct screen *s, u_int sx, u_int sy)
s->grid_attr[i] = xrealloc(s->grid_attr[i], sx, 1); s->grid_attr[i] = xrealloc(s->grid_attr[i], sx, 1);
s->grid_colr[i] = xrealloc(s->grid_colr[i], sx, 1); s->grid_colr[i] = xrealloc(s->grid_colr[i], sx, 1);
if (sx > ox) { if (sx > ox) {
screen_fill_end_of_line(s, ox, i, screen_display_fill_cells(s, ox, i, s->dx - ox,
SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFDATA, SCREEN_DEFATTR,
SCREEN_DEFCOLR); SCREEN_DEFCOLR);
} }
@ -217,7 +204,7 @@ screen_resize(struct screen *s, u_int sx, u_int sy)
void void
screen_destroy(struct screen *s) screen_destroy(struct screen *s)
{ {
screen_free_lines(s, 0, s->sy); screen_free_lines(s, 0, s->dy);
xfree(s->grid_data); xfree(s->grid_data);
xfree(s->grid_attr); xfree(s->grid_attr);
xfree(s->grid_colr); xfree(s->grid_colr);
@ -230,7 +217,7 @@ screen_draw(struct screen *s, struct buffer *b, u_int uy, u_int ly)
u_char attr, colr; u_char attr, colr;
u_int i, j; u_int i, j;
if (uy > screen_last_y(s) || ly > screen_last_y(s) || ly < uy) if (uy > s->dy - 1 || ly > s->dy - 1 || ly < uy)
fatalx("bad range"); fatalx("bad range");
/* XXX. This is naive and rough right now. */ /* XXX. This is naive and rough right now. */
@ -263,35 +250,28 @@ screen_draw(struct screen *s, struct buffer *b, u_int uy, u_int ly)
input_store_zero(b, CODE_CURSORON); input_store_zero(b, CODE_CURSORON);
} }
/* Make a range of lines. */ /* Create a range of lines. */
void void
screen_make_lines(struct screen *s, u_int py, u_int ny) screen_make_lines(struct screen *s, u_int py, u_int ny)
{ {
u_int i; u_int i;
log_debug("making lines %u,%u", py, ny);
if (py > screen_last_y(s) || py + ny - 1 > screen_last_y(s))
fatalx("bad range");
for (i = py; i < py + ny; i++) { for (i = py; i < py + ny; i++) {
s->grid_data[i] = xmalloc(s->sx); s->grid_data[i] = xmalloc(s->dx);
s->grid_attr[i] = xmalloc(s->sx); s->grid_attr[i] = xmalloc(s->dx);
s->grid_colr[i] = xmalloc(s->sx); s->grid_colr[i] = xmalloc(s->dx);
} }
screen_fill_lines(
s, py, ny, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
} }
/* Free a range of lines. */
/* Free a range of ny lines at py. */
void void
screen_free_lines(struct screen *s, u_int py, u_int ny) screen_free_lines(struct screen *s, u_int py, u_int ny)
{ {
u_int i; u_int i;
log_debug("freeing lines %u,%u", py, ny);
if (py > screen_last_y(s) || py + ny - 1 > screen_last_y(s))
fatalx("bad range");
for (i = py; i < py + ny; i++) { for (i = py; i < py + ny; i++) {
xfree(s->grid_data[i]); xfree(s->grid_data[i]);
xfree(s->grid_attr[i]); xfree(s->grid_attr[i]);
@ -303,15 +283,6 @@ screen_free_lines(struct screen *s, u_int py, u_int ny)
void void
screen_move_lines(struct screen *s, u_int dy, u_int py, u_int ny) screen_move_lines(struct screen *s, u_int dy, u_int py, u_int ny)
{ {
log_debug("moving lines %u,%u to %u", py, ny, dy);
if (py > screen_last_y(s) || py + ny - 1 > screen_last_y(s))
fatalx("bad range");
if (dy > screen_last_y(s) || dy == py)
fatalx("bad destination");
if (dy + ny - 1 > screen_last_y(s))
fatalx("bad size");
memmove( memmove(
&s->grid_data[dy], &s->grid_data[py], ny * (sizeof *s->grid_data)); &s->grid_data[dy], &s->grid_data[py], ny * (sizeof *s->grid_data));
memmove( memmove(
@ -327,383 +298,16 @@ screen_fill_lines(
{ {
u_int i; u_int i;
log_debug("filling lines %u,%u", py, ny);
if (py > screen_last_y(s) || py + ny - 1 > screen_last_y(s))
fatalx("bad range");
for (i = py; i < py + ny; i++) for (i = py; i < py + ny; i++)
screen_fill_line(s, i, data, attr, colr); screen_fill_cells(s, 0, i, s->dx, data, attr, colr);
} }
/* Write a single character to the screen at the cursor. */ /* Fill a range of cells. */
void void
screen_write_character(struct screen *s, u_char ch) screen_fill_cells(struct screen *s,
u_int px, u_int py, u_int nx, u_char data, u_char attr, u_char colr)
{ {
s->grid_data[s->cy][s->cx] = ch; memset(&s->grid_data[py][px], data, nx);
s->grid_attr[s->cy][s->cx] = s->attr; memset(&s->grid_attr[py][px], attr, nx);
s->grid_colr[s->cy][s->cx] = s->colr; memset(&s->grid_colr[py][px], colr, nx);
}
/* Move cursor up and scroll if necessary. */
void
screen_cursor_up_scroll(struct screen *s)
{
if (s->cy == s->rupper)
screen_scroll_region_down(s);
else if (s->cy > 0)
s->cy--;
}
/* Move cursor down and scroll if necessary. */
void
screen_cursor_down_scroll(struct screen *s)
{
if (s->cy == s->rlower)
screen_scroll_region_up(s);
else if (s->cy < screen_last_y(s))
s->cy++;
}
/* Scroll region up. */
void
screen_scroll_region_up(struct screen *s)
{
log_debug("scrolling region up: %u:%u", s->rupper, s->rlower);
/*
* Scroll scrolling region up:
* - delete rupper
* - move rupper + 1 to rlower to rupper
* - make new line at rlower
*
* Example: region is 12 to 24.
* rlower = 24, rupper = 12
* screen_free_lines(s, 12, 1);
* screen_move_lines(s, 12, 13, 12);
* screen_make_lines(s, 24, 1);
*/
screen_free_lines(s, s->rupper, 1);
if (s->rupper != s->rlower) {
screen_move_lines(s,
s->rupper, s->rupper + 1, s->rlower - s->rupper);
}
screen_make_lines(s, s->rlower, 1);
screen_fill_lines(
s, s->rlower, 1, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Scroll region down. */
void
screen_scroll_region_down(struct screen *s)
{
log_debug("scrolling region down: %u:%u", s->rupper, s->rlower);
/*
* Scroll scrolling region down:
* - delete rlower
* - move rupper to rlower - 1 to rupper + 1
* - make new line at rupper
*
* Example: region is 12 to 24.
* rlower = 24, rupper = 12
* screen_free_lines(s, 24, 1);
* screen_move_lines(s, 13, 12, 12);
* screen_make_lines(s, 12, 1);
*/
screen_free_lines(s, s->rlower, 1);
if (s->rupper != s->rlower) {
screen_move_lines(s,
s->rupper + 1, s->rupper, s->rlower - s->rupper);
}
screen_make_lines(s, s->rupper, 1);
screen_fill_lines(
s, s->rupper, 1, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Scroll screen up. */
void
screen_scroll_up(struct screen *s, u_int ny)
{
screen_delete_lines(s, 0, ny);
}
/* Scroll screen down. */
void
screen_scroll_down(struct screen *s, u_int ny)
{
screen_insert_lines(s, 0, ny);
}
/* Fill entire screen. */
void
screen_fill_screen(struct screen *s, u_char data, u_char attr, u_char colr)
{
screen_fill_end_of_screen(s, 0, 0, data, attr, colr);
}
/* Fill single line. */
void
screen_fill_line(
struct screen *s, u_int py, u_char data, u_char attr, u_char colr)
{
screen_fill_end_of_line(s, 0, py, data, attr, colr);
}
/* Fill to end of screen. */
void
screen_fill_end_of_screen(
struct screen *s, u_int px, u_int py, u_char data, u_char attr, u_char colr)
{
if (py > screen_last_y(s))
return;
if (px != 0) {
screen_fill_end_of_line(s, px, py, data, attr, colr);
if (py++ > screen_last_y(s))
return;
}
while (py <= screen_last_y(s)) {
screen_fill_line(s, py, data, attr, colr);
py++;
}
}
/* Fill to end of line. */
void
screen_fill_end_of_line(
struct screen *s, u_int px, u_int py, u_char data, u_char attr, u_char colr)
{
if (px > screen_last_x(s))
return;
if (py > screen_last_y(s))
return;
memset(&s->grid_data[py][px], data, s->sx - px);
memset(&s->grid_attr[py][px], attr, s->sx - px);
memset(&s->grid_colr[py][px], colr, s->sx - px);
}
/* Fill to start of line. */
void
screen_fill_start_of_line(
struct screen *s, u_int px, u_int py, u_char data, u_char attr, u_char colr)
{
if (px > screen_last_x(s))
return;
if (py > screen_last_y(s))
return;
memset(s->grid_data[py], data, px);
memset(s->grid_attr[py], attr, px);
memset(s->grid_colr[py], colr, px);
}
/* Insert lines. */
void
screen_insert_lines(struct screen *s, u_int py, u_int ny)
{
if (py > screen_last_y(s))
return;
if (py + ny > screen_last_y(s))
ny = screen_last_y(s) - py;
log_debug("inserting lines: %u,%u", py, ny);
/*
* Insert range of ny lines at py:
* - Free ny lines from end of screen.
* - Move from py to end of screen - ny to py + ny.
* - Create ny lines at py.
*
* Example: insert 2 lines at 4.
* sy = 10, py = 4, ny = 2
* screen_free_lines(s, 8, 2); - delete lines 8,9
* screen_move_lines(s, 6, 4, 4); - move 4,5,6,7 to 6,7,8,9
* screen_make_lines(s, 4, 2); - make lines 4,5
*/
screen_free_lines(s, s->sy - ny, ny);
if (py != screen_last_y(s))
screen_move_lines(s, py + ny, py, s->sy - py - ny);
screen_make_lines(s, py, ny);
screen_fill_lines(
s, py, ny, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Insert lines in region. */
void
screen_insert_lines_region(struct screen *s, u_int py, u_int ny)
{
if (py < s->rupper || py > s->rlower)
return;
if (py + ny > s->rlower)
ny = s->rlower - py;
log_debug("inserting lines in region: %u,%u (%u,%u)", py, ny,
s->rupper, s->rlower);
/*
* Insert range of ny lines at py:
* - Free ny lines from end of screen.
* - Move from py to end of screen - ny to py + ny.
* - Create ny lines at py.
*
* Example: insert 2 lines at 4.
* ryu = 11, ryl = 16, py = 13, ny = 2
* screen_free_lines(s, 15, 2); - delete lines 15,16
* screen_move_lines(s, 13, 15, 2);- move 13,14 to 15,16
* screen_make_lines(s, 13, 2); - make lines 13,14
*/
screen_free_lines(s, (s->rlower + 1) - ny, ny);
if (py != s->rlower)
screen_move_lines(s, py + ny, py, (s->rlower + 1) - py - ny);
screen_make_lines(s, py, ny);
screen_fill_lines(
s, py, ny, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Delete lines. */
void
screen_delete_lines(struct screen *s, u_int py, u_int ny)
{
if (py > screen_last_y(s))
return;
if (py + ny > screen_last_y(s))
ny = screen_last_y(s) - py;
log_debug("deleting lines: %u,%u", py, ny);
/*
* Delete range of ny lines at py:
* - Free ny lines at py.
* - Move from py + ny to end of screen to py.
* - Free and recreate last ny lines.
*
* Example: delete lines 3,4.
* sy = 10, py = 3, ny = 2
* screen_free_lines(s, 3, 2); - delete lines 3,4
* screen_move_lines(s, 3, 5, 5); - move 5,6,7,8,9 to 3
* screen_make_lines(s, 8, 2); - make lines 8,9
*/
screen_free_lines(s, py, ny);
if (py != screen_last_y(s))
screen_move_lines(s, py, py + ny, s->sy - py - ny);
screen_make_lines(s, s->sy - ny, ny);
screen_fill_lines(
s, s->sy - ny, ny, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Delete lines inside scroll region. */
void
screen_delete_lines_region(struct screen *s, u_int py, u_int ny)
{
if (py < s->rupper || py > s->rlower)
return;
if (py + ny > s->rlower)
ny = s->rlower - py;
log_debug("deleting lines in region: %u,%u (%u,%u)", py, ny,
s->rupper, s->rlower);
/*
* Delete range of ny lines at py:
* - Free ny lines at py.
* - Move from py + ny to end of region to py.
* - Free and recreate last ny lines.
*
* Example: delete lines 13,14.
* ryu = 11, ryl = 16, py = 13, ny = 2
* screen_free_lines(s, 13, 2); - delete lines 13,14
* screen_move_lines(s, 15, 16, 2);- move 15,16 to 13
* screen_make_lines(s, 15, 16); - make lines 15,16
*/
screen_free_lines(s, py, ny);
if (py != s->rlower)
screen_move_lines(s, py, py + ny, (s->rlower + 1) - py - ny);
screen_make_lines(s, (s->rlower + 1) - ny, ny);
screen_fill_lines(s, (s->rlower + 1) - ny,
ny, SCREEN_DEFDATA, SCREEN_DEFATTR, SCREEN_DEFCOLR);
}
/* Insert characters. */
void
screen_insert_characters(struct screen *s, u_int px, u_int py, u_int nx)
{
u_int lx, rx;
if (px > screen_last_x(s) || py > screen_last_y(s))
return;
lx = px;
rx = screen_offset_x(px, nx);
if (rx > screen_last_x(s))
rx = screen_last_x(s);
/*
* Inserting a range from lx to rx, inclusive.
*
* - If rx is not the last x, move from lx to rx + 1.
* - Clear the range from lx to rx.
*/
if (rx != screen_last_x(s)) {
nx = screen_range_x(rx + 1, screen_last_x(s));
memmove(&s->grid_data[py][rx + 1], &s->grid_data[py][lx], nx);
memmove(&s->grid_attr[py][rx + 1], &s->grid_attr[py][lx], nx);
memmove(&s->grid_colr[py][rx + 1], &s->grid_colr[py][lx], nx);
}
memset(&s->grid_data[py][lx], SCREEN_DEFDATA, screen_range_x(lx, rx));
memset(&s->grid_attr[py][lx], SCREEN_DEFATTR, screen_range_x(lx, rx));
memset(&s->grid_colr[py][lx], SCREEN_DEFCOLR, screen_range_x(lx, rx));
}
/* Delete characters. */
void
screen_delete_characters(struct screen *s, u_int px, u_int py, u_int nx)
{
u_int lx, rx;
if (px > screen_last_x(s) || py > screen_last_y(s))
return;
lx = px;
rx = screen_offset_x(px, nx);
if (rx > screen_last_x(s))
rx = screen_last_x(s);
/*
* Deleting the range from lx to rx, inclusive.
*
* - If rx is not the last x, move the range from rx + 1 to lx.
* - Clear the range from the last x - (rx - lx) to the last x.
*/
if (rx != screen_last_x(s)) {
nx = screen_range_x(rx + 1, screen_last_x(s));
memmove(&s->grid_data[py][lx], &s->grid_data[py][rx + 1], nx);
memmove(&s->grid_attr[py][lx], &s->grid_attr[py][rx + 1], nx);
memmove(&s->grid_colr[py][lx], &s->grid_colr[py][rx + 1], nx);
}
/* If lx == rx, then nx = 1. */
nx = screen_range_x(lx, rx);
memset(&s->grid_data[py][s->sx - nx], SCREEN_DEFDATA, nx);
memset(&s->grid_attr[py][s->sx - nx], SCREEN_DEFATTR, nx);
memset(&s->grid_colr[py][s->sx - nx], SCREEN_DEFCOLR, nx);
} }

View File

@ -1,4 +1,4 @@
/* $Id: server-fn.c,v 1.25 2007-11-20 18:11:37 nicm Exp $ */ /* $Id: server-fn.c,v 1.26 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -175,7 +175,7 @@ server_clear_client(struct client *c)
size = BUFFER_USED(c->out); size = BUFFER_USED(c->out);
input_store_zero(c->out, CODE_CURSOROFF); input_store_zero(c->out, CODE_CURSOROFF);
for (i = 0; i < s->sy; i++) { for (i = 0; i < screen_size_y(s); i++) {
input_store_two(c->out, CODE_CURSORMOVE, i + 1, 1); input_store_two(c->out, CODE_CURSORMOVE, i + 1, 1);
input_store_zero(c->out, CODE_CLEARLINE); input_store_zero(c->out, CODE_CLEARLINE);
} }
@ -200,7 +200,7 @@ server_redraw_client(struct client *c)
buffer_add(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr);
size = BUFFER_USED(c->out); size = BUFFER_USED(c->out);
screen_draw(s, c->out, 0, s->sy - 1); screen_draw(s, c->out, 0, screen_last_y(s));
size = BUFFER_USED(c->out) - size; size = BUFFER_USED(c->out) - size;
log_debug("redrawing screen, %zu bytes", size); log_debug("redrawing screen, %zu bytes", size);

81
tmux.h
View File

@ -1,4 +1,4 @@
/* $Id: tmux.h,v 1.83 2007-11-20 18:46:32 nicm Exp $ */ /* $Id: tmux.h,v 1.84 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -354,8 +354,11 @@ struct screen {
u_char **grid_attr; u_char **grid_attr;
u_char **grid_colr; u_char **grid_colr;
u_int sx; /* size x */ u_int dx; /* display x size */
u_int sy; /* size y */ u_int dy; /* display y size */
u_int ysize; /* actual y size */
u_int ylimit; /* maximum y size */
u_int rupper; /* scroll region top */ u_int rupper; /* scroll region top */
u_int rlower; /* scroll region bottom */ u_int rlower; /* scroll region bottom */
@ -373,6 +376,21 @@ struct screen {
int mode; int mode;
}; };
/* Screen display access macros. */
#define screen_x(s, x) (x)
#define screen_y(s, y) ((s)->ysize - (s)->dy + y)
#define screen_last_x(s) ((s)->dx - 1)
#define screen_last_y(s) ((s)->dy - 1)
#define screen_size_x(s) ((s)->dx)
#define screen_size_y(s) ((s)->dy)
#define screen_in_x(s, x) ((x) < screen_size_x(s))
#define screen_in_y(s, y) ((y) < screen_size_y(s))
#define screen_in_region(s, y) ((y) >= (s)->rupper && (y) <= (s)->rlower)
/* Screen default contents. */ /* Screen default contents. */
#define SCREEN_DEFDATA ' ' #define SCREEN_DEFDATA ' '
#define SCREEN_DEFATTR 0 #define SCREEN_DEFATTR 0
@ -676,6 +694,37 @@ void input_store_two(struct buffer *, u_char, uint16_t, uint16_t);
/* input-key.c */ /* input-key.c */
void input_translate_key(struct buffer *, int); void input_translate_key(struct buffer *, int);
/* screen-display.c */
void screen_display_make_lines(struct screen *, u_int, u_int);
void screen_display_free_lines(struct screen *, u_int, u_int);
void screen_display_move_lines(struct screen *, u_int, u_int, u_int);
void screen_display_fill_lines(
struct screen *, u_int, u_int, u_char, u_char, u_char);
void screen_display_fill_cells(
struct screen *, u_int, u_int, u_int, u_char, u_char, u_char);
void screen_display_fill_screen(struct screen *, u_char, u_char, u_char);
void screen_display_fill_cursor_eos(
struct screen *, u_char, u_char, u_char);
void screen_display_fill_cursor_bos(
struct screen *, u_char, u_char, u_char);
void screen_display_fill_line(
struct screen *, u_int, u_char, u_char, u_char);
void screen_display_fill_cursor_bol(
struct screen *, u_char, u_char, u_char);
void screen_display_fill_cursor_eol(
struct screen *, u_char, u_char, u_char);
void screen_display_cursor_set(struct screen *, u_char);
void screen_display_cursor_up(struct screen *);
void screen_display_cursor_down(struct screen *);
void screen_display_scroll_region_up(struct screen *);
void screen_display_scroll_region_down(struct screen *);
void screen_display_insert_lines(struct screen *, u_int, u_int);
void screen_display_insert_lines_region(struct screen *, u_int, u_int);
void screen_display_delete_lines(struct screen *, u_int, u_int);
void screen_display_delete_lines_region(struct screen *, u_int, u_int);
void screen_display_insert_characters(struct screen *, u_int, u_int, u_int);
void screen_display_delete_characters(struct screen *, u_int, u_int, u_int);
/* screen.c */ /* screen.c */
const char *screen_colourstring(u_char); const char *screen_colourstring(u_char);
u_char screen_stringcolour(const char *); u_char screen_stringcolour(const char *);
@ -683,27 +732,13 @@ void screen_create(struct screen *, u_int, u_int);
void screen_destroy(struct screen *); void screen_destroy(struct screen *);
void screen_resize(struct screen *, u_int, u_int); void screen_resize(struct screen *, u_int, u_int);
void screen_draw(struct screen *, struct buffer *, u_int, u_int); void screen_draw(struct screen *, struct buffer *, u_int, u_int);
void screen_write_character(struct screen *, u_char); void screen_make_lines(struct screen *, u_int, u_int);
void screen_insert_lines(struct screen *, u_int, u_int); void screen_free_lines(struct screen *, u_int, u_int);
void screen_insert_lines_region(struct screen *, u_int, u_int); void screen_move_lines(struct screen *, u_int, u_int, u_int);
void screen_delete_lines(struct screen *, u_int, u_int); void screen_fill_lines(
void screen_delete_lines_region(struct screen *, u_int, u_int);
void screen_insert_characters(struct screen *, u_int, u_int, u_int);
void screen_delete_characters(struct screen *, u_int, u_int, u_int);
void screen_cursor_up_scroll(struct screen *);
void screen_cursor_down_scroll(struct screen *);
void screen_scroll_region_up(struct screen *);
void screen_scroll_region_down(struct screen *);
void screen_scroll_up(struct screen *, u_int);
void screen_scroll_down(struct screen *, u_int);
void screen_fill_screen(struct screen *, u_char, u_char, u_char);
void screen_fill_line(struct screen *, u_int, u_char, u_char, u_char);
void screen_fill_end_of_screen(
struct screen *, u_int, u_int, u_char, u_char, u_char);
void screen_fill_end_of_line(
struct screen *, u_int, u_int, u_char, u_char, u_char);
void screen_fill_start_of_line(
struct screen *, u_int, u_int, u_char, u_char, u_char); struct screen *, u_int, u_int, u_char, u_char, u_char);
void screen_fill_cells(
struct screen *, u_int, u_int, u_int, u_char, u_char, u_char);
/* local.c */ /* local.c */
int local_init(struct buffer **, struct buffer **); int local_init(struct buffer **, struct buffer **);

View File

@ -1,4 +1,4 @@
/* $Id: window.c,v 1.26 2007-11-07 19:41:17 nicm Exp $ */ /* $Id: window.c,v 1.27 2007-11-20 21:42:29 nicm Exp $ */
/* /*
* Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net> * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
@ -256,7 +256,7 @@ window_resize(struct window *w, u_int sx, u_int sy)
{ {
struct winsize ws; struct winsize ws;
if (sx == w->screen.sx && sy == w->screen.sy) if (sx == screen_size_x(&w->screen) && sy == screen_size_y(&w->screen))
return (-1); return (-1);
memset(&ws, 0, sizeof ws); memset(&ws, 0, sizeof ws);