import {
    Conversion,
    createCumulativeConversion,
    createIndependentConversion,
    setupPassiveGeneration
} from "features/conversion";
import { createResource, Resource } from "features/resources/resource";
import { InvertibleIntegralFormula } from "game/formulas/types";
import { createLayer, Layer } from "game/layers";
import Decimal from "util/bignum";
import { 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: InvertibleIntegralFormula) => InvertibleIntegralFormula;
    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: Conversion;
            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);
            });
            test("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.currentGain)).compare_tolerance(0);
            });
        });
        describe("Calculates actualGain correctly", () => {
            let conversion: Conversion;
            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);
            });
            test("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.actualGain)).compare_tolerance(0);
            });
        });
        describe("Calculates currentAt correctly", () => {
            let conversion: Conversion;
            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)
                );
            });
            test("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.currentAt)).compare_tolerance(0);
            });
        });
        describe("Calculates nextAt correctly", () => {
            let conversion: Conversion;
            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("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.nextAt)).compare_tolerance(Decimal.dTen);
            });
        });
        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: Conversion;
            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);
            });
            test("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.currentGain)).compare_tolerance(1);
            });
        });
        describe("Calculates actualGain correctly", () => {
            let conversion: Conversion;
            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);
            });
            test("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.actualGain)).compare_tolerance(0);
            });
        });
        describe("Calculates currentAt correctly", () => {
            let conversion: Conversion;
            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)
                );
            });
            test("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.currentAt)).compare_tolerance(Decimal.pow(1, 2).times(10));
            });
        });
        describe("Calculates nextAt correctly", () => {
            let conversion: Conversion;
            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("Zero", () => {
                baseResource.value = Decimal.dZero;
                expect(unref(conversion.nextAt)).compare_tolerance(Decimal.pow(2, 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: Conversion;
            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: Conversion;
            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: InvertibleIntegralFormula) => InvertibleIntegralFormula;
    let conversion: Conversion;
    let layer: Layer;
    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);
    });
});