Skip to content

Instantly share code, notes, and snippets.

@busybox11
Last active June 14, 2024 22:19
Show Gist options
  • Save busybox11/de6ffc8f56a4d3281b7904642d26d6aa to your computer and use it in GitHub Desktop.
Save busybox11/de6ffc8f56a4d3281b7904642d26d6aa to your computer and use it in GitHub Desktop.
Adds a "Yeah!" button to every tweet that replies with Miiverse's "Yeah!" button picture
// ==UserScript==
// @name Miiverse Yeah Button for Twitter
// @namespace twitter scripts
// @match *://*.twitter.com/*
// @match *://*.x.com/*
// @grant none
// @version 1.0.3
// @author @chaoticvibing (busybox11 on GitHub)
// @description 6/14/2024
// @updateURL https://gist.githubusercontent.com/busybox11/de6ffc8f56a4d3281b7904642d26d6aa/raw/miiverse-yeah-btn-twitter.user.js
// @downloadURL https://gist.githubusercontent.com/busybox11/de6ffc8f56a4d3281b7904642d26d6aa/raw/miiverse-yeah-btn-twitter.user.js
// ==/UserScript==
// This script is free to use by anyone. You can modify it however you want,
// but I'd appreciate it if you could leave a mention to this script somewhere :)
//
// I don't like doing this, but I'd like to share donation links.
// I'm going through tough times, any help is appreciated - but absolutely not required!
// The following links directly help me:
// https://paypal.me/busybox11
// https://ko-fi.com/busybox11 (AKA https://moneyslave.penis.community/)
// https://github.com/sponsors/busybox11
const YEAH_B64 =
"";
const YEAH_BIG_B64 =
"";
let waitForPopup = false;
let waitForImage = false;
const setInput = async (el) => {
const dataTransfer = new DataTransfer();
const base64Data = YEAH_BIG_B64.split(",")[1];
// Convert the Base64 string to a Blob
const byteCharacters = atob(base64Data);
const byteNumbers = new Array(byteCharacters.length);
for (let i = 0; i < byteCharacters.length; i++) {
byteNumbers[i] = byteCharacters.charCodeAt(i);
}
const byteArray = new Uint8Array(byteNumbers);
const blob = new Blob([byteArray], { type: "image/jpeg" });
// Set generated image blob as the data for the clipboard
dataTransfer.setData("image/jpeg", blob);
// Add the file to the clipboard just in case it doesn't get recognized with the blob
const file = new File([blob], "image.jpg", { type: "image/jpeg" });
dataTransfer.items.add(file);
el.dispatchEvent(
new ClipboardEvent("paste", {
clipboardData: dataTransfer,
bubbles: true,
cancelable: true,
})
);
// Clear DataTransfer Data
dataTransfer.clearData();
};
const findClosestInput = (el) => {
const contentEditableEl = el.querySelector(
"div[data-testid^='tweetTextarea_'][contenteditable='true']"
);
if (contentEditableEl) {
return contentEditableEl;
}
if (!el.parentElement) {
return null;
} else {
return findClosestInput(el.parentElement);
}
};
function handleTweetFillInput(targetEl) {
const inputEl = findClosestInput(targetEl.parentElement.parentElement);
if (inputEl) {
setInput(inputEl);
}
waitForImage = true;
window.setTimeout(() => {
waitForImage = false;
}, 2000);
}
function handleTweetSendBtn() {
let sendTweetBtn = document.querySelector("[data-testid='tweetButton']");
if (!sendTweetBtn) {
sendTweetBtn = document.querySelector("[data-testid='tweetButtonInline']");
}
if (sendTweetBtn) {
sendTweetBtn.click();
}
}
/**
*
* @param {MutationRecord} mutationsList
* @param {MutationObserver} observer
*/
const handleMutation = (mutationsList, observer) => {
for (const mutation of mutationsList) {
if (mutation.type === "childList") {
/** @type {NodeList} */
const nodes = mutation.addedNodes;
for (const node of nodes) {
if (node && typeof node.querySelector === "function") {
let els = node.querySelectorAll(
"[role='group']:not(:has(.yeah-btn))"
);
if (els.length === 0) {
const contElem = node.querySelector(
"[data-testid='toolBar']:not(:has(.yeah-btn))"
);
if (contElem) {
els = contElem.querySelectorAll(
"[role='tablist']:not(:has(.yeah-btn))"
);
if (waitForPopup) {
// Wait for popup is handled here
waitForPopup = false;
handleTweetFillInput(contElem);
}
}
}
for (const el of els) {
if (el && el.childElementCount > 3) {
const elHeight = el.offsetHeight;
const isSmall = elHeight <= 24;
// Create "Yeah" button
const yeahBtn = document.createElement("button");
yeahBtn.classList.add("yeah-btn");
if (isSmall) {
yeahBtn.classList.add("small");
}
yeahBtn.innerHTML = `<img src="${YEAH_B64}" alt="Yeah!">`;
yeahBtn.addEventListener("click", () => {
if (isSmall) {
// TODO: Automatically like as well if not already liked
// Click on first nested button item of the tablist
// to bring up the reply box
el.firstElementChild.querySelector("button").click();
waitForPopup = true;
setTimeout(() => {
waitForPopup = false;
}, 2000);
} else {
handleTweetFillInput(el);
}
});
el.appendChild(yeahBtn);
}
}
// If image has been added
if (waitForImage) {
if (node.tagName === "IMG" && node.src.includes("blob:")) {
handleTweetSendBtn();
}
}
}
}
}
}
};
const sc = {
log: (...args) => {
console.log("[miiverse-yeah]", ...args);
},
};
const observer = new MutationObserver(handleMutation);
observer.observe(document, {
childList: true,
subtree: true,
attributeOldValue: true,
});
// Inject CSS
const style = document.createElement("style");
style.innerHTML = `
.yeah-btn {
display: flex;
align-items: center;
justify-content: center;
width: 36px;
min-height: 20px;
height: 36px;
margin: auto 0 auto 4px;
border-radius: 50%;
background-color: rgba(255, 255, 255, 0);
border: none !important;
cursor: pointer;
transition: background-color 0.4s;
}
.yeah-btn:hover {
background-color: rgba(255, 255, 255, 0.1);
}
.yeah-btn img {
width: 20px;
height: 20px;
}
.yeah-btn.small {
margin: -8px -8px -8px 4px;
}
.yeah-btn.small img {
width: 18px;
height: 18px;
}
`;
document.head.appendChild(style);
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment