Compare commits

...

15 Commits

Author SHA1 Message Date
Gabe Yuan
47f9635b10 v1.8.3 2024-02-26 16:42:36 +08:00
Gabe Yuan
68088f5e17 fix: remove excess line breaks 2024-02-26 16:35:29 +08:00
Gabe Yuan
77a37cc6df fix: remove excess line breaks 2024-02-26 16:34:53 +08:00
Gabe Yuan
420e59bf81 fix: translate text 2024-02-22 22:57:34 +08:00
Gabe Yuan
dbc5135301 fix: update default blacklist 2024-02-22 17:44:17 +08:00
Gabe Yuan
8c7d6bb552 fix: remove postMessage from iframe 2024-02-22 17:26:22 +08:00
Gabe Yuan
2b5c1952c0 fix: update default blacklist 2024-02-22 11:45:41 +08:00
Gabe Yuan
85a82618e5 fix: contextMenus text 2024-02-21 17:01:11 +08:00
Gabe Yuan
0280ac34c3 fix: dict voice not exist 2024-02-21 15:47:14 +08:00
Gabe Yuan
439900154b fix: dict voice not exist 2024-02-21 15:41:27 +08:00
Gabe Yuan
4a6e902684 fix: update readme 2024-02-06 10:22:56 +08:00
Gabe Yuan
71bbd2e54a v1.8.1 2024-02-06 10:09:33 +08:00
Gabe Yuan
3083d8e147 feat: context menu type 2024-02-05 11:28:34 +08:00
Gabe Yuan
e74883e9c2 fix: contextMenus: duplicate id err 2024-02-05 10:51:42 +08:00
Gabe Yuan
0816a9d167 fix: add BLOCKQUOTE to webfix 2024-02-05 10:02:30 +08:00
18 changed files with 118 additions and 115 deletions

2
.env
View File

@@ -2,7 +2,7 @@ GENERATE_SOURCEMAP=false
REACT_APP_NAME=KISS Translator REACT_APP_NAME=KISS Translator
REACT_APP_NAME_CN=简约翻译 REACT_APP_NAME_CN=简约翻译
REACT_APP_VERSION=1.8.1 REACT_APP_VERSION=1.8.3
REACT_APP_HOMEPAGE=https://github.com/fishjar/kiss-translator REACT_APP_HOMEPAGE=https://github.com/fishjar/kiss-translator

View File

@@ -44,7 +44,7 @@ A simple, open source [bilingual translation extension & Greasemonkey script](ht
> - Grease Monkey script will encounter more usage problems (cross domain issues, script conflicts, etc.) > - Grease Monkey script will encounter more usage problems (cross domain issues, script conflicts, etc.)
- [x] Browser extension - [x] Browser extension
- [x] Chrome [Installation address](https://chrome.google.com/webstore/detail/kiss-translator/bdiifdefkgmcblbcghdlonllpjhhjgof?hl=zh-CN) - [x] Chrome/Kiwi [Installation address](https://chrome.google.com/webstore/detail/kiss-translator/bdiifdefkgmcblbcghdlonllpjhhjgof?hl=zh-CN)
- [x] Edge [Installation address](https://microsoftedge.microsoft.com/addons/detail/%E7%AE%80%E7%BA%A6%E7%BF%BB%E8%AF%91/jemckldkclkinpjighnoilpbldbdmmlh?hl=zh-CN) - [x] Edge [Installation address](https://microsoftedge.microsoft.com/addons/detail/%E7%AE%80%E7%BA%A6%E7%BF%BB%E8%AF%91/jemckldkclkinpjighnoilpbldbdmmlh?hl=zh-CN)
- [x] Firefox [Installation address](https://addons.mozilla.org/zh-CN/firefox/addon/kiss-translator/) - [x] Firefox [Installation address](https://addons.mozilla.org/zh-CN/firefox/addon/kiss-translator/)
- [ ] Safari - [ ] Safari

View File

@@ -29,7 +29,7 @@
- [x] 自定义专业术语 - [x] 自定义专业术语
- [x] 自定义译文样式 - [x] 自定义译文样式
- [x] 自定义快捷键 - [x] 自定义快捷键
- `Alt+Q`翻译 - `Alt+Q` 启翻译
- `Alt+C` 切换样式 - `Alt+C` 切换样式
- `Alt+K` 打开设置弹窗 - `Alt+K` 打开设置弹窗
- `Alt+S` 打开翻译弹窗/翻译选中文字 - `Alt+S` 打开翻译弹窗/翻译选中文字
@@ -44,7 +44,7 @@
> - 油猴脚本会遇到更多使用上的问题(跨域问题、脚本冲突等) > - 油猴脚本会遇到更多使用上的问题(跨域问题、脚本冲突等)
- [x] 浏览器扩展 - [x] 浏览器扩展
- [x] Chrome [安装地址](https://chrome.google.com/webstore/detail/kiss-translator/bdiifdefkgmcblbcghdlonllpjhhjgof?hl=zh-CN) - [x] Chrome/Kiwi [安装地址](https://chrome.google.com/webstore/detail/kiss-translator/bdiifdefkgmcblbcghdlonllpjhhjgof?hl=zh-CN)
- [x] Edge [安装地址](https://microsoftedge.microsoft.com/addons/detail/%E7%AE%80%E7%BA%A6%E7%BF%BB%E8%AF%91/jemckldkclkinpjighnoilpbldbdmmlh?hl=zh-CN) - [x] Edge [安装地址](https://microsoftedge.microsoft.com/addons/detail/%E7%AE%80%E7%BA%A6%E7%BF%BB%E8%AF%91/jemckldkclkinpjighnoilpbldbdmmlh?hl=zh-CN)
- [x] Firefox [安装地址](https://addons.mozilla.org/zh-CN/firefox/addon/kiss-translator/) - [x] Firefox [安装地址](https://addons.mozilla.org/zh-CN/firefox/addon/kiss-translator/)
- [ ] Safari - [ ] Safari

View File

@@ -1,7 +1,7 @@
{ {
"name": "kiss-translator", "name": "kiss-translator",
"description": "A minimalist bilingual translation Extension & Greasemonkey Script", "description": "A minimalist bilingual translation Extension & Greasemonkey Script",
"version": "1.8.1", "version": "1.8.3",
"author": "Gabe<yugang2002@gmail.com>", "author": "Gabe<yugang2002@gmail.com>",
"private": true, "private": true,
"dependencies": { "dependencies": {

View File

@@ -6,7 +6,7 @@
"message": "一个简约的双语对照翻译扩展 & 油猴脚本" "message": "一个简约的双语对照翻译扩展 & 油猴脚本"
}, },
"toggle_translate": { "toggle_translate": {
"message": "启翻译" "message": "启翻译"
}, },
"toggle_style": { "toggle_style": {
"message": "切换样式" "message": "切换样式"

View File

@@ -2,7 +2,7 @@
"manifest_version": 2, "manifest_version": 2,
"name": "__MSG_app_name__", "name": "__MSG_app_name__",
"description": "__MSG_app_description__", "description": "__MSG_app_description__",
"version": "1.8.1", "version": "1.8.3",
"default_locale": "en", "default_locale": "en",
"author": "Gabe<yugang2002@gmail.com>", "author": "Gabe<yugang2002@gmail.com>",
"homepage_url": "https://github.com/fishjar/kiss-translator", "homepage_url": "https://github.com/fishjar/kiss-translator",

View File

@@ -2,7 +2,7 @@
"manifest_version": 3, "manifest_version": 3,
"name": "__MSG_app_name__", "name": "__MSG_app_name__",
"description": "__MSG_app_description__", "description": "__MSG_app_description__",
"version": "1.8.1", "version": "1.8.3",
"default_locale": "en", "default_locale": "en",
"author": "Gabe<yugang2002@gmail.com>", "author": "Gabe<yugang2002@gmail.com>",
"homepage_url": "https://github.com/fishjar/kiss-translator", "homepage_url": "https://github.com/fishjar/kiss-translator",

View File

@@ -196,7 +196,7 @@ export const apiTranslate = async ({
break; break;
case OPT_TRANS_GEMINI: case OPT_TRANS_GEMINI:
trText = res?.candidates trText = res?.candidates
?.map((item) => item.content.parts.map((item) => item.text).join(" ")) ?.map((item) => item.content?.parts.map((item) => item.text).join(" "))
.join(" "); .join(" ");
isSame = text === trText; isSame = text === trText;
break; break;

View File

@@ -28,7 +28,23 @@ globalThis.ContextType = "BACKGROUND";
/** /**
* 添加右键菜单 * 添加右键菜单
*/ */
function addContextMenus() { async function addContextMenus(contextMenuType = 1) {
// 添加前先删除,避免重复ID的错误
try {
await browser.contextMenus.removeAll();
} catch (err) {
//
}
switch (contextMenuType) {
case 1:
browser.contextMenus.create({
id: CMD_TOGGLE_TRANSLATE,
title: browser.i18n.getMessage("app_name"),
contexts: ["page", "selection"],
});
break;
case 2:
browser.contextMenus.create({ browser.contextMenus.create({
id: CMD_TOGGLE_TRANSLATE, id: CMD_TOGGLE_TRANSLATE,
title: browser.i18n.getMessage("toggle_translate"), title: browser.i18n.getMessage("toggle_translate"),
@@ -54,13 +70,9 @@ function addContextMenus() {
title: browser.i18n.getMessage("open_options"), title: browser.i18n.getMessage("open_options"),
contexts: ["page", "selection"], contexts: ["page", "selection"],
}); });
break;
default:
} }
/**
* 清除右键菜单
*/
function removeContextMenus() {
browser.contextMenus.removeAll();
} }
/** /**
@@ -80,11 +92,8 @@ browser.runtime.onStartup.addListener(async () => {
// 同步数据 // 同步数据
await trySyncSettingAndRules(); await trySyncSettingAndRules();
const { const { clearCache, contextMenuType, subrulesList } =
clearCache, await getSettingWithDefault();
contextMenus = true,
subrulesList,
} = await getSettingWithDefault();
// 清除缓存 // 清除缓存
if (clearCache) { if (clearCache) {
@@ -92,11 +101,8 @@ browser.runtime.onStartup.addListener(async () => {
} }
// 右键菜单 // 右键菜单
if (contextMenus) { // firefox重启后菜单会消失,故重复添加
addContextMenus(); addContextMenus(contextMenuType);
} else {
removeContextMenus();
}
// 同步订阅规则 // 同步订阅规则
trySyncAllSubRules({ subrulesList }); trySyncAllSubRules({ subrulesList });
@@ -134,12 +140,8 @@ browser.runtime.onMessage.addListener(
saveRule(args); saveRule(args);
break; break;
case MSG_CONTEXT_MENUS: case MSG_CONTEXT_MENUS:
const { contextMenus } = args; const { contextMenuType } = args;
if (contextMenus) { addContextMenus(contextMenuType);
addContextMenus();
} else {
removeContextMenus();
}
break; break;
case MSG_COMMAND_SHORTCUTS: case MSG_COMMAND_SHORTCUTS:
browser.commands browser.commands

View File

@@ -14,7 +14,7 @@ import {
} from "./config"; } from "./config";
import { getFabWithDefault, getSettingWithDefault } from "./libs/storage"; import { getFabWithDefault, getSettingWithDefault } from "./libs/storage";
import { Translator } from "./libs/translator"; import { Translator } from "./libs/translator";
import { isIframe, sendIframeMsg, sendParentMsg } from "./libs/iframe"; import { isIframe, sendIframeMsg } from "./libs/iframe";
import Slection from "./views/Selection"; import Slection from "./views/Selection";
import { touchTapListener } from "./libs/touch"; import { touchTapListener } from "./libs/touch";
import { debounce, genEventName } from "./libs/utils"; import { debounce, genEventName } from "./libs/utils";
@@ -79,10 +79,9 @@ function runtimeListener(translator) {
/** /**
* iframe 页面执行 * iframe 页面执行
* @param {*} setting * @param {*} translator
*/ */
function runIframe(setting) { function runIframe(translator) {
let translator;
window.addEventListener("message", (e) => { window.addEventListener("message", (e) => {
const { action, args } = e.data || {}; const { action, args } = e.data || {};
switch (action) { switch (action) {
@@ -93,16 +92,11 @@ function runIframe(setting) {
translator?.toggleStyle(); translator?.toggleStyle();
break; break;
case MSG_TRANS_PUTRULE: case MSG_TRANS_PUTRULE:
if (!translator) {
translator = new Translator(args, setting);
} else {
translator.updateRule(args || {}); translator.updateRule(args || {});
}
break; break;
default: default:
} }
}); });
sendParentMsg(MSG_TRANS_GETRULE);
} }
/** /**
@@ -140,7 +134,7 @@ async function showFab(translator) {
* @returns * @returns
*/ */
function showTransbox({ function showTransbox({
contextMenus = true, contextMenuType,
tranboxSetting = DEFAULT_TRANBOX_SETTING, tranboxSetting = DEFAULT_TRANBOX_SETTING,
transApis, transApis,
}) { }) {
@@ -165,7 +159,7 @@ function showTransbox({
<React.StrictMode> <React.StrictMode>
<CacheProvider value={cache}> <CacheProvider value={cache}>
<Slection <Slection
contextMenus={contextMenus} contextMenuType={contextMenuType}
tranboxSetting={tranboxSetting} tranboxSetting={tranboxSetting}
transApis={transApis} transApis={transApis}
/> />
@@ -174,22 +168,6 @@ function showTransbox({
); );
} }
/**
* 监听来自iframe页面消息
* @param {*} rule
*/
function windowListener(rule) {
window.addEventListener("message", (e) => {
const { action } = e.data || {};
switch (action) {
case MSG_TRANS_GETRULE:
sendIframeMsg(MSG_TRANS_PUTRULE, rule);
break;
default:
}
});
}
/** /**
* 显示错误信息到页面顶部 * 显示错误信息到页面顶部
* @param {*} message * @param {*} message
@@ -245,12 +223,6 @@ export async function run(isUserscript = false) {
return; return;
} }
// 适配iframe
if (isIframe) {
runIframe(setting);
return;
}
// 不规范网页修复 // 不规范网页修复
const fixerSetting = await matchFixer(href, setting); const fixerSetting = await matchFixer(href, setting);
@@ -258,8 +230,13 @@ export async function run(isUserscript = false) {
const rule = await matchRule(href, setting); const rule = await matchRule(href, setting);
const translator = new Translator(rule, setting, fixerSetting); const translator = new Translator(rule, setting, fixerSetting);
// 适配iframe
if (isIframe) {
runIframe(translator);
return;
}
// 监听消息 // 监听消息
windowListener(rule);
!isUserscript && runtimeListener(translator); !isUserscript && runtimeListener(translator);
// 输入框翻译 // 输入框翻译

View File

@@ -376,7 +376,7 @@ export const I18N = {
en: `1. Follow CSS selector syntax. 2. Leave blank to adopt the global setting. 3. Separate multiple CSS selectors with ";". 4. The "shadow root" selector and the internal selector are separated by ">>>".`, en: `1. Follow CSS selector syntax. 2. Leave blank to adopt the global setting. 3. Separate multiple CSS selectors with ";". 4. The "shadow root" selector and the internal selector are separated by ">>>".`,
}, },
translate_switch: { translate_switch: {
zh: `翻译`, zh: `启翻译`,
en: `Translate Switch`, en: `Translate Switch`,
}, },
default_enabled: { default_enabled: {
@@ -560,7 +560,7 @@ export const I18N = {
en: `Shortcuts Setting`, en: `Shortcuts Setting`,
}, },
toggle_translate_shortcut: { toggle_translate_shortcut: {
zh: `"启翻译"快捷键`, zh: `"启翻译"快捷键`,
en: `"Toggle Translate" Shortcut`, en: `"Toggle Translate" Shortcut`,
}, },
toggle_style_shortcut: { toggle_style_shortcut: {
@@ -723,9 +723,21 @@ export const I18N = {
zh: `此功能依赖准确的语言检测,建议启用远程语言检测。`, zh: `此功能依赖准确的语言检测,建议启用远程语言检测。`,
en: `This feature relies on accurate language detection. It is recommended to enable remote language detection.`, en: `This feature relies on accurate language detection. It is recommended to enable remote language detection.`,
}, },
add_context_menus: { context_menus: {
zh: `添加右键菜单`, zh: `右键菜单`,
en: `Add Context Menus`, en: `Context Menus`,
},
hide_context_menus: {
zh: `隐藏右键菜单`,
en: `Hide Context Menus`,
},
simple_context_menus: {
zh: `简单右键菜单`,
en: `Simple_context_menus Context Menus`,
},
secondary_context_menus: {
zh: `二级右键菜单`,
en: `Secondary Context Menus`,
}, },
mulkeys_help: { mulkeys_help: {
zh: `支持用换行或英文逗号“,”分隔多个KEY轮询调用。`, zh: `支持用换行或英文逗号“,”分隔多个KEY轮询调用。`,

View File

@@ -437,7 +437,7 @@ export const DEFAULT_BLACKLIST = [
"https://translate.google.com", "https://translate.google.com",
"https://www.deepl.com/translator", "https://www.deepl.com/translator",
"oapi.dingtalk.com", "oapi.dingtalk.com",
"login.dingtalk.com", "login.dingtalk.com"
]; // 禁用翻译名单 ]; // 禁用翻译名单
export const DEFAULT_SETTING = { export const DEFAULT_SETTING = {
@@ -452,7 +452,8 @@ export const DEFAULT_SETTING = {
injectRules: true, // 是否注入订阅规则 injectRules: true, // 是否注入订阅规则
injectWebfix: true, // 是否注入修复补丁 injectWebfix: true, // 是否注入修复补丁
detectRemote: false, // 是否使用远程语言检测 detectRemote: false, // 是否使用远程语言检测
contextMenus: true, // 是否添加右键菜单 contextMenus: true, // 是否添加右键菜单(作废)
contextMenuType: 1, // 右键菜单类型(0不显示1简单菜单2多级菜单)
transTitle: false, // 是否同时翻译页面标题 transTitle: false, // 是否同时翻译页面标题
subrulesList: DEFAULT_SUBRULES_LIST, // 订阅列表 subrulesList: DEFAULT_SUBRULES_LIST, // 订阅列表
owSubrule: DEFAULT_OW_RULE, // 覆写订阅规则 owSubrule: DEFAULT_OW_RULE, // 覆写订阅规则

View File

@@ -462,7 +462,12 @@ export class Translator {
}); });
if (keeps.length > 0) { if (keeps.length > 0) {
// textContent会保留些无用的换行符严重影响翻译质量
if (q.includes("\n")) {
q = text; q = text;
} else {
q = text.replaceAll("\n", " ");
}
} }
} }

View File

@@ -85,6 +85,7 @@ function brFixer(node, tag = "p") {
"HR", "HR",
"PRE", "PRE",
"TABLE", "TABLE",
"BLOCKQUOTE",
]; ];
let html = ""; let html = "";

View File

@@ -95,8 +95,8 @@ export default function Action({ translator, fab }) {
// 注册菜单 // 注册菜单
try { try {
const menuCommandIds = []; const menuCommandIds = [];
const { contextMenus = true } = translator.setting; const { contextMenuType } = translator.setting;
contextMenus && contextMenuType !== 0 &&
menuCommandIds.push( menuCommandIds.push(
GM.registerMenuCommand( GM.registerMenuCommand(
"Toggle Translate", "Toggle Translate",

View File

@@ -67,8 +67,8 @@ export default function Settings() {
case "touchTranslate": case "touchTranslate":
value = limitNumber(value, 0, 4); value = limitNumber(value, 0, 4);
break; break;
case "contextMenus": case "contextMenuType":
isExt && sendBgMsg(MSG_CONTEXT_MENUS, { contextMenus: value }); isExt && sendBgMsg(MSG_CONTEXT_MENUS, { contextMenuType: value });
break; break;
default: default:
} }
@@ -96,7 +96,7 @@ export default function Settings() {
newlineLength = TRANS_NEWLINE_LENGTH, newlineLength = TRANS_NEWLINE_LENGTH,
mouseKey = OPT_MOUSEKEY_DISABLE, mouseKey = OPT_MOUSEKEY_DISABLE,
detectRemote = false, detectRemote = false,
contextMenus = true, contextMenuType = 1,
transTitle = false, transTitle = false,
touchTranslate = 2, touchTranslate = 2,
blacklist = DEFAULT_BLACKLIST.join(",\n"), blacklist = DEFAULT_BLACKLIST.join(",\n"),
@@ -229,15 +229,16 @@ export default function Settings() {
</FormControl> </FormControl>
<FormControl size="small"> <FormControl size="small">
<InputLabel>{i18n("add_context_menus")}</InputLabel> <InputLabel>{i18n("context_menus")}</InputLabel>
<Select <Select
name="contextMenus" name="contextMenuType"
value={contextMenus} value={contextMenuType}
label={i18n("add_context_menus")} label={i18n("context_menus")}
onChange={handleChange} onChange={handleChange}
> >
<MenuItem value={false}>{i18n("disable")}</MenuItem> <MenuItem value={0}>{i18n("hide_context_menus")}</MenuItem>
<MenuItem value={true}>{i18n("enable")}</MenuItem> <MenuItem value={1}>{i18n("simple_context_menus")}</MenuItem>
<MenuItem value={2}>{i18n("secondary_context_menus")}</MenuItem>
</Select> </Select>
</FormControl> </FormControl>

View File

@@ -29,7 +29,7 @@ export default function DictCont({ dictResult }) {
<Typography component="div"> <Typography component="div">
<Typography> <Typography>
{dictResult.voice {dictResult.voice
.map(Object.entries) ?.map(Object.entries)
.map((item) => item[0]) .map((item) => item[0])
.map(([key, val]) => `${phonicMap[key] || key} ${val}`) .map(([key, val]) => `${phonicMap[key] || key} ${val}`)
.join(" ")} .join(" ")}

View File

@@ -7,7 +7,11 @@ import { isGm, isExt } from "../../libs/client";
import { MSG_OPEN_TRANBOX, DEFAULT_TRANBOX_SHORTCUT } from "../../config"; import { MSG_OPEN_TRANBOX, DEFAULT_TRANBOX_SHORTCUT } from "../../config";
import { isMobile } from "../../libs/mobile"; import { isMobile } from "../../libs/mobile";
export default function Slection({ contextMenus, tranboxSetting, transApis }) { export default function Slection({
contextMenuType,
tranboxSetting,
transApis,
}) {
const boxWidth = limitNumber(window.innerWidth, 300, 600); const boxWidth = limitNumber(window.innerWidth, 300, 600);
const boxHeight = limitNumber(window.innerHeight, 200, 400); const boxHeight = limitNumber(window.innerHeight, 200, 400);
@@ -106,7 +110,7 @@ export default function Slection({ contextMenus, tranboxSetting, transApis }) {
// 注册菜单 // 注册菜单
try { try {
const menuCommandIds = []; const menuCommandIds = [];
contextMenus && contextMenuType !== 0 &&
menuCommandIds.push( menuCommandIds.push(
GM.registerMenuCommand( GM.registerMenuCommand(
"Translate Selected Text", "Translate Selected Text",
@@ -125,7 +129,7 @@ export default function Slection({ contextMenus, tranboxSetting, transApis }) {
} catch (err) { } catch (err) {
console.log("[registerMenuCommand]", err); console.log("[registerMenuCommand]", err);
} }
}, [handleTranbox, contextMenus]); }, [handleTranbox, contextMenuType]);
return ( return (
<> <>