Compare commits
No commits in common. "100ff8b5560739eb25a509695688a07f6bd9635f" and "db6e8567c303dd82aba0ab075cd748eac3ac9ea0" have entirely different histories.
100ff8b556
...
db6e8567c3
1
.gitignore
vendored
1
.gitignore
vendored
@ -1 +0,0 @@
|
|||||||
*.src.rock
|
|
@ -17,7 +17,7 @@ lazy.nvim:
|
|||||||
```lua
|
```lua
|
||||||
-- Setting `lazy = true` ensures that the library is only loaded
|
-- Setting `lazy = true` ensures that the library is only loaded
|
||||||
-- when `require 'u.<utility>' is called.
|
-- when `require 'u.<utility>' is called.
|
||||||
{ 'jrop/u.nvim', lazy = true }
|
{ 'https://codeberg.org/jrop/u.nvim', lazy = true }
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
@ -1,86 +0,0 @@
|
|||||||
local vim_repeat = require 'u.repeat'
|
|
||||||
local CodeWriter = require 'u.codewriter'
|
|
||||||
local Range = require 'u.range'
|
|
||||||
|
|
||||||
local M = {}
|
|
||||||
|
|
||||||
---@param bracket_range Range
|
|
||||||
---@param left string
|
|
||||||
---@param right string
|
|
||||||
local function split(bracket_range, left, right)
|
|
||||||
local code = CodeWriter.from_pos(bracket_range.start)
|
|
||||||
code:write_raw(left)
|
|
||||||
|
|
||||||
local curr = bracket_range.start:next()
|
|
||||||
if curr == nil then return end
|
|
||||||
local last_start = curr
|
|
||||||
|
|
||||||
-- Sanity check: if we "skipped" past the start/stop of the overall range, then something is wrong:
|
|
||||||
-- This can happen with greater-/less- than signs that are mathematical, and not brackets:
|
|
||||||
while curr > bracket_range.start and curr < bracket_range.stop do
|
|
||||||
if vim.tbl_contains({ '{', '[', '(', '<' }, curr:char()) then
|
|
||||||
-- skip over bracketed groups:
|
|
||||||
local next = curr:find_match()
|
|
||||||
if next == nil then break end
|
|
||||||
curr = next
|
|
||||||
else
|
|
||||||
if vim.tbl_contains({ ',', ';' }, curr:char()) then
|
|
||||||
-- accumulate item:
|
|
||||||
local item = vim.trim(Range.new(last_start, curr):text())
|
|
||||||
if item ~= '' then code:indent():write(item) end
|
|
||||||
|
|
||||||
local next_last_start = curr:next()
|
|
||||||
if next_last_start == nil then break end
|
|
||||||
last_start = next_last_start
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Prepare for next iteration:
|
|
||||||
local next = curr:next()
|
|
||||||
if next == nil then break end
|
|
||||||
curr = next
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- accumulate last, unfinished item:
|
|
||||||
local pos_before_right = bracket_range.stop:must_next(-1)
|
|
||||||
if last_start < pos_before_right then
|
|
||||||
local item = vim.trim(Range.new(last_start, pos_before_right):text())
|
|
||||||
if item ~= '' then code:indent():write(item) end
|
|
||||||
end
|
|
||||||
|
|
||||||
code:write(right)
|
|
||||||
bracket_range:replace(code.lines)
|
|
||||||
end
|
|
||||||
|
|
||||||
---@param bracket_range Range
|
|
||||||
---@param left string
|
|
||||||
---@param right string
|
|
||||||
local function join(bracket_range, left, right)
|
|
||||||
local inner_range = Range.new(bracket_range.start:must_next(), bracket_range.stop:must_next(-1), bracket_range.mode)
|
|
||||||
local newline = vim
|
|
||||||
.iter(inner_range:lines())
|
|
||||||
:map(function(l) return vim.trim(l) end)
|
|
||||||
:filter(function(l) return l ~= '' end)
|
|
||||||
:join ' '
|
|
||||||
bracket_range:replace { left .. newline .. right }
|
|
||||||
end
|
|
||||||
|
|
||||||
local function splitjoin()
|
|
||||||
local bracket_range = Range.find_nearest_brackets()
|
|
||||||
if bracket_range == nil then return end
|
|
||||||
local lines = bracket_range:lines()
|
|
||||||
local left = lines[1]:sub(1, 1) -- left bracket
|
|
||||||
local right = lines[#lines]:sub(-1, -1) -- right bracket
|
|
||||||
|
|
||||||
if #lines == 1 then
|
|
||||||
split(bracket_range, left, right)
|
|
||||||
else
|
|
||||||
join(bracket_range, left, right)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
function M.setup()
|
|
||||||
vim.keymap.set('n', 'gS', function() vim_repeat.run(splitjoin) end)
|
|
||||||
end
|
|
||||||
|
|
||||||
return M
|
|
@ -1,254 +0,0 @@
|
|||||||
local vim_repeat = require 'u.repeat'
|
|
||||||
local opkeymap = require 'u.opkeymap'
|
|
||||||
local Pos = require 'u.pos'
|
|
||||||
local Range = require 'u.range'
|
|
||||||
local Buffer = require 'u.buffer'
|
|
||||||
local CodeWriter = require 'u.codewriter'
|
|
||||||
|
|
||||||
local M = {}
|
|
||||||
|
|
||||||
local surrounds = {
|
|
||||||
[')'] = { left = '(', right = ')' },
|
|
||||||
['('] = { left = '( ', right = ' )' },
|
|
||||||
[']'] = { left = '[', right = ']' },
|
|
||||||
['['] = { left = '[ ', right = ' ]' },
|
|
||||||
['}'] = { left = '{', right = '}' },
|
|
||||||
['{'] = { left = '{ ', right = ' }' },
|
|
||||||
['>'] = { left = '<', right = '>' },
|
|
||||||
['<'] = { left = '< ', right = ' >' },
|
|
||||||
["'"] = { left = "'", right = "'" },
|
|
||||||
['"'] = { left = '"', right = '"' },
|
|
||||||
['`'] = { left = '`', right = '`' },
|
|
||||||
}
|
|
||||||
|
|
||||||
---@return { left: string; right: string }|nil
|
|
||||||
local function prompt_for_bounds()
|
|
||||||
local cn = vim.fn.getchar()
|
|
||||||
-- Check for non-printable characters:
|
|
||||||
if type(cn) ~= 'number' or cn < 32 or cn > 126 then return end
|
|
||||||
local c = vim.fn.nr2char(cn)
|
|
||||||
|
|
||||||
if c == '<' then
|
|
||||||
-- Surround with a tag:
|
|
||||||
vim.keymap.set('c', '>', '><CR>')
|
|
||||||
local tag = '<' .. vim.fn.input '<'
|
|
||||||
if tag == '<' then return end
|
|
||||||
vim.keymap.del('c', '>')
|
|
||||||
local endtag = '</' .. tag:sub(2):match '[^ >]*' .. '>'
|
|
||||||
-- selene: allow(global_usage)
|
|
||||||
return { left = tag, right = endtag }
|
|
||||||
else
|
|
||||||
-- Default surround:
|
|
||||||
return (surrounds)[c] or { left = c, right = c }
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
---@param range Range
|
|
||||||
---@param bounds { left: string; right: string }
|
|
||||||
local function do_surround(range, bounds)
|
|
||||||
local left = bounds.left
|
|
||||||
local right = bounds.right
|
|
||||||
if range.mode == 'V' then
|
|
||||||
-- If we are surrounding multiple lines, we don't care about
|
|
||||||
-- space-padding:
|
|
||||||
left = vim.trim(left)
|
|
||||||
right = vim.trim(right)
|
|
||||||
end
|
|
||||||
|
|
||||||
if range.mode == 'v' then
|
|
||||||
range:replace(left .. range:text() .. right)
|
|
||||||
elseif range.mode == 'V' then
|
|
||||||
local buf = Buffer.current()
|
|
||||||
local cw = CodeWriter.from_line(buf:line0(range.start.lnum):text(), buf.buf)
|
|
||||||
|
|
||||||
-- write the left bound at the current indent level:
|
|
||||||
cw:write(left)
|
|
||||||
|
|
||||||
local curr_ident_prefix = cw.indent_str:rep(cw.indent_level)
|
|
||||||
cw:indent(function(cw2)
|
|
||||||
for _, line in ipairs(range:lines()) do
|
|
||||||
-- trim the current indent prefix from the line:
|
|
||||||
if line:sub(1, #curr_ident_prefix) == curr_ident_prefix then
|
|
||||||
--
|
|
||||||
line = line:sub(#curr_ident_prefix + 1)
|
|
||||||
end
|
|
||||||
|
|
||||||
cw2:write(line)
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
|
|
||||||
-- write the right bound at the current indent level:
|
|
||||||
cw:write(right)
|
|
||||||
|
|
||||||
range:replace(cw.lines)
|
|
||||||
end
|
|
||||||
|
|
||||||
range.start:save_to_pos '.'
|
|
||||||
end
|
|
||||||
|
|
||||||
function M.setup()
|
|
||||||
require('u.repeat').setup()
|
|
||||||
|
|
||||||
-- Visual
|
|
||||||
vim.keymap.set('v', 'S', function()
|
|
||||||
local c = vim.fn.getcharstr()
|
|
||||||
local range = Range.from_vtext()
|
|
||||||
local bounds = surrounds[c] or { left = c, right = c }
|
|
||||||
vim_repeat.run(function()
|
|
||||||
do_surround(range, bounds)
|
|
||||||
-- this is a visual mapping: end in normal mode:
|
|
||||||
vim.cmd { cmd = 'normal', args = { '' }, bang = true }
|
|
||||||
end)
|
|
||||||
end, { noremap = true, silent = true })
|
|
||||||
|
|
||||||
-- Change
|
|
||||||
vim.keymap.set('n', 'cs', function()
|
|
||||||
local from_cn = vim.fn.getchar()
|
|
||||||
-- Check for non-printable characters:
|
|
||||||
if from_cn < 32 or from_cn > 126 then return end
|
|
||||||
local from_c = vim.fn.nr2char(from_cn)
|
|
||||||
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 })
|
|
||||||
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_info2 = Range.new(arange.stop, arange.stop, 'v'):highlight('IncSearch', { priority = 999 })
|
|
||||||
local hl_clear = function()
|
|
||||||
if hl_info1 then hl_info1.clear() end
|
|
||||||
if hl_info2 then hl_info2.clear() end
|
|
||||||
end
|
|
||||||
|
|
||||||
local to = prompt_for_bounds()
|
|
||||||
hl_clear()
|
|
||||||
if to == nil then return end
|
|
||||||
|
|
||||||
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
|
|
||||||
-- For tags, we want to replace the inner text, not the tag:
|
|
||||||
local irange = Range.from_text_object('i' .. from_c, { user_defined = true })
|
|
||||||
if arange == nil or irange == nil then return nil end
|
|
||||||
|
|
||||||
local lrange = Range.new(arange.start, irange.start:must_next(-1))
|
|
||||||
local rrange = Range.new(irange.stop:must_next(1), arange.stop)
|
|
||||||
|
|
||||||
rrange:replace(to.right)
|
|
||||||
lrange:replace(to.left)
|
|
||||||
else
|
|
||||||
-- replace `from.right` with `to.right`:
|
|
||||||
local last_line = arange:line0(-1).text() --[[@as string]]
|
|
||||||
local from_right_match = last_line:match(vim.pesc(from.right) .. '$')
|
|
||||||
if from_right_match then
|
|
||||||
local match_start = arange.stop:clone()
|
|
||||||
match_start.col = match_start.col - #from_right_match + 1
|
|
||||||
Range.new(match_start, arange.stop):replace(to.right)
|
|
||||||
end
|
|
||||||
|
|
||||||
-- replace `from.left` with `to.left`:
|
|
||||||
local first_line = arange:line0(0).text() --[[@as string]]
|
|
||||||
local from_left_match = first_line:match('^' .. vim.pesc(from.left))
|
|
||||||
if from_left_match then
|
|
||||||
local match_end = arange.start:clone()
|
|
||||||
match_end.col = match_end.col + #from_left_match - 1
|
|
||||||
Range.new(arange.start, match_end):replace(to.left)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
end, { noremap = true, silent = true })
|
|
||||||
|
|
||||||
-- Delete
|
|
||||||
vim.keymap.set('n', 'ds', function()
|
|
||||||
local txt_obj = vim.fn.getcharstr()
|
|
||||||
vim_repeat.run(function()
|
|
||||||
local buf = Buffer.current()
|
|
||||||
local irange = Range.from_text_object('i' .. txt_obj)
|
|
||||||
local arange = Range.from_text_object('a' .. txt_obj)
|
|
||||||
if arange == nil or irange == nil then return nil end
|
|
||||||
local starting_cursor_pos = arange.start:clone()
|
|
||||||
|
|
||||||
-- Now, replace `arange` with the content of `irange`. If `arange` was multiple lines,
|
|
||||||
-- dedent the contents first, and operate in linewise mode
|
|
||||||
if arange.start.lnum ~= arange.stop.lnum then
|
|
||||||
-- Auto dedent:
|
|
||||||
vim.cmd.normal('<i' .. vim.trim(txt_obj))
|
|
||||||
-- Dedenting moves the cursor, so we need to set the cursor to a consistent starting spot:
|
|
||||||
arange.start:save_to_pos '.'
|
|
||||||
-- Dedenting also changed the inner text, so re-acquire it:
|
|
||||||
arange = Range.from_text_object('a' .. txt_obj)
|
|
||||||
irange = Range.from_text_object('i' .. txt_obj)
|
|
||||||
if arange == nil or irange == nil then return end -- should never be true
|
|
||||||
arange:replace(irange:lines())
|
|
||||||
|
|
||||||
local final_range = Range.new(
|
|
||||||
arange.start,
|
|
||||||
Pos.new(
|
|
||||||
arange.stop.buf,
|
|
||||||
irange.start.lnum + (arange.stop.lnum + arange.start.lnum),
|
|
||||||
arange.stop.col,
|
|
||||||
arange.stop.off
|
|
||||||
),
|
|
||||||
irange.mode
|
|
||||||
)
|
|
||||||
|
|
||||||
-- delete last line, if it is empty:
|
|
||||||
local last = buf:line0(final_range.stop.lnum)
|
|
||||||
if last:text():match '^%s*$' then last:replace(nil) end
|
|
||||||
|
|
||||||
-- delete first line, if it is empty:
|
|
||||||
local first = buf:line0(final_range.start.lnum)
|
|
||||||
if first:text():match '^%s*$' then first:replace(nil) end
|
|
||||||
else
|
|
||||||
-- trim start:
|
|
||||||
irange = irange:trim_start():trim_stop()
|
|
||||||
arange:replace(irange:lines())
|
|
||||||
end
|
|
||||||
|
|
||||||
starting_cursor_pos:save_to_pos '.'
|
|
||||||
end)
|
|
||||||
end, { noremap = true, silent = true })
|
|
||||||
|
|
||||||
opkeymap('n', 'ys', function(range)
|
|
||||||
local hl_info = range:highlight('IncSearch', { priority = 999 })
|
|
||||||
|
|
||||||
---@type { left: string; right: string }
|
|
||||||
local bounds
|
|
||||||
-- selene: allow(global_usage)
|
|
||||||
if _G.my_surround_bounds ~= nil then
|
|
||||||
-- This command was repeated with `.`, we don't need
|
|
||||||
-- to prompt for the bounds:
|
|
||||||
-- selene: allow(global_usage)
|
|
||||||
bounds = _G.my_surround_bounds
|
|
||||||
else
|
|
||||||
local prompted_bounds = prompt_for_bounds()
|
|
||||||
if prompted_bounds == nil and hl_info then return hl_info.clear() end
|
|
||||||
if prompted_bounds then bounds = prompted_bounds end
|
|
||||||
end
|
|
||||||
|
|
||||||
if hl_info then hl_info.clear() end
|
|
||||||
do_surround(range, bounds)
|
|
||||||
-- selene: allow(global_usage)
|
|
||||||
_G.my_surround_bounds = nil
|
|
||||||
|
|
||||||
-- return repeatable injection
|
|
||||||
return function()
|
|
||||||
-- on_repeat, we "stage" the bounds that we were originally called with:
|
|
||||||
-- selene: allow(global_usage)
|
|
||||||
_G.my_surround_bounds = bounds
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
|
|
||||||
return M
|
|
@ -1,126 +0,0 @@
|
|||||||
local utils = require 'u.utils'
|
|
||||||
local Pos = require 'u.pos'
|
|
||||||
local Range = require 'u.range'
|
|
||||||
local Buffer = require 'u.buffer'
|
|
||||||
|
|
||||||
local M = {}
|
|
||||||
|
|
||||||
function M.setup()
|
|
||||||
-- Select whole file:
|
|
||||||
utils.define_text_object('ag', function() return Buffer.current():all() end)
|
|
||||||
|
|
||||||
-- Select current line:
|
|
||||||
utils.define_text_object('a.', function()
|
|
||||||
local lnum = Pos.from_pos('.').lnum
|
|
||||||
return Buffer.current():line0(lnum)
|
|
||||||
end)
|
|
||||||
|
|
||||||
-- Select the nearest quote:
|
|
||||||
utils.define_text_object('aq', function() return Range.find_nearest_quotes() end)
|
|
||||||
utils.define_text_object('iq', function()
|
|
||||||
local range = Range.find_nearest_quotes()
|
|
||||||
if range == nil then return end
|
|
||||||
return range:shrink(1)
|
|
||||||
end)
|
|
||||||
|
|
||||||
---Selects the next quote object (searches forward)
|
|
||||||
---@param q string
|
|
||||||
local function define_quote_obj(q)
|
|
||||||
local function select_around()
|
|
||||||
-- Operator mappings are effectively running in visual mode, the way
|
|
||||||
-- `define_text_object` is implemented, so feed the keys `a${q}` to vim
|
|
||||||
-- to select the appropriate text-object
|
|
||||||
vim.cmd { cmd = 'normal', args = { 'a' .. q }, bang = true }
|
|
||||||
|
|
||||||
-- Now check on the visually selected text:
|
|
||||||
local range = Range.from_vtext()
|
|
||||||
if range:is_empty() then return range.start end
|
|
||||||
range.start = range.start:find_next(1, q) or range.start
|
|
||||||
range.stop = range.stop:find_next(-1, q) or range.stop
|
|
||||||
return range
|
|
||||||
end
|
|
||||||
|
|
||||||
utils.define_text_object('a' .. q, function() return select_around() end)
|
|
||||||
utils.define_text_object('i' .. q, function()
|
|
||||||
local range_or_pos = select_around()
|
|
||||||
if Range.is(range_or_pos) then
|
|
||||||
local start_next = range_or_pos.start:next(1)
|
|
||||||
local stop_prev = range_or_pos.stop:next(-1)
|
|
||||||
if start_next > stop_prev then return start_next end
|
|
||||||
|
|
||||||
local range = range_or_pos:shrink(1)
|
|
||||||
return range
|
|
||||||
else
|
|
||||||
return range_or_pos
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
define_quote_obj [["]]
|
|
||||||
define_quote_obj [[']]
|
|
||||||
define_quote_obj [[`]]
|
|
||||||
|
|
||||||
---Selects the "last" quote object (searches backward)
|
|
||||||
---@param q string
|
|
||||||
local function define_last_quote_obj(q)
|
|
||||||
local function select_around()
|
|
||||||
local curr = Pos.from_pos('.'):find_next(-1, q)
|
|
||||||
if not curr then return end
|
|
||||||
-- Reset visual selection to current context:
|
|
||||||
Range.new(curr, curr):set_visual_selection()
|
|
||||||
vim.cmd.normal('a' .. q)
|
|
||||||
local range = Range.from_vtext()
|
|
||||||
if range:is_empty() then return range.start end
|
|
||||||
range.start = range.start:find_next(1, q) or range.start
|
|
||||||
range.stop = range.stop:find_next(-1, q) or range.stop
|
|
||||||
return range
|
|
||||||
end
|
|
||||||
|
|
||||||
utils.define_text_object('al' .. q, function() return select_around() end)
|
|
||||||
utils.define_text_object('il' .. q, function()
|
|
||||||
local range_or_pos = select_around()
|
|
||||||
if range_or_pos == nil then return end
|
|
||||||
|
|
||||||
if Range.is(range_or_pos) then
|
|
||||||
local start_next = range_or_pos.start:next(1)
|
|
||||||
local stop_prev = range_or_pos.stop:next(-1)
|
|
||||||
if start_next > stop_prev then return start_next end
|
|
||||||
|
|
||||||
local range = range_or_pos:shrink(1)
|
|
||||||
return range
|
|
||||||
else
|
|
||||||
return range_or_pos
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
define_last_quote_obj [["]]
|
|
||||||
define_last_quote_obj [[']]
|
|
||||||
define_last_quote_obj [[`]]
|
|
||||||
|
|
||||||
-- Selects the "last" bracket object (searches backward):
|
|
||||||
local function define_last_bracket_obj(b, ...)
|
|
||||||
local function select_around()
|
|
||||||
local curr = Pos.from_pos('.'):find_next(-1, b)
|
|
||||||
if not curr then return end
|
|
||||||
|
|
||||||
local other = curr:find_match(1000)
|
|
||||||
if not other then return end
|
|
||||||
|
|
||||||
return Range.new(curr, other)
|
|
||||||
end
|
|
||||||
|
|
||||||
local keybinds = { ... }
|
|
||||||
table.insert(keybinds, b)
|
|
||||||
for _, k in ipairs(keybinds) do
|
|
||||||
utils.define_text_object('al' .. k, function() return select_around() end)
|
|
||||||
utils.define_text_object('il' .. k, function()
|
|
||||||
local range = select_around()
|
|
||||||
return range and range:shrink(1)
|
|
||||||
end)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
define_last_bracket_obj('}', 'B')
|
|
||||||
define_last_bracket_obj ']'
|
|
||||||
define_last_bracket_obj(')', 'b')
|
|
||||||
define_last_bracket_obj '>'
|
|
||||||
end
|
|
||||||
return M
|
|
@ -1,21 +1,16 @@
|
|||||||
local Range = require 'u.range'
|
local Range = require 'u.range'
|
||||||
local Renderer = require 'u.renderer'
|
|
||||||
|
|
||||||
---@class Buffer
|
---@class Buffer
|
||||||
---@field buf number
|
---@field buf number
|
||||||
---@field private renderer Renderer
|
|
||||||
local Buffer = {}
|
local Buffer = {}
|
||||||
|
|
||||||
---@param buf? number
|
---@param buf? number
|
||||||
---@return Buffer
|
---@return Buffer
|
||||||
function Buffer.from_nr(buf)
|
function Buffer.from_nr(buf)
|
||||||
if buf == nil or buf == 0 then buf = vim.api.nvim_get_current_buf() end
|
if buf == nil or buf == 0 then buf = vim.api.nvim_get_current_buf() end
|
||||||
|
local b = { buf = buf }
|
||||||
local renderer = Renderer.new(buf)
|
setmetatable(b, { __index = Buffer })
|
||||||
return setmetatable({
|
return b
|
||||||
buf = buf,
|
|
||||||
renderer = renderer,
|
|
||||||
}, { __index = Buffer })
|
|
||||||
end
|
end
|
||||||
|
|
||||||
---@return Buffer
|
---@return Buffer
|
||||||
@ -74,13 +69,4 @@ function Buffer:text_object(txt_obj, opts)
|
|||||||
return Range.from_text_object(txt_obj, opts)
|
return Range.from_text_object(txt_obj, opts)
|
||||||
end
|
end
|
||||||
|
|
||||||
--- @param event string|string[]
|
|
||||||
--- @param opts vim.api.keyset.create_autocmd
|
|
||||||
function Buffer:autocmd(event, opts)
|
|
||||||
vim.api.nvim_create_autocmd(event, vim.tbl_extend('force', opts, { buffer = self.buf }))
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param markup string
|
|
||||||
function Buffer:render(markup) return self.renderer:render(markup) end
|
|
||||||
|
|
||||||
return Buffer
|
return Buffer
|
||||||
|
126
lua/u/range.lua
126
lua/u/range.lua
@ -10,16 +10,16 @@ end
|
|||||||
|
|
||||||
---@class Range
|
---@class Range
|
||||||
---@field start Pos
|
---@field start Pos
|
||||||
---@field stop Pos|nil
|
---@field stop Pos
|
||||||
---@field mode 'v'|'V'
|
---@field mode 'v'|'V'
|
||||||
local Range = {}
|
local Range = {}
|
||||||
|
|
||||||
---@param start Pos
|
---@param start Pos
|
||||||
---@param stop Pos|nil
|
---@param stop Pos
|
||||||
---@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 ~= nil and stop < start then
|
if stop < start then
|
||||||
start, stop = stop, start
|
start, stop = stop, start
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -27,9 +27,7 @@ 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 == nil then
|
if p.off ~= 0 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)
|
||||||
@ -256,27 +254,22 @@ function Range.smallest(ranges)
|
|||||||
return result
|
return result
|
||||||
end
|
end
|
||||||
|
|
||||||
function Range:clone() return Range.new(self.start:clone(), self.stop ~= nil and self.stop:clone() or nil, self.mode) end
|
function Range:clone() return Range.new(self.start:clone(), self.stop:clone(), self.mode) end
|
||||||
function Range:line_count()
|
function Range:line_count() return self.stop.lnum - self.start.lnum + 1 end
|
||||||
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
|
||||||
if r.stop ~= nil then r.stop.col = Pos.MAX_COL end
|
r.stop.col = Pos.MAX_COL
|
||||||
|
|
||||||
return r
|
return r
|
||||||
end
|
end
|
||||||
|
|
||||||
function Range:is_empty() return self.stop == nil end
|
function Range:is_empty() return self.start == self.stop 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)
|
||||||
@ -287,8 +280,6 @@ 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)
|
||||||
@ -299,12 +290,10 @@ function Range:trim_stop()
|
|||||||
end
|
end
|
||||||
|
|
||||||
---@param p Pos
|
---@param p Pos
|
||||||
function Range:contains(p) return not self:is_empty() and p >= self.start and p <= self.stop end
|
function Range:contains(p) return 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)
|
||||||
@ -324,8 +313,6 @@ 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
|
||||||
|
|
||||||
@ -353,57 +340,30 @@ 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
|
||||||
|
|
||||||
local buf = self.start.buf
|
if replacement == nil and self.mode == 'V' then
|
||||||
-- convert to start-inclusive, stop-exclusive coordinates:
|
-- delete the lines:
|
||||||
local start_lnum, stop_lnum = self.start.lnum, (self.stop and self.stop.lnum or self.start.lnum) + 1
|
vim.api.nvim_buf_set_lines(self.start.buf, self.start.lnum, self.stop.lnum + 1, false, {})
|
||||||
local start_col, stop_col = self.start.col, (self.stop and self.stop.col or self.start.col) + 1
|
|
||||||
|
|
||||||
local replace_type = (self:is_empty() and 'insert') or (self.mode == 'v' and 'region') or 'lines'
|
|
||||||
|
|
||||||
---@param alnum number
|
|
||||||
---@param acol number
|
|
||||||
---@param blnum number
|
|
||||||
---@param bcol number
|
|
||||||
local function set_text(alnum, acol, blnum, bcol, repl)
|
|
||||||
-- row indices are end-inclusive, and column indices are end-exclusive.
|
|
||||||
vim.api.nvim_buf_set_text(buf, alnum, acol, blnum, bcol, repl)
|
|
||||||
|
|
||||||
local new_last_line_num = self.start.lnum + #replacement - 1
|
|
||||||
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
|
|
||||||
|
|
||||||
---@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
|
else
|
||||||
error 'unreachable'
|
if replacement == nil then replacement = { '' } end
|
||||||
|
|
||||||
|
if self.mode == 'v' then
|
||||||
|
-- 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(
|
||||||
|
self.start.buf,
|
||||||
|
self.start.lnum,
|
||||||
|
self.start.col,
|
||||||
|
self.stop.lnum,
|
||||||
|
math.min(self.stop.col + 1, max_col),
|
||||||
|
replacement
|
||||||
|
)
|
||||||
|
else
|
||||||
|
vim.api.nvim_buf_set_lines(self.start.buf, self.start.lnum, self.stop.lnum + 1, false, replacement)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -411,8 +371,6 @@ 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)
|
||||||
@ -421,7 +379,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 = nil end
|
if start > stop then stop = start end
|
||||||
return Range.new(start, stop, self.mode)
|
return Range.new(start, stop, self.mode)
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -435,30 +393,18 @@ 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(left)
|
self.stop:save_to_pos(right)
|
||||||
self.start:save_to_pos(right)
|
|
||||||
else
|
|
||||||
self.start:save_to_pos(left)
|
|
||||||
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(left)
|
self.stop:save_to_mark(right)
|
||||||
self.start:save_to_mark(right)
|
|
||||||
else
|
|
||||||
self.start:save_to_mark(left)
|
|
||||||
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)
|
||||||
@ -481,8 +427,6 @@ 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
|
||||||
|
|
||||||
|
@ -1,306 +0,0 @@
|
|||||||
local utils = require 'u.utils'
|
|
||||||
local str = require 'u.utils.string'
|
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
|
||||||
-- Renderer class
|
|
||||||
--------------------------------------------------------------------------------
|
|
||||||
--- @alias RendererHighlight { group: string; start: [number, number]; stop: [number, number ] }
|
|
||||||
|
|
||||||
--- @class Renderer
|
|
||||||
--- @field bufnr number
|
|
||||||
--- @field ns number
|
|
||||||
--- @field changedtick number
|
|
||||||
--- @field old { lines: string[]; hls: RendererHighlight[] }
|
|
||||||
--- @field curr { lines: string[]; hls: RendererHighlight[] }
|
|
||||||
local Renderer = {}
|
|
||||||
Renderer.__index = Renderer
|
|
||||||
|
|
||||||
--- @param bufnr number|nil
|
|
||||||
function Renderer.new(bufnr)
|
|
||||||
if bufnr == nil then bufnr = vim.api.nvim_get_current_buf() end
|
|
||||||
|
|
||||||
local self = setmetatable({
|
|
||||||
bufnr = bufnr,
|
|
||||||
ns = vim.api.nvim_create_namespace '',
|
|
||||||
changedtick = 0,
|
|
||||||
old = { lines = {}, hls = {} },
|
|
||||||
curr = { lines = {}, hls = {} },
|
|
||||||
}, Renderer)
|
|
||||||
return self
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param markup string
|
|
||||||
function Renderer:render(markup)
|
|
||||||
local changedtick = vim.b[self.bufnr].changedtick
|
|
||||||
if changedtick ~= self.changedtick then
|
|
||||||
self.curr = { lines = vim.api.nvim_buf_get_lines(self.bufnr, 0, -1, false), hls = {} }
|
|
||||||
self.changedtick = changedtick
|
|
||||||
end
|
|
||||||
|
|
||||||
local nodes = self._parse_markup(markup)
|
|
||||||
|
|
||||||
--- @type string[]
|
|
||||||
local lines = {}
|
|
||||||
--- @type RendererHighlight[]
|
|
||||||
local hls = {}
|
|
||||||
|
|
||||||
local curr_line1 = 1
|
|
||||||
local curr_col1 = 1 -- exclusive: sits one position **beyond** the last inserted text
|
|
||||||
--- @param s string
|
|
||||||
local function put(s)
|
|
||||||
lines[curr_line1] = (lines[curr_line1] or '') .. s
|
|
||||||
curr_col1 = #lines[curr_line1] + 1
|
|
||||||
end
|
|
||||||
local function put_line()
|
|
||||||
table.insert(lines, '')
|
|
||||||
curr_line1 = curr_line1 + 1
|
|
||||||
curr_col1 = 1
|
|
||||||
end
|
|
||||||
|
|
||||||
for _, node in ipairs(nodes) do
|
|
||||||
if node.kind == 'text' then
|
|
||||||
local node_lines = vim.split(node.value, '\n')
|
|
||||||
for lnum, s in ipairs(node_lines) do
|
|
||||||
if lnum > 1 then put_line() end
|
|
||||||
put(s)
|
|
||||||
end
|
|
||||||
elseif node.kind == 'tag' then
|
|
||||||
local function attr_num(nm)
|
|
||||||
if not node.attributes[nm] then return end
|
|
||||||
return tonumber(node.attributes[nm])
|
|
||||||
end
|
|
||||||
local function attr_bool(nm)
|
|
||||||
if not node.attributes[nm] then return end
|
|
||||||
return node.attributes[nm] and true or false
|
|
||||||
end
|
|
||||||
|
|
||||||
if node.name == 't' then
|
|
||||||
local value = node.attributes.value
|
|
||||||
if type(value) == 'string' then
|
|
||||||
local start0 = { curr_line1 - 1, curr_col1 - 1 }
|
|
||||||
local value_lines = vim.split(value, '\n')
|
|
||||||
for lnum, value_line in ipairs(value_lines) do
|
|
||||||
if lnum > 1 then put_line() end
|
|
||||||
put(value_line)
|
|
||||||
end
|
|
||||||
local stop0 = { curr_line1 - 1, curr_col1 - 1 }
|
|
||||||
|
|
||||||
local group = node.attributes.hl
|
|
||||||
if type(group) == 'string' then
|
|
||||||
local local_exists = #vim.tbl_keys(vim.api.nvim_get_hl(self.ns, { name = group })) > 0
|
|
||||||
local global_exists = #vim.tbl_keys(vim.api.nvim_get_hl(0, { name = group }))
|
|
||||||
local exists = local_exists or global_exists
|
|
||||||
|
|
||||||
if not exists or attr_bool 'hl:force' then
|
|
||||||
vim.api.nvim_set_hl_ns(self.ns)
|
|
||||||
vim.api.nvim_set_hl(self.ns, group, {
|
|
||||||
fg = node.attributes['hl:fg'],
|
|
||||||
bg = node.attributes['hl:bg'],
|
|
||||||
sp = node.attributes['hl:sp'],
|
|
||||||
blend = attr_num 'hl:blend',
|
|
||||||
bold = attr_bool 'hl:bold',
|
|
||||||
standout = attr_bool 'hl:standout',
|
|
||||||
underline = attr_bool 'hl:underline',
|
|
||||||
undercurl = attr_bool 'hl:undercurl',
|
|
||||||
underdouble = attr_bool 'hl:underdouble',
|
|
||||||
underdotted = attr_bool 'hl:underdotted',
|
|
||||||
underdashed = attr_bool 'hl:underdashed',
|
|
||||||
strikethrough = attr_bool 'hl:strikethrough',
|
|
||||||
italic = attr_bool 'hl:italic',
|
|
||||||
reverse = attr_bool 'hl:reverse',
|
|
||||||
nocombine = attr_bool 'hl:nocombine',
|
|
||||||
link = node.attributes['hl:link'],
|
|
||||||
default = attr_bool 'hl:default',
|
|
||||||
ctermfg = attr_num 'hl:ctermfg',
|
|
||||||
ctermbg = attr_num 'hl:ctermbg',
|
|
||||||
cterm = node.attributes['hl:cterm'],
|
|
||||||
force = attr_bool 'hl:force',
|
|
||||||
})
|
|
||||||
end
|
|
||||||
end
|
|
||||||
if type(group) == 'string' then table.insert(hls, { group = group, start = start0, stop = stop0 }) end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
self.old = self.curr
|
|
||||||
self.curr = { lines = lines, hls = hls }
|
|
||||||
self:_reconcile()
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @private
|
|
||||||
--- @param info string
|
|
||||||
--- @param start integer
|
|
||||||
--- @param end_ integer
|
|
||||||
--- @param strict_indexing boolean
|
|
||||||
--- @param replacement string[]
|
|
||||||
function Renderer:_set_lines(info, start, end_, strict_indexing, replacement)
|
|
||||||
self:_log { 'set_lines', self.bufnr, start, end_, strict_indexing, replacement }
|
|
||||||
vim.api.nvim_buf_set_lines(self.bufnr, start, end_, strict_indexing, replacement)
|
|
||||||
self:_log { 'after(' .. info .. ')', vim.api.nvim_buf_get_lines(self.bufnr, 0, -1, false) }
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @private
|
|
||||||
--- @param info string
|
|
||||||
--- @param start_row integer
|
|
||||||
--- @param start_col integer
|
|
||||||
--- @param end_row integer
|
|
||||||
--- @param end_col integer
|
|
||||||
--- @param replacement string[]
|
|
||||||
function Renderer:_set_text(info, start_row, start_col, end_row, end_col, replacement)
|
|
||||||
self:_log { 'set_text', self.bufnr, start_row, start_col, end_row, end_col, replacement }
|
|
||||||
vim.api.nvim_buf_set_text(self.bufnr, start_row, start_col, end_row, end_col, replacement)
|
|
||||||
self:_log { 'after(' .. info .. ')', vim.api.nvim_buf_get_lines(self.bufnr, 0, -1, false) }
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @private
|
|
||||||
function Renderer:_log(...)
|
|
||||||
--
|
|
||||||
-- vim.print(...)
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @private
|
|
||||||
function Renderer:_reconcile()
|
|
||||||
local line_changes = utils.levenshtein(self.old.lines, self.curr.lines)
|
|
||||||
self.old = self.curr
|
|
||||||
|
|
||||||
self:_log { line_changes = line_changes }
|
|
||||||
for _, line_change in ipairs(line_changes) do
|
|
||||||
local lnum0 = line_change.index - 1
|
|
||||||
|
|
||||||
if line_change.kind == 'add' then
|
|
||||||
self:_set_lines('add-line', lnum0, lnum0, true, { line_change.item })
|
|
||||||
elseif line_change.kind == 'change' then
|
|
||||||
-- Compute inter-line diff, and apply:
|
|
||||||
self:_log '--------------------------------------------------------------------------------'
|
|
||||||
local col_changes = utils.levenshtein(vim.split(line_change.from, ''), vim.split(line_change.to, ''))
|
|
||||||
|
|
||||||
for _, col_change in ipairs(col_changes) do
|
|
||||||
local cnum0 = col_change.index - 1
|
|
||||||
self:_log { line_change = col_change, cnum = cnum0, lnum = lnum0 }
|
|
||||||
if col_change.kind == 'add' then
|
|
||||||
self:_set_text('add-char', lnum0, cnum0, lnum0, cnum0, { col_change.item })
|
|
||||||
elseif col_change.kind == 'change' then
|
|
||||||
self:_set_text('change-char', lnum0, cnum0, lnum0, cnum0 + 1, { col_change.to })
|
|
||||||
elseif col_change.kind == 'delete' then
|
|
||||||
self:_set_text('del-char', lnum0, cnum0, lnum0, cnum0 + 1, {})
|
|
||||||
else
|
|
||||||
-- No change
|
|
||||||
end
|
|
||||||
end
|
|
||||||
elseif line_change.kind == 'delete' then
|
|
||||||
self:_set_lines('del-line', lnum0, lnum0 + 1, true, {})
|
|
||||||
else
|
|
||||||
-- No change
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- vim.api.nvim_buf_set_lines(self.bufnr, 0, -1, false, self.curr.lines)
|
|
||||||
vim.api.nvim_buf_clear_namespace(self.bufnr, self.ns, 0, -1)
|
|
||||||
for _, hl in ipairs(self.curr.hls) do
|
|
||||||
vim.highlight.range(self.bufnr, self.ns, hl.group, hl.start, hl.stop, {
|
|
||||||
inclusive = false,
|
|
||||||
priority = vim.highlight.priorities.user,
|
|
||||||
regtype = 'charwise',
|
|
||||||
})
|
|
||||||
end
|
|
||||||
|
|
||||||
self.changedtick = vim.b[self.bufnr].changedtick
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @private
|
|
||||||
--- @param markup string e.g., [[Something <t hl="My highlight" value="my text" />]]
|
|
||||||
function Renderer._parse_markup(markup)
|
|
||||||
--- @type ({ kind: 'text'; value: string } | { kind: 'tag'; name: string; attributes: table<string, string|boolean> })[]
|
|
||||||
local nodes = {}
|
|
||||||
|
|
||||||
--- @type 'text' | 'tag'
|
|
||||||
local mode = 'text'
|
|
||||||
local pos = 1
|
|
||||||
local watchdog = 0
|
|
||||||
|
|
||||||
local function skip_whitespace()
|
|
||||||
local _, new_pos = str.eat_while(markup, pos, str.is_whitespace)
|
|
||||||
pos = new_pos
|
|
||||||
end
|
|
||||||
local function check_infinite_loop()
|
|
||||||
watchdog = watchdog + 1
|
|
||||||
if watchdog > #markup then
|
|
||||||
vim.print('ERROR', {
|
|
||||||
num_nodes = #nodes,
|
|
||||||
last_node = nodes[#nodes],
|
|
||||||
pos = pos,
|
|
||||||
len = #markup,
|
|
||||||
})
|
|
||||||
error 'infinite loop'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
while pos <= #markup do
|
|
||||||
check_infinite_loop()
|
|
||||||
|
|
||||||
if mode == 'text' then
|
|
||||||
--
|
|
||||||
-- Parse contiguous regions of text
|
|
||||||
--
|
|
||||||
local eaten, new_pos = str.eat_while(markup, pos, function(c) return c ~= '<' end)
|
|
||||||
if #eaten > 0 then table.insert(nodes, { kind = 'text', value = eaten:gsub('<', '<') }) end
|
|
||||||
pos = new_pos
|
|
||||||
|
|
||||||
if markup:sub(pos, pos) == '<' then mode = 'tag' end
|
|
||||||
elseif mode == 'tag' then
|
|
||||||
--
|
|
||||||
-- Parse self-closing tags
|
|
||||||
--
|
|
||||||
if markup:sub(pos, pos) == '<' then pos = pos + 1 end
|
|
||||||
local tag_name, new_pos = str.eat_while(markup, pos, function(c) return not str.is_whitespace(c) end)
|
|
||||||
pos = new_pos
|
|
||||||
|
|
||||||
if tag_name == '/>' then
|
|
||||||
-- empty tag
|
|
||||||
table.insert(nodes, { kind = 'tag', name = '', attributes = {} })
|
|
||||||
else
|
|
||||||
local node = { kind = 'tag', name = tag_name, attributes = {} }
|
|
||||||
skip_whitespace()
|
|
||||||
|
|
||||||
while markup:sub(pos, pos + 1) ~= '/>' do
|
|
||||||
check_infinite_loop()
|
|
||||||
if pos > #markup then error 'unexpected end of markup' end
|
|
||||||
|
|
||||||
local attr_name
|
|
||||||
attr_name, new_pos = str.eat_while(markup, pos, function(c) return c ~= '=' and not str.is_whitespace(c) end)
|
|
||||||
pos = new_pos
|
|
||||||
|
|
||||||
local attr_value = nil
|
|
||||||
if markup:sub(pos, pos) == '=' then
|
|
||||||
pos = pos + 1
|
|
||||||
if markup:sub(pos, pos) == '"' then
|
|
||||||
pos = pos + 1
|
|
||||||
attr_value, new_pos = str.eat_while(markup, pos, function(c, i, s)
|
|
||||||
local prev_c = s:sub(i - 1, i - 1)
|
|
||||||
return c ~= '"' or (prev_c == '\\' and c == '"')
|
|
||||||
end)
|
|
||||||
pos = new_pos + 1 -- skip the closing '"'
|
|
||||||
else
|
|
||||||
attr_value, new_pos = str.eat_while(markup, pos, function(c) return not str.is_whitespace(c) end)
|
|
||||||
pos = new_pos
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
node.attributes[attr_name] = (attr_value and attr_value:gsub('\\"', '"')) or true
|
|
||||||
skip_whitespace()
|
|
||||||
end
|
|
||||||
pos = pos + 2 -- skip the '/>'
|
|
||||||
|
|
||||||
table.insert(nodes, node)
|
|
||||||
end
|
|
||||||
|
|
||||||
mode = 'text'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
return nodes
|
|
||||||
end
|
|
||||||
|
|
||||||
return Renderer
|
|
143
lua/u/signal.lua
143
lua/u/signal.lua
@ -1,143 +0,0 @@
|
|||||||
local M = {}
|
|
||||||
|
|
||||||
--- @class Signal
|
|
||||||
--- @field value any
|
|
||||||
--- @field subscribers table<function, boolean>
|
|
||||||
local Signal = {}
|
|
||||||
M.Signal = Signal
|
|
||||||
Signal.__index = Signal
|
|
||||||
|
|
||||||
--- @param value any
|
|
||||||
--- @return Signal
|
|
||||||
function Signal:new(value)
|
|
||||||
local obj = setmetatable({
|
|
||||||
value = value,
|
|
||||||
subscribers = {},
|
|
||||||
}, self)
|
|
||||||
return obj
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param value any
|
|
||||||
function Signal:set(value)
|
|
||||||
self.value = value
|
|
||||||
self:_notify()
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @return any
|
|
||||||
function Signal:get()
|
|
||||||
local ctx = M.ExecutionContext.current()
|
|
||||||
if ctx then ctx:track(self) end
|
|
||||||
return self.value
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param fn function
|
|
||||||
function Signal:update(fn) self:set(fn(self.value)) end
|
|
||||||
|
|
||||||
--- @private
|
|
||||||
function Signal:_notify()
|
|
||||||
for _, cb in ipairs(self.subscribers) do
|
|
||||||
cb(self.value)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param callback function
|
|
||||||
function Signal:subscribe(callback)
|
|
||||||
table.insert(self.subscribers, callback)
|
|
||||||
return function() self:unsubscribe(callback) end
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param callback function
|
|
||||||
function Signal:unsubscribe(callback)
|
|
||||||
for i, cb in ipairs(self.subscribers) do
|
|
||||||
if cb == callback then
|
|
||||||
table.remove(self.subscribers, i)
|
|
||||||
break
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
function Signal:dispose() self.subscribers = {} end
|
|
||||||
|
|
||||||
CURRENT_CONTEXT = nil
|
|
||||||
|
|
||||||
--- @class ExecutionContext
|
|
||||||
--- @field signals table<Signal, boolean>
|
|
||||||
--- @field subscribers table<function, boolean>
|
|
||||||
local ExecutionContext = {}
|
|
||||||
M.ExecutionContext = ExecutionContext
|
|
||||||
ExecutionContext.__index = ExecutionContext
|
|
||||||
|
|
||||||
--- @return ExecutionContext
|
|
||||||
function ExecutionContext:new()
|
|
||||||
return setmetatable({
|
|
||||||
signals = {},
|
|
||||||
subscribers = {},
|
|
||||||
}, ExecutionContext)
|
|
||||||
end
|
|
||||||
|
|
||||||
function ExecutionContext.current() return CURRENT_CONTEXT end
|
|
||||||
|
|
||||||
--- @param fn function
|
|
||||||
--- @param ctx ExecutionContext
|
|
||||||
function ExecutionContext:run(fn, ctx)
|
|
||||||
local oldCtx = CURRENT_CONTEXT
|
|
||||||
CURRENT_CONTEXT = ctx
|
|
||||||
local result
|
|
||||||
local success, err = pcall(function() result = fn() end)
|
|
||||||
|
|
||||||
CURRENT_CONTEXT = oldCtx
|
|
||||||
|
|
||||||
if not success then error(err) end
|
|
||||||
|
|
||||||
return result
|
|
||||||
end
|
|
||||||
|
|
||||||
function ExecutionContext:track(signal) self.signals[signal] = true end
|
|
||||||
|
|
||||||
--- @param callback function
|
|
||||||
function ExecutionContext:subscribe(callback)
|
|
||||||
self.subscribers[callback] = true
|
|
||||||
for signal in pairs(self.signals) do
|
|
||||||
signal:subscribe(function() callback() end)
|
|
||||||
end
|
|
||||||
return function() self:unsubscribe(callback) end
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param callback function
|
|
||||||
function ExecutionContext:unsubscribe(callback) self.subscribers[callback] = nil end
|
|
||||||
|
|
||||||
function ExecutionContext:dispose()
|
|
||||||
for signal, _ in pairs(self.signals) do
|
|
||||||
signal:dispose()
|
|
||||||
end
|
|
||||||
self.signals = {}
|
|
||||||
self.subscribers = {}
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param value any
|
|
||||||
--- @return Signal
|
|
||||||
function M.create_signal(value) return Signal:new(value) end
|
|
||||||
|
|
||||||
--- @param fn function
|
|
||||||
--- @return Signal
|
|
||||||
function M.create_memo(fn)
|
|
||||||
local result
|
|
||||||
M.create_effect(function()
|
|
||||||
local value = fn()
|
|
||||||
if result then
|
|
||||||
result:set(value)
|
|
||||||
else
|
|
||||||
result = M.create_signal(value)
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
return result
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param fn function
|
|
||||||
function M.create_effect(fn)
|
|
||||||
local ctx = M.ExecutionContext:new()
|
|
||||||
M.ExecutionContext:run(fn, ctx)
|
|
||||||
return ctx:subscribe(fn)
|
|
||||||
end
|
|
||||||
|
|
||||||
return M
|
|
@ -49,7 +49,6 @@ 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]]
|
||||||
@ -68,7 +67,6 @@ 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()
|
||||||
@ -131,95 +129,4 @@ function M.get_editor_dimensions()
|
|||||||
return { width = w, height = h }
|
return { width = w, height = h }
|
||||||
end
|
end
|
||||||
|
|
||||||
--- @alias LevenshteinChange<T> ({ kind: 'add'; item: T; index: number; } | { kind: 'delete'; item: T; index: number; } | { kind: 'change'; from: T; to: T; index: number; })
|
|
||||||
--- @private
|
|
||||||
--- @generic T
|
|
||||||
--- @param x `T`[]
|
|
||||||
--- @param y T[]
|
|
||||||
--- @param cost? { of_delete?: fun(x: T): number; of_add?: fun(x: T): number; of_change?: fun(x: T, y: T): number; }
|
|
||||||
--- @return LevenshteinChange<T>[]
|
|
||||||
function M.levenshtein(x, y, cost)
|
|
||||||
cost = cost or {}
|
|
||||||
local cost_of_delete_f = cost.of_delete or function() return 1 end
|
|
||||||
local cost_of_add_f = cost.of_add or function() return 1 end
|
|
||||||
local cost_of_change_f = cost.of_change or function() return 1 end
|
|
||||||
|
|
||||||
local m, n = #x, #y
|
|
||||||
-- Initialize the distance matrix
|
|
||||||
local dp = {}
|
|
||||||
for i = 0, m do
|
|
||||||
dp[i] = {}
|
|
||||||
for j = 0, n do
|
|
||||||
dp[i][j] = 0
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Fill the base cases
|
|
||||||
for i = 0, m do
|
|
||||||
dp[i][0] = i
|
|
||||||
end
|
|
||||||
for j = 0, n do
|
|
||||||
dp[0][j] = j
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Compute the Levenshtein distance dynamically
|
|
||||||
for i = 1, m do
|
|
||||||
for j = 1, n do
|
|
||||||
if x[i] == y[j] then
|
|
||||||
dp[i][j] = dp[i - 1][j - 1] -- no cost if items are the same
|
|
||||||
else
|
|
||||||
local costDelete = dp[i - 1][j] + cost_of_delete_f(x[i])
|
|
||||||
local costAdd = dp[i][j - 1] + cost_of_add_f(y[j])
|
|
||||||
local costChange = dp[i - 1][j - 1] + cost_of_change_f(x[i], y[j])
|
|
||||||
dp[i][j] = math.min(costDelete, costAdd, costChange)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
-- Backtrack to find the changes
|
|
||||||
local i = m
|
|
||||||
local j = n
|
|
||||||
--- @type LevenshteinChange[]
|
|
||||||
local changes = {}
|
|
||||||
|
|
||||||
while i > 0 or j > 0 do
|
|
||||||
local default_cost = dp[i][j]
|
|
||||||
local cost_of_change = (i > 0 and j > 0) and dp[i - 1][j - 1] or default_cost
|
|
||||||
local cost_of_add = j > 0 and dp[i][j - 1] or default_cost
|
|
||||||
local cost_of_delete = i > 0 and dp[i - 1][j] or default_cost
|
|
||||||
|
|
||||||
--- @param u number
|
|
||||||
--- @param v number
|
|
||||||
--- @param w number
|
|
||||||
local function is_first_min(u, v, w) return u <= v and u <= w end
|
|
||||||
|
|
||||||
if is_first_min(cost_of_change, cost_of_add, cost_of_delete) then
|
|
||||||
-- potential change
|
|
||||||
if x[i] ~= y[j] then
|
|
||||||
--- @type LevenshteinChange
|
|
||||||
local change = { kind = 'change', from = x[i], index = i, to = y[j] }
|
|
||||||
table.insert(changes, change)
|
|
||||||
end
|
|
||||||
i = i - 1
|
|
||||||
j = j - 1
|
|
||||||
elseif is_first_min(cost_of_add, cost_of_change, cost_of_delete) then
|
|
||||||
-- addition
|
|
||||||
--- @type LevenshteinChange
|
|
||||||
local change = { kind = 'add', item = y[j], index = i + 1 }
|
|
||||||
table.insert(changes, change)
|
|
||||||
j = j - 1
|
|
||||||
elseif is_first_min(cost_of_delete, cost_of_change, cost_of_add) then
|
|
||||||
-- deletion
|
|
||||||
--- @type LevenshteinChange
|
|
||||||
local change = { kind = 'delete', item = x[i], index = i }
|
|
||||||
table.insert(changes, change)
|
|
||||||
i = i - 1
|
|
||||||
else
|
|
||||||
error 'unreachable'
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
return changes
|
|
||||||
end
|
|
||||||
|
|
||||||
return M
|
return M
|
||||||
|
@ -1,29 +0,0 @@
|
|||||||
local M = {}
|
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
|
||||||
--- eat_while
|
|
||||||
--------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
--- @param s string
|
|
||||||
--- @param pos number
|
|
||||||
--- @param predicate fun(c: string, i: number, s: string): boolean
|
|
||||||
function M.eat_while(s, pos, predicate)
|
|
||||||
local eaten = ''
|
|
||||||
local curr = pos
|
|
||||||
local watchdog = 0
|
|
||||||
while curr <= #s do
|
|
||||||
watchdog = watchdog + 1
|
|
||||||
if watchdog > #s then error 'infinite loop' end
|
|
||||||
|
|
||||||
local c = s:sub(curr, curr)
|
|
||||||
if not predicate(c, curr, s) then break end
|
|
||||||
eaten = eaten .. c
|
|
||||||
curr = curr + 1
|
|
||||||
end
|
|
||||||
return eaten, curr
|
|
||||||
end
|
|
||||||
|
|
||||||
--- @param c string
|
|
||||||
function M.is_whitespace(c) return c == ' ' or c == '\t' or c == '\n' end
|
|
||||||
|
|
||||||
return M
|
|
@ -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), nil, 'v')
|
local range = Range.new(Pos.new(nil, 0, 0), Pos.new(nil, 0, 0), '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,90 +471,4 @@ 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)
|
||||||
|
@ -1,59 +0,0 @@
|
|||||||
local Renderer = require 'u.renderer'
|
|
||||||
|
|
||||||
--- @param markup string
|
|
||||||
local function parse(markup)
|
|
||||||
-- call private method:
|
|
||||||
return (Renderer --[[@as any]])._parse_markup(markup)
|
|
||||||
end
|
|
||||||
|
|
||||||
describe('Renderer', function()
|
|
||||||
it('_parse_markup: empty string', function()
|
|
||||||
local nodes = parse [[]]
|
|
||||||
assert.are.same({}, nodes)
|
|
||||||
end)
|
|
||||||
|
|
||||||
it('_parse_markup: only string', function()
|
|
||||||
local nodes = parse [[The quick brown fox jumps over the lazy dog.]]
|
|
||||||
assert.are.same({
|
|
||||||
{ kind = 'text', value = 'The quick brown fox jumps over the lazy dog.' },
|
|
||||||
}, nodes)
|
|
||||||
end)
|
|
||||||
|
|
||||||
it('_parse_markup: <', function()
|
|
||||||
local nodes = parse [[<t value="bleh" />]]
|
|
||||||
assert.are.same({
|
|
||||||
{ kind = 'text', value = '<t value="bleh" />' },
|
|
||||||
}, nodes)
|
|
||||||
end)
|
|
||||||
|
|
||||||
it('_parse_markup: empty tag', function()
|
|
||||||
local nodes = parse [[</>]]
|
|
||||||
assert.are.same({ { kind = 'tag', name = '', attributes = {} } }, nodes)
|
|
||||||
end)
|
|
||||||
|
|
||||||
it('_parse_markup: tag', function()
|
|
||||||
local nodes = parse [[<t value="Hello" />]]
|
|
||||||
assert.are.same({
|
|
||||||
{
|
|
||||||
kind = 'tag',
|
|
||||||
name = 't',
|
|
||||||
attributes = {
|
|
||||||
value = 'Hello',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, nodes)
|
|
||||||
end)
|
|
||||||
|
|
||||||
it('_parse_markup: attributes with quotes', function()
|
|
||||||
local nodes = parse [[<t value="Hello \"there\"" />]]
|
|
||||||
assert.are.same({
|
|
||||||
{
|
|
||||||
kind = 'tag',
|
|
||||||
name = 't',
|
|
||||||
attributes = {
|
|
||||||
value = 'Hello "there"',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, nodes)
|
|
||||||
end)
|
|
||||||
end)
|
|
@ -1,70 +0,0 @@
|
|||||||
local utils = require 'u.utils'
|
|
||||||
|
|
||||||
--- @param s string
|
|
||||||
local function split(s) return vim.split(s, '') end
|
|
||||||
|
|
||||||
--- @param original string
|
|
||||||
--- @param changes LevenshteinChange[]
|
|
||||||
local function morph(original, changes)
|
|
||||||
local t = split(original)
|
|
||||||
for _, change in ipairs(changes) do
|
|
||||||
if change.kind == 'add' then
|
|
||||||
table.insert(t, change.index, change.item)
|
|
||||||
elseif change.kind == 'delete' then
|
|
||||||
table.remove(t, change.index)
|
|
||||||
elseif change.kind == 'change' then
|
|
||||||
t[change.index] = change.to
|
|
||||||
end
|
|
||||||
end
|
|
||||||
return vim.iter(t):join ''
|
|
||||||
end
|
|
||||||
|
|
||||||
describe('utils', function()
|
|
||||||
it('levenshtein', function()
|
|
||||||
local original = 'abc'
|
|
||||||
local result = 'absece'
|
|
||||||
local changes = utils.levenshtein(split(original), split(result))
|
|
||||||
assert.are.same(changes, {
|
|
||||||
{
|
|
||||||
item = 'e',
|
|
||||||
kind = 'add',
|
|
||||||
index = 4,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
item = 'e',
|
|
||||||
kind = 'add',
|
|
||||||
index = 3,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
item = 's',
|
|
||||||
kind = 'add',
|
|
||||||
index = 3,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
assert.are.same(morph(original, changes), result)
|
|
||||||
|
|
||||||
original = 'jonathan'
|
|
||||||
result = 'ajoanthan'
|
|
||||||
changes = utils.levenshtein(split(original), split(result))
|
|
||||||
assert.are.same(changes, {
|
|
||||||
{
|
|
||||||
from = 'a',
|
|
||||||
index = 4,
|
|
||||||
kind = 'change',
|
|
||||||
to = 'n',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
from = 'n',
|
|
||||||
index = 3,
|
|
||||||
kind = 'change',
|
|
||||||
to = 'a',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
index = 1,
|
|
||||||
item = 'a',
|
|
||||||
kind = 'add',
|
|
||||||
},
|
|
||||||
})
|
|
||||||
assert.are.same(morph(original, changes), result)
|
|
||||||
end)
|
|
||||||
end)
|
|
@ -1,24 +0,0 @@
|
|||||||
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"
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
x
Reference in New Issue
Block a user