2022-12-04 20:26:21 -06:00
|
|
|
import Collapsible from "components/layout/Collapsible.vue";
|
2023-04-03 00:34:45 -05:00
|
|
|
import { GenericAchievement } from "features/achievements/achievement";
|
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";
|
2023-04-18 20:58:35 -05:00
|
|
|
import { Resource, 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";
|
2023-04-18 20:58:35 -05:00
|
|
|
import type { GenericFormula } from "game/formulas/types";
|
2023-10-11 21:39:01 -05:00
|
|
|
import { BaseLayer } from "game/layers";
|
2022-06-26 19:17:22 -05:00
|
|
|
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";
|
2023-02-15 21:58:06 -06:00
|
|
|
import settings from "game/settings";
|
2022-06-26 19:17:22 -05:00
|
|
|
import type { DecimalSource } from "util/bignum";
|
2023-02-15 15:24:36 -06:00
|
|
|
import Decimal, { format, formatSmall, formatTime } from "util/bignum";
|
2023-04-23 16:55:45 -05:00
|
|
|
import { WithRequired, camelToTitle } from "util/common";
|
2022-06-26 19:17:22 -05:00
|
|
|
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-12-04 20:26:21 -06:00
|
|
|
import { getFirstFeature, renderColJSX, renderJSX } from "util/vue";
|
2023-02-15 18:21:38 -06:00
|
|
|
import type { ComputedRef, Ref } from "vue";
|
2024-03-08 17:06:54 -06:00
|
|
|
import { computed, ref, unref } from "vue";
|
2024-03-17 23:28:57 -05:00
|
|
|
import { JSX } from "vue/jsx-runtime";
|
2022-04-30 16:42:03 -05:00
|
|
|
import "./common.css";
|
2022-01-13 22:25:47 -06:00
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/** An object that configures a {@link ResetButton} */
|
2022-01-13 22:25:47 -06:00
|
|
|
export interface ResetButtonOptions extends ClickableOptions {
|
2022-07-10 01:44:45 -05:00
|
|
|
/** The conversion the button uses to calculate how much resources will be gained on click */
|
2022-01-13 22:25:47 -06:00
|
|
|
conversion: GenericConversion;
|
2022-07-10 01:44:45 -05:00
|
|
|
/** The tree this reset button is apart of */
|
2022-01-13 22:25:47 -06:00
|
|
|
tree: GenericTree;
|
2022-07-10 01:44:45 -05:00
|
|
|
/** The specific tree node associated with this reset button */
|
2022-01-13 22:25:47 -06:00
|
|
|
treeNode: GenericTreeNode;
|
2022-07-26 15:25:31 -05:00
|
|
|
/**
|
2022-07-10 01:44:45 -05:00
|
|
|
* Text to display on low conversion amounts, describing what "resetting" is in this context.
|
|
|
|
* Defaults to "Reset for ".
|
|
|
|
*/
|
2022-01-13 22:25:47 -06:00
|
|
|
resetDescription?: Computable<string>;
|
2022-07-10 01:44:45 -05:00
|
|
|
/** Whether or not to show how much currency would be required to make the gain amount increase. */
|
2022-01-13 22:25:47 -06:00
|
|
|
showNextAt?: Computable<boolean>;
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* The content to display on the button.
|
|
|
|
* By default, this includes the reset description, and amount of currency to be gained.
|
|
|
|
*/
|
2022-01-13 22:25:47 -06:00
|
|
|
display?: Computable<CoercableComponent>;
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* Whether or not this button can currently be clicked.
|
|
|
|
* Defaults to checking the current gain amount is greater than {@link minimumGain}
|
|
|
|
*/
|
2022-01-13 22:25:47 -06:00
|
|
|
canClick?: Computable<boolean>;
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* When {@link canClick} is left to its default, minimumGain is used to only enable the reset button when a sufficient amount of currency to gain is available.
|
|
|
|
*/
|
2022-04-10 21:16:20 -05:00
|
|
|
minimumGain?: Computable<DecimalSource>;
|
2022-07-10 01:44:45 -05:00
|
|
|
/** A persistent ref to track how much time has passed since the last time this tree node was reset. */
|
2022-05-23 23:34:59 -05:00
|
|
|
resetTime?: Persistent<DecimalSource>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* A button that is used to control a conversion.
|
|
|
|
* It will show how much can be converted currently, and can show when that amount will go up, as well as handle only being clickable when a sufficient amount of currency can be gained.
|
|
|
|
* Assumes this button is associated with a specific node on a tree, and triggers that tree's reset propagation.
|
|
|
|
*/
|
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>;
|
2023-02-15 14:57:22 -06:00
|
|
|
onClick: (event?: MouseEvent | TouchEvent) => void;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
|
|
|
>;
|
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/** A type that matches any valid {@link ResetButton} object. */
|
2022-01-13 22:25:47 -06:00
|
|
|
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
|
|
|
}
|
|
|
|
>;
|
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* Lazily creates a reset button with the given options.
|
|
|
|
* @param optionsFunc A function that returns the options object for this reset button.
|
|
|
|
*/
|
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> {
|
2023-04-18 20:58:35 -05:00
|
|
|
return createClickable(feature => {
|
|
|
|
const resetButton = optionsFunc.call(feature, feature);
|
2022-02-27 13:49:34 -06:00
|
|
|
|
|
|
|
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}
|
2024-03-17 23:28:57 -05:00
|
|
|
{unref(resetButton.showNextAt as ProcessedComputable<boolean>) != null ? (
|
2022-06-07 19:07:04 -05:00
|
|
|
<div>
|
|
|
|
<br />
|
|
|
|
{unref(resetButton.conversion.buyMax) ? "Next:" : "Req:"}{" "}
|
|
|
|
{displayResource(
|
|
|
|
resetButton.conversion.baseResource,
|
2023-11-13 14:09:48 -08:00
|
|
|
!unref(resetButton.conversion.buyMax) &&
|
|
|
|
Decimal.gte(unref(resetButton.conversion.actualGain), 1)
|
2023-05-07 21:51:11 -05:00
|
|
|
? unref(resetButton.conversion.currentAt)
|
|
|
|
: unref(resetButton.conversion.nextAt)
|
2022-06-07 19:07:04 -05:00
|
|
|
)}{" "}
|
|
|
|
{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;
|
2023-02-15 14:57:22 -06:00
|
|
|
resetButton.onClick = function (event?: MouseEvent | TouchEvent) {
|
2022-12-20 21:26:25 -06:00
|
|
|
if (unref(resetButton.canClick) === false) {
|
2022-03-02 21:09:37 -06:00
|
|
|
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];
|
|
|
|
}
|
2023-02-15 14:57:22 -06:00
|
|
|
onClick?.(event);
|
2022-02-27 13:49:34 -06:00
|
|
|
};
|
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
|
|
|
}
|
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/** An object that configures a {@link LayerTreeNode} */
|
2022-01-13 22:25:47 -06:00
|
|
|
export interface LayerTreeNodeOptions extends TreeNodeOptions {
|
2022-07-10 01:44:45 -05:00
|
|
|
/** The ID of the layer this tree node is associated with */
|
2022-01-13 22:25:47 -06:00
|
|
|
layerID: string;
|
2022-07-10 01:44:45 -05:00
|
|
|
/** The color to display this tree node as */
|
2022-03-05 16:42:55 -06:00
|
|
|
color: Computable<string>; // marking as required
|
2022-07-10 01:44:45 -05:00
|
|
|
/** Whether or not to append the layer to the tabs list.
|
|
|
|
* If set to false, then the tree node will instead always remove all tabs to its right and then add the layer tab.
|
|
|
|
* Defaults to true.
|
|
|
|
*/
|
2022-03-05 16:42:55 -06:00
|
|
|
append?: Computable<boolean>;
|
2022-01-13 22:25:47 -06:00
|
|
|
}
|
2022-07-10 01:44:45 -05:00
|
|
|
/** A tree node that is associated with a given layer, and which opens the layer when clicked. */
|
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"]>;
|
|
|
|
}
|
|
|
|
>;
|
2022-07-10 01:44:45 -05:00
|
|
|
/** A type that matches any valid {@link LayerTreeNode} object. */
|
2022-03-05 16:42:55 -06:00
|
|
|
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-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* Lazily creates a tree node that's associated with a specific layer, with the given options.
|
|
|
|
* @param optionsFunc A function that returns the options object for this tree node.
|
|
|
|
*/
|
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> {
|
2023-04-18 20:58:35 -05:00
|
|
|
return createTreeNode(feature => {
|
|
|
|
const options = optionsFunc.call(feature, feature);
|
2023-04-23 16:55:45 -05:00
|
|
|
setDefault(options, "display", camelToTitle(options.layerID));
|
2022-02-27 13:49:34 -06:00
|
|
|
processComputable(options as T, "append");
|
|
|
|
return {
|
|
|
|
...options,
|
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
|
|
|
|
2022-07-15 17:27:51 -05:00
|
|
|
/** An option object for a modifier display as a single section. **/
|
|
|
|
export interface Section {
|
|
|
|
/** The header for this modifier. **/
|
2022-12-23 19:55:41 -06:00
|
|
|
title: Computable<string>;
|
2022-07-15 17:27:51 -05:00
|
|
|
/** A subtitle for this modifier, e.g. to explain the context for the modifier. **/
|
2022-12-23 19:55:41 -06:00
|
|
|
subtitle?: Computable<string>;
|
2022-07-15 17:27:51 -05:00
|
|
|
/** The modifier to be displaying in this section. **/
|
|
|
|
modifier: WithRequired<Modifier, "description">;
|
|
|
|
/** The base value being modified. **/
|
|
|
|
base?: Computable<DecimalSource>;
|
|
|
|
/** The unit of measurement for the base. **/
|
|
|
|
unit?: string;
|
|
|
|
/** The label to call the base amount. Defaults to "Base". **/
|
|
|
|
baseText?: Computable<CoercableComponent>;
|
|
|
|
/** Whether or not this section should be currently visible to the player. **/
|
|
|
|
visible?: Computable<boolean>;
|
2023-04-16 12:43:04 -05:00
|
|
|
/** Determines if numbers larger or smaller than the base should be displayed as red. */
|
|
|
|
smallerIsBetter?: boolean;
|
2022-07-15 17:27:51 -05:00
|
|
|
}
|
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* Takes an array of modifier "sections", and creates a JSXFunction that can render all those sections, and allow each section to be collapsed.
|
|
|
|
* Also returns a list of persistent refs that are used to control which sections are currently collapsed.
|
2022-07-26 15:25:31 -05:00
|
|
|
* @param sectionsFunc A function that returns the sections to display.
|
2022-07-10 01:44:45 -05:00
|
|
|
*/
|
2022-04-30 16:42:03 -05:00
|
|
|
export function createCollapsibleModifierSections(
|
2023-04-16 12:43:04 -05:00
|
|
|
sectionsFunc: () => Section[]
|
2022-07-27 22:56:34 -05:00
|
|
|
): [JSXFunction, Persistent<Record<number, boolean>>] {
|
2022-07-26 23:56:20 -05:00
|
|
|
const sections: Section[] = [];
|
|
|
|
const processed:
|
|
|
|
| {
|
|
|
|
base: ProcessedComputable<DecimalSource | undefined>[];
|
|
|
|
baseText: ProcessedComputable<CoercableComponent | undefined>[];
|
|
|
|
visible: ProcessedComputable<boolean | undefined>[];
|
2022-12-23 19:55:41 -06:00
|
|
|
title: ProcessedComputable<string | undefined>[];
|
|
|
|
subtitle: ProcessedComputable<string | undefined>[];
|
2022-07-26 23:56:20 -05:00
|
|
|
}
|
|
|
|
| Record<string, never> = {};
|
|
|
|
let calculated = false;
|
|
|
|
function calculateSections() {
|
|
|
|
if (!calculated) {
|
|
|
|
sections.push(...sectionsFunc());
|
|
|
|
processed.base = sections.map(s => convertComputable(s.base));
|
|
|
|
processed.baseText = sections.map(s => convertComputable(s.baseText));
|
|
|
|
processed.visible = sections.map(s => convertComputable(s.visible));
|
2022-12-23 19:55:41 -06:00
|
|
|
processed.title = sections.map(s => convertComputable(s.title));
|
|
|
|
processed.subtitle = sections.map(s => convertComputable(s.subtitle));
|
2022-07-26 23:56:20 -05:00
|
|
|
calculated = true;
|
|
|
|
}
|
|
|
|
return sections;
|
|
|
|
}
|
2022-04-30 16:42:03 -05:00
|
|
|
|
2023-02-22 20:48:04 -06:00
|
|
|
const collapsed = persistent<Record<number, boolean>>({}, false);
|
2022-07-26 23:56:20 -05:00
|
|
|
const jsxFunc = jsx(() => {
|
|
|
|
const sections = calculateSections();
|
2022-04-30 16:42:03 -05:00
|
|
|
|
2022-08-22 08:43:15 -05:00
|
|
|
let firstVisibleSection = true;
|
2022-07-26 23:56:20 -05:00
|
|
|
const sectionJSX = sections.map((s, i) => {
|
|
|
|
if (unref(processed.visible[i]) === false) return null;
|
|
|
|
const header = (
|
|
|
|
<h3
|
2022-07-27 22:56:34 -05:00
|
|
|
onClick={() => (collapsed.value[i] = !collapsed.value[i])}
|
2022-07-26 23:56:20 -05:00
|
|
|
style="cursor: pointer"
|
|
|
|
>
|
2022-07-27 22:56:34 -05:00
|
|
|
<span
|
|
|
|
class={"modifier-toggle" + (unref(collapsed.value[i]) ? " collapsed" : "")}
|
|
|
|
>
|
2022-07-26 23:56:20 -05:00
|
|
|
▼
|
|
|
|
</span>
|
2022-12-23 20:14:12 -06:00
|
|
|
{unref(processed.title[i])}
|
|
|
|
{unref(processed.subtitle[i]) != null ? (
|
|
|
|
<span class="subtitle"> ({unref(processed.subtitle[i])})</span>
|
2022-12-23 19:55:41 -06:00
|
|
|
) : null}
|
2022-07-26 23:56:20 -05:00
|
|
|
</h3>
|
|
|
|
);
|
2022-07-26 15:25:31 -05:00
|
|
|
|
2022-07-27 22:56:34 -05:00
|
|
|
const modifiers = unref(collapsed.value[i]) ? null : (
|
2022-07-26 23:56:20 -05:00
|
|
|
<>
|
|
|
|
<div class="modifier-container">
|
2022-12-10 02:24:14 -06:00
|
|
|
<span class="modifier-description">
|
|
|
|
{renderJSX(unref(processed.baseText[i]) ?? "Base")}
|
|
|
|
</span>
|
2022-07-26 23:56:20 -05:00
|
|
|
<span class="modifier-amount">
|
|
|
|
{format(unref(processed.base[i]) ?? 1)}
|
2022-07-26 15:25:31 -05:00
|
|
|
{s.unit}
|
2022-07-26 23:56:20 -05:00
|
|
|
</span>
|
|
|
|
</div>
|
2023-04-16 12:43:04 -05:00
|
|
|
{s.modifier.description == null
|
|
|
|
? null
|
|
|
|
: renderJSX(unref(s.modifier.description))}
|
2022-07-26 23:56:20 -05:00
|
|
|
</>
|
|
|
|
);
|
|
|
|
|
2022-08-22 08:43:15 -05:00
|
|
|
const hasPreviousSection = !firstVisibleSection;
|
|
|
|
firstVisibleSection = false;
|
|
|
|
|
2023-02-15 00:09:06 -06:00
|
|
|
const base = unref(processed.base[i]) ?? 1;
|
|
|
|
const total = s.modifier.apply(base);
|
|
|
|
|
2022-07-26 23:56:20 -05:00
|
|
|
return (
|
|
|
|
<>
|
2022-08-22 08:43:15 -05:00
|
|
|
{hasPreviousSection ? <br /> : null}
|
2023-02-15 21:58:06 -06:00
|
|
|
<div
|
|
|
|
style={{
|
|
|
|
"--unit":
|
|
|
|
settings.alignUnits && s.unit != null ? "'" + s.unit + "'" : ""
|
|
|
|
}}
|
|
|
|
>
|
2022-07-26 23:56:20 -05:00
|
|
|
{header}
|
|
|
|
<br />
|
|
|
|
{modifiers}
|
|
|
|
<hr />
|
2022-12-10 02:24:14 -06:00
|
|
|
<div class="modifier-container">
|
2023-02-15 00:09:06 -06:00
|
|
|
<span class="modifier-description">Total</span>
|
|
|
|
<span
|
|
|
|
class="modifier-amount"
|
|
|
|
style={
|
|
|
|
(
|
2023-04-16 12:43:04 -05:00
|
|
|
s.smallerIsBetter === true
|
2023-02-15 00:09:06 -06:00
|
|
|
? Decimal.gt(total, base ?? 1)
|
|
|
|
: Decimal.lt(total, base ?? 1)
|
|
|
|
)
|
|
|
|
? "color: var(--danger)"
|
|
|
|
: ""
|
|
|
|
}
|
|
|
|
>
|
|
|
|
{formatSmall(total)}
|
2022-12-10 02:24:14 -06:00
|
|
|
{s.unit}
|
|
|
|
</span>
|
|
|
|
</div>
|
2022-07-26 23:56:20 -05:00
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
);
|
2022-04-30 16:42:03 -05:00
|
|
|
});
|
2022-07-26 23:56:20 -05:00
|
|
|
return <>{sectionJSX}</>;
|
2022-04-30 16:42:03 -05:00
|
|
|
});
|
2022-07-26 23:56:20 -05:00
|
|
|
return [jsxFunc, collapsed];
|
2022-04-30 16:42:03 -05:00
|
|
|
}
|
2022-06-28 09:52:49 -05:00
|
|
|
|
2022-07-10 01:44:45 -05:00
|
|
|
/**
|
|
|
|
* Creates an HTML string for a span that writes some given text in a given color.
|
|
|
|
* @param textToColor The content to change the color of
|
|
|
|
* @param color The color to change the content to look like. Defaults to the current theme's accent 2 variable.
|
|
|
|
*/
|
2022-07-17 21:32:32 -05:00
|
|
|
export function colorText(textToColor: string, color = "var(--accent2)"): JSX.Element {
|
2022-07-25 21:30:19 -05:00
|
|
|
return <span style={{ color }}>{textToColor}</span>;
|
2022-06-28 09:52:49 -05:00
|
|
|
}
|
2022-12-04 20:26:21 -06:00
|
|
|
|
|
|
|
/**
|
2023-04-03 00:34:45 -05:00
|
|
|
* Creates a collapsible display of a list of achievements
|
|
|
|
* @param achievements A dictionary of the achievements to display, inserted in the order from easiest to hardest
|
2022-12-04 20:26:21 -06:00
|
|
|
*/
|
2023-04-03 00:34:45 -05:00
|
|
|
export function createCollapsibleAchievements(achievements: Record<string, GenericAchievement>) {
|
|
|
|
// Achievements are typically defined from easiest to hardest, and we want to show hardest first
|
|
|
|
const orderedAchievements = Object.values(achievements).reverse();
|
|
|
|
const collapseAchievements = persistent<boolean>(true, false);
|
|
|
|
const lockedAchievements = computed(() =>
|
|
|
|
orderedAchievements.filter(m => m.earned.value === false)
|
2022-12-04 20:26:21 -06:00
|
|
|
);
|
|
|
|
const { firstFeature, collapsedContent, hasCollapsedContent } = getFirstFeature(
|
2023-04-03 00:34:45 -05:00
|
|
|
orderedAchievements,
|
2022-12-04 20:26:21 -06:00
|
|
|
m => m.earned.value
|
|
|
|
);
|
|
|
|
const display = jsx(() => {
|
2023-04-03 00:34:45 -05:00
|
|
|
const achievementsToDisplay = [...lockedAchievements.value];
|
2022-12-04 20:26:21 -06:00
|
|
|
if (firstFeature.value) {
|
2023-04-03 00:34:45 -05:00
|
|
|
achievementsToDisplay.push(firstFeature.value);
|
2022-12-04 20:26:21 -06:00
|
|
|
}
|
|
|
|
return renderColJSX(
|
2023-04-03 00:34:45 -05:00
|
|
|
...achievementsToDisplay,
|
2022-12-04 20:26:21 -06:00
|
|
|
jsx(() => (
|
|
|
|
<Collapsible
|
2023-04-03 00:34:45 -05:00
|
|
|
collapsed={collapseAchievements}
|
2022-12-04 20:26:21 -06:00
|
|
|
content={collapsedContent}
|
|
|
|
display={
|
2023-04-03 00:34:45 -05:00
|
|
|
collapseAchievements.value
|
|
|
|
? "Show other completed achievements"
|
|
|
|
: "Hide other completed achievements"
|
2022-12-04 20:26:21 -06:00
|
|
|
}
|
|
|
|
v-show={unref(hasCollapsedContent)}
|
|
|
|
/>
|
|
|
|
))
|
|
|
|
);
|
|
|
|
});
|
|
|
|
return {
|
2023-04-03 00:34:45 -05:00
|
|
|
collapseAchievements: collapseAchievements,
|
2022-12-04 20:26:21 -06:00
|
|
|
display
|
|
|
|
};
|
|
|
|
}
|
2023-02-15 15:24:36 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility function for getting an ETA for when a target will be reached by a resource with a known (and assumed consistent) gain.
|
|
|
|
* @param resource The resource that will be increasing over time.
|
|
|
|
* @param rate The rate at which the resource is increasing.
|
|
|
|
* @param target The target amount of the resource to estimate the duration until.
|
|
|
|
*/
|
|
|
|
export function estimateTime(
|
|
|
|
resource: Resource,
|
|
|
|
rate: Computable<DecimalSource>,
|
|
|
|
target: Computable<DecimalSource>
|
|
|
|
) {
|
|
|
|
const processedRate = convertComputable(rate);
|
|
|
|
const processedTarget = convertComputable(target);
|
|
|
|
return computed(() => {
|
|
|
|
const currRate = unref(processedRate);
|
|
|
|
const currTarget = unref(processedTarget);
|
|
|
|
if (Decimal.gte(resource.value, currTarget)) {
|
|
|
|
return "Now";
|
2023-04-27 22:48:40 -05:00
|
|
|
} else if (Decimal.lte(currRate, 0)) {
|
2023-02-15 15:24:36 -06:00
|
|
|
return "Never";
|
|
|
|
}
|
|
|
|
return formatTime(Decimal.sub(currTarget, resource.value).div(currRate));
|
|
|
|
});
|
|
|
|
}
|
2023-02-15 18:21:38 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility function for displaying the result of a formula such that it will, when told to, preview how the formula's result will change.
|
|
|
|
* Requires a formula with a single variable inside.
|
|
|
|
* @param formula The formula to display the result of.
|
|
|
|
* @param showPreview Whether or not to preview how the formula's result will change.
|
|
|
|
* @param previewAmount The amount to _add_ to the current formula's variable amount to preview the change in result.
|
|
|
|
*/
|
|
|
|
export function createFormulaPreview(
|
|
|
|
formula: GenericFormula,
|
|
|
|
showPreview: Computable<boolean>,
|
|
|
|
previewAmount: Computable<DecimalSource> = 1
|
2023-04-22 19:54:12 -05:00
|
|
|
) {
|
2023-02-15 18:21:38 -06:00
|
|
|
const processedShowPreview = convertComputable(showPreview);
|
|
|
|
const processedPreviewAmount = convertComputable(previewAmount);
|
|
|
|
if (!formula.hasVariable()) {
|
2023-05-16 23:38:31 -05:00
|
|
|
console.error("Cannot create formula preview if the formula does not have a variable");
|
2023-02-15 18:21:38 -06:00
|
|
|
}
|
2023-04-22 19:54:12 -05:00
|
|
|
return jsx(() => {
|
2023-02-15 18:21:38 -06:00
|
|
|
if (unref(processedShowPreview)) {
|
|
|
|
const curr = formatSmall(formula.evaluate());
|
|
|
|
const preview = formatSmall(
|
|
|
|
formula.evaluate(
|
|
|
|
Decimal.add(
|
|
|
|
unref(formula.innermostVariable ?? 0),
|
|
|
|
unref(processedPreviewAmount)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
2023-04-22 19:54:12 -05:00
|
|
|
return (
|
2023-02-15 18:21:38 -06:00
|
|
|
<>
|
|
|
|
<b>
|
|
|
|
<i>
|
2023-04-22 19:54:12 -05:00
|
|
|
{curr} → {preview}
|
2023-02-15 18:21:38 -06:00
|
|
|
</i>
|
|
|
|
</b>
|
|
|
|
</>
|
2023-04-22 19:54:12 -05:00
|
|
|
);
|
2023-02-15 18:21:38 -06:00
|
|
|
}
|
2023-04-22 19:54:12 -05:00
|
|
|
return <>{formatSmall(formula.evaluate())}</>;
|
2023-02-15 18:21:38 -06:00
|
|
|
});
|
|
|
|
}
|
2023-10-11 21:39:01 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility function for getting a computed boolean for whether or not a given feature is currently rendered in the DOM.
|
|
|
|
* Note it will have a true value even if the feature is off screen.
|
|
|
|
* @param layer The layer the feature appears within
|
|
|
|
* @param id The ID of the feature
|
|
|
|
*/
|
|
|
|
export function isRendered(layer: BaseLayer, id: string): ComputedRef<boolean>;
|
2023-10-11 21:44:02 -05:00
|
|
|
/**
|
|
|
|
* Utility function for getting a computed boolean for whether or not a given feature is currently rendered in the DOM.
|
|
|
|
* Note it will have a true value even if the feature is off screen.
|
|
|
|
* @param layer The layer the feature appears within
|
|
|
|
* @param feature The feature that may be rendered
|
|
|
|
*/
|
2023-10-11 21:39:01 -05:00
|
|
|
export function isRendered(layer: BaseLayer, feature: { id: string }): ComputedRef<boolean>;
|
|
|
|
export function isRendered(layer: BaseLayer, idOrFeature: string | { id: string }) {
|
|
|
|
const id = typeof idOrFeature === "string" ? idOrFeature : idOrFeature.id;
|
|
|
|
return computed(() => id in layer.nodes.value);
|
|
|
|
}
|
2024-03-08 17:06:54 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Utility function for setting up a system where one of many things can be selected.
|
|
|
|
* It's recommended to use an ID or index rather than the object itself, so that you can wrap the ref in a persistent without breaking anything.
|
|
|
|
* @returns The ref containing the selection, as well as a select and deselect function
|
|
|
|
*/
|
|
|
|
export function setupSelectable<T>() {
|
|
|
|
const selected = ref<T>();
|
|
|
|
return {
|
|
|
|
select: function (node: T) {
|
|
|
|
selected.value = node;
|
|
|
|
},
|
|
|
|
deselect: function () {
|
|
|
|
selected.value = undefined;
|
|
|
|
},
|
|
|
|
selected
|
|
|
|
};
|
|
|
|
}
|