pokeemerald/src/battle_anim_mons.c

2509 lines
72 KiB
C
Raw Normal View History

2018-10-06 23:04:53 +02:00
#include "global.h"
#include "battle.h"
#include "battle_anim.h"
2018-11-14 00:01:50 +00:00
#include "bg.h"
2018-10-06 23:04:53 +02:00
#include "contest.h"
2019-04-04 23:53:06 +02:00
#include "data.h"
2018-10-06 23:04:53 +02:00
#include "decompress.h"
2018-11-14 00:01:50 +00:00
#include "dma3.h"
#include "gpu_regs.h"
#include "malloc.h"
2018-10-06 23:04:53 +02:00
#include "palette.h"
#include "pokemon_icon.h"
#include "sprite.h"
#include "task.h"
#include "trig.h"
#include "util.h"
2018-11-14 00:01:50 +00:00
#include "constants/battle_anim.h"
2018-10-06 23:04:53 +02:00
#define IS_DOUBLE_BATTLE() ((gBattleTypeFlags & BATTLE_TYPE_DOUBLE))
2019-10-15 01:26:47 +02:00
extern const struct OamData gOamData_AffineNormal_ObjNormal_64x64;
2018-10-06 23:04:53 +02:00
2019-02-06 13:17:09 -06:00
static void sub_80A6FB4(struct Sprite *sprite);
2021-01-23 00:40:46 -05:00
static void AnimFastTranslateLinearWaitEnd(struct Sprite *sprite);
2019-10-20 10:47:56 +02:00
static void AnimThrowProjectile_Step(struct Sprite *sprite);
2019-02-06 13:17:09 -06:00
static void sub_80A8DFC(struct Sprite *sprite);
2021-01-23 00:40:46 -05:00
static void AnimWeatherBallUp_Step(struct Sprite *sprite);
2019-02-06 13:17:09 -06:00
static u16 GetBattlerYDeltaFromSpriteId(u8 spriteId);
2020-02-20 00:04:42 -05:00
static void AnimTask_BlendPalInAndOutSetup(struct Task *task);
2021-01-23 00:40:46 -05:00
static void AnimTask_AlphaFadeIn_Step(u8 taskId);
static void AnimTask_AttackerPunchWithTrace_Step(u8 taskId);
static void AnimTask_BlendMonInAndOut_Step(u8 taskId);
2019-02-06 13:17:09 -06:00
static bool8 sub_80A7238(void);
static void sub_80A8D78(struct Task *task, u8 taskId);
2018-10-07 12:32:20 +02:00
// EWRAM vars
2018-12-17 22:08:08 -06:00
EWRAM_DATA static union AffineAnimCmd *gAnimTaskAffineAnim = NULL;
2018-10-06 23:04:53 +02:00
// Const rom data
2018-10-07 12:32:20 +02:00
static const struct UCoords8 sBattlerCoords[][4] =
2018-10-06 23:04:53 +02:00
{
{
{ 72, 80 },
{ 176, 40 },
{ 48, 40 },
{ 112, 80 },
},
{
{ 32, 80 },
{ 200, 40 },
{ 90, 88 },
{ 152, 32 },
},
};
// One entry for each of the four Castform forms.
const struct MonCoords gCastformFrontSpriteCoords[] =
{
{ .size = 0x44, .y_offset = 17 }, // NORMAL
{ .size = 0x66, .y_offset = 9 }, // SUN
{ .size = 0x46, .y_offset = 9 }, // RAIN
{ .size = 0x86, .y_offset = 8 }, // HAIL
2018-10-06 23:04:53 +02:00
};
2018-10-07 12:32:20 +02:00
static const u8 sCastformElevations[] =
2018-10-06 23:04:53 +02:00
{
13, // NORMAL
14, // SUN
13, // RAIN
13, // HAIL
};
// Y position of the backsprite for each of the four Castform forms.
2018-10-07 12:32:20 +02:00
static const u8 sCastformBackSpriteYCoords[] =
2018-10-06 23:04:53 +02:00
{
0, // NORMAL
0, // SUN
0, // RAIN
0, // HAIL
};
2021-01-22 23:22:37 -05:00
// Placeholders for pokemon sprites to be created for a move animation effect (e.g. Role Play / Snatch)
#define TAG_MOVE_EFFECT_MON_1 55125
#define TAG_MOVE_EFFECT_MON_2 55126
static const struct SpriteTemplate sSpriteTemplate_MoveEffectMons[] =
2018-10-06 23:04:53 +02:00
{
{
2021-01-22 23:22:37 -05:00
.tileTag = TAG_MOVE_EFFECT_MON_1,
.paletteTag = TAG_MOVE_EFFECT_MON_1,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjNormal_64x64,
2018-10-06 23:04:53 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
},
{
2021-01-22 23:22:37 -05:00
.tileTag = TAG_MOVE_EFFECT_MON_2,
.paletteTag = TAG_MOVE_EFFECT_MON_2,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjNormal_64x64,
2018-10-06 23:04:53 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
}
};
2021-01-22 23:22:37 -05:00
static const struct SpriteSheet sSpriteSheet_MoveEffectMons[] =
2018-10-06 23:04:53 +02:00
{
2021-03-29 09:38:19 -04:00
{ gMiscBlank_Gfx, MON_PIC_SIZE, TAG_MOVE_EFFECT_MON_1, },
{ gMiscBlank_Gfx, MON_PIC_SIZE, TAG_MOVE_EFFECT_MON_2, },
2018-10-06 23:04:53 +02:00
};
2018-12-17 22:08:08 -06:00
u8 GetBattlerSpriteCoord(u8 battlerId, u8 coordType)
2018-10-06 23:04:53 +02:00
{
u8 retVal;
u16 species;
2019-08-08 13:06:55 +02:00
struct Pokemon *mon, *illusionMon;
2018-10-06 23:04:53 +02:00
struct BattleSpriteInfo *spriteInfo;
if (IsContest())
{
2018-12-17 22:08:08 -06:00
if (coordType == BATTLER_COORD_Y_PIC_OFFSET && battlerId == 3)
coordType = BATTLER_COORD_Y;
2018-10-06 23:04:53 +02:00
}
2018-12-17 22:08:08 -06:00
switch (coordType)
2018-10-06 23:04:53 +02:00
{
case BATTLER_COORD_X:
case BATTLER_COORD_X_2:
retVal = sBattlerCoords[IS_DOUBLE_BATTLE()][GetBattlerPosition(battlerId)].x;
break;
case BATTLER_COORD_Y:
retVal = sBattlerCoords[IS_DOUBLE_BATTLE()][GetBattlerPosition(battlerId)].y;
break;
2018-12-17 22:08:08 -06:00
case BATTLER_COORD_Y_PIC_OFFSET:
case BATTLER_COORD_Y_PIC_OFFSET_DEFAULT:
2018-10-06 23:04:53 +02:00
default:
if (IsContest())
{
2020-08-13 19:10:23 -04:00
if (gContestResources->moveAnim->hasTargetAnim)
species = gContestResources->moveAnim->targetSpecies;
2018-10-06 23:04:53 +02:00
else
2020-08-13 19:10:23 -04:00
species = gContestResources->moveAnim->species;
2018-10-06 23:04:53 +02:00
}
else
{
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
2019-08-08 13:06:55 +02:00
mon = &gEnemyParty[gBattlerPartyIndexes[battlerId]];
2018-10-06 23:04:53 +02:00
else
2019-08-08 13:06:55 +02:00
mon = &gPlayerParty[gBattlerPartyIndexes[battlerId]];
illusionMon = GetIllusionMonPtr(battlerId);
if (illusionMon != NULL)
mon = illusionMon;
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
species = GetMonData(mon, MON_DATA_SPECIES);
else
species = spriteInfo[battlerId].transformSpecies;
2018-10-06 23:04:53 +02:00
}
2018-12-17 22:08:08 -06:00
if (coordType == BATTLER_COORD_Y_PIC_OFFSET)
2018-10-06 23:04:53 +02:00
retVal = GetBattlerSpriteFinal_Y(battlerId, species, TRUE);
else
retVal = GetBattlerSpriteFinal_Y(battlerId, species, FALSE);
break;
}
return retVal;
}
u8 GetBattlerYDelta(u8 battlerId, u16 species)
{
u16 letter;
u32 personality;
struct BattleSpriteInfo *spriteInfo;
u8 ret;
u16 coordSpecies;
if (GetBattlerSide(battlerId) == B_SIDE_PLAYER || IsContest())
{
if (species == SPECIES_UNOWN)
{
if (IsContest())
{
2020-08-13 19:10:23 -04:00
if (gContestResources->moveAnim->hasTargetAnim)
personality = gContestResources->moveAnim->targetPersonality;
2018-10-06 23:04:53 +02:00
else
2020-08-13 19:10:23 -04:00
personality = gContestResources->moveAnim->personality;
2018-10-06 23:04:53 +02:00
}
else
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
personality = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_PERSONALITY);
else
personality = gTransformedPersonalities[battlerId];
}
letter = GET_UNOWN_LETTER(personality);
if (!letter)
coordSpecies = species;
else
coordSpecies = letter + SPECIES_UNOWN_B - 1;
ret = gMonBackPicCoords[coordSpecies].y_offset;
}
else if (species == SPECIES_CASTFORM)
{
2018-10-07 12:32:20 +02:00
ret = sCastformBackSpriteYCoords[gBattleMonForms[battlerId]];
2018-10-06 23:04:53 +02:00
}
else if (species > NUM_SPECIES)
{
ret = gMonBackPicCoords[0].y_offset;
}
else
{
ret = gMonBackPicCoords[species].y_offset;
}
}
else
{
if (species == SPECIES_UNOWN)
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
personality = GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_PERSONALITY);
else
personality = gTransformedPersonalities[battlerId];
letter = GET_UNOWN_LETTER(personality);
if (!letter)
coordSpecies = species;
else
coordSpecies = letter + SPECIES_UNOWN_B - 1;
ret = gMonFrontPicCoords[coordSpecies].y_offset;
}
else if (species == SPECIES_CASTFORM)
{
ret = gCastformFrontSpriteCoords[gBattleMonForms[battlerId]].y_offset;
}
else if (species > NUM_SPECIES)
{
ret = gMonFrontPicCoords[0].y_offset;
}
else
{
ret = gMonFrontPicCoords[species].y_offset;
}
}
return ret;
}
u8 GetBattlerElevation(u8 battlerId, u16 species)
{
u8 ret = 0;
if (GetBattlerSide(battlerId) == B_SIDE_OPPONENT)
{
if (!IsContest())
{
if (species == SPECIES_CASTFORM)
2018-10-07 12:32:20 +02:00
ret = sCastformElevations[gBattleMonForms[battlerId]];
2018-10-06 23:04:53 +02:00
else if (species > NUM_SPECIES)
ret = gEnemyMonElevation[0];
else
ret = gEnemyMonElevation[species];
}
}
return ret;
}
u8 GetBattlerSpriteFinal_Y(u8 battlerId, u16 species, bool8 a3)
{
u16 offset;
u8 y;
if (GetBattlerSide(battlerId) == B_SIDE_PLAYER || IsContest())
{
offset = GetBattlerYDelta(battlerId, species);
}
else
{
offset = GetBattlerYDelta(battlerId, species);
offset -= GetBattlerElevation(battlerId, species);
}
y = offset + sBattlerCoords[IS_DOUBLE_BATTLE()][GetBattlerPosition(battlerId)].y;
if (a3)
{
if (GetBattlerSide(battlerId) == B_SIDE_PLAYER)
y += 8;
if (y > 104)
y = 104;
}
return y;
}
2018-12-17 22:08:08 -06:00
u8 GetBattlerSpriteCoord2(u8 battlerId, u8 coordType)
2018-10-06 23:04:53 +02:00
{
u16 species;
struct BattleSpriteInfo *spriteInfo;
2018-12-17 22:08:08 -06:00
if (coordType == BATTLER_COORD_Y_PIC_OFFSET || coordType == BATTLER_COORD_Y_PIC_OFFSET_DEFAULT)
2018-10-06 23:04:53 +02:00
{
if (IsContest())
{
2020-08-13 19:10:23 -04:00
if (gContestResources->moveAnim->hasTargetAnim)
species = gContestResources->moveAnim->targetSpecies;
2018-10-06 23:04:53 +02:00
else
2020-08-13 19:10:23 -04:00
species = gContestResources->moveAnim->species;
2018-10-06 23:04:53 +02:00
}
else
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
species = gAnimBattlerSpecies[battlerId];
else
species = spriteInfo[battlerId].transformSpecies;
}
2018-12-17 22:08:08 -06:00
if (coordType == BATTLER_COORD_Y_PIC_OFFSET)
2018-10-06 23:04:53 +02:00
return GetBattlerSpriteFinal_Y(battlerId, species, TRUE);
else
return GetBattlerSpriteFinal_Y(battlerId, species, FALSE);
}
else
{
2018-12-17 22:08:08 -06:00
return GetBattlerSpriteCoord(battlerId, coordType);
2018-10-06 23:04:53 +02:00
}
}
u8 GetBattlerSpriteDefault_Y(u8 battlerId)
{
2018-12-17 22:08:08 -06:00
return GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y_PIC_OFFSET_DEFAULT);
2018-10-06 23:04:53 +02:00
}
u8 GetSubstituteSpriteDefault_Y(u8 battlerId)
{
u16 y;
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
y = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y) + 16;
else
y = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y) + 17;
return y;
}
u8 GetBattlerYCoordWithElevation(u8 battlerId)
{
u16 species;
u8 y;
struct BattleSpriteInfo *spriteInfo;
y = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y);
if (!IsContest())
{
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
species = GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
else
species = spriteInfo[battlerId].transformSpecies;
}
else
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
else
species = spriteInfo[battlerId].transformSpecies;
}
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
y -= GetBattlerElevation(battlerId, species);
}
return y;
}
2018-12-17 22:08:08 -06:00
u8 GetAnimBattlerSpriteId(u8 animBattler)
2018-10-06 23:04:53 +02:00
{
u8 *sprites;
2018-12-17 22:08:08 -06:00
if (animBattler == ANIM_ATTACKER)
2018-10-06 23:04:53 +02:00
{
if (IsBattlerSpritePresent(gBattleAnimAttacker))
{
sprites = gBattlerSpriteIds;
return sprites[gBattleAnimAttacker];
}
else
{
return 0xff;
}
}
2018-12-17 22:08:08 -06:00
else if (animBattler == ANIM_TARGET)
2018-10-06 23:04:53 +02:00
{
if (IsBattlerSpritePresent(gBattleAnimTarget))
{
sprites = gBattlerSpriteIds;
return sprites[gBattleAnimTarget];
}
else
{
return 0xff;
}
}
2018-12-17 22:08:08 -06:00
else if (animBattler == ANIM_ATK_PARTNER)
2018-10-06 23:04:53 +02:00
{
if (!IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimAttacker)))
return 0xff;
else
return gBattlerSpriteIds[BATTLE_PARTNER(gBattleAnimAttacker)];
}
else
{
if (IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimTarget)))
return gBattlerSpriteIds[BATTLE_PARTNER(gBattleAnimTarget)];
else
return 0xff;
}
}
void StoreSpriteCallbackInData6(struct Sprite *sprite, void (*callback)(struct Sprite*))
{
sprite->data[6] = (u32)(callback) & 0xffff;
sprite->data[7] = (u32)(callback) >> 16;
}
void SetCallbackToStoredInData6(struct Sprite *sprite)
{
u32 callback = (u16)sprite->data[6] | (sprite->data[7] << 16);
sprite->callback = (void (*)(struct Sprite *))callback;
}
2018-12-17 22:08:08 -06:00
void TranslateSpriteInCircleOverDuration(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[3])
{
2021-07-07 09:11:52 -04:00
sprite->x2 = Sin(sprite->data[0], sprite->data[1]);
sprite->y2 = Cos(sprite->data[0], sprite->data[1]);
2018-10-06 23:04:53 +02:00
sprite->data[0] += sprite->data[2];
if (sprite->data[0] >= 0x100)
sprite->data[0] -= 0x100;
else if (sprite->data[0] < 0)
sprite->data[0] += 0x100;
sprite->data[3]--;
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2018-12-17 22:08:08 -06:00
void TranslateSpriteInGrowingCircleOverDuration(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[3])
{
2021-07-07 09:11:52 -04:00
sprite->x2 = Sin(sprite->data[0], (sprite->data[5] >> 8) + sprite->data[1]);
sprite->y2 = Cos(sprite->data[0], (sprite->data[5] >> 8) + sprite->data[1]);
2018-10-06 23:04:53 +02:00
sprite->data[0] += sprite->data[2];
sprite->data[5] += sprite->data[4];
if (sprite->data[0] >= 0x100)
sprite->data[0] -= 0x100;
else if (sprite->data[0] < 0)
sprite->data[0] += 0x100;
sprite->data[3]--;
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
void sub_80A63C8(struct Sprite *sprite)
{
if (sprite->data[3])
{
2021-07-07 09:11:52 -04:00
sprite->x2 = Sin(sprite->data[0], sprite->data[1]);
sprite->y2 = Cos(sprite->data[4], sprite->data[1]);
2018-10-06 23:04:53 +02:00
sprite->data[0] += sprite->data[2];
sprite->data[4] += sprite->data[5];
if (sprite->data[0] >= 0x100)
sprite->data[0] -= 0x100;
else if (sprite->data[0] < 0)
sprite->data[0] += 0x100;
if (sprite->data[4] >= 0x100)
sprite->data[4] -= 0x100;
else if (sprite->data[4] < 0)
sprite->data[4] += 0x100;
sprite->data[3]--;
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2019-02-06 13:17:09 -06:00
void TranslateSpriteInEllipseOverDuration(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[3])
{
2021-07-07 09:11:52 -04:00
sprite->x2 = Sin(sprite->data[0], sprite->data[1]);
sprite->y2 = Cos(sprite->data[0], sprite->data[4]);
2018-10-06 23:04:53 +02:00
sprite->data[0] += sprite->data[2];
if (sprite->data[0] >= 0x100)
sprite->data[0] -= 0x100;
else if (sprite->data[0] < 0)
sprite->data[0] += 0x100;
sprite->data[3]--;
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
// Simply waits until the sprite's data[0] hits zero.
// This is used to let sprite anims or affine anims to run for a designated
// duration.
2018-11-26 09:55:31 -06:00
void WaitAnimForDuration(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
sprite->data[0]--;
else
SetCallbackToStoredInData6(sprite);
}
2019-02-06 13:17:09 -06:00
static void sub_80A64D0(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
sub_80A64EC(sprite);
2019-02-06 13:17:09 -06:00
sprite->callback = TranslateSpriteLinear;
2018-10-06 23:04:53 +02:00
sprite->callback(sprite);
}
void sub_80A64EC(struct Sprite *sprite)
{
s16 old;
2019-02-06 13:17:09 -06:00
int xDiff;
2018-10-06 23:04:53 +02:00
if (sprite->data[1] > sprite->data[2])
sprite->data[0] = -sprite->data[0];
2019-02-06 13:17:09 -06:00
xDiff = sprite->data[2] - sprite->data[1];
2018-10-06 23:04:53 +02:00
old = sprite->data[0];
2019-02-06 13:17:09 -06:00
sprite->data[0] = abs(xDiff / sprite->data[0]);
2018-10-06 23:04:53 +02:00
sprite->data[2] = (sprite->data[4] - sprite->data[3]) / sprite->data[0];
sprite->data[1] = old;
}
2019-02-06 13:17:09 -06:00
void TranslateSpriteLinear(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
{
sprite->data[0]--;
2021-07-07 09:11:52 -04:00
sprite->x2 += sprite->data[1];
sprite->y2 += sprite->data[2];
2018-10-06 23:04:53 +02:00
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2019-02-06 13:17:09 -06:00
void TranslateSpriteLinearFixedPoint(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
{
sprite->data[0]--;
sprite->data[3] += sprite->data[1];
sprite->data[4] += sprite->data[2];
2021-07-07 09:11:52 -04:00
sprite->x2 = sprite->data[3] >> 8;
sprite->y2 = sprite->data[4] >> 8;
2018-10-06 23:04:53 +02:00
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2019-02-06 13:17:09 -06:00
static void TranslateSpriteLinearFixedPointIconFrame(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
{
sprite->data[0]--;
sprite->data[3] += sprite->data[1];
sprite->data[4] += sprite->data[2];
2021-07-07 09:11:52 -04:00
sprite->x2 = sprite->data[3] >> 8;
sprite->y2 = sprite->data[4] >> 8;
2018-10-06 23:04:53 +02:00
}
else
{
SetCallbackToStoredInData6(sprite);
}
2019-02-06 13:17:09 -06:00
2018-10-06 23:04:53 +02:00
UpdateMonIconFrame(sprite);
}
void sub_80A65EC(struct Sprite *sprite)
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x + sprite->x2;
sprite->data[3] = sprite->y + sprite->y2;
2019-02-06 13:17:09 -06:00
sprite->data[2] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_X_2);
sprite->data[4] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_Y_PIC_OFFSET);
2018-10-06 23:04:53 +02:00
sprite->callback = sub_80A64D0;
}
2019-02-06 13:17:09 -06:00
void TranslateMonSpriteLinear(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
{
sprite->data[0]--;
2021-07-07 09:11:52 -04:00
gSprites[sprite->data[3]].x2 += sprite->data[1];
gSprites[sprite->data[3]].y2 += sprite->data[2];
2018-10-06 23:04:53 +02:00
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2019-02-06 13:17:09 -06:00
void TranslateMonSpriteLinearFixedPoint(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
{
sprite->data[0]--;
sprite->data[3] += sprite->data[1];
sprite->data[4] += sprite->data[2];
2021-07-07 09:11:52 -04:00
gSprites[sprite->data[5]].x2 = sprite->data[3] >> 8;
gSprites[sprite->data[5]].y2 = sprite->data[4] >> 8;
2018-10-06 23:04:53 +02:00
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2019-02-06 13:17:09 -06:00
void TranslateSpriteLinearAndFlicker(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->data[0] > 0)
{
sprite->data[0]--;
2021-07-07 09:11:52 -04:00
sprite->x2 = sprite->data[2] >> 8;
2018-10-06 23:04:53 +02:00
sprite->data[2] += sprite->data[1];
2021-07-07 09:11:52 -04:00
sprite->y2 = sprite->data[4] >> 8;
2018-10-06 23:04:53 +02:00
sprite->data[4] += sprite->data[3];
if (sprite->data[0] % sprite->data[5] == 0)
{
if (sprite->data[5])
sprite->invisible ^= 1;
}
}
else
{
SetCallbackToStoredInData6(sprite);
}
}
2018-12-17 22:08:08 -06:00
void DestroySpriteAndMatrix(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
FreeSpriteOamMatrix(sprite);
DestroyAnimSprite(sprite);
}
void sub_80A6760(struct Sprite *sprite)
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x + sprite->x2;
sprite->data[3] = sprite->y + sprite->y2;
2019-02-06 13:17:09 -06:00
sprite->data[2] = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_X_2);
sprite->data[4] = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_Y_PIC_OFFSET);
2018-10-06 23:04:53 +02:00
sprite->callback = sub_80A64D0;
}
void sub_80A67A4(struct Sprite *sprite)
{
ResetPaletteStructByUid(sprite->data[5]);
2018-12-17 22:08:08 -06:00
DestroySpriteAndMatrix(sprite);
2018-10-06 23:04:53 +02:00
}
2018-12-17 22:08:08 -06:00
void RunStoredCallbackWhenAffineAnimEnds(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->affineAnimEnded)
SetCallbackToStoredInData6(sprite);
}
2018-12-17 22:08:08 -06:00
void RunStoredCallbackWhenAnimEnds(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
if (sprite->animEnded)
SetCallbackToStoredInData6(sprite);
}
2019-02-06 13:17:09 -06:00
void DestroyAnimSpriteAndDisableBlend(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
DestroyAnimSprite(sprite);
}
2018-12-31 11:01:37 -06:00
void DestroyAnimVisualTaskAndDisableBlend(u8 taskId)
2018-10-06 23:04:53 +02:00
{
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
DestroyAnimVisualTask(taskId);
}
2018-12-17 22:08:08 -06:00
void SetSpriteCoordsToAnimAttackerCoords(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_X_2);
sprite->y = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_Y_PIC_OFFSET);
2018-10-06 23:04:53 +02:00
}
2018-12-19 21:13:26 -06:00
// Sets the initial x offset of the anim sprite depending on the horizontal orientation
// of the two involved mons.
void SetAnimSpriteInitialXOffset(struct Sprite *sprite, s16 xOffset)
2018-10-06 23:04:53 +02:00
{
2018-12-17 22:08:08 -06:00
u16 attackerX = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_X);
u16 targetX = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_X);
2018-10-06 23:04:53 +02:00
2018-12-17 22:08:08 -06:00
if (attackerX > targetX)
2018-10-06 23:04:53 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x -= xOffset;
2018-10-06 23:04:53 +02:00
}
2018-12-17 22:08:08 -06:00
else if (attackerX < targetX)
2018-10-06 23:04:53 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x += xOffset;
2018-10-06 23:04:53 +02:00
}
else
{
2018-12-17 22:08:08 -06:00
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
2021-07-07 09:11:52 -04:00
sprite->x -= xOffset;
2018-10-06 23:04:53 +02:00
else
2021-07-07 09:11:52 -04:00
sprite->x += xOffset;
2018-10-06 23:04:53 +02:00
}
}
2018-11-25 17:00:18 -06:00
void InitAnimArcTranslation(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2018-10-06 23:04:53 +02:00
InitAnimLinearTranslation(sprite);
sprite->data[6] = 0x8000 / sprite->data[0];
sprite->data[7] = 0;
}
2019-02-06 13:17:09 -06:00
bool8 TranslateAnimHorizontalArc(struct Sprite *sprite)
2018-10-06 23:04:53 +02:00
{
2018-12-19 21:13:26 -06:00
if (AnimTranslateLinear(sprite))
2018-10-06 23:04:53 +02:00
return TRUE;
sprite->data[7] += sprite->data[6];
2021-07-07 09:11:52 -04:00
sprite->y2 += Sin((u8)(sprite->data[7] >> 8), sprite->data[5]);
2018-10-06 23:04:53 +02:00
return FALSE;
}
2018-10-07 12:32:20 +02:00
2019-02-06 13:17:09 -06:00
bool8 TranslateAnimVerticalArc(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
if (AnimTranslateLinear(sprite))
2018-10-07 12:32:20 +02:00
return TRUE;
sprite->data[7] += sprite->data[6];
2021-07-07 09:11:52 -04:00
sprite->x2 += Sin((u8)(sprite->data[7] >> 8), sprite->data[5]);
2018-10-07 12:32:20 +02:00
return FALSE;
}
2019-02-06 13:17:09 -06:00
void SetSpritePrimaryCoordsFromSecondaryCoords(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x += sprite->x2;
sprite->y += sprite->y2;
sprite->x2 = 0;
sprite->y2 = 0;
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
void InitSpritePosToAnimTarget(struct Sprite *sprite, bool8 respectMonPicOffsets)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
// Battle anim sprites are automatically created at the anim target's center, which
// is why there is no else clause for the "respectMonPicOffsets" check.
if (!respectMonPicOffsets)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x = GetBattlerSpriteCoord2(gBattleAnimTarget, BATTLER_COORD_X);
sprite->y = GetBattlerSpriteCoord2(gBattleAnimTarget, BATTLER_COORD_Y);
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
SetAnimSpriteInitialXOffset(sprite, gBattleAnimArgs[0]);
2021-07-07 09:11:52 -04:00
sprite->y += gBattleAnimArgs[1];
2018-10-07 12:32:20 +02:00
}
2018-12-24 10:48:57 -06:00
void InitSpritePosToAnimAttacker(struct Sprite *sprite, bool8 respectMonPicOffsets)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
if (!respectMonPicOffsets)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x = GetBattlerSpriteCoord2(gBattleAnimAttacker, BATTLER_COORD_X);
sprite->y = GetBattlerSpriteCoord2(gBattleAnimAttacker, BATTLER_COORD_Y);
2018-10-07 12:32:20 +02:00
}
else
{
2021-07-07 09:11:52 -04:00
sprite->x = GetBattlerSpriteCoord2(gBattleAnimAttacker, BATTLER_COORD_X_2);
sprite->y = GetBattlerSpriteCoord2(gBattleAnimAttacker, BATTLER_COORD_Y_PIC_OFFSET);
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
SetAnimSpriteInitialXOffset(sprite, gBattleAnimArgs[0]);
2021-07-07 09:11:52 -04:00
sprite->y += gBattleAnimArgs[1];
2018-10-07 12:32:20 +02:00
}
void InitSpritePosToAnimAttackerPartner(struct Sprite *sprite, bool8 respectMonPicOffsets)
{
if (!respectMonPicOffsets)
{
sprite->x = GetBattlerSpriteCoord2(BATTLE_PARTNER(gBattleAnimAttacker), BATTLER_COORD_X);
sprite->y = GetBattlerSpriteCoord2(BATTLE_PARTNER(gBattleAnimAttacker), BATTLER_COORD_Y);
}
else
{
sprite->x = GetBattlerSpriteCoord2(BATTLE_PARTNER(gBattleAnimAttacker), BATTLER_COORD_X_2);
sprite->y = GetBattlerSpriteCoord2(BATTLE_PARTNER(gBattleAnimAttacker), BATTLER_COORD_Y_PIC_OFFSET);
}
SetAnimSpriteInitialXOffset(sprite, gBattleAnimArgs[0]);
sprite->y += gBattleAnimArgs[1];
}
bool32 InitSpritePosToAnimBattler(u32 animBattlerId, struct Sprite *sprite, bool8 respectMonPicOffsets)
{
u32 battlerId = GetAnimBattlerId(animBattlerId);
if (GetAnimBattlerSpriteId(animBattlerId) == 0xFF || !IsBattlerSpriteVisible(battlerId))
{
DestroyAnimSprite(sprite);
return FALSE;
}
if (!respectMonPicOffsets)
{
sprite->x = GetBattlerSpriteCoord2(battlerId, BATTLER_COORD_X);
sprite->y = GetBattlerSpriteCoord2(battlerId, BATTLER_COORD_Y);
}
else if (animBattlerId != ANIM_TARGET)
{
sprite->x = GetBattlerSpriteCoord2(battlerId, BATTLER_COORD_X_2);
sprite->y = GetBattlerSpriteCoord2(battlerId, BATTLER_COORD_Y_PIC_OFFSET);
}
SetAnimSpriteInitialXOffset(sprite, gBattleAnimArgs[0]);
sprite->y += gBattleAnimArgs[1];
return TRUE;
}
2018-10-07 12:32:20 +02:00
u8 GetBattlerSide(u8 battlerId)
{
return GET_BATTLER_SIDE2(battlerId);
}
u8 GetBattlerPosition(u8 battlerId)
{
return GET_BATTLER_POSITION(battlerId);
}
u8 GetBattlerAtPosition(u8 position)
{
u8 i;
for (i = 0; i < gBattlersCount; i++)
{
if (gBattlerPositions[i] == position)
break;
}
return i;
}
bool8 IsBattlerSpritePresent(u8 battlerId)
{
if (IsContest())
{
if (gBattleAnimAttacker == battlerId)
return TRUE;
else if (gBattleAnimTarget == battlerId)
return TRUE;
else
return FALSE;
}
else
{
if (gBattlerPositions[battlerId] == 0xff)
return FALSE;
2019-08-08 13:06:55 +02:00
if (!gBattleStruct->spriteIgnore0Hp)
2018-10-07 12:32:20 +02:00
{
2019-08-08 13:06:55 +02:00
if (GetBattlerSide(battlerId) == B_SIDE_OPPONENT)
{
if (GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_HP) == 0)
return FALSE;
}
else
{
if (GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_HP) == 0)
return FALSE;
}
2018-10-07 12:32:20 +02:00
}
2019-08-08 13:06:55 +02:00
return TRUE;
2018-10-07 12:32:20 +02:00
}
}
bool8 IsDoubleBattle(void)
2018-10-07 12:32:20 +02:00
{
return IS_DOUBLE_BATTLE();
}
2021-02-27 17:41:30 -06:00
void GetBattleAnimBg1Data(struct BattleAnimBgData *out)
2018-10-07 12:32:20 +02:00
{
if (IsContest())
{
out->bgTiles = gUnknown_0202305C;
out->bgTilemap = (u16 *)gUnknown_02023060;
out->paletteId = 14;
out->bgId = 1;
out->tilesOffset = 0;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
else
{
out->bgTiles = gUnknown_0202305C;
out->bgTilemap = (u16 *)gUnknown_02023060;
out->paletteId = 8;
out->bgId = 1;
out->tilesOffset = 0x200;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
void GetBattleAnimBgData(struct BattleAnimBgData *out, u32 bgId)
2018-10-07 12:32:20 +02:00
{
if (IsContest())
{
2021-02-27 17:41:30 -06:00
out->bgTiles = gUnknown_0202305C;
out->bgTilemap = (u16 *)gUnknown_02023060;
out->paletteId = 14;
out->bgId = 1;
out->tilesOffset = 0;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
2021-02-27 17:41:30 -06:00
else if (bgId == 1)
2018-10-07 12:32:20 +02:00
{
2021-02-27 17:41:30 -06:00
GetBattleAnimBg1Data(out);
2018-10-07 12:32:20 +02:00
}
else
{
2021-02-27 17:41:30 -06:00
out->bgTiles = gUnknown_0202305C;
out->bgTilemap = (u16 *)gUnknown_02023060;
out->paletteId = 9;
out->bgId = 2;
out->tilesOffset = 0x300;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
void sub_80A6BFC(struct BattleAnimBgData *out, u8 unused)
2018-10-07 12:32:20 +02:00
{
2021-02-27 17:41:30 -06:00
out->bgTiles = gUnknown_0202305C;
out->bgTilemap = (u16 *)gUnknown_02023060;
2018-10-07 12:32:20 +02:00
if (IsContest())
{
2021-02-27 17:41:30 -06:00
out->paletteId = 14;
out->bgId = 1;
out->tilesOffset = 0;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
else if (GetBattlerSpriteBGPriorityRank(gBattleAnimAttacker) == 1)
2018-10-07 12:32:20 +02:00
{
2021-02-27 17:41:30 -06:00
out->paletteId = 8;
out->bgId = 1;
out->tilesOffset = 0x200;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
else
{
2021-02-27 17:41:30 -06:00
out->paletteId = 9;
out->bgId = 2;
out->tilesOffset = 0x300;
out->unused = 0;
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
void ClearBattleAnimBg(u32 bgId)
2018-10-07 12:32:20 +02:00
{
2021-02-27 17:41:30 -06:00
struct BattleAnimBgData bgAnimData;
2018-10-07 12:32:20 +02:00
2021-02-27 17:41:30 -06:00
GetBattleAnimBgData(&bgAnimData, bgId);
CpuFill32(0, bgAnimData.bgTiles, 0x2000);
LoadBgTiles(bgAnimData.bgId, bgAnimData.bgTiles, 0x2000, bgAnimData.tilesOffset);
FillBgTilemapBufferRect(bgAnimData.bgId, 0, 0, 0, 32, 64, 17);
CopyBgTilemapBufferToVram(bgAnimData.bgId);
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
void AnimLoadCompressedBgGfx(u32 bgId, const u32 *src, u32 tilesOffset)
2018-10-07 12:32:20 +02:00
{
CpuFill32(0, gUnknown_0202305C, 0x2000);
LZDecompressWram(src, gUnknown_0202305C);
LoadBgTiles(bgId, gUnknown_0202305C, 0x2000, tilesOffset);
}
2019-02-06 13:17:09 -06:00
static void InitAnimBgTilemapBuffer(u32 bgId, const void *src)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
FillBgTilemapBufferRect(bgId, 0, 0, 0, 32, 64, 17);
2018-10-07 12:32:20 +02:00
CopyToBgTilemapBuffer(bgId, src, 0, 0);
}
2019-02-06 13:17:09 -06:00
void AnimLoadCompressedBgTilemap(u32 bgId, const void *src)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
InitAnimBgTilemapBuffer(bgId, src);
2018-10-07 12:32:20 +02:00
CopyBgTilemapBufferToVram(bgId);
}
2020-02-21 12:38:20 -05:00
void AnimLoadCompressedBgTilemapHandleContest(struct BattleAnimBgData *unk, const void *src, u32 arg2)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
InitAnimBgTilemapBuffer(unk->bgId, src);
2018-10-07 12:32:20 +02:00
if (IsContest() == TRUE)
2019-02-06 13:17:09 -06:00
sub_80A4720(unk->paletteId, unk->bgTilemap, 0, arg2);
2018-10-07 12:32:20 +02:00
CopyBgTilemapBufferToVram(unk->bgId);
}
2019-11-24 18:00:51 +01:00
u8 GetBattleBgPaletteNum(void)
2018-10-07 12:32:20 +02:00
{
if (IsContest())
return 1;
else
return 2;
}
void UpdateAnimBg3ScreenSize(bool8 largeScreenSize)
2018-10-07 12:32:20 +02:00
{
if (!largeScreenSize || IsContest())
2018-10-07 12:32:20 +02:00
{
SetAnimBgAttribute(3, BG_ANIM_SCREEN_SIZE, 0);
SetAnimBgAttribute(3, BG_ANIM_AREA_OVERFLOW_MODE, 1);
}
else
{
SetAnimBgAttribute(3, BG_ANIM_SCREEN_SIZE, 1);
SetAnimBgAttribute(3, BG_ANIM_AREA_OVERFLOW_MODE, 0);
}
}
2019-10-04 07:42:45 -04:00
void TradeMenuBouncePartySprites(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2018-11-25 17:00:18 -06:00
InitSpriteDataForLinearTranslation(sprite);
2019-02-06 13:17:09 -06:00
sprite->callback = TranslateSpriteLinearFixedPointIconFrame;
2018-10-07 12:32:20 +02:00
sprite->callback(sprite);
}
2018-11-25 17:00:18 -06:00
void InitSpriteDataForLinearTranslation(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
s16 x = (sprite->data[2] - sprite->data[1]) << 8;
s16 y = (sprite->data[4] - sprite->data[3]) << 8;
sprite->data[1] = x / sprite->data[0];
sprite->data[2] = y / sprite->data[0];
sprite->data[4] = 0;
sprite->data[3] = 0;
}
void InitAnimLinearTranslation(struct Sprite *sprite)
{
int x = sprite->data[2] - sprite->data[1];
int y = sprite->data[4] - sprite->data[3];
bool8 movingLeft = x < 0;
bool8 movingUp = y < 0;
u16 xDelta = abs(x) << 8;
u16 yDelta = abs(y) << 8;
xDelta = xDelta / sprite->data[0];
yDelta = yDelta / sprite->data[0];
if (movingLeft)
xDelta |= 1;
else
xDelta &= ~1;
if (movingUp)
yDelta |= 1;
else
yDelta &= ~1;
sprite->data[1] = xDelta;
sprite->data[2] = yDelta;
sprite->data[4] = 0;
sprite->data[3] = 0;
}
2018-11-25 17:00:18 -06:00
void StartAnimLinearTranslation(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2018-10-07 12:32:20 +02:00
InitAnimLinearTranslation(sprite);
2021-01-23 00:40:46 -05:00
sprite->callback = AnimTranslateLinear_WaitEnd;
2018-10-07 12:32:20 +02:00
sprite->callback(sprite);
}
void sub_80A6F14(struct Sprite *sprite)
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2018-10-07 12:32:20 +02:00
InitAnimLinearTranslation(sprite);
sprite->callback = sub_80A6FB4;
sprite->callback(sprite);
}
2018-12-19 21:13:26 -06:00
bool8 AnimTranslateLinear(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
u16 v1, v2, x, y;
if (!sprite->data[0])
return TRUE;
v1 = sprite->data[1];
v2 = sprite->data[2];
x = sprite->data[3];
y = sprite->data[4];
x += v1;
y += v2;
if (v1 & 1)
2021-07-07 09:11:52 -04:00
sprite->x2 = -(x >> 8);
2018-10-07 12:32:20 +02:00
else
2021-07-07 09:11:52 -04:00
sprite->x2 = x >> 8;
2018-10-07 12:32:20 +02:00
if (v2 & 1)
2021-07-07 09:11:52 -04:00
sprite->y2 = -(y >> 8);
2018-10-07 12:32:20 +02:00
else
2021-07-07 09:11:52 -04:00
sprite->y2 = y >> 8;
2018-10-07 12:32:20 +02:00
sprite->data[3] = x;
sprite->data[4] = y;
sprite->data[0]--;
return FALSE;
}
2021-01-23 00:40:46 -05:00
void AnimTranslateLinear_WaitEnd(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
if (AnimTranslateLinear(sprite))
2018-10-07 12:32:20 +02:00
SetCallbackToStoredInData6(sprite);
}
2019-02-06 13:17:09 -06:00
static void sub_80A6FB4(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
sub_8039E9C(sprite);
2018-12-19 21:13:26 -06:00
if (AnimTranslateLinear(sprite))
2018-10-07 12:32:20 +02:00
SetCallbackToStoredInData6(sprite);
}
2021-01-23 00:40:46 -05:00
void InitAnimLinearTranslationWithSpeed(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
int v1 = abs(sprite->data[2] - sprite->data[1]) << 8;
sprite->data[0] = v1 / sprite->data[0];
InitAnimLinearTranslation(sprite);
}
2021-01-23 00:40:46 -05:00
void InitAnimLinearTranslationWithSpeedAndPos(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2021-01-23 00:40:46 -05:00
InitAnimLinearTranslationWithSpeed(sprite);
sprite->callback = AnimTranslateLinear_WaitEnd;
2018-10-07 12:32:20 +02:00
sprite->callback(sprite);
}
2018-12-19 21:13:26 -06:00
static void InitAnimFastLinearTranslation(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
int xDiff = sprite->data[2] - sprite->data[1];
int yDiff = sprite->data[4] - sprite->data[3];
bool8 x_sign = xDiff < 0;
bool8 y_sign = yDiff < 0;
u16 x2 = abs(xDiff) << 4;
u16 y2 = abs(yDiff) << 4;
2018-10-07 12:32:20 +02:00
x2 /= sprite->data[0];
y2 /= sprite->data[0];
if (x_sign)
x2 |= 1;
else
x2 &= ~1;
if (y_sign)
y2 |= 1;
else
y2 &= ~1;
sprite->data[1] = x2;
sprite->data[2] = y2;
sprite->data[4] = 0;
sprite->data[3] = 0;
}
2018-12-19 21:13:26 -06:00
void InitAndRunAnimFastLinearTranslation(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2018-12-19 21:13:26 -06:00
InitAnimFastLinearTranslation(sprite);
2021-01-23 00:40:46 -05:00
sprite->callback = AnimFastTranslateLinearWaitEnd;
2018-10-07 12:32:20 +02:00
sprite->callback(sprite);
}
2018-12-19 21:13:26 -06:00
bool8 AnimFastTranslateLinear(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
u16 v1, v2, x, y;
if (!sprite->data[0])
return TRUE;
v1 = sprite->data[1];
v2 = sprite->data[2];
x = sprite->data[3];
y = sprite->data[4];
x += v1;
y += v2;
if (v1 & 1)
2021-07-07 09:11:52 -04:00
sprite->x2 = -(x >> 4);
2018-10-07 12:32:20 +02:00
else
2021-07-07 09:11:52 -04:00
sprite->x2 = x >> 4;
2018-10-07 12:32:20 +02:00
if (v2 & 1)
2021-07-07 09:11:52 -04:00
sprite->y2 = -(y >> 4);
2018-10-07 12:32:20 +02:00
else
2021-07-07 09:11:52 -04:00
sprite->y2 = y >> 4;
2018-10-07 12:32:20 +02:00
sprite->data[3] = x;
sprite->data[4] = y;
sprite->data[0]--;
return FALSE;
}
2021-01-23 00:40:46 -05:00
static void AnimFastTranslateLinearWaitEnd(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
if (AnimFastTranslateLinear(sprite))
2018-10-07 12:32:20 +02:00
SetCallbackToStoredInData6(sprite);
}
2018-12-19 21:13:26 -06:00
void InitAnimFastLinearTranslationWithSpeed(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
int xDiff = abs(sprite->data[2] - sprite->data[1]) << 4;
sprite->data[0] = xDiff / sprite->data[0];
InitAnimFastLinearTranslation(sprite);
2018-10-07 12:32:20 +02:00
}
2021-01-23 00:40:46 -05:00
void InitAnimFastLinearTranslationWithSpeedAndPos(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->data[1] = sprite->x;
sprite->data[3] = sprite->y;
2018-12-19 21:13:26 -06:00
InitAnimFastLinearTranslationWithSpeed(sprite);
2021-01-23 00:40:46 -05:00
sprite->callback = AnimFastTranslateLinearWaitEnd;
2018-10-07 12:32:20 +02:00
sprite->callback(sprite);
}
2018-12-17 22:08:08 -06:00
void SetSpriteRotScale(u8 spriteId, s16 xScale, s16 yScale, u16 rotation)
2018-10-07 12:32:20 +02:00
{
int i;
struct ObjAffineSrcData src;
struct OamMatrix matrix;
src.xScale = xScale;
src.yScale = yScale;
src.rotation = rotation;
if (sub_80A7238())
src.xScale = -src.xScale;
i = gSprites[spriteId].oam.matrixNum;
ObjAffineSet(&src, &matrix, 1, 2);
gOamMatrices[i].a = matrix.a;
gOamMatrices[i].b = matrix.b;
gOamMatrices[i].c = matrix.c;
gOamMatrices[i].d = matrix.d;
}
2019-02-06 13:17:09 -06:00
static bool8 sub_80A7238(void)
2018-10-07 12:32:20 +02:00
{
if (IsContest())
{
if (gSprites[GetAnimBattlerSpriteId(ANIM_ATTACKER)].data[2] == SPECIES_UNOWN)
return FALSE;
else
return TRUE;
}
else
{
return FALSE;
}
}
2018-12-17 22:08:08 -06:00
void PrepareBattlerSpriteForRotScale(u8 spriteId, u8 objMode)
2018-10-07 12:32:20 +02:00
{
u8 battlerId = gSprites[spriteId].data[0];
if (IsContest() || IsBattlerSpriteVisible(battlerId))
gSprites[spriteId].invisible = FALSE;
gSprites[spriteId].oam.objMode = objMode;
gSprites[spriteId].affineAnimPaused = TRUE;
if (!IsContest() && !gSprites[spriteId].oam.affineMode)
2018-12-17 22:08:08 -06:00
gSprites[spriteId].oam.matrixNum = gBattleSpritesDataPtr->healthBoxesData[battlerId].matrixNum;
gSprites[spriteId].oam.affineMode = ST_OAM_AFFINE_DOUBLE;
2018-10-07 12:32:20 +02:00
CalcCenterToCornerVec(&gSprites[spriteId], gSprites[spriteId].oam.shape, gSprites[spriteId].oam.size, gSprites[spriteId].oam.affineMode);
}
2018-12-17 22:08:08 -06:00
void ResetSpriteRotScale(u8 spriteId)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
SetSpriteRotScale(spriteId, 0x100, 0x100, 0);
gSprites[spriteId].oam.affineMode = ST_OAM_AFFINE_NORMAL;
gSprites[spriteId].oam.objMode = ST_OAM_OBJ_NORMAL;
2018-10-07 12:32:20 +02:00
gSprites[spriteId].affineAnimPaused = FALSE;
CalcCenterToCornerVec(&gSprites[spriteId], gSprites[spriteId].oam.shape, gSprites[spriteId].oam.size, gSprites[spriteId].oam.affineMode);
}
2018-12-17 22:08:08 -06:00
// Sets the sprite's y offset equal to the y displacement caused by the
// matrix's rotation.
void SetBattlerSpriteYOffsetFromRotation(u8 spriteId)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
u16 matrixNum = gSprites[spriteId].oam.matrixNum;
// The "c" component of the battler sprite matrix contains the sine of the rotation angle divided by some scale amount.
s16 c = gOamMatrices[matrixNum].c;
2018-10-07 12:32:20 +02:00
if (c < 0)
c = -c;
2018-12-17 22:08:08 -06:00
2021-07-07 09:11:52 -04:00
gSprites[spriteId].y2 = c >> 3;
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
void TrySetSpriteRotScale(struct Sprite *sprite, bool8 recalcCenterVector, s16 xScale, s16 yScale, u16 rotation)
2018-10-07 12:32:20 +02:00
{
int i;
struct ObjAffineSrcData src;
struct OamMatrix matrix;
if (sprite->oam.affineMode & 1)
{
sprite->affineAnimPaused = TRUE;
2018-12-17 22:08:08 -06:00
if (recalcCenterVector)
2018-10-07 12:32:20 +02:00
CalcCenterToCornerVec(sprite, sprite->oam.shape, sprite->oam.size, sprite->oam.affineMode);
src.xScale = xScale;
src.yScale = yScale;
src.rotation = rotation;
if (sub_80A7238())
src.xScale = -src.xScale;
i = sprite->oam.matrixNum;
ObjAffineSet(&src, &matrix, 1, 2);
gOamMatrices[i].a = matrix.a;
gOamMatrices[i].b = matrix.b;
gOamMatrices[i].c = matrix.c;
gOamMatrices[i].d = matrix.d;
}
}
void ResetSpriteRotScale_PreserveAffine(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
TrySetSpriteRotScale(sprite, TRUE, 0x100, 0x100, 0);
2018-10-07 12:32:20 +02:00
sprite->affineAnimPaused = FALSE;
CalcCenterToCornerVec(sprite, sprite->oam.shape, sprite->oam.size, sprite->oam.affineMode);
}
static u16 ArcTan2_(s16 a, s16 b)
{
return ArcTan2(a, b);
}
u16 ArcTan2Neg(s16 a, s16 b)
{
u16 var = ArcTan2_(a, b);
return -var;
}
void SetGrayscaleOrOriginalPalette(u16 paletteNum, bool8 restoreOriginalColor)
2018-10-07 12:32:20 +02:00
{
int i;
2018-12-17 22:08:08 -06:00
struct PlttData *originalColor;
struct PlttData *destColor;
2018-10-07 12:32:20 +02:00
u16 average;
2018-12-17 22:08:08 -06:00
paletteNum *= 16;
2018-10-07 12:32:20 +02:00
2018-12-17 22:08:08 -06:00
if (!restoreOriginalColor)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
for (i = 0; i < 16; i++)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
originalColor = (struct PlttData *)&gPlttBufferUnfaded[paletteNum + i];
average = originalColor->r + originalColor->g + originalColor->b;
2018-10-07 12:32:20 +02:00
average /= 3;
2018-12-17 22:08:08 -06:00
destColor = (struct PlttData *)&gPlttBufferFaded[paletteNum + i];
destColor->r = average;
destColor->g = average;
destColor->b = average;
2018-10-07 12:32:20 +02:00
}
}
else
{
2018-12-17 22:08:08 -06:00
CpuCopy32(&gPlttBufferUnfaded[paletteNum], &gPlttBufferFaded[paletteNum], 32);
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
u32 GetBattleBgPalettesMask(u8 battleBackground, u8 attacker, u8 target, u8 attackerPartner, u8 targetPartner, u8 a6, u8 a7)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
u32 selectedPalettes = 0;
2018-10-07 12:32:20 +02:00
u32 shift;
2019-02-06 13:17:09 -06:00
if (battleBackground)
2018-10-07 12:32:20 +02:00
{
if (!IsContest())
2019-02-06 13:17:09 -06:00
selectedPalettes = 0xe;
2018-10-07 12:32:20 +02:00
else
2019-11-24 18:00:51 +01:00
selectedPalettes = 1 << GetBattleBgPaletteNum();
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
if (attacker)
2018-10-07 12:32:20 +02:00
{
shift = gBattleAnimAttacker + 16;
2019-02-06 13:17:09 -06:00
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
if (target)
{
2018-10-07 12:32:20 +02:00
shift = gBattleAnimTarget + 16;
2019-02-06 13:17:09 -06:00
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
if (attackerPartner)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
if (IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimAttacker)))
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
shift = BATTLE_PARTNER(gBattleAnimAttacker) + 16;
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
}
2019-02-06 13:17:09 -06:00
if (targetPartner)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
if (IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimTarget)))
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
shift = BATTLE_PARTNER(gBattleAnimTarget) + 16;
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
}
if (a6)
{
if (!IsContest())
2019-02-06 13:17:09 -06:00
selectedPalettes |= 0x100;
2018-10-07 12:32:20 +02:00
else
2019-02-06 13:17:09 -06:00
selectedPalettes |= 0x4000;
2018-10-07 12:32:20 +02:00
}
if (a7)
{
if (!IsContest())
2019-02-06 13:17:09 -06:00
selectedPalettes |= 0x200;
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
return selectedPalettes;
2018-10-07 12:32:20 +02:00
}
2021-02-27 17:41:30 -06:00
u32 GetBattleMonSpritePalettesMask(u8 playerLeft, u8 playerRight, u8 opponentLeft, u8 opponentRight)
2018-10-07 12:32:20 +02:00
{
2021-02-27 17:41:30 -06:00
u32 selectedPalettes = 0;
2018-10-07 12:32:20 +02:00
u32 shift;
if (IsContest())
{
2021-02-27 17:41:30 -06:00
if (playerLeft)
2018-10-07 12:32:20 +02:00
{
2021-02-27 17:41:30 -06:00
selectedPalettes |= 1 << 18;
return selectedPalettes;
2018-10-07 12:32:20 +02:00
}
}
else
{
2021-02-27 17:41:30 -06:00
if (playerLeft)
2018-10-07 12:32:20 +02:00
{
if (IsBattlerSpriteVisible(GetBattlerAtPosition(B_POSITION_PLAYER_LEFT)))
{
2021-02-27 17:41:30 -06:00
selectedPalettes |= 1 << (GetBattlerAtPosition(B_POSITION_PLAYER_LEFT) + 16);
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
if (playerRight)
2018-10-07 12:32:20 +02:00
{
if (IsBattlerSpriteVisible(GetBattlerAtPosition(B_POSITION_PLAYER_RIGHT)))
{
shift = GetBattlerAtPosition(B_POSITION_PLAYER_RIGHT) + 16;
2021-02-27 17:41:30 -06:00
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
if (opponentLeft)
2018-10-07 12:32:20 +02:00
{
if (IsBattlerSpriteVisible(GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT)))
{
shift = GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT) + 16;
2021-02-27 17:41:30 -06:00
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
}
2021-02-27 17:41:30 -06:00
if (opponentRight)
2018-10-07 12:32:20 +02:00
{
if (IsBattlerSpriteVisible(GetBattlerAtPosition(B_POSITION_OPPONENT_RIGHT)))
{
shift = GetBattlerAtPosition(B_POSITION_OPPONENT_RIGHT) + 16;
2021-02-27 17:41:30 -06:00
selectedPalettes |= 1 << shift;
2018-10-07 12:32:20 +02:00
}
}
}
2021-02-27 17:41:30 -06:00
return selectedPalettes;
2018-10-07 12:32:20 +02:00
}
2021-01-23 00:40:46 -05:00
// Presumably something commented here, just returns arg
u8 AnimDummyReturnArg(u8 battler)
2018-10-07 12:32:20 +02:00
{
2021-01-23 00:40:46 -05:00
return battler;
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
static u8 GetBattlerAtPosition_(u8 position)
2018-10-07 12:32:20 +02:00
{
return GetBattlerAtPosition(position);
}
2019-10-20 16:22:53 +02:00
void AnimSpriteOnMonPos(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
bool8 var;
if (!sprite->data[0])
{
if (!gBattleAnimArgs[3])
var = TRUE;
else
var = FALSE;
if (gBattleAnimArgs[2] == 0)
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimAttacker(sprite, var);
else if (gBattleAnimArgs[2] == 1)
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimTarget(sprite, var);
else if (gBattleAnimArgs[2] == 2)
InitSpritePosToAnimAttackerPartner(sprite, var);
2018-10-07 12:32:20 +02:00
sprite->data[0]++;
}
else if (sprite->animEnded || sprite->affineAnimEnded)
{
2018-12-17 22:08:08 -06:00
DestroySpriteAndMatrix(sprite);
2018-10-07 12:32:20 +02:00
}
}
// Linearly translates a sprite to a target position on the
// other mon's sprite.
// arg 0: initial x offset
// arg 1: initial y offset
// arg 2: target x offset
// arg 3: target y offset
// arg 4: duration
// arg 5: lower 8 bits = location on attacking mon, upper 8 bits = location on target mon pick to target
void TranslateAnimSpriteToTargetMonLocation(struct Sprite *sprite)
{
bool8 v1;
2018-12-17 22:08:08 -06:00
u8 coordType;
2018-10-07 12:32:20 +02:00
if (!(gBattleAnimArgs[5] & 0xff00))
v1 = TRUE;
else
v1 = FALSE;
if (!(gBattleAnimArgs[5] & 0xff))
2018-12-17 22:08:08 -06:00
coordType = BATTLER_COORD_Y_PIC_OFFSET;
2018-10-07 12:32:20 +02:00
else
2018-12-17 22:08:08 -06:00
coordType = BATTLER_COORD_Y;
2018-10-07 12:32:20 +02:00
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimAttacker(sprite, v1);
2018-10-07 12:32:20 +02:00
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
gBattleAnimArgs[2] = -gBattleAnimArgs[2];
sprite->data[0] = gBattleAnimArgs[4];
sprite->data[2] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_X_2) + gBattleAnimArgs[2];
2018-12-17 22:08:08 -06:00
sprite->data[4] = GetBattlerSpriteCoord(gBattleAnimTarget, coordType) + gBattleAnimArgs[3];
2018-11-25 17:00:18 -06:00
sprite->callback = StartAnimLinearTranslation;
2018-10-07 12:32:20 +02:00
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
}
2019-10-20 10:47:56 +02:00
void AnimThrowProjectile(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimAttacker(sprite, 1);
2018-10-07 12:32:20 +02:00
if (GetBattlerSide(gBattleAnimAttacker))
gBattleAnimArgs[2] = -gBattleAnimArgs[2];
sprite->data[0] = gBattleAnimArgs[4];
2019-02-06 13:17:09 -06:00
sprite->data[2] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_X_2) + gBattleAnimArgs[2];
sprite->data[4] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_Y_PIC_OFFSET) + gBattleAnimArgs[3];
2018-10-07 12:32:20 +02:00
sprite->data[5] = gBattleAnimArgs[5];
2018-11-25 17:00:18 -06:00
InitAnimArcTranslation(sprite);
2019-10-20 10:47:56 +02:00
sprite->callback = AnimThrowProjectile_Step;
2018-10-07 12:32:20 +02:00
}
2019-10-20 10:47:56 +02:00
static void AnimThrowProjectile_Step(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2019-02-06 13:17:09 -06:00
if (TranslateAnimHorizontalArc(sprite))
2018-10-07 12:32:20 +02:00
DestroyAnimSprite(sprite);
}
2020-02-14 18:11:04 -05:00
void AnimTravelDiagonally(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
bool8 r4;
2019-02-06 13:17:09 -06:00
u8 battlerId, coordType;
2018-10-07 12:32:20 +02:00
if (!gBattleAnimArgs[6])
{
r4 = TRUE;
2019-02-06 13:17:09 -06:00
coordType = BATTLER_COORD_Y_PIC_OFFSET;
2018-10-07 12:32:20 +02:00
}
else
{
r4 = FALSE;
2019-02-06 13:17:09 -06:00
coordType = BATTLER_COORD_Y;
2018-10-07 12:32:20 +02:00
}
2020-02-14 18:11:04 -05:00
if (gBattleAnimArgs[5] == ANIM_ATTACKER)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimAttacker(sprite, r4);
2018-10-07 12:32:20 +02:00
battlerId = gBattleAnimAttacker;
}
else
{
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimTarget(sprite, r4);
2018-10-07 12:32:20 +02:00
battlerId = gBattleAnimTarget;
}
if (GetBattlerSide(gBattleAnimAttacker))
gBattleAnimArgs[2] = -gBattleAnimArgs[2];
2018-12-19 21:13:26 -06:00
InitSpritePosToAnimTarget(sprite, r4);
2018-10-07 12:32:20 +02:00
sprite->data[0] = gBattleAnimArgs[4];
sprite->data[2] = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_X_2) + gBattleAnimArgs[2];
2019-02-06 13:17:09 -06:00
sprite->data[4] = GetBattlerSpriteCoord(battlerId, coordType) + gBattleAnimArgs[3];
2018-11-25 17:00:18 -06:00
sprite->callback = StartAnimLinearTranslation;
2018-10-07 12:32:20 +02:00
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
}
2018-12-17 22:08:08 -06:00
s16 CloneBattlerSpriteWithBlend(u8 animBattler)
2018-10-07 12:32:20 +02:00
{
u16 i;
2018-12-17 22:08:08 -06:00
u8 spriteId = GetAnimBattlerSpriteId(animBattler);
2018-10-07 12:32:20 +02:00
2021-02-20 00:30:37 -05:00
if (spriteId != SPRITE_NONE)
2018-10-07 12:32:20 +02:00
{
for (i = 0; i < MAX_SPRITES; i++)
{
if (!gSprites[i].inUse)
{
gSprites[i] = gSprites[spriteId];
2018-12-17 22:08:08 -06:00
gSprites[i].oam.objMode = ST_OAM_OBJ_BLEND;
2018-10-07 12:32:20 +02:00
gSprites[i].invisible = FALSE;
return i;
}
}
}
return -1;
2018-10-07 12:32:20 +02:00
}
void obj_delete_but_dont_free_vram(struct Sprite *sprite)
{
sprite->usingSheet = TRUE;
DestroySprite(sprite);
}
2020-02-20 00:04:42 -05:00
// Only used to fade Moonlight moon sprite in
void AnimTask_AlphaFadeIn(u8 taskId)
2018-10-07 12:32:20 +02:00
{
s16 v1 = 0;
s16 v2 = 0;
if (gBattleAnimArgs[2] > gBattleAnimArgs[0])
v2 = 1;
if (gBattleAnimArgs[2] < gBattleAnimArgs[0])
v2 = -1;
if (gBattleAnimArgs[3] > gBattleAnimArgs[1])
v1 = 1;
if (gBattleAnimArgs[3] < gBattleAnimArgs[1])
v1 = -1;
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = gBattleAnimArgs[4];
gTasks[taskId].data[2] = 0;
gTasks[taskId].data[3] = gBattleAnimArgs[0];
gTasks[taskId].data[4] = gBattleAnimArgs[1];
gTasks[taskId].data[5] = v2;
gTasks[taskId].data[6] = v1;
gTasks[taskId].data[7] = gBattleAnimArgs[2];
gTasks[taskId].data[8] = gBattleAnimArgs[3];
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(gBattleAnimArgs[0], gBattleAnimArgs[1]));
2021-01-23 00:40:46 -05:00
gTasks[taskId].func = AnimTask_AlphaFadeIn_Step;
2018-10-07 12:32:20 +02:00
}
2021-01-23 00:40:46 -05:00
static void AnimTask_AlphaFadeIn_Step(u8 taskId)
2018-10-07 12:32:20 +02:00
{
struct Task *task = &gTasks[taskId];
if (++task->data[0] > task->data[1])
{
task->data[0] = 0;
if (++task->data[2] & 1)
{
if (task->data[3] != task->data[7])
task->data[3] += task->data[5];
}
else
{
if (task->data[4] != task->data[8])
task->data[4] += task->data[6];
}
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(task->data[3], task->data[4]));
if (task->data[3] == task->data[7] && task->data[4] == task->data[8])
{
DestroyAnimVisualTask(taskId);
return;
}
}
}
// Linearly blends a mon's sprite colors with a target color with increasing
// strength, and then blends out to the original color.
// arg 0: anim bank
// arg 1: blend color
// arg 2: target blend coefficient
// arg 3: initial delay
// arg 4: number of times to blend in and out
void AnimTask_BlendMonInAndOut(u8 task)
{
u8 spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
2021-02-20 00:30:37 -05:00
if (spriteId == SPRITE_NONE)
2018-10-07 12:32:20 +02:00
{
DestroyAnimVisualTask(task);
return;
}
gTasks[task].data[0] = (gSprites[spriteId].oam.paletteNum * 0x10) + 0x101;
2020-02-20 00:04:42 -05:00
AnimTask_BlendPalInAndOutSetup(&gTasks[task]);
2018-10-07 12:32:20 +02:00
}
2020-02-20 00:04:42 -05:00
static void AnimTask_BlendPalInAndOutSetup(struct Task *task)
2018-10-07 12:32:20 +02:00
{
task->data[1] = gBattleAnimArgs[1];
task->data[2] = 0;
task->data[3] = gBattleAnimArgs[2];
task->data[4] = 0;
task->data[5] = gBattleAnimArgs[3];
task->data[6] = 0;
task->data[7] = gBattleAnimArgs[4];
2021-01-23 00:40:46 -05:00
task->func = AnimTask_BlendMonInAndOut_Step;
2018-10-07 12:32:20 +02:00
}
2021-01-23 00:40:46 -05:00
static void AnimTask_BlendMonInAndOut_Step(u8 taskId)
2018-10-07 12:32:20 +02:00
{
struct Task *task = &gTasks[taskId];
if (++task->data[4] >= task->data[5])
{
task->data[4] = 0;
if (!task->data[6])
{
task->data[2]++;
BlendPalette(task->data[0], 15, task->data[2], task->data[1]);
if (task->data[2] == task->data[3])
task->data[6] = 1;
}
else
{
task->data[2]--;
BlendPalette(task->data[0], 15, task->data[2], task->data[1]);
if (!task->data[2])
{
if (--task->data[7])
{
task->data[4] = 0;
task->data[6] = 0;
}
else
{
DestroyAnimVisualTask(taskId);
return;
}
}
}
}
}
2020-02-20 00:04:42 -05:00
// See AnimTask_BlendMonInAndOut. Same, but ANIM_TAG_* instead of mon
void AnimTask_BlendPalInAndOutByTag(u8 task)
2018-10-07 12:32:20 +02:00
{
u8 palette = IndexOfSpritePaletteTag(gBattleAnimArgs[0]);
if (palette == 0xff)
{
DestroyAnimVisualTask(task);
return;
}
gTasks[task].data[0] = (palette * 0x10) + 0x101;
2020-02-20 00:04:42 -05:00
AnimTask_BlendPalInAndOutSetup(&gTasks[task]);
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
void PrepareAffineAnimInTaskData(struct Task *task, u8 spriteId, const union AffineAnimCmd *affineAnimCmds)
2018-10-07 12:32:20 +02:00
{
task->data[7] = 0;
task->data[8] = 0;
task->data[9] = 0;
2018-12-17 22:08:08 -06:00
task->data[15] = spriteId;
2018-10-07 12:32:20 +02:00
task->data[10] = 0x100;
task->data[11] = 0x100;
task->data[12] = 0;
2018-12-17 22:08:08 -06:00
StorePointerInVars(&task->data[13], &task->data[14], affineAnimCmds);
PrepareBattlerSpriteForRotScale(spriteId, ST_OAM_OBJ_NORMAL);
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
bool8 RunAffineAnimFromTaskData(struct Task *task)
2018-10-07 12:32:20 +02:00
{
gAnimTaskAffineAnim = &((union AffineAnimCmd *)LoadPointerFromVars(task->data[13], task->data[14]))[task->data[7]];
2018-12-17 22:08:08 -06:00
switch (gAnimTaskAffineAnim->type)
2018-10-07 12:32:20 +02:00
{
default:
2018-12-17 22:08:08 -06:00
if (!gAnimTaskAffineAnim->frame.duration)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
task->data[10] = gAnimTaskAffineAnim->frame.xScale;
task->data[11] = gAnimTaskAffineAnim->frame.yScale;
task->data[12] = gAnimTaskAffineAnim->frame.rotation;
2018-10-07 12:32:20 +02:00
task->data[7]++;
2018-12-17 22:08:08 -06:00
gAnimTaskAffineAnim++;
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
task->data[10] += gAnimTaskAffineAnim->frame.xScale;
task->data[11] += gAnimTaskAffineAnim->frame.yScale;
task->data[12] += gAnimTaskAffineAnim->frame.rotation;
SetSpriteRotScale(task->data[15], task->data[10], task->data[11], task->data[12]);
SetBattlerSpriteYOffsetFromYScale(task->data[15]);
if (++task->data[8] >= gAnimTaskAffineAnim->frame.duration)
2018-10-07 12:32:20 +02:00
{
task->data[8] = 0;
task->data[7]++;
}
break;
case AFFINEANIMCMDTYPE_JUMP:
2018-12-17 22:08:08 -06:00
task->data[7] = gAnimTaskAffineAnim->jump.target;
2018-10-07 12:32:20 +02:00
break;
case AFFINEANIMCMDTYPE_LOOP:
2018-12-17 22:08:08 -06:00
if (gAnimTaskAffineAnim->loop.count)
2018-10-07 12:32:20 +02:00
{
if (task->data[9])
{
if (!--task->data[9])
{
task->data[7]++;
break;
}
}
else
{
2018-12-17 22:08:08 -06:00
task->data[9] = gAnimTaskAffineAnim->loop.count;
2018-10-07 12:32:20 +02:00
}
if (!task->data[7])
{
break;
}
for (;;)
{
task->data[7]--;
2018-12-17 22:08:08 -06:00
gAnimTaskAffineAnim--;
if (gAnimTaskAffineAnim->type == AFFINEANIMCMDTYPE_LOOP)
2018-10-07 12:32:20 +02:00
{
task->data[7]++;
return TRUE;
}
if (!task->data[7])
return TRUE;
}
}
task->data[7]++;
break;
case AFFINEANIMCMDTYPE_END:
2021-07-07 09:11:52 -04:00
gSprites[task->data[15]].y2 = 0;
2018-12-17 22:08:08 -06:00
ResetSpriteRotScale(task->data[15]);
2018-10-07 12:32:20 +02:00
return FALSE;
}
return TRUE;
}
2018-12-17 22:08:08 -06:00
// Sets the sprite's y offset equal to the y displacement caused by the
// matrix's scale in the y dimension.
void SetBattlerSpriteYOffsetFromYScale(u8 spriteId)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
int var = 64 - GetBattlerYDeltaFromSpriteId(spriteId) * 2;
2018-10-07 12:32:20 +02:00
u16 matrix = gSprites[spriteId].oam.matrixNum;
2021-03-19 18:17:43 -04:00
int var2 = SAFE_DIV(var << 8, gOamMatrices[matrix].d);
2018-10-07 12:32:20 +02:00
2018-12-17 22:08:08 -06:00
if (var2 > 128)
var2 = 128;
2021-07-07 09:11:52 -04:00
gSprites[spriteId].y2 = (var - var2) / 2;
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
// Sets the sprite's y offset equal to the y displacement caused by another sprite
// matrix's scale in the y dimension.
void SetBattlerSpriteYOffsetFromOtherYScale(u8 spriteId, u8 otherSpriteId)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
int var = 64 - GetBattlerYDeltaFromSpriteId(otherSpriteId) * 2;
2018-10-07 12:32:20 +02:00
u16 matrix = gSprites[spriteId].oam.matrixNum;
2021-03-19 18:17:43 -04:00
int var2 = SAFE_DIV(var << 8, gOamMatrices[matrix].d);
2018-10-07 12:32:20 +02:00
2018-12-17 22:08:08 -06:00
if (var2 > 128)
var2 = 128;
2021-07-07 09:11:52 -04:00
gSprites[spriteId].y2 = (var - var2) / 2;
2018-10-07 12:32:20 +02:00
}
2019-02-06 13:17:09 -06:00
static u16 GetBattlerYDeltaFromSpriteId(u8 spriteId)
2018-10-07 12:32:20 +02:00
{
struct BattleSpriteInfo *spriteInfo;
u8 battlerId = gSprites[spriteId].data[0];
u16 species;
u16 i;
for (i = 0; i < MAX_BATTLERS_COUNT; i++)
{
if (gBattlerSpriteIds[i] == spriteId)
{
if (IsContest())
{
2020-08-13 19:10:23 -04:00
species = gContestResources->moveAnim->species;
2018-10-07 12:32:20 +02:00
return gMonBackPicCoords[species].y_offset;
}
else
{
if (GetBattlerSide(i) == B_SIDE_PLAYER)
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[i]], MON_DATA_SPECIES);
else
species = spriteInfo[battlerId].transformSpecies;
if (species == SPECIES_CASTFORM)
return sCastformBackSpriteYCoords[gBattleMonForms[battlerId]];
else
return gMonBackPicCoords[species].y_offset;
}
else
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
species = GetMonData(&gEnemyParty[gBattlerPartyIndexes[i]], MON_DATA_SPECIES);
else
species = spriteInfo[battlerId].transformSpecies;
if (species == SPECIES_CASTFORM)
return sCastformElevations[gBattleMonForms[battlerId]];
else
return gMonFrontPicCoords[species].y_offset;
}
}
}
}
2018-12-17 22:08:08 -06:00
return 64;
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
void StorePointerInVars(s16 *lo, s16 *hi, const void *ptr)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
*lo = ((intptr_t) ptr) & 0xffff;
*hi = (((intptr_t) ptr) >> 16) & 0xffff;
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
void *LoadPointerFromVars(s16 lo, s16 hi)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
return (void *)((u16)lo | ((u16)hi << 16));
2018-10-07 12:32:20 +02:00
}
void PrepareEruptAnimTaskData(struct Task *task, u8 spriteId, s16 xScaleStart, s16 yScaleStart, s16 xScaleEnd, s16 yScaleEnd, u16 duration)
2018-10-07 12:32:20 +02:00
{
task->data[8] = duration;
task->data[15] = spriteId;
task->data[9] = xScaleStart;
task->data[10] = yScaleStart;
task->data[13] = xScaleEnd;
task->data[14] = yScaleEnd;
task->data[11] = (xScaleEnd - xScaleStart) / duration;
task->data[12] = (yScaleEnd - yScaleStart) / duration;
2018-10-07 12:32:20 +02:00
}
u8 UpdateEruptAnimTask(struct Task *task)
2018-10-07 12:32:20 +02:00
{
if (!task->data[8])
return 0;
if (--task->data[8] != 0)
{
task->data[9] += task->data[11];
task->data[10] += task->data[12];
}
else
{
task->data[9] = task->data[13];
task->data[10] = task->data[14];
}
2018-12-17 22:08:08 -06:00
SetSpriteRotScale(task->data[15], task->data[9], task->data[10], 0);
2018-10-07 12:32:20 +02:00
if (task->data[8])
2018-12-17 22:08:08 -06:00
SetBattlerSpriteYOffsetFromYScale(task->data[15]);
2018-10-07 12:32:20 +02:00
else
2021-07-07 09:11:52 -04:00
gSprites[task->data[15]].y2 = 0;
2018-10-07 12:32:20 +02:00
return task->data[8];
}
void AnimTask_GetFrustrationPowerLevel(u8 taskId)
{
u16 powerLevel;
if (gAnimFriendship <= 30)
powerLevel = 0;
else if (gAnimFriendship <= 100)
powerLevel = 1;
else if (gAnimFriendship <= 200)
powerLevel = 2;
else
powerLevel = 3;
2020-02-19 13:13:28 -05:00
gBattleAnimArgs[ARG_RET_ID] = powerLevel;
2018-10-07 12:32:20 +02:00
DestroyAnimVisualTask(taskId);
}
void sub_80A8174(u8 priority)
{
if (IsBattlerSpriteVisible(gBattleAnimTarget))
gSprites[gBattlerSpriteIds[gBattleAnimTarget]].oam.priority = priority;
if (IsBattlerSpriteVisible(gBattleAnimAttacker))
gSprites[gBattlerSpriteIds[gBattleAnimAttacker]].oam.priority = priority;
if (IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimTarget)))
gSprites[gBattlerSpriteIds[BATTLE_PARTNER(gBattleAnimTarget)]].oam.priority = priority;
if (IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimAttacker)))
gSprites[gBattlerSpriteIds[BATTLE_PARTNER(gBattleAnimAttacker)]].oam.priority = priority;
}
void sub_80A8278(void)
{
int i;
for (i = 0; i < gBattlersCount; i++)
{
if (IsBattlerSpriteVisible(i))
{
2018-12-17 22:08:08 -06:00
gSprites[gBattlerSpriteIds[i]].subpriority = GetBattlerSpriteSubpriority(i);
2018-10-07 12:32:20 +02:00
gSprites[gBattlerSpriteIds[i]].oam.priority = 2;
}
}
}
2018-12-17 22:08:08 -06:00
u8 GetBattlerSpriteSubpriority(u8 battlerId)
2018-10-07 12:32:20 +02:00
{
u8 position;
2018-12-17 22:08:08 -06:00
u8 subpriority;
2018-10-07 12:32:20 +02:00
if (IsContest())
{
if (battlerId == 2)
return 30;
else
return 40;
}
else
{
position = GetBattlerPosition(battlerId);
if (position == B_POSITION_PLAYER_LEFT)
2018-12-17 22:08:08 -06:00
subpriority = 30;
2018-10-07 12:32:20 +02:00
else if (position == B_POSITION_PLAYER_RIGHT)
2018-12-17 22:08:08 -06:00
subpriority = 20;
2018-10-07 12:32:20 +02:00
else if (position == B_POSITION_OPPONENT_LEFT)
2018-12-17 22:08:08 -06:00
subpriority = 40;
2018-10-07 12:32:20 +02:00
else
2018-12-17 22:08:08 -06:00
subpriority = 50;
2018-10-07 12:32:20 +02:00
}
2018-12-17 22:08:08 -06:00
return subpriority;
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
u8 GetBattlerSpriteBGPriority(u8 battlerId)
2018-10-07 12:32:20 +02:00
{
u8 position = GetBattlerPosition(battlerId);
if (IsContest())
return 2;
else if (position == B_POSITION_PLAYER_LEFT || position == B_POSITION_OPPONENT_RIGHT)
return GetAnimBgAttribute(2, BG_ANIM_PRIORITY);
else
return GetAnimBgAttribute(1, BG_ANIM_PRIORITY);
}
2018-12-19 21:13:26 -06:00
u8 GetBattlerSpriteBGPriorityRank(u8 battlerId)
2018-10-07 12:32:20 +02:00
{
if (!IsContest())
{
u8 position = GetBattlerPosition(battlerId);
if (position == B_POSITION_PLAYER_LEFT || position == B_POSITION_OPPONENT_RIGHT)
return 2;
else
return 1;
}
return 1;
}
2021-01-22 23:22:37 -05:00
// Create pokemon sprite to be used for a move animation effect (e.g. Role Play / Snatch)
u8 CreateAdditionalMonSpriteForMoveAnim(u16 species, bool8 isBackpic, u8 id, s16 x, s16 y, u8 subpriority, u32 personality, u32 trainerId, u32 battlerId, bool32 ignoreDeoxysForm)
2018-10-07 12:32:20 +02:00
{
u8 spriteId;
2021-01-22 23:22:37 -05:00
u16 sheet = LoadSpriteSheet(&sSpriteSheet_MoveEffectMons[id]);
u16 palette = AllocSpritePalette(sSpriteTemplate_MoveEffectMons[id].paletteTag);
2018-10-07 12:32:20 +02:00
2020-02-20 00:04:42 -05:00
if (gMonSpritesGfxPtr != NULL && gMonSpritesGfxPtr->buffer == NULL)
gMonSpritesGfxPtr->buffer = AllocZeroed(0x2000);
2018-10-07 12:32:20 +02:00
if (!isBackpic)
{
2019-07-25 18:56:08 +02:00
LoadCompressedPalette(GetMonSpritePalFromSpeciesAndPersonality(species, trainerId, personality), (palette * 0x10) + 0x100, 0x20);
2021-01-22 23:22:37 -05:00
if (ignoreDeoxysForm == TRUE || ShouldIgnoreDeoxysForm(5, battlerId) == TRUE || gBattleSpritesDataPtr->battlerData[battlerId].transformSpecies != 0)
2018-10-07 12:32:20 +02:00
LoadSpecialPokePic_DontHandleDeoxys(&gMonFrontPicTable[species],
2020-02-20 00:04:42 -05:00
gMonSpritesGfxPtr->buffer,
2018-10-07 12:32:20 +02:00
species,
personality,
TRUE);
else
LoadSpecialPokePic_2(&gMonFrontPicTable[species],
2020-02-20 00:04:42 -05:00
gMonSpritesGfxPtr->buffer,
2018-10-07 12:32:20 +02:00
species,
personality,
TRUE);
}
else
{
2019-07-25 18:56:08 +02:00
LoadCompressedPalette(GetMonSpritePalFromSpeciesAndPersonality(species, trainerId, personality), (palette * 0x10) + 0x100, 0x20);
2021-01-22 23:22:37 -05:00
if (ignoreDeoxysForm == TRUE || ShouldIgnoreDeoxysForm(5, battlerId) == TRUE || gBattleSpritesDataPtr->battlerData[battlerId].transformSpecies != 0)
2018-10-07 12:32:20 +02:00
LoadSpecialPokePic_DontHandleDeoxys(&gMonBackPicTable[species],
2020-02-20 00:04:42 -05:00
gMonSpritesGfxPtr->buffer,
2018-10-07 12:32:20 +02:00
species,
personality,
FALSE);
else
LoadSpecialPokePic_2(&gMonBackPicTable[species],
2020-02-20 00:04:42 -05:00
gMonSpritesGfxPtr->buffer,
2018-10-07 12:32:20 +02:00
species,
personality,
FALSE);
}
2021-03-29 09:38:19 -04:00
RequestDma3Copy(gMonSpritesGfxPtr->buffer, (void *)(OBJ_VRAM0 + (sheet * 0x20)), MON_PIC_SIZE, 1);
2020-02-20 00:04:42 -05:00
FREE_AND_SET_NULL(gMonSpritesGfxPtr->buffer);
2018-10-07 12:32:20 +02:00
if (!isBackpic)
2021-01-22 23:22:37 -05:00
spriteId = CreateSprite(&sSpriteTemplate_MoveEffectMons[id], x, y + gMonFrontPicCoords[species].y_offset, subpriority);
2018-10-07 12:32:20 +02:00
else
2021-01-22 23:22:37 -05:00
spriteId = CreateSprite(&sSpriteTemplate_MoveEffectMons[id], x, y + gMonBackPicCoords[species].y_offset, subpriority);
2018-10-07 12:32:20 +02:00
if (IsContest())
{
gSprites[spriteId].affineAnims = gAffineAnims_BattleSpriteContest;
StartSpriteAffineAnim(&gSprites[spriteId], BATTLER_AFFINE_NORMAL);
2018-10-07 12:32:20 +02:00
}
return spriteId;
}
2018-12-30 10:58:42 -06:00
void DestroySpriteAndFreeResources_(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
DestroySpriteAndFreeResources(sprite);
}
2018-12-30 10:58:42 -06:00
s16 GetBattlerSpriteCoordAttr(u8 battlerId, u8 attr)
2018-10-07 12:32:20 +02:00
{
u16 species;
u32 personality;
u16 letter;
2018-12-30 10:58:42 -06:00
u16 unownSpecies;
2018-10-07 12:32:20 +02:00
int ret;
const struct MonCoords *coords;
struct BattleSpriteInfo *spriteInfo;
if (IsContest())
{
2020-08-13 19:10:23 -04:00
if (gContestResources->moveAnim->hasTargetAnim)
2018-10-07 12:32:20 +02:00
{
2020-08-13 19:10:23 -04:00
species = gContestResources->moveAnim->targetSpecies;
personality = gContestResources->moveAnim->targetPersonality;
2018-10-07 12:32:20 +02:00
}
else
{
2020-08-13 19:10:23 -04:00
species = gContestResources->moveAnim->species;
personality = gContestResources->moveAnim->personality;
2018-10-07 12:32:20 +02:00
}
if (species == SPECIES_UNOWN)
{
letter = GET_UNOWN_LETTER(personality);
if (!letter)
2018-12-30 10:58:42 -06:00
unownSpecies = SPECIES_UNOWN;
2018-10-07 12:32:20 +02:00
else
2018-12-30 10:58:42 -06:00
unownSpecies = letter + SPECIES_UNOWN_B - 1;
coords = &gMonBackPicCoords[unownSpecies];
2018-10-07 12:32:20 +02:00
}
else if (species == SPECIES_CASTFORM)
{
coords = &gCastformFrontSpriteCoords[gBattleMonForms[battlerId]];
}
else if (species <= SPECIES_EGG)
{
coords = &gMonBackPicCoords[species];
}
else
{
coords = &gMonBackPicCoords[0];
}
}
else
{
if (GetBattlerSide(battlerId) == B_SIDE_PLAYER)
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
{
species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
personality = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_PERSONALITY);
}
else
{
species = spriteInfo[battlerId].transformSpecies;
personality = gTransformedPersonalities[battlerId];
}
2018-12-30 10:58:42 -06:00
2018-10-07 12:32:20 +02:00
if (species == SPECIES_UNOWN)
{
letter = GET_UNOWN_LETTER(personality);
if (!letter)
2018-12-30 10:58:42 -06:00
unownSpecies = SPECIES_UNOWN;
2018-10-07 12:32:20 +02:00
else
2018-12-30 10:58:42 -06:00
unownSpecies = letter + SPECIES_UNOWN_B - 1;
coords = &gMonBackPicCoords[unownSpecies];
2018-10-07 12:32:20 +02:00
}
2018-12-30 10:58:42 -06:00
else if (species > NUM_SPECIES)
2018-10-07 12:32:20 +02:00
{
coords = &gMonBackPicCoords[0];
}
else
{
coords = &gMonBackPicCoords[species];
}
}
else
{
spriteInfo = gBattleSpritesDataPtr->battlerData;
if (!spriteInfo[battlerId].transformSpecies)
{
species = GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
personality = GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_PERSONALITY);
}
else
{
species = spriteInfo[battlerId].transformSpecies;
personality = gTransformedPersonalities[battlerId];
}
2018-12-30 10:58:42 -06:00
2018-10-07 12:32:20 +02:00
if (species == SPECIES_UNOWN)
{
letter = GET_UNOWN_LETTER(personality);
if (!letter)
2018-12-30 10:58:42 -06:00
unownSpecies = SPECIES_UNOWN;
2018-10-07 12:32:20 +02:00
else
2018-12-30 10:58:42 -06:00
unownSpecies = letter + SPECIES_UNOWN_B - 1;
coords = &gMonFrontPicCoords[unownSpecies];
2018-10-07 12:32:20 +02:00
}
else if (species == SPECIES_CASTFORM)
{
coords = &gCastformFrontSpriteCoords[gBattleMonForms[battlerId]];
}
2018-12-30 10:58:42 -06:00
else if (species > NUM_SPECIES)
2018-10-07 12:32:20 +02:00
{
coords = &gMonFrontPicCoords[0];
}
else
{
coords = &gMonFrontPicCoords[species];
}
}
}
2018-12-30 10:58:42 -06:00
switch (attr)
2018-10-07 12:32:20 +02:00
{
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_HEIGHT:
return (coords->size & 0xf) * 8;
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_WIDTH:
return (coords->size >> 4) * 8;
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_LEFT:
return GetBattlerSpriteCoord(battlerId, BATTLER_COORD_X_2) - ((coords->size >> 4) * 4);
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_RIGHT:
return GetBattlerSpriteCoord(battlerId, BATTLER_COORD_X_2) + ((coords->size >> 4) * 4);
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_TOP:
return GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y_PIC_OFFSET) - ((coords->size & 0xf) * 4);
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_BOTTOM:
return GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y_PIC_OFFSET) + ((coords->size & 0xf) * 4);
2018-12-30 10:58:42 -06:00
case BATTLER_COORD_ATTR_RAW_BOTTOM:
ret = GetBattlerSpriteCoord(battlerId, BATTLER_COORD_Y) + 31;
2018-10-07 12:32:20 +02:00
return ret - coords->y_offset;
default:
return 0;
}
}
2018-12-19 21:13:26 -06:00
void SetAverageBattlerPositions(u8 battlerId, bool8 respectMonPicOffsets, s16 *x, s16 *y)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
u8 xCoordType, yCoordType;
s16 battlerX, battlerY;
s16 partnerX, partnerY;
2018-10-07 12:32:20 +02:00
2018-12-19 21:13:26 -06:00
if (!respectMonPicOffsets)
2018-10-07 12:32:20 +02:00
{
2018-12-19 21:13:26 -06:00
xCoordType = BATTLER_COORD_X;
yCoordType = BATTLER_COORD_Y;
2018-10-07 12:32:20 +02:00
}
else
{
2018-12-19 21:13:26 -06:00
xCoordType = BATTLER_COORD_X_2;
yCoordType = BATTLER_COORD_Y_PIC_OFFSET;
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
battlerX = GetBattlerSpriteCoord(battlerId, xCoordType);
battlerY = GetBattlerSpriteCoord(battlerId, yCoordType);
2018-10-07 12:32:20 +02:00
if (IsDoubleBattle() && !IsContest())
{
2018-12-19 21:13:26 -06:00
partnerX = GetBattlerSpriteCoord(BATTLE_PARTNER(battlerId), xCoordType);
partnerY = GetBattlerSpriteCoord(BATTLE_PARTNER(battlerId), yCoordType);
2018-10-07 12:32:20 +02:00
}
else
{
2018-12-19 21:13:26 -06:00
partnerX = battlerX;
partnerY = battlerY;
2018-10-07 12:32:20 +02:00
}
2018-12-19 21:13:26 -06:00
*x = (battlerX + partnerX) / 2;
*y = (battlerY + partnerY) / 2;
2018-10-07 12:32:20 +02:00
}
2021-02-27 17:41:30 -06:00
u8 CreateInvisibleSpriteCopy(int battlerId, u8 spriteId, int species)
2018-10-07 12:32:20 +02:00
{
u8 newSpriteId = CreateInvisibleSpriteWithCallback(SpriteCallbackDummy);
gSprites[newSpriteId] = gSprites[spriteId];
gSprites[newSpriteId].usingSheet = TRUE;
gSprites[newSpriteId].oam.priority = 0;
gSprites[newSpriteId].oam.objMode = ST_OAM_OBJ_WINDOW;
2018-10-07 12:32:20 +02:00
gSprites[newSpriteId].oam.tileNum = gSprites[spriteId].oam.tileNum;
gSprites[newSpriteId].callback = SpriteCallbackDummy;
return newSpriteId;
}
2021-04-24 18:26:25 -04:00
void AnimTranslateLinearAndFlicker_Flipped(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
SetSpriteCoordsToAnimAttackerCoords(sprite);
2018-10-07 12:32:20 +02:00
if (GetBattlerSide(gBattleAnimAttacker))
{
2021-07-07 09:11:52 -04:00
sprite->x -= gBattleAnimArgs[0];
2018-10-07 12:32:20 +02:00
gBattleAnimArgs[3] = -gBattleAnimArgs[3];
sprite->hFlip = TRUE;
}
else
{
2021-07-07 09:11:52 -04:00
sprite->x += gBattleAnimArgs[0];
2018-10-07 12:32:20 +02:00
}
2021-07-07 09:11:52 -04:00
sprite->y += gBattleAnimArgs[1];
2018-10-07 12:32:20 +02:00
sprite->data[0] = gBattleAnimArgs[2];
sprite->data[1] = gBattleAnimArgs[3];
sprite->data[3] = gBattleAnimArgs[4];
sprite->data[5] = gBattleAnimArgs[5];
2018-12-17 22:08:08 -06:00
StoreSpriteCallbackInData6(sprite, DestroySpriteAndMatrix);
2019-02-06 13:17:09 -06:00
sprite->callback = TranslateSpriteLinearAndFlicker;
2018-10-07 12:32:20 +02:00
}
2021-02-27 17:41:30 -06:00
// Used by three different unused battle anim sprite templates.
2021-04-24 18:26:25 -04:00
void AnimTranslateLinearAndFlicker(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
{
2021-07-07 09:11:52 -04:00
sprite->x -= gBattleAnimArgs[0];
2018-10-07 12:32:20 +02:00
gBattleAnimArgs[3] *= -1;
}
else
{
2021-07-07 09:11:52 -04:00
sprite->x += gBattleAnimArgs[0];
2018-10-07 12:32:20 +02:00
}
2021-07-07 09:11:52 -04:00
sprite->y += gBattleAnimArgs[1];
2018-10-07 12:32:20 +02:00
sprite->data[0] = gBattleAnimArgs[2];
sprite->data[1] = gBattleAnimArgs[3];
sprite->data[3] = gBattleAnimArgs[4];
sprite->data[5] = gBattleAnimArgs[5];
StartSpriteAnim(sprite, gBattleAnimArgs[6]);
2018-12-17 22:08:08 -06:00
StoreSpriteCallbackInData6(sprite, DestroySpriteAndMatrix);
2019-02-06 13:17:09 -06:00
sprite->callback = TranslateSpriteLinearAndFlicker;
2018-10-07 12:32:20 +02:00
}
2020-02-14 18:11:04 -05:00
// Used by Detect/Disable
void AnimSpinningSparkle(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
SetSpriteCoordsToAnimAttackerCoords(sprite);
2018-10-07 12:32:20 +02:00
if (GetBattlerSide(gBattleAnimAttacker))
2021-07-07 09:11:52 -04:00
sprite->x -= gBattleAnimArgs[0];
2018-10-07 12:32:20 +02:00
else
2021-07-07 09:11:52 -04:00
sprite->x += gBattleAnimArgs[0];
sprite->y += gBattleAnimArgs[1];
2018-12-17 22:08:08 -06:00
sprite->callback = RunStoredCallbackWhenAnimEnds;
2018-10-07 12:32:20 +02:00
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
}
2020-02-16 17:49:24 -05:00
// Slides attacker to right and back with a cloned trace of the specified color
// arg0: Trace palette blend color
// arg1: Trace palette blend coeff
void AnimTask_AttackerPunchWithTrace(u8 taskId)
2018-10-07 12:32:20 +02:00
{
u16 src;
u16 dest;
struct Task *task = &gTasks[taskId];
task->data[0] = GetAnimBattlerSpriteId(ANIM_ATTACKER);
task->data[1] = ((GetBattlerSide(gBattleAnimAttacker)) != B_SIDE_PLAYER) ? -8 : 8;
task->data[2] = 0;
task->data[3] = 0;
2021-07-07 09:11:52 -04:00
gSprites[task->data[0]].x2 -= task->data[0];
2021-01-22 20:03:21 -05:00
task->data[4] = AllocSpritePalette(ANIM_TAG_BENT_SPOON);
2018-10-07 12:32:20 +02:00
task->data[5] = 0;
dest = (task->data[4] + 0x10) * 0x10;
src = (gSprites[task->data[0]].oam.paletteNum + 0x10) * 0x10;
2018-12-17 22:08:08 -06:00
task->data[6] = GetBattlerSpriteSubpriority(gBattleAnimAttacker);
2018-10-07 12:32:20 +02:00
if (task->data[6] == 20 || task->data[6] == 40)
task->data[6] = 2;
else
task->data[6] = 3;
CpuCopy32(&gPlttBufferUnfaded[src], &gPlttBufferFaded[dest], 0x20);
BlendPalette(dest, 16, gBattleAnimArgs[1], gBattleAnimArgs[0]);
2021-01-23 00:40:46 -05:00
task->func = AnimTask_AttackerPunchWithTrace_Step;
2018-10-07 12:32:20 +02:00
}
2021-01-23 00:40:46 -05:00
static void AnimTask_AttackerPunchWithTrace_Step(u8 taskId)
2018-10-07 12:32:20 +02:00
{
struct Task *task = &gTasks[taskId];
switch (task->data[2])
{
case 0:
sub_80A8D78(task, taskId);
2021-07-07 09:11:52 -04:00
gSprites[task->data[0]].x2 += task->data[1];
2018-10-07 12:32:20 +02:00
if (++task->data[3] == 5)
{
task->data[3]--;
task->data[2]++;
}
break;
case 1:
sub_80A8D78(task, taskId);
2021-07-07 09:11:52 -04:00
gSprites[task->data[0]].x2 -= task->data[1];
2018-10-07 12:32:20 +02:00
if (--task->data[3] == 0)
{
2021-07-07 09:11:52 -04:00
gSprites[task->data[0]].x2 = 0;
2018-10-07 12:32:20 +02:00
task->data[2]++;
}
break;
case 2:
if (!task->data[5])
{
FreeSpritePaletteByTag(ANIM_TAG_BENT_SPOON);
DestroyAnimVisualTask(taskId);
}
break;
}
}
2019-02-06 13:17:09 -06:00
static void sub_80A8D78(struct Task *task, u8 taskId)
2018-10-07 12:32:20 +02:00
{
2018-12-17 22:08:08 -06:00
s16 spriteId = CloneBattlerSpriteWithBlend(0);
2018-10-07 12:32:20 +02:00
if (spriteId >= 0)
{
gSprites[spriteId].oam.priority = task->data[6];
gSprites[spriteId].oam.paletteNum = task->data[4];
gSprites[spriteId].data[0] = 8;
gSprites[spriteId].data[1] = taskId;
gSprites[spriteId].data[2] = spriteId;
2021-07-07 09:11:52 -04:00
gSprites[spriteId].x2 = gSprites[task->data[0]].x2;
2018-10-07 12:32:20 +02:00
gSprites[spriteId].callback = sub_80A8DFC;
task->data[5]++;
}
}
2019-02-06 13:17:09 -06:00
static void sub_80A8DFC(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
if (--sprite->data[0] == 0)
{
gTasks[sprite->data[1]].data[5]--;
obj_delete_but_dont_free_vram(sprite);
}
}
2020-02-19 20:22:53 -05:00
void AnimWeatherBallUp(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
2021-07-07 09:11:52 -04:00
sprite->x = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_X_2);
sprite->y = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_Y_PIC_OFFSET);
2018-10-07 12:32:20 +02:00
if (!GetBattlerSide(gBattleAnimAttacker))
sprite->data[0] = 5;
else
sprite->data[0] = -10;
sprite->data[1] = -40;
2021-01-23 00:40:46 -05:00
sprite->callback = AnimWeatherBallUp_Step;
2018-10-07 12:32:20 +02:00
}
2021-01-23 00:40:46 -05:00
static void AnimWeatherBallUp_Step(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
sprite->data[2] += sprite->data[0];
sprite->data[3] += sprite->data[1];
2021-07-07 09:11:52 -04:00
sprite->x2 = sprite->data[2] / 10;
sprite->y2 = sprite->data[3] / 10;
2018-10-07 12:32:20 +02:00
if (sprite->data[1] < -20)
sprite->data[1]++;
2021-07-07 09:11:52 -04:00
if (sprite->y + sprite->y2 < -32)
2018-10-07 12:32:20 +02:00
DestroyAnimSprite(sprite);
}
2020-02-19 20:22:53 -05:00
void AnimWeatherBallDown(struct Sprite *sprite)
2018-10-07 12:32:20 +02:00
{
int x;
sprite->data[0] = gBattleAnimArgs[2];
2021-07-07 09:11:52 -04:00
sprite->data[2] = sprite->x + gBattleAnimArgs[4];
sprite->data[4] = sprite->y + gBattleAnimArgs[5];
2018-10-07 12:32:20 +02:00
if (!GetBattlerSide(gBattleAnimTarget))
{
x = (u16)gBattleAnimArgs[4] + 30;
2021-07-07 09:11:52 -04:00
sprite->x += x;
sprite->y = gBattleAnimArgs[5] - 20;
2018-10-07 12:32:20 +02:00
}
else
{
x = (u16)gBattleAnimArgs[4] - 30;
2021-07-07 09:11:52 -04:00
sprite->x += x;
sprite->y = gBattleAnimArgs[5] - 80;
2018-10-07 12:32:20 +02:00
}
2018-11-25 17:00:18 -06:00
sprite->callback = StartAnimLinearTranslation;
2018-10-07 12:32:20 +02:00
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
}