Files
kiss-translator/src/common.js

183 lines
4.6 KiB
JavaScript
Raw Normal View History

2023-10-23 13:35:57 +08:00
import React from "react";
import ReactDOM from "react-dom/client";
import Action from "./views/Action";
import createCache from "@emotion/cache";
import { CacheProvider } from "@emotion/react";
import {
MSG_TRANS_TOGGLE,
MSG_TRANS_TOGGLE_STYLE,
MSG_TRANS_PUTRULE,
APP_CONSTS,
2023-10-23 13:35:57 +08:00
} from "./config";
2023-11-24 17:07:29 +08:00
import { getFabWithDefault, getSettingWithDefault } from "./libs/storage";
2023-10-23 13:35:57 +08:00
import { Translator } from "./libs/translator";
2024-02-22 17:26:22 +08:00
import { isIframe, sendIframeMsg } from "./libs/iframe";
2023-11-11 16:59:38 +08:00
import { touchTapListener } from "./libs/touch";
2023-11-24 17:07:29 +08:00
import { debounce, genEventName } from "./libs/utils";
import { handlePing, injectScript } from "./libs/gm";
import { matchRule } from "./libs/rules";
import { trySyncAllSubRules } from "./libs/subRules";
import { isInBlacklist } from "./libs/blacklist";
2023-10-23 13:35:57 +08:00
2023-11-24 17:07:29 +08:00
/**
* 油猴脚本设置页面
*/
function runSettingPage() {
if (GM?.info?.script?.grant?.includes("unsafeWindow")) {
unsafeWindow.GM = GM;
unsafeWindow.APP_INFO = {
name: process.env.REACT_APP_NAME,
version: process.env.REACT_APP_VERSION,
};
} else {
const ping = genEventName();
window.addEventListener(ping, handlePing);
// window.eval(`(${injectScript})("${ping}")`); // eslint-disable-line
const script = document.createElement("script");
script.textContent = `(${injectScript})("${ping}")`;
document.head.append(script);
}
}
2023-10-23 13:35:57 +08:00
2023-11-24 17:07:29 +08:00
/**
* iframe 页面执行
2024-02-22 17:26:22 +08:00
* @param {*} translator
2023-11-24 17:07:29 +08:00
*/
2024-02-22 17:26:22 +08:00
function runIframe(translator) {
2023-10-23 13:35:57 +08:00
window.addEventListener("message", (e) => {
const { action, args } = e.data || {};
switch (action) {
case MSG_TRANS_TOGGLE:
translator?.toggle();
break;
case MSG_TRANS_TOGGLE_STYLE:
translator?.toggleStyle();
break;
case MSG_TRANS_PUTRULE:
2024-02-22 17:26:22 +08:00
translator.updateRule(args || {});
2023-10-23 13:35:57 +08:00
break;
default:
}
});
}
2023-11-24 17:07:29 +08:00
/**
* 悬浮按钮
* @param {*} translator
* @returns
*/
async function showFab(translator) {
2023-10-23 13:35:57 +08:00
const fab = await getFabWithDefault();
2023-10-23 18:02:42 +08:00
const $action = document.createElement("div");
$action.setAttribute("id", APP_CONSTS.fabID);
2024-03-18 17:55:32 +08:00
$action.style.fontSize = "0";
$action.style.width = "0";
$action.style.height = "0";
2023-10-23 18:02:42 +08:00
document.body.parentElement.appendChild($action);
const shadowContainer = $action.attachShadow({ mode: "closed" });
const emotionRoot = document.createElement("style");
const shadowRootElement = document.createElement("div");
shadowRootElement.classList.add(`${APP_CONSTS.fabID}_warpper`);
2023-10-23 18:02:42 +08:00
shadowContainer.appendChild(emotionRoot);
shadowContainer.appendChild(shadowRootElement);
const cache = createCache({
key: APP_CONSTS.fabID,
2023-10-23 18:02:42 +08:00
prepend: true,
container: emotionRoot,
});
ReactDOM.createRoot(shadowRootElement).render(
<React.StrictMode>
<CacheProvider value={cache}>
<Action translator={translator} fab={fab} />
</CacheProvider>
</React.StrictMode>
);
}
2023-11-24 17:07:29 +08:00
/**
* 显示错误信息到页面顶部
* @param {*} message
*/
function showErr(message) {
2023-10-23 13:35:57 +08:00
const $err = document.createElement("div");
2023-11-11 16:59:38 +08:00
$err.innerText = `KISS-Translator: ${message}`;
2023-10-23 13:35:57 +08:00
$err.style.cssText = "background:red; color:#fff;";
document.body.prepend($err);
}
2023-11-11 16:59:38 +08:00
2023-11-24 17:07:29 +08:00
/**
* 监听触屏操作
* @param {*} translator
* @returns
*/
function touchOperation(translator) {
2023-11-11 16:59:38 +08:00
const { touchTranslate = 2 } = translator.setting;
if (touchTranslate === 0) {
return;
}
const handleTap = debounce(() => {
translator.toggle();
sendIframeMsg(MSG_TRANS_TOGGLE);
});
touchTapListener(handleTap, touchTranslate);
}
2023-11-24 17:07:29 +08:00
/**
* 入口函数
*/
export async function run(isUserscript = false) {
try {
const href = document.location.href;
// 设置页面
if (
isUserscript &&
(href.includes(process.env.REACT_APP_OPTIONSPAGE_DEV) ||
2024-04-12 14:33:29 +08:00
href.includes(process.env.REACT_APP_OPTIONSPAGE))
2023-11-24 17:07:29 +08:00
) {
runSettingPage();
return;
}
// 读取设置信息
const setting = await getSettingWithDefault();
// 黑名单
if (isInBlacklist(href, setting)) {
return;
}
// 翻译网页
const rule = await matchRule(href, setting);
2024-03-16 23:37:27 +08:00
const translator = new Translator(rule, setting);
2023-11-24 17:07:29 +08:00
2024-02-22 17:26:22 +08:00
// 适配iframe
if (isIframe) {
runIframe(translator);
return;
}
2023-11-24 17:07:29 +08:00
// 监听消息
2025-10-01 01:47:15 +08:00
// !isUserscript && runtimeListener(translator);
2023-11-24 17:07:29 +08:00
2023-12-18 11:46:37 +08:00
// 输入框翻译
2025-10-01 01:47:15 +08:00
// inputTranslate(setting);
2023-12-18 11:46:37 +08:00
2023-11-24 17:07:29 +08:00
// 划词翻译
2025-10-01 01:47:15 +08:00
// showTransbox(setting, rule);
2023-11-24 17:07:29 +08:00
// 浮球按钮
await showFab(translator);
// 触屏操作
touchOperation(translator);
// 同步订阅规则
isUserscript && (await trySyncAllSubRules(setting));
} catch (err) {
console.error("[KISS-Translator]", err);
showErr(err.message);
}
}