feat: supports plain text translation

This commit is contained in:
Gabe
2025-11-15 14:25:05 +08:00
parent 49a7698993
commit abcf2baad6
4 changed files with 51 additions and 6 deletions

View File

@@ -121,7 +121,7 @@ export async function run(isUserscript = false) {
// if (document?.documentElement?.tagName?.toUpperCase() !== "HTML") {
// return;
// }
if (!document?.contentType?.includes("html")) {
if (!document?.contentType?.includes("text")) {
return;
}

View File

@@ -2584,6 +2584,13 @@ export const I18N = {
ja: `原言語と目標言語が同じ場合、字幕は処理されません`,
ko: `원본 언어와 대상 언어가 동일한 경우, 자막은 처리되지 않습니다`,
},
plain_text_translate: {
zh: `纯文本翻译`,
en: `Plain text translation`,
zh_TW: `純文字翻譯`,
ja: `プレーンテキスト翻訳`,
ko: `순수 텍스트 번역`,
},
};
export const newI18n = (lang) => (key) => I18N[key]?.[lang] || "";

View File

@@ -274,8 +274,7 @@ export class Translator {
data, datalist, embed, head, iframe, input, noscript, map,
object, option, param, picture, progress,
select, script, style, track, textarea, template,
video, wbr, .notranslate, [contenteditable='true'], [translate='no'],
${Translator.KISS_IGNORE_SELECTOR}`;
video, wbr, .notranslate, [contenteditable='true'], [translate='no']`;
#setting; // 设置选项
#rule; // 规则
@@ -322,6 +321,10 @@ export class Translator {
// 忽略元素
get #ignoreSelector() {
if (this.#rule.isPlainText) {
return Translator.KISS_IGNORE_SELECTOR;
}
if (this.#rule.autoScan === "false") {
return `${Translator.KISS_IGNORE_SELECTOR}, ${this.#rule.ignoreSelector}`;
}
@@ -353,7 +356,7 @@ export class Translator {
constructor({ rule = {}, setting = {}, favWords = [] }) {
this.#setting = { ...Translator.DEFAULT_OPTIONS, ...setting };
this.#rule = { ...Translator.DEFAULT_RULE, ...rule };
this.#rule = { ...Translator.DEFAULT_RULE, ...rule, isPlainText: false };
this.#favWords = favWords;
this.#apisMap = new Map(
this.#setting.transApis.map((api) => [api.apiSlug, api])
@@ -413,6 +416,19 @@ export class Translator {
// 注入JS/CSS
this.#initInjector();
// 纯文本预处理
if (this.#rule.isPlainText) {
document
.querySelectorAll("pre")
.forEach(
(pre) =>
(pre.innerHTML = pre.innerHTML?.replace(
/(?:\r\n|\r|\n)/g,
"<br />"
))
);
}
// 查找根节点并扫描
document
.querySelectorAll(this.#rule.rootsSelector || "body")
@@ -1786,7 +1802,11 @@ export class Translator {
this.#rule[key] !== newRule[key]
) {
this.#rule[key] = newRule[key];
if (key === "autoScan" || key === "hasShadowroot") {
if (
key === "autoScan" ||
key === "hasShadowroot" ||
key === "isPlainText"
) {
needsRescan = true;
} else {
hasChanged = true;

View File

@@ -112,7 +112,10 @@ export default function PopupCont({
const handleChange = async (e) => {
try {
const { name, value } = e.target;
let { name, value, checked } = e.target;
if (name === "isPlainText") {
value = checked;
}
setRule((pre) => ({ ...pre, [name]: value }));
if (!processActions) {
@@ -204,6 +207,7 @@ export default function PopupCont({
transOnly,
hasRichText,
hasShadowroot,
isPlainText = false,
} = rule;
return (
@@ -322,6 +326,20 @@ export default function PopupCont({
label={i18n("input_translate")}
/>
</Grid>
<Grid item xs={6}>
<FormControlLabel
control={
<Switch
size="small"
name="isPlainText"
value={!isPlainText}
checked={isPlainText}
onChange={handleChange}
/>
}
label={i18n("plain_text_translate")}
/>
</Grid>
</Grid>
<TextField