forked from profectus/Profectus
Merge remote-tracking branch 'upstream/main'
This commit is contained in:
commit
3cac14d81d
5 changed files with 528 additions and 25 deletions
25
CHANGELOG.md
25
CHANGELOG.md
|
@ -9,9 +9,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
### Added
|
### Added
|
||||||
- **BREAKING** New requirements system
|
- **BREAKING** New requirements system
|
||||||
- Replaces many features' existing requirements with new generic form
|
- Replaces many features' existing requirements with new generic form
|
||||||
- Formulas, which can be used to calculate buy max for you
|
- **BREAKING** Formulas, which can be used to calculate buy max for you
|
||||||
- Action feature
|
- Requirements can use them so repeatables and challenges can be "buy max" without any extra effort
|
||||||
- ETA util
|
- Conversions now use formulas instead of the old scaling functions system, allowing for arbitrary functions that are much easier to follow
|
||||||
|
- There's a utility for converting modifiers to formulas, thus replacing things like the gain modifier on conversions
|
||||||
|
- Action feature, which is a clickable with a cooldown
|
||||||
|
- ETA util (calculates time until a specific amount of a resource, based on its current gain rate)
|
||||||
- createCollapsibleMilestones util
|
- createCollapsibleMilestones util
|
||||||
- deleteLowerSaves util
|
- deleteLowerSaves util
|
||||||
- Minimized layers can now display a component
|
- Minimized layers can now display a component
|
||||||
|
@ -35,6 +38,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
- Tweaked modifier displays, incl showing negative modifiers in red
|
- Tweaked modifier displays, incl showing negative modifiers in red
|
||||||
- Hotkeys now appear on key graphic
|
- Hotkeys now appear on key graphic
|
||||||
- Mofifier sections now accept computable strings for title and subtitle
|
- Mofifier sections now accept computable strings for title and subtitle
|
||||||
|
- Every VueFeature's `[Component]` property is now typed as GenericComponent
|
||||||
|
- Make errors throw objects instead of strings
|
||||||
- Updated b_e
|
- Updated b_e
|
||||||
### Fixed
|
### Fixed
|
||||||
- NaN detection stopped working
|
- NaN detection stopped working
|
||||||
|
@ -54,15 +59,19 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||||
- Tabs could sometimes not update correctly
|
- Tabs could sometimes not update correctly
|
||||||
- offlineTime not capping properly
|
- offlineTime not capping properly
|
||||||
- Tooltips being user-selectable
|
- Tooltips being user-selectable
|
||||||
|
- Pinnable tooltips causing stack overflow
|
||||||
- Workflows not working with submodules
|
- Workflows not working with submodules
|
||||||
- Various minor typing issues
|
- Various minor typing issues
|
||||||
|
### Removed
|
||||||
|
- **BREAKING** Removed milestones (achievements now have small and large displays)
|
||||||
### Documented
|
### Documented
|
||||||
- requirements.tsx
|
- every single feature
|
||||||
- formulas.tsx
|
|
||||||
- repeatables.tsx
|
|
||||||
### Tests
|
|
||||||
- requirements
|
|
||||||
- formulas
|
- formulas
|
||||||
|
- requirements
|
||||||
|
### Tests
|
||||||
|
- conversions
|
||||||
|
- formulas
|
||||||
|
- requirements
|
||||||
|
|
||||||
Contributors: thepaperpilot, escapee, adsaf, ducdat
|
Contributors: thepaperpilot, escapee, adsaf, ducdat
|
||||||
|
|
||||||
|
|
|
@ -492,6 +492,11 @@ export function createFormulaPreview(
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utility for converting a modifier into a formula. Takes the input for this formula as the base parameter.
|
||||||
|
* @param modifier The modifier to convert to the formula
|
||||||
|
* @param base An existing formula or processed DecimalSource that will be the input to the formula
|
||||||
|
*/
|
||||||
export function modifierToFormula<T extends GenericFormula>(
|
export function modifierToFormula<T extends GenericFormula>(
|
||||||
modifier: WithRequired<Modifier, "revert">,
|
modifier: WithRequired<Modifier, "revert">,
|
||||||
base: T
|
base: T
|
||||||
|
|
|
@ -139,11 +139,11 @@ export function createConversion<T extends ConversionOptions>(
|
||||||
);
|
);
|
||||||
if (conversion.currentGain == null) {
|
if (conversion.currentGain == null) {
|
||||||
conversion.currentGain = computed(() => {
|
conversion.currentGain = computed(() => {
|
||||||
let gain = (conversion as GenericConversion).formula.evaluate(
|
let gain = Decimal.floor(
|
||||||
conversion.baseResource.value
|
(conversion as GenericConversion).formula.evaluate(
|
||||||
);
|
conversion.baseResource.value
|
||||||
gain = Decimal.floor(gain).max(0);
|
)
|
||||||
|
).max(0);
|
||||||
if (unref(conversion.buyMax) === false) {
|
if (unref(conversion.buyMax) === false) {
|
||||||
gain = gain.min(1);
|
gain = gain.min(1);
|
||||||
}
|
}
|
||||||
|
@ -228,10 +228,11 @@ export function createIndependentConversion<S extends ConversionOptions>(
|
||||||
|
|
||||||
if (conversion.currentGain == null) {
|
if (conversion.currentGain == null) {
|
||||||
conversion.currentGain = computed(() => {
|
conversion.currentGain = computed(() => {
|
||||||
let gain = (conversion as unknown as GenericConversion).formula.evaluate(
|
let gain = Decimal.floor(
|
||||||
conversion.baseResource.value
|
(conversion as unknown as GenericConversion).formula.evaluate(
|
||||||
);
|
conversion.baseResource.value
|
||||||
gain = Decimal.floor(gain).max(conversion.gainResource.value);
|
)
|
||||||
|
).max(conversion.gainResource.value);
|
||||||
if (unref(conversion.buyMax) === false) {
|
if (unref(conversion.buyMax) === false) {
|
||||||
gain = gain.min(Decimal.add(conversion.gainResource.value, 1));
|
gain = gain.min(Decimal.add(conversion.gainResource.value, 1));
|
||||||
}
|
}
|
||||||
|
@ -245,7 +246,9 @@ export function createIndependentConversion<S extends ConversionOptions>(
|
||||||
conversion.baseResource.value
|
conversion.baseResource.value
|
||||||
),
|
),
|
||||||
conversion.gainResource.value
|
conversion.gainResource.value
|
||||||
).max(0);
|
)
|
||||||
|
.floor()
|
||||||
|
.max(0);
|
||||||
|
|
||||||
if (unref(conversion.buyMax) === false) {
|
if (unref(conversion.buyMax) === false) {
|
||||||
gain = gain.min(1);
|
gain = gain.min(1);
|
||||||
|
@ -273,13 +276,13 @@ export function createIndependentConversion<S extends ConversionOptions>(
|
||||||
* @param layer The layer this passive generation will be associated with. Typically `this` when calling this function from inside a layer's options function.
|
* @param layer The layer this passive generation will be associated with. Typically `this` when calling this function from inside a layer's options function.
|
||||||
* @param conversion The conversion that will determine how much generation there is.
|
* @param conversion The conversion that will determine how much generation there is.
|
||||||
* @param rate A multiplier to multiply against the conversion's currentGain.
|
* @param rate A multiplier to multiply against the conversion's currentGain.
|
||||||
* @param cap A value that should not be passed via passive generation. If null, no cap is applied.
|
* @param cap A value that should not be passed via passive generation.
|
||||||
*/
|
*/
|
||||||
export function setupPassiveGeneration(
|
export function setupPassiveGeneration(
|
||||||
layer: BaseLayer,
|
layer: BaseLayer,
|
||||||
conversion: GenericConversion,
|
conversion: GenericConversion,
|
||||||
rate: Computable<DecimalSource> = 1,
|
rate: Computable<DecimalSource> = 1,
|
||||||
cap: Computable<DecimalSource | null> = null
|
cap: Computable<DecimalSource> = Decimal.dInf
|
||||||
): void {
|
): void {
|
||||||
const processedRate = convertComputable(rate);
|
const processedRate = convertComputable(rate);
|
||||||
const processedCap = convertComputable(cap);
|
const processedCap = convertComputable(cap);
|
||||||
|
@ -290,7 +293,7 @@ export function setupPassiveGeneration(
|
||||||
conversion.gainResource.value,
|
conversion.gainResource.value,
|
||||||
Decimal.times(currRate, diff).times(Decimal.ceil(unref(conversion.actualGain)))
|
Decimal.times(currRate, diff).times(Decimal.ceil(unref(conversion.actualGain)))
|
||||||
)
|
)
|
||||||
.min(unref(processedCap) ?? Decimal.dInf)
|
.min(unref(processedCap))
|
||||||
.max(conversion.gainResource.value);
|
.max(conversion.gainResource.value);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -317,8 +317,8 @@ export default class Formula<T extends [FormulaSource] | FormulaSource[]> {
|
||||||
|
|
||||||
// TODO add integration support to step-wise functions
|
// TODO add integration support to step-wise functions
|
||||||
/**
|
/**
|
||||||
* Creates a step-wise formula. After {@ref start} the formula will have an additional modifier.
|
* Creates a step-wise formula. After {@link start} the formula will have an additional modifier.
|
||||||
* This function assumes the incoming {@ref value} will be continuous and monotonically increasing.
|
* This function assumes the incoming {@link value} will be continuous and monotonically increasing.
|
||||||
* @param value The value before applying the step
|
* @param value The value before applying the step
|
||||||
* @param start The value at which to start applying the step
|
* @param start The value at which to start applying the step
|
||||||
* @param formulaModifier How this step should modify the formula. The incoming value will be the unmodified formula value _minus the start value_. So for example if an incoming formula evaluates to 200 and has a step that starts at 150, the formulaModifier would be given 50 as the parameter
|
* @param formulaModifier How this step should modify the formula. The incoming value will be the unmodified formula value _minus the start value_. So for example if an incoming formula evaluates to 200 and has a step that starts at 150, the formulaModifier would be given 50 as the parameter
|
||||||
|
@ -1356,7 +1356,7 @@ export function printFormula(formula: FormulaSource): string {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility for calculating the maximum amount of purchases possible with a given formula and resource. If {@ref spendResources} is changed to false, the calculation will be much faster with higher numbers.
|
* Utility for calculating the maximum amount of purchases possible with a given formula and resource. If {@link spendResources} is changed to false, the calculation will be much faster with higher numbers.
|
||||||
* @param formula The formula to use for calculating buy max from
|
* @param formula The formula to use for calculating buy max from
|
||||||
* @param resource The resource used when purchasing (is only read from)
|
* @param resource The resource used when purchasing (is only read from)
|
||||||
* @param spendResources Whether or not to count spent resources on each purchase or not. If true, costs will be approximated for performance, skewing towards fewer purchases
|
* @param spendResources Whether or not to count spent resources on each purchase or not. If true, costs will be approximated for performance, skewing towards fewer purchases
|
||||||
|
@ -1424,7 +1424,7 @@ export function calculateMaxAffordable(
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility for calculating the cost of a formula for a given amount of purchases. If {@ref spendResources} is changed to false, the calculation will be much faster with higher numbers.
|
* Utility for calculating the cost of a formula for a given amount of purchases. If {@link spendResources} is changed to false, the calculation will be much faster with higher numbers.
|
||||||
* @param formula The formula to use for calculating buy max from
|
* @param formula The formula to use for calculating buy max from
|
||||||
* @param amountToBuy The amount of purchases to calculate the cost for
|
* @param amountToBuy The amount of purchases to calculate the cost for
|
||||||
* @param spendResources Whether or not to count spent resources on each purchase or not. If true, costs will be approximated for performance, skewing towards higher cost
|
* @param spendResources Whether or not to count spent resources on each purchase or not. If true, costs will be approximated for performance, skewing towards higher cost
|
||||||
|
|
486
tests/features/conversions.test.ts
Normal file
486
tests/features/conversions.test.ts
Normal file
|
@ -0,0 +1,486 @@
|
||||||
|
import {
|
||||||
|
createCumulativeConversion,
|
||||||
|
createIndependentConversion,
|
||||||
|
GenericConversion,
|
||||||
|
setupPassiveGeneration
|
||||||
|
} from "features/conversion";
|
||||||
|
import { createResource, Resource } from "features/resources/resource";
|
||||||
|
import { GenericFormula } from "game/formulas/types";
|
||||||
|
import { createLayer, GenericLayer } from "game/layers";
|
||||||
|
import Decimal from "util/bignum";
|
||||||
|
import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, test, vi } from "vitest";
|
||||||
|
import { ref, unref } from "vue";
|
||||||
|
import "../utils";
|
||||||
|
|
||||||
|
describe("Creating conversion", () => {
|
||||||
|
let baseResource: Resource;
|
||||||
|
let gainResource: Resource;
|
||||||
|
let formula: (x: GenericFormula) => GenericFormula;
|
||||||
|
beforeEach(() => {
|
||||||
|
baseResource = createResource(ref(40));
|
||||||
|
gainResource = createResource(ref(1));
|
||||||
|
formula = x => x.div(10).sqrt();
|
||||||
|
});
|
||||||
|
afterEach(() => {
|
||||||
|
vi.restoreAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Cumulative conversion", () => {
|
||||||
|
describe("Calculates currentGain correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(99);
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Calculates actualGain correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(99);
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Calculates currentAt correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(
|
||||||
|
Decimal.pow(100, 2).times(10)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(Decimal.pow(99, 2).times(10));
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(
|
||||||
|
Decimal.pow(100, 2).times(10)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Calculates nextAt correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(101, 2).times(10));
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(100, 2).times(10));
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(101, 2).times(10));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test("Converts correctly", () => {
|
||||||
|
const conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(baseResource.value).compare_tolerance(0);
|
||||||
|
expect(gainResource.value).compare_tolerance(3);
|
||||||
|
});
|
||||||
|
describe("Obeys buy max", () => {
|
||||||
|
test("buyMax = false", () => {
|
||||||
|
const conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: false
|
||||||
|
}));
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(1);
|
||||||
|
});
|
||||||
|
test("buyMax = true", () => {
|
||||||
|
const conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: true
|
||||||
|
}));
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test("Spends correctly", () => {
|
||||||
|
const conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(baseResource.value).compare_tolerance(0);
|
||||||
|
});
|
||||||
|
test("Calls onConvert", () => {
|
||||||
|
const onConvert = vi.fn();
|
||||||
|
const conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
onConvert
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(onConvert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Independent conversion", () => {
|
||||||
|
describe("Calculates currentGain correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: true
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(99);
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Calculates actualGain correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: true
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(99);
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(98);
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(99);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Calculates currentAt correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: true
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(
|
||||||
|
Decimal.pow(100, 2).times(10)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(Decimal.pow(99, 2).times(10));
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(
|
||||||
|
Decimal.pow(100, 2).times(10)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Calculates nextAt correctly", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
beforeEach(() => {
|
||||||
|
conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: true
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
test("Exactly enough", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10);
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(101, 2).times(10));
|
||||||
|
});
|
||||||
|
test("Just under", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).sub(1);
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(100, 2).times(10));
|
||||||
|
});
|
||||||
|
test("Just over", () => {
|
||||||
|
baseResource.value = Decimal.pow(100, 2).times(10).add(1);
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(101, 2).times(10));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test("Converts correctly", () => {
|
||||||
|
const conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(baseResource.value).compare_tolerance(0);
|
||||||
|
expect(gainResource.value).compare_tolerance(2);
|
||||||
|
});
|
||||||
|
describe("Obeys buy max", () => {
|
||||||
|
test("buyMax = false", () => {
|
||||||
|
const conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: false
|
||||||
|
}));
|
||||||
|
baseResource.value = 90;
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(1);
|
||||||
|
});
|
||||||
|
test("buyMax = true", () => {
|
||||||
|
const conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
buyMax: true
|
||||||
|
}));
|
||||||
|
baseResource.value = 90;
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test("Spends correctly", () => {
|
||||||
|
const conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(baseResource.value).compare_tolerance(0);
|
||||||
|
});
|
||||||
|
test("Calls onConvert", () => {
|
||||||
|
const onConvert = vi.fn();
|
||||||
|
const conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
onConvert
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(onConvert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Custom conversion", () => {
|
||||||
|
describe("Custom cumulative", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
const convert = vi.fn();
|
||||||
|
const spend = vi.fn();
|
||||||
|
const onConvert = vi.fn();
|
||||||
|
beforeAll(() => {
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
currentGain() {
|
||||||
|
return 10;
|
||||||
|
},
|
||||||
|
actualGain() {
|
||||||
|
return 5;
|
||||||
|
},
|
||||||
|
currentAt() {
|
||||||
|
return 100;
|
||||||
|
},
|
||||||
|
nextAt() {
|
||||||
|
return 1000;
|
||||||
|
},
|
||||||
|
convert,
|
||||||
|
spend,
|
||||||
|
onConvert
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
afterEach(() => {
|
||||||
|
vi.resetAllMocks();
|
||||||
|
});
|
||||||
|
test("Calculates currentGain correctly", () => {
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(10);
|
||||||
|
});
|
||||||
|
test("Calculates actualGain correctly", () => {
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(5);
|
||||||
|
});
|
||||||
|
test("Calculates currentAt correctly", () => {
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
test("Calculates nextAt correctly", () => {
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(1000);
|
||||||
|
});
|
||||||
|
test("Calls convert", () => {
|
||||||
|
conversion.convert();
|
||||||
|
expect(convert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
test("Calls spend and onConvert", () => {
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
spend,
|
||||||
|
onConvert
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(spend).toHaveBeenCalled();
|
||||||
|
expect(spend).toHaveBeenCalledWith(expect.compare_tolerance(2));
|
||||||
|
expect(onConvert).toHaveBeenCalled();
|
||||||
|
expect(onConvert).toHaveBeenCalledWith(expect.compare_tolerance(2));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
describe("Custom independent", () => {
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
const convert = vi.fn();
|
||||||
|
const spend = vi.fn();
|
||||||
|
const onConvert = vi.fn();
|
||||||
|
beforeAll(() => {
|
||||||
|
conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
currentGain() {
|
||||||
|
return 10;
|
||||||
|
},
|
||||||
|
actualGain() {
|
||||||
|
return 5;
|
||||||
|
},
|
||||||
|
currentAt() {
|
||||||
|
return 100;
|
||||||
|
},
|
||||||
|
nextAt() {
|
||||||
|
return 1000;
|
||||||
|
},
|
||||||
|
convert,
|
||||||
|
spend,
|
||||||
|
onConvert
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
afterEach(() => {
|
||||||
|
vi.resetAllMocks();
|
||||||
|
});
|
||||||
|
test("Calculates currentGain correctly", () => {
|
||||||
|
expect(unref(conversion.currentGain)).compare_tolerance(10);
|
||||||
|
});
|
||||||
|
test("Calculates actualGain correctly", () => {
|
||||||
|
expect(unref(conversion.actualGain)).compare_tolerance(5);
|
||||||
|
});
|
||||||
|
test("Calculates currentAt correctly", () => {
|
||||||
|
expect(unref(conversion.currentAt)).compare_tolerance(100);
|
||||||
|
});
|
||||||
|
test("Calculates nextAt correctly", () => {
|
||||||
|
expect(unref(conversion.nextAt)).compare_tolerance(1000);
|
||||||
|
});
|
||||||
|
test("Calls convert", () => {
|
||||||
|
conversion.convert();
|
||||||
|
expect(convert).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
test("Calls spend and onConvert", () => {
|
||||||
|
conversion = createIndependentConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula,
|
||||||
|
spend,
|
||||||
|
onConvert
|
||||||
|
}));
|
||||||
|
conversion.convert();
|
||||||
|
expect(spend).toHaveBeenCalled();
|
||||||
|
expect(spend).toHaveBeenCalledWith(expect.compare_tolerance(1));
|
||||||
|
expect(onConvert).toHaveBeenCalled();
|
||||||
|
expect(onConvert).toHaveBeenCalledWith(expect.compare_tolerance(1));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Passive generation", () => {
|
||||||
|
let baseResource: Resource;
|
||||||
|
let gainResource: Resource;
|
||||||
|
let formula: (x: GenericFormula) => GenericFormula;
|
||||||
|
let conversion: GenericConversion;
|
||||||
|
let layer: GenericLayer;
|
||||||
|
beforeEach(() => {
|
||||||
|
baseResource = createResource(ref(10));
|
||||||
|
gainResource = createResource(ref(1));
|
||||||
|
formula = x => x.div(10).sqrt();
|
||||||
|
conversion = createCumulativeConversion(() => ({
|
||||||
|
baseResource,
|
||||||
|
gainResource,
|
||||||
|
formula
|
||||||
|
}));
|
||||||
|
layer = createLayer("dummy", () => ({ display: "" }));
|
||||||
|
});
|
||||||
|
test("Rate is 0", () => {
|
||||||
|
setupPassiveGeneration(layer, conversion, 0);
|
||||||
|
layer.emit("preUpdate", 1);
|
||||||
|
expect(gainResource.value).compare_tolerance(1);
|
||||||
|
});
|
||||||
|
test("Rate is 1", () => {
|
||||||
|
setupPassiveGeneration(layer, conversion);
|
||||||
|
layer.emit("preUpdate", 1);
|
||||||
|
expect(gainResource.value).compare_tolerance(2);
|
||||||
|
})
|
||||||
|
test("Rate is 100", () => {
|
||||||
|
setupPassiveGeneration(layer, conversion, () => 100);
|
||||||
|
layer.emit("preUpdate", 1);
|
||||||
|
expect(gainResource.value).compare_tolerance(101);
|
||||||
|
})
|
||||||
|
test("Obeys cap", () => {
|
||||||
|
setupPassiveGeneration(layer, conversion, 100, () => 100);
|
||||||
|
layer.emit("preUpdate", 1);
|
||||||
|
expect(gainResource.value).compare_tolerance(100);
|
||||||
|
})
|
||||||
|
});
|
Loading…
Reference in a new issue