您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Upload/rehost images to catbox.moe directly from upload page
// ==UserScript== // @name Catbox Droptarget for Oppaitime // @namespace https://greasyfork.org/users/390979-parliament // @version 1.51 // @description Upload/rehost images to catbox.moe directly from upload page // @author Anakunda // @iconURL https://catbox.moe/pictures/favicon.ico // @match https://oppaiti.me/upload.php* // @match https://oppaiti.me/torrents.php?action=edit* // @match https://oppaiti.me/requests.php?action=new* // @match https://oppaiti.me/requests.php?action=edit* // @match https://oppaiti.me/reports.php?action=report* // @match https://oppaiti.me/reportsv2.php?* // @match https://oppaiti.me/artist.php?action=edit* // @connect catbox.moe // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @grant GM_deleteValue // @grant GM_log // ==/UserScript== 'use strict'; String.prototype.toASCII = function() { return this.normalize("NFKD").replace(/[\x00-\x1F\u0080-\uFFFF]/g, ''); } document.head.appendChild(document.createElement('style')).innerHTML = ` .catbox-droptarget { margin-left: 8px; padding: 5px; background-color: #fff9cc; border: solid thin black; align-content: center; vertical-align: 1px; } .catbox-img { height: 25px; vertical-align: middle; } `; var userhash = GM_getValue('userhash'); var image; bindAll(); onReportTypeChange(); if (document.location.pathname.toLowerCase() != '/requests.php') { var rlsTypeSelect = document.querySelector('select#categories'); if (rlsTypeSelect != null) rlsTypeSelect.addEventListener('change', onRlsTypeChange); } var reportTypeSelect = document.querySelector('select#type'); if (reportTypeSelect != null) reportTypeSelect.addEventListener('change', onReportTypeChange); GM_setValue('userhash', userhash || ''); function onReportTypeChange(evt) { setTimeout(function() { if (evt instanceof Event) { image = document.querySelector('input#proofimages') || document.querySelector('input#image'); if (image != null) image.parentNode.append(createDropTarget(image)); } bindToTextarea('extra'); }, 1000); } function onRlsTypeChange(evt) { setTimeout(bindAll, 1000) } function imageDropHandler(evt) { evt.preventDefault(); uploadFiles(evt.currentTarget, evt.dataTransfer.files); } function clickHandler(evt) { evt.preventDefault(); if (!evt.currentTarget.boundElement) throw new Error('boundElement not set'); if (evt.currentTarget.boundElement.nodeName == 'INPUT' && /^https?:\/\//i.test(evt.currentTarget.boundElement.value) && !evt.currentTarget.boundElement.value.toLowerCase().includes('catbox.moe/')) { rehostUrl(evt.currentTarget, evt.currentTarget.boundElement.value); } else { let currentTarget = evt.currentTarget; let inputElement = document.createElement("input"); inputElement.type = "file"; inputElement.accept = '.jpg, .jpeg, .jfif, .png, .gif, .webp'; inputElement.multiple = true; inputElement.onchange = evt => { uploadFiles(currentTarget, inputElement.files) }; inputElement.dispatchEvent(new MouseEvent("click")); } } function voidDragHandler(evt) { evt.preventDefault() } function uploadFiles(evtSrc, files) { if (files.length <= 0) return; if (!evtSrc.boundElement) throw new Error('boundElement not set'); if (evtSrc.busy) throw new Error('Wait till current upload finishes'); evtSrc.busy = true; if (evtSrc.hTimer) { clearTimeout(evtSrc.hTimer); delete evtSrc.hTimer; } evtSrc.style.backgroundColor = 'red'; Promise.all(upload2Catbox(files)) .then(function(results) { if (results.length > 0) { switch (evtSrc.boundElement.nodeName) { case 'INPUT': evtSrc.boundElement.value = results[0]; break; case 'TEXTAREA': evtSrc.boundElement.value += results.join('\n'); break; } evtSrc.style.backgroundColor = '#00C000'; evtSrc.hTimer = setTimeout(function() { evtSrc.style.backgroundColor = null; delete evtSrc.hTimer; }, 3000); } else evtSrc.style.backgroundColor = null; }).catch(function(e) { alert(e); evtSrc.style.backgroundColor = null; }).then(function() { evtSrc.busy = false; }); }; function rehostUrl(evtSrc, url) { if (!/^https?:\/\//i.test(url)) return; if (!evtSrc.boundElement) throw new Error('boundElement not set'); if (evtSrc.busy) throw new Error('Wait till current upload finishes'); evtSrc.busy = true; if (evtSrc.hTimer) { clearTimeout(evtSrc.hTimer); delete evtSrc.hTimer; } evtSrc.style.backgroundColor = 'red'; rehost2Catbox(evtSrc.boundElement.value).then(function(result) { evtSrc.boundElement.value = result; evtSrc.style.backgroundColor = '#00C000'; evtSrc.hTimer = setTimeout(function() { delete evtSrc.hTimer; evtSrc.style.backgroundColor = null; }, 3000); }).catch(function(e) { alert(e); evtSrc.style.backgroundColor = null; }).then(function() { evtSrc.busy = false; }); } function bindAll() { if ((image = document.getElementById('image')) != null) { image.parentNode.append(createDropTarget(image)); } else if ((image = document.querySelector('input[name="image"]')) != null) { image.parentNode.insertBefore(createDropTarget(image), image.parentNode.querySelector(':scope > br')); } ['album_desc', 'release_desc', 'desc', 'body', 'description', 'screenshots'].forEach(bindToTextarea); } function bindToTextarea(id) { var desc = document.querySelector('textarea#' + id); if (desc != null) { var btn = desc.parentNode.parentNode.querySelector('div > input[class^="button_preview"]'); if (btn != null) { btn.parentNode.append(createDropTarget(desc)); } else if ((btn = desc.parentNode.parentNode.querySelector(':scope > td.label')) != null) { var div = document.createElement('div'); div.style.marginTop = '60px'; div.append(createDropTarget(desc)); btn.append(div); } else if ((btn = desc.parentNode.querySelector('div#Bbcode_Toolbar > div[style]:last-of-type')) != null) { btn.parentNode.insertBefore(createDropTarget(desc), btn); } return btn != null; } else if ((desc = document.querySelector('textarea[name="' + id + '"]')) != null && (btn = desc.parentNode.querySelector(':scope > div > input[value="Submit"]')) != null) { btn.parentNode.append(createDropTarget(desc)); return true; } return false; } function createDropTarget(boundElement) { if (!(boundElement instanceof HTMLElement)) throw new Error('invalid boundElement'); var dropTarget = document.createElement('span'); dropTarget.boundElement = boundElement; dropTarget.className = 'catbox-droptarget'; dropTarget.ondragover = voidDragHandler; dropTarget.ondrop = imageDropHandler; // upload dropTarget.onclick = clickHandler; // rehost var img = document.createElement('img'); img.src = ''; img.onerror = function() { this.src = 'https://catbox.moe/pictures/logo.png' }; img.className = 'catbox-img'; dropTarget.append(img); return dropTarget; } function upload2Catbox(files) { if (!(files instanceof FileList)) return Promise.reject('Bad parameter (files)'); return Array.from(files) .sort((file1, file2) => file1.name.localeCompare(file2.name)) .map(file => new Promise(function(resolve, reject) { var fr = new Promise(function(resolve) { var reader = new FileReader(); reader.onload = function() { resolve(reader.result) } reader.readAsBinaryString/*readAsArrayBuffer(file)*/(file); }); fr.then(function(result) { const boundary = '----WebKitFormBoundaryTID_GM'; var data = '--' + boundary + '\r\n'; data += 'Content-Disposition: form-data; name="reqtype"\r\n\r\n'; data += 'fileupload\r\n'; if (userhash) { data += '--' + boundary + '\r\n'; data += 'Content-Disposition: form-data; name="userhash"\r\n\r\n'; data += userhash + '\r\n'; } data += '--' + boundary + '\r\n'; data += 'Content-Disposition: form-data; name="fileToUpload"; filename="' + file.name.toASCII() + '"\r\n'; data += 'Content-Type: ' + file.type + '\r\n\r\n'; data += result + '\r\n'; data += '--' + boundary + '--\r\n'; GM_xmlhttpRequest({ method: 'POST', url: 'https://catbox.moe/user/api.php', responseType: 'text', headers: { 'Content-Type': 'multipart/form-data; boundary=' + boundary, 'Content-Length': data.length, }, data: data, binary: true, onload: function(response) { if (response.status != 200) reject('Response error ' + response.status + ' (' + response.statusText + ')'); resolve(response.response); }, onerror: response => { reject('Response error ' + response.status + ' (' + response.statusText + ')') }, ontimeout: function() { reject('Timeout') }, }); }); })); } function rehost2Catbox(url) { if (typeof url != 'string' || !url) return Promise.reject('Bad parameter (url)'); return new Promise(function(resolve, reject) { var data = new URLSearchParams({ reqtype: 'urlupload', url: url.trim(), }); if (userhash) data.set('userhash', userhash); GM_xmlhttpRequest({ method: 'POST', url: 'https://catbox.moe/user/api.php', responseType: 'text', headers: { 'Content-Type': 'application/x-www-form-urlencoded', 'Content-Length': data.toString().length, }, data: data.toString(), onload: function(response) { if (response.status != 200) reject('Response error ' + response.status + ' (' + response.statusText + ')'); resolve(response.response); }, onerror: response => { reject('Response error ' + response.status + ' (' + response.statusText + ')') }, ontimeout: function() { reject('Timeout') }, }); }); }