Nexus Download Wabbajack Modlist

Download all mods from NexusMods for a Wabbajack Modlist with a single click

当前为 2025-03-24 提交的版本,查看 最新版本

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         Nexus Download Wabbajack Modlist
// @namespace    NDWM
// @version      0.3
// @description  Download all mods from NexusMods for a Wabbajack Modlist with a single click
// @author       Drigtime
// @match        https://www.nexusmods.com/
// @icon         https://www.google.com/s2/favicons?sz=64&domain=nexusmods.com
// @compatible   chrome
// @compatible   edge
// @compatible   firefox
// @compatible   safari
// @compatible   brave
// @grant        GM_addStyle
// @connect      nexusmods.com
// @require      https://cdn.jsdelivr.net/npm/@zip.js/[email protected]/dist/zip.min.js
// ==/UserScript==

// MDI : https://pictogrammers.com/library/mdi/
// MDI : https://github.com/MathewSachin/Captura/blob/master/src/Captura.Core/MaterialDesignIcons.cs

(async function () {

    /**
     * @typedef {Object} NexusModState
     * @property {string} $type
     * @property {string} [Author]
     * @property {string} [Description]
     * @property {number} FileID
     * @property {string} GameName
     * @property {string} [ImageURL]
     * @property {boolean} [IsNSFW]
     * @property {number} ModID
     * @property {string} [Name]
     * @property {string} [Version]
     */

    /**
     * @typedef {Object} NexusModArchive
     * @property {string} Hash
     * @property {string} Meta
     * @property {string} Name
     * @property {number} Size
     * @property {NexusModState} State
     */

    /**
     * @typedef {Object} WabbajackModlist
     * @property {NexusModArchive[]} Archives
     */

    // @ts-ignore
    GM_addStyle(`
        :root {
            --ndc-primary-color: rgb(217 143 64);
            --ndc-primary-color-subdued: rgb(200 123 40);
            --ndc-text-white: #fff;
        }

        .ndc\\:block { display: block; }
        .ndc\\:hidden { display: none; }
        .ndc\\:flex-1 { flex: 1; }

        .ndc\\:bg-primary-subdued { background-color: var(--ndc-primary-color-subdued); }

        .ndc\\:text-white { color: var(--ndc-text-white); }
        .ndc\\:text-primary { color: var(--ndc-primary-color); }
        
        .spinner-border {
            display: inline-block;
            width: 1.5rem;
            height: 1.5rem;
            vertical-align: text-bottom;
            border: 0.25em solid currentColor;
            border-right-color: transparent;
            border-radius: 50%;
            animation: spinner-border 0.75s linear infinite;
        }
        
        @keyframes spinner-border {
            to { transform: rotate(360deg); }
        }
        
        .ndc\\:badge-primary {
            padding: 0.25rem 0.5rem;
            border-radius: 1rem;
            font-size: 0.75rem;
            color: var(--ndc-text-white);
            background-color: var(--ndc-primary-color);
            white-space: nowrap;
        }
        
        .ndc\\:btn-outline-secondary {
            display: flex;
            align-items: center;
            justify-content: center;
            height: 36px;
            min-height: 36px;
            padding: 4px 8px;
            border: 1px solid rgb(212 212 216);
            border-radius: 0.25rem;
            background-color: rgb(41 41 46);
            color: rgb(212 212 216);
            font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            text-transform: uppercase;
            text-align: center;
            cursor: pointer;
            transition: color 0.15s, background-color 0.15s, border-color 0.15s;
            box-sizing: border-box;
            appearance: button;
        }
        
        .ndc\\:btn-outline-secondary:hover {
            background-color: rgb(51 51 56);
        }
        
        .ndc\\:btn-outline-secondary:disabled {
            background-color: rgba(51 51 56 / 0.5);
            cursor: not-allowed;
        }
        
        .ndc\\:btn-primary {
            min-height: 2.25rem;
            padding: 0.25rem;
            border-radius: 5px;
            background-color: var(--ndc-primary-color);
            color: var(--ndc-text-white);
            font: 600 0.875rem/1 "Montserrat", sans-serif;
            text-transform: uppercase;
            letter-spacing: 0.05em;
            cursor: pointer;
            transition: background-color 0.3s;
            border: none;
            outline: none;
        }
        
        .ndc\\:btn-primary:disabled {
            background-color: rgba(217 143 64 / 0.5);
            color: rgba(255 255 255 / 0.5);
            cursor: not-allowed;
        }
        
        .ndc-import-btn { border-radius: 0.25rem 0 0 0.25rem; }
        .ndc-import-btn-info { border-radius: 0 0.25rem 0.25rem 0; }
        .ndc-download-btn-all {
            display: flex;
            align-items: center;
            justify-content: space-between;
            gap: 0.5rem;
            width: 100%;
            border-radius: 0.25rem 0 0 0.25rem;
        }
        .ndc-download-btn-menu { border-radius: 0 0.25rem 0.25rem 0; }
        .ndc-pause-btn { border-radius: 0; }
        .ndc-stop-btn { border-radius: 0 0.25rem 0.25rem 0; }
        
        .ndc-dropdown {
            position: absolute;
            right: 0;
            top: 0;
            transform: translate3d(0, 38px, 0);
            min-width: 12rem;
            padding: 0.25rem 0;
            border: 1px solid rgba(255 255 255 / 0.2);
            border-radius: 6px;
            background-color: rgb(29 29 33);
            color: rgb(244 244 245);
            font: 400 16px/24px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            box-shadow: 0 9px 12px 1px rgba(0 0 0 / 0.14),
                        0 3px 16px 2px rgba(0 0 0 / 0.12),
                        0 5px 6px 0 rgba(0 0 0 / 0.2);
            z-index: 10;
            display: none;
        }
        
        .ndc-dropdown-item {
            display: flex;
            align-items: center;
            justify-content: space-between;
            height: 44px;
            padding: 8px;
            background-color: transparent;
            color: rgb(244 244 245);
            font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            text-transform: uppercase;
            white-space: nowrap;
            border: 0;
            cursor: pointer;
            width: 100%;
            text-align: left;
        }
        
        .ndc-dropdown-item:hover {
            background-color: var(--ndc-primary-color);
        }
        
        .ndc-progress-bar {
            display: block;
            flex: 1;
            height: 36px;
            min-height: 36px;
            border-radius: 0.25rem;
            background-color: rgb(41 41 46);
            color: rgb(244 244 245);
            font: 400 14px/24px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            overflow: hidden;
            position: relative;
            width: 100%;
        }
        
        .ndc-progress-bar-fill {
            position: absolute;
            top: 0;
            left: 0;
            height: 36px;
            width: 0;
            background-color: var(--ndc-primary-color);
            color: rgb(244 244 245);
            font: 400 14px/24px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            transition: width 0.3s ease;
        }
        
        .ndc-progress-bar-text-container {
            display: grid;
            grid-template-columns: repeat(3, 1fr);
            align-items: center;
            position: absolute;
            top: 0;
            left: 0;
            height: 36px;
            width: 100%;
            color: var(--ndc-text-white);
            font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            text-transform: uppercase;
            cursor: pointer;
        }
        
        .ndc-progress-bar-text-base {
            height: 14px;
            color: var(--ndc-text-white);
            font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
            text-transform: uppercase;
        }
        
        .ndc-progress-bar-text-progress { margin-left: 8px; }
        .ndc-progress-bar-text-center { text-align: center; }
        .ndc-progress-bar-text-right { margin-right: 8px; text-align: right; }
        
        .ndc-modal-backdrop {
            position: fixed;
            inset: 0;
            display: flex;
            justify-content: center;
            align-items: center;
            background-color: rgba(0 0 0 / 0.25);
            backdrop-filter: brightness(50%);
            z-index: 9999;
        }
        
        .ndc-modal {
            display: flex;
            flex-direction: column;
            width: 100%;
            max-width: 850px;
            height: calc(100vh - 3.5rem);
            padding: 1rem;
            border-radius: 0.5rem;
            background-color: rgb(29 29 33);
        }
        
        .ndc-modal-header,
        .ndc-modal-filter {
            display: flex;
            justify-content: space-between;
            align-items: center;
            margin-bottom: 0.5rem;
            gap: 0.5rem;
        }
        
        .ndc-modal-header-title {
            font: 600 1.125rem "Montserrat", sans-serif;
            text-transform: uppercase;
        }
        
        .ndc-modal-header-dropdown-btn {
            padding: 0.25rem;
            border-radius: 0.25rem;
        }
        
        .ndc-modal-filter input,
        .ndc-modal-filter select {
            padding: 0.25rem;
            border: 1px solid rgb(212 212 216);
            border-radius: 0.25rem;
            flex: 0 1 auto;
            color: #000;
            width: 100%;
            height: 100%;
            box-sizing: border-box;
        }
        
        .ndc-modal-mods-list {
            display: block;
            height: 100%;
            margin-bottom: 0.5rem;
            overflow-y: auto;
        }
        
        .ndc-modal-mods-list-header {
            display: none;
            gap: 0.5rem;
            border: 1px solid hsla(0 0% 100% / 0.2);
            padding: 0.5rem;
            border-radius: 0.25rem;
            cursor: pointer;
            user-select: none;
        }
        
        .ndc-modal-mods-list-header span {
            font: 600 0.875rem "Montserrat", sans-serif;
            text-transform: uppercase;
            color: rgb(161 161 170);
        }
        
        .ndc-modal-mods-list-body {
            display: flex;
            flex-direction: column;
            gap: 0.5rem;
        }
        
        .ndc-modal-mods-list-body-row {
            border: 1px solid hsla(0 0% 100% / 0.2);
            padding: 0.5rem;
            border-radius: 0.25rem;
            cursor: pointer;
            user-select: none;
        }
                
        .ndc-modal-actions {
            display: flex;
            justify-content: end;
            gap: 0.5rem;
        }
        
        @media (min-width: 640px) {
            .ndc-modal-filter input,
            .ndc-modal-filter select { width: auto; }
            .ndc-modal-mods-list-header { display: flex; border-radius: 0; }
            .ndc-modal-mods-list-body { gap: 0; }
            .ndc\\:sm\\:block { display: block; }
            .ndc\\:sm\\:hidden { display: none; }
            .ndc\\:sm\\:flex { display: flex; }
            .ndc\\:sm\\:flex-none { flex: none; }
            .ndc\\:sm\\:gap-0\\.5 { gap: 0.5rem; }
        }
        `);

    const convertSize = (/** @type {number} */ sizeInByte) => {
        // 3769655540 => 3.51 GB
        const units = ["B", "KB", "MB", "GB", "TB"];
        let i = 0;
        let size = sizeInByte;
        while (size >= 1024) {
            size /= 1024;
            i++;
        }
        return `${size.toFixed(2)} ${units[i]}`;
    };

    // Custom error classes
    class NDCDownloadError extends Error {
        /**
         * @param {string | undefined} message
         */
        constructor(message) {
            super(message);
            this.name = 'DownloadError';
        }
    }

    class NDCCaptchaError extends NDCDownloadError {
        /**
         * @param {string} url
         */
        constructor(url) {
            super(`Captcha required for ${url}`);
            this.name = 'CaptchaError';
            this.url = url;
        }
    }

    class NDCSuspendedError extends NDCDownloadError {
        constructor() {
            super('Account temporarily suspended');
            this.name = 'SuspendedError';
        }
    }

    class NDCRateLimitError extends NDCDownloadError {
        constructor() {
            super('Too many requests');
            this.name = 'RateLimitError';
        }
    }

    class Mod {
        /**
         * @param {string} modName
         * @param {string} url
         * @param {number} size
         * @param {number} gameId
         * @param {number} modId
         * @param {number} fileId
         * @param {string} fileName
         */
        constructor(modName, url, size, gameId, modId, fileId, fileName) {
            this.modName = modName;
            this.url = url;
            this.size = size;
            this.gameId = gameId;
            this.modId = modId;
            this.fileId = fileId;
            this.fileName = fileName;
        }
    }

    class NDC {
        /** @type {NDCDownloadButton} */ downloadButton
        /** @type {NDCProgressBar} */ progressBar
        /** @type {NDCLogConsole} */ console
        /** @type {Mod[]} */ mods = []
        /** @type {HTMLDivElement} */ element

        constructor() {
            this.element = this.createElement();
            this.initComponents();
        }

        /**
         * Creates a styled <div> element with predefined styles.
         *
         * @returns {HTMLDivElement} A <div> element with custom styles applied.
         */
        createElement() {
            const div = document.createElement("div");
            Object.assign(div.style, {
                borderRadius: "0.5rem",
                border: "2px solid rgb(217 143 64)",
                padding: "1rem",
                marginTop: "1rem",
                backgroundColor: "rgb(17 17 17)",
                backgroundImage: "url()",
                backgroundSize: "100%",
                backgroundPosition: "center",
                backgroundRepeat: "no-repeat",
                width: "100%"
            });
            return div;
        }

        /**
         * Initializes the UI components for the application.
         * Creates instances of NDCDownloadButton, NDCProgressBar, and NDCLogConsole,
         * and appends their elements to the parent element.
         */
        initComponents() {
            this.downloadButton = new NDCDownloadButton(this);
            this.progressBar = new NDCProgressBar(this);
            this.console = new NDCLogConsole(this);

            this.element.append(
                this.downloadButton.element,
                this.progressBar.element,
                this.console.element
            );
        }

        /**
         * Fetches the game ID associated with a given game domain from the Nexus Mods API.
         *
         * @param {string} gameDomain - The domain name of the game (e.g., "skyrim", "fallout4").
         * @returns {Promise<number|null>} A promise that resolves to the game ID as a string if found, or null if not found or an error occurs.
         */
        async fetchGameId(gameDomain) {
            try {
                const response = await fetch("https://api-router.nexusmods.com/graphql", {
                    method: "POST",
                    headers: { "content-type": "application/json" },
                    referrer: document.location.href,
                    referrerPolicy: "strict-origin-when-cross-origin",
                    body: JSON.stringify({
                        query: "query GameDomainToId($gameDomain: String!) { game(domainName: $gameDomain) { id } }",
                        variables: { gameDomain },
                        operationName: "GameDomainToId"
                    }),
                    mode: "cors",
                    credentials: "include"
                });
                const data = await response.json();
                return data?.data?.game?.id || null;
            } catch {
                return null;
            }
        }

        /**
         * Fetches the download link for a given mod from Nexus Mods.
         *
         * @param {Mod} mod - The mod object to fetch the download link for.
         * @returns {Promise<string>} The download URL of the mod.
         * @throws {NDCCaptchaError} If a CAPTCHA is encountered during the request.
         * @throws {NDCSuspendedError} If the account is temporarily suspended.
         * @throws {NDCRateLimitError} If the request is rate-limited.
         */
        async fetchDownloadLink(mod) {
            this.bypassNexusAdsCookie();

            const downloadResponse = await fetch(
                "https://www.nexusmods.com/Core/Libs/Common/Managers/Downloads?GenerateDownloadUrl",
                {
                    method: "POST",
                    headers: { "content-type": "application/x-www-form-urlencoded; charset=UTF-8" },
                    body: `fid=${mod.fileId}&game_id=${mod.gameId}`
                }
            );

            if (!downloadResponse.ok && downloadResponse.status === 429) {
                const text = await downloadResponse.text();
                if (text.includes("Just a moment...")) throw new NDCCaptchaError(mod.url);
                if (text.includes("temporarily suspended")) throw new NDCSuspendedError();
                throw new NDCRateLimitError();
            }

            const fileLink = await downloadResponse.json();

            return fileLink?.url || "";
        }


        /**
         * Sets a cookie to bypass Nexus Mods ads by simulating an "ab" cookie with a short expiration time.
         * 
         * The cookie is set to expire in 5 minutes and is scoped to the "nexusmods.com" domain.
         */
        bypassNexusAdsCookie() {
            const expiry = new Date(Date.now() + 5 * 60 * 1000).toUTCString();
            document.cookie = `ab=0|${Math.round(Date.now() / 1000) + 300};expires=${expiry};domain=nexusmods.com;path=/`;
        }

        /**
         * Downloads a list of mods while handling various errors and download states.
         * 
         * @async
         * @param {Mod[]} mods - The list of mods to download.
         * @throws {Error} Throws an error if a critical issue occurs during the download process.
         * 
         * @description
         * This method manages the download process for a list of mods. It initializes the download,
         * processes each mod sequentially, and handles errors such as rate limits, captchas, and account suspensions.
         * The method also supports pausing and resuming downloads and logs failed downloads for further review.
         * 
         * Error Handling:
         * - Captcha errors: Pauses the download and waits for the user to solve the captcha.
         * - Account suspension: Waits for 10 minutes before retrying.
         * - Rate limiting: Waits for 5 minutes before retrying.
         * - Other errors: Logs the error and stops the download process.
         * 
         * Workflow:
         * 1. Initializes the download process and progress bar.
         * 2. Iterates through the list of mods, skipping or stopping as necessary.
         * 3. Fetches the download link for each mod and handles success or failure.
         * 4. Waits for a delay between downloads if required.
         * 5. Logs any failed downloads at the end of the process.
         * 6. Finalizes the download process.
         */
        async downloadMods(mods) {
            this.initializeDownload(mods.length);

            try {
                const downloadState = { count: 0 };
                const failedMods = [];
                let currentIndex = 0;

                while (currentIndex < mods.length) {
                    const mod = mods[currentIndex];
                    if (this.shouldSkipDownload(currentIndex, mods.length)) {
                        currentIndex++;
                        continue;
                    }
                    if (this.progressBar.state.status === NDCProgressBar.STATUS.STOPPED) {
                        this.console.log("Download stopped.", NDCLogConsole.TYPE.INFO);
                        break;
                    }

                    const modNum = `${(currentIndex + 1).toString().padStart(mods.length.toString().length, "0")}/${mods.length}`;

                    try {
                        const downloadUrl = await this.fetchDownloadLink(mod);

                        if (!downloadUrl) {
                            this.handleDownloadError(mod, modNum, false, failedMods);
                            currentIndex++;
                        } else {
                            this.handleDownloadSuccess(mod, modNum, downloadUrl, downloadState);
                            currentIndex++;
                        }
                    } catch (error) {
                        if (error instanceof NDCCaptchaError) {
                            const url = error.url;
                            this.console.logError(
                                `You are rate limited by Cloudflare. <a href="${url}" target="_blank" class="ndc:text-primary">Solve captcha</a> then unpause to retry.`
                            );
                            this.progressBar.setStatus(NDCProgressBar.STATUS.PAUSED);
                            await this.waitForUnpause();
                        } else if (error instanceof NDCSuspendedError) {
                            this.console.logError("Account temporarily suspended. Waiting 10 minutes...");
                            await this.waitWithCountdown(10 * 60, "Waiting 10 minutes due to suspension...");
                        } else if (error instanceof NDCRateLimitError) {
                            this.console.logError("Too many requests. Waiting 5 minutes...");
                            await this.waitWithCountdown(5 * 60, "Waiting 5 minutes due to rate limit...");
                        } else {
                            this.console.logError(error.message);
                            this.handleDownloadError(mod, modNum, true, failedMods);
                            this.console.logError("Download forced to stop due to an error.");
                            break;
                        }
                    }

                    if (currentIndex < mods.length) {
                        await this.handleDownloadDelay(downloadState);
                    }
                }

                if (failedMods.length) this.logFailedDownloads(failedMods);
            } catch (error) {
                this.console.logError("An error occurred during the download.");
                console.error(error);
            }

            this.finalizeDownload();
        }

        /**
         * Waits for the progress bar to exit the paused state.
         * This function continuously checks the status of the progress bar
         * and resolves the promise once the status is no longer "PAUSED".
         *
         * @async
         * @returns {Promise<void>} A promise that resolves when the progress bar is unpaused.
         */
        async waitForUnpause() {
            return new Promise(resolve => {
                const checkUnpause = setInterval(() => {
                    if (this.progressBar.state.status !== NDCProgressBar.STATUS.PAUSED) {
                        clearInterval(checkUnpause);
                        resolve();
                    }
                }, 100);
            });
        }

        /**
         * Determines whether the current download should be skipped based on the progress bar's state.
         *
         * @param {number} index - The index of the current download in the list.
         * @param {number} total - The total number of downloads.
         * @returns {boolean} - Returns `true` if the download should be skipped, otherwise `false`.
         */
        shouldSkipDownload(index, total) {
            if (this.progressBar.state.skipTo && this.progressBar.state.skipToIndex - 1 > index) {
                this.console.log(`[${(index + 1).toString().padStart(total.toString().length, "0")}/${total}] Skipping <a href="${this.mods[index].url}" target="_blank" class="ndc:text-primary">${this.mods[index].modName}</a>`);
                this.progressBar.incrementProgress();
                if (this.progressBar.state.skipToIndex - 1 === index + 1) {
                    this.progressBar.state.skipTo = false;
                }
                return true;
            }
            this.progressBar.state.skipTo = false;
            return false;
        }

        /**
         * Handles the error that occurs when a download link for a mod cannot be retrieved.
         *
         * @param {Mod} mod - The mod object containing details about the mod.
         * @param {string} modNum - The numerical identifier of the mod.
         * @param {boolean} critical - Indicates whether the error is critical.
         * @param {Mod[]} failedMods - An array to store mods that failed to download if the error is not critical.
         */
        handleDownloadError(mod, modNum, critical, failedMods) {
            const logRow = this.console.logError(
                `[${modNum}] Failed to get download link for <a href="${mod.url}" target="_blank" class="ndc:text-primary">${mod.modName}</a> <button class="ndc:text-primary"><svg style="height: .75rem;" viewBox="0 3 24 17" xmlns="http://www.w3.org/2000/svg"><path style="fill: currentcolor;" d="M19,21H8V7H19M19,5H8A2,2 0 0,0 6,7V21A2,2 0 0,0 8,23H19A2,2 0 0,0 21,21V7A2,2 0 0,0 19,5M16,1H4A2,2 0 0,0 2,3V17H4V3H16V1Z"/></svg></button>`,
            );
            logRow.querySelector("button")?.addEventListener("click", () => {
                navigator.clipboard.writeText("Response not available");
                alert("Response copied to clipboard");
            });
            if (!critical) failedMods.push(mod);
        }

        /**
         * Handles the successful download of a mod by logging the download details,
         * creating a download link, and updating the progress bar and download state.
         *
         * @param {Mod} mod - The mod object containing details about the mod.
         * @param {string} modNum - The index or number of the mod being downloaded.
         * @param {string} downloadUrl - The URL from which the mod is being downloaded.
         * @param {{count: number}} downloadState - The download state object containing the download count.
         */
        handleDownloadSuccess(mod, modNum, downloadUrl, downloadState) {
            this.console.log(
                `[${modNum}] Downloading <a href="${mod.url}" target="_blank" class="ndc:text-primary">${mod.modName}</a><a href="${downloadUrl}"><svg style="height: .75rem;" class="ndc:text-white" xmlns="http://www.w3.org/2000/svg" viewBox="0 3 24 17"><path style="fill: currentcolor;" d="M5,20H19V18H5M19,9H15V3H9V9H5L12,16L19,9Z"/></svg></a><span style="font-size: .75rem; color: rgb(161 161 170)">(${convertSize(mod.size)})</span>`
            );
            const link = document.createElement("a");
            link.href = downloadUrl;
            link.download = mod.fileName;
            link.click();
            this.progressBar.incrementProgress();
            downloadState.count++;
        }

        /**
         * @param {{ count: number; }} downloadState
         */
        async handleDownloadDelay(downloadState) {
            if (downloadState.count >= 200) {
                await this.waitWithCountdown(5 * 60, "Waiting 5 minutes to avoid Nexus ban...");
                downloadState.count = 0;
            }

            await this.waitWithCountdown(1, "Waiting before next download...");
        }

        /**
         * Waits for a specified number of seconds while displaying a countdown message.
         * The countdown can be interrupted by certain states of the progress bar.
         * @async
         * 
         * @param {number} seconds - The number of seconds to wait.
         * @param {string} initialMessage - The initial message to display in the log.
         * @returns {Promise<void>} A promise that resolves when the countdown completes or is interrupted.
         */
        async waitWithCountdown(seconds, initialMessage) {
            let remaining = seconds;
            let logRow = this.console.logInfo(initialMessage);

            return new Promise(resolve => {
                const interval = setInterval(() => {
                    if (this.progressBar.state.skipPause || this.progressBar.state.skipTo ||
                        this.progressBar.state.status === NDCProgressBar.STATUS.STOPPED) {
                        this.progressBar.state.skipPause = false;
                        clearInterval(interval);
                        logRow.remove();
                        resolve();
                        return;
                    }

                    if (this.progressBar.state.status === NDCProgressBar.STATUS.PAUSED) return;

                    remaining--;
                    const mins = Math.floor(remaining / 60);
                    const secs = remaining % 60;
                    logRow.innerHTML = `Waiting ${mins} minutes and ${secs} seconds...`;

                    if (remaining <= 0) {
                        clearInterval(interval);
                        logRow.remove();
                        resolve();
                    }
                }, 1000);
            });
        }

        /**
         * Logs the list of failed mod downloads to the console.
         *
         * @param {Mod[]} failedMods - The list of mods that failed to download.
         */
        logFailedDownloads(failedMods) {
            this.console.logInfo(`Failed to download ${failedMods.length} mods:`);
            failedMods.forEach(mod =>
                this.console.logInfo(`<a href="${mod.url}" target="_blank" class="ndc:text-primary">${mod.modName}</a>`)
            );
        }

        /**
         * Initializes the download process by setting up the progress bar, 
         * updating its status, and hiding the download button.
         *
         * @param {number} modsCount - The total number of mods to be downloaded.
         */
        initializeDownload(modsCount) {
            this.progressBar.setModsCount(modsCount);
            this.progressBar.setProgress(0);
            this.progressBar.setStatus(NDCProgressBar.STATUS.DOWNLOADING);
            this.downloadButton.element.style.display = "none";
            this.progressBar.element.style.display = "flex";
            this.console.logInfo("Download started.");
        }

        /**
         * Finalizes the download process by updating the progress bar status,
         * hiding the progress bar, displaying the download button, and logging
         * a completion message to the console.
         */
        finalizeDownload() {
            this.progressBar.setStatus(NDCProgressBar.STATUS.FINISHED);
            this.progressBar.element.style.display = "none";
            this.downloadButton.element.style.display = "flex";
            this.console.logInfo("Download finished.");
        }
    }

    class NDCDownloadButton {
        /** @type {HTMLButtonElement | null} */ importBtn
        /** @type {HTMLButtonElement | null} */ infoBtn
        /** @type {HTMLButtonElement | null} */ downloadAllBtn
        /** @type {HTMLButtonElement | null} */ selectBtn
        /** @type {HTMLButtonElement | null} */ menuBtn
        /** @type {HTMLElement | null} */ dropdown
        /** @type {HTMLElement | null} */ modsCount

        /** @param {NDC} ndc */
        constructor(ndc) {
            this.ndc = ndc;
            this.element = this.createElement();
            this.setupElements();
            this.attachEventListeners();
            this.render();
        }

        createElement() {
            const div = document.createElement("div");
            div.id = "ndc-download-button";
            Object.assign(div.style, {
                display: "flex",
                flexDirection: "column",
                gap: "1rem",
                width: "100%"
            });

            div.innerHTML = `
                <div style="display: flex; justify-content: center;">
                    <button class="ndc:btn-outline-secondary ndc-import-btn ndc:flex-1 ndc:sm:flex-none">Import Wabbajack modlist</button>
                    <button class="ndc:btn-outline-secondary ndc-import-btn-info">
                        <svg style="width: 1.5rem; height: 1.5rem; cursor: pointer; fill: currentcolor;" 
                             xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
                            <title>information</title>
                            <path d="M13,9H11V7H13M13,17H11V11H13M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2Z"/>
                        </svg>
                    </button>
                </div>
                <div style="display: flex; width: 100%;">
                    <button class="ndc:btn-primary ndc-download-btn-all">
                        download all mods
                        <span style="padding: 0.5rem; background: rgba(29, 29, 33, 0.8); border-radius: 5px; font-size: 0.75rem;" class="ndc:text-white">
                            <span class="mods-number"></span> mods
                        </span>
                    </button>
                    <div style="position: relative;">
                        <button class="ndc:btn-primary ndc-download-btn-menu">
                            <svg style="width: 1.5rem; height: 1.5rem;" 
                                 xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
                                <path style="fill: currentcolor;" d="M7.41,8.58L12,13.17L16.59,8.58L18,10L12,16L6,10L7.41,8.58Z"/>
                            </svg>
                        </button>
                        <div class="ndc-dropdown">
                            <button class="ndc-dropdown-item">Select mods to download</button>
                        </div>
                    </div>
                </div>
            `;
            return div;
        }

        setupElements() {
            this.importBtn = this.element.querySelector(".ndc-import-btn");
            this.infoBtn = this.element.querySelector(".ndc-import-btn-info");
            this.downloadAllBtn = this.element.querySelector(".ndc-download-btn-all");
            this.modsCount = this.element.querySelector(".mods-number");
            this.menuBtn = this.element.querySelector(".ndc-download-btn-menu");
            this.selectBtn = this.element.querySelector(".ndc-dropdown-item");
            this.dropdown = this.element.querySelector(".ndc-dropdown");
        }

        attachEventListeners() {
            this.importBtn?.addEventListener("click", () => this.handleFileImport());
            this.infoBtn?.addEventListener("click", () => this.showImportInfo());
            this.downloadAllBtn?.addEventListener("click", () => this.ndc.downloadMods(this.ndc.mods));
            this.selectBtn?.addEventListener("click", () => this.showSelectModsModal());
            this.menuBtn?.addEventListener("click", () => this.toggleDropdown());
            document.addEventListener("click", (e) => this.closeDropdownOnOutsideClick(e));
        }

        /**
         * Handles the processing of a Wabbajack file, extracting and validating its contents,
         * and processing Nexus mods information for rendering.
         *
         * @async
         * @param {Blob} file - The Wabbajack file to process.
         * @returns {Promise<void>} Resolves when the file is successfully processed or logs errors if any issues occur.
         *
         * @throws {Error} Logs errors for various failure points, including:
         * - Missing or invalid file input.
         * - Issues reading or extracting the zip file.
         * - Invalid or missing "modlist" entry in the zip file.
         * - Parsing errors or invalid structure in the "modlist" JSON.
         * - Errors while processing individual mods.
         *
         * @example
         * const fileInput = document.querySelector('#fileInput');
         * fileInput.addEventListener('change', async (event) => {
         *     const file = event.target.files[0];
         *     await handleWabbajackFile(file);
         * });
         */
        async handleWabbajackFile(file) {
            try {
                // Validate input
                if (!file) {
                    this.ndc.console.logError("No file provided");
                    return;
                }

                // Initialize ZipReader with error handling
                let entries;
                try {
                    // @ts-ignore
                    const zipReader = new zip.ZipReader(new zip.BlobReader(file));
                    entries = await zipReader.getEntries({});
                } catch (zipError) {
                    this.ndc.console.logError("Failed to read zip file: " + zipError.message);
                    return;
                }

                // Check if entries exist
                if (!entries || entries.length === 0) {
                    this.ndc.console.logError("No entries found in zip file");
                    return;
                }

                // Find modlist entry
                const modListEntry = entries.find(entry => entry?.filename === "modlist");
                if (!modListEntry) {
                    this.ndc.console.logError("modlist file not found");
                    return;
                }

                // Extract and parse modlist data
                let modList;
                try {
                    // @ts-ignore
                    modList = await modListEntry.getData(new zip.TextWriter());
                } catch (extractError) {
                    this.ndc.console.logError("Failed to extract modlist: " + extractError.message);
                    return;
                }

                /** @type {WabbajackModlist} */
                let mods;
                try {
                    mods = JSON.parse(modList);
                    if (!mods?.Archives || !Array.isArray(mods.Archives)) {
                        throw new Error("Invalid modlist structure");
                    }
                } catch (parseError) {
                    this.ndc.console.logError("Invalid modlist format: " + parseError.message);
                    return;
                }

                /** @type {NexusModArchive[]} */
                const nexusMods = mods.Archives.filter((/** @type {NexusModArchive} */ mod) =>
                    mod?.State && mod.State['$type'] === "NexusDownloader, Wabbajack.Lib"
                );

                const games = {};
                /** @type {Mod[]} */
                const processedMods = [];

                for (const mod of nexusMods) {
                    try {
                        // Validate mod structure
                        if (!mod?.State || !mod.State.GameName || !mod.State.ModID || !mod.State.FileID) {
                            this.ndc.console.logError(`Skipping invalid mod: ${mod?.Name || 'unknown'}`);
                            continue;
                        }

                        const gameName = mod.State.GameName.toLowerCase() === "moddingtools" ? "site" : mod.State.GameName;
                        let gameId = games[gameName];

                        // Fetch game ID if not cached
                        if (!gameId) {
                            const gameDomain = gameName.toLowerCase();
                            gameId = await this.ndc.fetchGameId(gameDomain);

                            if (!gameId) {
                                this.ndc.console.logError(`Failed to get game id for ${gameName}`);
                                continue;
                            }
                            games[gameName] = gameId;
                        }

                        // Construct mod object with fallback values
                        processedMods.push(
                            new Mod(
                                mod.State.Name || "Unknown Mod",
                                `https://www.nexusmods.com/${mod.State.GameName.toLowerCase()}/mods/${mod.State.ModID}?tab=files&file_id=${mod.State.FileID}`,
                                mod.Size || 0,
                                gameId,
                                mod.State.ModID,
                                mod.State.FileID,
                                mod.Name || "Unknown File"
                            )
                        );
                    } catch (modError) {
                        this.ndc.console.logError(`Error processing mod ${mod?.Name || 'unknown'}: ${modError.message}`);
                        continue;
                    }
                }

                // Update mods array and render
                this.ndc.mods = processedMods;

                this.render();

                this.ndc.console.logInfo(`Wabbajack Modlist loaded successfully. Processed ${processedMods.length} mods.`);
            } catch (error) {
                this.ndc.console.logError("Unexpected error in handleWabbajackFile: " + error.message);
            }
        }

        async handleFileImport() {
            const input = document.createElement("input");
            input.type = "file";
            input.accept = ".wabbajack";

            input.addEventListener("change", async () => {
                if (this.importBtn) {
                    this.importBtn.disabled = true;
                    this.importBtn.innerHTML = `
                    <div class="spinner-border" style="margin-right: 0.25rem;"></div>
                    Importing...
                    `;
                }
                if (input.files && input.files[0]) {
                    await this.handleWabbajackFile(input.files[0]);
                } else {
                    this.ndc.console.logError("No file selected.");
                }
                if (this.importBtn) {
                    this.importBtn.disabled = false;
                    this.importBtn.innerHTML = "Import Wabbajack modlist";
                }
                input.remove();
            });

            input.click();
        }

        showImportInfo() {
            alert(
                "How to import a Wabbajack modlist?\n\n" +
                "1. Download the modlist from Wabbajack.\n" +
                "2. Click on 'Import Wabbajack modlist'.\n" +
                "3. Select the downloaded modlist file (.wabbajack).\n" +
                "This file should be in your Wabbajack installation folder.\n" +
                "(ex: C:\\Wabbajack\\3.7.5.3\\downloaded_mod_lists\\*.wabbajack)\n\n" +
                "The modlist will be loaded and you can download the mods."
            );
        }

        toggleDropdown() {
            if (this.dropdown) {
                this.dropdown.style.display = this.dropdown.style.display === "block" ? "none" : "block";
            }
        }

        /**
         * Handles the closing of a dropdown menu when a click occurs outside of the menu button.
         * 
         * @param {MouseEvent} event - The mouse event triggered by the user's click.
         */
        closeDropdownOnOutsideClick(event) {
            if (this.menuBtn && event.target instanceof Node && !this.menuBtn.contains(event.target) && this.dropdown) {
                this.dropdown.style.display = "none";
            }
        }

        showSelectModsModal() {
            const modal = new NDCSelectModsModal(this.ndc);
            document.body.appendChild(modal.element);
            modal.render();
        }

        updateModsCount() {
            const count = this.ndc.mods.length;
            if (this.modsCount) {
                this.modsCount.textContent = count.toString();
            }
            if (this.downloadAllBtn) {
                this.downloadAllBtn.disabled = count === 0;
            }
            if (this.menuBtn) {
                this.menuBtn.disabled = count === 0;
            }
        }

        render() {
            this.updateModsCount();
        }
    }

    /**
     * Represents a progress bar component for tracking the progress of mod downloads.
     * This class manages the visual representation of progress, including percentage completion,
     * status updates (e.g., downloading, paused, finished, stopped), and user interactions
     * such as pausing, stopping, or skipping downloads.
     */
    class NDCProgressBar {
        /**
         * Enum representing the various statuses of a process.
         * @enum {number}
         * @property {number} DOWNLOADING - Indicates the process is currently downloading.
         * @property {number} PAUSED - Indicates the process is paused.
         * @property {number} FINISHED - Indicates the process has finished.
         * @property {number} STOPPED - Indicates the process has been stopped.
         */
        static STATUS = {
            DOWNLOADING: 0,
            PAUSED: 1,
            FINISHED: 2,
            STOPPED: 3
        };

        /**
         * A mapping of progress bar statuses to their corresponding display text.
         * 
         * @constant {Object} STATUS_LABEL
         * @property {string} [NDCProgressBar.STATUS.DOWNLOADING] - Text displayed when the status is "Downloading...".
         * @property {string} [NDCProgressBar.STATUS.PAUSED] - Text displayed when the status is "Paused".
         * @property {string} [NDCProgressBar.STATUS.FINISHED] - Text displayed when the status is "Finished".
         * @property {string} [NDCProgressBar.STATUS.STOPPED] - Text displayed when the status is "Stopped".
         */
        static STATUS_LABEL = {
            [NDCProgressBar.STATUS.DOWNLOADING]: "Downloading...",
            [NDCProgressBar.STATUS.PAUSED]: "Paused",
            [NDCProgressBar.STATUS.FINISHED]: "Finished",
            [NDCProgressBar.STATUS.STOPPED]: "Stopped"
        };

        /** @type {HTMLElement | null} */ statusText
        /** @type {HTMLButtonElement | null} */ pauseBtn
        /** @type {HTMLButtonElement | null} */ stopBtn
        /** @type {HTMLButtonElement | null} */ skipPauseBtn
        /** @type {HTMLButtonElement | null} */ skipToBtn
        /** @type {HTMLInputElement | null} */ skipInput
        /** @type {HTMLElement | null} */ progressFill
        /** @type {HTMLElement | null} */ progressText
        /** @type {HTMLElement | null} */ countText

        /** @param {NDC} ndc */
        constructor(ndc) {
            this.ndc = ndc;
            this.state = {
                modsCount: 0,
                progress: 0,
                status: NDCProgressBar.STATUS.DOWNLOADING,
                skipPause: false,
                skipTo: false,
                skipToIndex: 0
            };

            this.element = this.createElement();
            this.setupElements();
            this.attachEventListeners();
        }

        createElement() {
            const div = document.createElement("div");
            Object.assign(div.style, {
                display: "none",
                flexWrap: "wrap",
                width: "100%"
            });

            div.innerHTML = `
                <div class="ndc-progress-bar">
                    <div class="ndc-progress-bar-fill"></div>
                    <div class="ndc-progress-bar-text-container">
                        <div class="ndc-progress-bar-text-base ndc-progress-bar-text-progress">0%</div>
                        <div class="ndc-progress-bar-text-base ndc-progress-bar-text-center">Downloading...</div>
                        <div class="ndc-progress-bar-text-base ndc-progress-bar-text-right">0/0</div>
                    </div>
                </div>
                <div style="display: flex;">
                    <button class="ndc:btn-primary ndc-pause-btn">
                        <svg style="width: 1.5rem; height: 1.5rem; fill: currentcolor;" 
                             viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
                            <path d="M14,19H18V5H14M6,19H10V5H6V19Z"/>
                        </svg>
                    </button>
                    <button class="ndc:btn-primary ndc-stop-btn">
                        <svg style="width: 1.5rem; height: 1.5rem; fill: currentcolor;" 
                             viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
                            <path d="M18,18H6V6H18V18Z"/>
                        </svg>
                    </button>
                </div>
                <div style="display: flex; margin: 0.5rem 0; justify-content: flex-end; flex-basis: 100%;">
                    <div style="display: flex; gap: 0.5rem; align-items: center;">
                        <button class="ndc:btn-primary ndc-skip-pause-btn">Skip pause</button>
                        <button class="ndc:btn-primary ndc-skip-to-index-btn">Skip to index</button>
                        <input class="ndc-skip-to-index-input"
                               style="background: rgb(41, 41, 46); border: 1px solid rgb(161, 161, 170); 
                               border-radius: 4px; color: rgb(161, 161, 170); font: 400 16px/24px 'Montserrat', sans-serif; 
                               height: 36px; padding: 0.5rem; width: 80px;" 
                               type="number" min="0" placeholder="Index">
                    </div>
                </div>
            `;
            return div;
        }

        setupElements() {
            this.progressFill = this.element.querySelector(".ndc-progress-bar-fill");
            this.progressText = this.element.querySelector(".ndc-progress-bar-text-progress");
            this.statusText = this.element.querySelector(".ndc-progress-bar-text-center");
            this.countText = this.element.querySelector(".ndc-progress-bar-text-right");
            this.pauseBtn = this.element.querySelector(".ndc-pause-btn");
            this.stopBtn = this.element.querySelector(".ndc-stop-btn");
            this.skipPauseBtn = this.element.querySelector(".ndc-skip-pause-btn");
            this.skipToBtn = this.element.querySelector(".ndc-skip-to-index-btn");
            this.skipInput = this.element.querySelector(".ndc-skip-to-index-input");
        }

        attachEventListeners() {
            this.pauseBtn?.addEventListener("click", () => this.togglePause());
            this.stopBtn?.addEventListener("click", () => this.setStatus(NDCProgressBar.STATUS.STOPPED));
            this.skipPauseBtn?.addEventListener("click", () => this.skipPauseDownload());
            this.skipToBtn?.addEventListener("click", () => this.skipToIndex());
        }

        togglePause() {
            const newStatus = this.state.status === NDCProgressBar.STATUS.DOWNLOADING
                ? NDCProgressBar.STATUS.PAUSED
                : NDCProgressBar.STATUS.DOWNLOADING;
            this.setStatus(newStatus);
        }

        skipPauseDownload() {
            this.setState({ skipPause: true });
            this.setStatus(NDCProgressBar.STATUS.DOWNLOADING);
        }

        skipToIndex() {
            const index = this.skipInput ? Number.parseInt(this.skipInput.value) : 0;
            if (index > this.state.progress && index <= this.state.modsCount) {
                this.setState({ skipTo: true, skipToIndex: index });
                this.setStatus(NDCProgressBar.STATUS.DOWNLOADING);
            }
        }

        /**
         * Updates the current state with the provided new state and triggers a re-render.
         * 
         * @param {Object} newState - An object containing the properties to update in the current state.
         */
        setState(newState) {
            Object.assign(this.state, newState);
            this.render();
        }

        /**
         * Updates the state with the given number of mods.
         *
         * @param {number} count - The number of mods to set.
         */
        setModsCount(count) {
            this.setState({ modsCount: count });
        }

        /**
         * Updates the progress state with the given value.
         *
         * @param {number} progress - The current progress value to set.
         */
        setProgress(progress) {
            this.setState({ progress });
        }

        incrementProgress() {
            this.setProgress(this.state.progress + 1);
        }

        /**
         * Updates the status of the progress bar and its associated text content.
         *
         * @param {number} status - The new status to set. This should correspond to a key in `NDCProgressBar.STATUS_TEXT`.
         */
        setStatus(status) {
            this.setState({ status });
            if (this.statusText) {
                this.statusText.textContent = NDCProgressBar.STATUS_LABEL[status];
            }
        }

        getProgressPercent() {
            return ((this.state.progress / this.state.modsCount) * 100).toFixed(2);
        }

        render() {
            const percent = this.getProgressPercent();
            if (this.progressFill) {
                this.progressFill.style.width = `${percent}%`;
            }
            if (this.progressText) {
                this.progressText.textContent = `${percent}%`;
            }
            if (this.countText) {
                this.countText.textContent = `${this.state.progress}/${this.state.modsCount}`;
            }

            if (this.pauseBtn) {
                this.pauseBtn.innerHTML = this.state.status === NDCProgressBar.STATUS.PAUSED
                    ? '<svg style="width: 1.5rem; height: 1.5rem;" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path style="fill: currentcolor;" d="M8,5.14V19.14L19,12.14L8,5.14Z"/></svg>'
                    : '<svg style="width: 1.5rem; height: 1.5rem;" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path style="fill: currentcolor;" d="M14,19H18V5H14M6,19H10V5H6V19Z"/></svg>';
            }
        }
    }

    class NDCSelectModsModal {
        /** @type {HTMLElement | null} */ dropdown
        /** @type {HTMLButtonElement | null} */ dropdownBtn
        /** @type {HTMLElement | null} */ modsList
        /** @type {HTMLElement | null} */ selectedCount
        /** @type {HTMLInputElement | null} */ searchInput
        /** @type {HTMLSelectElement | null} */ sortSelect
        /** @type {HTMLButtonElement | null} */ cancelBtn
        /** @type {HTMLButtonElement | null} */ downloadBtn

        /** @param {NDC} ndc */
        constructor(ndc) {
            this.ndc = ndc;
            this.element = this.createElement();
            this.setupElements();
            this.attachBasicListeners();
        }

        createElement() {
            const div = document.createElement("div");
            div.className = "ndc-modal-backdrop";
            div.innerHTML = `
            <div class="ndc-modal">
                <div class="ndc-modal-header">
                    <h2 class="ndc-modal-header-title">Select mods</h2>
                    <div style="display: flex; gap: .5rem;">
                        <div style="display: flex; align-items: center;">
                            <span class="ndc:badge-primary">0 mods selected</span>
                        </div>
                        <div style="position: relative;">
                            <button class="ndc:btn-outline-secondary ndc-modal-header-dropdown-btn">
                                <svg style="width: 1.5rem; height: 1.5rem; fill: currentcolor;" 
                                     xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
                                    <path d="M12,16A2,2 0 0,1 14,18A2,2 0 0,1 12,20A2,2 0 0,1 10,18A2,2 0 0,1 12,16M12,10A2,2 0 0,1 14,12A2,2 0 0,1 12,14A2,2 0 0,1 10,12A2,2 0 0,1 12,10M12,4A2,2 0 0,1 14,6A2,2 0 0,1 12,8A2,2 0 0,1 10,6A2,2 0 0,1 12,4Z"/>
                                </svg>
                            </button>
                            <div class="ndc-dropdown">
                                <button class="ndc-dropdown-item ndc-select-all">Select all<svg style="width: 1.5rem; height: 1.5rem;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: currentcolor;" d="M0.41,13.41L6,19L7.41,17.58L1.83,12M22.24,5.58L11.66,16.17L7.5,12L6.07,13.41L11.66,19L23.66,7M18,7L16.59,5.58L10.24,11.93L11.66,13.34L18,7Z"/></svg></button>
                                <button class="ndc-dropdown-item ndc-deselect-all">Deselect all<svg style="width: 1.5rem; height: 1.5rem;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: currentcolor;" d="M19,6.41L17.59,5L12,10.59L6.41,5L5,6.41L10.59,12L5,17.59L6.41,19L12,13.41L17.59,19L19,17.59L13.41,12L19,6.41Z"/></svg></button>
                                <button class="ndc-dropdown-item ndc-invert-selection">Invert selection<svg style="width: 1.5rem; height: 1.5rem;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: currentcolor;" d="M12 2C6.5 2 2 6.5 2 12S6.5 22 12 22 22 17.5 22 12 17.5 2 12 2M6.5 9L10 5.5L13.5 9H11V13H9V9H6.5M17.5 15L14 18.5L10.5 15H13V11H15V15H17.5Z"/></svg></button>
                                <div class="border-t border-stroke-subdued"></div>
                                <button class="ndc-dropdown-item">Export mods selection<svg style="width: 1.5rem; height: 1.5rem;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: currentcolor;" d="M9,16V10H5L12,3L19,10H15V16H9M5,20V18H19V20H5Z"/></svg></button>
                                <button class="ndc-dropdown-item">Import mods selection<svg style="width: 1.5rem; height: 1.5rem;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: currentcolor;" d="M5,20H19V18H5M19,9H15V3H9V9H5L12,16L19,9Z"/></svg></button>
                                <div class="border-t border-stroke-subdued"></div>
                                <button class="ndc-dropdown-item">Import downloaded mods<svg style="width: 1.5rem; height: 1.5rem;" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: currentcolor;" d="M5,20H19V18H5M19,9H15V3H9V9H5L12,16L19,9Z"/></svg></button>
                            </div>
                        </div>
                    </div>
                </div>
                <div class="ndc-modal-filter">
                    <input type="search" placeholder="Search mods...">
                    <select>
                        <option value="mod_name_asc">Order by mod name ASC</option>
                        <option value="mod_name_desc">Order by mod name DESC</option>
                        <option value="file_name_asc">Order by file name ASC</option>
                        <option value="file_name_desc">Order by file name DESC</option>
                        <option value="size_asc">Order by size ASC</option>
                        <option value="size_desc">Order by size DESC</option>
                    </select>
                </div>
                <div class="ndc-modal-mods-list">
                    <div class="ndc-modal-mods-list-header">
                        <span style="width: 3rem;">Index</span>
                        <span style="flex: 1;">Mod name</span>
                        <span style="flex: 1;">File name</span>
                        <span style="width: 5rem;">Size</span>
                    </div>
                    <div class="ndc-modal-mods-list-body"></div>
                </div>
                <div class="ndc-modal-actions">
                    <button class="ndc:btn-outline-secondary ndc-modal-cancel">Cancel</button>
                    <button class="ndc:btn-primary ndc-modal-download">Download selected mods</button>
                </div>
            </div>
        `;
            return div;
        }

        setupElements() {
            this.selectedCount = this.element.querySelector(".ndc\\:badge-primary");
            this.dropdownBtn = this.element.querySelector(".ndc-modal-header-dropdown-btn");
            this.dropdown = this.element.querySelector(".ndc-dropdown");
            this.selectAllBtn = this.dropdown?.querySelector(".ndc-select-all");
            this.deselectAllBtn = this.dropdown?.querySelector(".ndc-deselect-all");
            this.invertBtn = this.dropdown?.querySelector(".ndc-invert-selection");
            this.exportBtn = this.dropdown?.querySelector(".ndc-dropdown-item:nth-child(5)");
            this.importBtn = this.dropdown?.querySelector(".ndc-dropdown-item:nth-child(6)");
            this.importDownloadedBtn = this.dropdown?.querySelector(".ndc-dropdown-item:nth-child(8)");
            this.searchInput = this.element.querySelector("input[type='search']");
            this.sortSelect = this.element.querySelector("select");
            this.modsList = this.element.querySelector(".ndc-modal-mods-list-body");
            this.cancelBtn = this.element.querySelector(".ndc-modal-cancel");
            this.downloadBtn = this.element.querySelector(".ndc-modal-download");
        }

        attachBasicListeners() {
            this.dropdownBtn?.addEventListener("click", () => this.toggleDropdown());
            this.cancelBtn?.addEventListener("click", () => this.element.remove());
            this.downloadBtn?.addEventListener("click", () => this.downloadSelected());
            document.addEventListener("click", (e) => this.closeDropdownOnOutsideClick(e));
        }

        toggleDropdown() {
            if (this.dropdown) {
                this.dropdown.style.display = this.dropdown.style.display === "block" ? "none" : "block";
            }
        }

        /**
         * Handles the closing of a dropdown menu when a click occurs outside of the dropdown button.
         *
         * @param {MouseEvent} event - The mouse event triggered by the user's click.
         */
        closeDropdownOnOutsideClick(event) {
            if (this.dropdownBtn && event.target instanceof Node && !this.dropdownBtn.contains(event.target) && this.dropdown) {
                this.dropdown.style.display = "none";
            }
        }

        downloadSelected() {
            const selectedMods = this.ndc.mods.filter((mod) => {
                /** @type {HTMLInputElement|null} */
                const checkbox = this.element.querySelector(`#mod_${mod.fileId}`);
                if (checkbox) {
                    return checkbox.checked;
                }
                return false;
            });
            this.element.remove();
            this.ndc.downloadMods(selectedMods);
        }

        /**
         * Updates the mod list displayed in the UI with the provided mods data.
         * 
         * @param {Mod[]} mods - The list of mods to display.
         */
        updateModList(mods) {
            if (this.modsList) {
                // Save the checked state of checkboxes
                const checkedStates = {};
                this.modsList.querySelectorAll("input[type='checkbox']").forEach((checkbox) => {
                    if (checkbox instanceof HTMLInputElement) {
                        checkedStates[checkbox.id] = checkbox.checked;
                    }
                });

                // Update the mods list
                this.modsList.innerHTML = mods.map((mod, index) => `
                    <div class="ndc-modal-mods-list-body-row">
                        <input type="checkbox" id="mod_${mod.fileId}" style="display: none;">
                        <div class="ndc:hidden ndc:sm:flex ndc:sm:gap-0.5">
                            <span style="width: 3rem;" class="ndc:text-primary mod-list-index">#${index + 1}</span>
                            <span style="flex: 1;" class="ndc:text-white">${mod.modName}</span>
                            <span style="flex: 1;" class="ndc:text-white">${mod.fileName}</span>
                            <span style="width: 5rem;" class="ndc:text-white">${convertSize(mod.size)}</span>
                        </div>
                        <div class="ndc:block ndc:sm:hidden">
                            <div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 0.25rem;">
                                <div style="display: flex; item-align: center; gap: 0.5rem;">
                                    <span class="ndc:text-primary mod-list-index">#${index + 1}</span>
                                </div>
                                <div style="display: flex; gap: 0.5rem;">
                                    <span class="ndc:text-white">${convertSize(mod.size)}</span>
                                </div>
                            </div>
                            <div style="display: flex; flex-direction: column; gap: 0.25rem;">
                                <div class="ndc:text-white">${mod.modName}</div>
                                <div class="ndc:text-white">${mod.fileName}</div>
                            </div>
                        </div>
                    </div>
                `).join("");

                // Restore the checked state of checkboxes
                this.modsList.querySelectorAll("input[type='checkbox']").forEach((checkbox) => {
                    if (checkedStates[checkbox.id] !== undefined && checkbox instanceof HTMLInputElement) {
                        checkbox.checked = checkedStates[checkbox.id];
                        const parentElement = checkbox.parentElement;
                        if (parentElement) {
                            this.toggleRowSelection(parentElement, checkbox.checked);
                        }
                    }
                });

                // Reattach event listeners
                this.modsList.querySelectorAll(".ndc-modal-mods-list-body-row").forEach(row => {
                    row.addEventListener("click", (e) => this.handleModClick(row, e));
                });
            }
        }

        /**
         * Handles the click event on a mod row, toggling its selection state and updating the UI accordingly.
         * Supports shift-click functionality for selecting multiple rows at once.
         *
         * @param {HTMLElement|Element} row - The table row element representing the mod that was clicked.
         * @param {MouseEvent|Event} event - The mouse event triggered by the click.
         */
        handleModClick(row, event) {
            const checkbox = row.querySelector("input");
            if (checkbox) {
                checkbox.checked = !checkbox.checked;
                this.toggleRowSelection(row, checkbox.checked);
            }

            if (event instanceof MouseEvent && event.shiftKey && this.modsList?.dataset.lastChecked) {
                this.handleShiftSelection(row);
            }

            if (this.modsList) {
                this.modsList.dataset.lastChecked = Array.from(this.modsList.children).indexOf(row).toString();
            }
            this.updateSelectedCount();
        }

        /**
         * Toggles the checkbox state for a given mod and updates the row selection accordingly.
         *
         * @param {Mod} mod - The mod object containing information about the mod, including its fileId.
         * @param {boolean} [checked] - A boolean indicating whether the checkbox should be checked (true) or unchecked (false).
         * @returns {{ row: HTMLElement, checkbox: HTMLInputElement } | null} 
         *          An object containing the row element and the checkbox element, or null if not found.
         */
        toggleModCheckbox(mod, checked) {
            const parentNode = this.element.querySelector(`#mod_${mod.fileId}`)?.parentNode;
            const row = parentNode instanceof HTMLElement ? parentNode : null;
            if (!row) return null;

            const checkbox = row?.querySelector("input");
            if (!checkbox) return null;
            checkbox.checked = checked !== undefined ? checked : !checkbox.checked;

            this.toggleRowSelection(row, checkbox.checked);
            this.updateSelectedCount();

            return { row, checkbox };
        };

        /**
         * Toggles the selection state of a table row by adding or removing specific CSS classes.
         *
         * @param {HTMLElement | Element} row - The table row element to toggle selection for.
         * @param {boolean | undefined} checked - A boolean indicating whether the row should be marked as selected (true) or deselected (false).
         */
        toggleRowSelection(row, checked) {
            row.classList.toggle("ndc:bg-primary-subdued", checked);
            row.querySelector(".mod-list-index")?.classList.toggle("ndc:text-white", checked);
        }

        /**
         * Handles the selection of multiple rows in a list when the Shift key is pressed.
         * Toggles the checked state and applies/removes CSS classes for styling based on the state.
         *
         * @param {HTMLElement|Element} row - The row element where the Shift+click event occurred.
         */
        handleShiftSelection(row) {
            const start = this.modsList ? Array.from(this.modsList.children).indexOf(row) : -1;
            const end = this.modsList ? Number(this.modsList.dataset.lastChecked) : -1;
            const child = this.modsList?.children[end];
            const input = child?.querySelector("input");
            const checked = input?.checked || false;

            for (let i = Math.min(start, end); i <= Math.max(start, end); i++) {
                const modRow = this.modsList ? this.modsList.children[i] : null;
                const checkbox = modRow ? modRow.querySelector("input") : null;
                if (checkbox) {
                    checkbox.checked = checked;
                }
                if (modRow) {
                    this.toggleRowSelection(modRow, checked);
                }
            }
        }

        updateSelectedCount() {
            const count = this.element.querySelectorAll("input:checked").length;
            if (this.selectedCount) {
                this.selectedCount.textContent = `${count} mods selected`;
            }
        }

        render() {
            this.updateModList(this.ndc.mods);
            this.element.addEventListener("click", (e) => {
                if (e.target === this.element) this.element.remove();
            });

            this.searchInput?.addEventListener("input", () => this.filterMods());
            this.sortSelect?.addEventListener("change", () => this.sortMods());
            this.selectAllBtn?.addEventListener("click", () => this.selectAll());
            this.deselectAllBtn?.addEventListener("click", () => this.deselectAll());
            this.invertBtn?.addEventListener("click", () => this.invertSelection());
            this.exportBtn?.addEventListener("click", () => this.exportSelection());
            this.importBtn?.addEventListener("click", () => this.importSelection());
            this.importDownloadedBtn?.addEventListener("click", () => this.importDownloaded());
        }

        filterMods() {
            const search = this.searchInput?.value.toLowerCase();
            if (search) {
                this.ndc.mods.forEach((mod) => {
                    const row = this.element.querySelector(`#mod_${mod.fileId}`);
                    const parentNode = row?.parentNode instanceof HTMLElement ? row.parentNode : null;
                    if (parentNode) {
                        parentNode.style.display = (mod.modName.toLowerCase().includes(search) ||
                            mod.fileName.toLowerCase().includes(search)) ? "" : "none";
                    }
                });
            }
        }

        sortMods() {
            const sort = this.sortSelect?.value;
            const mods = [...this.ndc.mods].sort((a, b) => {
                switch (sort) {
                    case "mod_name_asc": return a.modName.localeCompare(b.modName);
                    case "mod_name_desc": return b.modName.localeCompare(a.modName);
                    case "file_name_asc": return a.fileName.localeCompare(b.fileName);
                    case "file_name_desc": return b.fileName.localeCompare(a.fileName);
                    case "size_asc": return a.size - b.size;
                    case "size_desc": return b.size - a.size;
                    default: return 0;
                }
            });
            this.updateModList(mods);
        }

        selectAll() { this.toggleAllCheckboxes(true); }

        deselectAll() { this.toggleAllCheckboxes(false); }

        invertSelection() {
            this.ndc.mods.forEach((mod) => this.toggleModCheckbox(mod));
        }

        /**
         * Toggles the state of all checkboxes in the mod list and updates the corresponding row styles.
         *
         * @param {boolean} state - The desired state for all checkboxes (true for checked, false for unchecked).
         */
        toggleAllCheckboxes(state) {
            this.ndc.mods.forEach((mod) => this.toggleModCheckbox(mod, state));
        }

        exportSelection() {
            const selectedMods = this.ndc.mods.filter((mod) => {
                /** @type {HTMLInputElement|null} */
                const row = this.element.querySelector(`#mod_${mod.fileId}`);
                return row && row.checked
            });
            if (!selectedMods.length) return alert("You must select at least one mod to export.");

            const blob = new Blob([JSON.stringify(selectedMods, null, 2)], { type: "application/json" });
            const url = URL.createObjectURL(blob);
            const a = document.createElement("a");
            a.href = url;
            a.download = `ndc_selected_mods_${Date.now()}.json`;
            a.click();
            URL.revokeObjectURL(url);
        }

        importSelection() {
            const input = document.createElement("input");
            input.type = "file";
            input.accept = ".json";
            input.addEventListener("change", () => {
                const reader = new FileReader();
                reader.onload = () => {

                    const result = reader.result;
                    if (typeof result !== "string") {
                        console.error("Unexpected reader result type: " + typeof result);
                        return;
                    }

                    const mods = JSON.parse(result);
                    mods.forEach((/** @type {Mod} */ mod) => this.toggleModCheckbox(mod, true));
                };
                if (input.files && input.files[0]) {
                    reader.readAsText(input.files[0]);
                } else {
                    console.error("No file selected or input.files is null.");
                }
            });
            input.click();
        }

        importDownloaded() {
            const input = document.createElement("input");
            input.type = "file";
            input.multiple = true;
            input.addEventListener("change", () => {
                const files = input.files ? Array.from(input.files) : [];
                const downloaded = this.ndc.mods.filter(mod =>
                    files.some(file => file.name.includes(mod.fileName)));
                const notDownloaded = this.ndc.mods.filter(mod => !downloaded.includes(mod));

                notDownloaded.forEach((mod) => this.toggleModCheckbox(mod, true));

                this.updateSelectedCount();
                alert(notDownloaded.length ?
                    `Selected ${notDownloaded.length} mods not yet downloaded.` :
                    "All mods are already downloaded.");
            });
            input.click();
        }
    }

    class NDCLogConsole {
        /**
         * An enumeration representing different types of messages.
         * @enum {string}
         * @property {string} NORMAL - Represents a normal message type.
         * @property {string} ERROR - Represents an error message type.
         * @property {string} INFO - Represents an informational message type.
         */
        static TYPE = {
            NORMAL: "NORMAL",
            ERROR: "ERROR",
            INFO: "INFO"
        };

        /** @type {boolean} */ hidden = false
        /** @type {HTMLButtonElement | null} */ toggleBtn
        /** @type {HTMLElement | null} */ logContainer

        /** @param {NDC} ndc */
        constructor(ndc) {
            this.ndc = ndc;
            this.element = this.createElement();
            this.setupElements();
            this.attachEventListeners();
        }

        createElement() {
            const div = document.createElement("div");
            Object.assign(div.style, {
                display: "flex",
                flexDirection: "column",
                width: "100%",
                gap: "1rem",
                marginTop: "1rem"
            });

            div.innerHTML = `
            <div style="display: flex; flex-direction: column; width: 100%; gap: 0.75rem;">
                <button style="background: none; border: 0; color: rgb(244, 244, 245); cursor: pointer; 
                              font: 400 16px/24px 'Montserrat', sans-serif; height: 24px; width: 100%;">
                    Hide logs
                </button>
                <div style="background: rgb(29 29 33 / 70%); border: 1px solid rgb(255, 255, 255); 
                           border-radius: 4px; color: rgb(255, 255, 255); 
                           font: 600 14px/21px monospace; height: 160px; 
                           overflow-y: auto; resize: vertical; width: 100%;">
                </div>
            </div>
        `;
            return div;
        }

        setupElements() {
            this.toggleBtn = this.element.querySelector("button");
            this.logContainer = this.element.querySelector("div > div:nth-child(2)");
        }

        attachEventListeners() {
            this.toggleBtn?.addEventListener("click", () => this.toggleVisibility());
        }

        toggleVisibility() {
            this.hidden = !this.hidden;
            if (this.logContainer) {
                this.logContainer.style.display = this.hidden ? "none" : "";
            }
            if (this.toggleBtn) {
                this.toggleBtn.textContent = this.hidden ? "Show logs" : "Hide logs";
            }
        }

        /**
         * Logs a message to the custom log console and the browser console.
         *
         * @param {string} message - The message to log.
         * @param {string} [type=NDCLogConsole.TYPE.NORMAL] - The type of log message. 
         *        Can be one of the following:
         *        - `NDCLogConsole.TYPE.NORMAL` (default): Standard log message.
         *        - `NDCLogConsole.TYPE.ERROR`: Error message, styled in red.
         *        - `NDCLogConsole.TYPE.INFO`: Informational message, styled in blue.
         * @returns {HTMLDivElement} The created log row element.
         */
        log(message, type = NDCLogConsole.TYPE.NORMAL) {
            const row = document.createElement("div");
            Object.assign(row.style, {
                display: "flex",
                gap: "0.25rem",
                padding: "0 0.5rem",
                ...(type === NDCLogConsole.TYPE.ERROR && { color: "rgb(229, 62, 62)" }),
                ...(type === NDCLogConsole.TYPE.INFO && { color: "rgb(96, 165, 250)" })
            });

            row.innerHTML = `<span>[${new Date().toLocaleTimeString()}]</span><span>${message}</span>`;
            if (this.logContainer) {
                this.logContainer.appendChild(row);
                this.logContainer.scrollTop = this.logContainer.scrollHeight;
            }
            console.log(message);

            return row;
        }

        /**
         * Logs a message with the normal log type.
         *
         * @param {string} message - The message to be logged.
         * @returns {HTMLDivElement}
         */
        logNormal(message) {
            return this.log(message, NDCLogConsole.TYPE.NORMAL);
        }

        /**
         * Logs an error message to the console with the error log type.
         *
         * @param {string} message - The error message to be logged.
         * @returns {HTMLDivElement}
         */
        logError(message) {
            return this.log(message, NDCLogConsole.TYPE.ERROR);
        }

        /**
         * Logs an informational message to the console.
         *
         * @param {string} message - The message to be logged.
         * @returns {HTMLDivElement} The result of the log operation.
         */
        logInfo(message) {
            return this.log(message, NDCLogConsole.TYPE.INFO);
        }

        clear() {
            if (this.logContainer) {
                this.logContainer.innerHTML = "";
            }
        }
    }

    let ndc = null;
    async function handleNextRouterChange() {
        ndc = new NDC();

        // set interval to check if ndc.element is still in the DOM, if not re add it
        setInterval(() => {
            if (!document.contains(ndc.element)) {
                document
                    .querySelector("#mainContent > section > div.home-intro")
                    ?.prepend(ndc.element);
            }
        }, 500);
    }

    // Monitor route changes using popstate
    window.addEventListener("popstate", handleNextRouterChange);

    // Handle programmatic navigation (optional, for pushState or replaceState)
    const originalPushState = history.pushState;
    const originalReplaceState = history.replaceState;

    history.pushState = function (...args) {
        originalPushState.apply(this, args);
        window.dispatchEvent(new Event("popstate"));
    };

    history.replaceState = function (...args) {
        originalReplaceState.apply(this, args);
        window.dispatchEvent(new Event("popstate"));
    };

    // Initial call to handle the current route
    handleNextRouterChange();
})();