local t = require('test.testutil') local n = require('test.functional.testnvim')() local Screen = require('test.functional.ui.screen') local clear, curbuf, curbuf_contents, curwin, eq, neq, matches, ok, feed, insert, eval = n.clear, n.api.nvim_get_current_buf, n.curbuf_contents, n.api.nvim_get_current_win, t.eq, t.neq, t.matches, t.ok, n.feed, n.insert, n.eval local poke_eventloop = n.poke_eventloop local exec = n.exec local exec_lua = n.exec_lua local fn = n.fn local request = n.request local NIL = vim.NIL local api = n.api local command = n.command local pcall_err = t.pcall_err local assert_alive = n.assert_alive describe('API/win', function() before_each(clear) describe('get_buf', function() it('works', function() eq(curbuf(), api.nvim_win_get_buf(api.nvim_list_wins()[1])) command('new') api.nvim_set_current_win(api.nvim_list_wins()[2]) eq(curbuf(), api.nvim_win_get_buf(api.nvim_list_wins()[2])) neq( api.nvim_win_get_buf(api.nvim_list_wins()[1]), api.nvim_win_get_buf(api.nvim_list_wins()[2]) ) end) end) describe('set_buf', function() it('works', function() command('new') local windows = api.nvim_list_wins() neq(api.nvim_win_get_buf(windows[2]), api.nvim_win_get_buf(windows[1])) api.nvim_win_set_buf(windows[2], api.nvim_win_get_buf(windows[1])) eq(api.nvim_win_get_buf(windows[2]), api.nvim_win_get_buf(windows[1])) end) it('validates args', function() eq('Invalid buffer id: 23', pcall_err(api.nvim_win_set_buf, api.nvim_get_current_win(), 23)) eq('Invalid window id: 23', pcall_err(api.nvim_win_set_buf, 23, api.nvim_get_current_buf())) end) it('disallowed in cmdwin if win=cmdwin_{old_cur}win or buf=cmdwin_buf', function() local new_buf = api.nvim_create_buf(true, true) local old_win = api.nvim_get_current_win() local new_win = api.nvim_open_win(new_buf, false, { relative = 'editor', row = 10, col = 10, width = 50, height = 10, }) feed('q:') eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_set_buf, 0, new_buf) ) eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_set_buf, old_win, new_buf) ) eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_set_buf, new_win, 0) ) matches( 'E11: Invalid in command%-line window; executes, CTRL%-C quits$', pcall_err( exec_lua, [[ local cmdwin_buf = vim.api.nvim_get_current_buf() local new_win, new_buf = ... vim._with({buf = new_buf}, function() vim.api.nvim_win_set_buf(new_win, cmdwin_buf) end) ]], new_win, new_buf ) ) matches( 'E11: Invalid in command%-line window; executes, CTRL%-C quits$', pcall_err( exec_lua, [[ local cmdwin_win = vim.api.nvim_get_current_win() local new_win, new_buf = ... vim._with({win = new_win}, function() vim.api.nvim_win_set_buf(cmdwin_win, new_buf) end) ]], new_win, new_buf ) ) local next_buf = api.nvim_create_buf(true, true) api.nvim_win_set_buf(new_win, next_buf) eq(next_buf, api.nvim_win_get_buf(new_win)) end) describe("with 'autochdir'", function() local topdir local otherbuf local oldwin local newwin before_each(function() command('set shellslash') topdir = fn.getcwd() t.mkdir(topdir .. '/Xacd') t.mkdir(topdir .. '/Xacd/foo') otherbuf = api.nvim_create_buf(false, true) api.nvim_buf_set_name(otherbuf, topdir .. '/Xacd/baz.txt') command('set autochdir') command('edit Xacd/foo/bar.txt') eq(topdir .. '/Xacd/foo', fn.getcwd()) oldwin = api.nvim_get_current_win() command('vsplit') newwin = api.nvim_get_current_win() end) after_each(function() n.rmdir(topdir .. '/Xacd') end) it('does not change cwd with non-current window', function() api.nvim_win_set_buf(oldwin, otherbuf) eq(topdir .. '/Xacd/foo', fn.getcwd()) end) it('changes cwd with current window', function() api.nvim_win_set_buf(newwin, otherbuf) eq(topdir .. '/Xacd', fn.getcwd()) end) end) end) describe('{get,set}_cursor', function() it('works', function() eq({ 1, 0 }, api.nvim_win_get_cursor(0)) command('normal ityping\027o some text') eq('typing\n some text', curbuf_contents()) eq({ 2, 10 }, api.nvim_win_get_cursor(0)) api.nvim_win_set_cursor(0, { 2, 6 }) command('normal i dumb') eq('typing\n some dumb text', curbuf_contents()) end) it('no memory leak when using invalid window ID with invalid pos', function() eq('Invalid window id: 1', pcall_err(api.nvim_win_set_cursor, 1, { 'b\na' })) end) it('updates the screen, and also when the window is unfocused', function() local screen = Screen.new(30, 9) insert('prologue') feed('100o') insert('epilogue') local win = curwin() feed('gg') screen:expect { grid = [[ ^prologue | |*8 ]], } -- cursor position is at beginning eq({ 1, 0 }, api.nvim_win_get_cursor(win)) -- move cursor to end api.nvim_win_set_cursor(win, { 101, 0 }) screen:expect { grid = [[ |*7 ^epilogue | | ]], } -- move cursor to the beginning again api.nvim_win_set_cursor(win, { 1, 0 }) screen:expect { grid = [[ ^prologue | |*8 ]], } -- move focus to new window command('new') neq(win, curwin()) -- sanity check, cursor position is kept eq({ 1, 0 }, api.nvim_win_get_cursor(win)) screen:expect { grid = [[ ^ | {1:~ }|*2 {3:[No Name] }| prologue | |*2 {2:[No Name] [+] }| | ]], } -- move cursor to end api.nvim_win_set_cursor(win, { 101, 0 }) screen:expect { grid = [[ ^ | {1:~ }|*2 {3:[No Name] }| |*2 epilogue | {2:[No Name] [+] }| | ]], } -- move cursor to the beginning again api.nvim_win_set_cursor(win, { 1, 0 }) screen:expect { grid = [[ ^ | {1:~ }|*2 {3:[No Name] }| prologue | |*2 {2:[No Name] [+] }| | ]], } -- curwin didn't change back neq(win, curwin()) end) it('remembers what column it wants to be in', function() insert('first line') feed('o') insert('second line') feed('gg') poke_eventloop() -- let nvim process the 'gg' command -- cursor position is at beginning local win = curwin() eq({ 1, 0 }, api.nvim_win_get_cursor(win)) -- move cursor to column 5 api.nvim_win_set_cursor(win, { 1, 5 }) -- move down a line feed('j') poke_eventloop() -- let nvim process the 'j' command -- cursor is still in column 5 eq({ 2, 5 }, api.nvim_win_get_cursor(win)) end) it('updates cursorline and statusline ruler in non-current window', function() local screen = Screen.new(60, 8) command('set ruler') command('set cursorline') insert([[ aaa bbb ccc ddd]]) local oldwin = curwin() command('vsplit') screen:expect([[ aaa │aaa | bbb │bbb | ccc │ccc | {21:dd^d }│{21:ddd }| {1:~ }│{1:~ }|*2 {3:[No Name] [+] 4,3 All }{2:[No Name] [+] 4,3 All}| | ]]) api.nvim_win_set_cursor(oldwin, { 1, 0 }) screen:expect([[ aaa │{21:aaa }| bbb │bbb | ccc │ccc | {21:dd^d }│ddd | {1:~ }│{1:~ }|*2 {3:[No Name] [+] 4,3 All }{2:[No Name] [+] 1,1 All}| | ]]) end) it('updates cursorcolumn in non-current window', function() local screen = Screen.new(60, 8) command('set cursorcolumn') insert([[ aaa bbb ccc ddd]]) local oldwin = curwin() command('vsplit') screen:expect([[ aa{21:a} │aa{21:a} | bb{21:b} │bb{21:b} | cc{21:c} │cc{21:c} | dd^d │ddd | {1:~ }│{1:~ }|*2 {3:[No Name] [+] }{2:[No Name] [+] }| | ]]) api.nvim_win_set_cursor(oldwin, { 2, 0 }) screen:expect([[ aa{21:a} │{21:a}aa | bb{21:b} │bbb | cc{21:c} │{21:c}cc | dd^d │{21:d}dd | {1:~ }│{1:~ }|*2 {3:[No Name] [+] }{2:[No Name] [+] }| | ]]) end) end) describe('{get,set}_height', function() it('works', function() command('vsplit') eq( api.nvim_win_get_height(api.nvim_list_wins()[2]), api.nvim_win_get_height(api.nvim_list_wins()[1]) ) api.nvim_set_current_win(api.nvim_list_wins()[2]) command('split') eq( api.nvim_win_get_height(api.nvim_list_wins()[2]), math.floor(api.nvim_win_get_height(api.nvim_list_wins()[1]) / 2) ) api.nvim_win_set_height(api.nvim_list_wins()[2], 2) eq(2, api.nvim_win_get_height(api.nvim_list_wins()[2])) end) it('failure modes', function() command('split') eq('Invalid window id: 999999', pcall_err(api.nvim_win_set_height, 999999, 10)) eq( 'Wrong type for argument 2 when calling nvim_win_set_height, expecting Integer', pcall_err(api.nvim_win_set_height, 0, 0.9) ) end) it('correctly handles height=1', function() command('split') api.nvim_set_current_win(api.nvim_list_wins()[1]) api.nvim_win_set_height(api.nvim_list_wins()[2], 1) eq(1, api.nvim_win_get_height(api.nvim_list_wins()[2])) end) it('correctly handles height=1 with a winbar', function() command('set winbar=foobar') command('set winminheight=0') command('split') api.nvim_set_current_win(api.nvim_list_wins()[1]) api.nvim_win_set_height(api.nvim_list_wins()[2], 1) eq(1, api.nvim_win_get_height(api.nvim_list_wins()[2])) end) it('do not cause ml_get errors with foldmethod=expr #19989', function() insert([[ aaaaa bbbbb ccccc]]) command('set foldmethod=expr') exec([[ new let w = nvim_get_current_win() wincmd w call nvim_win_set_height(w, 5) ]]) feed('l') eq('', api.nvim_get_vvar('errmsg')) end) end) describe('{get,set}_width', function() it('works', function() command('split') eq( api.nvim_win_get_width(api.nvim_list_wins()[2]), api.nvim_win_get_width(api.nvim_list_wins()[1]) ) api.nvim_set_current_win(api.nvim_list_wins()[2]) command('vsplit') eq( api.nvim_win_get_width(api.nvim_list_wins()[2]), math.floor(api.nvim_win_get_width(api.nvim_list_wins()[1]) / 2) ) api.nvim_win_set_width(api.nvim_list_wins()[2], 2) eq(2, api.nvim_win_get_width(api.nvim_list_wins()[2])) end) it('failure modes', function() command('vsplit') eq('Invalid window id: 999999', pcall_err(api.nvim_win_set_width, 999999, 10)) eq( 'Wrong type for argument 2 when calling nvim_win_set_width, expecting Integer', pcall_err(api.nvim_win_set_width, 0, 0.9) ) end) it('do not cause ml_get errors with foldmethod=expr #19989', function() insert([[ aaaaa bbbbb ccccc]]) command('set foldmethod=expr') exec([[ vnew let w = nvim_get_current_win() wincmd w call nvim_win_set_width(w, 5) ]]) feed('l') eq('', api.nvim_get_vvar('errmsg')) end) end) describe('{get,set,del}_var', function() it('works', function() api.nvim_win_set_var(0, 'lua', { 1, 2, { ['3'] = 1 } }) eq({ 1, 2, { ['3'] = 1 } }, api.nvim_win_get_var(0, 'lua')) eq({ 1, 2, { ['3'] = 1 } }, api.nvim_eval('w:lua')) eq(1, fn.exists('w:lua')) api.nvim_win_del_var(0, 'lua') eq(0, fn.exists('w:lua')) eq('Key not found: lua', pcall_err(api.nvim_win_del_var, 0, 'lua')) api.nvim_win_set_var(0, 'lua', 1) command('lockvar w:lua') eq('Key is locked: lua', pcall_err(api.nvim_win_del_var, 0, 'lua')) eq('Key is locked: lua', pcall_err(api.nvim_win_set_var, 0, 'lua', 1)) end) it('window_set_var returns the old value', function() local val1 = { 1, 2, { ['3'] = 1 } } local val2 = { 4, 7 } eq(NIL, request('window_set_var', 0, 'lua', val1)) eq(val1, request('window_set_var', 0, 'lua', val2)) end) it('window_del_var returns the old value', function() local val1 = { 1, 2, { ['3'] = 1 } } local val2 = { 4, 7 } eq(NIL, request('window_set_var', 0, 'lua', val1)) eq(val1, request('window_set_var', 0, 'lua', val2)) eq(val2, request('window_del_var', 0, 'lua')) end) end) describe('nvim_get_option_value, nvim_set_option_value', function() it('works', function() api.nvim_set_option_value('colorcolumn', '4,3', {}) eq('4,3', api.nvim_get_option_value('colorcolumn', {})) command('set modified hidden') command('enew') -- edit new buffer, window option is preserved eq('4,3', api.nvim_get_option_value('colorcolumn', {})) -- global-local option api.nvim_set_option_value('statusline', 'window-status', { win = 0 }) eq('window-status', api.nvim_get_option_value('statusline', { win = 0 })) eq('', api.nvim_get_option_value('statusline', { scope = 'global' })) command('set modified') command('enew') -- global-local: not preserved in new buffer -- confirm local value was not copied eq('', api.nvim_get_option_value('statusline', { win = 0 })) eq('', eval('&l:statusline')) end) it('after switching windows #15390', function() command('tabnew') local tab1 = unpack(api.nvim_list_tabpages()) local win1 = unpack(api.nvim_tabpage_list_wins(tab1)) api.nvim_set_option_value('statusline', 'window-status', { win = win1 }) command('split') command('wincmd J') command('wincmd j') eq('window-status', api.nvim_get_option_value('statusline', { win = win1 })) assert_alive() end) it('returns values for unset local options', function() eq(-1, api.nvim_get_option_value('scrolloff', { win = 0, scope = 'local' })) end) end) describe('get_position', function() it('works', function() local height = api.nvim_win_get_height(api.nvim_list_wins()[1]) local width = api.nvim_win_get_width(api.nvim_list_wins()[1]) command('split') command('vsplit') eq({ 0, 0 }, api.nvim_win_get_position(api.nvim_list_wins()[1])) local vsplit_pos = math.floor(width / 2) local split_pos = math.floor(height / 2) local win2row, win2col = unpack(api.nvim_win_get_position(api.nvim_list_wins()[2])) local win3row, win3col = unpack(api.nvim_win_get_position(api.nvim_list_wins()[3])) eq(0, win2row) eq(0, win3col) ok(vsplit_pos - 1 <= win2col and win2col <= vsplit_pos + 1) ok(split_pos - 1 <= win3row and win3row <= split_pos + 1) end) end) describe('get_position', function() it('works', function() command('tabnew') command('vsplit') eq(api.nvim_win_get_tabpage(api.nvim_list_wins()[1]), api.nvim_list_tabpages()[1]) eq(api.nvim_win_get_tabpage(api.nvim_list_wins()[2]), api.nvim_list_tabpages()[2]) eq(api.nvim_win_get_tabpage(api.nvim_list_wins()[3]), api.nvim_list_tabpages()[2]) end) end) describe('get_number', function() it('works', function() local wins = api.nvim_list_wins() eq(1, api.nvim_win_get_number(wins[1])) command('split') local win1, win2 = unpack(api.nvim_list_wins()) eq(1, api.nvim_win_get_number(win1)) eq(2, api.nvim_win_get_number(win2)) command('wincmd J') eq(2, api.nvim_win_get_number(win1)) eq(1, api.nvim_win_get_number(win2)) command('tabnew') local win3 = api.nvim_list_wins()[3] -- First tab page eq(2, api.nvim_win_get_number(win1)) eq(1, api.nvim_win_get_number(win2)) -- Second tab page eq(1, api.nvim_win_get_number(win3)) end) end) describe('is_valid', function() it('works', function() command('split') local win = api.nvim_list_wins()[2] api.nvim_set_current_win(win) ok(api.nvim_win_is_valid(win)) command('close') ok(not api.nvim_win_is_valid(win)) end) end) describe('close', function() it('can close current window', function() local oldwin = api.nvim_get_current_win() command('split') local newwin = api.nvim_get_current_win() api.nvim_win_close(newwin, false) eq({ oldwin }, api.nvim_list_wins()) end) it('can close noncurrent window', function() local oldwin = api.nvim_get_current_win() command('split') local newwin = api.nvim_get_current_win() api.nvim_win_close(oldwin, false) eq({ newwin }, api.nvim_list_wins()) end) it("handles changed buffer when 'hidden' is unset", function() command('set nohidden') local oldwin = api.nvim_get_current_win() insert('text') command('new') local newwin = api.nvim_get_current_win() eq( 'Vim:E37: No write since last change (add ! to override)', pcall_err(api.nvim_win_close, oldwin, false) ) eq({ newwin, oldwin }, api.nvim_list_wins()) end) it('handles changed buffer with force', function() local oldwin = api.nvim_get_current_win() insert('text') command('new') local newwin = api.nvim_get_current_win() api.nvim_win_close(oldwin, true) eq({ newwin }, api.nvim_list_wins()) end) it('in cmdline-window #9767', function() command('split') eq(2, #api.nvim_list_wins()) local oldbuf = api.nvim_get_current_buf() local oldwin = api.nvim_get_current_win() local otherwin = api.nvim_open_win(0, false, { relative = 'editor', row = 10, col = 10, width = 10, height = 10, }) -- Open cmdline-window. feed('q:') eq(4, #api.nvim_list_wins()) eq(':', fn.getcmdwintype()) -- Not allowed to close previous window from cmdline-window. eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_close, oldwin, true) ) -- Closing other windows is fine. api.nvim_win_close(otherwin, true) eq(false, api.nvim_win_is_valid(otherwin)) -- Close cmdline-window. api.nvim_win_close(0, true) eq(2, #api.nvim_list_wins()) eq('', fn.getcmdwintype()) -- Closing curwin in context of a different window shouldn't close cmdwin. otherwin = api.nvim_open_win(0, false, { relative = 'editor', row = 10, col = 10, width = 10, height = 10, }) feed('q:') exec_lua( [[ vim._with({win = ...}, function() vim.api.nvim_win_close(0, true) end) ]], otherwin ) eq(false, api.nvim_win_is_valid(otherwin)) eq(':', fn.getcmdwintype()) -- Closing cmdwin in context of a non-previous window is still OK. otherwin = api.nvim_open_win(oldbuf, false, { relative = 'editor', row = 10, col = 10, width = 10, height = 10, }) exec_lua( [[ local otherwin, cmdwin = ... vim._with({win = otherwin}, function() vim.api.nvim_win_close(cmdwin, true) end) ]], otherwin, api.nvim_get_current_win() ) eq('', fn.getcmdwintype()) eq(true, api.nvim_win_is_valid(otherwin)) end) it('closing current (float) window of another tabpage #15313', function() command('tabedit') command('botright split') local prevwin = curwin() eq(2, eval('tabpagenr()')) local win = api.nvim_open_win(0, true, { relative = 'editor', row = 10, col = 10, width = 50, height = 10, }) local tab = eval('tabpagenr()') command('tabprevious') eq(1, eval('tabpagenr()')) api.nvim_win_close(win, false) eq(prevwin, api.nvim_tabpage_get_win(tab)) assert_alive() end) end) describe('hide', function() it('can hide current window', function() local oldwin = api.nvim_get_current_win() command('split') local newwin = api.nvim_get_current_win() api.nvim_win_hide(newwin) eq({ oldwin }, api.nvim_list_wins()) end) it('can hide noncurrent window', function() local oldwin = api.nvim_get_current_win() command('split') local newwin = api.nvim_get_current_win() api.nvim_win_hide(oldwin) eq({ newwin }, api.nvim_list_wins()) end) it('does not close the buffer', function() local oldwin = api.nvim_get_current_win() local oldbuf = api.nvim_get_current_buf() local buf = api.nvim_create_buf(true, false) local newwin = api.nvim_open_win(buf, true, { relative = 'win', row = 3, col = 3, width = 12, height = 3, }) api.nvim_win_hide(newwin) eq({ oldwin }, api.nvim_list_wins()) eq({ oldbuf, buf }, api.nvim_list_bufs()) end) it('deletes the buffer when bufhidden=wipe', function() local oldwin = api.nvim_get_current_win() local oldbuf = api.nvim_get_current_buf() local buf = api.nvim_create_buf(true, false) local newwin = api.nvim_open_win(buf, true, { relative = 'win', row = 3, col = 3, width = 12, height = 3, }) api.nvim_set_option_value('bufhidden', 'wipe', { buf = buf }) api.nvim_win_hide(newwin) eq({ oldwin }, api.nvim_list_wins()) eq({ oldbuf }, api.nvim_list_bufs()) end) it('in the cmdwin', function() feed('q:') -- Can close the cmdwin. api.nvim_win_hide(0) eq('', fn.getcmdwintype()) local old_buf = api.nvim_get_current_buf() local old_win = api.nvim_get_current_win() local other_win = api.nvim_open_win(0, false, { relative = 'win', row = 3, col = 3, width = 12, height = 3, }) feed('q:') -- Cannot close the previous window. eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_hide, old_win) ) -- Can close other windows. api.nvim_win_hide(other_win) eq(false, api.nvim_win_is_valid(other_win)) -- Closing curwin in context of a different window shouldn't close cmdwin. other_win = api.nvim_open_win(old_buf, false, { relative = 'editor', row = 10, col = 10, width = 10, height = 10, }) exec_lua( [[ vim._with({win = ...}, function() vim.api.nvim_win_hide(0) end) ]], other_win ) eq(false, api.nvim_win_is_valid(other_win)) eq(':', fn.getcmdwintype()) -- Closing cmdwin in context of a non-previous window is still OK. other_win = api.nvim_open_win(old_buf, false, { relative = 'editor', row = 10, col = 10, width = 10, height = 10, }) exec_lua( [[ local otherwin, cmdwin = ... vim._with({win = otherwin}, function() vim.api.nvim_win_hide(cmdwin) end) ]], other_win, api.nvim_get_current_win() ) eq('', fn.getcmdwintype()) eq(true, api.nvim_win_is_valid(other_win)) end) end) describe('text_height', function() it('validation', function() local X = api.nvim_get_vvar('maxcol') insert([[ aaa bbb ccc ddd eee]]) eq('Invalid window id: 23', pcall_err(api.nvim_win_text_height, 23, {})) eq('Line index out of bounds', pcall_err(api.nvim_win_text_height, 0, { start_row = 5 })) eq('Line index out of bounds', pcall_err(api.nvim_win_text_height, 0, { start_row = -6 })) eq('Line index out of bounds', pcall_err(api.nvim_win_text_height, 0, { end_row = 5 })) eq('Line index out of bounds', pcall_err(api.nvim_win_text_height, 0, { end_row = -6 })) eq( "'start_row' is higher than 'end_row'", pcall_err(api.nvim_win_text_height, 0, { start_row = 3, end_row = 1 }) ) eq( "'start_vcol' specified without 'start_row'", pcall_err(api.nvim_win_text_height, 0, { end_row = 2, start_vcol = 0 }) ) eq( "'end_vcol' specified without 'end_row'", pcall_err(api.nvim_win_text_height, 0, { start_row = 2, end_vcol = 0 }) ) eq( "Invalid 'start_vcol': out of range", pcall_err(api.nvim_win_text_height, 0, { start_row = 2, start_vcol = -1 }) ) eq( "Invalid 'start_vcol': out of range", pcall_err(api.nvim_win_text_height, 0, { start_row = 2, start_vcol = X + 1 }) ) eq( "Invalid 'end_vcol': out of range", pcall_err(api.nvim_win_text_height, 0, { end_row = 2, end_vcol = -1 }) ) eq( "Invalid 'end_vcol': out of range", pcall_err(api.nvim_win_text_height, 0, { end_row = 2, end_vcol = X + 1 }) ) eq( "'start_vcol' is higher than 'end_vcol'", pcall_err( api.nvim_win_text_height, 0, { start_row = 2, end_row = 2, start_vcol = 10, end_vcol = 5 } ) ) end) it('with two diff windows', function() local X = api.nvim_get_vvar('maxcol') local screen = Screen.new(45, 22) exec([[ set diffopt+=context:2 number let expr = 'printf("%08d", v:val) .. repeat("!", v:val)' call setline(1, map(range(1, 20) + range(25, 45), expr)) vnew call setline(1, map(range(3, 20) + range(28, 50), expr)) windo diffthis ]]) feed('24gg') screen:expect { grid = [[ {7: }{8: }{23:----------------}│{7: }{8: 1 }{22:00000001! }| {7: }{8: }{23:----------------}│{7: }{8: 2 }{22:00000002!! }| {7: }{8: 1 }00000003!!! │{7: }{8: 3 }00000003!!! | {7: }{8: 2 }00000004!!!! │{7: }{8: 4 }00000004!!!! | {7:+ }{8: 3 }{13:+-- 14 lines: 00}│{7:+ }{8: 5 }{13:+-- 14 lines: 00}| {7: }{8: 17 }00000019!!!!!!!!│{7: }{8: 19 }00000019!!!!!!!!| {7: }{8: 18 }00000020!!!!!!!!│{7: }{8: 20 }00000020!!!!!!!!| {7: }{8: }{23:----------------}│{7: }{8: 21 }{22:00000025!!!!!!!!}| {7: }{8: }{23:----------------}│{7: }{8: 22 }{22:00000026!!!!!!!!}| {7: }{8: }{23:----------------}│{7: }{8: 23 }{22:00000027!!!!!!!!}| {7: }{8: 19 }00000028!!!!!!!!│{7: }{8: 24 }^00000028!!!!!!!!| {7: }{8: 20 }00000029!!!!!!!!│{7: }{8: 25 }00000029!!!!!!!!| {7:+ }{8: 21 }{13:+-- 14 lines: 00}│{7:+ }{8: 26 }{13:+-- 14 lines: 00}| {7: }{8: 35 }00000044!!!!!!!!│{7: }{8: 40 }00000044!!!!!!!!| {7: }{8: 36 }00000045!!!!!!!!│{7: }{8: 41 }00000045!!!!!!!!| {7: }{8: 37 }{22:00000046!!!!!!!!}│{7: }{8: }{23:----------------}| {7: }{8: 38 }{22:00000047!!!!!!!!}│{7: }{8: }{23:----------------}| {7: }{8: 39 }{22:00000048!!!!!!!!}│{7: }{8: }{23:----------------}| {7: }{8: 40 }{22:00000049!!!!!!!!}│{7: }{8: }{23:----------------}| {7: }{8: 41 }{22:00000050!!!!!!!!}│{7: }{8: }{23:----------------}| {2:[No Name] [+] }{3:[No Name] [+] }| | ]], } screen:try_resize(45, 3) screen:expect { grid = [[ {7: }{8: 19 }00000028!!!!!!!!│{7: }{8: 24 }^00000028!!!!!!!!| {2:[No Name] [+] }{3:[No Name] [+] }| | ]], } eq({ all = 20, fill = 5 }, api.nvim_win_text_height(1000, {})) eq({ all = 20, fill = 5 }, api.nvim_win_text_height(1001, {})) eq({ all = 20, fill = 5 }, api.nvim_win_text_height(1000, { start_row = 0 })) eq({ all = 20, fill = 5 }, api.nvim_win_text_height(1001, { start_row = 0 })) eq({ all = 15, fill = 0 }, api.nvim_win_text_height(1000, { end_row = -1 })) eq({ all = 15, fill = 0 }, api.nvim_win_text_height(1000, { end_row = 40 })) eq({ all = 20, fill = 5 }, api.nvim_win_text_height(1001, { end_row = -1 })) eq({ all = 20, fill = 5 }, api.nvim_win_text_height(1001, { end_row = 40 })) eq({ all = 10, fill = 5 }, api.nvim_win_text_height(1000, { start_row = 23 })) eq({ all = 13, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 18 })) eq({ all = 11, fill = 0 }, api.nvim_win_text_height(1000, { end_row = 23 })) eq({ all = 11, fill = 5 }, api.nvim_win_text_height(1001, { end_row = 18 })) eq({ all = 11, fill = 0 }, api.nvim_win_text_height(1000, { start_row = 3, end_row = 39 })) eq({ all = 11, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 1, end_row = 34 })) eq({ all = 9, fill = 0 }, api.nvim_win_text_height(1000, { start_row = 4, end_row = 38 })) eq({ all = 9, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 2, end_row = 33 })) eq({ all = 9, fill = 0 }, api.nvim_win_text_height(1000, { start_row = 5, end_row = 37 })) eq({ all = 9, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 3, end_row = 32 })) eq({ all = 9, fill = 0 }, api.nvim_win_text_height(1000, { start_row = 17, end_row = 25 })) eq({ all = 9, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 15, end_row = 20 })) eq({ all = 7, fill = 0 }, api.nvim_win_text_height(1000, { start_row = 18, end_row = 24 })) eq({ all = 7, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 16, end_row = 19 })) eq({ all = 6, fill = 5 }, api.nvim_win_text_height(1000, { start_row = -1 })) eq({ all = 5, fill = 5 }, api.nvim_win_text_height(1000, { start_row = -1, start_vcol = X })) eq( { all = 0, fill = 0 }, api.nvim_win_text_height(1000, { start_row = -1, start_vcol = X, end_row = -1 }) ) eq( { all = 0, fill = 0 }, api.nvim_win_text_height( 1000, { start_row = -1, start_vcol = X, end_row = -1, end_vcol = X } ) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height( 1000, { start_row = -1, start_vcol = 0, end_row = -1, end_vcol = X } ) ) eq({ all = 3, fill = 2 }, api.nvim_win_text_height(1001, { end_row = 0 })) eq({ all = 2, fill = 2 }, api.nvim_win_text_height(1001, { end_row = 0, end_vcol = 0 })) eq( { all = 2, fill = 2 }, api.nvim_win_text_height(1001, { start_row = 0, end_row = 0, end_vcol = 0 }) ) eq( { all = 0, fill = 0 }, api.nvim_win_text_height(1001, { start_row = 0, start_vcol = 0, end_row = 0, end_vcol = 0 }) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height(1001, { start_row = 0, start_vcol = 0, end_row = 0, end_vcol = X }) ) eq({ all = 11, fill = 5 }, api.nvim_win_text_height(1001, { end_row = 18 })) eq( { all = 9, fill = 3 }, api.nvim_win_text_height(1001, { start_row = 0, start_vcol = 0, end_row = 18 }) ) eq({ all = 10, fill = 5 }, api.nvim_win_text_height(1001, { end_row = 18, end_vcol = 0 })) eq( { all = 8, fill = 3 }, api.nvim_win_text_height( 1001, { start_row = 0, start_vcol = 0, end_row = 18, end_vcol = 0 } ) ) end) it('with wrapped lines', function() local X = api.nvim_get_vvar('maxcol') local screen = Screen.new(45, 22) exec([[ set number cpoptions+=n call setline(1, repeat([repeat('foobar-', 36)], 3)) ]]) local ns = api.nvim_create_namespace('') api.nvim_buf_set_extmark( 0, ns, 1, 100, { virt_text = { { ('?'):rep(15), 'Search' } }, virt_text_pos = 'inline' } ) api.nvim_buf_set_extmark( 0, ns, 2, 200, { virt_text = { { ('!'):rep(75), 'Search' } }, virt_text_pos = 'inline' } ) screen:expect { grid = [[ {8: 1 }^foobar-foobar-foobar-foobar-foobar-foobar| -foobar-foobar-foobar-foobar-foobar-foobar-fo| obar-foobar-foobar-foobar-foobar-foobar-fooba| r-foobar-foobar-foobar-foobar-foobar-foobar-f| oobar-foobar-foobar-foobar-foobar-foobar-foob| ar-foobar-foobar-foobar-foobar- | {8: 2 }foobar-foobar-foobar-foobar-foobar-foobar| -foobar-foobar-foobar-foobar-foobar-foobar-fo| obar-foobar-fo{10:???????????????}obar-foobar-foob| ar-foobar-foobar-foobar-foobar-foobar-foobar-| foobar-foobar-foobar-foobar-foobar-foobar-foo| bar-foobar-foobar-foobar-foobar-foobar-foobar| - | {8: 3 }foobar-foobar-foobar-foobar-foobar-foobar| -foobar-foobar-foobar-foobar-foobar-foobar-fo| obar-foobar-foobar-foobar-foobar-foobar-fooba| r-foobar-foobar-foobar-foobar-foobar-foobar-f| oobar-foobar-foobar-foob{10:!!!!!!!!!!!!!!!!!!!!!}| {10:!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!}| {10:!!!!!!!!!}ar-foobar-foobar-foobar-foobar-fooba| r-foobar-foobar- | | ]], } screen:try_resize(45, 2) screen:expect { grid = [[ {8: 1 }^foobar-foobar-foobar-foobar-foobar-foobar| | ]], } eq({ all = 21, fill = 0 }, api.nvim_win_text_height(0, {})) eq({ all = 6, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, end_row = 0 })) eq({ all = 7, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, end_row = 1 })) eq({ all = 8, fill = 0 }, api.nvim_win_text_height(0, { start_row = 2, end_row = 2 })) eq( { all = 0, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 0 }) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 41 }) ) eq( { all = 2, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 42 }) ) eq( { all = 2, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 86 }) ) eq( { all = 3, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 87 }) ) eq( { all = 6, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 266 }) ) eq( { all = 7, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 267 }) ) eq( { all = 7, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 311 }) ) eq( { all = 7, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = 312 }) ) eq( { all = 7, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 0, end_row = 1, end_vcol = X }) ) eq( { all = 7, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 40, end_row = 1, end_vcol = X }) ) eq( { all = 6, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 41, end_row = 1, end_vcol = X }) ) eq( { all = 6, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 85, end_row = 1, end_vcol = X }) ) eq( { all = 5, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 86, end_row = 1, end_vcol = X }) ) eq( { all = 2, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 265, end_row = 1, end_vcol = X }) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 266, end_row = 1, end_vcol = X }) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 310, end_row = 1, end_vcol = X }) ) eq( { all = 0, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 311, end_row = 1, end_vcol = X }) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height(0, { start_row = 1, start_vcol = 86, end_row = 1, end_vcol = 131 }) ) eq( { all = 1, fill = 0 }, api.nvim_win_text_height( 0, { start_row = 1, start_vcol = 221, end_row = 1, end_vcol = 266 } ) ) eq({ all = 18, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, start_vcol = 131 })) eq({ all = 19, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, start_vcol = 130 })) eq({ all = 20, fill = 0 }, api.nvim_win_text_height(0, { end_row = 2, end_vcol = 311 })) eq({ all = 21, fill = 0 }, api.nvim_win_text_height(0, { end_row = 2, end_vcol = 312 })) eq( { all = 17, fill = 0 }, api.nvim_win_text_height( 0, { start_row = 0, start_vcol = 131, end_row = 2, end_vcol = 311 } ) ) eq( { all = 19, fill = 0 }, api.nvim_win_text_height( 0, { start_row = 0, start_vcol = 130, end_row = 2, end_vcol = 312 } ) ) eq({ all = 16, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, start_vcol = 221 })) eq({ all = 17, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, start_vcol = 220 })) eq({ all = 14, fill = 0 }, api.nvim_win_text_height(0, { end_row = 2, end_vcol = 41 })) eq({ all = 15, fill = 0 }, api.nvim_win_text_height(0, { end_row = 2, end_vcol = 42 })) eq( { all = 9, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, start_vcol = 221, end_row = 2, end_vcol = 41 }) ) eq( { all = 11, fill = 0 }, api.nvim_win_text_height(0, { start_row = 0, start_vcol = 220, end_row = 2, end_vcol = 42 }) ) end) end) describe('open_win', function() it('disallowed in cmdwin if enter=true or buf=cmdwin_buf', function() local new_buf = api.nvim_create_buf(true, true) feed('q:') eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_open_win, new_buf, true, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, }) ) eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_open_win, 0, false, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, }) ) matches( 'E11: Invalid in command%-line window; executes, CTRL%-C quits$', pcall_err( exec_lua, [[ local cmdwin_buf = vim.api.nvim_get_current_buf() vim._with({buf = vim.api.nvim_create_buf(false, true)}, function() vim.api.nvim_open_win(cmdwin_buf, false, { relative='editor', row=5, col=5, width=5, height=5, }) end) ]] ) ) eq( new_buf, api.nvim_win_get_buf(api.nvim_open_win(new_buf, false, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, })) ) end) it('aborts if buffer is invalid', function() local wins_before = api.nvim_list_wins() eq( 'Invalid buffer id: 1337', pcall_err(api.nvim_open_win, 1337, false, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, }) ) eq(wins_before, api.nvim_list_wins()) end) describe('creates a split window above', function() local function test_open_win_split_above(key, val) local initial_win = api.nvim_get_current_win() local win = api.nvim_open_win(0, true, { [key] = val, height = 10, }) eq('', api.nvim_win_get_config(win).relative) eq(10, api.nvim_win_get_height(win)) local layout = fn.winlayout() eq({ 'col', { { 'leaf', win }, { 'leaf', initial_win }, }, }, layout) end it("with split = 'above'", function() test_open_win_split_above('split', 'above') end) it("with vertical = false and 'nosplitbelow'", function() api.nvim_set_option_value('splitbelow', false, {}) test_open_win_split_above('vertical', false) end) end) describe('creates a split window below', function() local function test_open_win_split_below(key, val) local initial_win = api.nvim_get_current_win() local win = api.nvim_open_win(0, true, { [key] = val, height = 15, }) eq('', api.nvim_win_get_config(win).relative) eq(15, api.nvim_win_get_height(win)) local layout = fn.winlayout() eq({ 'col', { { 'leaf', initial_win }, { 'leaf', win }, }, }, layout) end it("with split = 'below'", function() test_open_win_split_below('split', 'below') end) it("with vertical = false and 'splitbelow'", function() api.nvim_set_option_value('splitbelow', true, {}) test_open_win_split_below('vertical', false) end) end) describe('creates a split window to the left', function() local function test_open_win_split_left(key, val) local initial_win = api.nvim_get_current_win() local win = api.nvim_open_win(0, true, { [key] = val, width = 25, }) eq('', api.nvim_win_get_config(win).relative) eq(25, api.nvim_win_get_width(win)) local layout = fn.winlayout() eq({ 'row', { { 'leaf', win }, { 'leaf', initial_win }, }, }, layout) end it("with split = 'left'", function() test_open_win_split_left('split', 'left') end) it("with vertical = true and 'nosplitright'", function() api.nvim_set_option_value('splitright', false, {}) test_open_win_split_left('vertical', true) end) end) describe('creates a split window to the right', function() local function test_open_win_split_right(key, val) local initial_win = api.nvim_get_current_win() local win = api.nvim_open_win(0, true, { [key] = val, width = 30, }) eq('', api.nvim_win_get_config(win).relative) eq(30, api.nvim_win_get_width(win)) local layout = fn.winlayout() eq({ 'row', { { 'leaf', initial_win }, { 'leaf', win }, }, }, layout) end it("with split = 'right'", function() test_open_win_split_right('split', 'right') end) it("with vertical = true and 'splitright'", function() api.nvim_set_option_value('splitright', true, {}) test_open_win_split_right('vertical', true) end) end) it("doesn't change tp_curwin when splitting window in another tab with enter=false", function() local tab1 = api.nvim_get_current_tabpage() local tab1_win = api.nvim_get_current_win() n.command('tabnew') local tab2 = api.nvim_get_current_tabpage() local tab2_win = api.nvim_get_current_win() eq({ tab1_win, tab2_win }, api.nvim_list_wins()) eq({ tab1, tab2 }, api.nvim_list_tabpages()) api.nvim_set_current_tabpage(tab1) eq(tab1_win, api.nvim_get_current_win()) local tab2_prevwin = fn.tabpagewinnr(tab2, '#') -- split in tab2 whine in tab2, with enter = false local tab2_win2 = api.nvim_open_win(api.nvim_create_buf(false, true), false, { win = tab2_win, split = 'right', }) eq(tab1_win, api.nvim_get_current_win()) -- we should still be in the first tp eq(tab1_win, api.nvim_tabpage_get_win(tab1)) eq(tab2_win, api.nvim_tabpage_get_win(tab2)) -- tab2's tp_curwin should not have changed eq(tab2_prevwin, fn.tabpagewinnr(tab2, '#')) -- tab2's tp_prevwin should not have changed eq({ tab1_win, tab2_win, tab2_win2 }, api.nvim_list_wins()) eq({ tab2_win, tab2_win2 }, api.nvim_tabpage_list_wins(tab2)) end) it('creates splits in the correct location', function() local first_win = api.nvim_get_current_win() -- specifying window 0 should create a split next to the current window local win = api.nvim_open_win(0, true, { vertical = false, }) local layout = fn.winlayout() eq({ 'col', { { 'leaf', win }, { 'leaf', first_win }, }, }, layout) -- not specifying a window should create a top-level split local win2 = api.nvim_open_win(0, true, { split = 'left', win = -1, }) layout = fn.winlayout() eq({ 'row', { { 'leaf', win2 }, { 'col', { { 'leaf', win }, { 'leaf', first_win }, }, }, }, }, layout) -- specifying a window should create a split next to that window local win3 = api.nvim_open_win(0, true, { win = win, vertical = false, }) layout = fn.winlayout() eq({ 'row', { { 'leaf', win2 }, { 'col', { { 'leaf', win3 }, { 'leaf', win }, { 'leaf', first_win }, }, }, }, }, layout) end) it('opens floating windows in other tabpages', function() local first_win = api.nvim_get_current_win() local first_tab = api.nvim_get_current_tabpage() command('tabnew') local new_tab = api.nvim_get_current_tabpage() local win = api.nvim_open_win(0, false, { relative = 'win', win = first_win, width = 5, height = 5, row = 1, col = 1, }) eq(api.nvim_win_get_tabpage(win), first_tab) eq(api.nvim_get_current_tabpage(), new_tab) end) it('switches to new windows in non-current tabpages when enter=true', function() local first_win = api.nvim_get_current_win() local first_tab = api.nvim_get_current_tabpage() command('tabnew') local win = api.nvim_open_win(0, true, { relative = 'win', win = first_win, width = 5, height = 5, row = 1, col = 1, }) eq(api.nvim_win_get_tabpage(win), first_tab) eq(api.nvim_get_current_tabpage(), first_tab) end) local function setup_tabbed_autocmd_test() local info = {} info.orig_buf = api.nvim_get_current_buf() info.other_buf = api.nvim_create_buf(true, true) info.tab1_curwin = api.nvim_get_current_win() info.tab1 = api.nvim_get_current_tabpage() command('tab split | split') info.tab2_curwin = api.nvim_get_current_win() info.tab2 = api.nvim_get_current_tabpage() exec([=[ tabfirst let result = [] autocmd TabEnter * let result += [["TabEnter", nvim_get_current_tabpage()]] autocmd TabLeave * let result += [["TabLeave", nvim_get_current_tabpage()]] autocmd WinEnter * let result += [["WinEnter", win_getid()]] autocmd WinLeave * let result += [["WinLeave", win_getid()]] autocmd WinNew * let result += [["WinNew", win_getid()]] autocmd WinClosed * let result += [["WinClosed", str2nr(expand(""))]] autocmd BufEnter * let result += [["BufEnter", win_getid(), bufnr()]] autocmd BufLeave * let result += [["BufLeave", win_getid(), bufnr()]] autocmd BufWinEnter * let result += [["BufWinEnter", win_getid(), bufnr()]] autocmd BufWinLeave * let result += [["BufWinLeave", win_getid(), bufnr()]] ]=]) return info end it('noautocmd option works', function() local info = setup_tabbed_autocmd_test() api.nvim_open_win( info.other_buf, true, { split = 'left', win = info.tab2_curwin, noautocmd = true } ) eq({}, eval('result')) api.nvim_open_win( info.orig_buf, true, { relative = 'editor', row = 0, col = 0, width = 10, height = 10, noautocmd = true } ) eq({}, eval('result')) end) it('fires expected autocmds when creating splits without entering', function() local info = setup_tabbed_autocmd_test() -- For these, don't want BufWinEnter if visiting the same buffer, like :{s}buffer. -- Same tabpage, same buffer. local new_win = api.nvim_open_win(0, false, { split = 'left', win = info.tab1_curwin }) eq({ { 'WinNew', new_win }, }, eval('result')) eq(info.tab1_curwin, api.nvim_get_current_win()) -- Other tabpage, same buffer. command('let result = []') new_win = api.nvim_open_win(0, false, { split = 'left', win = info.tab2_curwin }) eq({ { 'WinNew', new_win }, }, eval('result')) eq(info.tab1_curwin, api.nvim_get_current_win()) -- Same tabpage, other buffer. command('let result = []') new_win = api.nvim_open_win(info.other_buf, false, { split = 'left', win = info.tab1_curwin }) eq({ { 'WinNew', new_win }, { 'BufWinEnter', new_win, info.other_buf }, }, eval('result')) eq(info.tab1_curwin, api.nvim_get_current_win()) -- Other tabpage, other buffer. command('let result = []') new_win = api.nvim_open_win(info.other_buf, false, { split = 'left', win = info.tab2_curwin }) eq({ { 'WinNew', new_win }, { 'BufWinEnter', new_win, info.other_buf }, }, eval('result')) eq(info.tab1_curwin, api.nvim_get_current_win()) end) it('fires expected autocmds when creating and entering splits', function() local info = setup_tabbed_autocmd_test() -- Same tabpage, same buffer. local new_win = api.nvim_open_win(0, true, { split = 'left', win = info.tab1_curwin }) eq({ { 'WinNew', new_win }, { 'WinLeave', info.tab1_curwin }, { 'WinEnter', new_win }, }, eval('result')) -- Same tabpage, other buffer. api.nvim_set_current_win(info.tab1_curwin) command('let result = []') new_win = api.nvim_open_win(info.other_buf, true, { split = 'left', win = info.tab1_curwin }) eq({ { 'WinNew', new_win }, { 'WinLeave', info.tab1_curwin }, { 'WinEnter', new_win }, { 'BufLeave', new_win, info.orig_buf }, { 'BufEnter', new_win, info.other_buf }, { 'BufWinEnter', new_win, info.other_buf }, }, eval('result')) -- For these, the other tabpage's prevwin and curwin will change like we switched from its old -- curwin to the new window, so the extra events near TabEnter reflect that. -- Other tabpage, same buffer. api.nvim_set_current_win(info.tab1_curwin) command('let result = []') new_win = api.nvim_open_win(0, true, { split = 'left', win = info.tab2_curwin }) eq({ { 'WinNew', new_win }, { 'WinLeave', info.tab1_curwin }, { 'TabLeave', info.tab1 }, { 'WinEnter', info.tab2_curwin }, { 'TabEnter', info.tab2 }, { 'WinLeave', info.tab2_curwin }, { 'WinEnter', new_win }, }, eval('result')) -- Other tabpage, other buffer. api.nvim_set_current_win(info.tab2_curwin) api.nvim_set_current_win(info.tab1_curwin) command('let result = []') new_win = api.nvim_open_win(info.other_buf, true, { split = 'left', win = info.tab2_curwin }) eq({ { 'WinNew', new_win }, { 'WinLeave', info.tab1_curwin }, { 'TabLeave', info.tab1 }, { 'WinEnter', info.tab2_curwin }, { 'TabEnter', info.tab2 }, { 'WinLeave', info.tab2_curwin }, { 'WinEnter', new_win }, { 'BufLeave', new_win, info.orig_buf }, { 'BufEnter', new_win, info.other_buf }, { 'BufWinEnter', new_win, info.other_buf }, }, eval('result')) -- Other tabpage, other buffer; but other tabpage's curwin has a new buffer active. api.nvim_set_current_win(info.tab2_curwin) local new_buf = api.nvim_create_buf(true, true) api.nvim_set_current_buf(new_buf) api.nvim_set_current_win(info.tab1_curwin) command('let result = []') new_win = api.nvim_open_win(info.other_buf, true, { split = 'left', win = info.tab2_curwin }) eq({ { 'WinNew', new_win }, { 'BufLeave', info.tab1_curwin, info.orig_buf }, { 'WinLeave', info.tab1_curwin }, { 'TabLeave', info.tab1 }, { 'WinEnter', info.tab2_curwin }, { 'TabEnter', info.tab2 }, { 'BufEnter', info.tab2_curwin, new_buf }, { 'WinLeave', info.tab2_curwin }, { 'WinEnter', new_win }, { 'BufLeave', new_win, new_buf }, { 'BufEnter', new_win, info.other_buf }, { 'BufWinEnter', new_win, info.other_buf }, }, eval('result')) end) it('OK when new window is moved to other tabpage by autocommands', function() -- Use nvim_win_set_config in the autocommands, as other methods of moving a window to a -- different tabpage (e.g: wincmd T) actually creates a new window. local tab0 = api.nvim_get_current_tabpage() local tab0_win = api.nvim_get_current_win() command('tabnew') local new_buf = api.nvim_create_buf(true, true) local tab1 = api.nvim_get_current_tabpage() local tab1_parent = api.nvim_get_current_win() command( 'tabfirst | autocmd WinNew * ++once call nvim_win_set_config(0, #{split: "left", win: ' .. tab1_parent .. '})' ) local new_win = api.nvim_open_win(new_buf, true, { split = 'left' }) eq(tab1, api.nvim_get_current_tabpage()) eq(new_win, api.nvim_get_current_win()) eq(new_buf, api.nvim_get_current_buf()) -- nvim_win_set_config called after entering. It doesn't follow a curwin that is moved to a -- different tabpage, but instead moves to the win filling the space, which is tab0_win. command( 'tabfirst | autocmd WinEnter * ++once call nvim_win_set_config(0, #{split: "left", win: ' .. tab1_parent .. '})' ) new_win = api.nvim_open_win(new_buf, true, { split = 'left' }) eq(tab0, api.nvim_get_current_tabpage()) eq(tab0_win, api.nvim_get_current_win()) eq(tab1, api.nvim_win_get_tabpage(new_win)) eq(new_buf, api.nvim_win_get_buf(new_win)) command( 'tabfirst | autocmd BufEnter * ++once call nvim_win_set_config(0, #{split: "left", win: ' .. tab1_parent .. '})' ) new_win = api.nvim_open_win(new_buf, true, { split = 'left' }) eq(tab0, api.nvim_get_current_tabpage()) eq(tab0_win, api.nvim_get_current_win()) eq(tab1, api.nvim_win_get_tabpage(new_win)) eq(new_buf, api.nvim_win_get_buf(new_win)) end) it('does not fire BufWinEnter if win_set_buf fails', function() exec([[ set nohidden modified autocmd WinNew * ++once only! let fired = v:false autocmd BufWinEnter * ++once let fired = v:true ]]) eq( 'Vim:E37: No write since last change (add ! to override)', pcall_err(api.nvim_open_win, api.nvim_create_buf(true, true), false, { split = 'left' }) ) eq(false, eval('fired')) end) it('fires Buf* autocommands when `!enter` if window is entered via autocommands', function() exec([[ autocmd WinNew * ++once only! let fired = v:false autocmd BufEnter * ++once let fired = v:true ]]) api.nvim_open_win(api.nvim_create_buf(true, true), false, { split = 'left' }) eq(true, eval('fired')) end) it('no heap-use-after-free if target buffer deleted by autocommands', function() local cur_buf = api.nvim_get_current_buf() local new_buf = api.nvim_create_buf(true, true) command('autocmd WinNew * ++once call nvim_buf_delete(' .. new_buf .. ', #{force: 1})') api.nvim_open_win(new_buf, true, { split = 'left' }) eq(cur_buf, api.nvim_get_current_buf()) end) it('checks if splitting disallowed', function() command('split | autocmd WinEnter * ++once call nvim_open_win(0, 0, #{split: "right"})') matches("E242: Can't split a window while closing another$", pcall_err(command, 'quit')) command('only | autocmd BufHidden * ++once call nvim_open_win(0, 0, #{split: "left"})') matches( 'E1159: Cannot split a window when closing the buffer$', pcall_err(command, 'new | quit') ) local w = api.nvim_get_current_win() command( 'only | new | autocmd BufHidden * ++once call nvim_open_win(0, 0, #{split: "left", win: ' .. w .. '})' ) matches( 'E1159: Cannot split a window when closing the buffer$', pcall_err(api.nvim_win_close, w, true) ) -- OK when using window to different buffer than `win`s. w = api.nvim_get_current_win() command( 'only | autocmd BufHidden * ++once call nvim_open_win(0, 0, #{split: "left", win: ' .. w .. '})' ) command('new | quit') end) it('restores last known cursor position if BufWinEnter did not move it', function() -- This test mostly exists to ensure BufWinEnter is executed before enter_buffer's epilogue. local buf = api.nvim_get_current_buf() insert([[ foo bar baz .etc i love autocommand bugs! supercalifragilisticexpialidocious marvim is actually a human llanfairpwllgwyngyllgogerychwyrndrobwllllantysiliogogogoch ]]) api.nvim_win_set_cursor(0, { 5, 2 }) command('set nostartofline | enew') local new_win = api.nvim_open_win(buf, false, { split = 'left' }) eq({ 5, 2 }, api.nvim_win_get_cursor(new_win)) exec([[ only! autocmd BufWinEnter * ++once normal! j6l ]]) new_win = api.nvim_open_win(buf, false, { split = 'left' }) eq({ 2, 6 }, api.nvim_win_get_cursor(new_win)) end) it('does not block all win_set_buf autocommands if !enter and !noautocmd', function() local new_buf = fn.bufadd('foobarbaz') exec([[ let triggered = "" autocmd BufReadCmd * ++once let triggered = bufname() ]]) api.nvim_open_win(new_buf, false, { split = 'left' }) eq('foobarbaz', eval('triggered')) end) it('sets error when no room', function() matches('E36: Not enough room$', pcall_err(command, 'execute "split|"->repeat(&lines)')) matches( 'E36: Not enough room$', pcall_err(api.nvim_open_win, 0, true, { split = 'above', win = 0 }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_open_win, 0, true, { split = 'below', win = 0 }) ) end) describe("with 'autochdir'", function() local topdir local otherbuf before_each(function() command('set shellslash') topdir = fn.getcwd() t.mkdir(topdir .. '/Xacd') t.mkdir(topdir .. '/Xacd/foo') otherbuf = api.nvim_create_buf(false, true) api.nvim_buf_set_name(otherbuf, topdir .. '/Xacd/baz.txt') command('set autochdir') command('edit Xacd/foo/bar.txt') eq(topdir .. '/Xacd/foo', fn.getcwd()) end) after_each(function() n.rmdir(topdir .. '/Xacd') end) it('does not change cwd with enter=false #15280', function() api.nvim_open_win( otherbuf, false, { relative = 'editor', height = 5, width = 5, row = 5, col = 5 } ) eq(topdir .. '/Xacd/foo', fn.getcwd()) end) it('changes cwd with enter=true', function() api.nvim_open_win( otherbuf, true, { relative = 'editor', height = 5, width = 5, row = 5, col = 5 } ) eq(topdir .. '/Xacd', fn.getcwd()) end) end) it('no memory leak with valid title and invalid footer', function() eq( 'title/footer must be string or array', pcall_err(api.nvim_open_win, 0, false, { relative = 'editor', row = 10, col = 10, height = 10, width = 10, border = 'single', title = { { 'TITLE' } }, footer = 0, }) ) end) it('no memory leak with invalid title and valid footer', function() eq( 'title/footer must be string or array', pcall_err(api.nvim_open_win, 0, false, { relative = 'editor', row = 10, col = 10, height = 10, width = 10, border = 'single', title = 0, footer = { { 'FOOTER' } }, }) ) end) end) describe('set_config', function() it('moves a split into a float', function() local win = api.nvim_open_win(0, true, { vertical = false, }) eq('', api.nvim_win_get_config(win).relative) api.nvim_win_set_config(win, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, }) eq('editor', api.nvim_win_get_config(win).relative) end) it('throws error when attempting to move the last window', function() local err = pcall_err(api.nvim_win_set_config, 0, { vertical = false, }) eq('Cannot move last window', err) end) it('passing retval of get_config results in no-op', function() -- simple split layout local win = api.nvim_open_win(0, true, { split = 'left', }) local layout = fn.winlayout() local config = api.nvim_win_get_config(win) api.nvim_win_set_config(win, config) eq(layout, fn.winlayout()) -- nested split layout local win2 = api.nvim_open_win(0, true, { vertical = true, }) local win3 = api.nvim_open_win(0, true, { win = win2, vertical = false, }) layout = fn.winlayout() config = api.nvim_win_get_config(win2) api.nvim_win_set_config(win2, config) eq(layout, fn.winlayout()) config = api.nvim_win_get_config(win3) api.nvim_win_set_config(win3, config) eq(layout, fn.winlayout()) end) it('moves a float into a split', function() local layout = fn.winlayout() eq('leaf', layout[1]) local win = api.nvim_open_win(0, true, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, }) api.nvim_win_set_config(win, { split = 'below', win = -1, }) eq('', api.nvim_win_get_config(win).relative) layout = fn.winlayout() eq('col', layout[1]) eq(2, #layout[2]) eq(win, layout[2][2][2]) end) it('respects the "split" option', function() local layout = fn.winlayout() eq('leaf', layout[1]) local first_win = layout[2] local win = api.nvim_open_win(0, true, { relative = 'editor', row = 5, col = 5, width = 5, height = 5, }) api.nvim_win_set_config(win, { split = 'right', win = first_win, }) layout = fn.winlayout() eq('row', layout[1]) eq(2, #layout[2]) eq(win, layout[2][2][2]) local config = api.nvim_win_get_config(win) eq('', config.relative) eq('right', config.split) api.nvim_win_set_config(win, { split = 'below', win = first_win, }) layout = fn.winlayout() eq('col', layout[1]) eq(2, #layout[2]) eq(win, layout[2][2][2]) config = api.nvim_win_get_config(win) eq('', config.relative) eq('below', config.split) eq( "non-float with 'win' requires at least 'split' or 'vertical'", pcall_err(api.nvim_win_set_config, 0, { win = 0 }) ) eq( "non-float with 'win' requires at least 'split' or 'vertical'", pcall_err(api.nvim_win_set_config, 0, { win = 0, relative = '' }) ) end) it('creates top-level splits', function() local win = api.nvim_open_win(0, true, { vertical = false, }) local win2 = api.nvim_open_win(0, true, { vertical = true, win = -1, }) local layout = fn.winlayout() eq('row', layout[1]) eq(2, #layout[2]) eq(win2, layout[2][1][2]) api.nvim_win_set_config(win, { split = 'below', win = -1, }) layout = fn.winlayout() eq('col', layout[1]) eq(2, #layout[2]) eq('row', layout[2][1][1]) eq(win, layout[2][2][2]) end) it('moves splits to other tabpages', function() local curtab = api.nvim_get_current_tabpage() local win = api.nvim_open_win(0, false, { split = 'left' }) command('tabnew') local tabnr = api.nvim_get_current_tabpage() command('tabprev') -- return to the initial tab api.nvim_win_set_config(win, { split = 'right', win = api.nvim_tabpage_get_win(tabnr), }) eq(tabnr, api.nvim_win_get_tabpage(win)) -- we are changing the config, the current tabpage should not change eq(curtab, api.nvim_get_current_tabpage()) command('tabnext') -- switch to the new tabpage so we can get the layout local layout = fn.winlayout() eq({ 'row', { { 'leaf', api.nvim_tabpage_get_win(tabnr) }, { 'leaf', win }, }, }, layout) end) it('correctly moves curwin when moving curwin to a different tabpage', function() local curtab = api.nvim_get_current_tabpage() command('tabnew') local tab2 = api.nvim_get_current_tabpage() local tab2_win = api.nvim_get_current_win() command('tabprev') -- return to the initial tab local neighbor = api.nvim_get_current_win() -- create and enter a new split local win = api.nvim_open_win(0, true, { vertical = false, }) eq(curtab, api.nvim_win_get_tabpage(win)) eq({ win, neighbor }, api.nvim_tabpage_list_wins(curtab)) -- move the current win to a different tabpage api.nvim_win_set_config(win, { split = 'right', win = api.nvim_tabpage_get_win(tab2), }) eq(curtab, api.nvim_get_current_tabpage()) -- win should have moved to tab2 eq(tab2, api.nvim_win_get_tabpage(win)) -- tp_curwin of tab2 should not have changed eq(tab2_win, api.nvim_tabpage_get_win(tab2)) -- win lists should be correct eq({ tab2_win, win }, api.nvim_tabpage_list_wins(tab2)) eq({ neighbor }, api.nvim_tabpage_list_wins(curtab)) -- current win should have moved to neighboring win eq(neighbor, api.nvim_tabpage_get_win(curtab)) end) it('splits windows in non-current tabpage', function() local curtab = api.nvim_get_current_tabpage() command('tabnew') local tabnr = api.nvim_get_current_tabpage() command('tabprev') -- return to the initial tab local win = api.nvim_open_win(0, false, { vertical = false, win = api.nvim_tabpage_get_win(tabnr), }) eq(tabnr, api.nvim_win_get_tabpage(win)) -- since enter = false, the current tabpage should not change eq(curtab, api.nvim_get_current_tabpage()) end) it('moves the current split window', function() local initial_win = api.nvim_get_current_win() local win = api.nvim_open_win(0, true, { vertical = true, }) local win2 = api.nvim_open_win(0, true, { vertical = true, }) api.nvim_set_current_win(win) eq({ 'row', { { 'leaf', win2 }, { 'leaf', win }, { 'leaf', initial_win }, }, }, fn.winlayout()) api.nvim_win_set_config(0, { vertical = false, win = 0, }) eq(win, api.nvim_get_current_win()) eq({ 'col', { { 'leaf', win }, { 'row', { { 'leaf', win2 }, { 'leaf', initial_win }, }, }, }, }, fn.winlayout()) api.nvim_set_current_win(win2) local win3 = api.nvim_open_win(0, true, { vertical = true, }) eq(win3, api.nvim_get_current_win()) eq({ 'col', { { 'leaf', win }, { 'row', { { 'leaf', win3 }, { 'leaf', win2 }, { 'leaf', initial_win }, }, }, }, }, fn.winlayout()) api.nvim_win_set_config(0, { vertical = false, win = 0, }) eq(win3, api.nvim_get_current_win()) eq({ 'col', { { 'leaf', win }, { 'row', { { 'col', { { 'leaf', win3 }, { 'leaf', win2 }, }, }, { 'leaf', initial_win }, }, }, }, }, fn.winlayout()) end) it('closing new curwin when moving window to other tabpage works', function() command('split | tabnew') local t2_win = api.nvim_get_current_win() command('tabfirst | autocmd WinEnter * ++once quit') local t1_move_win = api.nvim_get_current_win() -- win_set_config fails to switch away from "t1_move_win" because the WinEnter autocmd that -- closed the window we're switched to returns us to "t1_move_win", as it filled the space. eq( 'Failed to switch away from window ' .. t1_move_win, pcall_err(api.nvim_win_set_config, t1_move_win, { win = t2_win, split = 'left' }) ) eq(t1_move_win, api.nvim_get_current_win()) command('split | split | autocmd WinEnter * ++once quit') t1_move_win = api.nvim_get_current_win() -- In this case, we closed the window that we got switched to, but doing so didn't switch us -- back to "t1_move_win", which is fine. api.nvim_win_set_config(t1_move_win, { win = t2_win, split = 'left' }) neq(t1_move_win, api.nvim_get_current_win()) end) it('messing with "win" or "parent" when moving "win" to other tabpage', function() command('split | tabnew') local t2 = api.nvim_get_current_tabpage() local t2_win1 = api.nvim_get_current_win() command('split') local t2_win2 = api.nvim_get_current_win() command('split') local t2_win3 = api.nvim_get_current_win() command('tabfirst | autocmd WinEnter * ++once call nvim_win_close(' .. t2_win1 .. ', 1)') local cur_win = api.nvim_get_current_win() eq( 'Windows to split were closed', pcall_err(api.nvim_win_set_config, 0, { win = t2_win1, split = 'left' }) ) eq(cur_win, api.nvim_get_current_win()) command('split | autocmd WinLeave * ++once quit!') cur_win = api.nvim_get_current_win() eq( 'Windows to split were closed', pcall_err(api.nvim_win_set_config, 0, { win = t2_win2, split = 'left' }) ) neq(cur_win, api.nvim_get_current_win()) exec([[ split autocmd WinLeave * ++once \ call nvim_win_set_config(0, #{relative:'editor', row:0, col:0, width:5, height:5}) ]]) cur_win = api.nvim_get_current_win() eq( 'Floating state of windows to split changed', pcall_err(api.nvim_win_set_config, 0, { win = t2_win3, split = 'left' }) ) eq('editor', api.nvim_win_get_config(0).relative) eq(cur_win, api.nvim_get_current_win()) command('autocmd WinLeave * ++once wincmd J') cur_win = api.nvim_get_current_win() eq( 'Floating state of windows to split changed', pcall_err(api.nvim_win_set_config, 0, { win = t2_win3, split = 'left' }) ) eq('', api.nvim_win_get_config(0).relative) eq(cur_win, api.nvim_get_current_win()) -- Try to make "parent" floating. This should give the same error as before, but because -- changing a split from another tabpage into a float isn't supported yet, check for that -- error instead for now. -- Use ":silent!" to avoid the one second delay from printing the error message. exec(([[ autocmd WinLeave * ++once silent! \ call nvim_win_set_config(%d, #{relative:'editor', row:0, col:0, width:5, height:5}) ]]):format(t2_win3)) cur_win = api.nvim_get_current_win() api.nvim_win_set_config(0, { win = t2_win3, split = 'left' }) matches( 'Cannot change window from different tabpage into float$', api.nvim_get_vvar('errmsg') ) -- The error doesn't abort moving the window (or maybe it should, if that's wanted?) neq(cur_win, api.nvim_get_current_win()) eq(t2, api.nvim_win_get_tabpage(cur_win)) end) it('expected autocmds when moving window to other tabpage', function() local new_curwin = api.nvim_get_current_win() command('split') local win = api.nvim_get_current_win() command('tabnew') local parent = api.nvim_get_current_win() exec([[ tabfirst let result = [] autocmd WinEnter * let result += ["Enter", win_getid()] autocmd WinLeave * let result += ["Leave", win_getid()] autocmd WinNew * let result += ["New", win_getid()] ]]) api.nvim_win_set_config(0, { win = parent, split = 'left' }) -- Shouldn't see WinNew, as we're not creating any new windows, just moving existing ones. eq({ 'Leave', win, 'Enter', new_curwin }, eval('result')) end) it('no autocmds when moving window within same tabpage', function() local parent = api.nvim_get_current_win() exec([[ split let result = [] autocmd WinEnter * let result += ["Enter", win_getid()] autocmd WinLeave * let result += ["Leave", win_getid()] autocmd WinNew * let result += ["New", win_getid()] ]]) api.nvim_win_set_config(0, { win = parent, split = 'left' }) -- Shouldn't see any of those events, as we remain in the same window. eq({}, eval('result')) end) it('checks if splitting disallowed', function() command('split | autocmd WinEnter * ++once call nvim_win_set_config(0, #{split: "right"})') matches("E242: Can't split a window while closing another$", pcall_err(command, 'quit')) command('autocmd BufHidden * ++once call nvim_win_set_config(0, #{split: "left"})') matches( 'E1159: Cannot split a window when closing the buffer$', pcall_err(command, 'new | quit') ) -- OK when using window to different buffer. local w = api.nvim_get_current_win() command('autocmd BufHidden * ++once call nvim_win_set_config(' .. w .. ', #{split: "left"})') command('new | quit') end) --- Returns a function to get information about the window layout, sizes and positions of a --- tabpage. local function define_tp_info_function() exec_lua([[ function tp_info(tp) return { layout = vim.fn.winlayout(vim.api.nvim_tabpage_get_number(tp)), pos_sizes = vim.tbl_map( function(w) local pos = vim.fn.win_screenpos(w) return { row = pos[1], col = pos[2], width = vim.fn.winwidth(w), height = vim.fn.winheight(w) } end, vim.api.nvim_tabpage_list_wins(tp) ) } end ]]) return function(tp) return exec_lua('return tp_info(...)', tp) end end it('attempt to move window with no room', function() -- Fill the 2nd tabpage full of windows until we run out of room. -- Use &laststatus=0 to ensure restoring missing statuslines doesn't affect things. command('set laststatus=0 | tabnew') matches('E36: Not enough room$', pcall_err(command, 'execute "split|"->repeat(&lines)')) command('vsplit | wincmd | | wincmd p') local t2 = api.nvim_get_current_tabpage() local t2_cur_win = api.nvim_get_current_win() local t2_top_split = fn.win_getid(1) local t2_bot_split = fn.win_getid(fn.winnr('$')) local t2_float = api.nvim_open_win( 0, false, { relative = 'editor', row = 0, col = 0, width = 10, height = 10 } ) local t2_float_config = api.nvim_win_get_config(t2_float) local tp_info = define_tp_info_function() local t2_info = tp_info(t2) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_top_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_top_split, split = 'below' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_bot_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_bot_split, split = 'below' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t2_float, { win = t2_top_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t2_float, { win = t2_top_split, split = 'below' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t2_float, { win = t2_bot_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t2_float, { win = t2_bot_split, split = 'below' }) ) eq(t2_cur_win, api.nvim_get_current_win()) eq(t2_info, tp_info(t2)) eq(t2_float_config, api.nvim_win_get_config(t2_float)) -- Try to move windows from the 1st tabpage to the 2nd. command('tabfirst | split | wincmd _') local t1 = api.nvim_get_current_tabpage() local t1_cur_win = api.nvim_get_current_win() local t1_float = api.nvim_open_win( 0, false, { relative = 'editor', row = 5, col = 3, width = 7, height = 6 } ) local t1_float_config = api.nvim_win_get_config(t1_float) local t1_info = tp_info(t1) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_top_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_top_split, split = 'below' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_bot_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, 0, { win = t2_bot_split, split = 'below' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t1_float, { win = t2_top_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t1_float, { win = t2_top_split, split = 'below' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t1_float, { win = t2_bot_split, split = 'above' }) ) matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t1_float, { win = t2_bot_split, split = 'below' }) ) eq(t1_cur_win, api.nvim_get_current_win()) eq(t1_info, tp_info(t1)) eq(t1_float_config, api.nvim_win_get_config(t1_float)) end) it('attempt to move window from other tabpage with no room', function() -- Fill up the 1st tabpage with horizontal splits, then create a 2nd with only a few. Go back -- to the 1st and try to move windows from the 2nd (while it's non-current) to it. Check that -- window positions and sizes in the 2nd are unchanged. command('set laststatus=0') matches('E36: Not enough room$', pcall_err(command, 'execute "split|"->repeat(&lines)')) command('tab split') local t2 = api.nvim_get_current_tabpage() local t2_top = api.nvim_get_current_win() command('belowright split') local t2_mid_left = api.nvim_get_current_win() command('belowright vsplit') local t2_mid_right = api.nvim_get_current_win() command('split | wincmd J') local t2_bot = api.nvim_get_current_win() local tp_info = define_tp_info_function() local t2_info = tp_info(t2) eq({ 'col', { { 'leaf', t2_top }, { 'row', { { 'leaf', t2_mid_left }, { 'leaf', t2_mid_right }, }, }, { 'leaf', t2_bot }, }, }, t2_info.layout) local function try_move_t2_wins_to_t1() for _, w in ipairs({ t2_bot, t2_mid_left, t2_mid_right, t2_top }) do matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, w, { win = 0, split = 'below' }) ) eq(t2_info, tp_info(t2)) end end command('tabfirst') try_move_t2_wins_to_t1() -- Go to the 2nd tabpage to ensure nothing changes after win_comp_pos, last_status, .etc. -- from enter_tabpage. command('tabnext') eq(t2_info, tp_info(t2)) -- Check things are fine with the global statusline too, for good measure. -- Set it while the 2nd tabpage is current, so last_status runs for it. command('set laststatus=3') t2_info = tp_info(t2) command('tabfirst') try_move_t2_wins_to_t1() end) it('handles cmdwin and textlock restrictions', function() command('tabnew') local t2 = api.nvim_get_current_tabpage() local t2_win = api.nvim_get_current_win() command('tabfirst') local t1_move_win = api.nvim_get_current_win() command('split') -- Can't move the cmdwin, or its old curwin to a different tabpage. local old_curwin = api.nvim_get_current_win() feed('q:') eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_set_config, 0, { split = 'left', win = t2_win }) ) eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err(api.nvim_win_set_config, old_curwin, { split = 'left', win = t2_win }) ) -- But we can move other windows. api.nvim_win_set_config(t1_move_win, { split = 'left', win = t2_win }) eq(t2, api.nvim_win_get_tabpage(t1_move_win)) command('quit!') -- Can't configure windows such that the cmdwin would become the only non-float. command('only!') feed('q:') eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err( api.nvim_win_set_config, old_curwin, { relative = 'editor', row = 0, col = 0, width = 5, height = 5 } ) ) -- old_curwin is now no longer the only other non-float, so we can make it floating now. local t1_new_win = api.nvim_open_win( api.nvim_create_buf(true, true), false, { split = 'left', win = old_curwin } ) api.nvim_win_set_config( old_curwin, { relative = 'editor', row = 0, col = 0, width = 5, height = 5 } ) eq('editor', api.nvim_win_get_config(old_curwin).relative) -- ...which means we shouldn't be able to also make the new window floating too! eq( 'E11: Invalid in command-line window; executes, CTRL-C quits', pcall_err( api.nvim_win_set_config, t1_new_win, { relative = 'editor', row = 0, col = 0, width = 5, height = 5 } ) ) -- Nothing ought to stop us from making the cmdwin itself floating, though... api.nvim_win_set_config(0, { relative = 'editor', row = 0, col = 0, width = 5, height = 5 }) eq('editor', api.nvim_win_get_config(0).relative) -- We can't make our new window from before floating too, as it's now the only non-float. eq( 'Cannot change last window into float', pcall_err( api.nvim_win_set_config, t1_new_win, { relative = 'editor', row = 0, col = 0, width = 5, height = 5 } ) ) command('quit!') -- Can't switch away from window before moving it to a different tabpage during textlock. exec(([[ new call setline(1, 'foo') setlocal debug=throw indentexpr=nvim_win_set_config(0,#{split:'left',win:%d}) ]]):format(t2_win)) local cur_win = api.nvim_get_current_win() matches( 'E565: Not allowed to change text or change window$', pcall_err(command, 'normal! ==') ) eq(cur_win, api.nvim_get_current_win()) end) it('updates statusline when moving bottom split', function() local screen = Screen.new(10, 10) exec([[ set laststatus=0 belowright split call nvim_win_set_config(0, #{split: 'above', win: win_getid(winnr('#'))}) ]]) screen:expect([[ ^ | {1:~ }|*3 {3:[No Name] }| | {1:~ }|*3 | ]]) end) it("updates tp_curwin of moved window's original tabpage", function() local t1 = api.nvim_get_current_tabpage() command('tab split | split') local t2 = api.nvim_get_current_tabpage() local t2_alt_win = api.nvim_get_current_win() command('vsplit') local t2_cur_win = api.nvim_get_current_win() command('tabprevious') eq(t2_cur_win, api.nvim_tabpage_get_win(t2)) -- tp_curwin is unchanged when moved within the same tabpage. api.nvim_win_set_config(t2_cur_win, { split = 'left', win = t2_alt_win }) eq(t2_cur_win, api.nvim_tabpage_get_win(t2)) -- Also unchanged if the move failed. command('let &winwidth = &columns | let &winminwidth = &columns') matches( 'E36: Not enough room$', pcall_err(api.nvim_win_set_config, t2_cur_win, { split = 'left', win = 0 }) ) eq(t2_cur_win, api.nvim_tabpage_get_win(t2)) command('set winminwidth& winwidth&') -- But is changed if successfully moved to a different tabpage. api.nvim_win_set_config(t2_cur_win, { split = 'left', win = 0 }) eq(t2_alt_win, api.nvim_tabpage_get_win(t2)) eq(t1, api.nvim_win_get_tabpage(t2_cur_win)) -- Now do it for a float, which has different altwin logic. command('tabnext') t2_cur_win = api.nvim_open_win(0, true, { relative = 'editor', row = 5, col = 5, width = 5, height = 5 }) eq(t2_alt_win, fn.win_getid(fn.winnr('#'))) command('tabprevious') eq(t2_cur_win, api.nvim_tabpage_get_win(t2)) api.nvim_win_set_config(t2_cur_win, { split = 'left', win = 0 }) eq(t2_alt_win, api.nvim_tabpage_get_win(t2)) eq(t1, api.nvim_win_get_tabpage(t2_cur_win)) end) end) describe('get_config', function() it('includes border', function() local b = { 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h' } local win = api.nvim_open_win(0, true, { relative = 'win', row = 3, col = 3, width = 12, height = 3, border = b, }) local cfg = api.nvim_win_get_config(win) eq(b, cfg.border) end) it('includes border with highlight group', function() local b = { { 'a', 'Normal' }, { 'b', 'Special' }, { 'c', 'String' }, { 'd', 'Comment' }, { 'e', 'Visual' }, { 'f', 'Error' }, { 'g', 'Constant' }, { 'h', 'PreProc' }, } local win = api.nvim_open_win(0, true, { relative = 'win', row = 3, col = 3, width = 12, height = 3, border = b, }) local cfg = api.nvim_win_get_config(win) eq(b, cfg.border) end) it('includes title and footer', function() local title = { { 'A', { 'StatusLine', 'TabLine' } }, { 'B' }, { 'C', 'WinBar' } } local footer = { { 'A', 'WinBar' }, { 'B' }, { 'C', { 'StatusLine', 'TabLine' } } } local win = api.nvim_open_win(0, true, { relative = 'win', row = 3, col = 3, width = 12, height = 3, border = 'single', title = title, footer = footer, }) local cfg = api.nvim_win_get_config(win) eq(title, cfg.title) eq(footer, cfg.footer) end) it('includes split for normal windows', function() local win = api.nvim_open_win(0, true, { vertical = true, win = -1, }) eq('left', api.nvim_win_get_config(win).split) api.nvim_win_set_config(win, { vertical = false, win = -1, }) eq('above', api.nvim_win_get_config(win).split) api.nvim_win_set_config(win, { split = 'below', win = -1, }) eq('below', api.nvim_win_get_config(win).split) end) it('includes split when splitting with ex commands', function() local win = api.nvim_get_current_win() eq('left', api.nvim_win_get_config(win).split) command('vsplit') local win2 = api.nvim_get_current_win() -- initial window now be marked as right split -- since it was split with a vertical split -- and 'splitright' is false by default eq('right', api.nvim_win_get_config(win).split) eq('left', api.nvim_win_get_config(win2).split) api.nvim_set_option_value('splitbelow', true, { scope = 'global', }) api.nvim_win_close(win, true) command('split') local win3 = api.nvim_get_current_win() eq('below', api.nvim_win_get_config(win3).split) end) it("includes the correct 'split' option in complex layouts", function() local initial_win = api.nvim_get_current_win() local win = api.nvim_open_win(0, false, { split = 'right', win = -1, }) local win2 = api.nvim_open_win(0, false, { split = 'below', win = win, }) api.nvim_win_set_config(win2, { width = 50, }) api.nvim_win_set_config(win, { split = 'left', win = -1, }) local win3 = api.nvim_open_win(0, false, { split = 'above', win = -1, }) local float = api.nvim_open_win(0, false, { relative = 'editor', width = 40, height = 20, col = 20, row = 10, }) api.nvim_win_set_config(float, { split = 'right', win = -1, }) local layout = fn.winlayout() eq({ 'row', { { 'col', { { 'leaf', win3 }, { 'row', { { 'leaf', win }, { 'leaf', initial_win }, { 'leaf', win2 }, }, }, }, }, { 'leaf', float, }, }, }, layout) eq('above', api.nvim_win_get_config(win3).split) eq('left', api.nvim_win_get_config(win).split) eq('left', api.nvim_win_get_config(initial_win).split) eq('right', api.nvim_win_get_config(win2).split) eq('right', api.nvim_win_get_config(float).split) end) end) describe('set_config', function() it('no crash with invalid title', function() local win = api.nvim_open_win(0, true, { width = 10, height = 10, relative = 'editor', row = 10, col = 10, title = { { 'test' } }, border = 'single', }) eq( 'title/footer must be string or array', pcall_err(api.nvim_win_set_config, win, { title = 0 }) ) command('redraw!') assert_alive() eq( 'title/footer cannot be an empty array', pcall_err(api.nvim_win_set_config, win, { title = {} }) ) command('redraw!') assert_alive() end) it('no crash with invalid footer', function() local win = api.nvim_open_win(0, true, { width = 10, height = 10, relative = 'editor', row = 10, col = 10, footer = { { 'test' } }, border = 'single', }) eq( 'title/footer must be string or array', pcall_err(api.nvim_win_set_config, win, { footer = 0 }) ) command('redraw!') assert_alive() eq( 'title/footer cannot be an empty array', pcall_err(api.nvim_win_set_config, win, { footer = {} }) ) command('redraw!') assert_alive() end) describe('no crash or memory leak', function() local win before_each(function() win = api.nvim_open_win(0, false, { relative = 'editor', row = 10, col = 10, height = 10, width = 10, border = 'single', title = { { 'OLD_TITLE' } }, footer = { { 'OLD_FOOTER' } }, }) end) it('with valid title and invalid footer', function() eq( 'title/footer must be string or array', pcall_err(api.nvim_win_set_config, win, { title = { { 'NEW_TITLE' } }, footer = 0, }) ) command('redraw!') assert_alive() eq({ { 'OLD_TITLE' } }, api.nvim_win_get_config(win).title) end) it('with invalid title and valid footer', function() eq( 'title/footer must be string or array', pcall_err(api.nvim_win_set_config, win, { title = 0, footer = { { 'NEW_FOOTER' } }, }) ) command('redraw!') assert_alive() eq({ { 'OLD_FOOTER' } }, api.nvim_win_get_config(win).footer) end) end) end) end)