// ==UserScript==
// @name Nexus Download Wabbajack Modlist
// @namespace NDWM
// @version 0.4
// @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
/**
* @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;
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; }
}
`);
// 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 <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 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"
);
/** @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 = `
<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();