pokeemerald/src/pokeball.c

1298 lines
39 KiB
C
Raw Normal View History

2017-12-02 19:39:07 +01:00
#include "global.h"
#include "battle.h"
#include "battle_anim.h"
2018-11-14 00:01:50 +00:00
#include "decompress.h"
#include "graphics.h"
2017-12-02 19:39:07 +01:00
#include "main.h"
#include "m4a.h"
2018-11-14 00:01:50 +00:00
#include "pokeball.h"
#include "pokemon.h"
#include "sound.h"
#include "sprite.h"
#include "task.h"
#include "trig.h"
2017-12-02 20:38:26 +01:00
#include "util.h"
#include "data.h"
2018-11-14 00:01:50 +00:00
#include "constants/songs.h"
2017-12-02 19:39:07 +01:00
extern struct MusicPlayerInfo gMPlayInfo_BGM;
2017-12-02 19:39:07 +01:00
// this file's functions
2017-12-02 21:27:00 +01:00
static void Task_DoPokeballSendOutAnim(u8 taskId);
static void SpriteCB_PlayerMonSendOut_1(struct Sprite *sprite);
static void SpriteCB_PlayerMonSendOut_2(struct Sprite *sprite);
static void SpriteCB_OpponentMonSendOut(struct Sprite *sprite);
static void SpriteCB_BallThrow(struct Sprite *sprite);
static void SpriteCB_BallThrow_ReachMon(struct Sprite *sprite);
static void SpriteCB_BallThrow_StartShrinkMon(struct Sprite *sprite);
static void SpriteCB_BallThrow_ShrinkMon(struct Sprite *sprite);
static void SpriteCB_BallThrow_Close(struct Sprite *sprite);
static void SpriteCB_BallThrow_FallToGround(struct Sprite *sprite);
static void SpriteCB_BallThrow_StartShakes(struct Sprite *sprite);
static void SpriteCB_BallThrow_Shake(struct Sprite *sprite);
static void SpriteCB_BallThrow_StartCaptureMon(struct Sprite *sprite);
static void SpriteCB_BallThrow_CaptureMon(struct Sprite *sprite);
2017-12-02 21:27:00 +01:00
static void SpriteCB_ReleaseMonFromBall(struct Sprite *sprite);
static void SpriteCB_ReleaseMon2FromBall(struct Sprite *sprite);
static void HandleBallAnimEnd(struct Sprite *sprite);
2021-01-22 23:22:37 -05:00
static void SpriteCB_PokeballReleaseMon(struct Sprite *sprite);
static void SpriteCB_ReleasedMonFlyOut(struct Sprite *sprite);
static void SpriteCB_TradePokeball(struct Sprite *sprite);
static void SpriteCB_TradePokeballSendOff(struct Sprite *sprite);
static void SpriteCB_TradePokeballEnd(struct Sprite *sprite);
static void SpriteCB_HealthboxSlideInDelayed(struct Sprite *sprite);
static void SpriteCB_HealthboxSlideIn(struct Sprite *sprite);
2017-12-02 21:27:00 +01:00
static void SpriteCB_HitAnimHealthoxEffect(struct Sprite *sprite);
2018-03-01 00:59:52 +01:00
static u16 GetBattlerPokeballItemId(u8 battlerId);
2017-12-02 21:27:00 +01:00
// rom const data
#define GFX_TAG_POKEBALL 55000
#define GFX_TAG_GREATBALL 55001
#define GFX_TAG_SAFARIBALL 55002
#define GFX_TAG_ULTRABALL 55003
#define GFX_TAG_MASTERBALL 55004
#define GFX_TAG_NETBALL 55005
#define GFX_TAG_DIVEBALL 55006
#define GFX_TAG_NESTBALL 55007
#define GFX_TAG_REPEATBALL 55008
#define GFX_TAG_TIMERBALL 55009
#define GFX_TAG_LUXURYBALL 55010
#define GFX_TAG_PREMIERBALL 55011
const struct CompressedSpriteSheet gBallSpriteSheets[POKEBALL_COUNT] =
{
2021-01-22 23:22:37 -05:00
[BALL_POKE] = {gInterfaceGfx_PokeBall, 384, GFX_TAG_POKEBALL},
[BALL_GREAT] = {gInterfaceGfx_GreatBall, 384, GFX_TAG_GREATBALL},
[BALL_SAFARI] = {gInterfaceGfx_SafariBall, 384, GFX_TAG_SAFARIBALL},
[BALL_ULTRA] = {gInterfaceGfx_UltraBall, 384, GFX_TAG_ULTRABALL},
[BALL_MASTER] = {gInterfaceGfx_MasterBall, 384, GFX_TAG_MASTERBALL},
[BALL_NET] = {gInterfaceGfx_NetBall, 384, GFX_TAG_NETBALL},
[BALL_DIVE] = {gInterfaceGfx_DiveBall, 384, GFX_TAG_DIVEBALL},
[BALL_NEST] = {gInterfaceGfx_NestBall, 384, GFX_TAG_NESTBALL},
[BALL_REPEAT] = {gInterfaceGfx_RepeatBall, 384, GFX_TAG_REPEATBALL},
[BALL_TIMER] = {gInterfaceGfx_TimerBall, 384, GFX_TAG_TIMERBALL},
[BALL_LUXURY] = {gInterfaceGfx_LuxuryBall, 384, GFX_TAG_LUXURYBALL},
[BALL_PREMIER] = {gInterfaceGfx_PremierBall, 384, GFX_TAG_PREMIERBALL},
2017-12-02 21:27:00 +01:00
};
const struct CompressedSpritePalette gBallSpritePalettes[POKEBALL_COUNT] =
{
2021-01-22 23:22:37 -05:00
[BALL_POKE] = {gInterfacePal_PokeBall, GFX_TAG_POKEBALL},
[BALL_GREAT] = {gInterfacePal_GreatBall, GFX_TAG_GREATBALL},
[BALL_SAFARI] = {gInterfacePal_SafariBall, GFX_TAG_SAFARIBALL},
[BALL_ULTRA] = {gInterfacePal_UltraBall, GFX_TAG_ULTRABALL},
[BALL_MASTER] = {gInterfacePal_MasterBall, GFX_TAG_MASTERBALL},
[BALL_NET] = {gInterfacePal_NetBall, GFX_TAG_NETBALL},
[BALL_DIVE] = {gInterfacePal_DiveBall, GFX_TAG_DIVEBALL},
[BALL_NEST] = {gInterfacePal_NestBall, GFX_TAG_NESTBALL},
[BALL_REPEAT] = {gInterfacePal_RepeatBall, GFX_TAG_REPEATBALL},
[BALL_TIMER] = {gInterfacePal_TimerBall, GFX_TAG_TIMERBALL},
[BALL_LUXURY] = {gInterfacePal_LuxuryBall, GFX_TAG_LUXURYBALL},
[BALL_PREMIER] = {gInterfacePal_PremierBall, GFX_TAG_PREMIERBALL},
2017-12-02 21:27:00 +01:00
};
static const struct OamData sBallOamData =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_DOUBLE,
.objMode = ST_OAM_OBJ_NORMAL,
2017-12-02 21:27:00 +01:00
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(16x16),
2017-12-02 21:27:00 +01:00
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(16x16),
2017-12-02 21:27:00 +01:00
.tileNum = 0,
.priority = 2,
.paletteNum = 0,
.affineParam = 0,
};
static const union AnimCmd sBallAnimSeq3[] =
{
ANIMCMD_FRAME(0, 5),
ANIMCMD_JUMP(0),
};
static const union AnimCmd sBallAnimSeq5[] =
{
ANIMCMD_FRAME(4, 1),
ANIMCMD_JUMP(0),
};
static const union AnimCmd sBallAnimSeq4[] =
{
ANIMCMD_FRAME(8, 5),
ANIMCMD_JUMP(0),
};
static const union AnimCmd sBallAnimSeq6[] =
{
ANIMCMD_FRAME(12, 1),
ANIMCMD_JUMP(0),
};
static const union AnimCmd sBallAnimSeq0[] =
{
ANIMCMD_FRAME(0, 1),
ANIMCMD_END,
};
static const union AnimCmd sBallAnimSeq1[] =
{
ANIMCMD_FRAME(4, 5),
ANIMCMD_FRAME(8, 5),
ANIMCMD_END,
};
static const union AnimCmd sBallAnimSeq2[] =
{
ANIMCMD_FRAME(4, 5),
ANIMCMD_FRAME(0, 5),
ANIMCMD_END,
};
static const union AnimCmd *const sBallAnimSequences[] =
{
sBallAnimSeq0,
sBallAnimSeq1,
sBallAnimSeq2,
// unused?
sBallAnimSeq3,
sBallAnimSeq4,
sBallAnimSeq5,
sBallAnimSeq6,
};
static const union AffineAnimCmd sAffineAnim_BallRotate_0[] =
2017-12-02 21:27:00 +01:00
{
AFFINEANIMCMD_FRAME(0, 0, 0, 1),
AFFINEANIMCMD_JUMP(0),
};
static const union AffineAnimCmd sAffineAnim_BallRotate_Right[] =
2017-12-02 21:27:00 +01:00
{
AFFINEANIMCMD_FRAME(0, 0, -3, 1),
AFFINEANIMCMD_JUMP(0),
};
static const union AffineAnimCmd sAffineAnim_BallRotate_Left[] =
2017-12-02 21:27:00 +01:00
{
AFFINEANIMCMD_FRAME(0, 0, 3, 1),
AFFINEANIMCMD_JUMP(0),
};
static const union AffineAnimCmd sAffineAnim_BallRotate_3[] =
2017-12-02 21:27:00 +01:00
{
AFFINEANIMCMD_FRAME(256, 256, 0, 0),
AFFINEANIMCMD_END,
};
static const union AffineAnimCmd sAffineAnim_BallRotate_4[] =
2017-12-02 21:27:00 +01:00
{
AFFINEANIMCMD_FRAME(0, 0, 25, 1),
AFFINEANIMCMD_JUMP(0),
};
static const union AffineAnimCmd *const sAffineAnim_BallRotate[] =
2017-12-02 21:27:00 +01:00
{
[BALL_AFFINE_ANIM_0] = sAffineAnim_BallRotate_0,
[BALL_ROTATE_RIGHT] = sAffineAnim_BallRotate_Right,
[BALL_ROTATE_LEFT] = sAffineAnim_BallRotate_Left,
[BALL_AFFINE_ANIM_3] = sAffineAnim_BallRotate_3,
[BALL_AFFINE_ANIM_4] = sAffineAnim_BallRotate_4,
2017-12-02 21:27:00 +01:00
};
const struct SpriteTemplate gBallSpriteTemplates[POKEBALL_COUNT] =
{
{
.tileTag = GFX_TAG_POKEBALL,
.paletteTag = GFX_TAG_POKEBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_GREATBALL,
.paletteTag = GFX_TAG_GREATBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_SAFARIBALL,
.paletteTag = GFX_TAG_SAFARIBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_ULTRABALL,
.paletteTag = GFX_TAG_ULTRABALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_MASTERBALL,
.paletteTag = GFX_TAG_MASTERBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_NETBALL,
.paletteTag = GFX_TAG_NETBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_DIVEBALL,
.paletteTag = GFX_TAG_DIVEBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_NESTBALL,
.paletteTag = GFX_TAG_NESTBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_REPEATBALL,
.paletteTag = GFX_TAG_REPEATBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_TIMERBALL,
.paletteTag = GFX_TAG_TIMERBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_LUXURYBALL,
.paletteTag = GFX_TAG_LUXURYBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
{
.tileTag = GFX_TAG_PREMIERBALL,
.paletteTag = GFX_TAG_PREMIERBALL,
.oam = &sBallOamData,
.anims = sBallAnimSequences,
.images = NULL,
.affineAnims = sAffineAnim_BallRotate,
.callback = SpriteCB_BallThrow,
2017-12-02 21:27:00 +01:00
},
};
2017-12-02 19:39:07 +01:00
2018-03-01 00:59:52 +01:00
#define tFrames data[0]
#define tPan data[1]
#define tThrowId data[2]
#define tBattler data[3]
#define tOpponentBattler data[4]
2017-12-02 19:39:07 +01:00
u8 DoPokeballSendOutAnimation(s16 pan, u8 kindOfThrow)
{
u8 taskId;
gDoingBattleAnim = TRUE;
2018-02-05 19:46:59 -06:00
gBattleSpritesDataPtr->healthBoxesData[gActiveBattler].ballAnimActive = 1;
2017-12-02 19:39:07 +01:00
taskId = CreateTask(Task_DoPokeballSendOutAnim, 5);
gTasks[taskId].tPan = pan;
gTasks[taskId].tThrowId = kindOfThrow;
2018-03-01 00:59:52 +01:00
gTasks[taskId].tBattler = gActiveBattler;
2017-12-02 19:39:07 +01:00
return 0;
}
2018-03-01 00:59:52 +01:00
#define sBattler data[6]
2017-12-02 19:39:07 +01:00
2017-12-02 21:27:00 +01:00
static void Task_DoPokeballSendOutAnim(u8 taskId)
2017-12-02 19:39:07 +01:00
{
2019-08-08 13:06:55 +02:00
u32 throwCaseId, ballId, battlerId, ballSpriteId;
bool32 notSendOut = FALSE;
2017-12-02 19:39:07 +01:00
if (gTasks[taskId].tFrames == 0)
{
gTasks[taskId].tFrames++;
return;
}
throwCaseId = gTasks[taskId].tThrowId;
2018-03-01 00:59:52 +01:00
battlerId = gTasks[taskId].tBattler;
2019-08-08 13:06:55 +02:00
ballId = ItemIdToBallId(GetBattlerPokeballItemId(battlerId));
2017-12-02 19:39:07 +01:00
LoadBallGfx(ballId);
ballSpriteId = CreateSprite(&gBallSpriteTemplates[ballId], 32, 80, 29);
gSprites[ballSpriteId].data[0] = 0x80;
gSprites[ballSpriteId].data[1] = 0;
gSprites[ballSpriteId].data[7] = throwCaseId;
switch (throwCaseId)
{
case POKEBALL_PLAYER_SENDOUT:
2018-03-01 00:59:52 +01:00
gBattlerTarget = battlerId;
2021-07-07 09:11:52 -04:00
gSprites[ballSpriteId].x = 24;
gSprites[ballSpriteId].y = 68;
2017-12-02 19:39:07 +01:00
gSprites[ballSpriteId].callback = SpriteCB_PlayerMonSendOut_1;
break;
case POKEBALL_OPPONENT_SENDOUT:
2021-07-07 09:11:52 -04:00
gSprites[ballSpriteId].x = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_X);
gSprites[ballSpriteId].y = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y) + 24;
2018-03-01 00:59:52 +01:00
gBattlerTarget = battlerId;
2017-12-02 20:38:26 +01:00
gSprites[ballSpriteId].data[0] = 0;
2017-12-02 19:39:07 +01:00
gSprites[ballSpriteId].callback = SpriteCB_OpponentMonSendOut;
break;
default:
2018-02-06 16:09:39 -06:00
gBattlerTarget = GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT);
2017-12-02 19:39:07 +01:00
notSendOut = TRUE;
break;
}
2018-03-01 00:59:52 +01:00
gSprites[ballSpriteId].sBattler = gBattlerTarget;
2017-12-02 19:39:07 +01:00
if (!notSendOut)
{
DestroyTask(taskId);
return;
}
// this will perform an unused ball throw animation
2017-12-02 20:38:26 +01:00
gSprites[ballSpriteId].data[0] = 0x22;
2018-10-06 23:04:53 +02:00
gSprites[ballSpriteId].data[2] = GetBattlerSpriteCoord(gBattlerTarget, BATTLER_COORD_X);
gSprites[ballSpriteId].data[4] = GetBattlerSpriteCoord(gBattlerTarget, BATTLER_COORD_Y) - 16;
2017-12-02 20:38:26 +01:00
gSprites[ballSpriteId].data[5] = -40;
2018-11-25 17:00:18 -06:00
InitAnimArcTranslation(&gSprites[ballSpriteId]);
2017-12-02 19:39:07 +01:00
gSprites[ballSpriteId].oam.affineParam = taskId;
2018-03-01 00:59:52 +01:00
gTasks[taskId].tOpponentBattler = gBattlerTarget;
2017-12-02 19:39:07 +01:00
gTasks[taskId].func = TaskDummy;
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_THROW);
2017-12-02 19:39:07 +01:00
}
// This sequence of functions is very similar to those that get run when
// a Pokéball gets thrown at a wild Pokémon, starting at SpriteCB_Ball_Arc.
// These do not seem to get run.
static void SpriteCB_BallThrow(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
2019-02-06 13:17:09 -06:00
if (TranslateAnimHorizontalArc(sprite))
2017-12-02 19:39:07 +01:00
{
u16 ballId;
u8 taskId = sprite->oam.affineParam;
2018-03-01 00:59:52 +01:00
u8 opponentBattler = gTasks[taskId].tOpponentBattler;
2017-12-02 19:39:07 +01:00
u8 noOfShakes = gTasks[taskId].tThrowId;
StartSpriteAnim(sprite, 1);
sprite->affineAnimPaused = 1;
2021-07-07 09:11:52 -04:00
sprite->x += sprite->x2;
sprite->y += sprite->y2;
sprite->x2 = 0;
sprite->y2 = 0;
2017-12-02 20:38:26 +01:00
sprite->data[5] = 0;
2018-03-01 00:59:52 +01:00
ballId = ItemIdToBallId(GetBattlerPokeballItemId(opponentBattler));
2021-07-07 09:11:52 -04:00
AnimateBallOpenParticles(sprite->x, sprite->y - 5, 1, 0x1C, ballId);
2018-03-01 00:59:52 +01:00
sprite->data[0] = LaunchBallFadeMonTask(FALSE, opponentBattler, 14, ballId);
sprite->sBattler = opponentBattler;
2017-12-02 19:39:07 +01:00
sprite->data[7] = noOfShakes;
DestroyTask(taskId);
sprite->callback = SpriteCB_BallThrow_ReachMon;
2017-12-02 19:39:07 +01:00
}
}
#undef tFrames
#undef tPan
#undef tThrowId
2018-03-01 00:59:52 +01:00
#undef tBattler
#undef tOpponentBattler
2017-12-02 19:39:07 +01:00
static void SpriteCB_BallThrow_ReachMon(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
sprite->callback = SpriteCB_BallThrow_StartShrinkMon;
2017-12-02 19:39:07 +01:00
}
static void SpriteCB_BallThrow_StartShrinkMon(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
2017-12-02 20:38:26 +01:00
if (++sprite->data[5] == 10)
2017-12-02 19:39:07 +01:00
{
2017-12-02 20:38:26 +01:00
sprite->data[5] = 0;
sprite->callback = SpriteCB_BallThrow_ShrinkMon;
StartSpriteAffineAnim(&gSprites[gBattlerSpriteIds[sprite->sBattler]], BATTLER_AFFINE_RETURN);
2018-03-01 00:59:52 +01:00
AnimateSprite(&gSprites[gBattlerSpriteIds[sprite->sBattler]]);
gSprites[gBattlerSpriteIds[sprite->sBattler]].data[1] = 0;
2017-12-02 19:39:07 +01:00
}
}
static void SpriteCB_BallThrow_ShrinkMon(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
sprite->data[5]++;
if (sprite->data[5] == 11)
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_TRADE);
2018-03-01 00:59:52 +01:00
if (gSprites[gBattlerSpriteIds[sprite->sBattler]].affineAnimEnded)
2017-12-02 19:39:07 +01:00
{
StartSpriteAnim(sprite, 2);
2018-03-01 00:59:52 +01:00
gSprites[gBattlerSpriteIds[sprite->sBattler]].invisible = TRUE;
2017-12-02 19:39:07 +01:00
sprite->data[5] = 0;
sprite->callback = SpriteCB_BallThrow_Close;
2017-12-02 19:39:07 +01:00
}
else
{
2018-03-01 00:59:52 +01:00
gSprites[gBattlerSpriteIds[sprite->sBattler]].data[1] += 0x60;
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[sprite->sBattler]].y2 = -gSprites[gBattlerSpriteIds[sprite->sBattler]].data[1] >> 8;
2017-12-02 19:39:07 +01:00
}
}
static void SpriteCB_BallThrow_Close(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
if (sprite->animEnded)
{
sprite->data[5]++;
if (sprite->data[5] == 1)
{
sprite->data[3] = 0;
sprite->data[4] = 32;
sprite->data[5] = 0;
2021-07-07 09:11:52 -04:00
sprite->y += Cos(0, 32);
sprite->y2 = -Cos(0, sprite->data[4]);
sprite->callback = SpriteCB_BallThrow_FallToGround;
2017-12-02 19:39:07 +01:00
}
}
}
static void SpriteCB_BallThrow_FallToGround(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
bool8 r5 = FALSE;
switch (sprite->data[3] & 0xFF)
{
case 0:
2021-07-07 09:11:52 -04:00
sprite->y2 = -Cos(sprite->data[5], sprite->data[4]);
2017-12-02 19:39:07 +01:00
sprite->data[5] += 4 + (sprite->data[3] >> 8);
if (sprite->data[5] >= 64)
{
sprite->data[4] -= 10;
sprite->data[3] += 0x101;
if (sprite->data[3] >> 8 == 4)
r5 = TRUE;
switch (sprite->data[3] >> 8)
{
case 1:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_BOUNCE_1);
2017-12-02 19:39:07 +01:00
break;
case 2:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_BOUNCE_2);
2017-12-02 19:39:07 +01:00
break;
case 3:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_BOUNCE_3);
2017-12-02 19:39:07 +01:00
break;
default:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_BOUNCE_4);
2017-12-02 19:39:07 +01:00
break;
}
}
break;
case 1:
2021-07-07 09:11:52 -04:00
sprite->y2 = -Cos(sprite->data[5], sprite->data[4]);
2017-12-02 19:39:07 +01:00
sprite->data[5] -= 4 + (sprite->data[3] >> 8);
if (sprite->data[5] <= 0)
{
sprite->data[5] = 0;
sprite->data[3] &= 0xFF00;
}
break;
}
if (r5)
{
sprite->data[3] = 0;
2021-07-07 09:11:52 -04:00
sprite->y += Cos(64, 32);
sprite->y2 = 0;
2017-12-02 19:39:07 +01:00
if (sprite->data[7] == 0)
{
sprite->callback = SpriteCB_ReleaseMonFromBall;
}
else
{
sprite->callback = SpriteCB_BallThrow_StartShakes;
2017-12-02 19:39:07 +01:00
sprite->data[4] = 1;
sprite->data[5] = 0;
}
}
}
static void SpriteCB_BallThrow_StartShakes(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
sprite->data[3]++;
if (sprite->data[3] == 31)
{
sprite->data[3] = 0;
sprite->affineAnimPaused = TRUE;
StartSpriteAffineAnim(sprite, 1);
sprite->callback = SpriteCB_BallThrow_Shake;
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL);
2017-12-02 19:39:07 +01:00
}
}
static void SpriteCB_BallThrow_Shake(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
switch (sprite->data[3] & 0xFF)
{
case 0:
case 2:
2021-07-07 09:11:52 -04:00
sprite->x2 += sprite->data[4];
2017-12-02 19:39:07 +01:00
sprite->data[5] += sprite->data[4];
sprite->affineAnimPaused = FALSE;
if (sprite->data[5] > 3 || sprite->data[5] < -3)
{
sprite->data[3]++;
sprite->data[5] = 0;
}
break;
case 1:
sprite->data[5]++;
if (sprite->data[5] == 1)
{
sprite->data[5] = 0;
sprite->data[4] = -sprite->data[4];
sprite->data[3]++;
sprite->affineAnimPaused = FALSE;
if (sprite->data[4] < 0)
ChangeSpriteAffineAnim(sprite, 2);
else
ChangeSpriteAffineAnim(sprite, 1);
}
else
{
sprite->affineAnimPaused = TRUE;
}
break;
case 3:
sprite->data[3] += 0x100;
if (sprite->data[3] >> 8 == sprite->data[7])
{
sprite->callback = SpriteCB_ReleaseMonFromBall;
}
else
{
if (sprite->data[7] == 4 && sprite->data[3] >> 8 == 3)
{
sprite->callback = SpriteCB_BallThrow_StartCaptureMon;
2017-12-02 19:39:07 +01:00
sprite->affineAnimPaused = TRUE;
}
else
{
sprite->data[3]++;
sprite->affineAnimPaused = TRUE;
}
}
break;
case 4:
default:
sprite->data[5]++;
if (sprite->data[5] == 31)
{
sprite->data[5] = 0;
sprite->data[3] &= 0xFF00;
StartSpriteAffineAnim(sprite, 3);
if (sprite->data[4] < 0)
StartSpriteAffineAnim(sprite, 2);
else
StartSpriteAffineAnim(sprite, 1);
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL);
2017-12-02 19:39:07 +01:00
}
break;
}
}
#define tCryTaskSpecies data[0]
#define tCryTaskPan data[1]
#define tCryTaskWantedCry data[2]
2018-03-01 00:59:52 +01:00
#define tCryTaskBattler data[3]
2017-12-02 19:39:07 +01:00
#define tCryTaskMonSpriteId data[4]
#define tCryTaskMonPtr1 data[5]
#define tCryTaskMonPtr2 data[6]
#define tCryTaskFrames data[10]
#define tCryTaskState data[15]
2017-12-02 21:27:00 +01:00
static void Task_PlayCryWhenReleasedFromBall(u8 taskId)
2017-12-02 19:39:07 +01:00
{
u8 wantedCry = gTasks[taskId].tCryTaskWantedCry;
s8 pan = gTasks[taskId].tCryTaskPan;
u16 species = gTasks[taskId].tCryTaskSpecies;
2018-03-01 00:59:52 +01:00
u8 battlerId = gTasks[taskId].tCryTaskBattler;
2017-12-02 19:39:07 +01:00
u8 monSpriteId = gTasks[taskId].tCryTaskMonSpriteId;
2021-01-22 20:03:21 -05:00
struct Pokemon *mon = (void*)(u32)((gTasks[taskId].tCryTaskMonPtr1 << 16) | (u16)(gTasks[taskId].tCryTaskMonPtr2));
2017-12-02 19:39:07 +01:00
switch (gTasks[taskId].tCryTaskState)
{
case 0:
default:
if (gSprites[monSpriteId].affineAnimEnded)
gTasks[taskId].tCryTaskState = wantedCry + 1;
break;
case 1:
2018-06-19 00:43:15 +02:00
if (ShouldPlayNormalMonCry(mon) == TRUE)
2017-12-02 19:39:07 +01:00
PlayCry3(species, pan, 0);
else
PlayCry3(species, pan, 11);
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[battlerId].waitForCry = FALSE;
2017-12-02 19:39:07 +01:00
DestroyTask(taskId);
break;
case 2:
StopCryAndClearCrySongs();
gTasks[taskId].tCryTaskFrames = 3;
gTasks[taskId].tCryTaskState = 20;
break;
case 20:
if (gTasks[taskId].tCryTaskFrames == 0)
{
2018-06-19 00:43:15 +02:00
if (ShouldPlayNormalMonCry(mon) == TRUE)
2017-12-02 19:39:07 +01:00
PlayCry4(species, pan, 1);
else
PlayCry4(species, pan, 12);
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[battlerId].waitForCry = FALSE;
2017-12-02 19:39:07 +01:00
DestroyTask(taskId);
}
else
{
gTasks[taskId].tCryTaskFrames--;
}
break;
case 3:
gTasks[taskId].tCryTaskFrames = 6;
gTasks[taskId].tCryTaskState = 30;
break;
case 30:
if (gTasks[taskId].tCryTaskFrames != 0)
{
gTasks[taskId].tCryTaskFrames--;
break;
}
gTasks[taskId].tCryTaskState++;
// fall through
case 31:
if (!IsCryPlayingOrClearCrySongs())
{
StopCryAndClearCrySongs();
gTasks[taskId].tCryTaskFrames = 3;
gTasks[taskId].tCryTaskState++;
}
break;
case 32:
if (gTasks[taskId].tCryTaskFrames != 0)
{
gTasks[taskId].tCryTaskFrames--;
break;
}
2018-06-19 00:43:15 +02:00
if (ShouldPlayNormalMonCry(mon) == TRUE)
2017-12-02 19:39:07 +01:00
PlayCry4(species, pan, 0);
else
PlayCry4(species, pan, 11);
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[battlerId].waitForCry = FALSE;
2017-12-02 19:39:07 +01:00
DestroyTask(taskId);
break;
}
}
2017-12-02 21:27:00 +01:00
static void SpriteCB_ReleaseMonFromBall(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
2018-03-01 00:59:52 +01:00
u8 battlerId = sprite->sBattler;
2017-12-02 19:39:07 +01:00
u32 ballId;
StartSpriteAnim(sprite, 1);
2018-03-01 00:59:52 +01:00
ballId = ItemIdToBallId(GetBattlerPokeballItemId(battlerId));
2021-07-07 09:11:52 -04:00
AnimateBallOpenParticles(sprite->x, sprite->y - 5, 1, 0x1C, ballId);
sprite->data[0] = LaunchBallFadeMonTask(TRUE, sprite->sBattler, 14, ballId);
2017-12-02 19:39:07 +01:00
sprite->callback = HandleBallAnimEnd;
if (gMain.inBattle)
{
2019-08-08 13:06:55 +02:00
struct Pokemon *mon, *illusionMon;
2017-12-02 19:39:07 +01:00
s8 pan;
u16 wantedCryCase;
u8 taskId;
2018-03-01 00:59:52 +01:00
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
2017-12-02 19:39:07 +01:00
{
2018-03-01 00:59:52 +01:00
mon = &gEnemyParty[gBattlerPartyIndexes[battlerId]];
2017-12-02 19:39:07 +01:00
pan = 25;
}
else
{
2018-03-01 00:59:52 +01:00
mon = &gPlayerParty[gBattlerPartyIndexes[battlerId]];
2017-12-02 19:39:07 +01:00
pan = -25;
}
2018-03-01 00:59:52 +01:00
if ((battlerId == GetBattlerAtPosition(B_POSITION_PLAYER_LEFT) || battlerId == GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT))
2021-01-22 20:03:21 -05:00
&& IsDoubleBattle() && gBattleSpritesDataPtr->animationData->introAnimActive)
2017-12-02 19:39:07 +01:00
{
if (gBattleTypeFlags & BATTLE_TYPE_MULTI && gBattleTypeFlags & BATTLE_TYPE_LINK)
{
if (IsBGMPlaying())
m4aMPlayStop(&gMPlayInfo_BGM);
2017-12-02 19:39:07 +01:00
}
else
{
m4aMPlayVolumeControl(&gMPlayInfo_BGM, 0xFFFF, 128);
2017-12-02 19:39:07 +01:00
}
}
2021-01-22 20:03:21 -05:00
if (!IsDoubleBattle() || !gBattleSpritesDataPtr->animationData->introAnimActive)
2017-12-02 19:39:07 +01:00
wantedCryCase = 0;
2018-03-01 00:59:52 +01:00
else if (battlerId == GetBattlerAtPosition(B_POSITION_PLAYER_LEFT) || battlerId == GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT))
2017-12-02 19:39:07 +01:00
wantedCryCase = 1;
else
wantedCryCase = 2;
2021-01-22 20:03:21 -05:00
gBattleSpritesDataPtr->healthBoxesData[battlerId].waitForCry = TRUE;
2017-12-02 19:39:07 +01:00
taskId = CreateTask(Task_PlayCryWhenReleasedFromBall, 3);
2019-08-08 13:06:55 +02:00
illusionMon = GetIllusionMonPtr(battlerId);
if (illusionMon != NULL)
gTasks[taskId].tCryTaskSpecies = GetMonData(illusionMon, MON_DATA_SPECIES);
else
gTasks[taskId].tCryTaskSpecies = GetMonData(mon, MON_DATA_SPECIES);
2017-12-02 19:39:07 +01:00
gTasks[taskId].tCryTaskPan = pan;
gTasks[taskId].tCryTaskWantedCry = wantedCryCase;
2018-03-01 00:59:52 +01:00
gTasks[taskId].tCryTaskBattler = battlerId;
gTasks[taskId].tCryTaskMonSpriteId = gBattlerSpriteIds[sprite->sBattler];
2021-04-09 12:41:02 -04:00
gTasks[taskId].tCryTaskMonPtr1 = (u32)(mon) >> 16;
2017-12-02 19:39:07 +01:00
gTasks[taskId].tCryTaskMonPtr2 = (u32)(mon);
gTasks[taskId].tCryTaskState = 0;
}
StartSpriteAffineAnim(&gSprites[gBattlerSpriteIds[sprite->sBattler]], BATTLER_AFFINE_EMERGE);
2017-12-02 19:39:07 +01:00
2018-03-01 00:59:52 +01:00
if (GetBattlerSide(sprite->sBattler) == B_SIDE_OPPONENT)
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[sprite->sBattler]].callback = SpriteCb_OpponentMonFromBall;
2017-12-02 19:39:07 +01:00
else
2020-07-14 11:13:03 +02:00
gSprites[gBattlerSpriteIds[sprite->sBattler]].callback = SpriteCb_PlayerMonFromBall;
2017-12-02 19:39:07 +01:00
2018-03-01 00:59:52 +01:00
AnimateSprite(&gSprites[gBattlerSpriteIds[sprite->sBattler]]);
gSprites[gBattlerSpriteIds[sprite->sBattler]].data[1] = 0x1000;
2017-12-02 19:39:07 +01:00
}
#undef tCryTaskSpecies
#undef tCryTaskPan
#undef tCryTaskWantedCry
2018-03-01 00:59:52 +01:00
#undef tCryTaskBattler
2017-12-02 19:39:07 +01:00
#undef tCryTaskMonSpriteId
#undef tCryTaskMonPtr1
#undef tCryTaskMonPtr2
#undef tCryTaskFrames
#undef tCryTaskState
static void SpriteCB_BallThrow_StartCaptureMon(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
sprite->animPaused = TRUE;
sprite->callback = SpriteCB_BallThrow_CaptureMon;
2017-12-02 19:39:07 +01:00
sprite->data[3] = 0;
sprite->data[4] = 0;
sprite->data[5] = 0;
}
2017-12-02 21:27:00 +01:00
static void HandleBallAnimEnd(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
bool8 affineAnimEnded = FALSE;
2018-03-01 00:59:52 +01:00
u8 battlerId = sprite->sBattler;
2017-12-02 19:39:07 +01:00
2018-03-01 00:59:52 +01:00
gSprites[gBattlerSpriteIds[battlerId]].invisible = FALSE;
2017-12-02 19:39:07 +01:00
if (sprite->animEnded)
sprite->invisible = TRUE;
2018-03-01 00:59:52 +01:00
if (gSprites[gBattlerSpriteIds[battlerId]].affineAnimEnded)
2017-12-02 19:39:07 +01:00
{
StartSpriteAffineAnim(&gSprites[gBattlerSpriteIds[battlerId]], BATTLER_AFFINE_NORMAL);
2017-12-02 19:39:07 +01:00
affineAnimEnded = TRUE;
}
else
{
2018-03-01 00:59:52 +01:00
gSprites[gBattlerSpriteIds[battlerId]].data[1] -= 288;
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[battlerId]].y2 = gSprites[gBattlerSpriteIds[battlerId]].data[1] >> 8;
2017-12-02 19:39:07 +01:00
}
if (sprite->animEnded && affineAnimEnded)
{
2018-03-01 00:59:52 +01:00
s32 i, doneBattlers;
2017-12-02 19:39:07 +01:00
2021-07-07 09:11:52 -04:00
gSprites[gBattlerSpriteIds[battlerId]].y2 = 0;
2017-12-02 19:39:07 +01:00
gDoingBattleAnim = FALSE;
2018-03-01 00:59:52 +01:00
gBattleSpritesDataPtr->healthBoxesData[battlerId].ballAnimActive = 0;
2017-12-02 19:39:07 +01:00
FreeSpriteOamMatrix(sprite);
DestroySprite(sprite);
2018-03-01 00:59:52 +01:00
for (doneBattlers = 0, i = 0; i < MAX_BATTLERS_COUNT; i++)
2017-12-02 19:39:07 +01:00
{
if (gBattleSpritesDataPtr->healthBoxesData[i].ballAnimActive == 0)
2018-03-01 00:59:52 +01:00
doneBattlers++;
2017-12-02 19:39:07 +01:00
}
2018-03-01 00:59:52 +01:00
if (doneBattlers == MAX_BATTLERS_COUNT)
2017-12-02 19:39:07 +01:00
{
for (i = 0; i < POKEBALL_COUNT; i++)
FreeBallGfx(i);
}
}
}
static void SpriteCB_BallThrow_CaptureMon(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
2018-03-01 00:59:52 +01:00
u8 battlerId = sprite->sBattler;
2017-12-02 19:39:07 +01:00
sprite->data[4]++;
if (sprite->data[4] == 40)
{
return;
}
else if (sprite->data[4] == 95)
{
gDoingBattleAnim = FALSE;
m4aMPlayAllStop();
2020-08-20 18:02:00 -04:00
PlaySE(MUS_EVOLVED);
2017-12-02 19:39:07 +01:00
}
else if (sprite->data[4] == 315)
{
2018-03-01 00:59:52 +01:00
FreeOamMatrix(gSprites[gBattlerSpriteIds[sprite->sBattler]].oam.matrixNum);
DestroySprite(&gSprites[gBattlerSpriteIds[sprite->sBattler]]);
2017-12-02 19:39:07 +01:00
DestroySpriteAndFreeResources(sprite);
if (gMain.inBattle)
2018-03-01 00:59:52 +01:00
gBattleSpritesDataPtr->healthBoxesData[battlerId].ballAnimActive = 0;
2017-12-02 19:39:07 +01:00
}
}
2017-12-02 21:27:00 +01:00
static void SpriteCB_PlayerMonSendOut_1(struct Sprite *sprite)
2017-12-02 19:39:07 +01:00
{
sprite->data[0] = 25;
2018-03-01 00:59:52 +01:00
sprite->data[2] = GetBattlerSpriteCoord(sprite->sBattler, 2);
sprite->data[4] = GetBattlerSpriteCoord(sprite->sBattler, 3) + 24;
2017-12-02 19:39:07 +01:00
sprite->data[5] = -30;
2018-03-01 00:59:52 +01:00
sprite->oam.affineParam = sprite->sBattler;
2018-11-25 17:00:18 -06:00
InitAnimArcTranslation(sprite);
2017-12-02 19:39:07 +01:00
sprite->callback = SpriteCB_PlayerMonSendOut_2;
}
2017-12-02 20:38:26 +01:00
#define HIBYTE(x) (((x) >> 8) & 0xFF)
2017-12-02 21:27:00 +01:00
static void SpriteCB_PlayerMonSendOut_2(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
u32 r6;
u32 r7;
if (HIBYTE(sprite->data[7]) >= 35 && HIBYTE(sprite->data[7]) < 80)
{
s16 r4;
if ((sprite->oam.affineParam & 0xFF00) == 0)
{
r6 = sprite->data[1] & 1;
r7 = sprite->data[2] & 1;
sprite->data[1] = ((sprite->data[1] / 3) & ~1) | r6;
sprite->data[2] = ((sprite->data[2] / 3) & ~1) | r7;
StartSpriteAffineAnim(sprite, 4);
}
r4 = sprite->data[0];
2018-12-19 21:13:26 -06:00
AnimTranslateLinear(sprite);
2018-03-01 00:59:52 +01:00
sprite->data[7] += sprite->sBattler / 3;
2021-07-07 09:11:52 -04:00
sprite->y2 += Sin(HIBYTE(sprite->data[7]), sprite->data[5]);
2017-12-02 20:38:26 +01:00
sprite->oam.affineParam += 0x100;
if ((sprite->oam.affineParam >> 8) % 3 != 0)
sprite->data[0] = r4;
else
sprite->data[0] = r4 - 1;
if (HIBYTE(sprite->data[7]) >= 80)
{
r6 = sprite->data[1] & 1;
r7 = sprite->data[2] & 1;
sprite->data[1] = ((sprite->data[1] * 3) & ~1) | r6;
sprite->data[2] = ((sprite->data[2] * 3) & ~1) | r7;
}
}
else
{
2019-02-06 13:17:09 -06:00
if (TranslateAnimHorizontalArc(sprite))
2017-12-02 20:38:26 +01:00
{
2021-07-07 09:11:52 -04:00
sprite->x += sprite->x2;
sprite->y += sprite->y2;
sprite->y2 = 0;
sprite->x2 = 0;
2018-03-01 00:59:52 +01:00
sprite->sBattler = sprite->oam.affineParam & 0xFF;
2017-12-02 20:38:26 +01:00
sprite->data[0] = 0;
2021-01-22 20:03:21 -05:00
if (IsDoubleBattle() && gBattleSpritesDataPtr->animationData->introAnimActive
2018-03-01 00:59:52 +01:00
&& sprite->sBattler == GetBattlerAtPosition(B_POSITION_PLAYER_RIGHT))
2017-12-02 20:38:26 +01:00
sprite->callback = SpriteCB_ReleaseMon2FromBall;
else
sprite->callback = SpriteCB_ReleaseMonFromBall;
StartSpriteAffineAnim(sprite, 0);
}
}
}
2017-12-02 21:27:00 +01:00
static void SpriteCB_ReleaseMon2FromBall(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
if (sprite->data[0]++ > 24)
{
sprite->data[0] = 0;
sprite->callback = SpriteCB_ReleaseMonFromBall;
}
}
2017-12-02 21:27:00 +01:00
static void SpriteCB_OpponentMonSendOut(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
sprite->data[0]++;
if (sprite->data[0] > 15)
{
sprite->data[0] = 0;
2021-01-22 20:03:21 -05:00
if (IsDoubleBattle() && gBattleSpritesDataPtr->animationData->introAnimActive
2018-03-01 00:59:52 +01:00
&& sprite->sBattler == GetBattlerAtPosition(B_POSITION_OPPONENT_RIGHT))
2017-12-02 20:38:26 +01:00
sprite->callback = SpriteCB_ReleaseMon2FromBall;
else
sprite->callback = SpriteCB_ReleaseMonFromBall;
}
}
2018-03-01 00:59:52 +01:00
#undef sBattler
2017-12-02 20:38:26 +01:00
2018-12-13 21:33:54 -06:00
static u8 AnimateBallOpenParticlesForPokeball(u8 x, u8 y, u8 kindOfStars, u8 d)
2017-12-02 20:38:26 +01:00
{
2018-12-13 21:33:54 -06:00
return AnimateBallOpenParticles(x, y, kindOfStars, d, BALL_POKE);
2017-12-02 20:38:26 +01:00
}
2018-03-01 00:59:52 +01:00
static u8 LaunchBallFadeMonTaskForPokeball(bool8 unFadeLater, u8 battlerId, u32 arg2)
2017-12-02 20:38:26 +01:00
{
2018-03-01 00:59:52 +01:00
return LaunchBallFadeMonTask(unFadeLater, battlerId, arg2, BALL_POKE);
2017-12-02 20:38:26 +01:00
}
2021-01-22 23:22:37 -05:00
// Pokeball in Birch intro, and when receiving via trade
2018-03-01 00:59:52 +01:00
void CreatePokeballSpriteToReleaseMon(u8 monSpriteId, u8 battlerId, u8 x, u8 y, u8 oamPriority, u8 subpriortiy, u8 g, u32 h, u16 species)
2017-12-02 20:38:26 +01:00
{
u8 spriteId;
2018-12-17 23:00:08 +01:00
LoadCompressedSpriteSheetUsingHeap(&gBallSpriteSheets[0]);
LoadCompressedSpritePaletteUsingHeap(&gBallSpritePalettes[0]);
2017-12-02 20:38:26 +01:00
spriteId = CreateSprite(&gBallSpriteTemplates[0], x, y, subpriortiy);
gSprites[spriteId].data[0] = monSpriteId;
2021-07-07 09:11:52 -04:00
gSprites[spriteId].data[5] = gSprites[monSpriteId].x;
gSprites[spriteId].data[6] = gSprites[monSpriteId].y;
2017-12-02 20:38:26 +01:00
2021-07-07 09:11:52 -04:00
gSprites[monSpriteId].x = x;
gSprites[monSpriteId].y = y;
2017-12-02 20:38:26 +01:00
gSprites[monSpriteId].data[7] = species;
gSprites[spriteId].data[1] = g;
2018-03-01 00:59:52 +01:00
gSprites[spriteId].data[2] = battlerId;
2017-12-02 20:38:26 +01:00
gSprites[spriteId].data[3] = h;
gSprites[spriteId].data[4] = h >> 0x10;
gSprites[spriteId].oam.priority = oamPriority;
2021-01-22 23:22:37 -05:00
gSprites[spriteId].callback = SpriteCB_PokeballReleaseMon;
2017-12-02 20:38:26 +01:00
gSprites[monSpriteId].invisible = TRUE;
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_PokeballReleaseMon(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
if (sprite->data[1] == 0)
{
u8 r5;
u8 r7 = sprite->data[0];
2018-03-01 00:59:52 +01:00
u8 battlerId = sprite->data[2];
2017-12-02 20:38:26 +01:00
u32 r4 = (u16)sprite->data[3] | ((u16)sprite->data[4] << 16);
if (sprite->subpriority != 0)
r5 = sprite->subpriority - 1;
else
r5 = 0;
StartSpriteAnim(sprite, 1);
2021-07-07 09:11:52 -04:00
AnimateBallOpenParticlesForPokeball(sprite->x, sprite->y - 5, sprite->oam.priority, r5);
2018-03-01 00:59:52 +01:00
sprite->data[1] = LaunchBallFadeMonTaskForPokeball(1, battlerId, r4);
2021-01-22 23:22:37 -05:00
sprite->callback = SpriteCB_ReleasedMonFlyOut;
2017-12-02 20:38:26 +01:00
gSprites[r7].invisible = FALSE;
StartSpriteAffineAnim(&gSprites[r7], BATTLER_AFFINE_EMERGE);
2017-12-02 20:38:26 +01:00
AnimateSprite(&gSprites[r7]);
gSprites[r7].data[1] = 0x1000;
sprite->data[7] = 0;
}
else
{
sprite->data[1]--;
}
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_ReleasedMonFlyOut(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
bool8 r12 = FALSE;
bool8 r6 = FALSE;
u8 monSpriteId = sprite->data[0];
u16 var1;
u16 var2;
if (sprite->animEnded)
sprite->invisible = TRUE;
if (gSprites[monSpriteId].affineAnimEnded)
{
StartSpriteAffineAnim(&gSprites[monSpriteId], BATTLER_AFFINE_NORMAL);
2017-12-02 20:38:26 +01:00
r12 = TRUE;
}
2021-07-07 09:11:52 -04:00
var1 = (sprite->data[5] - sprite->x) * sprite->data[7] / 128 + sprite->x;
var2 = (sprite->data[6] - sprite->y) * sprite->data[7] / 128 + sprite->y;
gSprites[monSpriteId].x = var1;
gSprites[monSpriteId].y = var2;
2017-12-02 20:38:26 +01:00
if (sprite->data[7] < 128)
{
s16 sine = -(gSineTable[(u8)sprite->data[7]] / 8);
sprite->data[7] += 4;
2021-07-07 09:11:52 -04:00
gSprites[monSpriteId].x2 = sine;
gSprites[monSpriteId].y2 = sine;
2017-12-02 20:38:26 +01:00
}
else
{
2021-07-07 09:11:52 -04:00
gSprites[monSpriteId].x = sprite->data[5];
gSprites[monSpriteId].y = sprite->data[6];
gSprites[monSpriteId].x2 = 0;
gSprites[monSpriteId].y2 = 0;
2017-12-02 20:38:26 +01:00
r6 = TRUE;
}
if (sprite->animEnded && r12 && r6)
{
if (gSprites[monSpriteId].data[7] == SPECIES_EGG)
DoMonFrontSpriteAnimation(&gSprites[monSpriteId], gSprites[monSpriteId].data[7], TRUE, 0);
else
DoMonFrontSpriteAnimation(&gSprites[monSpriteId], gSprites[monSpriteId].data[7], FALSE, 0);
DestroySpriteAndFreeResources(sprite);
}
}
2019-10-09 05:56:44 -04:00
u8 CreateTradePokeballSprite(u8 a, u8 b, u8 x, u8 y, u8 oamPriority, u8 subPriority, u8 g, u32 h)
2017-12-02 20:38:26 +01:00
{
u8 spriteId;
2018-12-17 23:00:08 +01:00
LoadCompressedSpriteSheetUsingHeap(&gBallSpriteSheets[0]);
LoadCompressedSpritePaletteUsingHeap(&gBallSpritePalettes[0]);
2017-12-02 20:38:26 +01:00
spriteId = CreateSprite(&gBallSpriteTemplates[0], x, y, subPriority);
gSprites[spriteId].data[0] = a;
gSprites[spriteId].data[1] = g;
gSprites[spriteId].data[2] = b;
gSprites[spriteId].data[3] = h;
gSprites[spriteId].data[4] = h >> 16;
gSprites[spriteId].oam.priority = oamPriority;
2021-01-22 23:22:37 -05:00
gSprites[spriteId].callback = SpriteCB_TradePokeball;
2017-12-02 20:38:26 +01:00
return spriteId;
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_TradePokeball(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
if (sprite->data[1] == 0)
{
u8 r6;
2021-01-22 23:22:37 -05:00
u8 monSpriteId = sprite->data[0];
2017-12-02 20:38:26 +01:00
u8 r8 = sprite->data[2];
u32 r5 = (u16)sprite->data[3] | ((u16)sprite->data[4] << 16);
if (sprite->subpriority != 0)
r6 = sprite->subpriority - 1;
else
r6 = 0;
StartSpriteAnim(sprite, 1);
2021-07-07 09:11:52 -04:00
AnimateBallOpenParticlesForPokeball(sprite->x, sprite->y - 5, sprite->oam.priority, r6);
2017-12-02 20:38:26 +01:00
sprite->data[1] = LaunchBallFadeMonTaskForPokeball(1, r8, r5);
2021-01-22 23:22:37 -05:00
sprite->callback = SpriteCB_TradePokeballSendOff;
#ifdef BUGFIX
// FIX: If this is used on a sprite that has previously had an affine animation, it will not
// play the shrink anim properly due to being paused. Works together with the fix to `sub_817F77C`.
2021-01-22 23:22:37 -05:00
gSprites[monSpriteId].affineAnimPaused = FALSE;
#endif // BUGFIX
StartSpriteAffineAnim(&gSprites[monSpriteId], BATTLER_AFFINE_RETURN);
2021-01-22 23:22:37 -05:00
AnimateSprite(&gSprites[monSpriteId]);
gSprites[monSpriteId].data[1] = 0;
2017-12-02 20:38:26 +01:00
}
else
{
sprite->data[1]--;
}
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_TradePokeballSendOff(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
2021-01-22 23:22:37 -05:00
u8 monSpriteId;
2017-12-02 20:38:26 +01:00
sprite->data[5]++;
if (sprite->data[5] == 11)
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_TRADE);
2021-01-22 23:22:37 -05:00
monSpriteId = sprite->data[0];
if (gSprites[monSpriteId].affineAnimEnded)
2017-12-02 20:38:26 +01:00
{
StartSpriteAnim(sprite, 2);
2021-01-22 23:22:37 -05:00
gSprites[monSpriteId].invisible = TRUE;
2017-12-02 20:38:26 +01:00
sprite->data[5] = 0;
2021-01-22 23:22:37 -05:00
sprite->callback = SpriteCB_TradePokeballEnd;
2017-12-02 20:38:26 +01:00
}
else
{
2021-01-22 23:22:37 -05:00
gSprites[monSpriteId].data[1] += 96;
2021-07-07 09:11:52 -04:00
gSprites[monSpriteId].y2 = -gSprites[monSpriteId].data[1] >> 8;
2017-12-02 20:38:26 +01:00
}
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_TradePokeballEnd(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
if (sprite->animEnded)
sprite->callback = SpriteCallbackDummy;
}
2018-12-30 10:58:42 -06:00
static void Unref_DestroySpriteAndFreeResources(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
DestroySpriteAndFreeResources(sprite);
}
2021-01-22 23:22:37 -05:00
#define sSpeedX data[0]
#define sSpeedY data[1]
#define sDelayTimer data[1]
2021-01-22 20:03:21 -05:00
void StartHealthboxSlideIn(u8 battlerId)
2017-12-02 20:38:26 +01:00
{
2018-03-01 00:59:52 +01:00
struct Sprite *healthboxSprite = &gSprites[gHealthboxSpriteIds[battlerId]];
2017-12-02 20:38:26 +01:00
2021-01-22 23:22:37 -05:00
healthboxSprite->sSpeedX = 5;
healthboxSprite->sSpeedY = 0;
2021-07-07 09:11:52 -04:00
healthboxSprite->x2 = 0x73;
healthboxSprite->y2 = 0;
2021-01-22 23:22:37 -05:00
healthboxSprite->callback = SpriteCB_HealthboxSlideIn;
2018-03-01 00:59:52 +01:00
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
2017-12-02 20:38:26 +01:00
{
2021-01-22 23:22:37 -05:00
healthboxSprite->sSpeedX = -healthboxSprite->sSpeedX;
healthboxSprite->sSpeedY = -healthboxSprite->sSpeedY;
2021-07-07 09:11:52 -04:00
healthboxSprite->x2 = -healthboxSprite->x2;
healthboxSprite->y2 = -healthboxSprite->y2;
2017-12-02 20:38:26 +01:00
}
gSprites[healthboxSprite->data[5]].callback(&gSprites[healthboxSprite->data[5]]);
2018-03-01 00:59:52 +01:00
if (GetBattlerPosition(battlerId) == B_POSITION_PLAYER_RIGHT)
2021-01-22 23:22:37 -05:00
healthboxSprite->callback = SpriteCB_HealthboxSlideInDelayed;
2017-12-02 20:38:26 +01:00
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_HealthboxSlideInDelayed(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
2021-01-22 23:22:37 -05:00
sprite->sDelayTimer++;
if (sprite->sDelayTimer == 20)
2017-12-02 20:38:26 +01:00
{
2021-01-22 23:22:37 -05:00
sprite->sDelayTimer = 0;
sprite->callback = SpriteCB_HealthboxSlideIn;
2017-12-02 20:38:26 +01:00
}
}
2021-01-22 23:22:37 -05:00
static void SpriteCB_HealthboxSlideIn(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
2021-07-07 09:11:52 -04:00
sprite->x2 -= sprite->sSpeedX;
sprite->y2 -= sprite->sSpeedY;
if (sprite->x2 == 0 && sprite->y2 == 0)
2017-12-02 20:38:26 +01:00
sprite->callback = SpriteCallbackDummy;
}
2021-01-22 23:22:37 -05:00
#undef sSpeedX
#undef sSpeedY
#undef sDelayTimer
2018-03-01 00:59:52 +01:00
void DoHitAnimHealthboxEffect(u8 battlerId)
2017-12-02 20:38:26 +01:00
{
u8 spriteId;
spriteId = CreateInvisibleSpriteWithCallback(SpriteCB_HitAnimHealthoxEffect);
gSprites[spriteId].data[0] = 1;
2018-03-01 00:59:52 +01:00
gSprites[spriteId].data[1] = gHealthboxSpriteIds[battlerId];
2017-12-02 20:38:26 +01:00
gSprites[spriteId].callback = SpriteCB_HitAnimHealthoxEffect;
}
2017-12-02 21:27:00 +01:00
static void SpriteCB_HitAnimHealthoxEffect(struct Sprite *sprite)
2017-12-02 20:38:26 +01:00
{
u8 r1 = sprite->data[1];
2021-07-07 09:11:52 -04:00
gSprites[r1].y2 = sprite->data[0];
2017-12-02 20:38:26 +01:00
sprite->data[0] = -sprite->data[0];
sprite->data[2]++;
if (sprite->data[2] == 21)
{
2021-07-07 09:11:52 -04:00
gSprites[r1].x2 = 0;
gSprites[r1].y2 = 0;
2017-12-02 20:38:26 +01:00
DestroySprite(sprite);
}
}
void LoadBallGfx(u8 ballId)
{
u16 var;
2017-12-02 21:27:00 +01:00
if (GetSpriteTileStartByTag(gBallSpriteSheets[ballId].tag) == 0xFFFF)
2017-12-02 20:38:26 +01:00
{
2018-12-17 23:00:08 +01:00
LoadCompressedSpriteSheetUsingHeap(&gBallSpriteSheets[ballId]);
LoadCompressedSpritePaletteUsingHeap(&gBallSpritePalettes[ballId]);
2017-12-02 20:38:26 +01:00
}
switch (ballId)
{
case BALL_DIVE:
case BALL_LUXURY:
case BALL_PREMIER:
break;
default:
2017-12-02 21:27:00 +01:00
var = GetSpriteTileStartByTag(gBallSpriteSheets[ballId].tag);
2021-03-30 15:07:04 -04:00
LZDecompressVram(gOpenPokeballGfx, (void *)(OBJ_VRAM0 + 0x100 + var * 32));
2017-12-02 20:38:26 +01:00
break;
}
}
void FreeBallGfx(u8 ballId)
{
2017-12-02 21:27:00 +01:00
FreeSpriteTilesByTag(gBallSpriteSheets[ballId].tag);
FreeSpritePaletteByTag(gBallSpritePalettes[ballId].tag);
2017-12-02 20:38:26 +01:00
}
2018-03-01 00:59:52 +01:00
static u16 GetBattlerPokeballItemId(u8 battlerId)
2017-12-02 20:38:26 +01:00
{
2019-08-08 13:06:55 +02:00
struct Pokemon *mon, *illusionMon;
2018-03-01 00:59:52 +01:00
if (GetBattlerSide(battlerId) == B_SIDE_PLAYER)
2019-08-08 13:06:55 +02:00
mon = &gPlayerParty[gBattlerPartyIndexes[battlerId]];
2017-12-02 20:38:26 +01:00
else
2019-08-08 13:06:55 +02:00
mon = &gEnemyParty[gBattlerPartyIndexes[battlerId]];
illusionMon = GetIllusionMonPtr(battlerId);
if (illusionMon != NULL)
mon = illusionMon;
return GetMonData(mon, MON_DATA_POKEBALL);
2017-12-02 20:38:26 +01:00
}