123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849 |
- local helpers = require('test.functional.helpers')(after_each)
- local Screen = require('test.functional.ui.screen')
- local clear, nvim, buffer = helpers.clear, helpers.nvim, helpers.buffer
- local curbuf, curwin, eq = helpers.curbuf, helpers.curwin, helpers.eq
- local curbufmeths, ok = helpers.curbufmeths, helpers.ok
- local meths = helpers.meths
- local funcs = helpers.funcs
- local request = helpers.request
- local exc_exec = helpers.exc_exec
- local exec_lua = helpers.exec_lua
- local feed_command = helpers.feed_command
- local insert = helpers.insert
- local NIL = helpers.NIL
- local command = helpers.command
- local bufmeths = helpers.bufmeths
- local feed = helpers.feed
- local pcall_err = helpers.pcall_err
- describe('api/buf', function()
- before_each(clear)
- -- access deprecated functions
- local function curbuf_depr(method, ...)
- return request('buffer_'..method, 0, ...)
- end
- describe('nvim_buf_set_lines, nvim_buf_line_count', function()
- it('deprecated forms', function()
- eq(1, curbuf_depr('line_count'))
- curbuf_depr('insert', -1, {'line'})
- eq(2, curbuf_depr('line_count'))
- curbuf_depr('insert', -1, {'line'})
- eq(3, curbuf_depr('line_count'))
- curbuf_depr('del_line', -1)
- eq(2, curbuf_depr('line_count'))
- curbuf_depr('del_line', -1)
- curbuf_depr('del_line', -1)
- -- There's always at least one line
- eq(1, curbuf_depr('line_count'))
- end)
- it('cursor position is maintained after lines are inserted #9961', function()
- -- replace the buffer contents with these three lines.
- request('nvim_buf_set_lines', 0, 0, -1, 1, {"line1", "line2", "line3", "line4"})
- -- Set the current cursor to {3, 2}.
- curwin('set_cursor', {3, 2})
- -- add 2 lines and delete 1 line above the current cursor position.
- request('nvim_buf_set_lines', 0, 1, 2, 1, {"line5", "line6"})
- -- check the current set of lines in the buffer.
- eq({"line1", "line5", "line6", "line3", "line4"}, buffer('get_lines', 0, 0, -1, 1))
- -- cursor should be moved below by 1 line.
- eq({4, 2}, curwin('get_cursor'))
- -- add a line after the current cursor position.
- request('nvim_buf_set_lines', 0, 5, 5, 1, {"line7"})
- -- check the current set of lines in the buffer.
- eq({"line1", "line5", "line6", "line3", "line4", "line7"}, buffer('get_lines', 0, 0, -1, 1))
- -- cursor position is unchanged.
- eq({4, 2}, curwin('get_cursor'))
- -- overwrite current cursor line.
- request('nvim_buf_set_lines', 0, 3, 5, 1, {"line8", "line9"})
- -- check the current set of lines in the buffer.
- eq({"line1", "line5", "line6", "line8", "line9", "line7"}, buffer('get_lines', 0, 0, -1, 1))
- -- cursor position is unchanged.
- eq({4, 2}, curwin('get_cursor'))
- -- delete current cursor line.
- request('nvim_buf_set_lines', 0, 3, 5, 1, {})
- -- check the current set of lines in the buffer.
- eq({"line1", "line5", "line6", "line7"}, buffer('get_lines', 0, 0, -1, 1))
- -- cursor position is unchanged.
- eq({4, 2}, curwin('get_cursor'))
- end)
- it('line_count has defined behaviour for unloaded buffers', function()
- -- we'll need to know our bufnr for when it gets unloaded
- local bufnr = curbuf('get_number')
- -- replace the buffer contents with these three lines
- request('nvim_buf_set_lines', bufnr, 0, -1, 1, {"line1", "line2", "line3", "line4"})
- -- check the line count is correct
- eq(4, request('nvim_buf_line_count', bufnr))
- -- force unload the buffer (this will discard changes)
- command('new')
- command('bunload! '..bufnr)
- -- line count for an unloaded buffer should always be 0
- eq(0, request('nvim_buf_line_count', bufnr))
- end)
- it('get_lines has defined behaviour for unloaded buffers', function()
- -- we'll need to know our bufnr for when it gets unloaded
- local bufnr = curbuf('get_number')
- -- replace the buffer contents with these three lines
- buffer('set_lines', bufnr, 0, -1, 1, {"line1", "line2", "line3", "line4"})
- -- confirm that getting lines works
- eq({"line2", "line3"}, buffer('get_lines', bufnr, 1, 3, 1))
- -- force unload the buffer (this will discard changes)
- command('new')
- command('bunload! '..bufnr)
- -- attempting to get lines now always gives empty list
- eq({}, buffer('get_lines', bufnr, 1, 3, 1))
- -- it's impossible to get out-of-bounds errors for an unloaded buffer
- eq({}, buffer('get_lines', bufnr, 8888, 9999, 1))
- end)
- end)
- describe('deprecated: {get,set,del}_line', function()
- it('works', function()
- eq('', curbuf_depr('get_line', 0))
- curbuf_depr('set_line', 0, 'line1')
- eq('line1', curbuf_depr('get_line', 0))
- curbuf_depr('set_line', 0, 'line2')
- eq('line2', curbuf_depr('get_line', 0))
- curbuf_depr('del_line', 0)
- eq('', curbuf_depr('get_line', 0))
- end)
- it('get_line: out-of-bounds is an error', function()
- curbuf_depr('set_line', 0, 'line1.a')
- eq(1, curbuf_depr('line_count')) -- sanity
- eq(false, pcall(curbuf_depr, 'get_line', 1))
- eq(false, pcall(curbuf_depr, 'get_line', -2))
- end)
- it('set_line, del_line: out-of-bounds is an error', function()
- curbuf_depr('set_line', 0, 'line1.a')
- eq(false, pcall(curbuf_depr, 'set_line', 1, 'line1.b'))
- eq(false, pcall(curbuf_depr, 'set_line', -2, 'line1.b'))
- eq(false, pcall(curbuf_depr, 'del_line', 2))
- eq(false, pcall(curbuf_depr, 'del_line', -3))
- end)
- it('can handle NULs', function()
- curbuf_depr('set_line', 0, 'ab\0cd')
- eq('ab\0cd', curbuf_depr('get_line', 0))
- end)
- end)
- describe('deprecated: {get,set}_line_slice', function()
- it('get_line_slice: out-of-bounds returns empty array', function()
- curbuf_depr('set_line_slice', 0, 0, true, true, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, 2, true, true)) --sanity
- eq({}, curbuf_depr('get_line_slice', 2, 3, false, true))
- eq({}, curbuf_depr('get_line_slice', 3, 9, true, true))
- eq({}, curbuf_depr('get_line_slice', 3, -1, true, true))
- eq({}, curbuf_depr('get_line_slice', -3, -4, false, true))
- eq({}, curbuf_depr('get_line_slice', -4, -5, true, true))
- end)
- it('set_line_slice: out-of-bounds extends past end', function()
- curbuf_depr('set_line_slice', 0, 0, true, true, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, 2, true, true)) --sanity
- eq({'c'}, curbuf_depr('get_line_slice', -1, 4, true, true))
- eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, 5, true, true))
- curbuf_depr('set_line_slice', 4, 5, true, true, {'d'})
- eq({'a', 'b', 'c', 'd'}, curbuf_depr('get_line_slice', 0, 5, true, true))
- curbuf_depr('set_line_slice', -4, -5, true, true, {'e'})
- eq({'e', 'a', 'b', 'c', 'd'}, curbuf_depr('get_line_slice', 0, 5, true, true))
- end)
- it('works', function()
- eq({''}, curbuf_depr('get_line_slice', 0, -1, true, true))
- -- Replace buffer
- curbuf_depr('set_line_slice', 0, -1, true, true, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, -1, true, true))
- eq({'b', 'c'}, curbuf_depr('get_line_slice', 1, -1, true, true))
- eq({'b'}, curbuf_depr('get_line_slice', 1, 2, true, false))
- eq({}, curbuf_depr('get_line_slice', 1, 1, true, false))
- eq({'a', 'b'}, curbuf_depr('get_line_slice', 0, -1, true, false))
- eq({'b'}, curbuf_depr('get_line_slice', 1, -1, true, false))
- eq({'b', 'c'}, curbuf_depr('get_line_slice', -2, -1, true, true))
- curbuf_depr('set_line_slice', 1, 2, true, false, {'a', 'b', 'c'})
- eq({'a', 'a', 'b', 'c', 'c'}, curbuf_depr('get_line_slice', 0, -1, true, true))
- curbuf_depr('set_line_slice', -1, -1, true, true, {'a', 'b', 'c'})
- eq({'a', 'a', 'b', 'c', 'a', 'b', 'c'},
- curbuf_depr('get_line_slice', 0, -1, true, true))
- curbuf_depr('set_line_slice', 0, -3, true, false, {})
- eq({'a', 'b', 'c'}, curbuf_depr('get_line_slice', 0, -1, true, true))
- curbuf_depr('set_line_slice', 0, -1, true, true, {})
- eq({''}, curbuf_depr('get_line_slice', 0, -1, true, true))
- end)
- end)
- describe('nvim_buf_get_lines, nvim_buf_set_lines', function()
- local get_lines, set_lines = curbufmeths.get_lines, curbufmeths.set_lines
- local line_count = curbufmeths.line_count
- it('fails correctly when input is not valid', function()
- eq(1, curbufmeths.get_number())
- eq([[String cannot contain newlines]],
- pcall_err(bufmeths.set_lines, 1, 1, 2, false, {'b\na'}))
- end)
- it("fails if 'nomodifiable'", function()
- command('set nomodifiable')
- eq([[Buffer is not 'modifiable']],
- pcall_err(bufmeths.set_lines, 1, 1, 2, false, {'a','b'}))
- end)
- it('has correct line_count when inserting and deleting', function()
- eq(1, line_count())
- set_lines(-1, -1, true, {'line'})
- eq(2, line_count())
- set_lines(-1, -1, true, {'line'})
- eq(3, line_count())
- set_lines(-2, -1, true, {})
- eq(2, line_count())
- set_lines(-2, -1, true, {})
- set_lines(-2, -1, true, {})
- -- There's always at least one line
- eq(1, line_count())
- end)
- it('can get, set and delete a single line', function()
- eq({''}, get_lines(0, 1, true))
- set_lines(0, 1, true, {'line1'})
- eq({'line1'}, get_lines(0, 1, true))
- set_lines(0, 1, true, {'line2'})
- eq({'line2'}, get_lines(0, 1, true))
- set_lines(0, 1, true, {})
- eq({''}, get_lines(0, 1, true))
- end)
- it('can get a single line with strict indexing', function()
- set_lines(0, 1, true, {'line1.a'})
- eq(1, line_count()) -- sanity
- eq('Index out of bounds', pcall_err(get_lines, 1, 2, true))
- eq('Index out of bounds', pcall_err(get_lines, -3, -2, true))
- end)
- it('can get a single line with non-strict indexing', function()
- set_lines(0, 1, true, {'line1.a'})
- eq(1, line_count()) -- sanity
- eq({}, get_lines(1, 2, false))
- eq({}, get_lines(-3, -2, false))
- end)
- it('can set and delete a single line with strict indexing', function()
- set_lines(0, 1, true, {'line1.a'})
- eq('Index out of bounds', pcall_err(set_lines, 1, 2, true, {'line1.b'}))
- eq('Index out of bounds', pcall_err(set_lines, -3, -2, true, {'line1.c'}))
- eq({'line1.a'}, get_lines(0, -1, true))
- eq('Index out of bounds', pcall_err(set_lines, 1, 2, true, {}))
- eq('Index out of bounds', pcall_err(set_lines, -3, -2, true, {}))
- eq({'line1.a'}, get_lines(0, -1, true))
- end)
- it('can set and delete a single line with non-strict indexing', function()
- set_lines(0, 1, true, {'line1.a'})
- set_lines(1, 2, false, {'line1.b'})
- set_lines(-4, -3, false, {'line1.c'})
- eq({'line1.c', 'line1.a', 'line1.b'}, get_lines(0, -1, true))
- set_lines(3, 4, false, {})
- set_lines(-5, -4, false, {})
- eq({'line1.c', 'line1.a', 'line1.b'}, get_lines(0, -1, true))
- end)
- it('can handle NULs', function()
- set_lines(0, 1, true, {'ab\0cd'})
- eq({'ab\0cd'}, get_lines(0, -1, true))
- end)
- it('works with multiple lines', function()
- eq({''}, get_lines(0, -1, true))
- -- Replace buffer
- for _, mode in pairs({false, true}) do
- set_lines(0, -1, mode, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, get_lines(0, -1, mode))
- eq({'b', 'c'}, get_lines(1, -1, mode))
- eq({'b'}, get_lines(1, 2, mode))
- eq({}, get_lines(1, 1, mode))
- eq({'a', 'b'}, get_lines(0, -2, mode))
- eq({'b'}, get_lines(1, -2, mode))
- eq({'b', 'c'}, get_lines(-3, -1, mode))
- set_lines(1, 2, mode, {'a', 'b', 'c'})
- eq({'a', 'a', 'b', 'c', 'c'}, get_lines(0, -1, mode))
- set_lines(-2, -1, mode, {'a', 'b', 'c'})
- eq({'a', 'a', 'b', 'c', 'a', 'b', 'c'},
- get_lines(0, -1, mode))
- set_lines(0, -4, mode, {})
- eq({'a', 'b', 'c'}, get_lines(0, -1, mode))
- set_lines(0, -1, mode, {})
- eq({''}, get_lines(0, -1, mode))
- end
- end)
- it('can get line ranges with non-strict indexing', function()
- set_lines(0, -1, true, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
- eq({}, get_lines(3, 4, false))
- eq({}, get_lines(3, 10, false))
- eq({}, get_lines(-5, -5, false))
- eq({}, get_lines(3, -1, false))
- eq({}, get_lines(-3, -4, false))
- end)
- it('can get line ranges with strict indexing', function()
- set_lines(0, -1, true, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
- eq('Index out of bounds', pcall_err(get_lines, 3, 4, true))
- eq('Index out of bounds', pcall_err(get_lines, 3, 10, true))
- eq('Index out of bounds', pcall_err(get_lines, -5, -5, true))
- -- empty or inverted ranges are not errors
- eq({}, get_lines(3, -1, true))
- eq({}, get_lines(-3, -4, true))
- end)
- it('set_lines: out-of-bounds can extend past end', function()
- set_lines(0, -1, true, {'a', 'b', 'c'})
- eq({'a', 'b', 'c'}, get_lines(0, -1, true)) --sanity
- eq({'c'}, get_lines(-2, 5, false))
- eq({'a', 'b', 'c'}, get_lines(0, 6, false))
- eq('Index out of bounds', pcall_err(set_lines, 4, 6, true, {'d'}))
- set_lines(4, 6, false, {'d'})
- eq({'a', 'b', 'c', 'd'}, get_lines(0, -1, true))
- eq('Index out of bounds', pcall_err(set_lines, -6, -6, true, {'e'}))
- set_lines(-6, -6, false, {'e'})
- eq({'e', 'a', 'b', 'c', 'd'}, get_lines(0, -1, true))
- end)
- it("set_lines on alternate buffer does not access invalid line (E315)", function()
- feed_command('set hidden')
- insert('Initial file')
- command('enew')
- insert([[
- More
- Lines
- Than
- In
- The
- Other
- Buffer]])
- feed_command('$')
- local retval = exc_exec("call nvim_buf_set_lines(1, 0, 1, v:false, ['test'])")
- eq(0, retval)
- end)
- it("set_lines of invisible buffer doesn't move cursor in current window", function()
- local screen = Screen.new(20, 5)
- screen:set_default_attr_ids({
- [1] = {bold = true, foreground = Screen.colors.Blue1},
- [2] = {bold = true},
- })
- screen:attach()
- insert([[
- Who would win?
- A real window
- with proper text]])
- local buf = meths.create_buf(false,true)
- screen:expect([[
- Who would win? |
- A real window |
- with proper tex^t |
- {1:~ }|
- |
- ]])
- meths.buf_set_lines(buf, 0, -1, true, {'or some', 'scratchy text'})
- feed('i') -- provoke redraw
- screen:expect([[
- Who would win? |
- A real window |
- with proper tex^t |
- {1:~ }|
- {2:-- INSERT --} |
- ]])
- end)
- it('set_lines on hidden buffer preserves "previous window" #9741', function()
- insert([[
- visible buffer line 1
- line 2
- ]])
- local hiddenbuf = meths.create_buf(false,true)
- command('vsplit')
- command('vsplit')
- feed('<c-w>l<c-w>l<c-w>l')
- eq(3, funcs.winnr())
- feed('<c-w>h')
- eq(2, funcs.winnr())
- meths.buf_set_lines(hiddenbuf, 0, -1, true,
- {'hidden buffer line 1', 'line 2'})
- feed('<c-w>p')
- eq(3, funcs.winnr())
- end)
- end)
- describe('nvim_buf_set_text', function()
- local get_lines, set_text = curbufmeths.get_lines, curbufmeths.set_text
- it('works', function()
- insert([[
- hello foo!
- text
- ]])
- eq({'hello foo!'}, get_lines(0, 1, true))
- -- can replace a single word
- set_text(0, 6, 0, 9, {'world'})
- eq({'hello world!', 'text'}, get_lines(0, 2, true))
- -- can insert text
- set_text(0, 0, 0, 0, {'well '})
- eq({'well hello world!', 'text'}, get_lines(0, 2, true))
- -- can delete text
- set_text(0, 0, 0, 5, {''})
- eq({'hello world!', 'text'}, get_lines(0, 2, true))
- -- can replace with multiple lines
- set_text(0, 6, 0, 11, {'foo', 'wo', 'more'})
- eq({'hello foo', 'wo', 'more!', 'text'}, get_lines(0, 4, true))
- -- will join multiple lines if needed
- set_text(0, 6, 3, 4, {'bar'})
- eq({'hello bar'}, get_lines(0, 1, true))
- -- can use negative line numbers
- set_text(-2, 0, -2, 5, {'goodbye'})
- eq({'goodbye bar', ''}, get_lines(0, -1, true))
- set_text(-1, 0, -1, 0, {'text'})
- eq({'goodbye bar', 'text'}, get_lines(0, 2, true))
- -- can append to a line
- set_text(1, 4, -1, 4, {' and', 'more'})
- eq({'goodbye bar', 'text and', 'more'}, get_lines(0, 3, true))
- end)
- it('works with undo', function()
- insert([[
- hello world!
- foo bar
- ]])
- -- setting text
- set_text(0, 0, 0, 0, {'well '})
- feed('u')
- eq({'hello world!'}, get_lines(0, 1, true))
- -- deleting text
- set_text(0, 0, 0, 6, {''})
- feed('u')
- eq({'hello world!'}, get_lines(0, 1, true))
- -- inserting newlines
- set_text(0, 0, 0, 0, {'hello', 'mr '})
- feed('u')
- eq({'hello world!'}, get_lines(0, 1, true))
- -- deleting newlines
- set_text(0, 0, 1, 4, {'hello'})
- feed('u')
- eq({'hello world!'}, get_lines(0, 1, true))
- end)
- it('updates the cursor position', function()
- insert([[
- hello world!
- ]])
- -- position the cursor on `!`
- curwin('set_cursor', {1, 11})
- -- replace 'world' with 'foo'
- set_text(0, 6, 0, 11, {'foo'})
- eq('hello foo!', curbuf_depr('get_line', 0))
- -- cursor should be moved left by two columns (replacement is shorter by 2 chars)
- eq({1, 9}, curwin('get_cursor'))
- end)
- it('can handle NULs', function()
- set_text(0, 0, 0, 0, {'ab\0cd'})
- eq('ab\0cd', curbuf_depr('get_line', 0))
- end)
- it('adjusts extmarks', function()
- local ns = request('nvim_create_namespace', "my-fancy-plugin")
- insert([[
- foo bar
- baz
- ]])
- local id1 = curbufmeths.set_extmark(ns, 0, 1, {})
- local id2 = curbufmeths.set_extmark(ns, 0, 7, {})
- local id3 = curbufmeths.set_extmark(ns, 1, 1, {})
- set_text(0, 4, 0, 7, {"q"})
- eq({'foo q', 'baz'}, get_lines(0, 2, true))
- -- mark before replacement point is unaffected
- eq({0, 1}, curbufmeths.get_extmark_by_id(ns, id1, {}))
- -- mark gets shifted back because the replacement was shorter
- eq({0, 5}, curbufmeths.get_extmark_by_id(ns, id2, {}))
- -- mark on the next line is unaffected
- eq({1, 1}, curbufmeths.get_extmark_by_id(ns, id3, {}))
- -- replacing the text spanning two lines will adjust the mark on the next line
- set_text(0, 3, 1, 3, {"qux"})
- eq({'fooqux', ''}, get_lines(0, 2, true))
- eq({0, 6}, curbufmeths.get_extmark_by_id(ns, id3, {}))
- -- but mark before replacement point is still unaffected
- eq({0, 1}, curbufmeths.get_extmark_by_id(ns, id1, {}))
- -- and the mark in the middle was shifted to the end of the insertion
- eq({0, 6}, curbufmeths.get_extmark_by_id(ns, id2, {}))
- -- marks should be put back into the same place after undoing
- set_text(0, 0, 0, 2, {''})
- feed('u')
- eq({0, 1}, curbufmeths.get_extmark_by_id(ns, id1, {}))
- eq({0, 6}, curbufmeths.get_extmark_by_id(ns, id2, {}))
- eq({0, 6}, curbufmeths.get_extmark_by_id(ns, id3, {}))
- -- marks should be shifted over by the correct number of bytes for multibyte
- -- chars
- set_text(0, 0, 0, 0, {'Ø'})
- eq({0, 3}, curbufmeths.get_extmark_by_id(ns, id1, {}))
- eq({0, 8}, curbufmeths.get_extmark_by_id(ns, id2, {}))
- eq({0, 8}, curbufmeths.get_extmark_by_id(ns, id3, {}))
- end)
- it("correctly marks changed region for redraw #13890", function()
- local screen = Screen.new(20, 5)
- screen:attach()
- insert([[
- AAA
- BBB
- ]])
- curbufmeths.set_text(0, 0, 1, 3, {'XXX', 'YYY'})
- screen:expect([[
- XXX |
- YYY |
- ^ |
- ~ |
- |
- ]])
- end)
- it('errors on out-of-range', function()
- insert([[
- hello foo!
- text]])
- eq('start_row out of bounds', pcall_err(set_text, 2, 0, 3, 0, {}))
- eq('start_row out of bounds', pcall_err(set_text, -3, 0, 0, 0, {}))
- eq('end_row out of bounds', pcall_err(set_text, 0, 0, 2, 0, {}))
- eq('end_row out of bounds', pcall_err(set_text, 0, 0, -3, 0, {}))
- eq('start_col out of bounds', pcall_err(set_text, 1, 5, 1, 5, {}))
- eq('end_col out of bounds', pcall_err(set_text, 1, 0, 1, 5, {}))
- end)
- it('errors when start is greater than end', function()
- insert([[
- hello foo!
- text]])
- eq('start is higher than end', pcall_err(set_text, 1, 0, 0, 0, {}))
- eq('start is higher than end', pcall_err(set_text, 0, 1, 0, 0, {}))
- end)
- it('no heap-use-after-free when called consecutively #19643', function()
- set_text(0, 0, 0, 0, {'one', '', '', 'two'})
- eq({'one', '', '', 'two'}, get_lines(0, 4, true))
- meths.win_set_cursor(0, {1, 0})
- exec_lua([[
- vim.api.nvim_buf_set_text(0, 0, 3, 1, 0, {''})
- vim.api.nvim_buf_set_text(0, 0, 3, 1, 0, {''})
- ]])
- eq({'one', 'two'}, get_lines(0, 2, true))
- end)
- end)
- describe('nvim_buf_get_text', function()
- local get_text = curbufmeths.get_text
- before_each(function()
- insert([[
- hello foo!
- text]])
- end)
- it('works', function()
- eq({'hello'}, get_text(0, 0, 0, 5, {}))
- eq({'hello foo!'}, get_text(0, 0, 0, 42, {}))
- eq({'foo!'}, get_text(0, 6, 0, 10, {}))
- eq({'foo!', 'tex'}, get_text(0, 6, 1, 3, {}))
- eq({'foo!', 'tex'}, get_text(-2, 6, -1, 3, {}))
- eq({''}, get_text(0, 18, 0, 20, {}))
- eq({'ext'}, get_text(-1, 1, -1, 4, {}))
- end)
- it('errors on out-of-range', function()
- eq('Index out of bounds', pcall_err(get_text, 2, 0, 3, 0, {}))
- eq('Index out of bounds', pcall_err(get_text, -3, 0, 0, 0, {}))
- eq('Index out of bounds', pcall_err(get_text, 0, 0, 2, 0, {}))
- eq('Index out of bounds', pcall_err(get_text, 0, 0, -3, 0, {}))
- -- no ml_get errors should happen #19017
- eq('', meths.get_vvar('errmsg'))
- end)
- it('errors when start is greater than end', function()
- eq('start is higher than end', pcall_err(get_text, 1, 0, 0, 0, {}))
- eq('start_col must be less than end_col', pcall_err(get_text, 0, 1, 0, 0, {}))
- end)
- end)
- describe('nvim_buf_get_offset', function()
- local get_offset = curbufmeths.get_offset
- it('works', function()
- curbufmeths.set_lines(0,-1,true,{'Some\r','exa\000mple', '', 'buf\rfer', 'text'})
- eq(5, curbufmeths.line_count())
- eq(0, get_offset(0))
- eq(6, get_offset(1))
- eq(15, get_offset(2))
- eq(16, get_offset(3))
- eq(24, get_offset(4))
- eq(29, get_offset(5))
- eq('Index out of bounds', pcall_err(get_offset, 6))
- eq('Index out of bounds', pcall_err(get_offset, -1))
- curbufmeths.set_option('eol', false)
- curbufmeths.set_option('fixeol', false)
- eq(28, get_offset(5))
- -- fileformat is ignored
- curbufmeths.set_option('fileformat', 'dos')
- eq(0, get_offset(0))
- eq(6, get_offset(1))
- eq(15, get_offset(2))
- eq(16, get_offset(3))
- eq(24, get_offset(4))
- eq(28, get_offset(5))
- curbufmeths.set_option('eol', true)
- eq(29, get_offset(5))
- command("set hidden")
- command("enew")
- eq(6, bufmeths.get_offset(1,1))
- command("bunload! 1")
- eq(-1, bufmeths.get_offset(1,1))
- end)
- end)
- describe('nvim_buf_get_var, nvim_buf_set_var, nvim_buf_del_var', function()
- it('works', function()
- curbuf('set_var', 'lua', {1, 2, {['3'] = 1}})
- eq({1, 2, {['3'] = 1}}, curbuf('get_var', 'lua'))
- eq({1, 2, {['3'] = 1}}, nvim('eval', 'b:lua'))
- eq(1, funcs.exists('b:lua'))
- curbufmeths.del_var('lua')
- eq(0, funcs.exists('b:lua'))
- eq( 'Key not found: lua', pcall_err(curbufmeths.del_var, 'lua'))
- curbufmeths.set_var('lua', 1)
- command('lockvar b:lua')
- eq('Key is locked: lua', pcall_err(curbufmeths.del_var, 'lua'))
- eq('Key is locked: lua', pcall_err(curbufmeths.set_var, 'lua', 1))
- eq('Key is read-only: changedtick',
- pcall_err(curbufmeths.del_var, 'changedtick'))
- eq('Key is read-only: changedtick',
- pcall_err(curbufmeths.set_var, 'changedtick', 1))
- end)
- end)
- describe('nvim_buf_get_changedtick', function()
- it('works', function()
- eq(2, curbufmeths.get_changedtick())
- curbufmeths.set_lines(0, 1, false, {'abc\0', '\0def', 'ghi'})
- eq(3, curbufmeths.get_changedtick())
- eq(3, curbufmeths.get_var('changedtick'))
- end)
- it('buffer_set_var returns the old value', function()
- local val1 = {1, 2, {['3'] = 1}}
- local val2 = {4, 7}
- eq(NIL, request('buffer_set_var', 0, 'lua', val1))
- eq(val1, request('buffer_set_var', 0, 'lua', val2))
- end)
- it('buffer_del_var returns the old value', function()
- local val1 = {1, 2, {['3'] = 1}}
- local val2 = {4, 7}
- eq(NIL, request('buffer_set_var', 0, 'lua', val1))
- eq(val1, request('buffer_set_var', 0, 'lua', val2))
- eq(val2, request('buffer_del_var', 0, 'lua'))
- end)
- end)
- describe('nvim_buf_get_option, nvim_buf_set_option', function()
- it('works', function()
- eq(8, curbuf('get_option', 'shiftwidth'))
- curbuf('set_option', 'shiftwidth', 4)
- eq(4, curbuf('get_option', 'shiftwidth'))
- -- global-local option
- curbuf('set_option', 'define', 'test')
- eq('test', curbuf('get_option', 'define'))
- -- Doesn't change the global value
- eq([[^\s*#\s*define]], nvim('get_option', 'define'))
- end)
- it('returns values for unset local options', function()
- -- 'undolevels' is only set to its "unset" value when a new buffer is
- -- created
- command('enew')
- eq(-123456, curbuf('get_option', 'undolevels'))
- end)
- end)
- describe('nvim_buf_get_name, nvim_buf_set_name', function()
- it('works', function()
- nvim('command', 'new')
- eq('', curbuf('get_name'))
- local new_name = nvim('eval', 'resolve(tempname())')
- curbuf('set_name', new_name)
- eq(new_name, curbuf('get_name'))
- nvim('command', 'w!')
- eq(1, funcs.filereadable(new_name))
- os.remove(new_name)
- end)
- end)
- describe('nvim_buf_is_loaded', function()
- it('works', function()
- -- record our buffer number for when we unload it
- local bufnr = curbuf('get_number')
- -- api should report that the buffer is loaded
- ok(buffer('is_loaded', bufnr))
- -- hide the current buffer by switching to a new empty buffer
- -- Careful! we need to modify the buffer first or vim will just reuse it
- buffer('set_lines', bufnr, 0, -1, 1, {'line1'})
- command('hide enew')
- -- confirm the buffer is hidden, but still loaded
- local infolist = nvim('eval', 'getbufinfo('..bufnr..')')
- eq(1, #infolist)
- eq(1, infolist[1].hidden)
- eq(1, infolist[1].loaded)
- -- now force unload the buffer
- command('bunload! '..bufnr)
- -- confirm the buffer is unloaded
- infolist = nvim('eval', 'getbufinfo('..bufnr..')')
- eq(0, infolist[1].loaded)
- -- nvim_buf_is_loaded() should also report the buffer as unloaded
- eq(false, buffer('is_loaded', bufnr))
- end)
- end)
- describe('nvim_buf_is_valid', function()
- it('works', function()
- nvim('command', 'new')
- local b = nvim('get_current_buf')
- ok(buffer('is_valid', b))
- nvim('command', 'bw!')
- ok(not buffer('is_valid', b))
- end)
- end)
- describe('nvim_buf_delete', function()
- it('allows for just deleting', function()
- nvim('command', 'new')
- local b = nvim('get_current_buf')
- ok(buffer('is_valid', b))
- nvim('buf_delete', b, {})
- ok(not buffer('is_loaded', b))
- ok(not buffer('is_valid', b))
- end)
- it('allows for just unloading', function()
- nvim('command', 'new')
- local b = nvim('get_current_buf')
- ok(buffer('is_valid', b))
- nvim('buf_delete', b, { unload = true })
- ok(not buffer('is_loaded', b))
- ok(buffer('is_valid', b))
- end)
- end)
- describe('nvim_buf_get_mark', function()
- it('works', function()
- curbuf('set_lines', -1, -1, true, {'a', 'bit of', 'text'})
- curwin('set_cursor', {3, 4})
- nvim('command', 'mark v')
- eq({3, 0}, curbuf('get_mark', 'v'))
- end)
- end)
- describe('nvim_buf_set_mark', function()
- it('works with buffer local marks', function()
- curbufmeths.set_lines(-1, -1, true, {'a', 'bit of', 'text'})
- eq(true, curbufmeths.set_mark('z', 1, 1, {}))
- eq({1, 1}, curbufmeths.get_mark('z'))
- end)
- it('works with file/uppercase marks', function()
- curbufmeths.set_lines(-1, -1, true, {'a', 'bit of', 'text'})
- eq(true, curbufmeths.set_mark('Z', 3, 1, {}))
- eq({3, 1}, curbufmeths.get_mark('Z'))
- end)
- it('fails when invalid marks names are used', function()
- eq(false, pcall(curbufmeths.set_mark, '!', 1, 0, {}))
- eq(false, pcall(curbufmeths.set_mark, 'fail', 1, 0, {}))
- end)
- it('fails when invalid buffer number is used', function()
- eq(false, pcall(meths.buf_set_mark, 99, 'a', 1, 1, {}))
- end)
- end)
- describe('nvim_buf_del_mark', function()
- it('works with buffer local marks', function()
- curbufmeths.set_lines(-1, -1, true, {'a', 'bit of', 'text'})
- curbufmeths.set_mark('z', 3, 1, {})
- eq(true, curbufmeths.del_mark('z'))
- eq({0, 0}, curbufmeths.get_mark('z'))
- end)
- it('works with file/uppercase marks', function()
- curbufmeths.set_lines(-1, -1, true, {'a', 'bit of', 'text'})
- curbufmeths.set_mark('Z', 3, 3, {})
- eq(true, curbufmeths.del_mark('Z'))
- eq({0, 0}, curbufmeths.get_mark('Z'))
- end)
- it('returns false in marks not set in this buffer', function()
- local abuf = meths.create_buf(false,true)
- bufmeths.set_lines(abuf, -1, -1, true, {'a', 'bit of', 'text'})
- bufmeths.set_mark(abuf, 'A', 2, 2, {})
- eq(false, curbufmeths.del_mark('A'))
- eq({2, 2}, bufmeths.get_mark(abuf, 'A'))
- end)
- it('returns false if mark was not deleted', function()
- curbufmeths.set_lines(-1, -1, true, {'a', 'bit of', 'text'})
- curbufmeths.set_mark('z', 3, 1, {})
- eq(true, curbufmeths.del_mark('z'))
- eq(false, curbufmeths.del_mark('z')) -- Mark was already deleted
- end)
- it('fails when invalid marks names are used', function()
- eq(false, pcall(curbufmeths.del_mark, '!'))
- eq(false, pcall(curbufmeths.del_mark, 'fail'))
- end)
- it('fails when invalid buffer number is used', function()
- eq(false, pcall(meths.buf_del_mark, 99, 'a'))
- end)
- end)
- end)
|