// ==UserScript== // @name Nexus Download Wabbajack Modlist // @namespace NDWM // @version 0.5 // @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/zip.js@2.7.57/dist/zip.min.js // @downloadURL https://update.greasyfork.icu/scripts/530021/Nexus%20Download%20Wabbajack%20Modlist.user.js // @updateURL https://update.greasyfork.icu/scripts/530021/Nexus%20Download%20Wabbajack%20Modlist.meta.js // ==/UserScript== (function() { // MDI : https://pictogrammers.com/library/mdi/ // MDI : https://github.com/MathewSachin/Captura/blob/master/src/Captura.Core/MaterialDesignIcons.cs /** * @typedef {{ * $type: string, * Author?: string, * Description?: string, * FileID: number, * GameName: string, * ImageURL?: string, * IsNSFW?: boolean, * ModID: number, * Name?: string, * Version?: string * }} NexusModState * * @typedef {{ * Hash: string, * Meta: string, * Name: string, * Size: number, * State: NexusModState * }} NexusModArchive * * @typedef {{ * Archives: NexusModArchive[] * }} WabbajackModlist */ // @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; cursor: pointer; user-select: none; } .ndc-modal-mods-list-body-row:last-child { border-radius: 0 0 0.25rem 0.25rem; } .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; } } `); // https://github.com/wabbajack-tools/wabbajack/blob/main/Wabbajack.DTOs/Game/GameRegistry.cs const wabbajackGames = { "Morrowind": { "NexusName": "morrowind", "NexusGameId": 100 }, "Oblivion": { "NexusName": "oblivion", "NexusGameId": 101 }, "Fallout3": { "NexusName": "fallout3", "NexusGameId": 120 }, "FalloutNewVegas": { "NexusName": "newvegas", "NexusGameId": 130 }, "Skyrim": { "NexusName": "skyrim", "NexusGameId": 110 }, "SkyrimSpecialEdition": { "NexusName": "skyrimspecialedition", "NexusGameId": 1704 }, "Fallout4": { "NexusName": "fallout4", "NexusGameId": 1151 }, "SkyrimVR": { "NexusName": "skyrimspecialedition", "NexusGameId": 1704 }, "Enderal": { "NexusName": "enderal", "NexusGameId": 2736 }, "EnderalSpecialEdition": { "NexusName": "enderalspecialedition", "NexusGameId": 3685 }, "Fallout4VR": { "NexusName": "fallout4", "NexusGameId": 1151 }, "DarkestDungeon": { "NexusName": "darkestdungeon", "NexusGameId": 804 }, "Dishonored": { "NexusName": "dishonored", "NexusGameId": 802 }, "Witcher": { "NexusName": "witcher", "NexusGameId": 150 }, "Witcher3": { "NexusName": "witcher3", "NexusGameId": 952 }, "StardewValley": { "NexusName": "stardewvalley", "NexusGameId": 1303 }, "KingdomComeDeliverance": { "NexusName": "kingdomcomedeliverance", "NexusGameId": 2298 }, "MechWarrior5Mercenaries": { "NexusName": "mechwarrior5mercenaries", "NexusGameId": 3099 }, "NoMansSky": { "NexusName": "nomanssky", "NexusGameId": 1634 }, "DragonAgeOrigins": { "NexusName": "dragonage", "NexusGameId": 140 }, "DragonAge2": { "NexusName": "dragonage2", "NexusGameId": 141 }, "DragonAgeInquisition": { "NexusName": "dragonageinquisition", "NexusGameId": 728 }, "KerbalSpaceProgram": { "NexusName": "kerbalspaceprogram", "NexusGameId": 272 }, "Terraria": { "NexusName": null, "NexusGameId": null }, "Cyberpunk2077": { "NexusName": "cyberpunk2077", "NexusGameId": 3333 }, "Sims4": { "NexusName": "thesims4", "NexusGameId": 641 }, "DragonsDogma": { "NexusName": "dragonsdogma", "NexusGameId": 1249 }, "KarrynsPrison": { "NexusName": null, "NexusGameId": null }, "Valheim": { "NexusName": "valheim", "NexusGameId": 3667 }, "MountAndBlade2Bannerlord": { "NexusName": "mountandblade2bannerlord", "NexusGameId": 3174 }, "FinalFantasy7Remake": { "NexusName": "finalfantasy7remake", "NexusGameId": 4202 }, "BaldursGate3": { "NexusName": "baldursgate3", "NexusGameId": 3474 }, "Starfield": { "NexusName": "starfield", "NexusGameId": 4187 }, "SevenDaysToDie": { "NexusName": "7daystodie", "NexusGameId": 1059 }, "ModdingTools": { "NexusName": "site", "NexusGameId": 2295 } } 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
element with predefined styles. * * @returns {HTMLDivElement} A
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 download link for a given mod from Nexus Mods. * * @param {Mod} mod - The mod object to fetch the download link for. * @returns {Promise} 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. Solve captcha 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} 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 ${this.mods[index].modName}`); 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 ${mod.modName} `, ); 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 ${mod.modName}(${convertSize(mod.size)})` ); 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} 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(`${mod.modName}`) ); } /** * 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 = `
`; 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} 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" ); /** @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 { NexusGameId: gameId, NexusName: gameName } = wabbajackGames[mod.State.GameName] || {}; if (!gameId || !gameName) { this.ndc.console.logError(`Unsupported game: ${mod.State.GameName}`); continue; } // Construct mod object with fallback values processedMods.push( new Mod( mod.State.Name || "Unknown Mod", `https://www.nexusmods.com/${gameName}/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 = `
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 = `
0%
Downloading...
0/0
`; 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 ? '' : ''; } } } 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 = `

Select mods

0 mods selected
Index Mod name File name Size
`; 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) => `
#${index + 1} ${mod.modName} ${mod.fileName} ${convertSize(mod.size)}
#${index + 1}
${convertSize(mod.size)}
${mod.modName}
${mod.fileName}
`).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); const modListIndexes = row.querySelectorAll(".mod-list-index"); modListIndexes.forEach((modListIndex) => { modListIndex.classList.toggle("ndc:text-primary", !checked); modListIndex.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(); // split the search string into words const searchWords = search ? search.split(" ") : []; // check if any of the words are present in the mod name or file name if (searchWords.length > 0) { 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"; const matches = searchWords.every(word => mod.modName.toLowerCase().includes(word) || mod.fileName.toLowerCase().includes(word)); parentNode.style.display = matches ? "" : "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 = `
`; 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 = `[${new Date().toLocaleTimeString()}]${message}`; 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; setInterval(() => { if (window.location.href === "https://www.nexusmods.com/") { if (!ndc || (ndc instanceof NDC && !document.contains(ndc.element))) { ndc = new NDC(); // If NDC is async, await it if needed: await new NDC() document.querySelector("#mainContent > div > div.next-container > section") ?.prepend(ndc.element); } } }, 500); })();