123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915 |
- local t = require('test.testutil')
- local n = require('test.functional.testnvim')()
- local clear = n.clear
- local eq, ok = t.eq, t.ok
- local fn = n.fn
- local api = n.api
- local command, eval, next_msg = n.command, n.eval, n.next_msg
- local nvim_prog = n.nvim_prog
- local pcall_err = t.pcall_err
- local sleep = vim.uv.sleep
- local write_file = t.write_file
- local origlines = {
- 'original line 1',
- 'original line 2',
- 'original line 3',
- 'original line 4',
- 'original line 5',
- 'original line 6',
- }
- local function expectn(name, args)
- -- expect the next message to be the specified notification event
- eq({ 'notification', name, args }, next_msg())
- end
- local function sendkeys(keys)
- api.nvim_input(keys)
- -- Wait for Nvim to fully process pending input before possibly sending
- -- more key presses - otherwise they all pile up in the queue and get
- -- processed at once
- n.poke_eventloop()
- end
- local function open(activate, lines)
- local filename = t.tmpname()
- write_file(filename, table.concat(lines, '\n') .. '\n', true)
- command('edit ' .. filename)
- local b = api.nvim_get_current_buf()
- -- what is the value of b:changedtick?
- local tick = eval('b:changedtick')
- -- Enable buffer events, ensure that the nvim_buf_lines_event messages
- -- arrive as expected
- if activate then
- local firstline = 0
- ok(api.nvim_buf_attach(b, true, {}))
- expectn('nvim_buf_lines_event', { b, tick, firstline, -1, lines, false })
- end
- return b, tick, filename
- end
- local function editoriginal(activate, lines)
- if not lines then
- lines = origlines
- end
- -- load up the file with the correct contents
- clear()
- return open(activate, lines)
- end
- local function reopen(buf, expectedlines)
- ok(api.nvim_buf_detach(buf))
- expectn('nvim_buf_detach_event', { buf })
- -- for some reason the :edit! increments tick by 2
- command('edit!')
- local tick = eval('b:changedtick')
- ok(api.nvim_buf_attach(buf, true, {}))
- local firstline = 0
- expectn('nvim_buf_lines_event', { buf, tick, firstline, -1, expectedlines, false })
- command('normal! gg')
- return tick
- end
- local function reopenwithfolds(b)
- -- discard any changes to the buffer
- local tick = reopen(b, origlines)
- -- use markers for folds, make all folds open by default
- command('setlocal foldmethod=marker foldlevel=20 commentstring=/*%s*/')
- -- add a fold
- command('2,4fold')
- tick = tick + 1
- expectn('nvim_buf_lines_event', {
- b,
- tick,
- 1,
- 4,
- {
- 'original line 2/*{{{*/',
- 'original line 3',
- 'original line 4/*}}}*/',
- },
- false,
- })
- -- make a new fold that wraps lines 1-6
- command('1,6fold')
- tick = tick + 1
- expectn('nvim_buf_lines_event', {
- b,
- tick,
- 0,
- 6,
- {
- 'original line 1/*{{{*/',
- 'original line 2/*{{{*/',
- 'original line 3',
- 'original line 4/*}}}*/',
- 'original line 5',
- 'original line 6/*}}}*/',
- },
- false,
- })
- return tick
- end
- describe('API: buffer events:', function()
- before_each(clear)
- it('when lines are added', function()
- local b, tick = editoriginal(true)
- -- add a new line at the start of the buffer
- command('normal! GyyggP')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 0, { 'original line 6' }, false })
- -- add multiple lines at the start of the file
- command('normal! GkkyGggP')
- tick = tick + 1
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 0, 0, { 'original line 4', 'original line 5', 'original line 6' }, false }
- )
- -- add one line to the middle of the file, several times
- command('normal! ggYjjp')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 3, { 'original line 4' }, false })
- command('normal! p')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 4, 4, { 'original line 4' }, false })
- command('normal! p')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 5, 5, { 'original line 4' }, false })
- -- add multiple lines to the middle of the file
- command('normal! gg4Yjjp')
- tick = tick + 1
- expectn('nvim_buf_lines_event', {
- b,
- tick,
- 3,
- 3,
- {
- 'original line 4',
- 'original line 5',
- 'original line 6',
- 'original line 4',
- },
- false,
- })
- -- add one line to the end of the file
- command('normal! ggYGp')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 17, 17, { 'original line 4' }, false })
- -- add one line to the end of the file, several times
- command('normal! ggYGppp')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 18, 18, { 'original line 4' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 19, 19, { 'original line 4' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 20, 20, { 'original line 4' }, false })
- -- add several lines to the end of the file, several times
- command('normal! gg4YGp')
- command('normal! Gp')
- command('normal! Gp')
- local firstfour = { 'original line 4', 'original line 5', 'original line 6', 'original line 4' }
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 21, 21, firstfour, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 25, 25, firstfour, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 29, 29, firstfour, false })
- -- delete the current buffer to turn off buffer events
- command('bdelete!')
- expectn('nvim_buf_detach_event', { b })
- -- add a line at the start of an empty file
- command('enew')
- tick = eval('b:changedtick')
- local b2 = api.nvim_get_current_buf()
- ok(api.nvim_buf_attach(b2, true, {}))
- expectn('nvim_buf_lines_event', { b2, tick, 0, -1, { '' }, false })
- eval('append(0, ["new line 1"])')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b2, tick, 0, 0, { 'new line 1' }, false })
- -- turn off buffer events manually
- api.nvim_buf_detach(b2)
- expectn('nvim_buf_detach_event', { b2 })
- -- add multiple lines to a blank file
- command('enew!')
- local b3 = api.nvim_get_current_buf()
- ok(api.nvim_buf_attach(b3, true, {}))
- tick = eval('b:changedtick')
- expectn('nvim_buf_lines_event', { b3, tick, 0, -1, { '' }, false })
- eval('append(0, ["new line 1", "new line 2", "new line 3"])')
- tick = tick + 1
- expectn(
- 'nvim_buf_lines_event',
- { b3, tick, 0, 0, { 'new line 1', 'new line 2', 'new line 3' }, false }
- )
- -- use the API itself to add a line to the start of the buffer
- api.nvim_buf_set_lines(b3, 0, 0, true, { 'New First Line' })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b3, tick, 0, 0, { 'New First Line' }, false })
- end)
- it('when lines are removed', function()
- local b, tick = editoriginal(true)
- -- remove one line from start of file
- command('normal! dd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, {}, false })
- -- remove multiple lines from the start of the file
- command('normal! 4dd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 4, {}, false })
- -- remove multiple lines from middle of file
- tick = reopen(b, origlines)
- command('normal! jj3dd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 2, 5, {}, false })
- -- remove one line from the end of the file
- tick = reopen(b, origlines)
- command('normal! Gdd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 5, 6, {}, false })
- -- remove multiple lines from the end of the file
- tick = reopen(b, origlines)
- command('normal! 4G3dd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 6, {}, false })
- -- pretend to remove heaps lines from the end of the file but really
- -- just remove two
- tick = reopen(b, origlines)
- command('normal! Gk5dd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 4, 6, {}, false })
- end)
- it('when text is changed', function()
- local b, tick = editoriginal(true)
- -- some normal text editing
- command('normal! A555')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'original line 1555' }, false })
- command('normal! jj8X')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 2, 3, { 'origin3' }, false })
- -- modify multiple lines at once using visual block mode
- tick = reopen(b, origlines)
- command('normal! jjw')
- sendkeys('<C-v>jjllx')
- tick = tick + 1
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 2, 5, { 'original e 3', 'original e 4', 'original e 5' }, false }
- )
- -- replace part of a line line using :s
- tick = reopen(b, origlines)
- command('3s/line 3/foo/')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 2, 3, { 'original foo' }, false })
- -- replace parts of several lines line using :s
- tick = reopen(b, origlines)
- command('%s/line [35]/foo/')
- tick = tick + 1
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 2, 5, { 'original foo', 'original line 4', 'original foo' }, false }
- )
- -- type text into the first line of a blank file, one character at a time
- command('bdelete!')
- tick = 2
- expectn('nvim_buf_detach_event', { b })
- local bnew = api.nvim_get_current_buf()
- ok(api.nvim_buf_attach(bnew, true, {}))
- expectn('nvim_buf_lines_event', { bnew, tick, 0, -1, { '' }, false })
- sendkeys('i')
- sendkeys('h')
- sendkeys('e')
- sendkeys('l')
- sendkeys('l')
- sendkeys('o\nworld')
- expectn('nvim_buf_lines_event', { bnew, tick + 1, 0, 1, { 'h' }, false })
- expectn('nvim_buf_lines_event', { bnew, tick + 2, 0, 1, { 'he' }, false })
- expectn('nvim_buf_lines_event', { bnew, tick + 3, 0, 1, { 'hel' }, false })
- expectn('nvim_buf_lines_event', { bnew, tick + 4, 0, 1, { 'hell' }, false })
- expectn('nvim_buf_lines_event', { bnew, tick + 5, 0, 1, { 'hello' }, false })
- expectn('nvim_buf_lines_event', { bnew, tick + 6, 0, 1, { 'hello', '' }, false })
- expectn('nvim_buf_lines_event', { bnew, tick + 7, 1, 2, { 'world' }, false })
- end)
- it('when lines are replaced', function()
- local b, tick = editoriginal(true)
- -- blast away parts of some lines with visual mode
- command('normal! jjwvjjllx')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 2, 3, { 'original ' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 4, {}, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 4, { 'e 5' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 2, 3, { 'original e 5' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 4, {}, false })
- -- blast away a few lines using :g
- tick = reopen(b, origlines)
- command('global/line [35]/delete')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 2, 3, {}, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 4, {}, false })
- end)
- it('visual paste split empty line', function()
- local b, tick = editoriginal(true, { 'abc', '{', 'def', '}' })
- command('normal! ggyyjjvi{p')
- expectn('nvim_buf_lines_event', { b, tick + 1, 2, 3, { '' }, false })
- expectn('nvim_buf_lines_event', { b, tick + 2, 2, 3, { '}' }, false })
- expectn('nvim_buf_lines_event', { b, tick + 3, 3, 4, {}, false })
- expectn('nvim_buf_lines_event', { b, tick + 3, 2, 3, { '' }, false })
- expectn('nvim_buf_lines_event', { b, tick + 4, 3, 3, { 'abc', '}' }, false })
- end)
- it('when lines are filtered', function()
- -- Test filtering lines with !cat
- local b, tick = editoriginal(true, { 'A', 'C', 'E', 'B', 'D', 'F' })
- command('silent 2,5!cat')
- -- the change comes through as two changes:
- -- 1) addition of the new lines after the filtered lines
- -- 2) removal of the original lines
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 5, 5, { 'C', 'E', 'B', 'D' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 5, {}, false })
- end)
- it('when you use "o"', function()
- local b, tick = editoriginal(true, { 'AAA', 'BBB' })
- command('set noautoindent nosmartindent')
- -- use 'o' to start a new line from a line with no indent
- command('normal! o')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 1, { '' }, false })
- -- undo the change, indent line 1 a bit, and try again
- command('undo')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 2, {}, false })
- tick = tick + 1
- expectn('nvim_buf_changedtick_event', { b, tick })
- command('set autoindent')
- command('normal! >>')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { '\tAAA' }, false })
- command('normal! ommm')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 1, { '\t' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 2, { '\tmmm' }, false })
- -- undo the change, and try again with 'O'
- command('undo')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 2, { '\t' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 2, {}, false })
- tick = tick + 1
- expectn('nvim_buf_changedtick_event', { b, tick })
- command('normal! ggOmmm')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 0, { '\t' }, false })
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { '\tmmm' }, false })
- end)
- it('deactivates if the buffer is changed externally', function()
- -- Test changing file from outside vim and reloading using :edit
- local lines = { 'Line 1', 'Line 2' }
- local b, tick, filename = editoriginal(true, lines)
- command('normal! x')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'ine 1' }, false })
- command('undo')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'Line 1' }, false })
- tick = tick + 1
- expectn('nvim_buf_changedtick_event', { b, tick })
- -- change the file directly
- write_file(filename, 'another line\n', true, true)
- -- reopen the file and watch buffer events shut down
- command('edit')
- expectn('nvim_buf_detach_event', { b })
- end)
- it('channel can watch many buffers at once', function()
- -- edit 3 buffers, make sure they all have windows visible so that when we
- -- move between buffers, none of them are unloaded
- local b1, tick1 = editoriginal(true, { 'A1', 'A2' })
- local b1nr = eval('bufnr("")')
- command('split')
- local b2, tick2 = open(true, { 'B1', 'B2' })
- local b2nr = eval('bufnr("")')
- command('split')
- local b3, tick3 = open(true, { 'C1', 'C2' })
- local b3nr = eval('bufnr("")')
- -- make a new window for moving between buffers
- command('split')
- command('b' .. b1nr)
- command('normal! x')
- tick1 = tick1 + 1
- expectn('nvim_buf_lines_event', { b1, tick1, 0, 1, { '1' }, false })
- command('undo')
- tick1 = tick1 + 1
- expectn('nvim_buf_lines_event', { b1, tick1, 0, 1, { 'A1' }, false })
- tick1 = tick1 + 1
- expectn('nvim_buf_changedtick_event', { b1, tick1 })
- command('b' .. b2nr)
- command('normal! x')
- tick2 = tick2 + 1
- expectn('nvim_buf_lines_event', { b2, tick2, 0, 1, { '1' }, false })
- command('undo')
- tick2 = tick2 + 1
- expectn('nvim_buf_lines_event', { b2, tick2, 0, 1, { 'B1' }, false })
- tick2 = tick2 + 1
- expectn('nvim_buf_changedtick_event', { b2, tick2 })
- command('b' .. b3nr)
- command('normal! x')
- tick3 = tick3 + 1
- expectn('nvim_buf_lines_event', { b3, tick3, 0, 1, { '1' }, false })
- command('undo')
- tick3 = tick3 + 1
- expectn('nvim_buf_lines_event', { b3, tick3, 0, 1, { 'C1' }, false })
- tick3 = tick3 + 1
- expectn('nvim_buf_changedtick_event', { b3, tick3 })
- end)
- it('does not get confused if enabled/disabled many times', function()
- local channel = api.nvim_get_chan_info(0).id
- local b, tick = editoriginal(false)
- -- Enable buffer events many times.
- ok(api.nvim_buf_attach(b, true, {}))
- ok(api.nvim_buf_attach(b, true, {}))
- ok(api.nvim_buf_attach(b, true, {}))
- ok(api.nvim_buf_attach(b, true, {}))
- ok(api.nvim_buf_attach(b, true, {}))
- expectn('nvim_buf_lines_event', { b, tick, 0, -1, origlines, false })
- eval('rpcnotify(' .. channel .. ', "Hello There")')
- expectn('Hello There', {})
- -- Disable buffer events many times.
- ok(api.nvim_buf_detach(b))
- ok(api.nvim_buf_detach(b))
- ok(api.nvim_buf_detach(b))
- ok(api.nvim_buf_detach(b))
- ok(api.nvim_buf_detach(b))
- expectn('nvim_buf_detach_event', { b })
- eval('rpcnotify(' .. channel .. ', "Hello Again")')
- expectn('Hello Again', {})
- end)
- it('can notify several channels at once', function()
- clear()
- -- create several new sessions, in addition to our main API
- local sessions = {}
- local pipe = n.new_pipename()
- eval("serverstart('" .. pipe .. "')")
- sessions[1] = n.connect(pipe)
- sessions[2] = n.connect(pipe)
- sessions[3] = n.connect(pipe)
- local function request(sessionnr, method, ...)
- local status, rv = sessions[sessionnr]:request(method, ...)
- if not status then
- error(rv[2])
- end
- return rv
- end
- local function wantn(sessionid, name, args)
- local session = sessions[sessionid]
- eq({ 'notification', name, args }, session:next_message(10000))
- end
- -- Edit a new file, but don't enable buffer events.
- local lines = { 'AAA', 'BBB' }
- local b, tick = open(false, lines)
- -- Enable buffer events for sessions 1, 2 and 3.
- ok(request(1, 'nvim_buf_attach', b, true, {}))
- ok(request(2, 'nvim_buf_attach', b, true, {}))
- ok(request(3, 'nvim_buf_attach', b, true, {}))
- wantn(1, 'nvim_buf_lines_event', { b, tick, 0, -1, lines, false })
- wantn(2, 'nvim_buf_lines_event', { b, tick, 0, -1, lines, false })
- wantn(3, 'nvim_buf_lines_event', { b, tick, 0, -1, lines, false })
- -- Change the buffer.
- command('normal! x')
- tick = tick + 1
- wantn(1, 'nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- wantn(2, 'nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- wantn(3, 'nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- -- Stop watching on channel 1.
- ok(request(1, 'nvim_buf_detach', b))
- wantn(1, 'nvim_buf_detach_event', { b })
- -- Undo the change to buffer 1.
- command('undo')
- tick = tick + 1
- wantn(2, 'nvim_buf_lines_event', { b, tick, 0, 1, { 'AAA' }, false })
- wantn(3, 'nvim_buf_lines_event', { b, tick, 0, 1, { 'AAA' }, false })
- tick = tick + 1
- wantn(2, 'nvim_buf_changedtick_event', { b, tick })
- wantn(3, 'nvim_buf_changedtick_event', { b, tick })
- -- make sure there are no other pending nvim_buf_lines_event messages going to
- -- channel 1
- local channel1 = request(1, 'nvim_get_chan_info', 0).id
- eval('rpcnotify(' .. channel1 .. ', "Hello")')
- wantn(1, 'Hello', {})
- -- close the buffer and channels 2 and 3 should get a nvim_buf_detach_event
- -- notification
- command('edit')
- wantn(2, 'nvim_buf_detach_event', { b })
- wantn(3, 'nvim_buf_detach_event', { b })
- -- make sure there are no other pending nvim_buf_lines_event messages going to
- -- channel 1
- channel1 = request(1, 'nvim_get_chan_info', 0).id
- eval('rpcnotify(' .. channel1 .. ', "Hello Again")')
- wantn(1, 'Hello Again', {})
- end)
- it('works with :diffput and :diffget', function()
- local b1, tick1 = editoriginal(true, { 'AAA', 'BBB' })
- local channel = api.nvim_get_chan_info(0).id
- command('diffthis')
- command('rightbelow vsplit')
- local b2, tick2 = open(true, { 'BBB', 'CCC' })
- command('diffthis')
- -- go back to first buffer, and push the 'AAA' line to the second buffer
- command('1wincmd w')
- command('normal! gg')
- command('diffput')
- tick2 = tick2 + 1
- expectn('nvim_buf_lines_event', { b2, tick2, 0, 0, { 'AAA' }, false })
- -- use :diffget to grab the other change from buffer 2
- command('normal! G')
- command('diffget')
- tick1 = tick1 + 1
- expectn('nvim_buf_lines_event', { b1, tick1, 2, 2, { 'CCC' }, false })
- eval('rpcnotify(' .. channel .. ', "Goodbye")')
- expectn('Goodbye', {})
- end)
- it('works with :sort', function()
- -- test for :sort
- local b, tick = editoriginal(true, { 'B', 'D', 'C', 'A', 'E' })
- command('%sort')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 5, { 'A', 'B', 'C', 'D', 'E' }, false })
- end)
- it('works with :left', function()
- local b, tick = editoriginal(true, { ' A', ' B', 'B', '\tB', '\t\tC' })
- command('2,4left')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 4, { 'B', 'B', 'B' }, false })
- end)
- it('works with :right', function()
- local b, tick = editoriginal(true, { ' A', '\t B', '\t \tBB', ' \tB', '\t\tC' })
- command('set ts=2 et')
- command('2,4retab')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 4, { ' B', ' BB', ' B' }, false })
- end)
- it('works with :move', function()
- local b, tick = editoriginal(true, origlines)
- -- move text down towards the end of the file
- command('2,3move 4')
- tick = tick + 2
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 4, 4, { 'original line 2', 'original line 3' }, false }
- )
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 1, 3, {}, false })
- -- move text up towards the start of the file
- tick = reopen(b, origlines)
- command('4,5move 2')
- tick = tick + 2
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 2, 2, { 'original line 4', 'original line 5' }, false }
- )
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 5, 7, {}, false })
- end)
- it('when you manually add/remove folds', function()
- local b = editoriginal(true)
- local tick = reopenwithfolds(b)
- -- delete the inner fold
- command('normal! zR3Gzd')
- tick = tick + 1
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 1, 4, { 'original line 2', 'original line 3', 'original line 4' }, false }
- )
- -- delete the outer fold
- command('normal! zd')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 6, origlines, false })
- -- discard changes and put the folds back
- tick = reopenwithfolds(b)
- -- remove both folds at once
- command('normal! ggzczD')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 6, origlines, false })
- -- discard changes and put the folds back
- tick = reopenwithfolds(b)
- -- now delete all folds at once
- command('normal! zE')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 6, origlines, false })
- -- create a fold from line 4 to the end of the file
- command('normal! 4GA/*{{{*/')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 3, 4, { 'original line 4/*{{{*/' }, false })
- -- delete the fold which only has one marker
- command('normal! Gzd')
- tick = tick + 1
- expectn(
- 'nvim_buf_lines_event',
- { b, tick, 3, 6, { 'original line 4', 'original line 5', 'original line 6' }, false }
- )
- end)
- it('detaches if the buffer is closed', function()
- local b, tick = editoriginal(true, { 'AAA' })
- local channel = api.nvim_get_chan_info(0).id
- -- Test that buffer events are working.
- command('normal! x')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- command('undo')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AAA' }, false })
- tick = tick + 1
- expectn('nvim_buf_changedtick_event', { b, tick })
- -- close our buffer and create a new one
- command('bdelete')
- command('enew')
- expectn('nvim_buf_detach_event', { b })
- -- Reopen the original buffer, make sure there are no buffer events sent.
- command('b1')
- command('normal! x')
- eval('rpcnotify(' .. channel .. ', "Hello There")')
- expectn('Hello There', {})
- end)
- it(':edit! (reload) causes detach #9642', function()
- local b, tick = editoriginal(true, { 'AAA', 'BBB' })
- command('set undoreload=1')
- command('normal! x')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- command('edit!')
- expectn('nvim_buf_detach_event', { b })
- end)
- it(':enew! does not detach hidden buffer', function()
- local b, tick = editoriginal(true, { 'AAA', 'BBB' })
- local channel = api.nvim_get_chan_info(0).id
- command('set undoreload=1 hidden')
- command('normal! x')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- command('enew!')
- eval('rpcnotify(' .. channel .. ', "Hello There")')
- expectn('Hello There', {})
- end)
- it('stays attached if the buffer is hidden', function()
- local b, tick = editoriginal(true, { 'AAA' })
- local channel = api.nvim_get_chan_info(0).id
- -- Test that buffer events are working.
- command('normal! x')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- command('undo')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AAA' }, false })
- tick = tick + 1
- expectn('nvim_buf_changedtick_event', { b, tick })
- -- Close our buffer by creating a new one.
- command('set hidden')
- command('enew')
- -- Assert that no nvim_buf_detach_event is sent.
- eval('rpcnotify(' .. channel .. ', "Hello There")')
- expectn('Hello There', {})
- -- Reopen the original buffer, assert that buffer events are still active.
- command('b1')
- command('normal! x')
- tick = tick + 1
- expectn('nvim_buf_lines_event', { b, tick, 0, 1, { 'AA' }, false })
- end)
- it('detaches if the buffer is unloaded/deleted/wiped', function()
- -- start with a blank nvim
- clear()
- -- need to make a new window with a buffer because :bunload doesn't let you
- -- unload the last buffer
- for _, cmd in ipairs({ 'bunload', 'bdelete', 'bwipeout' }) do
- command('new')
- -- open a brand spanking new file
- local b = open(true, { 'AAA' })
- -- call :bunload or whatever the command is, and then check that we
- -- receive a nvim_buf_detach_event
- command(cmd)
- expectn('nvim_buf_detach_event', { b })
- end
- end)
- it('does not send the buffer content if not requested', function()
- clear()
- local b, tick = editoriginal(false)
- ok(api.nvim_buf_attach(b, false, {}))
- expectn('nvim_buf_changedtick_event', { b, tick })
- end)
- it('returns a proper error on nonempty options dict', function()
- clear()
- local b = editoriginal(false)
- eq("Invalid key: 'builtin'", pcall_err(api.nvim_buf_attach, b, false, { builtin = 'asfd' }))
- end)
- it('nvim_buf_attach returns response after delay #8634', function()
- clear()
- sleep(250)
- -- response
- eq(true, n.request('nvim_buf_attach', 0, false, {}))
- -- notification
- eq({
- [1] = 'notification',
- [2] = 'nvim_buf_changedtick_event',
- [3] = {
- [1] = 1,
- [2] = 2,
- },
- }, next_msg())
- end)
- end)
- describe('API: buffer events:', function()
- before_each(function()
- clear()
- end)
- local function lines_subset(first, second)
- for i = 1, #first do
- -- need to ignore trailing spaces
- if first[i]:gsub(' +$', '') ~= second[i]:gsub(' +$', '') then
- return false
- end
- end
- return true
- end
- local function lines_equal(f, s)
- return lines_subset(f, s) and lines_subset(s, f)
- end
- local function assert_match_somewhere(expected_lines, buffer_lines)
- local msg = next_msg()
- while msg ~= nil do
- local event = msg[2]
- if event == 'nvim_buf_lines_event' then
- local args = msg[3]
- local starts = args[3]
- local newlines = args[5]
- -- Size of the contained nvim instance is 23 lines, this might change
- -- with the test setup. Note updates are contiguous.
- assert(#newlines <= 23)
- for i = 1, #newlines do
- buffer_lines[starts + i] = newlines[i]
- end
- -- we don't compare the msg area of the embedded nvim, it's too flakey
- buffer_lines[23] = nil
- if lines_equal(buffer_lines, expected_lines) then
- -- OK
- return
- end
- end
- msg = next_msg()
- end
- assert(false, 'did not match/receive expected nvim_buf_lines_event lines')
- end
- it('when :terminal lines change', function()
- local buffer_lines = {}
- local expected_lines = {}
- fn.jobstart({ nvim_prog, '-u', 'NONE', '-i', 'NONE', '-n', '-c', 'set shortmess+=A' }, {
- term = true,
- env = { VIMRUNTIME = os.getenv('VIMRUNTIME') },
- })
- local b = api.nvim_get_current_buf()
- ok(api.nvim_buf_attach(b, true, {}))
- for _ = 1, 22 do
- table.insert(expected_lines, '~')
- end
- expected_lines[1] = ''
- expected_lines[22] = ('tmp_terminal_nvim' .. (' '):rep(45) .. '0,0-1 All')
- sendkeys('i:e tmp_terminal_nvim<Enter>')
- assert_match_somewhere(expected_lines, buffer_lines)
- expected_lines[1] = 'Blarg'
- expected_lines[22] = ('tmp_terminal_nvim [+]' .. (' '):rep(41) .. '1,6 All')
- sendkeys('iBlarg')
- assert_match_somewhere(expected_lines, buffer_lines)
- for i = 1, 21 do
- expected_lines[i] = 'xyz'
- end
- expected_lines[22] = ('tmp_terminal_nvim [+]' .. (' '):rep(41) .. '31,4 Bot')
- local s = string.rep('\nxyz', 30)
- sendkeys(s)
- assert_match_somewhere(expected_lines, buffer_lines)
- end)
- end)
|