7 Commits

Author SHA1 Message Date
6521bab7f6 update README to point to v3
All checks were successful
NeoVim tests / plenary-tests (push) Successful in 10s
2026-04-08 22:56:44 -06:00
44a97b5baa (README.md) point to v2
All checks were successful
NeoVim tests / plenary-tests (push) Successful in 10s
2025-05-04 15:42:46 -06:00
7fb60add94 move away from vim.opt_global to vim.go
All checks were successful
NeoVim tests / plenary-tests (push) Successful in 9s
2025-03-19 22:29:33 -06:00
79499e898c cleanup some vim.cmd calls
All checks were successful
NeoVim tests / plenary-tests (push) Successful in 9s
2025-03-12 23:12:28 -06:00
de01a95cdc update example surround plugin 2025-01-05 14:17:20 -07:00
c87cc7c387 add rockspec 2024-11-12 10:04:27 -07:00
5c244cfc0a support repeated Range:replace calls/empty ranges
All checks were successful
NeoVim tests / plenary-tests (push) Successful in 8s
- Range:replace now updates its bounds to reflect the replacement
- Support the notion of an empty range
2024-11-12 07:48:23 -07:00
10 changed files with 232 additions and 45 deletions

1
.gitignore vendored Normal file
View File

@@ -0,0 +1 @@
*.src.rock

View File

@@ -1,5 +1,9 @@
# u.nvim # u.nvim
🚨🚨 **BRANCH NOTICE: further development is happening on the `v3` branch. In the future, `v3` will be merged into `master`. If you want to pin to an older version of this library, please refer to a specific commit, or the `v1` branch.** 🚨🚨
🚨🚨[CLICK HERE FOR v3](https://github.com/jrop/u.nvim/tree/v3)🚨🚨
Welcome to **u.nvim** a powerful Lua library designed to enhance your text manipulation experience in NeoVim, focusing primarily on a context-aware "Range" utility. This utility allows you to work efficiently with text selections based on various conditions, in a variety of contexts, making coding and editing more intuitive and productive. Welcome to **u.nvim** a powerful Lua library designed to enhance your text manipulation experience in NeoVim, focusing primarily on a context-aware "Range" utility. This utility allows you to work efficiently with text selections based on various conditions, in a variety of contexts, making coding and editing more intuitive and productive.
This is meant to be used as a **library**, not a plugin. On its own, `u.nvim` does nothing. It is meant to be used by plugin authors, to make their lives easier based on the variety of utilities I found I needed while growing my NeoVim config. This is meant to be used as a **library**, not a plugin. On its own, `u.nvim` does nothing. It is meant to be used by plugin authors, to make their lives easier based on the variety of utilities I found I needed while growing my NeoVim config.

View File

@@ -108,14 +108,24 @@ function M.setup()
if from_cn < 32 or from_cn > 126 then return end if from_cn < 32 or from_cn > 126 then return end
local from_c = vim.fn.nr2char(from_cn) local from_c = vim.fn.nr2char(from_cn)
local from = surrounds[from_c] or { left = from_c, right = from_c } local from = surrounds[from_c] or { left = from_c, right = from_c }
local function get_fresh_arange()
local arange = Range.from_text_object('a' .. from_c, { user_defined = true }) local arange = Range.from_text_object('a' .. from_c, { user_defined = true })
if arange == nil then return nil end if arange == nil then return nil end
if from_c == 'q' then
from.left = arange.start:char()
from.right = arange.stop:char()
end
return arange
end
local arange = get_fresh_arange()
if arange == nil then return nil end
local hl_info1 = Range.new(arange.start, arange.start, 'v'):highlight('IncSearch', { priority = 999 }) local hl_info1 = Range.new(arange.start, arange.start, 'v'):highlight('IncSearch', { priority = 999 })
local hl_info2 = Range.new(arange.stop, arange.stop, 'v'):highlight('IncSearch', { priority = 999 }) local hl_info2 = Range.new(arange.stop, arange.stop, 'v'):highlight('IncSearch', { priority = 999 })
local hl_clear = function() local hl_clear = function()
hl_info1.clear() if hl_info1 then hl_info1.clear() end
hl_info2.clear() if hl_info2 then hl_info2.clear() end
end end
local to = prompt_for_bounds() local to = prompt_for_bounds()
@@ -123,6 +133,10 @@ function M.setup()
if to == nil then return end if to == nil then return end
vim_repeat.run(function() vim_repeat.run(function()
-- Re-fetch the arange, just in case this action is being repeated:
arange = get_fresh_arange()
if arange == nil then return nil end
if from_c == 't' then if from_c == 't' then
-- For tags, we want to replace the inner text, not the tag: -- For tags, we want to replace the inner text, not the tag:
local irange = Range.from_text_object('i' .. from_c, { user_defined = true }) local irange = Range.from_text_object('i' .. from_c, { user_defined = true })
@@ -219,11 +233,11 @@ function M.setup()
bounds = _G.my_surround_bounds bounds = _G.my_surround_bounds
else else
local prompted_bounds = prompt_for_bounds() local prompted_bounds = prompt_for_bounds()
if prompted_bounds == nil then return hl_info.clear() end if prompted_bounds == nil and hl_info then return hl_info.clear() end
bounds = prompted_bounds if prompted_bounds then bounds = prompted_bounds end
end end
hl_info.clear() if hl_info then hl_info.clear() end
do_surround(range, bounds) do_surround(range, bounds)
-- selene: allow(global_usage) -- selene: allow(global_usage)
_G.my_surround_bounds = nil _G.my_surround_bounds = nil

View File

@@ -10,16 +10,16 @@ end
---@class Range ---@class Range
---@field start Pos ---@field start Pos
---@field stop Pos ---@field stop Pos|nil
---@field mode 'v'|'V' ---@field mode 'v'|'V'
local Range = {} local Range = {}
---@param start Pos ---@param start Pos
---@param stop Pos ---@param stop Pos|nil
---@param mode? 'v'|'V' ---@param mode? 'v'|'V'
---@return Range ---@return Range
function Range.new(start, stop, mode) function Range.new(start, stop, mode)
if stop < start then if stop ~= nil and stop < start then
start, stop = stop, start start, stop = stop, start
end end
@@ -27,7 +27,9 @@ function Range.new(start, stop, mode)
local function str() local function str()
---@param p Pos ---@param p Pos
local function posstr(p) local function posstr(p)
if p.off ~= 0 then if p == nil then
return 'nil'
elseif p.off ~= 0 then
return string.format('Pos(%d:%d){off=%d}', p.lnum, p.col, p.off) return string.format('Pos(%d:%d){off=%d}', p.lnum, p.col, p.off)
else else
return string.format('Pos(%d:%d)', p.lnum, p.col) return string.format('Pos(%d:%d)', p.lnum, p.col)
@@ -125,7 +127,7 @@ function Range.from_text_object(text_obj, opts)
local prev_on_yank_enabled = on_yank_enabled local prev_on_yank_enabled = on_yank_enabled
on_yank_enabled = false on_yank_enabled = false
vim.cmd.normal { vim.cmd {
cmd = 'normal', cmd = 'normal',
bang = not opts.user_defined, bang = not opts.user_defined,
args = { '""y' .. text_obj }, args = { '""y' .. text_obj },
@@ -254,22 +256,27 @@ function Range.smallest(ranges)
return result return result
end end
function Range:clone() return Range.new(self.start:clone(), self.stop:clone(), self.mode) end function Range:clone() return Range.new(self.start:clone(), self.stop ~= nil and self.stop:clone() or nil, self.mode) end
function Range:line_count() return self.stop.lnum - self.start.lnum + 1 end function Range:line_count()
if self:is_empty() then return 0 end
return self.stop.lnum - self.start.lnum + 1
end
function Range:to_linewise() function Range:to_linewise()
local r = self:clone() local r = self:clone()
r.mode = 'V' r.mode = 'V'
r.start.col = 0 r.start.col = 0
r.stop.col = Pos.MAX_COL if r.stop ~= nil then r.stop.col = Pos.MAX_COL end
return r return r
end end
function Range:is_empty() return self.start == self.stop end function Range:is_empty() return self.stop == nil end
function Range:trim_start() function Range:trim_start()
if self:is_empty() then return end
local r = self:clone() local r = self:clone()
while r.start:char():match '%s' do while r.start:char():match '%s' do
local next = r.start:next(1) local next = r.start:next(1)
@@ -280,6 +287,8 @@ function Range:trim_start()
end end
function Range:trim_stop() function Range:trim_stop()
if self:is_empty() then return end
local r = self:clone() local r = self:clone()
while r.stop:char():match '%s' do while r.stop:char():match '%s' do
local next = r.stop:next(-1) local next = r.stop:next(-1)
@@ -290,10 +299,12 @@ function Range:trim_stop()
end end
---@param p Pos ---@param p Pos
function Range:contains(p) return p >= self.start and p <= self.stop end function Range:contains(p) return not self:is_empty() and p >= self.start and p <= self.stop end
---@return string[] ---@return string[]
function Range:lines() function Range:lines()
if self:is_empty() then return {} end
local lines = {} local lines = {}
for i = 0, self.stop.lnum - self.start.lnum do for i = 0, self.stop.lnum - self.start.lnum do
local line = self:line0(i) local line = self:line0(i)
@@ -313,6 +324,8 @@ function Range:sub(i, j) return self:text():sub(i, j) end
---@return { line: string; idx0: { start: number; stop: number; }; lnum: number; range: fun():Range; text: fun():string }|nil ---@return { line: string; idx0: { start: number; stop: number; }; lnum: number; range: fun():Range; text: fun():string }|nil
function Range:line0(l) function Range:line0(l)
if l < 0 then return self:line0(self:line_count() + l) end if l < 0 then return self:line0(self:line_count() + l) end
if l > self:line_count() then return end
local line = vim.api.nvim_buf_get_lines(self.start.buf, self.start.lnum + l, self.start.lnum + l + 1, false)[1] local line = vim.api.nvim_buf_get_lines(self.start.buf, self.start.lnum + l, self.start.lnum + l + 1, false)[1]
if line == nil then return end if line == nil then return end
@@ -340,30 +353,57 @@ end
---@param replacement nil|string|string[] ---@param replacement nil|string|string[]
function Range:replace(replacement) function Range:replace(replacement)
if replacement == nil then replacement = {} end
if type(replacement) == 'string' then replacement = vim.fn.split(replacement, '\n') end if type(replacement) == 'string' then replacement = vim.fn.split(replacement, '\n') end
if replacement == nil and self.mode == 'V' then local buf = self.start.buf
-- delete the lines: -- convert to start-inclusive, stop-exclusive coordinates:
vim.api.nvim_buf_set_lines(self.start.buf, self.start.lnum, self.stop.lnum + 1, false, {}) local start_lnum, stop_lnum = self.start.lnum, (self.stop and self.stop.lnum or self.start.lnum) + 1
else local start_col, stop_col = self.start.col, (self.stop and self.stop.col or self.start.col) + 1
if replacement == nil then replacement = { '' } end
if self.mode == 'v' then local replace_type = (self:is_empty() and 'insert') or (self.mode == 'v' and 'region') or 'lines'
-- Fixup the bounds:
local last_line = vim.api.nvim_buf_get_lines(self.stop.buf, self.stop.lnum, self.stop.lnum + 1, false)[1] or ''
local max_col = #last_line
vim.api.nvim_buf_set_text( ---@param alnum number
self.start.buf, ---@param acol number
self.start.lnum, ---@param blnum number
self.start.col, ---@param bcol number
self.stop.lnum, local function set_text(alnum, acol, blnum, bcol, repl)
math.min(self.stop.col + 1, max_col), -- row indices are end-inclusive, and column indices are end-exclusive.
replacement vim.api.nvim_buf_set_text(buf, alnum, acol, blnum, bcol, repl)
)
else local new_last_line_num = self.start.lnum + #replacement - 1
vim.api.nvim_buf_set_lines(self.start.buf, self.start.lnum, self.stop.lnum + 1, false, replacement) local new_last_col = #(replacement[#replacement] or '')
if new_last_line_num == start_lnum then new_last_col = new_last_col + start_col - 1 end
self.stop = Pos.new(buf, new_last_line_num, new_last_col)
end end
---@param alnum number
---@param blnum number
local function set_lines(alnum, blnum, repl)
-- indexing is zero-based, end-exclusive
vim.api.nvim_buf_set_lines(buf, alnum, blnum, false, repl)
if #repl == 0 then
self.stop = nil
else
local new_last_line_num = start_lnum + #replacement - 1
self.stop = Pos.new(self.start.buf, new_last_line_num, Pos.MAX_COL, self.stop.off)
end
self.mode = 'v'
end
if replace_type == 'insert' then
set_text(start_lnum, start_col, start_lnum, start_col, replacement)
elseif replace_type == 'region' then
-- Fixup the bounds:
local last_line = vim.api.nvim_buf_get_lines(buf, stop_lnum - 1, stop_lnum, false)[1] or ''
local max_col = #last_line
set_text(start_lnum, start_col, stop_lnum - 1, math.min(stop_col, max_col), replacement)
elseif replace_type == 'lines' then
set_lines(start_lnum, stop_lnum, replacement)
else
error 'unreachable'
end end
end end
@@ -371,6 +411,8 @@ end
function Range:shrink(amount) function Range:shrink(amount)
local start = self.start local start = self.start
local stop = self.stop local stop = self.stop
if stop == nil then return self:clone() end
for _ = 1, amount do for _ = 1, amount do
local next_start = start:next(1) local next_start = start:next(1)
local next_stop = stop:next(-1) local next_stop = stop:next(-1)
@@ -379,7 +421,7 @@ function Range:shrink(amount)
stop = next_stop stop = next_stop
if next_start > next_stop then break end if next_start > next_stop then break end
end end
if start > stop then stop = start end if start > stop then stop = nil end
return Range.new(start, stop, self.mode) return Range.new(start, stop, self.mode)
end end
@@ -393,18 +435,30 @@ end
---@param left string ---@param left string
---@param right string ---@param right string
function Range:save_to_pos(left, right) function Range:save_to_pos(left, right)
if self:is_empty() then
self.start:save_to_pos(left)
self.start:save_to_pos(right)
else
self.start:save_to_pos(left) self.start:save_to_pos(left)
self.stop:save_to_pos(right) self.stop:save_to_pos(right)
end end
end
---@param left string ---@param left string
---@param right string ---@param right string
function Range:save_to_marks(left, right) function Range:save_to_marks(left, right)
if self:is_empty() then
self.start:save_to_mark(left)
self.start:save_to_mark(right)
else
self.start:save_to_mark(left) self.start:save_to_mark(left)
self.stop:save_to_mark(right) self.stop:save_to_mark(right)
end end
end
function Range:set_visual_selection() function Range:set_visual_selection()
if self:is_empty() then return end
if vim.api.nvim_get_current_buf() ~= self.start.buf then vim.api.nvim_set_current_buf(self.start.buf) end if vim.api.nvim_get_current_buf() ~= self.start.buf then vim.api.nvim_set_current_buf(self.start.buf) end
State.run(self.start.buf, function(s) State.run(self.start.buf, function(s)
@@ -427,6 +481,8 @@ end
---@param group string ---@param group string
---@param opts? { timeout?: number, priority?: number, on_macro?: boolean } ---@param opts? { timeout?: number, priority?: number, on_macro?: boolean }
function Range:highlight(group, opts) function Range:highlight(group, opts)
if self:is_empty() then return end
opts = opts or { on_macro = false } opts = opts or { on_macro = false }
if opts.on_macro == nil then opts.on_macro = false end if opts.on_macro == nil then opts.on_macro = false end

View File

@@ -1,6 +1,6 @@
local M = {} local M = {}
local function _normal(cmd) vim.cmd.normal { cmd = 'normal', args = { cmd }, bang = true } end local function _normal(cmd) vim.cmd { cmd = 'normal', args = { cmd }, bang = true } end
M.native_repeat = function() _normal '.' end M.native_repeat = function() _normal '.' end
M.native_undo = function() _normal 'u' end M.native_undo = function() _normal 'u' end

View File

@@ -64,7 +64,7 @@ function State:track_mark(mark) self.marks[mark] = vim.api.nvim_buf_get_mark(sel
function State:track_pos(pos) self.positions[pos] = vim.fn.getpos(pos) end function State:track_pos(pos) self.positions[pos] = vim.fn.getpos(pos) end
---@param nm string ---@param nm string
function State:track_global_option(nm) self.global_options[nm] = vim.g[nm] end function State:track_global_option(nm) self.global_options[nm] = vim.go[nm] end
function State:track_winview() self.win_view = vim.fn.winsaveview() end function State:track_winview() self.win_view = vim.fn.winsaveview() end
@@ -82,7 +82,7 @@ function State:restore()
vim.keymap.set(map.mode, map.lhs, map.rhs, { buffer = map.buffer }) vim.keymap.set(map.mode, map.lhs, map.rhs, { buffer = map.buffer })
end end
for nm, val in pairs(self.global_options) do for nm, val in pairs(self.global_options) do
vim.g[nm] = val vim.go[nm] = val
end end
if self.win_view ~= nil then vim.fn.winrestview(self.win_view) end if self.win_view ~= nil then vim.fn.winrestview(self.win_view) end
end end

View File

@@ -49,6 +49,7 @@ function M.define_text_object(key_seq, fn, opts)
local function handle_visual() local function handle_visual()
local range_or_pos = fn(key_seq) local range_or_pos = fn(key_seq)
if range_or_pos == nil then return end if range_or_pos == nil then return end
if Range.is(range_or_pos) and range_or_pos:is_empty() then range_or_pos = range_or_pos.start end
if Range.is(range_or_pos) then if Range.is(range_or_pos) then
local range = range_or_pos --[[@as Range]] local range = range_or_pos --[[@as Range]]
@@ -67,6 +68,7 @@ function M.define_text_object(key_seq, fn, opts)
local range_or_pos = fn(key_seq) local range_or_pos = fn(key_seq)
if range_or_pos == nil then return end if range_or_pos == nil then return end
if Range.is(range_or_pos) and range_or_pos:is_empty() then range_or_pos = range_or_pos.start end
if Range.is(range_or_pos) then if Range.is(range_or_pos) then
range_or_pos:set_visual_selection() range_or_pos:set_visual_selection()
@@ -74,7 +76,7 @@ function M.define_text_object(key_seq, fn, opts)
local p = range_or_pos --[[@as Pos]] local p = range_or_pos --[[@as Pos]]
State.run(0, function(s) State.run(0, function(s)
s:track_global_option 'eventignore' s:track_global_option 'eventignore'
vim.opt_global.eventignore = 'all' vim.go.eventignore = 'all'
-- insert a single space, so we can select it: -- insert a single space, so we can select it:
vim.api.nvim_buf_set_text(0, p.lnum, p.col, p.lnum, p.col, { ' ' }) vim.api.nvim_buf_set_text(0, p.lnum, p.col, p.lnum, p.col, { ' ' })

View File

@@ -391,7 +391,7 @@ describe('Range', function()
it('is_empty', function() it('is_empty', function()
withbuf({ 'line one', 'and line two' }, function() withbuf({ 'line one', 'and line two' }, function()
local range = Range.new(Pos.new(nil, 0, 0), Pos.new(nil, 0, 0), 'v') local range = Range.new(Pos.new(nil, 0, 0), nil, 'v')
assert.is_true(range:is_empty()) assert.is_true(range:is_empty())
local range2 = Range.new(Pos.new(nil, 0, 0), Pos.new(nil, 0, 1), 'v') local range2 = Range.new(Pos.new(nil, 0, 0), Pos.new(nil, 0, 1), 'v')
@@ -471,4 +471,90 @@ describe('Range', function()
assert.are.same({ 'Rg bleh' }, vim.api.nvim_buf_get_lines(b, 0, -1, false)) assert.are.same({ 'Rg bleh' }, vim.api.nvim_buf_get_lines(b, 0, -1, false))
end) end)
end) end)
it('replace updates Range.stop: same line', function()
withbuf({ 'The quick brown fox jumps over the lazy dog' }, function()
local b = vim.api.nvim_get_current_buf()
local r = Range.new(Pos.new(b, 0, 4), Pos.new(b, 0, 8), 'v')
r:replace 'bleh1'
assert.are.same({ 'The bleh1 brown fox jumps over the lazy dog' }, vim.api.nvim_buf_get_lines(b, 0, -1, false))
r:replace 'bleh2'
assert.are.same({ 'The bleh2 brown fox jumps over the lazy dog' }, vim.api.nvim_buf_get_lines(b, 0, -1, false))
end)
end)
it('replace updates Range.stop: multi-line', function()
withbuf({
'The quick brown fox jumps',
'over the lazy dog',
}, function()
local b = vim.api.nvim_get_current_buf()
local r = Range.new(Pos.new(b, 0, 20), Pos.new(b, 1, 3), 'v')
assert.are.same({ 'jumps', 'over' }, r:lines())
r:replace 'bleh1'
assert.are.same({ 'The quick brown fox bleh1 the lazy dog' }, vim.api.nvim_buf_get_lines(b, 0, -1, false))
r:replace 'blehGoo2'
assert.are.same({ 'The quick brown fox blehGoo2 the lazy dog' }, vim.api.nvim_buf_get_lines(b, 0, -1, false))
end)
end)
it('replace updates Range.stop: multi-line (blockwise)', function()
withbuf({
'The quick brown',
'fox',
'jumps',
'over',
'the lazy dog',
}, function()
local b = vim.api.nvim_get_current_buf()
local r = Range.new(Pos.new(b, 1, 0), Pos.new(b, 3, Pos.MAX_COL), 'V')
assert.are.same({ 'fox', 'jumps', 'over' }, r:lines())
r:replace { 'bleh1', 'bleh2' }
assert.are.same({
'The quick brown',
'bleh1',
'bleh2',
'the lazy dog',
}, vim.api.nvim_buf_get_lines(b, 0, -1, false))
r:replace 'blehGoo2'
assert.are.same({
'The quick brown',
'blehGoo2',
'the lazy dog',
}, vim.api.nvim_buf_get_lines(b, 0, -1, false))
end)
end)
it('replace after delete', function()
withbuf({
'The quick brown',
'fox',
'jumps',
'over',
'the lazy dog',
}, function()
local b = vim.api.nvim_get_current_buf()
local r = Range.new(Pos.new(b, 1, 0), Pos.new(b, 3, Pos.MAX_COL), 'V')
assert.are.same({ 'fox', 'jumps', 'over' }, r:lines())
r:replace(nil)
assert.are.same({
'The quick brown',
'the lazy dog',
}, vim.api.nvim_buf_get_lines(b, 0, -1, false))
r:replace { 'blehGoo2', '' }
assert.are.same({
'The quick brown',
'blehGoo2',
'the lazy dog',
}, vim.api.nvim_buf_get_lines(b, 0, -1, false))
end)
end)
end) end)

View File

@@ -1,5 +1,5 @@
local function withbuf(lines, f) local function withbuf(lines, f)
vim.opt_global.swapfile = false vim.go.swapfile = false
vim.cmd.new() vim.cmd.new()
vim.api.nvim_buf_set_lines(0, 0, -1, false, lines) vim.api.nvim_buf_set_lines(0, 0, -1, false, lines)

24
u.nvim-0.2.0-1.rockspec Normal file
View File

@@ -0,0 +1,24 @@
package = "u.nvim"
version = "0.2.0-1"
source = {
url = "git+https://github.com/jrop/u.nvim"
}
description = {
summary = "nvim a powerful Lua library designed to enhance your text manipulation experience in NeoVim, focusing primarily on a context-aware \"Range\" utility.",
detailed = "Welcome to u.nvim a powerful Lua library designed to enhance your text manipulation experience in NeoVim, focusing primarily on a context-aware \"Range\" utility. This utility allows you to work efficiently with text selections based on various conditions, in a variety of contexts, making coding and editing more intuitive and productive.",
homepage = "https://github.com/jrop/u.nvim",
license = "MIT"
}
build = {
type = "builtin",
modules = {
["u.buffer"] = "lua/u/buffer.lua",
["u.codewriter"] = "lua/u/codewriter.lua",
["u.opkeymap"] = "lua/u/opkeymap.lua",
["u.pos"] = "lua/u/pos.lua",
["u.range"] = "lua/u/range.lua",
["u.repeat"] = "lua/u/repeat.lua",
["u.state"] = "lua/u/state.lua",
["u.utils"] = "lua/u/utils.lua"
}
}