#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" #include "constants/tv.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 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 gUnknown_02039F5C = 0; EWRAM_DATA u8 gUnknown_02039F5D = 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(eUnknownHeap1A004.cachedWindowPalettes[contestantWindowId], 16 * (5 + gContestantTurnOrder[i]), sizeof((eUnknownHeap1A004.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], eUnknownHeap1A004.savedJunk, sizeof(eUnknownHeap1A004.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, eUnknownHeap1A004.cachedWindowPalettes, sizeof(eUnknownHeap1A004.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 <= 160) 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, eUnknownHeap1A004.unk18204, 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 = 160; gBattle_BG2_Y = 160; 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_ColorLightShadowDarkGrey); } 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, eUnknownHeap1A004.unk18604, PLTT_BUFFER_SIZE * 2); LoadPalette(eUnknownHeap1A004.unk18204, 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].pos2.x = 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->pos2.x != 0) { sprite->pos2.x -= 2; } else { if (++sprite->data[0] == 31) { sprite->data[0] = 0; sprite->callback = SpriteCallbackDummy; } } } static void SpriteCB_MonSlideOut(struct Sprite *sprite) { sprite->pos2.x -= 6; if (sprite->pos1.x + sprite->pos2.x < -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, eUnknownHeap1A004.unk18204, 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 = 160; 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 = gUnknown_082FF6C0; else gSprites[spriteId].affineAnims = gAffineAnims_BattleSpriteOpponentSide; StartSpriteAffineAnim(gSprites + spriteId, 0); 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); if (gContestEffects[gContestMoves[a].effect].appeal == 0xFF) numHearts = 0; else numHearts = gContestEffects[gContestMoves[a].effect].appeal / 10; if (numHearts > 8) numHearts = 8; // Filled-in hearts ContestBG_FillBoxWithTile(0, 0x5035, 0x15, 0x1f, 0x08, 0x01, 0x11); // Empty hearts ContestBG_FillBoxWithTile(0, 0x5012, 0x15, 0x1f, numHearts, 0x01, 0x11); if (gContestEffects[gContestMoves[a].effect].jam == 0xFF) numHearts = 0; else numHearts = gContestEffects[gContestMoves[a].effect].jam / 10; if (numHearts > 8) numHearts = 8; // Filled-in hearts ContestBG_FillBoxWithTile(0, 0x5036, 0x15, 0x20, 0x08, 0x01, 0x11); // Empty hearts ContestBG_FillBoxWithTile(0, 0x5014, 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].pos2.x) 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->pos2.x == sprite->sTargetX) { eContestGfxState[sprite->sContestant].sliderUpdating = FALSE; sprite->callback = SpriteCallbackDummy; } else { sprite->pos2.x += 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].pos1.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].pos1.x = 180; else gSprites[eContestGfxState[i].sliderHeartSpriteId].pos1.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(eUnknownHeap1A004.cachedWindowPalettes[windowId], (gContestantTurnOrder[i] + 5) * 16, sizeof(eUnknownHeap1A004.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].pos2.x = -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->pos2.x += gTasks[taskId].data[10] >> 8; gTasks[taskId].data[10] = gTasks[taskId].data[10] & 0xFF; if (sprite->pos2.x > 0) sprite->pos2.x = 0; if (sprite->pos2.x == 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].pos2.x = 0; eContest.applauseMeterIsMoving = TRUE; } } static void Task_SlideApplauseMeterOut(u8 taskId) { struct Sprite *sprite = &gSprites[eContest.applauseMeterSpriteId]; gTasks[taskId].data[10] += 1664; sprite->pos2.x -= gTasks[taskId].data[10] >> 8; gTasks[taskId].data[10] = gTasks[taskId].data[10] & 0xFF; if (sprite->pos2.x < -70) sprite->pos2.x = -70; if (sprite->pos2.x == -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].pos2.x = 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].pos1.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->pos2.x = 0; sprite->pos2.y = 0; sprite->pos1.x = GetBattlerSpriteCoord(B_POSITION_OPPONENT_RIGHT, BATTLER_COORD_X); sprite->pos1.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 sub_80DEDA8(u8 rank) { s32 i; u8 r7 = Random() % 3; for (i = 0; i < CONTESTANT_COUNT - 1; i++) { if (gContestFinalStandings[i] == 0) break; } if (rank == 0xFF && i != gContestPlayerMonIndex) return FALSE; switch (gSpecialVar_ContestCategory) { case CONTEST_CATEGORY_COOL: r7 += 0; break; case CONTEST_CATEGORY_BEAUTY: r7 += 3; break; case CONTEST_CATEGORY_CUTE: r7 += 6; break; case CONTEST_CATEGORY_SMART: r7 += 9; break; case CONTEST_CATEGORY_TOUGH: r7 += 12; break; } if (rank != 0xFE) { u8 id = sub_80DEFA8(rank, 1); 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 != 0xFF) gSaveBlock1Ptr->contestWinners[id].contestCategory = gSpecialVar_ContestCategory; else gSaveBlock1Ptr->contestWinners[id].contestCategory = r7; } else { 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 = r7; } return TRUE; } u8 sub_80DEFA8(u8 rank, u8 b) { s32 i; switch (rank) { case CONTEST_RANK_NORMAL: case CONTEST_RANK_SUPER: case CONTEST_RANK_HYPER: case CONTEST_RANK_MASTER: if (b != 0) { for (i = NUM_CONTEST_HALL_WINNERS - 1; i >= 1; i--) memcpy(&gSaveBlock1Ptr->contestWinners[i], &gSaveBlock1Ptr->contestWinners[i - 1], sizeof(struct ContestWinner)); } return 0; default: 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 < 8; 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; } }