Skip to content

Instantly share code, notes, and snippets.

@mikhailsdv
Last active November 26, 2022 18:14
Show Gist options
  • Save mikhailsdv/7801a83cc8aac43d522fc2ca179d7836 to your computer and use it in GitHub Desktop.
Save mikhailsdv/7801a83cc8aac43d522fc2ca179d7836 to your computer and use it in GitHub Desktop.
Send any image to your Black Hole by pressing Ctrl + Q
// ==UserScript==
// @name Black Hole image downloader
// @version 1.0
// @description Send any image to your Black Hole by pressing Ctrl + Q
// @author https://github.com/mikhailsdv
// @license MIT
// @match https://*/*
// @match http://*/*
// @icon https://iili.io/HFm563g.png
// @grant GM_getValue
// @grant GM_setValue
// ==/UserScript==
;(function () {
"use strict"
const downloadImage =
""
const loadingImage =
""
const successImage =
""
let count = 0
const isValidIntegrationUrl = url => /^https?:\/\/.+?\/integration\/[A-Za-z0-9]+$/.test(url)
const removeButtons = () => {
Array.from(document.getElementsByClassName("bh-button")).forEach(element =>
element.remove()
)
document.getElementById("bh-change-url").remove()
count = 0
}
const saveFile = async (url, button) => {
button.style.backgroundImage = `url("${loadingImage}")`
const integrationURL =
GM_getValue("integration-url") ||
prompt("Set your Integration URL. This is one-time action.")
if (!integrationURL || !isValidIntegrationUrl(integrationURL)) {
return alert("Wrong Integration URL!")
}
GM_setValue("integration-url", integrationURL)
fetch(integrationURL, {
method: "POST",
body: JSON.stringify({url}),
headers: {"Content-Type": "application/json"},
})
.then(response => response.json())
.then(data => {
const {status, error, url} = data
if (error) {
button.style.backgroundImage = `url("${downloadImage}")`
return alert("Error: " + error)
}
button.style.backgroundImage = `url("${successImage}")`
setTimeout(() => {
removeButtons()
}, 2000)
})
}
const listenCombination = ({combination, callback, once, element = window}) => {
let combinationMemory = []
let emptyTimeout
const onKeyUp = e => {
const code = e.metaKey ? "MetaKey" : e.code
combinationMemory = combinationMemory.filter(item => item !== code)
emptyTimeout = setTimeout(() => {
combinationMemory = []
}, 500)
}
const onKeyDown = e => {
const code = e.metaKey ? "MetaKey" : e.code
clearTimeout(emptyTimeout)
if (!combinationMemory.includes(code)) {
combinationMemory.push(code)
}
if (combination.every((item, index) => item === combinationMemory[index])) {
if (once) {
element.removeEventListener("keydown", onKeyDown)
element.removeEventListener("keyup", onKeyUp)
}
callback()
}
}
element.addEventListener("keyup", onKeyUp)
element.addEventListener("keydown", onKeyDown)
}
const appendButton = ({width, height, top, left, element, url, fileName}) => {
const button = document.createElement("button")
const size = 20
let styles
if (!document.getElementById("#bh-button-styles")) {
styles = document.createElement("style")
styles.id = "bh-button-styles"
styles.innerHTML = `
.bh-button {
z-index: 99999;
position: absolute;
border: none;
outline: none;
padding: 0;
margin: 0;
cursor: pointer;
background-color: white;
background-size: 100%;
background-position: center;
background-repeat: no-repeat;
background-image: url("${downloadImage}");
border-radius: 100%;
height: ${size}px;
min-width: ${size}px;
min-height: ${size}px;
width: ${size}px;
box-shadow: 0 0 0 rgba(0, 0, 0, 0.3);
transition: all .2s ease-out;
}
.bh-button:hover {
transform: scale(1.1);
box-shadow: 0 0 10px rgba(0, 0, 0, 0.3);
}
.bh-button:active {
transform: scale(1.05);
box-shadow: 0 0 6px rgba(0, 0, 0, 0.3);
}
#bh-change-url {
position: fixed;
bottom: 14px;
left: 50%;
border: none;
transform: translateX(-50%);
z-index: 99999;
height: 54px;
padding: 14px 22px;
font-size: 16px;
font-weight: 500;
color: white;
cursor: pointer;
font-family: "Roboto", "Arial", sans-serif;
overflow: hidden;
user-select: none;
white-space: nowrap;
background-color: #ef39a8;
box-shadow: 0 2px 10px -2px #0000008c;
border-radius: 8px;
outline: none;
transition: background-color .2s ease-out;
}
#bh-change-url:hover {
background-color: #bd399c;
}
`
document.head.appendChild(styles)
}
button.style.top = top + height / 2 - size / 2 + "px"
button.style.left = left + width / 2 - size / 2 + "px"
button.classList.add("bh-button")
button.addEventListener("click", e => {
e.preventDefault()
e.stopImmediatePropagation()
e.stopPropagation()
saveFile(url, button)
})
element.prepend(button)
}
const getFileName = url => {
if (/^data:image\/([a-zA-Z0-9]+)/.test(url)) {
const extension = url.match(/^data:image\/([a-zA-Z0-9]+)/)
return "image." + extension[1]
} else {
const match = url.match(/^.*\/(.+?)(\?.+?)?$/)
if (match && match[1]) {
return match[1]
} else {
return "image.jpg"
}
}
}
const appendChangeUrlButton = () => {
const button = document.createElement("button")
button.id = "bh-change-url"
button.innerHTML = "Change Integration URL"
button.addEventListener("click", e => {
e.preventDefault()
e.stopImmediatePropagation()
e.stopPropagation()
const integrationURL = prompt("Set your Integration URL. This is one-time action.")
if (isValidIntegrationUrl(integrationURL)) {
GM_setValue("integration-url", integrationURL)
alert("Integration URL has been changed!")
} else {
alert("Wrong Integration URL!")
}
})
document.body.appendChild(button)
}
const callback = () => {
count++
if (count % 2 === 0) {
return removeButtons()
}
appendChangeUrlButton()
Array.from(document.querySelectorAll("*:not(img)")).forEach(element => {
if (element.offsetParent) {
const computedStyle = getComputedStyle(element)
const match = computedStyle.backgroundImage.match(/^url\(['"](.+?)['"]\)/)
if (match && match[1]) {
appendButton({
top: element.offsetTop,
left: element.offsetLeft,
width: element.offsetWidth,
height: element.offsetHeight,
element: element.offsetParent,
url: match[1],
fileName: getFileName(match[1]),
})
}
}
})
Array.from(document.getElementsByTagName("img")).forEach(element => {
if (element.offsetParent) {
element.setAttribute("crossorigin", "anonymous")
appendButton({
top: element.offsetTop,
left: element.offsetLeft,
width: element.offsetWidth,
height: element.offsetHeight,
element: element.offsetParent,
url: element.src,
fileName: getFileName(element.src),
})
}
})
}
listenCombination({
combination: ["MetaKey", "KeyQ"],
callback,
once: false,
})
listenCombination({
combination: ["ControlLeft", "KeyQ"],
callback,
once: false,
})
listenCombination({
combination: ["ControlRight", "KeyQ"],
callback,
once: false,
})
})()
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment