// ==UserScript==
// @name LynxChan Extended Minus Minus
// @namespace https://rentry.org/8chanMinusMinus
// @version 2.1.1
// @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 https://update.greasyfork.icu/scripts/533169/LynxChan%20Extended%20Minus%20Minus.user.js
// @updateURL https://update.greasyfork.icu/scripts/533169/LynxChan%20Extended%20Minus%20Minus.meta.js
// ==/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;
filter: drop-shadow(0px 0px 1px #000000BA);
}
.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", `
`);
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} ${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;
background-position: center;
& > 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;
background-position: center;
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;
background-position: center;
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);
})();