mirror of
https://github.com/neovim/neovim.git
synced 2025-01-01 17:23:36 -07:00
518 lines
17 KiB
Lua
518 lines
17 KiB
Lua
local Screen = require('test.functional.ui.screen')
|
|
local helpers = require('test.functional.helpers')(after_each)
|
|
local thelpers = require('test.functional.terminal.helpers')
|
|
local clear, eq, curbuf = helpers.clear, helpers.eq, helpers.curbuf
|
|
local feed, nvim_dir, feed_command = helpers.feed, helpers.nvim_dir, helpers.feed_command
|
|
local iswin = helpers.iswin
|
|
local eval = helpers.eval
|
|
local command = helpers.command
|
|
local wait = helpers.wait
|
|
local retry = helpers.retry
|
|
local curbufmeths = helpers.curbufmeths
|
|
local nvim = helpers.nvim
|
|
local feed_data = thelpers.feed_data
|
|
|
|
describe('terminal scrollback', function()
|
|
local screen
|
|
|
|
before_each(function()
|
|
clear()
|
|
screen = thelpers.screen_setup(nil, nil, 30)
|
|
end)
|
|
|
|
after_each(function()
|
|
screen:detach()
|
|
end)
|
|
|
|
describe('when the limit is exceeded', function()
|
|
before_each(function()
|
|
local lines = {}
|
|
for i = 1, 30 do
|
|
table.insert(lines, 'line'..tostring(i))
|
|
end
|
|
table.insert(lines, '')
|
|
feed_data(lines)
|
|
screen:expect([[
|
|
line26 |
|
|
line27 |
|
|
line28 |
|
|
line29 |
|
|
line30 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
|
|
it('will delete extra lines at the top', function()
|
|
feed('<c-\\><c-n>gg')
|
|
screen:expect([[
|
|
^line16 |
|
|
line17 |
|
|
line18 |
|
|
line19 |
|
|
line20 |
|
|
line21 |
|
|
|
|
|
]])
|
|
end)
|
|
end)
|
|
|
|
describe('with cursor at last row', function()
|
|
before_each(function()
|
|
feed_data({'line1', 'line2', 'line3', 'line4', ''})
|
|
screen:expect([[
|
|
tty ready |
|
|
line1 |
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
|
|
describe('and 1 line is printed', function()
|
|
before_each(function() feed_data({'line5', ''}) end)
|
|
|
|
it('will hide the top line', function()
|
|
screen:expect([[
|
|
line1 |
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
line5 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
eq(7, curbuf('line_count'))
|
|
end)
|
|
|
|
describe('and then 3 more lines are printed', function()
|
|
before_each(function() feed_data({'line6', 'line7', 'line8'}) end)
|
|
|
|
it('will hide the top 4 lines', function()
|
|
screen:expect([[
|
|
line3 |
|
|
line4 |
|
|
line5 |
|
|
line6 |
|
|
line7 |
|
|
line8{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
|
|
feed('<c-\\><c-n>6k')
|
|
screen:expect([[
|
|
^line2 |
|
|
line3 |
|
|
line4 |
|
|
line5 |
|
|
line6 |
|
|
line7 |
|
|
|
|
|
]])
|
|
|
|
feed('gg')
|
|
screen:expect([[
|
|
^tty ready |
|
|
line1 |
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
line5 |
|
|
|
|
|
]])
|
|
|
|
feed('G')
|
|
screen:expect([[
|
|
line3 |
|
|
line4 |
|
|
line5 |
|
|
line6 |
|
|
line7 |
|
|
^line8{2: } |
|
|
|
|
|
]])
|
|
end)
|
|
end)
|
|
end)
|
|
|
|
|
|
describe('and height decreased by 1', function()
|
|
if helpers.pending_win32(pending) then return end
|
|
local function will_hide_top_line()
|
|
feed([[<C-\><C-N>:]]) -- Go to cmdline-mode, so cursor is at bottom.
|
|
screen:try_resize(screen._width - 2, screen._height - 1)
|
|
screen:expect([[
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
rows: 5, cols: 28 |
|
|
{2: } |
|
|
:^ |
|
|
]])
|
|
end
|
|
|
|
it('will hide top line', will_hide_top_line)
|
|
|
|
describe('and then decreased by 2', function()
|
|
before_each(function()
|
|
will_hide_top_line()
|
|
screen:try_resize(screen._width - 2, screen._height - 2)
|
|
end)
|
|
|
|
it('will hide the top 3 lines', function()
|
|
screen:expect([[
|
|
rows: 5, cols: 28 |
|
|
rows: 3, cols: 26 |
|
|
{2: } |
|
|
:^ |
|
|
]])
|
|
eq(8, curbuf('line_count'))
|
|
feed([[<C-\><C-N>3k]])
|
|
screen:expect([[
|
|
^line4 |
|
|
rows: 5, cols: 28 |
|
|
rows: 3, cols: 26 |
|
|
|
|
|
]])
|
|
end)
|
|
end)
|
|
end)
|
|
end)
|
|
|
|
describe('with empty lines after the cursor', function()
|
|
-- XXX: Can't test this reliably on Windows unless the cursor is _moved_
|
|
-- by the resize. http://docs.libuv.org/en/v1.x/signal.html
|
|
-- See also: https://github.com/rprichard/winpty/issues/110
|
|
if helpers.pending_win32(pending) then return end
|
|
|
|
describe('and the height is decreased by 2', function()
|
|
before_each(function()
|
|
screen:try_resize(screen._width, screen._height - 2)
|
|
end)
|
|
|
|
local function will_delete_last_two_lines()
|
|
screen:expect([[
|
|
tty ready |
|
|
rows: 4, cols: 30 |
|
|
{1: } |
|
|
|
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
eq(4, curbuf('line_count'))
|
|
end
|
|
|
|
it('will delete the last two empty lines', will_delete_last_two_lines)
|
|
|
|
describe('and then decreased by 1', function()
|
|
before_each(function()
|
|
will_delete_last_two_lines()
|
|
screen:try_resize(screen._width, screen._height - 1)
|
|
end)
|
|
|
|
it('will delete the last line and hide the first', function()
|
|
screen:expect([[
|
|
rows: 4, cols: 30 |
|
|
rows: 3, cols: 30 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
eq(4, curbuf('line_count'))
|
|
feed('<c-\\><c-n>gg')
|
|
screen:expect([[
|
|
^tty ready |
|
|
rows: 4, cols: 30 |
|
|
rows: 3, cols: 30 |
|
|
|
|
|
]])
|
|
feed('a')
|
|
screen:expect([[
|
|
rows: 4, cols: 30 |
|
|
rows: 3, cols: 30 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
end)
|
|
end)
|
|
end)
|
|
|
|
describe('with 4 lines hidden in the scrollback', function()
|
|
before_each(function()
|
|
feed_data({'line1', 'line2', 'line3', 'line4', ''})
|
|
screen:expect([[
|
|
tty ready |
|
|
line1 |
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
screen:try_resize(screen._width, screen._height - 3)
|
|
screen:expect([[
|
|
line4 |
|
|
rows: 3, cols: 30 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
eq(7, curbuf('line_count'))
|
|
end)
|
|
|
|
describe('and the height is increased by 1', function()
|
|
-- XXX: Can't test this reliably on Windows unless the cursor is _moved_
|
|
-- by the resize. http://docs.libuv.org/en/v1.x/signal.html
|
|
-- See also: https://github.com/rprichard/winpty/issues/110
|
|
if helpers.pending_win32(pending) then return end
|
|
local function pop_then_push()
|
|
screen:try_resize(screen._width, screen._height + 1)
|
|
screen:expect([[
|
|
line4 |
|
|
rows: 3, cols: 30 |
|
|
rows: 4, cols: 30 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
end
|
|
|
|
it('will pop 1 line and then push it back', pop_then_push)
|
|
|
|
describe('and then by 3', function()
|
|
before_each(function()
|
|
pop_then_push()
|
|
eq(8, curbuf('line_count'))
|
|
screen:try_resize(screen._width, screen._height + 3)
|
|
end)
|
|
|
|
local function pop3_then_push1()
|
|
screen:expect([[
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
rows: 3, cols: 30 |
|
|
rows: 4, cols: 30 |
|
|
rows: 7, cols: 30 |
|
|
{1: } |
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
eq(9, curbuf('line_count'))
|
|
feed('<c-\\><c-n>gg')
|
|
screen:expect([[
|
|
^tty ready |
|
|
line1 |
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
rows: 3, cols: 30 |
|
|
rows: 4, cols: 30 |
|
|
|
|
|
]])
|
|
end
|
|
|
|
it('will pop 3 lines and then push one back', pop3_then_push1)
|
|
|
|
describe('and then by 4', function()
|
|
before_each(function()
|
|
pop3_then_push1()
|
|
feed('Gi')
|
|
screen:try_resize(screen._width, screen._height + 4)
|
|
end)
|
|
|
|
it('will show all lines and leave a blank one at the end', function()
|
|
screen:expect([[
|
|
tty ready |
|
|
line1 |
|
|
line2 |
|
|
line3 |
|
|
line4 |
|
|
rows: 3, cols: 30 |
|
|
rows: 4, cols: 30 |
|
|
rows: 7, cols: 30 |
|
|
rows: 11, cols: 30 |
|
|
{1: } |
|
|
|
|
|
{3:-- TERMINAL --} |
|
|
]])
|
|
-- since there's an empty line after the cursor, the buffer line
|
|
-- count equals the terminal screen height
|
|
eq(11, curbuf('line_count'))
|
|
end)
|
|
end)
|
|
end)
|
|
end)
|
|
end)
|
|
end)
|
|
|
|
describe('terminal prints more lines than the screen height and exits', function()
|
|
it('will push extra lines to scrollback', function()
|
|
clear()
|
|
local screen = Screen.new(30, 7)
|
|
screen:attach({rgb=false})
|
|
feed_command('call termopen(["'..nvim_dir..'/tty-test", "10"]) | startinsert')
|
|
wait()
|
|
screen:expect([[
|
|
line6 |
|
|
line7 |
|
|
line8 |
|
|
line9 |
|
|
|
|
|
[Process exited 0] |
|
|
-- TERMINAL -- |
|
|
]])
|
|
feed('<cr>')
|
|
-- closes the buffer correctly after pressing a key
|
|
screen:expect([[
|
|
^ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
~ |
|
|
|
|
|
]])
|
|
end)
|
|
end)
|
|
|
|
describe("'scrollback' option", function()
|
|
before_each(function()
|
|
clear()
|
|
end)
|
|
|
|
local function set_fake_shell()
|
|
-- shell-test.c is a fake shell that prints its arguments and exits.
|
|
nvim('set_option', 'shell', nvim_dir..'/shell-test')
|
|
nvim('set_option', 'shellcmdflag', 'EXE')
|
|
end
|
|
|
|
local function expect_lines(expected, epsilon)
|
|
local ep = epsilon and epsilon or 0
|
|
local actual = eval("line('$')")
|
|
if expected > actual + ep and expected < actual - ep then
|
|
error('expected (+/- '..ep..'): '..expected..', actual: '..tostring(actual))
|
|
end
|
|
end
|
|
|
|
it('set to 0 behaves as 1', function()
|
|
local screen
|
|
if iswin() then
|
|
screen = thelpers.screen_setup(nil,
|
|
"['powershell.exe', '-NoLogo', '-NoProfile', '-NoExit', '-Command', 'function global:prompt {return "..'"$"'.."}']", 30)
|
|
else
|
|
screen = thelpers.screen_setup(nil, "['sh']", 30)
|
|
end
|
|
|
|
curbufmeths.set_option('scrollback', 0)
|
|
if iswin() then
|
|
feed_data('for($i=1;$i -le 30;$i++){Write-Host \"line$i\"}\r')
|
|
else
|
|
feed_data('for i in $(seq 1 30); do echo "line$i"; done\n')
|
|
end
|
|
screen:expect('line30 ', nil, nil, nil, true)
|
|
retry(nil, nil, function() expect_lines(7) end)
|
|
|
|
screen:detach()
|
|
end)
|
|
|
|
it('deletes lines (only) if necessary', function()
|
|
local screen
|
|
if iswin() then
|
|
screen = thelpers.screen_setup(nil,
|
|
"['powershell.exe', '-NoLogo', '-NoProfile', '-NoExit', '-Command', 'function global:prompt {return "..'"$"'.."}']", 30)
|
|
else
|
|
screen = thelpers.screen_setup(nil, "['sh']", 30)
|
|
end
|
|
|
|
curbufmeths.set_option('scrollback', 200)
|
|
|
|
-- Wait for prompt.
|
|
screen:expect('$', nil, nil, nil, true)
|
|
|
|
wait()
|
|
if iswin() then
|
|
feed_data('for($i=1;$i -le 30;$i++){Write-Host \"line$i\"}\r')
|
|
else
|
|
feed_data('for i in $(seq 1 30); do echo "line$i"; done\n')
|
|
end
|
|
|
|
screen:expect('line30 ', nil, nil, nil, true)
|
|
|
|
retry(nil, nil, function() expect_lines(33, 2) end)
|
|
curbufmeths.set_option('scrollback', 10)
|
|
wait()
|
|
retry(nil, nil, function() expect_lines(16) end)
|
|
curbufmeths.set_option('scrollback', 10000)
|
|
retry(nil, nil, function() expect_lines(16) end)
|
|
-- Terminal job data is received asynchronously, may happen before the
|
|
-- 'scrollback' option is synchronized with the internal sb_buffer.
|
|
command('sleep 100m')
|
|
if iswin() then
|
|
feed_data('for($i=1;$i -le 40;$i++){Write-Host \"line$i\"}\r')
|
|
else
|
|
feed_data('for i in $(seq 1 40); do echo "line$i"; done\n')
|
|
end
|
|
|
|
screen:expect('line40 ', nil, nil, nil, true)
|
|
|
|
retry(nil, nil, function() expect_lines(58) end)
|
|
-- Verify off-screen state
|
|
eq('line35', eval("getline(line('w0') - 1)"))
|
|
eq('line26', eval("getline(line('w0') - 10)"))
|
|
|
|
screen:detach()
|
|
end)
|
|
|
|
it('defaults to 10000 in terminal buffers', function()
|
|
set_fake_shell()
|
|
command('terminal')
|
|
eq(10000, curbufmeths.get_option('scrollback'))
|
|
end)
|
|
|
|
it('error if set to invalid value', function()
|
|
local status, rv = pcall(command, 'set scrollback=-2')
|
|
eq(false, status) -- assert failure
|
|
eq('E474:', string.match(rv, "E%d*:"))
|
|
|
|
status, rv = pcall(command, 'set scrollback=100001')
|
|
eq(false, status) -- assert failure
|
|
eq('E474:', string.match(rv, "E%d*:"))
|
|
end)
|
|
|
|
it('defaults to -1 on normal buffers', function()
|
|
command('new')
|
|
eq(-1, curbufmeths.get_option('scrollback'))
|
|
end)
|
|
|
|
it(':setlocal in a normal buffer is an error', function()
|
|
command('new')
|
|
|
|
-- :setlocal to -1 is NOT an error.
|
|
feed_command('setlocal scrollback=-1')
|
|
eq(nil, string.match(eval("v:errmsg"), "E%d*:"))
|
|
feed('<CR>')
|
|
|
|
-- :setlocal to anything except -1 is an error.
|
|
feed_command('setlocal scrollback=42')
|
|
feed('<CR>')
|
|
eq('E474:', string.match(eval("v:errmsg"), "E%d*:"))
|
|
eq(-1, curbufmeths.get_option('scrollback'))
|
|
end)
|
|
|
|
it(':set updates local value and global default', function()
|
|
set_fake_shell()
|
|
command('set scrollback=42') -- set global and (attempt) local
|
|
eq(-1, curbufmeths.get_option('scrollback')) -- normal buffer: -1
|
|
command('terminal')
|
|
eq(42, curbufmeths.get_option('scrollback')) -- inherits global default
|
|
command('setlocal scrollback=99')
|
|
eq(99, curbufmeths.get_option('scrollback'))
|
|
command('set scrollback<') -- reset to global default
|
|
eq(42, curbufmeths.get_option('scrollback'))
|
|
command('setglobal scrollback=734') -- new global default
|
|
eq(42, curbufmeths.get_option('scrollback')) -- local value did not change
|
|
command('terminal')
|
|
eq(734, curbufmeths.get_option('scrollback'))
|
|
end)
|
|
|
|
end)
|