网页翻译

给每个非中文的网页左下角添加一个google翻译图标,直接调用 Google 的翻译接口对非中文网页进行翻译

As of 2020-07-02. See the latest version.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

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

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

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

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

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         网页翻译
// @author       Kaiter-Plus
// @namespace    https://gitee.com/Kaiter-Plus/TampermonkeyScript/tree/master/Translate
// @description  给每个非中文的网页左下角添加一个google翻译图标,直接调用 Google 的翻译接口对非中文网页进行翻译
// @include      *://*
// @exclude      /^[http|https].*[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+/
// @exclude      /lanzous\.com/
// @exclude      /w3school.*cn/
// @exclude      /iqiyi\.com/
// @exclude      /.*baidu.*/
// @exclude      /cnblogs\.com/
// @exclude      /runoob\.com/
// @exclude      /csdn\.net/
// @exclude      /zhku\.edu\.cn/
// @exclude      /zhihuishu\.com/
// @exclude      /aliyuncs\.com/
// @exclude      /.*chaoxing\.com/
// @exclude      /examcoo\.com/
// @exclude      /.*mooc\.com/
// @exclude      /52pojie\.cn/
// @exclude      /.*bilibili\.com/
// @exclude      /.*qq\.com/
// @exclude      /localhost/
// @icon         
// @version      1.22
// @noframes
// @note         2020/03/26 网页整页翻译功能
// @note         2020/04/13 排除纯ip网址
// @note         2020/04/14 移除翻译后顶边栏
// @note         2020/05/01 排除百度、QQ、超星等中文网址
// @note         2020/05/04 修复去除上边栏网页先向下再向上跳的Bug
// @note         2020/05/05 尝试修复百度出现超粗顶栏的Bug
// @note         2020/05/12 添加恢复原网页的按钮(翻译按钮旁边),有点丑,不过希望可以先用着,有时间再看看能不能弄好看一点ヾ(≧▽≦*)o
// @note         2020/05/23 稍微修改了一下恢复原网页的按钮的样式(还是不好看)
// @note         2020/05/26 修改脚本为原生javascript,兼容暴力猴
// @note         2020/05/26 修改翻译栏样式,固定宽高,防止在一些页面上出现太宽或太高的现象
// @note         2020/06/06 修复火狐浏览器(firefox),内存溢出的bug,精简了一点代码
// @note         2020/06/08 排除一些代码块的翻译,如果还有其它的网站的代码块需要排除,可以反馈给我,我排除一下
// @note         2020/06/17 修改恢复原网页按钮的样式(使用@picasso250的样式),排除标签 tt
// @note         2020/06/18 适配Quora
// @note         2020/06/26 翻译和恢复按钮修改为在页面边缘附着的半透明半圆 -> 鼠标移入弹出翻译或恢复按钮
// @note         2020/07/02 按钮向上移动了30像素,经测试,点击弹出按钮的方式不太友好,故放弃
// ==/UserScript==

(function () {
  'use strict'

  // 获取 head
  const head = document.head
  // 获取body
  const body = document.body
  // 获取当前页面的语言
  const lang = document.documentElement.lang

  // 判断是不是中文,如果不是则执行
  if (lang.substr(0, 2) !== 'zh') {

    // 创建网页元素方法
    function createElement(html, nodeText, attr, parent) {
      const element = document.createElement(nodeText)
      if(attr){
        element[attr] = html
      }else{
        element.innerHTML = html
      }
      parent.appendChild(element)
    }

    // 自定义样式,隐藏顶部栏
    createElement([
      '#google_translate_element {',
      '  position: fixed;',
      '  left: 0px;',
      '  bottom: 40px;',
      '  height: 22px;',
      '  border-radius: 11px;',
      '  z-index: 10000000;',
      '  overflow: hidden;',
      '  box-shadow: 1px 1px 3px 0 #888;',
      '  opacity: .5;',
      '  transform: translateX(-85%);',
      '  transition: all .3s;',
      '}',
      '#google_translate_element:hover {',
      '  opacity: 1;',
      '  transform: translateX(0);',
      '}',
      '#google_translate_element .goog-te-gadget-simple {',
      '  width: 100%;',
      '}',
      '.goog-te-banner-frame.skiptranslate {',
      '  display: none',
      '}',
      'html,body{',
      '  top: 0!important',
      '}',
      '.recoverPage {',
      '  width: 4em;',
      '  background-color: #fff;',
      '  position: fixed;',
      '  left: 0px;',
      '  z-index: 10000000;',
      '  bottom: 75px;',
      '  user-select: none;',
      '  text-align: center;',
      '  border: 1px solid #a8a8a8;',
      '  font-size: small;',
      '  line-height: 2em;',
      '  border-radius: 1em;',
      '  box-shadow: 1px 1px 3px 0 #888;',
      '  opacity: .5;',
      '  transform: translateX(-75%);',
      '  transition: all .3s;',
      '}',
      '.recoverPage:hover {',
      '  opacity: 1;',
      '  transform: translateX(0);',
      '}',
      '.recoverPage:active {',
      '  box-shadow: 1px 1px 3px 0 #888 inset;',
      '}'
    ].join('\n'), "style", '', head)

    createElement('google_translate_element', 'div', 'id', body)


    // 初始化
    const initHtml = [
      "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,zh-TW,en,ja,ru',",
      "         /*0,原生select,并且谷歌logo显示在按钮下方。",
      "           1,原生select,并且谷歌logo显示在右侧。",
      "           2,完全展开语言列表,适合pc。",
      "         */",
      "         layout: /mobile/i.test(navigator.userAgent) ? 0 : 2,",
      "       }, 'google_translate_element');",
      "       // 清除图片的请求,加快访问速度",
      "      img = [].slice.call(document.querySelectorAll('#goog-gt-tt img,#google_translate_element img'));",
      "      img.forEach(function(v, i) {",
      "        v.src = '';",
      "      });",
      "      const recoverPage = document.createElement('div')",
      "      recoverPage.setAttribute('class', 'notranslate recoverPage')",
      "      recoverPage.innerText = '恢复'",
      "      document.body.appendChild(recoverPage)",
      "      // 点击恢复原网页",
      "      recoverPage.onclick = (() => {",
      "        const recoverIframe = document.getElementById(':2.container')",
      "        if (recoverIframe) {",
      "          const recoverDocument = recoverIframe.contentWindow.document",
      "          recoverDocument.getElementById(':2.restore').click()",
      "        }",
      "      })",
      "    }",
      "  }, 300);",
      "}"
    ].join("\n")
    createElement(initHtml,'script', '', head)

    // 导入翻译接口
    if (/quora/i.test(location.href)) {
      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'
    ]
    noTranslateArray.forEach(selectorName => {
      [...document.querySelectorAll(selectorName)].forEach(node => {
        if (node.className.indexOf('notranslate') === -1) {
          node.classList.add('notranslate')
        }
      })
    })
  }
}());