pokeemerald/src/field_player_avatar.c

2214 lines
65 KiB
C
Raw Normal View History

2018-07-19 21:22:35 +04:30
#include "global.h"
#include "main.h"
2018-07-19 21:22:35 +04:30
#include "bike.h"
#include "event_data.h"
2018-09-06 03:11:33 +01:00
#include "event_object_movement.h"
#include "field_camera.h"
#include "field_effect.h"
#include "field_effect_helpers.h"
2018-11-13 14:19:04 +00:00
#include "field_player_avatar.h"
#include "fieldmap.h"
#include "menu.h"
2018-09-06 03:11:33 +01:00
#include "metatile_behavior.h"
#include "overworld.h"
#include "party_menu.h"
#include "random.h"
#include "rotating_gate.h"
#include "script.h"
#include "sound.h"
2018-09-06 03:11:33 +01:00
#include "sprite.h"
#include "strings.h"
2018-09-06 03:11:33 +01:00
#include "task.h"
#include "tv.h"
#include "wild_encounter.h"
#include "constants/abilities.h"
2018-09-06 03:11:33 +01:00
#include "constants/event_objects.h"
2019-11-21 14:03:35 -05:00
#include "constants/event_object_movement.h"
2018-11-13 14:19:04 +00:00
#include "constants/field_effects.h"
2018-09-06 03:11:33 +01:00
#include "constants/flags.h"
#include "constants/maps.h"
#include "constants/moves.h"
#include "constants/songs.h"
#include "constants/species.h"
2018-08-12 22:31:58 +04:30
static EWRAM_DATA u8 gUnknown_0203734C = 0;
EWRAM_DATA struct ObjectEvent gObjectEvents[OBJECT_EVENTS_COUNT] = {};
EWRAM_DATA struct PlayerAvatar gPlayerAvatar = {};
2018-11-13 14:19:04 +00:00
// static declarations
static u8 ObjectEventCB2_NoMovement2();
static bool8 TryInterruptObjectEventSpecialAnim(struct ObjectEvent *, u8);
static void npc_clear_strange_bits(struct ObjectEvent *);
2018-08-12 22:31:58 +04:30
static void MovePlayerAvatarUsingKeypadInput(u8, u16, u16);
static void PlayerAllowForcedMovementIfMovingSameDirection();
2018-11-13 14:19:04 +00:00
static bool8 TryDoMetatileBehaviorForcedMovement();
2018-08-12 22:31:58 +04:30
static u8 GetForcedMovementByMetatileBehavior();
static bool8 ForcedMovement_None(void);
static bool8 ForcedMovement_Slip(void);
static bool8 ForcedMovement_WalkSouth(void);
static bool8 ForcedMovement_WalkNorth(void);
static bool8 ForcedMovement_WalkWest(void);
static bool8 ForcedMovement_WalkEast(void);
static bool8 ForcedMovement_PushedSouthByCurrent(void);
static bool8 ForcedMovement_PushedNorthByCurrent(void);
static bool8 ForcedMovement_PushedWestByCurrent(void);
static bool8 ForcedMovement_PushedEastByCurrent(void);
static bool8 ForcedMovement_SlideSouth(void);
static bool8 ForcedMovement_SlideNorth(void);
static bool8 ForcedMovement_SlideWest(void);
static bool8 ForcedMovement_SlideEast(void);
2020-04-01 02:54:26 -04:00
static bool8 ForcedMovement_MatJump(void);
static bool8 ForcedMovement_MatSpin(void);
static bool8 ForcedMovement_MuddySlope(void);
2018-11-13 14:19:04 +00:00
static void MovePlayerNotOnBike(u8, u16);
static u8 CheckMovementInputNotOnBike(u8);
static void PlayerNotOnBikeNotMoving(u8, u16);
static void PlayerNotOnBikeTurningInPlace(u8, u16);
static void PlayerNotOnBikeMoving(u8, u16);
static u8 CheckForPlayerAvatarCollision(u8);
static u8 sub_808B028(u8);
static u8 sub_808B164(struct ObjectEvent *, s16, s16, u8, u8);
2019-10-06 15:40:10 -05:00
static bool8 CanStopSurfing(s16, s16, u8);
2018-11-13 14:19:04 +00:00
static bool8 ShouldJumpLedge(s16, s16, u8);
2019-10-06 15:40:10 -05:00
static bool8 TryPushBoulder(s16, s16, u8);
static void CheckAcroBikeCollision(s16, s16, u8, u8 *);
2018-11-13 14:19:04 +00:00
static void DoPlayerAvatarTransition(void);
static void PlayerAvatarTransition_Dummy(struct ObjectEvent *a);
static void PlayerAvatarTransition_Normal(struct ObjectEvent *a);
static void PlayerAvatarTransition_MachBike(struct ObjectEvent *a);
static void PlayerAvatarTransition_AcroBike(struct ObjectEvent *a);
static void PlayerAvatarTransition_Surfing(struct ObjectEvent *a);
static void PlayerAvatarTransition_Underwater(struct ObjectEvent *a);
static void PlayerAvatarTransition_ReturnToField(struct ObjectEvent *a);
2018-11-13 14:19:04 +00:00
static bool8 player_is_anim_in_certain_ranges(void);
static bool8 sub_808B618(void);
static bool8 PlayerIsAnimActive(void);
static bool8 PlayerCheckIfAnimFinishedOrInactive(void);
2019-02-01 11:50:45 +01:00
static void PlayerRun(u8);
2018-11-13 14:19:04 +00:00
static void PlayerNotOnBikeCollide(u8);
static void PlayerNotOnBikeCollideWithFarawayIslandMew(u8);
static void PlayCollisionSoundIfNotFacingWarp(u8 a);
static void sub_808C280(struct ObjectEvent *);
2018-11-13 14:19:04 +00:00
static void StartStrengthAnim(u8, u8);
2020-04-01 02:54:26 -04:00
static void Task_PushBoulder(u8 taskId);
static u8 PushBoulder_Start(struct Task *task, struct ObjectEvent *playerObject, struct ObjectEvent *strengthObject);
static u8 PushBoulder_Move(struct Task *task, struct ObjectEvent *playerObject, struct ObjectEvent *strengthObject);
static u8 PushBoulder_End(struct Task *task, struct ObjectEvent *playerObject, struct ObjectEvent *strengthObject);
2018-11-13 14:19:04 +00:00
static void DoPlayerMatJump(void);
static void DoPlayerAvatarSecretBaseMatJump(u8 taskId);
static u8 PlayerAvatar_DoSecretBaseMatJump(struct Task *task, struct ObjectEvent *objectEvent);
2018-11-13 14:19:04 +00:00
static void DoPlayerMatSpin(void);
static void PlayerAvatar_DoSecretBaseMatSpin(u8 taskId);
static bool8 PlayerAvatar_SecretBaseMatSpinStep0(struct Task *task, struct ObjectEvent *objectEvent);
static bool8 PlayerAvatar_SecretBaseMatSpinStep1(struct Task *task, struct ObjectEvent *objectEvent);
static bool8 PlayerAvatar_SecretBaseMatSpinStep2(struct Task *task, struct ObjectEvent *objectEvent);
static bool8 PlayerAvatar_SecretBaseMatSpinStep3(struct Task *task, struct ObjectEvent *objectEvent);
2019-10-06 15:40:10 -05:00
static void CreateStopSurfingTask(u8);
static void Task_StopSurfingInit(u8 taskId);
static void Task_WaitStopSurfing(u8 taskId);
2018-11-13 14:19:04 +00:00
static void Task_Fishing(u8 taskId);
static u8 Fishing1(struct Task *task);
static u8 Fishing2(struct Task *task);
static u8 Fishing3(struct Task *task);
static u8 Fishing4(struct Task *task);
static u8 Fishing5(struct Task *task);
static u8 Fishing6(struct Task *task);
static u8 Fishing7(struct Task *task);
static u8 Fishing8(struct Task *task);
static u8 Fishing9(struct Task *task);
static u8 Fishing10(struct Task *task);
static u8 Fishing11(struct Task *task);
static u8 Fishing12(struct Task *task);
static u8 Fishing13(struct Task *task);
static u8 Fishing14(struct Task *task);
static u8 Fishing15(struct Task *task);
static u8 Fishing16(struct Task *task);
2018-11-13 14:19:04 +00:00
static void AlignFishingAnimationFrames(void);
static u8 sub_808D38C(struct ObjectEvent *object, s16 *a1);
2018-11-13 14:19:04 +00:00
// .rodata
2018-09-06 03:11:33 +01:00
2018-10-16 14:55:16 +01:00
static bool8 (*const sForcedMovementTestFuncs[])(u8) =
{
MetatileBehavior_IsTrickHouseSlipperyFloor,
MetatileBehavior_IsIce_2,
MetatileBehavior_IsWalkSouth,
MetatileBehavior_IsWalkNorth,
MetatileBehavior_IsWalkWest,
MetatileBehavior_IsWalkEast,
MetatileBehavior_IsSouthwardCurrent,
MetatileBehavior_IsNorthwardCurrent,
MetatileBehavior_IsWestwardCurrent,
MetatileBehavior_IsEastwardCurrent,
MetatileBehavior_IsSlideSouth,
MetatileBehavior_IsSlideNorth,
MetatileBehavior_IsSlideWest,
MetatileBehavior_IsSlideEast,
MetatileBehavior_IsWaterfall,
MetatileBehavior_IsSecretBaseJumpMat,
MetatileBehavior_IsSecretBaseSpinMat,
MetatileBehavior_IsMuddySlope,
};
2018-08-12 22:31:58 +04:30
2018-10-16 14:55:16 +01:00
static bool8 (*const sForcedMovementFuncs[])(void) =
{
ForcedMovement_None,
ForcedMovement_Slip,
ForcedMovement_Slip,
ForcedMovement_WalkSouth,
ForcedMovement_WalkNorth,
ForcedMovement_WalkWest,
ForcedMovement_WalkEast,
ForcedMovement_PushedSouthByCurrent,
ForcedMovement_PushedNorthByCurrent,
ForcedMovement_PushedWestByCurrent,
ForcedMovement_PushedEastByCurrent,
ForcedMovement_SlideSouth,
ForcedMovement_SlideNorth,
ForcedMovement_SlideWest,
ForcedMovement_SlideEast,
ForcedMovement_PushedSouthByCurrent,
2020-04-01 02:54:26 -04:00
ForcedMovement_MatJump,
ForcedMovement_MatSpin,
ForcedMovement_MuddySlope,
};
2018-08-12 22:31:58 +04:30
static void (*const gUnknown_08497490[])(u8, u16) =
{
PlayerNotOnBikeNotMoving,
PlayerNotOnBikeTurningInPlace,
PlayerNotOnBikeMoving,
};
2019-10-06 15:40:10 -05:00
static bool8 (*const sAcroBikeTrickMetatiles[])(u8) =
{
MetatileBehavior_IsBumpySlope,
MetatileBehavior_IsIsolatedVerticalRail,
MetatileBehavior_IsIsolatedHorizontalRail,
MetatileBehavior_IsVerticalRail,
MetatileBehavior_IsHorizontalRail,
};
2018-08-12 22:31:58 +04:30
2019-10-06 15:40:10 -05:00
static const u8 sAcroBikeTrickCollisionTypes[] = {
COLLISION_WHEELIE_HOP,
COLLISION_ISOLATED_VERTICAL_RAIL,
COLLISION_ISOLATED_HORIZONTAL_RAIL,
COLLISION_VERTICAL_RAIL,
COLLISION_HORIZONTAL_RAIL,
};
2018-07-19 21:22:35 +04:30
2020-04-01 02:54:26 -04:00
static void (*const sPlayerAvatarTransitionFuncs[])(struct ObjectEvent *) =
2018-09-06 03:11:33 +01:00
{
2020-04-01 02:54:26 -04:00
[PLAYER_AVATAR_STATE_NORMAL] = PlayerAvatarTransition_Normal,
[PLAYER_AVATAR_STATE_MACH_BIKE] = PlayerAvatarTransition_MachBike,
[PLAYER_AVATAR_STATE_ACRO_BIKE] = PlayerAvatarTransition_AcroBike,
[PLAYER_AVATAR_STATE_SURFING] = PlayerAvatarTransition_Surfing,
[PLAYER_AVATAR_STATE_UNDERWATER] = PlayerAvatarTransition_Underwater,
[PLAYER_AVATAR_STATE_FIELD_MOVE] = PlayerAvatarTransition_ReturnToField,
[PLAYER_AVATAR_STATE_FISHING] = PlayerAvatarTransition_Dummy,
[PLAYER_AVATAR_STATE_WATERING] = PlayerAvatarTransition_Dummy,
2018-09-06 03:11:33 +01:00
};
static bool8 (*const sArrowWarpMetatileBehaviorChecks[])(u8) =
2018-09-06 03:11:33 +01:00
{
MetatileBehavior_IsSouthArrowWarp,
MetatileBehavior_IsNorthArrowWarp,
MetatileBehavior_IsWestArrowWarp,
MetatileBehavior_IsEastArrowWarp,
};
static const u8 sRivalAvatarGfxIds[][2] =
2018-09-06 03:11:33 +01:00
{
{OBJ_EVENT_GFX_RIVAL_BRENDAN_NORMAL, OBJ_EVENT_GFX_RIVAL_MAY_NORMAL},
{OBJ_EVENT_GFX_RIVAL_BRENDAN_MACH_BIKE, OBJ_EVENT_GFX_RIVAL_MAY_MACH_BIKE},
{OBJ_EVENT_GFX_RIVAL_BRENDAN_ACRO_BIKE, OBJ_EVENT_GFX_RIVAL_MAY_ACRO_BIKE},
{OBJ_EVENT_GFX_RIVAL_BRENDAN_SURFING, OBJ_EVENT_GFX_RIVAL_MAY_SURFING},
{OBJ_EVENT_GFX_BRENDAN_UNDERWATER, OBJ_EVENT_GFX_MAY_UNDERWATER},
{OBJ_EVENT_GFX_RIVAL_BRENDAN_FIELD_MOVE, OBJ_EVENT_GFX_RIVAL_MAY_FIELD_MOVE},
{OBJ_EVENT_GFX_BRENDAN_FISHING, OBJ_EVENT_GFX_MAY_FISHING},
{OBJ_EVENT_GFX_BRENDAN_WATERING, OBJ_EVENT_GFX_MAY_WATERING}
2018-09-06 03:11:33 +01:00
};
static const u8 sPlayerAvatarGfxIds[][2] =
2018-09-06 03:11:33 +01:00
{
{OBJ_EVENT_GFX_BRENDAN_NORMAL, OBJ_EVENT_GFX_MAY_NORMAL},
{OBJ_EVENT_GFX_BRENDAN_MACH_BIKE, OBJ_EVENT_GFX_MAY_MACH_BIKE},
{OBJ_EVENT_GFX_BRENDAN_ACRO_BIKE, OBJ_EVENT_GFX_MAY_ACRO_BIKE},
{OBJ_EVENT_GFX_BRENDAN_SURFING, OBJ_EVENT_GFX_MAY_SURFING},
{OBJ_EVENT_GFX_BRENDAN_UNDERWATER, OBJ_EVENT_GFX_MAY_UNDERWATER},
{OBJ_EVENT_GFX_BRENDAN_FIELD_MOVE, OBJ_EVENT_GFX_MAY_FIELD_MOVE},
{OBJ_EVENT_GFX_BRENDAN_FISHING, OBJ_EVENT_GFX_MAY_FISHING},
{OBJ_EVENT_GFX_BRENDAN_WATERING, OBJ_EVENT_GFX_MAY_WATERING},
2018-09-06 03:11:33 +01:00
};
static const u8 sFRLGAvatarGfxIds[] = {OBJ_EVENT_GFX_RED, OBJ_EVENT_GFX_LEAF};
2018-09-06 03:11:33 +01:00
static const u8 sRSAvatarGfxIds[] = {OBJ_EVENT_GFX_LINK_RS_BRENDAN, OBJ_EVENT_GFX_LINK_RS_MAY};
2018-09-06 03:11:33 +01:00
2020-04-01 02:54:26 -04:00
static const u8 sPlayerAvatarGfxToStateFlag[2][5][2] =
2018-09-06 03:11:33 +01:00
{
2020-04-01 02:54:26 -04:00
[MALE] =
2018-09-06 03:11:33 +01:00
{
2020-04-01 02:54:26 -04:00
{OBJ_EVENT_GFX_BRENDAN_NORMAL, PLAYER_AVATAR_FLAG_ON_FOOT},
{OBJ_EVENT_GFX_BRENDAN_MACH_BIKE, PLAYER_AVATAR_FLAG_MACH_BIKE},
{OBJ_EVENT_GFX_BRENDAN_ACRO_BIKE, PLAYER_AVATAR_FLAG_ACRO_BIKE},
{OBJ_EVENT_GFX_BRENDAN_SURFING, PLAYER_AVATAR_FLAG_SURFING},
{OBJ_EVENT_GFX_BRENDAN_UNDERWATER, PLAYER_AVATAR_FLAG_UNDERWATER},
2018-09-06 03:11:33 +01:00
},
2020-04-01 02:54:26 -04:00
[FEMALE] =
2018-09-06 03:11:33 +01:00
{
2020-04-01 02:54:26 -04:00
{OBJ_EVENT_GFX_MAY_NORMAL, PLAYER_AVATAR_FLAG_ON_FOOT},
{OBJ_EVENT_GFX_MAY_MACH_BIKE, PLAYER_AVATAR_FLAG_MACH_BIKE},
{OBJ_EVENT_GFX_MAY_ACRO_BIKE, PLAYER_AVATAR_FLAG_ACRO_BIKE},
{OBJ_EVENT_GFX_MAY_SURFING, PLAYER_AVATAR_FLAG_SURFING},
{OBJ_EVENT_GFX_MAY_UNDERWATER, PLAYER_AVATAR_FLAG_UNDERWATER},
2018-09-06 03:11:33 +01:00
}
};
static bool8 (*const sArrowWarpMetatileBehaviorChecks2[])(u8) = //Duplicate of sArrowWarpMetatileBehaviorChecks
2018-09-06 03:11:33 +01:00
{
MetatileBehavior_IsSouthArrowWarp,
MetatileBehavior_IsNorthArrowWarp,
MetatileBehavior_IsWestArrowWarp,
MetatileBehavior_IsEastArrowWarp,
};
2020-04-01 02:54:26 -04:00
static bool8 (*const sPushBoulderFuncs[])(struct Task *, struct ObjectEvent *, struct ObjectEvent *) =
2018-09-06 03:11:33 +01:00
{
2020-04-01 02:54:26 -04:00
PushBoulder_Start,
PushBoulder_Move,
PushBoulder_End,
2018-09-06 03:11:33 +01:00
};
static bool8 (*const sPlayerAvatarSecretBaseMatJump[])(struct Task *, struct ObjectEvent *) =
2018-09-06 03:11:33 +01:00
{
PlayerAvatar_DoSecretBaseMatJump,
2018-09-06 03:11:33 +01:00
};
static bool8 (*const sPlayerAvatarSecretBaseMatSpin[])(struct Task *, struct ObjectEvent *) =
2018-09-06 03:11:33 +01:00
{
PlayerAvatar_SecretBaseMatSpinStep0,
PlayerAvatar_SecretBaseMatSpinStep1,
PlayerAvatar_SecretBaseMatSpinStep2,
PlayerAvatar_SecretBaseMatSpinStep3,
2018-09-06 03:11:33 +01:00
};
// .text
2018-11-13 14:19:04 +00:00
2018-07-19 21:22:35 +04:30
void MovementType_Player(struct Sprite *sprite)
{
UpdateObjectEventCurrentMovement(&gObjectEvents[sprite->data[0]], sprite, ObjectEventCB2_NoMovement2);
2018-07-19 21:22:35 +04:30
}
static u8 ObjectEventCB2_NoMovement2(void)
2018-07-19 21:22:35 +04:30
{
return 0;
2018-07-19 21:22:35 +04:30
}
void player_step(u8 direction, u16 newKeys, u16 heldKeys)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-07-19 21:22:35 +04:30
2019-11-21 00:14:40 -05:00
sub_808C280(playerObjEvent);
2018-07-19 21:22:35 +04:30
if (gPlayerAvatar.preventStep == FALSE)
{
Bike_TryAcroBikeHistoryUpdate(newKeys, heldKeys);
2019-11-21 00:14:40 -05:00
if (TryInterruptObjectEventSpecialAnim(playerObjEvent, direction) == 0)
2018-07-19 21:22:35 +04:30
{
2019-11-21 00:14:40 -05:00
npc_clear_strange_bits(playerObjEvent);
2018-07-19 21:22:35 +04:30
DoPlayerAvatarTransition();
if (TryDoMetatileBehaviorForcedMovement() == 0)
{
MovePlayerAvatarUsingKeypadInput(direction, newKeys, heldKeys);
PlayerAllowForcedMovementIfMovingSameDirection();
}
}
}
}
2018-07-25 17:18:49 +04:30
2019-11-21 00:14:40 -05:00
static bool8 TryInterruptObjectEventSpecialAnim(struct ObjectEvent *playerObjEvent, u8 direction)
{
u8 r5 = direction;
u8 r6 = direction;
r6++; r6--;
2018-07-25 17:18:49 +04:30
2019-11-21 00:14:40 -05:00
if (ObjectEventIsMovementOverridden(playerObjEvent)
&& !ObjectEventClearHeldMovementIfFinished(playerObjEvent))
{
2019-11-21 00:14:40 -05:00
u8 heldMovementActionId = ObjectEventGetHeldMovementActionId(playerObjEvent);
2018-07-25 17:47:04 +04:30
if (heldMovementActionId > MOVEMENT_ACTION_WALK_FAST_RIGHT && heldMovementActionId < MOVEMENT_ACTION_WALK_IN_PLACE_NORMAL_DOWN)
{
2018-07-25 17:18:49 +04:30
if (direction == DIR_NONE)
{
return TRUE;
}
2018-07-25 17:18:49 +04:30
2019-11-21 00:14:40 -05:00
if (playerObjEvent->movementDirection != r5)
{
2019-11-21 00:14:40 -05:00
ObjectEventClearHeldMovement(playerObjEvent);
return FALSE;
}
2018-07-25 17:18:49 +04:30
if (!sub_808B028(r6))
{
2019-11-21 00:14:40 -05:00
ObjectEventClearHeldMovement(playerObjEvent);
2018-07-25 17:18:49 +04:30
return FALSE;
}
}
2018-07-25 17:18:49 +04:30
return TRUE;
}
2018-07-25 17:18:49 +04:30
return FALSE;
}
static void npc_clear_strange_bits(struct ObjectEvent *objEvent)
{
objEvent->inanimate = 0;
objEvent->disableAnim = 0;
objEvent->facingDirectionLocked = 0;
gPlayerAvatar.flags &= ~PLAYER_AVATAR_FLAG_DASH;
}
2018-08-12 22:31:58 +04:30
static void MovePlayerAvatarUsingKeypadInput(u8 direction, u16 newKeys, u16 heldKeys)
{
if ((gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_MACH_BIKE)
|| (gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_ACRO_BIKE))
MovePlayerOnBike(direction, newKeys, heldKeys);
else
MovePlayerNotOnBike(direction, heldKeys);
}
2018-08-12 22:31:58 +04:30
static void PlayerAllowForcedMovementIfMovingSameDirection(void)
{
if (gPlayerAvatar.runningState == MOVING)
gPlayerAvatar.flags &= ~PLAYER_AVATAR_FLAG_5;
}
2018-07-21 03:06:40 +04:30
static bool8 TryDoMetatileBehaviorForcedMovement(void)
2018-07-21 03:06:40 +04:30
{
return sForcedMovementFuncs[GetForcedMovementByMetatileBehavior()]();
2018-07-21 03:06:40 +04:30
}
2018-08-12 22:31:58 +04:30
static u8 GetForcedMovementByMetatileBehavior(void)
2018-07-21 03:06:40 +04:30
{
u8 i;
if (!(gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_5))
{
u8 metatileBehavior = gObjectEvents[gPlayerAvatar.objectEventId].currentMetatileBehavior;
2018-07-21 03:06:40 +04:30
for (i = 0; i < 18; i++)
{
2018-10-16 14:55:16 +01:00
if (sForcedMovementTestFuncs[i](metatileBehavior))
2018-07-21 03:06:40 +04:30
return i + 1;
}
}
return 0;
}
static bool8 ForcedMovement_None(void)
2018-07-21 03:06:40 +04:30
{
if (gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_6)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-07-21 03:06:40 +04:30
2019-11-21 00:14:40 -05:00
playerObjEvent->facingDirectionLocked = 0;
playerObjEvent->enableAnim = 1;
SetObjectEventDirection(playerObjEvent, playerObjEvent->facingDirection);
2018-07-21 03:06:40 +04:30
gPlayerAvatar.flags &= ~PLAYER_AVATAR_FLAG_6;
}
return FALSE;
}
static u8 DoForcedMovement(u8 direction, void (*b)(u8))
{
struct PlayerAvatar *playerAvatar = &gPlayerAvatar;
2019-10-06 15:40:10 -05:00
u8 collision = CheckForPlayerAvatarCollision(direction);
2018-07-21 03:06:40 +04:30
playerAvatar->flags |= PLAYER_AVATAR_FLAG_6;
2019-10-06 15:40:10 -05:00
if (collision)
2018-07-21 03:06:40 +04:30
{
ForcedMovement_None();
2019-10-06 15:40:10 -05:00
if (collision < COLLISION_STOP_SURFING)
2018-07-21 03:06:40 +04:30
{
return 0;
}
else
{
2019-10-06 15:40:10 -05:00
if (collision == COLLISION_LEDGE_JUMP)
2018-07-21 03:06:40 +04:30
PlayerJumpLedge(direction);
playerAvatar->flags |= PLAYER_AVATAR_FLAG_6;
playerAvatar->runningState = MOVING;
return 1;
}
}
else
{
playerAvatar->runningState = MOVING;
b(direction);
return 1;
}
}
static u8 DoForcedMovementInCurrentDirection(void (*a)(u8))
2018-07-21 03:06:40 +04:30
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-07-21 03:06:40 +04:30
2019-11-21 00:14:40 -05:00
playerObjEvent->disableAnim = 1;
return DoForcedMovement(playerObjEvent->movementDirection, a);
2018-07-21 03:06:40 +04:30
}
static bool8 ForcedMovement_Slip(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovementInCurrentDirection(PlayerGoSpeed2);
}
static bool8 ForcedMovement_WalkSouth(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_SOUTH, PlayerGoSpeed1);
}
static bool8 ForcedMovement_WalkNorth(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_NORTH, PlayerGoSpeed1);
}
static bool8 ForcedMovement_WalkWest(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_WEST, PlayerGoSpeed1);
}
static bool8 ForcedMovement_WalkEast(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_EAST, PlayerGoSpeed1);
}
static bool8 ForcedMovement_PushedSouthByCurrent(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_SOUTH, PlayerRideWaterCurrent);
2018-07-21 03:06:40 +04:30
}
static bool8 ForcedMovement_PushedNorthByCurrent(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_NORTH, PlayerRideWaterCurrent);
2018-07-21 03:06:40 +04:30
}
static bool8 ForcedMovement_PushedWestByCurrent(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_WEST, PlayerRideWaterCurrent);
2018-07-21 03:06:40 +04:30
}
static bool8 ForcedMovement_PushedEastByCurrent(void)
2018-07-21 03:06:40 +04:30
{
return DoForcedMovement(DIR_EAST, PlayerRideWaterCurrent);
2018-07-21 03:06:40 +04:30
}
u8 ForcedMovement_Slide(u8 direction, void (*b)(u8))
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-07-21 03:06:40 +04:30
2019-11-21 00:14:40 -05:00
playerObjEvent->disableAnim = 1;
playerObjEvent->facingDirectionLocked = 1;
2018-07-21 03:06:40 +04:30
return DoForcedMovement(direction, b);
}
static bool8 ForcedMovement_SlideSouth(void)
2018-07-21 03:06:40 +04:30
{
return ForcedMovement_Slide(DIR_SOUTH, PlayerGoSpeed2);
}
static bool8 ForcedMovement_SlideNorth(void)
2018-07-21 03:06:40 +04:30
{
return ForcedMovement_Slide(DIR_NORTH, PlayerGoSpeed2);
}
static bool8 ForcedMovement_SlideWest(void)
2018-07-21 03:06:40 +04:30
{
return ForcedMovement_Slide(DIR_WEST, PlayerGoSpeed2);
}
static bool8 ForcedMovement_SlideEast(void)
2018-07-21 03:06:40 +04:30
{
return ForcedMovement_Slide(DIR_EAST, PlayerGoSpeed2);
}
2020-04-01 02:54:26 -04:00
static bool8 ForcedMovement_MatJump(void)
2018-07-21 03:06:40 +04:30
{
DoPlayerMatJump();
2018-07-21 03:06:40 +04:30
return TRUE;
}
2020-04-01 02:54:26 -04:00
static bool8 ForcedMovement_MatSpin(void)
2018-07-21 03:06:40 +04:30
{
DoPlayerMatSpin();
2018-07-21 03:06:40 +04:30
return TRUE;
}
static bool8 ForcedMovement_MuddySlope(void)
2018-07-21 03:06:40 +04:30
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2018-07-21 03:06:40 +04:30
2019-11-21 00:14:40 -05:00
if (playerObjEvent->movementDirection != DIR_NORTH || GetPlayerSpeed() <= 3)
2018-07-21 03:06:40 +04:30
{
Bike_UpdateBikeCounterSpeed(0);
2019-11-21 00:14:40 -05:00
playerObjEvent->facingDirectionLocked = 1;
2018-07-21 03:06:40 +04:30
return DoForcedMovement(1, PlayerGoSpeed2);
}
else
{
return FALSE;
}
}
2018-08-12 22:31:58 +04:30
static void MovePlayerNotOnBike(u8 direction, u16 heldKeys)
{
gUnknown_08497490[CheckMovementInputNotOnBike(direction)](direction, heldKeys);
}
2018-08-12 22:31:58 +04:30
static u8 CheckMovementInputNotOnBike(u8 direction)
{
if (direction == DIR_NONE)
{
gPlayerAvatar.runningState = NOT_MOVING;
return 0;
}
else if (direction != GetPlayerMovementDirection() && gPlayerAvatar.runningState != MOVING)
{
gPlayerAvatar.runningState = TURN_DIRECTION;
return 1;
}
else
{
gPlayerAvatar.runningState = MOVING;
return 2;
}
}
2018-08-12 22:31:58 +04:30
static void PlayerNotOnBikeNotMoving(u8 direction, u16 heldKeys)
{
PlayerFaceDirection(GetPlayerFacingDirection());
}
2018-08-12 22:31:58 +04:30
static void PlayerNotOnBikeTurningInPlace(u8 direction, u16 heldKeys)
{
PlayerTurnInPlace(direction);
}
2018-08-12 22:31:58 +04:30
static void PlayerNotOnBikeMoving(u8 direction, u16 heldKeys)
{
2019-10-06 15:40:10 -05:00
u8 collision = CheckForPlayerAvatarCollision(direction);
2019-10-06 15:40:10 -05:00
if (collision)
{
2019-10-06 15:40:10 -05:00
if (collision == COLLISION_LEDGE_JUMP)
{
PlayerJumpLedge(direction);
return;
}
else if (collision == COLLISION_OBJECT_EVENT && IsPlayerCollidingWithFarawayIslandMew(direction))
{
PlayerNotOnBikeCollideWithFarawayIslandMew(direction);
return;
}
else
{
2019-10-06 15:40:10 -05:00
u8 adjustedCollision = collision - COLLISION_STOP_SURFING;
if (adjustedCollision > 3)
PlayerNotOnBikeCollide(direction);
2019-10-06 15:40:10 -05:00
return;
}
}
if (gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_SURFING)
{
// speed 2 is fast, same speed as running
PlayerGoSpeed2(direction);
return;
}
2018-10-17 01:11:44 +01:00
if (!(gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_UNDERWATER) && (heldKeys & B_BUTTON) && FlagGet(FLAG_SYS_B_DASH)
&& IsRunningDisallowed(gObjectEvents[gPlayerAvatar.objectEventId].currentMetatileBehavior) == 0)
{
PlayerRun(direction);
gPlayerAvatar.flags |= PLAYER_AVATAR_FLAG_DASH;
return;
}
else
{
PlayerGoSpeed1(direction);
}
}
2018-08-12 22:31:58 +04:30
static u8 CheckForPlayerAvatarCollision(u8 direction)
{
s16 x, y;
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
x = playerObjEvent->currentCoords.x;
y = playerObjEvent->currentCoords.y;
MoveCoords(direction, &x, &y);
2019-11-21 00:14:40 -05:00
return CheckForObjectEventCollision(playerObjEvent, x, y, direction, MapGridGetMetatileBehaviorAt(x, y));
}
2018-08-12 22:31:58 +04:30
static u8 sub_808B028(u8 direction)
{
s16 x, y;
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
x = playerObjEvent->currentCoords.x;
y = playerObjEvent->currentCoords.y;
MoveCoords(direction, &x, &y);
2019-11-21 00:14:40 -05:00
return sub_808B164(playerObjEvent, x, y, direction, MapGridGetMetatileBehaviorAt(x, y));
}
u8 CheckForObjectEventCollision(struct ObjectEvent *objectEvent, s16 x, s16 y, u8 direction, u8 metatileBehavior)
{
u8 collision = GetCollisionAtCoords(objectEvent, x, y, direction);
2019-10-06 15:40:10 -05:00
if (collision == COLLISION_ELEVATION_MISMATCH && CanStopSurfing(x, y, direction))
return COLLISION_STOP_SURFING;
if (ShouldJumpLedge(x, y, direction))
{
IncrementGameStat(GAME_STAT_JUMPED_DOWN_LEDGES);
return COLLISION_LEDGE_JUMP;
}
if (collision == COLLISION_OBJECT_EVENT && TryPushBoulder(x, y, direction))
2019-10-06 15:40:10 -05:00
return COLLISION_PUSHED_BOULDER;
2019-10-06 15:40:10 -05:00
if (collision == COLLISION_NONE)
{
if (CheckForRotatingGatePuzzleCollision(direction, x, y))
2019-10-06 15:40:10 -05:00
return COLLISION_ROTATING_GATE;
CheckAcroBikeCollision(x, y, metatileBehavior, &collision);
}
return collision;
}
static u8 sub_808B164(struct ObjectEvent *objectEvent, s16 x, s16 y, u8 direction, u8 metatileBehavior)
{
u8 collision = GetCollisionAtCoords(objectEvent, x, y, direction);
2019-10-06 15:40:10 -05:00
if (collision == COLLISION_NONE)
{
2019-10-06 15:40:10 -05:00
if (CheckForRotatingGatePuzzleCollisionWithoutAnimation(direction, x, y))
return COLLISION_ROTATING_GATE;
CheckAcroBikeCollision(x, y, metatileBehavior, &collision);
}
return collision;
}
2019-10-06 15:40:10 -05:00
static bool8 CanStopSurfing(s16 x, s16 y, u8 direction)
{
if ((gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_SURFING)
&& MapGridGetZCoordAt(x, y) == 3
&& GetObjectEventIdByXYZ(x, y, 3) == OBJECT_EVENTS_COUNT)
{
2019-10-06 15:40:10 -05:00
CreateStopSurfingTask(direction);
2018-07-25 17:18:49 +04:30
return TRUE;
}
else
{
2018-07-25 17:18:49 +04:30
return FALSE;
}
}
2018-08-12 22:31:58 +04:30
static bool8 ShouldJumpLedge(s16 x, s16 y, u8 z)
{
if (GetLedgeJumpDirection(x, y, z) != 0)
2018-07-25 17:18:49 +04:30
return TRUE;
else
2018-07-25 17:18:49 +04:30
return FALSE;
}
2019-10-06 15:40:10 -05:00
static bool8 TryPushBoulder(s16 x, s16 y, u8 direction)
{
if (FlagGet(FLAG_SYS_USE_STRENGTH))
{
u8 objectEventId = GetObjectEventIdByXY(x, y);
if (objectEventId != 16 && gObjectEvents[objectEventId].graphicsId == OBJ_EVENT_GFX_PUSHABLE_BOULDER)
2018-11-13 14:19:04 +00:00
{
x = gObjectEvents[objectEventId].currentCoords.x;
y = gObjectEvents[objectEventId].currentCoords.y;
2018-11-13 14:19:04 +00:00
MoveCoords(direction, &x, &y);
if (GetCollisionAtCoords(&gObjectEvents[objectEventId], x, y, direction) == COLLISION_NONE
2018-11-13 14:19:04 +00:00
&& MetatileBehavior_IsNonAnimDoor(MapGridGetMetatileBehaviorAt(x, y)) == 0)
{
StartStrengthAnim(objectEventId, direction);
2019-10-06 15:40:10 -05:00
return TRUE;
}
2018-11-13 14:19:04 +00:00
}
}
2019-10-06 15:40:10 -05:00
return FALSE;
}
2019-10-06 15:40:10 -05:00
static void CheckAcroBikeCollision(s16 x, s16 y, u8 metatileBehavior, u8 *collision)
{
u8 i;
2019-10-06 15:40:10 -05:00
for (i = 0; i < ARRAY_COUNT(sAcroBikeTrickMetatiles); i++)
{
2019-10-06 15:40:10 -05:00
if (sAcroBikeTrickMetatiles[i](metatileBehavior))
{
2019-10-06 15:40:10 -05:00
*collision = sAcroBikeTrickCollisionTypes[i];
return;
}
}
}
bool8 IsPlayerCollidingWithFarawayIslandMew(u8 direction)
{
u8 mewObjectId;
struct ObjectEvent *object;
s16 playerX;
s16 playerY;
s16 mewPrevX;
object = &gObjectEvents[gPlayerAvatar.objectEventId];
playerX = object->currentCoords.x;
playerY = object->currentCoords.y;
MoveCoords(direction, &playerX, &playerY);
mewObjectId = GetObjectEventIdByLocalIdAndMap(1, MAP_NUM(FARAWAY_ISLAND_INTERIOR), MAP_GROUP(FARAWAY_ISLAND_INTERIOR));
if (mewObjectId == OBJECT_EVENTS_COUNT)
return FALSE;
object = &gObjectEvents[mewObjectId];
mewPrevX = object->previousCoords.x;
if (mewPrevX == playerX)
{
if (object->previousCoords.y != playerY
|| object->currentCoords.x != mewPrevX
|| object->currentCoords.y != object->previousCoords.y)
{
if (object->previousCoords.x == playerX &&
object->previousCoords.y == playerY)
return TRUE;
}
}
return FALSE;
}
void SetPlayerAvatarTransitionFlags(u16 transitionFlags)
{
2020-04-01 02:54:26 -04:00
gPlayerAvatar.transitionFlags |= transitionFlags;
DoPlayerAvatarTransition();
}
static void DoPlayerAvatarTransition(void)
{
u8 i;
2020-04-01 02:54:26 -04:00
u8 flags = gPlayerAvatar.transitionFlags;
if (flags != 0)
{
2020-04-01 02:54:26 -04:00
for (i = 0; i < ARRAY_COUNT(sPlayerAvatarTransitionFuncs); i++, flags >>= 1)
{
if (flags & 1)
2020-04-01 02:54:26 -04:00
sPlayerAvatarTransitionFuncs[i](&gObjectEvents[gPlayerAvatar.objectEventId]);
}
2020-04-01 02:54:26 -04:00
gPlayerAvatar.transitionFlags = 0;
}
}
static void PlayerAvatarTransition_Dummy(struct ObjectEvent *objEvent)
{
}
static void PlayerAvatarTransition_Normal(struct ObjectEvent *objEvent)
{
ObjectEventSetGraphicsId(objEvent, GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_NORMAL));
ObjectEventTurn(objEvent, objEvent->movementDirection);
SetPlayerAvatarStateMask(PLAYER_AVATAR_FLAG_ON_FOOT);
}
static void PlayerAvatarTransition_MachBike(struct ObjectEvent *objEvent)
{
ObjectEventSetGraphicsId(objEvent, GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_MACH_BIKE));
ObjectEventTurn(objEvent, objEvent->movementDirection);
SetPlayerAvatarStateMask(PLAYER_AVATAR_FLAG_MACH_BIKE);
BikeClearState(0, 0);
}
static void PlayerAvatarTransition_AcroBike(struct ObjectEvent *objEvent)
{
ObjectEventSetGraphicsId(objEvent, GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_ACRO_BIKE));
ObjectEventTurn(objEvent, objEvent->movementDirection);
SetPlayerAvatarStateMask(PLAYER_AVATAR_FLAG_ACRO_BIKE);
BikeClearState(0, 0);
Bike_HandleBumpySlopeJump();
}
static void PlayerAvatarTransition_Surfing(struct ObjectEvent *objEvent)
{
u8 spriteId;
ObjectEventSetGraphicsId(objEvent, GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_SURFING));
ObjectEventTurn(objEvent, objEvent->movementDirection);
SetPlayerAvatarStateMask(PLAYER_AVATAR_FLAG_SURFING);
gFieldEffectArguments[0] = objEvent->currentCoords.x;
gFieldEffectArguments[1] = objEvent->currentCoords.y;
gFieldEffectArguments[2] = gPlayerAvatar.objectEventId;
spriteId = FieldEffectStart(FLDEFF_SURF_BLOB);
objEvent->fieldEffectSpriteId = spriteId;
sub_81555AC(spriteId, 1);
}
static void PlayerAvatarTransition_Underwater(struct ObjectEvent *objEvent)
{
ObjectEventSetGraphicsId(objEvent, GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_UNDERWATER));
ObjectEventTurn(objEvent, objEvent->movementDirection);
SetPlayerAvatarStateMask(PLAYER_AVATAR_FLAG_UNDERWATER);
objEvent->fieldEffectSpriteId = sub_8155800(objEvent->spriteId);
}
static void PlayerAvatarTransition_ReturnToField(struct ObjectEvent *objEvent)
{
gPlayerAvatar.flags |= PLAYER_AVATAR_FLAG_5;
}
void sub_808B578(void)
{
gPlayerAvatar.tileTransitionState = T_NOT_MOVING;
if (PlayerIsAnimActive())
{
if (!PlayerCheckIfAnimFinishedOrInactive())
{
if (!player_is_anim_in_certain_ranges())
gPlayerAvatar.tileTransitionState = T_TILE_TRANSITION;
}
else
{
if (!sub_808B618())
gPlayerAvatar.tileTransitionState = T_TILE_CENTER;
}
}
}
static bool8 player_is_anim_in_certain_ranges(void)
{
u8 movementActionId = gObjectEvents[gPlayerAvatar.objectEventId].movementActionId;
if (movementActionId <= MOVEMENT_ACTION_FACE_RIGHT
|| (movementActionId >= MOVEMENT_ACTION_DELAY_1 && movementActionId <= MOVEMENT_ACTION_DELAY_16)
|| (movementActionId >= MOVEMENT_ACTION_WALK_IN_PLACE_SLOW_DOWN && movementActionId <= MOVEMENT_ACTION_WALK_IN_PLACE_FASTEST_RIGHT)
|| (movementActionId >= MOVEMENT_ACTION_ACRO_WHEELIE_FACE_DOWN && movementActionId <= MOVEMENT_ACTION_ACRO_END_WHEELIE_FACE_RIGHT)
|| (movementActionId >= MOVEMENT_ACTION_ACRO_WHEELIE_IN_PLACE_DOWN && movementActionId <= MOVEMENT_ACTION_ACRO_WHEELIE_IN_PLACE_RIGHT))
return TRUE;
else
return FALSE;
}
static bool8 sub_808B618(void)
{
if (player_is_anim_in_certain_ranges() && gPlayerAvatar.runningState != TURN_DIRECTION)
return TRUE;
else
return FALSE;
}
static bool8 PlayerIsAnimActive(void)
{
return ObjectEventIsMovementOverridden(&gObjectEvents[gPlayerAvatar.objectEventId]);
}
static bool8 PlayerCheckIfAnimFinishedOrInactive(void)
{
return ObjectEventCheckHeldMovementStatus(&gObjectEvents[gPlayerAvatar.objectEventId]);
}
static void PlayerSetCopyableMovement(u8 a)
{
gObjectEvents[gPlayerAvatar.objectEventId].playerCopyableMovement = a;
}
u8 PlayerGetCopyableMovement(void)
{
return gObjectEvents[gPlayerAvatar.objectEventId].playerCopyableMovement;
}
static void sub_808B6BC(u8 a)
{
ObjectEventForceSetHeldMovement(&gObjectEvents[gPlayerAvatar.objectEventId], a);
}
void PlayerSetAnimId(u8 movementActionId, u8 copyableMovement)
{
if (!PlayerIsAnimActive())
{
PlayerSetCopyableMovement(copyableMovement);
ObjectEventSetHeldMovement(&gObjectEvents[gPlayerAvatar.objectEventId], movementActionId);
}
}
// normal speed (1 speed)
void PlayerGoSpeed1(u8 a)
{
PlayerSetAnimId(GetWalkNormalMovementAction(a), 2);
}
// fast speed (2 speed)
void PlayerGoSpeed2(u8 a)
{
PlayerSetAnimId(GetWalkFastMovementAction(a), 2);
}
void PlayerRideWaterCurrent(u8 a)
{
PlayerSetAnimId(GetRideWaterCurrentMovementAction(a), 2);
}
// fastest speed (4 speed)
void PlayerGoSpeed4(u8 a)
{
PlayerSetAnimId(GetWalkFastestMovementAction(a), 2);
}
2018-11-13 14:19:04 +00:00
static void PlayerRun(u8 a)
{
PlayerSetAnimId(GetPlayerRunMovementAction(a), 2);
}
void PlayerOnBikeCollide(u8 a)
{
PlayCollisionSoundIfNotFacingWarp(a);
PlayerSetAnimId(GetWalkInPlaceNormalMovementAction(a), 2);
}
void PlayerOnBikeCollideWithFarawayIslandMew(u8 a)
{
PlayerSetAnimId(GetWalkInPlaceNormalMovementAction(a), 2);
}
2018-11-13 14:19:04 +00:00
static void PlayerNotOnBikeCollide(u8 a)
{
PlayCollisionSoundIfNotFacingWarp(a);
PlayerSetAnimId(GetWalkInPlaceSlowMovementAction(a), 2);
}
2018-11-13 14:19:04 +00:00
static void PlayerNotOnBikeCollideWithFarawayIslandMew(u8 a)
{
PlayerSetAnimId(GetWalkInPlaceSlowMovementAction(a), 2);
}
void PlayerFaceDirection(u8 direction)
{
PlayerSetAnimId(GetFaceDirectionMovementAction(direction), 1);
}
void PlayerTurnInPlace(u8 direction)
{
PlayerSetAnimId(GetWalkInPlaceFastMovementAction(direction), 1);
}
void PlayerJumpLedge(u8 direction)
{
PlaySE(SE_DANSA);
PlayerSetAnimId(GetJump2MovementAction(direction), 8);
}
void sub_808B864(void)
{
if (gPlayerAvatar.tileTransitionState == T_TILE_CENTER || gPlayerAvatar.tileTransitionState == T_NOT_MOVING)
{
if (player_should_look_direction_be_enforced_upon_movement())
sub_808B6BC(GetFaceDirectionMovementAction(gObjectEvents[gPlayerAvatar.objectEventId].facingDirection));
}
}
// wheelie idle
void PlayerIdleWheelie(u8 direction)
{
PlayerSetAnimId(GetAcroWheelieFaceDirectionMovementAction(direction), 1);
}
// normal to wheelie
void PlayerStartWheelie(u8 direction)
{
PlayerSetAnimId(GetAcroPopWheelieFaceDirectionMovementAction(direction), 1);
}
// wheelie to normal
void PlayerEndWheelie(u8 direction)
{
PlayerSetAnimId(GetAcroEndWheelieFaceDirectionMovementAction(direction), 1);
}
// wheelie hopping standing
void PlayerStandingHoppingWheelie(u8 a)
{
PlaySE(SE_JITE_PYOKO);
PlayerSetAnimId(GetAcroWheelieHopFaceDirectionMovementAction(a), 1);
}
// wheelie hopping moving
void PlayerMovingHoppingWheelie(u8 a)
{
PlaySE(SE_JITE_PYOKO);
PlayerSetAnimId(GetAcroWheelieHopDirectionMovementAction(a), 2);
}
// wheelie hopping ledge
void PlayerLedgeHoppingWheelie(u8 a)
{
PlaySE(SE_JITE_PYOKO);
PlayerSetAnimId(GetAcroWheelieJumpDirectionMovementAction(a), 8);
}
// acro turn jump
void PlayerAcroTurnJump(u8 direction)
{
PlaySE(SE_JITE_PYOKO);
PlayerSetAnimId(GetJumpInPlaceTurnAroundMovementAction(direction), 1);
}
void sub_808B980(u8 direction)
{
PlaySE(SE_WALL_HIT);
PlayerSetAnimId(GetAcroWheelieInPlaceDirectionMovementAction(direction), 2);
}
void sub_808B9A4(u8 direction)
{
PlayerSetAnimId(GetAcroPopWheelieMoveDirectionMovementAction(direction), 2);
}
void sub_808B9BC(u8 direction)
{
PlayerSetAnimId(GetAcroWheelieMoveDirectionMovementAction(direction), 2);
}
void sub_808B9D4(u8 direction)
{
PlayerSetAnimId(GetAcroEndWheelieMoveDirectionMovementAction(direction), 2);
}
static void PlayCollisionSoundIfNotFacingWarp(u8 a)
{
s16 x, y;
u8 metatileBehavior = gObjectEvents[gPlayerAvatar.objectEventId].currentMetatileBehavior;
if (!sArrowWarpMetatileBehaviorChecks[a - 1](metatileBehavior))
{
if (a == 2)
{
PlayerGetDestCoords(&x, &y);
MoveCoords(2, &x, &y);
if (MetatileBehavior_IsWarpDoor(MapGridGetMetatileBehaviorAt(x, y)))
return;
}
PlaySE(SE_WALL_HIT);
}
}
void GetXYCoordsOneStepInFrontOfPlayer(s16 *x, s16 *y)
{
*x = gObjectEvents[gPlayerAvatar.objectEventId].currentCoords.x;
*y = gObjectEvents[gPlayerAvatar.objectEventId].currentCoords.y;
MoveCoords(GetPlayerFacingDirection(), x, y);
}
void PlayerGetDestCoords(s16 *x, s16 *y)
{
*x = gObjectEvents[gPlayerAvatar.objectEventId].currentCoords.x;
*y = gObjectEvents[gPlayerAvatar.objectEventId].currentCoords.y;
}
u8 player_get_pos_including_state_based_drift(s16 *x, s16 *y)
{
struct ObjectEvent *object = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-02-01 11:50:45 +01:00
if (object->heldMovementActive && !object->heldMovementFinished && !gSprites[object->spriteId].data[2])
{
*x = object->currentCoords.x;
*y = object->currentCoords.y;
switch (object->movementActionId)
{
case MOVEMENT_ACTION_WALK_NORMAL_DOWN:
case MOVEMENT_ACTION_PLAYER_RUN_DOWN:
(*y)++;
return TRUE;
case MOVEMENT_ACTION_WALK_NORMAL_UP:
case MOVEMENT_ACTION_PLAYER_RUN_UP:
(*y)--;
return TRUE;
case MOVEMENT_ACTION_WALK_NORMAL_LEFT:
case MOVEMENT_ACTION_PLAYER_RUN_LEFT:
(*x)--;
return TRUE;
case MOVEMENT_ACTION_WALK_NORMAL_RIGHT:
case MOVEMENT_ACTION_PLAYER_RUN_RIGHT:
(*x)++;
return TRUE;
}
}
*x = -1;
*y = -1;
return FALSE;
}
u8 GetPlayerFacingDirection(void)
{
return gObjectEvents[gPlayerAvatar.objectEventId].facingDirection;
}
u8 GetPlayerMovementDirection(void)
{
return gObjectEvents[gPlayerAvatar.objectEventId].movementDirection;
}
u8 PlayerGetZCoord(void)
{
return gObjectEvents[gPlayerAvatar.objectEventId].previousElevation;
}
2019-04-05 16:11:24 -05:00
void sub_808BC90(s16 x, s16 y)
{
MoveObjectEventToMapCoords(&gObjectEvents[gPlayerAvatar.objectEventId], x, y);
}
2020-04-01 02:54:26 -04:00
u8 TestPlayerAvatarFlags(u8 flag)
{
2020-04-01 02:54:26 -04:00
return gPlayerAvatar.flags & flag;
}
2020-04-01 02:54:26 -04:00
u8 GetPlayerAvatarFlags(void)
{
return gPlayerAvatar.flags;
}
u8 GetPlayerAvatarObjectId(void)
{
return gPlayerAvatar.spriteId;
}
void sub_808BCE8(void)
{
ForcedMovement_None();
}
void sub_808BCF4(void)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
npc_clear_strange_bits(playerObjEvent);
SetObjectEventDirection(playerObjEvent, playerObjEvent->facingDirection);
if (TestPlayerAvatarFlags(PLAYER_AVATAR_FLAG_MACH_BIKE | PLAYER_AVATAR_FLAG_ACRO_BIKE))
{
Bike_HandleBumpySlopeJump();
Bike_UpdateBikeCounterSpeed(0);
}
}
u8 GetRivalAvatarGraphicsIdByStateIdAndGender(u8 state, u8 gender)
{
return sRivalAvatarGfxIds[state][gender];
}
u8 GetPlayerAvatarGraphicsIdByStateIdAndGender(u8 state, u8 gender)
{
return sPlayerAvatarGfxIds[state][gender];
}
u8 GetFRLGAvatarGraphicsIdByGender(u8 gender)
{
return sFRLGAvatarGfxIds[gender];
}
u8 GetRSAvatarGraphicsIdByGender(u8 gender)
{
return sRSAvatarGfxIds[gender];
}
u8 GetPlayerAvatarGraphicsIdByStateId(u8 state)
{
return GetPlayerAvatarGraphicsIdByStateIdAndGender(state, gPlayerAvatar.gender);
}
u8 unref_GetRivalAvatarGenderByGraphicsId(u8 gfxId)
{
switch (gfxId)
{
case OBJ_EVENT_GFX_RIVAL_MAY_NORMAL:
case OBJ_EVENT_GFX_RIVAL_MAY_MACH_BIKE:
case OBJ_EVENT_GFX_RIVAL_MAY_ACRO_BIKE:
case OBJ_EVENT_GFX_RIVAL_MAY_SURFING:
case OBJ_EVENT_GFX_RIVAL_MAY_FIELD_MOVE:
case OBJ_EVENT_GFX_MAY_UNDERWATER:
case OBJ_EVENT_GFX_MAY_FISHING:
case OBJ_EVENT_GFX_MAY_WATERING:
return FEMALE;
default:
return MALE;
}
}
u8 GetPlayerAvatarGenderByGraphicsId(u8 gfxId)
{
switch (gfxId)
{
case OBJ_EVENT_GFX_MAY_NORMAL:
case OBJ_EVENT_GFX_MAY_MACH_BIKE:
case OBJ_EVENT_GFX_MAY_ACRO_BIKE:
case OBJ_EVENT_GFX_MAY_SURFING:
case OBJ_EVENT_GFX_MAY_FIELD_MOVE:
case OBJ_EVENT_GFX_MAY_UNDERWATER:
case OBJ_EVENT_GFX_MAY_FISHING:
case OBJ_EVENT_GFX_MAY_WATERING:
return FEMALE;
default:
return MALE;
}
}
bool8 PartyHasMonWithSurf(void)
{
u8 i;
if (!TestPlayerAvatarFlags(PLAYER_AVATAR_FLAG_SURFING))
{
2018-11-18 20:00:36 +01:00
for (i = 0; i < PARTY_SIZE; i++)
{
if (GetMonData(&gPlayerParty[i], MON_DATA_SPECIES) == SPECIES_NONE)
break;
2018-12-24 21:59:05 +01:00
if (MonKnowsMove(&gPlayerParty[i], MOVE_SURF))
return TRUE;
}
}
return FALSE;
}
bool8 IsPlayerSurfingNorth(void)
{
if (GetPlayerMovementDirection() == DIR_NORTH && TestPlayerAvatarFlags(PLAYER_AVATAR_FLAG_SURFING))
return TRUE;
else
return FALSE;
}
bool8 IsPlayerFacingSurfableFishableWater(void)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
s16 x = playerObjEvent->currentCoords.x;
s16 y = playerObjEvent->currentCoords.y;
2019-11-21 00:14:40 -05:00
MoveCoords(playerObjEvent->facingDirection, &x, &y);
if (GetCollisionAtCoords(playerObjEvent, x, y, playerObjEvent->facingDirection) == COLLISION_ELEVATION_MISMATCH
2019-10-06 15:40:10 -05:00
&& PlayerGetZCoord() == 3
&& MetatileBehavior_IsSurfableFishableWater(MapGridGetMetatileBehaviorAt(x, y)))
return TRUE;
else
return FALSE;
}
void ClearPlayerAvatarInfo(void)
{
memset(&gPlayerAvatar, 0, sizeof(struct PlayerAvatar));
}
void SetPlayerAvatarStateMask(u8 flags)
{
gPlayerAvatar.flags &= (PLAYER_AVATAR_FLAG_DASH | PLAYER_AVATAR_FLAG_6 | PLAYER_AVATAR_FLAG_5);
gPlayerAvatar.flags |= flags;
}
2019-05-25 20:26:29 +01:00
static u8 GetPlayerAvatarStateTransitionByGraphicsId(u8 graphicsId, u8 gender)
{
u8 i;
2020-04-01 02:54:26 -04:00
for (i = 0; i < ARRAY_COUNT(sPlayerAvatarGfxToStateFlag[0]); i++)
{
2020-04-01 02:54:26 -04:00
if (sPlayerAvatarGfxToStateFlag[gender][i][0] == graphicsId)
return sPlayerAvatarGfxToStateFlag[gender][i][1];
}
2020-04-01 02:54:26 -04:00
return PLAYER_AVATAR_FLAG_ON_FOOT;
}
u8 GetPlayerAvatarGraphicsIdByCurrentState(void)
{
u8 i;
2020-04-01 02:54:26 -04:00
u8 flags = gPlayerAvatar.flags;
2020-04-01 02:54:26 -04:00
for (i = 0; i < ARRAY_COUNT(sPlayerAvatarGfxToStateFlag[0]); i++)
{
2020-04-01 02:54:26 -04:00
if (sPlayerAvatarGfxToStateFlag[gPlayerAvatar.gender][i][1] & flags)
return sPlayerAvatarGfxToStateFlag[gPlayerAvatar.gender][i][0];
}
return 0;
}
2020-04-01 02:54:26 -04:00
void SetPlayerAvatarExtraStateTransition(u8 graphicsId, u8 transitionFlag)
{
2020-04-01 02:54:26 -04:00
u8 stateFlag = GetPlayerAvatarStateTransitionByGraphicsId(graphicsId, gPlayerAvatar.gender);
2020-04-01 02:54:26 -04:00
gPlayerAvatar.transitionFlags |= stateFlag | transitionFlag;
DoPlayerAvatarTransition();
}
void InitPlayerAvatar(s16 x, s16 y, u8 direction, u8 gender)
{
2019-11-21 00:14:40 -05:00
struct ObjectEventTemplate playerObjEventTemplate;
u8 objectEventId;
struct ObjectEvent *objectEvent;
2019-11-21 00:14:40 -05:00
playerObjEventTemplate.localId = OBJ_EVENT_ID_PLAYER;
playerObjEventTemplate.graphicsId = GetPlayerAvatarGraphicsIdByStateIdAndGender(PLAYER_AVATAR_STATE_NORMAL, gender);
playerObjEventTemplate.x = x - 7;
playerObjEventTemplate.y = y - 7;
playerObjEventTemplate.elevation = 0;
playerObjEventTemplate.movementType = MOVEMENT_TYPE_PLAYER;
playerObjEventTemplate.movementRangeX = 0;
playerObjEventTemplate.movementRangeY = 0;
playerObjEventTemplate.trainerType = 0;
playerObjEventTemplate.trainerRange_berryTreeId = 0;
playerObjEventTemplate.script = NULL;
playerObjEventTemplate.flagId = 0;
objectEventId = SpawnSpecialObjectEvent(&playerObjEventTemplate);
objectEvent = &gObjectEvents[objectEventId];
2020-04-01 02:54:26 -04:00
objectEvent->isPlayer = TRUE;
objectEvent->warpArrowSpriteId = CreateWarpArrowSprite();
ObjectEventTurn(objectEvent, direction);
ClearPlayerAvatarInfo();
gPlayerAvatar.runningState = NOT_MOVING;
gPlayerAvatar.tileTransitionState = T_NOT_MOVING;
gPlayerAvatar.objectEventId = objectEventId;
gPlayerAvatar.spriteId = objectEvent->spriteId;
gPlayerAvatar.gender = gender;
SetPlayerAvatarStateMask(PLAYER_AVATAR_FLAG_5 | PLAYER_AVATAR_FLAG_ON_FOOT);
}
2019-12-17 03:24:44 -05:00
void SetPlayerInvisibility(bool8 invisible)
{
2019-12-21 05:19:27 -05:00
gObjectEvents[gPlayerAvatar.objectEventId].invisible = invisible;
if (TestPlayerAvatarFlags(PLAYER_AVATAR_FLAG_SURFING))
2019-12-21 05:19:27 -05:00
gSprites[gObjectEvents[gPlayerAvatar.objectEventId].fieldEffectSpriteId].invisible = invisible;
}
void sub_808C114(void)
{
ObjectEventSetGraphicsId(&gObjectEvents[gPlayerAvatar.objectEventId], GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_FIELD_MOVE));
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], 0);
}
void sub_808C15C(u8 direction)
{
ObjectEventSetGraphicsId(&gObjectEvents[gPlayerAvatar.objectEventId], GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_FISHING));
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], GetFishingDirectionAnimNum(direction));
}
void sub_808C1B4(u8 direction)
{
ObjectEventSetGraphicsId(&gObjectEvents[gPlayerAvatar.objectEventId], GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_ACRO_BIKE));
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], GetAcroWheelieDirectionAnimNum(direction));
SeekSpriteAnim(&gSprites[gPlayerAvatar.spriteId], 1);
}
void sub_808C228(u8 direction)
{
ObjectEventSetGraphicsId(&gObjectEvents[gPlayerAvatar.objectEventId], GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_WATERING));
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], GetFaceDirectionAnimNum(direction));
}
static void sub_808C280(struct ObjectEvent *objectEvent)
{
s16 x;
s16 y;
u8 direction;
u8 metatileBehavior = objectEvent->currentMetatileBehavior;
for (x = 0, direction = DIR_SOUTH; x < 4; x++, direction++)
{
if (sArrowWarpMetatileBehaviorChecks2[x](metatileBehavior) && direction == objectEvent->movementDirection)
{
x = objectEvent->currentCoords.x;
y = objectEvent->currentCoords.y;
MoveCoords(direction, &x, &y);
ShowWarpArrowSprite(objectEvent->warpArrowSpriteId, direction, x, y);
return;
}
}
SetSpriteInvisible(objectEvent->warpArrowSpriteId);
}
/* Strength */
static void StartStrengthAnim(u8 a, u8 b)
{
2020-04-01 02:54:26 -04:00
u8 taskId = CreateTask(Task_PushBoulder, 0xFF);
gTasks[taskId].data[1] = a;
gTasks[taskId].data[2] = b;
2020-04-01 02:54:26 -04:00
Task_PushBoulder(taskId);
}
2020-04-01 02:54:26 -04:00
static void Task_PushBoulder(u8 taskId)
{
2020-04-01 02:54:26 -04:00
while (sPushBoulderFuncs[gTasks[taskId].data[0]](&gTasks[taskId],
&gObjectEvents[gPlayerAvatar.objectEventId],
&gObjectEvents[gTasks[taskId].data[1]]))
;
}
2020-04-01 02:54:26 -04:00
static bool8 PushBoulder_Start(struct Task *task, struct ObjectEvent *playerObject, struct ObjectEvent *strengthObject)
{
ScriptContext2_Enable();
gPlayerAvatar.preventStep = TRUE;
task->data[0]++;
return FALSE;
}
2020-04-01 02:54:26 -04:00
static bool8 PushBoulder_Move(struct Task *task, struct ObjectEvent *playerObject, struct ObjectEvent *strengthObject)
{
if (ObjectEventIsHeldMovementActive(playerObject))
{
ObjectEventClearHeldMovementIfFinished(playerObject);
}
if (ObjectEventIsHeldMovementActive(strengthObject))
{
ObjectEventClearHeldMovementIfFinished(strengthObject);
}
if (!ObjectEventIsMovementOverridden(playerObject)
&& !ObjectEventIsMovementOverridden(strengthObject))
{
ObjectEventClearHeldMovementIfFinished(playerObject);
ObjectEventClearHeldMovementIfFinished(strengthObject);
ObjectEventSetHeldMovement(playerObject, GetWalkInPlaceNormalMovementAction((u8)task->data[2]));
ObjectEventSetHeldMovement(strengthObject, GetWalkSlowMovementAction((u8)task->data[2]));
gFieldEffectArguments[0] = strengthObject->currentCoords.x;
gFieldEffectArguments[1] = strengthObject->currentCoords.y;
gFieldEffectArguments[2] = strengthObject->previousElevation;
gFieldEffectArguments[3] = gSprites[strengthObject->spriteId].oam.priority;
FieldEffectStart(FLDEFF_DUST);
PlaySE(SE_W070);
task->data[0]++;
}
return FALSE;
}
2020-04-01 02:54:26 -04:00
static bool8 PushBoulder_End(struct Task *task, struct ObjectEvent *playerObject, struct ObjectEvent *strengthObject)
{
if (ObjectEventCheckHeldMovementStatus(playerObject)
&& ObjectEventCheckHeldMovementStatus(strengthObject))
{
ObjectEventClearHeldMovementIfFinished(playerObject);
ObjectEventClearHeldMovementIfFinished(strengthObject);
gPlayerAvatar.preventStep = FALSE;
ScriptContext2_Disable();
2020-04-01 02:54:26 -04:00
DestroyTask(FindTaskIdByFunc(Task_PushBoulder));
}
return FALSE;
}
/* Some field effect */
static void DoPlayerMatJump(void)
{
DoPlayerAvatarSecretBaseMatJump(CreateTask(DoPlayerAvatarSecretBaseMatJump, 0xFF));
}
static void DoPlayerAvatarSecretBaseMatJump(u8 taskId)
{
while (sPlayerAvatarSecretBaseMatJump[gTasks[taskId].data[0]](&gTasks[taskId], &gObjectEvents[gPlayerAvatar.objectEventId]))
;
}
// because data[0] is used to call this, it can be inferred that there may have been multiple mat jump functions at one point, so the name for these groups of functions is appropriate in assuming the sole use of mat jump.
static u8 PlayerAvatar_DoSecretBaseMatJump(struct Task *task, struct ObjectEvent *objectEvent)
{
gPlayerAvatar.preventStep = TRUE;
if (ObjectEventClearHeldMovementIfFinished(objectEvent))
{
PlaySE(SE_DANSA);
ObjectEventSetHeldMovement(objectEvent, GetJumpInPlaceMovementAction(objectEvent->facingDirection));
task->data[1]++;
if (task->data[1] > 1)
{
gPlayerAvatar.preventStep = FALSE;
2020-04-01 02:54:26 -04:00
gPlayerAvatar.transitionFlags |= PLAYER_AVATAR_FLAG_5;
DestroyTask(FindTaskIdByFunc(DoPlayerAvatarSecretBaseMatJump));
}
}
return FALSE;
}
/* Some field effect */
static void DoPlayerMatSpin(void)
{
u8 taskId = CreateTask(PlayerAvatar_DoSecretBaseMatSpin, 0xFF);
PlayerAvatar_DoSecretBaseMatSpin(taskId);
}
static void PlayerAvatar_DoSecretBaseMatSpin(u8 taskId)
{
while (sPlayerAvatarSecretBaseMatSpin[gTasks[taskId].data[0]](&gTasks[taskId], &gObjectEvents[gPlayerAvatar.objectEventId]))
;
}
static bool8 PlayerAvatar_SecretBaseMatSpinStep0(struct Task *task, struct ObjectEvent *objectEvent)
{
task->data[0]++;
task->data[1] = objectEvent->movementDirection;
gPlayerAvatar.preventStep = TRUE;
ScriptContext2_Enable();
PlaySE(SE_TK_WARPIN);
return TRUE;
}
static bool8 PlayerAvatar_SecretBaseMatSpinStep1(struct Task *task, struct ObjectEvent *objectEvent)
{
u8 directions[] = {DIR_WEST, DIR_EAST, DIR_NORTH, DIR_SOUTH};
if (ObjectEventClearHeldMovementIfFinished(objectEvent))
{
u8 direction;
ObjectEventSetHeldMovement(objectEvent, GetFaceDirectionMovementAction(direction = directions[objectEvent->movementDirection - 1]));
if (direction == (u8)task->data[1])
task->data[2]++;
task->data[0]++;
if (task->data[2] > 3 && direction == GetOppositeDirection(task->data[1]))
task->data[0]++;
}
return FALSE;
}
static bool8 PlayerAvatar_SecretBaseMatSpinStep2(struct Task *task, struct ObjectEvent *objectEvent)
{
const u8 actions[] = {
MOVEMENT_ACTION_DELAY_1,
MOVEMENT_ACTION_DELAY_1,
MOVEMENT_ACTION_DELAY_2,
MOVEMENT_ACTION_DELAY_4,
MOVEMENT_ACTION_DELAY_8,
};
if (ObjectEventClearHeldMovementIfFinished(objectEvent))
{
ObjectEventSetHeldMovement(objectEvent, actions[task->data[2]]);
task->data[0] = 1;
}
return FALSE;
}
static bool8 PlayerAvatar_SecretBaseMatSpinStep3(struct Task *task, struct ObjectEvent *objectEvent)
{
if (ObjectEventClearHeldMovementIfFinished(objectEvent))
{
ObjectEventSetHeldMovement(objectEvent, GetWalkSlowMovementAction(GetOppositeDirection(task->data[1])));
ScriptContext2_Disable();
gPlayerAvatar.preventStep = FALSE;
DestroyTask(FindTaskIdByFunc(PlayerAvatar_DoSecretBaseMatSpin));
}
return FALSE;
}
2019-10-06 15:40:10 -05:00
static void CreateStopSurfingTask(u8 direction)
{
u8 taskId;
ScriptContext2_Enable();
Overworld_ClearSavedMusic();
Overworld_ChangeMusicToDefault();
gPlayerAvatar.flags &= ~PLAYER_AVATAR_FLAG_SURFING;
gPlayerAvatar.flags |= PLAYER_AVATAR_FLAG_ON_FOOT;
gPlayerAvatar.preventStep = TRUE;
2019-10-06 15:40:10 -05:00
taskId = CreateTask(Task_StopSurfingInit, 0xFF);
gTasks[taskId].data[0] = direction;
Task_StopSurfingInit(taskId);
}
2019-10-06 15:40:10 -05:00
static void Task_StopSurfingInit(u8 taskId)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
if (ObjectEventIsMovementOverridden(playerObjEvent))
{
2019-11-21 00:14:40 -05:00
if (!ObjectEventClearHeldMovementIfFinished(playerObjEvent))
return;
}
2019-11-21 00:14:40 -05:00
sub_81555AC(playerObjEvent->fieldEffectSpriteId, 2);
ObjectEventSetHeldMovement(playerObjEvent, GetJumpSpecialMovementAction((u8)gTasks[taskId].data[0]));
2019-10-06 15:40:10 -05:00
gTasks[taskId].func = Task_WaitStopSurfing;
}
2019-10-06 15:40:10 -05:00
static void Task_WaitStopSurfing(u8 taskId)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
if (ObjectEventClearHeldMovementIfFinished(playerObjEvent))
{
2019-11-21 00:14:40 -05:00
ObjectEventSetGraphicsId(playerObjEvent, GetPlayerAvatarGraphicsIdByStateId(PLAYER_AVATAR_STATE_NORMAL));
ObjectEventSetHeldMovement(playerObjEvent, GetFaceDirectionMovementAction(playerObjEvent->facingDirection));
gPlayerAvatar.preventStep = FALSE;
ScriptContext2_Disable();
2019-11-21 00:14:40 -05:00
DestroySprite(&gSprites[playerObjEvent->fieldEffectSpriteId]);
DestroyTask(taskId);
}
}
static bool8 (*const sFishingStateFuncs[])(struct Task *) =
{
Fishing1,
Fishing2,
Fishing3,
Fishing4,
Fishing5,
Fishing6,
Fishing7,
Fishing8,
Fishing9,
Fishing10,
Fishing11,
Fishing12,
Fishing13,
Fishing14,
Fishing15,
Fishing16,
};
#define tStep data[0]
#define tFrameCounter data[1]
#define tNumDots data[2]
#define tDotsRequired data[3]
#define tRoundsPlayed data[12]
#define tMinRoundsRequired data[13]
#define tPlayerGfxId data[14]
#define tFishingRod data[15]
#define FISHING_START_ROUND 3
#define FISHING_GOT_BITE 6
#define FISHING_ON_HOOK 9
#define FISHING_NO_BITE 11
#define FISHING_GOT_AWAY 12
#define FISHING_SHOW_RESULT 13
void StartFishing(u8 rod)
{
u8 taskId = CreateTask(Task_Fishing, 0xFF);
gTasks[taskId].tFishingRod = rod;
Task_Fishing(taskId);
}
static void Task_Fishing(u8 taskId)
{
while (sFishingStateFuncs[gTasks[taskId].tStep](&gTasks[taskId]))
;
}
static bool8 Fishing1(struct Task *task)
{
ScriptContext2_Enable();
gPlayerAvatar.preventStep = TRUE;
task->tStep++;
return FALSE;
}
static bool8 Fishing2(struct Task *task)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent;
const s16 arr1[] = {1, 1, 1};
const s16 arr2[] = {1, 3, 6};
task->tRoundsPlayed = 0;
task->tMinRoundsRequired = arr1[task->tFishingRod] + (Random() % arr2[task->tFishingRod]);
task->tPlayerGfxId = gObjectEvents[gPlayerAvatar.objectEventId].graphicsId;
2019-11-21 00:14:40 -05:00
playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
ObjectEventClearHeldMovementIfActive(playerObjEvent);
playerObjEvent->enableAnim = 1;
sub_808C15C(playerObjEvent->facingDirection);
task->tStep++;
return FALSE;
}
static bool8 Fishing3(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
2019-02-01 11:50:45 +01:00
// Wait one second
task->tFrameCounter++;
if (task->tFrameCounter >= 60)
task->tStep++;
return FALSE;
}
static bool8 Fishing4(struct Task *task)
{
u32 randVal;
sub_819786C(0, TRUE);
task->tStep++;
task->tFrameCounter = 0;
task->tNumDots = 0;
randVal = Random();
randVal %= 10;
task->tDotsRequired = randVal + 1;
if (task->tRoundsPlayed == 0)
task->tDotsRequired = randVal + 4;
if (task->tDotsRequired >= 10)
task->tDotsRequired = 10;
return TRUE;
}
// Play a round of the dot game
static bool8 Fishing5(struct Task *task)
{
const u8 dot[] = _("·");
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
task->tFrameCounter++;
if (gMain.newKeys & A_BUTTON)
{
task->tStep = FISHING_NO_BITE;
if (task->tRoundsPlayed != 0)
task->tStep = FISHING_GOT_AWAY;
return TRUE;
}
else
{
if (task->tFrameCounter >= 20)
{
task->tFrameCounter = 0;
if (task->tNumDots >= task->tDotsRequired)
{
task->tStep++;
if (task->tRoundsPlayed != 0)
task->tStep++;
task->tRoundsPlayed++;
}
else
{
AddTextPrinterParameterized(0, 1, dot, task->tNumDots * 8, 1, 0, NULL);
task->tNumDots++;
}
}
return FALSE;
}
}
// Determine if fish bites
static bool8 Fishing6(struct Task *task)
{
bool8 bite;
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
task->tStep++;
bite = FALSE;
if (!DoesCurrentMapHaveFishingMons())
{
task->tStep = FISHING_NO_BITE;
}
else
{
2018-12-15 23:58:47 +01:00
if (!GetMonData(&gPlayerParty[0], MON_DATA_SANITY_IS_EGG))
{
u8 ability = GetMonAbility(&gPlayerParty[0]);
if (ability == ABILITY_SUCTION_CUPS || ability == ABILITY_STICKY_HOLD)
{
if (Random() % 100 > 14)
{
bite = TRUE;
}
}
}
if (!bite)
{
if (Random() & 1)
{
task->tStep = FISHING_NO_BITE;
}
else
{
bite = TRUE;
}
}
if (bite == TRUE)
{
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], GetFishingBiteDirectionAnimNum(GetPlayerFacingDirection()));
}
}
return TRUE;
}
// Oh! A Bite!
static bool8 Fishing7(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
AddTextPrinterParameterized(0, 1, gText_OhABite, 0, 17, 0, NULL);
task->tStep++;
task->tFrameCounter = 0;
return FALSE;
}
// We have a bite. Now, wait for the player to press A, or the timer to expire.
static bool8 Fishing8(struct Task *task)
{
const s16 reelTimeouts[3] = {36, 33, 30};
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
task->tFrameCounter++;
if (task->tFrameCounter >= reelTimeouts[task->tFishingRod])
task->tStep = FISHING_GOT_AWAY;
else if (gMain.newKeys & A_BUTTON)
task->tStep++;
return FALSE;
}
// Determine if we're going to play the dot game again
static bool8 Fishing9(struct Task *task)
{
const s16 arr[][2] =
{
{0, 0},
{40, 10},
{70, 30}
};
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
task->tStep++;
if (task->tRoundsPlayed < task->tMinRoundsRequired)
{
task->tStep = FISHING_START_ROUND;
}
else if (task->tRoundsPlayed < 2)
{
// probability of having to play another round
s16 probability = Random() % 100;
if (arr[task->tFishingRod][task->tRoundsPlayed] > probability)
task->tStep = FISHING_START_ROUND;
}
return FALSE;
}
static bool8 Fishing10(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
FillWindowPixelBuffer(0, PIXEL_FILL(1));
AddTextPrinterParameterized2(0, 1, gText_PokemonOnHook, 1, 0, 2, 1, 3);
task->tStep++;
task->tFrameCounter = 0;
return FALSE;
}
static bool8 Fishing11(struct Task *task)
{
if (task->tFrameCounter == 0)
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
RunTextPrinters();
if (task->tFrameCounter == 0)
{
if (!IsTextPrinterActive(0))
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
ObjectEventSetGraphicsId(playerObjEvent, task->tPlayerGfxId);
ObjectEventTurn(playerObjEvent, playerObjEvent->movementDirection);
if (gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_SURFING)
sub_8155604(gObjectEvents[gPlayerAvatar.objectEventId].fieldEffectSpriteId, 0, 0);
gSprites[gPlayerAvatar.spriteId].pos2.x = 0;
gSprites[gPlayerAvatar.spriteId].pos2.y = 0;
ClearDialogWindowAndFrame(0, TRUE);
task->tFrameCounter++;
return FALSE;
}
}
if (task->tFrameCounter != 0)
{
gPlayerAvatar.preventStep = FALSE;
ScriptContext2_Disable();
FishingWildEncounter(task->tFishingRod);
sub_80ED950(1);
DestroyTask(FindTaskIdByFunc(Task_Fishing));
}
return FALSE;
}
// Not even a nibble
static bool8 Fishing12(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], GetFishingNoCatchDirectionAnimNum(GetPlayerFacingDirection()));
FillWindowPixelBuffer(0, PIXEL_FILL(1));
AddTextPrinterParameterized2(0, 1, gText_NotEvenANibble, 1, 0, 2, 1, 3);
task->tStep = FISHING_SHOW_RESULT;
return TRUE;
}
// It got away
static bool8 Fishing13(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
StartSpriteAnim(&gSprites[gPlayerAvatar.spriteId], GetFishingNoCatchDirectionAnimNum(GetPlayerFacingDirection()));
FillWindowPixelBuffer(0, PIXEL_FILL(1));
AddTextPrinterParameterized2(0, 1, gText_ItGotAway, 1, 0, 2, 1, 3);
task->tStep++;
return TRUE;
}
// Wait one second
static bool8 Fishing14(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
task->tStep++;
return FALSE;
}
static bool8 Fishing15(struct Task *task)
{
2018-11-03 20:20:28 +00:00
AlignFishingAnimationFrames();
if (gSprites[gPlayerAvatar.spriteId].animEnded)
{
2019-11-21 00:14:40 -05:00
struct ObjectEvent *playerObjEvent = &gObjectEvents[gPlayerAvatar.objectEventId];
2019-11-21 00:14:40 -05:00
ObjectEventSetGraphicsId(playerObjEvent, task->tPlayerGfxId);
ObjectEventTurn(playerObjEvent, playerObjEvent->movementDirection);
if (gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_SURFING)
sub_8155604(gObjectEvents[gPlayerAvatar.objectEventId].fieldEffectSpriteId, 0, 0);
gSprites[gPlayerAvatar.spriteId].pos2.x = 0;
gSprites[gPlayerAvatar.spriteId].pos2.y = 0;
task->tStep++;
}
return FALSE;
}
static bool8 Fishing16(struct Task *task)
{
RunTextPrinters();
if (!IsTextPrinterActive(0))
{
gPlayerAvatar.preventStep = FALSE;
ScriptContext2_Disable();
UnfreezeObjectEvents();
ClearDialogWindowAndFrame(0, TRUE);
sub_80ED950(0);
DestroyTask(FindTaskIdByFunc(Task_Fishing));
}
return FALSE;
}
#undef tStep
#undef tFrameCounter
#undef tFishingRod
2018-11-03 20:20:28 +00:00
static void AlignFishingAnimationFrames(void)
{
struct Sprite *playerSprite = &gSprites[gPlayerAvatar.spriteId];
u8 animCmdIndex;
u8 animType;
AnimateSprite(playerSprite);
playerSprite->pos2.x = 0;
playerSprite->pos2.y = 0;
animCmdIndex = playerSprite->animCmdIndex;
if (playerSprite->anims[playerSprite->animNum][animCmdIndex].type == -1)
{
animCmdIndex--;
}
else
{
playerSprite->animDelayCounter++;
if (playerSprite->anims[playerSprite->animNum][animCmdIndex].type == -1)
animCmdIndex--;
}
animType = playerSprite->anims[playerSprite->animNum][animCmdIndex].type;
if (animType == 1 || animType == 2 || animType == 3)
{
playerSprite->pos2.x = 8;
if (GetPlayerFacingDirection() == 3)
playerSprite->pos2.x = -8;
}
if (animType == 5)
playerSprite->pos2.y = -8;
if (animType == 10 || animType == 11)
playerSprite->pos2.y = 8;
if (gPlayerAvatar.flags & PLAYER_AVATAR_FLAG_SURFING)
sub_8155604(gObjectEvents[gPlayerAvatar.objectEventId].fieldEffectSpriteId, 1, playerSprite->pos2.y);
}
void sub_808D074(u8 a0)
{
gUnknown_0203734C = a0;
}
static u8 sub_808D080(void)
{
if (gUnknown_0203734C == 0)
{
return 1;
}
return gUnknown_0203734C;
}
static void sub_808D094(u8 taskId)
{
struct ObjectEvent *object = &gObjectEvents[gPlayerAvatar.objectEventId];
struct Sprite *sprite = &gSprites[object->spriteId];
s16 *data = gTasks[taskId].data;
switch (data[0])
{
case 0:
if (!ObjectEventClearHeldMovementIfFinished(object))
{
return;
}
sub_808D074(object->facingDirection);
data[1] = 0;
data[2] = 1;
data[3] = (u16)(sprite->pos1.y + sprite->pos2.y) * 16;
sprite->pos2.y = 0;
CameraObjectReset2();
object->fixedPriority = TRUE;
sprite->oam.priority = 0;
sprite->subpriority = 0;
sprite->subspriteMode = SUBSPRITES_OFF;
data[0]++;
case 1:
sub_808D38C(object, &data[1]);
data[3] -= data[2];
data[2] += 3;
sprite->pos1.y = data[3] >> 4;
if (sprite->pos1.y + (s16)gTotalCameraPixelOffsetY < -32)
{
data[0]++;
}
break;
case 2:
DestroyTask(taskId);
break;
}
}
static void sub_808D1FC(u8 taskId);
void sub_808D194(void)
{
sub_808D1FC(CreateTask(sub_808D1FC, 0));
}
2019-03-02 02:44:02 -05:00
bool32 sub_808D1B4(void)
{
return FuncIsActiveTask(sub_808D1FC);
}
void sub_808D1C8(void)
{
sub_808D094(CreateTask(sub_808D094, 0));
}
2019-03-02 02:44:02 -05:00
bool32 sub_808D1E8(void)
{
return FuncIsActiveTask(sub_808D094);
}
static const u8 gUnknown_084975BC[] = {DIR_SOUTH, DIR_WEST, DIR_EAST, DIR_NORTH, DIR_SOUTH};
static void sub_808D1FC(u8 taskId)
{
struct ObjectEvent *object = &gObjectEvents[gPlayerAvatar.objectEventId];
struct Sprite *sprite = &gSprites[object->spriteId];
s16 *data = gTasks[taskId].data;
switch (data[0])
{
case 0:
data[5] = sub_808D080();
ObjectEventForceSetHeldMovement(object, GetFaceDirectionMovementAction(gUnknown_084975BC[data[5]]));
data[1] = 0;
data[2] = 116;
data[4] = sprite->pos1.y;
data[6] = sprite->oam.priority;
data[7] = sprite->subpriority;
data[3] = -((u16)sprite->pos2.y + 32) * 16;
sprite->pos2.y = 0;
CameraObjectReset2();
object->fixedPriority = TRUE;
sprite->oam.priority = 1;
sprite->subpriority = 0;
sprite->subspriteMode = SUBSPRITES_OFF;
data[0]++;
case 1:
sub_808D38C(object, &data[1]);
data[3] += data[2];
data[2] -= 3;
if (data[2] < 4)
{
data[2] = 4;
}
sprite->pos1.y = data[3] >> 4;
if (sprite->pos1.y >= data[4])
{
sprite->pos1.y = data[4];
data[8] = 0;
data[0]++;
}
break;
case 2:
sub_808D38C(object, &data[1]);
data[8]++;
if (data[8] > 8)
{
data[0]++;
}
break;
case 3:
if (data[5] == sub_808D38C(object, &data[1]))
{
object->fixedPriority = 0;
sprite->oam.priority = data[6];
sprite->subpriority = data[7];
CameraObjectReset1();
DestroyTask(taskId);
}
break;
}
}
static u8 sub_808D38C(struct ObjectEvent *object, s16 *a1)
{
if (*a1 < 8 && ++(*a1) < 8)
{
return object->facingDirection;
}
if (!ObjectEventCheckHeldMovementStatus(object))
{
return object->facingDirection;
}
ObjectEventForceSetHeldMovement(object, GetFaceDirectionMovementAction(gUnknown_084975BC[object->facingDirection]));
*a1 = 0;
return gUnknown_084975BC[object->facingDirection];
}