// ==UserScript== // @name Wormlang RuneReader // @version 1.5 // @description automagically decode wormlang runes // @match *://*.libpol.org/* // @run-at document-idle // @grant none // @downloadURL https://raw.githubusercontent.com/Xx-hackerman-xX/brainworm-runereader/refs/heads/main/brainworm-runereader.js // @icon  // @author github.com/Xx-hackerman-xX // ==/UserScript== /* excellent idea gracelessly stolen from rune-reader-mobile-temp by isabelle & sam & The Worm. icon is bury smoking a fatty blunt. */ /* 1.5 - add support for way more rune types and their krazy kombinations - fix some runes not displaying their proper colours - removed herobrine 1.4.1 - new feature: click magnifying glass on a post to reveal its true contents - fix broken runes after update - code cleanup */ /* UTIL & CONSTANTS */ /* return context-appropriate window.require(path) */ function windowRequire(path) { try { window.require(path) } catch { // we're on a page that requires client/ to be prepended for whatever reason path = "client/" + path } return window.require(path) } /* pretty announce script details in console */ function announce() { console.log(`%c[[ ${GM_info.script.name} v${GM_info.script.version} loaded ]]`, "color:lime; background:black;") } /* async sleep */ function sleep(ms) { return new Promise(res => setTimeout(res, ms)) } var RUNEREADER_STATE = true // start on var vanilla_socket_onmessage // replaced later const RUNEREADER_CSS_ELM_ID = "runereader-css" // id of style elm with our fancy rules const TOGGLE_BUTTON_ID = "toggle-runereader" const RUNEREADER_ON = "runereader ON" const RUNEREADER_OFF = "runereader OFF" const ROOT_RUNE = ";" // the alpha and the omega, the root of all worm magicks // magnifying glass svg const REVEAL_SVG = `` // websocket message flags. stolen from vanilla message handler definition const MESSAGE = { insertPost: 1, // new post concat: 33, } /* CSS RUNE FACTORY */ const baseRunes = { targeted: '>', // act on quoted post masked: ',', // runespeak to quoted poster player: '.', // wormwatch image: ';', // image react: '+', // reactions value: '?', // idk... body: '*', // idk... // pseudoCommand: '#', // doesn't actually gain its own class when used, so we can't detect it. ignoring for now } const modifierRunes = { hidden: '|', // hide runes entirely public: '!', // reveal rune text // assuming neither value nor body are modifiers... } // definitions for standard runes that do normal things // rune: the specific runes needed to cast, displays in box, always visible // color: colour of the text and left border // classList: list of classes that identify this rune. all are combined in the querySelectors (e.g. 'class1.class2.class3') // description: human-readable description of the rune that appears on mouse hover const predefinedSwatches = [ // this one gotta be first so it can be overridden by more specific querySelectors // runes that are only public should always be revealed selfposts (;!). except when they're not. idk sometimes it goes weird { rune: ";!", color: "white", classList: ["public"], description: "public self post", }, // might not need this one, idk if it can ever be seen { rune: ";|", color: "white", classList: ["hidden"], description: "hidden self post", }, { rune: ";,", color: "magenta", classList: ["masked"], description: "runespeak", }, { rune: ";.", color: "pink", classList: ["player"], description: "wormwatch", }, { rune: ";>", color: "cyan", classList: ["targeted"], description: "target post", }, { rune: ";;", color: "white", classList: ["image"], description: "self image", }, { rune: ";;>", color: "lime", classList: ["targeted", "image"], description: "target image", }, { rune: ";+", color: "red", classList: ["react"], description: "self react", }, { rune: ";>+", color: "red", classList: ["targeted", "react"], description: "target react", }, { rune: ";?", color: "grey", classList: ["value"], description: "value", }, { rune: ";*", color: "grey", classList: ["body"], description: "body", } ] /* return all possible combinations of items (ignoring order) from array. stolen from https://stackoverflow.com/a/59942031 */ function getCombinations(valuesArray) { let combos = [] let temp var totalCombos = Math.pow(2, valuesArray.length) for (let i = 0; i < totalCombos; i++) { temp = [] for (let j = 0; j < valuesArray.length; j++) { if ((i & Math.pow(2, j))) { temp.push(valuesArray[j]) } } if (temp.length > 0) { combos.push(temp) } } combos.sort((a, b) => a.length - b.length) return combos } /* create css for a rune swatch */ function cssFactory(swatch) { const allClasses = swatch.classList.join('.') // default css let lines = [ `.def.${allClasses} {color: ${swatch.color} !important; border-color: ${swatch.color};}`, // set colours `.def:not(.revealed).${allClasses} {color: ${swatch.color} !important}`, // override vanilla colours in some cases `.def.${allClasses}:before {content: "${swatch.rune} "}`, // typed runes `.def.${allClasses}:hover:before {content: "[${swatch.description}] "}`, // description on hover ] // crazy combo styles // these currently also need to be set within the modifiers... maybe some day this will be nicer if (swatch.crazy) { lines.push( `.def.${allClasses} {border-style: double}`, // set doublethick border ) } // add every combination of modifiers getCombinations(Object.keys(modifierRunes)) .forEach((modifiers) => { // only add these modifiers if they are mututally exclusive from this current swatch if (modifiers.some( item => swatch.classList.includes(item) )) { return } let rune = swatch.rune modifiers.forEach((m) => { rune = rune.slice(0,1) + modifierRunes[m] + rune.slice(1) // slip the modifier just after the leading semicolon }) const description = modifiers.join(" ") + " " + swatch.description const modifierClasses = modifiers.join(".") lines.push( `.def.${modifierClasses}.${allClasses}:before {content: "${rune} "}`, `.def.${modifierClasses}.${allClasses}:hover:before {content: "[${description}] "}` ) if (swatch.crazy) { lines.push( `.def.${modifierClasses}.${allClasses} {border-style: double}`, // set doublethick border ) } }) return lines.join('\n') } /* return css for predefined runes */ function getNormalRuneCSS() { let output = "" for (let swatch of predefinedSwatches) { output += cssFactory(swatch) + "\n" } return output } /* return css for all nonsensical combinations of runes */ function getCrazyRuneCSS() { // normal pre-defined class combos let predefinedCombos = [] for (let swatch of predefinedSwatches) { predefinedCombos.push(swatch.classList.toSorted()) } const predefinedCombosStrings = predefinedCombos.map( (combo) => combo.toString() ) // for filtering, cause js thinks [1,2] != [1,2] if they're different objects... // all possible combinations of classes (mostly nonsense) let allCombos = getCombinations(Object.keys(baseRunes).toSorted()) for (let i in allCombos) { allCombos[i].sort() } // filter just the crazy stuff. anything not already defined is considered crazy (#society) let crazyCombos = [] allCombos.forEach((combo) => { if (!predefinedCombosStrings.includes(combo.toString())) { crazyCombos.push(combo) } }) // spit out the css for every crazy combo let output = "" for (const combo of crazyCombos) { let swatch = { classList: [], rune: "", description: "", color: "orange", crazy: true, } swatch.classList = combo let thisRune = ROOT_RUNE swatch.classList.forEach((className) => { thisRune += baseRunes[className] }) // add each rune swatch.rune = thisRune swatch.description = swatch.classList.join(' ') swatch.description = swatch.description.replace("player", "wormwatch").replace("masked", "runespeak") // human-readable description :) swatch.description += " " + "?".repeat(swatch.classList.length) // ????? output += cssFactory(swatch) + "\n" } return output } const LOVELY_CSS = ` /* base for all runetext */ .hide-live-body, .def.masked, .def.targeted, .def.player, .def.public, .def.react { display: block !important; width: 100%; padding: 10px; font-size: 10pt !important; letter-spacing: normal !important; font-family: monospace !important; text-shadow: none !important; font-weight: normal !important; background: black !important; color: white; border-left: 2px solid white; margin-top: 5px; } .def.masked:before, .def.targeted:before, .def.player:before { content: "[runereader broke... if you see this, pls tell me!!! :) thx]" /* you should never see this!! */ } /* predefined runes */ ${getNormalRuneCSS()} /* crazy runes */ ${getCrazyRuneCSS()} /* imshyuwu */ .hide-live-body { visibility: visible; color: teal; border-color: teal } /* "reveal true contents" button */ .reveal-button { display: initial; margin-left: 0.15em; color: var(--text-color); align-self: flex-start; } article:hover .reveal-button { opacity: 1; } .reveal-button svg { width: 1em; height: 1em; } .reveal-button:hover svg { transition: 0.1s; color: pink !important; transform: rotate(25deg); } /* tooltip for reveal button */ .reveal-button:before { position: absolute; left: 50%; transform: translateX(-50%); bottom: 100%; margin-bottom: 4px; width: auto; padding: 3px 6px; border-radius: 9px; background: black; color: white; text-align: center; opacity: 0; transition: 0.2s; pointer-events: none; } .reveal-button:hover:before { opacity: 1; } .reveal-button:before { content: "reveal true contents..."; } ` /* RUNEREADER */ const FUNCTION_ENCIPHER = windowRequire("util/cipher").convertToRandString // normal cypher function const FUNCTION_DECIPHER = function(e,t) { return e.replace(//g, ">") } // return uncophered text >:) /* show runes as original madoka runes */ function encipherRunes() { windowRequire("util/cipher").convertToRandString = FUNCTION_ENCIPHER } /* show all runes as plaintext */ function decipherRunes() { windowRequire("util/cipher").convertToRandString = FUNCTION_DECIPHER } /* toggle runereader on and off */ function toggleRunereader() { RUNEREADER_STATE = !RUNEREADER_STATE // invert let link = document.getElementById(TOGGLE_BUTTON_ID).firstChild link.innerText = RUNEREADER_STATE ? RUNEREADER_ON : RUNEREADER_OFF // set button text link.style = RUNEREADER_STATE ? "color:green;" : "" if (RUNEREADER_STATE) { addCSS() decipherRunes() } else { removeCSS() // encipherRunes() // just causes issues, disabling for the moment } } /* SHIMMING */ /* replace vanilla socket.onmessage with our own */ function replaceSocketOnmessage() { vanilla_socket_onmessage = windowRequire("connection/state").socket.onmessage // save windowRequire("connection/state").socket.onmessage = modified_socket_onmessage // shim console.debug("[reveal] socket.onmessage replaced :)") // hooray } /* keep trying the shim until it succeeds */ async function doTheShim(delay=100) { // can often fail cause window.require hasn't loaded or the method itself isn't defined in vanilla js yet let totalTime = 0 while (true) { try { replaceSocketOnmessage() break // succeeded, bail } catch { await sleep(delay) // failed, wait a mo and try again totalTime += delay } } console.debug("shim complete, took " + totalTime + "ms") } /* our modified socket.onmessage function. grabs the messages we care about and passes everything back to vanilla js as it was received */ function modified_socket_onmessage({data: e}, concatMessage=false) { let flag = e[0].codePointAt() // convert to int cause comparing unprintable characters is weird let payload = e.slice(1) // separate concatenated payloads and execute them separately if (flag === MESSAGE.concat) { payload = JSON.parse(payload) for (const msg of payload) { modified_socket_onmessage({data: msg}, concatMessage=true) // unroll and process each msg on our side } vanilla_socket_onmessage({data: e}) // send the original unrolled concat straight to vanilla js return // unconcatenated message } else { vanilla_socket_onmessage({data: e}) } // insert new post if (flag === MESSAGE.insertPost) { payload = JSON.parse(payload) addRevealButtonToPostID(payload.id) // go go gadget insert reveal button } } /* HTML - RUNEREADER */ /* add our pretty css rules to doc header */ function addCSS() { let style = document.createElement("style") style.id = RUNEREADER_CSS_ELM_ID style.innerText = LOVELY_CSS document.head.append(style) } /* remove our pretty css rules from doc header */ function removeCSS() { document.getElementById(RUNEREADER_CSS_ELM_ID).remove() } /* add footer button to toggle runes */ function addRunereaderToggleFooterButton() { let button = document.createElement("span") button.classList.add("act") button.id = TOGGLE_BUTTON_ID let link = document.createElement("a") link.innerText = RUNEREADER_STATE ? RUNEREADER_ON : RUNEREADER_OFF link.style = RUNEREADER_STATE ? "color:green;" : "" link.title = "click to toggle runereading" button.appendChild(link) let footer = document.getElementsByClassName("bottom-margin")[0] footer.appendChild(button) button.addEventListener('click', toggleRunereader) } /* HTML - REVEAL */ /* poop out a fresh reveal button that we can slap onto posts */ function createRevealButton() { let link = document.createElement("a") link.classList.add("svg-link", "noscript-hide", "concealed", "reveal-button") link.addEventListener( "click", (clickEvent) => { let post = clickEvent.target.closest("article") let allPosts = windowRequire("state").posts.models let trueBody = allPosts[post.id.slice(1)].body // slice the leading "p" from post id to lookup in model by post number alone alert(trueBody) } ) link.innerHTML = REVEAL_SVG return link } /* add reveal button to existing posts already on the page */ async function addRevealButtonToPosts() { while (!Object.keys(windowRequire("state").posts.models).length) { // wait here until posts model is populated await sleep(100) // probably a better way to do this..... } let allPosts = windowRequire("state").posts.models for (let postID in allPosts) { let header = document.getElementById(`p${postID}`).getElementsByTagName("header")[0] let revealButton = createRevealButton() header.append(revealButton) } } /* add a reveal button into the header of given postID */ function addRevealButtonToPostID(postID) { let postHeader = document.querySelector(`#p${postID} header`) // post doesn't exist... if (!postHeader) { console.error("tried to add reveal button to nonexisted post, id: " + postID) return } // button exists already if (postHeader.querySelector(".reveal-button")) { console.error("tried to add reveal button but post already has it, id: " + postID) return } postHeader.append(createRevealButton()) } async function main() { announce() addCSS() // runereader addRunereaderToggleFooterButton() decipherRunes() // reveal posts await sleep(2000) await doTheShim() // so we can listen for post insertion events addRevealButtonToPosts() } main()