2022-06-26 19:17:22 -05:00
|
|
|
import type { Clickable, ClickableOptions, GenericClickable } from "features/clickables/clickable";
|
|
|
|
import { createClickable } from "features/clickables/clickable";
|
|
|
|
import type { GenericConversion } from "features/conversion";
|
|
|
|
import type { CoercableComponent, JSXFunction, OptionsFunc, Replace } from "features/feature";
|
|
|
|
import { jsx, setDefault } from "features/feature";
|
2022-03-03 21:39:48 -06:00
|
|
|
import { displayResource } from "features/resources/resource";
|
2022-06-26 19:17:22 -05:00
|
|
|
import type { GenericTree, GenericTreeNode, TreeNode, TreeNodeOptions } from "features/trees/tree";
|
|
|
|
import { createTreeNode } from "features/trees/tree";
|
|
|
|
import type { Modifier } from "game/modifiers";
|
|
|
|
import type { Persistent } from "game/persistence";
|
|
|
|
import { DefaultValue, persistent } from "game/persistence";
|
2022-03-03 21:39:48 -06:00
|
|
|
import player from "game/player";
|
2022-06-26 19:17:22 -05:00
|
|
|
import type { DecimalSource } from "util/bignum";
|
|
|
|
import Decimal, { format } from "util/bignum";
|
|
|
|
import type { WithRequired } from "util/common";
|
|
|
|
import type {
|
2022-01-13 22:25:47 -06:00
|
|
|
Computable,
|
2022-03-05 16:42:55 -06:00
|
|
|
GetComputableType,
|
2022-01-13 22:25:47 -06:00
|
|
|
GetComputableTypeWithDefault,
|
|
|
|
ProcessedComputable
|
2022-03-03 21:39:48 -06:00
|
|
|
} from "util/computed";
|
2022-06-26 19:17:22 -05:00
|
|
|
import { convertComputable, processComputable } from "util/computed";
|
2022-04-30 16:42:03 -05:00
|
|
|
import { renderJSX } from "util/vue";
|
2022-06-26 19:17:22 -05:00
|
|
|
import type { Ref } from "vue";
|
|
|
|
import { computed, unref } from "vue";
|
2022-04-30 16:42:03 -05:00
|
|
|
import "./common.css";
|
2022-01-13 22:25:47 -06:00
|
|
|
|
|
|
|
export interface ResetButtonOptions extends ClickableOptions {
|
|
|
|
conversion: GenericConversion;
|
|
|
|
tree: GenericTree;
|
|
|
|
treeNode: GenericTreeNode;
|
|
|
|
resetDescription?: Computable<string>;
|
|
|
|
showNextAt?: Computable<boolean>;
|
|
|
|
display?: Computable<CoercableComponent>;
|
|
|
|
canClick?: Computable<boolean>;
|
2022-04-10 21:16:20 -05:00
|
|
|
minimumGain?: Computable<DecimalSource>;
|
2022-05-23 23:34:59 -05:00
|
|
|
resetTime?: Persistent<DecimalSource>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
|
2022-03-08 19:40:51 -06:00
|
|
|
export type ResetButton<T extends ResetButtonOptions> = Replace<
|
2022-01-13 22:25:47 -06:00
|
|
|
Clickable<T>,
|
|
|
|
{
|
|
|
|
resetDescription: GetComputableTypeWithDefault<T["resetDescription"], Ref<string>>;
|
|
|
|
showNextAt: GetComputableTypeWithDefault<T["showNextAt"], true>;
|
|
|
|
display: GetComputableTypeWithDefault<T["display"], Ref<JSX.Element>>;
|
|
|
|
canClick: GetComputableTypeWithDefault<T["canClick"], Ref<boolean>>;
|
2022-04-10 21:16:20 -05:00
|
|
|
minimumGain: GetComputableTypeWithDefault<T["minimumGain"], 1>;
|
2022-01-13 22:25:47 -06:00
|
|
|
onClick: VoidFunction;
|
|
|
|
}
|
|
|
|
>;
|
|
|
|
|
|
|
|
export type GenericResetButton = Replace<
|
|
|
|
GenericClickable & ResetButton<ResetButtonOptions>,
|
|
|
|
{
|
|
|
|
resetDescription: ProcessedComputable<string>;
|
|
|
|
showNextAt: ProcessedComputable<boolean>;
|
|
|
|
display: ProcessedComputable<CoercableComponent>;
|
|
|
|
canClick: ProcessedComputable<boolean>;
|
2022-04-10 21:16:20 -05:00
|
|
|
minimumGain: ProcessedComputable<DecimalSource>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
>;
|
|
|
|
|
|
|
|
export function createResetButton<T extends ClickableOptions & ResetButtonOptions>(
|
2022-04-10 19:04:56 -05:00
|
|
|
optionsFunc: OptionsFunc<T>
|
2022-01-13 22:25:47 -06:00
|
|
|
): ResetButton<T> {
|
2022-02-27 13:49:34 -06:00
|
|
|
return createClickable(() => {
|
|
|
|
const resetButton = optionsFunc();
|
|
|
|
|
|
|
|
processComputable(resetButton as T, "showNextAt");
|
|
|
|
setDefault(resetButton, "showNextAt", true);
|
2022-04-10 21:16:20 -05:00
|
|
|
setDefault(resetButton, "minimumGain", 1);
|
2022-02-27 13:49:34 -06:00
|
|
|
|
|
|
|
if (resetButton.resetDescription == null) {
|
|
|
|
resetButton.resetDescription = computed(() =>
|
|
|
|
Decimal.lt(resetButton.conversion.gainResource.value, 1e3) ? "Reset for " : ""
|
2022-01-13 22:25:47 -06:00
|
|
|
);
|
2022-02-27 13:49:34 -06:00
|
|
|
} else {
|
|
|
|
processComputable(resetButton as T, "resetDescription");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resetButton.display == null) {
|
|
|
|
resetButton.display = jsx(() => (
|
2022-01-13 22:25:47 -06:00
|
|
|
<span>
|
2022-02-27 13:49:34 -06:00
|
|
|
{unref(resetButton.resetDescription as ProcessedComputable<string>)}
|
2022-01-13 22:25:47 -06:00
|
|
|
<b>
|
|
|
|
{displayResource(
|
2022-02-27 13:49:34 -06:00
|
|
|
resetButton.conversion.gainResource,
|
2022-04-10 21:16:20 -05:00
|
|
|
Decimal.max(
|
|
|
|
unref(resetButton.conversion.actualGain),
|
|
|
|
unref(resetButton.minimumGain as ProcessedComputable<DecimalSource>)
|
|
|
|
)
|
2022-01-13 22:25:47 -06:00
|
|
|
)}
|
2022-02-27 13:49:34 -06:00
|
|
|
</b>{" "}
|
|
|
|
{resetButton.conversion.gainResource.displayName}
|
2022-06-07 19:07:04 -05:00
|
|
|
{unref(resetButton.showNextAt) ? (
|
|
|
|
<div>
|
|
|
|
<br />
|
|
|
|
{unref(resetButton.conversion.buyMax) ? "Next:" : "Req:"}{" "}
|
|
|
|
{displayResource(
|
|
|
|
resetButton.conversion.baseResource,
|
|
|
|
unref(resetButton.conversion.buyMax) ||
|
|
|
|
Decimal.floor(unref(resetButton.conversion.actualGain)).neq(1)
|
|
|
|
? unref(resetButton.conversion.nextAt)
|
|
|
|
: unref(resetButton.conversion.currentAt)
|
|
|
|
)}{" "}
|
|
|
|
{resetButton.conversion.baseResource.displayName}
|
|
|
|
</div>
|
|
|
|
) : null}
|
2022-01-13 22:25:47 -06:00
|
|
|
</span>
|
2022-02-27 13:49:34 -06:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resetButton.canClick == null) {
|
|
|
|
resetButton.canClick = computed(() =>
|
2022-04-10 21:16:20 -05:00
|
|
|
Decimal.gte(
|
|
|
|
unref(resetButton.conversion.actualGain),
|
|
|
|
unref(resetButton.minimumGain as ProcessedComputable<DecimalSource>)
|
|
|
|
)
|
2022-01-13 22:25:47 -06:00
|
|
|
);
|
2022-02-27 13:49:34 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
const onClick = resetButton.onClick;
|
|
|
|
resetButton.onClick = function () {
|
2022-03-02 21:09:37 -06:00
|
|
|
if (!unref(resetButton.canClick)) {
|
|
|
|
return;
|
|
|
|
}
|
2022-02-27 13:49:34 -06:00
|
|
|
resetButton.conversion.convert();
|
|
|
|
resetButton.tree.reset(resetButton.treeNode);
|
2022-05-23 23:34:59 -05:00
|
|
|
if (resetButton.resetTime) {
|
|
|
|
resetButton.resetTime.value = resetButton.resetTime[DefaultValue];
|
|
|
|
}
|
2022-02-27 13:49:34 -06:00
|
|
|
onClick?.();
|
|
|
|
};
|
2022-01-13 22:25:47 -06:00
|
|
|
|
2022-02-27 13:49:34 -06:00
|
|
|
return resetButton;
|
|
|
|
}) as unknown as ResetButton<T>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface LayerTreeNodeOptions extends TreeNodeOptions {
|
|
|
|
layerID: string;
|
2022-03-05 16:42:55 -06:00
|
|
|
color: Computable<string>; // marking as required
|
2022-04-23 23:54:23 -05:00
|
|
|
display?: Computable<CoercableComponent>;
|
2022-03-05 16:42:55 -06:00
|
|
|
append?: Computable<boolean>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
export type LayerTreeNode<T extends LayerTreeNodeOptions> = Replace<
|
|
|
|
TreeNode<T>,
|
|
|
|
{
|
2022-03-05 16:42:55 -06:00
|
|
|
display: GetComputableTypeWithDefault<T["display"], T["layerID"]>;
|
|
|
|
append: GetComputableType<T["append"]>;
|
|
|
|
}
|
|
|
|
>;
|
|
|
|
export type GenericLayerTreeNode = Replace<
|
|
|
|
LayerTreeNode<LayerTreeNodeOptions>,
|
|
|
|
{
|
2022-04-23 23:54:23 -05:00
|
|
|
display: ProcessedComputable<CoercableComponent>;
|
2022-03-05 16:42:55 -06:00
|
|
|
append?: ProcessedComputable<boolean>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
>;
|
|
|
|
|
2022-02-27 13:49:34 -06:00
|
|
|
export function createLayerTreeNode<T extends LayerTreeNodeOptions>(
|
2022-04-10 19:04:56 -05:00
|
|
|
optionsFunc: OptionsFunc<T>
|
2022-02-27 13:49:34 -06:00
|
|
|
): LayerTreeNode<T> {
|
|
|
|
return createTreeNode(() => {
|
|
|
|
const options = optionsFunc();
|
2022-03-05 16:42:55 -06:00
|
|
|
processComputable(options as T, "display");
|
|
|
|
setDefault(options, "display", options.layerID);
|
2022-02-27 13:49:34 -06:00
|
|
|
processComputable(options as T, "append");
|
|
|
|
return {
|
|
|
|
...options,
|
2022-04-23 23:54:23 -05:00
|
|
|
display: options.display,
|
2022-03-05 16:42:55 -06:00
|
|
|
onClick: unref((options as unknown as GenericLayerTreeNode).append)
|
|
|
|
? function () {
|
|
|
|
if (player.tabs.includes(options.layerID)) {
|
|
|
|
const index = player.tabs.lastIndexOf(options.layerID);
|
|
|
|
player.tabs.splice(index, 1);
|
|
|
|
} else {
|
|
|
|
player.tabs.push(options.layerID);
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
2022-03-05 16:42:55 -06:00
|
|
|
}
|
|
|
|
: function () {
|
|
|
|
player.tabs.splice(1, 1, options.layerID);
|
|
|
|
}
|
2022-02-27 13:49:34 -06:00
|
|
|
};
|
2022-01-24 22:23:30 -06:00
|
|
|
}) as unknown as LayerTreeNode<T>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
2022-04-30 16:42:03 -05:00
|
|
|
|
|
|
|
export function createCollapsibleModifierSections(
|
|
|
|
sections: {
|
|
|
|
title: string;
|
|
|
|
subtitle?: string;
|
2022-05-01 14:46:23 -05:00
|
|
|
modifier: WithRequired<Modifier, "description">;
|
2022-04-30 16:42:03 -05:00
|
|
|
base?: Computable<DecimalSource>;
|
|
|
|
unit?: string;
|
|
|
|
baseText?: Computable<CoercableComponent>;
|
|
|
|
visible?: Computable<boolean>;
|
|
|
|
}[]
|
|
|
|
): [JSXFunction, Persistent<boolean>[]] {
|
|
|
|
const processedBase = sections.map(s => convertComputable(s.base));
|
|
|
|
const processedBaseText = sections.map(s => convertComputable(s.baseText));
|
|
|
|
const processedVisible = sections.map(s => convertComputable(s.visible));
|
|
|
|
const collapsed = sections.map(() => persistent<boolean>(false));
|
|
|
|
const jsxFunc = jsx(() => {
|
|
|
|
const sectionJSX = sections.map((s, i) => {
|
|
|
|
if (unref(processedVisible[i]) === false) return null;
|
|
|
|
const header = (
|
|
|
|
<h3
|
|
|
|
onClick={() => (collapsed[i].value = !collapsed[i].value)}
|
|
|
|
style="cursor: pointer"
|
|
|
|
>
|
|
|
|
<span class={"modifier-toggle" + (unref(collapsed[i]) ? " collapsed" : "")}>
|
|
|
|
▼
|
|
|
|
</span>
|
|
|
|
{s.title}
|
|
|
|
{s.subtitle ? <span class="subtitle"> ({s.subtitle})</span> : null}
|
|
|
|
</h3>
|
|
|
|
);
|
|
|
|
|
|
|
|
const modifiers = unref(collapsed[i]) ? null : (
|
|
|
|
<>
|
|
|
|
<div class="modifier-container">
|
|
|
|
<span class="modifier-amount">
|
|
|
|
{format(unref(processedBase[i]) ?? 1)}
|
|
|
|
{s.unit}
|
|
|
|
</span>
|
|
|
|
<span class="modifier-description">
|
|
|
|
{renderJSX(unref(processedBaseText[i]) ?? "Base")}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
{renderJSX(unref(s.modifier.description))}
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
{i === 0 ? null : <br />}
|
|
|
|
<div>
|
|
|
|
{header}
|
|
|
|
<br />
|
|
|
|
{modifiers}
|
|
|
|
<hr />
|
|
|
|
Total: {format(s.modifier.apply(unref(processedBase[i]) ?? 1))}
|
|
|
|
{s.unit}
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
return <>{sectionJSX}</>;
|
|
|
|
});
|
|
|
|
return [jsxFunc, collapsed];
|
|
|
|
}
|