summaryrefslogtreecommitdiff
path: root/mnv/src/testdir/test_terminal3.mnv
diff options
context:
space:
mode:
authorMehmet Samet Duman <yongdohyun@projecttick.org>2026-04-04 12:41:27 +0300
committerMehmet Samet Duman <yongdohyun@projecttick.org>2026-04-04 12:41:27 +0300
commit4f2d36194b4f299aa7509d815c07121039ea833b (patch)
treef3ded014bad3a4c76ff6a22b8726ebaab68c3d13 /mnv/src/testdir/test_terminal3.mnv
parent5b578e70c314723a3cde5c9bfc2be0bf1dadc93b (diff)
downloadProject-Tick-4f2d36194b4f299aa7509d815c07121039ea833b.tar.gz
Project-Tick-4f2d36194b4f299aa7509d815c07121039ea833b.zip
NOISSUE change uvim folder name to mnv
Signed-off-by: Mehmet Samet Duman <yongdohyun@projecttick.org>
Diffstat (limited to 'mnv/src/testdir/test_terminal3.mnv')
-rw-r--r--mnv/src/testdir/test_terminal3.mnv1243
1 files changed, 1243 insertions, 0 deletions
diff --git a/mnv/src/testdir/test_terminal3.mnv b/mnv/src/testdir/test_terminal3.mnv
new file mode 100644
index 0000000000..dcf09801d5
--- /dev/null
+++ b/mnv/src/testdir/test_terminal3.mnv
@@ -0,0 +1,1243 @@
+" 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\<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.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()\<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 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()\<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 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)\<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 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\<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 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\<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 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<\<Esc>")
+ 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\<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 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\<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("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, "\<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 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'\<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
+ 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\<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 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)\<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 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\<CR>")
+ 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?\<CR>")
+ 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?\<CR>")
+ 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, "\<esc>")
+ call term_sendkeys(buf, ":set nu\<cr>")
+ 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, "\<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()
+ 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\<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 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