" Tests for the terminal window. " This is split in two, because it can take a lot of time. " See test_terminal.mnv and test_terminal2.mnv for further tests. CheckFeature terminal source util/screendump.mnv source util/mouse.mnv import './util/mnv9.mnv' as v9 let $PROMPT_COMMAND='' func Test_terminal_altscreen() " somehow doesn't work on MS-Windows CheckUnix let cmd = "cat Xtext\" let buf = term_start(&shell, {}) call TermWait(buf) call writefile(["\[?1047h"], 'Xtext', 'D') call term_sendkeys(buf, cmd) call WaitForAssert({-> assert_equal(1, term_getaltscreen(buf))}) call writefile(["\[?1047l"], 'Xtext') call term_sendkeys(buf, cmd) call WaitForAssert({-> assert_equal(0, term_getaltscreen(buf))}) call term_sendkeys(buf, "exit\r") exe buf . "bwipe!" endfunc func Test_terminal_shell_option() if has('unix') " exec is a shell builtin command, should fail without a shell. term exec ls runtest.mnv call WaitForAssert({-> assert_match('job failed', term_getline(bufnr(), 1))}) bwipe! term ++shell exec ls runtest.mnv call WaitForAssert({-> assert_match('runtest.mnv', term_getline(bufnr(), 1))}) bwipe! elseif has('win32') " dir is a shell builtin command, should fail without a shell. " However, if dir.exe (which might be provided by Cygwin/MSYS2) exists in " the %PATH%, "term dir" succeeds unintentionally. Use dir.com instead. try term dir.com /b runtest.mnv throw 'dir.com without a shell must fail, so you will never get here' catch /CreateProcess failed/ " ignore: " winpty simply fails with "CreateProcess failed". " compty fails with "CreateProcess failed: {localized failure reason}". endtry bwipe! " This should execute the dir builtin command even with ".com". term ++shell dir.com /b runtest.mnv call WaitForAssert({-> assert_match('runtest.mnv', term_getline(bufnr(), 1))}) bwipe! else throw 'Skipped: does not work on this platform' endif endfunc func Test_terminal_invalid_arg() call assert_fails('terminal ++xyz', 'E181:') endfunc " Check a terminal with different colors func Terminal_color(group_name, highlight_cmds, highlight_opt, open_cmds) CheckScreendump CheckRunMNVInTerminal CheckUnix let lines = [ \ 'call setline(1, range(20))', \ 'func NotifyParent()', \ ' call echoraw("' .. TermNotifyParentCmd(v:true) .. '")', \ 'endfunc', \ 'func OpenTerm()', \ ' set noruler', \ " call term_start('cat', #{vertical: 1, " \ .. 'exit_cb: {->NotifyParent()}, ' \ .. a:highlight_opt .. "})", \ ' call NotifyParent()', \ ] + a:open_cmds + [ \ 'endfunc', \ ] + a:highlight_cmds + [ \ 'call NotifyParent()', \ ] call writefile(lines, 'XtermStart', 'D') let buf = RunMNVInTerminal('-S XtermStart', #{rows: 15}) call WaitForChildNotification() call term_sendkeys(buf, ":call OpenTerm()\") call WaitForChildNotification() call term_sendkeys(buf, "hello\") call VerifyScreenDump(buf, 'Test_terminal_color_' .. a:group_name, {}) call term_sendkeys(buf, "\") call WaitForChildNotification() call StopMNVInTerminal(buf) endfunc func Test_terminal_color_Terminal() call Terminal_color("Terminal", [ \ "highlight Terminal ctermfg=blue ctermbg=yellow", \ ], "", []) endfunc func Test_terminal_color_group() call Terminal_color("MyTermCol", [ \ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue", \ ], "term_highlight: 'MyTermCol',", []) endfunc func Test_terminal_color_wincolor() call Terminal_color("MyWinCol", [ \ "highlight MyWinCol ctermfg=red ctermbg=darkyellow", \ ], "", [ \ 'set wincolor=MyWinCol', \ ]) endfunc func Test_terminal_color_group_over_Terminal() call Terminal_color("MyTermCol_over_Terminal", [ \ "highlight Terminal ctermfg=blue ctermbg=yellow", \ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue", \ ], "term_highlight: 'MyTermCol',", []) endfunc func Test_terminal_color_wincolor_over_group() call Terminal_color("MyWinCol_over_group", [ \ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue", \ "highlight MyWinCol ctermfg=red ctermbg=darkyellow", \ ], "term_highlight: 'MyTermCol',", [ \ 'set wincolor=MyWinCol', \ ]) endfunc func Test_terminal_color_wincolor_split() CheckScreendump CheckRunMNVInTerminal CheckUnix let lines = [ \ 'call setline(1, range(20))', \ 'func OpenTerm()', \ ' set noruler', \ " call term_start('cat', #{vertical: 1, term_highlight: 'MyTermCol'})", \ 'endfunc', \ 'highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue', \ 'highlight MyWinCol ctermfg=red ctermbg=darkyellow', \ 'highlight MyWinCol2 ctermfg=black ctermbg=blue', \ ] call writefile(lines, 'XtermStart', 'D') let buf = RunMNVInTerminal('-S XtermStart', #{rows: 15}) call TermWait(buf, 100) call term_sendkeys(buf, ":call OpenTerm()\") call TermWait(buf, 50) call term_sendkeys(buf, "hello\") call TermWait(buf, 50) call term_sendkeys(buf, "\:split\") call term_sendkeys(buf, "\:set wincolor=MyWinCol\") call VerifyScreenDump(buf, 'Test_terminal_wincolor_split_MyWinCol', {}) call term_sendkeys(buf, "\b:2sb\") call term_sendkeys(buf, "\:set wincolor=MyWinCol2\") call VerifyScreenDump(buf, 'Test_terminal_wincolor_split_MyWinCol2', {}) call term_sendkeys(buf, "\") call TermWait(buf, 50) call StopMNVInTerminal(buf) endfunc func Test_terminal_color_transp_Terminal() call Terminal_color("transp_Terminal", [ \ "highlight Terminal ctermfg=blue", \ ], "", []) endfunc func Test_terminal_color_transp_group() call Terminal_color("transp_MyTermCol", [ \ "highlight MyTermCol ctermfg=darkgreen", \ ], "term_highlight: 'MyTermCol',", []) endfunc func Test_terminal_color_transp_wincolor() call Terminal_color("transp_MyWinCol", [ \ "highlight MyWinCol ctermfg=red", \ ], "", [ \ 'set wincolor=MyWinCol', \ ]) endfunc func Test_terminal_color_gui_Terminal() CheckFeature termguicolors call Terminal_color("gui_Terminal", [ \ "set termguicolors", \ "highlight Terminal guifg=#3344ff guibg=#b0a700", \ ], "", []) endfunc func Test_terminal_color_gui_group() CheckFeature termguicolors call Terminal_color("gui_MyTermCol", [ \ "set termguicolors", \ "highlight MyTermCol guifg=#007800 guibg=#6789ff", \ ], "term_highlight: 'MyTermCol',", []) endfunc func Test_terminal_color_gui_wincolor() CheckFeature termguicolors call Terminal_color("gui_MyWinCol", [ \ "set termguicolors", \ "highlight MyWinCol guifg=#fe1122 guibg=#818100", \ ], "", [ \ 'set wincolor=MyWinCol', \ ]) endfunc func Test_terminal_color_gui_transp_Terminal() CheckFeature termguicolors call Terminal_color("gui_transp_Terminal", [ \ "set termguicolors", \ "highlight Terminal guifg=#3344ff", \ ], "", []) endfunc func Test_terminal_color_gui_transp_group() CheckFeature termguicolors call Terminal_color("gui_transp_MyTermCol", [ \ "set termguicolors", \ "highlight MyTermCol guifg=#007800", \ ], "term_highlight: 'MyTermCol',", []) endfunc func Test_terminal_color_gui_transp_wincolor() CheckFeature termguicolors call Terminal_color("gui_transp_MyWinCol", [ \ "set termguicolors", \ "highlight MyWinCol guifg=#fe1122", \ ], "", [ \ 'set wincolor=MyWinCol', \ ]) endfunc func Test_terminal_in_popup() CheckScreendump CheckRunMNVInTerminal let text =<< trim END some text to edit in a popup window END call writefile(text, 'Xtext', 'D') let cmd = GetMNVCommandCleanTerm() let lines = [ \ 'call setline(1, range(20))', \ 'hi PopTerm ctermbg=grey', \ 'func OpenTerm(setColor)', \ " set noruler", \ " let s:buf = term_start('" .. cmd .. " Xtext', #{hidden: 1, term_finish: 'close'})", \ ' let g:winid = popup_create(s:buf, #{minwidth: 45, minheight: 7, border: [], drag: 1, resize: 1})', \ ' if a:setColor', \ ' call win_execute(g:winid, "set wincolor=PopTerm")', \ ' endif', \ 'endfunc', \ 'func HidePopup()', \ ' call popup_hide(g:winid)', \ 'endfunc', \ 'func ClosePopup()', \ ' call popup_close(g:winid)', \ 'endfunc', \ 'func ReopenPopup()', \ ' call popup_create(s:buf, #{minwidth: 40, minheight: 6, border: []})', \ 'endfunc', \ ] call writefile(lines, 'XtermPopup', 'D') let buf = RunMNVInTerminal('-S XtermPopup', #{rows: 15}) call TermWait(buf,0) call term_sendkeys(buf, ":call OpenTerm(0)\") call TermWait(buf,0) call term_sendkeys(buf, ":\") call TermWait(buf,0) call term_sendkeys(buf, "\:echo getwinvar(g:winid, \"&buftype\") win_gettype(g:winid)\") call VerifyScreenDump(buf, 'Test_terminal_popup_1', {}) call term_sendkeys(buf, ":q\") call VerifyScreenDump(buf, 'Test_terminal_popup_2', {}) call term_sendkeys(buf, ":call OpenTerm(1)\") call TermWait(buf,0) call term_sendkeys(buf, ":set hlsearch\") call TermWait(buf,0) call term_sendkeys(buf, "/edit\") call VerifyScreenDump(buf, 'Test_terminal_popup_3', {}) call term_sendkeys(buf, "\:call HidePopup()\") call VerifyScreenDump(buf, 'Test_terminal_popup_4', {}) call term_sendkeys(buf, "\") call TermWait(buf,0) call term_sendkeys(buf, "\:call ClosePopup()\") call VerifyScreenDump(buf, 'Test_terminal_popup_5', {}) call term_sendkeys(buf, "\:call ReopenPopup()\") call VerifyScreenDump(buf, 'Test_terminal_popup_6', {}) " Go to terminal-Normal mode and visually select text. call term_sendkeys(buf, "\Ngg/in\vww") call VerifyScreenDump(buf, 'Test_terminal_popup_7', {}) " Back to job mode, redraws call term_sendkeys(buf, "A") call VerifyScreenDump(buf, 'Test_terminal_popup_8', {}) call TermWait(buf,0) call term_sendkeys(buf, ":q\") call WaitForAssert({-> assert_equal(0, match(term_getline(buf, 6), '^5\s*$'))}, 250) " wait for terminal to vanish call StopMNVInTerminal(buf) endfunc " Check a terminal in popup window uses the default minimum size. func Test_terminal_in_popup_min_size() CheckScreendump CheckRunMNVInTerminal let text =<< trim END another text to show in a popup window END call writefile(text, 'Xtext', 'D') let lines = [ \ 'call setline(1, range(20))', \ 'func OpenTerm()', \ " let s:buf = term_start('cat Xtext', #{hidden: 1})", \ ' let g:winid = popup_create(s:buf, #{ border: []})', \ 'endfunc', \ ] call writefile(lines, 'XtermPopup', 'D') let buf = RunMNVInTerminal('-S XtermPopup', #{rows: 15}) call TermWait(buf, 100) call term_sendkeys(buf, ":set noruler\") call term_sendkeys(buf, ":call OpenTerm()\") call TermWait(buf, 50) call term_sendkeys(buf, ":\") call VerifyScreenDump(buf, 'Test_terminal_popup_m1', {}) call TermWait(buf, 50) call term_sendkeys(buf, ":q\") call TermWait(buf, 50) " wait for terminal to vanish call StopMNVInTerminal(buf) endfunc " Check a terminal in popup window with different colors func Terminal_in_popup_color(group_name, highlight_cmds, highlight_opt, popup_cmds, popup_opt) CheckScreendump CheckRunMNVInTerminal CheckUnix let lines = [ \ 'call setline(1, range(20))', \ 'func NotifyParent(...)', \ ' call echoraw("' .. TermNotifyParentCmd(v:true) .. '")', \ 'endfunc', \ 'func OpenTerm()', \ " let s:buf = term_start('cat', #{hidden: 1, term_finish: 'close', " \ .. a:highlight_opt .. "})", \ ' let g:winid = popup_create(s:buf, #{border: [], ' \ .. 'callback: {->NotifyParent()}, ' \ .. a:popup_opt .. '})', \ ] + a:popup_cmds + [ \ ' call NotifyParent()', \ 'endfunc', \ ] + a:highlight_cmds + [ \ 'call NotifyParent()', \ ] call writefile(lines, 'XtermPopup', 'D') let buf = RunMNVInTerminal('-S XtermPopup', #{rows: 15}) call WaitForChildNotification() call term_sendkeys(buf, ":set noruler\") call term_sendkeys(buf, ":call OpenTerm()\") call WaitForChildNotification() call term_sendkeys(buf, "hello\") call VerifyScreenDump(buf, 'Test_terminal_popup_' .. a:group_name, {}) call term_sendkeys(buf, "\") call WaitForChildNotification() call StopMNVInTerminal(buf) endfunc func Test_terminal_in_popup_color_Terminal() call Terminal_in_popup_color("Terminal", [ \ "highlight Terminal ctermfg=blue ctermbg=yellow", \ ], "", [], "") endfunc func Test_terminal_in_popup_color_group() call Terminal_in_popup_color("MyTermCol", [ \ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue", \ ], "term_highlight: 'MyTermCol',", [], "") endfunc func Test_terminal_in_popup_color_wincolor() call Terminal_in_popup_color("MyWinCol", [ \ "highlight MyWinCol ctermfg=red ctermbg=darkyellow", \ ], "", [ \ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")', \ ], "") endfunc func Test_terminal_in_popup_color_popup_highlight() call Terminal_in_popup_color("MyPopupHlCol", [ \ "highlight MyPopupHlCol ctermfg=cyan ctermbg=green", \ ], "", [], "highlight: 'MyPopupHlCol'") endfunc func Test_terminal_in_popup_color_group_over_Terminal() call Terminal_in_popup_color("MyTermCol_over_Terminal", [ \ "highlight Terminal ctermfg=blue ctermbg=yellow", \ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue", \ ], "term_highlight: 'MyTermCol',", [], "") endfunc func Test_terminal_in_popup_color_wincolor_over_group() call Terminal_in_popup_color("MyWinCol_over_group", [ \ "highlight MyTermCol ctermfg=darkgreen ctermbg=lightblue", \ "highlight MyWinCol ctermfg=red ctermbg=darkyellow", \ ], "term_highlight: 'MyTermCol',", [ \ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")', \ ], "") endfunc func Test_terminal_in_popup_color_transp_Terminal() call Terminal_in_popup_color("transp_Terminal", [ \ "highlight Terminal ctermfg=blue", \ ], "", [], "") endfunc func Test_terminal_in_popup_color_transp_group() call Terminal_in_popup_color("transp_MyTermCol", [ \ "highlight MyTermCol ctermfg=darkgreen", \ ], "term_highlight: 'MyTermCol',", [], "") endfunc func Test_terminal_in_popup_color_transp_wincolor() call Terminal_in_popup_color("transp_MyWinCol", [ \ "highlight MyWinCol ctermfg=red", \ ], "", [ \ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")', \ ], "") endfunc func Test_terminal_in_popup_color_transp_popup_highlight() call Terminal_in_popup_color("transp_MyPopupHlCol", [ \ "highlight MyPopupHlCol ctermfg=cyan", \ ], "", [], "highlight: 'MyPopupHlCol'") endfunc func Test_terminal_in_popup_color_gui_Terminal() CheckFeature termguicolors call Terminal_in_popup_color("gui_Terminal", [ \ "set termguicolors", \ "highlight Terminal guifg=#3344ff guibg=#b0a700", \ ], "", [], "") endfunc func Test_terminal_in_popup_color_gui_group() CheckFeature termguicolors call Terminal_in_popup_color("gui_MyTermCol", [ \ "set termguicolors", \ "highlight MyTermCol guifg=#007800 guibg=#6789ff", \ ], "term_highlight: 'MyTermCol',", [], "") endfunc func Test_terminal_in_popup_color_gui_wincolor() CheckFeature termguicolors call Terminal_in_popup_color("gui_MyWinCol", [ \ "set termguicolors", \ "highlight MyWinCol guifg=#fe1122 guibg=#818100", \ ], "", [ \ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")', \ ], "") endfunc func Test_terminal_in_popup_color_gui_popup_highlight() CheckFeature termguicolors call Terminal_in_popup_color("gui_MyPopupHlCol", [ \ "set termguicolors", \ "highlight MyPopupHlCol guifg=#00e8f0 guibg=#126521", \ ], "", [], "highlight: 'MyPopupHlCol'") endfunc func Test_terminal_in_popup_color_gui_transp_Terminal() CheckFeature termguicolors call Terminal_in_popup_color("gui_transp_Terminal", [ \ "set termguicolors", \ "highlight Terminal guifg=#3344ff", \ ], "", [], "") endfunc func Test_terminal_in_popup_color_gui_transp_group() CheckFeature termguicolors call Terminal_in_popup_color("gui_transp_MyTermCol", [ \ "set termguicolors", \ "highlight MyTermCol guifg=#007800", \ ], "term_highlight: 'MyTermCol',", [], "") endfunc func Test_terminal_in_popup_color_gui_transp_wincolor() CheckFeature termguicolors call Terminal_in_popup_color("gui_transp_MyWinCol", [ \ "set termguicolors", \ "highlight MyWinCol guifg=#fe1122", \ ], "", [ \ 'call setwinvar(g:winid, "&wincolor", "MyWinCol")', \ ], "") endfunc func Test_terminal_in_popup_color_gui_transp_popup_highlight() CheckFeature termguicolors call Terminal_in_popup_color("gui_transp_MyPopupHlCol", [ \ "set termguicolors", \ "highlight MyPopupHlCol guifg=#00e8f0", \ ], "", [], "highlight: 'MyPopupHlCol'") endfunc func Test_double_popup_terminal() let buf1 = term_start(&shell, #{hidden: 1}) let win1 = popup_create(buf1, {}) let buf2 = term_start(&shell, #{hidden: 1}) call assert_fails('call popup_create(buf2, {})', 'E861:') call popup_close(win1) exe buf1 .. 'bwipe!' exe buf2 .. 'bwipe!' endfunc func Test_escape_popup_terminal() set hidden " Cannot escape a terminal popup window using win_gotoid let prev_win = win_getid() eval term_start('sh', #{hidden: 1, term_finish: 'close'})->popup_create({}) call assert_fails("call win_gotoid(" .. prev_win .. ")", 'E863:') call popup_clear(1) set hidden& endfunc func Test_issue_5607() let wincount = winnr('$') exe 'terminal' &shell &shellcmdflag 'exit' let job = term_getjob(bufnr()) call WaitForAssert({-> assert_equal("dead", job_status(job))}) let old_wincolor = &wincolor try set wincolor= finally let &wincolor = old_wincolor bw! endtry endfunc func Test_hidden_terminal() let buf = term_start(&shell, #{hidden: 1}) call assert_equal('', bufname('^$')) call StopShellInTerminal(buf) endfunc func Test_term_nasty_callback() CheckExecutable sh set hidden let g:buf0 = term_start('sh', #{hidden: 1, term_finish: 'close'}) call popup_create(g:buf0, {}) call assert_fails("call term_start(['sh', '-c'], #{curwin: 1})", 'E863:') call popup_clear(1) set hidden& endfunc func Test_term_and_startinsert() CheckRunMNVInTerminal CheckUnix let lines =<< trim EOL put='some text' term startinsert EOL call writefile(lines, 'XTest_startinsert', 'D') let buf = RunMNVInTerminal('-S XTest_startinsert', {}) call term_sendkeys(buf, "exit\r") call WaitForAssert({-> assert_equal("some text", term_getline(buf, 1))}) call term_sendkeys(buf, "0l") call term_sendkeys(buf, "A<\") call WaitForAssert({-> assert_equal("some text<", term_getline(buf, 1))}) call StopMNVInTerminal(buf) endfunc " Test for passing invalid arguments to terminal functions func Test_term_func_invalid_arg() call assert_fails('let b = term_getaltscreen([])', 'E745:') call assert_fails('let a = term_getattr(1, [])', 'E730:') call assert_fails('let c = term_getcursor([])', 'E745:') call assert_fails('let l = term_getline([], 1)', 'E745:') call assert_fails('let l = term_getscrolled([])', 'E745:') call assert_fails('let s = term_getsize([])', 'E745:') call assert_fails('let s = term_getstatus([])', 'E745:') call assert_fails('let s = term_scrape([], 1)', 'E745:') call assert_fails('call term_sendkeys([], "a")', 'E745:') call assert_fails('call term_setapi([], "")', 'E745:') call assert_fails('call term_setrestore([], "")', 'E745:') call assert_fails('call term_setkill([], "")', 'E745:') if has('gui') || has('termguicolors') call assert_fails('let p = term_getansicolors([])', 'E745:') call assert_fails('call term_setansicolors([], [])', 'E745:') endif let buf = term_start('echo') call assert_fails('call term_setapi(' .. buf .. ', {})', 'E731:') call assert_fails('call term_setkill(' .. buf .. ', {})', 'E731:') call assert_fails('call term_setrestore(' .. buf .. ', {})', 'E731:') exe buf . "bwipe!" endfunc " Test for sending various special keycodes to a terminal func Test_term_keycode_translation() CheckRunMNVInTerminal let buf = RunMNVInTerminal('', {}) call term_sendkeys(buf, ":set nocompatible\") call term_sendkeys(buf, ":set timeoutlen=20\") let keys = ["\", "\", "\", "\", "\", "\", "\", \ "\", "\", "\", "\", "\", "\", \ "\", "\", "\", "\", "\", \ "\", "\", "\", "\", "\", "\", \ "\", "\", "\", "\", "\", \ "\"] let output = ['', '', '', '', '', '', '', \ '', '', '', '', '', '', '', \ '', '', '', '', '', '', \ '', '', '', '', '', \ '', '', '', '', ''] call term_sendkeys(buf, "i") for i in range(len(keys)) call term_sendkeys(buf, "\\" .. keys[i]) call WaitForAssert({-> assert_equal(output[i], term_getline(buf, 1))}, 200) endfor let keypad_keys = ["\", "\", "\", "\", "\", "\", \ "\", "\", "\", "\", "\", "\", \ "\", "\", "\"] let keypad_output = ['0', '1', '2', '3', '4', '5', \ '6', '7', '8', '9', '.', '+', \ '-', '*', '/'] for i in range(len(keypad_keys)) " TODO: Mysteriously keypad 3 and 9 do not work on some systems. if keypad_output[i] == '3' || keypad_output[i] == '9' continue endif call term_sendkeys(buf, "\" .. keypad_keys[i]) call WaitForAssert({-> assert_equal(keypad_output[i], term_getline(buf, 1))}, 100) endfor call feedkeys("\\\one\.two", 'xt') call WaitForAssert({-> assert_equal('two', term_getline(buf, 1))}) call StopMNVInTerminal(buf) endfunc " Test for using the mouse in a terminal func Test_term_mouse() CheckNotGui CheckRunMNVInTerminal let save_mouse = &mouse let save_term = &term let save_ttymouse = &ttymouse let save_clipboard = &clipboard set mouse=a term=xterm ttymouse=sgr mousetime=200 clipboard= let lines =<< trim END one two three four five red green yellow red blue mnv emacs sublime nano END call writefile(lines, 'Xtest_mouse', 'D') " Create a terminal window running MNV for the test with mouse enabled let prev_win = win_getid() let buf = RunMNVInTerminal('Xtest_mouse -n', {}) call term_sendkeys(buf, ":set nocompatible\") call term_sendkeys(buf, ":set mouse=a term=xterm ttymouse=sgr\") call term_sendkeys(buf, ":set clipboard=\") call term_sendkeys(buf, ":set mousemodel=extend\") call TermWait(buf) redraw! " Funcref used in WaitFor() to check that the "Xbuf" file is readable and " has some contents. This avoids a "List index out of range" error when the " file hasn't been written yet. let XbufNotEmpty = {-> filereadable('Xbuf') && len(readfile('Xbuf')) > 0} " Use the mouse to enter the terminal window call win_gotoid(prev_win) call feedkeys(MouseLeftClickCode(1, 1), 'x') call feedkeys(MouseLeftReleaseCode(1, 1), 'x') call assert_equal(1, getwininfo(win_getid())[0].terminal) " Test for click/release call test_setmouse(2, 5) call feedkeys("\\", 'xt') call test_setmouse(3, 8) call term_sendkeys(buf, "\\") call TermWait(buf, 50) call delete('Xbuf') call term_sendkeys(buf, ":call writefile([json_encode(getpos('.'))], 'Xbuf')\") call TermWait(buf, 50) call WaitFor(XbufNotEmpty) let pos = json_decode(readfile('Xbuf')[0]) call assert_equal([3, 8], pos[1:2]) call delete('Xbuf') " Test for selecting text using mouse call test_setmouse(2, 11) call term_sendkeys(buf, "\") call test_setmouse(2, 16) call term_sendkeys(buf, "\y") call TermWait(buf, 50) call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\") call WaitFor(XbufNotEmpty) call WaitForAssert({-> assert_equal('yellow', readfile('Xbuf')[0])}) call delete('Xbuf') " Test for selecting text using double click call test_setmouse(1, 11) call term_sendkeys(buf, "\\\") call test_setmouse(1, 17) call term_sendkeys(buf, "\y") call TermWait(buf, 50) call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\") call WaitFor(XbufNotEmpty) call assert_equal('three four', readfile('Xbuf')[0]) call delete('Xbuf') " Test for selecting a line using triple click call test_setmouse(3, 2) call term_sendkeys(buf, "\\\\\\y") call TermWait(buf, 50) call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\") call WaitFor(XbufNotEmpty) call assert_equal("mnv emacs sublime nano\n", readfile('Xbuf')[0]) call delete('Xbuf') " Test for selecting a block using quadruple click call test_setmouse(1, 11) call term_sendkeys(buf, "\\\\\\\") call test_setmouse(3, 13) call term_sendkeys(buf, "\y") call TermWait(buf, 50) call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\") call WaitFor(XbufNotEmpty) call assert_equal("ree\nyel\nsub", readfile('Xbuf')[0]) call delete('Xbuf') " Test for extending a selection using right click call test_setmouse(2, 9) call term_sendkeys(buf, "\\") call test_setmouse(2, 16) call term_sendkeys(buf, "\\y") call TermWait(buf, 50) call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\") call WaitFor(XbufNotEmpty) call assert_equal("n yellow", readfile('Xbuf')[0]) call delete('Xbuf') " Test for pasting text using middle click call term_sendkeys(buf, ":let @r='bright '\") call test_setmouse(2, 22) call term_sendkeys(buf, "\"r\\") call TermWait(buf, 50) call term_sendkeys(buf, ":call writefile([getline(2)], 'Xbuf')\") call WaitFor(XbufNotEmpty) call assert_equal("red bright blue", readfile('Xbuf')[0][-15:]) call delete('Xbuf') " cleanup call TermWait(buf) call StopMNVInTerminal(buf) let &mouse = save_mouse let &term = save_term let &ttymouse = save_ttymouse let &clipboard = save_clipboard set mousetime& call delete('Xbuf') endfunc " Test for sync buffer cwd with shell's pwd func Test_terminal_sync_shell_dir() CheckUnix " The test always use sh (see src/testdir/util/unix.mnv). " BSD's sh doesn't seem to play well with the OSC 7 escape sequence. CheckNotBSD set asd " , is " 1. a valid character for directory names " 2. a reserved character in url-encoding let chars = ",a" " "," is url-encoded as '%2C' let chars_url = "%2Ca" let tmpfolder = fnamemodify(tempname(),':h') .. '/' .. chars let tmpfolder_url = fnamemodify(tempname(),':h') .. '/' .. chars_url call mkdir(tmpfolder, "p") let buf = Run_shell_in_terminal({}) call term_sendkeys(buf, "echo $'\\e\]7;file://" .. tmpfolder_url .. "\\a'\") "call term_sendkeys(buf, "cd " .. tmpfolder .. "\") call TermWait(buf) if has("mac") let expected = "/private" .. tmpfolder else let expected = tmpfolder endif call assert_equal(expected, getcwd(winnr())) set noasd endfunc " Test for modeless selection in a terminal func Test_term_modeless_selection() CheckUnix CheckNotGui CheckRunMNVInTerminal CheckFeature clipboard_working let save_mouse = &mouse let save_term = &term let save_ttymouse = &ttymouse set mouse=a term=xterm ttymouse=sgr mousetime=200 set clipboard=autoselectml let lines =<< trim END one two three four five red green yellow red blue mnv emacs sublime nano END call writefile(lines, 'Xtest_modeless', 'D') " Create a terminal window running MNV for the test with mouse disabled let prev_win = win_getid() let buf = RunMNVInTerminal('Xtest_modeless -n', {}) call term_sendkeys(buf, ":set nocompatible\") call term_sendkeys(buf, ":set mouse=\") call TermWait(buf) redraw! " Use the mouse to enter the terminal window call win_gotoid(prev_win) call feedkeys(MouseLeftClickCode(1, 1), 'x') call feedkeys(MouseLeftReleaseCode(1, 1), 'x') call TermWait(buf) call assert_equal(1, getwininfo(win_getid())[0].terminal) " Test for copying a modeless selection to clipboard let @* = 'clean' " communicating with X server may take a little time sleep 100m call feedkeys(MouseLeftClickCode(2, 3), 'x') call feedkeys(MouseLeftDragCode(2, 11), 'x') call feedkeys(MouseLeftReleaseCode(2, 11), 'x') call assert_equal("d green y", @*) " cleanup call TermWait(buf) call StopMNVInTerminal(buf) let &mouse = save_mouse let &term = save_term let &ttymouse = save_ttymouse set mousetime& clipboard& new | only! endfunc func Test_terminal_getwinpos() CheckRunMNVInTerminal " split, go to the bottom-right window split wincmd j set splitright let buf = RunMNVInTerminal('', {'cols': 60}) call TermWait(buf, 100) call term_sendkeys(buf, ":echo getwinpos(500)\") " Find the output of getwinpos() in the bottom line. let rows = term_getsize(buf)[0] call WaitForAssert({-> assert_match('\[\d\+, \d\+\]', term_getline(buf, rows))}) let line = term_getline(buf, rows) let xpos = str2nr(substitute(line, '\[\(\d\+\), \d\+\]', '\1', '')) let ypos = str2nr(substitute(line, '\[\d\+, \(\d\+\)\]', '\1', '')) " Position must be bigger than the getwinpos() result of MNV itself. " The calculation in the console assumes a 10 x 7 character cell. " In the GUI it can be more, let's assume a 20 x 14 cell. " And then add 100 / 200 tolerance. let [xroot, yroot] = getwinpos() let winpos = 50->getwinpos() call assert_equal(xroot, winpos[0]) call assert_equal(yroot, winpos[1]) let [winrow, wincol] = win_screenpos(0) let xoff = wincol * (has('gui_running') ? 14 : 7) + 100 let yoff = winrow * (has('gui_running') ? 20 : 10) + 200 call assert_inrange(xroot + 2, xroot + xoff, xpos) call assert_inrange(yroot + 2, yroot + yoff, ypos) call TermWait(buf) call term_sendkeys(buf, ":q\") call StopMNVInTerminal(buf) set splitright& only! endfunc func Test_terminal_term_start_error() func s:term_start_error() abort try return term_start([[]]) catch return v:exception finally " endtry endfunc autocmd WinEnter * call type(0) " Must not crash in s:term_start_error, nor the exception thrown. let result = s:term_start_error() call assert_match('^MNV(return):E730:', result) autocmd! WinEnter delfunc s:term_start_error endfunc func Test_terminal_vt420() CheckRunMNVInTerminal " For Termcap CheckUnix CheckExecutable infocmp let a = system('infocmp vt420') if v:shell_error " reset v:shell_error let a = system('true') throw 'Skipped: vt420 terminfo not available' endif let rows = 15 call writefile([':set term=vt420'], 'Xterm420', 'D') let buf = RunMNVInTerminal('-S Xterm420', #{rows: rows}) call TermWait(buf, 100) call term_sendkeys(buf, ":set t_xo?\") call WaitForAssert({-> assert_match('t_xo=y', term_getline(buf, rows))}) call StopMNVInTerminal(buf) call writefile([''], 'Xterm420') let buf = RunMNVInTerminal('-S Xterm420', #{rows: rows}) call TermWait(buf, 100) call term_sendkeys(buf, ":set t_xo?\") call WaitForAssert({-> assert_match('t_xo=\s\+', term_getline(buf, rows))}) call StopMNVInTerminal(buf) endfunc " Test for using 'vertical' with term_start(). If a following term_start(), " doesn't have the 'vertical' attribute, then it should be split horizontally. func Test_terminal_vertical() let lines =<< trim END call term_start("NONE", {'vertical': 1}) call term_start("NONE") VAR layout = winlayout() call assert_equal('row', layout[0], string(layout)) call assert_equal('col', layout[1][0][0], string(layout)) :%bw! END call v9.CheckLegacyAndMNV9Success(lines) endfunc " Needs to come before Test_hidden_terminal(), why? func Test_autocmd_buffilepost_with_hidden_term() CheckExecutable true new XTestFile defer delete('XTestFile') call setline(1, ['one', 'two', 'three']) call cursor(3, 10) augroup TestCursor au! autocmd BufFilePost * call setbufvar(3, '&tabstop', 4) augroup END let buf = term_start(['true'], #{hidden: 1, term_finish: 'close'}) call term_wait(buf) redraw! call assert_equal('XTestFile', bufname('%')) call assert_equal([0, 3, 5, 0], getpos('.')) augroup TestCursor au! augroup END augroup! TestCursor bw! XTestFile endfunc func Test_terminal_visual_empty_listchars() CheckScreendump CheckRunMNVInTerminal CheckUnix let lines = [ \ 'set listchars=', \ ':term sh -c "printf ''hello\\n\\nhello''"' \ ] call writefile(lines, 'XtermStart1', 'D') let buf = RunMNVInTerminal('-S XtermStart1', #{rows: 15}) call term_wait(buf) call term_sendkeys(buf, "V2k") call VerifyScreenDump(buf, 'Test_terminal_empty_listchars', {}) call term_sendkeys(buf, "\") call term_sendkeys(buf, ":set nu\") call term_sendkeys(buf, "ggV2j") call VerifyScreenDump(buf, 'Test_terminal_empty_listchars2', {}) call StopMNVInTerminal(buf) endfunc func Test_terminal_ansi_color_windows_cui() if !has('win32') || has('gui_running') throw 'Skipped: only for the Windows CUI' endif if exists('$APPVEYOR') throw 'Skipped: this test cannot be performed because AppVeyor does not support ANSI escape sequences' endif call assert_equal('dark', &background) " Outputs 16 ANSI colors as background colors let ansi = '' for i in range(16) let ansi ..= printf("\e[%dm%X", i + (i < 8 ? 40 : 92), i) endfor let ansi ..= "\e[40m\n" call writefile([ansi], 'XANSIcolor', 'D') let expected_chars = '0123456789ABCDEF' let expected_colors = [ \ '#000000', '#e00000', '#00e000', '#e0e000', \ '#0000e0', '#e000e0', '#00e0e0', '#e0e0e0', \ '#808080', '#ff4040', '#40ff40', '#ffff40', \ '#4040ff', '#ff40ff', '#40ffff', '#ffffff', \ ] " Ideally, 16 colors should be checked. However, in the current CI " environment, the 16th color is something other than white, and we don't " know the cause nor solution. After discussion, we decided to check only 15 " colors for the time being. " FIXME: Check all 16 colors in the future. let len_to_check = 15 let expected_colors = expected_colors[:len_to_check-1] " First, make sure mnv can display correct ANSI color text in terminal. let buf = term_start("cmd /C type XANSIcolor") call WaitForAssert({-> assert_equal(expected_chars, term_scrape(buf, 1)[:15]->map({_, v -> v['chars']})->join(''))}) call assert_equal(expected_colors, term_scrape(buf, 1)[:len_to_check-1]->map({_, v -> v['bg']})) bwipeout! " Next, check if mnv can do the same thing in the mnv terminal in terminal. let lines = [ \ 'call term_start("cmd /C type XANSIcolor")' \ ] call writefile(lines, 'XloadANSI', 'D') let cmd = GetMNVCommandCleanTerm() let buf = term_start(cmd .. '-S XloadANSI') call WaitForAssert({-> assert_equal(expected_chars, term_scrape(buf, 1)[:15]->map({_, v -> v['chars']})->join(''))}) call assert_equal(expected_colors, term_scrape(buf, 1)[:len_to_check-1]->map({_, v -> v['bg']})) endfunc func Test_terminal_backspace_on_windows() if !has('win32') throw 'Skipped: only for the Windows CUI' endif " Specify a simple prompt for easy comparison let save_prompt = $PROMPT let $PROMPT = '>' " Return the prompt line before the cursor func s:get_cmd_prompt(buf) let cur = term_getcursor(a:buf) return term_getline(a:buf, cur[0])[:cur[1]-2] endfunc let buf = term_start('cmd.exe') call WaitForAssert({-> assert_equal('>', s:get_cmd_prompt(buf))}, 100) " Verify sent characters are echoed back call term_sendkeys(buf, 'foo bar') call WaitForAssert({-> assert_equal('>foo bar', s:get_cmd_prompt(buf))}, 100) " Backspace should delete a character in front of the cursor call term_sendkeys(buf, "\") call WaitForAssert({-> assert_equal('>foo ba', s:get_cmd_prompt(buf))}, 100) " Ctrl+H behaves like Backspace call term_sendkeys(buf, "\") call WaitForAssert({-> assert_equal('>foo b', s:get_cmd_prompt(buf))}, 100) " Send a total of four BS and Ctrl+H to erase four characters. call term_sendkeys(buf, "\\\\") call WaitForAssert({-> assert_equal('>f', s:get_cmd_prompt(buf))}, 100) delfunc s:get_cmd_prompt let $PROMPT = save_prompt endfunc func Test_terminal_split_utf8() CheckUnix let buf = term_start('cat', {}) let chan = buf->term_getjob()->job_getchannel() call ch_sendraw(chan, "1: \xc3") call WaitForAssert({-> assert_equal('1: ', term_getline(buf, 1))}) call ch_sendraw(chan, "\xa5\xcc\xb2\n") call WaitForAssert({-> assert_equal('1: å̲', term_getline(buf, 1))}) call WaitForAssert({-> assert_equal('1: å̲', term_getline(buf, 2))}) call ch_sendraw(chan, "2: \xc3\xa5") call WaitForAssert({-> assert_equal('2: å', term_getline(buf, 3))}) call ch_sendraw(chan, "\xcc\xb2\n") call WaitForAssert({-> assert_equal('2: å̲', term_getline(buf, 3))}) call WaitForAssert({-> assert_equal('2: å̲', term_getline(buf, 4))}) call ch_sendraw(chan, "3: \xc3\xa5\xcc") call WaitForAssert({-> assert_equal('3: å', term_getline(buf, 5))}) call ch_sendraw(chan, "\xb2\n") call WaitForAssert({-> assert_equal('3: å̲', term_getline(buf, 5))}) call WaitForAssert({-> assert_equal('3: å̲', term_getline(buf, 6))}) exe buf .. "bwipe!" endfunc func Test_terminal_max_combining_chars() " somehow doesn't work on MS-Windows CheckUnix let cmd = "cat samples/terminal_max_combining_chars.txt\" let buf = Run_shell_in_terminal({'term_rows': 15, 'term_cols': 35}) call TermWait(buf) call term_sendkeys(buf, cmd) " last char is a space with many combining chars call WaitForAssert({-> assert_match("AAAAAAAAAAAAAAAAAAAAAAAAAAAA.", term_getline(buf, 14))}) call term_sendkeys(buf, "exit\r") exe buf . "bwipe!" endfunc func Test_term_getpos() CheckRunMNVInTerminal CheckUnix CheckExecutable seq defer delete('XTest_getpos_result') let lines =<< trim EOL term ++curwin sh EOL call writefile(lines, 'XTest_getpos', 'D') let buf = RunMNVInTerminal('-S XTest_getpos', {'rows': 15}) call term_sendkeys(buf, "for i in `seq 1 30`; do echo line$i; done\") call WaitForAssert({-> assert_match("line18", term_getline(buf, 1))}) call WaitForAssert({-> assert_match("line30", term_getline(buf, 13))}) call term_sendkeys(buf, "\:let g:job_w0 = line('w0')\") call term_sendkeys(buf, "\:let g:job_wdollar = line('w$')\") call term_sendkeys(buf, "\:call writefile([string(g:job_w0), string(g:job_wdollar)], 'XTest_getpos_result')\") call WaitForAssert({-> assert_true(filereadable('XTest_getpos_result'))}) call WaitForAssert({-> assert_equal(2, len(readfile('XTest_getpos_result')))}) let job_result = readfile('XTest_getpos_result') " 15 - 1: statusline - 1: prompt line call assert_equal(13, str2nr(job_result[1]) - str2nr(job_result[0])) call assert_true(str2nr(job_result[0]) > 1) call delete('XTest_getpos_result') " switch to Terminal-Normal mode and record w0/w$ call term_sendkeys(buf, "\N") call term_sendkeys(buf, ":let g:w0 = line('w0')\") call term_sendkeys(buf, ":let g:wdollar = line('w$')\") call term_sendkeys(buf, ":call writefile([string(g:w0), string(g:wdollar)], 'XTest_getpos_result')\") call WaitForAssert({-> assert_true(filereadable('XTest_getpos_result'))}) call WaitForAssert({-> assert_equal(2, len(readfile('XTest_getpos_result')))}) let result = readfile('XTest_getpos_result') " 15 - 1: statusline - 1: for prompt line call assert_equal(13, str2nr(result[1]) - str2nr(result[0])) call assert_true(str2nr(result[0]) > 1) " Regression: line('w0') and line('w$') must not move cursor position call term_sendkeys(buf, "gg") call term_sendkeys(buf, ":call line('w0')\") call term_sendkeys(buf, ":call line('w$')\") call term_wait(buf) call WaitForAssert({-> assert_match("for i in", term_getline(buf, 1))}) call WaitForAssert({-> assert_match("line12", term_getline(buf, 13))}) call StopMNVInTerminal(buf) " this crashed new setl buftype=terminal call assert_equal(2, line('w0') + line('w$')) bw endfunc func Test_term_autowrite() set autowrite new termautowritetestfile call setline(1, 'test content') term echo "test" call assert_equal(['test content'], readfile('termautowritetestfile')) call delete('termautowritetestfile') bwipe! set noautowrite endfunc " Test that CSI sequences with more than CSI_ARGS_MAX arguments do not crash func Test_terminal_csi_args_overflow() CheckExecutable printf let buf = term_start([&shell, &shellcmdflag, \ 'printf "\033[' . repeat('1;', 49) . '1m"']) " If we get here without a crash, the fix works call assert_equal('running', term_getstatus(buf)) call StopMNVInTerminal(buf) endfunc " mnv: shiftwidth=2 sts=2 expandtab