4chan Gallery

4chan grid-based image gallery with zoom mode support for threads that allows you to browse images, and soundposts (images with sounds, webms with sounds) along with other utility features.

// ==UserScript==
// @name         4chan Gallery
// @namespace    http://tampermonkey.net/
// @version      2025-01-12 (3.6)
// @description  4chan grid-based image gallery with zoom mode support for threads that allows you to browse images, and soundposts (images with sounds, webms with sounds) along with other utility features.
// @author       TheDarkEnjoyer
// @match        https://boards.4chan.org/*/thread/*
// @match        https://boards.4chan.org/*/archive
// @match        https://boards.4channel.org/*/thread/*
// @match        https://boards.4channel.org/*/archive
// @match        https://warosu.org/*/*
// @match        https://archived.moe/*/*
// @match        https://archive.palanq.win/*/*
// @match        https://archive.4plebs.org/*/*
// @match        https://desuarchive.org/*/*
// @match        https://thebarchive.com/*/*
// @match        https://archiveofsins.com/*/*
// @icon         
// @grant        none
// @license      GNU GPLv3
// ==/UserScript==

(function () {
    "use strict";
    // injectVideoJS();
    const defaultSettings = {
        Load_High_Res_Images_By_Default: {
            value: false,
            info: "When opening the gallery, load high quality images by default (no thumbnails)",
        },
        Add_Placeholder_Image_For_Zoom_Mode: {
            value: true,
            info: "Add a placeholder image for zoom mode so even if the thread has no images, you can still open the zoom mode",
        },
        Play_Webms_On_Hover: {
            value: true,
            info: "Autoplay webms on hover, pause on mouse leave",
        },
        Switch_Catbox_To_Pixstash_For_Soundposts: {
            value: false,
            info: "Switch all catbox.moe links to pixstash.moe links for soundposts",
        },
        Show_Arrow_Buttons_In_Zoom_Mode: {
            value: true,
            info: "Show clickable arrow buttons on screen edges in zoom mode",
        },
        Grid_Columns: {
            value: 3,
            info: "Number of columns in the grid view",
        },
        Grid_Cell_Max_Height: {
            value: 200,
            info: "Maximum height of each cell in pixels",
        },
        Embed_External_Links: {
            value: false,
            info: "Embed catbox/pixstash links found in post comments",
        },
        Strictly_Load_GIFs_As_Thumbnails_On_Hover: {
            value: false,
            info: "Only load GIF thumbnails until hovered"
        },
        Open_Close_Gallery_Key: {
            value: "i",
            info: "Key to open/close the gallery"
        },
        Hide_Gallery_Button: {
            value: false,
            info: "Hide the gallery button (You can still open the gallery with the keybind, default is 'i')"
        },
    };

    let threadURL = window.location.href;
    let lastScrollPosition = 0;
    let gallerySize = { width: 0, height: 0 };
    let gridContainer; // Add this line

    // store settings in local storage
    if (!localStorage.getItem("gallerySettings")) {
        localStorage.setItem("gallerySettings", JSON.stringify(defaultSettings));
    }
    let settings = JSON.parse(localStorage.getItem("gallerySettings"));

    // check if settings has all the keys from defaultSettings, if not, add the missing keys
    let missingSetting = false;
    for (const setting in defaultSettings) {
        if (!settings.hasOwnProperty(setting)) {
            settings[setting] = defaultSettings[setting];
            missingSetting = true;
        }
    }

    // update the settings in local storage if there are missing settings
    if (missingSetting) {
        localStorage.setItem("gallerySettings", JSON.stringify(settings));
    }

    function setStyles(element, styles) {
        for (const property in styles) {
            element.style[property] = styles[property];
        }
    }

    function getPosts(websiteUrl, doc) {
        switch (websiteUrl) {
            case "warosu.org":
                return doc.querySelectorAll(".comment, .highlight");
            case "archived.moe":
            case "archive.palanq.win":
            case "archive.4plebs.org":
            case "desuarchive.org":
            case "thebarchive.com":
            case "archiveofsins.com":
                return doc.querySelectorAll(".post, .thread");
            case "boards.4chan.org":
            case "boards.4channel.org":
            default:
                return doc.querySelectorAll(".postContainer");
        }
    }

    function getDocument(thread, threadURL) {
        return new Promise((resolve, reject) => {
            if (thread === threadURL) {
                resolve(document);
            } else {
                fetch(thread)
                    .then((response) => response.text())
                    .then((html) => {
                        const parser = new DOMParser();
                        const doc = parser.parseFromString(html, "text/html");
                        resolve(doc);
                    })
                    .catch((error) => {
                        reject(error);
                    });
            }
        });
    }

    function injectVideoJS() {
        const link = document.createElement("link");
        link.href = "https://vjs.zencdn.net/8.10.0/video-js.css";
        link.rel = "stylesheet";
        document.head.appendChild(link);

        // theme
        const theme = document.createElement("link");
        theme.href = "https://unpkg.com/@videojs/themes@1/dist/city/index.css";
        theme.rel = "stylesheet";
        document.head.appendChild(theme);

        const script = document.createElement("script");
        script.src = "https://vjs.zencdn.net/8.10.0/video.min.js";
        document.body.appendChild(script);
        ("VideoJS injected successfully!");
    }

    function createArrowButton(direction) {
        const button = document.createElement('button');
        setStyles(button, {
            position: 'fixed',
            top: '50%',
            [direction]: '20px',
            transform: 'translateY(-50%)',
            zIndex: '10001',
            backgroundColor: 'rgba(28, 28, 28, 0.7)',
            color: '#d9d9d9',
            padding: '15px',
            border: 'none',
            borderRadius: '50%',
            cursor: 'pointer',
            display: settings.Show_Arrow_Buttons_In_Zoom_Mode.value ? 'block' : 'none'
        });
        button.innerHTML = direction === 'left' ? '◀' : '▶';
        button.onclick = () => {
            const event = new KeyboardEvent('keydown', { key: direction === 'left' ? 'ArrowLeft' : 'ArrowRight' });
            document.dispatchEvent(event);
        };
        return button;
    }

    // Modify createMediaCell to accept mode and postURL parameters
    function createMediaCell(url, commentText, mode, postURL, board, threadID, postID) {
        if (!gridContainer) {
            gridContainer = document.createElement("div");
            setStyles(gridContainer, {
                display: "grid",
                gridTemplateColumns: `repeat(${settings.Grid_Columns.value}, 1fr)`,
                gap: "10px",
                padding: "20px",
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                maxWidth: "80%",
                maxHeight: "80%",
                overflowY: "auto",
                resize: "both",
                overflow: "auto",
                border: "1px solid #d9d9d9",
            });
        }
        const cell = document.createElement("div");
        setStyles(cell, {
            border: "1px solid #d9d9d9",
            position: "relative",
        });

        // Make the cell draggable
        cell.draggable = true;
        cell.addEventListener("dragstart", (e) => {
            e.dataTransfer.setData("text/plain", [...gridContainer.children].indexOf(cell));
            e.dataTransfer.dropEffect = "move";
        });

        // Allow drops on this cell
        cell.addEventListener("dragover", (e) => {
            e.preventDefault();
            e.dataTransfer.dropEffect = "move";
        });

        cell.addEventListener("drop", (e) => {
            e.preventDefault();
            const draggedIndex = e.dataTransfer.getData("text/plain");
            const containerChildren = [...gridContainer.children];
            const draggedCell = containerChildren[draggedIndex];
            if (draggedCell !== cell) {
                const dropIndex = containerChildren.indexOf(cell);
                if (draggedIndex < dropIndex) {
                    gridContainer.insertBefore(draggedCell, containerChildren[dropIndex].nextSibling);
                } else {
                    gridContainer.insertBefore(draggedCell, containerChildren[dropIndex]);
                }
            }
        });

        const mediaContainer = document.createElement("div");
        setStyles(mediaContainer, {
            position: "relative",
            display: "flex",
            justifyContent: "center",
            alignItems: "center",
        });

        const buttonDiv = document.createElement("div");
        setStyles(buttonDiv, {
            display: "flex",
            justifyContent: "space-between",
            alignItems: "center",
            padding: "5px",
        });

        // Add view post button for external media
        const viewPostButton = document.createElement("button");
        viewPostButton.textContent = "View Original";
        setStyles(viewPostButton, {
            backgroundColor: "#1c1c1c",
            color: "#d9d9d9",
            padding: "5px 10px",
            borderRadius: "3px",
            border: "none",
            cursor: "pointer",
            boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
        });
        viewPostButton.addEventListener("click", () => {
            window.location.href = postURL;
            gallerySize = {
                width: gridContainer.offsetWidth,
                height: gridContainer.offsetHeight,
            };
        });
        buttonDiv.appendChild(viewPostButton);

        if (url.match(/\.(webm|mp4)$/i)) {
            const video = document.createElement("video");
            video.src = url;
            video.controls = true;
            video.title = commentText;
            video.setAttribute("fileName", url.split('/').pop());
            video.setAttribute("board", board);
            video.setAttribute("threadID", threadID);
            video.setAttribute("postID", postID);
            setStyles(video, {
                maxWidth: "100%",
                maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                objectFit: "contain",
                cursor: "pointer",
            });
            mediaContainer.appendChild(video);

            const openInNewTabButton = document.createElement("button");
            openInNewTabButton.textContent = "Open";
            setStyles(openInNewTabButton, {
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                padding: "5px 10px",
                borderRadius: "3px",
                border: "none",
                cursor: "pointer",
                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
            });
            openInNewTabButton.onclick = () => {
                window.open(url, "_blank");
            };
            buttonDiv.appendChild(openInNewTabButton);
        } else if (url.match(/\.(jpg|jpeg|png|gif)$/i)) {
            // Only create image cell if mode is "all"
            if (mode === "all") {
                const image = document.createElement("img");
                image.src = url;
                image.title = commentText;
                image.setAttribute("fileName", url.split('/').pop());
                image.setAttribute("actualSrc", url);
                image.setAttribute("thumbnailUrl", url);
                image.setAttribute("board", board);
                image.setAttribute("threadID", threadID);
                image.setAttribute("postID", postID);
                setStyles(image, {
                    maxWidth: "100%",
                    maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                    objectFit: "contain",
                    cursor: "pointer",
                });
                image.loading = "lazy";

                if (
                    settings.Strictly_Load_GIFs_As_Thumbnails_On_Hover.value &&
                    url.match(/\.gif$/i)
                ) {
                    image.src = url;
                    image.addEventListener("mouseover", () => {
                        image.src = url;
                    });
                    image.addEventListener("mouseout", () => {
                        image.src = url;
                    });
                }

                mediaContainer.appendChild(image);
            } else {
                return; // Skip non-webm/soundpost media in webm mode
            }
        }

        cell.appendChild(mediaContainer);
        cell.appendChild(buttonDiv);
        gridContainer.appendChild(cell);
    }

    const loadButton = () => {
        const isArchivePage = window.location.pathname.includes("/archive");
        let addFakeImage = settings.Add_Placeholder_Image_For_Zoom_Mode.value;

        const button = document.createElement("button");
        button.textContent = "Open Image Gallery";
        button.id = "openImageGallery";
        setStyles(button, {
            position: "fixed",
            bottom: "20px",
            right: "20px",
            zIndex: "1000",
            backgroundColor: "#1c1c1c",
            color: "#d9d9d9",
            padding: "10px 20px",
            borderRadius: "5px",
            border: "none",
            cursor: "pointer",
            boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
            visibility: settings.Hide_Gallery_Button.value ? "hidden" : "visible",
        });

        const openImageGallery = () => {
            // new check to see if gallery is already in the DOM
            const existingGallery = document.getElementById("imageGallery");
            if (existingGallery) {
                existingGallery.style.display = "flex";
                return;
            }
            addFakeImage = settings.Add_Placeholder_Image_For_Zoom_Mode.value;

            const gallery = document.createElement("div");
            gallery.id = "imageGallery";
            setStyles(gallery, {
                position: "fixed",
                top: "0",
                left: "0",
                width: "100%",
                height: "100%",
                backgroundColor: "rgba(0, 0, 0, 0.8)",
                display: "flex",
                justifyContent: "center",
                alignItems: "center",
                zIndex: "9999",
            });

            gridContainer = document.createElement("div");
            setStyles(gridContainer, {
                display: "grid",
                gridTemplateColumns: `repeat(${settings.Grid_Columns.value}, 1fr)`,
                gap: "10px",
                padding: "20px",
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                maxWidth: "80%",
                maxHeight: "80%",
                overflowY: "auto",
                resize: "both",
                overflow: "auto",
                border: "1px solid #d9d9d9",
            });

            // Add dragover & drop listeners to the grid container
            gridContainer.addEventListener("dragover", (e) => {
                e.preventDefault();
            });
            gridContainer.addEventListener("drop", (e) => {
                e.preventDefault();
                const draggedIndex = e.dataTransfer.getData("text/plain");
                const targetCell = e.target.closest("div[draggable='true']");
                if (!targetCell) return;
                const containerChildren = [...gridContainer.children];
                const dropIndex = containerChildren.indexOf(targetCell);
                if (draggedIndex >= 0 && dropIndex >= 0) {
                    const draggedCell = containerChildren[draggedIndex];
                    if (draggedIndex < dropIndex) {
                        gridContainer.insertBefore(draggedCell, containerChildren[dropIndex].nextSibling);
                    } else {
                        gridContainer.insertBefore(draggedCell, containerChildren[dropIndex]);
                    }
                }
            });

            // Restore the previous grid container size
            if (gallerySize.width > 0 && gallerySize.height > 0) {
                gridContainer.style.width = `${gallerySize.width}px`;
                gridContainer.style.height = `${gallerySize.height}px`;
            }

            let mode = "all"; // Default mode is "all"
            let autoPlayWebms = false; // Default auto play webms without sound is false

            const mediaTypeButtonContainer = document.createElement("div");
            setStyles(mediaTypeButtonContainer, {
                position: "absolute",
                top: "10px",
                left: "10px",
                display: "flex",
                gap: "10px",
            });

            // Toggle mode button
            const toggleModeButton = document.createElement("button");
            toggleModeButton.textContent = "Toggle Mode (All)";
            setStyles(toggleModeButton, {
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                padding: "10px 20px",
                borderRadius: "5px",
                border: "none",
                cursor: "pointer",
                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
            });
            toggleModeButton.addEventListener("click", () => {
                mode = mode === "all" ? "webm" : "all";
                toggleModeButton.textContent = `Toggle Mode (${mode === "all" ? "All" : "Webm & Images with Sound"})`;
                gridContainer.innerHTML = ""; // Clear the grid
                loadPosts(mode, addFakeImage); // Reload posts based on the new mode
            });

            // Toggle auto play webms button
            const toggleAutoPlayButton = document.createElement("button");
            toggleAutoPlayButton.textContent = "Auto Play Webms without Sound";
            setStyles(toggleAutoPlayButton, {
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                padding: "10px 20px",
                borderRadius: "5px",
                border: "none",
                cursor: "pointer",
                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
            });
            toggleAutoPlayButton.addEventListener("click", () => {
                autoPlayWebms = !autoPlayWebms;
                toggleAutoPlayButton.textContent = autoPlayWebms
                    ? "Stop Auto Play Webms"
                    : "Auto Play Webms without Sound";
                gridContainer.innerHTML = ""; // Clear the grid
                loadPosts(mode, addFakeImage); // Reload posts based on the new mode and auto play setting
            });
            mediaTypeButtonContainer.appendChild(toggleModeButton);
            mediaTypeButtonContainer.appendChild(toggleAutoPlayButton);
            gallery.appendChild(mediaTypeButtonContainer);

            // settings button on the top right corner of the screen
            const settingsButton = document.createElement("button");
            settingsButton.id = "settingsButton";
            settingsButton.textContent = "Settings";
            setStyles(settingsButton, {
                position: "absolute",
                top: "20px",
                right: "20px",
                backgroundColor: "#007bff", // Primary color
                color: "#fff",
                padding: "10px 20px",
                borderRadius: "5px",
                border: "none",
                cursor: "pointer",
                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                transition: "background-color 0.3s ease",
            });
            settingsButton.addEventListener("click", () => {
                const settingsContainer = document.createElement("div");
                settingsContainer.id = "settingsContainer";
                setStyles(settingsContainer, {
                    position: "fixed",
                    top: "0",
                    left: "0",
                    width: "100%",
                    height: "100%",
                    backgroundColor: "rgba(0, 0, 0, 0.8)",
                    display: "flex",
                    justifyContent: "center",
                    alignItems: "center",
                    zIndex: "9999",
                    animation: "fadeIn 0.3s ease",
                });

                const settingsBox = document.createElement("div");
                setStyles(settingsBox, {
                    backgroundColor: "#000000", // Background color
                    color: "#ffffff", // Text color
                    padding: "30px",
                    borderRadius: "10px",
                    border: "1px solid #6c757d", // Secondary color
                    maxWidth: "80%",
                    maxHeight: "80%",
                    overflowY: "auto",
                    boxShadow: "0 4px 6px rgba(0, 0, 0, 0.1)",
                });

                const settingsTitle = document.createElement("h2");
                settingsTitle.id = "settingsTitle";
                settingsTitle.textContent = "Settings";
                setStyles(settingsTitle, {
                    textAlign: "center",
                    marginBottom: "20px",
                });

                const settingsList = document.createElement("ul");
                settingsList.id = "settingsList";
                setStyles(settingsList, {
                    listStyleType: "none",
                    padding: "0",
                    margin: "0",
                });

                // include default settings as existing settings inside the input fields
                // have an icon next to the setting that explains what the setting does
                for (const setting in settings) {
                    // remove settings that are not in the default settings
                    if (!(setting in defaultSettings)) {
                        delete settings[setting];
                        continue;
                    }

                    const settingItem = document.createElement("li");
                    setStyles(settingItem, {
                        display: "flex",
                        alignItems: "center",
                        marginBottom: "15px",
                    });

                    const settingLabel = document.createElement("label");
                    settingLabel.textContent = setting.replace(/_/g, " ");
                    settingLabel.title = settings[setting].info;
                    setStyles(settingLabel, {
                        flex: "1",
                        display: "flex",
                        alignItems: "center",
                    });

                    const settingIcon = document.createElement("span");
                    settingIcon.className = "material-icons-outlined";
                    settingIcon.textContent = settings[setting].icon;
                    settingIcon.style.marginRight = "10px";
                    settingLabel.prepend(settingIcon);

                    settingItem.appendChild(settingLabel);

                    const settingInput = document.createElement("input");
                    const settingValueType = typeof defaultSettings[setting].value;
                    if (settingValueType === "boolean") {
                        settingInput.type = "checkbox";
                        settingInput.checked = settings[setting].value;
                    } else if (settingValueType === "number") {
                        settingInput.type = "number";
                        settingInput.value = settings[setting].value;
                    } else {
                        settingInput.type = "text";
                        settingInput.value = settings[setting].value;
                    }
                    setStyles(settingInput, {
                        padding: "8px 12px",
                        borderRadius: "5px",
                        border: "1px solid #6c757d", // Secondary color
                        flex: "2",
                    });
                    settingInput.addEventListener("focus", () => {
                        setStyles(settingInput, {
                            borderColor: "#007bff", // Primary color
                            boxShadow: "0 0 0 2px rgba(0, 123, 255, 0.25)",
                            outline: "none",
                        });
                    });
                    settingInput.addEventListener("blur", () => {
                        setStyles(settingInput, {
                            borderColor: "#6c757d", // Secondary color
                            boxShadow: "none",
                        });
                    });

                    if (settingValueType === "boolean") {
                        settingInput.style.marginRight = "10px";
                    }

                    settingItem.appendChild(settingInput);
                    settingsList.appendChild(settingItem);
                }

                const saveButton = document.createElement("button");
                saveButton.id = "saveButton";
                saveButton.textContent = "Save";
                setStyles(saveButton, {
                    backgroundColor: "#007bff", // Primary color
                    color: "#fff",
                    padding: "10px 20px",
                    borderRadius: "5px",
                    border: "none",
                    cursor: "pointer",
                    boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                    transition: "background-color 0.3s ease",
                    marginRight: "10px",
                });
                saveButton.addEventListener("click", () => {
                    const newSettings = {};
                    // First copy default settings structure
                    for (const key in defaultSettings) {
                        newSettings[key] = { ...defaultSettings[key] };
                    }

                    const inputs = document.querySelectorAll("#settingsList input");
                    inputs.forEach((input) => {
                        const settingName = input.previousSibling.textContent.replace(
                            / /g,
                            "_"
                        );
                        if (settingName in defaultSettings) {
                            newSettings[settingName].value = input.type === "checkbox" ? input.checked : input.value;
                        }
                    });
                    localStorage.setItem("gallerySettings", JSON.stringify(newSettings));
                    settings = newSettings;
                    settingsContainer.remove();

                    const gallery = document.querySelector('#imageGallery');
                    if (gallery) {
                        document.body.removeChild(gallery);
                        setTimeout(() => {
                            document.querySelector('#openImageGallery').click();
                        }, 20);
                    }
                });

                // Close button
                const closeButton = document.createElement("button");
                closeButton.id = "closeButton";
                closeButton.textContent = "Close";
                setStyles(closeButton, {
                    backgroundColor: "#007bff", // Primary color
                    color: "#fff",
                    padding: "10px 20px",
                    borderRadius: "5px",
                    border: "none",
                    cursor: "pointer",
                    boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                    transition: "background-color 0.3s ease",
                });
                closeButton.addEventListener("click", () => {
                    settingsContainer.remove();
                });

                settingsBox.appendChild(settingsTitle);
                settingsBox.appendChild(settingsList);
                settingsBox.appendChild(saveButton);
                settingsBox.appendChild(closeButton);
                settingsContainer.appendChild(settingsBox);
                gallery.appendChild(settingsContainer);
            });

            // Hover effect for settings button
            settingsButton.addEventListener("mouseenter", () => {
                settingsButton.style.backgroundColor = "#0056b3";
            });
            settingsButton.addEventListener("mouseleave", () => {
                settingsButton.style.backgroundColor = "#007bff";
            });

            gallery.appendChild(settingsButton);

            const loadPosts = (mode, addFakeImage) => {
                const checkedThreads = isArchivePage
                    ? // Get all checked threads in the archive page or the current link if it's not an archive page
                    Array.from(
                        document.querySelectorAll(
                            ".flashListing input[type='checkbox']:checked"
                        )
                    ).map((checkbox) => {
                        let archiveSite =
                            checkbox.parentNode.parentNode.querySelector("a").href;
                        return archiveSite;
                    })
                    : [threadURL];

                const loadPostsFromThread = (thread, addFakeImage) => {
                    // get the website url without the protocol and next slash
                    let websiteUrl = thread.replace(/(^\w+:|^)\/\//, "").split("/")[0];

                    // const board = thread.split("/thread/")[0].split("/").pop();
                    // const threadNo = `${parseInt(thread.split("thread/").pop())}`
                    getDocument(thread, threadURL).then((doc) => {
                        let posts;

                        // use a case statement to deal with different websites
                        posts = getPosts(websiteUrl, doc);

                        // add thread and website url as attributes to every post
                        posts.forEach((post) => {
                            post.setAttribute("thread", thread);
                            post.setAttribute("websiteUrl", websiteUrl);
                        });

                        if (addFakeImage) {
                            // Add a fake image to the grid container to allow zoom mode to open even if the thread has no images
                            let placeholder_imageURL = "https://files.pixstash.moe/ecl8vh.png";
                            let examplePost = document.createElement("div");
                            examplePost.innerHTML = `
                                <div class="postContainer", id="1231232">
                                    <div class="fileText">
                                        <a href="${placeholder_imageURL}" download="${placeholder_imageURL}">OpenZoomMode[sound=https://files.catbox.moe/brugtt.mp3].jpg</a>
                                    </div>
                                    <div class="fileThumb">
                                        <img src="${placeholder_imageURL}" alt="Thumbnail">
                                    </div>
                                    <div class="postMessage">
                                        Just a placeholder image for zoom mode
                                    </div>
                                </div>
                            `;
                            examplePost.setAttribute("thread", "https://boards.4chan.org/b/thread/123456789");
                            examplePost.setAttribute("websiteUrl", "boards.4chan.org");
                            posts = [examplePost, ...posts];
                        }

                        posts.forEach((post) => {
                            let mediaLinkFlag = false;
                            let board;
                            let threadID;
                            let postID;
                            let postURL;
                            let thumbnailUrl;
                            let mediaLink;
                            let fileName;
                            let comment;

                            let isVideo;
                            let isImage;
                            let soundLink;
                            let encodedSoundPostLink;
                            let temp;
                            let hasEmbeddedMediaLink = false;
                            let matches;

                            websiteUrl = post.getAttribute("websiteUrl");
                            thread = post.getAttribute("thread");

                            // case statement for different websites
                            switch (websiteUrl) {
                                case "warosu.org":
                                    let thumbnailElement = post.querySelector(".thumb");

                                    fileName = post
                                        .querySelector(".fileinfo")
                                        ?.innerText.split(", ")[2];
                                    thumbnailUrl = thumbnailElement?.src;
                                    mediaLink = thumbnailElement?.parentNode.href;
                                    comment = post.querySelector("blockquote");

                                    threadID = post.getAttribute("thread").match(/thread\/(\d+)/)
                                    if (threadID) {
                                        threadID = threadID[1];
                                    } else {
                                        threadID = post.querySelector(".js").href.match(/thread\/(\d+)/)[1];
                                    }

                                    postID = post.id.replace("pc", "").replace("p", "");
                                    break;
                                case "archived.moe":
                                case "archive.palanq.win":
                                case "archive.4plebs.org":
                                case "desuarchive.org":
                                case "thebarchive.com":
                                case "archiveofsins.com":
                                    thumbnailUrl = post.querySelector(".post_image")?.src;
                                    mediaLink = post.querySelector(".thread_image_link")?.href;
                                    fileName = post.querySelector(
                                        ".post_file_filename"
                                    )?.title;
                                    comment = post.querySelector(".text");
                                    threadID = post.querySelector(".post_data > a")?.href.match(
                                        /thread\/(\d+)/
                                    )[1];
                                    postID = post.id
                                    break;
                                case "boards.4chan.org":
                                case "boards.4channel.org":
                                default:
                                    if (post.querySelector(".fileText")) {
                                        // if they have 4chanX installed, there will be a fileText-orignal class
                                        if (post.querySelector(".download-button")) {
                                            temp = post.querySelector(".download-button");
                                            mediaLink = temp.href;
                                            fileName = temp.download;
                                        } else {
                                            if (post.classList.contains("opContainer")) {
                                                mediaLink = post.querySelector(".fileText a");
                                                temp = mediaLink;
                                            } else {
                                                mediaLink = post.querySelector(".fileText");
                                                temp = mediaLink.querySelector("a");
                                            }
                                            if (mediaLink.title === "") {
                                                if (temp.title === "") {
                                                    fileName = temp.innerText;
                                                } else {
                                                    fileName = temp.title;
                                                }
                                            } else {
                                                fileName = mediaLink.title;
                                            }
                                            mediaLink = temp.href;
                                        }
                                        thumbnailUrl = post.querySelector(".fileThumb img")?.src;
                                    }

                                    comment = post.querySelector(".postMessage");

                                    threadID = thread.match(/thread\/(\d+)/)[1];
                                    postID = post.id.replace("pc", "").replace("p", "");
                            }

                            const fileExtRegex = /\.(webm|mp4|jpg|png|gif)$/i;
                            const linkRegex = /https:\/\/(files|litter)\.(catbox|pixstash)\.moe\/[a-z0-9]+\.(jpg|png|gif|webm|mp4)/g;

                            if (mediaLink) {
                                const ext = mediaLink.match(fileExtRegex)?.[1]?.toLowerCase();
                                isVideo = ext === 'webm' || ext === 'mp4';
                                isImage = ext === 'jpg' || ext === 'png' || ext === 'gif';
                                soundLink = fileName.match(/\[sound=(.+?)\]/);
                                mediaLinkFlag = true;
                            }
                            if (settings.Embed_External_Links.value && comment) {
                                matches = Array.from(comment.innerText.matchAll(linkRegex)).map(match => match[0]);
                                if (matches.length > 0) {
                                    if (!mediaLinkFlag) {
                                        mediaLink = matches[0];
                                        fileName = mediaLink.split("/").pop();
                                        thumbnailUrl = mediaLink;

                                        if (hasEmbeddedMediaLink) {
                                            matches.shift();
                                        }

                                        const ext = mediaLink.match(fileExtRegex)?.[1]?.toLowerCase();
                                        isVideo = ext === 'webm' || ext === 'mp4';
                                        isImage = ext === 'jpg' || ext === 'png' || ext === 'gif';
                                        soundLink = fileName.match(/\[sound=(.+?)\]/);
                                        mediaLinkFlag = true;
                                    }
                                    hasEmbeddedMediaLink = matches.length > 0;
                                }
                            }

                            // replace the "#pcXXXXXXX" or "#pXXXXXXX" with an empty string to get the actual thread url
                            if (thread.includes("#")) {
                                postURL = thread.replace(/#p\d+/, "");
                                postURL = postURL.replace(/#pc\d+/, "");
                            } else {
                                postURL = thread;
                            }

                            // post info (constant)
                            board = thread.match(/\/\/[^\/]+\/([^\/]+)/)[1];
                            if (soundLink) {
                                encodedSoundPostLink = `https://4chan.mahdeensky.top/${board}/thread/${threadID}/${postID}`;
                            }

                            if (mediaLinkFlag) {
                                // Check if the post should be loaded based on the mode
                                if (
                                    mode === "all" ||
                                    (mode === "webm" && (isVideo || (isImage && soundLink)))
                                ) {
                                    // Insert a button/link to open media in new tab for videos
                                    const cell = document.createElement("div");
                                    setStyles(cell, {
                                        border: "1px solid #d9d9d9",
                                        position: "relative",
                                    });

                                    // Make the cell draggable
                                    cell.draggable = true;
                                    cell.addEventListener("dragstart", (e) => {
                                        e.dataTransfer.setData("text/plain", [...gridContainer.children].indexOf(cell));
                                        e.dataTransfer.dropEffect = "move";
                                    });

                                    // Allow drops on this cell
                                    cell.addEventListener("dragover", (e) => {
                                        e.preventDefault();
                                        e.dataTransfer.dropEffect = "move";
                                    });

                                    cell.addEventListener("drop", (e) => {
                                        e.preventDefault();
                                        const draggedIndex = e.dataTransfer.getData("text/plain");
                                        const containerChildren = [...gridContainer.children];
                                        const draggedCell = containerChildren[draggedIndex];
                                        if (draggedCell !== cell) {
                                            const dropIndex = containerChildren.indexOf(cell);
                                            if (draggedIndex < dropIndex) {
                                                gridContainer.insertBefore(draggedCell, containerChildren[dropIndex].nextSibling);
                                            } else {
                                                gridContainer.insertBefore(draggedCell, containerChildren[dropIndex]);
                                            }
                                        }
                                    });

                                    const buttonDiv = document.createElement("div");
                                    setStyles(buttonDiv, {
                                        display: "flex",
                                        justifyContent: "space-between",
                                        alignItems: "center",
                                        padding: "5px",
                                    });

                                    if (isVideo) {
                                        const videoContainer = document.createElement("div");
                                        setStyles(videoContainer, {
                                            position: "relative",
                                            display: "flex",
                                            justifyContent: "center",
                                        });

                                        // if medialink is catbox.moe or pixstash.moe, then video thumbnail is a video element with no controls
                                        let videoThumbnail;
                                        if (mediaLink.match(/catbox.moe|pixstash.moe/)) {
                                            videoThumbnail = document.createElement("video");
                                        } else {
                                            videoThumbnail = document.createElement("img");
                                        }
                                        videoThumbnail.src = thumbnailUrl;
                                        videoThumbnail.alt = "Video Thumbnail";
                                        setStyles(videoThumbnail, {
                                            width: "100%",
                                            maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                                            objectFit: "contain",
                                            cursor: "pointer",
                                        });
                                        videoThumbnail.loading = "lazy";

                                        const video = document.createElement("video");
                                        video.src = mediaLink;
                                        video.controls = true;
                                        video.title = comment.innerText;
                                        video.videothumbnailDisplayed = "true";
                                        video.setAttribute("fileName", fileName);
                                        video.setAttribute("board", board);
                                        video.setAttribute("threadID", threadID);
                                        video.setAttribute("postID", postID);
                                        setStyles(video, {
                                            maxWidth: "100%",
                                            maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                                            objectFit: "contain",
                                            cursor: "pointer",
                                            display: "none",
                                        });

                                        // videoJS stuff (not working for some reason)
                                        // video.className = "video-js";
                                        // video.setAttribute("data-setup", "{}");
                                        // const source = document.createElement("source");
                                        // source.src = mediaLink;
                                        // source.type = "video/webm";
                                        // video.appendChild(source);

                                        videoThumbnail.addEventListener("click", () => {
                                            videoThumbnail.style.display = "none";
                                            video.style.display = "block";
                                            video.videothumbnailDisplayed = "false";
                                            // video.load();
                                        });

                                        // hide the video thumbnail and show the video when hovered
                                        videoThumbnail.addEventListener("mouseenter", () => {
                                            videoThumbnail.style.display = "none";
                                            video.style.display = "block";
                                            video.videothumbnailDisplayed = "false";
                                            // video.load();
                                        });

                                        // Play webms without sound automatically on hover or if autoPlayWebms is true
                                        if (!soundLink) {
                                            if (autoPlayWebms) {
                                                video.addEventListener("canplaythrough", () => {
                                                    video.play();
                                                    video.loop = true; // Loop webms when autoPlayWebms is true
                                                });
                                            } else {
                                                if (settings.Play_Webms_On_Hover.value) {
                                                    video.addEventListener("mouseenter", () => {
                                                        video.play();
                                                    });
                                                    video.addEventListener("mouseleave", () => {
                                                        video.pause();
                                                    });
                                                }
                                            }
                                        }

                                        videoContainer.appendChild(videoThumbnail);
                                        videoContainer.appendChild(video);

                                        if (soundLink) {
                                            // video.preload = "none"; // Disable video preload for better performance

                                            const audio = document.createElement("audio");
                                            audio.src = decodeURIComponent(
                                                soundLink[1].startsWith("http")
                                                    ? soundLink[1]
                                                    : `https://${soundLink[1]}`
                                            );

                                            // if switch catbox to pixstash is enabled, replace catbox.moe with pixstash.moe
                                            if (settings.Switch_Catbox_To_Pixstash_For_Soundposts.value) {
                                                audio.src = audio.src.replace("catbox.moe", "pixstash.moe");
                                            }

                                            // add attribute to the audio element with the encoded soundpost link
                                            audio.setAttribute(
                                                "encodedSoundPostLink",
                                                encodedSoundPostLink
                                            );
                                            videoContainer.appendChild(audio);

                                            const resetButton = document.createElement("button");
                                            resetButton.textContent = "Reset";
                                            setStyles(resetButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            resetButton.addEventListener("click", () => {
                                                video.currentTime = 0;
                                                audio.currentTime = 0;
                                            });
                                            buttonDiv.appendChild(resetButton);

                                            // html5 video play
                                            video.onplay = (event) => {
                                                audio.play();
                                            };

                                            video.onpause = (event) => {
                                                audio.pause();
                                            };

                                            let lastVideoTime = 0;
                                            // Sync audio with video on timeupdate event only if the difference is 2 seconds or more
                                            video.addEventListener("timeupdate", () => {
                                                if (Math.abs(video.currentTime - lastVideoTime) >= 2) {
                                                    audio.currentTime = video.currentTime;
                                                    lastVideoTime = video.currentTime;
                                                }
                                                lastVideoTime = video.currentTime;
                                            });
                                        }

                                        cell.appendChild(videoContainer);
                                    } else if (isImage) {
                                        const imageContainer = document.createElement("div");
                                        setStyles(imageContainer, {
                                            position: "relative",
                                            display: "flex",
                                            justifyContent: "center",
                                            alignItems: "center",
                                        });

                                        const image = document.createElement("img");
                                        image.src = thumbnailUrl;
                                        if (settings.Load_High_Res_Images_By_Default.value) {
                                            image.src = mediaLink;
                                        }
                                        if (mediaLink.includes(".gif")) {
                                            image.src = mediaLink;

                                            if (
                                                settings.Strictly_Load_GIFs_As_Thumbnails_On_Hover.value
                                            ) {
                                                mediaLink = thumbnailUrl;
                                                image.src = thumbnailUrl;
                                            }

                                        }
                                        image.setAttribute("fileName", fileName);
                                        image.setAttribute("actualSrc", mediaLink);
                                        image.setAttribute("thumbnailUrl", thumbnailUrl);
                                        image.setAttribute("board", board);
                                        image.setAttribute("threadID", threadID);
                                        image.setAttribute("postID", postID);
                                        setStyles(image, {
                                            maxWidth: "100%",
                                            maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                                            objectFit: "contain",
                                            cursor: "pointer",
                                        });

                                        let createDarkenBackground = () => {
                                            const background = document.createElement("div");
                                            background.id = "darkenBackground";
                                            setStyles(background, {
                                                position: "fixed",
                                                top: "0",
                                                left: "0",
                                                width: "100%",
                                                height: "100%",
                                                backgroundColor: "rgba(0, 0, 0, 0.3)",
                                                backdropFilter: "blur(5px)",
                                                zIndex: "9999",
                                            });
                                            return background;
                                        };

                                        let zoomImage = () => {
                                            // have the image pop up centered in front of the screen so that it fills about 80% of the screen
                                            image.style = "";
                                            image.src = mediaLink;
                                            setStyles(image, {
                                                position: "fixed",
                                                top: "50%",
                                                left: "50%",
                                                transform: "translate(-50%, -50%)",
                                                zIndex: "10000",
                                                height: "80%",
                                                width: "80%",
                                                objectFit: "contain",
                                                cursor: "pointer",
                                            });

                                            // darken and blur the background behind the image without affecting the image
                                            const background = createDarkenBackground();
                                            background.appendChild(createArrowButton('left'));
                                            background.appendChild(createArrowButton('right'));
                                            gallery.appendChild(background);

                                            // create a container for the buttons, number, and download buttons (even space between them)
                                            // position: fixed; bottom: 10px; display: flex; flex-direction: row; justify-content: space-around; z-index: 10000; width: 100%; margin:auto;
                                            const bottomContainer = document.createElement("div");
                                            setStyles(bottomContainer, {
                                                position: "fixed",
                                                bottom: "10px",
                                                display: "flex",
                                                flexDirection: "row",
                                                justifyContent: "space-around",
                                                zIndex: "10000",
                                                width: "100%",
                                                margin: "auto",
                                            });
                                            background.appendChild(bottomContainer);

                                            // buttons on the bottom left of the screen for reverse image search (SauceNAO, Google Lens, Yandex)
                                            const buttonContainer = document.createElement("div");
                                            setStyles(buttonContainer, {
                                                display: "flex",
                                                gap: "10px",
                                            });
                                            buttonContainer.setAttribute("mediaLink", mediaLink);

                                            const sauceNAOButton = document.createElement("button");
                                            sauceNAOButton.textContent = "SauceNAO";
                                            setStyles(sauceNAOButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                            });
                                            sauceNAOButton.addEventListener("click", () => {
                                                window.open(
                                                    `https://saucenao.com/search.php?url=${encodeURIComponent(
                                                        buttonContainer.getAttribute("mediaLink")
                                                    )}`
                                                );
                                            });
                                            buttonContainer.appendChild(sauceNAOButton);

                                            const googleLensButton = document.createElement("button");
                                            googleLensButton.textContent = "Google Lens";
                                            setStyles(googleLensButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            googleLensButton.addEventListener("click", () => {
                                                window.open(
                                                    `https://lens.google.com/uploadbyurl?url=${encodeURIComponent(
                                                        buttonContainer.getAttribute("mediaLink")
                                                    )}`
                                                );
                                            });
                                            buttonContainer.appendChild(googleLensButton);

                                            const yandexButton = document.createElement("button");
                                            yandexButton.textContent = "Yandex";
                                            setStyles(yandexButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            yandexButton.addEventListener("click", () => {
                                                window.open(
                                                    `https://yandex.com/images/search?rpt=imageview&url=${encodeURIComponent(
                                                        buttonContainer.getAttribute("mediaLink")
                                                    )}`
                                                );
                                            });
                                            buttonContainer.appendChild(yandexButton);

                                            bottomContainer.appendChild(buttonContainer);

                                            // download container for video/img and audio
                                            const downloadButtonContainer =
                                                document.createElement("div");
                                            setStyles(downloadButtonContainer, {
                                                display: "flex",
                                                gap: "10px",
                                            });
                                            bottomContainer.appendChild(downloadButtonContainer);

                                            const viewPostButton = document.createElement("a");
                                            viewPostButton.textContent = "View Post";
                                            viewPostButton.href = `https://boards.4chan.org/${board}/thread/${threadID}#p${postID}`;
                                            setStyles(viewPostButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            downloadButtonContainer.appendChild(viewPostButton);

                                            const downloadButton = document.createElement("a");
                                            downloadButton.textContent = "Download Video/Image";
                                            downloadButton.href = mediaLink;
                                            downloadButton.download = fileName;
                                            downloadButton.target = "_blank";
                                            setStyles(downloadButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            downloadButtonContainer.appendChild(downloadButton);

                                            const audioDownloadButton = document.createElement("a");
                                            audioDownloadButton.textContent = "Download Audio";
                                            audioDownloadButton.target = "_blank";
                                            setStyles(audioDownloadButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            if (soundLink) {
                                                audioDownloadButton.href = decodeURIComponent(
                                                    soundLink[1].startsWith("http")
                                                        ? soundLink[1]
                                                        : `https://${soundLink[1]}`
                                                );

                                                // if switch catbox to pixstash is enabled, replace catbox.moe with pixstash.moe
                                                if (settings.Switch_Catbox_To_Pixstash_For_Soundposts.value) {
                                                    audioDownloadButton.href = audioDownloadButton.href.replace(
                                                        "catbox.moe",
                                                        "pixstash.moe"
                                                    );
                                                }

                                                audioDownloadButton.download = soundLink[1]
                                                    .split("/")
                                                    .pop();
                                            } else {
                                                audioDownloadButton.style.display = "none";
                                            }
                                            downloadButtonContainer.appendChild(audioDownloadButton);

                                            // a button beside the download video and download audio button that says download encoded soundpost which links to the following url in a new tab "https://4chan.mahdeensky.top/<board>/thread/<thread>/<post>" where things between the <>, are variables to be replaced
                                            const encodedSoundPostButton =
                                                document.createElement("a");
                                            encodedSoundPostButton.textContent =
                                                "Download Encoded Soundpost";
                                            encodedSoundPostButton.target = "_blank";
                                            setStyles(encodedSoundPostButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            if (soundLink) {
                                                encodedSoundPostButton.href = `https://4chan.mahdeensky.top/${board}/thread/${threadID}/${postID}`;
                                            } else {
                                                encodedSoundPostButton.style.display = "none";
                                            }
                                            downloadButtonContainer.appendChild(
                                                encodedSoundPostButton
                                            );

                                            // number on the bottom right of the screen to show which image is currently being viewed
                                            const imageNumber = document.createElement("div");
                                            let currentImageNumber =
                                                Array.from(cell.parentNode.children).indexOf(cell) + 1;
                                            let imageTotal = cell.parentNode.children.length;
                                            imageNumber.textContent = `${currentImageNumber}/${imageTotal}`;
                                            setStyles(imageNumber, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                                position: "fixed",
                                                top: "10px",
                                                left: "10px",
                                            });
                                            background.appendChild(imageNumber);

                                            // title of the image/video on the top left of the screen
                                            const imageTitle = document.createElement("div");
                                            imageTitle.textContent = fileName;
                                            setStyles(imageTitle, {
                                                position: "fixed",
                                                top: "10px",
                                                right: "10px",
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                                zIndex: "10000",
                                            });
                                            background.appendChild(imageTitle);

                                            let currentCell = cell;

                                            function navigateImage(direction) {
                                                const targetCell = direction === 'left' ? currentCell.previousElementSibling : currentCell.nextElementSibling;
                                                if (!targetCell) return;

                                                // ...existing navigation code using targetCell instead of previousCell/nextCell...
                                                if (gallery.querySelector("#zoomedVideo")) {
                                                    if (
                                                        gallery
                                                            .querySelector("#zoomedVideo")
                                                            .querySelector("audio")
                                                    ) {
                                                        gallery
                                                            .querySelector("#zoomedVideo")
                                                            .querySelector("audio")
                                                            .pause();
                                                    }
                                                    gallery.removeChild(
                                                        gallery.querySelector("#zoomedVideo")
                                                    );
                                                } else if (gallery.querySelector("#zoomedImage")) {
                                                    gallery.removeChild(
                                                        gallery.querySelector("#zoomedImage")
                                                    );
                                                } else {
                                                    image.style = "";
                                                    // image.src = thumbnailUrl;
                                                    setStyles(image, {
                                                        maxWidth: "100%",
                                                        maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                                                        objectFit: "contain",
                                                    });
                                                }

                                                // check if it has a video
                                                const video = targetCell?.querySelector("video");
                                                if (video) {
                                                    const video = targetCell
                                                        .querySelector("video")
                                                        .cloneNode(true);
                                                    video.id = "zoomedVideo";
                                                    video.style = "";
                                                    setStyles(video, {
                                                        position: "fixed",
                                                        top: "50%",
                                                        left: "50%",
                                                        transform: "translate(-50%, -50%)",
                                                        zIndex: "10000",
                                                        height: "80%",
                                                        width: "80%",
                                                        objectFit: "contain",
                                                        cursor: "pointer",
                                                        preload: "auto",
                                                    });
                                                    gallery.appendChild(video);

                                                    // check if there is an audio element
                                                    let audio = targetCell.querySelector("audio");
                                                    if (audio) {
                                                        audio = audio.cloneNode(true);

                                                        // same event listeners as the video
                                                        video.onplay = (event) => {
                                                            audio.play();
                                                        };

                                                        video.onpause = (event) => {
                                                            audio.pause();
                                                        };

                                                        let lastVideoTime = 0;
                                                        video.addEventListener("timeupdate", () => {
                                                            if (
                                                                Math.abs(
                                                                    video.currentTime - lastVideoTime
                                                                ) >= 2
                                                            ) {
                                                                audio.currentTime = video.currentTime;
                                                                lastVideoTime = video.currentTime;
                                                            }
                                                            lastVideoTime = video.currentTime;
                                                        });
                                                        video.appendChild(audio);
                                                    }
                                                } else {
                                                    // if it doesn't have a video, it must have an image
                                                    const originalImage =
                                                        targetCell.querySelector("img");
                                                    const currentImage =
                                                        originalImage.cloneNode(true);
                                                    currentImage.id = "zoomedImage";
                                                    currentImage.style = "";
                                                    currentImage.src =
                                                        currentImage.getAttribute("actualSrc");
                                                    originalImage.src =
                                                        originalImage.getAttribute("actualSrc");
                                                    setStyles(currentImage, {
                                                        position: "fixed",
                                                        top: "50%",
                                                        left: "50%",
                                                        transform: "translate(-50%, -50%)",
                                                        zIndex: "10000",
                                                        height: "80%",
                                                        width: "80%",
                                                        objectFit: "contain",
                                                        cursor: "pointer",
                                                    });
                                                    gallery.appendChild(currentImage);
                                                    currentImage.addEventListener("click", () => {
                                                        gallery.removeChild(currentImage);
                                                        gallery.removeChild(background);
                                                        document.removeEventListener(
                                                            "keydown",
                                                            keybindHandler
                                                        );
                                                    });

                                                    let audio = targetCell.querySelector("audio");
                                                    if (audio) {
                                                        audio = audio.cloneNode(true);
                                                        currentImage.appendChild(audio);

                                                        // event listeners when hovering over the image
                                                        currentImage.addEventListener(
                                                            "mouseenter",
                                                            () => {
                                                                audio.play();
                                                            }
                                                        );
                                                        currentImage.addEventListener(
                                                            "mouseleave",
                                                            () => {
                                                                audio.pause();
                                                            }
                                                        );
                                                    }
                                                }

                                                if (targetCell) {
                                                    currentCell = targetCell;
                                                    buttonContainer.setAttribute(
                                                        "mediaLink",
                                                        targetCell.querySelector("img").src
                                                    );

                                                    currentImageNumber += direction === 'left' ? -1 : 1;
                                                    imageNumber.textContent = `${currentImageNumber}/${imageTotal}`;

                                                    // filename of the video if it has one, otherwise the filename of the image
                                                    imageTitle.textContent = video
                                                        ? video.getAttribute("fileName")
                                                        : targetCell
                                                            .querySelector("img")
                                                            .getAttribute("fileName");

                                                    // update view post button link
                                                    let targetMedia = video || targetCell.querySelector("img");
                                                    let targetBoard = targetMedia.getAttribute("board");
                                                    let targetThreadID = targetMedia.getAttribute("threadID");
                                                    let targetPostID = targetMedia.getAttribute("postID");
                                                    viewPostButton.href = `https://boards.4chan.org/${targetBoard}/thread/${targetThreadID}#p${targetPostID}`;

                                                    // update the download button links
                                                    downloadButton.href = targetMedia.src;
                                                    if (targetCell.querySelector("audio")) {
                                                        // updating audio button download link
                                                        audioDownloadButton.href =
                                                            targetCell.querySelector("audio").src;
                                                        audioDownloadButton.download = targetCell
                                                            .querySelector("audio")
                                                            .src.split("/")
                                                            .pop();
                                                        audioDownloadButton.style.display = "block";

                                                        // updating encoded soundpost button link
                                                        encodedSoundPostButton.href = targetCell.querySelector("audio")
                                                            .getAttribute("encodedSoundPostLink");
                                                        encodedSoundPostButton.style.display = "block";

                                                    } else {
                                                        audioDownloadButton.style.display = "none";
                                                        encodedSoundPostButton.style.display = "none";
                                                    }
                                                }
                                            }

                                            const keybindHandler = (event) => {
                                                if (event.key === "ArrowLeft") {
                                                    navigateImage('left');
                                                } else if (event.key === "ArrowRight") {
                                                    navigateImage('right');
                                                }
                                            };

                                            document.addEventListener("keydown", keybindHandler);

                                            image.addEventListener(
                                                "click",
                                                () => {
                                                    image.style = "";
                                                    // image.src = thumbnailUrl;
                                                    setStyles(image, {
                                                        maxWidth: "99%",
                                                        maxHeight: `${settings.Grid_Cell_Max_Height.value}px`,
                                                        objectFit: "contain",
                                                    });

                                                    if (gallery.querySelector("#darkenBackground")) {
                                                        gallery.removeChild(background);
                                                    }
                                                    document.removeEventListener(
                                                        "keydown",
                                                        keybindHandler
                                                    );

                                                    image.addEventListener("click", zoomImage, {
                                                        once: true,
                                                    });
                                                },
                                                { once: true }
                                            );
                                        };

                                        image.addEventListener("click", zoomImage, { once: true });
                                        image.title = comment.innerText;
                                        image.loading = "lazy";

                                        if (soundLink) {
                                            const audio = document.createElement("audio");
                                            audio.src = decodeURIComponent(
                                                soundLink[1].startsWith("http")
                                                    ? soundLink[1]
                                                    : `https://${soundLink[1]}`
                                            );

                                            // if switch catbox to pixstash is enabled, replace catbox.moe with pixstash.moe
                                            if (settings.Switch_Catbox_To_Pixstash_For_Soundposts.value) {
                                                audio.src = audio.src.replace("catbox.moe", "pixstash.moe");
                                            }

                                            audio.loop = true;
                                            // set the attribute to the audio element with the encoded soundpost link
                                            audio.setAttribute(
                                                "encodedSoundPostLink",
                                                encodedSoundPostLink
                                            );
                                            imageContainer.appendChild(audio);

                                            image.addEventListener("mouseenter", () => {
                                                audio.play();
                                            });
                                            image.addEventListener("mouseleave", () => {
                                                audio.pause();
                                            });

                                            const playPauseButton = document.createElement("button");
                                            playPauseButton.textContent = "Play/Pause";
                                            setStyles(playPauseButton, {
                                                backgroundColor: "#1c1c1c",
                                                color: "#d9d9d9",
                                                padding: "5px 10px",
                                                borderRadius: "3px",
                                                border: "none",
                                                cursor: "pointer",
                                                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                            });
                                            playPauseButton.addEventListener("click", () => {
                                                if (audio.paused) {
                                                    audio.play();
                                                } else {
                                                    audio.pause();
                                                }
                                            });
                                            buttonDiv.appendChild(playPauseButton);
                                        }
                                        imageContainer.appendChild(image);
                                        cell.appendChild(imageContainer);
                                    } else {
                                        return; // Skip non-video and non-image posts
                                    }

                                    // Add button that scrolls to the post in the thread
                                    const viewPostButton = document.createElement("button");
                                    viewPostButton.textContent = "View Post";
                                    setStyles(viewPostButton, {
                                        backgroundColor: "#1c1c1c",
                                        color: "#d9d9d9",
                                        padding: "5px 10px",
                                        borderRadius: "3px",
                                        border: "none",
                                        cursor: "pointer",
                                        boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                    });

                                    viewPostButton.addEventListener("click", () => {
                                        gallerySize = {
                                            width: gridContainer.offsetWidth,
                                            height: gridContainer.offsetHeight,
                                        };
                                        lastScrollPosition = gridContainer.scrollTop;
                                        window.location.href = postURL + "#" + post.id;
                                        // post id example: "pc77515440"
                                        gallery.style.display = "none"; // hide instead of removing
                                    });
                                    buttonDiv.appendChild(viewPostButton);

                                    // Add button that opens the media in a new tab if the media
                                    const openInNewTabButton = document.createElement("button");
                                    openInNewTabButton.textContent = "Open";
                                    setStyles(openInNewTabButton, {
                                        backgroundColor: "#1c1c1c",
                                        color: "#d9d9d9",
                                        padding: "5px 10px",
                                        borderRadius: "3px",
                                        border: "none",
                                        cursor: "pointer",
                                        boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
                                    });

                                    openInNewTabButton.addEventListener("click", () => {
                                        window.open(mediaLink, "_blank");
                                    });
                                    buttonDiv.appendChild(openInNewTabButton);

                                    cell.appendChild(buttonDiv);
                                    gridContainer.appendChild(cell);
                                }
                            }

                            // In the loadPosts function, update the embedded links section:
                            if (hasEmbeddedMediaLink) {
                                // Create a proper post link that includes the thread ID and post ID
                                const fullPostLink = postURL + "#" + post.id;
                                matches.forEach(url => {
                                    createMediaCell(url, comment.innerText, mode, fullPostLink, board, threadID, postID); // Pass the current post's URL
                                });
                            }
                        });
                    });
                };

                // only load the fake image in the first thread
                loadPostsFromThread(checkedThreads[0], addFakeImage);

                // load the rest of the threads with no fake image
                checkedThreads.slice(1).forEach((thread) => {
                    loadPostsFromThread(thread, false);
                });
            };

            loadPosts(mode, addFakeImage);

            gallery.appendChild(gridContainer);

            const closeButton = document.createElement("button");
            closeButton.textContent = "Close";
            closeButton.id = "closeGallery";
            setStyles(closeButton, {
                position: "absolute",
                bottom: "10px",
                right: "10px",
                zIndex: "10000",
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                padding: "10px 20px",
                borderRadius: "5px",
                border: "none",
                cursor: "pointer",
                boxShadow: "0 2px 4px rgba(0, 0, 0, 0.3)",
            });
            closeButton.addEventListener("click", () => {
                gallerySize = {
                    width: gridContainer.offsetWidth,
                    height: gridContainer.offsetHeight,
                };
                gallery.style.display = "none"; // hide instead of removing
            });

            gallery.appendChild(closeButton);

            // Add scroll to bottom button
            const scrollBottomButton = document.createElement("button");
            scrollBottomButton.textContent = "Scroll to Last";
            setStyles(scrollBottomButton, {
                position: "fixed",
                bottom: "20px",
                left: "20px",
                backgroundColor: "#1c1c1c",
                color: "#d9d9d9",
                padding: "10px 20px",
                borderRadius: "5px",
                border: "none",
                cursor: "pointer",
                zIndex: "10000",
            });
            scrollBottomButton.addEventListener("click", () => {
                const lastCell = gridContainer.lastElementChild;
                if (lastCell) {
                    lastCell.scrollIntoView({ behavior: "smooth" });
                }
            });
            gallery.appendChild(scrollBottomButton);

            // Add zoom mode arrow buttons
            const background = document.createElement('div');
            background.appendChild(createArrowButton('left'));
            background.appendChild(createArrowButton('right'));

            document.body.appendChild(gallery);

            // Store the current scroll position and grid container size when closing the gallery
            // (`Last scroll position: ${lastScrollPosition} px`);
            gridContainer.addEventListener("scroll", () => {
                lastScrollPosition = gridContainer.scrollTop;
                // (`Current scroll position: ${lastScrollPosition} px`);
            });

            // Restore the last scroll position and grid container size when opening the gallery after a timeout if the url is the same
            if (window.location.href.includes(threadURL.replace(/#.*$/, ""))) {
                setTimeout(() => {
                    if (gallerySize.width > 0 && gallerySize.height > 0) {
                        gridContainer.style.width = `${gallerySize.width}px`;
                        gridContainer.style.height = `${gallerySize.height}px`;
                    }
                    // (`Restored scroll position: ${lastScrollPosition} px`);
                    gridContainer.scrollTop = lastScrollPosition;
                }, 100);
            } else {
                // Reset the last scroll position and grid container size if the url is different
                threadURL = window.location.href;
                lastScrollPosition = 0;
                gallerySize = { width: 0, height: 0 };
            }

            gallery.addEventListener("click", (event) => {
                if (event.target === gallery) {
                    closeButton.click();
                }
            });
        };

        button.addEventListener("click", openImageGallery);

        // Append the button to the body
        document.body.appendChild(button);

        if (isArchivePage) {
            // adds the category to thead
            const thead = document.querySelector(".flashListing thead tr");
            const checkboxCell = document.createElement("td");
            checkboxCell.className = "postblock";
            checkboxCell.textContent = "Selected";
            thead.insertBefore(checkboxCell, thead.firstChild);

            // Add checkboxes to each thread row
            const threadRows = document.querySelectorAll(".flashListing tbody tr");
            threadRows.forEach((row) => {
                const checkbox = document.createElement("input");
                checkbox.type = "checkbox";
                const checkboxCell = document.createElement("td");
                checkboxCell.appendChild(checkbox);
                row.insertBefore(checkboxCell, row.firstChild);
            });
        }
    };

    // Use the "i" key to open and close the gallery/grid
    document.addEventListener("keydown", (event) => {
        if (event.target.tagName === "INPUT" || event.target.tagName === "TEXTAREA") {
            return;
        }
        if (event.key === settings.Open_Close_Gallery_Key.value) {
            if (!document.querySelector("#imageGallery")) {
                document.querySelector("#openImageGallery").click();
                return;
            }

            if (document.querySelector("#imageGallery").style.display === "none") {
                document.querySelector("#openImageGallery").click();
            } else {
                document.querySelector("#closeGallery").click();
            }
        }
    });

    loadButton();
    console.log("4chan Gallery loaded successfully!");
})();