mirror of
https://github.com/neovim/neovim.git
synced 2025-01-02 17:33:28 -07:00
4e4383ffa2
Problem: Divide by zero with 'smoothscroll' set and a narrow window.
Solution: Bail out when the window is too narrow.
870219c58c
Co-authored-by: Bram Moolenaar <Bram@vim.org>
650 lines
24 KiB
Lua
650 lines
24 KiB
Lua
local helpers = require('test.functional.helpers')(after_each)
|
|
local Screen = require('test.functional.ui.screen')
|
|
local clear = helpers.clear
|
|
local exec = helpers.exec
|
|
local feed = helpers.feed
|
|
|
|
before_each(clear)
|
|
|
|
describe('smoothscroll', function()
|
|
local screen
|
|
|
|
before_each(function()
|
|
screen = Screen.new(40, 12)
|
|
screen:attach()
|
|
end)
|
|
|
|
-- oldtest: Test_CtrlE_CtrlY_stop_at_end()
|
|
it('disabled does not break <C-E> and <C-Y> stop at end', function()
|
|
exec([[
|
|
enew
|
|
call setline(1, ['one', 'two'])
|
|
set number
|
|
]])
|
|
feed('<C-Y>')
|
|
screen:expect({any = " 1 ^one"})
|
|
feed('<C-E><C-E><C-E>')
|
|
screen:expect({any = " 2 ^two"})
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_CtrlE_CtrlY()
|
|
it('works with <C-E> and <C-E>', function()
|
|
exec([[
|
|
call setline(1, [ 'line one', 'word '->repeat(20), 'line three', 'long word '->repeat(7), 'line', 'line', 'line', ])
|
|
set smoothscroll scrolloff=5
|
|
:5
|
|
]])
|
|
local s1 = [[
|
|
word word word word word word word word |
|
|
word word word word word word word word |
|
|
word word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
^line |
|
|
line |
|
|
line |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s2 = [[
|
|
<<<d word word word word word word word |
|
|
word word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
^line |
|
|
line |
|
|
line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s3 = [[
|
|
<<<d word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
^line |
|
|
line |
|
|
line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s4 = [[
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
line |
|
|
line |
|
|
^line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s5 = [[
|
|
<<<d word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
line |
|
|
line |
|
|
^line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s6 = [[
|
|
<<<d word word word word word word word |
|
|
word word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
line |
|
|
line |
|
|
^line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s7 = [[
|
|
word word word word word word word word |
|
|
word word word word word word word word |
|
|
word word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
line |
|
|
line |
|
|
^line |
|
|
~ |
|
|
~ |
|
|
|
|
|
]]
|
|
local s8 = [[
|
|
line one |
|
|
word word word word word word word word |
|
|
word word word word word word word word |
|
|
word word word word |
|
|
line three |
|
|
long word long word long word long word |
|
|
long word long word long word |
|
|
line |
|
|
line |
|
|
^line |
|
|
~ |
|
|
|
|
|
]]
|
|
feed('<C-E>')
|
|
screen:expect(s1)
|
|
feed('<C-E>')
|
|
screen:expect(s2)
|
|
feed('<C-E>')
|
|
screen:expect(s3)
|
|
feed('<C-E>')
|
|
screen:expect(s4)
|
|
feed('<C-Y>')
|
|
screen:expect(s5)
|
|
feed('<C-Y>')
|
|
screen:expect(s6)
|
|
feed('<C-Y>')
|
|
screen:expect(s7)
|
|
feed('<C-Y>')
|
|
screen:expect(s8)
|
|
exec('set foldmethod=indent')
|
|
-- move the cursor so we can reuse the same dumps
|
|
feed('5G<C-E>')
|
|
screen:expect(s1)
|
|
feed('<C-E>')
|
|
screen:expect(s2)
|
|
feed('7G<C-Y>')
|
|
screen:expect(s7)
|
|
feed('<C-Y>')
|
|
screen:expect(s8)
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_number()
|
|
it("works 'number' and 'cpo'+=n", function()
|
|
exec([[
|
|
call setline(1, [ 'one ' .. 'word '->repeat(20), 'two ' .. 'long word '->repeat(7), 'line', 'line', 'line', ])
|
|
set smoothscroll scrolloff=5
|
|
set number cpo+=n
|
|
:3
|
|
func g:DoRel()
|
|
set number relativenumber scrolloff=0
|
|
:%del
|
|
call setline(1, [ 'one', 'very long text '->repeat(12), 'three', ])
|
|
exe "normal 2Gzt\<C-E>"
|
|
endfunc
|
|
]])
|
|
screen:expect([[
|
|
1 one word word word word word word wo|
|
|
rd word word word word word word word wo|
|
|
rd word word word word word |
|
|
2 two long word long word long word lo|
|
|
ng word long word long word long word |
|
|
3 ^line |
|
|
4 line |
|
|
5 line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('<C-E>')
|
|
screen:expect([[
|
|
<<<word word word word word word word wo|
|
|
rd word word word word word |
|
|
2 two long word long word long word lo|
|
|
ng word long word long word long word |
|
|
3 ^line |
|
|
4 line |
|
|
5 line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('<C-E>')
|
|
screen:expect([[
|
|
<<<word word word word word |
|
|
2 two long word long word long word lo|
|
|
ng word long word long word long word |
|
|
3 ^line |
|
|
4 line |
|
|
5 line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
exec('set cpo-=n')
|
|
screen:expect([[
|
|
<<< d word word word word word word |
|
|
2 two long word long word long word lo|
|
|
ng word long word long word long wor|
|
|
d |
|
|
3 ^line |
|
|
4 line |
|
|
5 line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('<C-Y>')
|
|
screen:expect([[
|
|
<<< rd word word word word word word wor|
|
|
d word word word word word word |
|
|
2 two long word long word long word lo|
|
|
ng word long word long word long wor|
|
|
d |
|
|
3 ^line |
|
|
4 line |
|
|
5 line |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('<C-Y>')
|
|
screen:expect([[
|
|
1 one word word word word word word wo|
|
|
rd word word word word word word wor|
|
|
d word word word word word word |
|
|
2 two long word long word long word lo|
|
|
ng word long word long word long wor|
|
|
d |
|
|
3 ^line |
|
|
4 line |
|
|
5 line |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
exec('call DoRel()')
|
|
screen:expect([[
|
|
2<<<^ong text very long text very long te|
|
|
xt very long text very long text ver|
|
|
y long text very long text very long|
|
|
text very long text very long text |
|
|
1 three |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
--No lines in buffer-- |
|
|
]])
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_list()
|
|
it("works with list mode", function()
|
|
screen:try_resize(40, 8)
|
|
exec([[
|
|
set smoothscroll scrolloff=0
|
|
set list
|
|
call setline(1, [ 'one', 'very long text '->repeat(12), 'three', ])
|
|
exe "normal 2Gzt\<C-E>"
|
|
]])
|
|
screen:expect([[
|
|
<<<t very long text very long text very |
|
|
^long text very long text very long text |
|
|
very long text very long text very long |
|
|
text very long text- |
|
|
three |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
exec('set listchars+=precedes:#')
|
|
screen:expect([[
|
|
#ext very long text very long text very |
|
|
^long text very long text very long text |
|
|
very long text very long text very long |
|
|
text very long text- |
|
|
three |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_diff_mode()
|
|
it("works with diff mode", function()
|
|
screen:try_resize(40, 8)
|
|
exec([[
|
|
let text = 'just some text here'
|
|
call setline(1, text)
|
|
set smoothscroll
|
|
diffthis
|
|
new
|
|
call setline(1, text)
|
|
set smoothscroll
|
|
diffthis
|
|
]])
|
|
screen:expect([[
|
|
- ^just some text here |
|
|
~ |
|
|
~ |
|
|
[No Name] [+] |
|
|
- just some text here |
|
|
~ |
|
|
[No Name] [+] |
|
|
|
|
|
]])
|
|
feed('<C-Y>')
|
|
screen:expect_unchanged()
|
|
feed('<C-E>')
|
|
screen:expect_unchanged()
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_wrap_scrolloff_zero()
|
|
it("works with zero 'scrolloff'", function()
|
|
screen:try_resize(40, 8)
|
|
exec([[
|
|
call setline(1, ['Line' .. (' with some text'->repeat(7))]->repeat(7))
|
|
set smoothscroll scrolloff=0 display=
|
|
:3
|
|
]])
|
|
screen:expect([[
|
|
<<<h some text with some text |
|
|
Line with some text with some text with |
|
|
some text with some text with some text |
|
|
with some text with some text |
|
|
^Line with some text with some text with |
|
|
some text with some text with some text |
|
|
with some text with some text |
|
|
|
|
|
]])
|
|
feed('j')
|
|
screen:expect_unchanged()
|
|
-- moving cursor down - whole bottom line shows
|
|
feed('<C-E>j')
|
|
screen:expect_unchanged()
|
|
feed('G')
|
|
screen:expect_unchanged()
|
|
-- moving cursor up right after the >>> marker - no need to show whole line
|
|
feed('2gj3l2k')
|
|
screen:expect([[
|
|
<<<^h some text with some text |
|
|
Line with some text with some text with |
|
|
some text with some text with some text |
|
|
with some text with some text |
|
|
Line with some text with some text with |
|
|
some text with some text with some text |
|
|
with some text with some text |
|
|
|
|
|
]])
|
|
-- moving cursor up where the >>> marker is - whole top line shows
|
|
feed('2j02k')
|
|
screen:expect([[
|
|
^Line with some text with some text with |
|
|
some text with some text with some text |
|
|
with some text with some text |
|
|
Line with some text with some text with |
|
|
some text with some text with some text |
|
|
with some text with some text |
|
|
@ |
|
|
|
|
|
]])
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_wrap_long_line()
|
|
it("adjusts the cursor position in a long line", function()
|
|
screen:try_resize(40, 6)
|
|
exec([[
|
|
call setline(1, ['one', 'two', 'Line' .. (' with lots of text'->repeat(30)) .. ' end', 'four'])
|
|
set smoothscroll scrolloff=0
|
|
normal 3G10|zt
|
|
]])
|
|
-- scrolling up, cursor moves screen line down
|
|
screen:expect([[
|
|
Line with^ lots of text with lots of text|
|
|
with lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text with lots of te|
|
|
|
|
|
]])
|
|
feed('<C-E>')
|
|
screen:expect([[
|
|
<<<th lot^s of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text with lots of te|
|
|
xt with lots of text with lots of text w|
|
|
|
|
|
]])
|
|
feed('5<C-E>')
|
|
screen:expect([[
|
|
<<< lots ^of text with lots of text with |
|
|
lots of text with lots of text with lots|
|
|
of text with lots of text with lots of |
|
|
text with lots of text with lots of text|
|
|
with lots of text with lots of text wit|
|
|
|
|
|
]])
|
|
-- scrolling down, cursor moves screen line up
|
|
feed('5<C-Y>')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text with lots of te|
|
|
xt with l^ots of text with lots of text w|
|
|
|
|
|
]])
|
|
feed('<C-Y>')
|
|
screen:expect([[
|
|
Line with lots of text with lots of text|
|
|
with lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text wi^th lots of text with lots of te|
|
|
|
|
|
]])
|
|
-- 'scrolloff' set to 1, scrolling up, cursor moves screen line down
|
|
exec('set scrolloff=1')
|
|
feed('10|<C-E>')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of^ text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text with lots of te|
|
|
xt with lots of text with lots of text w|
|
|
|
|
|
]])
|
|
-- 'scrolloff' set to 1, scrolling down, cursor moves screen line up
|
|
feed('<C-E>gjgj<C-Y>')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text wi^th lots of text with lots of te|
|
|
xt with lots of text with lots of text w|
|
|
|
|
|
]])
|
|
-- 'scrolloff' set to 2, scrolling up, cursor moves screen line down
|
|
exec('set scrolloff=2')
|
|
feed('10|<C-E>')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of tex^t with lots of text with lots o|
|
|
f text with lots of text with lots of te|
|
|
xt with lots of text with lots of text w|
|
|
|
|
|
]])
|
|
-- 'scrolloff' set to 2, scrolling down, cursor moves screen line up
|
|
feed('<C-E>gj<C-Y>')
|
|
screen:expect_unchanged()
|
|
-- '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.
|
|
exec('set scrolloff=0')
|
|
feed('0j')
|
|
screen:expect([[
|
|
<<<of text with lots of text with lots o|
|
|
f text with lots of text end |
|
|
^four |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
-- Test zt/zz/zb that they work properly when a long line is above it
|
|
feed('zb')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text end |
|
|
^four |
|
|
|
|
|
]])
|
|
feed('zz')
|
|
screen:expect([[
|
|
<<<of text with lots of text with lots o|
|
|
f text with lots of text end |
|
|
^four |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('zt')
|
|
screen:expect([[
|
|
^four |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
-- 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.
|
|
exec("call setline(1, ['one', 'two', 'Line' .. (' with lots of text'->repeat(10)) .. ' end', 'four'])")
|
|
feed('3Gztj')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text end |
|
|
^four |
|
|
|
|
|
]])
|
|
-- 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.
|
|
feed('3Gzt<C-E>j')
|
|
screen:expect([[
|
|
<<<th lots of text with lots of text wit|
|
|
h lots of text with lots of text with lo|
|
|
ts of text with lots of text with lots o|
|
|
f text with lots of text end |
|
|
fou^r |
|
|
|
|
|
]])
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_one_long_line()
|
|
it("scrolls correctly when moving the cursor", function()
|
|
screen:try_resize(40, 6)
|
|
exec([[
|
|
call setline(1, 'with lots of text '->repeat(7))
|
|
set smoothscroll scrolloff=0
|
|
]])
|
|
local s1 = [[
|
|
^with lots of text with lots of text with|
|
|
lots of text with lots of text with lot|
|
|
s of text with lots of text with lots of|
|
|
text |
|
|
~ |
|
|
|
|
|
]]
|
|
screen:expect(s1)
|
|
feed('<C-E>')
|
|
screen:expect([[
|
|
<<<ts of text with lots of text with lot|
|
|
^s of text with lots of text with lots of|
|
|
text |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('0')
|
|
screen:expect(s1)
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_long_line_showbreak()
|
|
it("cursor is not one screen line too far down", function()
|
|
screen:try_resize(40, 6)
|
|
-- a line that spans four screen lines
|
|
exec("call setline(1, 'with lots of text in one line '->repeat(6))")
|
|
exec('set smoothscroll scrolloff=0 showbreak=+++\\ ')
|
|
local s1 = [[
|
|
^with lots of text in one line with lots |
|
|
+++ of text in one line with lots of tex|
|
|
+++ t in one line with lots of text in o|
|
|
+++ ne line with lots of text in one lin|
|
|
+++ e with lots of text in one line |
|
|
|
|
|
]]
|
|
screen:expect(s1)
|
|
feed('<C-E>')
|
|
screen:expect([[
|
|
+++ ^of text in one line with lots of tex|
|
|
+++ t in one line with lots of text in o|
|
|
+++ ne line with lots of text in one lin|
|
|
+++ e with lots of text in one line |
|
|
~ |
|
|
|
|
|
]])
|
|
feed('0')
|
|
screen:expect(s1)
|
|
end)
|
|
|
|
-- oldtest: Test_smoothscroll_zero_width()
|
|
it("does not divide by zero with a narrow window", function()
|
|
screen:try_resize(12, 2)
|
|
screen:set_default_attr_ids({
|
|
[1] = {foreground = Screen.colors.Brown},
|
|
[2] = {foreground = Screen.colors.Blue1, bold = true},
|
|
})
|
|
exec([[
|
|
call setline(1, ['a'->repeat(100)])
|
|
set wrap smoothscroll number laststatus=0
|
|
wincmd v
|
|
wincmd v
|
|
wincmd v
|
|
wincmd v
|
|
]])
|
|
screen:expect([[
|
|
{1: 1^ }│{1: }│{1: }│{1: }│{1: }|
|
|
|
|
|
]])
|
|
feed('llllllllll<C-W>o')
|
|
screen:expect([[
|
|
{2:<<<}{1: }aa^aaaaaa|
|
|
|
|
|
]])
|
|
end)
|
|
end)
|