pokeemerald/src/battle_anim_sound_tasks.c

408 lines
12 KiB
C
Raw Normal View History

2018-04-18 16:03:15 +02:00
#include "global.h"
#include "battle.h"
#include "battle_anim.h"
#include "contest.h"
2018-11-14 01:01:50 +01:00
#include "sound.h"
#include "task.h"
#include "constants/battle_anim.h"
2018-04-18 16:03:15 +02:00
static void SoundTask_FireBlast_Step1(u8 taskId);
static void SoundTask_FireBlast_Step2(u8 taskId);
2020-02-22 00:22:18 +01:00
static void SoundTask_LoopSEAdjustPanning_Step(u8 taskId);
static void SoundTask_PlayDoubleCry_Step(u8 taskId);
static void SoundTask_PlayCryWithEcho_Step(u8 taskId);
static void SoundTask_AdjustPanningVar_Step(u8 taskId);
2018-04-18 16:03:15 +02:00
// Loops the specified sound effect and pans from the
// attacker to the target. The second specified sound effect
// is played at the very end. This task is effectively
// hardcoded to the move FIRE_BLAST due to the baked-in
// durations.
// arg 0: looped sound effect
// arg 1: ending sound effect
void SoundTask_FireBlast(u8 taskId)
2018-04-18 16:03:15 +02:00
{
s8 pan1, pan2, panIncrement;
gTasks[taskId].data[0] = gBattleAnimArgs[0];
gTasks[taskId].data[1] = gBattleAnimArgs[1];
pan1 = BattleAnimAdjustPanning(SOUND_PAN_ATTACKER);
pan2 = BattleAnimAdjustPanning(SOUND_PAN_TARGET);
2018-04-18 16:03:15 +02:00
panIncrement = CalculatePanIncrement(pan1, pan2, 2);
gTasks[taskId].data[2] = pan1;
gTasks[taskId].data[3] = pan2;
gTasks[taskId].data[4] = panIncrement;
gTasks[taskId].data[10] = 10;
gTasks[taskId].func = SoundTask_FireBlast_Step1;
2018-04-18 16:03:15 +02:00
}
static void SoundTask_FireBlast_Step1(u8 taskId)
2018-04-18 16:03:15 +02:00
{
s16 pan = gTasks[taskId].data[2];
s8 panIncrement = gTasks[taskId].data[4];
if (++gTasks[taskId].data[11] == 111)
{
gTasks[taskId].data[10] = 5;
gTasks[taskId].data[11] = 0;
gTasks[taskId].func = SoundTask_FireBlast_Step2;
2018-04-18 16:03:15 +02:00
}
else
{
if (++gTasks[taskId].data[10] == 11)
{
gTasks[taskId].data[10] = 0;
PlaySE12WithPanning(gTasks[taskId].data[0], pan);
}
pan += panIncrement;
2019-04-08 03:36:10 +02:00
gTasks[taskId].data[2] = KeepPanInRange(pan, panIncrement);
2018-04-18 16:03:15 +02:00
}
}
static void SoundTask_FireBlast_Step2(u8 taskId)
2018-04-18 16:03:15 +02:00
{
if (++gTasks[taskId].data[10] == 6)
{
s8 pan;
gTasks[taskId].data[10] = 0;
pan = BattleAnimAdjustPanning(SOUND_PAN_TARGET);
2018-04-18 16:03:15 +02:00
PlaySE12WithPanning(gTasks[taskId].data[1], pan);
if (++gTasks[taskId].data[11] == 2)
DestroyAnimSoundTask(taskId);
}
}
2020-02-20 06:04:42 +01:00
void SoundTask_LoopSEAdjustPanning(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 songId = gBattleAnimArgs[0];
s8 targetPan = gBattleAnimArgs[2];
s8 panIncrement = gBattleAnimArgs[3];
u8 r10 = gBattleAnimArgs[4];
u8 r7 = gBattleAnimArgs[5];
u8 r9 = gBattleAnimArgs[6];
s8 sourcePan = BattleAnimAdjustPanning(gBattleAnimArgs[1]);
targetPan = BattleAnimAdjustPanning(targetPan);
panIncrement = CalculatePanIncrement(sourcePan, targetPan, panIncrement);
gTasks[taskId].data[0] = songId;
gTasks[taskId].data[1] = sourcePan;
gTasks[taskId].data[2] = targetPan;
gTasks[taskId].data[3] = panIncrement;
gTasks[taskId].data[4] = r10;
gTasks[taskId].data[5] = r7;
gTasks[taskId].data[6] = r9;
gTasks[taskId].data[10] = 0;
gTasks[taskId].data[11] = sourcePan;
gTasks[taskId].data[12] = r9;
2020-02-22 00:22:18 +01:00
gTasks[taskId].func = SoundTask_LoopSEAdjustPanning_Step;
2022-08-17 17:44:20 +02:00
gTasks[taskId].func(taskId);
2018-04-18 16:03:15 +02:00
}
2020-02-22 00:22:18 +01:00
static void SoundTask_LoopSEAdjustPanning_Step(u8 taskId)
2018-04-18 16:03:15 +02:00
{
if (gTasks[taskId].data[12]++ == gTasks[taskId].data[6])
{
gTasks[taskId].data[12] = 0;
PlaySE12WithPanning(gTasks[taskId].data[0], gTasks[taskId].data[11]);
if (--gTasks[taskId].data[4] == 0)
{
DestroyAnimSoundTask(taskId);
return;
}
}
if (gTasks[taskId].data[10]++ == gTasks[taskId].data[5])
{
2019-04-08 03:38:01 +02:00
u16 dPan, oldPan;
2018-04-18 16:03:15 +02:00
gTasks[taskId].data[10] = 0;
2019-04-08 03:38:01 +02:00
dPan = gTasks[taskId].data[3];
2020-02-20 06:04:42 +01:00
oldPan = gTasks[taskId].data[11];
2019-04-08 03:36:10 +02:00
gTasks[taskId].data[11] = dPan + oldPan;
gTasks[taskId].data[11] = KeepPanInRange(gTasks[taskId].data[11], oldPan);
2018-04-18 16:03:15 +02:00
}
}
2020-02-20 06:04:42 +01:00
void SoundTask_PlayCryHighPitch(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 species = 0;
s8 pan = BattleAnimAdjustPanning(SOUND_PAN_ATTACKER);
2018-04-18 16:03:15 +02:00
if (IsContest())
{
if (gBattleAnimArgs[0] == ANIM_ATTACKER)
2020-08-14 01:10:23 +02:00
species = gContestResources->moveAnim->species;
// Destroying the task twice (here and at end of function)
// results in an incorrect value for gAnimVisualTaskCount
#ifndef BUGFIX
2018-04-18 16:03:15 +02:00
else
DestroyAnimVisualTask(taskId);
#endif
2018-04-18 16:03:15 +02:00
}
else
{
u8 battlerId;
2018-06-17 16:48:58 +02:00
// Get wanted battler.
2018-04-18 16:03:15 +02:00
if (gBattleAnimArgs[0] == ANIM_ATTACKER)
battlerId = gBattleAnimAttacker;
else if (gBattleAnimArgs[0] == ANIM_TARGET)
battlerId = gBattleAnimTarget;
else if (gBattleAnimArgs[0] == ANIM_ATK_PARTNER)
battlerId = BATTLE_PARTNER(gBattleAnimAttacker);
else
battlerId = BATTLE_PARTNER(gBattleAnimTarget);
2018-06-17 16:48:58 +02:00
// Check if battler is visible.
2018-04-18 16:03:15 +02:00
if ((gBattleAnimArgs[0] == ANIM_TARGET || gBattleAnimArgs[0] == ANIM_DEF_PARTNER) && !IsBattlerSpriteVisible(battlerId))
{
DestroyAnimVisualTask(taskId);
return;
}
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
species = GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
else
species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
}
if (species != SPECIES_NONE)
2021-11-07 19:54:44 +01:00
PlayCry_ByMode(species, pan, CRY_MODE_HIGH_PITCH);
2018-04-18 16:03:15 +02:00
DestroyAnimVisualTask(taskId);
}
2020-02-20 06:04:42 +01:00
void SoundTask_PlayDoubleCry(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 species = 0;
s8 pan = BattleAnimAdjustPanning(SOUND_PAN_ATTACKER);
2018-04-18 16:03:15 +02:00
if (IsContest())
{
if (gBattleAnimArgs[0] == ANIM_ATTACKER)
2020-08-14 01:10:23 +02:00
species = gContestResources->moveAnim->species;
// Destroying the task twice (here and at end of function)
// results in an incorrect value for gAnimVisualTaskCount
#ifndef BUGFIX
2018-04-18 16:03:15 +02:00
else
DestroyAnimVisualTask(taskId);
#endif
2018-04-18 16:03:15 +02:00
}
else
{
u8 battlerId;
2018-06-17 16:48:58 +02:00
// Get wanted battler.
2018-04-18 16:03:15 +02:00
if (gBattleAnimArgs[0] == ANIM_ATTACKER)
battlerId = gBattleAnimAttacker;
else if (gBattleAnimArgs[0] == ANIM_TARGET)
battlerId = gBattleAnimTarget;
else if (gBattleAnimArgs[0] == ANIM_ATK_PARTNER)
battlerId = BATTLE_PARTNER(gBattleAnimAttacker);
else
battlerId = BATTLE_PARTNER(gBattleAnimTarget);
2018-06-17 16:48:58 +02:00
// Check if battler is visible.
2018-04-18 16:03:15 +02:00
if ((gBattleAnimArgs[0] == ANIM_TARGET || gBattleAnimArgs[0] == ANIM_DEF_PARTNER) && !IsBattlerSpriteVisible(battlerId))
{
DestroyAnimVisualTask(taskId);
return;
}
if (GetBattlerSide(battlerId) != B_SIDE_PLAYER)
species = GetMonData(&gEnemyParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
else
species = GetMonData(&gPlayerParty[gBattlerPartyIndexes[battlerId]], MON_DATA_SPECIES);
}
gTasks[taskId].data[0] = gBattleAnimArgs[1];
gTasks[taskId].data[1] = species;
gTasks[taskId].data[2] = pan;
if (species != SPECIES_NONE)
{
2021-11-07 18:58:11 +01:00
if (gBattleAnimArgs[1] == DOUBLE_CRY_GROWL)
2021-11-07 19:54:44 +01:00
PlayCry_ByMode(species, pan, CRY_MODE_GROWL_1);
2021-11-07 18:58:11 +01:00
else // DOUBLE_CRY_ROAR
2021-11-07 19:54:44 +01:00
PlayCry_ByMode(species, pan, CRY_MODE_ROAR_1);
2018-04-18 16:03:15 +02:00
2020-02-22 00:22:18 +01:00
gTasks[taskId].func = SoundTask_PlayDoubleCry_Step;
2018-04-18 16:03:15 +02:00
}
else
{
DestroyAnimVisualTask(taskId);
}
}
2020-02-22 00:22:18 +01:00
static void SoundTask_PlayDoubleCry_Step(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 species = gTasks[taskId].data[1];
s8 pan = gTasks[taskId].data[2];
if (gTasks[taskId].data[9] < 2)
{
gTasks[taskId].data[9]++;
}
else
{
2021-11-07 18:58:11 +01:00
if (gTasks[taskId].data[0] == DOUBLE_CRY_GROWL)
2018-04-18 16:03:15 +02:00
{
if (!IsCryPlaying())
{
2021-11-07 19:54:44 +01:00
PlayCry_ByMode(species, pan, CRY_MODE_GROWL_2);
2018-04-18 16:03:15 +02:00
DestroyAnimVisualTask(taskId);
}
}
2021-11-07 18:58:11 +01:00
else // DOUBLE_CRY_ROAR
2018-04-18 16:03:15 +02:00
{
if (!IsCryPlaying())
{
2021-11-07 19:54:44 +01:00
PlayCry_ByMode(species, pan, CRY_MODE_ROAR_2);
2018-04-18 16:03:15 +02:00
DestroyAnimVisualTask(taskId);
}
}
}
}
2020-02-15 00:11:04 +01:00
void SoundTask_WaitForCry(u8 taskId)
2018-04-18 16:03:15 +02:00
{
if (gTasks[taskId].data[9] < 2)
{
gTasks[taskId].data[9]++;
}
else
{
if (!IsCryPlaying())
DestroyAnimVisualTask(taskId);
}
}
2021-11-07 19:54:44 +01:00
#define tSpecies data[1]
#define tPan data[2]
#define tState data[9]
#define tLastCry data[10] // If it's not the last cry, don't try to restore the BGM, because another is coming
2020-02-20 06:04:42 +01:00
void SoundTask_PlayCryWithEcho(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 species;
s8 pan;
2021-11-07 19:54:44 +01:00
gTasks[taskId].tLastCry = gBattleAnimArgs[0];
pan = BattleAnimAdjustPanning(SOUND_PAN_ATTACKER);
2018-04-18 16:03:15 +02:00
if (IsContest())
2020-08-14 01:10:23 +02:00
species = gContestResources->moveAnim->species;
2018-04-18 16:03:15 +02:00
else
species = gAnimBattlerSpecies[gBattleAnimAttacker];
2021-11-07 19:54:44 +01:00
gTasks[taskId].tSpecies = species;
gTasks[taskId].tPan = pan;
2018-04-18 16:03:15 +02:00
if (species != SPECIES_NONE)
2020-02-22 00:22:18 +01:00
gTasks[taskId].func = SoundTask_PlayCryWithEcho_Step;
2018-04-18 16:03:15 +02:00
else
DestroyAnimVisualTask(taskId);
}
2020-02-22 00:22:18 +01:00
static void SoundTask_PlayCryWithEcho_Step(u8 taskId)
2018-04-18 16:03:15 +02:00
{
2021-11-07 19:54:44 +01:00
u16 species = gTasks[taskId].tSpecies;
s8 pan = gTasks[taskId].tPan;
2018-04-18 16:03:15 +02:00
2021-11-07 18:58:11 +01:00
// Note the cases are not in order of execution
2021-11-07 19:54:44 +01:00
switch (gTasks[taskId].tState)
2018-04-18 16:03:15 +02:00
{
case 2:
2021-11-10 17:59:15 +01:00
PlayCry_DuckNoRestore(species, pan, CRY_MODE_ECHO_START);
2021-11-07 19:54:44 +01:00
gTasks[taskId].tState++;
2018-04-18 16:03:15 +02:00
break;
case 1:
case 3:
case 4:
2021-11-07 19:54:44 +01:00
gTasks[taskId].tState++;
2018-04-18 16:03:15 +02:00
break;
case 5:
if (IsCryPlaying())
break;
case 0:
StopCryAndClearCrySongs();
2021-11-07 19:54:44 +01:00
gTasks[taskId].tState++;
2018-04-18 16:03:15 +02:00
break;
default:
2021-11-07 19:54:44 +01:00
if (!gTasks[taskId].tLastCry)
2021-11-10 17:59:15 +01:00
PlayCry_DuckNoRestore(species, pan, CRY_MODE_ECHO_END);
2018-04-18 16:03:15 +02:00
else
2021-11-10 17:59:15 +01:00
PlayCry_ByMode(species, pan, CRY_MODE_ECHO_END);
2018-04-18 16:03:15 +02:00
DestroyAnimVisualTask(taskId);
break;
}
}
2021-11-07 19:54:44 +01:00
#undef tSpecies
#undef tPan
#undef tState
#undef tLastCry
2020-02-15 00:11:04 +01:00
void SoundTask_PlaySE1WithPanning(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 songId = gBattleAnimArgs[0];
s8 pan = BattleAnimAdjustPanning(gBattleAnimArgs[1]);
PlaySE1WithPanning(songId, pan);
DestroyAnimVisualTask(taskId);
}
2020-02-15 00:11:04 +01:00
void SoundTask_PlaySE2WithPanning(u8 taskId)
2018-04-18 16:03:15 +02:00
{
u16 songId = gBattleAnimArgs[0];
s8 pan = BattleAnimAdjustPanning(gBattleAnimArgs[1]);
PlaySE2WithPanning(songId, pan);
DestroyAnimVisualTask(taskId);
}
// Adjusts panning and assigns it to gAnimCustomPanning. Doesnt play sound.
2020-02-20 06:04:42 +01:00
// Used by Confuse Ray and Will-O-Wisp (see uses of gAnimCustomPanning)
void SoundTask_AdjustPanningVar(u8 taskId)
2018-04-18 16:03:15 +02:00
{
s8 targetPan = gBattleAnimArgs[1];
s8 panIncrement = gBattleAnimArgs[2];
u16 r9 = gBattleAnimArgs[3];
s8 sourcePan = BattleAnimAdjustPanning(gBattleAnimArgs[0]);
targetPan = BattleAnimAdjustPanning(targetPan);
panIncrement = CalculatePanIncrement(sourcePan, targetPan, panIncrement);
gTasks[taskId].data[1] = sourcePan;
gTasks[taskId].data[2] = targetPan;
gTasks[taskId].data[3] = panIncrement;
gTasks[taskId].data[5] = r9;
gTasks[taskId].data[10] = 0;
gTasks[taskId].data[11] = sourcePan;
2020-02-22 00:22:18 +01:00
gTasks[taskId].func = SoundTask_AdjustPanningVar_Step;
2022-08-17 17:44:20 +02:00
gTasks[taskId].func(taskId);
2018-04-18 16:03:15 +02:00
}
2020-02-22 00:22:18 +01:00
static void SoundTask_AdjustPanningVar_Step(u8 taskId)
2018-04-18 16:03:15 +02:00
{
2019-04-08 03:36:10 +02:00
u16 panIncrement = gTasks[taskId].data[3];
2018-04-18 16:03:15 +02:00
if (gTasks[taskId].data[10]++ == gTasks[taskId].data[5])
{
2019-04-08 03:38:01 +02:00
u16 oldPan;
2018-04-18 16:03:15 +02:00
gTasks[taskId].data[10] = 0;
2019-04-08 03:38:01 +02:00
oldPan = gTasks[taskId].data[11];
gTasks[taskId].data[11] = panIncrement + oldPan;
2019-04-08 03:36:10 +02:00
gTasks[taskId].data[11] = KeepPanInRange(gTasks[taskId].data[11], oldPan);
2018-04-18 16:03:15 +02:00
}
2020-02-20 06:04:42 +01:00
gAnimCustomPanning = gTasks[taskId].data[11];
2018-04-18 16:03:15 +02:00
if (gTasks[taskId].data[11] == gTasks[taskId].data[2])
DestroyAnimVisualTask(taskId);
}