1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-25 01:02:24 +00:00

Fix missing recalculation of window border.

If you open a Windows PuTTY session and press Win+Right, Windows
auto-sizes the terminal window to cover the right-hand half of the
screen. Then if you press Win+Up it will be auto-sized again, this
time to the top right quadrant. In the second resize (if you don't
have font-based resize handling turned on), the WM_SIZE handler code
will find a path through the twisty maze of ifs on which the border
between the text and the client-area edges is not recomputed, or
invalidated, or redrawn. So you can end up with half a line of text
from the previous window size still visible at the bottom of the new
window.

Fixed by factoring out the offset-recomputation code from the large
and complicated reset_window(), so that I can call just that snippet
on the dangerous code path.
This commit is contained in:
Simon Tatham 2021-02-12 17:50:47 +00:00
parent 6bf3b49471
commit e7771a21d9

View File

@ -1686,6 +1686,25 @@ static void wintw_request_resize(TermWin *tw, int w, int h)
InvalidateRect(wgs.term_hwnd, NULL, true); InvalidateRect(wgs.term_hwnd, NULL, true);
} }
static void recompute_window_offset(void)
{
RECT cr;
GetClientRect(wgs.term_hwnd, &cr);
int win_width = cr.right - cr.left;
int win_height = cr.bottom - cr.top;
int new_offset_width = (win_width-font_width*term->cols)/2;
int new_offset_height = (win_height-font_height*term->rows)/2;
if (offset_width != new_offset_width ||
offset_height != new_offset_height) {
offset_width = new_offset_width;
offset_height = new_offset_height;
InvalidateRect(wgs.term_hwnd, NULL, true);
}
}
static void reset_window(int reinit) { static void reset_window(int reinit) {
/* /*
* This function decides how to resize or redraw when the * This function decides how to resize or redraw when the
@ -1728,12 +1747,8 @@ static void reset_window(int reinit) {
return; return;
/* Is the window out of position ? */ /* Is the window out of position ? */
if ( !reinit && if (!reinit) {
(offset_width != (win_width-font_width*term->cols)/2 || recompute_window_offset();
offset_height != (win_height-font_height*term->rows)/2) ){
offset_width = (win_width-font_width*term->cols)/2;
offset_height = (win_height-font_height*term->rows)/2;
InvalidateRect(wgs.term_hwnd, NULL, true);
#ifdef RDB_DEBUG_PATCH #ifdef RDB_DEBUG_PATCH
debug("reset_window() -> Reposition terminal\n"); debug("reset_window() -> Reposition terminal\n");
#endif #endif
@ -2986,6 +3001,22 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
(resize_action == RESIZE_EITHER && (resize_action == RESIZE_EITHER &&
!is_alt_pressed())) { !is_alt_pressed())) {
wm_size_resize_term(lParam, true); wm_size_resize_term(lParam, true);
/*
* Sometimes, we can get a spontaneous resize event
* outside a WM_SIZING interactive drag which wants to
* set us to a new specific SIZE_RESTORED size. An
* example is what happens if you press Windows+Right
* and then Windows+Up: the first operation fits the
* window to the right-hand half of the screen, and
* the second one changes that for the top right
* quadrant. In that situation, if we've responded
* here by resizing the terminal, we may still need to
* recompute the border around the window and do a
* full redraw to clear the new border.
*/
if (!resizing)
recompute_window_offset();
} else { } else {
reset_window(0); reset_window(0);
} }