Coalesce redundant repaints, prepare to address some resizing issues (again!)

This commit is contained in:
ridiculousfish 2012-11-24 21:06:42 -08:00
parent e152cfac34
commit e557327860
2 changed files with 31 additions and 16 deletions

View file

@ -2696,6 +2696,9 @@ const wchar_t *reader_readline()
std::vector<completion_t> comp; std::vector<completion_t> comp;
int finished=0; int finished=0;
struct termios old_modes; struct termios old_modes;
/* Coalesce redundant repaints. When we get a repaint, we set this to true, and skip repaints until we get something else. */
bool coalescing_repaints = false;
/* The cycle index in our completion list */ /* The cycle index in our completion list */
size_t completion_cycle_idx = (size_t)(-1); size_t completion_cycle_idx = (size_t)(-1);
@ -2778,8 +2781,13 @@ const wchar_t *reader_readline()
break; break;
} }
/* If we get something other than a repaint, then stop coalescing them */
if (c != R_REPAINT)
coalescing_repaints = false;
if (last_char != R_YANK && last_char != R_YANK_POP) if (last_char != R_YANK && last_char != R_YANK_POP)
yank_len=0; yank_len=0;
const wchar_t *buff = data->command_line.c_str(); const wchar_t *buff = data->command_line.c_str();
switch (c) switch (c)
{ {
@ -2835,10 +2843,13 @@ const wchar_t *reader_readline()
case R_REPAINT: case R_REPAINT:
{ {
exec_prompt(); if (! coalescing_repaints)
write_loop(1, "\r", 1); {
s_reset(&data->screen, screen_reset_current_line_and_prompt); coalescing_repaints = true;
reader_repaint(); exec_prompt();
s_reset(&data->screen, screen_reset_current_line_contents);
reader_repaint();
}
break; break;
} }

View file

@ -1205,31 +1205,35 @@ void s_reset(screen_t *s, screen_reset_mode_t mode)
s->actual_lines_before_reset = maxi(s->actual_lines_before_reset, s->actual.line_count()); s->actual_lines_before_reset = maxi(s->actual_lines_before_reset, s->actual.line_count());
} }
int prev_line = s->actual.cursor.y;
if (repaint_prompt) if (repaint_prompt)
{ {
/* If the prompt is multi-line, we need to move up to the prompt's initial line. We do this by lying to ourselves and claiming that we're really below what we consider "line 0" (which is the last line of the prompt). This will cause is to move up to try to get back to line 0, but really we're getting back to the initial line of the prompt. */
const size_t prompt_line_count = calc_prompt_lines(s->actual_left_prompt);
assert(prompt_line_count >= 1);
prev_line += (prompt_line_count - 1);
/* Clear the prompt */ /* Clear the prompt */
s->actual_left_prompt.clear(); s->actual_left_prompt.clear();
} }
if (repaint_prompt && ! abandon_line)
{
/* If the prompt is multi-line, we need to move up to the prompt's initial line. We do this by lying to ourselves and claiming that we're really below what we consider "line 0" (which is the last line of the prompt). This will cause us to move up to try to get back to line 0, but really we're getting back to the initial line of the prompt. */
const size_t prompt_line_count = calc_prompt_lines(s->actual_left_prompt);
assert(prompt_line_count >= 1);
s->actual.cursor.y += (prompt_line_count - 1);
}
else if (abandon_line)
{
s->actual.cursor.y = 0;
}
s->actual.resize(0); s->actual.resize(0);
s->actual.cursor.x = 0;
s->actual.cursor.y = 0;
s->need_clear_lines = true; s->need_clear_lines = true;
s->need_clear_screen = s->need_clear_screen || clear_to_eos; s->need_clear_screen = s->need_clear_screen || clear_to_eos;
if (!abandon_line) if (! abandon_line)
{ {
/* This should prevent reseting the cursor position during the next repaint. */ /* This should prevent resetting the cursor position during the next repaint. */
write_loop(1, "\r", 1); write_loop(1, "\r", 1);
s->actual.cursor.y = prev_line; s->actual.cursor.x = 0;
} }
fstat(1, &s->prev_buff_1); fstat(1, &s->prev_buff_1);