local helpers = require('test.functional.helpers')(after_each) local Screen = require('test.functional.ui.screen') local spawn, set_session, clear = helpers.spawn, helpers.set_session, helpers.clear local feed, command = helpers.feed, helpers.command local insert = helpers.insert local eq = helpers.eq local eval = helpers.eval local iswin = helpers.iswin local funcs, meths, exec_lua = helpers.funcs, helpers.meths, helpers.exec_lua describe('screen', function() local screen local nvim_argv = {helpers.nvim_prog, '-u', 'NONE', '-i', 'NONE', '-N', '--cmd', 'set shortmess+=I background=light noswapfile belloff= noshowcmd noruler', '--embed'} before_each(function() local screen_nvim = spawn(nvim_argv) set_session(screen_nvim) screen = Screen.new() screen:attach() screen:set_default_attr_ids( { [0] = {bold=true, foreground=255}, [1] = {bold=true, reverse=true}, } ) end) it('default initial screen', function() screen:expect([[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | ]]) end) end) local function screen_tests(linegrid) local screen before_each(function() clear() screen = Screen.new() screen:attach({rgb=true,ext_linegrid=linegrid}) screen:set_default_attr_ids( { [0] = {bold=true, foreground=255}, [1] = {bold=true, reverse=true}, [2] = {bold=true}, [3] = {reverse=true}, [4] = {background = Screen.colors.LightGrey, underline = true}, [5] = {background = Screen.colors.LightGrey, underline = true, bold = true, foreground = Screen.colors.Fuchsia}, [6] = {bold = true, foreground = Screen.colors.Fuchsia}, [7] = {bold = true, foreground = Screen.colors.SeaGreen}, } ) end) describe(':suspend', function() it('is forwarded to the UI', function() local function check() eq(true, screen.suspended) end command('let g:ev = []') command('autocmd VimResume * :call add(g:ev, "r")') command('autocmd VimSuspend * :call add(g:ev, "s")') eq(false, screen.suspended) command('suspend') eq({ 's', 'r' }, eval('g:ev')) screen:expect(check) screen.suspended = false feed('') eq({ 's', 'r', 's', 'r' }, eval('g:ev')) screen:expect(check) screen.suspended = false command('suspend') eq({ 's', 'r', 's', 'r', 's', 'r' }, eval('g:ev')) end) end) describe('bell/visual bell', function() it('is forwarded to the UI', function() feed('') screen:expect(function() eq(true, screen.bell) eq(false, screen.visual_bell) end) screen.bell = false command('set visualbell') feed('') screen:expect(function() eq(true, screen.visual_bell) eq(false, screen.bell) end) end) end) describe(':set title', function() it('is forwarded to the UI', function() local expected = 'test-title' command('set titlestring='..expected) command('set title') screen:expect(function() eq(expected, screen.title) end) end) it('has correct default title with unnamed file', function() local expected = '[No Name] - NVIM' command('set title') screen:expect(function() eq(expected, screen.title) end) end) it('has correct default title with named file', function() local expected = (iswin() and 'myfile (C:\\mydir) - NVIM' or 'myfile (/mydir) - NVIM') command('set title') command(iswin() and 'file C:\\mydir\\myfile' or 'file /mydir/myfile') screen:expect(function() eq(expected, screen.title) end) end) describe('is not changed by', function() local file1 = iswin() and 'C:\\mydir\\myfile1' or '/mydir/myfile1' local file2 = iswin() and 'C:\\mydir\\myfile2' or '/mydir/myfile2' local expected = (iswin() and 'myfile1 (C:\\mydir) - NVIM' or 'myfile1 (/mydir) - NVIM') local buf2 before_each(function() command('edit '..file1) buf2 = funcs.bufadd(file2) command('set title') end) it('calling setbufvar() to set an option in a hidden buffer from i_CTRL-R', function() command([[inoremap =setbufvar(]]..buf2..[[, '&autoindent', 1) ? '' : '']]) feed('i') command('redraw!') screen:expect(function() eq(expected, screen.title) end) end) it('an RPC call to nvim_buf_set_option in a hidden buffer', function() meths.buf_set_option(buf2, 'autoindent', true) command('redraw!') screen:expect(function() eq(expected, screen.title) end) end) it('a Lua callback calling nvim_buf_set_option in a hidden buffer', function() exec_lua(string.format([[ vim.schedule(function() vim.api.nvim_buf_set_option(%d, 'autoindent', true) end) ]], buf2)) command('redraw!') screen:expect(function() eq(expected, screen.title) end) end) it('a Lua callback calling nvim_buf_call in a hidden buffer', function() exec_lua(string.format([[ vim.schedule(function() vim.api.nvim_buf_call(%d, function() end) end) ]], buf2)) command('redraw!') screen:expect(function() eq(expected, screen.title) end) end) end) end) describe(':set icon', function() it('is forwarded to the UI', function() local expected = 'test-icon' command('set iconstring='..expected) command('set icon') screen:expect(function() eq(expected, screen.icon) end) end) end) describe('statusline', function() it('is redrawn after ', function() command('set laststatus=2') screen:expect([[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | ]]) feed('') screen:expect{grid=[[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | ]], reset=true} command('split') screen:expect([[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {3:[No Name] }| | ]]) feed('') screen:expect{grid=[[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {3:[No Name] }| | ]], reset=true} end) end) describe('window', function() describe('split', function() it('horizontal', function() command('sp') screen:expect([[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {3:[No Name] }| | ]]) end) it('horizontal and resize', function() command('sp') command('resize 8') screen:expect([[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1:[No Name] }| | {0:~ }| {0:~ }| {3:[No Name] }| | ]]) end) it('horizontal and vertical', function() command('sp') command('vsp') command('vsp') screen:expect([[ ^ │ │ | {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {1:[No Name] }{3:[No Name] [No Name] }| | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {3:[No Name] }| | ]]) insert('hello') screen:expect([[ hell^o │hello │hello | {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {1:[No Name] [+] }{3:[No Name] [+] [No Name] [+] }| hello | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {3:[No Name] [+] }| | ]]) end) end) end) describe('tabs', function() it('tabnew creates a new buffer', function() command('sp') command('vsp') command('vsp') insert('hello') screen:expect([[ hell^o │hello │hello | {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {1:[No Name] [+] }{3:[No Name] [+] [No Name] [+] }| hello | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {3:[No Name] [+] }| | ]]) command('tabnew') insert('hello2') feed('h') screen:expect([[ {4: }{5:4}{4:+ [No Name] }{2: + [No Name] }{3: }{4:X}| hell^o2 | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) command('tabprevious') screen:expect([[ {2: }{6:4}{2:+ [No Name] }{4: + [No Name] }{3: }{4:X}| hell^o │hello │hello | {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {0:~ }│{0:~ }│{0:~ }| {1:[No Name] [+] }{3:[No Name] [+] [No Name] [+] }| hello | {0:~ }| {0:~ }| {0:~ }| {3:[No Name] [+] }| | ]]) end) it('tabline is redrawn after messages', function() command('tabnew') screen:expect([[ {4: [No Name] }{2: [No Name] }{3: }{4:X}| ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) feed(':echo "'..string.rep('x\\n', 11)..'"') screen:expect([[ {1: }| x | x | x | x | x | x | x | x | x | x | x | | {7:Press ENTER or type command to continue}^ | ]]) feed('') screen:expect([[ {4: [No Name] }{2: [No Name] }{3: }{4:X}| ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) feed(':echo "'..string.rep('x\\n', 12)..'"') screen:expect([[ x | x | x | x | x | x | x | x | x | x | x | x | | {7:Press ENTER or type command to continue}^ | ]]) feed('') screen:expect([[ {4: [No Name] }{2: [No Name] }{3: }{4:X}| ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) end) it('redraws properly with :tab split right after scroll', function() feed('15Ofoo15Obargg') command('vsplit') screen:expect([[ ^foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | foo │foo | {1:[No Name] [+] }{3:[No Name] [+] }| | ]]) feed('') screen:expect([[ ^foo │foo | foo │foo | foo │foo | foo │foo | bar │foo | bar │foo | bar │foo | bar │foo | bar │foo | bar │foo | bar │foo | bar │foo | {1:[No Name] [+] }{3:[No Name] [+] }| | ]]) command('tab split') screen:expect([[ {4: }{5:2}{4:+ [No Name] }{2: + [No Name] }{3: }{4:X}| ^foo | foo | foo | foo | bar | bar | bar | bar | bar | bar | bar | bar | | ]]) end) it('redraws unvisited tab #9152', function() insert('hello') -- create a tab without visiting it command('tabnew|tabnext') screen:expect([[ {2: + [No Name] }{4: [No Name] }{3: }{4:X}| hell^o | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) feed('gT') screen:expect([[ {4: + [No Name] }{2: [No Name] }{3: }{4:X}| ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) end) end) describe('insert mode', function() it('move to next line with ', function() feed('iline 1line 2') screen:expect([[ line 1 | line 2 | ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {2:-- INSERT --} | ]]) end) end) describe('normal mode', function() -- https://code.google.com/p/vim/issues/detail?id=339 it("setting 'ruler' doesn't reset the preferred column", function() command('set virtualedit=') feed('i0123456789kllj') command('set ruler') feed('k') screen:expect([[ 0123^456 | 789 | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| 1,5 All | ]]) end) end) describe('command mode', function() it('typing commands', function() feed(':ls') screen:expect([[ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| :ls^ | ]]) end) it('execute command with multi-line output without msgsep', function() command("set display-=msgsep") feed(':ls') screen:expect([[ {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| :ls | 1 %a "[No Name]" line 1 | {7:Press ENTER or type command to continue}^ | ]]) feed('') -- skip the "Press ENTER..." state or tests will hang end) it('execute command with multi-line output and with msgsep', function() command("set display+=msgsep") feed(':ls') screen:expect([[ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1: }| :ls | 1 %a "[No Name]" line 1 | {7:Press ENTER or type command to continue}^ | ]]) feed('') -- skip the "Press ENTER..." state or tests will hang end) end) describe('scrolling and clearing', function() before_each(function() insert([[ Inserting text with many lines to test scrolling and clearing in split windows ]]) command('sp') command('vsp') command('vsp') screen:expect([[ and │and │and | clearing │clearing │clearing | in │in │in | split │split │split | windows │windows │windows | ^ │ │ | {1:[No Name] [+] }{3:[No Name] [+] [No Name] [+] }| clearing | in | split | windows | | {3:[No Name] [+] }| | ]]) end) it('only affects the current scroll region', function() feed('6k') screen:expect([[ ^scrolling │and │and | and │clearing │clearing | clearing │in │in | in │split │split | split │windows │windows | windows │ │ | {1:[No Name] [+] }{3:[No Name] [+] [No Name] [+] }| clearing | in | split | windows | | {3:[No Name] [+] }| | ]]) feed('l') screen:expect([[ scrolling │and │and | and │clearing │clearing | clearing │in │in | in │split │split | split │windows │windows | windows │^ │ | {3:[No Name] [+] }{1:[No Name] [+] }{3::ls') screen:expect([[ resize | :ls^ | ]]) end) end) describe('press enter', function() it('does not crash on at “Press ENTER”', function() command('nnoremap :echo "TEST"') feed(':ls') screen:expect([[ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {1: }| :ls | 1 %a "[No Name]" line 1 | {7:Press ENTER or type command to continue}^ | ]]) feed('') screen:expect([[ ^ | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| TEST | ]]) end) end) -- Regression test for #8357 it('does not have artifacts after temporary chars in insert mode', function() command('set timeoutlen=10000') command('inoremap jk ') feed('ifooj') screen:expect([[ foo^j | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {2:-- INSERT --} | ]]) feed('k') screen:expect([[ fo^o | {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| | ]]) end) end describe("Screen (char-based)", function() screen_tests(false) end) describe("Screen (line-based)", function() screen_tests(true) end) describe('Screen default colors', function() local screen local function startup(light, termcolors) local extra = (light and ' background=light') or '' local nvim_argv = {helpers.nvim_prog, '-u', 'NONE', '-i', 'NONE', '-N', '--cmd', 'set shortmess+=I noswapfile belloff= noshowcmd noruler'..extra, '--embed'} local screen_nvim = spawn(nvim_argv) set_session(screen_nvim) screen = Screen.new() screen:attach(termcolors and {rgb=true,ext_termcolors=true} or {rgb=true}) end it('are dark per default', function() startup(false, false) screen:expect{condition=function() eq({rgb_bg=0, rgb_fg=Screen.colors.White, rgb_sp=Screen.colors.Red, cterm_bg=0, cterm_fg=0}, screen.default_colors) end} end) it('can be set to light', function() startup(true, false) screen:expect{condition=function() eq({rgb_fg=Screen.colors.White, rgb_bg=0, rgb_sp=Screen.colors.Red, cterm_bg=0, cterm_fg=0}, screen.default_colors) end} end) it('can be handled by external terminal', function() startup(false, true) screen:expect{condition=function() eq({rgb_bg=-1, rgb_fg=-1, rgb_sp=-1, cterm_bg=0, cterm_fg=0}, screen.default_colors) end} startup(true, true) screen:expect{condition=function() eq({rgb_bg=-1, rgb_fg=-1, rgb_sp=-1, cterm_bg=0, cterm_fg=0}, screen.default_colors) end} end) end) describe('screen with msgsep deactivated on startup', function() local screen before_each(function() clear('--cmd', 'set display-=msgsep') screen = Screen.new() screen:attach() screen:set_default_attr_ids { [0] = {bold=true, foreground=255}; [7] = {bold = true, foreground = Screen.colors.SeaGreen}; } end) it('execute command with multi-line output', function() feed ':ls' screen:expect([[ {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| {0:~ }| :ls | 1 %a "[No Name]" line 1 | {7:Press ENTER or type command to continue}^ | ]]) feed '' -- skip the "Press ENTER..." state or tests will hang end) end)