pokeemerald/src/battle_anim_normal.c

1107 lines
32 KiB
C
Raw Normal View History

2018-10-25 23:46:23 +02:00
#include "global.h"
#include "battle_anim.h"
2018-12-01 00:29:55 +01:00
#include "palette.h"
#include "random.h"
#include "task.h"
#include "trig.h"
2018-10-25 23:46:23 +02:00
#include "constants/rgb.h"
2020-02-20 22:22:33 +01:00
static void AnimConfusionDuck(struct Sprite *);
static void AnimSimplePaletteBlend(struct Sprite *);
static void AnimSimplePaletteBlend_Step(struct Sprite *);
static void AnimComplexPaletteBlend(struct Sprite *);
static void AnimComplexPaletteBlend_Step1(struct Sprite *);
static void AnimComplexPaletteBlend_Step2(struct Sprite *);
static void sub_81159B4(struct Sprite *);
2020-02-20 06:04:42 +01:00
static void AnimShakeMonOrBattleTerrain(struct Sprite *);
2020-02-20 22:22:33 +01:00
static void AnimShakeMonOrBattleTerrain_Step(struct Sprite *);
static void AnimShakeMonOrBattleTerrain_UpdateCoordOffsetEnabled(void);
2020-02-22 00:22:18 +01:00
static void AnimHitSplatPersistent(struct Sprite *);
static void AnimHitSplatHandleInvert(struct Sprite *);
2020-02-20 22:22:33 +01:00
static void AnimConfusionDuck_Step(struct Sprite *);
2020-02-20 06:04:42 +01:00
static void BlendColorCycle(u8, u8, u8);
static void AnimTask_BlendColorCycleLoop(u8);
static void BlendColorCycleExclude(u8, u8, u8);
static void AnimTask_BlendColorCycleExcludeLoop(u8);
static void BlendColorCycleByTag(u8, u8, u8);
static void AnimTask_BlendColorCycleByTagLoop(u8);
2020-02-20 22:22:33 +01:00
static void AnimTask_FlashAnimTagWithColor_Step1(u8);
static void AnimTask_FlashAnimTagWithColor_Step2(u8);
static void AnimTask_ShakeBattleTerrain_Step(u8);
2019-07-06 05:22:18 +02:00
static void AnimMovePowerSwapGuardSwap(struct Sprite *);
2018-12-01 00:29:55 +01:00
2020-02-21 18:38:20 +01:00
static const union AnimCmd sAnim_ConfusionDuck_0[] =
2018-10-25 23:46:23 +02:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(4, 8),
ANIMCMD_FRAME(0, 8, .hFlip = TRUE),
ANIMCMD_FRAME(8, 8),
ANIMCMD_JUMP(0),
};
2020-02-21 18:38:20 +01:00
static const union AnimCmd sAnim_ConfusionDuck_1[] =
2018-10-25 23:46:23 +02:00
{
ANIMCMD_FRAME(0, 8, .hFlip = TRUE),
ANIMCMD_FRAME(4, 8),
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(8, 8),
ANIMCMD_JUMP(0),
};
2020-02-21 18:38:20 +01:00
static const union AnimCmd *const sAnims_ConfusionDuck[] =
2018-10-25 23:46:23 +02:00
{
2020-02-21 18:38:20 +01:00
sAnim_ConfusionDuck_0,
sAnim_ConfusionDuck_1,
2018-10-25 23:46:23 +02:00
};
const struct SpriteTemplate gConfusionDuckSpriteTemplate =
{
.tileTag = ANIM_TAG_DUCK,
.paletteTag = ANIM_TAG_DUCK,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineOff_ObjNormal_16x16,
2020-02-21 18:38:20 +01:00
.anims = sAnims_ConfusionDuck,
2018-10-25 23:46:23 +02:00
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2018-12-01 00:29:55 +01:00
.callback = AnimConfusionDuck,
2018-10-25 23:46:23 +02:00
};
const struct SpriteTemplate gSimplePaletteBlendSpriteTemplate =
{
.tileTag = 0,
.paletteTag = 0,
.oam = &gDummyOamData,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2018-12-01 00:29:55 +01:00
.callback = AnimSimplePaletteBlend,
2018-10-25 23:46:23 +02:00
};
const struct SpriteTemplate gComplexPaletteBlendSpriteTemplate =
{
.tileTag = 0,
.paletteTag = 0,
.oam = &gDummyOamData,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2020-02-20 22:22:33 +01:00
.callback = AnimComplexPaletteBlend,
2018-10-25 23:46:23 +02:00
};
2019-07-06 05:22:18 +02:00
static const union AnimCmd sPowerSwapGuardSwapFrame0[] =
{
ANIMCMD_FRAME(0, 0),
ANIMCMD_END
};
static const union AnimCmd sPowerSwapGuardSwapFrame1[] =
{
ANIMCMD_FRAME(4, 0),
ANIMCMD_END
};
static const union AnimCmd sPowerSwapGuardSwapFrame2[] =
{
ANIMCMD_FRAME(8, 0),
ANIMCMD_END
};
static const union AnimCmd sPowerSwapGuardSwapFrame3[] =
{
ANIMCMD_FRAME(12, 0),
ANIMCMD_END
};
static const union AnimCmd sPowerSwapGuardSwapFrame4[] =
{
ANIMCMD_FRAME(16, 0),
ANIMCMD_END
};
static const union AnimCmd sPowerSwapGuardSwapFrame5[] =
{
ANIMCMD_FRAME(20, 0),
ANIMCMD_END
};
static const union AnimCmd * const sPowerSwapGuardSwapAnimTable[] =
{
sPowerSwapGuardSwapFrame0,
sPowerSwapGuardSwapFrame1,
sPowerSwapGuardSwapFrame2,
sPowerSwapGuardSwapFrame3,
sPowerSwapGuardSwapFrame4,
sPowerSwapGuardSwapFrame5
};
const struct SpriteTemplate gPowerSwapGuardSwapSpriteTemplate =
{
.tileTag = ANIM_TAG_COLORED_ORBS,
.paletteTag = ANIM_TAG_COLORED_ORBS,
2019-10-19 17:47:55 +02:00
.oam = &gOamData_AffineOff_ObjNormal_16x16,
2019-07-06 05:22:18 +02:00
.anims = sPowerSwapGuardSwapAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = AnimMovePowerSwapGuardSwap
};
2020-02-21 18:38:20 +01:00
static const union AnimCmd gUnknown_085972A4[] =
2018-10-25 23:46:23 +02:00
{
ANIMCMD_FRAME(0, 3),
ANIMCMD_FRAME(16, 3),
ANIMCMD_FRAME(32, 3),
ANIMCMD_FRAME(48, 3),
ANIMCMD_FRAME(64, 3),
ANIMCMD_JUMP(0),
};
2020-02-21 18:38:20 +01:00
static const union AnimCmd *const gUnknown_085972BC[] =
2018-10-25 23:46:23 +02:00
{
gUnknown_085972A4,
};
2020-02-21 18:38:20 +01:00
// Unused
2018-10-25 23:46:23 +02:00
const struct SpriteTemplate gUnknown_085972C0 =
{
.tileTag = ANIM_TAG_SPARKLE_4,
.paletteTag = ANIM_TAG_SPARKLE_4,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineOff_ObjNormal_32x32,
2018-10-25 23:46:23 +02:00
.anims = gUnknown_085972BC,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = sub_81159B4,
};
2020-02-20 06:04:42 +01:00
const struct SpriteTemplate gShakeMonOrTerrainSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = 0,
.paletteTag = 0,
.oam = &gDummyOamData,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2020-02-20 06:04:42 +01:00
.callback = AnimShakeMonOrBattleTerrain,
2018-10-25 23:46:23 +02:00
};
2020-02-21 18:38:20 +01:00
static const union AffineAnimCmd sAffineAnim_HitSplat_0[] =
2018-10-25 23:46:23 +02:00
{
AFFINEANIMCMD_FRAME(0x0, 0x0, 0, 8),
AFFINEANIMCMD_END,
};
2020-02-21 18:38:20 +01:00
static const union AffineAnimCmd sAffineAnim_HitSplat_1[] =
2018-10-25 23:46:23 +02:00
{
AFFINEANIMCMD_FRAME(0xD8, 0xD8, 0, 0),
AFFINEANIMCMD_FRAME(0x0, 0x0, 0, 8),
AFFINEANIMCMD_END,
};
2020-02-21 18:38:20 +01:00
static const union AffineAnimCmd sAffineAnim_HitSplat_2[] =
2018-10-25 23:46:23 +02:00
{
AFFINEANIMCMD_FRAME(0xB0, 0xB0, 0, 0),
AFFINEANIMCMD_FRAME(0x0, 0x0, 0, 8),
AFFINEANIMCMD_END,
};
2020-02-21 18:38:20 +01:00
static const union AffineAnimCmd sAffineAnim_HitSplat_3[] =
2018-10-25 23:46:23 +02:00
{
AFFINEANIMCMD_FRAME(0x80, 0x80, 0, 0),
AFFINEANIMCMD_FRAME(0x0, 0x0, 0, 8),
AFFINEANIMCMD_END,
};
2020-05-25 08:37:32 +02:00
const union AffineAnimCmd *const gAffineAnims_HitSplat[] =
2018-10-25 23:46:23 +02:00
{
2020-02-21 18:38:20 +01:00
sAffineAnim_HitSplat_0,
sAffineAnim_HitSplat_1,
sAffineAnim_HitSplat_2,
sAffineAnim_HitSplat_3,
2018-10-25 23:46:23 +02:00
};
const struct SpriteTemplate gBasicHitSplatSpriteTemplate =
{
.tileTag = ANIM_TAG_IMPACT,
.paletteTag = ANIM_TAG_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-22 00:22:18 +01:00
.callback = AnimHitSplatBasic,
2018-10-25 23:46:23 +02:00
};
2020-02-22 00:22:18 +01:00
const struct SpriteTemplate gHandleInvertHitSplatSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_IMPACT,
.paletteTag = ANIM_TAG_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-22 00:22:18 +01:00
.callback = AnimHitSplatHandleInvert,
2018-10-25 23:46:23 +02:00
};
2020-02-14 22:05:43 +01:00
const struct SpriteTemplate gWaterHitSplatSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_WATER_IMPACT,
.paletteTag = ANIM_TAG_WATER_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-22 00:22:18 +01:00
.callback = AnimHitSplatBasic,
2018-10-25 23:46:23 +02:00
};
2020-02-15 00:11:04 +01:00
const struct SpriteTemplate gRandomPosHitSplatSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_IMPACT,
.paletteTag = ANIM_TAG_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-15 00:11:04 +01:00
.callback = AnimHitSplatRandom,
2018-10-25 23:46:23 +02:00
};
2020-02-22 00:22:18 +01:00
const struct SpriteTemplate gMonEdgeHitSplatSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_IMPACT,
.paletteTag = ANIM_TAG_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-22 00:22:18 +01:00
.callback = AnimHitSplatOnMonEdge,
2018-10-25 23:46:23 +02:00
};
2020-02-15 00:11:04 +01:00
const struct SpriteTemplate gCrossImpactSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_CROSS_IMPACT,
.paletteTag = ANIM_TAG_CROSS_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineOff_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2020-02-20 22:22:33 +01:00
.callback = AnimCrossImpact,
2018-10-25 23:46:23 +02:00
};
2020-02-15 00:11:04 +01:00
const struct SpriteTemplate gFlashingHitSplatSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_IMPACT,
.paletteTag = ANIM_TAG_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjNormal_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-15 00:11:04 +01:00
.callback = AnimFlashingHitSplat,
2018-10-25 23:46:23 +02:00
};
2020-02-22 00:22:18 +01:00
const struct SpriteTemplate gPersistHitSplatSpriteTemplate =
2018-10-25 23:46:23 +02:00
{
.tileTag = ANIM_TAG_IMPACT,
.paletteTag = ANIM_TAG_IMPACT,
2019-10-15 01:26:47 +02:00
.oam = &gOamData_AffineNormal_ObjBlend_32x32,
2018-10-25 23:46:23 +02:00
.anims = gDummySpriteAnimTable,
.images = NULL,
2020-05-25 08:37:32 +02:00
.affineAnims = gAffineAnims_HitSplat,
2020-02-22 00:22:18 +01:00
.callback = AnimHitSplatPersistent,
2018-10-25 23:46:23 +02:00
};
2018-12-01 00:29:55 +01:00
2019-07-06 05:22:18 +02:00
static void AnimMovePowerSwapGuardSwapWait(struct Sprite* sprite)
{
if (TranslateAnimHorizontalArc(sprite))
DestroyAnimSprite(sprite);
}
// arg 0: initial x pixel offset
// arg 1: initial y pixel offset
// arg 2: orb type (0..5) - color and size
// arg 3: from user to target / target to user
// arg 4: wave period
// arg 5: wave amplitude
static void AnimMovePowerSwapGuardSwap(struct Sprite* sprite)
{
StartSpriteAnim(sprite, gBattleAnimArgs[2]);
if(gBattleAnimArgs[3] == 0)
{
InitSpritePosToAnimAttacker(sprite, TRUE);
sprite->data[2] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_X);
sprite->data[4] = GetBattlerSpriteCoord(gBattleAnimTarget, BATTLER_COORD_Y);
}
else
{
InitSpritePosToAnimTarget(sprite, TRUE);
sprite->data[2] = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_X);
sprite->data[4] = GetBattlerSpriteCoord(gBattleAnimAttacker, BATTLER_COORD_Y);
}
sprite->data[0] = gBattleAnimArgs[4];
sprite->data[5] = gBattleAnimArgs[5];
InitAnimArcTranslation(sprite);
sprite->callback = AnimMovePowerSwapGuardSwapWait;
}
2018-12-01 00:29:55 +01:00
// Moves a spinning duck around the mon's head.
// arg 0: initial x pixel offset
// arg 1: initial y pixel offset
// arg 2: initial wave offset
// arg 3: wave period (higher means faster wave)
// arg 4: duration
2020-02-20 22:22:33 +01:00
static void AnimConfusionDuck(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
sprite->pos1.x += gBattleAnimArgs[0];
sprite->pos1.y += gBattleAnimArgs[1];
sprite->data[0] = gBattleAnimArgs[2];
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
{
sprite->data[1] = -gBattleAnimArgs[3];
sprite->data[4] = 1;
}
else
{
sprite->data[1] = gBattleAnimArgs[3];
sprite->data[4] = 0;
StartSpriteAnim(sprite, 1);
}
sprite->data[3] = gBattleAnimArgs[4];
2020-02-20 22:22:33 +01:00
sprite->callback = AnimConfusionDuck_Step;
2018-12-01 00:29:55 +01:00
sprite->callback(sprite);
}
2020-02-20 22:22:33 +01:00
static void AnimConfusionDuck_Step(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
sprite->pos2.x = Cos(sprite->data[0], 30);
sprite->pos2.y = Sin(sprite->data[0], 10);
if ((u16)sprite->data[0] < 128)
sprite->oam.priority = 1;
else
sprite->oam.priority = 3;
sprite->data[0] = (sprite->data[0] + sprite->data[1]) & 0xFF;
if (++sprite->data[2] == sprite->data[3])
DestroyAnimSprite(sprite);
}
// Performs a simple color blend on a specified sprite.
// arg 0: palette selector
// arg 1: delay
// arg 2: start blend amount
// arg 3: end blend amount
// arg 4: blend color
2020-02-20 22:22:33 +01:00
static void AnimSimplePaletteBlend(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes = UnpackSelectedBattleAnimPalettes(gBattleAnimArgs[0]);
BeginNormalPaletteFade(selectedPalettes, gBattleAnimArgs[1], gBattleAnimArgs[2], gBattleAnimArgs[3], gBattleAnimArgs[4]);
2020-02-20 22:22:33 +01:00
sprite->invisible = TRUE;
sprite->callback = AnimSimplePaletteBlend_Step;
2018-12-01 00:29:55 +01:00
}
// Unpacks a bitfield and returns a bitmask of its selected palettes.
// Bits 0-6 of the selector parameter result in the following palettes being selected:
// 0: battle background palettes (BG palettes 1, 2, and 3)
// 1: gBattleAnimAttacker OBJ palette
// 2: gBattleAnimTarget OBJ palette
// 3: gBattleAnimAttacker partner OBJ palette
// 4: gBattleAnimTarget partner OBJ palette
// 5: BG palette 4
// 6: BG palette 5
u32 UnpackSelectedBattleAnimPalettes(s16 selector)
{
2019-02-06 20:17:09 +01:00
u8 battleBackground = selector & 1;
u8 attacker = (selector >> 1) & 1;
u8 target = (selector >> 2) & 1;
u8 attackerPartner = (selector >> 3) & 1;
u8 targetPartner = (selector >> 4) & 1;
2018-12-01 00:29:55 +01:00
u8 arg5 = (selector >> 5) & 1;
u8 arg6 = (selector >> 6) & 1;
2019-02-06 20:17:09 +01:00
return sub_80A75AC(battleBackground, attacker, target, attackerPartner, targetPartner, arg5, arg6);
2018-12-01 00:29:55 +01:00
}
2020-02-20 22:22:33 +01:00
static void AnimSimplePaletteBlend_Step(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
if (!gPaletteFade.active)
DestroyAnimSprite(sprite);
}
2020-02-20 22:22:33 +01:00
static void AnimComplexPaletteBlend(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes;
sprite->data[0] = gBattleAnimArgs[1];
sprite->data[1] = gBattleAnimArgs[1];
sprite->data[2] = gBattleAnimArgs[2];
sprite->data[3] = gBattleAnimArgs[3];
sprite->data[4] = gBattleAnimArgs[4];
sprite->data[5] = gBattleAnimArgs[5];
sprite->data[6] = gBattleAnimArgs[6];
sprite->data[7] = gBattleAnimArgs[0];
selectedPalettes = UnpackSelectedBattleAnimPalettes(sprite->data[7]);
BlendPalettes(selectedPalettes, gBattleAnimArgs[4], gBattleAnimArgs[3]);
2020-02-20 22:22:33 +01:00
sprite->invisible = TRUE;
sprite->callback = AnimComplexPaletteBlend_Step1;
2018-12-01 00:29:55 +01:00
}
2020-02-20 22:22:33 +01:00
static void AnimComplexPaletteBlend_Step1(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes;
if (sprite->data[0] > 0)
{
sprite->data[0]--;
return;
}
if (gPaletteFade.active)
return;
if (sprite->data[2] == 0)
{
2020-02-20 22:22:33 +01:00
sprite->callback = AnimComplexPaletteBlend_Step2;
2018-12-01 00:29:55 +01:00
return;
}
selectedPalettes = UnpackSelectedBattleAnimPalettes(sprite->data[7]);
if (sprite->data[1] & 0x100)
BlendPalettes(selectedPalettes, sprite->data[4], sprite->data[3]);
else
BlendPalettes(selectedPalettes, sprite->data[6], sprite->data[5]);
sprite->data[1] ^= 0x100;
sprite->data[0] = sprite->data[1] & 0xFF;
sprite->data[2]--;
}
2020-02-20 22:22:33 +01:00
static void AnimComplexPaletteBlend_Step2(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes;
if (!gPaletteFade.active)
{
selectedPalettes = UnpackSelectedBattleAnimPalettes(sprite->data[7]);
BlendPalettes(selectedPalettes, 0, 0);
DestroyAnimSprite(sprite);
}
}
2020-02-20 22:22:33 +01:00
static void sub_81159B4(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
sprite->pos1.x += gBattleAnimArgs[0];
sprite->pos1.y += gBattleAnimArgs[1];
sprite->data[0] = 0;
sprite->data[1] = 10;
sprite->data[2] = 8;
sprite->data[3] = 40;
sprite->data[4] = 112;
sprite->data[5] = 0;
2018-12-18 05:08:08 +01:00
StoreSpriteCallbackInData6(sprite, DestroySpriteAndMatrix);
sprite->callback = TranslateSpriteInGrowingCircleOverDuration;
2018-12-01 00:29:55 +01:00
sprite->callback(sprite);
}
2020-02-20 06:04:42 +01:00
// Task data for AnimTask_BlendColorCycle, AnimTask_BlendColorCycleExclude, and AnimTask_BlendColorCycleByTag
#define tPalSelector data[0] // AnimTask_BlendColorCycle
#define tPalTag data[0] // AnimTask_BlendColorCycleByTag
2020-02-19 19:13:28 +01:00
#define tDelay data[1]
#define tNumBlends data[2]
#define tInitialBlendY data[3]
#define tTargetBlendY data[4]
#define tBlendColor data[5]
2020-02-14 22:05:43 +01:00
#define tRestoreBlend data[8]
2020-02-20 06:04:42 +01:00
#define tPalSelectorHi data[9]
#define tPalSelectorLo data[10]
2020-02-14 22:05:43 +01:00
2020-02-29 13:59:41 +01:00
// Blends mon/screen to designated color or back alternately tNumBlends times
2020-02-14 22:05:43 +01:00
// Many uses of this task only set a tNumBlends of 2, which has the effect of blending to a color and back once
2020-02-16 23:49:24 +01:00
void AnimTask_BlendColorCycle(u8 taskId)
2020-02-14 22:05:43 +01:00
{
gTasks[taskId].tPalSelector = gBattleAnimArgs[0];
gTasks[taskId].tDelay = gBattleAnimArgs[1];
gTasks[taskId].tNumBlends = gBattleAnimArgs[2];
gTasks[taskId].tInitialBlendY = gBattleAnimArgs[3];
gTasks[taskId].tTargetBlendY = gBattleAnimArgs[4];
gTasks[taskId].tBlendColor = gBattleAnimArgs[5];
gTasks[taskId].tRestoreBlend = FALSE;
2020-02-20 06:04:42 +01:00
BlendColorCycle(taskId, 0, gTasks[taskId].tTargetBlendY);
gTasks[taskId].func = AnimTask_BlendColorCycleLoop;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
static void BlendColorCycle(u8 taskId, u8 startBlendAmount, u8 targetBlendAmount)
2018-12-01 00:29:55 +01:00
{
2020-02-14 22:05:43 +01:00
u32 selectedPalettes = UnpackSelectedBattleAnimPalettes(gTasks[taskId].tPalSelector);
2018-12-01 00:29:55 +01:00
BeginNormalPaletteFade(
selectedPalettes,
2020-02-14 22:05:43 +01:00
gTasks[taskId].tDelay,
startBlendAmount,
2018-12-01 00:29:55 +01:00
targetBlendAmount,
2020-02-14 22:05:43 +01:00
gTasks[taskId].tBlendColor);
2018-12-01 00:29:55 +01:00
2020-02-14 22:05:43 +01:00
gTasks[taskId].tNumBlends--;
gTasks[taskId].tRestoreBlend ^= 1;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
static void AnimTask_BlendColorCycleLoop(u8 taskId)
2018-12-01 00:29:55 +01:00
{
2020-02-14 22:05:43 +01:00
u8 startBlendAmount, targetBlendAmount;
2018-12-01 00:29:55 +01:00
if (!gPaletteFade.active)
{
2020-02-14 22:05:43 +01:00
if (gTasks[taskId].tNumBlends > 0)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
if (!gTasks[taskId].tRestoreBlend)
2018-12-01 00:29:55 +01:00
{
2020-02-14 22:05:43 +01:00
// Blend to designated color
startBlendAmount = gTasks[taskId].tInitialBlendY;
targetBlendAmount = gTasks[taskId].tTargetBlendY;
2018-12-01 00:29:55 +01:00
}
else
{
2020-02-14 22:05:43 +01:00
// Blend back to original color
startBlendAmount = gTasks[taskId].tTargetBlendY;
targetBlendAmount = gTasks[taskId].tInitialBlendY;
2018-12-01 00:29:55 +01:00
}
2020-02-14 22:05:43 +01:00
if (gTasks[taskId].tNumBlends == 1)
2018-12-01 00:29:55 +01:00
targetBlendAmount = 0;
2020-02-20 06:04:42 +01:00
BlendColorCycle(taskId, startBlendAmount, targetBlendAmount);
2018-12-01 00:29:55 +01:00
}
else
{
DestroyAnimVisualTask(taskId);
}
}
}
2020-02-20 06:04:42 +01:00
// See AnimTask_BlendColorCycle. Same, but excludes Attacker and Target
void AnimTask_BlendColorCycleExclude(u8 taskId)
2018-12-01 00:29:55 +01:00
{
int battler;
u32 selectedPalettes = 0;
gTasks[taskId].data[0] = gBattleAnimArgs[0];
2020-02-20 06:04:42 +01:00
gTasks[taskId].tDelay = gBattleAnimArgs[1];
gTasks[taskId].tNumBlends = gBattleAnimArgs[2];
gTasks[taskId].tInitialBlendY = gBattleAnimArgs[3];
gTasks[taskId].tTargetBlendY = gBattleAnimArgs[4];
gTasks[taskId].tBlendColor = gBattleAnimArgs[5];
gTasks[taskId].tRestoreBlend = 0;
2018-12-01 00:29:55 +01:00
for (battler = 0; battler < gBattlersCount; battler++)
{
if (battler != gBattleAnimAttacker && battler != gBattleAnimTarget)
selectedPalettes |= 1 << (battler + 16);
}
if (gBattleAnimArgs[0] == 1)
selectedPalettes |= 0xE;
2020-02-20 06:04:42 +01:00
gTasks[taskId].tPalSelectorHi = selectedPalettes >> 16;
gTasks[taskId].tPalSelectorLo = selectedPalettes & 0xFF;
BlendColorCycleExclude(taskId, 0, gTasks[taskId].tTargetBlendY);
gTasks[taskId].func = AnimTask_BlendColorCycleExcludeLoop;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
static void BlendColorCycleExclude(u8 taskId, u8 startBlendAmount, u8 targetBlendAmount)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
u32 selectedPalettes = ((u16)gTasks[taskId].tPalSelectorHi << 16) | (u16)gTasks[taskId].tPalSelectorLo;
2018-12-01 00:29:55 +01:00
BeginNormalPaletteFade(
selectedPalettes,
2020-02-20 06:04:42 +01:00
gTasks[taskId].tDelay,
startBlendAmount,
2018-12-01 00:29:55 +01:00
targetBlendAmount,
2020-02-20 06:04:42 +01:00
gTasks[taskId].tBlendColor);
2018-12-01 00:29:55 +01:00
2020-02-20 06:04:42 +01:00
gTasks[taskId].tNumBlends--;
gTasks[taskId].tRestoreBlend ^= 1;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
static void AnimTask_BlendColorCycleExcludeLoop(u8 taskId)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
u8 startBlendAmount, targetBlendAmount;
2018-12-01 00:29:55 +01:00
if (!gPaletteFade.active)
{
2020-02-20 06:04:42 +01:00
if (gTasks[taskId].tNumBlends > 0)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
if (!gTasks[taskId].tRestoreBlend)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
// Blend to designated color
startBlendAmount = gTasks[taskId].tInitialBlendY;
targetBlendAmount = gTasks[taskId].tTargetBlendY;
2018-12-01 00:29:55 +01:00
}
else
{
2020-02-20 06:04:42 +01:00
// Blend back to original color
startBlendAmount = gTasks[taskId].tTargetBlendY;
targetBlendAmount = gTasks[taskId].tInitialBlendY;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
if (gTasks[taskId].tNumBlends == 1)
2018-12-01 00:29:55 +01:00
targetBlendAmount = 0;
2020-02-20 06:04:42 +01:00
BlendColorCycleExclude(taskId, startBlendAmount, targetBlendAmount);
2018-12-01 00:29:55 +01:00
}
else
{
DestroyAnimVisualTask(taskId);
}
}
}
2020-02-20 06:04:42 +01:00
// See AnimTask_BlendColorCycle. Same, but selects palette by ANIM_TAG_*
void AnimTask_BlendColorCycleByTag(u8 taskId)
2018-12-01 00:29:55 +01:00
{
u8 paletteIndex;
2020-02-20 06:04:42 +01:00
gTasks[taskId].tPalTag = gBattleAnimArgs[0];
gTasks[taskId].tDelay = gBattleAnimArgs[1];
gTasks[taskId].tNumBlends = gBattleAnimArgs[2];
gTasks[taskId].tInitialBlendY = gBattleAnimArgs[3];
gTasks[taskId].tTargetBlendY = gBattleAnimArgs[4];
gTasks[taskId].tBlendColor = gBattleAnimArgs[5];
gTasks[taskId].tRestoreBlend = FALSE;
2018-12-01 00:29:55 +01:00
2020-02-20 06:04:42 +01:00
BlendColorCycleByTag(taskId, 0, gTasks[taskId].tTargetBlendY);
gTasks[taskId].func = AnimTask_BlendColorCycleByTagLoop;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
static void BlendColorCycleByTag(u8 taskId, u8 startBlendAmount, u8 targetBlendAmount)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
u8 paletteIndex = IndexOfSpritePaletteTag(gTasks[taskId].tPalTag);
2018-12-01 00:29:55 +01:00
BeginNormalPaletteFade(
1 << (paletteIndex + 16),
2020-02-20 06:04:42 +01:00
gTasks[taskId].tDelay,
startBlendAmount,
2018-12-01 00:29:55 +01:00
targetBlendAmount,
2020-02-20 06:04:42 +01:00
gTasks[taskId].tBlendColor);
2018-12-01 00:29:55 +01:00
2020-02-20 06:04:42 +01:00
gTasks[taskId].tNumBlends--;
gTasks[taskId].tRestoreBlend ^= 1;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
static void AnimTask_BlendColorCycleByTagLoop(u8 taskId)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
u8 startBlendAmount, targetBlendAmount;
2018-12-01 00:29:55 +01:00
if (!gPaletteFade.active)
{
2020-02-20 06:04:42 +01:00
if (gTasks[taskId].tNumBlends > 0)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
if (!gTasks[taskId].tRestoreBlend)
2018-12-01 00:29:55 +01:00
{
2020-02-20 06:04:42 +01:00
// Blend to designated color
startBlendAmount = gTasks[taskId].tInitialBlendY;
targetBlendAmount = gTasks[taskId].tTargetBlendY;
2018-12-01 00:29:55 +01:00
}
else
{
2020-02-20 06:04:42 +01:00
// Blend back to original color
startBlendAmount = gTasks[taskId].tTargetBlendY;
targetBlendAmount = gTasks[taskId].tInitialBlendY;
2018-12-01 00:29:55 +01:00
}
2020-02-20 06:04:42 +01:00
if (gTasks[taskId].tNumBlends == 1)
2018-12-01 00:29:55 +01:00
targetBlendAmount = 0;
2020-02-20 06:04:42 +01:00
BlendColorCycleByTag(taskId, startBlendAmount, targetBlendAmount);
2018-12-01 00:29:55 +01:00
}
else
{
DestroyAnimVisualTask(taskId);
}
}
}
2020-02-20 06:04:42 +01:00
#undef tPalSelector
#undef tPalTag
#undef tDelay
#undef tNumBlends
#undef tInitialBlendY
#undef tTargetBlendY
#undef tBlendColor
#undef tRestoreBlend
#undef tPalSelectorHi
#undef tPalSelectorLo
// Flashes the specified anim tag with given color. Used e.g. to flash the particles red in Hyper Beam
2020-02-20 02:22:53 +01:00
void AnimTask_FlashAnimTagWithColor(u8 taskId)
2018-12-01 00:29:55 +01:00
{
u8 paletteIndex;
gTasks[taskId].data[0] = gBattleAnimArgs[1];
gTasks[taskId].data[1] = gBattleAnimArgs[1];
gTasks[taskId].data[2] = gBattleAnimArgs[2];
gTasks[taskId].data[3] = gBattleAnimArgs[3];
gTasks[taskId].data[4] = gBattleAnimArgs[4];
gTasks[taskId].data[5] = gBattleAnimArgs[5];
gTasks[taskId].data[6] = gBattleAnimArgs[6];
gTasks[taskId].data[7] = gBattleAnimArgs[0];
paletteIndex = IndexOfSpritePaletteTag(gBattleAnimArgs[0]);
BeginNormalPaletteFade(
1 << (paletteIndex + 16),
0,
gBattleAnimArgs[4],
gBattleAnimArgs[4],
gBattleAnimArgs[3]);
2020-02-20 22:22:33 +01:00
gTasks[taskId].func = AnimTask_FlashAnimTagWithColor_Step1;
2018-12-01 00:29:55 +01:00
}
2020-02-20 22:22:33 +01:00
static void AnimTask_FlashAnimTagWithColor_Step1(u8 taskId)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes;
if (gTasks[taskId].data[0] > 0)
{
gTasks[taskId].data[0]--;
return;
}
if (gPaletteFade.active)
return;
if (gTasks[taskId].data[2] == 0)
{
2020-02-20 22:22:33 +01:00
gTasks[taskId].func = AnimTask_FlashAnimTagWithColor_Step2;
2018-12-01 00:29:55 +01:00
return;
}
selectedPalettes = 1 << (IndexOfSpritePaletteTag(gTasks[taskId].data[7]) + 16);
if (gTasks[taskId].data[1] & 0x100)
{
BeginNormalPaletteFade(
selectedPalettes,
0,
gTasks[taskId].data[4],
gTasks[taskId].data[4],
gTasks[taskId].data[3]);
}
else
{
BeginNormalPaletteFade(
selectedPalettes,
0,
gTasks[taskId].data[6],
gTasks[taskId].data[6],
gTasks[taskId].data[5]);
}
gTasks[taskId].data[1] ^= 0x100;
gTasks[taskId].data[0] = gTasks[taskId].data[1] & 0xFF;
gTasks[taskId].data[2]--;
}
2020-02-20 22:22:33 +01:00
static void AnimTask_FlashAnimTagWithColor_Step2(u8 taskId)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes;
if (!gPaletteFade.active)
{
selectedPalettes = 1 << (IndexOfSpritePaletteTag(gTasks[taskId].data[7]) + 16);
BeginNormalPaletteFade(selectedPalettes, 0, 0, 0, RGB(0, 0, 0));
DestroyAnimVisualTask(taskId);
}
}
2020-02-15 00:11:04 +01:00
void AnimTask_InvertScreenColor(u8 taskId)
2018-12-01 00:29:55 +01:00
{
u32 selectedPalettes = 0;
if (gBattleAnimArgs[0] & 0x1)
2018-12-01 00:29:55 +01:00
selectedPalettes = sub_80A75AC(1, 0, 0, 0, 0, 0, 0);
if (gBattleAnimArgs[0] & 0x2)
selectedPalettes |= (0x10000 << gBattleAnimAttacker);
if (gBattleAnimArgs[0] & 0x4)
selectedPalettes |= (0x10000 << gBattleAnimTarget);
if (gBattleAnimArgs[0] & 0x8 && IsBattlerAlive(BATTLE_PARTNER(gBattleAnimTarget)))
selectedPalettes |= (0x10000 << BATTLE_PARTNER(gBattleAnimTarget));
if (gBattleAnimArgs[0] & 0x10 && IsBattlerAlive(BATTLE_PARTNER(gBattleAnimAttacker)))
selectedPalettes |= (0x10000 << BATTLE_PARTNER(gBattleAnimAttacker));
2018-12-01 00:29:55 +01:00
InvertPlttBuffer(selectedPalettes);
DestroyAnimVisualTask(taskId);
}
void sub_8115F94(u8 taskId)
{
u8 attackerBattler;
u8 targetBattler;
u8 paletteIndex;
u32 selectedPalettes = 0;
if (gTasks[taskId].data[0] == 0)
{
gTasks[taskId].data[2] = gBattleAnimArgs[0];
gTasks[taskId].data[3] = gBattleAnimArgs[1];
gTasks[taskId].data[4] = gBattleAnimArgs[2];
gTasks[taskId].data[1] = gBattleAnimArgs[3];
gTasks[taskId].data[5] = gBattleAnimArgs[4];
gTasks[taskId].data[6] = gBattleAnimArgs[5];
gTasks[taskId].data[7] = gBattleAnimArgs[6];
}
gTasks[taskId].data[0]++;
attackerBattler = gBattleAnimAttacker;
targetBattler = gBattleAnimTarget;
if (gTasks[taskId].data[2] & 0x100)
selectedPalettes = 0x0000FFFF;
if (gTasks[taskId].data[2] & 0x1)
{
paletteIndex = IndexOfSpritePaletteTag(gSprites[gHealthboxSpriteIds[attackerBattler]].template->paletteTag);
selectedPalettes |= (1 << paletteIndex) << 16;
}
2019-10-19 17:47:55 +02:00
2018-12-01 00:29:55 +01:00
if (gTasks[taskId].data[3] & 0x100)
selectedPalettes |= (1 << attackerBattler) << 16;
if (gTasks[taskId].data[4] & 0x100)
selectedPalettes |= (1 << targetBattler) << 16;
TintPlttBuffer(selectedPalettes, gTasks[taskId].data[5], gTasks[taskId].data[6], gTasks[taskId].data[7]);
if (gTasks[taskId].data[0] == gTasks[taskId].data[1])
{
UnfadePlttBuffer(selectedPalettes);
DestroyAnimVisualTask(taskId);
}
}
2020-02-20 06:04:42 +01:00
static void AnimShakeMonOrBattleTerrain(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
u16 var0;
2020-02-20 22:22:33 +01:00
sprite->invisible = TRUE;
2018-12-01 00:29:55 +01:00
sprite->data[0] = -gBattleAnimArgs[0];
sprite->data[1] = gBattleAnimArgs[1];
sprite->data[2] = gBattleAnimArgs[1];
sprite->data[3] = gBattleAnimArgs[2];
switch (gBattleAnimArgs[3])
{
case 0:
StoreSpriteCallbackInData6(sprite, (void *)&gBattle_BG3_X);
break;
case 1:
StoreSpriteCallbackInData6(sprite, (void *)&gBattle_BG3_Y);
break;
case 2:
StoreSpriteCallbackInData6(sprite, (void *)&gSpriteCoordOffsetX);
break;
default:
StoreSpriteCallbackInData6(sprite, (void *)&gSpriteCoordOffsetY);
break;
}
sprite->data[4] = *(u16 *)(sprite->data[6] | (sprite->data[7] << 16));
sprite->data[5] = gBattleAnimArgs[3];
var0 = sprite->data[5] - 2;
if (var0 < 2)
2020-02-20 22:22:33 +01:00
AnimShakeMonOrBattleTerrain_UpdateCoordOffsetEnabled();
2018-12-01 00:29:55 +01:00
2020-02-20 22:22:33 +01:00
sprite->callback = AnimShakeMonOrBattleTerrain_Step;
2018-12-01 00:29:55 +01:00
}
2020-02-20 22:22:33 +01:00
static void AnimShakeMonOrBattleTerrain_Step(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
u8 i;
u16 var0;
if (sprite->data[3] > 0)
{
sprite->data[3]--;
if (sprite->data[1] > 0)
{
sprite->data[1]--;
}
else
{
sprite->data[1] = sprite->data[2];
*(u16 *)(sprite->data[6] | (sprite->data[7] << 16)) += sprite->data[0];
sprite->data[0] = -sprite->data[0];
}
}
else
{
*(u16 *)(sprite->data[6] | (sprite->data[7] << 16)) = sprite->data[4];
var0 = sprite->data[5] - 2;
if (var0 < 2)
{
for (i = 0; i < gBattlersCount; i++)
2020-02-20 22:22:33 +01:00
gSprites[gBattlerSpriteIds[i]].coordOffsetEnabled = FALSE;
2018-12-01 00:29:55 +01:00
}
DestroyAnimSprite(sprite);
}
}
2020-02-20 22:22:33 +01:00
static void AnimShakeMonOrBattleTerrain_UpdateCoordOffsetEnabled(void)
2018-12-01 00:29:55 +01:00
{
2020-02-20 22:22:33 +01:00
gSprites[gBattlerSpriteIds[gBattleAnimAttacker]].coordOffsetEnabled = FALSE;
gSprites[gBattlerSpriteIds[gBattleAnimTarget]].coordOffsetEnabled = FALSE;
2018-12-01 00:29:55 +01:00
if (gBattleAnimArgs[4] == 2)
{
2020-02-20 22:22:33 +01:00
gSprites[gBattlerSpriteIds[gBattleAnimAttacker]].coordOffsetEnabled = TRUE;
gSprites[gBattlerSpriteIds[gBattleAnimTarget]].coordOffsetEnabled = TRUE;
2018-12-01 00:29:55 +01:00
}
else
{
if (gBattleAnimArgs[4] == 0)
2020-02-20 22:22:33 +01:00
gSprites[gBattlerSpriteIds[gBattleAnimAttacker]].coordOffsetEnabled = TRUE;
2018-12-01 00:29:55 +01:00
else
2020-02-20 22:22:33 +01:00
gSprites[gBattlerSpriteIds[gBattleAnimTarget]].coordOffsetEnabled = TRUE;
2018-12-01 00:29:55 +01:00
}
}
2020-02-16 23:49:24 +01:00
// Task data for AnimTask_ShakeBattleTerrain
#define tXOffset data[0]
#define tYOffset data[1]
#define tNumShakes data[2]
#define tTimer data[3]
#define tShakeDelay data[8]
// Can shake battle terrain back and forth on the X or down and back to original pos on Y (cant shake up from orig pos)
// arg0: x offset of shake
// arg1: y offset of shake
// arg2: number of shakes
// arg3: time between shakes
void AnimTask_ShakeBattleTerrain(u8 taskId)
{
gTasks[taskId].tXOffset = gBattleAnimArgs[0];
gTasks[taskId].tYOffset = gBattleAnimArgs[1];
gTasks[taskId].tNumShakes = gBattleAnimArgs[2];
gTasks[taskId].tTimer = gBattleAnimArgs[3];
gTasks[taskId].tShakeDelay = gBattleAnimArgs[3];
2018-12-01 00:29:55 +01:00
gBattle_BG3_X = gBattleAnimArgs[0];
gBattle_BG3_Y = gBattleAnimArgs[1];
2020-02-20 22:22:33 +01:00
gTasks[taskId].func = AnimTask_ShakeBattleTerrain_Step;
2018-12-01 00:29:55 +01:00
gTasks[taskId].func(taskId);
}
2020-02-20 22:22:33 +01:00
static void AnimTask_ShakeBattleTerrain_Step(u8 taskId)
2018-12-01 00:29:55 +01:00
{
2020-02-16 23:49:24 +01:00
if (gTasks[taskId].tTimer == 0)
2018-12-01 00:29:55 +01:00
{
2020-02-16 23:49:24 +01:00
if (gBattle_BG3_X == gTasks[taskId].tXOffset)
gBattle_BG3_X = -gTasks[taskId].tXOffset;
2018-12-01 00:29:55 +01:00
else
2020-02-16 23:49:24 +01:00
gBattle_BG3_X = gTasks[taskId].tXOffset;
2018-12-01 00:29:55 +01:00
2020-02-16 23:49:24 +01:00
if (gBattle_BG3_Y == -gTasks[taskId].tYOffset)
2018-12-01 00:29:55 +01:00
gBattle_BG3_Y = 0;
else
2020-02-16 23:49:24 +01:00
gBattle_BG3_Y = -gTasks[taskId].tYOffset;
2018-12-01 00:29:55 +01:00
2020-02-16 23:49:24 +01:00
gTasks[taskId].tTimer = gTasks[taskId].tShakeDelay;
if (--gTasks[taskId].tNumShakes == 0)
2018-12-01 00:29:55 +01:00
{
gBattle_BG3_X = 0;
gBattle_BG3_Y = 0;
DestroyAnimVisualTask(taskId);
}
}
else
{
2020-02-16 23:49:24 +01:00
gTasks[taskId].tTimer--;
2018-12-01 00:29:55 +01:00
}
}
2020-02-16 23:49:24 +01:00
#undef tXOffset
#undef tYOffset
#undef tNumShakes
#undef tTimer
#undef tShakeDelay
2020-05-25 08:37:32 +02:00
void AnimHitSplatBasic(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
StartSpriteAffineAnim(sprite, gBattleAnimArgs[3]);
2020-02-15 00:11:04 +01:00
if (gBattleAnimArgs[2] == ANIM_ATTACKER)
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimAttacker(sprite, 1);
2018-12-01 00:29:55 +01:00
else
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimTarget(sprite, TRUE);
2018-12-01 00:29:55 +01:00
2018-12-18 05:08:08 +01:00
sprite->callback = RunStoredCallbackWhenAffineAnimEnds;
2018-12-01 00:29:55 +01:00
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
}
2020-02-22 00:22:18 +01:00
// Same as basic hit splat but takes a length of time to persist for (arg4)
static void AnimHitSplatPersistent(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
StartSpriteAffineAnim(sprite, gBattleAnimArgs[3]);
2020-02-15 00:11:04 +01:00
if (gBattleAnimArgs[2] == ANIM_ATTACKER)
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimAttacker(sprite, 1);
2018-12-01 00:29:55 +01:00
else
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimTarget(sprite, TRUE);
2018-12-01 00:29:55 +01:00
sprite->data[0] = gBattleAnimArgs[4];
2018-12-18 05:08:08 +01:00
sprite->callback = RunStoredCallbackWhenAffineAnimEnds;
2020-02-22 00:22:18 +01:00
StoreSpriteCallbackInData6(sprite, DestroyAnimSpriteAfterTimer);
2018-12-01 00:29:55 +01:00
}
2020-02-22 00:22:18 +01:00
// For paired hit splats whose position is inverted when used by the opponent on the player.
// Used by Twineedle and Spike Cannon
static void AnimHitSplatHandleInvert(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER && !IsContest())
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
2019-10-19 17:47:55 +02:00
2020-02-22 00:22:18 +01:00
AnimHitSplatBasic(sprite);
2018-12-01 00:29:55 +01:00
}
2020-05-25 08:37:32 +02:00
void AnimHitSplatRandom(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
if (gBattleAnimArgs[1] == -1)
gBattleAnimArgs[1] = Random2() & 3;
if (!InitSpritePosToAnimBattler(gBattleAnimArgs[0], sprite, FALSE))
return;
2018-12-01 00:29:55 +01:00
StartSpriteAffineAnim(sprite, gBattleAnimArgs[1]);
sprite->pos2.x += (Random2() % 48) - 24;
sprite->pos2.y += (Random2() % 24) - 12;
2018-12-18 05:08:08 +01:00
StoreSpriteCallbackInData6(sprite, DestroySpriteAndMatrix);
sprite->callback = RunStoredCallbackWhenAffineAnimEnds;
2018-12-01 00:29:55 +01:00
}
2020-05-25 08:37:32 +02:00
void AnimHitSplatOnMonEdge(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
sprite->data[0] = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
sprite->pos1.x = gSprites[sprite->data[0]].pos1.x + gSprites[sprite->data[0]].pos2.x;
sprite->pos1.y = gSprites[sprite->data[0]].pos1.y + gSprites[sprite->data[0]].pos2.y;
sprite->pos2.x = gBattleAnimArgs[1];
sprite->pos2.y = gBattleAnimArgs[2];
StartSpriteAffineAnim(sprite, gBattleAnimArgs[3]);
2018-12-18 05:08:08 +01:00
StoreSpriteCallbackInData6(sprite, DestroySpriteAndMatrix);
sprite->callback = RunStoredCallbackWhenAffineAnimEnds;
2018-12-01 00:29:55 +01:00
}
2020-05-25 08:37:32 +02:00
void AnimCrossImpact(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
2020-02-15 00:11:04 +01:00
if (gBattleAnimArgs[2] == ANIM_ATTACKER)
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimAttacker(sprite, 1);
2018-12-01 00:29:55 +01:00
else
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimTarget(sprite, TRUE);
2018-12-01 00:29:55 +01:00
sprite->data[0] = gBattleAnimArgs[3];
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
sprite->callback = WaitAnimForDuration;
}
2020-05-25 08:37:32 +02:00
void AnimFlashingHitSplat(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
StartSpriteAffineAnim(sprite, gBattleAnimArgs[3]);
2020-02-15 00:11:04 +01:00
if (gBattleAnimArgs[2] == ANIM_ATTACKER)
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimAttacker(sprite, 1);
2018-12-01 00:29:55 +01:00
else
2018-12-20 04:13:26 +01:00
InitSpritePosToAnimTarget(sprite, TRUE);
2018-12-01 00:29:55 +01:00
2020-02-20 22:22:33 +01:00
sprite->callback = AnimFlashingHitSplat_Step;
2018-12-01 00:29:55 +01:00
}
2020-05-25 08:37:32 +02:00
void AnimFlashingHitSplat_Step(struct Sprite *sprite)
2018-12-01 00:29:55 +01:00
{
sprite->invisible ^= 1;
if (sprite->data[0]++ > 12)
DestroyAnimSprite(sprite);
}