Add starter select ability selection
parent
bb86ab6250
commit
592ccc7c6a
Binary file not shown.
Before Width: | Height: | Size: 1.4 KiB After Width: | Height: | Size: 1.4 KiB |
|
@ -43,7 +43,7 @@ export class SelectStarterPhase extends BattlePhase {
|
||||||
const starterGender = starter.species.malePercent !== null
|
const starterGender = starter.species.malePercent !== null
|
||||||
? !starter.female ? Gender.MALE : Gender.FEMALE
|
? !starter.female ? Gender.MALE : Gender.FEMALE
|
||||||
: Gender.GENDERLESS;
|
: Gender.GENDERLESS;
|
||||||
const starterPokemon = new PlayerPokemon(this.scene, starter.species, startingLevel, 0, starter.formIndex, starterGender, starter.shiny);
|
const starterPokemon = new PlayerPokemon(this.scene, starter.species, startingLevel, starter.abilityIndex, starter.formIndex, starterGender, starter.shiny);
|
||||||
starterPokemon.setVisible(false);
|
starterPokemon.setVisible(false);
|
||||||
party.push(starterPokemon);
|
party.push(starterPokemon);
|
||||||
loadPokemonAssets.push(starterPokemon.loadAssets());
|
loadPokemonAssets.push(starterPokemon.loadAssets());
|
||||||
|
|
|
@ -36,6 +36,7 @@ export enum Button {
|
||||||
CYCLE_SHINY,
|
CYCLE_SHINY,
|
||||||
CYCLE_FORM,
|
CYCLE_FORM,
|
||||||
CYCLE_GENDER,
|
CYCLE_GENDER,
|
||||||
|
CYCLE_ABILITY,
|
||||||
QUICK_START,
|
QUICK_START,
|
||||||
AUTO,
|
AUTO,
|
||||||
SPEED_UP,
|
SPEED_UP,
|
||||||
|
@ -403,6 +404,7 @@ export default class BattleScene extends Phaser.Scene {
|
||||||
[Button.CYCLE_SHINY]: [keyCodes.R],
|
[Button.CYCLE_SHINY]: [keyCodes.R],
|
||||||
[Button.CYCLE_FORM]: [keyCodes.F],
|
[Button.CYCLE_FORM]: [keyCodes.F],
|
||||||
[Button.CYCLE_GENDER]: [keyCodes.G],
|
[Button.CYCLE_GENDER]: [keyCodes.G],
|
||||||
|
[Button.CYCLE_ABILITY]: [keyCodes.E],
|
||||||
[Button.QUICK_START]: [keyCodes.Q],
|
[Button.QUICK_START]: [keyCodes.Q],
|
||||||
[Button.AUTO]: [keyCodes.F2],
|
[Button.AUTO]: [keyCodes.F2],
|
||||||
[Button.SPEED_UP]: [keyCodes.PLUS],
|
[Button.SPEED_UP]: [keyCodes.PLUS],
|
||||||
|
@ -546,6 +548,8 @@ export default class BattleScene extends Phaser.Scene {
|
||||||
this.ui.processInput(Button.CYCLE_FORM);
|
this.ui.processInput(Button.CYCLE_FORM);
|
||||||
else if (this.isButtonPressed(Button.CYCLE_GENDER))
|
else if (this.isButtonPressed(Button.CYCLE_GENDER))
|
||||||
this.ui.processInput(Button.CYCLE_GENDER);
|
this.ui.processInput(Button.CYCLE_GENDER);
|
||||||
|
else if (this.isButtonPressed(Button.CYCLE_ABILITY))
|
||||||
|
this.ui.processInput(Button.CYCLE_ABILITY);
|
||||||
else
|
else
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,10 +59,18 @@ export abstract class PokemonSpeciesForm {
|
||||||
this.genderDiffs = genderDiffs;
|
this.genderDiffs = genderDiffs;
|
||||||
}
|
}
|
||||||
|
|
||||||
isOfType(type: integer) {
|
isOfType(type: integer): boolean {
|
||||||
return this.type1 === type || (this.type2 !== null && this.type2 === type);
|
return this.type1 === type || (this.type2 !== null && this.type2 === type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getAbilityCount(): integer {
|
||||||
|
return this.ability2 ? this.abilityHidden ? 3 : 2 : this.abilityHidden ? 2 : 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
getAbility(abilityIndex: integer): Abilities {
|
||||||
|
return !abilityIndex ? this.ability1 : abilityIndex === 1 && this.ability2 ? this.ability2 : this.abilityHidden
|
||||||
|
}
|
||||||
|
|
||||||
getSpriteAtlasPath(female: boolean, formIndex?: integer, shiny?: boolean): string {
|
getSpriteAtlasPath(female: boolean, formIndex?: integer, shiny?: boolean): string {
|
||||||
return this.getSpriteId(female, formIndex, shiny).replace(/\_{2}/g, '/');
|
return this.getSpriteId(female, formIndex, shiny).replace(/\_{2}/g, '/');
|
||||||
}
|
}
|
||||||
|
|
|
@ -879,6 +879,9 @@ export class PlayerPokemon extends Pokemon {
|
||||||
this.handleSpecialEvolutions(evolution);
|
this.handleSpecialEvolutions(evolution);
|
||||||
this.species = getPokemonSpecies(evolution.speciesId);
|
this.species = getPokemonSpecies(evolution.speciesId);
|
||||||
this.name = this.species.name.toUpperCase();
|
this.name = this.species.name.toUpperCase();
|
||||||
|
const abilityCount = this.species.getAbilityCount();
|
||||||
|
if (this.abilityIndex >= abilityCount) // Shouldn't happen
|
||||||
|
this.abilityIndex = abilityCount - 1;
|
||||||
this.getSpeciesForm().generateIconAnim(this.scene, this.gender === Gender.FEMALE, this.formIndex);
|
this.getSpeciesForm().generateIconAnim(this.scene, this.gender === Gender.FEMALE, this.formIndex);
|
||||||
this.compatibleTms.splice(0, this.compatibleTms.length);
|
this.compatibleTms.splice(0, this.compatibleTms.length);
|
||||||
this.generateCompatibleTms();
|
this.generateCompatibleTms();
|
||||||
|
|
|
@ -2,7 +2,7 @@ import BattleScene from "../battle-scene";
|
||||||
import { Gender } from "../data/gender";
|
import { Gender } from "../data/gender";
|
||||||
import Pokemon from "../pokemon";
|
import Pokemon from "../pokemon";
|
||||||
import { pokemonPrevolutions } from "../data/pokemon-evolutions";
|
import { pokemonPrevolutions } from "../data/pokemon-evolutions";
|
||||||
import PokemonSpecies, { allSpecies } from "../data/pokemon-species";
|
import PokemonSpecies, { allSpecies, getPokemonSpecies } from "../data/pokemon-species";
|
||||||
import { Species } from "../data/species";
|
import { Species } from "../data/species";
|
||||||
import * as Utils from "../utils";
|
import * as Utils from "../utils";
|
||||||
|
|
||||||
|
@ -10,6 +10,7 @@ interface SaveData {
|
||||||
trainerId: integer;
|
trainerId: integer;
|
||||||
secretId: integer;
|
secretId: integer;
|
||||||
dexData: DexData;
|
dexData: DexData;
|
||||||
|
timestamp: integer
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DexData {
|
export interface DexData {
|
||||||
|
@ -25,6 +26,7 @@ export interface DexEntryDetails {
|
||||||
shiny: boolean;
|
shiny: boolean;
|
||||||
formIndex: integer;
|
formIndex: integer;
|
||||||
female: boolean;
|
female: boolean;
|
||||||
|
abilityIndex: integer;
|
||||||
entry: DexEntry;
|
entry: DexEntry;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -32,6 +34,7 @@ export interface StarterDexUnlockTree {
|
||||||
shiny: boolean | Map<boolean, StarterDexUnlockTree>
|
shiny: boolean | Map<boolean, StarterDexUnlockTree>
|
||||||
formIndex: integer | Map<integer, StarterDexUnlockTree>
|
formIndex: integer | Map<integer, StarterDexUnlockTree>
|
||||||
female: boolean | Map<boolean, StarterDexUnlockTree>
|
female: boolean | Map<boolean, StarterDexUnlockTree>
|
||||||
|
abilityIndex: integer | Map<integer, StarterDexUnlockTree>
|
||||||
key: string,
|
key: string,
|
||||||
entry: DexEntry
|
entry: DexEntry
|
||||||
}
|
}
|
||||||
|
@ -59,7 +62,8 @@ export class GameData {
|
||||||
const data: SaveData = {
|
const data: SaveData = {
|
||||||
trainerId: this.trainerId,
|
trainerId: this.trainerId,
|
||||||
secretId: this.secretId,
|
secretId: this.secretId,
|
||||||
dexData: this.dexData
|
dexData: this.dexData,
|
||||||
|
timestamp: new Date().getTime()
|
||||||
};
|
};
|
||||||
|
|
||||||
localStorage.setItem('data', btoa(JSON.stringify(data)));
|
localStorage.setItem('data', btoa(JSON.stringify(data)));
|
||||||
|
@ -78,6 +82,9 @@ export class GameData {
|
||||||
this.secretId = data.secretId;
|
this.secretId = data.secretId;
|
||||||
this.dexData = data.dexData;
|
this.dexData = data.dexData;
|
||||||
|
|
||||||
|
if (data.timestamp === undefined)
|
||||||
|
this.convertDexData(data.dexData);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -108,12 +115,15 @@ export class GameData {
|
||||||
|
|
||||||
for (let species of allSpecies) {
|
for (let species of allSpecies) {
|
||||||
data[species.speciesId] = {};
|
data[species.speciesId] = {};
|
||||||
|
const abilityCount = species.getAbilityCount();
|
||||||
if (species.forms?.length)
|
if (species.forms?.length)
|
||||||
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => species.malePercent !== null ? initDexSubData(sd, species.forms.length).map(fd => initDexEntries(fd, 2)) : initDexEntries(sd, species.forms.length));
|
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)));
|
||||||
else if (species.malePercent !== null)
|
else if (species.malePercent !== null)
|
||||||
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => initDexEntries(sd, 2));
|
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => initDexSubData(sd, 2).map(gd => initDexEntries(gd, abilityCount)));
|
||||||
else
|
else
|
||||||
initDexEntries(data[species.speciesId] as DexData, 2)
|
initDexSubData(data[species.speciesId] as DexData, 2).map(sd => initDexEntries(sd, abilityCount))
|
||||||
}
|
}
|
||||||
|
|
||||||
const defaultStarters: Species[] = [
|
const defaultStarters: Species[] = [
|
||||||
|
@ -125,7 +135,7 @@ export class GameData {
|
||||||
];
|
];
|
||||||
|
|
||||||
for (let ds of defaultStarters) {
|
for (let ds of defaultStarters) {
|
||||||
let entry = data[ds][0][Gender.MALE] as DexEntry;
|
let entry = data[ds][0][Gender.MALE][0] as DexEntry;
|
||||||
entry.seen = true;
|
entry.seen = true;
|
||||||
entry.caught = true;
|
entry.caught = true;
|
||||||
}
|
}
|
||||||
|
@ -162,31 +172,37 @@ export class GameData {
|
||||||
}
|
}
|
||||||
|
|
||||||
getPokemonDexEntry(pokemon: Pokemon) {
|
getPokemonDexEntry(pokemon: Pokemon) {
|
||||||
return this.getDexEntry(pokemon.species, pokemon.shiny, pokemon.formIndex, pokemon.gender === Gender.FEMALE);
|
return this.getDexEntry(pokemon.species, pokemon.shiny, pokemon.formIndex, pokemon.gender === Gender.FEMALE, pokemon.abilityIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
getDexEntry(species: PokemonSpecies, shiny: boolean, formIndex: integer, female: boolean): DexEntry {
|
getDexEntry(species: PokemonSpecies, shiny: boolean, formIndex: integer, female: boolean, abilityIndex: integer): DexEntry {
|
||||||
const shinyIndex = !shiny ? 0 : 1;
|
const shinyIndex = !shiny ? 0 : 1;
|
||||||
const genderIndex = !female ? 0 : 1;
|
const genderIndex = !female ? 0 : 1;
|
||||||
const data = this.dexData[species.speciesId];
|
const data = this.dexData[species.speciesId];
|
||||||
if (species.forms?.length) {
|
if (species.forms?.length) {
|
||||||
if (species.malePercent !== null)
|
if (species.malePercent !== null)
|
||||||
return data[shinyIndex][formIndex][genderIndex];
|
return data[shinyIndex][formIndex][genderIndex][abilityIndex];
|
||||||
return data[shinyIndex][formIndex];
|
return data[shinyIndex][formIndex][abilityIndex];
|
||||||
} else if (species.malePercent !== null)
|
} else if (species.malePercent !== null)
|
||||||
return data[shinyIndex][genderIndex];
|
return data[shinyIndex][genderIndex][abilityIndex];
|
||||||
return data[shinyIndex] as DexEntry;
|
return data[shinyIndex][abilityIndex] as DexEntry;
|
||||||
}
|
}
|
||||||
|
|
||||||
getDefaultDexEntry(species: PokemonSpecies, forceShiny?: boolean, forceFormIndex?: integer, forceFemale?: boolean): DexEntryDetails {
|
getDefaultDexEntry(species: PokemonSpecies, forceShiny?: boolean, forceFormIndex?: integer, forceFemale?: boolean, forceAbilityIndex?: integer): DexEntryDetails {
|
||||||
const hasForms = !!species.forms?.length;
|
const hasForms = !!species.forms?.length;
|
||||||
|
const hasGender = species.malePercent !== null;
|
||||||
let shiny = false;
|
let shiny = false;
|
||||||
let formIndex = 0;
|
let formIndex = 0;
|
||||||
let female = false;
|
let female = false;
|
||||||
|
let abilityIndex = 0;
|
||||||
let entry = null;
|
let entry = null;
|
||||||
|
|
||||||
const traverseData = (data: DexData, level: integer) => {
|
const traverseData = (data: DexData, level: integer) => {
|
||||||
const keys = Object.keys(data);
|
const keys = Object.keys(data);
|
||||||
|
if ((!hasForms && level === 1) || (!hasGender && level === 2)) {
|
||||||
|
traverseData(data, level + 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
keys.forEach((key: string, k: integer) => {
|
keys.forEach((key: string, k: integer) => {
|
||||||
if (entry)
|
if (entry)
|
||||||
return;
|
return;
|
||||||
|
@ -207,13 +223,18 @@ export class GameData {
|
||||||
if (forceFemale !== undefined && female !== forceFemale)
|
if (forceFemale !== undefined && female !== forceFemale)
|
||||||
return
|
return
|
||||||
break;
|
break;
|
||||||
|
case 3:
|
||||||
|
abilityIndex = k;
|
||||||
|
if (forceAbilityIndex !== undefined && abilityIndex !== forceAbilityIndex)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ('caught' in data[key]) {
|
if ('caught' in data[key]) {
|
||||||
if (data[key].caught)
|
if (data[key].caught)
|
||||||
entry = data[key] as DexEntry;
|
entry = data[key] as DexEntry;
|
||||||
} else
|
} else
|
||||||
traverseData(data[key] as DexData, level + (hasForms ? 1 : 2));
|
traverseData(data[key] as DexData, level + 1);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -224,6 +245,7 @@ export class GameData {
|
||||||
shiny: shiny,
|
shiny: shiny,
|
||||||
formIndex: formIndex,
|
formIndex: formIndex,
|
||||||
female: female,
|
female: female,
|
||||||
|
abilityIndex: abilityIndex,
|
||||||
entry: entry
|
entry: entry
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -240,13 +262,14 @@ export class GameData {
|
||||||
case 'shiny':
|
case 'shiny':
|
||||||
const shinyMap = new Map<boolean, StarterDexUnlockTree>();
|
const shinyMap = new Map<boolean, StarterDexUnlockTree>();
|
||||||
for (let s = 0; s < 2; s++) {
|
for (let s = 0; s < 2; s++) {
|
||||||
const props = { shiny: !!s, formIndex: null };
|
const props = { shiny: !!s };
|
||||||
shinyMap.set(!!s, {
|
shinyMap.set(!!s, {
|
||||||
shiny: !!s,
|
shiny: !!s,
|
||||||
formIndex: hasForms ? getTreeOrValueMap('formIndex', props as StarterDexUnlockTree) : null,
|
formIndex: hasForms ? getTreeOrValueMap('formIndex', props as StarterDexUnlockTree) : null,
|
||||||
female: !hasForms && hasGender ? getTreeOrValueMap('female', props as StarterDexUnlockTree) : null,
|
female: !hasForms && hasGender ? getTreeOrValueMap('female', props as StarterDexUnlockTree) : null,
|
||||||
key: hasForms ? 'formIndex' : hasGender ? 'female' : 'entry',
|
abilityIndex: !hasForms && !hasGender ? getTreeOrValueMap('abilityIndex', props as StarterDexUnlockTree) : null,
|
||||||
entry: hasForms || hasGender ? null : this.dexData[species.speciesId][!s ? 0 : 1]
|
key: hasForms ? 'formIndex' : hasGender ? 'female' : 'abilityIndex',
|
||||||
|
entry: null,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
return shinyMap;
|
return shinyMap;
|
||||||
|
@ -258,25 +281,46 @@ export class GameData {
|
||||||
shiny: parent.shiny,
|
shiny: parent.shiny,
|
||||||
formIndex: f,
|
formIndex: f,
|
||||||
female: hasGender ? getTreeOrValueMap('female', props as StarterDexUnlockTree) : null,
|
female: hasGender ? getTreeOrValueMap('female', props as StarterDexUnlockTree) : null,
|
||||||
key: hasGender ? 'female' : 'entry',
|
abilityIndex: !hasGender ? getTreeOrValueMap('abilityIndex', props as StarterDexUnlockTree) : null,
|
||||||
entry: hasGender ? null : this.dexData[species.speciesId][!parent.shiny ? 0 : 1][f]
|
key: hasGender ? 'female' : 'abilityIndex',
|
||||||
|
entry: null
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
return formMap;
|
return formMap;
|
||||||
case 'female':
|
case 'female':
|
||||||
const genderMap = new Map<boolean, StarterDexUnlockTree>();
|
const genderMap = new Map<boolean, StarterDexUnlockTree>();
|
||||||
for (let g = 0; g < 2; g++) {
|
for (let g = 0; g < 2; g++) {
|
||||||
|
const props = { shiny: parent.shiny, formIndex: parent.formIndex, female: !!g };
|
||||||
genderMap.set(!!g, {
|
genderMap.set(!!g, {
|
||||||
shiny: parent.shiny,
|
shiny: parent.shiny,
|
||||||
formIndex: parent.formIndex,
|
formIndex: parent.formIndex,
|
||||||
female: !!g,
|
female: !!g,
|
||||||
key: 'entry',
|
abilityIndex: getTreeOrValueMap('abilityIndex', props as StarterDexUnlockTree),
|
||||||
entry: hasForms
|
key: 'abilityIndex',
|
||||||
? this.dexData[species.speciesId][!parent.shiny ? 0 : 1][parent.formIndex as integer][g]
|
entry: null
|
||||||
: this.dexData[species.speciesId][!parent.shiny ? 0 : 1][g]
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
return genderMap;
|
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,
|
||||||
|
key: 'entry',
|
||||||
|
entry: hasForms
|
||||||
|
? 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]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return abilityMap;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -284,10 +328,33 @@ export class GameData {
|
||||||
shiny: getTreeOrValueMap('shiny'),
|
shiny: getTreeOrValueMap('shiny'),
|
||||||
formIndex: null,
|
formIndex: null,
|
||||||
female: null,
|
female: null,
|
||||||
|
abilityIndex: null,
|
||||||
key: 'shiny',
|
key: 'shiny',
|
||||||
entry: null
|
entry: null
|
||||||
};
|
};
|
||||||
|
|
||||||
return root;
|
return root;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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]);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -1,13 +1,14 @@
|
||||||
import BattleScene, { Button } from "../battle-scene";
|
import BattleScene, { Button } from "../battle-scene";
|
||||||
import PokemonSpecies, { allSpecies } from "../data/pokemon-species";
|
import PokemonSpecies, { allSpecies } from "../data/pokemon-species";
|
||||||
import { Species } from "../data/species";
|
import { Species } from "../data/species";
|
||||||
import { TextStyle, addTextObject } from "./text";
|
import { TextStyle, addTextObject, getTextColor } from "./text";
|
||||||
import { Mode } from "./ui";
|
import { Mode } from "./ui";
|
||||||
import * as Utils from "../utils";
|
import * as Utils from "../utils";
|
||||||
import MessageUiHandler from "./message-ui-handler";
|
import MessageUiHandler from "./message-ui-handler";
|
||||||
import { DexEntryDetails, StarterDexUnlockTree } from "../system/game-data";
|
import { DexEntryDetails, StarterDexUnlockTree } from "../system/game-data";
|
||||||
import { Gender, getGenderColor, getGenderSymbol } from "../data/gender";
|
import { Gender, getGenderColor, getGenderSymbol } from "../data/gender";
|
||||||
import { pokemonPrevolutions } from "../data/pokemon-evolutions";
|
import { pokemonPrevolutions } from "../data/pokemon-evolutions";
|
||||||
|
import { abilities } from "../data/ability";
|
||||||
|
|
||||||
export type StarterSelectCallback = (starters: Starter[]) => void;
|
export type StarterSelectCallback = (starters: Starter[]) => void;
|
||||||
|
|
||||||
|
@ -16,6 +17,7 @@ export interface Starter {
|
||||||
shiny: boolean;
|
shiny: boolean;
|
||||||
formIndex: integer;
|
formIndex: integer;
|
||||||
female: boolean;
|
female: boolean;
|
||||||
|
abilityIndex: integer;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class StarterSelectUiHandler extends MessageUiHandler {
|
export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
|
@ -25,6 +27,8 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
private pokemonSprite: Phaser.GameObjects.Sprite;
|
private pokemonSprite: Phaser.GameObjects.Sprite;
|
||||||
private pokemonNameText: Phaser.GameObjects.Text;
|
private pokemonNameText: Phaser.GameObjects.Text;
|
||||||
private pokemonGenderText: Phaser.GameObjects.Text;
|
private pokemonGenderText: Phaser.GameObjects.Text;
|
||||||
|
private pokemonAbilityLabelText: Phaser.GameObjects.Text;
|
||||||
|
private pokemonAbilityText: Phaser.GameObjects.Text;
|
||||||
private instructionsText: Phaser.GameObjects.Text;
|
private instructionsText: Phaser.GameObjects.Text;
|
||||||
private starterSelectMessageBoxContainer: Phaser.GameObjects.Container;
|
private starterSelectMessageBoxContainer: Phaser.GameObjects.Container;
|
||||||
|
|
||||||
|
@ -32,6 +36,7 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
private shinyCursor: integer = 0;
|
private shinyCursor: integer = 0;
|
||||||
private formCursor: integer = 0;
|
private formCursor: integer = 0;
|
||||||
private genderCursor: integer = 0;
|
private genderCursor: integer = 0;
|
||||||
|
private abilityCursor: integer = 0;
|
||||||
private genCursor: integer = 0;
|
private genCursor: integer = 0;
|
||||||
|
|
||||||
private genSpecies: PokemonSpecies[][] = [];
|
private genSpecies: PokemonSpecies[][] = [];
|
||||||
|
@ -39,12 +44,13 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
private speciesLoaded: Map<Species, boolean> = new Map<Species, boolean>();
|
private speciesLoaded: Map<Species, boolean> = new Map<Species, boolean>();
|
||||||
private starterGens: integer[] = [];
|
private starterGens: integer[] = [];
|
||||||
private starterCursors: integer[] = [];
|
private starterCursors: integer[] = [];
|
||||||
private starterDetails: [boolean, integer, boolean][] = [];
|
private starterDetails: [boolean, integer, boolean, integer][] = [];
|
||||||
private speciesStarterDexEntry: DexEntryDetails;
|
private speciesStarterDexEntry: DexEntryDetails;
|
||||||
private speciesStarterDexTree: StarterDexUnlockTree;
|
private speciesStarterDexTree: StarterDexUnlockTree;
|
||||||
private canCycleShiny: boolean;
|
private canCycleShiny: boolean;
|
||||||
private canCycleForm: boolean;
|
private canCycleForm: boolean;
|
||||||
private canCycleGender: boolean;
|
private canCycleGender: boolean;
|
||||||
|
private canCycleAbility: boolean;
|
||||||
|
|
||||||
private assetLoadCancelled: Utils.BooleanHolder;
|
private assetLoadCancelled: Utils.BooleanHolder;
|
||||||
private cursorObj: Phaser.GameObjects.Image;
|
private cursorObj: Phaser.GameObjects.Image;
|
||||||
|
@ -86,6 +92,14 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
this.pokemonGenderText.setOrigin(0, 0);
|
this.pokemonGenderText.setOrigin(0, 0);
|
||||||
this.starterSelectContainer.add(this.pokemonGenderText);
|
this.starterSelectContainer.add(this.pokemonGenderText);
|
||||||
|
|
||||||
|
this.pokemonAbilityLabelText = addTextObject(this.scene, 6, 126, 'ABILITY:', TextStyle.SUMMARY, { fontSize: '64px' });
|
||||||
|
this.pokemonAbilityLabelText.setOrigin(0, 0);
|
||||||
|
this.starterSelectContainer.add(this.pokemonAbilityLabelText);
|
||||||
|
|
||||||
|
this.pokemonAbilityText = addTextObject(this.scene, 38, 126, '', TextStyle.SUMMARY, { fontSize: '64px' });
|
||||||
|
this.pokemonAbilityText.setOrigin(0, 0);
|
||||||
|
this.starterSelectContainer.add(this.pokemonAbilityText);
|
||||||
|
|
||||||
const genText = addTextObject(this.scene, 115, 6, 'I\nII\nIII\nIV\nV', TextStyle.WINDOW);
|
const genText = addTextObject(this.scene, 115, 6, 'I\nII\nIII\nIV\nV', TextStyle.WINDOW);
|
||||||
genText.setLineSpacing(16);
|
genText.setLineSpacing(16);
|
||||||
this.starterSelectContainer.add(genText);
|
this.starterSelectContainer.add(genText);
|
||||||
|
@ -161,7 +175,7 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
this.pokemonSprite = this.scene.add.sprite(53, 63, `pkmn__sub`);
|
this.pokemonSprite = this.scene.add.sprite(53, 63, `pkmn__sub`);
|
||||||
this.starterSelectContainer.add(this.pokemonSprite);
|
this.starterSelectContainer.add(this.pokemonSprite);
|
||||||
|
|
||||||
this.instructionsText = addTextObject(this.scene, 1, 132, '', TextStyle.PARTY, { fontSize: '52px' });
|
this.instructionsText = addTextObject(this.scene, 4, 140, '', TextStyle.PARTY, { fontSize: '42px' });
|
||||||
this.starterSelectContainer.add(this.instructionsText);
|
this.starterSelectContainer.add(this.instructionsText);
|
||||||
|
|
||||||
this.starterSelectMessageBoxContainer = this.scene.add.container(0, this.scene.game.canvas.height / 6);
|
this.starterSelectMessageBoxContainer = this.scene.add.container(0, this.scene.game.canvas.height / 6);
|
||||||
|
@ -248,7 +262,7 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
this.starterIcons[this.starterCursors.length].play(species.getIconKey(this.speciesStarterDexEntry?.female));
|
this.starterIcons[this.starterCursors.length].play(species.getIconKey(this.speciesStarterDexEntry?.female));
|
||||||
this.starterGens.push(this.genCursor);
|
this.starterGens.push(this.genCursor);
|
||||||
this.starterCursors.push(this.cursor);
|
this.starterCursors.push(this.cursor);
|
||||||
this.starterDetails.push([ !!this.shinyCursor, this.formCursor, !!this.genderCursor ]);
|
this.starterDetails.push([ !!this.shinyCursor, this.formCursor, !!this.genderCursor, this.abilityCursor ]);
|
||||||
if (this.speciesLoaded.get(species.speciesId))
|
if (this.speciesLoaded.get(species.speciesId))
|
||||||
species.cry(this.scene);
|
species.cry(this.scene);
|
||||||
if (this.starterCursors.length === 3) {
|
if (this.starterCursors.length === 3) {
|
||||||
|
@ -263,7 +277,8 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
species: thisObj.genSpecies[thisObj.starterGens[i]][thisObj.starterCursors[i]],
|
species: thisObj.genSpecies[thisObj.starterGens[i]][thisObj.starterCursors[i]],
|
||||||
shiny: thisObj.starterDetails[i][0],
|
shiny: thisObj.starterDetails[i][0],
|
||||||
formIndex: thisObj.starterDetails[i][1],
|
formIndex: thisObj.starterDetails[i][1],
|
||||||
female: thisObj.starterDetails[i][2]
|
female: thisObj.starterDetails[i][2],
|
||||||
|
abilityIndex: thisObj.starterDetails[i][3]
|
||||||
};
|
};
|
||||||
}));
|
}));
|
||||||
}, () => {
|
}, () => {
|
||||||
|
@ -292,7 +307,7 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
switch (button) {
|
switch (button) {
|
||||||
case Button.CYCLE_SHINY:
|
case Button.CYCLE_SHINY:
|
||||||
if (this.canCycleShiny) {
|
if (this.canCycleShiny) {
|
||||||
this.setSpeciesDetails(this.lastSpecies, !this.shinyCursor, undefined, undefined);
|
this.setSpeciesDetails(this.lastSpecies, !this.shinyCursor, undefined, undefined, undefined);
|
||||||
if (this.shinyCursor)
|
if (this.shinyCursor)
|
||||||
this.scene.sound.play('sparkle');
|
this.scene.sound.play('sparkle');
|
||||||
else
|
else
|
||||||
|
@ -301,13 +316,20 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
break;
|
break;
|
||||||
case Button.CYCLE_FORM:
|
case Button.CYCLE_FORM:
|
||||||
if (this.canCycleForm) {
|
if (this.canCycleForm) {
|
||||||
this.setSpeciesDetails(this.lastSpecies, undefined, (this.formCursor + 1) % this.lastSpecies.forms.length, undefined);
|
this.setSpeciesDetails(this.lastSpecies, undefined, (this.formCursor + 1) % this.lastSpecies.forms.length, undefined, undefined);
|
||||||
success = true;
|
success = true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case Button.CYCLE_GENDER:
|
case Button.CYCLE_GENDER:
|
||||||
if (this.canCycleGender) {
|
if (this.canCycleGender) {
|
||||||
this.setSpeciesDetails(this.lastSpecies, undefined, undefined, !this.genderCursor);
|
this.setSpeciesDetails(this.lastSpecies, undefined, undefined, !this.genderCursor, undefined);
|
||||||
|
success = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case Button.CYCLE_ABILITY:
|
||||||
|
if (this.canCycleAbility) {
|
||||||
|
const abilityCount = this.lastSpecies.getAbilityCount();
|
||||||
|
this.setSpeciesDetails(this.lastSpecies, undefined, undefined, undefined, (this.abilityCursor + 1) % abilityCount);
|
||||||
success = true;
|
success = true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -341,19 +363,31 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
let instructionLines = [
|
let instructionLines = [
|
||||||
'Arrow Keys/WASD: Move'
|
'Arrow Keys/WASD: Move'
|
||||||
];
|
];
|
||||||
|
let cycleInstructionLines = [];
|
||||||
if (!this.genMode)
|
if (!this.genMode)
|
||||||
instructionLines.push('A/Space/Enter: Select');
|
instructionLines.push('A/Space/Enter: Select');
|
||||||
if (this.starterCursors.length)
|
if (this.starterCursors.length)
|
||||||
instructionLines.push('X/Backspace/Esc: Undo');
|
instructionLines.push('X/Backspace/Esc: Undo');
|
||||||
if (this.speciesStarterDexTree) {
|
if (this.speciesStarterDexTree) {
|
||||||
if (this.canCycleShiny)
|
if (this.canCycleShiny)
|
||||||
instructionLines.push('R: Cycle Shiny');
|
cycleInstructionLines.push('R: Cycle Shiny');
|
||||||
if (this.canCycleForm)
|
if (this.canCycleForm)
|
||||||
instructionLines.push('F: Cycle Form');
|
cycleInstructionLines.push('F: Cycle Form');
|
||||||
if (this.canCycleGender)
|
if (this.canCycleGender)
|
||||||
instructionLines.push('G: Cycle Gender');
|
cycleInstructionLines.push('G: Cycle Gender');
|
||||||
|
if (this.canCycleAbility)
|
||||||
|
cycleInstructionLines.push('E: Cycle Ability');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cycleInstructionLines.length > 2) {
|
||||||
|
cycleInstructionLines[0] += ' | ' + cycleInstructionLines.splice(1, 1);
|
||||||
|
if (cycleInstructionLines.length > 2)
|
||||||
|
cycleInstructionLines[1] += ' | ' + cycleInstructionLines.splice(2, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let cil of cycleInstructionLines)
|
||||||
|
instructionLines.push(cil);
|
||||||
|
|
||||||
this.instructionsText.setText(instructionLines.join('\n'));
|
this.instructionsText.setText(instructionLines.join('\n'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -419,22 +453,24 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
this.pokemonNumberText.setText(Utils.padInt(species.speciesId, 3));
|
this.pokemonNumberText.setText(Utils.padInt(species.speciesId, 3));
|
||||||
this.pokemonNameText.setText(species.name.toUpperCase());
|
this.pokemonNameText.setText(species.name.toUpperCase());
|
||||||
|
|
||||||
this.setSpeciesDetails(species, !!this.speciesStarterDexEntry?.shiny, this.speciesStarterDexEntry?.formIndex || 0, !!this.speciesStarterDexEntry?.female);
|
this.setSpeciesDetails(species, !!this.speciesStarterDexEntry?.shiny, this.speciesStarterDexEntry?.formIndex, !!this.speciesStarterDexEntry?.female, this.speciesStarterDexEntry?.abilityIndex);
|
||||||
} else {
|
} else {
|
||||||
this.pokemonNumberText.setText(Utils.padInt(0, 3));
|
this.pokemonNumberText.setText(Utils.padInt(0, 3));
|
||||||
this.pokemonNameText.setText(species ? '???' : '');
|
this.pokemonNameText.setText(species ? '???' : '');
|
||||||
|
|
||||||
this.setSpeciesDetails(species, false, 0, false);
|
this.setSpeciesDetails(species, false, 0, false, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setSpeciesDetails(species: PokemonSpecies, shiny: boolean, formIndex: integer, female: boolean): void {
|
setSpeciesDetails(species: PokemonSpecies, shiny: boolean, formIndex: integer, female: boolean, abilityIndex: integer): void {
|
||||||
if (shiny !== undefined)
|
if (shiny !== undefined)
|
||||||
this.shinyCursor = !shiny ? 0 : 1;
|
this.shinyCursor = !shiny ? 0 : 1;
|
||||||
if (formIndex !== undefined)
|
if (formIndex !== undefined)
|
||||||
this.formCursor = formIndex;
|
this.formCursor = formIndex;
|
||||||
if (female !== undefined)
|
if (female !== undefined)
|
||||||
this.genderCursor = !female ? 0 : 1;
|
this.genderCursor = !female ? 0 : 1;
|
||||||
|
if (abilityIndex !== undefined)
|
||||||
|
this.abilityCursor = abilityIndex;
|
||||||
|
|
||||||
this.pokemonSprite.setVisible(false);
|
this.pokemonSprite.setVisible(false);
|
||||||
|
|
||||||
|
@ -444,8 +480,8 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (species) {
|
if (species) {
|
||||||
const defaultDexEntry = this.scene.gameData.getDefaultDexEntry(species, shiny, formIndex, female) || this.scene.gameData.getDefaultDexEntry(species);
|
const defaultDexEntry = this.scene.gameData.getDefaultDexEntry(species, shiny, formIndex, female, abilityIndex) || this.scene.gameData.getDefaultDexEntry(species);
|
||||||
const dexEntry = this.scene.gameData.getDexEntry(species, !!this.shinyCursor, this.formCursor, !!this.genderCursor);
|
const dexEntry = this.scene.gameData.getDexEntry(species, !!this.shinyCursor, this.formCursor, !!this.genderCursor, this.abilityCursor);
|
||||||
|
|
||||||
if (!dexEntry.caught) {
|
if (!dexEntry.caught) {
|
||||||
if (shiny === undefined || (defaultDexEntry && shiny !== defaultDexEntry.shiny))
|
if (shiny === undefined || (defaultDexEntry && shiny !== defaultDexEntry.shiny))
|
||||||
|
@ -454,10 +490,13 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
formIndex = defaultDexEntry.formIndex || 0;
|
formIndex = defaultDexEntry.formIndex || 0;
|
||||||
if (female === undefined || (defaultDexEntry && female !== defaultDexEntry.female))
|
if (female === undefined || (defaultDexEntry && female !== defaultDexEntry.female))
|
||||||
female = defaultDexEntry.female;
|
female = defaultDexEntry.female;
|
||||||
|
if (abilityIndex === undefined || (defaultDexEntry && abilityIndex !== defaultDexEntry.abilityIndex))
|
||||||
|
abilityIndex = defaultDexEntry.abilityIndex;
|
||||||
} else {
|
} else {
|
||||||
shiny = !!this.shinyCursor;
|
shiny = !!this.shinyCursor;
|
||||||
formIndex = this.formCursor;
|
formIndex = this.formCursor;
|
||||||
female = !!this.genderCursor;
|
female = !!this.genderCursor;
|
||||||
|
abilityIndex = this.abilityCursor;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.speciesStarterDexTree) {
|
if (this.speciesStarterDexTree) {
|
||||||
|
@ -484,6 +523,8 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
values = [];
|
values = [];
|
||||||
}
|
}
|
||||||
if (!tree.entry) {
|
if (!tree.entry) {
|
||||||
|
if (!tree[tree.key])
|
||||||
|
console.log(tree, tree.key);
|
||||||
for (let key of tree[tree.key].keys())
|
for (let key of tree[tree.key].keys())
|
||||||
calcUnlockedCount(tree[tree.key].get(key), prop);
|
calcUnlockedCount(tree[tree.key].get(key), prop);
|
||||||
} else if (tree.entry.caught) {
|
} else if (tree.entry.caught) {
|
||||||
|
@ -512,6 +553,12 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
this.canCycleGender = count > 1;
|
this.canCycleGender = count > 1;
|
||||||
} else
|
} else
|
||||||
this.canCycleGender = false;
|
this.canCycleGender = false;
|
||||||
|
|
||||||
|
if (this.lastSpecies.getAbilityCount() > 1) {
|
||||||
|
calcUnlockedCount(tree, 'abilityIndex', true);
|
||||||
|
this.canCycleAbility = count > 1;
|
||||||
|
} else
|
||||||
|
this.canCycleAbility = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (species.malePercent !== null) {
|
if (species.malePercent !== null) {
|
||||||
|
@ -521,8 +568,17 @@ export default class StarterSelectUiHandler extends MessageUiHandler {
|
||||||
this.pokemonGenderText.setShadowColor(getGenderColor(gender, true));
|
this.pokemonGenderText.setShadowColor(getGenderColor(gender, true));
|
||||||
} else
|
} else
|
||||||
this.pokemonGenderText.setText('');
|
this.pokemonGenderText.setText('');
|
||||||
} else
|
|
||||||
|
const ability = this.lastSpecies.getAbility(abilityIndex);
|
||||||
|
this.pokemonAbilityText.setText(abilities[ability].name.toUpperCase());
|
||||||
|
|
||||||
|
const isHidden = ability === this.lastSpecies.abilityHidden;
|
||||||
|
this.pokemonAbilityText.setColor(getTextColor(!isHidden ? TextStyle.SUMMARY : TextStyle.SUMMARY_GOLD));
|
||||||
|
this.pokemonAbilityText.setShadowColor(getTextColor(!isHidden ? TextStyle.SUMMARY : TextStyle.SUMMARY_GOLD, true));
|
||||||
|
} else {
|
||||||
this.pokemonGenderText.setText('');
|
this.pokemonGenderText.setText('');
|
||||||
|
this.pokemonAbilityText.setText('');
|
||||||
|
}
|
||||||
|
|
||||||
this.updateInstructions();
|
this.updateInstructions();
|
||||||
}
|
}
|
||||||
|
|
|
@ -435,7 +435,7 @@ export default class SummaryUiHandler extends UiHandler {
|
||||||
if (this.pokemon.species.type2)
|
if (this.pokemon.species.type2)
|
||||||
profileContainer.add(getTypeIcon(1, this.pokemon.species.type2));
|
profileContainer.add(getTypeIcon(1, this.pokemon.species.type2));
|
||||||
|
|
||||||
const ability = abilities[this.pokemon.species[`ability${!this.pokemon.abilityIndex ? '1' : '2'}`]];
|
const ability = abilities[this.pokemon.species.getAbility(this.pokemon.abilityIndex)];
|
||||||
|
|
||||||
const abilityNameText = addTextObject(this.scene, 7, 66, ability.name.toUpperCase(), TextStyle.SUMMARY);
|
const abilityNameText = addTextObject(this.scene, 7, 66, ability.name.toUpperCase(), TextStyle.SUMMARY);
|
||||||
abilityNameText.setOrigin(0, 1);
|
abilityNameText.setOrigin(0, 1);
|
||||||
|
|
|
@ -43,8 +43,13 @@ export function addTextObject(scene: Phaser.Scene, x: number, y: number, content
|
||||||
|
|
||||||
shadowColor = getTextColor(style, true);
|
shadowColor = getTextColor(style, true);
|
||||||
|
|
||||||
if (extraStyleOptions)
|
if (extraStyleOptions) {
|
||||||
|
if (extraStyleOptions.fontSize) {
|
||||||
|
const sizeRatio = parseInt(extraStyleOptions.fontSize.slice(0, -2)) / parseInt(styleOptions.fontSize.slice(0, -2));
|
||||||
|
shadowSize *= sizeRatio;
|
||||||
|
}
|
||||||
styleOptions = Object.assign(styleOptions, extraStyleOptions);
|
styleOptions = Object.assign(styleOptions, extraStyleOptions);
|
||||||
|
}
|
||||||
|
|
||||||
const ret = scene.add.text(x, y, content, styleOptions);
|
const ret = scene.add.text(x, y, content, styleOptions);
|
||||||
ret.setScale(0.1666666667);
|
ret.setScale(0.1666666667);
|
||||||
|
|
Loading…
Reference in New Issue