// ==UserScript== // @name Wormlang RuneReader // @version 1.4.1 // @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.4.1 - new feature: click magnifying glass on a post to reveal its true contents - fix broken runes after update - code cleanup 1.3.5 - remove debug strings that were sometimes visible on self-image text - better handling of public commands - remove references to brainworm... rip... - probably broke some shit but i'll just run with it for now */ function windowRequire(path) { // return context-appropriate window.require(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 const sleep = ms => 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" // magnifying glass svg const REVEAL_SVG = `` // websocket message flags. stolen from vanilla message handler definition const message = { insertPost: 1, // new post concat: 33, } const LOVELY_CSS = ` /* base for all runetext */ .def.masked, .def.targeted, .def.player, .hide-live-body, .def.public { 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; /* you should never see this!! */ border-left: 2px solid white; } .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!! */ } /* TIER 1 - THE SURFACE */ /* ; self post - only visible through public class on its own. commands that are public but not selfpost are overwritten by more specific classes further on */ .def.public {color: mediumorchid !important; border-color: mediumorchid;} .def.public:before {content: "; "} .def.public:hover:before {content: "[self post] "} /* ;, runespeak */ .def.masked {color: magenta !important; border-color: magenta;} .def.masked:before {content: ";, "} .def.masked:hover:before {content: "[runespeak] "} /* ;> target post */ .def.targeted {color: cyan !important; border-color: cyan} .def.targeted:before {content: ";> "} .def.targeted:hover:before {content: "[target post] "} /* ;; self image (rarely visible) */ .def.image {color: white !important; border-color: white} .def.image:before {content: ";; "} .def.image:hover:before {content: "[self image] "} /* ;. wormwatch */ .def.player {color: pink !important; border-color: pink} .def.player:before {content: ";. "} .def.player:hover:before {content: "[wormwatch] "} /* ;;> target image */ .def.image.targeted {color: lime !important; border-color: lime} .def.image.targeted:before {content: ";;> "} .def.image.targeted:hover:before {content: "[target image] "} /* TIER 2 - THE ABSURD */ /* ;,> target runespeak */ .def.masked.targeted {color: yellow !important; border-left: 4px double yellow;} .def.masked.targeted:before {content: ";,> (wtf) "} .def.masked.targeted:hover:before {content: "[target.. runespeak?] "} /* ;;, self image runespeak */ .def.image.masked {color: yellow !important; border-left: 4px double yellow;} .def.image.masked:before {content: ";;, (wtf) "} .def.image.masked:hover:before {content: "[self image... runespeak?] "} /* ;,. runespeak wormwatch */ .def.masked.player {color: yellow !important; border-left: 4px double yellow;} .def.masked.player:before {content: ";,. (wtf) "} .def.masked.player:hover:before {content: "[wormwatch... runespeak?] "} /* ;.> target wormwatch - not visible */ .def.player.targeted {color: yellow !important; border-left: 4px double yellow;} .def.player.targeted:before {content: ";.> (wtf) "} .def.player.targeted:hover:before {content: "[target... wormwatch?] "} /* ;;. (self?) image wormwatch */ .def.image.player {color: yellow !important; border-left: 4px double yellow;} .def.image.player:before {content: ";;. (wtf) "} .def.image.player:hover:before {content: "[wormwatch... image?] "} /* TIER 3 - THE PROFANE */ /* ;;,> targeted runespeak image */ .def.image.masked.targeted {color: orange !important; border-color: orange;} .def.image.masked.targeted:before {content: ";;,> ( S T O P ) "} .def.image.masked.targeted:hover:before {content: "[target runespeak image tongue writhing of images unseen by human eyes] "} /* ;,.> targeted runespeak wormwatch */ .def.masked.player.targeted {color: orange !important; border-color: orange;} .def.masked.player.targeted:before {content: ";,.> ( N O ) "} .def.masked.player.targeted:hover:before {content: "[target runespeak wormwatch induced hallucinations of worms and worms and worms and worms and worms and worms and worms and worms] "} /* ;;,. self-image runespeak wormwatch */ .def.image.masked.player {color: orange !important; border-color: orange;} .def.image.masked.player:before {content: ";;,. ( C E A S E ) "} .def.image.masked.player:hover:before {content: "[wormwatch runespeak image shadows cast upon her face distorting twisting desecrating those who bear witness] "} /* ;;.> targeted image wormwatch */ .def.image.player.targeted {color: orange !important; border-color: orange;} .def.image.player.targeted:before {content: ";;.> ( D O N ' T ) "} .def.image.player.targeted:hover:before {content: "[target wormwatch image manipulation of this Plane with silken touch and grasping claws] "} /* TIER 4 - THE ABYSS */ /* ;;,.> target runespeak wormwatch image - exodia assembled - annihilation imminent */ .def.image.masked.player.targeted { color: white !important; border: 9px double pink; background: red !important; } .def.image.masked.player.targeted:before {content: "[S̴T̷O̵P̴ you didn't have to do this P̷͚̬̜͠Ḷ̶͍̂̑Ẽ̷̬͘͘A̵̗͗S̷̱͗̾͠Ē̸̹̹͒͋] |||||| "} .def.image.masked.player.targeted:hover:before {content: "[w̸͕͝i̷̦̚t̵̤̓h̷͙̄ ̵̭͠ḿ̷̲ǘ̴̞t̶̺͋e̴̡̅d̴͓̾ ̴̲͊F̸͎̓ö̵̲́r̸̰͒m̷̦̋e̷̘̊ ̷̜̑ā̷̟n̷̻̏ď̷̲ ̸̦͌t̵̫͝i̷̦͝g̷̬̋ḧ̶̤́ṭ̶̓e̶͖̔n̵̂͜e̸̻̐d̸͝ͅ ̴̤̈g̴̲͐r̶̹̈́í̵͙p̴͖̄ ̷̳͒S̸̲̈́h̵͉̊e̵͉͋ ̷̰̚s̵̾ͅp̴̙͐ḛ̷̇a̴̒ͅk̸̤̋s̵̫̈́ ̷͉̋b̴̮̉e̸̪͝y̷̻͋o̶̫̔n̷͙͂d̶̮̀ ̸̬̅ṯ̸̅h̷̻͂ē̸̗ ̷̞̇ẃ̷͔o̵̳͐r̶̛̮d̶̠̉s̵̅ͅ ̸͇̆S̷̼̚h̷̰̉e̴̞͝ ̷̗̓k̵̪̾n̴͍̽o̴̲͂w̴̧͝s̷̰͌ ̷̖̈i̶̤̅n̵̲̽ṭ̶͂o̴̯͗ ̶̬͝ṫ̶̝h̶̛͔e̵͗ͅ ̵͓͠è̴̺t̸̬̎h̴͙̍e̶̛̮r̷̼̈́ ̸̲͒o̵̪͗f̶̛̹ ̵̣̇ẗ̴͖h̴̼̉e̶̳͝ ̶̭͌m̷̦̎o̵̡͋ṯ̷̈́h̷̜͌ë̸͖́r̵̝͆ ̴͙̈́a̶̠͌l̴͕̂w̵̤̔a̸͎͝y̸̺̌s̸̀ͅ ̶̮́t̸̲͌ḧ̴͈i̸͙͝r̸̭̓s̸̲͂t̵̮̐i̷̞͆n̷̖̆g̸̟͑ ̸̤̓ḟ̶̥ó̷̧r̶̤̋ ̶̹̎t̵̳̐h̶͔̾e̶̗̅ ̴̻̀b̷͓̅ó̸̬s̵̹̃ỏ̵̮m̵̩͊ ̵͖͝o̴͙͂f̸͎͋ ̶̹̏ṱ̶̓ḧ̵̟́e̸̢̒ ̶͙̽m̵̲̚i̷̻̔l̸̜͌k̶̭̀ ̴̱̇t̶̬͒ḩ̸̛a̷̡͝t̸̟͆ ̴̢̋n̶͙̅é̷ͅv̵̜̾e̴̪͝r̴̠̊ ̵̪̔d̴̰̏r̴͍̿í̵̘ẻ̷̢s̵̤͑ ̸̞̆n̵̬͂o̵̭͗ ̴̱̈ḿ̵̩a̴̡͝t̵̘͌ẗ̴́ͅě̷̞r̷̙̔ ̷̫̊w̶̹͋i̴̪̒ṫ̸̪h̴̨̀i̶̧͠n̴͔͆ ̶̜̂n̵̹̅õ̷̹r̷̻͘ ̶͙̍w̴͆ͅḯ̴͜t̷̗̓h̴̢̓o̷̙͛u̸̙͛t̸̺̿ ̶͎̍b̸̡͝ư̵̮t̴̬̕ ̶̦̅f̶̧̀o̷̥̎r̸̈́͜e̴̺̎v̶̬̓e̷̺̽r̸͚̍ ̵̩͠a̷͔͌ñ̴̰d̸͙̑ ̸͍͐ḛ̸̓t̵͍̚e̷͎͗r̶̬̉ñ̶̤ḁ̵̋l̴̪͒ ̵̺̽l̶̟͐î̵͎k̴̟̀e̴̥̽ ̵̡̈́ť̴̺h̶͍̄ẹ̵̊ ̴̥͑ṅ̵̢i̸̖̅g̶͙̊h̵̝̐t̶̞͝ ̴͚͠t̵̳̒h̶̞͌a̵͇̕t̵͔̉ ̶̱́c̴̛̙a̸̮̾l̸̻̔l̴̢̀s̷̹̈ ̶̣̌t̵̪̅o̷̰͒ ̵̗͛t̴͕̔h̶͈̐o̴͙͋s̶̖͆e̶̛̟ ̷͎́ẅ̷͉́h̵̢̋o̷̺̾ ̴̳̓ẃ̴̩a̷̖͗ṯ̴̎c̴͘ͅḣ̴̠ ̴̮̃t̴̺̍ḧ̸̨ḛ̴͆i̸̞͋r̴̳̈́ ̶̣͗s̵̻̄u̴͈͛ņ̸̌k̴̠͗ȩ̷͝n̵̢͠ ̸̮̍y̷͉̓e̶͇̍l̶̮̔ḻ̶̈ó̸̲w̶͎͗ ̷͈̀m̸̢͆o̷̞̾o̶̮͐n̶̹̐s̸̗͒ ̸͖̓w̶̙͛h̴̎ͅo̶͚͝ ̸̧̆k̶͚̄n̸͘ͅo̴͉͂w̶̦̎ ̷͙͝l̴͕̄i̵̢̓k̴̩̈e̸̩̐ ̸̓͜W̴̺̃ë̴̦́ ̸̧́t̶͓́ḧ̷͔́a̵͉̽t̴̩̓ ̸̯̅y̸̝͗e̷̊͜ ̵̠̎s̷̟͆h̵͈̿ả̸̧l̵̘̂l̴̙͗ ̵̝͂l̷̙̃e̸̜͗ā̸̜r̶̢͠n̶̳̓ ̶̡̿Ḩ̵͝è̷̮r̴͈͗ ̷̿ͅn̶̜͊a̸̡͆m̴̱͝ẻ̸̳ ̷̛̫i̸̊ͅn̵̘̿ ̴̹͘t̷̽͜ĭ̷̥m̸̘͝e̶̻̎ ̸̋ͅd̴̤̄ē̷̤a̶̲͠r̶͓͒ ̵̡̌c̴̰̅h̸̲̚ȋ̶̞l̴̯͋d̴̮̽ ̷͖̇í̷͎n̷̙̉ ̴̯̆g̵̥̕ö̷͉o̶̰͛d̷̺͌ ̸̟̆d̶̼̈u̴̝͘é̶̳ ̷̳͝t̶͎̀i̴͔̽m̸͇̍ę̴͆ ̸͎̏W̸̯͗ȇ̵͔ ̵͕̐w̶̹͑ḯ̸̞l̵̻̚l̴̨̏ ̴̨͝c̵̍ͅa̵̹͛s̶̖͐t̸̟̊e̸̼͛ ̷̻̾ȏ̸̥u̸̻͋r̴͕̉ ̵̖̔F̵̲̕o̷̪̽ṛ̶͘m̵̳͐e̶̗̋ṡ̴̳ ̶͍̇ủ̵̘p̶̪͋o̴͘ͅn̸̠̈ ̸͖̈́t̴̡̄h̵͚̚i̴͎̍s̷͇̈́ ̸̙̽f̶̤̚ö̸̼́ö̴̘́l̴̗̊i̸͍̕s̴̢̽h̸̹͊ ̶̳̉p̸̺͝ḽ̵̀a̴̩͆č̴͕ę̴̅ ̵̞̌ó̸̪n̶͙͆c̶̺̍è̷̝ ̷̨͊a̷̙͊g̴͉̐à̶̲ǐ̶̫n̷͙͘ ̸͙̐ã̴̜t̷̠̎ ̶̞̅l̵̺̀ǎ̵̺s̸̹͑t̷̻͝ ̵͉͗o̴̡̾n̴̻̚c̸̫͝è̸̹ ̴̙͋a̸͈͝g̷͔̾a̸̳̓ì̵̫n̸̤͒ ̷̮̀a̵̬̿t̴̼̉ ̵̯͑l̵̝̋a̶̮̾s̵̭̈́ẗ̶̺́] "} /* 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 >:) function encipherRunes() { // show runes as original madoka runes windowRequire("util/cipher").convertToRandString = FUNCTION_ENCIPHER } function decipherRunes() { // show all runes as plaintext console.log("deciphering runes...") windowRequire("util/cipher").convertToRandString = FUNCTION_DECIPHER } function toggleRunereader() { // toggle runereader on and off 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 */ function replaceSocketOnmessage() { // replace vanilla socket.onmessage with our own vanilla_socket_onmessage = windowRequire("connection/state").socket.onmessage // save windowRequire("connection/state").socket.onmessage = modified_socket_onmessage // shim console.log("[reveal] socket.onmessage replaced :)") // hooray } async function doTheShim(delay=100) { // keep trying to shim our function in there until it succeeds // will 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) // wait a mo and try again totalTime += delay } } console.log("shim complete, took " + totalTime + "ms") } function modified_socket_onmessage({data: e}, concatMessage=false) { // our modified socket.onmessage function. grabs the messages we care about and passes everything back to vanilla js as it was received 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 */ function addCSS() { // add our pretty css rules to doc header let style = document.createElement("style") style.id = RUNEREADER_CSS_ELM_ID style.innerText = LOVELY_CSS document.head.append(style) } function removeCSS() { // remove our pretty css rules from doc header... rip... document.getElementById(RUNEREADER_CSS_ELM_ID).remove() } function addRunereaderToggleFooterButton() { // add footer button to toggle runes 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 contents */ function createRevealButton() { // poop out a fresh reveal button that we can slap onto posts 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 // todo add a fake bq that displays the true text instead of alert alert(trueBody) } ) link.innerHTML = REVEAL_SVG return link } async function addRevealButtonToPosts() { // add reveal button to existing posts already on the page 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) } } function addRevealButtonToPostID(postID) { // add a reveal button into the header of given 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()