]> granicus.if.org Git - vim/commitdiff
patch 9.0.0908: with 'smoothscroll' cursor may end up in wrong position v9.0.0908
authorYee Cheng Chin <ychin.git@gmail.com>
Sat, 19 Nov 2022 12:25:16 +0000 (12:25 +0000)
committerBram Moolenaar <Bram@vim.org>
Sat, 19 Nov 2022 12:25:16 +0000 (12:25 +0000)
Problem:    With 'smoothscroll' cursor may end up in wrong position.
Solution:   Correct the computation of screen lines. (Yee Cheng Chin,
            closes #11502)

15 files changed:
src/move.c
src/testdir/dumps/Test_smooth_long_1.dump
src/testdir/dumps/Test_smooth_long_10.dump [new file with mode: 0644]
src/testdir/dumps/Test_smooth_long_11.dump [new file with mode: 0644]
src/testdir/dumps/Test_smooth_long_12.dump [new file with mode: 0644]
src/testdir/dumps/Test_smooth_long_2.dump
src/testdir/dumps/Test_smooth_long_3.dump
src/testdir/dumps/Test_smooth_long_4.dump
src/testdir/dumps/Test_smooth_long_5.dump
src/testdir/dumps/Test_smooth_long_6.dump
src/testdir/dumps/Test_smooth_long_7.dump
src/testdir/dumps/Test_smooth_long_8.dump
src/testdir/dumps/Test_smooth_long_9.dump
src/testdir/test_scroll_opt.vim
src/version.c

index 66e8c18c59e33d1bb57eadd3f9aaec0d122a4ab9..42054455bb541ca282f6c43f45de08b0f790797d 100644 (file)
@@ -2460,18 +2460,50 @@ scroll_cursor_bot(int min_scroll, int set_topbot)
     used = curwin->w_cline_height;
 #endif
 
-    // If the cursor is below botline, we will at least scroll by the height
-    // of the cursor line.  Correct for empty lines, which are really part of
-    // botline.
+    // If the cursor is on or below botline, we will at least scroll by the
+    // height of the cursor line, which is "used".  Correct for empty lines,
+    // which are really part of botline.
     if (cln >= curwin->w_botline)
     {
        scrolled = used;
        if (cln == curwin->w_botline)
            scrolled -= curwin->w_empty_rows;
        min_scrolled = scrolled;
-       if (cln > curwin->w_botline && curwin->w_p_sms && curwin->w_p_wrap)
-           for (linenr_T lnum = curwin->w_botline + 1; lnum <= cln; ++lnum)
-               min_scrolled += PLINES_NOFILL(lnum);
+       if (curwin->w_p_sms && curwin->w_p_wrap)
+       {
+           // 'smoothscroll' and 'wrap' are set
+           if (cln > curwin->w_botline)
+               // add screen lines below w_botline
+               for (linenr_T lnum = curwin->w_botline + 1; lnum <= cln; ++lnum)
+                   min_scrolled += PLINES_NOFILL(lnum);
+
+           // Calculate how many screen lines the current top line of window
+           // occupies. If it is occupying more than the entire window, we
+           // need to scroll the additional clipped lines to scroll past the
+           // top line before we can move on to the other lines.
+           int top_plines =
+#ifdef FEAT_DIFF
+                           plines_win_nofill
+#else
+                           plines_win
+#endif
+                                       (curwin, curwin->w_topline, FALSE);
+           int skip_lines = 0;
+           int width1 = curwin->w_width - curwin_col_off();
+           int width2 = width1 + curwin_col_off2();
+           // similar formula is used in curs_columns()
+           if (curwin->w_skipcol > width1)
+               skip_lines += (curwin->w_skipcol - width1) / width2 + 1;
+           else if (curwin->w_skipcol > 0)
+               skip_lines = 1;
+
+           top_plines -= skip_lines;
+           if (top_plines > curwin->w_height)
+           {
+               scrolled += (top_plines - curwin->w_height);
+               min_scrolled += (top_plines - curwin->w_height);
+           }
+       }
     }
 
     /*
index a6f4cfb1663a8157b2851cb93344f27fe3ef0df5..86c27ed064aff269263215c542591121c40b57ed 100644 (file)
@@ -3,4 +3,4 @@
 |h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o
 |t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
-| @21|3|,|1|0| @9|B|o|t
+| @21|3|,|1|0| @9|6@1|%
diff --git a/src/testdir/dumps/Test_smooth_long_10.dump b/src/testdir/dumps/Test_smooth_long_10.dump
new file mode 100644 (file)
index 0000000..a7aee24
--- /dev/null
@@ -0,0 +1,6 @@
+>f+0&#ffffff0|o|u|r| @35
+|~+0#4040ff13&| @38
+|~| @38
+|~| @38
+|~| @38
+|:+0#0000000&|s|e|t| |s|c|r|o|l@1|o| @9|4|,|1| @10|B|o|t| 
diff --git a/src/testdir/dumps/Test_smooth_long_11.dump b/src/testdir/dumps/Test_smooth_long_11.dump
new file mode 100644 (file)
index 0000000..7f5bff7
--- /dev/null
@@ -0,0 +1,6 @@
+|<+0#4040ff13#ffffff0@2|t+0#0000000&|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t
+|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o
+|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
+|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |e|n|d| @11
+>f|o|u|r| @35
+@22|4|,|1| @10|B|o|t| 
diff --git a/src/testdir/dumps/Test_smooth_long_12.dump b/src/testdir/dumps/Test_smooth_long_12.dump
new file mode 100644 (file)
index 0000000..b82fca5
--- /dev/null
@@ -0,0 +1,6 @@
+|<+0#4040ff13#ffffff0@2|t+0#0000000&|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t
+|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o
+|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
+|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |e|n|d| @11
+|f|o|u>r| @35
+@22|4|,|4| @10|B|o|t| 
index 0c2634f18c774857280837225f16ff02956b527f..6800ba41d5b85778a9b6de8069de6f6c16b2c1f2 100644 (file)
@@ -3,4 +3,4 @@
 |t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
 |x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w
-| @21|3|,|5|0| @9|B|o|t
+| @21|3|,|5|0| @9|6@1|%
index c06144c552c55cbcabccbf40c1d9efb1d4ead9ed..63d97940e79524080dd986c2fddb2b9005f34e1d 100644 (file)
@@ -3,4 +3,4 @@
 | |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| 
 |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t
 | |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t
-| @21|3|,|2|5|0| @8|B|o|t
+| @21|3|,|2|5|0| @8|6@1|%
index 98f2c62563d9be92220ec56188bc1baa4899a50a..479dee43a1364f47d4b76b28c871905335e29fe9 100644 (file)
@@ -3,4 +3,4 @@
 |t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
 |x|t| |w|i|t|h| |l>o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w
-| @21|3|,|2|1|0| @8|B|o|t
+| @21|3|,|2|1|0| @8|6@1|%
index 901baca90d888d63abfdfeab4fba9aaf85779dfb..0b728b2eb6404812206778d1c2d46ab784525833 100644 (file)
@@ -3,4 +3,4 @@
 |h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o
 |t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i>t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
-| @21|3|,|1|7|0| @8|B|o|t
+| @21|3|,|1|7|0| @8|6@1|%
index b7ef6bf44543bd455d636888999838925a64c251..ba48c2825fa4cf9a9bf672294125f9d45e43ba56 100644 (file)
@@ -3,4 +3,4 @@
 |t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
 |x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w
-| @21|3|,|9|0| @9|B|o|t
+| @21|3|,|9|0| @9|6@1|%
index 27cb393383ac6a593513645df61e68a42375414e..222e0019f5c51ed8a334da3de80d7b2927729b37 100644 (file)
@@ -3,4 +3,4 @@
 |t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i>t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
 |x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w
-| @21|3|,|1|7|0| @8|B|o|t
+| @21|3|,|1|7|0| @8|6@1|%
index 8bdc88971c9ad5411bf8439b5f77fe8691dcf812..f4687be513771bebbd9cca41da38d3cde953de44 100644 (file)
@@ -3,4 +3,4 @@
 |t|s| |o|f| |t|e|x>t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o
 |f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
 |x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w
-|:|s|e|t| |s|c|r|o|l@1|o| @9|3|,|1|3|0| @8|B|o|t
+|:|s|e|t| |s|c|r|o|l@1|o| @9|3|,|1|3|0| @8|6@1|%
index d3e6a08fde33ae7224cf3616475c5a137d4e47bd..8365ecc0a8cdc12c25d00626a0dc6ad8c74a3623 100644 (file)
@@ -3,4 +3,4 @@
 |f| |t|e|x|t| |w|i>t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e
 |x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w
 |i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| |l|o|t|s| |o|f| |t|e|x|t| |w|i|t|h| 
-@22|3|,|1|7|0| @8|B|o|t
+@22|3|,|1|7|0| @8|6@1|%
index 3975a1fe99e34b03d70b1bef47d5ac3f2dd9b129..d40c650f3ca89af867a430a45f83b22dfce3a9ce 100644 (file)
@@ -246,7 +246,7 @@ func Test_smoothscroll_wrap_long_line()
 
   let lines =<< trim END
       vim9script
-      setline(1, ['one', 'two', 'Line' .. (' with lots of text'->repeat(30))])
+      setline(1, ['one', 'two', 'Line' .. (' with lots of text'->repeat(30)) .. ' end', 'four'])
       set smoothscroll scrolloff=0
       normal 3G10|zt
   END
@@ -287,6 +287,30 @@ func Test_smoothscroll_wrap_long_line()
   call term_sendkeys(buf, "gj")
   call term_sendkeys(buf, "\<C-Y>")
   call VerifyScreenDump(buf, 'Test_smooth_long_9', {})
+
+  " 'scrolloff' set to 0, move cursor down one line.
+  " Cursor should move properly, and since this is a really long line, it will
+  " be put on top of the screen.
+  call term_sendkeys(buf, ":set scrolloff=0\<CR>")
+  call term_sendkeys(buf, "0j")
+  call VerifyScreenDump(buf, 'Test_smooth_long_10', {})
+
+  " Repeat the step and move the cursor down again.
+  " This time, use a shorter long line that is barely long enough to span more
+  " than one window. Note that the cursor is at the bottom this time because
+  " Vim prefers to do so if we are scrolling a few lines only.
+  call term_sendkeys(buf, ":call setline(1, ['one', 'two', 'Line' .. (' with lots of text'->repeat(10)) .. ' end', 'four'])\<CR>")
+  call term_sendkeys(buf, "3Gzt")
+  call term_sendkeys(buf, "j")
+  call VerifyScreenDump(buf, 'Test_smooth_long_11', {})
+
+  " Repeat the step but this time start it when the line is smooth-scrolled by
+  " one line. This tests that the offset calculation is still correct and
+  " still end up scrolling down to the next line with cursor at bottom of
+  " screen.
+  call term_sendkeys(buf, "3Gzt")
+  call term_sendkeys(buf, "\<C-E>j")
+  call VerifyScreenDump(buf, 'Test_smooth_long_12', {})
   
   call StopVimInTerminal(buf)
 endfunc
index 588c3126ab0c6b9aa0d7e11eed0287ee5768e2a6..f6640c34d82e14fa8d54394b18f48bee0f376d0e 100644 (file)
@@ -695,6 +695,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    908,
 /**/
     907,
 /**/