mirror of
https://github.com/irssi/irssi.git
synced 2025-01-03 14:56:47 -05:00
Virtualized cursor movement and clearing to end of line. Now neither is done
unless really needed. git-svn-id: http://svn.irssi.org/repos/irssi/trunk@1966 dbcabf3a-b0e7-0310-adc4-f8d773084564
This commit is contained in:
parent
d4d72e5583
commit
3578cea258
@ -36,8 +36,10 @@ struct _TERM_WINDOW {
|
|||||||
|
|
||||||
TERM_WINDOW *root_window;
|
TERM_WINDOW *root_window;
|
||||||
int term_width, term_height;
|
int term_width, term_height;
|
||||||
|
char *term_lines_empty; /* 1 if line is entirely empty */
|
||||||
|
|
||||||
static int vcx, vcy, curs_visible;
|
static int vcmove, vcx, vcy, curs_visible;
|
||||||
|
static int crealx, crealy, cforcemove;
|
||||||
static int curs_x, curs_y;
|
static int curs_x, curs_y;
|
||||||
static int last_fg, last_bg, last_attrs;
|
static int last_fg, last_bg, last_attrs;
|
||||||
static int redraw_needed, redraw_tag;
|
static int redraw_needed, redraw_tag;
|
||||||
@ -66,7 +68,8 @@ int term_init(void)
|
|||||||
|
|
||||||
last_fg = last_bg = -1;
|
last_fg = last_bg = -1;
|
||||||
last_attrs = 0;
|
last_attrs = 0;
|
||||||
vcx = vcy = -1;
|
vcx = vcy = 0; crealx = crealy = -1;
|
||||||
|
vcmove = FALSE; cforcemove = TRUE;
|
||||||
curs_visible = TRUE;
|
curs_visible = TRUE;
|
||||||
|
|
||||||
current_term = terminfo_core_init(stdin, stdout);
|
current_term = terminfo_core_init(stdin, stdout);
|
||||||
@ -85,6 +88,8 @@ int term_init(void)
|
|||||||
term_height = current_term->height;
|
term_height = current_term->height;
|
||||||
root_window = term_window_create(0, 0, term_width, term_height);
|
root_window = term_window_create(0, 0, term_width, term_height);
|
||||||
|
|
||||||
|
term_lines_empty = g_new0(char, term_height);
|
||||||
|
|
||||||
term_common_init();
|
term_common_init();
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
@ -97,6 +102,36 @@ void term_deinit(void)
|
|||||||
terminfo_core_deinit(current_term);
|
terminfo_core_deinit(current_term);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void term_move_real(void)
|
||||||
|
{
|
||||||
|
if (vcx != crealx || vcy != crealy || cforcemove) {
|
||||||
|
if (curs_visible) {
|
||||||
|
terminfo_set_cursor_visible(FALSE);
|
||||||
|
curs_visible = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cforcemove) {
|
||||||
|
crealx = crealy = -1;
|
||||||
|
cforcemove = FALSE;
|
||||||
|
}
|
||||||
|
terminfo_move_relative(crealx, crealy, vcx, vcy);
|
||||||
|
crealx = vcx; crealy = vcy;
|
||||||
|
}
|
||||||
|
|
||||||
|
vcmove = FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Cursor position is unknown - move it immediately to known position */
|
||||||
|
static void term_move_reset(int x, int y)
|
||||||
|
{
|
||||||
|
if (x >= term_width) x = term_width-1;
|
||||||
|
if (y >= term_height) y = term_height-1;
|
||||||
|
|
||||||
|
vcx = x; vcy = y;
|
||||||
|
cforcemove = TRUE;
|
||||||
|
term_move_real();
|
||||||
|
}
|
||||||
|
|
||||||
/* Resize terminal - if width or height is negative,
|
/* Resize terminal - if width or height is negative,
|
||||||
the new size is unknown and should be figured out somehow */
|
the new size is unknown and should be figured out somehow */
|
||||||
void term_resize(int width, int height)
|
void term_resize(int width, int height)
|
||||||
@ -110,10 +145,13 @@ void term_resize(int width, int height)
|
|||||||
if (term_width != width || term_height != height) {
|
if (term_width != width || term_height != height) {
|
||||||
term_width = current_term->width = width;
|
term_width = current_term->width = width;
|
||||||
term_height = current_term->height = height;
|
term_height = current_term->height = height;
|
||||||
term_window_move(root_window, 0, 0, term_width, term_height);
|
term_window_move(root_window, 0, 0, term_width, term_height);
|
||||||
|
|
||||||
|
g_free(term_lines_empty);
|
||||||
|
term_lines_empty = g_new0(char, term_height);
|
||||||
}
|
}
|
||||||
|
|
||||||
vcx = vcy = -1;
|
term_move_reset(0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void term_resize_final(int width, int height)
|
void term_resize_final(int width, int height)
|
||||||
@ -135,9 +173,11 @@ void term_force_colors(int set)
|
|||||||
/* Clear screen */
|
/* Clear screen */
|
||||||
void term_clear(void)
|
void term_clear(void)
|
||||||
{
|
{
|
||||||
vcx = vcy = -1;
|
|
||||||
term_set_color(root_window, 0);
|
term_set_color(root_window, 0);
|
||||||
terminfo_clear();
|
terminfo_clear();
|
||||||
|
term_move_reset(0, 0);
|
||||||
|
|
||||||
|
memset(term_lines_empty, 1, term_height);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Beep */
|
/* Beep */
|
||||||
@ -193,8 +233,24 @@ void term_window_clear(TERM_WINDOW *window)
|
|||||||
/* Scroll window up/down */
|
/* Scroll window up/down */
|
||||||
void term_window_scroll(TERM_WINDOW *window, int count)
|
void term_window_scroll(TERM_WINDOW *window, int count)
|
||||||
{
|
{
|
||||||
vcx = vcy = -1;
|
int y, idx;
|
||||||
terminfo_scroll(window->y, window->y+window->height-1, count);
|
|
||||||
|
terminfo_scroll(window->y, window->y+window->height-1, count);
|
||||||
|
|
||||||
|
/* set the newly scrolled lines dirty */
|
||||||
|
if (count < 0) {
|
||||||
|
/* scrolling down - dirty the top */
|
||||||
|
idx = window->y;
|
||||||
|
count = -count;
|
||||||
|
} else {
|
||||||
|
/* scrolling up - dirty the bottom */
|
||||||
|
idx = window->y+window->height-count;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (y = 0; y < count; y++)
|
||||||
|
term_lines_empty[idx+y] = FALSE;
|
||||||
|
|
||||||
|
term_move_reset(vcx, vcy);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change active color */
|
/* Change active color */
|
||||||
@ -265,43 +321,63 @@ void term_set_color(TERM_WINDOW *window, int col)
|
|||||||
|
|
||||||
void term_move(TERM_WINDOW *window, int x, int y)
|
void term_move(TERM_WINDOW *window, int x, int y)
|
||||||
{
|
{
|
||||||
int newx, newy;
|
vcmove = TRUE;
|
||||||
|
vcx = x+window->x;
|
||||||
|
vcy = y+window->y;
|
||||||
|
|
||||||
if (curs_visible) {
|
if (vcx >= term_width)
|
||||||
terminfo_set_cursor_visible(FALSE);
|
vcx = term_width-1;
|
||||||
curs_visible = FALSE;
|
if (vcy >= term_height)
|
||||||
}
|
vcy = term_height-1;
|
||||||
|
}
|
||||||
|
|
||||||
newx = x+window->x;
|
static void term_printed_text(int count)
|
||||||
newy = y+window->y;
|
{
|
||||||
if (vcx != newx || vcy != newy) {
|
term_lines_empty[vcy] = FALSE;
|
||||||
terminfo_move_relative(vcx, vcy, newx, newy);
|
|
||||||
vcx = newx; vcy = newy;
|
/* if we continued writing past the line, wrap to next line.
|
||||||
|
However, next term_move() really shouldn't try to cache
|
||||||
|
the move, otherwise terminals would try to combine the
|
||||||
|
last word in upper line with first word in lower line. */
|
||||||
|
cforcemove = TRUE;
|
||||||
|
vcx += count;
|
||||||
|
while (vcx >= term_width) {
|
||||||
|
vcx -= term_width;
|
||||||
|
if (vcy < term_height) vcy++;
|
||||||
|
if (vcx > 0) term_lines_empty[vcy] = FALSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void term_addch(TERM_WINDOW *window, int chr)
|
void term_addch(TERM_WINDOW *window, int chr)
|
||||||
{
|
{
|
||||||
|
if (vcmove) term_move_real();
|
||||||
putc(chr, window->term->out);
|
putc(chr, window->term->out);
|
||||||
vcx++; /* ignore if cursor gets past the screen */
|
term_printed_text(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void term_addstr(TERM_WINDOW *window, char *str)
|
void term_addstr(TERM_WINDOW *window, char *str)
|
||||||
{
|
{
|
||||||
|
if (vcmove) term_move_real();
|
||||||
fputs(str, window->term->out);
|
fputs(str, window->term->out);
|
||||||
vcx += strlen(str); /* ignore if cursor gets past the screen */
|
term_printed_text(strlen(str));
|
||||||
}
|
}
|
||||||
|
|
||||||
void term_clrtoeol(TERM_WINDOW *window)
|
void term_clrtoeol(TERM_WINDOW *window)
|
||||||
{
|
{
|
||||||
|
/* clrtoeol() doesn't necessarily understand colors */
|
||||||
if (last_fg == -1 && last_bg == -1 &&
|
if (last_fg == -1 && last_bg == -1 &&
|
||||||
(last_attrs & (ATTR_UNDERLINE|ATTR_REVERSE)) == 0) {
|
(last_attrs & (ATTR_UNDERLINE|ATTR_REVERSE)) == 0) {
|
||||||
/* clrtoeol() doesn't necessarily understand colors */
|
if (!term_lines_empty[vcy]) {
|
||||||
terminfo_clrtoeol();
|
if (vcmove) term_move_real();
|
||||||
|
terminfo_clrtoeol();
|
||||||
|
if (vcx == 0) term_lines_empty[vcy] = TRUE;
|
||||||
|
}
|
||||||
} else if (vcx < term_width) {
|
} else if (vcx < term_width) {
|
||||||
/* we'll need to fill the line ourself. */
|
/* we'll need to fill the line ourself. */
|
||||||
|
if (vcmove) term_move_real();
|
||||||
terminfo_repeat(' ', term_width-vcx);
|
terminfo_repeat(' ', term_width-vcx);
|
||||||
terminfo_move(vcx, vcy);
|
terminfo_move(vcx, vcy);
|
||||||
|
term_lines_empty[vcy] = FALSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -316,8 +392,9 @@ void term_refresh(TERM_WINDOW *window)
|
|||||||
if (freeze_counter > 0)
|
if (freeze_counter > 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (vcx != curs_x || vcy != curs_y)
|
term_move(root_window, curs_x, curs_y);
|
||||||
term_move(root_window, curs_x, curs_y);
|
term_move_real();
|
||||||
|
|
||||||
if (!curs_visible) {
|
if (!curs_visible) {
|
||||||
terminfo_set_cursor_visible(TRUE);
|
terminfo_set_cursor_visible(TRUE);
|
||||||
curs_visible = TRUE;
|
curs_visible = TRUE;
|
||||||
|
Loading…
Reference in New Issue
Block a user