网页翻译

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

2020-06-26 يوللانغان نەشرى. ئەڭ يېڭى نەشرىنى كۆرۈش.

  1. // ==UserScript==
  2. // @name 网页翻译
  3. // @author Kaiter-Plus
  4. // @namespace https://gitee.com/Kaiter-Plus/TampermonkeyScript/tree/master/Translate
  5. // @description 给每个非中文的网页左下角添加一个google翻译图标,直接调用 Google 的翻译接口对非中文网页进行翻译
  6. // @include *://*
  7. // @exclude /^[http|https].*[0-9]+\.[0-9]+\.[0-9]+\.[0-9]+/
  8. // @exclude /lanzous\.com/
  9. // @exclude /w3school.*cn/
  10. // @exclude /iqiyi\.com/
  11. // @exclude /.*baidu.*/
  12. // @exclude /cnblogs\.com/
  13. // @exclude /csdn\.net/
  14. // @exclude /zhku\.edu\.cn/
  15. // @exclude /zhihuishu\.com/
  16. // @exclude /runoob\.com/
  17. // @exclude /aliyuncs\.com/
  18. // @exclude /.*chaoxing\.com/
  19. // @exclude /examcoo\.com/
  20. // @exclude /.*mooc\.com/
  21. // @exclude /52pojie\.cn/
  22. // @exclude /.*bilibili\.com/
  23. // @exclude /.*qq\.com/
  24. // @exclude /localhost/
  25. // @icon 
  26. // @version 1.21
  27. // @noframes
  28. // @note 2020/03/26 网页整页翻译功能
  29. // @note 2020/04/13 排除纯ip网址
  30. // @note 2020/04/14 移除翻译后顶边栏
  31. // @note 2020/05/01 排除百度、QQ、超星等中文网址
  32. // @note 2020/05/04 修复去除上边栏网页先向下再向上跳的Bug
  33. // @note 2020/05/05 尝试修复百度出现超粗顶栏的Bug
  34. // @note 2020/05/12 添加恢复原网页的按钮(翻译按钮旁边),有点丑,不过希望可以先用着,有时间再看看能不能弄好看一点ヾ(≧▽≦*)o
  35. // @note 2020/05/23 稍微修改了一下恢复原网页的按钮的样式(还是不好看)
  36. // @note 2020/05/26 修改脚本为原生javascript,兼容暴力猴
  37. // @note 2020/05/26 修改翻译栏样式,固定宽高,防止在一些页面上出现太宽或太高的现象
  38. // @note 2020/06/06 修复火狐浏览器(firefox),内存溢出的bug,精简了一点代码
  39. // @note 2020/06/08 排除一些代码块的翻译,如果还有其它的网站的代码块需要排除,可以反馈给我,我排除一下
  40. // @note 2020/06/17 修改恢复原网页按钮的样式(使用@picasso250的样式),排除标签 tt
  41. // @note 2020/06/18 适配Quora
  42. // @note 2020/06/26 翻译和恢复按钮修改为在页面边缘附着的半透明半圆 -> 鼠标移入弹出翻译或恢复按钮
  43. // ==/UserScript==
  44.  
  45. (function () {
  46. 'use strict'
  47.  
  48. // 获取 head
  49. const head = document.head
  50. // 获取body
  51. const body = document.body
  52. // 获取当前页面的语言
  53. const lang = document.documentElement.lang
  54.  
  55. // 判断是不是中文,如果不是则执行
  56. if (lang.substr(0, 2) !== 'zh') {
  57.  
  58. // 创建网页元素方法
  59. function createElement(html, nodeText, attr, parent) {
  60. const element = document.createElement(nodeText)
  61. if(attr){
  62. element[attr] = html
  63. }else{
  64. element.innerHTML = html
  65. }
  66. parent.appendChild(element)
  67. }
  68.  
  69. // 自定义样式,隐藏顶部栏
  70. createElement([
  71. '#google_translate_element {',
  72. ' position: fixed;',
  73. ' left: 0px;',
  74. ' bottom: 10px;',
  75. ' height: 22px;',
  76. ' border-radius: 11px;',
  77. ' z-index: 10000000;',
  78. ' overflow: hidden;',
  79. ' box-shadow: 1px 1px 3px 0 #888;',
  80. ' opacity: .5;',
  81. ' transform: translateX(-85%);',
  82. ' transition: all .3s;',
  83. '}',
  84. '#google_translate_element:hover {',
  85. ' opacity: 1;',
  86. ' transform: translateX(0);',
  87. '}',
  88. '#google_translate_element .goog-te-gadget-simple {',
  89. ' width: 100%;',
  90. '}',
  91. '.goog-te-banner-frame.skiptranslate {',
  92. ' display: none',
  93. '}',
  94. 'html,body{',
  95. ' top: 0!important',
  96. '}',
  97. '.recoverPage {',
  98. ' width: 4em;',
  99. ' background-color: #fff;',
  100. ' position: fixed;',
  101. ' left: 0px;',
  102. ' z-index: 10000000;',
  103. ' bottom: 45px;',
  104. ' user-select: none;',
  105. ' text-align: center;',
  106. ' border: 1px solid #a8a8a8;',
  107. ' font-size: small;',
  108. ' line-height: 2em;',
  109. ' border-radius: 1em;',
  110. ' box-shadow: 1px 1px 3px 0 #888;',
  111. ' opacity: .5;',
  112. ' transform: translateX(-75%);',
  113. ' transition: all .3s;',
  114. '}',
  115. '.recoverPage:hover {',
  116. ' opacity: 1;',
  117. ' transform: translateX(0);',
  118. '}',
  119. '.recoverPage:active {',
  120. ' box-shadow: 1px 1px 3px 0 #888 inset;',
  121. '}'
  122. ].join('\n'), "style", '', head)
  123.  
  124. createElement('google_translate_element', 'div', 'id', body)
  125.  
  126.  
  127. // 初始化
  128. const initHtml = [
  129. "function googleTranslateElementInit() {",
  130. " let google_translate_element = document.getElementById('google_translate_element');",
  131. " let timer = setInterval(function() {",
  132. " google_translate_element = document.getElementById('google_translate_element');",
  133. " if (google_translate_element) {",
  134. " clearInterval(timer);",
  135. " new google.translate.TranslateElement({",
  136. " pageLanguage: 'auto',",
  137. " //包括的语言,中文简体,中文繁体,英语,日语,俄语",
  138. " includedLanguages: 'zh-CN,zh-TW,en,ja,ru',",
  139. " /*0,原生select,并且谷歌logo显示在按钮下方。",
  140. " 1,原生select,并且谷歌logo显示在右侧。",
  141. " 2,完全展开语言列表,适合pc。",
  142. " */",
  143. " layout: /mobile/i.test(navigator.userAgent) ? 0 : 2,",
  144. " }, 'google_translate_element');",
  145. " // 清除图片的请求,加快访问速度",
  146. " img = [].slice.call(document.querySelectorAll('#goog-gt-tt img,#google_translate_element img'));",
  147. " img.forEach(function(v, i) {",
  148. " v.src = '';",
  149. " });",
  150. " const recoverPage = document.createElement('div')",
  151. " recoverPage.setAttribute('class', 'notranslate recoverPage')",
  152. " recoverPage.innerText = '恢复'",
  153. " document.body.appendChild(recoverPage)",
  154. " // 点击恢复原网页",
  155. " recoverPage.onclick = (() => {",
  156. " const recoverIframe = document.getElementById(':2.container')",
  157. " if (recoverIframe) {",
  158. " const recoverDocument = recoverIframe.contentWindow.document",
  159. " recoverDocument.getElementById(':2.restore').click()",
  160. " }",
  161. " })",
  162. " }",
  163. " }, 300);",
  164. "}"
  165. ].join("\n")
  166. createElement(initHtml,'script', '', head)
  167.  
  168. // 导入翻译接口
  169. if (/quora/i.test(location.href)) {
  170. createElement('//translate.google.com/translate_a/element.js?&cb=googleTranslateElementInit','script','src', head)
  171. } else {
  172. createElement('//translate.google.cn/translate_a/element.js?&cb=googleTranslateElementInit','script','src', head)
  173. }
  174.  
  175. // 排除一些代码的翻译
  176. const noTranslateArray = [
  177. '.bbCodeCode',
  178. 'tt'
  179. ]
  180. noTranslateArray.forEach(selectorName => {
  181. [...document.querySelectorAll(selectorName)].forEach(node => {
  182. if (node.className.indexOf('notranslate') === -1) {
  183. node.classList.add('notranslate')
  184. }
  185. })
  186. })
  187. }
  188. }());