网页翻译——翻译为中文

给每个非中文的网页右下角(可以调整到左下角)添加一个google翻译图标,该版本为中文翻译版本,只把外语翻译为中文

当前为 2021-12-14 提交的版本,查看 最新版本

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         网页翻译——翻译为中文
// @author       Kaiter-Plus
// @namespace    https://gitee.com/Kaiter-Plus/TampermonkeyScript/tree/master/Translate/Translate_only_chinese.js
// @description  给每个非中文的网页右下角(可以调整到左下角)添加一个google翻译图标,该版本为中文翻译版本,只把外语翻译为中文
// @version      0.05
// @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
// @note         2021/04/13 新建副本,只把其它语言翻译为中文
// @note         2021/06/12 适配了移动端
// @note         2021/07/14 排除抖音,防止可能出现的 bug
// @note         2021/09/19 优化开启关闭自动检测中文逻辑
// @note         2021/12/14 直接使用 https 获取谷歌翻译接口(防止有可能火狐浏览器无法用于翻译本地文件的bug)@古海沉舟
// ==/UserScript==

;(function () {
  'use strict'

  // 获取 head
  const head = document.head
  // 获取body
  const body = document.body
  // 获取当前页面的语言
  const lang = document.documentElement.lang
  // 获取网页使用的主要语言
  const mainLang = document.characterSet.toLowerCase()
  // 获取按钮位置信息
  const currentPosition = GM_getValue('position')
  // 按钮位置:true 为左,false 为右, 默认为右
  let buttonPosition = currentPosition ? true : false
  // 获取是否自动检测中文配置信息
  const currentCheck = GM_getValue('isCheck')
  // 检测设置:true 关闭,false 开启, 默认开启
  let isCheck = currentCheck ? true : false

  // 判断是不是中文,如果是则直接return,否则执行
  if (!isCheck && (lang.substr(0, 2) === 'zh' || mainLang.substr(0, 2) === 'gb')) {
    addSwitchChecked()
    return
  } else {
    addSwitchChecked()
    // 创建网页元素方法
    function createElement(html, nodeText, attr, parent) {
      const element = document.createElement(nodeText)
      if (attr) {
        element[attr] = html
      } else {
        element.innerHTML = html
      }
      parent.appendChild(element)
    }

    // 自定义样式,隐藏顶部栏
    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
      }
    `)

    setButtonPosition()

    // 创建容器
    createElement('google_translate_element', 'div', 'id', body)

    // 初始化
    createElement(
      `function googleTranslateElementInit() {
        let google_translate_element = document.getElementById('google_translate_element')
        let timer = setInterval(function () {
          google_translate_element = document.getElementById('google_translate_element')
          if (google_translate_element) {
            clearInterval(timer)
            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(/<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 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)
          }
        }, 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']
    noTranslateArray.forEach(selectorName => {
      ;[...document.querySelectorAll(selectorName)].forEach(node => {
        if (node.className.indexOf('notranslate') === -1) {
          node.classList.add('notranslate')
        }
      })
    })
  }

  // 设置按钮位置
  function setButtonPosition() {
    if (buttonPosition) {
      GM_addStyle(`
        .buttonContainer {
          left: 0px;
          transform: translateX(-80%);
        }
      `)
    } else {
      GM_addStyle(`
        .buttonContainer {
          right: 0px;
          transform: translateX(80%);
        }
      `)
    }
  }

  // 添加注册菜单项
  function addSwitchChecked() {
    GM_registerMenuCommand('切换自动检测中文', function () {
      isCheck = !isCheck
      GM_setValue('isCheck', isCheck)
      isCheck ? GM_notification('已关闭自动检测中文') : GM_notification('已开启自动检测中文')
      location.reload()
    })
  }
  function switchButtonPosition() {
    GM_setValue('position', !buttonPosition)
    setButtonPosition()
    location.reload()
  }
  GM_registerMenuCommand('切换按钮位置', switchButtonPosition)
})()