import "components/common/modifiers.css"; import type { CoercableComponent } from "features/feature"; import { jsx } from "features/feature"; import type { DecimalSource } from "util/bignum"; import Decimal, { format } from "util/bignum"; import type { WithRequired } from "util/common"; import type { Computable, ProcessedComputable } from "util/computed"; import { convertComputable } from "util/computed"; import { createLazyProxy } from "util/proxies"; import { renderJSX } from "util/vue"; import { computed, unref } from "vue"; /** * An object that can be used to apply or unapply some modification to a number. * Being reversible requires the operation being invertible, but some features may rely on that. * Descriptions can be optionally included for displaying them to the player. * The built-in modifier creators are designed to display the modifiers using. * {@link createModifierSection}. */ export interface Modifier { /** Applies some operation on the input and returns the result. */ apply: (gain: DecimalSource) => DecimalSource; /** Reverses the operation applied by the apply property. Required by some features. */ revert?: (gain: DecimalSource) => DecimalSource; /** * Whether or not this modifier should be considered enabled. * Typically for use with modifiers passed into {@link createSequentialModifier}. */ enabled?: ProcessedComputable; /** * A description of this modifier. * @see {@link createModifierSection}. */ description?: ProcessedComputable; } /** * Utility type used to narrow down a modifier type that will have a description and/or enabled property based on optional parameters, T and S (respectively). */ export type ModifierFromOptionalParams = T extends undefined ? S extends undefined ? Omit, "description" | "enabled"> : Omit, "description"> : S extends undefined ? Omit, "enabled"> : WithRequired; /** An object that configures an additive modifier via {@link createAdditiveModifier}. */ export interface AdditiveModifierOptions { /** The amount to add to the input value. */ addend: Computable; /** Description of what this modifier is doing. */ description?: Computable | undefined; /** A computable that will be processed and passed directly into the returned modifier. */ enabled?: Computable | undefined; } /** * Create a modifier that adds some value to the input value. * @param optionsFunc Additive modifier options. */ export function createAdditiveModifier( optionsFunc: () => T ): ModifierFromOptionalParams { return createLazyProxy(() => { const { addend, description, enabled } = optionsFunc(); const processedAddend = convertComputable(addend); const processedDescription = convertComputable(description); const processedEnabled = enabled == null ? undefined : convertComputable(enabled); return { apply: (gain: DecimalSource) => Decimal.add(gain, unref(processedAddend)), revert: (gain: DecimalSource) => Decimal.sub(gain, unref(processedAddend)), enabled: processedEnabled, description: description == null ? undefined : jsx(() => (
{Decimal.gte(unref(processedAddend), 0) ? "+" : ""} {format(unref(processedAddend))} {unref(processedDescription) ? ( {/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */} {renderJSX(unref(processedDescription)!)} ) : null}
)) }; }) as unknown as ModifierFromOptionalParams; } /** An object that configures an multiplicative modifier via {@link createMultiplicativeModifier}. */ export interface MultiplicativeModifierOptions { /** The amount to multiply the input value by. */ multiplier: Computable; /** Description of what this modifier is doing. */ description?: Computable | undefined; /** A computable that will be processed and passed directly into the returned modifier. */ enabled?: Computable | undefined; } /** * Create a modifier that multiplies the input value by some value. * @param optionsFunc Multiplicative modifier options. */ export function createMultiplicativeModifier( optionsFunc: () => T ): ModifierFromOptionalParams { return createLazyProxy(() => { const { multiplier, description, enabled } = optionsFunc(); const processedMultiplier = convertComputable(multiplier); const processedDescription = convertComputable(description); const processedEnabled = enabled == null ? undefined : convertComputable(enabled); return { apply: (gain: DecimalSource) => Decimal.times(gain, unref(processedMultiplier)), revert: (gain: DecimalSource) => Decimal.div(gain, unref(processedMultiplier)), enabled: processedEnabled, description: description == null ? undefined : jsx(() => (
x{format(unref(processedMultiplier))} {unref(processedDescription) ? ( {/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */} {renderJSX(unref(processedDescription)!)} ) : null}
)) }; }) as unknown as ModifierFromOptionalParams; } /** An object that configures an exponential modifier via {@link createExponentialModifier}. */ export interface ExponentialModifierOptions { /** The amount to raise the input value to the power of. */ exponent: Computable; /** Description of what this modifier is doing. */ description?: Computable | undefined; /** A computable that will be processed and passed directly into the returned modifier. */ enabled?: Computable | undefined; /** Add 1 before calculating, then remove it afterwards. This prevents low numbers from becoming lower. */ supportLowNumbers?: boolean; } /** * Create a modifier that raises the input value to the power of some value. * @param optionsFunc Exponential modifier options. */ export function createExponentialModifier( optionsFunc: () => T ): ModifierFromOptionalParams { return createLazyProxy(() => { const { exponent, description, enabled, supportLowNumbers } = optionsFunc(); const processedExponent = convertComputable(exponent); const processedDescription = convertComputable(description); const processedEnabled = enabled == null ? undefined : convertComputable(enabled); return { apply: (gain: DecimalSource) => { let result = gain; if (supportLowNumbers) { result = Decimal.add(result, 1); } result = Decimal.pow(result, unref(processedExponent)); if (supportLowNumbers) { result = Decimal.sub(result, 1); } return result; }, revert: (gain: DecimalSource) => { let result = gain; if (supportLowNumbers) { result = Decimal.add(result, 1); } result = Decimal.root(result, unref(processedExponent)); if (supportLowNumbers) { result = Decimal.sub(result, 1); } return result; }, enabled: processedEnabled, description: description == null ? undefined : jsx(() => (
^{format(unref(processedExponent))} {unref(processedDescription) ? ( {/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */} {renderJSX(unref(processedDescription)!)} {supportLowNumbers ? " (+1 effective)" : null} ) : null}
)) }; }) as unknown as ModifierFromOptionalParams; } /** * Takes an array of modifiers and applies and reverses them in order. * Modifiers that are not enabled will not be applied nor reversed. * Also joins their descriptions together. * @param modifiersFunc The modifiers to perform sequentially. * @see {@link createModifierSection}. */ export function createSequentialModifier< T extends Modifier[], S = T extends WithRequired[] ? WithRequired : Omit, "revert"> >(modifiersFunc: () => T): S { return createLazyProxy(() => { const modifiers = modifiersFunc(); return { apply: (gain: DecimalSource) => modifiers .filter(m => unref(m.enabled) !== false) .reduce((gain, modifier) => modifier.apply(gain), gain), revert: modifiers.every(m => m.revert != null) ? (gain: DecimalSource) => modifiers .filter(m => unref(m.enabled) !== false) // eslint-disable-next-line @typescript-eslint/no-non-null-assertion .reduceRight((gain, modifier) => modifier.revert!(gain), gain) : undefined, enabled: computed(() => modifiers.filter(m => unref(m.enabled) !== false).length > 0), description: jsx(() => ( <> {( modifiers .filter(m => unref(m.enabled) !== false) .map(m => unref(m.description)) .filter(d => d) as CoercableComponent[] ).map(renderJSX)} )) }; }) as unknown as S; } /** * Create a JSX element that displays a modifier. * Intended to be used with the output from {@link createSequentialModifier}. * @param title The header for the section. * @param subtitle Smaller text that appears in the header after the title. * @param modifier The modifier to render. * @param base The base value that'll be passed into the modifier. * @param unit The unit of the value being modified, if any. * @param baseText The label to use for the base value. */ export function createModifierSection( title: string, subtitle: string, modifier: WithRequired, base: DecimalSource = 1, unit = "", baseText: CoercableComponent = "Base" ) { return (

{title} {subtitle ? ({subtitle}) : null}


{format(base)} {unit} {renderJSX(baseText)}
{renderJSX(unref(modifier.description))}
Total: {format(modifier.apply(base))} {unit}
); }