2023-04-28 12:03:42 -07:00
|
|
|
import BattleScene, { PokeballCounts } from "../battle-scene";
|
2023-04-20 12:46:05 -07:00
|
|
|
import { Gender } from "../data/gender";
|
2023-04-28 12:03:42 -07:00
|
|
|
import Pokemon, { EnemyPokemon, PlayerPokemon } from "../pokemon";
|
2023-04-20 12:46:05 -07:00
|
|
|
import { pokemonPrevolutions } from "../data/pokemon-evolutions";
|
2023-04-26 09:50:21 -07:00
|
|
|
import PokemonSpecies, { allSpecies, getPokemonSpecies } from "../data/pokemon-species";
|
2023-04-20 12:46:05 -07:00
|
|
|
import { Species } from "../data/species";
|
|
|
|
import * as Utils from "../utils";
|
2023-04-28 12:03:42 -07:00
|
|
|
import PokemonData from "./pokemon-data";
|
|
|
|
import PersistentModifierData from "./modifier-data";
|
|
|
|
import { PokemonHeldItemModifier } from "../modifier/modifier";
|
|
|
|
import ArenaData from "./arena-data";
|
2023-04-28 22:40:24 -07:00
|
|
|
import { Unlockables } from "./unlockables";
|
2023-05-31 16:54:57 -07:00
|
|
|
import { GameMode } from "../game-mode";
|
2023-10-07 13:08:33 -07:00
|
|
|
import { BattleType } from "../battle";
|
|
|
|
import TrainerData from "./trainer-data";
|
|
|
|
import { trainerConfigs } from "../data/trainer-type";
|
2023-10-26 13:33:59 -07:00
|
|
|
import { Setting, setSetting, settingDefaults } from "./settings";
|
2023-04-17 19:44:41 -07:00
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
interface SystemSaveData {
|
2023-04-17 22:32:26 -07:00
|
|
|
trainerId: integer;
|
|
|
|
secretId: integer;
|
|
|
|
dexData: DexData;
|
2023-04-28 22:40:24 -07:00
|
|
|
unlocks: Unlocks;
|
2023-04-28 12:03:42 -07:00
|
|
|
timestamp: integer;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface SessionSaveData {
|
2023-10-18 15:01:15 -07:00
|
|
|
seed: string;
|
2023-05-31 16:54:57 -07:00
|
|
|
gameMode: GameMode;
|
2023-04-28 12:03:42 -07:00
|
|
|
party: PokemonData[];
|
2023-10-07 13:08:33 -07:00
|
|
|
enemyParty: PokemonData[];
|
2023-05-18 08:11:06 -07:00
|
|
|
enemyField: PokemonData[];
|
2023-04-28 12:03:42 -07:00
|
|
|
modifiers: PersistentModifierData[];
|
|
|
|
enemyModifiers: PersistentModifierData[];
|
|
|
|
arena: ArenaData;
|
|
|
|
pokeballCounts: PokeballCounts;
|
|
|
|
waveIndex: integer;
|
2023-10-07 13:08:33 -07:00
|
|
|
battleType: BattleType;
|
|
|
|
trainer: TrainerData;
|
2023-04-28 12:03:42 -07:00
|
|
|
timestamp: integer;
|
2023-04-17 22:32:26 -07:00
|
|
|
}
|
|
|
|
|
2023-04-28 22:40:24 -07:00
|
|
|
interface Unlocks {
|
|
|
|
[key: integer]: boolean;
|
|
|
|
}
|
|
|
|
|
2023-04-17 19:44:41 -07:00
|
|
|
export interface DexData {
|
|
|
|
[key: integer]: DexData | DexEntry
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface DexEntry {
|
|
|
|
seen: boolean;
|
|
|
|
caught: boolean;
|
|
|
|
}
|
|
|
|
|
2023-04-17 22:32:26 -07:00
|
|
|
export interface DexEntryDetails {
|
2023-04-17 19:44:41 -07:00
|
|
|
shiny: boolean;
|
|
|
|
formIndex: integer;
|
|
|
|
female: boolean;
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: integer;
|
2023-04-17 19:44:41 -07:00
|
|
|
entry: DexEntry;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface StarterDexUnlockTree {
|
|
|
|
shiny: boolean | Map<boolean, StarterDexUnlockTree>
|
|
|
|
formIndex: integer | Map<integer, StarterDexUnlockTree>
|
|
|
|
female: boolean | Map<boolean, StarterDexUnlockTree>
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: integer | Map<integer, StarterDexUnlockTree>
|
2023-04-17 19:44:41 -07:00
|
|
|
key: string,
|
|
|
|
entry: DexEntry
|
|
|
|
}
|
|
|
|
|
2023-04-17 22:32:26 -07:00
|
|
|
export class GameData {
|
2023-04-17 19:44:41 -07:00
|
|
|
private scene: BattleScene;
|
|
|
|
|
|
|
|
public trainerId: integer;
|
|
|
|
public secretId: integer;
|
|
|
|
|
|
|
|
public dexData: DexData;
|
|
|
|
|
2023-04-28 22:40:24 -07:00
|
|
|
public unlocks: Unlocks;
|
|
|
|
|
2023-04-17 19:44:41 -07:00
|
|
|
constructor(scene: BattleScene) {
|
|
|
|
this.scene = scene;
|
2023-10-26 13:33:59 -07:00
|
|
|
this.loadSettings();
|
|
|
|
this.trainerId = Utils.randSeedInt(65536);
|
|
|
|
this.secretId = Utils.randSeedInt(65536);
|
2023-04-28 22:40:24 -07:00
|
|
|
this.unlocks = {
|
2023-05-31 16:54:57 -07:00
|
|
|
[Unlockables.ENDLESS_MODE]: false,
|
2023-11-04 16:46:48 -07:00
|
|
|
[Unlockables.MINI_BLACK_HOLE]: false,
|
|
|
|
[Unlockables.SPLICED_ENDLESS_MODE]: false
|
2023-04-28 22:40:24 -07:00
|
|
|
};
|
2023-04-26 13:07:29 -07:00
|
|
|
this.initDexData();
|
2023-04-28 12:03:42 -07:00
|
|
|
this.loadSystem();
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
2023-04-28 22:40:24 -07:00
|
|
|
public saveSystem(): boolean {
|
2023-04-19 11:07:38 -07:00
|
|
|
if (this.scene.quickStart)
|
|
|
|
return false;
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
const data: SystemSaveData = {
|
2023-04-17 22:32:26 -07:00
|
|
|
trainerId: this.trainerId,
|
|
|
|
secretId: this.secretId,
|
2023-04-26 09:50:21 -07:00
|
|
|
dexData: this.dexData,
|
2023-04-28 22:40:24 -07:00
|
|
|
unlocks: this.unlocks,
|
2023-04-26 09:50:21 -07:00
|
|
|
timestamp: new Date().getTime()
|
2023-04-17 22:32:26 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
localStorage.setItem('data', btoa(JSON.stringify(data)));
|
2023-04-17 19:44:41 -07:00
|
|
|
|
2023-04-17 22:32:26 -07:00
|
|
|
return true;
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
private loadSystem(): boolean {
|
2023-10-26 13:33:59 -07:00
|
|
|
if (!localStorage.hasOwnProperty('data'))
|
2023-04-17 22:32:26 -07:00
|
|
|
return false;
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
const data = JSON.parse(atob(localStorage.getItem('data'))) as SystemSaveData;
|
|
|
|
console.debug(data);
|
2023-04-17 22:32:26 -07:00
|
|
|
|
|
|
|
this.trainerId = data.trainerId;
|
|
|
|
this.secretId = data.secretId;
|
2023-04-17 19:44:41 -07:00
|
|
|
|
2023-04-28 22:40:24 -07:00
|
|
|
if (data.unlocks) {
|
|
|
|
for (let key of Object.keys(data.unlocks)) {
|
|
|
|
if (this.unlocks.hasOwnProperty(key))
|
|
|
|
this.unlocks[key] = data.unlocks[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-26 09:50:21 -07:00
|
|
|
if (data.timestamp === undefined)
|
|
|
|
this.convertDexData(data.dexData);
|
|
|
|
|
2023-04-26 13:07:29 -07:00
|
|
|
this.dexData = Object.assign(this.dexData, data.dexData);
|
|
|
|
|
2023-04-17 22:32:26 -07:00
|
|
|
return true;
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
2023-10-26 13:33:59 -07:00
|
|
|
public saveSetting(setting: Setting, valueIndex: integer): boolean {
|
|
|
|
let settings: object = {};
|
|
|
|
if (localStorage.hasOwnProperty('settings'))
|
|
|
|
settings = JSON.parse(localStorage.getItem('settings'));
|
|
|
|
|
|
|
|
setSetting(this.scene, setting as Setting, valueIndex);
|
|
|
|
|
|
|
|
Object.keys(settingDefaults).forEach(s => {
|
|
|
|
if (s === setting)
|
|
|
|
settings[s] = valueIndex;
|
|
|
|
});
|
|
|
|
|
|
|
|
localStorage.setItem('settings', JSON.stringify(settings));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private loadSettings(): boolean {
|
|
|
|
if (!localStorage.hasOwnProperty('settings'))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const settings = JSON.parse(localStorage.getItem('settings'));
|
|
|
|
|
|
|
|
for (let setting of Object.keys(settings))
|
|
|
|
setSetting(this.scene, setting as Setting, settings[setting]);
|
|
|
|
}
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
saveSession(scene: BattleScene): boolean {
|
|
|
|
const sessionData = {
|
2023-10-18 15:01:15 -07:00
|
|
|
seed: scene.seed,
|
2023-05-31 16:54:57 -07:00
|
|
|
gameMode: scene.gameMode,
|
2023-04-28 12:03:42 -07:00
|
|
|
party: scene.getParty().map(p => new PokemonData(p)),
|
2023-10-07 13:08:33 -07:00
|
|
|
enemyParty: scene.getEnemyParty().map(p => new PokemonData(p)),
|
2023-05-18 08:11:06 -07:00
|
|
|
modifiers: scene.findModifiers(() => true).map(m => new PersistentModifierData(m, true)),
|
|
|
|
enemyModifiers: scene.findModifiers(() => true, false).map(m => new PersistentModifierData(m, false)),
|
2023-04-28 12:03:42 -07:00
|
|
|
arena: new ArenaData(scene.arena),
|
|
|
|
pokeballCounts: scene.pokeballCounts,
|
|
|
|
waveIndex: scene.currentBattle.waveIndex,
|
2023-10-07 13:08:33 -07:00
|
|
|
battleType: scene.currentBattle.battleType,
|
|
|
|
trainer: scene.currentBattle.battleType == BattleType.TRAINER ? new TrainerData(scene.currentBattle.trainer) : null,
|
2023-04-28 12:03:42 -07:00
|
|
|
timestamp: new Date().getTime()
|
|
|
|
} as SessionSaveData;
|
|
|
|
|
|
|
|
localStorage.setItem('sessionData', btoa(JSON.stringify(sessionData)));
|
|
|
|
|
|
|
|
console.debug('Session data saved');
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hasSession() {
|
|
|
|
return !!localStorage.getItem('sessionData');
|
|
|
|
}
|
|
|
|
|
|
|
|
loadSession(scene: BattleScene): Promise<boolean> {
|
|
|
|
return new Promise(async (resolve, reject) => {
|
|
|
|
if (!this.hasSession())
|
|
|
|
return resolve(false);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const sessionData = JSON.parse(atob(localStorage.getItem('sessionData')), (k: string, v: any) => {
|
2023-10-07 13:08:33 -07:00
|
|
|
if (k === 'party' || k === 'enemyParty' || k === 'enemyField') {
|
2023-04-28 12:03:42 -07:00
|
|
|
const ret: PokemonData[] = [];
|
|
|
|
for (let pd of v)
|
|
|
|
ret.push(new PokemonData(pd));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-10-07 13:08:33 -07:00
|
|
|
if (k === 'trainer')
|
|
|
|
return v ? new TrainerData(v) : null;
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
if (k === 'modifiers' || k === 'enemyModifiers') {
|
|
|
|
const player = k === 'modifiers';
|
|
|
|
const ret: PersistentModifierData[] = [];
|
|
|
|
for (let md of v)
|
|
|
|
ret.push(new PersistentModifierData(md, player));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (k === 'arena')
|
|
|
|
return new ArenaData(v);
|
|
|
|
|
|
|
|
return v;
|
|
|
|
}) as SessionSaveData;
|
|
|
|
|
|
|
|
console.debug(sessionData);
|
|
|
|
|
2023-10-18 15:01:15 -07:00
|
|
|
scene.seed = sessionData.seed || this.scene.game.config.seed[0];
|
|
|
|
scene.resetSeed();
|
|
|
|
|
2023-05-31 16:54:57 -07:00
|
|
|
scene.gameMode = sessionData.gameMode || GameMode.CLASSIC;
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
const loadPokemonAssets: Promise<void>[] = [];
|
|
|
|
|
|
|
|
const party = scene.getParty();
|
|
|
|
party.splice(0, party.length);
|
|
|
|
|
|
|
|
for (let p of sessionData.party) {
|
|
|
|
const pokemon = p.toPokemon(scene) as PlayerPokemon;
|
|
|
|
pokemon.setVisible(false);
|
|
|
|
loadPokemonAssets.push(pokemon.loadAssets());
|
|
|
|
party.push(pokemon);
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.keys(scene.pokeballCounts).forEach((key: string) => {
|
|
|
|
scene.pokeballCounts[key] = sessionData.pokeballCounts[key] || 0;
|
|
|
|
});
|
|
|
|
|
2023-10-07 13:08:33 -07:00
|
|
|
// TODO: Remove this
|
|
|
|
if (sessionData.enemyField)
|
|
|
|
sessionData.enemyParty = sessionData.enemyField;
|
|
|
|
|
|
|
|
const battleType = sessionData.battleType || 0;
|
|
|
|
const battle = scene.newBattle(sessionData.waveIndex, battleType, sessionData.trainer, battleType === BattleType.TRAINER ? trainerConfigs[sessionData.trainer.trainerType].isDouble : sessionData.enemyParty.length > 1);
|
|
|
|
|
2023-10-18 15:01:15 -07:00
|
|
|
scene.newArena(sessionData.arena.biome, true);
|
|
|
|
|
2023-10-07 13:08:33 -07:00
|
|
|
sessionData.enemyParty.forEach((enemyData, e) => {
|
2023-10-28 07:51:34 -07:00
|
|
|
const enemyPokemon = enemyData.toPokemon(scene, battleType) as EnemyPokemon;
|
2023-10-07 13:08:33 -07:00
|
|
|
battle.enemyParty[e] = enemyPokemon;
|
2023-10-23 10:48:56 -07:00
|
|
|
if (battleType === BattleType.WILD)
|
|
|
|
battle.seenEnemyPartyMemberIds.add(enemyPokemon.id);
|
2023-05-18 08:11:06 -07:00
|
|
|
|
|
|
|
loadPokemonAssets.push(enemyPokemon.loadAssets());
|
|
|
|
});
|
2023-04-28 12:03:42 -07:00
|
|
|
|
|
|
|
scene.arena.weather = sessionData.arena.weather;
|
|
|
|
// TODO
|
|
|
|
//scene.arena.tags = sessionData.arena.tags;
|
|
|
|
|
|
|
|
const modifiersModule = await import('../modifier/modifier');
|
|
|
|
|
|
|
|
for (let modifierData of sessionData.modifiers) {
|
|
|
|
const modifier = modifierData.toModifier(scene, modifiersModule[modifierData.className]);
|
|
|
|
if (modifier)
|
2023-10-31 18:43:22 -07:00
|
|
|
scene.addModifier(modifier, true);
|
2023-04-28 12:03:42 -07:00
|
|
|
}
|
|
|
|
|
2023-10-31 18:43:22 -07:00
|
|
|
scene.updateModifiers(true);
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
for (let enemyModifierData of sessionData.enemyModifiers) {
|
2023-10-28 10:24:57 -07:00
|
|
|
const modifier = enemyModifierData.toModifier(scene, modifiersModule[enemyModifierData.className]);
|
2023-04-28 12:03:42 -07:00
|
|
|
if (modifier)
|
2023-10-31 18:43:22 -07:00
|
|
|
scene.addEnemyModifier(modifier, true);
|
2023-04-28 12:03:42 -07:00
|
|
|
}
|
|
|
|
|
2023-10-31 18:43:22 -07:00
|
|
|
scene.updateModifiers(false);
|
|
|
|
|
2023-04-28 12:03:42 -07:00
|
|
|
Promise.all(loadPokemonAssets).then(() => resolve(true));
|
|
|
|
} catch (err) {
|
|
|
|
reject(err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
clearSession(): void {
|
|
|
|
localStorage.removeItem('sessionData');
|
|
|
|
}
|
|
|
|
|
|
|
|
private initDexData(): void {
|
2023-04-17 19:44:41 -07:00
|
|
|
const data: DexData = {};
|
|
|
|
|
|
|
|
const initDexSubData = (dexData: DexData, count: integer): DexData[] => {
|
|
|
|
const ret: DexData[] = [];
|
|
|
|
for (let i = 0; i < count; i++) {
|
|
|
|
const newData: DexData = {};
|
|
|
|
dexData[i] = newData;
|
|
|
|
ret.push(newData);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
};
|
|
|
|
|
|
|
|
const initDexEntries = (dexData: DexData, count: integer): DexEntry[] => {
|
|
|
|
const ret: DexEntry[] = [];
|
|
|
|
for (let i = 0; i < count; i++) {
|
2023-04-17 22:32:26 -07:00
|
|
|
const entry: DexEntry = { seen: false, caught: false };
|
2023-04-17 19:44:41 -07:00
|
|
|
dexData[i] = entry;
|
|
|
|
ret.push(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
};
|
|
|
|
|
|
|
|
for (let species of allSpecies) {
|
|
|
|
data[species.speciesId] = {};
|
2023-04-26 09:50:21 -07:00
|
|
|
const abilityCount = species.getAbilityCount();
|
2023-04-17 19:44:41 -07:00
|
|
|
if (species.forms?.length)
|
2023-04-26 09:50:21 -07:00
|
|
|
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => species.malePercent !== null
|
|
|
|
? initDexSubData(sd, species.forms.length).map(fd => initDexSubData(fd, 2).map(gd => initDexEntries(gd, abilityCount)))
|
|
|
|
: initDexSubData(sd, species.forms.length).map(fd => initDexEntries(fd, abilityCount)));
|
2023-04-17 19:44:41 -07:00
|
|
|
else if (species.malePercent !== null)
|
2023-04-26 09:50:21 -07:00
|
|
|
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => initDexSubData(sd, 2).map(gd => initDexEntries(gd, abilityCount)));
|
2023-04-17 19:44:41 -07:00
|
|
|
else
|
2023-04-26 09:50:21 -07:00
|
|
|
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => initDexEntries(sd, abilityCount))
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
const defaultStarters: Species[] = [
|
|
|
|
Species.BULBASAUR, Species.CHARMANDER, Species.SQUIRTLE,
|
|
|
|
Species.CHIKORITA, Species.CYNDAQUIL, Species.TOTODILE,
|
|
|
|
Species.TREECKO, Species.TORCHIC, Species.MUDKIP,
|
|
|
|
Species.TURTWIG, Species.CHIMCHAR, Species.PIPLUP,
|
|
|
|
Species.SNIVY, Species.TEPIG, Species.OSHAWOTT
|
|
|
|
];
|
|
|
|
|
|
|
|
for (let ds of defaultStarters) {
|
2023-04-26 09:50:21 -07:00
|
|
|
let entry = data[ds][0][Gender.MALE][0] as DexEntry;
|
2023-04-17 19:44:41 -07:00
|
|
|
entry.seen = true;
|
|
|
|
entry.caught = true;
|
2023-04-17 22:32:26 -07:00
|
|
|
}
|
2023-04-17 19:44:41 -07:00
|
|
|
|
2023-04-17 22:32:26 -07:00
|
|
|
this.dexData = data;
|
|
|
|
}
|
|
|
|
|
|
|
|
setPokemonSeen(pokemon: Pokemon): void {
|
|
|
|
const dexEntry = this.getPokemonDexEntry(pokemon);
|
|
|
|
if (!dexEntry.seen) {
|
|
|
|
dexEntry.seen = true;
|
2023-04-28 12:03:42 -07:00
|
|
|
this.saveSystem();
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
2023-04-17 22:32:26 -07:00
|
|
|
}
|
2023-04-17 19:44:41 -07:00
|
|
|
|
2023-04-17 22:32:26 -07:00
|
|
|
setPokemonCaught(pokemon: Pokemon): Promise<void> {
|
2023-07-05 11:19:49 -07:00
|
|
|
return this.setPokemonSpeciesCaught(pokemon, pokemon.species);
|
|
|
|
}
|
|
|
|
|
|
|
|
setPokemonSpeciesCaught(pokemon: Pokemon, species: PokemonSpecies): Promise<void> {
|
|
|
|
return new Promise<void>((resolve) => {
|
2023-11-05 20:48:04 -08:00
|
|
|
const dexEntry = this.getDexEntry(species, pokemon.isShiny(), pokemon.formIndex, pokemon.gender === Gender.FEMALE, pokemon.abilityIndex);
|
2023-07-05 11:19:49 -07:00
|
|
|
const hasPrevolution = pokemonPrevolutions.hasOwnProperty(species.speciesId);
|
2023-04-17 22:32:26 -07:00
|
|
|
if (!dexEntry.caught) {
|
2023-07-05 11:19:49 -07:00
|
|
|
const newCatch = !this.getDefaultDexEntry(species);
|
2023-04-17 22:32:26 -07:00
|
|
|
|
|
|
|
dexEntry.caught = true;
|
2023-04-28 12:03:42 -07:00
|
|
|
this.saveSystem();
|
2023-04-17 22:32:26 -07:00
|
|
|
|
2023-07-05 11:19:49 -07:00
|
|
|
if (newCatch && !hasPrevolution) {
|
2023-04-17 22:32:26 -07:00
|
|
|
this.scene.playSoundWithoutBgm('level_up_fanfare', 1500);
|
2023-10-18 15:01:15 -07:00
|
|
|
this.scene.ui.showText(`${species.name} has been\nadded as a starter!`, null, () => resolve(), null, true);
|
2023-04-17 22:32:26 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-05 11:19:49 -07:00
|
|
|
if (hasPrevolution) {
|
|
|
|
const prevolutionSpecies = pokemonPrevolutions[species.speciesId];
|
|
|
|
this.setPokemonSpeciesCaught(pokemon, getPokemonSpecies(prevolutionSpecies)).then(() => resolve());
|
|
|
|
} else
|
|
|
|
resolve();
|
2023-04-17 22:32:26 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
getPokemonDexEntry(pokemon: Pokemon) {
|
2023-11-05 20:48:04 -08:00
|
|
|
return this.getDexEntry(pokemon.species, pokemon.isShiny(), pokemon.formIndex, pokemon.gender === Gender.FEMALE, pokemon.abilityIndex);
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
2023-04-26 09:50:21 -07:00
|
|
|
getDexEntry(species: PokemonSpecies, shiny: boolean, formIndex: integer, female: boolean, abilityIndex: integer): DexEntry {
|
2023-04-17 19:44:41 -07:00
|
|
|
const shinyIndex = !shiny ? 0 : 1;
|
|
|
|
const genderIndex = !female ? 0 : 1;
|
|
|
|
const data = this.dexData[species.speciesId];
|
|
|
|
if (species.forms?.length) {
|
2023-11-07 18:05:27 -08:00
|
|
|
const getEntry = () => {
|
2023-11-04 20:09:46 -07:00
|
|
|
if (species.malePercent !== null)
|
|
|
|
return data[shinyIndex][formIndex][genderIndex][abilityIndex];
|
|
|
|
return data[shinyIndex][formIndex][abilityIndex];
|
2023-11-07 18:05:27 -08:00
|
|
|
};
|
|
|
|
let entry: DexEntry;
|
|
|
|
try {
|
|
|
|
entry = getEntry();
|
|
|
|
} catch (err) { }
|
|
|
|
if (entry)
|
|
|
|
return entry;
|
|
|
|
else {
|
2023-11-04 20:09:46 -07:00
|
|
|
console.warn(`Form data not found for dex entry for ${species.name}: Restructuring dex entry`);
|
|
|
|
for (let s = 0; s < 2; s++) {
|
|
|
|
const oldData = Object.assign({}, data[s]);
|
|
|
|
data[s] = {};
|
|
|
|
for (let f = 0; f < species.forms.length; f++)
|
|
|
|
data[s][f] = oldData;
|
|
|
|
}
|
|
|
|
this.saveSystem();
|
|
|
|
}
|
2023-11-07 18:05:27 -08:00
|
|
|
return getEntry();
|
2023-04-17 19:44:41 -07:00
|
|
|
} else if (species.malePercent !== null)
|
2023-04-26 09:50:21 -07:00
|
|
|
return data[shinyIndex][genderIndex][abilityIndex];
|
|
|
|
return data[shinyIndex][abilityIndex] as DexEntry;
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
2023-04-26 09:50:21 -07:00
|
|
|
getDefaultDexEntry(species: PokemonSpecies, forceShiny?: boolean, forceFormIndex?: integer, forceFemale?: boolean, forceAbilityIndex?: integer): DexEntryDetails {
|
2023-04-17 19:44:41 -07:00
|
|
|
const hasForms = !!species.forms?.length;
|
2023-04-26 09:50:21 -07:00
|
|
|
const hasGender = species.malePercent !== null;
|
2023-04-17 19:44:41 -07:00
|
|
|
let shiny = false;
|
|
|
|
let formIndex = 0;
|
|
|
|
let female = false;
|
2023-04-26 09:50:21 -07:00
|
|
|
let abilityIndex = 0;
|
2023-04-17 19:44:41 -07:00
|
|
|
let entry = null;
|
|
|
|
|
|
|
|
const traverseData = (data: DexData, level: integer) => {
|
|
|
|
const keys = Object.keys(data);
|
2023-04-26 09:50:21 -07:00
|
|
|
if ((!hasForms && level === 1) || (!hasGender && level === 2)) {
|
|
|
|
traverseData(data, level + 1);
|
|
|
|
return;
|
|
|
|
}
|
2023-04-17 19:44:41 -07:00
|
|
|
keys.forEach((key: string, k: integer) => {
|
|
|
|
if (entry)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (level) {
|
|
|
|
case 0:
|
|
|
|
shiny = !!k;
|
|
|
|
if (forceShiny !== undefined && shiny !== forceShiny)
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
formIndex = k;
|
|
|
|
if (forceFormIndex !== undefined && formIndex !== forceFormIndex)
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
female = !!k;
|
|
|
|
if (forceFemale !== undefined && female !== forceFemale)
|
|
|
|
return
|
|
|
|
break;
|
2023-04-26 09:50:21 -07:00
|
|
|
case 3:
|
|
|
|
abilityIndex = k;
|
|
|
|
if (forceAbilityIndex !== undefined && abilityIndex !== forceAbilityIndex)
|
|
|
|
return;
|
|
|
|
break;
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if ('caught' in data[key]) {
|
|
|
|
if (data[key].caught)
|
|
|
|
entry = data[key] as DexEntry;
|
|
|
|
} else
|
2023-04-26 09:50:21 -07:00
|
|
|
traverseData(data[key] as DexData, level + 1);
|
2023-04-17 19:44:41 -07:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
traverseData(this.dexData[species.speciesId] as DexData, 0);
|
|
|
|
|
|
|
|
if (entry) {
|
2023-04-17 22:32:26 -07:00
|
|
|
return {
|
2023-04-17 19:44:41 -07:00
|
|
|
shiny: shiny,
|
|
|
|
formIndex: formIndex,
|
|
|
|
female: female,
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: abilityIndex,
|
2023-04-17 19:44:41 -07:00
|
|
|
entry: entry
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
getStarterDexUnlockTree(species: PokemonSpecies): StarterDexUnlockTree {
|
|
|
|
const hasForms = !!species.forms?.length;
|
|
|
|
const hasGender = species.malePercent !== null;
|
|
|
|
|
|
|
|
const getTreeOrValueMap = (key: string, parent?: StarterDexUnlockTree): (Map<any, any>) => {
|
|
|
|
switch (key) {
|
|
|
|
case 'shiny':
|
|
|
|
const shinyMap = new Map<boolean, StarterDexUnlockTree>();
|
|
|
|
for (let s = 0; s < 2; s++) {
|
2023-04-26 09:50:21 -07:00
|
|
|
const props = { shiny: !!s };
|
2023-04-17 19:44:41 -07:00
|
|
|
shinyMap.set(!!s, {
|
|
|
|
shiny: !!s,
|
|
|
|
formIndex: hasForms ? getTreeOrValueMap('formIndex', props as StarterDexUnlockTree) : null,
|
|
|
|
female: !hasForms && hasGender ? getTreeOrValueMap('female', props as StarterDexUnlockTree) : null,
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: !hasForms && !hasGender ? getTreeOrValueMap('abilityIndex', props as StarterDexUnlockTree) : null,
|
|
|
|
key: hasForms ? 'formIndex' : hasGender ? 'female' : 'abilityIndex',
|
|
|
|
entry: null,
|
2023-04-17 19:44:41 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return shinyMap;
|
|
|
|
case 'formIndex':
|
|
|
|
const formMap = new Map<integer, StarterDexUnlockTree>();
|
|
|
|
for (let f = 0; f < species.forms.length; f++) {
|
|
|
|
const props = { shiny: parent.shiny, formIndex: f };
|
|
|
|
formMap.set(f, {
|
|
|
|
shiny: parent.shiny,
|
|
|
|
formIndex: f,
|
|
|
|
female: hasGender ? getTreeOrValueMap('female', props as StarterDexUnlockTree) : null,
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: !hasGender ? getTreeOrValueMap('abilityIndex', props as StarterDexUnlockTree) : null,
|
|
|
|
key: hasGender ? 'female' : 'abilityIndex',
|
|
|
|
entry: null
|
2023-04-17 19:44:41 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return formMap;
|
|
|
|
case 'female':
|
|
|
|
const genderMap = new Map<boolean, StarterDexUnlockTree>();
|
|
|
|
for (let g = 0; g < 2; g++) {
|
2023-04-26 09:50:21 -07:00
|
|
|
const props = { shiny: parent.shiny, formIndex: parent.formIndex, female: !!g };
|
2023-04-17 19:44:41 -07:00
|
|
|
genderMap.set(!!g, {
|
|
|
|
shiny: parent.shiny,
|
|
|
|
formIndex: parent.formIndex,
|
|
|
|
female: !!g,
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: getTreeOrValueMap('abilityIndex', props as StarterDexUnlockTree),
|
|
|
|
key: 'abilityIndex',
|
|
|
|
entry: null
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return genderMap;
|
|
|
|
case 'abilityIndex':
|
|
|
|
const abilityMap = new Map<integer, StarterDexUnlockTree>();
|
|
|
|
const abilityCount = species.getAbilityCount();
|
|
|
|
for (let a = 0; a < abilityCount; a++) {
|
|
|
|
abilityMap.set(a, {
|
|
|
|
shiny: parent.shiny,
|
|
|
|
formIndex: parent.formIndex,
|
|
|
|
female: parent.female,
|
|
|
|
abilityIndex: a,
|
2023-04-17 19:44:41 -07:00
|
|
|
key: 'entry',
|
|
|
|
entry: hasForms
|
2023-04-26 09:50:21 -07:00
|
|
|
? hasGender
|
|
|
|
? this.dexData[species.speciesId][!parent.shiny ? 0 : 1][parent.formIndex as integer][!parent.female ? 0 : 1][a]
|
|
|
|
: this.dexData[species.speciesId][!parent.shiny ? 0 : 1][parent.formIndex as integer][a]
|
|
|
|
: hasGender
|
|
|
|
? this.dexData[species.speciesId][!parent.shiny ? 0 : 1][!parent.female ? 0 : 1][a]
|
|
|
|
: this.dexData[species.speciesId][!parent.shiny ? 0 : 1][a]
|
2023-04-17 19:44:41 -07:00
|
|
|
});
|
|
|
|
}
|
2023-04-26 09:50:21 -07:00
|
|
|
return abilityMap;
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const root = {
|
|
|
|
shiny: getTreeOrValueMap('shiny'),
|
|
|
|
formIndex: null,
|
|
|
|
female: null,
|
2023-04-26 09:50:21 -07:00
|
|
|
abilityIndex: null,
|
2023-04-17 19:44:41 -07:00
|
|
|
key: 'shiny',
|
|
|
|
entry: null
|
|
|
|
};
|
|
|
|
|
|
|
|
return root;
|
|
|
|
}
|
2023-04-26 09:50:21 -07:00
|
|
|
|
|
|
|
convertDexData(dexData: DexData): void {
|
|
|
|
const traverseData = (speciesId: Species, data: DexData) => {
|
|
|
|
const keys = Object.keys(data);
|
|
|
|
keys.forEach((key: string, k: integer) => {
|
|
|
|
if ('caught' in data[key]) {
|
|
|
|
const abilityCount = getPokemonSpecies(speciesId).getAbilityCount();
|
|
|
|
data[key] = {
|
|
|
|
0: data[key]
|
|
|
|
};
|
|
|
|
for (let a = 1; a < abilityCount; a++)
|
|
|
|
data[key][a] = { seen: false, caught: false };
|
|
|
|
} else
|
|
|
|
traverseData(speciesId, data[key]);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Object.keys(dexData).forEach((species: string, s: integer) => {
|
|
|
|
const speciesId = parseInt(species);
|
|
|
|
traverseData(speciesId, dexData[species]);
|
|
|
|
});
|
|
|
|
}
|
2023-04-17 19:44:41 -07:00
|
|
|
}
|