mirror of
				https://github.com/tmux/tmux.git
				synced 2025-11-04 01:34:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			446 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			446 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* $Id: screen.c,v 1.63 2008-06-29 07:04:30 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"
 | 
						|
 | 
						|
/*
 | 
						|
 * Virtual screen.
 | 
						|
 *
 | 
						|
 * A screen is stored as three arrays of lines of 8-bit values, one for the
 | 
						|
 * actual characters (data), one for attributes and one for colours. Three
 | 
						|
 * seperate blocks means memset and friends can be used. Each array is y by x
 | 
						|
 * in size, row then column order. Sizes are 0-based. There is an additional
 | 
						|
 * array of u_ints with the size of each line.
 | 
						|
 *
 | 
						|
 * Each screen has a history starting at the beginning of the arrays and
 | 
						|
 * extending for hsize lines. Beyond that is the screen display of size
 | 
						|
 * dy:
 | 
						|
 *
 | 
						|
 * ----------- array base
 | 
						|
 * |         |
 | 
						|
 * | history |
 | 
						|
 * ----------- array base + hsize
 | 
						|
 * |         |
 | 
						|
 * | display |
 | 
						|
 * |         |
 | 
						|
 * ----------- array base + hsize + dy
 | 
						|
 *
 | 
						|
 * The screen_x/screen_y macros are used to convert a cell on the displayed
 | 
						|
 * area to an absolute position in the arrays.
 | 
						|
 *
 | 
						|
 * Screen handling code is split into four files:
 | 
						|
 *
 | 
						|
 *	screen.c: Creation/deletion, utility functions, and basic functions to
 | 
						|
 *		  manipulate the screen based on offsets from the base.
 | 
						|
 *	screen-display.c: Basic functions for manipulating the displayed
 | 
						|
 *			  part of the screen. x,y coordinates passed to these
 | 
						|
 *			  are relative to the display. These are largely
 | 
						|
 *			  utility functions for screen-write.c.
 | 
						|
 *	screen-redraw.c: Functions for redrawing all or part of a screen to
 | 
						|
 *			 one or more ttys. A context is filled via one of the
 | 
						|
 *			 screen_redraw_start* variants which sets up (removes
 | 
						|
 *			 cursor etc) and figures out which tty_write_* function
 | 
						|
 *			 to use to write to the terminals, then the other
 | 
						|
 *			 screen_redraw_* functions are used to draw the screen,
 | 
						|
 *			 and screen_redraw_stop used to reset the cursor and
 | 
						|
 *			 clean up. These are used when changing window and a
 | 
						|
 *			 few other bits (status line).
 | 
						|
 * 	screen-write.c: Functions for modifying (writing into) the screen and
 | 
						|
 *			optionally simultaneously updating one or more ttys.
 | 
						|
 *			These are used in much the same way as the redraw
 | 
						|
 *			functions. These are used to update when parsing
 | 
						|
 *			input from the window (input.c) and for the various
 | 
						|
 *			other modes which maintain private screens.
 | 
						|
 *
 | 
						|
 * If you're thinking this all seems too complicated, that's because it is :-/.
 | 
						|
 */
 | 
						|
 | 
						|
/* Colour to string. */
 | 
						|
const char *
 | 
						|
screen_colourstring(u_char c)
 | 
						|
{
 | 
						|
	switch (c) {
 | 
						|
	case 0:
 | 
						|
		return ("black");
 | 
						|
	case 1:
 | 
						|
		return ("red");
 | 
						|
	case 2:
 | 
						|
		return ("green");
 | 
						|
	case 3:
 | 
						|
		return ("yellow");
 | 
						|
	case 4:
 | 
						|
		return ("blue");
 | 
						|
	case 5:
 | 
						|
		return ("magenta");
 | 
						|
	case 6:
 | 
						|
		return ("cyan");
 | 
						|
	case 7:
 | 
						|
		return ("white");
 | 
						|
	case 8:
 | 
						|
		return ("default");
 | 
						|
	}
 | 
						|
	return (NULL);
 | 
						|
}
 | 
						|
 | 
						|
/* String to colour. */
 | 
						|
u_char
 | 
						|
screen_stringcolour(const char *s)
 | 
						|
{
 | 
						|
	if (strcasecmp(s, "black") == 0 || (s[0] == '0' && s[1] == '\0'))
 | 
						|
		return (0);
 | 
						|
	if (strcasecmp(s, "red") == 0 || (s[0] == '1' && s[1] == '\0'))
 | 
						|
		return (1);
 | 
						|
	if (strcasecmp(s, "green") == 0 || (s[0] == '2' && s[1] == '\0'))
 | 
						|
		return (2);
 | 
						|
	if (strcasecmp(s, "yellow") == 0 || (s[0] == '3' && s[1] == '\0'))
 | 
						|
		return (3);
 | 
						|
	if (strcasecmp(s, "blue") == 0 || (s[0] == '4' && s[1] == '\0'))
 | 
						|
		return (4);
 | 
						|
	if (strcasecmp(s, "magenta") == 0 || (s[0] == '5' && s[1] == '\0'))
 | 
						|
		return (5);
 | 
						|
	if (strcasecmp(s, "cyan") == 0 || (s[0] == '6' && s[1] == '\0'))
 | 
						|
		return (6);
 | 
						|
	if (strcasecmp(s, "white") == 0 || (s[0] == '7' && s[1] == '\0'))
 | 
						|
		return (7);
 | 
						|
	if (strcasecmp(s, "default") == 0 || (s[0] == '8' && s[1] == '\0'))
 | 
						|
		return (8);
 | 
						|
	return (255);
 | 
						|
}
 | 
						|
 | 
						|
/* Create a new screen. */
 | 
						|
void
 | 
						|
screen_create(struct screen *s, u_int dx, u_int dy, u_int hlimit)
 | 
						|
{
 | 
						|
	s->dx = dx;
 | 
						|
	s->dy = dy;
 | 
						|
	s->cx = 0;
 | 
						|
	s->cy = 0;
 | 
						|
 | 
						|
	s->rupper = 0;
 | 
						|
	s->rlower = s->dy - 1;
 | 
						|
 | 
						|
	s->hsize = 0;
 | 
						|
	s->hlimit = hlimit;
 | 
						|
 | 
						|
	s->attr = SCREEN_DEFATTR;
 | 
						|
	s->colr = SCREEN_DEFCOLR;
 | 
						|
 | 
						|
	s->mode = MODE_CURSOR|MODE_KCURSOR|MODE_KKEYPAD;
 | 
						|
	s->title = xstrdup("");
 | 
						|
 | 
						|
	s->grid_data = xmalloc(dy * (sizeof *s->grid_data));
 | 
						|
	s->grid_attr = xmalloc(dy * (sizeof *s->grid_attr));
 | 
						|
	s->grid_colr = xmalloc(dy * (sizeof *s->grid_colr));
 | 
						|
	s->grid_size = xmalloc(dy * (sizeof *s->grid_size));
 | 
						|
	screen_make_lines(s, 0, dy);
 | 
						|
 | 
						|
	screen_clear_selection(s);
 | 
						|
}
 | 
						|
 | 
						|
/* Reinitialise screen. */
 | 
						|
void
 | 
						|
screen_reset(struct screen *s)
 | 
						|
{
 | 
						|
	s->cx = 0;
 | 
						|
	s->cy = 0;
 | 
						|
 | 
						|
	s->rupper = 0;
 | 
						|
	s->rlower = s->dy - 1;
 | 
						|
 | 
						|
	s->attr = SCREEN_DEFATTR;
 | 
						|
	s->colr = SCREEN_DEFCOLR;
 | 
						|
 | 
						|
	s->mode = MODE_CURSOR|MODE_KCURSOR|MODE_KKEYPAD;
 | 
						|
 | 
						|
	screen_display_fill_area(s, 0, 0, 
 | 
						|
	    screen_size_x(s), screen_size_y(s), ' ', 0, 0x88);
 | 
						|
	
 | 
						|
	screen_clear_selection(s);	
 | 
						|
}
 | 
						|
 | 
						|
/* Resize screen. */
 | 
						|
void
 | 
						|
screen_resize(struct screen *s, u_int sx, u_int sy)
 | 
						|
{
 | 
						|
	u_int	i, ox, oy, ny, my;
 | 
						|
 | 
						|
	if (sx < 1)
 | 
						|
		sx = 1;
 | 
						|
	if (sy < 1)
 | 
						|
		sy = 1;
 | 
						|
 | 
						|
	ox = s->dx;
 | 
						|
	oy = s->dy;
 | 
						|
	if (sx == ox && sy == oy)
 | 
						|
		return;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * X dimension.
 | 
						|
	 */
 | 
						|
	if (sx != ox) {
 | 
						|
		/*
 | 
						|
		 * If getting smaller, nuke any data in lines over the new
 | 
						|
		 * size.
 | 
						|
		 */
 | 
						|
		if (sx < ox) {
 | 
						|
			for (i = s->hsize; i < s->hsize + oy; i++) {
 | 
						|
				if (s->grid_size[i] > sx)
 | 
						|
					screen_reduce_line(s, i, sx);
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (s->cx >= sx)
 | 
						|
			s->cx = sx - 1;
 | 
						|
		s->dx = sx;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Y dimension.
 | 
						|
	 */
 | 
						|
	if (sy == oy)
 | 
						|
		return;
 | 
						|
 | 
						|
	/* Size decreasing. */
 | 
						|
	if (sy < oy) {
 | 
						|
 		ny = oy - sy;
 | 
						|
		if (s->cy != 0) {
 | 
						|
			/*
 | 
						|
			 * The cursor is not at the start. Try to remove as
 | 
						|
			 * many lines as possible from the top. (Up to the
 | 
						|
			 * cursor line.)
 | 
						|
			 */
 | 
						|
			my = s->cy;
 | 
						|
			if (my > ny)
 | 
						|
				my = ny;
 | 
						|
 | 
						|
			screen_free_lines(s, s->hsize, my);
 | 
						|
			screen_move_lines(s, s->hsize, s->hsize + my, oy - my);
 | 
						|
 | 
						|
			s->cy -= my;
 | 
						|
			oy -= my;
 | 
						|
		}
 | 
						|
 | 
						|
 		ny = oy - sy;
 | 
						|
		if (ny > 0) {
 | 
						|
			/*
 | 
						|
			 * Remove any remaining lines from the bottom.
 | 
						|
			 */
 | 
						|
			screen_free_lines(s, s->hsize + oy - ny, ny);
 | 
						|
			if (s->cy >= sy)
 | 
						|
				s->cy = sy - 1;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	/* Resize line arrays. */
 | 
						|
	ny = s->hsize + sy;
 | 
						|
	s->grid_data = xrealloc(s->grid_data, ny, sizeof *s->grid_data);
 | 
						|
	s->grid_attr = xrealloc(s->grid_attr, ny, sizeof *s->grid_attr);
 | 
						|
	s->grid_colr = xrealloc(s->grid_colr, ny, sizeof *s->grid_colr);
 | 
						|
	s->grid_size = xrealloc(s->grid_size, ny, sizeof *s->grid_size);
 | 
						|
	s->dy = sy;
 | 
						|
 | 
						|
	/* Size increasing. */
 | 
						|
	if (sy > oy)
 | 
						|
		screen_make_lines(s, s->hsize + oy, sy - oy);
 | 
						|
 | 
						|
	s->rupper = 0;
 | 
						|
	s->rlower = s->dy - 1;
 | 
						|
}
 | 
						|
 | 
						|
/* Expand line. */
 | 
						|
void
 | 
						|
screen_expand_line(struct screen *s, u_int py, u_int nx)
 | 
						|
{
 | 
						|
	u_int	ox;
 | 
						|
 | 
						|
	ox = s->grid_size[py];
 | 
						|
	s->grid_size[py] = nx;
 | 
						|
 | 
						|
	s->grid_data[py] = xrealloc(s->grid_data[py], 1, nx);
 | 
						|
	memset(&s->grid_data[py][ox], SCREEN_DEFDATA, nx - ox);
 | 
						|
	s->grid_attr[py] = xrealloc(s->grid_attr[py], 1, nx);
 | 
						|
	memset(&s->grid_attr[py][ox], SCREEN_DEFATTR, nx - ox);
 | 
						|
	s->grid_colr[py] = xrealloc(s->grid_colr[py], 1, nx);
 | 
						|
	memset(&s->grid_colr[py][ox], SCREEN_DEFCOLR, nx - ox);
 | 
						|
}
 | 
						|
 | 
						|
/* Reduce line. */
 | 
						|
void
 | 
						|
screen_reduce_line(struct screen *s, u_int py, u_int nx)
 | 
						|
{
 | 
						|
	s->grid_size[py] = nx;
 | 
						|
 | 
						|
	s->grid_data[py] = xrealloc(s->grid_data[py], 1, nx);
 | 
						|
	s->grid_attr[py] = xrealloc(s->grid_attr[py], 1, nx);
 | 
						|
	s->grid_colr[py] = xrealloc(s->grid_colr[py], 1, nx);
 | 
						|
}
 | 
						|
 | 
						|
/* Get cell. */
 | 
						|
void
 | 
						|
screen_get_cell(struct screen *s,
 | 
						|
    u_int cx, u_int cy, u_char *data, u_char *attr, u_char *colr)
 | 
						|
{
 | 
						|
	if (cx >= s->grid_size[cy]) {
 | 
						|
		*data = SCREEN_DEFDATA;
 | 
						|
		*attr = SCREEN_DEFATTR;
 | 
						|
		*colr = SCREEN_DEFCOLR;
 | 
						|
	} else {
 | 
						|
		*data = s->grid_data[cy][cx];
 | 
						|
		*attr = s->grid_attr[cy][cx];
 | 
						|
		*colr = s->grid_colr[cy][cx];
 | 
						|
	}
 | 
						|
 | 
						|
	if (screen_check_selection(s, cx, cy))
 | 
						|
		*attr |= ATTR_REVERSE;
 | 
						|
}
 | 
						|
 | 
						|
/* Set a cell. */
 | 
						|
void
 | 
						|
screen_set_cell(struct screen *s,
 | 
						|
    u_int cx, u_int cy, u_char data, u_char attr, u_char colr)
 | 
						|
{
 | 
						|
	if (cx >= s->grid_size[cy])
 | 
						|
		screen_expand_line(s, cy, cx + 1);
 | 
						|
 | 
						|
	s->grid_data[cy][cx] = data;
 | 
						|
	s->grid_attr[cy][cx] = attr;
 | 
						|
	s->grid_colr[cy][cx] = colr;
 | 
						|
}
 | 
						|
 | 
						|
/* Destroy a screen. */
 | 
						|
void
 | 
						|
screen_destroy(struct screen *s)
 | 
						|
{
 | 
						|
	xfree(s->title);
 | 
						|
	screen_free_lines(s, 0, s->dy + s->hsize);
 | 
						|
	xfree(s->grid_data);
 | 
						|
	xfree(s->grid_attr);
 | 
						|
	xfree(s->grid_colr);
 | 
						|
	xfree(s->grid_size);
 | 
						|
}
 | 
						|
 | 
						|
/* Create a range of lines. */
 | 
						|
void
 | 
						|
screen_make_lines(struct screen *s, u_int py, u_int ny)
 | 
						|
{
 | 
						|
	u_int	i;
 | 
						|
 | 
						|
	for (i = py; i < py + ny; i++) {
 | 
						|
		s->grid_data[i] = NULL;
 | 
						|
		s->grid_attr[i] = NULL;
 | 
						|
		s->grid_colr[i] = NULL;
 | 
						|
		s->grid_size[i] = 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Free a range of ny lines at py. */
 | 
						|
void
 | 
						|
screen_free_lines(struct screen *s, u_int py, u_int ny)
 | 
						|
{
 | 
						|
	u_int	i;
 | 
						|
 | 
						|
	for (i = py; i < py + ny; i++) {
 | 
						|
		if (s->grid_data[i] != NULL)
 | 
						|
			xfree(s->grid_data[i]);
 | 
						|
		s->grid_data[i] = NULL;
 | 
						|
		if (s->grid_attr[i] != NULL)
 | 
						|
			xfree(s->grid_attr[i]);
 | 
						|
		s->grid_attr[i] = NULL;
 | 
						|
		if (s->grid_colr[i] != NULL)
 | 
						|
			xfree(s->grid_colr[i]);
 | 
						|
		s->grid_colr[i] = NULL;
 | 
						|
		s->grid_size[i] = 0;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Move a range of lines. */
 | 
						|
void
 | 
						|
screen_move_lines(struct screen *s, u_int dy, u_int py, u_int ny)
 | 
						|
{
 | 
						|
	memmove(
 | 
						|
	    &s->grid_data[dy], &s->grid_data[py], ny * (sizeof *s->grid_data));
 | 
						|
	memmove(
 | 
						|
	    &s->grid_attr[dy], &s->grid_attr[py], ny * (sizeof *s->grid_attr));
 | 
						|
	memmove(
 | 
						|
	    &s->grid_colr[dy], &s->grid_colr[py], ny * (sizeof *s->grid_colr));
 | 
						|
	memmove(
 | 
						|
	    &s->grid_size[dy], &s->grid_size[py], ny * (sizeof *s->grid_size));
 | 
						|
}
 | 
						|
 | 
						|
/* Fill an area. */
 | 
						|
void
 | 
						|
screen_fill_area(struct screen *s, u_int px, u_int py,
 | 
						|
    u_int nx, u_int ny, u_char data, u_char attr, u_char colr)
 | 
						|
{
 | 
						|
	u_int	i, j;
 | 
						|
 | 
						|
	for (i = py; i < py + ny; i++) {
 | 
						|
		for (j = px; j < px + nx; j++)
 | 
						|
			screen_set_cell(s, j, i, data, attr, colr);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Set selection. */
 | 
						|
void
 | 
						|
screen_set_selection(struct screen *s, u_int sx, u_int sy, u_int ex, u_int ey)
 | 
						|
{
 | 
						|
	struct screen_sel	*sel = &s->sel;
 | 
						|
 | 
						|
	sel->flag = 1;
 | 
						|
	if (ey < sy || (sy == ey && ex < sx)) {
 | 
						|
		sel->sx = ex; sel->sy = ey;
 | 
						|
		sel->ex = sx; sel->ey = sy;
 | 
						|
	} else {
 | 
						|
		sel->sx = sx; sel->sy = sy;
 | 
						|
		sel->ex = ex; sel->ey = ey;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/* Clear selection. */
 | 
						|
void
 | 
						|
screen_clear_selection(struct screen *s)
 | 
						|
{
 | 
						|
	struct screen_sel	*sel = &s->sel;
 | 
						|
 | 
						|
	sel->flag = 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Check if cell in selection. */
 | 
						|
int
 | 
						|
screen_check_selection(struct screen *s, u_int px, u_int py)
 | 
						|
{
 | 
						|
	struct screen_sel	*sel = &s->sel;
 | 
						|
 | 
						|
	if (!sel->flag || py < sel->sy || py > sel->ey)
 | 
						|
		return (0);
 | 
						|
 | 
						|
	if (py == sel->sy && py == sel->ey) {
 | 
						|
		if (px < sel->sx || px > sel->ex)
 | 
						|
			return (0);
 | 
						|
		return (1);
 | 
						|
	}
 | 
						|
 | 
						|
	if ((py == sel->sy && px < sel->sx) || (py == sel->ey && px > sel->ex))
 | 
						|
		return (0);
 | 
						|
	return (1);
 | 
						|
}
 |