您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Tranlate images on boorus with cotrans.touhou.ai! Hit F8 to switch betweeen tranlated and original image. Hit F9 for configuration.
// ==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'; } }; }