1
0
mirror of https://github.com/tobspr/shapez.io.git synced 2025-06-13 13:04:03 +00:00
tobspr_shapez.io/src/js/game/hud/parts/statistics.js

256 lines
8.5 KiB
JavaScript
Raw Normal View History

2020-05-13 16:04:51 +00:00
import { InputReceiver } from "../../../core/input_receiver";
2020-08-06 09:28:28 +00:00
import { makeButton, makeDiv, removeAllChildren } from "../../../core/utils";
import { KeyActionMapper, KEYMAPPINGS } from "../../key_action_mapper";
2020-05-13 16:04:51 +00:00
import { enumAnalyticsDataSource } from "../../production_analytics";
2020-05-14 06:03:24 +00:00
import { BaseHUDPart } from "../base_hud_part";
import { DynamicDomAttach } from "../dynamic_dom_attach";
import { enumDisplayMode, HUDShapeStatisticsHandle } from "./statistics_handle";
2020-05-17 10:12:13 +00:00
import { T } from "../../../translations";
2020-05-14 08:25:00 +00:00
2020-08-06 09:28:28 +00:00
/**
* Capitalizes the first letter
* @param {string} str
*/
function capitalizeFirstLetter(str) {
return str.substr(0, 1).toUpperCase() + str.substr(1).toLowerCase();
}
2020-05-13 16:04:51 +00:00
export class HUDStatistics extends BaseHUDPart {
createElements(parent) {
this.background = makeDiv(parent, "ingame_HUD_Statistics", ["ingameDialog"]);
// DIALOG Inner / Wrapper
this.dialogInner = makeDiv(this.background, null, ["dialogInner"]);
2020-05-17 10:12:13 +00:00
this.title = makeDiv(this.dialogInner, null, ["title"], T.ingame.statistics.title);
2020-05-13 16:04:51 +00:00
this.closeButton = makeDiv(this.title, null, ["closeButton"]);
this.trackClicks(this.closeButton, this.close);
this.filterHeader = makeDiv(this.dialogInner, null, ["filterHeader"]);
2020-05-14 08:25:00 +00:00
this.sourceExplanation = makeDiv(this.dialogInner, null, ["sourceExplanation"]);
2020-05-13 16:04:51 +00:00
this.filtersDataSource = makeDiv(this.filterHeader, null, ["filtersDataSource"]);
this.filtersDisplayMode = makeDiv(this.filterHeader, null, ["filtersDisplayMode"]);
2020-05-17 10:12:13 +00:00
const dataSources = [
enumAnalyticsDataSource.produced,
enumAnalyticsDataSource.delivered,
enumAnalyticsDataSource.stored,
];
for (let i = 0; i < dataSources.length; ++i) {
const dataSource = dataSources[i];
const button = makeButton(
this.filtersDataSource,
["mode" + capitalizeFirstLetter(dataSource)],
T.ingame.statistics.dataSources[dataSource].title
);
this.trackClicks(button, () => this.setDataSource(dataSource));
}
2020-05-13 16:04:51 +00:00
const buttonDisplaySorted = makeButton(this.filtersDisplayMode, ["displaySorted"]);
2020-05-13 16:04:51 +00:00
const buttonDisplayDetailed = makeButton(this.filtersDisplayMode, ["displayDetailed"]);
const buttonDisplayIcons = makeButton(this.filtersDisplayMode, ["displayIcons"]);
this.trackClicks(buttonDisplaySorted, () => this.toggleSorted());
2020-05-13 16:04:51 +00:00
this.trackClicks(buttonDisplayIcons, () => this.setDisplayMode(enumDisplayMode.icons));
this.trackClicks(buttonDisplayDetailed, () => this.setDisplayMode(enumDisplayMode.detailed));
this.contentDiv = makeDiv(this.dialogInner, null, ["content"]);
}
/**
* @param {enumAnalyticsDataSource} source
*/
setDataSource(source) {
this.dataSource = source;
this.dialogInner.setAttribute("data-datasource", source);
2020-05-14 08:25:00 +00:00
2020-05-17 11:24:47 +00:00
this.sourceExplanation.innerText = T.ingame.statistics.dataSources[source].description;
2020-05-13 16:04:51 +00:00
if (this.visible) {
this.rerenderFull();
}
}
/**
* @param {enumDisplayMode} mode
*/
setDisplayMode(mode) {
this.displayMode = mode;
this.dialogInner.setAttribute("data-displaymode", mode);
if (this.visible) {
this.rerenderFull();
}
}
/**
* @param {boolean} sorted
*/
setSorted(sorted) {
this.sorted = sorted;
this.dialogInner.setAttribute("data-sorted", String(sorted));
if (this.visible) {
this.rerenderFull();
}
}
toggleSorted() {
this.setSorted(!this.sorted);
}
2020-05-13 16:04:51 +00:00
initialize() {
this.domAttach = new DynamicDomAttach(this.root, this.background, {
attachClass: "visible",
});
this.inputReciever = new InputReceiver("statistics");
this.keyActionMapper = new KeyActionMapper(this.root, this.inputReciever);
this.keyActionMapper.getBinding(KEYMAPPINGS.general.back).add(this.close, this);
this.keyActionMapper.getBinding(KEYMAPPINGS.ingame.menuClose).add(this.close, this);
this.keyActionMapper.getBinding(KEYMAPPINGS.ingame.menuOpenStats).add(this.close, this);
2020-05-13 16:04:51 +00:00
2020-05-14 06:03:24 +00:00
/** @type {Object.<string, HUDShapeStatisticsHandle>} */
2020-05-13 16:04:51 +00:00
this.activeHandles = {};
this.setSorted(true);
2020-05-13 16:04:51 +00:00
this.setDataSource(enumAnalyticsDataSource.produced);
this.setDisplayMode(enumDisplayMode.detailed);
this.intersectionObserver = new IntersectionObserver(this.intersectionCallback.bind(this), {
root: this.contentDiv,
});
this.lastFullRerender = 0;
this.close();
this.rerenderFull();
}
intersectionCallback(entries) {
for (let i = 0; i < entries.length; ++i) {
const entry = entries[i];
const handle = this.activeHandles[entry.target.getAttribute("data-shape-key")];
if (handle) {
handle.setVisible(entry.intersectionRatio > 0);
}
}
}
cleanup() {
document.body.classList.remove("ingameDialogOpen");
}
show() {
this.visible = true;
document.body.classList.add("ingameDialogOpen");
this.root.app.inputMgr.makeSureAttachedAndOnTop(this.inputReciever);
this.rerenderFull();
this.update();
}
close() {
this.visible = false;
document.body.classList.remove("ingameDialogOpen");
this.root.app.inputMgr.makeSureDetached(this.inputReciever);
this.update();
}
update() {
this.domAttach.update(this.visible);
if (this.visible) {
if (this.root.time.now() - this.lastFullRerender > 1) {
this.lastFullRerender = this.root.time.now();
this.lastPartialRerender = this.root.time.now();
this.rerenderFull();
}
this.rerenderPartial();
}
}
2020-05-14 06:03:24 +00:00
/**
* Performs a partial rerender, only updating graphs and counts
*/
2020-05-13 16:04:51 +00:00
rerenderPartial() {
for (const key in this.activeHandles) {
const handle = this.activeHandles[key];
handle.update(this.displayMode, this.dataSource);
}
}
2020-05-14 06:03:24 +00:00
/**
* Performs a full rerender, regenerating everything
*/
2020-05-13 16:04:51 +00:00
rerenderFull() {
2020-05-14 11:29:42 +00:00
for (const key in this.activeHandles) {
this.activeHandles[key].detach();
}
2020-05-13 16:04:51 +00:00
removeAllChildren(this.contentDiv);
// Now, attach new ones
2020-05-14 08:25:00 +00:00
let entries = null;
switch (this.dataSource) {
case enumAnalyticsDataSource.stored: {
entries = Object.entries(this.root.hubGoals.storedShapes);
break;
}
case enumAnalyticsDataSource.produced:
case enumAnalyticsDataSource.delivered: {
entries = Object.entries(this.root.productionAnalytics.getCurrentShapeRates(this.dataSource));
break;
}
}
const pinnedShapes = this.root.hud.parts.pinnedShapes;
entries.sort((a, b) => {
const aPinned = pinnedShapes.isShapePinned(a[0]);
const bPinned = pinnedShapes.isShapePinned(b[0]);
if (aPinned !== bPinned) {
return aPinned ? -1 : 1;
}
// Sort by shape key for some consistency
if (!this.sorted || b[1] == a[1]) {
return b[0].localeCompare(a[0]);
}
return b[1] - a[1];
});
2020-05-13 16:04:51 +00:00
let rendered = new Set();
for (let i = 0; i < Math.min(entries.length, 200); ++i) {
2020-05-13 16:04:51 +00:00
const entry = entries[i];
const shapeKey = entry[0];
let handle = this.activeHandles[shapeKey];
if (!handle) {
const definition = this.root.shapeDefinitionMgr.getShapeFromShortKey(shapeKey);
2020-05-14 06:03:24 +00:00
handle = this.activeHandles[shapeKey] = new HUDShapeStatisticsHandle(
2020-05-13 16:04:51 +00:00
this.root,
definition,
this.intersectionObserver
);
}
rendered.add(shapeKey);
handle.attach(this.contentDiv);
}
for (const key in this.activeHandles) {
if (!rendered.has(key)) {
this.activeHandles[key].destroy();
delete this.activeHandles[key];
}
}
if (entries.length === 0) {
this.contentDiv.innerHTML = `
2020-05-17 10:12:13 +00:00
<strong class="noEntries">${T.ingame.statistics.noShapesProduced}</strong>`;
2020-05-13 16:04:51 +00:00
}
this.contentDiv.classList.toggle("hasEntries", entries.length > 0);
}
}