pokeemerald/src/union_room_player_avatar.c

625 lines
18 KiB
C
Raw Normal View History

2019-04-01 18:05:58 +02:00
#include "global.h"
#include "event_data.h"
#include "event_object_movement.h"
#include "field_player_avatar.h"
#include "fieldmap.h"
2019-05-25 21:26:29 +02:00
#include "script.h"
2019-04-01 18:05:58 +02:00
#include "task.h"
2019-05-25 21:26:29 +02:00
#include "union_room.h"
#include "constants/event_objects.h"
2019-11-21 20:03:35 +01:00
#include "constants/event_object_movement.h"
2019-04-01 18:05:58 +02:00
2020-06-01 16:23:12 +02:00
#define UR_SPRITE_START_ID (MAX_SPRITES - MAX_UNION_ROOM_PLAYERS)
#define UR_PLAYER_SPRITE_ID(playerIdx, facingDir)(5 * playerIdx + facingDir)
2019-04-01 18:05:58 +02:00
2020-06-01 16:23:12 +02:00
static EWRAM_DATA struct UnionRoomObject * sUnionObjWork = NULL;
static EWRAM_DATA u32 sUnionObjRefreshTimer = 0;
2019-04-01 18:05:58 +02:00
2020-06-01 16:23:12 +02:00
static u8 CreateTask_AnimateUnionRoomPlayers(void);
static u32 IsUnionRoomPlayerInvisible(u32, u32);
static void SetUnionRoomObjectFacingDirection(s32, s32, u8);
static const u8 sUnionRoomObjGfxIds[GENDER_COUNT][MAX_UNION_ROOM_PLAYERS + 2] = {
2020-05-30 10:09:21 +02:00
[MALE] = {
OBJ_EVENT_GFX_MAN_3,
OBJ_EVENT_GFX_BLACK_BELT,
OBJ_EVENT_GFX_CAMPER,
OBJ_EVENT_GFX_YOUNGSTER,
OBJ_EVENT_GFX_PSYCHIC_M,
OBJ_EVENT_GFX_BUG_CATCHER,
OBJ_EVENT_GFX_MAN_4,
OBJ_EVENT_GFX_MAN_5
},
[FEMALE] = {
OBJ_EVENT_GFX_WOMAN_5,
OBJ_EVENT_GFX_HEX_MANIAC,
OBJ_EVENT_GFX_PICNICKER,
OBJ_EVENT_GFX_LASS,
OBJ_EVENT_GFX_LASS,
OBJ_EVENT_GFX_GIRL_3,
OBJ_EVENT_GFX_WOMAN_2,
OBJ_EVENT_GFX_BEAUTY
}
2019-04-01 18:05:58 +02:00
};
2020-06-01 16:23:12 +02:00
static const s16 sUnionRoomPlayerCoords[MAX_UNION_ROOM_PLAYERS][2] = {
2020-05-30 10:09:21 +02:00
{ 4, 6},
{13, 8},
{10, 6},
{ 1, 8},
{13, 4},
{ 7, 4},
{ 1, 4},
{ 7, 8}
2019-04-01 18:05:58 +02:00
};
2020-05-30 10:09:21 +02:00
static const s8 sFacingDirectionOffsets[][2] = {
[DIR_NONE] = { 0, 0},
[DIR_SOUTH] = { 1, 0},
[DIR_NORTH] = { 0, -1},
[DIR_WEST] = {-1, 0},
[DIR_EAST] = { 0, 1}
2019-04-01 18:05:58 +02:00
};
2020-06-01 16:23:12 +02:00
static const u8 sOppositeFacingDirection[] = {
[DIR_NONE] = DIR_NONE,
[DIR_SOUTH] = DIR_NORTH,
[DIR_NORTH] = DIR_SOUTH,
[DIR_WEST] = DIR_EAST,
[DIR_EAST] = DIR_WEST
2019-04-01 18:05:58 +02:00
};
2020-06-01 16:23:12 +02:00
static const u8 sNextFacingDirection[] = {
[DIR_NONE] = DIR_SOUTH,
[DIR_SOUTH] = DIR_WEST,
[DIR_NORTH] = DIR_SOUTH,
[DIR_WEST] = DIR_EAST,
[DIR_EAST] = DIR_NORTH
2019-04-01 18:05:58 +02:00
};
2020-06-09 00:16:57 +02:00
// Local id 1 is the Nurse/Attendant, 2-9 are link players
2020-06-01 16:23:12 +02:00
static const u8 sUnionRoomLocalIds[] = { 9, 8, 7, 2, 6, 5, 4, 3 };
static const u16 sUnknown[] = {
0x2BF,
0x2C0,
0x2C1,
0x2C2,
0x2C3,
0x2C4,
0x2C5,
0x2C6
2019-04-01 18:05:58 +02:00
};
2020-06-01 16:23:12 +02:00
static const u8 sMovement_UnionPlayerExit[2] = {
2019-04-01 18:05:58 +02:00
MOVEMENT_ACTION_FLY_UP,
MOVEMENT_ACTION_STEP_END
};
2020-06-01 16:23:12 +02:00
static const u8 sMovement_UnionPlayerEnter[2] = {
2019-04-01 18:05:58 +02:00
MOVEMENT_ACTION_FLY_DOWN,
MOVEMENT_ACTION_STEP_END
};
2020-06-09 00:16:57 +02:00
static bool32 IsPlayerStandingStill(void)
2019-04-01 18:05:58 +02:00
{
2020-06-09 00:16:57 +02:00
if (gPlayerAvatar.tileTransitionState == T_TILE_CENTER || gPlayerAvatar.tileTransitionState == T_NOT_MOVING)
2019-04-01 18:05:58 +02:00
return TRUE;
else
return FALSE;
}
2020-06-01 16:23:12 +02:00
static u8 GetUnionRoomPlayerGraphicsId(u32 gender, u32 id)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
return sUnionRoomObjGfxIds[gender][id % MAX_UNION_ROOM_PLAYERS];
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static void GetUnionRoomPlayerFacingCoords(u32 playerIdx, u32 direction, s32 * x, s32 * y)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
*x = sUnionRoomPlayerCoords[playerIdx][0] + sFacingDirectionOffsets[direction][0] + 7;
*y = sUnionRoomPlayerCoords[playerIdx][1] + sFacingDirectionOffsets[direction][1] + 7;
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static bool32 IsUnionRoomPlayerFacingTileAt(u32 playerIdx, u32 direction, s32 x, s32 y)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
if ((sUnionRoomPlayerCoords[playerIdx][0] + sFacingDirectionOffsets[direction][0] + 7 == x)
&& (sUnionRoomPlayerCoords[playerIdx][1] + sFacingDirectionOffsets[direction][1] + 7 == y))
2019-04-01 18:05:58 +02:00
return TRUE;
2020-06-01 16:23:12 +02:00
else
return FALSE;
2019-04-01 18:05:58 +02:00
}
2019-04-01 20:20:34 +02:00
static bool32 IsUnionRoomPlayerHidden(u32 player_idx)
2019-04-01 18:05:58 +02:00
{
return FlagGet(FLAG_HIDE_UNION_ROOM_PLAYER_1 + player_idx);
}
2019-04-01 20:20:34 +02:00
static void HideUnionRoomPlayer(u32 player_idx)
2019-04-01 18:05:58 +02:00
{
FlagSet(FLAG_HIDE_UNION_ROOM_PLAYER_1 + player_idx);
}
2019-04-01 20:20:34 +02:00
static void ShowUnionRoomPlayer(u32 player_idx)
2019-04-01 18:05:58 +02:00
{
FlagClear(FLAG_HIDE_UNION_ROOM_PLAYER_1 + player_idx);
}
2019-04-01 20:20:34 +02:00
static void SetUnionRoomPlayerGfx(u32 playerIdx, u32 gfxId)
2019-04-01 18:05:58 +02:00
{
VarSet(VAR_OBJ_GFX_ID_0 + playerIdx, gfxId);
}
static void CreateUnionRoomPlayerObjectEvent(u32 playerIdx)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
TrySpawnObjectEvent(sUnionRoomLocalIds[playerIdx], gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup);
2019-04-01 18:05:58 +02:00
}
static void RemoveUnionRoomPlayerObjectEvent(u32 playerIdx)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
RemoveObjectEventByLocalIdAndMap(sUnionRoomLocalIds[playerIdx], gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup);
2019-04-01 18:05:58 +02:00
}
2019-04-01 20:20:34 +02:00
static bool32 SetUnionRoomPlayerEnterExitMovement(u32 playerIdx, const u8 * movement)
2019-04-01 18:05:58 +02:00
{
u8 objectId;
struct ObjectEvent * object;
2020-06-01 16:23:12 +02:00
if (TryGetObjectEventIdByLocalIdAndMap(sUnionRoomLocalIds[playerIdx], gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, &objectId))
2019-04-01 18:05:58 +02:00
{
return FALSE;
}
object = &gObjectEvents[objectId];
if (ObjectEventIsMovementOverridden(object))
2019-04-01 18:05:58 +02:00
{
return FALSE;
}
if (ObjectEventSetHeldMovement(object, *movement))
2019-04-01 18:05:58 +02:00
{
return FALSE;
}
return TRUE;
}
2020-06-01 16:23:12 +02:00
static bool32 TryReleaseUnionRoomPlayerObjectEvent(u32 playerIdx)
2019-04-01 18:05:58 +02:00
{
u8 objectId;
struct ObjectEvent * object;
2020-06-01 16:23:12 +02:00
if (TryGetObjectEventIdByLocalIdAndMap(sUnionRoomLocalIds[playerIdx], gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, &objectId))
2019-04-01 18:05:58 +02:00
{
return TRUE;
}
object = &gObjectEvents[objectId];
if (!ObjectEventClearHeldMovementIfFinished(object))
2019-04-01 18:05:58 +02:00
{
return FALSE;
}
if (!ScriptContext2_IsEnabled())
{
UnfreezeObjectEvent(object);
2019-04-01 18:05:58 +02:00
}
else
{
FreezeObjectEvent(object);
2019-04-01 18:05:58 +02:00
}
return TRUE;
}
2020-06-01 16:23:12 +02:00
u8 InitUnionRoomPlayerObjects(struct UnionRoomObject * players)
2019-04-01 18:05:58 +02:00
{
s32 i;
2020-06-01 16:23:12 +02:00
sUnionObjRefreshTimer = 0;
sUnionObjWork = players;
for (i = 0; i < MAX_UNION_ROOM_PLAYERS; i++)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
players[i].state = 0;
players[i].gfxId = 0;
players[i].animState = 0;
players[i].schedAnim = UNION_ROOM_SPAWN_NONE;
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
return CreateTask_AnimateUnionRoomPlayers();
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static bool32 AnimateUnionRoomPlayerDespawn(s8 * state, u32 playerIdx, struct UnionRoomObject * ptr)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
switch (*state)
2019-04-01 18:05:58 +02:00
{
case 0:
2020-06-01 16:23:12 +02:00
if (SetUnionRoomPlayerEnterExitMovement(playerIdx, sMovement_UnionPlayerExit) == TRUE)
2019-04-01 18:05:58 +02:00
{
HideUnionRoomPlayer(playerIdx);
2020-06-01 16:23:12 +02:00
(*state)++;
2019-04-01 18:05:58 +02:00
}
break;
case 1:
2020-06-01 16:23:12 +02:00
if (TryReleaseUnionRoomPlayerObjectEvent(playerIdx))
2019-04-01 18:05:58 +02:00
{
RemoveUnionRoomPlayerObjectEvent(playerIdx);
2019-04-01 18:05:58 +02:00
HideUnionRoomPlayer(playerIdx);
2020-06-01 16:23:12 +02:00
*state = 0;
2019-04-01 18:05:58 +02:00
return TRUE;
}
break;
}
return FALSE;
}
2020-06-01 16:23:12 +02:00
static bool32 AnimateUnionRoomPlayerSpawn(s8 * state, u32 playerIdx, struct UnionRoomObject * ptr)
2019-04-01 18:05:58 +02:00
{
s16 x, y;
2020-06-01 16:23:12 +02:00
switch (*state)
2019-04-01 18:05:58 +02:00
{
case 0:
2020-06-09 00:16:57 +02:00
if (!IsPlayerStandingStill())
2019-04-01 18:05:58 +02:00
{
break;
}
PlayerGetDestCoords(&x, &y);
2020-06-01 16:23:12 +02:00
if (IsUnionRoomPlayerFacingTileAt(playerIdx, 0, x, y) == TRUE)
2019-04-01 18:05:58 +02:00
{
break;
}
player_get_pos_including_state_based_drift(&x, &y);
2020-06-01 16:23:12 +02:00
if (IsUnionRoomPlayerFacingTileAt(playerIdx, 0, x, y) == TRUE)
2019-04-01 18:05:58 +02:00
{
break;
}
2020-06-01 16:23:12 +02:00
SetUnionRoomPlayerGfx(playerIdx, ptr->gfxId);
CreateUnionRoomPlayerObjectEvent(playerIdx);
2019-04-01 18:05:58 +02:00
ShowUnionRoomPlayer(playerIdx);
2020-06-01 16:23:12 +02:00
(*state)++;
2019-04-01 18:05:58 +02:00
// fallthrough
case 3: // incorrect?
2020-06-01 16:23:12 +02:00
if (SetUnionRoomPlayerEnterExitMovement(playerIdx, sMovement_UnionPlayerEnter) == TRUE)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
(*state)++;
2019-04-01 18:05:58 +02:00
}
break;
case 2:
2020-06-01 16:23:12 +02:00
if (TryReleaseUnionRoomPlayerObjectEvent(playerIdx))
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
*state = 0;
2019-04-01 18:05:58 +02:00
return TRUE;
}
break;
}
return FALSE;
}
2020-06-01 16:23:12 +02:00
static bool32 SpawnGroupLeader(u32 playerIdx, u32 gender, u32 id)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
struct UnionRoomObject * ptr = &sUnionObjWork[playerIdx];
ptr->schedAnim = UNION_ROOM_SPAWN_IN;
ptr->gfxId = GetUnionRoomPlayerGraphicsId(gender, id);
if (ptr->state == 0)
2019-04-01 18:05:58 +02:00
return TRUE;
else
return FALSE;
}
2020-06-01 16:23:12 +02:00
static bool32 DespawnGroupLeader(u32 playerIdx)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
struct UnionRoomObject * ptr = &sUnionObjWork[playerIdx];
ptr->schedAnim = UNION_ROOM_SPAWN_OUT;
if (ptr->state == 1)
2019-04-01 18:05:58 +02:00
return TRUE;
else
return FALSE;
}
2020-06-01 16:23:12 +02:00
static void AnimateUnionRoomPlayer(u32 playerIdx, struct UnionRoomObject * ptr)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
switch (ptr->state)
2019-04-01 18:05:58 +02:00
{
case 0:
2020-06-01 16:23:12 +02:00
if (ptr->schedAnim == UNION_ROOM_SPAWN_IN)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
ptr->state = 2;
ptr->animState = 0;
2019-04-01 18:05:58 +02:00
}
else
{
break;
}
// fallthrough
case 2:
2020-06-01 16:23:12 +02:00
if (!IsUnionRoomPlayerInvisible(playerIdx, 0) && ptr->schedAnim == UNION_ROOM_SPAWN_OUT)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
ptr->state = 0;
ptr->animState = 0;
RemoveUnionRoomPlayerObjectEvent(playerIdx);
2019-04-01 18:05:58 +02:00
HideUnionRoomPlayer(playerIdx);
}
2020-06-01 16:23:12 +02:00
else if (AnimateUnionRoomPlayerSpawn(&ptr->animState, playerIdx, ptr) == TRUE)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
ptr->state = 1;
2019-04-01 18:05:58 +02:00
}
break;
case 1:
2020-06-01 16:23:12 +02:00
if (ptr->schedAnim == UNION_ROOM_SPAWN_OUT)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
ptr->state = 3;
ptr->animState = 0;
2019-04-01 18:05:58 +02:00
}
else
{
break;
}
// fallthrough
case 3:
2020-06-01 16:23:12 +02:00
if (AnimateUnionRoomPlayerDespawn(&ptr->animState, playerIdx, ptr) == 1)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
ptr->state = 0;
2019-04-01 18:05:58 +02:00
}
break;
}
2020-06-01 16:23:12 +02:00
ptr->schedAnim = UNION_ROOM_SPAWN_NONE;
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static void Task_AnimateUnionRoomPlayers(u8 taskId)
2019-04-01 18:05:58 +02:00
{
s32 i;
2020-06-01 16:23:12 +02:00
for (i = 0; i < MAX_UNION_ROOM_PLAYERS; i++)
AnimateUnionRoomPlayer(i, &sUnionObjWork[i]);
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static u8 CreateTask_AnimateUnionRoomPlayers(void)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
if (FuncIsActiveTask(Task_AnimateUnionRoomPlayers) == TRUE)
2019-04-01 18:05:58 +02:00
return NUM_TASKS;
else
2020-06-01 16:23:12 +02:00
return CreateTask(Task_AnimateUnionRoomPlayers, 5);
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static void DestroyTask_AnimateUnionRoomPlayers(void)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
u8 taskId = FindTaskIdByFunc(Task_AnimateUnionRoomPlayers);
2019-04-01 18:05:58 +02:00
if (taskId < NUM_TASKS)
{
DestroyTask(taskId);
}
}
2020-06-01 16:23:12 +02:00
void DestroyUnionRoomPlayerObjects(void)
2019-04-01 18:05:58 +02:00
{
s32 i;
2020-06-01 16:23:12 +02:00
for (i = 0; i < MAX_UNION_ROOM_PLAYERS; i++)
2019-04-01 18:05:58 +02:00
{
if (!IsUnionRoomPlayerHidden(i))
{
RemoveUnionRoomPlayerObjectEvent(i);
2019-04-01 18:05:58 +02:00
HideUnionRoomPlayer(i);
}
}
2020-06-01 16:23:12 +02:00
sUnionObjWork = NULL;
DestroyTask_AnimateUnionRoomPlayers();
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
void CreateGroupMemberSpritesInvisible(u8 * spriteIds, s32 playerIdx)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
s32 direction;
2019-04-01 18:05:58 +02:00
2020-06-01 16:23:12 +02:00
for (direction = DIR_NONE; direction <= DIR_EAST; direction++)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
s32 id = UR_PLAYER_SPRITE_ID(playerIdx, direction);
spriteIds[id] = CreateObjectSprite(OBJ_EVENT_GFX_MAN_4,
id - UR_SPRITE_START_ID,
sUnionRoomPlayerCoords[playerIdx][0] + sFacingDirectionOffsets[direction][0],
sUnionRoomPlayerCoords[playerIdx][1] + sFacingDirectionOffsets[direction][1],
3, 1);
SetObjectEventSpriteInvisibility(id - UR_SPRITE_START_ID, TRUE);
2019-04-01 18:05:58 +02:00
}
}
2020-06-01 16:23:12 +02:00
void DestroyGroupMemberSprites(u8 * spriteIds)
2019-04-01 18:05:58 +02:00
{
s32 i;
2020-06-01 16:23:12 +02:00
for (i = 0; i < UR_PLAYER_SPRITE_ID(MAX_UNION_ROOM_PLAYERS, 0); i++)
DestroySprite(&gSprites[spriteIds[i]]);
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
void SetTilesAroundUnionRoomPlayersPassable(void)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
s32 i, direction, x, y;
for (i = 0; i < MAX_UNION_ROOM_PLAYERS; i++)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
for (direction = DIR_NONE; direction <= DIR_EAST; direction++)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
GetUnionRoomPlayerFacingCoords(i, direction, &x, &y);
MapGridSetMetatileImpassabilityAt(x, y, FALSE);
2019-04-01 18:05:58 +02:00
}
}
}
2020-06-01 16:23:12 +02:00
static u8 GetNewFacingDirectionForUnionRoomPlayer(u32 direction, u32 playerIdx, struct GFtgtGname * gname)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
if (direction != DIR_NONE)
return sNextFacingDirection[direction];
else if (gname->activity == (ACTIVITY_CHAT | IN_UNION_ROOM))
return DIR_SOUTH;
2019-04-01 18:05:58 +02:00
else
2020-06-01 16:23:12 +02:00
return DIR_EAST;
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static bool32 IsUnionRoomPlayerInvisible(u32 playerIdx, u32 direction)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
return IsObjectEventSpriteInvisible(UR_PLAYER_SPRITE_ID(playerIdx, direction) - UR_SPRITE_START_ID);
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static void SpawnGroupMember(u32 playerIdx, u32 direction, u8 graphicsId, struct GFtgtGname * gname)
2019-04-01 18:05:58 +02:00
{
s32 x, y;
2020-06-01 16:23:12 +02:00
s32 id = UR_PLAYER_SPRITE_ID(playerIdx, direction);
if (IsUnionRoomPlayerInvisible(playerIdx, direction) == TRUE)
2019-04-01 18:05:58 +02:00
{
2020-06-01 16:23:12 +02:00
SetObjectEventSpriteInvisibility(id - UR_SPRITE_START_ID, FALSE);
SetObjectEventSpriteAnim(id - UR_SPRITE_START_ID, UNION_ROOM_SPAWN_IN);
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
SetObjectEventSpriteGraphics(id - UR_SPRITE_START_ID, graphicsId);
SetUnionRoomObjectFacingDirection(direction, playerIdx, GetNewFacingDirectionForUnionRoomPlayer(direction, playerIdx, gname));
GetUnionRoomPlayerFacingCoords(playerIdx, direction, &x, &y);
MapGridSetMetatileImpassabilityAt(x, y, TRUE);
2019-04-01 18:05:58 +02:00
}
2020-06-01 16:23:12 +02:00
static void DespawnGroupMember(u32 playerIdx, u32 direction)
2019-04-01 18:05:58 +02:00
{
s32 x, y;
2020-06-01 16:23:12 +02:00
SetObjectEventSpriteAnim(UR_PLAYER_SPRITE_ID(playerIdx, direction) - UR_SPRITE_START_ID, UNION_ROOM_SPAWN_OUT);
GetUnionRoomPlayerFacingCoords(playerIdx, direction, &x, &y);
MapGridSetMetatileImpassabilityAt(x, y, FALSE);
2019-04-01 18:05:58 +02:00
}
2019-04-01 20:09:56 +02:00
2020-06-01 16:23:12 +02:00
static void AssembleGroup(u32 playerIdx, struct GFtgtGname * gname)
2019-04-01 20:09:56 +02:00
{
s16 x, y, x2, y2;
s32 i;
PlayerGetDestCoords(&x, &y);
player_get_pos_including_state_based_drift(&x2, &y2);
2020-06-01 16:23:12 +02:00
if (IsObjectEventSpriteInvisible(UR_PLAYER_SPRITE_ID(playerIdx, 0) - UR_SPRITE_START_ID) == TRUE)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
if (IsUnionRoomPlayerFacingTileAt(playerIdx, 0, x, y) == TRUE || IsUnionRoomPlayerFacingTileAt(playerIdx, 0, x2, y2) == TRUE)
2019-04-01 20:09:56 +02:00
{
return;
}
2020-06-01 16:23:12 +02:00
SpawnGroupMember(playerIdx, 0, GetUnionRoomPlayerGraphicsId(gname->playerGender, gname->unk_00.playerTrainerId[0]), gname);
2019-04-01 20:09:56 +02:00
}
for (i = 1; i < 5; i++)
{
2020-06-01 16:23:12 +02:00
if (gname->child_sprite_gender[i - 1] == 0)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
DespawnGroupMember(playerIdx, i);
2019-04-01 20:09:56 +02:00
}
2020-06-01 16:23:12 +02:00
else if (IsUnionRoomPlayerFacingTileAt(playerIdx, i, x, y) == FALSE && IsUnionRoomPlayerFacingTileAt(playerIdx, i, x2, y2) == FALSE)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
SpawnGroupMember(playerIdx, i, GetUnionRoomPlayerGraphicsId((gname->child_sprite_gender[i - 1] >> 3) & 1, gname->child_sprite_gender[i - 1] & 7), gname);
2019-04-01 20:09:56 +02:00
}
}
}
2020-06-01 16:23:12 +02:00
static void SpawnGroupLeaderAndMembers(u32 playerIdx, struct GFtgtGname * gname)
2019-04-01 20:09:56 +02:00
{
u32 i;
2020-06-01 16:23:12 +02:00
switch (gname->activity)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
case ACTIVITY_NONE | IN_UNION_ROOM:
case ACTIVITY_PLYRTALK | IN_UNION_ROOM:
SpawnGroupLeader(playerIdx, gname->playerGender, gname->unk_00.playerTrainerId[0]);
2019-04-01 20:09:56 +02:00
for (i = 0; i < 5; i++)
{
2020-06-01 16:23:12 +02:00
DespawnGroupMember(playerIdx, i);
2019-04-01 20:09:56 +02:00
}
break;
2020-06-01 16:23:12 +02:00
case ACTIVITY_BATTLE_SINGLE | IN_UNION_ROOM:
case ACTIVITY_TRADE | IN_UNION_ROOM:
case ACTIVITY_CHAT | IN_UNION_ROOM:
case ACTIVITY_CARD | IN_UNION_ROOM:
case ACTIVITY_ACCEPT | IN_UNION_ROOM:
case ACTIVITY_DECLINE | IN_UNION_ROOM:
case ACTIVITY_NPCTALK | IN_UNION_ROOM:
DespawnGroupLeader(playerIdx);
AssembleGroup(playerIdx, gname);
2019-04-01 20:09:56 +02:00
break;
}
}
2020-06-01 16:23:12 +02:00
static void DespawnGroupLeaderAndMembers(u32 r5, struct GFtgtGname *gname)
2019-04-01 20:09:56 +02:00
{
s32 i;
2020-06-01 16:23:12 +02:00
DespawnGroupLeader(r5);
2019-04-01 20:09:56 +02:00
for (i = 0; i < 5; i++)
{
2020-06-01 16:23:12 +02:00
DespawnGroupMember(r5, i);
2019-04-01 20:09:56 +02:00
}
}
2020-06-01 16:23:12 +02:00
static void UpdateUnionRoomPlayerSprites(struct WirelessLink_URoom *uroom)
2019-04-01 20:09:56 +02:00
{
s32 i;
struct UnkStruct_x20 * r4;
2020-06-01 16:23:12 +02:00
sUnionObjRefreshTimer = 0;
for (i = 0, r4 = uroom->field_0->arr; i < MAX_UNION_ROOM_PLAYERS; i++)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
if (r4[i].groupScheduledAnim == UNION_ROOM_SPAWN_IN)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
SpawnGroupLeaderAndMembers(i, &r4[i].gname_uname.gname);
2019-04-01 20:09:56 +02:00
}
2020-06-01 16:23:12 +02:00
else if (r4[i].groupScheduledAnim == UNION_ROOM_SPAWN_OUT)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
DespawnGroupLeaderAndMembers(i, &r4[i].gname_uname.gname);
2019-04-01 20:09:56 +02:00
}
}
}
2020-06-01 16:23:12 +02:00
void ScheduleUnionRoomPlayerRefresh(struct WirelessLink_URoom *uroom)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
sUnionObjRefreshTimer = 300;
2019-04-01 20:09:56 +02:00
}
2020-06-01 16:23:12 +02:00
void HandleUnionRoomPlayerRefresh(struct WirelessLink_URoom *uroom)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
if (++sUnionObjRefreshTimer > 300)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
UpdateUnionRoomPlayerSprites(uroom);
2019-04-01 20:09:56 +02:00
}
}
2020-06-09 00:16:57 +02:00
bool32 TryInteractWithUnionRoomMember(struct UnkStruct_Main0 *main0, s16 *directionPtr, s16 *playerIdxPtr, u8 *spriteIds)
2019-04-01 20:09:56 +02:00
{
s16 x, y;
2020-06-01 16:23:12 +02:00
s32 i, direction;
2019-04-01 20:09:56 +02:00
struct UnkStruct_x20 * r4;
2020-06-09 00:16:57 +02:00
if (!IsPlayerStandingStill())
2019-04-01 20:09:56 +02:00
{
return FALSE;
}
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
2020-06-01 16:23:12 +02:00
for (i = 0, r4 = main0->arr; i < MAX_UNION_ROOM_PLAYERS; i++)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
for (direction = DIR_NONE; direction <= DIR_EAST; direction++)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
s32 id = UR_PLAYER_SPRITE_ID(i, direction);
if (x != sUnionRoomPlayerCoords[i][0] + sFacingDirectionOffsets[direction][0] + 7)
2019-04-01 20:09:56 +02:00
{
continue;
}
2020-06-01 16:23:12 +02:00
if (y != sUnionRoomPlayerCoords[i][1] + sFacingDirectionOffsets[direction][1] + 7)
2019-04-01 20:09:56 +02:00
{
continue;
}
2020-06-01 16:23:12 +02:00
if (IsObjectEventSpriteInvisible(id - UR_SPRITE_START_ID))
2019-04-01 20:09:56 +02:00
{
continue;
}
2020-06-01 16:23:12 +02:00
if (IsObjectEventSpriteAnimating(id - UR_SPRITE_START_ID))
2019-04-01 20:09:56 +02:00
{
continue;
}
2020-06-01 16:23:12 +02:00
if (r4[i].groupScheduledAnim != UNION_ROOM_SPAWN_IN)
2019-04-01 20:09:56 +02:00
{
continue;
}
2020-06-09 00:16:57 +02:00
// Face player
2020-06-01 16:23:12 +02:00
SetUnionRoomObjectFacingDirection(direction, i, sOppositeFacingDirection[GetPlayerFacingDirection()]);
*directionPtr = direction;
*playerIdxPtr = i;
2019-04-01 20:09:56 +02:00
return TRUE;
}
}
return FALSE;
}
2020-06-01 16:23:12 +02:00
static void SetUnionRoomObjectFacingDirection(s32 currDirection, s32 playerIdx, u8 newDirection)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
TurnObjectEventSprite(5 * playerIdx - UR_SPRITE_START_ID + currDirection, newDirection);
// should be line below, but order is swapped here
// TurnObjectEventSprite(UR_PLAYER_SPRITE_ID(playerIdx, currDirection) - UR_SPRITE_START_ID, newDirection);
2019-04-01 20:09:56 +02:00
}
2020-06-01 16:23:12 +02:00
void UpdateUnionRoomMemberFacing(u32 currDirection, u32 playerIdx, struct UnkStruct_Main0 *main0)
2019-04-01 20:09:56 +02:00
{
2020-06-01 16:23:12 +02:00
return SetUnionRoomObjectFacingDirection(currDirection, playerIdx, GetNewFacingDirectionForUnionRoomPlayer(currDirection, playerIdx, &main0->arr[playerIdx].gname_uname.gname));
2019-04-01 20:09:56 +02:00
}