/* eslint-disable no-multi-spaces */
// ==UserScript==
// @name Kemono中文翻译按钮
// @namespace Kemono-Translate
// @version 0.6.1
// @description 一键翻译Content内容到中文
// @author PY-DNG
// @license GPL-v3
// @match https://kemono.party/fanbox/user/*/post/*
// @match https://kemono.party/patreon/user/*/post/*
// @match https://kemono.su/fanbox/user/*/post/*
// @match https://kemono.su/patreon/user/*/post/*
// @require https://greasyfork.org/scripts/456034-basic-functions-for-userscripts/code/script.js?version=1226884
// @require https://greasyfork.org/scripts/452362-baidu-translate/code/Baidu%20Translate.js?version=1175971
// @connect baidu.com
// @icon 
// @grant GM_xmlhttpRequest
// @run-at document-start
// ==/UserScript==
/* global LogLevel DoLog Err $ $All $CrE $AEL $$CrE addStyle detectDom destroyEvent copyProp copyProps parseArgs escJsStr replaceText getUrlArgv dl_browser dl_GM AsyncManager */
/* global bdTransReady baidu_translate */
(function __MAIN__() {
'use strict';
// Constances
const CONST = {
TextAllLang: {
DEFAULT: 'zh-CN',
'zh-CN': {
Initing: '初始化...',
Translate: '翻译',
Translating: '翻译中...',
ShowOriginal: '显示原文',
ShowTrans: '显示译文'
}
}
};
// Init language
const i18n = !Object.keys(CONST.TextAllLang).includes(navigator.language) ? navigator.language : CONST.TextAllLang.DEFAULT;
CONST.Text = CONST.TextAllLang[i18n];
detectDom('.post__footer', content => detectDom('.post__actions>*:nth-child(2)', fav => main()));
function main() {
const content = $('.post__content');
if (!content) {return false;}
// Content
const container = $('.post__actions');
const button = container.children[1].cloneNode(true);
button.children[0].remove();
const button_text = button.children[0];
button_text.innerText = CONST.Text.Initing;
container.appendChild(button);
transButton(button, button_text, content);
// Comment
const comments = $All('.comment');
for (const comment of comments) {
const header = $(comment, '.comment__header');
const btn = $$CrE({
tagName: 'span',
styles: {
float: 'right',
cursor: 'pointer'
},
props: {
innerText: CONST.Text.Initing
}
});
const content = $(comment, '.comment__body');
header.appendChild(btn);
transButton(btn, btn, content);
}
function transButton(button, button_text, content) {
let instantTrans = false;
$AEL(button, 'click', function() {
instantTrans = true;
}, {once: true});
bdTransReady(function() {
button_text.innerText = CONST.Text.Translate;
const Translator = new NodeTranslator();
let tranlated = false;
let hasTrans = false;
$AEL(button, 'click', switchTrans);
instantTrans && switchTrans();
function switchTrans() {
(tranlated ? Translator.recover : Translator.translate)(content, onTrans);
function onTrans(status) {
const node = this;
if (node === content) {
switch (status) {
case -1: {
button_text.innerText = hasTrans ? CONST.Text.ShowTrans : CONST.Text.Translate;
tranlated = false;
break;
}
case 0: {
button_text.innerText = CONST.Text.Translating;
break;
}
case 1: {
button_text.innerText = CONST.Text.ShowOriginal;
tranlated = true;
hasTrans = true;
break;
}
}
}
}
}
});
}
}
function NodeTranslator() {
const NT = this;
NT.transMap = new Map();
NT.translate = translate;
NT.recover = recover;
// Translate whole node tree
// callback: [all nodes in tree].forEach(node => callback.call(node, status));
// status = {-1: untranslated, 0: translating, 1: translated}
// Returns a boolean repersents whether translate progress can start. (if already translated or during translating, retuns false)
function translate(node, callback) {
// Get node text obj
const text = getTextObj(node);
// Translate
if (text.status !== 0) {
if (node.childNodes.length) {
const AM = new AsyncManager();
for (const childNode of node.childNodes) {
translate(childNode, function(status) {
status === 1 && AM.finish();
callback.apply(this, arguments);
if (text.status === -1 && status === 0) {
text.status = status;
callback.call(node, text.status);
}
}) && AM.add();
}
AM.onfinish = function() {
text.status = 1;
callback.call(node, 1);
}
AM.finishEvent = true;
return true;
} else {
return transTerminalNode(node, callback);
}
} else {
return false;
}
}
// Show original text for whole node tree
// callback: [all nodes in tree].forEach(node => callback.call(node, status));
// status = {-1: untranslated, 0: translating, 1: translated}
// Returns a boolean repersents whether recover progress can start. (if not translated or during translating, retuns false)
function recover(node, callback) {
// Get node text obj
const text = getTextObj(node);
// Recover
if (text.status !== 0) {
if (node.childNodes.length) {
const AM = new AsyncManager();
for (const childNode of node.childNodes) {
recover(childNode, function(status) {
status === -1 && AM.finish();
callback.apply(this, arguments);
}) && AM.add();
}
AM.onfinish = function() {
text.status = -1;
callback.call(node, text.status);
}
AM.finishEvent = true;
return true;
} else {
return recoverTerminalNode(node, callback);
}
} else {
return false;
}
}
// Translate single terminal node
// callback: callback.call(node, status=1);
// status = {-1: untranslated, 0: translating, 1: translated}
// Returns a boolean repersents whether translate progress can start. (if already translated or during translating, retuns false)
function transTerminalNode(node, callback) {
// No need to translate if nothing more than whitespaces
if (!node.nodeValue || !node.nodeValue.trim()) {return false;}
// No translating for links
if (node.nodeValue.trim().match(/^https?:\/\//)) {return false;}
// Get node text obj
const text = getTextObj(node);
// Returns true only when untranslated
if (text.status === -1) {
if (text.hasTrans) {
node.nodeValue = text.trans;
text.status = 1;
// translated, callback
setTimeout(callback.bind(node, text.status), 0);
} else {
baidu_translate({
text: text.ori,
dst: 'zh',
callback: function(result_text) {
text.trans = node.nodeValue = result_text;
text.status = 1;
text.hasTrans = true;
// translated, callback
callback.call(node, text.status);
},
onerror: function(reason) {
DoLog(LogLevel.Error, 'translate error');
}
});
text.status = 0;
// translating, callback
callback.call(node, text.status);
}
return true;
} else {
return false;
}
}
// Show original text of single terminal node
// callback.call(node, status=-1);
// status = {-1: untranslated, 0: translating, 1: translated}
// Returns a boolean repersents whether recover progress can start. (if not translated or during translating, retuns false)
function recoverTerminalNode(node, callback) {
// Get node text obj
const text = getTextObj(node);
// Returns true only when translated
if (text.status === 1) {
node.nodeValue = text.ori;
text.status = -1;
// recovered, callback
setTimeout(callback.bind(node, text.status), 0);
return true;
} else {
return false;
}
}
function getTextObj(node) {
// Init node text obj
!NT.transMap.has(node) && NT.transMap.set(node, {
ori: node.nodeValue,
trans: null,
status: -1,
hasTrans: false
});
// Get node text obj
return NT.transMap.get(node);
}
}
})();