🏠 Home 

Nexus Download Wabbajack Modlist

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


Install this script?
// ==UserScript==
// @name         Nexus Download Wabbajack Modlist
// @namespace    NDWM
// @version      0.2
// @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.xmlHttpRequest
// @grant        GM_xmlhttpRequest
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM.setValue
// @grant        GM.getValue
// @grant        GM_addStyle
// @connect      next.nexusmods.com
// @connect      nexusmods.com
// @require      https://cdn.jsdelivr.net/npm/@zip.js/[email protected]/dist/zip.min.js
// ==/UserScript==
(async function () {
// MDI : https://pictogrammers.com/library/mdi/
// MDI : https://github.com/MathewSachin/Captura/blob/master/src/Captura.Core/MaterialDesignIcons.cs
GM_addStyle(`
.bg-primary-subdued {
background-color: rgb(200 123 40);
}
.text-white {
color: #fff;
}
.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: #fff;
background-color: rgb(217, 143, 64);
white-space: nowrap;
}
.btn-outline-secondary {
align-items: center;
appearance: button;
background-color: rgb(41, 41, 46);
border: 1px solid rgb(212, 212, 216);
box-sizing: border-box;
color: rgb(212, 212, 216);
cursor: pointer;
display: flex;
font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
height: 36px;
justify-content: center;
min-height: 36px;
padding: 4px 8px;
position: relative;
text-align: center;
text-transform: uppercase;
transition: color 0.15s, background-color 0.15s, border-color 0.15s;
border-radius: 0.25rem;
}
.import-btn {
border-radius: 0.25rem 0px 0px 0.25rem;
}
.import-btn-info {
border-radius: 0px 0.25rem 0.25rem 0px;
}
.btn-outline-secondary:hover {
background-color: rgb(51, 51, 56);
border-color: rgb(212, 212, 216);
}
.btn-outline-secondary:disabled {
background-color: rgb(51, 51, 56, 0.5);
border-color: rgb(212, 212, 216);
color: rgb(212, 212, 216);
cursor: not-allowed;
}
.btn-primary {
font-family: Montserrat, sans-serif;
font-weight: 600;
font-size: 0.875rem;
line-height: 1;
letter-spacing: 0.05em;
text-transform: uppercase;
transition: background-color 0.3s;
position: relative;
min-height: 2.25rem;
outline: none;
padding: 0.25rem;
cursor: pointer;
background-color: rgb(217, 143, 64);
color: rgb(255, 255, 255);
border: none;
border-radius: 5px;
}
.btn-primary:disabled {
background-color: rgb(217, 143, 64, 0.5);
color: rgb(255, 255, 255, 0.5);
cursor: not-allowed;
}
.download-btn-all {
width: 100%;
display: flex;
gap: 0.5rem;
justify-content: space-between;
align-items: center;
border-radius: 0.25rem 0 0 0.25rem;
}
.download-btn-menu {
border-radius: 0 0.25rem 0.25rem 0;
}
.ndc-dropdown {
background-color: rgb(29, 29, 33);
border: 1px solid rgba(255, 255, 255, 0.2);
border-radius: 6px;
box-shadow: 0px 9px 12px 1px rgba(0, 0, 0, 0.14),
0px 3px 16px 2px rgba(0, 0, 0, 0.12), 0px 5px 6px 0px rgba(0, 0, 0, 0.2);
box-sizing: border-box;
color: rgb(244, 244, 245);
display: none;
font: 400 16px/24px "Montserrat", ui-sans-serif, system-ui, sans-serif;
min-width: 12rem;
outline: 2px solid rgba(0, 0, 0, 0);
padding: 0.25rem 0;
position: absolute;
right: 0;
top: 0;
z-index: 10;
transform: translate3d(0px, 38px, 0px);
}
.ndc-dropdown-item {
align-items: center;
appearance: button;
background-color: transparent;
border: 0;
box-sizing: border-box;
color: rgb(244, 244, 245);
cursor: pointer;
display: flex;
font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
height: 44px;
justify-content: space-between;
padding: 8px;
position: relative;
text-align: left;
text-transform: uppercase;
white-space: nowrap;
width: 100%;
}
.ndc-dropdown-item:hover {
background-color: rgba(217, 143, 64);
}
.progress-bar {
background-color: rgb(41, 41, 46);
border-radius: 0.25rem;
border: 0 solid rgb(244, 244, 245);
box-sizing: border-box;
color: rgb(244, 244, 245);
display: block;
flex: 1 1 0%;
font: 400 14px/24px "Montserrat", ui-sans-serif, system-ui, sans-serif;
height: 36px;
min-height: 36px;
overflow: hidden;
position: relative;
width: 100%;
}
.progress-bar-fill {
background-color: rgb(217, 143, 64);
border: 0 solid rgb(244, 244, 245);
box-sizing: border-box;
color: rgb(244, 244, 245);
display: block;
font: 400 14px/24px "Montserrat", ui-sans-serif, system-ui, sans-serif;
height: 36px;
position: absolute;
top: 0;
left: 0;
transition: width 0.3s ease;
width: 0;
}
.progress-bar-text-container {
align-items: center;
border: 0 solid rgb(255, 255, 255);
box-sizing: border-box;
color: rgb(255, 255, 255);
cursor: pointer;
display: grid;
font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
grid-template-columns: repeat(3, minmax(0, 1fr));
height: 36px;
position: absolute;
top: 0;
left: 0;
text-transform: uppercase;
width: 100%;
}
.progress-bar-text-base {
border: 0 solid rgb(255, 255, 255);
box-sizing: border-box;
color: rgb(255, 255, 255);
cursor: pointer;
display: block;
font: 600 14px/14px "Montserrat", ui-sans-serif, system-ui, sans-serif;
height: 14px;
text-transform: uppercase;
}
.progress-bar-text-progress {
margin-left: 8px;
}
.progress-bar-text-center {
text-align: center;
}
.progress-bar-text-right {
margin-right: 8px;
text-align: right;
}
.pause-btn {
border-radius: 0;
}
.stop-btn {
border-radius: 0 0.25rem 0.25rem 0;
}
.ndc-modal-backdrop {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
z-index: 9999;
display: flex;
justify-content: center;
align-items: center;
background-color: rgba(0, 0, 0, 0.25);
backdrop-filter: brightness(50%);
}
.ndc-modal {
background-color: rgb(29, 29, 33);
padding: 1rem;
border-radius: 0.5rem;
display: flex;
flex-direction: column;
max-width: 850px;
width: 100%;
height: calc(100vh - 3.5rem);
}
.ndc-modal-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.5rem;
gap: 0.5rem;
}
.ndc-modal-header-title {
font-family: Montserrat, sans-serif;
font-weight: 600;
font-size: 1.125rem;
text-transform: uppercase;
}
.ndc-modal-header-dropdown-btn {
padding: 0.25rem;
border-radius: 0.25rem;
}
.ndc-modal-filter {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.5rem;
gap: 0.5rem;
}
.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: rgb(0, 0, 0);
width: 100%;
height: 100%;
}
.ndc-modal-filter input {
box-sizing: border-box;
}
@media (min-width: 640px) {
.ndc-modal-filter input,
.ndc-modal-filter select {
width: auto;
}
}
.ndc-modal-mods-list {
display: block;
margin-bottom: 0.5rem;
height: 100%;
overflow-y: auto;
}
.ndc-modal-mods-list-header {
display: hidden;
gap: 0.5rem;
border: 1px solid hsla(0, 0%, 100%, 0.2);
padding: 0.5rem;
border-radius: 0.25rem;
cursor: pointer;
user-select: none;
}
@media (min-width: 640px) {
.ndc-modal-mods-list-header {
display: flex;
border-radius: 0;
}
}
.ndc-modal-mods-list-header span {
font-family: Montserrat, sans-serif;
font-weight: 600;
font-size: 0.875rem;
text-transform: uppercase;
color: rgb(161 161 170);
}
.ndc-modal-mods-list-body {
display: flex;
flex-direction: column;
gap: 0.5rem;
}
@media (min-width: 640px) {
.ndc-modal-mods-list-body {
gap: 0;
}
}
.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-mods-list-body-row-desktop {
display: none;
}
@media (min-width: 640px) {
.ndc-modal-mods-list-body-row-desktop {
display: flex;
gap: 0.5rem;
}
}
.ndc-modal-mods-list-body-row-mobile {
display: block;
}
@media (min-width: 640px) {
.ndc-modal-mods-list-body-row-mobile {
display: none;
}
}
.ndc-modal-actions {
display: flex;
justify-content: end;
gap: 0.5rem;
}
`);
const convertSize = (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]}`;
};
class NDC {
mods = [];
constructor() {
this.element = document.createElement("div");
Object.assign(this.element.style, {
borderRadius: ".5rem",
border: "2px solid rgb(217 143 64/1)",
padding: "1rem",
marginTop: "1rem",
backgroundColor: "rgb(29 29 33 / 1)",
backgroundImage: "url(",
backgroundSize: "100%",
backgroundPosition: "center",
backgroundRepeat: "no-repeat",
width: "100%",
});
this.downloadButton = new NDCDownloadButton(this);
this.progressBar = new NDCProgressBar(this);
this.console = new NDCLogConsole(this);
}
async init() {
this.downloadButton.render();
this.element.innerHTML = "";
this.element.appendChild(this.downloadButton.element);
this.element.appendChild(this.progressBar.element);
this.element.appendChild(this.console.element);
}
async fetchGameId(gameDomain) {
const response = await fetch("https://api-router.nexusmods.com/graphql", {
headers: {
"content-type": "application/json",
},
referrer: document.location.href,
referrerPolicy: "strict-origin-when-cross-origin",
body: JSON.stringify({
query:
"query GameDomainToId($gameDomain: String!) { game(domainName: $gameDomain) { id } }",
variables: { gameDomain },
operationName: "GameDomainToId",
}),
method: "POST",
mode: "cors",
credentials: "include",
});
const responseJson = await response.json().catch((e) => {
return null;
});
return responseJson?.data?.game?.id || null;
}
async handleFileUpload(file) {
try {
// Validate input
if (!file) {
this.console.log("No file provided", NDCLogConsole.TYPE_ERROR);
return;
}
// Initialize ZipReader with error handling
let entries;
try {
const zipReader = new zip.ZipReader(new zip.BlobReader(file));
entries = await zipReader.getEntries({});
} catch (zipError) {
this.console.log("Failed to read zip file: " + zipError.message, NDCLogConsole.TYPE_ERROR);
return;
}
// Check if entries exist
if (!entries || entries.length === 0) {
this.console.log("No entries found in zip file", NDCLogConsole.TYPE_ERROR);
return;
}
// Find modlist entry
const modListEntry = entries.find(entry => entry?.filename === "modlist");
if (!modListEntry) {
this.console.log("modlist file not found", NDCLogConsole.TYPE_ERROR);
return;
}
// Extract and parse modlist data
let modList;
try {
modList = await modListEntry.getData(new zip.TextWriter());
} catch (extractError) {
this.console.log("Failed to extract modlist: " + extractError.message, NDCLogConsole.TYPE_ERROR);
return;
}
let mods;
try {
mods = JSON.parse(modList);
if (!mods?.Archives || !Array.isArray(mods.Archives)) {
throw new Error("Invalid modlist structure");
}
} catch (parseError) {
this.console.log("Invalid modlist format: " + parseError.message, NDCLogConsole.TYPE_ERROR);
return;
}
// Process nexus mods
const nexusMods = mods.Archives.filter(mod =>
mod?.State && mod.State['$type'] === "NexusDownloader, Wabbajack.Lib"
);
const games = {};
const processedMods = [];
for (const mod of nexusMods) {
try {
// Validate mod structure
if (!mod?.State || !mod.State.GameName || !mod.State.ModID || !mod.State.FileID) {
this.console.log(`Skipping invalid mod: ${mod?.Name || 'unknown'}`, NDCLogConsole.TYPE_WARNING);
continue;
}
const gameName = mod.State.GameName;
let gameId = games[gameName];
// Fetch game ID if not cached
if (!gameId) {
const gameDomain = gameName.toLowerCase();
gameId = await this.fetchGameId(gameDomain);
if (!gameId) {
this.console.log(`Failed to get game id for ${gameName}`, NDCLogConsole.TYPE_ERROR);
continue;
}
games[gameName] = gameId;
}
// Construct mod object with fallback values
processedMods.push({
fileName: mod.Name || "Unknown File",
modName: mod.State.Name || "Unknown Mod",
size: mod.Size || 0,
url: `https://www.nexusmods.com/${mod.State.GameName.toLowerCase()}/mods/${mod.State.ModID}?tab=files&file_id=${mod.State.FileID}`,
gameId,
modId: mod.State.ModID,
fileId: mod.State.FileID,
});
} catch (modError) {
this.console.log(`Error processing mod ${mod?.Name || 'unknown'}: ${modError.message}`, NDCLogConsole.TYPE_ERROR);
continue;
}
}
// Update mods array and render
this.mods = processedMods;
this.downloadButton.render();
this.console.log(`Wabbajack Modlist loaded successfully. Processed ${processedMods.length} mods.`, NDCLogConsole.TYPE_INFO);
} catch (error) {
this.console.log("Unexpected error in handleFileUpload: " + error.message, NDCLogConsole.TYPE_ERROR);
}
}
async fetchDownloadLink(mod) {
this.bypassNexusAdsCookie();
const response = await fetch(mod.url);
const text = await response.text();
if (text.match(/class="replaced-login-link"/)) {
this.console.log(
'You are not connected on NexusMods. <a href="https://users.nexusmods.com/auth/continue?client_id=nexus&redirect_uri=https://www.nexusmods.com/oauth/callback&response_type=code&referrer=https%3A%2F%2Fwww.nexusmods.com%2F" target="_blank" style="color: rgb(217 143 64)">Login</a> and try again.',
NDCLogConsole.TYPE_ERROR,
);
return { downloadUrl: "", text, stop: true };
} else if (text.match(/Just a moment.../)) {
this.console.log(
`You are rate limited by Cloudflare. Click on the link to solve the captcha and try again. <a href="${mod.url}" target="_blank" style="color: rgb(217 143 64)">Solve captcha</a>`,
NDCLogConsole.TYPE_ERROR,
);
return { downloadUrl: "", text, stop: true };
} else if (
text.match(/Your access to Nexus Mods has been temporarily suspended/)
) {
this.console.log(
"Du to too many requests, Nexus mods temporarily suspended your account for 10 minutes, try again later.",
NDCLogConsole.TYPE_ERROR,
);
return { downloadUrl: "", text, stop: true };
}
if (!response.ok) return { downloadUrl: "", text };
const generateDownloadUrlResponse = await fetch(
"https://www.nexusmods.com/Core/Libs/Common/Managers/Downloads?GenerateDownloadUrl",
{
headers: {
"content-type": "application/x-www-form-urlencoded; charset=UTF-8",
},
body: `fid=${mod.fileId}&game_id=${mod.gameId}`,
method: "POST"
},
);
const fileLink = await generateDownloadUrlResponse.json();
const downloadUrl = fileLink?.url || "";
return { downloadUrl, text };
}
bypassNexusAdsCookie() {
const now = Math.round(Date.now() / 1000);
const expirySeconds = 5 * 60; // 5 minutes in seconds
const expiryTimestamp = now + expirySeconds;
// Set expiry date for the cookie
const expiryDate = new Date(Date.now() + expirySeconds * 1000).toUTCString();
// Create and set the cookie
document.cookie = `ab=0|${expiryTimestamp};expires=${expiryDate};domain=nexusmods.com;path=/`;
}
async downloadMods(mods) {
this.startDownload(mods.length);
try {
const lauchedDownload = await GM.getValue("lauchedDownload", {
count: 0,
date: new Date().getTime(),
});
const failedDownload = [];
let forceStop = false;
for (const [index, mod] of mods.entries()) {
const modNumber = `${(index + 1)
.toString()
.padStart(mods.length.toString().length, "0")}/${mods.length}`;
if (lauchedDownload.date < new Date().getTime() - 1000 * 60 * 5) {
// 5 minutes
lauchedDownload.count = 0;
await GM.setValue("lauchedDownload", lauchedDownload);
}
if (this.progressBar.skipTo) {
if (this.progressBar.skipToIndex - 1 > index) {
this.console.log(
`[${modNumber}] Skipping <a href="${mod.url}" target="_blank" style="color: rgb(217 143 64)">${mod.modName}</a>`,
);
this.progressBar.incrementProgress();
if (this.progressBar.skipToIndex - 1 === index + 1) {
// if skip to index is the next index
this.progressBar.skipTo = false;
}
continue;
}
this.progressBar.skipTo = false;
}
if (this.progressBar.status === NDCProgressBar.STATUS_STOPPED) {
this.console.log("Download stopped.", NDCLogConsole.TYPE_INFO);
break;
}
const { downloadUrl, text, stop = false } = await this.fetchDownloadLink(mod);
if (downloadUrl === "") {
const logRow = this.console.log(
`
[${modNumber}] Failed to get download link for
<a href="${mod.url}" target="_blank" style="color: rgb(217 143 64)">${mod.modName}</a>
<button style="color: rgb(217 143 64)" title="Copy response to clipboard">
<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" role="presentation" style="width: 1rem; height: 1rem;">
<path 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" style="fill: currentcolor;"></path>
</svg>
</button>
`,
NDCLogConsole.TYPE_ERROR,
);
logRow.querySelector("button").addEventListener("click", () => {
navigator.clipboard.writeText(text);
alert("Response copied to clipboard");
});
// check if find .replaced-login-link in the html it is because the user is not connect on nexusmods
if (stop) {
forceStop = true;
} else {
failedDownload.push(mod);
}
} else {
this.console.log(
`[${modNumber}] Downloading <a href="${mod.url
}" target="_blank" style="color: rgb(217 143 64)" title="Nexus mod page">${mod.modName
}</a><a href="${downloadUrl}"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" role="presentation" style="width: 1.3rem; height: 1.3rem;"><title>Download link</title><path d="M5,20H19V18H5M19,9H15V3H9V9H5L12,16L19,9Z" style="fill: currentcolor;" /></svg></a><span style="font-size: .75rem; color: rgb(161 161 170)">(${convertSize(
mod.size,
)})</span>`,
);
const downloadLink = document.createElement("a");
downloadLink.href = downloadUrl;
downloadLink.download = mod.fileName;
downloadLink.click();
this.progressBar.incrementProgress();
lauchedDownload.count++;
lauchedDownload.date = new Date().getTime();
await GM.setValue("lauchedDownload", lauchedDownload);
}
if (forceStop) {
this.console.log(
"Download forced to stop due to an error.",
NDCLogConsole.TYPE_ERROR,
);
break;
}
if (index < mods.length - 1) {
if (lauchedDownload.count >= 200) {
// 200 is a safe number of downloads to avoid Nexus bans
let remainingTime = 5 * 60; // 5 minutes
this.console.log(
"Started the download of 200 mods. Waiting 5 minutes before continuing to avoid the temporary 10 minutes ban from Nexus.",
NDCLogConsole.TYPE_INFO,
);
let logRow = null;
await new Promise((resolve) => {
const intervalId = setInterval(async () => {
remainingTime--;
const minutes = Math.floor(remainingTime / 60);
const seconds = remainingTime % 60;
const logMessage = `Waiting for ${minutes} minutes and ${seconds} seconds before continuing...`;
if (!logRow) {
logRow = this.console.log(logMessage, NDCLogConsole.TYPE_INFO);
} else {
logRow.innerHTML = logMessage;
}
if (remainingTime <= 0) {
logRow.remove();
clearInterval(intervalId);
lauchedDownload.count = 0;
await GM.setValue("lauchedDownload", lauchedDownload);
return resolve();
}
}, 1000);
});
}
const pause = 1;
let logRow = null;
const startDateTime = new Date().getTime();
await new Promise((resolve) => {
const intervalId = setInterval(async () => {
const remainingTime = Math.max(
0,
Math.round(
(startDateTime + pause * 1000 - new Date().getTime()) / 1000,
),
);
const minutes = Math.max(0, Math.floor(remainingTime / 60));
const seconds = Math.max(0, remainingTime % 60);
const logMessage = `Waiting ${minutes} minutes and ${seconds} seconds before starting the next download...`;
if (!logRow) {
logRow = this.console.log(logMessage, NDCLogConsole.TYPE_INFO);
} else {
logRow.innerHTML = logMessage;
}
const shouldClearInterval = () => {
clearInterval(intervalId);
logRow.remove();
return resolve();
};
if (
this.progressBar.skipPause ||
this.progressBar.skipTo ||
this.progressBar.status === NDCProgressBar.STATUS_STOPPED
) {
if (this.progressBar.skipPause) {
this.progressBar.skipPause = false;
}
return shouldClearInterval();
}
if (this.progressBar.status === NDCProgressBar.STATUS_PAUSED) {
return;
}
if (new Date().getTime() >= startDateTime + pause * 1000) {
return shouldClearInterval();
}
}, 100);
});
}
}
if (failedDownload.length) {
this.console.log(
`Failed to download ${failedDownload.length} mods:`,
NDCLogConsole.TYPE_INFO,
);
for (const mod of failedDownload) {
this.console.log(
`<a href="${mod.url}" target="_blank" style="color: rgb(217 143 64)">${mod.modName}</a>`,
NDCLogConsole.TYPE_INFO,
);
}
}
} catch (error) {
this.console.log("An error occurred during the download.", NDCLogConsole.TYPE_ERROR);
console.error(error);
}
this.endDownload();
}
startDownload(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.log("Download started.", NDCLogConsole.TYPE_INFO);
}
endDownload() {
this.progressBar.setStatus(NDCProgressBar.STATUS_FINISHED);
this.progressBar.element.style.display = "none";
this.downloadButton.element.style.display = "flex";
this.console.log("Download finished.", NDCLogConsole.TYPE_INFO);
}
}
class NDCDownloadButton {
constructor(ndc) {
this.element = document.createElement("div");
Object.assign(this.element.style, {
display: "flex",
flexDirection: "column",
gap: "1rem",
width: "100%",
});
this.ndc = ndc;
this.html = `
<div style="display: flex; flex-direction: row; gap: 1rem; justify-content: center;">
<div style="display: flex;">
<button id="importWabbajackModsBtn" class="btn-outline-secondary import-btn">
Import Wabbajack modlist
</button>
<button id="importWabbajackModsBtnInfo" class="btn-outline-secondary import-btn-info">
<svg id="extraPauseInfo" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" role="presentation" style="width: 1.5rem; height: 1.5rem; cursor: pointer; fill: currentcolor;">
<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"></path>
</svg>
</button>
</div>
</div>
<div style="display: flex; width: 100%;">
<button id="mainBtn" class="btn-primary download-btn-all">
download all mods
<span id="mainModsCount" style="padding: 0.5rem; background-color: rgba(29, 29, 33, 0.8); border-radius: 5px; font-size: 0.75rem; color: white; white-space: nowrap;"></span>
</button>
<div style="position: relative;">
<button id="menuBtn" class="btn-primary download-btn-menu">
<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" role="presentation" style="width: 1.5rem; height: 1.5rem;">
<path d="M7.41,8.58L12,13.17L16.59,8.58L18,10L12,16L6,10L7.41,8.58Z" style="fill: currentcolor;"></path>
</svg>
</button>
<div id="otherOptionMenu" class="ndc-dropdown">
<button id="menuBtnSelect" class="ndc-dropdown-item">
Select mods to download
</button>
</div>
</div>
</div>
`;
this.element.innerHTML = this.html;
this.importWabbajackModsBtn = this.element.querySelector("#importWabbajackModsBtn",);
this.importWabbajackModsBtnInfo = this.element.querySelector("#importWabbajackModsBtnInfo",);
this.allBtn = this.element.querySelector("#mainBtn");
this.modsCount = this.element.querySelector("#mainModsCount");
this.selectBtn = this.element.querySelector("#menuBtnSelect");
this.menuBtn = this.element.querySelector("#menuBtn");
const otherOptionMenu = this.element.querySelector("#otherOptionMenu");
this.importWabbajackModsBtn.addEventListener("click", () => {
// create a temporary input element to select a folder
const input = document.createElement("input");
input.type = "file";
// allow only .wabbajack files
input.accept = ".wabbajack";
// input.multiple = true;
input.addEventListener("change", async () => {
// disable the button to prevent multiple uploads
this.importWabbajackModsBtn.disabled = true;
// Add a loading spinner
this.importWabbajackModsBtn.innerHTML = `
<div class="spinner-border" role="status" style="margin-right: 0.25rem;"></div>
Importing...
`;
await this.ndc.handleFileUpload(input.files[0]);
// re-enable the button
this.importWabbajackModsBtn.disabled = false;
this.importWabbajackModsBtn.innerHTML = "Import Wabbajack modlist";
// remove the input element
input.remove();
});
input.click();
});
this.importWabbajackModsBtnInfo.addEventListener("click", () => {
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 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 will be able to download the mods.'
);
});
this.menuBtn.addEventListener("click", () => {
otherOptionMenu.style.display = otherOptionMenu.style.display === "block" ? "none" : "block";
});
document.addEventListener("click", (event) => {
const isClickInside =
// otherOptionMenu.contains(event.target) ||
this.menuBtn.contains(event.target);
if (!isClickInside) {
otherOptionMenu.style.display = "none";
}
});
this.allBtn.addEventListener("click", () =>
this.ndc.downloadMods(this.ndc.mods, "all"),
);
this.selectBtn.addEventListener("click", () => {
const selectModsModal = new NDCSelectModsModal(this.ndc);
document.body.appendChild(selectModsModal.element);
selectModsModal.render();
});
}
updateModsCount() {
// if mods count is 0, gray and disable the button
if (this.ndc.mods.length === 0) {
this.allBtn.disabled = true;
this.menuBtn.disabled = true;
} else {
this.allBtn.disabled = false;
this.menuBtn.disabled = false;
}
this.modsCount.innerHTML = `${this.ndc.mods.length} mods`;
}
render() {
this.updateModsCount();
}
}
class NDCProgressBar {
static STATUS_DOWNLOADING = 0;
static STATUS_PAUSED = 1;
static STATUS_FINISHED = 2;
static STATUS_STOPPED = 3;
static STATUS_TEXT = {
[NDCProgressBar.STATUS_DOWNLOADING]: "Downloading...",
[NDCProgressBar.STATUS_PAUSED]: "Paused",
[NDCProgressBar.STATUS_FINISHED]: "Finished",
[NDCProgressBar.STATUS_STOPPED]: "Stopped",
};
constructor(ndc, options = {}) {
this.element = document.createElement("div");
Object.assign(this.element.style, {
// "flex", "flex-wrap", "w-100"
// display: "flex",
display: "none",
flexWrap: "wrap",
width: "100%",
});
this.ndc = ndc;
this.modsCount = 0;
this.progress = 0;
this.skipPause = false;
this.skipTo = false;
this.skipToIndex = 0;
this.status = NDCProgressBar.STATUS_DOWNLOADING;
this.html = `
<div class="progress-bar" id="progressBar">
<div class="progress-bar-fill" id="progressBarFill"></div>
<div class="progress-bar-text-container" id="progressBarText">
<div class="progress-bar-text-base progress-bar-text-progress" id="progressBarProgress">${this.progress}%</div>
<div class="progress-bar-text-base progress-bar-text-center" id="progressBarTextCenter">Downloading...</div>
<div class="progress-bar-text-base progress-bar-text-right" id="progressBarTextRight">${this.progress}/${this.modsCount}</div>
</div>
</div>
<div style="display: flex;" id="actionBtnGroup">
<button class="btn-primary pause-btn" id="playPauseBtn">
<svg style="width: 1.5rem; height: 1.5rem; fill: currentcolor;" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" role="presentation"><path d="M14,19H18V5H14M6,19H10V5H6V19Z" style="fill: currentcolor;"></path></svg>
</button>
<button class="btn-primary stop-btn" id="stopBtn">
<svg style="width: 1.5rem; height: 1.5rem; fill: currentcolor;" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" role="presentation"><path d="M18,18H6V6H18V18Z" style="fill: currentcolor;"></path></svg>
</button>
</div>
<div style="display: flex; margin: 0.5rem 0; justify-content: flex-end; flex-basis: 100%;" id="toolbarContainer">
<div style="display: flex; gap: 0.5rem; align-items: center;" id="skipContainer">
<button class="btn-primary" id="skipNextBtn">
Skip pause
</button>
<button class="btn-primary" id="skipToIndexBtn">
Skip to index
</button>
<input style="appearance: auto; background-color: rgb(41, 41, 46); border: 1px solid rgb(161, 161, 170); border-radius: 4px; box-sizing: border-box; color: rgb(161, 161, 170); cursor: text; display: block; font: 400 16px/24px 'Montserrat', ui-sans-serif, system-ui, sans-serif; height: 36px; outline: 2px solid rgba(0, 0, 0, 0); padding: 0.5rem; text-align: start; width: 80px;" type="number" min="0" placeholder="Index" id="skipToIndexInput">
</div>
</div>
`;
this.element.innerHTML = this.html;
this.progressBarFill = this.element.querySelector("#progressBarFill");
this.progressBarProgress = this.element.querySelector("#progressBarProgress");
this.progressBarTextCenter = this.element.querySelector("#progressBarTextCenter");
this.progressBarTextRight = this.element.querySelector("#progressBarTextRight");
this.playPauseBtn = this.element.querySelector("#playPauseBtn");
this.stopBtn = this.element.querySelector("#stopBtn");
this.skipNextBtn = this.element.querySelector("#skipNextBtn");
this.skipToIndexBtn = this.element.querySelector("#skipToIndexBtn");
this.skipToIndexInput = this.element.querySelector("#skipToIndexInput");
this.playPauseBtn.addEventListener("click", () => {
const status =
this.status === NDCProgressBar.STATUS_DOWNLOADING
? NDCProgressBar.STATUS_PAUSED
: NDCProgressBar.STATUS_DOWNLOADING;
this.setStatus(status);
});
this.stopBtn.addEventListener("click", () => {
this.setStatus(NDCProgressBar.STATUS_STOPPED);
});
this.skipNextBtn.addEventListener("click", () => {
this.skipPause = true;
this.setStatus(NDCProgressBar.STATUS_DOWNLOADING);
});
this.skipToIndexBtn.addEventListener("click", () => {
const index = Number.parseInt(this.skipToIndexInput.value);
if (index > this.progress && index <= this.modsCount) {
this.skipTo = true;
this.skipToIndex = index;
this.setStatus(NDCProgressBar.STATUS_DOWNLOADING);
}
});
}
setState(newState) {
Object.assign(this, newState);
this.render();
}
setModsCount(modsCount) {
this.setState({ modsCount });
}
setProgress(progress) {
this.setState({ progress });
}
incrementProgress() {
this.setState({ progress: this.progress + 1 });
}
setStatus(status) {
this.setState({ status });
this.progressBarTextCenter.innerHTML = NDCProgressBar.STATUS_TEXT[status];
}
getProgressPercent() {
return ((this.progress / this.modsCount) * 100).toFixed(2);
}
updateProgressBarFillWidth() {
this.progressBarFill.style.width = `${this.getProgressPercent()}%`;
}
updateProgressBarTextProgress() {
this.progressBarProgress.innerHTML = `${this.getProgressPercent()}%`;
}
updateProgressBarTextRight() {
this.progressBarTextRight.innerHTML = `${this.progress}/${this.modsCount}`;
}
updatePlayPauseBtn() {
this.playPauseBtn.innerHTML =
this.status === NDCProgressBar.STATUS_PAUSED
? '<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" role="presentation" style="width: 1.5rem; height: 1.5rem;"><path d="M8,5.14V19.14L19,12.14L8,5.14Z" style="fill: currentcolor;"></path></svg>'
: '<svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" role="presentation" style="width: 1.5rem; height: 1.5rem;"><path d="M14,19H18V5H14M6,19H10V5H6V19Z" style="fill: currentcolor;"></path></svg>';
}
render() {
this.updateProgressBarFillWidth();
this.updateProgressBarTextProgress();
this.updateProgressBarTextRight();
this.updatePlayPauseBtn();
}
}
class NDCSelectModsModal {
constructor(ndc) {
this.element = document.createElement("div");
this.element.classList.add(
"ndc-modal-backdrop",
);
this.ndc = ndc;
this.html = `
<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" id="selectedModsCount">0 mods selected</span>
</div>
<div style="position: relative;">
<button type="button" class="btn-outline-secondary ndc-modal-header-dropdown-btn" id="openSelectModsOptionMenu">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem; fill: currentcolor;">
<title>Options</title>
<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" id="selectModsOptionMenu">
<button class="ndc-dropdown-item" id="selectModsSelectAll">
Select all
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem;"><title>Check all mods</title><path 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" style="fill: currentcolor;"/></svg>
</button>
<button class="ndc-dropdown-item" id="selectModsDeselectAll">
Deselect all
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem;"><title>Clear selection</title><path 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" style="fill: currentcolor;"/></svg>
</button>
<button class="ndc-dropdown-item" id="selectModsInvertSelection">
Invert selection
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem;"><title>Invert selection</title><path 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" style="fill: currentcolor;"/></svg>
</button>
<div class="border-t border-stroke-subdued"></div>
<button class="ndc-dropdown-item" id="exportModsSelection">
Export mods selection
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem;"><title>Export</title><path d="M9,16V10H5L12,3L19,10H15V16H9M5,20V18H19V20H5Z" style="fill: currentcolor;"/></svg>
</button>
<button class="ndc-dropdown-item" id="importModsSelection">
Import mods selection
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem;"><title>Import</title><path d="M5,20H19V18H5M19,9H15V3H9V9H5L12,16L19,9Z" style="fill: currentcolor;"/></svg>
</button>
<div class="border-t border-stroke-subdued"></div>
<button class="ndc-dropdown-item" id="selectModsImportDownloadedMods">
Import downloaded mods
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style="width: 1.5rem; height: 1.5rem;"><title>Import</title><path d="M5,20H19V18H5M19,9H15V3H9V9H5L12,16L19,9Z" style="fill: currentcolor;"/></svg>
</button>
</div>
</div>
</div>
</div>
<div class="ndc-modal-filter">
<input type="search" id="searchMods" placeholder="Search mods...">
<select id="sortMods">
<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 class="mod-list-index" style="width: 3rem;">Index</span>
<span style="flex: 1 1 0%;">Mod name</span>
<span style="flex: 1 1 0%;">File name</span>
<span style="width: 5rem;">Size</span>
</div>
<div id="modsListMobile" class="ndc-modal-mods-list-body"></div>
</div>
<div class="ndc-modal-actions">
<button class="btn-outline-secondary" id="cancelSelectModsBtn">Cancel</button>
<button class="btn-primary" id="selectModsBtn">Download selected mods</button>
</div>
</div>
`;
this.element.innerHTML = this.html;
this.searchMods = this.element.querySelector("#searchMods");
this.sortMods = this.element.querySelector("#sortMods");
this.selectModsSelectAll = this.element.querySelector(
"#selectModsSelectAll",
);
this.selectModsInvertSelection = this.element.querySelector(
"#selectModsInvertSelection",
);
this.selectModsDeselectAll = this.element.querySelector(
"#selectModsDeselectAll",
);
this.modsListMobile = this.element.querySelector("#modsListMobile");
this.selectedModsCount = this.element.querySelector("#selectedModsCount");
this.openSelectModsOptionMenu = this.element.querySelector(
"#openSelectModsOptionMenu",
);
this.selectModsOptionMenu = this.element.querySelector(
"#selectModsOptionMenu",
);
this.exportModsSelection = this.element.querySelector(
"#exportModsSelection",
);
this.importModsSelection = this.element.querySelector(
"#importModsSelection",
);
this.selectModsImportDownloadedMods = this.element.querySelector(
"#selectModsImportDownloadedMods",
);
this.selectModsBtn = this.element.querySelector("#selectModsBtn");
this.cancelSelectModsBtn = this.element.querySelector(
"#cancelSelectModsBtn",
);
this.openSelectModsOptionMenu.addEventListener("click", () => {
this.selectModsOptionMenu.style.display =
this.selectModsOptionMenu.style.display === "block" ? "none" : "block";
});
this.selectModsBtn.addEventListener("click", () => {
const selectedMods = [];
for (const mod of this.ndc.mods) {
const checkbox = this.element.querySelector(`#mod_${mod.fileId}`);
if (checkbox.checked) {
selectedMods.push(mod);
}
}
this.element.remove();
this.ndc.downloadMods(selectedMods);
});
this.cancelSelectModsBtn.addEventListener("click", () => {
this.element.remove();
});
document.addEventListener("click", (event) => {
const isClickInside = this.openSelectModsOptionMenu.contains(
event.target,
);
// if the click on an option, close the menu
if (!isClickInside) {
this.selectModsOptionMenu.style.display = "none";
}
});
}
updateModList(mods) {
this.modsListMobile.innerHTML = "";
for (const [index, mod] of mods.entries()) {
const modElementMobile = document.createElement("div");
modElementMobile.classList.add("ndc-modal-mods-list-body-row");
modElementMobile.innerHTML = `
<input type="checkbox" id="mod_${mod.fileId}" style="display: none;">
<div class="ndc-modal-mods-list-body-row-desktop">
<span style="width: 3rem; color: rgb(217 143 64);" class="mod-list-index">#${index + 1}</span>
<span style="flex: 1 1 0%;" class="text-white">${mod.modName}</span>
<span style="flex: 1 1 0%;" class="text-white">${mod.fileName}</span>
<span style="width: 5rem;" class="text-white">${convertSize(mod.size)}</span>
</div>
<div class="ndc-modal-mods-list-body-row-mobile">
<div style="flex: 1 1 0%; justify-content: space-between; align-items: center; margin-bottom: 0.25rem;">
<div style="flex: 1 1 0%; gap: 0.5rem; align-items: center;">
<span class="mod-list-index" style="color: rgb(217 143 64);">#${index + 1}</span>
</div>
<div style="flex: 1 1 0%; gap: 0.5rem;">
<span class="text-white">${convertSize(mod.size)}</span>
</div>
</div>
<div style="flex: 1 1 0%; flex-direction: column; gap: 0.25rem;">
<div class="text-white">${mod.modName}</div>
<div class="text-white">${mod.fileName}</div>
</div>
</div>
`;
modElementMobile.addEventListener("click", (event) => {
// if check change color, if shiftkey is pressed, select all between the last checked and this one
const checkbox = modElementMobile.querySelector(
'input[type="checkbox"]',
);
checkbox.checked = !checkbox.checked;
const modElement = checkbox.parentNode;
modElement.classList.toggle("bg-primary-subdued");
modElement
.querySelector(".mod-list-index")
.classList.toggle("text-white");
// if shift key is pressed and there is a last checked element
if (event.shiftKey && modElement.parentNode.dataset.lastChecked) {
const start = Array.from(modElement.parentNode.children).indexOf(
modElement,
);
const end = modElement.parentNode.dataset.lastChecked;
const checkedState = modElement.parentNode.children[
end
].querySelector('input[type="checkbox"]').checked;
for (let i = Math.min(start, end); i <= Math.max(start, end); i++) {
const modEl = modElement.parentNode.children[i];
const checkboxEl = modEl.querySelector('input[type="checkbox"]');
checkboxEl.checked = checkedState;
modEl.classList.toggle("bg-primary-subdued", checkedState);
modEl
.querySelector(".mod-list-index")
.classList.toggle("text-white", checkedState);
}
}
// get the index of the element, and store it to the parent
const index = Array.from(modElement.parentNode.children).indexOf(
modElement,
);
modElement.parentNode.dataset.lastChecked = index;
this.selectedModsCount.firstChild.textContent = `${this.element.querySelectorAll('input[type="checkbox"]:checked').length
} mods selected`;
});
this.modsListMobile.appendChild(modElementMobile);
}
}
render() {
this.updateModList(this.ndc.mods);
// close the modal when clicking outside of it
this.element.addEventListener("click", (event) => {
if (event.target === this.element) {
this.element.remove();
}
});
// searchMods
this.searchMods.addEventListener("input", () => {
const search = this.searchMods.value.toLowerCase();
for (const mod of this.ndc.mods) {
const modElement = this.element.querySelector(
`#mod_${mod.fileId}`,
).parentNode;
if (
mod.modName.toLowerCase().includes(search) ||
mod.fileName.toLowerCase().includes(search)
) {
modElement.style.display = "";
} else {
modElement.style.display = "none";
}
}
});
// sortMods
this.sortMods.addEventListener("change", () => {
const sort = this.sortMods.value;
const mods = [...this.ndc.mods];
switch (sort) {
case "mod_name_asc":
mods.sort((a, b) => a.modName.localeCompare(b.modName));
break;
case "mod_name_desc":
mods.sort((a, b) => b.modName.localeCompare(a.modName));
break;
case "file_name_asc":
mods.sort((a, b) => a.fileName.localeCompare(b.fileName));
break;
case "file_name_desc":
mods.sort((a, b) => b.fileName.localeCompare(a.fileName));
break;
case "size_asc":
mods.sort((a, b) => a.size - b.size);
break;
case "size_desc":
mods.sort((a, b) => b.size - a.size);
break;
}
this.updateModList(mods);
});
this.selectModsSelectAll.addEventListener("click", () => {
for (const mod of this.ndc.mods) {
const checkbox = this.element.querySelector(`#mod_${mod.fileId}`);
checkbox.checked = true;
const modElement = checkbox.parentNode;
modElement.classList.add("bg-primary-subdued");
modElement.querySelector(".mod-list-index").classList.add("text-white");
}
this.selectedModsCount.firstChild.textContent = `${this.ndc.mods.length} mods selected`;
});
this.selectModsInvertSelection.addEventListener("click", () => {
for (const mod of this.ndc.mods) {
const checkbox = this.element.querySelector(`#mod_${mod.fileId}`);
checkbox.checked = !checkbox.checked;
const modElement = checkbox.parentNode;
modElement.classList.toggle("bg-primary-subdued");
modElement
.querySelector(".mod-list-index")
.classList.toggle("text-white");
}
this.selectedModsCount.firstChild.textContent =
`${this.element.querySelectorAll('input[type="checkbox"]:checked').length} mods selected`;
});
this.selectModsDeselectAll.addEventListener("click", () => {
for (const mod of this.ndc.mods) {
const checkbox = this.element.querySelector(`#mod_${mod.fileId}`);
checkbox.checked = false;
const modElement = checkbox.parentNode;
modElement.classList.remove("bg-primary-subdued");
modElement
.querySelector(".mod-list-index")
.classList.remove("text-white");
}
this.selectedModsCount.firstChild.textContent = "0 mods selected";
});
this.exportModsSelection.addEventListener("click", () => {
if (!this.element.querySelector('input[type="checkbox"]:checked')) {
alert("You must select at least one mod to export.");
return;
}
const selectedMods = [];
for (const mod of this.ndc.mods) {
const checkbox = this.element.querySelector(`#mod_${mod.fileId}`);
if (checkbox.checked) {
selectedMods.push(mod);
}
}
const selectedModsText = JSON.stringify(selectedMods, null, 2);
const blob = new Blob([selectedModsText], { type: "application/json" });
const url = URL.createObjectURL(blob);
const a = document.createElement("a");
a.href = url;
a.download = `ndc_selected_mods_${this.ndc.gameId}_${this.ndc.collectionId
}_${Date.now()}.json`;
a.click();
URL.revokeObjectURL(url);
});
this.importModsSelection.addEventListener("click", () => {
const input = document.createElement("input");
input.type = "file";
input.accept = ".json";
input.addEventListener("change", async () => {
const file = input.files[0];
const reader = new FileReader();
reader.onload = async () => {
const selectedMods = JSON.parse(reader.result);
for (const mod of selectedMods) {
const checkbox = this.element.querySelector(
`#mod_${mod.fileId}`,
);
if (checkbox == null) {
continue;
}
checkbox.checked = true;
const modElement = checkbox.parentNode;
modElement.classList.add("bg-primary-subdued");
modElement
.querySelector(".mod-list-index")
.classList.add("text-white");
}
this.selectedModsCount.firstChild.textContent = `${selectedMods.length} mods selected`;
};
reader.readAsText(file);
});
input.click();
});
this.selectModsImportDownloadedMods.addEventListener("click", () => {
const input = document.createElement("input");
input.type = "file";
input.multiple = true;
input.addEventListener("change", async () => {
const files = input.files;
const downloadedMods = this.ndc.mods.filter((mod) => {
for (const file of files) {
if (file.name.includes(mod.fileName)) {
return true;
}
}
return false;
}).reduce((acc, mod) => {
acc[mod.fileId] = mod;
return acc;
}, {});
// for each checkbox, check if the mod is in the downloadedMods array, if not, check the checkbox
const notDownloadedMods = [];
for (const modElement of this.modsListMobile.childNodes) {
const checkbox = modElement.querySelector('input[type="checkbox"]');
const modId = Number.parseInt(checkbox.id.split("_")[1]);
if (downloadedMods[modId] == null) {
notDownloadedMods.push(downloadedMods[modId]);
checkbox.checked = true;
modElement.classList.add("bg-primary-subdued");
modElement.querySelector(".mod-list-index").classList.add("text-white");
}
}
this.selectedModsCount.firstChild.textContent = `${notDownloadedMods.length} mods selected`;
if (notDownloadedMods.length === 0) {
alert("All mods are already downloaded.");
} else {
alert(
`Selected ${notDownloadedMods.length
} mods that are not downloaded yet.`,
);
}
});
input.click();
});
}
}
class NDCLogConsole {
static TYPE_NORMAL = "NORMAL";
static TYPE_ERROR = "ERROR";
static TYPE_INFO = "INFO";
constructor(ndc, options = {}) {
this.element = document.createElement("div");
Object.assign(this.element.style, {
// "flex", "flex-col", "w-100", "gap-3", "mt-3"
display: "flex",
flexDirection: "column",
width: "100%",
gap: "1rem",
marginTop: "1rem",
});
this.ndc = ndc;
this.hidden = false;
this.html = `
<div style="display: flex; flex-direction: column; width: 100%; gap: 0.75rem;">
<button style="appearance: button; background: none; border: 0; box-sizing: border-box; color: rgb(244, 244, 245); cursor: pointer; display: block; font: 400 16px/24px 'Montserrat', ui-sans-serif, system-ui, sans-serif; height: 24px; text-align: center; width: 100%;" id="toggleLogsButton">
Hide logs
</button>
<div style="background-color: rgb(29 29 33 / 70%); border: 1px solid rgb(255, 255, 255); border-radius: 4px; box-sizing: border-box; color: rgb(255, 255, 255); display: block; font: 600 14px/21px 'Montserrat', ui-sans-serif, system-ui, sans-serif; height: 160px; overflow-y: auto; resize: vertical; width: 100%;" id="logContainer">
</div>
</div>
`;
this.element.innerHTML = this.html;
this.toggle = this.element.querySelector("#toggleLogsButton");
this.logContainer = this.element.querySelector("#logContainer");
this.toggle.addEventListener("click", () => {
this.hidden = !this.hidden;
logContainer.style.display = this.hidden ? "none" : "";
this.toggle.innerHTML = this.hidden ? "Show logs" : "Hide logs";
});
}
log(message, type = NDCLogConsole.TYPE_NORMAL) {
const rowElement = document.createElement("div");
Object.assign(rowElement.style, {
display: "flex",
columnGap: ".25rem",
padding: "0.25rem .5rem",
});
if (type === NDCLogConsole.TYPE_ERROR) {
Object.assign(rowElement.style, {
color: "rgb(229, 62, 62)",
});
} else if (type === NDCLogConsole.TYPE_INFO) {
Object.assign(rowElement.style, {
color: "rgb(96 165 250)",
});
}
rowElement.innerHTML = `<span>[${new Date().toLocaleTimeString()}]</span><span class="ndc-log-message" style="display: flex; gap: 1rem;">${message}</span>`;
rowElement.message = rowElement.querySelector(".ndc-log-message");
this.logContainer.appendChild(rowElement);
this.logContainer.scrollTop = this.logContainer.scrollHeight;
console.log(`${message}`);
return rowElement;
}
clear() {
this.logContainer.innerHTML = "";
}
}
let ndc = null;
async function handleNextRouterChange() {
ndc = new NDC();
await ndc.init();
// 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();
})();