// ==UserScript==
// @name Tellonym Answer Revealer
// @namespace https://spin.rip/
// @match https://tellonym.me/*
// @grant none
// @version 1.0
// @author Spinfal
// @description Reveals the answers on a Tellonym user's page without logging in.
// @license AGPL-3.0 License
// ==/UserScript==
(function() {
'use strict';
// ✅ create status window
const statusDiv = document.createElement("div");
statusDiv.style.display = "block";
statusDiv.style.position = "fixed";
statusDiv.style.bottom = "10px";
statusDiv.style.right = "10px";
statusDiv.style.backgroundColor = "#1c1c1e";
statusDiv.style.color = "#fff";
statusDiv.style.padding = "10px";
statusDiv.style.borderRadius = "10px";
statusDiv.style.fontSize = "12px";
statusDiv.style.fontFamily = "monospace";
statusDiv.style.zIndex = "9999";
statusDiv.innerText = "🔎 waiting for profile data...";
document.body.appendChild(statusDiv);
let mainDiv;
const targetSelector = "#root > div > div > div.css-1dbjc4n.r-150rngu.r-eqz5dr.r-16y2uox.r-1wbh5a2.r-11yh6sk.r-1rnoaur > div > div";
// 👀 setup mutation observer to detect when mainDiv becomes available
const observer = new MutationObserver(() => {
const foundDiv = document.querySelector(targetSelector);
if (foundDiv) {
mainDiv = foundDiv;
console.log("✅ mainDiv found");
observer.disconnect();
}
});
observer.observe(document.body, { childList: true, subtree: true });
const originalFetch = window.fetch;
let capturedUserId = null;
window.fetch = async function(input, init = {}) {
const url = typeof input === "string" ? input : input.url;
const method = init?.method?.toUpperCase() || "GET";
const profileRegex = /^https:\/\/api\.tellonym\.me\/profiles\/name\/[^/?]+(\?limit=\d+)?$/;
const followingsRegex = /^https:\/\/api\.tellonym\.me\/followings\/id\/(\d+)\?userId=\1(&limit=\d+)?$/;
if (profileRegex.test(url) && method === "GET" && !location.href.includes("/followings")) {
updateStatus("📡 intercepted profile fetch");
const response = await originalFetch.call(this, input, init);
try {
const cloned = response.clone();
const json = await cloned.json();
if (json?.id) {
capturedUserId = json.id;
updateStatus(`🧐 captured userId: ${capturedUserId}`);
}
if (json && typeof json === "object") {
json.isAbleToComment = true;
json.followingCount === 0 ? json.followingCount = 1 : json.followingCount++;
localStorage.setItem("reduxPersist:user", JSON.stringify({ hasLoggedIn: true }));
updateStatus("✅ modified values");
}
if (json.answers) {
updateStatus("⌛ waiting for tellonym to finish DOM changes...");
setTimeout(() => {
const freshMainDiv = document.querySelector(targetSelector);
if (freshMainDiv) {
const cNodes = Array.from(freshMainDiv.childNodes).slice(1);
cNodes.forEach(i => i.remove());
json.answers.forEach(ans => {
const newTellDiv = document.createElement("div");
newTellDiv.classList.add("css-1dbjc4n");
newTellDiv.innerHTML = `
<div data-radium="true" style="display: flex; flex-direction: column;">
<div class="rmq-3ca56ea3" data-radium="true" style="padding: 16px; display: flex; flex-direction: column; margin-bottom: 8px; background-color: rgb(255, 255, 255);">
<div data-radium="true" style="display: flex; flex-direction: row; justify-content: space-between;">
<div data-radium="true" style="display: flex; flex-direction: row; align-items: center;">
<div data-radium="true" style="cursor: pointer; color: rgb(20, 23, 26); display: flex; flex-direction: column; border-radius: 8px; overflow: hidden; background-color: rgb(255, 255, 255);">
<div data-radium="true" style="display: flex; flex-direction: column; position: relative;">
<img alt="" src="https://userimg.tellonym.me/xs-v2/${json.avatarFileName}" resizemode="stretch" data-radium="true" style="justify-content: center; align-items: center; height: 38px; width: 29px;">
</div>
</div>
<div data-radium="true" style="display: flex; flex-direction: column; margin-left: 12px;">
<div data-radium="true" style="cursor: pointer; color: rgb(20, 23, 26); display: flex; flex-direction: row; max-width: 200px; margin-bottom: 2px;">
<div italic="false" data-radium="true" style="color: rgb(20, 23, 26); font-size: 14px; font-weight: bold; overflow-wrap: break-word; white-space: pre-wrap; word-break: break-word;">${json.username}</div>
</div>
<div data-radium="true" style="color: rgb(175, 175, 179); font-size: 12px; overflow-wrap: break-word; white-space: pre-wrap; word-break: break-word;">${timeAgo(ans.createdAt)}</div>
</div>
</div>
</div>
<div data-radium="true" style="display: ${ans.type === 1 ? "none;" : "flex; flex-direction: column; margin-top: 12px;"}">
<div data-radium="true" style="display: flex; flex-direction: row; border-left: 3px solid rgb(136, 137, 143); padding-left: 12px;">
<div data-radium="true" style="display: flex; flex-direction: column;">
<div data-radium="true" style="color: rgb(20, 23, 26); font-size: 16px; white-space: pre-wrap; word-break: break-word;">${ans.tell}</div>
</div>
</div>
</div>
<div data-radium="true" style="margin-top: 12px;">
<div data-radium="true" style="font-size: 16px; color: rgb(20, 23, 26); white-space: pre-wrap; word-break: break-word;">${ans.answer}</div>
</div>
<div data-radium="true" style="margin-top: 6px; font-size: 12px; color: #5264af; opacity: 0.5;">
<p>${ans.type === 1 ? "no tell available for this one" : 'answers revealed by <a href="https://spin.rip/" target="_blank">spin.rip</a>'}</p>
</div>
</div>
</div>
`.trim();
freshMainDiv.appendChild(newTellDiv);
});
updateStatus("✅ tells inserted!");
} else {
updateStatus("⚠️ mainDiv not found");
}
}, 3000);
}
return new Response(JSON.stringify(json), {
status: response.status,
statusText: response.statusText,
headers: response.headers
});
} catch (err) {
console.warn("❌ error parsing profile JSON:", err);
updateStatus("❌ error while handling profile data");
return response;
}
}
if (followingsRegex.test(url) && method === "GET") {
const response = await originalFetch.call(this, input, init);
try {
const cloned = response.clone();
const json = await cloned.json();
if (json && Array.isArray(json.followings)) {
const spinEntry = {
type: 0,
id: 113589800,
avatarFileName: "113589800_cukoc7ico5u4ro3gveqzwf8cgbdbagei.jpg",
displayName: "spin.rip",
username: "spin.rip",
isFollowingAnonymous: false,
aboutMe: "revealed by spin // i see all",
isVerified: true,
isBlocked: false,
isBlockedBy: false,
isFollowed: true,
isFollowedBy: true,
isActive: true
};
json.followings.unshift(spinEntry);
console.log("🧪 injected spin.rip into followings");
}
return new Response(JSON.stringify(json), {
status: response.status,
statusText: response.statusText,
headers: response.headers
});
} catch (err) {
console.warn("❌ error modifying followings JSON:", err);
return response;
}
}
return originalFetch.call(this, input, init);
};
function updateStatus(msg) {
if (msg === "CLEAR") statusDiv.style.display = "none";
statusDiv.innerText = msg;
}
function timeAgo(isoString) {
const now = new Date();
const then = new Date(isoString);
const seconds = Math.floor((now - then) / 1000);
if (seconds < 60) return "just now";
if (seconds < 3600) return `${Math.floor(seconds / 60)} minute(s) ago`;
if (seconds < 86400) return `${Math.floor(seconds / 3600)} hour(s) ago`;
if (seconds < 604800) return `${Math.floor(seconds / 86400)} day(s) ago`;
return then.toLocaleDateString();
}
})();