网页翻译——翻译为中文

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

As of 2021-07-14. See the latest version.

// ==UserScript==
// @name         网页翻译——翻译为中文
// @author       Kaiter-Plus
// @namespace    https://gitee.com/Kaiter-Plus/TampermonkeyScript/tree/master/Translate/Translate_only_chinese.js
// @description  给每个非中文的网页右下角(可以调整到左下角)添加一个google翻译图标,该版本为中文翻译版本,只把外语翻译为中文
// @version      0.03
// @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
// ==/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 (lang.substr(0, 2) === 'zh' || (mainLang.substr(0, 2) === 'gb' && !isCheck)) {
    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(
        '//translate.google.com/translate_a/element.js?&cb=googleTranslateElementInit',
        'script',
        'src',
        head
      )
    } else {
      createElement(
        '//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)
})()