pokeemerald/src/field_special_scene.c

356 lines
10 KiB
C
Raw Normal View History

2017-09-22 22:45:47 -04:00
#include "global.h"
2018-09-11 02:04:03 +01:00
#include "event_data.h"
#include "event_object_movement.h"
2018-12-27 16:30:47 -06:00
#include "field_camera.h"
#include "field_screen_effect.h"
#include "field_specials.h"
#include "fieldmap.h"
2018-09-11 02:04:03 +01:00
#include "main.h"
2018-12-27 16:30:47 -06:00
#include "overworld.h"
2017-09-22 22:45:47 -04:00
#include "palette.h"
#include "script.h"
2018-09-11 02:04:03 +01:00
#include "script_movement.h"
#include "sound.h"
#include "sprite.h"
#include "task.h"
2019-05-25 20:26:29 +01:00
#include "constants/event_objects.h"
2019-11-21 14:03:35 -05:00
#include "constants/event_object_movement.h"
2019-11-08 03:37:46 -05:00
#include "constants/field_specials.h"
2018-09-11 02:04:03 +01:00
#include "constants/songs.h"
#include "constants/metatile_labels.h"
2017-09-22 22:45:47 -04:00
#define SECONDS(value) ((signed) (60.0 * value + 0.5))
// porthole states
enum
{
INIT_PORTHOLE,
IDLE_CHECK,
EXECUTE_MOVEMENT,
EXIT_PORTHOLE,
};
2018-09-11 02:04:03 +01:00
//. rodata
static const s8 gTruckCamera_HorizontalTable[] = {0, 0, 0, 0, 0, 0, 0, 0, 1, 2, 2, 2, 2, 2, 2, -1, -1, -1, 0};
2019-11-08 03:37:46 -05:00
static const u8 sSSTidalSailEastMovementScript[] =
{
2019-11-14 18:58:33 -05:00
MOVEMENT_ACTION_WALK_FAST_RIGHT,
2019-11-08 03:37:46 -05:00
MOVEMENT_ACTION_STEP_END
};
static const u8 sSSTidalSailWestMovementScript[] =
{
2019-11-14 18:58:33 -05:00
MOVEMENT_ACTION_WALK_FAST_LEFT,
2019-11-08 03:37:46 -05:00
MOVEMENT_ACTION_STEP_END
};
2017-09-22 22:45:47 -04:00
2018-09-11 02:04:03 +01:00
// .text
2018-12-27 16:30:47 -06:00
static void Task_Truck3(u8);
2017-09-22 22:45:47 -04:00
s16 GetTruckCameraBobbingY(int a1)
2017-09-22 22:45:47 -04:00
{
if (!(a1 % 120))
return -1;
else if ((a1 % 10) <= 4)
return 1;
return 0;
}
s16 GetTruckBoxMovement(int a1) // for the box movement?
2017-09-22 22:45:47 -04:00
{
if (!((a1 + 120) % 180))
return -1;
return 0;
}
void Task_Truck1(u8 taskId)
{
s16 *data = gTasks[taskId].data;
2018-08-15 18:50:56 -07:00
s16 cameraXpan = 0, cameraYpan = 0;
s16 box1, box2, box3;
2017-09-22 22:45:47 -04:00
2021-07-05 13:54:43 -04:00
box1 = GetTruckBoxMovement(data[0] + 30) * 4;
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, 3 - cameraXpan, box1 + 3);
box2 = GetTruckBoxMovement(data[0]) * 2;
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -cameraXpan, box2 - 3);
box3 = GetTruckBoxMovement(data[0]) * 4;
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -3 - cameraXpan, box3);
2017-09-22 22:45:47 -04:00
2017-09-23 12:42:06 -04:00
if (++data[0] == SECONDS(500)) // this will never run
2018-08-15 18:50:56 -07:00
data[0] = 0; // reset the timer if it gets stuck.
2017-09-22 22:45:47 -04:00
cameraYpan = GetTruckCameraBobbingY(data[0]);
2018-08-15 18:50:56 -07:00
SetCameraPanning(cameraXpan, cameraYpan);
2017-09-22 22:45:47 -04:00
}
void Task_Truck2(u8 taskId)
{
s16 *data = gTasks[taskId].data;
s16 cameraYpan;
s16 cameraXpan;
s16 box1;
s16 box2;
s16 box3;
data[0]++;
data[2]++;
if (data[0] > 5)
{
data[0] = 0;
data[1]++;
}
if ((u16)data[1] == 19)
{
DestroyTask(taskId);
}
else
{
if (gTruckCamera_HorizontalTable[data[1]] == 2)
gTasks[taskId].func = Task_Truck3;
cameraXpan = gTruckCamera_HorizontalTable[data[1]];
cameraYpan = GetTruckCameraBobbingY(data[2]);
SetCameraPanning(cameraXpan, cameraYpan);
box1 = GetTruckBoxMovement(data[2] + 30) * 4;
2021-07-05 13:54:43 -04:00
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, 3 - cameraXpan, box1 + 3);
2017-09-22 22:45:47 -04:00
box2 = GetTruckBoxMovement(data[2]) * 2;
2021-07-05 13:54:43 -04:00
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -cameraXpan, box2 - 3);
2017-09-22 22:45:47 -04:00
box3 = GetTruckBoxMovement(data[2]) * 4;
2021-07-05 13:54:43 -04:00
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -3 - cameraXpan, box3);
2017-09-22 22:45:47 -04:00
}
}
2018-12-27 16:30:47 -06:00
static void Task_Truck3(u8 taskId)
2017-09-22 22:45:47 -04:00
{
s16 *data = gTasks[taskId].data;
s16 cameraXpan;
s16 cameraYpan;
data[0]++;
if (data[0] > 5)
{
data[0] = 0;
data[1]++;
}
if ((u16)data[1] == 19)
{
DestroyTask(taskId);
}
else
{
cameraXpan = gTruckCamera_HorizontalTable[data[1]];
cameraYpan = 0;
SetCameraPanning(cameraXpan, 0);
2021-07-05 13:54:43 -04:00
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, 3 - cameraXpan, cameraYpan + 3);
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -cameraXpan, cameraYpan - 3);
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -3 - cameraXpan, cameraYpan);
2017-09-22 22:45:47 -04:00
}
}
void Task_HandleTruckSequence(u8 taskId)
{
s16 *data = gTasks[taskId].data;
switch (data[0])
{
/*
Each case has a timer which is handled with data[1], incrementing
until it reaches the if function's condition, which sets the next task up.
*/
case 0:
data[1]++;
if (data[1] == SECONDS(1.5))
{
SetCameraPanningCallback(NULL);
data[1] = 0; // reset the timer.
data[2] = CreateTask(Task_Truck1, 0xA);
data[0] = 1; // run the next case.
2020-08-20 18:02:00 -04:00
PlaySE(SE_TRUCK_MOVE);
2017-09-22 22:45:47 -04:00
}
break;
case 1:
data[1]++;
if (data[1] == SECONDS(2.5))
{
2019-12-17 03:24:44 -05:00
FadeInFromBlack();
2017-09-22 22:45:47 -04:00
data[1] = 0;
data[0] = 2;
}
break;
case 2:
data[1]++;
if (!gPaletteFade.active && data[1] > SECONDS(5))
{
data[1] = 0;
DestroyTask(data[2]);
data[3] = CreateTask(Task_Truck2, 0xA);
data[0] = 3;
2020-08-20 18:02:00 -04:00
PlaySE(SE_TRUCK_STOP);
2017-09-22 22:45:47 -04:00
}
break;
case 3:
if (!gTasks[data[3]].isActive) // is Truck2 no longer active (is Truck3 active?)
{
InstallCameraPanAheadCallback();
data[1] = 0;
data[0] = 4;
}
break;
case 4:
data[1]++;
if (data[1] == 90)
{
2020-08-20 18:02:00 -04:00
PlaySE(SE_TRUCK_UNLOAD);
2017-09-22 22:45:47 -04:00
data[1] = 0;
data[0] = 5;
}
break;
case 5:
data[1]++;
if (data[1] == 120)
{
2020-07-02 00:17:14 -04:00
MapGridSetMetatileIdAt(11, 8, METATILE_InsideOfTruck_ExitLight_Top);
MapGridSetMetatileIdAt(11, 9, METATILE_InsideOfTruck_ExitLight_Mid);
MapGridSetMetatileIdAt(11, 10, METATILE_InsideOfTruck_ExitLight_Bottom);
2017-09-22 22:45:47 -04:00
DrawWholeMapView();
2020-08-20 18:02:00 -04:00
PlaySE(SE_TRUCK_DOOR);
2017-09-22 22:45:47 -04:00
DestroyTask(taskId);
ScriptContext2_Disable();
}
break;
}
}
void ExecuteTruckSequence(void)
{
2020-07-02 00:17:14 -04:00
MapGridSetMetatileIdAt(11, 8, METATILE_InsideOfTruck_DoorClosedFloor_Top);
MapGridSetMetatileIdAt(11, 9, METATILE_InsideOfTruck_DoorClosedFloor_Mid);
MapGridSetMetatileIdAt(11, 10, METATILE_InsideOfTruck_DoorClosedFloor_Bottom);
2017-09-22 22:45:47 -04:00
DrawWholeMapView();
ScriptContext2_Enable();
CpuFastFill(0, gPlttBufferFaded, 0x400);
CreateTask(Task_HandleTruckSequence, 0xA);
}
void EndTruckSequence(u8 taskId)
{
if (!FuncIsActiveTask(Task_HandleTruckSequence))
{
2021-07-05 13:54:43 -04:00
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, 3, 3);
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, 0, -3);
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, -3, 0);
2017-09-22 22:45:47 -04:00
}
}
2019-12-17 03:24:44 -05:00
bool8 TrySetPortholeWarpDestination(void)
2017-09-22 22:45:47 -04:00
{
s8 mapGroup, mapNum;
s16 x, y;
2019-12-17 03:24:44 -05:00
if (GetSSTidalLocation(&mapGroup, &mapNum, &x, &y) != SS_TIDAL_LOCATION_CURRENTS)
2017-09-22 22:45:47 -04:00
{
return FALSE;
}
else
{
2018-12-27 16:30:47 -06:00
SetWarpDestination(mapGroup, mapNum, -1, x, y);
2017-09-22 22:45:47 -04:00
return TRUE;
}
}
void Task_HandlePorthole(u8 taskId)
{
s16 *data = gTasks[taskId].data;
2019-11-08 03:37:46 -05:00
u16 *cruiseState = GetVarPointer(VAR_SS_TIDAL_STATE);
2017-09-22 22:45:47 -04:00
struct WarpData *location = &gSaveBlock1Ptr->location;
switch (data[0])
{
case INIT_PORTHOLE: // finish fading before making porthole finish.
if (!gPaletteFade.active)
{
data[1] = 0;
data[0] = EXECUTE_MOVEMENT; // execute movement before checking if should be exited. strange?
}
break;
2019-11-08 03:37:46 -05:00
case IDLE_CHECK:
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
2017-09-22 22:45:47 -04:00
data[1] = 1;
if (!ScriptMovement_IsObjectMovementFinished(OBJ_EVENT_ID_PLAYER, location->mapNum, location->mapGroup))
2017-09-22 22:45:47 -04:00
return;
if (CountSSTidalStep(1) == TRUE)
{
2019-11-08 03:37:46 -05:00
if (*cruiseState == SS_TIDAL_DEPART_SLATEPORT)
*cruiseState = SS_TIDAL_EXIT_CURRENTS_RIGHT;
2017-09-22 22:45:47 -04:00
else
2019-11-08 03:37:46 -05:00
*cruiseState = SS_TIDAL_EXIT_CURRENTS_LEFT;
data[0] = EXIT_PORTHOLE;
2017-09-22 22:45:47 -04:00
return;
}
2019-11-08 03:37:46 -05:00
data[0] = EXECUTE_MOVEMENT;
//fallthrough
case EXECUTE_MOVEMENT:
2017-09-22 22:45:47 -04:00
if (data[1])
{
2019-11-08 03:37:46 -05:00
data[0] = EXIT_PORTHOLE;
2017-09-22 22:45:47 -04:00
return;
}
2019-11-08 03:37:46 -05:00
if (*cruiseState == SS_TIDAL_DEPART_SLATEPORT)
2017-09-22 22:45:47 -04:00
{
ScriptMovement_StartObjectMovementScript(OBJ_EVENT_ID_PLAYER, location->mapNum, location->mapGroup, sSSTidalSailEastMovementScript);
2019-11-08 03:37:46 -05:00
data[0] = IDLE_CHECK;
2017-09-22 22:45:47 -04:00
}
else
{
ScriptMovement_StartObjectMovementScript(OBJ_EVENT_ID_PLAYER, location->mapNum, location->mapGroup, sSSTidalSailWestMovementScript);
2019-11-08 03:37:46 -05:00
data[0] = IDLE_CHECK;
2017-09-22 22:45:47 -04:00
}
break;
2019-11-08 03:37:46 -05:00
case EXIT_PORTHOLE:
FlagClear(FLAG_DONT_TRANSITION_MUSIC);
2019-03-10 08:10:59 -04:00
FlagClear(FLAG_HIDE_MAP_NAME_POPUP);
2018-12-27 16:30:47 -06:00
SetWarpDestinationToDynamicWarp(0);
2018-12-28 15:11:15 -06:00
DoDiveWarp();
2017-09-22 22:45:47 -04:00
DestroyTask(taskId);
break;
}
}
2019-11-08 03:37:46 -05:00
static void ShowSSTidalWhileSailing(void)
2017-09-22 22:45:47 -04:00
{
u8 spriteId = AddPseudoObjectEvent(OBJ_EVENT_GFX_SS_TIDAL, SpriteCallbackDummy, 112, 80, 0);
2017-09-22 22:45:47 -04:00
gSprites[spriteId].coordOffsetEnabled = FALSE;
2019-11-08 03:37:46 -05:00
if (VarGet(VAR_SS_TIDAL_STATE) == SS_TIDAL_DEPART_SLATEPORT)
StartSpriteAnim(&gSprites[spriteId], GetFaceDirectionAnimNum(DIR_EAST));
2017-09-22 22:45:47 -04:00
else
2019-11-08 03:37:46 -05:00
StartSpriteAnim(&gSprites[spriteId], GetFaceDirectionAnimNum(DIR_WEST));
2017-09-22 22:45:47 -04:00
}
2019-12-17 03:24:44 -05:00
void FieldCB_ShowPortholeView(void)
2017-09-22 22:45:47 -04:00
{
2019-11-08 03:37:46 -05:00
ShowSSTidalWhileSailing();
gObjectEvents[gPlayerAvatar.objectEventId].invisible = TRUE;
2019-12-17 03:24:44 -05:00
FadeInFromBlack();
2017-09-22 22:45:47 -04:00
CreateTask(Task_HandlePorthole, 80);
ScriptContext2_Enable();
}
2019-11-08 03:37:46 -05:00
void LookThroughPorthole(void)
2017-09-22 22:45:47 -04:00
{
2017-11-08 15:20:10 -06:00
FlagSet(FLAG_SYS_CRUISE_MODE);
FlagSet(FLAG_DONT_TRANSITION_MUSIC);
2019-03-10 08:10:59 -04:00
FlagSet(FLAG_HIDE_MAP_NAME_POPUP);
2018-12-27 16:30:47 -06:00
SetDynamicWarp(0, gSaveBlock1Ptr->location.mapGroup, gSaveBlock1Ptr->location.mapNum, -1);
2019-12-17 03:24:44 -05:00
TrySetPortholeWarpDestination();
DoPortholeWarp();
2017-09-22 22:45:47 -04:00
}