Profectus-Demo/src/game/requirements.tsx

324 lines
13 KiB
TypeScript
Raw Normal View History

2022-12-31 20:57:09 +00:00
import { isArray } from "@vue/shared";
import { CoercableComponent, isVisible, jsx, setDefault, Visibility } from "features/feature";
2022-12-31 20:57:09 +00:00
import { displayResource, Resource } from "features/resources/resource";
import Decimal, { DecimalSource } from "lib/break_eternity";
import {
Computable,
convertComputable,
processComputable,
ProcessedComputable
} from "util/computed";
import { createLazyProxy } from "util/proxies";
import { joinJSX, renderJSX } from "util/vue";
import { computed, unref } from "vue";
import Formula, { calculateCost, calculateMaxAffordable } from "./formulas/formulas";
import type { GenericFormula, InvertibleFormula } from "./formulas/types";
import { DefaultValue, Persistent } from "./persistence";
2022-12-31 20:57:09 +00:00
/**
* An object that can be used to describe a requirement to perform some purchase or other action.
* @see {@link createCostRequirement}
*/
export interface Requirement {
2023-02-05 10:23:53 +00:00
/**
* The display for this specific requirement. This is used for displays multiple requirements condensed. Required if {@link visibility} can be {@link Visibility.Visible}.
*/
2023-02-05 08:44:03 +00:00
partialDisplay?: (amount?: DecimalSource) => JSX.Element;
2023-02-05 10:23:53 +00:00
/**
* The display for this specific requirement. Required if {@link visibility} can be {@link Visibility.Visible}.
*/
2023-02-05 08:44:03 +00:00
display?: (amount?: DecimalSource) => JSX.Element;
2023-02-05 10:23:53 +00:00
/**
* Whether or not this requirement should be displayed in Vue Features. {@link displayRequirements} will respect this property.
*/
visibility: ProcessedComputable<Visibility.Visible | Visibility.None | boolean>;
2023-02-05 10:23:53 +00:00
/**
* Whether or not this requirement has been met.
*/
2023-02-05 08:44:03 +00:00
requirementMet: ProcessedComputable<DecimalSource | boolean>;
2023-02-05 10:23:53 +00:00
/**
* Whether or not this requirement will need to affect the game state when whatever is using this requirement gets triggered.
*/
2022-12-31 20:57:09 +00:00
requiresPay: ProcessedComputable<boolean>;
2023-02-05 10:23:53 +00:00
/**
* Whether or not this requirement can have multiple levels of requirements that can be met at once. Requirement is assumed to not have multiple levels if this property not present.
2023-02-05 10:23:53 +00:00
*/
canMaximize?: ProcessedComputable<boolean>;
2023-02-05 10:23:53 +00:00
/**
* Perform any effects to the game state that should happen when the requirement gets triggered.
* @param amount The amount of levels of requirements to pay for.
*/
2023-02-05 08:44:03 +00:00
pay?: (amount?: DecimalSource) => void;
2022-12-31 20:57:09 +00:00
}
2023-02-05 10:23:53 +00:00
/**
* Utility type for accepting 1 or more {@link Requirement}s.
*/
2022-12-31 20:57:09 +00:00
export type Requirements = Requirement | Requirement[];
2023-02-05 10:23:53 +00:00
/** An object that configures a {@link Requirement} based on a resource cost. */
2022-12-31 20:57:09 +00:00
export interface CostRequirementOptions {
2023-02-05 10:23:53 +00:00
/**
* The resource that will be checked for meeting the {@link cost}.
*/
2022-12-31 20:57:09 +00:00
resource: Resource;
2023-02-05 10:23:53 +00:00
/**
* The amount of {@link resource} that must be met for this requirement. You can pass a formula, in which case maximizing will work out of the box (assuming its invertible and, for more accurate calculations, its integral is invertible). If you don't pass a formula then you can still support maximizing by passing a custom {@link pay} function.
2023-02-05 10:23:53 +00:00
*/
2023-02-05 08:44:03 +00:00
cost: Computable<DecimalSource> | GenericFormula;
2023-02-05 10:23:53 +00:00
/**
* Pass-through to {@link Requirement.visibility}.
*/
visibility?: Computable<Visibility.Visible | Visibility.None | boolean>;
2023-02-05 10:23:53 +00:00
/**
* Pass-through to {@link Requirement.requiresPay}. If not set to false, the default {@link pay} function will remove {@link cost} from {@link resource}.
*/
2023-02-05 08:44:03 +00:00
requiresPay?: Computable<boolean>;
2023-02-05 10:23:53 +00:00
/**
* When calculating multiple levels to be handled at once, whether it should consider resources used for each level as spent. Setting this to false causes calculations to be faster with larger numbers and supports more math functions.
* @see {Formula}
*/
2023-03-21 05:15:28 +00:00
spendResources: Computable<boolean>;
2023-02-05 10:23:53 +00:00
/**
* Pass-through to {@link Requirement.pay}. May be required for maximizing support.
* @see {@link cost} for restrictions on maximizing support.
2023-02-05 10:23:53 +00:00
*/
2023-02-05 08:44:03 +00:00
pay?: (amount?: DecimalSource) => void;
2022-12-31 20:57:09 +00:00
}
export type CostRequirement = Requirement & CostRequirementOptions;
2023-02-05 10:23:53 +00:00
/**
* Lazily creates a requirement with the given options, that is based on meeting an amount of a resource.
* @param optionsFunc Cost requirement options.
*/
export function createCostRequirement<T extends CostRequirementOptions>(
optionsFunc: () => T
): CostRequirement {
2022-12-31 20:57:09 +00:00
return createLazyProxy(() => {
const req = optionsFunc() as T & Partial<Requirement>;
2023-02-05 08:44:03 +00:00
req.partialDisplay = amount => (
2022-12-31 20:57:09 +00:00
<span
style={
unref(req.requirementMet as ProcessedComputable<boolean>)
? ""
: "color: var(--danger)"
}
>
{displayResource(
req.resource,
2023-02-05 08:44:03 +00:00
req.cost instanceof Formula
? calculateCost(
req.cost,
amount ?? 1,
unref(
req.spendResources as ProcessedComputable<boolean> | undefined
) ?? true
)
: unref(req.cost as ProcessedComputable<DecimalSource>)
2022-12-31 20:57:09 +00:00
)}{" "}
{req.resource.displayName}
</span>
2023-02-05 08:44:03 +00:00
);
req.display = amount => (
2022-12-31 20:57:09 +00:00
<div>
{unref(req.requiresPay as ProcessedComputable<boolean>) ? "Costs: " : "Requires: "}
{displayResource(
req.resource,
2023-02-05 08:44:03 +00:00
req.cost instanceof Formula
? calculateCost(
req.cost,
amount ?? 1,
unref(
req.spendResources as ProcessedComputable<boolean> | undefined
) ?? true
)
: unref(req.cost as ProcessedComputable<DecimalSource>)
2022-12-31 20:57:09 +00:00
)}{" "}
{req.resource.displayName}
</div>
2023-02-05 08:44:03 +00:00
);
2022-12-31 20:57:09 +00:00
processComputable(req as T, "visibility");
setDefault(req, "visibility", Visibility.Visible);
processComputable(req as T, "cost");
processComputable(req as T, "requiresPay");
setDefault(req, "requiresPay", true);
2023-03-21 05:15:28 +00:00
processComputable(req as T, "spendResources");
setDefault(req, "spendResources", false);
2023-02-05 08:44:03 +00:00
setDefault(req, "pay", function (amount?: DecimalSource) {
const cost =
req.cost instanceof Formula
? calculateCost(
req.cost,
amount ?? 1,
unref(req.spendResources as ProcessedComputable<boolean> | undefined) ??
true
)
: unref(req.cost as ProcessedComputable<DecimalSource>);
req.resource.value = Decimal.sub(req.resource.value, cost).max(0);
2022-12-31 20:57:09 +00:00
});
2023-02-05 08:44:03 +00:00
req.canMaximize = req.cost instanceof Formula && req.cost.isInvertible();
if (req.canMaximize) {
2023-02-05 08:44:03 +00:00
req.requirementMet = calculateMaxAffordable(
req.cost as InvertibleFormula,
2023-02-05 08:44:03 +00:00
req.resource,
unref(req.spendResources as ProcessedComputable<boolean> | undefined) ?? true
);
} else {
req.requirementMet = computed(() => {
if (req.cost instanceof Formula) {
return Decimal.gte(req.resource.value, req.cost.evaluate());
} else {
return Decimal.gte(
req.resource.value,
unref(req.cost as ProcessedComputable<DecimalSource>)
);
}
});
}
2022-12-31 20:57:09 +00:00
return req as CostRequirement;
2022-12-31 20:57:09 +00:00
});
}
2023-02-05 10:23:53 +00:00
/**
* Utility function for creating a requirement that a specified vue feature is visible
* @param feature The feature to check the visibility of
*/
2022-12-31 20:57:09 +00:00
export function createVisibilityRequirement(feature: {
visibility: ProcessedComputable<Visibility | boolean>;
2022-12-31 20:57:09 +00:00
}): Requirement {
return createLazyProxy(() => ({
requirementMet: computed(() => isVisible(feature.visibility)),
2022-12-31 20:57:09 +00:00
visibility: Visibility.None,
requiresPay: false
}));
}
2023-02-05 10:23:53 +00:00
/**
* Creates a requirement based on a true/false value
* @param requirement The boolean requirement to use
* @param display How to display this requirement to the user
*/
2022-12-31 20:57:09 +00:00
export function createBooleanRequirement(
requirement: Computable<boolean>,
display?: CoercableComponent
): Requirement {
return createLazyProxy(() => ({
requirementMet: convertComputable(requirement),
partialDisplay: display == null ? undefined : jsx(() => renderJSX(display)),
display: display == null ? undefined : jsx(() => <>Req: {renderJSX(display)}</>),
visibility: display == null ? Visibility.None : Visibility.Visible,
requiresPay: false
}));
}
2023-02-05 10:23:53 +00:00
/**
* Utility for checking if 1+ requirements are all met
* @param requirements The 1+ requirements to check
*/
2023-02-05 08:44:03 +00:00
export function requirementsMet(requirements: Requirements): boolean {
if (isArray(requirements)) {
return requirements.every(requirementsMet);
}
const reqsMet = unref(requirements.requirementMet);
return typeof reqsMet === "boolean" ? reqsMet : Decimal.gt(reqsMet, 0);
}
2023-02-05 10:23:53 +00:00
/**
* Calculates the maximum number of levels that could be acquired with the current requirement states. True/false requirements will be counted as Infinity or 0.
* @param requirements The 1+ requirements to check
*/
2023-02-05 08:44:03 +00:00
export function maxRequirementsMet(requirements: Requirements): DecimalSource {
2022-12-31 20:57:09 +00:00
if (isArray(requirements)) {
2023-02-05 08:44:03 +00:00
return requirements.map(maxRequirementsMet).reduce(Decimal.min);
}
const reqsMet = unref(requirements.requirementMet);
if (typeof reqsMet === "boolean") {
return reqsMet ? Infinity : 0;
} else if (Decimal.gt(reqsMet, 1) && unref(requirements.canMaximize) !== true) {
return 1;
2022-12-31 20:57:09 +00:00
}
2023-02-05 08:44:03 +00:00
return reqsMet;
2022-12-31 20:57:09 +00:00
}
2023-02-05 10:23:53 +00:00
/**
* Utility function for display 1+ requirements compactly.
* @param requirements The 1+ requirements to display
* @param amount The amount of levels earned to be displayed
*/
2023-02-05 08:44:03 +00:00
export function displayRequirements(requirements: Requirements, amount: DecimalSource = 1) {
2022-12-31 20:57:09 +00:00
if (isArray(requirements)) {
requirements = requirements.filter(r => isVisible(r.visibility));
2022-12-31 20:57:09 +00:00
if (requirements.length === 1) {
requirements = requirements[0];
}
}
if (isArray(requirements)) {
requirements = requirements.filter(r => "partialDisplay" in r);
const withCosts = requirements.filter(r => unref(r.requiresPay));
const withoutCosts = requirements.filter(r => !unref(r.requiresPay));
return (
<>
{withCosts.length > 0 ? (
<div>
Costs:{" "}
{joinJSX(
2023-02-05 08:44:03 +00:00
withCosts.map(r => r.partialDisplay!(amount)),
2022-12-31 20:57:09 +00:00
<>, </>
)}
</div>
) : null}
{withoutCosts.length > 0 ? (
<div>
Requires:{" "}
{joinJSX(
2023-02-05 08:44:03 +00:00
withoutCosts.map(r => r.partialDisplay!(amount)),
2022-12-31 20:57:09 +00:00
<>, </>
)}
</div>
) : null}
</>
);
}
return requirements.display?.() ?? <></>;
}
2023-02-05 10:23:53 +00:00
/**
* Utility function for paying the costs for 1+ requirements
* @param requirements The 1+ requirements to pay
* @param amount How many levels to pay for
*/
2023-02-05 08:51:07 +00:00
export function payRequirements(requirements: Requirements, amount: DecimalSource = 1) {
2022-12-31 20:57:09 +00:00
if (isArray(requirements)) {
2023-02-05 08:44:03 +00:00
requirements.filter(r => unref(r.requiresPay)).forEach(r => r.pay?.(amount));
2022-12-31 20:57:09 +00:00
} else if (unref(requirements.requiresPay)) {
2023-02-05 08:44:03 +00:00
requirements.pay?.(amount);
2022-12-31 20:57:09 +00:00
}
}
export function payByDivision(this: CostRequirement, amount?: DecimalSource) {
const cost =
this.cost instanceof Formula
? calculateCost(
this.cost,
amount ?? 1,
unref(this.spendResources as ProcessedComputable<boolean> | undefined) ?? true
)
: unref(this.cost as ProcessedComputable<DecimalSource>);
this.resource.value = Decimal.div(this.resource.value, cost);
}
export function payByReset(overrideDefaultValue?: DecimalSource) {
return function (this: CostRequirement) {
this.resource.value =
overrideDefaultValue ??
(this.resource as Resource & Persistent<DecimalSource>)[DefaultValue] ??
0;
};
}