Add starter select ability selection

This commit is contained in:
Flashfyre 2023-04-26 12:50:21 -04:00
parent bb86ab6250
commit 592ccc7c6a
9 changed files with 190 additions and 47 deletions

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

After

Width:  |  Height:  |  Size: 1.4 KiB

View File

@ -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());

View File

@ -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;
} }

View File

@ -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, '/');
} }

View File

@ -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();

View File

@ -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]);
});
}
} }

View File

@ -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();
} }

View File

@ -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);

View File

@ -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);