Skip to content

Instantly share code, notes, and snippets.

@headzoo
Last active August 26, 2023 03:35
Show Gist options
  • Save headzoo/85528e64de6642457cb1675cf3d3be4b to your computer and use it in GitHub Desktop.
Save headzoo/85528e64de6642457cb1675cf3d3be4b to your computer and use it in GitHub Desktop.
Tampermonkey script for DiscImg.
// ==UserScript==
// @name DiscImg Upload
// @namespace http://discimg.com/
// @version 0.1
// @description Upload images to DiscImg.
// @author contact@discimg.com
// @updateURL https://gist.githubusercontent.com/headzoo/85528e64de6642457cb1675cf3d3be4b/raw/
// @downloadURL https://gist.githubusercontent.com/headzoo/85528e64de6642457cb1675cf3d3be4b/raw/
// @match https://discuit.net/*
// @icon 
// @grant none
// @require https://raw.githubusercontent.com/soufianesakhi/node-creation-observer-js/master/release/node-creation-observer-latest.js
// ==/UserScript==
(async function () {
'use strict';
NodeCreationObserver.onCreation('.page-new-post-body', (body) => {
const existing = document.querySelector('.discuit-image-downloader');
if (existing) {
existing.remove();
}
const downloader = document.createElement('div');
downloader.innerHTML = `
<div class="discuit-image-downloader">
<div style="display: flex; align-items: center; margin-bottom: 1rem">
<img
src="https://discimg.com/_next/image?url=%2Flogo.png&w=32&q=75"
alt=""
style="width: 26px; height: 26px; border-radius: 2px; margin-right: 0.5rem"
/>
<h3 style="font-size: 18px">DiscImg</h3>
</div>
<button class="discuit-image-btn" style="margin-bottom: 0.25rem;">Select Image</button>
<div style="margin-bottom: 0.25rem;">OR</div>
<input id="discuit-image-downloader__url" placeholder="Paste URL" />
<div class="discuit-image-progress" style="display: none;">Uploading...</div>
<input type="file" id="discuit-image-downloader__input" accept="image/*" />
</div>
`;
body.after(downloader);
const btn = downloader.querySelector('.discuit-image-btn');
const progress = downloader.querySelector('.discuit-image-progress');
const inputFile = downloader.querySelector('#discuit-image-downloader__input');
const inputUrl = downloader.querySelector('#discuit-image-downloader__url');
let selectionStart = 0;
const sendFile = (formData) => {
btn.disabled = true;
inputUrl.disabled = true;
progress.style.display = 'block';
fetch('https://discimg.com/api/upload', {
method: 'POST',
body: formData,
})
.then((resp) => {
resp.json().then((data) => {
const img = data.urls[0];
const textarea = document.querySelector('.page-new-post-body');
const tab = document.querySelector('.pn-tabs-item.is-selected');
const selected = tab.querySelector('svg').getAttribute('class');
// Prevents discuit from undoing the textarea changes. It doesn't work
// to simply set textarea.value = 'some value' because frontend framework
// removes it when done that way.
// @see https://stackoverflow.com/a/75430383
function setNativeValue(element, value) {
const { set: valueSetter } = Object.getOwnPropertyDescriptor(element, 'value') || {};
const prototype = Object.getPrototypeOf(element);
const { set: prototypeValueSetter } =
Object.getOwnPropertyDescriptor(prototype, 'value') || {};
if (prototypeValueSetter && valueSetter !== prototypeValueSetter) {
prototypeValueSetter.call(element, value);
} else if (valueSetter) {
valueSetter.call(element, value);
} else {
throw new Error('The given element does not have a value setter');
}
}
if (selected === 'page-new-icon-link') {
setNativeValue(textarea, img);
} else {
if (selectionStart) {
setNativeValue(
textarea,
(textarea.value =
textarea.value.substring(0, selectionStart) +
img +
textarea.value.substring(selectionStart)),
);
} else {
setNativeValue(textarea, img + textarea.value);
}
}
textarea.dispatchEvent(new Event('input', { bubbles: true }));
selectionStart = 0;
});
})
.finally(() => {
btn.disabled = false;
inputUrl.disabled = false;
inputUrl.value = '';
progress.style.display = 'none';
});
};
btn.addEventListener('click', () => {
const textarea = document.querySelector('.page-new-post-body');
selectionStart = textarea.selectionStart;
downloader.querySelector('#discuit-image-downloader__input').click();
});
inputUrl.addEventListener('input', (e) => {
try {
new URL(e.target.value);
} catch {
return;
}
const formData = new FormData();
formData.append('url', e.target.value);
sendFile(formData);
});
inputFile.addEventListener('change', (e) => {
const files = e.target.files;
if (files.length > 0) {
const formData = new FormData();
formData.append('files', files[0]);
sendFile(formData);
}
});
});
function GM_addStyle(css) {
const style =
document.getElementById('GM_addStyleBy8626') ||
(function () {
const style = document.createElement('style');
style.type = 'text/css';
style.id = 'GM_addStyleBy8626';
document.head.appendChild(style);
return style;
})();
style.textContent = css;
}
GM_addStyle(`
.discuit-image-downloader {
border: 2px dashed #ccc;
border-radius: 6px;
transition: border-color 0.2s;
padding: 20px;
margin: 1rem;
display: flex;
justify-content: center;
align-items: center;
flex-direction: column;
}
.discuit-image-downloader:hover {
border-color: #e0e0e0;
}
.discuit-image-downloader:focus {
outline: 0;
box-shadow: 0 0 0 5px rgba(255, 49, 186, 0.5);
}
.discuit-image-btn {
background-color: #0297db;
padding: 0.5rem 1rem;
border-radius: 6px;
transition: all 0.2s;
}
.discuit-image-btn:hover {
background-color: #12a8eb;
}
.discuit-image-btn:focus {
outline: 0;
box-shadow: 0 0 0 5px rgba(3, 174, 251, 0.51);
}
#discuit-image-downloader__input {
display: none;
}
`);
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment