diff --git a/lua/Trans/init.lua b/lua/Trans/init.lua index b88bec4..7ba4caa 100644 --- a/lua/Trans/init.lua +++ b/lua/Trans/init.lua @@ -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 diff --git a/lua/Trans/query/baidu.lua b/lua/Trans/query/baidu.lua index 61c5c8f..73c0d52 100644 --- a/lua/Trans/query/baidu.lua +++ b/lua/Trans/query/baidu.lua @@ -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) - if res and res.trans_result then - result = { - word = word, - translation = res.trans_result[1].dst, - } + local res = vim.json.decode(str) + if res and res.trans_result then + 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 + return result end diff --git a/lua/Trans/util/spinner.lua b/lua/Trans/ui/spinner.lua similarity index 100% rename from lua/Trans/util/spinner.lua rename to lua/Trans/ui/spinner.lua diff --git a/lua/Trans/theme.lua b/lua/Trans/ui/theme.lua similarity index 100% rename from lua/Trans/theme.lua rename to lua/Trans/ui/theme.lua diff --git a/lua/Trans/util/animation.lua b/lua/Trans/util/animation.lua new file mode 100644 index 0000000..81f7f77 --- /dev/null +++ b/lua/Trans/util/animation.lua @@ -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 diff --git a/lua/Trans/util/curl.lua b/lua/Trans/util/curl.lua index a136328..50af442 100644 --- a/lua/Trans/util/curl.lua +++ b/lua/Trans/util/curl.lua @@ -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)) - end, - } - end + + 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, + } vim.fn.jobstart(table.concat(cmd, ' '), option) end diff --git a/lua/Trans/view/hover.lua b/lua/Trans/view/hover.lua index d291ca4..ff6d76b 100644 --- a/lua/Trans/view/hover.lua +++ b/lua/Trans/view/hover.lua @@ -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,58 +300,53 @@ 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() - m_content:wipe() - for j = 1, size do - local res = wait[j]() - if res then - m_result = res - m_window:set_width(width) - handle() + require('Trans.util.animation')({ + times = m_window.width, + frame = function(self, times) + m_content:wipe() + for i, v in ipairs(lists) do + local res = v.value + if res then + m_result = res + m_window:set_width(width) + handle() + m_content:attach() + + m_window.height = m_content:actual_height(true) + m_window:open { + animation = 'fold', + } + + self.run = false + return + elseif res == 'false' then + table.remove(lists, i) + size = size - 1 + end + end + + if size == 0 then + m_content:addline( + it(error_msg, 'TransFailed') + ) m_content:attach() - m_window.height = m_content:actual_height(true) - m_window:open { - animation = 'fold', - } - return - - elseif res == false then - table.remove(wait, j) - size = size - 1 + else + m_content:addline( + it(f:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg') + ) + m_content:attach() end - end - - if i == m_window.width or size == 0 then - --- HACK : change framework - 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') - ) - 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, diff --git a/lua/Trans/window.lua b/lua/Trans/window.lua index b36b5ae..6d19f32 100644 --- a/lua/Trans/window.lua +++ b/lua/Trans/window.lua @@ -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 - - if animation then - check_busy() - - 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) - - else - busy = false - if callback then - callback() - end - end - end + local callback = function() + busy = false + self:set('wrap', wrap) + if opts.callback then + opts.callback() end + end - handler = { - slid = wrap('slid', 'width'), - fold = wrap('fold', 'height'), - } + lock() + if animation then + local interval = self.animation.interval + local field = ({ + fold = 'height', + slid = 'width', + })[animation] - handler[animation]() + 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() - elseif callback then + else callback() end end, @@ -117,63 +114,58 @@ local window = { ---安全的关闭窗口 try_close = function(self, opts) opts = opts or {} - local callback = opts.callback + self:set('wrap', false) + 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) - - else - vim.defer_fn(function() - api.nvim_win_close(self.winid, true) - self.winid = -1 - busy = false - - if type(callback) == 'function' then - callback() - end - - end, animation.interval + 2) - end - end - end - - handler = { - slid = wrap('slid', 'width'), - fold = wrap('fold', 'height'), - } - - handler[animation.close]() - - else + 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 = 'nvim_win_set_' .. field + new_animation({ + times = target, + frame = function(_, times) + api[method](self.winid, target - times) + end, + callback = callback, + interval = interval, + }):display() + + else + callback() end end end, reopen = function(self, opts) - local entry = opts.entry or false - local win_opt = opts.win_opt - local opt = opts.opt + assert(self.bufnr ~= -1) + local entry = opts.entry or false + 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 + for k, v in pairs(win_opt) do + self.config[k] = v end self.winid = api.nvim_open_win(self.bufnr, entry, self.config)