返回首頁 

Greasy Fork is available in English.

外语终结者

识别非中文字符,如果长度大于5且翻译文本中不含中文,则翻译并且替换原始文本到中文


Installer ce script?
Script suggéré par l'auteur

Vous pourriez également aimer Greasyfork Artisan de la colle.


Installer ce script
// ==UserScript==// @name              外语终结者// @namespace         https://github.com/#####GodMan/UserScripts// @version           1.4.0.0// @description       识别非中文字符,如果长度大于5且翻译文本中不含中文,则翻译并且替换原始文本到中文// @name:zh-CN        外语终结者// @description:zh-CN 识别非中文字符,如果长度大于5且翻译文本中不含中文,则翻译并且替换原始文本到中文// @license           MIT// @author            ##的勤务员 <[email protected]>// @match             *://*/*// @grant             GM_xmlhttpRequest// @icon               @iconbak           https://immersive-translate.owenyoung.com/favicon.png// @grant             GM_getValue// @grant             GM_addStyle// @grant             GM_setValue// @grant             GM_registerMenuCommand// @connect           translate.googleapis.com// @supportURL        https://github.com/#####GodMan/UserScripts/issues// @homepageURL       https://github.com/#####GodMan/UserScripts// ==/UserScript==//https://translate.googleapis.com/translate_a/single?client=gtx&sl=auto&tl=zh-CN&dj=1&dt=t&dt=rm&q=你好(function () {'use strict'// 添加一个对象来跟踪已经翻译过的节点var translatedNodes = {}var skipClasses = GM_getValue('skipClasses') || ['prettyprint', 'linenums', 'lang-js']// 创建一个 MutationObserver 实例var observer = new MutationObserver(function (mutationsList) {// 遍历每一个发生变化的 mutationfor (var mutation of mutationsList) {// 检查是否有子节点被添加到文档中//if (mutation.type === 'childList') {// 遍历新增加的节点mutation.addedNodes.forEach(function (addedNode) {// 如果新增节点是元素节点,可以递归扫描子节点以查找文本内容if (addedNode.nodeType === Node.ELEMENT_NODE) {if (!isInsidePreCode(addedNode)) {scanTextNodes(addedNode)}//  scanTextNodes(addedNode);// console.log("New element node added:", addedNode.nodeName);}})// }}})function isInsidePreCode(node) {while (node) {if (node.matches && node.matches('pre')) {return true}node = node.parentElement}return false}// 配置 MutationObserver 监听的目标节点和观察的属性var config = { childList: true, subtree: true }// 开始监听文档的变化observer.observe(document.body, config)// 添加或移除翻译功能function toggleTranslation() {var currentSite = window.location.hostnamevar enabledSites = GM_getValue('enabledSites') || []if (enabledSites.includes(currentSite)) {enabledSites = enabledSites.filter(function (site) {return site !== currentSite})GM_setValue('enabledSites', enabledSites)Toast('已移出翻译!', 3000, 'rgba(0, 128, 0, 0.7)', '#fff')location.reload()} else {enabledSites.push(currentSite)GM_setValue('enabledSites', enabledSites)Toast('已添加翻译!', 3000, 'rgba(0, 128, 0, 0.7)', '#fff')location.reload()}}function shouldSkipElement(element) {// 跳过类名包含 "prettyprint"、"linenums" 和 "lang-js" 的元素if (element.classList.contains('prettyprint') ||element.classList.contains('linenums') ||element.classList.contains('lang-js')) {return true}// 还可以根据其他需要跳过的条件进行判断return false}// 编辑生效站点列表function editSites() {// 检查是否已经存在编辑窗口if (document.getElementById('editorContainer')) {return}var enabledSites = GM_getValue('enabledSites') || []var siteList = enabledSites.join('\n')var editorWindow = document.createElement('div')editorWindow.innerHTML = `<div id="editorContainer" style="position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); background: white; padding: 20px; border: 1px solid #ccc; border-radius: 5px; z-index: 9999;"><textarea id="siteList" style="width: 300px; height: 200px; margin-bottom: 10px;">${siteList}</textarea><br><button id="saveButton" style="padding: 5px 10px; background: #007bff; color: #fff; border: none; border-radius: 3px; cursor: pointer;">保存</button><button id="cancelButton" style="margin-left: 10px; padding: 5px 10px; background: #ccc; color: #333; border: none; border-radius: 3px; cursor: pointer;">取消</button></div><div id="overlay" style="position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.5); z-index: 999;"></div>`document.body.appendChild(editorWindow)var saveButton = document.getElementById('saveButton')var cancelButton = document.getElementById('cancelButton')var siteListTextarea = document.getElementById('siteList')var overlay = document.getElementById('overlay')saveButton.addEventListener('click', function () {var editedSites = siteListTextarea.value.split('\n').map(function (site) {return site.trim()}).filter(Boolean)GM_setValue('enabledSites', editedSites)Toast('已保存生效站点列表!', 3000, 'rgba(0, 128, 0, 0.7)', '#fff')document.body.removeChild(editorWindow)document.body.removeChild(overlay)})cancelButton.addEventListener('click', function () {document.body.removeChild(editorWindow)document.body.removeChild(overlay)})}// 注册油猴菜单命令GM_registerMenuCommand(isCurrentSiteEnabled() ? '移出翻译' : '添加翻译', toggleTranslation)GM_registerMenuCommand('编辑生效站点', editSites)// GM_registerMenuCommand('开启/关闭高亮和删除功能', toggleMouseListeners)// 递归遍历给定节点及其后代(深度优先搜索)function scanTextNodes(node) {if (!node.parentNode || !document.body.contains(node)) {return}switch (node.nodeType) {case Node.ELEMENT_NODE:if (node.tagName.toLowerCase() === 'script' || node.isContentEditable) {return}if (shouldSkipElement(node)) {return}if (shouldTranslateNode(node)) { // 检查是否应该翻译该节点node.childNodes.forEach(scanTextNodes)}breakcase Node.TEXT_NODE:var text = node.nodeValue.trim()if (!containsChinese(text) && text.length > 5 && isCurrentSiteEnabled() && shouldTranslateNode(node.parentNode)) {translateText(node, text)}break}}// 检查当前站点是否在生效站点列表中function isCurrentSiteEnabled() {var currentSite = window.location.hostnamevar enabledSites = GM_getValue('enabledSites') || []return enabledSites.includes(currentSite)}// 检查字符串是否含中文或者片假字function containsChinese(str) {// 日文片假字范围:\u3040-\u30FF\uFF66-\uFF9Fvar katakanaRegex = /[\u3040-\u30FF\uFF66-\uFF9F]/if (katakanaRegex.test(str)) {return false // 如果包含日文片假字,则返回 false}// 中文字符范围:\u4E00-\u9FA5var chineseRegex = /[\u4E00-\u9FA5]/return chineseRegex.test(str) // 返回是否包含中文字符}// 检查字符串是否含有数字function containsNumbers(str) {return /^[a-z\u4E00-\u9FA5\s]+$/i.test(str)}// 检查字符串是否为网址function isURL(str) {return /((https?:\/\/|www\.)[\x21-\x7e]+[\w/=]|\w([\w.\-])+@\w[\w.\-]+\.(com|cn|org|net|info|tv|cc|###|edu)|(\w[\w.\-]+\.(com|cn|org|net|info|tv|cc|###|edu))(\/[\x21-\x7e]*[\w/])?|ed2k:\/\/[\x21-\x7e]+\|\/|thunder:\/\/[\x21-\x7e]+=)/i.test(str)}// 检查是否应该翻译该节点function shouldTranslateNode(node) {// 检查是否已经翻译过该节点return !translatedNodes[node.textContent]}// 翻译文本function translateText(node, text) {// 检查是否已经翻译过该节点if (translatedNodes[node.textContent]) {return}var api = 'https://translate.googleapis.com/translate_a/single'var params = {client: 'gtx',dt: 't',sl: 'auto',tl: 'zh-CN',q: text}GM_xmlhttpRequest({method: 'GET',url: api + buildQueryString(params),onload: function (response) {try {var data = JSON.parse(response.responseText.replace('\'', '\u2019'))var translatedText = data[0].reduce((acc, item) => acc + item[0], '')showTranslation(node, text, translatedText)// 标记该节点已经翻译过translatedNodes[node.textContent] = true} catch (error) {console.error('翻译失败:', error)}},onerror: function (response) {console.error('请求翻译失败:', response.statusText)}})}// 构建查询字符串function buildQueryString(params) {return '?' + Object.keys(params).map(function (key) {return encodeURIComponent(key) + '=' + encodeURIComponent(params[key])}).join('&')}// 显示翻译// 显示翻译function showTranslation(node, originalText, translatedText) {// 创建 <ruby> 元素和 <rt> 元素var rubyElement = document.createElement('ruby') // 创建包含音标的元素var rtElement = document.createElement('rt') // 创建音标元素// 设置音标元素的文本内容rtElement.textContent = translatedText// 设置音标元素的字体大小rtElement.style.fontSize = 'smaller'// 设置音标元素的文本左对齐rtElement.style.textAlign = 'left'rtElement.style.color = 'red' // 设置音标元素的颜色// 将音标元素添加到包含音标的元素中rubyElement.appendChild(rtElement)// 在中文翻译节点之后插入包含音标的元素//node.parentNode.insertBefore(rubyElement, node.nextSibling);// 将原始的非中文字符串替换为翻译后的文本node.textContent = translatedText // 替换原始的非中文字符串}// 主函数function main() {// 在页面加载时进行一次扫描scanTextNodes(document.body)}// 启动脚本main()// 弹出提示信息function Toast(msg, duration, backgroundColor, textColor) {duration = isNaN(duration) ? 3000 : durationbackgroundColor = backgroundColor || 'rgba(0, 0, 0, 0.7)'textColor = textColor || 'rgb(255, 255, 255)'var m = document.createElement('div')m.innerHTML = msgm.style.cssText = 'max-width:60%;min-width: 150px;padding:0 14px;height: 40px;color: ' + textColor + ';line-height: 40px;text-align: center;border-radius: 12px;position: fixed;top: 95%;left: 50%;transform: translate(-50%, -50%);z-index: 2147483647;background: ' + backgroundColor + ';font-size: 16px;'document.body.appendChild(m)setTimeout(function () {var d = 0.5m.style.transition = '-webkit-transform ' + d + 's ease-in, opacity ' + d + 's ease-in'm.style.opacity = '0'setTimeout(function () {document.body.removeChild(m)}, d * 1000)}, duration)}})()