mirror of
https://github.com/Ninjdai1/pokeemerald.git
synced 2025-01-21 19:00:18 +01:00
2571 lines
67 KiB
C
2571 lines
67 KiB
C
#include "global.h"
|
|
#include "naming_screen.h"
|
|
#include "malloc.h"
|
|
#include "palette.h"
|
|
#include "task.h"
|
|
#include "sprite.h"
|
|
#include "string_util.h"
|
|
#include "window.h"
|
|
#include "bg.h"
|
|
#include "gpu_regs.h"
|
|
#include "pokemon.h"
|
|
#include "field_specials.h"
|
|
#include "field_player_avatar.h"
|
|
#include "event_object_movement.h"
|
|
#include "event_data.h"
|
|
#include "constants/songs.h"
|
|
#include "pokemon_storage_system.h"
|
|
#include "graphics.h"
|
|
#include "sound.h"
|
|
#include "trig.h"
|
|
#include "field_effect.h"
|
|
#include "pokemon_icon.h"
|
|
#include "data.h"
|
|
#include "strings.h"
|
|
#include "menu.h"
|
|
#include "text_window.h"
|
|
#include "overworld.h"
|
|
#include "constants/event_objects.h"
|
|
#include "constants/rgb.h"
|
|
|
|
enum {
|
|
INPUT_NONE,
|
|
INPUT_DPAD_UP,
|
|
INPUT_DPAD_DOWN,
|
|
INPUT_DPAD_LEFT,
|
|
INPUT_DPAD_RIGHT,
|
|
INPUT_A_BUTTON,
|
|
INPUT_B_BUTTON,
|
|
INPUT_LR_BUTTON,
|
|
INPUT_SELECT,
|
|
INPUT_START,
|
|
};
|
|
|
|
#define KBROW_COUNT 4
|
|
#define KBCOL_COUNT 8
|
|
|
|
enum {
|
|
GFXTAG_BACK_BUTTON,
|
|
GFXTAG_OK_BUTTON,
|
|
GFXTAG_PAGE_SWAP_FRAME,
|
|
GFXTAG_PAGE_SWAP_BUTTON,
|
|
GFXTAG_PAGE_SWAP_UPPER,
|
|
GFXTAG_PAGE_SWAP_LOWER,
|
|
GFXTAG_PAGE_SWAP_OTHERS,
|
|
GFXTAG_CURSOR,
|
|
GFXTAG_CURSOR_SQUISHED,
|
|
GFXTAG_CURSOR_FILLED,
|
|
GFXTAG_INPUT_ARROW,
|
|
GFXTAG_UNDERSCORE,
|
|
};
|
|
|
|
enum {
|
|
PALTAG_PC_ICON,
|
|
PALTAG_PAGE_SWAP_UPPER,
|
|
PALTAG_PAGE_SWAP_LOWER,
|
|
PALTAG_PAGE_SWAP_OTHERS,
|
|
PALTAG_PAGE_SWAP,
|
|
PALTAG_CURSOR,
|
|
PALTAG_BACK_BUTTON,
|
|
PALTAG_OK_BUTTON,
|
|
};
|
|
|
|
enum {
|
|
WIN_KB_PAGE_1, // Which of these two windows is in front is cycled as the player swaps
|
|
WIN_KB_PAGE_2, // Initially WIN_KB_PAGE_1 is in front, with WIN_KB_PAGE_2 on deck
|
|
WIN_TEXT_ENTRY,
|
|
WIN_TEXT_ENTRY_BOX,
|
|
WIN_BANNER,
|
|
WIN_COUNT,
|
|
};
|
|
|
|
// The constants for the pages are needlessly complicated because GF didn't keep the indexing order consistent
|
|
// This set is used for sNamingScreen->currentPage. It uses the order that the pages are cycled in
|
|
enum {
|
|
KBPAGE_SYMBOLS,
|
|
KBPAGE_LETTERS_UPPER,
|
|
KBPAGE_LETTERS_LOWER,
|
|
KBPAGE_COUNT,
|
|
};
|
|
|
|
// This set is used for initializing a page's keyboard text and getting its number of columns
|
|
enum {
|
|
KEYBOARD_LETTERS_LOWER,
|
|
KEYBOARD_LETTERS_UPPER,
|
|
KEYBOARD_SYMBOLS,
|
|
};
|
|
|
|
// This set is used for getting the gfx/pal tags of the page's swap button
|
|
enum {
|
|
PAGE_SWAP_UPPER,
|
|
PAGE_SWAP_OTHERS,
|
|
PAGE_SWAP_LOWER,
|
|
};
|
|
|
|
enum {
|
|
KEY_ROLE_CHAR,
|
|
KEY_ROLE_PAGE,
|
|
KEY_ROLE_BACKSPACE,
|
|
KEY_ROLE_OK,
|
|
};
|
|
|
|
enum {
|
|
BUTTON_PAGE,
|
|
BUTTON_BACK,
|
|
BUTTON_OK,
|
|
BUTTON_COUNT,
|
|
};
|
|
|
|
// states for Task_NamingScreen
|
|
enum {
|
|
STATE_FADE_IN,
|
|
STATE_WAIT_FADE_IN,
|
|
STATE_HANDLE_INPUT,
|
|
STATE_MOVE_TO_OK_BUTTON,
|
|
STATE_START_PAGE_SWAP,
|
|
STATE_WAIT_PAGE_SWAP,
|
|
STATE_PRESSED_OK,
|
|
STATE_WAIT_SENT_TO_PC_MESSAGE,
|
|
STATE_FADE_OUT,
|
|
STATE_EXIT,
|
|
};
|
|
|
|
// sates for Task_HandleInput
|
|
enum
|
|
{
|
|
INPUT_STATE_DISABLED,
|
|
INPUT_STATE_ENABLED,
|
|
INPUT_STATE_OVERRIDE,
|
|
};
|
|
|
|
struct NamingScreenTemplate
|
|
{
|
|
u8 copyExistingString;
|
|
u8 maxChars;
|
|
u8 iconFunction;
|
|
u8 addGenderIcon;
|
|
u8 initialPage;
|
|
u8 unused;
|
|
const u8 *title;
|
|
};
|
|
|
|
struct NamingScreenData
|
|
{
|
|
u8 tilemapBuffer1[0x800];
|
|
u8 tilemapBuffer2[0x800];
|
|
u8 tilemapBuffer3[0x800];
|
|
u8 textBuffer[16];
|
|
u8 tileBuffer[0x600];
|
|
u8 state;
|
|
u8 windows[WIN_COUNT];
|
|
u16 inputCharBaseXPos;
|
|
u16 bg1vOffset;
|
|
u16 bg2vOffset;
|
|
u16 bg1Priority;
|
|
u16 bg2Priority;
|
|
u8 bgToReveal;
|
|
u8 bgToHide;
|
|
u8 currentPage;
|
|
u8 cursorSpriteId;
|
|
u8 swapBtnFrameSpriteId;
|
|
u8 keyRepeatStartDelayCopy;
|
|
const struct NamingScreenTemplate *template;
|
|
u8 templateNum;
|
|
u8 *destBuffer;
|
|
u16 monSpecies;
|
|
u16 monGender;
|
|
u32 monPersonality;
|
|
MainCallback returnCallback;
|
|
};
|
|
|
|
EWRAM_DATA static struct NamingScreenData *sNamingScreen = NULL;
|
|
extern u16 gKeyRepeatStartDelay;
|
|
|
|
// extern text
|
|
extern const u8 gText_MoveOkBack[];
|
|
extern const u8 gText_YourName[];
|
|
extern const u8 gText_BoxName[];
|
|
extern const u8 gText_PkmnsNickname[];
|
|
extern const u8 gText_TellHimTheWords[];
|
|
|
|
|
|
// start of .rodata
|
|
static const u8 sPCIconOff_Gfx[] = INCBIN_U8("graphics/naming_screen/pc_icon/off.4bpp");
|
|
static const u8 sPCIconOn_Gfx[] = INCBIN_U8("graphics/naming_screen/pc_icon/on.4bpp");
|
|
static const u16 sKeyboard_Pal[] = INCBIN_U16("graphics/naming_screen/keyboard.gbapal");
|
|
static const u16 sUnused_Pal[] = INCBIN_U16("graphics/naming_screen/unused.gbapal");
|
|
|
|
static const u8 *const sTransferredToPCMessages[] =
|
|
{
|
|
gText_PkmnTransferredSomeonesPC,
|
|
gText_PkmnTransferredLanettesPC,
|
|
gText_PkmnTransferredSomeonesPCBoxFull,
|
|
gText_PkmnTransferredLanettesPCBoxFull
|
|
};
|
|
|
|
static const u8 sText_AlphabetUpperLower[] = _("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz!");
|
|
|
|
static const struct BgTemplate sBgTemplates[] =
|
|
{
|
|
{
|
|
.bg = 0,
|
|
.charBaseIndex = 0,
|
|
.mapBaseIndex = 30,
|
|
.priority = 0
|
|
},
|
|
{
|
|
.bg = 1,
|
|
.charBaseIndex = 2,
|
|
.mapBaseIndex = 29,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.bg = 2,
|
|
.charBaseIndex = 2,
|
|
.mapBaseIndex = 28,
|
|
.priority = 2
|
|
},
|
|
{
|
|
.bg = 3,
|
|
.charBaseIndex = 3,
|
|
.mapBaseIndex = 31,
|
|
.priority = 3
|
|
}
|
|
};
|
|
|
|
static const struct WindowTemplate sWindowTemplates[WIN_COUNT + 1] =
|
|
{
|
|
[WIN_KB_PAGE_1] = {
|
|
.bg = 1,
|
|
.tilemapLeft = 3,
|
|
.tilemapTop = 10,
|
|
.width = 19,
|
|
.height = 8,
|
|
.paletteNum = 10,
|
|
.baseBlock = 0x030
|
|
},
|
|
[WIN_KB_PAGE_2] = {
|
|
.bg = 2,
|
|
.tilemapLeft = 3,
|
|
.tilemapTop = 10,
|
|
.width = 19,
|
|
.height = 8,
|
|
.paletteNum = 10,
|
|
.baseBlock = 0x0C8
|
|
},
|
|
[WIN_TEXT_ENTRY] = {
|
|
.bg = 3,
|
|
.tilemapLeft = 8,
|
|
.tilemapTop = 6,
|
|
.width = 17,
|
|
.height = 2,
|
|
.paletteNum = 10,
|
|
.baseBlock = 0x030
|
|
},
|
|
[WIN_TEXT_ENTRY_BOX] = {
|
|
.bg = 3,
|
|
.tilemapLeft = 8,
|
|
.tilemapTop = 4,
|
|
.width = 17,
|
|
.height = 2,
|
|
.paletteNum = 10,
|
|
.baseBlock = 0x052
|
|
},
|
|
[WIN_BANNER] = {
|
|
.bg = 0,
|
|
.tilemapLeft = 0,
|
|
.tilemapTop = 0,
|
|
.width = 30,
|
|
.height = 2,
|
|
.paletteNum = 11,
|
|
.baseBlock = 0x074
|
|
},
|
|
DUMMY_WIN_TEMPLATE
|
|
};
|
|
|
|
// This handles what characters get inserted when a key is pressed
|
|
// The keys shown on the keyboard are handled separately by sNamingScreenKeyboardText
|
|
static const u8 sKeyboardChars[KBPAGE_COUNT * KBROW_COUNT * KBCOL_COUNT] = __(
|
|
"abcdef ."
|
|
"ghijkl ,"
|
|
"mnopqrs "
|
|
"tuvwxyz "
|
|
"ABCDEF ."
|
|
"GHIJKL ,"
|
|
"MNOPQRS "
|
|
"TUVWXYZ "
|
|
"01234 "
|
|
"56789 "
|
|
"!?♂♀/- "
|
|
"…“”‘' ");
|
|
|
|
static const u8 sPageColumnCounts[KBPAGE_COUNT] = {
|
|
[KEYBOARD_LETTERS_LOWER] = KBCOL_COUNT,
|
|
[KEYBOARD_LETTERS_UPPER] = KBCOL_COUNT,
|
|
[KEYBOARD_SYMBOLS] = 6
|
|
};
|
|
static const u8 sPageColumnXPos[KBPAGE_COUNT * KBCOL_COUNT] = {
|
|
0, 12, 24, 56, 68, 80, 92, 123, // KEYBOARD_LETTERS_LOWER
|
|
0, 12, 24, 56, 68, 80, 92, 123, // KEYBOARD_LETTERS_UPPER
|
|
0, 22, 44, 66, 88, 110 // KEYBOARD_SYMBOLS
|
|
};
|
|
|
|
// forward declarations
|
|
static const struct NamingScreenTemplate *const sNamingScreenTemplates[];
|
|
static const struct SubspriteTable sSubspriteTable_PageSwapFrame[];
|
|
static const struct SubspriteTable sSubspriteTable_PageSwapText[];
|
|
static const struct SubspriteTable sSubspriteTable_Button[];
|
|
static const struct SubspriteTable sSubspriteTable_PCIcon[];
|
|
static const struct SpriteTemplate sSpriteTemplate_PageSwapFrame;
|
|
static const struct SpriteTemplate sSpriteTemplate_PageSwapButton;
|
|
static const struct SpriteTemplate sSpriteTemplate_PageSwapText;
|
|
static const struct SpriteTemplate sSpriteTemplate_BackButton;
|
|
static const struct SpriteTemplate sSpriteTemplate_OkButton;
|
|
static const struct SpriteTemplate sSpriteTemplate_Cursor;
|
|
static const struct SpriteTemplate sSpriteTemplate_InputArrow;
|
|
static const struct SpriteTemplate sSpriteTemplate_Underscore;
|
|
static const struct SpriteTemplate sSpriteTemplate_PCIcon;
|
|
static const u8* const sNamingScreenKeyboardText[KBPAGE_COUNT][KBROW_COUNT];
|
|
static const struct SpriteSheet sSpriteSheets[];
|
|
static const struct SpritePalette sSpritePalettes[];
|
|
|
|
static void CB2_LoadNamingScreen(void);
|
|
static void NamingScreen_Init(void);
|
|
static void NamingScreen_InitBGs(void);
|
|
static void CreateNamingScreenTask(void);
|
|
static void Task_NamingScreen(u8 taskId);
|
|
static bool8 MainState_FadeIn(void);
|
|
static bool8 MainState_WaitFadeIn(void);
|
|
static bool8 MainState_HandleInput(void);
|
|
static bool8 MainState_MoveToOKButton(void);
|
|
static bool8 MainState_PressedOKButton(void);
|
|
static bool8 MainState_FadeOut(void);
|
|
static bool8 MainState_Exit(void);
|
|
static void DisplaySentToPCMessage(void);
|
|
static bool8 MainState_WaitSentToPCMessage(void);
|
|
static bool8 MainState_StartPageSwap(void);
|
|
static bool8 MainState_WaitPageSwap(void);
|
|
static void StartPageSwapAnim(void);
|
|
static void Task_HandlePageSwapAnim(u8);
|
|
static bool8 IsPageSwapAnimNotInProgress(void);
|
|
static void TryStartButtonFlash(u8, bool8, bool8);
|
|
static void Task_UpdateButtonFlash(u8);
|
|
static u16 GetButtonPalOffset(u8);
|
|
static void RestoreButtonColor(u8);
|
|
static void StartButtonFlash(struct Task *, u8, bool8);
|
|
static void CreateSprites(void);
|
|
static void CreateCursorSprite(void);
|
|
static void SetCursorPos(s16, s16);
|
|
static void GetCursorPos(s16 *x, s16 *y);
|
|
static void MoveCursorToOKButton(void);
|
|
static void SetCursorInvisibility(u8);
|
|
static void SetCursorFlashing(bool8);
|
|
static u8 IsCursorAnimFinished(void);
|
|
static u8 GetCurrentPageColumnCount(void);
|
|
static void CreatePageSwapButtonSprites(void);
|
|
static void StartPageSwapButtonAnim(void);
|
|
static void SetPageSwapButtonGfx(u8, struct Sprite *, struct Sprite *);
|
|
static void CreateBackOkSprites(void);
|
|
static void CreateTextEntrySprites(void);
|
|
static void CreateInputTargetIcon(void);
|
|
static u8 HandleKeyboardEvent(void);
|
|
static u8 SwapKeyboardPage(void);
|
|
static u8 GetInputEvent(void);
|
|
static void SetInputState(u8);
|
|
static void DrawTextEntryBox(void);
|
|
static u8 GetTextEntryPosition(void);
|
|
static void DeleteTextCharacter(void);
|
|
static bool8 AddTextCharacter(void);
|
|
static void BufferCharacter(u8);
|
|
static void SaveInputText(void);
|
|
static void LoadGfx(void);
|
|
static void CreateHelperTasks(void);
|
|
static void LoadPalettes(void);
|
|
static void DrawBgTilemap(u8, const void *);
|
|
static void NamingScreen_Dummy(u8, u8);
|
|
static void DrawTextEntry(void);
|
|
static void PrintKeyboardKeys(u8, u8);
|
|
static void DrawKeyboardPageOnDeck(void);
|
|
static void PrintControls(void);
|
|
static void CB2_NamingScreen(void);
|
|
static void ResetVHBlank(void);
|
|
static void SetVBlank(void);
|
|
static void VBlankCB_NamingScreen(void);
|
|
static void NamingScreen_ShowBgs(void);
|
|
static bool8 IsWideLetter(u8);
|
|
|
|
void DoNamingScreen(u8 templateNum, u8 *destBuffer, u16 monSpecies, u16 monGender, u32 monPersonality, MainCallback returnCallback)
|
|
{
|
|
sNamingScreen = Alloc(sizeof(struct NamingScreenData));
|
|
if (!sNamingScreen)
|
|
{
|
|
SetMainCallback2(returnCallback);
|
|
}
|
|
else
|
|
{
|
|
sNamingScreen->templateNum = templateNum;
|
|
sNamingScreen->monSpecies = monSpecies;
|
|
sNamingScreen->monGender = monGender;
|
|
sNamingScreen->monPersonality = monPersonality;
|
|
sNamingScreen->destBuffer = destBuffer;
|
|
sNamingScreen->returnCallback = returnCallback;
|
|
|
|
if (templateNum == NAMING_SCREEN_PLAYER)
|
|
StartTimer1();
|
|
|
|
SetMainCallback2(CB2_LoadNamingScreen);
|
|
}
|
|
}
|
|
|
|
static void CB2_LoadNamingScreen(void)
|
|
{
|
|
switch (gMain.state)
|
|
{
|
|
case 0:
|
|
ResetVHBlank();
|
|
NamingScreen_Init();
|
|
gMain.state++;
|
|
break;
|
|
case 1:
|
|
NamingScreen_InitBGs();
|
|
gMain.state++;
|
|
break;
|
|
case 2:
|
|
ResetPaletteFade();
|
|
gMain.state++;
|
|
break;
|
|
case 3:
|
|
ResetSpriteData();
|
|
FreeAllSpritePalettes();
|
|
gMain.state++;
|
|
break;
|
|
case 4:
|
|
ResetTasks();
|
|
gMain.state++;
|
|
break;
|
|
case 5:
|
|
LoadPalettes();
|
|
gMain.state++;
|
|
break;
|
|
case 6:
|
|
LoadGfx();
|
|
gMain.state++;
|
|
break;
|
|
case 7:
|
|
CreateSprites();
|
|
UpdatePaletteFade();
|
|
NamingScreen_ShowBgs();
|
|
gMain.state++;
|
|
break;
|
|
default:
|
|
CreateHelperTasks();
|
|
CreateNamingScreenTask();
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void NamingScreen_Init(void)
|
|
{
|
|
sNamingScreen->state = STATE_FADE_IN;
|
|
sNamingScreen->bg1vOffset = 0;
|
|
sNamingScreen->bg2vOffset = 0;
|
|
sNamingScreen->bg1Priority = BGCNT_PRIORITY(1);
|
|
sNamingScreen->bg2Priority = BGCNT_PRIORITY(2);
|
|
sNamingScreen->bgToReveal = 0;
|
|
sNamingScreen->bgToHide = 1;
|
|
sNamingScreen->template = sNamingScreenTemplates[sNamingScreen->templateNum];
|
|
sNamingScreen->currentPage = sNamingScreen->template->initialPage;
|
|
sNamingScreen->inputCharBaseXPos = (240 - sNamingScreen->template->maxChars * 8) / 2 + 6;
|
|
if (sNamingScreen->templateNum == NAMING_SCREEN_WALDA)
|
|
sNamingScreen->inputCharBaseXPos += 11;
|
|
sNamingScreen->keyRepeatStartDelayCopy = gKeyRepeatStartDelay;
|
|
memset(sNamingScreen->textBuffer, EOS, sizeof(sNamingScreen->textBuffer));
|
|
if (sNamingScreen->template->copyExistingString)
|
|
StringCopy(sNamingScreen->textBuffer, sNamingScreen->destBuffer);
|
|
gKeyRepeatStartDelay = 16;
|
|
}
|
|
|
|
static void SetSpritesVisible(void)
|
|
{
|
|
u8 i;
|
|
for (i = 0; i < MAX_SPRITES; i++)
|
|
{
|
|
if (gSprites[i].inUse)
|
|
gSprites[i].invisible = FALSE;
|
|
}
|
|
SetCursorInvisibility(FALSE);
|
|
}
|
|
|
|
static void NamingScreen_InitBGs(void)
|
|
{
|
|
u8 i;
|
|
|
|
DmaClearLarge16(3, (void *)VRAM, VRAM_SIZE, 0x1000);
|
|
DmaClear32(3, (void *)OAM, OAM_SIZE);
|
|
DmaClear16(3, (void *)PLTT, PLTT_SIZE);
|
|
|
|
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0);
|
|
ResetBgsAndClearDma3BusyFlags(0);
|
|
InitBgsFromTemplates(0, sBgTemplates, ARRAY_COUNT(sBgTemplates));
|
|
|
|
ChangeBgX(0, 0, 0);
|
|
ChangeBgY(0, 0, 0);
|
|
ChangeBgX(1, 0, 0);
|
|
ChangeBgY(1, 0, 0);
|
|
ChangeBgX(2, 0, 0);
|
|
ChangeBgY(2, 0, 0);
|
|
ChangeBgX(3, 0, 0);
|
|
ChangeBgY(3, 0, 0);
|
|
|
|
InitStandardTextBoxWindows();
|
|
InitTextBoxGfxAndPrinters();
|
|
|
|
for (i = 0; i < WIN_COUNT; i++)
|
|
sNamingScreen->windows[i] = AddWindow(&sWindowTemplates[i]);
|
|
|
|
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_OBJ_1D_MAP | DISPCNT_OBJ_ON);
|
|
SetGpuReg(REG_OFFSET_BLDCNT, BLDCNT_EFFECT_BLEND | BLDCNT_TGT2_BG1 | BLDCNT_TGT2_BG2);
|
|
SetGpuReg(REG_OFFSET_BLDALPHA, BLDALPHA_BLEND(12, 8));
|
|
|
|
SetBgTilemapBuffer(1, sNamingScreen->tilemapBuffer1);
|
|
SetBgTilemapBuffer(2, sNamingScreen->tilemapBuffer2);
|
|
SetBgTilemapBuffer(3, sNamingScreen->tilemapBuffer3);
|
|
|
|
FillBgTilemapBufferRect_Palette0(1, 0, 0, 0, 0x20, 0x20);
|
|
FillBgTilemapBufferRect_Palette0(2, 0, 0, 0, 0x20, 0x20);
|
|
FillBgTilemapBufferRect_Palette0(3, 0, 0, 0, 0x20, 0x20);
|
|
}
|
|
|
|
static void CreateNamingScreenTask(void)
|
|
{
|
|
CreateTask(Task_NamingScreen, 2);
|
|
SetMainCallback2(CB2_NamingScreen);
|
|
}
|
|
|
|
static void Task_NamingScreen(u8 taskId)
|
|
{
|
|
switch (sNamingScreen->state)
|
|
{
|
|
case STATE_FADE_IN:
|
|
MainState_FadeIn();
|
|
SetSpritesVisible();
|
|
SetVBlank();
|
|
break;
|
|
case STATE_WAIT_FADE_IN:
|
|
MainState_WaitFadeIn();
|
|
break;
|
|
case STATE_HANDLE_INPUT:
|
|
MainState_HandleInput();
|
|
break;
|
|
case STATE_MOVE_TO_OK_BUTTON:
|
|
MainState_MoveToOKButton();
|
|
MainState_HandleInput();
|
|
break;
|
|
case STATE_START_PAGE_SWAP:
|
|
MainState_StartPageSwap();
|
|
break;
|
|
case STATE_WAIT_PAGE_SWAP:
|
|
MainState_WaitPageSwap();
|
|
break;
|
|
case STATE_PRESSED_OK:
|
|
MainState_PressedOKButton();
|
|
break;
|
|
case STATE_WAIT_SENT_TO_PC_MESSAGE:
|
|
MainState_WaitSentToPCMessage();
|
|
break;
|
|
case STATE_FADE_OUT:
|
|
MainState_FadeOut();
|
|
break;
|
|
case STATE_EXIT:
|
|
MainState_Exit();
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Which gfx/pal to load for the swap page button
|
|
static const u8 sPageToNextGfxId[KBPAGE_COUNT] =
|
|
{
|
|
[KBPAGE_SYMBOLS] = PAGE_SWAP_UPPER,
|
|
[KBPAGE_LETTERS_UPPER] = PAGE_SWAP_LOWER,
|
|
[KBPAGE_LETTERS_LOWER] = PAGE_SWAP_OTHERS
|
|
};
|
|
|
|
static const u8 sPageToNextKeyboardId[KBPAGE_COUNT] =
|
|
{
|
|
[KBPAGE_SYMBOLS] = KEYBOARD_LETTERS_UPPER,
|
|
[KBPAGE_LETTERS_UPPER] = KEYBOARD_LETTERS_LOWER,
|
|
[KBPAGE_LETTERS_LOWER] = KEYBOARD_SYMBOLS
|
|
};
|
|
|
|
static const u8 sPageToKeyboardId[KBPAGE_COUNT] =
|
|
{
|
|
[KBPAGE_SYMBOLS] = KEYBOARD_SYMBOLS,
|
|
[KBPAGE_LETTERS_UPPER] = KEYBOARD_LETTERS_UPPER,
|
|
[KBPAGE_LETTERS_LOWER] = KEYBOARD_LETTERS_LOWER
|
|
};
|
|
|
|
static u8 PageToNextGfxId(u8 page)
|
|
{
|
|
return sPageToNextGfxId[page];
|
|
}
|
|
|
|
static u8 CurrentPageToNextKeyboardId(void)
|
|
{
|
|
return sPageToNextKeyboardId[sNamingScreen->currentPage];
|
|
}
|
|
|
|
static u8 CurrentPageToKeyboardId(void)
|
|
{
|
|
return sPageToKeyboardId[sNamingScreen->currentPage];
|
|
}
|
|
|
|
static bool8 MainState_FadeIn(void)
|
|
{
|
|
DrawBgTilemap(3, gNamingScreenBackground_Tilemap);
|
|
sNamingScreen->currentPage = KBPAGE_LETTERS_UPPER;
|
|
DrawBgTilemap(2, gNamingScreenKeyboardLower_Tilemap);
|
|
DrawBgTilemap(1, gNamingScreenKeyboardUpper_Tilemap);
|
|
PrintKeyboardKeys(sNamingScreen->windows[WIN_KB_PAGE_2], KEYBOARD_LETTERS_LOWER);
|
|
PrintKeyboardKeys(sNamingScreen->windows[WIN_KB_PAGE_1], KEYBOARD_LETTERS_UPPER);
|
|
NamingScreen_Dummy(2, KEYBOARD_LETTERS_LOWER);
|
|
NamingScreen_Dummy(1, KEYBOARD_LETTERS_UPPER);
|
|
DrawTextEntry();
|
|
DrawTextEntryBox();
|
|
PrintControls();
|
|
CopyBgTilemapBufferToVram(1);
|
|
CopyBgTilemapBufferToVram(2);
|
|
CopyBgTilemapBufferToVram(3);
|
|
BlendPalettes(-1, 16, 0);
|
|
BeginNormalPaletteFade(0xFFFFFFFF, 0, 16, 0, RGB_BLACK);
|
|
sNamingScreen->state++;
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 MainState_WaitFadeIn(void)
|
|
{
|
|
if (!gPaletteFade.active)
|
|
{
|
|
SetInputState(INPUT_STATE_ENABLED);
|
|
SetCursorFlashing(TRUE);
|
|
sNamingScreen->state++;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 MainState_HandleInput(void)
|
|
{
|
|
return HandleKeyboardEvent();
|
|
}
|
|
|
|
static bool8 MainState_MoveToOKButton(void)
|
|
{
|
|
if (IsCursorAnimFinished())
|
|
{
|
|
SetInputState(INPUT_STATE_ENABLED);
|
|
MoveCursorToOKButton();
|
|
sNamingScreen->state = STATE_HANDLE_INPUT;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 MainState_PressedOKButton(void)
|
|
{
|
|
SaveInputText();
|
|
SetInputState(INPUT_STATE_DISABLED);
|
|
SetCursorFlashing(FALSE);
|
|
TryStartButtonFlash(BUTTON_COUNT, FALSE, TRUE);
|
|
if (sNamingScreen->templateNum == NAMING_SCREEN_CAUGHT_MON
|
|
&& CalculatePlayerPartyCount() >= PARTY_SIZE)
|
|
{
|
|
DisplaySentToPCMessage();
|
|
sNamingScreen->state = STATE_WAIT_SENT_TO_PC_MESSAGE;
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
sNamingScreen->state = STATE_FADE_OUT;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
static bool8 MainState_FadeOut(void)
|
|
{
|
|
BeginNormalPaletteFade(0xFFFFFFFF, 0, 0, 16, RGB_BLACK);
|
|
sNamingScreen->state++;
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 MainState_Exit(void)
|
|
{
|
|
if (!gPaletteFade.active)
|
|
{
|
|
if (sNamingScreen->templateNum == NAMING_SCREEN_PLAYER)
|
|
SeedRngAndSetTrainerId();
|
|
SetMainCallback2(sNamingScreen->returnCallback);
|
|
DestroyTask(FindTaskIdByFunc(Task_NamingScreen));
|
|
FreeAllWindowBuffers();
|
|
FREE_AND_SET_NULL(sNamingScreen);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static void DisplaySentToPCMessage(void)
|
|
{
|
|
u8 stringToDisplay = 0;
|
|
|
|
if (!IsDestinationBoxFull())
|
|
{
|
|
StringCopy(gStringVar1, GetBoxNamePtr(VarGet(VAR_PC_BOX_TO_SEND_MON)));
|
|
StringCopy(gStringVar2, sNamingScreen->destBuffer);
|
|
}
|
|
else
|
|
{
|
|
StringCopy(gStringVar1, GetBoxNamePtr(VarGet(VAR_PC_BOX_TO_SEND_MON)));
|
|
StringCopy(gStringVar2, sNamingScreen->destBuffer);
|
|
StringCopy(gStringVar3, GetBoxNamePtr(GetPCBoxToSendMon()));
|
|
stringToDisplay = 2;
|
|
}
|
|
|
|
if (FlagGet(FLAG_SYS_PC_LANETTE))
|
|
stringToDisplay++;
|
|
|
|
StringExpandPlaceholders(gStringVar4, sTransferredToPCMessages[stringToDisplay]);
|
|
DrawDialogueFrame(0, 0);
|
|
gTextFlags.canABSpeedUpPrint = TRUE;
|
|
AddTextPrinterParameterized2(0, 1, gStringVar4, GetPlayerTextSpeedDelay(), 0, 2, 1, 3);
|
|
CopyWindowToVram(0, 3);
|
|
}
|
|
|
|
static bool8 MainState_WaitSentToPCMessage(void)
|
|
{
|
|
RunTextPrinters();
|
|
if (!IsTextPrinterActive(0) && JOY_NEW(A_BUTTON))
|
|
sNamingScreen->state = STATE_FADE_OUT;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 MainState_StartPageSwap(void)
|
|
{
|
|
SetInputState(INPUT_STATE_DISABLED);
|
|
StartPageSwapButtonAnim();
|
|
StartPageSwapAnim();
|
|
SetCursorInvisibility(TRUE);
|
|
TryStartButtonFlash(BUTTON_PAGE, FALSE, TRUE);
|
|
PlaySE(SE_WIN_OPEN);
|
|
sNamingScreen->state = STATE_WAIT_PAGE_SWAP;
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 MainState_WaitPageSwap(void)
|
|
{
|
|
s16 cursorX;
|
|
s16 cursorY;
|
|
bool32 onLastColumn;
|
|
|
|
if (IsPageSwapAnimNotInProgress())
|
|
{
|
|
|
|
GetCursorPos(&cursorX, &cursorY);
|
|
onLastColumn = (cursorX == GetCurrentPageColumnCount());
|
|
|
|
sNamingScreen->state = STATE_HANDLE_INPUT;
|
|
sNamingScreen->currentPage++;
|
|
sNamingScreen->currentPage %= KBPAGE_COUNT;
|
|
|
|
if (onLastColumn)
|
|
{
|
|
cursorX = GetCurrentPageColumnCount();
|
|
}
|
|
else
|
|
{
|
|
if (cursorX >= GetCurrentPageColumnCount())
|
|
cursorX = GetCurrentPageColumnCount() - 1;
|
|
}
|
|
|
|
SetCursorPos(cursorX, cursorY);
|
|
DrawKeyboardPageOnDeck();
|
|
SetInputState(INPUT_STATE_ENABLED);
|
|
SetCursorInvisibility(FALSE);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
//--------------------------------------------------
|
|
// Page Swap
|
|
//--------------------------------------------------
|
|
|
|
#define tState data[0]
|
|
#define tFrameCount data[1]
|
|
|
|
static bool8 PageSwapAnimState_Init(struct Task *);
|
|
static bool8 PageSwapAnimState_1(struct Task *);
|
|
static bool8 PageSwapAnimState_2(struct Task *);
|
|
static bool8 PageSwapAnimState_Done(struct Task *);
|
|
|
|
static bool8 (*const sPageSwapAnimStateFuncs[])(struct Task *) =
|
|
{
|
|
PageSwapAnimState_Init,
|
|
PageSwapAnimState_1,
|
|
PageSwapAnimState_2,
|
|
PageSwapAnimState_Done,
|
|
};
|
|
|
|
static void StartPageSwapAnim(void)
|
|
{
|
|
u8 taskId;
|
|
|
|
taskId = CreateTask(Task_HandlePageSwapAnim, 0);
|
|
Task_HandlePageSwapAnim(taskId);
|
|
}
|
|
|
|
static void Task_HandlePageSwapAnim(u8 taskId)
|
|
{
|
|
while (sPageSwapAnimStateFuncs[gTasks[taskId].tState](&gTasks[taskId]) != 0);
|
|
}
|
|
|
|
static bool8 IsPageSwapAnimNotInProgress(void)
|
|
{
|
|
if (FindTaskIdByFunc(Task_HandlePageSwapAnim) == 0xFF)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 PageSwapAnimState_Init(struct Task *task)
|
|
{
|
|
sNamingScreen->bg1vOffset = 0;
|
|
sNamingScreen->bg2vOffset = 0;
|
|
task->tState++;
|
|
return 0;
|
|
}
|
|
|
|
static bool8 PageSwapAnimState_1(struct Task *task)
|
|
{
|
|
u16 *const vOffsets[] =
|
|
{
|
|
&sNamingScreen->bg2vOffset,
|
|
&sNamingScreen->bg1vOffset
|
|
};
|
|
|
|
task->tFrameCount += 4;
|
|
*vOffsets[sNamingScreen->bgToReveal] = Sin(task->tFrameCount, 40);
|
|
*vOffsets[sNamingScreen->bgToHide] = Sin((task->tFrameCount + 128) & 0xFF, 40);
|
|
if (task->tFrameCount >= 64)
|
|
{
|
|
u8 temp = sNamingScreen->bg1Priority; //Why u8 and not u16?
|
|
|
|
sNamingScreen->bg1Priority = sNamingScreen->bg2Priority;
|
|
sNamingScreen->bg2Priority = temp;
|
|
task->tState++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static bool8 PageSwapAnimState_2(struct Task *task)
|
|
{
|
|
u16 *const vOffsets[] =
|
|
{
|
|
&sNamingScreen->bg2vOffset,
|
|
&sNamingScreen->bg1vOffset
|
|
};
|
|
|
|
task->tFrameCount += 4;
|
|
*vOffsets[sNamingScreen->bgToReveal] = Sin(task->tFrameCount, 40);
|
|
*vOffsets[sNamingScreen->bgToHide] = Sin((task->tFrameCount + 128) & 0xFF, 40);
|
|
if (task->tFrameCount >= 128)
|
|
{
|
|
u8 temp = sNamingScreen->bgToReveal;
|
|
|
|
sNamingScreen->bgToReveal = sNamingScreen->bgToHide;
|
|
sNamingScreen->bgToHide = temp;
|
|
task->tState++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static bool8 PageSwapAnimState_Done(struct Task *task)
|
|
{
|
|
DestroyTask(FindTaskIdByFunc(Task_HandlePageSwapAnim));
|
|
return 0;
|
|
}
|
|
|
|
#undef tState
|
|
#undef tFrameCount
|
|
|
|
//--------------------------------------------------
|
|
//
|
|
//--------------------------------------------------
|
|
|
|
#define tButtonId data[0]
|
|
#define tKeepFlashing data[1]
|
|
#define tAllowFlash data[2]
|
|
#define tColor data[3]
|
|
#define tColorIncr data[4]
|
|
#define tColorDelay data[5]
|
|
#define tColorDelta data[6]
|
|
|
|
static void CreateButtonFlashTask(void)
|
|
{
|
|
u8 taskId;
|
|
|
|
taskId = CreateTask(Task_UpdateButtonFlash, 3);
|
|
gTasks[taskId].tButtonId = BUTTON_COUNT;
|
|
}
|
|
|
|
static void TryStartButtonFlash(u8 button, bool8 keepFlashing, bool8 interruptCurFlash)
|
|
{
|
|
struct Task *task = &gTasks[FindTaskIdByFunc(Task_UpdateButtonFlash)];
|
|
|
|
if (button == task->tButtonId && !interruptCurFlash)
|
|
{
|
|
task->tKeepFlashing = keepFlashing;
|
|
task->tAllowFlash = TRUE;
|
|
return;
|
|
}
|
|
if (button == BUTTON_COUNT && !task->tKeepFlashing && !interruptCurFlash)
|
|
return;
|
|
|
|
if (task->tButtonId != BUTTON_COUNT)
|
|
RestoreButtonColor(task->tButtonId);
|
|
|
|
StartButtonFlash(task, button, keepFlashing);
|
|
}
|
|
|
|
static void Task_UpdateButtonFlash(u8 taskId)
|
|
{
|
|
struct Task *task = &gTasks[taskId];
|
|
|
|
if (task->tButtonId == BUTTON_COUNT || !task->tAllowFlash)
|
|
return;
|
|
|
|
MultiplyInvertedPaletteRGBComponents(GetButtonPalOffset(task->tButtonId), task->tColor, task->tColor, task->tColor);
|
|
|
|
if (task->tColorDelay && --task->tColorDelay)
|
|
return;
|
|
|
|
task->tColorDelay = 2;
|
|
if (task->tColorIncr >= 0)
|
|
{
|
|
if (task->tColor < 14)
|
|
{
|
|
task->tColor += task->tColorIncr;
|
|
task->tColorDelta += task->tColorIncr;
|
|
}
|
|
else
|
|
{
|
|
task->tColor = 16;
|
|
task->tColorDelta++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
task->tColor += task->tColorIncr;
|
|
task->tColorDelta += task->tColorIncr;
|
|
}
|
|
|
|
if (task->tColor == 16 && task->tColorDelta == 22)
|
|
{
|
|
task->tColorIncr = -4;
|
|
}
|
|
else if (task->tColor == 0)
|
|
{
|
|
task->tAllowFlash = task->tKeepFlashing;
|
|
task->tColorIncr = 2;
|
|
task->tColorDelta = 0;
|
|
}
|
|
}
|
|
|
|
static u16 GetButtonPalOffset(u8 button)
|
|
{
|
|
const u16 palOffsets[BUTTON_COUNT + 1] =
|
|
{
|
|
[BUTTON_PAGE] = IndexOfSpritePaletteTag(PALTAG_PAGE_SWAP) * 16 + 0x10E,
|
|
[BUTTON_BACK] = IndexOfSpritePaletteTag(PALTAG_BACK_BUTTON) * 16 + 0x10E,
|
|
[BUTTON_OK] = IndexOfSpritePaletteTag(PALTAG_OK_BUTTON) * 16 + 0x10E,
|
|
[BUTTON_COUNT] = IndexOfSpritePaletteTag(PALTAG_OK_BUTTON) * 16 + 0x101,
|
|
};
|
|
|
|
return palOffsets[button];
|
|
}
|
|
|
|
static void RestoreButtonColor(u8 button)
|
|
{
|
|
u16 index = GetButtonPalOffset(button);
|
|
gPlttBufferFaded[index] = gPlttBufferUnfaded[index];
|
|
}
|
|
|
|
static void StartButtonFlash(struct Task *task, u8 button, bool8 keepFlashing)
|
|
{
|
|
task->tButtonId = button;
|
|
task->tKeepFlashing = keepFlashing;
|
|
task->tAllowFlash = TRUE;
|
|
task->tColor = 4;
|
|
task->tColorIncr = 2;
|
|
task->tColorDelay = 0;
|
|
task->tColorDelta = 4;
|
|
}
|
|
|
|
#undef tButtonId
|
|
#undef tColor
|
|
|
|
// Sprite data for the the cursor
|
|
#define sX data[0]
|
|
#define sY data[1]
|
|
#define sPrevX data[2]
|
|
#define sPrevY data[3]
|
|
#define sInvisible data[4] & 0x00FF
|
|
#define sFlashing data[4] & 0xFF00
|
|
#define sColor data[5]
|
|
#define sColorIncr data[6]
|
|
#define sColorDelay data[7]
|
|
|
|
static void SpriteCB_Cursor(struct Sprite *sprite)
|
|
{
|
|
if (sprite->animEnded)
|
|
StartSpriteAnim(sprite, 0);
|
|
|
|
// Hide cursor when on button column
|
|
sprite->invisible = sprite->sInvisible;
|
|
if (sprite->sX == GetCurrentPageColumnCount())
|
|
sprite->invisible = TRUE;
|
|
|
|
if (sprite->invisible
|
|
|| !(sprite->sFlashing)
|
|
|| sprite->sX != sprite->sPrevX
|
|
|| sprite->sY != sprite->sPrevY)
|
|
{
|
|
sprite->sColor = 0;
|
|
sprite->sColorIncr = 2;
|
|
sprite->sColorDelay = 2;
|
|
}
|
|
|
|
sprite->sColorDelay--;
|
|
if (sprite->sColorDelay == 0)
|
|
{
|
|
sprite->sColor += sprite->sColorIncr;
|
|
if (sprite->sColor == 16 || sprite->sColor == 0)
|
|
sprite->sColorIncr = -sprite->sColorIncr;
|
|
sprite->sColorDelay = 2;
|
|
}
|
|
|
|
if (sprite->sFlashing)
|
|
{
|
|
s8 gb = sprite->sColor;
|
|
s8 r = sprite->sColor >> 1;
|
|
u16 index = IndexOfSpritePaletteTag(PALTAG_CURSOR) * 16 + 0x0101;
|
|
|
|
MultiplyInvertedPaletteRGBComponents(index, r, gb, gb);
|
|
}
|
|
}
|
|
|
|
#define sDelay data[0]
|
|
#define sXPosId data[1]
|
|
|
|
static void SpriteCB_InputArrow(struct Sprite *sprite)
|
|
{
|
|
const s16 x[] = {0, -4, -2, -1};
|
|
|
|
if (sprite->sDelay == 0 || --sprite->sDelay == 0)
|
|
{
|
|
sprite->sDelay = 8;
|
|
sprite->sXPosId = (sprite->sXPosId + 1) & (ARRAY_COUNT(x) - 1);
|
|
}
|
|
sprite->pos2.x = x[sprite->sXPosId];
|
|
}
|
|
|
|
#undef sDelay
|
|
#undef sXPosId
|
|
|
|
#define sId data[0] // set in CreateTextEntrySprites
|
|
#define sYPosId data[1]
|
|
#define sDelay data[2]
|
|
|
|
static void SpriteCB_Underscore(struct Sprite *sprite)
|
|
{
|
|
const s16 y[] = {2, 3, 2, 1};
|
|
u8 pos;
|
|
|
|
pos = GetTextEntryPosition();
|
|
if (pos != (u8)sprite->sId)
|
|
{
|
|
sprite->pos2.y = 0;
|
|
sprite->sYPosId = 0;
|
|
sprite->sDelay = 0;
|
|
}
|
|
else
|
|
{
|
|
sprite->pos2.y = y[sprite->sYPosId];
|
|
sprite->sDelay++;
|
|
if (sprite->sDelay > 8)
|
|
{
|
|
sprite->sYPosId = (sprite->sYPosId + 1) & (ARRAY_COUNT(y) - 1);
|
|
sprite->sDelay = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
#undef sId
|
|
#undef sYPosId
|
|
#undef sDelay
|
|
|
|
static void CreateSprites(void)
|
|
{
|
|
CreateCursorSprite();
|
|
CreatePageSwapButtonSprites();
|
|
CreateBackOkSprites();
|
|
CreateTextEntrySprites();
|
|
CreateInputTargetIcon();
|
|
}
|
|
|
|
static void CreateCursorSprite(void)
|
|
{
|
|
sNamingScreen->cursorSpriteId = CreateSprite(&sSpriteTemplate_Cursor, 38, 88, 1);
|
|
SetCursorInvisibility(TRUE);
|
|
gSprites[sNamingScreen->cursorSpriteId].oam.priority = 1;
|
|
gSprites[sNamingScreen->cursorSpriteId].oam.objMode = ST_OAM_OBJ_BLEND;
|
|
gSprites[sNamingScreen->cursorSpriteId].sColorIncr = 1; // ? immediately overwritten
|
|
gSprites[sNamingScreen->cursorSpriteId].sColorIncr = 2;
|
|
SetCursorPos(0, 0);
|
|
}
|
|
|
|
static void SetCursorPos(s16 x, s16 y)
|
|
{
|
|
struct Sprite *cursorSprite = &gSprites[sNamingScreen->cursorSpriteId];
|
|
|
|
if (x < sPageColumnCounts[CurrentPageToKeyboardId()])
|
|
cursorSprite->pos1.x = sPageColumnXPos[x + CurrentPageToKeyboardId() * KBCOL_COUNT] + 38;
|
|
else
|
|
cursorSprite->pos1.x = 0;
|
|
|
|
cursorSprite->pos1.y = y * 16 + 88;
|
|
cursorSprite->sPrevX = cursorSprite->sX;
|
|
cursorSprite->sPrevY = cursorSprite->sY;
|
|
cursorSprite->sX = x;
|
|
cursorSprite->sY = y;
|
|
}
|
|
|
|
static void GetCursorPos(s16 *x, s16 *y)
|
|
{
|
|
struct Sprite *cursorSprite = &gSprites[sNamingScreen->cursorSpriteId];
|
|
|
|
*x = cursorSprite->sX;
|
|
*y = cursorSprite->sY;
|
|
}
|
|
|
|
static void MoveCursorToOKButton(void)
|
|
{
|
|
SetCursorPos(GetCurrentPageColumnCount(), 2);
|
|
}
|
|
|
|
static void SetCursorInvisibility(bool8 invisible)
|
|
{
|
|
gSprites[sNamingScreen->cursorSpriteId].data[4] &= 0xFF00;
|
|
gSprites[sNamingScreen->cursorSpriteId].data[4] |= invisible; // sInvisible
|
|
StartSpriteAnim(&gSprites[sNamingScreen->cursorSpriteId], 0);
|
|
}
|
|
|
|
static void SetCursorFlashing(bool8 flashing)
|
|
{
|
|
gSprites[sNamingScreen->cursorSpriteId].data[4] &= 0xFF;
|
|
gSprites[sNamingScreen->cursorSpriteId].data[4] |= flashing << 8; // sFlashing
|
|
}
|
|
|
|
static void SquishCursor(void)
|
|
{
|
|
StartSpriteAnim(&gSprites[sNamingScreen->cursorSpriteId], 1);
|
|
}
|
|
|
|
static bool8 IsCursorAnimFinished(void)
|
|
{
|
|
return gSprites[sNamingScreen->cursorSpriteId].animEnded;
|
|
}
|
|
|
|
static const u8 sButtonKeyRoles[] = {KEY_ROLE_PAGE, KEY_ROLE_BACKSPACE, KEY_ROLE_OK};
|
|
|
|
static u8 GetKeyRoleAtCursorPos(void)
|
|
{
|
|
s16 cursorX;
|
|
s16 cursorY;
|
|
|
|
GetCursorPos(&cursorX, &cursorY);
|
|
if (cursorX < GetCurrentPageColumnCount())
|
|
return KEY_ROLE_CHAR;
|
|
else
|
|
return sButtonKeyRoles[cursorY];
|
|
}
|
|
|
|
// If the cursor's x is equal to the column count, cursor is in the button column
|
|
static u8 GetCurrentPageColumnCount(void)
|
|
{
|
|
return sPageColumnCounts[CurrentPageToKeyboardId()];
|
|
}
|
|
|
|
#undef sX
|
|
#undef sY
|
|
#undef sPrevX
|
|
#undef sPrevY
|
|
#undef sInvisible
|
|
#undef sFlashing
|
|
#undef sColor
|
|
#undef sColorIncr
|
|
#undef sColorDelay
|
|
|
|
static bool8 PageSwapSprite_Init(struct Sprite *);
|
|
static bool8 PageSwapSprite_Idle(struct Sprite *);
|
|
static bool8 PageSwapSprite_SlideOff(struct Sprite *);
|
|
static bool8 PageSwapSprite_SlideOn(struct Sprite *);
|
|
|
|
#define sState data[0]
|
|
#define sPage data[1]
|
|
#define sTextSpriteId data[6]
|
|
#define sButtonSpriteId data[7]
|
|
|
|
static void CreatePageSwapButtonSprites(void)
|
|
{
|
|
u8 frameSpriteId;
|
|
u8 textSpriteId;
|
|
u8 buttonSpriteId;
|
|
|
|
frameSpriteId = CreateSprite(&sSpriteTemplate_PageSwapFrame, 204, 88, 0);
|
|
sNamingScreen->swapBtnFrameSpriteId = frameSpriteId;
|
|
SetSubspriteTables(&gSprites[frameSpriteId], sSubspriteTable_PageSwapFrame);
|
|
gSprites[frameSpriteId].invisible = TRUE;
|
|
|
|
textSpriteId = CreateSprite(&sSpriteTemplate_PageSwapText, 204, 84, 1);
|
|
gSprites[frameSpriteId].sTextSpriteId = textSpriteId;
|
|
SetSubspriteTables(&gSprites[textSpriteId], sSubspriteTable_PageSwapText);
|
|
gSprites[textSpriteId].invisible = TRUE;
|
|
|
|
buttonSpriteId = CreateSprite(&sSpriteTemplate_PageSwapButton, 204, 83, 2);
|
|
gSprites[buttonSpriteId].oam.priority = 1;
|
|
gSprites[frameSpriteId].sButtonSpriteId = buttonSpriteId;
|
|
gSprites[buttonSpriteId].invisible = TRUE;
|
|
}
|
|
|
|
static void StartPageSwapButtonAnim(void)
|
|
{
|
|
struct Sprite *sprite = &gSprites[sNamingScreen->swapBtnFrameSpriteId];
|
|
|
|
sprite->sState = 2; // go to PageSwapSprite_SlideOff
|
|
sprite->sPage = sNamingScreen->currentPage;
|
|
}
|
|
|
|
static u8 (*const sPageSwapSpriteFuncs[])(struct Sprite *) =
|
|
{
|
|
PageSwapSprite_Init,
|
|
PageSwapSprite_Idle,
|
|
PageSwapSprite_SlideOff,
|
|
PageSwapSprite_SlideOn,
|
|
};
|
|
|
|
static void SpriteCB_PageSwap(struct Sprite *sprite)
|
|
{
|
|
while (sPageSwapSpriteFuncs[sprite->sState](sprite));
|
|
}
|
|
|
|
static bool8 PageSwapSprite_Init(struct Sprite *sprite)
|
|
{
|
|
struct Sprite *text = &gSprites[sprite->sTextSpriteId];
|
|
struct Sprite *button = &gSprites[sprite->sButtonSpriteId];
|
|
|
|
SetPageSwapButtonGfx(PageToNextGfxId(sNamingScreen->currentPage), text, button);
|
|
sprite->sState++;
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 PageSwapSprite_Idle(struct Sprite *sprite)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 PageSwapSprite_SlideOff(struct Sprite *sprite)
|
|
{
|
|
struct Sprite *text = &gSprites[sprite->sTextSpriteId];
|
|
struct Sprite *button = &gSprites[sprite->sButtonSpriteId];
|
|
|
|
text->pos2.y++;
|
|
if (text->pos2.y > 7)
|
|
{
|
|
sprite->sState++;
|
|
text->pos2.y = -4;
|
|
text->invisible = TRUE;
|
|
SetPageSwapButtonGfx(PageToNextGfxId(((u8)sprite->sPage + 1) % KBPAGE_COUNT), text, button);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 PageSwapSprite_SlideOn(struct Sprite *sprite)
|
|
{
|
|
struct Sprite *text = &gSprites[sprite->sTextSpriteId];
|
|
|
|
text->invisible = FALSE;
|
|
text->pos2.y++;
|
|
if (text->pos2.y >= 0)
|
|
{
|
|
text->pos2.y = 0;
|
|
sprite->sState = 1; // go to PageSwapSprite_Idle
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static const u16 sPageSwapPalTags[] = {
|
|
[PAGE_SWAP_UPPER] = PALTAG_PAGE_SWAP_UPPER,
|
|
[PAGE_SWAP_OTHERS] = PALTAG_PAGE_SWAP_OTHERS,
|
|
[PAGE_SWAP_LOWER] = PALTAG_PAGE_SWAP_LOWER
|
|
};
|
|
|
|
static const u16 sPageSwapGfxTags[] = {
|
|
[PAGE_SWAP_UPPER] = GFXTAG_PAGE_SWAP_UPPER,
|
|
[PAGE_SWAP_OTHERS] = GFXTAG_PAGE_SWAP_OTHERS,
|
|
[PAGE_SWAP_LOWER] = GFXTAG_PAGE_SWAP_LOWER
|
|
};
|
|
|
|
static void SetPageSwapButtonGfx(u8 page, struct Sprite *text, struct Sprite *button)
|
|
{
|
|
button->oam.paletteNum = IndexOfSpritePaletteTag(sPageSwapPalTags[page]);
|
|
text->sheetTileStart = GetSpriteTileStartByTag(sPageSwapGfxTags[page]);
|
|
text->subspriteTableNum = page;
|
|
}
|
|
|
|
#undef sState
|
|
#undef sPage
|
|
#undef sTextSpriteId
|
|
#undef sButtonSpriteId
|
|
|
|
static void CreateBackOkSprites(void)
|
|
{
|
|
u8 spriteId;
|
|
|
|
spriteId = CreateSprite(&sSpriteTemplate_BackButton, 204, 116, 0);
|
|
SetSubspriteTables(&gSprites[spriteId], sSubspriteTable_Button);
|
|
gSprites[spriteId].invisible = TRUE;
|
|
|
|
spriteId = CreateSprite(&sSpriteTemplate_OkButton, 204, 140, 0);
|
|
SetSubspriteTables(&gSprites[spriteId], sSubspriteTable_Button);
|
|
gSprites[spriteId].invisible = TRUE;
|
|
}
|
|
|
|
static void CreateTextEntrySprites(void)
|
|
{
|
|
u8 spriteId;
|
|
s16 xPos;
|
|
u8 i;
|
|
|
|
xPos = sNamingScreen->inputCharBaseXPos - 5;
|
|
spriteId = CreateSprite(&sSpriteTemplate_InputArrow, xPos, 56, 0);
|
|
gSprites[spriteId].oam.priority = 3;
|
|
gSprites[spriteId].invisible = TRUE;
|
|
xPos = sNamingScreen->inputCharBaseXPos;
|
|
for (i = 0; i < sNamingScreen->template->maxChars; i++, xPos += 8)
|
|
{
|
|
spriteId = CreateSprite(&sSpriteTemplate_Underscore, xPos + 3, 60, 0);
|
|
gSprites[spriteId].oam.priority = 3;
|
|
gSprites[spriteId].data[0] = i;
|
|
gSprites[spriteId].invisible = TRUE;
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------
|
|
// Icon creation (the thing you're naming or giving input to)
|
|
//--------------------------------------------------
|
|
|
|
static void NamingScreen_NoIcon(void);
|
|
static void NamingScreen_CreatePlayerIcon(void);
|
|
static void NamingScreen_CreatePCIcon(void);
|
|
static void NamingScreen_CreateMonIcon(void);
|
|
static void NamingScreen_CreateWaldaDadIcon(void);
|
|
|
|
static void (*const sIconFunctions[])(void) =
|
|
{
|
|
NamingScreen_NoIcon,
|
|
NamingScreen_CreatePlayerIcon,
|
|
NamingScreen_CreatePCIcon,
|
|
NamingScreen_CreateMonIcon,
|
|
NamingScreen_CreateWaldaDadIcon,
|
|
};
|
|
|
|
static void CreateInputTargetIcon(void)
|
|
{
|
|
sIconFunctions[sNamingScreen->template->iconFunction]();
|
|
}
|
|
|
|
static void NamingScreen_NoIcon(void)
|
|
{
|
|
|
|
}
|
|
|
|
static void NamingScreen_CreatePlayerIcon(void)
|
|
{
|
|
u8 rivalGfxId;
|
|
u8 spriteId;
|
|
|
|
rivalGfxId = GetRivalAvatarGraphicsIdByStateIdAndGender(0, sNamingScreen->monSpecies);
|
|
spriteId = AddPseudoObjectEvent(rivalGfxId, SpriteCallbackDummy, 56, 37, 0);
|
|
gSprites[spriteId].oam.priority = 3;
|
|
StartSpriteAnim(&gSprites[spriteId], 4);
|
|
}
|
|
|
|
static void NamingScreen_CreatePCIcon(void)
|
|
{
|
|
u8 spriteId;
|
|
|
|
spriteId = CreateSprite(&sSpriteTemplate_PCIcon, 56, 41, 0);
|
|
SetSubspriteTables(&gSprites[spriteId], sSubspriteTable_PCIcon);
|
|
gSprites[spriteId].oam.priority = 3;
|
|
}
|
|
|
|
static void NamingScreen_CreateMonIcon(void)
|
|
{
|
|
u8 spriteId;
|
|
|
|
LoadMonIconPalettes();
|
|
spriteId = CreateMonIcon(sNamingScreen->monSpecies, SpriteCallbackDummy, 56, 40, 0, sNamingScreen->monPersonality, 1);
|
|
gSprites[spriteId].oam.priority = 3;
|
|
}
|
|
|
|
static void NamingScreen_CreateWaldaDadIcon(void)
|
|
{
|
|
u8 spriteId;
|
|
|
|
spriteId = AddPseudoObjectEvent(OBJ_EVENT_GFX_MAN_1, SpriteCallbackDummy, 56, 37, 0);
|
|
gSprites[spriteId].oam.priority = 3;
|
|
StartSpriteAnim(&gSprites[spriteId], 4);
|
|
}
|
|
|
|
//--------------------------------------------------
|
|
// Keyboard handling
|
|
//--------------------------------------------------
|
|
|
|
static bool8 KeyboardKeyHandler_Character(u8);
|
|
static bool8 KeyboardKeyHandler_Page(u8);
|
|
static bool8 KeyboardKeyHandler_Backspace(u8);
|
|
static bool8 KeyboardKeyHandler_OK(u8);
|
|
|
|
static bool8 (*const sKeyboardKeyHandlers[])(u8) =
|
|
{
|
|
[KEY_ROLE_CHAR] = KeyboardKeyHandler_Character,
|
|
[KEY_ROLE_PAGE] = KeyboardKeyHandler_Page,
|
|
[KEY_ROLE_BACKSPACE] = KeyboardKeyHandler_Backspace,
|
|
[KEY_ROLE_OK] = KeyboardKeyHandler_OK,
|
|
};
|
|
|
|
static bool8 HandleKeyboardEvent(void)
|
|
{
|
|
u8 input = GetInputEvent();
|
|
u8 keyRole = GetKeyRoleAtCursorPos();
|
|
|
|
if (input == INPUT_SELECT)
|
|
{
|
|
return SwapKeyboardPage();
|
|
}
|
|
else if (input == INPUT_B_BUTTON)
|
|
{
|
|
DeleteTextCharacter();
|
|
return FALSE;
|
|
}
|
|
else if (input == INPUT_START)
|
|
{
|
|
MoveCursorToOKButton();
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
return sKeyboardKeyHandlers[keyRole](input);
|
|
}
|
|
}
|
|
|
|
static bool8 KeyboardKeyHandler_Character(u8 input)
|
|
{
|
|
TryStartButtonFlash(BUTTON_COUNT, FALSE, FALSE);
|
|
if (input == INPUT_A_BUTTON)
|
|
{
|
|
bool8 textFull = AddTextCharacter();
|
|
|
|
SquishCursor();
|
|
if (textFull)
|
|
{
|
|
SetInputState(INPUT_STATE_OVERRIDE);
|
|
sNamingScreen->state = STATE_MOVE_TO_OK_BUTTON;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 KeyboardKeyHandler_Page(u8 input)
|
|
{
|
|
TryStartButtonFlash(BUTTON_PAGE, TRUE, FALSE);
|
|
if (input == INPUT_A_BUTTON)
|
|
return SwapKeyboardPage();
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 KeyboardKeyHandler_Backspace(u8 input)
|
|
{
|
|
TryStartButtonFlash(BUTTON_BACK, TRUE, FALSE);
|
|
if (input == INPUT_A_BUTTON)
|
|
DeleteTextCharacter();
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 KeyboardKeyHandler_OK(u8 input)
|
|
{
|
|
TryStartButtonFlash(BUTTON_OK, TRUE, FALSE);
|
|
if (input == INPUT_A_BUTTON)
|
|
{
|
|
PlaySE(SE_SELECT);
|
|
sNamingScreen->state = STATE_PRESSED_OK;
|
|
return TRUE;
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
static bool8 SwapKeyboardPage(void)
|
|
{
|
|
sNamingScreen->state = STATE_START_PAGE_SWAP;
|
|
return TRUE;
|
|
}
|
|
|
|
//--------------------------------------------------
|
|
// Input handling
|
|
//--------------------------------------------------
|
|
|
|
#define tState data[0]
|
|
#define tKeyboardEvent data[1]
|
|
#define tButtonId data[2]
|
|
|
|
|
|
static void Input_Disabled(struct Task *);
|
|
static void Input_Enabled(struct Task *);
|
|
static void Input_Override(struct Task *);
|
|
|
|
static void (*const sInputFuncs[])(struct Task *) =
|
|
{
|
|
[INPUT_STATE_DISABLED] = Input_Disabled,
|
|
[INPUT_STATE_ENABLED] = Input_Enabled,
|
|
[INPUT_STATE_OVERRIDE] = Input_Override,
|
|
};
|
|
|
|
static void Task_HandleInput(u8);
|
|
static void HandleDpadMovement(struct Task *);
|
|
|
|
static void CreateInputHandlerTask(void)
|
|
{
|
|
CreateTask(Task_HandleInput, 1);
|
|
}
|
|
|
|
static u8 GetInputEvent(void)
|
|
{
|
|
u8 taskId = FindTaskIdByFunc(Task_HandleInput);
|
|
|
|
return gTasks[taskId].tKeyboardEvent;
|
|
}
|
|
|
|
static void SetInputState(u8 state)
|
|
{
|
|
u8 taskId = FindTaskIdByFunc(Task_HandleInput);
|
|
|
|
gTasks[taskId].tState = state;
|
|
}
|
|
|
|
static void Task_HandleInput(u8 taskId)
|
|
{
|
|
sInputFuncs[gTasks[taskId].tState](&gTasks[taskId]);
|
|
}
|
|
|
|
static void Input_Disabled(struct Task *task)
|
|
{
|
|
task->tKeyboardEvent = INPUT_NONE;
|
|
}
|
|
|
|
static void Input_Enabled(struct Task *task)
|
|
{
|
|
task->tKeyboardEvent = INPUT_NONE;
|
|
|
|
if (JOY_NEW(A_BUTTON))
|
|
task->tKeyboardEvent = INPUT_A_BUTTON;
|
|
else if (JOY_NEW(B_BUTTON))
|
|
task->tKeyboardEvent = INPUT_B_BUTTON;
|
|
else if (JOY_NEW(SELECT_BUTTON))
|
|
task->tKeyboardEvent = INPUT_SELECT;
|
|
else if (JOY_NEW(START_BUTTON))
|
|
task->tKeyboardEvent = INPUT_START;
|
|
else
|
|
HandleDpadMovement(task);
|
|
}
|
|
|
|
static void Input_Override(struct Task *task)
|
|
{
|
|
task->tKeyboardEvent = INPUT_NONE;
|
|
}
|
|
|
|
static void HandleDpadMovement(struct Task *task)
|
|
{
|
|
const s16 sDpadDeltaX[] =
|
|
{
|
|
[INPUT_NONE] = 0,
|
|
[INPUT_DPAD_UP] = 0,
|
|
[INPUT_DPAD_DOWN] = 0,
|
|
[INPUT_DPAD_LEFT] = -1,
|
|
[INPUT_DPAD_RIGHT] = 1
|
|
};
|
|
|
|
const s16 sDpadDeltaY[] =
|
|
{
|
|
[INPUT_NONE] = 0,
|
|
[INPUT_DPAD_UP] = -1,
|
|
[INPUT_DPAD_DOWN] = 1,
|
|
[INPUT_DPAD_LEFT] = 0,
|
|
[INPUT_DPAD_RIGHT] = 0
|
|
};
|
|
|
|
const s16 sKeyRowToButtonRow[KBROW_COUNT] = {0, 1, 1, 2};
|
|
const s16 sButtonRowToKeyRow[BUTTON_COUNT] = {0, 0, 3};
|
|
|
|
s16 cursorX;
|
|
s16 cursorY;
|
|
u16 input;
|
|
s16 prevCursorX;
|
|
|
|
GetCursorPos(&cursorX, &cursorY);
|
|
input = INPUT_NONE;
|
|
if (JOY_REPEAT(DPAD_UP))
|
|
input = INPUT_DPAD_UP;
|
|
if (JOY_REPEAT(DPAD_DOWN))
|
|
input = INPUT_DPAD_DOWN;
|
|
if (JOY_REPEAT(DPAD_LEFT))
|
|
input = INPUT_DPAD_LEFT;
|
|
if (JOY_REPEAT(DPAD_RIGHT))
|
|
input = INPUT_DPAD_RIGHT;
|
|
|
|
// Get new cursor position
|
|
prevCursorX = cursorX;
|
|
cursorX += sDpadDeltaX[input];
|
|
cursorY += sDpadDeltaY[input];
|
|
|
|
// Wrap cursor position in the X direction
|
|
if (cursorX < 0)
|
|
cursorX = GetCurrentPageColumnCount();
|
|
if (cursorX > GetCurrentPageColumnCount())
|
|
cursorX = 0;
|
|
|
|
|
|
// Handle moving on/off the button column
|
|
if (sDpadDeltaX[input] != 0)
|
|
{
|
|
if (cursorX == GetCurrentPageColumnCount())
|
|
{
|
|
// Moved onto button column
|
|
// Save cursor pos in tButtonId for moving back onto keys
|
|
task->tButtonId = cursorY;
|
|
cursorY = sKeyRowToButtonRow[cursorY];
|
|
}
|
|
else if (prevCursorX == GetCurrentPageColumnCount())
|
|
{
|
|
// Moved off button column
|
|
// If you're on the middle button, go to the row that
|
|
// the cursor was on previously (saved in tButtonId above)
|
|
if (cursorY == BUTTON_COUNT / 2)
|
|
cursorY = task->tButtonId;
|
|
else
|
|
cursorY = sButtonRowToKeyRow[cursorY];
|
|
}
|
|
}
|
|
|
|
// Wrap cursor position in the y direction
|
|
if (cursorX == GetCurrentPageColumnCount())
|
|
{
|
|
// There are only 3 keys in the button column
|
|
// so wrap Y accordingly
|
|
if (cursorY < 0)
|
|
cursorY = BUTTON_COUNT - 1;
|
|
if (cursorY >= BUTTON_COUNT)
|
|
cursorY = 0;
|
|
|
|
if (cursorY == 0)
|
|
task->tButtonId = BUTTON_BACK;
|
|
else if (cursorY == BUTTON_COUNT - 1)
|
|
task->tButtonId = BUTTON_OK;
|
|
}
|
|
else
|
|
{
|
|
if (cursorY < 0)
|
|
cursorY = KBROW_COUNT - 1;
|
|
if (cursorY > KBROW_COUNT - 1)
|
|
cursorY = 0;
|
|
}
|
|
SetCursorPos(cursorX, cursorY);
|
|
}
|
|
|
|
#undef tState
|
|
#undef tKeyboardEvent
|
|
#undef tButtonId
|
|
|
|
static void DrawNormalTextEntryBox(void)
|
|
{
|
|
FillWindowPixelBuffer(sNamingScreen->windows[WIN_TEXT_ENTRY_BOX], PIXEL_FILL(1));
|
|
AddTextPrinterParameterized(sNamingScreen->windows[WIN_TEXT_ENTRY_BOX], 1, sNamingScreen->template->title, 8, 1, 0, 0);
|
|
PutWindowTilemap(sNamingScreen->windows[WIN_TEXT_ENTRY_BOX]);
|
|
}
|
|
|
|
static void DrawMonTextEntryBox(void)
|
|
{
|
|
u8 buffer[32];
|
|
|
|
StringCopy(buffer, gSpeciesNames[sNamingScreen->monSpecies]);
|
|
StringAppendN(buffer, sNamingScreen->template->title, 15);
|
|
FillWindowPixelBuffer(sNamingScreen->windows[WIN_TEXT_ENTRY_BOX], PIXEL_FILL(1));
|
|
AddTextPrinterParameterized(sNamingScreen->windows[WIN_TEXT_ENTRY_BOX], 1, buffer, 8, 1, 0, 0);
|
|
PutWindowTilemap(sNamingScreen->windows[WIN_TEXT_ENTRY_BOX]);
|
|
}
|
|
|
|
static void (*const sDrawTextEntryBoxFuncs[])(void) =
|
|
{
|
|
[NAMING_SCREEN_PLAYER] = DrawNormalTextEntryBox,
|
|
[NAMING_SCREEN_BOX] = DrawNormalTextEntryBox,
|
|
[NAMING_SCREEN_CAUGHT_MON] = DrawMonTextEntryBox,
|
|
[NAMING_SCREEN_NICKNAME] = DrawMonTextEntryBox,
|
|
[NAMING_SCREEN_WALDA] = DrawNormalTextEntryBox,
|
|
};
|
|
|
|
static void DrawTextEntryBox(void)
|
|
{
|
|
sDrawTextEntryBoxFuncs[sNamingScreen->templateNum]();
|
|
}
|
|
|
|
static void DummyGenderIcon(void);
|
|
static void DrawGenderIcon(void);
|
|
|
|
static void (*const sDrawGenderIconFuncs[])(void) =
|
|
{
|
|
[FALSE] = DummyGenderIcon,
|
|
[TRUE] = DrawGenderIcon,
|
|
};
|
|
|
|
static void TryDrawGenderIcon(void)
|
|
{
|
|
sDrawGenderIconFuncs[sNamingScreen->template->addGenderIcon]();
|
|
}
|
|
|
|
static void DummyGenderIcon(void)
|
|
{
|
|
|
|
}
|
|
|
|
static const u8 sGenderColors[2][3] =
|
|
{
|
|
{TEXT_COLOR_TRANSPARENT, TEXT_COLOR_LIGHT_BLUE, TEXT_COLOR_BLUE},
|
|
{TEXT_COLOR_TRANSPARENT, TEXT_COLOR_LIGHT_RED, TEXT_COLOR_RED}
|
|
};
|
|
|
|
static void DrawGenderIcon(void)
|
|
{
|
|
u8 text[2];
|
|
bool8 isFemale = FALSE;
|
|
|
|
StringCopy(text, gText_MaleSymbol);
|
|
if (sNamingScreen->monGender != MON_GENDERLESS)
|
|
{
|
|
if (sNamingScreen->monGender == MON_FEMALE)
|
|
{
|
|
StringCopy(text, gText_FemaleSymbol);
|
|
isFemale = TRUE;
|
|
}
|
|
AddTextPrinterParameterized3(sNamingScreen->windows[WIN_TEXT_ENTRY], 1, 0x68, 1, sGenderColors[isFemale], -1, text);
|
|
}
|
|
}
|
|
|
|
static u8 GetCharAtKeyboardPos(s16 x, s16 y)
|
|
{
|
|
return sKeyboardChars[x + y * KBCOL_COUNT + CurrentPageToKeyboardId() * KBCOL_COUNT * KBROW_COUNT];
|
|
}
|
|
|
|
|
|
static u8 GetTextEntryPosition(void)
|
|
{
|
|
u8 i;
|
|
|
|
for (i = 0; i < sNamingScreen->template->maxChars; i++)
|
|
{
|
|
if (sNamingScreen->textBuffer[i] == EOS)
|
|
return i;
|
|
}
|
|
return sNamingScreen->template->maxChars - 1;
|
|
}
|
|
|
|
static u8 GetPreviousTextCaretPosition(void)
|
|
{
|
|
s8 i;
|
|
|
|
for (i = sNamingScreen->template->maxChars - 1; i > 0; i--)
|
|
{
|
|
if (sNamingScreen->textBuffer[i] != EOS)
|
|
return i;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void DeleteTextCharacter(void)
|
|
{
|
|
u8 index;
|
|
u8 keyRole;
|
|
|
|
index = GetPreviousTextCaretPosition();
|
|
sNamingScreen->textBuffer[index] = 0;
|
|
DrawTextEntry();
|
|
CopyBgTilemapBufferToVram(3);
|
|
sNamingScreen->textBuffer[index] = EOS;
|
|
keyRole = GetKeyRoleAtCursorPos();
|
|
|
|
// The below flashes the Back key once on delete
|
|
// It incorrectly leaves the Back key 1 shade lighter than its default
|
|
if (keyRole == KEY_ROLE_CHAR || keyRole == KEY_ROLE_BACKSPACE)
|
|
TryStartButtonFlash(BUTTON_BACK, FALSE, TRUE);
|
|
PlaySE(SE_BALL);
|
|
}
|
|
|
|
// Returns TRUE if the text entry is now full
|
|
static bool8 AddTextCharacter(void)
|
|
{
|
|
s16 x;
|
|
s16 y;
|
|
|
|
GetCursorPos(&x, &y);
|
|
BufferCharacter(GetCharAtKeyboardPos(x, y));
|
|
DrawTextEntry();
|
|
CopyBgTilemapBufferToVram(3);
|
|
PlaySE(SE_SELECT);
|
|
|
|
if (GetPreviousTextCaretPosition() != sNamingScreen->template->maxChars - 1)
|
|
return FALSE;
|
|
else
|
|
return TRUE;
|
|
}
|
|
|
|
static void BufferCharacter(u8 ch)
|
|
{
|
|
u8 index = GetTextEntryPosition();
|
|
sNamingScreen->textBuffer[index] = ch;
|
|
}
|
|
|
|
static void SaveInputText(void)
|
|
{
|
|
u8 i;
|
|
|
|
for (i = 0; i < sNamingScreen->template->maxChars; i++)
|
|
{
|
|
if (sNamingScreen->textBuffer[i] != CHAR_SPACE && sNamingScreen->textBuffer[i] != EOS)
|
|
{
|
|
StringCopyN(sNamingScreen->destBuffer, sNamingScreen->textBuffer, sNamingScreen->template->maxChars + 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void LoadGfx(void)
|
|
{
|
|
LZ77UnCompWram(gNamingScreenMenu_Gfx, sNamingScreen->tileBuffer);
|
|
LoadBgTiles(1, sNamingScreen->tileBuffer, sizeof(sNamingScreen->tileBuffer), 0);
|
|
LoadBgTiles(2, sNamingScreen->tileBuffer, sizeof(sNamingScreen->tileBuffer), 0);
|
|
LoadBgTiles(3, sNamingScreen->tileBuffer, sizeof(sNamingScreen->tileBuffer), 0);
|
|
LoadSpriteSheets(sSpriteSheets);
|
|
LoadSpritePalettes(sSpritePalettes);
|
|
}
|
|
|
|
static void CreateHelperTasks(void)
|
|
{
|
|
CreateInputHandlerTask();
|
|
CreateButtonFlashTask();
|
|
}
|
|
|
|
static void LoadPalettes(void)
|
|
{
|
|
LoadPalette(gNamingScreenMenu_Pal, 0, 0xC0);
|
|
LoadPalette(sKeyboard_Pal, 0xA0, sizeof(sKeyboard_Pal));
|
|
LoadPalette(GetTextWindowPalette(2), 0xB0, 0x20);
|
|
}
|
|
|
|
static void DrawBgTilemap(u8 bg, const void *src)
|
|
{
|
|
CopyToBgTilemapBuffer(bg, src, 0, 0);
|
|
}
|
|
|
|
static void NamingScreen_Dummy(u8 bg, u8 page)
|
|
{
|
|
|
|
}
|
|
|
|
static void DrawTextEntry(void)
|
|
{
|
|
u8 i;
|
|
u8 temp[2];
|
|
u16 extraWidth;
|
|
u8 maxChars = sNamingScreen->template->maxChars;
|
|
u16 x = sNamingScreen->inputCharBaseXPos - 0x40;
|
|
|
|
FillWindowPixelBuffer(sNamingScreen->windows[WIN_TEXT_ENTRY], PIXEL_FILL(1));
|
|
|
|
for (i = 0; i < maxChars; i++)
|
|
{
|
|
temp[0] = sNamingScreen->textBuffer[i];
|
|
temp[1] = gText_ExpandedPlaceholder_Empty[0];
|
|
extraWidth = (IsWideLetter(temp[0]) == TRUE) ? 2 : 0;
|
|
|
|
AddTextPrinterParameterized(sNamingScreen->windows[WIN_TEXT_ENTRY], 1, temp, i * 8 + x + extraWidth, 1, 0xFF, NULL);
|
|
}
|
|
|
|
TryDrawGenderIcon();
|
|
CopyWindowToVram(sNamingScreen->windows[WIN_TEXT_ENTRY], 2);
|
|
PutWindowTilemap(sNamingScreen->windows[WIN_TEXT_ENTRY]);
|
|
}
|
|
|
|
struct TextColor // Needed because of alignment
|
|
{
|
|
u8 colors[3][4];
|
|
};
|
|
|
|
static const struct TextColor sTextColorStruct =
|
|
{
|
|
{
|
|
{TEXT_DYNAMIC_COLOR_4, TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GREY},
|
|
{TEXT_DYNAMIC_COLOR_5, TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GREY},
|
|
{TEXT_DYNAMIC_COLOR_6, TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GREY}
|
|
}
|
|
};
|
|
|
|
static const u8 sFillValues[KBPAGE_COUNT] =
|
|
{
|
|
[KEYBOARD_LETTERS_LOWER] = PIXEL_FILL(14),
|
|
[KEYBOARD_LETTERS_UPPER] = PIXEL_FILL(13),
|
|
[KEYBOARD_SYMBOLS] = PIXEL_FILL(15)
|
|
};
|
|
|
|
static const u8 *const sKeyboardTextColors[KBPAGE_COUNT] =
|
|
{
|
|
[KEYBOARD_LETTERS_LOWER] = sTextColorStruct.colors[1],
|
|
[KEYBOARD_LETTERS_UPPER] = sTextColorStruct.colors[0],
|
|
[KEYBOARD_SYMBOLS] = sTextColorStruct.colors[2]
|
|
};
|
|
|
|
static void PrintKeyboardKeys(u8 window, u8 page)
|
|
{
|
|
u8 i;
|
|
|
|
FillWindowPixelBuffer(window, sFillValues[page]);
|
|
|
|
for (i = 0; i < KBROW_COUNT; i++)
|
|
AddTextPrinterParameterized3(window, 1, 0, i * 16 + 1, sKeyboardTextColors[page], 0, sNamingScreenKeyboardText[page][i]);
|
|
|
|
PutWindowTilemap(window);
|
|
}
|
|
|
|
static const u8 *const sNextKeyboardPageTilemaps[] =
|
|
{
|
|
[KBPAGE_SYMBOLS] = gNamingScreenKeyboardUpper_Tilemap,
|
|
[KBPAGE_LETTERS_UPPER] = gNamingScreenKeyboardLower_Tilemap, // lower
|
|
[KBPAGE_LETTERS_LOWER] = gNamingScreenKeyboardSymbols_Tilemap // symbols
|
|
};
|
|
|
|
// There are always 2 keyboard pages drawn, the current page and the one that will shown next if the player swaps
|
|
// When the page swap is complete this function invisibly replaces the old page with the new next one
|
|
static void DrawKeyboardPageOnDeck(void)
|
|
{
|
|
u8 bg;
|
|
u8 bg_;
|
|
u8 windowId;
|
|
u8 bg1Priority = GetGpuReg(REG_OFFSET_BG1CNT) & 3;
|
|
u8 bg2Priority = GetGpuReg(REG_OFFSET_BG2CNT) & 3;
|
|
|
|
if (bg1Priority > bg2Priority)
|
|
{
|
|
bg = 1;
|
|
bg_ = 1;
|
|
windowId = sNamingScreen->windows[WIN_KB_PAGE_1];
|
|
}
|
|
else
|
|
{
|
|
bg = 2;
|
|
bg_ = 2;
|
|
windowId = sNamingScreen->windows[WIN_KB_PAGE_2];
|
|
}
|
|
|
|
DrawBgTilemap(bg, sNextKeyboardPageTilemaps[sNamingScreen->currentPage]);
|
|
PrintKeyboardKeys(windowId, CurrentPageToNextKeyboardId());
|
|
NamingScreen_Dummy(bg, CurrentPageToNextKeyboardId());
|
|
CopyBgTilemapBufferToVram(bg_);
|
|
}
|
|
|
|
static void PrintControls(void)
|
|
{
|
|
const u8 color[3] = { TEXT_DYNAMIC_COLOR_6, TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GREY };
|
|
|
|
FillWindowPixelBuffer(sNamingScreen->windows[WIN_BANNER], PIXEL_FILL(15));
|
|
AddTextPrinterParameterized3(sNamingScreen->windows[WIN_BANNER], 0, 2, 1, color, 0, gText_MoveOkBack);
|
|
PutWindowTilemap(sNamingScreen->windows[WIN_BANNER]);
|
|
CopyWindowToVram(sNamingScreen->windows[WIN_BANNER], 3);
|
|
}
|
|
|
|
static void CB2_NamingScreen(void)
|
|
{
|
|
RunTasks();
|
|
AnimateSprites();
|
|
BuildOamBuffer();
|
|
UpdatePaletteFade();
|
|
}
|
|
|
|
static void ResetVHBlank(void)
|
|
{
|
|
SetVBlankCallback(NULL);
|
|
SetHBlankCallback(NULL);
|
|
}
|
|
|
|
static void SetVBlank(void)
|
|
{
|
|
SetVBlankCallback(VBlankCB_NamingScreen);
|
|
}
|
|
|
|
static void VBlankCB_NamingScreen(void)
|
|
{
|
|
LoadOam();
|
|
ProcessSpriteCopyRequests();
|
|
TransferPlttBuffer();
|
|
SetGpuReg(REG_OFFSET_BG1VOFS, sNamingScreen->bg1vOffset);
|
|
SetGpuReg(REG_OFFSET_BG2VOFS, sNamingScreen->bg2vOffset);
|
|
SetGpuReg(REG_OFFSET_BG1CNT, GetGpuReg(REG_OFFSET_BG1CNT) & 0xFFFC);
|
|
SetGpuRegBits(REG_OFFSET_BG1CNT, sNamingScreen->bg1Priority);
|
|
SetGpuReg(REG_OFFSET_BG2CNT, GetGpuReg(REG_OFFSET_BG2CNT) & 0xFFFC);
|
|
SetGpuRegBits(REG_OFFSET_BG2CNT, sNamingScreen->bg2Priority);
|
|
}
|
|
|
|
static void NamingScreen_ShowBgs(void)
|
|
{
|
|
ShowBg(0);
|
|
ShowBg(1);
|
|
ShowBg(2);
|
|
ShowBg(3);
|
|
}
|
|
|
|
// Always false (presumably for non-latin languages)
|
|
static bool8 IsWideLetter(u8 character)
|
|
{
|
|
u8 i;
|
|
|
|
for (i = 0; sText_AlphabetUpperLower[i] != EOS; i++)
|
|
{
|
|
if (character == sText_AlphabetUpperLower[i])
|
|
return FALSE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
// Debug? Unused, and arguments aren't sensible for non-player screens.
|
|
static void Debug_NamingScreenPlayer(void)
|
|
{
|
|
DoNamingScreen(NAMING_SCREEN_PLAYER, gSaveBlock2Ptr->playerName, gSaveBlock2Ptr->playerGender, 0, 0, CB2_ReturnToFieldWithOpenMenu);
|
|
}
|
|
|
|
static void Debug_NamingScreenBox(void)
|
|
{
|
|
DoNamingScreen(NAMING_SCREEN_BOX, gSaveBlock2Ptr->playerName, gSaveBlock2Ptr->playerGender, 0, 0, CB2_ReturnToFieldWithOpenMenu);
|
|
}
|
|
|
|
static void Debug_NamingScreenCaughtMon(void)
|
|
{
|
|
DoNamingScreen(NAMING_SCREEN_CAUGHT_MON, gSaveBlock2Ptr->playerName, gSaveBlock2Ptr->playerGender, 0, 0, CB2_ReturnToFieldWithOpenMenu);
|
|
}
|
|
|
|
static void Debug_NamingScreenNickname(void)
|
|
{
|
|
DoNamingScreen(NAMING_SCREEN_NICKNAME, gSaveBlock2Ptr->playerName, gSaveBlock2Ptr->playerGender, 0, 0, CB2_ReturnToFieldWithOpenMenu);
|
|
}
|
|
|
|
//--------------------------------------------------
|
|
// Forward-declared variables
|
|
//--------------------------------------------------
|
|
|
|
// Initial pages below are pointless, they're overwritten with KBPAGE_LETTERS_UPPER in MainState_FadeIn()
|
|
static const struct NamingScreenTemplate sPlayerNamingScreenTemplate =
|
|
{
|
|
.copyExistingString = FALSE,
|
|
.maxChars = 7,
|
|
.iconFunction = 1,
|
|
.addGenderIcon = FALSE,
|
|
.initialPage = KBPAGE_LETTERS_UPPER,
|
|
.unused = 35,
|
|
.title = gText_YourName,
|
|
};
|
|
|
|
static const struct NamingScreenTemplate sPCBoxNamingTemplate =
|
|
{
|
|
.copyExistingString = FALSE,
|
|
.maxChars = 8,
|
|
.iconFunction = 2,
|
|
.addGenderIcon = FALSE,
|
|
.initialPage = KBPAGE_LETTERS_UPPER,
|
|
.unused = 19,
|
|
.title = gText_BoxName,
|
|
};
|
|
|
|
static const struct NamingScreenTemplate sMonNamingScreenTemplate =
|
|
{
|
|
.copyExistingString = FALSE,
|
|
.maxChars = 10,
|
|
.iconFunction = 3,
|
|
.addGenderIcon = TRUE,
|
|
.initialPage = KBPAGE_LETTERS_UPPER,
|
|
.unused = 35,
|
|
.title = gText_PkmnsNickname,
|
|
};
|
|
|
|
static const struct NamingScreenTemplate sWaldaWordsScreenTemplate =
|
|
{
|
|
.copyExistingString = TRUE,
|
|
.maxChars = 15,
|
|
.iconFunction = 4,
|
|
.addGenderIcon = FALSE,
|
|
.initialPage = KBPAGE_LETTERS_UPPER,
|
|
.unused = 11,
|
|
.title = gText_TellHimTheWords,
|
|
};
|
|
|
|
static const struct NamingScreenTemplate *const sNamingScreenTemplates[] =
|
|
{
|
|
[NAMING_SCREEN_PLAYER] = &sPlayerNamingScreenTemplate,
|
|
[NAMING_SCREEN_BOX] = &sPCBoxNamingTemplate,
|
|
[NAMING_SCREEN_CAUGHT_MON] = &sMonNamingScreenTemplate,
|
|
[NAMING_SCREEN_NICKNAME] = &sMonNamingScreenTemplate,
|
|
[NAMING_SCREEN_WALDA] = &sWaldaWordsScreenTemplate,
|
|
};
|
|
|
|
static const struct OamData sOam_8x8 =
|
|
{
|
|
.y = 0,
|
|
.affineMode = ST_OAM_AFFINE_OFF,
|
|
.objMode = ST_OAM_OBJ_NORMAL,
|
|
.bpp = ST_OAM_4BPP,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.x = 0,
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileNum = 0,
|
|
.priority = 0,
|
|
.paletteNum = 0,
|
|
};
|
|
|
|
static const struct OamData sOam_16x16 =
|
|
{
|
|
.y = 0,
|
|
.affineMode = ST_OAM_AFFINE_OFF,
|
|
.objMode = ST_OAM_OBJ_NORMAL,
|
|
.bpp = ST_OAM_4BPP,
|
|
.shape = SPRITE_SHAPE(16x16),
|
|
.x = 0,
|
|
.size = SPRITE_SIZE(16x16),
|
|
.tileNum = 0,
|
|
.priority = 0,
|
|
.paletteNum = 0,
|
|
};
|
|
|
|
static const struct OamData sOam_32x16 =
|
|
{
|
|
.y = 0,
|
|
.affineMode = ST_OAM_AFFINE_OFF,
|
|
.objMode = ST_OAM_OBJ_NORMAL,
|
|
.bpp = ST_OAM_4BPP,
|
|
.shape = SPRITE_SHAPE(32x16),
|
|
.x = 0,
|
|
.size = SPRITE_SIZE(32x16),
|
|
.tileNum = 0,
|
|
.priority = 0,
|
|
.paletteNum = 0,
|
|
};
|
|
|
|
static const struct Subsprite sSubsprites_PageSwapFrame[] =
|
|
{
|
|
{
|
|
.x = -20,
|
|
.y = -16,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 0,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = -16,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 4,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = -20,
|
|
.y = -8,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 5,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = -8,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 9,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = -20,
|
|
.y = 0,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 10,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = 0,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 14,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = -20,
|
|
.y = 8,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 15,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = 8,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 19,
|
|
.priority = 1
|
|
}
|
|
};
|
|
|
|
static const struct Subsprite sSubsprites_PageSwapText[] =
|
|
{
|
|
{
|
|
.x = -12,
|
|
.y = -4,
|
|
.shape = SPRITE_SHAPE(16x8),
|
|
.size = SPRITE_SIZE(16x8),
|
|
.tileOffset = 0,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 4,
|
|
.y = -4,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 2,
|
|
.priority = 1
|
|
}
|
|
};
|
|
|
|
static const struct Subsprite sSubsprites_Button[] =
|
|
{
|
|
{
|
|
.x = -20,
|
|
.y = -12,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 0,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = -12,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 4,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = -20,
|
|
.y = -4,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 5,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = -4,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 9,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = -20,
|
|
.y = 4,
|
|
.shape = SPRITE_SHAPE(32x8),
|
|
.size = SPRITE_SIZE(32x8),
|
|
.tileOffset = 10,
|
|
.priority = 1
|
|
},
|
|
{
|
|
.x = 12,
|
|
.y = 4,
|
|
.shape = SPRITE_SHAPE(8x8),
|
|
.size = SPRITE_SIZE(8x8),
|
|
.tileOffset = 14,
|
|
.priority = 1
|
|
}
|
|
};
|
|
|
|
static const struct Subsprite sSubsprites_PCIcon[] =
|
|
{
|
|
{
|
|
.x = -8,
|
|
.y = -12,
|
|
.shape = SPRITE_SHAPE(16x8),
|
|
.size = SPRITE_SIZE(16x8),
|
|
.tileOffset = 0,
|
|
.priority = 3
|
|
},
|
|
{
|
|
.x = -8,
|
|
.y = -4,
|
|
.shape = SPRITE_SHAPE(16x8),
|
|
.size = SPRITE_SIZE(16x8),
|
|
.tileOffset = 2,
|
|
.priority = 3
|
|
},
|
|
{
|
|
.x = -8,
|
|
.y = 4,
|
|
.shape = SPRITE_SHAPE(16x8),
|
|
.size = SPRITE_SIZE(16x8),
|
|
.tileOffset = 4,
|
|
.priority = 3
|
|
}
|
|
};
|
|
|
|
static const struct SubspriteTable sSubspriteTable_PageSwapFrame[] =
|
|
{
|
|
{ARRAY_COUNT(sSubsprites_PageSwapFrame), sSubsprites_PageSwapFrame}
|
|
};
|
|
|
|
static const struct SubspriteTable sSubspriteTable_PageSwapText[] =
|
|
{
|
|
{ARRAY_COUNT(sSubsprites_PageSwapText), sSubsprites_PageSwapText},
|
|
{ARRAY_COUNT(sSubsprites_PageSwapText), sSubsprites_PageSwapText},
|
|
{ARRAY_COUNT(sSubsprites_PageSwapText), sSubsprites_PageSwapText}
|
|
};
|
|
|
|
static const struct SubspriteTable sSubspriteTable_Button[] =
|
|
{
|
|
{ARRAY_COUNT(sSubsprites_Button), sSubsprites_Button}
|
|
};
|
|
|
|
static const struct SubspriteTable sSubspriteTable_PCIcon[] =
|
|
{
|
|
{ARRAY_COUNT(sSubsprites_PCIcon), sSubsprites_PCIcon}
|
|
};
|
|
|
|
static const struct SpriteFrameImage sImageTable_PCIcon[] =
|
|
{
|
|
{sPCIconOff_Gfx, sizeof(sPCIconOff_Gfx)},
|
|
{sPCIconOn_Gfx, sizeof(sPCIconOn_Gfx)},
|
|
};
|
|
|
|
static const union AnimCmd sAnim_Loop[] =
|
|
{
|
|
ANIMCMD_FRAME(0, 1),
|
|
ANIMCMD_JUMP(0)
|
|
};
|
|
|
|
static const union AnimCmd sAnim_CursorSquish[] =
|
|
{
|
|
ANIMCMD_FRAME(4, 8),
|
|
ANIMCMD_FRAME(8, 8),
|
|
ANIMCMD_END
|
|
};
|
|
|
|
static const union AnimCmd sAnim_PCIcon[] =
|
|
{
|
|
ANIMCMD_FRAME(0, 2),
|
|
ANIMCMD_FRAME(1, 2),
|
|
ANIMCMD_JUMP(0)
|
|
};
|
|
|
|
static const union AnimCmd *const sAnims_Loop[] =
|
|
{
|
|
sAnim_Loop
|
|
};
|
|
|
|
static const union AnimCmd *const sAnims_Cursor[] =
|
|
{
|
|
sAnim_Loop,
|
|
sAnim_CursorSquish
|
|
};
|
|
|
|
static const union AnimCmd *const sAnims_PCIcon[] =
|
|
{
|
|
sAnim_PCIcon
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_PageSwapFrame =
|
|
{
|
|
.tileTag = GFXTAG_PAGE_SWAP_FRAME,
|
|
.paletteTag = PALTAG_PAGE_SWAP,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCB_PageSwap
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_PageSwapButton =
|
|
{
|
|
.tileTag = GFXTAG_PAGE_SWAP_BUTTON,
|
|
.paletteTag = PALTAG_PAGE_SWAP_UPPER,
|
|
.oam = &sOam_32x16,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCallbackDummy
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_PageSwapText =
|
|
{
|
|
.tileTag = GFXTAG_PAGE_SWAP_UPPER,
|
|
.paletteTag = PALTAG_PAGE_SWAP,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCallbackDummy
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_BackButton =
|
|
{
|
|
.tileTag = GFXTAG_BACK_BUTTON,
|
|
.paletteTag = PALTAG_BACK_BUTTON,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCallbackDummy
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_OkButton =
|
|
{
|
|
.tileTag = GFXTAG_OK_BUTTON,
|
|
.paletteTag = PALTAG_OK_BUTTON,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCallbackDummy
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_Cursor =
|
|
{
|
|
.tileTag = GFXTAG_CURSOR,
|
|
.paletteTag = PALTAG_CURSOR,
|
|
.oam = &sOam_16x16,
|
|
.anims = sAnims_Cursor,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCB_Cursor
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_InputArrow =
|
|
{
|
|
.tileTag = GFXTAG_INPUT_ARROW,
|
|
.paletteTag = PALTAG_PAGE_SWAP_OTHERS,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCB_InputArrow
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_Underscore =
|
|
{
|
|
.tileTag = GFXTAG_UNDERSCORE,
|
|
.paletteTag = PALTAG_PAGE_SWAP_OTHERS,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_Loop,
|
|
.images = NULL,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCB_Underscore
|
|
};
|
|
|
|
static const struct SpriteTemplate sSpriteTemplate_PCIcon =
|
|
{
|
|
.tileTag = 0xFFFF,
|
|
.paletteTag = PALTAG_PC_ICON,
|
|
.oam = &sOam_8x8,
|
|
.anims = sAnims_PCIcon,
|
|
.images = sImageTable_PCIcon,
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
.callback = SpriteCallbackDummy
|
|
};
|
|
|
|
static const u8* const sNamingScreenKeyboardText[KBPAGE_COUNT][KBROW_COUNT] =
|
|
{
|
|
[KEYBOARD_LETTERS_LOWER] =
|
|
{
|
|
gText_NamingScreenKeyboard_abcdef,
|
|
gText_NamingScreenKeyboard_ghijkl,
|
|
gText_NamingScreenKeyboard_mnopqrs,
|
|
gText_NamingScreenKeyboard_tuvwxyz
|
|
},
|
|
[KEYBOARD_LETTERS_UPPER] =
|
|
{
|
|
gText_NamingScreenKeyboard_ABCDEF,
|
|
gText_NamingScreenKeyboard_GHIJKL,
|
|
gText_NamingScreenKeyboard_MNOPQRS,
|
|
gText_NamingScreenKeyboard_TUVWXYZ
|
|
},
|
|
[KEYBOARD_SYMBOLS] =
|
|
{
|
|
gText_NamingScreenKeyboard_01234,
|
|
gText_NamingScreenKeyboard_56789,
|
|
gText_NamingScreenKeyboard_Symbols1,
|
|
gText_NamingScreenKeyboard_Symbols2
|
|
},
|
|
};
|
|
|
|
static const struct SpriteSheet sSpriteSheets[] =
|
|
{
|
|
{gNamingScreenRWindow_Gfx + 0x280, 0x1E0, GFXTAG_BACK_BUTTON},
|
|
{gNamingScreenRWindow_Gfx + 0x460, 0x1E0, GFXTAG_OK_BUTTON},
|
|
{gNamingScreenRWindow_Gfx, 0x280, GFXTAG_PAGE_SWAP_FRAME},
|
|
{gNamingScreenPageButton_Gfx + 0x20, 0x100, GFXTAG_PAGE_SWAP_BUTTON},
|
|
{gNamingScreenROptions_Gfx, 0x060, GFXTAG_PAGE_SWAP_UPPER},
|
|
{gNamingScreenROptions_Gfx + 0xA0, 0x060, GFXTAG_PAGE_SWAP_LOWER},
|
|
{gNamingScreenROptions_Gfx + 0x140, 0x060, GFXTAG_PAGE_SWAP_OTHERS},
|
|
{gNamingScreenCursor_Gfx, 0x080, GFXTAG_CURSOR},
|
|
{gNamingScreenCursor_Gfx + 0xA0, 0x080, GFXTAG_CURSOR_SQUISHED},
|
|
{gNamingScreenCursor_Gfx + 0x140, 0x080, GFXTAG_CURSOR_FILLED},
|
|
{gNamingScreenInputArrow_Gfx, 0x020, GFXTAG_INPUT_ARROW},
|
|
{gNamingScreenUnderscore_Gfx, 0x020, GFXTAG_UNDERSCORE},
|
|
{}
|
|
};
|
|
|
|
static const struct SpritePalette sSpritePalettes[] =
|
|
{
|
|
{gNamingScreenMenu_Pal, PALTAG_PC_ICON},
|
|
{gNamingScreenMenu_Pal + 0x10, PALTAG_PAGE_SWAP_UPPER},
|
|
{gNamingScreenMenu_Pal + 0x20, PALTAG_PAGE_SWAP_LOWER},
|
|
{gNamingScreenMenu_Pal + 0x30, PALTAG_PAGE_SWAP_OTHERS},
|
|
{gNamingScreenMenu_Pal + 0x40, PALTAG_PAGE_SWAP},
|
|
{gNamingScreenMenu_Pal + 0x50, PALTAG_CURSOR},
|
|
{gNamingScreenMenu_Pal + 0x40, PALTAG_BACK_BUTTON},
|
|
{gNamingScreenMenu_Pal + 0x40, PALTAG_OK_BUTTON},
|
|
{}
|
|
};
|
|
|
|
|