mirror of
				https://github.com/pagefaultgames/pokerogue.git
				synced 2025-10-25 06:25:52 +02:00 
			
		
		
		
	* Rename `Abilities` to `AbilityId` * Rename `abilities.ts` to `ability-id.ts` * Rename `Moves` to `MoveId` * Rename `moves.ts` to `move-id.ts` * Rename `Species` to `SpeciesId` * Rename `species.ts` to `species-id.ts` * Rename `Biome` to `BiomeId` * Rename `biome.ts` to `biome-id.ts` * Replace `Abilities` with `AbilityId` in comments * Replace `Biome` with `BiomeId` in comments * Replace `Moves` with `MoveId` in comments * Replace `Species` with `SpeciesId` in comments
		
			
				
	
	
		
			548 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			548 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import GameManager from "#test/testUtils/gameManager";
 | |
| import Phaser from "phaser";
 | |
| import { afterEach, beforeAll, beforeEach, describe, expect, it, type MockInstance, vi } from "vitest";
 | |
| import PokedexUiHandler from "#app/ui/pokedex-ui-handler";
 | |
| import { FilterTextRow } from "#app/ui/filter-text";
 | |
| import { allAbilities } from "#app/data/data-lists";
 | |
| import { AbilityId } from "#enums/ability-id";
 | |
| import { SpeciesId } from "#enums/species-id";
 | |
| import { allSpecies, getPokemonSpecies, type PokemonForm } from "#app/data/pokemon-species";
 | |
| import { Button } from "#enums/buttons";
 | |
| import { DropDownColumn } from "#enums/drop-down-column";
 | |
| import type PokemonSpecies from "#app/data/pokemon-species";
 | |
| import { PokemonType } from "#enums/pokemon-type";
 | |
| import { UiMode } from "#enums/ui-mode";
 | |
| import PokedexPageUiHandler from "#app/ui/pokedex-page-ui-handler";
 | |
| import type { StarterAttributes } from "#app/system/game-data";
 | |
| 
 | |
| /*
 | |
| Information for the `data_pokedex_tests.psrv`:
 | |
| 
 | |
| Caterpie - Shiny 0
 | |
| Rattata - Shiny 1
 | |
| Ekans - Shiny 2
 | |
| 
 | |
| Chikorita has enough candies to unlock passive
 | |
| Cyndaquil has first cost reduction unlocked, enough candies to buy the second
 | |
| Totodile has first cost reduction unlocked, not enough candies to buy the second
 | |
| Treecko has both cost reduction unlocked
 | |
| Torchic has enough candies to do anything
 | |
| Mudkip has passive unlocked
 | |
| Turtwig has enough candies to purchase an egg
 | |
| */
 | |
| 
 | |
| /**
 | |
|  * Return all permutations of elements from an array
 | |
|  */
 | |
| function permutations<T>(array: T[], length: number): T[][] {
 | |
|   if (length === 0) {
 | |
|     return [[]];
 | |
|   }
 | |
|   return array.flatMap((item, index) =>
 | |
|     permutations([...array.slice(0, index), ...array.slice(index + 1)], length - 1).map(perm => [item, ...perm]),
 | |
|   );
 | |
| }
 | |
| 
 | |
| describe("UI - Pokedex", () => {
 | |
|   let phaserGame: Phaser.Game;
 | |
|   let game: GameManager;
 | |
|   const mocks: MockInstance[] = [];
 | |
| 
 | |
|   beforeAll(() => {
 | |
|     phaserGame = new Phaser.Game({
 | |
|       type: Phaser.HEADLESS,
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   afterEach(() => {
 | |
|     while (mocks.length > 0) {
 | |
|       mocks.pop()?.mockRestore();
 | |
|     }
 | |
|     game.phaseInterceptor.restoreOg();
 | |
|   });
 | |
| 
 | |
|   beforeEach(() => {
 | |
|     game = new GameManager(phaserGame);
 | |
|   });
 | |
| 
 | |
|   /**
 | |
|    * Run the game to open the pokedex UI.
 | |
|    * @returns The handler for the pokedex UI.
 | |
|    */
 | |
|   async function runToOpenPokedex(): Promise<PokedexUiHandler> {
 | |
|     // Open the pokedex UI.
 | |
|     await game.runToTitle();
 | |
| 
 | |
|     await game.phaseInterceptor.setOverlayMode(UiMode.POKEDEX);
 | |
| 
 | |
|     // Get the handler for the current UI.
 | |
|     const handler = game.scene.ui.getHandler();
 | |
|     expect(handler).toBeInstanceOf(PokedexUiHandler);
 | |
| 
 | |
|     return handler as PokedexUiHandler;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Run the game to open the pokedex UI.
 | |
|    * @returns The handler for the pokedex UI.
 | |
|    */
 | |
|   async function runToPokedexPage(
 | |
|     species: PokemonSpecies,
 | |
|     starterAttributes: StarterAttributes = {},
 | |
|   ): Promise<PokedexPageUiHandler> {
 | |
|     // Open the pokedex UI.
 | |
|     await game.runToTitle();
 | |
| 
 | |
|     await game.phaseInterceptor.setOverlayMode(UiMode.POKEDEX_PAGE, species, starterAttributes);
 | |
| 
 | |
|     // Get the handler for the current UI.
 | |
|     const handler = game.scene.ui.getHandler();
 | |
|     expect(handler).toBeInstanceOf(PokedexPageUiHandler);
 | |
| 
 | |
|     return handler as PokedexPageUiHandler;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Compute a set of pokemon that have a specific ability in allAbilities
 | |
|    * @param ability - The ability to filter for
 | |
|    */
 | |
|   function getSpeciesWithAbility(ability: AbilityId): Set<SpeciesId> {
 | |
|     const speciesSet = new Set<SpeciesId>();
 | |
|     for (const pkmn of allSpecies) {
 | |
|       if (
 | |
|         [pkmn.ability1, pkmn.ability2, pkmn.getPassiveAbility(), pkmn.abilityHidden].includes(ability) ||
 | |
|         pkmn.forms.some(form =>
 | |
|           [form.ability1, form.ability2, form.abilityHidden, form.getPassiveAbility()].includes(ability),
 | |
|         )
 | |
|       ) {
 | |
|         speciesSet.add(pkmn.speciesId);
 | |
|       }
 | |
|     }
 | |
|     return speciesSet;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Compute a set of pokemon that have one of the specified type(s)
 | |
|    *
 | |
|    * Includes all forms of the pokemon
 | |
|    * @param types - The types to filter for
 | |
|    */
 | |
|   function getSpeciesWithType(...types: PokemonType[]): Set<SpeciesId> {
 | |
|     const speciesSet = new Set<SpeciesId>();
 | |
|     const tySet = new Set<PokemonType>(types);
 | |
| 
 | |
|     // get the pokemon and its forms
 | |
|     outer: for (const pkmn of allSpecies) {
 | |
|       // @ts-expect-error We know that type2 might be null.
 | |
|       if (tySet.has(pkmn.type1) || tySet.has(pkmn.type2)) {
 | |
|         speciesSet.add(pkmn.speciesId);
 | |
|         continue;
 | |
|       }
 | |
|       for (const form of pkmn.forms) {
 | |
|         // @ts-expect-error We know that type2 might be null.
 | |
|         if (tySet.has(form.type1) || tySet.has(form.type2)) {
 | |
|           speciesSet.add(pkmn.speciesId);
 | |
|           continue outer;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     return speciesSet;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Create mocks for the abilities of a species.
 | |
|    * This is used to set the abilities of a species to a specific value.
 | |
|    * All abilities are optional. Not providing one will set it to NONE.
 | |
|    *
 | |
|    * This will override the ability of the pokemon species only, unless set forms is true
 | |
|    *
 | |
|    * @param species - The species to set the abilities for
 | |
|    * @param ability - The ability to set for the first ability
 | |
|    * @param ability2 - The ability to set for the second ability
 | |
|    * @param hidden - The ability to set for the hidden ability
 | |
|    * @param passive - The ability to set for the passive ability
 | |
|    * @param setForms - Whether to also overwrite the abilities for each of the species' forms (defaults to `true`)
 | |
|    */
 | |
|   function createAbilityMocks(
 | |
|     species: SpeciesId,
 | |
|     {
 | |
|       ability = AbilityId.NONE,
 | |
|       ability2 = AbilityId.NONE,
 | |
|       hidden = AbilityId.NONE,
 | |
|       passive = AbilityId.NONE,
 | |
|       setForms = true,
 | |
|     }: {
 | |
|       ability?: AbilityId;
 | |
|       ability2?: AbilityId;
 | |
|       hidden?: AbilityId;
 | |
|       passive?: AbilityId;
 | |
|       setForms?: boolean;
 | |
|     },
 | |
|   ) {
 | |
|     const pokemon = getPokemonSpecies(species);
 | |
|     const checks: [PokemonSpecies | PokemonForm] = [pokemon];
 | |
|     if (setForms) {
 | |
|       checks.push(...pokemon.forms);
 | |
|     }
 | |
|     for (const p of checks) {
 | |
|       mocks.push(vi.spyOn(p, "ability1", "get").mockReturnValue(ability));
 | |
|       mocks.push(vi.spyOn(p, "ability2", "get").mockReturnValue(ability2));
 | |
|       mocks.push(vi.spyOn(p, "abilityHidden", "get").mockReturnValue(hidden));
 | |
|       mocks.push(vi.spyOn(p, "getPassiveAbility").mockReturnValue(passive));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /***************************
 | |
|    *    Tests for Filters    *
 | |
|    ***************************/
 | |
| 
 | |
|   it("should filter to show only the pokemon with an ability when filtering by ability", async () => {
 | |
|     // await game.importData("test/testUtils/saves/everything.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // Get name of overgrow
 | |
|     const overgrow = allAbilities[AbilityId.OVERGROW].name;
 | |
| 
 | |
|     // @ts-expect-error `filterText` is private
 | |
|     pokedexHandler.filterText.setValue(FilterTextRow.ABILITY_1, overgrow);
 | |
| 
 | |
|     // filter all species to be the pokemon that have overgrow
 | |
|     const overgrowSpecies = getSpeciesWithAbility(AbilityId.OVERGROW);
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     const filteredSpecies = new Set(pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId));
 | |
| 
 | |
|     expect(filteredSpecies).toEqual(overgrowSpecies);
 | |
|   });
 | |
| 
 | |
|   it("should filter to show only pokemon with ability and passive when filtering by 2 abilities", async () => {
 | |
|     // Setup mocks for the ability and passive combinations
 | |
|     const whitelist: SpeciesId[] = [];
 | |
|     const blacklist: SpeciesId[] = [];
 | |
| 
 | |
|     const filter_ab1 = AbilityId.OVERGROW;
 | |
|     const filter_ab2 = AbilityId.ADAPTABILITY;
 | |
|     const ab1_instance = allAbilities[filter_ab1];
 | |
|     const ab2_instance = allAbilities[filter_ab2];
 | |
| 
 | |
|     // Create a species with passive set and each "ability" field
 | |
|     const baseObj = {
 | |
|       ability: AbilityId.BALL_FETCH,
 | |
|       ability2: AbilityId.NONE,
 | |
|       hidden: AbilityId.BLAZE,
 | |
|       passive: AbilityId.TORRENT,
 | |
|     };
 | |
| 
 | |
|     // Mock pokemon to have the exhaustive combination of the two selected abilities
 | |
|     const attrs: (keyof typeof baseObj)[] = ["ability", "ability2", "hidden", "passive"];
 | |
|     for (const [idx, value] of permutations(attrs, 2).entries()) {
 | |
|       createAbilityMocks(SpeciesId.BULBASAUR + idx, {
 | |
|         ...baseObj,
 | |
|         [value[0]]: filter_ab1,
 | |
|         [value[1]]: filter_ab2,
 | |
|       });
 | |
|       if (value.includes("passive")) {
 | |
|         whitelist.push(SpeciesId.BULBASAUR + idx);
 | |
|       } else {
 | |
|         blacklist.push(SpeciesId.BULBASAUR + idx);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error `filterText` is private
 | |
|     pokedexHandler.filterText.setValue(FilterTextRow.ABILITY_1, ab1_instance.name);
 | |
|     // @ts-expect-error `filterText` is private
 | |
|     pokedexHandler.filterText.setValue(FilterTextRow.ABILITY_2, ab2_instance.name);
 | |
| 
 | |
|     let whiteListCount = 0;
 | |
|     // @ts-expect-error `filteredPokemonData` is private
 | |
|     for (const species of pokedexHandler.filteredPokemonData) {
 | |
|       expect(blacklist, "entry must have one of the abilities as a passive").not.toContain(species.species.speciesId);
 | |
| 
 | |
|       const rawAbility = [species.species.ability1, species.species.ability2, species.species.abilityHidden];
 | |
|       const rawPassive = species.species.getPassiveAbility();
 | |
| 
 | |
|       const c1 = rawPassive === ab1_instance.id && rawAbility.includes(ab2_instance.id);
 | |
|       const c2 = c1 || (rawPassive === ab2_instance.id && rawAbility.includes(ab1_instance.id));
 | |
| 
 | |
|       expect(c2, "each filtered entry should have the ability and passive combination").toBe(true);
 | |
|       if (whitelist.includes(species.species.speciesId)) {
 | |
|         whiteListCount++;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     expect(whiteListCount).toBe(whitelist.length);
 | |
|   });
 | |
| 
 | |
|   it("should filter to show only the pokemon with a type when filtering by a single type", async () => {
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     pokedexHandler.filterBar.getFilter(DropDownColumn.TYPES).toggleOptionState(PokemonType.NORMAL + 1);
 | |
| 
 | |
|     const expectedPokemon = getSpeciesWithType(PokemonType.NORMAL);
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     const filteredPokemon = new Set(pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId));
 | |
| 
 | |
|     expect(filteredPokemon).toEqual(expectedPokemon);
 | |
|   });
 | |
| 
 | |
|   // Todo: Pokemon with a mega that adds a type do not show up in the filter, e.g. pinsir.
 | |
|   it.todo("should show only the pokemon with one of the types when filtering by multiple types", async () => {
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     pokedexHandler.filterBar.getFilter(DropDownColumn.TYPES).toggleOptionState(PokemonType.NORMAL + 1);
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     pokedexHandler.filterBar.getFilter(DropDownColumn.TYPES).toggleOptionState(PokemonType.FLYING + 1);
 | |
| 
 | |
|     const expectedPokemon = getSpeciesWithType(PokemonType.NORMAL, PokemonType.FLYING);
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     const filteredPokemon = new Set(pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId));
 | |
| 
 | |
|     expect(filteredPokemon).toEqual(expectedPokemon);
 | |
|   });
 | |
| 
 | |
|   it("filtering for unlockable cost reduction only shows species with sufficient candies", async () => {
 | |
|     // load the save file
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.UNLOCKS);
 | |
| 
 | |
|     // Cycling 4 times to get to the "can unlock" for cost reduction
 | |
|     for (let i = 0; i < 4; i++) {
 | |
|       // index 1 is the cost reduction
 | |
|       filter.toggleOptionState(1);
 | |
|     }
 | |
| 
 | |
|     const expectedPokemon = new Set([
 | |
|       SpeciesId.CHIKORITA,
 | |
|       SpeciesId.CYNDAQUIL,
 | |
|       SpeciesId.TORCHIC,
 | |
|       SpeciesId.TURTWIG,
 | |
|       SpeciesId.EKANS,
 | |
|       SpeciesId.MUDKIP,
 | |
|     ]);
 | |
|     expect(
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       pokedexHandler.filteredPokemonData.every(pokemon =>
 | |
|         expectedPokemon.has(pokedexHandler.getStarterSpeciesId(pokemon.species.speciesId)),
 | |
|       ),
 | |
|     ).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it("filtering by passive unlocked only shows species that have their passive", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.UNLOCKS);
 | |
| 
 | |
|     filter.toggleOptionState(0); // cycle to Passive: Yes
 | |
| 
 | |
|     expect(
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       pokedexHandler.filteredPokemonData.every(
 | |
|         pokemon => pokedexHandler.getStarterSpeciesId(pokemon.species.speciesId) === SpeciesId.MUDKIP,
 | |
|       ),
 | |
|     ).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it("filtering for pokemon that can unlock passive shows only species with sufficient candies", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.UNLOCKS);
 | |
| 
 | |
|     // Cycling 4 times to get to the "can unlock" for passive
 | |
|     const expectedPokemon = new Set([
 | |
|       SpeciesId.EKANS,
 | |
|       SpeciesId.CHIKORITA,
 | |
|       SpeciesId.CYNDAQUIL,
 | |
|       SpeciesId.TORCHIC,
 | |
|       SpeciesId.TURTWIG,
 | |
|     ]);
 | |
| 
 | |
|     // cycling twice to get to the "can unlock" for passive
 | |
|     filter.toggleOptionState(0);
 | |
|     filter.toggleOptionState(0);
 | |
| 
 | |
|     expect(
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       pokedexHandler.filteredPokemonData.every(pokemon =>
 | |
|         expectedPokemon.has(pokedexHandler.getStarterSpeciesId(pokemon.species.speciesId)),
 | |
|       ),
 | |
|     ).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it("filtering for pokemon that have any cost reduction shows only the species that have unlocked a cost reduction", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     const expectedPokemon = new Set([SpeciesId.TREECKO, SpeciesId.CYNDAQUIL, SpeciesId.TOTODILE]);
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.UNLOCKS);
 | |
|     // Cycle 1 time for cost reduction
 | |
|     filter.toggleOptionState(1);
 | |
| 
 | |
|     expect(
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       pokedexHandler.filteredPokemonData.every(pokemon =>
 | |
|         expectedPokemon.has(pokedexHandler.getStarterSpeciesId(pokemon.species.speciesId)),
 | |
|       ),
 | |
|     ).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it("filtering for pokemon that have a single cost reduction shows only the species that have unlocked a single cost reduction", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     const expectedPokemon = new Set([SpeciesId.CYNDAQUIL, SpeciesId.TOTODILE]);
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.UNLOCKS);
 | |
|     // Cycle 2 times for one cost reduction
 | |
|     filter.toggleOptionState(1);
 | |
|     filter.toggleOptionState(1);
 | |
| 
 | |
|     expect(
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       pokedexHandler.filteredPokemonData.every(pokemon =>
 | |
|         expectedPokemon.has(pokedexHandler.getStarterSpeciesId(pokemon.species.speciesId)),
 | |
|       ),
 | |
|     ).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it("filtering for pokemon that have two cost reductions sorts only shows the species that have unlocked both cost reductions", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.UNLOCKS);
 | |
|     // Cycle 3 time for two cost reductions
 | |
|     filter.toggleOptionState(1);
 | |
|     filter.toggleOptionState(1);
 | |
|     filter.toggleOptionState(1);
 | |
| 
 | |
|     expect(
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       pokedexHandler.filteredPokemonData.every(
 | |
|         pokemon => pokedexHandler.getStarterSpeciesId(pokemon.species.speciesId) === SpeciesId.TREECKO,
 | |
|       ),
 | |
|     ).toBe(true);
 | |
|   });
 | |
| 
 | |
|   it("filtering by shiny status shows the caught pokemon with the selected shiny tier", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests.prsv");
 | |
|     const pokedexHandler = await runToOpenPokedex();
 | |
|     // @ts-expect-error - `filterBar` is private
 | |
|     const filter = pokedexHandler.filterBar.getFilter(DropDownColumn.CAUGHT);
 | |
|     filter.toggleOptionState(3);
 | |
| 
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     let filteredPokemon = pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId);
 | |
| 
 | |
|     // Red shiny
 | |
|     expect(filteredPokemon.length).toBe(1);
 | |
|     expect(filteredPokemon[0], "tier 1 shiny").toBe(SpeciesId.CATERPIE);
 | |
| 
 | |
|     // tier 2 shiny
 | |
|     filter.toggleOptionState(3);
 | |
|     filter.toggleOptionState(2);
 | |
| 
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     filteredPokemon = pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId);
 | |
|     expect(filteredPokemon.length).toBe(1);
 | |
|     expect(filteredPokemon[0], "tier 2 shiny").toBe(SpeciesId.RATTATA);
 | |
| 
 | |
|     filter.toggleOptionState(2);
 | |
|     filter.toggleOptionState(1);
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     filteredPokemon = pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId);
 | |
|     expect(filteredPokemon.length).toBe(1);
 | |
|     expect(filteredPokemon[0], "tier 3 shiny").toBe(SpeciesId.EKANS);
 | |
| 
 | |
|     // filter by no shiny
 | |
|     filter.toggleOptionState(1);
 | |
|     filter.toggleOptionState(4);
 | |
| 
 | |
|     // @ts-expect-error - `filteredPokemonData` is private
 | |
|     filteredPokemon = pokedexHandler.filteredPokemonData.map(pokemon => pokemon.species.speciesId);
 | |
|     expect(filteredPokemon.length).toBe(27);
 | |
|     expect(filteredPokemon, "not shiny").not.toContain(SpeciesId.CATERPIE);
 | |
|     expect(filteredPokemon, "not shiny").not.toContain(SpeciesId.RATTATA);
 | |
|     expect(filteredPokemon, "not shiny").not.toContain(SpeciesId.EKANS);
 | |
|   });
 | |
| 
 | |
|   /****************************
 | |
|    *    Tests for UI Input    *
 | |
|    ****************************/
 | |
| 
 | |
|   // TODO: fix cursor wrapping
 | |
|   it.todo(
 | |
|     "should wrap the cursor to the top when moving to an empty entry when there are more than 81 pokemon",
 | |
|     async () => {
 | |
|       const pokedexHandler = await runToOpenPokedex();
 | |
| 
 | |
|       // Filter by gen 2 so we can pan a specific amount.
 | |
|       // @ts-expect-error `filterBar` is private
 | |
|       pokedexHandler.filterBar.getFilter(DropDownColumn.GEN).options[2].toggleOptionState();
 | |
|       pokedexHandler.updateStarters();
 | |
|       // @ts-expect-error - `filteredPokemonData` is private
 | |
|       expect(pokedexHandler.filteredPokemonData.length, "pokemon in gen2").toBe(100);
 | |
| 
 | |
|       // Let's try to pan to the right to see what the pokemon it points to is.
 | |
| 
 | |
|       // pan to the right once and down 11 times
 | |
|       pokedexHandler.processInput(Button.RIGHT);
 | |
|       // Nab the pokemon that is selected for comparison later.
 | |
| 
 | |
|       // @ts-expect-error - `lastSpecies` is private
 | |
|       const selectedPokemon = pokedexHandler.lastSpecies.speciesId;
 | |
|       for (let i = 0; i < 11; i++) {
 | |
|         pokedexHandler.processInput(Button.DOWN);
 | |
|       }
 | |
| 
 | |
|       // @ts-expect-error `lastSpecies` is private
 | |
|       expect(selectedPokemon).toEqual(pokedexHandler.lastSpecies.speciesId);
 | |
|     },
 | |
|   );
 | |
| 
 | |
|   /****************************
 | |
|    *    Tests for Pokédex Pages    *
 | |
|    ****************************/
 | |
| 
 | |
|   it("should show caught battle form as caught", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests_v2.prsv");
 | |
|     const pageHandler = await runToPokedexPage(getPokemonSpecies(SpeciesId.VENUSAUR), { form: 1 });
 | |
| 
 | |
|     // @ts-expect-error - `species` is private
 | |
|     expect(pageHandler.species.speciesId).toEqual(SpeciesId.VENUSAUR);
 | |
| 
 | |
|     // @ts-expect-error - `formIndex` is private
 | |
|     expect(pageHandler.formIndex).toEqual(1);
 | |
| 
 | |
|     expect(pageHandler.isFormCaught()).toEqual(true);
 | |
|     expect(pageHandler.isSeen()).toEqual(true);
 | |
|   });
 | |
| 
 | |
|   //TODO: check tint of the sprite
 | |
|   it("should show uncaught battle form as seen", async () => {
 | |
|     await game.importData("./test/testUtils/saves/data_pokedex_tests_v2.prsv");
 | |
|     const pageHandler = await runToPokedexPage(getPokemonSpecies(SpeciesId.VENUSAUR), { form: 2 });
 | |
| 
 | |
|     // @ts-expect-error - `species` is private
 | |
|     expect(pageHandler.species.speciesId).toEqual(SpeciesId.VENUSAUR);
 | |
| 
 | |
|     // @ts-expect-error - `formIndex` is private
 | |
|     expect(pageHandler.formIndex).toEqual(2);
 | |
| 
 | |
|     expect(pageHandler.isFormCaught()).toEqual(false);
 | |
|     expect(pageHandler.isSeen()).toEqual(true);
 | |
|   });
 | |
| });
 |