fix: backup

This commit is contained in:
JuanZoran 2023-03-14 13:18:53 +08:00
parent 2c490994db
commit 4931bdc74a
9 changed files with 79 additions and 348 deletions

View File

@ -1,5 +1,7 @@
---@class buf ---@class buf
---@field bufnr integer buffer handle ---@field bufnr integer buffer handle
---@type buf
local buffer = {} local buffer = {}
local api, fn = vim.api, vim.fn local api, fn = vim.api, vim.fn
@ -36,10 +38,11 @@ function buffer:option(name)
end end
---Destory buffer ---Destory buffer
function buffer:destory() function buffer:destroy()
api.nvim_buf_delete(self.bufnr, { force = true }) api.nvim_buf_delete(self.bufnr, { force = true })
end end
---Set buffer load keymap ---Set buffer load keymap
---@param key string ---@param key string
---@param operation function | string ---@param operation function | string
@ -166,8 +169,9 @@ buffer.__index = function(self, key)
elseif type(key) == 'number' then elseif type(key) == 'number' then
-- return fn.getbufoneline(self.bufnr, key) -- Vimscript Function Or Lua API ?? -- return fn.getbufoneline(self.bufnr, key) -- Vimscript Function Or Lua API ??
return api.nvim_buf_get_lines(self.bufnr, key - 1, key, true)[1] return api.nvim_buf_get_lines(self.bufnr, key - 1, key, true)[1]
else else
error('invalid key' .. key) error('invalid key: ' .. key)
end end
end end

View File

@ -51,17 +51,17 @@ function M.new(opts)
return setmetatable(data, M) return setmetatable(data, M)
end end
-- ---Get the first available result [return nil if no result] ---Get the first available result [return nil if no result]
-- ---@return table? ---@return table?
-- function M:get_available_result() function M:get_available_result()
-- local result = self.result local result = self.result
-- local backend = self.backend local backend = self.backend
-- for _, name in ipairs(backend) do for _, name in ipairs(backend) do
-- if result[name] then if result[name] then
-- return result[name] return result[name]
-- end end
-- end end
-- end end
return M return M

View File

@ -10,6 +10,7 @@ local function set_frontend_keymap(frontend)
for action, key in pairs(frontend.opts.keymap) do for action, key in pairs(frontend.opts.keymap) do
set('n', key, function() set('n', key, function()
local instance = frontend.get_active_instance() local instance = frontend.get_active_instance()
if instance then if instance then
instance:execute(action) instance:execute(action)
else else

View File

@ -47,8 +47,12 @@ local function process(opts)
-- Find in cache -- Find in cache
if Trans.cache[str] then if Trans.cache[str] then
local data = Trans.cache[str] local data = Trans.cache[str]
data.frontend:process(data)
return local result = data:get_available_result()
if result then
data.frontend:process(data, result)
return
end
end end

View File

@ -49,6 +49,8 @@ function M.get_str(mode)
end end
---Puase coroutine for {ms} milliseconds
---@param ms integer milliseconds
function M.pause(ms) function M.pause(ms)
local co = coroutine.running() local co = coroutine.running()
vim.defer_fn(function() vim.defer_fn(function()
@ -57,6 +59,10 @@ function M.pause(ms)
coroutine.yield() coroutine.yield()
end end
---Detect whether the string is English
---@param str string string to detect
---@return boolean
function M.is_English(str) function M.is_English(str)
local char = { str:byte(1, -1) } local char = { str:byte(1, -1) }
for i = 1, #str do for i = 1, #str do

View File

@ -54,11 +54,6 @@ function window:width()
return api.nvim_win_get_width(self.winid) return api.nvim_win_get_width(self.winid)
end end
function window:highlight_line(linenr, highlight)
self.buffer:highlight_line(linenr, highlight, self.ns)
end
---Get window height ---Get window height
function window:height() function window:height()
return api.nvim_win_get_height(self.winid) return api.nvim_win_get_height(self.winid)
@ -70,8 +65,8 @@ end
---|'target'integer ---|'target'integer
function window:smooth_expand(opts) function window:smooth_expand(opts)
local field = opts.field -- width | height local field = opts.field -- width | height
local from = self[field](self) local from = api['nvim_win_get_' .. field](self.winid)
local to = opts.target local to = opts.to
if from == to then return end if from == to then return end
@ -92,6 +87,27 @@ function window:smooth_expand(opts)
self:set('wrap', wrap) self:set('wrap', wrap)
end end
function M:resize(opts)
local width = opts[1]
local height = opts[2]
if self:width() ~= width then
self:smooth_expand {
field = 'width',
to = width
}
end
if self:height() ~= height then
self:smooth_expand {
field = 'height',
to = height
}
end
end
---Try to close window with animation? ---Try to close window with animation?
function window:try_close() function window:try_close()
local close_animation = self.animation.close local close_animation = self.animation.close
@ -103,7 +119,7 @@ function window:try_close()
self:smooth_expand({ self:smooth_expand({
field = field, field = field,
target = 1, to = 1,
}) })
end end
@ -132,23 +148,23 @@ function window:open()
self.winid = api.nvim_open_win(self.buffer.bufnr, self.enter, win_opts) self.winid = api.nvim_open_win(self.buffer.bufnr, self.enter, win_opts)
self:smooth_expand({ self:smooth_expand({
field = field, field = field,
target = to, to = to,
}) })
else else
self.winid = api.nvim_open_win(self.buffer.bufnr, self.enter, win_opts) self.winid = api.nvim_open_win(self.buffer.bufnr, self.enter, win_opts)
end end
end end
---buffer:addline() helper function
---@param node table
---@return table node formatted node
function window:center(node) function window:center(node)
local text = node[1] -- TODO :
local width = text:width() print('TODO Center')
local win_width = self.width -- local text = node[1]
local space = math.max(math.floor((win_width - width) / 2), 0) -- local width = text:width()
node[1] = (' '):rep(space) .. text -- local win_width = self.width
return node -- local space = math.max(math.floor((win_width - width) / 2), 0)
-- node[1] = (' '):rep(space) .. text
-- return node
end end
window.__index = window window.__index = window
@ -178,25 +194,6 @@ end
return window return window
-- local ns = opts.ns
-- local buf = opts.buf
-- local col = opts.col
-- local row = opts.row
-- local title = opts.title
-- local width = opts.width
-- local height = opts.height
-- local border = opts.border
-- local zindex = opts.zindex
-- local relative = opts.relative
-- local animation = opts.animation
-- local open = animation.open
-- local field = ({
-- slid = 'width',
-- fold = 'height',
-- })[open]
-- local win_opt = { -- local win_opt = {
-- focusable = false, -- focusable = false,
-- style = 'minimal', -- style = 'minimal',

View File

@ -1,5 +1,5 @@
local strategy = { local strategy = {
play = function(self) play = function()
print('TODO: play') print('TODO: play')
end, end,
pageup = function() pageup = function()
@ -11,8 +11,8 @@ local strategy = {
pin = function() pin = function()
print('TODO: pin') print('TODO: pin')
end, end,
close = function() close = function(hover)
print('TODO: close') hover:destroy()
end, end,
toggle_entry = function() toggle_entry = function()
print('TODO: toggle_entry') print('TODO: toggle_entry')
@ -21,7 +21,7 @@ local strategy = {
return function(self, action) return function(hover, action)
-- TODO : -- TODO :
strategy[action](self) coroutine.wrap(strategy[action])(hover)
end end

View File

@ -2,7 +2,7 @@ local Trans = require('Trans')
---@class hover ---@class hover
---@field queue table @hover queue for all hover instances ---@field queue table @hover queue for all hover instances
---@field buffer buffer @buffer for hover window ---@field buffer buf @buffer for hover window
---@field destroy_funcs table @functions to be executed when hover window is closed ---@field destroy_funcs table @functions to be executed when hover window is closed
---@field window window @hover window ---@field window window @hover window
---@field opts table @options for hover window ---@field opts table @options for hover window
@ -62,10 +62,12 @@ function M:destroy()
func(self) func(self)
end end
self.window:try_close()
self.buffer:destroy() if self.window:is_valid() then self.window:try_close() end
if self.buffer:is_valid() then self.buffer:destroy() end
end end
---Init hover window ---Init hover window
---@param opts table? @window options: width, height ---@param opts table? @window options: width, height
---@return unknown ---@return unknown
@ -127,9 +129,8 @@ function M:wait(tbl, name, timeout)
pause(interval) pause(interval)
end end
buffer[1] = ''
-- TODO : End waitting animation -- TODO : End waitting animation
buffer[1] = ''
end end
function M:process(_, result) function M:process(_, result)
@ -145,18 +146,10 @@ function M:process(_, result)
end end
local win = self.window local win = self.window
if not win then if win and win:is_valid() then
win:resize { self.opts.width, self.opts.height }
else
win = self:init_window() win = self:init_window()
elseif win:width() ~= opts.width then
win:expand {
field = 'width',
to = opts.width
}
elseif win:height() ~= opts.height then
win:expand {
field = 'height',
to = opts.height
}
end end
win:set('wrap', true) win:set('wrap', true)
@ -169,17 +162,6 @@ function M:is_available()
end end
return M return M
-- 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)
-- end
-- end
-- local lock = false
-- local cmd_id -- local cmd_id
-- local next -- local next
-- local action = { -- local action = {
@ -258,6 +240,7 @@ return M
-- set('n', key, action[act]) -- set('n', key, action[act])
-- end -- end
-- if hover.auto_close_events then -- if hover.auto_close_events then
-- cmd_id = api.nvim_create_autocmd( -- cmd_id = api.nvim_create_autocmd(
-- hover.auto_close_events, { -- hover.auto_close_events, {

View File

@ -1,264 +0,0 @@
local M = {}
local api, fn = vim.api, vim.fn
if fn.executable('sqlite3') ~= 1 then
error('Please check out sqlite3')
end
local win_title = fn.has('nvim-0.9') == 1 and {
{ '', 'TransTitleRound' },
{ ' Trans', 'TransTitle' },
{ '', 'TransTitleRound' },
} or nil
-- local title = {
-- "████████╗██████╗ █████╗ ███╗ ██╗███████╗",
-- "╚══██╔══╝██╔══██╗██╔══██╗████╗ ██║██╔════╝",
-- " ██║ ██████╔╝███████║██╔██╗ ██║███████╗",
-- " ██║ ██╔══██╗██╔══██║██║╚██╗██║╚════██║",
-- " ██║ ██║ ██║██║ ██║██║ ╚████║███████║",
-- " ╚═╝ ╚═╝ ╚═╝╚═╝ ╚═╝╚═╝ ╚═══╝╚══════╝",
--}
string.width = api.nvim_strwidth
string.isEn = function(self)
local char = { self:byte(1, -1) }
for i = 1, #self do
if char[i] > 128 then
return false
end
end
return true
end
string.play = fn.has('linux') == 1 and function(self)
local cmd = ([[echo "%s" | festival --tts]]):format(self)
fn.jobstart(cmd)
end or function(self)
local seperator = fn.has('unix') and '/' or '\\'
local file = debug.getinfo(1, "S").source:sub(2):match('(.*)lua') .. seperator .. 'tts' .. seperator .. 'say.js'
fn.jobstart('node ' .. file .. ' ' .. self)
end
M.conf = {
view = {
i = 'float',
n = 'hover',
v = 'hover',
},
hover = {
width = 37,
height = 27,
border = 'rounded',
title = win_title,
keymap = {
pageup = '[[',
pagedown = ']]',
pin = '<leader>[',
close = '<leader>]',
toggle_entry = '<leader>;',
play = '_',
},
animation = {
-- open = 'fold',
-- close = 'fold',
open = 'slid',
close = 'slid',
interval = 12,
},
auto_close_events = {
'InsertEnter',
'CursorMoved',
'BufLeave',
},
auto_play = true,
timeout = 2000,
spinner = 'dots', -- 查看所有样式: /lua/Trans/util/spinner
-- spinner = 'moon'
},
float = {
width = 0.8,
height = 0.8,
border = 'rounded',
title = win_title,
keymap = {
quit = 'q',
},
animation = {
open = 'fold',
close = 'fold',
interval = 10,
},
tag = {
wait = '#519aba',
fail = '#e46876',
success = '#10b981',
},
},
order = { -- only work on hover mode
'title',
'tag',
'pos',
'exchange',
'translation',
'definition',
},
icon = {
star = '',
notfound = '',
yes = '',
no = '',
-- --- char: ■ | □ | ▇ | ▏ ▎ ▍ ▌ ▋ ▊ ▉ █
-- --- ◖■■■■■■■◗▫◻ ▆ ▆ ▇⃞ ▉⃞
cell = '',
-- star = '⭐',
-- notfound = '❔',
-- yes = '✔️',
-- no = '❌'
},
theme = 'default',
-- theme = 'dracula',
-- theme = 'tokyonight',
db_path = '$HOME/.vim/dict/ultimate.db',
engine = {
youdao = {},
-- baidu = {
-- appid = '',
-- appPasswd = '',
-- },
-- -- youdao = {
-- appkey = '',
-- appPasswd = '',
-- },
},
-- TODO :
-- register word
-- history = {
-- -- TOOD
-- }
-- TODO :add online translate engine
}
local times = 0
M.setup = function(opts)
if opts then
M.conf = vim.tbl_deep_extend('force', M.conf, opts)
end
local conf = M.conf
local float = conf.float
if 0 < float.height and float.height <= 1 then
float.height = math.floor((vim.o.lines - vim.o.cmdheight - 1) * float.height)
end
if 0 < float.width and float.width <= 1 then
float.width = math.floor(vim.o.columns * float.width)
end
local engines = {}
local i = 1
for k, _ in pairs(conf.engine) do
engines[i] = k
i = i + 1
end
conf.engines = engines
times = times + 1
if times == 1 then
---@format disable
local new_command = api.nvim_create_user_command
new_command('Translate', function() M.translate() end, { desc = ' 单词翻译', })
new_command('TranslateInput', function() M.translate('i') end, { desc = ' 搜索翻译', })
new_command('TransPlay', function()
local word = M.get_word(api.nvim_get_mode().mode)
if word ~= '' and word:isEn() then
word:play()
end
end, { desc = ' 自动发音' })
local set_hl = api.nvim_set_hl
local hls = require('Trans.ui.theme')[conf.theme]
for hl, opt in pairs(hls) do
set_hl(0, hl, opt)
end
---@format enable
end
end
local function get_select()
local _start = fn.getpos("v")
local _end = fn.getpos('.')
if _start[2] > _end[2] or (_start[3] > _end[3] and _start[2] == _end[2]) then
_start, _end = _end, _start
end
local s_row, s_col = _start[2], _start[3]
local e_row, e_col = _end[2], _end[3]
-- print(s_row, e_row, s_col, e_col)
---@type string
---@diagnostic disable-next-line: assign-type-mismatch
local line = fn.getline(e_row)
local uidx = vim.str_utfindex(line, math.min(#line, e_col))
e_col = vim.str_byteindex(line, uidx)
if s_row == e_row then
return line:sub(s_col, e_col)
else
local lines = fn.getline(s_row, e_row)
local i = #lines
lines[1] = lines[1]:sub(s_col)
lines[i] = line:sub(1, e_col)
return table.concat(lines)
end
end
M.get_word = function(mode)
local word
if mode == 'n' then
word = fn.expand('<cword>')
elseif mode == 'v' then
api.nvim_input('<ESC>')
word = get_select()
elseif mode == 'i' then
-- TODO Use Telescope with fuzzy finder
---@diagnostic disable-next-line: param-type-mismatch
word = fn.input('请输入需要查询的单词:')
else
error('invalid mode: ' .. mode)
end
return word
end
M.translate = function(mode, view)
local function demo()
local frame = 1000
for i = 1, 10 do
print('now is ' .. i)
vim.wait(frame)
end
end
vim.loop.new_async(demo):send()
-- vim.validate {
-- mode = { mode, 's', true },
-- view = { view, 's', true }
-- }
-- mode = mode or api.nvim_get_mode().mode
-- view = view or M.conf.view[mode]
-- assert(mode and view)
-- local word = M.get_word(mode)
-- if word == nil or word == '' then
-- return
-- else
-- require('Trans.view.' .. view)(word:gsub('^%s+', '', 1))
-- end
end
M.ns = api.nvim_create_namespace('Trans')
return M