refactor: add animation utility
This commit is contained in:
parent
59f29f5a33
commit
50ff4980ef
@ -39,7 +39,7 @@ M.conf = {
|
||||
'BufLeave',
|
||||
},
|
||||
auto_play = true,
|
||||
timeout = 3000,
|
||||
timeout = 2000,
|
||||
spinner = 'dots', -- 查看所有样式: /lua/Trans/util/spinner
|
||||
-- spinner = 'moon'
|
||||
},
|
||||
@ -145,7 +145,7 @@ M.setup = function(opts)
|
||||
end, { desc = ' 搜索翻译' })
|
||||
|
||||
|
||||
local hls = require('Trans.theme')[M.conf.theme]
|
||||
local hls = require('Trans.ui.theme')[M.conf.theme]
|
||||
for hl, opt in pairs(hls) do
|
||||
vim.api.nvim_set_hl(0, hl, opt)
|
||||
end
|
||||
|
@ -28,10 +28,10 @@ end
|
||||
--- this is a nice plugin
|
||||
---返回一个channel
|
||||
---@param word string
|
||||
---@return function
|
||||
---@return table
|
||||
return function(word)
|
||||
local query = get_field(word)
|
||||
local result
|
||||
local result = {}
|
||||
|
||||
post(uri, {
|
||||
data = query,
|
||||
@ -39,26 +39,21 @@ return function(word)
|
||||
content_type = "application/x-www-form-urlencoded",
|
||||
},
|
||||
callback = function(str)
|
||||
if result then
|
||||
return
|
||||
elseif str ~= '' then
|
||||
local res = vim.fn.json_decode(str)
|
||||
local res = vim.json.decode(str)
|
||||
if res and res.trans_result then
|
||||
result = {
|
||||
result.value = {
|
||||
word = word,
|
||||
translation = res.trans_result[1].dst,
|
||||
}
|
||||
|
||||
else
|
||||
result = false
|
||||
if result.callback then
|
||||
result.callback(result.value)
|
||||
end
|
||||
else
|
||||
result = false
|
||||
result.value = false
|
||||
end
|
||||
end,
|
||||
})
|
||||
|
||||
return function()
|
||||
return result
|
||||
end
|
||||
end
|
||||
|
57
lua/Trans/util/animation.lua
Normal file
57
lua/Trans/util/animation.lua
Normal file
@ -0,0 +1,57 @@
|
||||
local animation = {
|
||||
display = function(self)
|
||||
local callback = self.callback or function ()
|
||||
|
||||
end
|
||||
|
||||
if self.sync then
|
||||
local times = self.times
|
||||
if times then
|
||||
for i = 1, self.times do
|
||||
if self.run then
|
||||
self:frame(i)
|
||||
end
|
||||
end
|
||||
else
|
||||
while self.run do
|
||||
self:frame()
|
||||
end
|
||||
callback()
|
||||
end
|
||||
|
||||
else
|
||||
local frame
|
||||
if self.times then
|
||||
local target = self.times
|
||||
local times = 0
|
||||
frame = function()
|
||||
if self.run and times < target then
|
||||
times = times + 1
|
||||
self:frame(times)
|
||||
vim.defer_fn(frame, self.interval)
|
||||
else
|
||||
callback()
|
||||
end
|
||||
end
|
||||
|
||||
else
|
||||
frame = function()
|
||||
if self.run then
|
||||
self:frame()
|
||||
vim.defer_fn(frame, self.interval)
|
||||
else
|
||||
callback()
|
||||
end
|
||||
end
|
||||
end
|
||||
frame()
|
||||
end
|
||||
end,
|
||||
}
|
||||
|
||||
animation.__index = animation
|
||||
|
||||
return function(opts)
|
||||
opts.run = true
|
||||
return setmetatable(opts, animation)
|
||||
end
|
@ -10,8 +10,6 @@ local curl = {}
|
||||
-- end,
|
||||
-- }
|
||||
|
||||
|
||||
|
||||
curl.GET = function(uri, opts)
|
||||
--- TODO :
|
||||
end
|
||||
@ -23,6 +21,8 @@ curl.POST = function(uri, opts)
|
||||
opts = { opts, 't' }
|
||||
}
|
||||
|
||||
local callback = opts.callback
|
||||
|
||||
local cmd = { 'curl', '-s', uri }
|
||||
local size = 3
|
||||
|
||||
@ -45,14 +45,20 @@ curl.POST = function(uri, opts)
|
||||
insert('-d', s:format(k, v))
|
||||
end
|
||||
|
||||
local option
|
||||
if opts.callback then
|
||||
option = {
|
||||
on_stdout = function (_, output)
|
||||
opts.callback(table.concat(output))
|
||||
|
||||
local output = ''
|
||||
local option = {
|
||||
stdin = 'null',
|
||||
on_stdout = function(_, stdout)
|
||||
local str = table.concat(stdout)
|
||||
if str ~= '' then
|
||||
output = output .. str
|
||||
end
|
||||
end,
|
||||
on_exit = function()
|
||||
callback(output)
|
||||
end,
|
||||
}
|
||||
end
|
||||
|
||||
vim.fn.jobstart(table.concat(cmd, ' '), option)
|
||||
end
|
||||
|
@ -23,7 +23,6 @@ local exist = function(str)
|
||||
return str and str ~= ''
|
||||
end
|
||||
|
||||
|
||||
local process = {
|
||||
title = function()
|
||||
local icon = conf.icon
|
||||
@ -105,9 +104,11 @@ local process = {
|
||||
t = '不定式标记infm ',
|
||||
d = '限定词determiner ',
|
||||
}
|
||||
|
||||
local f = '%s %s%%'
|
||||
for pos in vim.gsplit(m_result.pos, '/', true) do
|
||||
m_content:addline(
|
||||
it(m_indent .. pos_map[pos:sub(1, 1)] .. pos:sub(3) .. '%', 'TransPos')
|
||||
it(m_indent .. f:format(pos_map[pos:sub(1, 1)], pos:sub(3)), 'TransPos')
|
||||
)
|
||||
end
|
||||
|
||||
@ -131,7 +132,6 @@ local process = {
|
||||
['f'] = '第三人称单数',
|
||||
}
|
||||
local interval = ' '
|
||||
|
||||
for exc in vim.gsplit(m_result.exchange, '/', true) do
|
||||
m_content:addline(
|
||||
it(m_indent .. exchange_map[exc:sub(1, 1)] .. interval .. exc:sub(3), 'TransExchange')
|
||||
@ -195,9 +195,7 @@ action = {
|
||||
if pin then
|
||||
error('too many window')
|
||||
end
|
||||
|
||||
pcall(api.nvim_del_autocmd, cmd_id)
|
||||
m_window:set('wrap', false)
|
||||
|
||||
m_window:try_close {
|
||||
callback = function()
|
||||
@ -209,8 +207,8 @@ action = {
|
||||
},
|
||||
opt = {
|
||||
callback = function()
|
||||
m_window:set('wrap', true)
|
||||
m_window:bufset('bufhidden', 'wipe')
|
||||
m_window:set('wrap', true)
|
||||
end
|
||||
},
|
||||
}
|
||||
@ -239,8 +237,7 @@ action = {
|
||||
|
||||
close = function()
|
||||
pcall(api.nvim_del_autocmd, cmd_id)
|
||||
m_window:set('wrap', false)
|
||||
m_window:try_close()
|
||||
m_window:try_close { wipeout = true }
|
||||
try_del_keymap()
|
||||
end,
|
||||
|
||||
@ -284,13 +281,15 @@ end
|
||||
|
||||
local function online_query(word)
|
||||
-- TODO :Progress Bar
|
||||
local wait = {}
|
||||
local lists = {}
|
||||
local size = 0
|
||||
local icon = conf.icon
|
||||
for k, _ in pairs(conf.engine) do
|
||||
size = size + 1
|
||||
wait[size] = require('Trans.query.' .. k)(word)
|
||||
lists[size] = require('Trans.query.' .. k)(word)
|
||||
end
|
||||
local error_msg = conf.icon.notfound .. ' 没有找到相关的翻译'
|
||||
|
||||
local error_msg = icon.notfound .. ' 没有找到相关的翻译'
|
||||
|
||||
m_window:set_height(1)
|
||||
local width = m_window.width
|
||||
@ -301,21 +300,19 @@ local function online_query(word)
|
||||
return
|
||||
end
|
||||
|
||||
m_window:open()
|
||||
local icon = conf.icon
|
||||
local cell = icon.cell
|
||||
local spinner = require('Trans.util.spinner')[conf.hover.spinner]
|
||||
local spinner = require('Trans.ui.spinner')[conf.hover.spinner]
|
||||
local range = #spinner
|
||||
|
||||
local timeout = conf.hover.timeout
|
||||
local interval = math.floor(timeout / (m_window.width - spinner[1]:width()))
|
||||
local f = '%s %s'
|
||||
local i = 1
|
||||
local do_progress
|
||||
do_progress = function()
|
||||
require('Trans.util.animation')({
|
||||
times = m_window.width,
|
||||
frame = function(self, times)
|
||||
m_content:wipe()
|
||||
for j = 1, size do
|
||||
local res = wait[j]()
|
||||
for i, v in ipairs(lists) do
|
||||
local res = v.value
|
||||
if res then
|
||||
m_result = res
|
||||
m_window:set_width(width)
|
||||
@ -326,33 +323,30 @@ local function online_query(word)
|
||||
m_window:open {
|
||||
animation = 'fold',
|
||||
}
|
||||
return
|
||||
|
||||
elseif res == false then
|
||||
table.remove(wait, j)
|
||||
self.run = false
|
||||
return
|
||||
elseif res == 'false' then
|
||||
table.remove(lists, i)
|
||||
size = size - 1
|
||||
end
|
||||
end
|
||||
|
||||
if i == m_window.width or size == 0 then
|
||||
--- HACK : change framework
|
||||
if size == 0 then
|
||||
m_content:addline(
|
||||
it(error_msg, 'TransFailed')
|
||||
)
|
||||
|
||||
m_content:attach()
|
||||
|
||||
else
|
||||
m_content:addline(
|
||||
it(f:format(spinner[i % range + 1], cell:rep(i)), 'MoreMsg')
|
||||
it(f:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg')
|
||||
)
|
||||
i = i + 1
|
||||
m_content:attach()
|
||||
vim.defer_fn(do_progress, interval)
|
||||
end
|
||||
end
|
||||
|
||||
do_progress()
|
||||
end,
|
||||
interval = interval,
|
||||
}):display()
|
||||
end
|
||||
|
||||
return function(word)
|
||||
@ -372,22 +366,17 @@ return function(word)
|
||||
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()
|
||||
m_window:open({
|
||||
callback = function()
|
||||
m_window:set('wrap', true)
|
||||
end,
|
||||
})
|
||||
|
||||
local height = m_content:actual_height(true)
|
||||
if height < m_window.height then
|
||||
m_window:set_height(height)
|
||||
end
|
||||
m_window:open()
|
||||
else
|
||||
online_query(word)
|
||||
end
|
||||
@ -398,8 +387,7 @@ return function(word)
|
||||
hover.auto_close_events, {
|
||||
buffer = 0,
|
||||
callback = function()
|
||||
m_window:set('wrap', false)
|
||||
m_window:try_close()
|
||||
m_window:try_close { wipeout = true }
|
||||
try_del_keymap()
|
||||
api.nvim_del_autocmd(cmd_id)
|
||||
end,
|
||||
|
@ -1,5 +1,6 @@
|
||||
local api = vim.api
|
||||
local new_content = require('Trans.content')
|
||||
local new_animation = require('Trans.util.animation')
|
||||
|
||||
function string:width()
|
||||
---@diagnostic disable-next-line: param-type-mismatch
|
||||
@ -7,10 +8,11 @@ function string:width()
|
||||
end
|
||||
|
||||
local busy = false
|
||||
local function check_busy()
|
||||
local function lock()
|
||||
while busy do
|
||||
vim.wait(50)
|
||||
end
|
||||
busy = true
|
||||
end
|
||||
|
||||
---@class window
|
||||
@ -74,42 +76,37 @@ local window = {
|
||||
|
||||
open = function(self, opts)
|
||||
self:draw()
|
||||
local wrap = self:option('wrap')
|
||||
self:set('wrap', false)
|
||||
opts = opts or {}
|
||||
local interval = self.animation.interval
|
||||
local animation = opts.animation or self.animation.open
|
||||
local callback = opts.callback
|
||||
local callback = function()
|
||||
busy = false
|
||||
self:set('wrap', wrap)
|
||||
if opts.callback then
|
||||
opts.callback()
|
||||
end
|
||||
end
|
||||
|
||||
lock()
|
||||
if animation then
|
||||
check_busy()
|
||||
local interval = self.animation.interval
|
||||
local field = ({
|
||||
fold = 'height',
|
||||
slid = 'width',
|
||||
})[animation]
|
||||
|
||||
local handler
|
||||
local function wrap(name, target)
|
||||
local count = 0
|
||||
local action = 'nvim_win_set_' .. target
|
||||
return function()
|
||||
if count < self[target] then
|
||||
busy = true
|
||||
count = count + 1
|
||||
api[action](self.winid, count)
|
||||
vim.defer_fn(handler[name], interval)
|
||||
local method = 'nvim_win_set_' .. field
|
||||
new_animation({
|
||||
interval = interval,
|
||||
times = self[field],
|
||||
frame = function(_, times)
|
||||
api[method](self.winid, times)
|
||||
end,
|
||||
callback = callback,
|
||||
}):display()
|
||||
|
||||
else
|
||||
busy = false
|
||||
if callback then
|
||||
callback()
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
handler = {
|
||||
slid = wrap('slid', 'width'),
|
||||
fold = wrap('fold', 'height'),
|
||||
}
|
||||
|
||||
handler[animation]()
|
||||
|
||||
elseif callback then
|
||||
callback()
|
||||
end
|
||||
end,
|
||||
@ -117,64 +114,59 @@ local window = {
|
||||
---安全的关闭窗口
|
||||
try_close = function(self, opts)
|
||||
opts = opts or {}
|
||||
local callback = opts.callback
|
||||
if self:is_open() then
|
||||
check_busy()
|
||||
self.config = api.nvim_win_get_config(self.winid)
|
||||
local animation = self.animation
|
||||
if animation.close then
|
||||
local handler
|
||||
local function wrap(name, target)
|
||||
local count = self[target]
|
||||
local action = 'nvim_win_set_' .. target
|
||||
return function()
|
||||
if count > 1 then
|
||||
busy = true
|
||||
count = count - 1
|
||||
api[action](self.winid, count)
|
||||
vim.defer_fn(handler[name], animation.interval)
|
||||
self:set('wrap', false)
|
||||
|
||||
else
|
||||
vim.defer_fn(function()
|
||||
if self:is_open() then
|
||||
local callback = function()
|
||||
api.nvim_win_close(self.winid, true)
|
||||
self.winid = -1
|
||||
busy = false
|
||||
|
||||
if type(callback) == 'function' then
|
||||
callback()
|
||||
end
|
||||
|
||||
end, animation.interval + 2)
|
||||
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
|
||||
|
||||
handler = {
|
||||
slid = wrap('slid', 'width'),
|
||||
fold = wrap('fold', 'height'),
|
||||
}
|
||||
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]
|
||||
|
||||
handler[animation.close]()
|
||||
local target = self[field]
|
||||
local method = 'nvim_win_set_' .. field
|
||||
new_animation({
|
||||
times = target,
|
||||
frame = function(_, times)
|
||||
api[method](self.winid, target - times)
|
||||
end,
|
||||
callback = callback,
|
||||
interval = interval,
|
||||
}):display()
|
||||
|
||||
else
|
||||
api.nvim_win_close(self.winid, true)
|
||||
self.winid = -1
|
||||
callback()
|
||||
end
|
||||
end
|
||||
end,
|
||||
|
||||
reopen = function(self, opts)
|
||||
assert(self.bufnr ~= -1)
|
||||
local entry = opts.entry or false
|
||||
local win_opt = opts.win_opt
|
||||
local win_opt = opts.win_opt or {}
|
||||
local opt = opts.opt
|
||||
|
||||
check_busy()
|
||||
self.config.win = nil
|
||||
if win_opt then
|
||||
for k, v in pairs(win_opt) do
|
||||
self.config[k] = v
|
||||
end
|
||||
end
|
||||
|
||||
self.winid = api.nvim_open_win(self.bufnr, entry, self.config)
|
||||
self:open(opt)
|
||||
|
Loading…
x
Reference in New Issue
Block a user