- // ==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') },
- });
- });
- }