// ==UserScript==
// @name 网页翻译ver1.00
// @author xzsajo
// @namespace /
// @description 给每个非中文的网页右下角(可以调整到左下角)添加一个google翻译图标,直接调用 Google 的翻译接口对非中文网页进行翻译
// @version 1.62
// @license BSD-3-Clause
// @include *
// @icon 
// @noframes
// @grant GM_addStyle
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_notification
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// ==/UserScript==
;(function () {
'use strict'
// 取消没有用的图片请求
const pointTimer = setInterval(() => {
const banner = document.querySelector('.goog-te-banner-frame')
if (banner) {
const doc = banner.contentWindow.document || banner.contentDocument
const imgs = doc.getElementsByTagName('img')
for (let i = 0; i < imgs.length; i++) {
imgs[i].src = ''
}
clearInterval(pointTimer)
}
}, 10)
// 菜单
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_getValue(v.key) === null) 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(`
#google_translate_element {
${GM_getValue('position') ? 'left' : 'right'}: 0;
transform: translateX(${GM_getValue('position') ? '-' : ''}85%);
}
.recoverPage {
${GM_getValue('position') ? 'left' : 'right'}: 0;
transform: translateX(${GM_getValue('position') ? '-' : ''}73%);
}
@media handheld, only screen and (max-width: 768px) {
#google_translate_element {
transform: translateX(${GM_getValue('position') ? '-' : ''}60%);
}
.recoverPage {
transform: translateX(0);
}
}
`)
}
// 显示翻译建议信息
let tipStyle = null
function setShowTip() {
if (tipStyle) tipStyle.parentNode.removeChild(tipStyle)
tipStyle = 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()
// 判断是不是中文,不是则执行
// 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)
}
// 初始化按钮位置
setButtonPosition()
// 初始化是否显示更好的翻译建议
setShowTip()
// 设置网页自动把 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 {
position: fixed;
bottom: 30px;
height: 21px;
border-radius: 11px;
z-index: 10000000;
overflow: hidden;
box-shadow: 1px 1px 3px 0 #888;
opacity: .5;
transition: all .3s;
background-color: #646cff;
}
#google_translate_element .goog-te-gadget-simple {
border: 0;
background-color: transparent;
}
#google_translate_element .goog-te-gadget-simple span {
margin-right: 0;
border-radius: 11px;
color: rgba(255, 255, 255, .87);
}
.goog-te-banner-frame.skiptranslate {
display: none;
}
#lb {
display: inline-block;
}
.recoverPage {
width: 4em;
background-color: #646cff;
color: rgba(255, 255, 255, .87);
position: fixed;
z-index: 10000000;
bottom: 60px;
user-select: none;
text-align: center;
font-size: small;
line-height: 2em;
border-radius: 1em;
box-shadow: 1px 1px 3px 0 #888;
opacity: .5;
transition: all .3s;
}
#google_translate_element:hover, .recoverPage:hover {
opacity: 1;
transform: translateX(0);
}
.recoverPage:active {
box-shadow: 1px 1px 3px 0 #888 inset;
}
#google_translate_element .goog-te-gadget-simple {
width: 100%;
}
@media handheld, only screen and (max-width: 768px) {
#google_translate_element {
width: 104px;
color: unset;
background-color: #fff;
}
#google_translate_element .goog-te-combo {
margin: 0;
padding-top: 2px;
border: none;
color: unset;
background-color: transparent;
}
.recoverPage {
width: 1.5em;
color: unset;
line-height: 1.5em;
background-color: #fff;
}
}
`)
// 创建容器
createElement('google_translate_element', 'div', 'id', body)
// 初始化
createElement(
`
function googleTranslateElementInit() {
new google.translate.TranslateElement(
{
pageLanguage: 'auto',
//包括的语言,中文简体,中文繁体,英语,日语,俄语
includedLanguages: 'zh-CN,zh-TW,en,ja,ru',
/*
* 0,原生select,并且谷歌logo显示在按钮下方。
* 1,原生select,并且谷歌logo显示在右侧。
* 2,完全展开语言列表,适合pc。
*/
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(/<img(.*?)>/, () => {
return '<span id="lb"' + RegExp.$1 + '></span>'
})
const c = document.createElement('div')
c.innerHTML = b
a.parentNode.insertBefore(c.children[0], a.parentNode.children[0])
a.remove()
})
const recoverPage = document.createElement('div')
recoverPage.setAttribute('class', 'notranslate recoverPage')
recoverPage.innerText = '原'
document.body.appendChild(recoverPage)
// 点击恢复原网页
recoverPage.onclick = () => {
const phoneRecoverIframe = document.getElementById(':1.container') // 移动端
const PCRecoverIframe = document.getElementById(':2.container') // PC端
if (phoneRecoverIframe) {
const recoverDocument = phoneRecoverIframe.contentWindow.document
recoverDocument.getElementById(':1.restore').click()
} else if (PCRecoverIframe) {
const recoverDocument = PCRecoverIframe.contentWindow.document
recoverDocument.getElementById(':2.restore').click()
}
}
}
`,
'script',
'',
head
)
// 导入翻译接口
createElement(
'https://translate.google.com/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')
}
})
})
// 针对一些网站排除一些无需翻译的文字
const noTranslateList = [
{
site: 'cratchapixel.com',
selector: ['span.MathJax']
}
]
noTranslateList.forEach(item => {
if (~document.domain.indexOf(item.site)) {
item.selector.forEach(selectorName => {
let timer = null
let classList = document.querySelectorAll(selectorName)
if (!classList[0]) {
timer = setInterval(() => {
classList = document.querySelectorAll(selectorName)
if (classList[0]) {
clearInterval(timer)
;[...classList].forEach(node => {
if (!~node.className.indexOf('notranslate')) {
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()
// }
})()