0
0
mirror of https://github.com/vim/vim.git synced 2025-09-23 03:43:49 -04:00

patch 8.2.4389: screenpos() does not handle a position in a closed fold

Problem:    screenpos() does not handle a position in a closed fold.
Solution:   Check if the position is inside a closed fold. (closes #9778)
This commit is contained in:
Bram Moolenaar
2022-02-15 13:40:17 +00:00
parent 7745f14ef3
commit 4556a2e868
3 changed files with 64 additions and 30 deletions

View File

@@ -1236,39 +1236,54 @@ textpos2screenpos(
if (pos->lnum >= wp->w_topline && pos->lnum <= wp->w_botline) if (pos->lnum >= wp->w_topline && pos->lnum <= wp->w_botline)
{ {
colnr_T off; colnr_T off;
colnr_T col; colnr_T col;
int width; int width;
linenr_T lnum = pos->lnum;
#ifdef FEAT_FOLDING
int is_folded;
row = plines_m_win(wp, wp->w_topline, pos->lnum - 1) + 1; is_folded = hasFoldingWin(wp, lnum, &lnum, NULL, TRUE, NULL);
getvcol(wp, pos, &scol, &ccol, &ecol); #endif
row = plines_m_win(wp, wp->w_topline, lnum - 1) + 1;
// similar to what is done in validate_cursor_col() #ifdef FEAT_FOLDING
col = scol; if (is_folded)
off = win_col_off(wp);
col += off;
width = wp->w_width - off + win_col_off2(wp);
// long line wrapping, adjust row
if (wp->w_p_wrap
&& col >= (colnr_T)wp->w_width
&& width > 0)
{ {
// use same formula as what is used in curs_columns()
rowoff = ((col - wp->w_width) / width + 1);
col -= rowoff * width;
}
col -= wp->w_leftcol;
if (col >= wp->w_width)
col = -1;
if (col >= 0 && row + rowoff <= wp->w_height)
{
coloff = col - scol + wp->w_wincol + 1;
row += W_WINROW(wp); row += W_WINROW(wp);
coloff = wp->w_wincol + 1;
} }
else else
// character is left, right or below of the window #endif
row = rowoff = scol = ccol = ecol = 0; {
getvcol(wp, pos, &scol, &ccol, &ecol);
// similar to what is done in validate_cursor_col()
col = scol;
off = win_col_off(wp);
col += off;
width = wp->w_width - off + win_col_off2(wp);
// long line wrapping, adjust row
if (wp->w_p_wrap
&& col >= (colnr_T)wp->w_width
&& width > 0)
{
// use same formula as what is used in curs_columns()
rowoff = ((col - wp->w_width) / width + 1);
col -= rowoff * width;
}
col -= wp->w_leftcol;
if (col >= wp->w_width)
col = -1;
if (col >= 0 && row + rowoff <= wp->w_height)
{
coloff = col - scol + wp->w_wincol + 1;
row += W_WINROW(wp);
}
else
// character is left, right or below of the window
row = rowoff = scol = ccol = ecol = 0;
}
} }
*rowp = row + rowoff; *rowp = row + rowoff;
*scolp = scol + coloff; *scolp = scol + coloff;

View File

@@ -1,5 +1,7 @@
" Tests for cursor() and other functions that get/set the cursor position " Tests for cursor() and other functions that get/set the cursor position
source check.vim
func Test_wrong_arguments() func Test_wrong_arguments()
call assert_fails('call cursor(1. 3)', 'E474:') call assert_fails('call cursor(1. 3)', 'E474:')
call assert_fails('call cursor(test_null_list())', 'E474:') call assert_fails('call cursor(test_null_list())', 'E474:')
@@ -133,12 +135,27 @@ func Test_screenpos()
bwipe! bwipe!
set display& set display&
call assert_equal({'col': 1, 'row': 1, 'endcol': 1, 'curscol': 1}, screenpos(win_getid(), 1, 1)) call assert_equal(#{col: 1, row: 1, endcol: 1, curscol: 1}, screenpos(win_getid(), 1, 1))
nmenu WinBar.TEST : nmenu WinBar.TEST :
call assert_equal({'col': 1, 'row': 2, 'endcol': 1, 'curscol': 1}, screenpos(win_getid(), 1, 1)) call assert_equal(#{col: 1, row: 2, endcol: 1, curscol: 1}, screenpos(win_getid(), 1, 1))
nunmenu WinBar.TEST nunmenu WinBar.TEST
endfunc endfunc
func Test_screenpos_fold()
CheckFeature folding
enew!
call setline(1, range(10))
3,5fold
redraw
call assert_equal(2, screenpos(1, 2, 1).row)
call assert_equal(#{col: 1, row: 3, endcol: 1, curscol: 1}, screenpos(1, 3, 1))
call assert_equal(3, screenpos(1, 4, 1).row)
call assert_equal(3, screenpos(1, 5, 1).row)
call assert_equal(4, screenpos(1, 6, 1).row)
bwipe!
endfunc
func Test_screenpos_number() func Test_screenpos_number()
rightbelow new rightbelow new
rightbelow 73vsplit rightbelow 73vsplit

View File

@@ -750,6 +750,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
4389,
/**/ /**/
4388, 4388,
/**/ /**/