First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
import { CacheableFunction } from "@/typings/cacheableFunction";
|
|
|
|
import { Achievement } from "@/typings/features/achievement";
|
2021-08-19 00:25:49 -05:00
|
|
|
import { Board } from "@/typings/features/board";
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
import { Buyable } from "@/typings/features/buyable";
|
|
|
|
import { Challenge } from "@/typings/features/challenge";
|
|
|
|
import { Clickable } from "@/typings/features/clickable";
|
|
|
|
import {
|
|
|
|
Feature,
|
|
|
|
Features,
|
|
|
|
GridFeatures,
|
|
|
|
RawFeature,
|
|
|
|
RawFeatures,
|
|
|
|
RawGridFeatures
|
|
|
|
} from "@/typings/features/feature";
|
|
|
|
import { Grid } from "@/typings/features/grid";
|
|
|
|
import { Hotkey } from "@/typings/features/hotkey";
|
|
|
|
import { Milestone } from "@/typings/features/milestone";
|
|
|
|
import { Microtab, Subtab } from "@/typings/features/subtab";
|
|
|
|
import { Upgrade } from "@/typings/features/upgrade";
|
|
|
|
import { Layer, RawLayer } from "@/typings/layer";
|
|
|
|
import { PlayerData } from "@/typings/player";
|
|
|
|
import { State } from "@/typings/state";
|
|
|
|
import Decimal, { DecimalSource } from "@/util/bignum";
|
|
|
|
import { isFunction } from "@/util/common";
|
|
|
|
import {
|
|
|
|
defaultLayerProperties,
|
2021-08-19 00:25:49 -05:00
|
|
|
getStartingBoards,
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
getStartingBuyables,
|
|
|
|
getStartingChallenges,
|
|
|
|
getStartingClickables,
|
|
|
|
noCache
|
|
|
|
} from "@/util/layers";
|
|
|
|
import { createGridProxy, createLayerProxy } from "@/util/proxies";
|
|
|
|
import { applyPlayerData } from "@/util/save";
|
|
|
|
import clone from "lodash.clonedeep";
|
|
|
|
import { isRef } from "vue";
|
2021-08-20 23:21:13 -05:00
|
|
|
import { ProgressDisplay, Shape } from "./enums";
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
import { default as playerProxy } from "./player";
|
|
|
|
|
|
|
|
export const layers: Record<string, Readonly<Layer>> = {};
|
|
|
|
export const hotkeys: Hotkey[] = [];
|
|
|
|
window.layers = layers;
|
|
|
|
|
|
|
|
export function addLayer(layer: RawLayer, player?: Partial<PlayerData>): void {
|
|
|
|
player = player || playerProxy;
|
|
|
|
|
|
|
|
// Check for required properties
|
|
|
|
if (!("id" in layer)) {
|
|
|
|
console.error(`Cannot add layer without a "id" property!`, layer);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (layer.type === "static" || layer.type === "normal") {
|
|
|
|
const missingProperty = ["baseAmount", "requires"].find(prop => !(prop in layer));
|
|
|
|
if (missingProperty) {
|
|
|
|
console.error(`Cannot add layer without a "${missingProperty}" property!`, layer);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clone object to prevent modifying the original
|
|
|
|
layer = clone(layer);
|
|
|
|
|
|
|
|
setDefault(player, "layers", {});
|
2021-08-18 00:18:23 -05:00
|
|
|
player.layers[layer.id] = applyPlayerData(
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
{
|
|
|
|
points: new Decimal(0),
|
|
|
|
unlocked: false,
|
|
|
|
resetTime: new Decimal(0),
|
|
|
|
upgrades: [],
|
|
|
|
achievements: [],
|
|
|
|
milestones: [],
|
|
|
|
infoboxes: {},
|
|
|
|
buyables: getStartingBuyables(layer.buyables?.data),
|
|
|
|
clickables: getStartingClickables(layer.clickables?.data),
|
|
|
|
challenges: getStartingChallenges(layer.challenges?.data),
|
2021-08-20 21:42:11 -05:00
|
|
|
boards: player.layers[layer.id]?.boards || getStartingBoards(layer.boards?.data),
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
grids: {},
|
|
|
|
confirmRespecBuyables: false,
|
|
|
|
...(layer.startData?.() || {})
|
|
|
|
},
|
2021-08-18 00:18:23 -05:00
|
|
|
player.layers[layer.id]
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
// Set default property values
|
|
|
|
layer = Object.assign({}, defaultLayerProperties, layer);
|
|
|
|
layer.layer = layer.id;
|
|
|
|
if (layer.type === "static" && layer.base == undefined) {
|
|
|
|
layer.base = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process each feature
|
|
|
|
const uncachedProperties = ["startData", "click", "update", "reset", "hardReset"];
|
|
|
|
for (const property of uncachedProperties) {
|
|
|
|
if (layer[property] && !isRef(layer.property) && isFunction(layer[property])) {
|
|
|
|
(layer[property] as CacheableFunction).forceCached = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.upgrades) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["upgrades"]>, Upgrade>(layer.id, layer.upgrades);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
setRowCol(layer.upgrades);
|
|
|
|
for (const id in layer.upgrades.data) {
|
|
|
|
layer.upgrades.data[id].bought = function() {
|
|
|
|
return (
|
|
|
|
!layers[this.layer].deactivated &&
|
|
|
|
playerProxy.layers[this.layer].upgrades.some(
|
|
|
|
(upgrade: string | number) => upgrade == id
|
|
|
|
)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
setDefault(layer.upgrades.data[id], "canAfford", function() {
|
|
|
|
if (this.currencyInternalName) {
|
|
|
|
const name = this.currencyInternalName;
|
|
|
|
if (this.currencyLocation) {
|
|
|
|
return !Decimal.lt(this.currencyLocation[name], this.cost);
|
|
|
|
} else if (this.currencyLayer) {
|
|
|
|
return !Decimal.lt(
|
|
|
|
playerProxy.layers[this.currencyLayer][name] as DecimalSource,
|
|
|
|
this.cost
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return !Decimal.lt(playerProxy[name] as DecimalSource, this.cost);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return !playerProxy.layers[this.layer].points.lt(this.cost);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
setDefault(
|
|
|
|
layer.upgrades.data[id],
|
|
|
|
"pay",
|
|
|
|
function() {
|
|
|
|
if (this.bought || !this.canAfford) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (this.currencyInternalName) {
|
|
|
|
const name = this.currencyInternalName;
|
|
|
|
if (this.currencyLocation) {
|
|
|
|
if (Decimal.lt(this.currencyLocation[name], this.cost)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.currencyLocation[name] = Decimal.sub(
|
|
|
|
this.currencyLocation[name],
|
|
|
|
this.cost
|
|
|
|
);
|
|
|
|
} else if (this.currencyLayer) {
|
|
|
|
const lr = this.currencyLayer;
|
|
|
|
if (
|
|
|
|
Decimal.lt(playerProxy.layers[lr][name] as DecimalSource, this.cost)
|
|
|
|
) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
playerProxy.layers[lr][name] = Decimal.sub(
|
|
|
|
playerProxy.layers[lr][name] as DecimalSource,
|
|
|
|
this.cost
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
if (Decimal.lt(playerProxy[name] as DecimalSource, this.cost)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
playerProxy[name] = Decimal.sub(
|
|
|
|
playerProxy[name] as DecimalSource,
|
|
|
|
this.cost
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (playerProxy.layers[this.layer].points.lt(this.cost)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
playerProxy.layers[this.layer].points = playerProxy.layers[
|
|
|
|
this.layer
|
|
|
|
].points.sub(this.cost);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
false
|
|
|
|
);
|
|
|
|
setDefault(
|
|
|
|
layer.upgrades.data[id],
|
|
|
|
"buy",
|
|
|
|
function() {
|
|
|
|
if (this.bought || !this.canAfford) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.pay();
|
|
|
|
playerProxy.layers[this.layer].upgrades.push(this.id);
|
|
|
|
this.onPurchase?.();
|
|
|
|
},
|
|
|
|
false
|
|
|
|
);
|
|
|
|
setDefault(layer.upgrades.data[id], "onPurchase", undefined, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.achievements) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["achievements"]>, Achievement>(
|
|
|
|
layer.id,
|
|
|
|
layer.achievements
|
|
|
|
);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
setRowCol(layer.achievements);
|
|
|
|
for (const id in layer.achievements.data) {
|
|
|
|
layer.achievements.data[id].earned = function() {
|
|
|
|
return (
|
|
|
|
!layers[this.layer].deactivated &&
|
|
|
|
playerProxy.layers[this.layer].achievements.some(
|
|
|
|
(achievement: string | number) => achievement == id
|
|
|
|
)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
setDefault(layer.achievements.data[id], "onComplete", undefined, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.challenges) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["challenges"]>, Challenge>(layer.id, layer.challenges);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
setRowCol(layer.challenges);
|
|
|
|
layer.activeChallenge = function() {
|
|
|
|
return Object.values(this.challenges!.data).find(
|
|
|
|
(challenge: Challenge) => challenge.active
|
|
|
|
);
|
|
|
|
};
|
|
|
|
for (const id in layer.challenges.data) {
|
|
|
|
layer.challenges.data[id].shown = function() {
|
|
|
|
return (
|
|
|
|
this.unlocked !== false && (playerProxy.hideChallenges === false || !this.maxed)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
layer.challenges.data[id].completed = function() {
|
|
|
|
return (
|
|
|
|
!layers[this.layer].deactivated &&
|
|
|
|
playerProxy.layers[this.layer].challenges[id]?.gt(0)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
layer.challenges.data[id].completions = function() {
|
|
|
|
return playerProxy.layers[this.layer].challenges[id];
|
|
|
|
};
|
|
|
|
layer.challenges.data[id].maxed = function() {
|
|
|
|
return (
|
|
|
|
!layers[this.layer].deactivated &&
|
|
|
|
Decimal.gte(playerProxy.layers[this.layer].challenges[id], this.completionLimit)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
layer.challenges.data[id].active = function() {
|
|
|
|
return (
|
|
|
|
!layers[this.layer].deactivated &&
|
|
|
|
playerProxy.layers[this.layer].activeChallenge === id
|
|
|
|
);
|
|
|
|
};
|
|
|
|
layer.challenges.data[id].toggle = noCache(function(this: Challenge) {
|
|
|
|
const exiting = playerProxy.layers[this.layer].activeChallenge === id;
|
|
|
|
if (exiting) {
|
|
|
|
if (this.canComplete && !this.maxed) {
|
|
|
|
let completions: boolean | DecimalSource = this.canComplete;
|
|
|
|
if (completions === true) {
|
|
|
|
completions = 1;
|
|
|
|
}
|
|
|
|
playerProxy.layers[this.layer].challenges[id] = Decimal.min(
|
|
|
|
playerProxy.layers[this.layer].challenges[id].add(completions),
|
|
|
|
this.completionLimit
|
|
|
|
);
|
|
|
|
this.onComplete?.();
|
|
|
|
}
|
|
|
|
playerProxy.layers[this.layer].activeChallenge = null;
|
|
|
|
this.onExit?.();
|
|
|
|
layers[this.layer].reset(true);
|
|
|
|
} else if (!exiting && this.canStart) {
|
|
|
|
layers[this.layer].reset(true);
|
|
|
|
playerProxy.layers[this.layer].activeChallenge = id;
|
|
|
|
this.onEnter?.();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
setDefault(layer.challenges.data[id], "onComplete", undefined, false);
|
|
|
|
setDefault(layer.challenges.data[id], "onEnter", undefined, false);
|
|
|
|
setDefault(layer.challenges.data[id], "onExit", undefined, false);
|
|
|
|
setDefault(layer.challenges.data[id], "canStart", true);
|
|
|
|
setDefault(layer.challenges.data[id], "completionLimit", new Decimal(1));
|
|
|
|
setDefault(layer.challenges.data[id], "mark", function() {
|
|
|
|
return Decimal.gt(this.completionLimit, 1) && this.maxed;
|
|
|
|
});
|
|
|
|
setDefault(layer.challenges.data[id], "canComplete", function() {
|
|
|
|
if (!this.active) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (this.currencyInternalName) {
|
|
|
|
const name = this.currencyInternalName;
|
|
|
|
if (this.currencyLocation) {
|
|
|
|
return !Decimal.lt(this.currencyLocation[name], this.goal);
|
|
|
|
} else if (this.currencyLayer) {
|
|
|
|
const lr = this.currencyLayer;
|
|
|
|
return !Decimal.lt(
|
|
|
|
playerProxy.layers[lr][name] as DecimalSource,
|
|
|
|
this.goal
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return !Decimal.lt(playerProxy[name] as DecimalSource, this.goal);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return !playerProxy.points.lt(this.goal);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.buyables) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["buyables"]>, Buyable>(layer.id, layer.buyables);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
setRowCol(layer.buyables);
|
|
|
|
setDefault(layer.buyables, "respec", undefined, false);
|
|
|
|
setDefault(
|
|
|
|
layer.buyables,
|
|
|
|
"reset",
|
|
|
|
function(this: NonNullable<Layer["buyables"]>) {
|
|
|
|
playerProxy.layers[this.layer].buyables = getStartingBuyables(layer.buyables?.data);
|
|
|
|
},
|
|
|
|
false
|
|
|
|
);
|
|
|
|
for (const id in layer.buyables.data) {
|
|
|
|
layer.buyables.data[id].amount = function() {
|
|
|
|
return playerProxy.layers[this.layer].buyables[id];
|
|
|
|
};
|
|
|
|
layer.buyables.data[id].amountSet = function(amount: Decimal) {
|
|
|
|
playerProxy.layers[this.layer].buyables[id] = amount;
|
|
|
|
};
|
|
|
|
layer.buyables.data[id].canBuy = function() {
|
|
|
|
return (
|
|
|
|
!layers[this.layer].deactivated &&
|
|
|
|
this.unlocked !== false &&
|
|
|
|
this.canAfford !== false &&
|
|
|
|
Decimal.lt(playerProxy.layers[this.layer].buyables[id], this.purchaseLimit)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
setDefault(layer.buyables.data[id], "purchaseLimit", new Decimal(Infinity));
|
|
|
|
setDefault(layer.buyables.data[id], "sellOne", undefined, false);
|
|
|
|
setDefault(layer.buyables.data[id], "sellAll", undefined, false);
|
|
|
|
if (layer.buyables.data[id].cost != undefined) {
|
|
|
|
setDefault(
|
|
|
|
layer.buyables.data[id],
|
|
|
|
"buy",
|
|
|
|
function() {
|
|
|
|
if (this.canBuy) {
|
|
|
|
playerProxy.layers[this.layer].points = playerProxy.layers[
|
|
|
|
this.layer
|
|
|
|
].points.sub(this.cost!);
|
|
|
|
this.amount = this.amount.add(1);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
false
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.clickables) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["clickables"]>, Clickable>(layer.id, layer.clickables);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
setRowCol(layer.clickables);
|
|
|
|
setDefault(layer.clickables, "masterButtonClick", undefined, false);
|
|
|
|
if (layer.clickables.masterButtonDisplay != undefined) {
|
|
|
|
setDefault(layer.clickables, "showMasterButton", true);
|
|
|
|
}
|
|
|
|
for (const id in layer.clickables.data) {
|
|
|
|
layer.clickables.data[id].state = function() {
|
|
|
|
return playerProxy.layers[this.layer].clickables[id];
|
|
|
|
};
|
|
|
|
layer.clickables.data[id].stateSet = function(state: State) {
|
|
|
|
playerProxy.layers[this.layer].clickables[id] = state;
|
|
|
|
};
|
|
|
|
setDefault(layer.clickables.data[id], "canClick", true);
|
|
|
|
setDefault(layer.clickables.data[id], "click", undefined, false);
|
|
|
|
setDefault(layer.clickables.data[id], "hold", undefined, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.milestones) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["milestones"]>, Milestone>(layer.id, layer.milestones);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
for (const id in layer.milestones.data) {
|
|
|
|
layer.milestones.data[id].earned = function() {
|
|
|
|
return (
|
|
|
|
!layer.deactivated &&
|
|
|
|
playerProxy.layers[this.layer].milestones.some(
|
|
|
|
(milestone: string | number) => milestone == id
|
|
|
|
)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
layer.milestones.data[id].shown = function() {
|
|
|
|
if (!this.unlocked) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
switch (playerProxy.msDisplay) {
|
|
|
|
default:
|
|
|
|
case "all":
|
|
|
|
return true;
|
|
|
|
case "last":
|
|
|
|
return (
|
|
|
|
this.optionsDisplay ||
|
|
|
|
!this.earned ||
|
|
|
|
playerProxy.layers[this.layer].milestones[
|
|
|
|
playerProxy.layers[this.layer].milestones.length - 1
|
|
|
|
] === this.id
|
|
|
|
);
|
|
|
|
case "configurable":
|
|
|
|
return this.optionsDisplay || !this.earned;
|
|
|
|
case "incomplete":
|
|
|
|
return !this.earned;
|
|
|
|
case "none":
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
setDefault(layer.milestones.data[id], "done", false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.grids) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setupFeatures<NonNullable<RawLayer["grids"]>, Grid>(layer.id, layer.grids);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
for (const id in layer.grids.data) {
|
2021-08-18 00:18:23 -05:00
|
|
|
setDefault(player.layers[layer.id].grids, id, {});
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
layer.grids.data[id].getData = function(cell): State {
|
|
|
|
if (playerProxy.layers[this.layer].grids[id][cell] != undefined) {
|
|
|
|
return playerProxy.layers[this.layer].grids[id][cell];
|
|
|
|
}
|
|
|
|
if (isFunction(this.getStartData)) {
|
|
|
|
return (this.getStartData as (this: Grid, cell: string | number) => State)(
|
|
|
|
cell
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return this.getStartData;
|
|
|
|
};
|
|
|
|
layer.grids.data[id].setData = function(cell, data) {
|
|
|
|
playerProxy.layers[this.layer].grids[id][cell] = data;
|
|
|
|
};
|
|
|
|
setDefault(layer.grids.data[id], "getUnlocked", true, false);
|
|
|
|
setDefault(layer.grids.data[id], "getCanClick", true, false);
|
|
|
|
setDefault(layer.grids.data[id], "getStartData", "", false);
|
|
|
|
setDefault(layer.grids.data[id], "getStyle", undefined, false);
|
|
|
|
setDefault(layer.grids.data[id], "click", undefined, false);
|
|
|
|
setDefault(layer.grids.data[id], "hold", undefined, false);
|
|
|
|
setDefault(layer.grids.data[id], "getTitle", undefined, false);
|
|
|
|
layer.grids.data[id] = createGridProxy(layer.grids.data[id]) as Grid;
|
|
|
|
}
|
|
|
|
}
|
2021-08-19 00:25:49 -05:00
|
|
|
if (layer.boards) {
|
|
|
|
setupFeatures<NonNullable<RawLayer["boards"]>, Board>(layer.id, layer.boards);
|
|
|
|
for (const id in layer.boards.data) {
|
|
|
|
setDefault(layer.boards.data[id], "width", "100%");
|
|
|
|
setDefault(layer.boards.data[id], "height", "400px");
|
2021-08-22 01:50:03 -05:00
|
|
|
setDefault(layer.boards.data[id], "nodes", function() {
|
|
|
|
return playerProxy.layers[this.layer].boards[this.id].nodes;
|
|
|
|
});
|
|
|
|
setDefault(layer.boards.data[id], "selectedNode", function() {
|
|
|
|
return playerProxy.layers[this.layer].boards[this.id].nodes.find(
|
|
|
|
node => node.id === playerProxy.layers[this.layer].boards[this.id].selectedNode
|
|
|
|
);
|
|
|
|
});
|
|
|
|
setDefault(layer.boards.data[id], "selectedAction", function() {
|
|
|
|
if (this.selectedNode == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const nodeType = layers[this.layer].boards!.data[this.id].types[
|
|
|
|
this.selectedNode.type
|
|
|
|
];
|
|
|
|
if (nodeType.actions === null) {
|
|
|
|
return null;
|
|
|
|
}
|
2021-08-22 22:57:59 -05:00
|
|
|
const actions =
|
|
|
|
typeof nodeType.actions === "function"
|
|
|
|
? nodeType.actions(this.selectedNode)
|
|
|
|
: nodeType.actions;
|
|
|
|
return actions?.find(
|
|
|
|
action =>
|
|
|
|
action.id === playerProxy.layers[this.layer].boards[this.id].selectedAction
|
|
|
|
);
|
|
|
|
});
|
|
|
|
setDefault(layer.boards.data[id], "links", function() {
|
|
|
|
if (this.selectedAction == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (this.selectedAction.links) {
|
|
|
|
if (typeof this.selectedAction.links === "function") {
|
|
|
|
return this.selectedAction.links(this.selectedNode);
|
|
|
|
}
|
|
|
|
return this.selectedAction.links;
|
2021-08-22 01:50:03 -05:00
|
|
|
}
|
2021-08-22 22:57:59 -05:00
|
|
|
return null;
|
2021-08-22 01:50:03 -05:00
|
|
|
});
|
2021-08-19 00:25:49 -05:00
|
|
|
for (const nodeType in layer.boards.data[id].types) {
|
|
|
|
layer.boards.data[id].types[nodeType].layer = layer.id;
|
|
|
|
layer.boards.data[id].types[nodeType].id = id;
|
|
|
|
layer.boards.data[id].types[nodeType].type = nodeType;
|
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "size", 50);
|
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "draggable", false);
|
2021-08-20 23:21:13 -05:00
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "shape", Shape.Circle);
|
2021-08-19 00:25:49 -05:00
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "canAccept", false);
|
2021-08-22 01:50:03 -05:00
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "actionDistance", Math.PI / 6);
|
2021-08-19 00:25:49 -05:00
|
|
|
setDefault(
|
|
|
|
layer.boards.data[id].types[nodeType],
|
|
|
|
"progressDisplay",
|
|
|
|
ProgressDisplay.Fill
|
|
|
|
);
|
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "nodes", function() {
|
2021-08-22 01:50:03 -05:00
|
|
|
return playerProxy.layers[this.layer].boards[this.id].nodes.filter(
|
2021-08-19 00:25:49 -05:00
|
|
|
node => node.type === this.type
|
|
|
|
);
|
|
|
|
});
|
2021-08-22 01:50:03 -05:00
|
|
|
setDefault(layer.boards.data[id].types[nodeType], "onClick", function(node) {
|
|
|
|
playerProxy.layers[this.layer].boards[this.id].selectedNode = node.id;
|
|
|
|
});
|
2021-08-19 00:25:49 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
if (layer.subtabs) {
|
|
|
|
layer.activeSubtab = function() {
|
|
|
|
if (
|
|
|
|
layers[this.layer].subtabs![playerProxy.subtabs[this.layer].mainTabs!] &&
|
|
|
|
layers[this.layer].subtabs![playerProxy.subtabs[this.layer].mainTabs!].unlocked !==
|
|
|
|
false
|
|
|
|
) {
|
|
|
|
return layers[this.layer].subtabs![playerProxy.subtabs[this.layer].mainTabs!];
|
|
|
|
}
|
|
|
|
// Default to first unlocked tab
|
|
|
|
return Object.values(layers[this.layer].subtabs!).find(
|
|
|
|
(subtab: Subtab) => subtab.unlocked !== false
|
|
|
|
);
|
|
|
|
};
|
|
|
|
setDefault(player, "subtabs", {});
|
2021-08-17 18:39:11 -05:00
|
|
|
setDefault(player.subtabs, layer.id, {});
|
|
|
|
setDefault(player.subtabs[layer.id], "mainTabs", Object.keys(layer.subtabs)[0]);
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
for (const id in layer.subtabs) {
|
|
|
|
layer.subtabs[id].active = function() {
|
|
|
|
return playerProxy.subtabs[this.layer].mainTabs === this.id;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.microtabs) {
|
|
|
|
setDefault(player, "subtabs", {});
|
2021-08-17 18:39:11 -05:00
|
|
|
setDefault(player.subtabs, layer.id, {});
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
for (const family in layer.microtabs) {
|
|
|
|
if (Object.keys(layer.microtabs[family]).length === 0) {
|
|
|
|
console.warn(
|
|
|
|
"Cannot create microtab family with 0 tabs",
|
|
|
|
layer.id,
|
|
|
|
family,
|
|
|
|
layer.microtabs[family]
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
layer.microtabs[family].activeMicrotab = function() {
|
|
|
|
if (
|
|
|
|
this.data[playerProxy.subtabs[this.layer as string][family]] &&
|
|
|
|
this.data[playerProxy.subtabs[this.layer as string][family]].unlocked !== false
|
|
|
|
) {
|
|
|
|
return this[playerProxy.subtabs[this.layer as string][family]];
|
|
|
|
}
|
|
|
|
// Default to first unlocked tab
|
|
|
|
const firstUnlocked: string | undefined = Object.keys(this).find(
|
|
|
|
microtab =>
|
|
|
|
microtab !== "activeMicrotab" && this.data[microtab].unlocked !== false
|
|
|
|
);
|
|
|
|
return firstUnlocked != undefined ? this[firstUnlocked] : undefined;
|
|
|
|
};
|
|
|
|
setDefault(
|
2021-08-17 18:39:11 -05:00
|
|
|
player.subtabs[layer.id],
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
family,
|
|
|
|
Object.keys(layer.microtabs[family]).find(tab => tab !== "activeMicrotab")!
|
|
|
|
);
|
|
|
|
layer.microtabs[family].layer = layer.id;
|
|
|
|
layer.microtabs[family].family = family;
|
|
|
|
for (const id in layer.microtabs[family].data) {
|
|
|
|
const microtab: RawFeature<Microtab> = layer.microtabs[family].data[id];
|
|
|
|
microtab.layer = layer.id;
|
|
|
|
microtab.family = family;
|
|
|
|
microtab.id = id;
|
|
|
|
microtab.active = function() {
|
|
|
|
return playerProxy.subtabs[this.layer][this.family] === this.id;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (layer.hotkeys) {
|
|
|
|
for (const id in layer.hotkeys) {
|
|
|
|
setDefault(layer.hotkeys[id], "press", undefined, false);
|
|
|
|
setDefault(layer.hotkeys[id], "unlocked", function() {
|
|
|
|
return layers[this.layer].unlocked;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create layer proxy
|
|
|
|
layer = createLayerProxy(layer) as Layer;
|
|
|
|
|
|
|
|
// Register layer
|
|
|
|
layers[layer.id] = layer as Layer;
|
|
|
|
|
|
|
|
// Register hotkeys
|
|
|
|
if (layers[layer.id].hotkeys) {
|
|
|
|
for (const hotkey of layers[layer.id].hotkeys!) {
|
|
|
|
hotkeys.push(hotkey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function removeLayer(layer: string): void {
|
|
|
|
// Un-set hotkeys
|
|
|
|
if (layers[layer].hotkeys) {
|
|
|
|
for (const hotkey of Object.values(layers[layer].hotkeys!)) {
|
|
|
|
const index = hotkeys.indexOf(hotkey);
|
|
|
|
if (index >= 0) {
|
|
|
|
hotkeys.splice(index, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete layers[layer];
|
|
|
|
}
|
|
|
|
|
|
|
|
export function reloadLayer(layer: Layer): void {
|
|
|
|
removeLayer(layer.id);
|
|
|
|
|
|
|
|
// Re-create layer
|
|
|
|
addLayer(layer);
|
|
|
|
}
|
|
|
|
|
|
|
|
function setRowCol<T extends GridFeatures<S>, S extends Feature>(features: RawGridFeatures<T, S>) {
|
|
|
|
if (features.rows && features.cols) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let maxRow = 0;
|
|
|
|
let maxCol = 0;
|
|
|
|
for (const id in features) {
|
|
|
|
const index = Number(id);
|
|
|
|
if (!isNaN(index)) {
|
|
|
|
if (Math.floor(index / 10) > maxRow) {
|
|
|
|
maxRow = Math.floor(index / 10);
|
|
|
|
}
|
|
|
|
if (index % 10 > maxCol) {
|
|
|
|
maxCol = index % 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
features.rows = maxRow;
|
|
|
|
features.cols = maxCol;
|
|
|
|
}
|
|
|
|
|
2021-08-18 00:18:23 -05:00
|
|
|
function setupFeatures<
|
|
|
|
T extends RawFeatures<R, S, unknown>,
|
|
|
|
S extends Feature,
|
|
|
|
R extends Features<S> = Features<S>
|
|
|
|
>(layer: string, features: T) {
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
features.layer = layer;
|
|
|
|
for (const id in features.data) {
|
|
|
|
const feature = features.data[id];
|
2021-08-18 00:18:23 -05:00
|
|
|
(feature as S).id = id;
|
|
|
|
(feature as S).layer = layer;
|
|
|
|
if ((feature as S).unlocked == undefined) {
|
|
|
|
(feature as S).unlocked = true;
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-17 18:39:11 -05:00
|
|
|
function setDefault<T, K extends keyof T>(
|
|
|
|
object: T,
|
|
|
|
key: K,
|
|
|
|
value: T[K],
|
|
|
|
forceCached?: boolean
|
|
|
|
): asserts object is Exclude<T, K> & Required<Pick<T, K>> {
|
2021-08-18 00:18:23 -05:00
|
|
|
if (object[key] === undefined && value != undefined) {
|
First pass at typescript support
Oh man did this end up requiring a *ton* of other work as well.
There's still a few typing issues I still can't quite work out,
and others I'd like to improve when I have time. In fact, this version
doesn't even really work, it has a stack overflow error caused by
a tooltip for some reason have a tree inside it, which in turn has
another tooltip, etc. There's also 17 errors that I *really* feel like
shouldn't be there, but they are, and 113 warnings - mostly using !
to assert that things are non-null. Lots of work left to do, to sum up.
The reason I'm committing this now is because I really need to get to
work on my game jam, and since it won't use a tree or really many of
TMT-X's features, I can get away with using a broken engine :)
2021-08-16 23:30:54 -05:00
|
|
|
object[key] = value;
|
|
|
|
}
|
|
|
|
if (object[key] != undefined && isFunction(object[key]) && forceCached != undefined) {
|
|
|
|
Object.assign(object[key], { forceCached });
|
|
|
|
}
|
|
|
|
}
|