pokerogue/src/ui/modifier-select-ui-handler.ts

566 lines
19 KiB
TypeScript
Raw Normal View History

import BattleScene, { Button } from "../battle-scene";
import { getPlayerShopModifierTypeOptionsForWave, ModifierTypeOption } from "../modifier/modifier-type";
2023-04-20 20:46:05 +01:00
import { getPokeballAtlasKey, PokeballType } from "../data/pokeball";
import { addTextObject, getModifierTierTextTint, getTextColor, TextStyle } from "./text";
2023-03-28 19:54:52 +01:00
import AwaitableUiHandler from "./awaitable-ui-handler";
import { Mode } from "./ui";
2023-04-21 19:05:16 +01:00
import { PokemonHeldItemModifier } from "../modifier/modifier";
2024-02-14 15:44:55 +00:00
import { handleTutorial, Tutorial } from "../tutorial";
2023-03-28 19:54:52 +01:00
export const SHOP_OPTIONS_ROW_LIMIT = 6;
2023-03-28 19:54:52 +01:00
export default class ModifierSelectUiHandler extends AwaitableUiHandler {
private modifierContainer: Phaser.GameObjects.Container;
private rerollButtonContainer: Phaser.GameObjects.Container;
2023-04-21 19:05:16 +01:00
private transferButtonContainer: Phaser.GameObjects.Container;
private rerollCostText: Phaser.GameObjects.Text;
2023-04-21 19:05:16 +01:00
private rowCursor: integer = 0;
private player: boolean;
private rerollCost: integer;
2023-04-21 19:05:16 +01:00
2023-03-29 05:31:25 +01:00
public options: ModifierOption[];
public shopOptionsRows: ModifierOption[][];
2023-03-28 19:54:52 +01:00
private cursorObj: Phaser.GameObjects.Image;
constructor(scene: BattleScene) {
super(scene, Mode.CONFIRM);
2023-03-28 19:54:52 +01:00
this.options = [];
this.shopOptionsRows = [];
2023-03-28 19:54:52 +01:00
}
setup() {
const ui = this.getUi();
this.modifierContainer = this.scene.add.container(0, 0);
ui.add(this.modifierContainer);
2023-04-21 19:05:16 +01:00
this.transferButtonContainer = this.scene.add.container((this.scene.game.canvas.width / 6) - 1, -64);
this.transferButtonContainer.setVisible(false);
ui.add(this.transferButtonContainer);
const transferButtonText = addTextObject(this.scene, -4, -2, 'Transfer', TextStyle.PARTY);
transferButtonText.setOrigin(1, 0);
this.transferButtonContainer.add(transferButtonText);
this.rerollButtonContainer = this.scene.add.container(16, -64);
this.rerollButtonContainer.setVisible(false);
ui.add(this.rerollButtonContainer);
const rerollButtonText = addTextObject(this.scene, -4, -2, 'Reroll', TextStyle.PARTY);
rerollButtonText.setOrigin(0, 0);
this.rerollButtonContainer.add(rerollButtonText);
this.rerollCostText = addTextObject(this.scene, 0, 0, '', TextStyle.MONEY);
this.rerollCostText.setOrigin(0, 0);
this.rerollCostText.setPositionRelative(rerollButtonText, rerollButtonText.displayWidth + 5, 1);
this.rerollButtonContainer.add(this.rerollCostText);
2023-03-28 19:54:52 +01:00
}
2023-12-30 23:41:25 +00:00
show(args: any[]): boolean {
2023-04-10 00:15:21 +01:00
if (this.active) {
if (args.length >= 3) {
2023-04-10 00:15:21 +01:00
this.awaitingActionInput = true;
this.onActionInput = args[2];
2023-04-10 00:15:21 +01:00
}
2023-12-30 23:41:25 +00:00
return false;
2023-04-10 00:15:21 +01:00
}
if (args.length !== 4 || !(args[1] instanceof Array) || !args[1].length || !(args[2] instanceof Function))
2023-12-30 23:41:25 +00:00
return false;
2023-03-28 19:54:52 +01:00
super.show(args);
this.getUi().clearText();
this.player = args[0];
const partyHasHeldItem = this.player && !!this.scene.findModifiers(m => m instanceof PokemonHeldItemModifier && (m as PokemonHeldItemModifier).getTransferrable(true)).length;
2023-04-21 19:05:16 +01:00
this.transferButtonContainer.setVisible(false);
this.transferButtonContainer.setAlpha(0);
this.rerollButtonContainer.setVisible(false);
this.rerollButtonContainer.setAlpha(0);
this.rerollCost = args[3] as integer;
this.updateRerollCostText();
const typeOptions = args[1] as ModifierTypeOption[];
2024-03-17 02:06:56 +00:00
const shopTypeOptions = !this.scene.gameMode.hasNoShop
? getPlayerShopModifierTypeOptionsForWave(this.scene.currentBattle.waveIndex, this.scene.getWaveMoneyAmount(1))
: [];
const optionsYOffset = shopTypeOptions.length >= SHOP_OPTIONS_ROW_LIMIT ? -8 : -24;
2023-04-12 17:48:02 +01:00
for (let m = 0; m < typeOptions.length; m++) {
const sliceWidth = (this.scene.game.canvas.width / 6) / (typeOptions.length + 2);
const option = new ModifierOption(this.scene, sliceWidth * (m + 1) + (sliceWidth * 0.5), -this.scene.game.canvas.height / 12 + optionsYOffset, typeOptions[m]);
2023-03-28 19:54:52 +01:00
option.setScale(0.5);
this.scene.add.existing(option);
this.modifierContainer.add(option);
this.options.push(option);
}
for (let m = 0; m < shopTypeOptions.length; m++) {
const row = m < SHOP_OPTIONS_ROW_LIMIT ? 0 : 1;
const col = m < SHOP_OPTIONS_ROW_LIMIT ? m : m - SHOP_OPTIONS_ROW_LIMIT;
const rowOptions = shopTypeOptions.slice(row ? SHOP_OPTIONS_ROW_LIMIT : 0, row ? undefined : SHOP_OPTIONS_ROW_LIMIT);
const sliceWidth = (this.scene.game.canvas.width / SHOP_OPTIONS_ROW_LIMIT) / (rowOptions.length + 2);
const option = new ModifierOption(this.scene, sliceWidth * (col + 1) + (sliceWidth * 0.5), ((-this.scene.game.canvas.height / 12) - (this.scene.game.canvas.height / 32) - (40 - (28 * row - 1))), shopTypeOptions[m]);
option.setScale(0.375);
this.scene.add.existing(option);
this.modifierContainer.add(option);
if (row >= this.shopOptionsRows.length)
this.shopOptionsRows.push([]);
this.shopOptionsRows[row].push(option);
}
const maxUpgradeCount = typeOptions.map(to => to.upgradeCount).reduce((max, current) => Math.max(current, max), 0);
2023-04-12 17:48:02 +01:00
this.scene.showFieldOverlay(750);
2023-03-28 19:54:52 +01:00
let i = 0;
this.scene.tweens.addCounter({
ease: 'Sine.easeIn',
duration: 1250,
onUpdate: t => {
const value = t.getValue();
2023-04-12 17:48:02 +01:00
const index = Math.floor(value * typeOptions.length);
if (index > i && index <= typeOptions.length) {
const option = this.options[i];
option?.show(Math.floor((1 - value) * 1250) * 0.325 + 2000 * maxUpgradeCount, -(maxUpgradeCount - typeOptions[i].upgradeCount));
i++;
2023-03-29 05:31:25 +01:00
}
2023-03-28 19:54:52 +01:00
}
});
this.scene.time.delayedCall(1000 + maxUpgradeCount * 2000, () => {
for (let shopOption of this.shopOptionsRows.flat())
shopOption.show(0, 0);
});
this.scene.time.delayedCall(4000 + maxUpgradeCount * 2000, () => {
2023-04-21 19:05:16 +01:00
if (partyHasHeldItem) {
this.transferButtonContainer.setAlpha(0);
this.transferButtonContainer.setVisible(true);
this.scene.tweens.add({
targets: this.transferButtonContainer,
alpha: 1,
duration: 250
});
}
if (this.scene.currentBattle.waveIndex % 10) {
this.rerollButtonContainer.setAlpha(0);
this.rerollButtonContainer.setVisible(true);
this.scene.tweens.add({
targets: this.rerollButtonContainer,
alpha: 1,
duration: 250
});
}
2023-03-28 19:54:52 +01:00
this.setCursor(0);
this.setRowCursor(1);
2024-02-14 15:44:55 +00:00
handleTutorial(this.scene, Tutorial.Select_Item).then(() => {
this.setCursor(0);
this.awaitingActionInput = true;
this.onActionInput = args[2];
});
2023-03-28 19:54:52 +01:00
});
2023-12-30 23:41:25 +00:00
return true;
2023-03-28 19:54:52 +01:00
}
processInput(button: Button): boolean {
2023-03-28 19:54:52 +01:00
const ui = this.getUi();
if (!this.awaitingActionInput)
return false;
2023-03-28 19:54:52 +01:00
let success = false;
if (button === Button.ACTION) {
2023-03-28 19:54:52 +01:00
success = true;
2023-04-09 05:22:14 +01:00
if (this.onActionInput) {
const originalOnActionInput = this.onActionInput;
2023-04-10 00:15:21 +01:00
this.awaitingActionInput = false;
2023-04-09 05:22:14 +01:00
this.onActionInput = null;
if (!originalOnActionInput(this.rowCursor, this.cursor)) {
this.awaitingActionInput = true;
this.onActionInput = originalOnActionInput;
}
2023-04-09 05:22:14 +01:00
}
} else if (button === Button.CANCEL) {
if (this.player) {
success = true;
if (this.onActionInput) {
const originalOnActionInput = this.onActionInput;
this.awaitingActionInput = false;
this.onActionInput = null;
originalOnActionInput(-1);
}
2023-04-09 05:22:14 +01:00
}
2023-03-28 19:54:52 +01:00
} else {
switch (button) {
2023-04-21 19:05:16 +01:00
case Button.UP:
if (this.rowCursor < this.shopOptionsRows.length + 1)
success = this.setRowCursor(this.rowCursor + 1);
2023-04-21 19:05:16 +01:00
break;
case Button.DOWN:
if (this.rowCursor)
success = this.setRowCursor(this.rowCursor - 1);
2023-04-21 19:05:16 +01:00
break;
case Button.LEFT:
if (!this.rowCursor)
success = this.rerollButtonContainer.visible && this.setCursor(0);
else if (this.cursor)
success = this.setCursor(this.cursor - 1);
else if (this.rowCursor === 1 && this.rerollButtonContainer.visible)
success = this.setRowCursor(0);
2023-03-28 19:54:52 +01:00
break;
case Button.RIGHT:
if (!this.rowCursor)
success = this.transferButtonContainer.visible && this.setCursor(1);
else if (this.cursor < this.getRowItems(this.rowCursor) - 1)
2023-03-28 19:54:52 +01:00
success = this.setCursor(this.cursor + 1);
else if (this.rowCursor === 1 && this.transferButtonContainer.visible)
success = this.setRowCursor(0);
2023-03-28 19:54:52 +01:00
break;
}
}
if (success)
ui.playSelect();
return success;
2023-03-28 19:54:52 +01:00
}
setCursor(cursor: integer): boolean {
const ui = this.getUi();
const ret = super.setCursor(cursor);
if (!this.cursorObj) {
this.cursorObj = this.scene.add.image(0, 0, 'cursor');
this.modifierContainer.add(this.cursorObj);
}
const options = (this.rowCursor === 1 ? this.options : this.shopOptionsRows[this.shopOptionsRows.length - (this.rowCursor - 1)]);
this.cursorObj.setScale(this.rowCursor === 1 ? 2 : this.rowCursor >= 2 ? 1.5 : 1);
2023-04-21 19:05:16 +01:00
if (this.rowCursor) {
let sliceWidth = (this.scene.game.canvas.width / 6) / (options.length + 2);
if (this.rowCursor < 2)
this.cursorObj.setPosition(sliceWidth * (cursor + 1) + (sliceWidth * 0.5) - 20, (-this.scene.game.canvas.height / 12) - (this.shopOptionsRows.length > 1 ? 6 : 22));
else
this.cursorObj.setPosition(sliceWidth * (cursor + 1) + (sliceWidth * 0.5) - 16, (-this.scene.game.canvas.height / 12 - this.scene.game.canvas.height / 32) - (-16 + 28 * (this.rowCursor - (this.shopOptionsRows.length - 1))));
ui.showText(options[this.cursor].modifierTypeOption.type.getDescription(this.scene));
} else if (!cursor) {
this.cursorObj.setPosition(6, -60);
ui.showText('Spend money to reroll your item options');
2023-04-21 19:05:16 +01:00
} else {
this.cursorObj.setPosition((this.scene.game.canvas.width / 6) - 50, -60);
2024-03-31 00:40:48 +00:00
ui.showText('Transfer a held item from one Pokémon to another');
2023-04-21 19:05:16 +01:00
}
2023-03-28 19:54:52 +01:00
return ret;
}
setRowCursor(rowCursor: integer): boolean {
const lastRowCursor = this.rowCursor;
if (rowCursor !== lastRowCursor && (rowCursor || this.rerollButtonContainer.visible || this.transferButtonContainer.visible)) {
this.rowCursor = rowCursor;
let newCursor = Math.round(this.cursor / Math.max(this.getRowItems(lastRowCursor) - 1, 1) * (this.getRowItems(rowCursor) - 1));
if (!rowCursor) {
if (!newCursor && !this.rerollButtonContainer.visible)
newCursor = 1;
else if (newCursor && !this.transferButtonContainer.visible)
newCursor = 0;
}
this.cursor = -1;
this.setCursor(newCursor);
return true;
}
return false;
}
private getRowItems(rowCursor: integer): integer {
switch (rowCursor) {
case 0:
return 2;
case 1:
return this.options.length;
default:
return this.shopOptionsRows[this.shopOptionsRows.length - (rowCursor - 1)].length;
}
}
updateCostText(): void {
const shopOptions = this.shopOptionsRows.flat();
for (let shopOption of shopOptions)
shopOption.updateCostText();
this.updateRerollCostText();
}
updateRerollCostText(): void {
const canReroll = this.scene.money >= this.rerollCost;
this.rerollCostText.setText(`${this.rerollCost.toLocaleString('en-US')}`);
2024-03-31 21:49:53 +01:00
this.rerollCostText.setColor(this.getTextColor(canReroll ? TextStyle.MONEY : TextStyle.PARTY_RED));
this.rerollCostText.setShadowColor(this.getTextColor(canReroll ? TextStyle.MONEY : TextStyle.PARTY_RED, true));
}
2023-03-28 19:54:52 +01:00
clear() {
super.clear();
this.awaitingActionInput = false;
this.onActionInput = null;
this.getUi().clearText();
this.eraseCursor();
this.scene.hideFieldOverlay(250);
const options = this.options.concat(this.shopOptionsRows.flat());
this.options.splice(0, this.options.length);
this.shopOptionsRows.splice(0, this.shopOptionsRows.length);
2023-03-28 19:54:52 +01:00
this.scene.tweens.add({
targets: options,
2023-03-28 19:54:52 +01:00
scale: 0.01,
duration: 250,
ease: 'Cubic.easeIn',
onComplete: () => options.forEach(o => o.destroy())
2023-03-28 19:54:52 +01:00
});
2023-11-10 21:41:02 +00:00
[ this.rerollButtonContainer, this.transferButtonContainer ].forEach(container => {
if (container.visible) {
this.scene.tweens.add({
targets: container,
alpha: 0,
duration: 250,
ease: 'Cubic.easeIn',
onComplete: () => {
if (!this.options.length)
container.setVisible(false);
else
container.setAlpha(1);
}
2023-11-10 21:41:02 +00:00
});
}
});
2023-03-28 19:54:52 +01:00
}
eraseCursor() {
if (this.cursorObj)
this.cursorObj.destroy();
this.cursorObj = null;
}
}
class ModifierOption extends Phaser.GameObjects.Container {
2023-04-12 17:48:02 +01:00
public modifierTypeOption: ModifierTypeOption;
2023-03-28 19:54:52 +01:00
private pb: Phaser.GameObjects.Sprite;
2023-04-12 17:48:02 +01:00
private pbTint: Phaser.GameObjects.Sprite;
2023-03-28 19:54:52 +01:00
private itemContainer: Phaser.GameObjects.Container;
private item: Phaser.GameObjects.Sprite;
private itemTint: Phaser.GameObjects.Sprite;
private itemText: Phaser.GameObjects.Text;
private itemCostText: Phaser.GameObjects.Text;
2023-03-28 19:54:52 +01:00
2023-04-12 17:48:02 +01:00
constructor(scene: BattleScene, x: number, y: number, modifierTypeOption: ModifierTypeOption) {
2023-03-28 19:54:52 +01:00
super(scene, x, y);
2023-04-12 17:48:02 +01:00
this.modifierTypeOption = modifierTypeOption;
2023-03-28 19:54:52 +01:00
this.setup();
}
setup() {
if (!this.modifierTypeOption.cost) {
const getPb = (): Phaser.GameObjects.Sprite => {
const pb = this.scene.add.sprite(0, -182, 'pb', this.getPbAtlasKey(-this.modifierTypeOption.upgradeCount));
pb.setScale(2);
return pb;
};
this.pb = getPb();
this.add(this.pb);
this.pbTint = getPb();
this.pbTint.setVisible(false);
this.add(this.pbTint);
}
2023-04-12 17:48:02 +01:00
2023-03-28 19:54:52 +01:00
this.itemContainer = this.scene.add.container(0, 0);
this.itemContainer.setScale(0.5);
this.itemContainer.setAlpha(0);
this.add(this.itemContainer);
const getItem = () => {
2023-04-12 17:48:02 +01:00
const item = this.scene.add.sprite(0, 0, 'items', this.modifierTypeOption.type.iconImage);
2023-03-28 19:54:52 +01:00
return item;
};
this.item = getItem();
this.itemContainer.add(this.item);
if (!this.modifierTypeOption.cost) {
this.itemTint = getItem();
this.itemTint.setTintFill(Phaser.Display.Color.GetColor(255, 192, 255));
this.itemContainer.add(this.itemTint);
}
2023-03-28 19:54:52 +01:00
2023-04-12 17:48:02 +01:00
this.itemText = addTextObject(this.scene, 0, 35, this.modifierTypeOption.type.name, TextStyle.PARTY, { align: 'center' });
2023-03-28 19:54:52 +01:00
this.itemText.setOrigin(0.5, 0);
this.itemText.setAlpha(0);
2023-11-02 04:55:20 +00:00
this.itemText.setTint(getModifierTierTextTint(this.modifierTypeOption.type.tier));
2023-03-28 19:54:52 +01:00
this.add(this.itemText);
if (this.modifierTypeOption.cost) {
this.itemCostText = addTextObject(this.scene, 0, 45, '', TextStyle.MONEY, { align: 'center' });
this.itemCostText.setOrigin(0.5, 0);
this.itemCostText.setAlpha(0);
this.add(this.itemCostText);
this.updateCostText();
}
2023-03-28 19:54:52 +01:00
}
show(remainingDuration: integer, upgradeCountOffset: integer) {
if (!this.modifierTypeOption.cost) {
this.scene.tweens.add({
targets: this.pb,
y: 0,
duration: 1250,
ease: 'Bounce.Out'
});
2023-03-28 19:54:52 +01:00
let lastValue = 1;
let bounceCount = 0;
let bounce = false;
this.scene.tweens.addCounter({
from: 1,
to: 0,
duration: 1250,
ease: 'Bounce.Out',
onUpdate: t => {
if (!this.scene)
return;
const value = t.getValue();
if (!bounce && value > lastValue) {
(this.scene as BattleScene).playSound('pb_bounce_1', { volume: 1 / ++bounceCount });
bounce = true;
} else if (bounce && value < lastValue)
bounce = false;
lastValue = value;
}
});
2023-03-28 19:54:52 +01:00
for (let u = 0; u < this.modifierTypeOption.upgradeCount; u++) {
const upgradeIndex = u;
this.scene.time.delayedCall(remainingDuration - 2000 * (this.modifierTypeOption.upgradeCount - (upgradeIndex + 1 + upgradeCountOffset)), () => {
(this.scene as BattleScene).playSound('upgrade', { rate: 1 + 0.25 * upgradeIndex });
this.pbTint.setPosition(this.pb.x, this.pb.y);
this.pbTint.setTintFill(0xFFFFFF);
this.pbTint.setAlpha(0);
this.pbTint.setVisible(true);
this.scene.tweens.add({
targets: this.pbTint,
alpha: 1,
duration: 1000,
ease: 'Sine.easeIn',
onComplete: () => {
this.pb.setTexture('pb', this.getPbAtlasKey(-this.modifierTypeOption.upgradeCount + (upgradeIndex + 1)));
this.scene.tweens.add({
targets: this.pbTint,
alpha: 0,
duration: 750,
ease: 'Sine.easeOut',
onComplete: () => {
this.pbTint.setVisible(false);
}
});
}
});
2023-04-12 17:48:02 +01:00
});
}
2023-04-12 17:48:02 +01:00
}
2023-03-28 19:54:52 +01:00
this.scene.time.delayedCall(remainingDuration + 2000, () => {
2023-03-29 05:31:25 +01:00
if (!this.scene)
return;
if (!this.modifierTypeOption.cost) {
this.pb.setTexture('pb', `${this.getPbAtlasKey(0)}_open`);
(this.scene as BattleScene).playSound('pb_rel');
this.scene.tweens.add({
targets: this.pb,
duration: 500,
delay: 250,
ease: 'Sine.easeIn',
alpha: 0,
onComplete: () => this.pb.destroy()
});
}
2023-03-28 19:54:52 +01:00
this.scene.tweens.add({
targets: this.itemContainer,
duration: 500,
ease: 'Elastic.Out',
scale: 2,
alpha: 1
});
if (!this.modifierTypeOption.cost) {
this.scene.tweens.add({
targets: this.itemTint,
alpha: 0,
duration: 500,
ease: 'Sine.easeIn',
onComplete: () => this.itemTint.destroy()
});
}
2023-03-28 19:54:52 +01:00
this.scene.tweens.add({
targets: this.itemText,
duration: 500,
alpha: 1,
y: 25,
ease: 'Cubic.easeInOut'
});
if (this.itemCostText) {
this.scene.tweens.add({
targets: this.itemCostText,
duration: 500,
alpha: 1,
y: 35,
ease: 'Cubic.easeInOut'
});
}
});
2023-03-28 19:54:52 +01:00
}
getPbAtlasKey(tierOffset: integer = 0) {
return getPokeballAtlasKey((this.modifierTypeOption.type.tier + tierOffset) as integer as PokeballType);
2023-03-28 19:54:52 +01:00
}
updateCostText(): void {
2024-03-31 21:49:53 +01:00
const scene = this.scene as BattleScene;
const textStyle = this.modifierTypeOption.cost <= scene.money ? TextStyle.MONEY : TextStyle.PARTY_RED;
this.itemCostText.setText(`${this.modifierTypeOption.cost.toLocaleString('en-US')}`);
2024-03-31 21:49:53 +01:00
this.itemCostText.setColor(getTextColor(textStyle, false, scene.uiTheme));
this.itemCostText.setShadowColor(getTextColor(textStyle, true, scene.uiTheme));
}
2023-03-28 19:54:52 +01:00
}