commit 019b2050e13583f8a26dc5dc1c7e20ca07bb51e5
parent 8d68dbf906f9559dfd6e8cb1dbc1bb26fffca74d
Author: Luuk van Baal <luukvbaal@gmail.com>
Date: Tue, 22 Apr 2025 21:35:10 +0200
fix(snippet): use <cmd>call cursor() for visual range
Problem: Change applied in d3e495ce uses a byte-offset where a virtual
column is expected.
Solution: Set the cursor directly through a <Cmd> mapping, while making
sure the commands are ordered correctly by adding them to the
type-ahead buffer.
Diffstat:
2 files changed, 16 insertions(+), 16 deletions(-)
diff --git a/runtime/lua/vim/snippet.lua b/runtime/lua/vim/snippet.lua
@@ -278,15 +278,6 @@ local function select_tabstop(tabstop)
vim.api.nvim_feedkeys(keys, 'n', true)
end
- --- NOTE: We don't use `vim.api.nvim_win_set_cursor` here because it causes the cursor to end
- --- at the end of the selection instead of the start.
- ---
- --- @param row integer
- --- @param col integer
- local function move_cursor_to(row, col)
- feedkeys(string.format('%sG%s|', row, col))
- end
-
local range = tabstop:get_range()
local mode = vim.fn.mode()
@@ -311,13 +302,16 @@ local function select_tabstop(tabstop)
end
else
-- Else, select the tabstop's text.
- if mode ~= 'n' then
- feedkeys('<Esc>')
- end
- move_cursor_to(range[1] + 1, range[2] + 1)
- feedkeys('v')
- move_cursor_to(range[3] + 1, range[4])
- feedkeys('o<c-g><c-r>_')
+ -- Need this exact order so cannot mix regular API calls with feedkeys, which
+ -- are not executed immediately. Use <Cmd> to set the cursor position.
+ local keys = {
+ mode ~= 'n' and '<Esc>' or '',
+ ('<Cmd>call cursor(%s,%s)<CR>'):format(range[1] + 1, range[2] + 1),
+ 'v',
+ ('<Cmd>call cursor(%s,%s)<CR>'):format(range[3] + 1, range[4]),
+ 'o<c-g><c-r>_',
+ }
+ feedkeys(table.concat(keys))
end
end
diff --git a/test/functional/lua/snippet_spec.lua b/test/functional/lua/snippet_spec.lua
@@ -307,4 +307,10 @@ describe('vim.snippet', function()
]]
)
end)
+
+ it('correct visual selection with multi-byte text', function()
+ test_expand_success({ 'function(${1:var})' }, { '口口function(var)' }, nil, '口口')
+ feed('foo')
+ eq({ '口口function(foo)' }, buf_lines(0))
+ end)
end)