mirror of
https://github.com/neovim/neovim.git
synced 2024-12-19 18:55:14 -07:00
e61228a214
Before calling "attach" a screen object is just a dummy container for (row, col) values whose purpose is to be sent as part of the "attach" function call anyway. Just create the screen in an attached state directly. Keep the complete (row, col, options) config together. It is still completely valid to later detach and re-attach as needed, including to another session.
798 lines
27 KiB
Lua
798 lines
27 KiB
Lua
local t = require('test.testutil')
|
|
local n = require('test.functional.testnvim')()
|
|
local Screen = require('test.functional.ui.screen')
|
|
|
|
local eq = t.eq
|
|
local feed = n.feed
|
|
local clear = n.clear
|
|
local api = n.api
|
|
local fn = n.fn
|
|
local source = n.source
|
|
local exec_capture = n.exec_capture
|
|
local dedent = t.dedent
|
|
local command = n.command
|
|
|
|
local screen
|
|
|
|
-- Bug in input() handling: :redraw! will erase the whole prompt up until
|
|
-- user types something. It exists in Vim as well, so using `h<BS>` as
|
|
-- a workaround.
|
|
local function redraw_input()
|
|
feed('{REDRAW}h<BS>')
|
|
end
|
|
|
|
before_each(function()
|
|
clear()
|
|
screen = Screen.new(40, 8)
|
|
source([[
|
|
highlight RBP1 guibg=Red
|
|
highlight RBP2 guibg=Yellow
|
|
highlight RBP3 guibg=Green
|
|
highlight RBP4 guibg=Blue
|
|
let g:NUM_LVLS = 4
|
|
function Redraw()
|
|
mode
|
|
return "\<Ignore>"
|
|
endfunction
|
|
let g:id = ''
|
|
cnoremap <expr> {REDRAW} Redraw()
|
|
function DoPrompt(do_return) abort
|
|
let id = g:id
|
|
let Cb = g:Nvim_color_input{g:id}
|
|
let out = input({'prompt': ':', 'highlight': Cb})
|
|
let g:out{id} = out
|
|
return (a:do_return ? out : "\<Ignore>")
|
|
endfunction
|
|
nnoremap <expr> {PROMPT} DoPrompt(0)
|
|
cnoremap <expr> {PROMPT} DoPrompt(1)
|
|
function RainBowParens(cmdline)
|
|
let ret = []
|
|
let i = 0
|
|
let lvl = 0
|
|
while i < len(a:cmdline)
|
|
if a:cmdline[i] is# '('
|
|
call add(ret, [i, i + 1, 'RBP' . ((lvl % g:NUM_LVLS) + 1)])
|
|
let lvl += 1
|
|
elseif a:cmdline[i] is# ')'
|
|
let lvl -= 1
|
|
call add(ret, [i, i + 1, 'RBP' . ((lvl % g:NUM_LVLS) + 1)])
|
|
endif
|
|
let i += 1
|
|
endwhile
|
|
return ret
|
|
endfunction
|
|
function SplitMultibyteStart(cmdline)
|
|
let ret = []
|
|
let i = 0
|
|
while i < len(a:cmdline)
|
|
let char = nr2char(char2nr(a:cmdline[i:]))
|
|
if a:cmdline[i:i + len(char) - 1] is# char
|
|
if len(char) > 1
|
|
call add(ret, [i + 1, i + len(char), 'RBP2'])
|
|
endif
|
|
let i += len(char)
|
|
else
|
|
let i += 1
|
|
endif
|
|
endwhile
|
|
return ret
|
|
endfunction
|
|
function SplitMultibyteEnd(cmdline)
|
|
let ret = []
|
|
let i = 0
|
|
while i < len(a:cmdline)
|
|
let char = nr2char(char2nr(a:cmdline[i:]))
|
|
if a:cmdline[i:i + len(char) - 1] is# char
|
|
if len(char) > 1
|
|
call add(ret, [i, i + 1, 'RBP1'])
|
|
endif
|
|
let i += len(char)
|
|
else
|
|
let i += 1
|
|
endif
|
|
endwhile
|
|
return ret
|
|
endfunction
|
|
function Echoing(cmdline)
|
|
echo 'HERE'
|
|
return v:_null_list
|
|
endfunction
|
|
function Echoning(cmdline)
|
|
echon 'HERE'
|
|
return v:_null_list
|
|
endfunction
|
|
function Echomsging(cmdline)
|
|
echomsg 'HERE'
|
|
return v:_null_list
|
|
endfunction
|
|
function Echoerring(cmdline)
|
|
echoerr 'HERE'
|
|
return v:_null_list
|
|
endfunction
|
|
function Redrawing(cmdline)
|
|
redraw!
|
|
return v:_null_list
|
|
endfunction
|
|
function Throwing(cmdline)
|
|
throw "ABC"
|
|
return v:_null_list
|
|
endfunction
|
|
function Halting(cmdline)
|
|
while 1
|
|
endwhile
|
|
endfunction
|
|
function ReturningGlobal(cmdline)
|
|
return g:callback_return
|
|
endfunction
|
|
function ReturningGlobal2(cmdline)
|
|
return g:callback_return[:len(a:cmdline)-1]
|
|
endfunction
|
|
function ReturningGlobalN(n, cmdline)
|
|
return g:callback_return{a:n}
|
|
endfunction
|
|
let g:recording_calls = []
|
|
function Recording(cmdline)
|
|
call add(g:recording_calls, a:cmdline)
|
|
return []
|
|
endfunction
|
|
]])
|
|
screen:set_default_attr_ids({
|
|
RBP1 = { background = Screen.colors.Red },
|
|
RBP2 = { background = Screen.colors.Yellow },
|
|
RBP3 = { background = Screen.colors.Green },
|
|
RBP4 = { background = Screen.colors.Blue },
|
|
EOB = { bold = true, foreground = Screen.colors.Blue1 },
|
|
ERR = { foreground = Screen.colors.Grey100, background = Screen.colors.Red },
|
|
SK = { foreground = Screen.colors.Blue },
|
|
PE = { bold = true, foreground = Screen.colors.SeaGreen4 },
|
|
NUM = { foreground = Screen.colors.Blue2 },
|
|
NPAR = { foreground = Screen.colors.Yellow },
|
|
SQ = { foreground = Screen.colors.Blue3 },
|
|
SB = { foreground = Screen.colors.Blue4 },
|
|
E = { foreground = Screen.colors.Red, background = Screen.colors.Blue },
|
|
M = { bold = true },
|
|
MSEP = { bold = true, reverse = true },
|
|
})
|
|
end)
|
|
|
|
local function set_color_cb(funcname, callback_return, id)
|
|
api.nvim_set_var('id', id or '')
|
|
if id and id ~= '' and fn.exists('*' .. funcname .. 'N') then
|
|
command(('let g:Nvim_color_input%s = {cmdline -> %sN(%s, cmdline)}'):format(id, funcname, id))
|
|
if callback_return then
|
|
api.nvim_set_var('callback_return' .. id, callback_return)
|
|
end
|
|
else
|
|
api.nvim_set_var('Nvim_color_input', funcname)
|
|
if callback_return then
|
|
api.nvim_set_var('callback_return', callback_return)
|
|
end
|
|
end
|
|
end
|
|
local function start_prompt(text)
|
|
feed('{PROMPT}' .. (text or ''))
|
|
end
|
|
|
|
describe('Command-line coloring', function()
|
|
it('works', function()
|
|
set_color_cb('RainBowParens')
|
|
api.nvim_set_option_value('more', false, {})
|
|
start_prompt()
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:^ |
|
|
]])
|
|
feed('e')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:e^ |
|
|
]])
|
|
feed('cho ')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo ^ |
|
|
]])
|
|
feed('(')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}^ |
|
|
]])
|
|
feed('(')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}{RBP2:(}^ |
|
|
]])
|
|
feed('42')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}{RBP2:(}42^ |
|
|
]])
|
|
feed('))')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}{RBP2:(}42{RBP2:)}{RBP1:)}^ |
|
|
]])
|
|
feed('<BS>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}{RBP2:(}42{RBP2:)}^ |
|
|
]])
|
|
redraw_input()
|
|
screen:expect {
|
|
grid = [[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}{RBP2:(}42{RBP2:)}^ |
|
|
]],
|
|
reset = true,
|
|
}
|
|
end)
|
|
for _, func_part in ipairs({ '', 'n', 'msg' }) do
|
|
it('disables :echo' .. func_part .. ' messages', function()
|
|
set_color_cb('Echo' .. func_part .. 'ing')
|
|
start_prompt('echo')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo^ |
|
|
]])
|
|
end)
|
|
end
|
|
it('does the right thing when hl start appears to split multibyte char', function()
|
|
set_color_cb('SplitMultibyteStart')
|
|
start_prompt('echo "«')
|
|
screen:expect {
|
|
grid = [[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:echo " |
|
|
{ERR:E5405: Chunk 0 start 7 splits multibyte }|
|
|
{ERR:character} |
|
|
:echo "«^ |
|
|
]],
|
|
}
|
|
feed('»')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:echo " |
|
|
{ERR:E5405: Chunk 0 start 7 splits multibyte }|
|
|
{ERR:character} |
|
|
:echo "«»^ |
|
|
]])
|
|
end)
|
|
it('does the right thing when hl end appears to split multibyte char', function()
|
|
set_color_cb('SplitMultibyteEnd')
|
|
start_prompt('echo "«')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:echo " |
|
|
{ERR:E5406: Chunk 0 end 7 splits multibyte ch}|
|
|
{ERR:aracter} |
|
|
:echo "«^ |
|
|
]])
|
|
end)
|
|
it('does the right thing when erroring', function()
|
|
set_color_cb('Echoerring')
|
|
start_prompt('e')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|
|
|
{MSEP: }|
|
|
: |
|
|
{ERR:E5407: Callback has thrown an exception:}|
|
|
{ERR: function DoPrompt[3]..Echoerring, line }|
|
|
{ERR:1: Vim(echoerr):HERE} |
|
|
:e^ |
|
|
]])
|
|
end)
|
|
it('silences :echo', function()
|
|
set_color_cb('Echoing')
|
|
start_prompt('e')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:e^ |
|
|
]])
|
|
eq('', exec_capture('messages'))
|
|
end)
|
|
it('silences :echon', function()
|
|
set_color_cb('Echoning')
|
|
start_prompt('e')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:e^ |
|
|
]])
|
|
eq('', exec_capture('messages'))
|
|
end)
|
|
it('silences :echomsg', function()
|
|
set_color_cb('Echomsging')
|
|
start_prompt('e')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:e^ |
|
|
]])
|
|
eq('', exec_capture('messages'))
|
|
end)
|
|
it('does the right thing when throwing', function()
|
|
set_color_cb('Throwing')
|
|
start_prompt('e')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|
|
|
{MSEP: }|
|
|
: |
|
|
{ERR:E5407: Callback has thrown an exception:}|
|
|
{ERR: function DoPrompt[3]..Throwing, line 1:}|
|
|
{ERR: ABC} |
|
|
:e^ |
|
|
]])
|
|
end)
|
|
it('stops executing callback after a number of errors', function()
|
|
set_color_cb('SplitMultibyteStart')
|
|
start_prompt('let x = "«»«»«»«»«»"')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:let x = " |
|
|
{ERR:E5405: Chunk 0 start 10 splits multibyte}|
|
|
{ERR: character} |
|
|
:let x = "«»«»«»«»«»"^ |
|
|
]])
|
|
feed('\n')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
|
|
|
]])
|
|
feed('\n')
|
|
eq('let x = "«»«»«»«»«»"', api.nvim_get_var('out'))
|
|
local msg = '\nE5405: Chunk 0 start 10 splits multibyte character'
|
|
eq(msg:rep(1), fn.execute('messages'))
|
|
end)
|
|
it('allows interrupting callback with <C-c>', function()
|
|
set_color_cb('Halting')
|
|
start_prompt('echo 42')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
|
|
|
]])
|
|
screen:sleep(500)
|
|
feed('<C-c>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
: |
|
|
{ERR:E5407: Callback has thrown an exception:}|
|
|
{ERR: Keyboard interrupt} |
|
|
:echo 42^ |
|
|
]])
|
|
redraw_input()
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo 42^ |
|
|
]])
|
|
feed('\n')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
:echo 42 |
|
|
]])
|
|
feed('\n')
|
|
eq('echo 42', api.nvim_get_var('out'))
|
|
feed('<C-c>')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
Type :qa and pre...nter> to exit Nvim |
|
|
]])
|
|
end)
|
|
it('works fine with NUL, NL, CR', function()
|
|
set_color_cb('RainBowParens')
|
|
start_prompt('echo ("<C-v><CR><C-v><Nul><C-v><NL>")')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}"{SK:^M^@^@}"{RBP1:)}^ |
|
|
]])
|
|
end)
|
|
it('errors out when callback returns something wrong', function()
|
|
command('cnoremap + ++')
|
|
set_color_cb('ReturningGlobal', '')
|
|
start_prompt('#')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*3
|
|
{MSEP: }|
|
|
: |
|
|
{ERR:E5400: Callback should return list} |
|
|
:#^ |
|
|
]])
|
|
|
|
feed('<CR><CR><CR>')
|
|
set_color_cb('ReturningGlobal', { { 0, 1, 'Normal' }, 42 })
|
|
start_prompt('#')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*3
|
|
{MSEP: }|
|
|
: |
|
|
{ERR:E5401: List item 1 is not a List} |
|
|
:#^ |
|
|
]])
|
|
|
|
feed('<CR><CR><CR>')
|
|
set_color_cb('ReturningGlobal2', { { 0, 1, 'Normal' }, { 1 } })
|
|
start_prompt('+')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:+ |
|
|
{ERR:E5402: List item 1 has incorrect length:}|
|
|
{ERR: 1 /= 3} |
|
|
:++^ |
|
|
]])
|
|
|
|
feed('<CR><CR><CR>')
|
|
set_color_cb('ReturningGlobal2', { { 0, 1, 'Normal' }, { 2, 3, 'Normal' } })
|
|
start_prompt('+')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:+ |
|
|
{ERR:E5403: Chunk 1 start 2 not in range [1, }|
|
|
{ERR:2)} |
|
|
:++^ |
|
|
]])
|
|
|
|
feed('<CR><CR><CR>')
|
|
set_color_cb('ReturningGlobal2', { { 0, 1, 'Normal' }, { 1, 3, 'Normal' } })
|
|
start_prompt('+')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*3
|
|
{MSEP: }|
|
|
:+ |
|
|
{ERR:E5404: Chunk 1 end 3 not in range (1, 2]}|
|
|
:++^ |
|
|
]])
|
|
end)
|
|
it('does not error out when called from a errorred out cycle', function()
|
|
set_color_cb('ReturningGlobal', { { 0, 1, 'Normal' } })
|
|
feed(dedent([[
|
|
:set regexpengine=2
|
|
:for pat in [' \ze*', ' \zs*']
|
|
: try
|
|
: let l = matchlist('x x', pat)
|
|
: $put =input({'prompt':'>','highlight':'ReturningGlobal'})
|
|
:
|
|
: $put ='E888 NOT detected for ' . pat
|
|
: catch
|
|
: $put =input({'prompt':'>','highlight':'ReturningGlobal'})
|
|
:
|
|
: $put ='E888 detected for ' . pat
|
|
: endtry
|
|
:endfor
|
|
:
|
|
:
|
|
:
|
|
:
|
|
:
|
|
:
|
|
]]))
|
|
eq(
|
|
{ '', ':', 'E888 detected for \\ze*', ':', 'E888 detected for \\zs*' },
|
|
api.nvim_buf_get_lines(0, 0, -1, false)
|
|
)
|
|
eq('', fn.execute('messages'))
|
|
end)
|
|
it('allows nesting input()s', function()
|
|
set_color_cb('ReturningGlobal', { { 0, 1, 'RBP1' } }, '')
|
|
start_prompt('1')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP1:1}^ |
|
|
]])
|
|
|
|
set_color_cb('ReturningGlobal', { { 0, 1, 'RBP2' } }, '1')
|
|
start_prompt('2')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP2:2}^ |
|
|
]])
|
|
|
|
set_color_cb('ReturningGlobal', { { 0, 1, 'RBP3' } }, '2')
|
|
start_prompt('3')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP3:3}^ |
|
|
]])
|
|
|
|
set_color_cb('ReturningGlobal', { { 0, 1, 'RBP4' } }, '3')
|
|
start_prompt('4')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP4:4}^ |
|
|
]])
|
|
|
|
feed('<CR>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP3:3}4^ |
|
|
]])
|
|
feed('<CR>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP2:2}34^ |
|
|
]])
|
|
feed('<CR>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:{RBP1:1}234^ |
|
|
]])
|
|
feed('<CR><CR><C-l>')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
|
|
|
]])
|
|
eq('1234', api.nvim_get_var('out'))
|
|
eq('234', api.nvim_get_var('out1'))
|
|
eq('34', api.nvim_get_var('out2'))
|
|
eq('4', api.nvim_get_var('out3'))
|
|
eq(0, fn.exists('g:out4'))
|
|
end)
|
|
it('runs callback with the same data only once', function()
|
|
local function new_recording_calls(...)
|
|
eq({ ... }, api.nvim_get_var('recording_calls'))
|
|
api.nvim_set_var('recording_calls', {})
|
|
end
|
|
set_color_cb('Recording')
|
|
start_prompt('')
|
|
-- Regression test. Disambiguation:
|
|
--
|
|
-- new_recording_calls(expected_result) -- (actual_before_fix)
|
|
--
|
|
feed('a')
|
|
new_recording_calls('a') -- ('a', 'a')
|
|
feed('b')
|
|
new_recording_calls('ab') -- ('a', 'ab', 'ab')
|
|
feed('c')
|
|
new_recording_calls('abc') -- ('ab', 'abc', 'abc')
|
|
feed('<BS>')
|
|
new_recording_calls('ab') -- ('abc', 'ab', 'ab')
|
|
feed('<BS>')
|
|
new_recording_calls('a') -- ('ab', 'a', 'a')
|
|
feed('<BS>')
|
|
new_recording_calls() -- ('a')
|
|
feed('<CR><CR>')
|
|
eq('', api.nvim_get_var('out'))
|
|
end)
|
|
it('does not crash when callback has caught not-a-editor-command exception', function()
|
|
source([[
|
|
function CaughtExc(cmdline) abort
|
|
try
|
|
gibberish
|
|
catch
|
|
" Do nothing
|
|
endtry
|
|
return []
|
|
endfunction
|
|
]])
|
|
set_color_cb('CaughtExc')
|
|
start_prompt('1')
|
|
eq(1, api.nvim_eval('1'))
|
|
end)
|
|
end)
|
|
describe('Ex commands coloring', function()
|
|
it('works', function()
|
|
api.nvim_set_var('Nvim_color_cmdline', 'RainBowParens')
|
|
feed(':echo (((1)))')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:echo {RBP1:(}{RBP2:(}{RBP3:(}1{RBP3:)}{RBP2:)}{RBP1:)}^ |
|
|
]])
|
|
end)
|
|
it('still executes command-line even if errored out', function()
|
|
api.nvim_set_var('Nvim_color_cmdline', 'SplitMultibyteStart')
|
|
feed(':let x = "«"\n')
|
|
eq('«', api.nvim_get_var('x'))
|
|
local msg = 'E5405: Chunk 0 start 10 splits multibyte character'
|
|
eq('\n' .. msg, fn.execute('messages'))
|
|
end)
|
|
it('does not error out when called from a errorred out cycle', function()
|
|
-- Apparently when there is a cycle in which one of the commands errors out
|
|
-- this error may be caught by color_cmdline before it is presented to the
|
|
-- user.
|
|
feed(dedent([[
|
|
:set regexpengine=2
|
|
:for pat in [' \ze*', ' \zs*']
|
|
: try
|
|
: let l = matchlist('x x', pat)
|
|
: $put ='E888 NOT detected for ' . pat
|
|
: catch
|
|
: $put ='E888 detected for ' . pat
|
|
: endtry
|
|
:endfor
|
|
]]))
|
|
eq(
|
|
{ '', 'E888 detected for \\ze*', 'E888 detected for \\zs*' },
|
|
api.nvim_buf_get_lines(0, 0, -1, false)
|
|
)
|
|
eq('', fn.execute('messages'))
|
|
end)
|
|
it('does not crash when using `n` in debug mode', function()
|
|
feed(':debug execute "echo 1"\n')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
Entering Debug mode. Type "cont" to con|
|
|
tinue. |
|
|
cmd: execute "echo 1" |
|
|
>^ |
|
|
]])
|
|
feed('n\n')
|
|
screen:expect([[
|
|
|
|
|
{MSEP: }|
|
|
Entering Debug mode. Type "cont" to con|
|
|
tinue. |
|
|
cmd: execute "echo 1" |
|
|
>n |
|
|
1 |
|
|
{PE:Press ENTER or type command to continue}^ |
|
|
]])
|
|
feed('\n')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
|
|
|
]])
|
|
end)
|
|
it('mapping error does not cancel prompt', function()
|
|
command("cnoremap <expr> x execute('throw 42')[-1]")
|
|
feed(':#x')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*2
|
|
{MSEP: }|
|
|
:# |
|
|
{ERR:Error detected while processing :} |
|
|
{ERR:E605: Exception not caught: 42} |
|
|
:#^ |
|
|
]])
|
|
feed('<CR>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|
|
|
{MSEP: }|
|
|
:# |
|
|
{ERR:Error detected while processing :} |
|
|
{ERR:E605: Exception not caught: 42} |
|
|
{ERR:E749: Empty buffer} |
|
|
{PE:Press ENTER or type command to continue}^ |
|
|
]])
|
|
feed('<CR>')
|
|
eq(
|
|
'Error detected while processing :\nE605: Exception not caught: 42\nE749: Empty buffer',
|
|
exec_capture('messages')
|
|
)
|
|
end)
|
|
it('errors out when failing to get callback', function()
|
|
api.nvim_set_var('Nvim_color_cmdline', 42)
|
|
feed(':#')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|
|
|
{MSEP: }|
|
|
: |
|
|
{ERR:E5408: Unable to get g:Nvim_color_cmdlin}|
|
|
{ERR:e callback: Vim:E6000: Argument is not a}|
|
|
{ERR: function or function name} |
|
|
:#^ |
|
|
]])
|
|
end)
|
|
end)
|
|
describe('Expressions coloring support', function()
|
|
it('works', function()
|
|
command('hi clear NvimNumber')
|
|
command('hi clear NvimNestingParenthesis')
|
|
command('hi NvimNumber guifg=Blue2')
|
|
command('hi NvimNestingParenthesis guifg=Yellow')
|
|
feed(':echo <C-r>=(((1)))')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
={NPAR:(((}{NUM:1}{NPAR:)))}^ |
|
|
]])
|
|
end)
|
|
it('does not use Nvim_color_expr', function()
|
|
api.nvim_set_var('Nvim_color_expr', 42)
|
|
-- Used to error out due to failing to get callback.
|
|
command('hi clear NvimNumber')
|
|
command('hi NvimNumber guifg=Blue2')
|
|
feed(':<C-r>=1')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
={NUM:1}^ |
|
|
]])
|
|
end)
|
|
it('works correctly with non-ASCII and control characters', function()
|
|
command('hi clear NvimStringBody')
|
|
command('hi clear NvimStringQuote')
|
|
command('hi clear NvimInvalid')
|
|
command('hi NvimStringQuote guifg=Blue3')
|
|
command('hi NvimStringBody guifg=Blue4')
|
|
command('hi NvimInvalid guifg=Red guibg=Blue')
|
|
feed('i<C-r>="«»"«»')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
={SQ:"}{SB:«»}{SQ:"}{E:«»}^ |
|
|
]])
|
|
feed('<C-c>')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
{M:-- INSERT --} |
|
|
]])
|
|
feed('<Esc>')
|
|
screen:expect([[
|
|
^ |
|
|
{EOB:~ }|*6
|
|
|
|
|
]])
|
|
feed(':<C-\\>e"<C-v><C-x>"<C-v><C-x>')
|
|
-- TODO(ZyX-I): Parser highlighting should not override special character
|
|
-- highlighting.
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
={SQ:"}{SB:^X}{SQ:"}{ERR:^X}^ |
|
|
]])
|
|
feed('<C-c>')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
:^ |
|
|
]])
|
|
fn.setreg('a', { '\192' })
|
|
feed('<C-r>="<C-r><C-r>a"<C-r><C-r>a"foo"')
|
|
screen:expect([[
|
|
|
|
|
{EOB:~ }|*6
|
|
={SQ:"}{SB:<c0>}{SQ:"}{E:<c0>"}{SB:foo}{E:"}^ |
|
|
]])
|
|
end)
|
|
end)
|