refactor: 重写了类的逻辑, 引入了部分bug待解决

This commit is contained in:
JuanZoran 2023-02-03 11:57:42 +08:00
parent 6737090710
commit 8b8879b2cd
9 changed files with 695 additions and 653 deletions

144
lua/Trans/buffer.lua Normal file
View File

@ -0,0 +1,144 @@
local api = vim.api
local fn = vim.fn
local buffer = {
addline = function(self, nodes, index)
local size = self.size
if index then
assert(index <= size + 1)
index = index
else
index = size + 1
end
local append = index == size + 1
local line = index - 1
if type(nodes) == 'string' then
self[index] = nodes
else
local bufnr = self.bufnr
local col = 0
if type(nodes[1]) == 'string' then
self[index] = nodes[1]
nodes:load(bufnr, line, col)
else
local strs = {}
local num = #nodes
for i = 1, num do
strs[i] = nodes[i][1]
end
self[index] = table.concat(strs)
for i = 1, num do
local node = nodes[i]
node:load(bufnr, line, col)
col = col + #node[1]
end
end
end
if append then
self.size = self.size + 1
end
end,
del = function(self, _start, _end)
if not _start then
fn.deletebufline(self.bufnr, '$')
else
_end = _end or _start
fn.deletebufline(self.bufnr, _start, _end)
end
self.size = api.nvim_buf_line_count(self.bufnr)
end,
set = function(self, name, option)
api.nvim_buf_set_option(self.bufnr, name, option)
end,
option = function(self, name)
return api.nvim_buf_get_option(self.bufnr, name)
end,
is_valid = function(self)
return api.nvim_buf_is_valid(self.bufnr)
end,
delete = function(self)
api.nvim_buf_delete(self.bufnr, { force = true })
end,
len = function(self)
return api.nvim_buf_line_count(self.bufnr) - 1
end,
map = function(self, key, operation)
vim.keymap.set('n', key, operation, {
buffer = self.bufnr,
silent = true,
})
end,
normal = function(self, key)
api.nvim_buf_call(self.bufnr, function()
vim.cmd([[normal! ]] .. key)
end)
end,
lines = function(self, i, j)
i = i and i - 1 or 0
j = j and j - 1 or -1
return api.nvim_buf_get_lines(self.bufnr, i, j, false)
end,
height = function(self, opts)
local width = opts.width
local wrap = opts.wrap or false
local lines = self:lines()
local size = #lines
if wrap then
local height = 0
for i = 1, size do
height = height + math.max(1, (math.ceil(lines[i]:width() / width)))
end
return height
else
return size
end
end,
init = function(self)
self.bufnr = api.nvim_create_buf(false, false)
self:set('filetype', 'Trans')
self:set('buftype', 'nofile')
self.size = 0
end,
}
buffer.__index = function(self, key)
local res = buffer[key]
if res then
return res
elseif type(key) == 'number' then
return fn.getbufoneline(self.bufnr, key)
else
error('invalid key' .. key)
end
end
buffer.__newindex = function(self, key, text)
assert(key <= self.size + 1)
fn.setbufline(self.bufnr, key, text)
end
return function()
return setmetatable({
bufnr = -1,
size = 0,
}, buffer)
end

View File

@ -1,138 +0,0 @@
local api = vim.api
local content = {
newline = function(self, value)
local index = self.size + 1
self.size = index
self.lines[index] = value
end,
newhl = function(self, opt)
local index = self.hl_size + 1
self.hl_size = index
self.highlights[index] = opt
end,
wipe = function(self)
local clear = require('table.clear')
clear(self.lines)
clear(self.highlights)
self.size = 0
self.hl_size = 0
end,
---将内容连接上对应的窗口
---@param self table content对象
---@param offset integer 起始行
attach = function(self, offset)
if self.size == 0 then
return
end
offset = offset or 0
local win = self.window
win:bufset('modifiable', true)
--- NOTE : 使用-1 则需要按顺序设置
api.nvim_buf_set_lines(win.bufnr, offset, -1, true, self.lines)
local hl
local highlights = self.highlights
local method = api.nvim_buf_add_highlight
for i = 1, self.hl_size do
hl = highlights[i]
method(win.bufnr, win.hl, hl.name, offset + hl.line, hl._start, hl._end)
end
win:bufset('modifiable', false)
end,
actual_height = function(self, wrap)
wrap = wrap or self.window:option('wrap')
if wrap then
local height = 0
local width = self.window.width
local lines = self.lines
for i = 1, self.size do
height = height + math.max(1, (math.ceil(lines[i]:width() / width)))
end
return height
else
return self.size
end
end,
format = function(self, opt)
local win_width = opt.width or self.window.width
local nodes = opt.nodes
local size = #nodes
assert(size > 1, 'check items size')
local tot_width = 0
local strs = {}
local str
for i = 1, size do
str = nodes[i].text
strs[i] = str
tot_width = tot_width + str:width()
end
local space = math.floor(((win_width - tot_width) / (size - 1)))
if opt.strict and space < 0 then
return false
end
local interval = (' '):rep(space)
return {
text = table.concat(strs, interval),
load_hl = function(_, content, line, col)
for _, item in ipairs(nodes) do
item:load_hl(content, line, col)
col = col + #item.text + space
end
end
}
end,
center = function(self, item)
local text = item.text
local space = bit.rshift(self.window.width - text:width(), 1)
item.text = (' '):rep(space) .. text
local load_hl = item.load_hl
item.load_hl = function(this, content, line, col)
load_hl(this, content, line, col + space)
end
return item
end,
addline = function(self, ...)
local strs = {}
local col = 0
local str
local line = self.size -- line is zero index
for i, node in ipairs { ... } do
str = node.text
strs[i] = str
node:load_hl(self, line, col)
col = col + #str
end
self:newline(table.concat(strs))
end
}
content.__index = content
---content的构造函数
---@param window table 链接的窗口
---@return table 构造好的content
return function(window)
vim.validate {
window = { window, 't' },
}
return setmetatable({
window = window,
size = 0,
hl_size = 0,
lines = {},
highlights = {},
}, content)
end

View File

@ -255,4 +255,6 @@ M.translate = function(mode, view)
end end
end end
M.ns = api.nvim_create_namespace('Trans')
return M return M

View File

@ -1,39 +1,73 @@
-- NOTE : 设置content的node local api = vim.api
local item_load = function(self, content, line, col) local add_hl = api.nvim_buf_add_highlight
if self.hl then local ns = require('Trans').ns
content:newhl {
name = self.hl, local item_meta = {
line = line, load = function(self, bufnr, line, col)
_start = col, if self[2] then
_end = col + #self.text, add_hl(bufnr, ns, self[2], line, col, col + #self[1])
end
end,
} }
local text_meta = {
load = function(self, bufnr, line, col)
local items = self.items
local step = self.step or ''
local len = #step
for i = 1, self.size do
local item = items[i]
item:load(bufnr, line, col)
col = col + #item[1] + len
end
end
}
item_meta.__index = item_meta
text_meta.__index = function(self, key)
local res = text_meta[key]
if res then
return res
elseif key == 1 then
return table.concat(self.strs, self.step)
end end
end end
return { return {
item = function(text, hl) item = function(text, highlight)
return { return setmetatable({
text = text, [1] = text,
hl = hl, [2] = highlight,
load_hl = item_load, }, item_meta)
}
end, end,
text = function(...) text = function(items)
local items = { ... }
local strs = {} local strs = {}
for i, item in ipairs(items) do local size = #items
strs[i] = item.text assert(size > 1)
for i = 1, size do
strs[i] = items[i][1]
end end
return { return setmetatable({
text = table.concat(strs), strs = strs,
load_hl = function(_, content, line, col) size = size,
for _, item in ipairs(items) do items = items,
item:load_hl(content, line, col) }, text_meta)
col = col + #item.text end,
format = function(opts)
local text = opts.text
local width = opts.width
local spin = opts.spin or ' '
local size = text.size
local text_width = text[1]:width()
local space = math.max(math.floor((width - text_width) / (size - 1)), 0)
if space > 0 then
text.step = spin:rep(space)
end end
end return text
}
end, end,
} }

View File

@ -41,16 +41,16 @@ return function(word)
callback = function(str) callback = function(str)
local ok, res = pcall(vim.json.decode, str) local ok, res = pcall(vim.json.decode, str)
if ok and res and res.trans_result then if ok and res and res.trans_result then
result.value = { result[1] = {
word = word, title = { word = word },
[isEn and 'translation' or 'definition'] = res.trans_result[1].dst, [isEn and 'translation' or 'definition'] = res.trans_result[1].dst,
} }
if result.callback then if result.callback then
result.callback(result.value) result.callback(result[1])
end end
else else
result.value = false result[1] = false
end end
end, end,
}) })

View File

@ -16,8 +16,9 @@ vim.api.nvim_create_autocmd('VimLeavePre', {
end end
}) })
return function(word) return function(word)
local res = dict:select('stardict', { local res = (dict:select('stardict', {
where = { where = {
word = word, word = word,
}, },
@ -33,6 +34,20 @@ return function(word)
'exchange', 'exchange',
}, },
limit = 1, limit = 1,
}) }))[1]
return res[1]
if res then
res.title = {
word = res.word,
oxford = res.oxford,
collins = res.collins,
phonetic = res.phonetic,
}
res.word = nil
res.oxford = nil
res.collins = nil
res.phonetic = nil
end
return res
end end

View File

@ -1,33 +1,35 @@
local display = function(self) return function(opts)
local callback = self.callback or function() local callback = opts.callback or function()
end end
opts.run = true
local target = self.times local target = opts.times
if self.sync then if opts.sync then
if target then if target then
for i = 1, target do for i = 1, target do
if self.run then if opts.run then
self:frame(i) opts:frame(i)
end end
end end
else else
while self.run do while opts.run do
self:frame() opts:frame()
end end
end end
callback() callback()
else else
local frame local frame
if target then if target then
local times = 0 local times = 0
frame = function() frame = function()
if self.run and times < target then if opts.run and times < target then
times = times + 1 times = times + 1
self:frame(times) opts:frame(times)
vim.defer_fn(frame, self.interval) vim.defer_fn(frame, opts.interval)
else else
callback() callback()
end end
@ -35,9 +37,9 @@ local display = function(self)
else else
frame = function() frame = function()
if self.run then if opts.run then
self:frame() opts:frame()
vim.defer_fn(frame, self.interval) vim.defer_fn(frame, opts.interval)
else else
callback() callback()
end end
@ -45,11 +47,5 @@ local display = function(self)
end end
frame() frame()
end end
end
return function(opts)
opts.run = true
opts.display = display
return opts return opts
end end

View File

@ -1,55 +1,56 @@
local api = vim.api local api = vim.api
local conf = require('Trans').conf local conf = require('Trans').conf
local new_window = require('Trans.window') local hover = conf.hover
local buffer = require('Trans.buffer')()
local error_msg = conf.icon.notfound .. ' 没有找到相关的翻译'
local m_window local node = require('Trans.node')
local m_result
local m_content
-- content utility
local node = require("Trans.node")
local t = node.text
local it = node.item local it = node.item
local t = node.text
local f = node.format
local m_indent = ' ' local function handle_result(result)
local icon = conf.icon
local notfound = icon.notfound
local indent = ' '
local title = function(str) local addtitle = function(title)
m_content:addline( buffer:addline {
t(it('', 'TransTitleRound'), it(str, 'TransTitle'), it('', 'TransTitleRound')) it('', 'TransTitleRound'),
) it(title, 'TransTitle'),
end it('', 'TransTitleRound'),
}
local exist = function(str)
return str and str ~= ''
end end
local process = { local process = {
title = function() title = function(title)
local icon = conf.icon local word = title.word
local line local oxford = title.oxford
if m_result.word:find(' ', 1, true) then local collins = title.collins
line = it(m_result.word, 'TransWord') local phonetic = title.phonetic
if not phonetic and not collins and not oxford then
buffer:addline(it(result.word, 'TransWord'))
else else
line = m_content:format { buffer:addline(f {
nodes = { width = hover.width,
it(m_result.word, 'TransWord'), text = t {
t( it(word, 'TransWord'),
t {
it('['), it('['),
it(exist(m_result.phonetic) and m_result.phonetic or icon.notfound, 'TransPhonetic'), it((phonetic and phonetic ~= '') and phonetic or notfound, 'TransPhonetic'),
it(']') it(']')
),
it(m_result.collins and icon.star:rep(m_result.collins) or icon.notfound, 'TransCollins'),
it(m_result.oxford == 1 and icon.yes or icon.no)
}, },
} it(collins and icon.star:rep(collins) or notfound, 'TransCollins'),
it(oxford == 1 and icon.yes or icon.no)
},
})
end end
m_content:addline(line)
end, end,
tag = function() tag = function(tag)
if exist(m_result.tag) then addtitle('标签')
title('标签')
local tag_map = { local tag_map = {
zk = '中考', zk = '中考',
gk = '高考', gk = '高考',
@ -64,17 +65,16 @@ local process = {
local tags = {} local tags = {}
local size = 0 local size = 0
local interval = ' ' local interval = ' '
for tag in vim.gsplit(m_result.tag, ' ', true) do for _tag in vim.gsplit(tag, ' ', true) do
size = size + 1 size = size + 1
tags[size] = tag_map[tag] tags[size] = tag_map[_tag]
end end
for i = 1, size, 3 do for i = 1, size, 3 do
m_content:addline( buffer:addline(
it( it(
m_indent .. indent .. tags[i] ..
tags[i] ..
(tags[i + 1] and interval .. tags[i + 1] .. (tags[i + 1] and interval .. tags[i + 1] ..
(tags[i + 2] and interval .. tags[i + 2] or '') or ''), (tags[i + 2] and interval .. tags[i + 2] or '') or ''),
'TransTag' 'TransTag'
@ -82,13 +82,11 @@ local process = {
) )
end end
m_content:newline('') buffer:addline('')
end
end, end,
pos = function() pos = function(pos)
if exist(m_result.pos) then addtitle('词性')
title('词性')
local pos_map = { local pos_map = {
a = '代词pron ', a = '代词pron ',
c = '连接词conj ', c = '连接词conj ',
@ -105,20 +103,18 @@ local process = {
d = '限定词determiner ', d = '限定词determiner ',
} }
local f = '%s %2s%%' local s = '%s %2s%%'
for pos in vim.gsplit(m_result.pos, '/', true) do for _pos in vim.gsplit(pos, '/', true) do
m_content:addline( buffer:addline(
it(m_indent .. f:format(pos_map[pos:sub(1, 1)], pos:sub(3)), 'TransPos') it(indent .. s:format(pos_map[_pos:sub(1, 1)], _pos:sub(3)), 'TransPos')
) )
end end
m_content:newline('') buffer:addline('')
end
end, end,
exchange = function() exchange = function(exchange)
if exist(m_result.exchange) then addtitle('词形变化')
title('词形变化')
local exchange_map = { local exchange_map = {
['p'] = '过去式 ', ['p'] = '过去式 ',
['d'] = '过去分词 ', ['d'] = '过去分词 ',
@ -132,264 +128,306 @@ local process = {
['f'] = '第三人称单数', ['f'] = '第三人称单数',
} }
local interval = ' ' local interval = ' '
for exc in vim.gsplit(m_result.exchange, '/', true) do for exc in vim.gsplit(exchange, '/', true) do
m_content:addline( buffer:addline(
it(m_indent .. exchange_map[exc:sub(1, 1)] .. interval .. exc:sub(3), 'TransExchange') it(indent .. exchange_map[exc:sub(1, 1)] .. interval .. exc:sub(3), 'TransExchange')
) )
end end
m_content:newline('') buffer:addline('')
end
end, end,
translation = function() translation = function(translation)
if exist(m_result.translation) then if hover.auto_play then
title('中文翻译') result.title.word:play()
end
for trs in vim.gsplit(m_result.translation, '\n', true) do addtitle('中文翻译')
m_content:addline(
it(m_indent .. trs, 'TransTranslation') for trs in vim.gsplit(translation, '\n', true) do
buffer:addline(
it(indent .. trs, 'TransTranslation')
) )
end end
end
m_content:newline('') buffer:addline('')
end, end,
definition = function() definition = function(definition)
if exist(m_result.definition) then addtitle('英文注释')
title('英文注释')
for def in vim.gsplit(m_result.definition, '\n', true) do for def in vim.gsplit(definition, '\n', true) do
def = def:gsub('^%s+', '', 1) -- TODO :判断是否需要分割空格 def = def:gsub('^%s+', '', 1) -- TODO :判断是否需要分割空格
m_content:addline( buffer:addline(
it(m_indent .. def, 'TransDefinition') it(indent .. def, 'TransDefinition')
) )
end end
m_content:newline('') buffer:addline('')
end
end, end,
} }
buffer:set('modifiable', true)
for _, field in ipairs(conf.order) do
local value = result[field]
if value and value ~= '' then
process[field](value)
end
end
buffer:set('modifiable', false)
end
local try_del_keymap = function() local function open_window(opts)
for _, key in pairs(conf.hover.keymap) do opts = opts or {}
pcall(vim.keymap.del, 'n', key, { buffer = true }) local col = opts.col or 1
local row = opts.row or 1
local width = opts.width or hover.width
local height = opts.height or hover.height
local relative = opts.relative or 'cursor'
local task = opts.task
local win = require('Trans.window') {
col = col,
row = row,
task = task,
buf = buffer,
relative = relative,
width = width,
height = height,
title = hover.title,
border = hover.border,
animation = hover.animation,
zindex = 100,
enter = false,
ns = require('Trans').ns,
}
return win
end
local function handle_keymap(win, word)
local keymap = hover.keymap
local cur_buf = api.nvim_get_current_buf()
local del = vim.keymap.del
local function try_del_keymap()
for _, key in pairs(keymap) do
pcall(del, 'n', key, { buffer = cur_buf })
end end
end end
local lock = false
local cmd_id local cmd_id
local pin local next = win.id
local next local action = {
local action
action = {
pageup = function() pageup = function()
m_window:normal('gg') buffer:normal('gg')
end, end,
pagedown = function() pagedown = function()
m_window:normal('G') buffer:normal('G')
end, end,
pin = function() pin = function()
if pin then if lock then
error('too many window') error('too many window')
else
lock = true
end end
pcall(api.nvim_del_autocmd, cmd_id) pcall(api.nvim_del_autocmd, cmd_id)
local width = win.width
m_window:try_close { local height = win.height
callback = function() local col = vim.o.columns - width - 3
m_window:reopen { local buf = buffer.bufnr
win_opt = { win:try_close()
win.tasks:add(function()
win = open_window {
width = width,
height = height,
relative = 'editor', relative = 'editor',
row = 1, col = col,
col = vim.o.columns - m_window.width - 3, task = function(self)
}, self:set('wrap', true)
opt = { end,
callback = function()
m_window:bufset('bufhidden', 'wipe')
m_window:set('wrap', true)
end
},
} }
vim.keymap.del('n', conf.hover.keymap.pin, { buffer = true }) del('n', keymap.pin, { buffer = cur_buf })
--- NOTE : 只允许存在一个pin窗口
local buf = m_window.bufnr
pin = true
local toggle = conf.hover.keymap.toggle_entry
if toggle then
next = m_window.winid
vim.keymap.set('n', toggle, action.toggle_entry, { silent = true, buffer = buf })
end
api.nvim_create_autocmd('BufWipeOut', { api.nvim_create_autocmd('BufWipeOut', {
callback = function(opt) callback = function(opt)
if opt.buf == buf then if opt.buf == buf or opt.buf == cur_buf then
pin = false lock = false
api.nvim_del_autocmd(opt.id) api.nvim_del_autocmd(opt.id)
end end
end end
}) })
end end)
}
end, end,
close = function() close = function()
pcall(api.nvim_del_autocmd, cmd_id) pcall(api.nvim_del_autocmd, cmd_id)
m_window:try_close { wipeout = true } win:try_close()
win.tasts:add(function()
buffer:delete()
end)
try_del_keymap() try_del_keymap()
end, end,
toggle_entry = function() toggle_entry = function()
if pin and m_window:is_open() then if lock and win:is_valid() then
local prev = api.nvim_get_current_win() local prev = api.nvim_get_current_win()
api.nvim_set_current_win(next) api.nvim_set_current_win(next)
next = prev next = prev
else else
vim.keymap.del('n', conf.hover.keymap.toggle_entry, { buffer = true }) del('n', keymap.toggle_entry, { buffer = cur_buf })
end end
end, end,
play = function() play = function()
m_result.word:play() if word then
word:play()
end
end, end,
} }
local set = vim.keymap.set
local function handle() local opts = { buffer = cur_buf, silent = true }
local hover = conf.hover
if m_result.translation and hover.auto_play then
local ok = pcall(action.play)
if not ok then
vim.notify('自动发音失败, 请检查README发音部分', vim.log.WARN)
end
end
for _, field in ipairs(conf.order) do
process[field]()
end
for act, key in pairs(hover.keymap) do for act, key in pairs(hover.keymap) do
vim.keymap.set('n', key, action[act], { buffer = true, silent = true }) set('n', key, action[act], opts)
end
if hover.auto_close_events then
cmd_id = api.nvim_create_autocmd(
hover.auto_close_events, {
buffer = 0,
callback = function(opt)
win:try_close()
win.tasks:add(function()
buffer:delete()
try_del_keymap()
end)
api.nvim_del_autocmd(opt.id)
end,
})
end end
end end
local function online_query(word) local function online_query(win, word)
local lists = {} -- FIXME :
local lists = {
remove = table.remove
}
local engines = conf.engines local engines = conf.engines
local size = #engines local size = #engines
local icon = conf.icon local icon = conf.icon
local error_msg = icon.notfound .. ' 没有找到相关的翻译' local error_line = it(error_msg, 'TransFailed')
m_window:set_height(1)
local origin_width = m_window.width
m_window:set_width(error_msg:width())
if size == 0 then if size == 0 then
m_content:addline(it(error_msg, 'TransFailed')) buffer:addline(error_line)
m_window:open()
return
else else
m_window:open()
for i = 1, size do for i = 1, size do
lists[size] = require('Trans.query.' .. engines[i])(word) lists[size] = require('Trans.query.' .. engines[i])(word)
end end
end local win_width = win.width
local cell = icon.cell local cell = icon.cell
local spinner = require('Trans.ui.spinner')[conf.hover.spinner] local spinner = require('Trans.ui.spinner')[hover.spinner]
local range = #spinner local range = #spinner
local timeout = conf.hover.timeout local timeout = hover.timeout
local interval = math.floor(timeout / (m_window.width - spinner[1]:width())) local interval = math.floor(timeout / (win.width - spinner[1]:width()))
local width = m_window.width
local f = '%s %s' local s = '%s %s'
require('Trans.util.animation')({ local width = hover.width
times = width, local height = hover.height
buffer:set('modifiable', true)
require('Trans.util.display') {
times = win_width,
interval = interval, interval = interval,
frame = function(self, times) frame = function(self, times)
m_content:wipe()
for i, v in ipairs(lists) do for i, v in ipairs(lists) do
local res = v.value local res = v[1]
if res then if res then
m_result = res vim.pretty_print(res)
m_window:set_width(origin_width) buffer:del(1)
handle() win:set_width(width)
m_content:attach() handle_result(res)
local actual_height = buffer:height {
m_window.height = m_content:actual_height(true) width = width,
m_window:open { wrap = true,
animation = 'fold',
} }
height = math.min(height, actual_height)
win:expand {
field = 'height',
target = height,
}
win.tasks:add(function(this)
this:set('wrap', true)
handle_keymap(this, word)
end)
self.run = false self.run = false
return return
elseif res == false then elseif res == false then
table.remove(lists, i) lists:remove(i)
size = size - 1 size = size - 1
end end
end end
local line local line
if size == 0 or times == width then if size == 0 or times == win_width then
line = it(error_msg, 'TransFailed') line = error_line
self.run = false self.run = false
win:set('wrap', true)
handle_keymap(win, word)
else else
line = it(f:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg') line = it(s:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg')
end end
m_content:addline(line) buffer:addline(line, 1)
m_content:attach()
end, end,
}):display()
callback = function()
buffer:set('modifiable', false)
end,
}
end
end end
return function(word) return function(word)
vim.validate { buffer:init()
word = { word, 's' }, local result = require('Trans.query.offline')(word)
local opts
if result then
handle_result(result)
local width = hover.width
local height = math.min(buffer:height {
width = width,
wrap = true,
}, hover.height)
opts = {
width = width,
height = height,
task = function(self)
self:set('wrap', true)
handle_keymap(self, word)
end
} }
local hover = conf.hover
m_window = new_window(false, {
relative = 'cursor',
width = hover.width,
height = hover.height,
title = hover.title,
border = hover.border,
animation = hover.animation,
col = 1,
row = 1,
})
m_window:set('wrap', true)
m_content = m_window:new_content()
m_result = require('Trans.query.offline')(word)
if m_result then
handle()
local height = m_content:actual_height(true)
if height < m_window.height then
m_window:set_height(height)
end
m_window:open()
else else
online_query(word) opts = {
width = error_msg:width(),
height = 1,
task = function(win)
online_query(win, word)
end
}
end end
-- Auto Close open_window(opts)
if hover.auto_close_events then
cmd_id = api.nvim_create_autocmd(
hover.auto_close_events, {
buffer = 0,
callback = function()
m_window:try_close { wipeout = true }
try_del_keymap()
api.nvim_del_autocmd(cmd_id)
end,
})
end
end end

View File

@ -1,29 +1,23 @@
local api = vim.api local api = vim.api
local new_content = require('Trans.content') local display = require('Trans.util.display')
local new_animation = require('Trans.util.animation')
local busy = false
local function lock()
while busy do
vim.wait(50)
end
busy = true
end
---@class window
---@field winid integer 窗口的handle
---@field bufnr integer 窗口对应buffer的handle
---@field width integer 窗口当前的宽度
---@field height integer 窗口当前的高度
---@field hl integer 窗口highlight的namespace
---@field contents table[] 窗口内容的对象数组
---@type window
local window = { local window = {
set_buf = function(self, buf)
api.nvim_win_set_buf(self.winid, buf)
end,
is_valid = function(self)
return api.nvim_win_is_valid(self.winid)
end,
set = function(self, option, value) set = function(self, option, value)
api.nvim_win_set_option(self.winid, option, value) api.nvim_win_set_option(self.winid, option, value)
end, end,
option = function(self, name)
return api.nvim_win_get_option(self.winid, name)
end,
set_height = function(self, height) set_height = function(self, height)
api.nvim_win_set_height(self.winid, height) api.nvim_win_set_height(self.winid, height)
self.height = height self.height = height
@ -34,214 +28,171 @@ local window = {
self.width = width self.width = width
end, end,
bufset = function(self, option, value) expand = function(self, opts)
api.nvim_buf_set_option(self.bufnr, option, value) self:lock()
end,
---@nodiscard
option = function(self, name)
return api.nvim_win_get_option(self.winid, name)
end,
map = function(self, key, operation)
vim.keymap.set('n', key, operation, {
buffer = self.bufnr,
silent = true,
})
end,
---@nodiscard
is_open = function(self)
return api.nvim_win_is_valid(self.winid)
end,
normal = function(self, key)
api.nvim_buf_call(self.bufnr, function()
vim.cmd([[normal! ]] .. key)
end)
end,
draw = function(self)
local offset = 0
for _, content in ipairs(self.contents) do
content:attach(offset)
offset = offset + content.size
end
end,
open = function(self, opts)
self:draw()
local wrap = self:option('wrap') local wrap = self:option('wrap')
self:set('wrap', false) self:set('wrap', false)
opts = opts or {} local field = opts.field
local animation = opts.animation or self.animation.open local target = opts.target
local interval = opts.interval or self.animation.interval
local callback = function() local callback = function()
busy = false
self:set('wrap', wrap) self:set('wrap', wrap)
if opts.callback then local tasks = self.tasks
opts.callback() for i = 1, #tasks do
tasks[i](self)
tasks[i] = nil
end
self:unlock()
end
local cur = self[field]
local times = math.abs(target - cur)
if times ~= 0 then
local frame
local method = 'set_' .. field
if target > cur then
frame = function(_, cur_times)
self[method](self, cur + cur_times)
end
elseif target < cur then
frame = function(_, cur_times)
self[method](self, cur - cur_times)
end end
end end
lock() display {
if animation then times = times,
local interval = self.animation.interval frame = frame,
local field = ({
fold = 'height',
slid = 'width',
})[animation]
local method = api['nvim_win_set_' .. field]
local winid = self.winid
new_animation({
interval = interval, interval = interval,
times = self[field],
frame = function(_, times)
method(winid, times)
end,
callback = callback, callback = callback,
}):display() }
else else
callback() callback()
end end
end, end,
---安全的关闭窗口 try_close = function(self)
try_close = function(self, opts) if self:is_valid() then
opts = opts or {}
self:set('wrap', false)
if self:is_open() then
local callback = function()
api.nvim_win_close(self.winid, true)
self.winid = -1
busy = false
if opts.callback then
opts.callback()
end
if api.nvim_buf_is_valid(self.bufnr) and opts.wipeout then
api.nvim_buf_delete(self.bufnr, { force = true })
self.bufnr = -1
end
end
lock()
self.config = api.nvim_win_get_config(self.winid)
local animation = self.animation.close
if animation then
local interval = self.animation.interval
local field = ({
fold = 'height',
slid = 'width',
})[animation]
local target = self[field]
local method = api['nvim_win_set_' .. field]
local winid = self.winid local winid = self.winid
new_animation({ self.tasks:add(function()
times = target, api.nvim_win_close(winid, true)
frame = function(_, times) end)
method(winid, target - times)
end,
callback = callback,
interval = interval,
}):display()
else local animation = self.animation
callback() local field = ({
slid = 'width',
fold = 'height',
})[animation.close]
if field then
--- 播放动画
self:expand {
field = field,
target = 1,
debug = true,
}
end end
end end
end, end,
reopen = function(self, opts) lock = function(self)
assert(self.bufnr ~= -1) while self.busy do
local entry = opts.entry or false vim.wait(50)
local win_opt = opts.win_opt or {}
local opt = opts.opt
self.config.win = nil
for k, v in pairs(win_opt) do
self.config[k] = v
end end
self.busy = true
self.winid = api.nvim_open_win(self.bufnr, entry, self.config) end,
self:open(opt)
unlock = function(self)
self.busy = false
end, end,
set_hl = function(self, name, opts) set_hl = function(self, name, opts)
api.nvim_set_hl(self.hl, name, opts) api.nvim_set_hl(self.ns, name, opts)
end, end,
new_content = function(self) center = function(self, node)
local index = self.size + 1 local text = node[1]
self.size = index local width = text:width()
self.contents[index] = new_content(self) local win_width = self.width
local space = math.max(math.floor((win_width - width) / 2), 0)
return self.contents[index] node[1] = (' '):rep(space) .. text
return node
end, end,
} }
window.__index = window window.__index = window
return function(opts)
assert(type(opts) == 'table')
local buf = opts.buf
local height = opts.height
local width = opts.width
local col = opts.col
local row = opts.row
local border = opts.border
local title = opts.title
local relative = opts.relative
local zindex = opts.zindex or 100
local enter = opts.enter
local ns = opts.ns
local animation = opts.animation
local task = opts.task
local open = animation.open
---窗口对象的构造器 local field = ({
---@param entry boolean 光标初始化时是否应该进入窗口 slid = 'width',
---@param option table 需要设置的选项 fold = 'height',
---@return window win })[open]
---@nodiscard
return function(entry, option)
vim.validate {
entry = { entry, 'b' },
option = { option, 't' },
}
local opt = {
relative = option.relative,
width = option.width,
height = option.height,
border = option.border,
title = option.title,
col = option.col,
row = option.row,
local win_opt = {
title_pos = nil, title_pos = nil,
focusable = false, focusable = false,
zindex = option.zindex or 100,
style = 'minimal', style = 'minimal',
zindex = zindex,
width = width,
height = height,
col = col,
row = row,
border = border,
title = title,
relative = relative,
} }
if opt.title then if field then
opt.title_pos = 'center' win_opt[field] = 1
end end
local bufnr = api.nvim_create_buf(false, true) if win_opt.title then
local ok, winid = pcall(api.nvim_open_win, bufnr, entry, opt) win_opt.title_pos = 'center'
if not ok then
error('open window faild: ' .. vim.inspect(opt))
end end
local win local win = setmetatable({
win = { buf = buf,
winid = winid, ns = ns,
bufnr = bufnr, tasks = {
width = opt.width, add = table.insert,
height = opt.height, },
animation = option.animation, height = win_opt.height,
hl = api.nvim_create_namespace('TransWinHl'), width = win_opt.width,
size = 0, animation = animation,
contents = {} winid = api.nvim_open_win(buf.bufnr, enter, win_opt),
}, window)
if task then
win.tasks:add(task)
end
win:expand {
field = field,
target = opts[field],
} }
---@diagnostic disable-next-line: param-type-mismatch api.nvim_win_set_hl_ns(win.winid, win.ns)
setmetatable(win, window)
win:bufset('filetype', 'Trans')
win:bufset('buftype', 'nofile')
api.nvim_win_set_hl_ns(win.winid, win.hl)
win:set_hl('Normal', { link = 'TransWin' }) win:set_hl('Normal', { link = 'TransWin' })
win:set_hl('FloatBorder', { link = 'TransBorder' }) win:set_hl('FloatBorder', { link = 'TransBorder' })
---@diagnostic disable-next-line: return-type-mismatch
return win return win
end end