1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Windows: make the need_backend_resize mechanism consistent.

There were three separate clauses in the WM_SIZE message handler which
potentially called term_size() to resize the actual Terminal object.
Two of them (for maximisation and normal non-maximised resizing drags)
first checked if an interactive resize was in progress, and if so,
instead set the need_backend_resize, to defer the term_size call to
the end of the interactive operation. But the third, for
_un_-maximising a window, didn't have that check.

As a result, if you start with a maximised window, drag its title bar
downward from the top of the screen (which unmaximises it), and
without letting go, drag it back up again (which maximises it), the
effect would be that you'd get one call to term_size in the middle of
the drag, and a second at the end. This isn't what I intended, and it
can also cause a redraw failure in full-screen applications on the
server (such as a terminal-based text editor - I reproduced this with
emacs), in which after the second term_size the terminal doesn't
manage to redraw itself.

Now I've pulled out the common logic that was in two of those three
pieces of code (and should have been in all three) into a subroutine
wm_size_resize_term, and arranged to call that in all three cases.
This fixes the inconsistency, and also fixes the emacs redraw problem
in the edge case I describe above.
This commit is contained in:
Simon Tatham 2021-02-07 19:59:21 +00:00
parent 334688db81
commit 07d334c61d

View File

@ -2049,12 +2049,44 @@ static void free_hdc(HDC hdc)
ReleaseDC(wgs.term_hwnd, hdc); ReleaseDC(wgs.term_hwnd, hdc);
} }
static bool need_backend_resize = false;
static void wm_size_resize_term(LPARAM lParam, bool border)
{
int width = LOWORD(lParam);
int height = HIWORD(lParam);
int border_size = border ? conf_get_int(conf, CONF_window_border) : 0;
int w = (width - border_size*2) / font_width;
int h = (height - border_size*2) / font_height;
if (w < 1) w = 1;
if (h < 1) h = 1;
if (resizing) {
/*
* If we're in the middle of an interactive resize, we don't
* call term_size. This means that, firstly, the user can drag
* the size back and forth indecisively without wiping out any
* actual terminal contents, and secondly, the Terminal
* doesn't call back->size in turn for each increment of the
* resizing drag, so we don't spam the server with huge
* numbers of resize events.
*/
need_backend_resize = true;
conf_set_int(conf, CONF_height, h);
conf_set_int(conf, CONF_width, w);
} else {
term_size(term, h, w,
conf_get_int(conf, CONF_savelines));
}
}
static LRESULT CALLBACK WndProc(HWND hwnd, UINT message, static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
WPARAM wParam, LPARAM lParam) WPARAM wParam, LPARAM lParam)
{ {
HDC hdc; HDC hdc;
static bool ignore_clip = false; static bool ignore_clip = false;
static bool need_backend_resize = false;
static bool fullscr_on_max = false; static bool fullscr_on_max = false;
static bool processed_resize = false; static bool processed_resize = false;
static UINT last_mousemove = 0; static UINT last_mousemove = 0;
@ -2932,48 +2964,17 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
/* A resize, well it better be a minimize. */ /* A resize, well it better be a minimize. */
reset_window(-1); reset_window(-1);
} else { } else {
int width, height, w, h;
int window_border = conf_get_int(conf, CONF_window_border);
width = LOWORD(lParam);
height = HIWORD(lParam);
if (wParam == SIZE_MAXIMIZED) { if (wParam == SIZE_MAXIMIZED) {
was_zoomed = true; was_zoomed = true;
prev_rows = term->rows; prev_rows = term->rows;
prev_cols = term->cols; prev_cols = term->cols;
if (resize_action == RESIZE_TERM) { if (resize_action == RESIZE_TERM)
w = width / font_width; wm_size_resize_term(lParam, false);
if (w < 1) w = 1;
h = height / font_height;
if (h < 1) h = 1;
if (resizing) {
/*
* As below, if we're in the middle of an
* interactive resize we don't call
* back->size. In Windows 7, this case can
* arise in maximisation as well via the Aero
* snap UI.
*/
need_backend_resize = true;
conf_set_int(conf, CONF_height, h);
conf_set_int(conf, CONF_width, w);
} else {
term_size(term, h, w,
conf_get_int(conf, CONF_savelines));
}
}
reset_window(0); reset_window(0);
} else if (wParam == SIZE_RESTORED && was_zoomed) { } else if (wParam == SIZE_RESTORED && was_zoomed) {
was_zoomed = false; was_zoomed = false;
if (resize_action == RESIZE_TERM) { if (resize_action == RESIZE_TERM) {
w = (width-window_border*2) / font_width; wm_size_resize_term(lParam, true);
if (w < 1) w = 1;
h = (height-window_border*2) / font_height;
if (h < 1) h = 1;
term_size(term, h, w, conf_get_int(conf, CONF_savelines));
reset_window(2); reset_window(2);
} else if (resize_action != RESIZE_FONT) } else if (resize_action != RESIZE_FONT)
reset_window(2); reset_window(2);
@ -2984,24 +2985,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
} else if (resize_action == RESIZE_TERM || } else if (resize_action == RESIZE_TERM ||
(resize_action == RESIZE_EITHER && (resize_action == RESIZE_EITHER &&
!is_alt_pressed())) { !is_alt_pressed())) {
w = (width-window_border*2) / font_width; wm_size_resize_term(lParam, true);
if (w < 1) w = 1;
h = (height-window_border*2) / font_height;
if (h < 1) h = 1;
if (resizing) {
/*
* Don't call back->size in mid-resize. (To
* prevent massive numbers of resize events
* getting sent down the connection during an NT
* opaque drag.)
*/
need_backend_resize = true;
conf_set_int(conf, CONF_height, h);
conf_set_int(conf, CONF_width, w);
} else {
term_size(term, h, w, conf_get_int(conf, CONF_savelines));
}
} else { } else {
reset_window(0); reset_window(0);
} }