LynxChan Extended Minus Minus

LynxChan Extended with even more features

当前为 2025-04-27 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         LynxChan Extended Minus Minus
// @namespace    https://rentry.org/8chanMinusMinus
// @version      2.0.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
// ==/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,
			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. <span class='altText'>Previously spoilered images will have a red border around them indicating that they're spoilers.</span>",
				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 <b style="color: var(--link-color);">"ThreadSpoiler.jpg"</b> from the current thread. <span class="altText">(first posted jpg, png or webp image with that filename)</span>`,
				threadAlt:`same as above with the filename <b style="color: var(--link-color);">"ThreadSpoilerAlt.jpg"</b> <span class="altText">(.jpg, .png or .webp)</span>`,
				//test:`[TEST OPTION] Randomly pick spoiler image from /gacha/ board <span class='altText'>(This is a test option. It selects the spoiler from var(--spoiler-img) after setting.)</span>`
			},
			nonewline:true
		},
		overrideBoardSpoilerImage: {
			default:true,
			parent:"spoilerImageType",
			//Not implemented yet
			//depends: function() {return settings.spoilerImageType != "off"},
			desc:"Override spoiler thumbnail even if the board has a custom thumbnail set. <span class='altText'>(For example, /v/'s spoiler thumbnail is an image of a monitor with a ? inside it)</span>"
		},
		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 <a href="https://www.dropbox.com/s/n6ys414nviitr9y/MadokaRunes-2.0.ttf"><u>MadokaRunes.ttf</u></a> for it to show up properly.`
			}
		},
		markYourPosts:{
			default:true,
			desc:"<span class='boldText'>Style:</span> Mark your posts and replies <span class='altText'>(with a left border)</span>",
			nonewline:true
		},
		compactPosts:{
			default:true,
			desc:"<span class='boldText'>Style:</span> Make thumbnails and posts more compact",
			nonewline:true
		},
		showStubs:{
			default:true,
			desc:"<span class='boldText'>Style:</span> Show post stubs when filtering",
			nonewline:true
		},
		//I swear this used to be a built in option on 8chan
		halfchanGreentexts:{
			default:false,
			desc:"<span class='boldText'>Style:</span> Make the greentext brighter like 4chan"
		},
		glowFirstPostByID:{
			default:true,
			desc:"Mark new/unique posters by adding a glow effect to their ID"
		},
		showPostIndex:{
			default:true,
			desc:"Show the current index of a post on the thread. <span class='altText'>That is, the topmost post will start at 1 and count up from there.</span>"
		},
		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);
	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]]));

	function addMyStyle(newID, newStyle) {
		let myStyle = document.createElement("style");
		//myStyle.type = 'text/css';
		myStyle.id = newID;
		myStyle.textContent = newStyle;
		document.head.appendChild(myStyle);
	}

	// 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.markYourPosts = false;
			settings.compactPosts = false;
			await GM.setValue("markYourPosts", false);
			await GM.setValue("compactPosts", false);
		}
		// Remove the old setting
		await GM.deleteValue("useExtraStylingFixes");
	}

	//Apply all the styles as soon as possible
	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;
		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.75;
		}

		& .boldText {
			color: var(--link-color);
			font-weight: bold;
		}

		button {
			padding: 8px 16px;
			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:70px;
	}
	@media screen and (max-width: 1000px) {
		#lynxExtendedMenu{
			right:0;
			width:90%;
			/*bottom:15px;*/
		}
	}

	.lynxExtendedButton::before {
		content: "\\e0da";
	`);

	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;
			}
		`);
	}

	if (settings.markYourPosts) {
		addMyStyle("lynx-mark-posts", `
			/* Make your name in your post red */
			.youName { color: red; }
			.you { --link-color: red; }

			/* mark your posts and replies */
			#divThreads .postCell > .innerPost:has(> .postInfo.title > .youName) {
					border-left: 3px dashed #4BB2FFC2;
					padding-left: 1px;
			}

			#divThreads .postCell > .innerPost:has(> .divMessage .quoteLink.you) {
					border-left: 2px solid rgb(0, 102, 255);
					padding-left: 1px;
			}
		`);
	}

	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;
			}
		`)
	}

	//Everything in afterReady runs after document has loaded (like @run-at document-end)
	//Everything in afterReady runs after document has loaded (like @run-at document-end)
	//Everything in afterReady runs after document has loaded (like @run-at document-end)
	async function afterReady() {
		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", `
			<span>/</span>
			<a id="navigation-lynxextended" class="coloredIcon lynxExtendedButton" title="LynxChan Extended-- Settings"></a>
			`);
			//Mobile
			document.querySelector("#sidebar-menu > ul > li > .settingsButton").parentElement.insertAdjacentHTML("afterend", `
				<li>
					<a id="navigation-lynxextended-mobile" class="coloredIcon lynxExtendedButton" title="LynxChan Extended-- Settings">Lynx Ex-- Settings</a>
				</li>
			`);
			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();
				document.getElementById("qrbody").value = "";
				document.getElementById('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 = `<h3 style="text-align: center; color: var(--subject-color);" class='settings-header'>LynxChan Extended-- Options</h3><br>`;
			
			//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 = `<span>${setting.desc}</span><br><form id="${name}" action='#'>`
					for (const [value, description] of Object.entries(setting.options)) {
						html += `
						<label>
							<input name="${name}" type="radio" value="${value}" ${settings[name]==value ? "checked" : ""}
							<span>${description}</span>
						</label><br>
						`
					}
					html += "</form><br>"
					settings_content.innerHTML += html;
				} else {
	
					settings_content.innerHTML += `
					<label>
						<input type="checkbox" id="${name}" ${settings[name] ? "checked" : ""}>
						${setting.desc}
					</label><br>${setting.nonewline ? '' : '<br>'}`
				}
			})
			menu.appendChild(settings_content);
			menu.innerHTML += `
				<div class='settings-footer'>
					<button id="saveSettings">Save</button>
					<button id="closeMenu">Close</button>
				</div>
			`
			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 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");
			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;

			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";
			if (localCount < Infinity) //knownBumpLimit
			{
				newNode.style = "color: rgb(123, 59, 200); font-weight: bold;"
			}
			else
			{
				newNode.style = "color: rgb(255, 4, 4); font-weight: bold;"
			}
			postInfoDiv.insertBefore(newNode, posterNameDiv);
			let foo = document.createTextNode("\u00A0");
			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", `<span class='reverseImageSearchDetails'><a href="https://pixiv.net/i/${m[1]}">pixiv</a></span>`)
				}
			}
		}
	
		/*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.style.boxShadow = "0 0 15px #26bf47";
					poster.title = "This is the first post from this ID.";
				} else if (!newpost && idMap[bgColor] == postNumber) {
					//inlines and hovers check the postID
					poster.style.boxShadow = "0 0 15px #26bf47";
					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;
					border: 2px dotted red;
					${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;
		function setThreadSpoiler(post) {
			if (threadSpoilerFound) return;
	
			let spoilerImageUrl = null;
	
			if (settings.spoilerImageType=="thread") {
				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;
			} else 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;
			} 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) {
				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 #ff0000f5;
						& > img {
							opacity: 0;
						}
					}
				`);
				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});

	} //afterReady close

	//Starting afterReady
	if (document.readyState === "loading") {
		//Loading hasn't finished yet. Wait for the inital document to load and start.
		document.addEventListener("DOMContentLoaded", afterReady);
	} else {
		//Document has already loaded. Start.
		afterReady();
	}
})();