pokeemerald/src/berry_crush.c

3511 lines
104 KiB
C
Raw Normal View History

2019-03-31 09:05:32 -05:00
#include "global.h"
2020-02-10 00:47:00 -05:00
#include "battle_anim.h"
2019-12-12 10:16:05 +08:00
#include "berry.h"
2019-03-31 09:05:32 -05:00
#include "berry_powder.h"
#include "bg.h"
2019-12-12 10:16:05 +08:00
#include "decompress.h"
#include "dynamic_placeholder_text_util.h"
2019-03-31 09:05:32 -05:00
#include "event_data.h"
#include "gpu_regs.h"
#include "graphics.h"
2019-12-12 10:16:05 +08:00
#include "international_string_util.h"
2019-03-31 09:05:32 -05:00
#include "item_icon.h"
#include "item_menu.h"
#include "link.h"
#include "link_rfu.h"
#include "main.h"
2019-12-12 10:16:05 +08:00
#include "malloc.h"
2019-03-31 09:05:32 -05:00
#include "math_util.h"
#include "menu.h"
#include "overworld.h"
#include "palette.h"
2020-04-08 17:23:32 -04:00
#include "minigame_countdown.h"
2019-12-17 15:20:38 +08:00
#include "random.h"
2020-04-08 16:24:30 -04:00
#include "digit_obj_util.h"
2019-12-18 03:35:41 +08:00
#include "save.h"
2019-03-31 09:05:32 -05:00
#include "scanline_effect.h"
2019-12-12 10:16:05 +08:00
#include "script.h"
2019-03-31 09:05:32 -05:00
#include "sound.h"
#include "sprite.h"
#include "string_util.h"
2019-12-12 10:16:05 +08:00
#include "strings.h"
2019-03-31 09:05:32 -05:00
#include "task.h"
#include "text.h"
2019-12-12 10:16:05 +08:00
#include "text_window.h"
2019-03-31 09:05:32 -05:00
#include "trig.h"
#include "window.h"
#include "constants/items.h"
#include "constants/rgb.h"
#include "constants/songs.h"
2021-03-16 05:40:42 -04:00
#define MAX_TIME (10 * 60 * 60) // Timer can go up to 9:59:59
#define TAG_CRUSHER_BASE 1
#define PALTAG_EFFECT 2 // The next two gfx tags share this pal tag
#define GFXTAG_IMPACT 2
#define GFXTAG_SPARKLE 3
#define TAG_TIMER_DIGITS 4
#define TAG_PLAYER1_BERRY 5
#define TAG_PLAYER2_BERRY 6
#define TAG_PLAYER3_BERRY 7
#define TAG_PLAYER4_BERRY 8
#define TAG_PLAYER5_BERRY 9
2021-03-11 14:16:40 -05:00
#define TAG_COUNTDOWN 0x1000
2021-03-16 05:40:42 -04:00
#define CRUSHER_START_Y (-104)
enum {
RUN_CMD,
SCHEDULE_CMD,
};
// IDs for the main berry crush game functions
enum {
CMD_NONE,
CMD_FADE,
CMD_WAIT_FADE,
CMD_PRINT_MSG,
CMD_SHOW_GAME,
CMD_HIDE_GAME,
CMD_READY_BEGIN,
CMD_ASK_PICK_BERRY,
CMD_PICK_BERRY,
CMD_WAIT_BERRIES,
CMD_DROP_BERRIES,
CMD_DROP_LID,
CMD_COUNTDOWN,
CMD_PLAY_GAME_LEADER,
CMD_PLAY_GAME_MEMBER,
CMD_FINISH_GAME,
CMD_TIMES_UP,
CMD_CALC_RESULTS,
CMD_SHOW_RESULTS,
CMD_SAVE,
CMD_ASK_PLAY_AGAIN,
CMD_COMM_PLAY_AGAIN,
CMD_PLAY_AGAIN_YES,
CMD_PLAY_AGAIN_NO,
CMD_CLOSE_LINK,
CMD_QUIT,
};
enum {
MSG_PICK_BERRY,
MSG_WAIT_PICK,
MSG_POWDER,
MSG_SAVING,
MSG_PLAY_AGAIN,
MSG_NO_BERRIES,
MSG_DROPPED,
MSG_TIMES_UP,
MSG_COMM_STANDBY,
};
#define F_MSG_CLEAR (1 << 0)
#define F_MSG_EXPAND (1 << 1)
// Main states for the game. Many are assigned but never checked
enum {
STATE_INIT = 1,
STATE_RESET,
STATE_PICK_BERRY,
STATE_DROP_BERRIES,
STATE_DROP_LID,
STATE_COUNTDOWN,
STATE_PLAYING,
STATE_FINISHED,
STATE_TIMES_UP,
STATE_10, // Unused
STATE_RESULTS_PRESSES,
STATE_RESULTS_RANDOM,
STATE_RESULTS_CRUSHING,
STATE_14, // Unused
STATE_PLAY_AGAIN,
};
#define RESULTS_STATE_START STATE_RESULTS_PRESSES
#define RESULTS_STATE_END STATE_RESULTS_CRUSHING
// IDs for each results page that shows in succession at the game's end.
// Only 3 pages are shown for a given game. Presses and Crushing are always shown 1st and 3rd.
// The 2nd page is random, and can be rankings for either Neatness, Cooperative, or Power.
enum {
RESULTS_PAGE_PRESSES,
RESULTS_PAGE_RANDOM,
RESULTS_PAGE_CRUSHING,
NUM_RESULTS_PAGES,
};
// Random pages, see above
// "Neatness" is how many of the player's inputs were at a regular interval
// "Cooperative" is how often the player pressed A at the same time as others
// "Power" is how much of the time the player spent pressing A
enum {
RESULTS_PAGE_NEATNESS,
RESULTS_PAGE_COOPERATIVE,
RESULTS_PAGE_POWER,
NUM_RANDOM_RESULTS_PAGES
};
#define PLAY_AGAIN_YES 0
#define PLAY_AGAIN_NO 1
#define PLAY_AGAIN_NO_BERRIES 3
enum {
2021-04-09 22:39:34 -04:00
COLORID_GRAY,
2021-03-16 05:40:42 -04:00
COLORID_BLACK,
2021-04-09 22:39:34 -04:00
COLORID_LIGHT_GRAY,
2021-03-16 05:40:42 -04:00
COLORID_BLUE,
COLORID_GREEN,
COLORID_RED,
2019-03-31 09:05:32 -05:00
};
2021-03-16 05:40:42 -04:00
// Flags for the inputFlags field
// Field is 16 bits; 3 bits for each player, last bit is unused
// The first two bits are interchangeable
// Needlessly complicated system, the inputState field is sufficient by itself
#define F_INPUT_HIT_A (1 << 0)
#define F_INPUT_HIT_B (1 << 1)
#define F_INPUT_HIT_SYNC (1 << 2) // Input at same time as another player
#define INPUT_FLAGS_PER_PLAYER 3
#define INPUT_FLAG_MASK ((1 << INPUT_FLAGS_PER_PLAYER) - 1)
// Values for the inputState field
enum {
2021-03-16 05:40:42 -04:00
INPUT_STATE_NONE,
INPUT_STATE_HIT, // Hit the crusher
INPUT_STATE_HIT_SYNC, // Hit the crusher at same time as another player
2019-12-15 14:04:15 +08:00
};
2021-03-16 05:40:42 -04:00
// No reason for this to be 2
// Simply a flag for whether a given player has sent their data this round
// Data is only sent if the player is the leader or if they pressed A
#define SEND_GAME_STATE 2
struct BerryCrushGame_Player
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
u8 name[PLAYER_NAME_LENGTH + 1 + 4];
u16 berryId;
u16 inputTime;
u16 neatInputStreak;
u16 timeSincePrevInput;
u16 maxNeatInputStreak;
u16 numAPresses;
u16 numSyncedAPresses;
u16 timePressingA;
u8 inputFlags;
u8 inputState;
2019-12-15 14:04:15 +08:00
};
2021-03-16 05:40:42 -04:00
// This data is set locally and sent to the other players
struct BerryCrushGame_LocalState
{
u16 sendFlag;
bool8 endGame:1;
bool8 bigSparkle:1;
bool8 pushedAButton:1;
u8 playerPressedAFlags:5; // 1 bit for each player
s8 vibration;
u16 depth;
u16 timer;
u16 inputFlags;
u16 sparkleAmount;
2019-12-15 14:04:15 +08:00
};
2021-03-16 05:40:42 -04:00
// This data is read from another player's local state above by casting the recvCmd buffer
// It is identical with exception to the addition of rfuCmd
struct BerryCrushGame_LinkState
{
u16 rfuCmd;
u16 sendFlag;
bool8 endGame:1;
bool8 bigSparkle:1;
bool8 pushedAButton:1;
u8 playerPressedAFlags:5;
s8 vibration;
u16 depth;
u16 timer;
u16 inputFlags;
u16 sparkleAmount;
2019-12-15 14:04:15 +08:00
};
2021-03-16 05:40:42 -04:00
struct BerryCrushGame_Results
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
u32 powder;
u16 time;
u16 targetPressesPerSec; // Never read
u16 silkiness;
u16 totalAPresses;
u16 stats[2][MAX_RFU_PLAYERS];
u8 playerIdsRanked[2][MAX_RFU_PLAYERS + 3];
};
// playerIdsRanked above has 3 additional elements after the players.
// Only 1 of these 2*3 is ever used, and it stores the id for which
2021-03-16 05:40:42 -04:00
// random results page to show. Its define below is for readability.
#define randomPageId playerIdsRanked[0][7]
// Holds position data for various player-associated graphics
struct BerryCrushPlayerCoords
{
u8 playerId;
u8 windowGfxX;
u8 windowGfxY;
s16 impactXOffset;
s16 impactYOffset;
s16 berryXOffset;
s16 berryXDest;
2019-03-31 09:05:32 -05:00
};
2021-03-16 05:40:42 -04:00
struct BerryCrushGame_Gfx
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
u8 counter;
u8 vibrationIdx;
u8 numVibrations;
bool8 vibrating;
s16 minutes;
s16 secondsInt;
s16 secondsFrac;
2021-03-16 05:40:42 -04:00
const struct BerryCrushPlayerCoords *playerCoords[MAX_RFU_PLAYERS];
struct Sprite *coreSprite;
2021-03-16 05:40:42 -04:00
struct Sprite *impactSprites[MAX_RFU_PLAYERS];
struct Sprite *berrySprites[MAX_RFU_PLAYERS];
struct Sprite *sparkleSprites[11];
struct Sprite *timerSprites[2];
2021-03-16 05:40:42 -04:00
u8 resultsState;
u8 unused;
u8 resultsWindowId;
u8 nameWindowIds[MAX_RFU_PLAYERS];
2021-03-12 19:24:03 -06:00
u16 bgBuffers[4][0x800];
2019-03-31 09:05:32 -05:00
};
struct BerryCrushGame
{
2021-03-16 05:40:42 -04:00
MainCallback exitCallback;
u32 (*cmdCallback)(struct BerryCrushGame *, u8 *);
u8 localId;
u8 playerCount;
2021-03-16 05:40:42 -04:00
u8 taskId;
u8 textSpeed;
u8 cmdState;
2021-03-16 05:40:42 -04:00
u8 unused; // Never read
u8 nextCmd;
u8 afterPalFadeCmd;
2021-03-16 05:40:42 -04:00
u16 cmdTimer;
u16 gameState;
2021-03-16 05:40:42 -04:00
u16 playAgainState;
u16 pressingSpeed;
2021-03-16 05:40:42 -04:00
s16 targetAPresses;
s16 totalAPresses;
s32 powder;
2021-03-16 05:40:42 -04:00
s32 targetDepth;
u8 newDepth;
bool8 noRoomForPowder:1; // Never read
bool8 newRecord:1;
bool8 playedSound:1;
bool8 endGame:1;
bool8 bigSparkle:1;
u8 sparkleAmount:3;
u16 leaderTimer;
u16 timer;
s16 depth;
s16 vibration;
2021-03-16 05:40:42 -04:00
s16 bigSparkleCounter;
s16 numBigSparkles;
s16 numBigSparkleChecks;
s16 sparkleCounter;
u8 commandArgs[12];
u16 sendCmd[6];
u16 recvCmd[7];
2021-03-16 05:40:42 -04:00
struct BerryCrushGame_LocalState localState;
struct BerryCrushGame_Results results;
struct BerryCrushGame_Player players[MAX_RFU_PLAYERS];
struct BerryCrushGame_Gfx gfx;
2019-03-31 09:05:32 -05:00
};
2020-09-13 00:26:01 -04:00
static void VBlankCB(void);
static void MainCB(void);
static void MainTask(u8);
2021-03-16 05:40:42 -04:00
static void SetNamesAndTextSpeed(struct BerryCrushGame *);
static void RunOrScheduleCommand(u16, u8, u8 *);
static void SetPaletteFadeArgs(u8 *, bool8, u32, s8, u8, u8, u16);
static s32 UpdateGame(struct BerryCrushGame *);
static void CreatePlayerNameWindows(struct BerryCrushGame *);
static void DrawPlayerNameWindows(struct BerryCrushGame *);
static void CopyPlayerNameWindowGfxToBg(struct BerryCrushGame *);
static void CreateGameSprites(struct BerryCrushGame *);
static void DestroyGameSprites(struct BerryCrushGame *);
static void PrintTimer(struct BerryCrushGame_Gfx *, u16);
static void SpriteCB_Sparkle_Init(struct Sprite *);
static void HideTimer(struct BerryCrushGame_Gfx *);
static void ResetGame(struct BerryCrushGame *);
static void SetPrintMessageArgs(u8 *, u8, u8, u16, u8);
static void SpriteCB_Impact(struct Sprite *);
static u32 Cmd_BeginNormalPaletteFade(struct BerryCrushGame *, u8 *);
static u32 Cmd_WaitPaletteFade(struct BerryCrushGame *, u8 *);
static u32 Cmd_PrintMessage(struct BerryCrushGame *, u8 *);
static u32 Cmd_ShowGameDisplay(struct BerryCrushGame *, u8 *);
static u32 Cmd_HideGameDisplay(struct BerryCrushGame *, u8 *);
static u32 Cmd_SignalReadyToBegin(struct BerryCrushGame *, u8 *);
static u32 Cmd_AskPickBerry(struct BerryCrushGame *, u8 *);
static u32 Cmd_GoToBerryPouch(struct BerryCrushGame *, u8 *);
static u32 Cmd_WaitForOthersToPickBerries(struct BerryCrushGame *, u8 *);
static u32 Cmd_DropBerriesIntoCrusher(struct BerryCrushGame *, u8 *);
static u32 Cmd_DropLid(struct BerryCrushGame *, u8 *);
static u32 Cmd_Countdown(struct BerryCrushGame *, u8 *);
static u32 Cmd_PlayGame_Leader(struct BerryCrushGame *, u8 *);
static u32 Cmd_PlayGame_Member(struct BerryCrushGame *, u8 *);
static u32 Cmd_FinishGame(struct BerryCrushGame *, u8 *);
static u32 Cmd_HandleTimeUp(struct BerryCrushGame *, u8 *);
static u32 Cmd_TabulateResults(struct BerryCrushGame *, u8 *);
static u32 Cmd_ShowResults(struct BerryCrushGame *, u8 *);
static u32 Cmd_SaveGame(struct BerryCrushGame *, u8 *);
static u32 Cmd_AskPlayAgain(struct BerryCrushGame *, u8 *);
static u32 Cmd_CommunicatePlayAgainResponses(struct BerryCrushGame *, u8 *);
static u32 Cmd_PlayAgain(struct BerryCrushGame *, u8 *);
static u32 Cmd_StopGame(struct BerryCrushGame *, u8 *);
static u32 Cmd_CloseLink(struct BerryCrushGame *, u8 *);
static u32 Cmd_Quit(struct BerryCrushGame *, u8 *);
static EWRAM_DATA struct BerryCrushGame *sGame = NULL;
static const u8 sBitTable[] = {
1 << 0,
1 << 1,
1 << 2,
1 << 3,
1 << 4,
1 << 5,
1 << 6,
1 << 7
2021-03-16 05:40:42 -04:00
};
// Additional A presses are counted depending on the number of players
// The bonus of 5 is unobtainable
static const u8 sSyncPressBonus[MAX_RFU_PLAYERS] = { 0, 1, 2, 3, 5 };
ALIGNED(4)
static const s8 sIntroOutroVibrationData[][7] =
2020-02-10 00:47:00 -05:00
{
{ 4, 1, 0, -1, 0, 0, 0},
{ 4, 2, 0, -1, 0, 0, 0},
{ 4, 2, 0, -2, 0, 0, 0},
{ 6, 3, 1, -1, -3, -1, 0},
2020-02-10 00:47:00 -05:00
{ 6, 4, 1, -2, -4, -2, 0},
};
2021-03-16 05:40:42 -04:00
ALIGNED(4)
static const u8 sVibrationData[MAX_RFU_PLAYERS][4] =
2020-02-10 00:47:00 -05:00
{
{3, 2, 1, 0},
2020-02-10 00:47:00 -05:00
{3, 3, 1, 0},
{3, 3, 2, 0},
{3, 4, 2, 0},
{3, 5, 3, 0},
};
2021-03-16 05:40:42 -04:00
static const u8 *const sMessages[] =
{
[MSG_PICK_BERRY] = gText_ReadyPickBerry,
[MSG_WAIT_PICK] = gText_WaitForAllChooseBerry,
[MSG_POWDER] = gText_EndedWithXUnitsPowder,
[MSG_SAVING] = gText_RecordingGameResults,
[MSG_PLAY_AGAIN] = gText_PlayBerryCrushAgain,
[MSG_NO_BERRIES] = gText_YouHaveNoBerries,
[MSG_DROPPED] = gText_MemberDroppedOut,
[MSG_TIMES_UP] = gText_TimesUpNoGoodPowder,
[MSG_COMM_STANDBY] = gText_CommunicationStandby2,
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const struct BgTemplate sBgTemplates[4] =
2020-02-10 00:47:00 -05:00
{
{
.bg = 0,
.charBaseIndex = 2,
.mapBaseIndex = 15,
.screenSize = 0,
.paletteMode = 0,
.priority = 0,
.baseTile = 0,
},
{
.bg = 1,
.charBaseIndex = 0,
.mapBaseIndex = 13,
.screenSize = 2,
.paletteMode = 0,
.priority = 1,
.baseTile = 0,
2020-02-10 00:47:00 -05:00
},
{
.bg = 2,
.charBaseIndex = 0,
.mapBaseIndex = 12,
.screenSize = 0,
.paletteMode = 0,
.priority = 2,
.baseTile = 0,
2020-02-10 00:47:00 -05:00
},
{
.bg = 3,
.charBaseIndex = 0,
.mapBaseIndex = 11,
.screenSize = 0,
.paletteMode = 0,
.priority = 3,
.baseTile = 0,
2020-02-10 00:47:00 -05:00
},
};
2021-03-16 05:40:42 -04:00
static const u8 sTextColorTable[][3] =
2020-02-10 00:47:00 -05:00
{
2021-04-09 22:39:34 -04:00
[COLORID_GRAY] = {TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GRAY, TEXT_COLOR_LIGHT_GRAY},
[COLORID_BLACK] = {TEXT_COLOR_TRANSPARENT, TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GRAY},
[COLORID_LIGHT_GRAY] = {TEXT_COLOR_TRANSPARENT, TEXT_COLOR_LIGHT_GRAY, TEXT_COLOR_RED},
2021-03-16 05:40:42 -04:00
[COLORID_BLUE] = {TEXT_COLOR_WHITE, TEXT_COLOR_BLUE, TEXT_COLOR_LIGHT_BLUE},
[COLORID_GREEN] = {TEXT_COLOR_WHITE, TEXT_COLOR_GREEN, TEXT_COLOR_LIGHT_GREEN},
[COLORID_RED] = {TEXT_COLOR_WHITE, TEXT_COLOR_RED, TEXT_COLOR_LIGHT_RED},
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const struct WindowTemplate sWindowTemplate_Rankings =
2020-02-10 00:47:00 -05:00
{
.bg = 0,
.tilemapLeft = 3,
.tilemapTop = 4,
.width = 24,
.height = 13,
2020-02-10 00:47:00 -05:00
.paletteNum = 15,
.baseBlock = 1
};
2021-03-16 05:40:42 -04:00
static const struct WindowTemplate sWindowTemplates_PlayerNames[MAX_RFU_PLAYERS + 1] =
2020-02-10 00:47:00 -05:00
{
{
.bg = 0,
.tilemapLeft = 0,
.tilemapTop = 0,
.width = 9,
.height = 2,
.paletteNum = 8,
2020-02-10 00:47:00 -05:00
.baseBlock = 1005
},
{
.bg = 0,
.tilemapLeft = 0,
.tilemapTop = 3,
.width = 9,
.height = 2,
.paletteNum = 8,
2020-02-10 00:47:00 -05:00
.baseBlock = 987
},
{
.bg = 0,
.tilemapLeft = 0,
.tilemapTop = 6,
.width = 9,
.height = 2,
.paletteNum = 8,
2020-02-10 00:47:00 -05:00
.baseBlock = 969
},
{
.bg = 0,
.tilemapLeft = 21,
.tilemapTop = 3,
.width = 9,
.height = 2,
.paletteNum = 8,
2020-02-10 00:47:00 -05:00
.baseBlock = 951
},
{
.bg = 0,
.tilemapLeft = 21,
.tilemapTop = 6,
.width = 9,
.height = 2,
.paletteNum = 8,
2020-02-10 00:47:00 -05:00
.baseBlock = 933
},
DUMMY_WIN_TEMPLATE,
};
2021-03-16 05:40:42 -04:00
static const struct WindowTemplate sWindowTemplates_Results[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
[STATE_RESULTS_PRESSES - RESULTS_STATE_START] = {
.bg = 0,
.tilemapLeft = 5,
.tilemapTop = 2,
.width = 20,
.height = 16,
.paletteNum = 15,
2020-02-10 00:47:00 -05:00
.baseBlock = 1
},
2021-03-16 05:40:42 -04:00
[STATE_RESULTS_RANDOM - RESULTS_STATE_START] = {
.bg = 0,
.tilemapLeft = 5,
.tilemapTop = 2,
.width = 20,
.height = 16,
.paletteNum = 15,
2020-02-10 00:47:00 -05:00
.baseBlock = 1
},
2021-03-16 05:40:42 -04:00
[STATE_RESULTS_CRUSHING - RESULTS_STATE_START] = {
.bg = 0,
.tilemapLeft = 4,
.tilemapTop = 2,
.width = 22,
.height = 16,
.paletteNum = 15,
2020-02-10 00:47:00 -05:00
.baseBlock = 1
},
DUMMY_WIN_TEMPLATE,
};
2021-03-16 05:40:42 -04:00
// The height of the results window depending on the number of players
// 2 players, 3 players, 4 players, or 5 players
static const u8 sResultsWindowHeights[][MAX_RFU_PLAYERS - 1] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
{6, 8, 9, 11}, // "Presses" and "Neatness/Cooperative/Power" pages
{12, 14, 15, 16}, // "Crushing" page
2020-02-10 00:47:00 -05:00
};
static const u32 sPressingSpeedConversionTable[] =
2020-02-10 00:47:00 -05:00
{
50000000, // 50
25000000, // 25
12500000, // 12.5
6250000, // 6.25
3125000, // 3.125
1562500, // 1.5625
781250, // 0.78125
390625 // 0.390625
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const u16 sCrusherBase_Pal[] = INCBIN_U16("graphics/berry_crush/crusher_base.gbapal");
static const u16 sEffects_Pal[] = INCBIN_U16("graphics/berry_crush/effects.gbapal");
static const u16 sTimerDigits_Pal[] = INCBIN_U16("graphics/berry_crush/timer_digits.gbapal");
static const u32 sCrusherBase_Gfx[] = INCBIN_U32("graphics/berry_crush/crusher_base.4bpp.lz");
static const u32 sImpact_Gfx[] = INCBIN_U32("graphics/berry_crush/impact.4bpp.lz");
static const u32 sSparkle_Gfx[] = INCBIN_U32("graphics/berry_crush/sparkle.4bpp.lz");
static const u32 sTimerDigits_Gfx[] = INCBIN_U32("graphics/berry_crush/timer_digits.4bpp.lz");
static const u8 sCrusherTop_Tilemap[] = INCBIN_U8("graphics/berry_crush/crusher_top.bin.lz");
static const u8 sContainerCap_Tilemap[] = INCBIN_U8("graphics/berry_crush/container_cap.bin.lz");
static const u8 sBg_Tilemap[] = INCBIN_U8("graphics/berry_crush/bg.bin.lz");
// Takes the number of players - 2 and a player id and returns the
2021-03-16 05:40:42 -04:00
// index into sPlayerCoords where that player should be seated
static const u8 sPlayerIdToPosId[MAX_RFU_PLAYERS - 1][MAX_RFU_PLAYERS] =
2021-03-16 05:40:42 -04:00
{
{1, 3},
{0, 1, 3},
{1, 3, 2, 4},
2020-02-10 00:47:00 -05:00
{0, 1, 3, 2, 4},
};
2021-03-16 05:40:42 -04:00
static const struct BerryCrushPlayerCoords sPlayerCoords[MAX_RFU_PLAYERS] =
2020-02-10 00:47:00 -05:00
{
{
2021-03-16 05:40:42 -04:00
.playerId = 0,
.windowGfxX = 0,
.windowGfxY = 0,
.impactXOffset = 0,
.impactYOffset = -16,
.berryXOffset = 0,
.berryXDest = 0,
2020-02-10 00:47:00 -05:00
},
{
2021-03-16 05:40:42 -04:00
.playerId = 1,
.windowGfxX = 0,
.windowGfxY = 3,
.impactXOffset = -28,
.impactYOffset = -4,
.berryXOffset = -24,
.berryXDest = 16,
2020-02-10 00:47:00 -05:00
},
{
2021-03-16 05:40:42 -04:00
.playerId = 2,
.windowGfxX = 0,
.windowGfxY = 6,
.impactXOffset = -16,
.impactYOffset = 20,
.berryXOffset = -8,
.berryXDest = 16,
2020-02-10 00:47:00 -05:00
},
{
2021-03-16 05:40:42 -04:00
.playerId = 3,
.windowGfxX = 20,
.windowGfxY = 3,
.impactXOffset = 28,
.impactYOffset = -4,
.berryXOffset = 32,
.berryXDest = -8,
2020-02-10 00:47:00 -05:00
},
{
2021-03-16 05:40:42 -04:00
.playerId = 4,
.windowGfxX = 20,
.windowGfxY = 6,
.impactXOffset = 16,
.impactYOffset = 20,
.berryXOffset = 16,
.berryXDest = -8,
2020-02-10 00:47:00 -05:00
}
};
2021-03-16 05:40:42 -04:00
static const s8 sImpactCoords[][2] =
2020-02-10 00:47:00 -05:00
{
{ 0, 0},
{-1, 0},
{ 1, 1},
};
2021-03-16 05:40:42 -04:00
static const s8 sSparkleCoords[][2] =
2020-02-10 00:47:00 -05:00
{
{ 0, 0},
{-16, -4},
{ 16, -4},
{ -8, -2},
{ 8, -2},
{-24, -8},
{ 24, -8},
{-32, -12},
{ 32, -12},
{-40, -16},
{ 40, -16},
};
static const u16 sPlayerBerrySpriteTags[MAX_RFU_PLAYERS] =
2020-02-10 00:47:00 -05:00
{
TAG_PLAYER1_BERRY,
TAG_PLAYER2_BERRY,
TAG_PLAYER3_BERRY,
TAG_PLAYER4_BERRY,
2021-03-16 05:40:42 -04:00
TAG_PLAYER5_BERRY
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
// sTimerDigits_Gfx is part of this array but is (apparently) uncompressed
// It gets cast to raw uncompressed data when used in sDigitObjTemplates
static const struct CompressedSpriteSheet sSpriteSheets[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
{ .data = sCrusherBase_Gfx, .size = 0x800, .tag = TAG_CRUSHER_BASE },
{ .data = sImpact_Gfx, .size = 0xE00, .tag = GFXTAG_IMPACT },
{ .data = sSparkle_Gfx, .size = 0x700, .tag = GFXTAG_SPARKLE },
{ .data = sTimerDigits_Gfx, .size = 0x2C0, .tag = TAG_TIMER_DIGITS },
2020-02-10 00:47:00 -05:00
{}
};
static const struct SpritePalette sSpritePals[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
{ .data = sCrusherBase_Pal, .tag = TAG_CRUSHER_BASE },
{ .data = sEffects_Pal, .tag = PALTAG_EFFECT }, // For the impact and sparkle effects
{ .data = sTimerDigits_Pal, .tag = TAG_TIMER_DIGITS },
2020-02-10 00:47:00 -05:00
{}
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_CrusherBase[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(0, 0),
ANIMCMD_END
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_Impact_Small[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(0, 4),
ANIMCMD_FRAME(16, 4),
ANIMCMD_FRAME(32, 4),
ANIMCMD_END
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_Impact_Big[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(48, 2),
ANIMCMD_FRAME(64, 2),
ANIMCMD_FRAME(80, 2),
ANIMCMD_FRAME(96, 2),
ANIMCMD_END
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_Sparkle_Small[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(0, 2),
ANIMCMD_FRAME(4, 2),
2020-02-10 00:47:00 -05:00
ANIMCMD_FRAME(8, 2),
ANIMCMD_FRAME(12, 2),
ANIMCMD_FRAME(16, 2),
ANIMCMD_FRAME(20, 2),
2020-02-10 00:47:00 -05:00
ANIMCMD_JUMP(0)
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_Sparkle_Big[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(24, 4),
ANIMCMD_FRAME(28, 4),
ANIMCMD_FRAME(32, 4),
2020-02-10 00:47:00 -05:00
ANIMCMD_FRAME(36, 4),
ANIMCMD_FRAME(40, 4),
ANIMCMD_FRAME(44, 4),
ANIMCMD_FRAME(48, 4),
2020-02-10 00:47:00 -05:00
ANIMCMD_FRAME(52, 4),
ANIMCMD_JUMP(0)
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_Timer[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(20, 0),
ANIMCMD_END
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd sAnim_PlayerBerry[] =
2020-02-10 00:47:00 -05:00
{
ANIMCMD_FRAME(0, 0),
2020-02-10 00:47:00 -05:00
ANIMCMD_END
};
2021-03-16 05:40:42 -04:00
static const union AffineAnimCmd sAffineAnim_PlayerBerry_0[] =
2020-02-10 00:47:00 -05:00
{
AFFINEANIMCMD_FRAME(256, 256, 0, 0),
AFFINEANIMCMD_FRAME(0, 0, 2, 1),
AFFINEANIMCMD_JUMP(1)
};
2021-03-16 05:40:42 -04:00
static const union AffineAnimCmd sAffineAnim_PlayerBerry_1[] =
2020-02-10 00:47:00 -05:00
{
AFFINEANIMCMD_FRAME(256, 256, 0, 0),
AFFINEANIMCMD_FRAME(0, 0, -2, 1),
AFFINEANIMCMD_JUMP(1)
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd *const sAnims_CrusherBase[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
sAnim_CrusherBase
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd *const sAnims_Impact[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
sAnim_Impact_Small,
sAnim_Impact_Big,
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd *const sAnims_Sparkle[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
sAnim_Sparkle_Small,
sAnim_Sparkle_Big,
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd *const sAnims_Timer[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
sAnim_Timer
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const union AnimCmd *const sAnims_PlayerBerry[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
sAnim_PlayerBerry
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const union AffineAnimCmd *const sAffineAnims_PlayerBerry[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
sAffineAnim_PlayerBerry_0,
sAffineAnim_PlayerBerry_1,
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const struct SpriteTemplate sSpriteTemplate_CrusherBase =
2020-02-10 00:47:00 -05:00
{
.tileTag = TAG_CRUSHER_BASE,
.paletteTag = TAG_CRUSHER_BASE,
.oam = &gOamData_AffineOff_ObjNormal_64x64,
.anims = sAnims_CrusherBase,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2020-02-10 00:47:00 -05:00
.callback = SpriteCallbackDummy
};
2021-03-16 05:40:42 -04:00
static const struct SpriteTemplate sSpriteTemplate_Impact =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
.tileTag = GFXTAG_IMPACT,
.paletteTag = PALTAG_EFFECT,
.oam = &gOamData_AffineOff_ObjNormal_32x32,
.anims = sAnims_Impact,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2021-03-16 05:40:42 -04:00
.callback = SpriteCB_Impact
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static const struct SpriteTemplate sSpriteTemplate_Sparkle =
2020-02-10 00:47:00 -05:00
{
.tileTag = GFXTAG_SPARKLE,
.paletteTag = PALTAG_EFFECT,
.oam = &gOamData_AffineOff_ObjNormal_16x16,
.anims = sAnims_Sparkle,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2020-02-10 00:47:00 -05:00
.callback = SpriteCallbackDummy
};
2021-03-16 05:40:42 -04:00
static const struct SpriteTemplate sSpriteTemplate_Timer =
2020-02-10 00:47:00 -05:00
{
.tileTag = TAG_TIMER_DIGITS,
.paletteTag = TAG_TIMER_DIGITS,
.oam = &gOamData_AffineOff_ObjNormal_8x16,
.anims = sAnims_Timer,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
2020-02-10 00:47:00 -05:00
.callback = SpriteCallbackDummy
};
2021-02-27 15:49:43 -05:00
static const struct SpriteTemplate sSpriteTemplate_PlayerBerry =
2020-02-10 00:47:00 -05:00
{
.tileTag = TAG_PLAYER1_BERRY,
.paletteTag = TAG_PLAYER1_BERRY,
.oam = &gOamData_AffineDouble_ObjNormal_32x32,
.anims = sAnims_PlayerBerry,
.images = NULL,
.affineAnims = sAffineAnims_PlayerBerry,
2020-02-10 00:47:00 -05:00
.callback = SpriteCallbackDummy
};
static const struct DigitObjUtilTemplate sDigitObjTemplates[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
{ // Minutes
2020-04-08 16:24:30 -04:00
.strConvMode = 1,
2020-02-10 00:47:00 -05:00
.shape = 2,
.size = 0,
.priority = 0,
.oamCount = 2,
.xDelta = 8,
2020-02-10 00:47:00 -05:00
.x = 156,
.y = 0,
2022-07-29 10:52:35 -04:00
.spriteSheet = (void *) &sSpriteSheets[3],
.spritePal = &sSpritePals[2],
2020-02-10 00:47:00 -05:00
},
2021-03-16 05:40:42 -04:00
{ // Seconds
2020-04-08 16:24:30 -04:00
.strConvMode = 0,
2020-02-10 00:47:00 -05:00
.shape = 2,
.size = 0,
.priority = 0,
.oamCount = 2,
.xDelta = 8,
2020-02-10 00:47:00 -05:00
.x = 180,
.y = 0,
2022-07-29 10:52:35 -04:00
.spriteSheet = (void *) &sSpriteSheets[3],
.spritePal = &sSpritePals[2],
2020-02-10 00:47:00 -05:00
},
2021-03-16 05:40:42 -04:00
{ // 1/60ths of a second
2020-04-08 16:24:30 -04:00
.strConvMode = 0,
2020-02-10 00:47:00 -05:00
.shape = 2,
.size = 0,
.priority = 0,
.oamCount = 2,
.xDelta = 8,
2020-02-10 00:47:00 -05:00
.x = 204,
.y = 0,
2022-07-29 10:52:35 -04:00
.spriteSheet = (void *) &sSpriteSheets[3],
.spritePal = &sSpritePals[2],
2020-02-10 00:47:00 -05:00
}
};
2021-03-16 05:40:42 -04:00
static const u8 *const sResultsTexts[] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
[RESULTS_PAGE_PRESSES] = gText_SpaceTimes2, // " times"
[RESULTS_PAGE_RANDOM] = gText_XDotY, // "##.##", for Neatness, Cooperation, or Power value
[RESULTS_PAGE_CRUSHING] = gText_Var1Berry,
[RESULTS_PAGE_NEATNESS + NUM_RESULTS_PAGES] = gText_NeatnessRankings,
[RESULTS_PAGE_COOPERATIVE + NUM_RESULTS_PAGES] = gText_CoopRankings,
[RESULTS_PAGE_POWER + NUM_RESULTS_PAGES] = gText_PressingPowerRankings,
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
static u32 (*const sBerryCrushCommands[])(struct BerryCrushGame * game, u8 * data) =
{
[CMD_NONE] = NULL,
[CMD_FADE] = Cmd_BeginNormalPaletteFade,
[CMD_WAIT_FADE] = Cmd_WaitPaletteFade,
[CMD_PRINT_MSG] = Cmd_PrintMessage,
[CMD_SHOW_GAME] = Cmd_ShowGameDisplay,
[CMD_HIDE_GAME] = Cmd_HideGameDisplay,
[CMD_READY_BEGIN] = Cmd_SignalReadyToBegin,
[CMD_ASK_PICK_BERRY] = Cmd_AskPickBerry,
[CMD_PICK_BERRY] = Cmd_GoToBerryPouch,
[CMD_WAIT_BERRIES] = Cmd_WaitForOthersToPickBerries,
[CMD_DROP_BERRIES] = Cmd_DropBerriesIntoCrusher,
[CMD_DROP_LID] = Cmd_DropLid,
[CMD_COUNTDOWN] = Cmd_Countdown,
[CMD_PLAY_GAME_LEADER] = Cmd_PlayGame_Leader,
[CMD_PLAY_GAME_MEMBER] = Cmd_PlayGame_Member,
[CMD_FINISH_GAME] = Cmd_FinishGame,
[CMD_TIMES_UP] = Cmd_HandleTimeUp,
[CMD_CALC_RESULTS] = Cmd_TabulateResults,
[CMD_SHOW_RESULTS] = Cmd_ShowResults,
[CMD_SAVE] = Cmd_SaveGame,
[CMD_ASK_PLAY_AGAIN] = Cmd_AskPlayAgain,
[CMD_COMM_PLAY_AGAIN] = Cmd_CommunicatePlayAgainResponses,
[CMD_PLAY_AGAIN_YES] = Cmd_PlayAgain,
[CMD_PLAY_AGAIN_NO] = Cmd_StopGame,
[CMD_CLOSE_LINK] = Cmd_CloseLink,
[CMD_QUIT] = Cmd_Quit,
2020-02-10 00:47:00 -05:00
};
// Per group size, the number of A presses required to increase the number of sparkles.
2021-03-16 05:40:42 -04:00
static const u8 sSparkleThresholds[MAX_RFU_PLAYERS - 1][4] =
2020-02-10 00:47:00 -05:00
{
2021-03-16 05:40:42 -04:00
{2, 4, 6, 7}, // 2 players
{3, 5, 8, 11}, // 3 players
{3, 7, 11, 15}, // 4 players
{4, 8, 12, 17}, // 5 players
2020-02-10 00:47:00 -05:00
};
2021-03-16 05:40:42 -04:00
// Per group size, the number of A presses required to get big sparkles
static const u8 sBigSparkleThresholds[MAX_RFU_PLAYERS - 1] = {5, 7, 9, 12};
static const u8 sReceivedPlayerBitmasks[] = {0x03, 0x07, 0x0F, 0x1F};
2020-02-10 00:47:00 -05:00
2021-03-16 05:40:42 -04:00
static struct BerryCrushGame * GetBerryCrushGame(void)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
return sGame;
2019-03-31 09:05:32 -05:00
}
2021-03-16 05:40:42 -04:00
static u32 QuitBerryCrush(MainCallback exitCallback)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
if (!sGame)
2019-03-31 09:05:32 -05:00
return 2;
2021-03-16 05:40:42 -04:00
if (!exitCallback)
exitCallback = sGame->exitCallback;
2019-03-31 09:05:32 -05:00
2021-03-16 05:40:42 -04:00
DestroyTask(sGame->taskId);
FREE_AND_SET_NULL(sGame);
SetMainCallback2(exitCallback);
if (exitCallback == CB2_ReturnToField)
2019-03-31 09:05:32 -05:00
{
2019-12-18 03:35:41 +08:00
gTextFlags.autoScroll = TRUE;
2020-08-20 18:02:00 -04:00
PlayNewMapMusic(MUS_POKE_CENTER);
2019-03-31 09:05:32 -05:00
SetMainCallback1(CB1_Overworld);
}
return 0;
}
#define ERROR_EXIT(exitCallback) \
{ \
SetMainCallback2(exitCallback); \
gRfu.errorParam0 = 0; \
gRfu.errorParam1 = 0; \
gRfu.errorState = RFU_ERROR_STATE_OCCURRED; \
2021-03-16 05:40:42 -04:00
}
void StartBerryCrush(MainCallback exitCallback)
2019-03-31 09:05:32 -05:00
{
u8 playerCount = 0;
u8 multiplayerId;
if (!gReceivedRemoteLinkPlayers || gWirelessCommType == 0)
{
2021-03-16 05:40:42 -04:00
// Link disconnected
ERROR_EXIT(exitCallback);
2019-03-31 09:05:32 -05:00
return;
}
playerCount = GetLinkPlayerCount();
multiplayerId = GetMultiplayerId();
if (playerCount < 2 || multiplayerId >= playerCount)
{
2021-03-16 05:40:42 -04:00
// Too few players, or invalid id
ERROR_EXIT(exitCallback);
2019-03-31 09:05:32 -05:00
return;
}
2021-03-16 05:40:42 -04:00
sGame = AllocZeroed(sizeof(*sGame));
if (!sGame)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
// Alloc failed
ERROR_EXIT(exitCallback);
2019-03-31 09:05:32 -05:00
return;
}
2021-03-16 05:40:42 -04:00
sGame->exitCallback = exitCallback;
sGame->localId = multiplayerId;
sGame->playerCount = playerCount;
SetNamesAndTextSpeed(sGame);
sGame->gameState = STATE_INIT;
sGame->nextCmd = CMD_FADE;
sGame->afterPalFadeCmd = CMD_READY_BEGIN;
SetPaletteFadeArgs(sGame->commandArgs, TRUE, PALETTES_ALL, 0, 16, 0, RGB_BLACK);
RunOrScheduleCommand(CMD_SHOW_GAME, 1, sGame->commandArgs);
2020-09-13 00:26:01 -04:00
SetMainCallback2(MainCB);
2021-03-16 05:40:42 -04:00
sGame->taskId = CreateTask(MainTask, 8);
gTextFlags.autoScroll = FALSE;
2019-03-31 09:05:32 -05:00
}
2020-09-13 00:26:01 -04:00
static void GetBerryFromBag(void)
2019-03-31 09:05:32 -05:00
{
if (gSpecialVar_ItemId < FIRST_BERRY_INDEX || gSpecialVar_ItemId > LAST_BERRY_INDEX + 1)
gSpecialVar_ItemId = FIRST_BERRY_INDEX;
2019-03-31 09:05:32 -05:00
else
RemoveBagItem(gSpecialVar_ItemId, 1);
2021-03-16 05:40:42 -04:00
sGame->players[sGame->localId].berryId = gSpecialVar_ItemId - FIRST_BERRY_INDEX;
sGame->nextCmd = CMD_FADE;
sGame->afterPalFadeCmd = CMD_WAIT_BERRIES;
SetPaletteFadeArgs(sGame->commandArgs, FALSE, PALETTES_ALL, 0, 16, 0, RGB_BLACK);
RunOrScheduleCommand(CMD_SHOW_GAME, 1, sGame->commandArgs);
sGame->taskId = CreateTask(MainTask, 8);
2020-09-13 00:26:01 -04:00
SetMainCallback2(MainCB);
2019-03-31 09:05:32 -05:00
}
2021-03-16 05:40:42 -04:00
static void ChooseBerry(void)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
DestroyTask(sGame->taskId);
2020-09-13 00:26:01 -04:00
ChooseBerryForMachine(GetBerryFromBag);
2019-03-31 09:05:32 -05:00
}
static void BerryCrush_SetVBlankCB(void)
2019-03-31 09:05:32 -05:00
{
2020-09-13 00:26:01 -04:00
SetVBlankCallback(VBlankCB);
2019-03-31 09:05:32 -05:00
}
static void BerryCrush_InitVBlankCB(void)
2019-03-31 09:05:32 -05:00
{
SetVBlankCallback(NULL);
}
2021-03-16 05:40:42 -04:00
static void SaveResults(void)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
u32 time, presses;
// Calculate pressing speed ((time / 60) / presses)
time = sGame->results.time;
time = Q_24_8(time);
time = MathUtil_Div32(time, Q_24_8(60));
presses = sGame->results.totalAPresses;
presses = Q_24_8(presses);
presses = MathUtil_Div32(presses, time) & 0xFFFF;
sGame->pressingSpeed = presses;
2019-03-31 09:05:32 -05:00
2021-03-16 05:40:42 -04:00
switch (sGame->playerCount)
2019-03-31 09:05:32 -05:00
{
case 2:
2021-03-16 05:40:42 -04:00
if (sGame->pressingSpeed > gSaveBlock2Ptr->berryCrush.pressingSpeeds[0])
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
// New 2-player record
sGame->newRecord = TRUE;
gSaveBlock2Ptr->berryCrush.pressingSpeeds[0] = sGame->pressingSpeed;
2019-03-31 09:05:32 -05:00
}
break;
case 3:
2021-03-16 05:40:42 -04:00
if (sGame->pressingSpeed > gSaveBlock2Ptr->berryCrush.pressingSpeeds[1])
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
// New 3-player record
sGame->newRecord = TRUE;
gSaveBlock2Ptr->berryCrush.pressingSpeeds[1] = sGame->pressingSpeed;
2019-03-31 09:05:32 -05:00
}
break;
case 4:
2021-03-16 05:40:42 -04:00
if (sGame->pressingSpeed > gSaveBlock2Ptr->berryCrush.pressingSpeeds[2])
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
// New 4-player record
sGame->newRecord = TRUE;
gSaveBlock2Ptr->berryCrush.pressingSpeeds[2] = sGame->pressingSpeed;
2019-03-31 09:05:32 -05:00
}
break;
case 5:
2021-03-16 05:40:42 -04:00
if (sGame->pressingSpeed > gSaveBlock2Ptr->berryCrush.pressingSpeeds[3])
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
// New 5-player record
sGame->newRecord = TRUE;
gSaveBlock2Ptr->berryCrush.pressingSpeeds[3] = sGame->pressingSpeed;
2019-03-31 09:05:32 -05:00
}
break;
}
2021-03-16 05:40:42 -04:00
sGame->powder = sGame->results.powder;
if (GiveBerryPowder(sGame->powder))
2019-03-31 09:05:32 -05:00
return;
2021-03-16 05:40:42 -04:00
sGame->noRoomForPowder = TRUE;
2019-03-31 09:05:32 -05:00
}
2020-09-13 00:26:01 -04:00
static void VBlankCB(void)
2019-03-31 09:05:32 -05:00
{
TransferPlttBuffer();
LoadOam();
ProcessSpriteCopyRequests();
}
2020-09-13 00:26:01 -04:00
static void MainCB(void)
2019-03-31 09:05:32 -05:00
{
RunTasks();
RunTextPrinters();
AnimateSprites();
BuildOamBuffer();
}
2020-09-13 00:26:01 -04:00
static void MainTask(u8 taskId)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
if (sGame->cmdCallback)
sGame->cmdCallback(sGame, sGame->commandArgs);
2019-03-31 09:05:32 -05:00
2021-03-16 05:40:42 -04:00
UpdateGame(sGame);
2019-03-31 09:05:32 -05:00
}
2021-03-16 05:40:42 -04:00
static void SetNamesAndTextSpeed(struct BerryCrushGame *game)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
u8 i;
for (i = 0; i < game->playerCount; i++)
StringCopy(game->players[i].name, gLinkPlayers[i].name);
for (; i < MAX_RFU_PLAYERS; i++)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
memset(game->players[i].name, 1, PLAYER_NAME_LENGTH);
game->players[i].name[PLAYER_NAME_LENGTH] = EOS;
2019-03-31 09:05:32 -05:00
}
switch (gSaveBlock2Ptr->optionsTextSpeed)
{
case OPTIONS_TEXT_SPEED_SLOW:
2021-03-16 05:40:42 -04:00
game->textSpeed = 8;
2019-03-31 09:05:32 -05:00
break;
case OPTIONS_TEXT_SPEED_MID:
2021-03-16 05:40:42 -04:00
game->textSpeed = 4;
2019-03-31 09:05:32 -05:00
break;
case OPTIONS_TEXT_SPEED_FAST:
2021-03-16 05:40:42 -04:00
game->textSpeed = 1;
2019-03-31 09:05:32 -05:00
break;
}
}
2021-03-16 05:40:42 -04:00
static s32 ShowGameDisplay(void)
2019-03-31 09:05:32 -05:00
{
2020-09-13 00:26:01 -04:00
struct BerryCrushGame *game = GetBerryCrushGame();
if (!game)
2019-03-31 09:05:32 -05:00
return -1;
switch (game->cmdState)
2019-03-31 09:05:32 -05:00
{
case 0:
SetVBlankCallback(NULL);
SetHBlankCallback(NULL);
SetGpuReg(REG_OFFSET_DISPCNT, 0);
ScanlineEffect_Stop();
2020-05-14 01:37:09 -07:00
ResetTempTileDataBuffers();
2019-03-31 09:05:32 -05:00
break;
case 1:
CpuFill16(0, (void *)OAM, OAM_SIZE);
gReservedSpritePaletteCount = 0;
2020-04-08 16:24:30 -04:00
DigitObjUtil_Init(3);
2019-03-31 09:05:32 -05:00
break;
case 2:
ResetPaletteFade();
ResetSpriteData();
FreeAllSpritePalettes();
break;
case 3:
ResetBgsAndClearDma3BusyFlags(0);
2021-03-16 05:40:42 -04:00
InitBgsFromTemplates(0, sBgTemplates, ARRAY_COUNT(sBgTemplates));
SetBgTilemapBuffer(1, game->gfx.bgBuffers[0]);
SetBgTilemapBuffer(2, game->gfx.bgBuffers[2]);
SetBgTilemapBuffer(3, game->gfx.bgBuffers[3]);
2021-11-03 23:02:06 -04:00
ChangeBgX(0, 0, BG_COORD_SET);
ChangeBgY(0, 0, BG_COORD_SET);
ChangeBgX(2, 0, BG_COORD_SET);
ChangeBgY(2, 0, BG_COORD_SET);
ChangeBgX(3, 0, BG_COORD_SET);
ChangeBgY(3, 0, BG_COORD_SET);
2019-03-31 09:05:32 -05:00
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
break;
case 4:
FillBgTilemapBufferRect_Palette0(0, 0, 0, 0, 32, 32);
FillBgTilemapBufferRect_Palette0(1, 0, 0, 0, 32, 64);
FillBgTilemapBufferRect_Palette0(2, 0, 0, 0, 32, 32);
FillBgTilemapBufferRect_Palette0(3, 0, 0, 0, 32, 32);
break;
case 5:
CopyBgTilemapBufferToVram(0);
CopyBgTilemapBufferToVram(1);
CopyBgTilemapBufferToVram(2);
CopyBgTilemapBufferToVram(3);
2021-03-16 05:40:42 -04:00
DecompressAndCopyTileDataToVram(1, gBerryCrush_Crusher_Gfx, 0, 0, 0);
2019-03-31 09:05:32 -05:00
break;
case 6:
2020-05-14 01:37:09 -07:00
if (FreeTempTileDataBuffersIfPossible())
2019-03-31 09:05:32 -05:00
return 0;
InitStandardTextBoxWindows();
InitTextBoxGfxAndPrinters();
2021-03-16 05:40:42 -04:00
CreatePlayerNameWindows(game);
DrawPlayerNameWindows(game);
2019-12-18 03:35:41 +08:00
gPaletteFade.bufferTransferDisabled = TRUE;
2019-03-31 09:05:32 -05:00
break;
case 7:
2021-03-16 05:40:42 -04:00
LoadPalette(gBerryCrush_Crusher_Pal, 0, 0x180);
CopyToBgTilemapBuffer(1, sCrusherTop_Tilemap, 0, 0);
CopyToBgTilemapBuffer(2, sContainerCap_Tilemap, 0, 0);
CopyToBgTilemapBuffer(3, sBg_Tilemap, 0, 0);
CopyPlayerNameWindowGfxToBg(game);
2019-03-31 09:05:32 -05:00
CopyBgTilemapBufferToVram(1);
CopyBgTilemapBufferToVram(2);
CopyBgTilemapBufferToVram(3);
break;
case 8:
2019-04-04 17:05:46 -04:00
LoadWirelessStatusIndicatorSpriteGfx();
2019-03-31 09:05:32 -05:00
CreateWirelessStatusIndicatorSprite(0, 0);
2021-03-16 05:40:42 -04:00
CreateGameSprites(game);
2019-03-31 09:05:32 -05:00
SetGpuReg(REG_OFFSET_BG1VOFS, -gSpriteCoordOffsetY);
2021-11-03 23:02:06 -04:00
ChangeBgX(1, 0, BG_COORD_SET);
ChangeBgY(1, 0, BG_COORD_SET);
2019-03-31 09:05:32 -05:00
break;
case 9:
2019-12-18 03:35:41 +08:00
gPaletteFade.bufferTransferDisabled = FALSE;
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 16, RGB_BLACK);
2019-03-31 09:05:32 -05:00
ShowBg(0);
ShowBg(1);
ShowBg(2);
ShowBg(3);
SetGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_OBJ_ON | DISPCNT_OBJ_1D_MAP);
BerryCrush_SetVBlankCB();
game->cmdState = 0;
2019-03-31 09:05:32 -05:00
return 1;
}
game->cmdState++;
2019-03-31 09:05:32 -05:00
return 0;
}
2021-03-16 05:40:42 -04:00
static s32 HideGameDisplay(void)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
struct BerryCrushGame *game = GetBerryCrushGame();
if (!game)
2019-03-31 09:05:32 -05:00
return -1;
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-03-31 09:05:32 -05:00
{
case 0:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-03-31 09:05:32 -05:00
break;
case 1:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
// fall through
// This will call BeginNormalPaletteFade() twice.
#ifdef BUGFIX
break;
#endif
2019-03-31 09:05:32 -05:00
case 2:
2021-02-24 11:01:02 -05:00
BeginNormalPaletteFade(PALETTES_ALL, 0, 0, 16, RGB_BLACK);
2019-03-31 09:05:32 -05:00
UpdatePaletteFade();
break;
case 3:
if (UpdatePaletteFade())
return 0;
break;
case 4:
FillBgTilemapBufferRect_Palette0(0, 0, 0, 0, 32, 32);
FillBgTilemapBufferRect_Palette0(1, 0, 0, 0, 32, 32);
FillBgTilemapBufferRect_Palette0(2, 0, 0, 0, 32, 32);
FillBgTilemapBufferRect_Palette0(3, 0, 0, 0, 32, 32);
CopyBgTilemapBufferToVram(0);
CopyBgTilemapBufferToVram(1);
CopyBgTilemapBufferToVram(2);
CopyBgTilemapBufferToVram(3);
break;
case 5:
FreeAllWindowBuffers();
HideBg(0);
UnsetBgTilemapBuffer(0);
HideBg(1);
UnsetBgTilemapBuffer(1);
HideBg(2);
UnsetBgTilemapBuffer(2);
HideBg(3);
UnsetBgTilemapBuffer(3);
ClearGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_OBJ_ON | DISPCNT_OBJ_1D_MAP);
break;
case 6:
2019-03-31 18:59:52 -04:00
DestroyWirelessStatusIndicatorSprite();
2021-03-16 05:40:42 -04:00
DestroyGameSprites(game);
2020-04-08 16:24:30 -04:00
DigitObjUtil_Free();
2019-03-31 09:05:32 -05:00
break;
case 7:
2021-03-16 05:40:42 -04:00
game->cmdState = 0;
2019-03-31 09:05:32 -05:00
return 1;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-03-31 09:05:32 -05:00
return 0;
}
2021-03-16 05:40:42 -04:00
// Handles the crusher vibration and the timer
static s32 UpdateGame(struct BerryCrushGame *game)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
gSpriteCoordOffsetY = game->depth + game->vibration;
2019-03-31 09:05:32 -05:00
SetGpuReg(REG_OFFSET_BG1VOFS, -gSpriteCoordOffsetY);
2021-03-16 05:40:42 -04:00
if (game->gameState == STATE_PLAYING)
PrintTimer(&game->gfx, game->timer);
2019-03-31 09:05:32 -05:00
return 0;
}
2021-03-16 05:40:42 -04:00
static void ResetCrusherPos(struct BerryCrushGame *game)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
game->depth = CRUSHER_START_Y;
game->vibration = 0;
2019-03-31 09:05:32 -05:00
gSpriteCoordOffsetX = 0;
2021-03-16 05:40:42 -04:00
gSpriteCoordOffsetY = CRUSHER_START_Y;
}
// Sprite data for berry sprites. Identical to fields for sparkle sprites
#define sX data[0]
#define sYSpeed data[1]
#define sYAccel data[2]
#define sXSpeed data[3]
#define sSinIdx data[4]
#define sSinSpeed data[5]
#define sAmplitude data[6]
// The last element (data[7]) is a bitfield.
// The first 15 bits are the y coord to stop at.
// The last bit is a flag for whether or not to move horizontally too
#define sBitfield data[7]
#define MASK_TARGET_Y 0x7FFF
#define F_MOVE_HORIZ 0x8000
static void CreateBerrySprites(struct BerryCrushGame *game, struct BerryCrushGame_Gfx *gfx)
2019-03-31 09:05:32 -05:00
{
u8 i;
u8 spriteId;
2021-03-16 05:40:42 -04:00
s16 distance, var1;
2019-03-31 09:05:32 -05:00
s16 *data;
2021-03-16 05:40:42 -04:00
s32 amplitude;
s16 speed;
u32 var2;
2019-03-31 09:05:32 -05:00
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount; i++)
2019-03-31 09:05:32 -05:00
{
spriteId = AddCustomItemIconSprite(
2021-02-27 15:49:43 -05:00
&sSpriteTemplate_PlayerBerry,
sPlayerBerrySpriteTags[i],
sPlayerBerrySpriteTags[i],
2021-03-16 05:40:42 -04:00
game->players[i].berryId + FIRST_BERRY_INDEX);
gfx->berrySprites[i] = &gSprites[spriteId];
gfx->berrySprites[i]->oam.priority = 3;
gfx->berrySprites[i]->affineAnimPaused = TRUE;
2021-07-07 09:11:52 -04:00
gfx->berrySprites[i]->x = gfx->playerCoords[i]->berryXOffset + 120;
gfx->berrySprites[i]->y = -16;
2021-03-16 05:40:42 -04:00
data = gfx->berrySprites[i]->data;
speed = 512;
sYSpeed = speed;
sYAccel = 32;
sBitfield = 112; // Setting bits in MASK_TARGET_Y
distance = gfx->playerCoords[i]->berryXDest - gfx->playerCoords[i]->berryXOffset;
amplitude = distance;
if (distance < 0)
amplitude += 3;
sAmplitude = amplitude >> 2;
distance *= 128;
var2 = speed + 32;
var2 = var2 / 2;
var1 = MathUtil_Div16Shift(7, Q_8_8(63.5), var2);
2021-07-07 09:11:52 -04:00
sX = (u16)gfx->berrySprites[i]->x * 128;
2021-03-16 05:40:42 -04:00
sXSpeed = MathUtil_Div16Shift(7, distance, var1);
2020-05-20 14:22:41 -04:00
var1 = MathUtil_Mul16Shift(7, var1, 85);
2021-03-16 05:40:42 -04:00
sSinIdx = 0;
sSinSpeed = MathUtil_Div16Shift(7, Q_8_8(63.5), var1);
sBitfield |= F_MOVE_HORIZ;
if (gfx->playerCoords[i]->berryXOffset < 0)
StartSpriteAffineAnim(gfx->berrySprites[i], 1);
2019-03-31 09:05:32 -05:00
}
}
2021-02-27 15:49:43 -05:00
static void SpriteCB_DropBerryIntoCrusher(struct Sprite *sprite)
2019-03-31 09:05:32 -05:00
{
s16 *data = sprite->data;
2021-03-16 05:40:42 -04:00
sYSpeed += sYAccel;
2021-07-07 09:11:52 -04:00
sprite->y2 += sYSpeed >> 8;
2021-03-16 05:40:42 -04:00
if (sBitfield & F_MOVE_HORIZ)
2019-03-31 09:05:32 -05:00
{
2021-03-16 05:40:42 -04:00
sprite->sX += sXSpeed;
sSinIdx += sSinSpeed;
2021-07-07 09:11:52 -04:00
sprite->x2 = Sin(sSinIdx >> 7, sAmplitude);
2021-03-16 05:40:42 -04:00
if ((sBitfield & F_MOVE_HORIZ) && (sSinIdx >> 7) > 126)
2019-03-31 09:05:32 -05:00
{
2021-07-07 09:11:52 -04:00
sprite->x2 = 0;
2021-03-16 05:40:42 -04:00
sBitfield &= MASK_TARGET_Y;
2019-03-31 09:05:32 -05:00
}
}
2021-07-07 09:11:52 -04:00
sprite->x = sX >> 7;
if (sprite->y + sprite->y2 >= (sBitfield & MASK_TARGET_Y))
2019-03-31 09:05:32 -05:00
{
sprite->callback = SpriteCallbackDummy;
FreeSpriteOamMatrix(sprite);
DestroySprite(sprite);
}
}
2021-03-16 05:40:42 -04:00
#undef sX
#undef sYSpeed
#undef sYAccel
#undef sXSpeed
#undef sSinIdx
#undef sSinSpeed
#undef sAmplitude
#undef sBitfield
#undef MASK_TARGET_Y
#undef F_MOVE_HORIZ
static void BerryCrushFreeBerrySpriteGfx(struct BerryCrushGame *game, struct BerryCrushGame_Gfx *gfx)
2019-03-31 09:05:32 -05:00
{
u8 i;
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount; i++)
2019-03-31 09:05:32 -05:00
{
2021-02-27 15:49:43 -05:00
FreeSpritePaletteByTag(sPlayerBerrySpriteTags[i]);
FreeSpriteTilesByTag(sPlayerBerrySpriteTags[i]);
2019-03-31 09:05:32 -05:00
}
}
2021-03-16 05:40:42 -04:00
static void UpdateInputEffects(struct BerryCrushGame *game, struct BerryCrushGame_Gfx *gfx)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
u8 numPlayersPressed;
struct BerryCrushGame_LinkState *linkState;
2019-07-03 11:28:44 +02:00
u8 i;
2021-03-16 05:40:42 -04:00
u16 temp1, xModifier;
2019-07-03 11:28:44 +02:00
2021-03-16 05:40:42 -04:00
numPlayersPressed = 0;
linkState = (struct BerryCrushGame_LinkState *)game->recvCmd;
2021-03-16 05:40:42 -04:00
// Read inputs and update impact effects
for (i = 0; i < game->playerCount; i++)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
#define flags temp1
flags = linkState->inputFlags >> (i * INPUT_FLAGS_PER_PLAYER);
flags &= INPUT_FLAG_MASK;
if (flags)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
numPlayersPressed++;
if (flags & F_INPUT_HIT_SYNC)
StartSpriteAnim(gfx->impactSprites[i], 1); // Big impact sprite
2019-07-03 11:28:44 +02:00
else
2021-03-16 05:40:42 -04:00
StartSpriteAnim(gfx->impactSprites[i], 0); // Small impact sprite
2019-07-03 11:28:44 +02:00
2021-03-16 05:40:42 -04:00
gfx->impactSprites[i]->invisible = FALSE;
gfx->impactSprites[i]->animPaused = FALSE;
2021-07-07 09:11:52 -04:00
gfx->impactSprites[i]->x2 = sImpactCoords[(flags % (ARRAY_COUNT(sImpactCoords) + 1)) - 1][0];
gfx->impactSprites[i]->y2 = sImpactCoords[(flags % (ARRAY_COUNT(sImpactCoords) + 1)) - 1][1];
2019-07-03 11:28:44 +02:00
}
2021-03-16 05:40:42 -04:00
#undef flags
2019-07-03 11:28:44 +02:00
}
2021-03-16 05:40:42 -04:00
if (numPlayersPressed == 0)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
game->playedSound = FALSE;
2019-07-03 11:28:44 +02:00
}
else
{
2021-03-16 05:40:42 -04:00
// Update sparkle effect
#define yModifier temp1
yModifier = (u8)(game->timer % 3);
xModifier = yModifier;
for (i = 0; i < linkState->sparkleAmount * 2 + 3; i++)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
if (gfx->sparkleSprites[i]->invisible)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
gfx->sparkleSprites[i]->callback = SpriteCB_Sparkle_Init;
2021-07-07 09:11:52 -04:00
gfx->sparkleSprites[i]->x = sSparkleCoords[i][0] + 120;
gfx->sparkleSprites[i]->y = sSparkleCoords[i][1] + 136 - (yModifier * 4);
gfx->sparkleSprites[i]->x2 = sSparkleCoords[i][0] + (sSparkleCoords[i][0] / (xModifier * 4));
gfx->sparkleSprites[i]->y2 = sSparkleCoords[i][1];
2021-03-16 05:40:42 -04:00
if (linkState->bigSparkle)
StartSpriteAnim(gfx->sparkleSprites[i], 1);
2019-07-03 11:28:44 +02:00
else
2021-03-16 05:40:42 -04:00
StartSpriteAnim(gfx->sparkleSprites[i], 0);
2019-07-03 11:28:44 +02:00
2021-03-16 05:40:42 -04:00
yModifier++;
if (yModifier > 3)
yModifier = 0;
2019-07-03 11:28:44 +02:00
}
}
2021-03-16 05:40:42 -04:00
#undef yModifier
if (game->playedSound)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
game->playedSound = FALSE;
2019-07-03 11:28:44 +02:00
}
else
{
2021-03-16 05:40:42 -04:00
if (numPlayersPressed == 1)
2020-08-20 18:02:00 -04:00
PlaySE(SE_MUD_BALL);
2019-07-03 11:28:44 +02:00
else
2020-08-20 18:02:00 -04:00
PlaySE(SE_BREAKABLE_DOOR);
2019-07-03 11:28:44 +02:00
2021-03-16 05:40:42 -04:00
game->playedSound = TRUE;
2019-07-03 11:28:44 +02:00
}
}
}
2021-03-16 05:40:42 -04:00
static bool32 AreEffectsFinished(struct BerryCrushGame *game, struct BerryCrushGame_Gfx *gfx)
2019-07-03 11:28:44 +02:00
{
u8 i;
2021-03-16 05:40:42 -04:00
// Are any impact sprites active
for (i = 0; i < game->playerCount; i++)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
if (!gfx->impactSprites[i]->invisible)
2019-07-03 11:28:44 +02:00
return FALSE;
}
2021-03-16 05:40:42 -04:00
// Are any sparkle sprites active
for (i = 0; i < ARRAY_COUNT(gfx->sparkleSprites); i++)
2019-07-03 11:28:44 +02:00
{
2021-03-16 05:40:42 -04:00
if (!gfx->sparkleSprites[i]->invisible)
2019-07-03 11:28:44 +02:00
return FALSE;
}
2021-03-16 05:40:42 -04:00
if (game->vibration != 0)
game->vibration = 0;
2019-07-03 11:28:44 +02:00
return TRUE;
}
2021-03-16 05:40:42 -04:00
static void FramesToMinSec(struct BerryCrushGame_Gfx *gfx, u16 frames)
2019-07-03 11:28:44 +02:00
{
u8 i = 0;
2021-02-27 15:49:43 -05:00
u32 fractionalFrames = 0;
2019-07-03 11:28:44 +02:00
s16 r3 = 0;
2021-03-16 05:40:42 -04:00
gfx->minutes = frames / (60 * 60);
gfx->secondsInt = (frames % (60 * 60)) / 60;
r3 = MathUtil_Mul16(Q_8_8(frames % 60), 4);
2019-07-03 11:28:44 +02:00
for (i = 0; i < 8; i++)
{
if ((r3 >> (7 - i)) & 1)
2021-02-27 15:49:43 -05:00
fractionalFrames += sPressingSpeedConversionTable[i];
2019-07-03 11:28:44 +02:00
}
2021-03-16 05:40:42 -04:00
gfx->secondsFrac = fractionalFrames / 1000000;
2019-07-03 11:28:44 +02:00
}
2021-02-27 15:49:43 -05:00
static void PrintTextCentered(u8 windowId, u8 left, u8 colorId, const u8 *string)
2019-07-03 11:28:44 +02:00
{
2021-10-30 16:47:37 -04:00
left = (left * 4) - (GetStringWidth(FONT_SHORT, string, -1) / 2u);
AddTextPrinterParameterized3(windowId, FONT_SHORT, left, 0, sTextColorTable[colorId], 0, string);
2019-07-03 11:28:44 +02:00
}
2019-12-12 10:16:05 +08:00
2021-03-16 05:40:42 -04:00
static void PrintResultsText(struct BerryCrushGame * game, u8 page, u8 sp14, u8 baseY)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i, j;
u8 playerId = 0;
u8 ranking = 0;
s32 x;
u8 stat;
struct BerryCrushGame_Results * results = &game->results;
2019-12-12 10:16:05 +08:00
u32 xOffset;
2021-03-16 05:40:42 -04:00
s32 y;
baseY -= 16;
if (page == RESULTS_PAGE_CRUSHING)
baseY -= 42;
y = baseY - 14 * game->playerCount;
if (y > 0)
y = y / 2 + 16;
2019-12-12 10:16:05 +08:00
else
2021-03-16 05:40:42 -04:00
y = 16;
2020-03-06 06:59:32 +08:00
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount; y += 14, i++)
2019-12-12 10:16:05 +08:00
{
DynamicPlaceholderTextUtil_Reset();
2021-03-16 05:40:42 -04:00
switch (page)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
case RESULTS_PAGE_PRESSES:
playerId = results->playerIdsRanked[page][i];
if (i != 0 && results->stats[page][i] != results->stats[page][i - 1])
ranking = i;
ConvertIntToDecimalStringN(gStringVar4, results->stats[page][i], STR_CONV_MODE_RIGHT_ALIGN, 4);
StringAppend(gStringVar4, sResultsTexts[page]);
2019-12-12 10:16:05 +08:00
break;
2021-03-16 05:40:42 -04:00
case RESULTS_PAGE_RANDOM:
playerId = results->playerIdsRanked[page][i];
if (i != 0 && results->stats[page][i] != results->stats[page][i - 1])
ranking = i;
ConvertIntToDecimalStringN(gStringVar1, results->stats[page][i] >> 4, STR_CONV_MODE_RIGHT_ALIGN, 3);
2020-12-21 20:24:36 -05:00
xOffset = 0;
2021-03-16 05:40:42 -04:00
stat = results->stats[page][i] & 15;
for (j = 0; j < 4; j++)
if ((stat >> (3 - j)) & 1)
xOffset += sPressingSpeedConversionTable[j];
stat = xOffset / 1000000u;
ConvertIntToDecimalStringN(gStringVar2, stat, STR_CONV_MODE_LEADING_ZEROS, 2);
StringExpandPlaceholders(gStringVar4, sResultsTexts[page]);
2019-12-12 10:16:05 +08:00
break;
2021-03-16 05:40:42 -04:00
case RESULTS_PAGE_CRUSHING:
playerId = i;
ranking = i;
j = game->players[i].berryId;
if (j >= LAST_BERRY_INDEX - FIRST_BERRY_INDEX + 2)
j = 0;
StringCopy(gStringVar1, gBerries[j].name);
StringExpandPlaceholders(gStringVar4, sResultsTexts[page]);
2019-12-12 10:16:05 +08:00
break;
}
2021-10-30 16:47:37 -04:00
x = GetStringRightAlignXOffset(FONT_SHORT, gStringVar4, sp14 - 4);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2021-03-16 05:40:42 -04:00
if (playerId == game->localId)
2019-12-12 10:16:05 +08:00
StringCopy(gStringVar3, gText_1DotBlueF700);
else
StringCopy(gStringVar3, gText_1DotF700);
2021-03-16 05:40:42 -04:00
gStringVar3[0] = ranking + CHAR_1;
DynamicPlaceholderTextUtil_SetPlaceholderPtr(0, game->players[playerId].name);
2019-12-12 10:16:05 +08:00
DynamicPlaceholderTextUtil_ExpandPlaceholders(gStringVar4, gStringVar3);
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, 4, y, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2019-12-12 10:16:05 +08:00
}
}
2021-03-16 05:40:42 -04:00
static void PrintCrushingResults(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i = 0;
u8 x = 0;
u32 pressingSpeedFrac = 0;
struct BerryCrushGame_Results *results = &game->results;
u8 y = GetWindowAttribute(game->gfx.resultsWindowId, WINDOW_HEIGHT) * 8 - 42;
FramesToMinSec(&game->gfx, results->time);
// Print time text
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gText_TimeColon);
2019-12-12 10:16:05 +08:00
2021-03-16 05:40:42 -04:00
// Print seconds text
2021-10-30 16:47:37 -04:00
x = 176 - (u8)GetStringWidth(FONT_SHORT, gText_SpaceSec, -1);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gText_SpaceSec);
2021-03-16 05:40:42 -04:00
// Print seconds value
ConvertIntToDecimalStringN(gStringVar1, game->gfx.secondsInt, STR_CONV_MODE_LEADING_ZEROS, 2);
ConvertIntToDecimalStringN(gStringVar2, game->gfx.secondsFrac, STR_CONV_MODE_LEADING_ZEROS, 2);
2019-12-12 10:16:05 +08:00
StringExpandPlaceholders(gStringVar4, gText_XDotY2);
2021-10-30 16:47:37 -04:00
x -= GetStringWidth(FONT_SHORT, gStringVar4, -1);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2021-03-16 05:40:42 -04:00
// Print minutes text
2021-10-30 16:47:37 -04:00
x -= GetStringWidth(FONT_SHORT, gText_SpaceMin, -1);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gText_SpaceMin);
2021-03-16 05:40:42 -04:00
// Print minutes value
ConvertIntToDecimalStringN(gStringVar1, game->gfx.minutes, STR_CONV_MODE_LEADING_ZEROS, 1);
2019-12-12 10:16:05 +08:00
StringExpandPlaceholders(gStringVar4, gText_StrVar1);
2021-10-30 16:47:37 -04:00
x -= GetStringWidth(FONT_SHORT, gStringVar4, -1);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2021-03-16 05:40:42 -04:00
// Print pressing speed text
y += 14;
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, 0, y, sTextColorTable[COLORID_GRAY], 0, gText_PressingSpeed);
x = 176 - (u8)GetStringWidth(FONT_SHORT, gText_TimesPerSec, -1);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gText_TimesPerSec);
2021-03-16 05:40:42 -04:00
// Print pressing speed value
for (i = 0; i < 8; i++)
if (((u8)game->pressingSpeed >> (7 - i)) & 1)
pressingSpeedFrac += *(i + sPressingSpeedConversionTable); // It's accessed in a different way here for unknown reason
ConvertIntToDecimalStringN(gStringVar1, game->pressingSpeed >> 8, STR_CONV_MODE_RIGHT_ALIGN, 3);
ConvertIntToDecimalStringN(gStringVar2, pressingSpeedFrac / 1000000, STR_CONV_MODE_LEADING_ZEROS, 2);
2019-12-12 10:16:05 +08:00
StringExpandPlaceholders(gStringVar4, gText_XDotY3);
2021-10-30 16:47:37 -04:00
x -= GetStringWidth(FONT_SHORT, gStringVar4, -1);
2021-03-16 05:40:42 -04:00
if (game->newRecord)
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_RED], 0, gStringVar4);
2019-12-12 10:16:05 +08:00
else
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2021-03-16 05:40:42 -04:00
// Print silkiness text
y += 14;
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, 0, y, sTextColorTable[COLORID_GRAY], 0, gText_Silkiness);
2021-03-16 05:40:42 -04:00
// Print silkiness value
ConvertIntToDecimalStringN(gStringVar1, results->silkiness, STR_CONV_MODE_RIGHT_ALIGN, 3);
2019-12-12 10:16:05 +08:00
StringExpandPlaceholders(gStringVar4, gText_Var1Percent);
2021-10-30 16:47:37 -04:00
x = 176 - (u8)GetStringWidth(FONT_SHORT, gStringVar4, -1);
AddTextPrinterParameterized3(game->gfx.resultsWindowId, FONT_SHORT, x, y, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
static bool32 OpenResultsWindow(struct BerryCrushGame *game, struct BerryCrushGame_Gfx *gfx)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 playerCountIdx;
2019-12-12 10:16:05 +08:00
struct WindowTemplate template;
2021-03-16 05:40:42 -04:00
switch (gfx->resultsState)
2019-12-12 10:16:05 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
playerCountIdx = game->playerCount - 2;
HideTimer(gfx);
memcpy(&template, &sWindowTemplates_Results[game->gameState - RESULTS_STATE_START], sizeof(struct WindowTemplate));
if (game->gameState == STATE_RESULTS_CRUSHING)
template.height = sResultsWindowHeights[1][playerCountIdx];
2019-12-12 10:16:05 +08:00
else
2021-03-16 05:40:42 -04:00
template.height = sResultsWindowHeights[0][playerCountIdx];
gfx->resultsWindowId = AddWindow(&template);
2019-12-12 10:16:05 +08:00
break;
case 1:
2021-03-16 05:40:42 -04:00
PutWindowTilemap(gfx->resultsWindowId);
FillWindowPixelBuffer(gfx->resultsWindowId, PIXEL_FILL(0));
2019-12-12 10:16:05 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
LoadUserWindowBorderGfx_(gfx->resultsWindowId, 541, 208);
2022-07-25 14:59:14 -04:00
DrawStdFrameWithCustomTileAndPalette(gfx->resultsWindowId, FALSE, 541, 13);
2019-12-12 10:16:05 +08:00
break;
case 3:
2021-03-16 05:40:42 -04:00
playerCountIdx = game->playerCount - 2;
switch (game->gameState)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
case STATE_RESULTS_PRESSES:
PrintTextCentered(gfx->resultsWindowId, 20, COLORID_BLUE, gText_PressesRankings);
PrintResultsText(game, RESULTS_PAGE_PRESSES, 0xA0, 8 * sResultsWindowHeights[0][playerCountIdx]);
gfx->resultsState = 5; // Skip past Crushing Results text
2019-12-12 10:16:05 +08:00
return FALSE;
2021-03-16 05:40:42 -04:00
case STATE_RESULTS_RANDOM:
PrintTextCentered(gfx->resultsWindowId, 20, COLORID_GREEN, sResultsTexts[game->results.randomPageId + NUM_RESULTS_PAGES]);
PrintResultsText(game, RESULTS_PAGE_RANDOM, 0xA0, 8 * sResultsWindowHeights[0][playerCountIdx]);
gfx->resultsState = 5; // Skip past Crushing Results text
2019-12-12 10:16:05 +08:00
return FALSE;
2021-03-16 05:40:42 -04:00
case STATE_RESULTS_CRUSHING:
PrintTextCentered(gfx->resultsWindowId, 22, COLORID_BLUE, gText_CrushingResults);
PrintResultsText(game, RESULTS_PAGE_CRUSHING, 0xB0, 8 * sResultsWindowHeights[1][playerCountIdx]);
2019-12-12 10:16:05 +08:00
break;
}
break;
case 4:
2021-03-16 05:40:42 -04:00
PrintCrushingResults(game);
2019-12-12 10:16:05 +08:00
break;
case 5:
2021-11-03 15:29:18 -04:00
CopyWindowToVram(gfx->resultsWindowId, COPYWIN_FULL);
2021-03-16 05:40:42 -04:00
gfx->resultsState = 0;
2019-12-12 10:16:05 +08:00
return TRUE;
}
2021-03-16 05:40:42 -04:00
gfx->resultsState++;
2019-12-12 10:16:05 +08:00
return FALSE;
}
2021-03-16 05:40:42 -04:00
static void CloseResultsWindow(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2022-07-25 14:59:14 -04:00
ClearStdWindowAndFrameToTransparent(game->gfx.resultsWindowId, TRUE);
2021-03-16 05:40:42 -04:00
RemoveWindow(game->gfx.resultsWindowId);
DrawPlayerNameWindows(game);
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
#define tState data[0]
#define tWindowId data[1]
#define tPressingSpeeds(i) data[2 + (i)] // data[2]-[5], for different group sizes
static void Task_ShowRankings(u8 taskId)
2019-12-12 10:16:05 +08:00
{
2020-05-21 12:47:16 -04:00
u8 i = 0, j, xPos, yPos;
u32 score = 0;
s16 *data = gTasks[taskId].data;
2021-03-16 05:40:42 -04:00
switch (tState)
2019-12-12 10:16:05 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
tWindowId = AddWindow(&sWindowTemplate_Rankings);
PutWindowTilemap(tWindowId);
FillWindowPixelBuffer(tWindowId, PIXEL_FILL(0));
LoadUserWindowBorderGfx_(tWindowId, 541, 208);
2022-07-25 14:59:14 -04:00
DrawStdFrameWithCustomTileAndPalette(tWindowId, FALSE, 541, 13);
2019-12-12 10:16:05 +08:00
break;
case 1:
2021-03-16 05:40:42 -04:00
// Print header text
2021-10-30 16:47:37 -04:00
xPos = 96 - GetStringWidth(FONT_NORMAL, gText_BerryCrush2, -1) / 2u;
AddTextPrinterParameterized3(tWindowId, FONT_NORMAL, xPos, 1, sTextColorTable[COLORID_BLUE], 0, gText_BerryCrush2);
xPos = 96 - GetStringWidth(FONT_NORMAL, gText_PressingSpeedRankings, -1) / 2u;
AddTextPrinterParameterized3(tWindowId, FONT_NORMAL, xPos, 17, sTextColorTable[COLORID_BLUE], 0, gText_PressingSpeedRankings);
2021-03-16 05:40:42 -04:00
// Print pressing speed record for each group size, ranked
2020-05-21 12:47:16 -04:00
yPos = 41;
2021-03-16 05:40:42 -04:00
for (i = 0; i < MAX_RFU_PLAYERS - 1; i++)
2019-12-12 10:16:05 +08:00
{
2020-05-21 12:47:16 -04:00
ConvertIntToDecimalStringN(gStringVar1, i + 2, STR_CONV_MODE_LEFT_ALIGN, 1);
2019-12-12 10:16:05 +08:00
StringExpandPlaceholders(gStringVar4, gText_Var1Players);
2021-10-30 16:47:37 -04:00
AddTextPrinterParameterized3(tWindowId, FONT_NORMAL, 0, yPos, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
xPos = 192 - (u8)GetStringWidth(FONT_NORMAL, gText_TimesPerSec, -1);
AddTextPrinterParameterized3(tWindowId, FONT_NORMAL, xPos, yPos, sTextColorTable[COLORID_GRAY], 0, gText_TimesPerSec);
2021-03-16 05:40:42 -04:00
for (j = 0; j < 8; j++)
2020-05-21 12:47:16 -04:00
{
2021-03-16 05:40:42 -04:00
if (((tPressingSpeeds(i) & 0xFF) >> (7 - j)) & 1)
2020-05-21 12:47:16 -04:00
score += sPressingSpeedConversionTable[j];
}
2021-03-16 05:40:42 -04:00
ConvertIntToDecimalStringN(gStringVar1, (u16)tPressingSpeeds(i) >> 8, STR_CONV_MODE_RIGHT_ALIGN, 3);
2020-05-21 12:47:16 -04:00
ConvertIntToDecimalStringN(gStringVar2, score / 1000000, STR_CONV_MODE_LEADING_ZEROS, 2);
2019-12-12 10:16:05 +08:00
StringExpandPlaceholders(gStringVar4, gText_XDotY3);
2021-10-30 16:47:37 -04:00
xPos -= GetStringWidth(FONT_NORMAL, gStringVar4, -1);
AddTextPrinterParameterized3(tWindowId, FONT_NORMAL, xPos, yPos, sTextColorTable[COLORID_GRAY], 0, gStringVar4);
2020-05-21 12:47:16 -04:00
yPos += 16;
score = 0;
2019-12-12 10:16:05 +08:00
}
2021-11-03 15:29:18 -04:00
CopyWindowToVram(tWindowId, COPYWIN_FULL);
2019-12-12 10:16:05 +08:00
break;
case 2:
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON | B_BUTTON))
2019-12-12 10:16:05 +08:00
break;
else
return;
case 3:
2022-07-25 14:59:14 -04:00
ClearStdWindowAndFrameToTransparent(tWindowId, TRUE);
2021-03-16 05:40:42 -04:00
ClearWindowTilemap(tWindowId);
RemoveWindow(tWindowId);
2020-05-21 12:47:16 -04:00
DestroyTask(taskId);
2019-12-12 10:16:05 +08:00
EnableBothScriptContexts();
ScriptContext2_Disable();
2021-03-16 05:40:42 -04:00
tState = 0;
2019-12-12 10:16:05 +08:00
return;
}
2021-03-16 05:40:42 -04:00
tState++;
2019-12-12 10:16:05 +08:00
}
void ShowBerryCrushRankings(void)
{
u8 taskId;
ScriptContext2_Enable();
2021-03-16 05:40:42 -04:00
taskId = CreateTask(Task_ShowRankings, 0);
gTasks[taskId].tPressingSpeeds(0) = gSaveBlock2Ptr->berryCrush.pressingSpeeds[0];
gTasks[taskId].tPressingSpeeds(1) = gSaveBlock2Ptr->berryCrush.pressingSpeeds[1];
gTasks[taskId].tPressingSpeeds(2) = gSaveBlock2Ptr->berryCrush.pressingSpeeds[2];
gTasks[taskId].tPressingSpeeds(3) = gSaveBlock2Ptr->berryCrush.pressingSpeeds[3];
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
static void PrintTimer(struct BerryCrushGame_Gfx *gfx, u16 timer)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
FramesToMinSec(gfx, timer);
DigitObjUtil_PrintNumOn(0, gfx->minutes);
DigitObjUtil_PrintNumOn(1, gfx->secondsInt);
DigitObjUtil_PrintNumOn(2, gfx->secondsFrac);
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
static void HideTimer(struct BerryCrushGame_Gfx *gfx)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
gfx->timerSprites[0]->invisible = TRUE;
gfx->timerSprites[1]->invisible = TRUE;
2022-07-25 14:59:14 -04:00
DigitObjUtil_HideOrShow(2, TRUE);
DigitObjUtil_HideOrShow(1, TRUE);
DigitObjUtil_HideOrShow(0, TRUE);
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
static void CreatePlayerNameWindows(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i;
for (i = 0; i < game->playerCount; i++)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
game->gfx.playerCoords[i] = &sPlayerCoords[sPlayerIdToPosId[game->playerCount - 2][i]];
game->gfx.nameWindowIds[i] = AddWindow(&sWindowTemplates_PlayerNames[game->gfx.playerCoords[i]->playerId]);
PutWindowTilemap(game->gfx.nameWindowIds[i]);
FillWindowPixelBuffer(game->gfx.nameWindowIds[i], 0);
2019-12-12 10:16:05 +08:00
}
}
2021-03-16 05:40:42 -04:00
static void DrawPlayerNameWindows(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i;
for (i = 0; i < game->playerCount; i++)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
PutWindowTilemap(game->gfx.nameWindowIds[i]);
if (i == game->localId)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
// Print the player's name
2019-12-12 10:16:05 +08:00
AddTextPrinterParameterized4(
2021-03-16 05:40:42 -04:00
game->gfx.nameWindowIds[i],
2021-10-30 16:47:37 -04:00
FONT_SHORT,
36 - GetStringWidth(FONT_SHORT, game->players[i].name, 0) / 2u,
2019-12-12 10:16:05 +08:00
1,
0,
0,
2021-03-16 05:40:42 -04:00
sTextColorTable[COLORID_BLACK],
2019-12-12 10:16:05 +08:00
0,
2021-03-16 05:40:42 -04:00
game->players[i].name
2019-12-12 10:16:05 +08:00
);
}
else
{
2021-03-16 05:40:42 -04:00
// Print a partner's name
2019-12-12 10:16:05 +08:00
AddTextPrinterParameterized4(
2021-03-16 05:40:42 -04:00
game->gfx.nameWindowIds[i],
2021-10-30 16:47:37 -04:00
FONT_SHORT,
36 - GetStringWidth(FONT_SHORT, game->players[i].name, 0) / 2u,
2019-12-12 10:16:05 +08:00
1,
0,
0,
2021-04-09 22:39:34 -04:00
sTextColorTable[COLORID_LIGHT_GRAY],
2019-12-12 10:16:05 +08:00
0,
2021-03-16 05:40:42 -04:00
game->players[i].name
2019-12-12 10:16:05 +08:00
);
}
2021-11-03 15:29:18 -04:00
CopyWindowToVram(game->gfx.nameWindowIds[i], COPYWIN_FULL);
2019-12-12 10:16:05 +08:00
}
CopyBgTilemapBufferToVram(0);
}
2021-03-16 05:40:42 -04:00
// Each player name window border uses a color that corresponds to a slot of the crusher lid
static void CopyPlayerNameWindowGfxToBg(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i = 0;
2022-01-11 10:42:00 -05:00
u8 * windowGfx;
2022-01-11 10:42:00 -05:00
LZ77UnCompWram(gBerryCrush_TextWindows_Tilemap, gDecompressionBuffer);
2022-01-11 10:42:00 -05:00
for (windowGfx = gDecompressionBuffer; i < game->playerCount; i++)
2019-12-12 10:16:05 +08:00
{
CopyToBgTilemapBufferRect(
3,
2022-01-11 10:42:00 -05:00
&windowGfx[game->gfx.playerCoords[i]->playerId * 40],
2021-03-16 05:40:42 -04:00
game->gfx.playerCoords[i]->windowGfxX,
game->gfx.playerCoords[i]->windowGfxY,
2019-12-12 10:16:05 +08:00
10,
2
);
}
CopyBgTilemapBufferToVram(3);
}
2021-03-16 05:40:42 -04:00
static void CreateGameSprites(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i = 0;
u8 spriteId;
2019-12-12 10:16:05 +08:00
2021-03-16 05:40:42 -04:00
game->depth = CRUSHER_START_Y;
game->vibration = 0;
2019-12-12 10:16:05 +08:00
gSpriteCoordOffsetX = 0;
2021-03-16 05:40:42 -04:00
gSpriteCoordOffsetY = CRUSHER_START_Y;
for (i = 0; i < ARRAY_COUNT(sSpriteSheets) - 1; i++)
LoadCompressedSpriteSheet(&sSpriteSheets[i]);
LoadSpritePalettes(sSpritePals);
2021-03-16 05:40:42 -04:00
// Create sprite for crusher base
spriteId = CreateSprite(&sSpriteTemplate_CrusherBase, 120, 88, 5);
game->gfx.coreSprite = &gSprites[spriteId];
game->gfx.coreSprite->oam.priority = 3;
game->gfx.coreSprite->coordOffsetEnabled = TRUE;
game->gfx.coreSprite->animPaused = TRUE;
// Create sprites for the impact effect
for (i = 0; i < game->playerCount; i++)
{
spriteId = CreateSprite(
&sSpriteTemplate_Impact,
game->gfx.playerCoords[i]->impactXOffset + 120,
game->gfx.playerCoords[i]->impactYOffset + 32,
2019-12-12 10:16:05 +08:00
0
);
2021-03-16 05:40:42 -04:00
game->gfx.impactSprites[i] = &gSprites[spriteId];
game->gfx.impactSprites[i]->oam.priority = 1;
game->gfx.impactSprites[i]->invisible = TRUE;
game->gfx.impactSprites[i]->coordOffsetEnabled = TRUE;
game->gfx.impactSprites[i]->animPaused = TRUE;
}
// Create sprites for sparkle effect
for (i = 0; i < ARRAY_COUNT(game->gfx.sparkleSprites); i++)
{
spriteId = CreateSprite(
&sSpriteTemplate_Sparkle,
sSparkleCoords[i][0] + 120,
sSparkleCoords[i][1] + 136,
2019-12-12 10:16:05 +08:00
6
);
2021-03-16 05:40:42 -04:00
game->gfx.sparkleSprites[i] = &gSprites[spriteId];
game->gfx.sparkleSprites[i]->oam.priority = 3;
game->gfx.sparkleSprites[i]->invisible = TRUE;
game->gfx.sparkleSprites[i]->animPaused = TRUE;
game->gfx.sparkleSprites[i]->data[0] = i;
}
// Create sprites for timer
for (i = 0; i < ARRAY_COUNT(game->gfx.timerSprites); i++)
{
spriteId = CreateSprite(&sSpriteTemplate_Timer, 24 * i + 176, 8, 0);
game->gfx.timerSprites[i] = &gSprites[spriteId];
game->gfx.timerSprites[i]->oam.priority = 0;
game->gfx.timerSprites[i]->invisible = FALSE;
game->gfx.timerSprites[i]->animPaused = FALSE;
}
DigitObjUtil_CreatePrinter(0, 0, &sDigitObjTemplates[0]);
DigitObjUtil_CreatePrinter(1, 0, &sDigitObjTemplates[1]);
DigitObjUtil_CreatePrinter(2, 0, &sDigitObjTemplates[2]);
2021-03-16 05:40:42 -04:00
if (game->gameState == STATE_INIT)
HideTimer(&game->gfx);
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
static void DestroyGameSprites(struct BerryCrushGame *game)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i = 0;
FreeSpriteTilesByTag(TAG_TIMER_DIGITS);
FreeSpriteTilesByTag(GFXTAG_SPARKLE);
FreeSpriteTilesByTag(GFXTAG_IMPACT);
FreeSpriteTilesByTag(TAG_CRUSHER_BASE);
FreeSpritePaletteByTag(TAG_TIMER_DIGITS);
FreeSpritePaletteByTag(PALTAG_EFFECT);
FreeSpritePaletteByTag(TAG_CRUSHER_BASE);
for (i = 0; i < ARRAY_COUNT(game->gfx.timerSprites); i++)
DestroySprite(game->gfx.timerSprites[i]);
2020-04-08 16:24:30 -04:00
DigitObjUtil_DeletePrinter(2);
DigitObjUtil_DeletePrinter(1);
DigitObjUtil_DeletePrinter(0);
2021-03-16 05:40:42 -04:00
for (i = 0; i < ARRAY_COUNT(game->gfx.sparkleSprites); i++)
DestroySprite(game->gfx.sparkleSprites[i]);
for (i = 0; i < game->playerCount; i++)
DestroySprite(game->gfx.impactSprites[i]);
if (game->gfx.coreSprite->inUse)
DestroySprite(game->gfx.coreSprite);
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
static void SpriteCB_Impact(struct Sprite *sprite)
2019-12-12 10:16:05 +08:00
{
if (sprite->animEnded)
{
sprite->invisible = TRUE;
sprite->animPaused = TRUE;
}
}
2021-03-16 05:40:42 -04:00
static void SpriteCB_Sparkle_End(struct Sprite *sprite)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i;
for (i = 0; i < ARRAY_COUNT(sprite->data); i++)
sprite->data[i] = 0;
2021-07-07 09:11:52 -04:00
sprite->x2 = 0;
sprite->y2 = 0;
2019-12-12 10:16:05 +08:00
sprite->invisible = TRUE;
sprite->animPaused = TRUE;
2021-03-16 05:40:42 -04:00
sprite->callback = SpriteCallbackDummy;
2019-12-12 10:16:05 +08:00
}
2021-03-16 05:40:42 -04:00
#define sX data[0]
#define sYSpeed data[1]
#define sYAccel data[2]
#define sXSpeed data[3]
#define sSinIdx data[4]
#define sSinSpeed data[5]
#define sAmplitude data[6]
// The last element (data[7]) is a bitfield.
// The first 15 bits are the y coord to stop at.
// The last bit is a flag for whether or not to move on the x too
#define sBitfield data[7]
#define MASK_TARGET_Y 0x7FFF
#define F_MOVE_HORIZ 0x8000
static void SpriteCB_Sparkle(struct Sprite *sprite)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
s16 *data = sprite->data;
2019-12-12 10:16:05 +08:00
2021-03-16 05:40:42 -04:00
sYSpeed += sYAccel;
2021-07-07 09:11:52 -04:00
sprite->y2 += sYSpeed >> 8;
2021-03-16 05:40:42 -04:00
if (sBitfield & F_MOVE_HORIZ)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
sprite->sX += sXSpeed;
sSinIdx += sSinSpeed;
2021-07-07 09:11:52 -04:00
sprite->x2 = Sin(sSinIdx >> 7, sAmplitude);
2021-03-16 05:40:42 -04:00
if (sBitfield & F_MOVE_HORIZ && sSinIdx >> 7 > 126)
2019-12-12 10:16:05 +08:00
{
2021-07-07 09:11:52 -04:00
sprite->x2 = 0;
2021-03-16 05:40:42 -04:00
sBitfield &= MASK_TARGET_Y;
2019-12-12 10:16:05 +08:00
}
}
2021-07-07 09:11:52 -04:00
sprite->x = sX >> 7;
if (sprite->y + sprite->y2 > (sBitfield & MASK_TARGET_Y))
2021-03-16 05:40:42 -04:00
sprite->callback = SpriteCB_Sparkle_End;
}
static void SpriteCB_Sparkle_Init(struct Sprite *sprite)
{
s16 *data = sprite->data;
s16 xMult, xDiv;
s32 var;
u32 zero = 0;
var = 640;
sYSpeed = var;
sYAccel = 32;
sBitfield = 168; // Setting bits in MASK_TARGET_Y
2021-07-07 09:11:52 -04:00
xMult = sprite->x2 * 128;
xDiv = MathUtil_Div16Shift(7, (168 - sprite->y) << 7, (var + 32) >> 1);
sprite->sX = sprite->x << 7;
2021-03-16 05:40:42 -04:00
sXSpeed = MathUtil_Div16Shift(7, xMult, xDiv);
var = MathUtil_Mul16Shift(7, xDiv, 85);
sSinIdx = zero;
sSinSpeed = MathUtil_Div16Shift(7, Q_8_8(63.5), var);
2021-07-07 09:11:52 -04:00
sAmplitude = sprite->x2 / 4;
2021-03-16 05:40:42 -04:00
sBitfield |= F_MOVE_HORIZ;
2021-07-07 09:11:52 -04:00
sprite->y2 = zero;
sprite->x2 = zero;
2021-03-16 05:40:42 -04:00
sprite->callback = SpriteCB_Sparkle;
2019-12-12 10:16:05 +08:00
sprite->animPaused = FALSE;
sprite->invisible = FALSE;
}
2021-03-16 05:40:42 -04:00
#undef sX
#undef sYSpeed
#undef sYAccel
#undef sXSpeed
#undef sSinIdx
#undef sSinSpeed
#undef sAmplitude
#undef sBitfield
#undef MASK_TARGET_Y
#undef F_MOVE_HORIZ
static void RunOrScheduleCommand(u16 cmdId, u8 mode, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
struct BerryCrushGame *game = GetBerryCrushGame();
2019-12-12 10:16:05 +08:00
2021-03-16 05:40:42 -04:00
if (cmdId >= ARRAY_COUNT(sBerryCrushCommands))
cmdId = CMD_NONE;
switch (mode)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
case RUN_CMD:
if (cmdId != CMD_NONE)
sBerryCrushCommands[cmdId](game, args);
if (game->nextCmd >= ARRAY_COUNT(sBerryCrushCommands))
game->nextCmd = CMD_NONE;
game->cmdCallback = sBerryCrushCommands[game->nextCmd];
2019-12-12 10:16:05 +08:00
break;
2021-03-16 05:40:42 -04:00
case SCHEDULE_CMD:
game->cmdCallback = sBerryCrushCommands[cmdId];
2019-12-12 10:16:05 +08:00
break;
}
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_BeginNormalPaletteFade(struct BerryCrushGame *game, u8 *args)
2020-05-21 12:49:41 -04:00
{
2021-03-16 05:40:42 -04:00
// args points to packed values:
2020-05-21 12:49:41 -04:00
// bytes 0-3: selectedPals (bitfield)
// byte 4: delay
// byte 5: startY
// byte 6: stopY
// bytes 7-8: fade color
// byte 9: if TRUE, communicate on fade complete
u16 color;
2020-09-10 10:07:16 -04:00
u32 selectedPals[2];
2021-03-16 05:40:42 -04:00
selectedPals[0] = (u32)args[0];
selectedPals[1] = (u32)args[1];
2020-09-10 18:07:46 -04:00
selectedPals[1] <<= 8;
2020-05-21 12:49:41 -04:00
2020-09-10 18:07:46 -04:00
selectedPals[0] |= selectedPals[1];
2021-03-16 05:40:42 -04:00
selectedPals[1] = (u32)args[2];
2020-09-10 18:07:46 -04:00
selectedPals[1] <<= 16;
2020-05-21 12:49:41 -04:00
2020-09-10 18:07:46 -04:00
selectedPals[0] |= selectedPals[1];
2021-03-16 05:40:42 -04:00
selectedPals[1] = (u32)args[3];
2020-09-10 18:07:46 -04:00
selectedPals[1] <<= 24;
2020-09-10 10:07:16 -04:00
2020-09-10 18:07:46 -04:00
selectedPals[0] |= selectedPals[1];
2021-03-16 05:40:42 -04:00
args[0] = args[9];
2020-09-10 10:07:16 -04:00
2021-03-16 05:40:42 -04:00
color = args[8];
2020-09-10 18:07:46 -04:00
color <<= 8;
2021-03-16 05:40:42 -04:00
color |= args[7];
2019-12-12 10:16:05 +08:00
gPaletteFade.bufferTransferDisabled = FALSE;
2021-03-16 05:40:42 -04:00
BeginNormalPaletteFade(selectedPals[0], args[4], args[5], args[6], color);
2019-12-12 10:16:05 +08:00
UpdatePaletteFade();
2021-03-16 05:40:42 -04:00
game->nextCmd = CMD_WAIT_FADE;
2019-12-12 10:16:05 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_WaitPaletteFade(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-12 10:16:05 +08:00
{
case 0:
if (UpdatePaletteFade())
return 0;
2021-03-16 05:40:42 -04:00
if(args[0] != 0)
game->cmdState++;
2019-12-12 10:16:05 +08:00
else
2021-03-16 05:40:42 -04:00
game->cmdState = 3;
2019-12-12 10:16:05 +08:00
return 0;
case 1:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-12 10:16:05 +08:00
return 0;
case 2:
if (IsLinkTaskFinished())
{
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-12 10:16:05 +08:00
return 0;
}
return 0;
case 3:
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(game->afterPalFadeCmd, SCHEDULE_CMD, NULL);
game->cmdState = 0;
2019-12-12 10:16:05 +08:00
return 0;
default:
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-12 10:16:05 +08:00
return 0;
}
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_PrintMessage(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
u16 keys = args[3];
keys <<= 8;
keys |= args[2];
2019-12-12 10:16:05 +08:00
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-12 10:16:05 +08:00
{
case 0:
2022-05-27 02:18:52 +02:00
DrawDialogueFrame(0, FALSE);
2021-03-16 05:40:42 -04:00
if (args[1] & F_MSG_EXPAND)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
StringExpandPlaceholders(gStringVar4, sMessages[args[0]]);
AddTextPrinterParameterized2(0, FONT_NORMAL, gStringVar4, game->textSpeed, 0, TEXT_COLOR_DARK_GRAY, TEXT_COLOR_WHITE, TEXT_COLOR_LIGHT_GRAY);
2019-12-12 10:16:05 +08:00
}
else
{
AddTextPrinterParameterized2(0, FONT_NORMAL, sMessages[args[0]], game->textSpeed, 0, TEXT_COLOR_DARK_GRAY, TEXT_COLOR_WHITE, TEXT_COLOR_LIGHT_GRAY);
2019-12-12 10:16:05 +08:00
}
2021-11-03 15:29:18 -04:00
CopyWindowToVram(0, COPYWIN_FULL);
2019-12-12 10:16:05 +08:00
break;
case 1:
if (!IsTextPrinterActive(0))
{
2021-03-16 05:40:42 -04:00
// If no wait keys are given, skip
// waiting state below
if (keys == 0)
game->cmdState++;
2019-12-12 10:16:05 +08:00
break;
}
return 0;
case 2:
2021-03-16 05:40:42 -04:00
if (!JOY_NEW(keys))
2019-12-12 10:16:05 +08:00
return 0;
break;
case 3:
2021-03-16 05:40:42 -04:00
if (args[1] & F_MSG_CLEAR)
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, TRUE);
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(game->nextCmd, SCHEDULE_CMD, NULL);
game->cmdState = args[4];
2019-12-12 10:16:05 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-12 10:16:05 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_ShowGameDisplay(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
if (ShowGameDisplay())
RunOrScheduleCommand(game->nextCmd, RUN_CMD, game->commandArgs);
2019-12-12 10:16:05 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_HideGameDisplay(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
if (HideGameDisplay())
RunOrScheduleCommand(game->nextCmd, RUN_CMD, game->commandArgs);
2019-12-12 10:16:05 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_SignalReadyToBegin(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-12 10:16:05 +08:00
{
case 0:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-12 10:16:05 +08:00
break;
case 1:
if (IsLinkTaskFinished())
{
2020-08-20 18:02:00 -04:00
PlayNewMapMusic(MUS_RG_GAME_CORNER);
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_ASK_PICK_BERRY, SCHEDULE_CMD, NULL);
game->gameState = STATE_PICK_BERRY;
game->cmdState = 0;
2019-12-12 10:16:05 +08:00
}
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-12 10:16:05 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_AskPickBerry(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-12 10:16:05 +08:00
{
default:
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-12 10:16:05 +08:00
break;
case 0:
2021-03-16 05:40:42 -04:00
ResetGame(game);
SetPrintMessageArgs(args, MSG_PICK_BERRY, F_MSG_CLEAR, 0, 1);
game->nextCmd = CMD_ASK_PICK_BERRY;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
2019-12-12 10:16:05 +08:00
break;
case 1:
2021-03-16 05:40:42 -04:00
game->nextCmd = CMD_PICK_BERRY;
RunOrScheduleCommand(CMD_HIDE_GAME, SCHEDULE_CMD, NULL);
game->cmdState = 2;
2019-12-12 10:16:05 +08:00
break;
}
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_GoToBerryPouch(struct BerryCrushGame *game, u8 *args)
2019-12-12 10:16:05 +08:00
{
2021-03-16 05:40:42 -04:00
game->cmdCallback = NULL;
SetMainCallback2(ChooseBerry);
2019-12-12 10:16:05 +08:00
return 0;
}
2019-12-15 14:04:15 +08:00
2021-03-16 05:40:42 -04:00
static u32 Cmd_WaitForOthersToPickBerries(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i;
2019-12-15 14:04:15 +08:00
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-15 14:04:15 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
SetPrintMessageArgs(args, MSG_WAIT_PICK, 0, 0, 1);
game->nextCmd = CMD_WAIT_BERRIES;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
2019-12-15 14:04:15 +08:00
return 0;
case 1:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-15 14:04:15 +08:00
break;
case 2:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
// Send player's chosen berry to partners
memset(game->sendCmd, 0, sizeof(game->sendCmd));
game->sendCmd[0] = game->players[game->localId].berryId;
SendBlock(0, game->sendCmd, 2);
2019-12-15 14:04:15 +08:00
break;
case 3:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-15 14:04:15 +08:00
break;
case 4:
2021-03-16 05:40:42 -04:00
// Wait for partners responses
if (GetBlockReceivedStatus() != sReceivedPlayerBitmasks[game->playerCount - 2])
2019-12-15 14:04:15 +08:00
return 0;
2021-03-16 05:40:42 -04:00
// Read partners chosen berries
for (i = 0; i < game->playerCount; i++)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->players[i].berryId = gBlockRecvBuffer[i][0];
if (game->players[i].berryId > LAST_BERRY_INDEX + 1)
game->players[i].berryId = 0;
game->targetAPresses += gBerryCrush_BerryData[game->players[i].berryId].difficulty;
game->powder += gBerryCrush_BerryData[game->players[i].berryId].powder;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-15 14:04:15 +08:00
ResetBlockReceivedFlags();
2021-03-16 05:40:42 -04:00
game->targetDepth = MathUtil_Div32(Q_24_8(game->targetAPresses), Q_24_8(32));
2019-12-15 14:04:15 +08:00
break;
case 5:
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, TRUE);
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_DROP_BERRIES, SCHEDULE_CMD, NULL);
game->gameState = STATE_DROP_BERRIES;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_DropBerriesIntoCrusher(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-15 14:04:15 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
CreateBerrySprites(game, &game->gfx);
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-15 14:04:15 +08:00
break;
case 1:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
game->gfx.counter = 0;
game->gfx.vibrationIdx = 0;
game->gfx.numVibrations = 0;
game->gfx.vibrating = FALSE;
2019-12-15 14:04:15 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
game->gfx.berrySprites[game->gfx.counter]->callback = SpriteCB_DropBerryIntoCrusher;
game->gfx.berrySprites[game->gfx.counter]->affineAnimPaused = FALSE;
2020-08-20 18:02:00 -04:00
PlaySE(SE_BALL_THROW);
2019-12-15 14:04:15 +08:00
break;
case 3:
2021-03-16 05:40:42 -04:00
if (game->gfx.berrySprites[game->gfx.counter]->callback == SpriteCB_DropBerryIntoCrusher)
2019-12-15 14:04:15 +08:00
return 0;
2021-03-16 05:40:42 -04:00
game->gfx.berrySprites[game->gfx.counter] = NULL;
game->gfx.counter++;
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-15 14:04:15 +08:00
break;
case 4:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
if (game->gfx.counter < game->playerCount)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->cmdState = 2;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->gfx.counter = 0;
2019-12-15 14:04:15 +08:00
break;
case 5:
2021-03-16 05:40:42 -04:00
BerryCrushFreeBerrySpriteGfx(game, &game->gfx);
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-15 14:04:15 +08:00
break;
case 6:
if (!IsLinkTaskFinished())
return 0;
2020-08-20 18:02:00 -04:00
PlaySE(SE_FALL);
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_DROP_LID, SCHEDULE_CMD, NULL);
game->gameState = STATE_DROP_LID;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_DropLid(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-15 14:04:15 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
game->depth += 4;
if (game->depth < 0)
2019-12-15 14:04:15 +08:00
return 0;
2021-03-16 05:40:42 -04:00
game->depth = 0;
game->gfx.vibrationIdx = 4;
game->gfx.counter = 0;
game->gfx.numVibrations = sIntroOutroVibrationData[game->gfx.vibrationIdx][0];
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_STRENGTH);
2019-12-15 14:04:15 +08:00
break;
case 1:
2021-03-16 05:40:42 -04:00
game->vibration = sIntroOutroVibrationData[game->gfx.vibrationIdx][game->gfx.counter];
SetGpuReg(REG_OFFSET_BG0VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG2VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG3VOFS, -game->vibration);
game->gfx.counter++;
if (game->gfx.counter < game->gfx.numVibrations)
2019-12-15 14:04:15 +08:00
return 0;
2021-03-16 05:40:42 -04:00
if (game->gfx.vibrationIdx == 0)
2019-12-15 14:04:15 +08:00
break;
2021-03-16 05:40:42 -04:00
game->gfx.vibrationIdx--;
game->gfx.numVibrations = sIntroOutroVibrationData[game->gfx.vibrationIdx][0];
game->gfx.counter = 0;
2019-12-15 14:04:15 +08:00
return 0;
case 2:
2021-03-16 05:40:42 -04:00
game->vibration = 0;
2019-12-15 14:04:15 +08:00
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
SetGpuReg(REG_OFFSET_BG2VOFS, 0);
SetGpuReg(REG_OFFSET_BG3VOFS, 0);
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-15 14:04:15 +08:00
break;
case 3:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_COUNTDOWN, SCHEDULE_CMD, NULL);
game->gameState = STATE_COUNTDOWN;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_Countdown(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-15 14:04:15 +08:00
{
case 1:
if (!IsLinkTaskFinished())
return 0;
2021-03-11 14:16:40 -05:00
StartMinigameCountdown(TAG_COUNTDOWN, TAG_COUNTDOWN, 120, 80, 0);
2019-12-15 14:04:15 +08:00
break;
case 2:
2020-04-08 17:23:32 -04:00
if (IsMinigameCountdownRunning())
2019-12-15 14:04:15 +08:00
return 0;
// fallthrough
case 0:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-15 14:04:15 +08:00
break;
case 3:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
game->gfx.counter = 0;
game->gfx.vibrationIdx = 0;
game->gfx.numVibrations = 0;
game->gfx.vibrating = FALSE;
game->cmdTimer = 0;
if (game->localId == 0)
RunOrScheduleCommand(CMD_PLAY_GAME_LEADER, SCHEDULE_CMD, NULL);
2019-12-15 14:04:15 +08:00
else
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_PLAY_GAME_MEMBER, SCHEDULE_CMD, NULL);
game->gameState = STATE_PLAYING;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
// Receive and process data from all players
// Only used by the link leader
static void HandlePartnerInput(struct BerryCrushGame *game)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
u8 numPlayersPressed = 0;
u8 i = 0;
u16 timeDiff;
s32 temp = 0;
struct BerryCrushGame_LinkState *linkState;
2019-12-15 14:04:15 +08:00
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount; i++)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
linkState = (struct BerryCrushGame_LinkState *)gRecvCmds[i];
2021-03-16 05:40:42 -04:00
// Skip player if we have not received a packet from them
if ((linkState->rfuCmd & RFUCMD_MASK) != RFUCMD_SEND_PACKET)
continue;
2021-03-16 05:40:42 -04:00
if (linkState->sendFlag != SEND_GAME_STATE)
continue;
2021-03-16 05:40:42 -04:00
if (linkState->pushedAButton)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->localState.playerPressedAFlags |= sBitTable[i];
game->players[i].inputState = INPUT_STATE_HIT;
game->players[i].numAPresses++;
numPlayersPressed++;
timeDiff = game->timer - game->players[i].inputTime;
2021-03-16 05:40:42 -04:00
// If the interval between inputs is regular, the input is considered "neat"
// This counts toward the player's neatness score
if (timeDiff >= game->players[i].timeSincePrevInput - 1
2021-03-16 05:40:42 -04:00
&& timeDiff <= game->players[i].timeSincePrevInput + 1)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
// On neat input streak
game->players[i].neatInputStreak++;
game->players[i].timeSincePrevInput = timeDiff;
if (game->players[i].neatInputStreak > game->players[i].maxNeatInputStreak)
game->players[i].maxNeatInputStreak = game->players[i].neatInputStreak;
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
// End neat input streak
game->players[i].neatInputStreak = 0;
game->players[i].timeSincePrevInput = timeDiff;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->players[i].inputTime = game->timer;
game->players[i].inputFlags++;
if (game->players[i].inputFlags > F_INPUT_HIT_B)
game->players[i].inputFlags = 0;
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
game->players[i].inputState = INPUT_STATE_NONE;
}
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
if (numPlayersPressed > 1)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
// For each player that pressed A, flag their input as synchronous
// This is used to change their impact sprite to a big impact
for (i = 0; i < game->playerCount; i++)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->players[i].inputState == INPUT_STATE_NONE)
continue;
2021-03-16 05:40:42 -04:00
game->players[i].inputState |= INPUT_STATE_HIT_SYNC;
game->players[i].numSyncedAPresses++;
2019-12-15 14:04:15 +08:00
}
}
2021-03-16 05:40:42 -04:00
if (numPlayersPressed == 0)
return;
2021-03-16 05:40:42 -04:00
game->bigSparkleCounter += numPlayersPressed;
numPlayersPressed += sSyncPressBonus[numPlayersPressed - 1];
game->sparkleCounter += numPlayersPressed;
game->totalAPresses += numPlayersPressed;
if (game->targetAPresses - game->totalAPresses > 0)
{
temp = (s32)game->totalAPresses;
temp = Q_24_8(temp);
temp = MathUtil_Div32(temp, game->targetDepth);
temp = Q_24_8_TO_INT(temp);
game->newDepth = (u8)temp;
return;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
// Target number of A presses has been reached, game is complete
game->newDepth = 32;
game->localState.endGame = TRUE;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
// Updates the crusher, input flags, and timer to send to group members
// Only used by the link leader
static void UpdateLeaderGameState(struct BerryCrushGame *game)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
u8 numPlayersPressed = 0;
u16 flags = 0;
u16 temp = 0;
u8 i = 0;
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount; i++)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->players[i].inputState != INPUT_STATE_NONE)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
numPlayersPressed++;
flags = game->players[i].inputFlags + F_INPUT_HIT_A;
if (game->players[i].inputState & INPUT_STATE_HIT_SYNC)
flags |= F_INPUT_HIT_SYNC;
flags <<= INPUT_FLAGS_PER_PLAYER * i;
game->localState.inputFlags |= flags;
2019-12-15 14:04:15 +08:00
}
}
2021-03-16 05:40:42 -04:00
temp = (u16)game->newDepth;
game->localState.depth = temp;
if (numPlayersPressed == 0)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->gfx.vibrating)
game->gfx.counter++;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
else if (game->gfx.vibrating)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (numPlayersPressed != game->gfx.vibrationIdx)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->gfx.vibrationIdx = numPlayersPressed - 1;
game->gfx.numVibrations = sVibrationData[numPlayersPressed - 1][0];
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
game->gfx.counter++;
2019-12-15 14:04:15 +08:00
}
}
else
{
2021-03-16 05:40:42 -04:00
game->gfx.counter = 0;
game->gfx.vibrationIdx = numPlayersPressed - 1;
game->gfx.numVibrations = sVibrationData[numPlayersPressed - 1][0];
game->gfx.vibrating = TRUE;
}
2021-03-16 05:40:42 -04:00
if (game->gfx.vibrating)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->gfx.counter >= game->gfx.numVibrations)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->gfx.counter = 0;
game->gfx.vibrationIdx = 0;
game->gfx.numVibrations = 0;
game->gfx.vibrating = FALSE;
temp = 0;
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
temp = sVibrationData[game->gfx.vibrationIdx][game->gfx.counter + 1];
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->localState.vibration = (u8)temp;
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
game->localState.vibration = 0;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->localState.timer = game->leaderTimer;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
// Checks for input and sends data to group members
static void HandlePlayerInput(struct BerryCrushGame *game)
2019-12-15 14:04:15 +08:00
{
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
2021-03-16 05:40:42 -04:00
game->localState.pushedAButton = TRUE;
2020-09-04 21:11:55 -04:00
if (JOY_HELD(A_BUTTON))
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->players[game->localId].timePressingA < game->timer)
game->players[game->localId].timePressingA++;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
// Only send data to other players if you are the leader or you pressed A
if (game->localId != 0 && !game->localState.pushedAButton)
2019-12-15 14:04:15 +08:00
return;
2021-03-16 05:40:42 -04:00
game->localState.sendFlag = SEND_GAME_STATE;
2021-03-16 05:40:42 -04:00
// Every 30 frames, check whether the sparkles produced should be big,
// depending on how many A presses there were in that time
if (game->timer % 30 == 0)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->bigSparkleCounter > sBigSparkleThresholds[game->playerCount - 2])
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->numBigSparkles++;
game->bigSparkle = TRUE;
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
game->bigSparkle = FALSE;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->bigSparkleCounter = 0;
game->numBigSparkleChecks++;
}
2021-03-16 05:40:42 -04:00
// Every 15 frames, update the amount of sparkles that should be produced,
// depending on how many A presses there were in that time (including the bonus)
if (game->timer % 15 == 0)
{
// BUG: The wrong field is used twice below
// As a result, only a sparkleAmount of 0, 1, or 4 is attainable
#ifdef BUGFIX
#define field sparkleAmount
#else
#define field sparkleCounter
#endif
if (game->sparkleCounter < sSparkleThresholds[game->playerCount - 2][0])
game->sparkleAmount = 0;
else if (game->sparkleCounter < sSparkleThresholds[game->playerCount - 2][1])
game->sparkleAmount = 1;
else if (game->sparkleCounter < sSparkleThresholds[game->playerCount - 2][2])
game->field = 2;
else if (game->sparkleCounter < sSparkleThresholds[game->playerCount - 2][3])
game->field = 3;
2019-12-15 14:04:15 +08:00
else
2021-03-16 05:40:42 -04:00
game->sparkleAmount = 4;
game->sparkleCounter = 0;
#undef field
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
game->cmdTimer++;
if (game->cmdTimer > 60)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->cmdTimer > 70)
2019-12-15 14:04:15 +08:00
{
2021-03-01 01:54:51 -05:00
ClearRecvCommands();
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
else if (game->localState.playerPressedAFlags == 0)
2019-12-15 14:04:15 +08:00
{
2021-03-01 01:54:51 -05:00
ClearRecvCommands();
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-15 14:04:15 +08:00
}
}
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
if (game->timer >= MAX_TIME)
game->localState.endGame = TRUE;
game->localState.bigSparkle = game->bigSparkle;
game->localState.sparkleAmount = game->sparkleAmount;
memcpy(game->sendCmd, &game->localState, sizeof(game->sendCmd));
Rfu_SendPacket(game->sendCmd);
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
static void RecvLinkData(struct BerryCrushGame *game)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i = 0;
struct BerryCrushGame_LinkState *linkState = NULL;
for (i = 0; i < game->playerCount; i++)
game->players[i].inputState = INPUT_STATE_NONE;
2019-12-15 14:04:15 +08:00
if ((gRecvCmds[0][0] & RFUCMD_MASK) != RFUCMD_SEND_PACKET)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->playedSound = FALSE;
return;
2019-12-15 14:04:15 +08:00
}
if (gRecvCmds[0][1] != 2)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->playedSound = FALSE;
return;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
memcpy(game->recvCmd, gRecvCmds[0], sizeof(game->recvCmd));
linkState = (struct BerryCrushGame_LinkState *)&game->recvCmd;
game->depth = linkState->depth;
game->vibration = (s16)linkState->vibration;
game->timer = linkState->timer;
UpdateInputEffects(game, &(game->gfx));
if (linkState->endGame)
game->endGame = TRUE;
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_PlayGame_Leader(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
memset(&game->localState, 0, sizeof(game->localState));
memset(&game->recvCmd, 0, sizeof(game->recvCmd));
RecvLinkData(game);
SetGpuReg(REG_OFFSET_BG0VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG2VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG3VOFS, -game->vibration);
if (game->endGame)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->timer >= MAX_TIME)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->timer = MAX_TIME;
RunOrScheduleCommand(CMD_TIMES_UP, SCHEDULE_CMD, NULL);
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_FINISH_GAME, SCHEDULE_CMD, NULL);
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
else
{
2021-03-16 05:40:42 -04:00
game->leaderTimer++;
HandlePartnerInput(game);
UpdateLeaderGameState(game);
HandlePlayerInput(game);
2019-12-15 14:04:15 +08:00
return 0;
}
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_PlayGame_Member(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
memset(&game->localState, 0, sizeof(game->localState));
memset(&game->recvCmd, 0, sizeof(game->recvCmd));
RecvLinkData(game);
SetGpuReg(REG_OFFSET_BG0VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG2VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG3VOFS, -game->vibration);
if (game->endGame)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
if (game->timer >= MAX_TIME)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->timer = MAX_TIME;
RunOrScheduleCommand(CMD_TIMES_UP, SCHEDULE_CMD, NULL);
2019-12-15 14:04:15 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_FINISH_GAME, SCHEDULE_CMD, NULL);
2019-12-15 14:04:15 +08:00
}
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
else
{
2021-03-16 05:40:42 -04:00
HandlePlayerInput(game);
2019-12-15 14:04:15 +08:00
return 0;
}
}
2021-03-16 05:40:42 -04:00
// Game was 'won', crusher was pushed down fully before time was up
static u32 Cmd_FinishGame(struct BerryCrushGame *game, u8 *args)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-15 14:04:15 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
game->gameState = STATE_FINISHED;
2020-08-20 18:02:00 -04:00
PlaySE(SE_M_STRENGTH);
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 8, RGB(31, 31, 0));
2021-03-16 05:40:42 -04:00
game->gfx.counter = 2;
2019-12-15 14:04:15 +08:00
break;
case 1:
2021-03-16 05:40:42 -04:00
if (--game->gfx.counter != (u8)-1)
2019-12-15 14:04:15 +08:00
return 0;
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 0, RGB(31, 31, 0));
2021-03-16 05:40:42 -04:00
game->gfx.vibrationIdx = 4;
game->gfx.counter = 0;
game->gfx.numVibrations = sIntroOutroVibrationData[game->gfx.vibrationIdx][0];
2019-12-15 14:04:15 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
game->vibration = sIntroOutroVibrationData[game->gfx.vibrationIdx][game->gfx.counter];
SetGpuReg(REG_OFFSET_BG0VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG2VOFS, -game->vibration);
SetGpuReg(REG_OFFSET_BG3VOFS, -game->vibration);
if (++game->gfx.counter < game->gfx.numVibrations)
2019-12-15 14:04:15 +08:00
return 0;
2021-03-16 05:40:42 -04:00
if (game->gfx.vibrationIdx != 0)
2019-12-15 14:04:15 +08:00
{
2021-03-16 05:40:42 -04:00
game->gfx.vibrationIdx--;
game->gfx.numVibrations = sIntroOutroVibrationData[game->gfx.vibrationIdx][0];
game->gfx.counter = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
break;
case 3:
2021-03-16 05:40:42 -04:00
game->vibration = 0;
2019-12-15 14:04:15 +08:00
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
SetGpuReg(REG_OFFSET_BG2VOFS, 0);
SetGpuReg(REG_OFFSET_BG3VOFS, 0);
break;
case 4:
2021-03-16 05:40:42 -04:00
if (!AreEffectsFinished(game, &game->gfx))
2019-12-15 14:04:15 +08:00
return 0;
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-15 14:04:15 +08:00
break;
case 5:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_CALC_RESULTS, SCHEDULE_CMD, NULL);
game->cmdTimer = 0;
game->cmdState = 0;
2019-12-15 14:04:15 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-15 14:04:15 +08:00
return 0;
}
2019-12-17 15:20:38 +08:00
2021-03-16 05:40:42 -04:00
static u32 Cmd_HandleTimeUp(struct BerryCrushGame *game, u8 *args)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-17 15:20:38 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
game->gameState = STATE_TIMES_UP;
2020-08-20 18:02:00 -04:00
PlaySE(SE_FAILURE);
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 8, RGB(31, 0, 0));
2021-03-16 05:40:42 -04:00
game->gfx.counter = 4;
2019-12-17 15:20:38 +08:00
break;
case 1:
2021-03-16 05:40:42 -04:00
if (--game->gfx.counter != (u8)-1)
2019-12-17 15:20:38 +08:00
return 0;
2021-02-24 11:01:02 -05:00
BlendPalettes(PALETTES_ALL, 0, RGB(31, 0, 0));
2021-03-16 05:40:42 -04:00
game->gfx.counter = 0;
2019-12-17 15:20:38 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
if (!AreEffectsFinished(game, &game->gfx))
2019-12-17 15:20:38 +08:00
return 0;
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-17 15:20:38 +08:00
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
SetGpuReg(REG_OFFSET_BG2VOFS, 0);
SetGpuReg(REG_OFFSET_BG3VOFS, 0);
break;
case 3:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
ConvertIntToDecimalStringN(gStringVar1, game->powder, STR_CONV_MODE_LEFT_ALIGN, 6);
SetPrintMessageArgs(args, MSG_TIMES_UP, F_MSG_CLEAR, 0, 0);
game->nextCmd = CMD_SAVE;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
game->cmdTimer = 0;
game->cmdState = 0;
2019-12-17 15:20:38 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-17 15:20:38 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_TabulateResults(struct BerryCrushGame *game, u8 *args)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i, j, tempPlayerId;
s32 temp1, temp2;
u16 tempStat;
2019-12-17 15:20:38 +08:00
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-17 15:20:38 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
memset(game->sendCmd, 0, 2 * sizeof(u16));
if (game->players[game->localId].timePressingA > game->timer)
game->players[game->localId].timePressingA = game->timer;
game->sendCmd[0] = game->players[game->localId].timePressingA;
SendBlock(0, game->sendCmd, 2);
2019-12-17 15:20:38 +08:00
break;
case 1:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-17 15:20:38 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
if (GetBlockReceivedStatus() != sReceivedPlayerBitmasks[game->playerCount - 2])
2019-12-17 15:20:38 +08:00
return 0;
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount; i++)
game->players[i].timePressingA = gBlockRecvBuffer[i][0];
game->cmdTimer = 0;
game->sendCmd[0] = 0;
2019-12-17 15:20:38 +08:00
ResetBlockReceivedFlags();
2021-03-16 05:40:42 -04:00
// If player is not leader, skip the steps
// where the results are calculated and sent.
// Group members just read the results sent
// to them by the leader.
if (game->localId == 0)
game->cmdState = 3;
2019-12-17 15:20:38 +08:00
else
2021-03-16 05:40:42 -04:00
game->cmdState = 6;
2019-12-17 15:20:38 +08:00
return 0;
case 3:
2021-03-16 05:40:42 -04:00
memset(&game->results, 0, sizeof(game->results));
game->results.time = game->timer;
game->results.targetPressesPerSec = game->targetAPresses / (game->timer / 60);
// Calculate silkiness
// Silkiness is the percentage of times big sparkles were produced when possible,
// which itself depends on the number of A presses every 30 frames
temp1 = MathUtil_Mul32(Q_24_8(game->numBigSparkles), Q_24_8(50));
temp1 = MathUtil_Div32(temp1, Q_24_8(game->numBigSparkleChecks)) + Q_24_8(50);
temp1 = Q_24_8_TO_INT(temp1);
game->results.silkiness = temp1 & 0x7F;
// Calculate amount of powder
temp1 = Q_24_8(temp1);
temp1 = MathUtil_Div32(temp1, Q_24_8(100));
temp2 = Q_24_8(game->powder * game->playerCount);
temp2 = MathUtil_Mul32(temp2, temp1);
game->results.powder = Q_24_8_TO_INT(temp2);
// Choose random second results page
game->results.randomPageId = Random() % NUM_RANDOM_RESULTS_PAGES;
for (i = 0; i < game->playerCount; i++)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
game->results.playerIdsRanked[RESULTS_PAGE_PRESSES][i] = i;
game->results.playerIdsRanked[RESULTS_PAGE_RANDOM][i] = i;
game->results.stats[RESULTS_PAGE_PRESSES][i] = game->players[i].numAPresses;
game->results.totalAPresses += game->results.stats[RESULTS_PAGE_PRESSES][i];
2021-03-16 05:40:42 -04:00
// Calculate value for random second results page
switch (game->results.randomPageId)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
case RESULTS_PAGE_NEATNESS:
if (game->players[i].numAPresses != 0)
2019-12-17 15:20:38 +08:00
{
// Calculate percentage of inputs that were in largest "neat" streak
2021-03-16 05:40:42 -04:00
// "Neat" inputs are those done at a regular interval
temp1 = game->players[i].maxNeatInputStreak;
temp1 = Q_24_8(temp1);
temp1 = MathUtil_Mul32(temp1, Q_24_8(100));
temp2 = game->players[i].numAPresses;
temp2 = Q_24_8(temp2);
temp2 = MathUtil_Div32(temp1, temp2);
2019-12-17 15:20:38 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
temp2 = 0;
2019-12-17 15:20:38 +08:00
}
break;
2021-03-16 05:40:42 -04:00
case RESULTS_PAGE_COOPERATIVE:
if (game->players[i].numAPresses != 0)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
// Calculate percentage of inputs that were
// done at the same time as another player
temp1 = game->players[i].numSyncedAPresses;
temp1 = Q_24_8(temp1);
temp1 = MathUtil_Mul32(temp1, Q_24_8(100));
temp2 = game->players[i].numAPresses;
temp2 = Q_24_8(temp2);
temp2 = MathUtil_Div32(temp1, temp2);
2019-12-17 15:20:38 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
temp2 = 0;
2019-12-17 15:20:38 +08:00
}
break;
2021-03-16 05:40:42 -04:00
case RESULTS_PAGE_POWER:
if (game->players[i].numAPresses == 0)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
temp2 = 0;
2019-12-17 15:20:38 +08:00
}
2021-03-16 05:40:42 -04:00
else if (game->players[i].timePressingA >= game->timer)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
// Spent 100% of the time pressing A
temp2 = Q_24_8(100);
2019-12-17 15:20:38 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
// Calculate percentage of time the
// player spent pressing A
temp1 = game->players[i].timePressingA;
temp1 = Q_24_8(temp1);
temp1 = MathUtil_Mul32(temp1, Q_24_8(100));
temp2 = game->timer;
temp2 = Q_24_8(temp2);
temp2 = MathUtil_Div32(temp1, temp2);
2019-12-17 15:20:38 +08:00
}
break;
}
2021-03-16 05:40:42 -04:00
temp2 >>= 4;
game->results.stats[RESULTS_PAGE_RANDOM][i] = temp2;
2019-12-17 15:20:38 +08:00
}
break;
case 4:
2021-03-16 05:40:42 -04:00
for (i = 0; i < game->playerCount - 1; i++)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
for (j = game->playerCount - 1; j > i; j--)
2019-12-17 15:20:38 +08:00
{
2021-03-16 05:40:42 -04:00
// Calculate player rankings for "Number of Presses" by sorting arrays
if (game->results.stats[RESULTS_PAGE_PRESSES][j - 1] < game->results.stats[RESULTS_PAGE_PRESSES][j])
2019-12-17 15:20:38 +08:00
{
SWAP(game->results.stats[RESULTS_PAGE_PRESSES][j],
2021-03-16 05:40:42 -04:00
game->results.stats[RESULTS_PAGE_PRESSES][j - 1],
tempStat);
SWAP(game->results.playerIdsRanked[RESULTS_PAGE_PRESSES][j],
game->results.playerIdsRanked[RESULTS_PAGE_PRESSES][j - 1],
tempPlayerId);
2019-12-17 15:20:38 +08:00
}
2021-03-16 05:40:42 -04:00
// Calculate player rankings for random second results page by sorting arrays
if (game->results.stats[RESULTS_PAGE_RANDOM][j - 1] < game->results.stats[RESULTS_PAGE_RANDOM][j])
2019-12-17 15:20:38 +08:00
{
SWAP(game->results.stats[RESULTS_PAGE_RANDOM][j],
2021-03-16 05:40:42 -04:00
game->results.stats[RESULTS_PAGE_RANDOM][j - 1],
tempStat);
SWAP(game->results.playerIdsRanked[RESULTS_PAGE_RANDOM][j],
game->results.playerIdsRanked[RESULTS_PAGE_RANDOM][j - 1],
tempPlayerId);
2019-12-17 15:20:38 +08:00
}
}
}
2021-03-16 05:40:42 -04:00
SendBlock(0, &game->results, sizeof(game->results));
2019-12-17 15:20:38 +08:00
break;
case 5:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-17 15:20:38 +08:00
break;
case 6:
if (GetBlockReceivedStatus() != 1)
return 0;
2021-03-16 05:40:42 -04:00
// Receive results calculated by leader
memset(&game->results, 0, sizeof(game->results));
memcpy(&game->results, gBlockRecvBuffer, sizeof(game->results));
2019-12-17 15:20:38 +08:00
ResetBlockReceivedFlags();
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-17 15:20:38 +08:00
break;
case 7:
2021-03-16 05:40:42 -04:00
SaveResults();
RunOrScheduleCommand(CMD_SHOW_RESULTS, SCHEDULE_CMD, NULL);
game->gameState = STATE_RESULTS_PRESSES;
game->cmdState = 0;
game->newDepth = 0;
2019-12-17 15:20:38 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-17 15:20:38 +08:00
return 0;
}
2019-12-18 03:35:41 +08:00
2021-03-16 05:40:42 -04:00
static u32 Cmd_ShowResults(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
if (!OpenResultsWindow(game, &game->gfx))
2019-12-18 03:35:41 +08:00
return 0;
break;
case 1:
CopyBgTilemapBufferToVram(0);
2021-03-16 05:40:42 -04:00
game->gfx.counter = 30;
2019-12-18 03:35:41 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
if (game->gfx.counter != 0)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
game->gfx.counter--;
2019-12-18 03:35:41 +08:00
return 0;
}
2020-09-04 21:11:55 -04:00
if (!(JOY_NEW(A_BUTTON)))
2019-12-18 03:35:41 +08:00
return 0;
PlaySE(SE_SELECT);
2021-03-16 05:40:42 -04:00
CloseResultsWindow(game);
2019-12-18 03:35:41 +08:00
break;
case 3:
2021-03-16 05:40:42 -04:00
// Progress through each page of the results
if (game->gameState < RESULTS_STATE_END)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
game->gameState++;
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
return 0;
}
break;
case 4:
2021-03-16 05:40:42 -04:00
// Print message showing how much powder was created
ConvertIntToDecimalStringN(gStringVar1, game->powder, STR_CONV_MODE_LEFT_ALIGN, 6);
2019-12-18 04:17:10 +08:00
ConvertIntToDecimalStringN(gStringVar2, GetBerryPowder(), STR_CONV_MODE_LEFT_ALIGN, 6);
2021-03-16 05:40:42 -04:00
SetPrintMessageArgs(args, MSG_POWDER, F_MSG_CLEAR | F_MSG_EXPAND, 0, 0);
game->nextCmd = CMD_SAVE;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_SaveGame(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
if (game->timer >= MAX_TIME)
HideTimer(&game->gfx);
SetPrintMessageArgs(args, MSG_COMM_STANDBY, 0, 0, 1);
game->nextCmd = CMD_SAVE;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
game->cmdState = 0; // State is progressed by CMD_PRINT_MSG
2019-12-18 03:35:41 +08:00
return 0;
case 1:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-18 03:35:41 +08:00
break;
case 2:
if (!IsLinkTaskFinished())
return 0;
2022-05-27 02:18:52 +02:00
DrawDialogueFrame(0, FALSE);
AddTextPrinterParameterized2(0, FONT_NORMAL, gText_SavingDontTurnOffPower, 0, 0, TEXT_COLOR_DARK_GRAY, TEXT_COLOR_WHITE, TEXT_COLOR_LIGHT_GRAY);
2021-11-03 15:29:18 -04:00
CopyWindowToVram(0, COPYWIN_FULL);
2021-10-28 22:54:41 -04:00
CreateTask(Task_LinkFullSave, 0);
2019-12-18 03:35:41 +08:00
break;
case 3:
2021-10-28 22:54:41 -04:00
if (FuncIsActiveTask(Task_LinkFullSave))
2019-12-18 03:35:41 +08:00
return 0;
break;
case 4:
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_ASK_PLAY_AGAIN, SCHEDULE_CMD, NULL);
game->gameState = STATE_PLAY_AGAIN;
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_AskPlayAgain(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
s8 input = 0;
2019-12-18 03:35:41 +08:00
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
SetPrintMessageArgs(args, MSG_PLAY_AGAIN, 0, 0, 1);
game->nextCmd = CMD_ASK_PLAY_AGAIN;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
game->cmdState = 0; // State is progressed by CMD_PRINT_MSG
2019-12-18 03:35:41 +08:00
return 0;
case 1:
DisplayYesNoMenuDefaultYes();
break;
case 2:
2021-03-16 05:40:42 -04:00
input = Menu_ProcessInputNoWrapClearOnChoose();
if (input != -2)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
memset(game->sendCmd, 0, sizeof(game->sendCmd));
if (input == 0)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
// Selected Yes
2019-12-18 03:35:41 +08:00
if (HasAtLeastOneBerry())
2021-03-16 05:40:42 -04:00
game->playAgainState = PLAY_AGAIN_YES;
2019-12-18 03:35:41 +08:00
else
2021-03-16 05:40:42 -04:00
game->playAgainState = PLAY_AGAIN_NO_BERRIES;
2019-12-18 03:35:41 +08:00
}
else
{
2021-03-16 05:40:42 -04:00
// Selected No
game->playAgainState = PLAY_AGAIN_NO;
2019-12-18 03:35:41 +08:00
}
2021-03-16 05:40:42 -04:00
// Close Yes/No and start communication
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, TRUE);
2021-03-16 05:40:42 -04:00
SetPrintMessageArgs(args, MSG_COMM_STANDBY, 0, 0, 0);
game->nextCmd = CMD_COMM_PLAY_AGAIN;
RunOrScheduleCommand(CMD_PRINT_MSG, SCHEDULE_CMD, NULL);
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
}
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_CommunicatePlayAgainResponses(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
u8 i = 0;
2019-12-18 03:35:41 +08:00
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-18 03:35:41 +08:00
break;
case 1:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
// Send player's Yes/No response to partners
game->sendCmd[0] = game->playAgainState;
game->recvCmd[0] = 0;
SendBlock(0, game->sendCmd, sizeof(u16));
2019-12-18 03:35:41 +08:00
break;
case 2:
if (!IsLinkTaskFinished())
return 0;
2021-03-16 05:40:42 -04:00
game->cmdTimer = 0;
2019-12-18 03:35:41 +08:00
break;
case 3:
2021-03-16 05:40:42 -04:00
// Wait for partners responses
if (GetBlockReceivedStatus() != sReceivedPlayerBitmasks[game->playerCount - 2])
2019-12-18 03:35:41 +08:00
return 0;
2021-03-16 05:40:42 -04:00
// Read partners responses
for (i = 0; i < game->playerCount; i++)
game->recvCmd[0] += gBlockRecvBuffer[i][0];
if (game->recvCmd[0] != PLAY_AGAIN_YES)
RunOrScheduleCommand(CMD_PLAY_AGAIN_NO, SCHEDULE_CMD, NULL);
2019-12-18 03:35:41 +08:00
else
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_PLAY_AGAIN_YES, SCHEDULE_CMD, NULL);
2019-12-18 03:35:41 +08:00
ResetBlockReceivedFlags();
2021-03-16 05:40:42 -04:00
game->sendCmd[0] = 0;
game->recvCmd[0] = 0;
game->cmdTimer = 0;
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_PlayAgain(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2021-03-16 05:40:42 -04:00
BeginNormalPaletteFade(PALETTES_ALL, 1, 0, 16, RGB_BLACK);
2019-12-18 03:35:41 +08:00
UpdatePaletteFade();
break;
case 1:
if (UpdatePaletteFade())
return 0;
break;
case 2:
2022-07-25 14:59:14 -04:00
ClearDialogWindowAndFrame(0, TRUE);
2021-03-16 05:40:42 -04:00
ResetCrusherPos(game);
BeginNormalPaletteFade(PALETTES_ALL, 0, 16, 0, RGB_BLACK);
2019-12-18 03:35:41 +08:00
UpdatePaletteFade();
break;
case 3:
if (UpdatePaletteFade())
return 0;
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_ASK_PICK_BERRY, SCHEDULE_CMD, NULL);
game->gameState = STATE_PICK_BERRY;
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_StopGame(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2022-05-27 02:18:52 +02:00
DrawDialogueFrame(0, FALSE);
2021-03-16 05:40:42 -04:00
if (game->playAgainState == PLAY_AGAIN_NO_BERRIES)
AddTextPrinterParameterized2(0, FONT_NORMAL, sMessages[MSG_NO_BERRIES], game->textSpeed, 0, TEXT_COLOR_DARK_GRAY, TEXT_COLOR_WHITE, TEXT_COLOR_LIGHT_GRAY);
2019-12-18 03:35:41 +08:00
else
AddTextPrinterParameterized2(0, FONT_NORMAL, sMessages[MSG_DROPPED], game->textSpeed, 0, TEXT_COLOR_DARK_GRAY, TEXT_COLOR_WHITE, TEXT_COLOR_LIGHT_GRAY);
2021-11-03 15:29:18 -04:00
CopyWindowToVram(0, COPYWIN_FULL);
2019-12-18 03:35:41 +08:00
break;
case 1:
if (IsTextPrinterActive(0))
return 0;
2021-03-16 05:40:42 -04:00
game->gfx.counter = 120;
2019-12-18 03:35:41 +08:00
break;
case 2:
2021-03-16 05:40:42 -04:00
if (game->gfx.counter != 0)
game->gfx.counter--;
2019-12-18 03:35:41 +08:00
else
{
2021-03-16 05:40:42 -04:00
RunOrScheduleCommand(CMD_CLOSE_LINK, SCHEDULE_CMD, NULL);
game->cmdState = 0;
2019-12-18 03:35:41 +08:00
}
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_CloseLink(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2021-03-16 05:40:42 -04:00
switch (game->cmdState)
2019-12-18 03:35:41 +08:00
{
case 0:
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2019-12-18 03:35:41 +08:00
break;
case 1:
if (!IsLinkTaskFinished())
return 0;
2020-08-13 03:09:47 -04:00
SetCloseLinkCallback();
2019-12-18 03:35:41 +08:00
break;
case 2:
if (gReceivedRemoteLinkPlayers != 0)
return 0;
2021-03-16 05:40:42 -04:00
game->nextCmd = CMD_QUIT;
RunOrScheduleCommand(CMD_HIDE_GAME, SCHEDULE_CMD, NULL);
game->cmdState = 2; // ???
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
game->cmdState++;
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static u32 Cmd_Quit(struct BerryCrushGame *game, u8 *args)
2019-12-18 03:35:41 +08:00
{
2020-09-13 00:26:01 -04:00
QuitBerryCrush(NULL);
2019-12-18 03:35:41 +08:00
return 0;
}
2021-03-16 05:40:42 -04:00
static void ResetGame(struct BerryCrushGame *game)
{
u8 i = 0;
IncrementGameStat(GAME_STAT_PLAYED_BERRY_CRUSH);
game->unused = 0;
game->cmdTimer = 0;
game->gameState = STATE_RESET;
game->playAgainState = 0;
game->powder = 0;
game->targetAPresses = 0;
game->totalAPresses = 0;
game->targetDepth = 0;
game->newDepth = 0;
game->noRoomForPowder = FALSE;
game->newRecord = FALSE;
game->playedSound = FALSE;
game->endGame = FALSE;
game->bigSparkle = FALSE;
game->sparkleAmount = 0;
game->leaderTimer = 0;
game->timer = 0;
game->bigSparkleCounter = 0;
game->numBigSparkleChecks = -1;
game->numBigSparkles = 0;
game->sparkleCounter = 0;
for (i = 0; i < MAX_RFU_PLAYERS; i++)
{
game->players[i].berryId = -1;
game->players[i].inputTime = 0;
game->players[i].neatInputStreak = 0;
game->players[i].timeSincePrevInput = 1;
game->players[i].maxNeatInputStreak = 0;
game->players[i].numAPresses = 0;
game->players[i].numSyncedAPresses = 0;
game->players[i].timePressingA = 0;
game->players[i].inputFlags = 0;
game->players[i].inputState = INPUT_STATE_NONE;
}
}
static void SetPaletteFadeArgs(u8 *args, bool8 communicateAfter, u32 selectedPals, s8 delay, u8 startY, u8 targetY, u16 palette)
{
args[0] = ((u8 *)&selectedPals)[0];
args[1] = ((u8 *)&selectedPals)[1];
args[2] = ((u8 *)&selectedPals)[2];
args[3] = ((u8 *)&selectedPals)[3];
args[4] = delay;
args[5] = startY;
args[6] = targetY;
args[7] = ((u8 *)&palette)[0];
args[8] = ((u8 *)&palette)[1];
args[9] = communicateAfter;
}
static void SetPrintMessageArgs(u8 *args, u8 msgId, u8 flags, u16 waitKeys, u8 followupState)
{
args[0] = msgId;
args[1] = flags;
args[2] = ((u8 *)&waitKeys)[0];
args[3] = ((u8 *)&waitKeys)[1];
args[4] = followupState;
2019-12-18 03:35:41 +08:00
}