commit
f5732f58d7
3
.gitignore
vendored
3
.gitignore
vendored
@ -1,6 +1,7 @@
|
|||||||
lua/Trans/util/
|
lua/Trans/util/
|
||||||
lua/Trans/test/
|
lua/Trans/test/
|
||||||
note/
|
note/
|
||||||
go/
|
|
||||||
demo.mp4
|
demo.mp4
|
||||||
screenshot.gif
|
screenshot.gif
|
||||||
|
tts/node_modules/
|
||||||
|
tts/package-lock.json
|
||||||
|
54
README.md
54
README.md
@ -103,6 +103,12 @@ use {
|
|||||||
|
|
||||||
> 后续会增加 `healthcheck` 进行检查
|
> 后续会增加 `healthcheck` 进行检查
|
||||||
|
|
||||||
|
- **`auto_play`** 使用步骤:
|
||||||
|
- 需要确保安装了`nodejs`
|
||||||
|
- 进入插件的`tts`目录运行`npm install`
|
||||||
|
> 如果`install.sh`运行正常则自动安装,如果安装失败,请尝试手动安装
|
||||||
|
- linux 用户需要额外安装以下依赖:
|
||||||
|
> sudo apt-get install festival festvox-kallpc16k
|
||||||
|
|
||||||
## 配置
|
## 配置
|
||||||
```lua
|
```lua
|
||||||
@ -125,14 +131,24 @@ require'Trans'.setup {
|
|||||||
-- TODO :
|
-- TODO :
|
||||||
pageup = '[[',
|
pageup = '[[',
|
||||||
pagedown = ']]',
|
pagedown = ']]',
|
||||||
pin = '_', -- 将窗口固定在右上角, 参考demo
|
pin = '<leader>[',
|
||||||
close = '+',
|
close = '<leader>]',
|
||||||
|
toggle_entry = '<leader>;',
|
||||||
|
play = '_',
|
||||||
},
|
},
|
||||||
animation = {
|
animation = {
|
||||||
open = 'slid', -- 可选的样式: slid , fold
|
-- open = 'fold',
|
||||||
|
-- close = 'fold',
|
||||||
|
open = 'slid',
|
||||||
close = 'slid',
|
close = 'slid',
|
||||||
interval = 12, -- 动画的帧间隔
|
interval = 12,
|
||||||
}
|
},
|
||||||
|
auto_close_events = {
|
||||||
|
'InsertEnter',
|
||||||
|
'CursorMoved',
|
||||||
|
'BufLeave',
|
||||||
|
},
|
||||||
|
auto_play = true, -- WARN : 请阅读说明
|
||||||
},
|
},
|
||||||
float = {
|
float = {
|
||||||
width = 0.8,
|
width = 0.8,
|
||||||
@ -167,14 +183,15 @@ require'Trans'.setup {
|
|||||||
},
|
},
|
||||||
icon = {
|
icon = {
|
||||||
star = '',
|
star = '',
|
||||||
notfound = '❔',
|
-- notfound = '❔',
|
||||||
yes = '✔️',
|
notfound = ' ',
|
||||||
no = '❌'
|
yes = ' ',
|
||||||
|
no = ''
|
||||||
|
-- yes = '✔️',
|
||||||
|
-- no = '❌'
|
||||||
-- star = '⭐',
|
-- star = '⭐',
|
||||||
-- notfound = '',
|
|
||||||
-- yes = '',
|
|
||||||
-- no = ''
|
|
||||||
},
|
},
|
||||||
|
|
||||||
db_path = '$HOME/.vim/dict/ultimate.db',
|
db_path = '$HOME/.vim/dict/ultimate.db',
|
||||||
|
|
||||||
-- TODO :
|
-- TODO :
|
||||||
@ -264,11 +281,10 @@ vim.keymap.set('n', 'mi', '<Cmd>TranslateInput<CR>')
|
|||||||
- [T.vim](https://github.com/sicong-li/T.vim) 灵感来源
|
- [T.vim](https://github.com/sicong-li/T.vim) 灵感来源
|
||||||
|
|
||||||
## 待办 (画大饼)
|
## 待办 (画大饼)
|
||||||
- ~~移动光标自动关闭窗口~~
|
- [x] 多风格样式查询
|
||||||
- 多风格样式
|
- [x] 重新录制屏幕截图示例
|
||||||
- 历史查询结果保存
|
- [ ] 历史查询结果保存
|
||||||
- 在线多引擎异步查询
|
- [ ] 在线多引擎异步查询
|
||||||
- 快捷键定义
|
- [ ] 快捷键定义
|
||||||
- 自动读音
|
- [ ] 自动读音
|
||||||
- `句子翻译` | `中翻英` 的支持
|
- [ ] `句子翻译` | `中翻英` 的支持
|
||||||
- 重新录制屏幕截图示例
|
|
||||||
|
@ -15,3 +15,4 @@ wget https://github.com/skywind3000/ECDICT-ultimate/releases/download/1.0.0/ecdi
|
|||||||
unzip /tmp/dict.zip -d $HOME/.vim/dict
|
unzip /tmp/dict.zip -d $HOME/.vim/dict
|
||||||
|
|
||||||
rm -rf /tmp/dict.zip
|
rm -rf /tmp/dict.zip
|
||||||
|
cd ./tts/ && npm install
|
||||||
|
@ -41,15 +41,16 @@ local content = {
|
|||||||
self.size = 0
|
self.size = 0
|
||||||
end,
|
end,
|
||||||
|
|
||||||
attach = function(self)
|
---将内容连接上对应的窗口
|
||||||
|
---@param self table content对象
|
||||||
|
---@param offset integer 起始行
|
||||||
|
attach = function(self, offset)
|
||||||
if self.size == 0 then
|
if self.size == 0 then
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
self.window:bufset('modifiable', true)
|
self.window:bufset('modifiable', true)
|
||||||
local window = self.window
|
local window = self.window
|
||||||
local offset = self.offset
|
|
||||||
|
|
||||||
api.nvim_buf_set_lines(window.bufnr, offset, offset + 1, true, self.lines)
|
api.nvim_buf_set_lines(window.bufnr, offset, offset + 1, true, self.lines)
|
||||||
|
|
||||||
for _, hl in ipairs(self.highlights) do
|
for _, hl in ipairs(self.highlights) do
|
||||||
@ -58,8 +59,9 @@ local content = {
|
|||||||
self.window:bufset('modifiable', false)
|
self.window:bufset('modifiable', false)
|
||||||
end,
|
end,
|
||||||
|
|
||||||
actual_height = function(self)
|
actual_height = function(self, wrap)
|
||||||
if self.window:option('wrap') then
|
wrap = wrap or self.window:option('wrap')
|
||||||
|
if wrap then
|
||||||
local height = 0
|
local height = 0
|
||||||
local width = self.window.width
|
local width = self.window.width
|
||||||
local lines = self.lines
|
local lines = self.lines
|
||||||
@ -67,6 +69,7 @@ local content = {
|
|||||||
height = height + math.max(1, (math.ceil(lines[i]:width() / width)))
|
height = height + math.max(1, (math.ceil(lines[i]:width() / width)))
|
||||||
end
|
end
|
||||||
return height
|
return height
|
||||||
|
|
||||||
else
|
else
|
||||||
return self.size
|
return self.size
|
||||||
end
|
end
|
||||||
@ -151,13 +154,12 @@ local content = {
|
|||||||
---content的构造函数
|
---content的构造函数
|
||||||
---@param window table 链接的窗口
|
---@param window table 链接的窗口
|
||||||
---@return table 构造好的content
|
---@return table 构造好的content
|
||||||
return function(window, offset)
|
return function(window)
|
||||||
vim.validate {
|
vim.validate {
|
||||||
window = { window, 't' },
|
window = { window, 't' },
|
||||||
}
|
}
|
||||||
return setmetatable({
|
return setmetatable({
|
||||||
modifiable = true,
|
modifiable = true,
|
||||||
offset = offset or 0,
|
|
||||||
window = window,
|
window = window,
|
||||||
size = 0,
|
size = 0,
|
||||||
hl_size = 0,
|
hl_size = 0,
|
||||||
|
@ -19,9 +19,10 @@ M.conf = {
|
|||||||
-- TODO :
|
-- TODO :
|
||||||
pageup = '[[',
|
pageup = '[[',
|
||||||
pagedown = ']]',
|
pagedown = ']]',
|
||||||
pin = '+',
|
pin = '<leader>[',
|
||||||
close = '_',
|
close = '<leader>]',
|
||||||
toggle_entry = '--',
|
toggle_entry = '<leader>;',
|
||||||
|
play = '_',
|
||||||
},
|
},
|
||||||
animation = {
|
animation = {
|
||||||
-- open = 'fold',
|
-- open = 'fold',
|
||||||
@ -35,6 +36,7 @@ M.conf = {
|
|||||||
'CursorMoved',
|
'CursorMoved',
|
||||||
'BufLeave',
|
'BufLeave',
|
||||||
},
|
},
|
||||||
|
auto_play = true,
|
||||||
},
|
},
|
||||||
float = {
|
float = {
|
||||||
width = 0.8,
|
width = 0.8,
|
||||||
@ -69,14 +71,15 @@ M.conf = {
|
|||||||
},
|
},
|
||||||
icon = {
|
icon = {
|
||||||
star = '',
|
star = '',
|
||||||
notfound = '❔',
|
-- notfound = '❔',
|
||||||
yes = '✔️',
|
notfound = ' ',
|
||||||
no = '❌'
|
yes = ' ',
|
||||||
|
no = ''
|
||||||
|
-- yes = '✔️',
|
||||||
|
-- no = '❌'
|
||||||
-- star = '⭐',
|
-- star = '⭐',
|
||||||
-- notfound = '',
|
|
||||||
-- yes = '',
|
|
||||||
-- no = ''
|
|
||||||
},
|
},
|
||||||
|
|
||||||
db_path = '$HOME/.vim/dict/ultimate.db',
|
db_path = '$HOME/.vim/dict/ultimate.db',
|
||||||
|
|
||||||
-- TODO :
|
-- TODO :
|
||||||
|
@ -2,6 +2,7 @@ if vim.fn.executable('sqlite3') ~= 1 then
|
|||||||
error('Please check out sqlite3')
|
error('Please check out sqlite3')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
vim.api.nvim_create_user_command('Translate', function()
|
vim.api.nvim_create_user_command('Translate', function()
|
||||||
require("Trans").translate()
|
require("Trans").translate()
|
||||||
end, { desc = ' 单词翻译', })
|
end, { desc = ' 单词翻译', })
|
||||||
|
@ -31,6 +31,7 @@ local function get_word(mode)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
local function translate(mode, view)
|
local function translate(mode, view)
|
||||||
vim.validate {
|
vim.validate {
|
||||||
mode = { mode, 's', true },
|
mode = { mode, 's', true },
|
||||||
|
@ -2,7 +2,7 @@ local m_window
|
|||||||
local m_result
|
local m_result
|
||||||
|
|
||||||
local function set_title()
|
local function set_title()
|
||||||
local title = m_window.title
|
local title = m_window.contents[1]
|
||||||
local github = 'https://github.com/JuanZoran/Trans.nvim'
|
local github = 'https://github.com/JuanZoran/Trans.nvim'
|
||||||
|
|
||||||
-- TODO :config this
|
-- TODO :config this
|
||||||
@ -10,7 +10,7 @@ local function set_title()
|
|||||||
end
|
end
|
||||||
|
|
||||||
local action = {
|
local action = {
|
||||||
quit = function ()
|
quit = function()
|
||||||
m_window:try_close()
|
m_window:try_close()
|
||||||
end,
|
end,
|
||||||
}
|
}
|
||||||
|
@ -186,6 +186,8 @@ end
|
|||||||
|
|
||||||
local action
|
local action
|
||||||
local next
|
local next
|
||||||
|
local _word
|
||||||
|
|
||||||
action = {
|
action = {
|
||||||
pageup = function()
|
pageup = function()
|
||||||
m_window:normal('gg')
|
m_window:normal('gg')
|
||||||
@ -251,7 +253,12 @@ action = {
|
|||||||
else
|
else
|
||||||
vim.keymap.del('n', conf.hover.keymap.toggle_entry, { buffer = true })
|
vim.keymap.del('n', conf.hover.keymap.toggle_entry, { buffer = true })
|
||||||
end
|
end
|
||||||
end
|
end,
|
||||||
|
|
||||||
|
play = function()
|
||||||
|
local file = debug.getinfo(1, "S").source:sub(2):match('(.*)lua/') .. 'tts/say.js'
|
||||||
|
vim.fn.jobstart('node ' .. file .. ' ' .. _word)
|
||||||
|
end,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -259,10 +266,14 @@ return function(word)
|
|||||||
vim.validate {
|
vim.validate {
|
||||||
word = { word, 's' },
|
word = { word, 's' },
|
||||||
}
|
}
|
||||||
|
_word = word
|
||||||
-- 目前只处理了本地数据库的查询
|
-- 目前只处理了本地数据库的查询
|
||||||
m_result = require('Trans.query.offline')(word)
|
m_result = require('Trans.query.offline')(word)
|
||||||
local hover = conf.hover
|
local hover = conf.hover
|
||||||
|
if hover.auto_play then
|
||||||
|
action.play()
|
||||||
|
end
|
||||||
|
|
||||||
local opt = {
|
local opt = {
|
||||||
relative = 'cursor',
|
relative = 'cursor',
|
||||||
width = hover.width,
|
width = hover.width,
|
||||||
@ -275,7 +286,7 @@ return function(word)
|
|||||||
|
|
||||||
m_window = require("Trans.window")(false, opt)
|
m_window = require("Trans.window")(false, opt)
|
||||||
m_window.animation = hover.animation
|
m_window.animation = hover.animation
|
||||||
m_content = m_window.content
|
m_content = m_window.contents[1]
|
||||||
|
|
||||||
if m_result then
|
if m_result then
|
||||||
for _, field in ipairs(conf.order) do
|
for _, field in ipairs(conf.order) do
|
||||||
@ -283,9 +294,16 @@ return function(word)
|
|||||||
end
|
end
|
||||||
else
|
else
|
||||||
process.failed()
|
process.failed()
|
||||||
|
m_window:set_width(m_content.lines[1]:width())
|
||||||
|
end
|
||||||
|
|
||||||
|
m_window:draw()
|
||||||
|
|
||||||
|
local height = m_content:actual_height(true)
|
||||||
|
if height < m_window.height then
|
||||||
|
m_window:set_height(height)
|
||||||
end
|
end
|
||||||
|
|
||||||
m_window:draw(true)
|
|
||||||
m_window:open(function()
|
m_window:open(function()
|
||||||
m_window:set('wrap', true)
|
m_window:set('wrap', true)
|
||||||
end)
|
end)
|
||||||
@ -304,7 +322,9 @@ return function(word)
|
|||||||
end,
|
end,
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if m_result then
|
||||||
for act, key in pairs(hover.keymap) do
|
for act, key in pairs(hover.keymap) do
|
||||||
vim.keymap.set('n', key, action[act], { buffer = true, silent = true })
|
vim.keymap.set('n', key, action[act], { buffer = true, silent = true })
|
||||||
end
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
@ -83,23 +83,12 @@ local window = {
|
|||||||
|
|
||||||
---**第一次**绘制窗口的内容
|
---**第一次**绘制窗口的内容
|
||||||
---@param self table 窗口的对象
|
---@param self table 窗口的对象
|
||||||
---@param adjust boolean 是否需要调整窗口的高度和宽度 (只有窗口只有一行时才会调整宽度)
|
draw = function(self)
|
||||||
draw = function(self, adjust)
|
|
||||||
-- TODO :
|
-- TODO :
|
||||||
if self.title then
|
local offset = 0
|
||||||
self.title:attach()
|
for _, content in ipairs(self.contents) do
|
||||||
end
|
content:attach(offset)
|
||||||
self.content:attach()
|
offset = offset + content.size
|
||||||
|
|
||||||
if adjust then
|
|
||||||
local height = self.content:actual_height() + self.title:actual_height()
|
|
||||||
if self.height > height then
|
|
||||||
self:set_height(height)
|
|
||||||
end
|
|
||||||
|
|
||||||
if self.content.size == 1 and self.title.size == 0 then
|
|
||||||
self:set_width(self.content.lines[1]:width())
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end,
|
end,
|
||||||
|
|
||||||
@ -207,20 +196,18 @@ local window = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
---@class window
|
---@class window
|
||||||
---@field title table 窗口不变的title对象,载入后不可修改
|
|
||||||
---@field winid integer 窗口的handle
|
---@field winid integer 窗口的handle
|
||||||
---@field bufnr integer 窗口对应buffer的handle
|
---@field bufnr integer 窗口对应buffer的handle
|
||||||
---@field content table 窗口内容的对象, 和title一样是content类
|
|
||||||
---@field width integer 窗口当前的宽度
|
---@field width integer 窗口当前的宽度
|
||||||
---@field height integer 窗口当前的高度
|
---@field height integer 窗口当前的高度
|
||||||
---@field hl integer 窗口highlight的namespace
|
---@field hl integer 窗口highlight的namespace
|
||||||
|
---@field contents table[] 窗口内容的对象数组
|
||||||
|
|
||||||
|
|
||||||
---窗口对象的构造器
|
---窗口对象的构造器
|
||||||
---@param entry boolean 光标初始化时是否应该进入窗口
|
---@param entry boolean 光标初始化时是否应该进入窗口
|
||||||
---@param option table 需要设置的选项
|
---@param option table 需要设置的选项
|
||||||
---@return window
|
---@return window win
|
||||||
---@nodiscard
|
---@nodiscard
|
||||||
return function(entry, option)
|
return function(entry, option)
|
||||||
vim.validate {
|
vim.validate {
|
||||||
@ -249,34 +236,24 @@ return function(entry, option)
|
|||||||
local bufnr = api.nvim_create_buf(false, true)
|
local bufnr = api.nvim_create_buf(false, true)
|
||||||
local winid = api.nvim_open_win(bufnr, entry, opt)
|
local winid = api.nvim_open_win(bufnr, entry, opt)
|
||||||
|
|
||||||
local win = setmetatable({
|
|
||||||
|
local win
|
||||||
|
win = {
|
||||||
winid = winid,
|
winid = winid,
|
||||||
bufnr = bufnr,
|
bufnr = bufnr,
|
||||||
title = nil,
|
|
||||||
content = nil,
|
|
||||||
width = opt.width,
|
width = opt.width,
|
||||||
height = opt.height,
|
height = opt.height,
|
||||||
hl = api.nvim_create_namespace('TransWinHl'),
|
hl = api.nvim_create_namespace('TransWinHl'),
|
||||||
},
|
contents = setmetatable({}, {
|
||||||
{ __index = function(tbl, key)
|
__index = function(self, key)
|
||||||
if key == 'content' then
|
assert(type(key) == 'number')
|
||||||
if tbl.title then
|
self[key] = require('Trans.content')(win)
|
||||||
tbl.content = require('Trans.content')(tbl, tbl.title.size)
|
return self[key]
|
||||||
tbl.title.modifiable = false
|
|
||||||
else
|
|
||||||
tbl.content = require('Trans.content')(tbl)
|
|
||||||
end
|
end
|
||||||
return tbl.content
|
})
|
||||||
|
}
|
||||||
elseif key == 'title' then
|
|
||||||
tbl.title = require('Trans.content')(tbl, 0)
|
|
||||||
return tbl.title
|
|
||||||
|
|
||||||
else
|
|
||||||
return window[key]
|
|
||||||
end
|
|
||||||
end })
|
|
||||||
|
|
||||||
|
setmetatable(win, { __index = window })
|
||||||
win:set('winhl', 'Normal:TransWin,FloatBorder:TransBorder')
|
win:set('winhl', 'Normal:TransWin,FloatBorder:TransBorder')
|
||||||
win:bufset('filetype', 'Trans')
|
win:bufset('filetype', 'Trans')
|
||||||
win:bufset('buftype', 'nofile')
|
win:bufset('buftype', 'nofile')
|
||||||
|
5
tts/package.json
Normal file
5
tts/package.json
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"dependencies": {
|
||||||
|
"say": "^0.16.0"
|
||||||
|
}
|
||||||
|
}
|
6
tts/say.js
Normal file
6
tts/say.js
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
const say = require('say')
|
||||||
|
|
||||||
|
word = process.argv
|
||||||
|
|
||||||
|
// console.log(word)
|
||||||
|
say.speak(word.slice(2))
|
Loading…
x
Reference in New Issue
Block a user