LynxChan Extended Minus Minus

LynxChan Extended with even more features

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         LynxChan Extended Minus Minus
// @namespace    https://rentry.org/8chanMinusMinus
// @version      1.53
// @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-idle
// ==/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 <b style="color: #6bc9ff;">"ThreadSpoiler.jpg"</b> (or .png or .webp)`,
				threadAlt:`same as above with the filename <b style="color: #6bc9ff;">"ThreadSpoilerAlt.jpg"</b> (or .png or .webp)`
			}
		},
		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. (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 <a href="https://www.dropbox.com/s/n6ys414nviitr9y/MadokaRunes-2.0.ttf">MadokaRunes.ttf</a> for it to show up properly.`
			}
		},
		useExtraStylingFixes:{
			default:true,
			desc:"Apply some styling fixes (Mark your posts and replies, smaller thumbnails etc.)"
		},
		//I swear this used to be a built in option on 8chan
		halfchanGreentexts:{
			default:false,
			desc:"Make the greentext brighter like 4chan"
		},
		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"
		},
		preserveQuickReply:{
			default:false,
			desc:"Preserve the quick reply text when pressing the Esc key or X button"
		}
		//Would anyone ever want this off?
		/*ctrlEnterPost:{
			default:false,
			desc:"Enable Ctrl+Enter to post your reply"
		}*/
		/*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);
	if (typeof api !== "undefined") {
		console.log("The script is not sandboxed. Adding quick reply shortcut.")
		
		function quickReplyShortcut(ev) {
			if (ev.ctrlKey && ev.key == "r") {
				ev.preventDefault(); qr.showQr(); document.getElementById('qrbody')?.focus();
			};
		}
		document.addEventListener("keydown",quickReplyShortcut);

	} else {
		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 addMyStyle(newID, newStyle) {
		let myStyle = document.createElement("style");
		//myStyle.type = 'text/css';
		myStyle.id = newID;
		myStyle.textContent = newStyle;
		document.querySelector("head").appendChild(myStyle);
	}

	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: #353535;
		border: 1px solid #737373;
		color: #ddd;
		border-radius: 4px;
		max-height:100%;
		overflow-y: auto;
	}
	/*What the fuck is up with CSS */
	/*.settings-content {
		max-height: 90%; 
	}*/
	.settings-footer {
		height:70px;
	}
	@media screen and (max-width: 1000px) {
		#lynxExtendedMenu{
			right:0;
			width:90%;
			/*bottom:15px;*/
		}
	}

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

	// 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;
				}
			}
			//Ctrl+Enter to send reply
			if (ev.key=="Enter") {
				document.getElementById("qrbutton")?.click()
			}
		//I wanted to change this so it propagates upwards for any input but I decided it was kind of pointless.
		//The quick reply field is focused... EXCEPT when you're pressing the button at the bottom. Why??? JUST ALWAYS FOCUS IT
		} else if (ev.key == "Escape") {
			//Because some script managers cannot access the JS of the page we have to do some funny stuff
			//(So far violentmonkey isn't sandboxed)
			document.getElementById("quick-reply").querySelector(".close-btn").click()
		}
	}
	document.getElementById("qrbody").addEventListener("keydown", replyKeyboardShortcuts);

	//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 = "";
		})
	}

	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:#6bc9ff;" 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><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 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);
	}

	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"))
		
		//This goes bottom to top so we stop when we've reached posts
		//that have already had a number inserted
		for (let i = posts.length-1; i>=0; i--)
		{
			//We've reached posts where we already added numbers, 
			// there's no need to keep going.
			if (posts[i].querySelector(".postNum")) {
				break;
			}

			const postInfoDiv = posts[i].getElementsByClassName("title")[0]
			if (!postInfoDiv) {
				console.error("[Lynx--] Failed to find post for div ",posts[i])
				continue;
			}
			const posterNameDiv = postInfoDiv.getElementsByClassName("linkName")[0];
			
			var newNode = document.createElement("span");
			newNode.innerText = i+1;
			newNode.className="postNum"
			if (i < 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);
			var foo = document.createTextNode("\u00A0");
			postInfoDiv.insertBefore(foo, posterNameDiv);
		}
	}

	const indicateCrossLinks = function() {
		const crossLinks = document.querySelectorAll(`a.quoteLink:not(.crossThread):not([href*='${api.boardUri}/res/${api.threadId}'])`);
		crossLinks.forEach(crossLink => {
			crossLink.classList.add("crossThread")
			//This is kind of an expensive operation isn't it? Plus checking :not(.crossThread)? Oh well who cares
			const hrefTokens = crossLink.href.split("#");
			const quoteLinkId = hrefTokens[1];
			crossLink.innerHTML = ">>" + quoteLinkId;
		});
	}
	function addDeletedChecks() {
		const postLinks = document.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
					//postLinks[i].removeEventListener("mouseenter",evListener)
				}
			}
			postLinks[i].addEventListener("mouseenter", evListener);
			//Why does js allow this
			postLinks[i].hasMouseOverEvent = true;
		}
	}

	addMyStyle("linkHelpers",`
		.quoteLink.crossThread::after {
			content: " \(Cross-thread\)";
		}
		.quoteLink.deleted::after {
			content: " \(Deleted\)";
		}
	`)


	const revealSpoilerImages = function() {
		const spoilers = document.querySelectorAll(".imgLink > img:is([src='/spoiler.png'],[src*='/custom.spoiler'])");
		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";
		});
	}

	const executePostModifiers = function() {
		indicateCrossLinks();
		addDeletedChecks();
		// Recreate markers because the page grew taller. Is this heavy? probably not.
		if (settings.showScrollbarMarkers)
			recreateScrollMarkers();
		if (settings.showPostIndex)
			addPostCount();
		if (settings.spoilerImageType=="reveal")
			revealSpoilerImages();
	}

	// Create markers 1 second after page load
	setTimeout(() => {
		executePostModifiers();
	}, 1500);

	const observer = new MutationObserver((mt_callback) => {
		mt_callback.forEach(mut => {
			if (mut.type=="childList") {
				//console.log("MutationObserver!!!");
				executePostModifiers();
			}
		})
	})
	observer.observe(document.querySelector(".divPosts"), {'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; object-fit:contain }

			.uploadCell .imgLink {
				margin-right: 1.5em;
			}

			/* smaller post spacing (not too much) */
			.divMessage {
				margin: .8em .8em .5em 3em;
			}

			/* 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.halfchanGreentexts) {
		addMyStyle("halfchanGreentexts",
			`.greenText {
				filter: brightness(110%);
			}`
		);
	}
	
	if (settings.showStubs === false) {
		addMyStyle("hide-stubs",`
		.postCell:has(> span.unhideButton.glowOnHover) {
			display: none;
		}
		`);
	}

	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", `
				${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;
					}
				}
			`);
		}
	}
	else if (settings.spoilerImageType=="kachina") {
		addMyStyle("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: 1;        
					transform: translate(0, -25%) scale(0.5);    
				}
			}
		`)
	}
})();