Skip to content

Commit 32148c5

Browse files
committed
feat: Sticky cursor for visual mode(s).
1 parent 2e3f0e6 commit 32148c5

File tree

2 files changed

+84
-3
lines changed

2 files changed

+84
-3
lines changed

lua/nvim-surround/init.lua

+3-3
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,7 @@ M.visual_surround = function(args)
9595
return
9696
end
9797

98+
local sticky_mark = buffer.set_extmark(args.curpos)
9899
if vim.fn.visualmode() == "\22" then -- Visual block mode case (add delimiters to every line)
99100
if vim.o.selection == "exclusive" then
100101
last_pos[2] = last_pos[2] - 1
@@ -144,13 +145,12 @@ M.visual_surround = function(args)
144145
buffer.insert_text(first_pos, delimiters[1])
145146
end
146147

147-
config.get_opts().indent_lines(first_pos[1], last_pos[1] + #delimiters[1] + #delimiters[2] - 2)
148-
149-
-- TODO: We should be updating the cursor after every insertion, in case the cursor is sticky.
150148
buffer.restore_curpos({
151149
first_pos = first_pos,
150+
sticky_pos = buffer.get_extmark(sticky_mark),
152151
old_pos = args.curpos,
153152
})
153+
config.get_opts().indent_lines(first_pos[1], last_pos[1] + #delimiters[1] + #delimiters[2] - 2)
154154
end
155155

156156
-- Delete a surrounding delimiter pair, if it exists.

tests/configuration_spec.lua

+81
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
local cr = vim.api.nvim_replace_termcodes("<CR>", true, false, true)
22
local esc = vim.api.nvim_replace_termcodes("<Esc>", true, false, true)
3+
local ctrl_v = vim.api.nvim_replace_termcodes("<C-v>", true, false, true)
34
local get_curpos = function()
45
local curpos = vim.api.nvim_win_get_cursor(0)
56
return { curpos[1], curpos[2] + 1 }
@@ -325,6 +326,86 @@ describe("configuration", function()
325326
check_curpos({ 1, 2 })
326327
end)
327328

329+
it("can make the cursor 'stick' to the text (visual)", function()
330+
require("nvim-surround").buffer_setup({
331+
move_cursor = "sticky",
332+
})
333+
334+
set_lines({
335+
"this is a line",
336+
})
337+
set_curpos({ 1, 9 })
338+
vim.cmd("normal vllS'")
339+
check_curpos({ 1, 12 })
340+
341+
set_lines({
342+
"this is a line",
343+
"with some more text",
344+
})
345+
set_curpos({ 1, 6 })
346+
vim.cmd("normal vjeSb")
347+
check_curpos({ 2, 9 })
348+
349+
set_lines({
350+
"this is a line",
351+
"with some more text",
352+
})
353+
set_curpos({ 1, 6 })
354+
vim.cmd("normal vjeoSb")
355+
check_curpos({ 1, 7 })
356+
end)
357+
358+
it("can make the cursor 'stick' to the text (visual line)", function()
359+
require("nvim-surround").buffer_setup({
360+
move_cursor = "sticky",
361+
})
362+
363+
set_lines({
364+
"this is a line",
365+
})
366+
set_curpos({ 1, 9 })
367+
vim.cmd("normal VSb")
368+
check_curpos({ 2, 9 })
369+
370+
set_lines({
371+
"this is a line",
372+
"with some more text",
373+
})
374+
set_curpos({ 1, 6 })
375+
vim.cmd("normal VjStdiv" .. cr)
376+
check_curpos({ 3, 6 })
377+
end)
378+
379+
it("can make the cursor 'stick' to the text (visual block)", function()
380+
require("nvim-surround").buffer_setup({
381+
move_cursor = "sticky",
382+
surrounds = {
383+
["x"] = {
384+
add = { { "|", "" }, { "", "|" } },
385+
},
386+
},
387+
})
388+
389+
set_lines({
390+
"this is a line",
391+
"this is another line",
392+
})
393+
set_curpos({ 1, 5 })
394+
vim.cmd("normal! " .. ctrl_v .. "jf ")
395+
vim.cmd("normal Sb")
396+
check_curpos({ 2, 9 })
397+
398+
set_lines({
399+
"this is a line",
400+
"this is another line",
401+
"some more random text",
402+
})
403+
set_curpos({ 1, 4 })
404+
vim.cmd("normal! " .. ctrl_v .. "jjww")
405+
vim.cmd("normal Sx")
406+
set_curpos({ 8, 8 })
407+
end)
408+
328409
it("can make the cursor 'stick' to the text (delete)", function()
329410
require("nvim-surround").buffer_setup({
330411
move_cursor = "sticky",

0 commit comments

Comments
 (0)