// ==UserScript== // @name LynxChan Extended Minus Minus // @namespace https://rentry.org/8chanMinusMinus // @version 2.1.0 // @description LynxChan Extended with even more features // @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-start // @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)" }, addKeyboardHandlers:{ default:true, desc:"Add keyboard Ctrl+ hotkeys to the quick reply box (Disable this for 8chanSS compatibility)" }, showScrollbarMarkers:{ default:true, type:"checkbox_with_colors", desc:"Show your posts and replies on the scrollbar", color1Default:"#0092ff", color1Desc:"Your marker:", color2Default:"#a8d8f8", color2Desc:"Reply marker:" }, 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.)", reveal_blur:"Change to a blurred thumbnail (Unblurred when you hover your mouse over.)", kachina:"Makes the spoiler image Kachina from Genshin Impact.", thread:`Use "ThreadSpoiler.jpg" from the current thread (first posted jpg, png or webp image with that filename)`, threadAlt:`same as above with the filename "ThreadSpoilerAlt.jpg" (jpg, png or webp; uses ThreadSpoiler.jpg until this is found)`, //test:`[TEST OPTION] Randomly pick spoiler image from /gacha/ board (This is a test option. It selects the spoiler from var(--spoiler-img) after setting.)` }, nonewline:true }, overrideBoardSpoilerImage: { default:true, parent:"spoilerImageType", //Not implemented yet //depends: function() {return settings.spoilerImageType != "off"}, desc:"Also override board's custom thumbnail image (for example, /v/'s spoiler thumbnail is an image of a monitor with a ? inside it)" }, 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 always be shown by turning the text white.", madoka:`Spoilers will turn into madoka runes. Please install MadokaRunes.ttf for it to show up properly.` } }, markPostEdge:{ default:true, type:"checkbox_with_colors", desc:"Style: Mark your posts and replies (with a left border)", color1Default:"#4BB2FF", color1Desc:"Your border:", color2Default:"#0066ff", color2Desc:"Reply border:", nonewline:true }, markYouText:{ default:true, type:"checkbox_with_colors", desc:"Style: Color your name and (You) links", color1Default:"#ff2222", color1Desc:"Color:", nonewline:true }, compactPosts:{ default:true, desc:"Style: Make thumbnails and posts more compact", nonewline:true }, showStubs:{ default:true, desc:"Style: Show post stubs when filtering", nonewline:true }, //I swear this used to be a built in option on 8chan halfchanGreentexts:{ default:false, desc:"Style: Make the greentext brighter like 4chan" }, glowFirstPostByID:{ default:true, type:"checkbox_with_colors", desc:"Mark new/unique posters by adding a glow effect to their ID", color1Default:"#26bf47", color1Desc:"Glow color:" }, showPostIndex:{ default:true, type:"checkbox_with_colors", desc:"Show the current index of a post on the thread. (OP: 1, first post: 2 etc.)", color1Default:"#7b3bcc", color1Desc:"Index color:" }, preserveQuickReply:{ default:false, desc:"Preserve the quick reply text when closing the box or refreshing the page" } /*redirectToCatalog:{ default:false, desc:"Redirect to catalog when clicking on the index." }*/ } const settingsNames = Object.keys(SETTINGS_DEFINITIONS); //Collect all color fields for checkbox_with_colors settings //In the userscript storage they look like settingName_color1 etc. const colorSettingKeys = []; settingsNames.forEach(key => { const def = SETTINGS_DEFINITIONS[key]; if (def.type === "checkbox_with_colors") { Object.keys(def).forEach(k => { const match = k.match(/^color(\d+)Default$/); if (match) { colorSettingKeys.push(`${key}_color${match[1]}`); } }); } }); //Compose all keys to load: main settings + color fields const allSettingKeys = [...settingsNames, ...colorSettingKeys]; //For each color field, get its default from the definition function getDefaultForKey(key) { const colorMatch = key.match(/^(.+)_color(\d+)$/); if (colorMatch) { const [_, base, idx] = colorMatch; const def = SETTINGS_DEFINITIONS[base]; //Return color setting default like color1Default return def && def[`color${idx}Default`] ? def[`color${idx}Default`] : undefined; } //Return regular setting return SETTINGS_DEFINITIONS[key]?.default; } const allSettingDefaults = allSettingKeys.map(getDefaultForKey); const allSettingValues = await Promise.all(allSettingKeys.map((key, i) => GM.getValue(key, allSettingDefaults[i]))); const settings = Object.fromEntries(allSettingKeys.map((key, i) => [key, allSettingValues[i]])); function addMyStyle(newID, newStyle) { let myStyle = document.createElement("style"); //myStyle.type = 'text/css'; myStyle.id = newID; myStyle.textContent = newStyle; document.head.appendChild(myStyle); } function waitForDom(callback) { if (document.readyState === "loading") { //Loading hasn't finished yet. Wait for the inital document to load and start. document.addEventListener("DOMContentLoaded", callback); } else { //Document has already loaded. Start. callback(); } } if (document?.head) { runASAP(); } else { //On some environments document.head doesn't exist yet? waitForDom(runASAP); } async function runASAP() { // Migrations can be removed in a few weeks // Migrate old useExtraStylingFixes setting if present const oldStyling = await GM.getValue("useExtraStylingFixes", undefined); if (typeof oldStyling !== "undefined") { // If oldStyling is false, set both new options to false if (oldStyling === false) { settings.markPostEdge = false; settings.compactPosts = false; await GM.setValue("markPostEdge", false); await GM.setValue("compactPosts", false); } // Remove the old setting await GM.deleteValue("useExtraStylingFixes"); } // Migrate old markYourPosts setting if present const oldMarkYourPosts = await GM.getValue("markYourPosts", undefined); if (typeof oldMarkYourPosts !== "undefined") { settings.markPostEdge = oldMarkYourPosts; settings.markYouText = oldMarkYourPosts; await GM.setValue("markPostEdge", oldMarkYourPosts); await GM.setValue("markYouText", oldMarkYourPosts); await GM.deleteValue("markYourPosts"); } //Secret tip for anyone manually editing colors: //if you edit the saved value in your userscript manager's settings database manually, you can use semi-transparent colors for the color pickers (until you click save on the settings menu). //or easier: just copy the relevant part of the css and paste it to the css box in the website settings. Add !important if you want to force it like: color: red !important; //Apply all the styles as soon as possible if (settings.compactPosts) { addMyStyle("lynx-compact-posts", ` /* 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; object-fit:contain } .uploadCell .imgLink { margin-right: 1.5em; } /* smaller post spacing (not too much) */ .divMessage { margin: .8em .8em .5em 3em; } `); } const markerColor1 = settings.showScrollbarMarkers_color1 || SETTINGS_DEFINITIONS.showScrollbarMarkers.color1Default; const markerColor2 = settings.showScrollbarMarkers_color2 || SETTINGS_DEFINITIONS.showScrollbarMarkers.color2Default; const indexColor = settings.showPostIndex_color1 || SETTINGS_DEFINITIONS.showPostIndex.color1Default; const glowColor = settings.glowFirstPostByID_color1 || SETTINGS_DEFINITIONS.glowFirstPostByID.color1Default; addMyStyle("lynx-extended-css", ` :root { --showScrollbarMarkers_color1: ${markerColor1}; --showScrollbarMarkers_color2: ${markerColor2}; --showPostIndex_color1: ${indexColor}; --glowFirstPostByID_color1: ${glowColor}; } .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: var(--showScrollbarMarkers_color1); cursor: pointer; pointer-events: auto; border-radius: 40% 0 0 40%; z-index: 5; } .marker.alt { background: var(--showScrollbarMarkers_color2); z-index: 2; } .postNum.index { color: var(--showPostIndex_color1); font-weight: bold; } .labelId.glows { box-shadow: 0 0 15px var(--glowFirstPostByID_color1); } #lynxExtendedMenu { position: fixed; top: 15px; left: 50%; transform: TranslateX(-50%); 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: var(--contrast-color); color: var(--text-color); border: 1px solid #737373; border-radius: 4px; max-height:100%; overflow-y: auto; & .altText { opacity: 0.8; font-size: 0.9em; &.lineBefore:before { content: "—— "; } } & .boldText { color: var(--link-color); font-weight: bold; } & input[type="color"] { width: 40px; height: 20px; padding: 1px; transform: translate(0, 2px); } & button { padding: 10px 20px; margin-right: 4px; filter: contrast(115%) brightness(110%); &:hover { filter: brightness(130%); } } } /*What the fuck is up with CSS */ /*#lynxExtendedMenu.settings-content { max-height: 90%; }*/ #lynxExtendedMenu > .settings-footer { height: auto; } @media screen and (max-width: 1000px) { #lynxExtendedMenu{ right:0; width:90%; /*bottom:15px;*/ } } .lynxExtendedButton::before { content: "\\e0da"; `); if (settings.markPostEdge) { const color1 = settings.markPostEdge_color1 || SETTINGS_DEFINITIONS.markPostEdge.color1Default; const color2 = settings.markPostEdge_color2 || SETTINGS_DEFINITIONS.markPostEdge.color2Default; addMyStyle("lynx-mark-posts", ` /* mark your posts and replies */ #divThreads .postCell .innerPost:has(> .postInfo.title > .youName) { border-left: 3px dashed var(--markPostEdge_color1, ${color1}); padding-left: 1px; } #divThreads .postCell .innerPost:has(> .divMessage .quoteLink.you) { border-left: 2px solid var(--markPostEdge_color2, ${color2}); padding-left: 1px; } `); } if (settings.markYouText) { const color1 = settings.markYouText_color1 || SETTINGS_DEFINITIONS.markYouText.color1Default; addMyStyle("lynx-mark-you-text", ` .youName { color: var(--markYouText_color1, ${color1}); } .you { --link-color: var(--markYouText_color1, ${color1}); } `); } if (settings.halfchanGreentexts) { addMyStyle("lynx-halfchanGreentexts", `.greenText { filter: brightness(110%); } `); } if (settings.showStubs === false) { addMyStyle("lynx-hide-stubs",` .postCell:has(> span.unhideButton.glowOnHover) { display: none; } `); } if (settings.revealSpoilerText=="on") { addMyStyle("lynx-reveal-spoilertext1",` span.spoiler { color: white } `); } else if (settings.revealSpoilerText=="madoka") { addMyStyle("lynx-reveal-spoilertext2",` span.spoiler:not(:hover) { color: white; font-family: MadokaRunes !important; } `); } } //End of runASAP() //Everything in runAfterDom runs after document has loaded (like @run-at document-end) //Everything in runAfterDom runs after document has loaded (like @run-at document-end) //Everything in runAfterDom runs after document has loaded (like @run-at document-end) async function runAfterDom() { console.log("%cLynx Minus Minus Started with settings:", "color:rgb(0, 140, 255)", settings); if (typeof api !== "undefined") { console.log("The script is not sandboxed. Adding quick reply shortcut.") function quickReplyShortcut(ev) { if ((ev.ctrlKey && ev.key == "q") || (ev.altKey && ev.key=="r")) { ev.preventDefault(); //8chan's HTML will keep the text after a reload so attempt to clear it again if (settings.preserveQuickReply===false) { document.getElementById("qrbody").value = ""; } qr.showQr(); document.getElementById('qrbody')?.focus(); }; } document.addEventListener("keydown",quickReplyShortcut); } else { //I think greasemonkey sandboxes the script. I use violentmonkey though console.log("JS script is sandboxed and can't access page JS... (If you can read this, let me know what browser/extension does this. Or maybe the site just failed to load?)") } function createSettingsButton() { //Desktop document.querySelector("#navLinkSpan > .settingsButton").insertAdjacentHTML("afterend", ` / `); //Mobile document.querySelector("#sidebar-menu > ul > li > .settingsButton").parentElement.insertAdjacentHTML("afterend", `
  • Lynx Ex-- Settings
  • `); document.querySelector("#navigation-lynxextended").addEventListener("click", openMenu); document.querySelector("#navigation-lynxextended-mobile").addEventListener("click", openMenu); } //Register menu command for the settings button 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; } } //Ctrl+Enter to send reply if (ev.key=="Enter") { document.getElementById("qrbutton")?.click() } } } if (settings.addKeyboardHandlers) { document.getElementById("qrbody").addEventListener("keydown", replyKeyboardShortcuts); document.getElementById("quick-reply").addEventListener('keydown',function(ev) { if (ev.key == "Escape") { document.getElementById("quick-reply").querySelector(".close-btn").click() } }) } //I'm not sure who would ever want this on but I'm making it an option anyways if (settings.preserveQuickReply===false) { document.getElementById("quick-reply").querySelector(".close-btn").addEventListener("click", function(ev){ document.getElementById("qrbody").value = ""; }); //This doesn't replace the built in onclick but adds to it so the original onclick will still bring up the qr document.getElementById("replyButton")?.addEventListener("click", function(ev){ ev.preventDefault(); const qrBody = document.getElementById("qrbody"); if (qrBody) { qrBody.value = ""; qrBody?.focus(); } }); } 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


    `; //we use createElement() here instead of setting innerHTML so we can attach onclick to elements //...In the future, at least. There aren't any onclicks added yet. let settings_content = document.createElement("div"); settings_content.classList.add("settings-content"); 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 += `
    ${setting.nonewline ? '' : '
    '}`; settings_content.innerHTML += html; } else if (setting.type == "checkbox_with_colors") { let colorHtml = ""; let colorFields = Object.keys(setting).filter(k => /^color\d+Default$/.test(k)); colorFields.forEach((colorKey) => { const idx = colorKey.match(/^color(\d+)Default$/)[1]; const colorValue = settings[`${name}_color${idx}`] || setting[`color${idx}Default`]; const colorDesc = setting[`color${idx}Desc`] || ""; colorHtml += ` `; }); settings_content.innerHTML += ` ${colorHtml}
    ${setting.nonewline ? '' : '
    '}`; } else { settings_content.innerHTML += `
    ${setting.nonewline ? '' : '
    '}`; } }) menu.appendChild(settings_content); 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] = menu.querySelector(`input[name="${name}"]:checked`).value } else if (setting.type=="checkbox_with_colors") { settings[name] = document.getElementById(name).checked; let colorFields = Object.keys(setting).filter(k => /^color\d+Default$/.test(k)); colorFields.forEach((colorKey) => { const idx = colorKey.match(/^color(\d+)Default$/)[1]; const colorName = `${name}_color${idx}`; const colorValue = document.getElementById(colorName).value; settings[colorName] = colorValue; // Set CSS variable on body (so it can be used without a refresh) document.body.style.setProperty(`--${colorName}`, colorValue); }); } 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))); setTimeout(()=>{ alert("Settings saved!\nFor most settings you must refresh the page for the changes to take effect.\n\n(only color pickers don't need a refresh)"); }, 1); // menu.remove(); }); // Reset button functionality document.getElementById("resetSettings").addEventListener("click", async () => { if (!confirm("Are you sure you want to reset all settings? This will delete all saved data.")) return; const keys = await GM.listValues(); await Promise.all(keys.map(key => GM.deleteValue(key))); alert("All settings have been reset.\nRefreshing automatically for the changes to take effect."); menu.remove(); location.reload(); }); // Close button functionality document.getElementById("closeMenu").addEventListener("click", () => { menu.remove(); }); } 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; if (elem) 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"); 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); }); } let postCount = 1; const postIndexLookup = {}; function addPostCount(post, newpost = true) { // const posts = Array.from(document.querySelectorAll(".innerOP, .divPosts > .postCell")); if (post.querySelector(".postNum")) { return; } const postInfoDiv = post.getElementsByClassName("title")[0] if (!postInfoDiv) { console.error("[Lynx--] Failed to find post for div ", post); return; } const posterNameDiv = postInfoDiv.getElementsByClassName("linkName")[0]; const postNumber = post.querySelector(".linkQuote")?.textContent; if (!postNumber) return; let localCount = postCount; if (newpost) { postIndexLookup[postNumber] = localCount; postCount++; } else { //Show cached post count for inlines & hovers localCount = postIndexLookup[postNumber]; if (!localCount) return; } let newNode = document.createElement("span"); newNode.innerText = localCount; newNode.className = "postNum index"; if (localCount < Infinity) //knownBumpLimit { // color is handled by .postNum.index newNode.style = ""; } else { newNode.style = "color: rgb(255, 4, 4); font-weight: bold;" } postInfoDiv.insertBefore(newNode, posterNameDiv); let foo = document.createTextNode("\u00A0"); // Non-breaking space postInfoDiv.insertBefore(foo, posterNameDiv); } //mark cross-thread links. const indicateCrossLinks = function(post) { const crossLinks = post.querySelectorAll(`a.quoteLink:not(.crossThread):not([href*='${api.boardUri}/res/${api.threadId}'])`); crossLinks.forEach(crossLink => { //ignore cross-board links (they look obvious like >>>/board/123456 ) if (!crossLink.href.includes(`/${api.boardUri}/`)) { return; } crossLink.classList.add("crossThread"); const hrefTokens = crossLink.href.split("#"); const quoteLinkId = hrefTokens[1]; crossLink.innerHTML = ">>" + quoteLinkId; }); } function addDeletedChecks(post) { const postLinks = post.querySelectorAll(`a.quoteLink[href*='${api.boardUri}/res/${api.threadId}']`); //This goes bottom to top so we stop when we've reached a post with a check attached for (let i = postLinks.length-1; i>=0; i--) { //We've reached posts where we already added numbers, // there's no need to keep going. if (postLinks[i].hasMouseOverEvent) { break; } var evListener = function(ev) { if (!document.getElementById(ev.target.href.split("#").pop())) { ev.target.classList.add("deleted") //Sadly this doesn't actually work and I don't know why (S.Panda: postlinks[i] is gone by the time the event is ran) //postLinks[i].removeEventListener("mouseenter",evListener) ev.target.closest("a.quoteLink")?.removeEventListener("mouseenter", evListener); } } postLinks[i].addEventListener("mouseenter", evListener); //Why does js allow this postLinks[i].hasMouseOverEvent = true; } } addMyStyle("lynx-linkHelpers",` .quoteLink.crossThread::after { content: " \(Cross-thread\)"; } .quoteLink.deleted::after { content: " \(Deleted\)"; } `) function imageSearchHooks(post) { //You ever think about how we're iterating over every single post every single time for all these different functions instead of just looping once? //S.Panda: yeah, thankfully no more. const fileNameElements = Array.from(post.querySelectorAll(".originalNameLink[href]")); const regex = /(\d+)_p\d+/; for (let i = fileNameElements.length-1; i>=0; i--) { const parent = fileNameElements[i].parentElement if (parent.querySelector(".reverseImageSearchDetails")) { return; } let m; if ((m = regex.exec(fileNameElements[i].innerText)) !== null) { parent.insertAdjacentHTML("beforeend", `pixiv`) } } } /*function glowpost() { // Create a frequency map to track occurrences of each item const list = document.querySelectorAll(".labelId"); const countMap = Array.from(list).reduce((acc, item) => { acc[item.style.backgroundColor] = (acc[item.style.backgroundColor] || 0) + 1; return acc; }, {}); // Filter the list to keep only items with a count of 1 Array.from(list).filter(item => countMap[item.style.backgroundColor] === 1).forEach((item) => { item.style.boxShadow = "0 0 15px #26bf47"; item.title = "This is the first post from this ID."; }); }*/ var idMap = {}; const glowpost = function(post, newpost = true) { const list = post.querySelectorAll(".labelId"); const postNumber = post.querySelector(".linkQuote")?.textContent; list.forEach((poster) => { const bgColor = poster.style.backgroundColor; if (newpost && idMap[bgColor] === undefined) { idMap[bgColor] = postNumber; poster.classList.add("glows"); poster.title = "This is the first post from this ID."; } else if (!newpost && idMap[bgColor] == postNumber) { poster.classList.add("glows"); poster.title = "This is the first post from this ID."; } }); } const revealSpoilerImages = function(post) { const spoilers = post.querySelectorAll(".imgLink > img:is([src='/spoiler.png'],[src*='/custom.spoiler'])"); spoilers.forEach(spoiler => { spoiler.classList.add('spoiler-thumb'); 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 = "2px dotted red"; }); } if (settings.spoilerImageType.startsWith("reveal")) { addMyStyle("lynx-reveal-spoilerimage",` img.spoiler-thumb { transition: 0.2s; outline: 2px dotted #ff0000ee; ${settings.spoilerImageType=="reveal_blur" ? "filter: blur(10px);" : ""} } img.spoiler-thumb:hover { filter: blur(0); } `) } // Add functionality to apply the custom spoiler image CSS let threadSpoilerFound = false; let tsFallbackUsed = false; function setThreadSpoiler(post) { if (threadSpoilerFound) return; let spoilerImageUrl = null; //When the option is "threadAlt", fallback to "thread" if "threadAlt" doesn't exist yet. if (settings.spoilerImageType == "threadAlt") { const altSpoilerLink = Array.from(post.querySelectorAll('a.originalNameLink[download^="ThreadSpoilerAlt"]')).find(link => /\.(jpg|jpeg|png|webp)$/i.test(link.download)); spoilerImageUrl = altSpoilerLink ? altSpoilerLink.href : null; tsFallbackUsed = false; //stop looking for threadAlt } if (settings.spoilerImageType == "thread" || (!spoilerImageUrl && !tsFallbackUsed && settings.spoilerImageType == "threadAlt")) { const spoilerLink = Array.from(post.querySelectorAll('a.originalNameLink[download^="ThreadSpoiler"]')).find(link => /\.(jpg|jpeg|png|webp)$/i.test(link.download)); spoilerImageUrl = spoilerLink ? spoilerLink.href : null; if (settings.spoilerImageType == "threadAlt") { tsFallbackUsed = true; //Keep looking for threadAlt } } else if (settings.spoilerImageType == "test") { const myArray = [ 'https://8chan.moe/.media/f76e9657d6b506115ccd0ade73d3d562777a441e4b6bb396610669396ff3032a.png', 'https://8chan.moe/.media/1074fdb6eea4ba609910581e7824106736a1bcad446ace1ae0792b231b52cf9a.png', 'https://8chan.moe/.media/c32b4de8490d7e77987f0e2a381d5935ffa6fec9b98c78ea7c05bd4381d6f64b.png', 'https://8chan.moe/.media/bb225302110d52494ec2bea68693d566acee09767212ce4ee8c0d83d49cfa05b.png' ]; spoilerImageUrl = myArray[Math.floor(Math.random() * myArray.length)]; addMyStyle("lynx-thread-spoiler-css1", ` body { --spoiler-img: url("${spoilerImageUrl}") } .uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]), .uploadCell:not(.expandedCell) a.imgLink:has(>img[src*="/custom.spoiler"]) { background-image: var(--spoiler-img); background-size: cover; & > img { opacity: 0; } } `); threadSpoilerFound = true; return; } if (spoilerImageUrl) { document.head?.querySelector("#lynx-thread-spoiler-css2")?.remove(); //Remove if the style already exists (from fallback) addMyStyle("lynx-thread-spoiler-css2", ` ${settings.overrideBoardSpoilerImage ? '.uploadCell:not(.expandedCell) a.imgLink:has(>img[src*="/custom.spoiler"]),' : "" } .uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]) { background-image: url("${spoilerImageUrl}"); background-size: cover; outline: dashed 2px #ff000090; & > img { opacity: 0; } } `); if (!tsFallbackUsed) { threadSpoilerFound = true; } } } if (settings.spoilerImageType=="kachina") { addMyStyle("lynx-kachinaSpoilers",` ${settings.overrideBoardSpoilerImage ? '.uploadCell:not(.expandedCell) a.imgLink:has(>img[src*="/custom.spoiler"]),' : "" } .uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]) { background-size: cover; margin-right:5px; background-image: url(""); & > img { opacity: 0; } } `) } function iterateAllPosts() { //Get ALL posts (this does NOT include inlined posts and hovered posts) const allPosts = document.querySelectorAll("#divThreads > .opCell > .innerOP, .divPosts > .postCell"); const postsArray = Array.from(allPosts); //use an array to find the last post postsArray.forEach((post, index) => { if (index == postsArray.length-1) { //only the last post sends batching=false iterateSinglePost(post, true, false); } else { iterateSinglePost(post, true, true); } }); } /** * Processes a single post element. * * @param {HTMLElement} post - The post here can be an .innerPost or one of its containers * @param {boolean} newpost - True if this is a new post in the thread (i.e. not a tooltip or inline) * @param {boolean} batching - False if this is not from a batch from iterateAllPosts (or not the last post of the batch) */ function iterateSinglePost(post, newpost = true, batching = false) { // console.log("Lynx-- processing post", {post}, {newpost}, {batching}); indicateCrossLinks(post); addDeletedChecks(post); imageSearchHooks(post); if (settings.glowFirstPostByID) glowpost(post, newpost); if (settings.spoilerImageType.startsWith("reveal")) revealSpoilerImages(post); if (settings.showPostIndex) addPostCount(post, newpost); //Run only if its a new post in the thread if (newpost) { if (settings.spoilerImageType=="thread" || settings.spoilerImageType=="threadAlt") setThreadSpoiler(post); //This still has to iterate all posts, do it last and only when necessary. if (batching === false && settings.showScrollbarMarkers) recreateScrollMarkers(); } } //Start running and observing //At startup, iterate over all posts after a delay // setTimeout(() => { // iterateAllPosts(); // }, 100); //I guess we don't need a delay anymore iterateAllPosts(); //Observe posts and all their children const observer = new MutationObserver((mt_callback) => { mt_callback.forEach(mut => { if (mut.type=="childList" && mut.addedNodes?.length > 0) { //console.log("MutationObserver!!!"); mut.addedNodes.forEach(node => { //New posts, new inlined posts, new hovered posts all contain .innerPost and are always in a div container. //New posts are div.postCell and new inlines are div.inlineQuote if (node.tagName === "DIV" && node.querySelector(".innerPost,.innerOP")) { // console.log("lynx ~ observer:", {node}, {mut}); if (node.classList?.contains("postCell")) { iterateSinglePost(node, true); } else { iterateSinglePost(node, false); } } }); } }) }); observer.observe(document.querySelector(".divPosts"), {childList: true, subtree: true}); //Observe the hover tooltip (ignore everything else) const toolObserver = new MutationObserver((mutationsList) => { for (const mutation of mutationsList) { if (mutation.type === 'childList') { mutation.addedNodes.forEach(node => { if (node.classList?.contains("quoteTooltip")) { //New hover tooltip div.quoteTooltip found iterateSinglePost(node, false); } }); } } }); toolObserver.observe(document.body, {childList: true}); } //End of runAfterDom() //Starting runAfterDom when the document is ready waitForDom(runAfterDom); })();