// ==UserScript== // @name LynxChan Extended Minus Minus // @namespace https://rentry.org/8chanMinusMinus // @version 1.32 // @description It's like 4chanXT but worse // @author SaddestPanda & Dandelion & /gfg/ // @license UNLICENSE // @match *://8chan.moe/*/res/* // @match *://8chan.se/*/res/* // @match *://8chan.cc/*/res/* // @grant GM.getValue // @grant GM.setValue // @grant GM.deleteValue // @grant GM.registerMenuCommand // @run-at document-idle // @downloadURL none // ==/UserScript== (async function () { "use strict"; const SETTINGS_DEFINITIONS = { firstRun:{ default:true, hidden:true, desc:"You shouldn't be able to see this setting! (firstRun)" }, showScrollbarMarkers:{ default:true, desc:"Show your posts and replies on the scrollbar" }, spoilerImageType:{ default:"off", desc:"Override how the spoiler thumbnail looks", type:"radio", options:{ off:"Don't change the thumbnail.", reveal:"Reveal spoilers. Previously spoilered images will have a red border around them indicating that they're spoilers.", kachina:"Makes the spoiler image Kachina from Genshin Impact.", thread:`Uses the first image of the first visible post on the current thread with the filename "ThreadSpoiler.jpg" (or .png or .webp)`, threadAlt:`same as above with the filename "ThreadSpoilerAlt.jpg" (or .png or .webp)` } }, useExtraStylingFixes:{ default:true, desc:"Apply some styling fixes (Mark your posts and replies, smaller thumbnails etc.)" }, revealSpoilerText:{ default:"off", desc:"Reveal the spoiler text. Or make it into madoka runes.", type:"radio", options:{ off:"Don't reveal spoilers.", on:"Spoilers will be shown by turning the text white.", madoka:`Spoilers will turn into madoka runes. Please install MadokaRunes.ttf for it to show up properly.` } }, showPostIndex:{ default:true, desc:"Show the current index of a post on the thread. That is, the topmost post will start at 1 and count up from there." }, /*showStubs:{ default:true, desc:"Show post stubs when filtering." }, redirectToCatalog:{ default:false, desc:"Redirect to catalog when clicking on the index." }*/ } const settingsNames = Object.keys(SETTINGS_DEFINITIONS); const settingsValues = await Promise.all(settingsNames.map(key => GM.getValue(key, SETTINGS_DEFINITIONS[key]['default']))); const settings = Object.fromEntries(settingsNames.map((key, index) => [key, settingsValues[index]])); console.log("%cLynx Minus Minus Started with settings:", "color:rgb(0, 140, 255)", settings); addMyStyle("lynx-extended-css", ` .marker-container { position: fixed; top: 16px; right: 0; width: 10px; height: calc(100vh - 40px); z-index: 11000; pointer-events: none; } .marker { position: absolute; width: 100%; height: 6px; background: #0092ff; cursor: pointer; pointer-events: auto; border-radius: 40% 0 0 40%; z-index: 5; } .marker.alt { background: #a8d8f8; z-index: 2; } #lynxExtendedMenu { position: fixed; top: 15px; right: 100px; padding: 10px; z-index: 10000; font-family: Arial, sans-serif; font-size: 14px; box-shadow: 0 2px 5px rgba(0, 0, 0, 0.5); background: #353535; border: 1px solid #737373; color: #ddd; border-radius: 4px; } `); // Register menu command GM.registerMenuCommand("Show Options Menu", openMenu); try { createSettingsButton(); } catch (error) { console.log("Error while creating settings button:", error); } //Open the settings menu on the first run if (settings.firstRun) { settings.firstRun = false; await GM.setValue("firstRun", settings.firstRun); openMenu(); } function replyKeyboardShortcuts(ev) { if (ev.ctrlKey) { let combinations = { "s":["[spoiler]","[/spoiler]"], "b":["'''","'''"], "u":["__","__"], "i":["''","''"], "d":["[doom]","[/doom]"], "m":["[moe]","[/moe]"] } for (var key in combinations) { if (ev.key == key) { ev.preventDefault(); console.log("ctrl+"+key+" pressed in textbox") const textBox = ev.target; let newText = textBox.value; const tags = combinations[key] const selectionStart = textBox.selectionStart const selectionEnd = textBox.selectionEnd if (selectionStart == selectionEnd) { //If there is nothing selected, make empty tags and center the cursor between it document.execCommand("insertText",false, tags[0] + tags[1]); //Center the cursor between tags textBox.selectionStart = textBox.selectionEnd = (textBox.selectionEnd - tags[1].length); } else { //Insert text and keep undo/redo support (Only replaces highlighted text) document.execCommand("insertText",false, tags[0] + newText.slice(selectionStart, selectionEnd) + tags[1]) } return; } } } else if (ev.key == "Escape") { //Because greasemonkey cannot access the JS of the page we have to do some funny stuff document.getElementById("quick-reply").querySelector(".close-btn").click() } } document.getElementById("qrbody").addEventListener("keydown", replyKeyboardShortcuts); // Create markers 1 second after page load setTimeout(() => { recreateScrollMarkers(); }, 1500); if (settings.showPostIndex) { setTimeout(() => { addPostCount(); }, 1400); } function openMenu() { const oldMenu = document.getElementById("lynxExtendedMenu"); if (oldMenu) { oldMenu.remove(); return; } // Create options menu const menu = document.createElement("div"); menu.id = "lynxExtendedMenu"; menu.innerHTML = `

LynxChan Extended-- Options


`; Object.keys(SETTINGS_DEFINITIONS).forEach((name) => { const setting = SETTINGS_DEFINITIONS[name]; if (setting.hidden) { //pass } else if (setting.type == "radio") { let html = `${setting.desc}
` for (const [value, description] of Object.entries(setting.options)) { html += `
` } html += "

" menu.innerHTML += html; } else { menu.innerHTML += `

` } }) menu.innerHTML += ` ` document.body.appendChild(menu); // Save button functionality document.getElementById("saveSettings").addEventListener("click", async () => { Object.keys(SETTINGS_DEFINITIONS).forEach((name) => { const setting = SETTINGS_DEFINITIONS[name]; if (!('hidden' in setting)) { if (setting.type=="radio") { settings[name] = document.querySelector(`input[name="${name}"]:checked`).value } else { settings[name] = document.getElementById(name).checked; } } }) console.log("Saving settings ",settings) await Promise.all(Object.entries(settings).map(([key, value]) => GM.setValue(key, value))); alert("Settings saved!\nRefresh the page for the changes to take effect."); menu.remove(); }); // Close button functionality document.getElementById("closeMenu").addEventListener("click", () => { menu.remove(); }); } function createSettingsButton() { document.querySelector("#navLinkSpan > .settingsButton").insertAdjacentHTML("afterend", ` / `); document.querySelector("#navigation-lynxextended").addEventListener("click", openMenu); } function addMyStyle(newID, newStyle) { let myStyle = document.createElement("style"); //myStyle.type = 'text/css'; myStyle.id = newID; myStyle.textContent = newStyle; document.querySelector("head").appendChild(myStyle); } function createMarker(element, container, isReply) { const pageHeight = document.body.scrollHeight; const offsetTop = element.offsetTop; const percent = offsetTop / pageHeight; const marker = document.createElement("div"); marker.classList.add("marker"); if (isReply) { marker.classList.add("alt"); } marker.style.top = `${percent * 100}%`; marker.dataset.postid = element.id; marker.addEventListener("click", () => { let elem = element?.previousElementSibling || element; elem.scrollIntoView({ behavior: "smooth", block: "start" }); }); container.appendChild(marker); } function recreateScrollMarkers() { let oldContainer = document.querySelector(".marker-container"); if (oldContainer) { oldContainer.remove(); } // Create marker container const markerContainer = document.createElement("div"); if (settings.showScrollbarMarkers) { markerContainer.classList.add("marker-container"); document.body.appendChild(markerContainer); } // Match and create markers for "my posts" (matches native & dollchan) document.querySelectorAll(".postCell:has(.innerPost > .postInfo.title :is(.linkName.youName, .de-post-counter-you)),.postCell:has(.innerPost.de-mypost)") .forEach((elem) => { createMarker(elem, markerContainer, false); }); // Match and create markers for "replies" (matches native & dollchan) document.querySelectorAll(".postCell:has(.innerPost > .divMessage :is(.quoteLink.you, .de-ref-you)),.postCell:has(.innerPost.de-mypost-reply)") .forEach((elem) => { createMarker(elem, markerContainer, true); }); } function addPostCount() { //This function causes a DOMException, I don't know why, just ignore it const posts = Array.from(document.getElementsByClassName("divPosts")[0].children); //Why is the insert method called unshift???? This inserts it at the beginning //(This is also insanely inefficient since we only need to do it once) posts.unshift(document.querySelector(".innerOP")) for (let i=0; i img[src='/spoiler.png']"); spoilers.forEach(spoiler => { const parent = spoiler.parentElement; const hrefTokens = parent.href.split("/"); const fileNameTokens = hrefTokens[4].split("."); const thumbUrl = `/.media/t_${fileNameTokens[0]}`; spoiler.src = thumbUrl; spoiler.style.border = "thin dotted red"; spoiler.style.borderWidth = "2px"; }); } if (settings.showScrollbarMarkers || settings.showPostIndex) { const observer = new MutationObserver((mt_callback) => { mt_callback.forEach(mut => { if (mut.type=="childList") { //console.log("MutationObserver!!!"); // Recreate markers because the page grew taller. Is this heavy? probably not. recreateScrollMarkers(); if (settings.showPostIndex) addPostCount(); if (settings.spoilerImageType=="reveal") revealSpoilerImages(); } }) }) observer.observe(document.querySelector(".divPosts"), {'childList':true}) // I'm not sure why but this doesn't work // // Add a second observer for #threadList (new posts) // const threadObserver = new MutationObserver((mutationsList) => { // for (const mutation of mutationsList) { // if (mutation.type === 'childList') { // mutation.addedNodes.forEach((node) => { // if (node.classList && node.classList.contains("postCell")) { // console.log("ThreadObverver!!!") // // Recreate markers because the page grew taller. Is this heavy? probably not. // recreateScrollMarkers(); // if (settings.showPostIndex) // addPostCount(); // if (settings.spoilerImageType=="reveal") // revealSpoilerImages(); // } // }); // } // } // }); // const threadList = document.querySelector("#threadList"); // if (threadList) { // threadObserver.observe(threadList, { childList: true }); // } } // Apply the CSS if the setting is enabled if (settings.useExtraStylingFixes) { addMyStyle("extra-styling-css", ` /* smaller thumbnails & image paddings */ body .uploadCell img:not(.imgExpanded) { max-width: 160px; max-height: 125px; object-fit: contain; height: auto; width: auto; margin-right: 0em; margin-bottom: 0em; } .imgExpanded { max-height:100vh; } .uploadCell .imgLink { margin-right: 1.5em; } /* smaller post spacing (not too much) */ .divMessage { margin: .8em .8em .5em 3em; } /*.greenText { filter: brightness(110%); }*/ /* Make your name in your post red */ .youName { color: red; } .you { --link-color: red; } /* mark your posts and replies (same selectors are also used for detection above) */ .postCell:has(.innerPost > .postInfo.title :is(.linkName.youName, .de-post-counter-you)), .postCell:has(.innerPost.de-mypost) { & > .innerPost { border-left: 3px dashed; border-left-color: #4BB2FFC2; padding-left: 0px; } } .postCell:has(.innerPost > .divMessage :is(.quoteLink.you, .de-ref-you)), .postCell:has(.innerPost.de-mypost-reply) { & > .innerPost { border-left: 2px solid; border-left-color:rgb(0, 102, 255); padding-left: 1px; } } `); } if (settings.revealSpoilerText=="on") { addMyStyle("reveal-spoilers",` .span.spoiler { color: white} `) } else if (settings.revealSpoilerText="madoka") { addMyStyle("reveal-spoilers",` span.spoiler:not(:hover) { color: white; font-family:MadokaRunes!important; } `) } // Add functionality to apply the custom spoiler image CSS if (settings.spoilerImageType=="thread" || settings.spoilerImageType=="threadAlt") { let spoilerImageUrl = null; if (settings.spoilerImageType=="thread") { const spoilerLink = Array.from(document.querySelectorAll('a.originalNameLink[download^="ThreadSpoiler"]')).find(link => /\.(jpg|png|webp)$/i.test(link.download)); spoilerImageUrl = spoilerLink ? spoilerLink.href : null; } else if (settings.spoilerImageType=="threadAlt") { const altSpoilerLink = Array.from(document.querySelectorAll('a.originalNameLink[download^="ThreadSpoilerAlt"]')).find(link => /\.(jpg|png|webp)$/i.test(link.download)); spoilerImageUrl = altSpoilerLink ? altSpoilerLink.href : null; } if (spoilerImageUrl) { addMyStyle("thread-spoiler-css", ` .uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]) { background-image: url("${spoilerImageUrl}"); background-size: cover; outline: dashed 2px #ff0000f5; & > img[src="/spoiler.png"] { opacity: 0; } } `); } } else if (settings.spoilerImageType=="reveal") { revealSpoilerImages(); } else if (settings.spoilerImageType=="kachina") { addMyStyle("kachinaSpoilers",` .uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]) { background-size: cover; margin-right:5px; background-image: url(""); & > img[src="/spoiler.png"] { opacity: 1; transform: translate(0, -25%) scale(0.5); }} `) } })();