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