pokeemerald/src/battle_controller_player.c

3276 lines
124 KiB
C
Raw Normal View History

#include "global.h"
#include "battle.h"
2018-11-14 00:01:50 +00:00
#include "battle_anim.h"
#include "battle_arena.h"
#include "battle_controllers.h"
2018-11-14 00:01:50 +00:00
#include "battle_dome.h"
#include "battle_interface.h"
2018-11-14 00:01:50 +00:00
#include "battle_message.h"
#include "battle_setup.h"
2018-02-27 20:40:09 +01:00
#include "battle_tv.h"
2018-11-14 00:01:50 +00:00
#include "bg.h"
2019-04-04 23:53:06 +02:00
#include "data.h"
2018-11-14 00:01:50 +00:00
#include "item.h"
#include "item_menu.h"
#include "link.h"
#include "main.h"
#include "m4a.h"
#include "palette.h"
2018-11-14 00:01:50 +00:00
#include "party_menu.h"
2017-10-22 18:43:15 +02:00
#include "pokeball.h"
2018-11-14 00:01:50 +00:00
#include "pokemon.h"
#include "random.h"
2018-02-08 00:00:25 +01:00
#include "recorded_battle.h"
2018-11-14 00:01:50 +00:00
#include "reshow_battle_screen.h"
#include "sound.h"
#include "string_util.h"
#include "task.h"
#include "text.h"
#include "util.h"
#include "window.h"
#include "constants/battle_anim.h"
#include "constants/battle_config.h"
2018-11-14 00:01:50 +00:00
#include "constants/items.h"
#include "constants/moves.h"
2019-10-17 19:22:03 -04:00
#include "constants/party_menu.h"
2018-11-14 00:01:50 +00:00
#include "constants/songs.h"
#include "constants/trainers.h"
2019-04-04 17:05:46 -04:00
#include "constants/rgb.h"
2018-02-08 00:00:25 +01:00
extern struct MusicPlayerInfo gMPlayInfo_BGM;
// this file's functions
2017-10-22 18:43:15 +02:00
static void PlayerHandleGetMonData(void);
static void PlayerHandleSetMonData(void);
static void PlayerHandleSetRawMonData(void);
static void PlayerHandleLoadMonSprite(void);
static void PlayerHandleSwitchInAnim(void);
static void PlayerHandleReturnMonToBall(void);
static void PlayerHandleDrawTrainerPic(void);
static void PlayerHandleTrainerSlide(void);
static void PlayerHandleTrainerSlideBack(void);
static void PlayerHandleFaintAnimation(void);
static void PlayerHandlePaletteFade(void);
static void PlayerHandleSuccessBallThrowAnim(void);
static void PlayerHandleBallThrowAnim(void);
static void PlayerHandlePause(void);
static void PlayerHandleMoveAnimation(void);
static void PlayerHandlePrintString(void);
static void PlayerHandlePrintSelectionString(void);
2017-10-22 18:43:15 +02:00
static void PlayerHandleChooseAction(void);
static void PlayerHandleYesNoBox(void);
2017-10-22 18:43:15 +02:00
static void PlayerHandleChooseMove(void);
static void PlayerHandleChooseItem(void);
static void PlayerHandleChoosePokemon(void);
static void PlayerHandleCmd23(void);
static void PlayerHandleHealthBarUpdate(void);
static void PlayerHandleExpUpdate(void);
static void PlayerHandleStatusIconUpdate(void);
static void PlayerHandleStatusAnimation(void);
static void PlayerHandleStatusXor(void);
static void PlayerHandleDataTransfer(void);
static void PlayerHandleDMA3Transfer(void);
static void PlayerHandlePlayBGM(void);
static void PlayerHandleCmd32(void);
static void PlayerHandleTwoReturnValues(void);
static void PlayerHandleChosenMonReturnValue(void);
static void PlayerHandleOneReturnValue(void);
static void PlayerHandleOneReturnValue_Duplicate(void);
2020-12-29 16:51:44 -05:00
static void PlayerHandleClearUnkVar(void);
static void PlayerHandleSetUnkVar(void);
static void PlayerHandleClearUnkFlag(void);
static void PlayerHandleToggleUnkFlag(void);
2017-10-22 18:43:15 +02:00
static void PlayerHandleHitAnimation(void);
2021-04-03 12:38:07 -04:00
static void PlayerHandleCantSwitch(void);
static void PlayerHandlePlaySE(void);
2017-10-22 18:43:15 +02:00
static void PlayerHandlePlayFanfareOrBGM(void);
static void PlayerHandleFaintingCry(void);
static void PlayerHandleIntroSlide(void);
static void PlayerHandleIntroTrainerBallThrow(void);
static void PlayerHandleDrawPartyStatusSummary(void);
2018-06-28 21:06:32 +02:00
static void PlayerHandleHidePartyStatusSummary(void);
2018-06-20 23:07:51 +02:00
static void PlayerHandleEndBounceEffect(void);
2017-10-22 18:43:15 +02:00
static void PlayerHandleSpriteInvisibility(void);
static void PlayerHandleBattleAnimation(void);
static void PlayerHandleLinkStandbyMsg(void);
static void PlayerHandleResetActionMoveSelection(void);
static void PlayerHandleEndLinkBattle(void);
2018-07-13 23:00:56 +02:00
static void PlayerHandleBattleDebug(void);
static void PlayerCmdEnd(void);
2017-10-22 18:43:15 +02:00
static void PlayerBufferRunCommand(void);
static void HandleInputChooseTarget(void);
static void HandleInputChooseMove(void);
static void MoveSelectionCreateCursorAt(u8 cursorPos, u8 arg1);
static void MoveSelectionDestroyCursorAt(u8 cursorPos);
static void MoveSelectionDisplayPpNumber(void);
static void MoveSelectionDisplayPpString(void);
static void MoveSelectionDisplayMoveType(void);
static void MoveSelectionDisplayMoveNames(void);
2018-11-22 01:10:50 +00:00
static void HandleMoveSwitching(void);
2021-01-22 20:03:21 -05:00
static void SwitchIn_HandleSoundAndEnd(void);
2018-07-07 19:57:09 +02:00
static void WaitForMonSelection(void);
2017-10-22 20:38:23 +02:00
static void CompleteWhenChoseItem(void);
static void Task_LaunchLvlUpAnim(u8 taskId);
2017-10-22 18:43:15 +02:00
static void Task_PrepareToGiveExpWithExpBar(u8 taskId);
static void DestroyExpTaskAndCompleteOnInactiveTextPrinter(u8 taskId);
2021-01-22 20:03:21 -05:00
static void Task_GiveExpWithExpBar(u8 taskId);
static void Task_UpdateLvlInHealthbox(u8 taskId);
2017-10-22 18:43:15 +02:00
static void PrintLinkStandbyMsg(void);
static u32 CopyPlayerMonData(u8 monId, u8 *dst);
static void SetPlayerMonData(u8 monId);
2021-01-22 20:03:21 -05:00
static void StartSendOutAnim(u8 battlerId, bool8 dontClearSubstituteBit);
2017-10-22 18:43:15 +02:00
static void DoSwitchOutAnimation(void);
static void PlayerDoMoveAnimation(void);
2021-01-22 20:03:21 -05:00
static void Task_StartSendOutAnim(u8 taskId);
static void EndDrawPartyStatusSummary(void);
2017-10-22 18:43:15 +02:00
2017-10-31 18:04:08 +01:00
static void (*const sPlayerBufferCommands[CONTROLLER_CMDS_COUNT])(void) =
{
2020-12-29 16:51:44 -05:00
[CONTROLLER_GETMONDATA] = PlayerHandleGetMonData,
[CONTROLLER_GETRAWMONDATA] = PlayerHandleGetRawMonData,
[CONTROLLER_SETMONDATA] = PlayerHandleSetMonData,
[CONTROLLER_SETRAWMONDATA] = PlayerHandleSetRawMonData,
[CONTROLLER_LOADMONSPRITE] = PlayerHandleLoadMonSprite,
[CONTROLLER_SWITCHINANIM] = PlayerHandleSwitchInAnim,
[CONTROLLER_RETURNMONTOBALL] = PlayerHandleReturnMonToBall,
[CONTROLLER_DRAWTRAINERPIC] = PlayerHandleDrawTrainerPic,
[CONTROLLER_TRAINERSLIDE] = PlayerHandleTrainerSlide,
[CONTROLLER_TRAINERSLIDEBACK] = PlayerHandleTrainerSlideBack,
[CONTROLLER_FAINTANIMATION] = PlayerHandleFaintAnimation,
[CONTROLLER_PALETTEFADE] = PlayerHandlePaletteFade,
[CONTROLLER_SUCCESSBALLTHROWANIM] = PlayerHandleSuccessBallThrowAnim,
[CONTROLLER_BALLTHROWANIM] = PlayerHandleBallThrowAnim,
[CONTROLLER_PAUSE] = PlayerHandlePause,
[CONTROLLER_MOVEANIMATION] = PlayerHandleMoveAnimation,
[CONTROLLER_PRINTSTRING] = PlayerHandlePrintString,
[CONTROLLER_PRINTSTRINGPLAYERONLY] = PlayerHandlePrintSelectionString,
[CONTROLLER_CHOOSEACTION] = PlayerHandleChooseAction,
[CONTROLLER_YESNOBOX] = PlayerHandleYesNoBox,
2020-12-29 16:51:44 -05:00
[CONTROLLER_CHOOSEMOVE] = PlayerHandleChooseMove,
[CONTROLLER_OPENBAG] = PlayerHandleChooseItem,
[CONTROLLER_CHOOSEPOKEMON] = PlayerHandleChoosePokemon,
[CONTROLLER_23] = PlayerHandleCmd23,
[CONTROLLER_HEALTHBARUPDATE] = PlayerHandleHealthBarUpdate,
[CONTROLLER_EXPUPDATE] = PlayerHandleExpUpdate,
[CONTROLLER_STATUSICONUPDATE] = PlayerHandleStatusIconUpdate,
[CONTROLLER_STATUSANIMATION] = PlayerHandleStatusAnimation,
[CONTROLLER_STATUSXOR] = PlayerHandleStatusXor,
[CONTROLLER_DATATRANSFER] = PlayerHandleDataTransfer,
[CONTROLLER_DMA3TRANSFER] = PlayerHandleDMA3Transfer,
[CONTROLLER_PLAYBGM] = PlayerHandlePlayBGM,
[CONTROLLER_32] = PlayerHandleCmd32,
[CONTROLLER_TWORETURNVALUES] = PlayerHandleTwoReturnValues,
[CONTROLLER_CHOSENMONRETURNVALUE] = PlayerHandleChosenMonReturnValue,
[CONTROLLER_ONERETURNVALUE] = PlayerHandleOneReturnValue,
[CONTROLLER_ONERETURNVALUE_DUPLICATE] = PlayerHandleOneReturnValue_Duplicate,
[CONTROLLER_CLEARUNKVAR] = PlayerHandleClearUnkVar,
[CONTROLLER_SETUNKVAR] = PlayerHandleSetUnkVar,
[CONTROLLER_CLEARUNKFLAG] = PlayerHandleClearUnkFlag,
[CONTROLLER_TOGGLEUNKFLAG] = PlayerHandleToggleUnkFlag,
[CONTROLLER_HITANIMATION] = PlayerHandleHitAnimation,
2021-04-03 12:38:07 -04:00
[CONTROLLER_CANTSWITCH] = PlayerHandleCantSwitch,
2020-12-29 16:51:44 -05:00
[CONTROLLER_PLAYSE] = PlayerHandlePlaySE,
[CONTROLLER_PLAYFANFAREORBGM] = PlayerHandlePlayFanfareOrBGM,
[CONTROLLER_FAINTINGCRY] = PlayerHandleFaintingCry,
[CONTROLLER_INTROSLIDE] = PlayerHandleIntroSlide,
[CONTROLLER_INTROTRAINERBALLTHROW] = PlayerHandleIntroTrainerBallThrow,
[CONTROLLER_DRAWPARTYSTATUSSUMMARY] = PlayerHandleDrawPartyStatusSummary,
[CONTROLLER_HIDEPARTYSTATUSSUMMARY] = PlayerHandleHidePartyStatusSummary,
[CONTROLLER_ENDBOUNCE] = PlayerHandleEndBounceEffect,
[CONTROLLER_SPRITEINVISIBILITY] = PlayerHandleSpriteInvisibility,
[CONTROLLER_BATTLEANIMATION] = PlayerHandleBattleAnimation,
[CONTROLLER_LINKSTANDBYMSG] = PlayerHandleLinkStandbyMsg,
[CONTROLLER_RESETACTIONMOVESELECTION] = PlayerHandleResetActionMoveSelection,
[CONTROLLER_ENDLINKBATTLE] = PlayerHandleEndLinkBattle,
[CONTROLLER_DEBUGMENU] = PlayerHandleBattleDebug,
2020-12-29 16:51:44 -05:00
[CONTROLLER_TERMINATOR_NOP] = PlayerCmdEnd
};
// unknown unused data
2020-12-29 16:51:44 -05:00
static const u8 sUnused[] = {0x48, 0x48, 0x20, 0x5a, 0x50, 0x50, 0x50, 0x58};
void BattleControllerDummy(void)
{
}
2017-10-26 23:12:48 +02:00
void SetControllerToPlayer(void)
{
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = PlayerBufferRunCommand;
gDoingBattleAnim = FALSE;
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames = 0;
}
2017-10-22 18:43:15 +02:00
static void PlayerBufferExecCompleted(void)
{
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = PlayerBufferRunCommand;
if (gBattleTypeFlags & BATTLE_TYPE_LINK)
{
u8 playerId = GetMultiplayerId();
PrepareBufferDataTransferLink(2, 4, &playerId);
gBattleResources->bufferA[gActiveBattler][0] = CONTROLLER_TERMINATOR_NOP;
}
else
{
2018-02-06 13:48:02 -06:00
gBattleControllerExecFlags &= ~gBitTable[gActiveBattler];
}
}
2017-10-22 18:43:15 +02:00
static void PlayerBufferRunCommand(void)
{
2018-02-06 13:48:02 -06:00
if (gBattleControllerExecFlags & gBitTable[gActiveBattler])
{
if (gBattleResources->bufferA[gActiveBattler][0] < ARRAY_COUNT(sPlayerBufferCommands))
sPlayerBufferCommands[gBattleResources->bufferA[gActiveBattler][0]]();
else
PlayerBufferExecCompleted();
}
}
2017-10-22 18:43:15 +02:00
static void CompleteOnBankSpritePosX_0(void)
{
2021-07-07 09:11:52 -04:00
if (gSprites[gBattlerSpriteIds[gActiveBattler]].x2 == 0)
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void HandleInputChooseAction(void)
{
u16 itemId = gBattleResources->bufferA[gActiveBattler][2] | (gBattleResources->bufferA[gActiveBattler][3] << 8);
2018-06-20 23:07:51 +02:00
DoBounceEffect(gActiveBattler, BOUNCE_HEALTHBOX, 7, 1);
DoBounceEffect(gActiveBattler, BOUNCE_MON, 7, 1);
2020-09-04 21:11:55 -04:00
if (JOY_REPEAT(DPAD_ANY) && gSaveBlock2Ptr->optionsButtonMode == OPTIONS_BUTTON_MODE_L_EQUALS_A)
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames++;
else
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames = 0;
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
switch (gActionSelectionCursor[gActiveBattler])
{
2017-10-22 20:38:23 +02:00
case 0:
2018-02-06 16:09:39 -06:00
BtlController_EmitTwoReturnValues(1, B_ACTION_USE_MOVE, 0);
break;
2017-10-22 20:38:23 +02:00
case 1:
2018-02-06 16:09:39 -06:00
BtlController_EmitTwoReturnValues(1, B_ACTION_USE_ITEM, 0);
break;
2017-10-22 20:38:23 +02:00
case 2:
2018-02-06 16:09:39 -06:00
BtlController_EmitTwoReturnValues(1, B_ACTION_SWITCH, 0);
break;
2017-10-22 20:38:23 +02:00
case 3:
2018-02-06 16:09:39 -06:00
BtlController_EmitTwoReturnValues(1, B_ACTION_RUN, 0);
break;
}
PlayerBufferExecCompleted();
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_LEFT))
{
2018-02-06 16:09:39 -06:00
if (gActionSelectionCursor[gActiveBattler] & 1) // if is B_ACTION_USE_ITEM or B_ACTION_RUN
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
ActionSelectionDestroyCursorAt(gActionSelectionCursor[gActiveBattler]);
gActionSelectionCursor[gActiveBattler] ^= 1;
ActionSelectionCreateCursorAt(gActionSelectionCursor[gActiveBattler], 0);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_RIGHT))
{
2018-02-06 16:09:39 -06:00
if (!(gActionSelectionCursor[gActiveBattler] & 1)) // if is B_ACTION_USE_MOVE or B_ACTION_SWITCH
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
ActionSelectionDestroyCursorAt(gActionSelectionCursor[gActiveBattler]);
gActionSelectionCursor[gActiveBattler] ^= 1;
ActionSelectionCreateCursorAt(gActionSelectionCursor[gActiveBattler], 0);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_UP))
{
2018-02-06 16:09:39 -06:00
if (gActionSelectionCursor[gActiveBattler] & 2) // if is B_ACTION_SWITCH or B_ACTION_RUN
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
ActionSelectionDestroyCursorAt(gActionSelectionCursor[gActiveBattler]);
gActionSelectionCursor[gActiveBattler] ^= 2;
ActionSelectionCreateCursorAt(gActionSelectionCursor[gActiveBattler], 0);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_DOWN))
{
2018-02-06 16:09:39 -06:00
if (!(gActionSelectionCursor[gActiveBattler] & 2)) // if is B_ACTION_USE_MOVE or B_ACTION_USE_ITEM
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
ActionSelectionDestroyCursorAt(gActionSelectionCursor[gActiveBattler]);
gActionSelectionCursor[gActiveBattler] ^= 2;
ActionSelectionCreateCursorAt(gActionSelectionCursor[gActiveBattler], 0);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(B_BUTTON) || gPlayerDpadHoldFrames > 59)
{
if ((gBattleTypeFlags & BATTLE_TYPE_DOUBLE)
2018-02-05 19:46:59 -06:00
&& GetBattlerPosition(gActiveBattler) == B_POSITION_PLAYER_RIGHT
&& !(gAbsentBattlerFlags & gBitTable[GetBattlerAtPosition(B_POSITION_PLAYER_LEFT)])
&& !(gBattleTypeFlags & BATTLE_TYPE_MULTI))
{
if (gBattleResources->bufferA[gActiveBattler][1] == B_ACTION_USE_ITEM)
{
// Add item to bag if it is a ball
2020-07-12 01:49:32 -04:00
if (itemId <= LAST_BALL)
AddBagItem(itemId, 1);
else
return;
}
PlaySE(SE_SELECT);
2018-02-06 16:09:39 -06:00
BtlController_EmitTwoReturnValues(1, B_ACTION_CANCEL_PARTNER, 0);
PlayerBufferExecCompleted();
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(START_BUTTON))
{
SwapHpBarsWithHpText();
}
2020-12-23 14:04:14 -03:00
else if (B_ENABLE_DEBUG && gMain.newKeys & SELECT_BUTTON)
2018-07-13 23:00:56 +02:00
{
BtlController_EmitTwoReturnValues(1, B_ACTION_DEBUG, 0);
PlayerBufferExecCompleted();
}
}
2021-01-22 20:03:21 -05:00
static void UnusedEndBounceEffect(void)
{
2018-06-20 23:07:51 +02:00
EndBounceEffect(gActiveBattler, BOUNCE_HEALTHBOX);
EndBounceEffect(gActiveBattler, BOUNCE_MON);
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseTarget;
}
2017-10-22 18:43:15 +02:00
static void HandleInputChooseTarget(void)
{
s32 i;
static const u8 identities[MAX_BATTLERS_COUNT] = {B_POSITION_PLAYER_LEFT, B_POSITION_PLAYER_RIGHT, B_POSITION_OPPONENT_RIGHT, B_POSITION_OPPONENT_LEFT};
2018-11-24 21:06:53 +01:00
u16 move = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_MOVE1 + gMoveSelectionCursor[gActiveBattler]);
2018-06-20 23:07:51 +02:00
DoBounceEffect(gMultiUsePlayerCursor, BOUNCE_HEALTHBOX, 15, 1);
for (i = 0; i < gBattlersCount; i++)
{
if (i != gMultiUsePlayerCursor)
EndBounceEffect(i, BOUNCE_HEALTHBOX);
}
2020-09-04 21:11:55 -04:00
if (JOY_HELD(DPAD_ANY) && gSaveBlock2Ptr->optionsButtonMode == OPTIONS_BUTTON_MODE_L_EQUALS_A)
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames++;
else
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames = 0;
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
{
PlaySE(SE_SELECT);
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_HideAsMoveTarget;
if (gBattleStruct->mega.playerSelect)
BtlController_EmitTwoReturnValues(1, 10, gMoveSelectionCursor[gActiveBattler] | RET_MEGA_EVOLUTION | (gMultiUsePlayerCursor << 8));
2018-09-16 21:08:49 +02:00
else
BtlController_EmitTwoReturnValues(1, 10, gMoveSelectionCursor[gActiveBattler] | (gMultiUsePlayerCursor << 8));
2018-06-20 23:07:51 +02:00
EndBounceEffect(gMultiUsePlayerCursor, BOUNCE_HEALTHBOX);
HideMegaTriggerSprite();
PlayerBufferExecCompleted();
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(B_BUTTON) || gPlayerDpadHoldFrames > 59)
{
PlaySE(SE_SELECT);
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_HideAsMoveTarget;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseMove;
2018-06-20 23:07:51 +02:00
DoBounceEffect(gActiveBattler, BOUNCE_HEALTHBOX, 7, 1);
DoBounceEffect(gActiveBattler, BOUNCE_MON, 7, 1);
EndBounceEffect(gMultiUsePlayerCursor, BOUNCE_HEALTHBOX);
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_LEFT | DPAD_UP))
{
PlaySE(SE_SELECT);
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_HideAsMoveTarget;
2018-11-24 21:06:53 +01:00
if (gBattleMoves[move].target == (MOVE_TARGET_USER | MOVE_TARGET_ALLY))
{
gMultiUsePlayerCursor ^= BIT_FLANK;
}
else
{
do
{
2018-11-24 21:06:53 +01:00
u8 currSelIdentity = GetBattlerPosition(gMultiUsePlayerCursor);
for (i = 0; i < MAX_BATTLERS_COUNT; i++)
{
if (currSelIdentity == identities[i])
break;
}
do
{
if (--i < 0)
i = MAX_BATTLERS_COUNT - 1;
2018-11-24 21:06:53 +01:00
gMultiUsePlayerCursor = GetBattlerAtPosition(identities[i]);
} while (gMultiUsePlayerCursor == gBattlersCount);
2018-11-24 21:06:53 +01:00
i = 0;
switch (GetBattlerPosition(gMultiUsePlayerCursor))
{
case B_POSITION_PLAYER_LEFT:
case B_POSITION_PLAYER_RIGHT:
if (gActiveBattler != gMultiUsePlayerCursor)
i++;
else if (gBattleMoves[move].target & MOVE_TARGET_USER_OR_SELECTED)
i++;
break;
case B_POSITION_OPPONENT_LEFT:
case B_POSITION_OPPONENT_RIGHT:
i++;
2018-11-24 21:06:53 +01:00
break;
}
2018-11-24 21:06:53 +01:00
if (gAbsentBattlerFlags & gBitTable[gMultiUsePlayerCursor])
i = 0;
} while (i == 0);
}
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_ShowAsMoveTarget;
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_RIGHT | DPAD_DOWN))
{
PlaySE(SE_SELECT);
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_HideAsMoveTarget;
2018-11-24 21:06:53 +01:00
if (gBattleMoves[move].target == (MOVE_TARGET_USER | MOVE_TARGET_ALLY))
{
gMultiUsePlayerCursor ^= BIT_FLANK;
}
else
{
do
{
2018-11-24 21:06:53 +01:00
u8 currSelIdentity = GetBattlerPosition(gMultiUsePlayerCursor);
for (i = 0; i < MAX_BATTLERS_COUNT; i++)
{
if (currSelIdentity == identities[i])
break;
}
do
{
if (++i > 3)
i = 0;
gMultiUsePlayerCursor = GetBattlerAtPosition(identities[i]);
} while (gMultiUsePlayerCursor == gBattlersCount);
2018-11-24 21:06:53 +01:00
i = 0;
switch (GetBattlerPosition(gMultiUsePlayerCursor))
{
case B_POSITION_PLAYER_LEFT:
case B_POSITION_PLAYER_RIGHT:
if (gActiveBattler != gMultiUsePlayerCursor)
i++;
else if (gBattleMoves[move].target & MOVE_TARGET_USER_OR_SELECTED)
i++;
break;
case B_POSITION_OPPONENT_LEFT:
case B_POSITION_OPPONENT_RIGHT:
i++;
2018-11-24 21:06:53 +01:00
break;
}
if (gAbsentBattlerFlags & gBitTable[gMultiUsePlayerCursor])
i = 0;
} while (i == 0);
}
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_ShowAsMoveTarget;
}
}
static void HideShownTargets(void)
{
s32 i;
for (i = 0; i < MAX_BATTLERS_COUNT; i++)
{
if (IsBattlerAlive(i) && gBattleSpritesDataPtr->healthBoxesData[i].healthboxIsBouncing && i != gActiveBattler)
{
2020-07-15 10:13:19 +02:00
gSprites[gBattlerSpriteIds[i]].callback = SpriteCb_HideAsMoveTarget;
EndBounceEffect(i, BOUNCE_HEALTHBOX);
}
}
}
static void HandleInputShowTargets(void)
{
2020-09-04 21:11:55 -04:00
if (JOY_HELD(DPAD_ANY) && gSaveBlock2Ptr->optionsButtonMode == OPTIONS_BUTTON_MODE_L_EQUALS_A)
gPlayerDpadHoldFrames++;
else
gPlayerDpadHoldFrames = 0;
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
{
PlaySE(SE_SELECT);
HideShownTargets();
if (gBattleStruct->mega.playerSelect)
BtlController_EmitTwoReturnValues(1, 10, gMoveSelectionCursor[gActiveBattler] | RET_MEGA_EVOLUTION | (gMultiUsePlayerCursor << 8));
else
BtlController_EmitTwoReturnValues(1, 10, gMoveSelectionCursor[gActiveBattler] | (gMultiUsePlayerCursor << 8));
HideMegaTriggerSprite();
PlayerBufferExecCompleted();
}
else if (gMain.newKeys & B_BUTTON || gPlayerDpadHoldFrames > 59)
{
PlaySE(SE_SELECT);
HideShownTargets();
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseMove;
DoBounceEffect(gActiveBattler, BOUNCE_HEALTHBOX, 7, 1);
DoBounceEffect(gActiveBattler, BOUNCE_MON, 7, 1);
}
}
static void TryShowAsTarget(u32 battlerId)
{
if (IsBattlerAlive(battlerId))
{
DoBounceEffect(battlerId, BOUNCE_HEALTHBOX, 15, 1);
2020-07-15 10:13:19 +02:00
gSprites[gBattlerSpriteIds[battlerId]].callback = SpriteCb_ShowAsMoveTarget;
}
}
2017-10-22 18:43:15 +02:00
static void HandleInputChooseMove(void)
{
u8 moveTarget;
u32 canSelectTarget = 0;
struct ChooseMoveStruct *moveInfo = (struct ChooseMoveStruct*)(&gBattleResources->bufferA[gActiveBattler][4]);
2018-06-28 21:06:32 +02:00
if (gMain.heldKeys & DPAD_ANY && gSaveBlock2Ptr->optionsButtonMode == OPTIONS_BUTTON_MODE_L_EQUALS_A)
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames++;
else
2017-10-27 13:29:18 +02:00
gPlayerDpadHoldFrames = 0;
if (gMain.newKeys & A_BUTTON)
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
if (moveInfo->moves[gMoveSelectionCursor[gActiveBattler]] == MOVE_CURSE)
{
2018-11-17 12:10:24 +01:00
if (moveInfo->monType1 != TYPE_GHOST && moveInfo->monType2 != TYPE_GHOST && moveInfo->monType3 != TYPE_GHOST)
2018-02-26 13:24:46 +01:00
moveTarget = MOVE_TARGET_USER;
else
moveTarget = MOVE_TARGET_SELECTED;
}
else
{
2018-02-05 19:46:59 -06:00
moveTarget = gBattleMoves[moveInfo->moves[gMoveSelectionCursor[gActiveBattler]]].target;
}
2018-02-26 13:24:46 +01:00
if (moveTarget & MOVE_TARGET_USER)
2018-02-05 19:46:59 -06:00
gMultiUsePlayerCursor = gActiveBattler;
else
2018-02-05 19:46:59 -06:00
gMultiUsePlayerCursor = GetBattlerAtPosition((GetBattlerPosition(gActiveBattler) & BIT_SIDE) ^ BIT_SIDE);
if (!gBattleResources->bufferA[gActiveBattler][1]) // not a double battle
{
if (moveTarget & MOVE_TARGET_USER_OR_SELECTED && !gBattleResources->bufferA[gActiveBattler][2])
canSelectTarget = 1;
}
else // double battle
{
2018-11-25 12:59:22 +01:00
if (!(moveTarget & (MOVE_TARGET_RANDOM | MOVE_TARGET_BOTH | MOVE_TARGET_DEPENDS | MOVE_TARGET_FOES_AND_ALLY | MOVE_TARGET_OPPONENTS_FIELD | MOVE_TARGET_USER | MOVE_TARGET_ALLY)))
canSelectTarget = 1; // either selected or user
2018-11-24 21:06:53 +01:00
if (moveTarget == (MOVE_TARGET_USER | MOVE_TARGET_ALLY) && IsBattlerAlive(BATTLE_PARTNER(gActiveBattler)))
canSelectTarget = 1;
2018-02-05 19:46:59 -06:00
if (moveInfo->currentPp[gMoveSelectionCursor[gActiveBattler]] == 0)
{
canSelectTarget = 0;
}
2018-02-26 13:24:46 +01:00
else if (!(moveTarget & (MOVE_TARGET_USER | MOVE_TARGET_USER_OR_SELECTED)) && CountAliveMonsInBattle(BATTLE_ALIVE_EXCEPT_ACTIVE) <= 1)
{
2018-02-05 19:46:59 -06:00
gMultiUsePlayerCursor = GetDefaultMoveTarget(gActiveBattler);
canSelectTarget = 0;
}
// Show all available targets for multi-target moves
if (B_SHOW_TARGETS && moveTarget & (MOVE_TARGET_OPPONENTS_FIELD | MOVE_TARGET_BOTH | MOVE_TARGET_FOES_AND_ALLY))
{
TryShowAsTarget(gMultiUsePlayerCursor);
TryShowAsTarget(BATTLE_PARTNER(gMultiUsePlayerCursor));
if (moveTarget & MOVE_TARGET_FOES_AND_ALLY)
TryShowAsTarget(BATTLE_PARTNER(gActiveBattler));
canSelectTarget = 2;
}
}
if (canSelectTarget == 0)
{
if (gBattleStruct->mega.playerSelect)
BtlController_EmitTwoReturnValues(1, 10, gMoveSelectionCursor[gActiveBattler] | RET_MEGA_EVOLUTION | (gMultiUsePlayerCursor << 8));
2018-09-16 21:08:49 +02:00
else
BtlController_EmitTwoReturnValues(1, 10, gMoveSelectionCursor[gActiveBattler] | (gMultiUsePlayerCursor << 8));
HideMegaTriggerSprite();
PlayerBufferExecCompleted();
}
else if (canSelectTarget == 1)
{
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseTarget;
2018-02-26 13:24:46 +01:00
if (moveTarget & (MOVE_TARGET_USER | MOVE_TARGET_USER_OR_SELECTED))
2018-02-05 19:46:59 -06:00
gMultiUsePlayerCursor = gActiveBattler;
else if (gAbsentBattlerFlags & gBitTable[GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT)])
gMultiUsePlayerCursor = GetBattlerAtPosition(B_POSITION_OPPONENT_RIGHT);
else
2018-02-05 19:46:59 -06:00
gMultiUsePlayerCursor = GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT);
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[gMultiUsePlayerCursor]].callback = SpriteCb_ShowAsMoveTarget;
}
else
{
gBattlerControllerFuncs[gActiveBattler] = HandleInputShowTargets;
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(B_BUTTON) || gPlayerDpadHoldFrames > 59)
{
PlaySE(SE_SELECT);
gBattleStruct->mega.playerSelect = FALSE;
2018-02-06 13:48:02 -06:00
BtlController_EmitTwoReturnValues(1, 10, 0xFFFF);
HideMegaTriggerSprite();
PlayerBufferExecCompleted();
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_LEFT))
{
2018-02-05 19:46:59 -06:00
if (gMoveSelectionCursor[gActiveBattler] & 1)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 1;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_RIGHT))
{
2018-02-05 19:46:59 -06:00
if (!(gMoveSelectionCursor[gActiveBattler] & 1)
&& (gMoveSelectionCursor[gActiveBattler] ^ 1) < gNumberOfMovesToChoose)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 1;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_UP))
{
2018-02-05 19:46:59 -06:00
if (gMoveSelectionCursor[gActiveBattler] & 2)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 2;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_DOWN))
{
2018-02-05 19:46:59 -06:00
if (!(gMoveSelectionCursor[gActiveBattler] & 2)
&& (gMoveSelectionCursor[gActiveBattler] ^ 2) < gNumberOfMovesToChoose)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 2;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(SELECT_BUTTON))
{
if (gNumberOfMovesToChoose > 1 && !(gBattleTypeFlags & BATTLE_TYPE_LINK))
{
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 29);
2018-02-05 19:46:59 -06:00
if (gMoveSelectionCursor[gActiveBattler] != 0)
gMultiUsePlayerCursor = 0;
else
2018-02-05 19:46:59 -06:00
gMultiUsePlayerCursor = gMoveSelectionCursor[gActiveBattler] + 1;
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 27);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gText_BattleSwitchWhich, 0xB);
2018-11-22 01:10:50 +00:00
gBattlerControllerFuncs[gActiveBattler] = HandleMoveSwitching;
}
}
2018-09-16 21:08:49 +02:00
else if (gMain.newKeys & START_BUTTON)
{
if (CanMegaEvolve(gActiveBattler))
2018-09-16 21:08:49 +02:00
{
gBattleStruct->mega.playerSelect ^= 1;
2018-09-20 20:59:00 +02:00
ChangeMegaTriggerSprite(gBattleStruct->mega.triggerSpriteId, gBattleStruct->mega.playerSelect);
2018-09-16 21:08:49 +02:00
PlaySE(SE_SELECT);
}
}
}
2021-01-22 20:03:21 -05:00
static u32 HandleMoveInputUnused(void)
{
u32 var = 0;
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
{
PlaySE(SE_SELECT);
var = 1;
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(B_BUTTON))
{
PlaySE(SE_SELECT);
gBattle_BG0_X = 0;
2021-04-15 02:04:01 -04:00
gBattle_BG0_Y = DISPLAY_HEIGHT * 2;
var = 0xFF;
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(DPAD_LEFT) && gMoveSelectionCursor[gActiveBattler] & 1)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 1;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(DPAD_RIGHT) && !(gMoveSelectionCursor[gActiveBattler] & 1)
2018-02-05 19:46:59 -06:00
&& (gMoveSelectionCursor[gActiveBattler] ^ 1) < gNumberOfMovesToChoose)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 1;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(DPAD_UP) && gMoveSelectionCursor[gActiveBattler] & 2)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 2;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(DPAD_DOWN) && !(gMoveSelectionCursor[gActiveBattler] & 2)
2018-02-05 19:46:59 -06:00
&& (gMoveSelectionCursor[gActiveBattler] ^ 2) < gNumberOfMovesToChoose)
{
2018-02-05 19:46:59 -06:00
MoveSelectionDestroyCursorAt(gMoveSelectionCursor[gActiveBattler]);
gMoveSelectionCursor[gActiveBattler] ^= 2;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
}
return var;
}
2018-11-22 01:10:50 +00:00
static void HandleMoveSwitching(void)
{
2019-09-08 11:53:48 -04:00
u8 perMovePPBonuses[MAX_MON_MOVES];
struct ChooseMoveStruct moveStruct;
u8 totalPPBonuses;
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON | SELECT_BUTTON))
{
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
if (gMoveSelectionCursor[gActiveBattler] != gMultiUsePlayerCursor)
{
struct ChooseMoveStruct *moveInfo = (struct ChooseMoveStruct*)(&gBattleResources->bufferA[gActiveBattler][4]);
s32 i;
// swap moves and pp
2018-02-05 19:46:59 -06:00
i = moveInfo->moves[gMoveSelectionCursor[gActiveBattler]];
moveInfo->moves[gMoveSelectionCursor[gActiveBattler]] = moveInfo->moves[gMultiUsePlayerCursor];
moveInfo->moves[gMultiUsePlayerCursor] = i;
2018-02-05 19:46:59 -06:00
i = moveInfo->currentPp[gMoveSelectionCursor[gActiveBattler]];
moveInfo->currentPp[gMoveSelectionCursor[gActiveBattler]] = moveInfo->currentPp[gMultiUsePlayerCursor];
moveInfo->currentPp[gMultiUsePlayerCursor] = i;
2018-02-05 19:46:59 -06:00
i = moveInfo->maxPp[gMoveSelectionCursor[gActiveBattler]];
moveInfo->maxPp[gMoveSelectionCursor[gActiveBattler]] = moveInfo->maxPp[gMultiUsePlayerCursor];
moveInfo->maxPp[gMultiUsePlayerCursor] = i;
if (gDisableStructs[gActiveBattler].mimickedMoves & gBitTable[gMoveSelectionCursor[gActiveBattler]])
{
gDisableStructs[gActiveBattler].mimickedMoves &= (~gBitTable[gMoveSelectionCursor[gActiveBattler]]);
gDisableStructs[gActiveBattler].mimickedMoves |= gBitTable[gMultiUsePlayerCursor];
}
MoveSelectionDisplayMoveNames();
for (i = 0; i < MAX_MON_MOVES; i++)
2018-02-05 19:46:59 -06:00
perMovePPBonuses[i] = (gBattleMons[gActiveBattler].ppBonuses & (3 << (i * 2))) >> (i * 2);
2018-02-05 19:46:59 -06:00
totalPPBonuses = perMovePPBonuses[gMoveSelectionCursor[gActiveBattler]];
perMovePPBonuses[gMoveSelectionCursor[gActiveBattler]] = perMovePPBonuses[gMultiUsePlayerCursor];
perMovePPBonuses[gMultiUsePlayerCursor] = totalPPBonuses;
totalPPBonuses = 0;
for (i = 0; i < MAX_MON_MOVES; i++)
totalPPBonuses |= perMovePPBonuses[i] << (i * 2);
2018-02-05 19:46:59 -06:00
gBattleMons[gActiveBattler].ppBonuses = totalPPBonuses;
for (i = 0; i < MAX_MON_MOVES; i++)
{
2018-02-05 19:46:59 -06:00
gBattleMons[gActiveBattler].moves[i] = moveInfo->moves[i];
gBattleMons[gActiveBattler].pp[i] = moveInfo->currentPp[i];
}
2018-02-05 19:46:59 -06:00
if (!(gBattleMons[gActiveBattler].status2 & STATUS2_TRANSFORMED))
{
for (i = 0; i < MAX_MON_MOVES; i++)
{
2018-02-06 13:48:02 -06:00
moveStruct.moves[i] = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_MOVE1 + i);
moveStruct.currentPp[i] = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_PP1 + i);
}
2018-02-06 13:48:02 -06:00
totalPPBonuses = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_PP_BONUSES);
for (i = 0; i < MAX_MON_MOVES; i++)
perMovePPBonuses[i] = (totalPPBonuses & (3 << (i * 2))) >> (i * 2);
2018-02-05 19:46:59 -06:00
i = moveStruct.moves[gMoveSelectionCursor[gActiveBattler]];
moveStruct.moves[gMoveSelectionCursor[gActiveBattler]] = moveStruct.moves[gMultiUsePlayerCursor];
moveStruct.moves[gMultiUsePlayerCursor] = i;
2018-02-05 19:46:59 -06:00
i = moveStruct.currentPp[gMoveSelectionCursor[gActiveBattler]];
moveStruct.currentPp[gMoveSelectionCursor[gActiveBattler]] = moveStruct.currentPp[gMultiUsePlayerCursor];
moveStruct.currentPp[gMultiUsePlayerCursor] = i;
2018-02-05 19:46:59 -06:00
totalPPBonuses = perMovePPBonuses[gMoveSelectionCursor[gActiveBattler]];
perMovePPBonuses[gMoveSelectionCursor[gActiveBattler]] = perMovePPBonuses[gMultiUsePlayerCursor];
perMovePPBonuses[gMultiUsePlayerCursor] = totalPPBonuses;
totalPPBonuses = 0;
for (i = 0; i < MAX_MON_MOVES; i++)
totalPPBonuses |= perMovePPBonuses[i] << (i * 2);
for (i = 0; i < MAX_MON_MOVES; i++)
{
2018-02-06 13:48:02 -06:00
SetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_MOVE1 + i, &moveStruct.moves[i]);
SetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_PP1 + i, &moveStruct.currentPp[i]);
}
2018-02-06 13:48:02 -06:00
SetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_PP_BONUSES, &totalPPBonuses);
}
}
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseMove;
2018-02-05 19:46:59 -06:00
gMoveSelectionCursor[gActiveBattler] = gMultiUsePlayerCursor;
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
MoveSelectionDisplayPpString();
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(B_BUTTON | SELECT_BUTTON))
{
PlaySE(SE_SELECT);
MoveSelectionDestroyCursorAt(gMultiUsePlayerCursor);
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseMove;
MoveSelectionDisplayPpString();
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_LEFT))
{
if (gMultiUsePlayerCursor & 1)
{
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 29);
else
MoveSelectionDestroyCursorAt(gMultiUsePlayerCursor);
gMultiUsePlayerCursor ^= 1;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 0);
else
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 27);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_RIGHT))
{
if (!(gMultiUsePlayerCursor & 1) && (gMultiUsePlayerCursor ^ 1) < gNumberOfMovesToChoose)
{
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 29);
else
MoveSelectionDestroyCursorAt(gMultiUsePlayerCursor);
gMultiUsePlayerCursor ^= 1;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 0);
else
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 27);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_UP))
{
if (gMultiUsePlayerCursor & 2)
{
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 29);
else
MoveSelectionDestroyCursorAt(gMultiUsePlayerCursor);
gMultiUsePlayerCursor ^= 2;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 0);
else
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 27);
}
}
2020-09-04 21:11:55 -04:00
else if (JOY_NEW(DPAD_DOWN))
{
if (!(gMultiUsePlayerCursor & 2) && (gMultiUsePlayerCursor ^ 2) < gNumberOfMovesToChoose)
{
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 29);
else
MoveSelectionDestroyCursorAt(gMultiUsePlayerCursor);
gMultiUsePlayerCursor ^= 2;
PlaySE(SE_SELECT);
2018-02-05 19:46:59 -06:00
if (gMultiUsePlayerCursor == gMoveSelectionCursor[gActiveBattler])
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 0);
else
MoveSelectionCreateCursorAt(gMultiUsePlayerCursor, 27);
}
}
}
2021-01-22 20:03:21 -05:00
static void SetLinkBattleEndCallbacks(void)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == 0)
{
if (gReceivedRemoteLinkPlayers == 0)
{
2020-08-20 18:02:00 -04:00
m4aSongNumStop(SE_LOW_HEALTH);
gMain.inBattle = 0;
gMain.callback1 = gPreBattleCallback1;
2021-01-22 20:03:21 -05:00
SetMainCallback2(CB2_InitEndLinkBattle);
2018-01-16 15:12:38 -06:00
if (gBattleOutcome == B_OUTCOME_WON)
2018-02-27 20:40:09 +01:00
TryPutLinkBattleTvShowOnAir();
FreeAllWindowBuffers();
}
}
else
{
2018-12-31 02:22:21 -06:00
if (IsLinkTaskFinished())
{
2020-08-20 18:02:00 -04:00
m4aSongNumStop(SE_LOW_HEALTH);
gMain.inBattle = 0;
gMain.callback1 = gPreBattleCallback1;
2021-01-22 20:03:21 -05:00
SetMainCallback2(CB2_InitEndLinkBattle);
2018-01-16 15:12:38 -06:00
if (gBattleOutcome == B_OUTCOME_WON)
2018-02-27 20:40:09 +01:00
TryPutLinkBattleTvShowOnAir();
FreeAllWindowBuffers();
}
}
}
2021-01-22 20:03:21 -05:00
// Despite handling link battles separately, this is only ever used by link battles
void SetBattleEndCallbacks(void)
{
if (!gPaletteFade.active)
{
if (gBattleTypeFlags & BATTLE_TYPE_LINK)
{
2018-12-31 02:22:21 -06:00
if (IsLinkTaskFinished())
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == 0)
2020-08-13 03:09:47 -04:00
SetCloseLinkCallback();
else
2020-08-13 03:09:47 -04:00
SetLinkStandbyCallback();
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = SetLinkBattleEndCallbacks;
}
}
else
{
2020-08-20 18:02:00 -04:00
m4aSongNumStop(SE_LOW_HEALTH);
gMain.inBattle = 0;
gMain.callback1 = gPreBattleCallback1;
SetMainCallback2(gMain.savedCallback);
}
}
}
static void CompleteOnBattlerSpriteCallbackDummy(void)
{
2018-02-06 13:48:02 -06:00
if (gSprites[gBattlerSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy)
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void CompleteOnBankSpriteCallbackDummy2(void)
{
2018-02-06 13:48:02 -06:00
if (gSprites[gBattlerSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy)
PlayerBufferExecCompleted();
}
2021-01-22 20:03:21 -05:00
static void FreeTrainerSpriteAfterSlide(void)
{
2018-02-06 13:48:02 -06:00
if (gSprites[gBattlerSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy)
{
2021-01-22 20:03:21 -05:00
BattleGfxSfxDummy3(gSaveBlock2Ptr->playerGender);
2018-02-06 13:48:02 -06:00
FreeSpriteOamMatrix(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
DestroySprite(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
PlayerBufferExecCompleted();
}
}
2021-01-22 20:03:21 -05:00
static void Intro_DelayAndEnd(void)
{
2021-01-22 20:03:21 -05:00
if (--gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].introEndDelay == (u8)-1)
{
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].introEndDelay = 0;
PlayerBufferExecCompleted();
}
}
2021-01-22 20:03:21 -05:00
static void Intro_WaitForShinyAnimAndHealthbox(void)
{
2021-01-22 20:03:21 -05:00
bool8 healthboxAnimDone = FALSE;
2021-01-22 20:03:21 -05:00
// Check if healthbox has finished sliding in
if (!IsDoubleBattle() || (IsDoubleBattle() && (gBattleTypeFlags & BATTLE_TYPE_MULTI)))
{
2018-02-06 13:48:02 -06:00
if (gSprites[gHealthboxSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy)
2021-01-22 20:03:21 -05:00
healthboxAnimDone = TRUE;
}
else
{
2018-02-06 13:48:02 -06:00
if (gSprites[gHealthboxSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy
&& gSprites[gHealthboxSpriteIds[gActiveBattler ^ BIT_FLANK]].callback == SpriteCallbackDummy)
2021-01-22 20:03:21 -05:00
healthboxAnimDone = TRUE;
}
2021-01-22 20:03:21 -05:00
// If healthbox and shiny anim are done
if (healthboxAnimDone && gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].finishedShinyMonAnim
2020-08-30 15:11:44 -04:00
&& gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].finishedShinyMonAnim)
{
2021-01-22 20:03:21 -05:00
// Reset shiny anim (even if it didn't occur)
2020-08-30 15:11:44 -04:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].triedShinyMonAnim = FALSE;
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].finishedShinyMonAnim = FALSE;
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].triedShinyMonAnim = FALSE;
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].finishedShinyMonAnim = FALSE;
2021-01-22 20:03:21 -05:00
FreeSpriteTilesByTag(ANIM_TAG_GOLD_STARS);
FreeSpritePaletteByTag(ANIM_TAG_GOLD_STARS);
2018-02-06 13:48:02 -06:00
HandleLowHpMusicChange(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
if (IsDoubleBattle())
2018-02-06 13:48:02 -06:00
HandleLowHpMusicChange(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler ^ BIT_FLANK]], gActiveBattler ^ BIT_FLANK);
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].introEndDelay = 3;
gBattlerControllerFuncs[gActiveBattler] = Intro_DelayAndEnd;
}
}
2021-01-22 20:03:21 -05:00
static void Intro_TryShinyAnimShowHealthbox(void)
{
2021-01-22 20:03:21 -05:00
bool32 bgmRestored = FALSE;
bool32 battlerAnimsDone = FALSE;
2021-01-22 20:03:21 -05:00
// Start shiny animation if applicable for 1st pokemon
2021-06-03 23:17:44 +02:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].triedShinyMonAnim
2020-08-30 15:11:44 -04:00
&& !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].ballAnimActive)
TryShinyAnimation(gActiveBattler, &gPlayerParty[gBattlerPartyIndexes[gActiveBattler]]);
2020-08-30 15:11:44 -04:00
2021-01-22 20:03:21 -05:00
// Start shiny animation if applicable for 2nd pokemon
2020-09-08 12:48:28 +02:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].triedShinyMonAnim
2020-08-30 15:11:44 -04:00
&& !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].ballAnimActive)
TryShinyAnimation(gActiveBattler ^ BIT_FLANK, &gPlayerParty[gBattlerPartyIndexes[gActiveBattler ^ BIT_FLANK]]);
2021-01-22 20:03:21 -05:00
// Show healthbox after ball anim
2021-06-03 23:17:44 +02:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].ballAnimActive
2021-01-22 20:03:21 -05:00
&& !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].ballAnimActive)
{
2021-01-22 20:03:21 -05:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].healthboxSlideInStarted)
{
if (IsDoubleBattle() && !(gBattleTypeFlags & BATTLE_TYPE_MULTI))
{
2018-02-06 13:48:02 -06:00
UpdateHealthboxAttribute(gHealthboxSpriteIds[gActiveBattler ^ BIT_FLANK], &gPlayerParty[gBattlerPartyIndexes[gActiveBattler ^ BIT_FLANK]], HEALTHBOX_ALL);
2021-01-22 20:03:21 -05:00
StartHealthboxSlideIn(gActiveBattler ^ BIT_FLANK);
2018-02-06 13:48:02 -06:00
SetHealthboxSpriteVisible(gHealthboxSpriteIds[gActiveBattler ^ BIT_FLANK]);
}
2018-02-06 13:48:02 -06:00
UpdateHealthboxAttribute(gHealthboxSpriteIds[gActiveBattler], &gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], HEALTHBOX_ALL);
2021-01-22 20:03:21 -05:00
StartHealthboxSlideIn(gActiveBattler);
2018-02-06 13:48:02 -06:00
SetHealthboxSpriteVisible(gHealthboxSpriteIds[gActiveBattler]);
}
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].healthboxSlideInStarted = TRUE;
}
2021-01-22 20:03:21 -05:00
// Restore bgm after cry has played and healthbox anim is started
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].waitForCry
&& gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].healthboxSlideInStarted
&& !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler ^ BIT_FLANK].waitForCry
&& !IsCryPlayingOrClearCrySongs())
{
2021-01-22 20:03:21 -05:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].bgmRestored)
{
if (gBattleTypeFlags & BATTLE_TYPE_MULTI && gBattleTypeFlags & BATTLE_TYPE_LINK)
m4aMPlayContinue(&gMPlayInfo_BGM);
else
m4aMPlayVolumeControl(&gMPlayInfo_BGM, 0xFFFF, 0x100);
}
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].bgmRestored = TRUE;
bgmRestored = TRUE;
}
2021-01-22 20:03:21 -05:00
// Wait for battler anims
if (!IsDoubleBattle() || (IsDoubleBattle() && (gBattleTypeFlags & BATTLE_TYPE_MULTI)))
{
2021-01-22 20:03:21 -05:00
if (gSprites[gBattleControllerData[gActiveBattler]].callback == SpriteCallbackDummy
2018-02-06 13:48:02 -06:00
&& gSprites[gBattlerSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy)
{
2021-01-22 20:03:21 -05:00
battlerAnimsDone = TRUE;
}
}
else
{
2021-01-22 20:03:21 -05:00
if (gSprites[gBattleControllerData[gActiveBattler]].callback == SpriteCallbackDummy
2018-02-06 13:48:02 -06:00
&& gSprites[gBattlerSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy
2021-01-22 20:03:21 -05:00
&& gSprites[gBattleControllerData[gActiveBattler ^ BIT_FLANK]].callback == SpriteCallbackDummy
2018-02-06 13:48:02 -06:00
&& gSprites[gBattlerSpriteIds[gActiveBattler ^ BIT_FLANK]].callback == SpriteCallbackDummy)
{
2021-01-22 20:03:21 -05:00
battlerAnimsDone = TRUE;
}
}
2021-01-22 20:03:21 -05:00
// Clean up
if (bgmRestored && battlerAnimsDone)
{
if (IsDoubleBattle() && !(gBattleTypeFlags & BATTLE_TYPE_MULTI))
2021-01-22 20:03:21 -05:00
DestroySprite(&gSprites[gBattleControllerData[gActiveBattler ^ BIT_FLANK]]);
DestroySprite(&gSprites[gBattleControllerData[gActiveBattler]]);
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->animationData->introAnimActive = FALSE;
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].bgmRestored = FALSE;
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].healthboxSlideInStarted = FALSE;
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = Intro_WaitForShinyAnimAndHealthbox;
}
}
2021-01-22 20:03:21 -05:00
static void SwitchIn_CleanShinyAnimShowSubstitute(void)
{
2018-02-06 13:48:02 -06:00
if (gSprites[gHealthboxSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy
2020-08-30 15:11:44 -04:00
&& gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].finishedShinyMonAnim
2018-02-06 13:48:02 -06:00
&& gSprites[gBattlerSpriteIds[gActiveBattler]].callback == SpriteCallbackDummy)
{
2018-02-05 19:46:59 -06:00
CopyBattleSpriteInvisibility(gActiveBattler);
2021-06-03 23:17:44 +02:00
2021-01-22 20:03:21 -05:00
// Reset shiny anim (even if it didn't occur)
2020-08-30 15:11:44 -04:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].triedShinyMonAnim = FALSE;
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].finishedShinyMonAnim = FALSE;
2021-01-22 20:03:21 -05:00
FreeSpriteTilesByTag(ANIM_TAG_GOLD_STARS);
FreeSpritePaletteByTag(ANIM_TAG_GOLD_STARS);
2021-01-22 20:03:21 -05:00
// Check if Substitute should be shown
2018-02-06 16:09:39 -06:00
if (gBattleSpritesDataPtr->battlerData[gActiveBattler].behindSubstitute)
2018-02-05 19:46:59 -06:00
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gActiveBattler, B_ANIM_MON_TO_SUBSTITUTE);
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = SwitchIn_HandleSoundAndEnd;
}
}
2021-01-22 20:03:21 -05:00
static void SwitchIn_HandleSoundAndEnd(void)
{
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive
&& !IsCryPlayingOrClearCrySongs())
{
m4aMPlayVolumeControl(&gMPlayInfo_BGM, 0xFFFF, 0x100);
2018-02-06 13:48:02 -06:00
HandleLowHpMusicChange(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
PlayerBufferExecCompleted();
}
}
2021-01-22 20:03:21 -05:00
static void SwitchIn_TryShinyAnimShowHealthbox(void)
{
2021-01-22 20:03:21 -05:00
// Start shiny animation if applicable
2020-08-30 15:11:44 -04:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].triedShinyMonAnim
2018-02-05 19:46:59 -06:00
&& !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].ballAnimActive)
TryShinyAnimation(gActiveBattler, &gPlayerParty[gBattlerPartyIndexes[gActiveBattler]]);
2021-01-22 20:03:21 -05:00
// Wait for ball anim, then show healthbox
if (gSprites[gBattleControllerData[gActiveBattler]].callback == SpriteCallbackDummy
2018-02-05 19:46:59 -06:00
&& !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].ballAnimActive)
{
2021-01-22 20:03:21 -05:00
DestroySprite(&gSprites[gBattleControllerData[gActiveBattler]]);
2018-02-06 13:48:02 -06:00
UpdateHealthboxAttribute(gHealthboxSpriteIds[gActiveBattler], &gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], HEALTHBOX_ALL);
2021-01-22 20:03:21 -05:00
StartHealthboxSlideIn(gActiveBattler);
2018-02-06 13:48:02 -06:00
SetHealthboxSpriteVisible(gHealthboxSpriteIds[gActiveBattler]);
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = SwitchIn_CleanShinyAnimShowSubstitute;
}
}
2021-01-22 20:03:21 -05:00
void Task_PlayerController_RestoreBgmAfterCry(u8 taskId)
{
if (!IsCryPlayingOrClearCrySongs())
{
m4aMPlayVolumeControl(&gMPlayInfo_BGM, 0xFFFF, 0x100);
DestroyTask(taskId);
}
}
2017-10-22 18:43:15 +02:00
static void CompleteOnHealthbarDone(void)
{
2018-06-19 00:43:15 +02:00
s16 hpValue = MoveBattleBar(gActiveBattler, gHealthboxSpriteIds[gActiveBattler], HEALTH_BAR, 0);
2018-02-06 13:48:02 -06:00
SetHealthboxSpriteVisible(gHealthboxSpriteIds[gActiveBattler]);
if (hpValue != -1)
{
2018-02-06 13:48:02 -06:00
UpdateHpTextInHealthbox(gHealthboxSpriteIds[gActiveBattler], hpValue, HP_CURRENT);
}
else
{
2018-02-06 13:48:02 -06:00
HandleLowHpMusicChange(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
PlayerBufferExecCompleted();
}
}
2017-10-22 18:43:15 +02:00
static void CompleteOnInactiveTextPrinter(void)
{
if (!IsTextPrinterActive(0))
PlayerBufferExecCompleted();
}
2021-06-03 23:17:44 +02:00
#define tExpTask_monId data[0]
#define tExpTask_battler data[2]
#define tExpTask_gainedExp_1 data[3]
#define tExpTask_gainedExp_2 data[4] // Stored as two half-words containing a word.
#define tExpTask_frames data[10]
static s32 GetTaskExpValue(u8 taskId)
{
return (u16)(gTasks[taskId].tExpTask_gainedExp_1) | (gTasks[taskId].tExpTask_gainedExp_2 << 16);
}
2017-10-22 18:43:15 +02:00
static void Task_GiveExpToMon(u8 taskId)
{
2017-10-22 18:43:15 +02:00
u32 monId = (u8)(gTasks[taskId].tExpTask_monId);
u8 battlerId = gTasks[taskId].tExpTask_battler;
2021-06-03 23:17:44 +02:00
s32 gainedExp = GetTaskExpValue(taskId);
if (IsDoubleBattle() == TRUE || monId != gBattlerPartyIndexes[battlerId]) // Give exp without moving the expbar.
{
struct Pokemon *mon = &gPlayerParty[monId];
u16 species = GetMonData(mon, MON_DATA_SPECIES);
u8 level = GetMonData(mon, MON_DATA_LEVEL);
u32 currExp = GetMonData(mon, MON_DATA_EXP);
u32 nextLvlExp = gExperienceTables[gBaseStats[species].growthRate][level + 1];
if (currExp + gainedExp >= nextLvlExp)
{
u8 savedActiveBattler;
SetMonData(mon, MON_DATA_EXP, &nextLvlExp);
CalculateMonStats(mon);
gainedExp -= nextLvlExp - currExp;
savedActiveBattler = gActiveBattler;
2018-03-01 00:59:52 +01:00
gActiveBattler = battlerId;
2018-07-16 20:47:30 +02:00
BtlController_EmitTwoReturnValues(1, RET_VALUE_LEVELED_UP, gainedExp);
gActiveBattler = savedActiveBattler;
if (IsDoubleBattle() == TRUE
2018-03-01 00:59:52 +01:00
&& ((u16)(monId) == gBattlerPartyIndexes[battlerId] || (u16)(monId) == gBattlerPartyIndexes[battlerId ^ BIT_FLANK]))
gTasks[taskId].func = Task_LaunchLvlUpAnim;
else
gTasks[taskId].func = DestroyExpTaskAndCompleteOnInactiveTextPrinter;
}
else
{
currExp += gainedExp;
SetMonData(mon, MON_DATA_EXP, &currExp);
2018-03-01 00:59:52 +01:00
gBattlerControllerFuncs[battlerId] = CompleteOnInactiveTextPrinter;
DestroyTask(taskId);
}
}
else
{
2017-10-22 18:43:15 +02:00
gTasks[taskId].func = Task_PrepareToGiveExpWithExpBar;
}
}
2017-10-22 18:43:15 +02:00
static void Task_PrepareToGiveExpWithExpBar(u8 taskId)
{
2017-10-22 18:43:15 +02:00
u8 monIndex = gTasks[taskId].tExpTask_monId;
2021-06-03 23:17:44 +02:00
s32 gainedExp = GetTaskExpValue(taskId);
u8 battlerId = gTasks[taskId].tExpTask_battler;
struct Pokemon *mon = &gPlayerParty[monIndex];
u8 level = GetMonData(mon, MON_DATA_LEVEL);
u16 species = GetMonData(mon, MON_DATA_SPECIES);
u32 exp = GetMonData(mon, MON_DATA_EXP);
u32 currLvlExp = gExperienceTables[gBaseStats[species].growthRate][level];
u32 expToNextLvl;
exp -= currLvlExp;
expToNextLvl = gExperienceTables[gBaseStats[species].growthRate][level + 1] - currLvlExp;
2018-03-01 00:59:52 +01:00
SetBattleBarStruct(battlerId, gHealthboxSpriteIds[battlerId], expToNextLvl, exp, -gainedExp);
PlaySE(SE_EXP);
2021-01-22 20:03:21 -05:00
gTasks[taskId].func = Task_GiveExpWithExpBar;
}
2021-01-22 20:03:21 -05:00
static void Task_GiveExpWithExpBar(u8 taskId)
{
2017-10-22 18:43:15 +02:00
if (gTasks[taskId].tExpTask_frames < 13)
{
2017-10-22 18:43:15 +02:00
gTasks[taskId].tExpTask_frames++;
}
else
{
2017-10-22 18:43:15 +02:00
u8 monId = gTasks[taskId].tExpTask_monId;
2021-06-03 23:17:44 +02:00
s32 gainedExp = GetTaskExpValue(taskId);
u8 battlerId = gTasks[taskId].tExpTask_battler;
2021-06-03 23:17:44 +02:00
s32 newExpPoints;
newExpPoints = MoveBattleBar(battlerId, gHealthboxSpriteIds[battlerId], EXP_BAR, 0);
2018-03-01 00:59:52 +01:00
SetHealthboxSpriteVisible(gHealthboxSpriteIds[battlerId]);
if (newExpPoints == -1) // The bar has been filled with given exp points.
{
u8 level;
s32 currExp;
u16 species;
s32 expOnNextLvl;
m4aSongNumStop(SE_EXP);
level = GetMonData(&gPlayerParty[monId], MON_DATA_LEVEL);
currExp = GetMonData(&gPlayerParty[monId], MON_DATA_EXP);
species = GetMonData(&gPlayerParty[monId], MON_DATA_SPECIES);
expOnNextLvl = gExperienceTables[gBaseStats[species].growthRate][level + 1];
if (currExp + gainedExp >= expOnNextLvl)
{
u8 savedActiveBattler;
SetMonData(&gPlayerParty[monId], MON_DATA_EXP, &expOnNextLvl);
CalculateMonStats(&gPlayerParty[monId]);
gainedExp -= expOnNextLvl - currExp;
savedActiveBattler = gActiveBattler;
2018-03-01 00:59:52 +01:00
gActiveBattler = battlerId;
2018-07-16 20:47:30 +02:00
BtlController_EmitTwoReturnValues(1, RET_VALUE_LEVELED_UP, gainedExp);
gActiveBattler = savedActiveBattler;
gTasks[taskId].func = Task_LaunchLvlUpAnim;
}
else
{
currExp += gainedExp;
SetMonData(&gPlayerParty[monId], MON_DATA_EXP, &currExp);
2018-03-01 00:59:52 +01:00
gBattlerControllerFuncs[battlerId] = CompleteOnInactiveTextPrinter;
DestroyTask(taskId);
}
}
}
}
static void Task_LaunchLvlUpAnim(u8 taskId)
{
u8 battlerId = gTasks[taskId].tExpTask_battler;
2017-10-22 18:43:15 +02:00
u8 monIndex = gTasks[taskId].tExpTask_monId;
2018-03-01 00:59:52 +01:00
if (IsDoubleBattle() == TRUE && monIndex == gBattlerPartyIndexes[battlerId ^ BIT_FLANK])
battlerId ^= BIT_FLANK;
2018-03-01 00:59:52 +01:00
InitAndLaunchSpecialAnimation(battlerId, battlerId, battlerId, B_ANIM_LVL_UP);
gTasks[taskId].func = Task_UpdateLvlInHealthbox;
}
static void Task_UpdateLvlInHealthbox(u8 taskId)
{
u8 battlerId = gTasks[taskId].tExpTask_battler;
2018-03-01 00:59:52 +01:00
if (!gBattleSpritesDataPtr->healthBoxesData[battlerId].specialAnimActive)
{
2017-10-22 18:43:15 +02:00
u8 monIndex = gTasks[taskId].tExpTask_monId;
GetMonData(&gPlayerParty[monIndex], MON_DATA_LEVEL); // Unused return value.
2018-03-01 00:59:52 +01:00
if (IsDoubleBattle() == TRUE && monIndex == gBattlerPartyIndexes[battlerId ^ BIT_FLANK])
UpdateHealthboxAttribute(gHealthboxSpriteIds[battlerId ^ BIT_FLANK], &gPlayerParty[monIndex], HEALTHBOX_ALL);
else
2018-03-01 00:59:52 +01:00
UpdateHealthboxAttribute(gHealthboxSpriteIds[battlerId], &gPlayerParty[monIndex], HEALTHBOX_ALL);
gTasks[taskId].func = DestroyExpTaskAndCompleteOnInactiveTextPrinter;
}
}
2017-10-22 18:43:15 +02:00
static void DestroyExpTaskAndCompleteOnInactiveTextPrinter(u8 taskId)
{
u8 monIndex;
2018-03-01 00:59:52 +01:00
u8 battlerId;
2017-10-22 18:43:15 +02:00
monIndex = gTasks[taskId].tExpTask_monId;
GetMonData(&gPlayerParty[monIndex], MON_DATA_LEVEL); // Unused return value.
battlerId = gTasks[taskId].tExpTask_battler;
2018-03-01 00:59:52 +01:00
gBattlerControllerFuncs[battlerId] = CompleteOnInactiveTextPrinter;
DestroyTask(taskId);
}
2021-01-22 20:03:21 -05:00
static void FreeMonSpriteAfterFaintAnim(void)
{
2021-07-07 09:11:52 -04:00
if (gSprites[gBattlerSpriteIds[gActiveBattler]].y + gSprites[gBattlerSpriteIds[gActiveBattler]].y2 > DISPLAY_HEIGHT)
{
2018-02-06 13:48:02 -06:00
u16 species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_SPECIES);
2021-01-22 20:03:21 -05:00
BattleGfxSfxDummy2(species);
2018-02-06 13:48:02 -06:00
FreeOamMatrix(gSprites[gBattlerSpriteIds[gActiveBattler]].oam.matrixNum);
DestroySprite(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
SetHealthboxSpriteInvisible(gHealthboxSpriteIds[gActiveBattler]);
PlayerBufferExecCompleted();
}
}
2021-01-22 20:03:21 -05:00
static void FreeMonSpriteAfterSwitchOutAnim(void)
{
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive)
{
2018-02-06 13:48:02 -06:00
FreeSpriteOamMatrix(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
DestroySprite(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
SetHealthboxSpriteInvisible(gHealthboxSpriteIds[gActiveBattler]);
PlayerBufferExecCompleted();
}
}
2017-10-22 18:43:15 +02:00
static void CompleteOnInactiveTextPrinter2(void)
{
if (!IsTextPrinterActive(0))
PlayerBufferExecCompleted();
}
2018-07-07 19:57:09 +02:00
static void OpenPartyMenuToChooseMon(void)
{
if (!gPaletteFade.active)
{
2018-07-07 19:57:09 +02:00
u8 caseId;
2018-07-07 19:57:09 +02:00
gBattlerControllerFuncs[gActiveBattler] = WaitForMonSelection;
2021-01-22 20:03:21 -05:00
caseId = gTasks[gBattleControllerData[gActiveBattler]].data[0];
DestroyTask(gBattleControllerData[gActiveBattler]);
FreeAllWindowBuffers();
2018-07-07 19:57:09 +02:00
OpenPartyMenuInBattle(caseId);
}
}
2018-07-07 19:57:09 +02:00
static void WaitForMonSelection(void)
{
if (gMain.callback2 == BattleMainCB2 && !gPaletteFade.active)
{
2019-10-25 21:55:01 -04:00
if (gPartyMenuUseExitCallback == TRUE)
BtlController_EmitChosenMonReturnValue(1, gSelectedMonPartyId, gBattlePartyCurrentOrder);
else
2019-10-17 19:22:03 -04:00
BtlController_EmitChosenMonReturnValue(1, PARTY_SIZE, NULL);
if ((gBattleResources->bufferA[gActiveBattler][1] & 0xF) == 1)
PrintLinkStandbyMsg();
PlayerBufferExecCompleted();
}
}
2017-10-22 20:38:23 +02:00
static void OpenBagAndChooseItem(void)
{
if (!gPaletteFade.active)
{
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteWhenChoseItem;
2021-01-22 20:03:21 -05:00
ReshowBattleScreenDummy();
FreeAllWindowBuffers();
2020-05-14 01:37:09 -07:00
CB2_BagMenuFromBattle();
}
}
2017-10-22 20:38:23 +02:00
static void CompleteWhenChoseItem(void)
{
if (gMain.callback2 == BattleMainCB2 && !gPaletteFade.active)
{
2018-02-06 13:48:02 -06:00
BtlController_EmitOneReturnValue(1, gSpecialVar_ItemId);
PlayerBufferExecCompleted();
}
}
2017-10-22 18:43:15 +02:00
static void CompleteOnSpecialAnimDone(void)
{
2018-02-05 19:46:59 -06:00
if (!gDoingBattleAnim || !gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive)
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void DoHitAnimBlinkSpriteEffect(void)
{
2018-02-06 13:48:02 -06:00
u8 spriteId = gBattlerSpriteIds[gActiveBattler];
2017-12-02 21:44:50 +01:00
if (gSprites[spriteId].data[1] == 32)
{
2017-12-02 21:44:50 +01:00
gSprites[spriteId].data[1] = 0;
gSprites[spriteId].invisible = FALSE;
gDoingBattleAnim = FALSE;
PlayerBufferExecCompleted();
}
else
{
2017-12-02 21:44:50 +01:00
if ((gSprites[spriteId].data[1] % 4) == 0)
gSprites[spriteId].invisible ^= 1;
2017-12-02 21:44:50 +01:00
gSprites[spriteId].data[1]++;
}
}
static void PlayerHandleYesNoInput(void)
{
2020-09-04 21:11:55 -04:00
if (JOY_NEW(DPAD_UP) && gMultiUsePlayerCursor != 0)
{
PlaySE(SE_SELECT);
BattleDestroyYesNoCursorAt(gMultiUsePlayerCursor);
gMultiUsePlayerCursor = 0;
BattleCreateYesNoCursorAt(0);
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(DPAD_DOWN) && gMultiUsePlayerCursor == 0)
{
PlaySE(SE_SELECT);
BattleDestroyYesNoCursorAt(gMultiUsePlayerCursor);
gMultiUsePlayerCursor = 1;
BattleCreateYesNoCursorAt(1);
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
{
HandleBattleWindow(0x18, 8, 0x1D, 0xD, WINDOW_CLEAR);
PlaySE(SE_SELECT);
if (gMultiUsePlayerCursor != 0)
2018-02-06 13:48:02 -06:00
BtlController_EmitTwoReturnValues(1, 0xE, 0);
else
2018-02-06 13:48:02 -06:00
BtlController_EmitTwoReturnValues(1, 0xD, 0);
PlayerBufferExecCompleted();
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(B_BUTTON))
{
HandleBattleWindow(0x18, 8, 0x1D, 0xD, WINDOW_CLEAR);
PlaySE(SE_SELECT);
PlayerBufferExecCompleted();
}
}
2017-10-22 18:43:15 +02:00
static void MoveSelectionDisplayMoveNames(void)
{
s32 i;
struct ChooseMoveStruct *moveInfo = (struct ChooseMoveStruct*)(&gBattleResources->bufferA[gActiveBattler][4]);
gNumberOfMovesToChoose = 0;
for (i = 0; i < MAX_MON_MOVES; i++)
{
MoveSelectionDestroyCursorAt(i);
StringCopy(gDisplayedStringBattle, gMoveNames[moveInfo->moves[i]]);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gDisplayedStringBattle, i + 3);
if (moveInfo->moves[i] != MOVE_NONE)
gNumberOfMovesToChoose++;
}
}
2017-10-22 18:43:15 +02:00
static void MoveSelectionDisplayPpString(void)
{
StringCopy(gDisplayedStringBattle, gText_MoveInterfacePP);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gDisplayedStringBattle, 7);
}
2017-10-22 18:43:15 +02:00
static void MoveSelectionDisplayPpNumber(void)
{
u8 *txtPtr;
struct ChooseMoveStruct *moveInfo;
if (gBattleResources->bufferA[gActiveBattler][2] == TRUE) // check if we didn't want to display pp number
return;
SetPpNumbersPaletteInMoveSelection();
moveInfo = (struct ChooseMoveStruct*)(&gBattleResources->bufferA[gActiveBattler][4]);
2018-02-05 19:46:59 -06:00
txtPtr = ConvertIntToDecimalStringN(gDisplayedStringBattle, moveInfo->currentPp[gMoveSelectionCursor[gActiveBattler]], STR_CONV_MODE_RIGHT_ALIGN, 2);
2020-08-10 23:50:49 -04:00
*(txtPtr)++ = CHAR_SLASH;
2018-02-05 19:46:59 -06:00
ConvertIntToDecimalStringN(txtPtr, moveInfo->maxPp[gMoveSelectionCursor[gActiveBattler]], STR_CONV_MODE_RIGHT_ALIGN, 2);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gDisplayedStringBattle, 9);
}
2017-10-22 18:43:15 +02:00
static void MoveSelectionDisplayMoveType(void)
{
u8 *txtPtr;
struct ChooseMoveStruct *moveInfo = (struct ChooseMoveStruct*)(&gBattleResources->bufferA[gActiveBattler][4]);
txtPtr = StringCopy(gDisplayedStringBattle, gText_MoveInterfaceType);
2020-08-10 23:50:49 -04:00
*(txtPtr)++ = EXT_CTRL_CODE_BEGIN;
*(txtPtr)++ = EXT_CTRL_CODE_SIZE;
*(txtPtr)++ = 1;
2018-02-05 19:46:59 -06:00
StringCopy(txtPtr, gTypeNames[gBattleMoves[moveInfo->moves[gMoveSelectionCursor[gActiveBattler]]].type]);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gDisplayedStringBattle, 10);
}
2017-10-22 18:43:15 +02:00
static void MoveSelectionCreateCursorAt(u8 cursorPosition, u8 arg1)
{
u16 src[2];
src[0] = arg1 + 1;
src[1] = arg1 + 2;
CopyToBgTilemapBufferRect_ChangePalette(0, src, 9 * (cursorPosition & 1) + 1, 55 + (cursorPosition & 2), 1, 2, 0x11);
CopyBgTilemapBufferToVram(0);
}
2017-10-22 18:43:15 +02:00
static void MoveSelectionDestroyCursorAt(u8 cursorPosition)
{
u16 src[2];
src[0] = 0x1016;
src[1] = 0x1016;
CopyToBgTilemapBufferRect_ChangePalette(0, src, 9 * (cursorPosition & 1) + 1, 55 + (cursorPosition & 2), 1, 2, 0x11);
CopyBgTilemapBufferToVram(0);
}
void ActionSelectionCreateCursorAt(u8 cursorPosition, u8 arg1)
{
u16 src[2];
src[0] = 1;
src[1] = 2;
CopyToBgTilemapBufferRect_ChangePalette(0, src, 7 * (cursorPosition & 1) + 16, 35 + (cursorPosition & 2), 1, 2, 0x11);
CopyBgTilemapBufferToVram(0);
}
void ActionSelectionDestroyCursorAt(u8 cursorPosition)
{
u16 src[2];
src[0] = 0x1016;
src[1] = 0x1016;
CopyToBgTilemapBufferRect_ChangePalette(0, src, 7 * (cursorPosition & 1) + 16, 35 + (cursorPosition & 2), 1, 2, 0x11);
CopyBgTilemapBufferToVram(0);
}
2019-10-17 19:22:03 -04:00
void CB2_SetUpReshowBattleScreenAfterMenu(void)
{
SetMainCallback2(ReshowBattleScreenAfterMenu);
}
2019-10-17 19:22:03 -04:00
void CB2_SetUpReshowBattleScreenAfterMenu2(void)
{
SetMainCallback2(ReshowBattleScreenAfterMenu);
}
2017-10-22 18:43:15 +02:00
static void CompleteOnFinishedStatusAnimation(void)
{
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].statusAnimActive)
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void CompleteOnFinishedBattleAnimation(void)
{
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animFromTableActive)
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void PrintLinkStandbyMsg(void)
{
if (gBattleTypeFlags & BATTLE_TYPE_LINK)
{
gBattle_BG0_X = 0;
gBattle_BG0_Y = 0;
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gText_LinkStandby, 0);
}
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleGetMonData(void)
{
u8 monData[sizeof(struct Pokemon) * 2 + 56]; // this allows to get full data of two pokemon, trying to get more will result in overwriting data
u32 size = 0;
2017-10-31 18:04:08 +01:00
u8 monToCheck;
s32 i;
if (gBattleResources->bufferA[gActiveBattler][2] == 0)
{
2018-02-06 13:48:02 -06:00
size += CopyPlayerMonData(gBattlerPartyIndexes[gActiveBattler], monData);
}
else
{
monToCheck = gBattleResources->bufferA[gActiveBattler][2];
2018-11-18 20:00:36 +01:00
for (i = 0; i < PARTY_SIZE; i++)
{
2017-10-31 18:04:08 +01:00
if (monToCheck & 1)
size += CopyPlayerMonData(i, monData + size);
2017-10-31 18:04:08 +01:00
monToCheck >>= 1;
}
}
2018-02-06 13:48:02 -06:00
BtlController_EmitDataTransfer(1, size, monData);
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static u32 CopyPlayerMonData(u8 monId, u8 *dst)
{
struct BattlePokemon battleMon;
struct MovePpInfo moveData;
u8 nickname[20];
u8 *src;
s16 data16;
u32 data32;
s32 size = 0;
switch (gBattleResources->bufferA[gActiveBattler][1])
{
case REQUEST_ALL_BATTLE:
battleMon.species = GetMonData(&gPlayerParty[monId], MON_DATA_SPECIES);
battleMon.item = GetMonData(&gPlayerParty[monId], MON_DATA_HELD_ITEM);
for (size = 0; size < MAX_MON_MOVES; size++)
{
battleMon.moves[size] = GetMonData(&gPlayerParty[monId], MON_DATA_MOVE1 + size);
battleMon.pp[size] = GetMonData(&gPlayerParty[monId], MON_DATA_PP1 + size);
}
battleMon.ppBonuses = GetMonData(&gPlayerParty[monId], MON_DATA_PP_BONUSES);
battleMon.friendship = GetMonData(&gPlayerParty[monId], MON_DATA_FRIENDSHIP);
battleMon.experience = GetMonData(&gPlayerParty[monId], MON_DATA_EXP);
battleMon.hpIV = GetMonData(&gPlayerParty[monId], MON_DATA_HP_IV);
battleMon.attackIV = GetMonData(&gPlayerParty[monId], MON_DATA_ATK_IV);
battleMon.defenseIV = GetMonData(&gPlayerParty[monId], MON_DATA_DEF_IV);
2017-10-24 15:35:36 +02:00
battleMon.speedIV = GetMonData(&gPlayerParty[monId], MON_DATA_SPEED_IV);
battleMon.spAttackIV = GetMonData(&gPlayerParty[monId], MON_DATA_SPATK_IV);
battleMon.spDefenseIV = GetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_IV);
battleMon.personality = GetMonData(&gPlayerParty[monId], MON_DATA_PERSONALITY);
battleMon.status1 = GetMonData(&gPlayerParty[monId], MON_DATA_STATUS);
battleMon.level = GetMonData(&gPlayerParty[monId], MON_DATA_LEVEL);
battleMon.hp = GetMonData(&gPlayerParty[monId], MON_DATA_HP);
battleMon.maxHP = GetMonData(&gPlayerParty[monId], MON_DATA_MAX_HP);
battleMon.attack = GetMonData(&gPlayerParty[monId], MON_DATA_ATK);
battleMon.defense = GetMonData(&gPlayerParty[monId], MON_DATA_DEF);
2017-10-24 15:35:36 +02:00
battleMon.speed = GetMonData(&gPlayerParty[monId], MON_DATA_SPEED);
battleMon.spAttack = GetMonData(&gPlayerParty[monId], MON_DATA_SPATK);
battleMon.spDefense = GetMonData(&gPlayerParty[monId], MON_DATA_SPDEF);
2019-05-14 15:22:16 +02:00
battleMon.abilityNum = GetMonData(&gPlayerParty[monId], MON_DATA_ABILITY_NUM);
battleMon.otId = GetMonData(&gPlayerParty[monId], MON_DATA_OT_ID);
GetMonData(&gPlayerParty[monId], MON_DATA_NICKNAME, nickname);
StringCopy10(battleMon.nickname, nickname);
GetMonData(&gPlayerParty[monId], MON_DATA_OT_NAME, battleMon.otName);
src = (u8 *)&battleMon;
for (size = 0; size < sizeof(battleMon); size++)
dst[size] = src[size];
break;
case REQUEST_SPECIES_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_SPECIES);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_HELDITEM_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_HELD_ITEM);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_MOVES_PP_BATTLE:
for (size = 0; size < MAX_MON_MOVES; size++)
{
moveData.moves[size] = GetMonData(&gPlayerParty[monId], MON_DATA_MOVE1 + size);
moveData.pp[size] = GetMonData(&gPlayerParty[monId], MON_DATA_PP1 + size);
}
moveData.ppBonuses = GetMonData(&gPlayerParty[monId], MON_DATA_PP_BONUSES);
src = (u8*)(&moveData);
for (size = 0; size < sizeof(moveData); size++)
dst[size] = src[size];
break;
case REQUEST_MOVE1_BATTLE:
case REQUEST_MOVE2_BATTLE:
case REQUEST_MOVE3_BATTLE:
case REQUEST_MOVE4_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_MOVE1 + gBattleResources->bufferA[gActiveBattler][1] - REQUEST_MOVE1_BATTLE);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_PP_DATA_BATTLE:
for (size = 0; size < MAX_MON_MOVES; size++)
dst[size] = GetMonData(&gPlayerParty[monId], MON_DATA_PP1 + size);
dst[size] = GetMonData(&gPlayerParty[monId], MON_DATA_PP_BONUSES);
size++;
break;
case REQUEST_PPMOVE1_BATTLE:
case REQUEST_PPMOVE2_BATTLE:
case REQUEST_PPMOVE3_BATTLE:
case REQUEST_PPMOVE4_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_PP1 + gBattleResources->bufferA[gActiveBattler][1] - REQUEST_PPMOVE1_BATTLE);
size = 1;
break;
case REQUEST_OTID_BATTLE:
data32 = GetMonData(&gPlayerParty[monId], MON_DATA_OT_ID);
dst[0] = (data32 & 0x000000FF);
dst[1] = (data32 & 0x0000FF00) >> 8;
dst[2] = (data32 & 0x00FF0000) >> 16;
size = 3;
break;
case REQUEST_EXP_BATTLE:
data32 = GetMonData(&gPlayerParty[monId], MON_DATA_EXP);
dst[0] = (data32 & 0x000000FF);
dst[1] = (data32 & 0x0000FF00) >> 8;
dst[2] = (data32 & 0x00FF0000) >> 16;
size = 3;
break;
case REQUEST_HP_EV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_HP_EV);
size = 1;
break;
case REQUEST_ATK_EV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_ATK_EV);
size = 1;
break;
case REQUEST_DEF_EV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_DEF_EV);
size = 1;
break;
case REQUEST_SPEED_EV_BATTLE:
2017-10-24 15:35:36 +02:00
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SPEED_EV);
size = 1;
break;
case REQUEST_SPATK_EV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SPATK_EV);
size = 1;
break;
case REQUEST_SPDEF_EV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_EV);
size = 1;
break;
case REQUEST_FRIENDSHIP_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_FRIENDSHIP);
size = 1;
break;
case REQUEST_POKERUS_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_POKERUS);
size = 1;
break;
case REQUEST_MET_LOCATION_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_MET_LOCATION);
size = 1;
break;
case REQUEST_MET_LEVEL_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_MET_LEVEL);
size = 1;
break;
case REQUEST_MET_GAME_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_MET_GAME);
size = 1;
break;
case REQUEST_POKEBALL_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_POKEBALL);
size = 1;
break;
case REQUEST_ALL_IVS_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_HP_IV);
dst[1] = GetMonData(&gPlayerParty[monId], MON_DATA_ATK_IV);
dst[2] = GetMonData(&gPlayerParty[monId], MON_DATA_DEF_IV);
2017-10-24 15:35:36 +02:00
dst[3] = GetMonData(&gPlayerParty[monId], MON_DATA_SPEED_IV);
dst[4] = GetMonData(&gPlayerParty[monId], MON_DATA_SPATK_IV);
dst[5] = GetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_IV);
size = 6;
break;
case REQUEST_HP_IV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_HP_IV);
size = 1;
break;
case REQUEST_ATK_IV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_ATK_IV);
size = 1;
break;
case REQUEST_DEF_IV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_DEF_IV);
size = 1;
break;
case REQUEST_SPEED_IV_BATTLE:
2017-10-24 15:35:36 +02:00
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SPEED_IV);
size = 1;
break;
case REQUEST_SPATK_IV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SPATK_IV);
size = 1;
break;
case REQUEST_SPDEF_IV_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_IV);
size = 1;
break;
case REQUEST_PERSONALITY_BATTLE:
data32 = GetMonData(&gPlayerParty[monId], MON_DATA_PERSONALITY);
dst[0] = (data32 & 0x000000FF);
dst[1] = (data32 & 0x0000FF00) >> 8;
dst[2] = (data32 & 0x00FF0000) >> 16;
dst[3] = (data32 & 0xFF000000) >> 24;
size = 4;
break;
case REQUEST_CHECKSUM_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_CHECKSUM);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_STATUS_BATTLE:
data32 = GetMonData(&gPlayerParty[monId], MON_DATA_STATUS);
dst[0] = (data32 & 0x000000FF);
dst[1] = (data32 & 0x0000FF00) >> 8;
dst[2] = (data32 & 0x00FF0000) >> 16;
dst[3] = (data32 & 0xFF000000) >> 24;
size = 4;
break;
case REQUEST_LEVEL_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_LEVEL);
size = 1;
break;
case REQUEST_HP_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_HP);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_MAX_HP_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_MAX_HP);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_ATK_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_ATK);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_DEF_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_DEF);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_SPEED_BATTLE:
2017-10-24 15:35:36 +02:00
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_SPEED);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_SPATK_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_SPATK);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_SPDEF_BATTLE:
data16 = GetMonData(&gPlayerParty[monId], MON_DATA_SPDEF);
dst[0] = data16;
dst[1] = data16 >> 8;
size = 2;
break;
case REQUEST_COOL_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_COOL);
size = 1;
break;
case REQUEST_BEAUTY_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_BEAUTY);
size = 1;
break;
case REQUEST_CUTE_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_CUTE);
size = 1;
break;
case REQUEST_SMART_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SMART);
size = 1;
break;
case REQUEST_TOUGH_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_TOUGH);
size = 1;
break;
case REQUEST_SHEEN_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SHEEN);
size = 1;
break;
case REQUEST_COOL_RIBBON_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_COOL_RIBBON);
size = 1;
break;
case REQUEST_BEAUTY_RIBBON_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_BEAUTY_RIBBON);
size = 1;
break;
case REQUEST_CUTE_RIBBON_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_CUTE_RIBBON);
size = 1;
break;
case REQUEST_SMART_RIBBON_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_SMART_RIBBON);
size = 1;
break;
case REQUEST_TOUGH_RIBBON_BATTLE:
dst[0] = GetMonData(&gPlayerParty[monId], MON_DATA_TOUGH_RIBBON);
size = 1;
break;
}
return size;
}
void PlayerHandleGetRawMonData(void)
{
struct BattlePokemon battleMon;
u8 *src = (u8 *)&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]] + gBattleResources->bufferA[gActiveBattler][1];
u8 *dst = (u8 *)&battleMon + gBattleResources->bufferA[gActiveBattler][1];
u8 i;
for (i = 0; i < gBattleResources->bufferA[gActiveBattler][2]; i++)
dst[i] = src[i];
BtlController_EmitDataTransfer(1, gBattleResources->bufferA[gActiveBattler][2], dst);
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleSetMonData(void)
{
2017-10-31 18:04:08 +01:00
u8 monToCheck;
u8 i;
if (gBattleResources->bufferA[gActiveBattler][2] == 0)
{
2018-02-06 13:48:02 -06:00
SetPlayerMonData(gBattlerPartyIndexes[gActiveBattler]);
}
else
{
monToCheck = gBattleResources->bufferA[gActiveBattler][2];
2018-11-18 20:00:36 +01:00
for (i = 0; i < PARTY_SIZE; i++)
{
2017-10-31 18:04:08 +01:00
if (monToCheck & 1)
SetPlayerMonData(i);
2017-10-31 18:04:08 +01:00
monToCheck >>= 1;
}
}
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void SetPlayerMonData(u8 monId)
{
struct BattlePokemon *battlePokemon = (struct BattlePokemon *)&gBattleResources->bufferA[gActiveBattler][3];
struct MovePpInfo *moveData = (struct MovePpInfo *)&gBattleResources->bufferA[gActiveBattler][3];
s32 i;
switch (gBattleResources->bufferA[gActiveBattler][1])
{
case REQUEST_ALL_BATTLE:
{
u8 iv;
SetMonData(&gPlayerParty[monId], MON_DATA_SPECIES, &battlePokemon->species);
SetMonData(&gPlayerParty[monId], MON_DATA_HELD_ITEM, &battlePokemon->item);
for (i = 0; i < MAX_MON_MOVES; i++)
{
SetMonData(&gPlayerParty[monId], MON_DATA_MOVE1 + i, &battlePokemon->moves[i]);
SetMonData(&gPlayerParty[monId], MON_DATA_PP1 + i, &battlePokemon->pp[i]);
}
SetMonData(&gPlayerParty[monId], MON_DATA_PP_BONUSES, &battlePokemon->ppBonuses);
SetMonData(&gPlayerParty[monId], MON_DATA_FRIENDSHIP, &battlePokemon->friendship);
SetMonData(&gPlayerParty[monId], MON_DATA_EXP, &battlePokemon->experience);
iv = battlePokemon->hpIV;
SetMonData(&gPlayerParty[monId], MON_DATA_HP_IV, &iv);
iv = battlePokemon->attackIV;
SetMonData(&gPlayerParty[monId], MON_DATA_ATK_IV, &iv);
iv = battlePokemon->defenseIV;
SetMonData(&gPlayerParty[monId], MON_DATA_DEF_IV, &iv);
iv = battlePokemon->speedIV;
2017-10-24 15:35:36 +02:00
SetMonData(&gPlayerParty[monId], MON_DATA_SPEED_IV, &iv);
iv = battlePokemon->spAttackIV;
SetMonData(&gPlayerParty[monId], MON_DATA_SPATK_IV, &iv);
iv = battlePokemon->spDefenseIV;
SetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_IV, &iv);
SetMonData(&gPlayerParty[monId], MON_DATA_PERSONALITY, &battlePokemon->personality);
SetMonData(&gPlayerParty[monId], MON_DATA_STATUS, &battlePokemon->status1);
SetMonData(&gPlayerParty[monId], MON_DATA_LEVEL, &battlePokemon->level);
SetMonData(&gPlayerParty[monId], MON_DATA_HP, &battlePokemon->hp);
SetMonData(&gPlayerParty[monId], MON_DATA_MAX_HP, &battlePokemon->maxHP);
SetMonData(&gPlayerParty[monId], MON_DATA_ATK, &battlePokemon->attack);
SetMonData(&gPlayerParty[monId], MON_DATA_DEF, &battlePokemon->defense);
2017-10-24 15:35:36 +02:00
SetMonData(&gPlayerParty[monId], MON_DATA_SPEED, &battlePokemon->speed);
SetMonData(&gPlayerParty[monId], MON_DATA_SPATK, &battlePokemon->spAttack);
SetMonData(&gPlayerParty[monId], MON_DATA_SPDEF, &battlePokemon->spDefense);
}
break;
case REQUEST_SPECIES_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPECIES, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_HELDITEM_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_HELD_ITEM, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_MOVES_PP_BATTLE:
for (i = 0; i < MAX_MON_MOVES; i++)
{
SetMonData(&gPlayerParty[monId], MON_DATA_MOVE1 + i, &moveData->moves[i]);
SetMonData(&gPlayerParty[monId], MON_DATA_PP1 + i, &moveData->pp[i]);
}
SetMonData(&gPlayerParty[monId], MON_DATA_PP_BONUSES, &moveData->ppBonuses);
break;
case REQUEST_MOVE1_BATTLE:
case REQUEST_MOVE2_BATTLE:
case REQUEST_MOVE3_BATTLE:
case REQUEST_MOVE4_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_MOVE1 + gBattleResources->bufferA[gActiveBattler][1] - REQUEST_MOVE1_BATTLE, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_PP_DATA_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_PP1, &gBattleResources->bufferA[gActiveBattler][3]);
SetMonData(&gPlayerParty[monId], MON_DATA_PP2, &gBattleResources->bufferA[gActiveBattler][4]);
SetMonData(&gPlayerParty[monId], MON_DATA_PP3, &gBattleResources->bufferA[gActiveBattler][5]);
SetMonData(&gPlayerParty[monId], MON_DATA_PP4, &gBattleResources->bufferA[gActiveBattler][6]);
SetMonData(&gPlayerParty[monId], MON_DATA_PP_BONUSES, &gBattleResources->bufferA[gActiveBattler][7]);
break;
case REQUEST_PPMOVE1_BATTLE:
case REQUEST_PPMOVE2_BATTLE:
case REQUEST_PPMOVE3_BATTLE:
case REQUEST_PPMOVE4_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_PP1 + gBattleResources->bufferA[gActiveBattler][1] - REQUEST_PPMOVE1_BATTLE, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_OTID_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_OT_ID, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_EXP_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_EXP, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_HP_EV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_HP_EV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_ATK_EV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_ATK_EV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_DEF_EV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_DEF_EV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPEED_EV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPEED_EV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPATK_EV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPATK_EV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPDEF_EV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_EV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_FRIENDSHIP_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_FRIENDSHIP, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_POKERUS_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_POKERUS, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_MET_LOCATION_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_MET_LOCATION, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_MET_LEVEL_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_MET_LEVEL, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_MET_GAME_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_MET_GAME, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_POKEBALL_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_POKEBALL, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_ALL_IVS_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_HP_IV, &gBattleResources->bufferA[gActiveBattler][3]);
SetMonData(&gPlayerParty[monId], MON_DATA_ATK_IV, &gBattleResources->bufferA[gActiveBattler][4]);
SetMonData(&gPlayerParty[monId], MON_DATA_DEF_IV, &gBattleResources->bufferA[gActiveBattler][5]);
SetMonData(&gPlayerParty[monId], MON_DATA_SPEED_IV, &gBattleResources->bufferA[gActiveBattler][6]);
SetMonData(&gPlayerParty[monId], MON_DATA_SPATK_IV, &gBattleResources->bufferA[gActiveBattler][7]);
SetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_IV, &gBattleResources->bufferA[gActiveBattler][8]);
break;
case REQUEST_HP_IV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_HP_IV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_ATK_IV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_ATK_IV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_DEF_IV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_DEF_IV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPEED_IV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPEED_IV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPATK_IV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPATK_IV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPDEF_IV_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPDEF_IV, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_PERSONALITY_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_PERSONALITY, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_CHECKSUM_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_CHECKSUM, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_STATUS_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_STATUS, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_LEVEL_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_LEVEL, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_HP_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_HP, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_MAX_HP_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_MAX_HP, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_ATK_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_ATK, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_DEF_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_DEF, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPEED_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPEED, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPATK_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPATK, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SPDEF_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SPDEF, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_COOL_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_COOL, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_BEAUTY_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_BEAUTY, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_CUTE_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_CUTE, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SMART_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SMART, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_TOUGH_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_TOUGH, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SHEEN_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SHEEN, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_COOL_RIBBON_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_COOL_RIBBON, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_BEAUTY_RIBBON_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_BEAUTY_RIBBON, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_CUTE_RIBBON_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_CUTE_RIBBON, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_SMART_RIBBON_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_SMART_RIBBON, &gBattleResources->bufferA[gActiveBattler][3]);
break;
case REQUEST_TOUGH_RIBBON_BATTLE:
SetMonData(&gPlayerParty[monId], MON_DATA_TOUGH_RIBBON, &gBattleResources->bufferA[gActiveBattler][3]);
break;
}
2018-02-06 13:48:02 -06:00
HandleLowHpMusicChange(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleSetRawMonData(void)
{
u8 *dst = (u8 *)&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]] + gBattleResources->bufferA[gActiveBattler][1];
u8 i;
for (i = 0; i < gBattleResources->bufferA[gActiveBattler][2]; i++)
dst[i] = gBattleResources->bufferA[gActiveBattler][3 + i];
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleLoadMonSprite(void)
{
2018-02-06 13:48:02 -06:00
BattleLoadPlayerMonSpriteGfx(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
gSprites[gBattlerSpriteIds[gActiveBattler]].oam.paletteNum = gActiveBattler;
gBattlerControllerFuncs[gActiveBattler] = CompleteOnBankSpritePosX_0;
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleSwitchInAnim(void)
{
ClearTemporarySpeciesSpriteData(gActiveBattler, gBattleResources->bufferA[gActiveBattler][2]);
gBattlerPartyIndexes[gActiveBattler] = gBattleResources->bufferA[gActiveBattler][1];
2018-02-06 13:48:02 -06:00
BattleLoadPlayerMonSpriteGfx(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
2018-02-05 19:46:59 -06:00
gActionSelectionCursor[gActiveBattler] = 0;
gMoveSelectionCursor[gActiveBattler] = 0;
StartSendOutAnim(gActiveBattler, gBattleResources->bufferA[gActiveBattler][2]);
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = SwitchIn_TryShinyAnimShowHealthbox;
}
2021-01-22 20:03:21 -05:00
static void StartSendOutAnim(u8 battlerId, bool8 dontClearSubstituteBit)
{
u16 species;
2018-03-01 00:59:52 +01:00
ClearTemporarySpeciesSpriteData(battlerId, dontClearSubstituteBit);
gBattlerPartyIndexes[battlerId] = gBattleResources->bufferA[battlerId][1];
2018-03-01 00:59:52 +01:00
species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
2021-01-22 20:03:21 -05:00
gBattleControllerData[battlerId] = CreateInvisibleSpriteWithCallback(SpriteCB_WaitForBattlerBallReleaseAnim);
SetMultiuseSpriteTemplateToPokemon(species, GetBattlerPosition(battlerId));
2018-03-01 00:59:52 +01:00
gBattlerSpriteIds[battlerId] = CreateSprite(
&gMultiuseSpriteTemplate,
2018-03-01 00:59:52 +01:00
GetBattlerSpriteCoord(battlerId, 2),
GetBattlerSpriteDefault_Y(battlerId),
2018-12-17 22:08:08 -06:00
GetBattlerSpriteSubpriority(battlerId));
2021-01-22 20:03:21 -05:00
gSprites[gBattleControllerData[battlerId]].data[1] = gBattlerSpriteIds[battlerId];
gSprites[gBattleControllerData[battlerId]].data[2] = battlerId;
2018-03-01 00:59:52 +01:00
gSprites[gBattlerSpriteIds[battlerId]].data[0] = battlerId;
gSprites[gBattlerSpriteIds[battlerId]].data[2] = species;
gSprites[gBattlerSpriteIds[battlerId]].oam.paletteNum = battlerId;
2018-03-01 00:59:52 +01:00
StartSpriteAnim(&gSprites[gBattlerSpriteIds[battlerId]], gBattleMonForms[battlerId]);
2018-03-01 00:59:52 +01:00
gSprites[gBattlerSpriteIds[battlerId]].invisible = TRUE;
gSprites[gBattlerSpriteIds[battlerId]].callback = SpriteCallbackDummy;
2021-01-22 20:03:21 -05:00
gSprites[gBattleControllerData[battlerId]].data[0] = DoPokeballSendOutAnimation(0, POKEBALL_PLAYER_SENDOUT);
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleReturnMonToBall(void)
{
if (gBattleResources->bufferA[gActiveBattler][1] == 0)
{
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 0;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = DoSwitchOutAnimation;
}
else
{
2018-02-06 13:48:02 -06:00
FreeSpriteOamMatrix(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
DestroySprite(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
SetHealthboxSpriteInvisible(gHealthboxSpriteIds[gActiveBattler]);
PlayerBufferExecCompleted();
}
}
2017-10-22 18:43:15 +02:00
static void DoSwitchOutAnimation(void)
{
2018-02-05 19:46:59 -06:00
switch (gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState)
{
case 0:
2018-02-06 16:09:39 -06:00
if (gBattleSpritesDataPtr->battlerData[gActiveBattler].behindSubstitute)
2018-02-05 19:46:59 -06:00
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gActiveBattler, B_ANIM_SUBSTITUTE_TO_MON);
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 1;
break;
case 1:
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive)
{
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 0;
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gActiveBattler, B_ANIM_SWITCH_OUT_PLAYER_MON);
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = FreeMonSpriteAfterSwitchOutAnim;
}
break;
}
}
2021-01-22 20:03:21 -05:00
#define sSpeedX data[0]
// In emerald it's possible to have a tag battle in the battle frontier facilities with AI
// which use the front sprite for both the player and the partner as opposed to any other battles (including the one with Steven)
// that use an animated back pic.
2017-10-22 18:43:15 +02:00
static void PlayerHandleDrawTrainerPic(void)
{
s16 xPos, yPos;
u32 trainerPicId;
if (gBattleTypeFlags & BATTLE_TYPE_LINK)
{
if ((gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_FIRE_RED
|| (gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_LEAF_GREEN)
{
trainerPicId = gLinkPlayers[GetMultiplayerId()].gender + TRAINER_BACK_PIC_RED;
}
else if ((gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_RUBY
|| (gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_SAPPHIRE)
{
trainerPicId = gLinkPlayers[GetMultiplayerId()].gender + TRAINER_BACK_PIC_RUBY_SAPPHIRE_BRENDAN;
}
else
{
trainerPicId = gLinkPlayers[GetMultiplayerId()].gender + TRAINER_BACK_PIC_BRENDAN;
}
}
else
{
trainerPicId = gSaveBlock2Ptr->playerGender;
}
if (gBattleTypeFlags & BATTLE_TYPE_MULTI)
{
if ((GetBattlerPosition(gActiveBattler) & BIT_FLANK) != B_FLANK_LEFT) // Second mon, on the right.
xPos = 90;
else // First mon, on the left.
xPos = 32;
2018-11-24 01:02:02 +01:00
if (gBattleTypeFlags & BATTLE_TYPE_INGAME_PARTNER && gPartnerTrainerId != TRAINER_STEVEN_PARTNER && gPartnerTrainerId < TRAINER_CUSTOM_PARTNER)
{
xPos = 90;
yPos = (8 - gTrainerFrontPicCoords[trainerPicId].size) * 4 + 80;
}
else
{
yPos = (8 - gTrainerBackPicCoords[trainerPicId].size) * 4 + 80;
}
}
else
{
xPos = 80;
yPos = (8 - gTrainerBackPicCoords[trainerPicId].size) * 4 + 80;
}
// Use front pic table for any tag battles unless your partner is Steven.
2018-11-24 01:02:02 +01:00
if (gBattleTypeFlags & BATTLE_TYPE_INGAME_PARTNER && gPartnerTrainerId != TRAINER_STEVEN_PARTNER && gPartnerTrainerId < TRAINER_CUSTOM_PARTNER)
{
trainerPicId = PlayerGenderToFrontTrainerPicId(gSaveBlock2Ptr->playerGender);
2018-02-05 19:46:59 -06:00
DecompressTrainerFrontPic(trainerPicId, gActiveBattler);
SetMultiuseSpriteTemplateToTrainerFront(trainerPicId, GetBattlerPosition(gActiveBattler));
2018-12-17 22:08:08 -06:00
gBattlerSpriteIds[gActiveBattler] = CreateSprite(&gMultiuseSpriteTemplate, xPos, yPos, GetBattlerSpriteSubpriority(gActiveBattler));
2018-02-05 19:46:59 -06:00
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].oam.paletteNum = IndexOfSpritePaletteTag(gTrainerFrontPicPaletteTable[trainerPicId].tag);
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[gActiveBattler]].x2 = DISPLAY_WIDTH;
gSprites[gBattlerSpriteIds[gActiveBattler]].y2 = 48;
2021-01-22 20:03:21 -05:00
gSprites[gBattlerSpriteIds[gActiveBattler]].sSpeedX = -2;
gSprites[gBattlerSpriteIds[gActiveBattler]].callback = SpriteCB_TrainerSlideIn;
gSprites[gBattlerSpriteIds[gActiveBattler]].oam.affineMode = ST_OAM_AFFINE_OFF;
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].hFlip = 1;
}
// Use the back pic in any other scenario.
else
{
2018-02-05 19:46:59 -06:00
DecompressTrainerBackPic(trainerPicId, gActiveBattler);
SetMultiuseSpriteTemplateToTrainerBack(trainerPicId, GetBattlerPosition(gActiveBattler));
2018-12-17 22:08:08 -06:00
gBattlerSpriteIds[gActiveBattler] = CreateSprite(&gMultiuseSpriteTemplate, xPos, yPos, GetBattlerSpriteSubpriority(gActiveBattler));
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].oam.paletteNum = gActiveBattler;
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[gActiveBattler]].x2 = DISPLAY_WIDTH;
2021-01-22 20:03:21 -05:00
gSprites[gBattlerSpriteIds[gActiveBattler]].sSpeedX = -2;
gSprites[gBattlerSpriteIds[gActiveBattler]].callback = SpriteCB_TrainerSlideIn;
}
gBattlerControllerFuncs[gActiveBattler] = CompleteOnBattlerSpriteCallbackDummy;
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleTrainerSlide(void)
{
u32 trainerPicId;
if (gBattleTypeFlags & BATTLE_TYPE_LINK)
{
if ((gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_FIRE_RED
|| (gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_LEAF_GREEN)
{
trainerPicId = gLinkPlayers[GetMultiplayerId()].gender + TRAINER_BACK_PIC_RED;
}
else if ((gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_RUBY
|| (gLinkPlayers[GetMultiplayerId()].version & 0xFF) == VERSION_SAPPHIRE)
{
trainerPicId = gLinkPlayers[GetMultiplayerId()].gender + TRAINER_BACK_PIC_RUBY_SAPPHIRE_BRENDAN;
}
else
{
trainerPicId = gLinkPlayers[GetMultiplayerId()].gender + TRAINER_BACK_PIC_BRENDAN;
}
}
else
{
trainerPicId = gSaveBlock2Ptr->playerGender + TRAINER_BACK_PIC_BRENDAN;
}
2018-02-05 19:46:59 -06:00
DecompressTrainerBackPic(trainerPicId, gActiveBattler);
SetMultiuseSpriteTemplateToTrainerBack(trainerPicId, GetBattlerPosition(gActiveBattler));
gBattlerSpriteIds[gActiveBattler] = CreateSprite(&gMultiuseSpriteTemplate, 80, (8 - gTrainerBackPicCoords[trainerPicId].size) * 4 + 80, 30);
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].oam.paletteNum = gActiveBattler;
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[gActiveBattler]].x2 = -96;
2021-01-22 20:03:21 -05:00
gSprites[gBattlerSpriteIds[gActiveBattler]].sSpeedX = 2;
gSprites[gBattlerSpriteIds[gActiveBattler]].callback = SpriteCB_TrainerSlideIn;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnBankSpriteCallbackDummy2;
}
2021-01-22 20:03:21 -05:00
#undef sSpeedX
2017-10-22 18:43:15 +02:00
static void PlayerHandleTrainerSlideBack(void)
{
2019-02-06 13:17:09 -06:00
SetSpritePrimaryCoordsFromSecondaryCoords(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].data[0] = 50;
gSprites[gBattlerSpriteIds[gActiveBattler]].data[2] = -40;
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[gActiveBattler]].data[4] = gSprites[gBattlerSpriteIds[gActiveBattler]].y;
2018-11-25 17:00:18 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].callback = StartAnimLinearTranslation;
2018-02-06 13:48:02 -06:00
StoreSpriteCallbackInData6(&gSprites[gBattlerSpriteIds[gActiveBattler]], SpriteCallbackDummy);
StartSpriteAnim(&gSprites[gBattlerSpriteIds[gActiveBattler]], 1);
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = FreeTrainerSpriteAfterSlide;
}
2021-01-22 20:03:21 -05:00
#define sSpeedX data[1]
#define sSpeedY data[2]
2017-10-22 18:43:15 +02:00
static void PlayerHandleFaintAnimation(void)
{
2018-02-05 19:46:59 -06:00
if (gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState == 0)
{
2018-02-06 16:09:39 -06:00
if (gBattleSpritesDataPtr->battlerData[gActiveBattler].behindSubstitute)
2018-02-05 19:46:59 -06:00
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gActiveBattler, B_ANIM_SUBSTITUTE_TO_MON);
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState++;
}
else
{
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive)
{
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 0;
2018-02-06 13:48:02 -06:00
HandleLowHpMusicChange(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
2020-08-20 18:02:00 -04:00
PlaySE12WithPanning(SE_FAINT, SOUND_PAN_ATTACKER);
2021-01-22 20:03:21 -05:00
gSprites[gBattlerSpriteIds[gActiveBattler]].sSpeedX = 0;
gSprites[gBattlerSpriteIds[gActiveBattler]].sSpeedY = 5;
gSprites[gBattlerSpriteIds[gActiveBattler]].callback = SpriteCB_FaintSlideAnim;
gBattlerControllerFuncs[gActiveBattler] = FreeMonSpriteAfterFaintAnim;
}
}
}
2021-01-22 20:03:21 -05:00
#undef sSpeedX
#undef sSpeedY
2017-10-22 18:43:15 +02:00
static void PlayerHandlePaletteFade(void)
{
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 2, 0, 16, RGB_BLACK);
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleSuccessBallThrowAnim(void)
{
2017-10-22 18:43:15 +02:00
gBattleSpritesDataPtr->animationData->ballThrowCaseId = BALL_3_SHAKES_SUCCESS;
gDoingBattleAnim = TRUE;
if (IsCriticalCapture())
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gBattlerTarget, B_ANIM_CRITICAL_CAPTURE_THROW);
else
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gBattlerTarget, B_ANIM_BALL_THROW);
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnSpecialAnimDone;
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleBallThrowAnim(void)
{
u8 ballThrowCaseId = gBattleResources->bufferA[gActiveBattler][1];
2017-10-22 18:43:15 +02:00
gBattleSpritesDataPtr->animationData->ballThrowCaseId = ballThrowCaseId;
gDoingBattleAnim = TRUE;
if (IsCriticalCapture())
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gBattlerTarget, B_ANIM_CRITICAL_CAPTURE_THROW);
else
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gBattlerTarget, B_ANIM_BALL_THROW);
2020-07-15 10:13:19 +02:00
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnSpecialAnimDone;
}
2017-10-22 18:43:15 +02:00
static void PlayerHandlePause(void)
{
u8 timer = gBattleResources->bufferA[gActiveBattler][1];
2021-01-22 20:03:21 -05:00
while (timer != 0)
timer--;
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleMoveAnimation(void)
{
2021-01-22 20:03:21 -05:00
if (!IsBattleSEPlaying(gActiveBattler))
{
u16 move = gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8);
gAnimMoveTurn = gBattleResources->bufferA[gActiveBattler][3];
gAnimMovePower = gBattleResources->bufferA[gActiveBattler][4] | (gBattleResources->bufferA[gActiveBattler][5] << 8);
gAnimMoveDmg = gBattleResources->bufferA[gActiveBattler][6] | (gBattleResources->bufferA[gActiveBattler][7] << 8) | (gBattleResources->bufferA[gActiveBattler][8] << 16) | (gBattleResources->bufferA[gActiveBattler][9] << 24);
gAnimFriendship = gBattleResources->bufferA[gActiveBattler][10];
gWeatherMoveAnim = gBattleResources->bufferA[gActiveBattler][12] | (gBattleResources->bufferA[gActiveBattler][13] << 8);
gAnimDisableStructPtr = (struct DisableStruct *)&gBattleResources->bufferA[gActiveBattler][16];
2018-02-05 19:46:59 -06:00
gTransformedPersonalities[gActiveBattler] = gAnimDisableStructPtr->transformedMonPersonality;
if (IsMoveWithoutAnimation(move, gAnimMoveTurn)) // Always returns FALSE.
{
PlayerBufferExecCompleted();
}
else
{
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 0;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = PlayerDoMoveAnimation;
2018-02-27 20:40:09 +01:00
BattleTv_SetDataBasedOnMove(move, gWeatherMoveAnim, gAnimDisableStructPtr);
}
}
}
2017-10-22 18:43:15 +02:00
static void PlayerDoMoveAnimation(void)
{
u16 move = gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8);
u8 multihit = gBattleResources->bufferA[gActiveBattler][11];
2018-02-05 19:46:59 -06:00
switch (gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState)
{
case 0:
2018-02-06 16:09:39 -06:00
if (gBattleSpritesDataPtr->battlerData[gActiveBattler].behindSubstitute
&& !gBattleSpritesDataPtr->battlerData[gActiveBattler].flag_x8)
{
2018-02-06 16:09:39 -06:00
gBattleSpritesDataPtr->battlerData[gActiveBattler].flag_x8 = 1;
2018-02-05 19:46:59 -06:00
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gActiveBattler, B_ANIM_SUBSTITUTE_TO_MON);
}
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 1;
break;
case 1:
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive)
{
2021-01-22 20:03:21 -05:00
SetBattlerSpriteAffineMode(ST_OAM_AFFINE_OFF);
DoMoveAnim(move);
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 2;
}
break;
case 2:
gAnimScriptCallback();
if (!gAnimScriptActive)
{
2021-01-22 20:03:21 -05:00
SetBattlerSpriteAffineMode(ST_OAM_AFFINE_NORMAL);
2018-02-06 16:09:39 -06:00
if (gBattleSpritesDataPtr->battlerData[gActiveBattler].behindSubstitute && multihit < 2)
{
2018-02-05 19:46:59 -06:00
InitAndLaunchSpecialAnimation(gActiveBattler, gActiveBattler, gActiveBattler, B_ANIM_MON_TO_SUBSTITUTE);
2018-02-06 16:09:39 -06:00
gBattleSpritesDataPtr->battlerData[gActiveBattler].flag_x8 = 0;
}
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 3;
}
break;
case 3:
2018-02-05 19:46:59 -06:00
if (!gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].specialAnimActive)
{
2017-11-04 16:11:13 +01:00
CopyAllBattleSpritesInvisibilities();
TrySetBehindSubstituteSpriteBit(gActiveBattler, gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8));
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].animationState = 0;
PlayerBufferExecCompleted();
}
break;
}
}
2017-10-22 18:43:15 +02:00
static void PlayerHandlePrintString(void)
{
u16 *stringId;
gBattle_BG0_X = 0;
gBattle_BG0_Y = 0;
stringId = (u16*)(&gBattleResources->bufferA[gActiveBattler][2]);
BufferStringBattle(*stringId);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gDisplayedStringBattle, 0);
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnInactiveTextPrinter2;
2018-02-27 20:40:09 +01:00
BattleTv_SetDataBasedOnString(*stringId);
2018-11-11 18:33:16 +01:00
BattleArena_DeductMindPoints(gActiveBattler, *stringId);
}
static void PlayerHandlePrintSelectionString(void)
{
2018-02-05 19:46:59 -06:00
if (GetBattlerSide(gActiveBattler) == B_SIDE_PLAYER)
PlayerHandlePrintString();
else
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
static void HandleChooseActionAfterDma3(void)
{
if (!IsDma3ManagerBusyWithBgCopy())
{
gBattle_BG0_X = 0;
2021-04-15 02:04:01 -04:00
gBattle_BG0_Y = DISPLAY_HEIGHT;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseAction;
}
}
2017-10-22 18:43:15 +02:00
static void PlayerHandleChooseAction(void)
{
s32 i;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleChooseActionAfterDma3;
2018-02-27 20:40:09 +01:00
BattleTv_ClearExplosionFaintCause();
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gText_BattleMenu, 2);
for (i = 0; i < 4; i++)
ActionSelectionDestroyCursorAt(i);
2018-02-05 19:46:59 -06:00
ActionSelectionCreateCursorAt(gActionSelectionCursor[gActiveBattler], 0);
BattleStringExpandPlaceholdersToDisplayedString(gText_WhatWillPkmnDo);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gDisplayedStringBattle, 1);
}
static void PlayerHandleYesNoBox(void)
{
2018-02-05 19:46:59 -06:00
if (GetBattlerSide(gActiveBattler) == B_SIDE_PLAYER)
{
HandleBattleWindow(0x18, 8, 0x1D, 0xD, 0);
2018-06-17 16:48:58 +02:00
BattlePutTextOnWindow(gText_BattleYesNoChoice, 12);
gMultiUsePlayerCursor = 1;
BattleCreateYesNoCursorAt(1);
gBattlerControllerFuncs[gActiveBattler] = PlayerHandleYesNoInput;
2017-10-22 18:43:15 +02:00
}
else
{
PlayerBufferExecCompleted();
}
}
static void HandleChooseMoveAfterDma3(void)
{
if (!IsDma3ManagerBusyWithBgCopy())
{
gBattle_BG0_X = 0;
2021-04-15 02:04:01 -04:00
gBattle_BG0_Y = DISPLAY_HEIGHT * 2;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = HandleInputChooseMove;
2017-10-22 18:43:15 +02:00
}
}
2018-11-11 18:33:16 +01:00
// arenaMindPoints is used here as a placeholder for a timer.
2017-10-22 18:43:15 +02:00
static void PlayerChooseMoveInBattlePalace(void)
{
2018-11-11 18:33:16 +01:00
if (--*(gBattleStruct->arenaMindPoints + gActiveBattler) == 0)
2017-10-22 18:43:15 +02:00
{
gBattlePalaceMoveSelectionRngValue = gRngValue;
2018-02-06 13:48:02 -06:00
BtlController_EmitTwoReturnValues(1, 10, ChooseMoveAndTargetInBattlePalace());
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
}
static void PlayerHandleChooseMove(void)
{
if (gBattleTypeFlags & BATTLE_TYPE_PALACE)
{
2018-11-11 18:33:16 +01:00
*(gBattleStruct->arenaMindPoints + gActiveBattler) = 8;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = PlayerChooseMoveInBattlePalace;
2017-10-22 18:43:15 +02:00
}
else
{
InitMoveSelectionsVarsAndStrings();
gBattleStruct->mega.playerSelect = FALSE;
if (!IsMegaTriggerSpriteActive())
gBattleStruct->mega.triggerSpriteId = 0xFF;
2018-09-17 22:40:49 +02:00
if (CanMegaEvolve(gActiveBattler))
CreateMegaTriggerSprite(gActiveBattler, 0);
gBattlerControllerFuncs[gActiveBattler] = HandleChooseMoveAfterDma3;
2017-10-22 18:43:15 +02:00
}
}
void InitMoveSelectionsVarsAndStrings(void)
{
MoveSelectionDisplayMoveNames();
gMultiUsePlayerCursor = 0xFF;
2018-02-05 19:46:59 -06:00
MoveSelectionCreateCursorAt(gMoveSelectionCursor[gActiveBattler], 0);
2017-10-22 18:43:15 +02:00
MoveSelectionDisplayPpString();
MoveSelectionDisplayPpNumber();
MoveSelectionDisplayMoveType();
}
static void PlayerHandleChooseItem(void)
{
s32 i;
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 0x10, RGB_BLACK);
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = OpenBagAndChooseItem;
2018-02-07 22:53:40 +01:00
gBattlerInMenuId = gActiveBattler;
2017-10-22 18:43:15 +02:00
2019-11-03 19:55:19 +01:00
for (i = 0; i < ARRAY_COUNT(gBattlePartyCurrentOrder); i++)
gBattlePartyCurrentOrder[i] = gBattleResources->bufferA[gActiveBattler][1 + i];
2017-10-22 18:43:15 +02:00
}
static void PlayerHandleChoosePokemon(void)
{
s32 i;
2019-11-03 19:55:19 +01:00
for (i = 0; i < ARRAY_COUNT(gBattlePartyCurrentOrder); i++)
gBattlePartyCurrentOrder[i] = gBattleResources->bufferA[gActiveBattler][4 + i];
2017-10-22 18:43:15 +02:00
2019-11-03 19:55:19 +01:00
if (gBattleTypeFlags & BATTLE_TYPE_ARENA && (gBattleResources->bufferA[gActiveBattler][1] & 0xF) != PARTY_ACTION_CANT_SWITCH)
2017-10-22 18:43:15 +02:00
{
2019-10-17 19:22:03 -04:00
BtlController_EmitChosenMonReturnValue(1, gBattlerPartyIndexes[gActiveBattler] + 1, gBattlePartyCurrentOrder);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
else
{
2021-01-22 20:03:21 -05:00
gBattleControllerData[gActiveBattler] = CreateTask(TaskDummy, 0xFF);
gTasks[gBattleControllerData[gActiveBattler]].data[0] = gBattleResources->bufferA[gActiveBattler][1] & 0xF;
*(&gBattleStruct->battlerPreventingSwitchout) = gBattleResources->bufferA[gActiveBattler][1] >> 4;
*(&gBattleStruct->prevSelectedPartySlot) = gBattleResources->bufferA[gActiveBattler][2];
*(&gBattleStruct->abilityPreventingSwitchout) = (gBattleResources->bufferA[gActiveBattler][3] & 0xFF) | (gBattleResources->bufferA[gActiveBattler][7] << 8);
2019-04-04 17:05:46 -04:00
BeginNormalPaletteFade(0xFFFFFFFF, 0, 0, 0x10, RGB_BLACK);
2018-07-07 19:57:09 +02:00
gBattlerControllerFuncs[gActiveBattler] = OpenPartyMenuToChooseMon;
2018-02-07 22:53:40 +01:00
gBattlerInMenuId = gActiveBattler;
2017-10-22 18:43:15 +02:00
}
}
static void PlayerHandleCmd23(void)
{
2017-11-04 16:11:13 +01:00
BattleStopLowHpSound();
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 2, 0, 16, RGB_BLACK);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleHealthBarUpdate(void)
{
s16 hpVal;
LoadBattleBarGfx(0);
hpVal = gBattleResources->bufferA[gActiveBattler][2] | (gBattleResources->bufferA[gActiveBattler][3] << 8);
2017-10-22 18:43:15 +02:00
// gPlayerPartyLostHP used by Battle Dome, but never read
2017-10-22 18:43:15 +02:00
if (hpVal > 0)
gPlayerPartyLostHP += hpVal;
2017-10-22 18:43:15 +02:00
if (hpVal != INSTANT_HP_BAR_DROP)
{
2018-02-06 13:48:02 -06:00
u32 maxHP = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_MAX_HP);
u32 curHP = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_HP);
2017-10-22 18:43:15 +02:00
2018-02-06 13:48:02 -06:00
SetBattleBarStruct(gActiveBattler, gHealthboxSpriteIds[gActiveBattler], maxHP, curHP, hpVal);
2017-10-22 18:43:15 +02:00
}
else
{
2018-02-06 13:48:02 -06:00
u32 maxHP = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_MAX_HP);
2017-10-22 18:43:15 +02:00
2018-02-06 13:48:02 -06:00
SetBattleBarStruct(gActiveBattler, gHealthboxSpriteIds[gActiveBattler], maxHP, 0, hpVal);
UpdateHpTextInHealthbox(gHealthboxSpriteIds[gActiveBattler], 0, HP_CURRENT);
2017-10-22 18:43:15 +02:00
}
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnHealthbarDone;
2017-10-22 18:43:15 +02:00
}
static void PlayerHandleExpUpdate(void)
{
u8 monId = gBattleResources->bufferA[gActiveBattler][1];
2021-06-03 23:17:44 +02:00
s32 taskId, expPointsToGive;
2017-10-22 18:43:15 +02:00
2018-10-16 21:47:08 -05:00
if (GetMonData(&gPlayerParty[monId], MON_DATA_LEVEL) >= MAX_LEVEL)
2017-10-22 18:43:15 +02:00
{
PlayerBufferExecCompleted();
}
else
{
LoadBattleBarGfx(1);
GetMonData(&gPlayerParty[monId], MON_DATA_SPECIES); // Unused return value.
2021-06-03 23:17:44 +02:00
expPointsToGive = T1_READ_32(&gBattleResources->bufferA[gActiveBattler][2]);
2017-10-22 18:43:15 +02:00
taskId = CreateTask(Task_GiveExpToMon, 10);
gTasks[taskId].tExpTask_monId = monId;
2021-06-03 23:17:44 +02:00
gTasks[taskId].tExpTask_gainedExp_1 = expPointsToGive;
gTasks[taskId].tExpTask_gainedExp_2 = expPointsToGive >> 16;
gTasks[taskId].tExpTask_battler = gActiveBattler;
gBattlerControllerFuncs[gActiveBattler] = BattleControllerDummy;
2017-10-22 18:43:15 +02:00
}
}
#undef tExpTask_monId
#undef tExpTask_battler
2021-06-03 23:17:44 +02:00
#undef tExpTask_gainedExp_1
#undef tExpTask_gainedExp_2
2017-10-22 18:43:15 +02:00
#undef tExpTask_frames
static void PlayerHandleStatusIconUpdate(void)
{
2021-01-22 20:03:21 -05:00
if (!IsBattleSEPlaying(gActiveBattler))
2017-10-22 18:43:15 +02:00
{
2018-03-01 00:59:52 +01:00
u8 battlerId;
2017-10-22 18:43:15 +02:00
2018-02-06 13:48:02 -06:00
UpdateHealthboxAttribute(gHealthboxSpriteIds[gActiveBattler], &gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], HEALTHBOX_STATUS_ICON);
2018-03-01 00:59:52 +01:00
battlerId = gActiveBattler;
gBattleSpritesDataPtr->healthBoxesData[battlerId].statusAnimActive = 0;
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnFinishedStatusAnimation;
2017-10-22 18:43:15 +02:00
}
}
static void PlayerHandleStatusAnimation(void)
{
2021-01-22 20:03:21 -05:00
if (!IsBattleSEPlaying(gActiveBattler))
2017-10-22 18:43:15 +02:00
{
InitAndLaunchChosenStatusAnimation(gBattleResources->bufferA[gActiveBattler][1],
gBattleResources->bufferA[gActiveBattler][2] | (gBattleResources->bufferA[gActiveBattler][3] << 8) | (gBattleResources->bufferA[gActiveBattler][4] << 16) | (gBattleResources->bufferA[gActiveBattler][5] << 24));
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnFinishedStatusAnimation;
2017-10-22 18:43:15 +02:00
}
}
static void PlayerHandleStatusXor(void)
{
u8 val = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_STATUS) ^ gBattleResources->bufferA[gActiveBattler][1];
2017-10-22 18:43:15 +02:00
2018-02-06 13:48:02 -06:00
SetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_STATUS, &val);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleDataTransfer(void)
{
PlayerBufferExecCompleted();
}
static void PlayerHandleDMA3Transfer(void)
{
u32 dstArg = gBattleResources->bufferA[gActiveBattler][1]
| (gBattleResources->bufferA[gActiveBattler][2] << 8)
| (gBattleResources->bufferA[gActiveBattler][3] << 16)
| (gBattleResources->bufferA[gActiveBattler][4] << 24);
u16 sizeArg = gBattleResources->bufferA[gActiveBattler][5] | (gBattleResources->bufferA[gActiveBattler][6] << 8);
2017-10-22 18:43:15 +02:00
const u8 *src = &gBattleResources->bufferA[gActiveBattler][7];
2017-10-22 18:43:15 +02:00
u8 *dst = (u8*)(dstArg);
u32 size = sizeArg;
while (1)
{
if (size <= 0x1000)
{
DmaCopy16(3, src, dst, size);
break;
}
DmaCopy16(3, src, dst, 0x1000);
src += 0x1000;
dst += 0x1000;
size -= 0x1000;
}
PlayerBufferExecCompleted();
}
static void PlayerHandlePlayBGM(void)
{
PlayBGM(gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8));
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleCmd32(void)
{
PlayerBufferExecCompleted();
}
static void PlayerHandleTwoReturnValues(void)
{
2018-02-06 13:48:02 -06:00
BtlController_EmitTwoReturnValues(1, 0, 0);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleChosenMonReturnValue(void)
{
2018-02-06 13:48:02 -06:00
BtlController_EmitChosenMonReturnValue(1, 0, NULL);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleOneReturnValue(void)
{
2018-02-06 13:48:02 -06:00
BtlController_EmitOneReturnValue(1, 0);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleOneReturnValue_Duplicate(void)
{
2018-02-06 13:48:02 -06:00
BtlController_EmitOneReturnValue_Duplicate(1, 0);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
2020-12-29 16:51:44 -05:00
static void PlayerHandleClearUnkVar(void)
2017-10-22 18:43:15 +02:00
{
2020-12-29 16:51:44 -05:00
gUnusedControllerStruct.unk = 0;
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
2020-12-29 16:51:44 -05:00
static void PlayerHandleSetUnkVar(void)
2017-10-22 18:43:15 +02:00
{
gUnusedControllerStruct.unk = gBattleResources->bufferA[gActiveBattler][1];
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
2020-12-29 16:51:44 -05:00
static void PlayerHandleClearUnkFlag(void)
2017-10-22 18:43:15 +02:00
{
2020-12-29 16:51:44 -05:00
gUnusedControllerStruct.flag = 0;
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
2020-12-29 16:51:44 -05:00
static void PlayerHandleToggleUnkFlag(void)
2017-10-22 18:43:15 +02:00
{
2020-12-29 16:51:44 -05:00
gUnusedControllerStruct.flag ^= 1;
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleHitAnimation(void)
{
2018-02-06 13:48:02 -06:00
if (gSprites[gBattlerSpriteIds[gActiveBattler]].invisible == TRUE)
2017-10-22 18:43:15 +02:00
{
PlayerBufferExecCompleted();
}
else
{
2017-10-24 15:25:20 +02:00
gDoingBattleAnim = TRUE;
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].data[1] = 0;
2018-02-05 19:46:59 -06:00
DoHitAnimHealthboxEffect(gActiveBattler);
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = DoHitAnimBlinkSpriteEffect;
2017-10-22 18:43:15 +02:00
}
}
2021-04-03 12:38:07 -04:00
static void PlayerHandleCantSwitch(void)
2017-10-22 18:43:15 +02:00
{
PlayerBufferExecCompleted();
}
static void PlayerHandlePlaySE(void)
2017-10-22 18:43:15 +02:00
{
s8 pan;
2018-02-05 19:46:59 -06:00
if (GetBattlerSide(gActiveBattler) == B_SIDE_PLAYER)
pan = SOUND_PAN_ATTACKER;
2017-10-22 18:43:15 +02:00
else
pan = SOUND_PAN_TARGET;
2017-10-22 18:43:15 +02:00
PlaySE12WithPanning(gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8), pan);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandlePlayFanfareOrBGM(void)
{
if (gBattleResources->bufferA[gActiveBattler][3])
2017-10-22 18:43:15 +02:00
{
2017-11-04 16:11:13 +01:00
BattleStopLowHpSound();
PlayBGM(gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8));
2017-10-22 18:43:15 +02:00
}
else
{
PlayFanfare(gBattleResources->bufferA[gActiveBattler][1] | (gBattleResources->bufferA[gActiveBattler][2] << 8));
2017-10-22 18:43:15 +02:00
}
PlayerBufferExecCompleted();
}
static void PlayerHandleFaintingCry(void)
{
2018-02-06 13:48:02 -06:00
u16 species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], MON_DATA_SPECIES);
2017-10-22 18:43:15 +02:00
PlayCry3(species, -25, 5);
PlayerBufferExecCompleted();
}
static void PlayerHandleIntroSlide(void)
{
HandleIntroSlide(gBattleResources->bufferA[gActiveBattler][1]);
2018-02-08 11:17:41 +01:00
gIntroSlideFlags |= 1;
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
2021-01-22 20:03:21 -05:00
// Task data for Task_StartSendOutAnim
#define tBattlerId data[0]
#define tStartTimer data[1]
#define sBattlerId data[5]
2017-10-22 18:43:15 +02:00
static void PlayerHandleIntroTrainerBallThrow(void)
{
u8 paletteNum;
u8 taskId;
2019-02-06 13:17:09 -06:00
SetSpritePrimaryCoordsFromSecondaryCoords(&gSprites[gBattlerSpriteIds[gActiveBattler]]);
2017-10-22 18:43:15 +02:00
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].data[0] = 50;
gSprites[gBattlerSpriteIds[gActiveBattler]].data[2] = -40;
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[gActiveBattler]].data[4] = gSprites[gBattlerSpriteIds[gActiveBattler]].y;
2018-11-25 17:00:18 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].callback = StartAnimLinearTranslation;
2021-01-22 20:03:21 -05:00
gSprites[gBattlerSpriteIds[gActiveBattler]].sBattlerId = gActiveBattler;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
StoreSpriteCallbackInData6(&gSprites[gBattlerSpriteIds[gActiveBattler]], SpriteCB_FreePlayerSpriteLoadMonSprite);
2018-02-06 13:48:02 -06:00
StartSpriteAnim(&gSprites[gBattlerSpriteIds[gActiveBattler]], 1);
2017-10-22 18:43:15 +02:00
paletteNum = AllocSpritePalette(0xD6F8);
LoadCompressedPalette(gTrainerBackPicPaletteTable[gSaveBlock2Ptr->playerGender].data, 0x100 + paletteNum * 16, 32);
2018-02-06 13:48:02 -06:00
gSprites[gBattlerSpriteIds[gActiveBattler]].oam.paletteNum = paletteNum;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
taskId = CreateTask(Task_StartSendOutAnim, 5);
gTasks[taskId].tBattlerId = gActiveBattler;
2017-10-22 18:43:15 +02:00
2018-06-28 21:06:32 +02:00
if (gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusSummaryShown)
gTasks[gBattlerStatusSummaryTaskId[gActiveBattler]].func = Task_HidePartyStatusSummary;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->animationData->introAnimActive = TRUE;
gBattlerControllerFuncs[gActiveBattler] = BattleControllerDummy;
2017-10-22 18:43:15 +02:00
}
2021-01-22 20:03:21 -05:00
void SpriteCB_FreePlayerSpriteLoadMonSprite(struct Sprite *sprite)
2017-10-22 18:43:15 +02:00
{
2021-01-22 20:03:21 -05:00
u8 battlerId = sprite->sBattlerId;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
// Free player trainer sprite
2017-10-22 18:43:15 +02:00
FreeSpriteOamMatrix(sprite);
FreeSpritePaletteByTag(GetSpritePaletteTagByPaletteNum(sprite->oam.paletteNum));
DestroySprite(sprite);
2021-01-22 20:03:21 -05:00
// Load mon sprite
2018-03-01 00:59:52 +01:00
BattleLoadPlayerMonSpriteGfx(&gPlayerParty[gBattlerPartyIndexes[battlerId]], battlerId);
StartSpriteAnim(&gSprites[gBattlerSpriteIds[battlerId]], 0);
2017-10-22 18:43:15 +02:00
}
2021-01-22 20:03:21 -05:00
#undef sBattlerId
// Send out at start of battle
static void Task_StartSendOutAnim(u8 taskId)
2017-10-22 18:43:15 +02:00
{
2021-01-22 20:03:21 -05:00
if (gTasks[taskId].tStartTimer < 31)
2017-10-22 18:43:15 +02:00
{
2021-01-22 20:03:21 -05:00
gTasks[taskId].tStartTimer++;
}
else
2017-10-22 18:43:15 +02:00
{
u8 savedActiveBattler = gActiveBattler;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
gActiveBattler = gTasks[taskId].tBattlerId;
2017-10-22 18:43:15 +02:00
if (!IsDoubleBattle() || (gBattleTypeFlags & BATTLE_TYPE_MULTI))
{
gBattleResources->bufferA[gActiveBattler][1] = gBattlerPartyIndexes[gActiveBattler];
2021-01-22 20:03:21 -05:00
StartSendOutAnim(gActiveBattler, FALSE);
2017-10-22 18:43:15 +02:00
}
else
{
gBattleResources->bufferA[gActiveBattler][1] = gBattlerPartyIndexes[gActiveBattler];
2021-01-22 20:03:21 -05:00
StartSendOutAnim(gActiveBattler, FALSE);
2018-02-05 19:46:59 -06:00
gActiveBattler ^= BIT_FLANK;
gBattleResources->bufferA[gActiveBattler][1] = gBattlerPartyIndexes[gActiveBattler];
2018-02-06 13:48:02 -06:00
BattleLoadPlayerMonSpriteGfx(&gPlayerParty[gBattlerPartyIndexes[gActiveBattler]], gActiveBattler);
2021-01-22 20:03:21 -05:00
StartSendOutAnim(gActiveBattler, FALSE);
2018-02-05 19:46:59 -06:00
gActiveBattler ^= BIT_FLANK;
2017-10-22 18:43:15 +02:00
}
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = Intro_TryShinyAnimShowHealthbox;
gActiveBattler = savedActiveBattler;
2017-10-22 18:43:15 +02:00
DestroyTask(taskId);
}
}
2021-01-22 20:03:21 -05:00
#undef tBattlerId
#undef tStartTimer
2017-10-22 18:43:15 +02:00
static void PlayerHandleDrawPartyStatusSummary(void)
{
if (gBattleResources->bufferA[gActiveBattler][1] != 0 && GetBattlerSide(gActiveBattler) == B_SIDE_PLAYER)
{
PlayerBufferExecCompleted();
}
2017-10-22 18:43:15 +02:00
else
{
2018-06-28 21:06:32 +02:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusSummaryShown = 1;
gBattlerStatusSummaryTaskId[gActiveBattler] = CreatePartyStatusSummarySprites(gActiveBattler, (struct HpAndStatus *)&gBattleResources->bufferA[gActiveBattler][4], gBattleResources->bufferA[gActiveBattler][1], gBattleResources->bufferA[gActiveBattler][2]);
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusDelayTimer = 0;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
// If intro, skip the delay after drawing
if (gBattleResources->bufferA[gActiveBattler][2] != 0)
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusDelayTimer = 93;
2017-10-22 18:43:15 +02:00
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = EndDrawPartyStatusSummary;
2017-10-22 18:43:15 +02:00
}
}
2021-01-22 20:03:21 -05:00
static void EndDrawPartyStatusSummary(void)
2017-10-22 18:43:15 +02:00
{
2021-01-22 20:03:21 -05:00
if (gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusDelayTimer++ > 92)
2017-10-22 18:43:15 +02:00
{
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusDelayTimer = 0;
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
}
2018-06-28 21:06:32 +02:00
static void PlayerHandleHidePartyStatusSummary(void)
2017-10-22 18:43:15 +02:00
{
2018-06-28 21:06:32 +02:00
if (gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].partyStatusSummaryShown)
gTasks[gBattlerStatusSummaryTaskId[gActiveBattler]].func = Task_HidePartyStatusSummary;
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
2018-06-20 23:07:51 +02:00
static void PlayerHandleEndBounceEffect(void)
2017-10-22 18:43:15 +02:00
{
2018-06-20 23:07:51 +02:00
EndBounceEffect(gActiveBattler, BOUNCE_HEALTHBOX);
EndBounceEffect(gActiveBattler, BOUNCE_MON);
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
}
static void PlayerHandleSpriteInvisibility(void)
{
2018-02-06 13:48:02 -06:00
if (IsBattlerSpritePresent(gActiveBattler))
2017-10-22 18:43:15 +02:00
{
gSprites[gBattlerSpriteIds[gActiveBattler]].invisible = gBattleResources->bufferA[gActiveBattler][1];
2018-02-05 19:46:59 -06:00
CopyBattleSpriteInvisibility(gActiveBattler);
2017-10-22 18:43:15 +02:00
}
PlayerBufferExecCompleted();
}
static void PlayerHandleBattleAnimation(void)
{
2021-01-22 20:03:21 -05:00
if (!IsBattleSEPlaying(gActiveBattler))
2017-10-22 18:43:15 +02:00
{
u8 animationId = gBattleResources->bufferA[gActiveBattler][1];
u16 argument = gBattleResources->bufferA[gActiveBattler][2] | (gBattleResources->bufferA[gActiveBattler][3] << 8);
2017-10-22 18:43:15 +02:00
2018-02-05 19:46:59 -06:00
if (TryHandleLaunchBattleTableAnimation(gActiveBattler, gActiveBattler, gActiveBattler, animationId, argument))
2017-10-22 18:43:15 +02:00
PlayerBufferExecCompleted();
else
2018-02-06 13:48:02 -06:00
gBattlerControllerFuncs[gActiveBattler] = CompleteOnFinishedBattleAnimation;
2017-10-22 18:43:15 +02:00
2018-02-27 20:40:09 +01:00
BattleTv_SetDataBasedOnAnimation(animationId);
2017-10-22 18:43:15 +02:00
}
}
static void PlayerHandleLinkStandbyMsg(void)
{
RecordedBattle_RecordAllBattlerData(&gBattleResources->bufferA[gActiveBattler][2]);
switch (gBattleResources->bufferA[gActiveBattler][1])
2017-10-22 18:43:15 +02:00
{
case 0:
PrintLinkStandbyMsg();
// fall through
case 1:
2018-06-20 23:07:51 +02:00
EndBounceEffect(gActiveBattler, BOUNCE_HEALTHBOX);
EndBounceEffect(gActiveBattler, BOUNCE_MON);
2017-10-22 18:43:15 +02:00
break;
case 2:
PrintLinkStandbyMsg();
break;
}
PlayerBufferExecCompleted();
}
static void PlayerHandleResetActionMoveSelection(void)
{
switch (gBattleResources->bufferA[gActiveBattler][1])
2017-10-22 18:43:15 +02:00
{
case RESET_ACTION_MOVE_SELECTION:
2018-02-05 19:46:59 -06:00
gActionSelectionCursor[gActiveBattler] = 0;
gMoveSelectionCursor[gActiveBattler] = 0;
2017-10-22 18:43:15 +02:00
break;
case RESET_ACTION_SELECTION:
2018-02-05 19:46:59 -06:00
gActionSelectionCursor[gActiveBattler] = 0;
2017-10-22 18:43:15 +02:00
break;
case RESET_MOVE_SELECTION:
2018-02-05 19:46:59 -06:00
gMoveSelectionCursor[gActiveBattler] = 0;
2017-10-22 18:43:15 +02:00
break;
}
PlayerBufferExecCompleted();
}
static void PlayerHandleEndLinkBattle(void)
2017-10-22 18:43:15 +02:00
{
RecordedBattle_RecordAllBattlerData(&gBattleResources->bufferA[gActiveBattler][4]);
gBattleOutcome = gBattleResources->bufferA[gActiveBattler][1];
2019-12-30 18:52:23 +01:00
gSaveBlock2Ptr->frontier.disableRecordBattle = gBattleResources->bufferA[gActiveBattler][2];
2017-10-22 18:43:15 +02:00
FadeOutMapMusic(5);
BeginFastPaletteFade(3);
PlayerBufferExecCompleted();
2021-01-22 20:03:21 -05:00
gBattlerControllerFuncs[gActiveBattler] = SetBattleEndCallbacks;
2017-10-22 18:43:15 +02:00
}
2018-07-13 23:00:56 +02:00
static void WaitForDebug(void)
{
if (gMain.callback2 == BattleMainCB2 && !gPaletteFade.active)
{
PlayerBufferExecCompleted();
}
}
static void PlayerHandleBattleDebug(void)
{
BeginNormalPaletteFade(-1, 0, 0, 0x10, 0);
SetMainCallback2(CB2_BattleDebugMenu);
gBattlerControllerFuncs[gActiveBattler] = WaitForDebug;
}
static void PlayerCmdEnd(void)
2017-10-22 18:43:15 +02:00
{
}