Compare commits
19 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
72ccfc8aec | ||
|
|
d117c5dc10 | ||
|
|
9312783f44 | ||
|
|
e5b16ebfd3 | ||
|
|
5d1d65c2d3 | ||
|
|
9ca1309cec | ||
|
|
a03afc05f5 | ||
|
|
0198963584 | ||
|
|
58e745d967 | ||
|
|
377e347d68 | ||
|
|
bac0704d3d | ||
|
|
d2ff46edf6 | ||
|
|
f908372b4e | ||
|
|
5d44ff4913 | ||
|
|
4c9aa66048 | ||
|
|
b6a09b99ab | ||
|
|
3a0dcb1a52 | ||
|
|
5015503b4c | ||
|
|
16423feea4 |
2
.env
2
.env
@@ -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.8
|
REACT_APP_VERSION=1.8.11
|
||||||
|
|
||||||
REACT_APP_HOMEPAGE=https://github.com/fishjar/kiss-translator
|
REACT_APP_HOMEPAGE=https://github.com/fishjar/kiss-translator
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
# KISS Translator
|
# KISS Translator
|
||||||
|
|
||||||
|
English | [简体中文](README.md)
|
||||||
|
|
||||||
A simple, open source [bilingual translation extension & Greasemonkey script](https://github.com/fishjar/kiss-translator).
|
A simple, open source [bilingual translation extension & Greasemonkey script](https://github.com/fishjar/kiss-translator).
|
||||||
|
|
||||||
[kiss-translator.webm](https://github.com/fishjar/kiss-translator/assets/1157624/f7ba8a5c-e4a8-4d5a-823a-5c5c67a0a47f)
|
[kiss-translator.webm](https://github.com/fishjar/kiss-translator/assets/1157624/f7ba8a5c-e4a8-4d5a-823a-5c5c67a0a47f)
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
# 简约翻译
|
# 简约翻译
|
||||||
|
|
||||||
|
[English](README.en.md) | 简体中文
|
||||||
|
|
||||||
一个简约、开源的 [双语对照翻译扩展 & 油猴脚本](https://github.com/fishjar/kiss-translator)。
|
一个简约、开源的 [双语对照翻译扩展 & 油猴脚本](https://github.com/fishjar/kiss-translator)。
|
||||||
|
|
||||||
[kiss-translator.webm](https://github.com/fishjar/kiss-translator/assets/1157624/f7ba8a5c-e4a8-4d5a-823a-5c5c67a0a47f)
|
[kiss-translator.webm](https://github.com/fishjar/kiss-translator/assets/1157624/f7ba8a5c-e4a8-4d5a-823a-5c5c67a0a47f)
|
||||||
|
|||||||
@@ -112,6 +112,8 @@ const userscriptWebpack = (config, env) => {
|
|||||||
// @connect 127.0.0.1:3000
|
// @connect 127.0.0.1:3000
|
||||||
// @connect localhost:1188
|
// @connect localhost:1188
|
||||||
// @connect 127.0.0.1:1188
|
// @connect 127.0.0.1:1188
|
||||||
|
// @connect localhost:11434
|
||||||
|
// @connect 127.0.0.1:11434
|
||||||
// @run-at document-end
|
// @run-at document-end
|
||||||
// ==/UserScript==
|
// ==/UserScript==
|
||||||
|
|
||||||
|
|||||||
@@ -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.8",
|
"version": "1.8.11",
|
||||||
"author": "Gabe<yugang2002@gmail.com>",
|
"author": "Gabe<yugang2002@gmail.com>",
|
||||||
"private": true,
|
"private": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -17,6 +17,7 @@
|
|||||||
"react-markdown": "^8.0.7",
|
"react-markdown": "^8.0.7",
|
||||||
"react-router-dom": "^6.16.0",
|
"react-router-dom": "^6.16.0",
|
||||||
"react-scripts": "5.0.1",
|
"react-scripts": "5.0.1",
|
||||||
|
"sval": "^0.5.2",
|
||||||
"webdav": "^5.3.0",
|
"webdav": "^5.3.0",
|
||||||
"webextension-polyfill": "^0.10.0"
|
"webextension-polyfill": "^0.10.0"
|
||||||
},
|
},
|
||||||
@@ -31,7 +32,7 @@
|
|||||||
"build:userscript": "rm -rf build/userscript && mkdir build/userscript && cp build/web/*.user.js build/userscript/",
|
"build:userscript": "rm -rf build/userscript && mkdir build/userscript && cp build/web/*.user.js build/userscript/",
|
||||||
"build:rules": "babel-node src/rules.js",
|
"build:rules": "babel-node src/rules.js",
|
||||||
"build": "pnpm build:chrome && pnpm build:edge && pnpm build:firefox && pnpm build:web && pnpm build:userscript-ios && pnpm build:userscript && pnpm build:rules",
|
"build": "pnpm build:chrome && pnpm build:edge && pnpm build:firefox && pnpm build:web && pnpm build:userscript-ios && pnpm build:userscript && pnpm build:rules",
|
||||||
"pack": "cd build && zip -r chrome.zip chrome && zip -r edge.zip edge && cd firefox && zip -r ../firefox.zip .",
|
"zip": "cd build && zip -r chrome.zip chrome && zip -r edge.zip edge && cd firefox && zip -r ../firefox.zip .",
|
||||||
"test": "react-app-rewired test",
|
"test": "react-app-rewired test",
|
||||||
"eject": "react-scripts eject"
|
"eject": "react-scripts eject"
|
||||||
},
|
},
|
||||||
|
|||||||
11
pnpm-lock.yaml
generated
11
pnpm-lock.yaml
generated
@@ -41,6 +41,9 @@ dependencies:
|
|||||||
react-scripts:
|
react-scripts:
|
||||||
specifier: 5.0.1
|
specifier: 5.0.1
|
||||||
version: 5.0.1(@babel/plugin-syntax-flow@7.24.1)(@babel/plugin-transform-react-jsx@7.23.4)(eslint@8.57.0)(react@18.2.0)(typescript@5.4.5)
|
version: 5.0.1(@babel/plugin-syntax-flow@7.24.1)(@babel/plugin-transform-react-jsx@7.23.4)(eslint@8.57.0)(react@18.2.0)(typescript@5.4.5)
|
||||||
|
sval:
|
||||||
|
specifier: ^0.5.2
|
||||||
|
version: 0.5.2
|
||||||
webdav:
|
webdav:
|
||||||
specifier: ^5.3.0
|
specifier: ^5.3.0
|
||||||
version: 5.3.0
|
version: 5.3.0
|
||||||
@@ -6504,7 +6507,7 @@ packages:
|
|||||||
optional: true
|
optional: true
|
||||||
dependencies:
|
dependencies:
|
||||||
abab: 2.0.6
|
abab: 2.0.6
|
||||||
acorn: 8.10.0
|
acorn: 8.11.3
|
||||||
acorn-globals: 6.0.0
|
acorn-globals: 6.0.0
|
||||||
cssom: 0.4.4
|
cssom: 0.4.4
|
||||||
cssstyle: 2.3.0
|
cssstyle: 2.3.0
|
||||||
@@ -9348,6 +9351,12 @@ packages:
|
|||||||
resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==}
|
resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==}
|
||||||
engines: {node: '>= 0.4'}
|
engines: {node: '>= 0.4'}
|
||||||
|
|
||||||
|
/sval@0.5.2:
|
||||||
|
resolution: {integrity: sha512-cMN4SWqQ8K2DypYVZ1DVsicvXsr4gQmAYR2faKwHttJFJAqjfc4+taG9esMIP0hMP5+4Caun99n6y+4T6nCPEA==}
|
||||||
|
dependencies:
|
||||||
|
acorn: 8.11.3
|
||||||
|
dev: false
|
||||||
|
|
||||||
/svg-parser@2.0.4:
|
/svg-parser@2.0.4:
|
||||||
resolution: {integrity: sha512-e4hG1hRwoOdRb37cIMSgzNsxyzKfayW6VOflrwvR+/bzrkyxY/31WkbgnQpgtrNp1SdpJvpUAGTa/ZoiPNDuRQ==}
|
resolution: {integrity: sha512-e4hG1hRwoOdRb37cIMSgzNsxyzKfayW6VOflrwvR+/bzrkyxY/31WkbgnQpgtrNp1SdpJvpUAGTa/ZoiPNDuRQ==}
|
||||||
|
|
||||||
|
|||||||
@@ -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.8",
|
"version": "1.8.11",
|
||||||
"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",
|
||||||
|
|||||||
@@ -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.8",
|
"version": "1.8.11",
|
||||||
"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",
|
||||||
|
|||||||
@@ -10,8 +10,13 @@ import {
|
|||||||
OPT_TRANS_BAIDU,
|
OPT_TRANS_BAIDU,
|
||||||
OPT_TRANS_TENCENT,
|
OPT_TRANS_TENCENT,
|
||||||
OPT_TRANS_OPENAI,
|
OPT_TRANS_OPENAI,
|
||||||
|
OPT_TRANS_OPENAI_2,
|
||||||
|
OPT_TRANS_OPENAI_3,
|
||||||
OPT_TRANS_GEMINI,
|
OPT_TRANS_GEMINI,
|
||||||
OPT_TRANS_CLOUDFLAREAI,
|
OPT_TRANS_CLOUDFLAREAI,
|
||||||
|
OPT_TRANS_OLLAMA,
|
||||||
|
OPT_TRANS_OLLAMA_2,
|
||||||
|
OPT_TRANS_OLLAMA_3,
|
||||||
OPT_TRANS_CUSTOMIZE,
|
OPT_TRANS_CUSTOMIZE,
|
||||||
OPT_TRANS_CUSTOMIZE_2,
|
OPT_TRANS_CUSTOMIZE_2,
|
||||||
OPT_TRANS_CUSTOMIZE_3,
|
OPT_TRANS_CUSTOMIZE_3,
|
||||||
@@ -19,6 +24,8 @@ import {
|
|||||||
OPT_TRANS_CUSTOMIZE_5,
|
OPT_TRANS_CUSTOMIZE_5,
|
||||||
URL_CACHE_TRAN,
|
URL_CACHE_TRAN,
|
||||||
KV_SALT_SYNC,
|
KV_SALT_SYNC,
|
||||||
|
URL_GOOGLE_TRAN,
|
||||||
|
URL_MICROSOFT_LANGDETECT,
|
||||||
URL_BAIDU_LANGDETECT,
|
URL_BAIDU_LANGDETECT,
|
||||||
URL_BAIDU_SUGGEST,
|
URL_BAIDU_SUGGEST,
|
||||||
URL_BAIDU_TTS,
|
URL_BAIDU_TTS,
|
||||||
@@ -26,8 +33,11 @@ import {
|
|||||||
URL_TENCENT_TRANSMART,
|
URL_TENCENT_TRANSMART,
|
||||||
OPT_LANGS_TENCENT,
|
OPT_LANGS_TENCENT,
|
||||||
OPT_LANGS_SPECIAL,
|
OPT_LANGS_SPECIAL,
|
||||||
|
OPT_LANGS_MICROSOFT,
|
||||||
} from "../config";
|
} from "../config";
|
||||||
import { sha256 } from "../libs/utils";
|
import { sha256 } from "../libs/utils";
|
||||||
|
import interpreter from "../libs/interpreter";
|
||||||
|
import { msAuth } from "../libs/auth";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 同步数据
|
* 同步数据
|
||||||
@@ -53,6 +63,52 @@ export const apiSyncData = async (url, key, data) =>
|
|||||||
*/
|
*/
|
||||||
export const apiFetch = (url) => fetchData(url);
|
export const apiFetch = (url) => fetchData(url);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Google语言识别
|
||||||
|
* @param {*} text
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
export const apiGoogleLangdetect = async (text) => {
|
||||||
|
const params = {
|
||||||
|
client: "gtx",
|
||||||
|
dt: "t",
|
||||||
|
dj: 1,
|
||||||
|
ie: "UTF-8",
|
||||||
|
sl: "auto",
|
||||||
|
tl: "zh-CN",
|
||||||
|
q: text,
|
||||||
|
};
|
||||||
|
const input = `${URL_GOOGLE_TRAN}?${queryString.stringify(params)}`;
|
||||||
|
const res = await fetchData(input, {
|
||||||
|
headers: {
|
||||||
|
"Content-type": "application/json",
|
||||||
|
},
|
||||||
|
useCache: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
return res.src;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Microsoft语言识别
|
||||||
|
* @param {*} text
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
export const apiMicrosoftLangdetect = async (text) => {
|
||||||
|
const [token] = await msAuth();
|
||||||
|
const res = await fetchData(URL_MICROSOFT_LANGDETECT, {
|
||||||
|
headers: {
|
||||||
|
"Content-type": "application/json",
|
||||||
|
Authorization: `Bearer ${token}`,
|
||||||
|
},
|
||||||
|
method: "POST",
|
||||||
|
body: JSON.stringify([{ Text: text }]),
|
||||||
|
useCache: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
return OPT_LANGS_MICROSOFT.get(res[0].language) ?? res[0].language;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 百度语言识别
|
* 百度语言识别
|
||||||
* @param {*} text
|
* @param {*} text
|
||||||
@@ -244,6 +300,8 @@ export const apiTranslate = async ({
|
|||||||
isSame = text === trText;
|
isSame = text === trText;
|
||||||
break;
|
break;
|
||||||
case OPT_TRANS_OPENAI:
|
case OPT_TRANS_OPENAI:
|
||||||
|
case OPT_TRANS_OPENAI_2:
|
||||||
|
case OPT_TRANS_OPENAI_3:
|
||||||
trText = res?.choices?.map((item) => item.message.content).join(" ");
|
trText = res?.choices?.map((item) => item.message.content).join(" ");
|
||||||
isSame = text === trText;
|
isSame = text === trText;
|
||||||
break;
|
break;
|
||||||
@@ -257,32 +315,25 @@ export const apiTranslate = async ({
|
|||||||
trText = res?.result?.translated_text;
|
trText = res?.result?.translated_text;
|
||||||
isSame = text === trText;
|
isSame = text === trText;
|
||||||
break;
|
break;
|
||||||
|
case OPT_TRANS_OLLAMA:
|
||||||
|
case OPT_TRANS_OLLAMA_2:
|
||||||
|
case OPT_TRANS_OLLAMA_3:
|
||||||
|
trText = res?.response;
|
||||||
|
isSame = text === trText;
|
||||||
|
break;
|
||||||
case OPT_TRANS_CUSTOMIZE:
|
case OPT_TRANS_CUSTOMIZE:
|
||||||
case OPT_TRANS_CUSTOMIZE_2:
|
case OPT_TRANS_CUSTOMIZE_2:
|
||||||
case OPT_TRANS_CUSTOMIZE_3:
|
case OPT_TRANS_CUSTOMIZE_3:
|
||||||
case OPT_TRANS_CUSTOMIZE_4:
|
case OPT_TRANS_CUSTOMIZE_4:
|
||||||
case OPT_TRANS_CUSTOMIZE_5:
|
case OPT_TRANS_CUSTOMIZE_5:
|
||||||
|
const { resHook } = apiSetting;
|
||||||
|
if (resHook?.trim()) {
|
||||||
|
interpreter.run(`exports.resHook = ${resHook}`);
|
||||||
|
[trText, isSame] = interpreter.exports.resHook(res, text, from, to);
|
||||||
|
} else {
|
||||||
trText = res.text;
|
trText = res.text;
|
||||||
isSame = to === res.from;
|
isSame = to === res.from;
|
||||||
|
|
||||||
const { customOption } = apiSetting;
|
|
||||||
if (customOption?.trim()) {
|
|
||||||
try {
|
|
||||||
const opt = JSON.parse(customOption);
|
|
||||||
const textPattern = opt.resPattern?.text;
|
|
||||||
const fromPattern = opt.resPattern?.from;
|
|
||||||
if (textPattern) {
|
|
||||||
trText = textPattern.split(".").reduce((pre, cur) => pre[cur], res);
|
|
||||||
}
|
}
|
||||||
if (fromPattern) {
|
|
||||||
isSame =
|
|
||||||
to === fromPattern.split(".").reduce((pre, cur) => pre[cur], res);
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
throw new Error(`custom option parse err: ${err}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,8 +9,13 @@ import {
|
|||||||
OPT_TRANS_BAIDU,
|
OPT_TRANS_BAIDU,
|
||||||
OPT_TRANS_TENCENT,
|
OPT_TRANS_TENCENT,
|
||||||
OPT_TRANS_OPENAI,
|
OPT_TRANS_OPENAI,
|
||||||
|
OPT_TRANS_OPENAI_2,
|
||||||
|
OPT_TRANS_OPENAI_3,
|
||||||
OPT_TRANS_GEMINI,
|
OPT_TRANS_GEMINI,
|
||||||
OPT_TRANS_CLOUDFLAREAI,
|
OPT_TRANS_CLOUDFLAREAI,
|
||||||
|
OPT_TRANS_OLLAMA,
|
||||||
|
OPT_TRANS_OLLAMA_2,
|
||||||
|
OPT_TRANS_OLLAMA_3,
|
||||||
OPT_TRANS_CUSTOMIZE,
|
OPT_TRANS_CUSTOMIZE,
|
||||||
OPT_TRANS_CUSTOMIZE_2,
|
OPT_TRANS_CUSTOMIZE_2,
|
||||||
OPT_TRANS_CUSTOMIZE_3,
|
OPT_TRANS_CUSTOMIZE_3,
|
||||||
@@ -23,10 +28,12 @@ import {
|
|||||||
INPUT_PLACE_TO,
|
INPUT_PLACE_TO,
|
||||||
INPUT_PLACE_TEXT,
|
INPUT_PLACE_TEXT,
|
||||||
INPUT_PLACE_KEY,
|
INPUT_PLACE_KEY,
|
||||||
|
INPUT_PLACE_MODEL,
|
||||||
} from "../config";
|
} from "../config";
|
||||||
import { msAuth } from "../libs/auth";
|
import { msAuth } from "../libs/auth";
|
||||||
import { genDeeplFree } from "./deepl";
|
import { genDeeplFree } from "./deepl";
|
||||||
import { genBaidu } from "./baidu";
|
import { genBaidu } from "./baidu";
|
||||||
|
import interpreter from "../libs/interpreter";
|
||||||
|
|
||||||
const keyMap = new Map();
|
const keyMap = new Map();
|
||||||
const urlMap = new Map();
|
const urlMap = new Map();
|
||||||
@@ -178,7 +185,17 @@ const genTencent = ({ text, from, to }) => {
|
|||||||
return [URL_TENCENT_TRANSMART, init];
|
return [URL_TENCENT_TRANSMART, init];
|
||||||
};
|
};
|
||||||
|
|
||||||
const genOpenAI = ({ text, from, to, url, key, prompt, model }) => {
|
const genOpenAI = ({
|
||||||
|
text,
|
||||||
|
from,
|
||||||
|
to,
|
||||||
|
url,
|
||||||
|
key,
|
||||||
|
prompt,
|
||||||
|
model,
|
||||||
|
temperature,
|
||||||
|
maxTokens,
|
||||||
|
}) => {
|
||||||
prompt = prompt
|
prompt = prompt
|
||||||
.replaceAll(INPUT_PLACE_FROM, from)
|
.replaceAll(INPUT_PLACE_FROM, from)
|
||||||
.replaceAll(INPUT_PLACE_TO, to);
|
.replaceAll(INPUT_PLACE_TO, to);
|
||||||
@@ -195,8 +212,8 @@ const genOpenAI = ({ text, from, to, url, key, prompt, model }) => {
|
|||||||
content: text,
|
content: text,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
temperature: 0,
|
temperature,
|
||||||
max_tokens: 256,
|
max_tokens: maxTokens,
|
||||||
};
|
};
|
||||||
|
|
||||||
const init = {
|
const init = {
|
||||||
@@ -213,6 +230,9 @@ const genOpenAI = ({ text, from, to, url, key, prompt, model }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const genGemini = ({ text, from, to, url, key, prompt, model }) => {
|
const genGemini = ({ text, from, to, url, key, prompt, model }) => {
|
||||||
|
url = url
|
||||||
|
.replaceAll(INPUT_PLACE_MODEL, model)
|
||||||
|
.replaceAll(INPUT_PLACE_KEY, key);
|
||||||
prompt = prompt
|
prompt = prompt
|
||||||
.replaceAll(INPUT_PLACE_FROM, from)
|
.replaceAll(INPUT_PLACE_FROM, from)
|
||||||
.replaceAll(INPUT_PLACE_TO, to)
|
.replaceAll(INPUT_PLACE_TO, to)
|
||||||
@@ -231,7 +251,6 @@ const genGemini = ({ text, from, to, url, key, prompt, model }) => {
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
|
|
||||||
const input = `${url}/${model}:generateContent?key=${key}`;
|
|
||||||
const init = {
|
const init = {
|
||||||
headers: {
|
headers: {
|
||||||
"Content-type": "application/json",
|
"Content-type": "application/json",
|
||||||
@@ -240,7 +259,33 @@ const genGemini = ({ text, from, to, url, key, prompt, model }) => {
|
|||||||
body: JSON.stringify(data),
|
body: JSON.stringify(data),
|
||||||
};
|
};
|
||||||
|
|
||||||
return [input, init];
|
return [url, init];
|
||||||
|
};
|
||||||
|
|
||||||
|
const genOllama = ({ text, from, to, url, key, prompt, model }) => {
|
||||||
|
prompt = prompt
|
||||||
|
.replaceAll(INPUT_PLACE_FROM, from)
|
||||||
|
.replaceAll(INPUT_PLACE_TO, to)
|
||||||
|
.replaceAll(INPUT_PLACE_TEXT, text);
|
||||||
|
|
||||||
|
const data = {
|
||||||
|
model,
|
||||||
|
prompt,
|
||||||
|
stream: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
const init = {
|
||||||
|
headers: {
|
||||||
|
"Content-type": "application/json",
|
||||||
|
},
|
||||||
|
method: "POST",
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
};
|
||||||
|
if (key) {
|
||||||
|
init.headers.Authorization = `Bearer ${key}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [url, init];
|
||||||
};
|
};
|
||||||
|
|
||||||
const genCloudflareAI = ({ text, from, to, url, key }) => {
|
const genCloudflareAI = ({ text, from, to, url, key }) => {
|
||||||
@@ -262,20 +307,27 @@ const genCloudflareAI = ({ text, from, to, url, key }) => {
|
|||||||
return [url, init];
|
return [url, init];
|
||||||
};
|
};
|
||||||
|
|
||||||
const genCustom = ({ text, from, to, url, key, customOption }) => {
|
const genCustom = ({ text, from, to, url, key, reqHook }) => {
|
||||||
const replaceInput = (str) =>
|
url = url
|
||||||
str
|
|
||||||
.replaceAll(INPUT_PLACE_URL, url)
|
.replaceAll(INPUT_PLACE_URL, url)
|
||||||
.replaceAll(INPUT_PLACE_FROM, from)
|
.replaceAll(INPUT_PLACE_FROM, from)
|
||||||
.replaceAll(INPUT_PLACE_TO, to)
|
.replaceAll(INPUT_PLACE_TO, to)
|
||||||
.replaceAll(INPUT_PLACE_TEXT, text.replaceAll(`"`, `\n`))
|
.replaceAll(INPUT_PLACE_TEXT, text)
|
||||||
.replaceAll(INPUT_PLACE_KEY, key);
|
.replaceAll(INPUT_PLACE_KEY, key);
|
||||||
|
let init = {};
|
||||||
|
|
||||||
|
if (reqHook?.trim()) {
|
||||||
|
interpreter.run(`exports.reqHook = ${reqHook}`);
|
||||||
|
[url, init] = interpreter.exports.reqHook(text, from, to, url, key);
|
||||||
|
return [url, init];
|
||||||
|
}
|
||||||
|
|
||||||
const data = {
|
const data = {
|
||||||
text,
|
text,
|
||||||
from,
|
from,
|
||||||
to,
|
to,
|
||||||
};
|
};
|
||||||
const init = {
|
init = {
|
||||||
headers: {
|
headers: {
|
||||||
"Content-type": "application/json",
|
"Content-type": "application/json",
|
||||||
},
|
},
|
||||||
@@ -285,23 +337,6 @@ const genCustom = ({ text, from, to, url, key, customOption }) => {
|
|||||||
if (key) {
|
if (key) {
|
||||||
init.headers.Authorization = `Bearer ${key}`;
|
init.headers.Authorization = `Bearer ${key}`;
|
||||||
}
|
}
|
||||||
url = replaceInput(url);
|
|
||||||
|
|
||||||
if (customOption?.trim()) {
|
|
||||||
try {
|
|
||||||
const opt = JSON.parse(replaceInput(customOption));
|
|
||||||
opt.url && (url = opt.url);
|
|
||||||
opt.headers && (init.headers = opt.headers);
|
|
||||||
opt.method && (init.method = opt.method);
|
|
||||||
if (init.method === "GET") {
|
|
||||||
delete init.body;
|
|
||||||
} else {
|
|
||||||
opt.body && (init.body = JSON.stringify(opt.body));
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
throw new Error(`custom option parse err: ${err}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return [url, init];
|
return [url, init];
|
||||||
};
|
};
|
||||||
@@ -317,8 +352,13 @@ export const genTransReq = ({ translator, text, from, to }, apiSetting) => {
|
|||||||
switch (translator) {
|
switch (translator) {
|
||||||
case OPT_TRANS_DEEPL:
|
case OPT_TRANS_DEEPL:
|
||||||
case OPT_TRANS_OPENAI:
|
case OPT_TRANS_OPENAI:
|
||||||
|
case OPT_TRANS_OPENAI_2:
|
||||||
|
case OPT_TRANS_OPENAI_3:
|
||||||
case OPT_TRANS_GEMINI:
|
case OPT_TRANS_GEMINI:
|
||||||
case OPT_TRANS_CLOUDFLAREAI:
|
case OPT_TRANS_CLOUDFLAREAI:
|
||||||
|
case OPT_TRANS_OLLAMA:
|
||||||
|
case OPT_TRANS_OLLAMA_2:
|
||||||
|
case OPT_TRANS_OLLAMA_3:
|
||||||
case OPT_TRANS_NIUTRANS:
|
case OPT_TRANS_NIUTRANS:
|
||||||
args.key = keyPick(translator, args.key, keyMap);
|
args.key = keyPick(translator, args.key, keyMap);
|
||||||
break;
|
break;
|
||||||
@@ -346,11 +386,17 @@ export const genTransReq = ({ translator, text, from, to }, apiSetting) => {
|
|||||||
case OPT_TRANS_TENCENT:
|
case OPT_TRANS_TENCENT:
|
||||||
return genTencent(args);
|
return genTencent(args);
|
||||||
case OPT_TRANS_OPENAI:
|
case OPT_TRANS_OPENAI:
|
||||||
|
case OPT_TRANS_OPENAI_2:
|
||||||
|
case OPT_TRANS_OPENAI_3:
|
||||||
return genOpenAI(args);
|
return genOpenAI(args);
|
||||||
case OPT_TRANS_GEMINI:
|
case OPT_TRANS_GEMINI:
|
||||||
return genGemini(args);
|
return genGemini(args);
|
||||||
case OPT_TRANS_CLOUDFLAREAI:
|
case OPT_TRANS_CLOUDFLAREAI:
|
||||||
return genCloudflareAI(args);
|
return genCloudflareAI(args);
|
||||||
|
case OPT_TRANS_OLLAMA:
|
||||||
|
case OPT_TRANS_OLLAMA_2:
|
||||||
|
case OPT_TRANS_OLLAMA_3:
|
||||||
|
return genOllama(args);
|
||||||
case OPT_TRANS_CUSTOMIZE:
|
case OPT_TRANS_CUSTOMIZE:
|
||||||
case OPT_TRANS_CUSTOMIZE_2:
|
case OPT_TRANS_CUSTOMIZE_2:
|
||||||
case OPT_TRANS_CUSTOMIZE_3:
|
case OPT_TRANS_CUSTOMIZE_3:
|
||||||
|
|||||||
@@ -141,6 +141,7 @@ function showTransbox({
|
|||||||
transApis,
|
transApis,
|
||||||
darkMode,
|
darkMode,
|
||||||
uiLang,
|
uiLang,
|
||||||
|
langDetector,
|
||||||
}) {
|
}) {
|
||||||
if (!tranboxSetting?.transOpen) {
|
if (!tranboxSetting?.transOpen) {
|
||||||
return;
|
return;
|
||||||
@@ -172,6 +173,7 @@ function showTransbox({
|
|||||||
tranboxSetting={tranboxSetting}
|
tranboxSetting={tranboxSetting}
|
||||||
transApis={transApis}
|
transApis={transApis}
|
||||||
uiLang={uiLang}
|
uiLang={uiLang}
|
||||||
|
langDetector={langDetector}
|
||||||
/>
|
/>
|
||||||
</CacheProvider>
|
</CacheProvider>
|
||||||
</React.StrictMode>
|
</React.StrictMode>
|
||||||
|
|||||||
@@ -42,7 +42,23 @@ const customApiLangs = `["en", "English - English"],
|
|||||||
["vi", "Vietnamese - Tiếng Việt"],
|
["vi", "Vietnamese - Tiếng Việt"],
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const customDefaultOption = `{
|
const hookExample = `// URL
|
||||||
|
https://translate.googleapis.com/translate_a/single?client=gtx&dj=1&dt=t&ie=UTF-8&q={{text}}&sl=en&tl=zh-CN
|
||||||
|
|
||||||
|
// Request Hook
|
||||||
|
(text, from, to, url, key) => [url, {
|
||||||
|
headers: {
|
||||||
|
"Content-type": "application/json",
|
||||||
|
},
|
||||||
|
method: "GET",
|
||||||
|
body: null,
|
||||||
|
}]
|
||||||
|
|
||||||
|
// Response Hook
|
||||||
|
(res, text, from, to) => [res.sentences.map((item) => item.trans).join(" "), to === res.src]`;
|
||||||
|
|
||||||
|
const customApiHelpZH = `// 请求数据默认格式
|
||||||
|
{
|
||||||
"url": "{{url}}",
|
"url": "{{url}}",
|
||||||
"method": "POST",
|
"method": "POST",
|
||||||
"headers": {
|
"headers": {
|
||||||
@@ -50,18 +66,12 @@ const customDefaultOption = `{
|
|||||||
"Authorization": "Bearer {{key}}"
|
"Authorization": "Bearer {{key}}"
|
||||||
},
|
},
|
||||||
"body": {
|
"body": {
|
||||||
"text": "{{text}}",
|
"text": "{{text}}", // 待翻译文字
|
||||||
"from": "{{from}}",
|
"from": "{{from}}", // 文字的语言(可能为空)
|
||||||
"to": "{{to}}"
|
"to": "{{to}}", // 目标语言
|
||||||
},
|
},
|
||||||
"resPattern": {
|
}
|
||||||
"text": "text",
|
|
||||||
"from": "from"
|
|
||||||
}
|
|
||||||
}`;
|
|
||||||
|
|
||||||
const customApiHelpZH = `// 自定义选项范例
|
|
||||||
${customDefaultOption}
|
|
||||||
|
|
||||||
// 返回数据默认格式
|
// 返回数据默认格式
|
||||||
{
|
{
|
||||||
@@ -70,20 +80,43 @@ ${customDefaultOption}
|
|||||||
to: "", // 目标语言(可选)
|
to: "", // 目标语言(可选)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Hook 范例
|
||||||
|
${hookExample}
|
||||||
|
|
||||||
|
|
||||||
// 支持的语言代码如下
|
// 支持的语言代码如下
|
||||||
${customApiLangs}
|
${customApiLangs}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const customApiHelpEN = `// Example of custom options
|
const customApiHelpEN = `// Default request
|
||||||
${customDefaultOption}
|
{
|
||||||
|
"url": "{{url}}",
|
||||||
|
"method": "POST",
|
||||||
|
"headers": {
|
||||||
|
"Content-type": "application/json",
|
||||||
|
"Authorization": "Bearer {{key}}"
|
||||||
|
},
|
||||||
|
"body": {
|
||||||
|
"text": "{{text}}", // Text to be translated
|
||||||
|
"from": "{{from}}", // The language of the text (may be empty)
|
||||||
|
"to": "{{to}}", // Target language
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
// Return data default format
|
|
||||||
|
// Default response
|
||||||
{
|
{
|
||||||
text: "", // translated text
|
text: "", // translated text
|
||||||
from: "", // Recognized source language
|
from: "", // Recognized source language
|
||||||
to: "", // Target language (optional)
|
to: "", // Target language (optional)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// Hook Example
|
||||||
|
${hookExample}
|
||||||
|
|
||||||
|
|
||||||
// The supported language codes are as follows
|
// The supported language codes are as follows
|
||||||
${customApiLangs}
|
${customApiLangs}
|
||||||
`;
|
`;
|
||||||
@@ -623,7 +656,7 @@ export const I18N = {
|
|||||||
},
|
},
|
||||||
use_simple_style: {
|
use_simple_style: {
|
||||||
zh: `使用简洁界面`,
|
zh: `使用简洁界面`,
|
||||||
en: `Click outside to close the pop-up window`,
|
en: `Use a simple interface`,
|
||||||
},
|
},
|
||||||
show: {
|
show: {
|
||||||
zh: `显示`,
|
zh: `显示`,
|
||||||
@@ -873,4 +906,32 @@ export const I18N = {
|
|||||||
zh: `翻译框跟随选中文本`,
|
zh: `翻译框跟随选中文本`,
|
||||||
en: `Transbox Follow Selection`,
|
en: `Transbox Follow Selection`,
|
||||||
},
|
},
|
||||||
|
translate_start_hook: {
|
||||||
|
zh: `翻译开始钩子函数`,
|
||||||
|
en: `Translate Start Hook`,
|
||||||
|
},
|
||||||
|
translate_start_hook_helper: {
|
||||||
|
zh: `翻译开始时运行,入参为: 翻译节点,原文文本。`,
|
||||||
|
en: `Run when translation starts, the input parameters are: translation node, original text.`,
|
||||||
|
},
|
||||||
|
translate_end_hook: {
|
||||||
|
zh: `翻译完成钩子函数`,
|
||||||
|
en: `Translate End Hook`,
|
||||||
|
},
|
||||||
|
translate_end_hook_helper: {
|
||||||
|
zh: `翻译完成时运行,入参为: 翻译节点,原文文本,译文文本,保留元素。`,
|
||||||
|
en: `Run when the translation is completed, the input parameters are: translation node, original text, translation text, retained elements.`,
|
||||||
|
},
|
||||||
|
translate_remove_hook: {
|
||||||
|
zh: `翻译移除钩子函数`,
|
||||||
|
en: `Translate Removed Hook`,
|
||||||
|
},
|
||||||
|
translate_remove_hook_helper: {
|
||||||
|
zh: `翻译移除时运行,入参为: 翻译节点。`,
|
||||||
|
en: `Run when translation is removed, the input parameters are: translation node.`,
|
||||||
|
},
|
||||||
|
english_dict: {
|
||||||
|
zh: `英文词典`,
|
||||||
|
en: `English Dictionary`,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -78,9 +78,16 @@ export const URL_RAW_PREFIX =
|
|||||||
"https://raw.githubusercontent.com/fishjar/kiss-translator/master";
|
"https://raw.githubusercontent.com/fishjar/kiss-translator/master";
|
||||||
|
|
||||||
export const URL_CACHE_TRAN = `https://${APP_LCNAME}/translate`;
|
export const URL_CACHE_TRAN = `https://${APP_LCNAME}/translate`;
|
||||||
|
|
||||||
|
// api.cognitive.microsofttranslator.com
|
||||||
export const URL_MICROSOFT_TRAN =
|
export const URL_MICROSOFT_TRAN =
|
||||||
"https://api-edge.cognitive.microsofttranslator.com/translate";
|
"https://api-edge.cognitive.microsofttranslator.com/translate";
|
||||||
export const URL_MICROSOFT_AUTH = "https://edge.microsoft.com/translate/auth";
|
export const URL_MICROSOFT_AUTH = "https://edge.microsoft.com/translate/auth";
|
||||||
|
export const URL_MICROSOFT_LANGDETECT =
|
||||||
|
"https://api-edge.cognitive.microsofttranslator.com/detect?api-version=3.0";
|
||||||
|
|
||||||
|
export const URL_GOOGLE_TRAN =
|
||||||
|
"https://translate.googleapis.com/translate_a/single";
|
||||||
export const URL_BAIDU_LANGDETECT = "https://fanyi.baidu.com/langdetect";
|
export const URL_BAIDU_LANGDETECT = "https://fanyi.baidu.com/langdetect";
|
||||||
export const URL_BAIDU_SUGGEST = "https://fanyi.baidu.com/sug";
|
export const URL_BAIDU_SUGGEST = "https://fanyi.baidu.com/sug";
|
||||||
export const URL_BAIDU_TTS = "https://fanyi.baidu.com/gettts";
|
export const URL_BAIDU_TTS = "https://fanyi.baidu.com/gettts";
|
||||||
@@ -95,6 +102,8 @@ export const URL_NIUTRANS_REG =
|
|||||||
export const DEFAULT_USER_AGENT =
|
export const DEFAULT_USER_AGENT =
|
||||||
"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36";
|
"Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36";
|
||||||
|
|
||||||
|
export const OPT_DICT_BAIDU = "Baidu";
|
||||||
|
|
||||||
export const OPT_TRANS_GOOGLE = "Google";
|
export const OPT_TRANS_GOOGLE = "Google";
|
||||||
export const OPT_TRANS_MICROSOFT = "Microsoft";
|
export const OPT_TRANS_MICROSOFT = "Microsoft";
|
||||||
export const OPT_TRANS_DEEPL = "DeepL";
|
export const OPT_TRANS_DEEPL = "DeepL";
|
||||||
@@ -104,8 +113,13 @@ export const OPT_TRANS_NIUTRANS = "NiuTrans";
|
|||||||
export const OPT_TRANS_BAIDU = "Baidu";
|
export const OPT_TRANS_BAIDU = "Baidu";
|
||||||
export const OPT_TRANS_TENCENT = "Tencent";
|
export const OPT_TRANS_TENCENT = "Tencent";
|
||||||
export const OPT_TRANS_OPENAI = "OpenAI";
|
export const OPT_TRANS_OPENAI = "OpenAI";
|
||||||
|
export const OPT_TRANS_OPENAI_2 = "OpenAI2";
|
||||||
|
export const OPT_TRANS_OPENAI_3 = "OpenAI3";
|
||||||
export const OPT_TRANS_GEMINI = "Gemini";
|
export const OPT_TRANS_GEMINI = "Gemini";
|
||||||
export const OPT_TRANS_CLOUDFLAREAI = "CloudflareAI";
|
export const OPT_TRANS_CLOUDFLAREAI = "CloudflareAI";
|
||||||
|
export const OPT_TRANS_OLLAMA = "Ollama";
|
||||||
|
export const OPT_TRANS_OLLAMA_2 = "Ollama2";
|
||||||
|
export const OPT_TRANS_OLLAMA_3 = "Ollama3";
|
||||||
export const OPT_TRANS_CUSTOMIZE = "Custom";
|
export const OPT_TRANS_CUSTOMIZE = "Custom";
|
||||||
export const OPT_TRANS_CUSTOMIZE_2 = "Custom2";
|
export const OPT_TRANS_CUSTOMIZE_2 = "Custom2";
|
||||||
export const OPT_TRANS_CUSTOMIZE_3 = "Custom3";
|
export const OPT_TRANS_CUSTOMIZE_3 = "Custom3";
|
||||||
@@ -121,8 +135,13 @@ export const OPT_TRANS_ALL = [
|
|||||||
OPT_TRANS_DEEPLX,
|
OPT_TRANS_DEEPLX,
|
||||||
OPT_TRANS_NIUTRANS,
|
OPT_TRANS_NIUTRANS,
|
||||||
OPT_TRANS_OPENAI,
|
OPT_TRANS_OPENAI,
|
||||||
|
OPT_TRANS_OPENAI_2,
|
||||||
|
OPT_TRANS_OPENAI_3,
|
||||||
OPT_TRANS_GEMINI,
|
OPT_TRANS_GEMINI,
|
||||||
OPT_TRANS_CLOUDFLAREAI,
|
OPT_TRANS_CLOUDFLAREAI,
|
||||||
|
OPT_TRANS_OLLAMA,
|
||||||
|
OPT_TRANS_OLLAMA_2,
|
||||||
|
OPT_TRANS_OLLAMA_3,
|
||||||
OPT_TRANS_CUSTOMIZE,
|
OPT_TRANS_CUSTOMIZE,
|
||||||
OPT_TRANS_CUSTOMIZE_2,
|
OPT_TRANS_CUSTOMIZE_2,
|
||||||
OPT_TRANS_CUSTOMIZE_3,
|
OPT_TRANS_CUSTOMIZE_3,
|
||||||
@@ -130,6 +149,13 @@ export const OPT_TRANS_ALL = [
|
|||||||
OPT_TRANS_CUSTOMIZE_5,
|
OPT_TRANS_CUSTOMIZE_5,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export const OPT_LANGDETECTOR_ALL = [
|
||||||
|
OPT_TRANS_GOOGLE,
|
||||||
|
OPT_TRANS_MICROSOFT,
|
||||||
|
OPT_TRANS_BAIDU,
|
||||||
|
OPT_TRANS_TENCENT,
|
||||||
|
];
|
||||||
|
|
||||||
export const OPT_LANGS_TO = [
|
export const OPT_LANGS_TO = [
|
||||||
["en", "English - English"],
|
["en", "English - English"],
|
||||||
["zh-CN", "Simplified Chinese - 简体中文"],
|
["zh-CN", "Simplified Chinese - 简体中文"],
|
||||||
@@ -192,7 +218,7 @@ export const OPT_LANGS_SPECIAL = {
|
|||||||
]),
|
]),
|
||||||
[OPT_TRANS_DEEPLX]: new Map([
|
[OPT_TRANS_DEEPLX]: new Map([
|
||||||
...OPT_LANGS_FROM.map(([key]) => [key, key.toUpperCase()]),
|
...OPT_LANGS_FROM.map(([key]) => [key, key.toUpperCase()]),
|
||||||
["auto", ""],
|
["auto", "auto"],
|
||||||
["zh-CN", "ZH"],
|
["zh-CN", "ZH"],
|
||||||
["zh-TW", "ZH"],
|
["zh-TW", "ZH"],
|
||||||
]),
|
]),
|
||||||
@@ -254,9 +280,24 @@ export const OPT_LANGS_SPECIAL = {
|
|||||||
[OPT_TRANS_OPENAI]: new Map(
|
[OPT_TRANS_OPENAI]: new Map(
|
||||||
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
),
|
),
|
||||||
|
[OPT_TRANS_OPENAI_2]: new Map(
|
||||||
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
|
),
|
||||||
|
[OPT_TRANS_OPENAI_3]: new Map(
|
||||||
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
|
),
|
||||||
[OPT_TRANS_GEMINI]: new Map(
|
[OPT_TRANS_GEMINI]: new Map(
|
||||||
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
),
|
),
|
||||||
|
[OPT_TRANS_OLLAMA]: new Map(
|
||||||
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
|
),
|
||||||
|
[OPT_TRANS_OLLAMA_2]: new Map(
|
||||||
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
|
),
|
||||||
|
[OPT_TRANS_OLLAMA_3]: new Map(
|
||||||
|
OPT_LANGS_FROM.map(([key, val]) => [key, val.split(" - ")[0]])
|
||||||
|
),
|
||||||
[OPT_TRANS_CLOUDFLAREAI]: new Map([
|
[OPT_TRANS_CLOUDFLAREAI]: new Map([
|
||||||
["auto", ""],
|
["auto", ""],
|
||||||
["zh-CN", "chinese"],
|
["zh-CN", "chinese"],
|
||||||
@@ -293,6 +334,12 @@ export const OPT_LANGS_SPECIAL = {
|
|||||||
]),
|
]),
|
||||||
};
|
};
|
||||||
export const OPT_LANGS_LIST = OPT_LANGS_TO.map(([lang]) => lang);
|
export const OPT_LANGS_LIST = OPT_LANGS_TO.map(([lang]) => lang);
|
||||||
|
export const OPT_LANGS_MICROSOFT = new Map(
|
||||||
|
Array.from(OPT_LANGS_SPECIAL[OPT_TRANS_MICROSOFT].entries()).map(([k, v]) => [
|
||||||
|
v,
|
||||||
|
k,
|
||||||
|
])
|
||||||
|
);
|
||||||
export const OPT_LANGS_BAIDU = new Map(
|
export const OPT_LANGS_BAIDU = new Map(
|
||||||
Array.from(OPT_LANGS_SPECIAL[OPT_TRANS_BAIDU].entries()).map(([k, v]) => [
|
Array.from(OPT_LANGS_SPECIAL[OPT_TRANS_BAIDU].entries()).map(([k, v]) => [
|
||||||
v,
|
v,
|
||||||
@@ -359,6 +406,7 @@ export const INPUT_PLACE_FROM = "{{from}}"; // 占位符
|
|||||||
export const INPUT_PLACE_TO = "{{to}}"; // 占位符
|
export const INPUT_PLACE_TO = "{{to}}"; // 占位符
|
||||||
export const INPUT_PLACE_TEXT = "{{text}}"; // 占位符
|
export const INPUT_PLACE_TEXT = "{{text}}"; // 占位符
|
||||||
export const INPUT_PLACE_KEY = "{{key}}"; // 占位符
|
export const INPUT_PLACE_KEY = "{{key}}"; // 占位符
|
||||||
|
export const INPUT_PLACE_MODEL = "{{model}}"; // 占位符
|
||||||
|
|
||||||
export const DEFAULT_COLOR = "#209CEE"; // 默认高亮背景色/线条颜色
|
export const DEFAULT_COLOR = "#209CEE"; // 默认高亮背景色/线条颜色
|
||||||
|
|
||||||
@@ -391,6 +439,9 @@ export const GLOBLA_RULE = {
|
|||||||
skipLangs: [], // 不翻译的语言
|
skipLangs: [], // 不翻译的语言
|
||||||
fixerSelector: "", // 修复函数选择器
|
fixerSelector: "", // 修复函数选择器
|
||||||
fixerFunc: "-", // 修复函数
|
fixerFunc: "-", // 修复函数
|
||||||
|
transStartHook: "", // 钩子函数
|
||||||
|
transEndHook: "", // 钩子函数
|
||||||
|
transRemoveHook: "", // 钩子函数
|
||||||
};
|
};
|
||||||
|
|
||||||
// 输入框翻译
|
// 输入框翻译
|
||||||
@@ -438,6 +489,7 @@ export const DEFAULT_TRANBOX_SETTING = {
|
|||||||
followSelection: false, // 翻译框是否跟随选中文本
|
followSelection: false, // 翻译框是否跟随选中文本
|
||||||
triggerMode: OPT_TRANBOX_TRIGGER_CLICK, // 触发翻译方式
|
triggerMode: OPT_TRANBOX_TRIGGER_CLICK, // 触发翻译方式
|
||||||
extStyles: "", // 附加样式
|
extStyles: "", // 附加样式
|
||||||
|
enDict: OPT_DICT_BAIDU, // 英文词典
|
||||||
};
|
};
|
||||||
|
|
||||||
// 订阅列表
|
// 订阅列表
|
||||||
@@ -460,13 +512,33 @@ export const DEFAULT_SUBRULES_LIST = [
|
|||||||
const defaultCustomApi = {
|
const defaultCustomApi = {
|
||||||
url: "",
|
url: "",
|
||||||
key: "",
|
key: "",
|
||||||
customOption: "",
|
customOption: "", // (作废)
|
||||||
|
reqHook: "", // request 钩子函数
|
||||||
|
resHook: "", // response 钩子函数
|
||||||
fetchLimit: DEFAULT_FETCH_LIMIT,
|
fetchLimit: DEFAULT_FETCH_LIMIT,
|
||||||
fetchInterval: DEFAULT_FETCH_INTERVAL,
|
fetchInterval: DEFAULT_FETCH_INTERVAL,
|
||||||
};
|
};
|
||||||
|
const defaultOpenaiApi = {
|
||||||
|
url: "https://api.openai.com/v1/chat/completions",
|
||||||
|
key: "",
|
||||||
|
model: "gpt-4",
|
||||||
|
prompt: `You will be provided with a sentence in ${INPUT_PLACE_FROM}, and your task is to translate it into ${INPUT_PLACE_TO}.`,
|
||||||
|
temperature: 0,
|
||||||
|
maxTokens: 256,
|
||||||
|
fetchLimit: 1,
|
||||||
|
fetchInterval: 500,
|
||||||
|
};
|
||||||
|
const defaultOllamaApi = {
|
||||||
|
url: "http://localhost:11434/api/generate",
|
||||||
|
key: "",
|
||||||
|
model: "llama3",
|
||||||
|
prompt: `Translate the following text from ${INPUT_PLACE_FROM} to ${INPUT_PLACE_TO}:\n\n${INPUT_PLACE_TEXT}`,
|
||||||
|
fetchLimit: 1,
|
||||||
|
fetchInterval: 500,
|
||||||
|
};
|
||||||
export const DEFAULT_TRANS_APIS = {
|
export const DEFAULT_TRANS_APIS = {
|
||||||
[OPT_TRANS_GOOGLE]: {
|
[OPT_TRANS_GOOGLE]: {
|
||||||
url: "https://translate.googleapis.com/translate_a/single",
|
url: URL_GOOGLE_TRAN,
|
||||||
key: "",
|
key: "",
|
||||||
fetchLimit: DEFAULT_FETCH_LIMIT, // 最大任务数量
|
fetchLimit: DEFAULT_FETCH_LIMIT, // 最大任务数量
|
||||||
fetchInterval: DEFAULT_FETCH_INTERVAL, // 任务间隔时间
|
fetchInterval: DEFAULT_FETCH_INTERVAL, // 任务间隔时间
|
||||||
@@ -507,16 +579,11 @@ export const DEFAULT_TRANS_APIS = {
|
|||||||
fetchLimit: DEFAULT_FETCH_LIMIT,
|
fetchLimit: DEFAULT_FETCH_LIMIT,
|
||||||
fetchInterval: DEFAULT_FETCH_INTERVAL,
|
fetchInterval: DEFAULT_FETCH_INTERVAL,
|
||||||
},
|
},
|
||||||
[OPT_TRANS_OPENAI]: {
|
[OPT_TRANS_OPENAI]: defaultOpenaiApi,
|
||||||
url: "https://api.openai.com/v1/chat/completions",
|
[OPT_TRANS_OPENAI_2]: defaultOpenaiApi,
|
||||||
key: "",
|
[OPT_TRANS_OPENAI_3]: defaultOpenaiApi,
|
||||||
model: "gpt-4",
|
|
||||||
prompt: `You will be provided with a sentence in ${INPUT_PLACE_FROM}, and your task is to translate it into ${INPUT_PLACE_TO}.`,
|
|
||||||
fetchLimit: 1,
|
|
||||||
fetchInterval: 500,
|
|
||||||
},
|
|
||||||
[OPT_TRANS_GEMINI]: {
|
[OPT_TRANS_GEMINI]: {
|
||||||
url: "https://generativelanguage.googleapis.com/v1/models",
|
url: `https://generativelanguage.googleapis.com/v1/models/${INPUT_PLACE_MODEL}:generateContent?key=${INPUT_PLACE_KEY}`,
|
||||||
key: "",
|
key: "",
|
||||||
model: "gemini-pro",
|
model: "gemini-pro",
|
||||||
prompt: `Translate the following text from ${INPUT_PLACE_FROM} to ${INPUT_PLACE_TO}:\n\n${INPUT_PLACE_TEXT}`,
|
prompt: `Translate the following text from ${INPUT_PLACE_FROM} to ${INPUT_PLACE_TO}:\n\n${INPUT_PLACE_TEXT}`,
|
||||||
@@ -524,11 +591,14 @@ export const DEFAULT_TRANS_APIS = {
|
|||||||
fetchInterval: 500,
|
fetchInterval: 500,
|
||||||
},
|
},
|
||||||
[OPT_TRANS_CLOUDFLAREAI]: {
|
[OPT_TRANS_CLOUDFLAREAI]: {
|
||||||
url: "https://api.cloudflare.com/client/v4/accounts/{ACCOUNT_ID}/ai/run/@cf/meta/m2m100-1.2b",
|
url: "https://api.cloudflare.com/client/v4/accounts/{{ACCOUNT_ID}}/ai/run/@cf/meta/m2m100-1.2b",
|
||||||
key: "",
|
key: "",
|
||||||
fetchLimit: 1,
|
fetchLimit: 1,
|
||||||
fetchInterval: 500,
|
fetchInterval: 500,
|
||||||
},
|
},
|
||||||
|
[OPT_TRANS_OLLAMA]: defaultOllamaApi,
|
||||||
|
[OPT_TRANS_OLLAMA_2]: defaultOllamaApi,
|
||||||
|
[OPT_TRANS_OLLAMA_3]: defaultOllamaApi,
|
||||||
[OPT_TRANS_CUSTOMIZE]: defaultCustomApi,
|
[OPT_TRANS_CUSTOMIZE]: defaultCustomApi,
|
||||||
[OPT_TRANS_CUSTOMIZE_2]: defaultCustomApi,
|
[OPT_TRANS_CUSTOMIZE_2]: defaultCustomApi,
|
||||||
[OPT_TRANS_CUSTOMIZE_3]: defaultCustomApi,
|
[OPT_TRANS_CUSTOMIZE_3]: defaultCustomApi,
|
||||||
@@ -589,6 +659,7 @@ export const DEFAULT_SETTING = {
|
|||||||
csplist: DEFAULT_CSPLIST.join(",\n"), // 禁用CSP名单
|
csplist: DEFAULT_CSPLIST.join(",\n"), // 禁用CSP名单
|
||||||
// disableLangs: [], // 不翻译的语言(移至rule,作废)
|
// disableLangs: [], // 不翻译的语言(移至rule,作废)
|
||||||
transInterval: 500, // 翻译间隔时间
|
transInterval: 500, // 翻译间隔时间
|
||||||
|
langDetector: OPT_TRANS_MICROSOFT, // 远程语言识别服务
|
||||||
};
|
};
|
||||||
|
|
||||||
export const DEFAULT_RULES = [GLOBLA_RULE];
|
export const DEFAULT_RULES = [GLOBLA_RULE];
|
||||||
|
|||||||
@@ -30,6 +30,9 @@ export const DEFAULT_RULE = {
|
|||||||
skipLangs: [], // 不翻译的语言
|
skipLangs: [], // 不翻译的语言
|
||||||
fixerSelector: "", // 修复函数选择器
|
fixerSelector: "", // 修复函数选择器
|
||||||
fixerFunc: GLOBAL_KEY, // 修复函数
|
fixerFunc: GLOBAL_KEY, // 修复函数
|
||||||
|
transStartHook: "", // 钩子函数
|
||||||
|
transEndHook: "", // 钩子函数
|
||||||
|
transRemoveHook: "", // 钩子函数
|
||||||
};
|
};
|
||||||
|
|
||||||
const DEFAULT_DIY_STYLE = `color: #666;
|
const DEFAULT_DIY_STYLE = `color: #666;
|
||||||
|
|||||||
@@ -30,7 +30,11 @@ export function useTranslate(q, rule, setting) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const deLang = await tryDetectLang(q, detectRemote === "true");
|
const deLang = await tryDetectLang(
|
||||||
|
q,
|
||||||
|
detectRemote === "true",
|
||||||
|
setting.langDetector
|
||||||
|
);
|
||||||
if (deLang && (toLang.includes(deLang) || skipLangs.includes(deLang))) {
|
if (deLang && (toLang.includes(deLang) || skipLangs.includes(deLang))) {
|
||||||
setSamelang(true);
|
setSamelang(true);
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -1,8 +1,26 @@
|
|||||||
import { CACHE_NAME } from "../config";
|
import {
|
||||||
|
CACHE_NAME,
|
||||||
|
OPT_TRANS_GOOGLE,
|
||||||
|
OPT_TRANS_MICROSOFT,
|
||||||
|
OPT_TRANS_BAIDU,
|
||||||
|
OPT_TRANS_TENCENT,
|
||||||
|
} from "../config";
|
||||||
import { browser } from "./browser";
|
import { browser } from "./browser";
|
||||||
import { apiBaiduLangdetect } from "../apis";
|
import {
|
||||||
|
apiGoogleLangdetect,
|
||||||
|
apiMicrosoftLangdetect,
|
||||||
|
apiBaiduLangdetect,
|
||||||
|
apiTencentLangdetect,
|
||||||
|
} from "../apis";
|
||||||
import { kissLog } from "./log";
|
import { kissLog } from "./log";
|
||||||
|
|
||||||
|
const langdetectMap = {
|
||||||
|
[OPT_TRANS_GOOGLE]: apiGoogleLangdetect,
|
||||||
|
[OPT_TRANS_MICROSOFT]: apiMicrosoftLangdetect,
|
||||||
|
[OPT_TRANS_BAIDU]: apiBaiduLangdetect,
|
||||||
|
[OPT_TRANS_TENCENT]: apiTencentLangdetect,
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 清除缓存数据
|
* 清除缓存数据
|
||||||
*/
|
*/
|
||||||
@@ -19,12 +37,16 @@ export const tryClearCaches = async () => {
|
|||||||
* @param {*} q
|
* @param {*} q
|
||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
export const tryDetectLang = async (q, useRemote = false) => {
|
export const tryDetectLang = async (
|
||||||
|
q,
|
||||||
|
useRemote = false,
|
||||||
|
langDetector = OPT_TRANS_MICROSOFT
|
||||||
|
) => {
|
||||||
let lang = "";
|
let lang = "";
|
||||||
|
|
||||||
if (useRemote) {
|
if (useRemote) {
|
||||||
try {
|
try {
|
||||||
lang = await apiBaiduLangdetect(q);
|
lang = await langdetectMap[langDetector](q);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
kissLog(err, "detect lang remote");
|
kissLog(err, "detect lang remote");
|
||||||
}
|
}
|
||||||
|
|||||||
16
src/libs/interpreter.js
Normal file
16
src/libs/interpreter.js
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import Sval from "sval";
|
||||||
|
|
||||||
|
const interpreter = new Sval({
|
||||||
|
// ECMA Version of the code
|
||||||
|
// 3 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15
|
||||||
|
// or 2015 | 2016 | 2017 | 2018 | 2019 | 2020 | 2021 | 2022 | 2023 | 2024
|
||||||
|
// or "latest"
|
||||||
|
ecmaVer: "latest",
|
||||||
|
// Code source type
|
||||||
|
// "script" or "module"
|
||||||
|
sourceType: "script",
|
||||||
|
// Whether the code runs in a sandbox
|
||||||
|
sandBox: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default interpreter;
|
||||||
@@ -74,6 +74,9 @@ export const matchRule = async (
|
|||||||
"injectJs",
|
"injectJs",
|
||||||
"injectCss",
|
"injectCss",
|
||||||
"fixerSelector",
|
"fixerSelector",
|
||||||
|
"transStartHook",
|
||||||
|
"transEndHook",
|
||||||
|
"transRemoveHook",
|
||||||
].forEach((key) => {
|
].forEach((key) => {
|
||||||
if (!rule[key]?.trim()) {
|
if (!rule[key]?.trim()) {
|
||||||
rule[key] = globalRule[key];
|
rule[key] = globalRule[key];
|
||||||
@@ -162,6 +165,9 @@ export const checkRules = (rules) => {
|
|||||||
skipLangs,
|
skipLangs,
|
||||||
fixerSelector,
|
fixerSelector,
|
||||||
fixerFunc,
|
fixerFunc,
|
||||||
|
transStartHook,
|
||||||
|
transEndHook,
|
||||||
|
transRemoveHook,
|
||||||
}) => ({
|
}) => ({
|
||||||
pattern: pattern.trim(),
|
pattern: pattern.trim(),
|
||||||
selector: type(selector) === "string" ? selector : "",
|
selector: type(selector) === "string" ? selector : "",
|
||||||
@@ -185,6 +191,10 @@ export const checkRules = (rules) => {
|
|||||||
detectRemote: matchValue([GLOBAL_KEY, "true", "false"], detectRemote),
|
detectRemote: matchValue([GLOBAL_KEY, "true", "false"], detectRemote),
|
||||||
skipLangs: type(skipLangs) === "array" ? skipLangs : [],
|
skipLangs: type(skipLangs) === "array" ? skipLangs : [],
|
||||||
fixerSelector: type(fixerSelector) === "string" ? fixerSelector : "",
|
fixerSelector: type(fixerSelector) === "string" ? fixerSelector : "",
|
||||||
|
transStartHook: type(transStartHook) === "string" ? transStartHook : "",
|
||||||
|
transEndHook: type(transEndHook) === "string" ? transEndHook : "",
|
||||||
|
transRemoveHook:
|
||||||
|
type(transRemoveHook) === "string" ? transRemoveHook : "",
|
||||||
fixerFunc: matchValue([GLOBAL_KEY, ...FIXER_ALL], fixerFunc),
|
fixerFunc: matchValue([GLOBAL_KEY, ...FIXER_ALL], fixerFunc),
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ import { sendBgMsg } from "./msg";
|
|||||||
import { isExt } from "./client";
|
import { isExt } from "./client";
|
||||||
import { injectInlineJs, injectInternalCss } from "./injector";
|
import { injectInlineJs, injectInternalCss } from "./injector";
|
||||||
import { kissLog } from "./log";
|
import { kissLog } from "./log";
|
||||||
|
import interpreter from "./interpreter";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 翻译类
|
* 翻译类
|
||||||
@@ -405,6 +406,7 @@ export class Translator {
|
|||||||
// 移除键盘监听
|
// 移除键盘监听
|
||||||
window.removeEventListener("keydown", this._handleKeydown);
|
window.removeEventListener("keydown", this._handleKeydown);
|
||||||
|
|
||||||
|
const { transRemoveHook } = this._rule;
|
||||||
this._tranNodes.forEach((innerHTML, node) => {
|
this._tranNodes.forEach((innerHTML, node) => {
|
||||||
if (
|
if (
|
||||||
!this._rule.transTiming ||
|
!this._rule.transTiming ||
|
||||||
@@ -420,11 +422,18 @@ export class Translator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 移除/恢复元素
|
// 移除/恢复元素
|
||||||
if (innerHTML && this._rule.transOnly === "true") {
|
if (innerHTML) {
|
||||||
|
if (this._rule.transOnly === "true") {
|
||||||
node.innerHTML = innerHTML;
|
node.innerHTML = innerHTML;
|
||||||
} else {
|
} else {
|
||||||
node.querySelector(APP_LCNAME)?.remove();
|
node.querySelector(APP_LCNAME)?.remove();
|
||||||
}
|
}
|
||||||
|
// 钩子函数
|
||||||
|
if (transRemoveHook?.trim()) {
|
||||||
|
interpreter.run(`exports.transRemoveHook = ${transRemoveHook}`);
|
||||||
|
interpreter.exports.transRemoveHook(node);
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// 移除用户JS/CSS
|
// 移除用户JS/CSS
|
||||||
@@ -490,6 +499,13 @@ export class Translator {
|
|||||||
}
|
}
|
||||||
const keeps = [];
|
const keeps = [];
|
||||||
|
|
||||||
|
// 翻译开始钩子函数
|
||||||
|
const { transStartHook } = this._rule;
|
||||||
|
if (transStartHook?.trim()) {
|
||||||
|
interpreter.run(`exports.transStartHook = ${transStartHook}`);
|
||||||
|
interpreter.exports.transStartHook(el, q);
|
||||||
|
}
|
||||||
|
|
||||||
// 保留元素
|
// 保留元素
|
||||||
const [matchSelector, subSelector] = this._keepSelector;
|
const [matchSelector, subSelector] = this._keepSelector;
|
||||||
if (matchSelector || subSelector) {
|
if (matchSelector || subSelector) {
|
||||||
@@ -538,18 +554,22 @@ export class Translator {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
traEl = document.createElement(APP_LCNAME);
|
// 附加样式
|
||||||
traEl.style.visibility = "visible";
|
|
||||||
// if (this._rule.transOnly === "true") {
|
|
||||||
// el.innerHTML = "";
|
|
||||||
// }
|
|
||||||
const { selectStyle, parentStyle } = this._rule;
|
const { selectStyle, parentStyle } = this._rule;
|
||||||
el.appendChild(traEl);
|
|
||||||
el.style.cssText += selectStyle;
|
el.style.cssText += selectStyle;
|
||||||
if (el.parentElement) {
|
if (el.parentElement) {
|
||||||
el.parentElement.style.cssText += parentStyle;
|
el.parentElement.style.cssText += parentStyle;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 插入译文节点
|
||||||
|
traEl = document.createElement(APP_LCNAME);
|
||||||
|
traEl.style.visibility = "visible";
|
||||||
|
// if (this._rule.transOnly === "true") {
|
||||||
|
// el.innerHTML = "";
|
||||||
|
// }
|
||||||
|
el.appendChild(traEl);
|
||||||
|
|
||||||
|
// 渲染译文节点
|
||||||
const root = createRoot(traEl);
|
const root = createRoot(traEl);
|
||||||
root.render(<Content q={q} keeps={keeps} translator={this} $el={el} />);
|
root.render(<Content q={q} keeps={keeps} translator={this} $el={el} />);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -15,6 +15,16 @@ export const limitNumber = (num, min = 0, max = 100) => {
|
|||||||
return number;
|
return number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const limitFloat = (num, min = 0, max = 100) => {
|
||||||
|
const number = parseFloat(num);
|
||||||
|
if (Number.isNaN(number) || number < min) {
|
||||||
|
return min;
|
||||||
|
} else if (number > max) {
|
||||||
|
return max;
|
||||||
|
}
|
||||||
|
return number;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 匹配是否为数组中的值
|
* 匹配是否为数组中的值
|
||||||
* @param {*} arr
|
* @param {*} arr
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import {
|
|||||||
import { useTranslate } from "../../hooks/Translate";
|
import { useTranslate } from "../../hooks/Translate";
|
||||||
import { styled, css } from "@mui/material/styles";
|
import { styled, css } from "@mui/material/styles";
|
||||||
import { APP_LCNAME } from "../../config";
|
import { APP_LCNAME } from "../../config";
|
||||||
|
import interpreter from "../../libs/interpreter";
|
||||||
|
|
||||||
const LINE_STYLES = {
|
const LINE_STYLES = {
|
||||||
[OPT_STYLE_LINE]: "solid",
|
[OPT_STYLE_LINE]: "solid",
|
||||||
@@ -85,8 +86,15 @@ const StyledSpan = styled("span")`
|
|||||||
export default function Content({ q, keeps, translator, $el }) {
|
export default function Content({ q, keeps, translator, $el }) {
|
||||||
const [rule, setRule] = useState(translator.rule);
|
const [rule, setRule] = useState(translator.rule);
|
||||||
const { text, sameLang, loading } = useTranslate(q, rule, translator.setting);
|
const { text, sameLang, loading } = useTranslate(q, rule, translator.setting);
|
||||||
const { transOpen, textStyle, bgColor, textDiyStyle, transOnly, transTag } =
|
const {
|
||||||
rule;
|
transOpen,
|
||||||
|
textStyle,
|
||||||
|
bgColor,
|
||||||
|
textDiyStyle,
|
||||||
|
transOnly,
|
||||||
|
transTag,
|
||||||
|
transEndHook,
|
||||||
|
} = rule;
|
||||||
|
|
||||||
const { newlineLength } = translator.setting;
|
const { newlineLength } = translator.setting;
|
||||||
|
|
||||||
@@ -107,6 +115,14 @@ export default function Content({ q, keeps, translator, $el }) {
|
|||||||
};
|
};
|
||||||
}, [translator.eventName]);
|
}, [translator.eventName]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// 运行钩子函数
|
||||||
|
if (text && transEndHook?.trim()) {
|
||||||
|
interpreter.run(`exports.transEndHook = ${transEndHook}`);
|
||||||
|
interpreter.exports.transEndHook($el, q, text, keeps);
|
||||||
|
}
|
||||||
|
}, [$el, q, text, keeps, transEndHook]);
|
||||||
|
|
||||||
const gap = useMemo(() => {
|
const gap = useMemo(() => {
|
||||||
if (transOnly === "true") {
|
if (transOnly === "true") {
|
||||||
return "";
|
return "";
|
||||||
|
|||||||
@@ -11,8 +11,13 @@ import {
|
|||||||
OPT_TRANS_BAIDU,
|
OPT_TRANS_BAIDU,
|
||||||
OPT_TRANS_TENCENT,
|
OPT_TRANS_TENCENT,
|
||||||
OPT_TRANS_OPENAI,
|
OPT_TRANS_OPENAI,
|
||||||
|
OPT_TRANS_OPENAI_2,
|
||||||
|
OPT_TRANS_OPENAI_3,
|
||||||
OPT_TRANS_GEMINI,
|
OPT_TRANS_GEMINI,
|
||||||
OPT_TRANS_CLOUDFLAREAI,
|
OPT_TRANS_CLOUDFLAREAI,
|
||||||
|
OPT_TRANS_OLLAMA,
|
||||||
|
OPT_TRANS_OLLAMA_2,
|
||||||
|
OPT_TRANS_OLLAMA_3,
|
||||||
OPT_TRANS_CUSTOMIZE,
|
OPT_TRANS_CUSTOMIZE,
|
||||||
OPT_TRANS_NIUTRANS,
|
OPT_TRANS_NIUTRANS,
|
||||||
URL_KISS_PROXY,
|
URL_KISS_PROXY,
|
||||||
@@ -33,7 +38,7 @@ import { useApi } from "../../hooks/Api";
|
|||||||
import { apiTranslate } from "../../apis";
|
import { apiTranslate } from "../../apis";
|
||||||
import Box from "@mui/material/Box";
|
import Box from "@mui/material/Box";
|
||||||
import Link from "@mui/material/Link";
|
import Link from "@mui/material/Link";
|
||||||
import { limitNumber } from "../../libs/utils";
|
import { limitNumber, limitFloat } from "../../libs/utils";
|
||||||
|
|
||||||
function TestButton({ translator, api }) {
|
function TestButton({ translator, api }) {
|
||||||
const i18n = useI18n();
|
const i18n = useI18n();
|
||||||
@@ -114,7 +119,10 @@ function ApiFields({ translator }) {
|
|||||||
fetchInterval = DEFAULT_FETCH_INTERVAL,
|
fetchInterval = DEFAULT_FETCH_INTERVAL,
|
||||||
dictNo = "",
|
dictNo = "",
|
||||||
memoryNo = "",
|
memoryNo = "",
|
||||||
customOption = "",
|
reqHook = "",
|
||||||
|
resHook = "",
|
||||||
|
temperature = 0,
|
||||||
|
maxTokens = 256,
|
||||||
} = api;
|
} = api;
|
||||||
|
|
||||||
const handleChange = (e) => {
|
const handleChange = (e) => {
|
||||||
@@ -126,6 +134,12 @@ function ApiFields({ translator }) {
|
|||||||
case "fetchInterval":
|
case "fetchInterval":
|
||||||
value = limitNumber(value, 0, 5000);
|
value = limitNumber(value, 0, 5000);
|
||||||
break;
|
break;
|
||||||
|
case "temperature":
|
||||||
|
value = limitFloat(value, 0, 2);
|
||||||
|
break;
|
||||||
|
case "maxTokens":
|
||||||
|
value = limitNumber(value, 0, 2 ** 15);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
updateApi({
|
updateApi({
|
||||||
@@ -133,7 +147,7 @@ function ApiFields({ translator }) {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const buildinTranslators = [
|
const builtinTranslators = [
|
||||||
OPT_TRANS_MICROSOFT,
|
OPT_TRANS_MICROSOFT,
|
||||||
OPT_TRANS_DEEPLFREE,
|
OPT_TRANS_DEEPLFREE,
|
||||||
OPT_TRANS_BAIDU,
|
OPT_TRANS_BAIDU,
|
||||||
@@ -143,8 +157,13 @@ function ApiFields({ translator }) {
|
|||||||
const mulkeysTranslators = [
|
const mulkeysTranslators = [
|
||||||
OPT_TRANS_DEEPL,
|
OPT_TRANS_DEEPL,
|
||||||
OPT_TRANS_OPENAI,
|
OPT_TRANS_OPENAI,
|
||||||
|
OPT_TRANS_OPENAI_2,
|
||||||
|
OPT_TRANS_OPENAI_3,
|
||||||
OPT_TRANS_GEMINI,
|
OPT_TRANS_GEMINI,
|
||||||
OPT_TRANS_CLOUDFLAREAI,
|
OPT_TRANS_CLOUDFLAREAI,
|
||||||
|
OPT_TRANS_OLLAMA,
|
||||||
|
OPT_TRANS_OLLAMA_2,
|
||||||
|
OPT_TRANS_OLLAMA_3,
|
||||||
OPT_TRANS_NIUTRANS,
|
OPT_TRANS_NIUTRANS,
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -164,7 +183,7 @@ function ApiFields({ translator }) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack spacing={3}>
|
<Stack spacing={3}>
|
||||||
{!buildinTranslators.includes(translator) && (
|
{!builtinTranslators.includes(translator) && (
|
||||||
<>
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
size="small"
|
size="small"
|
||||||
@@ -191,7 +210,9 @@ function ApiFields({ translator }) {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{(translator === OPT_TRANS_OPENAI || translator === OPT_TRANS_GEMINI) && (
|
{(translator.startsWith(OPT_TRANS_OPENAI) ||
|
||||||
|
translator.startsWith(OPT_TRANS_OLLAMA) ||
|
||||||
|
translator === OPT_TRANS_GEMINI) && (
|
||||||
<>
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
size="small"
|
size="small"
|
||||||
@@ -212,6 +233,27 @@ function ApiFields({ translator }) {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{translator.startsWith(OPT_TRANS_OPENAI) && (
|
||||||
|
<>
|
||||||
|
<TextField
|
||||||
|
size="small"
|
||||||
|
label={"Temperature"}
|
||||||
|
type="number"
|
||||||
|
name="temperature"
|
||||||
|
value={temperature}
|
||||||
|
onChange={handleChange}
|
||||||
|
/>
|
||||||
|
<TextField
|
||||||
|
size="small"
|
||||||
|
label={"Max Tokens"}
|
||||||
|
type="number"
|
||||||
|
name="maxTokens"
|
||||||
|
value={maxTokens}
|
||||||
|
onChange={handleChange}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
{translator === OPT_TRANS_NIUTRANS && (
|
{translator === OPT_TRANS_NIUTRANS && (
|
||||||
<>
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
@@ -232,15 +274,26 @@ function ApiFields({ translator }) {
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{translator.startsWith(OPT_TRANS_CUSTOMIZE) && (
|
{translator.startsWith(OPT_TRANS_CUSTOMIZE) && (
|
||||||
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
size="small"
|
size="small"
|
||||||
label={i18n("custom_option")}
|
label={"Request Hook"}
|
||||||
name="customOption"
|
name="reqHook"
|
||||||
value={customOption}
|
value={reqHook}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
multiline
|
multiline
|
||||||
maxRows={10}
|
maxRows={10}
|
||||||
/>
|
/>
|
||||||
|
<TextField
|
||||||
|
size="small"
|
||||||
|
label={"Response Hook"}
|
||||||
|
name="resHook"
|
||||||
|
value={resHook}
|
||||||
|
onChange={handleChange}
|
||||||
|
multiline
|
||||||
|
maxRows={10}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<TextField
|
<TextField
|
||||||
|
|||||||
@@ -97,6 +97,9 @@ function RuleFields({ rule, rules, setShow, setKeyword }) {
|
|||||||
skipLangs = [],
|
skipLangs = [],
|
||||||
fixerSelector = "",
|
fixerSelector = "",
|
||||||
fixerFunc = "-",
|
fixerFunc = "-",
|
||||||
|
transStartHook = "",
|
||||||
|
transEndHook = "",
|
||||||
|
transRemoveHook = "",
|
||||||
} = formValues;
|
} = formValues;
|
||||||
|
|
||||||
const hasSamePattern = (str) => {
|
const hasSamePattern = (str) => {
|
||||||
@@ -458,6 +461,7 @@ function RuleFields({ rule, rules, setShow, setKeyword }) {
|
|||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
multiline
|
multiline
|
||||||
|
maxRows={10}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<TextField
|
<TextField
|
||||||
@@ -468,6 +472,7 @@ function RuleFields({ rule, rules, setShow, setKeyword }) {
|
|||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
multiline
|
multiline
|
||||||
|
maxRows={10}
|
||||||
/>
|
/>
|
||||||
<TextField
|
<TextField
|
||||||
select
|
select
|
||||||
@@ -487,6 +492,40 @@ function RuleFields({ rule, rules, setShow, setKeyword }) {
|
|||||||
))}
|
))}
|
||||||
</TextField>
|
</TextField>
|
||||||
|
|
||||||
|
<TextField
|
||||||
|
size="small"
|
||||||
|
label={i18n("translate_start_hook")}
|
||||||
|
helperText={i18n("translate_start_hook_helper")}
|
||||||
|
name="transStartHook"
|
||||||
|
value={transStartHook}
|
||||||
|
disabled={disabled}
|
||||||
|
onChange={handleChange}
|
||||||
|
multiline
|
||||||
|
maxRows={10}
|
||||||
|
/>
|
||||||
|
<TextField
|
||||||
|
size="small"
|
||||||
|
label={i18n("translate_end_hook")}
|
||||||
|
helperText={i18n("translate_end_hook_helper")}
|
||||||
|
name="transEndHook"
|
||||||
|
value={transEndHook}
|
||||||
|
disabled={disabled}
|
||||||
|
onChange={handleChange}
|
||||||
|
multiline
|
||||||
|
maxRows={10}
|
||||||
|
/>
|
||||||
|
<TextField
|
||||||
|
size="small"
|
||||||
|
label={i18n("translate_remove_hook")}
|
||||||
|
helperText={i18n("translate_remove_hook_helper")}
|
||||||
|
name="transRemoveHook"
|
||||||
|
value={transRemoveHook}
|
||||||
|
disabled={disabled}
|
||||||
|
onChange={handleChange}
|
||||||
|
multiline
|
||||||
|
maxRows={10}
|
||||||
|
/>
|
||||||
|
|
||||||
<TextField
|
<TextField
|
||||||
size="small"
|
size="small"
|
||||||
label={i18n("selector_style")}
|
label={i18n("selector_style")}
|
||||||
|
|||||||
@@ -17,6 +17,8 @@ import {
|
|||||||
UI_LANGS,
|
UI_LANGS,
|
||||||
TRANS_NEWLINE_LENGTH,
|
TRANS_NEWLINE_LENGTH,
|
||||||
CACHE_NAME,
|
CACHE_NAME,
|
||||||
|
OPT_TRANS_MICROSOFT,
|
||||||
|
OPT_LANGDETECTOR_ALL,
|
||||||
OPT_SHORTCUT_TRANSLATE,
|
OPT_SHORTCUT_TRANSLATE,
|
||||||
OPT_SHORTCUT_STYLE,
|
OPT_SHORTCUT_STYLE,
|
||||||
OPT_SHORTCUT_POPUP,
|
OPT_SHORTCUT_POPUP,
|
||||||
@@ -31,6 +33,8 @@ import ShortcutInput from "./ShortcutInput";
|
|||||||
import { useFab } from "../../hooks/Fab";
|
import { useFab } from "../../hooks/Fab";
|
||||||
import { sendBgMsg } from "../../libs/msg";
|
import { sendBgMsg } from "../../libs/msg";
|
||||||
import { kissLog } from "../../libs/log";
|
import { kissLog } from "../../libs/log";
|
||||||
|
import UploadButton from "./UploadButton";
|
||||||
|
import DownloadButton from "./DownloadButton";
|
||||||
|
|
||||||
function ShortcutItem({ action, label }) {
|
function ShortcutItem({ action, label }) {
|
||||||
const { shortcut, setShortcut } = useShortcut(action);
|
const { shortcut, setShortcut } = useShortcut(action);
|
||||||
@@ -92,6 +96,14 @@ export default function Settings() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleImport = async (data) => {
|
||||||
|
try {
|
||||||
|
await updateSetting(JSON.parse(data));
|
||||||
|
} catch (err) {
|
||||||
|
kissLog(err, "import setting");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const {
|
const {
|
||||||
uiLang,
|
uiLang,
|
||||||
minLength,
|
minLength,
|
||||||
@@ -103,12 +115,28 @@ export default function Settings() {
|
|||||||
blacklist = DEFAULT_BLACKLIST.join(",\n"),
|
blacklist = DEFAULT_BLACKLIST.join(",\n"),
|
||||||
csplist = DEFAULT_CSPLIST.join(",\n"),
|
csplist = DEFAULT_CSPLIST.join(",\n"),
|
||||||
transInterval = 500,
|
transInterval = 500,
|
||||||
|
langDetector = OPT_TRANS_MICROSOFT,
|
||||||
} = setting;
|
} = setting;
|
||||||
const { isHide = false } = fab || {};
|
const { isHide = false } = fab || {};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box>
|
<Box>
|
||||||
<Stack spacing={3}>
|
<Stack spacing={3}>
|
||||||
|
<Stack
|
||||||
|
direction="row"
|
||||||
|
alignItems="center"
|
||||||
|
spacing={2}
|
||||||
|
useFlexGap
|
||||||
|
flexWrap="wrap"
|
||||||
|
>
|
||||||
|
<UploadButton text={i18n("import")} handleImport={handleImport} />
|
||||||
|
<DownloadButton
|
||||||
|
handleData={() => JSON.stringify(setting, null, 2)}
|
||||||
|
text={i18n("export")}
|
||||||
|
fileName={`kiss-setting_${Date.now()}.json`}
|
||||||
|
/>
|
||||||
|
</Stack>
|
||||||
|
|
||||||
<FormControl size="small">
|
<FormControl size="small">
|
||||||
<InputLabel>{i18n("ui_lang")}</InputLabel>
|
<InputLabel>{i18n("ui_lang")}</InputLabel>
|
||||||
<Select
|
<Select
|
||||||
@@ -206,6 +234,22 @@ export default function Settings() {
|
|||||||
</Select>
|
</Select>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|
||||||
|
<FormControl size="small">
|
||||||
|
<InputLabel>{i18n("detect_lang_remote")}</InputLabel>
|
||||||
|
<Select
|
||||||
|
name="langDetector"
|
||||||
|
value={langDetector}
|
||||||
|
label={i18n("detect_lang_remote")}
|
||||||
|
onChange={handleChange}
|
||||||
|
>
|
||||||
|
{OPT_LANGDETECTOR_ALL.map((item) => (
|
||||||
|
<MenuItem value={item} key={item}>
|
||||||
|
{item}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</Select>
|
||||||
|
</FormControl>
|
||||||
|
|
||||||
{isExt ? (
|
{isExt ? (
|
||||||
<>
|
<>
|
||||||
<FormControl size="small">
|
<FormControl size="small">
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import {
|
|||||||
OPT_LANGS_TO,
|
OPT_LANGS_TO,
|
||||||
OPT_TRANBOX_TRIGGER_CLICK,
|
OPT_TRANBOX_TRIGGER_CLICK,
|
||||||
OPT_TRANBOX_TRIGGER_ALL,
|
OPT_TRANBOX_TRIGGER_ALL,
|
||||||
|
OPT_DICT_BAIDU,
|
||||||
} from "../../config";
|
} from "../../config";
|
||||||
import ShortcutInput from "./ShortcutInput";
|
import ShortcutInput from "./ShortcutInput";
|
||||||
import FormControlLabel from "@mui/material/FormControlLabel";
|
import FormControlLabel from "@mui/material/FormControlLabel";
|
||||||
@@ -63,6 +64,7 @@ export default function Tranbox() {
|
|||||||
followSelection = false,
|
followSelection = false,
|
||||||
triggerMode = OPT_TRANBOX_TRIGGER_CLICK,
|
triggerMode = OPT_TRANBOX_TRIGGER_CLICK,
|
||||||
extStyles = "",
|
extStyles = "",
|
||||||
|
enDict = OPT_DICT_BAIDU,
|
||||||
} = tranboxSetting;
|
} = tranboxSetting;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -143,6 +145,18 @@ export default function Tranbox() {
|
|||||||
))}
|
))}
|
||||||
</TextField>
|
</TextField>
|
||||||
|
|
||||||
|
<TextField
|
||||||
|
select
|
||||||
|
size="small"
|
||||||
|
name="enDict"
|
||||||
|
value={enDict}
|
||||||
|
label={i18n("english_dict")}
|
||||||
|
onChange={handleChange}
|
||||||
|
>
|
||||||
|
<MenuItem value={"-"}>{i18n("disable")}</MenuItem>
|
||||||
|
<MenuItem value={OPT_DICT_BAIDU}>{OPT_DICT_BAIDU}</MenuItem>
|
||||||
|
</TextField>
|
||||||
|
|
||||||
<TextField
|
<TextField
|
||||||
size="small"
|
size="small"
|
||||||
label={i18n("tranbtn_offset_x")}
|
label={i18n("tranbtn_offset_x")}
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ import DictCont from "./DictCont";
|
|||||||
import SugCont from "./SugCont";
|
import SugCont from "./SugCont";
|
||||||
import CopyBtn from "./CopyBtn";
|
import CopyBtn from "./CopyBtn";
|
||||||
import { isValidWord } from "../../libs/utils";
|
import { isValidWord } from "../../libs/utils";
|
||||||
|
import { isMobile } from "../../libs/mobile";
|
||||||
|
|
||||||
function Header({
|
function Header({
|
||||||
setShowPopup,
|
setShowPopup,
|
||||||
@@ -34,7 +35,12 @@ function Header({
|
|||||||
setHideClickAway,
|
setHideClickAway,
|
||||||
followSelection,
|
followSelection,
|
||||||
setFollowSelection,
|
setFollowSelection,
|
||||||
|
mouseHover,
|
||||||
}) {
|
}) {
|
||||||
|
if (!isMobile && simpleStyle && !mouseHover) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<Box
|
||||||
className="KT-transbox-header"
|
className="KT-transbox-header"
|
||||||
@@ -95,7 +101,15 @@ function Header({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function TranForm({ text, setText, tranboxSetting, transApis, simpleStyle }) {
|
function TranForm({
|
||||||
|
text,
|
||||||
|
setText,
|
||||||
|
tranboxSetting,
|
||||||
|
transApis,
|
||||||
|
simpleStyle,
|
||||||
|
langDetector,
|
||||||
|
enDict,
|
||||||
|
}) {
|
||||||
const i18n = useI18n();
|
const i18n = useI18n();
|
||||||
|
|
||||||
const [editMode, setEditMode] = useState(false);
|
const [editMode, setEditMode] = useState(false);
|
||||||
@@ -227,7 +241,10 @@ function TranForm({ text, setText, tranboxSetting, transApis, simpleStyle }) {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{(!simpleStyle || !isValidWord(text) || !toLang.startsWith("zh")) && (
|
{(!simpleStyle ||
|
||||||
|
!isValidWord(text) ||
|
||||||
|
!toLang.startsWith("zh") ||
|
||||||
|
enDict === "-") && (
|
||||||
<TranCont
|
<TranCont
|
||||||
text={text}
|
text={text}
|
||||||
translator={translator}
|
translator={translator}
|
||||||
@@ -236,11 +253,16 @@ function TranForm({ text, setText, tranboxSetting, transApis, simpleStyle }) {
|
|||||||
toLang2={tranboxSetting.toLang2}
|
toLang2={tranboxSetting.toLang2}
|
||||||
transApis={transApis}
|
transApis={transApis}
|
||||||
simpleStyle={simpleStyle}
|
simpleStyle={simpleStyle}
|
||||||
|
langDetector={langDetector}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{enDict !== "-" && (
|
||||||
|
<>
|
||||||
<DictCont text={text} />
|
<DictCont text={text} />
|
||||||
<SugCont text={text} />
|
<SugCont text={text} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</Stack>
|
</Stack>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -262,7 +284,10 @@ export default function TranBox({
|
|||||||
followSelection,
|
followSelection,
|
||||||
setFollowSelection,
|
setFollowSelection,
|
||||||
extStyles,
|
extStyles,
|
||||||
|
langDetector,
|
||||||
|
enDict,
|
||||||
}) {
|
}) {
|
||||||
|
const [mouseHover, setMouseHover] = useState(false);
|
||||||
return (
|
return (
|
||||||
<SettingProvider>
|
<SettingProvider>
|
||||||
<ThemeProvider styles={extStyles}>
|
<ThemeProvider styles={extStyles}>
|
||||||
@@ -280,9 +305,12 @@ export default function TranBox({
|
|||||||
setHideClickAway={setHideClickAway}
|
setHideClickAway={setHideClickAway}
|
||||||
followSelection={followSelection}
|
followSelection={followSelection}
|
||||||
setFollowSelection={setFollowSelection}
|
setFollowSelection={setFollowSelection}
|
||||||
|
mouseHover={mouseHover}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
onClick={(e) => e.stopPropagation()}
|
onClick={(e) => e.stopPropagation()}
|
||||||
|
onMouseEnter={() => setMouseHover(true)}
|
||||||
|
onMouseLeave={() => setMouseHover(false)}
|
||||||
>
|
>
|
||||||
<TranForm
|
<TranForm
|
||||||
text={text}
|
text={text}
|
||||||
@@ -290,6 +318,8 @@ export default function TranBox({
|
|||||||
tranboxSetting={tranboxSetting}
|
tranboxSetting={tranboxSetting}
|
||||||
transApis={transApis}
|
transApis={transApis}
|
||||||
simpleStyle={simpleStyle}
|
simpleStyle={simpleStyle}
|
||||||
|
langDetector={langDetector}
|
||||||
|
enDict={enDict}
|
||||||
/>
|
/>
|
||||||
</DraggableResizable>
|
</DraggableResizable>
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
|
|||||||
@@ -5,10 +5,11 @@ import Stack from "@mui/material/Stack";
|
|||||||
import { useI18n } from "../../hooks/I18n";
|
import { useI18n } from "../../hooks/I18n";
|
||||||
import { DEFAULT_TRANS_APIS } from "../../config";
|
import { DEFAULT_TRANS_APIS } from "../../config";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { apiTranslate, apiBaiduLangdetect } from "../../apis";
|
import { apiTranslate } from "../../apis";
|
||||||
import CopyBtn from "./CopyBtn";
|
import CopyBtn from "./CopyBtn";
|
||||||
import Typography from "@mui/material/Typography";
|
import Typography from "@mui/material/Typography";
|
||||||
import Alert from "@mui/material/Alert";
|
import Alert from "@mui/material/Alert";
|
||||||
|
import { tryDetectLang } from "../../libs";
|
||||||
|
|
||||||
export default function TranCont({
|
export default function TranCont({
|
||||||
text,
|
text,
|
||||||
@@ -18,6 +19,7 @@ export default function TranCont({
|
|||||||
toLang2 = "en",
|
toLang2 = "en",
|
||||||
transApis,
|
transApis,
|
||||||
simpleStyle,
|
simpleStyle,
|
||||||
|
langDetector,
|
||||||
}) {
|
}) {
|
||||||
const i18n = useI18n();
|
const i18n = useI18n();
|
||||||
const [trText, setTrText] = useState("");
|
const [trText, setTrText] = useState("");
|
||||||
@@ -33,7 +35,7 @@ export default function TranCont({
|
|||||||
|
|
||||||
let to = toLang;
|
let to = toLang;
|
||||||
if (toLang !== toLang2 && toLang2 !== "none") {
|
if (toLang !== toLang2 && toLang2 !== "none") {
|
||||||
const detectLang = await apiBaiduLangdetect(text);
|
const detectLang = await tryDetectLang(text, true, langDetector);
|
||||||
if (detectLang === toLang) {
|
if (detectLang === toLang) {
|
||||||
to = toLang2;
|
to = toLang2;
|
||||||
}
|
}
|
||||||
@@ -55,7 +57,7 @@ export default function TranCont({
|
|||||||
setLoading(false);
|
setLoading(false);
|
||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
}, [text, translator, fromLang, toLang, toLang2, transApis]);
|
}, [text, translator, fromLang, toLang, toLang2, transApis, langDetector]);
|
||||||
|
|
||||||
if (simpleStyle) {
|
if (simpleStyle) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import {
|
|||||||
OPT_TRANBOX_TRIGGER_CLICK,
|
OPT_TRANBOX_TRIGGER_CLICK,
|
||||||
OPT_TRANBOX_TRIGGER_HOVER,
|
OPT_TRANBOX_TRIGGER_HOVER,
|
||||||
OPT_TRANBOX_TRIGGER_SELECT,
|
OPT_TRANBOX_TRIGGER_SELECT,
|
||||||
|
OPT_DICT_BAIDU,
|
||||||
} from "../../config";
|
} from "../../config";
|
||||||
import { isMobile } from "../../libs/mobile";
|
import { isMobile } from "../../libs/mobile";
|
||||||
import { kissLog } from "../../libs/log";
|
import { kissLog } from "../../libs/log";
|
||||||
@@ -20,6 +21,7 @@ export default function Slection({
|
|||||||
tranboxSetting,
|
tranboxSetting,
|
||||||
transApis,
|
transApis,
|
||||||
uiLang,
|
uiLang,
|
||||||
|
langDetector,
|
||||||
}) {
|
}) {
|
||||||
const {
|
const {
|
||||||
hideTranBtn = false,
|
hideTranBtn = false,
|
||||||
@@ -33,6 +35,7 @@ export default function Slection({
|
|||||||
btnOffsetY,
|
btnOffsetY,
|
||||||
boxOffsetX = 0,
|
boxOffsetX = 0,
|
||||||
boxOffsetY = 10,
|
boxOffsetY = 10,
|
||||||
|
enDict = OPT_DICT_BAIDU,
|
||||||
} = tranboxSetting;
|
} = tranboxSetting;
|
||||||
|
|
||||||
const boxWidth =
|
const boxWidth =
|
||||||
@@ -234,6 +237,8 @@ export default function Slection({
|
|||||||
followSelection={followSelection}
|
followSelection={followSelection}
|
||||||
setFollowSelection={setFollowSelection}
|
setFollowSelection={setFollowSelection}
|
||||||
extStyles={extStyles}
|
extStyles={extStyles}
|
||||||
|
langDetector={langDetector}
|
||||||
|
enDict={enDict}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user