Greasy Fork

LynxChan Extended Minus Minus

It's like 4chanXT but worse

目前为 2025-04-17 提交的版本。查看 最新版本

// ==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
// ==/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)`
			}
		},
		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 <a href="https://www.dropbox.com/s/n6ys414nviitr9y/MadokaRunes-2.0.ttf">MadokaRunes.ttf</a> 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 = `
			<h3 style="text-align: center; color:#6bc9ff;">LynxChan Extended-- Options</h3><br>
		`;
		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>"
				menu.innerHTML += html;
			} else {

				menu.innerHTML += `
				<label>
					<input type="checkbox" id="${name}" ${settings[name] ? "checked" : ""}>
					${setting.desc}
				</label><br><br>`
			}
		})
		menu.innerHTML += `
			<button id="saveSettings">Save</button>
			<button id="closeMenu">Close</button>
		`
		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", `
		<span>/</span>
		<a id="navigation-lynxextended" class="lynxExtendedSettings" title="LynxChan Extended Settings"
			style="width: 13px;height: 13px;display: inline-block;fill: var(--link-color); vertical-align: middle;margin-left: 1px;"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512">
				<path
					d="M352 320c88.4 0 160-71.6 160-160c0-15.3-2.2-30.1-6.2-44.2c-3.1-10.8-16.4-13.2-24.3-5.3l-76.8 76.8c-3 3-7.1 4.7-11.3 4.7L336 192c-8.8 0-16-7.2-16-16l0-57.4c0-4.2 1.7-8.3 4.7-11.3l76.8-76.8c7.9-7.9 5.4-21.2-5.3-24.3C382.1 2.2 367.3 0 352 0C263.6 0 192 71.6 192 160c0 19.1 3.4 37.5 9.5 54.5L19.9 396.1C7.2 408.8 0 426.1 0 444.1C0 481.6 30.4 512 67.9 512c18 0 35.3-7.2 48-19.9L297.5 310.5c17 6.2 35.4 9.5 54.5 9.5zM80 408a24 24 0 1 1 0 48 24 24 0 1 1 0-48z">
				</path>
			</svg>
		</a>
		`);
		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<posts.length; i++)
		{
			//Already added, ignore
			if (posts[i].querySelector(".postNum")) {
				continue;
			}

			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 revealSpoilerImages = function() {
		const spoilers = document.querySelectorAll(".imgLink > 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);    }}
		`)
	}
})();