您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
在网页中的日语外来语上方标注英文原词
当前为
// ==UserScript== // @name Katakana Terminator // @description Convert gairaigo (Japanese loan words) back to English // @author Arnie97 // @license MIT; https://mit-license.org // @copyright 2017-2021, Katakana Terminator Contributors (https://github.com/Arnie97/katakana-terminator/graphs/contributors) // @namespace https://github.com/Arnie97 // @homepageURL https://github.com/Arnie97/katakana-terminator // @supportURL https://gf.qytechs.cn/scripts/33268/feedback // @icon https://upload.wikimedia.org/wikipedia/commons/2/28/Ja-Ruby.png // @match *://*/* // @exclude *://*.bilibili.com/video/* // @grant GM.xmlHttpRequest // @grant GM_xmlhttpRequest // @grant GM_addStyle // @connect translate.google.cn // @version 2021.05.24 // @name:ja-JP カタカナターミネーター // @name:zh-CN 片假名终结者 // @description:zh-CN 在网页中的日语外来语上方标注英文原词 // ==/UserScript== // define some shorthands var _ = document; var queue = {}; // {"カタカナ": [rtNodeA, rtNodeB]} var cachedTranslations = {}; // {"ターミネーター": "Terminator"} // Inspired by https://www.the-art-of-web.com/javascript/search-highlight/ function scanTextNodes(node) { var excludeTags = {ruby: true, script: true, select: true, textarea: true}; if (node.nodeName.toLowerCase() in excludeTags) { return; } else if (node.hasChildNodes()) { return Array.prototype.slice.call(node.childNodes).forEach(scanTextNodes); } else if (node.nodeType == 3) { while ((node = addRuby(node))); } } // Inspired by http://www.the-art-of-web.com/javascript/search-highlight/ function addRuby(node) { var katakana = /[\u30A1-\u30FA\u30FD-\u30FF][\u3099\u309A\u30A1-\u30FF]*[\u3099\u309A\u30A1-\u30FA\u30FC-\u30FF]|[\uFF66-\uFF6F\uFF71-\uFF9D][\uFF65-\uFF9F]*[\uFF66-\uFF9F]/, match; if (!node.nodeValue || !(match = katakana.exec(node.nodeValue))) { return false; } var ruby = _.createElement('ruby'); ruby.appendChild(_.createTextNode(match[0])); var rt = _.createElement('rt'); rt.classList.add('katakana-terminator-rt'); queue[match[0]] = queue[match[0]] || []; queue[match[0]].push(rt); ruby.appendChild(rt); var after = node.splitText(match.index); node.parentNode.insertBefore(ruby, after); after.nodeValue = after.nodeValue.substring(match[0].length); return after; } // Forked from https://stackoverflow.com/a/34209399 function buildURL(base, params) { var query = Object.keys(params).map(function(k) { var esc = encodeURIComponent; // Support arrays in parameters, just like Python Requests // {keyA: 1, keyB: [2, 3]} => '?keyA=1&keyB=2&keyB=3' if (params[k] instanceof Array) { return params[k].map(function(v) { return esc(k) + '=' + esc(v); }).join('&'); } return esc(k) + '=' + esc(params[k]); }).join('&'); return base + '?' + query; } // Split word list into chunks to limit the length of API requests function translateTextNodes() { var apiRequestCount = 0; var phraseCount = 0; var chunkSize = 200; var chunk = []; for (var phrase in queue) { if (!queue.hasOwnProperty(phrase)) { continue; } phraseCount++; if (phrase in cachedTranslations) { updateRubyByCachedTranslations(phrase); continue; } chunk.push(phrase); if (chunk.length >= chunkSize) { apiRequestCount++; googleTranslate('ja', 'en', chunk); chunk = []; } } if (chunk.length) { apiRequestCount++; googleTranslate('ja', 'en', chunk); } console.debug('Katakana Terminator:', phraseCount, 'phrases translated in', apiRequestCount, 'requests, frame', window.location.href); } function googleTranslate(srcLang, destLang, phrases) { // Prevent duplicate HTTP requests before the request completes phrases.forEach(function(phrase) { cachedTranslations[phrase] = null; }); var joinedText = phrases.join('\n').trim(); var api = 'https://translate.google.cn/translate_a/single'; var params = { client: 't', sl: srcLang, tl: destLang, dt: ['rm', 't'], tk: googleToken(joinedText), q: joinedText, }; GM_xmlhttpRequest({ method: "GET", url: buildURL(api, params), onload: function(dom) { var escapedResult = dom.responseText.replace("'", '\u2019'); var translations = JSON.parse(escapedResult)[0]; phrases.forEach(function(phrase, i) { cachedTranslations[phrase] = translations[i][0].trim(); updateRubyByCachedTranslations(phrase); }); }, onerror: function() { phrases.forEach(function(phrase) { if (cachedTranslations[phrase]) { delete cachedTranslations[phrase]; } }); }, }); } function updateRubyByCachedTranslations(phrase) { if (!cachedTranslations[phrase]) { return; } (queue[phrase] || []).forEach(function(node) { node.dataset.rt = cachedTranslations[phrase]; }); delete queue[phrase]; } // Forked from https://github.com/cocoa520/Google_TK function googleToken(r) { for(var a=406644,e=[],h=0,n=0;n<r.length;n++){var o=r.charCodeAt(n);128>o?e[h++]=o:(2048>o?e[h++]=o>>6|192:(55296==(64512&o)&&n+1<r.length&&56320==(64512&r.charCodeAt(n+1))?(o=65536+((1023&o)<<10)+(1023&r.charCodeAt(++n)),e[h++]=o>>18|240,e[h++]=o>>12&63|128):e[h++]=o>>12|224,e[h++]=o>>6&63|128),e[h++]=63&o|128)}function t(r,t){for(var a=0;a<t.length-2;a+=3){var e=(e=t.charAt(a+2))>="a"?e.charCodeAt(0)-87:Number(e),e="+"==t.charAt(a+1)?r>>>e:r<<e;r="+"==t.charAt(a)?r+e&4294967295:r^e}return r}for(r=a,h=0;h<e.length;h++)r+=e[h],r=t(r,"+-a^+6");return r=t(r,"+-3^+b+-f"),0>(r^=3293161072)&&(r=2147483648+(2147483647&r)),(r%=1e6).toString()+"."+(r^a) } function main() { GM_addStyle("rt.katakana-terminator-rt::before { content: attr(data-rt); }"); var domChangedSinceLastScan = true; var observer = new MutationObserver(function() { domChangedSinceLastScan = true; }); observer.observe(_.body, {childList: true, subtree: true}); function rescanTextNodes() { if (!domChangedSinceLastScan) { return; } // Deplete buffered mutations observer.takeRecords(); domChangedSinceLastScan = false; scanTextNodes(_.body); translateTextNodes(); } // Limit the frequency of API requests rescanTextNodes(); setInterval(rescanTextNodes, 500); } // Polyfill for Greasemonkey 4 if (typeof GM_xmlhttpRequest === 'undefined' && typeof GM === 'object' && typeof GM.xmlHttpRequest === 'function') { GM_xmlhttpRequest = GM.xmlHttpRequest; } if (typeof GM_addStyle === 'undefined') { GM_addStyle = function(css) { var head = _.getElementsByTagName('head')[0]; if (!head) { return null; } var style = _.createElement('style'); style.setAttribute('type', 'text/css'); style.textContent = css; head.appendChild(style); return style; }; } main();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址