325 lines
11 KiB
JavaScript
325 lines
11 KiB
JavaScript
import { KeybindsInput } from "./keybindsInput.js";
|
|
import winCounter from "./winCounter.js";
|
|
import WindowManager from "./windowManager.js";
|
|
import versionData from '../version.json';
|
|
import { displayChangelog } from './changelog.js';
|
|
|
|
window.__fx = window.__fx || {};
|
|
const __fx = window.__fx;
|
|
|
|
var settings = {
|
|
//"fontName": "Trebuchet MS",
|
|
//"showBotDonations": false,
|
|
displayWinCounter: true,
|
|
useFullscreenMode: false,
|
|
hoveringTooltip: true,
|
|
//"hideAllLinks": false,
|
|
realisticNames: false,
|
|
showPlayerDensity: true,
|
|
coloredDensity: true,
|
|
densityDisplayStyle: "percentage",
|
|
hideBotNames: false,
|
|
highlightClanSpawns: false,
|
|
detailedTeamPercentage: false,
|
|
//"customMapFileBtn": true
|
|
customBackgroundUrl: "",
|
|
keybindButtons: false,
|
|
attackPercentageKeybinds: [],
|
|
};
|
|
__fx.settings = settings;
|
|
const discontinuedSettings = ["hideAllLinks", "fontName"];
|
|
__fx.makeMainMenuTransparent = false;
|
|
|
|
/*var settingsGearIcon = document.createElement('img');
|
|
settingsGearIcon.setAttribute('src', 'assets/geari_white.png');*/
|
|
|
|
const settingsManager = new (function () {
|
|
const settingsStructure = [
|
|
{
|
|
for: "displayWinCounter",
|
|
type: "checkbox",
|
|
label: "Display win counter",
|
|
note: "The win counter tracks multiplayer solo wins (not in team games)",
|
|
},
|
|
{
|
|
type: "button",
|
|
text: "Reset win counter",
|
|
action: winCounter.removeWins,
|
|
},
|
|
{
|
|
for: "useFullscreenMode",
|
|
type: "checkbox",
|
|
label: "Use fullscreen mode",
|
|
note: "Note: fullscreen mode will trigger after you click anywhere on the page due to browser policy restrictions.",
|
|
},
|
|
{
|
|
for: "hoveringTooltip",
|
|
type: "checkbox",
|
|
label: "Hovering tooltip",
|
|
note: "Display map territory info constantly (on mouse hover) instead of only when right clicking on the map",
|
|
},
|
|
//{ for: "hideAllLinks", type: "checkbox", label: "Hide Links option also hides app store links" },
|
|
{ for: "realisticNames", type: "checkbox", label: "Realistic Bot Names" },
|
|
{
|
|
for: "showPlayerDensity",
|
|
type: "checkbox",
|
|
label: "Show player density",
|
|
},
|
|
{
|
|
for: "coloredDensity",
|
|
type: "checkbox",
|
|
label: "Colored density",
|
|
note: "Display the density with a color between red and green depending on the density value",
|
|
},
|
|
{
|
|
for: "densityDisplayStyle",
|
|
type: "selectMenu",
|
|
label: "Density value display style:",
|
|
tooltip: "Controls how the territorial density value should be rendered",
|
|
options: [
|
|
{ value: "percentage", label: "Percentage" },
|
|
{
|
|
value: "absoluteQuotient",
|
|
label: "Value from 0 to 150 (BetterTT style)",
|
|
},
|
|
],
|
|
},
|
|
{ for: "hideBotNames", type: "checkbox", label: "Hide bot names" },
|
|
{
|
|
for: "highlightClanSpawns",
|
|
type: "checkbox",
|
|
label: "Highlight clan spawnpoints",
|
|
note: "Increases the spawnpoint glow size for members of your clan",
|
|
},
|
|
{
|
|
for: "detailedTeamPercentage", type: "checkbox",
|
|
label: "Detailed team pie chart percentage",
|
|
note: "For example: this would show 25.82% instead of 26% on the pie chart in team games"
|
|
},
|
|
{
|
|
for: "customBackgroundUrl",
|
|
type: "textInput",
|
|
label: "Custom main menu background:",
|
|
placeholder: "Enter an image URL here",
|
|
tooltip:
|
|
"A custom image to be shown as the main menu background instead of the currently selected map.",
|
|
},
|
|
KeybindsInput,
|
|
{
|
|
for: "keybindButtons", type: "checkbox",
|
|
label: "Keybind buttons", note: "Show keybind buttons above the troop selector (max 6)"
|
|
},
|
|
function Footer(container) {
|
|
const versionInfo = document.createElement("p");
|
|
versionInfo.innerText = `FX Client v${versionData.version}`;
|
|
const links = document.createElement("p");
|
|
links.innerHTML = `<a href="https://discord.gg/mtWFTQhTT9" target="_blank">Discord server</a> |
|
|
<a href="https://github.com/fxclient/FXclient#readme">Github repository</a>`;
|
|
const changelogButton = document.createElement("button");
|
|
changelogButton.innerText = "Changelog";
|
|
changelogButton.addEventListener("click", displayChangelog);
|
|
container.append(versionInfo, links, changelogButton);
|
|
}
|
|
];
|
|
const settingsContainer = document.querySelector(".settings .scrollable");
|
|
var inputFields = {}; // (includes select menus)
|
|
var checkboxFields = {};
|
|
var customElements = [];
|
|
settingsStructure.forEach((item) => {
|
|
if (typeof item === "function") {
|
|
const container = document.createElement("div");
|
|
customElements.push(new item(container));
|
|
return settingsContainer.append(container);
|
|
}
|
|
const label = document.createElement("label");
|
|
if (item.tooltip) label.title = item.tooltip;
|
|
const isValueInput = item.type.endsWith("Input");
|
|
const element = document.createElement(
|
|
isValueInput || item.type === "checkbox"
|
|
? "input"
|
|
: item.type === "selectMenu"
|
|
? "select"
|
|
: "button"
|
|
);
|
|
if (item.type === "textInput") element.type = "text";
|
|
if (item.placeholder) element.placeholder = item.placeholder;
|
|
if (isValueInput || item.type === "selectMenu")
|
|
inputFields[item.for] = element;
|
|
if (item.text) element.innerText = item.text;
|
|
if (item.action) element.addEventListener("click", item.action);
|
|
if (item.label) label.append(item.label + " ");
|
|
if (item.note) {
|
|
const note = document.createElement("small");
|
|
note.innerText = item.note;
|
|
label.append(document.createElement("br"), note);
|
|
}
|
|
if (item.options)
|
|
item.options.forEach((option) => {
|
|
const optionElement = document.createElement("option");
|
|
optionElement.setAttribute("value", option.value);
|
|
optionElement.innerText = option.label;
|
|
element.append(optionElement);
|
|
});
|
|
label.append(element);
|
|
if (item.type === "checkbox") {
|
|
element.type = "checkbox";
|
|
const checkmark = document.createElement("span");
|
|
checkmark.className = "checkmark";
|
|
label.className = "checkbox";
|
|
label.append(checkmark);
|
|
checkboxFields[item.for] = element;
|
|
} else label.append(document.createElement("br"));
|
|
settingsContainer.append(label, document.createElement("br"));
|
|
});
|
|
this.save = function () {
|
|
Object.keys(inputFields).forEach(function (key) {
|
|
settings[key] = inputFields[key].value.trim();
|
|
});
|
|
Object.keys(checkboxFields).forEach(function (key) {
|
|
settings[key] = checkboxFields[key].checked;
|
|
});
|
|
this.applySettings();
|
|
WindowManager.closeWindow("settings");
|
|
discontinuedSettings.forEach((settingName) => delete settings[settingName]);
|
|
localStorage.setItem("fx_settings", JSON.stringify(settings));
|
|
// should probably firgure out a way to do this without reloading - // You can't do it, localstorages REQUIRE you to reload
|
|
window.location.reload();
|
|
};
|
|
|
|
const fileInput = document.createElement("input");
|
|
fileInput.type = "file";
|
|
function handleFileSelect(event) {
|
|
const input = event.target;
|
|
/** @type {File} */
|
|
const selectedFile = input.files[0];
|
|
if (!selectedFile) return;
|
|
|
|
input.removeEventListener("change", handleFileSelect);
|
|
input.value = "";
|
|
if (!selectedFile.name.endsWith(".json"))
|
|
return alert("Invalid file format");
|
|
const fileReader = new FileReader();
|
|
fileReader.onload = function () {
|
|
let result;
|
|
try {
|
|
result = JSON.parse(fileReader.result);
|
|
if (
|
|
confirm(
|
|
'Warning: This will override all current settings, click "OK" to confirm'
|
|
)
|
|
)
|
|
__fx.settings = settings = result;
|
|
localStorage.setItem("fx_settings", JSON.stringify(settings));
|
|
window.location.reload();
|
|
} catch (error) {
|
|
alert("Error\n" + error);
|
|
}
|
|
};
|
|
fileReader.readAsText(selectedFile);
|
|
}
|
|
this.importFromFile = function () {
|
|
fileInput.click();
|
|
fileInput.addEventListener("change", handleFileSelect);
|
|
};
|
|
// https://stackoverflow.com/a/34156339
|
|
function saveFile(content, fileName, contentType) {
|
|
var a = document.createElement("a");
|
|
var file = new Blob([content], { type: contentType });
|
|
a.href = URL.createObjectURL(file);
|
|
a.download = fileName;
|
|
a.click();
|
|
URL.revokeObjectURL(a.href);
|
|
}
|
|
this.exportToFile = function () {
|
|
saveFile(
|
|
JSON.stringify(settings),
|
|
"FX_client_settings.json",
|
|
"application/json"
|
|
);
|
|
};
|
|
|
|
this.syncFields = function () {
|
|
Object.keys(inputFields).forEach(function (key) {
|
|
inputFields[key].value = settings[key];
|
|
});
|
|
Object.keys(checkboxFields).forEach(function (key) {
|
|
checkboxFields[key].checked = settings[key];
|
|
});
|
|
customElements.forEach((element) => element.update?.(settings));
|
|
};
|
|
this.resetAll = function () {
|
|
if (
|
|
!confirm(
|
|
"Are you Really SURE you want to RESET ALL SETTINGS back to the default?"
|
|
)
|
|
)
|
|
return;
|
|
localStorage.removeItem("fx_settings");
|
|
window.location.reload();
|
|
};
|
|
this.applySettings = function () {
|
|
//setVarByName("bu", "px " + settings.fontName);
|
|
|
|
if (settings.customBackgroundUrl !== "") {
|
|
document.body.style.backgroundImage =
|
|
"url(" + settings.customBackgroundUrl + ")";
|
|
document.body.style.backgroundSize = "cover";
|
|
document.body.style.backgroundPosition = "center";
|
|
}
|
|
__fx.makeMainMenuTransparent = settings.customBackgroundUrl !== "";
|
|
};
|
|
|
|
if (settings.useFullscreenMode) tryEnterFullscreen();
|
|
})();
|
|
|
|
export function tryEnterFullscreen() {
|
|
if (document.fullscreenElement !== null || !document.fullscreenEnabled) return;
|
|
document.documentElement
|
|
.requestFullscreen({ navigationUI: "hide" })
|
|
.then(() => {
|
|
console.log("Fullscreen mode activated");
|
|
})
|
|
.catch((error) => {
|
|
console.warn("Could not enter fullscreen mode:", error);
|
|
});
|
|
}
|
|
|
|
const openCustomBackgroundFilePicker = () => {
|
|
const fileInput = document.getElementById("customBackgroundFileInput");
|
|
fileInput.click();
|
|
fileInput.addEventListener("change", handleFileSelect);
|
|
};
|
|
function handleFileSelect(event) {
|
|
const fileInput = event.target;
|
|
const selectedFile = fileInput.files[0];
|
|
console.log(fileInput.files);
|
|
console.log(fileInput.files[0]);
|
|
if (selectedFile) {
|
|
const fileUrl = URL.createObjectURL(selectedFile);
|
|
console.log("File URL:", fileUrl);
|
|
fileInput.value = "";
|
|
fileInput.removeEventListener("change", handleFileSelect);
|
|
}
|
|
}
|
|
|
|
WindowManager.add({
|
|
name: "settings",
|
|
element: document.querySelector(".settings"),
|
|
beforeOpen: function () {
|
|
settingsManager.syncFields();
|
|
},
|
|
});
|
|
|
|
if (localStorage.getItem("fx_settings") !== null) {
|
|
__fx.settings = settings = {
|
|
...settings,
|
|
...JSON.parse(localStorage.getItem("fx_settings")),
|
|
};
|
|
}
|
|
settingsManager.applySettings();
|
|
|
|
export default settingsManager;
|
|
export function getSettings() {
|
|
return settings;
|
|
}
|