mirror of
https://github.com/neovim/neovim.git
synced 2025-02-25 18:55:25 -06:00
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.
163 lines
5.1 KiB
Lua
163 lines
5.1 KiB
Lua
-- Test suite for testing interactions with API bindings
|
|
local t = require('test.testutil')
|
|
local n = require('test.functional.testnvim')()
|
|
local Screen = require('test.functional.ui.screen')
|
|
|
|
local fn = n.fn
|
|
local api = n.api
|
|
local clear = n.clear
|
|
local sleep = vim.uv.sleep
|
|
local feed = n.feed
|
|
local eq = t.eq
|
|
local eval = n.eval
|
|
local matches = t.matches
|
|
local exec_lua = n.exec_lua
|
|
local retry = t.retry
|
|
|
|
before_each(clear)
|
|
|
|
describe('vim.uv', function()
|
|
it('version', function()
|
|
assert(fn.luaeval('vim.uv.version()') >= 72961, 'libuv version too old')
|
|
matches('(%d+)%.(%d+)%.(%d+)', fn.luaeval('vim.uv.version_string()'))
|
|
end)
|
|
|
|
it('timer', function()
|
|
exec_lua('vim.api.nvim_set_var("coroutine_cnt", 0)', {})
|
|
|
|
local code = function()
|
|
local touch = 0
|
|
local function wait(ms)
|
|
local this = coroutine.running()
|
|
assert(this)
|
|
local timer = assert(vim.uv.new_timer())
|
|
timer:start(
|
|
ms,
|
|
0,
|
|
vim.schedule_wrap(function()
|
|
timer:close()
|
|
touch = touch + 1
|
|
coroutine.resume(this)
|
|
touch = touch + 1
|
|
assert(touch == 3)
|
|
vim.api.nvim_set_var('coroutine_cnt_1', touch)
|
|
end)
|
|
)
|
|
coroutine.yield()
|
|
touch = touch + 1
|
|
return touch
|
|
end
|
|
coroutine.wrap(function()
|
|
local touched = wait(10)
|
|
assert(touched == touch)
|
|
vim.api.nvim_set_var('coroutine_cnt', touched)
|
|
end)()
|
|
end
|
|
|
|
eq(0, api.nvim_get_var('coroutine_cnt'))
|
|
exec_lua(code)
|
|
retry(2, nil, function()
|
|
sleep(50)
|
|
eq(2, api.nvim_get_var('coroutine_cnt'))
|
|
end)
|
|
eq(3, api.nvim_get_var('coroutine_cnt_1'))
|
|
end)
|
|
|
|
it('is API safe', function()
|
|
local screen = Screen.new(50, 10)
|
|
screen:set_default_attr_ids({
|
|
[1] = { bold = true, foreground = Screen.colors.Blue1 },
|
|
[2] = { bold = true, reverse = true },
|
|
[3] = { foreground = Screen.colors.Grey100, background = Screen.colors.Red },
|
|
[4] = { bold = true, foreground = Screen.colors.SeaGreen4 },
|
|
[5] = { bold = true },
|
|
})
|
|
|
|
-- deferred API functions are disabled, as their safety can't be guaranteed
|
|
exec_lua([[
|
|
local timer = vim.uv.new_timer()
|
|
timer:start(20, 0, function ()
|
|
_G.is_fast = vim.in_fast_event()
|
|
timer:close()
|
|
vim.api.nvim_set_var("valid", true)
|
|
vim.api.nvim_command("echomsg 'howdy'")
|
|
end)
|
|
]])
|
|
|
|
screen:expect([[
|
|
|
|
|
{2: }|
|
|
{3:Error executing luv callback:} |
|
|
{3:[string "<nvim>"]:5: E5560: nvim_set_var must not }|
|
|
{3:be called in a lua loop callback} |
|
|
{3:stack traceback:} |
|
|
{3: [C]: in function 'nvim_set_var'} |
|
|
{3: [string "<nvim>"]:5: in function <[string }|
|
|
{3:"<nvim>"]:2>} |
|
|
{4:Press ENTER or type command to continue}^ |
|
|
]])
|
|
feed('<cr>')
|
|
eq(false, eval("get(g:, 'valid', v:false)"))
|
|
eq(true, exec_lua('return _G.is_fast'))
|
|
|
|
-- callbacks can be scheduled to be executed in the main event loop
|
|
-- where the entire API is available
|
|
exec_lua(function()
|
|
local timer = assert(vim.uv.new_timer())
|
|
timer:start(
|
|
20,
|
|
0,
|
|
vim.schedule_wrap(function()
|
|
_G.is_fast = vim.in_fast_event()
|
|
timer:close()
|
|
vim.api.nvim_set_var('valid', true)
|
|
vim.api.nvim_command("echomsg 'howdy'")
|
|
end)
|
|
)
|
|
end)
|
|
|
|
screen:expect([[
|
|
^ |
|
|
{1:~ }|*8
|
|
howdy |
|
|
]])
|
|
eq(true, eval("get(g:, 'valid', v:false)"))
|
|
eq(false, exec_lua('return _G.is_fast'))
|
|
|
|
-- fast (not deferred) API functions are allowed to be called directly
|
|
exec_lua(function()
|
|
local timer = assert(vim.uv.new_timer())
|
|
timer:start(20, 0, function()
|
|
timer:close()
|
|
-- input is queued for processing after the callback returns
|
|
vim.api.nvim_input('isneaky')
|
|
_G.mode = vim.api.nvim_get_mode()
|
|
end)
|
|
end)
|
|
screen:expect([[
|
|
sneaky^ |
|
|
{1:~ }|*8
|
|
{5:-- INSERT --} |
|
|
]])
|
|
eq({ blocking = false, mode = 'n' }, exec_lua('return _G.mode'))
|
|
|
|
exec_lua(function()
|
|
local timer = assert(vim.uv.new_timer())
|
|
timer:start(20, 0, function()
|
|
_G.is_fast = vim.in_fast_event()
|
|
timer:close()
|
|
_G.value = vim.fn.has('nvim-0.5')
|
|
_G.unvalue = vim.fn.has('python3')
|
|
end)
|
|
end)
|
|
|
|
screen:expect({ any = [[{3:Vim:E5560: Vimscript function must not be called i}]] })
|
|
feed('<cr>')
|
|
eq({ 1, nil }, exec_lua('return {_G.value, _G.unvalue}'))
|
|
end)
|
|
|
|
it("is equal to require('luv')", function()
|
|
eq(true, exec_lua("return vim.uv == require('luv')"))
|
|
end)
|
|
end)
|