171 lines
5.2 KiB
Lua
171 lines
5.2 KiB
Lua
local type_check = vim.validate
|
||
|
||
-- NOTE :中文字符及占两个字节宽,但是在lua里是3个字节长度
|
||
-- 为了解决中文字符在lua的长度和neovim显示不一致的问题
|
||
local get_width = vim.fn.strdisplaywidth
|
||
|
||
local function format(win_width, items)
|
||
local size = #items
|
||
local tot_width = 0
|
||
|
||
if items.indent then
|
||
win_width = win_width - items.indent
|
||
end
|
||
|
||
for i = 1, size do
|
||
if type(items[i]) == 'string' then
|
||
items[i] = { items[i] }
|
||
end
|
||
tot_width = tot_width + #items[i][1] + 4
|
||
end
|
||
|
||
|
||
-- 判断宽度是否超过最大宽度
|
||
if tot_width > win_width + 4 then
|
||
-- 放不下则需要分成多行
|
||
local lines = {}
|
||
|
||
-- 行内字符串按照宽度排序
|
||
table.sort(items, function(a, b)
|
||
return #a[1] > #b[1]
|
||
end)
|
||
|
||
local cols = 1
|
||
win_width = win_width - #items[1][1]
|
||
|
||
while win_width > 0 and cols < size do
|
||
cols = cols + 1
|
||
win_width = win_width - #items[cols][1] + 4
|
||
end
|
||
if cols > 1 then
|
||
cols = cols - 1
|
||
end
|
||
|
||
if cols == 1 then -- 只能放在一行时就对齐了
|
||
for i = size, 1, -1 do
|
||
lines[i] = {
|
||
items[i][1],
|
||
highlight = items.highlight,
|
||
indent = items.indent,
|
||
}
|
||
end
|
||
return lines, true
|
||
end
|
||
|
||
|
||
local rows = math.ceil(size / cols)
|
||
local rest = size % cols
|
||
if rest == 0 then
|
||
rest = cols
|
||
end
|
||
|
||
local max_width = get_width(items[1][1])
|
||
local index = 1 -- 当前操作的字符串下标
|
||
for i = rows, 1, -1 do -- 当前操作的行号
|
||
lines[i] = {
|
||
highlight = items.highlight,
|
||
indent = items.indent,
|
||
}
|
||
|
||
local item = items[index]
|
||
-- if not item then
|
||
-- error('item nil ' .. tostring(index) .. ' rows:' .. tostring(rows) .. vim.inspect(items) )
|
||
-- end
|
||
|
||
item[1] = item[1] .. (' '):rep(max_width - get_width(item[1]))
|
||
lines[i][1] = items[index]
|
||
index = index + 1
|
||
end
|
||
|
||
|
||
for col = 2, cols do
|
||
max_width = get_width(items[index][1])
|
||
local _end = col > rest and rows - 1 or rows
|
||
|
||
for i = _end, 1, -1 do
|
||
local item = items[index]
|
||
item[1] = item[1] .. (' '):rep(max_width - get_width(item[1]))
|
||
|
||
|
||
lines[i][col] = item
|
||
index = index + 1
|
||
end
|
||
end
|
||
|
||
return lines, true
|
||
end
|
||
return items
|
||
end
|
||
|
||
local function process(opts)
|
||
type_check {
|
||
opts = { opts, 'table' },
|
||
['opts.field'] = { opts.field, 'table', true },
|
||
['opts.order'] = { opts.order, 'table' },
|
||
['opts.win'] = { opts.win, 'table' },
|
||
['opts.engine'] = { opts.engine, 'table' },
|
||
}
|
||
|
||
if opts.field == nil then
|
||
local lines = { '⚠️ 本地没有找到相关释义' }
|
||
vim.api.nvim_buf_set_lines(opts.bufnr, 0, -1, false, lines)
|
||
vim.api.nvim_win_set_height(opts.winid, 1)
|
||
vim.api.nvim_win_set_width(opts.winid, get_width(lines[1]))
|
||
else
|
||
local content = require('Trans.component.content'):new(opts)
|
||
for _, v in ipairs(opts.order) do
|
||
local component
|
||
if type(v) == 'table' then
|
||
component = require("Trans.component." .. 'offline' --[[ opts.engine ]] .. '.' .. v[1]).component(opts.field
|
||
, v.max_size)
|
||
else
|
||
component = require("Trans.component." .. 'offline' --[[ opts.engine ]] .. '.' .. v).component(opts.field)
|
||
end
|
||
if component then
|
||
for _, items in ipairs(component) do
|
||
|
||
if items.needformat then
|
||
local formatted_items, split = format(opts.win.width, items)
|
||
if split then
|
||
for _, itms in ipairs(formatted_items) do
|
||
content:insert(itms)
|
||
end
|
||
else
|
||
content:insert(formatted_items)
|
||
end
|
||
else
|
||
content:insert(items)
|
||
end
|
||
|
||
if items.emptyline then
|
||
content:insert({ '' })
|
||
end
|
||
end
|
||
end
|
||
|
||
end
|
||
|
||
content:attach()
|
||
end
|
||
|
||
vim.api.nvim_buf_set_option(opts.bufnr, 'modifiable', false)
|
||
vim.api.nvim_buf_set_option(opts.bufnr, 'filetype', 'Trans')
|
||
|
||
vim.api.nvim_win_set_option(opts.winid, 'wrap', true)
|
||
vim.api.nvim_win_set_option(opts.winid, 'winhl', 'Normal:TransCursorWin,FloatBorder:TransCursorBorder')
|
||
if opts.win.style == 'cursor' then
|
||
vim.api.nvim_create_autocmd(
|
||
{ 'InsertEnter', 'CursorMoved', 'BufLeave', }, {
|
||
buffer = 0,
|
||
once = true,
|
||
callback = function()
|
||
if vim.api.nvim_win_is_valid(opts.winid) then
|
||
vim.api.nvim_win_close(opts.winid, true)
|
||
end
|
||
end,
|
||
})
|
||
end
|
||
end
|
||
|
||
return process
|