pokeemerald/src/slot_machine.c
2019-06-26 16:42:57 -04:00

4699 lines
133 KiB
C

#include "global.h"
#include "constants/songs.h"
#include "overworld.h"
#include "field_effect.h"
#include "random.h"
#include "sound.h"
#include "main.h"
#include "slot_machine.h"
#include "string_util.h"
#include "decompress.h"
#include "trig.h"
#include "graphics.h"
#include "palette.h"
#include "util.h"
#include "text.h"
#include "menu.h"
#include "alloc.h"
#include "bg.h"
#include "gpu_regs.h"
#include "coins.h"
#include "tv.h"
#include "text_window.h"
#include "constants/rgb.h"
#include "main_menu.h"
#include "bg.h"
#include "window.h"
// Text
extern const u8 gText_YouDontHaveThreeCoins[];
extern const u8 gText_QuitTheGame[];
extern const u8 gText_YouveGot9999Coins[];
extern const u8 gText_YouveRunOutOfCoins[];
extern const u8 gText_ReelTimeHelp[];
enum
{
SLOT_MACHINE_TAG_7_RED,
SLOT_MACHINE_TAG_7_BLUE,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_REPLAY
};
enum
{
SLOT_MACHINE_MATCHED_1CHERRY,
SLOT_MACHINE_MATCHED_2CHERRY,
SLOT_MACHINE_MATCHED_REPLAY,
SLOT_MACHINE_MATCHED_LOTAD,
SLOT_MACHINE_MATCHED_AZURILL,
SLOT_MACHINE_MATCHED_POWER,
SLOT_MACHINE_MATCHED_777_MIXED,
SLOT_MACHINE_MATCHED_777_RED,
SLOT_MACHINE_MATCHED_777_BLUE,
SLOT_MACHINE_MATCHED_NONE
};
struct SlotMachineEwramStruct
{
/*0x00*/ u8 state;
/*0x01*/ u8 machineId;
/*0x02*/ u8 pikaPower;
/*0x03*/ u8 luckyGame;
/*0x04*/ u8 luckyFlags;
/*0x05*/ u8 reelTimeDraw;
/*0x06*/ u8 luckySpinsLeft; // tentative
/*0x07*/ u8 biasTag;
/*0x08*/ u16 matchedSymbols;
/*0x0A*/ u8 fairRollsLeft; // only happens if you win reeltime
/*0x0B*/ u8 fairRollsUsed;
/*0x0C*/ s16 coins;
/*0x0E*/ s16 payout;
/*0x10*/ s16 netCoinLoss; // coins lost to machine (but never goes below 0)
/*0x12*/ s16 bet;
/*0x14*/ s16 reeltimePixelOffset;
/*0x16*/ s16 reeltimePosition;
/*0x18*/ s16 currReel;
/*0x1A*/ s16 reelIncrement; // speed of reel
/*0x1C*/ s16 reelPixelOffsets[3];
/*0x22*/ u16 reelPixelOffsetsWhileStopping[3];
/*0x28*/ s16 reelTagOffsets[3];
/*0x2E*/ s16 reelExtraTurns[3];
/*0x34*/ s16 winnerRows[3];
/*0x3A*/ u8 slotReelTasks[3];
/*0x3D*/ u8 unkTaskPointer3D;
/*0x3E*/ u8 unkTaskPointer3E;
/*0x3F*/ u8 reelTimeSprite3F;
/*0x40*/ u8 unk40;
/*0x41*/ u8 unk41;
/*0x42*/ u8 unk42;
/*0x43*/ u8 unk43;
/*0x44*/ u8 unk44[5];
/*0x49*/ u8 unk49[2];
/*0x49*/ u8 unk4B[3];
/*0x4E*/ u8 unk4E[2];
/*0x50*/ u8 reelTimeSprites1[2];
/*0x52*/ u8 reelTimeSprites2[2];
/*0x54*/ u8 unk54[4];
/*0x58*/ u16 win0h;
/*0x5a*/ u16 win0v;
/*0x5c*/ u16 winIn;
/*0x5e*/ u16 winOut;
/*0x60*/ u16 backupMapMusic;
/*0x64*/ MainCallback prevMainCb;
};
struct UnkStruct1
{
/*0x00*/ u8 unk00;
/*0x01*/ u8 unk01;
/*0x02*/ s16 unk02;
};
#define SLOTMACHINE_GFX_TILES 233
static void CB2_SlotMachineSetup(void);
static void CB2_SlotMachineLoop(void);
static void PlaySlotMachine_Internal(u8 slotMachineIndex, MainCallback cb);
static void SlotMachineDummyTask(u8 taskId);
static void SlotMachineSetup_0_0(void);
static void SlotMachineSetup_6_2(void);
static void SlotMachineSetup_1_0(void);
static void SlotMachineSetup_2_0(void);
static void SlotMachineSetup_2_1(void);
static void SlotMachineSetup_0_1(void);
static void SlotMachineSetup_3_0(void);
static void SlotMachineSetup_4_0(void);
static void SlotMachineSetup_5_0(void);
static void SlotMachineSetup_6_0(void);
static void SlotMachineSetup_6_1(void);
static void SlotMachineSetup_8_0(void);
static void SlotMachineSetup_9_0(void);
static void SlotMachineSetup_10_0(void);
static void SlotMachineSetupGameplayTasks(void);
static void GameplayTasks_Slots(void);
static void sub_8104DA4(void);
static void RunSlotActions(u8 taskId);
static bool8 SlotAction_UnfadeScreen(struct Task *task);
static bool8 SlotAction_WaitForUnfade(struct Task *task);
static bool8 SlotAction_SetSlotMachineVars(struct Task *task);
static bool8 SlotAction3(struct Task *task);
static bool8 SlotAction4(struct Task *task);
static bool8 SlotAction_AwaitPlayerInput(struct Task *task);
static bool8 SlotAction_PrintYouDontHaveThreeCoins(struct Task *task);
static bool8 SlotAction_ExitYouDontHaveThreeCoinsMessage(struct Task *task);
static bool8 SlotAction_GivingInformation(struct Task *task);
static bool8 SlotAction9(struct Task *task);
static bool8 SlotAction10(struct Task *task);
static bool8 SlotAction_SetLuckySpins(struct Task *task);
static bool8 SlotAction_AwaitReelStop(struct Task *task);
static bool8 SlotAction_WaitForAllReelsToStop(struct Task *task);
static bool8 SlotAction_CheckMatches(struct Task *task);
static bool8 SlotAction_WaitForPayoutToBeAwarded(struct Task *task);
static bool8 SlotAction_EndOfRoll(struct Task *task);
static bool8 SlotAction_MatchedPower(struct Task *task);
static bool8 SlotAction18(struct Task *task);
static bool8 SlotAction_Loop(struct Task *task);
static bool8 SlotAction_NoMatches(struct Task *task);
static bool8 SlotAction_PrintQuitTheGame(struct Task *task);
static bool8 SlotAction_SeeIfPlayerQuits(struct Task *task);
static bool8 SlotAction_PrintMessage_9999Coins(struct Task *task);
static bool8 SlotAction_ExitMessage_9999Coins(struct Task *task);
static bool8 SlotAction_PrintMessage_NoMoreCoins(struct Task *task);
static bool8 SlotAction_ExitMessage_NoMoreCoins(struct Task *task);
static bool8 SlotAction_EndGame(struct Task *task);
static bool8 SlotAction_FreeDataStructures(struct Task *task);
static void DrawLuckyFlags(void);
static void SetLuckySpins(void);
static bool8 IsThisRoundLucky(void);
static u8 AttemptsAtLuckyFlags_Top3(void);
static u16 SlowReelSpeed(void);
static u8 AttemptsAtLuckyFlags_NotTop3(void);
static void CheckMatch(void);
static void CheckMatch_CenterRow(void);
static void CheckMatch_TopAndBottom(void);
static void CheckMatch_Diagonals(void);
static u8 GetMatchFromSymbolsInRow(u8 c1, u8 c2, u8 c3);
static void AwardPayout(void);
static void RunAwardPayoutActions(u8 taskId);
static bool8 IsFinalTask_RunAwardPayoutActions(void);
static bool8 AwardPayoutAction0(struct Task *task);
static bool8 AwardPayoutAction_GivePayoutToPlayer(struct Task *task);
static bool8 AwardPayoutAction_FreeTask(struct Task *task);
static u8 GetNearbyTag_Quantized(u8 x, s16 y);
static void GameplayTask_StopSlotReel(void);
static void ReelTasks_SetUnkTaskData(u8 a0);
static void sub_8102E1C(u8 a0);
static bool8 IsSlotReelMoving(u8 a0);
static void RunSlotReelActions(u8 taskId);
static bool8 SlotReelAction_StayStill(struct Task *task);
static bool8 SlotReelAction_Spin(struct Task *task);
static bool8 SlotReelAction_DecideWhereToStop(struct Task *task);
static bool8 SlotReelAction_MoveToStop(struct Task *task);
static bool8 SlotReelAction_OscillatingStop(struct Task *task);
static bool8 DecideReelTurns_BiasTag_Reel1(void);
static bool8 DecideReelTurns_BiasTag_Reel1_Bet1(u8 a0, u8 a1);
static bool8 DecideReelTurns_BiasTag_Reel1_Bet2or3(u8 a0, u8 a1);
static bool8 DecideReelTurns_BiasTag_Reel2(void);
static bool8 DecideReelTurns_BiasTag_Reel2_Bet1or2(void);
static bool8 DecideReelTurns_BiasTag_Reel2_Bet3(void);
static bool8 DecideReelTurns_BiasTag_Reel3(void);
static bool8 DecideReelTurns_BiasTag_Reel3_Bet1or2(u8 a0);
static bool8 DecideReelTurns_BiasTag_Reel3_Bet3(u8 a0);
static void DecideReelTurns_NoBiasTag_Reel1(void);
static void DecideReelTurns_NoBiasTag_Reel2(void);
static void DecideReelTurns_NoBiasTag_Reel2_Bet1(void);
static void DecideReelTurns_NoBiasTag_Reel2_Bet2(void);
static void DecideReelTurns_NoBiasTag_Reel2_Bet3(void);
static void DecideReelTurns_NoBiasTag_Reel3(void);
static void DecideReelTurns_NoBiasTag_Reel3_Bet1(void);
static void DecideReelTurns_NoBiasTag_Reel3_Bet2(void);
static void DecideReelTurns_NoBiasTag_Reel3_Bet3(void);
static void sub_8103C14(u8 a0);
static void sub_8103C48(u8 taskId);
static void LoadBetTiles(u8 a0);
static void sub_8103C78(struct Task *task, u8 taskId);
static void sub_8103CAC(struct Task *task, u8 taskId);
static void sub_8103CC8(struct Task *task, u8 taskId);
static void sub_8103D8C(u8 a0);
static void sub_8103DC8(void);
static void sub_8103E04(u8 a0);
static bool8 sub_8103E38(void);
static bool8 sub_8103E7C(void);
static bool8 sub_8103EAC(u8 spriteId);
static void sub_8103EE4(struct Sprite *sprite);
static void sub_8103F70(void);
static bool8 sub_8103FA0(void);
static void sub_8103FE8_(u8 taskId);
static void GameplayTask_PikaPower(void);
static void DisplayPikaPower(u8 pikaPower);
static bool8 sub_81040C8(void);
static void sub_81040E8(u8 taskId);
static void nullsub_68(struct Task *task);
static void sub_810411C(struct Task *task);
static void sub_8104144(struct Task *task);
static void sub_81041AC(struct Task *task);
static void ClearTaskDataFields_2orHigher(struct Task *task);
static void sub_810423C(u8 pikaPower);
static void BeginReeltime(void);
static bool8 IsFinalTask_RunReelTimeActions(void);
static void RunReeltimeActions(u8 taskId);
static void ReeltimeAction0(struct Task *task);
static void ReeltimeAction1(struct Task *task);
static void ReeltimeAction2(struct Task *task);
static void ReeltimeAction3(struct Task *task);
static void ReeltimeAction4(struct Task *task);
static void ReeltimeAction5(struct Task *task);
static void ReeltimeAction6(struct Task *task);
static void ReelTimeAction_LandOnOutcome(struct Task *task);
static void ReeltimeAction8(struct Task *task);
static void ReeltimeAction9(struct Task *task);
static void ReeltimeAction10(struct Task *task);
static void ReeltimeAction11(struct Task *task);
static void ReeltimeAction12(struct Task *task);
static void ReeltimeAction13(struct Task *task);
static void ReeltimeAction14(struct Task *task);
static void ReeltimeAction15(struct Task *task);
static void ReeltimeAction16(struct Task *task);
static void ReeltimeAction17(struct Task *task);
static void sub_8104A40(s16 a0, s16 a1);
static void sub_8104A88(s16 a0);
static void OpenInfoBox(u8 a0);
static bool8 IsInfoBoxClosed(void);
static void RunInfoBoxActions(u8 taskId);
static void InfoBox_FadeIn(struct Task *task);
static void InfoBox_WaitForFade(struct Task *task);
static void InfoBox_8104B80(struct Task *task);
static void InfoBox_AwaitPlayerInput(struct Task *task);
static void InfoBox_AddText(struct Task *task);
static void InfoBox_8104BFC(struct Task *task);
static void InfoBox_812DE14(struct Task *task);
static void InfoBox_812DE30(struct Task *task);
static void InfoBox_FreeTask(struct Task *task);
static void sub_8104C5C(void);
static void sub_8104CAC(u8 arg0);
static bool8 sub_8104E18(void);
static void nullsub_69(struct Task *task);
static void sub_8104E74_(u8 taskId);
static void sub_8104EA8(void);
static void sub_8104F8C(void);
static void sub_8104FF4(s16 x, s16 y, u8 a2, s16 a3);
static void sub_81050C4(void);
static void sub_8105100(void);
static void sub_810514C(void);
static void sub_81051C0(void);
static void sub_8105284_(void);
static void sub_81052EC(void);
static void sub_81053A0(void);
static void sub_810545C(void);
static void sub_81054B8(void);
static void sub_8105524(void);
static void sub_8105554(void);
static void CreateReelTimeSprites1(void);
static void sub_8105688(s16 a0);
static void sub_81056C0(void);
static void CreateReelTimeSprite2(void);
static void sub_81057E8(s16 a0);
static void sub_8105804(void);
static void sub_8105854(void);
static void sub_81058A0(void);
static void sub_81058C4(void);
static void sub_81059B8(void);
static void sub_81059E8(void);
static bool8 sub_8105ACC(void);
static void sub_8105AEC(void);
static u8 sub_8105B1C(s16 x, s16 y);
static void sub_8105B88(u8 spriteId);
static u8 sub_8105BF8(u8 templateIdx, void (*callback)(struct Sprite*), s16 x, s16 y, s16 a4);
static void sub_81063C0(void);
static void sub_8106404(void);
static void sub_8106448(void);
static void sub_81064B8(void);
static void sub_81065A8(s16 arg0, u16 arg1, u16 arg2, u16 arg3, u16 arg4);
static void sub_81065DC(void);
static void sub_812F958(void);
static void sub_812F968(void);
static void LoadSlotMachineWheelOverlay(void);
static u8 sub_8105BB4(u8 templateIdx, u8 cbAndCoordsIdx, s16 a2);
// Ewram variables
static EWRAM_DATA u16 *sUnknown_0203AAC8 = NULL;
static EWRAM_DATA u16 *sSelectedPikaPowerTile = NULL;
static EWRAM_DATA u16 *sUnknown_0203AAD0 = NULL;
static EWRAM_DATA u8 *sUnknown_0203AAD4 = NULL;
static EWRAM_DATA u8 *sUnknown_0203AAD8 = NULL;
static EWRAM_DATA u16 *sUnknown_0203AADC = NULL;
static EWRAM_DATA u8 *sUnknown_0203AAE0 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAE4 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAE8 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAEC = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAF0 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAF4 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAF8 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AAFC = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB00 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB04 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB08 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB0C = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB10 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB14 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB18 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB1C = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB20 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB24 = NULL;
static EWRAM_DATA struct SpriteFrameImage *sUnknown_0203AB28 = NULL;
static EWRAM_DATA struct SpriteSheet *sUnknown_0203AB2C = NULL;
static EWRAM_DATA struct SpriteSheet *sUnknown_0203AB30 = NULL;
static EWRAM_DATA struct SlotMachineEwramStruct *sSlotMachine = NULL;
// IWRAM bss
static struct SpriteFrameImage *gUnknown_03001188[26];
// Const rom data.
extern const struct UnkStruct1 *const gUnknown_083ED048[];
extern const u16 gPalette_83EDE24[];
extern const u8 gLuckyRoundProbabilities[][3];
extern const u8 gBiasTags[];
extern const u16 gLuckyFlagSettings_Top3[];
extern const u16 gLuckyFlagSettings_NotTop3[];
extern const s16 gUnknown_083ECE7E[][2];
extern const SpriteCallback gUnknown_083ECF0C[];
extern const struct SpriteTemplate *const gUnknown_083EDB5C[];
extern const struct SubspriteTable *const gUnknown_083EDBC4[];
extern const struct SpriteTemplate gSpriteTemplate_83ED6CC;
extern const struct SpriteTemplate gSpriteTemplate_83ED564;
extern const struct SpriteTemplate gSpriteTemplate_83ED54C;
extern const struct SpriteTemplate gSpriteTemplate_83ED534;
extern const u8 gUnknown_083ECC58[2];
extern const struct SpriteTemplate gSpriteTemplate_83ED51C;
extern const u16 gProbabilityTable_SkipToReeltimeAction14[];
extern const u16 *const gUnknown_083EDE10[];
extern const u16 gReelIncrementTable[][2];
extern const u16 gReelTimeBonusIncrementTable[];
extern const u16 gSlotMatchFlags[];
extern const u16 gSlotPayouts[];
extern const u8 *const gUnknown_083EDCE4;
extern const u8 *const gUnknown_083EDCDC;
extern const u32 gReelTimeGfx[];
extern const struct SpriteSheet gSlotMachineSpriteSheets[];
extern const struct SpritePalette gSlotMachineSpritePalettes[];
extern const u16 *const gUnknown_083EDE20;
extern const s16 gInitialReelPositions[][2];
extern const struct BgTemplate gUnknown_085A7424[4];
extern const struct WindowTemplate gUnknown_085A7434[];
extern const u8 gLuckyFlagProbabilities_Top3[][6];
extern const u8 gLuckyFlagProbabilities_NotTop3[][6];
extern const u8 gReeltimeProbabilities_UnluckyGame[][17];
extern const u8 gReelTimeProbabilities_LuckyGame[][17];
extern const u8 gSym2Match[];
extern const u8 gReelTimeTags[];
extern const u8 gReelSymbols[][REEL_NUM_TAGS];
extern const u16 *const gUnknown_083EDD08[];
extern const u16 *const gUnknown_083EDD1C[];
extern const u8 gUnknown_083EDD30[];
extern const u8 gBettingTilesId[][2];
extern const u8 gNumberBettingTiles[];
extern const u16 *const gUnknown_083EDDA0[];
extern const u16 *const gUnknown_083EDDAC;
extern const u16 gReelTimeWindowTilemap[];
extern const u16 gUnknown_085A9898[];
extern void (*const gUnknown_083ED064[])(void);
const struct BgTemplate gUnknown_085A7424[] =
{
{
.bg = 0,
.charBaseIndex = 2,
.mapBaseIndex = 31,
.screenSize = 0,
.paletteMode = 0,
.priority = 0,
.baseTile = 0
},
{
.bg = 1,
.charBaseIndex = 1,
.mapBaseIndex = 28,
.screenSize = 0,
.paletteMode = 0,
.priority = 1,
.baseTile = 0
},
{
.bg = 2,
.charBaseIndex = 1,
.mapBaseIndex = 29,
.screenSize = 0,
.paletteMode = 0,
.priority = 2,
.baseTile = 0
},
{
.bg = 3,
.charBaseIndex = 1,
.mapBaseIndex = 30,
.screenSize = 0,
.paletteMode = 0,
.priority = 1,
.baseTile = 0
},
};
const struct WindowTemplate gUnknown_085A7434[] =
{
{0, 2, 15, 0x1B, 4, 15, 0x194},
DUMMY_WIN_TEMPLATE
};
const struct WindowTemplate gUnknown_085A7444 =
{
0, 1, 3, 20, 13, 13, 1
};
const u8 gColors_ReeltimeHelp[] = {3, 1, 2, 0};
bool8 (*const SlotActions[])(struct Task *task) =
{
SlotAction_UnfadeScreen, // 0
SlotAction_WaitForUnfade, // 1
SlotAction_SetSlotMachineVars, // 2
SlotAction3, // 3
SlotAction4, // 4
SlotAction_AwaitPlayerInput, // 5
SlotAction_PrintYouDontHaveThreeCoins, // 6
SlotAction_ExitYouDontHaveThreeCoinsMessage, // 7
SlotAction_GivingInformation, // 8
SlotAction9, // 9
SlotAction10, // 10
SlotAction_SetLuckySpins, // 11
SlotAction_AwaitReelStop, // 12
SlotAction_WaitForAllReelsToStop, // 13
SlotAction_CheckMatches, // 14
SlotAction_WaitForPayoutToBeAwarded, // 15
SlotAction_EndOfRoll, // 16
SlotAction_MatchedPower, // 17
SlotAction18, // 18
SlotAction_Loop, // 19
SlotAction_NoMatches, // 20
SlotAction_PrintQuitTheGame, // 21
SlotAction_SeeIfPlayerQuits, // 22
SlotAction_PrintMessage_9999Coins, // 23
SlotAction_ExitMessage_9999Coins, // 24
SlotAction_PrintMessage_NoMoreCoins, // 25
SlotAction_ExitMessage_NoMoreCoins, // 26
SlotAction_EndGame, // 27
SlotAction_FreeDataStructures, // 28
};
bool8 (*const AwardPayoutActions[])(struct Task *task) =
{
AwardPayoutAction0,
AwardPayoutAction_GivePayoutToPlayer,
AwardPayoutAction_FreeTask
};
bool8 (*const SlotReelActions[])(struct Task *task) =
{
SlotReelAction_StayStill,
SlotReelAction_Spin,
SlotReelAction_DecideWhereToStop,
SlotReelAction_MoveToStop,
SlotReelAction_OscillatingStop
};
// returns True if a match with the biasTag is possible in that reel
// also modifies data in sSlotMachine reel arrays to indicate how to get to the matching state
bool8 (*const DecideReelTurns_BiasTag[])(void) =
{
DecideReelTurns_BiasTag_Reel1,
DecideReelTurns_BiasTag_Reel2,
DecideReelTurns_BiasTag_Reel3
};
void (*const DecideReelTurns_NoBiasTag[])(void) =
{
DecideReelTurns_NoBiasTag_Reel1,
DecideReelTurns_NoBiasTag_Reel2,
DecideReelTurns_NoBiasTag_Reel3
};
const u16 ReelStopShocks[] = {2, 4, 4, 4, 8};
bool8 (*const DecideReelTurns_BiasTag_Reel1_Bets[])(u8 tag1, u8 tag2) =
{
DecideReelTurns_BiasTag_Reel1_Bet1,
DecideReelTurns_BiasTag_Reel1_Bet2or3,
DecideReelTurns_BiasTag_Reel1_Bet2or3
};
bool8 (*const DecideReelTurns_BiasTag_Reel2_Bets[])(void) =
{
DecideReelTurns_BiasTag_Reel2_Bet1or2,
DecideReelTurns_BiasTag_Reel2_Bet1or2,
DecideReelTurns_BiasTag_Reel2_Bet3
};
bool8 (*const DecideReelTurns_BiasTag_Reel3_Bets[])(u8 biasTag) =
{
DecideReelTurns_BiasTag_Reel3_Bet1or2,
DecideReelTurns_BiasTag_Reel3_Bet1or2,
DecideReelTurns_BiasTag_Reel3_Bet3
};
void (*const DecideReelTurns_NoBiasTag_Reel2_Bets[])(void) =
{
DecideReelTurns_NoBiasTag_Reel2_Bet1,
DecideReelTurns_NoBiasTag_Reel2_Bet2,
DecideReelTurns_NoBiasTag_Reel2_Bet3
};
void (*const DecideReelTurns_NoBiasTag_Reel3_Bets[])(void) =
{
DecideReelTurns_NoBiasTag_Reel3_Bet1,
DecideReelTurns_NoBiasTag_Reel3_Bet2,
DecideReelTurns_NoBiasTag_Reel3_Bet3
};
void (*const gUnknown_083ECBA0[])(struct Task *task, u8 taskId) =
{
sub_8103C78,
sub_8103CAC,
sub_8103CC8
};
const s16 gUnknown_083ECBAC[] = {5, 10, 15};
void (*const gUnknown_083ECBB4[])(struct Task *task) =
{
nullsub_68,
sub_810411C,
sub_8104144,
sub_81041AC
};
const u16 pikaPowerTileTable[][2] =
{
{0x9e, 0x6e},
{0x9f, 0x6f},
{0xaf, 0x7f},
};
void (*const ReeltimeActions[])(struct Task *task) =
{
ReeltimeAction0,
ReeltimeAction1,
ReeltimeAction2,
ReeltimeAction3,
ReeltimeAction4,
ReeltimeAction5,
ReeltimeAction6,
ReelTimeAction_LandOnOutcome,
ReeltimeAction8,
ReeltimeAction9,
ReeltimeAction10,
ReeltimeAction11,
ReeltimeAction12,
ReeltimeAction13,
ReeltimeAction14,
ReeltimeAction15,
ReeltimeAction16,
ReeltimeAction10,
ReeltimeAction17
};
const u8 gUnknown_085A75C0[] = {1, 1, 2, 2};
const s16 gUnknown_085A75C4[] = {0x40, 0x30, 0x18, 0x08};
const s16 gUnknown_085A75CC[] = {10, 8, 6, 4};
void (*const InfoBoxActions[])(struct Task *task) =
{
InfoBox_FadeIn,
InfoBox_WaitForFade,
InfoBox_8104B80,
InfoBox_WaitForFade,
InfoBox_AddText,
InfoBox_WaitForFade,
InfoBox_AwaitPlayerInput,
InfoBox_WaitForFade,
InfoBox_812DE14,
InfoBox_WaitForFade,
InfoBox_812DE30,
InfoBox_WaitForFade,
InfoBox_8104BFC,
InfoBox_WaitForFade,
InfoBox_FreeTask,
};
void (*const gUnknown_083ECC54[])(struct Task *task) =
{
nullsub_69,
};
extern const struct SpriteTemplate gSpriteTemplate_83ED504;
extern const struct SpriteTemplate gSpriteTemplate_83ED4EC;
extern const struct SubspriteTable gSubspriteTables_83ED7D4[];
extern const struct SpriteTemplate gSpriteTemplate_83ED4D4;
extern const struct SubspriteTable gSubspriteTables_83ED7B4[];
extern const struct SpriteTemplate gSpriteTemplate_83ED4BC;
extern const struct SpriteTemplate gSpriteTemplate_83ED4A4;
extern const struct SubspriteTable gSubspriteTables_83ED78C[];
extern const struct SpriteTemplate gSpriteTemplate_83ED474;
extern const struct SpriteTemplate gSpriteTemplate_83ED48C;
extern const struct SubspriteTable gSubspriteTables_83ED73C[];
extern const struct SubspriteTable gSubspriteTables_83ED75C[];
extern const struct SpriteTemplate gSpriteTemplate_83ED444;
extern const struct SubspriteTable gSubspriteTables_83ED704[];
extern const struct SpriteTemplate gSpriteTemplate_83ED42C;
extern const struct SpriteTemplate gSpriteTemplate_83ED414;
extern const struct SpriteTemplate gSpriteTemplate_83ED45C;
// code
#define tState data[0]
static void Task_FadeToSlotMachine(u8 taskId)
{
switch (gTasks[taskId].tState)
{
case 0:
BeginNormalPaletteFade(0xFFFFFFFF, 0, 0, 0x10, RGB_BLACK);
gTasks[taskId].tState++;
break;
case 1:
if (!gPaletteFade.active)
{
SetMainCallback2(CB2_SlotMachineSetup);
DestroyTask(taskId);
}
break;
}
}
void PlaySlotMachine(u8 slotMachineIndex, MainCallback cb)
{
u8 taskId;
sSlotMachine = AllocZeroed(sizeof(*sSlotMachine));
PlaySlotMachine_Internal(slotMachineIndex, cb);
taskId = CreateTask(Task_FadeToSlotMachine, 0);
gTasks[taskId].tState = 0;
}
#undef tState
static void CB2_SlotMachineSetup(void)
{
switch (gMain.state)
{
case 0:
SlotMachineSetup_0_0();
SlotMachineSetup_0_1();
gMain.state++;
break;
case 1:
SlotMachineSetup_1_0();
gMain.state++;
break;
case 2:
SlotMachineSetup_2_0();
SlotMachineSetup_2_1();
gMain.state++;
break;
case 3:
SlotMachineSetup_3_0();
gMain.state++;
break;
case 4:
SlotMachineSetup_4_0();
gMain.state++;
break;
case 5:
SlotMachineSetup_5_0();
gMain.state++;
break;
case 6:
SlotMachineSetup_6_0();
gMain.state++;
break;
case 7:
BeginNormalPaletteFade(-1, 0, 0x10, 0, RGB_BLACK);
ShowBg(0);
ShowBg(1);
ShowBg(2);
ShowBg(3);
gMain.state++;
break;
case 8:
SlotMachineSetup_8_0();
gMain.state++;
break;
case 9:
SlotMachineSetup_9_0();
gMain.state++;
break;
case 10:
SlotMachineSetup_10_0();
SlotMachineSetupGameplayTasks();
gMain.state++;
break;
case 11:
SetMainCallback2(CB2_SlotMachineLoop);
break;
}
}
static void CB2_SlotMachineLoop(void)
{
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void SlotMachine_VBlankCallback(void)
{
LoadOam();
ProcessSpriteCopyRequests();
TransferPlttBuffer();
SetGpuReg(REG_OFFSET_WIN0H, sSlotMachine->win0h);
SetGpuReg(REG_OFFSET_WIN0V, sSlotMachine->win0v);
SetGpuReg(REG_OFFSET_WININ, sSlotMachine->winIn);
SetGpuReg(REG_OFFSET_WINOUT, sSlotMachine->winOut);
}
static void PlaySlotMachine_Internal(u8 slotMachineIndex, MainCallback cb)
{
struct Task *task = &gTasks[CreateTask(SlotMachineDummyTask, 0xFF)];
task->data[0] = slotMachineIndex;
StoreWordInTwoHalfwords(task->data + 1, (intptr_t)cb);
}
static void sub_81019EC(void)
{
struct Task *task = &gTasks[FindTaskIdByFunc(SlotMachineDummyTask)];
sSlotMachine->machineId = task->data[0];
LoadWordFromTwoHalfwords((u16 *)(task->data + 1), (u32 *)&sSlotMachine->prevMainCb);
}
static void SlotMachineDummyTask(u8 taskId)
{
}
static void SlotMachineSetup_0_0(void)
{
SetVBlankCallback(NULL);
SetHBlankCallback(NULL);
CpuFill32(0, (void *)VRAM, VRAM_SIZE);
ResetBgsAndClearDma3BusyFlags(0);
InitBgsFromTemplates(0, gUnknown_085A7424, ARRAY_COUNT(gUnknown_085A7424));
InitWindows(gUnknown_085A7434);
DeactivateAllTextPrinters();
}
static void SlotMachineSetup_6_0(void)
{
SetVBlankCallback(SlotMachine_VBlankCallback);
EnableInterrupts(INTR_FLAG_VBLANK);
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_OBJ_1D_MAP | DISPCNT_OBJ_ON | DISPCNT_WIN0_ON);
}
static void SlotMachineSetup_1_0(void)
{
DmaClearLarge16(3, (u16 *)(BG_VRAM), BG_VRAM_SIZE, 0x1000);
}
static void SlotMachineSetup_2_0(void)
{
DmaClear16(3, (u16 *)OAM, OAM_SIZE);
}
static void SlotMachineSetup_2_1(void)
{
SetGpuReg(REG_OFFSET_BG0CNT, 0);
SetGpuReg(REG_OFFSET_BG1CNT, 0);
SetGpuReg(REG_OFFSET_BG2CNT, 0);
SetGpuReg(REG_OFFSET_BG3CNT, 0);
SetGpuReg(REG_OFFSET_BG0HOFS, 0);
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
SetGpuReg(REG_OFFSET_BG2HOFS, 0);
SetGpuReg(REG_OFFSET_BG2VOFS, 0);
SetGpuReg(REG_OFFSET_BG3HOFS, 0);
SetGpuReg(REG_OFFSET_BG3VOFS, 0);
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG0 | WININ_WIN0_BG1 | WININ_WIN0_BG2 | WININ_WIN0_BG3 | WININ_WIN0_OBJ | WININ_WIN0_CLR);
SetGpuReg(REG_OFFSET_WINOUT, WINOUT_WIN01_BG0 | WINOUT_WIN01_BG1 | WINOUT_WIN01_BG2 | WINOUT_WIN01_BG3 | WINOUT_WIN01_OBJ | WINOUT_WIN01_CLR);
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_TGT1_BG3 | BLDCNT_EFFECT_BLEND | BLDCNT_TGT2_OBJ);
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(9, 8));
}
// set up initial state of slot machine
static void SlotMachineSetup_0_1(void)
{
u8 i;
sub_81019EC(); // assigns sSlotMachine->machineId, etc.
sSlotMachine->state = 0;
sSlotMachine->pikaPower = 0;
sSlotMachine->luckyGame = Random() & 1;
sSlotMachine->luckyFlags = 0;
sSlotMachine->matchedSymbols = 0;
sSlotMachine->fairRollsLeft = 0;
sSlotMachine->fairRollsUsed = 0;
sSlotMachine->coins = GetCoins();
sSlotMachine->payout = 0;
sSlotMachine->netCoinLoss = 0;
sSlotMachine->bet = 0;
sSlotMachine->currReel = 0;
sSlotMachine->reelIncrement = 8;
sSlotMachine->win0h = 0xf0;
sSlotMachine->win0v = 0xa0;
sSlotMachine->winIn = 0x3f;
sSlotMachine->winOut = 0x3f;
sSlotMachine->backupMapMusic = GetCurrentMapMusic();
for (i = 0; i < NUM_REELS; i++)
{
sSlotMachine->reelPixelOffsetsWhileStopping[i] = 0;
sSlotMachine->reelTagOffsets[i] = gInitialReelPositions[i][sSlotMachine->luckyGame] % REEL_NUM_TAGS;
sSlotMachine->reelPixelOffsets[i] = REEL_NUM_TAGS * REEL_TAG_HEIGHT - sSlotMachine->reelTagOffsets[i] * REEL_TAG_HEIGHT;
sSlotMachine->reelPixelOffsets[i] %= REEL_NUM_TAGS * REEL_TAG_HEIGHT;
}
AlertTVThatPlayerPlayedSlotMachine(GetCoins());
}
static void SlotMachineSetup_3_0(void)
{
ResetPaletteFade();
ResetSpriteData();
gOamLimit = 0x80;
FreeAllSpritePalettes();
ResetTasks();
}
static void SlotMachineSetup_4_0(void)
{
sSelectedPikaPowerTile = Alloc(8);
sUnknown_0203AAD0 = AllocZeroed(0xE);
sUnknown_0203AADC = AllocZeroed(8);
// several of these are 1 bit off from each other
sUnknown_0203AAD0[0] = 0x2051;
sUnknown_0203AAD0[1] = 0x2851;
sUnknown_0203AAD0[2] = 0x2061;
sUnknown_0203AAD0[3] = 0x2861;
sUnknown_0203AAD0[4] = 0x20BE;
sUnknown_0203AAD0[5] = 0x28BE;
sUnknown_0203AAD0[6] = 0x20BF;
}
static void SlotMachineSetup_5_0(void)
{
sub_8106448();
sub_81064B8();
sub_81063C0();
LoadMessageBoxGfx(0, 0x200, 0xF0);
LoadUserWindowBorderGfx(0, 0x214, 0xE0);
PutWindowTilemap(0);
}
static void SlotMachineSetup_10_0(void)
{
sub_8104EA8();
sub_8104F8C();
sub_8103DC8();
sub_81050C4();
}
// create gameplay tasks
static void SlotMachineSetupGameplayTasks(void)
{
GameplayTask_PikaPower();
GameplayTask_StopSlotReel();
sub_8104C5C();
GameplayTasks_Slots();
}
static void GameplayTasks_Slots(void)
{
RunSlotActions(CreateTask(RunSlotActions, 0));
}
// task->data[0] is a timer
static void RunSlotActions(u8 taskId)
{
while (SlotActions[sSlotMachine->state](&gTasks[taskId]))
;
}
static bool8 SlotAction_UnfadeScreen(struct Task *task)
{
BeginNormalPaletteFade(0xFFFFFFFF, 0, 16, 0, RGB(0, 0, 0));
sub_810423C(sSlotMachine->pikaPower);
sSlotMachine->state++;
return FALSE;
}
static bool8 SlotAction_WaitForUnfade(struct Task *task)
{
if (!gPaletteFade.active)
sSlotMachine->state++;
return FALSE;
}
static bool8 SlotAction_SetSlotMachineVars(struct Task *task)
{
sSlotMachine->payout = 0;
sSlotMachine->bet = 0;
sSlotMachine->currReel = 0;
sSlotMachine->luckyFlags &= (LUCKY_BIAS_777 | LUCKY_BIAS_MIXED_777);
sSlotMachine->state = 4;
if (sSlotMachine->coins <= 0)
{
sSlotMachine->state = 25;
}
else if (sSlotMachine->fairRollsLeft)
{
sSlotMachine->state = 3;
sub_8104CAC(4);
}
return TRUE;
}
static bool8 SlotAction3(struct Task *task)
{
if (sub_8104E18())
sSlotMachine->state = 4;
return FALSE;
}
static bool8 SlotAction4(struct Task *task)
{
sub_8104CAC(0);
sSlotMachine->state = 5;
if (sSlotMachine->coins >= 9999)
sSlotMachine->state = 23;
return TRUE;
}
static bool8 SlotAction_AwaitPlayerInput(struct Task *task)
{
s16 i;
if (gMain.newKeys & SELECT_BUTTON)
{
OpenInfoBox(0);
sSlotMachine->state = 8;
}
else if (gMain.newKeys & R_BUTTON) // bet the max amount
{
if (sSlotMachine->coins - (3 - sSlotMachine->bet) >= 0)
{
for (i = sSlotMachine->bet; i < 3; i++)
LoadBetTiles(i);
sSlotMachine->coins -= (3 - sSlotMachine->bet);
sSlotMachine->bet = 3;
sSlotMachine->state = 9;
PlaySE(SE_REGI);
}
else // you didn't have enough coins to bet the max
{
sSlotMachine->state = 6;
}
}
else
{
if (gMain.newKeys & DPAD_DOWN && sSlotMachine->coins != 0)
{
PlaySE(SE_REGI);
LoadBetTiles(sSlotMachine->bet);
sSlotMachine->coins--;
sSlotMachine->bet++;
}
// player maxed out or finished betting
if (sSlotMachine->bet >= 3 || (sSlotMachine->bet != 0 && gMain.newKeys & A_BUTTON))
sSlotMachine->state = 9;
// player wants to quit
if (gMain.newKeys & B_BUTTON)
sSlotMachine->state = 21;
}
return FALSE;
}
static bool8 SlotAction_PrintYouDontHaveThreeCoins(struct Task *task)
{
DrawDialogueFrame(0, 0);
AddTextPrinterParameterized(0, 1, gText_YouDontHaveThreeCoins, 0, 1, 0, 0);
CopyWindowToVram(0, 3);
sSlotMachine->state = 7;
return FALSE;
}
static bool8 SlotAction_ExitYouDontHaveThreeCoinsMessage(struct Task *task)
{
if (gMain.newKeys & (A_BUTTON | B_BUTTON))
{
ClearDialogWindowAndFrame(0, TRUE);
sSlotMachine->state = 5;
}
return FALSE;
}
static bool8 SlotAction_GivingInformation(struct Task *task)
{
if (IsInfoBoxClosed())
sSlotMachine->state = 5;
return FALSE;
}
// probably make all the slots roll
static bool8 SlotAction9(struct Task *task)
{
DrawLuckyFlags();
sub_8104DA4();
// for each reel...
ReelTasks_SetUnkTaskData(0);
ReelTasks_SetUnkTaskData(1);
ReelTasks_SetUnkTaskData(2);
sub_80EEC80(); // something with daily slot variable
task->data[0] = 0;
if (sSlotMachine->luckyFlags & LUCKY_BIAS_REELTIME)
{
BeginReeltime();
sSlotMachine->state = 10;
}
else
{
sub_8104CAC(1);
sSlotMachine->state = 11;
}
sSlotMachine->reelIncrement = 8;
if (sSlotMachine->fairRollsLeft)
sSlotMachine->reelIncrement = SlowReelSpeed();
return FALSE;
}
static bool8 SlotAction10(struct Task *task)
{
if (IsFinalTask_RunReelTimeActions())
{
sub_8104CAC(1);
sSlotMachine->luckyFlags &= ~LUCKY_BIAS_REELTIME;
sSlotMachine->state = 11;
}
return FALSE;
}
static bool8 SlotAction_SetLuckySpins(struct Task *task)
{
if (++task->data[0] >= 30)
{
SetLuckySpins();
sSlotMachine->state = 12;
}
return FALSE;
}
static bool8 SlotAction_AwaitReelStop(struct Task *task)
{
if (gMain.newKeys & A_BUTTON)
{
PlaySE(SE_JYUNI);
sub_8102E1C(sSlotMachine->currReel);
sub_8103C14(sSlotMachine->currReel);
sSlotMachine->state = 13;
}
return FALSE;
}
static bool8 SlotAction_WaitForAllReelsToStop(struct Task *task)
{
if (!IsSlotReelMoving(sSlotMachine->currReel))
{
sSlotMachine->currReel++;
sSlotMachine->state = 12;
if (sSlotMachine->currReel > 2)
{
sSlotMachine->state = 14;
}
return TRUE;
}
return FALSE;
}
// once all reels have stopped
static bool8 SlotAction_CheckMatches(struct Task *task)
{
sSlotMachine->luckyFlags &= (LUCKY_BIAS_777 | LUCKY_BIAS_MIXED_777);
CheckMatch();
if (sSlotMachine->fairRollsLeft)
{
sSlotMachine->fairRollsLeft--;
sSlotMachine->fairRollsUsed++;
}
if (sSlotMachine->matchedSymbols)
{
sSlotMachine->state = 15;
AwardPayout();
sub_8103F70();
if ((sSlotMachine->netCoinLoss -= sSlotMachine->payout) < 0)
{
sSlotMachine->netCoinLoss = 0;
}
if (sSlotMachine->matchedSymbols & ((1 << SLOT_MACHINE_MATCHED_777_BLUE) | (1 << SLOT_MACHINE_MATCHED_777_RED)))
{
PlayFanfare(MUS_ME_B_BIG);
sub_8104CAC(6);
}
else if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_777_MIXED))
{
PlayFanfare(MUS_ME_B_BIG);
sub_8104CAC(5);
}
else
{
PlayFanfare(MUS_ME_B_SMALL);
sub_8104CAC(2);
}
// if you matched 777...
if (sSlotMachine->matchedSymbols & ((1 << SLOT_MACHINE_MATCHED_777_MIXED) | (1 << SLOT_MACHINE_MATCHED_777_BLUE) | (1 << SLOT_MACHINE_MATCHED_777_RED)))
{
sSlotMachine->luckyFlags &= ~(LUCKY_BIAS_777 | LUCKY_BIAS_MIXED_777);
if (sSlotMachine->matchedSymbols & ((1 << SLOT_MACHINE_MATCHED_777_BLUE) | (1 << SLOT_MACHINE_MATCHED_777_RED)))
{
sSlotMachine->fairRollsLeft = 0;
sSlotMachine->fairRollsUsed = 0;
sSlotMachine->luckyGame = FALSE;
if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_777_BLUE))
// this may be an error, but if you get blue 777, the game becomes lucky
sSlotMachine->luckyGame = TRUE;
}
}
if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_POWER) && sSlotMachine->pikaPower < 16)
{
sSlotMachine->pikaPower++;
DisplayPikaPower(sSlotMachine->pikaPower);
}
}
else
{
sub_8104CAC(3);
sSlotMachine->state = 20;
if ((sSlotMachine->netCoinLoss += sSlotMachine->bet) > 9999)
sSlotMachine->netCoinLoss = 9999;
}
return FALSE;
}
static bool8 SlotAction_WaitForPayoutToBeAwarded(struct Task *task)
{
if (IsFinalTask_RunAwardPayoutActions())
sSlotMachine->state = 16;
return FALSE;
}
static bool8 SlotAction_EndOfRoll(struct Task *task)
{
if (sub_8103FA0())
{
sSlotMachine->state = 19;
if (sSlotMachine->matchedSymbols & ((1 << SLOT_MACHINE_MATCHED_777_RED) | (1 << SLOT_MACHINE_MATCHED_777_BLUE)))
IncrementGameStat(GAME_STAT_SLOT_JACKPOTS);
if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_REPLAY))
{
sSlotMachine->currReel = 0;
sSlotMachine->state = 9;
}
if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_POWER))
sSlotMachine->state = 17;
if (sSlotMachine->fairRollsLeft && sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_REPLAY))
{
sub_8104CAC(4);
sSlotMachine->state = 18;
}
}
return FALSE;
}
static bool8 SlotAction_MatchedPower(struct Task *task)
{
if (!sub_81040C8())
{
sSlotMachine->state = 19;
if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_REPLAY))
{
sSlotMachine->state = 9;
if (sSlotMachine->fairRollsLeft)
{
sub_8104CAC(4);
sSlotMachine->state = 18;
}
}
}
return FALSE;
}
static bool8 SlotAction18(struct Task *task)
{
if (sub_8104E18())
{
sSlotMachine->state = 19;
if (sSlotMachine->matchedSymbols & (1 << SLOT_MACHINE_MATCHED_REPLAY))
{
sSlotMachine->state = 9;
}
}
return FALSE;
}
static bool8 SlotAction_Loop(struct Task *task)
{
sub_8103D8C(0);
sub_8103D8C(1);
sub_8103D8C(2);
sSlotMachine->state = 2;
return FALSE;
}
static bool8 SlotAction_NoMatches(struct Task *task)
{
if (++task->data[1] > 64)
{
task->data[1] = 0;
sSlotMachine->state = 19;
}
return FALSE;
}
static bool8 SlotAction_PrintQuitTheGame(struct Task *task)
{
DrawDialogueFrame(0, 0);
AddTextPrinterParameterized(0, 1, gText_QuitTheGame, 0, 1, 0, 0);
CopyWindowToVram(0, 3);
CreateYesNoMenuParameterized(0x15, 7, 0x214, 0x180, 0xE, 0xF);
sSlotMachine->state = 22;
return FALSE;
}
static bool8 SlotAction_SeeIfPlayerQuits(struct Task *task)
{
s8 input = Menu_ProcessInputNoWrapClearOnChoose();
if (input == 0) // player chooses to quit
{
ClearDialogWindowAndFrame(0, TRUE);
sub_8103D8C(0);
sub_8103D8C(1);
sub_8103D8C(2);
sSlotMachine->coins += sSlotMachine->bet;
sSlotMachine->state = 27;
}
else if (input == 1 || input == -1) // player chooses not to quit
{
ClearDialogWindowAndFrame(0, TRUE);
sSlotMachine->state = 5;
}
return FALSE;
}
static bool8 SlotAction_PrintMessage_9999Coins(struct Task *task)
{
DrawDialogueFrame(0, 0);
AddTextPrinterParameterized(0, 1, gText_YouveGot9999Coins, 0, 1, 0, 0);
CopyWindowToVram(0, 3);
sSlotMachine->state = 24;
return FALSE;
}
static bool8 SlotAction_ExitMessage_9999Coins(struct Task *task)
{
if (gMain.newKeys & (A_BUTTON | B_BUTTON))
{
ClearDialogWindowAndFrame(0, TRUE);
sSlotMachine->state = 5;
}
return FALSE;
}
static bool8 SlotAction_PrintMessage_NoMoreCoins(struct Task *task)
{
DrawDialogueFrame(0, 0);
AddTextPrinterParameterized(0, 1, gText_YouveRunOutOfCoins, 0, 1, 0, 0);
CopyWindowToVram(0, 3);
sSlotMachine->state = 26;
return FALSE;
}
static bool8 SlotAction_ExitMessage_NoMoreCoins(struct Task *task)
{
if (gMain.newKeys & (A_BUTTON | B_BUTTON))
{
ClearDialogWindowAndFrame(0, TRUE);
sSlotMachine->state = 27;
}
return FALSE;
}
static bool8 SlotAction_EndGame(struct Task *task)
{
SetCoins(sSlotMachine->coins);
AlertTVOfNewCoinTotal(GetCoins());
BeginNormalPaletteFade(0xFFFFFFFF, 0, 0, 16, RGB(0, 0, 0));
sSlotMachine->state++;
return FALSE;
}
static bool8 SlotAction_FreeDataStructures(struct Task *task)
{
if (!gPaletteFade.active)
{
SetMainCallback2(sSlotMachine->prevMainCb);
FREE_AND_SET_NULL(sUnknown_0203AAF4);
FREE_AND_SET_NULL(sUnknown_0203AAF8);
FREE_AND_SET_NULL(sUnknown_0203AAFC);
FREE_AND_SET_NULL(sUnknown_0203AB00);
FREE_AND_SET_NULL(sUnknown_0203AB04);
FREE_AND_SET_NULL(sUnknown_0203AB08);
FREE_AND_SET_NULL(sUnknown_0203AB0C);
FREE_AND_SET_NULL(sUnknown_0203AB10);
FREE_AND_SET_NULL(sUnknown_0203AB14);
FREE_AND_SET_NULL(sUnknown_0203AB18);
FREE_AND_SET_NULL(sUnknown_0203AB1C);
FREE_AND_SET_NULL(sUnknown_0203AB20);
FREE_AND_SET_NULL(sUnknown_0203AB24);
FREE_AND_SET_NULL(sUnknown_0203AB28);
if (sUnknown_0203AAE4 != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAE4);
if (sUnknown_0203AAE8 != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAE8);
if (sUnknown_0203AAEC != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAEC);
if (sUnknown_0203AAF0 != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAF0);
FREE_AND_SET_NULL(sUnknown_0203AAC8);
FREE_AND_SET_NULL(sSelectedPikaPowerTile);
FREE_AND_SET_NULL(sUnknown_0203AAD0);
FREE_AND_SET_NULL(sUnknown_0203AAD4);
FREE_AND_SET_NULL(sUnknown_0203AAD8);
FREE_AND_SET_NULL(sUnknown_0203AADC);
FREE_AND_SET_NULL(sUnknown_0203AAE0);
FREE_AND_SET_NULL(sUnknown_0203AB2C);
FREE_AND_SET_NULL(sUnknown_0203AB30);
FREE_AND_SET_NULL(sSlotMachine);
}
return FALSE;
}
static void DrawLuckyFlags(void)
{
u8 attempts;
if (sSlotMachine->fairRollsLeft == 0)
{
if (!(sSlotMachine->luckyFlags & (LUCKY_BIAS_777 | LUCKY_BIAS_MIXED_777)))
{
if (IsThisRoundLucky())
{
attempts = AttemptsAtLuckyFlags_Top3();
if (attempts != 3) // if you found a lucky number
{
// attempts == 1: reelTime flag set
sSlotMachine->luckyFlags |= gLuckyFlagSettings_Top3[attempts];
if (attempts != 1)
{
return;
}
}
}
// if it's not a lucky round or you got reel time, roll for the lower lucky flags
attempts = AttemptsAtLuckyFlags_NotTop3();
if (attempts != 5) // if you found a lucky number
{
sSlotMachine->luckyFlags |= gLuckyFlagSettings_NotTop3[attempts];
}
}
}
}
static void SetLuckySpins(void)
{
sSlotMachine->luckySpinsLeft = 0;
if (sSlotMachine->luckyFlags)
sSlotMachine->luckySpinsLeft = 1;
}
static u8 GetBiasTag(u8 luckyFlags)
{
u8 i;
for (i = 0; i < 8; i++)
{
if (luckyFlags & 1)
return gBiasTags[i];
luckyFlags >>= 1;
}
return 0;
}
// you have way more luck betting 3 coins than anything lower
static bool8 IsThisRoundLucky(void)
{
u8 rval = Random();
if (gLuckyRoundProbabilities[sSlotMachine->machineId][sSlotMachine->bet - 1] > rval)
return TRUE;
return FALSE;
}
static u8 AttemptsAtLuckyFlags_Top3(void)
{
s16 count;
for (count = 0; count < 3; count++)
{
s16 rval = Random() & 0xff;
s16 value = gLuckyFlagProbabilities_Top3[count][sSlotMachine->machineId];
if (value > rval)
break;
}
return count;
}
static u8 AttemptsAtLuckyFlags_NotTop3(void)
{
s16 count;
for (count = 0; count < 5; count++)
{
s16 rval = Random() & 0xff; // random byte
s16 value = gLuckyFlagProbabilities_NotTop3[count][sSlotMachine->machineId];
// make first attempt easier if it's a lucky game
if (count == 0 && sSlotMachine->luckyGame == TRUE)
{
value += 10;
if (value > 0x100)
value = 0x100;
}
// make last attempt harder if it's a lucky game
else if (count == 4 && sSlotMachine->luckyGame == TRUE)
{
value -= 10;
if (value < 0)
value = 0;
}
if (value > rval)
break;
}
return count;
}
static u8 GetReelTimeProbability(u8 reelTimeDraw)
{
if (sSlotMachine->luckyGame == FALSE)
return gReeltimeProbabilities_UnluckyGame[reelTimeDraw][sSlotMachine->pikaPower];
else
return gReelTimeProbabilities_LuckyGame[reelTimeDraw][sSlotMachine->pikaPower];
}
static void GetReeltimeDraw(void)
{
u8 rval;
s16 reelTimeDraw;
sSlotMachine->reelTimeDraw = 0;
rval = Random();
if (rval < GetReelTimeProbability(0))
return;
for (reelTimeDraw = 5; reelTimeDraw > 0; reelTimeDraw--)
{
rval = Random();
if (rval < GetReelTimeProbability(reelTimeDraw))
break;
}
sSlotMachine->reelTimeDraw = reelTimeDraw;
}
static bool8 SkipToReeltimeAction14(u16 i)
{
u16 rval = Random() & 0xff;
if (rval < gProbabilityTable_SkipToReeltimeAction14[i])
return TRUE;
else
return FALSE;
}
static u16 SlowReelSpeed(void)
{
u8 i = 0;
u8 rval;
u8 value;
if (sSlotMachine->netCoinLoss >= 300)
i = 4;
else if (sSlotMachine->netCoinLoss >= 250)
i = 3;
else if (sSlotMachine->netCoinLoss >= 200)
i = 2;
else if (sSlotMachine->netCoinLoss >= 150)
i = 1;
rval = Random() % 100;
value = gReelIncrementTable[i][0];
if (rval < value)
return 4;
rval = Random() % 100;
value = gReelIncrementTable[i][1] + gReelTimeBonusIncrementTable[sSlotMachine->fairRollsUsed];
if (rval < value)
return 2;
return 8;
}
static void CheckMatch(void)
{
sSlotMachine->matchedSymbols = 0;
CheckMatch_CenterRow();
if (sSlotMachine->bet > 1)
CheckMatch_TopAndBottom();
if (sSlotMachine->bet > 2)
CheckMatch_Diagonals();
}
static void CheckMatch_CenterRow(void)
{
u8 c1, c2, c3, match;
c1 = GetNearbyTag_Quantized(0, 2);
c2 = GetNearbyTag_Quantized(1, 2);
c3 = GetNearbyTag_Quantized(2, 2);
match = GetMatchFromSymbolsInRow(c1, c2, c3);
if (match != SLOT_MACHINE_MATCHED_NONE)
{
sSlotMachine->payout += gSlotPayouts[match];
sSlotMachine->matchedSymbols |= gSlotMatchFlags[match];
sub_8103E04(0);
}
}
static void CheckMatch_TopAndBottom(void)
{
u8 c1, c2, c3, match;
c1 = GetNearbyTag_Quantized(0, 1);
c2 = GetNearbyTag_Quantized(1, 1);
c3 = GetNearbyTag_Quantized(2, 1);
match = GetMatchFromSymbolsInRow(c1, c2, c3);
if (match != SLOT_MACHINE_MATCHED_NONE)
{
if (match == SLOT_MACHINE_MATCHED_1CHERRY)
match = SLOT_MACHINE_MATCHED_2CHERRY;
sSlotMachine->payout += gSlotPayouts[match];
sSlotMachine->matchedSymbols |= gSlotMatchFlags[match];
sub_8103E04(1);
}
c1 = GetNearbyTag_Quantized(0, 3);
c2 = GetNearbyTag_Quantized(1, 3);
c3 = GetNearbyTag_Quantized(2, 3);
match = GetMatchFromSymbolsInRow(c1, c2, c3);
if (match != SLOT_MACHINE_MATCHED_NONE)
{
if (match == SLOT_MACHINE_MATCHED_1CHERRY)
match = SLOT_MACHINE_MATCHED_2CHERRY;
sSlotMachine->payout += gSlotPayouts[match];
sSlotMachine->matchedSymbols |= gSlotMatchFlags[match];
sub_8103E04(2);
}
}
static void CheckMatch_Diagonals(void)
{
u8 c1, c2, c3, match;
c1 = GetNearbyTag_Quantized(0, 1);
c2 = GetNearbyTag_Quantized(1, 2);
c3 = GetNearbyTag_Quantized(2, 3);
match = GetMatchFromSymbolsInRow(c1, c2, c3);
if (match != SLOT_MACHINE_MATCHED_NONE)
{
if (match != SLOT_MACHINE_MATCHED_1CHERRY)
{
sSlotMachine->payout += gSlotPayouts[match];
sSlotMachine->matchedSymbols |= gSlotMatchFlags[match];
}
sub_8103E04(3);
}
c1 = GetNearbyTag_Quantized(0, 3);
c2 = GetNearbyTag_Quantized(1, 2);
c3 = GetNearbyTag_Quantized(2, 1);
match = GetMatchFromSymbolsInRow(c1, c2, c3);
if (match != SLOT_MACHINE_MATCHED_NONE)
{
if (match != SLOT_MACHINE_MATCHED_1CHERRY)
{
sSlotMachine->payout += gSlotPayouts[match];
sSlotMachine->matchedSymbols |= gSlotMatchFlags[match];
}
sub_8103E04(4);
}
}
static u8 GetMatchFromSymbolsInRow(u8 c1, u8 c2, u8 c3)
{
if (c1 == c2 && c1 == c3)
return gSym2Match[c1];
if (c1 == SLOT_MACHINE_TAG_7_RED && c2 == SLOT_MACHINE_TAG_7_RED && c3 == SLOT_MACHINE_TAG_7_BLUE)
return SLOT_MACHINE_MATCHED_777_MIXED;
if (c1 == SLOT_MACHINE_TAG_7_BLUE && c2 == SLOT_MACHINE_TAG_7_BLUE && c3 == SLOT_MACHINE_TAG_7_RED)
return SLOT_MACHINE_MATCHED_777_MIXED;
if (c1 == SLOT_MACHINE_TAG_CHERRY)
return SLOT_MACHINE_MATCHED_1CHERRY;
return SLOT_MACHINE_MATCHED_NONE;
}
static void AwardPayout(void)
{
RunAwardPayoutActions(CreateTask(RunAwardPayoutActions, 4));
}
static bool8 IsFinalTask_RunAwardPayoutActions(void)
{
if (FindTaskIdByFunc(RunAwardPayoutActions) == TAIL_SENTINEL)
return TRUE;
else
return FALSE;
}
static void RunAwardPayoutActions(u8 taskId)
{
while (AwardPayoutActions[gTasks[taskId].data[0]](&gTasks[taskId]))
;
}
static bool8 AwardPayoutAction0(struct Task *task)
{
if (sub_8103E38())
{
task->data[0]++;
if (sSlotMachine->payout == 0)
{
task->data[0] = 2;
return TRUE;
}
}
return FALSE;
}
// task->data[1]: timer
static bool8 AwardPayoutAction_GivePayoutToPlayer(struct Task *task)
{
if (!task->data[1]--)
{
if (IsFanfareTaskInactive())
PlaySE(SE_PIN);
sSlotMachine->payout--;
if (sSlotMachine->coins < 9999)
sSlotMachine->coins++;
task->data[1] = 8;
if (gMain.heldKeys & A_BUTTON)
task->data[1] = 4;
}
if (IsFanfareTaskInactive() && gMain.newKeys & START_BUTTON)
{
PlaySE(SE_PIN);
sSlotMachine->coins += sSlotMachine->payout;
if (sSlotMachine->coins > 9999)
sSlotMachine->coins = 9999;
sSlotMachine->payout = 0;
}
if (sSlotMachine->payout == 0)
task->data[0]++;
return FALSE;
}
static bool8 AwardPayoutAction_FreeTask(struct Task *task)
{
if (sub_8103E7C())
DestroyTask(FindTaskIdByFunc(RunAwardPayoutActions));
return FALSE;
}
// Returns the tag that is posOffset below the tag at the top of reelIndex's tape
static u8 GetNearbyTag_Quantized(u8 reelIndex, s16 posOffset)
{
s16 tagIndex = (sSlotMachine->reelTagOffsets[reelIndex] + posOffset) % REEL_NUM_TAGS;
if (tagIndex < 0)
tagIndex += REEL_NUM_TAGS;
return gReelSymbols[reelIndex][tagIndex];
}
// Calculates GetNearbyTag_Quantized as if the reel was snapped downwards into place
static u8 GetNearbyTag(u8 reelIndex, s16 posOffset)
{
s16 tagOffset = 0;
s16 result = sSlotMachine->reelPixelOffsets[reelIndex] % 24;
if (result != 0)
tagOffset = -1;
return GetNearbyTag_Quantized(reelIndex, posOffset + tagOffset);
}
static u8 GetNearbyReelTimeTag(s16 n)
{
s16 newPosition = (sSlotMachine->reeltimePosition + n) % 6;
if (newPosition < 0)
newPosition += 6;
return gReelTimeTags[newPosition];
}
static void AdvanceSlotReel(u8 reelIndex, s16 value)
{
sSlotMachine->reelPixelOffsets[reelIndex] += value;
sSlotMachine->reelPixelOffsets[reelIndex] %= 504;
sSlotMachine->reelTagOffsets[reelIndex] = REEL_NUM_TAGS - sSlotMachine->reelPixelOffsets[reelIndex] / 24;
}
s16 AdvanceSlotReelToNextTag(u8 reelIndex, s16 value)
{
s16 offset = sSlotMachine->reelPixelOffsets[reelIndex] % 24;
if (offset != 0)
{
if (offset < value)
value = offset;
AdvanceSlotReel(reelIndex, value);
offset = sSlotMachine->reelPixelOffsets[reelIndex] % 24;
}
return offset;
}
static void AdvanceReeltimeReel(s16 value)
{
sSlotMachine->reeltimePixelOffset += value;
sSlotMachine->reeltimePixelOffset %= 120;
sSlotMachine->reeltimePosition = 6 - sSlotMachine->reeltimePixelOffset / 20;
}
s16 AdvanceReeltimeReelToNextTag(s16 value)
{
s16 offset = sSlotMachine->reeltimePixelOffset % 20;
if (offset != 0)
{
if (offset < value)
value = offset;
AdvanceReeltimeReel(value);
offset = sSlotMachine->reeltimePixelOffset % 20;
}
return offset;
}
static void GameplayTask_StopSlotReel(void)
{
u8 i;
for (i = 0; i < 3; i++)
{
u8 taskId = CreateTask(RunSlotReelActions, 2);
gTasks[taskId].data[15] = i;
sSlotMachine->slotReelTasks[i] = taskId;
RunSlotReelActions(taskId);
}
}
static void ReelTasks_SetUnkTaskData(u8 reelIndex)
{
gTasks[sSlotMachine->slotReelTasks[reelIndex]].data[0] = 1;
gTasks[sSlotMachine->slotReelTasks[reelIndex]].data[14] = 1;
}
static void sub_8102E1C(u8 reelIndex)
{
gTasks[sSlotMachine->slotReelTasks[reelIndex]].data[0] = 2;
}
static bool8 IsSlotReelMoving(u8 reelIndex)
{
return gTasks[sSlotMachine->slotReelTasks[reelIndex]].data[14];
}
static void RunSlotReelActions(u8 taskId)
{
while (SlotReelActions[gTasks[taskId].data[0]](&gTasks[taskId]))
;
}
// task->data[1] reel turns
// task->data[15] reelIndex
static bool8 SlotReelAction_StayStill(struct Task *task)
{
return FALSE;
}
static bool8 SlotReelAction_Spin(struct Task *task)
{
AdvanceSlotReel(task->data[15], sSlotMachine->reelIncrement);
return FALSE;
}
// As in previous generations, the slot machine often doesn't stop exactly when you press stop
static bool8 SlotReelAction_DecideWhereToStop(struct Task *task)
{
task->data[0]++;
// initialize data for that reel --> these will be changed if gBiasTags can be lined up
sSlotMachine->winnerRows[task->data[15]] = 0;
sSlotMachine->reelExtraTurns[task->data[15]] = 0;
if (sSlotMachine->fairRollsLeft == 0 && (sSlotMachine->luckyFlags == 0 || sSlotMachine->luckySpinsLeft == 0 || !DecideReelTurns_BiasTag[task->data[15]]()))
{
sSlotMachine->luckySpinsLeft = 0;
DecideReelTurns_NoBiasTag[task->data[15]]();
}
task->data[1] = sSlotMachine->reelExtraTurns[task->data[15]];
return TRUE;
}
// go to next tag and then do any additional turns
static bool8 SlotReelAction_MoveToStop(struct Task *task)
{
u16 reelStopShocks[ARRAY_COUNT(ReelStopShocks)];
s16 reelPixelPos;
memcpy(reelStopShocks, ReelStopShocks, sizeof(ReelStopShocks));
reelPixelPos = sSlotMachine->reelPixelOffsets[task->data[15]] % 24;
if (reelPixelPos != 0)
reelPixelPos = AdvanceSlotReelToNextTag(task->data[15], sSlotMachine->reelIncrement);
else if (sSlotMachine->reelExtraTurns[task->data[15]])
{
sSlotMachine->reelExtraTurns[task->data[15]]--;
AdvanceSlotReel(task->data[15], sSlotMachine->reelIncrement);
reelPixelPos = sSlotMachine->reelPixelOffsets[task->data[15]] % 24;
}
if (reelPixelPos == 0 && sSlotMachine->reelExtraTurns[task->data[15]] == 0)
{
task->data[0]++;
task->data[1] = reelStopShocks[task->data[1]];
task->data[2] = 0;
}
return FALSE;
}
// make selected tag oscillate before it becomes still
static bool8 SlotReelAction_OscillatingStop(struct Task *task)
{
sSlotMachine->reelPixelOffsetsWhileStopping[task->data[15]] = task->data[1];
task->data[1] = -task->data[1];
task->data[2]++;
if ((task->data[2] & 0x3) == 0)
task->data[1] >>= 1;
if (task->data[1] == 0)
{
task->data[0] = 0;
task->data[14] = 0;
sSlotMachine->reelPixelOffsetsWhileStopping[task->data[15]] = 0;
}
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel1(void)
{
u8 tag2 = GetBiasTag(sSlotMachine->luckyFlags);
u8 tag1 = tag2;
if (sSlotMachine->luckyFlags & (LUCKY_BIAS_777 | LUCKY_BIAS_MIXED_777))
{
tag1 = SLOT_MACHINE_TAG_7_RED;
tag2 = SLOT_MACHINE_TAG_7_BLUE;
}
return DecideReelTurns_BiasTag_Reel1_Bets[sSlotMachine->bet - 1](tag1, tag2);
}
static bool8 AreTagsAtPosition_Reel1(s16 pos, u8 tag1, u8 tag2)
{
u8 tag = GetNearbyTag(0, pos);
if (tag == tag1 || tag == tag2)
{
sSlotMachine->biasTag = tag;
return TRUE;
}
return FALSE;
}
static bool8 AreCherriesOnScreen_Reel1(s16 offsetFromCenter)
{
if (GetNearbyTag(0, 1 - offsetFromCenter) == SLOT_MACHINE_TAG_CHERRY || GetNearbyTag(0, 2 - offsetFromCenter) == SLOT_MACHINE_TAG_CHERRY || GetNearbyTag(0, 3 - offsetFromCenter) == SLOT_MACHINE_TAG_CHERRY)
return TRUE;
else
return FALSE;
}
static bool8 IsBiasTowardsCherryOr7s(void)
{
if (sSlotMachine->luckyFlags & (LUCKY_BIAS_777 | LUCKY_BIAS_MIXED_777 | LUCKY_BIAS_CHERRY))
return TRUE;
else
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel1_Bet1(u8 tag1, u8 tag2)
{
s16 i;
for (i = 0; i < 5; i++)
{
// if a lucky tag appears in the center row within 4 turns
if (AreTagsAtPosition_Reel1(2 - i, tag1, tag2))
{
sSlotMachine->winnerRows[0] = 2;
sSlotMachine->reelExtraTurns[0] = i;
return TRUE;
}
}
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel1_Bet2or3(u8 tag1, u8 tag2)
{
s16 i;
bool8 biased = IsBiasTowardsCherryOr7s();
// if lucky numbers or no cherries are currently on screen in reel 1...
if (biased || !AreCherriesOnScreen_Reel1(0))
{
for (i = 1; i < 4; i++)
{
// if a bias tag is currently on the screen
if (AreTagsAtPosition_Reel1(i, tag1, tag2))
{
sSlotMachine->winnerRows[0] = i;
sSlotMachine->reelExtraTurns[0] = 0;
return TRUE;
}
}
}
for (i = 1; i < 5; i++)
{
bool8 biasedCopy = biased; // redundant
// if biased or if in the next 4 turns there is a screen with no cherries...
if (biasedCopy || !AreCherriesOnScreen_Reel1(i))
{
//...and if a bias tag is in top row of that screen
if (AreTagsAtPosition_Reel1(1 - i, tag1, tag2))
{
//...and if it only took 1 turn and the lucky tag could also be the bottom row of a screen with no cherries...
if (i == 1 && (biasedCopy || !AreCherriesOnScreen_Reel1(3)))
{
sSlotMachine->winnerRows[0] = 3;
sSlotMachine->reelExtraTurns[0] = 3;
return TRUE;
}
//...or if it isn't the last turn and the lucky tag could be in the center row of a screen with no cherries...
if (i < 4 && (biasedCopy || !AreCherriesOnScreen_Reel1(i + 1)))
{
sSlotMachine->winnerRows[0] = 2;
sSlotMachine->reelExtraTurns[0] = i + 1;
return TRUE;
}
//...else
sSlotMachine->winnerRows[0] = 1;
sSlotMachine->reelExtraTurns[0] = i;
return TRUE;
}
}
}
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel2(void)
{
return DecideReelTurns_BiasTag_Reel2_Bets[sSlotMachine->bet - 1]();
}
static bool8 DecideReelTurns_BiasTag_Reel2_Bet1or2(void)
{
s16 i;
s16 biasTagLocation_Reel1 = sSlotMachine->winnerRows[0];
for (i = 0; i < 5; i++)
{
// if biasTag appears in the same row within 4 turns
if (GetNearbyTag(1, biasTagLocation_Reel1 - i) == sSlotMachine->biasTag)
{
sSlotMachine->winnerRows[1] = biasTagLocation_Reel1;
sSlotMachine->reelExtraTurns[1] = i;
return TRUE;
}
}
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel2_Bet3(void)
{
s16 i;
// if biasTag appears in the same row within 4 turns...
if (DecideReelTurns_BiasTag_Reel2_Bet1or2())
{
//...and if the biasTag is not in middle row of reel 1 and if biasTag appears in middle row of reel 2 in 2 or 3 turns...
if (sSlotMachine->winnerRows[0] != 2 && sSlotMachine->reelExtraTurns[1] > 1 && sSlotMachine->reelExtraTurns[1] != 4)
{
for (i = 0; i < 5; i++)
{
//...and if the bias tag will appear in the middle row within 4 turns
if (GetNearbyTag(1, 2 - i) == sSlotMachine->biasTag)
{
sSlotMachine->winnerRows[1] = 2;
sSlotMachine->reelExtraTurns[1] = i;
break;
}
}
}
return TRUE;
}
// else if the biasTag is not in middle row of reel 1...
if (sSlotMachine->winnerRows[0] != 2)
{
for (i = 0; i < 5; i++)
{
//...and if the biasTag will appear in the center row of reel 2 within 4 turns
if (GetNearbyTag(1, 2 - i) == sSlotMachine->biasTag)
{
sSlotMachine->winnerRows[1] = 2;
sSlotMachine->reelExtraTurns[1] = i;
return TRUE;
}
}
}
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel3(void)
{
u8 biasTag = sSlotMachine->biasTag;
if (sSlotMachine->luckyFlags & LUCKY_BIAS_MIXED_777)
{
biasTag = SLOT_MACHINE_TAG_7_RED;
if (sSlotMachine->biasTag == SLOT_MACHINE_TAG_7_RED)
{
biasTag = SLOT_MACHINE_TAG_7_BLUE;
}
}
return DecideReelTurns_BiasTag_Reel3_Bets[sSlotMachine->bet - 1](biasTag);
}
static bool8 DecideReelTurns_BiasTag_Reel3_Bet1or2(u8 biasTag)
{
s16 i;
s16 biasTagLocation_Reel2 = sSlotMachine->winnerRows[1];
for (i = 0; i < 5; i++)
{
// if the biasTag appears in the same row as in reel 2 within 4 turns
if (GetNearbyTag(2, biasTagLocation_Reel2 - i) == biasTag)
{
sSlotMachine->winnerRows[2] = biasTagLocation_Reel2;
sSlotMachine->reelExtraTurns[2] = i;
return TRUE;
}
}
return FALSE;
}
static bool8 DecideReelTurns_BiasTag_Reel3_Bet3(u8 biasTag)
{
s16 i;
s16 biasTagFinalPos;
// if the final position of the biasTag matches in reel 1 and reel 2...
if (sSlotMachine->winnerRows[0] == sSlotMachine->winnerRows[1])
//...then try to line it up in reel 3
return DecideReelTurns_BiasTag_Reel3_Bet1or2(biasTag);
// else place it in the row opposite reel 1's
if (sSlotMachine->winnerRows[0] == 1)
biasTagFinalPos = 3;
else
biasTagFinalPos = 1;
for (i = 0; i < 5; i++)
{
// if the biasTag lands in that position within 4 turns
if (GetNearbyTag(2, biasTagFinalPos - i) == biasTag)
{
sSlotMachine->reelExtraTurns[2] = i;
sSlotMachine->winnerRows[2] = biasTagFinalPos;
return TRUE;
}
}
return FALSE;
}
// Advance until there are no cherries on screen in reel 1
static void DecideReelTurns_NoBiasTag_Reel1(void)
{
s16 i = 0;
while (AreCherriesOnScreen_Reel1(i) != 0)
i++;
sSlotMachine->reelExtraTurns[0] = i;
}
static bool8 IsBiasTag777_SwitchColor(u8 *biasTagPtr)
{
if (*biasTagPtr == SLOT_MACHINE_TAG_7_RED)
{
*biasTagPtr = SLOT_MACHINE_TAG_7_BLUE;
return TRUE;
}
if (*biasTagPtr == SLOT_MACHINE_TAG_7_BLUE)
{
*biasTagPtr = SLOT_MACHINE_TAG_7_RED;
return TRUE;
}
return FALSE;
}
static void DecideReelTurns_NoBiasTag_Reel2(void)
{
DecideReelTurns_NoBiasTag_Reel2_Bets[sSlotMachine->bet - 1]();
}
// only does stuff if the biasTag is one of the 7's, plus other conditions
static void DecideReelTurns_NoBiasTag_Reel2_Bet1(void)
{
if (sSlotMachine->winnerRows[0] != 0 && sSlotMachine->luckyFlags & LUCKY_BIAS_777)
{
u8 biasTag = GetNearbyTag(0, 2 - sSlotMachine->reelExtraTurns[0]);
//...and if biasTag is one of the 7's...
if (IsBiasTag777_SwitchColor(&biasTag))
//...swap color of biasTag...
{
s16 i;
for (i = 0; i < 5; i++)
{
//...and if the biasTag appears within 4 turns
if (biasTag == GetNearbyTag(1, 2 - i))
{
sSlotMachine->winnerRows[1] = 2;
sSlotMachine->reelExtraTurns[1] = i;
break;
}
}
}
}
}
static void DecideReelTurns_NoBiasTag_Reel2_Bet2(void)
{
if (sSlotMachine->winnerRows[0] != 0 && sSlotMachine->luckyFlags & LUCKY_BIAS_777)
{
u8 biasTag = GetNearbyTag(0, sSlotMachine->winnerRows[0] - sSlotMachine->reelExtraTurns[0]);
//...and if biasTag is one of the 7's...
if (IsBiasTag777_SwitchColor(&biasTag))
//...swap color of biasTag...
{
s16 i;
for (i = 0; i < 5; i++)
{
//...and if the biasTag appears in same row in reel 2 within 4 turns
if (biasTag == GetNearbyTag(1, sSlotMachine->winnerRows[0] - i))
{
sSlotMachine->winnerRows[1] = sSlotMachine->winnerRows[0];
sSlotMachine->reelExtraTurns[1] = i;
break;
}
}
}
}
}
static void DecideReelTurns_NoBiasTag_Reel2_Bet3(void)
{
s16 i;
s16 j;
// if reel 1 has a biasTag and bit 7 is set in luckyFlags...
if (sSlotMachine->winnerRows[0] != 0 && sSlotMachine->luckyFlags & LUCKY_BIAS_777)
{
//...and if biasTag appeared in the center row of reel 1
if (sSlotMachine->winnerRows[0] == 2)
{
DecideReelTurns_NoBiasTag_Reel2_Bet2();
}
else
{
u8 biasTag = GetNearbyTag(0, sSlotMachine->winnerRows[0] - sSlotMachine->reelExtraTurns[0]);
//...and if biasTag is one of the 7's...
if (IsBiasTag777_SwitchColor(&biasTag))
//...swap the color of the 7...
{
j = 2;
if (sSlotMachine->winnerRows[0] == 3)
j = 3;
for (i = 0; i < 2; i++, j--)
{
if (biasTag == GetNearbyTag(1, j))
{
sSlotMachine->winnerRows[1] = j;
sSlotMachine->reelExtraTurns[1] = 0;
return;
}
}
for (j = 1; j < 5; j++)
{
if (biasTag == GetNearbyTag(1, sSlotMachine->winnerRows[0] - j))
{
if (sSlotMachine->winnerRows[0] == 1)
{
if (j < 3)
{
sSlotMachine->winnerRows[1] = 2;
sSlotMachine->reelExtraTurns[1] = j + 1;
}
else
{
sSlotMachine->winnerRows[1] = 1;
sSlotMachine->reelExtraTurns[1] = j;
}
}
else
{
if (j < 3)
{
sSlotMachine->winnerRows[1] = 3;
sSlotMachine->reelExtraTurns[1] = j;
}
else
{
sSlotMachine->winnerRows[1] = 2;
sSlotMachine->reelExtraTurns[1] = j - 1;
}
}
return;
}
}
}
}
}
}
static bool8 AreTagsMixed77(u8 tag1, u8 tag2)
{
if ((tag1 == SLOT_MACHINE_TAG_7_RED && tag2 == SLOT_MACHINE_TAG_7_BLUE) || (tag1 == SLOT_MACHINE_TAG_7_BLUE && tag2 == SLOT_MACHINE_TAG_7_RED))
return TRUE;
else
return FALSE;
}
static bool8 AreTagsMixed777(u8 tag1, u8 tag2, u8 tag3)
{
if ((tag1 == SLOT_MACHINE_TAG_7_RED && tag2 == SLOT_MACHINE_TAG_7_BLUE && tag3 == SLOT_MACHINE_TAG_7_RED) ||
(tag1 == SLOT_MACHINE_TAG_7_BLUE && tag2 == SLOT_MACHINE_TAG_7_RED && tag3 == SLOT_MACHINE_TAG_7_BLUE))
return TRUE;
else
return FALSE;
}
static bool8 TagsDontMatchOrHaveAny7s(u8 tag1, u8 tag2, u8 tag3)
{
if ((tag1 == SLOT_MACHINE_TAG_7_RED && tag2 == SLOT_MACHINE_TAG_7_BLUE && tag3 == SLOT_MACHINE_TAG_7_RED) ||
(tag1 == SLOT_MACHINE_TAG_7_BLUE && tag2 == SLOT_MACHINE_TAG_7_RED && tag3 == SLOT_MACHINE_TAG_7_BLUE) ||
(tag1 == SLOT_MACHINE_TAG_7_RED && tag2 == SLOT_MACHINE_TAG_7_RED && tag3 == SLOT_MACHINE_TAG_7_BLUE) ||
(tag1 == SLOT_MACHINE_TAG_7_BLUE && tag2 == SLOT_MACHINE_TAG_7_BLUE && tag3 == SLOT_MACHINE_TAG_7_RED) ||
(tag1 == tag2 && tag1 == tag3))
{
return FALSE;
}
return TRUE;
}
static void DecideReelTurns_NoBiasTag_Reel3(void)
{
DecideReelTurns_NoBiasTag_Reel3_Bets[sSlotMachine->bet - 1]();
}
static void DecideReelTurns_NoBiasTag_Reel3_Bet1(void)
{
s16 i = 0;
u8 tag1 = GetNearbyTag(0, 2 - sSlotMachine->reelExtraTurns[0]);
u8 tag2 = GetNearbyTag(1, 2 - sSlotMachine->reelExtraTurns[1]);
// if tags match in first 2 reels...
if (tag1 == tag2)
{
//...spin until you get non-matching tag
while (1)
{
u8 tag3;
if (!(tag1 == (tag3 = GetNearbyTag(2, 2 - i)) || (tag1 == SLOT_MACHINE_TAG_7_RED && tag3 == SLOT_MACHINE_TAG_7_BLUE) || (tag1 == SLOT_MACHINE_TAG_7_BLUE && tag3 == SLOT_MACHINE_TAG_7_RED)))
break;
i++;
}
}
else if (AreTagsMixed77(tag1, tag2))
{
if (sSlotMachine->luckyFlags & LUCKY_BIAS_777)
{
//...see if you can match with reel 1 within 4 turns
for (i = 0; i < 5; i++)
{
if (tag1 == GetNearbyTag(2, 2 - i))
{
sSlotMachine->reelExtraTurns[2] = i;
return;
}
}
}
// turn until you aren't matching with reel 1
i = 0;
while (1)
{
if (tag1 != GetNearbyTag(2, 2 - i))
break;
i++;
}
}
sSlotMachine->reelExtraTurns[2] = i;
}
static void DecideReelTurns_NoBiasTag_Reel3_Bet2(void)
{
s16 extraTurns = 0;
s16 i;
u8 tag1;
u8 tag2;
u8 tag3;
if (sSlotMachine->winnerRows[1] != 0 && sSlotMachine->winnerRows[0] == sSlotMachine->winnerRows[1] && sSlotMachine->luckyFlags & LUCKY_BIAS_777)
{
tag1 = GetNearbyTag(0, sSlotMachine->winnerRows[0] - sSlotMachine->reelExtraTurns[0]);
tag2 = GetNearbyTag(1, sSlotMachine->winnerRows[1] - sSlotMachine->reelExtraTurns[1]);
//...and if tags are mixed 7s...
if (AreTagsMixed77(tag1, tag2))
{
//...try to match with reel 1 within 4 turns
for (i = 0; i < 5; i++)
{
tag3 = GetNearbyTag(2, sSlotMachine->winnerRows[1] - i);
if (tag1 == tag3)
{
extraTurns = i;
break;
}
}
}
}
// GUESS: spin until there's no possible match within 4 turns of you stopping
while (1)
{
s16 loopExit;
for (i = 1, loopExit = 0; i < 4; i++)
{
tag1 = GetNearbyTag(0, i - sSlotMachine->reelExtraTurns[0]); // why does this update with i
tag2 = GetNearbyTag(1, i - sSlotMachine->reelExtraTurns[1]);
tag3 = GetNearbyTag(2, i - extraTurns);
// if bit 7 of luckyFlags is unset...
//...and if all 3 tags match and they're not mixed 7s
if (!TagsDontMatchOrHaveAny7s(tag1, tag2, tag3) && (!AreTagsMixed777(tag1, tag2, tag3) || !(sSlotMachine->luckyFlags & LUCKY_BIAS_777)))
{
loopExit++;
break;
}
}
if (loopExit == 0)
break;
extraTurns++;
}
sSlotMachine->reelExtraTurns[2] = extraTurns;
}
static void DecideReelTurns_NoBiasTag_Reel3_Bet3(void)
{
u8 tag1;
u8 tag2;
u8 tag3;
s16 j;
s16 i;
DecideReelTurns_NoBiasTag_Reel3_Bet2();
if (sSlotMachine->winnerRows[1] != 0 && sSlotMachine->winnerRows[0] != sSlotMachine->winnerRows[1] && sSlotMachine->luckyFlags & LUCKY_BIAS_777)
{
tag1 = GetNearbyTag(0, sSlotMachine->winnerRows[0] - sSlotMachine->reelExtraTurns[0]);
tag2 = GetNearbyTag(1, sSlotMachine->winnerRows[1] - sSlotMachine->reelExtraTurns[1]);
//..and if tags are mixed 7s...
if (AreTagsMixed77(tag1, tag2))
{
j = 1;
if (sSlotMachine->winnerRows[0] == 1)
j = 3;
for (i = 0; i < 5; i++)
{
tag3 = GetNearbyTag(2, j - (sSlotMachine->reelExtraTurns[2] + i));
if (tag1 == tag3)
{
sSlotMachine->reelExtraTurns[2] += i;
break;
}
}
}
}
while (1)
{
tag1 = GetNearbyTag(0, 1 - sSlotMachine->reelExtraTurns[0]);
tag2 = GetNearbyTag(1, 2 - sSlotMachine->reelExtraTurns[1]);
tag3 = GetNearbyTag(2, 3 - sSlotMachine->reelExtraTurns[2]);
if (TagsDontMatchOrHaveAny7s(tag1, tag2, tag3) || (AreTagsMixed777(tag1, tag2, tag3) && sSlotMachine->luckyFlags & LUCKY_BIAS_777))
break;
sSlotMachine->reelExtraTurns[2]++;
}
while (1)
{
tag1 = GetNearbyTag(0, 3 - sSlotMachine->reelExtraTurns[0]);
tag2 = GetNearbyTag(1, 2 - sSlotMachine->reelExtraTurns[1]);
tag3 = GetNearbyTag(2, 1 - sSlotMachine->reelExtraTurns[2]);
if (TagsDontMatchOrHaveAny7s(tag1, tag2, tag3) || (AreTagsMixed777(tag1, tag2, tag3) && sSlotMachine->luckyFlags & LUCKY_BIAS_777))
break;
sSlotMachine->reelExtraTurns[2]++;
}
}
static void sub_8103C14(u8 a0)
{
u8 taskId = CreateTask(sub_8103C48, 5);
gTasks[taskId].data[15] = a0;
sub_8103C48(taskId);
}
static void sub_8103C48(u8 taskId)
{
gUnknown_083ECBA0[gTasks[taskId].data[0]](&gTasks[taskId], taskId);
}
static void sub_8103C78(struct Task *task, u8 taskId)
{
sub_81065A8(gUnknown_083ECBAC[task->data[15]], 0x62, 0x63, 0x72, 0x73);
task->data[0]++;
}
static void sub_8103CAC(struct Task *task, u8 taskId)
{
if (++task->data[1] > 11)
task->data[0]++;
}
static void sub_8103CC8(struct Task *task, u8 taskId)
{
sub_81065A8(gUnknown_083ECBAC[task->data[15]], 0x42, 0x43, 0x52, 0x53);
DestroyTask(taskId);
}
static void LoadLitTile(u8 tileId)
{
LoadPalette(gUnknown_083EDD08[tileId], gUnknown_083EDD30[tileId], 2);
}
static void sub_8103D28(u8 a0)
{
LoadPalette(gUnknown_083EDD1C[a0], gUnknown_083EDD30[a0], 2);
}
// light up the value bet by the player
static void LoadBetTiles(u8 betVal)
{
u8 i;
for (i = 0; i < gNumberBettingTiles[betVal]; i++)
LoadLitTile(gBettingTilesId[betVal][i]);
}
static void sub_8103D8C(u8 a0)
{
u8 i;
for (i = 0; i < gNumberBettingTiles[a0]; i++)
sub_8103D28(gBettingTilesId[a0][i]);
}
static void sub_8103DC8(void)
{
u8 i;
for (i = 0; i < 5; i++)
{
u8 spriteId = CreateInvisibleSprite(sub_8103EE4);
gSprites[spriteId].data[0] = i;
sSlotMachine->unk44[i] = spriteId;
}
}
static void sub_8103E04(u8 a0)
{
struct Sprite *sprite = gSprites + sSlotMachine->unk44[a0];
sprite->data[1] = 1;
sprite->data[2] = 4;
sprite->data[3] = 0;
sprite->data[4] = 0;
sprite->data[5] = 2;
sprite->data[7] = 0;
}
// something with payout digits
static bool8 sub_8103E38(void)
{
u8 i;
for (i = 0; i < 5; i++)
{
struct Sprite *sprite = &gSprites[sSlotMachine->unk44[i]];
if (sprite->data[1] && sprite->data[2])
return FALSE;
}
return TRUE;
}
static bool8 sub_8103E7C(void)
{
u8 i;
for (i = 0; i < 5; i++)
{
if (!sub_8103EAC(sSlotMachine->unk44[i]))
return FALSE;
}
return TRUE;
}
static bool8 sub_8103EAC(u8 spriteId)
{
struct Sprite *sprite = gSprites + spriteId;
if (!sprite->data[1])
return TRUE;
if (sprite->data[7])
sprite->data[1] = 0;
return sprite->data[7];
}
static void sub_8103EE4(struct Sprite *sprite)
{
s16 r4;
if (sprite->data[1])
{
if (!sprite->data[3]--)
{
sprite->data[7] = 0;
sprite->data[3] = 1;
sprite->data[4] += sprite->data[5];
r4 = 4;
if (sprite->data[2])
r4 = 8;
if (sprite->data[4] <= 0)
{
sprite->data[7] = 1;
sprite->data[5] = -sprite->data[5];
if (sprite->data[2])
sprite->data[2]--;
}
else if (sprite->data[4] >= r4)
sprite->data[5] = -sprite->data[5];
if (sprite->data[2])
sprite->data[3] <<= 1;
}
MultiplyPaletteRGBComponents(gUnknown_083EDD30[sprite->data[0]], sprite->data[4], sprite->data[4], sprite->data[4]);
}
}
static void sub_8103F70(void)
{
u8 taskId = CreateTask(sub_8103FE8_, 6);
gTasks[taskId].data[3] = 1;
sub_8103FE8_(taskId);
}
static bool8 sub_8103FA0(void)
{
u8 taskId = FindTaskIdByFunc(sub_8103FE8_);
if (!gTasks[taskId].data[2])
{
DestroyTask(taskId);
LoadPalette(gUnknown_083EDDAC, 0x10, 0x20);
return TRUE;
}
return FALSE;
}
static void sub_8103FE8_(u8 taskId)
{
struct Task *task = &gTasks[taskId];
if (!task->data[1]--)
{
task->data[1] = 4;
task->data[2] += task->data[3];
if (task->data[2] == 0 || task->data[2] == 2)
task->data[3] = -task->data[3];
}
LoadPalette(gUnknown_083EDDA0[task->data[2]], 0x10, 0x20);
}
static void GameplayTask_PikaPower(void)
{
sSlotMachine->unkTaskPointer3E = CreateTask(sub_81040E8, 8);
}
static void DisplayPikaPower(u8 pikaPower)
{
struct Task *task = &gTasks[sSlotMachine->unkTaskPointer3E];
ClearTaskDataFields_2orHigher(task);
task->data[0] = 1;
task->data[1]++;
task->data[15] = 1; // points to a reelIndex
}
static void sub_8104098(void)
{
struct Task *task = &gTasks[sSlotMachine->unkTaskPointer3E];
ClearTaskDataFields_2orHigher(task);
task->data[0] = 3;
task->data[15] = 1; // points to a reelIndex
}
static bool8 sub_81040C8(void)
{
return gTasks[sSlotMachine->unkTaskPointer3E].data[15];
}
static void sub_81040E8(u8 taskId)
{
gUnknown_083ECBB4[gTasks[taskId].data[0]](&gTasks[taskId]);
}
static void nullsub_68(struct Task *task)
{
}
static void sub_810411C(struct Task *task)
{
task->data[2] = sub_8105B1C((task->data[1] << 3) + 20, 20);
task->data[0]++;
}
static void sub_8104144(struct Task *task)
{
if (gSprites[task->data[2]].data[7])
{
s16 r5 = task->data[1] + 2;
s16 r3 = 0;
s16 r2 = 0;
if (task->data[1] == 1)
r3 = 1, r2 = 1;
else if (task->data[1] == 16)
r3 = 2, r2 = 2;
sSelectedPikaPowerTile[r2] = pikaPowerTileTable[r3][0];
LoadBgTilemap(2, &sSelectedPikaPowerTile[r2], 2, r5 + 0x40);
sub_8105B88(task->data[2]);
task->data[0] = 0;
task->data[15] = 0; // points to a reelIndex
}
}
static void sub_81041AC(struct Task *task)
{
s16 r5 = task->data[1] + 2;
s16 r3 = 0;
s16 r2 = 3;
if (task->data[1] == 1)
r3 = 1, r2 = 1;
else if (task->data[1] == 16)
r3 = 2, r2 = 2;
if (task->data[2] == 0)
{
sSelectedPikaPowerTile[r2] = pikaPowerTileTable[r3][1];
LoadBgTilemap(2, &sSelectedPikaPowerTile[r2], 2, r5 + 0x40);
task->data[1]--;
}
if (++task->data[2] >= 20)
task->data[2] = 0;
if (task->data[1] == 0)
{
task->data[0] = 0;
task->data[15] = 0;
}
}
static void ClearTaskDataFields_2orHigher(struct Task *task)
{
u8 i;
for (i = 2; i < 16; i++)
task->data[i] = 0;
}
// possibly load tiles for pika power meter
static void sub_810423C(u8 pikaPower)
{
s16 i;
s16 r3 = 0, r1 = 0;
s16 r4 = 3;
for (i = 0; i < pikaPower; i++, r4++)
{
r3 = 0, r1 = 0;
if (i == 0)
r3 = 1, r1 = 1;
else if (i == 15) // pikaPower meter is full
r3 = 2, r1 = 2;
sSelectedPikaPowerTile[r1] = pikaPowerTileTable[r3][0];
LoadBgTilemap(2, &sSelectedPikaPowerTile[r1], 2, r4 + 0x40);
}
for (; i < 16; i++, r4++)
{
r3 = 0, r1 = 3;
if (i == 0)
r3 = 1, r1 = 1;
else if (i == 15)
r3 = 2, r1 = 2;
sSelectedPikaPowerTile[r1] = pikaPowerTileTable[r3][1];
LoadBgTilemap(2, &sSelectedPikaPowerTile[r1], 2, r4 + 0x40);
}
gTasks[sSlotMachine->unkTaskPointer3E].data[1] = pikaPower;
}
static void BeginReeltime(void)
{
u8 taskId = CreateTask(RunReeltimeActions, 7);
RunReeltimeActions(taskId);
}
static bool8 IsFinalTask_RunReelTimeActions(void)
{
if (FindTaskIdByFunc(RunReeltimeActions) == TAIL_SENTINEL)
return TRUE;
return FALSE;
}
static void RunReeltimeActions(u8 taskId)
{
// task.data[0] points to which ReelTimeAction to do, and starts at 0
// task.data[1] has something to do with the threshold
// task.data[4] says how many pixels to advance the reel
// task.data[5] is a timer
ReeltimeActions[gTasks[taskId].data[0]](&gTasks[taskId]);
}
static void ReeltimeAction0(struct Task *task)
{
sSlotMachine->fairRollsLeft = 0;
sSlotMachine->reeltimePixelOffset = 0;
sSlotMachine->reeltimePosition = 0;
task->data[0]++;
task->data[1] = 0;
task->data[2] = 30;
task->data[4] = 1280; // reel speed
gSpriteCoordOffsetX = 0;
gSpriteCoordOffsetY = 0;
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
sub_8104A40(REG_OFFSET_BG3VOFS, 0);
sub_81051C0();
sub_8105100();
sub_81052EC();
sub_81053A0();
sub_810545C();
GetReeltimeDraw();
StopMapMusic();
PlayNewMapMusic(MUS_BD_TIME);
}
static void ReeltimeAction1(struct Task *task)
{
s16 r3;
gSpriteCoordOffsetX -= 8;
task->data[1] += 8;
r3 = ((task->data[1] + 240) & 0xff) >> 3;
SetGpuReg(REG_OFFSET_BG1HOFS, task->data[1] & 0x1ff);
if (r3 != task->data[2] && task->data[3] <= 18)
{
task->data[2] = r3;
task->data[3] = task->data[1] >> 3;
sub_8104A40(r3, task->data[3]);
}
if (task->data[1] >= 200)
{
task->data[0]++;
task->data[3] = 0;
}
AdvanceReeltimeReel(task->data[4] >> 8);
}
static void ReeltimeAction2(struct Task *task)
{
AdvanceReeltimeReel(task->data[4] >> 8);
if (++task->data[5] >= 60)
{
task->data[0]++;
CreateReelTimeSprites1();
CreateReelTimeSprite2();
}
}
static void ReeltimeAction3(struct Task *task)
{
int r5;
u8 sp0[ARRAY_COUNT(gUnknown_085A75C0)];
s16 sp4[ARRAY_COUNT(gUnknown_085A75C4)];
s16 spC[ARRAY_COUNT(gUnknown_085A75CC)];
memcpy(sp0, gUnknown_085A75C0, sizeof(gUnknown_085A75C0));
memcpy(sp4, gUnknown_085A75C4, sizeof(gUnknown_085A75C4));
memcpy(spC, gUnknown_085A75CC, sizeof(gUnknown_085A75CC));
AdvanceReeltimeReel(task->data[4] >> 8);
// gradually slow down the reel
task->data[4] -= 4;
r5 = 4 - (task->data[4] >> 8);
sub_8105688(sp4[r5]);
sub_81057E8(spC[r5]);
StartSpriteAnimIfDifferent(gSprites + sSlotMachine->reelTimeSprite3F, sp0[r5]);
// once speed goes below 256, go to next ReelTimeAction and keep the speed level
if (task->data[4] <= 0x100)
{
task->data[0]++;
task->data[4] = 0x100;
task->data[5] = 0;
}
}
static void ReeltimeAction4(struct Task *task)
{
AdvanceReeltimeReel(task->data[4] >> 8);
if (++task->data[5] >= 80)
{
task->data[0]++;
task->data[5] = 0;
sub_81057E8(2);
StartSpriteAnimIfDifferent(gSprites + sSlotMachine->reelTimeSprite3F, 3);
}
}
static void ReeltimeAction5(struct Task *task)
{
AdvanceReeltimeReel(task->data[4] >> 8);
task->data[4] = (u8)task->data[4] + 0x80;
if (++task->data[5] >= 80)
{
task->data[0]++;
task->data[5] = 0;
}
}
static void ReeltimeAction6(struct Task *task)
{
AdvanceReeltimeReel(task->data[4] >> 8);
task->data[4] = (u8)task->data[4] + 0x40;
if (++task->data[5] >= 40)
{
task->data[5] = 0;
if (sSlotMachine->reelTimeDraw)
{
if (sSlotMachine->fairRollsLeft <= task->data[6])
task->data[0]++;
}
else if (task->data[6] > 3)
{
task->data[0]++;
}
else if (SkipToReeltimeAction14(task->data[6]))
{
task->data[0] = 14;
}
task->data[6]++;
}
}
static void ReelTimeAction_LandOnOutcome(struct Task *task)
{
s16 reeltimePixelOffset = sSlotMachine->reeltimePixelOffset % 20;
if (reeltimePixelOffset)
{
reeltimePixelOffset = AdvanceReeltimeReelToNextTag(task->data[4] >> 8);
task->data[4] = (u8)task->data[4] + 0x40;
}
else if (GetNearbyReelTimeTag(1) != sSlotMachine->reelTimeDraw)
{
AdvanceReeltimeReel(task->data[4] >> 8);
reeltimePixelOffset = sSlotMachine->reeltimePixelOffset % 20;
task->data[4] = (u8)task->data[4] + 0x40;
}
if (reeltimePixelOffset == 0 && GetNearbyReelTimeTag(1) == sSlotMachine->reelTimeDraw)
{
task->data[4] = 0; // stop moving
task->data[0]++;
}
}
static void ReeltimeAction8(struct Task *task)
{
if (++task->data[4] >= 60)
{
StopMapMusic();
sub_81056C0();
sub_8105804();
task->data[0]++;
if(sSlotMachine->reelTimeDraw == 0)
{
task->data[4] = 0xa0;
StartSpriteAnimIfDifferent(gSprites + sSlotMachine->reelTimeSprite3F, 5);
PlayFanfare(MUS_ME_ZANNEN);
}
else
{
task->data[4] = 0xc0;
StartSpriteAnimIfDifferent(gSprites + sSlotMachine->reelTimeSprite3F, 4);
gSprites[sSlotMachine->reelTimeSprite3F].animCmdIndex = 0;
if (sSlotMachine->pikaPower)
{
sub_8104098();
sSlotMachine->pikaPower = 0;
}
PlayFanfare(MUS_ME_B_SMALL);
}
}
}
static void ReeltimeAction9(struct Task *task)
{
if ((task->data[4] == 0 || --task->data[4] == 0) && !sub_81040C8())
task->data[0]++;
}
static void ReeltimeAction10(struct Task *task)
{
s16 r4;
gSpriteCoordOffsetX -= 8;
task->data[1] += 8;
task->data[3] += 8;
r4 = ((task->data[1] - 8) & 0xff) >> 3;
SetGpuReg(REG_OFFSET_BG1HOFS, task->data[1] & 0x1ff);
if (task->data[3] >> 3 <= 25)
sub_8104A88(r4);
else
task->data[0]++;
}
static void ReeltimeAction11(struct Task *task)
{
sSlotMachine->fairRollsUsed = 0;
sSlotMachine->fairRollsLeft = sSlotMachine->reelTimeDraw;
gSpriteCoordOffsetX = 0;
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
sSlotMachine->reelIncrement = 8;
sub_810514C();
sub_81054B8();
sub_8105524();
PlayNewMapMusic(sSlotMachine->backupMapMusic);
if (sSlotMachine->fairRollsLeft == 0)
{
DestroyTask(FindTaskIdByFunc(RunReeltimeActions));
}
else
{
sub_8104CAC(4);
task->data[1] = SlowReelSpeed();
task->data[2] = 0;
task->data[3] = 0;
task->data[0]++;
}
}
static void ReeltimeAction12(struct Task *task)
{
if (sSlotMachine->reelIncrement == task->data[1])
task->data[0]++;
else if (sSlotMachine->reelPixelOffsets[0] % 24 == 0 && (++task->data[2]& 0x07) == 0)
sSlotMachine->reelIncrement >>= 1;
}
static void ReeltimeAction13(struct Task *task)
{
if (sub_8104E18())
DestroyTask(FindTaskIdByFunc(RunReeltimeActions));
}
static void ReeltimeAction14(struct Task *task)
{
sub_81054B8();
sub_81056C0();
sub_8105804();
sub_8105854();
gSprites[sSlotMachine->unk4E[0]].invisible = TRUE;
StartSpriteAnimIfDifferent(gSprites + sSlotMachine->reelTimeSprite3F, 5);
task->data[0]++;
task->data[4] = 4;
task->data[5] = 0;
StopMapMusic();
PlayFanfare(MUS_ME_ZANNEN);
PlaySE(SE_W153);
}
static void ReeltimeAction15(struct Task *task)
{
gSpriteCoordOffsetY = task->data[4];
SetGpuReg(REG_OFFSET_BG1VOFS, task->data[4]);
if (task->data[5] & 0x01)
task->data[4] = -task->data[4];
if ((++task->data[5] & 0x1f) == 0)
task->data[4] >>= 1;
if (task->data[4] == 0)
{
sub_81058A0();
sub_81058C4();
sub_8105284_();
sub_81059E8();
gSprites[sSlotMachine->unk4E[0]].invisible = FALSE;
task->data[0]++;
task->data[5] = 0;
}
}
static void ReeltimeAction16(struct Task *task)
{
gSpriteCoordOffsetY = 0;
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
if (sub_8105ACC())
{
task->data[0]++;
sub_8105AEC();
}
}
static void ReeltimeAction17(struct Task *task)
{
gSpriteCoordOffsetX = 0;
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
PlayNewMapMusic(sSlotMachine->backupMapMusic);
sub_810514C();
sub_8105554();
sub_8105524();
sub_81059B8();
DestroyTask(FindTaskIdByFunc(RunReeltimeActions));
}
static void sub_8104A40(s16 a0, s16 a1)
{
s16 i;
for (i = 4; i < 15; i++)
{
LoadBgTilemap(1, &gReelTimeWindowTilemap[a1 + (i - 4) * 20], 2, 32 * i + a0);
}
}
static void sub_8104A88(s16 a0)
{
u8 i;
for (i = 4; i < 15; i++)
{
LoadBgTilemap(1, gUnknown_085A9898, 2, 32 * i + a0);
}
}
static void OpenInfoBox(u8 seemsUnused)
{
u8 taskId = CreateTask(RunInfoBoxActions, 1);
gTasks[taskId].data[1] = seemsUnused;
RunInfoBoxActions(taskId);
}
static bool8 IsInfoBoxClosed(void)
{
if (FindTaskIdByFunc(RunInfoBoxActions) == 0xFF)
return TRUE;
else
return FALSE;
}
static void RunInfoBoxActions(u8 taskId)
{
InfoBoxActions[gTasks[taskId].data[0]](&gTasks[taskId]);
}
static void InfoBox_FadeIn(struct Task *task)
{
BeginNormalPaletteFade(0xFFFFFFFF, 0, 0, 16, RGB(0, 0, 0));
task->data[0]++;
}
static void InfoBox_WaitForFade(struct Task *task)
{
if (!gPaletteFade.active)
task->data[0]++;
}
static void InfoBox_8104B80(struct Task *task)
{
sub_8104DA4();
sub_81065DC();
AddWindow(&gUnknown_085A7444);
PutWindowTilemap(1);
FillWindowPixelBuffer(1, PIXEL_FILL(0));
task->data[0]++;
}
static void InfoBox_AddText(struct Task *task)
{
AddTextPrinterParameterized3(1, 1, 2, 5, gColors_ReeltimeHelp, 0, gText_ReelTimeHelp);
CopyWindowToVram(1, 3);
BeginNormalPaletteFade(0xFFFFFFFF, 0, 16, 0, RGB(0, 0, 0));
task->data[0]++;
}
static void InfoBox_AwaitPlayerInput(struct Task *task)
{
if (gMain.newKeys & (B_BUTTON | SELECT_BUTTON))
{
FillWindowPixelBuffer(1, PIXEL_FILL(0));
ClearWindowTilemap(1);
CopyWindowToVram(1, 1);
RemoveWindow(1);
BeginNormalPaletteFade(0xFFFFFFFF, 0, 0, 16, RGB(0, 0, 0));
task->data[0]++;
}
}
static void InfoBox_812DE14(struct Task *task)
{
sub_812F968();
ShowBg(3);
task->data[0]++;
}
static void InfoBox_812DE30(struct Task *task)
{
sub_8104CAC(task->data[1]);
task->data[0]++;
}
static void InfoBox_8104BFC(struct Task *task)
{
sub_810423C(sSlotMachine->pikaPower);
BeginNormalPaletteFade(0xFFFFFFFF, 0, 16, 0, RGB(0, 0, 0));
task->data[0]++;
}
static void InfoBox_FreeTask(struct Task *task)
{
DestroyTask(FindTaskIdByFunc(RunInfoBoxActions));
}
static void sub_8104C5C(void)
{
u8 i;
struct Task *task;
i = CreateTask(sub_8104E74_, 3);
sSlotMachine->unkTaskPointer3D = i;
task = &gTasks[i];
task->data[1] = -1;
for (i = 4; i < 16; i++)
task->data[i] = MAX_SPRITES;
}
// possibly initialize each wheel
static void sub_8104CAC(u8 arg0)
{
u8 i;
struct Task *task;
sub_8104DA4();
task = &gTasks[sSlotMachine->unkTaskPointer3D];
task->data[1] = arg0;
for (i = 0; gUnknown_083ED048[arg0][i].unk00 != 0xFF; i++)
{
u8 spriteId;
spriteId = sub_8105BB4(
gUnknown_083ED048[arg0][i].unk00,
gUnknown_083ED048[arg0][i].unk01,
gUnknown_083ED048[arg0][i].unk02
);
task->data[4 + i] = spriteId;
}
}
static void sub_8104D30(u8 a0, SpriteCallback a1, s16 a2, s16 a3, s16 a4)
{
u8 i;
struct Task *task = &gTasks[sSlotMachine->unkTaskPointer3D];
for (i = 4; i < 16; i++)
{
if (task->data[i] == MAX_SPRITES)
{
task->data[i] = sub_8105BF8(a0, a1, a2, a3, a4);
break;
}
}
}
static void sub_8104DA4(void)
{
u8 i;
struct Task *task = &gTasks[sSlotMachine->unkTaskPointer3D];
if ((u16)task->data[1] != 0xFFFF)
gUnknown_083ED064[task->data[1]]();
for (i = 4; i < 16; i++)
{
if (task->data[i] != MAX_SPRITES)
{
DestroySprite(&gSprites[task->data[i]]);
task->data[i] = MAX_SPRITES;
}
}
}
static bool8 sub_8104E18(void)
{
u8 i;
struct Task *task = &gTasks[sSlotMachine->unkTaskPointer3D];
for (i = 4; i < 16; i++)
{
if (task->data[i] != MAX_SPRITES)
{
if (gSprites[task->data[i]].data[7])
return FALSE;
}
}
return TRUE;
}
static void sub_8104E74_(u8 taskId)
{
gUnknown_083ECC54[gTasks[taskId].data[0]](&gTasks[taskId]);
}
static void nullsub_69(struct Task *task)
{
}
static void sub_8104EA8(void)
{
s16 i;
s16 j;
s16 x;
for (i = 0, x = 0x30; i < 3; i++, x += 0x28)
{
for (j = 0; j < 120; j += 24)
{
struct Sprite *sprite = gSprites + CreateSprite(&gSpriteTemplate_83ED414, x, 0, 14);
sprite->oam.priority = 3;
sprite->data[0] = i;
sprite->data[1] = j;
sprite->data[3] = -1;
}
}
}
/*static */void sub_8104F18(struct Sprite *sprite)
{
sprite->data[2] = sSlotMachine->reelPixelOffsets[sprite->data[0]] + sprite->data[1];
sprite->data[2] %= 120;
sprite->pos1.y = sSlotMachine->reelPixelOffsetsWhileStopping[sprite->data[0]] + 28 + sprite->data[2];
sprite->sheetTileStart = GetSpriteTileStartByTag(GetNearbyTag_Quantized(sprite->data[0], sprite->data[2] / 24));
SetSpriteSheetFrameTileNum(sprite);
}
static void sub_8104F8C(void)
{
s16 i;
s16 x;
for (x = 203, i = 1; i < 10000; i *= 10, x -= 7)
sub_8104FF4(x, 23, 0, i);
for (x = 235, i = 1; i < 10000; i *= 10, x -= 7)
sub_8104FF4(x, 23, 1, i);
}
static void sub_8104FF4(s16 x, s16 y, u8 a2, s16 a3)
{
struct Sprite *sprite = gSprites + CreateSprite(&gSpriteTemplate_83ED42C, x, y, 13);
sprite->oam.priority = 2;
sprite->data[0] = a2;
sprite->data[1] = a3;
sprite->data[2] = a3 * 10;
sprite->data[3] = -1;
}
/*static */void sub_810506C(struct Sprite *sprite)
{
u16 tag = sSlotMachine->coins;
if (sprite->data[0])
tag = sSlotMachine->payout;
if (sprite->data[3] != tag)
{
sprite->data[3] = tag;
tag %= (u16)sprite->data[2];
tag /= (u16)sprite->data[1];
tag += 7;
sprite->sheetTileStart = GetSpriteTileStartByTag(tag);
SetSpriteSheetFrameTileNum(sprite);
}
}
static void sub_81050C4(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED444, 0x58, 0x48, 15);
gSprites[spriteId].oam.priority = 3;
SetSubspriteTables(gSprites + spriteId, gSubspriteTables_83ED704);
}
static void sub_8105100(void)
{
struct SpriteTemplate spriteTemplate;
u8 spriteId;
if (sUnknown_0203AAE4 == NULL)
sUnknown_0203AAE4 = AllocZeroed(sizeof(struct SpriteFrameImage) * 5);
sUnknown_0203AAE4[0].data = sUnknown_0203AAD8 + (0 * 0x800);
sUnknown_0203AAE4[0].size = 0x800;
sUnknown_0203AAE4[1].data = sUnknown_0203AAD8 + (1 * 0x800);
sUnknown_0203AAE4[1].size = 0x800;
sUnknown_0203AAE4[2].data = sUnknown_0203AAD8 + (2 * 0x800);
sUnknown_0203AAE4[2].size = 0x800;
sUnknown_0203AAE4[3].data = sUnknown_0203AAD8 + (3 * 0x800);
sUnknown_0203AAE4[3].size = 0x800;
sUnknown_0203AAE4[4].data = sUnknown_0203AAD8 + (4 * 0x800);
sUnknown_0203AAE4[4].size = 0x800;
spriteTemplate = gSpriteTemplate_83ED45C;
spriteTemplate.images = sUnknown_0203AAE4;
spriteId = CreateSprite(&spriteTemplate, 280, 80, 1);
gSprites[spriteId].oam.priority = 1;
gSprites[spriteId].coordOffsetEnabled = TRUE;
sSlotMachine->reelTimeSprite3F = spriteId;
}
static void sub_810514C(void)
{
DestroySprite(gSprites + sSlotMachine->reelTimeSprite3F);
if (sUnknown_0203AAE4 != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAE4);
}
/*static */void sub_8105170(struct Sprite *sprite)
{
sprite->pos2.y = sprite->pos2.x = 0;
if (sprite->animNum == 4)
{
sprite->pos2.y = sprite->pos2.x = 8;
if ((sprite->animCmdIndex != 0 && sprite->animDelayCounter != 0) || (sprite->animCmdIndex == 0 && sprite->animDelayCounter == 0))
sprite->pos2.y = -8;
}
}
static void sub_81051C0(void)
{
struct SpriteTemplate spriteTemplate;
u8 spriteId;
struct Sprite *sprite;
if (sUnknown_0203AAE8 == NULL)
sUnknown_0203AAE8 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AAE8[0].data = sUnknown_0203AAD8 + 0x2800;
sUnknown_0203AAE8[0].size = 0x300;
spriteTemplate = gSpriteTemplate_83ED474;
spriteTemplate.images = sUnknown_0203AAE8;
spriteId = CreateSprite(&spriteTemplate, 368, 52, 7);
sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->coordOffsetEnabled = TRUE;
SetSubspriteTables(sprite, gSubspriteTables_83ED73C);
sSlotMachine->unk49[0] = spriteId;
if (sUnknown_0203AAEC == NULL)
sUnknown_0203AAEC = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AAEC[0].data = sUnknown_0203AAD8 + 0x2800 + 0x300;
sUnknown_0203AAEC[0].size = 0x500;
spriteTemplate = gSpriteTemplate_83ED48C;
spriteTemplate.images = sUnknown_0203AAEC;
spriteId = CreateSprite(&spriteTemplate, 368, 84, 7);
sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->coordOffsetEnabled = TRUE;
SetSubspriteTables(sprite, gSubspriteTables_83ED75C);
sSlotMachine->unk49[1] = spriteId;
}
static void sub_8105284_(void)
{
struct SpriteTemplate spriteTemplate;
u8 spriteId;
struct Sprite *sprite;
if (sUnknown_0203AAF0 == NULL)
sUnknown_0203AAF0 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AAF0[0].data = sUnknown_0203AAD8 + 0x3000;
sUnknown_0203AAF0[0].size = 0x600;
spriteTemplate = gSpriteTemplate_83ED4A4;
spriteTemplate.images = sUnknown_0203AAF0;
spriteId = CreateSprite(&spriteTemplate, 0xa8 - gSpriteCoordOffsetX, 0x50, 7);
sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->coordOffsetEnabled = TRUE;
SetSubspriteTables(sprite, gSubspriteTables_83ED78C);
sSlotMachine->unk42 = spriteId;
}
static void sub_81052EC(void)
{
u8 i;
s16 r5;
for (i = 0, r5 = 0; i < 3; i++, r5 += 20)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED4BC, 0x170, 0, 10);
struct Sprite *sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->coordOffsetEnabled = TRUE;
sprite->data[7] = r5;
sSlotMachine->unk4B[i] = spriteId;
}
}
/*static */void sub_810535C(struct Sprite *sprite)
{
s16 r0 = (u16)(sSlotMachine->reeltimePixelOffset + sprite->data[7]);
r0 %= 40;
sprite->pos1.y = r0 + 59;
StartSpriteAnimIfDifferent(sprite, GetNearbyReelTimeTag(r0 / 20));
}
static void sub_81053A0(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED4D4, 0x170, 0x64, 9);
struct Sprite *sprite = &gSprites[spriteId];
sprite->coordOffsetEnabled = TRUE;
sprite->oam.priority = 1;
SetSubspriteTables(sprite, gSubspriteTables_83ED7B4);
sSlotMachine->unk4E[0] = spriteId;
spriteId = CreateSprite(&gSpriteTemplate_83ED4D4, 0x120, 0x68, 4);
sprite = &gSprites[spriteId];
sprite->coordOffsetEnabled = TRUE;
sprite->oam.priority = 1;
SetSubspriteTables(sprite, gSubspriteTables_83ED7B4);
sSlotMachine->unk4E[1] = spriteId;
}
static void sub_810545C(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED4EC, 0x170, 0x4c, 11);
struct Sprite *sprite = &gSprites[spriteId];
sprite->coordOffsetEnabled = TRUE;
sprite->oam.priority = 1;
SetSubspriteTables(sprite, gSubspriteTables_83ED7D4);
sSlotMachine->unk40 = spriteId;
}
static void sub_81054B8(void)
{
u8 i;
DestroySprite(&gSprites[sSlotMachine->unk40]);
for (i = 0; i < 2; i++)
DestroySprite(&gSprites[sSlotMachine->unk49[i]]);
if (sUnknown_0203AAE8 != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAE8);
if (sUnknown_0203AAEC != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAEC);
for (i = 0; i < 3; i++)
DestroySprite(&gSprites[sSlotMachine->unk4B[i]]);
}
static void sub_8105524(void)
{
u8 i;
for (i = 0; i < 2; i++)
DestroySprite(&gSprites[sSlotMachine->unk4E[i]]);
}
static void sub_8105554(void)
{
DestroySprite(&gSprites[sSlotMachine->unk42]);
if (sUnknown_0203AAF0 != NULL)
FREE_AND_SET_NULL(sUnknown_0203AAF0);
}
// TODO: check if this is true
static void CreateReelTimeSprites1(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED504, 0x98, 0x20, 5);
struct Sprite *sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->hFlip = TRUE;
sSlotMachine->reelTimeSprites1[0] = spriteId;
sprite->data[0] = 8;
sprite->data[1] = -1;
sprite->data[2] = -1;
sprite->data[7] = 0x20;
spriteId = CreateSprite(&gSpriteTemplate_83ED504, 0xb8, 0x20, 5);
sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sSlotMachine->reelTimeSprites1[1] = spriteId;
sprite->data[1] = 1;
sprite->data[2] = -1;
sprite->data[7] = 0x20;
}
/*static */void sub_810562C(struct Sprite *sprite)
{
if (sprite->data[0] != 0)
{
sprite->data[0]--;
sprite->pos2.x = 0;
sprite->pos2.y = 0;
sprite->invisible = TRUE;
}
else
{
sprite->invisible = FALSE;
sprite->pos2.x += sprite->data[1];
sprite->pos2.y += sprite->data[2];
if (++sprite->data[3] >= 8)
{
sprite->data[0] = sprite->data[7];
sprite->data[3] = 0;
}
}
}
static void sub_8105688(s16 a0)
{
gSprites[sSlotMachine->reelTimeSprites1[0]].data[7] = a0;
gSprites[sSlotMachine->reelTimeSprites1[1]].data[7] = a0;
}
static void sub_81056C0(void)
{
u8 i;
for (i = 0; i < 2; i++)
DestroySprite(&gSprites[sSlotMachine->reelTimeSprites1[i]]);
}
static void CreateReelTimeSprite2(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED51C, 0x48, 0x50, 3);
gSprites[spriteId].oam.priority = 1;
gSprites[spriteId].data[0] = 1;
gSprites[spriteId].data[5] = 0;
gSprites[spriteId].data[6] = 16;
gSprites[spriteId].data[7] = 8;
sSlotMachine->reelTimeSprites2[0] = spriteId;
spriteId = CreateSprite(&gSpriteTemplate_83ED51C, 0x68, 0x50, 3);
gSprites[spriteId].oam.priority = 1;
gSprites[spriteId].hFlip = TRUE;
sSlotMachine->reelTimeSprites2[1] = spriteId;
}
/*static */void sub_8105784(struct Sprite *sprite)
{
u8 sp[] = {16, 0};
if (sprite->data[0] && --sprite->data[6] <= 0)
{
MultiplyInvertedPaletteRGBComponents((IndexOfSpritePaletteTag(7) << 4) + 0x103, sp[sprite->data[5]], sp[sprite->data[5]], sp[sprite->data[5]]);
++sprite->data[5];
sprite->data[5] &= 1;
sprite->data[6] = sprite->data[7];
}
}
static void sub_81057E8(s16 a0)
{
gSprites[sSlotMachine->reelTimeSprites2[0]].data[7] = a0;
}
static void sub_8105804(void)
{
u8 i;
MultiplyInvertedPaletteRGBComponents((IndexOfSpritePaletteTag(7) << 4) + 0x103, 0, 0, 0);
for (i = 0; i < 2; i++)
DestroySprite(&gSprites[sSlotMachine->reelTimeSprites2[i]]);
}
static void sub_8105854(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED534, 0xa8, 0x50, 6);
gSprites[spriteId].oam.priority = 1;
sSlotMachine->unk41 = spriteId;
}
/*static */void sub_8105894(struct Sprite *sprite)
{
sprite->pos2.y = gSpriteCoordOffsetY;
}
static void sub_81058A0(void)
{
DestroySprite(&gSprites[sSlotMachine->unk41]);
}
static void sub_81058C4(void)
{
u8 i;
u16 sp[] = {0x0, 0x40, 0x80, 0xC0};
for (i = 0; i < 4; i++)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED54C, 0x50 - gSpriteCoordOffsetX, 0x44, 0);
struct Sprite *sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->coordOffsetEnabled = TRUE;
sprite->data[0] = sp[i];
sSlotMachine->unk54[i] = spriteId;
}
}
/*static */void sub_810594C(struct Sprite *sprite)
{
sprite->data[0] -= 2;
sprite->data[0] &= 0xff;
sprite->pos2.x = Cos(sprite->data[0], 20);
sprite->pos2.y = Sin(sprite->data[0], 6);
sprite->subpriority = 0;
if (sprite->data[0] >= 0x80)
{
sprite->subpriority = 2;
}
if (++sprite->data[1] >= 16)
{
sprite->hFlip ^= 1;
sprite->data[1] = 0;
}
}
static void sub_81059B8(void)
{
u8 i;
for (i = 0; i < 4; i++)
{
DestroySprite(&gSprites[sSlotMachine->unk54[i]]);
}
}
static void sub_81059E8(void)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED564, 0xa8, 0x3c, 8);
struct Sprite *sprite = &gSprites[spriteId];
sprite->oam.priority = 1;
sprite->oam.affineMode = ST_OAM_AFFINE_DOUBLE;
InitSpriteAffineAnim(sprite);
sSlotMachine->unk43 = spriteId;
}
/*static */void sub_8105A38(struct Sprite *sprite)
{
if (sprite->data[0] == 0)
{
if (sprite->affineAnimEnded)
sprite->data[0]++;
}
else if (sprite->data[0] == 1)
{
sprite->invisible ^= 1;
if (++sprite->data[2] >= 24)
{
sprite->data[0]++;
sprite->data[2] = 0;
}
}
else
{
sprite->invisible = TRUE;
if (++sprite->data[2] >= 16)
sprite->data[7] = 1;
}
sprite->data[1] &= 0xff;
sprite->data[1] += 16;
sprite->pos2.y -= (sprite->data[1] >> 8);
}
static u8 sub_8105ACC(void)
{
return gSprites[sSlotMachine->unk43].data[7];
}
static void sub_8105AEC(void)
{
struct Sprite *sprite = &gSprites[sSlotMachine->unk43];
FreeOamMatrix(sprite->oam.matrixNum);
DestroySprite(sprite);
}
static u8 sub_8105B1C(s16 x, s16 y)
{
u8 spriteId = CreateSprite(&gSpriteTemplate_83ED6CC, x, y, 12);
struct Sprite *sprite = &gSprites[spriteId];
sprite->oam.priority = 2;
sprite->oam.affineMode = ST_OAM_AFFINE_DOUBLE;
InitSpriteAffineAnim(sprite);
return spriteId;
}
/*static */void sub_8105B70(struct Sprite *sprite)
{
if (sprite->affineAnimEnded)
sprite->data[7] = 1;
}
static void sub_8105B88(u8 spriteId)
{
struct Sprite *sprite = &gSprites[spriteId];
FreeOamMatrix(sprite->oam.matrixNum);
DestroySprite(sprite);
}
static u8 sub_8105BB4(u8 templateIdx, u8 cbAndCoordsIdx, s16 a2)
{
return sub_8105BF8(templateIdx, gUnknown_083ECF0C[cbAndCoordsIdx], gUnknown_083ECE7E[cbAndCoordsIdx][0], gUnknown_083ECE7E[cbAndCoordsIdx][1], a2);
}
static u8 sub_8105BF8(u8 templateIdx, SpriteCallback callback, s16 x, s16 y, s16 a4)
{
struct SpriteTemplate spriteTemplate;
u8 spriteId;
struct Sprite *sprite;
spriteTemplate = *gUnknown_083EDB5C[templateIdx];
spriteTemplate.images = gUnknown_03001188[templateIdx];
spriteId = CreateSprite(&spriteTemplate, x, y, 16);
sprite = &gSprites[spriteId];
sprite->oam.priority = 3;
sprite->callback = callback;
sprite->data[6] = a4;
sprite->data[7] = 1;
if (gUnknown_083EDBC4[templateIdx])
SetSubspriteTables(sprite, gUnknown_083EDBC4[templateIdx]);
return spriteId;
}
static void sub_8105C64(struct Sprite *sprite)
{
sprite->data[7] = 0;
}
static void sub_8105C6C(struct Sprite *sprite)
{
s16 sp0[] = {4, -4, 4, -4};
s16 sp8[] = {4, 4, -4, -4};
if (sprite->data[1]++ >= 16)
{
sprite->subspriteTableNum ^= 1;
sprite->data[1] = 0;
}
sprite->pos2.x = 0;
sprite->pos2.y = 0;
if (sprite->subspriteTableNum != 0)
{
sprite->pos2.x = sp0[sprite->data[6]];
sprite->pos2.y = sp8[sprite->data[6]];
}
}
static void sub_8105CF0(struct Sprite *sprite)
{
sprite->hFlip = TRUE;
sub_8105C6C(sprite);
}
static void sub_8105D08(struct Sprite *sprite)
{
sprite->vFlip = TRUE;
sub_8105C6C(sprite);
}
static void sub_8105D20(struct Sprite *sprite)
{
sprite->hFlip = TRUE;
sprite->vFlip = TRUE;
sub_8105C6C(sprite);
}
static void sub_8105D3C(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
sprite->pos1.x += 4;
if (sprite->pos1.x >= 0xd0)
{
sprite->pos1.x = 0xd0;
sprite->data[0]++;
}
break;
case 1:
if (++sprite->data[1] > 90)
sprite->data[0]++;
break;
case 2:
sprite->pos1.x += 4;
if (sprite->pos1.x >= 0x110)
sprite->data[0]++;
break;
case 3:
sprite->data[7] = 0;
break;
}
}
static void sub_8105DA4(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
sprite->pos1.x -= 4;
if (sprite->pos1.x <= 0xd0)
{
sprite->pos1.x = 0xd0;
sprite->data[0]++;
}
break;
case 1:
if (++sprite->data[1] > 90)
sprite->data[0]++;
break;
case 2:
sprite->pos1.x -= 4;
if (sprite->pos1.x <= 0x90)
sprite->data[0]++;
break;
case 3:
sprite->data[7] = 0;
break;
}
}
static void sub_8105E08(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
StartSpriteAnim(sprite, sSlotMachine->fairRollsLeft - 1);
sprite->data[0]++;
// fallthrough
case 1:
if (++sprite->data[1] >= 4)
{
sprite->data[0]++;
sprite->data[1] = 0;
}
break;
case 2:
sprite->pos1.x += 4;
if (sprite->pos1.x >= 0xd0)
{
sprite->pos1.x = 0xd0;
sprite->data[0]++;
}
break;
case 3:
if (++sprite->data[1] > 90)
sprite->data[0]++;
break;
case 4:
sprite->pos1.x += 4;
if (sprite->pos1.x >= 0xf8)
sprite->data[0]++;
break;
case 5:
sprite->data[7] = 0;
break;
}
}
static void sub_8105EB4(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
sprite->animPaused = TRUE;
sprite->data[0]++;
// fallthrough
case 1:
sprite->pos1.y += 8;
if (sprite->pos1.y >= 0x70)
{
sprite->pos1.y = 0x70;
sprite->data[1] = 16;
sprite->data[0]++;
}
break;
case 2:
if (sprite->data[2] == 0)
{
sprite->pos1.y -= sprite->data[1];
sprite->data[1] = -sprite->data[1];
if (++sprite->data[3] >= 2)
{
sprite->data[1] >>= 2;
sprite->data[3] = 0;
if (sprite->data[1] == 0)
{
sprite->data[0]++;
sprite->data[7] = 0;
sprite->animPaused = FALSE;
}
}
}
sprite->data[2]++;
sprite->data[2] &= 0x07;
break;
}
}
static void sub_8105F54(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
if (++sprite->data[1] > 8)
sprite->data[0]++;
break;
case 1:
sprite->pos1.y += 2;
if (sprite->pos1.y >= 0x30)
{
sprite->pos1.y = 0x30;
sprite->data[0]++;
sprite->data[7] = 0;
}
break;
}
}
static void sub_8105F9C(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
sprite->invisible = TRUE;
if (++sprite->data[1] > 0x20)
{
sprite->data[0]++;
sprite->data[1] = 5;
sprite->oam.mosaic = TRUE;
sprite->invisible = FALSE;
StartSpriteAnim(sprite, 1);
SetGpuReg(REG_OFFSET_MOSAIC, ((sprite->data[1] << 4) | sprite->data[1]) << 8);
}
break;
case 1:
sprite->data[1] -= (sprite->data[2] >> 8);
if (sprite->data[1] < 0)
sprite->data[1] = 0;
SetGpuReg(REG_OFFSET_MOSAIC, ((sprite->data[1] << 4) | sprite->data[1]) << 8);
sprite->data[2] &= 0xff;
sprite->data[2] += 0x80;
if (sprite->data[1] == 0)
{
sprite->data[0]++;
sprite->data[7] = 0;
sprite->oam.mosaic = FALSE;
StartSpriteAnim(sprite, 0);
}
break;
}
}
static void sub_8106058(struct Sprite *sprite)
{
if (sprite->data[1] < 3)
{
LoadPalette(gUnknown_083EDE10[sprite->data[1]], (IndexOfSpritePaletteTag(6) << 4) + 0x100, 0x20);
if (++sprite->data[2] >= 4)
{
sprite->data[1]++;
sprite->data[2] = 0;
}
}
else
{
LoadPalette(gUnknown_083EDE10[sprite->data[1]], (IndexOfSpritePaletteTag(6) << 4) + 0x100, 0x20);
if (++sprite->data[2] >= 25)
{
sprite->data[1] = 0;
sprite->data[2] = 0;
}
}
StartSpriteAnimIfDifferent(sprite, 1);
sprite->data[7] = 0;
}
static void sub_81060FC(struct Sprite *sprite)
{
s16 sp00[] = {0, -40, 0, 0, 48, 0, 24, 0};
s16 sp10[] = {-32, 0, -32, -48, 0, -48, 0, -48};
s16 sp20[] = {16, 12, 16, 0, 0, 4, 8, 8};
switch (sprite->data[0])
{
case 0:
sprite->pos2.x = sp00[sprite->data[6]];
sprite->pos2.y = sp10[sprite->data[6]];
sprite->data[1] = sp20[sprite->data[6]];
sprite->data[0]++;
// fallthrough
case 1:
if (sprite->data[1]-- == 0)
sprite->data[0]++;
break;
case 2:
if (sprite->pos2.x > 0)
sprite->pos2.x -= 4;
else if (sprite->pos2.x < 0)
sprite->pos2.x += 4;
if (sprite->pos2.y > 0)
sprite->pos2.y -= 4;
else if (sprite->pos2.y < 0)
sprite->pos2.y += 4;
if (sprite->pos2.x == 0 && sprite->pos2.y == 0)
sprite->data[0]++;
break;
}
}
static void sub_81061C8(struct Sprite *sprite)
{
s16 sp0[] = {160, 192, 224, 104, 80, 64, 48, 24};
if (sprite->data[0] == 0)
{
sprite->data[0]++;
sprite->data[1] = 12;
}
sprite->pos2.x = Cos(sp0[sprite->data[6]], sprite->data[1]);
sprite->pos2.y = Sin(sp0[sprite->data[6]], sprite->data[1]);
if (sprite->data[1] != 0)
sprite->data[1]--;
}
static void sub_8106230(struct Sprite *sprite)
{
switch (sprite->data[0])
{
case 0:
sSlotMachine->winIn = 0x2f;
sSlotMachine->winOut = 0x3f;
sSlotMachine->win0v = 0x2088;
sprite->invisible = TRUE;
sprite->data[0]++;
// fallthrough
case 1:
sprite->data[1] += 2;
sprite->data[2] = sprite->data[1] + 0xb0;
sprite->data[3] = 0xf0 - sprite->data[1];
if (sprite->data[2] > 0xd0)
sprite->data[2] = 0xd0;
if (sprite->data[3] < 0xd0)
sprite->data[3] = 0xd0;
sSlotMachine->win0h = (sprite->data[2] << 8) | sprite->data[3];
if (sprite->data[1] > 0x33)
{
sprite->data[0]++;
sSlotMachine->winIn = 0x3f;
}
break;
case 2:
if (sSlotMachine->bet == 0)
break;
sub_8104D30(5, SpriteCallbackDummy, 0xd0, 0x74, 0);
sSlotMachine->win0h = 0xc0e0;
sSlotMachine->win0v = 0x6880;
sSlotMachine->winIn = 0x2f;
sprite->data[0]++;
sprite->data[1] = 0;
// fallthrough
case 3:
sprite->data[1] += 2;
sprite->data[2] = sprite->data[1] + 0xc0;
sprite->data[3] = 0xe0 - sprite->data[1];
if (sprite->data[2] > 0xd0)
sprite->data[2] = 0xd0;
if (sprite->data[3] < 0xd0)
sprite->data[3] = 0xd0;
sSlotMachine->win0h = (sprite->data[2] << 8) | sprite->data[3];
if (sprite->data[1] > 0x0f)
{
sprite->data[0]++;
sSlotMachine->winIn = 0x3f;
}
break;
}
}
static void nullsub_70(void)
{
}
static void sub_8106364(void)
{
SetGpuReg(REG_OFFSET_MOSAIC, 0);
}
static void sub_8106370(void)
{
LoadPalette(gUnknown_083EDE20, (IndexOfSpritePaletteTag(6) << 4) + 0x100, 0x20);
}
static void sub_810639C(void)
{
sSlotMachine->win0h = 0xf0;
sSlotMachine->win0v = 0xa0;
sSlotMachine->winIn = 0x3f;
sSlotMachine->winOut = 0x3f;
}
static void sub_81063C0(void)
{
u8 i;
sub_8106404();
sUnknown_0203AAD4 = Alloc(0x3200);
LZDecompressWram(gSlotMachineReelTime_Gfx, sUnknown_0203AAD4);
sUnknown_0203AAD8 = Alloc(0x3600);
LZDecompressWram(gReelTimeGfx, sUnknown_0203AAD8);
sUnknown_0203AB30 = AllocZeroed(sizeof(struct SpriteSheet) * 22);
for (i = 0; i < 22; i++)
{
sUnknown_0203AB30[i].data = gSlotMachineSpriteSheets[i].data;
sUnknown_0203AB30[i].size = gSlotMachineSpriteSheets[i].size;
sUnknown_0203AB30[i].tag = gSlotMachineSpriteSheets[i].tag;
}
sUnknown_0203AB30[17].data = sUnknown_0203AAD4 + 0xA00;
sUnknown_0203AB30[18].data = sUnknown_0203AAD4 + 0x1400;
sUnknown_0203AB30[19].data = sUnknown_0203AAD4 + 0x1600;
sUnknown_0203AB30[20].data = sUnknown_0203AAD4 + 0x1900;
LoadSpriteSheets(sUnknown_0203AB30);
LoadSpritePalettes(gSlotMachineSpritePalettes);
}
static void sub_8106404(void)
{
u8 *dest;
u8 i, j;
sUnknown_0203AB2C = AllocZeroed(sizeof(struct SpriteSheet));
sUnknown_0203AAE0 = AllocZeroed(0x2000);
dest = sUnknown_0203AAE0;
for (i = 0; i < 0x40; i++)
{
for (j = 0; j < 0x20; j++, dest++)
*dest = gUnknown_083EDCDC[j];
}
sUnknown_0203AB2C->data = sUnknown_0203AAE0;
sUnknown_0203AB2C->size = 0x800;
sUnknown_0203AB2C->tag = 0x11;
LoadSpriteSheet(sUnknown_0203AB2C);
}
extern const u32 gSlotMachineMenu_Gfx[];
extern const u16 gSlotMachineMenu_Tilemap[];
extern const u16 gUnknown_08DCEC70[];
extern const u16 gSlotMachineMenu_Pal[];
static void sub_8106448(void)
{
sUnknown_0203AAC8 = Alloc(0x2200);
LZDecompressWram(gSlotMachineMenu_Gfx, sUnknown_0203AAC8);
LoadBgTiles(2, sUnknown_0203AAC8, 0x2200, 0);
LoadPalette(gSlotMachineMenu_Pal, 0, 160);
LoadPalette(gPalette_83EDE24, 208, 32);
}
static void sub_81064B8(void)
{
sub_812F968();
LoadSlotMachineWheelOverlay();
}
static void sub_812F968(void)
{
LoadBgTilemap(2, gSlotMachineMenu_Tilemap, 0x500, 0);
}
static void LoadSlotMachineWheelOverlay(void)
{
s16 x, y, dx;
for (x = 4; x < 18; x += 5)
{
for (dx = 0; dx < 4; dx++)
{
LoadBgTilemap(3, sUnknown_0203AAD0, 2, x + dx + 5 * 32);
LoadBgTilemap(3, sUnknown_0203AAD0 + 1, 2, x + dx + 13 * 32);
LoadBgTilemap(3, sUnknown_0203AAD0 + 2, 2, x + dx + 6 * 32);
LoadBgTilemap(3, sUnknown_0203AAD0 + 3, 2, x + dx + 12 * 32);
}
LoadBgTilemap(3, sUnknown_0203AAD0 + 4, 2, x + 6 * 32);
LoadBgTilemap(3, sUnknown_0203AAD0 + 5, 2, x + 12 * 32);
for (y = 7; y <= 11; y++)
LoadBgTilemap(3, sUnknown_0203AAD0 + 6, 2, x + y * 32);
}
}
static void sub_81065A8(s16 arg0, u16 arg1, u16 arg2, u16 arg3, u16 arg4)
{
sUnknown_0203AADC[0] = arg1;
sUnknown_0203AADC[1] = arg2;
sUnknown_0203AADC[2] = arg3;
sUnknown_0203AADC[3] = arg4;
LoadBgTilemap(2, sUnknown_0203AADC, 2, 15 * 32 + arg0);
LoadBgTilemap(2, sUnknown_0203AADC + 1, 2, 15 * 32 + 1 + arg0);
LoadBgTilemap(2, sUnknown_0203AADC + 2, 2, 16 * 32 + arg0);
LoadBgTilemap(2, sUnknown_0203AADC + 3, 2, 16 * 32 + 1 + arg0);
}
static void sub_81065DC(void)
{
LoadBgTilemap(2, gUnknown_08DCEC70, 0x500, 0);
HideBg(3);
}
static void SlotMachineSetup_9_0(void)
{
gUnknown_03001188[0] = sUnknown_0203AAF4;
gUnknown_03001188[1] = sUnknown_0203AAF8;
gUnknown_03001188[2] = sUnknown_0203AAFC;
gUnknown_03001188[3] = sUnknown_0203AB04;
gUnknown_03001188[4] = sUnknown_0203AB08;
gUnknown_03001188[5] = sUnknown_0203AB18;
gUnknown_03001188[6] = sUnknown_0203AB1C;
gUnknown_03001188[7] = sUnknown_0203AB20;
gUnknown_03001188[8] = sUnknown_0203AB24;
gUnknown_03001188[9] = sUnknown_0203AB28;
gUnknown_03001188[10] = sUnknown_0203AB00;
gUnknown_03001188[11] = sUnknown_0203AB00;
gUnknown_03001188[12] = sUnknown_0203AB00;
gUnknown_03001188[13] = sUnknown_0203AB00;
gUnknown_03001188[14] = sUnknown_0203AB0C;
gUnknown_03001188[15] = sUnknown_0203AB0C;
gUnknown_03001188[16] = sUnknown_0203AB0C;
gUnknown_03001188[17] = sUnknown_0203AB0C;
gUnknown_03001188[18] = sUnknown_0203AB0C;
gUnknown_03001188[19] = sUnknown_0203AB10;
gUnknown_03001188[20] = sUnknown_0203AB10;
gUnknown_03001188[21] = sUnknown_0203AB10;
gUnknown_03001188[22] = sUnknown_0203AB14;
gUnknown_03001188[23] = sUnknown_0203AB14;
gUnknown_03001188[24] = sUnknown_0203AB14;
gUnknown_03001188[25] = NULL;
}
static void SlotMachineSetup_8_0(void)
{
sUnknown_0203AAF4 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AAF4[0].data = sUnknown_0203AAD4;
sUnknown_0203AAF4[0].size = 0x600;
sUnknown_0203AAF8 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AAF8[0].data = sUnknown_0203AAD4 + 0x600;
sUnknown_0203AAF8[0].size = 0x200;
sUnknown_0203AAFC = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AAFC[0].data = sUnknown_0203AAD4 + 0x800;
sUnknown_0203AAFC[0].size = 0x200;
sUnknown_0203AB00 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB00[0].data = sUnknown_0203AAD4 + 0xA00;
sUnknown_0203AB00[0].size = 0x200;
sUnknown_0203AB04 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB04[0].data = sUnknown_0203AAD4 + 0xC00;
sUnknown_0203AB04[0].size = 0x300;
sUnknown_0203AB08 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB08[0].data = sUnknown_0203AAD4 + 0x1000;
sUnknown_0203AB08[0].size = 0x400;
sUnknown_0203AB0C = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB0C[0].data = sUnknown_0203AAD4 + 0x1400;
sUnknown_0203AB0C[0].size = 0x200;
sUnknown_0203AB10 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB10[0].data = sUnknown_0203AAD4 + 0x1600;
sUnknown_0203AB10[0].size = 0x300;
sUnknown_0203AB14 = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB14[0].data = sUnknown_0203AAD4 + 0x1900;
sUnknown_0203AB14[0].size = 0x300;
sUnknown_0203AB18 = AllocZeroed(sizeof(struct SpriteFrameImage) * 2);
sUnknown_0203AB18[0].data = sUnknown_0203AAD4 + 0x1C00;
sUnknown_0203AB18[0].size = 0x200;
sUnknown_0203AB18[1].data = sUnknown_0203AAD4 + 0x1E00;
sUnknown_0203AB18[1].size = 0x200;
sUnknown_0203AB1C = AllocZeroed(sizeof(struct SpriteFrameImage) * 1);
sUnknown_0203AB1C[0].data = sUnknown_0203AAD4 + 0x2000;
sUnknown_0203AB1C[0].size = 640;
sUnknown_0203AB20 = AllocZeroed(sizeof(struct SpriteFrameImage) * 5);
sUnknown_0203AB20[0].data = sUnknown_0203AAD4 + 0x2280;
sUnknown_0203AB20[0].size = 0x80;
sUnknown_0203AB20[1].data = sUnknown_0203AAD4 + 0x2300;
sUnknown_0203AB20[1].size = 0x80;
sUnknown_0203AB20[2].data = sUnknown_0203AAD4 + 0x2380;
sUnknown_0203AB20[2].size = 0x80;
sUnknown_0203AB20[3].data = sUnknown_0203AAD4 + 0x2400;
sUnknown_0203AB20[3].size = 0x80;
sUnknown_0203AB20[4].data = sUnknown_0203AAD4 + 0x2480;
sUnknown_0203AB20[4].size = 0x80;
sUnknown_0203AB24 = AllocZeroed(sizeof(struct SpriteFrameImage) * 2);
sUnknown_0203AB24[0].data = sUnknown_0203AAD4 + 0x2600;
sUnknown_0203AB24[0].size = 0x480;
sUnknown_0203AB24[1].data = sUnknown_0203AAD4 + 10880;
sUnknown_0203AB24[1].size = 0x480;
sUnknown_0203AB28 = AllocZeroed(sizeof(struct SpriteFrameImage) * 2);
sUnknown_0203AB28[0].data = sUnknown_0203AAD4 + 0x2F00;
sUnknown_0203AB28[0].size = 0x180;
sUnknown_0203AB28[1].data = sUnknown_0203AAD4 + 0x3080;
sUnknown_0203AB28[1].size = 0x180;
}
const u8 gReelSymbols[][REEL_NUM_TAGS] =
{
{
SLOT_MACHINE_TAG_7_RED,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_7_BLUE,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_7_RED,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_7_BLUE,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_REPLAY
},
{
SLOT_MACHINE_TAG_7_RED,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_7_BLUE,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_CHERRY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_CHERRY
},
{
SLOT_MACHINE_TAG_7_RED,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_7_BLUE,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_AZURILL,
SLOT_MACHINE_TAG_POWER,
SLOT_MACHINE_TAG_REPLAY,
SLOT_MACHINE_TAG_LOTAD,
SLOT_MACHINE_TAG_CHERRY
},
};
const u8 gReelTimeTags[] = {
1, 0, 5, 4, 3, 2
};
const s16 gInitialReelPositions[][2] = {
{0, 6},
{0, 10},
{0, 2}
};
const u8 gLuckyRoundProbabilities[][3] = {
{1, 1, 12},
{1, 1, 14},
{2, 2, 14},
{2, 2, 14},
{2, 3, 16},
{3, 3, 16}
};
const u8 gLuckyFlagProbabilities_Top3[][6] = {
{25, 25, 30, 40, 40, 50},
{25, 25, 30, 30, 35, 35},
{25, 25, 30, 25, 25, 30}
};
const u8 gLuckyFlagProbabilities_NotTop3[][6] = {
{20, 25, 25, 20, 25, 25},
{12, 15, 15, 18, 19, 22},
{25, 25, 25, 30, 30, 40},
{25, 25, 20, 20, 15, 15},
{40, 40, 35, 35, 40, 40}
};
const u8 gReeltimeProbabilities_UnluckyGame[][17] = {
{243, 243, 243, 80, 80, 80, 80, 40, 40, 40, 40, 40, 40, 5, 5, 5, 5},
{ 5, 5, 5, 150, 150, 150, 150, 130, 130, 130, 130, 130, 130, 100, 100, 100, 5},
{ 4, 4, 4, 20, 20, 20, 20, 80, 80, 80, 80, 80, 80, 100, 100, 100, 40},
{ 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 45, 45, 45, 100},
{ 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 5, 5, 5, 100},
{ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 6}
};
const u8 gReelTimeProbabilities_LuckyGame[][17] = {
{ 243, 243, 243, 200, 200, 200, 200, 160, 160, 160, 160, 160, 160, 70, 70, 70, 5},
{ 5, 5, 5, 25, 25, 25, 25, 5, 5, 5, 5, 5, 5, 2, 2, 2, 6},
{ 4, 4, 4, 25, 25, 25, 25, 30, 30, 30, 30, 30, 30, 40, 40, 40, 35},
{ 2, 2, 2, 3, 3, 3, 3, 30, 30, 30, 30, 30, 30, 100, 100, 100, 50},
{ 1, 1, 1, 2, 2, 2, 2, 30, 30, 30, 30, 30, 30, 40, 40, 40, 100},
{ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 4, 4, 4, 60}
};
const u16 gProbabilityTable_SkipToReeltimeAction14[] = {
128, 175, 200, 225, 256
};
const u16 gReelIncrementTable[][2] = {
{10, 5},
{10, 10},
{10, 15},
{10, 25},
{10, 35}
};
const u16 gReelTimeBonusIncrementTable[] = {
0, 5, 10, 15, 20
};
// tentative name
const u8 gBiasTags[] = {
SLOT_MACHINE_TAG_REPLAY, SLOT_MACHINE_TAG_CHERRY, SLOT_MACHINE_TAG_LOTAD, SLOT_MACHINE_TAG_AZURILL, SLOT_MACHINE_TAG_POWER, SLOT_MACHINE_TAG_7_RED, SLOT_MACHINE_TAG_7_RED, SLOT_MACHINE_TAG_7_RED
};
const u16 gLuckyFlagSettings_Top3[] = {
LUCKY_BIAS_777, LUCKY_BIAS_REELTIME, LUCKY_BIAS_MIXED_777
};
const u16 gLuckyFlagSettings_NotTop3[] = {
LUCKY_BIAS_POWER, LUCKY_BIAS_AZURILL, LUCKY_BIAS_LOTAD, LUCKY_BIAS_CHERRY, LUCKY_BIAS_REPLAY
};
const u8 gSym2Match[] = {
SLOT_MACHINE_MATCHED_777_RED,
SLOT_MACHINE_MATCHED_777_BLUE,
SLOT_MACHINE_MATCHED_AZURILL,
SLOT_MACHINE_MATCHED_LOTAD,
SLOT_MACHINE_MATCHED_1CHERRY,
SLOT_MACHINE_MATCHED_POWER,
SLOT_MACHINE_MATCHED_REPLAY
};
const u16 gSlotMatchFlags[] = {
1 << SLOT_MACHINE_MATCHED_1CHERRY,
1 << SLOT_MACHINE_MATCHED_2CHERRY,
1 << SLOT_MACHINE_MATCHED_REPLAY,
1 << SLOT_MACHINE_MATCHED_LOTAD,
1 << SLOT_MACHINE_MATCHED_AZURILL,
1 << SLOT_MACHINE_MATCHED_POWER,
1 << SLOT_MACHINE_MATCHED_777_MIXED,
1 << SLOT_MACHINE_MATCHED_777_RED,
1 << SLOT_MACHINE_MATCHED_777_BLUE
};
const u16 gSlotPayouts[] = {
2, 4, 0, 6, 12, 3, 90, 300, 300
};
const s16 gUnknown_083ECE7E[][2] = {
{ 0xd0, 0x38},
{ 0xb8, 0x00},
{ 0xc8, 0x08},
{ 0xd8, 0x10},
{ 0xe8, 0x18},
{ 0xd0, 0x48},
{ 0xd0, 0x08},
{ 0xd0, 0x40},
{ 0xd0, 0x38},
{ 0xc0, 0x58},
{ 0xe0, 0x58},
{ 0xc0, 0x78},
{ 0xe0, 0x78},
{ 0x90, 0x38},
{0x110, 0x58},
{ 0xa8, 0x70},
{ 0xd0, 0x54},
{ 0xd0, 0x70},
{ 0xbc, 0x34},
{ 0xd0, 0x34},
{ 0xe4, 0x34},
{ 0xb8, 0x48},
{ 0xc4, 0x48},
{ 0xd0, 0x48},
{ 0xdc, 0x48},
{ 0xe8, 0x48},
{ 0xbc, 0x34},
{ 0xd0, 0x34},
{ 0xe4, 0x34},
{ 0xb8, 0x48},
{ 0xc4, 0x48},
{ 0xd0, 0x48},
{ 0xdc, 0x48},
{ 0xe8, 0x48},
{ 0x00, 0x00}
};
const SpriteCallback gUnknown_083ECF0C[] = {
sub_8105C64,
sub_8105F54,
sub_8105F54,
sub_8105F54,
sub_8105F54,
sub_8105F9C,
sub_8105EB4,
sub_8105C64,
sub_8105C64,
sub_8105C6C,
sub_8105CF0,
sub_8105D08,
sub_8105D20,
sub_8105D3C,
sub_8105DA4,
sub_8105E08,
sub_8105C64,
sub_8106058,
sub_81060FC,
sub_81060FC,
sub_81060FC,
sub_81060FC,
sub_81060FC,
sub_81060FC,
sub_81060FC,
sub_81060FC,
sub_81061C8,
sub_81061C8,
sub_81061C8,
sub_81061C8,
sub_81061C8,
sub_81061C8,
sub_81061C8,
sub_81061C8,
sub_8106230
};
const struct UnkStruct1 Unknown_83ECF98[] = {
{25, 34, 0},
{2, 0, 0},
{9, 16, 0},
{255, 0, 0}
};
const struct UnkStruct1 Unknown_83ECFA8[] = {
{10, 1, 0},
{11, 2, 0},
{12, 3, 0},
{13, 4, 0},
{5, 5, 0},
{8, 6, 0},
{255, 0, 0}
};
const struct UnkStruct1 Unknown_83ECFC4[] = {
{3, 7, 0},
{8, 17, 0},
{255, 0, 0}
};
const struct UnkStruct1 Unknown_83ECFD0[] = {
{4, 8, 0},
{6, 9, 0},
{6, 10, 1},
{6, 11, 2},
{6, 12, 3},
{255, 0, 0}
};
const struct UnkStruct1 Unknown_83ECFE8[] = {
{0, 13, 0},
{1, 14, 0},
{7, 15, 0},
{255, 0, 0}
};
const struct UnkStruct1 Unknown_83ECFF8[] = {
{19, 26, 0},
{20, 27, 1},
{21, 28, 2},
{14, 29, 3},
{15, 30, 4},
{16, 31, 5},
{17, 32, 6},
{18, 33, 7},
{8, 17, 0},
{255, 0, 0}
};
const struct UnkStruct1 Unknown_83ED020[] = {
{22, 18, 0},
{23, 19, 1},
{24, 20, 2},
{14, 21, 3},
{15, 22, 4},
{16, 23, 5},
{17, 24, 6},
{18, 25, 7},
{8, 17, 0},
{255, 0, 0}
};
const struct UnkStruct1 *const gUnknown_083ED048[] = {
Unknown_83ECF98,
Unknown_83ECFA8,
Unknown_83ECFC4,
Unknown_83ECFD0,
Unknown_83ECFE8,
Unknown_83ED020,
Unknown_83ECFF8
};
void (*const gUnknown_083ED064[])(void) = {
sub_810639C,
sub_8106364,
sub_8106370,
nullsub_70,
nullsub_70,
sub_8106370,
sub_8106370
};