mirror of
https://github.com/neovim/neovim.git
synced 2025-09-06 11:28:22 +00:00

Problem: in terminal mode, adjust_topline moves curwin's cursor to the last row so set_topline tails the non-scrollback area. This may result in the observed cursor position remaining tailed in events within the focused terminal, rather than reflecting the actual cursor position. Solution: use the focused terminal's actual cursor position immediately, rather than relying on the next terminal_check_cursor call in terminal_check to set it. Note: Maybe also possible for terminal mode cursor position to be stale (reporting the normal mode position) when switching buffers in events to another terminal until the next terminal_check call? (or until refresh_terminal is called for it) Maybe not worth fixing that, though.
1217 lines
54 KiB
Lua
1217 lines
54 KiB
Lua
local t = require('test.testutil')
|
|
local n = require('test.functional.testnvim')()
|
|
local tt = require('test.functional.testterm')
|
|
|
|
local feed, clear = n.feed, n.clear
|
|
local testprg, command = n.testprg, n.command
|
|
local eq, eval = t.eq, n.eval
|
|
local api = n.api
|
|
local exec_lua = n.exec_lua
|
|
local matches = t.matches
|
|
local call = n.call
|
|
local hide_cursor = tt.hide_cursor
|
|
local show_cursor = tt.show_cursor
|
|
local retry = t.retry
|
|
local is_os = t.is_os
|
|
local skip = t.skip
|
|
|
|
describe(':terminal cursor', function()
|
|
local screen
|
|
|
|
local terminal_mode_idx ---@type number
|
|
|
|
before_each(function()
|
|
clear()
|
|
screen = tt.setup_screen()
|
|
|
|
if terminal_mode_idx == nil then
|
|
for i, v in ipairs(screen._mode_info) do
|
|
if v.name == 'terminal' then
|
|
terminal_mode_idx = i
|
|
end
|
|
end
|
|
assert(terminal_mode_idx)
|
|
end
|
|
end)
|
|
|
|
it('moves the screen cursor when focused', function()
|
|
tt.feed_data('testing cursor')
|
|
screen:expect([[
|
|
tty ready |
|
|
testing cursor^ |
|
|
|*4
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
|
|
it('is highlighted when not focused', function()
|
|
feed('<c-\\><c-n>')
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
end)
|
|
|
|
describe('with number column', function()
|
|
before_each(function()
|
|
feed('<c-\\><c-n>:set number<cr>')
|
|
end)
|
|
|
|
it('is positioned correctly when unfocused', function()
|
|
screen:expect([[
|
|
{121: 1 }tty ready |
|
|
{121: 2 }^rows: 6, cols: 46 |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 } |
|
|
{121: 6 } |
|
|
:set number |
|
|
]])
|
|
end)
|
|
|
|
it('is positioned correctly when focused', function()
|
|
screen:expect([[
|
|
{121: 1 }tty ready |
|
|
{121: 2 }^rows: 6, cols: 46 |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 } |
|
|
{121: 6 } |
|
|
:set number |
|
|
]])
|
|
feed('i')
|
|
n.poke_eventloop()
|
|
screen:expect([[
|
|
{121: 1 }tty ready |
|
|
{121: 2 }rows: 6, cols: 46 |
|
|
{121: 3 }^ |
|
|
{121: 4 } |
|
|
{121: 5 } |
|
|
{121: 6 } |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
end)
|
|
|
|
describe('when invisible', function()
|
|
it('is not highlighted', function()
|
|
skip(is_os('win'), '#31587')
|
|
hide_cursor()
|
|
screen:expect([[
|
|
tty ready |
|
|
|*5
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
show_cursor()
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*4
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
-- same for when the terminal is unfocused
|
|
feed('<c-\\><c-n>')
|
|
hide_cursor()
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
show_cursor()
|
|
screen:expect_unchanged()
|
|
end)
|
|
|
|
it('becomes visible when exiting Terminal mode', function()
|
|
skip(is_os('win'), '#31587')
|
|
hide_cursor()
|
|
screen:expect([[
|
|
tty ready |
|
|
|*5
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
feed('<c-\\><c-n>')
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
feed('i')
|
|
screen:expect([[
|
|
tty ready |
|
|
|*5
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
|
|
-- Cursor is hidden; now request to show it while in a TermLeave autocmd.
|
|
-- Process events (via :sleep) to handle the escape sequence now.
|
|
command([[autocmd TermLeave * ++once call chansend(&channel, "\e[?25h") | sleep 1m]])
|
|
feed([[<C-\><C-N>]]) -- Exit terminal mode; cursor should not remain hidden
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
|
|
command('bwipeout! | let chan = nvim_open_term(0, {})')
|
|
feed('i')
|
|
-- Hide the cursor, switch to a non-terminal buffer, then show the cursor; it shouldn't remain
|
|
-- hidden after we're kicked out of terminal mode in the new buffer.
|
|
-- Must ensure these actions happen within the same terminal_execute call. The stream is
|
|
-- internal, so polling the event loop isn't necessary (terminal_receive is directly called).
|
|
command([[call chansend(chan, "\e[?25l") | new floob | call chansend(chan, "\e[?25h")]])
|
|
screen:expect([[
|
|
^ |
|
|
{100:~ }|
|
|
{3:floob }|
|
|
|*2
|
|
{119:[Scratch] [-] }|
|
|
|
|
|
]])
|
|
|
|
feed('<C-W>pi')
|
|
screen:expect([[
|
|
|
|
|
{100:~ }|
|
|
{2:floob }|
|
|
^ |
|
|
|
|
|
{120:[Scratch] [-] }|
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
|
|
it('becomes visible on TermLeave if hidden immediately by events #32456', function()
|
|
skip(is_os('win'), '#31587')
|
|
-- Reproducing the issue is quite fragile; it's easiest done in a lone test case like this
|
|
-- with no prior commands.
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
|
|
-- Hide the cursor such that the escape sequence is processed as a side effect of showmode in
|
|
-- terminal_enter handling events (skip_showmode -> char_avail -> vpeekc -> os_breakcheck).
|
|
-- This requires a particular set of actions; :startinsert repros better than feed('i') here.
|
|
hide_cursor()
|
|
command('mode | startinsert')
|
|
screen:expect([[
|
|
tty ready |
|
|
|*5
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
end)
|
|
end)
|
|
|
|
it('can be modified by application #3681 #31685', function()
|
|
skip(is_os('win'), '#31587')
|
|
|
|
local states = {
|
|
[1] = { blink = true, shape = 'block' },
|
|
[2] = { blink = false, shape = 'block' },
|
|
[3] = { blink = true, shape = 'horizontal' },
|
|
[4] = { blink = false, shape = 'horizontal' },
|
|
[5] = { blink = true, shape = 'vertical' },
|
|
[6] = { blink = false, shape = 'vertical' },
|
|
}
|
|
|
|
for k, v in pairs(states) do
|
|
tt.feed_csi(('%d q'):format(k))
|
|
screen:expect({
|
|
grid = [[
|
|
tty ready |
|
|
^ |
|
|
|*4
|
|
{5:-- TERMINAL --} |
|
|
]],
|
|
condition = function()
|
|
if v.blink then
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
else
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
end
|
|
|
|
eq(v.shape, screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
|
|
-- Cell percentages are hard coded for each shape in terminal.c
|
|
if v.shape == 'horizontal' then
|
|
eq(20, screen._mode_info[terminal_mode_idx].cell_percentage)
|
|
elseif v.shape == 'vertical' then
|
|
eq(25, screen._mode_info[terminal_mode_idx].cell_percentage)
|
|
end
|
|
end,
|
|
})
|
|
end
|
|
|
|
feed([[<C-\><C-N>]])
|
|
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
|
|
-- Cursor returns to default on TermLeave
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
eq('block', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
end)
|
|
|
|
it('can be modified per terminal', function()
|
|
skip(is_os('win'), '#31587')
|
|
|
|
-- Set cursor to vertical bar with blink
|
|
tt.feed_csi('5 q')
|
|
screen:expect({
|
|
grid = [[
|
|
tty ready |
|
|
^ |
|
|
|*4
|
|
{5:-- TERMINAL --} |
|
|
]],
|
|
condition = function()
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
eq('vertical', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
end,
|
|
})
|
|
|
|
tt.hide_cursor()
|
|
screen:expect({
|
|
grid = [[
|
|
tty ready |
|
|
|
|
|
|*4
|
|
{5:-- TERMINAL --} |
|
|
]],
|
|
condition = function()
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
eq('vertical', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
end,
|
|
})
|
|
|
|
-- Exit terminal mode to reset terminal cursor settings to default and
|
|
-- create a new terminal window
|
|
feed([[<C-\><C-N>]])
|
|
command('set statusline=~~~')
|
|
command('new')
|
|
call('jobstart', { testprg('tty-test') }, { term = true })
|
|
feed('i')
|
|
screen:expect({
|
|
grid = [[
|
|
tty ready |
|
|
^ |
|
|
{120:~~~ }|
|
|
rows: 2, cols: 50 |
|
|
|
|
|
{119:~~~ }|
|
|
{5:-- TERMINAL --} |
|
|
]],
|
|
condition = function()
|
|
-- New terminal, cursor resets to defaults
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
eq('block', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
end,
|
|
})
|
|
|
|
-- Set cursor to underline, no blink
|
|
tt.feed_csi('4 q')
|
|
screen:expect({
|
|
grid = [[
|
|
tty ready |
|
|
^ |
|
|
{120:~~~ }|
|
|
rows: 2, cols: 50 |
|
|
|
|
|
{119:~~~ }|
|
|
{5:-- TERMINAL --} |
|
|
]],
|
|
condition = function()
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
eq('horizontal', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
end,
|
|
})
|
|
|
|
-- Switch back to first terminal, cursor should still be hidden
|
|
command('wincmd p')
|
|
screen:expect({
|
|
grid = [[
|
|
tty ready |
|
|
|
|
|
{119:~~~ }|
|
|
rows: 2, cols: 50 |
|
|
|
|
|
{120:~~~ }|
|
|
{5:-- TERMINAL --} |
|
|
]],
|
|
condition = function()
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
eq('vertical', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
end,
|
|
})
|
|
end)
|
|
|
|
it('can be positioned arbitrarily', function()
|
|
clear()
|
|
screen = tt.setup_child_nvim({
|
|
'-u',
|
|
'NONE',
|
|
'-i',
|
|
'NONE',
|
|
'--cmd',
|
|
n.nvim_set .. ' noshowmode',
|
|
})
|
|
screen:expect([[
|
|
^ |
|
|
~ |*4
|
|
|
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
|
|
feed('i<Tab>')
|
|
screen:expect([[
|
|
^ |
|
|
~ |*4
|
|
|
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
end)
|
|
|
|
it('preserves guicursor value on TermLeave #31612', function()
|
|
eq(3, screen._mode_info[terminal_mode_idx].hl_id)
|
|
|
|
-- Change 'guicursor' while terminal mode is active
|
|
command('set guicursor+=t:Error')
|
|
|
|
local error_hl_id = call('hlID', 'Error')
|
|
|
|
screen:expect({
|
|
condition = function()
|
|
eq(error_hl_id, screen._mode_info[terminal_mode_idx].hl_id)
|
|
end,
|
|
})
|
|
|
|
-- Exit terminal mode
|
|
feed([[<C-\><C-N>]])
|
|
|
|
screen:expect([[
|
|
tty ready |
|
|
^ |
|
|
|*5
|
|
]])
|
|
|
|
eq(error_hl_id, screen._mode_info[terminal_mode_idx].hl_id)
|
|
end)
|
|
|
|
it('uses the correct attributes', function()
|
|
feed([[<C-\><C-N>]])
|
|
command([[
|
|
bwipeout!
|
|
let chan1 = nvim_open_term(0, {})
|
|
vnew
|
|
let chan2 = nvim_open_term(0, {})
|
|
]])
|
|
feed('i')
|
|
screen:expect([[
|
|
^ │ |
|
|
│ |*4
|
|
{120:[Scratch] [-] }{119:[Scratch] [-] }|
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq('block', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
|
|
-- Modify cursor in the non-current terminal; should not affect this cursor.
|
|
command([[call chansend(chan1, "\e[4 q")]])
|
|
screen:expect_unchanged()
|
|
eq('block', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(500, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
|
|
-- Modify cursor in the current terminal.
|
|
command([[call chansend(chan2, "\e[6 q")]])
|
|
screen:expect_unchanged()
|
|
eq('vertical', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
|
|
-- Check the cursor in the other terminal reflects our changes from before.
|
|
command('wincmd p')
|
|
screen:expect([[
|
|
│^ |
|
|
│ |*4
|
|
{119:[Scratch] [-] }{120:[Scratch] [-] }|
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq('horizontal', screen._mode_info[terminal_mode_idx].cursor_shape)
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkon)
|
|
eq(0, screen._mode_info[terminal_mode_idx].blinkoff)
|
|
end)
|
|
|
|
it('position correct within events', function()
|
|
local term, term_unfocused = exec_lua(function()
|
|
vim.cmd 'bwipeout!'
|
|
local term_unfocused = vim.api.nvim_open_term(0, {})
|
|
vim.cmd.vnew()
|
|
vim.cmd.wincmd '|'
|
|
local term = vim.api.nvim_open_term(0, {})
|
|
-- We'll use this keymap to pause the main loop while we send events, as we want the test to
|
|
-- run within the same terminal_execute call (while using test suite facilities like retry).
|
|
vim.keymap.set('t', '<F1>', '<Cmd>let g:sleepy = 1 | sleep 5000 | let g:sleepy = 0<CR>')
|
|
return term, term_unfocused
|
|
end)
|
|
feed('i<F1>')
|
|
|
|
local function check_pos(expected_pos, expected_virtcol, chan, data)
|
|
api.nvim_chan_send(chan, data) -- Using nvim_chan_send so terminal_receive is immediate.
|
|
|
|
-- Results won't be visible until refresh_terminal is called, which happens on a timer.
|
|
retry(nil, nil, function()
|
|
eq(expected_pos, eval("getpos('.')[1:]"))
|
|
end)
|
|
eq(expected_virtcol, eval("virtcol('.', 1)"))
|
|
eq(1, eval('g:sleepy')) -- :sleep shouldn't have timed out.
|
|
end
|
|
|
|
check_pos({ 1, 4, 0 }, { 4, 4 }, term, 'foo')
|
|
-- double-width char at end (3 bytes)
|
|
check_pos({ 2, 13, 0 }, { 12, 12 }, term, '\r\nbarbaaaar哦')
|
|
-- Move to 1,12 (beyond eol; sets coladd)
|
|
check_pos({ 1, 4, 8 }, { 12, 12 }, term, '\27[1;12H')
|
|
-- Move to 4,1
|
|
check_pos({ 4, 1, 0 }, { 1, 1 }, term, '\27[4;1H')
|
|
-- Move to 4,5 (beyond eol; sets coladd)
|
|
check_pos({ 4, 1, 4 }, { 5, 5 }, term, '\27[4;5H')
|
|
-- Move to 2,10 (head of wide char)
|
|
check_pos({ 2, 10, 0 }, { 10, 11 }, term, '\27[2;10H')
|
|
-- Move to 2,11 (non-head of wide char)
|
|
check_pos({ 2, 10, 0 }, { 10, 11 }, term, '\27[2;11H')
|
|
-- Move to 2,12 (after wide char)
|
|
check_pos({ 2, 13, 0 }, { 12, 12 }, term, '\27[2;12H')
|
|
-- Move to 2,13 (beyond eol; sets coladd)
|
|
check_pos({ 2, 13, 1 }, { 13, 13 }, term, '\27[2;13H')
|
|
-- Cursor movement in unfocused terminal shouldn't affect us
|
|
check_pos({ 2, 13, 1 }, { 13, 13 }, term_unfocused, 'amogus')
|
|
end)
|
|
end)
|
|
|
|
describe('buffer cursor position is correct in terminal without number column', function()
|
|
local screen
|
|
|
|
local function setup_ex_register(str)
|
|
screen = tt.setup_child_nvim({
|
|
'-u',
|
|
'NONE',
|
|
'-i',
|
|
'NONE',
|
|
'-E',
|
|
'--cmd',
|
|
string.format('let @r = "%s"', str),
|
|
-- <Left> and <Right> don't always work
|
|
'--cmd',
|
|
'cnoremap <C-X> <Left>',
|
|
'--cmd',
|
|
'cnoremap <C-O> <Right>',
|
|
'--cmd',
|
|
'set notermguicolors',
|
|
}, {
|
|
cols = 70,
|
|
})
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
end
|
|
|
|
before_each(clear)
|
|
|
|
describe('in a line with no multibyte chars or trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('aaaaaaaa')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:aaaaaaaa^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 9 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:aaaaaaa^a |
|
|
|
|
|
]])
|
|
eq({ 6, 8 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:aaaaaa^aa |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 7 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:aaaaa^aaa |
|
|
|
|
|
]])
|
|
eq({ 6, 6 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:a^aaaaaaa |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 2 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:^aaaaaaaa |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('in a line with single-cell multibyte chars and no trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('µµµµµµµµ')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µµµµµµµµ^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 17 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µµµµµµµ^µ |
|
|
|
|
|
]])
|
|
eq({ 6, 15 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µµµµµµ^µµ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 13 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µµµµµ^µµµ |
|
|
|
|
|
]])
|
|
eq({ 6, 11 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µ^µµµµµµµ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 3 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:^µµµµµµµµ |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('in a line with single-cell composed multibyte chars and no trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('µ̳µ̳µ̳µ̳µ̳µ̳µ̳µ̳')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µ̳µ̳µ̳µ̳µ̳µ̳µ̳µ̳^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 33 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µ̳µ̳µ̳µ̳µ̳µ̳µ̳^µ̳ |
|
|
|
|
|
]])
|
|
eq({ 6, 29 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
skip(is_os('win'))
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µ̳µ̳µ̳µ̳µ̳µ̳^µ̳µ̳ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 25 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µ̳µ̳µ̳µ̳µ̳^µ̳µ̳µ̳ |
|
|
|
|
|
]])
|
|
eq({ 6, 21 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
skip(is_os('win'))
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:µ̳^µ̳µ̳µ̳µ̳µ̳µ̳µ̳ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 5 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:^µ̳µ̳µ̳µ̳µ̳µ̳µ̳µ̳ |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('in a line with double-cell multibyte chars and no trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('哦哦哦哦哦哦哦哦')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:哦哦哦哦哦哦哦哦^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 25 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:哦哦哦哦哦哦哦^哦 |
|
|
|
|
|
]])
|
|
eq({ 6, 22 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:哦哦哦哦哦哦^哦哦 |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 19 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:哦哦哦哦哦^哦哦哦 |
|
|
|
|
|
]])
|
|
eq({ 6, 16 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:哦^哦哦哦哦哦哦哦 |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 4 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:^哦哦哦哦哦哦哦哦 |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
it('at the end of a line with trailing spaces #16234', function()
|
|
setup_ex_register('aaaaaaaa ')
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:aaaaaaaa ^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
matches('^:aaaaaaaa [ ]*$', eval('nvim_get_current_line()'))
|
|
eq({ 6, 13 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
|*4
|
|
Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
:aaaaaaaa ^ |
|
|
|
|
|
]])
|
|
eq({ 6, 12 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('buffer cursor position is correct in terminal with number column', function()
|
|
if t.is_ci('cirrus') then
|
|
return
|
|
end
|
|
|
|
local screen
|
|
|
|
local function setup_ex_register(str)
|
|
screen = tt.setup_child_nvim({
|
|
'-u',
|
|
'NONE',
|
|
'-i',
|
|
'NONE',
|
|
'-E',
|
|
'--cmd',
|
|
string.format('let @r = "%s"', str),
|
|
-- <Left> and <Right> don't always work
|
|
'--cmd',
|
|
'cnoremap <C-X> <Left>',
|
|
'--cmd',
|
|
'cnoremap <C-O> <Right>',
|
|
'--cmd',
|
|
'set notermguicolors',
|
|
}, {
|
|
cols = 70,
|
|
})
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
end
|
|
|
|
before_each(function()
|
|
clear()
|
|
command('au TermOpen * set number')
|
|
end)
|
|
|
|
describe('in a line with no multibyte chars or trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('aaaaaaaa')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:aaaaaaaa^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 9 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:aaaaaaa^a |
|
|
|
|
|
]])
|
|
eq({ 6, 8 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:aaaaaa^aa |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 7 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:aaaaa^aaa |
|
|
|
|
|
]])
|
|
eq({ 6, 6 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:a^aaaaaaa |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 2 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:^aaaaaaaa |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('in a line with single-cell multibyte chars and no trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('µµµµµµµµ')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µµµµµµµµ^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 17 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µµµµµµµ^µ |
|
|
|
|
|
]])
|
|
eq({ 6, 15 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µµµµµµ^µµ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 13 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µµµµµ^µµµ |
|
|
|
|
|
]])
|
|
eq({ 6, 11 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µ^µµµµµµµ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 3 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:^µµµµµµµµ |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('in a line with single-cell composed multibyte chars and no trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('µ̳µ̳µ̳µ̳µ̳µ̳µ̳µ̳')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µ̳µ̳µ̳µ̳µ̳µ̳µ̳µ̳^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 33 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µ̳µ̳µ̳µ̳µ̳µ̳µ̳^µ̳ |
|
|
|
|
|
]])
|
|
eq({ 6, 29 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
skip(is_os('win'))
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µ̳µ̳µ̳µ̳µ̳µ̳^µ̳µ̳ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 25 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µ̳µ̳µ̳µ̳µ̳^µ̳µ̳µ̳ |
|
|
|
|
|
]])
|
|
eq({ 6, 21 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
skip(is_os('win'))
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:µ̳^µ̳µ̳µ̳µ̳µ̳µ̳µ̳ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 5 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:^µ̳µ̳µ̳µ̳µ̳µ̳µ̳µ̳ |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
describe('in a line with double-cell multibyte chars and no trailing spaces,', function()
|
|
before_each(function()
|
|
setup_ex_register('哦哦哦哦哦哦哦哦')
|
|
end)
|
|
|
|
it('at the end', function()
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:哦哦哦哦哦哦哦哦^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 25 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:哦哦哦哦哦哦哦^哦 |
|
|
|
|
|
]])
|
|
eq({ 6, 22 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the end', function()
|
|
feed('<C-R>r<C-X><C-X>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:哦哦哦哦哦哦^哦哦 |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 19 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:哦哦哦哦哦^哦哦哦 |
|
|
|
|
|
]])
|
|
eq({ 6, 16 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
|
|
it('near the start', function()
|
|
feed('<C-R>r<C-B><C-O>')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:哦^哦哦哦哦哦哦哦 |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
eq({ 6, 4 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:^哦哦哦哦哦哦哦哦 |
|
|
|
|
|
]])
|
|
eq({ 6, 1 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|
|
|
|
it('at the end of a line with trailing spaces #16234', function()
|
|
setup_ex_register('aaaaaaaa ')
|
|
feed('<C-R>r')
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:aaaaaaaa ^ |
|
|
{5:-- TERMINAL --} |
|
|
]])
|
|
matches('^:aaaaaaaa [ ]*$', eval('nvim_get_current_line()'))
|
|
eq({ 6, 13 }, eval('nvim_win_get_cursor(0)'))
|
|
feed([[<C-\><C-N>]])
|
|
screen:expect([[
|
|
{121: 1 } |
|
|
{121: 2 } |
|
|
{121: 3 } |
|
|
{121: 4 } |
|
|
{121: 5 }Entering Ex mode. Type "visual" to go to Normal mode. |
|
|
{121: 6 }:aaaaaaaa ^ |
|
|
|
|
|
]])
|
|
eq({ 6, 12 }, eval('nvim_win_get_cursor(0)'))
|
|
end)
|
|
end)
|