mirror of
https://github.com/Ninjdai1/pokeemerald.git
synced 2025-01-22 11:20:17 +01:00
1060 lines
32 KiB
C
1060 lines
32 KiB
C
#include "global.h"
|
|
#include "battle.h"
|
|
#include "battle_anim.h"
|
|
#include "sprite.h"
|
|
#include "task.h"
|
|
#include "trig.h"
|
|
|
|
// This file's functions.
|
|
static void AnimTask_ShakeMon_Step(u8 taskId);
|
|
static void AnimTask_ShakeMon2_Step(u8 taskId);
|
|
static void AnimTask_ShakeMonInPlace_Step(u8 taskId);
|
|
static void AnimTask_ShakeAndSinkMon_Step(u8 taskId);
|
|
static void AnimTask_TranslateMonElliptical_Step(u8 taskId);
|
|
static void DoHorizontalLunge(struct Sprite *sprite);
|
|
static void ReverseHorizontalLungeDirection(struct Sprite *sprite);
|
|
static void DoVerticalDip(struct Sprite *sprite);
|
|
static void ReverseVerticalDipDirection(struct Sprite* sprite);
|
|
static void SlideMonToOriginalPos(struct Sprite *sprite);
|
|
static void SlideMonToOriginalPos_Step(struct Sprite *sprite);
|
|
static void SlideMonToOffset(struct Sprite *sprite);
|
|
static void SlideMonToOffsetAndBack(struct Sprite *sprite);
|
|
static void SlideMonToOffsetAndBack_End(struct Sprite *sprite);
|
|
static void AnimTask_WindUpLunge_Step1(u8 taskId);
|
|
static void AnimTask_WindUpLunge_Step2(u8 taskId);
|
|
static void AnimTask_SwayMonStep(u8 taskId);
|
|
static void AnimTask_ScaleMonAndRestore_Step(u8 taskId);
|
|
static void AnimTask_RotateMonSpriteToSide_Step(u8 taskId);
|
|
static void AnimTask_ShakeTargetBasedOnMovePowerOrDmg_Step(u8 taskId);
|
|
static void AnimTask_SlideOffScreen_Step(u8 taskId);
|
|
|
|
const struct SpriteTemplate gHorizontalLungeSpriteTemplate =
|
|
{
|
|
.tileTag = 0,
|
|
.paletteTag = 0,
|
|
.oam = &gDummyOamData,
|
|
.anims = gDummySpriteAnimTable,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = DoHorizontalLunge,
|
|
};
|
|
|
|
const struct SpriteTemplate gVerticalDipSpriteTemplate =
|
|
{
|
|
.tileTag = 0,
|
|
.paletteTag = 0,
|
|
.oam = &gDummyOamData,
|
|
.anims = gDummySpriteAnimTable,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = DoVerticalDip,
|
|
};
|
|
|
|
const struct SpriteTemplate gSlideMonToOriginalPosSpriteTemplate =
|
|
{
|
|
.tileTag = 0,
|
|
.paletteTag = 0,
|
|
.oam = &gDummyOamData,
|
|
.anims = gDummySpriteAnimTable,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SlideMonToOriginalPos,
|
|
};
|
|
|
|
const struct SpriteTemplate gSlideMonToOffsetSpriteTemplate =
|
|
{
|
|
.tileTag = 0,
|
|
.paletteTag = 0,
|
|
.oam = &gDummyOamData,
|
|
.anims = gDummySpriteAnimTable,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SlideMonToOffset,
|
|
};
|
|
|
|
const struct SpriteTemplate gSlideMonToOffsetAndBackSpriteTemplate =
|
|
{
|
|
.tileTag = 0,
|
|
.paletteTag = 0,
|
|
.oam = &gDummyOamData,
|
|
.anims = gDummySpriteAnimTable,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SlideMonToOffsetAndBack,
|
|
};
|
|
|
|
// Task to facilitate simple shaking of a pokemon's picture in battle.
|
|
// The shaking alternates between the original position and the target position.
|
|
// arg 0: anim battler
|
|
// arg 1: x pixel offset
|
|
// arg 2: y pixel offset
|
|
// arg 3: num times to shake
|
|
// arg 4: frame delay
|
|
void AnimTask_ShakeMon(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
if (spriteId == SPRITE_NONE)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
gSprites[spriteId].pos2.x = gBattleAnimArgs[1];
|
|
gSprites[spriteId].pos2.y = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[0] = spriteId;
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[4] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[5] = gBattleAnimArgs[2];
|
|
gTasks[taskId].func = AnimTask_ShakeMon_Step;
|
|
AnimTask_ShakeMon_Step(taskId);
|
|
}
|
|
|
|
static void AnimTask_ShakeMon_Step(u8 taskId)
|
|
{
|
|
if (gTasks[taskId].data[3] == 0)
|
|
{
|
|
if (gSprites[gTasks[taskId].data[0]].pos2.x == 0)
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x = gTasks[taskId].data[4];
|
|
}
|
|
else
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x = 0;
|
|
}
|
|
if (gSprites[gTasks[taskId].data[0]].pos2.y == 0)
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.y = gTasks[taskId].data[5];
|
|
}
|
|
else
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.y = 0;
|
|
}
|
|
gTasks[taskId].data[3] = gTasks[taskId].data[2];
|
|
if (--gTasks[taskId].data[1] == 0)
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x = 0;
|
|
gSprites[gTasks[taskId].data[0]].pos2.y = 0;
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[3]--;
|
|
}
|
|
}
|
|
|
|
// Task to facilitate simple shaking of a pokemon's picture in battle.
|
|
// The shaking alternates between the positive and negative versions of the specified pixel offsets.
|
|
// arg 0: anim battler
|
|
// arg 1: x pixel offset
|
|
// arg 2: y pixel offset
|
|
// arg 3: num times to shake
|
|
// arg 4: frame delay
|
|
void AnimTask_ShakeMon2(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
bool8 destroy = FALSE;
|
|
u8 battlerId;
|
|
|
|
if (gBattleAnimArgs[0] < MAX_BATTLERS_COUNT)
|
|
{
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
if (spriteId == SPRITE_NONE)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
else if (gBattleAnimArgs[0] != 8)
|
|
{
|
|
switch (gBattleAnimArgs[0])
|
|
{
|
|
case 4:
|
|
battlerId = GetBattlerAtPosition(B_POSITION_PLAYER_LEFT);
|
|
break;
|
|
case 5:
|
|
battlerId = GetBattlerAtPosition(B_POSITION_PLAYER_RIGHT);
|
|
break;
|
|
case 6:
|
|
battlerId = GetBattlerAtPosition(B_POSITION_OPPONENT_LEFT);
|
|
break;
|
|
case 7:
|
|
default:
|
|
battlerId = GetBattlerAtPosition(B_POSITION_OPPONENT_RIGHT);
|
|
break;
|
|
}
|
|
|
|
if (IsBattlerSpriteVisible(battlerId) == FALSE)
|
|
destroy = TRUE;
|
|
|
|
spriteId = gBattlerSpriteIds[battlerId];
|
|
}
|
|
else
|
|
{
|
|
spriteId = gBattlerSpriteIds[gBattleAnimAttacker];
|
|
}
|
|
|
|
if (destroy)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
|
|
gSprites[spriteId].pos2.x = gBattleAnimArgs[1];
|
|
gSprites[spriteId].pos2.y = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[0] = spriteId;
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[4] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[5] = gBattleAnimArgs[2];
|
|
gTasks[taskId].func = AnimTask_ShakeMon2_Step;
|
|
gTasks[taskId].func(taskId);
|
|
}
|
|
|
|
static void AnimTask_ShakeMon2_Step(u8 taskId)
|
|
{
|
|
if (gTasks[taskId].data[3] == 0)
|
|
{
|
|
if (gSprites[gTasks[taskId].data[0]].pos2.x == gTasks[taskId].data[4])
|
|
gSprites[gTasks[taskId].data[0]].pos2.x = -gTasks[taskId].data[4];
|
|
else
|
|
gSprites[gTasks[taskId].data[0]].pos2.x = gTasks[taskId].data[4];
|
|
|
|
if (gSprites[gTasks[taskId].data[0]].pos2.y == gTasks[taskId].data[5])
|
|
gSprites[gTasks[taskId].data[0]].pos2.y = -gTasks[taskId].data[5];
|
|
else
|
|
gSprites[gTasks[taskId].data[0]].pos2.y = gTasks[taskId].data[5];
|
|
|
|
gTasks[taskId].data[3] = gTasks[taskId].data[2];
|
|
if (--gTasks[taskId].data[1] == 0)
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x = 0;
|
|
gSprites[gTasks[taskId].data[0]].pos2.y = 0;
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[3]--;
|
|
}
|
|
}
|
|
|
|
// Task to facilitate simple shaking of a pokemon's picture in battle.
|
|
// The shaking alternates between the positive and negative versions of the specified pixel offsets
|
|
// with respect to the current location of the mon's picture.
|
|
// arg 0: battler
|
|
// arg 1: x offset
|
|
// arg 2: y offset
|
|
// arg 3: num shakes
|
|
// arg 4: delay
|
|
void AnimTask_ShakeMonInPlace(u8 taskId)
|
|
{
|
|
u8 spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
if (spriteId == SPRITE_NONE)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
|
|
gSprites[spriteId].pos2.x += gBattleAnimArgs[1];
|
|
gSprites[spriteId].pos2.y += gBattleAnimArgs[2];
|
|
gTasks[taskId].data[0] = spriteId;
|
|
gTasks[taskId].data[1] = 0;
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[3] = 0;
|
|
gTasks[taskId].data[4] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[5] = gBattleAnimArgs[1] * 2;
|
|
gTasks[taskId].data[6] = gBattleAnimArgs[2] * 2;
|
|
gTasks[taskId].func = AnimTask_ShakeMonInPlace_Step;
|
|
gTasks[taskId].func(taskId);
|
|
}
|
|
|
|
static void AnimTask_ShakeMonInPlace_Step(u8 taskId)
|
|
{
|
|
if (gTasks[taskId].data[3] == 0)
|
|
{
|
|
if (gTasks[taskId].data[1] & 1)
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x += gTasks[taskId].data[5];
|
|
gSprites[gTasks[taskId].data[0]].pos2.y += gTasks[taskId].data[6];
|
|
}
|
|
else
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x -= gTasks[taskId].data[5];
|
|
gSprites[gTasks[taskId].data[0]].pos2.y -= gTasks[taskId].data[6];
|
|
}
|
|
gTasks[taskId].data[3] = gTasks[taskId].data[4];
|
|
if (++gTasks[taskId].data[1] >= gTasks[taskId].data[2])
|
|
{
|
|
if (gTasks[taskId].data[1] & 1)
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x += gTasks[taskId].data[5] / 2;
|
|
gSprites[gTasks[taskId].data[0]].pos2.y += gTasks[taskId].data[6] / 2;
|
|
}
|
|
else
|
|
{
|
|
gSprites[gTasks[taskId].data[0]].pos2.x -= gTasks[taskId].data[5] / 2;
|
|
gSprites[gTasks[taskId].data[0]].pos2.y -= gTasks[taskId].data[6] / 2;
|
|
}
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[3]--;
|
|
}
|
|
}
|
|
|
|
// Shakes a mon bg horizontally and moves it downward linearly.
|
|
// arg 0: battler
|
|
// arg 1: x offset
|
|
// arg 2: frame delay between each movement
|
|
// arg 3: downward speed (subpixel)
|
|
// arg 4: duration
|
|
void AnimTask_ShakeAndSinkMon(u8 taskId)
|
|
{
|
|
u8 spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
gSprites[spriteId].pos2.x = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[0] = spriteId;
|
|
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].func = AnimTask_ShakeAndSinkMon_Step;
|
|
gTasks[taskId].func(taskId);
|
|
}
|
|
|
|
static void AnimTask_ShakeAndSinkMon_Step(u8 taskId)
|
|
{
|
|
s16 x;
|
|
u8 spriteId;
|
|
spriteId = gTasks[taskId].data[0];
|
|
x = gTasks[taskId].data[1];
|
|
if (gTasks[taskId].data[2] == gTasks[taskId].data[8]++)
|
|
{
|
|
gTasks[taskId].data[8] = 0;
|
|
if (gSprites[spriteId].pos2.x == x)
|
|
x = -x;
|
|
|
|
gSprites[spriteId].pos2.x += x;
|
|
}
|
|
|
|
gTasks[taskId].data[1] = x;
|
|
gTasks[taskId].data[9] += gTasks[taskId].data[3];
|
|
gSprites[spriteId].pos2.y = gTasks[taskId].data[9] >> 8;
|
|
if (--gTasks[taskId].data[4] == 0)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Moves a mon bg picture along an elliptical path that begins
|
|
// and ends at the mon's origin location.
|
|
// arg 0: battler
|
|
// arg 1: ellipse width
|
|
// arg 2: ellipse height
|
|
// arg 3: num loops
|
|
// arg 4: speed (valid values are 0-5)
|
|
void AnimTask_TranslateMonElliptical(u8 taskId)
|
|
{
|
|
u8 i;
|
|
u8 spriteId;
|
|
u8 wavePeriod;
|
|
|
|
wavePeriod = 1;
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
if (gBattleAnimArgs[4] > 5)
|
|
gBattleAnimArgs[4] = 5;
|
|
|
|
for (i = 0; i < gBattleAnimArgs[4]; i++)
|
|
{
|
|
wavePeriod <<= 1;
|
|
}
|
|
|
|
gTasks[taskId].data[0] = spriteId;
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[4] = wavePeriod;
|
|
gTasks[taskId].func = AnimTask_TranslateMonElliptical_Step;
|
|
gTasks[taskId].func(taskId);
|
|
}
|
|
|
|
static void AnimTask_TranslateMonElliptical_Step(u8 taskId)
|
|
{
|
|
u8 spriteId = gTasks[taskId].data[0];
|
|
gSprites[spriteId].pos2.x = Sin(gTasks[taskId].data[5], gTasks[taskId].data[1]);
|
|
gSprites[spriteId].pos2.y = -Cos(gTasks[taskId].data[5], gTasks[taskId].data[2]);
|
|
gSprites[spriteId].pos2.y += gTasks[taskId].data[2];
|
|
gTasks[taskId].data[5] += gTasks[taskId].data[4];
|
|
gTasks[taskId].data[5] &= 0xff;
|
|
|
|
if (gTasks[taskId].data[5] == 0)
|
|
gTasks[taskId].data[3]--;
|
|
|
|
if (gTasks[taskId].data[3] == 0)
|
|
{
|
|
gSprites[spriteId].pos2.x = 0;
|
|
gSprites[spriteId].pos2.y = 0;
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Moves a mon bg picture along an elliptical path that begins
|
|
// and ends at the mon's origin location. Reverses the direction
|
|
// of the path if it's not on the player's side of the battle.
|
|
// arg 0: battler
|
|
// arg 1: ellipse width
|
|
// arg 2: ellipse height
|
|
// arg 3: num loops
|
|
// arg 4: speed (valid values are 0-5)
|
|
void AnimTask_TranslateMonEllipticalRespectSide(u8 taskId)
|
|
{
|
|
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
|
|
AnimTask_TranslateMonElliptical(taskId);
|
|
}
|
|
|
|
// Performs a simple horizontal lunge, where the mon moves
|
|
// horizontally, and then moves back in the opposite direction.
|
|
// arg 0: duration of single lunge direction
|
|
// arg 1: x pixel delta that is applied each frame
|
|
static void DoHorizontalLunge(struct Sprite *sprite)
|
|
{
|
|
sprite->invisible = TRUE;
|
|
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
|
|
sprite->data[1] = -gBattleAnimArgs[1];
|
|
else
|
|
sprite->data[1] = gBattleAnimArgs[1];
|
|
|
|
sprite->data[0] = gBattleAnimArgs[0];
|
|
sprite->data[2] = 0;
|
|
sprite->data[3] = gBattlerSpriteIds[gBattleAnimAttacker];
|
|
sprite->data[4] = gBattleAnimArgs[0];
|
|
StoreSpriteCallbackInData6(sprite, ReverseHorizontalLungeDirection);
|
|
sprite->callback = TranslateMonSpriteLinear;
|
|
}
|
|
|
|
static void ReverseHorizontalLungeDirection(struct Sprite *sprite)
|
|
{
|
|
sprite->data[0] = sprite->data[4];
|
|
sprite->data[1] = -sprite->data[1];
|
|
sprite->callback = TranslateMonSpriteLinear;
|
|
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
|
|
}
|
|
|
|
// Performs a simple vertical dipping motion, where moves vertically, and then
|
|
// moves back in the opposite direction.
|
|
// arg 0: duration of single dip direction
|
|
// arg 1: y pixel delta that is applied each frame
|
|
// arg 2: battler
|
|
static void DoVerticalDip(struct Sprite *sprite)
|
|
{
|
|
u8 spriteId;
|
|
sprite->invisible = TRUE;
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[2]);
|
|
sprite->data[0] = gBattleAnimArgs[0];
|
|
sprite->data[1] = 0;
|
|
sprite->data[2] = gBattleAnimArgs[1];
|
|
sprite->data[3] = spriteId;
|
|
sprite->data[4] = gBattleAnimArgs[0];
|
|
StoreSpriteCallbackInData6(sprite, ReverseVerticalDipDirection);
|
|
sprite->callback = TranslateMonSpriteLinear;
|
|
}
|
|
|
|
static void ReverseVerticalDipDirection(struct Sprite *sprite)
|
|
{
|
|
sprite->data[0] = sprite->data[4];
|
|
sprite->data[2] = -sprite->data[2];
|
|
sprite->callback = TranslateMonSpriteLinear;
|
|
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
|
|
}
|
|
|
|
// Linearly slides a mon's bg picture back to its original sprite position.
|
|
// The sprite parameter is a dummy sprite used for facilitating the movement with its callback.
|
|
// arg 0: 1 = target or 0 = attacker
|
|
// arg 1: direction (0 = horizontal and vertical, 1 = horizontal only, 2 = vertical only)
|
|
// arg 2: duration
|
|
static void SlideMonToOriginalPos(struct Sprite *sprite)
|
|
{
|
|
u32 monSpriteId;
|
|
if (!gBattleAnimArgs[0])
|
|
monSpriteId = gBattlerSpriteIds[gBattleAnimAttacker];
|
|
else
|
|
monSpriteId = gBattlerSpriteIds[gBattleAnimTarget];
|
|
|
|
sprite->data[0] = gBattleAnimArgs[2];
|
|
sprite->data[1] = gSprites[monSpriteId].pos1.x + gSprites[monSpriteId].pos2.x;
|
|
sprite->data[2] = gSprites[monSpriteId].pos1.x;
|
|
sprite->data[3] = gSprites[monSpriteId].pos1.y + gSprites[monSpriteId].pos2.y;
|
|
sprite->data[4] = gSprites[monSpriteId].pos1.y;
|
|
InitSpriteDataForLinearTranslation(sprite);
|
|
sprite->data[3] = 0;
|
|
sprite->data[4] = 0;
|
|
sprite->data[5] = gSprites[monSpriteId].pos2.x;
|
|
sprite->data[6] = gSprites[monSpriteId].pos2.y;
|
|
sprite->invisible = TRUE;
|
|
|
|
if (gBattleAnimArgs[1] == 1)
|
|
sprite->data[2] = 0;
|
|
else if (gBattleAnimArgs[1] == 2)
|
|
sprite->data[1] = 0;
|
|
|
|
sprite->data[7] = gBattleAnimArgs[1];
|
|
sprite->data[7] |= monSpriteId << 8;
|
|
sprite->callback = SlideMonToOriginalPos_Step;
|
|
}
|
|
|
|
static void SlideMonToOriginalPos_Step(struct Sprite *sprite)
|
|
{
|
|
s8 monSpriteId;
|
|
u8 lo;
|
|
struct Sprite *monSprite;
|
|
|
|
lo = sprite->data[7] & 0xff;
|
|
monSpriteId = sprite->data[7] >> 8;
|
|
monSprite = &gSprites[monSpriteId];
|
|
if (sprite->data[0] == 0)
|
|
{
|
|
if (lo < 2)
|
|
monSprite->pos2.x = 0;
|
|
|
|
if (lo == 2 || lo == 0)
|
|
monSprite->pos2.y = 0;
|
|
|
|
DestroyAnimSprite(sprite);
|
|
}
|
|
else
|
|
{
|
|
sprite->data[0]--;
|
|
sprite->data[3] += sprite->data[1];
|
|
sprite->data[4] += sprite->data[2];
|
|
monSprite->pos2.x = (s8)(sprite->data[3] >> 8) + sprite->data[5];
|
|
monSprite->pos2.y = (s8)(sprite->data[4] >> 8) + sprite->data[6];
|
|
}
|
|
}
|
|
|
|
// Linearly translates a mon to a target offset. The horizontal offset
|
|
// is mirrored for the opponent's pokemon, and the vertical offset
|
|
// is only mirrored if arg 3 is set to 1.
|
|
// arg 0: 0 = attacker, 1 = target
|
|
// arg 1: target x pixel offset
|
|
// arg 2: target y pixel offset
|
|
// arg 3: mirror vertical translation for opposite battle side
|
|
// arg 4: duration
|
|
static void SlideMonToOffset(struct Sprite *sprite)
|
|
{
|
|
u8 battler;
|
|
u8 monSpriteId;
|
|
if (!gBattleAnimArgs[0])
|
|
battler = gBattleAnimAttacker;
|
|
else
|
|
battler = gBattleAnimTarget;
|
|
|
|
monSpriteId = gBattlerSpriteIds[battler];
|
|
if (GetBattlerSide(battler) != B_SIDE_PLAYER)
|
|
{
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
if (gBattleAnimArgs[3] == 1)
|
|
{
|
|
gBattleAnimArgs[2] = -gBattleAnimArgs[2];
|
|
}
|
|
}
|
|
|
|
sprite->data[0] = gBattleAnimArgs[4];
|
|
sprite->data[1] = gSprites[monSpriteId].pos1.x;
|
|
sprite->data[2] = gSprites[monSpriteId].pos1.x + gBattleAnimArgs[1];
|
|
sprite->data[3] = gSprites[monSpriteId].pos1.y;
|
|
sprite->data[4] = gSprites[monSpriteId].pos1.y + gBattleAnimArgs[2];
|
|
InitSpriteDataForLinearTranslation(sprite);
|
|
sprite->data[3] = 0;
|
|
sprite->data[4] = 0;
|
|
sprite->data[5] = monSpriteId;
|
|
sprite->invisible = TRUE;
|
|
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
|
|
sprite->callback = TranslateMonSpriteLinearFixedPoint;
|
|
}
|
|
|
|
static void SlideMonToOffsetAndBack(struct Sprite *sprite)
|
|
{
|
|
u8 spriteId;
|
|
u8 battlerId;
|
|
sprite->invisible = TRUE;
|
|
|
|
if (gBattleAnimArgs[0] == ANIM_ATTACKER)
|
|
battlerId = gBattleAnimAttacker;
|
|
else
|
|
battlerId = gBattleAnimTarget;
|
|
|
|
spriteId = gBattlerSpriteIds[battlerId];
|
|
if (GetBattlerSide(battlerId))
|
|
{
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
if (gBattleAnimArgs[3] == 1)
|
|
{
|
|
gBattleAnimArgs[2] = -gBattleAnimArgs[2];
|
|
}
|
|
}
|
|
sprite->data[0] = gBattleAnimArgs[4];
|
|
sprite->data[1] = gSprites[spriteId].pos1.x + gSprites[spriteId].pos2.x;
|
|
sprite->data[2] = sprite->data[1] + gBattleAnimArgs[1];
|
|
sprite->data[3] = gSprites[spriteId].pos1.y + gSprites[spriteId].pos2.y;
|
|
sprite->data[4] = sprite->data[3] + gBattleAnimArgs[2];
|
|
InitSpriteDataForLinearTranslation(sprite);
|
|
sprite->data[3] = gSprites[spriteId].pos2.x << 8;
|
|
sprite->data[4] = gSprites[spriteId].pos2.y << 8;
|
|
sprite->data[5] = spriteId;
|
|
sprite->data[6] = gBattleAnimArgs[5];
|
|
if (!gBattleAnimArgs[5])
|
|
{
|
|
StoreSpriteCallbackInData6(sprite, DestroyAnimSprite);
|
|
}
|
|
else
|
|
{
|
|
StoreSpriteCallbackInData6(sprite, SlideMonToOffsetAndBack_End);
|
|
}
|
|
sprite->callback = TranslateMonSpriteLinearFixedPoint;
|
|
}
|
|
|
|
|
|
static void SlideMonToOffsetAndBack_End(struct Sprite *sprite)
|
|
{
|
|
gSprites[sprite->data[5]].pos2.x = 0;
|
|
gSprites[sprite->data[5]].pos2.y = 0;
|
|
DestroyAnimSprite(sprite);
|
|
}
|
|
|
|
// Task to facilitate a two-part translation animation, in which the sprite
|
|
// is first translated in an arc to one position. Then, it "lunges" to a target
|
|
// x offset. Used in TAKE_DOWN, for example.
|
|
// arg 0: anim bank
|
|
// arg 1: horizontal speed (subpixel)
|
|
// arg 2: wave amplitude
|
|
// arg 3: first duration
|
|
// arg 4: delay before starting lunge
|
|
// arg 5: target x offset for lunge
|
|
// arg 6: lunge duration
|
|
void AnimTask_WindUpLunge(u8 taskId)
|
|
{
|
|
s16 wavePeriod = 0x8000 / gBattleAnimArgs[3];
|
|
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
|
|
{
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
gBattleAnimArgs[5] = -gBattleAnimArgs[5];
|
|
}
|
|
gTasks[taskId].data[0] = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
gTasks[taskId].data[1] = (gBattleAnimArgs[1] << 8) / gBattleAnimArgs[3];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[4] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[5] = (gBattleAnimArgs[5] << 8) / gBattleAnimArgs[6];
|
|
gTasks[taskId].data[6] = gBattleAnimArgs[6];
|
|
gTasks[taskId].data[7] = wavePeriod;
|
|
gTasks[taskId].func = AnimTask_WindUpLunge_Step1;
|
|
}
|
|
|
|
static void AnimTask_WindUpLunge_Step1(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
spriteId = gTasks[taskId].data[0];
|
|
gTasks[taskId].data[11] += gTasks[taskId].data[1];
|
|
gSprites[spriteId].pos2.x = gTasks[taskId].data[11] >> 8;
|
|
gSprites[spriteId].pos2.y = Sin((u8)(gTasks[taskId].data[10] >> 8), gTasks[taskId].data[2]);
|
|
gTasks[taskId].data[10] += gTasks[taskId].data[7];
|
|
if (--gTasks[taskId].data[3] == 0)
|
|
{
|
|
gTasks[taskId].func = AnimTask_WindUpLunge_Step2;
|
|
}
|
|
}
|
|
|
|
static void AnimTask_WindUpLunge_Step2(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
if (gTasks[taskId].data[4] > 0)
|
|
{
|
|
gTasks[taskId].data[4]--;
|
|
}
|
|
else
|
|
{
|
|
spriteId = gTasks[taskId].data[0];
|
|
gTasks[taskId].data[12] += gTasks[taskId].data[5];
|
|
gSprites[spriteId].pos2.x = (gTasks[taskId].data[12] >> 8) + (gTasks[taskId].data[11] >> 8);
|
|
if (--gTasks[taskId].data[6] == 0)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// To move a mon off-screen when pushed out by Roar/Whirlwind
|
|
void AnimTask_SlideOffScreen(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
switch (gBattleAnimArgs[0])
|
|
{
|
|
case ANIM_ATTACKER:
|
|
case ANIM_TARGET:
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[0]);
|
|
break;
|
|
case ANIM_ATK_PARTNER:
|
|
if (!IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimAttacker)))
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
spriteId = gBattlerSpriteIds[BATTLE_PARTNER(gBattleAnimAttacker)];
|
|
break;
|
|
case ANIM_DEF_PARTNER:
|
|
if (!IsBattlerSpriteVisible(BATTLE_PARTNER(gBattleAnimTarget)))
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
spriteId = gBattlerSpriteIds[BATTLE_PARTNER(gBattleAnimTarget)];
|
|
break;
|
|
default:
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
gTasks[taskId].data[0] = spriteId;
|
|
if (GetBattlerSide(gBattleAnimTarget) != B_SIDE_PLAYER)
|
|
{
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[1];
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[1] = -gBattleAnimArgs[1];
|
|
}
|
|
gTasks[taskId].func = AnimTask_SlideOffScreen_Step;
|
|
}
|
|
|
|
static void AnimTask_SlideOffScreen_Step(u8 taskId)
|
|
{
|
|
u8 spriteId = gTasks[taskId].data[0];
|
|
gSprites[spriteId].pos2.x += gTasks[taskId].data[1];
|
|
if (gSprites[spriteId].pos2.x + gSprites[spriteId].pos1.x + 0x20 > 0x130u)
|
|
{
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Task that facilitates translating the mon bg picture back and forth
|
|
// in a swaying motion (uses Sine wave). It can sway either horizontally
|
|
// or vertically, but not both.
|
|
// arg 0: direction (0 = horizontal, 1 = vertical)
|
|
// arg 1: wave amplitude
|
|
// arg 2: wave period
|
|
// arg 3: num sways
|
|
// arg 4: which mon (0 = attacker, 1`= target)
|
|
void AnimTask_SwayMon(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
if (GetBattlerSide(gBattleAnimAttacker) != B_SIDE_PLAYER)
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[4]);
|
|
gTasks[taskId].data[0] = gBattleAnimArgs[0];
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[4] = spriteId;
|
|
|
|
if (gBattleAnimArgs[4] == 0)
|
|
gTasks[taskId].data[5] = gBattleAnimAttacker;
|
|
else
|
|
gTasks[taskId].data[5] = gBattleAnimTarget;
|
|
|
|
gTasks[taskId].data[12] = 1;
|
|
gTasks[taskId].func = AnimTask_SwayMonStep;
|
|
}
|
|
|
|
static void AnimTask_SwayMonStep(u8 taskId)
|
|
{
|
|
s16 sineValue;
|
|
u8 spriteId;
|
|
int waveIndex;
|
|
u16 sineIndex;
|
|
|
|
spriteId = gTasks[taskId].data[4];
|
|
sineIndex = gTasks[taskId].data[10] + gTasks[taskId].data[2];
|
|
gTasks[taskId].data[10] = sineIndex;
|
|
waveIndex = sineIndex >> 8;
|
|
sineValue = Sin(waveIndex, gTasks[taskId].data[1]);
|
|
|
|
if (gTasks[taskId].data[0] == 0)
|
|
{
|
|
gSprites[spriteId].pos2.x = sineValue;
|
|
}
|
|
else
|
|
{
|
|
if (GetBattlerSide(gTasks[taskId].data[5]) == B_SIDE_PLAYER)
|
|
{
|
|
gSprites[spriteId].pos2.y = (sineValue >= 0) ? sineValue : -sineValue;
|
|
}
|
|
else
|
|
{
|
|
gSprites[spriteId].pos2.y = (sineValue >= 0) ? -sineValue : sineValue;
|
|
}
|
|
}
|
|
|
|
if (((waveIndex >= 0x80u) && (gTasks[taskId].data[11] == 0) && (gTasks[taskId].data[12] == 1))
|
|
|| ((waveIndex < 0x7fu) && (gTasks[taskId].data[11] == 1) && (gTasks[taskId].data[12] == 0)))
|
|
{
|
|
gTasks[taskId].data[11] ^= 1;
|
|
gTasks[taskId].data[12] ^= 1;
|
|
if (--gTasks[taskId].data[3] == 0)
|
|
{
|
|
gSprites[spriteId].pos2.x = 0;
|
|
gSprites[spriteId].pos2.y = 0;
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Scales a mon's sprite, and then scales back to its original dimensions.
|
|
// arg 0: x scale delta
|
|
// arg 1: y scale delta
|
|
// arg 2: duration
|
|
// arg 3: anim bank
|
|
// arg 4: sprite object mode
|
|
void AnimTask_ScaleMonAndRestore(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[3]);
|
|
PrepareBattlerSpriteForRotScale(spriteId, gBattleAnimArgs[4]);
|
|
gTasks[taskId].data[0] = gBattleAnimArgs[0];
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[2];
|
|
gTasks[taskId].data[4] = spriteId;
|
|
gTasks[taskId].data[10] = 0x100;
|
|
gTasks[taskId].data[11] = 0x100;
|
|
gTasks[taskId].func = AnimTask_ScaleMonAndRestore_Step;
|
|
}
|
|
|
|
static void AnimTask_ScaleMonAndRestore_Step(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
gTasks[taskId].data[10] += gTasks[taskId].data[0];
|
|
gTasks[taskId].data[11] += gTasks[taskId].data[1];
|
|
spriteId = gTasks[taskId].data[4];
|
|
SetSpriteRotScale(spriteId, gTasks[taskId].data[10], gTasks[taskId].data[11], 0);
|
|
if (--gTasks[taskId].data[2] == 0)
|
|
{
|
|
if (gTasks[taskId].data[3] > 0)
|
|
{
|
|
gTasks[taskId].data[0] = -gTasks[taskId].data[0];
|
|
gTasks[taskId].data[1] = -gTasks[taskId].data[1];
|
|
gTasks[taskId].data[2] = gTasks[taskId].data[3];
|
|
gTasks[taskId].data[3] = 0;
|
|
}
|
|
else
|
|
{
|
|
ResetSpriteRotScale(spriteId);
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AnimTask_RotateMonSpriteToSide(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[2]);
|
|
PrepareBattlerSpriteForRotScale(spriteId, ST_OAM_OBJ_NORMAL);
|
|
gTasks[taskId].data[1] = 0;
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[0];
|
|
if (gBattleAnimArgs[3] != 1)
|
|
{
|
|
gTasks[taskId].data[3] = 0;
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[0] * gBattleAnimArgs[1];
|
|
}
|
|
gTasks[taskId].data[4] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[5] = spriteId;
|
|
gTasks[taskId].data[6] = gBattleAnimArgs[3];
|
|
if (IsContest())
|
|
{
|
|
gTasks[taskId].data[7] = 1;
|
|
}
|
|
else
|
|
{
|
|
if (gBattleAnimArgs[2] == 0)
|
|
{
|
|
gTasks[taskId].data[7] = !GetBattlerSide(gBattleAnimAttacker);
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[7] = !GetBattlerSide(gBattleAnimTarget);
|
|
}
|
|
}
|
|
if (gTasks[taskId].data[7])
|
|
{
|
|
if (!IsContest())
|
|
{
|
|
gTasks[taskId].data[3] *= -1;
|
|
gTasks[taskId].data[4] *= -1;
|
|
}
|
|
}
|
|
gTasks[taskId].func = AnimTask_RotateMonSpriteToSide_Step;
|
|
}
|
|
|
|
// Rotates mon to side and back to original position. For Peck and when a held item activates
|
|
void AnimTask_RotateMonToSideAndRestore(u8 taskId)
|
|
{
|
|
u8 spriteId;
|
|
spriteId = GetAnimBattlerSpriteId(gBattleAnimArgs[2]);
|
|
PrepareBattlerSpriteForRotScale(spriteId, ST_OAM_OBJ_NORMAL);
|
|
gTasks[taskId].data[1] = 0;
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[0];
|
|
if (gBattleAnimArgs[2] == ANIM_ATTACKER)
|
|
{
|
|
if (GetBattlerSide(gBattleAnimAttacker))
|
|
{
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (GetBattlerSide(gBattleAnimTarget))
|
|
{
|
|
gBattleAnimArgs[1] = -gBattleAnimArgs[1];
|
|
}
|
|
}
|
|
if (gBattleAnimArgs[3] != 1)
|
|
{
|
|
gTasks[taskId].data[3] = 0;
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[3] = gBattleAnimArgs[0] * gBattleAnimArgs[1];
|
|
}
|
|
gTasks[taskId].data[4] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[5] = spriteId;
|
|
gTasks[taskId].data[6] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[7] = 1;
|
|
gTasks[taskId].data[3] *= -1;
|
|
gTasks[taskId].data[4] *= -1;
|
|
gTasks[taskId].func = AnimTask_RotateMonSpriteToSide_Step;
|
|
}
|
|
|
|
static void AnimTask_RotateMonSpriteToSide_Step(u8 taskId)
|
|
{
|
|
gTasks[taskId].data[3] += gTasks[taskId].data[4];
|
|
SetSpriteRotScale(gTasks[taskId].data[5], 0x100, 0x100, gTasks[taskId].data[3]);
|
|
if (gTasks[taskId].data[7])
|
|
{
|
|
SetBattlerSpriteYOffsetFromRotation(gTasks[taskId].data[5]);
|
|
}
|
|
if (++gTasks[taskId].data[1] >= gTasks[taskId].data[2])
|
|
{
|
|
switch (gTasks[taskId].data[6])
|
|
{
|
|
case 1:
|
|
ResetSpriteRotScale(gTasks[taskId].data[5]);
|
|
case 0:
|
|
default:
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
case 2:
|
|
gTasks[taskId].data[1] = 0;
|
|
gTasks[taskId].data[4] *= -1;
|
|
gTasks[taskId].data[6] = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AnimTask_ShakeTargetBasedOnMovePowerOrDmg(u8 taskId)
|
|
{
|
|
if (!gBattleAnimArgs[0])
|
|
{
|
|
gTasks[taskId].data[15] = gAnimMovePower / 12;
|
|
if (gTasks[taskId].data[15] < 1)
|
|
{
|
|
gTasks[taskId].data[15] = 1;
|
|
}
|
|
if (gTasks[taskId].data[15] > 16)
|
|
{
|
|
gTasks[taskId].data[15] = 16;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gTasks[taskId].data[15] = gAnimMoveDmg / 12;
|
|
if (gTasks[taskId].data[15] < 1)
|
|
{
|
|
gTasks[taskId].data[15] = 1;
|
|
}
|
|
if (gTasks[taskId].data[15] > 16)
|
|
{
|
|
gTasks[taskId].data[15] = 16;
|
|
}
|
|
}
|
|
gTasks[taskId].data[14] = gTasks[taskId].data[15] / 2;
|
|
gTasks[taskId].data[13] = gTasks[taskId].data[14] + (gTasks[taskId].data[15] & 1);
|
|
gTasks[taskId].data[12] = 0;
|
|
gTasks[taskId].data[10] = gBattleAnimArgs[3];
|
|
gTasks[taskId].data[11] = gBattleAnimArgs[4];
|
|
gTasks[taskId].data[7] = GetAnimBattlerSpriteId(ANIM_TARGET);
|
|
gTasks[taskId].data[8] = gSprites[gTasks[taskId].data[7]].pos2.x;
|
|
gTasks[taskId].data[9] = gSprites[gTasks[taskId].data[7]].pos2.y;
|
|
gTasks[taskId].data[0] = 0;
|
|
gTasks[taskId].data[1] = gBattleAnimArgs[1];
|
|
gTasks[taskId].data[2] = gBattleAnimArgs[2];
|
|
gTasks[taskId].func = AnimTask_ShakeTargetBasedOnMovePowerOrDmg_Step;
|
|
}
|
|
|
|
static void AnimTask_ShakeTargetBasedOnMovePowerOrDmg_Step(u8 taskId)
|
|
{
|
|
struct Task *task = &gTasks[taskId];
|
|
if (++task->data[0] > task->data[1])
|
|
{
|
|
task->data[0] = 0;
|
|
task->data[12] = (task->data[12] + 1) & 1;
|
|
if (task->data[10])
|
|
{
|
|
if (task->data[12])
|
|
{
|
|
gSprites[task->data[7]].pos2.x = task->data[8] + task->data[13];
|
|
}
|
|
else
|
|
{
|
|
gSprites[task->data[7]].pos2.x = task->data[8] - task->data[14];
|
|
}
|
|
}
|
|
if (task->data[11])
|
|
{
|
|
if (task->data[12])
|
|
{
|
|
gSprites[task->data[7]].pos2.y = task->data[15];
|
|
}
|
|
else
|
|
{
|
|
gSprites[task->data[7]].pos2.y = 0;
|
|
}
|
|
}
|
|
if (!--task->data[2])
|
|
{
|
|
gSprites[task->data[7]].pos2.x = 0;
|
|
gSprites[task->data[7]].pos2.y = 0;
|
|
DestroyAnimVisualTask(taskId);
|
|
return;
|
|
}
|
|
}
|
|
}
|