// ==UserScript==
// @name LynxChan Extended Minus Minus
// @namespace https://rentry.org/8chanMinusMinus
// @version 1.32
// @description It's like 4chanXT but worse
// @author SaddestPanda & Dandelion & /gfg/
// @license UNLICENSE
// @match *://8chan.moe/*/res/*
// @match *://8chan.se/*/res/*
// @match *://8chan.cc/*/res/*
// @grant GM.getValue
// @grant GM.setValue
// @grant GM.deleteValue
// @grant GM.registerMenuCommand
// @run-at document-idle
// @downloadURL none
// ==/UserScript==
(async function () {
"use strict";
const SETTINGS_DEFINITIONS = {
firstRun:{
default:true,
hidden:true,
desc:"You shouldn't be able to see this setting! (firstRun)"
},
showScrollbarMarkers:{
default:true,
desc:"Show your posts and replies on the scrollbar"
},
spoilerImageType:{
default:"off",
desc:"Override how the spoiler thumbnail looks",
type:"radio",
options:{
off:"Don't change the thumbnail.",
reveal:"Reveal spoilers. Previously spoilered images will have a red border around them indicating that they're spoilers.",
kachina:"Makes the spoiler image Kachina from Genshin Impact.",
thread:`Uses the first image of the first visible post on the current thread with the filename "ThreadSpoiler.jpg" (or .png or .webp)`,
threadAlt:`same as above with the filename "ThreadSpoilerAlt.jpg" (or .png or .webp)`
}
},
useExtraStylingFixes:{
default:true,
desc:"Apply some styling fixes (Mark your posts and replies, smaller thumbnails etc.)"
},
revealSpoilerText:{
default:"off",
desc:"Reveal the spoiler text. Or make it into madoka runes.",
type:"radio",
options:{
off:"Don't reveal spoilers.",
on:"Spoilers will be shown by turning the text white.",
madoka:`Spoilers will turn into madoka runes. Please install MadokaRunes.ttf for it to show up properly.`
}
},
showPostIndex:{
default:true,
desc:"Show the current index of a post on the thread. That is, the topmost post will start at 1 and count up from there."
},
/*showStubs:{
default:true,
desc:"Show post stubs when filtering."
},
redirectToCatalog:{
default:false,
desc:"Redirect to catalog when clicking on the index."
}*/
}
const settingsNames = Object.keys(SETTINGS_DEFINITIONS);
const settingsValues = await Promise.all(settingsNames.map(key => GM.getValue(key, SETTINGS_DEFINITIONS[key]['default'])));
const settings = Object.fromEntries(settingsNames.map((key, index) => [key, settingsValues[index]]));
console.log("%cLynx Minus Minus Started with settings:", "color:rgb(0, 140, 255)", settings);
addMyStyle("lynx-extended-css", `
.marker-container {
position: fixed;
top: 16px;
right: 0;
width: 10px;
height: calc(100vh - 40px);
z-index: 11000;
pointer-events: none;
}
.marker {
position: absolute;
width: 100%;
height: 6px;
background: #0092ff;
cursor: pointer;
pointer-events: auto;
border-radius: 40% 0 0 40%;
z-index: 5;
}
.marker.alt {
background: #a8d8f8;
z-index: 2;
}
#lynxExtendedMenu {
position: fixed;
top: 15px;
right: 100px;
padding: 10px;
z-index: 10000;
font-family: Arial, sans-serif;
font-size: 14px;
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.5);
background: #353535;
border: 1px solid #737373;
color: #ddd;
border-radius: 4px;
}
`);
// Register menu command
GM.registerMenuCommand("Show Options Menu", openMenu);
try {
createSettingsButton();
} catch (error) {
console.log("Error while creating settings button:", error);
}
//Open the settings menu on the first run
if (settings.firstRun) {
settings.firstRun = false;
await GM.setValue("firstRun", settings.firstRun);
openMenu();
}
function replyKeyboardShortcuts(ev) {
if (ev.ctrlKey) {
let combinations = {
"s":["[spoiler]","[/spoiler]"],
"b":["'''","'''"],
"u":["__","__"],
"i":["''","''"],
"d":["[doom]","[/doom]"],
"m":["[moe]","[/moe]"]
}
for (var key in combinations)
{
if (ev.key == key)
{
ev.preventDefault();
console.log("ctrl+"+key+" pressed in textbox")
const textBox = ev.target;
let newText = textBox.value;
const tags = combinations[key]
const selectionStart = textBox.selectionStart
const selectionEnd = textBox.selectionEnd
if (selectionStart == selectionEnd) { //If there is nothing selected, make empty tags and center the cursor between it
document.execCommand("insertText",false, tags[0] + tags[1]);
//Center the cursor between tags
textBox.selectionStart = textBox.selectionEnd = (textBox.selectionEnd - tags[1].length);
} else {
//Insert text and keep undo/redo support (Only replaces highlighted text)
document.execCommand("insertText",false, tags[0] + newText.slice(selectionStart, selectionEnd) + tags[1])
}
return;
}
}
} else if (ev.key == "Escape") {
//Because greasemonkey cannot access the JS of the page we have to do some funny stuff
document.getElementById("quick-reply").querySelector(".close-btn").click()
}
}
document.getElementById("qrbody").addEventListener("keydown", replyKeyboardShortcuts);
// Create markers 1 second after page load
setTimeout(() => {
recreateScrollMarkers();
}, 1500);
if (settings.showPostIndex) {
setTimeout(() => {
addPostCount();
}, 1400);
}
function openMenu() {
const oldMenu = document.getElementById("lynxExtendedMenu");
if (oldMenu) {
oldMenu.remove();
return;
}
// Create options menu
const menu = document.createElement("div");
menu.id = "lynxExtendedMenu";
menu.innerHTML = `
LynxChan Extended-- Options
`;
Object.keys(SETTINGS_DEFINITIONS).forEach((name) => {
const setting = SETTINGS_DEFINITIONS[name];
if (setting.hidden) {
//pass
}
else if (setting.type == "radio") {
let html = `${setting.desc} "
menu.innerHTML += html;
} else {
menu.innerHTML += `
${setting.desc}
`
}
})
menu.innerHTML += `
Save
`
document.body.appendChild(menu);
// Save button functionality
document.getElementById("saveSettings").addEventListener("click", async () => {
Object.keys(SETTINGS_DEFINITIONS).forEach((name) => {
const setting = SETTINGS_DEFINITIONS[name];
if (!('hidden' in setting)) {
if (setting.type=="radio") {
settings[name] = document.querySelector(`input[name="${name}"]:checked`).value
} else {
settings[name] = document.getElementById(name).checked;
}
}
})
console.log("Saving settings ",settings)
await Promise.all(Object.entries(settings).map(([key, value]) => GM.setValue(key, value)));
alert("Settings saved!\nRefresh the page for the changes to take effect.");
menu.remove();
});
// Close button functionality
document.getElementById("closeMenu").addEventListener("click", () => {
menu.remove();
});
}
function createSettingsButton() {
document.querySelector("#navLinkSpan > .settingsButton").insertAdjacentHTML("afterend", `
/
`);
document.querySelector("#navigation-lynxextended").addEventListener("click", openMenu);
}
function addMyStyle(newID, newStyle) {
let myStyle = document.createElement("style");
//myStyle.type = 'text/css';
myStyle.id = newID;
myStyle.textContent = newStyle;
document.querySelector("head").appendChild(myStyle);
}
function createMarker(element, container, isReply) {
const pageHeight = document.body.scrollHeight;
const offsetTop = element.offsetTop;
const percent = offsetTop / pageHeight;
const marker = document.createElement("div");
marker.classList.add("marker");
if (isReply) {
marker.classList.add("alt");
}
marker.style.top = `${percent * 100}%`;
marker.dataset.postid = element.id;
marker.addEventListener("click", () => {
let elem = element?.previousElementSibling || element;
elem.scrollIntoView({ behavior: "smooth", block: "start" });
});
container.appendChild(marker);
}
function recreateScrollMarkers() {
let oldContainer = document.querySelector(".marker-container");
if (oldContainer) {
oldContainer.remove();
}
// Create marker container
const markerContainer = document.createElement("div");
if (settings.showScrollbarMarkers) {
markerContainer.classList.add("marker-container");
document.body.appendChild(markerContainer);
}
// Match and create markers for "my posts" (matches native & dollchan)
document.querySelectorAll(".postCell:has(.innerPost > .postInfo.title :is(.linkName.youName, .de-post-counter-you)),.postCell:has(.innerPost.de-mypost)")
.forEach((elem) => {
createMarker(elem, markerContainer, false);
});
// Match and create markers for "replies" (matches native & dollchan)
document.querySelectorAll(".postCell:has(.innerPost > .divMessage :is(.quoteLink.you, .de-ref-you)),.postCell:has(.innerPost.de-mypost-reply)")
.forEach((elem) => {
createMarker(elem, markerContainer, true);
});
}
function addPostCount() {
//This function causes a DOMException, I don't know why, just ignore it
const posts = Array.from(document.getElementsByClassName("divPosts")[0].children);
//Why is the insert method called unshift???? This inserts it at the beginning
//(This is also insanely inefficient since we only need to do it once)
posts.unshift(document.querySelector(".innerOP"))
for (let i=0; i img[src='/spoiler.png']");
spoilers.forEach(spoiler => {
const parent = spoiler.parentElement;
const hrefTokens = parent.href.split("/");
const fileNameTokens = hrefTokens[4].split(".");
const thumbUrl = `/.media/t_${fileNameTokens[0]}`;
spoiler.src = thumbUrl;
spoiler.style.border = "thin dotted red";
spoiler.style.borderWidth = "2px";
});
}
if (settings.showScrollbarMarkers || settings.showPostIndex) {
const observer = new MutationObserver((mt_callback) => {
mt_callback.forEach(mut => {
if (mut.type=="childList") {
//console.log("MutationObserver!!!");
// Recreate markers because the page grew taller. Is this heavy? probably not.
recreateScrollMarkers();
if (settings.showPostIndex)
addPostCount();
if (settings.spoilerImageType=="reveal")
revealSpoilerImages();
}
})
})
observer.observe(document.querySelector(".divPosts"), {'childList':true})
// I'm not sure why but this doesn't work
// // Add a second observer for #threadList (new posts)
// const threadObserver = new MutationObserver((mutationsList) => {
// for (const mutation of mutationsList) {
// if (mutation.type === 'childList') {
// mutation.addedNodes.forEach((node) => {
// if (node.classList && node.classList.contains("postCell")) {
// console.log("ThreadObverver!!!")
// // Recreate markers because the page grew taller. Is this heavy? probably not.
// recreateScrollMarkers();
// if (settings.showPostIndex)
// addPostCount();
// if (settings.spoilerImageType=="reveal")
// revealSpoilerImages();
// }
// });
// }
// }
// });
// const threadList = document.querySelector("#threadList");
// if (threadList) {
// threadObserver.observe(threadList, { childList: true });
// }
}
// Apply the CSS if the setting is enabled
if (settings.useExtraStylingFixes) {
addMyStyle("extra-styling-css", `
/* smaller thumbnails & image paddings */
body .uploadCell img:not(.imgExpanded) {
max-width: 160px;
max-height: 125px;
object-fit: contain;
height: auto;
width: auto;
margin-right: 0em;
margin-bottom: 0em;
}
.imgExpanded { max-height:100vh; }
.uploadCell .imgLink {
margin-right: 1.5em;
}
/* smaller post spacing (not too much) */
.divMessage {
margin: .8em .8em .5em 3em;
}
/*.greenText {
filter: brightness(110%);
}*/
/* Make your name in your post red */
.youName { color: red; }
.you { --link-color: red; }
/* mark your posts and replies (same selectors are also used for detection above) */
.postCell:has(.innerPost > .postInfo.title :is(.linkName.youName, .de-post-counter-you)),
.postCell:has(.innerPost.de-mypost) {
& > .innerPost {
border-left: 3px dashed;
border-left-color: #4BB2FFC2;
padding-left: 0px;
}
}
.postCell:has(.innerPost > .divMessage :is(.quoteLink.you, .de-ref-you)),
.postCell:has(.innerPost.de-mypost-reply) {
& > .innerPost {
border-left: 2px solid;
border-left-color:rgb(0, 102, 255);
padding-left: 1px;
}
}
`);
}
if (settings.revealSpoilerText=="on") {
addMyStyle("reveal-spoilers",`
.span.spoiler { color: white}
`)
} else if (settings.revealSpoilerText="madoka") {
addMyStyle("reveal-spoilers",`
span.spoiler:not(:hover) {
color: white;
font-family:MadokaRunes!important;
}
`)
}
// Add functionality to apply the custom spoiler image CSS
if (settings.spoilerImageType=="thread" || settings.spoilerImageType=="threadAlt") {
let spoilerImageUrl = null;
if (settings.spoilerImageType=="thread") {
const spoilerLink = Array.from(document.querySelectorAll('a.originalNameLink[download^="ThreadSpoiler"]')).find(link => /\.(jpg|png|webp)$/i.test(link.download));
spoilerImageUrl = spoilerLink ? spoilerLink.href : null;
} else if (settings.spoilerImageType=="threadAlt") {
const altSpoilerLink = Array.from(document.querySelectorAll('a.originalNameLink[download^="ThreadSpoilerAlt"]')).find(link => /\.(jpg|png|webp)$/i.test(link.download));
spoilerImageUrl = altSpoilerLink ? altSpoilerLink.href : null;
}
if (spoilerImageUrl) {
addMyStyle("thread-spoiler-css", `
.uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]) {
background-image: url("${spoilerImageUrl}");
background-size: cover;
outline: dashed 2px #ff0000f5;
& > img[src="/spoiler.png"] {
opacity: 0;
}
}
`);
}
}
else if (settings.spoilerImageType=="reveal") {
revealSpoilerImages();
}
else if (settings.spoilerImageType=="kachina") {
addMyStyle("kachinaSpoilers",`
.uploadCell:not(.expandedCell) a.imgLink:has(>img[src="/spoiler.png"]) { background-size: cover; margin-right:5px; background-image: url(""); & > img[src="/spoiler.png"] { opacity: 1; transform: translate(0, -25%) scale(0.5); }}
`)
}
})();