1cb659df8c
- Determined how the various script contexts were used and renamed accordingly. - ScriptContext2_Enable/Disable => Lock/UnlockPlayerFieldControls - The sole purpose of the flag is to make sure the player can't move around in the overworld. It has nothing to do with script contexts. - ScriptContext1 => ScriptContext - It is the global script context used to set up scripts which run over many frames. - ScriptContext2_RunNewScript => RunScriptImmediately - ScriptContext2's sole purpose was to run scripts immediately and in a separate context, usually while the global context is waiting for things like map loads or screen changes.
385 lines
12 KiB
C
385 lines
12 KiB
C
#include "global.h"
|
|
#include "event_data.h"
|
|
#include "event_object_movement.h"
|
|
#include "field_camera.h"
|
|
#include "field_screen_effect.h"
|
|
#include "field_specials.h"
|
|
#include "fieldmap.h"
|
|
#include "main.h"
|
|
#include "overworld.h"
|
|
#include "palette.h"
|
|
#include "script.h"
|
|
#include "script_movement.h"
|
|
#include "sound.h"
|
|
#include "sprite.h"
|
|
#include "task.h"
|
|
#include "constants/event_objects.h"
|
|
#include "constants/event_object_movement.h"
|
|
#include "constants/field_specials.h"
|
|
#include "constants/songs.h"
|
|
#include "constants/metatile_labels.h"
|
|
|
|
// Most of the boxes in the moving truck are map tiles, with the
|
|
// exception of three boxes that are map events that jostle around
|
|
// while the truck is driving. In addition, their sprite's placement
|
|
// is slightly offset to make them look less perfectly stacked.
|
|
// Box 1 (LOCALID_TRUCK_BOX_TOP)
|
|
#define BOX1_X_OFFSET 3
|
|
#define BOX1_Y_OFFSET 3
|
|
// Box 2 (LOCALID_TRUCK_BOX_BOTTOM_L)
|
|
#define BOX2_X_OFFSET 0
|
|
#define BOX2_Y_OFFSET -3
|
|
// Box 3 (LOCALID_TRUCK_BOX_BOTTOM_R)
|
|
#define BOX3_X_OFFSET -3
|
|
#define BOX3_Y_OFFSET 0
|
|
|
|
// porthole states
|
|
enum
|
|
{
|
|
INIT_PORTHOLE,
|
|
IDLE_CHECK,
|
|
EXECUTE_MOVEMENT,
|
|
EXIT_PORTHOLE,
|
|
};
|
|
|
|
static const s8 sTruckCamera_HorizontalTable[] = {0, 0, 0, 0, 0, 0, 0, 0, 1, 2, 2, 2, 2, 2, 2, -1, -1, -1, 0};
|
|
|
|
static const u8 sSSTidalSailEastMovementScript[] =
|
|
{
|
|
MOVEMENT_ACTION_WALK_FAST_RIGHT,
|
|
MOVEMENT_ACTION_STEP_END
|
|
};
|
|
|
|
static const u8 sSSTidalSailWestMovementScript[] =
|
|
{
|
|
MOVEMENT_ACTION_WALK_FAST_LEFT,
|
|
MOVEMENT_ACTION_STEP_END
|
|
};
|
|
|
|
static void Task_Truck3(u8);
|
|
|
|
static s16 GetTruckCameraBobbingY(int time)
|
|
{
|
|
if (!(time % 120))
|
|
return -1;
|
|
else if ((time % 10) <= 4)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
// Determines the frequency that the truck boxes bounce at.
|
|
// The return value of this function is multiplied and added
|
|
// to the boxes resting y offset, the result of which is that
|
|
// when it returns 0 they remain vertically still and when it
|
|
// returns -1 they jump upward.
|
|
// Box 1 has 30 added to the time so it jumps earlier, and
|
|
// box 2 has the return value multiplied by less, so it doesn't
|
|
// jump as high.
|
|
static s16 GetTruckBoxYMovement(int time)
|
|
{
|
|
if (!((time + 120) % 180))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define tTimer data[0]
|
|
|
|
static void Task_Truck1(u8 taskId)
|
|
{
|
|
s16 *data = gTasks[taskId].data;
|
|
s16 cameraXpan = 0, cameraYpan = 0;
|
|
s16 yBox1, yBox2, yBox3;
|
|
|
|
yBox1 = GetTruckBoxYMovement(tTimer + 30) * 4;
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX1_X_OFFSET - cameraXpan, BOX1_Y_OFFSET + yBox1);
|
|
yBox2 = GetTruckBoxYMovement(tTimer) * 2;
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX2_X_OFFSET - cameraXpan, BOX2_Y_OFFSET + yBox2);
|
|
yBox3 = GetTruckBoxYMovement(tTimer) * 4;
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX3_X_OFFSET - cameraXpan, BOX3_Y_OFFSET + yBox3);
|
|
|
|
// Arbitrary timer limit that won't be reached
|
|
if (++tTimer == 30000)
|
|
tTimer = 0;
|
|
|
|
cameraYpan = GetTruckCameraBobbingY(tTimer);
|
|
SetCameraPanning(cameraXpan, cameraYpan);
|
|
}
|
|
|
|
#undef tTimer
|
|
|
|
#define tTimerHorizontal data[0]
|
|
#define tMoveStep data[1]
|
|
#define tTimerVertical data[2]
|
|
|
|
static void Task_Truck2(u8 taskId)
|
|
{
|
|
s16 *data = gTasks[taskId].data;
|
|
s16 cameraYpan, cameraXpan;
|
|
s16 yBox1, yBox2, yBox3;
|
|
|
|
tTimerHorizontal++;
|
|
tTimerVertical++;
|
|
|
|
if (tTimerHorizontal > 5)
|
|
{
|
|
tTimerHorizontal = 0;
|
|
tMoveStep++;
|
|
}
|
|
if ((u16)tMoveStep == ARRAY_COUNT(sTruckCamera_HorizontalTable))
|
|
{
|
|
// Never reached, the task function is changed below before finishing the table
|
|
DestroyTask(taskId);
|
|
}
|
|
else
|
|
{
|
|
if (sTruckCamera_HorizontalTable[tMoveStep] == 2)
|
|
gTasks[taskId].func = Task_Truck3;
|
|
|
|
cameraXpan = sTruckCamera_HorizontalTable[tMoveStep];
|
|
cameraYpan = GetTruckCameraBobbingY(tTimerVertical);
|
|
SetCameraPanning(cameraXpan, cameraYpan);
|
|
yBox1 = GetTruckBoxYMovement(tTimerVertical + 30) * 4;
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX1_X_OFFSET - cameraXpan, BOX1_Y_OFFSET + yBox1);
|
|
yBox2 = GetTruckBoxYMovement(tTimerVertical) * 2;
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX2_X_OFFSET - cameraXpan, BOX2_Y_OFFSET + yBox2);
|
|
yBox3 = GetTruckBoxYMovement(tTimerVertical) * 4;
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX3_X_OFFSET - cameraXpan, BOX3_Y_OFFSET + yBox3);
|
|
}
|
|
}
|
|
|
|
static void Task_Truck3(u8 taskId)
|
|
{
|
|
s16 *data = gTasks[taskId].data;
|
|
s16 cameraXpan, cameraYpan;
|
|
|
|
tTimerHorizontal++;
|
|
|
|
if (tTimerHorizontal > 5)
|
|
{
|
|
tTimerHorizontal = 0;
|
|
tMoveStep++;
|
|
}
|
|
|
|
if ((u16)tMoveStep == ARRAY_COUNT(sTruckCamera_HorizontalTable))
|
|
{
|
|
DestroyTask(taskId);
|
|
}
|
|
else
|
|
{
|
|
cameraXpan = sTruckCamera_HorizontalTable[tMoveStep];
|
|
cameraYpan = 0;
|
|
SetCameraPanning(cameraXpan, cameraYpan);
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX1_X_OFFSET - cameraXpan, BOX1_Y_OFFSET + cameraYpan);
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX2_X_OFFSET - cameraXpan, BOX2_Y_OFFSET + cameraYpan);
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX3_X_OFFSET - cameraXpan, BOX3_Y_OFFSET + cameraYpan);
|
|
}
|
|
}
|
|
|
|
#undef tTimerHorizontal
|
|
#undef tMoveStep
|
|
#undef tTimerVertical
|
|
|
|
#define tState data[0]
|
|
#define tTimer data[1]
|
|
#define tTaskId1 data[2]
|
|
#define tTaskId2 data[3]
|
|
|
|
static void Task_HandleTruckSequence(u8 taskId)
|
|
{
|
|
s16 *data = gTasks[taskId].data;
|
|
|
|
switch (tState)
|
|
{
|
|
case 0:
|
|
tTimer++;
|
|
if (tTimer == 90)
|
|
{
|
|
SetCameraPanningCallback(NULL);
|
|
tTimer = 0;
|
|
tTaskId1 = CreateTask(Task_Truck1, 0xA);
|
|
tState = 1;
|
|
PlaySE(SE_TRUCK_MOVE);
|
|
}
|
|
break;
|
|
case 1:
|
|
tTimer++;
|
|
if (tTimer == 150)
|
|
{
|
|
FadeInFromBlack();
|
|
tTimer = 0;
|
|
tState = 2;
|
|
}
|
|
break;
|
|
case 2:
|
|
tTimer++;
|
|
if (!gPaletteFade.active && tTimer > 300)
|
|
{
|
|
tTimer = 0;
|
|
DestroyTask(tTaskId1);
|
|
tTaskId2 = CreateTask(Task_Truck2, 0xA);
|
|
tState = 3;
|
|
PlaySE(SE_TRUCK_STOP);
|
|
}
|
|
break;
|
|
case 3:
|
|
if (!gTasks[tTaskId2].isActive)
|
|
{
|
|
// Task_Truck2 / Task_Truck3 has finished
|
|
InstallCameraPanAheadCallback();
|
|
tTimer = 0;
|
|
tState = 4;
|
|
}
|
|
break;
|
|
case 4:
|
|
tTimer++;
|
|
if (tTimer == 90)
|
|
{
|
|
PlaySE(SE_TRUCK_UNLOAD);
|
|
tTimer = 0;
|
|
tState = 5;
|
|
}
|
|
break;
|
|
case 5:
|
|
tTimer++;
|
|
if (tTimer == 120)
|
|
{
|
|
MapGridSetMetatileIdAt(4 + MAP_OFFSET, 1 + MAP_OFFSET, METATILE_InsideOfTruck_ExitLight_Top);
|
|
MapGridSetMetatileIdAt(4 + MAP_OFFSET, 2 + MAP_OFFSET, METATILE_InsideOfTruck_ExitLight_Mid);
|
|
MapGridSetMetatileIdAt(4 + MAP_OFFSET, 3 + MAP_OFFSET, METATILE_InsideOfTruck_ExitLight_Bottom);
|
|
DrawWholeMapView();
|
|
PlaySE(SE_TRUCK_DOOR);
|
|
DestroyTask(taskId);
|
|
UnlockPlayerFieldControls();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ExecuteTruckSequence(void)
|
|
{
|
|
MapGridSetMetatileIdAt(4 + MAP_OFFSET, 1 + MAP_OFFSET, METATILE_InsideOfTruck_DoorClosedFloor_Top);
|
|
MapGridSetMetatileIdAt(4 + MAP_OFFSET, 2 + MAP_OFFSET, METATILE_InsideOfTruck_DoorClosedFloor_Mid);
|
|
MapGridSetMetatileIdAt(4 + MAP_OFFSET, 3 + MAP_OFFSET, METATILE_InsideOfTruck_DoorClosedFloor_Bottom);
|
|
DrawWholeMapView();
|
|
LockPlayerFieldControls();
|
|
CpuFastFill(0, gPlttBufferFaded, 0x400);
|
|
CreateTask(Task_HandleTruckSequence, 0xA);
|
|
}
|
|
|
|
void EndTruckSequence(u8 taskId)
|
|
{
|
|
if (!FuncIsActiveTask(Task_HandleTruckSequence))
|
|
{
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_TOP, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX1_X_OFFSET, BOX1_Y_OFFSET);
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_L, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX2_X_OFFSET, BOX2_Y_OFFSET);
|
|
SetObjectEventSpritePosByLocalIdAndMap(LOCALID_TRUCK_BOX_BOTTOM_R, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, BOX3_X_OFFSET, BOX3_Y_OFFSET);
|
|
}
|
|
}
|
|
|
|
bool8 TrySetPortholeWarpDestination(void)
|
|
{
|
|
s8 mapGroup, mapNum;
|
|
s16 x, y;
|
|
|
|
if (GetSSTidalLocation(&mapGroup, &mapNum, &x, &y) != SS_TIDAL_LOCATION_CURRENTS)
|
|
{
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
SetWarpDestination(mapGroup, mapNum, WARP_ID_NONE, x, y);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
void Task_HandlePorthole(u8 taskId)
|
|
{
|
|
s16 *data = gTasks[taskId].data;
|
|
u16 *cruiseState = GetVarPointer(VAR_SS_TIDAL_STATE);
|
|
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;
|
|
case IDLE_CHECK:
|
|
if (JOY_NEW(A_BUTTON))
|
|
data[1] = 1;
|
|
if (!ScriptMovement_IsObjectMovementFinished(OBJ_EVENT_ID_PLAYER, location->mapNum, location->mapGroup))
|
|
return;
|
|
if (CountSSTidalStep(1) == TRUE)
|
|
{
|
|
if (*cruiseState == SS_TIDAL_DEPART_SLATEPORT)
|
|
*cruiseState = SS_TIDAL_EXIT_CURRENTS_RIGHT;
|
|
else
|
|
*cruiseState = SS_TIDAL_EXIT_CURRENTS_LEFT;
|
|
data[0] = EXIT_PORTHOLE;
|
|
return;
|
|
}
|
|
data[0] = EXECUTE_MOVEMENT;
|
|
//fallthrough
|
|
case EXECUTE_MOVEMENT:
|
|
if (data[1])
|
|
{
|
|
data[0] = EXIT_PORTHOLE;
|
|
return;
|
|
}
|
|
|
|
if (*cruiseState == SS_TIDAL_DEPART_SLATEPORT)
|
|
{
|
|
ScriptMovement_StartObjectMovementScript(OBJ_EVENT_ID_PLAYER, location->mapNum, location->mapGroup, sSSTidalSailEastMovementScript);
|
|
data[0] = IDLE_CHECK;
|
|
}
|
|
else
|
|
{
|
|
ScriptMovement_StartObjectMovementScript(OBJ_EVENT_ID_PLAYER, location->mapNum, location->mapGroup, sSSTidalSailWestMovementScript);
|
|
data[0] = IDLE_CHECK;
|
|
}
|
|
break;
|
|
case EXIT_PORTHOLE:
|
|
FlagClear(FLAG_DONT_TRANSITION_MUSIC);
|
|
FlagClear(FLAG_HIDE_MAP_NAME_POPUP);
|
|
SetWarpDestinationToDynamicWarp(0);
|
|
DoDiveWarp();
|
|
DestroyTask(taskId);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void ShowSSTidalWhileSailing(void)
|
|
{
|
|
u8 spriteId = CreateObjectGraphicsSprite(OBJ_EVENT_GFX_SS_TIDAL, SpriteCallbackDummy, 112, 80, 0);
|
|
|
|
gSprites[spriteId].coordOffsetEnabled = FALSE;
|
|
|
|
if (VarGet(VAR_SS_TIDAL_STATE) == SS_TIDAL_DEPART_SLATEPORT)
|
|
StartSpriteAnim(&gSprites[spriteId], GetFaceDirectionAnimNum(DIR_EAST));
|
|
else
|
|
StartSpriteAnim(&gSprites[spriteId], GetFaceDirectionAnimNum(DIR_WEST));
|
|
}
|
|
|
|
void FieldCB_ShowPortholeView(void)
|
|
{
|
|
ShowSSTidalWhileSailing();
|
|
gObjectEvents[gPlayerAvatar.objectEventId].invisible = TRUE;
|
|
FadeInFromBlack();
|
|
CreateTask(Task_HandlePorthole, 80);
|
|
LockPlayerFieldControls();
|
|
}
|
|
|
|
void LookThroughPorthole(void)
|
|
{
|
|
FlagSet(FLAG_SYS_CRUISE_MODE);
|
|
FlagSet(FLAG_DONT_TRANSITION_MUSIC);
|
|
FlagSet(FLAG_HIDE_MAP_NAME_POPUP);
|
|
SetDynamicWarp(0, gSaveBlock1Ptr->location.mapGroup, gSaveBlock1Ptr->location.mapNum, WARP_ID_NONE);
|
|
TrySetPortholeWarpDestination();
|
|
DoPortholeWarp();
|
|
}
|