pokeemerald/src/secret_base.c

2073 lines
70 KiB
C
Raw Normal View History

2017-10-23 18:35:06 -04:00
#include "global.h"
#include "malloc.h"
2019-01-01 23:19:07 +00:00
#include "battle.h"
#include "battle_setup.h"
#include "decoration.h"
#include "event_data.h"
#include "event_object_movement.h"
#include "event_scripts.h"
2017-10-24 20:19:33 -04:00
#include "field_camera.h"
2019-01-01 23:19:07 +00:00
#include "field_effect.h"
2017-10-23 19:29:16 -04:00
#include "field_player_avatar.h"
2018-12-08 22:05:11 +01:00
#include "field_screen_effect.h"
2019-01-01 23:19:07 +00:00
#include "field_specials.h"
2017-10-24 21:27:46 -04:00
#include "field_weather.h"
2019-01-01 23:19:07 +00:00
#include "fieldmap.h"
2018-12-24 21:59:05 +01:00
#include "fldeff.h"
2018-12-19 22:19:54 -05:00
#include "fldeff_misc.h"
2019-01-01 23:19:07 +00:00
#include "international_string_util.h"
#include "item_menu.h"
2019-01-01 23:19:07 +00:00
#include "link.h"
#include "list_menu.h"
#include "main.h"
2017-10-24 21:41:22 -04:00
#include "map_name_popup.h"
2019-01-01 23:19:07 +00:00
#include "menu.h"
#include "menu_helpers.h"
#include "metatile_behavior.h"
#include "overworld.h"
#include "palette.h"
2017-10-24 21:41:22 -04:00
#include "script.h"
2019-01-01 23:19:07 +00:00
#include "secret_base.h"
#include "sound.h"
#include "string_util.h"
2017-10-26 21:39:34 -04:00
#include "strings.h"
2019-01-01 23:19:07 +00:00
#include "task.h"
2017-10-27 22:12:11 -04:00
#include "tv.h"
2019-01-01 23:19:07 +00:00
#include "window.h"
2019-11-21 14:03:35 -05:00
#include "constants/event_bg.h"
2019-01-01 23:19:07 +00:00
#include "constants/decorations.h"
2019-04-05 16:11:24 -05:00
#include "constants/event_objects.h"
2020-06-17 17:48:20 -04:00
#include "constants/field_specials.h"
2019-04-05 16:11:24 -05:00
#include "constants/items.h"
2018-07-12 22:39:38 -05:00
#include "constants/map_types.h"
2019-01-01 23:19:07 +00:00
#include "constants/metatile_behaviors.h"
#include "constants/metatile_labels.h"
2019-04-05 16:11:24 -05:00
#include "constants/moves.h"
2019-01-05 20:12:50 +00:00
#include "constants/secret_bases.h"
2019-01-01 23:19:07 +00:00
#include "constants/songs.h"
2018-10-21 20:13:12 +02:00
#include "constants/trainers.h"
2017-10-23 18:35:06 -04:00
2021-02-05 12:08:25 -05:00
// Values for registryStatus
enum {
UNREGISTERED,
REGISTERED,
NEW, // Temporary, so new secret bases mixed at the same time don't overwrite each other
};
2019-04-05 16:11:24 -05:00
struct SecretBaseRegistryMenu
{
2017-10-23 18:56:18 -04:00
struct ListMenuItem items[11];
u8 names[11][32];
};
2019-04-05 16:11:24 -05:00
struct SecretBaseRecordMixer
{
struct SecretBase *secretBases;
2017-10-27 21:17:17 -04:00
u32 version;
u32 language;
};
2019-04-05 16:11:24 -05:00
struct SecretBaseEntranceMetatiles
{
u16 closedMetatileId;
u16 openMetatileId;
2017-10-27 23:21:09 -04:00
};
2019-04-05 16:11:24 -05:00
static EWRAM_DATA u8 sCurSecretBaseId = 0;
static EWRAM_DATA bool8 sInFriendSecretBase = FALSE;
static EWRAM_DATA struct SecretBaseRegistryMenu *sRegistryMenu = NULL;
2021-02-05 12:08:25 -05:00
static void Task_ShowSecretBaseRegistryMenu(u8);
static void BuildRegistryMenuItems(u8);
static void RegistryMenu_OnCursorMove(s32, bool8, struct ListMenu *);
static void FinalizeRegistryMenu(u8);
static void AddRegistryMenuScrollArrows(u8);
static void HandleRegistryMenuInput(u8);
static void ShowRegistryMenuActions(u8);
static void HandleRegistryMenuActionsInput(u8);
static void ShowRegistryMenuDeleteConfirmation(u8);
static void ShowRegistryMenuDeleteYesNo(u8);
static void DeleteRegistry_Yes(u8);
static void DeleteRegistry_No(u8);
static void ReturnToMainRegistryMenu(u8);
static void GoToSecretBasePCRegisterMenu(u8);
static u8 GetSecretBaseOwnerType(u8);
2019-04-05 16:11:24 -05:00
static const struct SecretBaseEntranceMetatiles sSecretBaseEntranceMetatiles[] =
{
{.closedMetatileId = METATILE_General_SecretBase_TreeLeft, .openMetatileId = METATILE_General_SecretBase_VineLeft},
{.closedMetatileId = METATILE_General_SecretBase_TreeRight, .openMetatileId = METATILE_General_SecretBase_VineRight},
{.closedMetatileId = METATILE_General_RedCaveIndent, .openMetatileId = METATILE_General_RedCaveOpen},
{.closedMetatileId = METATILE_General_YellowCaveIndent, .openMetatileId = METATILE_General_YellowCaveOpen},
{.closedMetatileId = METATILE_General_BlueCaveIndent, .openMetatileId = METATILE_General_BlueCaveOpen},
{.closedMetatileId = METATILE_Fallarbor_BrownCaveIndent, .openMetatileId = METATILE_Fallarbor_BrownCaveOpen},
{.closedMetatileId = METATILE_Fortree_SecretBase_Shrub, .openMetatileId = METATILE_Fortree_SecretBase_ShrubOpen},
2017-10-27 23:21:09 -04:00
};
2019-04-05 16:11:24 -05:00
// mapNum, warpId, x, y
// x, y positions are for when the player warps in for the first time (in front of the computer)
2021-02-05 12:08:25 -05:00
static const u8 sSecretBaseEntrancePositions[NUM_SECRET_BASE_GROUPS * 4] =
{
[SECRET_BASE_RED_CAVE1] = MAP_NUM(SECRET_BASE_RED_CAVE1), 0, 1, 3,
[SECRET_BASE_RED_CAVE2] = MAP_NUM(SECRET_BASE_RED_CAVE2), 0, 5, 9,
[SECRET_BASE_RED_CAVE3] = MAP_NUM(SECRET_BASE_RED_CAVE3), 0, 1, 3,
[SECRET_BASE_RED_CAVE4] = MAP_NUM(SECRET_BASE_RED_CAVE4), 0, 7, 13,
[SECRET_BASE_BROWN_CAVE1] = MAP_NUM(SECRET_BASE_BROWN_CAVE1), 0, 2, 3,
[SECRET_BASE_BROWN_CAVE2] = MAP_NUM(SECRET_BASE_BROWN_CAVE2), 0, 9, 2,
[SECRET_BASE_BROWN_CAVE3] = MAP_NUM(SECRET_BASE_BROWN_CAVE3), 0, 13, 4,
[SECRET_BASE_BROWN_CAVE4] = MAP_NUM(SECRET_BASE_BROWN_CAVE4), 0, 1, 2,
[SECRET_BASE_BLUE_CAVE1] = MAP_NUM(SECRET_BASE_BLUE_CAVE1), 0, 1, 3,
[SECRET_BASE_BLUE_CAVE2] = MAP_NUM(SECRET_BASE_BLUE_CAVE2), 0, 1, 2,
[SECRET_BASE_BLUE_CAVE3] = MAP_NUM(SECRET_BASE_BLUE_CAVE3), 0, 3, 15,
[SECRET_BASE_BLUE_CAVE4] = MAP_NUM(SECRET_BASE_BLUE_CAVE4), 0, 3, 14,
[SECRET_BASE_YELLOW_CAVE1] = MAP_NUM(SECRET_BASE_YELLOW_CAVE1), 0, 9, 3,
[SECRET_BASE_YELLOW_CAVE2] = MAP_NUM(SECRET_BASE_YELLOW_CAVE2), 0, 8, 7,
[SECRET_BASE_YELLOW_CAVE3] = MAP_NUM(SECRET_BASE_YELLOW_CAVE3), 0, 3, 6,
[SECRET_BASE_YELLOW_CAVE4] = MAP_NUM(SECRET_BASE_YELLOW_CAVE4), 0, 5, 9,
[SECRET_BASE_TREE1] = MAP_NUM(SECRET_BASE_TREE1), 0, 2, 3,
[SECRET_BASE_TREE2] = MAP_NUM(SECRET_BASE_TREE2), 0, 5, 6,
[SECRET_BASE_TREE3] = MAP_NUM(SECRET_BASE_TREE3), 0, 15, 3,
[SECRET_BASE_TREE4] = MAP_NUM(SECRET_BASE_TREE4), 0, 4, 10,
[SECRET_BASE_SHRUB1] = MAP_NUM(SECRET_BASE_SHRUB1), 0, 3, 3,
[SECRET_BASE_SHRUB2] = MAP_NUM(SECRET_BASE_SHRUB2), 0, 1, 2,
[SECRET_BASE_SHRUB3] = MAP_NUM(SECRET_BASE_SHRUB3), 0, 7, 8,
[SECRET_BASE_SHRUB4] = MAP_NUM(SECRET_BASE_SHRUB4), 0, 9, 6,
2017-10-27 23:21:09 -04:00
};
2021-02-05 12:08:25 -05:00
#define GET_BASE_MAP_NUM(group) (sSecretBaseEntrancePositions[(group) + 0])
#define GET_BASE_WARP_ID(group) (sSecretBaseEntrancePositions[(group) + 1])
#define GET_BASE_COMPUTER_X(group) (sSecretBaseEntrancePositions[(group) + 2])
#define GET_BASE_COMPUTER_Y(group) (sSecretBaseEntrancePositions[(group) + 3])
2019-04-05 16:11:24 -05:00
static const struct MenuAction sRegistryMenuActions[] =
{
{
.text = gText_DelRegist,
.func = { .void_u8 = ShowRegistryMenuDeleteConfirmation },
},
{
.text = gText_Cancel,
.func = { .void_u8 = ReturnToMainRegistryMenu },
},
2017-10-27 23:21:09 -04:00
};
2019-04-05 16:11:24 -05:00
static const struct YesNoFuncTable sDeleteRegistryYesNoFuncs =
{
.yesFunc = DeleteRegistry_Yes,
.noFunc = DeleteRegistry_No,
2017-10-27 23:21:09 -04:00
};
2017-10-24 20:19:33 -04:00
2019-04-05 16:11:24 -05:00
static const u8 sSecretBaseOwnerGfxIds[10] =
{
// Male
OBJ_EVENT_GFX_YOUNGSTER,
OBJ_EVENT_GFX_BUG_CATCHER,
OBJ_EVENT_GFX_RICH_BOY,
OBJ_EVENT_GFX_CAMPER,
OBJ_EVENT_GFX_MAN_3,
2019-04-05 16:11:24 -05:00
// Female
OBJ_EVENT_GFX_LASS,
OBJ_EVENT_GFX_GIRL_3,
OBJ_EVENT_GFX_WOMAN_2,
OBJ_EVENT_GFX_PICNICKER,
OBJ_EVENT_GFX_WOMAN_5,
2017-10-27 23:21:09 -04:00
};
2019-04-05 16:11:24 -05:00
static const struct WindowTemplate sRegistryWindowTemplates[] =
{
{
.bg = 0,
.tilemapLeft = 18,
.tilemapTop = 1,
.width = 11,
.height = 18,
.paletteNum = 15,
.baseBlock = 0x01,
},
{
.bg = 0,
.tilemapLeft = 2,
.tilemapTop = 1,
.width = 28,
.height = 4,
.paletteNum = 15,
.baseBlock = 0xc7,
}
2017-10-27 23:21:09 -04:00
};
2017-10-24 21:10:58 -04:00
2019-04-05 16:11:24 -05:00
static const struct ListMenuTemplate sRegistryListMenuTemplate =
{
.items = NULL,
.moveCursorFunc = RegistryMenu_OnCursorMove,
.itemPrintFunc = NULL,
.totalItems = 0,
.maxShowed = 0,
.windowId = 0,
.header_X = 0,
.item_X = 8,
.cursor_X = 0,
.upText_Y = 9,
.cursorPal = 2,
.fillValue = 1,
.cursorShadowPal = 3,
.lettersSpacing = 0,
.itemVerticalPadding = 0,
.scrollMultiple = LIST_NO_MULTIPLE_SCROLL,
2021-10-30 16:47:37 -04:00
.fontId = FONT_NORMAL,
2022-11-09 15:14:41 -05:00
.cursorKind = CURSOR_BLACK_ARROW,
2019-04-05 16:11:24 -05:00
};
2017-10-23 19:29:16 -04:00
2019-04-05 16:11:24 -05:00
static void ClearSecretBase(struct SecretBase *secretBase)
2017-10-23 18:35:06 -04:00
{
u16 i;
2019-04-05 16:11:24 -05:00
CpuFastFill16(0, secretBase, sizeof(struct SecretBase));
for (i = 0; i < PLAYER_NAME_LENGTH; i++)
secretBase->trainerName[i] = EOS;
2017-10-23 18:35:06 -04:00
}
2019-04-05 16:11:24 -05:00
void ClearSecretBases(void)
2017-10-23 18:56:18 -04:00
{
u16 i;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
ClearSecretBase(&gSaveBlock1Ptr->secretBases[i]);
2017-10-23 18:56:18 -04:00
}
2019-04-05 16:11:24 -05:00
static void SetCurSecretBaseId(void)
2017-10-23 18:56:18 -04:00
{
2017-10-23 19:21:08 -04:00
sCurSecretBaseId = gSpecialVar_0x8004;
}
2019-04-05 16:11:24 -05:00
void TrySetCurSecretBaseIndex(void)
2017-10-23 19:21:08 -04:00
{
u16 i;
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = FALSE;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-23 19:21:08 -04:00
{
2019-04-05 16:11:24 -05:00
if (sCurSecretBaseId == gSaveBlock1Ptr->secretBases[i].secretBaseId)
2017-10-23 19:21:08 -04:00
{
2019-04-05 16:11:24 -05:00
gSpecialVar_Result = TRUE;
VarSet(VAR_CURRENT_SECRET_BASE, i);
break;
2017-10-23 19:21:08 -04:00
}
}
}
2019-04-05 16:11:24 -05:00
void CheckPlayerHasSecretBase(void)
2017-10-23 19:21:08 -04:00
{
2019-04-05 16:11:24 -05:00
// The player's secret base is always the first in the array.
if (gSaveBlock1Ptr->secretBases[0].secretBaseId)
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = TRUE;
2017-10-23 19:21:08 -04:00
else
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = FALSE;
2017-10-23 18:56:18 -04:00
}
2019-04-05 16:11:24 -05:00
static u8 GetSecretBaseTypeInFrontOfPlayer_(void)
2017-10-23 19:29:16 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-23 19:29:16 -04:00
s16 behavior;
2017-10-23 18:35:06 -04:00
2017-10-23 19:29:16 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
behavior = MapGridGetMetatileBehaviorAt(x, y) & 0xFFF;
2019-01-01 23:19:07 +00:00
if (behavior == MB_SECRET_BASE_SPOT_RED_CAVE || behavior == MB_SECRET_BASE_SPOT_RED_CAVE_OPEN)
2019-01-05 20:12:50 +00:00
return SECRET_BASE_RED_CAVE;
2019-04-05 16:11:24 -05:00
2019-01-01 23:19:07 +00:00
if (behavior == MB_SECRET_BASE_SPOT_BROWN_CAVE || behavior == MB_SECRET_BASE_SPOT_BROWN_CAVE_OPEN)
2019-01-05 20:12:50 +00:00
return SECRET_BASE_BROWN_CAVE;
2019-04-05 16:11:24 -05:00
2019-01-01 23:19:07 +00:00
if (behavior == MB_SECRET_BASE_SPOT_BLUE_CAVE || behavior == MB_SECRET_BASE_SPOT_BLUE_CAVE_OPEN)
2019-01-05 20:12:50 +00:00
return SECRET_BASE_BLUE_CAVE;
2019-04-05 16:11:24 -05:00
2019-01-01 23:19:07 +00:00
if (behavior == MB_SECRET_BASE_SPOT_YELLOW_CAVE || behavior == MB_SECRET_BASE_SPOT_YELLOW_CAVE_OPEN)
2019-01-05 20:12:50 +00:00
return SECRET_BASE_YELLOW_CAVE;
2019-04-05 16:11:24 -05:00
if (behavior == MB_SECRET_BASE_SPOT_TREE_LEFT || behavior == MB_SECRET_BASE_SPOT_TREE_LEFT_OPEN
|| behavior == MB_SECRET_BASE_SPOT_TREE_RIGHT || behavior == MB_SECRET_BASE_SPOT_TREE_RIGHT_OPEN)
2019-01-05 20:12:50 +00:00
return SECRET_BASE_TREE;
2019-04-05 16:11:24 -05:00
2019-01-01 23:19:07 +00:00
if (behavior == MB_SECRET_BASE_SPOT_SHRUB || behavior == MB_SECRET_BASE_SPOT_SHRUB_OPEN)
2019-01-05 20:12:50 +00:00
return SECRET_BASE_SHRUB;
2019-04-05 16:11:24 -05:00
2017-10-23 19:29:16 -04:00
return 0;
}
2017-10-24 20:19:33 -04:00
2019-04-05 16:11:24 -05:00
void GetSecretBaseTypeInFrontOfPlayer(void)
2017-10-24 20:19:33 -04:00
{
2019-04-05 16:11:24 -05:00
gSpecialVar_0x8007 = GetSecretBaseTypeInFrontOfPlayer_();
2017-10-24 20:19:33 -04:00
}
2019-04-05 16:11:24 -05:00
static void FindMetatileIdMapCoords(s16 *x, s16 *y, u16 metatileId)
2017-10-24 20:19:33 -04:00
{
2019-04-05 16:11:24 -05:00
s16 i, j;
const struct MapLayout *mapLayout = gMapHeader.mapLayout;
2017-10-24 20:19:33 -04:00
2019-04-05 16:11:24 -05:00
for (j = 0; j < mapLayout->height; j++)
2017-10-24 20:19:33 -04:00
{
2019-04-05 16:11:24 -05:00
for (i = 0; i < mapLayout->width; i++)
2017-10-24 20:19:33 -04:00
{
2022-01-19 10:15:32 -05:00
if ((mapLayout->map[j * mapLayout->width + i] & MAPGRID_METATILE_ID_MASK) == metatileId)
2017-10-24 20:19:33 -04:00
{
2019-04-05 16:11:24 -05:00
*x = i;
*y = j;
2017-10-24 20:19:33 -04:00
return;
}
}
}
}
2019-04-05 16:11:24 -05:00
// Opens or closes the secret base entrance metatile in front of the player.
void ToggleSecretBaseEntranceMetatile(void)
2017-10-24 20:19:33 -04:00
{
u16 i;
2019-04-05 16:11:24 -05:00
s16 x, y;
s16 metatileId;
2017-10-24 20:19:33 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
2019-04-05 16:11:24 -05:00
metatileId = MapGridGetMetatileIdAt(x, y);
2021-02-05 12:08:25 -05:00
// Look for entrance metatiles to open
2019-04-05 16:11:24 -05:00
for (i = 0; i < ARRAY_COUNT(sSecretBaseEntranceMetatiles); i++)
2017-10-24 20:19:33 -04:00
{
2019-04-05 16:11:24 -05:00
if (sSecretBaseEntranceMetatiles[i].closedMetatileId == metatileId)
2017-10-24 20:19:33 -04:00
{
2022-01-19 10:15:32 -05:00
MapGridSetMetatileIdAt(x, y, sSecretBaseEntranceMetatiles[i].openMetatileId | MAPGRID_COLLISION_MASK);
2017-10-24 20:19:33 -04:00
CurrentMapDrawMetatileAt(x, y);
return;
}
}
2019-04-05 16:11:24 -05:00
2021-02-05 12:08:25 -05:00
// Look for entrance metatiles to close
2019-04-05 16:11:24 -05:00
for (i = 0; i < ARRAY_COUNT(sSecretBaseEntranceMetatiles); i++)
2017-10-24 20:19:33 -04:00
{
2019-04-05 16:11:24 -05:00
if (sSecretBaseEntranceMetatiles[i].openMetatileId == metatileId)
2017-10-24 20:19:33 -04:00
{
2022-01-19 10:15:32 -05:00
MapGridSetMetatileIdAt(x, y, sSecretBaseEntranceMetatiles[i].closedMetatileId | MAPGRID_COLLISION_MASK);
2017-10-24 20:19:33 -04:00
CurrentMapDrawMetatileAt(x, y);
return;
}
}
}
2017-10-24 20:27:07 -04:00
2019-04-05 16:11:24 -05:00
static u8 GetNameLength(const u8 *secretBaseOwnerName)
2017-10-24 20:27:07 -04:00
{
u8 i;
2019-04-05 16:11:24 -05:00
for (i = 0; i < PLAYER_NAME_LENGTH; i++)
2017-10-24 20:27:07 -04:00
{
2019-04-05 16:11:24 -05:00
if (secretBaseOwnerName[i] == EOS)
2017-10-24 20:27:07 -04:00
return i;
}
2019-04-05 16:11:24 -05:00
return PLAYER_NAME_LENGTH;
2017-10-24 20:27:07 -04:00
}
2019-04-05 16:11:24 -05:00
void SetPlayerSecretBase(void)
2017-10-24 20:27:07 -04:00
{
u16 i;
gSaveBlock1Ptr->secretBases[0].secretBaseId = sCurSecretBaseId;
2019-09-08 11:53:48 -04:00
for (i = 0; i < TRAINER_ID_LENGTH; i++)
2017-10-24 20:27:07 -04:00
gSaveBlock1Ptr->secretBases[0].trainerId[i] = gSaveBlock2Ptr->playerTrainerId[i];
2019-04-05 16:11:24 -05:00
2018-12-09 19:22:59 -05:00
VarSet(VAR_CURRENT_SECRET_BASE, 0);
2019-04-05 16:11:24 -05:00
StringCopyN(gSaveBlock1Ptr->secretBases[0].trainerName, gSaveBlock2Ptr->playerName, GetNameLength(gSaveBlock2Ptr->playerName));
2017-10-24 20:27:07 -04:00
gSaveBlock1Ptr->secretBases[0].gender = gSaveBlock2Ptr->playerGender;
gSaveBlock1Ptr->secretBases[0].language = GAME_LANGUAGE;
VarSet(VAR_SECRET_BASE_MAP, gMapHeader.regionMapSectionId);
}
2017-10-24 20:37:38 -04:00
2021-02-05 12:08:25 -05:00
// Set the 'open' entrance metatile for any occupied secret base on this map
2019-04-05 16:11:24 -05:00
void SetOccupiedSecretBaseEntranceMetatiles(struct MapEvents const *events)
2017-10-24 20:37:38 -04:00
{
2019-04-05 16:11:24 -05:00
u16 bgId;
u16 i, j;
2017-10-24 20:37:38 -04:00
2019-04-05 16:11:24 -05:00
for (bgId = 0; bgId < events->bgEventCount; bgId++)
2017-10-24 20:37:38 -04:00
{
2019-04-05 16:11:24 -05:00
if (events->bgEvents[bgId].kind == BG_EVENT_SECRET_BASE)
2017-10-24 20:37:38 -04:00
{
2019-04-05 16:11:24 -05:00
for (j = 0; j < SECRET_BASES_COUNT; j++)
2017-10-24 20:37:38 -04:00
{
2019-04-05 16:11:24 -05:00
if (gSaveBlock1Ptr->secretBases[j].secretBaseId == events->bgEvents[bgId].bgUnion.secretBaseId)
2017-10-24 20:37:38 -04:00
{
2021-10-09 12:12:18 -04:00
s16 x = events->bgEvents[bgId].x + MAP_OFFSET;
s16 y = events->bgEvents[bgId].y + MAP_OFFSET;
2019-04-05 16:11:24 -05:00
s16 tile_id = MapGridGetMetatileIdAt(x, y);
for (i = 0; i < ARRAY_COUNT(sSecretBaseEntranceMetatiles); i++)
2017-10-24 20:37:38 -04:00
{
2019-04-05 16:11:24 -05:00
if (sSecretBaseEntranceMetatiles[i].closedMetatileId == tile_id)
2017-10-24 20:37:38 -04:00
{
2022-01-19 10:15:32 -05:00
MapGridSetMetatileIdAt(x, y, sSecretBaseEntranceMetatiles[i].openMetatileId | MAPGRID_COLLISION_MASK);
2017-10-24 20:37:38 -04:00
break;
}
}
break;
}
}
}
}
}
2017-10-24 21:10:58 -04:00
2019-04-05 16:11:24 -05:00
static void SetSecretBaseWarpDestination(void)
2017-10-24 21:10:58 -04:00
{
2021-02-05 12:08:25 -05:00
s8 secretBaseGroup = SECRET_BASE_ID_TO_GROUP(sCurSecretBaseId);
SetWarpDestinationToMapWarp(MAP_GROUP(SECRET_BASE_RED_CAVE1), GET_BASE_MAP_NUM(secretBaseGroup), GET_BASE_WARP_ID(secretBaseGroup));
2017-10-24 21:10:58 -04:00
}
2017-10-24 21:27:46 -04:00
2021-02-05 12:08:25 -05:00
#define tState data[0]
2019-04-05 16:11:24 -05:00
static void Task_EnterSecretBase(u8 taskId)
2017-10-24 21:27:46 -04:00
{
2021-02-05 12:08:25 -05:00
u16 secretBaseIdx;
2017-10-24 21:27:46 -04:00
2021-02-05 12:08:25 -05:00
switch (gTasks[taskId].tState)
2017-10-24 21:27:46 -04:00
{
2019-04-05 16:11:24 -05:00
case 0:
if (!gPaletteFade.active)
2021-02-05 12:08:25 -05:00
gTasks[taskId].tState = 1;
2019-04-05 16:11:24 -05:00
break;
case 1:
2021-02-05 12:08:25 -05:00
secretBaseIdx = VarGet(VAR_CURRENT_SECRET_BASE);
if (gSaveBlock1Ptr->secretBases[secretBaseIdx].numTimesEntered < 255)
gSaveBlock1Ptr->secretBases[secretBaseIdx].numTimesEntered++;
2019-04-05 16:11:24 -05:00
SetSecretBaseWarpDestination();
WarpIntoMap();
2019-12-17 03:24:44 -05:00
gFieldCallback = FieldCB_ContinueScriptHandleMusic;
2019-04-05 16:11:24 -05:00
SetMainCallback2(CB2_LoadMap);
DestroyTask(taskId);
break;
2017-10-24 21:27:46 -04:00
}
}
2021-02-05 12:08:25 -05:00
#undef tState
2019-04-05 16:11:24 -05:00
void EnterSecretBase(void)
2017-10-24 21:27:46 -04:00
{
2019-04-05 16:11:24 -05:00
CreateTask(Task_EnterSecretBase, 0);
2019-12-15 11:42:50 -05:00
FadeScreen(FADE_TO_BLACK, 0);
SetDynamicWarp(0, gSaveBlock1Ptr->location.mapGroup, gSaveBlock1Ptr->location.mapNum, WARP_ID_NONE);
2017-10-24 21:27:46 -04:00
}
2019-04-05 16:11:24 -05:00
bool8 SecretBaseMapPopupEnabled(void)
2017-10-24 21:27:46 -04:00
{
2019-03-01 01:49:11 -05:00
if (gMapHeader.mapType == MAP_TYPE_SECRET_BASE && VarGet(VAR_INIT_SECRET_BASE) == 0)
2017-10-24 21:27:46 -04:00
return FALSE;
2019-04-05 16:11:24 -05:00
2017-10-24 21:27:46 -04:00
return TRUE;
}
2017-10-24 21:41:22 -04:00
2019-04-05 16:11:24 -05:00
static void EnterNewlyCreatedSecretBase_WaitFadeIn(u8 taskId)
2017-10-24 21:41:22 -04:00
{
ObjectEventTurn(&gObjectEvents[gPlayerAvatar.objectEventId], DIR_NORTH);
if (IsWeatherNotFadingIn() == TRUE)
2017-10-24 21:41:22 -04:00
{
ScriptContext_Enable();
2017-10-24 21:41:22 -04:00
DestroyTask(taskId);
}
}
2019-04-05 16:11:24 -05:00
static void EnterNewlyCreatedSecretBase_StartFadeIn(void)
2017-10-24 21:41:22 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-24 21:41:22 -04:00
LockPlayerFieldControls();
2017-10-24 21:41:22 -04:00
HideMapNamePopUpWindow();
FindMetatileIdMapCoords(&x, &y, METATILE_SecretBase_PC);
2021-10-09 12:12:18 -04:00
x += MAP_OFFSET;
y += MAP_OFFSET;
2022-01-19 10:15:32 -05:00
MapGridSetMetatileIdAt(x, y, METATILE_SecretBase_PC | MAPGRID_COLLISION_MASK);
2017-10-24 21:41:22 -04:00
CurrentMapDrawMetatileAt(x, y);
2019-12-17 03:24:44 -05:00
FadeInFromBlack();
2019-04-05 16:11:24 -05:00
CreateTask(EnterNewlyCreatedSecretBase_WaitFadeIn, 0);
2017-10-24 21:41:22 -04:00
}
2017-10-24 22:00:29 -04:00
2019-04-05 16:11:24 -05:00
static void Task_EnterNewlyCreatedSecretBase(u8 taskId)
2017-10-24 22:00:29 -04:00
{
if (!gPaletteFade.active)
{
2021-02-05 12:08:25 -05:00
s8 secretBaseGroup = SECRET_BASE_ID_TO_GROUP(sCurSecretBaseId);
2019-04-05 16:11:24 -05:00
SetWarpDestination(
gSaveBlock1Ptr->location.mapGroup,
gSaveBlock1Ptr->location.mapNum,
WARP_ID_NONE,
2021-02-05 12:08:25 -05:00
GET_BASE_COMPUTER_X(secretBaseGroup),
GET_BASE_COMPUTER_Y(secretBaseGroup));
WarpIntoMap();
2019-04-05 16:11:24 -05:00
gFieldCallback = EnterNewlyCreatedSecretBase_StartFadeIn;
2018-02-14 00:58:22 +01:00
SetMainCallback2(CB2_LoadMap);
2017-10-24 22:00:29 -04:00
DestroyTask(taskId);
}
}
2017-10-24 22:10:35 -04:00
2019-04-05 16:11:24 -05:00
void EnterNewlyCreatedSecretBase(void)
2017-10-24 22:10:35 -04:00
{
2019-04-05 16:11:24 -05:00
CreateTask(Task_EnterNewlyCreatedSecretBase, 0);
FadeScreen(FADE_TO_BLACK, 0);
2017-10-24 22:10:35 -04:00
}
2019-04-05 16:11:24 -05:00
bool8 CurMapIsSecretBase(void)
2017-10-24 22:10:35 -04:00
{
2019-04-05 16:11:24 -05:00
if (gSaveBlock1Ptr->location.mapGroup == MAP_GROUP(SECRET_BASE_RED_CAVE1)
&& (u8)gSaveBlock1Ptr->location.mapNum <= MAP_NUM(SECRET_BASE_SHRUB4))
2017-10-24 22:10:35 -04:00
return TRUE;
2019-04-05 16:11:24 -05:00
else
return FALSE;
2017-10-24 22:10:35 -04:00
}
2017-10-24 22:17:26 -04:00
2019-04-05 16:11:24 -05:00
void InitSecretBaseAppearance(bool8 hidePC)
2017-10-24 22:17:26 -04:00
{
2021-02-05 12:08:25 -05:00
u16 secretBaseIdx;
2019-04-05 16:11:24 -05:00
u16 x, y;
2017-10-24 22:17:26 -04:00
u8 *decorations;
u8 *decorPos;
2019-04-05 16:11:24 -05:00
if (CurMapIsSecretBase())
2017-10-24 22:17:26 -04:00
{
2021-02-05 12:08:25 -05:00
secretBaseIdx = VarGet(VAR_CURRENT_SECRET_BASE);
decorations = gSaveBlock1Ptr->secretBases[secretBaseIdx].decorations;
decorPos = gSaveBlock1Ptr->secretBases[secretBaseIdx].decorationPositions;
for (x = 0; x < DECOR_MAX_SECRET_BASE; x++)
2017-10-24 22:17:26 -04:00
{
2019-04-05 16:11:24 -05:00
if (decorations[x] > 0 && decorations[x] <= NUM_DECORATIONS && gDecorations[decorations[x]].permission != DECORPERM_SPRITE)
2021-10-09 12:12:18 -04:00
ShowDecorationOnMap((decorPos[x] >> 4) + MAP_OFFSET, (decorPos[x] & 0xF) + MAP_OFFSET, decorations[x]);
2017-10-24 22:17:26 -04:00
}
2019-04-05 16:11:24 -05:00
2021-02-05 12:08:25 -05:00
if (secretBaseIdx != 0)
2017-10-24 22:17:26 -04:00
{
2019-04-05 16:11:24 -05:00
// Another player's secret base. Change PC type to the "Register" PC.
FindMetatileIdMapCoords(&x, &y, METATILE_SecretBase_PC);
2022-01-19 10:15:32 -05:00
MapGridSetMetatileIdAt(x + MAP_OFFSET, y + MAP_OFFSET, METATILE_SecretBase_RegisterPC | MAPGRID_COLLISION_MASK);
2017-10-24 22:17:26 -04:00
}
2019-04-05 16:11:24 -05:00
else if (hidePC == TRUE && VarGet(VAR_SECRET_BASE_INITIALIZED) == 1)
2017-10-24 22:17:26 -04:00
{
2019-04-05 16:11:24 -05:00
// Change PC to regular ground tile.
FindMetatileIdMapCoords(&x, &y, METATILE_SecretBase_PC);
2022-01-19 10:15:32 -05:00
MapGridSetMetatileIdAt(x + MAP_OFFSET, y + MAP_OFFSET, METATILE_SecretBase_Ground | MAPGRID_COLLISION_MASK);
2017-10-24 22:17:26 -04:00
}
}
}
2017-10-24 22:34:52 -04:00
2019-04-05 16:11:24 -05:00
void InitSecretBaseDecorationSprites(void)
2017-10-24 22:34:52 -04:00
{
2019-04-05 16:11:24 -05:00
u8 i;
u8 *decorations;
u8 *decorationPositions;
u8 objectEventId;
2019-04-05 16:11:24 -05:00
u8 metatileBehavior;
2017-10-24 22:34:52 -04:00
u8 category;
u8 permission;
2019-04-05 16:11:24 -05:00
u8 numDecorations;
2017-10-24 22:34:52 -04:00
objectEventId = 0;
2019-04-05 16:11:24 -05:00
if (!CurMapIsSecretBase())
2017-10-24 22:34:52 -04:00
{
2020-02-23 15:46:22 +00:00
decorations = gSaveBlock1Ptr->playerRoomDecorations;
decorationPositions = gSaveBlock1Ptr->playerRoomDecorationPositions;
numDecorations = DECOR_MAX_PLAYERS_HOUSE;
2017-10-24 22:34:52 -04:00
}
else
{
2021-02-05 12:08:25 -05:00
u16 secretBaseIdx = VarGet(VAR_CURRENT_SECRET_BASE);
decorations = gSaveBlock1Ptr->secretBases[secretBaseIdx].decorations;
decorationPositions = gSaveBlock1Ptr->secretBases[secretBaseIdx].decorationPositions;
2020-02-23 15:46:22 +00:00
numDecorations = DECOR_MAX_SECRET_BASE;
2017-10-24 22:34:52 -04:00
}
2019-04-05 16:11:24 -05:00
for (i = 0; i < numDecorations; i++)
2017-10-24 22:34:52 -04:00
{
2019-04-05 16:11:24 -05:00
if (decorations[i] == DECOR_NONE)
continue;
2020-02-23 15:46:22 +00:00
2019-04-05 16:11:24 -05:00
permission = gDecorations[decorations[i]].permission;
category = gDecorations[decorations[i]].category;
if (permission == DECORPERM_SPRITE)
2017-10-24 22:34:52 -04:00
{
for (objectEventId = 0; objectEventId < gMapHeader.events->objectEventCount; objectEventId++)
2017-10-24 22:34:52 -04:00
{
if (gMapHeader.events->objectEvents[objectEventId].flagId == FLAG_DECORATION_1 + gSpecialVar_0x8004)
2019-04-05 16:11:24 -05:00
break;
}
if (objectEventId == gMapHeader.events->objectEventCount)
2019-04-05 16:11:24 -05:00
continue;
gSpecialVar_0x8006 = decorationPositions[i] >> 4;
gSpecialVar_0x8007 = decorationPositions[i] & 0xF;
2021-10-09 12:12:18 -04:00
metatileBehavior = MapGridGetMetatileBehaviorAt(gSpecialVar_0x8006 + MAP_OFFSET, gSpecialVar_0x8007 + MAP_OFFSET);
2019-04-05 16:11:24 -05:00
if (MetatileBehavior_HoldsSmallDecoration(metatileBehavior) == TRUE
|| MetatileBehavior_HoldsLargeDecoration(metatileBehavior) == TRUE)
{
gSpecialVar_Result = VAR_OBJ_GFX_ID_0 + (gMapHeader.events->objectEvents[objectEventId].graphicsId - OBJ_EVENT_GFX_VAR_0);
2019-04-05 16:11:24 -05:00
VarSet(gSpecialVar_Result, gDecorations[decorations[i]].tiles[0]);
gSpecialVar_Result = gMapHeader.events->objectEvents[objectEventId].localId;
2019-04-05 16:11:24 -05:00
FlagClear(FLAG_DECORATION_1 + gSpecialVar_0x8004);
TrySpawnObjectEvent(gSpecialVar_Result, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup);
TryMoveObjectEventToMapCoords(gSpecialVar_Result, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup, gSpecialVar_0x8006, gSpecialVar_0x8007);
TryOverrideObjectEventTemplateCoords(gSpecialVar_Result, gSaveBlock1Ptr->location.mapNum, gSaveBlock1Ptr->location.mapGroup);
2019-04-05 16:11:24 -05:00
if (CurMapIsSecretBase() == TRUE && VarGet(VAR_CURRENT_SECRET_BASE) != 0)
2017-10-24 22:34:52 -04:00
{
2019-04-05 16:11:24 -05:00
if (category == DECORCAT_DOLL)
2017-10-24 22:34:52 -04:00
{
2019-04-05 16:11:24 -05:00
OverrideSecretBaseDecorationSpriteScript(
gSpecialVar_Result,
gSaveBlock1Ptr->location.mapNum,
gSaveBlock1Ptr->location.mapGroup,
DECORCAT_DOLL);
2017-10-24 22:34:52 -04:00
}
2019-04-05 16:11:24 -05:00
else if (category == DECORCAT_CUSHION)
2017-10-24 22:34:52 -04:00
{
2019-04-05 16:11:24 -05:00
OverrideSecretBaseDecorationSpriteScript(
gSpecialVar_Result,
gSaveBlock1Ptr->location.mapNum,
gSaveBlock1Ptr->location.mapGroup,
DECORCAT_CUSHION);
2017-10-24 22:34:52 -04:00
}
}
2019-04-05 16:11:24 -05:00
gSpecialVar_0x8004++;
2017-10-24 22:34:52 -04:00
}
}
}
}
2017-10-24 22:47:12 -04:00
2019-04-05 16:11:24 -05:00
void HideSecretBaseDecorationSprites(void)
2017-10-24 22:47:12 -04:00
{
u8 objectEventId;
2019-04-05 16:11:24 -05:00
u16 flag;
2017-10-24 22:47:12 -04:00
for (objectEventId = 0; objectEventId < gMapHeader.events->objectEventCount; objectEventId++)
2017-10-24 22:47:12 -04:00
{
flag = gMapHeader.events->objectEvents[objectEventId].flagId;
2019-04-05 16:11:24 -05:00
if (flag >= FLAG_DECORATION_1 && flag <= FLAG_DECORATION_14)
2017-10-24 22:47:12 -04:00
{
RemoveObjectEventByLocalIdAndMap(
gMapHeader.events->objectEvents[objectEventId].localId,
2019-04-05 16:11:24 -05:00
gSaveBlock1Ptr->location.mapNum,
gSaveBlock1Ptr->location.mapGroup);
FlagSet(flag);
2017-10-24 22:47:12 -04:00
}
}
}
2017-10-25 08:22:36 -04:00
2019-04-05 16:11:24 -05:00
void SetSecretBaseOwnerGfxId(void)
2017-10-25 08:22:36 -04:00
{
2019-04-05 16:11:24 -05:00
VarSet(VAR_OBJ_GFX_ID_F, sSecretBaseOwnerGfxIds[GetSecretBaseOwnerType(VarGet(VAR_CURRENT_SECRET_BASE))]);
2017-10-25 08:22:36 -04:00
}
2019-04-05 16:11:24 -05:00
void SetCurSecretBaseIdFromPosition(const struct MapPosition *position, const struct MapEvents *events)
2017-10-25 08:22:36 -04:00
{
2019-04-05 16:11:24 -05:00
s16 i;
for (i = 0; i < events->bgEventCount; i++)
2017-10-25 08:22:36 -04:00
{
2021-10-09 12:12:18 -04:00
if (events->bgEvents[i].kind == BG_EVENT_SECRET_BASE
&& position->x == events->bgEvents[i].x + MAP_OFFSET
&& position->y == events->bgEvents[i].y + MAP_OFFSET)
2017-10-25 08:22:36 -04:00
{
2019-04-05 16:11:24 -05:00
sCurSecretBaseId = events->bgEvents[i].bgUnion.secretBaseId;
2017-10-25 08:22:36 -04:00
break;
}
}
}
2018-12-28 15:11:15 -06:00
void WarpIntoSecretBase(const struct MapPosition *position, const struct MapEvents *events)
2017-10-25 08:22:36 -04:00
{
2019-04-05 16:11:24 -05:00
SetCurSecretBaseIdFromPosition(position, events);
TrySetCurSecretBaseIndex();
ScriptContext_SetupScript(SecretBase_EventScript_Enter);
2017-10-25 08:22:36 -04:00
}
2017-10-25 08:34:41 -04:00
2019-04-05 16:11:24 -05:00
bool8 TrySetCurSecretBase(void)
2017-10-25 08:34:41 -04:00
{
2019-04-05 16:11:24 -05:00
SetCurSecretBaseId();
TrySetCurSecretBaseIndex();
2017-11-10 18:12:18 -06:00
if (gSpecialVar_Result == TRUE)
2017-10-25 08:34:41 -04:00
return FALSE;
2019-04-05 16:11:24 -05:00
2017-10-25 08:34:41 -04:00
return TRUE;
}
2019-04-05 16:11:24 -05:00
static void Task_WarpOutOfSecretBase(u8 taskId)
2017-10-25 08:34:41 -04:00
{
switch (gTasks[taskId].data[0])
{
2019-04-05 16:11:24 -05:00
case 0:
LockPlayerFieldControls();
2019-04-05 16:11:24 -05:00
gTasks[taskId].data[0] = 1;
break;
case 1:
if (!gPaletteFade.active)
gTasks[taskId].data[0] = 2;
break;
case 2:
SetWarpDestinationToDynamicWarp(WARP_ID_SECRET_BASE);
2019-04-05 16:11:24 -05:00
WarpIntoMap();
2019-12-17 03:24:44 -05:00
gFieldCallback = FieldCB_DefaultWarpExit;
2019-04-05 16:11:24 -05:00
SetMainCallback2(CB2_LoadMap);
UnlockPlayerFieldControls();
2019-04-05 16:11:24 -05:00
DestroyTask(taskId);
break;
2017-10-25 08:34:41 -04:00
}
}
2019-04-05 16:11:24 -05:00
static void WarpOutOfSecretBase(void)
2017-10-25 08:34:41 -04:00
{
2019-04-05 16:11:24 -05:00
CreateTask(Task_WarpOutOfSecretBase, 0);
FadeScreen(FADE_TO_BLACK, 0);
2017-10-25 08:34:41 -04:00
}
2019-04-05 16:11:24 -05:00
void IsCurSecretBaseOwnedByAnotherPlayer(void)
2017-10-25 08:34:41 -04:00
{
if (gSaveBlock1Ptr->secretBases[0].secretBaseId != sCurSecretBaseId)
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = TRUE;
2017-10-25 08:34:41 -04:00
else
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = FALSE;
2017-10-25 08:34:41 -04:00
}
2017-10-25 08:41:25 -04:00
2021-02-05 12:08:25 -05:00
static u8 *GetSecretBaseName(u8 *dest, u8 secretBaseIdx)
2017-10-25 08:41:25 -04:00
{
2021-02-05 12:08:25 -05:00
*StringCopyN(dest, gSaveBlock1Ptr->secretBases[secretBaseIdx].trainerName, GetNameLength(gSaveBlock1Ptr->secretBases[secretBaseIdx].trainerName)) = EOS;
ConvertInternationalString(dest, gSaveBlock1Ptr->secretBases[secretBaseIdx].language);
2017-10-25 08:41:25 -04:00
return StringAppend(dest, gText_ApostropheSBase);
}
2017-10-25 08:46:08 -04:00
u8 *GetSecretBaseMapName(u8 *dest)
{
2019-04-05 16:11:24 -05:00
return GetSecretBaseName(dest, VarGet(VAR_CURRENT_SECRET_BASE));
2017-10-25 08:46:08 -04:00
}
2019-04-05 16:11:24 -05:00
void CopyCurSecretBaseOwnerName_StrVar1(void)
2017-10-25 08:46:08 -04:00
{
2021-02-05 12:08:25 -05:00
u8 secretBaseIdx;
2019-04-05 16:11:24 -05:00
const u8 *name;
2017-10-25 08:46:08 -04:00
2021-02-05 12:08:25 -05:00
secretBaseIdx = VarGet(VAR_CURRENT_SECRET_BASE);
name = gSaveBlock1Ptr->secretBases[secretBaseIdx].trainerName;
2019-04-05 16:11:24 -05:00
*StringCopyN(gStringVar1, name, GetNameLength(name)) = EOS;
2021-02-05 12:08:25 -05:00
ConvertInternationalString(gStringVar1, gSaveBlock1Ptr->secretBases[secretBaseIdx].language);
2017-10-25 08:46:08 -04:00
}
2017-10-25 18:40:44 -04:00
2021-02-05 12:08:25 -05:00
static bool8 IsSecretBaseRegistered(u8 secretBaseIdx)
2017-10-25 18:40:44 -04:00
{
2021-02-05 12:08:25 -05:00
if (gSaveBlock1Ptr->secretBases[secretBaseIdx].registryStatus)
2017-10-25 18:40:44 -04:00
return TRUE;
2019-04-05 16:11:24 -05:00
2017-10-25 18:40:44 -04:00
return FALSE;
}
2019-04-05 16:11:24 -05:00
static u8 GetAverageEVs(struct Pokemon *pokemon)
2017-10-25 18:40:44 -04:00
{
u16 evTotal;
evTotal = GetMonData(pokemon, MON_DATA_HP_EV);
evTotal += GetMonData(pokemon, MON_DATA_ATK_EV);
evTotal += GetMonData(pokemon, MON_DATA_DEF_EV);
2017-11-01 23:37:57 -04:00
evTotal += GetMonData(pokemon, MON_DATA_SPEED_EV);
2017-10-25 18:40:44 -04:00
evTotal += GetMonData(pokemon, MON_DATA_SPATK_EV);
evTotal += GetMonData(pokemon, MON_DATA_SPDEF_EV);
return evTotal / 6;
}
2017-10-25 19:54:24 -04:00
2019-04-05 16:11:24 -05:00
void SetPlayerSecretBaseParty(void)
2017-10-25 19:54:24 -04:00
{
2019-04-05 16:11:24 -05:00
u16 i;
2017-11-01 23:59:48 -04:00
u16 moveIdx;
2019-04-05 16:11:24 -05:00
u16 partyId;
2018-02-07 13:49:33 +01:00
struct SecretBaseParty *party;
2017-10-25 19:54:24 -04:00
2019-04-05 16:11:24 -05:00
partyId = 0;
2018-02-07 13:49:33 +01:00
party = &gSaveBlock1Ptr->secretBases[0].party;
2019-04-05 16:11:24 -05:00
if (gSaveBlock1Ptr->secretBases[0].secretBaseId)
2017-10-25 19:54:24 -04:00
{
2019-04-05 16:11:24 -05:00
for (i = 0; i < PARTY_SIZE; i++)
2017-10-25 19:54:24 -04:00
{
2019-04-05 16:11:24 -05:00
for (moveIdx = 0; moveIdx < MAX_MON_MOVES; moveIdx++)
party->moves[i * MAX_MON_MOVES + moveIdx] = MOVE_NONE;
party->species[i] = SPECIES_NONE;
party->heldItems[i] = ITEM_NONE;
party->levels[i] = 0;
party->personality[i] = 0;
party->EVs[i] = 0;
if (GetMonData(&gPlayerParty[i], MON_DATA_SPECIES) != SPECIES_NONE
&& !GetMonData(&gPlayerParty[i], MON_DATA_IS_EGG))
2017-10-25 19:54:24 -04:00
{
2019-04-05 16:11:24 -05:00
for (moveIdx = 0; moveIdx < MAX_MON_MOVES; moveIdx++)
party->moves[partyId * MAX_MON_MOVES + moveIdx] = GetMonData(&gPlayerParty[i], MON_DATA_MOVE1 + moveIdx);
party->species[partyId] = GetMonData(&gPlayerParty[i], MON_DATA_SPECIES);
party->heldItems[partyId] = GetMonData(&gPlayerParty[i], MON_DATA_HELD_ITEM);
party->levels[partyId] = GetMonData(&gPlayerParty[i], MON_DATA_LEVEL);
party->personality[partyId] = GetMonData(&gPlayerParty[i], MON_DATA_PERSONALITY);
party->EVs[partyId] = GetAverageEVs(&gPlayerParty[i]);
partyId++;
2017-10-25 19:54:24 -04:00
}
}
}
}
2017-10-25 20:03:55 -04:00
2019-04-05 16:11:24 -05:00
void ClearAndLeaveSecretBase(void)
2017-10-25 20:03:55 -04:00
{
2019-04-05 16:11:24 -05:00
u16 temp = gSaveBlock1Ptr->secretBases[0].numSecretBasesReceived;
ClearSecretBase(&gSaveBlock1Ptr->secretBases[0]);
2019-04-05 16:11:24 -05:00
gSaveBlock1Ptr->secretBases[0].numSecretBasesReceived = temp;
WarpOutOfSecretBase();
2017-10-25 20:03:55 -04:00
}
2017-10-26 08:40:08 -04:00
2019-04-05 16:11:24 -05:00
void MoveOutOfSecretBase(void)
2017-10-26 08:40:08 -04:00
{
IncrementGameStat(GAME_STAT_MOVED_SECRET_BASE);
2019-04-05 16:11:24 -05:00
ClearAndLeaveSecretBase();
2017-10-26 08:40:08 -04:00
}
2019-04-05 16:11:24 -05:00
static void ClosePlayerSecretBaseEntrance(void)
2017-10-26 08:40:08 -04:00
{
u16 i;
u16 j;
2019-04-05 16:11:24 -05:00
s16 metatileId;
const struct MapEvents *events = gMapHeader.events;
2017-10-26 08:40:08 -04:00
2019-04-05 16:11:24 -05:00
for (i = 0; i < events->bgEventCount; i++)
2017-10-26 08:40:08 -04:00
{
2019-04-05 16:11:24 -05:00
if (events->bgEvents[i].kind == BG_EVENT_SECRET_BASE
&& gSaveBlock1Ptr->secretBases[0].secretBaseId == events->bgEvents[i].bgUnion.secretBaseId)
2017-10-26 08:40:08 -04:00
{
2021-10-09 12:12:18 -04:00
metatileId = MapGridGetMetatileIdAt(events->bgEvents[i].x + MAP_OFFSET, events->bgEvents[i].y + MAP_OFFSET);
2021-02-05 12:08:25 -05:00
for (j = 0; j < ARRAY_COUNT(sSecretBaseEntranceMetatiles); j++)
2017-10-26 08:40:08 -04:00
{
2019-04-05 16:11:24 -05:00
if (sSecretBaseEntranceMetatiles[j].openMetatileId == metatileId)
2017-10-26 08:40:08 -04:00
{
2021-10-09 12:12:18 -04:00
MapGridSetMetatileIdAt(events->bgEvents[i].x + MAP_OFFSET,
events->bgEvents[i].y + MAP_OFFSET,
2022-01-19 10:15:32 -05:00
sSecretBaseEntranceMetatiles[j].closedMetatileId | MAPGRID_COLLISION_MASK);
2017-10-26 08:40:08 -04:00
break;
}
}
2019-04-05 16:11:24 -05:00
2017-10-26 08:40:08 -04:00
DrawWholeMapView();
break;
}
}
}
2017-10-26 08:45:41 -04:00
2019-04-05 16:11:24 -05:00
// When the player moves to a new secret base by interacting with a new secret base
// entrance in the overworld.
void MoveOutOfSecretBaseFromOutside(void)
2017-10-26 08:45:41 -04:00
{
2019-04-05 16:11:24 -05:00
u16 temp;
2017-10-26 08:45:41 -04:00
2019-04-05 16:11:24 -05:00
ClosePlayerSecretBaseEntrance();
2017-10-26 08:45:41 -04:00
IncrementGameStat(GAME_STAT_MOVED_SECRET_BASE);
2019-04-05 16:11:24 -05:00
temp = gSaveBlock1Ptr->secretBases[0].numSecretBasesReceived;
ClearSecretBase(&gSaveBlock1Ptr->secretBases[0]);
2019-04-05 16:11:24 -05:00
gSaveBlock1Ptr->secretBases[0].numSecretBasesReceived = temp;
2017-10-26 08:45:41 -04:00
}
2019-04-05 16:11:24 -05:00
static u8 GetNumRegisteredSecretBases(void)
2017-10-26 08:45:41 -04:00
{
s16 i;
2019-04-05 16:11:24 -05:00
u8 count = 0;
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-26 08:45:41 -04:00
{
2019-04-05 16:11:24 -05:00
if (IsSecretBaseRegistered(i) == TRUE)
count++;
2017-10-26 08:45:41 -04:00
}
2019-04-05 16:11:24 -05:00
return count;
2017-10-26 08:45:41 -04:00
}
2017-10-26 21:39:34 -04:00
2019-04-05 16:11:24 -05:00
void GetCurSecretBaseRegistrationValidity(void)
2017-10-26 21:39:34 -04:00
{
2019-04-05 16:11:24 -05:00
if (IsSecretBaseRegistered(VarGet(VAR_CURRENT_SECRET_BASE)) == TRUE)
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = 1;
else if (GetNumRegisteredSecretBases() >= 10)
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = 2;
2017-10-26 21:39:34 -04:00
else
2017-11-10 18:12:18 -06:00
gSpecialVar_Result = 0;
2017-10-26 21:39:34 -04:00
}
2019-04-05 16:11:24 -05:00
void ToggleCurSecretBaseRegistry(void)
2017-10-26 21:39:34 -04:00
{
2019-04-05 16:11:24 -05:00
gSaveBlock1Ptr->secretBases[VarGet(VAR_CURRENT_SECRET_BASE)].registryStatus ^= 1;
FlagSet(FLAG_SECRET_BASE_REGISTRY_ENABLED);
2017-10-26 21:39:34 -04:00
}
2019-04-05 16:11:24 -05:00
void ShowSecretBaseDecorationMenu(void)
2017-10-26 21:39:34 -04:00
{
CreateTask(DoSecretBaseDecorationMenu, 0);
2017-10-26 21:39:34 -04:00
}
2019-04-05 16:11:24 -05:00
void ShowSecretBaseRegistryMenu(void)
2017-10-26 21:39:34 -04:00
{
2019-04-05 16:11:24 -05:00
CreateTask(Task_ShowSecretBaseRegistryMenu, 0);
2017-10-26 21:39:34 -04:00
}
2021-02-05 12:08:25 -05:00
#define tNumBases data[0]
#define tSelectedRow data[1]
#define tScrollOffset data[2]
#define tMaxShownItems data[3]
#define tSelectedBaseId data[4]
#define tListTaskId data[5]
#define tMainWindowId data[6]
#define tActionWindowId data[7]
#define tArrowTaskId data[8]
2019-04-05 16:11:24 -05:00
static void Task_ShowSecretBaseRegistryMenu(u8 taskId)
2017-10-26 21:39:34 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
LockPlayerFieldControls();
2021-02-05 12:08:25 -05:00
tNumBases = GetNumRegisteredSecretBases();
if (tNumBases != 0)
2017-10-26 21:39:34 -04:00
{
2021-02-05 12:08:25 -05:00
tSelectedRow = 0;
tScrollOffset = 0;
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, FALSE);
2022-09-16 20:34:22 -03:00
sRegistryMenu = AllocZeroed(sizeof(*sRegistryMenu));
2021-02-05 12:08:25 -05:00
tMainWindowId = AddWindow(&sRegistryWindowTemplates[0]);
2019-04-05 16:11:24 -05:00
BuildRegistryMenuItems(taskId);
FinalizeRegistryMenu(taskId);
gTasks[taskId].func = HandleRegistryMenuInput;
2017-10-26 21:39:34 -04:00
}
else
{
2019-09-05 16:16:21 -04:00
DisplayItemMessageOnField(taskId, gText_NoRegistry, GoToSecretBasePCRegisterMenu);
2017-10-26 21:39:34 -04:00
}
}
2019-04-05 16:11:24 -05:00
static void BuildRegistryMenuItems(u8 taskId)
2017-10-26 21:39:34 -04:00
{
s16 *data;
u8 i;
2017-11-01 23:49:01 -04:00
u8 count;
2017-10-26 21:39:34 -04:00
data = gTasks[taskId].data;
2017-11-01 23:49:01 -04:00
count = 0;
2019-04-05 16:11:24 -05:00
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-26 21:39:34 -04:00
{
2019-04-05 16:11:24 -05:00
if (IsSecretBaseRegistered(i))
2017-10-26 21:39:34 -04:00
{
2019-04-05 16:11:24 -05:00
GetSecretBaseName(sRegistryMenu->names[count], i);
sRegistryMenu->items[count].name = sRegistryMenu->names[count];
sRegistryMenu->items[count].id = i;
count++;
2017-10-26 21:39:34 -04:00
}
}
2019-04-05 16:11:24 -05:00
sRegistryMenu->items[count].name = gText_Cancel;
2022-11-09 15:53:47 -05:00
sRegistryMenu->items[count].id = LIST_CANCEL;
2021-02-05 12:08:25 -05:00
tNumBases = count + 1;
if (tNumBases < 8)
tMaxShownItems = tNumBases;
2017-10-26 21:39:34 -04:00
else
2021-02-05 12:08:25 -05:00
tMaxShownItems = 8;
2019-04-05 16:11:24 -05:00
gMultiuseListMenuTemplate = sRegistryListMenuTemplate;
2021-02-05 12:08:25 -05:00
gMultiuseListMenuTemplate.windowId = tMainWindowId;
gMultiuseListMenuTemplate.totalItems = tNumBases;
2019-04-05 16:11:24 -05:00
gMultiuseListMenuTemplate.items = sRegistryMenu->items;
2021-02-05 12:08:25 -05:00
gMultiuseListMenuTemplate.maxShowed = tMaxShownItems;
2017-10-26 21:39:34 -04:00
}
2017-10-26 21:53:00 -04:00
2019-07-18 20:46:00 -04:00
static void RegistryMenu_OnCursorMove(s32 unused, bool8 flag, struct ListMenu *menu)
2017-10-26 21:53:00 -04:00
{
if (flag != TRUE)
PlaySE(SE_SELECT);
}
2019-04-05 16:11:24 -05:00
static void FinalizeRegistryMenu(u8 taskId)
2017-10-26 21:53:00 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
2022-07-25 14:59:14 -04:00
SetStandardWindowBorderStyle(tMainWindowId, FALSE);
2021-02-05 12:08:25 -05:00
tListTaskId = ListMenuInit(&gMultiuseListMenuTemplate, tScrollOffset, tSelectedRow);
2019-04-05 16:11:24 -05:00
AddRegistryMenuScrollArrows(taskId);
2020-05-14 01:37:09 -07:00
ScheduleBgCopyTilemapToVram(0);
2017-10-26 21:53:00 -04:00
}
2019-04-05 16:11:24 -05:00
static void AddRegistryMenuScrollArrows(u8 taskId)
2017-10-26 21:53:00 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
2021-02-05 12:08:25 -05:00
tArrowTaskId = AddScrollIndicatorArrowPairParameterized(SCROLL_ARROW_UP, 188, 12, 148, tNumBases - tMaxShownItems, 0x13f8, 0x13f8, &tScrollOffset);
2017-10-26 21:53:00 -04:00
}
2017-10-26 21:59:54 -04:00
2019-04-05 16:11:24 -05:00
static void HandleRegistryMenuInput(u8 taskId)
2017-10-26 21:59:54 -04:00
{
2021-02-05 12:08:25 -05:00
s16 *data = gTasks[taskId].data;
s32 input = ListMenu_ProcessInput(tListTaskId);
ListMenuGetScrollAndRow(tListTaskId, &tScrollOffset, &tSelectedRow);
2017-10-26 21:59:54 -04:00
switch (input)
{
2019-04-05 16:11:24 -05:00
case LIST_NOTHING_CHOSEN:
break;
case LIST_CANCEL:
PlaySE(SE_SELECT);
2021-02-05 12:08:25 -05:00
DestroyListMenuTask(tListTaskId, NULL, NULL);
RemoveScrollIndicatorArrowPair(tArrowTaskId);
2022-07-25 14:59:14 -04:00
ClearStdWindowAndFrame(tMainWindowId, FALSE);
2021-02-05 12:08:25 -05:00
ClearWindowTilemap(tMainWindowId);
RemoveWindow(tMainWindowId);
2020-05-14 01:37:09 -07:00
ScheduleBgCopyTilemapToVram(0);
2022-09-16 20:36:44 -03:00
Free(sRegistryMenu);
2019-09-05 16:16:21 -04:00
GoToSecretBasePCRegisterMenu(taskId);
2019-04-05 16:11:24 -05:00
break;
default:
PlaySE(SE_SELECT);
2021-02-05 12:08:25 -05:00
tSelectedBaseId = input;
2019-04-05 16:11:24 -05:00
ShowRegistryMenuActions(taskId);
break;
2017-10-26 21:59:54 -04:00
}
}
2017-10-26 22:10:38 -04:00
2019-04-05 16:11:24 -05:00
static void ShowRegistryMenuActions(u8 taskId)
2017-10-26 22:10:38 -04:00
{
struct WindowTemplate template;
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
2021-02-05 12:08:25 -05:00
RemoveScrollIndicatorArrowPair(tArrowTaskId);
2019-04-05 16:11:24 -05:00
template = sRegistryWindowTemplates[1];
template.width = GetMaxWidthInMenuTable(sRegistryMenuActions, 2);
2021-02-05 12:08:25 -05:00
tActionWindowId = AddWindow(&template);
2022-07-25 14:59:14 -04:00
SetStandardWindowBorderStyle(tActionWindowId, FALSE);
2021-02-05 12:08:25 -05:00
PrintMenuTable(tActionWindowId, ARRAY_COUNT(sRegistryMenuActions), sRegistryMenuActions);
2022-02-27 13:47:50 -05:00
InitMenuInUpperLeftCornerNormal(tActionWindowId, ARRAY_COUNT(sRegistryMenuActions), 0);
2020-05-14 01:37:09 -07:00
ScheduleBgCopyTilemapToVram(0);
2019-04-05 16:11:24 -05:00
gTasks[taskId].func = HandleRegistryMenuActionsInput;
2017-10-26 22:10:38 -04:00
}
2019-04-05 16:11:24 -05:00
static void HandleRegistryMenuActionsInput(u8 taskId)
2017-10-26 22:10:38 -04:00
{
2019-04-05 16:11:24 -05:00
s8 input = Menu_ProcessInputNoWrap();
2017-10-26 22:10:38 -04:00
switch (input)
{
2019-04-05 16:11:24 -05:00
case MENU_B_PRESSED:
PlaySE(SE_SELECT);
ReturnToMainRegistryMenu(taskId);
break;
case MENU_NOTHING_CHOSEN:
break;
default:
PlaySE(SE_SELECT);
sRegistryMenuActions[input].func.void_u8(taskId);
break;
2017-10-26 22:10:38 -04:00
}
}
2017-10-26 22:21:20 -04:00
2019-04-05 16:11:24 -05:00
static void ShowRegistryMenuDeleteConfirmation(u8 taskId)
2017-10-26 22:21:20 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
2021-02-05 12:08:25 -05:00
ClearStdWindowAndFrame(tMainWindowId, FALSE);
ClearStdWindowAndFrame(tActionWindowId, FALSE);
ClearWindowTilemap(tMainWindowId);
ClearWindowTilemap(tActionWindowId);
RemoveWindow(tActionWindowId);
2020-05-14 01:37:09 -07:00
ScheduleBgCopyTilemapToVram(0);
2021-02-05 12:08:25 -05:00
GetSecretBaseName(gStringVar1, tSelectedBaseId);
2017-10-26 22:21:20 -04:00
StringExpandPlaceholders(gStringVar4, gText_OkayToDeleteFromRegistry);
2019-04-05 16:11:24 -05:00
DisplayItemMessageOnField(taskId, gStringVar4, ShowRegistryMenuDeleteYesNo);
2017-10-26 22:21:20 -04:00
}
2019-04-05 16:11:24 -05:00
static void ShowRegistryMenuDeleteYesNo(u8 taskId)
2017-10-26 22:21:20 -04:00
{
2019-02-26 22:30:40 -05:00
DisplayYesNoMenuDefaultYes();
2019-04-05 16:11:24 -05:00
DoYesNoFuncWithChoice(taskId, &sDeleteRegistryYesNoFuncs);
2017-10-26 22:21:20 -04:00
}
2019-04-05 16:11:24 -05:00
void DeleteRegistry_Yes_Callback(u8 taskId)
2017-10-26 22:21:20 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, FALSE);
2021-02-05 12:08:25 -05:00
DestroyListMenuTask(tListTaskId, &tScrollOffset, &tSelectedRow);
gSaveBlock1Ptr->secretBases[tSelectedBaseId].registryStatus = UNREGISTERED;
2019-04-05 16:11:24 -05:00
BuildRegistryMenuItems(taskId);
2021-04-25 12:07:08 -04:00
SetCursorWithinListBounds(&tScrollOffset, &tSelectedRow, tMaxShownItems, tNumBases);
2019-04-05 16:11:24 -05:00
FinalizeRegistryMenu(taskId);
gTasks[taskId].func = HandleRegistryMenuInput;
2017-10-26 22:21:20 -04:00
}
2017-10-26 22:37:16 -04:00
2019-04-05 16:11:24 -05:00
static void DeleteRegistry_Yes(u8 taskId)
2017-10-26 22:37:16 -04:00
{
2019-04-05 16:11:24 -05:00
DisplayItemMessageOnField(taskId, gText_RegisteredDataDeleted, DeleteRegistry_Yes_Callback);
2017-10-26 22:37:16 -04:00
}
2019-04-05 16:11:24 -05:00
static void DeleteRegistry_No(u8 taskId)
2017-10-26 22:37:16 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, FALSE);
2021-02-05 12:08:25 -05:00
DestroyListMenuTask(tListTaskId, &tScrollOffset, &tSelectedRow);
2019-04-05 16:11:24 -05:00
FinalizeRegistryMenu(taskId);
gTasks[taskId].func = HandleRegistryMenuInput;
2017-10-26 22:37:16 -04:00
}
2019-04-05 16:11:24 -05:00
static void ReturnToMainRegistryMenu(u8 taskId)
2017-10-26 22:37:16 -04:00
{
2019-04-05 16:11:24 -05:00
s16 *data = gTasks[taskId].data;
AddRegistryMenuScrollArrows(taskId);
2022-07-25 14:59:14 -04:00
ClearStdWindowAndFrame(tActionWindowId, FALSE);
2021-02-05 12:08:25 -05:00
ClearWindowTilemap(tActionWindowId);
RemoveWindow(tActionWindowId);
2020-05-14 01:37:09 -07:00
ScheduleBgCopyTilemapToVram(0);
2019-04-05 16:11:24 -05:00
gTasks[taskId].func = HandleRegistryMenuInput;
2017-10-26 22:37:16 -04:00
}
2019-09-05 16:16:21 -04:00
static void GoToSecretBasePCRegisterMenu(u8 taskId)
2017-10-26 22:37:16 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) == 0)
ScriptContext_SetupScript(SecretBase_EventScript_PCCancel);
2017-10-26 22:37:16 -04:00
else
ScriptContext_SetupScript(SecretBase_EventScript_ShowRegisterMenu);
2019-04-05 16:11:24 -05:00
2017-10-26 22:37:16 -04:00
DestroyTask(taskId);
}
2021-02-05 12:08:25 -05:00
#undef tNumBases
#undef tSelectedRow
#undef tScrollOffset
#undef tMaxShownItems
#undef tSelectedBaseId
#undef tListTaskId
#undef tMainWindowId
#undef tActionWindowId
#undef tArrowTaskId
static u8 GetSecretBaseOwnerType(u8 secretBaseIdx)
2017-10-26 22:37:16 -04:00
{
2021-02-05 12:08:25 -05:00
return (gSaveBlock1Ptr->secretBases[secretBaseIdx].trainerId[0] % 5)
+ (gSaveBlock1Ptr->secretBases[secretBaseIdx].gender * 5);
2017-10-26 22:37:16 -04:00
}
2017-10-26 22:49:51 -04:00
2017-12-01 21:25:13 +01:00
const u8 *GetSecretBaseTrainerLoseText(void)
2017-10-26 22:49:51 -04:00
{
2019-04-05 16:11:24 -05:00
u8 ownerType = GetSecretBaseOwnerType(VarGet(VAR_CURRENT_SECRET_BASE));
if (ownerType == 0)
return SecretBase_Text_Trainer0Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 1)
return SecretBase_Text_Trainer1Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 2)
return SecretBase_Text_Trainer2Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 3)
return SecretBase_Text_Trainer3Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 4)
return SecretBase_Text_Trainer4Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 5)
return SecretBase_Text_Trainer5Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 6)
return SecretBase_Text_Trainer6Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 7)
return SecretBase_Text_Trainer7Defeated;
2019-04-05 16:11:24 -05:00
else if (ownerType == 8)
return SecretBase_Text_Trainer8Defeated;
2019-04-05 16:11:24 -05:00
else
return SecretBase_Text_Trainer9Defeated;
2017-10-26 22:49:51 -04:00
}
2019-04-05 16:11:24 -05:00
void PrepSecretBaseBattleFlags(void)
2017-10-26 22:49:51 -04:00
{
2020-06-17 17:48:20 -04:00
TryGainNewFanFromCounter(FANCOUNTER_BATTLED_AT_BASE);
2018-10-21 20:13:12 +02:00
gTrainerBattleOpponent_A = TRAINER_SECRET_BASE;
2017-10-26 22:49:51 -04:00
gBattleTypeFlags = BATTLE_TYPE_TRAINER | BATTLE_TYPE_SECRET_BASE;
}
void SetBattledOwnerFromResult(void)
2017-10-26 22:49:51 -04:00
{
2019-04-05 16:11:24 -05:00
gSaveBlock1Ptr->secretBases[VarGet(VAR_CURRENT_SECRET_BASE)].battledOwnerToday = gSpecialVar_Result;
2017-10-26 22:49:51 -04:00
}
2017-10-26 22:54:11 -04:00
void GetSecretBaseOwnerAndState(void)
2017-10-26 22:54:11 -04:00
{
2021-02-05 12:08:25 -05:00
u16 secretBaseIdx;
2017-10-26 22:54:11 -04:00
u8 i;
2021-02-05 12:08:25 -05:00
secretBaseIdx = VarGet(VAR_CURRENT_SECRET_BASE);
2019-01-05 20:12:50 +00:00
if (!FlagGet(FLAG_DAILY_SECRET_BASE))
2017-10-26 22:54:11 -04:00
{
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
gSaveBlock1Ptr->secretBases[i].battledOwnerToday = FALSE;
2019-01-05 20:12:50 +00:00
FlagSet(FLAG_DAILY_SECRET_BASE);
2017-10-26 22:54:11 -04:00
}
2021-02-05 12:08:25 -05:00
gSpecialVar_0x8004 = GetSecretBaseOwnerType(secretBaseIdx);
gSpecialVar_Result = gSaveBlock1Ptr->secretBases[secretBaseIdx].battledOwnerToday;
2017-10-26 22:54:11 -04:00
}
2017-10-26 23:35:41 -04:00
2021-02-05 12:08:25 -05:00
#define tStepCb data[0] // See Task_RunPerStepCallback
#define tState data[1]
#define tPlayerX data[2]
#define tPlayerY data[3]
#define tFldEff data[4]
2018-12-31 12:22:27 -06:00
void SecretBasePerStepCallback(u8 taskId)
2017-10-26 23:35:41 -04:00
{
s16 x;
s16 y;
u8 behavior;
u16 tileId;
s16 *data;
data = gTasks[taskId].data;
2021-02-05 12:08:25 -05:00
switch (tState)
2017-10-26 23:35:41 -04:00
{
2019-04-05 16:11:24 -05:00
case 0:
2021-02-05 12:08:25 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE))
2019-04-05 16:11:24 -05:00
sInFriendSecretBase = TRUE;
else
sInFriendSecretBase = FALSE;
2021-02-05 12:08:25 -05:00
PlayerGetDestCoords(&tPlayerX, &tPlayerY);
tState = 1;
2019-04-05 16:11:24 -05:00
break;
case 1:
2021-11-16 21:34:47 -05:00
// End if player hasn't moved
2019-04-05 16:11:24 -05:00
PlayerGetDestCoords(&x, &y);
2021-11-16 21:34:47 -05:00
if (x == tPlayerX && y == tPlayerY)
return;
tPlayerX = x;
tPlayerY = y;
VarSet(VAR_SECRET_BASE_STEP_COUNTER, VarGet(VAR_SECRET_BASE_STEP_COUNTER) + 1);
behavior = MapGridGetMetatileBehaviorAt(x, y);
tileId = MapGridGetMetatileIdAt(x, y);
if (tileId == METATILE_SecretBase_SolidBoard_Top || tileId == METATILE_SecretBase_SolidBoard_Bottom)
2019-04-05 16:11:24 -05:00
{
2021-11-16 21:34:47 -05:00
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_SOLID_BOARD);
}
else if (tileId == METATILE_SecretBase_SmallChair
|| tileId == METATILE_SecretBase_PokemonChair
|| tileId == METATILE_SecretBase_HeavyChair
|| tileId == METATILE_SecretBase_PrettyChair
|| tileId == METATILE_SecretBase_ComfortChair
|| tileId == METATILE_SecretBase_RaggedChair
|| tileId == METATILE_SecretBase_BrickChair
|| tileId == METATILE_SecretBase_CampChair
|| tileId == METATILE_SecretBase_HardChair)
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_CHAIR);
}
else if (tileId == METATILE_SecretBase_RedTent_DoorTop
|| tileId == METATILE_SecretBase_RedTent_Door
|| tileId == METATILE_SecretBase_BlueTent_DoorTop
|| tileId == METATILE_SecretBase_BlueTent_Door)
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_TENT);
}
else if ((behavior == MB_IMPASSABLE_NORTHEAST && tileId == METATILE_SecretBase_Stand_CornerRight)
|| (behavior == MB_IMPASSABLE_NORTHWEST && MapGridGetMetatileIdAt(x, y) == METATILE_SecretBase_Stand_CornerLeft))
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_STAND);
}
else if (behavior == MB_IMPASSABLE_WEST_AND_EAST && tileId == METATILE_SecretBase_Slide_StairLanding)
{
if (sInFriendSecretBase == TRUE)
2019-04-05 16:11:24 -05:00
{
2021-11-16 21:34:47 -05:00
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) ^ SECRET_BASE_USED_SLIDE);
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_DECLINED_SLIDE);
2019-04-05 16:11:24 -05:00
}
2021-11-16 21:34:47 -05:00
}
else if (behavior == MB_SLIDE_SOUTH && tileId == METATILE_SecretBase_Slide_SlideTop)
{
if (sInFriendSecretBase == TRUE)
2019-04-05 16:11:24 -05:00
{
2021-11-16 21:34:47 -05:00
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_SLIDE);
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) ^ SECRET_BASE_DECLINED_SLIDE);
2019-04-05 16:11:24 -05:00
}
2021-11-16 21:34:47 -05:00
}
else if (MetatileBehavior_IsSecretBaseGlitterMat(behavior) == TRUE)
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_GLITTER_MAT);
}
else if (MetatileBehavior_IsSecretBaseBalloon(behavior) == TRUE)
{
PopSecretBaseBalloon(MapGridGetMetatileIdAt(x, y), x, y);
if (sInFriendSecretBase == TRUE)
2019-04-05 16:11:24 -05:00
{
2021-11-16 21:34:47 -05:00
switch ((int)MapGridGetMetatileIdAt(x, y))
2017-10-26 23:35:41 -04:00
{
2021-11-16 21:34:47 -05:00
case METATILE_SecretBase_RedBalloon:
case METATILE_SecretBase_BlueBalloon:
case METATILE_SecretBase_YellowBalloon:
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_BALLOON);
break;
case METATILE_SecretBase_MudBall:
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_MUD_BALL);
break;
2017-10-26 23:35:41 -04:00
}
}
2021-11-16 21:34:47 -05:00
}
else if (MetatileBehavior_IsSecretBaseBreakableDoor(behavior) == TRUE)
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_BREAKABLE_DOOR);
2019-04-05 16:11:24 -05:00
2021-11-16 21:34:47 -05:00
ShatterSecretBaseBreakableDoor(x, y);
}
else if (MetatileBehavior_IsSecretBaseSoundMat(behavior) == TRUE){
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_NOTE_MAT);
}
else if (MetatileBehavior_IsSecretBaseJumpMat(behavior) == TRUE)
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_JUMP_MAT);
}
else if (MetatileBehavior_IsSecretBaseSpinMat(behavior) == TRUE)
{
if (sInFriendSecretBase == TRUE)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_SPIN_MAT);
2019-04-05 16:11:24 -05:00
}
break;
case 2:
2021-02-05 12:08:25 -05:00
// This state is never reached, and tFldEff is never set
if (!FieldEffectActiveListContains(tFldEff))
tState = 1;
2019-04-05 16:11:24 -05:00
break;
2017-10-26 23:35:41 -04:00
}
}
2017-10-27 08:29:58 -04:00
2021-02-05 12:08:25 -05:00
#undef tStepCb
#undef tState
#undef tPlayerX
#undef tPlayerY
#undef tFldEff
static void SaveSecretBase(u8 secretBaseIdx, struct SecretBase *secretBase, u32 version, u32 language)
2017-10-27 08:29:58 -04:00
{
2017-11-01 23:49:01 -04:00
int stringLength;
2017-10-27 08:29:58 -04:00
u8 *name;
2021-02-05 12:08:25 -05:00
gSaveBlock1Ptr->secretBases[secretBaseIdx] = *secretBase;
gSaveBlock1Ptr->secretBases[secretBaseIdx].registryStatus = NEW;
2017-10-27 08:29:58 -04:00
if (version == VERSION_SAPPHIRE || version == VERSION_RUBY)
2021-02-05 12:08:25 -05:00
gSaveBlock1Ptr->secretBases[secretBaseIdx].language = GAME_LANGUAGE;
2019-04-05 16:11:24 -05:00
2017-10-27 08:29:58 -04:00
if (version == VERSION_EMERALD && language == LANGUAGE_JAPANESE)
{
2021-02-05 12:08:25 -05:00
name = gSaveBlock1Ptr->secretBases[secretBaseIdx].trainerName;
for (stringLength = 0; stringLength < PLAYER_NAME_LENGTH; stringLength++)
2017-10-27 08:29:58 -04:00
{
2017-11-01 23:49:01 -04:00
if (name[stringLength] == EOS)
2017-10-27 08:29:58 -04:00
break;
}
2019-04-05 16:11:24 -05:00
2017-11-01 23:49:01 -04:00
if (stringLength > 5)
2021-02-05 12:08:25 -05:00
gSaveBlock1Ptr->secretBases[secretBaseIdx].language = GAME_LANGUAGE;
2017-10-27 08:29:58 -04:00
}
}
2017-10-27 08:38:54 -04:00
2019-04-05 16:11:24 -05:00
static bool8 SecretBasesHaveSameTrainerId(struct SecretBase *secretBase1, struct SecretBase *secretBase2)
2017-10-27 08:38:54 -04:00
{
u8 i;
2019-09-08 11:53:48 -04:00
for (i = 0; i < TRAINER_ID_LENGTH; i++)
2017-10-27 08:38:54 -04:00
{
2019-04-05 16:11:24 -05:00
if (secretBase1->trainerId[i] != secretBase2->trainerId[i])
2017-10-27 08:38:54 -04:00
return FALSE;
}
2019-04-05 16:11:24 -05:00
2017-10-27 08:38:54 -04:00
return TRUE;
}
2019-04-05 16:11:24 -05:00
static bool8 SecretBasesHaveSameTrainerName(struct SecretBase *sbr1, struct SecretBase *sbr2)
2017-10-27 08:38:54 -04:00
{
u8 i;
2018-09-01 22:03:21 +02:00
for (i = 0; i < PLAYER_NAME_LENGTH && (sbr1->trainerName[i] != EOS || sbr2->trainerName[i] != EOS); i++)
2017-10-27 08:38:54 -04:00
{
if (sbr1->trainerName[i] != sbr2->trainerName[i])
return FALSE;
}
2019-04-05 16:11:24 -05:00
2017-10-27 08:38:54 -04:00
return TRUE;
}
2019-04-05 16:11:24 -05:00
static bool8 SecretBasesBelongToSamePlayer(struct SecretBase *secretBase1, struct SecretBase *secretBase2)
2017-10-27 08:38:54 -04:00
{
2019-04-05 16:11:24 -05:00
if (secretBase1->gender == secretBase2->gender
&& SecretBasesHaveSameTrainerId(secretBase1, secretBase2)
&& SecretBasesHaveSameTrainerName(secretBase1, secretBase2))
2017-10-27 08:38:54 -04:00
{
return TRUE;
}
2019-04-05 16:11:24 -05:00
2017-10-27 08:38:54 -04:00
return FALSE;
}
2019-04-05 16:11:24 -05:00
static s16 GetSecretBaseIndexFromId(u8 secretBaseId)
2017-10-27 08:38:54 -04:00
{
s16 i;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-27 08:38:54 -04:00
{
2019-04-05 16:11:24 -05:00
if (gSaveBlock1Ptr->secretBases[i].secretBaseId == secretBaseId)
2017-10-27 08:38:54 -04:00
return i;
}
2019-04-05 16:11:24 -05:00
2017-10-27 08:38:54 -04:00
return -1;
}
2017-10-27 08:43:10 -04:00
2019-04-05 16:11:24 -05:00
static u8 FindAvailableSecretBaseIndex(void)
2017-10-27 08:43:10 -04:00
{
s16 i;
2019-04-05 16:11:24 -05:00
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-27 08:43:10 -04:00
{
if (gSaveBlock1Ptr->secretBases[i].secretBaseId == 0)
return i;
}
2019-04-05 16:11:24 -05:00
2017-10-27 08:43:10 -04:00
return 0;
}
2021-02-05 12:08:25 -05:00
static u8 FindUnregisteredSecretBaseIndex(void)
2017-10-27 08:43:10 -04:00
{
s16 i;
2019-04-05 16:11:24 -05:00
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-27 08:43:10 -04:00
{
2021-02-05 12:08:25 -05:00
if (gSaveBlock1Ptr->secretBases[i].registryStatus == UNREGISTERED && gSaveBlock1Ptr->secretBases[i].toRegister == FALSE)
2017-10-27 08:43:10 -04:00
return i;
}
2019-04-05 16:11:24 -05:00
2017-10-27 08:43:10 -04:00
return 0;
}
2017-10-27 20:32:05 -04:00
2021-02-05 12:08:25 -05:00
static u8 TrySaveFriendsSecretBase(struct SecretBase *secretBase, u32 version, u32 language)
2017-10-27 20:32:05 -04:00
{
2019-04-05 16:11:24 -05:00
s16 index;
2017-10-27 20:32:05 -04:00
2021-02-05 12:08:25 -05:00
// Secret base has no location
2019-04-05 16:11:24 -05:00
if (!secretBase->secretBaseId)
2017-10-27 20:32:05 -04:00
return 0;
2019-04-05 16:11:24 -05:00
index = GetSecretBaseIndexFromId(secretBase->secretBaseId);
if (index != 0)
2017-10-27 20:32:05 -04:00
{
2021-02-05 12:08:25 -05:00
// An existing secret base is using this location
2019-04-05 16:11:24 -05:00
if (index != -1)
2017-10-27 20:32:05 -04:00
{
2021-02-05 12:08:25 -05:00
if (gSaveBlock1Ptr->secretBases[index].toRegister == TRUE)
2017-10-27 20:32:05 -04:00
return 0;
2019-04-05 16:11:24 -05:00
2021-02-05 12:08:25 -05:00
if (gSaveBlock1Ptr->secretBases[index].registryStatus != NEW || secretBase->toRegister == TRUE)
2017-10-27 20:32:05 -04:00
{
2021-02-05 12:08:25 -05:00
// Overwrite unregistered base at this location
SaveSecretBase(index, secretBase, version, language);
2019-04-05 16:11:24 -05:00
return index;
2017-10-27 20:32:05 -04:00
}
}
2021-02-05 12:08:25 -05:00
// No secret base is using this location, find a spot to save it
2017-10-27 20:32:05 -04:00
else
{
2019-04-05 16:11:24 -05:00
index = FindAvailableSecretBaseIndex();
if (index != 0)
2017-10-27 20:32:05 -04:00
{
2021-02-05 12:08:25 -05:00
// Save in empty space
SaveSecretBase(index, secretBase, version, language);
2019-04-05 16:11:24 -05:00
return index;
2017-10-27 20:32:05 -04:00
}
2019-04-05 16:11:24 -05:00
2021-02-05 12:08:25 -05:00
index = FindUnregisteredSecretBaseIndex();
2019-04-05 16:11:24 -05:00
if (index != 0)
2017-10-27 20:32:05 -04:00
{
2021-02-05 12:08:25 -05:00
// Overwrite unregistered base
SaveSecretBase(index, secretBase, version, language);
2019-04-05 16:11:24 -05:00
return index;
2017-10-27 20:32:05 -04:00
}
}
}
2021-02-05 12:08:25 -05:00
// Unable to save. Either...
// - This was the player's base
// - A registered base exists at this location
// - The secret base limit has been filled with registered bases
2017-10-27 20:32:05 -04:00
return 0;
}
2017-10-27 20:39:02 -04:00
2019-04-05 16:11:24 -05:00
// Moves the registered secret bases to the beginning of the array, so that
// they won't be forgotten during record mixing.
static void SortSecretBasesByRegistryStatus(void)
2017-10-27 20:39:02 -04:00
{
u8 i;
u8 j;
2019-04-05 16:11:24 -05:00
struct SecretBase *secretBases;
2017-10-27 20:39:02 -04:00
secretBases = gSaveBlock1Ptr->secretBases;
2021-02-05 12:08:25 -05:00
for (i = 1; i < SECRET_BASES_COUNT - 1; i++)
2017-10-27 20:39:02 -04:00
{
2019-04-05 16:11:24 -05:00
for (j = i + 1; j < SECRET_BASES_COUNT; j++)
2017-10-27 20:39:02 -04:00
{
if ((secretBases[i].registryStatus == UNREGISTERED && secretBases[j].registryStatus == REGISTERED)
2021-02-05 12:08:25 -05:00
|| (secretBases[i].registryStatus == NEW && secretBases[j].registryStatus != NEW))
2017-10-27 20:39:02 -04:00
{
struct SecretBase temp;
SWAP(secretBases[i], secretBases[j], temp)
2017-10-27 20:39:02 -04:00
}
}
}
}
2017-10-27 20:47:07 -04:00
2021-02-05 12:08:25 -05:00
// Used to save a record mixing friends' bases other than their own
// registryStatus is so registered bases can be attempted first
static void TrySaveFriendsSecretBases(struct SecretBaseRecordMixer *mixer, u8 registryStatus)
2017-10-27 20:47:07 -04:00
{
u16 i;
2019-04-05 16:11:24 -05:00
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-27 20:47:07 -04:00
{
2019-04-05 16:11:24 -05:00
if (mixer->secretBases[i].registryStatus == registryStatus)
2021-02-05 12:08:25 -05:00
TrySaveFriendsSecretBase(&mixer->secretBases[i], mixer->version, mixer->language);
2017-10-27 20:47:07 -04:00
}
}
2017-10-27 20:50:32 -04:00
2021-02-05 12:08:25 -05:00
static bool8 SecretBaseBelongsToPlayer(struct SecretBase *secretBase)
2017-10-27 20:50:32 -04:00
{
u8 i;
if (secretBase->secretBaseId == 0)
return FALSE;
if (secretBase->secretBaseId && secretBase->gender != gSaveBlock2Ptr->playerGender)
return FALSE;
// Check if the player's trainer Id matches the secret base's id.
2019-09-08 11:53:48 -04:00
for (i = 0; i < TRAINER_ID_LENGTH; i++)
2017-10-27 20:50:32 -04:00
{
if (secretBase->trainerId[i] != gSaveBlock2Ptr->playerTrainerId[i])
return FALSE;
}
2019-04-05 16:11:24 -05:00
for (i = 0; i < PLAYER_NAME_LENGTH && (secretBase->trainerName[i] != EOS || gSaveBlock2Ptr->playerName[i] != EOS); i++)
2017-10-27 20:50:32 -04:00
{
if (secretBase->trainerName[i] != gSaveBlock2Ptr->playerName[i])
return FALSE;
}
return TRUE;
}
#define DELETED_BASE_A (1 << 0)
#define DELETED_BASE_B (1 << 1)
#define DELETED_BASE_C (1 << 2)
2021-02-05 12:08:25 -05:00
static void DeleteFirstOldBaseFromPlayerInRecordMixingFriendsRecords(struct SecretBase *basesA, struct SecretBase *basesB, struct SecretBase *basesC)
{
u8 i;
u8 sbFlags = 0;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
{
if (!(sbFlags & DELETED_BASE_A))
{
2019-04-05 16:11:24 -05:00
if (SecretBaseBelongsToPlayer(&basesA[i]) == TRUE)
{
ClearSecretBase(&basesA[i]);
sbFlags |= DELETED_BASE_A;
}
}
if (!(sbFlags & DELETED_BASE_B))
{
2019-04-05 16:11:24 -05:00
if (SecretBaseBelongsToPlayer(&basesB[i]) == TRUE)
{
ClearSecretBase(&basesB[i]);
sbFlags |= DELETED_BASE_B;
}
}
if (!(sbFlags & DELETED_BASE_C))
{
2019-04-05 16:11:24 -05:00
if (SecretBaseBelongsToPlayer(&basesC[i]) == TRUE)
{
ClearSecretBase(&basesC[i]);
sbFlags |= DELETED_BASE_C;
}
}
if (sbFlags == (DELETED_BASE_A | DELETED_BASE_B | DELETED_BASE_C))
{
break;
}
}
}
2017-10-27 21:00:59 -04:00
#undef DELETED_BASE_A
#undef DELETED_BASE_B
#undef DELETED_BASE_C
// returns TRUE if secretBase was deleted, FALSE otherwise
2021-02-05 12:08:25 -05:00
static bool8 ClearDuplicateOwnedSecretBase(struct SecretBase *secretBase, struct SecretBase *secretBases, u8 idx)
2017-10-27 21:00:59 -04:00
{
u8 i;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:00:59 -04:00
{
if (secretBases[i].secretBaseId != 0)
{
2019-04-05 16:11:24 -05:00
if (SecretBasesBelongToSamePlayer(secretBase, &secretBases[i]) == TRUE)
2017-10-27 21:00:59 -04:00
{
2021-02-05 12:08:25 -05:00
if (idx == 0)
2017-10-27 21:00:59 -04:00
{
ClearSecretBase(&secretBases[i]);
return FALSE;
}
2019-04-05 16:11:24 -05:00
if (secretBase->numSecretBasesReceived > secretBases[i].numSecretBasesReceived)
2017-10-27 21:00:59 -04:00
{
ClearSecretBase(&secretBases[i]);
return FALSE;
}
2021-02-05 12:08:25 -05:00
secretBases[i].toRegister = secretBase->toRegister;
2019-04-05 16:11:24 -05:00
ClearSecretBase(secretBase);
2017-10-27 21:00:59 -04:00
return TRUE;
}
}
}
return FALSE;
}
2017-10-27 21:03:27 -04:00
2021-02-05 12:08:25 -05:00
static void ClearDuplicateOwnedSecretBases(struct SecretBase *playersBases, struct SecretBase *friendsBasesA, struct SecretBase *friendsBasesB, struct SecretBase *friendsBasesC)
2017-10-27 21:03:27 -04:00
{
u8 i;
2019-04-05 16:11:24 -05:00
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
if (playersBases[i].secretBaseId)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
if (playersBases[i].registryStatus == REGISTERED)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
// Mark registered bases, so if they're deleted as a duplicate they
// will be re-registered later
playersBases[i].toRegister = TRUE;
2017-10-27 21:03:27 -04:00
}
2021-02-05 12:08:25 -05:00
if (!ClearDuplicateOwnedSecretBase(&playersBases[i], friendsBasesA, i))
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
if (!ClearDuplicateOwnedSecretBase(&playersBases[i], friendsBasesB, i))
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
ClearDuplicateOwnedSecretBase(&playersBases[i], friendsBasesC, i);
2017-10-27 21:03:27 -04:00
}
}
}
}
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
if (friendsBasesA[i].secretBaseId)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
friendsBasesA[i].battledOwnerToday = 0;
if (!ClearDuplicateOwnedSecretBase(&friendsBasesA[i], friendsBasesB, i))
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
ClearDuplicateOwnedSecretBase(&friendsBasesA[i], friendsBasesC, i);
2017-10-27 21:03:27 -04:00
}
}
}
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
if (friendsBasesB[i].secretBaseId)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
friendsBasesB[i].battledOwnerToday = 0;
ClearDuplicateOwnedSecretBase(&friendsBasesB[i], friendsBasesC, i);
2017-10-27 21:03:27 -04:00
}
2021-02-05 12:08:25 -05:00
if (friendsBasesC[i].secretBaseId)
2017-10-27 21:03:27 -04:00
{
2021-02-05 12:08:25 -05:00
friendsBasesC[i].battledOwnerToday = 0;
2017-10-27 21:03:27 -04:00
}
}
}
2017-10-27 21:17:17 -04:00
2021-02-05 12:08:25 -05:00
static void TrySaveRegisteredDuplicate(struct SecretBase *base, u32 version, u32 language)
2017-10-27 21:17:17 -04:00
{
2021-02-05 12:08:25 -05:00
if (base->toRegister == TRUE)
2017-10-27 21:17:17 -04:00
{
2021-02-05 12:08:25 -05:00
TrySaveFriendsSecretBase(base, version, language);
2017-10-27 21:17:17 -04:00
ClearSecretBase(base);
}
}
2021-02-05 12:08:25 -05:00
static void TrySaveRegisteredDuplicates(struct SecretBaseRecordMixer *mixers)
2017-10-27 21:17:17 -04:00
{
u16 i;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:17:17 -04:00
{
2021-02-05 12:08:25 -05:00
TrySaveRegisteredDuplicate(&mixers[0].secretBases[i], mixers[0].version, mixers[0].language);
TrySaveRegisteredDuplicate(&mixers[1].secretBases[i], mixers[1].version, mixers[1].language);
TrySaveRegisteredDuplicate(&mixers[2].secretBases[i], mixers[2].version, mixers[2].language);
2017-10-27 21:17:17 -04:00
}
}
2021-02-05 12:08:25 -05:00
static void SaveRecordMixBases(struct SecretBaseRecordMixer *mixers)
2017-10-27 21:17:17 -04:00
{
2019-04-05 16:11:24 -05:00
DeleteFirstOldBaseFromPlayerInRecordMixingFriendsRecords(mixers[0].secretBases, mixers[1].secretBases, mixers[2].secretBases);
2021-02-05 12:08:25 -05:00
ClearDuplicateOwnedSecretBases(gSaveBlock1Ptr->secretBases, mixers[0].secretBases, mixers[1].secretBases, mixers[2].secretBases);
// First, save any registered secret bases that were deleted as duplicates
TrySaveRegisteredDuplicates(mixers);
// Then try to save the record mixing friends' own bases
TrySaveFriendsSecretBase(mixers[0].secretBases, mixers[0].version, mixers[0].language);
TrySaveFriendsSecretBase(mixers[1].secretBases, mixers[1].version, mixers[1].language);
TrySaveFriendsSecretBase(mixers[2].secretBases, mixers[2].version, mixers[2].language);
// Then try to save as many of their registered bases as possible
TrySaveFriendsSecretBases(&mixers[0], REGISTERED);
TrySaveFriendsSecretBases(&mixers[1], REGISTERED);
TrySaveFriendsSecretBases(&mixers[2], REGISTERED);
// Lastly save as many of their unregistered bases as possible
TrySaveFriendsSecretBases(&mixers[0], UNREGISTERED);
TrySaveFriendsSecretBases(&mixers[1], UNREGISTERED);
TrySaveFriendsSecretBases(&mixers[2], UNREGISTERED);
2017-10-27 21:55:07 -04:00
}
#define INIT_SECRET_BASE_RECORD_MIXER(linkId1, linkId2, linkId3) \
mixers[0].secretBases = secretBases + linkId1 * recordSize; \
mixers[0].version = gLinkPlayers[linkId1].version & 0xFF; \
mixers[0].language = gLinkPlayers[linkId1].language; \
mixers[1].secretBases = secretBases + linkId2 * recordSize; \
mixers[1].version = gLinkPlayers[linkId2].version & 0xFF; \
mixers[1].language = gLinkPlayers[linkId2].language; \
mixers[2].secretBases = secretBases + linkId3 * recordSize; \
mixers[2].version = gLinkPlayers[linkId3].version & 0xFF; \
mixers[2].language = gLinkPlayers[linkId3].language;
2019-04-05 16:11:24 -05:00
void ReceiveSecretBasesData(void *secretBases, size_t recordSize, u8 linkIdx)
2017-10-27 21:55:07 -04:00
{
struct SecretBaseRecordMixer mixers[3];
u16 i;
2017-11-15 22:12:18 +01:00
2019-01-05 20:12:50 +00:00
if (FlagGet(FLAG_RECEIVED_SECRET_POWER))
2017-10-27 21:55:07 -04:00
{
switch (GetLinkPlayerCount())
{
2019-04-05 16:11:24 -05:00
case 2:
memset(secretBases + 2 * recordSize, 0, recordSize);
memset(secretBases + 3 * recordSize, 0, recordSize);
break;
case 3:
memset(secretBases + 3 * recordSize, 0, recordSize);
break;
2017-10-27 21:55:07 -04:00
}
2019-04-05 16:11:24 -05:00
2017-10-27 21:55:07 -04:00
switch (linkIdx)
{
2019-04-05 16:11:24 -05:00
case 0:
INIT_SECRET_BASE_RECORD_MIXER(1, 2, 3)
2019-04-05 16:11:24 -05:00
break;
case 1:
INIT_SECRET_BASE_RECORD_MIXER(2, 3, 0)
2019-04-05 16:11:24 -05:00
break;
case 2:
INIT_SECRET_BASE_RECORD_MIXER(3, 0, 1)
2019-04-05 16:11:24 -05:00
break;
case 3:
INIT_SECRET_BASE_RECORD_MIXER(0, 1, 2)
2019-04-05 16:11:24 -05:00
break;
2017-10-27 21:55:07 -04:00
}
2019-04-05 16:11:24 -05:00
2021-02-05 12:08:25 -05:00
SaveRecordMixBases(mixers);
2019-04-05 16:11:24 -05:00
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:55:07 -04:00
{
2021-02-05 12:08:25 -05:00
// In the process of deleting duplicate bases, if a base the player has registered is deleted it is
// flagged with the temporary toRegister flag, so it can be re-registered after it has been newly saved
if (gSaveBlock1Ptr->secretBases[i].toRegister == TRUE)
2017-10-27 21:55:07 -04:00
{
2021-02-05 12:08:25 -05:00
gSaveBlock1Ptr->secretBases[i].registryStatus = REGISTERED;
gSaveBlock1Ptr->secretBases[i].toRegister = FALSE;
2017-10-27 21:55:07 -04:00
}
}
2019-04-05 16:11:24 -05:00
SortSecretBasesByRegistryStatus();
for (i = 1; i < SECRET_BASES_COUNT; i++)
2017-10-27 21:55:07 -04:00
{
2021-02-05 12:08:25 -05:00
// Unmark "new" bases, they've been saved now and are no longer important
if (gSaveBlock1Ptr->secretBases[i].registryStatus == NEW)
gSaveBlock1Ptr->secretBases[i].registryStatus = UNREGISTERED;
2017-10-27 21:55:07 -04:00
}
2019-04-05 16:11:24 -05:00
if (gSaveBlock1Ptr->secretBases[0].secretBaseId != 0
&& gSaveBlock1Ptr->secretBases[0].numSecretBasesReceived != 0xFFFF)
2017-10-27 21:55:07 -04:00
{
2019-04-05 16:11:24 -05:00
gSaveBlock1Ptr->secretBases[0].numSecretBasesReceived++;
2017-10-27 21:55:07 -04:00
}
}
2017-10-27 21:17:17 -04:00
}
2017-10-27 22:12:11 -04:00
2019-04-05 16:11:24 -05:00
void ClearJapaneseSecretBases(struct SecretBase *bases)
2017-10-27 22:12:11 -04:00
{
u32 i;
2019-04-05 16:11:24 -05:00
for (i = 0; i < SECRET_BASES_COUNT; i++)
2017-10-27 22:12:11 -04:00
{
if (bases[i].language == LANGUAGE_JAPANESE)
ClearSecretBase(&bases[i]);
}
}
void InitSecretBaseVars(void)
2017-10-27 22:12:11 -04:00
{
2019-02-27 23:54:51 -05:00
VarSet(VAR_SECRET_BASE_STEP_COUNTER, 0);
VarSet(VAR_SECRET_BASE_LAST_ITEM_USED, 0);
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, 0);
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, 0);
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
2019-02-27 23:54:51 -05:00
VarSet(VAR_SECRET_BASE_IS_NOT_LOCAL, TRUE);
2017-10-27 22:12:11 -04:00
else
2019-02-27 23:54:51 -05:00
VarSet(VAR_SECRET_BASE_IS_NOT_LOCAL, FALSE);
2019-04-05 16:11:24 -05:00
sInFriendSecretBase = FALSE;
2017-10-27 22:12:11 -04:00
}
void CheckLeftFriendsSecretBase(void)
2017-10-27 22:12:11 -04:00
{
2019-04-05 16:11:24 -05:00
if (VarGet(VAR_SECRET_BASE_IS_NOT_LOCAL) && sInFriendSecretBase == TRUE && !CurMapIsSecretBase())
2017-10-27 22:12:11 -04:00
{
2019-02-27 23:54:51 -05:00
VarSet(VAR_SECRET_BASE_IS_NOT_LOCAL, FALSE);
2019-04-05 16:11:24 -05:00
sInFriendSecretBase = FALSE;
TryPutSecretBaseSecretsOnAir();
2019-02-27 23:54:51 -05:00
VarSet(VAR_SECRET_BASE_STEP_COUNTER, 0);
VarSet(VAR_SECRET_BASE_LAST_ITEM_USED, 0);
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, 0);
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, 0);
VarSet(VAR_SECRET_BASE_IS_NOT_LOCAL, FALSE);
2017-10-27 22:12:11 -04:00
}
}
2017-10-27 22:20:13 -04:00
void CheckInteractedWithFriendsDollDecor(void)
2017-10-27 22:20:13 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_DOLL);
2017-10-27 22:20:13 -04:00
}
void CheckInteractedWithFriendsCushionDecor(void)
2017-10-27 22:20:13 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_CUSHION);
2017-10-27 22:20:13 -04:00
}
void DeclinedSecretBaseBattle(void)
2017-10-27 22:20:13 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
2017-10-27 22:20:13 -04:00
{
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) & ~(SECRET_BASE_BATTLED_WON | SECRET_BASE_BATTLED_LOST | SECRET_BASE_DECLINED_BATTLE));
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) & ~(SECRET_BASE_BATTLED_DRAW));
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_DECLINED_BATTLE);
2017-10-27 22:20:13 -04:00
}
}
void WonSecretBaseBattle(void)
2017-10-27 22:20:13 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
2017-10-27 22:20:13 -04:00
{
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) & ~(SECRET_BASE_BATTLED_WON | SECRET_BASE_BATTLED_LOST | SECRET_BASE_DECLINED_BATTLE));
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) & ~(SECRET_BASE_BATTLED_DRAW));
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_BATTLED_WON);
2017-10-27 22:20:13 -04:00
}
}
void LostSecretBaseBattle(void)
2017-10-27 22:20:13 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
2017-10-27 22:20:13 -04:00
{
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) & ~(SECRET_BASE_BATTLED_WON | SECRET_BASE_BATTLED_LOST | SECRET_BASE_DECLINED_BATTLE));
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) & ~(SECRET_BASE_BATTLED_DRAW));
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_BATTLED_LOST);
2017-10-27 22:20:13 -04:00
}
}
void DrewSecretBaseBattle(void)
2017-10-27 22:20:13 -04:00
{
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
2017-10-27 22:20:13 -04:00
{
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) & ~(SECRET_BASE_BATTLED_WON | SECRET_BASE_BATTLED_LOST | SECRET_BASE_DECLINED_BATTLE));
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) & ~(SECRET_BASE_BATTLED_DRAW));
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_BATTLED_DRAW);
2017-10-27 22:20:13 -04:00
}
}
2017-10-27 22:24:54 -04:00
void CheckInteractedWithFriendsPosterDecor(void)
2017-10-27 22:24:54 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-27 22:24:54 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
switch (MapGridGetMetatileIdAt(x, y))
{
case METATILE_SecretBase_PikaPoster_Left:
case METATILE_SecretBase_PikaPoster_Right:
case METATILE_SecretBase_LongPoster_Left:
case METATILE_SecretBase_LongPoster_Right:
case METATILE_SecretBase_SeaPoster_Left:
case METATILE_SecretBase_SeaPoster_Right:
case METATILE_SecretBase_SkyPoster_Left:
case METATILE_SecretBase_SkyPoster_Right:
case METATILE_SecretBase_KissPoster_Left:
case METATILE_SecretBase_KissPoster_Right:
case METATILE_SecretBase_BallPoster:
case METATILE_SecretBase_GreenPoster:
case METATILE_SecretBase_RedPoster:
case METATILE_SecretBase_BluePoster:
case METATILE_SecretBase_CutePoster:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_POSTER);
2017-10-27 22:24:54 -04:00
break;
}
}
2017-10-27 22:34:38 -04:00
void CheckInteractedWithFriendsFurnitureBottom(void)
2017-10-27 22:34:38 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-27 22:34:38 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
switch (MapGridGetMetatileIdAt(x, y))
{
case METATILE_SecretBase_GlassOrnament_Base1:
case METATILE_SecretBase_GlassOrnament_Base2:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_GLASS_ORNAMENT);
2017-10-27 22:34:38 -04:00
break;
case METATILE_SecretBase_RedPlant_Base1:
case METATILE_SecretBase_RedPlant_Base2:
case METATILE_SecretBase_TropicalPlant_Base1:
case METATILE_SecretBase_TropicalPlant_Base2:
2020-04-13 14:43:50 -04:00
case METATILE_SecretBase_PrettyFlowers_Base1:
case METATILE_SecretBase_PrettyFlowers_Base2:
case METATILE_SecretBase_ColorfulPlant_BaseLeft1:
case METATILE_SecretBase_ColorfulPlant_BaseRight1:
case METATILE_SecretBase_ColorfulPlant_BaseLeft2:
case METATILE_SecretBase_ColorfulPlant_BaseRight2:
case METATILE_SecretBase_BigPlant_BaseLeft1:
case METATILE_SecretBase_BigPlant_BaseRight1:
case METATILE_SecretBase_BigPlant_BaseLeft2:
case METATILE_SecretBase_BigPlant_BaseRight2:
case METATILE_SecretBase_GorgeousPlant_BaseLeft1:
case METATILE_SecretBase_GorgeousPlant_BaseRight1:
case METATILE_SecretBase_GorgeousPlant_BaseLeft2:
case METATILE_SecretBase_GorgeousPlant_BaseRight2:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_LOW_TV_FLAGS, VarGet(VAR_SECRET_BASE_LOW_TV_FLAGS) | SECRET_BASE_USED_PLANT);
2017-10-27 22:34:38 -04:00
break;
case METATILE_SecretBase_Fence_Horizontal:
case METATILE_SecretBase_Fence_Vertical:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_FENCE);
2017-10-27 22:34:38 -04:00
break;
case METATILE_SecretBase_Tire_BottomLeft:
case METATILE_SecretBase_Tire_BottomRight:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_TIRE);
2017-10-27 22:34:38 -04:00
break;
case METATILE_SecretBase_RedBrick_Bottom:
case METATILE_SecretBase_YellowBrick_Bottom:
case METATILE_SecretBase_BlueBrick_Bottom:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_BRICK);
2017-10-27 22:34:38 -04:00
break;
case METATILE_SecretBase_SmallDesk:
case METATILE_SecretBase_PokemonDesk:
case METATILE_SecretBase_HeavyDesk_BottomLeft:
case METATILE_SecretBase_HeavyDesk_BottomMid:
case METATILE_SecretBase_HeavyDesk_BottomRight:
case METATILE_SecretBase_RaggedDesk_BottomLeft:
case METATILE_SecretBase_RaggedDesk_BottomMid:
case METATILE_SecretBase_RaggedDesk_BottomRight:
case METATILE_SecretBase_ComfortDesk_BottomLeft:
case METATILE_SecretBase_ComfortDesk_BottomMid:
case METATILE_SecretBase_ComfortDesk_BottomRight:
case METATILE_SecretBase_BrickDesk_BottomLeft:
case METATILE_SecretBase_BrickDesk_BottomMid:
case METATILE_SecretBase_BrickDesk_BottomRight:
case METATILE_SecretBase_CampDesk_BottomLeft:
case METATILE_SecretBase_CampDesk_BottomMid:
case METATILE_SecretBase_CampDesk_BottomRight:
case METATILE_SecretBase_HardDesk_BottomLeft:
case METATILE_SecretBase_HardDesk_BottomMid:
case METATILE_SecretBase_HardDesk_BottomRight:
case METATILE_SecretBase_PrettyDesk_BottomLeft:
case METATILE_SecretBase_PrettyDesk_BottomMid:
case METATILE_SecretBase_PrettyDesk_BottomRight:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_DESK);
2017-10-27 22:34:38 -04:00
break;
}
}
2017-10-27 22:38:33 -04:00
void CheckInteractedWithFriendsFurnitureMiddle(void)
2017-10-27 22:38:33 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-27 22:38:33 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
switch (MapGridGetMetatileIdAt(x, y))
{
case METATILE_SecretBase_HeavyDesk_TopMid:
case METATILE_SecretBase_RaggedDesk_TopMid:
case METATILE_SecretBase_ComfortDesk_TopMid:
case METATILE_SecretBase_BrickDesk_TopMid:
case METATILE_SecretBase_BrickDesk_Center:
case METATILE_SecretBase_CampDesk_TopMid:
case METATILE_SecretBase_CampDesk_Center:
case METATILE_SecretBase_HardDesk_TopMid:
case METATILE_SecretBase_HardDesk_Center:
case METATILE_SecretBase_PrettyDesk_TopMid:
case METATILE_SecretBase_PrettyDesk_Center:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_DESK);
2017-10-27 22:38:33 -04:00
break;
}
}
2017-10-27 22:43:11 -04:00
void CheckInteractedWithFriendsFurnitureTop(void)
2017-10-27 22:43:11 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-27 22:43:11 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
switch (MapGridGetMetatileIdAt(x, y))
{
case METATILE_SecretBase_HeavyDesk_TopLeft:
case METATILE_SecretBase_HeavyDesk_TopRight:
case METATILE_SecretBase_RaggedDesk_TopLeft:
case METATILE_SecretBase_RaggedDesk_TopRight:
case METATILE_SecretBase_ComfortDesk_TopLeft:
case METATILE_SecretBase_ComfortDesk_TopRight:
case METATILE_SecretBase_BrickDesk_TopLeft:
case METATILE_SecretBase_BrickDesk_TopRight:
case METATILE_SecretBase_BrickDesk_MidLeft:
case METATILE_SecretBase_BrickDesk_MidRight:
case METATILE_SecretBase_CampDesk_TopLeft:
case METATILE_SecretBase_CampDesk_TopRight:
case METATILE_SecretBase_CampDesk_MidLeft:
case METATILE_SecretBase_CampDesk_MidRight:
case METATILE_SecretBase_HardDesk_TopLeft:
case METATILE_SecretBase_HardDesk_TopRight:
case METATILE_SecretBase_HardDesk_MidLeft:
case METATILE_SecretBase_HardDesk_MidRight:
case METATILE_SecretBase_PrettyDesk_TopLeft:
case METATILE_SecretBase_PrettyDesk_TopRight:
case METATILE_SecretBase_PrettyDesk_MidLeft:
case METATILE_SecretBase_PrettyDesk_MidRight:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_DESK);
2017-10-27 22:43:11 -04:00
break;
case METATILE_SecretBase_Tire_TopLeft:
case METATILE_SecretBase_Tire_TopRight:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_TIRE);
2017-10-27 22:43:11 -04:00
break;
case METATILE_SecretBase_RedBrick_Top:
case METATILE_SecretBase_YellowBrick_Top:
case METATILE_SecretBase_BlueBrick_Top:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_BRICK);
2017-10-27 22:43:11 -04:00
break;
}
}
2017-10-27 22:46:57 -04:00
void CheckInteractedWithFriendsSandOrnament(void)
2017-10-27 22:46:57 -04:00
{
2019-04-05 16:11:24 -05:00
s16 x, y;
2017-10-27 22:46:57 -04:00
GetXYCoordsOneStepInFrontOfPlayer(&x, &y);
switch ((int)MapGridGetMetatileIdAt(x, y))
{
case METATILE_SecretBase_SandOrnament_Base1:
case METATILE_SecretBase_SandOrnament_Base2:
2018-12-09 19:22:59 -05:00
if (VarGet(VAR_CURRENT_SECRET_BASE) != 0)
VarSet(VAR_SECRET_BASE_HIGH_TV_FLAGS, VarGet(VAR_SECRET_BASE_HIGH_TV_FLAGS) | SECRET_BASE_USED_SAND_ORNAMENT);
2017-10-27 22:46:57 -04:00
break;
}
}