123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416 |
- -- 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 command = n.command
- local api = n.api
- local fn = n.fn
- local clear = n.clear
- local eq = t.eq
- local fail = t.fail
- local exec_lua = n.exec_lua
- local feed = n.feed
- local expect_events = t.expect_events
- local write_file = t.write_file
- local dedent = t.dedent
- local origlines = {
- 'original line 1',
- 'original line 2',
- 'original line 3',
- 'original line 4',
- 'original line 5',
- 'original line 6',
- ' indented line',
- }
- before_each(function()
- clear()
- exec_lua(function()
- local events = {}
- function _G.test_register(bufnr, evname, id, changedtick, utf_sizes, preview)
- local function callback(...)
- table.insert(events, { id, ... })
- if _G.test_unreg == id then
- return true
- end
- end
- local opts = {
- [evname] = callback,
- on_detach = callback,
- on_reload = callback,
- utf_sizes = utf_sizes,
- preview = preview,
- }
- if changedtick then
- opts.on_changedtick = callback
- end
- vim.api.nvim_buf_attach(bufnr, false, opts)
- end
- function _G.get_events()
- local ret_events = events
- events = {}
- return ret_events
- end
- end)
- end)
- describe('lua buffer event callbacks: on_lines', function()
- local function setup_eventcheck(verify, utf_sizes, lines)
- local lastsize
- api.nvim_buf_set_lines(0, 0, -1, true, lines)
- if verify then
- lastsize = api.nvim_buf_get_offset(0, api.nvim_buf_line_count(0))
- end
- exec_lua('return test_register(...)', 0, 'on_lines', 'test1', false, utf_sizes)
- local verify_name = 'test1'
- local function check_events(expected)
- local events = exec_lua('return get_events(...)')
- if utf_sizes then
- -- this test case uses ASCII only, so sizes should be the same.
- -- Unicode is tested below.
- for _, event in ipairs(expected) do
- event[9] = event[9] or event[8]
- event[10] = event[10] or event[9]
- end
- end
- expect_events(expected, events, 'line updates')
- if verify then
- for _, event in ipairs(events) do
- if event[1] == verify_name and event[2] == 'lines' then
- local startline, endline = event[5], event[7]
- local newrange = api.nvim_buf_get_offset(0, endline)
- - api.nvim_buf_get_offset(0, startline)
- local newsize = api.nvim_buf_get_offset(0, api.nvim_buf_line_count(0))
- local oldrange = newrange + lastsize - newsize
- eq(oldrange, event[8])
- lastsize = newsize
- end
- end
- end
- end
- return check_events, function(new)
- verify_name = new
- end
- end
- -- verifying the sizes with nvim_buf_get_offset is nice (checks we cannot
- -- assert the wrong thing), but masks errors with unflushed lines (as
- -- nvim_buf_get_offset forces a flush of the memline). To be safe run the
- -- test both ways.
- local function check(verify, utf_sizes)
- local check_events, verify_name = setup_eventcheck(verify, utf_sizes, origlines)
- local tick = api.nvim_buf_get_changedtick(0)
- command('set autoindent')
- command('normal! GyyggP')
- tick = tick + 1
- check_events { { 'test1', 'lines', 1, tick, 0, 0, 1, 0 } }
- api.nvim_buf_set_lines(0, 3, 5, true, { 'changed line' })
- tick = tick + 1
- check_events { { 'test1', 'lines', 1, tick, 3, 5, 4, 32 } }
- exec_lua('return test_register(...)', 0, 'on_lines', 'test2', true, utf_sizes)
- tick = tick + 1
- command('undo')
- -- plugins can opt in to receive changedtick events, or choose
- -- to only receive actual changes.
- check_events {
- { 'test1', 'lines', 1, tick, 3, 4, 5, 13 },
- { 'test2', 'lines', 1, tick, 3, 4, 5, 13 },
- { 'test2', 'changedtick', 1, tick + 1 },
- }
- tick = tick + 1
- tick = tick + 1
- command('redo')
- check_events {
- { 'test1', 'lines', 1, tick, 3, 5, 4, 32 },
- { 'test2', 'lines', 1, tick, 3, 5, 4, 32 },
- { 'test2', 'changedtick', 1, tick + 1 },
- }
- tick = tick + 1
- tick = tick + 1
- command('undo!')
- check_events {
- { 'test1', 'lines', 1, tick, 3, 4, 5, 13 },
- { 'test2', 'lines', 1, tick, 3, 4, 5, 13 },
- { 'test2', 'changedtick', 1, tick + 1 },
- }
- tick = tick + 1
- -- simulate next callback returning true
- exec_lua("test_unreg = 'test1'")
- api.nvim_buf_set_lines(0, 6, 7, true, { 'x1', 'x2', 'x3' })
- tick = tick + 1
- -- plugins can opt in to receive changedtick events, or choose
- -- to only receive actual changes.
- check_events {
- { 'test1', 'lines', 1, tick, 6, 7, 9, 16 },
- { 'test2', 'lines', 1, tick, 6, 7, 9, 16 },
- }
- verify_name 'test2'
- api.nvim_buf_set_lines(0, 1, 1, true, { 'added' })
- tick = tick + 1
- check_events { { 'test2', 'lines', 1, tick, 1, 1, 2, 0 } }
- feed('wix')
- tick = tick + 1
- check_events { { 'test2', 'lines', 1, tick, 4, 5, 5, 16 } }
- -- check hot path for multiple insert
- feed('yz')
- tick = tick + 1
- check_events { { 'test2', 'lines', 1, tick, 4, 5, 5, 17 } }
- feed('<bs>')
- tick = tick + 1
- check_events { { 'test2', 'lines', 1, tick, 4, 5, 5, 19 } }
- feed('<esc>Go')
- tick = tick + 1
- check_events { { 'test2', 'lines', 1, tick, 11, 11, 12, 0 } }
- feed('x')
- tick = tick + 1
- check_events { { 'test2', 'lines', 1, tick, 11, 12, 12, 5 } }
- command('bwipe!')
- check_events { { 'test2', 'detach', 1 } }
- end
- it('works', function()
- check(false)
- end)
- it('works with verify', function()
- check(true)
- end)
- it('works with utf_sizes and ASCII text', function()
- check(false, true)
- end)
- local function check_unicode(verify)
- local unicode_text = {
- 'ascii text',
- 'latin text åäö',
- 'BMP text ɧ αλφά',
- 'BMP text 汉语 ↥↧',
- 'SMP 🤦 🦄🦃',
- 'combining å بِيَّة',
- }
- local check_events, verify_name = setup_eventcheck(verify, true, unicode_text)
- local tick = api.nvim_buf_get_changedtick(0)
- feed('ggdd')
- tick = tick + 1
- check_events { { 'test1', 'lines', 1, tick, 0, 1, 0, 11, 11, 11 } }
- feed('A<bs>')
- tick = tick + 1
- check_events { { 'test1', 'lines', 1, tick, 0, 1, 1, 18, 15, 15 } }
- feed('<esc>jylp')
- tick = tick + 1
- check_events { { 'test1', 'lines', 1, tick, 1, 2, 2, 21, 16, 16 } }
- feed('+eea<cr>')
- tick = tick + 1
- check_events { { 'test1', 'lines', 1, tick, 2, 3, 4, 23, 15, 15 } }
- feed('<esc>jdw')
- tick = tick + 1
- -- non-BMP chars count as 2 UTF-2 codeunits
- check_events { { 'test1', 'lines', 1, tick, 4, 5, 5, 18, 9, 12 } }
- feed('+rx')
- tick = tick + 1
- -- count the individual codepoints of a composed character.
- check_events { { 'test1', 'lines', 1, tick, 5, 6, 6, 27, 20, 20 } }
- feed('kJ')
- tick = tick + 1
- -- verification fails with multiple line updates, sorry about that
- verify_name ''
- -- NB: this is inefficient (but not really wrong).
- check_events {
- { 'test1', 'lines', 1, tick, 4, 5, 5, 14, 5, 8 },
- { 'test1', 'lines', 1, tick + 1, 5, 6, 5, 27, 20, 20 },
- }
- end
- it('works with utf_sizes and unicode text', function()
- check_unicode(false)
- end)
- it('works with utf_sizes and unicode text with verify', function()
- check_unicode(true)
- end)
- it('has valid cursor position while shifting', function()
- api.nvim_buf_set_lines(0, 0, -1, true, { 'line1' })
- exec_lua(function()
- vim.api.nvim_buf_attach(0, false, {
- on_lines = function()
- vim.api.nvim_set_var('listener_cursor_line', vim.api.nvim_win_get_cursor(0)[1])
- end,
- })
- end)
- feed('>>')
- eq(1, api.nvim_get_var('listener_cursor_line'))
- end)
- it('has valid cursor position while deleting lines', function()
- api.nvim_buf_set_lines(0, 0, -1, true, { 'line_1', 'line_2', 'line_3', 'line_4' })
- api.nvim_win_set_cursor(0, { 2, 0 })
- eq(2, api.nvim_win_get_cursor(0)[1])
- api.nvim_buf_set_lines(0, 0, -1, true, { 'line_1', 'line_2', 'line_3' })
- eq(2, api.nvim_win_get_cursor(0)[1])
- end)
- it('does not SEGFAULT when accessing window buffer info in on_detach #14998', function()
- local code = function()
- local buf = vim.api.nvim_create_buf(false, false)
- vim.cmd 'split'
- vim.api.nvim_win_set_buf(0, buf)
- vim.api.nvim_buf_attach(buf, false, {
- on_detach = function(_, buf0)
- vim.fn.tabpagebuflist()
- vim.fn.win_findbuf(buf0)
- end,
- })
- end
- exec_lua(code)
- command('q!')
- n.assert_alive()
- exec_lua(code)
- command('bd!')
- n.assert_alive()
- end)
- it('no invalid lnum error for closed memline in on_detach #31251', function()
- eq(vim.NIL, exec_lua('return _G.did_detach'))
- exec_lua([[
- vim.api.nvim_buf_set_lines(0, 0, -1, false, { '' })
- local bufname = 'buf2'
- local buf = vim.api.nvim_create_buf(false, true)
- vim.api.nvim_buf_set_name(buf, bufname)
- vim.bo[buf].bufhidden = 'wipe'
- vim.cmd('vertical diffsplit '..bufname)
- vim.api.nvim_buf_attach(0, false, { on_detach = function()
- vim.cmd("redraw")
- _G.did_detach = true
- end})
- vim.cmd.bdelete()
- ]])
- eq(true, exec_lua('return _G.did_detach'))
- end)
- it('#12718 lnume', function()
- api.nvim_buf_set_lines(0, 0, -1, true, { '1', '2', '3' })
- exec_lua(function()
- vim.api.nvim_buf_attach(0, false, {
- on_lines = function(...)
- vim.api.nvim_set_var('linesev', { ... })
- end,
- })
- end)
- feed('1G0')
- feed('y<C-v>2j')
- feed('G0')
- feed('p')
- -- Is the last arg old_byte_size correct? Doesn't matter for this PR
- eq({ 'lines', 1, 4, 2, 3, 5, 4 }, api.nvim_get_var('linesev'))
- feed('2G0')
- feed('p')
- eq({ 'lines', 1, 5, 1, 4, 4, 8 }, api.nvim_get_var('linesev'))
- feed('1G0')
- feed('P')
- eq({ 'lines', 1, 6, 0, 3, 3, 9 }, api.nvim_get_var('linesev'))
- end)
- it('nvim_buf_call() from callback does not cause wrong Normal mode CTRL-A #16729', function()
- exec_lua(function()
- vim.api.nvim_buf_attach(0, false, {
- on_lines = function()
- vim.api.nvim_buf_call(0, function() end)
- end,
- })
- end)
- feed('itest123<Esc><C-A>')
- eq('test124', api.nvim_get_current_line())
- end)
- it('setting extmark in on_lines callback works', function()
- local screen = Screen.new(40, 6)
- api.nvim_buf_set_lines(0, 0, -1, true, { 'aaa', 'bbb', 'ccc' })
- exec_lua(function()
- local ns = vim.api.nvim_create_namespace('')
- vim.api.nvim_buf_attach(0, false, {
- on_lines = function(_, _, _, row, _, end_row)
- vim.api.nvim_buf_clear_namespace(0, ns, row, end_row)
- for i = row, end_row - 1 do
- vim.api.nvim_buf_set_extmark(0, ns, i, 0, {
- virt_text = { { 'NEW' .. tostring(i), 'WarningMsg' } },
- })
- end
- end,
- })
- end)
- feed('o')
- screen:expect({
- grid = [[
- aaa |
- ^ {19:NEW1} |
- bbb |
- ccc |
- {1:~ }|
- {5:-- INSERT --} |
- ]],
- })
- feed('<CR>')
- screen:expect({
- grid = [[
- aaa |
- {19:NEW1} |
- ^ {19:NEW2} |
- bbb |
- ccc |
- {5:-- INSERT --} |
- ]],
- })
- end)
- it('line lengths are correct when pressing TAB with folding #29119', function()
- api.nvim_buf_set_lines(0, 0, -1, true, { 'a', 'b' })
- exec_lua(function()
- _G.res = {}
- vim.o.foldmethod = 'indent'
- vim.o.softtabstop = -1
- vim.api.nvim_buf_attach(0, false, {
- on_lines = function(_, bufnr, _, row, _, end_row)
- local lines = vim.api.nvim_buf_get_lines(bufnr, row, end_row, true)
- table.insert(_G.res, lines)
- end,
- })
- end)
- feed('i<Tab>')
- eq({ '\ta' }, exec_lua('return _G.res[#_G.res]'))
- end)
- it('quickfix buffer send change', function()
- command('copen')
- exec_lua(function()
- vim.api.nvim_buf_attach(vim.api.nvim_get_current_buf(), false, {
- on_lines = function(...)
- vim.g.qf_on_lines = { ... }
- end,
- on_bytes = function(...)
- vim.g.qf_on_bytes = { ... }
- end,
- })
- end)
- command('caddexpr "foo"')
- eq({ 'bytes', 2, 2, 0, 0, 0, 0, 0, 0, 0, 6, 6 }, api.nvim_get_var('qf_on_bytes'))
- eq({ 'lines', 2, 3, 0, 1, 1, 1 }, api.nvim_get_var('qf_on_lines'))
- command('caddexpr "bar"')
- eq({ 'bytes', 2, 3, 0, 6, 6, 0, 0, 0, 1, 6, 6 }, api.nvim_get_var('qf_on_bytes'))
- eq({ 'lines', 2, 4, 1, 1, 2, 0 }, api.nvim_get_var('qf_on_lines'))
- command('caddexpr ["line1", "line2", "line3"]')
- eq({ 'bytes', 2, 4, 1, 6, 13, 0, 0, 0, 3, 8, 26 }, api.nvim_get_var('qf_on_bytes'))
- eq({ 'lines', 2, 5, 2, 2, 5, 0 }, api.nvim_get_var('qf_on_lines'))
- command('cexpr "replace"')
- eq({ 'bytes', 2, 5, 0, 0, 0, 4, 0, 40, 0, 10, 10 }, api.nvim_get_var('qf_on_bytes'))
- eq({ 'lines', 2, 6, 0, 5, 1, 42 }, api.nvim_get_var('qf_on_lines'))
- end)
- end)
- describe('lua: nvim_buf_attach on_bytes', function()
- -- verifying the sizes with nvim_buf_get_offset is nice (checks we cannot
- -- assert the wrong thing), but masks errors with unflushed lines (as
- -- nvim_buf_get_offset forces a flush of the memline). To be safe run the
- -- test both ways.
- local function setup_eventcheck(verify, start_txt)
- if start_txt then
- api.nvim_buf_set_lines(0, 0, -1, true, start_txt)
- else
- start_txt = api.nvim_buf_get_lines(0, 0, -1, true)
- end
- local shadowbytes = table.concat(start_txt, '\n') .. '\n'
- -- TODO: while we are brewing the real strong coffee,
- -- verify should check buf_get_offset after every check_events
- if verify then
- local len = api.nvim_buf_get_offset(0, api.nvim_buf_line_count(0))
- eq(len == -1 and 1 or len, string.len(shadowbytes))
- end
- exec_lua('return test_register(...)', 0, 'on_bytes', 'test1', false, false, true)
- api.nvim_buf_get_changedtick(0)
- local verify_name = 'test1'
- local function check_events(expected)
- local events = exec_lua('return get_events(...)')
- expect_events(expected, events, 'byte updates')
- if not verify then
- return
- end
- for _, event in ipairs(events) do
- for _, elem in ipairs(event) do
- if type(elem) == 'number' and elem < 0 then
- fail(string.format('Received event has negative values'))
- end
- end
- if event[1] == verify_name and event[2] == 'bytes' then
- local _, _, _, _, _, _, start_byte, _, _, old_byte, _, _, new_byte = unpack(event)
- local before = string.sub(shadowbytes, 1, start_byte)
- -- no text in the tests will contain 0xff bytes (invalid UTF-8)
- -- so we can use it as marker for unknown bytes
- local unknown = string.rep('\255', new_byte)
- local after = string.sub(shadowbytes, start_byte + old_byte + 1)
- shadowbytes = before .. unknown .. after
- elseif event[1] == verify_name and event[2] == 'reload' then
- shadowbytes = table.concat(api.nvim_buf_get_lines(0, 0, -1, true), '\n') .. '\n'
- end
- end
- local text = api.nvim_buf_get_lines(0, 0, -1, true)
- local bytes = table.concat(text, '\n') .. '\n'
- eq(
- string.len(bytes),
- string.len(shadowbytes),
- '\non_bytes: total bytecount of buffer is wrong'
- )
- for i = 1, string.len(shadowbytes) do
- local shadowbyte = string.sub(shadowbytes, i, i)
- if shadowbyte ~= '\255' then
- eq(string.sub(bytes, i, i), shadowbyte, i)
- end
- end
- end
- return check_events
- end
- -- Yes, we can do both
- local function do_both(verify)
- it('single and multiple join', function()
- local check_events = setup_eventcheck(verify, origlines)
- feed 'ggJ'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 15, 15, 1, 0, 1, 0, 1, 1 },
- }
- feed '3J'
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 31, 31, 1, 0, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 5, 0, 47, 47, 1, 0, 1, 0, 1, 1 },
- }
- end)
- it('opening lines', function()
- local check_events = setup_eventcheck(verify, origlines)
- api.nvim_set_option_value('autoindent', false, {})
- feed 'Go'
- check_events {
- { 'test1', 'bytes', 1, 3, 7, 0, 114, 0, 0, 0, 1, 0, 1 },
- }
- feed '<cr>'
- check_events {
- { 'test1', 'bytes', 1, 4, 7, 0, 114, 0, 0, 0, 1, 0, 1 },
- }
- end)
- it('opening lines with autoindent', function()
- local check_events = setup_eventcheck(verify, origlines)
- api.nvim_set_option_value('autoindent', true, {})
- feed 'Go'
- check_events {
- { 'test1', 'bytes', 1, 3, 7, 0, 114, 0, 0, 0, 1, 0, 5 },
- }
- feed '<cr>'
- check_events {
- { 'test1', 'bytes', 1, 4, 7, 0, 114, 0, 4, 4, 0, 0, 0 },
- { 'test1', 'bytes', 1, 4, 7, 0, 114, 0, 0, 0, 1, 4, 5 },
- }
- end)
- it('setline(num, line)', function()
- local check_events = setup_eventcheck(verify, origlines)
- fn.setline(2, 'babla')
- check_events {
- { 'test1', 'bytes', 1, 3, 1, 0, 16, 0, 15, 15, 0, 5, 5 },
- }
- fn.setline(2, { 'foo', 'bar' })
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 0, 16, 0, 5, 5, 0, 3, 3 },
- { 'test1', 'bytes', 1, 5, 2, 0, 20, 0, 15, 15, 0, 3, 3 },
- }
- local buf_len = api.nvim_buf_line_count(0)
- fn.setline(buf_len + 1, 'baz')
- check_events {
- { 'test1', 'bytes', 1, 6, 7, 0, 90, 0, 0, 0, 1, 0, 4 },
- }
- end)
- it('continuing comments with fo=or', function()
- local check_events = setup_eventcheck(verify, { '// Comment' })
- api.nvim_set_option_value('formatoptions', 'ro', {})
- api.nvim_set_option_value('filetype', 'c', {})
- feed 'A<CR>'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 10, 10, 0, 0, 0, 1, 3, 4 },
- }
- feed '<ESC>'
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 2, 13, 0, 1, 1, 0, 0, 0 },
- }
- feed 'ggo' -- goto first line to continue testing
- check_events {
- { 'test1', 'bytes', 1, 5, 1, 0, 11, 0, 0, 0, 1, 0, 4 },
- }
- feed '<CR>'
- check_events {
- { 'test1', 'bytes', 1, 6, 1, 2, 13, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 6, 1, 2, 13, 0, 0, 0, 1, 3, 4 },
- }
- end)
- it('editing empty buffers', function()
- local check_events = setup_eventcheck(verify, {})
- feed 'ia'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 0, 0, 0, 1, 1 },
- }
- end)
- it('deleting lines', function()
- local check_events = setup_eventcheck(verify, origlines)
- feed('dd')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 1, 0, 16, 0, 0, 0 },
- }
- feed('d2j')
- check_events {
- { 'test1', 'bytes', 1, 4, 0, 0, 0, 3, 0, 48, 0, 0, 0 },
- }
- feed('ld<c-v>2j')
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 5, 1, 1, 16, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 5, 2, 1, 31, 0, 1, 1, 0, 0, 0 },
- }
- feed('vjwd')
- check_events {
- { 'test1', 'bytes', 1, 10, 0, 1, 1, 1, 9, 23, 0, 0, 0 },
- }
- end)
- it('changing lines', function()
- local check_events = setup_eventcheck(verify, origlines)
- feed 'cc'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 15, 15, 0, 0, 0 },
- }
- feed '<ESC>'
- check_events {}
- feed 'c3j'
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 0, 1, 3, 0, 48, 0, 0, 0 },
- }
- end)
- it('visual charwise paste', function()
- local check_events = setup_eventcheck(verify, { '1234567890' })
- fn.setreg('a', '___')
- feed '1G1|vll'
- check_events {}
- feed '"ap'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 3, 3, 0, 0, 0 },
- { 'test1', 'bytes', 1, 5, 0, 0, 0, 0, 0, 0, 0, 3, 3 },
- }
- end)
- it('blockwise paste', function()
- local check_events = setup_eventcheck(verify, { '1', '2', '3' })
- feed('1G0')
- feed('y<C-v>2j')
- feed('G0')
- feed('p')
- check_events {
- { 'test1', 'bytes', 1, 3, 2, 1, 5, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 3, 3, 0, 7, 0, 0, 0, 0, 3, 3 },
- { 'test1', 'bytes', 1, 3, 4, 0, 10, 0, 0, 0, 0, 3, 3 },
- }
- feed('2G0')
- feed('p')
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 1, 3, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 4, 2, 1, 6, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 4, 3, 1, 10, 0, 0, 0, 0, 1, 1 },
- }
- feed('1G0')
- feed('P')
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 0, 0, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 5, 1, 0, 3, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 5, 2, 0, 7, 0, 0, 0, 0, 1, 1 },
- }
- end)
- it('linewise paste', function()
- local check_events = setup_eventcheck(verify, origlines)
- feed 'yyp'
- check_events {
- { 'test1', 'bytes', 1, 3, 1, 0, 16, 0, 0, 0, 1, 0, 16 },
- }
- feed 'Gyyp'
- check_events {
- { 'test1', 'bytes', 1, 4, 8, 0, 130, 0, 0, 0, 1, 0, 18 },
- }
- end)
- it('inccomand=nosplit and substitute', function()
- local check_events = setup_eventcheck(verify, { 'abcde', '12345' })
- api.nvim_set_option_value('inccommand', 'nosplit', {})
- -- linewise substitute
- feed(':%s/bcd/')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 3, 3, 0, 0, 0 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 0, 0, 0, 3, 3 },
- }
- feed('a')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 3, 3, 0, 1, 1 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 1, 1, 0, 3, 3 },
- }
- feed('<esc>')
- -- splitting lines
- feed([[:%s/abc/\r]])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 3, 3, 1, 0, 1 },
- { 'test1', 'bytes', 1, 6, 0, 0, 0, 1, 0, 1, 0, 3, 3 },
- }
- feed('<esc>')
- -- multi-line regex
- feed([[:%s/de\n123/a]])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 3, 3, 1, 3, 6, 0, 1, 1 },
- { 'test1', 'bytes', 1, 6, 0, 3, 3, 0, 1, 1, 1, 3, 6 },
- }
- feed('<esc>')
- -- replacing with unicode
- feed(':%s/b/→')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 1, 1, 0, 3, 3 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 3, 3, 0, 1, 1 },
- }
- feed('<esc>')
- -- replacing with expression register
- feed([[:%s/b/\=5+5]])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 1, 1, 0, 2, 2 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 2, 2, 0, 1, 1 },
- }
- feed('<esc>')
- -- replacing with backslash
- feed([[:%s/b/\\]])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 1, 1, 0, 1, 1 },
- }
- feed('<esc>')
- -- replacing with backslash from expression register
- feed([[:%s/b/\='\']])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 1, 1, 0, 1, 1 },
- }
- feed('<esc>')
- -- replacing with backslash followed by another character
- feed([[:%s/b/\\!]])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 1, 1, 0, 2, 2 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 2, 2, 0, 1, 1 },
- }
- feed('<esc>')
- -- replacing with backslash followed by another character from expression register
- feed([[:%s/b/\='\!']])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 1, 1, 0, 2, 2 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 2, 2, 0, 1, 1 },
- }
- end)
- it('nvim_buf_set_text insert', function()
- local check_events = setup_eventcheck(verify, { 'bastext' })
- api.nvim_buf_set_text(0, 0, 3, 0, 3, { 'fiol', 'kontra' })
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 3, 3, 0, 0, 0, 1, 6, 11 },
- }
- api.nvim_buf_set_text(0, 1, 6, 1, 6, { 'punkt', 'syntgitarr', 'övnings' })
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 6, 14, 0, 0, 0, 2, 8, 25 },
- }
- eq(
- { 'basfiol', 'kontrapunkt', 'syntgitarr', 'övningstext' },
- api.nvim_buf_get_lines(0, 0, -1, true)
- )
- end)
- it('nvim_buf_set_text replace', function()
- local check_events = setup_eventcheck(verify, origlines)
- api.nvim_buf_set_text(0, 2, 3, 2, 8, { 'very text' })
- check_events {
- { 'test1', 'bytes', 1, 3, 2, 3, 35, 0, 5, 5, 0, 9, 9 },
- }
- api.nvim_buf_set_text(0, 3, 5, 3, 7, { ' splitty', 'line ' })
- check_events {
- { 'test1', 'bytes', 1, 4, 3, 5, 57, 0, 2, 2, 1, 5, 14 },
- }
- api.nvim_buf_set_text(0, 0, 8, 1, 2, { 'JOINY' })
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 8, 8, 1, 2, 10, 0, 5, 5 },
- }
- api.nvim_buf_set_text(0, 4, 0, 6, 0, { 'was 5,6', '' })
- check_events {
- { 'test1', 'bytes', 1, 6, 4, 0, 75, 2, 0, 32, 1, 0, 8 },
- }
- eq({
- 'originalJOINYiginal line 2',
- 'orivery text line 3',
- 'origi splitty',
- 'line l line 4',
- 'was 5,6',
- ' indented line',
- }, api.nvim_buf_get_lines(0, 0, -1, true))
- end)
- it('nvim_buf_set_text delete', function()
- local check_events = setup_eventcheck(verify, origlines)
- -- really {""} but accepts {} as a shorthand
- api.nvim_buf_set_text(0, 0, 0, 1, 0, {})
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 1, 0, 16, 0, 0, 0 },
- }
- -- TODO(bfredl): this works but is not as convenient as set_lines
- api.nvim_buf_set_text(0, 4, 15, 5, 17, { '' })
- check_events {
- { 'test1', 'bytes', 1, 4, 4, 15, 79, 1, 17, 18, 0, 0, 0 },
- }
- eq({
- 'original line 2',
- 'original line 3',
- 'original line 4',
- 'original line 5',
- 'original line 6',
- }, api.nvim_buf_get_lines(0, 0, -1, true))
- end)
- it('checktime autoread', function()
- write_file(
- 'Xtest-reload',
- dedent [[
- old line 1
- old line 2]]
- )
- local atime = os.time() - 10
- vim.uv.fs_utime('Xtest-reload', atime, atime)
- command 'e Xtest-reload'
- command 'set autoread'
- local check_events = setup_eventcheck(verify, nil)
- write_file(
- 'Xtest-reload',
- dedent [[
- new line 1
- new line 2
- new line 3]]
- )
- command 'checktime'
- check_events {
- { 'test1', 'reload', 1 },
- }
- feed 'ggJ'
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 10, 10, 1, 0, 1, 0, 1, 1 },
- }
- eq({ 'new line 1 new line 2', 'new line 3' }, api.nvim_buf_get_lines(0, 0, -1, true))
- -- check we can undo and redo a reload event.
- feed 'u'
- check_events {
- { 'test1', 'bytes', 1, 8, 0, 10, 10, 0, 1, 1, 1, 0, 1 },
- }
- feed 'u'
- check_events {
- { 'test1', 'reload', 1 },
- }
- feed '<c-r>'
- check_events {
- { 'test1', 'reload', 1 },
- }
- feed '<c-r>'
- check_events {
- { 'test1', 'bytes', 1, 14, 0, 10, 10, 1, 0, 1, 0, 1, 1 },
- }
- end)
- it('tab with noexpandtab and softtabstop', function()
- command('set noet')
- command('set ts=4')
- command('set sw=2')
- command('set sts=4')
- local check_events = setup_eventcheck(verify, { 'asdfasdf' })
- feed('gg0i<tab>')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 4, 0, 1, 1, 0, 0, 0, 0, 1, 1 },
- }
- feed('<tab>')
- -- when spaces are merged into a tabstop
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 2, 2, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 6, 0, 3, 3, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 7, 0, 0, 0, 0, 4, 4, 0, 1, 1 },
- }
- feed('<esc>u')
- check_events {
- { 'test1', 'bytes', 1, 9, 0, 0, 0, 0, 1, 1, 0, 4, 4 },
- { 'test1', 'bytes', 1, 9, 0, 0, 0, 0, 4, 4, 0, 0, 0 },
- }
- -- in REPLACE mode
- feed('R<tab><tab>')
- check_events {
- { 'test1', 'bytes', 1, 10, 0, 0, 0, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 11, 0, 1, 1, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 12, 0, 2, 2, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 13, 0, 3, 3, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 14, 0, 0, 0, 0, 4, 4, 0, 1, 1 },
- }
- feed('<esc>u')
- check_events {
- { 'test1', 'bytes', 1, 16, 0, 0, 0, 0, 1, 1, 0, 4, 4 },
- { 'test1', 'bytes', 1, 16, 0, 2, 2, 0, 2, 2, 0, 1, 1 },
- { 'test1', 'bytes', 1, 16, 0, 0, 0, 0, 2, 2, 0, 1, 1 },
- }
- -- in VISUALREPLACE mode
- feed('gR<tab><tab>')
- check_events {
- { 'test1', 'bytes', 1, 17, 0, 0, 0, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 18, 0, 1, 1, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 19, 0, 2, 2, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 20, 0, 3, 3, 0, 1, 1, 0, 1, 1 },
- { 'test1', 'bytes', 1, 21, 0, 3, 3, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 22, 0, 3, 3, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 24, 0, 2, 2, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 25, 0, 2, 2, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 27, 0, 1, 1, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 28, 0, 1, 1, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 30, 0, 0, 0, 0, 1, 1, 0, 0, 0 },
- { 'test1', 'bytes', 1, 31, 0, 0, 0, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 33, 0, 0, 0, 0, 4, 4, 0, 1, 1 },
- }
- -- inserting tab after other tabs
- command('set sw=4')
- feed('<esc>0a<tab>')
- check_events {
- { 'test1', 'bytes', 1, 34, 0, 1, 1, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 35, 0, 2, 2, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 36, 0, 3, 3, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 37, 0, 4, 4, 0, 0, 0, 0, 1, 1 },
- { 'test1', 'bytes', 1, 38, 0, 1, 1, 0, 4, 4, 0, 1, 1 },
- }
- end)
- it('retab', function()
- command('set noet')
- command('set ts=4')
- local check_events = setup_eventcheck(verify, { ' asdf' })
- command('retab 8')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 7, 7, 0, 9, 9 },
- }
- end)
- it('sends events when undoing with undofile', function()
- write_file(
- 'Xtest-undofile',
- dedent([[
- 12345
- hello world
- ]])
- )
- command('e! Xtest-undofile')
- command('set undodir=. | set undofile')
- local ns = n.request('nvim_create_namespace', 'ns1')
- api.nvim_buf_set_extmark(0, ns, 0, 0, {})
- eq({ '12345', 'hello world' }, api.nvim_buf_get_lines(0, 0, -1, true))
- -- splice
- feed('gg0d2l')
- eq({ '345', 'hello world' }, api.nvim_buf_get_lines(0, 0, -1, true))
- -- move
- command('.m+1')
- eq({ 'hello world', '345' }, api.nvim_buf_get_lines(0, 0, -1, true))
- -- reload undofile and undo changes
- command('w')
- command('set noundofile')
- command('bw!')
- command('e! Xtest-undofile')
- command('set undofile')
- local check_events = setup_eventcheck(verify, nil)
- feed('u')
- eq({ '345', 'hello world' }, api.nvim_buf_get_lines(0, 0, -1, true))
- check_events {
- { 'test1', 'bytes', 2, 6, 1, 0, 12, 1, 0, 4, 0, 0, 0 },
- { 'test1', 'bytes', 2, 6, 0, 0, 0, 0, 0, 0, 1, 0, 4 },
- }
- feed('u')
- eq({ '12345', 'hello world' }, api.nvim_buf_get_lines(0, 0, -1, true))
- check_events {
- { 'test1', 'bytes', 2, 8, 0, 0, 0, 0, 0, 0, 0, 2, 2 },
- }
- command('bw!')
- end)
- it('blockwise paste with uneven line lengths', function()
- local check_events = setup_eventcheck(verify, { 'aaaa', 'aaa', 'aaa' })
- -- eq({}, api.nvim_buf_get_lines(0, 0, -1, true))
- feed('gg0<c-v>jj$d')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 4, 4, 0, 0, 0 },
- { 'test1', 'bytes', 1, 3, 1, 0, 1, 0, 3, 3, 0, 0, 0 },
- { 'test1', 'bytes', 1, 3, 2, 0, 2, 0, 3, 3, 0, 0, 0 },
- }
- feed('p')
- check_events {
- { 'test1', 'bytes', 1, 4, 0, 0, 0, 0, 0, 0, 0, 4, 4 },
- { 'test1', 'bytes', 1, 4, 1, 0, 5, 0, 0, 0, 0, 3, 3 },
- { 'test1', 'bytes', 1, 4, 2, 0, 9, 0, 0, 0, 0, 3, 3 },
- }
- end)
- it(':luado', function()
- local check_events = setup_eventcheck(verify, { 'abc', '12345' })
- command(".luado return 'a'")
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 3, 3, 0, 1, 1 },
- }
- command('luado return 10')
- check_events {
- { 'test1', 'bytes', 1, 4, 0, 0, 0, 0, 1, 1, 0, 2, 2 },
- { 'test1', 'bytes', 1, 5, 1, 0, 3, 0, 5, 5, 0, 2, 2 },
- }
- end)
- it('flushes deleted bytes on move', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB', 'CCC', 'DDD' })
- feed(':.move+1<cr>')
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 0, 0, 1, 0, 4, 0, 0, 0 },
- { 'test1', 'bytes', 1, 5, 1, 0, 4, 0, 0, 0, 1, 0, 4 },
- }
- feed('jd2j')
- check_events {
- { 'test1', 'bytes', 1, 6, 2, 0, 8, 2, 0, 8, 0, 0, 0 },
- }
- end)
- it('virtual edit', function()
- local check_events = setup_eventcheck(verify, { '', ' ' })
- api.nvim_set_option_value('virtualedit', 'all', {})
- feed [[<Right><Right>iab<ESC>]]
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 0, 0, 0, 2, 2 },
- { 'test1', 'bytes', 1, 4, 0, 2, 2, 0, 0, 0, 0, 2, 2 },
- }
- feed [[j<Right><Right>iab<ESC>]]
- check_events {
- { 'test1', 'bytes', 1, 5, 1, 0, 5, 0, 1, 1, 0, 8, 8 },
- { 'test1', 'bytes', 1, 6, 1, 5, 10, 0, 0, 0, 0, 2, 2 },
- }
- end)
- it('block visual paste', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB', 'CCC', 'DDD', 'EEE', 'FFF' })
- fn.setreg('a', '___')
- feed([[gg0l<c-v>3jl"ap]])
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 1, 1, 0, 2, 2, 0, 0, 0 },
- { 'test1', 'bytes', 1, 3, 1, 1, 3, 0, 2, 2, 0, 0, 0 },
- { 'test1', 'bytes', 1, 3, 2, 1, 5, 0, 2, 2, 0, 0, 0 },
- { 'test1', 'bytes', 1, 3, 3, 1, 7, 0, 2, 2, 0, 0, 0 },
- { 'test1', 'bytes', 1, 5, 0, 1, 1, 0, 0, 0, 0, 3, 3 },
- { 'test1', 'bytes', 1, 6, 1, 1, 6, 0, 0, 0, 0, 3, 3 },
- { 'test1', 'bytes', 1, 7, 2, 1, 11, 0, 0, 0, 0, 3, 3 },
- { 'test1', 'bytes', 1, 8, 3, 1, 16, 0, 0, 0, 0, 3, 3 },
- }
- end)
- it('visual paste', function()
- local check_events = setup_eventcheck(verify, { 'aaa {', 'b', '}' })
- -- Setting up
- feed [[jdd]]
- check_events {
- { 'test1', 'bytes', 1, 3, 1, 0, 6, 1, 0, 2, 0, 0, 0 },
- }
- -- Actually testing
- feed [[v%p]]
- check_events {
- { 'test1', 'bytes', 1, 8, 0, 4, 4, 1, 1, 3, 0, 0, 0 },
- { 'test1', 'bytes', 1, 8, 0, 4, 4, 0, 0, 0, 2, 0, 3 },
- }
- end)
- it('visual paste 2: splitting an empty line', function()
- local check_events = setup_eventcheck(verify, { 'abc', '{', 'def', '}' })
- feed('ggyyjjvi{p')
- check_events {
- { 'test1', 'bytes', 1, 6, 2, 0, 6, 1, 0, 4, 0, 0, 0 },
- { 'test1', 'bytes', 1, 6, 2, 0, 6, 0, 0, 0, 2, 0, 5 },
- }
- end)
- it('nvim_buf_set_lines', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB' })
- -- delete
- api.nvim_buf_set_lines(0, 0, 1, true, {})
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 1, 0, 4, 0, 0, 0 },
- }
- -- add
- api.nvim_buf_set_lines(0, 0, 0, true, { 'asdf' })
- check_events {
- { 'test1', 'bytes', 1, 4, 0, 0, 0, 0, 0, 0, 1, 0, 5 },
- }
- -- replace
- api.nvim_buf_set_lines(0, 0, 1, true, { 'asdf', 'fdsa' })
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 0, 0, 1, 0, 5, 2, 0, 10 },
- }
- end)
- it('flushes delbytes on substitute', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB', 'CCC' })
- feed('gg0')
- command('s/AAA/GGG/')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 3, 3, 0, 3, 3 },
- }
- -- check that byte updates for :delete (which uses curbuf->deleted_bytes2)
- -- are correct
- command('delete')
- check_events {
- { 'test1', 'bytes', 1, 4, 0, 0, 0, 1, 0, 4, 0, 0, 0 },
- }
- end)
- it('flushes delbytes on join', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB', 'CCC' })
- feed('gg0J')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 3, 3, 1, 0, 1, 0, 1, 1 },
- }
- command('delete')
- check_events {
- { 'test1', 'bytes', 1, 5, 0, 0, 0, 1, 0, 8, 0, 0, 0 },
- }
- end)
- it('sends updates on U', function()
- feed('ggiAAA<cr>BBB')
- feed('<esc>gg$a CCC')
- local check_events = setup_eventcheck(verify, nil)
- feed('ggU')
- check_events {
- { 'test1', 'bytes', 1, 6, 0, 7, 7, 0, 0, 0, 0, 3, 3 },
- }
- end)
- it('delete in completely empty buffer', function()
- local check_events = setup_eventcheck(verify, nil)
- command 'delete'
- check_events {}
- end)
- it('delete the only line of a buffer', function()
- local check_events = setup_eventcheck(verify, { 'AAA' })
- command 'delete'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 1, 0, 4, 1, 0, 1 },
- }
- end)
- it('delete the last line of a buffer with two lines', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB' })
- command '2delete'
- check_events {
- { 'test1', 'bytes', 1, 3, 1, 0, 4, 1, 0, 4, 0, 0, 0 },
- }
- end)
- it(':sort lines', function()
- local check_events = setup_eventcheck(verify, { 'CCC', 'BBB', 'AAA' })
- command '%sort'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 3, 0, 12, 3, 0, 12 },
- }
- end)
- it('handles already sorted lines', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB', 'CCC' })
- command '%sort'
- check_events {}
- end)
- it('works with accepting spell suggestions', function()
- local check_events = setup_eventcheck(verify, { 'hallo world', 'hallo world' })
- feed('gg0z=4<cr><cr>') -- accepts 'Hello'
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 2, 2, 0, 2, 2 },
- }
- command('spellrepall') -- replaces whole words
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 0, 12, 0, 5, 5, 0, 5, 5 },
- }
- end)
- it('works with :diffput and :diffget', function()
- local check_events = setup_eventcheck(verify, { 'AAA' })
- command('diffthis')
- command('new')
- command('diffthis')
- api.nvim_buf_set_lines(0, 0, -1, true, { 'AAA', 'BBB' })
- feed('G')
- command('diffput')
- check_events {
- { 'test1', 'bytes', 1, 3, 1, 0, 4, 0, 0, 0, 1, 0, 4 },
- }
- api.nvim_buf_set_lines(0, 0, -1, true, { 'AAA', 'CCC' })
- feed('<C-w>pG')
- command('diffget')
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 0, 4, 1, 0, 4, 1, 0, 4 },
- }
- end)
- it('prompt buffer', function()
- local check_events = setup_eventcheck(verify, {})
- api.nvim_set_option_value('buftype', 'prompt', {})
- feed('i')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 0, 0, 0, 0, 2, 2 },
- }
- feed('<CR>')
- check_events {
- { 'test1', 'bytes', 1, 4, 1, 0, 3, 0, 0, 0, 1, 0, 1 },
- { 'test1', 'bytes', 1, 5, 1, 0, 3, 0, 0, 0, 0, 2, 2 },
- }
- feed('<CR>')
- check_events {
- { 'test1', 'bytes', 1, 6, 2, 0, 6, 0, 0, 0, 1, 0, 1 },
- { 'test1', 'bytes', 1, 7, 2, 0, 6, 0, 0, 0, 0, 2, 2 },
- }
- end)
- local function test_lockmarks(mode)
- local description = (mode ~= '') and mode or '(baseline)'
- it('test_lockmarks ' .. description .. ' %delete _', function()
- local check_events = setup_eventcheck(verify, { 'AAA', 'BBB', 'CCC' })
- command(mode .. ' %delete _')
- check_events {
- { 'test1', 'bytes', 1, 3, 0, 0, 0, 3, 0, 12, 1, 0, 1 },
- }
- end)
- it('test_lockmarks ' .. description .. ' append()', function()
- local check_events = setup_eventcheck(verify)
- command(mode .. " call append(0, 'CCC')")
- check_events {
- { 'test1', 'bytes', 1, 2, 0, 0, 0, 0, 0, 0, 1, 0, 4 },
- }
- command(mode .. " call append(1, 'BBBB')")
- check_events {
- { 'test1', 'bytes', 1, 3, 1, 0, 4, 0, 0, 0, 1, 0, 5 },
- }
- command(mode .. " call append(2, '')")
- check_events {
- { 'test1', 'bytes', 1, 4, 2, 0, 9, 0, 0, 0, 1, 0, 1 },
- }
- command(mode .. ' $delete _')
- check_events {
- { 'test1', 'bytes', 1, 5, 3, 0, 10, 1, 0, 1, 0, 0, 0 },
- }
- eq('CCC|BBBB|', table.concat(api.nvim_buf_get_lines(0, 0, -1, true), '|'))
- end)
- end
- -- check that behavior is identical with and without "lockmarks"
- test_lockmarks ''
- test_lockmarks 'lockmarks'
- teardown(function()
- os.remove 'Xtest-reload'
- os.remove 'Xtest-undofile'
- os.remove '.Xtest-undofile.un~'
- end)
- end
- describe('(with verify) handles', function()
- do_both(true)
- end)
- describe('(without verify) handles', function()
- do_both(false)
- end)
- end)
|