2023-03-14 18:17:07 +08:00
|
|
|
local api, fn = vim.api, vim.fn
|
2023-03-14 13:18:53 +08:00
|
|
|
|
2023-03-14 18:17:07 +08:00
|
|
|
---@class TransBuffer
|
|
|
|
---@field bufnr integer buffer handle
|
2023-04-24 20:58:46 +08:00
|
|
|
---@field [integer] string|TransNode|TransNode[] buffer[line] content
|
2023-02-04 15:03:02 +08:00
|
|
|
local buffer = {}
|
|
|
|
|
2023-03-15 20:57:28 +08:00
|
|
|
-- INFO : corountine can't invoke C function
|
2023-02-04 15:03:02 +08:00
|
|
|
---Clear all content in buffer
|
2023-03-15 22:12:44 +08:00
|
|
|
function buffer:wipe()
|
2023-03-23 09:52:44 +08:00
|
|
|
api.nvim_buf_set_lines(self.bufnr, 0, -1, false, {})
|
2023-03-15 22:12:44 +08:00
|
|
|
end
|
2023-02-04 15:03:02 +08:00
|
|
|
|
2023-03-14 18:17:07 +08:00
|
|
|
---Delete buffer [_start, _end] line content [one index]
|
2023-03-15 00:18:31 +08:00
|
|
|
---@param _start? integer start line index
|
|
|
|
---@param _end? integer end line index
|
2023-03-15 22:12:44 +08:00
|
|
|
function buffer:deleteline(_start, _end)
|
2023-03-23 09:52:44 +08:00
|
|
|
---@diagnostic disable-next-line: cast-local-type
|
|
|
|
_start = _start and _start - 1 or self:line_count() - 1
|
2023-03-29 11:59:58 +08:00
|
|
|
_end = _end or _start + 1 -- because of end exclusive
|
2023-03-23 09:52:44 +08:00
|
|
|
api.nvim_buf_set_lines(self.bufnr, _start, _end, false, {})
|
2023-02-04 15:03:02 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
---Set buffer option
|
|
|
|
---@param name string option name
|
|
|
|
---@param value any option value
|
|
|
|
function buffer:set(name, value)
|
|
|
|
api.nvim_buf_set_option(self.bufnr, name, value)
|
|
|
|
end
|
|
|
|
|
2023-03-14 18:17:07 +08:00
|
|
|
---Get buffer option
|
2023-02-04 15:03:02 +08:00
|
|
|
---@param name string option name
|
|
|
|
---@return any
|
|
|
|
function buffer:option(name)
|
|
|
|
return api.nvim_buf_get_option(self.bufnr, name)
|
|
|
|
end
|
|
|
|
|
2023-03-13 11:51:46 +08:00
|
|
|
---Destory buffer
|
2023-03-14 13:18:53 +08:00
|
|
|
function buffer:destroy()
|
2023-04-07 19:05:24 +08:00
|
|
|
pcall(api.nvim_buf_delete, self.bufnr, { force = true })
|
2023-02-04 15:03:02 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
---Set buffer load keymap
|
|
|
|
---@param key string
|
|
|
|
---@param operation function | string
|
|
|
|
function buffer:map(key, operation)
|
|
|
|
vim.keymap.set('n', key, operation, {
|
|
|
|
buffer = self.bufnr,
|
|
|
|
silent = true,
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2023-03-14 18:17:07 +08:00
|
|
|
---Execute keycode in normal this buffer[no recursive]
|
2023-02-04 15:03:02 +08:00
|
|
|
---@param key string key code
|
|
|
|
function buffer:normal(key)
|
|
|
|
api.nvim_buf_call(self.bufnr, function()
|
|
|
|
vim.cmd([[normal! ]] .. key)
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
---@nodiscard
|
2023-03-14 18:17:07 +08:00
|
|
|
---@return boolean
|
2023-02-04 15:03:02 +08:00
|
|
|
function buffer:is_valid()
|
|
|
|
return api.nvim_buf_is_valid(self.bufnr)
|
|
|
|
end
|
|
|
|
|
|
|
|
---Get buffer [i, j] line content
|
|
|
|
---@param i integer? start line index
|
|
|
|
---@param j integer? end line index
|
|
|
|
---@return string[]
|
|
|
|
function buffer:lines(i, j)
|
|
|
|
i = i and i - 1 or 0
|
2023-03-29 11:59:58 +08:00
|
|
|
j = j or -1 -- because of end exclusive
|
2023-02-04 15:03:02 +08:00
|
|
|
return api.nvim_buf_get_lines(self.bufnr, i, j, false)
|
|
|
|
end
|
|
|
|
|
2023-03-13 11:51:46 +08:00
|
|
|
---Add highlight to buffer
|
|
|
|
---@param linenr number line number should be set[one index]
|
|
|
|
---@param hl_group string highlight group
|
2023-03-14 22:30:25 +08:00
|
|
|
---@param col_start? number column start [zero index]
|
|
|
|
---@param col_end? number column end
|
2023-03-13 11:51:46 +08:00
|
|
|
---@param ns number? highlight namespace
|
2023-03-13 19:50:28 +08:00
|
|
|
function buffer:add_highlight(linenr, hl_group, col_start, col_end, ns)
|
2023-03-15 00:18:31 +08:00
|
|
|
-- vim.print(linenr, hl_group, col_start, col_end, ns)
|
2023-03-23 09:52:44 +08:00
|
|
|
linenr = linenr - 1
|
|
|
|
col_start = col_start or 0
|
|
|
|
api.nvim_buf_add_highlight(self.bufnr, ns or -1, hl_group, linenr, col_start, col_end or -1)
|
2023-03-13 11:51:46 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
---Get buffer line count
|
|
|
|
---@return integer
|
|
|
|
function buffer:line_count()
|
2023-03-29 11:59:58 +08:00
|
|
|
local line_count = api.nvim_buf_line_count(self.bufnr)
|
2023-04-01 09:54:35 +08:00
|
|
|
return line_count == 1 and self[1] == '' and 0 or line_count
|
2023-03-13 11:51:46 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
---Set line content
|
2023-02-04 15:03:02 +08:00
|
|
|
---@param nodes string|table|table[] string -> as line content | table -> as a node | table[] -> as node[]
|
2023-03-13 22:39:40 +08:00
|
|
|
---@param one_index number? line number should be set[one index] or let it be nil to append
|
|
|
|
function buffer:setline(nodes, one_index)
|
2023-03-27 00:36:28 +08:00
|
|
|
local append_line_index = self:line_count() + 1
|
|
|
|
one_index = one_index or append_line_index
|
|
|
|
if one_index > append_line_index then
|
|
|
|
for i = append_line_index, one_index - 1 do
|
|
|
|
self:setline('', i)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-03-15 00:18:31 +08:00
|
|
|
|
2023-02-04 15:03:02 +08:00
|
|
|
if type(nodes) == 'string' then
|
2023-03-15 00:18:31 +08:00
|
|
|
fn.setbufline(self.bufnr, one_index, nodes)
|
2023-03-15 22:12:44 +08:00
|
|
|
return
|
2023-02-04 15:03:02 +08:00
|
|
|
end
|
2023-03-13 21:41:17 +08:00
|
|
|
|
2023-03-15 22:12:44 +08:00
|
|
|
if type(nodes[1]) == 'string' then
|
|
|
|
---@diagnostic disable-next-line: assign-type-mismatch, param-type-mismatch
|
|
|
|
fn.setbufline(self.bufnr, one_index, nodes[1])
|
|
|
|
nodes:render(self, one_index, 0)
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local strs = {}
|
|
|
|
local num = #nodes
|
|
|
|
for i = 1, num do
|
|
|
|
strs[i] = nodes[i][1]
|
|
|
|
end
|
|
|
|
|
|
|
|
fn.setbufline(self.bufnr, one_index, table.concat(strs))
|
|
|
|
local col = 0
|
|
|
|
for i = 1, num do
|
|
|
|
local node = nodes[i]
|
|
|
|
node:render(self, one_index, col)
|
|
|
|
col = col + #node[1]
|
|
|
|
end
|
2023-02-04 15:03:02 +08:00
|
|
|
end
|
|
|
|
|
2023-02-03 11:57:42 +08:00
|
|
|
buffer.__index = function(self, key)
|
|
|
|
local res = buffer[key]
|
|
|
|
if res then
|
|
|
|
return res
|
|
|
|
elseif type(key) == 'number' then
|
2023-03-17 22:17:17 +08:00
|
|
|
-- return fn.getbufoneline(self.bufnr, key) -- Vimscript Function Or Lua API ?? -- INFO :only work on neovim-nightly
|
2023-03-16 21:15:00 +08:00
|
|
|
return api.nvim_buf_get_lines(self.bufnr, key - 1, key, true)[1]
|
2023-02-03 11:57:42 +08:00
|
|
|
else
|
2023-03-14 13:18:53 +08:00
|
|
|
error('invalid key: ' .. key)
|
2023-02-03 11:57:42 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-04-01 09:54:35 +08:00
|
|
|
buffer.__newindex = function(self, key, values)
|
2023-03-13 11:51:46 +08:00
|
|
|
if type(key) == 'number' then
|
2023-04-01 09:54:35 +08:00
|
|
|
self:setline(values, key)
|
2023-03-13 11:51:46 +08:00
|
|
|
else
|
2023-04-01 09:54:35 +08:00
|
|
|
rawset(self, key, values)
|
2023-03-13 11:51:46 +08:00
|
|
|
end
|
2023-02-03 11:57:42 +08:00
|
|
|
end
|
|
|
|
|
2023-03-15 14:30:01 +08:00
|
|
|
---Init buffer with bufnr
|
|
|
|
---@param bufnr? integer buffer handle
|
|
|
|
function buffer:init(bufnr)
|
|
|
|
self.bufnr = bufnr or api.nvim_create_buf(false, false)
|
|
|
|
self:set('filetype', 'Trans')
|
|
|
|
self:set('buftype', 'nofile')
|
|
|
|
end
|
2023-03-14 18:17:07 +08:00
|
|
|
|
|
|
|
---@nodiscard
|
|
|
|
---TransBuffer constructor
|
2023-03-24 11:09:04 +08:00
|
|
|
---@param bufnr? integer buffer handle
|
2023-03-14 18:17:07 +08:00
|
|
|
---@return TransBuffer
|
2023-03-24 11:09:04 +08:00
|
|
|
function buffer.new(bufnr)
|
2023-03-15 14:30:01 +08:00
|
|
|
local new_buf = setmetatable({}, buffer)
|
2023-03-24 11:09:04 +08:00
|
|
|
new_buf:init(bufnr)
|
2023-03-12 23:17:36 +08:00
|
|
|
return new_buf
|
2023-02-03 11:57:42 +08:00
|
|
|
end
|
2023-03-12 21:33:00 +08:00
|
|
|
|
2023-03-23 20:24:17 +08:00
|
|
|
--- HACK :available options:
|
|
|
|
--- - id
|
|
|
|
--- - end_row
|
|
|
|
--- - end_col
|
|
|
|
--- - hl_eol
|
|
|
|
--- - virt_text
|
|
|
|
--- - virt_text_pos
|
|
|
|
--- - virt_text_win_col
|
|
|
|
--- - hl_mode
|
|
|
|
--- - virt_lines
|
|
|
|
--- - virt_lines_above
|
|
|
|
--- - virt_lines_leftcol
|
|
|
|
--- - ephemeral
|
|
|
|
--- - right_gravity
|
|
|
|
--- - end_right_gravity
|
|
|
|
--- - priority
|
|
|
|
--- - strict
|
|
|
|
--- - sign_text
|
|
|
|
--- - sign_hl_group
|
|
|
|
--- - number_hl_group
|
|
|
|
--- - line_hl_group
|
|
|
|
--- - cursorline_hl_group
|
|
|
|
--- - conceal
|
|
|
|
--- - ui_watched
|
|
|
|
|
|
|
|
---Add Extmark to buffer
|
|
|
|
---@param ns number highlight namespace
|
|
|
|
---@param linenr number line number should be set[one index]
|
|
|
|
---@param col_start number column start
|
|
|
|
function buffer:set_extmark(ns, linenr, col_start, opts)
|
|
|
|
linenr = linenr and linenr - 1 or -1
|
|
|
|
return api.nvim_buf_set_extmark(self.bufnr, ns, linenr, col_start, opts)
|
|
|
|
end
|
|
|
|
|
2023-03-14 18:17:07 +08:00
|
|
|
---@class Trans
|
|
|
|
---@field buffer TransBuffer
|
2023-03-12 21:33:00 +08:00
|
|
|
return buffer
|