pokeemerald/src/trade.c

5056 lines
167 KiB
C
Raw Normal View History

2018-12-19 02:29:41 +01:00
#include "global.h"
#include "malloc.h"
2018-12-21 19:20:28 +01:00
#include "battle_anim.h"
2018-12-24 10:03:02 +01:00
#include "battle_interface.h"
2018-12-19 02:29:41 +01:00
#include "bg.h"
#include "cable_club.h"
2019-04-04 23:53:06 +02:00
#include "data.h"
2018-12-23 08:26:38 +01:00
#include "daycare.h"
2018-12-27 09:05:09 +01:00
#include "decompress.h"
2018-12-21 19:20:28 +01:00
#include "event_data.h"
2018-12-28 09:11:41 +01:00
#include "evolution_scene.h"
2018-12-29 19:46:15 +01:00
#include "field_screen_effect.h"
2018-12-19 02:29:41 +01:00
#include "gpu_regs.h"
#include "graphics.h"
#include "international_string_util.h"
#include "librfu.h"
#include "link.h"
#include "link_rfu.h"
2018-12-29 19:46:15 +01:00
#include "load_save.h"
2018-12-27 09:05:09 +01:00
#include "mail.h"
2018-12-19 02:29:41 +01:00
#include "main.h"
2019-03-30 19:36:52 +01:00
#include "mystery_gift.h"
#include "mystery_gift_menu.h"
2018-12-21 19:20:28 +01:00
#include "overworld.h"
2018-12-19 02:29:41 +01:00
#include "palette.h"
#include "party_menu.h"
2018-12-28 09:11:41 +01:00
#include "pokeball.h"
2018-12-27 09:05:09 +01:00
#include "pokedex.h"
2018-12-19 02:29:41 +01:00
#include "pokemon_icon.h"
2018-12-21 19:20:28 +01:00
#include "pokemon_summary_screen.h"
2019-01-26 20:56:06 +01:00
#include "pokemon_storage_system.h"
2018-12-29 19:46:15 +01:00
#include "random.h"
#include "save.h"
#include "script.h"
2018-12-19 02:29:41 +01:00
#include "sound.h"
#include "string_util.h"
2018-12-28 09:11:41 +01:00
#include "strings.h"
2018-12-19 02:29:41 +01:00
#include "task.h"
#include "text.h"
#include "text_window.h"
#include "trainer_card.h"
#include "trade.h"
2019-03-30 19:36:52 +01:00
#include "union_room.h"
2018-12-27 09:05:09 +01:00
#include "util.h"
2018-12-19 02:29:41 +01:00
#include "window.h"
#include "constants/contest.h"
2018-12-28 09:11:41 +01:00
#include "constants/items.h"
2018-12-22 08:26:53 +01:00
#include "constants/moves.h"
#include "constants/region_map_sections.h"
#include "constants/rgb.h"
2018-12-19 02:29:41 +01:00
#include "constants/songs.h"
2020-05-30 10:09:21 +02:00
#include "constants/union_room.h"
2018-12-19 02:29:41 +01:00
// The following tags are offsets from GFXTAG_MENU_TEXT
// They're looped over in CB2_CreateTradeMenu and CB2_ReturnToTradeMenu
// and used as indexes into sMenuTextTileBuffers
enum {
GFXTAG_PLAYER_NAME_L,
GFXTAG_PLAYER_NAME_M,
GFXTAG_PLAYER_NAME_R,
GFXTAG_PARTNER_NAME_L,
GFXTAG_PARTNER_NAME_M,
GFXTAG_PARTNER_NAME_R,
GFXTAG_CANCEL_L,
GFXTAG_CANCEL_R,
GFXTAG_CHOOSE_PKMN_L,
GFXTAG_CHOOSE_PKMN_M,
GFXTAG_CHOOSE_PKMN_R,
GFXTAG_CHOOSE_PKMN_EMPTY_1, // 6 sprites to cover the full bottom bar, but only first 3 are needed
GFXTAG_CHOOSE_PKMN_EMPTY_2,
GFXTAG_CHOOSE_PKMN_EMPTY_3,
GFXTAG_MENU_TEXT_COUNT
};
#define GFXTAG_PLAYER_NAME (1 + GFXTAG_PLAYER_NAME_R - GFXTAG_PLAYER_NAME_L)
#define GFXTAG_PARTNER_NAME (1 + GFXTAG_PARTNER_NAME_R - GFXTAG_PARTNER_NAME_L)
#define GFXTAG_CHOOSE_PKMN (1 + GFXTAG_CHOOSE_PKMN_EMPTY_3 - GFXTAG_CHOOSE_PKMN_L)
2018-12-29 19:46:15 +01:00
2018-12-28 09:11:41 +01:00
struct InGameTrade {
/*0x00*/ u8 nickname[POKEMON_NAME_LENGTH + 1];
2018-12-28 09:11:41 +01:00
/*0x0C*/ u16 species;
/*0x0E*/ u8 ivs[NUM_STATS];
/*0x14*/ u8 abilityNum;
2018-12-28 09:11:41 +01:00
/*0x18*/ u32 otId;
/*0x1C*/ u8 conditions[CONTEST_CATEGORIES_COUNT];
2018-12-28 09:11:41 +01:00
/*0x24*/ u32 personality;
/*0x28*/ u16 heldItem;
/*0x2A*/ u8 mailNum;
/*0x2B*/ u8 otName[11];
/*0x36*/ u8 otGender;
/*0x37*/ u8 sheen;
/*0x38*/ u16 requestedSpecies;
2018-12-28 09:11:41 +01:00
};
static EWRAM_DATA u8 *sMenuTextAllocBuffer = NULL;
2019-10-05 16:41:37 +02:00
// Bytes 0-2 are used for the player's name text
// Bytes 3-5 are used for the partner's name text
// Bytes 6-7 are used for the Cancel text
// Bytes 8-13 are used for the Choose a Pokemon text
// See the corresponding GFXTAGs in src/data/trade.h
static EWRAM_DATA u8 *sMenuTextTileBuffers[GFXTAG_MENU_TEXT_COUNT] = {NULL};
2019-10-05 16:41:37 +02:00
2021-10-23 16:55:46 +02:00
EWRAM_DATA struct Mail gTradeMail[PARTY_SIZE] = {0};
2019-10-04 01:39:37 +02:00
EWRAM_DATA u8 gSelectedTradeMonPositions[2] = {0};
static EWRAM_DATA struct {
2019-10-04 01:39:37 +02:00
/*0x0000*/ u8 bg2hofs;
/*0x0001*/ u8 bg3hofs;
/*0x0002*/ u8 filler_2[38];
2019-10-07 04:58:25 +02:00
/*0x0028*/ u8 partySpriteIds[2][PARTY_SIZE];
/*0x0034*/ u8 cursorSpriteId;
2019-10-04 01:39:37 +02:00
/*0x0035*/ u8 cursorPosition;
/*0x0036*/ u8 partyCounts[2];
2019-10-04 13:42:45 +02:00
/*0x0038*/ bool8 monPresent[PARTY_SIZE * 2];
2019-10-09 11:56:44 +02:00
/*0x0044*/ bool8 neverRead_44;
2019-10-04 01:39:37 +02:00
/*0x0045*/ bool8 isLiveMon[2][PARTY_SIZE];
/*0x0051*/ bool8 isEgg[2][PARTY_SIZE];
/*0x005D*/ u8 hpBarLevels[2][PARTY_SIZE];
2019-10-11 00:46:42 +02:00
/*0x0069*/ u8 bufferPartyState;
/*0x006A*/ u8 filler_6A[5];
2019-10-10 09:50:51 +02:00
/*0x006F*/ u8 tradeMenuFunc;
2019-10-04 13:42:45 +02:00
/*0x0070*/ u8 neverRead_70;
2018-12-28 09:11:41 +01:00
/*0x0071*/ u8 filler_71;
2019-10-10 09:50:51 +02:00
/*0x0072*/ u16 bottomTextTileStart;
2019-10-04 13:42:45 +02:00
/*0x0074*/ u8 drawPartyState[2];
2019-10-04 01:39:37 +02:00
/*0x0076*/ u8 selectedMonIdx[2];
2019-10-07 04:58:25 +02:00
/*0x0078*/ u8 playerLinkFlagChoseAction;
/*0x0079*/ u8 partnerLinkFlagChoseAction;
2019-10-09 11:56:44 +02:00
/*0x007A*/ u8 playerLinkFlagStatus;
/*0x007B*/ u8 partnerLinkFlagStatus;
/*0x007C*/ u8 filler_7C[2];
2019-10-04 01:39:37 +02:00
/*0x007E*/ u8 partnerCursorPosition;
2019-10-04 13:42:45 +02:00
/*0x007F*/ u8 unused_7F;
2018-12-31 09:22:21 +01:00
/*0x0080*/ u16 linkData[20];
2019-10-07 04:58:25 +02:00
/*0x00A8*/ u8 timer;
/*0x00A9*/ u8 giftRibbons[GIFT_RIBBONS_COUNT];
/*0x00B4*/ u8 filler_B4[0x81C];
2018-12-28 09:11:41 +01:00
/*0x08D0*/ struct {
2019-10-07 04:58:25 +02:00
bool8 queued;
u16 queueDelay;
u8 actionId;
} queuedActions[4];
2018-12-28 09:11:41 +01:00
/*0x08F0*/ u16 tilemapBuffer[0x400];
2019-10-09 11:56:44 +02:00
} *sTradeMenuData = {NULL};
static EWRAM_DATA struct {
2021-10-23 16:55:46 +02:00
/*0x00*/ struct Pokemon tempMon; // Used as a temp variable when swapping Pokémon
/*0x64*/ u32 timer;
2019-10-04 13:42:45 +02:00
/*0x68*/ u32 monPersonalities[2];
2018-12-28 09:11:41 +01:00
/*0x70*/ u8 filler_70[2];
/*0x72*/ u8 playerLinkFlagFinishTrade;
/*0x73*/ u8 partnerLinkFlagFinishTrade;
2018-12-29 19:46:15 +01:00
/*0x74*/ u16 linkData[10];
/*0x88*/ u8 linkTimeoutZero1;
/*0x89*/ u8 linkTimeoutZero2;
2019-10-05 16:41:37 +02:00
/*0x8A*/ u16 linkTimeoutCounter;
2019-10-04 13:42:45 +02:00
/*0x8C*/ u16 neverRead_8C;
/*0x8E*/ u8 monSpriteIds[2];
/*0x90*/ u8 connectionSpriteId1; // Multi-purpose sprite ids used during the transfer sequence
/*0x91*/ u8 connectionSpriteId2;
/*0x92*/ u8 cableEndSpriteId;
2019-10-10 09:50:51 +02:00
/*0x93*/ u8 sendTradeFinishState;
/*0x94*/ u16 state;
2018-12-28 09:11:41 +01:00
/*0x96*/ u8 filler_96[0xD2 - 0x96];
/*0xD2*/ u8 releasePokeballSpriteId;
/*0xD3*/ u8 bouncingPokeballSpriteId;
2019-10-04 13:42:45 +02:00
/*0xD4*/ u16 texX;
/*0xD6*/ u16 texY;
/*0xD8*/ u16 neverRead_D8;
/*0xDA*/ u16 neverRead_DA;
/*0xDC*/ u16 scrX;
/*0xDE*/ u16 scrY;
2018-12-28 09:11:41 +01:00
/*0xE0*/ s16 bg1vofs;
/*0xE2*/ s16 bg1hofs;
/*0xE4*/ s16 bg2vofs;
/*0xE6*/ s16 bg2hofs;
2019-10-04 13:42:45 +02:00
/*0xE8*/ u16 sXY;
/*0xEA*/ u16 gbaScale;
2019-10-04 13:42:45 +02:00
/*0xEC*/ u16 alpha;
2018-12-28 09:11:41 +01:00
/*0xEE*/ bool8 isLinkTrade;
2019-10-04 13:42:45 +02:00
/*0xF0*/ u16 monSpecies[2];
/*0xF4*/ u16 cachedMapMusic;
2019-10-04 01:39:37 +02:00
/*0xF6*/ u8 textColors[3];
2018-12-29 19:46:15 +01:00
/*0xF9*/ u8 filler_F9;
2019-10-05 16:41:37 +02:00
/*0xFA*/ bool8 isCableTrade;
/*0xFB*/ u8 wirelessWinLeft;
/*0xFC*/ u8 wirelessWinTop;
/*0xFD*/ u8 wirelessWinRight;
/*0xFE*/ u8 wirelessWinBottom;
2019-10-09 11:56:44 +02:00
} *sTradeData = {NULL};
2019-10-04 13:42:45 +02:00
static bool32 IsWirelessTrade(void);
2019-10-09 11:56:44 +02:00
static void CB2_CreateTradeMenu(void);
static void VBlankCB_TradeMenu(void);
2019-10-07 04:58:25 +02:00
static void CB2_TradeMenu(void);
2019-10-09 11:56:44 +02:00
static void LoadTradeBgGfx(u8);
2019-10-07 04:58:25 +02:00
static void SetTradePartyMonsVisible(void);
static bool8 BufferTradeParties(void);
2019-10-09 11:56:44 +02:00
static void CB1_SendOrReactToLinkTradeData(void);
static void CallTradeMenuFunc(void);
2019-10-04 01:39:37 +02:00
static void SetSelectedMon(u8);
2019-10-04 13:42:45 +02:00
static void DrawTradeMenuParty(u8);
2019-10-07 04:58:25 +02:00
static u8 GetMonNicknameWidth(u8 *, u8, u8);
static void BufferTradeMonMoves(u8 *, u8, u8);
2019-10-04 01:39:37 +02:00
static void PrintPartyNicknamesForTradeMenu(u8);
2019-10-04 13:42:45 +02:00
static void DrawTradeMenuPartyMonInfo(u8, u8, u8, u8, u8, u8);
2019-10-09 11:56:44 +02:00
static void DrawTradeMenuPartyInfo(u8);
2019-10-10 09:50:51 +02:00
static void PrintNicknamesForTradeMenu(void);
2019-10-07 04:58:25 +02:00
static void RedrawTradeMenuParty(u8);
static void Task_DrawSelectionSummary(u8);
static void Task_DrawSelectionTrade(u8);
2019-10-07 04:58:25 +02:00
static void QueueAction(u16, u8);
static u32 GetNumQueuedActions(void);
static void DoQueuedActions(void);
2019-10-04 01:39:37 +02:00
static void PrintTradeMessage(u8);
2019-10-07 04:58:25 +02:00
static bool8 LoadTradeMenuSpriteSheetsAndPalettes(void);
static void DrawBottomRowText(const u8 *, u8 *, u8);
2019-10-07 04:58:25 +02:00
static void SetTradePartyLiveStatuses(u8);
2019-10-04 01:39:37 +02:00
static void GetTradePartyHPBarLevels(u8);
static void SetTradePartyHPBarSprites(void);
2019-10-09 11:56:44 +02:00
static void SaveTradeGiftRibbons(void);
2019-10-07 04:58:25 +02:00
static u32 CanTradeSelectedMon(struct Pokemon *, int, int);
static void SpriteCB_LinkMonGlow(struct Sprite *);
static void SpriteCB_LinkMonShadow(struct Sprite *);
static void SpriteCB_CableEndSending(struct Sprite *);
static void SpriteCB_CableEndReceiving(struct Sprite *);
static void SpriteCB_GbaScreen(struct Sprite *);
2019-10-05 16:41:37 +02:00
static void InitTradeBgInternal(void);
static void CB2_UpdateInGameTrade(void);
static void SetTradeSequenceBgGpuRegs(u8);
2019-10-05 16:41:37 +02:00
static void LoadTradeSequenceSpriteSheetsAndPalettes(void);
static void BufferTradeSceneStrings(void);
2019-10-05 16:41:37 +02:00
static bool8 AnimateTradeSequence(void);
static bool8 AnimateTradeSequenceCable(void);
static bool8 AnimateTradeSequenceWireless(void);
static void SpriteCB_BouncingPokeball(struct Sprite *);
static void SpriteCB_BouncingPokeballDepart(struct Sprite *);
static void SpriteCB_BouncingPokeballDepartEnd(struct Sprite *);
static void SpriteCB_BouncingPokeballArrive(struct Sprite *);
2020-01-08 19:19:41 +01:00
static void BufferInGameTradeMonName(void);
2021-10-23 16:55:46 +02:00
static void SetInGameTradeMail(struct Mail *, const struct InGameTrade *);
static void CB2_UpdateLinkTrade(void);
static void CB2_TryFinishTrade(void);
static void CB2_SaveAndEndTrade(void);
static void CB2_FreeTradeData(void);
2019-10-04 13:42:45 +02:00
static void Task_InGameTrade(u8);
2019-10-04 01:39:37 +02:00
static void CheckPartnersMonForRibbons(void);
static void Task_AnimateWirelessSignal(u8);
static void Task_NarrowWindowForCrossing_Wireless(u8);
static void Task_NarrowWindowForCrossing_Cable(u8);
static void CB2_SaveAndEndWirelessTrade(void);
#include "data/trade.h"
2018-12-28 09:11:41 +01:00
2019-10-04 01:39:37 +02:00
static bool8 SendLinkData(const void *linkData, u32 size)
2018-12-19 02:29:41 +01:00
{
2020-05-30 10:09:21 +02:00
if (gPlayerCurrActivity == ACTIVITY_29)
2018-12-19 02:29:41 +01:00
{
2020-02-16 19:49:36 +01:00
rfu_NI_setSendData(lman.acceptSlot_flag, 84, linkData, size);
2018-12-19 02:29:41 +01:00
return TRUE;
}
else
{
2019-10-04 01:39:37 +02:00
return SendBlock(0, linkData, size);
2018-12-19 02:29:41 +01:00
}
}
2020-08-13 09:09:47 +02:00
static void RequestLinkData(u8 type)
2018-12-19 02:29:41 +01:00
{
2020-08-13 09:09:47 +02:00
SendBlockRequest(type);
2018-12-19 02:29:41 +01:00
}
static bool32 IsLinkTradeTaskFinished(void)
2018-12-19 02:29:41 +01:00
{
2020-05-30 10:09:21 +02:00
if (gPlayerCurrActivity == ACTIVITY_29)
2018-12-19 02:29:41 +01:00
{
if (gRfuSlotStatusNI[Rfu_GetIndexOfNewestChild(lman.acceptSlot_flag)]->send.state == 0)
2018-12-19 02:29:41 +01:00
return TRUE;
else
return FALSE;
}
else
{
2018-12-31 09:22:21 +01:00
return IsLinkTaskFinished();
2018-12-19 02:29:41 +01:00
}
}
2018-12-31 09:22:21 +01:00
static u32 _GetBlockReceivedStatus(void)
2018-12-19 02:29:41 +01:00
{
return GetBlockReceivedStatus();
}
2019-10-04 13:42:45 +02:00
static void TradeResetReceivedFlags(void)
2018-12-19 02:29:41 +01:00
{
2019-10-04 13:42:45 +02:00
if (IsWirelessTrade())
2020-02-16 19:49:36 +01:00
rfu_clearSlot(12, lman.acceptSlot_flag);
2018-12-19 02:29:41 +01:00
else
ResetBlockReceivedFlags();
}
2019-10-04 13:42:45 +02:00
static void TradeResetReceivedFlag(u32 who)
2018-12-19 02:29:41 +01:00
{
2019-10-04 13:42:45 +02:00
if (IsWirelessTrade())
2020-02-16 19:49:36 +01:00
rfu_clearSlot(12, lman.acceptSlot_flag);
2018-12-19 02:29:41 +01:00
else
2019-10-04 13:42:45 +02:00
ResetBlockReceivedFlag(who);
2018-12-19 02:29:41 +01:00
}
2019-10-04 13:42:45 +02:00
static bool32 IsWirelessTrade(void)
2018-12-19 02:29:41 +01:00
{
2020-05-30 10:09:21 +02:00
if (gWirelessCommType && gPlayerCurrActivity == ACTIVITY_29)
2018-12-19 02:29:41 +01:00
return TRUE;
else
return FALSE;
}
2020-08-13 09:09:47 +02:00
static void SetTradeLinkStandbyCallback(u8 unused)
2018-12-19 02:29:41 +01:00
{
2020-08-13 09:09:47 +02:00
SetLinkStandbyCallback();
2018-12-19 02:29:41 +01:00
}
2018-12-31 09:22:21 +01:00
static bool32 _IsLinkTaskFinished(void)
2018-12-19 02:29:41 +01:00
{
2018-12-31 09:22:21 +01:00
return IsLinkTaskFinished();
2018-12-19 02:29:41 +01:00
}
2019-10-04 13:42:45 +02:00
static void InitTradeMenu(void)
2018-12-19 02:29:41 +01:00
{
ResetSpriteData();
FreeAllSpritePalettes();
ResetTasks();
ResetPaletteFade();
gPaletteFade.bufferTransferDisabled = TRUE;
2019-10-09 11:56:44 +02:00
SetVBlankCallback(VBlankCB_TradeMenu);
2021-10-26 22:52:23 +02:00
LoadPalette(gStandardMenuPalette, 0xF0, 20);
LoadPalette(gStandardMenuPalette, 0xD0, 20);
2018-12-19 02:29:41 +01:00
ResetBgsAndClearDma3BusyFlags(0);
2019-10-07 04:58:25 +02:00
InitBgsFromTemplates(0, sTradeMenuBgTemplates, ARRAY_COUNT(sTradeMenuBgTemplates));
2019-10-09 11:56:44 +02:00
SetBgTilemapBuffer(1, sTradeMenuData->tilemapBuffer);
2018-12-19 02:29:41 +01:00
2019-10-07 04:58:25 +02:00
if (InitWindows(sTradeMenuWindowTemplates))
2018-12-19 02:29:41 +01:00
{
u32 i;
DeactivateAllTextPrinters();
for (i = 0; i < 18; i++)
{
ClearWindowTilemap(i);
FillWindowPixelBuffer(i, PIXEL_FILL(0));
2018-12-19 02:29:41 +01:00
}
FillBgTilemapBufferRect(0, 0, 0, 0, 30, 20, 15);
LoadUserWindowBorderGfx_(0, 20, 0xC0);
LoadUserWindowBorderGfx(2, 1, 0xE0);
LoadMonIconPalettes();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState = 0;
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = 0;
sTradeMenuData->neverRead_70 = 0;
sTradeMenuData->drawPartyState[TRADE_PLAYER] = 0;
sTradeMenuData->drawPartyState[TRADE_PARTNER] = 0;
sTradeMenuData->playerLinkFlagStatus = 0;
sTradeMenuData->partnerLinkFlagStatus = 0;
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
}
}
2019-10-09 11:56:44 +02:00
void CB2_StartCreateTradeMenu(void)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
SetMainCallback2(CB2_CreateTradeMenu);
2018-12-19 02:29:41 +01:00
gMain.callback1 = NULL;
gEnemyPartyCount = 0;
}
2019-10-09 11:56:44 +02:00
static void CB2_CreateTradeMenu(void)
2018-12-19 02:29:41 +01:00
{
int i;
struct SpriteTemplate temp;
u8 id;
u32 xPos;
switch (gMain.state)
{
case 0:
2019-10-09 11:56:44 +02:00
sTradeMenuData = AllocZeroed(sizeof(*sTradeMenuData));
2019-10-04 13:42:45 +02:00
InitTradeMenu();
sMenuTextAllocBuffer = AllocZeroed(GFXTAG_MENU_TEXT_COUNT * 256);
2018-12-25 15:14:15 +01:00
for (i = 0; i < GFXTAG_MENU_TEXT_COUNT; i++)
sMenuTextTileBuffers[i] = &sMenuTextAllocBuffer[i * 256];
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 1:
gPaletteFade.bufferTransferDisabled = FALSE;
for (i = 0; i < PARTY_SIZE; i++)
CreateMon(&gEnemyParty[i], SPECIES_NONE, 0, USE_RANDOM_IVS, FALSE, 0, OT_ID_PLAYER_ID, 0);
2018-12-19 02:29:41 +01:00
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_STANDBY);
2018-12-19 02:29:41 +01:00
ShowBg(0);
2018-12-31 09:22:21 +01:00
if (!gReceivedRemoteLinkPlayers)
2018-12-19 02:29:41 +01:00
{
2020-06-04 00:00:53 +02:00
gLinkType = LINKTYPE_TRADE_CONNECTING;
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
2018-12-31 09:22:21 +01:00
if (gWirelessCommType)
2018-12-19 02:29:41 +01:00
{
2020-05-30 10:09:21 +02:00
SetWirelessCommType1();
2018-12-19 02:29:41 +01:00
OpenLink();
2020-06-06 22:46:19 +02:00
CreateTask_RfuIdle();
2018-12-19 02:29:41 +01:00
}
else
{
OpenLink();
gMain.state++;
2020-06-04 00:00:53 +02:00
CreateTask(Task_WaitForLinkPlayerConnection, 1);
2018-12-19 02:29:41 +01:00
}
}
else
{
gMain.state = 4;
}
break;
case 2:
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer++;
if (sTradeMenuData->timer > 11)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
gMain.state++;
}
break;
case 3:
if (GetLinkPlayerCount_2() >= GetSavedPlayerCount())
2018-12-19 02:29:41 +01:00
{
if (IsLinkMaster())
{
2019-10-09 11:56:44 +02:00
if (++sTradeMenuData->timer > 30)
2018-12-19 02:29:41 +01:00
{
CheckShouldAdvanceLinkState();
2018-12-19 02:29:41 +01:00
gMain.state++;
}
}
else
{
gMain.state++;
}
}
break;
case 4:
2018-12-31 09:22:21 +01:00
if (gReceivedRemoteLinkPlayers == TRUE && IsLinkPlayerDataExchangeComplete() == TRUE)
2018-12-19 02:29:41 +01:00
{
2020-06-06 22:46:19 +02:00
DestroyTask_RfuIdle();
2018-12-19 02:29:41 +01:00
CalculatePlayerPartyCount();
gMain.state++;
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-31 09:22:21 +01:00
if (gWirelessCommType)
2018-12-19 02:29:41 +01:00
{
Rfu_SetLinkRecovery(TRUE);
2020-08-13 09:09:47 +02:00
SetLinkStandbyCallback();
2018-12-19 02:29:41 +01:00
}
}
break;
case 5:
2018-12-31 09:22:21 +01:00
if (gWirelessCommType)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
if (IsLinkRfuTaskFinished())
2018-12-19 02:29:41 +01:00
{
gMain.state++;
2019-04-04 23:05:46 +02:00
LoadWirelessStatusIndicatorSpriteGfx();
2018-12-19 02:29:41 +01:00
CreateWirelessStatusIndicatorSprite(0, 0);
}
}
else
{
gMain.state++;
}
break;
case 6:
if (BufferTradeParties())
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
SaveTradeGiftRibbons();
2018-12-19 02:29:41 +01:00
gMain.state++;
}
break;
case 7:
CalculateEnemyPartyCount();
SetGpuReg(REG_OFFSET_DISPCNT, 0);
SetGpuReg(REG_OFFSET_BLDCNT, 0);
2019-10-09 11:56:44 +02:00
sTradeMenuData->partyCounts[TRADE_PLAYER] = gPlayerPartyCount;
sTradeMenuData->partyCounts[TRADE_PARTNER] = gEnemyPartyCount;
2018-12-19 02:29:41 +01:00
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PLAYER]; i++)
2018-12-19 02:29:41 +01:00
{
struct Pokemon *mon = &gPlayerParty[i];
2019-10-09 11:56:44 +02:00
sTradeMenuData->partySpriteIds[TRADE_PLAYER][i] = CreateMonIcon(GetMonData(mon, MON_DATA_SPECIES2),
2019-10-31 19:11:55 +01:00
SpriteCB_MonIcon,
2019-10-11 00:46:42 +02:00
(sTradeMonSpriteCoords[i][0] * 8) + 14,
(sTradeMonSpriteCoords[i][1] * 8) - 12,
2018-12-19 02:29:41 +01:00
1,
GetMonData(mon, MON_DATA_PERSONALITY),
TRUE);
}
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PARTNER]; i++)
2018-12-19 02:29:41 +01:00
{
struct Pokemon *mon = &gEnemyParty[i];
2019-10-09 11:56:44 +02:00
sTradeMenuData->partySpriteIds[TRADE_PARTNER][i] = CreateMonIcon(GetMonData(mon, MON_DATA_SPECIES2, NULL),
2019-10-31 19:11:55 +01:00
SpriteCB_MonIcon,
2019-10-11 00:46:42 +02:00
(sTradeMonSpriteCoords[i + PARTY_SIZE][0] * 8) + 14,
(sTradeMonSpriteCoords[i + PARTY_SIZE][1] * 8) - 12,
2018-12-19 02:29:41 +01:00
1,
GetMonData(mon, MON_DATA_PERSONALITY),
FALSE);
}
gMain.state++;
break;
case 8:
LoadHeldItemIcons();
DrawHeldItemIconsForTrade(&sTradeMenuData->partyCounts[0], sTradeMenuData->partySpriteIds[0], TRADE_PLAYER);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 9:
DrawHeldItemIconsForTrade(&sTradeMenuData->partyCounts[0], sTradeMenuData->partySpriteIds[0], TRADE_PARTNER);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 10:
DrawTextWindowAndBufferTiles(gSaveBlock2Ptr->playerName, sMenuTextTileBuffers[GFXTAG_PLAYER_NAME_L], 0, 0, 3);
2018-12-19 02:29:41 +01:00
id = GetMultiplayerId();
DrawTextWindowAndBufferTiles(gLinkPlayers[id ^ 1].name, sMenuTextTileBuffers[GFXTAG_PARTNER_NAME_L], 0, 0, 3);
DrawTextWindowAndBufferTiles(sTradeActionTexts[TRADE_TEXT_CANCEL], sMenuTextTileBuffers[GFXTAG_CANCEL_L], 0, 0, 2);
DrawBottomRowText(sTradeActionTexts[TRADE_TEXT_CHOOSE_MON], sMenuTextTileBuffers[GFXTAG_CHOOSE_PKMN_L], 24);
2018-12-19 02:29:41 +01:00
gMain.state++;
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
break;
case 11:
2019-10-07 04:58:25 +02:00
if (LoadTradeMenuSpriteSheetsAndPalettes())
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 12:
// Create player's name text sprites
2021-10-30 22:47:37 +02:00
xPos = GetStringCenterAlignXOffset(FONT_NORMAL, gSaveBlock2Ptr->playerName, 120);
for (i = 0; i < GFXTAG_PLAYER_NAME; i++)
2018-12-19 02:29:41 +01:00
{
temp = sSpriteTemplate_MenuText;
temp.tileTag += i + GFXTAG_PLAYER_NAME_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, xPos + (i * 32) + 16, 10, 1);
}
// Create partner's name text sprites
2021-10-30 22:47:37 +02:00
xPos = GetStringCenterAlignXOffset(FONT_NORMAL, gLinkPlayers[GetMultiplayerId() ^ 1].name, 120);
for (i = 0; i < GFXTAG_PARTNER_NAME; i++)
2018-12-19 02:29:41 +01:00
{
temp = sSpriteTemplate_MenuText;
temp.tileTag += i + GFXTAG_PARTNER_NAME_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, xPos + (i * 32) + 136, 10, 1);
}
gMain.state++;
break;
case 13:
// Create Cancel text sprites
temp = sSpriteTemplate_MenuText;
temp.tileTag += GFXTAG_CANCEL_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, 215, 152, 1);
temp = sSpriteTemplate_MenuText;
temp.tileTag += GFXTAG_CANCEL_R;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, 247, 152, 1);
// Create Choose a Pokémon text sprites (only 3 are needed, other 3 are empty)
for (i = 0; i < GFXTAG_CHOOSE_PKMN; i++)
2018-12-19 02:29:41 +01:00
{
temp = sSpriteTemplate_MenuText;
temp.tileTag += i + GFXTAG_CHOOSE_PKMN_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, (i * 32) + 24, 150, 1);
}
sTradeMenuData->cursorSpriteId = CreateSprite(&sSpriteTemplate_Cursor, sTradeMonSpriteCoords[0][0] * 8 + 32, sTradeMonSpriteCoords[0][1] * 8, 2);
2019-10-09 11:56:44 +02:00
sTradeMenuData->cursorPosition = 0;
2018-12-19 02:29:41 +01:00
gMain.state++;
rbox_fill_rectangle(0);
break;
case 14:
2019-10-07 04:58:25 +02:00
SetTradePartyLiveStatuses(TRADE_PLAYER);
2019-10-04 01:39:37 +02:00
PrintPartyNicknamesForTradeMenu(TRADE_PLAYER);
2019-10-09 11:56:44 +02:00
sTradeMenuData->bg2hofs = 0;
sTradeMenuData->bg3hofs = 0;
2019-10-07 04:58:25 +02:00
SetTradePartyMonsVisible();
2018-12-19 02:29:41 +01:00
gMain.state++;
2020-08-21 00:02:00 +02:00
PlayBGM(MUS_SCHOOL);
2018-12-19 02:29:41 +01:00
break;
case 15:
2019-10-07 04:58:25 +02:00
SetTradePartyLiveStatuses(TRADE_PARTNER);
2019-10-04 01:39:37 +02:00
PrintPartyNicknamesForTradeMenu(TRADE_PARTNER);
2018-12-19 02:29:41 +01:00
gMain.state++;
// fallthrough
case 16:
2019-10-09 11:56:44 +02:00
LoadTradeBgGfx(0);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 17:
2019-10-09 11:56:44 +02:00
LoadTradeBgGfx(1);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 18:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 19:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_OBJ_1D_MAP | DISPCNT_OBJ_ON);
2019-10-09 11:56:44 +02:00
LoadTradeBgGfx(2);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 20:
2019-10-04 01:39:37 +02:00
GetTradePartyHPBarLevels(TRADE_PLAYER);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 21:
2019-10-04 01:39:37 +02:00
GetTradePartyHPBarLevels(TRADE_PARTNER);
SetTradePartyHPBarSprites();
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 22:
if (!gPaletteFade.active)
{
2019-10-09 11:56:44 +02:00
gMain.callback1 = CB1_SendOrReactToLinkTradeData;
2019-10-07 04:58:25 +02:00
SetMainCallback2(CB2_TradeMenu);
2018-12-19 02:29:41 +01:00
}
break;
}
RunTextPrinters();
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
2019-10-09 11:56:44 +02:00
static void CB2_ReturnToTradeMenu(void)
2018-12-19 02:29:41 +01:00
{
int i;
struct SpriteTemplate temp;
u8 id;
u32 xPos;
switch (gMain.state)
{
case 0:
2019-10-04 13:42:45 +02:00
InitTradeMenu();
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 1:
gMain.state++;
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
break;
case 2:
gMain.state++;
break;
case 3:
gMain.state++;
break;
case 4:
CalculatePlayerPartyCount();
gMain.state++;
break;
case 5:
2018-12-31 09:22:21 +01:00
if (gWirelessCommType)
2018-12-19 02:29:41 +01:00
{
2019-04-04 23:05:46 +02:00
LoadWirelessStatusIndicatorSpriteGfx();
2018-12-19 02:29:41 +01:00
CreateWirelessStatusIndicatorSprite(0, 0);
}
gMain.state++;
break;
case 6:
gMain.state++;
break;
case 7:
CalculateEnemyPartyCount();
2019-10-09 11:56:44 +02:00
sTradeMenuData->partyCounts[TRADE_PLAYER] = gPlayerPartyCount;
sTradeMenuData->partyCounts[TRADE_PARTNER] = gEnemyPartyCount;
2018-12-19 02:29:41 +01:00
ClearWindowTilemap(0);
2019-10-04 01:39:37 +02:00
PrintPartyNicknamesForTradeMenu(TRADE_PLAYER);
PrintPartyNicknamesForTradeMenu(TRADE_PARTNER);
2018-12-19 02:29:41 +01:00
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PLAYER]; i++)
2018-12-19 02:29:41 +01:00
{
struct Pokemon *mon = &gPlayerParty[i];
2019-10-09 11:56:44 +02:00
sTradeMenuData->partySpriteIds[TRADE_PLAYER][i] = CreateMonIcon(GetMonData(mon, MON_DATA_SPECIES2, NULL),
2019-10-31 19:11:55 +01:00
SpriteCB_MonIcon,
2019-10-11 00:46:42 +02:00
(sTradeMonSpriteCoords[i][0] * 8) + 14,
(sTradeMonSpriteCoords[i][1] * 8) - 12,
2018-12-19 02:29:41 +01:00
1,
GetMonData(mon, MON_DATA_PERSONALITY),
TRUE);
}
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PARTNER]; i++)
2018-12-19 02:29:41 +01:00
{
struct Pokemon *mon = &gEnemyParty[i];
2019-10-09 11:56:44 +02:00
sTradeMenuData->partySpriteIds[TRADE_PARTNER][i] = CreateMonIcon(GetMonData(mon, MON_DATA_SPECIES2, NULL),
2019-10-31 19:11:55 +01:00
SpriteCB_MonIcon,
2019-10-11 00:46:42 +02:00
(sTradeMonSpriteCoords[i + PARTY_SIZE][0] * 8) + 14,
(sTradeMonSpriteCoords[i + PARTY_SIZE][1] * 8) - 12,
2018-12-19 02:29:41 +01:00
1,
GetMonData(mon, MON_DATA_PERSONALITY),
FALSE);
}
gMain.state++;
break;
case 8:
LoadHeldItemIcons();
DrawHeldItemIconsForTrade(&sTradeMenuData->partyCounts[0], sTradeMenuData->partySpriteIds[0], TRADE_PLAYER);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 9:
DrawHeldItemIconsForTrade(&sTradeMenuData->partyCounts[0], sTradeMenuData->partySpriteIds[0], TRADE_PARTNER);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 10:
DrawTextWindowAndBufferTiles(gSaveBlock2Ptr->playerName, sMenuTextTileBuffers[GFXTAG_PLAYER_NAME_L], 0, 0, 3);
2018-12-19 02:29:41 +01:00
id = GetMultiplayerId();
DrawTextWindowAndBufferTiles(gLinkPlayers[id ^ 1].name, sMenuTextTileBuffers[GFXTAG_PARTNER_NAME_L], 0, 0, 3);
DrawTextWindowAndBufferTiles(sTradeActionTexts[TRADE_TEXT_CANCEL], sMenuTextTileBuffers[GFXTAG_CANCEL_L], 0, 0, 2);
DrawBottomRowText(sTradeActionTexts[TRADE_TEXT_CHOOSE_MON], sMenuTextTileBuffers[GFXTAG_CHOOSE_PKMN_L], 24);
2018-12-19 02:29:41 +01:00
gMain.state++;
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
break;
case 11:
2019-10-07 04:58:25 +02:00
if (LoadTradeMenuSpriteSheetsAndPalettes())
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 12:
// Create player's name text sprites
2021-10-30 22:47:37 +02:00
xPos = GetStringCenterAlignXOffset(FONT_NORMAL, gSaveBlock2Ptr->playerName, 120);
for (i = 0; i < GFXTAG_PLAYER_NAME; i++)
2018-12-19 02:29:41 +01:00
{
temp = sSpriteTemplate_MenuText;
temp.tileTag += i + GFXTAG_PLAYER_NAME_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, xPos + (i * 32) + 16, 10, 1);
}
// Create partner's name text sprites
2021-10-30 22:47:37 +02:00
xPos = GetStringCenterAlignXOffset(FONT_NORMAL, gLinkPlayers[GetMultiplayerId() ^ 1].name, 120);
for (i = 0; i < GFXTAG_PARTNER_NAME; i++)
2018-12-19 02:29:41 +01:00
{
temp = sSpriteTemplate_MenuText;
temp.tileTag += i + GFXTAG_PARTNER_NAME_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, xPos + (i * 32) + 136, 10, 1);
}
gMain.state++;
break;
case 13:
// Create Cancel text sprites
temp = sSpriteTemplate_MenuText;
temp.tileTag += GFXTAG_CANCEL_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, 215, 152, 1);
temp = sSpriteTemplate_MenuText;
temp.tileTag += GFXTAG_CANCEL_R;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, 247, 152, 1);
// Create Choose a Pokémon text sprites
for (i = 0; i < GFXTAG_CHOOSE_PKMN; i++)
2018-12-19 02:29:41 +01:00
{
temp = sSpriteTemplate_MenuText;
temp.tileTag += i + GFXTAG_CHOOSE_PKMN_L;
2018-12-19 02:29:41 +01:00
CreateSprite(&temp, (i * 32) + 24, 150, 1);
}
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->cursorPosition < PARTY_SIZE)
sTradeMenuData->cursorPosition = gLastViewedMonIndex;
2018-12-19 02:29:41 +01:00
else
2019-10-09 11:56:44 +02:00
sTradeMenuData->cursorPosition = gLastViewedMonIndex + PARTY_SIZE;
2018-12-19 02:29:41 +01:00
sTradeMenuData->cursorSpriteId = CreateSprite(&sSpriteTemplate_Cursor,
sTradeMonSpriteCoords[sTradeMenuData->cursorPosition][0] * 8 + 32,
sTradeMonSpriteCoords[sTradeMenuData->cursorPosition][1] * 8, 2);
2018-12-19 02:29:41 +01:00
gMain.state = 16;
break;
case 16:
2019-10-09 11:56:44 +02:00
LoadTradeBgGfx(0);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 17:
2019-10-09 11:56:44 +02:00
LoadTradeBgGfx(1);
sTradeMenuData->bg2hofs = 0;
sTradeMenuData->bg3hofs = 0;
2019-10-07 04:58:25 +02:00
SetTradePartyMonsVisible();
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 18:
gPaletteFade.bufferTransferDisabled = FALSE;
2021-02-24 17:01:02 +01:00
BlendPalettes(PALETTES_ALL, 16, RGB_BLACK);
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 19:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_OBJ_1D_MAP | DISPCNT_OBJ_ON);
2019-10-09 11:56:44 +02:00
LoadTradeBgGfx(2);
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 20:
gMain.state++;
break;
case 21:
2019-10-04 01:39:37 +02:00
SetTradePartyHPBarSprites();
2018-12-19 02:29:41 +01:00
gMain.state++;
break;
case 22:
if (!gPaletteFade.active)
{
2019-10-07 04:58:25 +02:00
SetMainCallback2(CB2_TradeMenu);
2018-12-19 02:29:41 +01:00
}
break;
}
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
2019-10-09 11:56:44 +02:00
static void VBlankCB_TradeMenu(void)
2018-12-19 02:29:41 +01:00
{
LoadOam();
ProcessSpriteCopyRequests();
TransferPlttBuffer();
}
2019-10-09 11:56:44 +02:00
static void LinkTradeFadeOut(void)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
if (++sTradeMenuData->timer > 15)
2018-12-19 02:29:41 +01:00
{
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_LINK_TRADE_WAIT_FADE;
2018-12-19 02:29:41 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void LinkTradeWaitForFade(void)
2018-12-19 02:29:41 +01:00
{
if (!gPaletteFade.active)
{
2019-10-09 11:56:44 +02:00
gSelectedTradeMonPositions[TRADE_PLAYER] = sTradeMenuData->cursorPosition;
gSelectedTradeMonPositions[TRADE_PARTNER] = sTradeMenuData->partnerCursorPosition;
2018-12-19 02:29:41 +01:00
2018-12-31 09:22:21 +01:00
if (gWirelessCommType)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_LINK_TRADE_WAIT_QUEUE;
2018-12-19 02:29:41 +01:00
}
else
{
2020-08-13 09:09:47 +02:00
SetCloseLinkCallbackAndType(32);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_START_LINK_TRADE;
2018-12-19 02:29:41 +01:00
}
}
}
2019-10-09 11:56:44 +02:00
static void SetLinkTradeCallbacks(void)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
gMain.savedCallback = CB2_StartCreateTradeMenu;
2018-12-19 02:29:41 +01:00
2019-10-05 16:41:37 +02:00
// Wireless Link Trade
2018-12-31 09:22:21 +01:00
if (gWirelessCommType)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
if (IsLinkRfuTaskFinished())
2018-12-19 02:29:41 +01:00
{
Free(sMenuTextAllocBuffer);
2018-12-19 02:29:41 +01:00
FreeAllWindowBuffers();
2019-10-09 11:56:44 +02:00
Free(sTradeMenuData);
2018-12-19 02:29:41 +01:00
gMain.callback1 = NULL;
2019-04-01 00:59:52 +02:00
DestroyWirelessStatusIndicatorSprite();
2019-10-05 16:41:37 +02:00
SetMainCallback2(CB2_LinkTrade);
2018-12-19 02:29:41 +01:00
}
}
2019-10-05 16:41:37 +02:00
// Cable Link Trade
2018-12-19 02:29:41 +01:00
else
{
2018-12-31 09:22:21 +01:00
if (!gReceivedRemoteLinkPlayers)
2018-12-19 02:29:41 +01:00
{
Free(sMenuTextAllocBuffer);
2018-12-19 02:29:41 +01:00
FreeAllWindowBuffers();
2019-10-09 11:56:44 +02:00
Free(sTradeMenuData);
2018-12-19 02:29:41 +01:00
gMain.callback1 = NULL;
2019-10-05 16:41:37 +02:00
SetMainCallback2(CB2_LinkTrade);
2018-12-19 02:29:41 +01:00
}
}
}
2019-10-07 04:58:25 +02:00
static void CB2_TradeMenu(void)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
CallTradeMenuFunc();
2019-10-07 04:58:25 +02:00
DoQueuedActions();
2019-10-04 13:42:45 +02:00
DrawTradeMenuParty(TRADE_PLAYER);
DrawTradeMenuParty(TRADE_PARTNER);
2018-12-25 15:14:15 +01:00
2019-10-09 11:56:44 +02:00
SetGpuReg(REG_OFFSET_BG2HOFS, sTradeMenuData->bg2hofs++);
SetGpuReg(REG_OFFSET_BG3HOFS, sTradeMenuData->bg3hofs--);
2018-12-19 02:29:41 +01:00
RunTextPrintersAndIsPrinter0Active();
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
2019-10-09 11:56:44 +02:00
static void LoadTradeBgGfx(u8 state)
2018-12-19 02:29:41 +01:00
{
int i;
2019-10-07 04:58:25 +02:00
switch (state)
2018-12-19 02:29:41 +01:00
{
case 0:
LoadPalette(gTradeMenu_Pal, 0, 0x60);
LoadBgTiles(1, gTradeMenu_Gfx, 0x1280, 0);
2021-10-26 22:52:23 +02:00
CopyToBgTilemapBufferRect_ChangePalette(1, gTradeMenu_Tilemap, 0, 0, 32, 20, 0);
2019-10-10 09:50:51 +02:00
LoadBgTilemap(2, sTradeStripesBG2Tilemap, 0x800, 0);
2018-12-19 02:29:41 +01:00
break;
case 1:
2019-10-10 09:50:51 +02:00
LoadBgTilemap(3, sTradeStripesBG3Tilemap, 0x800, 0);
2019-10-09 11:56:44 +02:00
DrawTradeMenuPartyInfo(TRADE_PLAYER);
DrawTradeMenuPartyInfo(TRADE_PARTNER);
2018-12-19 02:29:41 +01:00
CopyBgTilemapBufferToVram(1);
break;
case 2:
for (i = 0; i < 4; i++)
SetGpuReg(REG_OFFSET_BG0HOFS + (i * 2), 0);
ShowBg(0);
ShowBg(1);
ShowBg(2);
ShowBg(3);
break;
}
}
2019-10-07 04:58:25 +02:00
static void SetTradePartyMonsVisible(void)
2018-12-19 02:29:41 +01:00
{
int i;
2018-12-21 19:20:28 +01:00
for (i = 0; i < PARTY_SIZE; i++)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
if (i < sTradeMenuData->partyCounts[TRADE_PLAYER])
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
gSprites[sTradeMenuData->partySpriteIds[TRADE_PLAYER][i]].invisible = FALSE;
sTradeMenuData->monPresent[i] = TRUE;
2018-12-19 02:29:41 +01:00
}
else
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->monPresent[i] = FALSE;
2018-12-19 02:29:41 +01:00
}
2018-12-25 15:14:15 +01:00
2019-10-09 11:56:44 +02:00
if (i < sTradeMenuData->partyCounts[TRADE_PARTNER])
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
gSprites[sTradeMenuData->partySpriteIds[TRADE_PARTNER][i]].invisible = FALSE;
sTradeMenuData->monPresent[i + PARTY_SIZE] = TRUE;
2018-12-19 02:29:41 +01:00
}
else
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->monPresent[i + PARTY_SIZE] = FALSE;
2018-12-19 02:29:41 +01:00
}
}
2019-10-09 11:56:44 +02:00
sTradeMenuData->neverRead_44 = TRUE;
2018-12-19 02:29:41 +01:00
}
2018-12-31 09:22:21 +01:00
// why not just use memcpy?
static void Trade_Memcpy(void *dataDest, const void *dataSrc, u32 count)
2018-12-19 02:29:41 +01:00
{
2018-12-31 09:22:21 +01:00
u8 *dest = dataDest;
const u8 *src = dataSrc;
2018-12-19 02:29:41 +01:00
u32 i;
for (i = 0; i < count; i++)
{
dest[i] = src[i];
}
}
static bool8 BufferTradeParties(void)
2018-12-19 02:29:41 +01:00
{
u8 id = GetMultiplayerId();
int i;
struct Pokemon *mon;
2019-10-11 00:46:42 +02:00
switch (sTradeMenuData->bufferPartyState)
2018-12-19 02:29:41 +01:00
{
case 0:
// The parties are sent in pairs rather than all at once
2018-12-31 09:22:21 +01:00
Trade_Memcpy(gBlockSendBuffer, &gPlayerParty[0], 2 * sizeof(struct Pokemon));
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2018-12-19 02:29:41 +01:00
break;
case 1:
if (IsLinkTradeTaskFinished())
2018-12-19 02:29:41 +01:00
{
2018-12-31 09:22:21 +01:00
if (_GetBlockReceivedStatus() == 0)
2018-12-19 02:29:41 +01:00
{
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
else
{
2019-10-04 13:42:45 +02:00
TradeResetReceivedFlags();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
}
break;
case 3:
if (id == 0)
RequestLinkData(BLOCK_REQ_SIZE_200);
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 4:
2018-12-31 09:22:21 +01:00
if (_GetBlockReceivedStatus() == 3)
2018-12-19 02:29:41 +01:00
{
2018-12-31 09:22:21 +01:00
Trade_Memcpy(&gEnemyParty[0], gBlockRecvBuffer[id ^ 1], 2 * sizeof(struct Pokemon));
2019-10-04 13:42:45 +02:00
TradeResetReceivedFlags();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
break;
case 5:
2018-12-31 09:22:21 +01:00
Trade_Memcpy(gBlockSendBuffer, &gPlayerParty[2], 2 * sizeof(struct Pokemon));
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 7:
if (id == 0)
RequestLinkData(BLOCK_REQ_SIZE_200);
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 8:
2018-12-31 09:22:21 +01:00
if (_GetBlockReceivedStatus() == 3)
2018-12-19 02:29:41 +01:00
{
Trade_Memcpy(&gEnemyParty[2], gBlockRecvBuffer[id ^ 1], 2 * sizeof(struct Pokemon));
2019-10-04 13:42:45 +02:00
TradeResetReceivedFlags();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
break;
case 9:
Trade_Memcpy(gBlockSendBuffer, &gPlayerParty[4], 2 * sizeof(struct Pokemon));
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 11:
if (id == 0)
RequestLinkData(BLOCK_REQ_SIZE_200);
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 12:
2018-12-31 09:22:21 +01:00
if (_GetBlockReceivedStatus() == 3)
2018-12-19 02:29:41 +01:00
{
Trade_Memcpy(&gEnemyParty[4], gBlockRecvBuffer[id ^ 1], 2 * sizeof(struct Pokemon));
2019-10-04 13:42:45 +02:00
TradeResetReceivedFlags();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
break;
case 13:
2021-10-23 16:55:46 +02:00
Trade_Memcpy(gBlockSendBuffer, gSaveBlock1Ptr->mail, PARTY_SIZE * sizeof(struct Mail) + 4);
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 15:
if (id == 0)
RequestLinkData(BLOCK_REQ_SIZE_220);
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 16:
2018-12-31 09:22:21 +01:00
if (_GetBlockReceivedStatus() == 3)
2018-12-19 02:29:41 +01:00
{
2021-10-23 16:55:46 +02:00
Trade_Memcpy(gTradeMail, gBlockRecvBuffer[id ^ 1], PARTY_SIZE * sizeof(struct Mail));
2019-10-04 13:42:45 +02:00
TradeResetReceivedFlags();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
break;
case 17:
Trade_Memcpy(gBlockSendBuffer, gSaveBlock1Ptr->giftRibbons, sizeof(sTradeMenuData->giftRibbons));
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 19:
if (id == 0)
RequestLinkData(BLOCK_REQ_SIZE_40);
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
break;
case 20:
2018-12-31 09:22:21 +01:00
if (_GetBlockReceivedStatus() == 3)
2018-12-19 02:29:41 +01:00
{
Trade_Memcpy(sTradeMenuData->giftRibbons, gBlockRecvBuffer[id ^ 1], sizeof(sTradeMenuData->giftRibbons));
2019-10-04 13:42:45 +02:00
TradeResetReceivedFlags();
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
break;
case 21:
2018-12-21 19:20:28 +01:00
for (i = 0, mon = gEnemyParty; i < PARTY_SIZE; mon++, i++)
2018-12-19 02:29:41 +01:00
{
u8 name[POKEMON_NAME_LENGTH + 1];
u16 species = GetMonData(mon, MON_DATA_SPECIES);
if (species != SPECIES_NONE)
{
if (species == SPECIES_SHEDINJA && GetMonData(mon, MON_DATA_LANGUAGE) != LANGUAGE_JAPANESE)
{
GetMonData(mon, MON_DATA_NICKNAME, name);
2019-10-04 01:39:37 +02:00
if (!StringCompareWithoutExtCtrlCodes(name, sJPText_Shedinja))
2018-12-19 02:29:41 +01:00
{
SetMonData(mon, MON_DATA_NICKNAME, gSpeciesNames[SPECIES_SHEDINJA]);
}
}
}
}
return TRUE;
2019-10-11 00:46:42 +02:00
// Delay until next state
2018-12-19 02:29:41 +01:00
case 2:
case 6:
case 10:
case 14:
case 18:
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer++;
if (sTradeMenuData->timer > 10)
2018-12-19 02:29:41 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
2019-10-11 00:46:42 +02:00
sTradeMenuData->bufferPartyState++;
2018-12-19 02:29:41 +01:00
}
break;
}
return FALSE;
}
static void DrawIsThisTradeOkay(void)
2018-12-19 02:29:41 +01:00
{
DrawBottomRowText(sText_IsThisTradeOkay, (void *)(OBJ_VRAM0 + (sTradeMenuData->bottomTextTileStart * 32)), 24);
2018-12-19 02:29:41 +01:00
}
2018-12-21 19:20:28 +01:00
// mpId is unused
static void UpdateLinkTradeFlags(u8 mpId, u8 status)
2018-12-21 19:20:28 +01:00
{
if (status & 1)
2018-12-21 19:20:28 +01:00
{
switch (gBlockRecvBuffer[0][0])
{
2019-10-09 11:56:44 +02:00
case LINKCMD_REQUEST_CANCEL:
sTradeMenuData->playerLinkFlagChoseAction = WANTS_TO_CANCEL;
2018-12-21 19:20:28 +01:00
break;
2019-10-09 11:56:44 +02:00
case LINKCMD_READY_TO_TRADE:
sTradeMenuData->playerLinkFlagChoseAction = WANTS_TO_TRADE;
2018-12-21 19:20:28 +01:00
break;
2019-10-09 11:56:44 +02:00
case LINKCMD_INIT_BLOCK:
sTradeMenuData->playerLinkFlagStatus = INITIATE_TRADE;
2018-12-21 19:20:28 +01:00
break;
2019-10-11 00:46:42 +02:00
case LINKCMD_READY_CANCEL_TRADE:
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagStatus = CANCEL_TRADE;
2018-12-21 19:20:28 +01:00
break;
}
2019-10-09 11:56:44 +02:00
TradeResetReceivedFlag(0);
2018-12-21 19:20:28 +01:00
}
if (status & 2)
2018-12-21 19:20:28 +01:00
{
switch (gBlockRecvBuffer[1][0])
{
2019-10-09 11:56:44 +02:00
case LINKCMD_REQUEST_CANCEL:
sTradeMenuData->partnerLinkFlagChoseAction = WANTS_TO_CANCEL;
2018-12-21 19:20:28 +01:00
break;
2019-10-09 11:56:44 +02:00
case LINKCMD_READY_TO_TRADE:
sTradeMenuData->partnerCursorPosition = gBlockRecvBuffer[1][1] + 6;
sTradeMenuData->partnerLinkFlagChoseAction = WANTS_TO_TRADE;
2018-12-21 19:20:28 +01:00
break;
2019-10-09 11:56:44 +02:00
case LINKCMD_INIT_BLOCK:
sTradeMenuData->partnerLinkFlagStatus = INITIATE_TRADE;
2018-12-21 19:20:28 +01:00
break;
2019-10-11 00:46:42 +02:00
case LINKCMD_READY_CANCEL_TRADE:
2019-10-09 11:56:44 +02:00
sTradeMenuData->partnerLinkFlagStatus = CANCEL_TRADE;
2018-12-21 19:20:28 +01:00
break;
}
2019-10-09 11:56:44 +02:00
TradeResetReceivedFlag(1);
2018-12-21 19:20:28 +01:00
}
}
// mpId is unused
static void ReactToLinkTradeData(u8 mpId, u8 status)
2018-12-21 19:20:28 +01:00
{
if (status & 1)
2018-12-21 19:20:28 +01:00
{
switch (gBlockRecvBuffer[0][0])
{
2021-03-19 19:10:13 +01:00
case LINKCMD_BOTH_CANCEL_TRADE:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_WAITING_FOR_FRIEND);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CANCEL_TRADE_1;
2018-12-21 19:20:28 +01:00
break;
2021-03-19 19:10:13 +01:00
case LINKCMD_PARTNER_CANCEL_TRADE:
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_FRIEND_WANTS_TO_TRADE);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
break;
2019-10-09 11:56:44 +02:00
case LINKCMD_SET_MONS_TO_TRADE:
sTradeMenuData->partnerCursorPosition = gBlockRecvBuffer[0][1] + 6;
2018-12-21 19:20:28 +01:00
rbox_fill_rectangle(0);
2019-10-09 11:56:44 +02:00
SetSelectedMon(sTradeMenuData->cursorPosition);
SetSelectedMon(sTradeMenuData->partnerCursorPosition);
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CONFIRM_TRADE_PROMPT;
2018-12-21 19:20:28 +01:00
break;
2019-10-09 11:56:44 +02:00
case LINKCMD_START_TRADE:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_LINK_TRADE_WAIT_FADE;
2018-12-21 19:20:28 +01:00
break;
2021-03-19 19:10:13 +01:00
case LINKCMD_PLAYER_CANCEL_TRADE:
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_CANCELED);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
TradeResetReceivedFlag(0);
2018-12-21 19:20:28 +01:00
}
if (status & 2)
2019-10-09 11:56:44 +02:00
TradeResetReceivedFlag(1);
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
static void QueueLinkTradeData(void)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->playerLinkFlagChoseAction && sTradeMenuData->partnerLinkFlagChoseAction)
2018-12-21 19:20:28 +01:00
{
2020-05-18 14:03:01 +02:00
if (sTradeMenuData->playerLinkFlagChoseAction == WANTS_TO_TRADE
2019-10-09 11:56:44 +02:00
&& sTradeMenuData->partnerLinkFlagChoseAction == WANTS_TO_TRADE)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_BOTH_MONS_SELECTED;
sTradeMenuData->linkData[0] = LINKCMD_SET_MONS_TO_TRADE;
sTradeMenuData->linkData[1] = sTradeMenuData->cursorPosition;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagChoseAction = sTradeMenuData->partnerLinkFlagChoseAction = 0;
2018-12-21 19:20:28 +01:00
}
2020-05-18 14:03:01 +02:00
else if (sTradeMenuData->playerLinkFlagChoseAction == WANTS_TO_TRADE
2019-10-09 11:56:44 +02:00
&& sTradeMenuData->partnerLinkFlagChoseAction == WANTS_TO_CANCEL)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_CANCELED);
2021-03-19 19:10:13 +01:00
sTradeMenuData->linkData[0] = LINKCMD_PARTNER_CANCEL_TRADE;
2019-10-09 11:56:44 +02:00
sTradeMenuData->linkData[1] = 0;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagStatus = sTradeMenuData->partnerLinkFlagStatus = 0;
sTradeMenuData->playerLinkFlagChoseAction = sTradeMenuData->partnerLinkFlagChoseAction = 0;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
}
2020-05-18 14:03:01 +02:00
else if (sTradeMenuData->playerLinkFlagChoseAction == WANTS_TO_CANCEL
2019-10-09 11:56:44 +02:00
&& sTradeMenuData->partnerLinkFlagChoseAction == WANTS_TO_TRADE)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_FRIEND_WANTS_TO_TRADE);
2021-03-19 19:10:13 +01:00
sTradeMenuData->linkData[0] = LINKCMD_PLAYER_CANCEL_TRADE;
2019-10-09 11:56:44 +02:00
sTradeMenuData->linkData[1] = 0;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagStatus = sTradeMenuData->partnerLinkFlagStatus = 0;
sTradeMenuData->playerLinkFlagChoseAction = sTradeMenuData->partnerLinkFlagChoseAction = 0;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
}
2020-05-18 14:03:01 +02:00
else if (sTradeMenuData->playerLinkFlagChoseAction == WANTS_TO_CANCEL
2019-10-09 11:56:44 +02:00
&& sTradeMenuData->partnerLinkFlagChoseAction == WANTS_TO_CANCEL)
2018-12-21 19:20:28 +01:00
{
2021-03-19 19:10:13 +01:00
sTradeMenuData->linkData[0] = LINKCMD_BOTH_CANCEL_TRADE;
2019-10-09 11:56:44 +02:00
sTradeMenuData->linkData[1] = 0;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagChoseAction = sTradeMenuData->partnerLinkFlagChoseAction = 0;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CANCEL_TRADE_1;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->playerLinkFlagStatus && sTradeMenuData->partnerLinkFlagStatus)
2018-12-21 19:20:28 +01:00
{
2020-05-18 14:03:01 +02:00
if (sTradeMenuData->playerLinkFlagStatus == INITIATE_TRADE
2019-10-09 11:56:44 +02:00
&& sTradeMenuData->partnerLinkFlagStatus == INITIATE_TRADE)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->linkData[0] = LINKCMD_START_TRADE;
sTradeMenuData->linkData[1] = 0;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagStatus = 0;
sTradeMenuData->partnerLinkFlagStatus = 0;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_LINK_TRADE_FADE_OUT;
2018-12-21 19:20:28 +01:00
}
2020-05-18 14:03:01 +02:00
if (sTradeMenuData->playerLinkFlagStatus == CANCEL_TRADE
2019-10-09 11:56:44 +02:00
|| sTradeMenuData->partnerLinkFlagStatus == CANCEL_TRADE)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_CANCELED);
2021-03-19 19:10:13 +01:00
sTradeMenuData->linkData[0] = LINKCMD_PLAYER_CANCEL_TRADE;
2019-10-09 11:56:44 +02:00
sTradeMenuData->linkData[1] = 0;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagStatus = 0;
sTradeMenuData->partnerLinkFlagStatus = 0;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
}
}
}
2019-10-07 04:58:25 +02:00
static void _SetLinkData(u16 *linkData, u16 linkCmd, u16 cursorPosition)
2018-12-21 19:20:28 +01:00
{
2019-10-07 04:58:25 +02:00
linkData[0] = linkCmd;
2019-10-04 01:39:37 +02:00
linkData[1] = cursorPosition;
2019-10-07 04:58:25 +02:00
QueueAction(QUEUE_DELAY_DATA, QUEUE_SEND_DATA);
2018-12-21 19:20:28 +01:00
}
2019-10-07 04:58:25 +02:00
static void SetLinkData(u16 linkCmd, u16 cursorPosition)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
_SetLinkData(sTradeMenuData->linkData, linkCmd, cursorPosition);
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
static void CB1_SendOrReactToLinkTradeData(void)
2018-12-21 19:20:28 +01:00
{
2018-12-31 09:22:21 +01:00
u8 mpId = GetMultiplayerId();
u8 status;
2018-12-21 19:20:28 +01:00
2018-12-31 09:22:21 +01:00
if ((status = _GetBlockReceivedStatus()))
2018-12-21 19:20:28 +01:00
{
2018-12-31 09:22:21 +01:00
if (mpId == 0)
UpdateLinkTradeFlags(mpId, status);
2018-12-21 19:20:28 +01:00
else
2019-10-09 11:56:44 +02:00
ReactToLinkTradeData(mpId, status);
2018-12-21 19:20:28 +01:00
}
2018-12-31 09:22:21 +01:00
if (mpId == 0)
2019-10-09 11:56:44 +02:00
QueueLinkTradeData();
2018-12-21 19:20:28 +01:00
}
2019-10-05 16:41:37 +02:00
static u8 GetNewTradeMenuPosition(u8 oldPosition, u8 direction)
2018-12-21 19:20:28 +01:00
{
int i;
2018-12-31 09:22:21 +01:00
u8 newPosition = 0;
2018-12-21 19:20:28 +01:00
for (i = 0; i < PARTY_SIZE; i++)
{
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->monPresent[sTradeNextSelectedMonTable[oldPosition][direction][i]] == TRUE)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
newPosition = sTradeNextSelectedMonTable[oldPosition][direction][i];
2018-12-21 19:20:28 +01:00
break;
}
}
2018-12-31 09:22:21 +01:00
return newPosition;
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
static void TradeMenuMoveCursor(u8 *cursorPosition, u8 direction)
2018-12-21 19:20:28 +01:00
{
2019-10-05 16:41:37 +02:00
u8 newPosition = GetNewTradeMenuPosition(*cursorPosition, direction);
2018-12-21 19:20:28 +01:00
2019-10-05 16:41:37 +02:00
if (newPosition == (PARTY_SIZE * 2)) // CANCEL
2018-12-21 19:20:28 +01:00
{
StartSpriteAnim(&gSprites[sTradeMenuData->cursorSpriteId], CURSOR_ANIM_ON_CANCEL);
2021-07-07 15:11:52 +02:00
gSprites[sTradeMenuData->cursorSpriteId].x = DISPLAY_WIDTH - 16;
gSprites[sTradeMenuData->cursorSpriteId].y = DISPLAY_HEIGHT;
2018-12-21 19:20:28 +01:00
}
else
{
StartSpriteAnim(&gSprites[sTradeMenuData->cursorSpriteId], CURSOR_ANIM_NORMAL);
2021-07-07 15:11:52 +02:00
gSprites[sTradeMenuData->cursorSpriteId].x = sTradeMonSpriteCoords[newPosition][0] * 8 + 32;
gSprites[sTradeMenuData->cursorSpriteId].y = sTradeMonSpriteCoords[newPosition][1] * 8;
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
if (*cursorPosition != newPosition)
2018-12-21 19:20:28 +01:00
{
PlaySE(SE_SELECT);
}
2019-10-04 01:39:37 +02:00
*cursorPosition = newPosition;
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
static void SetReadyToTrade(void)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
PrintTradeMessage(TRADE_MSG_STANDBY);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_STANDBY;
2018-12-21 19:20:28 +01:00
if (GetMultiplayerId() == 1)
2019-10-09 11:56:44 +02:00
SetLinkData(LINKCMD_READY_TO_TRADE, sTradeMenuData->cursorPosition);
2018-12-21 19:20:28 +01:00
else
2019-10-09 11:56:44 +02:00
sTradeMenuData->playerLinkFlagChoseAction = WANTS_TO_TRADE;
2018-12-21 19:20:28 +01:00
}
2019-10-05 16:41:37 +02:00
static void TradeMenuProcessInput(void)
2018-12-21 19:20:28 +01:00
{
2020-09-05 03:11:55 +02:00
if (JOY_REPEAT(DPAD_UP))
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
TradeMenuMoveCursor(&sTradeMenuData->cursorPosition, 0);
2018-12-21 19:20:28 +01:00
}
2020-09-05 03:11:55 +02:00
else if (JOY_REPEAT(DPAD_DOWN))
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
TradeMenuMoveCursor(&sTradeMenuData->cursorPosition, 1);
2018-12-21 19:20:28 +01:00
}
2020-09-05 03:11:55 +02:00
else if (JOY_REPEAT(DPAD_LEFT))
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
TradeMenuMoveCursor(&sTradeMenuData->cursorPosition, 2);
2018-12-21 19:20:28 +01:00
}
2020-09-05 03:11:55 +02:00
else if (JOY_REPEAT(DPAD_RIGHT))
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
TradeMenuMoveCursor(&sTradeMenuData->cursorPosition, 3);
2018-12-21 19:20:28 +01:00
}
2020-09-05 03:11:55 +02:00
if (JOY_NEW(A_BUTTON))
2018-12-21 19:20:28 +01:00
{
PlaySE(SE_SELECT);
2019-10-04 01:39:37 +02:00
// Cursor is in player's party
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->cursorPosition < PARTY_SIZE)
2018-12-21 19:20:28 +01:00
{
2019-04-01 00:59:52 +02:00
DrawTextBorderOuter(1, 1, 14);
FillWindowPixelBuffer(1, PIXEL_FILL(1));
2019-10-04 01:39:37 +02:00
PrintMenuTable(1, ARRAY_COUNT(sSelectTradeMonActions), sSelectTradeMonActions);
2022-02-27 19:47:50 +01:00
InitMenuInUpperLeftCornerNormal(1, ARRAY_COUNT(sSelectTradeMonActions), 0);
2018-12-21 19:20:28 +01:00
PutWindowTilemap(1);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(1, COPYWIN_FULL);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_SELECTED_MON;
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
// Cursor is in partner's party
2019-10-09 11:56:44 +02:00
else if (sTradeMenuData->cursorPosition < PARTY_SIZE * 2)
2018-12-21 19:20:28 +01:00
{
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_SHOW_MON_SUMMARY;
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
// Cursor is on Cancel
2019-10-09 11:56:44 +02:00
else if (sTradeMenuData->cursorPosition == PARTY_SIZE * 2)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
CreateYesNoMenu(&sTradeYesNoWindowTemplate, 1, 14, 0);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CANCEL_TRADE_PROMPT;
DrawBottomRowText(sTradeActionTexts[TRADE_TEXT_CANCEL_TRADE], (void *)(OBJ_VRAM0 + sTradeMenuData->bottomTextTileStart * 32), 24);
2018-12-21 19:20:28 +01:00
}
}
}
2019-10-09 11:56:44 +02:00
static void TradeMenuChooseMon(void)
2018-12-21 19:20:28 +01:00
{
2019-10-10 09:50:51 +02:00
PrintNicknamesForTradeMenu();
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_MAIN_MENU;
gSprites[sTradeMenuData->cursorSpriteId].invisible = FALSE;
DrawBottomRowText(sTradeActionTexts[TRADE_TEXT_CHOOSE_MON], (void *)(OBJ_VRAM0 + sTradeMenuData->bottomTextTileStart * 32), 24);
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
static void TradeMenuProcessInput_SelectedMon(void)
2018-12-21 19:20:28 +01:00
{
switch (Menu_ProcessInputNoWrap())
{
case MENU_B_PRESSED:
PlaySE(SE_SELECT);
TradeMenuChooseMon();
break;
case MENU_NOTHING_CHOSEN:
break;
2019-10-10 09:50:51 +02:00
case 0: // Summary
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_SHOW_MON_SUMMARY;
break;
2019-10-10 09:50:51 +02:00
case 1: // Trade
switch (CanTradeSelectedMon(gPlayerParty, gPlayerPartyCount, sTradeMenuData->cursorPosition))
{
case CAN_TRADE_MON:
SetReadyToTrade();
gSprites[sTradeMenuData->cursorSpriteId].invisible = TRUE;
2018-12-21 19:20:28 +01:00
break;
case CANT_TRADE_LAST_MON:
QueueAction(QUEUE_DELAY_MSG, QUEUE_ONLY_MON2);
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
break;
case CANT_TRADE_NATIONAL:
case CANT_TRADE_INVALID_MON:
QueueAction(QUEUE_DELAY_MSG, QUEUE_MON_CANT_BE_TRADED);
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
break;
2019-10-18 01:22:03 +02:00
case CANT_TRADE_EGG_YET:
case CANT_TRADE_EGG_YET2:
QueueAction(QUEUE_DELAY_MSG, QUEUE_EGG_CANT_BE_TRADED);
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_REDRAW_MAIN_MENU;
2018-12-21 19:20:28 +01:00
break;
}
break;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void ChooseMonAfterButtonPress(void)
2018-12-21 19:20:28 +01:00
{
2020-09-05 03:11:55 +02:00
if ((JOY_NEW(A_BUTTON)) || (JOY_NEW(B_BUTTON)))
2018-12-21 19:20:28 +01:00
{
PlaySE(SE_SELECT);
2019-10-09 11:56:44 +02:00
TradeMenuChooseMon();
2018-12-21 19:20:28 +01:00
}
}
2019-10-04 13:42:45 +02:00
static void TradeMenuShowMonSummaryScreen(void)
2018-12-21 19:20:28 +01:00
{
if (!gPaletteFade.active)
{
2019-10-04 13:42:45 +02:00
// Player's party
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->cursorPosition < PARTY_SIZE)
ShowPokemonSummaryScreen(SUMMARY_MODE_LOCK_MOVES, gPlayerParty, sTradeMenuData->cursorPosition, sTradeMenuData->partyCounts[TRADE_PLAYER] - 1, CB2_ReturnToTradeMenu);
2019-10-04 13:42:45 +02:00
// Partner's party
2018-12-21 19:20:28 +01:00
else
ShowPokemonSummaryScreen(SUMMARY_MODE_LOCK_MOVES, gEnemyParty, sTradeMenuData->cursorPosition - PARTY_SIZE, sTradeMenuData->partyCounts[TRADE_PARTNER] - 1, CB2_ReturnToTradeMenu);
2018-12-21 19:20:28 +01:00
FreeAllWindowBuffers();
}
}
2019-10-04 01:39:37 +02:00
static u8 CheckValidityOfTradeMons(u8 *aliveMons, u8 playerPartyCount, u8 playerMonIdx, u8 partnerMonIdx)
2018-12-21 19:20:28 +01:00
{
int i;
2019-10-11 00:46:42 +02:00
u16 partnerSpecies;
2019-10-04 01:39:37 +02:00
u8 hasLiveMon = 0;
2018-12-21 19:20:28 +01:00
2019-10-04 01:39:37 +02:00
// Make sure mon to be traded isn't player's last alive mon
for (i = 0; i < playerPartyCount; i++)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
if (playerMonIdx != i)
hasLiveMon += aliveMons[i];
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
partnerMonIdx %= PARTY_SIZE;
2019-10-11 00:46:42 +02:00
partnerSpecies = GetMonData(&gEnemyParty[partnerMonIdx], MON_DATA_SPECIES);
2018-12-21 19:20:28 +01:00
2019-10-04 01:39:37 +02:00
// Partner cant trade illegitimate Deoxys or Mew
2019-10-11 00:46:42 +02:00
if (partnerSpecies == SPECIES_DEOXYS || partnerSpecies == SPECIES_MEW)
2018-12-21 19:20:28 +01:00
{
if (!GetMonData(&gEnemyParty[partnerMonIdx], MON_DATA_EVENT_LEGAL))
2019-10-04 01:39:37 +02:00
return PARTNER_MON_INVALID;
2018-12-21 19:20:28 +01:00
}
2019-10-04 01:39:37 +02:00
// Partner cant trade Egg or non-Hoenn mon if player doesn't have National Dex
2018-12-21 19:20:28 +01:00
if (!IsNationalPokedexEnabled())
{
2019-10-11 00:46:42 +02:00
if (sTradeMenuData->isEgg[TRADE_PARTNER][partnerMonIdx] || !IsSpeciesInHoennDex(partnerSpecies))
2019-10-04 01:39:37 +02:00
return PARTNER_MON_INVALID;
2018-12-21 19:20:28 +01:00
}
2018-12-25 15:14:15 +01:00
2019-10-04 01:39:37 +02:00
if (hasLiveMon)
2019-10-10 09:50:51 +02:00
hasLiveMon = BOTH_MONS_VALID;
2018-12-21 19:20:28 +01:00
2020-05-18 14:03:01 +02:00
return hasLiveMon; //PLAYER_MON_INVALID or BOTH_MONS_VALID
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
// Returns TRUE if the partner's selected mon is invalid, FALSE otherwise
2019-10-11 00:46:42 +02:00
static bool32 CheckMonsBeforeTrade(void)
2018-12-21 19:20:28 +01:00
{
int i;
2019-10-04 01:39:37 +02:00
u8 aliveMons[PARTY_SIZE * 2];
2018-12-21 19:20:28 +01:00
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PLAYER]; i++)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
aliveMons[i] = sTradeMenuData->isLiveMon[TRADE_PLAYER][i];
2018-12-21 19:20:28 +01:00
}
switch (CheckValidityOfTradeMons(aliveMons, sTradeMenuData->partyCounts[TRADE_PLAYER],
sTradeMenuData->cursorPosition,
sTradeMenuData->partnerCursorPosition))
2018-12-21 19:20:28 +01:00
{
case PLAYER_MON_INVALID:
QueueAction(QUEUE_DELAY_MSG, QUEUE_ONLY_MON2);
2019-10-11 00:46:42 +02:00
SetLinkData(LINKCMD_READY_CANCEL_TRADE, 0);
break;
2019-10-10 09:50:51 +02:00
case BOTH_MONS_VALID:
QueueAction(QUEUE_DELAY_MSG, QUEUE_STANDBY);
SetLinkData(LINKCMD_INIT_BLOCK, 0);
break;
case PARTNER_MON_INVALID:
QueueAction(QUEUE_DELAY_MSG, QUEUE_FRIENDS_MON_CANT_BE_TRADED);
return TRUE;
2018-12-21 19:20:28 +01:00
}
return FALSE;
}
2019-10-09 11:56:44 +02:00
static void ConfirmOrCancelTrade(void)
2018-12-21 19:20:28 +01:00
{
switch (Menu_ProcessInputNoWrapClearOnChoose())
{
2019-10-10 09:50:51 +02:00
case 0: // Confirm Trade
2019-10-11 00:46:42 +02:00
if (!CheckMonsBeforeTrade())
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_STANDBY;
else
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_PARTNER_MON_INVALID;
2019-10-11 00:46:42 +02:00
PutWindowTilemap(17);
break;
2019-10-10 09:50:51 +02:00
case 1: // Cancel Trade
case MENU_B_PRESSED:
QueueAction(QUEUE_DELAY_MSG, QUEUE_STANDBY);
if (IsLinkTradeTaskFinished())
2019-10-11 00:46:42 +02:00
SetLinkData(LINKCMD_READY_CANCEL_TRADE, 0);
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_STANDBY;
PutWindowTilemap(17);
break;
2018-12-21 19:20:28 +01:00
}
}
// Only when choosing Yes to cancel, when No is chosen all are redrawn anyway
static void RestoreNicknamesCoveredByYesNo(void)
2018-12-21 19:20:28 +01:00
{
int i;
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[1] - 4; i++)
2018-12-21 19:20:28 +01:00
{
PutWindowTilemap(i + 12);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(i + 12, COPYWIN_MAP);
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void CancelTradeYesNo(void)
2018-12-21 19:20:28 +01:00
{
switch (Menu_ProcessInputNoWrapClearOnChoose())
{
case 0: // YES, Cancel
PrintTradeMessage(TRADE_MSG_WAITING_FOR_FRIEND);
SetLinkData(LINKCMD_REQUEST_CANCEL, 0);
gSprites[sTradeMenuData->cursorSpriteId].invisible = TRUE;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_STANDBY;
RestoreNicknamesCoveredByYesNo();
break;
case 1: // NO, Continue
case MENU_B_PRESSED:
PlaySE(SE_SELECT);
TradeMenuChooseMon();
break;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void SetBothSelectedMons(void)
2018-12-21 19:20:28 +01:00
{
if (GetMultiplayerId() == 0)
{
rbox_fill_rectangle(0);
2019-10-09 11:56:44 +02:00
SetSelectedMon(sTradeMenuData->cursorPosition);
SetSelectedMon(sTradeMenuData->partnerCursorPosition);
2018-12-21 19:20:28 +01:00
}
2019-10-11 00:46:42 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CONFIRM_TRADE_PROMPT;
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
static void ConfirmTradePrompt(void)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->drawPartyState[TRADE_PLAYER] == DRAW_PARTY_FINISH
&& sTradeMenuData->drawPartyState[TRADE_PARTNER] == DRAW_PARTY_FINISH)
2018-12-21 19:20:28 +01:00
{
DrawIsThisTradeOkay();
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_DELAY_TRADE_CONFIRM;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void DelayTradeConfirmation(void)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer++;
2018-12-25 15:14:15 +01:00
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->timer > 120)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
CreateYesNoMenu(&sTradeYesNoWindowTemplate, 1, 14, 0);
2019-10-09 11:56:44 +02:00
sTradeMenuData->timer = 0;
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CONFIRM_OR_CANCEL_TRADE;
2018-12-21 19:20:28 +01:00
}
}
2019-10-07 04:58:25 +02:00
static void RedrawTradeMenuAfterPressA(void)
2018-12-21 19:20:28 +01:00
{
int i;
2020-09-05 03:11:55 +02:00
if (JOY_NEW(A_BUTTON))
2018-12-21 19:20:28 +01:00
{
PlaySE(SE_SELECT);
rbox_fill_rectangle(0);
rbox_fill_rectangle(1);
for (i = 0; i < 4; i++)
{
FillWindowPixelBuffer(i + 14, PIXEL_FILL(0));
2018-12-21 19:20:28 +01:00
rbox_fill_rectangle(i + 14);
}
2019-10-07 04:58:25 +02:00
RedrawTradeMenuParty(TRADE_PLAYER);
RedrawTradeMenuParty(TRADE_PARTNER);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_MAIN_MENU;
gSprites[sTradeMenuData->cursorSpriteId].invisible = FALSE;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void CancelTrade_1(void)
2018-12-21 19:20:28 +01:00
{
if (!gPaletteFade.active)
{
if (gWirelessCommType)
{
2020-08-13 09:09:47 +02:00
SetLinkStandbyCallback();
2018-12-21 19:20:28 +01:00
}
else
{
2020-08-13 09:09:47 +02:00
SetCloseLinkCallbackAndType(12);
2018-12-21 19:20:28 +01:00
}
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_CANCEL_TRADE_2;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void CancelTrade_2(void)
2018-12-21 19:20:28 +01:00
{
if (gWirelessCommType)
{
if (IsLinkTradeTaskFinished() && GetNumQueuedActions() == 0)
2018-12-21 19:20:28 +01:00
{
Free(sMenuTextAllocBuffer);
2019-10-09 11:56:44 +02:00
Free(sTradeMenuData);
2018-12-21 19:20:28 +01:00
FreeAllWindowBuffers();
2019-04-01 00:59:52 +02:00
DestroyWirelessStatusIndicatorSprite();
SetMainCallback2(CB2_ReturnToFieldFromMultiplayer);
2018-12-21 19:20:28 +01:00
}
}
else
{
2018-12-31 09:22:21 +01:00
if (!gReceivedRemoteLinkPlayers)
2018-12-21 19:20:28 +01:00
{
Free(sMenuTextAllocBuffer);
2019-10-09 11:56:44 +02:00
Free(sTradeMenuData);
2018-12-21 19:20:28 +01:00
FreeAllWindowBuffers();
SetMainCallback2(CB2_ReturnToFieldFromMultiplayer);
2018-12-21 19:20:28 +01:00
}
}
}
2019-10-09 11:56:44 +02:00
static void LinkTradeWaitForQueue(void)
2018-12-21 19:20:28 +01:00
{
if (!Rfu_SetLinkRecovery(FALSE) && GetNumQueuedActions() == 0)
2018-12-21 19:20:28 +01:00
{
2020-08-13 09:09:47 +02:00
SetLinkStandbyCallback();
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_START_LINK_TRADE;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void PartnersMonWasInvalid(void)
2018-12-21 19:20:28 +01:00
{
2020-09-05 03:11:55 +02:00
if (JOY_NEW(A_BUTTON))
2018-12-21 19:20:28 +01:00
{
2019-10-11 00:46:42 +02:00
SetLinkData(LINKCMD_READY_CANCEL_TRADE, 0);
2019-10-09 11:56:44 +02:00
sTradeMenuData->tradeMenuFunc = TRADEMENUFUNC_STANDBY;
2018-12-21 19:20:28 +01:00
}
}
2019-10-09 11:56:44 +02:00
static void CallTradeMenuFunc(void)
2018-12-21 19:20:28 +01:00
{
2019-10-09 11:56:44 +02:00
switch (sTradeMenuData->tradeMenuFunc)
2018-12-21 19:20:28 +01:00
{
case TRADEMENUFUNC_MAIN_MENU:
TradeMenuProcessInput();
break;
case TRADEMENUFUNC_SELECTED_MON:
TradeMenuProcessInput_SelectedMon();
break;
case TRADEMENUFUNC_SHOW_MON_SUMMARY:
TradeMenuShowMonSummaryScreen();
break;
case TRADEMENUFUNC_CONFIRM_OR_CANCEL_TRADE:
ConfirmOrCancelTrade();
break;
case TRADEMENUFUNC_CANCEL_TRADE_PROMPT:
CancelTradeYesNo();
break;
case TRADEMENUFUNC_BOTH_MONS_SELECTED:
SetBothSelectedMons();
break;
case TRADEMENUFUNC_CONFIRM_TRADE_PROMPT:
ConfirmTradePrompt();
break;
case TRADEMENUFUNC_REDRAW_MAIN_MENU:
RedrawTradeMenuAfterPressA();
break;
case TRADEMENUFUNC_LINK_TRADE_FADE_OUT:
LinkTradeFadeOut();
break;
case TRADEMENUFUNC_LINK_TRADE_WAIT_FADE:
LinkTradeWaitForFade();
break;
case TRADEMENUFUNC_CANCEL_TRADE_1:
CancelTrade_1();
break;
case TRADEMENUFUNC_CANCEL_TRADE_2:
CancelTrade_2();
break;
case TRADEMENUFUNC_START_LINK_TRADE:
SetLinkTradeCallbacks();
break;
case TRADEMENUFUNC_DELAY_TRADE_CONFIRM:
DelayTradeConfirmation();
break;
case TRADEMENUFUNC_UNUSED_15:
ChooseMonAfterButtonPress();
break;
case TRADEMENUFUNC_LINK_TRADE_WAIT_QUEUE:
LinkTradeWaitForQueue();
break;
case TRADEMENUFUNC_PARTNER_MON_INVALID:
PartnersMonWasInvalid();
break;
//case TRADEMENUFUNC_STANDBY: is nop
2018-12-21 19:20:28 +01:00
}
}
2019-10-04 01:39:37 +02:00
static void SetSelectedMon(u8 cursorPosition)
2018-12-21 19:20:28 +01:00
{
2019-10-04 01:39:37 +02:00
//cursorPosition 0-5 are the player's mons, 6-11 are the partner's
u8 whichParty = cursorPosition / PARTY_SIZE;
2018-12-21 19:20:28 +01:00
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->drawPartyState[whichParty] == 0)
2018-12-21 19:20:28 +01:00
{
2019-10-10 09:50:51 +02:00
sTradeMenuData->drawPartyState[whichParty] = 1;
2019-10-09 11:56:44 +02:00
sTradeMenuData->selectedMonIdx[whichParty] = cursorPosition;
2018-12-21 19:20:28 +01:00
}
}
2019-10-04 13:42:45 +02:00
static void DrawTradeMenuParty(u8 whichParty)
2018-12-21 19:20:28 +01:00
{
2018-12-31 09:22:21 +01:00
s8 nameStringWidth;
u8 nickname[20];
u8 movesString[56];
2018-12-21 19:20:28 +01:00
u8 i;
2018-12-31 09:22:21 +01:00
u8 partyIdx;
2019-10-04 13:42:45 +02:00
u8 selectedMonParty;
2019-10-09 11:56:44 +02:00
u8 selectedMonIdx = sTradeMenuData->selectedMonIdx[whichParty];
2018-12-25 15:14:15 +01:00
2019-10-04 13:42:45 +02:00
selectedMonParty = TRADE_PARTNER;
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->selectedMonIdx[whichParty] < PARTY_SIZE)
2019-10-04 13:42:45 +02:00
selectedMonParty = TRADE_PLAYER;
2019-10-04 01:39:37 +02:00
partyIdx = selectedMonIdx % PARTY_SIZE;
2018-12-31 09:22:21 +01:00
nameStringWidth = 0;
2018-12-25 15:14:15 +01:00
2019-10-09 11:56:44 +02:00
switch (sTradeMenuData->drawPartyState[whichParty])
2018-12-21 19:20:28 +01:00
{
2019-10-10 09:50:51 +02:00
case 1:
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[whichParty]; i++)
2018-12-25 15:14:15 +01:00
{
2019-10-09 11:56:44 +02:00
gSprites[sTradeMenuData->partySpriteIds[0][i + (selectedMonParty * PARTY_SIZE)]].invisible = TRUE;
2018-12-25 15:14:15 +01:00
}
2018-12-21 19:20:28 +01:00
2018-12-25 15:14:15 +01:00
for (i = 0; i < 6; i++)
{
2019-10-04 13:42:45 +02:00
ClearWindowTilemap(i + (whichParty * PARTY_SIZE + 2));
2018-12-25 15:14:15 +01:00
}
2019-10-09 11:56:44 +02:00
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].invisible = FALSE;
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].data[0] = 20;
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].data[2] = (sTradeMonSpriteCoords[selectedMonParty * PARTY_SIZE][0]
+ sTradeMonSpriteCoords[selectedMonParty * PARTY_SIZE + 1][0]) / 2 * 8 + 14;
2019-10-11 00:46:42 +02:00
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].data[4] = (sTradeMonSpriteCoords[selectedMonParty * PARTY_SIZE][1] * 8) - 12;
2019-10-31 19:11:55 +01:00
StoreSpriteCallbackInData6(&gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]], SpriteCB_MonIcon);
2019-10-09 11:56:44 +02:00
sTradeMenuData->drawPartyState[whichParty]++;
TradeMenuBouncePartySprites(&gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]]);
2019-10-10 09:50:51 +02:00
CopyToBgTilemapBufferRect_ChangePalette(1, sTradePartyBoxTilemap, whichParty * 15, 0, 15, 17, 0);
2018-12-25 15:14:15 +01:00
CopyBgTilemapBufferToVram(1);
CopyBgTilemapBufferToVram(0);
2019-10-04 13:42:45 +02:00
if (selectedMonParty == TRADE_PLAYER)
2019-10-10 09:50:51 +02:00
PrintNicknamesForTradeMenu();
2018-12-25 15:14:15 +01:00
break;
case 2:
2019-10-31 19:11:55 +01:00
if (gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].callback == SpriteCB_MonIcon)
2019-10-09 11:56:44 +02:00
sTradeMenuData->drawPartyState[whichParty] = 3;
2018-12-25 15:14:15 +01:00
break;
case 3:
2019-10-10 09:50:51 +02:00
CopyToBgTilemapBufferRect_ChangePalette(1, sTradeMovesBoxTilemap, selectedMonParty * 15, 0, 15, 17, 0);
2018-12-25 15:14:15 +01:00
CopyBgTilemapBufferToVram(1);
2021-07-07 15:11:52 +02:00
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].x = (sTradeMonSpriteCoords[selectedMonParty * PARTY_SIZE][0]
+ sTradeMonSpriteCoords[selectedMonParty * PARTY_SIZE + 1][0]) / 2 * 8 + 14;
2021-07-07 15:11:52 +02:00
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].y = (sTradeMonSpriteCoords[selectedMonParty * PARTY_SIZE][1] * 8) - 12;
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].x2 = 0;
gSprites[sTradeMenuData->partySpriteIds[0][partyIdx + (selectedMonParty * PARTY_SIZE)]].y2 = 0;
2019-10-07 04:58:25 +02:00
nameStringWidth = GetMonNicknameWidth(nickname, selectedMonParty, partyIdx);
2021-10-30 22:47:37 +02:00
AddTextPrinterParameterized3((whichParty * 2) + 14, FONT_SMALL, (80 - nameStringWidth) / 2, 4, sTradeTextColors, 0, nickname);
2019-10-04 13:42:45 +02:00
BufferTradeMonMoves(movesString, selectedMonParty, partyIdx);
2021-10-30 22:47:37 +02:00
AddTextPrinterParameterized4((whichParty * 2) + 15, FONT_NORMAL, 0, 0, 0, 0, sTradeTextColors, 0, movesString);
2019-10-04 13:42:45 +02:00
PutWindowTilemap((whichParty * 2) + 14);
2021-11-03 20:29:18 +01:00
CopyWindowToVram((whichParty * 2) + 14, COPYWIN_FULL);
2019-10-04 13:42:45 +02:00
PutWindowTilemap((whichParty * 2) + 15);
2021-11-03 20:29:18 +01:00
CopyWindowToVram((whichParty * 2) + 15, COPYWIN_FULL);
2019-10-09 11:56:44 +02:00
sTradeMenuData->drawPartyState[whichParty]++;
2018-12-25 15:14:15 +01:00
break;
case 4:
2020-05-18 14:03:01 +02:00
DrawTradeMenuPartyMonInfo(whichParty, partyIdx,
sTradeMenuPartyMonBoxDimensions[whichParty][0] + 4,
sTradeMenuPartyMonBoxDimensions[whichParty][1] + 1,
sTradeMenuPartyMonBoxDimensions[whichParty][0],
2019-10-04 13:42:45 +02:00
sTradeMenuPartyMonBoxDimensions[whichParty][1]);
2019-10-09 11:56:44 +02:00
sTradeMenuData->drawPartyState[whichParty]++;
2018-12-25 15:14:15 +01:00
break;
2018-12-21 19:20:28 +01:00
}
}
2018-12-22 08:26:53 +01:00
2019-10-07 04:58:25 +02:00
static u8 GetMonNicknameWidth(u8 *str, u8 whichParty, u8 monIdx)
2018-12-22 08:26:53 +01:00
{
2018-12-31 09:22:21 +01:00
u8 nickname[12];
2018-12-22 08:26:53 +01:00
2019-10-07 04:58:25 +02:00
if (whichParty == TRADE_PLAYER)
2018-12-31 09:22:21 +01:00
GetMonData(&gPlayerParty[monIdx], MON_DATA_NICKNAME, nickname);
2018-12-22 08:26:53 +01:00
else
2018-12-31 09:22:21 +01:00
GetMonData(&gEnemyParty[monIdx], MON_DATA_NICKNAME, nickname);
2018-12-22 08:26:53 +01:00
StringCopy_Nickname(str, nickname);
2021-10-30 22:47:37 +02:00
return GetStringWidth(FONT_SMALL, str, GetFontAttribute(FONT_SMALL, FONTATTR_LETTER_SPACING));
2018-12-22 08:26:53 +01:00
}
static void BufferTradeMonMoves(u8 *str, u8 whichParty, u8 partyIdx)
2018-12-22 08:26:53 +01:00
{
2019-09-08 17:53:48 +02:00
u16 moves[MAX_MON_MOVES];
2018-12-22 08:26:53 +01:00
u16 i;
2019-10-09 11:56:44 +02:00
if (!sTradeMenuData->isEgg[whichParty][partyIdx])
2018-12-22 08:26:53 +01:00
{
for (i = 0; i < MAX_MON_MOVES; i++)
{
if (whichParty == TRADE_PLAYER)
2018-12-22 08:26:53 +01:00
{
moves[i] = GetMonData(&gPlayerParty[partyIdx], i + MON_DATA_MOVE1, NULL);
2018-12-22 08:26:53 +01:00
}
else
{
moves[i] = GetMonData(&gEnemyParty[partyIdx], i + MON_DATA_MOVE1, NULL);
2018-12-22 08:26:53 +01:00
}
}
2019-10-11 00:46:42 +02:00
StringCopy(str, sText_EmptyString);
2018-12-22 08:26:53 +01:00
for (i = 0; i < MAX_MON_MOVES; i++)
{
2019-09-08 17:53:48 +02:00
if (moves[i] != MOVE_NONE)
2018-12-22 08:26:53 +01:00
{
StringAppend(str, gMoveNames[moves[i]]);
2018-12-22 08:26:53 +01:00
}
2019-10-10 09:50:51 +02:00
StringAppend(str, sText_NewLine);
2018-12-22 08:26:53 +01:00
}
}
else
{
2019-10-11 00:46:42 +02:00
StringCopy(str, sText_EmptyString);
2019-10-10 09:50:51 +02:00
StringAppend(str, sText_FourQuestionMarks);
2018-12-22 08:26:53 +01:00
}
}
2019-10-04 01:39:37 +02:00
static void PrintMonNicknameForTradeMenu(u8 whichParty, u8 windowId, u8 *nickname)
2018-12-22 08:26:53 +01:00
{
u8 xPos;
2018-12-31 09:22:21 +01:00
windowId += (whichParty * PARTY_SIZE) + 2;
2021-10-30 22:47:37 +02:00
xPos = GetStringCenterAlignXOffset(FONT_SMALL, nickname, 64);
AddTextPrinterParameterized3(windowId, FONT_SMALL, xPos, 4, sTradeTextColors, 0, nickname);
2018-12-31 09:22:21 +01:00
PutWindowTilemap(windowId);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(windowId, COPYWIN_FULL);
2018-12-22 08:26:53 +01:00
}
2018-12-23 08:26:38 +01:00
2019-10-04 01:39:37 +02:00
static void PrintPartyNicknamesForTradeMenu(u8 whichParty)
2018-12-23 08:26:38 +01:00
{
u8 i;
2019-10-04 01:39:37 +02:00
u8 nickname[20];
u8 str[32];
struct Pokemon *party = (whichParty == TRADE_PLAYER) ? gPlayerParty : gEnemyParty;
2018-12-23 08:26:38 +01:00
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[whichParty]; i++)
2018-12-23 08:26:38 +01:00
{
2019-10-04 01:39:37 +02:00
GetMonData(&party[i], MON_DATA_NICKNAME, nickname);
StringCopy_Nickname(str, nickname);
2019-10-04 01:39:37 +02:00
PrintMonNicknameForTradeMenu(whichParty, i, str);
2018-12-23 08:26:38 +01:00
}
}
2019-10-04 13:42:45 +02:00
static void DrawTradeMenuPartyMonInfo(u8 whichParty, u8 monIdx, u8 x, u8 y, u8 width, u8 height)
2018-12-23 08:26:38 +01:00
{
2018-12-31 09:22:21 +01:00
u8 level;
2019-10-07 04:58:25 +02:00
u32 symbolTile;
2018-12-23 08:26:38 +01:00
u8 gender;
2018-12-31 09:22:21 +01:00
u8 nickname[12];
2018-12-23 08:26:38 +01:00
2019-10-09 11:56:44 +02:00
CopyToBgTilemapBufferRect_ChangePalette(1, gTradeMenuMonBox_Tilemap, width, height, 6, 3, 0);
2018-12-23 08:26:38 +01:00
CopyBgTilemapBufferToVram(1);
if (whichParty == TRADE_PLAYER)
2018-12-31 09:22:21 +01:00
level = GetMonData(&gPlayerParty[monIdx], MON_DATA_LEVEL, NULL);
2018-12-23 08:26:38 +01:00
else
2018-12-31 09:22:21 +01:00
level = GetMonData(&gEnemyParty[monIdx], MON_DATA_LEVEL, NULL);
2018-12-23 08:26:38 +01:00
2019-10-09 11:56:44 +02:00
if (!sTradeMenuData->isEgg[whichParty][monIdx])
2018-12-23 08:26:38 +01:00
{
2018-12-31 09:22:21 +01:00
if (level / 10 != 0)
2019-10-09 11:56:44 +02:00
sTradeMenuData->tilemapBuffer[x + (y * 32)] = (level / 10) + 0x60;
2018-12-31 09:22:21 +01:00
2019-10-09 11:56:44 +02:00
sTradeMenuData->tilemapBuffer[x + (y * 32) + 1] = (level % 10) + 0x70;
2018-12-23 08:26:38 +01:00
}
else
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->tilemapBuffer[x + (y * 32) - 32] = sTradeMenuData->tilemapBuffer[x + (y * 32) - 33];
sTradeMenuData->tilemapBuffer[x + (y * 32) - 31] = sTradeMenuData->tilemapBuffer[x + (y * 32) - 36] | 0x400;
2018-12-23 08:26:38 +01:00
}
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->isEgg[whichParty][monIdx])
2018-12-23 08:26:38 +01:00
{
2019-10-07 04:58:25 +02:00
symbolTile = 0x480;
2018-12-23 08:26:38 +01:00
}
else
{
if (whichParty == TRADE_PLAYER)
2018-12-23 08:26:38 +01:00
{
2018-12-31 09:22:21 +01:00
gender = GetMonGender(&gPlayerParty[monIdx]);
GetMonData(&gPlayerParty[monIdx], MON_DATA_NICKNAME, nickname);
2018-12-23 08:26:38 +01:00
}
else
{
2018-12-31 09:22:21 +01:00
gender = GetMonGender(&gEnemyParty[monIdx]);
GetMonData(&gEnemyParty[monIdx], MON_DATA_NICKNAME, nickname);
2018-12-23 08:26:38 +01:00
}
switch (gender)
{
case MON_MALE:
symbolTile = !NameHasGenderSymbol(nickname, MON_MALE) ? 0x84 : 0x83;
break;
case MON_FEMALE:
symbolTile = !NameHasGenderSymbol(nickname, MON_FEMALE) ? 0x85 : 0x83;
break;
default:
symbolTile = 0x83;
break;
2018-12-23 08:26:38 +01:00
}
}
2019-10-09 11:56:44 +02:00
sTradeMenuData->tilemapBuffer[(y - 1) * 32 + x + 1] = symbolTile;
2018-12-23 08:26:38 +01:00
}
2019-10-09 11:56:44 +02:00
static void DrawTradeMenuPartyInfo(u8 whichParty)
2018-12-23 08:26:38 +01:00
{
2019-12-20 23:09:59 +01:00
s32 i;
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[whichParty]; i++)
2018-12-23 08:26:38 +01:00
{
2019-12-20 23:09:59 +01:00
const u8 (*r5)[2];
const u8 (*r4)[2];
u32 r0 = 3 * whichParty;
const u8 (*r1)[2][2] = sTradeMonLevelCoords;
r5 = r1[r0];
r4 = sTradeMonBoxCoords[r0];
DrawTradeMenuPartyMonInfo(
whichParty,
i,
r5[i][0],
r5[i][1],
r4[i][0],
r4[i][1]
);
2018-12-23 08:26:38 +01:00
}
}
2018-12-24 10:03:02 +01:00
2019-10-07 04:58:25 +02:00
static void ResetTradeMenuPartyPositions(u8 whichParty)
2018-12-24 10:03:02 +01:00
{
int i;
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[whichParty]; i++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
gSprites[sTradeMenuData->partySpriteIds[whichParty][i]].invisible = FALSE;
2021-07-07 15:11:52 +02:00
gSprites[sTradeMenuData->partySpriteIds[whichParty][i]].x = sTradeMonSpriteCoords[(whichParty * PARTY_SIZE) + i][0] * 8 + 14;
gSprites[sTradeMenuData->partySpriteIds[whichParty][i]].y = sTradeMonSpriteCoords[(whichParty * PARTY_SIZE) + i][1] * 8 - 12;
gSprites[sTradeMenuData->partySpriteIds[whichParty][i]].x2 = 0;
gSprites[sTradeMenuData->partySpriteIds[whichParty][i]].y2 = 0;
2018-12-24 10:03:02 +01:00
}
}
2019-10-10 09:50:51 +02:00
static void PrintNicknamesForTradeMenu(void)
2018-12-24 10:03:02 +01:00
{
rbox_fill_rectangle(1);
2020-05-18 14:03:01 +02:00
//PrintPartyNicknamesForTradeMenu(TRADE_PLAYER); ?
2019-10-04 01:39:37 +02:00
PrintPartyNicknamesForTradeMenu(TRADE_PARTNER);
2018-12-24 10:03:02 +01:00
}
2019-10-07 04:58:25 +02:00
static void RedrawTradeMenuParty(u8 whichParty)
2018-12-24 10:03:02 +01:00
{
2019-10-10 09:50:51 +02:00
CopyToBgTilemapBufferRect_ChangePalette(1, sTradePartyBoxTilemap, whichParty * 15, 0, 15, 17, 0);
2018-12-24 10:03:02 +01:00
CopyBgTilemapBufferToVram(1);
2019-10-09 11:56:44 +02:00
DrawTradeMenuPartyInfo(whichParty);
2019-10-04 01:39:37 +02:00
PrintPartyNicknamesForTradeMenu(whichParty);
2019-10-07 04:58:25 +02:00
ResetTradeMenuPartyPositions(whichParty);
DrawBottomRowText(sTradeActionTexts[TRADE_TEXT_CHOOSE_MON], (void *)(OBJ_VRAM0 + (sTradeMenuData->bottomTextTileStart * 32)), 24);
2019-10-09 11:56:44 +02:00
sTradeMenuData->drawPartyState[whichParty] = 0;
2018-12-24 10:03:02 +01:00
}
2019-10-10 09:50:51 +02:00
static void Task_DrawSelectionSummary(u8 taskId)
2018-12-24 10:03:02 +01:00
{
FillBgTilemapBufferRect_Palette0(0, 0, 0, 0, 30, 20);
CopyBgTilemapBufferToVram(0);
}
2019-10-10 09:50:51 +02:00
static void Task_DrawSelectionTrade(u8 taskId)
2018-12-24 10:03:02 +01:00
{
FillBgTilemapBufferRect_Palette0(0, 0, 0, 0, 30, 20);
CopyBgTilemapBufferToVram(0);
}
2019-10-07 04:58:25 +02:00
static void QueueAction(u16 delay, u8 actionId)
2018-12-24 10:03:02 +01:00
{
int i;
2019-10-09 11:56:44 +02:00
for (i = 0; i < (int)ARRAY_COUNT(sTradeMenuData->queuedActions); i++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
if (!sTradeMenuData->queuedActions[i].queued)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->queuedActions[i].queueDelay = delay;
sTradeMenuData->queuedActions[i].actionId = actionId;
sTradeMenuData->queuedActions[i].queued = TRUE;
2018-12-24 10:03:02 +01:00
break;
}
}
}
2019-10-07 04:58:25 +02:00
static u32 GetNumQueuedActions(void)
2018-12-24 10:03:02 +01:00
{
2019-10-07 04:58:25 +02:00
u32 numActions = 0;
2018-12-24 10:03:02 +01:00
int i;
2018-12-25 15:14:15 +01:00
2019-10-09 11:56:44 +02:00
for (i = 0; i < (int)ARRAY_COUNT(sTradeMenuData->queuedActions); i++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
numActions += sTradeMenuData->queuedActions[i].queued;
2018-12-24 10:03:02 +01:00
}
2019-10-07 04:58:25 +02:00
return numActions;
2018-12-24 10:03:02 +01:00
}
2019-10-07 04:58:25 +02:00
static void DoQueuedActions(void)
2018-12-24 10:03:02 +01:00
{
int i;
2019-10-09 11:56:44 +02:00
for (i = 0; i < (int)ARRAY_COUNT(sTradeMenuData->queuedActions); i++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->queuedActions[i].queued)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->queuedActions[i].queueDelay)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->queuedActions[i].queueDelay--;
2018-12-24 10:03:02 +01:00
}
else
{
2019-10-09 11:56:44 +02:00
switch (sTradeMenuData->queuedActions[i].actionId)
2018-12-24 10:03:02 +01:00
{
case QUEUE_SEND_DATA:
SendLinkData(sTradeMenuData->linkData, 20);
break;
case QUEUE_STANDBY:
PrintTradeMessage(TRADE_MSG_STANDBY);
break;
case QUEUE_ONLY_MON1:
PrintTradeMessage(TRADE_MSG_ONLY_MON1);
break;
case QUEUE_ONLY_MON2:
case QUEUE_UNUSED1:
case QUEUE_UNUSED2:
PrintTradeMessage(TRADE_MSG_ONLY_MON2);
break;
case QUEUE_MON_CANT_BE_TRADED:
PrintTradeMessage(TRADE_MSG_MON_CANT_BE_TRADED);
break;
case QUEUE_EGG_CANT_BE_TRADED:
PrintTradeMessage(TRADE_MSG_EGG_CANT_BE_TRADED);
break;
case QUEUE_FRIENDS_MON_CANT_BE_TRADED:
PrintTradeMessage(TRADE_MSG_FRIENDS_MON_CANT_BE_TRADED);
break;
2018-12-24 10:03:02 +01:00
}
2019-10-09 11:56:44 +02:00
sTradeMenuData->queuedActions[i].queued = FALSE;
2018-12-24 10:03:02 +01:00
}
}
}
}
2019-10-04 01:39:37 +02:00
static void PrintTradeMessage(u8 messageId)
2018-12-24 10:03:02 +01:00
{
FillWindowPixelBuffer(0, PIXEL_FILL(1));
AddTextPrinterParameterized(0, FONT_NORMAL, sTradeMessages[messageId], 0, 1, TEXT_SKIP_DRAW, NULL);
2019-04-01 00:59:52 +02:00
DrawTextBorderOuter(0, 20, 12);
2018-12-24 10:03:02 +01:00
PutWindowTilemap(0);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(0, COPYWIN_FULL);
2018-12-24 10:03:02 +01:00
}
2019-10-07 04:58:25 +02:00
static bool8 LoadTradeMenuSpriteSheetsAndPalettes(void)
2018-12-24 10:03:02 +01:00
{
struct SpriteSheet sheet;
if (sTradeMenuData->timer < GFXTAG_MENU_TEXT_COUNT)
2018-12-24 10:03:02 +01:00
{
sheet.data = sMenuTextTileBuffers[sTradeMenuData->timer];
2018-12-24 10:03:02 +01:00
sheet.size = 0x100;
sheet.tag = GFXTAG_MENU_TEXT + sTradeMenuData->timer;
2018-12-24 10:03:02 +01:00
}
2019-10-09 11:56:44 +02:00
switch (sTradeMenuData->timer)
2018-12-24 10:03:02 +01:00
{
case GFXTAG_PLAYER_NAME_L:
case GFXTAG_PLAYER_NAME_M:
case GFXTAG_PLAYER_NAME_R:
case GFXTAG_PARTNER_NAME_L:
case GFXTAG_PARTNER_NAME_M:
case GFXTAG_PARTNER_NAME_R:
case GFXTAG_CANCEL_L:
case GFXTAG_CANCEL_R:
LoadSpriteSheet(&sheet);
sTradeMenuData->timer++;
break;
case GFXTAG_CHOOSE_PKMN_L:
2019-10-10 09:50:51 +02:00
sTradeMenuData->bottomTextTileStart = LoadSpriteSheet(&sheet);
sTradeMenuData->timer++;
break;
case GFXTAG_CHOOSE_PKMN_M:
case GFXTAG_CHOOSE_PKMN_R:
case GFXTAG_CHOOSE_PKMN_EMPTY_1:
case GFXTAG_CHOOSE_PKMN_EMPTY_2:
case GFXTAG_CHOOSE_PKMN_EMPTY_3:
LoadSpriteSheet(&sheet);
sTradeMenuData->timer++;
break;
case GFXTAG_MENU_TEXT_COUNT:
2022-05-17 19:51:54 +02:00
LoadSpritePalette(&sSpritePalette_TradeScreenText);
sTradeMenuData->timer++;
break;
case GFXTAG_MENU_TEXT_COUNT + 1:
LoadSpritePalette(&sCursor_SpritePalette);
sTradeMenuData->timer++;
break;
case GFXTAG_MENU_TEXT_COUNT + 2:
LoadSpriteSheet(&sCursor_SpriteSheet);
sTradeMenuData->timer++;
break;
case GFXTAG_MENU_TEXT_COUNT + 3:
sTradeMenuData->timer = 0;
return TRUE;
2018-12-24 10:03:02 +01:00
}
return FALSE;
}
static void DrawBottomRowText(const u8 *str, u8 *dest, u8 unused)
2018-12-24 10:03:02 +01:00
{
2019-10-07 04:58:25 +02:00
DrawTextWindowAndBufferTiles(str, dest, 0, 0, 6);
2018-12-24 10:03:02 +01:00
}
2019-10-07 04:58:25 +02:00
static void SetTradePartyLiveStatuses(u8 whichParty)
2018-12-24 10:03:02 +01:00
{
int i;
switch (whichParty)
2018-12-24 10:03:02 +01:00
{
case TRADE_PLAYER:
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[whichParty]; i++)
{
if (GetMonData(&gPlayerParty[i], MON_DATA_IS_EGG) == TRUE)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->isLiveMon[whichParty][i] = FALSE;
sTradeMenuData->isEgg[whichParty][i] = TRUE;
2018-12-24 10:03:02 +01:00
}
else if (GetMonData(&gPlayerParty[i], MON_DATA_HP) == 0)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->isLiveMon[whichParty][i] = FALSE;
sTradeMenuData->isEgg[whichParty][i] = FALSE;
2018-12-24 10:03:02 +01:00
}
else
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->isLiveMon[whichParty][i] = TRUE;
sTradeMenuData->isEgg[whichParty][i] = FALSE;
}
}
break;
case TRADE_PARTNER:
2019-10-09 11:56:44 +02:00
for (i = 0; i < sTradeMenuData->partyCounts[whichParty]; i++)
{
if (GetMonData(&gEnemyParty[i], MON_DATA_IS_EGG) == TRUE)
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->isLiveMon[whichParty][i] = FALSE;
sTradeMenuData->isEgg[whichParty][i] = TRUE;
}
else if (GetMonData(&gEnemyParty[i], MON_DATA_HP) == 0)
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->isLiveMon[whichParty][i] = FALSE;
sTradeMenuData->isEgg[whichParty][i] = FALSE;
}
else
{
2019-10-09 11:56:44 +02:00
sTradeMenuData->isLiveMon[whichParty][i] = TRUE;
sTradeMenuData->isEgg[whichParty][i] = FALSE;
}
}
break;
2018-12-24 10:03:02 +01:00
}
}
2019-10-04 01:39:37 +02:00
static void GetTradePartyHPBarLevels(u8 who)
2018-12-24 10:03:02 +01:00
{
2018-12-28 09:11:41 +01:00
u16 i, curHp, maxHp;
2018-12-24 10:03:02 +01:00
2018-12-28 09:11:41 +01:00
switch (who)
2018-12-24 10:03:02 +01:00
{
case TRADE_PLAYER:
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PLAYER]; i++)
{
curHp = GetMonData(&gPlayerParty[i], MON_DATA_HP);
maxHp = GetMonData(&gPlayerParty[i], MON_DATA_MAX_HP);
sTradeMenuData->hpBarLevels[TRADE_PLAYER][i] = GetHPBarLevel(curHp, maxHp);
}
break;
case TRADE_PARTNER:
for (i = 0; i < sTradeMenuData->partyCounts[TRADE_PARTNER]; i++)
{
curHp = GetMonData(&gEnemyParty[i], MON_DATA_HP);
maxHp = GetMonData(&gEnemyParty[i], MON_DATA_MAX_HP);
sTradeMenuData->hpBarLevels[TRADE_PARTNER][i] = GetHPBarLevel(curHp, maxHp);
}
break;
2018-12-24 10:03:02 +01:00
}
}
2019-10-04 01:39:37 +02:00
static void SetTradePartyHPBarSprites(void)
2018-12-24 10:03:02 +01:00
{
int i, j;
2018-12-25 15:14:15 +01:00
2019-10-04 01:39:37 +02:00
for (i = 0; i < 2; i++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
for (j = 0; j < sTradeMenuData->partyCounts[i]; j++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
SetPartyHPBarSprite(&gSprites[sTradeMenuData->partySpriteIds[i][j]], 4 - sTradeMenuData->hpBarLevels[i][j]);
2018-12-24 10:03:02 +01:00
}
}
}
2019-10-09 11:56:44 +02:00
static void SaveTradeGiftRibbons(void)
2018-12-24 10:03:02 +01:00
{
int i;
2019-10-09 11:56:44 +02:00
for (i = 0; i < (int)ARRAY_COUNT(sTradeMenuData->giftRibbons); i++)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
if (gSaveBlock1Ptr->giftRibbons[i] == 0 && sTradeMenuData->giftRibbons[i] != 0)
2018-12-24 10:03:02 +01:00
{
2019-10-09 11:56:44 +02:00
if (sTradeMenuData->giftRibbons[i] < 64)
gSaveBlock1Ptr->giftRibbons[i] = sTradeMenuData->giftRibbons[i];
2018-12-24 10:03:02 +01:00
}
}
}
2019-10-07 04:58:25 +02:00
static u32 CanTradeSelectedMon(struct Pokemon *playerParty, int partyCount, int monIdx)
2018-12-24 10:03:02 +01:00
{
2019-10-10 09:50:51 +02:00
int i, numMonsLeft;
2018-12-24 10:03:02 +01:00
struct LinkPlayer *player;
2019-10-04 01:39:37 +02:00
u32 species[PARTY_SIZE];
u32 species2[PARTY_SIZE];
2018-12-24 10:03:02 +01:00
2019-10-04 01:39:37 +02:00
for (i = 0; i < partyCount; i++)
2018-12-24 10:03:02 +01:00
{
2019-10-04 01:39:37 +02:00
species2[i] = GetMonData(&playerParty[i], MON_DATA_SPECIES2);
species[i] = GetMonData(&playerParty[i], MON_DATA_SPECIES);
2018-12-24 10:03:02 +01:00
}
2019-10-07 04:58:25 +02:00
// Cant trade Eggs or non-Hoenn mons if player doesn't have National Dex
2018-12-24 10:03:02 +01:00
if (!IsNationalPokedexEnabled())
{
2018-12-31 09:22:21 +01:00
if (species2[monIdx] == SPECIES_EGG)
2019-10-18 01:22:03 +02:00
return CANT_TRADE_EGG_YET;
2018-12-24 10:03:02 +01:00
2018-12-31 09:22:21 +01:00
if (!IsSpeciesInHoennDex(species2[monIdx]))
2019-10-07 04:58:25 +02:00
return CANT_TRADE_NATIONAL;
2018-12-24 10:03:02 +01:00
}
player = &gLinkPlayers[GetMultiplayerId() ^ 1];
if ((player->version & 0xFF) != VERSION_RUBY &&
(player->version & 0xFF) != VERSION_SAPPHIRE)
{
2019-10-10 09:50:51 +02:00
// Does partner not have National Dex
if (!(player->progressFlagsCopy & 0xF))
2018-12-24 10:03:02 +01:00
{
2018-12-31 09:22:21 +01:00
if (species2[monIdx] == SPECIES_EGG)
2019-10-18 01:22:03 +02:00
return CANT_TRADE_EGG_YET2;
2018-12-25 15:14:15 +01:00
2018-12-31 09:22:21 +01:00
if (!IsSpeciesInHoennDex(species2[monIdx]))
2019-10-07 04:58:25 +02:00
return CANT_TRADE_INVALID_MON;
2018-12-24 10:03:02 +01:00
}
}
2018-12-31 09:22:21 +01:00
if (species[monIdx] == SPECIES_DEOXYS || species[monIdx] == SPECIES_MEW)
2018-12-24 10:03:02 +01:00
{
if (!GetMonData(&playerParty[monIdx], MON_DATA_EVENT_LEGAL))
2019-10-07 04:58:25 +02:00
return CANT_TRADE_INVALID_MON;
2018-12-24 10:03:02 +01:00
}
2019-10-10 09:50:51 +02:00
// Make Eggs not count for numMonsLeft
2019-10-04 01:39:37 +02:00
for (i = 0; i < partyCount; i++)
2018-12-24 10:03:02 +01:00
{
if (species2[i] == SPECIES_EGG)
species2[i] = SPECIES_NONE;
}
2019-10-07 04:58:25 +02:00
// Count alive mons in party, excluding selected trade mon
2019-10-10 09:50:51 +02:00
for (numMonsLeft = 0, i = 0; i < partyCount; i++)
2018-12-24 10:03:02 +01:00
{
2018-12-31 09:22:21 +01:00
if (i != monIdx)
2019-10-10 09:50:51 +02:00
numMonsLeft += species2[i];
2018-12-24 10:03:02 +01:00
}
2019-10-10 09:50:51 +02:00
if (numMonsLeft != 0)
2019-10-07 04:58:25 +02:00
return CAN_TRADE_MON;
2018-12-24 10:03:02 +01:00
else
2019-10-07 04:58:25 +02:00
return CANT_TRADE_LAST_MON;
2018-12-24 10:03:02 +01:00
}
2019-10-10 09:50:51 +02:00
s32 GetGameProgressForLinkTrade(void)
{
2019-10-10 09:50:51 +02:00
// isGameFrLg could have been a bool but they use 2 and > 0 instead
// possible other checks (for other game versions?) were planned/removed
s32 isGameFrLg;
2018-12-25 15:14:15 +01:00
u16 version;
2019-01-05 19:25:46 +01:00
if (gReceivedRemoteLinkPlayers != 0)
{
2019-10-10 09:50:51 +02:00
isGameFrLg = 0;
version = (gLinkPlayers[GetMultiplayerId() ^ 1].version & 0xFF);
2018-12-25 15:14:15 +01:00
if (version == VERSION_RUBY || version == VERSION_SAPPHIRE || version == VERSION_EMERALD)
2019-10-10 09:50:51 +02:00
isGameFrLg = 0;
2018-12-25 15:14:15 +01:00
else if (version == VERSION_FIRE_RED || version == VERSION_LEAF_GREEN)
2019-10-10 09:50:51 +02:00
isGameFrLg = 2;
2019-10-11 00:46:42 +02:00
// If trading with FRLG, both players must be champion
2019-10-10 09:50:51 +02:00
if (isGameFrLg > 0)
{
2019-10-10 09:50:51 +02:00
// Is player champion
if (gLinkPlayers[GetMultiplayerId()].progressFlagsCopy & 0xF0)
{
2019-10-10 09:50:51 +02:00
if (isGameFrLg == 2) //unnecessary check, isGameFrLg always 2 here
{
2019-10-10 09:50:51 +02:00
// Is partner champion
2019-10-09 11:56:44 +02:00
if (gLinkPlayers[GetMultiplayerId() ^ 1].progressFlagsCopy & 0xF0)
2019-10-10 09:50:51 +02:00
return TRADE_BOTH_PLAYERS_READY;
else
2019-10-10 09:50:51 +02:00
return TRADE_PARTNER_NOT_READY;
}
}
else
{
2019-10-10 09:50:51 +02:00
return TRADE_PLAYER_NOT_READY;
}
}
}
2019-10-10 09:50:51 +02:00
return TRADE_BOTH_PLAYERS_READY;
}
2018-12-25 03:18:06 +01:00
static bool32 IsDeoxysOrMewUntradable(u16 species, bool8 isEventLegal)
2018-12-25 03:18:06 +01:00
{
2018-12-31 09:22:21 +01:00
if (species == SPECIES_DEOXYS || species == SPECIES_MEW)
2018-12-25 03:18:06 +01:00
{
if (!isEventLegal)
2018-12-25 03:18:06 +01:00
return TRUE;
}
return FALSE;
}
2022-06-01 18:41:57 +02:00
int GetUnionRoomTradeMessageId(struct RfuGameCompatibilityData player, struct RfuGameCompatibilityData partner, u16 playerSpecies2, u16 partnerSpecies, u8 requestedType, u16 playerSpecies, bool8 isEventLegal)
2018-12-25 03:18:06 +01:00
{
bool8 playerHasNationalDex = player.hasNationalDex;
bool8 playerIsChampion = player.isChampion;
bool8 partnerHasNationalDex = partner.hasNationalDex;
bool8 partnerIsChampion = partner.isChampion;
u8 partnerVersion = partner.version;
2018-12-25 03:18:06 +01:00
// If partner is not using Emerald, both players must be champion
if (partnerVersion != VERSION_EMERALD)
2018-12-25 03:18:06 +01:00
{
2019-10-04 23:24:03 +02:00
if (!playerIsChampion)
return UR_TRADE_MSG_CANT_TRADE_WITH_PARTNER_1;
else if (!partnerIsChampion)
return UR_TRADE_MSG_CANT_TRADE_WITH_PARTNER_2;
2018-12-25 03:18:06 +01:00
}
// Cannot trade illegitimate Deoxys/Mew
if (IsDeoxysOrMewUntradable(playerSpecies, isEventLegal))
2019-10-04 23:24:03 +02:00
return UR_TRADE_MSG_MON_CANT_BE_TRADED_2;
2018-12-25 03:18:06 +01:00
2019-10-04 23:24:03 +02:00
if (partnerSpecies == SPECIES_EGG)
2018-12-25 03:18:06 +01:00
{
// If partner is trading an Egg then the player must also be trading an Egg
2019-10-04 23:24:03 +02:00
if (playerSpecies2 != partnerSpecies)
return UR_TRADE_MSG_NOT_EGG;
2018-12-25 03:18:06 +01:00
}
else
{
// Player's Pokémon must be of the type the partner requested
if (gBaseStats[playerSpecies2].type1 != requestedType
&& gBaseStats[playerSpecies2].type2 != requestedType)
2019-10-04 23:24:03 +02:00
return UR_TRADE_MSG_NOT_MON_PARTNER_WANTS;
2018-12-25 03:18:06 +01:00
}
// If the player is trading an Egg then the partner must also be trading an Egg
// Odd that this wasn't checked earlier, as by this point we know either the partner doesn't have an Egg or that both do.
2019-10-04 23:24:03 +02:00
if (playerSpecies2 == SPECIES_EGG && playerSpecies2 != partnerSpecies)
return UR_TRADE_MSG_MON_CANT_BE_TRADED_1;
2018-12-25 03:18:06 +01:00
// If the player doesn't have the National Dex then Eggs and non-Hoenn Pokémon can't be traded
2019-10-04 23:24:03 +02:00
if (!playerHasNationalDex)
2018-12-25 03:18:06 +01:00
{
2019-10-04 23:24:03 +02:00
if (playerSpecies2 == SPECIES_EGG)
return UR_TRADE_MSG_EGG_CANT_BE_TRADED;
2019-01-05 19:25:46 +01:00
2019-10-04 23:24:03 +02:00
if (!IsSpeciesInHoennDex(playerSpecies2))
return UR_TRADE_MSG_MON_CANT_BE_TRADED_2;
2018-12-25 03:18:06 +01:00
2019-10-04 23:24:03 +02:00
if (!IsSpeciesInHoennDex(partnerSpecies))
return UR_TRADE_MSG_PARTNERS_MON_CANT_BE_TRADED;
2018-12-25 03:18:06 +01:00
}
// If the partner doesn't have the National Dex then the player's offer has to be a Hoenn Pokémon
2019-10-04 23:24:03 +02:00
if (!partnerHasNationalDex && !IsSpeciesInHoennDex(playerSpecies2))
return UR_TRADE_MSG_PARTNER_CANT_ACCEPT_MON;
2018-12-25 03:18:06 +01:00
// Trade is allowed
2019-10-04 23:24:03 +02:00
return UR_TRADE_MSG_NONE;
2018-12-25 03:18:06 +01:00
}
2018-12-27 09:05:09 +01:00
2022-06-01 18:41:57 +02:00
int CanRegisterMonForTradingBoard(struct RfuGameCompatibilityData player, u16 species2, u16 species, bool8 isEventLegal)
2018-12-27 09:05:09 +01:00
{
bool8 hasNationalDex = player.hasNationalDex;
2018-12-27 09:05:09 +01:00
if (IsDeoxysOrMewUntradable(species, isEventLegal))
return CANT_REGISTER_MON;
2018-12-27 09:05:09 +01:00
if (hasNationalDex)
return CAN_REGISTER_MON;
2018-12-27 09:05:09 +01:00
// Eggs can only be traded if the player has the National Dex
2019-10-04 13:42:45 +02:00
if (species2 == SPECIES_EGG)
return CANT_REGISTER_EGG;
2018-12-27 09:05:09 +01:00
2019-10-04 13:42:45 +02:00
if (IsSpeciesInHoennDex(species2))
return CAN_REGISTER_MON;
2018-12-27 09:05:09 +01:00
return CANT_REGISTER_MON;
2018-12-27 09:05:09 +01:00
}
2019-10-18 01:22:03 +02:00
// Spin Trade wasnt fully implemented, but this checks if a mon would be valid to Spin Trade
// Unlike later generations, this version of Spin Trade isnt only for Eggs
int CanSpinTradeMon(struct Pokemon *mon, u16 monIdx)
2018-12-27 09:05:09 +01:00
{
2019-10-11 00:46:42 +02:00
int i, version, versions, canTradeAnyMon, numMonsLeft;
2018-12-31 09:22:21 +01:00
int speciesArray[PARTY_SIZE];
2018-12-27 09:05:09 +01:00
2019-10-10 09:50:51 +02:00
// Make Eggs not count for numMonsLeft
2018-12-27 09:05:09 +01:00
for (i = 0; i < gPlayerPartyCount; i++)
{
2018-12-31 09:22:21 +01:00
speciesArray[i] = GetMonData(&mon[i], MON_DATA_SPECIES2);
if (speciesArray[i] == SPECIES_EGG)
2019-10-10 09:50:51 +02:00
speciesArray[i] = SPECIES_NONE;
2018-12-27 09:05:09 +01:00
}
2019-10-10 09:50:51 +02:00
versions = 0;
2019-12-20 23:15:41 +01:00
canTradeAnyMon = TRUE;
2018-12-27 09:05:09 +01:00
for (i = 0; i < GetLinkPlayerCount(); i++)
{
version = gLinkPlayers[i].version & 0xFF;
if (version == VERSION_FIRE_RED ||
version == VERSION_LEAF_GREEN)
versions = 0;
else
versions |= 1;
}
for (i = 0; i < GetLinkPlayerCount(); i++)
{
struct LinkPlayer *player = &gLinkPlayers[i];
2019-10-09 11:56:44 +02:00
2019-10-10 09:50:51 +02:00
// Does player not have National Dex
2019-12-20 23:15:41 +01:00
do
{
if (!(player->progressFlags & 0xF))
canTradeAnyMon = FALSE;
2019-01-05 19:25:46 +01:00
2019-12-20 23:15:41 +01:00
if (versions && (player->progressFlags / 16))
canTradeAnyMon = FALSE;
} while (0);
2018-12-27 09:05:09 +01:00
}
2019-10-11 00:46:42 +02:00
if (canTradeAnyMon == FALSE)
2018-12-27 09:05:09 +01:00
{
2018-12-31 09:22:21 +01:00
if (!IsSpeciesInHoennDex(speciesArray[monIdx]))
2019-10-10 09:50:51 +02:00
return CANT_TRADE_NATIONAL;
2019-01-05 19:25:46 +01:00
2018-12-31 09:22:21 +01:00
if (speciesArray[monIdx] == SPECIES_NONE)
2019-10-18 01:22:03 +02:00
return CANT_TRADE_EGG_YET;
2018-12-27 09:05:09 +01:00
}
2019-10-10 09:50:51 +02:00
numMonsLeft = 0;
2018-12-27 09:05:09 +01:00
for (i = 0; i < gPlayerPartyCount; i++)
{
2018-12-31 09:22:21 +01:00
if (monIdx != i)
2019-10-10 09:50:51 +02:00
numMonsLeft += speciesArray[i];
2018-12-27 09:05:09 +01:00
}
2019-10-10 09:50:51 +02:00
if (!numMonsLeft)
return CANT_TRADE_LAST_MON;
2018-12-27 09:05:09 +01:00
else
2019-10-10 09:50:51 +02:00
return CAN_TRADE_MON;
2018-12-27 09:05:09 +01:00
}
static void SpriteCB_LinkMonGlow(struct Sprite *sprite)
2018-12-27 09:05:09 +01:00
{
if (++sprite->data[0] == 10)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_BALL);
2018-12-27 09:05:09 +01:00
sprite->data[0] = 0;
}
}
static void SpriteCB_LinkMonGlowWireless(struct Sprite *sprite)
2018-12-27 09:05:09 +01:00
{
if (!sprite->invisible && ++sprite->data[0] == 10)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_M_SWAGGER2);
2018-12-27 09:05:09 +01:00
sprite->data[0] = 0;
}
}
static void SpriteCB_LinkMonShadow(struct Sprite *sprite)
2018-12-27 09:05:09 +01:00
{
if (!sprite->data[1])
{
if (++sprite->data[0] == 12)
sprite->data[0] = 0;
LoadPalette(&sLinkMonShadow_Pal[sprite->data[0]], (sprite->oam.paletteNum + 16) * 16 + 4, 2);
2018-12-27 09:05:09 +01:00
}
}
// Move cable down offscreen
static void SpriteCB_CableEndSending(struct Sprite *sprite)
2018-12-27 09:05:09 +01:00
{
sprite->data[0]++;
2021-07-07 15:11:52 +02:00
sprite->y2++;
2018-12-27 09:05:09 +01:00
if (sprite->data[0] == 10)
DestroySprite(sprite);
}
// Move cable up onscreen
static void SpriteCB_CableEndReceiving(struct Sprite *sprite)
2018-12-27 09:05:09 +01:00
{
sprite->data[0]++;
2021-07-07 15:11:52 +02:00
sprite->y2--;
2018-12-27 09:05:09 +01:00
if (sprite->data[0] == 10)
DestroySprite(sprite);
}
static void SpriteCB_GbaScreen(struct Sprite *sprite)
2018-12-27 09:05:09 +01:00
{
if (++sprite->data[0] == 15)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_M_MINIMIZE);
2018-12-27 09:05:09 +01:00
sprite->data[0] = 0;
}
}
2019-10-05 16:41:37 +02:00
static void SetTradeBGAffine(void)
2018-12-27 09:05:09 +01:00
{
struct BgAffineDstData affine;
2019-10-09 11:56:44 +02:00
DoBgAffineSet(&affine, sTradeData->texX * 0x100, sTradeData->texY * 0x100, sTradeData->scrX, sTradeData->scrY, sTradeData->sXY, sTradeData->sXY, sTradeData->alpha);
2018-12-27 09:05:09 +01:00
SetGpuReg(REG_OFFSET_BG2PA, affine.pa);
SetGpuReg(REG_OFFSET_BG2PB, affine.pb);
SetGpuReg(REG_OFFSET_BG2PC, affine.pc);
SetGpuReg(REG_OFFSET_BG2PD, affine.pd);
SetGpuReg(REG_OFFSET_BG2X_L, affine.dx);
SetGpuReg(REG_OFFSET_BG2X_H, affine.dx >> 16);
SetGpuReg(REG_OFFSET_BG2Y_L, affine.dy);
SetGpuReg(REG_OFFSET_BG2Y_H, affine.dy >> 16);
}
2019-10-05 16:41:37 +02:00
static void SetTradeGpuRegs(void)
2018-12-27 09:05:09 +01:00
{
u16 dispcnt;
2019-10-09 11:56:44 +02:00
SetGpuReg(REG_OFFSET_BG1VOFS, sTradeData->bg1vofs);
SetGpuReg(REG_OFFSET_BG1HOFS, sTradeData->bg1hofs);
2018-12-27 09:05:09 +01:00
dispcnt = GetGpuReg(REG_OFFSET_DISPCNT);
2018-12-31 09:22:21 +01:00
if ((dispcnt & 7) == DISPCNT_MODE_0)
2018-12-27 09:05:09 +01:00
{
2019-10-09 11:56:44 +02:00
SetGpuReg(REG_OFFSET_BG2VOFS, sTradeData->bg2vofs);
SetGpuReg(REG_OFFSET_BG2HOFS, sTradeData->bg2hofs);
2018-12-27 09:05:09 +01:00
}
else
{
2019-10-05 16:41:37 +02:00
SetTradeBGAffine();
2018-12-27 09:05:09 +01:00
}
}
2019-10-05 16:41:37 +02:00
static void VBlankCB_Trade(void)
2018-12-27 09:05:09 +01:00
{
2019-10-05 16:41:37 +02:00
SetTradeGpuRegs();
2018-12-27 09:05:09 +01:00
LoadOam();
ProcessSpriteCopyRequests();
TransferPlttBuffer();
}
2019-10-05 16:41:37 +02:00
static void ClearLinkTimeoutCounter(void)
2018-12-27 09:05:09 +01:00
{
2019-10-09 11:56:44 +02:00
sTradeData->linkTimeoutCounter = 0;
sTradeData->linkTimeoutZero1 = 0;
sTradeData->linkTimeoutZero2 = 0;
2018-12-27 09:05:09 +01:00
}
2019-10-05 16:41:37 +02:00
static void CheckForLinkTimeout(void)
2018-12-27 09:05:09 +01:00
{
if (sTradeData->linkTimeoutZero1 == sTradeData->linkTimeoutZero2)
2019-10-09 11:56:44 +02:00
sTradeData->linkTimeoutCounter++;
2018-12-27 09:05:09 +01:00
else
2019-10-09 11:56:44 +02:00
sTradeData->linkTimeoutCounter = 0;
2018-12-27 09:05:09 +01:00
2019-10-09 11:56:44 +02:00
if (sTradeData->linkTimeoutCounter > LINK_TRADE_TIMEOUT)
2018-12-27 09:05:09 +01:00
{
CloseLink();
SetMainCallback2(CB2_LinkError);
2019-10-09 11:56:44 +02:00
sTradeData->linkTimeoutCounter = 0;
sTradeData->linkTimeoutZero2 = 0;
sTradeData->linkTimeoutZero1 = 0;
2018-12-27 09:05:09 +01:00
}
sTradeData->linkTimeoutZero2 = sTradeData->linkTimeoutZero1;
2018-12-27 09:05:09 +01:00
}
2019-10-04 13:42:45 +02:00
static u32 TradeGetMultiplayerId(void)
2018-12-27 09:05:09 +01:00
{
2018-12-31 09:22:21 +01:00
if (gReceivedRemoteLinkPlayers)
2018-12-27 09:05:09 +01:00
return GetMultiplayerId();
return 0;
}
static void LoadTradeMonPic(u8 whichParty, u8 state)
2018-12-27 09:05:09 +01:00
{
2018-12-31 09:22:21 +01:00
int pos = 0;
2018-12-27 09:05:09 +01:00
struct Pokemon *mon = NULL;
u16 species;
u32 personality;
if (whichParty == TRADE_PLAYER)
2018-12-27 09:05:09 +01:00
{
2019-10-04 01:39:37 +02:00
mon = &gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]];
pos = B_POSITION_OPPONENT_LEFT;
2018-12-27 09:05:09 +01:00
}
if (whichParty == TRADE_PARTNER)
2018-12-27 09:05:09 +01:00
{
2019-10-04 01:39:37 +02:00
mon = &gEnemyParty[gSelectedTradeMonPositions[TRADE_PARTNER] % PARTY_SIZE];
pos = B_POSITION_OPPONENT_RIGHT;
2018-12-27 09:05:09 +01:00
}
switch (state)
2018-12-27 09:05:09 +01:00
{
case 0:
species = GetMonData(mon, MON_DATA_SPECIES2);
personality = GetMonData(mon, MON_DATA_PERSONALITY);
2018-12-27 09:05:09 +01:00
if (whichParty == TRADE_PLAYER)
2021-10-03 05:47:59 +02:00
HandleLoadSpecialPokePic_2(&gMonFrontPicTable[species], gMonSpritesGfxPtr->sprites.ptr[B_POSITION_OPPONENT_LEFT], species, personality);
else
2021-10-03 05:47:59 +02:00
HandleLoadSpecialPokePic_DontHandleDeoxys(&gMonFrontPicTable[species], gMonSpritesGfxPtr->sprites.ptr[whichParty * 2 + B_POSITION_OPPONENT_LEFT], species, personality);
2018-12-28 09:11:41 +01:00
LoadCompressedSpritePalette(GetMonSpritePalStruct(mon));
sTradeData->monSpecies[whichParty] = species;
sTradeData->monPersonalities[whichParty] = personality;
break;
case 1:
SetMultiuseSpriteTemplateToPokemon(GetMonSpritePalStruct(mon)->tag, pos);
sTradeData->monSpriteIds[whichParty] = CreateSprite(&gMultiuseSpriteTemplate, 120, 60, 6);
gSprites[sTradeData->monSpriteIds[whichParty]].invisible = TRUE;
gSprites[sTradeData->monSpriteIds[whichParty]].callback = SpriteCallbackDummy;
break;
2018-12-27 09:05:09 +01:00
}
}
2019-10-05 16:41:37 +02:00
void CB2_LinkTrade(void)
2018-12-27 09:05:09 +01:00
{
switch (gMain.state)
{
case 0:
if (!gReceivedRemoteLinkPlayers)
{
2020-06-04 00:00:53 +02:00
gLinkType = LINKTYPE_TRADE_DISCONNECTED;
CloseLink();
}
sTradeData = AllocZeroed(sizeof(*sTradeData));
AllocateMonSpritesGfx();
ResetTasks();
ResetSpriteData();
FreeAllSpritePalettes();
SetVBlankCallback(VBlankCB_Trade);
InitTradeBgInternal();
ClearLinkTimeoutCounter();
gMain.state++;
sTradeData->neverRead_8C = 0;
sTradeData->state = 0;
sTradeData->isLinkTrade = TRUE;
sTradeData->texX = 64;
sTradeData->texY = 64;
sTradeData->neverRead_D8 = 0;
sTradeData->neverRead_DA = 0;
sTradeData->scrX = 120;
sTradeData->scrY = 80;
sTradeData->sXY = 256;
sTradeData->alpha = 0;
break;
case 1:
if (!gReceivedRemoteLinkPlayers)
{
sTradeData->isCableTrade = TRUE;
OpenLink();
2018-12-27 09:05:09 +01:00
gMain.state++;
sTradeData->timer = 0;
}
else
{
gMain.state = 4;
}
break;
case 2:
if (++sTradeData->timer > 60)
{
sTradeData->timer = 0;
gMain.state++;
}
break;
case 3:
if (IsLinkMaster())
{
if (GetLinkPlayerCount_2() >= GetSavedPlayerCount())
2018-12-27 09:05:09 +01:00
{
if (++sTradeData->timer > 30)
2018-12-27 09:05:09 +01:00
{
CheckShouldAdvanceLinkState();
gMain.state++;
2018-12-27 09:05:09 +01:00
}
}
else
{
CheckForLinkTimeout();
2018-12-27 09:05:09 +01:00
}
}
else
{
2018-12-27 09:05:09 +01:00
gMain.state++;
}
break;
case 4:
CheckForLinkTimeout();
if (gReceivedRemoteLinkPlayers == TRUE && IsLinkPlayerDataExchangeComplete() == TRUE)
2018-12-27 09:05:09 +01:00
gMain.state++;
break;
case 5:
sTradeData->playerLinkFlagFinishTrade = 0;
sTradeData->partnerLinkFlagFinishTrade = 0;
2019-10-10 09:50:51 +02:00
sTradeData->sendTradeFinishState = 0;
LoadTradeMonPic(TRADE_PLAYER, 0);
gMain.state++;
break;
case 6:
LoadTradeMonPic(TRADE_PLAYER, 1);
gMain.state++;
break;
case 7:
LoadTradeMonPic(TRADE_PARTNER, 0);
gMain.state++;
break;
case 8:
LoadTradeMonPic(TRADE_PARTNER, 1);
2019-10-10 09:50:51 +02:00
LinkTradeDrawWindow();
gMain.state++;
break;
case 9:
LoadTradeSequenceSpriteSheetsAndPalettes();
LoadSpriteSheet(&sPokeBallSpriteSheet);
LoadSpritePalette(&sPokeBallSpritePalette);
gMain.state++;
break;
case 10:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
ShowBg(0);
gMain.state++;
break;
case 11:
2019-10-10 09:50:51 +02:00
InitTradeSequenceBgGpuRegs();
BufferTradeSceneStrings();
gMain.state++;
break;
case 12:
if (!gPaletteFade.active)
{
if (gWirelessCommType)
{
LoadWirelessStatusIndicatorSpriteGfx();
CreateWirelessStatusIndicatorSprite(0, 0);
}
SetMainCallback2(CB2_UpdateLinkTrade);
}
break;
}
RunTasks();
RunTextPrinters();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
2019-10-10 09:50:51 +02:00
void InitTradeSequenceBgGpuRegs(void)
{
SetTradeSequenceBgGpuRegs(5);
SetTradeSequenceBgGpuRegs(0);
}
2019-10-10 09:50:51 +02:00
void LinkTradeDrawWindow(void)
{
FillWindowPixelBuffer(0, PIXEL_FILL(15));
PutWindowTilemap(0);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(0, COPYWIN_FULL);
}
static void InitTradeBgInternal(void)
2018-12-27 09:05:09 +01:00
{
SetGpuReg(REG_OFFSET_DISPCNT, 0);
ResetBgsAndClearDma3BusyFlags(0);
2019-10-07 04:58:25 +02:00
InitBgsFromTemplates(0, sTradeSequenceBgTemplates, ARRAY_COUNT(sTradeSequenceBgTemplates));
2021-11-04 04:02:06 +01:00
ChangeBgX(0, 0, BG_COORD_SET);
ChangeBgY(0, 0, BG_COORD_SET);
2021-03-29 15:38:19 +02:00
SetBgTilemapBuffer(0, Alloc(BG_SCREEN_SIZE));
SetBgTilemapBuffer(1, Alloc(BG_SCREEN_SIZE));
SetBgTilemapBuffer(3, Alloc(BG_SCREEN_SIZE));
2018-12-27 09:05:09 +01:00
DeactivateAllTextPrinters();
DecompressAndLoadBgGfxUsingHeap(0, gBattleTextboxTiles, 0, 0, 0);
LZDecompressWram(gBattleTextboxTilemap, gDecompressionBuffer);
CopyToBgTilemapBuffer(0, gDecompressionBuffer, 0x800, 0);
LoadCompressedPalette(gBattleTextboxPalette, 0, 0x20);
2019-10-07 04:58:25 +02:00
InitWindows(sTradeSequenceWindowTemplates);
2018-12-27 09:05:09 +01:00
DecompressAndLoadBgGfxUsingHeap(0, gBattleTextboxTiles, 0, 0, 0);
LZDecompressWram(gBattleTextboxTilemap, gDecompressionBuffer);
CopyToBgTilemapBuffer(0, gDecompressionBuffer, 0x800, 0);
LoadCompressedPalette(gBattleTextboxPalette, 0, 0x20);
}
2019-10-04 13:42:45 +02:00
static void CB2_InGameTrade(void)
2018-12-27 09:05:09 +01:00
{
2018-12-31 09:22:21 +01:00
u8 otName[11];
2018-12-27 09:05:09 +01:00
switch (gMain.state)
{
case 0:
gSelectedTradeMonPositions[TRADE_PLAYER] = gSpecialVar_0x8005;
gSelectedTradeMonPositions[TRADE_PARTNER] = PARTY_SIZE;
StringCopy(gLinkPlayers[0].name, gSaveBlock2Ptr->playerName);
GetMonData(&gEnemyParty[0], MON_DATA_OT_NAME, otName);
StringCopy(gLinkPlayers[1].name, otName);
gLinkPlayers[0].language = GAME_LANGUAGE;
gLinkPlayers[1].language = GetMonData(&gEnemyParty[0], MON_DATA_LANGUAGE);
sTradeData = AllocZeroed(sizeof(*sTradeData));
AllocateMonSpritesGfx();
ResetTasks();
ResetSpriteData();
FreeAllSpritePalettes();
SetVBlankCallback(VBlankCB_Trade);
InitTradeBgInternal();
sTradeData->isLinkTrade = FALSE;
sTradeData->neverRead_8C = 0;
sTradeData->state = 0;
sTradeData->texX = 64;
sTradeData->texY = 64;
sTradeData->neverRead_D8 = 0;
sTradeData->neverRead_DA = 0;
sTradeData->scrX = 120;
sTradeData->scrY = 80;
sTradeData->sXY = 256;
sTradeData->alpha = 0;
sTradeData->timer = 0;
gMain.state = 5;
break;
case 5:
LoadTradeMonPic(TRADE_PLAYER, 0);
gMain.state++;
break;
case 6:
LoadTradeMonPic(TRADE_PLAYER, 1);
gMain.state++;
break;
case 7:
LoadTradeMonPic(TRADE_PARTNER, 0);
ShowBg(0);
gMain.state++;
break;
case 8:
LoadTradeMonPic(TRADE_PARTNER, 1);
FillWindowPixelBuffer(0, PIXEL_FILL(15));
PutWindowTilemap(0);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(0, COPYWIN_FULL);
gMain.state++;
break;
case 9:
LoadTradeSequenceSpriteSheetsAndPalettes();
LoadSpriteSheet(&sPokeBallSpriteSheet);
LoadSpritePalette(&sPokeBallSpritePalette);
gMain.state++;
break;
case 10:
ShowBg(0);
gMain.state++;
break;
case 11:
SetTradeSequenceBgGpuRegs(5);
SetTradeSequenceBgGpuRegs(0);
BufferTradeSceneStrings();
gMain.state++;
break;
case 12:
SetMainCallback2(CB2_UpdateInGameTrade);
break;
2018-12-27 09:05:09 +01:00
}
RunTasks();
RunTextPrinters();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
2019-10-04 13:42:45 +02:00
static void UpdatePokedexForReceivedMon(u8 partyIdx)
2018-12-27 09:05:09 +01:00
{
2018-12-31 09:22:21 +01:00
struct Pokemon *mon = &gPlayerParty[partyIdx];
2018-12-28 09:11:41 +01:00
2018-12-27 09:05:09 +01:00
if (!GetMonData(mon, MON_DATA_IS_EGG))
{
2018-12-28 09:11:41 +01:00
u16 species = GetMonData(mon, MON_DATA_SPECIES, NULL);
u32 personality = GetMonData(mon, MON_DATA_PERSONALITY, NULL);
2018-12-27 09:05:09 +01:00
species = SpeciesToNationalPokedexNum(species);
GetSetPokedexFlag(species, FLAG_SET_SEEN);
HandleSetPokedexFlag(species, FLAG_SET_CAUGHT, personality);
}
}
2019-10-04 13:42:45 +02:00
// Functionally nop after commented code
static void TryEnableNationalDexFromLinkPartner(void)
2018-12-27 09:05:09 +01:00
{
2018-12-28 09:11:41 +01:00
u8 mpId = GetMultiplayerId();
// Originally in Ruby but commented out
/*if (gLinkPlayers[mpId ^ 1].lp_field_2 == 0x8000)
EnableNationalPokedex();*/
2018-12-27 09:05:09 +01:00
}
2019-10-04 13:42:45 +02:00
static void TradeMons(u8 playerPartyIdx, u8 partnerPartyIdx)
2018-12-27 09:05:09 +01:00
{
u8 friendship;
2019-10-04 13:42:45 +02:00
struct Pokemon *playerMon = &gPlayerParty[playerPartyIdx];
2018-12-31 09:22:21 +01:00
u16 playerMail = GetMonData(playerMon, MON_DATA_MAIL);
2018-12-27 09:05:09 +01:00
2019-10-04 13:42:45 +02:00
struct Pokemon *partnerMon = &gEnemyParty[partnerPartyIdx];
2018-12-31 09:22:21 +01:00
u16 partnerMail = GetMonData(partnerMon, MON_DATA_MAIL);
2021-02-20 21:15:38 +01:00
if (playerMail != MAIL_NONE)
2021-10-23 16:55:46 +02:00
ClearMail(&gSaveBlock1Ptr->mail[playerMail]);
2018-12-27 09:05:09 +01:00
2021-10-23 16:55:46 +02:00
SWAP(*playerMon, *partnerMon, sTradeData->tempMon);
2018-12-27 09:05:09 +01:00
friendship = 70;
2018-12-31 09:22:21 +01:00
if (!GetMonData(playerMon, MON_DATA_IS_EGG))
SetMonData(playerMon, MON_DATA_FRIENDSHIP, &friendship);
2018-12-27 09:05:09 +01:00
2021-02-20 21:15:38 +01:00
if (partnerMail != MAIL_NONE)
2021-10-23 16:55:46 +02:00
GiveMailToMon(playerMon, &gTradeMail[partnerMail]);
2018-12-27 09:05:09 +01:00
2019-10-04 13:42:45 +02:00
UpdatePokedexForReceivedMon(playerPartyIdx);
2018-12-31 09:22:21 +01:00
if (gReceivedRemoteLinkPlayers)
TryEnableNationalDexFromLinkPartner();
2018-12-27 09:05:09 +01:00
}
2019-10-10 09:50:51 +02:00
static void TrySendTradeFinishData(void)
2018-12-27 09:05:09 +01:00
{
2019-10-10 09:50:51 +02:00
switch (sTradeData->sendTradeFinishState)
2018-12-27 09:05:09 +01:00
{
case 1:
if (IsLinkTaskFinished())
{
SendBlock(BitmaskAllOtherLinkPlayers(), sTradeData->linkData, sizeof(sTradeData->linkData));
2019-10-10 09:50:51 +02:00
sTradeData->sendTradeFinishState++;
}
// fallthrough
case 2:
2019-10-10 09:50:51 +02:00
sTradeData->sendTradeFinishState = 0;
break;
2018-12-27 09:05:09 +01:00
}
}
static void CB2_UpdateInGameTrade(void)
2018-12-27 09:05:09 +01:00
{
2019-10-05 16:41:37 +02:00
AnimateTradeSequence();
2018-12-27 09:05:09 +01:00
RunTasks();
RunTextPrinters();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void SetTradeSequenceBgGpuRegs(u8 state)
2018-12-27 09:05:09 +01:00
{
switch (state)
2018-12-27 09:05:09 +01:00
{
case 0:
sTradeData->bg2vofs = 0;
sTradeData->bg2hofs = 180;
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG0_ON |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
SetGpuReg(REG_OFFSET_BG2CNT, BGCNT_PRIORITY(2) |
BGCNT_CHARBASE(1) |
BGCNT_16COLOR |
BGCNT_SCREENBASE(18) |
BGCNT_TXT512x256);
LoadPalette(gTradeGba2_Pal, 16, 0x60);
DmaCopyLarge16(3, gTradeGba_Gfx, (void *) BG_CHAR_ADDR(1), 0x1420, 0x1000);
DmaCopy16Defvars(3, gTradePlatform_Tilemap, (void *) BG_SCREEN_ADDR(18), 0x1000);
break;
case 1:
sTradeData->bg1hofs = 0;
sTradeData->bg1vofs = 348;
SetGpuReg(REG_OFFSET_BG1VOFS, 348);
SetGpuReg(REG_OFFSET_BG1CNT, BGCNT_PRIORITY(2) |
BGCNT_CHARBASE(0) |
BGCNT_16COLOR |
BGCNT_SCREENBASE(5) |
BGCNT_TXT256x512);
SetGpuReg(REG_OFFSET_BG2CNT, BGCNT_PRIORITY(2) |
BGCNT_CHARBASE(1) |
BGCNT_16COLOR |
BGCNT_SCREENBASE(18) |
BGCNT_TXT256x512);
if (sTradeData->isCableTrade)
{
DmaCopy16Defvars(3, sGbaCable_Tilemap, (void *) BG_SCREEN_ADDR(5), 0x1000);
}
else
{
DmaCopy16Defvars(3, sGbaWireless_Tilemap, (void *) BG_SCREEN_ADDR(5), 0x1000);
}
2018-12-27 09:05:09 +01:00
DmaCopyLarge16(3, gTradeGba_Gfx, (void *) BG_CHAR_ADDR(0), 0x1420, 0x1000);
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_OBJ_ON);
break;
case 2:
sTradeData->bg1vofs = 0;
sTradeData->bg1hofs = 0;
if (!sTradeData->isCableTrade)
{
2018-12-27 09:05:09 +01:00
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_1 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
2018-12-27 09:05:09 +01:00
DISPCNT_OBJ_ON);
LZ77UnCompVram(sCrossingHighlightWireless_Tilemap, (void *) BG_SCREEN_ADDR(5));
BlendPalettes(0x8, 16, RGB_BLACK);
}
else
{
2018-12-27 09:05:09 +01:00
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_1 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
2018-12-27 09:05:09 +01:00
DISPCNT_OBJ_ON);
DmaCopy16Defvars(3, sCrossingHighlightCable_Tilemap, (void *) BG_SCREEN_ADDR(5), 0x800);
BlendPalettes(0x1, 16, RGB_BLACK);
}
break;
case 3:
LoadPalette(sWirelessSignalNone_Pal, 48, 0x20);
LZ77UnCompVram(sWirelessSignal_Gfx, (void *) BG_CHAR_ADDR(1));
LZ77UnCompVram(sWirelessSignal_Tilemap, (void *) BG_SCREEN_ADDR(18));
sTradeData->bg2vofs = 80;
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
break;
case 4:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_1 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
SetGpuReg(REG_OFFSET_BG2CNT, BGCNT_PRIORITY(3) |
BGCNT_CHARBASE(1) |
BGCNT_256COLOR |
BGCNT_SCREENBASE(18) |
BGCNT_AFF128x128);
sTradeData->texX = 64;
sTradeData->texY = 92;
sTradeData->sXY = 32;
sTradeData->gbaScale = 1024;
sTradeData->alpha = 0;
DmaCopyLarge16(3, sGbaAffine_Gfx, (void *) BG_CHAR_ADDR(1), 0x2840, 0x1000);
if (sTradeData->isCableTrade)
{
DmaCopy16Defvars(3, sGbaCable_AffineTilemap, (void *) BG_SCREEN_ADDR(18), 0x100);
}
else
{
DmaCopy16Defvars(3, sGbaWireless_AffineTilemap, (void *) BG_SCREEN_ADDR(18), 0x100);
}
break;
case 5:
sTradeData->bg1vofs = 0;
sTradeData->bg1hofs = 0;
break;
case 6:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_1 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
SetGpuReg(REG_OFFSET_BG2CNT, BGCNT_PRIORITY(3) |
BGCNT_CHARBASE(1) |
BGCNT_256COLOR |
BGCNT_SCREENBASE(18) |
BGCNT_AFF128x128);
sTradeData->texX = 64;
sTradeData->texY = 92;
sTradeData->sXY = 256;
sTradeData->gbaScale = 128;
sTradeData->scrX = 120;
sTradeData->scrY = 80;
sTradeData->alpha = 0;
DmaCopyLarge16(3, sGbaAffine_Gfx, (void *) BG_CHAR_ADDR(1), 0x2840, 0x1000);
if (sTradeData->isCableTrade)
{
DmaCopy16Defvars(3, sGbaCable_AffineTilemap, (void *) BG_SCREEN_ADDR(18), 0x100);
}
else
{
DmaCopy16Defvars(3, sGbaWireless_AffineTilemap, (void *) BG_SCREEN_ADDR(18), 0x100);
}
break;
case 7:
sTradeData->bg2vofs = 0;
sTradeData->bg2hofs = 0;
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BG2CNT, BGCNT_PRIORITY(2) |
BGCNT_CHARBASE(1) |
BGCNT_16COLOR |
BGCNT_SCREENBASE(18) |
BGCNT_TXT512x256);
LoadPalette(gTradeGba2_Pal, 16, 0x60);
DmaCopyLarge16(3, gTradeGba_Gfx, (void *) BG_CHAR_ADDR(1), 0x1420, 0x1000);
DmaCopy16Defvars(3, gTradePlatform_Tilemap, (void *) BG_SCREEN_ADDR(18), 0x1000);
break;
2018-12-27 09:05:09 +01:00
}
}
2018-12-28 09:11:41 +01:00
2019-10-05 16:41:37 +02:00
static void LoadTradeSequenceSpriteSheetsAndPalettes(void)
2018-12-28 09:11:41 +01:00
{
LoadSpriteSheet(&sSpriteSheet_LinkMonGlow);
LoadSpriteSheet(&sSpriteSheet_LinkMonShadow);
LoadSpriteSheet(&sSpriteSheet_CableEnd);
LoadSpriteSheet(&sSpriteSheet_GbaScreen);
LoadSpritePalette(&sSpritePalette_LinkMon);
LoadSpritePalette(&sSpritePalette_Gba);
2018-12-28 09:11:41 +01:00
}
// Buffers "[Pokemon] will be sent to [Trainer]" strings
static void BufferTradeSceneStrings(void)
2018-12-28 09:11:41 +01:00
{
u8 mpId;
u8 name[20];
const struct InGameTrade *ingameTrade;
2019-10-09 11:56:44 +02:00
if (sTradeData->isLinkTrade)
2018-12-28 09:11:41 +01:00
{
mpId = GetMultiplayerId();
StringCopy(gStringVar1, gLinkPlayers[mpId ^ 1].name);
2019-10-04 01:39:37 +02:00
GetMonData(&gEnemyParty[gSelectedTradeMonPositions[TRADE_PARTNER] % PARTY_SIZE], MON_DATA_NICKNAME, name);
StringCopy_Nickname(gStringVar3, name);
2019-10-04 01:39:37 +02:00
GetMonData(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], MON_DATA_NICKNAME, name);
StringCopy_Nickname(gStringVar2, name);
2018-12-28 09:11:41 +01:00
}
else
{
ingameTrade = &sIngameTrades[gSpecialVar_0x8004];
2018-12-28 09:11:41 +01:00
StringCopy(gStringVar1, ingameTrade->otName);
StringCopy_Nickname(gStringVar3, ingameTrade->nickname);
2018-12-28 09:11:41 +01:00
GetMonData(&gPlayerParty[gSpecialVar_0x8005], MON_DATA_NICKNAME, name);
StringCopy_Nickname(gStringVar2, name);
2018-12-28 09:11:41 +01:00
}
}
// returns TRUE if it finished a link trade, FALSE if it finished an in-game trade or if sequence is still going
2019-10-05 16:41:37 +02:00
static bool8 AnimateTradeSequence(void)
2018-12-28 09:11:41 +01:00
{
2019-10-09 11:56:44 +02:00
if (sTradeData->isCableTrade)
2019-10-05 16:41:37 +02:00
return AnimateTradeSequenceCable();
2018-12-28 09:11:41 +01:00
else
2019-10-05 16:41:37 +02:00
return AnimateTradeSequenceWireless();
2018-12-28 09:11:41 +01:00
}
// Below are the states for the main switch in AnimateTradeSequenceCable and AnimateTradeSequenceWireless
// When AnimateTradeSequenceWireless has a unique version of a
// state used by AnimateTradeSequenceCable, it adds the below modifier
#define TS_WIRELESS_STATE 100
enum {
TS_STATE_START,
TS_STATE_MON_SLIDE_IN,
// 2-9 unused
TS_STATE_SEND_MSG = 10,
TS_STATE_BYE_BYE,
TS_STATE_POKEBALL_DEPART,
TS_STATE_POKEBALL_DEPART_WAIT,
TS_STATE_FADE_OUT_TO_GBA_SEND,
// 15-19 unused
TS_STATE_WAIT_FADE_OUT_TO_GBA_SEND = 20,
TS_STATE_FADE_IN_TO_GBA_SEND,
TS_STATE_WAIT_FADE_IN_TO_GBA_SEND,
TS_STATE_GBA_ZOOM_OUT,
TS_STATE_GBA_FLASH_SEND,
TS_STATE_GBA_STOP_FLASH_SEND,
TS_STATE_PAN_AWAY_GBA,
TS_STATE_CREATE_LINK_MON_LEAVING,
TS_STATE_LINK_MON_TRAVEL_OUT,
TS_STATE_FADE_OUT_TO_CROSSING,
TS_STATE_WAIT_FADE_OUT_TO_CROSSING,
TS_STATE_FADE_IN_TO_CROSSING,
TS_STATE_WAIT_FADE_IN_TO_CROSSING,
TS_STATE_CROSSING_LINK_MONS_ENTER,
TS_STATE_CROSSING_BLEND_WHITE_1,
TS_STATE_CROSSING_BLEND_WHITE_2,
TS_STATE_CROSSING_BLEND_WHITE_3,
TS_STATE_CROSSING_CREATE_MON_PICS,
TS_STATE_CROSSING_MON_PICS_MOVE,
TS_STATE_CROSSING_LINK_MONS_EXIT,
TS_STATE_CREATE_LINK_MON_ARRIVING,
TS_STATE_FADE_OUT_TO_GBA_RECV,
TS_STATE_WAIT_FADE_OUT_TO_GBA_RECV,
TS_STATE_LINK_MON_TRAVEL_IN,
TS_STATE_PAN_TO_GBA,
TS_STATE_DESTROY_LINK_MON,
TS_STATE_LINK_MON_ARRIVED_DELAY,
TS_STATE_MOVE_GBA_TO_CENTER,
TS_STATE_GBA_FLASH_RECV,
TS_STATE_UNUSED,
TS_STATE_GBA_STOP_FLASH_RECV,
TS_STATE_GBA_ZOOM_IN,
TS_STATE_FADE_OUT_TO_NEW_MON,
// 53-59 unused
TS_STATE_WAIT_FADE_OUT_TO_NEW_MON = 60,
TS_STATE_FADE_IN_TO_NEW_MON,
TS_STATE_WAIT_FADE_IN_TO_NEW_MON,
TS_STATE_POKEBALL_ARRIVE,
TS_STATE_FADE_POKEBALL_TO_NORMAL,
TS_STATE_POKEBALL_ARRIVE_WAIT,
TS_STATE_SHOW_NEW_MON,
TS_STATE_NEW_MON_MSG,
TS_STATE_TAKE_CARE_OF_MON,
TS_STATE_AFTER_NEW_MON_DELAY,
TS_STATE_CHECK_RIBBONS,
TS_STATE_END_LINK_TRADE,
TS_STATE_TRY_EVOLUTION,
TS_STATE_FADE_OUT_END,
TS_STATE_WAIT_FADE_OUT_END,
// Special states
TS_STATE_GBA_FLASH_SEND_WIRELESS = TS_STATE_GBA_FLASH_SEND + TS_WIRELESS_STATE,
TS_STATE_GBA_STOP_FLASH_SEND_WIRELESS,
TS_STATE_WAIT_WIRELESS_SIGNAL_SEND,
TS_STATE_PAN_TO_GBA_WIRELESS = TS_STATE_PAN_TO_GBA + TS_WIRELESS_STATE,
TS_STATE_DESTROY_LINK_MON_WIRELESS,
TS_STATE_WAIT_WIRELESS_SIGNAL_RECV,
TS_STATE_DELAY_FOR_MON_ANIM = 167,
TS_STATE_LINK_MON_TRAVEL_OFFSCREEN = 200,
TS_STATE_WAIT_FOR_MON_CRY = 267,
};
2019-10-05 16:41:37 +02:00
static bool8 AnimateTradeSequenceCable(void)
2018-12-28 09:11:41 +01:00
{
u16 evoTarget;
2019-10-09 11:56:44 +02:00
switch (sTradeData->state)
2018-12-28 09:11:41 +01:00
{
case TS_STATE_START:
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].invisible = FALSE;
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x2 = -180;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 = gMonFrontPicCoords[sTradeData->monSpecies[TRADE_PLAYER]].y_offset;
sTradeData->state++;
sTradeData->cachedMapMusic = GetCurrentMapMusic();
2020-08-21 00:02:00 +02:00
PlayNewMapMusic(MUS_EVOLUTION);
break;
case TS_STATE_MON_SLIDE_IN:
if (sTradeData->bg2hofs > 0)
{
// Sliding
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x2 += 3;
sTradeData->bg2hofs -= 3;
}
else
{
// Pokémon has arrived onscreen
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x2 = 0;
sTradeData->bg2hofs = 0;
sTradeData->state = TS_STATE_SEND_MSG;
}
break;
case TS_STATE_SEND_MSG:
StringExpandPlaceholders(gStringVar4, gText_XWillBeSentToY);
DrawTextOnTradeWindow(0, gStringVar4, 0);
2018-12-28 09:11:41 +01:00
if (sTradeData->monSpecies[TRADE_PLAYER] != SPECIES_EGG)
2021-11-07 19:54:44 +01:00
PlayCry_Normal(sTradeData->monSpecies[TRADE_PLAYER], 0);
2018-12-28 09:11:41 +01:00
sTradeData->state = TS_STATE_BYE_BYE;
sTradeData->timer = 0;
break;
case TS_STATE_BYE_BYE:
if (++sTradeData->timer == 80)
{
sTradeData->releasePokeballSpriteId = CreateTradePokeballSprite(sTradeData->monSpriteIds[0], gSprites[sTradeData->monSpriteIds[0]].oam.paletteNum, 120, 32, 2, 1, 0x14, 0xfffff);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
StringExpandPlaceholders(gStringVar4, gText_ByeByeVar1);
DrawTextOnTradeWindow(0, gStringVar4, 0);
}
break;
case TS_STATE_POKEBALL_DEPART:
if (gSprites[sTradeData->releasePokeballSpriteId].callback == SpriteCallbackDummy)
{
sTradeData->bouncingPokeballSpriteId = CreateSprite(&sSpriteTemplate_Pokeball, 120, 32, 0);
gSprites[sTradeData->bouncingPokeballSpriteId].callback = SpriteCB_BouncingPokeballDepart;
DestroySprite(&gSprites[sTradeData->releasePokeballSpriteId]);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_POKEBALL_DEPART_WAIT:
// The game waits here for the sprite to finish its animation sequence.
break;
case TS_STATE_FADE_OUT_TO_GBA_SEND:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeData->state = TS_STATE_WAIT_FADE_OUT_TO_GBA_SEND;
break;
case TS_STATE_WAIT_FADE_OUT_TO_GBA_SEND:
if (!gPaletteFade.active)
{
SetTradeSequenceBgGpuRegs(4);
FillWindowPixelBuffer(0, PIXEL_FILL(15));
2021-11-03 20:29:18 +01:00
CopyWindowToVram(0, COPYWIN_FULL);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_FADE_IN_TO_GBA_SEND:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_IN_TO_GBA_SEND:
if (!gPaletteFade.active)
sTradeData->state = TS_STATE_GBA_ZOOM_OUT;
break;
case TS_STATE_GBA_ZOOM_OUT:
if (sTradeData->gbaScale > 0x100)
{
sTradeData->gbaScale -= 0x34;
}
else
{
SetTradeSequenceBgGpuRegs(1);
sTradeData->gbaScale = 0x80;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
sTradeData->timer = 0;
}
sTradeData->sXY = 0x8000 / sTradeData->gbaScale;
break;
case TS_STATE_GBA_FLASH_SEND:
if (++sTradeData->timer > 20)
{
SetTradeBGAffine();
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_GbaScreenFlash_Long, 120, 80, 0);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_GBA_STOP_FLASH_SEND:
if (gSprites[sTradeData->connectionSpriteId2].animEnded)
{
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_EFFECT_BLEND |
BLDCNT_TGT2_BG1 |
BLDCNT_TGT2_BG2);
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(12, 4));
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_PAN_AWAY_GBA:
if (--sTradeData->bg1vofs == 316)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
if (sTradeData->bg1vofs == 328)
sTradeData->cableEndSpriteId = CreateSprite(&sSpriteTemplate_CableEnd, 128, 65, 0);
break;
case TS_STATE_CREATE_LINK_MON_LEAVING:
sTradeData->connectionSpriteId1 = CreateSprite(&sSpriteTemplate_LinkMonGlow, 128, 80, 3);
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 128, 80, 0);
StartSpriteAnim(&gSprites[sTradeData->connectionSpriteId2], ANIM_LINKMON_SMALL);
sTradeData->state++;
break;
case TS_STATE_LINK_MON_TRAVEL_OUT:
if ((sTradeData->bg1vofs -= 2) == 166)
sTradeData->state = TS_STATE_LINK_MON_TRAVEL_OFFSCREEN;
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_1 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_OBJ_ON);
break;
case TS_STATE_LINK_MON_TRAVEL_OFFSCREEN:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y -= 2;
gSprites[sTradeData->connectionSpriteId2].y -= 2;
if (gSprites[sTradeData->connectionSpriteId1].y < -8)
sTradeData->state = TS_STATE_FADE_OUT_TO_CROSSING;
break;
case TS_STATE_FADE_OUT_TO_CROSSING:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 0, 16, RGB_BLACK);
sTradeData->state = TS_STATE_WAIT_FADE_OUT_TO_CROSSING;
break;
case TS_STATE_WAIT_FADE_OUT_TO_CROSSING:
if (!gPaletteFade.active)
{
DestroySprite(&gSprites[sTradeData->connectionSpriteId1]);
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
SetTradeSequenceBgGpuRegs(2);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_FADE_IN_TO_CROSSING:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
sTradeData->connectionSpriteId1 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 111, 170, 0);
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 129, -10, 0);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_IN_TO_CROSSING:
if (!gPaletteFade.active)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_WARP_OUT);
sTradeData->state++;
}
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 -= 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
break;
case TS_STATE_CROSSING_LINK_MONS_ENTER:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 -= 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
if (gSprites[sTradeData->connectionSpriteId1].y2 <= -90)
{
gSprites[sTradeData->connectionSpriteId1].data[1] = 1;
gSprites[sTradeData->connectionSpriteId2].data[1] = 1;
sTradeData->state++;
}
break;
case TS_STATE_CROSSING_BLEND_WHITE_1:
BlendPalettes(0x1, 16, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_CROSSING_BLEND_WHITE_2:
BlendPalettes(0x1, 0, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_CROSSING_BLEND_WHITE_3:
BlendPalettes(0x1, 16, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_CROSSING_CREATE_MON_PICS:
if (!IsMonSpriteNotFlipped(sTradeData->monSpecies[TRADE_PLAYER]))
{
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].affineAnims = sAffineAnims_CrossingMonPics;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].oam.affineMode = ST_OAM_AFFINE_DOUBLE;
CalcCenterToCornerVec(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]], SPRITE_SHAPE(64x64), SPRITE_SIZE(64x64), ST_OAM_AFFINE_DOUBLE);
StartSpriteAffineAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]], 0);
}
else
{
StartSpriteAffineAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]], 0);
}
StartSpriteAffineAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]], 0);
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x = 60;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].x = 180;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y = 192;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y = -32;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].invisible = FALSE;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].invisible = FALSE;
sTradeData->state++;
break;
case TS_STATE_CROSSING_MON_PICS_MOVE:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 -= 3;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y2 += 3;
if (gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 < -DISPLAY_HEIGHT
2021-07-07 15:11:52 +02:00
&& gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 >= -DISPLAY_HEIGHT - 3)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_WARP_IN);
}
2021-07-07 15:11:52 +02:00
if (gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 < -222)
{
gSprites[sTradeData->connectionSpriteId1].data[1] = 0;
gSprites[sTradeData->connectionSpriteId2].data[1] = 0;
sTradeData->state++;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].invisible = TRUE;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].invisible = TRUE;
BlendPalettes(0x1, 0, RGB_WHITEALPHA);
}
break;
case TS_STATE_CROSSING_LINK_MONS_EXIT:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 -= 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
if (gSprites[sTradeData->connectionSpriteId1].y2 <= -222)
{
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 0, 16, RGB_BLACK);
sTradeData->state++;
DestroySprite(&gSprites[sTradeData->connectionSpriteId1]);
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
}
break;
case TS_STATE_CREATE_LINK_MON_ARRIVING:
if (!gPaletteFade.active)
{
sTradeData->state++;
SetTradeSequenceBgGpuRegs(1);
sTradeData->bg1vofs = 166;
sTradeData->connectionSpriteId1 = CreateSprite(&sSpriteTemplate_LinkMonGlow, 128, -20, 3);
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 128, -20, 0);
StartSpriteAnim(&gSprites[sTradeData->connectionSpriteId2], ANIM_LINKMON_SMALL);
}
break;
case TS_STATE_FADE_OUT_TO_GBA_RECV:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_OUT_TO_GBA_RECV:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_OBJ_ON);
if (!gPaletteFade.active)
sTradeData->state++;
break;
case TS_STATE_LINK_MON_TRAVEL_IN:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 += 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
if (gSprites[sTradeData->connectionSpriteId1].y2 + gSprites[sTradeData->connectionSpriteId1].y == 64)
{
sTradeData->state++;
}
break;
case TS_STATE_PAN_TO_GBA:
if ((sTradeData->bg1vofs += 2) > 316)
{
sTradeData->bg1vofs = 316;
sTradeData->state++;
}
break;
case TS_STATE_DESTROY_LINK_MON:
DestroySprite(&gSprites[sTradeData->connectionSpriteId1]);
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
sTradeData->state++;
sTradeData->timer = 0;
break;
case TS_STATE_LINK_MON_ARRIVED_DELAY:
if (++sTradeData->timer == 10)
sTradeData->state++;
break;
case TS_STATE_MOVE_GBA_TO_CENTER:
if (++sTradeData->bg1vofs > 348)
{
sTradeData->bg1vofs = 348;
sTradeData->state++;
}
if (sTradeData->bg1vofs == 328 && sTradeData->isCableTrade)
{
sTradeData->cableEndSpriteId = CreateSprite(&sSpriteTemplate_CableEnd, 128, 65, 0);
gSprites[sTradeData->cableEndSpriteId].callback = SpriteCB_CableEndReceiving;
}
break;
case TS_STATE_GBA_FLASH_RECV:
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_GbaScreenFlash_Long, 120, 80, 0);
sTradeData->state = TS_STATE_GBA_STOP_FLASH_RECV;
break;
case TS_STATE_GBA_STOP_FLASH_RECV:
if (gSprites[sTradeData->connectionSpriteId2].animEnded)
{
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
SetTradeSequenceBgGpuRegs(6);
sTradeData->state++;
2020-08-21 00:02:00 +02:00
PlaySE(SE_M_SAND_ATTACK);
}
break;
case TS_STATE_GBA_ZOOM_IN:
if (sTradeData->gbaScale < 0x400)
{
sTradeData->gbaScale += 0x34;
}
else
{
sTradeData->gbaScale = 0x400;
sTradeData->state++;
}
sTradeData->sXY = 0x8000 / sTradeData->gbaScale;
break;
case TS_STATE_FADE_OUT_TO_NEW_MON:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeData->state = TS_STATE_WAIT_FADE_OUT_TO_NEW_MON;
break;
2018-12-28 09:11:41 +01:00
case TS_STATE_WAIT_FADE_OUT_TO_NEW_MON:
if (!gPaletteFade.active)
{
SetTradeSequenceBgGpuRegs(5);
SetTradeSequenceBgGpuRegs(7);
gPaletteFade.bufferTransferDisabled = TRUE;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_FADE_IN_TO_NEW_MON:
gPaletteFade.bufferTransferDisabled = FALSE;
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_IN_TO_NEW_MON:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
if (!gPaletteFade.active)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
break;
case TS_STATE_POKEBALL_ARRIVE:
sTradeData->bouncingPokeballSpriteId = CreateSprite(&sSpriteTemplate_Pokeball, 120, -8, 0);
gSprites[sTradeData->bouncingPokeballSpriteId].data[3] = 74;
gSprites[sTradeData->bouncingPokeballSpriteId].callback = SpriteCB_BouncingPokeballArrive;
StartSpriteAnim(&gSprites[sTradeData->bouncingPokeballSpriteId], 1);
StartSpriteAffineAnim(&gSprites[sTradeData->bouncingPokeballSpriteId], 2);
BlendPalettes(1 << (16 + gSprites[sTradeData->bouncingPokeballSpriteId].oam.paletteNum), 16, RGB_WHITEALPHA);
sTradeData->state++;
sTradeData->timer = 0;
break;
case TS_STATE_FADE_POKEBALL_TO_NORMAL:
BeginNormalPaletteFade(1 << (16 + gSprites[sTradeData->bouncingPokeballSpriteId].oam.paletteNum), 1, 16, 0, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_POKEBALL_ARRIVE_WAIT:
if (gSprites[sTradeData->bouncingPokeballSpriteId].callback == SpriteCallbackDummy)
{
2021-10-03 05:47:59 +02:00
HandleLoadSpecialPokePic_2(&gMonFrontPicTable[sTradeData->monSpecies[TRADE_PARTNER]], gMonSpritesGfxPtr->sprites.ptr[B_POSITION_OPPONENT_RIGHT], sTradeData->monSpecies[TRADE_PARTNER], sTradeData->monPersonalities[TRADE_PARTNER]);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_SHOW_NEW_MON:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].x = 120;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y = gMonFrontPicCoords[sTradeData->monSpecies[TRADE_PARTNER]].y_offset + 60;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].x2 = 0;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y2 = 0;
StartSpriteAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]], 0);
2022-06-01 18:41:57 +02:00
CreatePokeballSpriteToReleaseMon(sTradeData->monSpriteIds[TRADE_PARTNER], gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].oam.paletteNum, 120, 84, 2, 1, 20, PALETTES_BG | (0xF << 16), sTradeData->monSpecies[TRADE_PARTNER]);
FreeSpriteOamMatrix(&gSprites[sTradeData->bouncingPokeballSpriteId]);
DestroySprite(&gSprites[sTradeData->bouncingPokeballSpriteId]);
sTradeData->state++;
break;
case TS_STATE_NEW_MON_MSG:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG0_ON |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
StringExpandPlaceholders(gStringVar4, gText_XSentOverY);
DrawTextOnTradeWindow(0, gStringVar4, 0);
sTradeData->state = TS_STATE_DELAY_FOR_MON_ANIM;
sTradeData->timer = 0;
break;
case TS_STATE_DELAY_FOR_MON_ANIM:
if (++sTradeData->timer > 60)
{
sTradeData->state = TS_STATE_WAIT_FOR_MON_CRY;
sTradeData->timer = 0;
}
break;
case TS_STATE_WAIT_FOR_MON_CRY:
if (IsCryFinished())
sTradeData->state = TS_STATE_TAKE_CARE_OF_MON;
break;
case TS_STATE_TAKE_CARE_OF_MON:
if (++sTradeData->timer == 10)
2020-08-21 00:02:00 +02:00
PlayFanfare(MUS_EVOLVED);
if (sTradeData->timer == 250)
{
2019-10-09 11:56:44 +02:00
sTradeData->state++;
StringExpandPlaceholders(gStringVar4, gText_TakeGoodCareOfX);
2019-10-04 01:39:37 +02:00
DrawTextOnTradeWindow(0, gStringVar4, 0);
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
}
break;
case TS_STATE_AFTER_NEW_MON_DELAY:
if (++sTradeData->timer == 60)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
break;
case TS_STATE_CHECK_RIBBONS:
CheckPartnersMonForRibbons();
sTradeData->state++;
break;
case TS_STATE_END_LINK_TRADE:
if (sTradeData->isLinkTrade)
{
return TRUE;
}
2020-09-05 03:11:55 +02:00
else if (JOY_NEW(A_BUTTON))
{
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_TRY_EVOLUTION: // Only if in-game trade, link trades use CB2_TryLinkTradeEvolution
TradeMons(gSpecialVar_0x8005, 0);
gCB2_AfterEvolution = CB2_UpdateInGameTrade;
2021-02-12 20:54:11 +01:00
evoTarget = GetEvolutionTargetSpecies(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], EVO_MODE_TRADE, ITEM_NONE);
if (evoTarget != SPECIES_NONE)
{
TradeEvolutionScene(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], evoTarget, sTradeData->monSpriteIds[TRADE_PARTNER], gSelectedTradeMonPositions[TRADE_PLAYER]);
}
sTradeData->state++;
break;
case TS_STATE_FADE_OUT_END:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_OUT_END:
if (!gPaletteFade.active)
{
PlayNewMapMusic(sTradeData->cachedMapMusic);
if (sTradeData)
{
FreeAllWindowBuffers();
Free(GetBgTilemapBuffer(3));
Free(GetBgTilemapBuffer(1));
Free(GetBgTilemapBuffer(0));
FreeMonSpritesGfx();
FREE_AND_SET_NULL(sTradeData);
}
SetMainCallback2(CB2_ReturnToField);
2020-01-08 19:19:41 +01:00
BufferInGameTradeMonName();
}
break;
2018-12-28 20:42:34 +01:00
}
return FALSE;
}
2019-10-05 16:41:37 +02:00
static bool8 AnimateTradeSequenceWireless(void)
2018-12-28 20:42:34 +01:00
{
u16 evoTarget;
2019-10-09 11:56:44 +02:00
switch (sTradeData->state)
2018-12-28 20:42:34 +01:00
{
case TS_STATE_START:
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].invisible = FALSE;
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x2 = -180;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 = gMonFrontPicCoords[sTradeData->monSpecies[TRADE_PLAYER]].y_offset;
sTradeData->state++;
sTradeData->cachedMapMusic = GetCurrentMapMusic();
2020-08-21 00:02:00 +02:00
PlayNewMapMusic(MUS_EVOLUTION);
break;
case TS_STATE_MON_SLIDE_IN:
if (sTradeData->bg2hofs > 0)
{
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x2 += 3;
sTradeData->bg2hofs -= 3;
}
else
{
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x2 = 0;
sTradeData->bg2hofs = 0;
sTradeData->state = TS_STATE_SEND_MSG;
}
break;
case TS_STATE_SEND_MSG:
StringExpandPlaceholders(gStringVar4, gText_XWillBeSentToY);
DrawTextOnTradeWindow(0, gStringVar4, 0);
2018-12-28 20:42:34 +01:00
if (sTradeData->monSpecies[TRADE_PLAYER] != SPECIES_EGG)
2021-11-07 19:54:44 +01:00
PlayCry_Normal(sTradeData->monSpecies[TRADE_PLAYER], 0);
2018-12-28 20:42:34 +01:00
sTradeData->state = TS_STATE_BYE_BYE;
sTradeData->timer = 0;
break;
case TS_STATE_BYE_BYE:
if (++sTradeData->timer == 80)
{
sTradeData->releasePokeballSpriteId = CreateTradePokeballSprite(sTradeData->monSpriteIds[0], gSprites[sTradeData->monSpriteIds[0]].oam.paletteNum, 120, 32, 2, 1, 0x14, 0xfffff);
sTradeData->state++;
StringExpandPlaceholders(gStringVar4, gText_ByeByeVar1);
DrawTextOnTradeWindow(0, gStringVar4, 0);
}
break;
case TS_STATE_POKEBALL_DEPART:
if (gSprites[sTradeData->releasePokeballSpriteId].callback == SpriteCallbackDummy)
{
sTradeData->bouncingPokeballSpriteId = CreateSprite(&sSpriteTemplate_Pokeball, 120, 32, 0);
gSprites[sTradeData->bouncingPokeballSpriteId].callback = SpriteCB_BouncingPokeballDepart;
DestroySprite(&gSprites[sTradeData->releasePokeballSpriteId]);
sTradeData->state++;
}
break;
case TS_STATE_POKEBALL_DEPART_WAIT:
// The game waits here for the sprite to finish its animation sequence.
break;
case TS_STATE_FADE_OUT_TO_GBA_SEND:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeData->state = TS_STATE_WAIT_FADE_OUT_TO_GBA_SEND;
break;
case TS_STATE_WAIT_FADE_OUT_TO_GBA_SEND:
if (!gPaletteFade.active)
{
SetTradeSequenceBgGpuRegs(4);
FillWindowPixelBuffer(0, PIXEL_FILL(15));
2021-11-03 20:29:18 +01:00
CopyWindowToVram(0, COPYWIN_FULL);
sTradeData->state++;
}
break;
case TS_STATE_FADE_IN_TO_GBA_SEND:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_IN_TO_GBA_SEND:
if (!gPaletteFade.active)
sTradeData->state = TS_STATE_GBA_ZOOM_OUT;
break;
case TS_STATE_GBA_ZOOM_OUT:
if (sTradeData->gbaScale > 0x100)
{
sTradeData->gbaScale -= 0x34;
}
else
{
SetTradeSequenceBgGpuRegs(1);
sTradeData->gbaScale = 0x80;
sTradeData->state = TS_STATE_GBA_FLASH_SEND_WIRELESS;
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
}
sTradeData->sXY = 0x8000 / sTradeData->gbaScale;
break;
case TS_STATE_GBA_FLASH_SEND_WIRELESS:
if (++sTradeData->timer > 20)
{
SetTradeSequenceBgGpuRegs(3);
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_GbaScreenFlash_Short, 120, 80, 0);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_GBA_STOP_FLASH_SEND_WIRELESS:
if (gSprites[sTradeData->connectionSpriteId2].animEnded)
{
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_TGT1_BG1 |
BLDCNT_TGT1_OBJ |
BLDCNT_EFFECT_BLEND |
BLDCNT_TGT2_BG2);
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(16, 4));
// Start wireless signal effect
CreateTask(Task_AnimateWirelessSignal, 5);
sTradeData->state++;
}
break;
case TS_STATE_WAIT_WIRELESS_SIGNAL_SEND:
if (!FuncIsActiveTask(Task_AnimateWirelessSignal))
sTradeData->state = TS_STATE_PAN_AWAY_GBA;
break;
case TS_STATE_PAN_AWAY_GBA:
if (--sTradeData->bg1vofs == 316)
sTradeData->state++;
break;
case TS_STATE_CREATE_LINK_MON_LEAVING:
sTradeData->connectionSpriteId1 = CreateSprite(&sSpriteTemplate_LinkMonGlow, 120, 80, 3);
gSprites[sTradeData->connectionSpriteId1].callback = SpriteCB_LinkMonGlowWireless;
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 120, 80, 0);
StartSpriteAnim(&gSprites[sTradeData->connectionSpriteId2], ANIM_LINKMON_SMALL);
sTradeData->state++;
break;
case TS_STATE_LINK_MON_TRAVEL_OUT:
if ((sTradeData->bg1vofs -= 3) == 166)
sTradeData->state = TS_STATE_LINK_MON_TRAVEL_OFFSCREEN;
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_1 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_OBJ_ON);
break;
case TS_STATE_LINK_MON_TRAVEL_OFFSCREEN:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y -= 2;
gSprites[sTradeData->connectionSpriteId2].y -= 2;
if (gSprites[sTradeData->connectionSpriteId1].y < -8)
{
sTradeData->state = TS_STATE_FADE_OUT_TO_CROSSING;
}
break;
case TS_STATE_FADE_OUT_TO_CROSSING:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 0, 16, RGB_BLACK);
sTradeData->state = TS_STATE_WAIT_FADE_OUT_TO_CROSSING;
break;
case TS_STATE_WAIT_FADE_OUT_TO_CROSSING:
if (!gPaletteFade.active)
{
DestroySprite(&gSprites[sTradeData->connectionSpriteId1]);
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
SetTradeSequenceBgGpuRegs(2);
sTradeData->state++;
}
break;
case TS_STATE_FADE_IN_TO_CROSSING:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
sTradeData->connectionSpriteId1 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 111, 170, 0);
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 129, -10, 0);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_IN_TO_CROSSING:
if (!gPaletteFade.active)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_WARP_OUT);
sTradeData->state++;
}
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 -= 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
break;
case TS_STATE_CROSSING_LINK_MONS_ENTER:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 -= 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
if (gSprites[sTradeData->connectionSpriteId1].y2 <= -90)
{
gSprites[sTradeData->connectionSpriteId1].data[1] = 1;
gSprites[sTradeData->connectionSpriteId2].data[1] = 1;
sTradeData->state++;
CreateTask(Task_NarrowWindowForCrossing_Wireless, 5);
}
break;
case TS_STATE_CROSSING_BLEND_WHITE_1:
BlendPalettes(0x8, 16, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_CROSSING_BLEND_WHITE_2:
BlendPalettes(0x8, 16, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_CROSSING_BLEND_WHITE_3:
BlendPalettes(0x8, 16, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_CROSSING_CREATE_MON_PICS:
if (!IsMonSpriteNotFlipped(sTradeData->monSpecies[TRADE_PLAYER]))
{
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].affineAnims = sAffineAnims_CrossingMonPics;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].oam.affineMode = ST_OAM_AFFINE_DOUBLE;
CalcCenterToCornerVec(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]], SPRITE_SHAPE(64x64), SPRITE_SIZE(64x64), ST_OAM_AFFINE_DOUBLE);
StartSpriteAffineAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]], 0);
}
else
{
StartSpriteAffineAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]], 0);
}
StartSpriteAffineAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]], 0);
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].x = 40;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].x = 200;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y = 192;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y = -32;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].invisible = FALSE;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].invisible = FALSE;
sTradeData->state++;
break;
case TS_STATE_CROSSING_MON_PICS_MOVE:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 -= 3;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y2 += 3;
if (gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 < -DISPLAY_HEIGHT
&& gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 >= -DISPLAY_HEIGHT - 3)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_WARP_IN);
}
2021-07-07 15:11:52 +02:00
if (gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].y2 < -222)
{
gSprites[sTradeData->connectionSpriteId1].data[1] = 0;
gSprites[sTradeData->connectionSpriteId2].data[1] = 0;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]].invisible = TRUE;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].invisible = TRUE;
CreateTask(Task_NarrowWindowForCrossing_Cable, 5);
}
break;
case TS_STATE_CROSSING_LINK_MONS_EXIT:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 -= 3;
gSprites[sTradeData->connectionSpriteId2].y2 += 3;
if (gSprites[sTradeData->connectionSpriteId1].y2 <= -222)
{
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 0, 16, RGB_BLACK);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
DestroySprite(&gSprites[sTradeData->connectionSpriteId1]);
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
}
break;
case TS_STATE_CREATE_LINK_MON_ARRIVING:
if (!gPaletteFade.active)
{
2019-10-09 11:56:44 +02:00
sTradeData->state++;
SetTradeSequenceBgGpuRegs(1);
sTradeData->bg1vofs = 166;
SetTradeSequenceBgGpuRegs(3);
sTradeData->bg2vofs = 412;
sTradeData->connectionSpriteId1 = CreateSprite(&sSpriteTemplate_LinkMonGlow, 120, -20, 3);
gSprites[sTradeData->connectionSpriteId1].callback = SpriteCB_LinkMonGlowWireless;
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_LinkMonShadow, 120, -20, 0);
StartSpriteAnim(&gSprites[sTradeData->connectionSpriteId2], ANIM_LINKMON_SMALL);
}
break;
case TS_STATE_FADE_OUT_TO_GBA_RECV:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_OUT_TO_GBA_RECV:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_OBJ_ON);
if (!gPaletteFade.active)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
break;
case TS_STATE_LINK_MON_TRAVEL_IN:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->connectionSpriteId1].y2 += 4;
gSprites[sTradeData->connectionSpriteId2].y2 += 4;
if (gSprites[sTradeData->connectionSpriteId1].y2 + gSprites[sTradeData->connectionSpriteId1].y == 64)
{
sTradeData->state = TS_STATE_PAN_TO_GBA_WIRELESS;
sTradeData->timer = 0;
}
break;
case TS_STATE_PAN_TO_GBA_WIRELESS:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG1_ON |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
sTradeData->bg1vofs += 3;
sTradeData->bg2vofs += 3;
if (++sTradeData->timer == 10)
{
u8 taskId = CreateTask(Task_AnimateWirelessSignal, 5);
gTasks[taskId].data[2] = TRUE;
}
if (sTradeData->bg1vofs > 316)
{
sTradeData->bg1vofs = 316;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_DESTROY_LINK_MON_WIRELESS:
DestroySprite(&gSprites[sTradeData->connectionSpriteId1]);
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
sTradeData->state++;
sTradeData->timer = 0;
break;
case TS_STATE_WAIT_WIRELESS_SIGNAL_RECV:
if (!FuncIsActiveTask(Task_AnimateWirelessSignal))
{
sTradeData->state = TS_STATE_LINK_MON_ARRIVED_DELAY;
sTradeData->timer = 0;
}
break;
case TS_STATE_LINK_MON_ARRIVED_DELAY:
if (++sTradeData->timer == 10)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
break;
case TS_STATE_MOVE_GBA_TO_CENTER:
if (++sTradeData->bg1vofs > 348)
{
sTradeData->bg1vofs = 348;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_GBA_FLASH_RECV:
sTradeData->connectionSpriteId2 = CreateSprite(&sSpriteTemplate_GbaScreenFlash_Long, 120, 80, 0);
sTradeData->state = TS_STATE_GBA_STOP_FLASH_RECV;
break;
case TS_STATE_GBA_STOP_FLASH_RECV:
if (gSprites[sTradeData->connectionSpriteId2].animEnded)
{
DestroySprite(&gSprites[sTradeData->connectionSpriteId2]);
SetTradeSequenceBgGpuRegs(6);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
2020-08-21 00:02:00 +02:00
PlaySE(SE_M_SAND_ATTACK);
}
break;
case TS_STATE_GBA_ZOOM_IN:
if (sTradeData->gbaScale < 0x400)
{
sTradeData->gbaScale += 0x34;
}
else
{
sTradeData->gbaScale = 0x400;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
sTradeData->sXY = 0x8000 / sTradeData->gbaScale;
break;
case TS_STATE_FADE_OUT_TO_NEW_MON:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeData->state = TS_STATE_WAIT_FADE_OUT_TO_NEW_MON;
break;
case TS_STATE_WAIT_FADE_OUT_TO_NEW_MON:
if (!gPaletteFade.active)
{
SetTradeSequenceBgGpuRegs(5);
SetTradeSequenceBgGpuRegs(7);
gPaletteFade.bufferTransferDisabled = TRUE;
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_FADE_IN_TO_NEW_MON:
gPaletteFade.bufferTransferDisabled = FALSE;
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_IN_TO_NEW_MON:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
if (!gPaletteFade.active)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
break;
case TS_STATE_POKEBALL_ARRIVE:
sTradeData->bouncingPokeballSpriteId = CreateSprite(&sSpriteTemplate_Pokeball, 120, -8, 0);
gSprites[sTradeData->bouncingPokeballSpriteId].data[3] = 74;
gSprites[sTradeData->bouncingPokeballSpriteId].callback = SpriteCB_BouncingPokeballArrive;
StartSpriteAnim(&gSprites[sTradeData->bouncingPokeballSpriteId], 1);
StartSpriteAffineAnim(&gSprites[sTradeData->bouncingPokeballSpriteId], 2);
BlendPalettes(1 << (16 + gSprites[sTradeData->bouncingPokeballSpriteId].oam.paletteNum), 16, RGB_WHITEALPHA);
sTradeData->state++;
sTradeData->timer = 0;
break;
case TS_STATE_FADE_POKEBALL_TO_NORMAL:
BeginNormalPaletteFade(1 << (16 + gSprites[sTradeData->bouncingPokeballSpriteId].oam.paletteNum), 1, 16, 0, RGB_WHITEALPHA);
sTradeData->state++;
break;
case TS_STATE_POKEBALL_ARRIVE_WAIT:
if (gSprites[sTradeData->bouncingPokeballSpriteId].callback == SpriteCallbackDummy)
{
HandleLoadSpecialPokePic_2(&gMonFrontPicTable[sTradeData->monSpecies[TRADE_PARTNER]],
2021-10-03 05:47:59 +02:00
gMonSpritesGfxPtr->sprites.ptr[B_POSITION_OPPONENT_RIGHT],
sTradeData->monSpecies[TRADE_PARTNER],
sTradeData->monPersonalities[TRADE_PARTNER]);
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_SHOW_NEW_MON:
2021-07-07 15:11:52 +02:00
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].x = 120;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y = gMonFrontPicCoords[sTradeData->monSpecies[TRADE_PARTNER]].y_offset + 60;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].x2 = 0;
gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].y2 = 0;
StartSpriteAnim(&gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]], 0);
2022-06-01 18:41:57 +02:00
CreatePokeballSpriteToReleaseMon(sTradeData->monSpriteIds[TRADE_PARTNER], gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]].oam.paletteNum, 120, 84, 2, 1, 20, PALETTES_BG | (0xF << 16), sTradeData->monSpecies[TRADE_PARTNER]);
FreeSpriteOamMatrix(&gSprites[sTradeData->bouncingPokeballSpriteId]);
DestroySprite(&gSprites[sTradeData->bouncingPokeballSpriteId]);
sTradeData->state++;
break;
case TS_STATE_NEW_MON_MSG:
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 |
DISPCNT_OBJ_1D_MAP |
DISPCNT_BG0_ON |
DISPCNT_BG2_ON |
DISPCNT_OBJ_ON);
StringExpandPlaceholders(gStringVar4, gText_XSentOverY);
DrawTextOnTradeWindow(0, gStringVar4, 0);
sTradeData->state = TS_STATE_DELAY_FOR_MON_ANIM;
sTradeData->timer = 0;
break;
case TS_STATE_DELAY_FOR_MON_ANIM:
if (++sTradeData->timer > 60)
{
sTradeData->state = TS_STATE_WAIT_FOR_MON_CRY;
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
}
break;
case TS_STATE_WAIT_FOR_MON_CRY:
if (IsCryFinished())
sTradeData->state = TS_STATE_TAKE_CARE_OF_MON;
break;
case TS_STATE_TAKE_CARE_OF_MON:
if (++sTradeData->timer == 10)
2020-08-21 00:02:00 +02:00
PlayFanfare(MUS_EVOLVED);
if (sTradeData->timer == 250)
{
2019-10-09 11:56:44 +02:00
sTradeData->state++;
StringExpandPlaceholders(gStringVar4, gText_TakeGoodCareOfX);
DrawTextOnTradeWindow(0, gStringVar4, 0);
sTradeData->timer = 0;
}
break;
case TS_STATE_AFTER_NEW_MON_DELAY:
if (++sTradeData->timer == 60)
2019-10-09 11:56:44 +02:00
sTradeData->state++;
break;
case TS_STATE_CHECK_RIBBONS:
CheckPartnersMonForRibbons();
sTradeData->state++;
break;
case TS_STATE_END_LINK_TRADE:
if (sTradeData->isLinkTrade)
{
return TRUE;
}
2020-09-05 03:11:55 +02:00
else if (JOY_NEW(A_BUTTON))
{
2019-10-09 11:56:44 +02:00
sTradeData->state++;
}
break;
case TS_STATE_TRY_EVOLUTION: // Only if in-game trade, link trades use CB2_TryLinkTradeEvolution
TradeMons(gSpecialVar_0x8005, 0);
gCB2_AfterEvolution = CB2_UpdateInGameTrade;
2021-02-12 20:54:11 +01:00
evoTarget = GetEvolutionTargetSpecies(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], EVO_MODE_TRADE, ITEM_NONE);
if (evoTarget != SPECIES_NONE)
{
TradeEvolutionScene(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], evoTarget, sTradeData->monSpriteIds[TRADE_PARTNER], gSelectedTradeMonPositions[TRADE_PLAYER]);
}
sTradeData->state++;
break;
case TS_STATE_FADE_OUT_END:
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
sTradeData->state++;
break;
case TS_STATE_WAIT_FADE_OUT_END:
if (!gPaletteFade.active)
{
PlayNewMapMusic(sTradeData->cachedMapMusic);
if (sTradeData)
{
FreeAllWindowBuffers();
Free(GetBgTilemapBuffer(3));
Free(GetBgTilemapBuffer(1));
Free(GetBgTilemapBuffer(0));
FreeMonSpritesGfx();
FREE_AND_SET_NULL(sTradeData);
}
SetMainCallback2(CB2_ReturnToField);
2020-01-08 19:19:41 +01:00
BufferInGameTradeMonName();
}
break;
2018-12-28 09:11:41 +01:00
}
return FALSE;
}
2018-12-29 19:46:15 +01:00
// Try to evolve a Pokémon received in a link trade
// In-game trades resolve evolution during the trade sequence, in TS_STATE_TRY_EVOLUTION
static void CB2_TryLinkTradeEvolution(void)
2018-12-29 19:46:15 +01:00
{
u16 evoTarget;
switch (gMain.state)
{
case 0:
gMain.state = 4;
gSoftResetDisabled = TRUE;
break;
case 4:
gCB2_AfterEvolution = CB2_SaveAndEndTrade;
2021-02-12 20:54:11 +01:00
evoTarget = GetEvolutionTargetSpecies(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], EVO_MODE_TRADE, ITEM_NONE);
if (evoTarget != SPECIES_NONE)
TradeEvolutionScene(&gPlayerParty[gSelectedTradeMonPositions[TRADE_PLAYER]], evoTarget, sTradeData->monSpriteIds[TRADE_PARTNER], gSelectedTradeMonPositions[TRADE_PLAYER]);
else if (IsWirelessTrade())
SetMainCallback2(CB2_SaveAndEndWirelessTrade);
else
SetMainCallback2(CB2_SaveAndEndTrade);
gSelectedTradeMonPositions[TRADE_PLAYER] = 255;
break;
2018-12-29 19:46:15 +01:00
}
if (!HasLinkErrorOccurred())
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void UpdateTradeFinishFlags(void)
2018-12-29 19:46:15 +01:00
{
u8 blockReceivedStatus;
2019-10-04 13:42:45 +02:00
TradeGetMultiplayerId(); // no effect call, ret val ignored
2018-12-29 19:46:15 +01:00
blockReceivedStatus = GetBlockReceivedStatus();
if (blockReceivedStatus & 0x01)
{
if (gBlockRecvBuffer[0][0] == LINKCMD_CONFIRM_FINISH_TRADE)
SetMainCallback2(CB2_TryLinkTradeEvolution);
if (gBlockRecvBuffer[0][0] == LINKCMD_READY_FINISH_TRADE)
sTradeData->playerLinkFlagFinishTrade = READY_FINISH_TRADE;
2018-12-29 19:46:15 +01:00
ResetBlockReceivedFlag(0);
}
if (blockReceivedStatus & 0x02)
{
if (gBlockRecvBuffer[1][0] == LINKCMD_READY_FINISH_TRADE)
sTradeData->partnerLinkFlagFinishTrade = READY_FINISH_TRADE;
2018-12-29 19:46:15 +01:00
ResetBlockReceivedFlag(1);
}
}
static void SpriteCB_BouncingPokeball(struct Sprite *sprite)
2018-12-29 19:46:15 +01:00
{
2021-07-07 15:11:52 +02:00
sprite->y += sprite->data[0] / 10;
2018-12-29 19:46:15 +01:00
sprite->data[5] += sprite->data[1];
2021-07-07 15:11:52 +02:00
sprite->x = sprite->data[5] / 10;
if (sprite->y > 0x4c)
2018-12-29 19:46:15 +01:00
{
2021-07-07 15:11:52 +02:00
sprite->y = 0x4c;
2018-12-29 19:46:15 +01:00
sprite->data[0] = -(sprite->data[0] * sprite->data[2]) / 100;
sprite->data[3] ++;
}
2021-07-07 15:11:52 +02:00
if (sprite->x == 0x78)
2018-12-29 19:46:15 +01:00
sprite->data[1] = 0;
sprite->data[0] += sprite->data[4];
if (sprite->data[3] == 4)
{
sprite->data[7] = 1;
sprite->callback = SpriteCallbackDummy;
}
}
static void SpriteCB_BouncingPokeballDepart(struct Sprite *sprite)
2018-12-29 19:46:15 +01:00
{
2021-07-07 15:11:52 +02:00
sprite->y2 += sTradeBallVerticalVelocityTable[sprite->data[0]];
2018-12-29 19:46:15 +01:00
if (sprite->data[0] == 22)
2020-08-21 00:02:00 +02:00
PlaySE(SE_BALL_BOUNCE_1);
2018-12-29 19:46:15 +01:00
if (++ sprite->data[0] == 44)
{
2020-08-21 00:02:00 +02:00
PlaySE(SE_M_MEGA_KICK);
sprite->callback = SpriteCB_BouncingPokeballDepartEnd;
2018-12-29 19:46:15 +01:00
sprite->data[0] = 0;
BeginNormalPaletteFade(1 << (16 + sprite->oam.paletteNum), -1, 0, 16, RGB_WHITEALPHA);
}
}
static void SpriteCB_BouncingPokeballDepartEnd(struct Sprite *sprite)
2018-12-29 19:46:15 +01:00
{
if (sprite->data[1] == 20)
StartSpriteAffineAnim(sprite, 1);
if (++ sprite->data[1] > 20)
{
2021-07-07 15:11:52 +02:00
sprite->y2 -= sTradeBallVerticalVelocityTable[sprite->data[0]];
2018-12-29 19:46:15 +01:00
if (++ sprite->data[0] == 23)
{
DestroySprite(sprite);
sTradeData->state = TS_STATE_FADE_OUT_TO_GBA_SEND; // Resume the master trade animation
2018-12-29 19:46:15 +01:00
}
}
}
static void SpriteCB_BouncingPokeballArrive(struct Sprite *sprite)
2018-12-29 19:46:15 +01:00
{
if (sprite->data[2] == 0)
{
2021-07-07 15:11:52 +02:00
if ((sprite->y += 4) > sprite->data[3])
2018-12-29 19:46:15 +01:00
{
sprite->data[2] ++;
sprite->data[0] = 0x16;
2020-08-21 00:02:00 +02:00
PlaySE(SE_BALL_BOUNCE_1);
2018-12-29 19:46:15 +01:00
}
}
else
{
if (sprite->data[0] == 0x42)
2020-08-21 00:02:00 +02:00
PlaySE(SE_BALL_BOUNCE_2);
2018-12-29 19:46:15 +01:00
if (sprite->data[0] == 0x5c)
2020-08-21 00:02:00 +02:00
PlaySE(SE_BALL_BOUNCE_3);
2018-12-29 19:46:15 +01:00
if (sprite->data[0] == 0x6b)
2020-08-21 00:02:00 +02:00
PlaySE(SE_BALL_BOUNCE_4);
2021-07-07 15:11:52 +02:00
sprite->y2 += sTradeBallVerticalVelocityTable[sprite->data[0]];
2018-12-29 19:46:15 +01:00
if (++sprite->data[0] == 0x6c)
sprite->callback = SpriteCallbackDummy;
}
}
u16 GetInGameTradeSpeciesInfo(void)
{
const struct InGameTrade *inGameTrade = &sIngameTrades[gSpecialVar_0x8004];
StringCopy(gStringVar1, gSpeciesNames[inGameTrade->requestedSpecies]);
2018-12-29 19:46:15 +01:00
StringCopy(gStringVar2, gSpeciesNames[inGameTrade->species]);
return inGameTrade->requestedSpecies;
2018-12-29 19:46:15 +01:00
}
2020-01-08 19:19:41 +01:00
static void BufferInGameTradeMonName(void)
2018-12-29 19:46:15 +01:00
{
u8 nickname[32];
const struct InGameTrade *inGameTrade = &sIngameTrades[gSpecialVar_0x8004];
2018-12-29 19:46:15 +01:00
GetMonData(&gPlayerParty[gSpecialVar_0x8005], MON_DATA_NICKNAME, nickname);
StringCopy_Nickname(gStringVar1, nickname);
2018-12-29 19:46:15 +01:00
StringCopy(gStringVar2, gSpeciesNames[inGameTrade->species]);
}
static void _CreateInGameTradePokemon(u8 whichPlayerMon, u8 whichInGameTrade)
2018-12-29 19:46:15 +01:00
{
const struct InGameTrade *inGameTrade = &sIngameTrades[whichInGameTrade];
2018-12-29 19:46:15 +01:00
u8 level = GetMonData(&gPlayerParty[whichPlayerMon], MON_DATA_LEVEL);
2021-10-23 16:55:46 +02:00
struct Mail mail;
u8 metLocation = METLOC_IN_GAME_TRADE;
2018-12-29 19:46:15 +01:00
u8 isMail;
struct Pokemon *pokemon = &gEnemyParty[0];
CreateMon(pokemon, inGameTrade->species, level, USE_RANDOM_IVS, TRUE, inGameTrade->personality, OT_ID_PRESET, inGameTrade->otId);
2018-12-29 19:46:15 +01:00
SetMonData(pokemon, MON_DATA_HP_IV, &inGameTrade->ivs[0]);
SetMonData(pokemon, MON_DATA_ATK_IV, &inGameTrade->ivs[1]);
SetMonData(pokemon, MON_DATA_DEF_IV, &inGameTrade->ivs[2]);
SetMonData(pokemon, MON_DATA_SPEED_IV, &inGameTrade->ivs[3]);
SetMonData(pokemon, MON_DATA_SPATK_IV, &inGameTrade->ivs[4]);
SetMonData(pokemon, MON_DATA_SPDEF_IV, &inGameTrade->ivs[5]);
SetMonData(pokemon, MON_DATA_NICKNAME, inGameTrade->nickname);
2018-12-29 19:46:15 +01:00
SetMonData(pokemon, MON_DATA_OT_NAME, inGameTrade->otName);
SetMonData(pokemon, MON_DATA_OT_GENDER, &inGameTrade->otGender);
SetMonData(pokemon, MON_DATA_ABILITY_NUM, &inGameTrade->abilityNum);
SetMonData(pokemon, MON_DATA_BEAUTY, &inGameTrade->conditions[1]);
SetMonData(pokemon, MON_DATA_CUTE, &inGameTrade->conditions[2]);
SetMonData(pokemon, MON_DATA_COOL, &inGameTrade->conditions[0]);
SetMonData(pokemon, MON_DATA_SMART, &inGameTrade->conditions[3]);
SetMonData(pokemon, MON_DATA_TOUGH, &inGameTrade->conditions[4]);
2018-12-29 19:46:15 +01:00
SetMonData(pokemon, MON_DATA_SHEEN, &inGameTrade->sheen);
SetMonData(pokemon, MON_DATA_MET_LOCATION, &metLocation);
isMail = FALSE;
if (inGameTrade->heldItem != ITEM_NONE)
{
if (ItemIsMail(inGameTrade->heldItem))
{
2019-10-04 01:39:37 +02:00
SetInGameTradeMail(&mail, inGameTrade);
2019-10-04 13:42:45 +02:00
gTradeMail[0] = mail;
2018-12-29 19:46:15 +01:00
SetMonData(pokemon, MON_DATA_MAIL, &isMail);
SetMonData(pokemon, MON_DATA_HELD_ITEM, &inGameTrade->heldItem);
}
else
{
SetMonData(pokemon, MON_DATA_HELD_ITEM, &inGameTrade->heldItem);
}
}
CalculateMonStats(&gEnemyParty[0]);
}
2021-12-14 18:47:04 +01:00
static void SetInGameTradeMail(struct Mail *mail, const struct InGameTrade *trade)
{
2018-12-29 19:46:15 +01:00
s32 i;
2019-10-05 16:41:37 +02:00
for (i = 0; i < MAIL_WORDS_COUNT; i++)
2018-12-29 19:46:15 +01:00
{
2019-10-04 01:39:37 +02:00
mail->words[i] = sIngameTradeMail[trade->mailNum][i];
2018-12-29 19:46:15 +01:00
}
StringCopy(mail->playerName, trade->otName);
PadNameString(mail->playerName, CHAR_SPACE);
mail->trainerId[0] = trade->otId >> 24;
mail->trainerId[1] = trade->otId >> 16;
mail->trainerId[2] = trade->otId >> 8;
mail->trainerId[3] = trade->otId;
mail->species = trade->species;
mail->itemId = trade->heldItem;
}
u16 GetTradeSpecies(void)
{
if (GetMonData(&gPlayerParty[gSpecialVar_0x8005], MON_DATA_IS_EGG))
return SPECIES_NONE;
return GetMonData(&gPlayerParty[gSpecialVar_0x8005], MON_DATA_SPECIES);
}
void CreateInGameTradePokemon(void)
{
_CreateInGameTradePokemon(gSpecialVar_0x8005, gSpecialVar_0x8004);
}
static void CB2_UpdateLinkTrade(void)
2018-12-29 19:46:15 +01:00
{
2019-10-05 16:41:37 +02:00
if (AnimateTradeSequence() == TRUE)
2018-12-29 19:46:15 +01:00
{
DestroySprite(&gSprites[sTradeData->monSpriteIds[TRADE_PLAYER]]);
FreeSpriteOamMatrix(&gSprites[sTradeData->monSpriteIds[TRADE_PARTNER]]);
2019-10-04 13:42:45 +02:00
TradeMons(gSelectedTradeMonPositions[TRADE_PLAYER], gSelectedTradeMonPositions[TRADE_PARTNER] % PARTY_SIZE);
if (!IsWirelessTrade())
2018-12-29 19:46:15 +01:00
{
sTradeData->linkData[0] = LINKCMD_READY_FINISH_TRADE;
2019-10-10 09:50:51 +02:00
sTradeData->sendTradeFinishState = 1;
2018-12-29 19:46:15 +01:00
}
SetMainCallback2(CB2_TryFinishTrade);
2018-12-29 19:46:15 +01:00
}
2019-10-10 09:50:51 +02:00
TrySendTradeFinishData();
UpdateTradeFinishFlags();
2018-12-29 19:46:15 +01:00
RunTasks();
RunTextPrinters();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void CB2_TryFinishTrade(void)
2018-12-29 19:46:15 +01:00
{
2019-10-04 13:42:45 +02:00
u8 mpId = TradeGetMultiplayerId();
if (IsWirelessTrade())
2018-12-29 19:46:15 +01:00
{
SetMainCallback2(CB2_TryLinkTradeEvolution);
2018-12-29 19:46:15 +01:00
}
else
{
UpdateTradeFinishFlags();
2020-05-18 14:03:01 +02:00
if (mpId == 0
&& sTradeData->playerLinkFlagFinishTrade == READY_FINISH_TRADE
&& sTradeData->partnerLinkFlagFinishTrade == READY_FINISH_TRADE)
2018-12-29 19:46:15 +01:00
{
sTradeData->linkData[0] = LINKCMD_CONFIRM_FINISH_TRADE;
SendBlock(BitmaskAllOtherLinkPlayers(), sTradeData->linkData, sizeof(sTradeData->linkData));
sTradeData->playerLinkFlagFinishTrade = FINISH_TRADE;
sTradeData->partnerLinkFlagFinishTrade = FINISH_TRADE;
2018-12-29 19:46:15 +01:00
}
}
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void CB2_SaveAndEndTrade(void)
2018-12-29 19:46:15 +01:00
{
switch (gMain.state)
{
case 0:
gMain.state++;
StringExpandPlaceholders(gStringVar4, gText_CommunicationStandby5);
DrawTextOnTradeWindow(0, gStringVar4, 0);
break;
case 1:
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(0);
gMain.state = 100;
sTradeData->timer = 0;
break;
case 100:
if (++sTradeData->timer > 180)
{
gMain.state = 101;
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
}
if (_IsLinkTaskFinished())
{
gMain.state = 2;
}
break;
case 101:
if (_IsLinkTaskFinished())
{
gMain.state = 2;
}
break;
case 2:
gMain.state = 50;
StringExpandPlaceholders(gStringVar4, gText_SavingDontTurnOffPower);
DrawTextOnTradeWindow(0, gStringVar4, 0);
break;
case 50:
if (!InUnionRoom())
IncrementGameStat(GAME_STAT_POKEMON_TRADES);
if (gWirelessCommType)
2021-10-17 07:35:26 +02:00
MysteryGift_TryIncrementStat(CARD_STAT_NUM_TRADES, gLinkPlayers[GetMultiplayerId() ^ 1].trainerId);
SetContinueGameWarpStatusToDynamicWarp();
2021-10-29 04:54:41 +02:00
LinkFullSave_Init();
gMain.state++;
sTradeData->timer = 0;
break;
case 51:
if (++sTradeData->timer == 5)
2018-12-29 19:46:15 +01:00
gMain.state++;
break;
case 52:
2021-10-29 04:54:41 +02:00
if (LinkFullSave_WriteSector())
{
ClearContinueGameWarpStatus2();
gMain.state = 4;
}
else
{
2021-10-29 04:54:41 +02:00
// Save isn't finished, delay again
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
gMain.state = 51;
}
break;
case 4:
2021-10-29 04:54:41 +02:00
LinkFullSave_ReplaceLastSector();
gMain.state = 40;
sTradeData->timer = 0;
break;
case 40:
if (++sTradeData->timer > 50)
{
if (GetMultiplayerId() == 0)
2018-12-29 19:46:15 +01:00
{
sTradeData->timer = Random() % 30;
2018-12-29 19:46:15 +01:00
}
else
{
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
2018-12-29 19:46:15 +01:00
}
gMain.state = 41;
}
break;
case 41:
if (sTradeData->timer == 0)
{
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(1);
gMain.state = 42;
}
else
{
sTradeData->timer--;
}
break;
case 42:
if (_IsLinkTaskFinished())
{
2022-08-05 16:31:50 +02:00
LinkFullSave_SetLastSectorSignature();
gMain.state = 5;
}
break;
case 5:
if (++sTradeData->timer > 60)
{
gMain.state++;
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(2);
}
break;
case 6:
if (_IsLinkTaskFinished())
{
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
gMain.state ++;
}
break;
case 7:
if (!gPaletteFade.active)
{
FadeOutBGM(3);
gMain.state++;
}
break;
case 8:
if (IsBGMStopped() == TRUE)
{
if (gWirelessCommType && gMain.savedCallback == CB2_StartCreateTradeMenu)
2018-12-29 19:46:15 +01:00
{
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(3);
2018-12-29 19:46:15 +01:00
}
else
{
2020-08-13 09:09:47 +02:00
SetCloseLinkCallback();
2018-12-29 19:46:15 +01:00
}
gMain.state++;
}
break;
case 9:
if (gWirelessCommType && gMain.savedCallback == CB2_StartCreateTradeMenu)
{
2018-12-31 09:22:21 +01:00
if (_IsLinkTaskFinished())
2018-12-29 19:46:15 +01:00
{
gSoftResetDisabled = FALSE;
SetMainCallback2(CB2_FreeTradeData);
2018-12-29 19:46:15 +01:00
}
}
else if (!gReceivedRemoteLinkPlayers)
{
gSoftResetDisabled = FALSE;
SetMainCallback2(CB2_FreeTradeData);
}
break;
2018-12-29 19:46:15 +01:00
}
if (!HasLinkErrorOccurred())
{
RunTasks();
}
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void CB2_FreeTradeData(void)
2018-12-29 19:46:15 +01:00
{
if (!gPaletteFade.active)
{
FreeAllWindowBuffers();
Free(GetBgTilemapBuffer(3));
Free(GetBgTilemapBuffer(1));
Free(GetBgTilemapBuffer(0));
FreeMonSpritesGfx();
2019-10-09 11:56:44 +02:00
FREE_AND_SET_NULL(sTradeData);
2018-12-29 19:46:15 +01:00
if (gWirelessCommType)
2019-04-01 00:59:52 +02:00
DestroyWirelessStatusIndicatorSprite();
2018-12-29 19:46:15 +01:00
SetMainCallback2(gMain.savedCallback);
}
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
void DoInGameTradeScene(void)
{
ScriptContext2_Enable();
2019-10-04 13:42:45 +02:00
CreateTask(Task_InGameTrade, 10);
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2018-12-29 19:46:15 +01:00
}
2019-10-04 13:42:45 +02:00
static void Task_InGameTrade(u8 taskId)
2018-12-29 19:46:15 +01:00
{
if (!gPaletteFade.active)
{
2019-10-04 13:42:45 +02:00
SetMainCallback2(CB2_InGameTrade);
2019-12-17 09:24:44 +01:00
gFieldCallback = FieldCB_ContinueScriptHandleMusic;
2018-12-29 19:46:15 +01:00
DestroyTask(taskId);
}
}
2019-10-04 01:39:37 +02:00
static void CheckPartnersMonForRibbons(void)
2018-12-29 19:46:15 +01:00
{
u8 i;
u8 numRibbons = 0;
2022-06-09 19:59:21 +02:00
for (i = 0; i < (MON_DATA_UNUSED_RIBBONS - MON_DATA_CHAMPION_RIBBON); i ++)
2018-12-29 19:46:15 +01:00
{
2019-10-04 01:39:37 +02:00
numRibbons += GetMonData(&gEnemyParty[gSelectedTradeMonPositions[TRADE_PARTNER] % PARTY_SIZE], MON_DATA_CHAMPION_RIBBON + i);
2018-12-29 19:46:15 +01:00
}
if (numRibbons != 0)
FlagSet(FLAG_SYS_RIBBON_GET);
}
2022-07-29 23:00:07 +02:00
void LoadTradeAnimGfx(void)
2018-12-29 19:46:15 +01:00
{
2019-10-05 16:41:37 +02:00
InitTradeBgInternal();
2018-12-29 19:46:15 +01:00
}
2019-10-04 01:39:37 +02:00
void DrawTextOnTradeWindow(u8 windowId, const u8 *str, u8 speed)
2018-12-29 19:46:15 +01:00
{
FillWindowPixelBuffer(windowId, PIXEL_FILL(15));
2019-10-09 11:56:44 +02:00
sTradeData->textColors[0] = TEXT_DYNAMIC_COLOR_6;
sTradeData->textColors[1] = TEXT_COLOR_WHITE;
sTradeData->textColors[2] = TEXT_COLOR_GREEN;
2021-10-30 22:47:37 +02:00
AddTextPrinterParameterized4(windowId, FONT_NORMAL, 0, 2, 0, 0, sTradeData->textColors, speed, str);
2021-11-03 20:29:18 +01:00
CopyWindowToVram(windowId, COPYWIN_FULL);
2018-12-29 19:46:15 +01:00
}
#define idx data[0]
#define counter data[1]
#define signalComingBack data[2]
2019-10-09 11:56:44 +02:00
static void Task_AnimateWirelessSignal(u8 taskId)
2018-12-29 19:46:15 +01:00
{
s16 *data = gTasks[taskId].data;
u16 paletteIdx = sWirelessSignalTiming[idx][0] * 16;
2018-12-29 19:46:15 +01:00
2019-10-09 11:56:44 +02:00
if (!signalComingBack)
2018-12-29 19:46:15 +01:00
{
if (paletteIdx == 256)
LoadPalette(sWirelessSignalNone_Pal, 0x30, 32);
2018-12-29 19:46:15 +01:00
else
LoadPalette(&sWirelessSignalSend_Pal[paletteIdx], 0x30, 32);
2018-12-29 19:46:15 +01:00
}
else
{
if (paletteIdx == 256)
LoadPalette(sWirelessSignalNone_Pal, 0x30, 32);
2018-12-29 19:46:15 +01:00
else
LoadPalette(&sWirelessSignalRecv_Pal[paletteIdx], 0x30, 32);
2018-12-29 19:46:15 +01:00
}
if (sWirelessSignalTiming[idx][0] == 0 && counter == 0)
2020-08-21 00:02:00 +02:00
PlaySE(SE_M_HEAL_BELL);
2018-12-29 19:46:15 +01:00
if (counter == sWirelessSignalTiming[idx][1])
2018-12-29 19:46:15 +01:00
{
idx++;
counter = 0;
if (sWirelessSignalTiming[idx][1] == 0xFF)
2018-12-29 19:46:15 +01:00
{
DestroyTask(taskId);
}
}
else
{
counter++;
2018-12-29 19:46:15 +01:00
}
}
#undef idx
#undef counter
2019-10-09 11:56:44 +02:00
#undef signalComingBack
static void Task_NarrowWindowForCrossing_Wireless(u8 taskId)
2018-12-29 19:46:15 +01:00
{
s16 *data = gTasks[taskId].data;
if (data[0] == 0)
{
sTradeData->wirelessWinLeft = sTradeData->wirelessWinRight = DISPLAY_WIDTH / 2;
sTradeData->wirelessWinTop = 0;
2021-04-15 08:04:01 +02:00
sTradeData->wirelessWinBottom = DISPLAY_HEIGHT;
2018-12-29 19:46:15 +01:00
SetGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON);
SetGpuReg(REG_OFFSET_WINOUT, WINOUT_WIN01_OBJ);
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG0 |
WININ_WIN0_BG1 |
WININ_WIN0_OBJ);
}
SetGpuReg(REG_OFFSET_WIN0H, WIN_RANGE2(sTradeData->wirelessWinLeft, sTradeData->wirelessWinRight));
SetGpuReg(REG_OFFSET_WIN0V, WIN_RANGE2(sTradeData->wirelessWinTop, sTradeData->wirelessWinBottom));
2019-01-05 19:25:46 +01:00
2018-12-29 19:46:15 +01:00
data[0]++;
sTradeData->wirelessWinLeft -= 5;
sTradeData->wirelessWinRight += 5;
2018-12-29 19:46:15 +01:00
if (sTradeData->wirelessWinLeft < 80)
2018-12-29 19:46:15 +01:00
DestroyTask(taskId);
}
2018-12-29 20:50:24 +01:00
static void Task_NarrowWindowForCrossing_Cable(u8 taskId)
2018-12-29 20:50:24 +01:00
{
s16 *data = gTasks[taskId].data;
if (data[0] == 0)
{
sTradeData->wirelessWinLeft = 80;
sTradeData->wirelessWinRight = DISPLAY_WIDTH - 80;
2018-12-29 20:50:24 +01:00
SetGpuReg(REG_OFFSET_WINOUT, WINOUT_WIN01_OBJ);
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG0 |
WININ_WIN0_BG1 |
WININ_WIN0_OBJ);
}
SetGpuReg(REG_OFFSET_WIN0H, WIN_RANGE2(sTradeData->wirelessWinLeft, sTradeData->wirelessWinRight));
SetGpuReg(REG_OFFSET_WIN0V, WIN_RANGE2(sTradeData->wirelessWinTop, sTradeData->wirelessWinBottom));
2019-01-05 19:25:46 +01:00
if (sTradeData->wirelessWinLeft != DISPLAY_WIDTH / 2)
2018-12-29 20:50:24 +01:00
{
data[0]++;
sTradeData->wirelessWinLeft += 5;
sTradeData->wirelessWinRight -= 5;
2018-12-29 20:50:24 +01:00
if (sTradeData->wirelessWinLeft > DISPLAY_WIDTH / 2 - 5)
2018-12-29 20:50:24 +01:00
BlendPalettes(0x8, 0, RGB_WHITEALPHA);
}
else
{
ClearGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON);
DestroyTask(taskId);
}
}
static void CB2_SaveAndEndWirelessTrade(void)
2018-12-29 20:50:24 +01:00
{
switch (gMain.state)
{
case 0:
gMain.state = 1;
StringExpandPlaceholders(gStringVar4, gText_CommunicationStandby5);
DrawTextOnTradeWindow(0, gStringVar4, 0);
break;
case 1:
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(0);
gMain.state = 2;
sTradeData->timer = 0;
break;
case 2:
if (_IsLinkTaskFinished())
{
gMain.state = 3;
StringExpandPlaceholders(gStringVar4, gText_SavingDontTurnOffPower);
2019-10-04 01:39:37 +02:00
DrawTextOnTradeWindow(0, gStringVar4, 0);
IncrementGameStat(GAME_STAT_POKEMON_TRADES);
2021-10-29 04:54:41 +02:00
LinkFullSave_Init();
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
}
break;
case 3:
if (++sTradeData->timer == 5)
gMain.state = 4;
break;
case 4:
2021-10-29 04:54:41 +02:00
if (LinkFullSave_WriteSector())
{
gMain.state = 5;
}
else
{
2019-10-09 11:56:44 +02:00
sTradeData->timer = 0;
gMain.state = 3;
}
break;
case 5:
2021-10-29 04:54:41 +02:00
LinkFullSave_ReplaceLastSector();
gMain.state = 6;
sTradeData->timer = 0;
break;
case 6:
if (++sTradeData->timer > 10)
{
if (GetMultiplayerId() == 0)
sTradeData->timer = Random() % 30;
2018-12-29 20:50:24 +01:00
else
sTradeData->timer = 0;
gMain.state = 7;
}
break;
case 7:
if (sTradeData->timer == 0)
{
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(1);
gMain.state = 8;
}
else
{
sTradeData->timer--;
}
break;
case 8:
if (_IsLinkTaskFinished())
{
2022-08-05 16:31:50 +02:00
LinkFullSave_SetLastSectorSignature();
gMain.state = 9;
}
break;
case 9:
if (++sTradeData->timer > 60)
{
gMain.state++;
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(2);
}
break;
case 10:
if (_IsLinkTaskFinished())
{
FadeOutBGM(3);
2021-02-24 17:01:02 +01:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
gMain.state = 11;
}
break;
case 11:
if (!gPaletteFade.active && IsBGMStopped() == TRUE)
{
2020-08-13 09:09:47 +02:00
SetTradeLinkStandbyCallback(3);
gMain.state = 12;
}
break;
case 12:
if (_IsLinkTaskFinished())
{
gSoftResetDisabled = FALSE;
SetMainCallback2(CB2_FreeTradeData);
}
break;
2018-12-29 20:50:24 +01:00
}
if (!HasLinkErrorOccurred())
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}