mirror of
https://github.com/macvim-dev/macvim.git
synced 2026-06-07 15:37:14 +02:00
54ffef7eb8
Problem: In the bundled libvterm the CSI 8 ; rows ; cols t sequence reaches
on_resize() without validating its arguments. Missing, zero or
negative dimensions cause a negative-size memmove() in
resize_buffer() and out-of-bounds accesses in set_lineinfo() and
DECALN, all reachable from output rendered in a terminal window
(Yukihiro Nakamura).
Solution: Reject missing, zero or negative dimensions before calling
on_resize(). Also clamp a negative cell width in on_text() as
hardening for the bundled libvterm.
Signed-off-by: Christian Brabandt <cb@256bit.org>
1321 lines
43 KiB
VimL
1321 lines
43 KiB
VimL
" Tests for the terminal window.
|
|
" This is split in two, because it can take a lot of time.
|
|
" See test_terminal.vim and test_terminal2.vim for further tests.
|
|
|
|
CheckFeature terminal
|
|
|
|
source util/screendump.vim
|
|
source util/mouse.vim
|
|
|
|
import './util/vim9.vim' as v9
|
|
|
|
let $PROMPT_COMMAND=''
|
|
|
|
func Test_terminal_altscreen()
|
|
" somehow doesn't work on MS-Windows
|
|
CheckUnix
|
|
let cmd = "cat Xtext\<CR>"
|
|
|
|
let buf = term_start(&shell, {})
|
|
call TermWait(buf)
|
|
call writefile(["\<Esc>[?1047h"], 'Xtext', 'D')
|
|
call term_sendkeys(buf, cmd)
|
|
call WaitForAssert({-> assert_equal(1, term_getaltscreen(buf))})
|
|
|
|
call writefile(["\<Esc>[?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.vim
|
|
call WaitForAssert({-> assert_match('job failed', term_getline(bufnr(), 1))})
|
|
bwipe!
|
|
|
|
term ++shell exec ls runtest.vim
|
|
call WaitForAssert({-> assert_match('runtest.vim', 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.vim
|
|
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.vim
|
|
call WaitForAssert({-> assert_match('runtest.vim', 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
|
|
CheckRunVimInTerminal
|
|
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 = RunVimInTerminal('-S XtermStart', #{rows: 15})
|
|
call WaitForChildNotification()
|
|
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
|
|
call WaitForChildNotification()
|
|
call term_sendkeys(buf, "hello\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_color_' .. a:group_name, {})
|
|
|
|
call term_sendkeys(buf, "\<C-D>")
|
|
call WaitForChildNotification()
|
|
call StopVimInTerminal(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
|
|
CheckRunVimInTerminal
|
|
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 = RunVimInTerminal('-S XtermStart', #{rows: 15})
|
|
call TermWait(buf, 100)
|
|
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, "hello\<CR>")
|
|
call TermWait(buf, 50)
|
|
|
|
call term_sendkeys(buf, "\<C-W>:split\<CR>")
|
|
call term_sendkeys(buf, "\<C-W>:set wincolor=MyWinCol\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_wincolor_split_MyWinCol', {})
|
|
|
|
call term_sendkeys(buf, "\<C-W>b:2sb\<CR>")
|
|
call term_sendkeys(buf, "\<C-W>:set wincolor=MyWinCol2\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_wincolor_split_MyWinCol2', {})
|
|
|
|
call term_sendkeys(buf, "\<C-D>")
|
|
call TermWait(buf, 50)
|
|
call StopVimInTerminal(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
|
|
CheckRunVimInTerminal
|
|
|
|
let text =<< trim END
|
|
some text
|
|
to edit
|
|
in a popup window
|
|
END
|
|
call writefile(text, 'Xtext', 'D')
|
|
let cmd = GetVimCommandCleanTerm()
|
|
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 = RunVimInTerminal('-S XtermPopup', #{rows: 15})
|
|
call TermWait(buf,0)
|
|
call term_sendkeys(buf, ":call OpenTerm(0)\<CR>")
|
|
call TermWait(buf,0)
|
|
call term_sendkeys(buf, ":\<CR>")
|
|
call TermWait(buf,0)
|
|
call term_sendkeys(buf, "\<C-W>:echo getwinvar(g:winid, \"&buftype\") win_gettype(g:winid)\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_1', {})
|
|
|
|
call term_sendkeys(buf, ":q\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_2', {})
|
|
|
|
call term_sendkeys(buf, ":call OpenTerm(1)\<CR>")
|
|
call TermWait(buf,0)
|
|
call term_sendkeys(buf, ":set hlsearch\<CR>")
|
|
call TermWait(buf,0)
|
|
call term_sendkeys(buf, "/edit\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_3', {})
|
|
|
|
call term_sendkeys(buf, "\<C-W>:call HidePopup()\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_4', {})
|
|
call term_sendkeys(buf, "\<CR>")
|
|
call TermWait(buf,0)
|
|
|
|
call term_sendkeys(buf, "\<C-W>:call ClosePopup()\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_5', {})
|
|
|
|
call term_sendkeys(buf, "\<C-W>:call ReopenPopup()\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_6', {})
|
|
|
|
" Go to terminal-Normal mode and visually select text.
|
|
call term_sendkeys(buf, "\<C-W>Ngg/in\<CR>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\<CR>")
|
|
call WaitForAssert({-> assert_equal(0, match(term_getline(buf, 6), '^5\s*$'))}, 250) " wait for terminal to vanish
|
|
|
|
call StopVimInTerminal(buf)
|
|
endfunc
|
|
|
|
" Check a terminal in popup window uses the default minimum size.
|
|
func Test_terminal_in_popup_min_size()
|
|
CheckScreendump
|
|
CheckRunVimInTerminal
|
|
|
|
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 = RunVimInTerminal('-S XtermPopup', #{rows: 15})
|
|
call TermWait(buf, 100)
|
|
call term_sendkeys(buf, ":set noruler\<CR>")
|
|
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_m1', {})
|
|
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":q\<CR>")
|
|
call TermWait(buf, 50) " wait for terminal to vanish
|
|
call StopVimInTerminal(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
|
|
CheckRunVimInTerminal
|
|
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 = RunVimInTerminal('-S XtermPopup', #{rows: 15})
|
|
call WaitForChildNotification()
|
|
call term_sendkeys(buf, ":set noruler\<CR>")
|
|
call term_sendkeys(buf, ":call OpenTerm()\<CR>")
|
|
call WaitForChildNotification()
|
|
call term_sendkeys(buf, "hello\<CR>")
|
|
call VerifyScreenDump(buf, 'Test_terminal_popup_' .. a:group_name, {})
|
|
|
|
call term_sendkeys(buf, "\<C-D>")
|
|
call WaitForChildNotification()
|
|
call StopVimInTerminal(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()
|
|
CheckRunVimInTerminal
|
|
CheckUnix
|
|
|
|
let lines =<< trim EOL
|
|
put='some text'
|
|
term
|
|
startinsert
|
|
EOL
|
|
call writefile(lines, 'XTest_startinsert', 'D')
|
|
let buf = RunVimInTerminal('-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<\<Esc>")
|
|
call WaitForAssert({-> assert_equal("some text<", term_getline(buf, 1))})
|
|
|
|
call StopVimInTerminal(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()
|
|
CheckRunVimInTerminal
|
|
|
|
let buf = RunVimInTerminal('', {})
|
|
call term_sendkeys(buf, ":set nocompatible\<CR>")
|
|
call term_sendkeys(buf, ":set timeoutlen=20\<CR>")
|
|
|
|
let keys = ["\<F1>", "\<F2>", "\<F3>", "\<F4>", "\<F5>", "\<F6>", "\<F7>",
|
|
\ "\<F8>", "\<F9>", "\<F10>", "\<F11>", "\<F12>", "\<Home>",
|
|
\ "\<S-Home>", "\<C-Home>", "\<End>", "\<S-End>", "\<C-End>",
|
|
\ "\<Ins>", "\<Del>", "\<Left>", "\<S-Left>", "\<C-Left>", "\<Right>",
|
|
\ "\<S-Right>", "\<C-Right>", "\<Up>", "\<S-Up>", "\<Down>",
|
|
\ "\<S-Down>"]
|
|
let output = ['<F1>', '<F2>', '<F3>', '<F4>', '<F5>', '<F6>', '<F7>',
|
|
\ '<F8>', '<F9>', '<F10>', '<F11>', '<F12>', '<Home>', '<S-Home>',
|
|
\ '<C-Home>', '<End>', '<S-End>', '<C-End>', '<Insert>', '<Del>',
|
|
\ '<Left>', '<S-Left>', '<C-Left>', '<Right>', '<S-Right>',
|
|
\ '<C-Right>', '<Up>', '<S-Up>', '<Down>', '<S-Down>']
|
|
|
|
call term_sendkeys(buf, "i")
|
|
for i in range(len(keys))
|
|
call term_sendkeys(buf, "\<C-U>\<C-K>" .. keys[i])
|
|
call WaitForAssert({-> assert_equal(output[i], term_getline(buf, 1))}, 200)
|
|
endfor
|
|
|
|
let keypad_keys = ["\<k0>", "\<k1>", "\<k2>", "\<k3>", "\<k4>", "\<k5>",
|
|
\ "\<k6>", "\<k7>", "\<k8>", "\<k9>", "\<kPoint>", "\<kPlus>",
|
|
\ "\<kMinus>", "\<kMultiply>", "\<kDivide>"]
|
|
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, "\<C-U>" .. keypad_keys[i])
|
|
call WaitForAssert({-> assert_equal(keypad_output[i], term_getline(buf, 1))}, 100)
|
|
endfor
|
|
|
|
call feedkeys("\<C-U>\<kEnter>\<BS>one\<C-W>.two", 'xt')
|
|
call WaitForAssert({-> assert_equal('two', term_getline(buf, 1))})
|
|
|
|
call StopVimInTerminal(buf)
|
|
endfunc
|
|
|
|
" Test for using the mouse in a terminal
|
|
func Test_term_mouse()
|
|
CheckNotGui
|
|
CheckRunVimInTerminal
|
|
|
|
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
|
|
vim emacs sublime nano
|
|
END
|
|
call writefile(lines, 'Xtest_mouse', 'D')
|
|
|
|
" Create a terminal window running Vim for the test with mouse enabled
|
|
let prev_win = win_getid()
|
|
let buf = RunVimInTerminal('Xtest_mouse -n', {})
|
|
call term_sendkeys(buf, ":set nocompatible\<CR>")
|
|
call term_sendkeys(buf, ":set mouse=a term=xterm ttymouse=sgr\<CR>")
|
|
call term_sendkeys(buf, ":set clipboard=\<CR>")
|
|
call term_sendkeys(buf, ":set mousemodel=extend\<CR>")
|
|
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 <LeftMouse> click/release
|
|
call test_setmouse(2, 5)
|
|
call feedkeys("\<LeftMouse>\<LeftRelease>", 'xt')
|
|
call test_setmouse(3, 8)
|
|
call term_sendkeys(buf, "\<LeftMouse>\<LeftRelease>")
|
|
call TermWait(buf, 50)
|
|
call delete('Xbuf')
|
|
call term_sendkeys(buf, ":call writefile([json_encode(getpos('.'))], 'Xbuf')\<CR>")
|
|
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, "\<LeftMouse>")
|
|
call test_setmouse(2, 16)
|
|
call term_sendkeys(buf, "\<LeftRelease>y")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
|
|
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, "\<LeftMouse>\<LeftRelease>\<LeftMouse>")
|
|
call test_setmouse(1, 17)
|
|
call term_sendkeys(buf, "\<LeftRelease>y")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
|
|
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, "\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>y")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
|
|
call WaitFor(XbufNotEmpty)
|
|
call assert_equal("vim 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, "\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>\<LeftMouse>\<LeftRelease>\<LeftMouse>")
|
|
call test_setmouse(3, 13)
|
|
call term_sendkeys(buf, "\<LeftRelease>y")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
|
|
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, "\<LeftMouse>\<LeftRelease>")
|
|
call test_setmouse(2, 16)
|
|
call term_sendkeys(buf, "\<RightMouse>\<RightRelease>y")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":call writefile([@\"], 'Xbuf')\<CR>")
|
|
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 '\<CR>")
|
|
call test_setmouse(2, 22)
|
|
call term_sendkeys(buf, "\"r\<MiddleMouse>\<MiddleRelease>")
|
|
call TermWait(buf, 50)
|
|
call term_sendkeys(buf, ":call writefile([getline(2)], 'Xbuf')\<CR>")
|
|
call WaitFor(XbufNotEmpty)
|
|
call assert_equal("red bright blue", readfile('Xbuf')[0][-15:])
|
|
call delete('Xbuf')
|
|
|
|
" cleanup
|
|
call TermWait(buf)
|
|
call StopVimInTerminal(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.vim).
|
|
" 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'\<CR>")
|
|
"call term_sendkeys(buf, "cd " .. tmpfolder .. "\<CR>")
|
|
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
|
|
CheckRunVimInTerminal
|
|
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
|
|
vim emacs sublime nano
|
|
END
|
|
call writefile(lines, 'Xtest_modeless', 'D')
|
|
|
|
" Create a terminal window running Vim for the test with mouse disabled
|
|
let prev_win = win_getid()
|
|
let buf = RunVimInTerminal('Xtest_modeless -n', {})
|
|
call term_sendkeys(buf, ":set nocompatible\<CR>")
|
|
call term_sendkeys(buf, ":set mouse=\<CR>")
|
|
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 StopVimInTerminal(buf)
|
|
let &mouse = save_mouse
|
|
let &term = save_term
|
|
let &ttymouse = save_ttymouse
|
|
set mousetime& clipboard&
|
|
new | only!
|
|
endfunc
|
|
|
|
func Test_terminal_getwinpos()
|
|
CheckRunVimInTerminal
|
|
|
|
" split, go to the bottom-right window
|
|
split
|
|
wincmd j
|
|
set splitright
|
|
|
|
let buf = RunVimInTerminal('', {'cols': 60})
|
|
call TermWait(buf, 100)
|
|
call term_sendkeys(buf, ":echo getwinpos(500)\<CR>")
|
|
|
|
" 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 Vim 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\<CR>")
|
|
call StopVimInTerminal(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('^Vim(return):E730:', result)
|
|
|
|
autocmd! WinEnter
|
|
delfunc s:term_start_error
|
|
endfunc
|
|
|
|
func Test_terminal_vt420()
|
|
CheckRunVimInTerminal
|
|
" 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 = RunVimInTerminal('-S Xterm420', #{rows: rows})
|
|
call TermWait(buf, 100)
|
|
call term_sendkeys(buf, ":set t_xo?\<CR>")
|
|
call WaitForAssert({-> assert_match('t_xo=y', term_getline(buf, rows))})
|
|
call StopVimInTerminal(buf)
|
|
|
|
call writefile([''], 'Xterm420')
|
|
let buf = RunVimInTerminal('-S Xterm420', #{rows: rows})
|
|
call TermWait(buf, 100)
|
|
call term_sendkeys(buf, ":set t_xo?\<CR>")
|
|
call WaitForAssert({-> assert_match('t_xo=\s\+', term_getline(buf, rows))})
|
|
call StopVimInTerminal(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.CheckLegacyAndVim9Success(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
|
|
CheckRunVimInTerminal
|
|
CheckUnix
|
|
|
|
let lines = [
|
|
\ 'set listchars=',
|
|
\ ':term sh -c "printf ''hello\\n\\nhello''"'
|
|
\ ]
|
|
call writefile(lines, 'XtermStart1', 'D')
|
|
let buf = RunVimInTerminal('-S XtermStart1', #{rows: 15})
|
|
call term_wait(buf)
|
|
call term_sendkeys(buf, "V2k")
|
|
call VerifyScreenDump(buf, 'Test_terminal_empty_listchars', {})
|
|
call term_sendkeys(buf, "\<esc>")
|
|
call term_sendkeys(buf, ":set nu\<cr>")
|
|
call term_sendkeys(buf, "ggV2j")
|
|
call VerifyScreenDump(buf, 'Test_terminal_empty_listchars2', {})
|
|
|
|
call StopVimInTerminal(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 vim 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 vim can do the same thing in the vim terminal in terminal.
|
|
let lines = [
|
|
\ 'call term_start("cmd /C type XANSIcolor")'
|
|
\ ]
|
|
call writefile(lines, 'XloadANSI', 'D')
|
|
let cmd = GetVimCommandCleanTerm()
|
|
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, "\<BS>")
|
|
call WaitForAssert({-> assert_equal('>foo ba', s:get_cmd_prompt(buf))}, 100)
|
|
" Ctrl+H behaves like Backspace
|
|
call term_sendkeys(buf, "\<C-H>")
|
|
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, "\<BS>\<BS>\<C-H>\<C-H>")
|
|
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\<CR>"
|
|
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()
|
|
CheckRunVimInTerminal
|
|
CheckUnix
|
|
CheckExecutable seq
|
|
defer delete('XTest_getpos_result')
|
|
|
|
let lines =<< trim EOL
|
|
term ++curwin sh
|
|
EOL
|
|
call writefile(lines, 'XTest_getpos', 'D')
|
|
let buf = RunVimInTerminal('-S XTest_getpos', {'rows': 15})
|
|
call term_sendkeys(buf, "for i in `seq 1 30`; do echo line$i; done\<cr>")
|
|
|
|
call WaitForAssert({-> assert_match("line18", term_getline(buf, 1))})
|
|
call WaitForAssert({-> assert_match("line30", term_getline(buf, 13))})
|
|
|
|
call term_sendkeys(buf, "\<c-w>:let g:job_w0 = line('w0')\<cr>")
|
|
call term_sendkeys(buf, "\<c-w>:let g:job_wdollar = line('w$')\<cr>")
|
|
call term_sendkeys(buf, "\<c-w>:call writefile([string(g:job_w0), string(g:job_wdollar)], 'XTest_getpos_result')\<cr>")
|
|
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, "\<c-w>N")
|
|
call term_sendkeys(buf, ":let g:w0 = line('w0')\<cr>")
|
|
call term_sendkeys(buf, ":let g:wdollar = line('w$')\<cr>")
|
|
call term_sendkeys(buf, ":call writefile([string(g:w0), string(g:wdollar)], 'XTest_getpos_result')\<cr>")
|
|
|
|
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')\<cr>")
|
|
call term_sendkeys(buf, ":call line('w$')\<cr>")
|
|
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 StopVimInTerminal(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 seq = "\033[" .. repeat('1;', 49) .. '1m'
|
|
let seq ..= "\033[1111111111111111111m"
|
|
let buf = term_start([&shell, &shellcmdflag, 'printf "' .. seq .. '"'])
|
|
|
|
" If we get here without a crash, the fix works
|
|
call assert_equal('running', term_getstatus(buf))
|
|
call StopVimInTerminal(buf)
|
|
endfunc
|
|
|
|
func Test_terminal_output_combining_chars()
|
|
CheckUnix
|
|
new
|
|
let cmd = "cat samples/combining_chars.txt"
|
|
let buf = term_start(cmd, {'curwin': 1, 'term_finish': 'open', 'term_rows': 10, 'term_cols': 30})
|
|
call WaitForAssert({-> assert_match('finished', term_getstatus(buf))})
|
|
call TermWait(buf)
|
|
let lines = getbufline(buf, 1, '$')
|
|
" get byte lengths to confirm combining chars present
|
|
let lens = map(copy(lines), 'len(v:val)')
|
|
let expected = repeat([11], 190) + repeat([14], 10)
|
|
call assert_equal(expected, lens)
|
|
bw!
|
|
endfunc
|
|
|
|
" This caused a Crash
|
|
func Test_terminal_csi_resize_oob()
|
|
return
|
|
CheckUnix
|
|
CheckExecutable printf
|
|
|
|
" CSI 8 ; rows ; cols t with missing, zero or negative dimensions reached
|
|
" on_resize()/resize_buffer() unvalidated, causing a negative-size memmove()
|
|
" and out-of-bounds set_lineinfo()/DECALN accesses in libvterm. Rendering
|
|
" these must not crash Vim.
|
|
|
|
" Sequences:
|
|
" 1 resize_buffer negative-size memmove
|
|
" 2 set_lineinfo OOB after corrupt resize
|
|
" 3 DECALN putglyph OOB after corrupt resize
|
|
let seqs = ["\<ESC>[8;0;t",
|
|
\ "\<ESC>[8;;t\<ESC>[J",
|
|
\ "\<ESC>[8;;0t\<ESC>#8"]
|
|
|
|
for seq in seqs
|
|
let buf = term_start([&shell, &shellcmdflag, 'printf "%s" ' .. shellescape(seq)],
|
|
\ #{term_rows: 10, term_cols: 40})
|
|
call TermWait(buf)
|
|
" Getting here without a crash (and no ASAN report) is the test.
|
|
call assert_true(bufexists(buf))
|
|
exe 'bwipe! ' .. buf
|
|
endfor
|
|
endfunc
|
|
|
|
" This caused a Crash, but Vim builds libvterm using -DWCWIDTH_FUNCTION=utf_uint2cells
|
|
" which wouldn't return -1 and therefore does not reproduce here
|
|
func Test_terminal_negative_col_oob()
|
|
CheckUnix
|
|
CheckExecutable printf
|
|
|
|
" A wcwidth() == -1 codepoint (U+0087, \302\207 in UTF-8) drove
|
|
" state->pos.col negative, then used as an array index in the tabstop
|
|
" helpers and moverect_internal(). These are single-byte / single-bit
|
|
" out-of-bounds accesses that do not crash a normal build, so this test
|
|
" is only meaningful under a sanitizer build; otherwise it just confirms
|
|
" Vim does not crash.
|
|
|
|
" Sequences:
|
|
" 1. clear_col_tabstop OOB read
|
|
" 2. is_col_tabstop OOB read
|
|
" 3. set_col_tabstop OOB write (HTS)
|
|
" 4. moverect_internal memmove (insert mode)
|
|
|
|
let seqs = ["\302\207\<ESC>[g",
|
|
\ "\302\207\302\207\t",
|
|
\ "\302\207\<ESC>H",
|
|
\ "\<ESC>[4h\302\2070"]
|
|
for seq in seqs
|
|
let buf = term_start([&shell, &shellcmdflag, 'printf "%s" ' .. shellescape(seq)],
|
|
\ #{term_rows: 10, term_cols: 40})
|
|
call TermWait(buf)
|
|
call assert_true(bufexists(buf))
|
|
exe 'bwipe! ' .. buf
|
|
endfor
|
|
endfunc
|
|
|
|
" vim: shiftwidth=2 sts=2 expandtab
|