// ==UserScript== // @name 网页翻译——翻译为中文 // @author Kaiter-Plus // @namespace https://gitee.com/Kaiter-Plus/TampermonkeyScript/tree/master/Translate/Translate_only_chinese.js // @description 给每个非中文的网页右下角(可以调整到左下角)添加一个google翻译图标,该版本为中文翻译版本,只把外语翻译为中文 // @version 0.13 // @license BSD-3-Clause // @include *://* // @exclude /^(http|https).*[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+/ // @exclude /.*duyaoss\.com/ // @exclude /.*lanzous\.com/ // @exclude /.*w3school.*cn/ // @exclude /.*iqiyi\.com/ // @exclude /.*baidu.*/ // @exclude /.*cnblogs\.com/ // @exclude /.*csdn\.net/ // @exclude /.*zhku\.edu\.cn/ // @exclude /.*zhihuishu\.com/ // @exclude /.*aliyuncs\.com/ // @exclude /.*chaoxing\.com/ // @exclude /.*youku\.com/ // @exclude /.*examcoo\.com/ // @exclude /.*mooc\.com/ // @exclude /.*bilibili\.com/ // @exclude /.*qq\.com/ // @exclude /.*yy\.com/ // @exclude /.*huya\.com/ // @exclude /localhost/ // @exclude /.*acfun\.cn/ // @exclude /.*eleme\.cn/ // @exclude /.*douyin\.com/ // @icon  // @noframes // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @grant GM_notification // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @note 2021/04/13 新建副本,只把其它语言翻译为中文 // @note 2021/06/12 适配了移动端 // @note 2021/07/14 排除抖音,防止可能出现的 bug // @note 2021/09/19 优化开启关闭自动检测中文逻辑 // @note 2021/12/12 应用户反馈,去除显示“提供更好的翻译建议”弹框 // @note 2021/12/14 直接使用 https 获取谷歌翻译接口(防止有可能火狐浏览器无法用于翻译本地文件的bug)@古海沉舟 // @note 2021/12/21 优化菜单切换逻辑,优化交互体验 // @note 2021/12/28 优化判断网页是否是中文逻辑 // @note 2022/01/08 修复上一个版本更新后大多数网站不能使用的 Bug, 解决一些网站开启脚本之后不能滚动 // @note 2022/01/10 修复访问站内 http 链接自动跳转 https 的问题 // @note 2022/01/18 增加排除网页元素 // @note 2022/03/09 增加排除网页元素 // @note 2022/03/19 增加配置选项【显示翻译建议】,默认关闭,打开之后通过悬浮文字可以看到原文 // @downloadURL none // ==/UserScript== ;(function () { 'use strict' // 菜单 const menu = [ { key: 'position', name: '按钮位置', value: true, tip: { open: '👈', close: '👉' }, click: setButtonPosition }, { key: 'isCheck', name: '自动检测中文', value: true, tip: { open: '✅', close: '❌' }, click: null }, { key: 'isShowTip', name: '显示翻译建议', value: false, tip: { open: '✅', close: '❌' }, click: setShowTip } ] // 保存已注册的菜单 const munuRegister = [] // 配置默认菜单 menu.forEach(v => { if (GM_getValue(v.key) === undefined) GM_setValue(v.key, v.value) }) // 注册菜单 function registerMenuCommand() { if (munuRegister.length === menu.length) { munuRegister.forEach(v => { GM_unregisterMenuCommand(v) }) } menu.forEach((v, i) => { v.value = GM_getValue(v.key) munuRegister[i] = GM_registerMenuCommand(`${v.value ? v.tip.open : v.tip.close} ${v.name}`, () => { menuSwitch(v) }) }) } // 切换菜单 function menuSwitch(item) { // 设置数据 item.value = !item.value GM_setValue(item.key, item.value) // 系统通知 GM_notification({ text: `已${item.value ? item.tip.open : item.tip.close}[${item.name}] 功能`, title: '网页翻译', timeout: 1000 }) // 如果有点击事件,执行 if (item.click) item.click() // 重新注册 registerMenuCommand() } // 获取 head const head = document.head // 获取body const body = document.body // 获取当前页面的语言 const lang = document.documentElement.lang // 获取网页的标题 const pageTitle = document.title // 获取网页使用的主要语言 const mainLang = document.characterSet.toLowerCase() // 判断是不是中文网页 function isChinesePage() { return ( GM_getValue('isCheck') && (lang.substring(0, 2) === 'zh' || mainLang.substring(0, 2) === 'gb' || /[\u4E00-\u9FFF]/.test(pageTitle)) ) } // 位置信息样式 let positionStyle = null // 设置按钮位置 function setButtonPosition() { if (positionStyle) positionStyle.parentNode.removeChild(positionStyle) positionStyle = GM_addStyle(` .buttonContainer { ${GM_getValue('position') ? 'left' : 'right'}: 0; transform: translateX(${GM_getValue('position') ? '-' : ''}80%); } `) } // 显示翻译建议信息 let tipStyle = null function setShowTip() { if (tipStyle) tipStyle.parentNode.removeChild(tipStyle) positionStyle = GM_addStyle(` #goog-gt-tt { visibility: ${GM_getValue('isShowTip') ? 'visible' : 'hidden'}!important; display: ${GM_getValue('isShowTip') ? 'block' : 'none'}!important; } .goog-text-highlight { background-color: ${GM_getValue('isShowTip') ? '#c9d7f1' : 'inherit'}!important; box-shadow: ${GM_getValue('isShowTip') ? '2 2 4 #99a' : '0 0 0 0 transparent'}!important; } `) } // 注册菜单 registerMenuCommand() // 判断是不是中文,如果是则直接return,否则执行 if (!isChinesePage()) { // 创建网页元素方法 function createElement(html, nodeText, attr, parent) { const element = document.createElement(nodeText) if (attr) { element[attr] = html } else { element.innerHTML = html } parent.appendChild(element) } // 设置网页自动把 http 升级为 https // const e = document.createElement('meta') // e.setAttribute('http-equiv', 'Content-Security-Policy') // e.setAttribute('content', 'upgrade-insecure-requests') // head.appendChild(e) // 自定义样式,隐藏顶部栏 GM_addStyle(` html,body{ top: 0!important; } #google_translate_element { display: none; } .buttonContainer { width: 6em; position: fixed; bottom: 30px; z-index: 10000000; user-select: none; overflow: hidden; text-align: center; font-size: 13px; line-height: 2em; border-radius: 1em; box-shadow: 1px 1px 3px 0 #888; opacity: .5; transition: all .3s; } .recoverPage, .translateButton { float: left; width: 50%; box-sizing: border-box; } .recoverPage { border-radius: 1em 0 0 1rem; background-color: #fff; } .translateButton { color: #fff; border-radius: 0 1rem 1rem 0; background-color: #55b9f3; } .buttonContainer:hover { opacity: 1; transform: translateX(0); } .recoverPage:active, .translateButton:active { box-shadow: 1px 1px 3px 0 #888 inset; } .goog-te-banner-frame.skiptranslate { display: none } #goog-gt-tt { visibility: hidden!important; display: none!important; } .goog-text-highlight { background-color: inherit!important; box-shadow: 0 0 0 0 transparent!important; } `) setButtonPosition() // 创建容器 createElement('google_translate_element', 'div', 'id', body) // 初始化 createElement( ` function googleTranslateElementInit() { new google.translate.TranslateElement( { pageLanguage: 'auto', includedLanguages: 'zh-CN', layout: /mobile/i.test(navigator.userAgent) ? 0 : 2 }, 'google_translate_element' ) // 清除图片的请求,加快访问速度 let img = [].slice.call(document.querySelectorAll('#goog-gt-tt img,#google_translate_element img')) img.forEach(function (v) { const a = v a.src = '' let b = a.outerHTML.replace(//, () => { return '' }) const c = document.createElement('div') c.innerHTML = b a.parentNode.insertBefore(c.children[0], a.parentNode.children[0]) a.remove() }) // 获取设备信息 const device = navigator.userAgent.indexOf('Mobile') // 按钮容器 const buttonContainer = document.createElement('div') buttonContainer.setAttribute('class', 'notranslate buttonContainer') document.body.appendChild(buttonContainer) // 恢复按钮 const recoverPage = document.createElement('div') recoverPage.setAttribute('class', 'notranslate recoverPage') recoverPage.innerText = '恢复' buttonContainer.appendChild(recoverPage) // 点击恢复原网页 recoverPage.onclick = () => { let recoverIframe = null if (~device) { // 移动端 const recoverDocument = document.getElementById(':1.container').contentWindow.document recoverDocument.getElementById(':1.restore').click() } else { // PC端 const recoverDocument = document.getElementById(':2.container').contentWindow.document recoverDocument.getElementById(':2.restore').click() } } // 翻译按钮 let langIframe = document.querySelector('.goog-te-menu-frame') const langIframeTimer = setInterval(() => { if (langIframe) { const langDocument = langIframe.contentWindow.document || langIframe.contentDocument let translateLang const translateTimer = setInterval(() => { if (~device) { translateLang = document.querySelector('.goog-te-combo') } else { translateLang = langDocument.querySelectorAll('table a')[1] } if (translateLang) { clearInterval(translateTimer) // 添加翻译按钮 const translateButton = document.createElement('div') translateButton.setAttribute('class', 'notranslate translateButton') translateButton.innerText = '翻译' buttonContainer.appendChild(translateButton) // 点击翻译 translateButton.onclick = () => { if (~device) { const event = document.createEvent('HTMLEvents') event.initEvent('change', true, true) event.eventType = 'message' document.querySelector('.goog-te-combo').dispatchEvent(event) } else { translateLang.click() } } } }, 300) clearInterval(langIframeTimer) } else { langIframe = document.querySelector('.goog-te-menu-frame') } }, 300) } `, 'script', '', head ) // 导入翻译接口 if (/quora/i.test(location.href)) { // 这里主要是适配quora createElement( 'https://translate.google.com/translate_a/element.js?&cb=googleTranslateElementInit', 'script', 'src', head ) } else { createElement( 'https://translate.google.cn/translate_a/element.js?&cb=googleTranslateElementInit', 'script', 'src', head ) } // 排除一些代码的翻译 const noTranslateArray = [ '.bbCodeCode', 'tt', 'pre[translate="no"]', 'pre', '.post_spoiler_show', '.c-article-section__content sub', '.c-article-section__content sup', '.c-article-equation', '.mathjax-tex' ] noTranslateArray.forEach(selectorName => { ;[...document.querySelectorAll(selectorName)].forEach(node => { if (node.className.indexOf('notranslate') === -1) { node.classList.add('notranslate') } }) }) // 解决一些网站开启脚本之后不能滚动 function CanIScroll() { const noScrollSite = ['curseforge.com'] noScrollSite.forEach(site => { if (~document.domain.indexOf(site)) { GM_addStyle(` html { height: auto!important; } `) } }) } CanIScroll() } })()