pokeemerald/src/contest.c
2021-07-07 09:11:52 -04:00

6115 lines
193 KiB
C

#include "global.h"
#include "gpu_regs.h"
#include "bg.h"
#include "malloc.h"
#include "battle.h"
#include "battle_anim.h"
#include "contest.h"
#include "contest_link.h"
#include "data.h"
#include "decompress.h"
#include "graphics.h"
#include "link.h"
#include "m4a.h"
#include "main.h"
#include "menu.h"
#include "overworld.h"
#include "palette.h"
#include "random.h"
#include "new_game.h"
#include "script.h"
#include "sound.h"
#include "sprite.h"
#include "string_util.h"
#include "task.h"
#include "text.h"
#include "tv.h"
#include "scanline_effect.h"
#include "util.h"
#include "contest_util.h"
#include "dma3.h"
#include "battle_message.h"
#include "event_scripts.h"
#include "event_data.h"
#include "strings.h"
#include "contest_effect.h"
#include "contest_link.h"
#include "international_string_util.h"
#include "data.h"
#include "contest_ai.h"
#include "constants/event_objects.h"
#include "constants/items.h"
#include "constants/moves.h"
#include "constants/rgb.h"
#include "constants/songs.h"
// This file's functions.
static void LoadContestPalettes(void);
static void Task_StartContestWaitFade(u8 taskId);
static void Task_TryStartLinkContest(u8 taskId);
static void Task_CommunicateMonIdxs(u8 taskId);
static void Task_EndCommunicateMonIdxs(u8 taskId);
static void Task_ReadyStartLinkContest(u8 taskId);
static bool8 SetupContestGraphics(u8 *stateVar);
static void Task_WaitToRaiseCurtainAtStart(u8 taskId);
static void Task_RaiseCurtainAtStart(u8 taskId);
static void VBlankCB_Contest(void);
static void CB2_ContestMain(void);
static void Task_DisplayAppealNumberText(u8 taskId);
static void Task_TryShowMoveSelectScreen(u8 taskId);
static void Task_ShowMoveSelectScreen(u8 taskId);
static void Task_HandleMoveSelectInput(u8 taskId);
static void DrawMoveSelectArrow(s8);
static void EraseMoveSelectArrow(s8);
static void Task_SelectedMove(u8 taskId);
static void Task_EndCommunicateMoveSelections(u8 taskId);
static void Task_HideMoveSelectScreen(u8 taskId);
static void Task_HideApplauseMeterForAppealStart(u8 taskId);
static void Task_WaitHideApplauseMeterForAppealStart(u8 taskId);
static void Task_AppealSetup(u8 taskId);
static void Task_DoAppeals(u8 taskId);
static void Task_EndWaitForLink(u8);
static void SpriteCB_MonSlideIn(struct Sprite *);
static void SpriteCB_MonSlideOut(struct Sprite *);
static void Task_FinishRoundOfAppeals(u8);
static void Task_ReadyUpdateHeartSliders(u8);
static void Task_UpdateHeartSliders(u8);
static void Task_WaitForHeartSliders(u8);
static void sub_80DA348(u8);
static void Task_WaitPrintRoundResult(u8);
static void Task_PrintRoundResultText(u8);
static void Task_ReUpdateHeartSliders(u8);
static void Task_WaitForHeartSlidersAgain(u8);
static void Task_DropCurtainAtRoundEnd(u8);
static void Task_TryStartNextRoundOfAppeals(u8);
static void Task_StartNewRoundOfAppeals(u8);
static void Task_EndAppeals(u8);
static void Task_WaitForOutOfTimeMsg(u8);
static void Task_DropCurtainAtAppealsEnd(u8);
static void Task_TryCommunicateFinalStandings(u8);
static void Task_CommunicateFinalStandings(u8);
static void Task_EndCommunicateFinalStandings(u8);
static void Task_ContestReturnToField(u8);
static void FieldCB_ContestReturnToField(void);
static bool8 IsPlayerLinkLeader(void);
static void PrintContestantTrainerName(u8);
static void PrintContestantTrainerNameWithColor(u8 a0, u8 a1);
static void PrintContestantMonName(u8);
static void PrintContestantMonNameWithColor(u8, u8);
static u8 CreateJudgeSprite(void);
static u8 CreateJudgeSpeechBubbleSprite(void);
static u8 CreateContestantSprite(u16, u32, u32, u32);
static void PrintContestMoveDescription(u16);
static u16 SanitizeSpecies(u16);
static void ContestClearGeneralTextWindow(void);
static u16 GetChosenMove(u8);
static void GetAllChosenMoves(void);
static void ContestPrintLinkStandby(void);
static void FillContestantWindowBgs(void);
static void CreateSliderHeartSprites(void);
static void SetBottomSliderHeartsInvisibility(bool8);
static void CreateNextTurnSprites(void);
static void CreateApplauseMeterSprite(void);
static void CreateJudgeAttentionEyeTask(void);
static void CreateUnusedBlendTask(void);
static void ContestDebugDoPrint(void);
static void DrawContestantWindows(void);
static void ApplyNextTurnOrder(void);
static void SlideApplauseMeterIn(void);
static void SlideApplauseMeterOut(void);
static void SetBgForCurtainDrop(void);
static void UpdateContestantBoxOrder(void);
static void Task_StartDropCurtainAtRoundEnd(u8);
static void AnimateSliderHearts(u8);
static void CreateInvisibleBattleTargetSprite(void);
static void Contest_StartTextPrinter(const u8 *, u32);
static void ContestBG_FillBoxWithIncrementingTile(u8, u16, u8, u8, u8, u8, u8, s16);
static bool32 Contest_RunTextPrinters(void);
static void Contest_SetBgCopyFlags(u32 flagIndex);
static void CalculateFinalScores(void);
static void CalculateAppealMoveImpact(u8);
static void SetMoveAnimAttackerData(u8);
static void BlinkContestantBox(u8, u8);
static u8 CreateContestantBoxBlinkSprites(u8);
static u16 SanitizeMove(u16);
static void SetMoveSpecificAnimData(u8);
static void SetMoveTargetPosition(u16);
static void ClearMoveAnimData(u8);
static void StopFlashJudgeAttentionEye(u8);
static void DrawUnnervedSymbols(void);
static void PrintAppealMoveResultText(u8, u8);
static void DoJudgeSpeechBubble(u8);
static void ShowHideNextTurnGfx(bool8);
static u8 UpdateAppealHearts(s16, s16, u8);
static bool8 UpdateConditionStars(u8, u8);
static bool8 DrawStatusSymbol(u8);
static void DrawStatusSymbols(void);
static void StartStopFlashJudgeAttentionEye(u8);
static void BlendAudienceBackground(s8, s8);
static void ShowAndUpdateApplauseMeter(s8 unused);
static void AnimateAudience(void);
static void UpdateApplauseMeter(void);
static void RankContestants(void);
static void SetAttentionLevels(void);
static void UpdateHeartSliders(void);
static bool8 SlidersDoneUpdating(void);
static void ContestBG_FillBoxWithTile(u8, u16, u8, u8, u8, u8, u8);
static void Contest_PrintTextToBg0WindowStd(u32, const u8 *);
static s16 GetContestantRound2Points(u8);
static void DetermineFinalStandings(void);
static bool8 DidContestantPlaceHigher(s32, s32, struct ContestFinalStandings *);
static void Task_UpdateAppealHearts(u8);
static void SpriteCB_UpdateHeartSlider(struct Sprite *);
static void Task_FlashJudgeAttentionEye(u8);
static void Task_StopFlashJudgeAttentionEye(u8);
static void Task_UnusedBlend(u8);
static void InitUnusedBlendTaskData(u8);
static void UpdateBlendTaskContestantData(u8);
static void SpriteCB_BlinkContestantBox(struct Sprite *);
static void SpriteCB_EndBlinkContestantBox(struct Sprite *sprite);
static u8 StartApplauseOverflowAnimation(void);
static void Task_ApplauseOverflowAnimation(u8);
static void Task_SlideApplauseMeterIn(u8);
static void Task_SlideApplauseMeterOut(u8);
static void Task_ShowAndUpdateApplauseMeter(u8);
static void Task_AnimateAudience(u8);
static void Task_BlendAudienceBackground(u8);
static const u8 *GetTurnOrderNumberGfx(u8);
static void Task_UpdateCurtainDropAtRoundEnd(u8);
static void Task_ResetForNextRound(u8);
static void Task_WaitRaiseCurtainAtRoundEnd(u8);
static void Task_StartRaiseCurtainAtRoundEnd(u8);
static void Task_WaitForSliderHeartAnim(u8);
static void SetBattleTargetSpritePosition(void);
static void CalculateContestLiveUpdateData(void);
static void SetConestLiveUpdateTVData(void);
static void SetContestLiveUpdateFlags(u8);
static void ContestDebugPrintBitStrings(void);
static void StripPlayerNameForLinkContest(u8 *);
static void StripMonNameForLinkContest(u8 *, s32);
static void SwapMoveDescAndContestTilemaps(void);
// An index into a palette where the text color for each contestant is stored.
// Contestant 0 will use palette color 10, contestant 1 will use color 11, etc.
#define CONTESTANT_TEXT_COLOR_START 10
enum {
// The "{Pokemon Name} / {Trainer Name}" windows.
WIN_CONTESTANT0_NAME,
WIN_CONTESTANT1_NAME,
WIN_CONTESTANT2_NAME,
WIN_CONTESTANT3_NAME,
WIN_GENERAL_TEXT,
// The available moves, from top to bottom
WIN_MOVE0,
WIN_MOVE1,
WIN_MOVE2,
WIN_MOVE3,
// The small "/" character between the move category and the
// appeal/jam display
WIN_SLASH,
WIN_MOVE_DESCRIPTION
};
enum {
JUDGE_SYMBOL_SWIRL,
JUDGE_SYMBOL_SWIRL_UNUSED,
JUDGE_SYMBOL_ONE_EXCLAMATION,
JUDGE_SYMBOL_TWO_EXCLAMATIONS,
JUDGE_SYMBOL_NUMBER_ONE_UNUSED,
JUDGE_SYMBOL_NUMBER_ONE,
JUDGE_SYMBOL_NUMBER_FOUR,
JUDGE_SYMBOL_QUESTION_MARK,
JUDGE_SYMBOL_STAR,
};
enum {
STAT_SYMBOL_CIRCLE,
STAT_SYMBOL_WAVE,
STAT_SYMBOL_X,
STAT_SYMBOL_SWIRL,
STAT_SYMBOL_SQUARE,
};
enum {
CONTEST_DEBUG_MODE_OFF,
CONTEST_DEBUG_MODE_PRINT_POINT_TOTAL,
CONTEST_DEBUG_MODE_PRINT_WINNER_FLAGS,
CONTEST_DEBUG_MODE_PRINT_LOSER_FLAGS
};
#define MOVE_WINDOWS_START WIN_MOVE0
#define TAG_CONTEST_SYMBOLS_PAL 0xABE0
#define TAG_JUDGE_SYMBOLS_GFX 0xABE0
#define TAG_FACES_GFX 0xABE1
#define TAG_APPLAUSE_METER 0xABE2
#define TAG_SLIDER_HEART 0x4E20
#define TAG_JUDGE 0x4E21
#define TAG_NEXT_TURN_PAL 0x4E22
#define TAG_NEXT_TURN_1_GFX 0x4E22
#define TAG_NEXT_TURN_2_GFX 0x4E23
#define TAG_NEXT_TURN_3_GFX 0x4E24
#define TAG_NEXT_TURN_4_GFX 0x4E25
#define TAG_BLINK_EFFECT_CONTESTANT0 0x80E8
#define TAG_BLINK_EFFECT_CONTESTANT1 0x80E9
#define TAG_BLINK_EFFECT_CONTESTANT2 0x80EA
#define TAG_BLINK_EFFECT_CONTESTANT3 0x80EB
#define TILE_FILLED_APPEAL_HEART 0x5012
#define TILE_FILLED_JAM_HEART 0x5014
#define TILE_EMPTY_APPEAL_HEART 0x5035
#define TILE_EMPTY_JAM_HEART 0x5036
enum {
SLIDER_HEART_ANIM_NORMAL,
SLIDER_HEART_ANIM_DISAPPEAR,
SLIDER_HEART_ANIM_APPEAR,
};
// States for Task_DoAppeals
enum {
APPEALSTATE_START_TURN,
APPEALSTATE_WAIT_LINK,
APPEALSTATE_CHECK_SKIP_TURN,
APPEALSTATE_SLIDE_MON_IN,
APPEALSTATE_WAIT_SLIDE_MON,
APPEALSTATE_PRINT_USED_MOVE_MSG,
APPEALSTATE_WAIT_USED_MOVE_MSG,
APPEALSTATE_MOVE_ANIM,
APPEALSTATE_WAIT_MOVE_ANIM,
APPEALSTATE_MOVE_ANIM_MULTITURN,
APPEALSTATE_SLIDE_MON_OUT,
APPEALSTATE_FREE_MON_SPRITE,
APPEALSTATE_UPDATE_MOVE_USERS_HEARTS,
APPEALSTATE_WAIT_MOVE_USERS_HEARTS,
APPEALSTATE_PRINT_COMBO_MSG,
APPEALSTATE_TRY_UPDATE_HEARTS_FROM_COMBO,
APPEALSTATE_WAIT_HEARTS_FROM_COMBO,
APPEALSTATE_CHECK_REPEATED_MOVE,
APPEALSTATE_WAIT_HEARTS_FROM_REPEAT,
APPEALSTATE_UPDATE_HEARTS_FROM_REPEAT,
APPEALSTATE_START_TURN_END_DELAY,
APPEALSTATE_TURN_END_DELAY,
APPEALSTATE_START_NEXT_TURN,
APPEALSTATE_TRY_PRINT_MOVE_RESULT,
APPEALSTATE_WAIT_MOVE_RESULT_MSG,
APPEALSTATE_UPDATE_OPPONENTS,
APPEALSTATE_UPDATE_OPPONENT,
APPEALSTATE_WAIT_OPPONENT_RESPONSE_MSG,
APPEALSTATE_UPDATE_OPPONENT_HEARTS,
APPEALSTATE_WAIT_OPPONENT_HEARTS,
APPEALSTATE_UPDATE_OPPONENT_STATUS,
APPEALSTATE_PRINT_SKIP_TURN_MSG,
APPEALSTATE_WAIT_SKIP_TURN_MSG,
APPEALSTATE_PRINT_TOO_NERVOUS_MSG,
APPEALSTATE_WAIT_TOO_NERVOUS_MSG,
APPEALSTATE_TRY_JUDGE_STAR,
APPEALSTATE_WAIT_JUDGE_STAR,
APPEALSTATE_UPDATE_MOVE_USERS_STARS,
APPEALSTATE_WAIT_MOVE_USERS_STARS,
APPEALSTATE_UPDATE_OPPONENT_STARS,
APPEALSTATE_WAIT_OPPONENT_STARS,
APPEALSTATE_UPDATE_CROWD,
APPEALSTATE_42, // Unused state
APPEALSTATE_WAIT_EXCITEMENT_HEARTS,
APPEALSTATE_44, // Unused state
APPEALSTATE_WAIT_JUDGE_COMBO,
APPEALSTATE_WAIT_JUDGE_REPEATED_MOVE,
APPEALSTATE_TRY_SHOW_NEXT_TURN_GFX,
APPEALSTATE_CHECK_TURN_ORDER_MOD,
APPEALSTATE_WAIT_JUDGE_TURN_ORDER,
APPEALSTATE_UPDATE_MOVE_USERS_STATUS,
APPEALSTATE_TRY_PRINT_SKIP_NEXT_TURN_MSG,
APPEALSTATE_WAIT_SKIP_NEXT_TURN_MSG,
APPEALSTATE_DO_CROWD_UNEXCITED,
APPEALSTATE_DO_CROWD_EXCITED,
APPEALSTATE_SLIDE_APPLAUSE_OUT,
APPEALSTATE_WAIT_SLIDE_APPLAUSE,
APPEALSTATE_PRINT_CROWD_WATCHES_MSG,
APPEALSTATE_PRINT_MON_MOVE_IGNORED_MSG,
APPEALSTATE_WAIT_MON_MOVE_IGNORED_MSG,
};
// EWRAM vars.
EWRAM_DATA struct ContestPokemon gContestMons[CONTESTANT_COUNT] = {0};
EWRAM_DATA s16 gContestMonRound1Points[CONTESTANT_COUNT] = {0}; // "Round 1" points are based on condition
EWRAM_DATA s16 gContestMonTotalPoints[CONTESTANT_COUNT] = {0}; // Round 1 points + Round 2 points
EWRAM_DATA s16 gContestMonAppealPointTotals[CONTESTANT_COUNT] = {0};
EWRAM_DATA s16 gContestMonRound2Points[CONTESTANT_COUNT] = {0}; // "Round 2" points are just appeal points * 2
EWRAM_DATA u8 gContestFinalStandings[CONTESTANT_COUNT] = {0};
EWRAM_DATA u8 gContestMonPartyIndex = 0;
EWRAM_DATA u8 gContestPlayerMonIndex = 0;
EWRAM_DATA u8 gContestantTurnOrder[CONTESTANT_COUNT] = {0};
EWRAM_DATA u8 gLinkContestFlags = 0;
// Bit 0: Is a link contest
// Bit 1: Link contest uses wireless adapter
EWRAM_DATA u8 gContestLinkLeaderIndex = 0;
EWRAM_DATA u16 gSpecialVar_ContestCategory = 0;
EWRAM_DATA u16 gSpecialVar_ContestRank = 0;
EWRAM_DATA u8 gNumLinkContestPlayers = 0;
EWRAM_DATA u8 gHighestRibbonRank = 0;
EWRAM_DATA struct ContestResources *gContestResources = NULL;
EWRAM_DATA u8 sContestBgCopyFlags = 0;
EWRAM_DATA struct ContestWinner gCurContestWinner = {0};
EWRAM_DATA bool8 gCurContestWinnerIsForArtist = 0;
EWRAM_DATA u8 gCurContestWinnerSaveIdx = 0;
// IWRAM common vars.
u32 gContestRngValue;
extern const u8 gText_LinkStandby4[];
extern const u8 gText_BDot[];
extern const u8 gText_CDot[];
extern void (*const gContestEffectFuncs[])(void);
static const u8 sSliderHeartYPositions[CONTESTANT_COUNT] =
{
36, 76, 116, 156
};
// The "Next Turn" sprites cover up the slider, so the y positions are the same as above
static const u8 sNextTurnSpriteYPositions[CONTESTANT_COUNT] =
{
36, 76, 116, 156
};
static const struct SpriteSheet sSpriteSheet_SliderHeart =
{
.data = gContestSliderHeart_Gfx,
.size = 0x20,
.tag = TAG_SLIDER_HEART
};
static const struct OamData sOam_SliderHeart =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(8x8),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(8x8),
.tileNum = 0,
.priority = 0,
.paletteNum = 0,
.affineParam = 0
};
static const union AffineAnimCmd sAffineAnim_SliderHeart_Normal[] =
{
AFFINEANIMCMD_FRAME(256, 256, 0, 0),
AFFINEANIMCMD_END,
};
static const union AffineAnimCmd sAffineAnim_SliderHeart_SpinDisappear[] =
{
AFFINEANIMCMD_FRAME(256, 256, 0, 0),
AFFINEANIMCMD_FRAME(-10, -10, -20, 20),
AFFINEANIMCMD_END
};
static const union AffineAnimCmd sAffineAnim_SliderHeart_SpinAppear[] =
{
AFFINEANIMCMD_FRAME(56, 56, 0, 0),
AFFINEANIMCMD_FRAME(10, 10, 20, 20),
AFFINEANIMCMD_END
};
static const union AffineAnimCmd* const sAffineAnims_SliderHeart[] =
{
[SLIDER_HEART_ANIM_NORMAL] = sAffineAnim_SliderHeart_Normal,
[SLIDER_HEART_ANIM_DISAPPEAR] = sAffineAnim_SliderHeart_SpinDisappear,
[SLIDER_HEART_ANIM_APPEAR] = sAffineAnim_SliderHeart_SpinAppear
};
static const struct SpriteTemplate sSpriteTemplate_SliderHeart =
{
.tileTag = TAG_SLIDER_HEART,
.paletteTag = TAG_CONTEST_SYMBOLS_PAL,
.oam = &sOam_SliderHeart,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = sAffineAnims_SliderHeart,
.callback = SpriteCallbackDummy
};
static const struct CompressedSpriteSheet sSpriteSheet_NextTurn[CONTESTANT_COUNT] =
{
{
.data = gContestNextTurnGfx,
.size = 0x100,
.tag = TAG_NEXT_TURN_1_GFX
},
{
.data = gContestNextTurnGfx,
.size = 0x100,
.tag = TAG_NEXT_TURN_2_GFX
},
{
.data = gContestNextTurnGfx,
.size = 0x100,
.tag = TAG_NEXT_TURN_3_GFX
},
{
.data = gContestNextTurnGfx,
.size = 0x100,
.tag = TAG_NEXT_TURN_4_GFX
}
};
static const struct SpritePalette sSpritePalette_NextTurn =
{
.data = gContestPal,
.tag = TAG_NEXT_TURN_PAL
};
static const struct OamData sOam_NextTurn =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(32x8),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(32x8),
.tileNum = 0,
.priority = 0,
.paletteNum = 0,
.affineParam = 0
};
static const struct SpriteTemplate sSpriteTemplates_NextTurn[CONTESTANT_COUNT] =
{
{
.tileTag = TAG_NEXT_TURN_1_GFX,
.paletteTag = TAG_NEXT_TURN_PAL,
.oam = &sOam_NextTurn,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
},
{
.tileTag = TAG_NEXT_TURN_2_GFX,
.paletteTag = TAG_NEXT_TURN_PAL,
.oam = &sOam_NextTurn,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
},
{
.tileTag = TAG_NEXT_TURN_3_GFX,
.paletteTag = TAG_NEXT_TURN_PAL,
.oam = &sOam_NextTurn,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
},
{
.tileTag = TAG_NEXT_TURN_4_GFX,
.paletteTag = TAG_NEXT_TURN_PAL,
.oam = &sOam_NextTurn,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
}
};
static const struct Subsprite sSubsprites_NextTurn[] =
{
{
.x = -28,
.y = -4,
.shape = SPRITE_SHAPE(32x8),
.size = SPRITE_SIZE(32x8),
.tileOffset = 0,
.priority = 0
},
{
.x = 4,
.y = -4,
.shape = SPRITE_SHAPE(32x8),
.size = SPRITE_SIZE(32x8),
.tileOffset = 4,
.priority = 0
}
};
static const struct SubspriteTable sSubspriteTable_NextTurn[] =
{
{
.subspriteCount = ARRAY_COUNT(sSubsprites_NextTurn),
.subsprites = sSubsprites_NextTurn
}
};
// Unused
static const struct CompressedSpriteSheet sSpriteSheet_Faces =
{
.data = gContestFaces_Gfx,
.size = 0x180,
.tag = TAG_FACES_GFX
};
static const struct OamData sOam_Faces =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(16x16),
.x = 0,
.size = SPRITE_SIZE(16x16),
.tileNum = 0,
.priority = 0,
.paletteNum = 0,
};
// Unused
static const struct SpriteTemplate sSpriteTemplate_Faces =
{
.tileTag = TAG_FACES_GFX,
.paletteTag = TAG_CONTEST_SYMBOLS_PAL,
.oam = &sOam_Faces,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy
};
static const struct CompressedSpriteSheet sSpriteSheet_ApplauseMeter =
{
.data = gContestApplauseGfx,
.size = 0x400,
.tag = TAG_APPLAUSE_METER
};
static const struct SpritePalette sSpritePalette_ApplauseMeter =
{
.data = gContestPal,
.tag = TAG_APPLAUSE_METER
};
static const struct OamData sOam_ApplauseMeter =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x32),
.x = 0,
.size = SPRITE_SIZE(64x32),
.tileNum = 0,
.priority = 0,
.paletteNum = 0,
};
static const struct SpriteTemplate sSpriteTemplate_ApplauseMeter =
{
.tileTag = TAG_APPLAUSE_METER,
.paletteTag = TAG_APPLAUSE_METER,
.oam = &sOam_ApplauseMeter,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy
};
const struct OamData sOam_Judge =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x64),
.x = 0,
.size = SPRITE_SIZE(64x64),
.tileNum = 0,
.priority = 3,
.paletteNum = 2,
};
const struct SpriteTemplate sSpriteTemplate_Judge =
{
.tileTag = TAG_JUDGE,
.paletteTag = TAG_JUDGE,
.oam = &sOam_Judge,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy
};
const struct CompressedSpriteSheet sSpriteSheet_Judge =
{
.data = gContestJudgeGfx,
.size = 0x800,
.tag = TAG_JUDGE
};
static const struct CompressedSpriteSheet sSpriteSheet_JudgeSymbols =
{
.data = gContestJudgeSymbolsGfx,
.size = 0x380,
.tag = TAG_JUDGE_SYMBOLS_GFX
};
const struct CompressedSpritePalette sSpritePalette_JudgeSymbols =
{
.data = gContest3Pal,
.tag = TAG_CONTEST_SYMBOLS_PAL
};
const struct SpriteTemplate sSpriteTemplate_JudgeSpeechBubble =
{
.tileTag = TAG_JUDGE_SYMBOLS_GFX,
.paletteTag = TAG_CONTEST_SYMBOLS_PAL,
.oam = &gOamData_AffineOff_ObjNormal_16x16,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy
};
static const u16 sText_Pal[] = INCBIN_U16("graphics/contest/text.gbapal");
#include "data/contest_text_tables.h"
static const struct BgTemplate sContestBgTemplates[] =
{
{
.bg = 0,
.charBaseIndex = 0,
.mapBaseIndex = 0x18,
.screenSize = 2,
.paletteMode = 0,
.priority = 0,
.baseTile = 0
},
{
.bg = 1,
.charBaseIndex = 2,
.mapBaseIndex = 0x1E,
.screenSize = 2,
.paletteMode = 0,
.priority = 1,
.baseTile = 0
},
{
.bg = 2,
.charBaseIndex = 0,
.mapBaseIndex = 0x1C,
.screenSize = 2,
.paletteMode = 0,
.priority = 0,
.baseTile = 0
},
{
.bg = 3,
.charBaseIndex = 0,
.mapBaseIndex = 0x1A,
.screenSize = 0,
.paletteMode = 0,
.priority = 3,
.baseTile = 0
}
};
static const struct WindowTemplate sContestWindowTemplates[] =
{
[WIN_CONTESTANT0_NAME] = {
.bg = 0,
.tilemapLeft = 18,
.tilemapTop = 0,
.width = 12,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x200
},
[WIN_CONTESTANT1_NAME] = {
.bg = 0,
.tilemapLeft = 18,
.tilemapTop = 5,
.width = 12,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x218
},
[WIN_CONTESTANT2_NAME] = {
.bg = 0,
.tilemapLeft = 18,
.tilemapTop = 10,
.width = 12,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x230
},
[WIN_CONTESTANT3_NAME] = {
.bg = 0,
.tilemapLeft = 18,
.tilemapTop = 15,
.width = 12,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x248
},
[WIN_GENERAL_TEXT] = {
.bg = 0,
.tilemapLeft = 1,
.tilemapTop = 15,
.width = 17,
.height = 4,
.paletteNum = 0xF,
.baseBlock = 0x260
},
[WIN_MOVE0] = {
.bg = 0,
.tilemapLeft = 1,
.tilemapTop = 31,
.width = 9,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x2A4
},
[WIN_MOVE1] = {
.bg = 0,
.tilemapLeft = 1,
.tilemapTop = 33,
.width = 9,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x2B6
},
[WIN_MOVE2] = {
.bg = 0,
.tilemapLeft = 1,
.tilemapTop = 35,
.width = 9,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x2C8
},
[WIN_MOVE3] = {
.bg = 0,
.tilemapLeft = 1,
.tilemapTop = 37,
.width = 9,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x2DA
},
[WIN_SLASH] = {
.bg = 0,
.tilemapLeft = 16,
.tilemapTop = 31,
.width = 1,
.height = 2,
.paletteNum = 0xF,
.baseBlock = 0x2EC
},
[WIN_MOVE_DESCRIPTION] = {
.bg = 0,
.tilemapLeft = 11,
.tilemapTop = 35,
.width = 18,
.height = 4,
.paletteNum = 0xF,
.baseBlock = 0x2EE
},
DUMMY_WIN_TEMPLATE
};
#include "data/contest_opponents.h"
static const struct CompressedSpriteSheet sSpriteSheets_ContestantsTurnBlinkEffect[CONTESTANT_COUNT] =
{
{
.data = gBlankGfxCompressed,
.size = 0x1000,
.tag = TAG_BLINK_EFFECT_CONTESTANT0
},
{
.data = gBlankGfxCompressed,
.size = 0x1000,
.tag = TAG_BLINK_EFFECT_CONTESTANT1
},
{
.data = gBlankGfxCompressed,
.size = 0x1000,
.tag = TAG_BLINK_EFFECT_CONTESTANT2
},
{
.data = gBlankGfxCompressed,
.size = 0x1000,
.tag = TAG_BLINK_EFFECT_CONTESTANT3
}
};
// Yup this is super dangerous but that's how it is here
static const struct SpritePalette sSpritePalettes_ContestantsTurnBlinkEffect[CONTESTANT_COUNT] =
{
{
.data = (u16*)(gHeap + 0x1A0A4),
.tag = TAG_BLINK_EFFECT_CONTESTANT0
},
{
.data = (u16*)(gHeap + 0x1A0C4),
.tag = TAG_BLINK_EFFECT_CONTESTANT1
},
{
.data = (u16*)(gHeap + 0x1A0E4),
.tag = TAG_BLINK_EFFECT_CONTESTANT2
},
{
.data = (u16*)(gHeap + 0x1A104),
.tag = TAG_BLINK_EFFECT_CONTESTANT3
}
};
const struct OamData sOam_ContestantsTurnBlinkEffect =
{
.y = 0,
.affineMode = ST_OAM_AFFINE_DOUBLE,
.objMode = ST_OAM_OBJ_BLEND,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x64),
.x = 0,
.size = SPRITE_SIZE(64x64),
.tileNum = 0,
.priority = 0,
.paletteNum = 0,
.affineParam = 0,
};
const union AffineAnimCmd sAffineAnim_ContestantsTurnBlinkEffect_0[] =
{
AFFINEANIMCMD_FRAME(0x100, 0x100, 0, 0),
AFFINEANIMCMD_END
};
const union AffineAnimCmd sAffineAnim_ContestantsTurnBlinkEffect_1[] =
{
AFFINEANIMCMD_FRAME(3, 3, 0, 15),
AFFINEANIMCMD_FRAME(-3, -3, 0, 15),
AFFINEANIMCMD_FRAME(3, 3, 0, 15),
AFFINEANIMCMD_FRAME(-3, -3, 0, 15),
AFFINEANIMCMD_END
};
const union AffineAnimCmd *const sAffineAnims_ContestantsTurnBlinkEffect[] =
{
sAffineAnim_ContestantsTurnBlinkEffect_0,
sAffineAnim_ContestantsTurnBlinkEffect_1
};
const struct SpriteTemplate sSpriteTemplates_ContestantsTurnBlinkEffect[CONTESTANT_COUNT] =
{
{
.tileTag = TAG_BLINK_EFFECT_CONTESTANT0,
.paletteTag = TAG_BLINK_EFFECT_CONTESTANT0,
.oam = &sOam_ContestantsTurnBlinkEffect,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = sAffineAnims_ContestantsTurnBlinkEffect,
.callback = SpriteCallbackDummy
},
{
.tileTag = TAG_BLINK_EFFECT_CONTESTANT1,
.paletteTag = TAG_BLINK_EFFECT_CONTESTANT1,
.oam = &sOam_ContestantsTurnBlinkEffect,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = sAffineAnims_ContestantsTurnBlinkEffect,
.callback = SpriteCallbackDummy
},
{
.tileTag = TAG_BLINK_EFFECT_CONTESTANT2,
.paletteTag = TAG_BLINK_EFFECT_CONTESTANT2,
.oam = &sOam_ContestantsTurnBlinkEffect,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = sAffineAnims_ContestantsTurnBlinkEffect,
.callback = SpriteCallbackDummy
},
{
.tileTag = TAG_BLINK_EFFECT_CONTESTANT3,
.paletteTag = TAG_BLINK_EFFECT_CONTESTANT3,
.oam = &sOam_ContestantsTurnBlinkEffect,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = sAffineAnims_ContestantsTurnBlinkEffect,
.callback = SpriteCallbackDummy
}
};
static const s8 gContestExcitementTable[CONTEST_CATEGORIES_COUNT][CONTEST_CATEGORIES_COUNT] =
{
[CONTEST_CATEGORY_COOL] = {
[CONTEST_CATEGORY_COOL] = +1,
[CONTEST_CATEGORY_BEAUTY] = 0,
[CONTEST_CATEGORY_CUTE] = -1,
[CONTEST_CATEGORY_SMART] = -1,
[CONTEST_CATEGORY_TOUGH] = 0
},
[CONTEST_CATEGORY_BEAUTY] = {
[CONTEST_CATEGORY_COOL] = 0,
[CONTEST_CATEGORY_BEAUTY] = +1,
[CONTEST_CATEGORY_CUTE] = 0,
[CONTEST_CATEGORY_SMART] = -1,
[CONTEST_CATEGORY_TOUGH] = -1
},
[CONTEST_CATEGORY_CUTE] = {
[CONTEST_CATEGORY_COOL] = -1,
[CONTEST_CATEGORY_BEAUTY] = 0,
[CONTEST_CATEGORY_CUTE] = +1,
[CONTEST_CATEGORY_SMART] = 0,
[CONTEST_CATEGORY_TOUGH] = -1
},
[CONTEST_CATEGORY_SMART] = {
[CONTEST_CATEGORY_COOL] = -1,
[CONTEST_CATEGORY_BEAUTY] = -1,
[CONTEST_CATEGORY_CUTE] = 0,
[CONTEST_CATEGORY_SMART] = +1,
[CONTEST_CATEGORY_TOUGH] = 0
},
[CONTEST_CATEGORY_TOUGH] = {
[CONTEST_CATEGORY_COOL] = 0,
[CONTEST_CATEGORY_BEAUTY] = -1,
[CONTEST_CATEGORY_CUTE] = -1,
[CONTEST_CATEGORY_SMART] = 0,
[CONTEST_CATEGORY_TOUGH] = +1
}
};
static void TaskDummy1(u8 taskId)
{
}
void ResetLinkContestBoolean(void)
{
gLinkContestFlags = 0;
}
static void SetupContestGpuRegs(void)
{
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_OBJ_1D_MAP);
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
SetGpuReg(REG_OFFSET_BLDY, 0);
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG0 | WININ_WIN0_BG1 | WININ_WIN0_BG2 | WININ_WIN0_BG3 | WININ_WIN0_OBJ | WININ_WIN0_CLR | WININ_WIN1_BG0 | WININ_WIN1_BG1 | WININ_WIN1_BG2 | WININ_WIN1_BG3 | WININ_WIN1_OBJ | WININ_WIN1_CLR);
SetGpuReg(REG_OFFSET_WINOUT, WINOUT_WIN01_BG0 | WINOUT_WIN01_BG1 | WINOUT_WIN01_BG2 | WINOUT_WIN01_BG3 | WINOUT_WIN01_OBJ | WINOUT_WIN01_CLR | WINOUT_WINOBJ_BG0 | WINOUT_WINOBJ_BG1 | WINOUT_WINOBJ_BG2 | WINOUT_WINOBJ_BG3 | WINOUT_WINOBJ_OBJ | WINOUT_WINOBJ_CLR);
SetGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_BG0_ON | DISPCNT_BG1_ON | DISPCNT_BG2_ON | DISPCNT_BG3_ON | DISPCNT_OBJ_ON | DISPCNT_WIN0_ON | DISPCNT_WIN1_ON);
gBattle_BG0_X = 0;
gBattle_BG0_Y = 0;
gBattle_BG1_X = 0;
gBattle_BG1_Y = 0;
gBattle_BG2_X = 0;
gBattle_BG2_Y = 0;
gBattle_BG3_X = 0;
gBattle_BG3_Y = 0;
gBattle_WIN0H = 0;
gBattle_WIN0V = 0;
gBattle_WIN1H = 0;
gBattle_WIN1V = 0;
}
void LoadContestBgAfterMoveAnim(void)
{
s32 i;
LZDecompressVram(gContestMiscGfx, (void *)VRAM);
LZDecompressVram(gContestAudienceGfx, (void *)(BG_SCREEN_ADDR(4)));
CopyToBgTilemapBuffer(3, gOldContestGfx, 0, 0);
CopyBgTilemapBufferToVram(3);
LoadCompressedPalette(gOldContestPalette, 0, 0x200);
LoadContestPalettes();
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u32 contestantWindowId = 5 + i;
LoadPalette(eContestTempSave.cachedWindowPalettes[contestantWindowId], 16 * (5 + gContestantTurnOrder[i]), sizeof((eContestTempSave.cachedWindowPalettes[contestantWindowId])));
}
}
static void InitContestInfoBgs(void)
{
s32 i;
ResetBgsAndClearDma3BusyFlags(0);
InitBgsFromTemplates(0, sContestBgTemplates, ARRAY_COUNT(sContestBgTemplates));
SetBgAttribute(3, BG_ATTR_WRAPAROUND, 1);
for (i = 0; i < CONTESTANT_COUNT; i++)
{
SetBgTilemapBuffer(i, gContestResources->contestBgTilemaps[i]);
}
}
static void InitContestWindows(void)
{
InitWindows(sContestWindowTemplates);
DeactivateAllTextPrinters();
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
gTextFlags.canABSpeedUpPrint = FALSE;
}
else
{
gTextFlags.canABSpeedUpPrint = TRUE;
}
}
static void LoadContestPalettes(void)
{
s32 i;
LoadPalette(sText_Pal, 0xf0, 0x20);
FillPalette(RGB_BLACK, 0, 2);
for (i = 10; i < 14; i++)
LoadPalette(gPlttBufferUnfaded + 241, 240 + i, 2);
FillPalette(RGB(31, 17, 31), 0xF3, 2);
}
static void InitContestResources(void)
{
s32 i;
eContest = (struct Contest){};
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContest.unk[i] = 0xFF;
}
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i] = (struct ContestantStatus){};
}
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i].ranking = 0;
eContestantStatus[i].effectStringId = CONTEST_STRING_NONE;
eContestantStatus[i].effectStringId2 = CONTEST_STRING_NONE;
}
eContestAppealResults = (struct ContestAppealMoveResults){};
eContestAI = (struct ContestAIInfo){};
*gContestResources->excitement = (struct ContestExcitement){};
memset(eContestGfxState, 0, CONTESTANT_COUNT * sizeof(struct ContestGraphicsState));
if (!(gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK))
SortContestants(FALSE);
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i].nextTurnOrder = 0xFF;
eContest.prevTurnOrder[i] = gContestantTurnOrder[i];
}
// Calling this here while all the nextTurnOrder values are 0xFF will actually
// just reverse the turn order.
ApplyNextTurnOrder();
memset(gContestResources->tv, 0, sizeof(*gContestResources->tv) * CONTESTANT_COUNT);
}
static void AllocContestResources(void)
{
gContestResources = AllocZeroed(sizeof(struct ContestResources));
gContestResources->contest = AllocZeroed(sizeof(struct Contest));
gContestResources->status = AllocZeroed(sizeof(struct ContestantStatus) * CONTESTANT_COUNT);
gContestResources->appealResults = AllocZeroed(sizeof(struct ContestAppealMoveResults));
gContestResources->aiData = AllocZeroed(sizeof(struct ContestAIInfo));
gContestResources->excitement = AllocZeroed(sizeof(struct ContestExcitement) * CONTESTANT_COUNT);
gContestResources->gfxState = AllocZeroed(sizeof(struct ContestGraphicsState) * CONTESTANT_COUNT);
gContestResources->moveAnim = AllocZeroed(sizeof(struct ContestMoveAnimData));
gContestResources->tv = AllocZeroed(sizeof(struct ContestTV) * CONTESTANT_COUNT);
gContestResources->unused = AllocZeroed(sizeof(struct ContestUnused));
gContestResources->contestBgTilemaps[0] = AllocZeroed(0x1000);
gContestResources->contestBgTilemaps[1] = AllocZeroed(0x1000);
gContestResources->contestBgTilemaps[2] = AllocZeroed(0x1000);
gContestResources->contestBgTilemaps[3] = AllocZeroed(0x1000);
gContestResources->boxBlinkTiles1 = AllocZeroed(0x800);
gContestResources->boxBlinkTiles2 = AllocZeroed(0x800);
gContestResources->field_3c = AllocZeroed(0x2000);
gUnknown_0202305C = gContestResources->field_3c;
gUnknown_02023060 = gContestResources->contestBgTilemaps[1];
}
static void FreeContestResources(void)
{
FREE_AND_SET_NULL(gContestResources->contest);
FREE_AND_SET_NULL(gContestResources->status);
FREE_AND_SET_NULL(gContestResources->appealResults);
FREE_AND_SET_NULL(gContestResources->aiData);
FREE_AND_SET_NULL(gContestResources->excitement);
FREE_AND_SET_NULL(gContestResources->gfxState);
FREE_AND_SET_NULL(gContestResources->moveAnim);
FREE_AND_SET_NULL(gContestResources->tv);
FREE_AND_SET_NULL(gContestResources->unused);
FREE_AND_SET_NULL(gContestResources->contestBgTilemaps[0]);
FREE_AND_SET_NULL(gContestResources->contestBgTilemaps[1]);
FREE_AND_SET_NULL(gContestResources->contestBgTilemaps[2]);
FREE_AND_SET_NULL(gContestResources->contestBgTilemaps[3]);
FREE_AND_SET_NULL(gContestResources->boxBlinkTiles1);
FREE_AND_SET_NULL(gContestResources->boxBlinkTiles2);
FREE_AND_SET_NULL(gContestResources->field_3c);
FREE_AND_SET_NULL(gContestResources);
gUnknown_0202305C = NULL;
gUnknown_02023060 = NULL;
}
void CB2_StartContest(void)
{
switch (gMain.state)
{
case 0:
sContestBgCopyFlags = 0;
AllocContestResources();
AllocateMonSpritesGfx();
FREE_AND_SET_NULL(gMonSpritesGfxPtr->firstDecompressed);
gMonSpritesGfxPtr->firstDecompressed = AllocZeroed(0x4000);
SetVBlankCallback(NULL);
InitContestInfoBgs();
InitContestWindows();
SetupContestGpuRegs();
ScanlineEffect_Clear();
ResetPaletteFade();
gPaletteFade.bufferTransferDisabled = TRUE;
ResetSpriteData();
ResetTasks();
FreeAllSpritePalettes();
gReservedSpritePaletteCount = 4;
eContestDebugMode = CONTEST_DEBUG_MODE_OFF;
ClearBattleMonForms();
InitContestResources();
gMain.state++;
break;
case 1:
gMain.state++;
break;
case 2:
if (SetupContestGraphics(&eContest.contestSetupState))
{
eContest.contestSetupState = 0;
gMain.state++;
}
break;
case 3:
SetBgForCurtainDrop();
gBattle_BG1_X = 0;
gBattle_BG1_Y = 0;
BeginFastPaletteFade(2);
gPaletteFade.bufferTransferDisabled = FALSE;
SetVBlankCallback(VBlankCB_Contest);
eContest.mainTaskId = CreateTask(Task_StartContestWaitFade, 10);
SetMainCallback2(CB2_ContestMain);
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_WIRELESS)
{
LoadWirelessStatusIndicatorSpriteGfx();
CreateWirelessStatusIndicatorSprite(8, 8);
}
break;
}
}
static void Task_StartContestWaitFade(u8 taskId)
{
if (!gPaletteFade.active)
{
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_TryStartLinkContest;
}
}
// If this is a link contest try to start appeals communication
// Otherwise skip ahead
static void Task_TryStartLinkContest(u8 taskId)
{
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_WIRELESS)
{
switch (gTasks[taskId].data[0])
{
case 0:
ContestPrintLinkStandby();
gTasks[taskId].data[0]++;
// fallthrough
case 1:
if (IsLinkTaskFinished())
{
SetLinkStandbyCallback();
gTasks[taskId].data[0]++;
}
return;
case 2:
if (IsLinkTaskFinished() != TRUE)
return;
gTasks[taskId].data[0]++;
break;
}
}
if (!gPaletteFade.active)
{
gPaletteFade.bufferTransferDisabled = FALSE;
if (!(gLinkContestFlags & LINK_CONTEST_FLAG_IS_WIRELESS))
ContestPrintLinkStandby();
CreateTask(Task_CommunicateMonIdxs, 0);
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = TaskDummy1;
}
}
else
{
gTasks[taskId].func = Task_WaitToRaiseCurtainAtStart;
}
}
static void Task_CommunicateMonIdxs(u8 taskId)
{
SetTaskFuncWithFollowupFunc(taskId, Task_LinkContest_CommunicateMonIdxs, Task_EndCommunicateMonIdxs);
}
static void Task_EndCommunicateMonIdxs(u8 taskId)
{
gTasks[taskId].data[0] = 1;
gTasks[taskId].func = Task_ReadyStartLinkContest;
}
static void Task_ReadyStartLinkContest(u8 taskId)
{
// data[0] always 1 here
gTasks[taskId].data[0]--;
if (gTasks[taskId].data[0] <= 0)
{
GetMultiplayerId(); // unused return value
DestroyTask(taskId);
gTasks[eContest.mainTaskId].func = Task_WaitToRaiseCurtainAtStart;
gRngValue = gContestRngValue;
}
}
static bool8 SetupContestGraphics(u8 *stateVar)
{
u16 tempPalette1[16];
u16 tempPalette2[16];
switch (*stateVar)
{
case 0:
gPaletteFade.bufferTransferDisabled = TRUE;
RequestDma3Fill(0, (void *)VRAM, 0x8000, 1);
RequestDma3Fill(0, (void *)VRAM + 0x8000, 0x8000, 1);
RequestDma3Fill(0, (void *)VRAM + 0x10000, 0x8000, 1);
break;
case 1:
LZDecompressVram(gContestMiscGfx, (void *)VRAM);
break;
case 2:
LZDecompressVram(gContestAudienceGfx, (void *)(BG_SCREEN_ADDR(4)));
DmaCopyLarge32(3, (void *)(BG_SCREEN_ADDR(4)), eUnzippedContestAudience_Gfx, 0x2000, 0x1000);
break;
case 3:
CopyToBgTilemapBuffer(3, gOldContestGfx, 0, 0);
CopyBgTilemapBufferToVram(3);
break;
case 4:
CopyToBgTilemapBuffer(2, gUnknown_08C17170, 0, 0);
CopyBgTilemapBufferToVram(2);
// This is a bug, and copies random junk. savedJunk is never read.
DmaCopy32Defvars(3, gContestResources->contestBgTilemaps[2], eContestTempSave.savedJunk, sizeof(eContestTempSave.savedJunk));
break;
case 5:
LoadCompressedPalette(gOldContestPalette, 0, 0x200);
CpuCopy32(gPlttBufferUnfaded + 128, tempPalette1, 16 * sizeof(u16));
CpuCopy32(gPlttBufferUnfaded + (5 + gContestPlayerMonIndex) * 16, tempPalette2, 16 * sizeof(u16));
CpuCopy32(tempPalette2, gPlttBufferUnfaded + 128, 16 * sizeof(u16));
CpuCopy32(tempPalette1, gPlttBufferUnfaded + (5 + gContestPlayerMonIndex) * 16, 16 * sizeof(u16));
DmaCopy32Defvars(3, gPlttBufferUnfaded, eContestTempSave.cachedWindowPalettes, sizeof(eContestTempSave.cachedWindowPalettes));
LoadContestPalettes();
break;
case 6:
DrawContestantWindows();
FillContestantWindowBgs();
SwapMoveDescAndContestTilemaps();
eContest.judgeSpeechBubbleSpriteId = CreateJudgeSpeechBubbleSprite();
CreateSliderHeartSprites();
CreateNextTurnSprites();
CreateApplauseMeterSprite();
CreateJudgeAttentionEyeTask();
CreateUnusedBlendTask();
gBattlerPositions[0] = B_POSITION_PLAYER_LEFT;
gBattlerPositions[1] = B_POSITION_OPPONENT_LEFT;
gBattlerPositions[2] = B_POSITION_OPPONENT_RIGHT;
gBattlerPositions[3] = B_POSITION_PLAYER_RIGHT;
gBattleTypeFlags = 0;
gBattlerAttacker = B_POSITION_PLAYER_RIGHT;
gBattlerTarget = B_POSITION_OPPONENT_RIGHT;
// Unclear why judge sprite is assigned here
// Overwritten in APPEALSTATE_SLIDE_MON_IN with the attacking contest mon
gBattlerSpriteIds[gBattlerAttacker] = CreateJudgeSprite();
CreateInvisibleBattleTargetSprite();
CopyBgTilemapBufferToVram(3);
CopyBgTilemapBufferToVram(2);
CopyBgTilemapBufferToVram(1);
ShowBg(3);
ShowBg(2);
ShowBg(0);
ShowBg(1);
break;
default:
*stateVar = 0;
return TRUE;
}
(*stateVar)++;
return FALSE;
}
static void Task_WaitToRaiseCurtainAtStart(u8 taskId)
{
gPaletteFade.bufferTransferDisabled = FALSE;
if (!gPaletteFade.active)
{
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].func = Task_RaiseCurtainAtStart;
}
}
static void Task_RaiseCurtainAtStart(u8 taskId)
{
switch (gTasks[taskId].data[0])
{
case 0:
if (gTasks[taskId].data[1]++ <= 60)
break;
gTasks[taskId].data[1] = 0;
PlaySE12WithPanning(SE_CONTEST_CURTAIN_RISE, 0);
gTasks[taskId].data[0]++;
break;
case 1:
*(s16*)&gBattle_BG1_Y += 7;
if ((s16)gBattle_BG1_Y <= DISPLAY_HEIGHT)
break;
gTasks[taskId].data[0]++;
break;
case 2:
UpdateContestantBoxOrder();
gTasks[taskId].data[0]++;
break;
case 3:
{
u16 bg0Cnt = GetGpuReg(REG_OFFSET_BG0CNT);
u16 bg2Cnt = GetGpuReg(REG_OFFSET_BG2CNT);
((struct BgCnt *)&bg0Cnt)->priority = 0;
((struct BgCnt *)&bg2Cnt)->priority = 0;
SetGpuReg(REG_OFFSET_BG0CNT, bg0Cnt);
SetGpuReg(REG_OFFSET_BG2CNT, bg2Cnt);
SlideApplauseMeterIn();
gTasks[taskId].data[0]++;
break;
}
case 4:
default:
if (eContest.applauseMeterIsMoving)
break;
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].func = Task_DisplayAppealNumberText;
break;
}
}
static void CB2_ContestMain(void)
{
s32 i;
AnimateSprites();
RunTasks();
BuildOamBuffer();
UpdatePaletteFade();
for (i = 0; i < 4; i++)
{
if ((sContestBgCopyFlags >> i) & 1)
CopyBgTilemapBufferToVram(i);
}
sContestBgCopyFlags = 0;
}
static void VBlankCB_Contest(void)
{
SetGpuReg(REG_OFFSET_BG0HOFS, gBattle_BG0_X);
SetGpuReg(REG_OFFSET_BG0VOFS, gBattle_BG0_Y);
SetGpuReg(REG_OFFSET_BG1HOFS, gBattle_BG1_X);
SetGpuReg(REG_OFFSET_BG1VOFS, gBattle_BG1_Y);
SetGpuReg(REG_OFFSET_BG2HOFS, gBattle_BG2_X);
SetGpuReg(REG_OFFSET_BG2VOFS, gBattle_BG2_Y);
SetGpuReg(REG_OFFSET_BG3HOFS, gBattle_BG3_X);
SetGpuReg(REG_OFFSET_BG3VOFS, gBattle_BG3_Y);
SetGpuReg(REG_OFFSET_WIN0H, gBattle_WIN0H);
SetGpuReg(REG_OFFSET_WIN0V, gBattle_WIN0V);
SetGpuReg(REG_OFFSET_WIN1H, gBattle_WIN1H);
SetGpuReg(REG_OFFSET_WIN1V, gBattle_WIN1V);
TransferPlttBuffer();
LoadOam();
ProcessSpriteCopyRequests();
ScanlineEffect_InitHBlankDmaTransfer();
}
static void Task_DisplayAppealNumberText(u8 taskId)
{
if (gTasks[taskId].data[0] == 0)
{
gBattle_BG0_Y = 0;
gBattle_BG2_Y = 0;
ContestDebugDoPrint();
DmaCopy32Defvars(3, gPlttBufferUnfaded, eContestTempSave.cachedPlttBufferUnfaded, PLTT_BUFFER_SIZE * 2);
ConvertIntToDecimalStringN(gStringVar1, eContest.appealNumber + 1, STR_CONV_MODE_LEFT_ALIGN, 1);
if (!Contest_IsMonsTurnDisabled(gContestPlayerMonIndex))
StringCopy(gDisplayedStringBattle, gText_AppealNumWhichMoveWillBePlayed);
else
StringCopy(gDisplayedStringBattle, gText_AppealNumButItCantParticipate);
ContestClearGeneralTextWindow();
StringExpandPlaceholders(gStringVar4, gDisplayedStringBattle);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].data[0]++;
}
else
{
if (!Contest_RunTextPrinters())
{
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_TryShowMoveSelectScreen;
}
}
}
static void Task_TryShowMoveSelectScreen(u8 taskId)
{
// Wait for button press to show move select screen
if ((JOY_NEW(A_BUTTON)) || (gMain.newKeys == B_BUTTON))
{
PlaySE(SE_SELECT);
if (!Contest_IsMonsTurnDisabled(gContestPlayerMonIndex))
{
SetBottomSliderHeartsInvisibility(TRUE);
gTasks[taskId].func = Task_ShowMoveSelectScreen;
}
else
{
// Skip move selection (selected move will be MOVE_NONE)
gTasks[taskId].func = Task_SelectedMove;
}
}
}
static void Task_ShowMoveSelectScreen(u8 taskId)
{
u8 i;
u8 moveName[32];
gBattle_BG0_Y = DISPLAY_HEIGHT;
gBattle_BG2_Y = DISPLAY_HEIGHT;
for (i = 0; i < MAX_MON_MOVES; i++)
{
u16 move = gContestMons[gContestPlayerMonIndex].moves[i];
u8 *moveNameBuffer = moveName;
if (eContestantStatus[gContestPlayerMonIndex].prevMove != MOVE_NONE
&& IsContestantAllowedToCombo(gContestPlayerMonIndex)
&& AreMovesContestCombo(eContestantStatus[gContestPlayerMonIndex].prevMove, move)
&& eContestantStatus[gContestPlayerMonIndex].hasJudgesAttention)
{
// Highlight the text because it's a combo move
moveNameBuffer = StringCopy(moveName, gText_ColorLightShadowDarkGray);
}
else if (move != MOVE_NONE
&& eContestantStatus[gContestPlayerMonIndex].prevMove == move
&& gContestMoves[move].effect != CONTEST_EFFECT_REPETITION_NOT_BORING)
{
// Gray the text because it's a repeated move
moveNameBuffer = StringCopy(moveName, gText_ColorBlue);
}
moveNameBuffer = StringCopy(moveNameBuffer, gMoveNames[move]);
FillWindowPixelBuffer(i + MOVE_WINDOWS_START, PIXEL_FILL(0));
Contest_PrintTextToBg0WindowAt(i + MOVE_WINDOWS_START, moveName, 5, 1, 7);
}
DrawMoveSelectArrow(eContest.playerMoveChoice);
PrintContestMoveDescription(gContestMons[gContestPlayerMonIndex].moves[eContest.playerMoveChoice]);
gTasks[taskId].func = Task_HandleMoveSelectInput;
}
static void Task_HandleMoveSelectInput(u8 taskId)
{
u8 numMoves = 0;
s32 i;
for (i = 0; i < MAX_MON_MOVES; i++)
{
if (gContestMons[gContestPlayerMonIndex].moves[i] != MOVE_NONE)
numMoves++;
}
if (JOY_NEW(A_BUTTON))
{
PlaySE(SE_SELECT);
gTasks[taskId].func = Task_SelectedMove;
}
else
{
switch (gMain.newAndRepeatedKeys)
{
case B_BUTTON:
// Cancel move selection
PlaySE(SE_SELECT);
SetBottomSliderHeartsInvisibility(FALSE);
ConvertIntToDecimalStringN(gStringVar1, eContest.appealNumber + 1, STR_CONV_MODE_LEFT_ALIGN, 1);
if (!Contest_IsMonsTurnDisabled(gContestPlayerMonIndex))
StringCopy(gDisplayedStringBattle, gText_AppealNumWhichMoveWillBePlayed);
else
StringCopy(gDisplayedStringBattle, gText_AppealNumButItCantParticipate);
ContestClearGeneralTextWindow();
StringExpandPlaceholders(gStringVar4, gDisplayedStringBattle);
Contest_StartTextPrinter(gStringVar4, 0);
gBattle_BG0_Y = 0;
gBattle_BG2_Y = 0;
gTasks[taskId].func = Task_TryShowMoveSelectScreen;
break;
case DPAD_LEFT:
case DPAD_RIGHT:
break;
case DPAD_UP:
EraseMoveSelectArrow(eContest.playerMoveChoice);
if (eContest.playerMoveChoice == 0)
eContest.playerMoveChoice = numMoves - 1;
else
eContest.playerMoveChoice--;
DrawMoveSelectArrow(eContest.playerMoveChoice);
PrintContestMoveDescription(gContestMons[gContestPlayerMonIndex].moves[eContest.playerMoveChoice]);
if (numMoves > 1)
PlaySE(SE_SELECT);
break;
case DPAD_DOWN:
EraseMoveSelectArrow(eContest.playerMoveChoice);
if (eContest.playerMoveChoice == numMoves - 1)
eContest.playerMoveChoice = 0;
else
eContest.playerMoveChoice++;
DrawMoveSelectArrow(eContest.playerMoveChoice);
PrintContestMoveDescription(gContestMons[gContestPlayerMonIndex].moves[eContest.playerMoveChoice]);
if (numMoves > 1)
PlaySE(SE_SELECT);
break;
}
}
}
static void DrawMoveSelectArrow(s8 moveIndex)
{
ContestBG_FillBoxWithIncrementingTile(2, 55, 0, 31 + moveIndex * 2, 2, 2, 17, 1);
}
static void EraseMoveSelectArrow(s8 moveIndex)
{
ContestBG_FillBoxWithIncrementingTile(2, 11, 0, 31 + moveIndex * 2, 2, 1, 17, 1);
ContestBG_FillBoxWithIncrementingTile(2, 11, 0, 32 + moveIndex * 2, 2, 1, 17, 1);
}
static void Task_SelectedMove(u8 taskId)
{
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
u16 move = GetChosenMove(gContestPlayerMonIndex);
u8 taskId2;
eContestantStatus[gContestPlayerMonIndex].currMove = move;
taskId2 = CreateTask(Task_LinkContest_CommunicateMoveSelections, 0);
SetTaskFuncWithFollowupFunc(taskId2, Task_LinkContest_CommunicateMoveSelections, Task_EndCommunicateMoveSelections);
gTasks[taskId].func = TaskDummy1;
ContestPrintLinkStandby();
SetBottomSliderHeartsInvisibility(FALSE);
}
else
{
GetAllChosenMoves();
gTasks[taskId].func = Task_HideMoveSelectScreen;
}
}
static void Task_EndCommunicateMoveSelections(u8 taskId)
{
DestroyTask(taskId);
gTasks[eContest.mainTaskId].func = Task_HideMoveSelectScreen;
}
static void Task_HideMoveSelectScreen(u8 taskId)
{
s32 i;
ContestClearGeneralTextWindow();
gBattle_BG0_Y = 0;
gBattle_BG2_Y = 0;
SetBottomSliderHeartsInvisibility(FALSE);
for (i = 0; i < MAX_MON_MOVES; i++)
{
FillWindowPixelBuffer(MOVE_WINDOWS_START + i, PIXEL_FILL(0));
PutWindowTilemap(MOVE_WINDOWS_START + i);
CopyWindowToVram(MOVE_WINDOWS_START + i, 2);
}
Contest_SetBgCopyFlags(0);
// This seems to be a bug; it should have just copied PLTT_BUFFER_SIZE.
DmaCopy32Defvars(3, gPlttBufferFaded, eContestTempSave.cachedPlttBufferFaded, PLTT_BUFFER_SIZE * 2);
LoadPalette(eContestTempSave.cachedPlttBufferUnfaded, 0, PLTT_BUFFER_SIZE * 2);
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].func = Task_HideApplauseMeterForAppealStart;
}
static void Task_HideApplauseMeterForAppealStart(u8 taskId)
{
if (++gTasks[taskId].data[0] > 2)
{
gTasks[taskId].data[0] = 0;
if (++gTasks[taskId].data[1] == 2)
{
SlideApplauseMeterOut();
AnimateSliderHearts(SLIDER_HEART_ANIM_DISAPPEAR);
gTasks[taskId].func = Task_WaitHideApplauseMeterForAppealStart;
}
}
}
static void Task_WaitHideApplauseMeterForAppealStart(u8 taskId)
{
if (!eContest.applauseMeterIsMoving && !eContest.sliderHeartsAnimating)
gTasks[taskId].func = Task_AppealSetup;
}
#define tState data[0]
#define tMonSpriteId data[2]
#define tCounter data[10]
static void Task_AppealSetup(u8 taskId)
{
if (++gTasks[taskId].data[0] > 19)
{
eContest.turnNumber = 0;
eContest.unusedRng = gRngValue;
if ((gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK) && IsPlayerLinkLeader())
{
s32 i;
for (i = 0; i + gNumLinkContestPlayers < CONTESTANT_COUNT; i++)
{
eContestantStatus[gNumLinkContestPlayers + i].currMove = GetChosenMove(gNumLinkContestPlayers + i);
}
}
gTasks[taskId].tState = APPEALSTATE_START_TURN;
gTasks[taskId].func = Task_DoAppeals;
}
}
static void Task_DoAppeals(u8 taskId)
{
u8 spriteId;
s32 i;
u8 contestant = eContest.currentContestant;
s8 r3;
switch (gTasks[taskId].tState)
{
case APPEALSTATE_START_TURN:
ContestDebugDoPrint();
for (i = 0; eContest.turnNumber != eContestAppealResults.turnOrder[i]; i++)
;
eContest.currentContestant = i;
contestant = eContest.currentContestant;
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
u8 taskId2;
eContest.waitForLink = TRUE;
if (IsPlayerLinkLeader())
CalculateAppealMoveImpact(eContest.currentContestant);
taskId2 = CreateTask(Task_LinkContest_CommunicateAppealsState, 0);
SetTaskFuncWithFollowupFunc(taskId2, Task_LinkContest_CommunicateAppealsState, Task_EndWaitForLink);
ContestPrintLinkStandby();
gTasks[taskId].tState = APPEALSTATE_WAIT_LINK;
}
else
{
CalculateAppealMoveImpact(eContest.currentContestant);
gTasks[taskId].tState = APPEALSTATE_CHECK_SKIP_TURN;
}
return;
case APPEALSTATE_WAIT_LINK:
if (!eContest.waitForLink)
gTasks[taskId].tState = APPEALSTATE_CHECK_SKIP_TURN;
return;
case APPEALSTATE_CHECK_SKIP_TURN:
SetContestLiveUpdateFlags(contestant);
ContestDebugPrintBitStrings();
if (eContestantStatus[contestant].numTurnsSkipped != 0
|| eContestantStatus[contestant].noMoreTurns)
{
gTasks[taskId].tState = APPEALSTATE_PRINT_SKIP_TURN_MSG;
}
else
{
ContestClearGeneralTextWindow();
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_SLIDE_MON_IN;
}
return;
case APPEALSTATE_SLIDE_MON_IN:
for (i = 0; i < CONTESTANT_COUNT; i++)
gBattleMonForms[i] = 0;
memset(gContestResources->moveAnim, 0, sizeof(*gContestResources->moveAnim));
SetMoveAnimAttackerData(eContest.currentContestant);
spriteId = CreateContestantSprite(
gContestMons[eContest.currentContestant].species,
gContestMons[eContest.currentContestant].otId,
gContestMons[eContest.currentContestant].personality,
eContest.currentContestant);
gSprites[spriteId].x2 = 120;
gSprites[spriteId].callback = SpriteCB_MonSlideIn;
gTasks[taskId].tMonSpriteId = spriteId;
gBattlerSpriteIds[gBattlerAttacker] = spriteId;
BlinkContestantBox(CreateContestantBoxBlinkSprites(eContest.currentContestant), FALSE);
gTasks[taskId].tState = APPEALSTATE_WAIT_SLIDE_MON;
return;
case APPEALSTATE_WAIT_SLIDE_MON:
spriteId = gTasks[taskId].tMonSpriteId;
if (gSprites[spriteId].callback == SpriteCallbackDummy)
{
// Once mon has slid in, also wait for box to finish blinking
if (!eContestGfxState[contestant].boxBlinking)
gTasks[taskId].tState = APPEALSTATE_PRINT_USED_MOVE_MSG;
}
return;
case APPEALSTATE_PRINT_USED_MOVE_MSG:
if (eContestantStatus[contestant].nervous)
{
gTasks[taskId].tState = APPEALSTATE_PRINT_TOO_NERVOUS_MSG;
}
else
{
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[contestant].nickname);
if (eContestantStatus[contestant].currMove < MOVES_COUNT)
StringCopy(gStringVar2, gMoveNames[eContestantStatus[contestant].currMove]);
else
StringCopy(gStringVar2, sInvalidContestMoveNames[eContestantStatus[contestant].moveCategory]);
StringExpandPlaceholders(gStringVar4, gText_MonAppealedWithMove);
Contest_StartTextPrinter(gStringVar4, 1);
gTasks[taskId].tState = APPEALSTATE_WAIT_USED_MOVE_MSG;
}
return;
case APPEALSTATE_WAIT_USED_MOVE_MSG:
if (!Contest_RunTextPrinters())
{
eContest.moveAnimTurnCount = 0;
gTasks[taskId].tState = APPEALSTATE_MOVE_ANIM;
}
return;
case APPEALSTATE_MOVE_ANIM:
{
u16 move = SanitizeMove(eContestantStatus[eContest.currentContestant].currMove);
SetMoveSpecificAnimData(eContest.currentContestant);
SetMoveAnimAttackerData(eContest.currentContestant);
SetMoveTargetPosition(move);
DoMoveAnim(move);
gTasks[taskId].tState = APPEALSTATE_WAIT_MOVE_ANIM;
}
return;
case APPEALSTATE_WAIT_MOVE_ANIM:
gAnimScriptCallback();
if (!gAnimScriptActive)
{
ClearMoveAnimData(contestant);
if (eContest.moveAnimTurnCount != 0)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_MOVE_ANIM_MULTITURN;
}
else
{
if (!eContestantStatus[contestant].hasJudgesAttention)
StopFlashJudgeAttentionEye(contestant);
DrawUnnervedSymbols();
gTasks[taskId].tState = APPEALSTATE_TRY_PRINT_MOVE_RESULT;
}
}
return;
case APPEALSTATE_MOVE_ANIM_MULTITURN:
if (gTasks[taskId].tCounter++ > 30)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_MOVE_ANIM;
}
return;
case APPEALSTATE_TRY_PRINT_MOVE_RESULT:
gTasks[taskId].data[1] = 0;
if (eContestantStatus[contestant].effectStringId != CONTEST_STRING_NONE)
{
PrintAppealMoveResultText(contestant, eContestantStatus[contestant].effectStringId);
eContestantStatus[contestant].effectStringId = CONTEST_STRING_NONE;
gTasks[taskId].tState = APPEALSTATE_WAIT_MOVE_RESULT_MSG;
}
else
{
if (eContestantStatus[contestant].effectStringId2 != CONTEST_STRING_NONE)
{
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (i != contestant && eContestantStatus[i].effectStringId != CONTEST_STRING_NONE)
break;
}
if (i == CONTESTANT_COUNT)
{
PrintAppealMoveResultText(contestant, eContestantStatus[contestant].effectStringId2);
eContestantStatus[contestant].effectStringId2 = CONTEST_STRING_NONE;
gTasks[taskId].tState = APPEALSTATE_WAIT_MOVE_RESULT_MSG;
}
else
{
gTasks[taskId].tState = APPEALSTATE_CHECK_TURN_ORDER_MOD;
}
}
else
{
gTasks[taskId].tState = APPEALSTATE_CHECK_TURN_ORDER_MOD;
}
}
return;
case APPEALSTATE_WAIT_MOVE_RESULT_MSG:
if (!Contest_RunTextPrinters())
gTasks[taskId].tState = APPEALSTATE_TRY_PRINT_MOVE_RESULT;
return;
case APPEALSTATE_CHECK_TURN_ORDER_MOD:
if (eContestantStatus[contestant].turnOrderModAction == 1)
{
DoJudgeSpeechBubble(JUDGE_SYMBOL_NUMBER_ONE);
}
else if (eContestantStatus[contestant].turnOrderModAction == 2)
{
DoJudgeSpeechBubble(JUDGE_SYMBOL_NUMBER_FOUR);
}
else if (eContestantStatus[contestant].turnOrderModAction == 3)
{
DoJudgeSpeechBubble(JUDGE_SYMBOL_QUESTION_MARK);
}
else
{
gTasks[taskId].tState = APPEALSTATE_TRY_SHOW_NEXT_TURN_GFX;
return;
}
gTasks[taskId].tState = APPEALSTATE_WAIT_JUDGE_TURN_ORDER;
return;
case APPEALSTATE_WAIT_JUDGE_TURN_ORDER:
if (!eContest.waitForJudgeSpeechBubble)
gTasks[taskId].tState = APPEALSTATE_TRY_SHOW_NEXT_TURN_GFX;
return;
case APPEALSTATE_TRY_SHOW_NEXT_TURN_GFX:
ShowHideNextTurnGfx(TRUE);
gTasks[taskId].tState = APPEALSTATE_UPDATE_MOVE_USERS_HEARTS;
return;
case APPEALSTATE_UPDATE_MOVE_USERS_HEARTS:
UpdateAppealHearts(0, eContestantStatus[contestant].appeal, contestant);
gTasks[taskId].tState = APPEALSTATE_WAIT_MOVE_USERS_HEARTS;
return;
case APPEALSTATE_WAIT_MOVE_USERS_HEARTS:
if (!eContestGfxState[eContest.currentContestant].updatingAppealHearts)
gTasks[taskId].tState = APPEALSTATE_TRY_JUDGE_STAR;
return;
case APPEALSTATE_TRY_JUDGE_STAR:
if (eContestantStatus[contestant].conditionMod == CONDITION_GAIN)
DoJudgeSpeechBubble(JUDGE_SYMBOL_STAR);
gTasks[taskId].tState = APPEALSTATE_WAIT_JUDGE_STAR;
return;
case APPEALSTATE_WAIT_JUDGE_STAR:
if (!eContest.waitForJudgeSpeechBubble)
gTasks[taskId].tState = APPEALSTATE_UPDATE_MOVE_USERS_STARS;
return;
case APPEALSTATE_UPDATE_MOVE_USERS_STARS:
if (UpdateConditionStars(contestant, TRUE))
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_WAIT_MOVE_USERS_STARS;
}
else
{
gTasks[taskId].tState = APPEALSTATE_UPDATE_MOVE_USERS_STATUS;
}
return;
case APPEALSTATE_WAIT_MOVE_USERS_STARS:
if (++gTasks[taskId].tCounter > 20)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_UPDATE_MOVE_USERS_STATUS;
}
return;
case APPEALSTATE_UPDATE_MOVE_USERS_STATUS:
if (DrawStatusSymbol(contestant))
PlaySE(SE_CONTEST_ICON_CHANGE);
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENTS;
return;
case APPEALSTATE_UPDATE_OPPONENTS:
gTasks[taskId].data[1] = 0;
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENT;
return;
case APPEALSTATE_UPDATE_OPPONENT:
{
// Update each other contestant as a result of this move appeal
s32 j = 0;
r3 = FALSE; // Can't get this to use local variable. Should be "needsUpdate"
for (i = gTasks[taskId].data[1]; i < CONTESTANT_COUNT; i++)
{
r3 = FALSE;
for (j = 0; j < CONTESTANT_COUNT; j++)
{
if (j != contestant && gContestantTurnOrder[j] == i
&& eContestantStatus[j].effectStringId != CONTEST_STRING_NONE)
{
r3 = TRUE;
break;
}
}
if (r3)
break;
}
if (r3)
{
// Update contestant
gTasks[taskId].data[1] = gContestantTurnOrder[j];
PrintAppealMoveResultText(j, eContestantStatus[j].effectStringId);
eContestantStatus[j].effectStringId = CONTEST_STRING_NONE;
gTasks[taskId].tState = APPEALSTATE_WAIT_OPPONENT_RESPONSE_MSG;
}
else
{
// Done updating contestants
gTasks[taskId].data[1] = 0;
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_TRY_PRINT_SKIP_NEXT_TURN_MSG;
DrawStatusSymbols();
}
}
return;
case APPEALSTATE_WAIT_OPPONENT_RESPONSE_MSG:
// Wait for contestants response to current appeal
// i.e. "Contestant managed to avert its gaze"
if (!Contest_RunTextPrinters())
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENT_HEARTS;
return;
case APPEALSTATE_UPDATE_OPPONENT_HEARTS:
for (i = 0; gTasks[taskId].data[1] != gContestantTurnOrder[i]; i++)
;
UpdateAppealHearts(eContestantStatus[i].appeal + eContestantStatus[i].jam, -eContestantStatus[i].jam, i);
gTasks[taskId].tState = APPEALSTATE_WAIT_OPPONENT_HEARTS;
return;
case APPEALSTATE_WAIT_OPPONENT_HEARTS:
for (i = 0; gTasks[taskId].data[1] != gContestantTurnOrder[i]; i++)
;
if (!eContestGfxState[i].updatingAppealHearts)
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENT_STARS;
return;
case APPEALSTATE_UPDATE_OPPONENT_STARS:
for (i = 0; gTasks[taskId].data[1] != gContestantTurnOrder[i]; i++)
;
if (UpdateConditionStars(i, TRUE))
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_WAIT_OPPONENT_STARS;
}
else
{
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENT_STATUS;
}
return;
case APPEALSTATE_WAIT_OPPONENT_STARS:
if (++gTasks[taskId].tCounter > 20)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENT_STATUS;
}
return;
case APPEALSTATE_UPDATE_OPPONENT_STATUS:
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (gContestantTurnOrder[i] == gTasks[taskId].data[1])
break;
}
if (DrawStatusSymbol(i))
PlaySE(SE_CONTEST_ICON_CHANGE);
else
PlaySE(SE_CONTEST_ICON_CLEAR);
if (eContestantStatus[i].judgesAttentionWasRemoved)
{
StopFlashJudgeAttentionEye(i);
eContestantStatus[i].judgesAttentionWasRemoved = FALSE;
}
gTasks[taskId].data[1]++;
gTasks[taskId].tState = APPEALSTATE_UPDATE_OPPONENT;
return;
case APPEALSTATE_TRY_PRINT_SKIP_NEXT_TURN_MSG:
if (gTasks[taskId].tCounter++ > 9)
{
gTasks[taskId].tCounter = 0;
if (eContestantStatus[contestant].numTurnsSkipped != 0
|| eContestantStatus[contestant].turnSkipped)
{
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringExpandPlaceholders(gStringVar4, gText_MonCantAppealNextTurn);
Contest_StartTextPrinter(gStringVar4, 1);
}
gTasks[taskId].tState = APPEALSTATE_WAIT_SKIP_NEXT_TURN_MSG;
}
return;
case APPEALSTATE_WAIT_SKIP_NEXT_TURN_MSG:
if (!Contest_RunTextPrinters())
{
if (!eContestantStatus[contestant].usedComboMove)
gTasks[taskId].tState = APPEALSTATE_CHECK_REPEATED_MOVE;
else
gTasks[taskId].tState = APPEALSTATE_PRINT_COMBO_MSG;
}
return;
case APPEALSTATE_PRINT_COMBO_MSG:
{
s8 completedCombo = eContestantStatus[contestant].completedCombo;
if (eContestantStatus[contestant].completedCombo)
{
// Finished combo
// Looks like there were originally meant to be move combos of
// different effectivenesses. completedCombo however is only ever 0 or 1,
// so in here only "Went over well" will ever be used
ContestClearGeneralTextWindow();
if (completedCombo == 1)
Contest_StartTextPrinter(gText_AppealComboWentOverWell, TRUE);
else if (completedCombo == 2)
Contest_StartTextPrinter(gText_AppealComboWentOverVeryWell, TRUE);
else
Contest_StartTextPrinter(gText_AppealComboWentOverExcellently, TRUE);
DoJudgeSpeechBubble(JUDGE_SYMBOL_TWO_EXCLAMATIONS);
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_WAIT_JUDGE_COMBO;
}
else
{
// Started combo
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringExpandPlaceholders(gStringVar4, gText_JudgeLookedAtMonExpectantly);
Contest_StartTextPrinter(gStringVar4, 1);
DoJudgeSpeechBubble(JUDGE_SYMBOL_ONE_EXCLAMATION);
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_WAIT_JUDGE_COMBO;
}
return;
}
case APPEALSTATE_WAIT_JUDGE_COMBO:
if (!eContest.waitForJudgeSpeechBubble)
{
StartStopFlashJudgeAttentionEye(eContest.currentContestant);
gTasks[taskId].tState = APPEALSTATE_TRY_UPDATE_HEARTS_FROM_COMBO;
}
return;
case APPEALSTATE_TRY_UPDATE_HEARTS_FROM_COMBO:
if (!Contest_RunTextPrinters())
{
if (++gTasks[taskId].tCounter > 50)
{
if (!eContestantStatus[contestant].hasJudgesAttention)
{
UpdateAppealHearts(
eContestantStatus[contestant].appeal,
eContestantStatus[contestant].comboAppealBonus,
contestant);
eContestantStatus[contestant].appeal += eContestantStatus[contestant].comboAppealBonus;
}
gTasks[taskId].tState = APPEALSTATE_WAIT_HEARTS_FROM_COMBO;
}
}
return;
case APPEALSTATE_WAIT_HEARTS_FROM_COMBO:
if (!eContestGfxState[contestant].updatingAppealHearts)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_CHECK_REPEATED_MOVE;
}
return;
case APPEALSTATE_CHECK_REPEATED_MOVE:
if (eContestantStatus[contestant].repeatedMove)
{
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringExpandPlaceholders(gStringVar4, gText_RepeatedAppeal);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].tCounter = 0;
DoJudgeSpeechBubble(JUDGE_SYMBOL_SWIRL);
gTasks[taskId].tState = APPEALSTATE_WAIT_JUDGE_REPEATED_MOVE;
}
else
{
gTasks[taskId].tState = APPEALSTATE_UPDATE_CROWD;
}
return;
case APPEALSTATE_WAIT_JUDGE_REPEATED_MOVE:
if (!eContest.waitForJudgeSpeechBubble)
gTasks[taskId].tState = APPEALSTATE_UPDATE_HEARTS_FROM_REPEAT;
return;
case APPEALSTATE_UPDATE_HEARTS_FROM_REPEAT:
if (!Contest_RunTextPrinters())
{
UpdateAppealHearts(eContestantStatus[contestant].appeal, -eContestantStatus[contestant].repeatJam, contestant);
eContestantStatus[contestant].appeal -= eContestantStatus[contestant].repeatJam;
gTasks[taskId].tState = APPEALSTATE_WAIT_HEARTS_FROM_REPEAT;
}
return;
case APPEALSTATE_WAIT_HEARTS_FROM_REPEAT:
ContestDebugDoPrint();
if (!eContestGfxState[contestant].updatingAppealHearts)
{
gTasks[taskId].tCounter = 0;
ContestClearGeneralTextWindow();
gTasks[taskId].tState = APPEALSTATE_UPDATE_CROWD;
}
return;
case APPEALSTATE_UPDATE_CROWD:
if (eContestExcitement.frozen && contestant != eContestExcitement.freezer)
{
gTasks[taskId].tState = APPEALSTATE_PRINT_CROWD_WATCHES_MSG;
}
else
{
r3 = eContestExcitement.moveExcitement; // Can't get this to use local variable. Should be "moveExcitement"
if (eContestantStatus[contestant].overrideCategoryExcitementMod)
{
r3 = 1;
StringCopy(gStringVar3, gMoveNames[eContestantStatus[contestant].currMove]);
}
else
{
StringCopy(gStringVar3, sContestConditions[gContestMoves[eContestantStatus[contestant].currMove].contestCategory]);
}
if (r3 > 0 && eContestantStatus[contestant].repeatedMove)
r3 = 0;
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[contestant].nickname);
eContest.applauseLevel += r3;
if (eContest.applauseLevel < 0)
eContest.applauseLevel = 0;
if (r3 == 0)
{
gTasks[taskId].tState = APPEALSTATE_SLIDE_APPLAUSE_OUT;
}
else
{
if (r3 < 0)
StringExpandPlaceholders(gStringVar4, gText_MonsXDidntGoOverWell);
else if (r3 > 0 && eContest.applauseLevel <= 4)
StringExpandPlaceholders(gStringVar4, gText_MonsXWentOverGreat);
else
StringExpandPlaceholders(gStringVar4, gText_MonsXGotTheCrowdGoing);
Contest_StartTextPrinter(gStringVar4, 1);
gTasks[taskId].tCounter = 0;
gTasks[taskId].data[11] = 0;
if (r3 < 0)
gTasks[taskId].tState = APPEALSTATE_DO_CROWD_UNEXCITED;
else
gTasks[taskId].tState = APPEALSTATE_DO_CROWD_EXCITED;
}
}
return;
case APPEALSTATE_DO_CROWD_UNEXCITED:
switch (gTasks[taskId].tCounter)
{
case 0:
BlendAudienceBackground(-1, 1);
PlayFanfare(MUS_TOO_BAD);
gTasks[taskId].tCounter++;
break;
case 1:
if (!eContest.waitForAudienceBlend && !Contest_RunTextPrinters())
{
ShowAndUpdateApplauseMeter(-1);
gTasks[taskId].tCounter++;
}
break;
case 2:
if (!eContest.isShowingApplauseMeter)
{
if (gTasks[taskId].data[11]++ > 29)
{
gTasks[taskId].data[11] = 0;
BlendAudienceBackground(-1, -1);
gTasks[taskId].tCounter++;
}
}
break;
case 3:
if (!gPaletteFade.active)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].data[11] = 0;
gTasks[taskId].tState = APPEALSTATE_WAIT_EXCITEMENT_HEARTS;
}
break;
}
return;
case APPEALSTATE_DO_CROWD_EXCITED:
switch (gTasks[taskId].tCounter)
{
case 0:
if (!Contest_RunTextPrinters())
{
BlendAudienceBackground(1, 1);
gTasks[taskId].tCounter++;
}
break;
case 1:
if (!eContest.waitForAudienceBlend)
{
AnimateAudience();
PlaySE(SE_M_ENCORE2);
ShowAndUpdateApplauseMeter(1);
gTasks[taskId].tCounter++;
}
break;
case 2:
if (!eContest.isShowingApplauseMeter)
{
if (gTasks[taskId].data[11]++ > 29)
{
gTasks[taskId].data[11] = 0;
UpdateAppealHearts(eContestantStatus[contestant].appeal, eContestExcitement.excitementAppealBonus, contestant);
eContestantStatus[contestant].appeal += eContestExcitement.excitementAppealBonus;
gTasks[taskId].tCounter++;
}
}
break;
case 3:
if (!eContestGfxState[contestant].updatingAppealHearts)
{
if (!eContest.animatingAudience)
{
BlendAudienceBackground(1, -1);
gTasks[taskId].tCounter++;
}
}
break;
case 4:
if (!gPaletteFade.active)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].data[11] = 0;
gTasks[taskId].tState = APPEALSTATE_WAIT_EXCITEMENT_HEARTS;
}
break;
}
return;
case APPEALSTATE_WAIT_EXCITEMENT_HEARTS:
if (!eContestGfxState[contestant].updatingAppealHearts)
{
ContestClearGeneralTextWindow();
gTasks[taskId].tState = APPEALSTATE_SLIDE_APPLAUSE_OUT;
}
return;
case APPEALSTATE_PRINT_CROWD_WATCHES_MSG:
ContestClearGeneralTextWindow();
StringCopy(gStringVar3, gContestMons[eContestExcitement.freezer].nickname);
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringCopy(gStringVar2, gMoveNames[eContestantStatus[contestant].currMove]);
StringExpandPlaceholders(gStringVar4, gText_CrowdContinuesToWatchMon);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].tState = APPEALSTATE_PRINT_MON_MOVE_IGNORED_MSG;
return;
case APPEALSTATE_PRINT_MON_MOVE_IGNORED_MSG:
if (!Contest_RunTextPrinters())
{
ContestClearGeneralTextWindow();
StringExpandPlaceholders(gStringVar4, gText_MonsMoveIsIgnored);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].tState = APPEALSTATE_WAIT_MON_MOVE_IGNORED_MSG;
}
return;
case APPEALSTATE_WAIT_MON_MOVE_IGNORED_MSG:
if (!Contest_RunTextPrinters())
{
ContestClearGeneralTextWindow();
gTasks[taskId].tState = APPEALSTATE_SLIDE_APPLAUSE_OUT;
}
return;
case APPEALSTATE_PRINT_TOO_NERVOUS_MSG:
if (eContestantStatus[contestant].hasJudgesAttention)
eContestantStatus[contestant].hasJudgesAttention = FALSE;
StartStopFlashJudgeAttentionEye(contestant);
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringCopy(gStringVar2, gMoveNames[eContestantStatus[contestant].currMove]);
StringExpandPlaceholders(gStringVar4, gText_MonWasTooNervousToMove);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].tState = APPEALSTATE_WAIT_TOO_NERVOUS_MSG;
return;
case APPEALSTATE_WAIT_TOO_NERVOUS_MSG:
if (!Contest_RunTextPrinters())
gTasks[taskId].tState = APPEALSTATE_SLIDE_APPLAUSE_OUT;
return;
case APPEALSTATE_SLIDE_APPLAUSE_OUT:
SlideApplauseMeterOut();
gTasks[taskId].tState = APPEALSTATE_WAIT_SLIDE_APPLAUSE;
return;
case APPEALSTATE_WAIT_SLIDE_APPLAUSE:
if (!eContest.applauseMeterIsMoving)
{
if (eContest.applauseLevel > 4)
{
eContest.applauseLevel = 0;
UpdateApplauseMeter();
}
gTasks[taskId].tState = APPEALSTATE_SLIDE_MON_OUT;
}
return;
case APPEALSTATE_SLIDE_MON_OUT:
spriteId = gTasks[taskId].tMonSpriteId;
gSprites[spriteId].callback = SpriteCB_MonSlideOut;
gTasks[taskId].tState = APPEALSTATE_FREE_MON_SPRITE;
return;
case APPEALSTATE_FREE_MON_SPRITE:
spriteId = gTasks[taskId].tMonSpriteId;
if (gSprites[spriteId].invisible)
{
FreeSpriteOamMatrix(&gSprites[spriteId]);
DestroySprite(&gSprites[spriteId]);
gTasks[taskId].tState = APPEALSTATE_START_TURN_END_DELAY;
}
return;
case APPEALSTATE_START_TURN_END_DELAY:
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_TURN_END_DELAY;
return;
case APPEALSTATE_PRINT_SKIP_TURN_MSG:
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringExpandPlaceholders(gStringVar4, gText_MonWasWatchingOthers);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].tState = APPEALSTATE_WAIT_SKIP_TURN_MSG;
return;
case APPEALSTATE_WAIT_SKIP_TURN_MSG:
if (!Contest_RunTextPrinters())
gTasks[taskId].tState = APPEALSTATE_TURN_END_DELAY;
return;
case APPEALSTATE_TURN_END_DELAY:
if (++gTasks[taskId].tCounter > 29)
{
gTasks[taskId].tCounter = 0;
gTasks[taskId].tState = APPEALSTATE_START_NEXT_TURN;
}
return;
case APPEALSTATE_START_NEXT_TURN:
if (++eContest.turnNumber == CONTESTANT_COUNT)
{
gTasks[taskId].tState = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].tMonSpriteId = 0;
gTasks[taskId].func = Task_FinishRoundOfAppeals;
}
else
{
gTasks[taskId].tState = APPEALSTATE_START_TURN;
}
return;
}
}
static void Task_EndWaitForLink(u8 taskId)
{
eContest.waitForLink = FALSE;
DestroyTask(taskId);
}
static void SpriteCB_MonSlideIn(struct Sprite *sprite)
{
if (sprite->x2 != 0)
{
sprite->x2 -= 2;
}
else
{
if (++sprite->data[0] == 31)
{
sprite->data[0] = 0;
sprite->callback = SpriteCallbackDummy;
}
}
}
static void SpriteCB_MonSlideOut(struct Sprite *sprite)
{
sprite->x2 -= 6;
if (sprite->x + sprite->x2 < -32)
{
sprite->callback = SpriteCallbackDummy;
sprite->invisible = TRUE;
}
}
static void Task_FinishRoundOfAppeals(u8 taskId)
{
switch (gTasks[taskId].data[0])
{
case 0:
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
u8 taskId2;
eContest.waitForLink = TRUE;
if (IsPlayerLinkLeader())
{
RankContestants();
SetAttentionLevels();
}
taskId2 = CreateTask(Task_LinkContest_CommunicateAppealsState, 0);
SetTaskFuncWithFollowupFunc(taskId2, Task_LinkContest_CommunicateAppealsState, Task_EndWaitForLink);
ContestPrintLinkStandby();
gTasks[taskId].data[0] = 1;
}
else
{
RankContestants();
SetAttentionLevels();
gTasks[taskId].data[0] = 2;
}
break;
case 1:
if (!eContest.waitForLink)
gTasks[taskId].data[0] = 2;
break;
case 2:
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_ReadyUpdateHeartSliders;
break;
}
}
static void Task_ReadyUpdateHeartSliders(u8 taskId)
{
ShowHideNextTurnGfx(FALSE);
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].func = Task_UpdateHeartSliders;
}
static void Task_UpdateHeartSliders(u8 taskId)
{
switch (gTasks[taskId].data[0])
{
case 0:
if (++gTasks[taskId].data[1] > 20)
{
AnimateSliderHearts(SLIDER_HEART_ANIM_APPEAR);
gTasks[taskId].data[1] = 0;
gTasks[taskId].data[0]++;
}
break;
case 1:
if (!eContest.sliderHeartsAnimating)
{
if (++gTasks[taskId].data[1] > 20)
{
gTasks[taskId].data[1] = 0;
gTasks[taskId].data[0]++;
}
}
break;
case 2:
UpdateHeartSliders();
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].func = Task_WaitForHeartSliders;
break;
}
}
static void Task_WaitForHeartSliders(u8 taskId)
{
if (SlidersDoneUpdating())
gTasks[taskId].func = sub_80DA348;
}
static void sub_80DA348(u8 taskId)
{
DmaCopy32Defvars(3, eContestTempSave.cachedPlttBufferUnfaded, gPlttBufferUnfaded, PLTT_BUFFER_SIZE * 2);
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 2;
gTasks[taskId].func = Task_WaitPrintRoundResult;
}
static void Task_WaitPrintRoundResult(u8 taskId)
{
if (++gTasks[taskId].data[0] > 2)
{
gTasks[taskId].data[0] = 0;
if (--gTasks[taskId].data[1] == 0)
gTasks[taskId].func = Task_PrintRoundResultText;
}
}
static void Task_PrintRoundResultText(u8 taskId)
{
if (gTasks[taskId].data[0] == 0)
{
u8 attention = eContestantStatus[gContestPlayerMonIndex].attentionLevel;
ContestClearGeneralTextWindow();
StringCopy(gStringVar1, gContestMons[gContestPlayerMonIndex].nickname);
StringExpandPlaceholders(gStringVar4, sRoundResultTexts[attention]);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].data[0]++;
}
else
{
if (!Contest_RunTextPrinters())
{
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_ReUpdateHeartSliders;
ContestDebugDoPrint();
}
}
}
static void Task_ReUpdateHeartSliders(u8 taskId)
{
if (gTasks[taskId].data[0]++ > 29)
{
gTasks[taskId].data[0] = 0;
UpdateHeartSliders(); // ? Sliders have already been updated
gTasks[taskId].func = Task_WaitForHeartSlidersAgain;
}
}
static void Task_WaitForHeartSlidersAgain(u8 taskId)
{
if (SlidersDoneUpdating())
{
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_DropCurtainAtRoundEnd;
}
}
static void Task_DropCurtainAtRoundEnd(u8 taskId)
{
SetBgForCurtainDrop();
gTasks[taskId].func = Task_StartDropCurtainAtRoundEnd;
}
static void Task_UpdateContestantBoxOrder(u8 taskId)
{
UpdateContestantBoxOrder();
gTasks[taskId].func = Task_TryStartNextRoundOfAppeals;
}
static void Task_TryStartNextRoundOfAppeals(u8 taskId)
{
vu16 sp0 = GetGpuReg(REG_OFFSET_BG0CNT);
vu16 sp2 = GetGpuReg(REG_OFFSET_BG2CNT);
((vBgCnt *)&sp0)->priority = 0;
((vBgCnt *)&sp2)->priority = 0;
SetGpuReg(REG_OFFSET_BG0CNT, sp0);
SetGpuReg(REG_OFFSET_BG2CNT, sp2);
eContest.appealNumber++;
if (eContest.appealNumber == CONTEST_NUM_APPEALS)
{
gTasks[taskId].func = Task_EndAppeals;
}
else
{
SlideApplauseMeterIn();
gTasks[taskId].func = Task_StartNewRoundOfAppeals;
}
}
static void Task_StartNewRoundOfAppeals(u8 taskId)
{
if (!eContest.applauseMeterIsMoving)
gTasks[taskId].func = Task_DisplayAppealNumberText;
}
static void Task_EndAppeals(u8 taskId)
{
s32 i;
gBattle_BG0_Y = 0;
gBattle_BG2_Y = 0;
for (i = 0; i < CONTESTANT_COUNT; i++)
gContestMonAppealPointTotals[i] = eContestantStatus[i].pointTotal;
CalculateFinalScores();
ContestClearGeneralTextWindow();
if (!(gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK))
BravoTrainerPokemonProfile_BeforeInterview1(eContestantStatus[gContestPlayerMonIndex].prevMove);
else
{
CalculateContestLiveUpdateData();
SetConestLiveUpdateTVData();
ContestDebugPrintBitStrings();
}
gContestRngValue = gRngValue;
StringExpandPlaceholders(gStringVar4, gText_AllOutOfAppealTime);
Contest_StartTextPrinter(gStringVar4, TRUE);
gTasks[taskId].data[2] = 0;
gTasks[taskId].func = Task_WaitForOutOfTimeMsg;
}
static void Task_WaitForOutOfTimeMsg(u8 taskId)
{
if (!Contest_RunTextPrinters())
{
SetBgForCurtainDrop();
gBattle_BG1_X = 0;
gBattle_BG1_Y = DISPLAY_HEIGHT;
PlaySE12WithPanning(SE_CONTEST_CURTAIN_FALL, 0);
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_DropCurtainAtAppealsEnd;
}
}
static void Task_DropCurtainAtAppealsEnd(u8 taskId)
{
gBattle_BG1_Y -= 7;
if ((s16)gBattle_BG1_Y < 0)
gBattle_BG1_Y = 0;
if (gBattle_BG1_Y == 0)
{
gTasks[taskId].func = Task_TryCommunicateFinalStandings;
gTasks[taskId].data[0] = 0;
}
}
static void Task_TryCommunicateFinalStandings(u8 taskId)
{
if (gTasks[taskId].data[0]++ >= 50)
{
gTasks[taskId].data[0] = 0;
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
gTasks[taskId].func = Task_CommunicateFinalStandings;
}
else
{
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
gTasks[taskId].func = Task_ContestReturnToField;
}
}
}
static void Task_CommunicateFinalStandings(u8 taskId)
{
u8 taskId2 = CreateTask(Task_LinkContest_CommunicateFinalStandings, 0);
SetTaskFuncWithFollowupFunc(taskId2, Task_LinkContest_CommunicateFinalStandings, Task_EndCommunicateFinalStandings);
gTasks[taskId].func = TaskDummy1;
ContestPrintLinkStandby();
SetBottomSliderHeartsInvisibility(FALSE);
}
static void Task_EndCommunicateFinalStandings(u8 taskId)
{
DestroyTask(taskId);
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
gTasks[eContest.mainTaskId].func = Task_ContestReturnToField;
}
static void Task_ContestReturnToField(u8 taskId)
{
if (!gPaletteFade.active)
{
DestroyTask(taskId);
gFieldCallback = FieldCB_ContestReturnToField;
FreeAllWindowBuffers();
FreeContestResources();
FreeMonSpritesGfx();
SetMainCallback2(CB2_ReturnToField);
}
}
static void FieldCB_ContestReturnToField(void)
{
ScriptContext2_Disable();
EnableBothScriptContexts();
}
static void TryPutPlayerLast(void)
{
if (!(gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK))
gContestPlayerMonIndex = CONTESTANT_COUNT - 1;
}
static bool8 IsPlayerLinkLeader(void)
{
if (gContestPlayerMonIndex == gContestLinkLeaderIndex)
return TRUE;
return FALSE;
}
void CreateContestMonFromParty(u8 partyIndex)
{
u8 name[20];
u16 heldItem;
s16 cool;
s16 beauty;
s16 cute;
s16 smart;
s16 tough;
StringCopy(name, gSaveBlock2Ptr->playerName);
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
StripPlayerNameForLinkContest(name);
}
memcpy(gContestMons[gContestPlayerMonIndex].trainerName, name, 8);
if (gSaveBlock2Ptr->playerGender == MALE)
gContestMons[gContestPlayerMonIndex].trainerGfxId = OBJ_EVENT_GFX_LINK_BRENDAN;
else
gContestMons[gContestPlayerMonIndex].trainerGfxId = OBJ_EVENT_GFX_LINK_MAY;
gContestMons[gContestPlayerMonIndex].aiFlags = 0;
gContestMons[gContestPlayerMonIndex].highestRank = 0;
gContestMons[gContestPlayerMonIndex].species = GetMonData(&gPlayerParty[partyIndex], MON_DATA_SPECIES);
GetMonData(&gPlayerParty[partyIndex], MON_DATA_NICKNAME, name);
StringGetEnd10(name);
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
StripMonNameForLinkContest(name, GetMonData(&gPlayerParty[partyIndex], MON_DATA_LANGUAGE));
}
memcpy(gContestMons[gContestPlayerMonIndex].nickname, name, POKEMON_NAME_LENGTH + 1);
StringCopy(gContestMons[gContestPlayerMonIndex].nickname, name);
gContestMons[gContestPlayerMonIndex].cool = GetMonData(&gPlayerParty[partyIndex], MON_DATA_COOL);
gContestMons[gContestPlayerMonIndex].beauty = GetMonData(&gPlayerParty[partyIndex], MON_DATA_BEAUTY);
gContestMons[gContestPlayerMonIndex].cute = GetMonData(&gPlayerParty[partyIndex], MON_DATA_CUTE);
gContestMons[gContestPlayerMonIndex].smart = GetMonData(&gPlayerParty[partyIndex], MON_DATA_SMART);
gContestMons[gContestPlayerMonIndex].tough = GetMonData(&gPlayerParty[partyIndex], MON_DATA_TOUGH);
gContestMons[gContestPlayerMonIndex].sheen = GetMonData(&gPlayerParty[partyIndex], MON_DATA_SHEEN);
gContestMons[gContestPlayerMonIndex].moves[0] = GetMonData(&gPlayerParty[partyIndex], MON_DATA_MOVE1);
gContestMons[gContestPlayerMonIndex].moves[1] = GetMonData(&gPlayerParty[partyIndex], MON_DATA_MOVE2);
gContestMons[gContestPlayerMonIndex].moves[2] = GetMonData(&gPlayerParty[partyIndex], MON_DATA_MOVE3);
gContestMons[gContestPlayerMonIndex].moves[3] = GetMonData(&gPlayerParty[partyIndex], MON_DATA_MOVE4);
gContestMons[gContestPlayerMonIndex].personality = GetMonData(&gPlayerParty[partyIndex], MON_DATA_PERSONALITY);
gContestMons[gContestPlayerMonIndex].otId = GetMonData(&gPlayerParty[partyIndex], MON_DATA_OT_ID);
heldItem = GetMonData(&gPlayerParty[partyIndex], MON_DATA_HELD_ITEM);
cool = gContestMons[gContestPlayerMonIndex].cool;
beauty = gContestMons[gContestPlayerMonIndex].beauty;
cute = gContestMons[gContestPlayerMonIndex].cute;
smart = gContestMons[gContestPlayerMonIndex].smart;
tough = gContestMons[gContestPlayerMonIndex].tough;
if (heldItem == ITEM_RED_SCARF)
cool += 20;
else if (heldItem == ITEM_BLUE_SCARF)
beauty += 20;
else if (heldItem == ITEM_PINK_SCARF)
cute += 20;
else if (heldItem == ITEM_GREEN_SCARF)
smart += 20;
else if (heldItem == ITEM_YELLOW_SCARF)
tough += 20;
if (cool > 255)
cool = 255;
if (beauty > 255)
beauty = 255;
if (cute > 255)
cute = 255;
if (smart > 255)
smart = 255;
if (tough > 255)
tough = 255;
gContestMons[gContestPlayerMonIndex].cool = cool;
gContestMons[gContestPlayerMonIndex].beauty = beauty;
gContestMons[gContestPlayerMonIndex].cute = cute;
gContestMons[gContestPlayerMonIndex].smart = smart;
gContestMons[gContestPlayerMonIndex].tough = tough;
}
void SetContestants(u8 contestType, u8 rank)
{
s32 i;
u8 opponentsCount = 0;
u8 opponents[100];
bool8 allowPostgameContestants = FALSE;
const u8 * filter;
TryPutPlayerLast();
if (FlagGet(FLAG_SYS_GAME_CLEAR) && !(gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK))
allowPostgameContestants = TRUE;
// Find all suitable opponents
filter = gPostgameContestOpponentFilter;
for (i = 0; i < ARRAY_COUNT(gContestOpponents); i++)
{
if (rank == gContestOpponents[i].whichRank)
{
if (allowPostgameContestants == TRUE)
{
if (filter[i] == CONTEST_FILTER_NO_POSTGAME)
continue;
}
else
{
if (filter[i] == CONTEST_FILTER_ONLY_POSTGAME)
continue;
}
if (contestType == CONTEST_CATEGORY_COOL && gContestOpponents[i].aiPool_Cool)
opponents[opponentsCount++] = i;
else if (contestType == CONTEST_CATEGORY_BEAUTY && gContestOpponents[i].aiPool_Beauty)
opponents[opponentsCount++] = i;
else if (contestType == CONTEST_CATEGORY_CUTE && gContestOpponents[i].aiPool_Cute)
opponents[opponentsCount++] = i;
else if (contestType == CONTEST_CATEGORY_SMART && gContestOpponents[i].aiPool_Smart)
opponents[opponentsCount++] = i;
else if (contestType == CONTEST_CATEGORY_TOUGH && gContestOpponents[i].aiPool_Tough)
opponents[opponentsCount++] = i;
}
}
opponents[opponentsCount] = 0xFF;
// Choose three random opponents from the list
for (i = 0; i < CONTESTANT_COUNT - 1; i++)
{
u16 rnd = Random() % opponentsCount;
s32 j;
gContestMons[i] = gContestOpponents[opponents[rnd]];
for (j = rnd; opponents[j] != 0xFF; j++)
opponents[j] = opponents[j + 1];
opponentsCount--;
}
CreateContestMonFromParty(gContestMonPartyIndex);
}
void SetLinkAIContestants(u8 contestType, u8 rank, bool32 isPostgame)
{
s32 i, j;
u8 opponentsCount = 0;
u8 opponents[100];
if (gNumLinkContestPlayers == CONTESTANT_COUNT)
return;
// Find all suitable AI opponents
for (i = 0; i < ARRAY_COUNT(gContestOpponents); i++)
{
if (rank != gContestOpponents[i].whichRank)
continue;
if (isPostgame == TRUE)
{
if (gPostgameContestOpponentFilter[i] == CONTEST_FILTER_NO_POSTGAME)
continue;
}
else
{
if (gPostgameContestOpponentFilter[i] == CONTEST_FILTER_ONLY_POSTGAME)
continue;
}
if ((contestType == CONTEST_CATEGORY_COOL && gContestOpponents[i].aiPool_Cool)
|| (contestType == CONTEST_CATEGORY_BEAUTY && gContestOpponents[i].aiPool_Beauty)
|| (contestType == CONTEST_CATEGORY_CUTE && gContestOpponents[i].aiPool_Cute)
|| (contestType == CONTEST_CATEGORY_SMART && gContestOpponents[i].aiPool_Smart)
|| (contestType == CONTEST_CATEGORY_TOUGH && gContestOpponents[i].aiPool_Tough))
opponents[opponentsCount++] = i;
}
opponents[opponentsCount] = 0xFF;
// Fill remaining contestant slots with random AI opponents from the list
for (i = 0; i < CONTESTANT_COUNT - gNumLinkContestPlayers; i++)
{
u16 rnd = GetContestRand() % opponentsCount;
gContestMons[gNumLinkContestPlayers + i] = gContestOpponents[opponents[rnd]];
StripPlayerNameForLinkContest(gContestMons[gNumLinkContestPlayers + i].trainerName);
StripMonNameForLinkContest(gContestMons[gNumLinkContestPlayers + i].nickname, GAME_LANGUAGE);
for (j = rnd; opponents[j] != 0xFF; j++)
opponents[j] = opponents[j + 1];
opponentsCount--;
}
}
u8 GetContestEntryEligibility(struct Pokemon *pkmn)
{
u8 ribbon;
u8 eligibility;
if (GetMonData(pkmn, MON_DATA_IS_EGG))
return CANT_ENTER_CONTEST_EGG;
if (GetMonData(pkmn, MON_DATA_HP) == 0)
return CANT_ENTER_CONTEST_FAINTED;
switch (gSpecialVar_ContestCategory)
{
case CONTEST_CATEGORY_COOL:
ribbon = GetMonData(pkmn, MON_DATA_COOL_RIBBON);
break;
case CONTEST_CATEGORY_BEAUTY:
ribbon = GetMonData(pkmn, MON_DATA_BEAUTY_RIBBON);
break;
case CONTEST_CATEGORY_CUTE:
ribbon = GetMonData(pkmn, MON_DATA_CUTE_RIBBON);
break;
case CONTEST_CATEGORY_SMART:
ribbon = GetMonData(pkmn, MON_DATA_SMART_RIBBON);
break;
case CONTEST_CATEGORY_TOUGH:
ribbon = GetMonData(pkmn, MON_DATA_TOUGH_RIBBON);
break;
default:
return CANT_ENTER_CONTEST;
}
// Couldn't get this to match any other way.
// Returns 2, 1, or 0 respectively if ribbon's rank is above, equal, or below
// the current contest rank.
if (ribbon > gSpecialVar_ContestRank)
eligibility = CAN_ENTER_CONTEST_HIGH_RANK;
else if (ribbon >= gSpecialVar_ContestRank)
eligibility = CAN_ENTER_CONTEST_EQUAL_RANK;
else
eligibility = CANT_ENTER_CONTEST;
return eligibility;
}
static void DrawContestantWindowText(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
FillWindowPixelBuffer(gContestantTurnOrder[i], PIXEL_FILL(0));
PrintContestantTrainerName(i);
PrintContestantMonName(i);
}
}
static u8 *Contest_CopyStringWithColor(const u8 *string, u8 color)
{
u8 * ptr = StringCopy(gDisplayedStringBattle, gText_ColorTransparent);
ptr[-1] = color; // Overwrites the "{COLOR TRANSPARENT}" part of the string.
ptr = StringCopy(ptr, string);
return ptr;
}
static void PrintContestantTrainerName(u8 contestant)
{
PrintContestantTrainerNameWithColor(contestant, contestant + CONTESTANT_TEXT_COLOR_START);
}
static void PrintContestantTrainerNameWithColor(u8 contestant, u8 color)
{
u8 buffer[32];
s32 offset;
StringCopy(buffer, gText_Slash);
StringAppend(buffer, gContestMons[contestant].trainerName);
Contest_CopyStringWithColor(buffer, color);
offset = GetStringRightAlignXOffset(7, gDisplayedStringBattle, 0x60);
if (offset > 55)
offset = 55;
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[contestant], gDisplayedStringBattle, offset, 1, 7);
}
static void PrintContestantMonName(u8 contestant)
{
PrintContestantMonNameWithColor(contestant, contestant + CONTESTANT_TEXT_COLOR_START);
}
static void PrintContestantMonNameWithColor(u8 contestant, u8 color)
{
Contest_CopyStringWithColor(gContestMons[contestant].nickname, color);
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[contestant], gDisplayedStringBattle, 5, 1, 7);
}
static u16 CalculateContestantRound1Points(u8 who, u8 contestCategory)
{
u8 statMain;
u8 statSub1;
u8 statSub2;
switch (contestCategory)
{
case CONTEST_CATEGORY_COOL:
statMain = gContestMons[who].cool;
statSub1 = gContestMons[who].tough;
statSub2 = gContestMons[who].beauty;
break;
case CONTEST_CATEGORY_BEAUTY:
statMain = gContestMons[who].beauty;
statSub1 = gContestMons[who].cool;
statSub2 = gContestMons[who].cute;
break;
case CONTEST_CATEGORY_CUTE:
statMain = gContestMons[who].cute;
statSub1 = gContestMons[who].beauty;
statSub2 = gContestMons[who].smart;
break;
case CONTEST_CATEGORY_SMART:
statMain = gContestMons[who].smart;
statSub1 = gContestMons[who].cute;
statSub2 = gContestMons[who].tough;
break;
case CONTEST_CATEGORY_TOUGH:
default:
statMain = gContestMons[who].tough;
statSub1 = gContestMons[who].smart;
statSub2 = gContestMons[who].cool;
break;
}
return statMain + (statSub1 + statSub2 + gContestMons[who].sheen) / 2;
}
void CalculateRound1Points(u8 contestCategory)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
gContestMonRound1Points[i] = CalculateContestantRound1Points(i, contestCategory);
}
static u8 CreateJudgeSprite(void)
{
u8 spriteId;
LoadCompressedSpriteSheet(&sSpriteSheet_Judge);
LoadCompressedPalette(gContest2Pal, 0x110, 32);
spriteId = CreateSprite(&sSpriteTemplate_Judge, 112, 36, 30);
gSprites[spriteId].oam.paletteNum = 1;
gSprites[spriteId].callback = SpriteCallbackDummy;
return spriteId;
}
static u8 CreateJudgeSpeechBubbleSprite(void)
{
u8 spriteId;
LoadCompressedSpriteSheet(&sSpriteSheet_JudgeSymbols);
LoadCompressedSpritePalette(&sSpritePalette_JudgeSymbols);
spriteId = CreateSprite(&sSpriteTemplate_JudgeSpeechBubble, 96, 10, 29);
gSprites[spriteId].invisible = TRUE;
gSprites[spriteId].data[0] = gSprites[spriteId].oam.tileNum;
return spriteId;
}
static u8 CreateContestantSprite(u16 species, u32 otId, u32 personality, u32 index)
{
u8 spriteId;
species = SanitizeSpecies(species);
if (index == gContestPlayerMonIndex)
HandleLoadSpecialPokePic_2(&gMonBackPicTable[species], gMonSpritesGfxPtr->sprites.ptr[0], species, personality);
else
HandleLoadSpecialPokePic_DontHandleDeoxys(&gMonBackPicTable[species], gMonSpritesGfxPtr->sprites.ptr[0], species, personality);
LoadCompressedPalette(GetMonSpritePalFromSpeciesAndPersonality(species, otId, personality), 0x120, 0x20);
SetMultiuseSpriteTemplateToPokemon(species, 0);
spriteId = CreateSprite(&gMultiuseSpriteTemplate, 0x70, GetBattlerSpriteFinal_Y(2, species, FALSE), 30);
gSprites[spriteId].oam.paletteNum = 2;
gSprites[spriteId].oam.priority = 2;
gSprites[spriteId].subpriority = GetBattlerSpriteSubpriority(2);
gSprites[spriteId].callback = SpriteCallbackDummy;
gSprites[spriteId].data[0] = gSprites[spriteId].oam.paletteNum;
gSprites[spriteId].data[2] = species;
if (IsSpeciesNotUnown(species))
gSprites[spriteId].affineAnims = gAffineAnims_BattleSpriteContest;
else
gSprites[spriteId].affineAnims = gAffineAnims_BattleSpriteOpponentSide;
StartSpriteAffineAnim(&gSprites[spriteId], BATTLER_AFFINE_NORMAL);
return spriteId;
}
bool8 IsSpeciesNotUnown(u16 species)
{
if (species == SPECIES_UNOWN)
return FALSE;
else
return TRUE;
}
// The contestant info windows and general-purpose text box are drawn on one half, while
// the moves and move description windows are drawn on another screen. Only the first 32 * 20
// tiles are actually drawn on screen.
static void SwapMoveDescAndContestTilemaps(void)
{
CpuCopy16(gContestResources->contestBgTilemaps[0], gContestResources->contestBgTilemaps[0] + 0x500, 32 * 20);
CpuCopy16(gContestResources->contestBgTilemaps[2], gContestResources->contestBgTilemaps[2] + 0x500, 32 * 20);
}
// Functionally unused
static u16 GetMoveEffectSymbolTileOffset(u16 move, u8 contestant)
{
u16 offset;
switch (gContestEffects[gContestMoves[move].effect].effectType)
{
case 0:
case 1:
case 8:
offset = 0x9082;
break;
case 2:
case 3:
offset = 0x9088;
break;
default:
offset = 0x9086;
break;
}
offset += 0x9000 + (contestant << 12);
return offset;
}
static void PrintContestMoveDescription(u16 a)
{
u8 category;
u16 categoryTile;
u8 numHearts;
// The contest category icon is implemented as a 5x2 group of tiles.
category = gContestMoves[a].contestCategory;
if (category == CONTEST_CATEGORY_COOL)
categoryTile = 0x4040;
else if (category == CONTEST_CATEGORY_BEAUTY)
categoryTile = 0x4045;
else if (category == CONTEST_CATEGORY_CUTE)
categoryTile = 0x404A;
else if (category == CONTEST_CATEGORY_SMART)
categoryTile = 0x406A;
else
categoryTile = 0x408A;
ContestBG_FillBoxWithIncrementingTile(0, categoryTile, 0x0b, 0x1f, 0x05, 0x01, 0x11, 0x01);
ContestBG_FillBoxWithIncrementingTile(0, categoryTile + 0x10, 0x0b, 0x20, 0x05, 0x01, 0x11, 0x01);
// Appeal hearts
if (gContestEffects[gContestMoves[a].effect].appeal == 0xFF)
numHearts = 0;
else
numHearts = gContestEffects[gContestMoves[a].effect].appeal / 10;
if (numHearts > MAX_CONTEST_MOVE_HEARTS)
numHearts = MAX_CONTEST_MOVE_HEARTS;
ContestBG_FillBoxWithTile(0, TILE_EMPTY_APPEAL_HEART, 0x15, 0x1f, MAX_CONTEST_MOVE_HEARTS, 0x01, 0x11);
ContestBG_FillBoxWithTile(0, TILE_FILLED_APPEAL_HEART, 0x15, 0x1f, numHearts, 0x01, 0x11);
// Jam hearts
if (gContestEffects[gContestMoves[a].effect].jam == 0xFF)
numHearts = 0;
else
numHearts = gContestEffects[gContestMoves[a].effect].jam / 10;
if (numHearts > MAX_CONTEST_MOVE_HEARTS)
numHearts = MAX_CONTEST_MOVE_HEARTS;
ContestBG_FillBoxWithTile(0, TILE_EMPTY_JAM_HEART, 0x15, 0x20, MAX_CONTEST_MOVE_HEARTS, 0x01, 0x11);
ContestBG_FillBoxWithTile(0, TILE_FILLED_JAM_HEART, 0x15, 0x20, numHearts, 0x01, 0x11);
FillWindowPixelBuffer(WIN_MOVE_DESCRIPTION, PIXEL_FILL(0));
Contest_PrintTextToBg0WindowStd(WIN_MOVE_DESCRIPTION, gContestEffectDescriptionPointers[gContestMoves[a].effect]);
Contest_PrintTextToBg0WindowStd(WIN_SLASH, gText_Slash);
}
static void DrawMoveEffectSymbol(u16 move, u8 contestant)
{
u8 contestantOffset = gContestantTurnOrder[contestant] * 5 + 2;
if (!Contest_IsMonsTurnDisabled(contestant) && move != MOVE_NONE)
{
u16 tile = GetMoveEffectSymbolTileOffset(move, contestant);
ContestBG_FillBoxWithIncrementingTile(0, tile, 20, contestantOffset, 2, 1, 17, 1);
ContestBG_FillBoxWithIncrementingTile(0, tile + 16, 20, contestantOffset + 1, 2, 1, 17, 1);
}
else
{
ContestBG_FillBoxWithTile(0, 0, 20, contestantOffset, 2, 2, 17);
}
}
// Unused
static void DrawMoveEffectSymbols(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
DrawMoveEffectSymbol(eContestantStatus[i].currMove, i);
}
static u16 GetStarTileOffset(void)
{
return 0x2034;
}
static bool8 UpdateConditionStars(u8 contestantIdx, bool8 resetMod)
{
u8 contestantOffset;
s32 numStars;
if (eContestantStatus[contestantIdx].conditionMod == CONDITION_NO_CHANGE)
return FALSE;
contestantOffset = gContestantTurnOrder[contestantIdx] * 5 + 2;
numStars = eContestantStatus[contestantIdx].condition / 10;
if (eContestantStatus[contestantIdx].conditionMod == CONDITION_GAIN)
{
ContestBG_FillBoxWithTile(0, GetStarTileOffset(), 19, contestantOffset, 1, numStars, 17);
if (resetMod)
{
PlaySE(SE_EXP_MAX);
eContestantStatus[contestantIdx].conditionMod = CONDITION_NO_CHANGE;
}
}
else // CONDITION_LOSE
{
ContestBG_FillBoxWithTile(0, 0, 19, contestantOffset + numStars, 1, 3 - numStars, 17);
if (resetMod)
{
PlaySE(SE_CONTEST_CONDITION_LOSE);
eContestantStatus[contestantIdx].conditionMod = CONDITION_NO_CHANGE;
}
}
return TRUE;
}
static void DrawConditionStars(void)
{
s32 i;
s32 numStars;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u8 contestantOffset = gContestantTurnOrder[i] * 5 + 2;
u16 starOffset = GetStarTileOffset();
numStars = eContestantStatus[i].condition / 10;
ContestBG_FillBoxWithTile(0, starOffset, 19, contestantOffset, 1, numStars, 17);
ContestBG_FillBoxWithTile(0, 0, 19, contestantOffset + numStars, 1, 3 - numStars, 17);
}
}
static u16 GetStatusSymbolTileOffset(u8 status)
{
u16 offset = 0;
switch (status)
{
case STAT_SYMBOL_CIRCLE: // For resistant
offset = 0x80;
break;
case STAT_SYMBOL_WAVE: // For nervous
offset = 0x84;
break;
case STAT_SYMBOL_X: // For turn skipped
offset = 0x86;
break;
case STAT_SYMBOL_SWIRL: // For jammed/unnerved
offset = 0x88;
break;
case STAT_SYMBOL_SQUARE: // Never used
offset = 0x82;
break;
}
offset += 0x9000;
return offset;
}
static bool8 DrawStatusSymbol(u8 contestant)
{
bool8 statused = TRUE;
u16 symbolOffset = 0;
u8 contestantOffset = gContestantTurnOrder[contestant] * 5 + 2;
if (eContestantStatus[contestant].resistant
|| eContestantStatus[contestant].immune
|| eContestantStatus[contestant].jamSafetyCount != 0
|| eContestantStatus[contestant].jamReduction != 0)
symbolOffset = GetStatusSymbolTileOffset(STAT_SYMBOL_CIRCLE);
else if (eContestantStatus[contestant].nervous)
symbolOffset = GetStatusSymbolTileOffset(STAT_SYMBOL_WAVE);
else if (eContestantStatus[contestant].numTurnsSkipped != 0 || eContestantStatus[contestant].noMoreTurns)
symbolOffset = GetStatusSymbolTileOffset(STAT_SYMBOL_X);
else
statused = FALSE;
if (statused)
{
ContestBG_FillBoxWithIncrementingTile(0, symbolOffset, 20, contestantOffset, 2, 1, 17, 1);
ContestBG_FillBoxWithIncrementingTile(0, symbolOffset + 16, 20, contestantOffset + 1, 2, 1, 17, 1);
}
else
{
ContestBG_FillBoxWithTile(0, 0, 20, contestantOffset, 2, 2, 17);
}
return statused;
}
static void DrawStatusSymbols(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
DrawStatusSymbol(i);
}
static void ContestClearGeneralTextWindow(void)
{
FillWindowPixelBuffer(WIN_GENERAL_TEXT, PIXEL_FILL(0));
CopyWindowToVram(WIN_GENERAL_TEXT, 2);
Contest_SetBgCopyFlags(0);
}
static u16 GetChosenMove(u8 contestant)
{
if (Contest_IsMonsTurnDisabled(contestant))
return MOVE_NONE;
if (contestant == gContestPlayerMonIndex)
{
return gContestMons[contestant].moves[eContest.playerMoveChoice];
}
else
{
u8 moveChoice;
ContestAI_ResetAI(contestant);
moveChoice = ContestAI_GetActionToUse();
return gContestMons[contestant].moves[moveChoice];
}
}
static void GetAllChosenMoves(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
eContestantStatus[i].currMove = GetChosenMove(i);
}
static void RankContestants(void)
{
s32 i;
s32 j;
s16 arr[CONTESTANT_COUNT];
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i].pointTotal += eContestantStatus[i].appeal;
arr[i] = eContestantStatus[i].pointTotal;
}
// Sort the point totals using bubble-sort.
for (i = 0; i < CONTESTANT_COUNT - 1; i++)
{
for (j = CONTESTANT_COUNT - 1; j > i; j--)
{
if (arr[j - 1] < arr[j])
{
u16 temp;
SWAP(arr[j], arr[j - 1], temp);
}
}
}
// For each contestant, find the best rank with their point total.
// Normally, each point total is different, and this will output the
// rankings as expected. However, if two pokemon are tied, then they
// both get the best rank for that point total.
//
// For example if the point totals are [100, 80, 80, 50], the ranks will
// be [1, 2, 2, 4]. The pokemon with a point total of 80 stop looking
// when they see the first 80 in the array, so they both share the '2'
// rank.
for (i = 0; i < CONTESTANT_COUNT; i++)
{
for (j = 0; j < CONTESTANT_COUNT; j++)
{
if (eContestantStatus[i].pointTotal == arr[j])
{
eContestantStatus[i].ranking = j;
break;
}
}
}
SortContestants(TRUE);
ApplyNextTurnOrder();
}
static void SetAttentionLevels(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u8 attentionLevel;
if (eContestantStatus[i].currMove == MOVE_NONE)
attentionLevel = 5;
else if (eContestantStatus[i].appeal <= 0)
attentionLevel = 0;
else if (eContestantStatus[i].appeal < 30)
attentionLevel = 1;
else if (eContestantStatus[i].appeal < 60)
attentionLevel = 2;
else if (eContestantStatus[i].appeal < 80)
attentionLevel = 3;
else
attentionLevel = 4;
eContestantStatus[i].attentionLevel = attentionLevel;
}
}
static bool8 ContestantCanUseTurn(u8 contestant)
{
if (eContestantStatus[contestant].numTurnsSkipped != 0 || eContestantStatus[contestant].noMoreTurns)
return FALSE;
else
return TRUE;
}
static void SetContestantStatusesForNextRound(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i].appeal = 0;
eContestantStatus[i].baseAppeal = 0;
eContestantStatus[i].jamSafetyCount = 0;
if (eContestantStatus[i].numTurnsSkipped > 0)
eContestantStatus[i].numTurnsSkipped--;
eContestantStatus[i].jam = 0;
eContestantStatus[i].resistant = 0;
eContestantStatus[i].jamReduction = 0;
eContestantStatus[i].immune = 0;
eContestantStatus[i].moreEasilyStartled = 0;
eContestantStatus[i].usedRepeatableMove = 0;
eContestantStatus[i].nervous = FALSE;
eContestantStatus[i].effectStringId = CONTEST_STRING_NONE;
eContestantStatus[i].effectStringId2 = CONTEST_STRING_NONE;
eContestantStatus[i].conditionMod = CONDITION_NO_CHANGE;
eContestantStatus[i].repeatedPrevMove = eContestantStatus[i].repeatedMove;
eContestantStatus[i].repeatedMove = FALSE;
eContestantStatus[i].turnOrderModAction = 0;
eContestantStatus[i].appealTripleCondition = 0;
if (eContestantStatus[i].turnSkipped)
{
eContestantStatus[i].numTurnsSkipped = 1;
eContestantStatus[i].turnSkipped = 0;
}
if (eContestantStatus[i].exploded)
{
eContestantStatus[i].noMoreTurns = TRUE;
eContestantStatus[i].exploded = FALSE;
}
eContestantStatus[i].overrideCategoryExcitementMod = 0;
}
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i].prevMove = eContestantStatus[i].currMove;
eContest.moveHistory[eContest.appealNumber][i] = eContestantStatus[i].currMove;
eContest.excitementHistory[eContest.appealNumber][i] = Contest_GetMoveExcitement(eContestantStatus[i].currMove);
eContestantStatus[i].currMove = MOVE_NONE;
}
eContestExcitement.frozen = FALSE;
}
bool8 Contest_IsMonsTurnDisabled(u8 contestant)
{
if (eContestantStatus[contestant].numTurnsSkipped != 0 || eContestantStatus[contestant].noMoreTurns)
return TRUE;
else
return FALSE;
}
static void CalculateTotalPointsForContestant(u8 contestant)
{
gContestMonRound2Points[contestant] = GetContestantRound2Points(contestant);
gContestMonTotalPoints[contestant] = gContestMonRound1Points[contestant] + gContestMonRound2Points[contestant];
}
static void CalculateFinalScores(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
CalculateTotalPointsForContestant(i);
DetermineFinalStandings();
}
static s16 GetContestantRound2Points(u8 contestant)
{
return gContestMonAppealPointTotals[contestant] * 2;
}
static void DetermineFinalStandings(void)
{
u16 randomOrdering[CONTESTANT_COUNT] = {0};
struct ContestFinalStandings standings[CONTESTANT_COUNT];
s32 i;
// Seed random order in case of ties
for (i = 0; i < CONTESTANT_COUNT; i++)
{
s32 j;
randomOrdering[i] = Random();
for (j = 0; j < i; j++)
{
if (randomOrdering[i] == randomOrdering[j])
{
i--;
break;
}
}
}
// Init data for ranking contestants
for (i = 0; i < CONTESTANT_COUNT; i++)
{
standings[i].totalPoints = gContestMonTotalPoints[i];
standings[i].round1Points = gContestMonRound1Points[i];
standings[i].random = randomOrdering[i];
standings[i].contestant = i;
}
// Rank contestants
for (i = 0; i < CONTESTANT_COUNT - 1; i++)
{
s32 j;
for (j = CONTESTANT_COUNT - 1; j > i; j--)
{
if (DidContestantPlaceHigher(j - 1, j, standings))
{
// Swap contestants in array
struct ContestFinalStandings temp;
temp.totalPoints = standings[j - 1].totalPoints;
temp.round1Points = standings[j - 1].round1Points;
temp.random = standings[j - 1].random;
temp.contestant = standings[j - 1].contestant;
standings[j - 1].totalPoints = standings[j].totalPoints;
standings[j - 1].round1Points = standings[j].round1Points;
standings[j - 1].random = standings[j].random;
standings[j - 1].contestant = standings[j].contestant;
standings[j].totalPoints = temp.totalPoints;
standings[j].round1Points = temp.round1Points;
standings[j].random = temp.random;
standings[j].contestant = temp.contestant;
}
}
}
// Assign placements. i is the placing (0 is 1st, 1 is 2nd...)
for (i = 0; i < CONTESTANT_COUNT; i++)
gContestFinalStandings[standings[i].contestant] = i;
}
void SaveLinkContestResults(void)
{
if ((gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK))
{
gSaveBlock2Ptr->contestLinkResults[gSpecialVar_ContestCategory][gContestFinalStandings[gContestPlayerMonIndex]] =
((gSaveBlock2Ptr->contestLinkResults[gSpecialVar_ContestCategory][gContestFinalStandings[gContestPlayerMonIndex]] + 1) > 9999) ? 9999 :
(gSaveBlock2Ptr->contestLinkResults[gSpecialVar_ContestCategory][gContestFinalStandings[gContestPlayerMonIndex]] + 1);
}
}
static bool8 DidContestantPlaceHigher(s32 a, s32 b, struct ContestFinalStandings *standings)
{
bool8 retVal;
// Rank contestants first based on total points
if (standings[a].totalPoints < standings[b].totalPoints)
retVal = TRUE;
else if (standings[a].totalPoints > standings[b].totalPoints)
retVal = FALSE;
// If tied, rank on round 1 points
else if (standings[a].round1Points < standings[b].round1Points)
retVal = TRUE;
else if (standings[a].round1Points > standings[b].round1Points)
retVal = FALSE;
// If tied again, choose randomly
else if (standings[a].random < standings[b].random)
retVal = TRUE;
else
retVal = FALSE;
return retVal;
}
static void ContestPrintLinkStandby(void)
{
gBattle_BG0_Y = 0;
gBattle_BG2_Y = 0;
ContestClearGeneralTextWindow();
Contest_StartTextPrinter(gText_LinkStandby4, 0);
}
static void FillContestantWindowBgs(void)
{
int i;
for(i = 0; i < CONTESTANT_COUNT; i++)
ContestBG_FillBoxWithTile(0, 0, 0x16, 2 + i * 5, 8, 2, 0x11);
}
static u16 GetAppealHeartTileOffset(u8 contestant)
{
u16 offset;
if (contestant == 0)
offset = 0x5011;
else if (contestant == 1)
offset = 0x6011;
else if (contestant == 2)
offset = 0x7011;
else
offset = 0x8011;
return offset + 1;
}
static s8 GetNumHeartsFromAppealPoints(s16 appeal)
{
s8 hearts = appeal / 10;
if (hearts > 16)
hearts = 16;
else if (hearts < -16)
hearts = -16;
return hearts;
}
#define tNumHearts data[0]
#define tHeartsDelta data[1]
#define tHeartsSign data[2]
#define tContestant data[3]
#define tDelayTimer data[10]
static u8 UpdateAppealHearts(s16 startAppeal, s16 appealDelta, u8 contestant)
{
u8 taskId;
s8 startHearts;
s8 heartsDelta;
eContestGfxState[contestant].updatingAppealHearts = TRUE;
taskId = CreateTask(Task_UpdateAppealHearts, 20);
startHearts = GetNumHeartsFromAppealPoints(startAppeal);
heartsDelta = GetNumHeartsFromAppealPoints(startAppeal + appealDelta) - startHearts;
GetAppealHeartTileOffset(contestant); // unused return value
gTasks[taskId].tNumHearts = abs(startHearts);
gTasks[taskId].tHeartsDelta = heartsDelta;
if (startHearts > 0 || (startHearts == 0 && heartsDelta > 0))
gTasks[taskId].tHeartsSign = 1;
else
gTasks[taskId].tHeartsSign = -1;
gTasks[taskId].tContestant = contestant;
return taskId;
}
static void Task_UpdateAppealHearts(u8 taskId)
{
u8 contestant = gTasks[taskId].tContestant;
s16 startHearts = gTasks[taskId].tNumHearts;
s16 heartsDelta = gTasks[taskId].tHeartsDelta;
if (++gTasks[taskId].tDelayTimer > 14)
{
u16 heartOffset;
u8 newNumHearts;
u8 pitchMod;
bool8 onSecondLine;
gTasks[taskId].tDelayTimer = 0;
if (gTasks[taskId].tHeartsDelta == 0)
{
// No more hearts to add/remove, end
DestroyTask(taskId);
eContestGfxState[contestant].updatingAppealHearts = FALSE;
return;
}
else if (startHearts == 0)
{
if (heartsDelta < 0)
{
// Losing hearts, get black heart offset
heartOffset = GetAppealHeartTileOffset(contestant) + 2;
gTasks[taskId].tHeartsDelta++;
}
else
{
// Gaining hearts, get red heart offset
heartOffset = GetAppealHeartTileOffset(contestant);
gTasks[taskId].tHeartsDelta--;
}
newNumHearts = gTasks[taskId].tNumHearts++;
}
else
{
if (gTasks[taskId].tHeartsSign < 0)
{
// Hearts currently black (negative)
if (heartsDelta < 0)
{
// Losing points, add black heart
newNumHearts = gTasks[taskId].tNumHearts++;
gTasks[taskId].tHeartsDelta++;
heartOffset = GetAppealHeartTileOffset(contestant) + 2;
}
else
{
// Gaining points, remove black heart
newNumHearts = --gTasks[taskId].tNumHearts;
heartOffset = 0;
gTasks[taskId].tHeartsDelta--;
}
}
else
{
// Hearts currently red (positive)
if (heartsDelta < 0)
{
// Losing points, remove red heart
newNumHearts = --gTasks[taskId].tNumHearts;
heartOffset = 0;
gTasks[taskId].tHeartsDelta++;
}
else
{
// Gaining points, add red heart
newNumHearts = gTasks[taskId].tNumHearts++;
gTasks[taskId].tHeartsDelta--;
heartOffset = GetAppealHeartTileOffset(contestant);
}
}
}
pitchMod = newNumHearts;
onSecondLine = FALSE;
// Check if wrapping to second line of hearts
if (newNumHearts > 7)
{
onSecondLine = TRUE;
newNumHearts -= 8;
}
ContestBG_FillBoxWithTile(0, heartOffset, newNumHearts + 22, gContestantTurnOrder[contestant] * 5 + 2 + onSecondLine, 1, 1, 17);
if (heartsDelta > 0)
{
PlaySE(SE_CONTEST_HEART);
m4aMPlayImmInit(&gMPlayInfo_SE1);
m4aMPlayPitchControl(&gMPlayInfo_SE1, 0xFFFF, pitchMod * 256);
}
else
{
PlaySE(SE_BOO);
}
if (!onSecondLine && newNumHearts == 0 && heartOffset == 0)
gTasks[taskId].tHeartsSign = -gTasks[taskId].tHeartsSign;
}
}
static void CreateSliderHeartSprites(void)
{
s32 i;
LoadSpriteSheet(&sSpriteSheet_SliderHeart);
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u8 y = sSliderHeartYPositions[gContestantTurnOrder[i]];
eContestGfxState[i].sliderHeartSpriteId = CreateSprite(&sSpriteTemplate_SliderHeart, 180, y, 1);
}
}
#define sContestant data[0]
#define sTargetX data[1]
#define sMoveX data[2]
static void UpdateHeartSlider(u8 contestant)
{
u8 spriteId;
s16 slideTarget;
eContestGfxState[contestant].sliderUpdating = TRUE;
spriteId = eContestGfxState[contestant].sliderHeartSpriteId;
slideTarget = eContestantStatus[contestant].pointTotal / 10 * 2;
if (slideTarget > 56)
slideTarget = 56;
else if (slideTarget < 0)
slideTarget = 0;
gSprites[spriteId].invisible = FALSE;
gSprites[spriteId].sContestant = contestant;
gSprites[spriteId].sTargetX = slideTarget;
if (gSprites[spriteId].sTargetX > gSprites[spriteId].x2)
gSprites[spriteId].sMoveX = 1;
else
gSprites[spriteId].sMoveX = -1;
gSprites[spriteId].callback = SpriteCB_UpdateHeartSlider;
}
static void UpdateHeartSliders(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
UpdateHeartSlider(i);
}
static bool8 SlidersDoneUpdating(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (eContestGfxState[i].sliderUpdating)
break;
}
if (i == CONTESTANT_COUNT)
return TRUE;
else
return FALSE;
}
static void SpriteCB_UpdateHeartSlider(struct Sprite *sprite)
{
if (sprite->x2 == sprite->sTargetX)
{
eContestGfxState[sprite->sContestant].sliderUpdating = FALSE;
sprite->callback = SpriteCallbackDummy;
}
else
{
sprite->x2 += sprite->sMoveX;
}
}
#undef sContestant
#undef sTargetX
#undef sMoveX
// Y positions change as the contestants change order
static void UpdateSliderHeartSpriteYPositions(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
gSprites[eContestGfxState[i].sliderHeartSpriteId].y = sSliderHeartYPositions[gContestantTurnOrder[i]];
}
// Used to hide (or subsequently reshow) the bottom two slider hearts that get hidden by text windows by moving them offscreen
static void SetBottomSliderHeartsInvisibility(bool8 invisible)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
// Skip the top two contestants sliders
if (gContestantTurnOrder[i] > 1)
{
if (!invisible)
gSprites[eContestGfxState[i].sliderHeartSpriteId].x = 180;
else
gSprites[eContestGfxState[i].sliderHeartSpriteId].x = 256;
}
}
}
static void CreateNextTurnSprites(void)
{
s32 i;
LoadSpritePalette(&sSpritePalette_NextTurn);
for (i = 0; i < CONTESTANT_COUNT; i++)
{
LoadCompressedSpriteSheet(&sSpriteSheet_NextTurn[i]);
eContestGfxState[i].nextTurnSpriteId = CreateSprite(&sSpriteTemplates_NextTurn[i],
204,
sNextTurnSpriteYPositions[gContestantTurnOrder[i]],
0);
SetSubspriteTables(&gSprites[eContestGfxState[i].nextTurnSpriteId], sSubspriteTable_NextTurn);
gSprites[eContestGfxState[i].nextTurnSpriteId].invisible = TRUE;
}
}
static void CreateApplauseMeterSprite(void)
{
u8 spriteId;
LoadCompressedSpriteSheet(&sSpriteSheet_ApplauseMeter);
LoadSpritePalette(&sSpritePalette_ApplauseMeter);
spriteId = CreateSprite(&sSpriteTemplate_ApplauseMeter, 30, 44, 1);
gSprites[spriteId].invisible = TRUE;
eContest.applauseMeterSpriteId = spriteId;
}
static void CreateJudgeAttentionEyeTask(void)
{
u8 i;
u8 taskId = CreateTask(Task_FlashJudgeAttentionEye, 30);
eContest.judgeAttentionTaskId = taskId;
for (i = 0; i < CONTESTANT_COUNT; i++)
gTasks[taskId].data[i * 4] = 0xFF;
}
static void StartFlashJudgeAttentionEye(u8 contestant)
{
gTasks[eContest.judgeAttentionTaskId].data[contestant * 4 + 0] = 0;
gTasks[eContest.judgeAttentionTaskId].data[contestant * 4 + 1] = 0;
}
static void StopFlashJudgeAttentionEye(u8 contestant)
{
u8 taskId = CreateTask(Task_StopFlashJudgeAttentionEye, 31);
gTasks[taskId].data[0] = contestant;
}
static void Task_StopFlashJudgeAttentionEye(u8 taskId)
{
u8 contestant = gTasks[taskId].data[0];
if (gTasks[eContest.judgeAttentionTaskId].data[contestant * 4 + 0] == 0
|| gTasks[eContest.judgeAttentionTaskId].data[contestant * 4 + 0] == 0xFF)
{
gTasks[eContest.judgeAttentionTaskId].data[contestant * 4 + 0] = 0xFF;
gTasks[eContest.judgeAttentionTaskId].data[contestant * 4 + 1] = 0;
BlendPalette((eContest.prevTurnOrder[contestant] + 5) * 16 + 6, 2, 0, RGB(31, 31, 18));
DestroyTask(taskId);
}
}
static void Task_FlashJudgeAttentionEye(u8 taskId)
{
u8 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u8 offset = i * 4;
if (gTasks[taskId].data[offset + 0] != 0xFF)
{
if (gTasks[taskId].data[offset + 1] == 0)
gTasks[taskId].data[offset + 0]++;
else
gTasks[taskId].data[offset + 0]--;
if (gTasks[taskId].data[offset + 0] == 16
|| gTasks[taskId].data[offset + 0] == 0)
gTasks[taskId].data[offset + 1] ^= 1;
BlendPalette((eContest.prevTurnOrder[i] + 5) * 16 + 6, 2, gTasks[taskId].data[offset + 0], RGB(31, 31, 18));
}
}
}
// Note: While the below task is run for the entire Appeals portion of the contest,
// because data[i * 4] is always 0xFF it never does anything
// If turned on by setting that data between 0 and 16, it blends
// an odd selection of palette colors (e.g. the text box, the appeal hearts
// for only one contestant, the heart outlines in the move selection box, etc)
// Given the similarities, it's possible this was an incorrect attempt
// at something similar to what CreateJudgeAttentionEyeTask does
static void CreateUnusedBlendTask(void)
{
s32 i;
eContest.blendTaskId = CreateTask(Task_UnusedBlend, 30);
for (i = 0; i < CONTESTANT_COUNT; i++)
InitUnusedBlendTaskData(i);
}
static void InitUnusedBlendTaskData(u8 contestant)
{
gTasks[eContest.blendTaskId].data[contestant * 4] = 0xFF;
gTasks[eContest.blendTaskId].data[contestant * 4 + 1] = 0;
}
static void UpdateBlendTaskContestantsData(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
UpdateBlendTaskContestantData(i);
}
static void UpdateBlendTaskContestantData(u8 contestant)
{
u32 palOffset1;
u32 palOffset2;
InitUnusedBlendTaskData(contestant);
palOffset1 = contestant + 5;
DmaCopy16Defvars(3,
gPlttBufferUnfaded + palOffset1 * 16 + 10,
gPlttBufferFaded + palOffset1 * 16 + 10,
2);
palOffset2 = (contestant + 5) * 16 + 12 + contestant;
DmaCopy16Defvars(3,
gPlttBufferUnfaded + palOffset2,
gPlttBufferFaded + palOffset2,
2);
}
// See comments on CreateUnusedBlendTask
static void Task_UnusedBlend(u8 taskId)
{
u8 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u8 idx = i * 4;
// Below is never true
if (gTasks[taskId].data[idx] != 0xFF)
{
if (++gTasks[taskId].data[idx + 2] > 2)
{
gTasks[taskId].data[idx + 2] = 0;
if (gTasks[taskId].data[idx + 1] == 0)
gTasks[taskId].data[idx]++;
else
gTasks[taskId].data[idx]--;
if (gTasks[taskId].data[idx] == 16
|| gTasks[taskId].data[idx] == 0)
gTasks[taskId].data[idx + 1] ^= 1;
BlendPalette((i + 5) * 16 + 10, 1, gTasks[taskId].data[idx + 0], RGB(31, 31, 18));
BlendPalette((i + 5) * 16 + 12 + i, 1, gTasks[taskId].data[idx + 0], RGB(31, 31, 18));
}
}
}
}
static void StartStopFlashJudgeAttentionEye(u8 contestant)
{
if (eContestantStatus[contestant].hasJudgesAttention)
StartFlashJudgeAttentionEye(contestant);
else
StopFlashJudgeAttentionEye(contestant);
}
static u8 CreateContestantBoxBlinkSprites(u8 contestant)
{
u8 spriteId1, spriteId2;
u8 x = gContestantTurnOrder[contestant] * 40 + 32;
LoadCompressedSpriteSheet(&sSpriteSheets_ContestantsTurnBlinkEffect[contestant]);
LoadSpritePalette(&sSpritePalettes_ContestantsTurnBlinkEffect[contestant]);
spriteId1 = CreateSprite(&sSpriteTemplates_ContestantsTurnBlinkEffect[contestant], 184, x, 29);
spriteId2 = CreateSprite(&sSpriteTemplates_ContestantsTurnBlinkEffect[contestant], 248, x, 29);
gSprites[spriteId2].oam.tileNum += 64;
CopySpriteTiles(0,
3,
(void *)VRAM,
(u16 *)(BG_SCREEN_ADDR(28) + gContestantTurnOrder[contestant] * 5 * 64 + 0x26),
gContestResources->boxBlinkTiles1);
CopySpriteTiles(0,
3, (void *)VRAM,
(u16 *)(BG_SCREEN_ADDR(28) + gContestantTurnOrder[contestant] * 5 * 64 + 0x36),
gContestResources->boxBlinkTiles2);
CpuFill32(0, gContestResources->boxBlinkTiles1 + 0x500, 0x300);
CpuFill32(0, gContestResources->boxBlinkTiles2 + 0x500, 0x300);
RequestDma3Copy(gContestResources->boxBlinkTiles1,
(u8 *)(OBJ_VRAM0 + gSprites[spriteId1].oam.tileNum * 32),
0x800,
1);
RequestDma3Copy(gContestResources->boxBlinkTiles2,
(u8 *)(OBJ_VRAM0 + gSprites[spriteId2].oam.tileNum * 32),
0x800,
1);
gSprites[spriteId1].data[0] = spriteId2;
gSprites[spriteId2].data[0] = spriteId1;
gSprites[spriteId1].data[1] = contestant;
gSprites[spriteId2].data[1] = contestant;
return spriteId1;
}
static void DestroyContestantBoxBlinkSprites(u8 spriteId)
{
u8 spriteId2 = gSprites[spriteId].data[0];
FreeSpriteOamMatrix(&gSprites[spriteId2]);
DestroySprite(&gSprites[spriteId2]);
DestroySpriteAndFreeResources(&gSprites[spriteId]);
}
static void SetBlendForContestantBoxBlink(void)
{
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_TGT2_ALL | BLDCNT_EFFECT_BLEND);
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(7, 9));
}
static void ResetBlendForContestantBoxBlink(void)
{
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
}
// To indicate whose turn is up
static void BlinkContestantBox(u8 spriteId, bool8 b)
{
u8 spriteId2;
SetBlendForContestantBoxBlink();
eContestGfxState[gSprites[spriteId].data[1]].boxBlinking = TRUE;
spriteId2 = gSprites[spriteId].data[0];
StartSpriteAffineAnim(&gSprites[spriteId], 1);
StartSpriteAffineAnim(&gSprites[spriteId2], 1);
gSprites[spriteId].callback = SpriteCB_BlinkContestantBox;
gSprites[spriteId2].callback = SpriteCallbackDummy;
if (b == FALSE)
PlaySE(SE_CONTEST_MONS_TURN);
else
PlaySE(SE_PC_LOGIN);
}
static void SpriteCB_BlinkContestantBox(struct Sprite *sprite)
{
if (sprite->affineAnimEnded)
{
u8 spriteId2 = sprite->data[0];
if (gSprites[spriteId2].affineAnimEnded)
{
sprite->invisible = TRUE;
gSprites[spriteId2].invisible = TRUE;
sprite->callback = SpriteCB_EndBlinkContestantBox;
}
}
}
static void SpriteCB_EndBlinkContestantBox(struct Sprite *sprite)
{
eContestGfxState[sprite->data[1]].boxBlinking = FALSE;
DestroyContestantBoxBlinkSprites(sprite->data[0]);
ResetBlendForContestantBoxBlink();
}
// Unused.
static void ContestDebugTogglePointTotal(void)
{
if(eContestDebugMode == CONTEST_DEBUG_MODE_PRINT_POINT_TOTAL)
eContestDebugMode = CONTEST_DEBUG_MODE_OFF;
else
eContestDebugMode = CONTEST_DEBUG_MODE_PRINT_POINT_TOTAL;
if(eContestDebugMode == CONTEST_DEBUG_MODE_OFF)
{
DrawContestantWindowText();
SwapMoveDescAndContestTilemaps();
}
else
{
ContestDebugDoPrint();
}
}
static void ContestDebugDoPrint(void)
{
u8 i;
s16 value;
u8 *txtPtr;
u8 text[8];
if (!gEnableContestDebugging)
return;
switch (eContestDebugMode)
{
case CONTEST_DEBUG_MODE_OFF:
break;
case CONTEST_DEBUG_MODE_PRINT_WINNER_FLAGS:
case CONTEST_DEBUG_MODE_PRINT_LOSER_FLAGS:
ContestDebugPrintBitStrings();
break;
// The only other possible value is 1, which is only set by ContestDebugTogglePointTotal.
//
// case CONTEST_DEBUG_MODE_PRINT_POINT_TOTAL:
default:
for (i = 0; i < CONTESTANT_COUNT; i++)
FillWindowPixelBuffer(i, PIXEL_FILL(0));
for (i = 0; i < CONTESTANT_COUNT; i++)
{
value = eContestantStatus[i].pointTotal;
txtPtr = text;
if (eContestantStatus[i].pointTotal < 0)
{
value *= -1;
txtPtr = StringCopy(txtPtr, gText_OneDash);
}
ConvertIntToDecimalStringN(txtPtr, value, STR_CONV_MODE_LEFT_ALIGN, 4);
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text, 55, 1, 7);
}
for (i = 0; i < CONTESTANT_COUNT; i++)
{
value = eContestantStatus[i].appeal;
txtPtr = text;
if (eContestantStatus[i].appeal < 0)
{
value *= -1;
txtPtr = StringCopy(txtPtr, gText_OneDash);
}
ConvertIntToDecimalStringN(txtPtr, value, STR_CONV_MODE_LEFT_ALIGN, 4);
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text, 5, 1, 7);
}
SwapMoveDescAndContestTilemaps();
break;
}
}
void SortContestants(bool8 useRanking)
{
u8 scratch[CONTESTANT_COUNT];
u16 randomOrdering[CONTESTANT_COUNT] = {0};
s32 i;
s32 v3;
// Generate a unique random number for each contestant.
for (i = 0; i < CONTESTANT_COUNT; i++)
{
s32 j;
randomOrdering[i] = Random();
// Loop through all the numbers generated so far.
for (j = 0; j < i; j++)
{
if (randomOrdering[i] == randomOrdering[j])
{
// This number isn't unique; try generating again.
i--;
break;
}
}
}
if (!useRanking)
{
// Order based on the results of the Conditions round using Insertion Sort.
// Use the randomOrdering to break ties.
for (i = 0; i < CONTESTANT_COUNT; i++)
{
// Append this contestant to the list.
gContestantTurnOrder[i] = i;
// Determine where the contestant should be ordered.
for (v3 = 0; v3 < i; v3++)
{
if (gContestMonRound1Points[gContestantTurnOrder[v3]] < gContestMonRound1Points[i]
|| (gContestMonRound1Points[gContestantTurnOrder[v3]] == gContestMonRound1Points[i] && randomOrdering[gContestantTurnOrder[v3]] < randomOrdering[i]))
{
// Shift everything larger up to make room.
s32 j;
for (j = i; j > v3; j--)
gContestantTurnOrder[j] = gContestantTurnOrder[j - 1];
// Insert into the new spot.
gContestantTurnOrder[v3] = i;
break;
}
}
// This is redundant.
// Perhaps GF switched from true insertion sort to in-place insertion sort, and forgot to
// remove this check?
if (v3 == i)
gContestantTurnOrder[i] = i;
}
// Invert gContestantTurnOrder; above, it was a list of contestant IDs. Now it's a list of turn orderings.
//
// For example, if contestant 3 had the first turn, then `gContestantTurnOrder[1] = 3`. The turn is the index,
// the contestant is the data. After inverting the list, `gContestantTurnOrder[3] = 1`. The contestant is the index,
// and the turn is the data.
memcpy(scratch, gContestantTurnOrder, sizeof(scratch));
for (i = 0; i < CONTESTANT_COUNT; i++)
gContestantTurnOrder[scratch[i]] = i;
}
else
{
// Order contestants based on their ranking.
// If contestants have tied ranking, fill in the next available slot.
//
// Note that ranking is calculated so that shared places still take up a ranking
// space. A ranking like [1, 2, 2, 3] is not possible; it would be [1, 2, 2, 4]
// instead.
memset(scratch, 0xFF, sizeof(scratch));
for (i = 0; i < CONTESTANT_COUNT; i++)
{
u8 j = eContestantStatus[i].ranking;
while (1)
{
u8 *ptr = &scratch[j];
if (*ptr == 0xFF)
{
*ptr = i;
gContestantTurnOrder[i] = j;
break;
}
j++;
}
}
// Randomize the order of contestants with tied rankings using Selection Sort.
//
// Look through the array for tied ranks, and use randomOrdering to break the tie.
// This ensures that contestants with the same rank will be randomly ordered. This
// uses an in-place slection sort, which involves a lot of extra swapping.
for (i = 0; i < CONTESTANT_COUNT - 1; i++)
{
for (v3 = CONTESTANT_COUNT - 1; v3 > i; v3--)
{
if (eContestantStatus[v3 - 1].ranking == eContestantStatus[v3].ranking
&& gContestantTurnOrder[v3 - 1] < gContestantTurnOrder[v3]
&& randomOrdering[v3 - 1] < randomOrdering[v3])
{
u8 temp = gContestantTurnOrder[v3];
gContestantTurnOrder[v3] = gContestantTurnOrder[v3 - 1];
gContestantTurnOrder[v3 - 1] = temp;
}
}
}
}
}
static void DrawContestantWindows(void)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
s32 windowId = i + 5;
LoadPalette(eContestTempSave.cachedWindowPalettes[windowId], (gContestantTurnOrder[i] + 5) * 16, sizeof(eContestTempSave.cachedWindowPalettes[0]));
}
DrawContestantWindowText();
}
static void CalculateAppealMoveImpact(u8 contestant)
{
u16 move;
u8 effect;
u8 rnd;
s32 i;
eContestantStatus[contestant].appeal = 0;
eContestantStatus[contestant].baseAppeal = 0;
if (!ContestantCanUseTurn(contestant))
return;
move = eContestantStatus[contestant].currMove;
effect = gContestMoves[move].effect;
eContestantStatus[contestant].moveCategory = gContestMoves[eContestantStatus[contestant].currMove].contestCategory;
if (eContestantStatus[contestant].currMove == eContestantStatus[contestant].prevMove && eContestantStatus[contestant].currMove != MOVE_NONE)
{
eContestantStatus[contestant].repeatedMove = TRUE;
eContestantStatus[contestant].moveRepeatCount++;
}
else
{
eContestantStatus[contestant].moveRepeatCount = 0;
}
eContestantStatus[contestant].baseAppeal = gContestEffects[effect].appeal;
eContestantStatus[contestant].appeal = eContestantStatus[contestant].baseAppeal;
eContestAppealResults.jam = gContestEffects[effect].jam;
eContestAppealResults.jam2 = eContestAppealResults.jam;
eContestAppealResults.contestant = contestant;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestantStatus[i].jam = 0;
eContestAppealResults.unnervedPokes[i] = 0;
}
if (eContestantStatus[contestant].hasJudgesAttention
&& !AreMovesContestCombo(eContestantStatus[contestant].prevMove, eContestantStatus[contestant].currMove))
eContestantStatus[contestant].hasJudgesAttention = FALSE;
gContestEffectFuncs[effect]();
if (eContestantStatus[contestant].conditionMod == CONDITION_GAIN)
eContestantStatus[contestant].appeal += eContestantStatus[contestant].condition - 10;
else if (eContestantStatus[contestant].appealTripleCondition)
eContestantStatus[contestant].appeal += eContestantStatus[contestant].condition * 3;
else
eContestantStatus[contestant].appeal += eContestantStatus[contestant].condition;
eContestantStatus[contestant].completedCombo = FALSE;
eContestantStatus[contestant].usedComboMove = FALSE;
if (IsContestantAllowedToCombo(contestant))
{
bool8 completedCombo = AreMovesContestCombo(eContestantStatus[contestant].prevMove, eContestantStatus[contestant].currMove);
if (completedCombo && eContestantStatus[contestant].hasJudgesAttention)
{
eContestantStatus[contestant].completedCombo = completedCombo;
eContestantStatus[contestant].usedComboMove = TRUE;
eContestantStatus[contestant].hasJudgesAttention = FALSE;
eContestantStatus[contestant].comboAppealBonus = eContestantStatus[contestant].baseAppeal * eContestantStatus[contestant].completedCombo;
eContestantStatus[contestant].completedComboFlag = TRUE; // Redundant with completedCombo, used by AI
}
else
{
if (gContestMoves[eContestantStatus[contestant].currMove].comboStarterId != 0)
{
eContestantStatus[contestant].hasJudgesAttention = TRUE;
eContestantStatus[contestant].usedComboMove = TRUE;
}
else
{
eContestantStatus[contestant].hasJudgesAttention = FALSE;
}
}
}
if (eContestantStatus[contestant].repeatedMove)
eContestantStatus[contestant].repeatJam = (eContestantStatus[contestant].moveRepeatCount + 1) * 10;
if (eContestantStatus[contestant].nervous)
{
eContestantStatus[contestant].hasJudgesAttention = FALSE;
eContestantStatus[contestant].appeal = 0;
eContestantStatus[contestant].baseAppeal = 0;
}
eContestExcitement.moveExcitement = Contest_GetMoveExcitement(eContestantStatus[contestant].currMove);
if (eContestantStatus[contestant].overrideCategoryExcitementMod)
eContestExcitement.moveExcitement = 1;
if (eContestExcitement.moveExcitement > 0)
{
if (eContest.applauseLevel + eContestExcitement.moveExcitement > 4)
eContestExcitement.excitementAppealBonus = 60;
else
eContestExcitement.excitementAppealBonus = 10;
}
else
{
eContestExcitement.excitementAppealBonus = 0;
}
// Transform and Role Play require a visible target mon
// so randomly choose a contestant to be the "target"
rnd = Random() % (CONTESTANT_COUNT - 1);
for (i = 0; i < CONTESTANT_COUNT; i++)
{
// Target can't be the attacker
if (i != contestant)
{
if (rnd == 0)
break;
rnd--;
}
}
eContestantStatus[contestant].contestantAnimTarget = i;
}
void SetContestantEffectStringID(u8 a, u8 b)
{
eContestantStatus[a].effectStringId = b;
}
void SetContestantEffectStringID2(u8 a, u8 b)
{
eContestantStatus[a].effectStringId2 = b;
}
void SetStartledString(u8 contestant, u8 jam)
{
if (jam >= 60)
SetContestantEffectStringID(contestant, CONTEST_STRING_TRIPPED_OVER);
else if (jam >= 40)
SetContestantEffectStringID(contestant, CONTEST_STRING_LEAPT_UP);
else if (jam >= 30)
SetContestantEffectStringID(contestant, CONTEST_STRING_UTTER_CRY);
else if (jam >= 20)
SetContestantEffectStringID(contestant, CONTEST_STRING_TURNED_BACK);
else if (jam >= 10)
SetContestantEffectStringID(contestant, CONTEST_STRING_LOOKED_DOWN);
}
static void PrintAppealMoveResultText(u8 contestant, u8 stringId)
{
StringCopy(gStringVar1, gContestMons[contestant].nickname);
StringCopy(gStringVar2, gMoveNames[eContestantStatus[contestant].currMove]);
if (gContestMoves[eContestantStatus[eContestAppealResults.contestant].currMove].contestCategory == CONTEST_CATEGORY_COOL)
StringCopy(gStringVar3, gText_Contest_Shyness);
else if (gContestMoves[eContestantStatus[eContestAppealResults.contestant].currMove].contestCategory == CONTEST_CATEGORY_BEAUTY)
StringCopy(gStringVar3, gText_Contest_Anxiety);
else if (gContestMoves[eContestantStatus[eContestAppealResults.contestant].currMove].contestCategory == CONTEST_CATEGORY_CUTE)
StringCopy(gStringVar3, gText_Contest_Laziness);
else if (gContestMoves[eContestantStatus[eContestAppealResults.contestant].currMove].contestCategory == CONTEST_CATEGORY_SMART)
StringCopy(gStringVar3, gText_Contest_Hesitancy);
else
StringCopy(gStringVar3, gText_Contest_Fear);
StringExpandPlaceholders(gStringVar4, sAppealResultTexts[stringId]);
ContestClearGeneralTextWindow();
Contest_StartTextPrinter(gStringVar4, 1);
}
void MakeContestantNervous(u8 p)
{
eContestantStatus[p].nervous = TRUE;
eContestantStatus[p].currMove = MOVE_NONE;
}
// This function calculates the new turn order for the next round. The
// algorithm first checks for explicit turn assignments in the
// ContestantStatus::nextTurnOrder field of each contestant. The remaining
// turns are assigned such that the turn order will reverse.
//
// For example, if no pokemon have a defined nextTurnOrder, then the 4th
// will become 1st, the 3rd will become 2nd, etc.
//
// Note: This function assumes that multiple pokemon cannot have the same
// nextTurnOrder value.
static void ApplyNextTurnOrder(void)
{
u8 nextContestant = 0;
s32 i;
s32 j;
u8 newTurnOrder[CONTESTANT_COUNT];
bool8 isContestantOrdered[CONTESTANT_COUNT];
// Copy the current turn order.
for (i = 0; i < CONTESTANT_COUNT; i++)
{
newTurnOrder[i] = gContestantTurnOrder[i];
isContestantOrdered[i] = FALSE;
}
// For each turn, assign a contestant to that turn.
for (i = 0; i < CONTESTANT_COUNT; i++)
{
// Look for explicit turn assignments.
for (j = 0; j < CONTESTANT_COUNT; j++)
{
if (eContestantStatus[j].nextTurnOrder == i)
{
newTurnOrder[j] = i;
isContestantOrdered[j] = TRUE;
break;
}
}
if (j == CONTESTANT_COUNT)
{
// No contestant was assigned to this turn. Look for the unassigned contestant
// with the highest turn order.
//
// First, look for the first unassigned contestant.
for (j = 0; j < CONTESTANT_COUNT; j++)
{
if (!isContestantOrdered[j] && eContestantStatus[j].nextTurnOrder == 0xFF)
{
nextContestant = j;
j++;
break;
}
}
// Then, look for a better candidate, with a higher turn order.
for (; j < CONTESTANT_COUNT; j++)
{
if (!isContestantOrdered[j] && eContestantStatus[j].nextTurnOrder == 0xFF
&& gContestantTurnOrder[nextContestant] > gContestantTurnOrder[j])
nextContestant = j;
}
// Assign the contestant to this turn.
newTurnOrder[nextContestant] = i;
isContestantOrdered[nextContestant] = TRUE;
}
}
for (i = 0; i < CONTESTANT_COUNT; i++)
{
eContestAppealResults.turnOrder[i] = newTurnOrder[i];
eContestantStatus[i].nextTurnOrder = 0xFF;
eContestantStatus[i].turnOrderMod = 0;
gContestantTurnOrder[i] = newTurnOrder[i];
}
}
static void SpriteCB_JudgeSpeechBubble(struct Sprite *sprite)
{
if (sprite->data[1]++ > 84)
{
sprite->data[1] = 0;
sprite->invisible = TRUE;
sprite->callback = SpriteCallbackDummy;
eContest.waitForJudgeSpeechBubble = FALSE;
}
}
static void DoJudgeSpeechBubble(u8 symbolId)
{
u8 spriteId = eContest.judgeSpeechBubbleSpriteId;
switch (symbolId)
{
case JUDGE_SYMBOL_SWIRL:
case JUDGE_SYMBOL_SWIRL_UNUSED:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0];
PlaySE(SE_FAILURE);
break;
case JUDGE_SYMBOL_ONE_EXCLAMATION:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 4;
PlaySE(SE_SUCCESS);
break;
case JUDGE_SYMBOL_TWO_EXCLAMATIONS:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 8;
PlaySE(SE_SUCCESS);
break;
case JUDGE_SYMBOL_NUMBER_ONE_UNUSED: // Identical to JUDGE_SYMBOL_NUMBER_ONE
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 12;
PlaySE(SE_WARP_IN);
break;
case JUDGE_SYMBOL_NUMBER_ONE:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 12;
PlaySE(SE_WARP_IN);
break;
case JUDGE_SYMBOL_NUMBER_FOUR:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 16;
PlaySE(SE_WARP_IN);
break;
case JUDGE_SYMBOL_STAR:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 24;
PlaySE(SE_M_HEAL_BELL);
break;
case JUDGE_SYMBOL_QUESTION_MARK:
default:
gSprites[spriteId].oam.tileNum = gSprites[spriteId].data[0] + 20;
PlaySE(SE_WARP_IN);
break;
}
gSprites[spriteId].data[1] = 0;
gSprites[spriteId].invisible = FALSE;
gSprites[spriteId].callback = SpriteCB_JudgeSpeechBubble;
eContest.waitForJudgeSpeechBubble = TRUE;
}
static void UpdateApplauseMeter(void)
{
s32 i;
for (i = 0; i < APPLAUSE_METER_SIZE; i++)
{
const u8 *src;
if (i < eContest.applauseLevel)
src = &gContestApplauseMeterGfx[64];
else
src = gContestApplauseMeterGfx;
CpuCopy32(src, (void *)(OBJ_VRAM0 + (gSprites[eContest.applauseMeterSpriteId].oam.tileNum + 17 + i) * 32), 32);
CpuCopy32(src + 32, (void *)(OBJ_VRAM0 + (gSprites[eContest.applauseMeterSpriteId].oam.tileNum + 25 + i) * 32), 32);
if (eContest.applauseLevel > 4)
StartApplauseOverflowAnimation();
}
}
s8 Contest_GetMoveExcitement(u16 move)
{
return gContestExcitementTable[gSpecialVar_ContestCategory][gContestMoves[move].contestCategory];
}
static u8 StartApplauseOverflowAnimation(void)
{
u8 taskId = CreateTask(Task_ApplauseOverflowAnimation, 10);
gTasks[taskId].data[1] = 1;
gTasks[taskId].data[2] = IndexOfSpritePaletteTag(TAG_APPLAUSE_METER);
return taskId;
}
static void Task_ApplauseOverflowAnimation(u8 taskId)
{
// Skip every other frame.
if (++gTasks[taskId].data[0] == 1)
{
gTasks[taskId].data[0] = 0;
// Alternate between normal colors and white.
if (gTasks[taskId].data[3] == 0)
gTasks[taskId].data[4]++;
else
gTasks[taskId].data[4]--;
BlendPalette(264 + gTasks[taskId].data[2] * 16, 1, gTasks[taskId].data[4], RGB_WHITE);
// At the maximum or minimum blending, switch directions.
if (gTasks[taskId].data[4] == 0 || gTasks[taskId].data[4] == 16)
{
gTasks[taskId].data[3] ^= 1;
// Continue the animation until the applause meter is cleared.
if (eContest.applauseLevel < 5)
{
BlendPalette(264 + gTasks[taskId].data[2] * 16, 1, 0, RGB_RED);
DestroyTask(taskId);
}
}
}
}
static void SlideApplauseMeterIn(void)
{
CreateTask(Task_SlideApplauseMeterIn, 10);
gSprites[eContest.applauseMeterSpriteId].x2 = -70;
gSprites[eContest.applauseMeterSpriteId].invisible = FALSE;
eContest.applauseMeterIsMoving = TRUE;
}
static void Task_SlideApplauseMeterIn(u8 taskId)
{
struct Sprite *sprite = &gSprites[eContest.applauseMeterSpriteId];
gTasks[taskId].data[10] += 1664;
sprite->x2 += gTasks[taskId].data[10] >> 8;
gTasks[taskId].data[10] = gTasks[taskId].data[10] & 0xFF;
if (sprite->x2 > 0)
sprite->x2 = 0;
if (sprite->x2 == 0)
{
eContest.applauseMeterIsMoving = FALSE;
DestroyTask(taskId);
}
}
static void SlideApplauseMeterOut(void)
{
if (gSprites[eContest.applauseMeterSpriteId].invisible == TRUE)
{
eContest.applauseMeterIsMoving = FALSE;
}
else
{
CreateTask(Task_SlideApplauseMeterOut, 10);
gSprites[eContest.applauseMeterSpriteId].x2 = 0;
eContest.applauseMeterIsMoving = TRUE;
}
}
static void Task_SlideApplauseMeterOut(u8 taskId)
{
struct Sprite *sprite = &gSprites[eContest.applauseMeterSpriteId];
gTasks[taskId].data[10] += 1664;
sprite->x2 -= gTasks[taskId].data[10] >> 8;
gTasks[taskId].data[10] = gTasks[taskId].data[10] & 0xFF;
if (sprite->x2 < -70)
sprite->x2 = -70;
if (sprite->x2 == -70)
{
sprite->invisible = TRUE;
eContest.applauseMeterIsMoving = FALSE;
DestroyTask(taskId);
}
}
static void ShowAndUpdateApplauseMeter(s8 unused)
{
u8 taskId = CreateTask(Task_ShowAndUpdateApplauseMeter, 5);
gTasks[taskId].data[0] = unused;
eContest.isShowingApplauseMeter = TRUE;
}
static void Task_ShowAndUpdateApplauseMeter(u8 taskId)
{
switch (gTasks[taskId].data[10])
{
case 0:
SlideApplauseMeterIn();
gTasks[taskId].data[10]++;
break;
case 1:
if (!eContest.applauseMeterIsMoving)
{
gTasks[taskId].data[10]++;
}
break;
case 2:
if (gTasks[taskId].data[11]++ > 20)
{
gTasks[taskId].data[11] = 0;
UpdateApplauseMeter();
eContest.isShowingApplauseMeter = FALSE;
DestroyTask(taskId);
}
break;
}
}
// Unused.
static void HideApplauseMeterNoAnim(void)
{
gSprites[eContest.applauseMeterSpriteId].x2 = 0;
gSprites[eContest.applauseMeterSpriteId].invisible = FALSE;
}
// Unused.
static void ShowApplauseMeterNoAnim(void)
{
gSprites[eContest.applauseMeterSpriteId].invisible = TRUE;
}
#define tDelay data[10]
#define tFrame data[11]
#define tCycles data[12]
static void AnimateAudience(void)
{
CreateTask(Task_AnimateAudience, 15);
eContest.animatingAudience = TRUE;
}
static void Task_AnimateAudience(u8 taskId)
{
if (gTasks[taskId].tDelay++ > 6)
{
gTasks[taskId].tDelay = 0;
if (gTasks[taskId].tFrame == 0)
{
RequestDma3Copy(eContestAudienceFrame2_Gfx, (void *)(BG_SCREEN_ADDR(4)), 0x1000, 1);
}
else
{
RequestDma3Copy(eUnzippedContestAudience_Gfx, (void *)(BG_SCREEN_ADDR(4)), 0x1000, 1);
gTasks[taskId].tCycles++;
}
gTasks[taskId].tFrame ^= 1;
if (gTasks[taskId].tCycles == 9)
{
eContest.animatingAudience = FALSE;
DestroyTask(taskId);
}
}
}
#undef tDelay
#undef tFrame
#undef tCycles
#define tBlendColor data[0]
#define tBlendCoeff data[1]
#define tBlendDir data[2]
#define tTargetBlendCoeff data[3]
#define tBlendDelay data[10]
static void BlendAudienceBackground(s8 excitementDir, s8 blendDir)
{
u8 taskId = CreateTask(Task_BlendAudienceBackground, 10);
u16 blendColor;
u8 blendCoeff;
u8 targetBlendCoeff;
if (excitementDir > 0)
{
blendColor = RGB(30, 27, 8);
if (blendDir > 0)
{
// Blend to yellow (amount depends on applause meter)
blendCoeff = 0;
targetBlendCoeff = eContest.applauseLevel * 3;
}
else
{
// Blend back to original
blendCoeff = eContest.applauseLevel * 3;
targetBlendCoeff = 0;
}
}
else
{
blendColor = RGB_BLACK;
if (blendDir > 0)
{
// Blend to black
blendCoeff = 0;
targetBlendCoeff = 12;
}
else
{
// Black back to original
blendCoeff = 12;
targetBlendCoeff = 0;
}
}
gTasks[taskId].tBlendColor = blendColor;
gTasks[taskId].tBlendCoeff = blendCoeff;
gTasks[taskId].tBlendDir = blendDir;
gTasks[taskId].tTargetBlendCoeff = targetBlendCoeff;
// Because this isn't set to TRUE here, the main task doesn't wait for the color blend
// Unclear if this was intentional or not (perhaps waiting added too much delay). In any case it does nothing now
eContest.waitForAudienceBlend = FALSE;
}
static void Task_BlendAudienceBackground(u8 taskId)
{
if (gTasks[taskId].tBlendDelay++ >= 0)
{
gTasks[taskId].tBlendDelay = 0;
if (gTasks[taskId].tBlendDir > 0)
gTasks[taskId].tBlendCoeff++;
else
gTasks[taskId].tBlendCoeff--;
BlendPalette(17, 1, gTasks[taskId].tBlendCoeff, gTasks[taskId].tBlendColor);
BlendPalette(26, 1, gTasks[taskId].tBlendCoeff, gTasks[taskId].tBlendColor);
if (gTasks[taskId].tBlendCoeff == gTasks[taskId].tTargetBlendCoeff)
{
DestroyTask(taskId);
eContest.waitForAudienceBlend = FALSE;
}
}
}
#undef tBlendColor
#undef tBlendCoeff
#undef tTargetBlendCoeff
#undef tBlendDelay
static void ShowHideNextTurnGfx(bool8 show)
{
s32 i;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (eContestantStatus[i].turnOrderMod != 0 && show)
{
CpuCopy32(GetTurnOrderNumberGfx(i), (void *)(OBJ_VRAM0 + (gSprites[eContestGfxState[i].nextTurnSpriteId].oam.tileNum + 6) * 32), 32);
gSprites[eContestGfxState[i].nextTurnSpriteId].y = sNextTurnSpriteYPositions[gContestantTurnOrder[i]];
gSprites[eContestGfxState[i].nextTurnSpriteId].invisible = FALSE;
}
else
{
gSprites[eContestGfxState[i].nextTurnSpriteId].invisible = TRUE;
}
}
}
static const u8 *GetTurnOrderNumberGfx(u8 contestant)
{
if (eContestantStatus[contestant].turnOrderMod != 1)
return gContestNextTurnRandomGfx;
else
return gContestNextTurnNumbersGfx + eContestantStatus[contestant].nextTurnOrder * 32;
}
static void DrawUnnervedSymbols(void)
{
s32 i = 0;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (eContestAppealResults.unnervedPokes[i] != 0 && !Contest_IsMonsTurnDisabled(i))
{
u32 contestantOffset = gContestantTurnOrder[i] * 5 + 2;
u16 symbolOffset = GetStatusSymbolTileOffset(STAT_SYMBOL_SWIRL);
ContestBG_FillBoxWithIncrementingTile(0, symbolOffset, 20, contestantOffset, 2, 1, 17, 1);
symbolOffset += 16;
ContestBG_FillBoxWithIncrementingTile(0, symbolOffset, 20, contestantOffset + 1, 2, 1, 17, 1);
PlaySE(SE_CONTEST_ICON_CHANGE);
}
}
}
bool8 IsContestantAllowedToCombo(u8 contestant)
{
if (eContestantStatus[contestant].repeatedMove || eContestantStatus[contestant].nervous)
return FALSE;
else
return TRUE;
}
static void SetBgForCurtainDrop(void)
{
s32 i;
u16 bg0Cnt, bg1Cnt, bg2Cnt;
bg1Cnt = GetGpuReg(REG_OFFSET_BG1CNT);
((vBgCnt *)&bg1Cnt)->priority = 0;
((vBgCnt *)&bg1Cnt)->screenSize = 2;
((vBgCnt *)&bg1Cnt)->areaOverflowMode = 0;
((vBgCnt *)&bg1Cnt)->charBaseBlock = 0;
SetGpuReg(REG_OFFSET_BG1CNT, bg1Cnt);
bg0Cnt = GetGpuReg(REG_OFFSET_BG0CNT);
bg2Cnt = GetGpuReg(REG_OFFSET_BG2CNT);
((vBgCnt *)&bg0Cnt)->priority = 1;
((vBgCnt *)&bg2Cnt)->priority = 1;
SetGpuReg(REG_OFFSET_BG0CNT, bg0Cnt);
SetGpuReg(REG_OFFSET_BG2CNT, bg2Cnt);
gBattle_BG1_X = DISPLAY_WIDTH;
gBattle_BG1_Y = DISPLAY_HEIGHT;
SetGpuReg(REG_OFFSET_BG1HOFS, gBattle_BG1_X);
SetGpuReg(REG_OFFSET_BG1VOFS, gBattle_BG1_Y);
CpuFill32(0, gContestResources->contestBgTilemaps[1], 0x1000);
CopyToBgTilemapBuffer(1, gUnknown_08C17980, 0, 0);
Contest_SetBgCopyFlags(1);
for (i = 0; i < CONTESTANT_COUNT; i++)
{
gSprites[eContestGfxState[i].sliderHeartSpriteId].oam.priority = 1;
gSprites[eContestGfxState[i].nextTurnSpriteId].oam.priority = 1;
}
}
static void UpdateContestantBoxOrder(void)
{
s32 i;
u16 bg1Cnt;
RequestDma3Fill(0,(void *)(BG_CHAR_ADDR(2)), 0x2000, 1);
CpuFill32(0, gContestResources->contestBgTilemaps[1], 0x1000);
Contest_SetBgCopyFlags(1);
bg1Cnt = GetGpuReg(REG_OFFSET_BG1CNT);
((vBgCnt *) &bg1Cnt)->priority = 1;
((vBgCnt *) &bg1Cnt)->screenSize = 0;
((vBgCnt *) &bg1Cnt)->areaOverflowMode = 0;
((vBgCnt *) &bg1Cnt)->charBaseBlock = 2;
SetGpuReg(REG_OFFSET_BG1CNT, bg1Cnt);
gBattle_BG1_X = 0;
gBattle_BG1_Y = 0;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
gSprites[eContestGfxState[i].sliderHeartSpriteId].oam.priority = 0;
gSprites[eContestGfxState[i].nextTurnSpriteId].oam.priority = 0;
}
}
static void Task_StartDropCurtainAtRoundEnd(u8 taskId)
{
gBattle_BG1_X = 0;
gBattle_BG1_Y = DISPLAY_HEIGHT;
PlaySE12WithPanning(SE_CONTEST_CURTAIN_FALL, 0);
gTasks[taskId].func = Task_UpdateCurtainDropAtRoundEnd;
}
static void Task_UpdateCurtainDropAtRoundEnd(u8 taskId)
{
if ((s16)(gBattle_BG1_Y -= 7) < 0)
gBattle_BG1_Y = 0;
if (gBattle_BG1_Y == 0)
{
gTasks[taskId].data[0] = 0;
gTasks[taskId].data[1] = 0;
gTasks[taskId].data[2] = 0;
gTasks[taskId].func = Task_ResetForNextRound;
}
}
static void Task_ResetForNextRound(u8 taskId)
{
s32 i;
switch (gTasks[taskId].data[0])
{
case 0:
for (i = 0; i < CONTESTANT_COUNT; i++)
eContest.prevTurnOrder[i] = gContestantTurnOrder[i];
FillContestantWindowBgs();
UpdateBlendTaskContestantsData();
DrawConditionStars();
DrawContestantWindows();
ShowHideNextTurnGfx(TRUE);
UpdateSliderHeartSpriteYPositions();
gTasks[taskId].data[0] = 1;
break;
case 1:
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
{
u8 taskId2;
eContest.waitForLink = TRUE;
if (IsPlayerLinkLeader())
SetContestantStatusesForNextRound();
taskId2 = CreateTask(Task_LinkContest_CommunicateAppealsState, 0);
SetTaskFuncWithFollowupFunc(taskId2, Task_LinkContest_CommunicateAppealsState, Task_EndWaitForLink);
ContestPrintLinkStandby();
gTasks[taskId].data[0] = 2;
}
else
{
SetContestantStatusesForNextRound();
gTasks[taskId].data[0] = 3;
}
break;
case 2:
if (!eContest.waitForLink)
gTasks[taskId].data[0] = 3;
break;
case 3:
DrawStatusSymbols();
SwapMoveDescAndContestTilemaps();
gTasks[taskId].data[0] = 0;
gTasks[taskId].func = Task_WaitRaiseCurtainAtRoundEnd;
break;
}
}
static void Task_UpdateRaiseCurtainAtRoundEnd(u8 taskId)
{
if ((s16)(gBattle_BG1_Y += 7) > DISPLAY_HEIGHT)
gTasks[taskId].func = Task_UpdateContestantBoxOrder;
}
static void Task_WaitRaiseCurtainAtRoundEnd(u8 taskId)
{
if (gTasks[taskId].data[2] < 10)
{
gTasks[taskId].data[2]++;
}
else
{
if (gTasks[taskId].data[1] == 0)
{
if (gTasks[taskId].data[0] == 16)
gTasks[taskId].data[1]++;
else
gTasks[taskId].data[0]++;
}
else
{
if (gTasks[taskId].data[0] == 0)
{
gTasks[taskId].data[1] = 0;
gTasks[taskId].data[2] = 0;
gTasks[taskId].func = Task_StartRaiseCurtainAtRoundEnd;
}
else
{
gTasks[taskId].data[0]--;
}
}
}
}
static void Task_StartRaiseCurtainAtRoundEnd(u8 taskId)
{
if (gTasks[taskId].data[2] < 10)
{
gTasks[taskId].data[2]++;
}
else
{
gTasks[taskId].data[2] = 0;
PlaySE12WithPanning(SE_CONTEST_CURTAIN_RISE, 0);
gTasks[taskId].func = Task_UpdateRaiseCurtainAtRoundEnd;
}
}
#define tAnimId data[0]
static void AnimateSliderHearts(u8 animId)
{
s32 i;
u8 taskId;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
gSprites[eContestGfxState[i].sliderHeartSpriteId].oam.matrixNum = AllocOamMatrix();
gSprites[eContestGfxState[i].sliderHeartSpriteId].oam.affineMode = ST_OAM_AFFINE_NORMAL;
StartSpriteAffineAnim(&gSprites[eContestGfxState[i].sliderHeartSpriteId], animId);
if (animId == SLIDER_HEART_ANIM_APPEAR)
{
AnimateSprite(&gSprites[eContestGfxState[i].sliderHeartSpriteId]);
gSprites[eContestGfxState[i].sliderHeartSpriteId].invisible = FALSE;
}
}
taskId = CreateTask(Task_WaitForSliderHeartAnim, 5);
gTasks[taskId].tAnimId = animId;
eContest.sliderHeartsAnimating = TRUE;
}
static void Task_WaitForSliderHeartAnim(u8 taskId)
{
s32 i;
if (gSprites[eContestGfxState[0].sliderHeartSpriteId].affineAnimEnded)
{
if ((u8)gTasks[taskId].tAnimId == SLIDER_HEART_ANIM_DISAPPEAR)
{
for (i = 0; i < CONTESTANT_COUNT; i++)
gSprites[eContestGfxState[i].sliderHeartSpriteId].invisible = TRUE;
}
for (i = 0; i < CONTESTANT_COUNT; i++)
FreeSpriteOamMatrix(&gSprites[eContestGfxState[i].sliderHeartSpriteId]);
eContest.sliderHeartsAnimating = FALSE;
DestroyTask(taskId);
}
}
#undef tAnimId
static u16 SanitizeMove(u16 move)
{
if (move >= MOVES_COUNT)
move = MOVE_POUND;
return move;
}
static u16 SanitizeSpecies(u16 species)
{
if (species >= NUM_SPECIES)
species = SPECIES_NONE;
return species;
}
static void SetMoveSpecificAnimData(u8 contestant)
{
s32 i;
u16 move = SanitizeMove(eContestantStatus[contestant].currMove);
u16 species = SanitizeSpecies(gContestMons[contestant].species);
u8 targetContestant;
memset(&gContestResources->moveAnim->species, 0, 20);
ClearBattleAnimationVars();
for (i = 0; i < CONTESTANT_COUNT; i++)
gBattleMonForms[i] = 0;
switch (move)
{
case MOVE_CURSE:
if (gBaseStats[species].type1 == TYPE_GHOST || gBaseStats[species].type2 == TYPE_GHOST)
gAnimMoveTurn = 0;
else
gAnimMoveTurn = 1;
break;
case MOVE_TRANSFORM:
case MOVE_ROLE_PLAY:
targetContestant = eContestantStatus[contestant].contestantAnimTarget;
gContestResources->moveAnim->targetSpecies = SanitizeSpecies(gContestMons[targetContestant].species);
gContestResources->moveAnim->targetPersonality = gContestMons[targetContestant].personality;
gContestResources->moveAnim->hasTargetAnim = TRUE;
break;
case MOVE_RETURN:
gAnimFriendship = MAX_FRIENDSHIP;
break;
case MOVE_FRUSTRATION:
gAnimFriendship = 0;
break;
case MOVE_SOLAR_BEAM:
case MOVE_RAZOR_WIND:
case MOVE_SKULL_BASH:
case MOVE_SKY_ATTACK:
if (eContest.moveAnimTurnCount == 0)
{
eContest.moveAnimTurnCount = 2;
gAnimMoveTurn = 0;
}
else
{
gAnimMoveTurn = 1;
}
break;
}
SetBattleTargetSpritePosition();
}
static void ClearMoveAnimData(u8 contestant)
{
memset(gContestResources->moveAnim, 0, sizeof(struct ContestMoveAnimData));
if (eContest.moveAnimTurnCount != 0)
eContest.moveAnimTurnCount--;
}
static void SetMoveAnimAttackerData(u8 contestant)
{
gContestResources->moveAnim->contestant = contestant;
gContestResources->moveAnim->species = SanitizeSpecies(gContestMons[contestant].species);
gContestResources->moveAnim->personality = gContestMons[contestant].personality;
gContestResources->moveAnim->otId = gContestMons[contestant].otId;
}
static void CreateInvisibleBattleTargetSprite(void)
{
gBattlerSpriteIds[B_POSITION_OPPONENT_RIGHT] = CreateInvisibleSpriteWithCallback(SpriteCallbackDummy);
InitSpriteAffineAnim(&gSprites[gBattlerSpriteIds[gBattlerTarget]]);
SetBattleTargetSpritePosition();
}
static void SetBattleTargetSpritePosition(void)
{
struct Sprite *sprite = &gSprites[gBattlerSpriteIds[B_POSITION_OPPONENT_RIGHT]];
sprite->x2 = 0;
sprite->y2 = 0;
sprite->x = GetBattlerSpriteCoord(B_POSITION_OPPONENT_RIGHT, BATTLER_COORD_X);
sprite->y = GetBattlerSpriteCoord(B_POSITION_OPPONENT_RIGHT, BATTLER_COORD_Y);
sprite->invisible = TRUE;
}
static void SetMoveTargetPosition(u16 move)
{
switch (gBattleMoves[move].target)
{
case MOVE_TARGET_USER_OR_SELECTED:
case MOVE_TARGET_USER:
gBattlerTarget = B_POSITION_PLAYER_RIGHT;
break;
case MOVE_TARGET_SELECTED:
case MOVE_TARGET_RANDOM:
case MOVE_TARGET_BOTH:
case MOVE_TARGET_FOES_AND_ALLY:
default:
gBattlerTarget = B_POSITION_OPPONENT_RIGHT;
break;
}
}
static void Contest_PrintTextToBg0WindowStd(u32 windowId, const u8 *b)
{
struct TextPrinterTemplate printerTemplate;
printerTemplate.currentChar = b;
printerTemplate.windowId = windowId;
printerTemplate.fontId = 1;
printerTemplate.x = 0;
printerTemplate.y = 1;
printerTemplate.currentX = 0;
printerTemplate.currentY = 1;
printerTemplate.letterSpacing = 0;
printerTemplate.lineSpacing = 0;
printerTemplate.unk = 0;
printerTemplate.fgColor = 15;
printerTemplate.bgColor = 0;
printerTemplate.shadowColor = 8;
AddTextPrinter(&printerTemplate, 0, 0);
PutWindowTilemap(windowId);
Contest_SetBgCopyFlags(0);
}
void Contest_PrintTextToBg0WindowAt(u32 windowId, u8 *currChar, s32 x, s32 y, s32 fontId)
{
struct TextPrinterTemplate printerTemplate;
printerTemplate.currentChar = currChar;
printerTemplate.windowId = windowId;
printerTemplate.fontId = fontId;
printerTemplate.x = x;
printerTemplate.y = y;
printerTemplate.currentX = x;
printerTemplate.currentY = y;
printerTemplate.letterSpacing = 0;
printerTemplate.lineSpacing = 0;
printerTemplate.unk = 0;
printerTemplate.fgColor = 15;
printerTemplate.bgColor = 0;
printerTemplate.shadowColor = 8;
AddTextPrinter(&printerTemplate, 0, 0);
PutWindowTilemap(windowId);
Contest_SetBgCopyFlags(0);
}
static void Contest_StartTextPrinter(const u8 *currChar, bool32 b)
{
struct TextPrinterTemplate printerTemplate;
u8 speed;
printerTemplate.currentChar = currChar;
printerTemplate.windowId = 4;
printerTemplate.fontId = 1;
printerTemplate.x = 0;
printerTemplate.y = 1;
printerTemplate.currentX = 0;
printerTemplate.currentY = 1;
printerTemplate.letterSpacing = 0;
printerTemplate.lineSpacing = 0;
printerTemplate.unk = 0;
printerTemplate.fgColor = 1;
printerTemplate.bgColor = 0;
printerTemplate.shadowColor = 8;
if (!b)
{
AddTextPrinter(&printerTemplate, 0, 0);
}
else
{
if (gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
speed = 4;
else
speed = GetPlayerTextSpeedDelay();
AddTextPrinter(&printerTemplate, speed, 0);
}
PutWindowTilemap(4);
Contest_SetBgCopyFlags(0);
}
static void ContestBG_FillBoxWithIncrementingTile(u8 bg, u16 firstTileNum, u8 x, u8 y, u8 width, u8 height, u8 paletteSlot, s16 tileNumData)
{
WriteSequenceToBgTilemapBuffer(bg, firstTileNum, x, y, width, height, paletteSlot, tileNumData);
Contest_SetBgCopyFlags(bg);
}
static void ContestBG_FillBoxWithTile(u8 bg, u16 firstTileNum, u8 x, u8 y, u8 width, u8 height, u8 paletteSlot)
{
ContestBG_FillBoxWithIncrementingTile(bg, firstTileNum, x, y, width, height, paletteSlot, 0);
}
static bool32 Contest_RunTextPrinters(void)
{
RunTextPrinters();
return IsTextPrinterActive(4);
}
static void Contest_SetBgCopyFlags(u32 flagIndex)
{
sContestBgCopyFlags |= 1 << flagIndex;
}
void ResetContestLinkResults(void)
{
s32 i;
s32 j;
for(i = 0; i < CONTEST_CATEGORIES_COUNT; i++)
for(j = 0; j < CONTESTANT_COUNT; j++)
gSaveBlock2Ptr->contestLinkResults[i][j] = 0;
}
bool8 SaveContestWinner(u8 rank)
{
s32 i;
u8 captionId = Random() % NUM_PAINTING_CAPTIONS;
// Get the index of the winner among the contestants
for (i = 0; i < CONTESTANT_COUNT - 1; i++)
if (gContestFinalStandings[i] == 0)
break;
// Exit if attempting to save a Pokémon other than the player's to the museum
if (rank == CONTEST_SAVE_FOR_MUSEUM && i != gContestPlayerMonIndex)
return FALSE;
// Adjust the random painting caption depending on the category
switch (gSpecialVar_ContestCategory)
{
case CONTEST_CATEGORY_COOL:
captionId += NUM_PAINTING_CAPTIONS * CONTEST_CATEGORY_COOL;
break;
case CONTEST_CATEGORY_BEAUTY:
captionId += NUM_PAINTING_CAPTIONS * CONTEST_CATEGORY_BEAUTY;
break;
case CONTEST_CATEGORY_CUTE:
captionId += NUM_PAINTING_CAPTIONS * CONTEST_CATEGORY_CUTE;
break;
case CONTEST_CATEGORY_SMART:
captionId += NUM_PAINTING_CAPTIONS * CONTEST_CATEGORY_SMART;
break;
case CONTEST_CATEGORY_TOUGH:
captionId += NUM_PAINTING_CAPTIONS * CONTEST_CATEGORY_TOUGH;
break;
}
if (rank != CONTEST_SAVE_FOR_ARTIST)
{
// Save winner in the saveblock
// Used to save any winner for the Contest Hall or the Museum
// but excludes the temporary save used by the artist
u8 id = GetContestWinnerSaveIdx(rank, TRUE);
gSaveBlock1Ptr->contestWinners[id].personality = gContestMons[i].personality;
gSaveBlock1Ptr->contestWinners[id].species = gContestMons[i].species;
gSaveBlock1Ptr->contestWinners[id].trainerId = gContestMons[i].otId;
StringCopy(gSaveBlock1Ptr->contestWinners[id].monName, gContestMons[i].nickname);
StringCopy(gSaveBlock1Ptr->contestWinners[id].trainerName, gContestMons[i].trainerName);
if(gLinkContestFlags & LINK_CONTEST_FLAG_IS_LINK)
gSaveBlock1Ptr->contestWinners[id].contestRank = CONTEST_RANK_LINK;
else
gSaveBlock1Ptr->contestWinners[id].contestRank = gSpecialVar_ContestRank;
if (rank != CONTEST_SAVE_FOR_MUSEUM)
gSaveBlock1Ptr->contestWinners[id].contestCategory = gSpecialVar_ContestCategory;
else
gSaveBlock1Ptr->contestWinners[id].contestCategory = captionId;
}
else
{
// Set the most recent winner so the artist can show the player their painting
gCurContestWinner.personality = gContestMons[i].personality;
gCurContestWinner.trainerId = gContestMons[i].otId;
gCurContestWinner.species = gContestMons[i].species;
StringCopy(gCurContestWinner.monName, gContestMons[i].nickname);
StringCopy(gCurContestWinner.trainerName, gContestMons[i].trainerName);
gCurContestWinner.contestCategory = captionId;
}
return TRUE;
}
// Rank is either a regular contest rank (for saving winners to show in the Contest Hall)
// Or one of two special IDs listed below (for saving winners to show in Museum, or from the artist)
// If just retrieving the index where the winner *would* go, shift is FALSE
// If actually preparing to insert the winner into the saveblock, shift is TRUE
u8 GetContestWinnerSaveIdx(u8 rank, bool8 shift)
{
s32 i;
switch (rank)
{
case CONTEST_RANK_NORMAL:
case CONTEST_RANK_SUPER:
case CONTEST_RANK_HYPER:
case CONTEST_RANK_MASTER:
if (shift)
{
for (i = NUM_CONTEST_HALL_WINNERS - 1; i > 0; i--)
memcpy(&gSaveBlock1Ptr->contestWinners[i], &gSaveBlock1Ptr->contestWinners[i - 1], sizeof(struct ContestWinner));
}
return CONTEST_WINNER_HALL_1 - 1;
default:
// case CONTEST_SAVE_FOR_MUSEUM:
// case CONTEST_SAVE_FOR_ARTIST:
switch (gSpecialVar_ContestCategory)
{
case CONTEST_CATEGORY_COOL:
return CONTEST_WINNER_MUSEUM_COOL - 1;
case CONTEST_CATEGORY_BEAUTY:
return CONTEST_WINNER_MUSEUM_BEAUTY - 1;
case CONTEST_CATEGORY_CUTE:
return CONTEST_WINNER_MUSEUM_CUTE - 1;
case CONTEST_CATEGORY_SMART:
return CONTEST_WINNER_MUSEUM_SMART - 1;
case CONTEST_CATEGORY_TOUGH:
default:
return CONTEST_WINNER_MUSEUM_TOUGH - 1;
}
}
}
void ClearContestWinnerPicsInContestHall(void)
{
s32 i;
for (i = 0; i < MUSEUM_CONTEST_WINNERS_START; i++)
gSaveBlock1Ptr->contestWinners[i] = gDefaultContestWinners[i];
}
static void SetContestLiveUpdateFlags(u8 contestant)
{
s32 i;
if (!eContestExcitement.frozen
&& eContestExcitement.moveExcitement > 0
&& !eContestantStatus[contestant].repeatedMove)
{
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_EXCITING_APPEAL;
gContestResources->tv[contestant].madeExcitingAppeal = TRUE;
}
if (eContestantStatus[contestant].nervous)
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_GOT_NERVOUS;
if (!eContestExcitement.frozen
&& eContestExcitement.moveExcitement != 0
&& eContestExcitement.excitementAppealBonus == 60)
{
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_MAXED_EXCITEMENT;
}
if (eContestantStatus[contestant].usedComboMove
&& eContestantStatus[contestant].completedCombo)
{
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_USED_COMBO;
}
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (i != contestant && eContestantStatus[i].jam != 0)
{
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_STARTLED_OTHER;
gContestResources->tv[i].winnerFlags |= CONTESTLIVE_FLAG_GOT_STARTLED;
}
}
if (eContestantStatus[contestant].numTurnsSkipped != 0
|| eContestantStatus[contestant].noMoreTurns)
{
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_SKIPPED_TURN;
}
else if (!eContestantStatus[contestant].nervous)
{
gContestResources->tv[contestant].winnerFlags |= CONTESTLIVE_FLAG_MADE_APPEAL;
gContestResources->tv[contestant].madeAppeal = TRUE;
gContestResources->tv[contestant].appeals[eContest.appealNumber] = eContestantStatus[contestant].currMove;
}
if (eContestantStatus[contestant].repeatedMove)
gContestResources->tv[contestant].loserFlags |= CONTESTLIVE_FLAG_REPEATED_MOVE;
if (eContest.applauseLevel == 4
&& !eContestExcitement.frozen
&& eContestExcitement.moveExcitement < 0)
{
gContestResources->tv[contestant].loserFlags |= CONTESTLIVE_FLAG_MISSED_EXCITEMENT;
}
}
static void CalculateContestLiveUpdateData(void)
{
u8 loser;
s32 i, j;
bool32 notLastInRound1, notLastInRound2;
u16 appealMoves[CONTEST_NUM_APPEALS + 1];
u8 numMoveUses[CONTEST_NUM_APPEALS + 1];
u16 moveCandidates[CONTEST_NUM_APPEALS];
u8 winner;
u8 mostUses;
u8 numMoveCandidates;
loser = 0;
winner = 0;
// Get loser/winner ids
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (gContestFinalStandings[i] == 0)
winner = i;
else if (gContestFinalStandings[i] == CONTESTANT_COUNT - 1)
loser = i;
}
// Set flags for commenting on loser
gContestResources->tv[loser].loserFlags |= CONTESTLIVE_FLAG_LOST;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (i != winner && gContestMonTotalPoints[winner] - gContestMonTotalPoints[i] <= 50)
gContestResources->tv[i].loserFlags |= CONTESTLIVE_FLAG_LOST_SMALL_MARGIN;
if (!gContestResources->tv[i].madeExcitingAppeal)
gContestResources->tv[i].loserFlags |= CONTESTLIVE_FLAG_NO_EXCITEMENT;
for (j = 0; j < CONTESTANT_COUNT; j++)
{
if (gContestMonRound1Points[i] < gContestMonRound1Points[j])
break;
}
if (j == CONTESTANT_COUNT && gContestFinalStandings[i] != 0)
gContestResources->tv[i].loserFlags |= CONTESTLIVE_FLAG_BLEW_LEAD;
notLastInRound1 = FALSE;
notLastInRound2 = FALSE;
for (j = 0; j < CONTESTANT_COUNT; j++)
{
if (gContestMonRound1Points[i] > gContestMonRound1Points[j])
notLastInRound1 = TRUE;
if (gContestMonRound2Points[i] > gContestMonRound2Points[j])
notLastInRound2 = TRUE;
}
if (!notLastInRound1 && !notLastInRound2)
gContestResources->tv[i].loserFlags |= CONTESTLIVE_FLAG_LAST_BOTH_ROUNDS;
if (!gContestResources->tv[i].madeAppeal)
gContestResources->tv[i].loserFlags |= CONTESTLIVE_FLAG_NO_APPEALS;
}
// Get what moves the winner used and how many times they used them
for (i = 0; i < CONTEST_NUM_APPEALS; i++)
{
appealMoves[i] = MOVE_NONE;
numMoveUses[i] = 0;
}
appealMoves[CONTEST_NUM_APPEALS] = 0xFFFF;
numMoveUses[CONTEST_NUM_APPEALS] = 0;
for (i = 0; i < CONTEST_NUM_APPEALS; i++)
{
if (gContestResources->tv[winner].appeals[i] != MOVE_NONE)
{
for (j = 0; j < CONTEST_NUM_APPEALS; j++)
{
if (gContestResources->tv[winner].appeals[i] != appealMoves[j])
{
if (appealMoves[j] == MOVE_NONE)
{
appealMoves[j] = gContestResources->tv[winner].appeals[i];
numMoveUses[j]++;
}
}
else
{
numMoveUses[j]++;
}
}
}
}
// Choose an appeal move to comment on for the winner (most commonly used)
moveCandidates[0] = appealMoves[0];
mostUses = numMoveUses[0];
numMoveCandidates = 0;
for (i = 1; appealMoves[i] != 0xFFFF; i++)
{
if (mostUses < numMoveUses[i])
{
moveCandidates[0] = appealMoves[i];
mostUses = numMoveUses[i];
numMoveCandidates = 1;
}
else if (mostUses == numMoveUses[i])
{
moveCandidates[numMoveCandidates] = appealMoves[i];
numMoveCandidates++;
}
}
gContestResources->tv[winner].move = moveCandidates[Random() % numMoveCandidates];
}
static void SetConestLiveUpdateTVData(void)
{
s32 i;
u32 flags;
u8 winner;
u8 round1Placing, round2Placing;
u8 count;
u8 randAction;
u8 numLoserCandidates;
u8 flagId;
u16 winnerFlag;
u8 loserFlag;
u8 loser;
u8 loserCandidates[CONTESTANT_COUNT - 1];
// Players mon didn't win, don't generate show
if (gContestFinalStandings[gContestPlayerMonIndex] != 0)
return;
// Get winner id (unnecessary, we now know it's gContestPlayerMonIndex)
winner = 0;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (gContestFinalStandings[i] == 0)
winner = i;
}
// Get winner's placement in Round 1 and 2
round1Placing = 0;
round2Placing = 0;
for (i = 0; i < CONTESTANT_COUNT; i++)
{
if (gContestMonRound1Points[winner] < gContestMonRound1Points[i])
round1Placing++;
if (gContestMonRound2Points[winner] < gContestMonRound2Points[i])
round2Placing++;
}
// Count how many TV comment-worthy actions the winner took
flags = gContestResources->tv[winner].winnerFlags;
count = 0;
for (i = 0; i < 8; flags >>= 1, i++)
{
if (flags & 1)
count++;
}
// Randomly choose one of these actions to comment on
randAction = Random() % count;
flags = gContestResources->tv[winner].winnerFlags;
count = 0;
flagId = 0;
for (i = 0; i < 8; flags >>= 1, flagId++, i++)
{
if (!(flags & 1))
continue;
if (randAction == count)
break;
count++;
}
winnerFlag = 1 << flagId;
// Pick a losing player with the highest severity of bad actions to comment on
if (winner == 0)
{
loserCandidates[0] = 1;
loserFlag = gContestResources->tv[1].loserFlags;
i = 2;
}
else
{
loserCandidates[0] = 0;
loserFlag = gContestResources->tv[0].loserFlags;
i = 1;
}
numLoserCandidates = 1;
for (; i < CONTESTANT_COUNT; i++)
{
if (i != winner)
{
if (loserFlag < gContestResources->tv[i].loserFlags)
{
// Losing player currently has the worst (highest) set of flags, only candidate
loserCandidates[0] = i;
loserFlag = gContestResources->tv[i].loserFlags;
numLoserCandidates = 1;
}
else if (loserFlag == gContestResources->tv[i].loserFlags)
{
// Tie, increment number of loser candidates
loserCandidates[numLoserCandidates] = i;
numLoserCandidates++;
}
}
}
loser = loserCandidates[Random() % numLoserCandidates];
// Choose the "worst" action to comment on (flag with highest value)
flagId = CONTESTLIVE_FLAG_NO_APPEALS;
for (i = 0; i < 8; flagId >>= 1, i++)
{
loserFlag = gContestResources->tv[loser].loserFlags & flagId;
if (loserFlag)
break;
}
ContestLiveUpdates_Init(round1Placing);
ContestLiveUpdates_SetRound2Placing(round2Placing);
ContestLiveUpdates_SetWinnerAppealFlag(winnerFlag);
ContestLiveUpdates_SetWinnerMoveUsed(gContestResources->tv[winner].move);
ContestLiveUpdates_SetLoserData(loserFlag, loser);
}
// Unused
void ContestDebugToggleBitfields(bool8 loserFlags)
{
if (eContestDebugMode == CONTEST_DEBUG_MODE_OFF)
{
if (!loserFlags)
eContestDebugMode = CONTEST_DEBUG_MODE_PRINT_WINNER_FLAGS;
else
eContestDebugMode = CONTEST_DEBUG_MODE_PRINT_LOSER_FLAGS;
}
else
{
eContestDebugMode = CONTEST_DEBUG_MODE_OFF;
}
if (eContestDebugMode == CONTEST_DEBUG_MODE_OFF)
{
DrawContestantWindowText();
SwapMoveDescAndContestTilemaps();
}
else
{
ContestDebugPrintBitStrings();
}
}
static void ContestDebugPrintBitStrings(void)
{
u8 i;
s8 j;
u8 text1[20];
u8 text2[20];
u8 *txtPtr;
u32 bits;
if (!gEnableContestDebugging)
return;
if (eContestDebugMode != CONTEST_DEBUG_MODE_PRINT_WINNER_FLAGS && eContestDebugMode != CONTEST_DEBUG_MODE_PRINT_LOSER_FLAGS)
return;
for (i = 0; i < CONTESTANT_COUNT; i++)
FillWindowPixelBuffer(i, PIXEL_FILL(0));
if (eContestDebugMode == CONTEST_DEBUG_MODE_PRINT_WINNER_FLAGS)
{
for (i = 0; i < CONTESTANT_COUNT; i++)
{
txtPtr = StringCopy(text1, gText_CDot);
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text1, 5, 1, 7);
bits = gContestResources->tv[i].winnerFlags;
for (j = 7; j > -1; j--) // Weird loop.
{
txtPtr = ConvertIntToDecimalStringN(txtPtr, bits & 1, STR_CONV_MODE_LEFT_ALIGN, 1);
bits >>= 1;
}
for (j = 0; j < 5; j++)
text2[j] = text1[j];
text2[j] = EOS;
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text2, 5, 1, 7);
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text1 + j, 55, 1, 7);
}
}
else // CONTEST_DEBUG_MODE_PRINT_LOSER_FLAGS
{
for (i = 0; i < CONTESTANT_COUNT; i++)
{
StringCopy(text1, gText_BDot);
bits = gContestResources->tv[i].loserFlags;
txtPtr = &text1[2];
for (j = 7; j > -1; j--) // Weird loop.
{
txtPtr = ConvertIntToDecimalStringN(txtPtr, bits & 1, STR_CONV_MODE_LEFT_ALIGN, 1);
bits >>= 1;
}
for (j = 0; j < 5; j++)
text2[j] = text1[j];
text2[j] = EOS;
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text2, 5, 1, 7);
Contest_PrintTextToBg0WindowAt(gContestantTurnOrder[i], text1 + j, 55, 1, 7);
}
}
SwapMoveDescAndContestTilemaps();
}
static u8 GetMonNicknameLanguage(u8 *nickname)
{
u8 ret = GAME_LANGUAGE;
if (nickname[0] == EXT_CTRL_CODE_BEGIN && nickname[1] == EXT_CTRL_CODE_JPN)
return GAME_LANGUAGE;
if (StringLength(nickname) < PLAYER_NAME_LENGTH - 1)
{
while (*nickname != EOS)
{
if ((*nickname >= CHAR_A && *nickname <= CHAR_z)
|| (*nickname >= CHAR_0 && *nickname <= CHAR_9)
|| *nickname == CHAR_SPACE
|| *nickname == CHAR_PERIOD
|| *nickname == CHAR_COMMA
|| *nickname == CHAR_EXCL_MARK
|| *nickname == CHAR_QUESTION_MARK
|| *nickname == CHAR_MALE
|| *nickname == CHAR_FEMALE
|| *nickname == CHAR_SLASH
|| *nickname == CHAR_HYPHEN
|| *nickname == CHAR_ELLIPSIS
|| *nickname == CHAR_DBL_QUOT_LEFT
|| *nickname == CHAR_DBL_QUOT_RIGHT
|| *nickname == CHAR_SGL_QUOT_LEFT
|| *nickname == CHAR_DBL_QUOT_LEFT) // Most likely a typo, CHAR_SGL_QUOT_RIGHT should be here instead.
{
nickname++;
}
else
{
ret = LANGUAGE_JAPANESE;
break;
}
}
}
return ret;
}
static void StripPlayerNameForLinkContest(u8 *playerName)
{
u8 chr = playerName[5];
playerName[5] = EOS;
playerName[PLAYER_NAME_LENGTH] = chr;
}
static void StripMonNameForLinkContest(u8 *monName, s32 language)
{
u8 chr;
StripExtCtrlCodes(monName);
if (language == LANGUAGE_JAPANESE)
{
monName[5] = EOS;
monName[POKEMON_NAME_LENGTH] = EXT_CTRL_CODE_BEGIN;
}
else
{
chr = monName[5];
monName[5] = EOS;
monName[POKEMON_NAME_LENGTH] = chr;
}
}
void StripPlayerAndMonNamesForLinkContest(struct ContestPokemon *mon, s32 language)
{
u8 *name = mon->nickname;
if (language == LANGUAGE_JAPANESE)
{
ConvertInternationalString(name, GetMonNicknameLanguage(name));
}
else if (name[POKEMON_NAME_LENGTH] == EXT_CTRL_CODE_BEGIN)
{
ConvertInternationalString(name, LANGUAGE_JAPANESE);
}
else
{
name[5] = name[POKEMON_NAME_LENGTH];
name[POKEMON_NAME_LENGTH] = EOS;
}
name = mon->trainerName;
if (language == LANGUAGE_JAPANESE)
{
name[PLAYER_NAME_LENGTH] = EOS;
name[6] = name[4];
name[5] = name[3];
name[4] = name[2];
name[3] = name[1];
name[2] = mon->trainerName[0];
name[1] = EXT_CTRL_CODE_JPN;
name[0] = EXT_CTRL_CODE_BEGIN;
}
else
{
name[5] = name[PLAYER_NAME_LENGTH];
name[PLAYER_NAME_LENGTH] = EOS;
}
}