pokeemerald/src/credits.c

1634 lines
46 KiB
C
Raw Normal View History

2018-02-17 03:55:32 -06:00
#include "global.h"
#include "palette.h"
#include "main.h"
#include "task.h"
#include "bg.h"
#include "malloc.h"
2018-02-17 03:55:32 -06:00
#include "window.h"
#include "text.h"
#include "menu.h"
#include "international_string_util.h"
#include "constants/songs.h"
#include "gpu_regs.h"
#include "m4a.h"
#include "constants/rgb.h"
2018-08-19 01:06:10 +02:00
#include "trainer_pokemon_sprites.h"
2018-02-17 03:55:32 -06:00
#include "starter_choose.h"
#include "decompress.h"
#include "intro_credits_graphics.h"
#include "sound.h"
#include "trig.h"
#include "graphics.h"
#include "pokedex.h"
#include "event_data.h"
#include "random.h"
#define COLOR_DARK_GREEN RGB(7, 11, 6)
#define COLOR_LIGHT_GREEN RGB(13, 20, 12)
2021-03-28 23:56:42 -04:00
#define TAG_MON_BG 1001
// Positions for the Pokémon images
enum {
POS_LEFT,
POS_CENTER,
POS_RIGHT,
};
2021-03-29 12:39:49 -04:00
enum {
MODE_NONE,
MODE_BIKE_SCENE,
MODE_SHOW_MONS,
2018-02-17 03:55:32 -06:00
};
2021-03-29 12:39:49 -04:00
#define tState data[0]
// Task data for the main Credits tasks
#define tTaskId_BgScenery data[0] // ID for Task_BicycleBgAnimation (created by CreateBicycleBgAnimationTask)
#define tTaskId_BikeScene data[1] // ID for Task_BikeScene
#define tTaskId_SceneryPal data[2] // ID for Task_CycleSceneryPalette
#define tTaskId_ShowMons data[3] // ID for Task_ShowMons
#define tEndCredits data[4]
#define tPlayerSpriteId data[5]
#define tRivalSpriteId data[6]
#define tSceneNum data[7]
// data[8]-[10] are unused
#define tNextMode data[11]
#define tTheEndDelay data[12]
#define tCurrentMode data[13]
#define tPrintedPage data[14]
#define tTaskId_UpdatePage data[15]
2019-11-04 11:38:09 -05:00
#define NUM_MON_SLIDES 71
2021-03-29 12:39:49 -04:00
2019-11-03 01:33:46 -04:00
struct CreditsData
{
2019-11-04 11:38:09 -05:00
u16 monToShow[NUM_MON_SLIDES]; // List of Pokemon species ids that will show during the credits
2019-11-03 01:33:46 -04:00
u16 imgCounter; //how many mon images have been shown
u16 nextImgPos; //if the next image spawns left/center/right
2019-11-04 11:38:09 -05:00
u16 currShownMon; //index into monToShow
u16 numMonToShow; //number of pokemon to show, always NUM_MON_SLIDES after determine function
u16 caughtMonIds[NATIONAL_DEX_COUNT]; //temporary location to hold a condensed array of all caught pokemon
u16 numCaughtMon; //count of filled spaces in caughtMonIds
u16 unused[7];
2018-02-17 03:55:32 -06:00
};
struct CreditsEntry
{
2021-03-28 23:56:42 -04:00
u8 unk; // Never read
2019-08-17 13:32:26 -04:00
bool8 isTitle;
2018-02-17 03:55:32 -06:00
const u8 *text;
};
2021-03-28 23:56:42 -04:00
static EWRAM_DATA s16 sUnkVar = 0; // Never read, only set to 0
static EWRAM_DATA u16 sSavedTaskId = 0;
2018-02-17 03:55:32 -06:00
EWRAM_DATA bool8 gHasHallOfFameRecords = 0;
2021-03-28 23:56:42 -04:00
static EWRAM_DATA bool8 sUsedSpeedUp = 0; // Never read
2019-11-03 01:33:46 -04:00
static EWRAM_DATA struct CreditsData *sCreditsData = {0};
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
static const u16 sCredits_Pal[] = INCBIN_U16("graphics/credits/credits.gbapal");
static const u32 sCreditsCopyrightEnd_Gfx[] = INCBIN_U32("graphics/credits/the_end_copyright.4bpp.lz");
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
static void SpriteCB_CreditsMonBg(struct Sprite *);
static void Task_WaitPaletteFade(u8);
2021-03-29 12:39:49 -04:00
static void Task_CreditsMain(u8);
static void Task_ReadyBikeScene(u8);
static void Task_SetBikeScene(u8);
static void Task_LoadShowMons(u8);
static void Task_ReadyShowMons(u8);
2021-03-28 23:56:42 -04:00
static void Task_CreditsTheEnd1(u8);
static void Task_CreditsTheEnd2(u8);
static void Task_CreditsTheEnd3(u8);
static void Task_CreditsTheEnd4(u8);
static void Task_CreditsTheEnd5(u8);
static void Task_CreditsTheEnd6(u8);
static void Task_CreditsSoftReset(u8);
static void ResetGpuAndVram(void);
2021-03-29 12:39:49 -04:00
static void Task_UpdatePage(u8);
2021-03-28 23:56:42 -04:00
static u8 CheckChangeScene(u8, u8);
2021-03-29 12:39:49 -04:00
static void Task_ShowMons(u8);
static void Task_CycleSceneryPalette(u8);
static void Task_BikeScene(u8);
static bool8 LoadBikeScene(u8 data, u8);
2021-03-28 23:56:42 -04:00
static void ResetCreditsTasks(u8);
static void LoadTheEndScreen(u16, u16, u16);
static void DrawTheEnd(u16, u16);
2021-03-29 12:39:49 -04:00
static void SpriteCB_Player(struct Sprite *);
static void SpriteCB_Rival(struct Sprite *);
2021-03-28 23:56:42 -04:00
static u8 CreateCreditsMonSprite(u16, s16, s16, u16);
static void DeterminePokemonToShow(void);
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
static const u8 sTheEnd_LetterMap_T[] =
2018-02-17 03:55:32 -06:00
{
0, 1, 0,
0xFF, 1, 0xFF,
0xFF, 1, 0xFF,
0xFF, 1, 0xFF,
0xFF, 1, 0xFF,
};
2021-03-28 23:56:42 -04:00
static const u8 sTheEnd_LetterMap_H[] =
2018-02-17 03:55:32 -06:00
{
1, 0xFF, 1,
1, 0xFF, 1,
1, 2, 1,
1, 0xFF, 1,
1, 0xFF, 1,
};
2021-03-28 23:56:42 -04:00
static const u8 sTheEnd_LetterMap_E[] =
2018-02-17 03:55:32 -06:00
{
1, 0, 0,
1, 0xFF, 0xFF,
1, 2, 2,
1, 0xFF, 0xFF,
1, 0x80, 0x80,
};
2021-03-28 23:56:42 -04:00
static const u8 sTheEnd_LetterMap_N[] =
2018-02-17 03:55:32 -06:00
{
1, 3, 1,
1, 4, 1,
1, 5, 1,
1, 0xC4, 1,
1, 0xC3, 1,
};
2021-03-28 23:56:42 -04:00
static const u8 sTheEnd_LetterMap_D[] =
2018-02-17 03:55:32 -06:00
{
1, 6, 7,
1, 8, 9,
1, 0xFF, 1,
1, 0x88, 0x89,
1, 0x86, 0x87,
};
2021-03-28 23:56:42 -04:00
#include "data/credits.h"
2018-02-17 03:55:32 -06:00
2019-08-17 12:01:38 -04:00
static const struct BgTemplate sBackgroundTemplates[] =
2018-02-17 03:55:32 -06:00
{
{
.bg = 0,
.charBaseIndex = 2,
.mapBaseIndex = 28,
.screenSize = 0,
.paletteMode = 0,
.priority = 0,
.baseTile = 0
},
};
2019-08-17 12:01:38 -04:00
static const struct WindowTemplate sWindowTemplates[] =
2018-02-17 03:55:32 -06:00
{
2018-09-02 17:53:52 +01:00
{
2018-10-27 00:53:07 +02:00
.bg = 0,
2018-09-02 17:53:52 +01:00
.tilemapLeft = 0,
.tilemapTop = 9,
.width = 30,
.height = 12,
.paletteNum = 8,
.baseBlock = 1
},
2018-02-17 03:55:32 -06:00
DUMMY_WIN_TEMPLATE,
};
2019-11-03 01:33:46 -04:00
static const u8 sMonSpritePos[][2] =
2018-02-17 03:55:32 -06:00
{
{104, 36},
{120, 36},
{136, 36},
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Player_Slow[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(64, 8),
ANIMCMD_FRAME(128, 8),
ANIMCMD_FRAME(192, 8),
ANIMCMD_JUMP(0),
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Player_Fast[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(0, 4),
ANIMCMD_FRAME(64, 4),
ANIMCMD_FRAME(128, 4),
ANIMCMD_FRAME(192, 4),
ANIMCMD_JUMP(0),
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Player_LookBack[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(256, 4),
ANIMCMD_FRAME(320, 4),
ANIMCMD_FRAME(384, 4),
ANIMCMD_END,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Player_LookForward[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(384, 30),
ANIMCMD_FRAME(320, 30),
ANIMCMD_FRAME(256, 30),
ANIMCMD_FRAME(256, 30),
ANIMCMD_END,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd *const sAnims_Player[] =
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
sAnim_Player_Slow,
sAnim_Player_Fast,
sAnim_Player_LookBack,
sAnim_Player_LookForward,
2018-02-17 03:55:32 -06:00
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Rival_Slow[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_FRAME(64, 8),
ANIMCMD_FRAME(128, 8),
ANIMCMD_FRAME(192, 8),
ANIMCMD_JUMP(0),
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Rival_Fast[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(0, 4),
ANIMCMD_FRAME(64, 4),
ANIMCMD_FRAME(128, 4),
ANIMCMD_FRAME(192, 4),
ANIMCMD_JUMP(0),
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_Rival_Still[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(0, 4),
ANIMCMD_END,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd *const sAnims_Rival[] =
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
sAnim_Rival_Slow,
sAnim_Rival_Fast,
sAnim_Rival_Still,
2018-02-17 03:55:32 -06:00
};
2021-03-29 09:38:19 -04:00
#define MONBG_OFFSET (MON_PIC_SIZE * 3)
2021-03-28 23:56:42 -04:00
static const struct SpriteSheet sSpriteSheet_MonBg[] = {
{ gDecompressionBuffer, MONBG_OFFSET, TAG_MON_BG },
{},
2018-02-17 03:55:32 -06:00
};
2021-03-28 23:56:42 -04:00
static const struct SpritePalette sSpritePalette_MonBg[] = {
{ (const u16 *)&gDecompressionBuffer[MONBG_OFFSET], TAG_MON_BG },
{},
2018-02-17 03:55:32 -06:00
};
2021-03-28 23:56:42 -04:00
static const struct OamData sOamData_MonBg =
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
.y = DISPLAY_HEIGHT,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
2018-02-17 03:55:32 -06:00
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x64),
2018-02-17 03:55:32 -06:00
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(64x64),
2018-02-17 03:55:32 -06:00
.tileNum = 0,
.priority = 1,
.paletteNum = 0,
.affineParam = 0,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_MonBg_Yellow[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(0, 8),
ANIMCMD_END,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_MonBg_Red[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(64, 8),
ANIMCMD_END,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd sAnim_MonBg_Blue[] =
2018-02-17 03:55:32 -06:00
{
ANIMCMD_FRAME(128, 8),
ANIMCMD_END,
};
2021-03-28 23:56:42 -04:00
static const union AnimCmd *const sAnims_MonBg[] =
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
[POS_LEFT] = sAnim_MonBg_Yellow,
[POS_CENTER] = sAnim_MonBg_Red,
[POS_RIGHT] = sAnim_MonBg_Blue,
2018-02-17 03:55:32 -06:00
};
2021-03-28 23:56:42 -04:00
static const struct SpriteTemplate sSpriteTemplate_CreditsMonBg =
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
.tileTag = TAG_MON_BG,
.paletteTag = TAG_MON_BG,
.oam = &sOamData_MonBg,
.anims = sAnims_MonBg,
2018-02-17 03:55:32 -06:00
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2021-03-28 23:56:42 -04:00
.callback = SpriteCB_CreditsMonBg,
2018-02-17 03:55:32 -06:00
};
2021-03-28 23:56:42 -04:00
static void VBlankCB_Credits(void)
2018-02-17 03:55:32 -06:00
{
LoadOam();
ProcessSpriteCopyRequests();
TransferPlttBuffer();
}
2021-03-28 23:56:42 -04:00
static void CB2_Credits(void)
2018-02-17 03:55:32 -06:00
{
RunTasks();
AnimateSprites();
2018-10-27 00:53:07 +02:00
2020-09-04 21:11:55 -04:00
if ((JOY_HELD(B_BUTTON))
2021-03-28 23:56:42 -04:00
&& gHasHallOfFameRecords
2021-03-29 12:39:49 -04:00
&& gTasks[sSavedTaskId].func == Task_CreditsMain)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
// Speed up credits
VBlankCB_Credits();
2018-02-17 03:55:32 -06:00
RunTasks();
AnimateSprites();
2021-03-28 23:56:42 -04:00
sUsedSpeedUp = TRUE;
2018-02-17 03:55:32 -06:00
}
BuildOamBuffer();
UpdatePaletteFade();
}
static void InitCreditsBgsAndWindows(void)
2018-02-17 03:55:32 -06:00
{
ResetBgsAndClearDma3BusyFlags(0);
InitBgsFromTemplates(0, sBackgroundTemplates, ARRAY_COUNT(sBackgroundTemplates));
2020-08-23 18:50:57 -04:00
SetBgTilemapBuffer(0, AllocZeroed(BG_SCREEN_SIZE));
2021-03-28 23:56:42 -04:00
LoadPalette(sCredits_Pal, 0x80, 64);
2019-08-17 12:01:38 -04:00
InitWindows(sWindowTemplates);
2018-02-17 03:55:32 -06:00
DeactivateAllTextPrinters();
PutWindowTilemap(0);
CopyWindowToVram(0, 3);
ShowBg(0);
}
2021-03-29 12:39:49 -04:00
static void FreeCreditsBgsAndWindows(void)
2018-02-17 03:55:32 -06:00
{
void *ptr;
FreeAllWindowBuffers();
ptr = GetBgTilemapBuffer(0);
if (ptr)
Free(ptr);
}
2019-08-17 13:32:26 -04:00
static void PrintCreditsText(const u8 *string, u8 y, bool8 isTitle)
2018-02-17 03:55:32 -06:00
{
u8 x;
u8 color[3];
2018-10-27 00:53:07 +02:00
2021-03-28 23:56:42 -04:00
color[0] = TEXT_COLOR_TRANSPARENT;
2018-10-27 00:53:07 +02:00
2019-08-17 13:32:26 -04:00
if (isTitle == TRUE)
2018-02-17 03:55:32 -06:00
{
2021-04-09 22:39:34 -04:00
color[1] = TEXT_COLOR_LIGHT_GRAY;
2021-03-28 23:56:42 -04:00
color[2] = TEXT_COLOR_RED;
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-28 23:56:42 -04:00
color[1] = TEXT_COLOR_WHITE;
2021-04-09 22:39:34 -04:00
color[2] = TEXT_COLOR_DARK_GRAY;
2018-02-17 03:55:32 -06:00
}
2018-10-27 00:53:07 +02:00
2021-03-28 23:56:42 -04:00
x = GetStringCenterAlignXOffsetWithLetterSpacing(1, string, DISPLAY_WIDTH, 1);
AddTextPrinterParameterized4(0, 1, x, y, 1, 0, color, -1, string);
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
#define tMainTaskId data[1]
2019-08-17 12:01:38 -04:00
void CB2_StartCreditsSequence(void)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
u8 taskId;
s16 bikeTaskId;
u8 pageTaskId;
2018-02-17 03:55:32 -06:00
2019-08-17 12:01:38 -04:00
ResetGpuAndVram();
2018-02-17 03:55:32 -06:00
SetVBlankCallback(NULL);
InitHeap(gHeap, HEAP_SIZE);
ResetPaletteFade();
ResetTasks();
InitCreditsBgsAndWindows();
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
taskId = CreateTask(Task_WaitPaletteFade, 0);
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
gTasks[taskId].tEndCredits = FALSE;
gTasks[taskId].tSceneNum = SCENE_OCEAN_MORNING;
gTasks[taskId].tNextMode = MODE_NONE;
gTasks[taskId].tCurrentMode = MODE_BIKE_SCENE;
2018-02-17 03:55:32 -06:00
while (TRUE)
{
2021-03-29 12:39:49 -04:00
if (LoadBikeScene(SCENE_OCEAN_MORNING, taskId))
2018-02-17 03:55:32 -06:00
break;
}
2021-03-29 12:39:49 -04:00
bikeTaskId = gTasks[taskId].tTaskId_BikeScene;
gTasks[bikeTaskId].tState = 40;
2018-02-17 03:55:32 -06:00
SetGpuReg(REG_OFFSET_BG0VOFS, 0xFFFC);
2018-10-27 00:53:07 +02:00
2021-03-29 12:39:49 -04:00
pageTaskId = CreateTask(Task_UpdatePage, 0);
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
gTasks[pageTaskId].tMainTaskId = taskId;
gTasks[taskId].tTaskId_UpdatePage = pageTaskId;
2018-02-17 03:55:32 -06:00
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
2018-02-17 03:55:32 -06:00
EnableInterrupts(INTR_FLAG_VBLANK);
2021-03-28 23:56:42 -04:00
SetVBlankCallback(VBlankCB_Credits);
2020-08-20 18:02:00 -04:00
m4aSongNumStart(MUS_CREDITS);
2021-03-28 23:56:42 -04:00
SetMainCallback2(CB2_Credits);
sUsedSpeedUp = FALSE;
2019-11-03 01:33:46 -04:00
sCreditsData = AllocZeroed(sizeof(struct CreditsData));
2018-02-17 03:55:32 -06:00
2019-11-03 01:33:46 -04:00
DeterminePokemonToShow();
2018-02-17 03:55:32 -06:00
2019-11-03 01:33:46 -04:00
sCreditsData->imgCounter = 0;
2021-03-28 23:56:42 -04:00
sCreditsData->nextImgPos = POS_LEFT;
2019-11-04 11:38:09 -05:00
sCreditsData->currShownMon = 0;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
sSavedTaskId = taskId;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
static void Task_WaitPaletteFade(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_CreditsMain;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
static void Task_CreditsMain(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
u16 mode;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tEndCredits)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
s16 bikeTaskId = gTasks[taskId].tTaskId_BikeScene;
gTasks[bikeTaskId].tState = 30;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
gTasks[taskId].tTheEndDelay = 256;
gTasks[taskId].func = Task_CreditsTheEnd1;
2018-02-17 03:55:32 -06:00
return;
}
2021-03-28 23:56:42 -04:00
sUnkVar = 0;
2021-03-29 12:39:49 -04:00
mode = gTasks[taskId].tNextMode;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tNextMode == MODE_BIKE_SCENE)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
// Start a bike cutscene
gTasks[taskId].tCurrentMode = mode;
gTasks[taskId].tNextMode = MODE_NONE;
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_ReadyBikeScene;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
else if (gTasks[taskId].tNextMode == MODE_SHOW_MONS)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
// Start a Pokémon interlude
gTasks[taskId].tCurrentMode = mode;
gTasks[taskId].tNextMode = MODE_NONE;
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_ReadyShowMons;
2018-02-17 03:55:32 -06:00
}
}
2021-03-29 12:39:49 -04:00
static void Task_ReadyBikeScene(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
{
SetGpuReg(REG_OFFSET_DISPCNT, 0);
2021-03-29 12:39:49 -04:00
ResetCreditsTasks(taskId);
gTasks[taskId].func = Task_SetBikeScene;
2018-02-17 03:55:32 -06:00
}
}
2021-03-29 12:39:49 -04:00
static void Task_SetBikeScene(u8 taskId)
2018-02-17 03:55:32 -06:00
{
SetVBlankCallback(NULL);
2021-03-29 12:39:49 -04:00
if (LoadBikeScene(gTasks[taskId].tSceneNum, taskId))
2018-02-17 03:55:32 -06:00
{
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
2018-02-17 03:55:32 -06:00
EnableInterrupts(INTR_FLAG_VBLANK);
2021-03-28 23:56:42 -04:00
SetVBlankCallback(VBlankCB_Credits);
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_WaitPaletteFade;
2018-02-17 03:55:32 -06:00
}
}
2021-03-29 12:39:49 -04:00
static void Task_ReadyShowMons(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
{
SetGpuReg(REG_OFFSET_DISPCNT, 0);
2021-03-29 12:39:49 -04:00
ResetCreditsTasks(taskId);
gTasks[taskId].func = Task_LoadShowMons;
2018-02-17 03:55:32 -06:00
}
}
2021-03-29 12:39:49 -04:00
static void Task_LoadShowMons(u8 taskId)
2018-02-17 03:55:32 -06:00
{
switch (gMain.state)
{
default:
case 0:
{
u16 i;
u16 *temp;
ResetSpriteData();
2018-08-19 01:06:10 +02:00
ResetAllPicSprites();
2018-02-17 03:55:32 -06:00
FreeAllSpritePalettes();
gReservedSpritePaletteCount = 8;
2018-10-21 02:24:57 -05:00
LZ77UnCompVram(gBirchHelpGfx, (void *)VRAM);
LZ77UnCompVram(gBirchGrassTilemap, (void *)(BG_SCREEN_ADDR(7)));
2018-02-17 03:55:32 -06:00
LoadPalette(gBirchBagGrassPal[0] + 1, 1, 31 * 2);
2021-03-29 09:38:19 -04:00
for (i = 0; i < MON_PIC_SIZE; i++)
2018-02-17 03:55:32 -06:00
gDecompressionBuffer[i] = 0x11;
2021-03-29 09:38:19 -04:00
for (i = 0; i < MON_PIC_SIZE; i++)
(gDecompressionBuffer + MON_PIC_SIZE)[i] = 0x22;
for (i = 0; i < MON_PIC_SIZE; i++)
(gDecompressionBuffer + MON_PIC_SIZE * 2)[i] = 0x33;
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
temp = (u16 *)(&gDecompressionBuffer[MONBG_OFFSET]);
2018-02-17 03:55:32 -06:00
temp[0] = RGB_BLACK;
temp[1] = RGB(31, 31, 20); // light yellow
temp[2] = RGB(31, 20, 20); // light red
temp[3] = RGB(20, 20, 31); // light blue
2021-03-28 23:56:42 -04:00
LoadSpriteSheet(sSpriteSheet_MonBg);
LoadSpritePalette(sSpritePalette_MonBg);
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
gMain.state++;
2018-02-17 03:55:32 -06:00
break;
}
case 1:
2021-03-29 12:39:49 -04:00
gTasks[taskId].tTaskId_ShowMons = CreateTask(Task_ShowMons, 0);
gTasks[gTasks[taskId].tTaskId_ShowMons].tState = 1;
gTasks[gTasks[taskId].tTaskId_ShowMons].tMainTaskId = taskId;
gTasks[gTasks[taskId].tTaskId_ShowMons].data[2] = gTasks[taskId].tSceneNum; // data[2] never read
2018-02-17 03:55:32 -06:00
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
2018-02-17 03:55:32 -06:00
SetGpuReg(REG_OFFSET_BG3HOFS, 0);
SetGpuReg(REG_OFFSET_BG3VOFS, 32);
SetGpuReg(REG_OFFSET_BG3CNT, BGCNT_PRIORITY(3)
| BGCNT_CHARBASE(0)
| BGCNT_SCREENBASE(7)
| BGCNT_16COLOR
| BGCNT_TXT256x256);
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0
| DISPCNT_OBJ_1D_MAP
| DISPCNT_BG0_ON
| DISPCNT_BG3_ON
| DISPCNT_OBJ_ON);
gMain.state = 0;
2021-03-28 23:56:42 -04:00
gIntroCredits_MovingSceneryState = INTROCRED_SCENERY_NORMAL;
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_WaitPaletteFade;
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-29 12:39:49 -04:00
static void Task_CreditsTheEnd1(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tTheEndDelay)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tTheEndDelay--;
2018-02-17 03:55:32 -06:00
return;
}
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 12, 0, 16, RGB_BLACK);
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_CreditsTheEnd2;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
static void Task_CreditsTheEnd2(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
{
2021-03-29 12:39:49 -04:00
ResetCreditsTasks(taskId);
gTasks[taskId].func = Task_CreditsTheEnd3;
2018-02-17 03:55:32 -06:00
}
}
2021-03-28 23:56:42 -04:00
#define tDelay data[0]
2021-03-29 12:39:49 -04:00
static void Task_CreditsTheEnd3(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2019-08-17 12:01:38 -04:00
ResetGpuAndVram();
2018-02-17 03:55:32 -06:00
ResetPaletteFade();
2019-08-17 12:01:38 -04:00
LoadTheEndScreen(0, 0x3800, 0);
2018-02-17 03:55:32 -06:00
ResetSpriteData();
FreeAllSpritePalettes();
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 8, 16, 0, RGB_BLACK);
2018-02-17 03:55:32 -06:00
SetGpuReg(REG_OFFSET_BG0CNT, BGCNT_PRIORITY(0)
| BGCNT_CHARBASE(0)
| BGCNT_SCREENBASE(7)
| BGCNT_16COLOR
| BGCNT_TXT256x256);
EnableInterrupts(INTR_FLAG_VBLANK);
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0
| DISPCNT_OBJ_1D_MAP
| DISPCNT_BG0_ON);
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay = 235; //set this to 215 to actually show "THE END" in time to the last song beat
gTasks[taskId].func = Task_CreditsTheEnd4;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
static void Task_CreditsTheEnd4(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay--;
2018-02-17 03:55:32 -06:00
return;
}
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 6, 0, 16, RGB_BLACK);
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_CreditsTheEnd5;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
static void Task_CreditsTheEnd5(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
{
2021-03-28 23:56:42 -04:00
DrawTheEnd(0x3800, 0);
2018-02-17 03:55:32 -06:00
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 0, RGB_BLACK);
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay = 7200;
gTasks[taskId].func = Task_CreditsTheEnd6;
2018-02-17 03:55:32 -06:00
}
}
2021-03-29 12:39:49 -04:00
static void Task_CreditsTheEnd6(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
{
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay == 0 || gMain.newKeys)
2018-02-17 03:55:32 -06:00
{
FadeOutBGM(4);
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 8, 0, 16, RGB_WHITEALPHA);
2021-03-29 12:39:49 -04:00
gTasks[taskId].func = Task_CreditsSoftReset;
2018-02-17 03:55:32 -06:00
return;
}
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay == 7144)
2018-02-17 03:55:32 -06:00
FadeOutBGM(8);
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay == 6840)
2018-02-17 03:55:32 -06:00
m4aSongNumStart(MUS_END);
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay--;
2018-02-17 03:55:32 -06:00
}
}
2021-03-28 23:56:42 -04:00
#undef tDelay
2021-03-29 12:39:49 -04:00
static void Task_CreditsSoftReset(u8 taskId)
2018-02-17 03:55:32 -06:00
{
if (!gPaletteFade.active)
2021-03-28 23:56:42 -04:00
SoftReset(RESET_ALL);
2018-02-17 03:55:32 -06:00
}
2019-08-17 12:01:38 -04:00
static void ResetGpuAndVram(void)
2018-02-17 03:55:32 -06:00
{
SetGpuReg(REG_OFFSET_DISPCNT, 0);
SetGpuReg(REG_OFFSET_BG3HOFS, 0);
SetGpuReg(REG_OFFSET_BG3VOFS, 0);
SetGpuReg(REG_OFFSET_BG2HOFS, 0);
SetGpuReg(REG_OFFSET_BG2VOFS, 0);
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
SetGpuReg(REG_OFFSET_BG0HOFS, 0);
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
SetGpuReg(REG_OFFSET_BLDY, 0);
DmaFill16(3, 0, (void *)VRAM, VRAM_SIZE);
DmaFill32(3, 0, (void *)OAM, OAM_SIZE);
DmaFill16(3, 0, (void *)(PLTT + 2), PLTT_SIZE - 2);
}
2021-03-29 12:39:49 -04:00
#define tCurrentPage data[2]
#define tDelay data[3]
static void Task_UpdatePage(u8 taskId)
2018-02-17 03:55:32 -06:00
{
int i;
2021-03-29 12:39:49 -04:00
switch (gTasks[taskId].tState)
2018-02-17 03:55:32 -06:00
{
case 0:
case 6:
case 7:
case 8:
case 9:
default:
if (!gPaletteFade.active)
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState = 1;
gTasks[taskId].tDelay = 72;
gTasks[gTasks[taskId].tMainTaskId].tPrintedPage = FALSE;
2021-03-28 23:56:42 -04:00
sUnkVar = 0;
2018-02-17 03:55:32 -06:00
}
return;
case 1:
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay--;
2018-02-17 03:55:32 -06:00
return;
}
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
return;
case 2:
2021-03-29 12:39:49 -04:00
if (gTasks[gTasks[taskId].tMainTaskId].func == Task_CreditsMain)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tCurrentPage < PAGE_COUNT)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
// Print text for this Credits page
2021-03-28 23:56:42 -04:00
for (i = 0; i < ENTRIES_PER_PAGE; i++)
2019-08-17 12:01:38 -04:00
PrintCreditsText(
2021-03-29 12:39:49 -04:00
sCreditsEntryPointerTable[gTasks[taskId].tCurrentPage][i]->text,
5 + i * 16,
2021-03-29 12:39:49 -04:00
sCreditsEntryPointerTable[gTasks[taskId].tCurrentPage][i]->isTitle);
2018-02-17 03:55:32 -06:00
CopyWindowToVram(0, 2);
2018-10-27 00:53:07 +02:00
2021-03-29 12:39:49 -04:00
gTasks[taskId].tCurrentPage++;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
gTasks[gTasks[taskId].tMainTaskId].tPrintedPage = TRUE;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
if (gTasks[gTasks[taskId].tMainTaskId].tCurrentMode == MODE_BIKE_SCENE)
BeginNormalPaletteFade(0x300, 0, 16, 0, COLOR_LIGHT_GREEN);
else // MODE_SHOW_MONS
BeginNormalPaletteFade(0x300, 0, 16, 0, COLOR_DARK_GREEN);
2018-02-17 03:55:32 -06:00
return;
}
2021-03-29 12:39:49 -04:00
// Reached final page of Credits, end task
gTasks[taskId].tState = 10;
2018-02-17 03:55:32 -06:00
return;
}
2021-03-29 12:39:49 -04:00
gTasks[gTasks[taskId].tMainTaskId].tPrintedPage = FALSE;
2018-02-17 03:55:32 -06:00
return;
case 3:
if (!gPaletteFade.active)
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay = 115;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
}
return;
case 4:
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay--;
2018-02-17 03:55:32 -06:00
return;
}
2021-03-29 12:39:49 -04:00
if (CheckChangeScene((u8)gTasks[taskId].tCurrentPage, (u8)gTasks[taskId].tMainTaskId))
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
return;
}
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState++;
if (gTasks[gTasks[taskId].tMainTaskId].tCurrentMode == MODE_BIKE_SCENE)
BeginNormalPaletteFade(0x300, 0, 0, 16, COLOR_LIGHT_GREEN);
else // MODE_SHOW_MONS
BeginNormalPaletteFade(0x300, 0, 0, 16, COLOR_DARK_GREEN);
2018-02-17 03:55:32 -06:00
return;
case 5:
if (!gPaletteFade.active)
{
2021-03-29 12:39:49 -04:00
// Still more Credits pages to show, return to state 2 to print
FillWindowPixelBuffer(0, PIXEL_FILL(0));
2018-02-17 03:55:32 -06:00
CopyWindowToVram(0, 2);
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState = 2;
2018-02-17 03:55:32 -06:00
}
return;
case 10:
2021-03-29 12:39:49 -04:00
gTasks[gTasks[taskId].tMainTaskId].tEndCredits = TRUE;
DestroyTask(taskId);
FreeCreditsBgsAndWindows();
2019-11-03 01:33:46 -04:00
FREE_AND_SET_NULL(sCreditsData);
2018-02-17 03:55:32 -06:00
return;
}
}
2021-03-29 12:39:49 -04:00
#undef tDelay
#define PAGE_INTERVAL (PAGE_COUNT / 9) // 9 scenes (5 bike scenes, 4 Pokémon interludes)
static u8 CheckChangeScene(u8 page, u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
// Starts with bike + ocean + morning (SCENE_OCEAN_MORNING)
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 1)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
// Pokémon interlude
2021-03-29 12:39:49 -04:00
gTasks[taskId].tNextMode = MODE_SHOW_MONS;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 2)
2018-02-17 03:55:32 -06:00
{
// Bike + ocean + sunset
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSceneNum = SCENE_OCEAN_SUNSET;
gTasks[taskId].tNextMode = MODE_BIKE_SCENE;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 3)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
// Pokémon interlude
2021-03-29 12:39:49 -04:00
gTasks[taskId].tNextMode = MODE_SHOW_MONS;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 4)
2018-02-17 03:55:32 -06:00
{
// Bike + forest + sunset
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSceneNum = SCENE_FOREST_RIVAL_ARRIVE;
gTasks[taskId].tNextMode = MODE_BIKE_SCENE;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 5)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
// Pokémon interlude
2021-03-29 12:39:49 -04:00
gTasks[taskId].tNextMode = MODE_SHOW_MONS;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 6)
2018-02-17 03:55:32 -06:00
{
// Bike + forest + sunset
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSceneNum = SCENE_FOREST_CATCH_RIVAL;
gTasks[taskId].tNextMode = MODE_BIKE_SCENE;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 7)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
// Pokémon interlude
2021-03-29 12:39:49 -04:00
gTasks[taskId].tNextMode = MODE_SHOW_MONS;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (page == PAGE_INTERVAL * 8)
2018-02-17 03:55:32 -06:00
{
// Bike + town + night
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSceneNum = SCENE_CITY_NIGHT;
gTasks[taskId].tNextMode = MODE_BIKE_SCENE;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tNextMode != MODE_NONE)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
// Returns true if changed
2018-02-17 03:55:32 -06:00
return TRUE;
}
return FALSE;
}
2021-03-29 12:39:49 -04:00
#define tDelay data[3]
static void Task_ShowMons(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
u8 spriteId;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
switch (gTasks[taskId].tState)
2018-02-17 03:55:32 -06:00
{
case 0:
break;
case 1:
2021-03-29 12:39:49 -04:00
if (sCreditsData->nextImgPos == POS_LEFT && gTasks[gTasks[taskId].tMainTaskId].tPrintedPage == FALSE)
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
break;
case 2:
2021-03-29 12:39:49 -04:00
if (sCreditsData->imgCounter == NUM_MON_SLIDES || gTasks[gTasks[taskId].tMainTaskId].func != Task_CreditsMain)
2018-02-17 03:55:32 -06:00
break;
spriteId = CreateCreditsMonSprite(sCreditsData->monToShow[sCreditsData->currShownMon],
sMonSpritePos[sCreditsData->nextImgPos][0],
sMonSpritePos[sCreditsData->nextImgPos][1],
2021-03-28 23:56:42 -04:00
sCreditsData->nextImgPos);
2019-11-04 11:38:09 -05:00
if (sCreditsData->currShownMon < sCreditsData->numMonToShow - 1)
2018-02-17 03:55:32 -06:00
{
2019-11-04 11:38:09 -05:00
sCreditsData->currShownMon++;
2021-03-29 12:39:49 -04:00
gSprites[spriteId].data[3] = 50;
2018-02-17 03:55:32 -06:00
}
else
{
2019-11-04 11:38:09 -05:00
sCreditsData->currShownMon = 0;
2021-03-29 12:39:49 -04:00
gSprites[spriteId].data[3] = 512;
2018-02-17 03:55:32 -06:00
}
2019-11-03 01:33:46 -04:00
sCreditsData->imgCounter++;
2021-03-28 23:56:42 -04:00
if (sCreditsData->nextImgPos == POS_RIGHT)
sCreditsData->nextImgPos = POS_LEFT;
2018-02-17 03:55:32 -06:00
else
2019-11-03 01:33:46 -04:00
sCreditsData->nextImgPos++;
2021-03-28 23:56:42 -04:00
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay = 50;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
break;
case 3:
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay != 0)
gTasks[taskId].tDelay--;
2018-02-17 03:55:32 -06:00
else
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState = 1;
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-29 12:39:49 -04:00
#undef tMainTaskId
#undef tDelay
#define tPlayer data[2]
#define tRival data[3]
#define tDelay data[4]
#define tSinIdx data[5]
static void Task_BikeScene(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
switch (gTasks[taskId].tState)
2018-02-17 03:55:32 -06:00
{
case 0:
2021-03-29 12:39:49 -04:00
gIntroCredits_MovingSceneryVOffset = Sin((gTasks[taskId].tSinIdx >> 1) & 0x7F, 12);
gTasks[taskId].tSinIdx++;
2018-02-17 03:55:32 -06:00
break;
case 1:
if (gIntroCredits_MovingSceneryVOffset != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gIntroCredits_MovingSceneryVOffset = Sin((gTasks[taskId].tSinIdx >> 1) & 0x7F, 12);
gTasks[taskId].tSinIdx++;
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayer].data[0] = 2;
gTasks[taskId].tSinIdx = 0;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
}
break;
case 2:
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tSinIdx < 64)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSinIdx++;
gIntroCredits_MovingSceneryVOffset = Sin(gTasks[taskId].tSinIdx & 0x7F, 20);
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
}
break;
case 3:
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayer].data[0] = 3;
gSprites[gTasks[taskId].tRival].data[0] = 1;
gTasks[taskId].tDelay = 120;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
break;
case 4:
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tDelay != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tDelay--;
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSinIdx = 64;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
}
break;
case 5:
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tSinIdx > 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tSinIdx--;
gIntroCredits_MovingSceneryVOffset = Sin(gTasks[taskId].tSinIdx & 0x7F, 20);
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayer].data[0] = 1;
gTasks[taskId].tState++;
2018-02-17 03:55:32 -06:00
}
break;
case 6:
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState = 50;
2018-02-17 03:55:32 -06:00
break;
case 10:
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tRival].data[0] = 2;
gTasks[taskId].tState = 50;
2018-02-17 03:55:32 -06:00
break;
case 20:
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayer].data[0] = 4;
gTasks[taskId].tState = 50;
2018-02-17 03:55:32 -06:00
break;
case 30:
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayer].data[0] = 5;
gSprites[gTasks[taskId].tRival].data[0] = 3;
gTasks[taskId].tState = 50;
2018-02-17 03:55:32 -06:00
break;
case 50:
2021-03-29 12:39:49 -04:00
gTasks[taskId].tState = 0;
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-29 12:39:49 -04:00
#define TIMER_STOP 0x7FFF
#define tTimer data[1]
#define tMainTaskId data[2]
static void Task_CycleSceneryPalette(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
s16 bikeTaskId;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
switch (gTasks[taskId].tState)
2018-02-17 03:55:32 -06:00
{
default:
2021-03-29 12:39:49 -04:00
case SCENE_OCEAN_MORNING:
if (gTasks[taskId].tTimer != TIMER_STOP)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
if (gTasks[gTasks[gTasks[taskId].tMainTaskId].tTaskId_UpdatePage].tCurrentPage == 2)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[gTasks[gTasks[taskId].tMainTaskId].tTaskId_BikeScene].tState = 20;
gTasks[taskId].tTimer = TIMER_STOP;
2018-02-17 03:55:32 -06:00
}
}
CycleSceneryPalette(0);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_OCEAN_SUNSET:
CycleSceneryPalette(0);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_FOREST_RIVAL_ARRIVE:
if (gTasks[taskId].tTimer != TIMER_STOP)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
bikeTaskId = gTasks[gTasks[taskId].tMainTaskId].tTaskId_BikeScene;
2018-02-17 03:55:32 -06:00
// Floor to multiple of 128
2021-03-29 12:39:49 -04:00
if ((gTasks[bikeTaskId].tSinIdx & -128) == 640)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[bikeTaskId].tState = 1;
gTasks[taskId].tTimer = TIMER_STOP;
2018-02-17 03:55:32 -06:00
}
}
CycleSceneryPalette(1);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_FOREST_CATCH_RIVAL:
if (gTasks[taskId].tTimer != TIMER_STOP)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
if (gTasks[taskId].tTimer == 584)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
gTasks[gTasks[gTasks[taskId].tMainTaskId].tTaskId_BikeScene].tState = 10;
gTasks[taskId].tTimer = TIMER_STOP;
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-29 12:39:49 -04:00
gTasks[taskId].tTimer++;
2018-02-17 03:55:32 -06:00
}
}
CycleSceneryPalette(1);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_CITY_NIGHT:
CycleSceneryPalette(2);
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-29 12:39:49 -04:00
static void SetBikeScene(u8 scene, u8 taskId)
2018-02-17 03:55:32 -06:00
{
switch (scene)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
case SCENE_OCEAN_MORNING:
gSprites[gTasks[taskId].tPlayerSpriteId].invisible = FALSE;
gSprites[gTasks[taskId].tRivalSpriteId].invisible = FALSE;
2021-07-07 09:11:52 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].x = DISPLAY_WIDTH + 32;
gSprites[gTasks[taskId].tRivalSpriteId].x = DISPLAY_WIDTH + 32;
gSprites[gTasks[taskId].tPlayerSpriteId].y = 46;
gSprites[gTasks[taskId].tRivalSpriteId].y = 46;
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].data[0] = 0;
gSprites[gTasks[taskId].tRivalSpriteId].data[0] = 0;
gTasks[taskId].tTaskId_BgScenery = CreateBicycleBgAnimationTask(0, 0x2000, 0x20, 8);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_OCEAN_SUNSET:
gSprites[gTasks[taskId].tPlayerSpriteId].invisible = FALSE;
gSprites[gTasks[taskId].tRivalSpriteId].invisible = FALSE;
2021-07-07 09:11:52 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].x = 120;
gSprites[gTasks[taskId].tRivalSpriteId].x = DISPLAY_WIDTH + 32;
gSprites[gTasks[taskId].tPlayerSpriteId].y = 46;
gSprites[gTasks[taskId].tRivalSpriteId].y = 46;
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].data[0] = 0;
gSprites[gTasks[taskId].tRivalSpriteId].data[0] = 0;
gTasks[taskId].tTaskId_BgScenery = CreateBicycleBgAnimationTask(0, 0x2000, 0x20, 8);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_FOREST_RIVAL_ARRIVE:
gSprites[gTasks[taskId].tPlayerSpriteId].invisible = FALSE;
gSprites[gTasks[taskId].tRivalSpriteId].invisible = FALSE;
2021-07-07 09:11:52 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].x = 120;
gSprites[gTasks[taskId].tRivalSpriteId].x = DISPLAY_WIDTH + 32;
gSprites[gTasks[taskId].tPlayerSpriteId].y = 46;
gSprites[gTasks[taskId].tRivalSpriteId].y = 46;
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].data[0] = 0;
gSprites[gTasks[taskId].tRivalSpriteId].data[0] = 0;
gTasks[taskId].tTaskId_BgScenery = CreateBicycleBgAnimationTask(1, 0x2000, 0x200, 8);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_FOREST_CATCH_RIVAL:
gSprites[gTasks[taskId].tPlayerSpriteId].invisible = FALSE;
gSprites[gTasks[taskId].tRivalSpriteId].invisible = FALSE;
2021-07-07 09:11:52 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].x = 120;
gSprites[gTasks[taskId].tRivalSpriteId].x = -32;
gSprites[gTasks[taskId].tPlayerSpriteId].y = 46;
gSprites[gTasks[taskId].tRivalSpriteId].y = 46;
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].data[0] = 0;
gSprites[gTasks[taskId].tRivalSpriteId].data[0] = 0;
gTasks[taskId].tTaskId_BgScenery = CreateBicycleBgAnimationTask(1, 0x2000, 0x200, 8);
2018-02-17 03:55:32 -06:00
break;
2021-03-29 12:39:49 -04:00
case SCENE_CITY_NIGHT:
gSprites[gTasks[taskId].tPlayerSpriteId].invisible = FALSE;
gSprites[gTasks[taskId].tRivalSpriteId].invisible = FALSE;
2021-07-07 09:11:52 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].x = 88;
gSprites[gTasks[taskId].tRivalSpriteId].x = 152;
gSprites[gTasks[taskId].tPlayerSpriteId].y = 46;
gSprites[gTasks[taskId].tRivalSpriteId].y = 46;
2021-03-29 12:39:49 -04:00
gSprites[gTasks[taskId].tPlayerSpriteId].data[0] = 0;
gSprites[gTasks[taskId].tRivalSpriteId].data[0] = 0;
gTasks[taskId].tTaskId_BgScenery = CreateBicycleBgAnimationTask(2, 0x2000, 0x200, 8);
2018-02-17 03:55:32 -06:00
break;
}
2021-03-29 12:39:49 -04:00
gTasks[taskId].tTaskId_SceneryPal = CreateTask(Task_CycleSceneryPalette, 0);
gTasks[gTasks[taskId].tTaskId_SceneryPal].tState = scene;
gTasks[gTasks[taskId].tTaskId_SceneryPal].tTimer = 0;
gTasks[gTasks[taskId].tTaskId_SceneryPal].tMainTaskId = taskId;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
gTasks[taskId].tTaskId_BikeScene = CreateTask(Task_BikeScene, 0);
gTasks[gTasks[taskId].tTaskId_BikeScene].tState = 0;
gTasks[gTasks[taskId].tTaskId_BikeScene].data[1] = taskId; // data[1] is never read
gTasks[gTasks[taskId].tTaskId_BikeScene].tPlayer = gTasks[taskId].tPlayerSpriteId;
gTasks[gTasks[taskId].tTaskId_BikeScene].tRival = gTasks[taskId].tRivalSpriteId;
gTasks[gTasks[taskId].tTaskId_BikeScene].tDelay = 0;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
if (scene == SCENE_FOREST_RIVAL_ARRIVE)
gTasks[gTasks[taskId].tTaskId_BikeScene].tSinIdx = 69;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
#undef tTimer
#undef tDelay
#undef tSinIdx
#undef tRival
#undef tPlayer
static bool8 LoadBikeScene(u8 scene, u8 taskId)
2018-02-17 03:55:32 -06:00
{
u8 spriteId;
switch (gMain.state)
{
default:
case 0:
SetGpuReg(REG_OFFSET_DISPCNT, 0);
SetGpuReg(REG_OFFSET_BG3HOFS, 8);
SetGpuReg(REG_OFFSET_BG3VOFS, 0);
SetGpuReg(REG_OFFSET_BG2HOFS, 0);
SetGpuReg(REG_OFFSET_BG2VOFS, 0);
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
ResetSpriteData();
FreeAllSpritePalettes();
gMain.state = 1;
break;
case 1:
gIntroCredits_MovingSceneryVBase = 34;
gIntroCredits_MovingSceneryVOffset = 0;
LoadCreditsSceneGraphics(scene);
2021-03-28 23:56:42 -04:00
gMain.state++;
2018-02-17 03:55:32 -06:00
break;
case 2:
if (gSaveBlock2Ptr->playerGender == MALE)
{
LoadCompressedSpriteSheet(gSpriteSheet_CreditsBrendan);
LoadCompressedSpriteSheet(gSpriteSheet_CreditsRivalMay);
LoadCompressedSpriteSheet(gSpriteSheet_CreditsBicycle);
LoadSpritePalettes(gSpritePalettes_Credits);
2018-02-17 03:55:32 -06:00
spriteId = CreateIntroBrendanSprite(120, 46);
2021-03-29 12:39:49 -04:00
gTasks[taskId].tPlayerSpriteId = spriteId;
gSprites[spriteId].callback = SpriteCB_Player;
2021-03-28 23:56:42 -04:00
gSprites[spriteId].anims = sAnims_Player;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
spriteId = CreateIntroMaySprite(DISPLAY_WIDTH + 32, 46);
gTasks[taskId].tRivalSpriteId = spriteId;
gSprites[spriteId].callback = SpriteCB_Rival;
2021-03-28 23:56:42 -04:00
gSprites[spriteId].anims = sAnims_Rival;
2018-02-17 03:55:32 -06:00
}
else
{
LoadCompressedSpriteSheet(gSpriteSheet_CreditsMay);
LoadCompressedSpriteSheet(gSpriteSheet_CreditsRivalBrendan);
LoadCompressedSpriteSheet(gSpriteSheet_CreditsBicycle);
LoadSpritePalettes(gSpritePalettes_Credits);
2018-02-17 03:55:32 -06:00
spriteId = CreateIntroMaySprite(120, 46);
2021-03-29 12:39:49 -04:00
gTasks[taskId].tPlayerSpriteId = spriteId;
gSprites[spriteId].callback = SpriteCB_Player;
2021-03-28 23:56:42 -04:00
gSprites[spriteId].anims = sAnims_Player;
2018-02-17 03:55:32 -06:00
2021-03-29 12:39:49 -04:00
spriteId = CreateIntroBrendanSprite(DISPLAY_WIDTH + 32, 46);
gTasks[taskId].tRivalSpriteId = spriteId;
gSprites[spriteId].callback = SpriteCB_Rival;
2021-03-28 23:56:42 -04:00
gSprites[spriteId].anims = sAnims_Rival;
2018-02-17 03:55:32 -06:00
};
2021-03-28 23:56:42 -04:00
gMain.state++;
2018-02-17 03:55:32 -06:00
break;
case 3:
2021-03-29 12:39:49 -04:00
SetBikeScene(scene, taskId);
SetCreditsSceneBgCnt(scene);
2018-02-17 03:55:32 -06:00
gMain.state = 0;
return TRUE;
}
return FALSE;
}
2021-03-29 12:39:49 -04:00
static void ResetCreditsTasks(u8 taskId)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
// Destroy Task_BicycleBgAnimation, if running
if (gTasks[taskId].tTaskId_BgScenery != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
DestroyTask(gTasks[taskId].tTaskId_BgScenery);
gTasks[taskId].tTaskId_BgScenery = 0;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
// Destroy Task_BikeScene, if running
if (gTasks[taskId].tTaskId_BikeScene != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
DestroyTask(gTasks[taskId].tTaskId_BikeScene);
gTasks[taskId].tTaskId_BikeScene = 0;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
// Destroy Task_CycleSceneryPalette, if running
if (gTasks[taskId].tTaskId_SceneryPal != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
DestroyTask(gTasks[taskId].tTaskId_SceneryPal);
gTasks[taskId].tTaskId_SceneryPal = 0;
2018-02-17 03:55:32 -06:00
}
2021-03-29 12:39:49 -04:00
// Destroy Task_ShowMons, if running
if (gTasks[taskId].tTaskId_ShowMons != 0)
2018-02-17 03:55:32 -06:00
{
2021-03-29 12:39:49 -04:00
DestroyTask(gTasks[taskId].tTaskId_ShowMons);
gTasks[taskId].tTaskId_ShowMons = 0;
2018-02-17 03:55:32 -06:00
}
gIntroCredits_MovingSceneryState = INTROCRED_SCENERY_DESTROY;
2018-02-17 03:55:32 -06:00
}
2021-03-28 23:56:42 -04:00
static void LoadTheEndScreen(u16 arg0, u16 arg1, u16 palOffset)
2018-02-17 03:55:32 -06:00
{
u16 baseTile;
u16 i;
LZ77UnCompVram(sCreditsCopyrightEnd_Gfx, (void *)(VRAM + arg0));
2021-03-28 23:56:42 -04:00
LoadPalette(gIntroCopyright_Pal, palOffset, sizeof(gIntroCopyright_Pal));
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
baseTile = (palOffset / 16) << 12;
2018-02-17 03:55:32 -06:00
for (i = 0; i < 32 * 32; i++)
((u16 *) (VRAM + arg1))[i] = baseTile + 1;
}
2021-03-28 23:56:42 -04:00
static u16 GetLetterMapTile(u8 baseTiles)
2018-02-17 03:55:32 -06:00
{
u16 out = (baseTiles & 0x3F) + 80;
2018-02-17 03:55:32 -06:00
if (baseTiles == 0xFF)
2018-02-17 03:55:32 -06:00
return 1;
if (baseTiles & (1 << 7))
2018-02-17 03:55:32 -06:00
out |= 1 << 11;
if (baseTiles & (1 << 6))
2018-02-17 03:55:32 -06:00
out |= 1 << 10;
return out;
}
2021-03-28 23:56:42 -04:00
static void DrawLetterMapTiles(const u8 baseTiles[], u8 baseX, u8 baseY, u16 offset, u16 palette)
2018-02-17 03:55:32 -06:00
{
u8 y, x;
const u16 tileOffset = (palette / 16) << 12;
for (y = 0; y < 5; y++)
{
for (x = 0; x < 3; x++)
2021-03-28 23:56:42 -04:00
((u16 *) (VRAM + offset + (baseY + y) * 64))[baseX + x] = tileOffset + GetLetterMapTile(baseTiles[y * 3 + x]);
2018-02-17 03:55:32 -06:00
}
}
2021-03-28 23:56:42 -04:00
static void DrawTheEnd(u16 offset, u16 palette)
2018-02-17 03:55:32 -06:00
{
u16 pos;
u16 baseTile = (palette / 16) << 12;
for (pos = 0; pos < 32 * 32; pos++)
2021-03-28 23:56:42 -04:00
((u16 *) (VRAM + offset))[pos] = baseTile + 1;
DrawLetterMapTiles(sTheEnd_LetterMap_T, 3, 7, offset, palette);
DrawLetterMapTiles(sTheEnd_LetterMap_H, 7, 7, offset, palette);
DrawLetterMapTiles(sTheEnd_LetterMap_E, 11, 7, offset, palette);
DrawLetterMapTiles(sTheEnd_LetterMap_E, 16, 7, offset, palette);
DrawLetterMapTiles(sTheEnd_LetterMap_N, 20, 7, offset, palette);
DrawLetterMapTiles(sTheEnd_LetterMap_D, 24, 7, offset, palette);
2018-02-17 03:55:32 -06:00
}
2021-03-28 23:56:42 -04:00
#define sState data[0]
2021-03-29 12:39:49 -04:00
static void SpriteCB_Player(struct Sprite *sprite)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
if (gIntroCredits_MovingSceneryState != INTROCRED_SCENERY_NORMAL)
2018-02-17 03:55:32 -06:00
{
DestroySprite(sprite);
return;
}
2021-03-28 23:56:42 -04:00
switch (sprite->sState)
2018-02-17 03:55:32 -06:00
{
case 0:
StartSpriteAnimIfDifferent(sprite, 0);
break;
case 1:
StartSpriteAnimIfDifferent(sprite, 1);
2021-07-07 09:11:52 -04:00
if (sprite->x > -32)
sprite->x--;
2018-02-17 03:55:32 -06:00
break;
case 2:
StartSpriteAnimIfDifferent(sprite, 2);
break;
case 3:
StartSpriteAnimIfDifferent(sprite, 3);
break;
case 4:
StartSpriteAnimIfDifferent(sprite, 0);
2021-07-07 09:11:52 -04:00
if (sprite->x > 120)
sprite->x--;
2018-02-17 03:55:32 -06:00
break;
case 5:
StartSpriteAnimIfDifferent(sprite, 0);
2021-07-07 09:11:52 -04:00
if (sprite->x > -32)
sprite->x--;
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-29 12:39:49 -04:00
static void SpriteCB_Rival(struct Sprite *sprite)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
if (gIntroCredits_MovingSceneryState != INTROCRED_SCENERY_NORMAL)
2018-02-17 03:55:32 -06:00
{
DestroySprite(sprite);
return;
}
2021-03-28 23:56:42 -04:00
switch (sprite->sState)
2018-02-17 03:55:32 -06:00
{
case 0:
2021-07-07 09:11:52 -04:00
sprite->y2 = 0;
2018-02-17 03:55:32 -06:00
StartSpriteAnimIfDifferent(sprite, 0);
break;
case 1:
2021-07-07 09:11:52 -04:00
if (sprite->x > 200)
2018-02-17 03:55:32 -06:00
StartSpriteAnimIfDifferent(sprite, 1);
else
StartSpriteAnimIfDifferent(sprite, 2);
2021-07-07 09:11:52 -04:00
if (sprite->x > -32)
sprite->x -= 2;
sprite->y2 = -gIntroCredits_MovingSceneryVOffset;
2018-02-17 03:55:32 -06:00
break;
case 2:
2021-03-28 23:56:42 -04:00
sprite->data[7]++;
2018-02-17 03:55:32 -06:00
StartSpriteAnimIfDifferent(sprite, 0);
if ((sprite->data[7] & 3) == 0)
2021-07-07 09:11:52 -04:00
sprite->x++;
2018-02-17 03:55:32 -06:00
break;
case 3:
StartSpriteAnimIfDifferent(sprite, 0);
2021-07-07 09:11:52 -04:00
if (sprite->x > -32)
sprite->x--;
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-28 23:56:42 -04:00
#define sPosition data[1]
#define sSpriteId data[6]
static void SpriteCB_CreditsMon(struct Sprite *sprite)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
if (gIntroCredits_MovingSceneryState != INTROCRED_SCENERY_NORMAL)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
FreeAndDestroyMonPicSprite(sprite->sSpriteId);
2018-02-17 03:55:32 -06:00
return;
}
2021-03-28 23:56:42 -04:00
sprite->data[7]++;
switch (sprite->sState)
2018-02-17 03:55:32 -06:00
{
case 0:
default:
sprite->oam.affineMode = ST_OAM_AFFINE_NORMAL;
2021-03-29 12:39:49 -04:00
sprite->oam.matrixNum = sprite->sPosition;
2018-02-17 03:55:32 -06:00
sprite->data[2] = 16;
2021-03-29 12:39:49 -04:00
SetOamMatrix(sprite->sPosition, 0x10000 / sprite->data[2], 0, 0, 0x10000 / sprite->data[2]);
2018-02-17 03:55:32 -06:00
sprite->invisible = FALSE;
2021-03-28 23:56:42 -04:00
sprite->sState = 1;
2018-02-17 03:55:32 -06:00
break;
case 1:
if (sprite->data[2] < 256)
{
sprite->data[2] += 8;
2021-03-29 12:39:49 -04:00
SetOamMatrix(sprite->sPosition, 0x10000 / sprite->data[2], 0, 0, 0x10000 / sprite->data[2]);
2018-02-17 03:55:32 -06:00
}
else
{
2021-03-28 23:56:42 -04:00
sprite->sState++;
2018-02-17 03:55:32 -06:00
}
2021-03-28 23:56:42 -04:00
switch (sprite->sPosition)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
case POS_LEFT + 1:
2018-02-17 03:55:32 -06:00
if ((sprite->data[7] & 3) == 0)
2021-07-07 09:11:52 -04:00
sprite->y++;
sprite->x -= 2;
2018-02-17 03:55:32 -06:00
break;
2021-03-28 23:56:42 -04:00
case POS_CENTER + 1:
2018-02-17 03:55:32 -06:00
break;
2021-03-28 23:56:42 -04:00
case POS_RIGHT + 1:
2018-02-17 03:55:32 -06:00
if ((sprite->data[7] & 3) == 0)
2021-07-07 09:11:52 -04:00
sprite->y++;
sprite->x += 2;
2018-02-17 03:55:32 -06:00
break;
}
break;
case 2:
if (sprite->data[3] != 0)
{
2021-03-28 23:56:42 -04:00
sprite->data[3]--;
2018-02-17 03:55:32 -06:00
}
else
{
2018-12-26 13:43:07 +01:00
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_EFFECT_BLEND | BLDCNT_TGT2_BG0 | BLDCNT_TGT2_BG1 | BLDCNT_TGT2_BG2 | BLDCNT_TGT2_BG3);
2018-12-26 14:03:48 +01:00
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(16, 0));
sprite->oam.objMode = ST_OAM_OBJ_BLEND;
2018-02-17 03:55:32 -06:00
sprite->data[3] = 16;
2021-03-28 23:56:42 -04:00
sprite->sState++;
2018-02-17 03:55:32 -06:00
}
break;
case 3:
if (sprite->data[3] != 0)
{
int data3;
2021-03-28 23:56:42 -04:00
sprite->data[3]--;
2018-02-17 03:55:32 -06:00
data3 = 16 - sprite->data[3];
SetGpuReg(REG_OFFSET_BLDALPHA, (data3 << 8) + sprite->data[3]);
}
else
{
sprite->invisible = TRUE;
2021-03-28 23:56:42 -04:00
sprite->sState = 9;
2018-02-17 03:55:32 -06:00
}
break;
case 9:
2021-03-28 23:56:42 -04:00
sprite->sState++;
2018-02-17 03:55:32 -06:00
break;
case 10:
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
2018-08-19 01:06:10 +02:00
FreeAndDestroyMonPicSprite(sprite->data[6]);
2018-02-17 03:55:32 -06:00
break;
}
}
2021-03-28 23:56:42 -04:00
#define sMonSpriteId data[0]
static u8 CreateCreditsMonSprite(u16 nationalDexNum, s16 x, s16 y, u16 position)
2018-02-17 03:55:32 -06:00
{
2021-03-28 23:56:42 -04:00
u8 monSpriteId;
u8 bgSpriteId;
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
monSpriteId = CreateMonSpriteFromNationalDexNumber(nationalDexNum, x, y, position);
gSprites[monSpriteId].oam.priority = 1;
gSprites[monSpriteId].sPosition = position + 1;
gSprites[monSpriteId].invisible = TRUE;
gSprites[monSpriteId].callback = SpriteCB_CreditsMon;
gSprites[monSpriteId].sSpriteId = monSpriteId;
2018-02-17 03:55:32 -06:00
2021-07-07 09:11:52 -04:00
bgSpriteId = CreateSprite(&sSpriteTemplate_CreditsMonBg, gSprites[monSpriteId].x, gSprites[monSpriteId].y, 1);
2021-03-28 23:56:42 -04:00
gSprites[bgSpriteId].sMonSpriteId = monSpriteId;
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
StartSpriteAnimIfDifferent(&gSprites[bgSpriteId], position);
2018-02-17 03:55:32 -06:00
2021-03-28 23:56:42 -04:00
return monSpriteId;
2018-02-17 03:55:32 -06:00
}
2021-03-28 23:56:42 -04:00
static void SpriteCB_CreditsMonBg(struct Sprite *sprite)
2018-02-17 03:55:32 -06:00
{
if (gSprites[sprite->sMonSpriteId].data[0] == 10
2021-03-28 23:56:42 -04:00
|| gIntroCredits_MovingSceneryState != INTROCRED_SCENERY_NORMAL)
2018-02-17 03:55:32 -06:00
{
DestroySprite(sprite);
return;
}
2021-03-28 23:56:42 -04:00
// Copy sprite data from the associated Pokémon
sprite->invisible = gSprites[sprite->sMonSpriteId].invisible;
sprite->oam.objMode = gSprites[sprite->sMonSpriteId].oam.objMode;
sprite->oam.affineMode = gSprites[sprite->sMonSpriteId].oam.affineMode;
sprite->oam.matrixNum = gSprites[sprite->sMonSpriteId].oam.matrixNum;
2021-07-07 09:11:52 -04:00
sprite->x = gSprites[sprite->sMonSpriteId].x;
sprite->y = gSprites[sprite->sMonSpriteId].y;
2018-02-17 03:55:32 -06:00
}
2019-11-03 01:33:46 -04:00
static void DeterminePokemonToShow(void)
2018-02-17 03:55:32 -06:00
{
u16 starter = SpeciesToNationalPokedexNum(GetStarterPokemon(VarGet(VAR_STARTER_MON)));
2018-02-17 03:55:32 -06:00
u16 page;
u16 dexNum;
u16 j;
2019-11-03 01:33:46 -04:00
// Go through the Pokedex, and anything that has gotten caught we put into our massive array.
// This basically packs all of the caught pokemon into the front of the array
for (dexNum = 1, j = 0; dexNum < NATIONAL_DEX_COUNT; dexNum++)
2018-02-17 03:55:32 -06:00
{
if (GetSetPokedexFlag(dexNum, FLAG_GET_CAUGHT))
2018-02-17 03:55:32 -06:00
{
2019-11-04 11:38:09 -05:00
sCreditsData->caughtMonIds[j] = dexNum;
2018-02-17 03:55:32 -06:00
j++;
}
}
2019-11-03 01:33:46 -04:00
// Fill the rest of the array with zeroes
for (dexNum = j; dexNum < NATIONAL_DEX_COUNT; dexNum++)
sCreditsData->caughtMonIds[dexNum] = NATIONAL_DEX_NONE;
2018-02-17 03:55:32 -06:00
2019-11-04 11:38:09 -05:00
// Cap the number of pokemon we care about to NUM_MON_SLIDES, the max we show in the credits scene (-1 for the starter)
sCreditsData->numCaughtMon = j;
if (sCreditsData->numCaughtMon < NUM_MON_SLIDES)
sCreditsData->numMonToShow = j;
2018-02-17 03:55:32 -06:00
else
2019-11-04 11:38:09 -05:00
sCreditsData->numMonToShow = NUM_MON_SLIDES;
2018-02-17 03:55:32 -06:00
2019-11-03 01:33:46 -04:00
// Loop through our list of caught pokemon and select randomly from it to fill the images to show
2018-02-17 03:55:32 -06:00
j = 0;
do
{
2019-11-03 01:33:46 -04:00
// Select a random mon, insert into array
2019-11-04 11:38:09 -05:00
page = Random() % sCreditsData->numCaughtMon;
sCreditsData->monToShow[j] = sCreditsData->caughtMonIds[page];
2019-11-03 01:33:46 -04:00
// Remove the select mon from the array, and condense array entries
2018-02-17 03:55:32 -06:00
j++;
2019-11-04 11:38:09 -05:00
sCreditsData->caughtMonIds[page] = 0;
sCreditsData->numCaughtMon--;
if (page != sCreditsData->numCaughtMon)
2018-02-17 03:55:32 -06:00
{
2019-11-03 01:33:46 -04:00
// Instead of looping through and moving everything down, just take from the end. Order doesn't matter after all.
2019-11-04 11:38:09 -05:00
sCreditsData->caughtMonIds[page] = sCreditsData->caughtMonIds[sCreditsData->numCaughtMon];
sCreditsData->caughtMonIds[sCreditsData->numCaughtMon] = 0;
2018-02-17 03:55:32 -06:00
}
}
2019-11-04 11:38:09 -05:00
while (sCreditsData->numCaughtMon != 0 && j < NUM_MON_SLIDES);
2018-02-17 03:55:32 -06:00
2019-11-03 01:33:46 -04:00
// If we don't have enough pokemon in the dex to fill everything, copy the selected mon into the end of the array, so it loops
2019-11-04 11:38:09 -05:00
if (sCreditsData->numMonToShow < NUM_MON_SLIDES)
2018-02-17 03:55:32 -06:00
{
2019-11-04 11:38:09 -05:00
for (j = sCreditsData->numMonToShow, page = 0; j < NUM_MON_SLIDES; j++)
2018-02-17 03:55:32 -06:00
{
2019-11-04 11:38:09 -05:00
sCreditsData->monToShow[j] = sCreditsData->monToShow[page];
2018-02-17 03:55:32 -06:00
page++;
2019-11-04 11:38:09 -05:00
if (page == sCreditsData->numMonToShow)
2018-02-17 03:55:32 -06:00
page = 0;
}
2019-11-03 01:33:46 -04:00
// Ensure the last pokemon is our starter
sCreditsData->monToShow[NUM_MON_SLIDES - 1] = starter;
2018-02-17 03:55:32 -06:00
}
else
{
2019-11-03 01:33:46 -04:00
// Check to see if our starter has already appeared in this list, break if it has
2019-11-04 11:38:09 -05:00
for (dexNum = 0; sCreditsData->monToShow[dexNum] != starter && dexNum < NUM_MON_SLIDES; dexNum++);
2018-02-17 03:55:32 -06:00
2019-11-03 01:33:46 -04:00
// If it has, swap it with the last pokemon, to ensure our starter is the last image
2019-11-04 11:38:09 -05:00
if (dexNum < sCreditsData->numMonToShow - 1)
2018-02-17 03:55:32 -06:00
{
2019-11-04 11:38:09 -05:00
sCreditsData->monToShow[dexNum] = sCreditsData->monToShow[NUM_MON_SLIDES-1];
sCreditsData->monToShow[NUM_MON_SLIDES - 1] = starter;
2018-02-17 03:55:32 -06:00
}
else
{
2019-11-03 01:33:46 -04:00
// Ensure the last pokemon is our starter
sCreditsData->monToShow[NUM_MON_SLIDES - 1] = starter;
2018-02-17 03:55:32 -06:00
}
}
2019-11-04 11:38:09 -05:00
sCreditsData->numMonToShow = NUM_MON_SLIDES;
2018-02-17 03:55:32 -06:00
}