pokeemerald/src/fldeff_misc.c

1321 lines
37 KiB
C
Raw Normal View History

2018-11-20 09:51:54 -05:00
#include "global.h"
#include "gpu_regs.h"
#include "palette.h"
#include "script.h"
#include "sound.h"
#include "task.h"
#include "strings.h"
#include "party_menu.h"
2018-12-09 19:22:59 -05:00
#include "fieldmap.h"
#include "field_effect.h"
#include "field_camera.h"
#include "field_player_avatar.h"
2018-12-24 21:59:05 +01:00
#include "fldeff.h"
2018-12-22 19:33:45 +01:00
#include "fldeff_misc.h"
2018-12-09 19:22:59 -05:00
#include "secret_base.h"
#include "event_data.h"
2018-12-22 19:33:45 +01:00
#include "event_scripts.h"
2018-12-09 19:22:59 -05:00
#include "event_object_movement.h"
#include "metatile_behavior.h"
2018-12-09 19:22:59 -05:00
#include "string_util.h"
#include "constants/field_effects.h"
#include "constants/metatile_behaviors.h"
#include "constants/metatile_labels.h"
#include "constants/songs.h"
2018-11-20 09:51:54 -05:00
EWRAM_DATA struct MapPosition gPlayerFacingPosition = {0};
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
static void Task_ComputerScreenOpenEffect(u8);
static void Task_ComputerScreenCloseEffect(u8);
static void CreateComputerScreenEffectTask(TaskFunc, u16, u16, u8);
2018-12-22 19:33:45 +01:00
static void Task_SecretBasePCTurnOn(u8);
2018-12-22 19:33:45 +01:00
static void Task_PopSecretBaseBalloon(u8);
static void DoBalloonSoundEffect(s16);
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
static void Task_WateringBerryTreeAnim_Start(u8);
static void Task_WateringBerryTreeAnim_Continue(u8);
static void Task_WateringBerryTreeAnim_End(u8);
2018-12-22 19:33:45 +01:00
static void FieldCallback_SecretBaseCave(void);
2020-06-29 11:41:09 -04:00
static void SpriteCB_CaveEntranceInit(struct Sprite *);
static void SpriteCB_CaveEntranceOpen(struct Sprite *);
static void SpriteCB_CaveEntranceEnd(struct Sprite *);
static void StartSecretBaseCaveFieldEffect(void);
2018-12-22 19:33:45 +01:00
static void FieldCallback_SecretBaseTree(void);
2020-06-29 11:41:09 -04:00
static void SpriteCB_TreeEntranceInit(struct Sprite *);
static void SpriteCB_TreeEntranceOpen(struct Sprite *);
static void SpriteCB_TreeEntranceEnd(struct Sprite *);
static void StartSecretBaseTreeFieldEffect(void);
2018-12-22 19:33:45 +01:00
static void FieldCallback_SecretBaseShrub(void);
2020-06-29 11:41:09 -04:00
static void SpriteCB_ShrubEntranceInit(struct Sprite *);
static void SpriteCB_ShrubEntranceOpen(struct Sprite *);
static void SpriteCB_ShrubEntranceEnd(struct Sprite *);
static void StartSecretBaseShrubFieldEffect(void);
2020-06-29 11:41:09 -04:00
static void SpriteCB_SandPillar_BreakTop(struct Sprite *);
static void SpriteCB_SandPillar_BreakBase(struct Sprite *);
static void SpriteCB_SandPillar_End(struct Sprite *);
static const u8 sSecretPowerCave_Gfx[] = INCBIN_U8("graphics/field_effects/pics/secret_power_cave.4bpp");
static const u8 sFiller[32] = {0};
2020-07-02 04:59:52 -04:00
static const u16 sSecretPowerCave_Pal[] = INCBIN_U16("graphics/field_effects/palettes/secret_power_cave.gbapal");
2020-06-29 11:41:09 -04:00
static const u8 sSecretPowerShrub_Gfx[] = INCBIN_U8("graphics/field_effects/pics/secret_power_shrub.4bpp");
static const u8 sSecretPowerTree_Gfx[] = INCBIN_U8("graphics/field_effects/pics/secret_power_tree.4bpp");
2020-07-02 04:59:52 -04:00
static const u16 sSecretPowerPlant_Pal[] = INCBIN_U16("graphics/field_effects/palettes/secret_power_plant.gbapal");
2020-06-29 11:41:09 -04:00
2020-07-02 04:59:52 -04:00
// TODO: These should also be combined into a single image, not matching for some reason
2020-06-29 11:41:09 -04:00
static const u8 sSandPillar0_Gfx[] = INCBIN_U8("graphics/field_effects/pics/sand_pillar/0.4bpp");
static const u8 sSandPillar1_Gfx[] = INCBIN_U8("graphics/field_effects/pics/sand_pillar/1.4bpp");
static const u8 sSandPillar2_Gfx[] = INCBIN_U8("graphics/field_effects/pics/sand_pillar/2.4bpp");
static const struct OamData sOam_SecretPower =
2018-12-09 19:22:59 -05:00
{
.y = 0,
.x = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(16x16),
.size = SPRITE_SIZE(16x16),
2018-12-09 19:22:59 -05:00
.priority = 2,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_SecretPowerCave[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(1, 8),
ANIMCMD_FRAME(2, 8),
ANIMCMD_FRAME(3, 8),
ANIMCMD_FRAME(4, 8),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_VineDropLeft[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(1, 8),
ANIMCMD_FRAME(2, 8),
ANIMCMD_FRAME(3, 8),
ANIMCMD_FRAME(4, 8),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_VineRiseLeft[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(4, 8),
ANIMCMD_FRAME(3, 8),
ANIMCMD_FRAME(2, 8),
ANIMCMD_FRAME(1, 8),
ANIMCMD_FRAME(0, 8),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_VineDropRight[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(0, 8, .hFlip = TRUE),
ANIMCMD_FRAME(1, 8, .hFlip = TRUE),
ANIMCMD_FRAME(2, 8, .hFlip = TRUE),
ANIMCMD_FRAME(3, 8, .hFlip = TRUE),
ANIMCMD_FRAME(4, 8, .hFlip = TRUE),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_VineRiseRight[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(4, 8, .hFlip = TRUE),
ANIMCMD_FRAME(3, 8, .hFlip = TRUE),
ANIMCMD_FRAME(2, 8, .hFlip = TRUE),
ANIMCMD_FRAME(1, 8, .hFlip = TRUE),
ANIMCMD_FRAME(0, 8, .hFlip = TRUE),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_SecretPowerShrub[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(1, 8),
ANIMCMD_FRAME(2, 8),
ANIMCMD_FRAME(3, 8),
ANIMCMD_FRAME(4, 8),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd *const sAnimTable_SecretPowerCave[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
sAnim_SecretPowerCave,
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd *const sAnimTable_SecretPowerTree[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
sAnim_VineDropLeft,
sAnim_VineRiseLeft,
sAnim_VineDropRight,
sAnim_VineRiseRight,
2018-12-09 19:22:59 -05:00
};
2018-11-20 09:51:54 -05:00
2020-06-29 11:41:09 -04:00
static const union AnimCmd *const sAnimTable_SecretPowerShrub[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
sAnim_SecretPowerShrub,
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteFrameImage sPicTable_SecretPowerCave[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
overworld_frame(sSecretPowerCave_Gfx, 2, 2, 0),
overworld_frame(sSecretPowerCave_Gfx, 2, 2, 1),
overworld_frame(sSecretPowerCave_Gfx, 2, 2, 2),
overworld_frame(sSecretPowerCave_Gfx, 2, 2, 3),
overworld_frame(sSecretPowerCave_Gfx, 2, 2, 4),
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteFrameImage sPicTable_SecretPowerTree[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
overworld_frame(sSecretPowerTree_Gfx, 2, 2, 0),
overworld_frame(sSecretPowerTree_Gfx, 2, 2, 1),
overworld_frame(sSecretPowerTree_Gfx, 2, 2, 2),
overworld_frame(sSecretPowerTree_Gfx, 2, 2, 3),
overworld_frame(sSecretPowerTree_Gfx, 2, 2, 4),
// 6th frame exists but isnt accessed, the tree vine metatile is used instead
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteFrameImage sPicTable_SecretPowerShrub[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
overworld_frame(sSecretPowerShrub_Gfx, 2, 2, 0),
overworld_frame(sSecretPowerShrub_Gfx, 2, 2, 1),
overworld_frame(sSecretPowerShrub_Gfx, 2, 2, 2),
overworld_frame(sSecretPowerShrub_Gfx, 2, 2, 3),
overworld_frame(sSecretPowerShrub_Gfx, 2, 2, 4),
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteTemplate sSpriteTemplate_SecretPowerCave =
2018-12-09 19:22:59 -05:00
{
.tileTag = 0xFFFF,
2020-07-02 04:59:52 -04:00
.paletteTag = FLDEFF_PAL_TAG_SECRET_POWER_TREE,
2020-06-29 11:41:09 -04:00
.oam = &sOam_SecretPower,
.anims = sAnimTable_SecretPowerCave,
.images = sPicTable_SecretPowerCave,
2018-12-09 19:22:59 -05:00
.affineAnims = gDummySpriteAffineAnimTable,
2020-06-29 11:41:09 -04:00
.callback = SpriteCB_CaveEntranceInit,
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteTemplate sSpriteTemplate_SecretPowerTree =
2018-12-09 19:22:59 -05:00
{
.tileTag = 0xFFFF,
2020-07-02 04:59:52 -04:00
.paletteTag = FLDEFF_PAL_TAG_SECRET_POWER_PLANT,
2020-06-29 11:41:09 -04:00
.oam = &sOam_SecretPower,
.anims = sAnimTable_SecretPowerTree,
.images = sPicTable_SecretPowerTree,
2018-12-09 19:22:59 -05:00
.affineAnims = gDummySpriteAffineAnimTable,
2020-06-29 11:41:09 -04:00
.callback = SpriteCB_TreeEntranceInit,
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteTemplate sSpriteTemplate_SecretPowerShrub =
2018-12-09 19:22:59 -05:00
{
.tileTag = 0xFFFF,
2020-07-02 04:59:52 -04:00
.paletteTag = FLDEFF_PAL_TAG_SECRET_POWER_PLANT,
2020-06-29 11:41:09 -04:00
.oam = &sOam_SecretPower,
.anims = sAnimTable_SecretPowerShrub,
.images = sPicTable_SecretPowerShrub,
2018-12-09 19:22:59 -05:00
.affineAnims = gDummySpriteAffineAnimTable,
2020-06-29 11:41:09 -04:00
.callback = SpriteCB_ShrubEntranceInit,
2018-12-09 19:22:59 -05:00
};
2020-07-02 04:59:52 -04:00
const struct SpritePalette gSpritePalette_SecretPower_Cave = {sSecretPowerCave_Pal, FLDEFF_PAL_TAG_SECRET_POWER_TREE};
const struct SpritePalette gSpritePalette_SecretPower_Plant = {sSecretPowerPlant_Pal, FLDEFF_PAL_TAG_SECRET_POWER_PLANT};
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
static const struct OamData sOam_SandPillar =
2018-12-09 19:22:59 -05:00
{
.x = 0,
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(16x32),
.size = SPRITE_SIZE(16x32),
2018-12-09 19:22:59 -05:00
.priority = 2,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_SandPillar[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(0, 6),
ANIMCMD_FRAME(1, 6),
ANIMCMD_FRAME(2, 6),
ANIMCMD_END,
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd *const sAnimTable_SandPillar[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
sAnim_SandPillar,
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteFrameImage sPicTable_SandPillar[] =
2018-12-09 19:22:59 -05:00
{
2020-07-02 04:59:52 -04:00
{sSandPillar0_Gfx, sizeof(sSandPillar0_Gfx)},
{sSandPillar1_Gfx, sizeof(sSandPillar1_Gfx)},
{sSandPillar2_Gfx, sizeof(sSandPillar2_Gfx)},
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteTemplate sSpriteTemplate_SandPillar =
2018-12-09 19:22:59 -05:00
{
.tileTag = 0xFFFF,
2020-07-02 04:59:52 -04:00
.paletteTag = FLDEFF_PAL_TAG_SAND_PILLAR,
2020-06-29 11:41:09 -04:00
.oam = &sOam_SandPillar,
.anims = sAnimTable_SandPillar,
.images = sPicTable_SandPillar,
2018-12-09 19:22:59 -05:00
.affineAnims = gDummySpriteAffineAnimTable,
2020-06-29 11:41:09 -04:00
.callback = SpriteCB_SandPillar_BreakTop,
2018-12-09 19:22:59 -05:00
};
2020-07-02 04:59:52 -04:00
const struct SpritePalette gSpritePalette_SandPillar = {gTilesetPalettes_SecretBase[5], FLDEFF_PAL_TAG_SAND_PILLAR};
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
static const u8 sRecordMixLights_Gfx[] = INCBIN_U8("graphics/field_effects/pics/record_mix_lights.4bpp");
static const u16 sRecordMixLights_Pal[] = INCBIN_U16("graphics/field_effects/palettes/record_mix_lights.gbapal");
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
static const struct SpriteFrameImage sPicTable_RecordMixLights[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
overworld_frame(sRecordMixLights_Gfx, 4, 1, 0),
overworld_frame(sRecordMixLights_Gfx, 4, 1, 1),
overworld_frame(sRecordMixLights_Gfx, 4, 1, 2),
2018-12-09 19:22:59 -05:00
};
2020-07-02 04:59:52 -04:00
static const struct SpritePalette sSpritePalette_RecordMixLights = {sRecordMixLights_Pal, 0x1000};
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
static const union AnimCmd sAnim_RecordMixLights[] =
2018-12-09 19:22:59 -05:00
{
ANIMCMD_FRAME(0, 30),
ANIMCMD_FRAME(1, 30),
ANIMCMD_FRAME(2, 30),
ANIMCMD_JUMP(0),
};
2020-06-29 11:41:09 -04:00
static const union AnimCmd *const sAnimTable_RecordMixLights[] =
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
sAnim_RecordMixLights,
2018-12-09 19:22:59 -05:00
};
2020-06-29 11:41:09 -04:00
static const struct SpriteTemplate sSpriteTemplate_RecordMixLights =
2018-12-09 19:22:59 -05:00
{
.tileTag = 0xFFFF,
2020-07-02 04:59:52 -04:00
.paletteTag = 0x1000,
.oam = &gObjectEventBaseOam_32x8,
2020-06-29 11:41:09 -04:00
.anims = sAnimTable_RecordMixLights,
.images = sPicTable_RecordMixLights,
2018-12-09 19:22:59 -05:00
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
};
2018-11-20 09:51:54 -05:00
2020-06-29 11:41:09 -04:00
// For accessing pokemon storage PC or the Hall of Fame PC
void ComputerScreenOpenEffect(u16 increment, u16 unused, u8 priority)
2018-11-20 09:51:54 -05:00
{
2020-06-29 11:41:09 -04:00
CreateComputerScreenEffectTask(Task_ComputerScreenOpenEffect, increment, unused, priority);
2018-11-20 09:51:54 -05:00
}
2020-06-29 11:41:09 -04:00
void ComputerScreenCloseEffect(u16 increment, u16 unused, u8 priority)
2018-11-20 09:51:54 -05:00
{
2020-06-29 11:41:09 -04:00
CreateComputerScreenEffectTask(Task_ComputerScreenCloseEffect, increment, unused, priority);
2018-11-20 09:51:54 -05:00
}
2020-06-29 11:41:09 -04:00
bool8 IsComputerScreenOpenEffectActive(void)
2018-11-20 09:51:54 -05:00
{
2020-06-29 11:41:09 -04:00
return FuncIsActiveTask(Task_ComputerScreenOpenEffect);
2018-11-20 09:51:54 -05:00
}
2020-06-29 11:41:09 -04:00
bool8 IsComputerScreenCloseEffectActive(void)
2018-11-20 09:51:54 -05:00
{
2020-06-29 11:41:09 -04:00
return FuncIsActiveTask(Task_ComputerScreenCloseEffect);
2018-11-20 09:51:54 -05:00
}
2020-06-29 11:41:09 -04:00
#define tState data[0]
#define tHorzIncrement data[1]
#define tVertIncrement data[2]
#define tWinLeft data[3]
#define tWinRight data[4]
#define tWinTop data[5]
#define tWinBottom data[6]
#define tBlendCnt data[7]
#define tBlendY data[8]
static void CreateComputerScreenEffectTask(void (*taskfunc) (u8), u16 increment, u16 unused, u8 priority)
2018-11-20 09:51:54 -05:00
{
2020-06-29 11:41:09 -04:00
u8 taskId = CreateTask(taskfunc, priority);
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
gTasks[taskId].tState = 0;
gTasks[taskId].tHorzIncrement = increment == 0 ? 16 : increment;
gTasks[taskId].tVertIncrement = increment == 0 ? 20 : increment;
2018-12-09 19:22:59 -05:00
gTasks[taskId].func(taskId);
}
2020-06-29 11:41:09 -04:00
static void Task_ComputerScreenOpenEffect(u8 taskId)
2018-12-09 19:22:59 -05:00
{
struct Task *task = &gTasks[taskId];
2020-06-29 11:41:09 -04:00
switch (task->tState)
2018-12-09 19:22:59 -05:00
{
2018-12-19 20:52:41 -05:00
case 0:
2020-06-29 11:41:09 -04:00
task->tWinLeft = DISPLAY_WIDTH / 2;
task->tWinRight = DISPLAY_WIDTH / 2;
task->tWinTop = DISPLAY_HEIGHT / 2;
task->tWinBottom = DISPLAY_HEIGHT / 2 + 1;
2018-12-09 19:22:59 -05:00
SetGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON);
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_WIN0H, WIN_RANGE(task->tWinLeft, task->tWinRight));
SetGpuReg(REG_OFFSET_WIN0V, WIN_RANGE(task->tWinTop, task->tWinBottom));
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG_ALL | WININ_WIN0_OBJ | WININ_WIN0_CLR);
2018-12-09 19:22:59 -05:00
SetGpuReg(REG_OFFSET_WINOUT, 0);
break;
2018-12-19 20:52:41 -05:00
case 1:
2020-06-29 11:41:09 -04:00
task->tBlendCnt = GetGpuReg(REG_OFFSET_BLDCNT);
task->tBlendY = GetGpuReg(REG_OFFSET_BLDY);
2018-12-09 19:22:59 -05:00
2018-12-26 13:43:07 +01:00
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_TGT1_ALL | BLDCNT_EFFECT_LIGHTEN);
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_BLDY, 16);
2018-12-09 19:22:59 -05:00
break;
case 2:
2020-06-29 11:41:09 -04:00
task->tWinLeft -= task->tHorzIncrement;
task->tWinRight += task->tHorzIncrement;
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinLeft < 1 || task->tWinRight > DISPLAY_WIDTH - 1)
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
task->tWinLeft = 0;
task->tWinRight = DISPLAY_WIDTH;
2018-12-09 19:22:59 -05:00
SetGpuReg(REG_OFFSET_BLDY, 0);
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_BLDCNT, task->tBlendCnt);
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 0, 0);
2018-12-19 20:52:41 -05:00
gPlttBufferFaded[0] = 0;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_WIN0H, WIN_RANGE(task->tWinLeft, task->tWinRight));
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinLeft != 0)
2018-12-19 22:19:54 -05:00
return;
2018-12-09 19:22:59 -05:00
break;
case 3:
2020-06-29 11:41:09 -04:00
task->tWinTop -= task->tVertIncrement;
task->tWinBottom += task->tVertIncrement;
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinTop < 1 || task->tWinBottom > DISPLAY_HEIGHT - 1)
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
task->tWinTop = 0;
task->tWinBottom = DISPLAY_HEIGHT;
2018-12-09 19:22:59 -05:00
ClearGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON);
}
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_WIN0V, WIN_RANGE(task->tWinTop, task->tWinBottom));
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinTop != 0)
2018-12-19 22:19:54 -05:00
return;
2018-12-09 19:22:59 -05:00
break;
default:
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_BLDCNT, task->tBlendCnt);
2018-12-09 19:22:59 -05:00
DestroyTask(taskId);
return;
}
2020-06-29 11:41:09 -04:00
task->tState++;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
static void Task_ComputerScreenCloseEffect(u8 taskId)
2018-12-09 19:22:59 -05:00
{
struct Task *task = &gTasks[taskId];
2020-06-29 11:41:09 -04:00
switch (task->tState)
2018-12-09 19:22:59 -05:00
{
case 0:
2020-06-29 11:41:09 -04:00
gPlttBufferFaded[0] = 0;
2018-12-09 19:22:59 -05:00
break;
case 1:
2020-06-29 11:41:09 -04:00
task->tWinLeft = 0;
task->tWinRight = DISPLAY_WIDTH;
task->tWinTop = 0;
task->tWinBottom = DISPLAY_HEIGHT;
2018-12-09 19:22:59 -05:00
SetGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON);
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_WIN0H, WIN_RANGE(task->tWinLeft, task->tWinRight));
SetGpuReg(REG_OFFSET_WIN0V, WIN_RANGE(task->tWinTop, task->tWinBottom));
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG_ALL | WININ_WIN0_OBJ | WININ_WIN0_CLR);
2018-12-09 19:22:59 -05:00
SetGpuReg(REG_OFFSET_WINOUT, 0);
break;
case 2:
2020-06-29 11:41:09 -04:00
task->tWinTop += task->tVertIncrement;
task->tWinBottom -= task->tVertIncrement;
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinTop >= DISPLAY_HEIGHT / 2 || task->tWinBottom <= DISPLAY_HEIGHT / 2 + 1)
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
task->tWinTop = DISPLAY_HEIGHT / 2;
task->tWinBottom = DISPLAY_HEIGHT / 2 + 1;
2018-12-26 13:43:07 +01:00
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_TGT1_ALL | BLDCNT_EFFECT_LIGHTEN);
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_BLDY, 16);
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_WIN0V, WIN_RANGE(task->tWinTop, task->tWinBottom));
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinTop != DISPLAY_HEIGHT / 2)
2018-12-19 22:19:54 -05:00
return;
2018-12-09 19:22:59 -05:00
break;
case 3:
2020-06-29 11:41:09 -04:00
task->tWinLeft += task->tHorzIncrement;
task->tWinRight -= task->tHorzIncrement;
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinLeft >= DISPLAY_WIDTH / 2 || task->tWinRight <= DISPLAY_WIDTH / 2)
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
task->tWinLeft = DISPLAY_WIDTH / 2;
task->tWinRight = DISPLAY_WIDTH / 2;
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 16, 0);
2018-12-09 19:22:59 -05:00
gPlttBufferFaded[0] = 0;
}
2020-06-29 11:41:09 -04:00
SetGpuReg(REG_OFFSET_WIN0H, WIN_RANGE(task->tWinLeft, task->tWinRight));
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
if (task->tWinLeft != DISPLAY_WIDTH / 2)
2018-12-19 22:19:54 -05:00
return;
2018-12-09 19:22:59 -05:00
break;
default:
ClearGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON);
SetGpuReg(REG_OFFSET_BLDY, 0);
SetGpuReg(REG_OFFSET_BLDCNT, 0);
DestroyTask(taskId);
return;
}
2020-06-29 11:41:09 -04:00
task->tState++;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
#undef tState
#undef tHorzIncrement
#undef tVertIncrement
#undef tWinLeft
#undef tWinRight
#undef tWinTop
#undef tWinBottom
#undef tBlendCnt
#undef tBlendY
2018-12-22 19:33:45 +01:00
static void SetCurrentSecretBase(void)
2018-12-09 19:22:59 -05:00
{
2019-04-05 16:11:24 -05:00
SetCurSecretBaseIdFromPosition(&gPlayerFacingPosition, gMapHeader.events);
TrySetCurSecretBaseIndex();
2018-12-09 19:22:59 -05:00
}
2018-12-22 19:33:45 +01:00
static void AdjustSecretPowerSpritePixelOffsets(void)
2018-12-09 19:22:59 -05:00
{
if (gPlayerAvatar.flags & (PLAYER_AVATAR_FLAG_MACH_BIKE | PLAYER_AVATAR_FLAG_ACRO_BIKE))
2018-12-09 19:22:59 -05:00
{
switch (gFieldEffectArguments[1])
{
case DIR_SOUTH:
gFieldEffectArguments[5] = 16;
gFieldEffectArguments[6] = 40;
break;
case DIR_NORTH:
gFieldEffectArguments[5] = 16;
gFieldEffectArguments[6] = 8;
break;
case DIR_WEST:
gFieldEffectArguments[5] = -8;
gFieldEffectArguments[6] = 24;
break;
case DIR_EAST:
gFieldEffectArguments[5] = 24;
gFieldEffectArguments[6] = 24;
break;
}
}
else
{
switch (gFieldEffectArguments[1])
{
case DIR_SOUTH:
gFieldEffectArguments[5] = 8;
gFieldEffectArguments[6] = 40;
break;
case DIR_NORTH:
gFieldEffectArguments[5] = 8;
gFieldEffectArguments[6] = 8;
break;
case DIR_WEST:
gFieldEffectArguments[5] = -8;
gFieldEffectArguments[6] = 24;
break;
case DIR_EAST:
gFieldEffectArguments[5] = 24;
gFieldEffectArguments[6] = 24;
break;
}
}
}
bool8 SetUpFieldMove_SecretPower(void)
2018-12-09 19:22:59 -05:00
{
u8 mb;
2019-04-05 16:11:24 -05:00
CheckPlayerHasSecretBase();
2018-12-09 19:22:59 -05:00
if (gSpecialVar_Result == 1 || GetPlayerFacingDirection() != DIR_NORTH)
return FALSE;
GetXYCoordsOneStepInFrontOfPlayer(&gPlayerFacingPosition.x, &gPlayerFacingPosition.y);
mb = MapGridGetMetatileBehaviorAt(gPlayerFacingPosition.x, gPlayerFacingPosition.y);
if (MetatileBehavior_IsSecretBaseCave(mb) == TRUE)
{
SetCurrentSecretBase();
2018-12-09 19:22:59 -05:00
gFieldCallback2 = FieldCallback_PrepareFadeInFromMenu;
gPostMenuFieldCallback = FieldCallback_SecretBaseCave;
2018-12-09 19:22:59 -05:00
return TRUE;
}
if (MetatileBehavior_IsSecretBaseTree(mb) == TRUE)
{
SetCurrentSecretBase();
2018-12-09 19:22:59 -05:00
gFieldCallback2 = FieldCallback_PrepareFadeInFromMenu;
gPostMenuFieldCallback = FieldCallback_SecretBaseTree;
2018-12-09 19:22:59 -05:00
return TRUE;
}
if (MetatileBehavior_IsSecretBaseShrub(mb) == TRUE)
{
SetCurrentSecretBase();
2018-12-09 19:22:59 -05:00
gFieldCallback2 = FieldCallback_PrepareFadeInFromMenu;
gPostMenuFieldCallback = FieldCallback_SecretBaseShrub;
2018-12-09 19:22:59 -05:00
return TRUE;
}
return FALSE;
}
2018-12-22 19:33:45 +01:00
static void FieldCallback_SecretBaseCave(void)
2018-12-09 19:22:59 -05:00
{
gFieldEffectArguments[0] = GetCursorSelectionMonId();
ScriptContext1_SetupScript(SecretBase_EventScript_CaveUseSecretPower);
2018-12-09 19:22:59 -05:00
}
bool8 FldEff_UseSecretPowerCave(void)
2018-12-09 19:22:59 -05:00
{
u8 taskId = CreateFieldMoveTask();
2018-12-09 19:22:59 -05:00
gTasks[taskId].data[8] = (u32)StartSecretBaseCaveFieldEffect >> 16;
gTasks[taskId].data[9] = (u32)StartSecretBaseCaveFieldEffect;
2018-12-09 19:22:59 -05:00
return FALSE;
}
static void StartSecretBaseCaveFieldEffect(void)
2018-12-09 19:22:59 -05:00
{
FieldEffectActiveListRemove(FLDEFF_USE_SECRET_POWER_CAVE);
FieldEffectStart(FLDEFF_SECRET_POWER_CAVE);
}
bool8 FldEff_SecretPowerCave(void)
2018-12-09 19:22:59 -05:00
{
AdjustSecretPowerSpritePixelOffsets();
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SecretPowerCave,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x + gFieldEffectArguments[5],
gSprites[gPlayerAvatar.spriteId].oam.y + gFieldEffectArguments[6],
148);
return FALSE;
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_CaveEntranceInit(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_ROCK_THROW);
2018-12-09 19:22:59 -05:00
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_CaveEntranceOpen;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_CaveEntranceOpen(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
if (sprite->data[0] < 40)
{
2019-04-05 16:11:24 -05:00
if (++sprite->data[0] == 20)
ToggleSecretBaseEntranceMetatile();
2018-12-09 19:22:59 -05:00
}
else
{
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_CaveEntranceEnd;
2018-12-09 19:22:59 -05:00
}
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_CaveEntranceEnd(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
FieldEffectStop(sprite, FLDEFF_SECRET_POWER_CAVE);
EnableBothScriptContexts();
}
2018-12-22 19:33:45 +01:00
static void FieldCallback_SecretBaseTree(void)
2018-12-09 19:22:59 -05:00
{
gFieldEffectArguments[0] = GetCursorSelectionMonId();
ScriptContext1_SetupScript(SecretBase_EventScript_TreeUseSecretPower);
2018-12-09 19:22:59 -05:00
}
bool8 FldEff_UseSecretPowerTree(void)
2018-12-09 19:22:59 -05:00
{
u8 taskId = CreateFieldMoveTask();
2018-12-09 19:22:59 -05:00
gTasks[taskId].data[8] = (u32)StartSecretBaseTreeFieldEffect >> 16;
gTasks[taskId].data[9] = (u32)StartSecretBaseTreeFieldEffect;
2018-12-09 19:22:59 -05:00
return FALSE;
}
static void StartSecretBaseTreeFieldEffect(void)
2018-12-09 19:22:59 -05:00
{
FieldEffectActiveListRemove(FLDEFF_USE_SECRET_POWER_TREE);
FieldEffectStart(FLDEFF_SECRET_POWER_TREE);
}
bool8 FldEff_SecretPowerTree(void)
2018-12-09 19:22:59 -05:00
{
s16 mb = MapGridGetMetatileBehaviorAt(gPlayerFacingPosition.x, gPlayerFacingPosition.y) & 0xFFF;
if (mb == MB_SECRET_BASE_SPOT_TREE_LEFT)
gFieldEffectArguments[7] = 0;
if (mb == MB_SECRET_BASE_SPOT_TREE_RIGHT)
gFieldEffectArguments[7] = 2;
AdjustSecretPowerSpritePixelOffsets();
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SecretPowerTree,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x + gFieldEffectArguments[5],
gSprites[gPlayerAvatar.spriteId].oam.y + gFieldEffectArguments[6],
148);
if (gFieldEffectArguments[7] == 1 || gFieldEffectArguments[7] == 3)
2019-04-05 16:11:24 -05:00
ToggleSecretBaseEntranceMetatile();
2018-12-09 19:22:59 -05:00
return FALSE;
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_TreeEntranceInit(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_SCRATCH);
2018-12-09 19:22:59 -05:00
sprite->animNum = gFieldEffectArguments[7];
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_TreeEntranceOpen;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_TreeEntranceOpen(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
sprite->data[0]++;
if (sprite->data[0] >= 40)
{
if (gFieldEffectArguments[7] == 0 || gFieldEffectArguments[7] == 2)
2019-04-05 16:11:24 -05:00
ToggleSecretBaseEntranceMetatile();
2018-12-09 19:22:59 -05:00
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_TreeEntranceEnd;
2018-12-09 19:22:59 -05:00
}
2018-11-20 09:51:54 -05:00
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_TreeEntranceEnd(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
FieldEffectStop(sprite, FLDEFF_SECRET_POWER_TREE);
EnableBothScriptContexts();
}
2018-12-22 19:33:45 +01:00
static void FieldCallback_SecretBaseShrub(void)
2018-12-09 19:22:59 -05:00
{
gFieldEffectArguments[0] = GetCursorSelectionMonId();
ScriptContext1_SetupScript(SecretBase_EventScript_ShrubUseSecretPower);
2018-12-09 19:22:59 -05:00
}
bool8 FldEff_UseSecretPowerShrub(void)
2018-12-09 19:22:59 -05:00
{
u8 taskId = CreateFieldMoveTask();
2018-12-09 19:22:59 -05:00
gTasks[taskId].data[8] = (u32)StartSecretBaseShrubFieldEffect >> 16;
gTasks[taskId].data[9] = (u32)StartSecretBaseShrubFieldEffect;
2018-12-09 19:22:59 -05:00
return FALSE;
}
static void StartSecretBaseShrubFieldEffect(void)
2018-12-09 19:22:59 -05:00
{
FieldEffectActiveListRemove(FLDEFF_USE_SECRET_POWER_SHRUB);
FieldEffectStart(FLDEFF_SECRET_POWER_SHRUB);
}
bool8 FldEff_SecretPowerShrub(void)
2018-12-09 19:22:59 -05:00
{
AdjustSecretPowerSpritePixelOffsets();
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SecretPowerShrub,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x + gFieldEffectArguments[5],
gSprites[gPlayerAvatar.spriteId].oam.y + gFieldEffectArguments[6],
148);
return FALSE;
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_ShrubEntranceInit(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_POISON_POWDER);
2018-12-09 19:22:59 -05:00
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_ShrubEntranceOpen;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_ShrubEntranceOpen(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
if (sprite->data[0] < 40)
{
sprite->data[0]++;
if (sprite->data[0] == 20)
2019-04-05 16:11:24 -05:00
ToggleSecretBaseEntranceMetatile();
2018-12-09 19:22:59 -05:00
}
else
{
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_ShrubEntranceEnd;
2018-12-09 19:22:59 -05:00
}
}
2020-06-29 11:41:09 -04:00
static void SpriteCB_ShrubEntranceEnd(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
FieldEffectStop(sprite, FLDEFF_SECRET_POWER_SHRUB);
EnableBothScriptContexts();
}
2020-06-29 11:41:09 -04:00
#define tX data[0]
#define tY data[1]
#define tState data[2]
2018-12-22 19:33:45 +01:00
bool8 FldEff_SecretBasePCTurnOn(void)
2018-12-09 19:22:59 -05:00
{
s16 x, y;
u8 taskId;
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
taskId = CreateTask(Task_SecretBasePCTurnOn, 0);
2020-06-29 11:41:09 -04:00
gTasks[taskId].tX = x;
gTasks[taskId].tY = y;
gTasks[taskId].tState = 0;
2018-12-09 19:22:59 -05:00
2018-12-22 19:33:45 +01:00
return FALSE;
2018-12-09 19:22:59 -05:00
}
2018-12-22 19:33:45 +01:00
static void Task_SecretBasePCTurnOn(u8 taskId)
2018-12-09 19:22:59 -05:00
{
s16 *data = gTasks[taskId].data;
2020-06-29 11:41:09 -04:00
switch (tState)
2018-12-09 19:22:59 -05:00
{
case 4:
case 12:
2020-06-29 11:41:09 -04:00
MapGridSetMetatileIdAt(tX, tY, METATILE_SecretBase_PC_On);
CurrentMapDrawMetatileAt(tX, tY);
2018-12-09 19:22:59 -05:00
break;
case 8:
case 16:
2020-06-29 11:41:09 -04:00
MapGridSetMetatileIdAt(tX, tY, METATILE_SecretBase_PC);
CurrentMapDrawMetatileAt(tX, tY);
2018-12-09 19:22:59 -05:00
break;
case 20:
2020-06-29 11:41:09 -04:00
MapGridSetMetatileIdAt(tX, tY, METATILE_SecretBase_PC_On);
CurrentMapDrawMetatileAt(tX, tY);
2018-12-09 19:22:59 -05:00
FieldEffectActiveListRemove(FLDEFF_PCTURN_ON);
EnableBothScriptContexts();
DestroyTask(taskId);
return;
}
2020-06-29 11:41:09 -04:00
tState++;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
#undef tX
#undef tY
#undef tState
void DoSecretBasePCTurnOffEffect(void)
2018-12-09 19:22:59 -05:00
{
s16 x, y;
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
PlaySE(SE_PC_OFF);
if (!VarGet(VAR_CURRENT_SECRET_BASE))
MapGridSetMetatileIdAt(x, y, METATILE_SecretBase_PC | METATILE_COLLISION_MASK);
2018-12-09 19:22:59 -05:00
else
MapGridSetMetatileIdAt(x, y, METATILE_SecretBase_RegisterPC | METATILE_COLLISION_MASK);
2018-12-09 19:22:59 -05:00
CurrentMapDrawMetatileAt(x, y);
}
void PopSecretBaseBalloon(s16 metatileId, s16 x, s16 y)
2018-12-09 19:22:59 -05:00
{
u8 taskId = CreateTask(Task_PopSecretBaseBalloon, 0);
2018-12-09 19:22:59 -05:00
gTasks[taskId].data[0] = metatileId;
gTasks[taskId].data[1] = x;
gTasks[taskId].data[2] = y;
gTasks[taskId].data[3] = 0;
gTasks[taskId].data[4] = 1;
}
2018-12-22 19:33:45 +01:00
static void Task_PopSecretBaseBalloon(u8 taskId)
2018-12-09 19:22:59 -05:00
{
s16 *data = gTasks[taskId].data;
if (data[3] == 6)
data[3] = 0;
else
data[3]++;
if (data[3] == 0)
{
if (data[4] == 2)
DoBalloonSoundEffect(data[0]);
2018-12-09 19:22:59 -05:00
MapGridSetMetatileIdAt(data[1], data[2], data[0] + data[4]);
CurrentMapDrawMetatileAt(data[1], data[2]);
if (data[4] == 3)
DestroyTask(taskId);
else
data[4]++;
}
}
2018-12-22 19:33:45 +01:00
static void DoBalloonSoundEffect(s16 metatileId)
2018-12-09 19:22:59 -05:00
{
switch (metatileId)
{
case METATILE_SecretBase_RedBalloon:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALLOON_RED);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_BlueBalloon:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALLOON_BLUE);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_YellowBalloon:
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALLOON_YELLOW);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_MudBall:
2020-08-20 18:02:00 -04:00
PlaySE(SE_MUD_BALL);
2018-12-09 19:22:59 -05:00
break;
}
}
2020-06-29 11:41:09 -04:00
bool8 FldEff_Nop47(void)
2018-12-09 19:22:59 -05:00
{
return FALSE;
}
2020-06-29 11:41:09 -04:00
bool8 FldEff_Nop48(void)
2018-12-09 19:22:59 -05:00
{
return FALSE;
}
2018-12-22 19:33:45 +01:00
static void DoSecretBaseBreakableDoorEffect(s16 x, s16 y)
2018-12-09 19:22:59 -05:00
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_BREAKABLE_DOOR);
MapGridSetMetatileIdAt(x, y, METATILE_SecretBase_BreakableDoor_BottomOpen);
MapGridSetMetatileIdAt(x, y - 1, METATILE_SecretBase_BreakableDoor_TopOpen);
2018-12-09 19:22:59 -05:00
CurrentMapDrawMetatileAt(x, y);
CurrentMapDrawMetatileAt(x, y - 1);
}
2018-12-22 19:33:45 +01:00
static void Task_ShatterSecretBaseBreakableDoor(u8 taskId)
2018-12-09 19:22:59 -05:00
{
if (gTasks[taskId].data[0] == 7)
{
DoSecretBaseBreakableDoorEffect(gTasks[taskId].data[1], gTasks[taskId].data[2]);
2018-12-09 19:22:59 -05:00
DestroyTask(taskId);
}
else
{
gTasks[taskId].data[0]++;
}
}
void ShatterSecretBaseBreakableDoor(s16 x, s16 y)
2018-12-09 19:22:59 -05:00
{
u8 dir = GetPlayerFacingDirection();
if (dir == DIR_SOUTH)
{
DoSecretBaseBreakableDoorEffect(x, y);
2018-12-09 19:22:59 -05:00
}
else if (dir == DIR_NORTH)
{
u8 taskId = CreateTask(Task_ShatterSecretBaseBreakableDoor, 5);
2018-12-09 19:22:59 -05:00
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = x;
gTasks[taskId].data[2] = y;
}
}
#define tMetatileID data[0]
2018-12-22 19:33:45 +01:00
static void Task_SecretBaseMusicNoteMatSound(u8 taskId)
2018-12-09 19:22:59 -05:00
{
if (gTasks[taskId].data[1] == 7)
{
switch (gTasks[taskId].tMetatileID)
2018-12-09 19:22:59 -05:00
{
2020-04-13 14:43:50 -04:00
case METATILE_SecretBase_NoteMat_C_Low:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_C);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_NoteMat_D:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_D);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_NoteMat_E:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_E);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_NoteMat_F:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_F);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_NoteMat_G:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_G);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_NoteMat_A:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_A);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_NoteMat_B:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_B);
2018-12-09 19:22:59 -05:00
break;
2020-04-13 14:43:50 -04:00
case METATILE_SecretBase_NoteMat_C_High:
2020-08-20 18:02:00 -04:00
PlaySE(SE_NOTE_C_HIGH);
2018-12-09 19:22:59 -05:00
break;
}
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
DestroyTask(taskId);
}
else
{
gTasks[taskId].data[1]++;
}
}
2018-11-20 09:51:54 -05:00
void PlaySecretBaseMusicNoteMatSound(s16 metatileId)
2018-12-09 19:22:59 -05:00
{
u8 taskId = CreateTask(Task_SecretBaseMusicNoteMatSound, 5);
2018-12-09 19:22:59 -05:00
gTasks[taskId].tMetatileID = metatileId;
gTasks[taskId].data[1] = 0;
}
#undef tMetatileID
2018-11-20 09:51:54 -05:00
2018-12-22 19:33:45 +01:00
static void SpriteCB_GlitterMatSparkle(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
sprite->data[0]++;
if (sprite->data[0] == 8)
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_HEAL_BELL);
2018-12-09 19:22:59 -05:00
if (sprite->data[0] >= 32)
DestroySprite(sprite);
}
void DoSecretBaseGlitterMatSparkle(void)
2018-12-09 19:22:59 -05:00
{
s16 x = gObjectEvents[gPlayerAvatar.objectEventId].currentCoords.x;
s16 y = gObjectEvents[gPlayerAvatar.objectEventId].currentCoords.y;
2018-12-09 19:22:59 -05:00
u8 spriteId;
2018-11-20 09:51:54 -05:00
2020-06-01 10:23:12 -04:00
SetSpritePosToOffsetMapCoords(&x, &y, 8, 4);
2018-11-20 09:51:54 -05:00
spriteId = CreateSpriteAtEnd(gFieldEffectObjectTemplatePointers[FLDEFFOBJ_SPARKLE], x, y, 0);
2018-12-09 19:22:59 -05:00
if (spriteId != MAX_SPRITES)
{
gSprites[spriteId].coordOffsetEnabled = TRUE;
gSprites[spriteId].oam.priority = 1;
gSprites[spriteId].oam.paletteNum = 5;
gSprites[spriteId].callback = SpriteCB_GlitterMatSparkle;
2018-12-09 19:22:59 -05:00
gSprites[spriteId].data[0] = 0;
}
}
bool8 FldEff_SandPillar(void)
2018-12-09 19:22:59 -05:00
{
s16 x, y;
ScriptContext2_Enable();
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
gFieldEffectArguments[5] = x;
gFieldEffectArguments[6] = y;
switch (GetPlayerFacingDirection())
{
case DIR_SOUTH:
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SandPillar,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x + 8,
gSprites[gPlayerAvatar.spriteId].oam.y + 32,
0);
break;
case DIR_NORTH:
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SandPillar,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x + 8,
gSprites[gPlayerAvatar.spriteId].oam.y,
148);
2018-12-22 19:33:45 +01:00
2018-12-09 19:22:59 -05:00
break;
2018-12-22 19:33:45 +01:00
2018-12-09 19:22:59 -05:00
case DIR_WEST:
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SandPillar,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x - 8,
gSprites[gPlayerAvatar.spriteId].oam.y + 16,
148);
2018-12-22 19:33:45 +01:00
2018-12-09 19:22:59 -05:00
break;
2018-12-22 19:33:45 +01:00
2018-12-09 19:22:59 -05:00
case DIR_EAST:
2020-06-29 11:41:09 -04:00
CreateSprite(&sSpriteTemplate_SandPillar,
2018-12-09 19:22:59 -05:00
gSprites[gPlayerAvatar.spriteId].oam.x + 24,
gSprites[gPlayerAvatar.spriteId].oam.y + 16,
148);
2018-12-22 19:33:45 +01:00
2018-12-09 19:22:59 -05:00
break;
}
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
return FALSE;
}
2018-11-20 09:51:54 -05:00
2020-06-29 11:41:09 -04:00
static void SpriteCB_SandPillar_BreakTop(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_ROCK_THROW);
2018-11-20 09:51:54 -05:00
if (MapGridGetMetatileIdAt(gFieldEffectArguments[5], gFieldEffectArguments[6] - 1) == METATILE_SecretBase_SandOrnament_TopWall)
MapGridSetMetatileIdAt(gFieldEffectArguments[5], gFieldEffectArguments[6] - 1, METATILE_SecretBase_Wall_TopMid | METATILE_COLLISION_MASK);
2018-12-09 19:22:59 -05:00
else
MapGridSetMetatileIdAt(gFieldEffectArguments[5], gFieldEffectArguments[6] - 1, METATILE_SecretBase_SandOrnament_BrokenTop);
2018-12-22 19:33:45 +01:00
MapGridSetMetatileIdAt(gFieldEffectArguments[5], gFieldEffectArguments[6], METATILE_SecretBase_Ground);
2018-12-09 19:22:59 -05:00
CurrentMapDrawMetatileAt(gFieldEffectArguments[5], gFieldEffectArguments[6] - 1);
CurrentMapDrawMetatileAt(gFieldEffectArguments[5], gFieldEffectArguments[6]);
2018-12-22 19:33:45 +01:00
2018-12-09 19:22:59 -05:00
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_SandPillar_BreakBase;
2018-12-09 19:22:59 -05:00
}
2018-11-20 09:51:54 -05:00
2020-06-29 11:41:09 -04:00
static void SpriteCB_SandPillar_BreakBase(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
if (sprite->data[0] < 18)
{
sprite->data[0]++;
}
else
{
MapGridSetMetatileIdAt(gFieldEffectArguments[5], gFieldEffectArguments[6], METATILE_SecretBase_SandOrnament_BrokenBase | METATILE_COLLISION_MASK);
2018-12-09 19:22:59 -05:00
CurrentMapDrawMetatileAt(gFieldEffectArguments[5], gFieldEffectArguments[6]);
sprite->data[0] = 0;
2020-06-29 11:41:09 -04:00
sprite->callback = SpriteCB_SandPillar_End;
2018-12-09 19:22:59 -05:00
}
}
2018-11-20 09:51:54 -05:00
2020-06-29 11:41:09 -04:00
static void SpriteCB_SandPillar_End(struct Sprite *sprite)
2018-12-09 19:22:59 -05:00
{
FieldEffectStop(sprite, FLDEFF_SAND_PILLAR);
EnableBothScriptContexts();
}
2018-11-20 09:51:54 -05:00
void InteractWithShieldOrTVDecoration(void)
2018-12-09 19:22:59 -05:00
{
s16 x, y;
s32 metatileId;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
metatileId = MapGridGetMetatileIdAt(x, y);
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
switch (metatileId)
{
case METATILE_SecretBase_GoldShield_Base1:
2018-12-09 19:22:59 -05:00
ConvertIntToDecimalStringN(gStringVar1, 100, STR_CONV_MODE_LEFT_ALIGN, 3);
StringCopy(gStringVar2, gText_Gold);
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
gSpecialVar_Result = 0;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
if (!VarGet(VAR_CURRENT_SECRET_BASE))
return;
2018-11-20 09:51:54 -05:00
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_GOLD_SHIELD);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_SilverShield_Base1:
2018-12-09 19:22:59 -05:00
ConvertIntToDecimalStringN(gStringVar1, 50, STR_CONV_MODE_LEFT_ALIGN, 2);
StringCopy(gStringVar2, gText_Silver);
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
gSpecialVar_Result = 0;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
if (!VarGet(VAR_CURRENT_SECRET_BASE))
return;
2018-11-20 09:51:54 -05:00
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_SILVER_SHIELD);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_TV:
2018-12-09 19:22:59 -05:00
gSpecialVar_Result = 1;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
if (!VarGet(VAR_CURRENT_SECRET_BASE))
return;
2018-11-20 09:51:54 -05:00
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_TV);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_RoundTV:
2018-12-09 19:22:59 -05:00
gSpecialVar_Result = 2;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
if (!VarGet(VAR_CURRENT_SECRET_BASE))
return;
2018-11-20 09:51:54 -05:00
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_TV);
2018-12-09 19:22:59 -05:00
break;
case METATILE_SecretBase_CuteTV:
2018-12-09 19:22:59 -05:00
gSpecialVar_Result = 3;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
if (!VarGet(VAR_CURRENT_SECRET_BASE))
return;
2018-11-20 09:51:54 -05:00
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_TV);
2018-12-09 19:22:59 -05:00
break;
}
}
2018-11-20 09:51:54 -05:00
2020-06-29 11:41:09 -04:00
// As opposed to a small one (single metatile) like the balloons
bool8 IsLargeBreakableDecoration(u16 metatileId, bool8 checkBase)
2018-12-09 19:22:59 -05:00
{
2019-04-05 16:11:24 -05:00
if (!CurMapIsSecretBase())
2018-12-22 19:33:45 +01:00
return FALSE;
2020-06-29 11:41:09 -04:00
if (!checkBase)
2018-12-09 19:22:59 -05:00
{
if (metatileId == METATILE_SecretBase_SandOrnament_Top || metatileId == METATILE_SecretBase_SandOrnament_TopWall)
2018-12-22 19:33:45 +01:00
return TRUE;
if (metatileId == METATILE_SecretBase_BreakableDoor_TopClosed)
2018-12-22 19:33:45 +01:00
return TRUE;
2018-12-09 19:22:59 -05:00
}
2018-12-22 19:33:45 +01:00
else
{
if (metatileId == METATILE_SecretBase_SandOrnament_Base1)
2018-12-22 19:33:45 +01:00
return TRUE;
if (metatileId == METATILE_SecretBase_BreakableDoor_BottomClosed)
2018-12-22 19:33:45 +01:00
return TRUE;
}
2018-12-09 19:22:59 -05:00
return FALSE;
}
2018-11-20 09:51:54 -05:00
2018-12-22 19:33:45 +01:00
static void Task_FieldPoisonEffect(u8 taskId)
2018-12-09 19:22:59 -05:00
{
s16 *data = gTasks[taskId].data;
2018-11-20 09:51:54 -05:00
2018-12-09 19:22:59 -05:00
switch (data[0])
{
case 0:
data[1] += 2;
if (data[1] > 8)
data[0]++;
break;
case 1:
data[1] -= 2;
if (data[1] == 0)
data[0]++;
break;
case 2:
DestroyTask(taskId);
return;
}
SetGpuReg(REG_OFFSET_MOSAIC, (data[1] << 4) | data[1]);
}
2018-12-19 22:19:54 -05:00
void FldEffPoison_Start(void)
2018-12-09 19:22:59 -05:00
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_FIELD_POISON);
CreateTask(Task_FieldPoisonEffect, 80);
2018-12-09 19:22:59 -05:00
}
2018-12-19 22:19:54 -05:00
bool32 FldEffPoison_IsActive(void)
2018-12-09 19:22:59 -05:00
{
return FuncIsActiveTask(Task_FieldPoisonEffect);
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
static void Task_WateringBerryTreeAnim(u8 taskId)
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
gTasks[taskId].func = Task_WateringBerryTreeAnim_Start;
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
static void Task_WateringBerryTreeAnim_Start(u8 taskId)
2018-12-09 19:22:59 -05:00
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-12-22 19:33:45 +01:00
2019-11-21 00:14:40 -05:00
if (!ObjectEventIsMovementOverridden(playerObjEvent)
|| ObjectEventClearHeldMovementIfFinished(playerObjEvent))
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
// Start watering
SetPlayerAvatarWatering(GetPlayerFacingDirection());
2019-11-21 00:14:40 -05:00
ObjectEventSetHeldMovement(playerObjEvent, GetWalkInPlaceNormalMovementAction(GetPlayerFacingDirection()));
2020-06-29 11:41:09 -04:00
gTasks[taskId].func = Task_WateringBerryTreeAnim_Continue;
2018-12-09 19:22:59 -05:00
}
}
2020-06-29 11:41:09 -04:00
static void Task_WateringBerryTreeAnim_Continue(u8 taskId)
2018-12-09 19:22:59 -05:00
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-12-22 19:33:45 +01:00
2019-11-21 00:14:40 -05:00
if (ObjectEventClearHeldMovementIfFinished(playerObjEvent))
2018-12-09 19:22:59 -05:00
{
s16 value = gTasks[taskId].data[1]++;
2018-12-22 19:33:45 +01:00
2020-06-29 11:41:09 -04:00
// Continue holding watering action 10 times, then end
2018-12-09 19:22:59 -05:00
if (value < 10)
2019-11-21 00:14:40 -05:00
ObjectEventSetHeldMovement(playerObjEvent, GetWalkInPlaceNormalMovementAction(GetPlayerFacingDirection()));
2018-12-09 19:22:59 -05:00
else
2020-06-29 11:41:09 -04:00
gTasks[taskId].func = Task_WateringBerryTreeAnim_End;
2018-12-09 19:22:59 -05:00
}
}
2020-06-29 11:41:09 -04:00
static void Task_WateringBerryTreeAnim_End(u8 taskId)
2018-12-09 19:22:59 -05:00
{
2020-04-01 02:54:26 -04:00
SetPlayerAvatarTransitionFlags(GetPlayerAvatarFlags());
2018-12-09 19:22:59 -05:00
DestroyTask(taskId);
EnableBothScriptContexts();
}
void DoWateringBerryTreeAnim(void)
2018-12-09 19:22:59 -05:00
{
2020-06-29 11:41:09 -04:00
CreateTask(Task_WateringBerryTreeAnim, 80);
2018-12-09 19:22:59 -05:00
}
2020-06-29 11:41:09 -04:00
// The lights that blink on the counter when mixing records in the cable club
u8 CreateRecordMixingLights(void)
2018-12-09 19:22:59 -05:00
{
u8 spriteId;
2020-06-29 11:41:09 -04:00
LoadSpritePalette(&sSpritePalette_RecordMixLights);
2018-12-09 19:22:59 -05:00
2020-06-29 11:41:09 -04:00
spriteId = CreateSprite(&sSpriteTemplate_RecordMixLights, 0, 0, 82);
2018-12-09 19:22:59 -05:00
if (spriteId == MAX_SPRITES)
{
return MAX_SPRITES;
}
else
{
struct Sprite *sprite = &gSprites[spriteId];
2021-07-07 09:11:52 -04:00
GetMapCoordsFromSpritePos(16, 13, &sprite->x, &sprite->y);
2018-12-09 19:22:59 -05:00
sprite->coordOffsetEnabled = TRUE;
2021-07-07 09:11:52 -04:00
sprite->x += 16;
sprite->y += 2;
2018-12-09 19:22:59 -05:00
}
return spriteId;
}
2020-06-29 11:41:09 -04:00
void DestroyRecordMixingLights(void)
2018-12-09 19:22:59 -05:00
{
int i;
for (i = 0; i < MAX_SPRITES; i++)
{
2020-06-29 11:41:09 -04:00
if (gSprites[i].template == &sSpriteTemplate_RecordMixLights)
2018-12-09 19:22:59 -05:00
{
FreeSpritePalette(&gSprites[i]);
DestroySprite(&gSprites[i]);
}
}
}