// ==UserScript==
// @name Booru Cotrans
// @namespace http://tampermonkey.net/
// @version 0.1
// @description Tranlate images on boorus with cotrans.touhou.ai! Hit F8 to switch betweeen tranlated and original image. Hit F9 for configuration.
// @author Rhodan81
// @match https://gelbooru.com/index.php?page=post*
// @match https://rule34.xxx/index.php?page=post*
// @match https://chan.sankakucomplex.com/de/posts/*
// @match https://booru.allthefallen.moe/posts/*
// @match https://*.booru.org/index.php?page=post*
// @connect gelbooru.com
// @connect img3.gelbooru.com
// @connect rule34.xxx
// @connect wimg.rule34.xxx
// @connect api.cotrans.touhou.ai
// @connect s.sankakucomplex.com
// @connect chan.sankakucomplex.com
// @connect booru.allthefallen.moe
// @connect *.booru.org
// @connect img.booru.org
// @require https://openuserjs.org/src/libs/sizzle/GM_config.js
// @grant GM.xmlHttpRequest
// @grant GM_xmlhttpRequest
// @grant GM.setValue
// @grant GM_setValue
// @grant GM.getValue
// @grant GM_getValue
// @grant GM.deleteValue
// @grant GM_deleteValue
// @grant GM.addValueChangeListener
// @grant GM_addValueChangeListener
// @grant GM.removeValueChangeListener
// @grant GM_removeValueChangeListener
// ==/UserScript==
let gmc = new GM_config(
{
'id': 'BooruCotransConfig', // The id used for this instance of GM_config
'title': 'Booru Cotrans Settings', // Panel Title
'fields': // Fields object
{
'target_language': // This is the id of the field
{
'label': 'Target language', // Appears next to field
'type': 'select', // Makes this setting a text field
'options': ['English', 'Chinese (Simplified)', 'Chinese (Traditional)', 'Czech', 'Dutch', 'French', 'German', 'Hungarian', 'Italian', 'Japanese', 'Korean', 'Polish', 'Portuguese (Brazil)', 'Romanian', 'Russian', 'Spanish', 'Türkish', 'Ukrainian', 'Vietnames', 'Arabic', 'Serbian', 'Croation', 'Thai'], // Possible choices
'default': 'English' // Default value if user doesn't change it
},
'translator':
{
'label': 'Translation engine',
'type': 'select',
'options': ['gpt3.5', 'google','youdao','baidu','deepl','papago','offline'],
'default': 'gpt3.5'
},
'size':
{
'label': 'Translation resolution',
'type': 'select',
'options': ['1024x1024', '1536x1536','2048x2048','2560x2560'],
'default': '1536x1536'
},
'direction':
{
'label': 'Text direction',
'type': 'select',
'options': ['Automatic', 'Horizontal','Vertical'],
'default': 'Automatic'
},
},
'events':
{
'save': function () { // runs after values are saved
if (tranlated_image_src != null) {
tranlated_image_src = null;
}
if (translated == true) {
untranlate_image();
tranlate_image();
}
this.close();
}
}
});
let GMP
{
// polyfill functions
const GMPFunctionMap = {
xmlHttpRequest: typeof GM_xmlhttpRequest !== 'undefined' ? GM_xmlhttpRequest : undefined,
setValue: typeof GM_setValue !== 'undefined' ? GM_setValue : undefined,
getValue: typeof GM_getValue !== 'undefined' ? GM_getValue : undefined,
deleteValue: typeof GM_deleteValue !== 'undefined' ? GM_deleteValue : undefined,
addValueChangeListener: typeof GM_addValueChangeListener !== 'undefined' ? GM_addValueChangeListener : undefined,
removeValueChangeListener: typeof GM_removeValueChangeListener !== 'undefined' ? GM_removeValueChangeListener : undefined,
}
const xmlHttpRequest = GM.xmlHttpRequest.bind(GM) || GMPFunctionMap.xmlHttpRequest
GMP = new Proxy(GM, {
get(target, prop) {
if (prop === 'xmlHttpRequest') {
return (context) => {
return new Promise((resolve, reject) => {
xmlHttpRequest({
...context,
onload(event) {
context.onload?.()
resolve(event)
},
onerror(event) {
context.onerror?.()
reject(event)
},
})
})
}
}
if (prop in target) {
const v = target[prop]
return typeof v === 'function' ? v.bind(target) : v
}
if (prop in GMPFunctionMap && typeof GMPFunctionMap[prop] === 'function')
return GMPFunctionMap[prop]
console.error(`[Cotrans Manga Translator] GM.${prop} isn't supported in your userscript engine and it's required by this script. This may lead to unexpected behavior.`)
},
})
}
let image = document.getElementById('image');
let translated = false;
let tranlated_image_src;
let original_image_src;
const translating_idle_image = document.createElement("img");
translating_idle_image.src = '';
translating_idle_image.style.position = 'fixed';
translating_idle_image.style.bottom = '50px';
translating_idle_image.style.left = '50px';
translating_idle_image.style.display = 'none';
document.body.appendChild(translating_idle_image);
function key_up(e) {
if (e.code === 'F8') {
if (translated == false)
tranlate_image();
else
untranlate_image();
}
else if (e.code === 'F9') {
gmc.open();
}
}
function get_language() {
switch(gmc.get('target_language')) {
case 'English':
return 'ENG';
case 'Chinese (Simplified)':
return 'CHS';
case 'Chinese (Traditional)':
return 'CHT';
case 'Czech':
return 'CSY';
case 'Dutch':
return 'NLD';
case 'French':
return 'FRA';
case 'German':
return 'DEU';
case 'Hungarian':
return 'HUN';
case 'Italian':
return 'ITA';
case 'Japanese':
return 'JPN';
case 'Korean':
return 'KOR';
case 'Polish':
return 'PLK';
case 'Portuguese (Brazil)':
return 'PTB';
case 'Romanian':
return 'ROM';
case 'Russian':
return 'RUS';
case 'Spanish':
return 'ESP';
case 'Türkish':
return 'TRK';
case 'Ukrainian':
return 'UKR';
case 'Vietnames':
return 'VIN';
case 'Arabic':
return 'ARA';
case 'Serbian':
return 'SRP';
case 'Croation':
return 'HRV';
case 'Thai':
return 'THA';
default:
return 'ENG';
}
}
function get_translator() {
return gmc.get('translator');
}
function get_size() {
switch(gmc.get('size')) {
case '1024x1024':
return 'S';
case '1536x1536':
return 'M';
case '2048x2048':
return 'L';
case '2560x2560':
return 'X';
default:
return 'M';
}
}
function get_direction() {
switch(gmc.get('direction')) {
case 'Automatic':
return 'auto';
case 'Horizontal':
return 'h';
case 'Vertical':
return 'v';
default:
return 'auto';
}
}
document.addEventListener('keyup', key_up, false);
async function pullTranslationStatusPolling(id) {
while (true) {
console.info('Polling translation result');
const res = await GMP.xmlHttpRequest({
method: "GET",
url: `https://api.cotrans.touhou.ai/task/${id}/status/v1`
});
const msg = JSON.parse(res.responseText);
if (msg.type === "result") {
return msg.result;
}
await new Promise(resolve => setTimeout(resolve, 1e3));
}
}
async function untranlate_image() {
translated = false;
image.src = original_image_src;
}
async function tranlate_image() {
if (tranlated_image_src != null) {
image.src = tranlated_image_src;
translated = true;
return;
}
console.info('Tranlating image');
translating_idle_image.style.display = 'unset';
const result_get_blob = await GMP.xmlHttpRequest({
method: "GET",
responseType: "blob",
url: image.src,
overrideMimeType: "text/plain; charset=x-user-defined"});
let file_blob = result_get_blob.response;
const form_data = new FormData();
form_data.append("file", file_blob);
form_data.append("target_language", get_language());
form_data.append("detector", "default");
form_data.append("direction", get_direction());
form_data.append("translator", get_translator());
form_data.append("size", get_size());
form_data.append("retry", "false");
const result = await GMP.xmlHttpRequest({
method: "POST",
url: "https://api.cotrans.touhou.ai/task/upload/v1",
// @ts-expect-error FormData is supported
data: form_data
});
let responseText = JSON.parse(result.responseText);
let mask_url = responseText.result?.translation_mask;
if (!mask_url) {
const res = await pullTranslationStatusPolling(responseText.id);
mask_url = res.translation_mask;
}
const c = document.createElement("canvas");
var ctx=c.getContext("2d");
var imageObj1 = new Image();
var imageObj2 = new Image();
imageObj1.src = URL.createObjectURL(file_blob);
imageObj1.onload = function() {
c.width = imageObj1.width;
c.height = imageObj1.height;
ctx.drawImage(imageObj1, 0, 0);
imageObj2.src = mask_url;
imageObj2.crossOrigin = "anonymous";
imageObj2.onload = function() {
ctx.drawImage(imageObj2, 0, 0);
original_image_src = image.src;
tranlated_image_src = c.toDataURL("image/png");
image.src = tranlated_image_src;
translated = true;
translating_idle_image.style.display = 'none';
}
};
}