From 673709071025d8184b09f3ef1ea0b6fa190ce06a Mon Sep 17 00:00:00 2001 From: JuanZoran <1430359574@qq.com> Date: Thu, 2 Feb 2023 12:01:23 +0800 Subject: [PATCH 1/3] fix: fix vistual selection region mistakes --- lua/Trans/content.lua | 2 +- lua/Trans/init.lua | 68 ++++++++++++++++++++++--------------- lua/Trans/query/offline.lua | 2 +- lua/Trans/window.lua | 1 - 4 files changed, 42 insertions(+), 31 deletions(-) diff --git a/lua/Trans/content.lua b/lua/Trans/content.lua index 911fb54..ee5f400 100644 --- a/lua/Trans/content.lua +++ b/lua/Trans/content.lua @@ -135,4 +135,4 @@ return function(window) lines = {}, highlights = {}, }, content) -end \ No newline at end of file +end diff --git a/lua/Trans/init.lua b/lua/Trans/init.lua index 4822031..39dba3c 100644 --- a/lua/Trans/init.lua +++ b/lua/Trans/init.lua @@ -1,13 +1,15 @@ -local M = {} +local M = {} +local api = vim.api +local fn = vim.fn -local title = vim.fn.has('nvim-0.9') == 1 and { +local title = fn.has('nvim-0.9') == 1 and { { '', 'TransTitleRound' }, { ' Trans', 'TransTitle' }, { '', 'TransTitleRound' }, } or nil -string.width = vim.fn.strwidth +string.width = api.nvim_strwidth string.isEn = function(self) local char = { self:byte(1, -1) } for i = 1, #self do @@ -19,13 +21,13 @@ string.isEn = function(self) end -string.play = vim.fn.has('linux') == 1 and function(self) +string.play = fn.has('linux') == 1 and function(self) local cmd = ([[echo "%s" | festival --tts]]):format(self) - vim.fn.jobstart(cmd) + fn.jobstart(cmd) end or function(self) - local seperator = vim.fn.has('unix') and '/' or '\\' + local seperator = fn.has('unix') and '/' or '\\' local file = debug.getinfo(1, "S").source:sub(2):match('(.*)lua') .. seperator .. 'tts' .. seperator .. 'say.js' - vim.fn.jobstart('node ' .. file .. ' ' .. self) + fn.jobstart('node ' .. file .. ' ' .. self) end @@ -154,13 +156,12 @@ M.setup = function(opts) times = times + 1 if times == 1 then - local api = vim.api local get_mode = api.nvim_get_mode local set_hl = api.nvim_set_hl local new_command = api.nvim_create_user_command - if vim.fn.executable('sqlite3') ~= 1 then + if fn.executable('sqlite3') ~= 1 then error('Please check out sqlite3') end @@ -187,30 +188,44 @@ M.setup = function(opts) end local function get_select() - local s_start = vim.fn.getpos("v") - local s_end = vim.fn.getpos(".") - if s_start[2] > s_end[2] or s_start[3] > s_end[3] then - s_start, s_end = s_end, s_start - end + local _start = fn.getpos("v") + local _end = fn.getpos('.') - local n_lines = math.abs(s_end[2] - s_start[2]) + 1 - local lines = vim.api.nvim_buf_get_lines(0, s_start[2] - 1, s_end[2], false) - lines[1] = string.sub(lines[1], s_start[3], -1) - if n_lines == 1 then - lines[n_lines] = string.sub(lines[n_lines], 1, s_end[3] - s_start[3] + 1) - else - lines[n_lines] = string.sub(lines[n_lines], 1, s_end[3]) + if _start[2] > _end[2] or (_start[3] > _end[3] and _start[2] == _end[2]) then + _start, _end = _end, _start + end + local s_row = _start[2] + local e_row = _end[2] + local s_col = _start[3] + local e_col = _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 - return table.concat(lines, '') end M.get_word = function(mode) local word if mode == 'n' then - word = vim.fn.expand('') + word = fn.expand('') + elseif mode == 'v' then - vim.api.nvim_input('') + api.nvim_input('') word = get_select() + elseif mode == 'i' then -- TODO Use Telescope with fuzzy finder vim.ui.input({ prompt = '请输入需要查询的单词: ' }, function(input) @@ -229,7 +244,7 @@ M.translate = function(mode, view) view = { view, 's', true } } - mode = mode or vim.api.nvim_get_mode().mode + 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) @@ -240,7 +255,4 @@ M.translate = function(mode, view) end end - -M.augroup = vim.api.nvim_create_augroup('Trans', { clear = true }) - return M diff --git a/lua/Trans/query/offline.lua b/lua/Trans/query/offline.lua index 703d84f..46a2733 100644 --- a/lua/Trans/query/offline.lua +++ b/lua/Trans/query/offline.lua @@ -8,7 +8,7 @@ local path = require('Trans').conf.db_path local dict = db:open(path) vim.api.nvim_create_autocmd('VimLeavePre', { - group = require("Trans").augroup, + once = true, callback = function() if db:isopen() then db:close() diff --git a/lua/Trans/window.lua b/lua/Trans/window.lua index 4eedb42..b7a19bf 100644 --- a/lua/Trans/window.lua +++ b/lua/Trans/window.lua @@ -237,7 +237,6 @@ return function(entry, option) ---@diagnostic disable-next-line: param-type-mismatch setmetatable(win, window) - win:bufset('filetype', 'Trans') win:bufset('buftype', 'nofile') api.nvim_win_set_hl_ns(win.winid, win.hl) From 8b8879b2cdf9435028b2a60333d463071302d3fd Mon Sep 17 00:00:00 2001 From: JuanZoran <1430359574@qq.com> Date: Fri, 3 Feb 2023 11:57:42 +0800 Subject: [PATCH 2/3] =?UTF-8?q?refactor:=20=E9=87=8D=E5=86=99=E4=BA=86?= =?UTF-8?q?=E7=B1=BB=E7=9A=84=E9=80=BB=E8=BE=91,=20=E5=BC=95=E5=85=A5?= =?UTF-8?q?=E4=BA=86=E9=83=A8=E5=88=86bug=E5=BE=85=E8=A7=A3=E5=86=B3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- lua/Trans/buffer.lua | 144 +++++ lua/Trans/content.lua | 138 ---- lua/Trans/init.lua | 2 + lua/Trans/node.lua | 90 ++- lua/Trans/query/baidu.lua | 8 +- lua/Trans/query/offline.lua | 21 +- lua/Trans/util/{animation.lua => display.lua} | 36 +- lua/Trans/view/hover.lua | 598 ++++++++++-------- lua/Trans/window.lua | 311 ++++----- 9 files changed, 695 insertions(+), 653 deletions(-) create mode 100644 lua/Trans/buffer.lua delete mode 100644 lua/Trans/content.lua rename lua/Trans/util/{animation.lua => display.lua} (53%) diff --git a/lua/Trans/buffer.lua b/lua/Trans/buffer.lua new file mode 100644 index 0000000..78ac88e --- /dev/null +++ b/lua/Trans/buffer.lua @@ -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 diff --git a/lua/Trans/content.lua b/lua/Trans/content.lua deleted file mode 100644 index ee5f400..0000000 --- a/lua/Trans/content.lua +++ /dev/null @@ -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 diff --git a/lua/Trans/init.lua b/lua/Trans/init.lua index 39dba3c..2d91f4e 100644 --- a/lua/Trans/init.lua +++ b/lua/Trans/init.lua @@ -255,4 +255,6 @@ M.translate = function(mode, view) end end +M.ns = api.nvim_create_namespace('Trans') + return M diff --git a/lua/Trans/node.lua b/lua/Trans/node.lua index 7c8b48c..0de11b4 100644 --- a/lua/Trans/node.lua +++ b/lua/Trans/node.lua @@ -1,39 +1,73 @@ --- NOTE : 设置content的node -local item_load = function(self, content, line, col) - if self.hl then - content:newhl { - name = self.hl, - line = line, - _start = col, - _end = col + #self.text, - } +local api = vim.api +local add_hl = api.nvim_buf_add_highlight +local ns = require('Trans').ns + +local item_meta = { + load = function(self, bufnr, line, col) + if self[2] then + 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 return { - item = function(text, hl) - return { - text = text, - hl = hl, - load_hl = item_load, - } + item = function(text, highlight) + return setmetatable({ + [1] = text, + [2] = highlight, + }, item_meta) end, - text = function(...) - local items = { ... } + text = function(items) local strs = {} - for i, item in ipairs(items) do - strs[i] = item.text + local size = #items + assert(size > 1) + for i = 1, size do + strs[i] = items[i][1] end - return { - text = table.concat(strs), - load_hl = function(_, content, line, col) - for _, item in ipairs(items) do - item:load_hl(content, line, col) - col = col + #item.text - end - end - } + return setmetatable({ + strs = strs, + size = size, + items = items, + }, text_meta) + 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 + return text end, } diff --git a/lua/Trans/query/baidu.lua b/lua/Trans/query/baidu.lua index d98c0ec..cd23dd6 100644 --- a/lua/Trans/query/baidu.lua +++ b/lua/Trans/query/baidu.lua @@ -41,16 +41,16 @@ return function(word) callback = function(str) local ok, res = pcall(vim.json.decode, str) if ok and res and res.trans_result then - result.value = { - word = word, + result[1] = { + title = { word = word }, [isEn and 'translation' or 'definition'] = res.trans_result[1].dst, } if result.callback then - result.callback(result.value) + result.callback(result[1]) end else - result.value = false + result[1] = false end end, }) diff --git a/lua/Trans/query/offline.lua b/lua/Trans/query/offline.lua index 46a2733..4cfdbcb 100644 --- a/lua/Trans/query/offline.lua +++ b/lua/Trans/query/offline.lua @@ -16,8 +16,9 @@ vim.api.nvim_create_autocmd('VimLeavePre', { end }) + return function(word) - local res = dict:select('stardict', { + local res = (dict:select('stardict', { where = { word = word, }, @@ -33,6 +34,20 @@ return function(word) 'exchange', }, limit = 1, - }) - return res[1] + }))[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 diff --git a/lua/Trans/util/animation.lua b/lua/Trans/util/display.lua similarity index 53% rename from lua/Trans/util/animation.lua rename to lua/Trans/util/display.lua index 8e74d7c..049d16e 100644 --- a/lua/Trans/util/animation.lua +++ b/lua/Trans/util/display.lua @@ -1,33 +1,35 @@ -local display = function(self) - local callback = self.callback or function() +return function(opts) + local callback = opts.callback or function() end + opts.run = true - local target = self.times - if self.sync then + local target = opts.times + if opts.sync then if target then for i = 1, target do - if self.run then - self:frame(i) + if opts.run then + opts:frame(i) end end else - while self.run do - self:frame() + while opts.run do + opts:frame() end end callback() + else local frame if target then local times = 0 frame = function() - if self.run and times < target then + if opts.run and times < target then times = times + 1 - self:frame(times) - vim.defer_fn(frame, self.interval) + opts:frame(times) + vim.defer_fn(frame, opts.interval) else callback() end @@ -35,9 +37,9 @@ local display = function(self) else frame = function() - if self.run then - self:frame() - vim.defer_fn(frame, self.interval) + if opts.run then + opts:frame() + vim.defer_fn(frame, opts.interval) else callback() end @@ -45,11 +47,5 @@ local display = function(self) end frame() end -end - - -return function(opts) - opts.run = true - opts.display = display return opts end diff --git a/lua/Trans/view/hover.lua b/lua/Trans/view/hover.lua index 3a00c71..8e41ae8 100644 --- a/lua/Trans/view/hover.lua +++ b/lua/Trans/view/hover.lua @@ -1,55 +1,56 @@ local api = vim.api 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 m_result -local m_content - --- content utility -local node = require("Trans.node") -local t = node.text +local node = require('Trans.node') 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) - m_content:addline( - t(it('', 'TransTitleRound'), it(str, 'TransTitle'), it('', 'TransTitleRound')) - ) -end + local addtitle = function(title) + buffer:addline { + it('', 'TransTitleRound'), + it(title, 'TransTitle'), + it('', 'TransTitleRound'), + } + end -local exist = function(str) - return str and str ~= '' -end + local process = { + title = function(title) + local word = title.word + local oxford = title.oxford + local collins = title.collins + local phonetic = title.phonetic -local process = { - title = function() - local icon = conf.icon - local line - if m_result.word:find(' ', 1, true) then - line = it(m_result.word, 'TransWord') + if not phonetic and not collins and not oxford then + buffer:addline(it(result.word, 'TransWord')) - else - line = m_content:format { - nodes = { - it(m_result.word, 'TransWord'), - t( - it('['), - it(exist(m_result.phonetic) and m_result.phonetic or icon.notfound, 'TransPhonetic'), - 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) - }, - } - end - m_content:addline(line) - end, + else + buffer:addline(f { + width = hover.width, + text = t { + it(word, 'TransWord'), + t { + it('['), + it((phonetic and phonetic ~= '') and phonetic or notfound, 'TransPhonetic'), + it(']') + }, + it(collins and icon.star:rep(collins) or notfound, 'TransCollins'), + it(oxford == 1 and icon.yes or icon.no) + }, + }) + end + end, - tag = function() - if exist(m_result.tag) then - title('标签') + tag = function(tag) + addtitle('标签') local tag_map = { zk = '中考', gk = '高考', @@ -64,17 +65,16 @@ local process = { local tags = {} local size = 0 local interval = ' ' - for tag in vim.gsplit(m_result.tag, ' ', true) do + for _tag in vim.gsplit(tag, ' ', true) do size = size + 1 - tags[size] = tag_map[tag] + tags[size] = tag_map[_tag] end for i = 1, size, 3 do - m_content:addline( + buffer:addline( it( - m_indent .. - tags[i] .. + indent .. tags[i] .. (tags[i + 1] and interval .. tags[i + 1] .. (tags[i + 2] and interval .. tags[i + 2] or '') or ''), 'TransTag' @@ -82,13 +82,11 @@ local process = { ) end - m_content:newline('') - end - end, + buffer:addline('') + end, - pos = function() - if exist(m_result.pos) then - title('词性') + pos = function(pos) + addtitle('词性') local pos_map = { a = '代词pron ', c = '连接词conj ', @@ -105,20 +103,18 @@ local process = { d = '限定词determiner ', } - local f = '%s %2s%%' - for pos in vim.gsplit(m_result.pos, '/', true) do - m_content:addline( - it(m_indent .. f:format(pos_map[pos:sub(1, 1)], pos:sub(3)), 'TransPos') + local s = '%s %2s%%' + for _pos in vim.gsplit(pos, '/', true) do + buffer:addline( + it(indent .. s:format(pos_map[_pos:sub(1, 1)], _pos:sub(3)), 'TransPos') ) end - m_content:newline('') - end - end, + buffer:addline('') + end, - exchange = function() - if exist(m_result.exchange) then - title('词形变化') + exchange = function(exchange) + addtitle('词形变化') local exchange_map = { ['p'] = '过去式 ', ['d'] = '过去分词 ', @@ -132,264 +128,306 @@ 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') + for exc in vim.gsplit(exchange, '/', true) do + buffer:addline( + it(indent .. exchange_map[exc:sub(1, 1)] .. interval .. exc:sub(3), 'TransExchange') ) end - m_content:newline('') - end - end, + buffer:addline('') + end, - translation = function() - if exist(m_result.translation) then - title('中文翻译') + translation = function(translation) + if hover.auto_play then + result.title.word:play() + end - for trs in vim.gsplit(m_result.translation, '\n', true) do - m_content:addline( - it(m_indent .. trs, 'TransTranslation') + addtitle('中文翻译') + + for trs in vim.gsplit(translation, '\n', true) do + buffer:addline( + it(indent .. trs, 'TransTranslation') ) end - end - m_content:newline('') - end, + buffer:addline('') + end, - definition = function() - if exist(m_result.definition) then - title('英文注释') + definition = function(definition) + addtitle('英文注释') - 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 :判断是否需要分割空格 - m_content:addline( - it(m_indent .. def, 'TransDefinition') + buffer:addline( + it(indent .. def, 'TransDefinition') ) end - m_content:newline('') + buffer:addline('') + 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, -} - - -local try_del_keymap = function() - for _, key in pairs(conf.hover.keymap) do - pcall(vim.keymap.del, 'n', key, { buffer = true }) end + buffer:set('modifiable', false) end +local function open_window(opts) + opts = opts or {} + 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 cmd_id -local pin -local next -local action -action = { - pageup = function() - m_window:normal('gg') - end, + 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 - pagedown = function() - m_window:normal('G') - end, - - pin = function() - if pin then - error('too many window') +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 - pcall(api.nvim_del_autocmd, cmd_id) + end - m_window:try_close { - callback = function() - m_window:reopen { - win_opt = { - relative = 'editor', - row = 1, - col = vim.o.columns - m_window.width - 3, - }, - opt = { - callback = function() - m_window:bufset('bufhidden', 'wipe') - m_window:set('wrap', true) - end - }, + local lock = false + local cmd_id + local next = win.id + local action = { + pageup = function() + buffer:normal('gg') + end, + + pagedown = function() + buffer:normal('G') + end, + + pin = function() + if lock then + error('too many window') + else + lock = true + end + pcall(api.nvim_del_autocmd, cmd_id) + local width = win.width + local height = win.height + local col = vim.o.columns - width - 3 + local buf = buffer.bufnr + win:try_close() + win.tasks:add(function() + win = open_window { + width = width, + height = height, + relative = 'editor', + col = col, + task = function(self) + self:set('wrap', true) + end, } - vim.keymap.del('n', conf.hover.keymap.pin, { buffer = true }) - --- 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 - + del('n', keymap.pin, { buffer = cur_buf }) api.nvim_create_autocmd('BufWipeOut', { callback = function(opt) - if opt.buf == buf then - pin = false + if opt.buf == buf or opt.buf == cur_buf then + lock = false api.nvim_del_autocmd(opt.id) end end }) - end - } - end, - - close = function() - pcall(api.nvim_del_autocmd, cmd_id) - m_window:try_close { wipeout = true } - try_del_keymap() - end, - - toggle_entry = function() - if pin and m_window:is_open() then - local prev = api.nvim_get_current_win() - api.nvim_set_current_win(next) - next = prev - else - vim.keymap.del('n', conf.hover.keymap.toggle_entry, { buffer = true }) - end - end, - - play = function() - m_result.word:play() - end, -} - - -local function handle() - 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 - vim.keymap.set('n', key, action[act], { buffer = true, silent = true }) - end -end - -local function online_query(word) - local lists = {} - local engines = conf.engines - local size = #engines - local icon = conf.icon - local error_msg = icon.notfound .. ' 没有找到相关的翻译' - m_window:set_height(1) - local origin_width = m_window.width - m_window:set_width(error_msg:width()) - - if size == 0 then - m_content:addline(it(error_msg, 'TransFailed')) - m_window:open() - return - else - m_window:open() - for i = 1, size do - lists[size] = require('Trans.query.' .. engines[i])(word) - end - end - - local cell = icon.cell - 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 width = m_window.width - - local f = '%s %s' - require('Trans.util.animation')({ - times = width, - interval = interval, - 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(origin_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 - - local line - if size == 0 or times == width then - line = it(error_msg, 'TransFailed') - self.run = false - else - line = it(f:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg') - end - - m_content:addline(line) - m_content:attach() + end) end, - }):display() -end -return function(word) - vim.validate { - word = { word, 's' }, + close = function() + pcall(api.nvim_del_autocmd, cmd_id) + win:try_close() + win.tasts:add(function() + buffer:delete() + end) + try_del_keymap() + end, + + toggle_entry = function() + if lock and win:is_valid() then + local prev = api.nvim_get_current_win() + api.nvim_set_current_win(next) + next = prev + else + del('n', keymap.toggle_entry, { buffer = cur_buf }) + end + end, + + play = function() + if word then + word:play() + end + 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 - online_query(word) + local set = vim.keymap.set + local opts = { buffer = cur_buf, silent = true } + for act, key in pairs(hover.keymap) do + set('n', key, action[act], opts) end - -- Auto Close 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) + 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 + +local function online_query(win, word) + -- FIXME : + local lists = { + remove = table.remove + } + local engines = conf.engines + local size = #engines + local icon = conf.icon + local error_line = it(error_msg, 'TransFailed') + + if size == 0 then + buffer:addline(error_line) + + else + for i = 1, size do + lists[size] = require('Trans.query.' .. engines[i])(word) + end + local win_width = win.width + local cell = icon.cell + local spinner = require('Trans.ui.spinner')[hover.spinner] + local range = #spinner + + local timeout = hover.timeout + local interval = math.floor(timeout / (win.width - spinner[1]:width())) + + local s = '%s %s' + local width = hover.width + local height = hover.height + buffer:set('modifiable', true) + + require('Trans.util.display') { + times = win_width, + interval = interval, + frame = function(self, times) + for i, v in ipairs(lists) do + local res = v[1] + if res then + vim.pretty_print(res) + buffer:del(1) + win:set_width(width) + handle_result(res) + local actual_height = buffer:height { + width = width, + wrap = true, + } + 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 + return + + elseif res == false then + lists:remove(i) + size = size - 1 + end + end + + local line + if size == 0 or times == win_width then + line = error_line + self.run = false + win:set('wrap', true) + handle_keymap(win, word) + + else + line = it(s:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg') + end + + buffer:addline(line, 1) + end, + + callback = function() + buffer:set('modifiable', false) + end, + } + end +end + +return function(word) + buffer:init() + 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 + } + + else + opts = { + width = error_msg:width(), + height = 1, + task = function(win) + online_query(win, word) + end + } + end + + open_window(opts) +end diff --git a/lua/Trans/window.lua b/lua/Trans/window.lua index b7a19bf..55be0fd 100644 --- a/lua/Trans/window.lua +++ b/lua/Trans/window.lua @@ -1,29 +1,23 @@ local api = vim.api -local new_content = require('Trans.content') -local new_animation = require('Trans.util.animation') +local display = require('Trans.util.display') -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 = { + 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) api.nvim_win_set_option(self.winid, option, value) end, + option = function(self, name) + return api.nvim_win_get_option(self.winid, name) + end, + set_height = function(self, height) api.nvim_win_set_height(self.winid, height) self.height = height @@ -34,214 +28,171 @@ local window = { self.width = width end, - bufset = function(self, option, value) - api.nvim_buf_set_option(self.bufnr, option, value) - 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() + expand = function(self, opts) + self:lock() local wrap = self:option('wrap') self:set('wrap', false) - opts = opts or {} - local animation = opts.animation or self.animation.open - local callback = function() - busy = false + local field = opts.field + local target = opts.target + local interval = opts.interval or self.animation.interval + local callback = function() self:set('wrap', wrap) - if opts.callback then - opts.callback() + local tasks = self.tasks + for i = 1, #tasks do + tasks[i](self) + tasks[i] = nil end + self:unlock() end - lock() - if animation then - local interval = self.animation.interval - local field = ({ - fold = 'height', - slid = 'width', - })[animation] + local cur = self[field] + local times = math.abs(target - cur) - local method = api['nvim_win_set_' .. field] - local winid = self.winid - new_animation({ + 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 + + display { + times = times, + frame = frame, interval = interval, - times = self[field], - frame = function(_, times) - method(winid, times) - end, callback = callback, - }):display() + } else callback() end end, - ---安全的关闭窗口 - try_close = function(self, opts) - opts = opts or {} - self:set('wrap', false) + try_close = function(self) + if self:is_valid() then + local winid = self.winid + self.tasks:add(function() + api.nvim_win_close(winid, true) + end) - 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 + local animation = self.animation + local field = ({ + slid = 'width', + fold = 'height', + })[animation.close] - 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 - new_animation({ - times = target, - frame = function(_, times) - method(winid, target - times) - end, - callback = callback, - interval = interval, - }):display() - - else - callback() + if field then + --- 播放动画 + self:expand { + field = field, + target = 1, + debug = true, + } end end end, - reopen = function(self, opts) - assert(self.bufnr ~= -1) - local entry = opts.entry or false - 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 + lock = function(self) + while self.busy do + vim.wait(50) end + self.busy = true - self.winid = api.nvim_open_win(self.bufnr, entry, self.config) - self:open(opt) + end, + + unlock = function(self) + self.busy = false end, set_hl = function(self, name, opts) - api.nvim_set_hl(self.hl, name, opts) + api.nvim_set_hl(self.ns, name, opts) end, - new_content = function(self) - local index = self.size + 1 - self.size = index - self.contents[index] = new_content(self) - - return self.contents[index] + center = function(self, node) + local text = node[1] + local width = text:width() + local win_width = self.width + local space = math.max(math.floor((win_width - width) / 2), 0) + node[1] = (' '):rep(space) .. text + return node end, } - 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 ----窗口对象的构造器 ----@param entry boolean 光标初始化时是否应该进入窗口 ----@param option table 需要设置的选项 ----@return window win ----@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 field = ({ + slid = 'width', + fold = 'height', + })[open] + local win_opt = { title_pos = nil, focusable = false, - zindex = option.zindex or 100, style = 'minimal', + zindex = zindex, + width = width, + height = height, + col = col, + row = row, + border = border, + title = title, + relative = relative, } - if opt.title then - opt.title_pos = 'center' + if field then + win_opt[field] = 1 end - local bufnr = api.nvim_create_buf(false, true) - local ok, winid = pcall(api.nvim_open_win, bufnr, entry, opt) - if not ok then - error('open window faild: ' .. vim.inspect(opt)) + if win_opt.title then + win_opt.title_pos = 'center' end - local win - win = { - winid = winid, - bufnr = bufnr, - width = opt.width, - height = opt.height, - animation = option.animation, - hl = api.nvim_create_namespace('TransWinHl'), - size = 0, - contents = {} + local win = setmetatable({ + buf = buf, + ns = ns, + tasks = { + add = table.insert, + }, + height = win_opt.height, + width = win_opt.width, + animation = animation, + 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 - setmetatable(win, window) - - win:bufset('filetype', 'Trans') - win:bufset('buftype', 'nofile') - api.nvim_win_set_hl_ns(win.winid, win.hl) + api.nvim_win_set_hl_ns(win.winid, win.ns) win:set_hl('Normal', { link = 'TransWin' }) win:set_hl('FloatBorder', { link = 'TransBorder' }) - ---@diagnostic disable-next-line: return-type-mismatch return win end From b62478cf2d52e945d64b5b5e8facb67195098da3 Mon Sep 17 00:00:00 2001 From: JuanZoran <1430359574@qq.com> Date: Fri, 3 Feb 2023 15:27:00 +0800 Subject: [PATCH 3/3] =?UTF-8?q?fix:=20=E5=8F=96=E6=B6=88=E4=BA=86=E5=9B=9E?= =?UTF-8?q?=E8=B0=83=E7=9A=84=E8=AE=BE=E8=AE=A1,=20=E4=BF=AE=E5=A4=8D?= =?UTF-8?q?=E4=BA=86=E9=83=A8=E5=88=86bug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit 将回调的接口换成了run的函数接口, 修复了在线查询, 自动命令, 窗口显示的bug --- lua/Trans/buffer.lua | 17 ++-- lua/Trans/node.lua | 6 +- lua/Trans/util/display.lua | 77 +++++++++---------- lua/Trans/view/float.lua | 154 +++++++++++++++++++------------------ lua/Trans/view/hover.lua | 152 ++++++++++++++---------------------- lua/Trans/window.lua | 106 ++++++++++--------------- 6 files changed, 227 insertions(+), 285 deletions(-) diff --git a/lua/Trans/buffer.lua b/lua/Trans/buffer.lua index 78ac88e..0a64ebb 100644 --- a/lua/Trans/buffer.lua +++ b/lua/Trans/buffer.lua @@ -42,6 +42,11 @@ local buffer = { end end, + wipe = function(self) + api.nvim_buf_set_lines(self.bufnr, 0, -1, false, {}) + self.size = 0 + end, + del = function(self, _start, _end) if not _start then fn.deletebufline(self.bufnr, '$') @@ -91,14 +96,10 @@ local buffer = { 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 + height = function(self, width) + local size = self.size + if width then + local lines = self:lines() local height = 0 for i = 1, size do height = height + math.max(1, (math.ceil(lines[i]:width() / width))) diff --git a/lua/Trans/node.lua b/lua/Trans/node.lua index 0de11b4..1deb1e6 100644 --- a/lua/Trans/node.lua +++ b/lua/Trans/node.lua @@ -59,12 +59,12 @@ return { format = function(opts) local text = opts.text + local size = text.size 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) + local wid = text[1]:width() + local space = math.max(math.floor((width - wid) / (size - 1)), 0) if space > 0 then text.step = spin:rep(space) end diff --git a/lua/Trans/util/display.lua b/lua/Trans/util/display.lua index 049d16e..b740f73 100644 --- a/lua/Trans/util/display.lua +++ b/lua/Trans/util/display.lua @@ -1,51 +1,48 @@ return function(opts) - local callback = opts.callback or function() - - end - opts.run = true - local target = opts.times - if opts.sync then - if target then - for i = 1, target do - if opts.run then - opts:frame(i) - end - end + opts.run = target ~= 0 - else - while opts.run do - opts:frame() + ---@type function[] + local tasks = {} + local function do_task() + for _, task in ipairs(tasks) do + task() + end + end + + local frame + if target then + local times = 0 + frame = function() + if opts.run and times < target then + times = times + 1 + opts:frame(times) + vim.defer_fn(frame, opts.interval) + + else + do_task() end end - callback() - else - local frame - if target then - local times = 0 - frame = function() - if opts.run and times < target then - times = times + 1 - opts:frame(times) - vim.defer_fn(frame, opts.interval) - else - callback() - end - end - - else - frame = function() - if opts.run then - opts:frame() - vim.defer_fn(frame, opts.interval) - else - callback() - end + frame = function() + if opts.run then + opts:frame() + vim.defer_fn(frame, opts.interval) + else + do_task() end end - frame() end - return opts + frame() + + ---任务句柄, 如果任务结束了则立即执行, 否则立即执行 + ---@param task function + return function(task) + if opts.run then + tasks[#tasks + 1] = task + else + task() + end + end end diff --git a/lua/Trans/view/float.lua b/lua/Trans/view/float.lua index 0123b9c..16fae4a 100644 --- a/lua/Trans/view/float.lua +++ b/lua/Trans/view/float.lua @@ -1,11 +1,11 @@ +local api = vim.api local conf = require('Trans').conf -local m_window -local m_result -local m_content +local buffer = require('Trans.buffer')() local node = require("Trans.node") local t = node.text local it = node.item +local f = node.format local engine_map = { @@ -16,102 +16,104 @@ local engine_map = { } local function set_tag_hl(name, status) - local hl = conf.float.tag[status] - m_window:set_hl(name, { - fg = '#000000', - bg = hl, - }) + -- local hl = conf.float.tag[status] + -- m_window:set_hl(name, { + -- fg = '#000000', + -- bg = hl, + -- }) - m_window:set_hl(name .. 'round', { - fg = hl, - }) + -- m_window:set_hl(name .. 'round', { + -- fg = hl, + -- }) end local function set_title() - local title = m_window:new_content() - local github = ' https://github.com/JuanZoran/Trans.nvim' + -- local title = m_window:new_content() + -- local github = ' https://github.com/JuanZoran/Trans.nvim' - title:addline( - title:center(it(github, '@text.uri')) - ) + -- title:addline( + -- title:center(it(github, '@text.uri')) + -- ) - local f = '%s(%d)' + -- local f = '%s(%d)' - local tags = {} - local load_tag = function(engine, index) - set_tag_hl(engine, 'wait') - local round = engine .. 'round' - table.insert(tags, t( - it('', round), - it(f:format(engine_map[engine], index), engine), - it('', round) - )) - end - load_tag('offline', 1) - title:addline(unpack(tags)) - title:newline('') + -- local tags = {} + -- local load_tag = function(engine, index) + -- set_tag_hl(engine, 'wait') + -- local round = engine .. 'round' + -- table.insert(tags, t( + -- it('', round), + -- it(f:format(engine_map[engine], index), engine), + -- it('', round) + -- )) + -- end + -- load_tag('offline', 1) + -- title:addline(unpack(tags)) + -- title:newline('') end local action = { quit = function() - m_window:try_close() + -- m_window:try_close() end, } -local exist = function (str) +local exist = function(str) return str and str ~= '' end local function process() -- TODO : - local icon = conf.icon - m_content:addline(m_content:format { - nodes = { - it(m_result.word, 'TransWord'), - t( - it('['), - it(exist(m_result.phonetic) and m_result.phonetic or icon.notfound, 'TransPhonetic'), - 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) - }, - width = math.floor(m_window.width * 0.5) - }) - m_content:addline(it('该窗口还属于实验性功能 .... ')) + -- local icon = conf.icon + -- m_content:addline(m_content:format { + -- nodes = { + -- it(m_result.word, 'TransWord'), + -- t( + -- it('['), + -- it(exist(m_result.phonetic) and m_result.phonetic or icon.notfound, 'TransPhonetic'), + -- 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) + -- }, + -- width = math.floor(m_window.width * 0.5) + -- }) + -- m_content:addline(it('该窗口还属于实验性功能 .... ')) end return function(word) -- TODO :online query - local float = conf.float - vim.notify('[注意]: float窗口目前还待开发, 如果需要input查询功能, 请将窗口改成hover', - vim.log.WARN) - local opt = { - relative = 'editor', - width = float.width, - height = float.height, - border = float.border, - title = float.title, - animation = float.animation, - row = bit.rshift((vim.o.lines - float.height), 1), - col = bit.rshift((vim.o.columns - float.width), 1), - zindex = 20, - } - m_window = require('Trans.window')(true, opt) - set_title() - m_content = m_window:new_content() - m_result = require('Trans.query.offline')(word) - if m_result then - set_tag_hl('offline', 'success') - process() - else - set_tag_hl('offline', 'fail') - end + -- local float = conf.float + vim.notify([[ +[注意]: +float窗口目前还待开发 +如果需要input查询功能, 请将窗口改成hover]]) + -- local opt = { + -- relative = 'editor', + -- width = float.width, + -- height = float.height, + -- border = float.border, + -- title = float.title, + -- animation = float.animation, + -- row = bit.rshift((vim.o.lines - float.height), 1), + -- col = bit.rshift((vim.o.columns - float.width), 1), + -- zindex = 20, + -- } + -- m_window = require('Trans.window')(true, opt) + -- set_title() + -- m_content = m_window:new_content() + -- m_result = require('Trans.query.offline')(word) + -- if m_result then + -- set_tag_hl('offline', 'success') + -- process() + -- else + -- set_tag_hl('offline', 'fail') + -- end - m_window:open() - m_window:bufset('bufhidden', 'wipe') + -- m_window:open() + -- m_window:bufset('bufhidden', 'wipe') - for act, key in pairs(float.keymap) do - m_window:map(key, action[act]) - end + -- for act, key in pairs(float.keymap) do + -- m_window:map(key, action[act]) + -- end end diff --git a/lua/Trans/view/hover.lua b/lua/Trans/view/hover.lua index 8e41ae8..e9fe8c8 100644 --- a/lua/Trans/view/hover.lua +++ b/lua/Trans/view/hover.lua @@ -14,6 +14,12 @@ local function handle_result(result) local notfound = icon.notfound local indent = ' ' + local word = result.title.word + if hover.auto_play then + string.play(word:isEn() and word or result.definition) + end + + local addtitle = function(title) buffer:addline { it('', 'TransTitleRound'), @@ -24,13 +30,12 @@ local function handle_result(result) local process = { title = function(title) - local word = title.word local oxford = title.oxford local collins = title.collins local phonetic = title.phonetic if not phonetic and not collins and not oxford then - buffer:addline(it(result.word, 'TransWord')) + buffer:addline(it(word, 'TransWord')) else buffer:addline(f { @@ -138,10 +143,6 @@ local function handle_result(result) end, translation = function(translation) - if hover.auto_play then - result.title.word:play() - end - addtitle('中文翻译') for trs in vim.gsplit(translation, '\n', true) do @@ -178,18 +179,17 @@ local function handle_result(result) end local function open_window(opts) - opts = opts or {} + opts = opts or {} + 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') { + return require('Trans.window') { col = col, row = row, - task = task, buf = buffer, relative = relative, width = width, @@ -197,11 +197,10 @@ local function open_window(opts) title = hover.title, border = hover.border, animation = hover.animation, - zindex = 100, + zindex = 80, enter = false, ns = require('Trans').ns, } - return win end local function handle_keymap(win, word) @@ -210,13 +209,13 @@ local function handle_keymap(win, word) local del = vim.keymap.del local function try_del_keymap() for _, key in pairs(keymap) do - pcall(del, 'n', key, { buffer = cur_buf }) + pcall(del, 'n', key) end end local lock = false local cmd_id - local next = win.id + local next local action = { pageup = function() buffer:normal('gg') @@ -228,7 +227,7 @@ local function handle_keymap(win, word) pin = function() if lock then - error('too many window') + error('请先关闭窗口') else lock = true end @@ -237,19 +236,22 @@ local function handle_keymap(win, word) local height = win.height local col = vim.o.columns - width - 3 local buf = buffer.bufnr - win:try_close() - win.tasks:add(function() - win = open_window { + local run = win:try_close() + run(function() + local w, r = open_window { width = width, height = height, relative = 'editor', col = col, - task = function(self) - self:set('wrap', true) - end, } - del('n', keymap.pin, { buffer = cur_buf }) + next = w.winid + win = w + r(function() + w:set('wrap', true) + end) + + del('n', keymap.pin) api.nvim_create_autocmd('BufWipeOut', { callback = function(opt) if opt.buf == buf or opt.buf == cur_buf then @@ -263,8 +265,8 @@ local function handle_keymap(win, word) close = function() pcall(api.nvim_del_autocmd, cmd_id) - win:try_close() - win.tasts:add(function() + local run = win:try_close() + run(function() buffer:delete() end) try_del_keymap() @@ -276,7 +278,7 @@ local function handle_keymap(win, word) api.nvim_set_current_win(next) next = prev else - del('n', keymap.toggle_entry, { buffer = cur_buf }) + del('n', keymap.toggle_entry) end end, @@ -286,31 +288,21 @@ local function handle_keymap(win, word) end end, } - local set = vim.keymap.set - local opts = { buffer = cur_buf, silent = true } for act, key in pairs(hover.keymap) do - set('n', key, action[act], opts) + set('n', key, action[act]) 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, + callback = action.close, }) end end local function online_query(win, word) - -- FIXME : local lists = { remove = table.remove } @@ -318,81 +310,61 @@ local function online_query(win, word) local size = #engines local icon = conf.icon local error_line = it(error_msg, 'TransFailed') - if size == 0 then buffer:addline(error_line) - else for i = 1, size do lists[size] = require('Trans.query.' .. engines[i])(word) end + local cell = icon.cell + local timeout = hover.timeout + local spinner = require('Trans.ui.spinner')[hover.spinner] + local range = #spinner + local interval = math.floor(timeout / (win.width - spinner[1]:width())) local win_width = win.width - local cell = icon.cell - local spinner = require('Trans.ui.spinner')[hover.spinner] - local range = #spinner - - local timeout = hover.timeout - local interval = math.floor(timeout / (win.width - spinner[1]:width())) local s = '%s %s' local width = hover.width local height = hover.height - buffer:set('modifiable', true) - require('Trans.util.display') { + buffer:set('modifiable', true) + local run = require('Trans.util.display') { times = win_width, interval = interval, frame = function(self, times) - for i, v in ipairs(lists) do - local res = v[1] + for i = 1, size do + local res = lists[i][1] if res then - vim.pretty_print(res) - buffer:del(1) + buffer:wipe() win:set_width(width) handle_result(res) - local actual_height = buffer:height { - width = width, - wrap = true, - } + local actual_height = buffer:height(width) 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 return - elseif res == false then lists:remove(i) size = size - 1 end end - local line if size == 0 or times == win_width then - line = error_line + buffer:addline(error_line, 1) self.run = false - win:set('wrap', true) - handle_keymap(win, word) - else - line = it(s:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg') + buffer:addline(it(s:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg'), 1) end - - buffer:addline(line, 1) - end, - - callback = function() - buffer:set('modifiable', false) end, } + + run(function() + buffer:set('modifiable', false) + end) end end @@ -400,34 +372,28 @@ return function(word) buffer:init() 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 { + local win, run = open_window { width = width, - wrap = true, - }, hover.height) - - opts = { - width = width, - height = height, - task = function(self) - self:set('wrap', true) - handle_keymap(self, word) - end + height = math.min(buffer:height(width), hover.height) } + run(function() + win:set('wrap', true) + handle_keymap(win, word) + end) else - opts = { + local win, run = open_window { width = error_msg:width(), height = 1, - task = function(win) - online_query(win, word) - end } - end - open_window(opts) + run(function() + win:set('wrap', true) + handle_keymap(win, word) + online_query(win, word) + end) + end end diff --git a/lua/Trans/window.lua b/lua/Trans/window.lua index 55be0fd..e9dea82 100644 --- a/lua/Trans/window.lua +++ b/lua/Trans/window.lua @@ -30,71 +30,52 @@ local window = { expand = function(self, opts) self:lock() + local field = opts.field + local target = opts.target + local cur = self[field] + local times = math.abs(target - cur) + local wrap = self:option('wrap') self:set('wrap', false) - local field = opts.field - local target = opts.target local interval = opts.interval or self.animation.interval - local callback = function() + local method = 'set_' .. field + + local frame = target > cur and function(_, cur_times) + self[method](self, cur + cur_times) + end or function(_, cur_times) + self[method](self, cur - cur_times) + end + + local run = display { + times = times, + frame = frame, + interval = interval, + } + + run(function() self:set('wrap', wrap) - local tasks = self.tasks - 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 - - display { - times = times, - frame = frame, - interval = interval, - callback = callback, - } - - else - callback() - end + end) + return run end, try_close = function(self) if self:is_valid() then local winid = self.winid - self.tasks:add(function() - api.nvim_win_close(winid, true) - end) - - local animation = self.animation local field = ({ slid = 'width', fold = 'height', - })[animation.close] + })[self.animation.close] - if field then - --- 播放动画 - self:expand { - field = field, - target = 1, - debug = true, - } - end + --- 播放动画 + local run = self:expand { + field = field, + target = 1, + } + run(function() + api.nvim_win_close(winid, true) + end) + return run end end, @@ -125,6 +106,10 @@ local window = { } window.__index = window +---window的构造函数 +---@param opts table +---@return table +---@return function return function(opts) assert(type(opts) == 'table') local buf = opts.buf @@ -139,7 +124,6 @@ return function(opts) local enter = opts.enter local ns = opts.ns local animation = opts.animation - local task = opts.task local open = animation.open @@ -173,26 +157,18 @@ return function(opts) local win = setmetatable({ buf = buf, ns = ns, - tasks = { - add = table.insert, - }, height = win_opt.height, width = win_opt.width, animation = animation, 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], - } - api.nvim_win_set_hl_ns(win.winid, win.ns) win:set_hl('Normal', { link = 'TransWin' }) win:set_hl('FloatBorder', { link = 'TransBorder' }) - return win + + return win, win:expand { + field = field, + target = opts[field], + } end