diff --git a/README.md b/README.md
index 568ea62..fa5851c 100644
--- a/README.md
+++ b/README.md
@@ -1,6 +1,7 @@
# Trans.nvim
+
- [Trans.nvim](#transnvim)
- [特点](#特点)
- [屏幕截图](#屏幕截图)
@@ -14,15 +15,17 @@
- [感谢](#感谢)
- [贡献](#贡献)
- [待办 (画大饼)](#待办-画大饼)
-
-
+
## 特点
-- 使用纯lua编写, 速度极快
-> `Lazy.nvim`的记录: `➜ Trans.nvim 0.82ms`
+
+- 使用纯 lua 编写, 速度极快
+
+ > `Lazy.nvim`的记录: `➜ Trans.nvim 0.82ms`
- **可以定义快捷键读英文单词**
-> 见wiki
+
+ > 见 wiki
- 大部分功能可以自定义:
- 高亮
@@ -34,19 +37,20 @@
- **完全离线** 的单词翻译体验 (可能后面会支持在线翻译)
- 支持显示:
- 柯林斯星级
- - 牛津3000词汇
+ - 牛津 3000 词汇
- 中文翻译
- - 英文翻译 (不是英译中, 而是用英文解释)
+ - 英文翻译 (不是英译中, 而是用英文解释)
- 词根
- etc
- 舒服的排版和`动画`
- 支持 `normal`和 `visual`模式
- > 不支持 visual-block mode
-
+ > 不支持 visual-block mode
- 本地词库单词量: `430w`
-
+
## 屏幕截图
+
### 演示
+
https://user-images.githubusercontent.com/107862700/213752097-2eee026a-ddee-4531-bf80-ba2cbc8b44ef.mp4
> 视频演示的在线查询, 查询速度取决于你的网络状况
@@ -55,22 +59,23 @@ https://user-images.githubusercontent.com/107862700/213752097-2eee026a-ddee-4531
https://user-images.githubusercontent.com/107862700/215941500-3293c571-20a1-44e2-b202-77079f158ce9.mp4
### 主题
-> 如果你有更美观或者更适合的配色, 欢迎提PR
+
+> 如果你有更美观或者更适合的配色, 欢迎提 PR
> 主题配色在: `lua/Trans/theme.lua`文件中,你只需要添加你主题的表就可以了
-
- `default`
-
+ 
- `dracula`
-
+ 
- `tokyonight`
-
-
+ 
## 安装
-*安装之前, 首先需要明确本插件的依赖:*
+
+_安装之前, 首先需要明确本插件的依赖:_
+
- [ECDICT](https://github.com/skywind3000/ECDICT): 插件所用的离线单词数据库
- sqlite.lua: 操作数据库所用的库
- sqlite3: 数据库
@@ -92,13 +97,14 @@ use {
}
```
-**如果你想要使用Packer的惰性加载,这里有一个例子**
+**如果你想要使用 Packer 的惰性加载,这里有一个例子**
+
```lua
use {
"JuanZoran/Trans.nvim",
keys = {
{ {'n', 'x'}, 'mm' }, -- 换成其他你想用的key即可
- { {'n', 'x'}, 'mk' },
+ { {'n', 'x'}, 'mk' },
{ 'n', 'mi' },
},
run = 'bash ./install.sh', -- 自动下载使用的本地词库
@@ -111,6 +117,7 @@ use {
end
}
```
+
@@ -133,38 +140,46 @@ use {
}
}
```
+
-**注意事项**:
+**注意事项**:
+
- `install.sh`
- - 使用了 `wget`下载词库, 安装请确保你的环境变量中存在wget
+
+ - 使用了 `wget`下载词库, 安装请确保你的环境变量中存在 wget
- install.sh 下载后会自动将词库解压, 并移动到 `$HOME/.vim/dict`文件夹下
- 目前仅在 `Ubuntu22.04`的环境下测试通过
> 如果上述条件不符合, 请删掉 `run = 'install.sh'`部分, 考虑手动安装词库
- > 如果上述条件满足, 仍出现问题, 欢迎在issue里向我反馈,我会及时尝试解决
+ > 如果上述条件满足, 仍出现问题, 欢迎在 issue 里向我反馈,我会及时尝试解决
- 下载词典的过程中, 需要能够 `流畅的访问github下载`
- > 词库文件压缩包大小为: **281M**
- > 解压缩后的大小大概为: 1.2G
+
+ > 词库文件压缩包大小为: **281M**
+ > 解压缩后的大小大概为: 1.2G
- 安装后如果不能正常运行, 请尝试检查一下问题:
+
- 本机是否已经安装了 `sqlite3`
- > Linux下安装:
+ > Linux 下安装:
> `sudo pacman -S sqlite # Arch`
> `sudo apt-get install sqlite3 libsqlite3-dev # Ubuntu`
-
+
> 后续会增加 `healthcheck` 进行检查
- **`auto_play`** 使用步骤:
- > linux 只需要安装`festival`
- > sudo apt-get install festival festvox-kallpc16k
- > ***如果你想要设置音色,发音可以访问:*** [Festival官方](https://www.cstr.ed.ac.uk/projects/festival/morevoices.html)
- > 可以选择英音、美音、男声、女声
- > 其他操作系统
- - 需要确保安装了`nodejs`
- - 进入插件的`tts`目录运行`npm install`
- > 如果`install.sh`运行正常则自动安装,如果安装失败,请尝试手动安装
+ > linux 只需要安装`festival`
+ > sudo apt-get install festival festvox-kallpc16k
+ > **_如果你想要设置音色,发音可以访问:_** [Festival 官方](https://www.cstr.ed.ac.uk/projects/festival/morevoices.html)
+ > 可以选择英音、美音、男声、女声
+
+ > 其他操作系统
+
+ - 需要确保安装了`nodejs`
+ - 进入插件的`tts`目录运行`npm install`
+ > 如果`install.sh`运行正常则自动安装,如果安装失败,请尝试手动安装
+
- `title`的配置,只对`neovim 0.9`版本有效
@@ -174,41 +189,49 @@ use {
- 用户配置: `~/.festivalrc`
- 更改声音
- - 在festival的voices文件内建立自己的文件夹
- > 一般其默认配置目录在`/usr/share/festival/voices`
- 示例:
- > `sudo mkdir /usr/share/festival/voices/my_voices`
+ - 在 festival 的 voices 文件内建立自己的文件夹
- - 下载想要的voices文件并解压
- > 可能需要
+ > 一般其默认配置目录在`/usr/share/festival/voices`
- - 试听[在这里](https://www.cstr.ed.ac.uk/projects/festival/morevoices.html))
- - 下载[在这里](http://festvox.org/packed/festival/2.5/voices/))
- > 假设下载的文件在`Downloads`文件夹, 下载的文件为:`festvox_cmu_us_aew_cg.tar.gz`
+ 示例:
- 示例:
- > `cd ~/Downloads && tar -xf festvox_cmu_us_aew_cg.tar.gz`
+ > `sudo mkdir /usr/share/festival/voices/my_voices`
- - 将音频文件拷贝到festival文件夹
- 示例:
- > `sudo cp -r festival/lib/voices/us/cmu_us_aew_cg/ /usr/share/festival/voices/my_voices/`
+ - 下载想要的 voices 文件并解压
- - 在配置文件中设置默认的声音
- 示例:
- > 加入`(set! voice_default voice_cmu_indic_hin_ab_cg)`到配置文件
+ > 可能需要
- - 安装完成
+ - 试听[在这里](https://www.cstr.ed.ac.uk/projects/festival/morevoices.html))
+ - 下载[在这里](http://festvox.org/packed/festival/2.5/voices/))
+ > 假设下载的文件在`Downloads`文件夹, 下载的文件为:`festvox_cmu_us_aew_cg.tar.gz`
+
+ 示例:
+
+ > `cd ~/Downloads && tar -xf festvox_cmu_us_aew_cg.tar.gz`
+
+ - 将音频文件拷贝到 festival 文件夹
+ 示例:
+
+ > `sudo cp -r festival/lib/voices/us/cmu_us_aew_cg/ /usr/share/festival/voices/my_voices/`
+
+ - 在配置文件中设置默认的声音
+ 示例:
+
+ > 加入`(set! voice_default voice_cmu_indic_hin_ab_cg)`到配置文件
+
+ - 安装完成
- 相关说明网站
- > 可能需要
- - [wiki](https://archlinux.org/packages/community/any/festival-us/) 查看更多详细配置
- - [官方网站](http://festvox.org/dbs/index.html)
- - [用户手册](http://www.festvox.org/docs/manual-2.4.0/festival_toc.html)
+ > 可能需要
+ - [wiki](https://archlinux.org/packages/community/any/festival-us/) 查看更多详细配置
+ - [官方网站](http://festvox.org/dbs/index.html)
+ - [用户手册](http://www.festvox.org/docs/manual-2.4.0/festival_toc.html)
## 配置
+
```lua
require'Trans'.setup {
view = {
@@ -314,12 +337,15 @@ require'Trans'.setup {
-- TODO :add online translate engine
}
-
+
```
## 快捷键绑定
+
**示例:**
+
> 示例中展示, 将`mm`映射成快捷键
+
```lua
vim.keymap.set({'n', 'x'}, 'mm', 'Translate')
vim.keymap.set({'n', 'x'}, 'mk', 'TransPlay') -- 自动发音选中或者光标下的单词
@@ -328,7 +354,9 @@ vim.keymap.set('n', 'mi', 'TranslateInput')
```
## 高亮组
+
> 默认定义
+
```lua
{
TransWord = {
@@ -378,23 +406,27 @@ vim.keymap.set('n', 'mi', 'TranslateInput')
```
## 声明
+
- 本插件词典基于[ECDICT](https://github.com/skywind3000/ECDICT)
## 感谢
-- [ECDICT](https://github.com/skywind3000/ECDICT) 本地词典的提供
-- [sqlite.lua](https://github.com/kharji/sqlite.lua) 数据库访问
-- [T.vim](https://github.com/sicong-li/T.vim) 灵感来源
+
+- [ECDICT](https://github.com/skywind3000/ECDICT) 本地词典的提供
+- [sqlite.lua](https://github.com/kharji/sqlite.lua) 数据库访问
+- [T.vim](https://github.com/sicong-li/T.vim) 灵感来源
## 贡献
+
> 更新比较频繁, 文档先鸽了
-> 如果你想要参加这个项目, 可以提issue, 我会把文档补齐
+> 如果你想要参加这个项目, 可以提 issue, 我会把文档补齐
## 待办 (画大饼)
+
- [x] 多风格样式查询
-- [x] 重新录制屏幕截图示例
-- [x] 快捷键定义
-- [x] 自动读音
+- [x] 重新录制屏幕截图示例
+- [x] 快捷键定义
+- [x] 自动读音
- [ ] 变量命名的支持
- [ ] 历史查询结果保存
-- [ ] 在线多引擎异步查询
-- [ ] `句子翻译` | `中翻英` 的支持
+- [ ] 在线多引擎异步查询
+- [ ] `句子翻译` | `中翻英` 的支持
diff --git a/install.sh b/install.sh
index 57f24e0..629d14f 100755
--- a/install.sh
+++ b/install.sh
@@ -2,18 +2,17 @@
set -e
if test -e "$HOME/.vim/dict/ultimate.db"; then
- exit
+ exit
fi
-
mkdir -p "$HOME/.vim/dict"
wget https://github.com/skywind3000/ECDICT-ultimate/releases/download/1.0.0/ecdict-ultimate-sqlite.zip -O /tmp/dict.zip
unzip /tmp/dict.zip -d "$HOME/.vim/dict" && rm -rf /tmp/dict.zip
-uNames=`uname -s`
-osName=${uNames: 0: 4}
-if [ "$osName" != "Linux" ];then
- cd ./tts/ && npm install
+uNames=$(uname -s)
+osName=${uNames:0:4}
+if [ "$osName" != "Linux" ]; then
+ cd ./tts/ && npm install
fi
diff --git a/lua/Trans/buffer.lua b/lua/Trans/buffer.lua
new file mode 100644
index 0000000..698f8dc
--- /dev/null
+++ b/lua/Trans/buffer.lua
@@ -0,0 +1,167 @@
+local api, fn = vim.api, vim.fn
+
+---@class buf
+---@field bufnr integer buffer handle
+---@field size integer buffer line count
+local buffer = {}
+
+---Clear all content in buffer
+function buffer:wipe()
+ api.nvim_buf_set_lines(self.bufnr, 0, -1, false, {})
+ self.size = 0
+end
+
+---delete buffer [_start, _end] line content [one index]
+---@param _start integer start line index
+---@param _end integer end line index
+function buffer:del(_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 buffer option
+---@param name string option name
+---@param value any option value
+function buffer:set(name, value)
+ api.nvim_buf_set_option(self.bufnr, name, value)
+end
+
+---get buffer option
+---@param name string option name
+---@return any
+function buffer:option(name)
+ return api.nvim_buf_get_option(self.bufnr, name)
+end
+
+function buffer:delete()
+ api.nvim_buf_delete(self.bufnr, { force = true })
+end
+
+---Set buffer load keymap
+---@param key string
+---@param operation function | string
+function buffer:map(key, operation)
+ vim.keymap.set('n', key, operation, {
+ buffer = self.bufnr,
+ silent = true,
+ })
+end
+
+---Execute normal keycode in this buffer[no recursive]
+---@param key string key code
+function buffer:normal(key)
+ api.nvim_buf_call(self.bufnr, function()
+ vim.cmd([[normal! ]] .. key)
+ end)
+end
+
+---@return boolean
+---@nodiscard
+function buffer:is_valid()
+ return api.nvim_buf_is_valid(self.bufnr)
+end
+
+---Get buffer [i, j] line content
+---@param i integer? start line index
+---@param j integer? end line index
+---@return string[]
+function buffer:lines(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
+
+---Calculate buffer content display height
+---@param width integer
+---@return integer height
+function buffer:height(width)
+ local size = self.size
+ local lines = self:lines()
+ local height = 0
+ for i = 1, size do
+ height = height + math.max(1, (math.ceil(lines[i]:width() / width)))
+ end
+ return height
+end
+
+---Add|Set line content
+---@param nodes string|table|table[] string -> as line content | table -> as a node | table[] -> as node[]
+---@param index number? line number should be set[one index]
+function buffer:addline(nodes, index)
+ local newsize = self.size + 1
+ assert(index == nil or index <= newsize)
+ index = index or newsize
+ if index == newsize then
+ self.size = newsize
+ end
+
+ if type(nodes) == 'string' then
+ self[index] = nodes
+ return
+ end
+
+
+ local line = index - 1
+ local bufnr = self.bufnr
+ local col = 0
+ if type(nodes[1]) == 'string' then
+ self[index] = nodes[1]
+ nodes:load(bufnr, line, col)
+ return
+ end
+
+
+ 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
+
+function buffer:init()
+ self.bufnr = api.nvim_create_buf(false, false)
+ self:set('filetype', 'Trans')
+ self:set('buftype', 'nofile')
+ self.size = 0
+end
+
+---@private
+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
+
+---@private
+buffer.__newindex = function(self, key, text)
+ assert(key <= self.size + 1)
+ fn.setbufline(self.bufnr, key, text)
+end
+
+---buffer constructor
+---@return buf
+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 911fb54..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
\ No newline at end of file
diff --git a/lua/Trans/init.lua b/lua/Trans/init.lua
index 4822031..94796a8 100644
--- a/lua/Trans/init.lua
+++ b/lua/Trans/init.lua
@@ -1,13 +1,26 @@
local M = {}
+local api, fn = vim.api, vim.fn
-local title = vim.fn.has('nvim-0.9') == 1 and {
+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 = vim.fn.strwidth
+string.width = api.nvim_strwidth
string.isEn = function(self)
local char = { self:byte(1, -1) }
for i = 1, #self do
@@ -18,17 +31,15 @@ string.isEn = function(self)
return true
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
-
M.conf = {
view = {
i = 'float',
@@ -39,7 +50,7 @@ M.conf = {
width = 37,
height = 27,
border = 'rounded',
- title = title,
+ title = win_title,
keymap = {
pageup = '[[',
pagedown = ']]',
@@ -69,7 +80,7 @@ M.conf = {
width = 0.8,
height = 0.8,
border = 'rounded',
- title = title,
+ title = win_title,
keymap = {
quit = 'q',
},
@@ -110,8 +121,8 @@ M.conf = {
-- theme = 'tokyonight',
db_path = '$HOME/.vim/dict/ultimate.db',
-
engine = {
+ youdao = {},
-- baidu = {
-- appid = '',
-- appPasswd = '',
@@ -121,7 +132,6 @@ M.conf = {
-- appPasswd = '',
-- },
},
-
-- TODO :
-- register word
-- history = {
@@ -147,75 +157,78 @@ M.setup = function(opts)
end
local engines = {}
+ local i = 1
for k, _ in pairs(conf.engine) do
- table.insert(engines, k)
+ engines[i] = k
+ i = i + 1
end
- conf.engines = engines
+ conf.engines = engines
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
+ ---@format disable
local new_command = api.nvim_create_user_command
-
- if vim.fn.executable('sqlite3') ~= 1 then
- error('Please check out sqlite3')
- end
-
- 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(get_mode().mode)
+ 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 hls = require('Trans.ui.theme')[conf.theme]
+
+ 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 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, 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
- 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)
- word = input
- end)
+ ---@diagnostic disable-next-line: param-type-mismatch
+ word = fn.input('请输入需要查询的单词:')
else
error('invalid mode: ' .. mode)
end
@@ -229,7 +242,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 +253,6 @@ M.translate = function(mode, view)
end
end
-
-M.augroup = vim.api.nvim_create_augroup('Trans', { clear = true })
+M.ns = api.nvim_create_namespace('Trans')
return M
diff --git a/lua/Trans/node.lua b/lua/Trans/node.lua
index 7c8b48c..81729df 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 ns = require('Trans').ns
+local add_hl = api.nvim_buf_add_highlight
+
+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 size = text.size
+ local width = opts.width
+ local spin = opts.spin or ' '
+
+ 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
+ return text
end,
}
diff --git a/lua/Trans/query/baidu.lua b/lua/Trans/query/baidu.lua
index d98c0ec..bf0e8ff 100644
--- a/lua/Trans/query/baidu.lua
+++ b/lua/Trans/query/baidu.lua
@@ -41,19 +41,25 @@ 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,
})
return result
end
+
+
+
+-- NOTE :free tts:
+-- https://zj.v.api.aa1.cn/api/baidu-01/?msg=我爱你&choose=0&su=100&yd=5
+-- 选择转音频的人物,女生1 输入0 | 女生2输入:5|男生1 输入:1|男生2 输入:2|男生3 输入:3
diff --git a/lua/Trans/query/offline.lua b/lua/Trans/query/offline.lua
index 703d84f..96fdd3b 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()
@@ -16,11 +16,10 @@ vim.api.nvim_create_autocmd('VimLeavePre', {
end
})
+
return function(word)
- local res = dict:select('stardict', {
- where = {
- word = word,
- },
+ local res = (dict:select('stardict', {
+ where = { word = word, },
keys = {
'word',
'phonetic',
@@ -33,6 +32,16 @@ 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,
+ }
+ end
+
+ return res
end
diff --git a/lua/Trans/query/youdao.lua b/lua/Trans/query/youdao.lua
index 43e7dfd..ac9fe66 100644
--- a/lua/Trans/query/youdao.lua
+++ b/lua/Trans/query/youdao.lua
@@ -1,48 +1,74 @@
-local youdao = require("Trans").conf.engine.youdao
-local appKey = youdao.appKey
-local appPasswd = youdao.appPasswd
-local uri = 'https://openapi.youdao.com/api'
-local salt = tostring(math.random(bit.rshift(1, 5)))
-
-
-local ok, curl = pcall(require, 'plenary.curl')
-if not ok then
- error('plenary not found')
-end
-
-
-local function get_field(word)
- local len = #word
- local curtime = tostring(os.time())
- local input = len > 20 and
- word:sub(1, 10) .. len .. word:sub(-10) or word
-
- -- sign=sha256(应用ID+input+salt+curtime+应用密钥);
- local hash = appKey .. input .. salt .. curtime .. appPasswd
- local sign = vim.fn.sha256(hash)
-
- return {
- q = word,
- from = 'auto',
- to = 'zh-CHS',
- signType = 'v3',
- appKey = appKey,
- salt = salt,
- curtime = curtime,
- sign = sign,
- }
-end
+local GET = require("Trans.util.curl").GET
return function(word)
- -- return result
- local field = get_field(word)
- local output = curl.post(uri, {
- body = field,
- })
- if output.exit == 0 and output.status == 200 then
- local result = vim.fn.json_decode(output.body)
- if result and result.errorCode == 0 then
- --- TODO :
+ local isEn = word:isEn()
+ local result = {}
+
+ local uri = ('https://v.api.aa1.cn/api/api-fanyi-yd/index.php?msg=%s&type=%d'):format(word, isEn and 2 or 1)
+ GET(uri, {
+ callback = function(str)
+ local ok, res = pcall(vim.json.decode, str)
+ if not ok or not res or not res.text or isEn and res.text:isEn() then
+ result[1] = false
+ return
+ end
+
+ result[1] = {
+ title = { word = word },
+ [isEn and 'translation' or 'definition'] = res.text,
+ }
+
+ if result.callback then
+ result.callback(result[1])
+ end
end
- end
+ })
+
+ return result
end
+
+-- local youdao = require("Trans").conf.engine.youdao
+-- local uri = 'https://openapi.youdao.com/api'
+-- local salt = tostring(math.random(bit.lshift(1, 15)))
+-- local appid = youdao.appid
+-- local appPasswd = youdao.appPasswd
+
+-- local post = require('Trans.util.curl').POST
+
+-- local function get_field(word)
+-- -- local to = isEn and 'zh-'
+-- local len = #word
+-- local curtime = tostring(os.time())
+-- local input = len > 20 and
+-- word:sub(1, 10) .. len .. word:sub(-10) or word
+
+-- -- sign=sha256(应用ID+input+salt+curtime+应用密钥);
+-- local hash = appid .. input .. salt .. curtime .. appPasswd
+-- local sign = vim.fn.sha256(hash)
+
+-- return {
+-- q = word,
+-- from = 'auto',
+-- to = 'zh-CHS',
+-- signType = 'v3',
+-- appKey = appid,
+-- salt = salt,
+-- curtime = curtime,
+-- sign = sign,
+-- }
+-- end
+
+-- return function(word)
+-- -- return result
+-- -- local field = get_field(word)
+-- -- local output = post(uri, {
+-- -- body = field,
+-- -- })
+
+-- -- if output.exit == 0 and output.status == 200 then
+-- -- local result = vim.fn.json_decode(output.body)
+-- -- if result and result.errorCode == 0 then
+-- -- --- TODO :
+-- -- end
+-- -- end
+-- end
diff --git a/lua/Trans/util/animation.lua b/lua/Trans/util/animation.lua
deleted file mode 100644
index 8e74d7c..0000000
--- a/lua/Trans/util/animation.lua
+++ /dev/null
@@ -1,55 +0,0 @@
-local display = function(self)
- local callback = self.callback or function()
-
- end
-
- local target = self.times
- if self.sync then
- if target then
- for i = 1, target do
- if self.run then
- self:frame(i)
- end
- end
-
- else
- while self.run do
- self:frame()
- end
- end
-
- callback()
- else
- local frame
- if target then
- 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
-
-
-return function(opts)
- opts.run = true
- opts.display = display
- return opts
-end
diff --git a/lua/Trans/util/curl.lua b/lua/Trans/util/curl.lua
index 50af442..0b54edf 100644
--- a/lua/Trans/util/curl.lua
+++ b/lua/Trans/util/curl.lua
@@ -12,9 +12,32 @@ local curl = {}
curl.GET = function(uri, opts)
--- TODO :
+ vim.validate {
+ uri = { uri, 's' },
+ opts = { opts, 't' }
+ }
+ local cmd = {'curl', '-s', ('"%s"'):format(uri)}
+ local callback = opts.callback
+
+ 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
+
curl.POST = function(uri, opts)
vim.validate {
uri = { uri, 's' },
@@ -23,7 +46,7 @@ curl.POST = function(uri, opts)
local callback = opts.callback
- local cmd = { 'curl', '-s', uri }
+ local cmd = { 'curl', '-s', ('"%s"'):format(uri) }
local size = 3
local function insert(...)
@@ -63,5 +86,4 @@ curl.POST = function(uri, opts)
vim.fn.jobstart(table.concat(cmd, ' '), option)
end
-
return curl
diff --git a/lua/Trans/util/display.lua b/lua/Trans/util/display.lua
new file mode 100644
index 0000000..b740f73
--- /dev/null
+++ b/lua/Trans/util/display.lua
@@ -0,0 +1,48 @@
+return function(opts)
+ local target = opts.times
+ opts.run = target ~= 0
+
+ ---@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
+
+ else
+ frame = function()
+ if opts.run then
+ opts:frame()
+ vim.defer_fn(frame, opts.interval)
+ else
+ do_task()
+ end
+ end
+ end
+ 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 3a00c71..ab19611 100644
--- a/lua/Trans/view/hover.lua
+++ b/lua/Trans/view/hover.lua
@@ -1,55 +1,58 @@
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
+local node = require('Trans.node')
+local it, t, f = node.item, node.text, node.format
--- content utility
-local node = require("Trans.node")
-local t = node.text
-local it = node.item
+local function handle_result(result)
+ local icon = conf.icon
+ local notfound = icon.notfound
+ local indent = ' '
-local m_indent = ' '
+ local word = result.title.word
+ if hover.auto_play then
+ string.play(word:isEn() and word or result.definition)
+ end
-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 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(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 +67,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 +84,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 +105,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 +130,266 @@ 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)
+ addtitle('中文翻译')
- for trs in vim.gsplit(m_result.translation, '\n', true) do
- m_content:addline(
- it(m_indent .. trs, 'TransTranslation')
+ 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 cmd_id
-local pin
-local next
-local action
-action = {
- pageup = function()
- m_window:normal('gg')
- end,
+ 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'
- pagedown = function()
- m_window:normal('G')
- end,
+ return require('Trans.window') {
+ col = col,
+ row = row,
+ buf = buffer,
+ relative = relative,
+ width = width,
+ height = height,
+ title = hover.title,
+ border = hover.border,
+ animation = hover.animation,
+ ns = require('Trans').ns,
+ }
+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)
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
+ local action = {
+ pageup = function()
+ buffer:normal('gg')
+ end,
+
+ pagedown = function()
+ buffer:normal('G')
+ end,
+
+ pin = function()
+ if lock then
+ error('请先关闭窗口')
+ else
+ lock = true
+ end
+ pcall(api.nvim_del_autocmd, cmd_id)
+ local width, height = win.width, win.height
+ local col = vim.o.columns - width - 3
+ local buf = buffer.bufnr
+ local run = win:try_close()
+ run(function()
+ local w, r = open_window {
+ width = width,
+ height = height,
+ relative = 'editor',
+ col = col,
}
- 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
+ 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 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)
+ local run = win:try_close()
+ run(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)
+ 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
+ for act, key in pairs(hover.keymap) do
+ set('n', key, action[act])
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)
- end,
+ callback = action.close,
})
end
end
+
+local function online_query(win, word)
+ 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)
+ return
+ end
+
+ for i = 1, size do
+ lists[i] = 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 s = '%s %s'
+ local width, height = hover.width, hover.height
+ local function waitting_result(this, times)
+ for i = 1, size do
+ local res = lists[i][1]
+ if res then
+ buffer:wipe()
+ win:set_width(width)
+ handle_result(res)
+ height = math.min(height, buffer:height(width))
+
+ win:expand {
+ field = 'height',
+ target = height,
+ }
+ this.run = false
+ return
+ elseif res == false then
+ lists:remove(i)
+ size = size - 1
+ end
+ end
+
+ if size == 0 or times == win_width then
+ buffer:addline(error_line, 1)
+ this.run = false
+ else
+ buffer:addline(it(s:format(spinner[times % range + 1], cell:rep(times)), 'MoreMsg'), 1)
+ end
+ end
+
+ buffer:set('modifiable', true)
+ local run = require('Trans.util.display') {
+ times = win_width,
+ interval = interval,
+ frame = waitting_result,
+ }
+
+ run(function()
+ buffer:set('modifiable', false)
+ end)
+end
+
+---处理不同hover模式的窗口
+---@param word string 待查询的单词
+return function(word)
+ buffer:init()
+ local result = require('Trans.query.offline')(word)
+
+ if result then
+ handle_result(result)
+ local width = hover.width
+ local win, run = open_window {
+ width = width,
+ height = math.min(buffer:height(width), hover.height)
+ }
+ run(function()
+ win:set('wrap', true)
+ handle_keymap(win, word)
+ end)
+
+ else
+ local win, run = open_window {
+ width = error_msg:width(),
+ height = 1,
+ }
+
+ 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 4eedb42..cd85546 100644
--- a/lua/Trans/window.lua
+++ b/lua/Trans/window.lua
@@ -1,248 +1,221 @@
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
+---@class win
+---@field winid integer window handle
+---@field width integer
+---@field height integer
+---@field ns integer namespace for highlight
+---@field animation table window animation
+---@field buf buf buffer for attached
+local window = {}
+
+---Change window attached buffer
+---@param buf buf
+function window:set_buf(buf)
+ api.nvim_win_set_buf(self.winid, buf.bufnr)
+ self.buf = buf
+end
+
+---Check window valid
+---@return boolean
+function window:is_valid()
+ return api.nvim_win_is_valid(self.winid)
+end
+
+---Set window option
+---@param option string option name
+---@param value any
+function window:set(option, value)
+ api.nvim_win_set_option(self.winid, option, value)
+end
+
+---@param name string option name
+---@return any
+function window:option(name)
+ return api.nvim_win_get_option(self.winid, name)
+end
+
+---@param height integer
+function window:set_height(height)
+ api.nvim_win_set_height(self.winid, height)
+ self.height = height
+end
+
+---@param width integer
+function window:set_width(width)
+ api.nvim_win_set_width(self.winid, width)
+ self.width = width
+end
+
+---Expand window [width | height] value
+---@param opts table 窗口的配置
+---|'field'string [width | height]
+---|'target'integer
+---@return function
+function window:expand(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 interval = opts.interval or self.animation.interval
+ local method = api['nvim_win_set_' .. field]
+
+ local winid = self.winid
+ local frame = target > cur and function(_, cur_times)
+ method(winid, cur + cur_times)
+ end or function(_, cur_times)
+ method(winid, cur - cur_times)
+ end
+
+ local run = display {
+ times = times,
+ frame = frame,
+ interval = interval,
+ }
+
+ run(function()
+ self:set('wrap', wrap)
+ self[field] = target
+ self:unlock()
+ end)
+ return run
+end
+
+---Close window
+---@return function run run until close done
+function window:try_close()
+ local field = ({
+ slid = 'width',
+ fold = 'height',
+ })[self.animation.close]
+
+ --- 播放动画
+ local run = self:expand {
+ field = field,
+ target = 1,
+ }
+ run(function()
+ api.nvim_win_close(self.winid, true)
+ end)
+ return run
+end
+
+---lock window [open | close] operation
+function window:lock()
+ while self.busy do
vim.wait(50)
end
- busy = true
+ self.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[] 窗口内容的对象数组
+function window:unlock()
+ self.busy = false
+end
----@type window
-local window = {
- set = function(self, option, value)
- api.nvim_win_set_option(self.winid, option, value)
- end,
+---设置窗口本地的高亮组
+---@param name string 高亮组的名称
+---@param opts table 高亮选项
+function window:set_hl(name, opts)
+ api.nvim_set_hl(self.ns, name, opts)
+end
- set_height = function(self, height)
- api.nvim_win_set_height(self.winid, height)
- self.height = height
- end,
-
- set_width = function(self, width)
- api.nvim_win_set_width(self.winid, width)
- 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()
- 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
- self:set('wrap', wrap)
- if opts.callback then
- opts.callback()
- end
- end
-
- lock()
- if animation then
- local interval = self.animation.interval
- local field = ({
- fold = 'height',
- slid = 'width',
- })[animation]
-
- local method = api['nvim_win_set_' .. field]
- local winid = self.winid
- new_animation({
- 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)
-
- 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
-
- 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()
- 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
- end
-
- self.winid = api.nvim_open_win(self.bufnr, entry, self.config)
- self:open(opt)
- end,
-
- set_hl = function(self, name, opts)
- api.nvim_set_hl(self.hl, 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]
- end,
-}
+---buffer:addline() helper function
+---@param node table
+---@return table node formatted node
+function window:center(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
+---@private
window.__index = window
+---@class win_opts
+---@field buf buf buffer for attached
+---@field height integer
+---@field width integer
+---@field col integer
+---@field row integer
+---@field border string
+---@field title string | nil | table
+---@field relative string
+---@field ns integer namespace for highlight
+---@field zindex? integer
+---@field enter? boolean cursor should [enter] window
+---@field animation table window animation
+---window constructor
+---@param opts win_opts
+---@return table
+---@return function
+return function(opts)
+ assert(type(opts) == 'table')
+ 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 enter = opts.enter or false
+ local height = opts.height
+ local border = opts.border
+ local zindex = opts.zindex
+ local relative = opts.relative
+ local animation = opts.animation
----窗口对象的构造器
----@param entry boolean 光标初始化时是否应该进入窗口
----@param option table 需要设置的选项
----@return window win
----@nodiscard
-return function(entry, option)
- vim.validate {
- entry = { entry, 'b' },
- option = { option, 't' },
- }
+ local open = animation.open
- 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,
+ height = win_opt.height,
+ width = win_opt.width,
+ animation = animation,
+ winid = api.nvim_open_win(buf.bufnr, enter, win_opt),
+ }, window)
- ---@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
+
+ return win, win:expand {
+ field = field,
+ target = opts[field],
+ }
end