Greasy Fork is available in English.

网页翻译

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

Pada tanggal 26 Juni 2020. Lihat %(latest_version_link).

// ==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      /csdn\.net/
// @exclude      /zhku\.edu\.cn/
// @exclude      /zhihuishu\.com/
// @exclude      /runoob\.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.21
// @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 翻译和恢复按钮修改为在页面边缘附着的半透明半圆 -> 鼠标移入弹出翻译或恢复按钮
// ==/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: 10px;',
      '  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: 45px;',
      '  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')
        }
      })
    })
  }
}());