diff --git a/userscript.js b/userscript.js new file mode 100644 index 0000000..43f771a --- /dev/null +++ b/userscript.js @@ -0,0 +1,182 @@ + // ==UserScript== +// @name UEX汉化脚本 +// @namespace Violentmonkey Scripts +// @match https://uexcorp.space/* +// @grant none +// @version 0.2 +// @author CxJuice +// @description 2022/7/28 12:48:01 + +// ==/UserScript== +(function() { + 'use strict'; + + const SUPPORT_LANG = ["zh-CN", "ja"]; + const lang = (navigator.language || navigator.userLanguage); + const locales = getLocales(lang) + + translateByCssSelector(); + translateDesc(); + traverseElement(document.body); + watchUpdate(); + + function getLocales(lang) { + if(lang.startsWith("zh")) { // zh zh-TW --> zh-CN + lang = "zh-CN"; + } + if(SUPPORT_LANG.includes(lang)) { + return JSON.parse(GM_getResourceText(lang)); + } + return { + css: [], + dict: {} + }; + } + + function translateRelativeTimeEl(el) { + const datetime = $(el).attr('datetime'); + $(el).text(timeago.format(datetime, lang.replace('-', '_'))); + } + + function translateElement(el) { + // Get the text field name + let k; + if(el.tagName === "INPUT") { + if (el.type === 'button' || el.type === 'submit') { + k = 'value'; + } else { + k = 'placeholder'; + } + } else { + k = 'data'; + } + + const txtSrc = el[k].trim(); + const key = txtSrc.toLowerCase() + .replace(/\xa0/g, ' ') // replace ' ' + .replace(/\s{2,}/g, ' '); + + if(locales.dict[key]) { + el[k] = el[k].replace(txtSrc, locales.dict[key]) + } + } + + function shoudTranslateEl(el) { + const blockIds = ["readme", "wiki-content"]; + const blockClass = [ + "CodeMirror", + "css-truncate" // 过滤文件目录 + ]; + const blockTags = ["CODE", "SCRIPT", "LINK", "IMG", "svg", "TABLE", "ARTICLE", "PRE"]; + + if(blockTags.includes(el.tagName)) { + return false; + } + + if(el.id && blockIds.includes(el.id)) { + return false; + } + + if(el.classList) { + for(let clazz of blockClass) { + if(el.classList.contains(clazz)) { + return false; + } + } + } + + return true; + } + + function traverseElement(el) { + if(!shoudTranslateEl(el)) { + return + } + + for(const child of el.childNodes) { + if(["RELATIVE-TIME", "TIME-AGO"].includes(el.tagName)) { + translateRelativeTimeEl(el); + return; + } + + if(child.nodeType === Node.TEXT_NODE) { + translateElement(child); + } + else if(child.nodeType === Node.ELEMENT_NODE) { + if(child.tagName === "INPUT") { + translateElement(child); + } else { + traverseElement(child); + } + } else { + // pass + } + } + } + + function watchUpdate() { + const m = window.MutationObserver || window.WebKitMutationObserver; + const observer = new m(function (mutations, observer) { + for(let mutationRecord of mutations) { + for(let node of mutationRecord.addedNodes) { + traverseElement(node); + } + } + }); + + observer.observe(document.body, { + subtree: true, + characterData: true, + childList: true, + }); + } + + // translate "about" + function translateDesc() { + $(".repository-content .f4").append("
"); + $(".repository-content .f4").append("翻译"); + $("#translate-me").click(function() { + // get description text + const desc = $(".repository-content .f4") + .clone() + .children() + .remove() + .end() + .text() + .trim(); + + if(!desc) { + return; + } + + GM_xmlhttpRequest({ + onload: function(res) { + if (res.status === 200) { + $("#translate-me").hide(); + // render result + const text = res.responseText; + $(".repository-content .f4").append("TK翻译"); + $(".repository-content .f4").append("
"); + $(".repository-content .f4").append(text); + } else { + alert("翻译失败"); + } + } + }); + }); + } + + function translateByCssSelector() { + if(locales.css) { + for(var css of locales.css) { + if($(css.selector).length > 0) { + if(css.key === '!html') { + $(css.selector).html(css.replacement); + } else { + $(css.selector).attr(css.key, css.replacement); + } + } + } + } + } +})();