// ==UserScript==
// @name 流畅阅读
// @license GPL-3.0 license
// @namespace https://fr.unmeta.cn/
// @version 0.3
// @description 基于上下文语境的人工智能翻译引擎,为部分网站提供精准翻译,让所有人都能够拥有基于母语般的阅读体验。程序Github开源:https://github.com/Bistutu/FluentRead,欢迎 star。
// @author ThinkStu
// @match *://*/*
// @icon 
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_listValues
// @grant GM_deleteValue
// @grant GM_xmlhttpRequest
// @connect fr.unmeta.cn
// @connect 127.0.0.1
// @run-at document-end
// ==/UserScript==
// URL 相关
const METHOD = "POST";
let url = new URL(window.location.href.split('?')[0]);
// cacheKey
const checkKey = "fluent_read_check";
// 时间
const expiringTime = 86400000 / 4;
const debouncedTime = 200; // 200毫秒
// 服务端地址
const readLink = "https://fr.unmeta.cn/read";
const preReadLink = "https://fr.unmeta.cn/preread";
//const readLink = "http://127.0.0.1:80/read";
//const preReadLink = "http://127.0.0.1:80/preread";
// const url
const Maven = "mvnrepository.com";
const DockerHub = "hub.docker.com";
// 防抖包装观察函数
const debouncedObserveDOM = debounce(observeDOM, debouncedTime);
(function () {
'use strict';
// 初始化,判断是否需要清空缓存
clearCacheIfNeeded()
// 检查是否需要拉取数据
checkNeedToRun(function (shouldRun) {
if (shouldRun) {
// 添加监听器:使用MutationObserver监听DOM变化,并配置和启动观察器
const observer = new MutationObserver(function (mutations, obs) {
mutations.forEach(mutation => {
if (mutation.target === null || mutation.target === undefined) return;
// console.log("变更记录: ", mutation.target);
// 处理每个变更记录(包含 body)
if (["div", "button", "svg", "span", "nav", "body"].includes(mutation.target.tagName.toLowerCase())) {
handleDOMUpdate(mutation.target);
}
});
});
observer.observe(document.body, {childList: true, subtree: true});
handleDOMUpdate(document.body);
}
});
// remove all cache
document.addEventListener('keydown', function (event) {
if (event.key === 'F2') {
console.log('Cache cleared!');
let listValues = GM_listValues();
listValues.forEach(e => {
GM_deleteValue(e)
})
}
});
})();
// 异步返回 callback,表示是否需要拉取数据
function checkNeedToRun(callback) {
// 1、检查缓存
let PagesMapCache = GM_getValue(checkKey, undefined);
if (PagesMapCache !== undefined && PagesMapCache !== null) {
PagesMapCache[url.host] ? callback(true) : callback(false);
return;
}
// 2、网络请求
GM_xmlhttpRequest({
method: METHOD,
url: preReadLink,
onload: function (response) {
let pagesMap = JSON.parse(response.responseText).Data;
// 设置缓存
GM_setValue(checkKey, pagesMap);
pagesMap !== null && pagesMap !== undefined && pagesMap[url.host] ? callback(true) : callback(false);
},
onerror: function (error) {
console.error("请求失败: ", error);
callback(false);
}
});
}
// 初始化函数
function clearCacheIfNeeded() {
const lastRun = GM_getValue("lastRun");
const now = new Date().getTime();
if (lastRun === null || lastRun === undefined || now - lastRun > expiringTime) {
console.log("time to start");
updateCache();
GM_setValue("lastRun", now.toString());
}
}
function updateCache() {
GM_xmlhttpRequest({
method: METHOD,
url: preReadLink,
onload: handleCacheUpdateResponse,
onerror: (error) => console.error("请求失败: ", error)
});
}
function handleCacheUpdateResponse(response) {
const pagesMap = JSON.parse(response.responseText).Data;
const pageMapCache = GM_getValue(checkKey) || {};
GM_setValue(checkKey, pagesMap);
const listValues = GM_listValues();
listValues.forEach(host => {
if (pageMapCache[host] !== pagesMap[host]) {
GM_deleteValue(host);
console.log("删除缓存: ", host);
}
});
}
// 处理 DOM 更新
function handleDOMUpdate(node) {
let cached = getPageCached();
if (cached !== null && cached !== undefined) {
parseDfs(node, cached);
} else {
debouncedObserveDOM();
}
}
// 监听器配置
function observeDOM() {
sendRequest(function (respMap) {
parseDfs(document.body, respMap);
});
}
function getPageCached() {
const cachedData = GM_getValue(url.host, undefined);
if (cachedData !== undefined && cachedData !== null) {
return cachedData;
}
return null;
}
// 发送请求获取 host 对应的翻译数据
function sendRequest(callback) {
let param = {page: url.origin};
GM_xmlhttpRequest({
method: METHOD,
url: readLink,
data: JSON.stringify(param),
onload: function (response) {
if (callback) {
let parse = JSON.parse(response.responseText);
GM_setValue(url.host, parse.Data);
callback(parse.Data);
console.log("新请求: ", url.host);
}
},
onerror: function (error) {
console.error("请求失败: ", error);
}
});
}
// 递归提取节点的文本内容
function parseDfs(node, respMap) {
// 检查node是否为null
if (node === null || node === undefined) {
return;
}
switch (true) {
// 元素节点
case node.nodeType === Node.ELEMENT_NODE:
// console.log("元素节点》 ", node);
if (["head", "path", "script", "style", "img", "noscript"].includes(node.tagName.toLowerCase())
// 适配 OpenAI
|| node.hasAttribute("data-message-author-role")
// 适配 stackoverflow
|| node.classList.contains("s-post-summary--content")
|| node.classList.contains("thread-item")
) {
// console.log("忽略节点: ", node);
return;
}
if (["input", "button", "textarea"].includes(node.tagName.toLowerCase())) {
processInput(node, respMap);
}
break;
// 文本节点
case node.nodeType === Node.TEXT_NODE:
// console.log("文本节点》 ", node);
switch (url.host) {
case Maven:
processTextNode_maven(node, respMap);
break;
case DockerHub:
processTextNode_dockerhub(node, respMap);
break;
default:
processTextNode(node, respMap);
}
}
// 递归处理子节点
let child = node.firstChild;
while (child) {
parseDfs(child, respMap);
child = child.nextSibling;
}
}
// 处理 input placeholder
function processInput(node, respMap) {
if (node.placeholder) {
let placeholder = node.placeholder.replace(/\u00A0/g, ' ').trim();
if (placeholder.length > 0 && isNonChinese(placeholder)) {
signature(url.host + placeholder).then((value) => {
// 在这里添加一个检查以确保 respMap 是有效的
if (respMap && respMap[value] !== undefined && respMap[value] !== "") {
node.placeholder = respMap[value];
}
}).catch((error) => {
// 处理任何可能的错误
console.error("Error in signature promise: ", error);
});
}
}
if (node.value) {
let value = node.value.replace(/\u00A0/g, ' ').trim();
if (value.length > 0 && isNonChinese(value)) {
signature(url.host + value).then((value) => {
// 在这里添加一个检查以确保 respMap 是有效的
if (respMap && respMap[value] !== undefined && respMap[value] !== "") {
node.value = respMap[value];
}
}).catch((error) => {
// 处理任何可能的错误
console.error("Error in signature promise: ", error);
});
}
}
}
// 处理文本内容
function processTextNode(node, respMap) {
let text = node.textContent.replace(/\u00A0/g, ' ').trim();
if (text.length > 0 && isNonChinese(text)) {
// console.log(text);
signature(url.host + text).then((value) => {
// 添加一个检查以确保 respMap 是有效的
if (respMap && respMap[value] !== undefined && respMap[value] !== "") {
node.textContent = respMap[value];
}
}).catch((error) => {
// 处理任何可能的错误
console.error("Error in signature promise: ", error);
});
}
}
// 防抖函数
function debounce(func, wait) {
let timeout;
return function executedFunction(...args) {
const later = () => {
clearTimeout(timeout);
func(...args);
};
clearTimeout(timeout);
timeout = setTimeout(later, wait);
};
};
// 判断字符串是否不包含中文
function isNonChinese(text) {
return !/[\u4e00-\u9fa5]/.test(text);
}
// 计算SHA-1散列,取最后20个字符
async function signature(text) {
if (text === "") {
return "";
}
const hashBuffer = await crypto.subtle.digest('SHA-1', new TextEncoder().encode(text));
const hashArray = Array.from(new Uint8Array(hashBuffer));
const hashHex = hashArray.map(b => b.toString(16).padStart(2, '0')).join('');
return hashHex.slice(-20);
}
// 适配 maven
function processTextNode_maven(node, respMap) {
let text = node.textContent.replace(/\u00A0/g, ' ').trim();
if (text.length > 0 && isNonChinese(text)) {
// 如果是 Maven,且为日期格式,则改变其格式 May 09, 2019 变为 2019-05-09
if (text.match(/^(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)\s\d{1,2},\s\d{4}$/)) {
let date = new Date(text);
let year = date.getFullYear();
let month = date.getMonth() + 1; // 月份是从0开始的
let day = date.getDate();
// 确保月份和日期为两位数
month = month < 10 ? '0' + month : month;
day = day < 10 ? '0' + day : day;
text = year + "-" + month + "-" + day;
node.textContent = text;
return
} else {
let match = text.match(/Last Release on (Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)\s(\d{1,2}),\s(\d{4})/);
if (match) {
let date = new Date(match[1] + " " + match[2] + ", " + match[3]);
let year = date.getFullYear();
let month = date.getMonth() + 1; // 月份是从0开始的
let day = date.getDate();
// 构建新的日期格式
text = `最近更新 ${year}年${month}月${day}日`;
node.textContent = text;
return;
}
// 处理依赖类型的翻译
let dependencyMatch = text.match(/(Test|Provided|Compile) Dependencies \((\d+)\)/);
if (dependencyMatch) {
let type = dependencyMatch[1];
let count = dependencyMatch[2];
switch (type) {
case 'Test':
text = `测试依赖 Test (${count})`;
break;
case 'Provided':
text = `提供依赖 Provided (${count})`;
break;
case 'Compile':
text = `编译依赖 Compile (${count})`;
break;
// 可以根据需要添加更多的情况
}
node.textContent = text;
return;
}
}
// 处理 "#数字 in" 格式的字符串
let rankMatch = text.match(/#(\d+) in\s*(.*)/);
if (rankMatch) {
let number = rankMatch[1];
let context = rankMatch[2]; // 如 "MvnRepository" 或为空
if (context) {
text = `第 ${number} 位 ${context}`;
} else {
text = `第 ${number} 位 `;
}
node.textContent = text;
return;
}
// 处理 "21,687 artifacts" 到 "被引用 21,687 次" 的转换
// 处理 artifacts 的翻译
// 处理 artifacts 被引用的翻译
let artifactsMatch = text.match(/^([\d,]+)\s+artifacts$/);
if (artifactsMatch) {
let count = artifactsMatch[1];
text = `被引用 ${count} 次`;
node.textContent = text;
return;
}
// 处理漏洞数量的翻译
let vulnerabilityMatch = text.match(/^(\d+)\s+vulnerabilit(y|ies)$/);
if (vulnerabilityMatch) {
let count = vulnerabilityMatch[1];
text = `${count}个漏洞`;
node.textContent = text;
return;
}
// 如果都不符合,则进行普通哈希替换
processTextNode(node, respMap)
}
}
function processTextNode_dockerhub(node, respMap) {
let text = node.textContent.replace(/\u00A0/g, ' ').trim();
if (text.length > 0 && isNonChinese(text)) {
// 处理更新时间的翻译
let timeMatch = text.match(/^(a|an|\d+)\s+(minute|hour|day|month|year)(s)?\s+ago$/);
if (timeMatch) {
let quantity = timeMatch[1];
let unit = timeMatch[2];
let isPlural = timeMatch[3];
// 将 'a' 或 'an' 转换为 '1'
if (quantity === 'a' || quantity === 'an') {
quantity = ' 1';
} else {
quantity = ' ' + quantity
}
// 单位转换
switch (unit) {
case 'minute':
unit = '分钟';
break;
case 'hour':
unit = '小时';
break;
case 'day':
unit = '天';
break;
case 'month':
unit = '月';
break;
case 'year':
unit = '年';
break;
}
// 构建新的文本格式
text = `${quantity} ${unit}之前`;
node.textContent = text;
return;
}
// 处理分页信息的翻译
let paginationMatch = text.match(/^(\d+)\s*-\s*(\d+)\s+of\s+([\d,]+)$/);
if (paginationMatch) {
let start = paginationMatch[1];
let end = paginationMatch[2];
let total = paginationMatch[3].replace(/,/g, ''); // 去除数字中的逗号
// 构建新的文本格式
text = `当前第 ${start} - ${end} 项,共 ${total} `;
node.textContent = text;
return;
}
// 如果都不符合,则进行普通哈希替换
processTextNode(node, respMap)
}
}