LynxChan Extended Minus Minus

LynxChan Extended with even more features

Instalar este script¿?
Script recomendado por el autor

Puede que también te guste 8chan-buffs.

Instalar este script
// ==UserScript==
// @name         LynxChan Extended Minus Minus
// @namespace    https://rentry.org/8chanMinusMinus
// @version      2.2.4
// @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==

//TODO LATER MAYBE: combine all CSS into one <style> and use classes on html or body instead.

(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:"<b>Your marker:</b>",
            color2Default:"#a8d8f8",
            color2Desc:"<b>Reply marker:</b>"
        },
        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 lineBefore'>(Previously spoilered images will have a red border around them indicating that they're spoilers.)</span>",
                reveal_blur:"Change to a blurred thumbnail <span class='altText lineBefore'>(Unblurred when you hover your mouse over.)</span>",
                kachina:"Makes the spoiler image Kachina from Genshin Impact.",
                thread:`<b>Use <b style="color: var(--link-color);">"ThreadSpoiler.jpg"</b> from the current thread <span class="altText lineBefore">(first posted jpg, png or webp image with that filename)</span></b>`,
                threadAlt:`same as above with the filename <b style="color: var(--link-color);">"ThreadSpoilerAlt.jpg"</b> <span class="altText lineBefore">(jpg, png or webp; uses ThreadSpoiler.jpg until this is found)</span>`,
                //test:`[TEST OPTION] Set custom spoiler thumb per-thread (For /gacha/ only!)`
            },
            nonewline:true
        },
        overrideBoardSpoilerImage: {
            default:true,
            parent:"spoilerImageType",
            //Not implemented yet
            //depends: function() {return settings.spoilerImageType != "off"},
            desc:"Also override board's custom thumbnail image <span class='altText lineBefore'>(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.`
            }
        },
        markPostEdge:{
            default:true,
            type:"checkbox_with_colors",
            desc:"<span class='boldText'>Style:</span> Mark your posts and replies <span class='altText'>(with a left border)</span>",
            color1Default:"#4BB2FF",
            color1Desc:"<b>Your border:</b>",
            color2Default:"#0066ff",
            color2Desc:"<b>Reply border:</b>",
            nonewline:true
        },
        markYouText:{
            default:true,
            type:"checkbox_with_colors",
            desc:"<span class='boldText'>Style:</span> Color your name and (You) links",
            color1Default:"#ff2222",
            color1Desc:"<b>Color:</b>",
            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,
            type:"checkbox_with_colors",
            desc:"Mark new/unique posters by adding a glow effect to their ID",
            color1Default:"#26bf47",
            color1Desc:"<b>Glow color:</b>"
        },
        showPostIndex:{
            default:true,
            type:"checkbox_with_colors",
            desc:"Show the current index of a post on the thread. <span class='altText'>(OP: 1, first post: 2 etc.)</span>",
            color1Default:"#7b3bcc",
            color1Desc:"<b>Index color:</b>"
        },
        preserveQuickReply:{
            default:false,
            desc:"Preserve the quick reply text when closing the box or refreshing the page"
        },
        /*preserveName:{
            default:false,
            desc:"Preserve the last used name when refreshing the page"
        },*/
        reverseSearchBooruSite:{
            desc:"Booru to link to when a valid md5 is found in an attachment filename",
            type:"dropdown",
            default:"gelbooru",
            choices:{
                "gelbooru":"https://gelbooru.com/index.php?page=post&s=list&tags=md5%3a",
                "danbooru":"https://danbooru.donmai.us/posts?tags=md5%253A",
                "safebooru":"https://safebooru.org/index.php?page=post&s=list&tags=md5%3a"
            }
        }
        /*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:90%;
            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;
                margin-bottom: 0;
                filter: contrast(115%) brightness(110%);
                &:hover {
                    filter: brightness(130%);
                }
            }
        }
        #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", `
                /* 
                README:
                Mark your posts and replies with a left border. Specificity order: (you) > (reply). 
                Important: The :not(#SP1) selectors are used for extra specificity.
                These are made extra specific so we can override ones from other userscripts.
                (because Lynx-- has an option to disable only this and also has the ability to customize the color)
                */
                /* Match your posts. This is easy. */
                body:not(#SP1#SP1) .innerPost.yourPost {
                    border-left: 3px dashed var(--markPostEdge_color1, ${color1});
                }

                /* 
                * Match replies:
                * This can be a simple .divMessage > .quoteLink 
                * or it can be a .divMessage > details > .spoiler > s > u > .quoteLink (or something like that)
                */
                body:not(#SP1) .innerPost.replyPost {
                    border-left: 2px solid var(--markPostEdge_color2, ${color2});
                }
            `);
        }

        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", `
            <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();
                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 = `
            <h3 style="text-align: center; color: var(--subject-color);" class='settings-header'>LynxChan Extended-- Options</h3>
            <p style="text-align: center;">Version ${GM.info.script.version}</p><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>${setting.nonewline ? '' : '<br>'}`;
                    settings_content.innerHTML += html;
                } else if (setting.type == "dropdown") {
                    let html = `<label for="${name}">${setting.desc}:</label><select id="${name}">`
                    Object.keys(setting['choices']).forEach(value => {
                        html+=`<option value="${value}" ${settings[name]==value ? "selected" : ""}>${value}</option>`
                    })
                    html+=`</select>`
                    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 += `
                        <label style="margin-left:0.5em;">
                            ${colorDesc}
                            <input type="color" id="${name}_color${idx}" value="${colorValue}" ${settings[name] ? '' : 'disabled'}>
                        </label>
                        `;
                    });
                    settings_content.innerHTML += `
                    <label>
                        <input type="checkbox" id="${name}" ${settings[name] ? "checked" : ""}>
                        ${setting.desc}
                    </label>
                    ${colorHtml}
                    <br>${setting.nonewline ? '' : '<br>'}`;
                } 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>
                    <button id="resetSettings" style="float: right;">Reset</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] = menu.querySelector(`input[name="${name}"]:checked`).value;
                        } else if (setting.type=="dropdown") {
                            settings[name] = document.getElementById(name).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_md5sum = /[0-9a-f]{32}/g;
            const regex_pixiv = /(\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_pixiv.exec(fileNameElements[i].innerText)) !== null) {
                    parent.insertAdjacentHTML("beforeend", `<span class='reverseImageSearchDetails'><a href="https://pixiv.net/i/${m[1]}">pixiv</a></span>`)
                }
                //Careful with this insane abuse of conditionals, the order of operations matters (&& is before = without parenthesis)
                //And we don't want to match >1 because that could be an 8chan hash (There should only be 1 md5 hash in a file name anyways)
                else if ((m = [...fileNameElements[i].innerText.matchAll(regex_md5sum)]) && m?.length == 1) {
                    parent.insertAdjacentHTML("beforeend", `<span class='reverseImageSearchDetails'><a href="${SETTINGS_DEFINITIONS['reverseSearchBooruSite']['choices'][settings.reverseSearchBooruSite]}${m[0]}">${settings.reverseSearchBooruSite}</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.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);
                }
            });
        }

        /**
         * Adds .yourPost to your posts and adds .replyPost to reply posts. 
         * Works on new posts, tooltip posts, and inline posts. 
         * 
         * .yourPost: if .innerPost:has(> .postInfo.title > .youName)
         * 
         * .replyPost: if .innerPost:has(> .divMessage .quoteLink.you)
         */
        function setPostType(node) {
            const inner = node.querySelector(":scope > .innerPost");
            if (!inner) return;

            // Check for your post
            if (inner.querySelector(":scope > .postInfo.title > .youName")) {
                inner.classList.add("yourPost");
            } else if (inner.querySelector(":scope > .divMessage .quoteLink.you")) {
                inner.classList.add("replyPost");
            } else {
                //Testing: mark hovered posts as replies (missing from native js)
                try {
                    if (node.classList.contains("quoteTooltip") && posting.yous) {
                        const quoteLinks = inner.querySelectorAll(".quoteLink");
                        for (const qlink of quoteLinks) {
                            const extractedNumber = parseInt(qlink.textContent.replace(/[^\d]/g, ''), 10);
                            if (!isNaN(extractedNumber) && posting.yous.includes(extractedNumber)) {
                                qlink.classList.add("you");
                                inner.classList.add("replyPost");
                                // break;
                            }
                        }
                    }
                } catch (error) {
                    console.log("Lynx-- ~ setPostType ~ error when trying to parse tooltip:", error);
                }
            }
        }

        /**
         * 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.markPostEdge)
                setPostType(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.
                //These are manually ran in the mutation observer
                if (batching === false && settings.showScrollbarMarkers)
                    recreateScrollMarkers();
            }
        }

        //Start running and observing, no need for a delay.
        iterateAllPosts();
        //Observe posts and all their children
        const observer = new MutationObserver((mt_callback) => {
            let foundNewPost = false;
            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")) {
                                foundNewPost = true;
                                iterateSinglePost(node, true, true); //batching=true for both
                            } else {
                                iterateSinglePost(node, false, true);
                            }
                        }
                    });
                }
            });
            //Manually run all batching=false actions here
            if (foundNewPost && settings.showScrollbarMarkers) {
                recreateScrollMarkers();
            }
        });
        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);
})();