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/pinned_shapes.js

292 lines
8.8 KiB
JavaScript
Raw Normal View History

2020-05-14 11:29:42 +00:00
import { ClickDetector } from "../../../core/click_detector";
import { formatBigNumber, makeDiv, arrayDelete, arrayDeleteValue } from "../../../core/utils";
2020-05-14 11:29:42 +00:00
import { ShapeDefinition } from "../../shape_definition";
2020-05-14 17:12:58 +00:00
import { BaseHUDPart } from "../base_hud_part";
import { blueprintShape, UPGRADES } from "../../upgrades";
2020-05-28 13:36:38 +00:00
import { enumHubGoalRewards } from "../../tutorial_goals";
2020-05-14 11:29:42 +00:00
/**
* Manages the pinned shapes on the left side of the screen
*/
2020-05-14 11:29:42 +00:00
export class HUDPinnedShapes extends BaseHUDPart {
constructor(root) {
super(root);
/**
* Store a list of pinned shapes
* @type {Array<string>}
*/
this.pinnedShapes = [];
/**
* Store handles to the currently rendered elements, so we can update them more
* convenient. Also allows for cleaning up handles.
* @type {Array<{
* key: string,
* amountLabel: HTMLElement,
* lastRenderedValue: string,
* element: HTMLElement,
2020-06-22 12:32:24 +00:00
* detector?: ClickDetector,
* infoDetector?: ClickDetector
* }>}
*/
this.handles = [];
}
2020-05-14 11:29:42 +00:00
createElements(parent) {
this.element = makeDiv(parent, "ingame_HUD_PinnedShapes", []);
}
/**
* Serializes the pinned shapes
*/
serialize() {
return {
shapes: this.pinnedShapes,
};
}
/**
* Deserializes the pinned shapes
* @param {{ shapes: Array<string>}} data
*/
deserialize(data) {
if (!data || !data.shapes || !Array.isArray(data.shapes)) {
return "Invalid pinned shapes data";
}
this.pinnedShapes = data.shapes;
}
/**
* Initializes the hud component
*/
2020-05-14 11:29:42 +00:00
initialize() {
// Connect to any relevant signals
2020-05-14 11:29:42 +00:00
this.root.signals.storyGoalCompleted.add(this.rerenderFull, this);
this.root.signals.upgradePurchased.add(this.updateShapesAfterUpgrade, this);
this.root.signals.postLoadHook.add(this.rerenderFull, this);
2020-05-14 11:29:42 +00:00
this.root.hud.signals.shapePinRequested.add(this.pinNewShape, this);
this.root.hud.signals.shapeUnpinRequested.add(this.unpinShape, this);
// Perform initial render
this.updateShapesAfterUpgrade();
2020-05-14 11:29:42 +00:00
}
/**
* Updates all shapes after an upgrade has been purchased and removes the unused ones
2020-05-14 11:29:42 +00:00
*/
updateShapesAfterUpgrade() {
for (let i = 0; i < this.pinnedShapes.length; ++i) {
const key = this.pinnedShapes[i];
if (key === blueprintShape) {
// Ignore blueprint shapes
continue;
}
let goal = this.findGoalValueForShape(key);
if (!goal) {
// Seems no longer relevant
this.pinnedShapes.splice(i, 1);
i -= 1;
}
2020-05-14 11:29:42 +00:00
}
this.rerenderFull();
}
/**
* Finds the current goal for the given key. If the key is the story goal, returns
* the story goal. If its the blueprint shape, no goal is returned. Otherwise
* it's searched for upgrades.
* @param {string} key
*/
findGoalValueForShape(key) {
2020-05-14 11:40:38 +00:00
if (key === this.root.hubGoals.currentGoal.definition.getHash()) {
return this.root.hubGoals.currentGoal.required;
2020-05-14 11:40:38 +00:00
}
2020-05-28 13:36:38 +00:00
if (key === blueprintShape) {
return null;
2020-05-28 13:36:38 +00:00
}
// Check if this shape is required for any upgrade
for (const upgradeId in UPGRADES) {
const { tiers } = UPGRADES[upgradeId];
const currentTier = this.root.hubGoals.getUpgradeLevel(upgradeId);
const tierHandle = tiers[currentTier];
if (!tierHandle) {
// Max level
continue;
}
for (let i = 0; i < tierHandle.required.length; ++i) {
const { shape, amount } = tierHandle.required[i];
if (shape === key) {
return amount;
}
2020-05-14 11:40:38 +00:00
}
}
return null;
2020-05-14 11:29:42 +00:00
}
/**
* Returns whether a given shape is currently pinned
* @param {string} key
*/
isShapePinned(key) {
if (key === this.root.hubGoals.currentGoal.definition.getHash() || key === blueprintShape) {
// This is a "special" shape which is always pinned
return true;
}
return this.pinnedShapes.indexOf(key) >= 0;
}
/**
* Rerenders the whole component
*/
2020-05-14 11:29:42 +00:00
rerenderFull() {
2020-05-14 11:40:38 +00:00
const currentGoal = this.root.hubGoals.currentGoal;
const currentKey = currentGoal.definition.getHash();
2020-05-14 11:29:42 +00:00
// First, remove all old shapes
2020-05-14 11:29:42 +00:00
for (let i = 0; i < this.handles.length; ++i) {
this.handles[i].element.remove();
const detector = this.handles[i].detector;
if (detector) {
detector.cleanup();
}
2020-06-22 12:32:24 +00:00
const infoDetector = this.handles[i].infoDetector;
if (infoDetector) {
infoDetector.cleanup();
}
2020-05-14 11:29:42 +00:00
}
this.handles = [];
// Pin story goal
this.internalPinShape(currentKey, false, "goal");
2020-05-14 11:29:42 +00:00
// Pin blueprint shape as well
2020-05-28 13:36:38 +00:00
if (this.root.hubGoals.isRewardUnlocked(enumHubGoalRewards.reward_blueprints)) {
this.internalPinShape(blueprintShape, false, "blueprint");
2020-05-28 13:36:38 +00:00
}
// Pin manually pinned shapes
2020-05-14 11:29:42 +00:00
for (let i = 0; i < this.pinnedShapes.length; ++i) {
const key = this.pinnedShapes[i];
2020-05-14 11:40:38 +00:00
if (key !== currentKey) {
this.internalPinShape(key);
2020-05-14 11:29:42 +00:00
}
}
}
/**
* Pins a new shape
2020-05-14 11:29:42 +00:00
* @param {string} key
* @param {boolean} canUnpin
* @param {string=} className
2020-05-14 11:29:42 +00:00
*/
internalPinShape(key, canUnpin = true, className = null) {
2020-05-14 11:29:42 +00:00
const definition = this.root.shapeDefinitionMgr.getShapeFromShortKey(key);
const element = makeDiv(this.element, null, ["shape"]);
const canvas = definition.generateAsCanvas(120);
element.appendChild(canvas);
if (className) {
element.classList.add(className);
}
2020-05-14 11:29:42 +00:00
let detector = null;
if (canUnpin) {
element.classList.add("unpinable");
detector = new ClickDetector(element, {
consumeEvents: true,
preventDefault: true,
2020-06-22 12:32:24 +00:00
targetOnly: true,
2020-05-14 11:29:42 +00:00
});
detector.click.add(() => this.unpinShape(key));
} else {
element.classList.add("marked");
}
2020-06-22 12:32:24 +00:00
// Show small info icon
const infoButton = document.createElement("button");
infoButton.classList.add("infoButton");
element.appendChild(infoButton);
const infoDetector = new ClickDetector(infoButton, {
consumeEvents: true,
preventDefault: true,
targetOnly: true,
});
infoDetector.click.add(() => this.root.hud.signals.viewShapeDetailsRequested.dispatch(definition));
2020-05-14 11:40:38 +00:00
const amountLabel = makeDiv(element, null, ["amountLabel"], "");
2020-05-28 16:07:57 +00:00
const goal = this.findGoalValueForShape(key);
2020-05-28 16:07:57 +00:00
if (goal) {
makeDiv(element, null, ["goalLabel"], "/" + formatBigNumber(goal));
}
2020-05-14 11:29:42 +00:00
this.handles.push({
key,
element,
amountLabel,
lastRenderedValue: "",
2020-06-22 12:32:24 +00:00
detector,
infoDetector,
2020-05-14 11:29:42 +00:00
});
}
/**
* Updates all amount labels
*/
2020-05-14 11:29:42 +00:00
update() {
for (let i = 0; i < this.handles.length; ++i) {
const handle = this.handles[i];
const currentValue = this.root.hubGoals.getShapesStoredByKey(handle.key);
const currentValueFormatted = formatBigNumber(currentValue);
if (currentValueFormatted !== handle.lastRenderedValue) {
handle.lastRenderedValue = currentValueFormatted;
handle.amountLabel.innerText = currentValueFormatted;
const goal = this.findGoalValueForShape(handle.key);
handle.element.classList.toggle("completed", goal && currentValue > goal);
2020-05-14 11:29:42 +00:00
}
}
}
2020-05-14 11:40:38 +00:00
/**
* Unpins a shape
* @param {string} key
*/
2020-05-14 11:29:42 +00:00
unpinShape(key) {
arrayDeleteValue(this.pinnedShapes, key);
this.rerenderFull();
2020-05-14 11:29:42 +00:00
}
/**
* Requests to pin a new shape
2020-05-14 11:29:42 +00:00
* @param {ShapeDefinition} definition
*/
pinNewShape(definition) {
2020-05-14 11:29:42 +00:00
const key = definition.getHash();
if (key === this.root.hubGoals.currentGoal.definition.getHash()) {
// Can not pin current goal
return;
}
2020-05-28 13:36:38 +00:00
if (key === blueprintShape) {
// Can not pin the blueprint shape
2020-05-28 13:36:38 +00:00
return;
}
// Check if its already pinned
if (this.pinnedShapes.indexOf(key) >= 0) {
return;
2020-05-14 11:29:42 +00:00
}
2020-05-14 11:40:38 +00:00
this.pinnedShapes.push(key);
2020-05-14 11:40:38 +00:00
this.rerenderFull();
2020-05-14 11:29:42 +00:00
}
}