网站推荐、资源下载等 | 个人网站

【转】GitHub汉化脚本(谷歌浏览器)

  1 // ==UserScript==
  2 // @name         GitHub 汉化插件
  3 // @description  汉化 GitHub 界面的部分菜单及内容。
  4 // @copyright    2016, 楼教主 (http://www.52cik.com/)
  5 // @icon         https://assets-cdn.github.com/pinned-octocat.svg
  6 // @version      1.6.4
  7 // @author       楼教主
  8 // @license      MIT
  9 // @homepageURL  https://github.com/52cik/github-hans
 10 // @match        http://*.github.com/*
 11 // @match        https://*.github.com/*
 12 // @require      https://52cik.github.io/github-hans/locals.js?v1.6.4
 13 // @run-at       document-end
 14 // @grant        none
 15 // ==/UserScript==
 16 
 17 (function (window, document, undefined) {
 18     'use strict';
 19 
 20     var lang = 'zh'; // 中文
 21 
 22     // 2016-04-18 github 将 jquery 以 amd 加载,不暴露到全局了。
 23     // var $ = require('github/jquery')['default'];
 24 
 25     // 要翻译的页面
 26     var page = getPage();
 27 
 28     transTitle(); // 页面标题翻译
 29     timeElement(); // 时间节点翻译
 30     // setTimeout(contributions, 100); // 贡献日历翻译 (日历是内嵌或ajax的, 所以基于回调事件处理)
 31     walk(document.body); // 立即翻译页面
 32 
 33     // 2017-03-19 github 屏蔽 require 改为 Promise 形式的 ghImport
 34     define('github-hans-ajax', ['./jquery'], function($) {
 35         $(document).ajaxComplete(function () {
 36             transTitle();
 37             walk(document.body); // ajax 请求后再次翻译页面
 38         });
 39     });
 40     ghImport('github-hans-ajax')['catch'](function(e) {
 41         setTimeout(function() { throw e });
 42     });
 43 
 44     /**
 45      * 遍历节点
 46      *
 47      * @param {Element} node 节点
 48      */
 49     function walk(node) {
 50         var nodes = node.childNodes;
 51 
 52         for (var i = 0, len = nodes.length; i < len; i++) {
 53             var el = nodes[i];
 54             // todo 1. 修复多属性翻译问题; 2. 添加事件翻译, 如论预览信息;
 55 
 56             if (el.nodeType === Node.ELEMENT_NODE) { // 元素节点处理
 57 
 58                 // 元素节点属性翻译
 59                 if (el.tagName === 'INPUT' || el.tagName === 'TEXTAREA') { // 输入框 按钮 文本域
 60                     if (el.type === 'button' || el.type === 'submit') {
 61                         transElement(el, 'value');
 62                     } else {
 63                         transElement(el, 'placeholder');
 64                     }
 65                 } else if (el.hasAttribute('aria-label')) { // 带提示的元素,类似 tooltip 效果的
 66                     transElement(el, 'aria-label', true);
 67 
 68                     if (el.hasAttribute('data-copied-hint')) { // 复制成功提示
 69                         transElement(el.dataset, 'copiedHint');
 70                     }
 71                 } else if (el.tagName === 'OPTGROUP') { // 翻译 <optgroup> 的 label 属性
 72                     transElement(el, 'label');
 73                 }
 74 
 75                 if (el.hasAttribute('data-disable-with')) { // 按钮等待提示
 76                     transElement(el.dataset, 'disableWith');
 77                 }
 78 
 79                 // 跳过 readme, 文件列表, 代码显示
 80                 if (el.id !== 'readme' && !I18N.conf.reIgnore.test(el.className)) {
 81                     walk(el); // 遍历子节点
 82                 }
 83             } else if (el.nodeType === Node.TEXT_NODE) { // 文本节点翻译
 84                 transElement(el, 'data');
 85             }
 86 
 87         }
 88     }
 89 
 90     /**
 91      * 获取翻译页面
 92      */
 93     function getPage() {
 94         // 先匹配 body 的 class
 95         var page = document.body.className.match(I18N.conf.rePageClass);
 96 
 97         if (!page) { // 扩展 url 匹配
 98             page = location.href.match(I18N.conf.rePageUrl);
 99         }
100 
101         if (!page) { // 扩展 pathname 匹配
102             page = location.pathname.match(I18N.conf.rePagePath);
103         }
104 
105         return page ? page[1] || 'homepage' : false; // 取页面 key
106     }
107 
108     /**
109      * 翻译页面标题
110      */
111     function transTitle() {
112         var title = translate(document.title, 'title');
113 
114         if (title === false) { // 无翻译则退出
115             return false;
116         }
117 
118         document.title = title;
119     }
120 
121 
122     /**
123      * 翻译节点对应属性内容
124      *
125      * @param {object} el 对象
126      * @param {string} field 属性字段
127      * @param {boolean} isAttr 是否是 attr 属性
128      *
129      * @returns {boolean}
130      */
131     function transElement(el, field, isAttr) {
132         var transText = false; // 翻译后的文本
133 
134         if (isAttr === undefined) { // 非属性翻译
135             transText = translate(el[field], page);
136         } else {
137             transText = translate(el.getAttribute(field), page);
138         }
139 
140         if (transText === false) { // 无翻译则退出
141             return false;
142         }
143 
144         // 替换翻译后的内容
145         if (isAttr === undefined) {
146             el[field] = transText;
147         } else {
148             el.setAttribute(field, transText);
149         }
150     }
151 
152 
153     /**
154      * 翻译文本
155      *
156      * @param {string} text 待翻译字符串
157      * @param {string} page 页面字段
158      *
159      * @returns {string|boolean}
160      */
161     function translate(text, page) { // 翻译
162         var str;
163         var _key = text.trim(); // 去除首尾空格的 key
164         var _key_neat = _key
165             .replace(/\xa0/g, ' ') // 替换 &nbsp; 空格导致的 bug
166             .replace(/\s{2,}/g, ' '); // 去除多余换行空格等字符,(试验测试阶段,有问题再恢复)
167 
168         if (_key_neat === '') {
169             return false;
170         } // 内容为空不翻译
171 
172         str = transPage('pubilc', _key_neat); // 公共翻译
173 
174         if (str !== false && str !== _key_neat) { // 公共翻译完成
175             str = transPage('pubilc', str) || str;  // 二次公共翻译(为了弥补正则部分翻译的情况)
176             return text.replace(_key, str);  // 替换原字符,保留空白部分
177         }
178 
179         if (page === false) {
180             return false;
181         } // 未知页面不翻译
182 
183         str = transPage(page, _key_neat); // 翻译已知页面
184         if (str === false || str === '') {
185             return false;
186         } // 未知内容不翻译
187 
188         str = transPage('pubilc', str) || str; // 二次公共翻译(为了弥补正则部分翻译的情况)
189         return text.replace(_key, str); // 替换原字符,保留空白部分
190     }
191 
192 
193     /**
194      * 翻译页面内容
195      *
196      * @param {string} page 页面
197      * @param {string} key 待翻译内容
198      *
199      * @returns {string|boolean}
200      */
201     function transPage(page, key) {
202         var str; // 翻译结果
203         var res; // 正则数组
204 
205         // 静态翻译
206         str = I18N[lang][page]['static'][key];
207         if (str) {
208             return str;
209         }
210 
211         // 正则翻译
212         res = I18N[lang][page].regexp;
213         if (res) {
214             for (var i = 0, len = res.length; i < len; i++) {
215                 str = key.replace(res[i][0], res[i][1]);
216                 if (str !== key) {
217                     return str;
218                 }
219             }
220         }
221 
222         return false; // 没有翻译条目
223     }
224 
225 
226     /**
227      * 时间节点翻译
228      */
229     function timeElement() {
230         if (!window.RelativeTimeElement) { // 防止报错
231             return;
232         }
233 
234         var RelativeTimeElement$getFormattedDate = RelativeTimeElement.prototype.getFormattedDate;
235         var TimeAgoElement$getFormattedDate = TimeAgoElement.prototype.getFormattedDate;
236         // var LocalTimeElement$getFormattedDate = LocalTimeElement.prototype.getFormattedDate;
237 
238         var RelativeTime = function (str, el) { // 相对时间解析
239             if (/^on ([\w ]+)$/.test(str)) {
240                 return '于 ' + el.title.replace(/ .+$/, '');
241             }
242 
243             // 使用字典公共翻译的第二个正则翻译相对时间
244             var time_ago = I18N[lang].pubilc.regexp[1];
245             return str.replace(time_ago[0], time_ago[1]);
246         };
247 
248         RelativeTimeElement.prototype.getFormattedDate = function () {
249             var str = RelativeTimeElement$getFormattedDate.call(this);
250             return RelativeTime(str, this);
251         };
252 
253         TimeAgoElement.prototype.getFormattedDate = function () {
254             var str = TimeAgoElement$getFormattedDate.call(this);
255             return RelativeTime(str, this);
256         };
257 
258         LocalTimeElement.prototype.getFormattedDate = function () {
259             return this.title.replace(/ .+$/, '');
260         };
261 
262         // 遍历 time 元素进行翻译
263         // 2016-04-16 github 改版,不再用 time 标签了。
264         var times = document.querySelectorAll('time, relative-time, time-ago, local-time');
265         Array.prototype.forEach.call(times, function (el) {
266             if (el.getFormattedDate) { // 跳过未注册的 time 元素
267                 el.textContent = el.getFormattedDate();
268             }
269         });
270     }
271 
272 
273     /**
274      * 贡献日历 基于事件翻译
275      */
276     function contributions() {
277         var tip = document.getElementsByClassName('svg-tip-one-line');
278 
279         // 等待 IncludeFragmentElement 元素加载完毕后绑定事件
280         // var observe = require('github/observe').observe;
281 
282         define('github/hans-contributions', ['./observe'], function (observe) {
283             observe(".js-calendar-graph-svg", function () {
284                 setTimeout(function () { // 延时绑定 mouseover 事件,否则没法翻译
285                     var $calendar = $('.js-calendar-graph');
286                     walk($calendar[0]); // 翻译日历部分
287 
288                     $calendar.on('mouseover', '.day', function () {
289                         if (tip.length === 0) { // 没有 tip 元素时退出防止报错
290                             return true;
291                         }
292 
293                         var data = $(this).data(); // 获取节点上的 data
294                         var $tip = $(tip[0]);
295 
296                         $tip.html(data.count + ' 次贡献 ' + data.date);
297 
298                         var rect = this.getBoundingClientRect(); // 获取元素位置
299                         var left = rect.left + window.pageXOffset - tip[0].offsetWidth / 2 + 5.5;
300 
301                         $tip.css('left', left);
302                     });
303                 }, 999);
304             });
305         });
306 
307         ghImport('github/hans-contributions')['catch'](function(e) {
308             setTimeout(function() { throw e });
309         });
310     }
311 
312 })(window, document);

转载于楼教主GitHub:https://github.com/52cik/github-hans/blob/gh-pages/main.js


 

posted @ 2018-09-10 19:49  xiaostudy  阅读(1937)  评论(0编辑  收藏  举报
网站推荐
[理工最爱]小时百科 |  GitHub |  Gitee |  开源中国社区 |  牛客网 |  不学网论坛 |  r2coding |  冷熊简历 |  爱盘 |  零散坑 |  bootstrap中文网 |  vue.js官网教程 |  源码分享站 |  maven仓库 |  楼教主网站 |  廖雪峰网站 |  w3cschool |  在线API |  代码在线运行 |  [不学网]代码在线运行 |  JS在线运行 |  PHP中文网 |  深度开源eclipse插件 |  文字在线加密解密 |  菜鸟教程 |  慕课网 |  千图网 |  手册网 |  素材兔 |  盘多多 |  悦书PDF |  sumatra PDF |  calibre PDF |  Snipaste截图 |  shareX截图 |  vlc-media-player播放器 |  MCMusic player |  IDM下载器 |  格式工厂 |  插件网 |  谷歌浏览器插件 |  Crx搜搜 |  懒人在线计算器 |  leetcode算法题库 |  layer官网 |  layui官网 |  formSelects官网 |  Fly社区 |  程序员客栈 |  融云 |  华为云 |  阿里云 |  ztree官网API |  teamviewer官网 |  sonarlint官网 |  editormd |  pcmark10官网 |  crx4chrome官网 |  apipost官网 |  花生壳官网 |  serv-u官网 |  杀毒eset官网 |  分流抢票bypass官网 |  懒猴子CG代码生成器官网 |  IT猿网 |  natapp[内网穿透] |  ngrok[内网穿透] |  深蓝穿透[内网穿透] |  WakeMeOnLan[查看ip] |  iis7 |  [漏洞扫描]Dependency_Check官网 |  [图标UI]fontawesome官网 |  idea插件官网 |  路过图床官网 |  sha256在线解密 |  在线正则表达式测试 |  在线文件扫毒 |  KuangStudy | 
资源下载
电脑相关: Windows原装下载msdn我告诉你 |  U盘制作微PE工具官网下载 |  Linux_CentOS官网下载 |  Linux_Ubuntu官网下载 |  Linux_OpenSUSE官网下载 |  IE浏览器官网下载 |  firefox浏览器官网下载 |  百分浏览器官网下载 |  谷歌google浏览器历史版本下载 |  深度deepin系统官网下载 |  中兴新支点操作系统官网下载 |  文件对比工具Beyond Compare官网下载 |  开机启动程序startup-delayer官网下载 |  openoffice官网下载 |  utorrent官网下载 |  qbittorrent官网下载 |  cpu-z官网下载 |  蜘蛛校色仪displaycal官网下载 |  单文件制作greenone下载 |  win清理工具Advanced SystemCare官网下载 |  解压bandizip官网下载 |  内存检测工具memtest官网下载 |  磁盘坏道检测与修复DiskGenius官网下载 |  磁盘占用可视化SpaceSniffer官网下载 |  [磁盘可视化]WizTree官网下载 |  win快速定位文件Everything官网下载 |  文件定位listary官网下载 |  动图gifcam官网下载 |  7-Zip官网下载 |  磁盘分区工具diskgenius官网下载 |  CEB文件查看工具Apabi Reader官网下载 |  罗技鼠标options官网下载 |  [去除重复文件]doublekiller官网下载 | 
编程相关: ApacheServer官网下载 |  Apache官网下载 |  Git官网下载 |  Git高速下载 |  Jboss官网下载 |  Mysql官网下载 |  Mysql官网历史版本下载 |  NetBeans IDE官网下载 |  Spring官网下载 |  Nginx官网下载 |  Resin官网下载 |  Tomcat官网下载 |  jQuery历史版本下载 |  nosql官网下载 |  mongodb官网下载 |  mongodb_linux历史版本下载 |  mongodb客户端下载 |  VScode官网下载 |  cxf官网下载 |  maven官网下载 |  QT官网下载 |  SVN官网下载 |  SVN历史版本下载 |  nodeJS官网下载 |  oracle官网下载 |  jdk官网下载 |  STS官网下载 |  STS历史版本官网下载 |  vue官网下载 |  virtualbox官网下载 |  docker desktop官网下载 |  github desktop官网下载 |  EditPlus官网下载 |  zTree下载 |  layui官网下载 |  jqgrid官网下载 |  jqueryui官网下载 |  solr历史版本下载 |  solr分词器ik-analyzer-solr历史版本下载 |  zookeeper历史版本官网下载 |  nssm官网下载 |  elasticsearch官网下载 |  elasticsearch历史版本官网下载 |  redis官网下载 |  redis历史版本官网下载 |  redis的win版本下载 |  putty官网下载 |  查看svn密码TSvnPD官网下载 |  MongoDB连接工具Robo官网下载 |  dll查看exescope官网下载 |  dll2c官网下载 |  接口测试apipost官网下载 |  接口测试postman官网下载 |  原型设计工具AxureRP官网下载 |  canal官网下载 |  idea主题样式下载 |  vue的GitHub下载 |  finalShell官网下载 |  ETL工具kafka官网下载 |  cavaj[java反编译]官网下载 |  jd-gui[java反编译]官网下载 |  radmin[远程连接]官网下载 |  tcping[win ping端口]下载 |  jQueryUploadFile官网下载 |  RedisPlus下载 |  aiXcoder智能编程助手官网下载 |  [表单效验]validform官网下载 |  idea官网下载 |  RedisStudio下载 |  MD转word含公式pandoc官网下载 |  logviewer官网下载 |  Kafka官网下载 |  hbase高速下载 |  hadoop官网下载 |  hadooponwindows的GitHub下载 |  hive官网下载 |  soapui官网下载 |  flink官网下载 |  kafkatool官网下载 |  MinIO官网下载 |  MinIO中国镜像下载 | 
办公相关工具
免费在线拆分PDF【不超过30M】 |  免费在线PDF转Word【不超过10M】 |  在线文字识别转换【不超过1M】 |  PDF转换成Word【不超过50M】 |  在线OCR识别 |  Smallpdf |  文件转换器Convertio |  迅捷PDF转换器 |  字母大小写转换工具 |  档铺 |  快传airportal[可文字] |  快传-文叔叔 |  P2P-小鹿快传 |  [图床]ImgURL | 
网站入口
腾讯文档 |  有道云笔记网页版 |  为知笔记网页版 |  印象笔记网页版 |  蓝奏云 |  QQ邮箱 |  MindMaster在线思维导图 |  bilibili |  PDM文件在线打开 |  MPP文件在线打开 |  在线PS软件 |  在线WPS |  阿里云企业邮箱登陆入口 | 
其他
PDF转换 |  悦书PDF转换 |  手机号注册查询 |  Reg007 |  akmsg |  ip8_ip查询 |  ipip_ip查询 |  天体运行testtubegames |  测试帧率 |  在线网速测试 |