// ==UserScript== // @name UEX——CN // @name:zh-CN UEX汉化脚本 // @namespace https://github.com/CxJuice/Uex_Chinese_Translate // @version 0.8.0 // @description UEX——CN // @description:zh UEX汉化插件 // @description:zh-CN UEX汉化插件 // @author CxJuice // @match https://uexcorp.space/* // @match https://www.erkul.games/* // @grant GM_xmlhttpRequest // @grant GM_getResourceText // @resource zh-CN https://cdn.jsdelivr.net/gh/CxJuice/Uex_Chinese_Translate@main/zh-CN-uex1.7.json // @require https://cdn.bootcdn.net/ajax/libs/timeago.js/4.0.2/timeago.full.min.js // @require https://cdn.bootcdn.net/ajax/libs/jquery/3.5.1/jquery.min.js // ==/UserScript== window.addEventListener('load', function(event) { // 当页面加载完成时,执行我们的脚本 myScript(); }); // 监听页面中的鼠标点击事件 document.addEventListener('click', function(event) { // 当发生鼠标点击事件时,执行我们的脚本 myScript(); }); function myScript() { (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 = []; const blockClass = [ "css-truncate" // 过滤文件目录 ]; const blockTags = [ "IMG", "svg"]; 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); } } } } } })(); }