1
0
mirror of https://git.savannah.gnu.org/git/emacs.git synced 2025-01-02 11:21:42 +00:00

Fix bug #18545 with lack of scrolling a window when point goes out of view.

src/xdisp.c (cursor_row_fully_visible_p): Update commentary.
 (redisplay_window): Treat the frame's frozen_window_starts flag
 the same way as the optional_new_start flag for the window: only
 obey it if the glyph row showing point will be fully visible.
 Likewise when the window start is in a continuation line.  If,
 after trying everything under the 'force_start' label, point is
 still not fully visible, give up and scroll the window.  Add
 debugging traces.
 src/window.c (Frecenter): Set the window's redisplay flag.
This commit is contained in:
Eli Zaretskii 2014-09-29 22:10:28 +03:00
parent 95e509140e
commit 5bb6d037f0
3 changed files with 64 additions and 11 deletions

View File

@ -1,3 +1,16 @@
2014-09-29 Eli Zaretskii <eliz@gnu.org>
* xdisp.c (cursor_row_fully_visible_p): Update commentary.
(redisplay_window): Treat the frame's frozen_window_starts flag
the same way as the optional_new_start flag for the window: only
obey it if the glyph row showing point will be fully visible.
Likewise when the window start is in a continuation line. If,
after trying everything under the 'force_start' label, point is
still not fully visible, give up and scroll the window. Add
debugging traces. (Bug#18545)
* window.c (Frecenter): Set the window's redisplay flag.
2014-09-24 Eli Zaretskii <eliz@gnu.org> 2014-09-24 Eli Zaretskii <eliz@gnu.org>
* w32term.c (w32_read_socket): Don't use frame dimensions for * w32term.c (w32_read_socket): Don't use frame dimensions for

View File

@ -5897,6 +5897,8 @@ and redisplay normally--don't erase and redraw the frame. */)
w->start_at_line_beg = (bytepos == BEGV_BYTE || w->start_at_line_beg = (bytepos == BEGV_BYTE ||
FETCH_BYTE (bytepos - 1) == '\n'); FETCH_BYTE (bytepos - 1) == '\n');
wset_redisplay (w);
set_buffer_internal (obuf); set_buffer_internal (obuf);
return Qnil; return Qnil;
} }

View File

@ -15027,6 +15027,10 @@ run_window_scroll_functions (Lisp_Object window, struct text_pos startp)
If FORCE_P is non-zero, return 0 even if partial visible cursor row If FORCE_P is non-zero, return 0 even if partial visible cursor row
is higher than window. is higher than window.
If CURRENT_MATRIX_P is non-zero, use the information from the
window's current glyph matrix; otherwise uze the desired glyph
matrix.
A value of 0 means the caller should do scrolling A value of 0 means the caller should do scrolling
as if point had gone off the screen. */ as if point had gone off the screen. */
@ -16136,26 +16140,48 @@ redisplay_window (Lisp_Object window, bool just_this_one_p)
/* If someone specified a new starting point but did not insist, /* If someone specified a new starting point but did not insist,
check whether it can be used. */ check whether it can be used. */
if (w->optional_new_start if ((w->optional_new_start || window_frozen_p (w))
&& CHARPOS (startp) >= BEGV && CHARPOS (startp) >= BEGV
&& CHARPOS (startp) <= ZV) && CHARPOS (startp) <= ZV)
{ {
ptrdiff_t it_charpos;
w->optional_new_start = 0; w->optional_new_start = 0;
start_display (&it, w, startp); start_display (&it, w, startp);
move_it_to (&it, PT, 0, it.last_visible_y, -1, move_it_to (&it, PT, 0, it.last_visible_y, -1,
MOVE_TO_POS | MOVE_TO_X | MOVE_TO_Y); MOVE_TO_POS | MOVE_TO_X | MOVE_TO_Y);
if (IT_CHARPOS (it) == PT) /* Record IT's position now, since line_bottom_y might change
w->force_start = 1; that. */
/* IT may overshoot PT if text at PT is invisible. */ it_charpos = IT_CHARPOS (it);
else if (IT_CHARPOS (it) > PT && CHARPOS (startp) <= PT) /* Make sure we set the force_start flag only if the cursor row
w->force_start = 1; will be fully visible. Otherwise, the code under force_start
label below will try to move point back into view, which is
not what the code which sets optional_new_start wants. */
if ((it.current_y == 0 || line_bottom_y (&it) < it.last_visible_y)
&& !w->force_start)
{
if (it_charpos == PT)
w->force_start = 1;
/* IT may overshoot PT if text at PT is invisible. */
else if (it_charpos > PT && CHARPOS (startp) <= PT)
w->force_start = 1;
#ifdef GLYPH_DEBUG
if (w->force_start)
{
if (window_frozen_p (w))
debug_method_add (w, "set force_start from frozen window start");
else
debug_method_add (w, "set force_start from optional_new_start");
}
#endif
}
} }
force_start: force_start:
/* Handle case where place to start displaying has been specified, /* Handle case where place to start displaying has been specified,
unless the specified location is outside the accessible range. */ unless the specified location is outside the accessible range. */
if (w->force_start || window_frozen_p (w)) if (w->force_start)
{ {
/* We set this later on if we have to adjust point. */ /* We set this later on if we have to adjust point. */
int new_vpos = -1; int new_vpos = -1;
@ -16200,7 +16226,7 @@ redisplay_window (Lisp_Object window, bool just_this_one_p)
goto need_larger_matrices; goto need_larger_matrices;
} }
if (w->cursor.vpos < 0 && !window_frozen_p (w)) if (w->cursor.vpos < 0)
{ {
/* If point does not appear, try to move point so it does /* If point does not appear, try to move point so it does
appear. The desired matrix has been built above, so we appear. The desired matrix has been built above, so we
@ -16293,6 +16319,11 @@ redisplay_window (Lisp_Object window, bool just_this_one_p)
} }
*/ */
} }
if (w->cursor.vpos < 0 || !cursor_row_fully_visible_p (w, 0, 0))
{
clear_glyph_matrix (w->desired_matrix);
goto try_to_scroll;
}
#ifdef GLYPH_DEBUG #ifdef GLYPH_DEBUG
debug_method_add (w, "forced window start"); debug_method_add (w, "forced window start");
@ -16357,7 +16388,8 @@ redisplay_window (Lisp_Object window, bool just_this_one_p)
|| CHARPOS (startp) == BEGV || CHARPOS (startp) == BEGV
|| !window_outdated (w))) || !window_outdated (w)))
{ {
int d1, d2, d3, d4, d5, d6; int d1, d2, d5, d6;
int rtop, rbot;
/* If first window line is a continuation line, and window start /* If first window line is a continuation line, and window start
is inside the modified region, but the first change is before is inside the modified region, but the first change is before
@ -16382,14 +16414,20 @@ redisplay_window (Lisp_Object window, bool just_this_one_p)
&& compute_window_start_on_continuation_line (w) && compute_window_start_on_continuation_line (w)
/* It doesn't make sense to force the window start like we /* It doesn't make sense to force the window start like we
do at label force_start if it is already known that point do at label force_start if it is already known that point
will not be visible in the resulting window, because will not be fully visible in the resulting window, because
doing so will move point from its correct position doing so will move point from its correct position
instead of scrolling the window to bring point into view. instead of scrolling the window to bring point into view.
See bug#9324. */ See bug#9324. */
&& pos_visible_p (w, PT, &d1, &d2, &d3, &d4, &d5, &d6)) && pos_visible_p (w, PT, &d1, &d2, &rtop, &rbot, &d5, &d6)
/* A very tall row could need more than the window height,
in which case we accept that it is partially visible. */
&& (rtop != 0) == (rbot != 0))
{ {
w->force_start = 1; w->force_start = 1;
SET_TEXT_POS_FROM_MARKER (startp, w->start); SET_TEXT_POS_FROM_MARKER (startp, w->start);
#ifdef GLYPH_DEBUG
debug_method_add (w, "recomputed window start in continuation line");
#endif
goto force_start; goto force_start;
} }