pokeemerald/src/easy_chat.c
2021-02-24 11:03:51 -05:00

5841 lines
161 KiB
C

#include "global.h"
#include "malloc.h"
#include "bard_music.h"
#include "bg.h"
#include "data.h"
#include "decompress.h"
#include "dewford_trend.h"
#include "dynamic_placeholder_text_util.h"
#include "easy_chat.h"
#include "event_data.h"
#include "event_object_movement.h"
#include "field_message_box.h"
#include "field_weather.h"
#include "gpu_regs.h"
#include "graphics.h"
#include "international_string_util.h"
#include "main.h"
#include "mevent.h"
#include "menu.h"
#include "overworld.h"
#include "palette.h"
#include "pokedex.h"
#include "random.h"
#include "sound.h"
#include "string_util.h"
#include "strings.h"
#include "task.h"
#include "text_window.h"
#include "window.h"
#include "constants/easy_chat.h"
#include "constants/event_objects.h"
#include "constants/lilycove_lady.h"
#include "constants/mauville_old_man.h"
#include "constants/songs.h"
#include "constants/rgb.h"
static EWRAM_DATA struct EasyChatScreen *sEasyChatScreen = NULL;
static EWRAM_DATA struct EasyChatScreenControl *sScreenControl = NULL;
static EWRAM_DATA struct EasyChatScreenWordData *sWordData = NULL;
static void Task_InitEasyChatScreen(u8);
static void CB2_EasyChatScreen(void);
static bool8 InitEasyChatScreen(u8);
static void Task_EasyChatScreen(u8);
static void ExitEasyChatScreen(MainCallback);
static bool32 IsFuncIdForQuizLadyScreen(u16);
static void EnterQuizLadyScreen(u16);
static bool8 InitEasyChatScreenStruct(u8, u16 *, u8);
static void FreeEasyChatScreenStruct(void);
static u16 HandleEasyChatInput(void);
static u16 HandleEasyChatInput_Phrase(void);
static u16 HandleEasyChatInput_MainScreenButtons(void);
static u16 HandleEasyChatInput_Keyboard(void);
static u16 HandleEasyChatInput_WordSelect(void);
static u16 HandleEasyChatInput_ExitPrompt(void);
static u16 HandleEasyChatInput_ConfirmWordsYesNo(void);
static u16 HandleEasyChatInput_DeleteAllYesNo(void);
static u16 HandleEasyChatInput_QuizQuestion(void);
static u16 HandleEasyChatInput_WaitForMsg(void);
static u16 HandleEasyChatInput_StartConfirmLyrics(void);
static u16 HandleEasyChatInput_ConfirmLyricsYesNo(void);
static u16 StartConfirmExitPrompt(void);
static u16 TryConfirmWords(void);
static u8 GetEasyChatScreenFrameId(void);
static u8 GetEachChatScreenTemplateId(u8);
static void GetQuizTitle(u8 *);
static void ClearUnusedField(void);
static bool8 InitEasyChatScreenControl(void);
static bool8 LoadEasyChatScreen(void);
static void FreeEasyChatScreenControl(void);
static void StartEasyChatFunction(u16);
static bool8 RunEasyChatFunction(void);
static bool8 InitEasyChatScreenWordData(void);
static void FreeEasyChatScreenWordData(void);
static u8 GetNumUnlockedEasyChatGroups(void);
static int FooterHasFourOptions(void);
static int DoDeleteAllButton(void);
static int DoQuizButton(void);
static int ExitKeyboardToMainScreen(void);
static int SelectKeyboardGroup(void);
static int StartSwitchKeyboardMode(void);
static int DeleteSelectedWord(void);
static u16 MoveKeyboardCursor(int);
static u16 MoveWordSelectCursor(u32);
static int SelectNewWord(void);
static u8 GetEasyChatBackupState(void);
static void SaveCurrentPhrase(void);
static void SetSpecialEasyChatResult(void);
static bool32 GetEasyChatCompleted(void);
static void ResetCurrentPhrase(void);
static void ResetCurrentPhraseToSaved(void);
static int IsQuizQuestionEmpty(void);
static int IsQuizAnswerEmpty(void);
static bool32 IsCurrentPhraseFull(void);
static bool32 IsCurrentPhraseEmpty(void);
static u16 GetSelectedGroupIndex(void);
static u8 GetUnlockedEasyChatGroupId(u8);
static void SetSelectedWordGroup(bool32, u16);
static int GetSelectedAlphabetGroupId(void);
static u16 GetNumWordsInSelectedGroup(void);
static void SetSelectedWord(u16);
static u16 GetSelectedWordIndex(void);
static u16 GetWordFromSelectedGroup(u16);
static bool32 DummyWordCheck(int);
static u16 GetWordIndexToReplace(void);
static int MoveKeyboardCursor_GroupNames(u32);
static int MoveKeyboardCursor_Alphabet(u32);
static int MoveKeyboardCursor_ButtonWindow(u32);
static void ReduceToValidKeyboardColumn(void);
static void SetKeyboardCursorInButtonWindow(void);
static bool8 IsSelectedKeyboardIndexInvalid(void);
static void SetKeyboardCursorToLastColumn(void);
static u8 GetLastAlphabetColumn(u8);
static void ReduceToValidWordSelectColumn(void);
static bool8 IsSelectedWordIndexInvalid(void);
static int DidPlayerInputMysteryGiftPhrase(void);
static u16 DidPlayerInputABerryMasterWifePhrase(void);
static bool8 InitEasyChatScreenControl_(void);
static void LoadEasyChatPalettes(void);
static void InitEasyChatBgs(void);
static void AdjustBgTilemapForFooter(void);
static void BufferFrameTilemap(u16 *);
static void AddPhraseWindow(void);
static void AddMainScreenButtonWindow(void);
static void PrintTitle(void);
static void PrintInitialInstructions(void);
static void PrintCurrentPhrase(void);
static void DrawLowerWindow(void);
static void LoadEasyChatGfx(void);
static void CreateMainCursorSprite(void);
static void SpriteCB_Cursor(struct Sprite *);
static void SetWindowDimensions(u8, u8, u8, u8);
static void CreateScrollIndicatorSprites(void);
static void CreateStartSelectButtonSprites(void);
static void TryAddInterviewObjectEvents(void);
static bool8 ReprintPhrase(void);
static bool8 UpdateMainCursor(void);
static bool8 UpdateMainCursorOnButtons(void);
static bool8 ShowConfirmDeleteAllPrompt(void);
static bool8 ShowConfirmExitPrompt(void);
static bool8 ShowConfirmPrompt(void);
static bool8 ClosePrompt(void);
static bool8 ClosePromptAfterDeleteAll(void);
static bool8 OpenKeyboard(void);
static bool8 CloseKeyboard(void);
static bool8 OpenWordSelect(void);
static bool8 CloseWordSelect(void);
static bool8 ShowConfirmLyricsPrompt(void);
static bool8 ReturnToKeyboard(void);
static bool8 UpdateKeyboardCursor(void);
static bool8 GroupNamesScrollDown(void);
static bool8 GroupNamesScrollUp(void);
static bool8 UpdateWordSelectCursor(void);
static bool8 WordSelectScrollUp(void);
static bool8 WordSelectScrollDown(void);
static bool8 WordSelectPageScrollUp(void);
static bool8 WordSelectPageScrollDown(void);
static bool8 SwitchKeyboardMode(void);
static bool8 ShowCreateQuizMsg(void);
static bool8 ShowSelectAnswerMsg(void);
static bool8 ShowSongTooShortMsg(void);
static bool8 ShowCantDeleteLyricsMsg(void);
static bool8 ShowCombineTwoWordsMsg(void);
static bool8 ShowCantExitMsg(void);
static void SetMainCursorPos(u8, u8);
static int GetFooterOptionXOffset(int);
static void StopMainCursorAnim(void);
static void PrintEasyChatStdMessage(u8);
static void CreateEasyChatYesNoMenu(u8);
static void StartMainCursorAnim(void);
static void PrintKeyboardText(void);
static void InitLowerWindowAnim(int);
static void CreateSideWindowSprites(void);
static bool8 ShowSideWindow(void);
static void CreateRectangleCursorSprites(void);
static void SetScrollIndicatorXPos(bool32);
static bool8 UpdateLowerWindowAnim(void);
static void UpdateScrollIndicatorsVisibility(void);
static void DestroyRectangleCursorSprites(void);
static void HideModeWindow(void);
static void HideScrollIndicators(void);
static void SetModeWindowToTransition(void);
static bool8 DestroySideWindowSprites(void);
static bool8 IsModeWindowAnimActive(void);
static void UpdateModeWindowAnim(void);
static void UpdateRectangleCursorPos(void);
static void InitLowerWindowScroll(s16, u8);
static bool8 UpdateLowerWindowScroll(void);
static void ClearWordSelectWindow(void);
static void InitLowerWindowText(u32);
static void CreateWordSelectCursorSprite(void);
static void UpdateStartSelectButtonsVisibility(void);
static void DestroyWordSelectCursorSprite(void);
static void HideStartSelectButtons(void);
static void UpdateWordSelectCursorPos(void);
static void PrintWordSelectNextRowDown(void);
static void PrintWordSelectNextRowUp(void);
static int GetLowerWindowScrollOffset(void);
static void PrintWordSelectRowsPageDown(void);
static void PrintWordSelectRowsPageUp(void);
static void PrintEasyChatTextWithColors(u8, u8, const u8 *, u8, u8, u8, u8, u8, u8);
static void ResetLowerWindowScroll(void);
static void PrintKeyboardGroupNames(void);
static void PrintKeyboardAlphabet(void);
static void PrintInitialWordSelectText(void);
static const u8 *GetEasyChatWordGroupName(u8);
static void PrintWordSelectText(u8, u8);
static void EraseWordSelectRows(u8, u8);
static void DrawLowerWindowFrame(u8);
static void BufferLowerWindowFrame(int, int, int, int);
static void SetRectangleCursorPos_GroupMode(s8, s8);
static void SetRectangleCursorPos_AlphabetMode(s8, s8);
static void SpriteCB_WordSelectCursor(struct Sprite *);
static void SetWordSelectCursorPos(u8, u8);
static bool8 EasyChatIsNationalPokedexEnabled(void);
static u16 GetRandomUnlockedEasyChatPokemon(void);
static void SetUnlockedEasyChatGroups(void);
static void SetUnlockedWordsByAlphabet(void);
static u8 *CopyEasyChatWordPadded(u8 *, u16, u16);
static u8 IsEasyChatWordUnlocked(u16);
static u16 SetSelectedWordGroup_GroupMode(u16);
static u16 SetSelectedWordGroup_AlphabetMode(u16);
static bool8 IsEasyChatIndexAndGroupUnlocked(u16, u8);
static int IsRestrictedWordSpecies(u16);
static void DoQuizAnswerEasyChatScreen(void);
static void DoQuizQuestionEasyChatScreen(void);
static void DoQuizSetAnswerEasyChatScreen(void);
static void DoQuizSetQuestionEasyChatScreen(void);
#define PALTAG_TRIANGLE_CURSOR 0
#define PALTAG_RECTANGLE_CURSOR 1
#define PALTAG_MISC_UI 2
#define PALTAG_3 3
#define GFXTAG_TRIANGLE_CURSOR 0
#define GFXTAG_RECTANGLE_CURSOR 1
#define GFXTAG_SCROLL_INDICATOR 2
#define GFXTAG_START_SELECT_BUTTONS 3
#define GFXTAG_MODE_WINDOW 4
#define GFXTAG_5 5
#define GFXTAG_BUTTON_WINDOW 6
// State values for sEasyChatScreen->inputState
// Control which input handler to use in HandleEasyChatInput
enum {
INPUTSTATE_PHRASE,
INPUTSTATE_MAIN_SCREEN_BUTTONS,
INPUTSTATE_KEYBOARD,
INPUTSTATE_WORD_SELECT,
INPUTSTATE_EXIT_PROMPT,
INPUTSTATE_DELETE_ALL_YES_NO,
INPUTSTATE_CONFIRM_WORDS_YES_NO,
INPUTSTATE_QUIZ_QUESTION,
INPUTSTATE_WAIT_FOR_MSG,
INPUTSTATE_START_CONFIRM_LYRICS,
INPUTSTATE_CONFIRM_LYRICS_YES_NO,
};
// Task states for the 'main' task, Task_EasyChatScreen
enum {
MAINSTATE_FADE_IN,
MAINSTATE_HANDLE_INPUT,
MAINSTATE_RUN_FUNC,
MAINSTATE_TO_QUIZ_LADY,
MAINSTATE_EXIT,
MAINSTATE_WAIT_FADE_IN,
};
// IDs provided to PrintEasyChatStdMessage to print a standard message
enum {
MSG_INSTRUCTIONS,
MSG_CONFIRM_DELETE,
MSG_CONFIRM_EXIT,
MSG_CONFIRM,
MSG_CREATE_QUIZ,
MSG_SELECT_ANSWER,
MSG_SONG_TOO_SHORT,
MSG_CANT_DELETE_LYRICS,
MSG_COMBINE_TWO_WORDS,
MSG_CANT_QUIT,
};
// IDs for supplementary Easy Chat functions
// Returned by the input handler functions, and run
// in the main task (MAINSTATE_RUN_FUNC)
enum {
ECFUNC_NONE,
ECFUNC_REPRINT_PHRASE,
ECFUNC_UPDATE_MAIN_CURSOR,
ECFUNC_UPDATE_MAIN_CURSOR_ON_BUTTONS,
ECFUNC_PROMPT_DELETE_ALL,
ECFUNC_PROMPT_EXIT,
ECFUNC_PROMPT_CONFIRM,
ECFUNC_CLOSE_PROMPT,
ECFUNC_CLOSE_PROMPT_AFTER_DELETE,
ECFUNC_OPEN_KEYBOARD,
ECFUNC_CLOSE_KEYBOARD,
ECFUNC_OPEN_WORD_SELECT,
ECFUNC_CLOSE_WORD_SELECT,
ECFUNC_PROMPT_CONFIRM_LYRICS,
ECFUNC_RETURN_TO_KEYBOARD,
ECFUNC_UPDATE_KEYBOARD_CURSOR,
ECFUNC_GROUP_NAMES_SCROLL_DOWN,
ECFUNC_GROUP_NAMES_SCROLL_UP,
ECFUNC_UPDATE_WORD_SELECT_CURSOR,
ECFUNC_WORD_SELECT_SCROLL_UP,
ECFUNC_WORD_SELECT_SCROLL_DOWN,
ECFUNC_WORD_SELECT_PAGE_UP,
ECFUNC_WORD_SELECT_PAGE_DOWN,
ECFUNC_SWITCH_KEYBOARD_MODE,
ECFUNC_EXIT,
ECFUNC_QUIZ_QUESTION,
ECFUNC_QUIZ_ANSWER,
ECFUNC_SET_QUIZ_QUESTION,
ECFUNC_SET_QUIZ_ANSWER,
ECFUNC_MSG_CREATE_QUIZ,
ECFUNC_MSG_SELECT_ANSWER,
ECFUNC_MSG_SONG_TOO_SHORT,
ECFUNC_MSG_CANT_DELETE_LYRICS,
ECFUNC_MSG_COMBINE_TWO_WORDS,
ECFUNC_MSG_CANT_EXIT,
};
// IDs for InitLowerWindowText
enum {
TEXT_GROUPS,
TEXT_ALPHABET,
TEXT_WORD_SELECT,
};
#define NUM_ALPHABET_ROWS 4
#define NUM_GROUP_NAME_ROWS 4
#define NUM_WORD_SELECT_ROWS 4
#define NUM_BUTTON_ROWS 3
#define NUM_ALPHABET_COLUMNS 7
#define NUM_GROUP_NAME_COLUMNS 2
#define NUM_WORD_SELECT_COLUMNS 2
enum {
FRAMEID_GENERAL_2x2,
FRAMEID_GENERAL_2x3,
FRAMEID_MAIL,
FRAMEID_COMBINE_TWO_WORDS,
FRAMEID_INTERVIEW_SHOW_PERSON,
FRAMEID_INTERVIEW,
FRAMEID_QUIZ_ANSWER,
FRAMEID_QUIZ_QUESTION,
FRAMEID_QUIZ_SET_QUESTION,
};
// IDs for the footer row of buttons on the main screen
enum {
FOOTER_NORMAL,
FOOTER_QUIZ,
FOOTER_ANSWER,
NUM_FOOTER_TYPES
};
enum {
INPUT_RIGHT,
INPUT_LEFT,
INPUT_UP,
INPUT_DOWN,
INPUT_START,
INPUT_SELECT,
};
// Types of animations for the lower window (keyboard/word select), given to InitLowerWindowAnim
enum {
WINANIM_OPEN_KEYBOARD,
WINANIM_CLOSE_KEYBOARD,
WINANIM_OPEN_WORD_SELECT,
WINANIM_CLOSE_WORD_SELECT,
WINANIM_RETURN_TO_KEYBOARD,
WINANIM_KEYBOARD_SWITCH_OUT,
WINANIM_KEYBOARD_SWITCH_IN,
};
// Values for text frame tilemap
#define FRAME_OFFSET_ORANGE 0x1000 // Orange frame, for phrase text
#define FRAME_OFFSET_GREEN 0x4000 // Green frame, for keyboard/word select
#define FRAME_TILE_TRANSPARENT 0x0
#define FRAME_TILE_TOP_L_CORNER 0x1
#define FRAME_TILE_TOP_EDGE 0x2
#define FRAME_TILE_TOP_R_CORNER 0x3
#define FRAME_TILE_L_EDGE 0x5
#define FRAME_TILE_R_EDGE 0x7
#define FRAME_TILE_BOTTOM_L_CORNER 0x9
#define FRAME_TILE_BOTTOM_EDGE 0xA
#define FRAME_TILE_BOTTOM_R_CORNER 0xB
struct
{
u16 funcId;
MainCallback callback;
} static const sQuizLadyEasyChatScreens[] = {
{
.funcId = ECFUNC_QUIZ_ANSWER,
.callback = DoQuizAnswerEasyChatScreen,
},
{
.funcId = ECFUNC_QUIZ_QUESTION,
.callback = DoQuizQuestionEasyChatScreen,
},
{
.funcId = ECFUNC_SET_QUIZ_ANSWER,
.callback = DoQuizSetAnswerEasyChatScreen,
},
{
.funcId = ECFUNC_SET_QUIZ_QUESTION,
.callback = DoQuizSetQuestionEasyChatScreen,
},
};
static const struct EasyChatScreenTemplate sEasyChatScreenTemplates[] = {
{
.type = EASY_CHAT_TYPE_PROFILE,
.numColumns = 2,
.numRows = 2,
.frameId = FRAMEID_GENERAL_2x2,
.fourFooterOptions = FALSE,
.titleText = gText_Profile,
.instructionsText1 = gText_CombineFourWordsOrPhrases,
.instructionsText2 = gText_AndMakeYourProfile,
.confirmText1 = gText_YourProfile,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_BATTLE_START,
.numColumns = 2,
.numRows = 3,
.frameId = FRAMEID_GENERAL_2x3,
.fourFooterOptions = FALSE,
.titleText = gText_AtTheBattlesStart,
.instructionsText1 = gText_CombineSixWordsOrPhrases,
.instructionsText2 = gText_AndMakeAMessage,
.confirmText1 = gText_YourFeelingAtTheBattlesStart,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_BATTLE_WON,
.numColumns = 2,
.numRows = 3,
.frameId = FRAMEID_GENERAL_2x3,
.fourFooterOptions = FALSE,
.titleText = gText_UponWinningABattle,
.instructionsText1 = gText_CombineSixWordsOrPhrases,
.instructionsText2 = gText_AndMakeAMessage,
.confirmText1 = gText_WhatYouSayIfYouWin,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_BATTLE_LOST,
.numColumns = 2,
.numRows = 3,
.frameId = FRAMEID_GENERAL_2x3,
.fourFooterOptions = FALSE,
.titleText = gText_UponLosingABattle,
.instructionsText1 = gText_CombineSixWordsOrPhrases,
.instructionsText2 = gText_AndMakeAMessage,
.confirmText1 = gText_WhatYouSayIfYouLose,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_MAIL,
.numColumns = 2,
.numRows = 5,
.frameId = FRAMEID_MAIL,
.fourFooterOptions = FALSE,
.titleText = NULL,
.instructionsText1 = gText_CombineNineWordsOrPhrases,
.instructionsText2 = gText_AndMakeAMessage2,
.confirmText1 = gText_TheMailMessage,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_INTERVIEW,
.numColumns = 2,
.numRows = 2,
.frameId = FRAMEID_INTERVIEW,
.fourFooterOptions = FALSE,
.titleText = gText_Interview,
.instructionsText1 = gText_CombineFourWordsOrPhrases,
.instructionsText2 = gText_LetsReplyToTheInterview,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_BARD_SONG,
.numColumns = 2,
.numRows = 3,
.frameId = FRAMEID_GENERAL_2x3,
.fourFooterOptions = FALSE,
.titleText = gText_TheBardsSong,
.instructionsText1 = gText_ChangeJustOneWordOrPhrase,
.instructionsText2 = gText_AndImproveTheBardsSong,
.confirmText1 = gText_TheBardsSong2,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_FAN_CLUB,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_INTERVIEW_SHOW_PERSON,
.fourFooterOptions = FALSE,
.titleText = gText_Interview,
.instructionsText1 = gText_FindWordsThatDescribeYour,
.instructionsText2 = gText_FeelingsRightNow,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_TRENDY_PHRASE,
.numColumns = 2,
.numRows = 1,
.frameId = FRAMEID_COMBINE_TWO_WORDS,
.fourFooterOptions = FALSE,
.titleText = gText_WhatsHipAndHappening,
.instructionsText1 = gText_CombineTwoWordsOrPhrases,
.instructionsText2 = gText_AndMakeATrendySaying,
.confirmText1 = gText_TheTrendySaying,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_QUIZ_QUESTION,
.numColumns = 2,
.numRows = 5,
.frameId = FRAMEID_QUIZ_QUESTION,
.fourFooterOptions = TRUE,
.titleText = NULL,
.instructionsText1 = gText_AfterYouHaveReadTheQuiz,
.instructionsText2 = gText_QuestionPressTheAButton,
.confirmText1 = NULL,
.confirmText2 = NULL,
},
{
.type = EASY_CHAT_TYPE_QUIZ_ANSWER,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_QUIZ_ANSWER,
.fourFooterOptions = TRUE,
.titleText = gText_TheQuizAnswerIs,
.instructionsText1 = gText_OutOfTheListedChoices,
.instructionsText2 = gText_SelectTheAnswerToTheQuiz,
.confirmText1 = gText_TheAnswerColon,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_QUIZ_SET_QUESTION,
.numColumns = 2,
.numRows = 5,
.frameId = FRAMEID_QUIZ_SET_QUESTION,
.fourFooterOptions = TRUE,
.titleText = NULL,
.instructionsText1 = gText_CombineNineWordsOrPhrases,
.instructionsText2 = gText_AndCreateAQuiz,
.confirmText1 = gText_IsThisQuizOK,
.confirmText2 = NULL,
},
{
.type = EASY_CHAT_TYPE_QUIZ_SET_ANSWER,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_QUIZ_ANSWER,
.fourFooterOptions = TRUE,
.titleText = gText_TheQuizAnswerIs,
.instructionsText1 = gText_PickAWordOrPhraseAnd,
.instructionsText2 = gText_SetTheQuizAnswer,
.confirmText1 = gText_IsThisQuizOK,
.confirmText2 = NULL,
},
{
.type = EASY_CHAT_TYPE_BARD_SONG,
.numColumns = 2,
.numRows = 3,
.frameId = FRAMEID_GENERAL_2x3,
.fourFooterOptions = FALSE,
.titleText = gText_TheBardsSong,
.instructionsText1 = gText_ChangeJustOneWordOrPhrase,
.instructionsText2 = gText_AndImproveTheBardsSong,
.confirmText1 = gText_TheBardsSong2,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_APPRENTICE,
.numColumns = 2,
.numRows = 3,
.frameId = FRAMEID_GENERAL_2x3,
.fourFooterOptions = FALSE,
.titleText = gText_ApprenticesPhrase,
.instructionsText1 = gText_FindWordsWhichFit,
.instructionsText2 = gText_TheTrainersImage,
.confirmText1 = gText_ApprenticePhrase,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_GOOD_SAYING,
.numColumns = 2,
.numRows = 1,
.frameId = FRAMEID_COMBINE_TWO_WORDS,
.fourFooterOptions = FALSE,
.titleText = gText_GoodSaying,
.instructionsText1 = gText_CombineTwoWordsOrPhrases2,
.instructionsText2 = gText_ToTeachHerAGoodSaying,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_GABBY_AND_TY,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_INTERVIEW_SHOW_PERSON,
.fourFooterOptions = FALSE,
.titleText = gText_Interview,
.instructionsText1 = gText_FindWordsThatDescribeYour,
.instructionsText2 = gText_FeelingsRightNow,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_BATTLE_TOWER_INTERVIEW,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_INTERVIEW_SHOW_PERSON,
.fourFooterOptions = FALSE,
.titleText = gText_Interview,
.instructionsText1 = gText_FindWordsThatDescribeYour,
.instructionsText2 = gText_FeelingsRightNow,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_CONTEST_INTERVIEW,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_INTERVIEW_SHOW_PERSON,
.fourFooterOptions = FALSE,
.titleText = gText_Interview,
.instructionsText1 = gText_FindWordsThatDescribeYour,
.instructionsText2 = gText_FeelingsRightNow,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_FAN_QUESTION,
.numColumns = 1,
.numRows = 1,
.frameId = FRAMEID_INTERVIEW_SHOW_PERSON,
.fourFooterOptions = FALSE,
.titleText = gText_FansQuestion,
.instructionsText1 = gText_FindWordsWhichFit,
.instructionsText2 = gText_TheTrainersImage,
.confirmText1 = gText_TheImage,
.confirmText2 = gText_IsAsShownOkay,
},
{
.type = EASY_CHAT_TYPE_QUESTIONNAIRE,
.numColumns = 2,
.numRows = 2,
.frameId = FRAMEID_GENERAL_2x2,
.fourFooterOptions = FALSE,
.titleText = gText_Questionnaire,
.instructionsText1 = gText_CombineFourWordsOrPhrases,
.instructionsText2 = gText_AndFillOutTheQuestionnaire,
.confirmText1 = gText_TheAnswer,
.confirmText2 = gText_IsAsShownOkay,
},
};
// IDs are used indirectly as indexes into gEasyChatWordsByLetterPointers
// 0 is 'Others', 1-26 are the letters A-Z
// This array maps the group IDs to the alphabet keyboard
static const u8 sAlphabetGroupIdMap[NUM_ALPHABET_ROWS][NUM_ALPHABET_COLUMNS] = {
{ 1, 2, 3, 4, 5, 6, 0},
{ 7, 8, 9, 10, 11, 12, 0},
{13, 14, 15, 16, 17, 18, 19},
{20, 21, 22, 23, 24, 25, 26},
};
static const u16 sMysteryGiftPhrase[NUM_QUESTIONNAIRE_WORDS] = {
EC_WORD_LINK,
EC_WORD_TOGETHER,
EC_WORD_WITH,
EC_WORD_ALL,
};
static const u16 sBerryMasterWifePhrases[][2] = {
[PHRASE_GREAT_BATTLE - 1] = {EC_WORD_GREAT, EC_WORD_BATTLE},
[PHRASE_CHALLENGE_CONTEST - 1] = {EC_WORD_CHALLENGE, EC_WORD_CONTEST},
[PHRASE_OVERWHELMING_LATIAS - 1] = {EC_WORD_OVERWHELMING, EC_POKEMON(LATIAS)},
[PHRASE_COOL_LATIOS - 1] = {EC_WORD_COOL, EC_POKEMON(LATIOS)},
[PHRASE_SUPER_HUSTLE - 1] = {EC_WORD_SUPER, EC_WORD_HUSTLE},
};
static const u16 sTriangleCursor_Pal[] = INCBIN_U16("graphics/easy_chat/triangle_cursor.gbapal");
static const u32 sTriangleCursor_Gfx[] = INCBIN_U32("graphics/easy_chat/triangle_cursor.4bpp");
static const u32 sScrollIndicator_Gfx[] = INCBIN_U32("graphics/easy_chat/scroll_indicator.4bpp");
static const u32 sStartSelectButtons_Gfx[] = INCBIN_U32("graphics/easy_chat/start_select_buttons.4bpp");
static const u16 sUnknown_085979C0[] = INCBIN_U16("graphics/misc/interview_frame.gbapal");
static const u32 sUnknown_085979E0[] = INCBIN_U32("graphics/misc/interview_frame.4bpp.lz");
static const u16 sTextInputFrameOrange_Pal[] = INCBIN_U16("graphics/easy_chat/text_input_frame_orange.gbapal");
static const u16 sTextInputFrameGreen_Pal[] = INCBIN_U16("graphics/easy_chat/text_input_frame_green.gbapal");
static const u32 sTextInputFrame_Gfx[] = INCBIN_U32("graphics/easy_chat/text_input_frame.4bpp.lz");
static const u16 sUnknown_08597C1C[] = INCBIN_U16("graphics/misc/8597C1C.gbapal");
static const u16 sUnknown_08597C24[] = INCBIN_U16("graphics/misc/8597C24.gbapal");
static const struct EasyChatPhraseFrameDimensions sPhraseFrameDimensions[] = {
[FRAMEID_GENERAL_2x2] = {
.left = 3,
.top = 4,
.width = 24,
.height = 4,
.footerId = FOOTER_NORMAL,
},
[FRAMEID_GENERAL_2x3] = {
.left = 3,
.top = 3,
.width = 24,
.height = 6,
.footerId = FOOTER_NORMAL,
},
[FRAMEID_MAIL] = {
.left = 3,
.top = 0,
.width = 24,
.height = 10,
.footerId = FOOTER_NORMAL,
},
[FRAMEID_COMBINE_TWO_WORDS] = {
.left = 3,
.top = 5,
.width = 24,
.height = 2,
.footerId = FOOTER_NORMAL,
},
[FRAMEID_INTERVIEW_SHOW_PERSON] = {
.left = 16,
.top = 5,
.width = 12,
.height = 2,
.footerId = FOOTER_NORMAL,
},
[FRAMEID_INTERVIEW] = {
.left = 3,
.top = 4,
.width = 24,
.height = 4,
.footerId = FOOTER_NORMAL,
},
[FRAMEID_QUIZ_ANSWER] = {
.left = 9,
.top = 4,
.width = 12,
.height = 2,
.footerId = FOOTER_QUIZ,
},
[FRAMEID_QUIZ_QUESTION] = {
.left = 5,
.top = 3,
.width = 20,
.height = 10,
.footerId = NUM_FOOTER_TYPES,
},
[FRAMEID_QUIZ_SET_QUESTION] = {
.left = 3,
.top = 0,
.width = 24,
.height = 10,
.footerId = FOOTER_ANSWER,
},
};
static const struct BgTemplate sEasyChatBgTemplates[] = {
{
.bg = 0,
.charBaseIndex = 0,
.mapBaseIndex = 28,
.screenSize = 0,
.paletteMode = 0,
.priority = 0,
.baseTile = 0,
},
{
.bg = 1,
.charBaseIndex = 3,
.mapBaseIndex = 29,
.screenSize = 0,
.paletteMode = 0,
.priority = 1,
.baseTile = 0,
},
{
.bg = 2,
.charBaseIndex = 0,
.mapBaseIndex = 30,
.screenSize = 0,
.paletteMode = 0,
.priority = 2,
.baseTile = 0x80,
},
{
.bg = 3,
.charBaseIndex = 2,
.mapBaseIndex = 31,
.screenSize = 0,
.paletteMode = 0,
.priority = 3,
.baseTile = 0,
},
};
static const struct WindowTemplate sEasyChatWindowTemplates[] = {
{
.bg = 1,
.tilemapLeft = 6,
.tilemapTop = 0,
.width = 18,
.height = 2,
.paletteNum = 10,
.baseBlock = 0x10,
},
{
.bg = 0,
.tilemapLeft = 3,
.tilemapTop = 15,
.width = 24,
.height = 4,
.paletteNum = 15,
.baseBlock = 0xA,
},
{
.bg = 2,
.tilemapLeft = 1,
.tilemapTop = 0,
.width = 28,
.height = 32,
.paletteNum = 3,
.baseBlock = 0,
},
DUMMY_WIN_TEMPLATE,
};
static const struct WindowTemplate sEasyChatYesNoWindowTemplate = {
.bg = 0,
.tilemapLeft = 22,
.tilemapTop = 9,
.width = 5,
.height = 4,
.paletteNum = 15,
.baseBlock = 0x6A,
};
static const u8 sText_Clear17[] = _("{CLEAR 17}");
static const u8 *const sEasyChatKeyboardAlphabet[NUM_ALPHABET_ROWS] =
{
gText_EasyChatKeyboard_ABCDEFothers,
gText_EasyChatKeyboard_GHIJKL,
gText_EasyChatKeyboard_MNOPQRS,
gText_EasyChatKeyboard_TUVWXYZ,
};
static const struct SpriteSheet sSpriteSheets[] = {
{
.data = sTriangleCursor_Gfx,
.size = 0x20,
.tag = GFXTAG_TRIANGLE_CURSOR
},
{
.data = sScrollIndicator_Gfx,
.size = 0x100,
.tag = GFXTAG_SCROLL_INDICATOR
},
{
.data = sStartSelectButtons_Gfx,
.size = 0x100,
.tag = GFXTAG_START_SELECT_BUTTONS
},
{0}
};
static const struct SpritePalette sSpritePalettes[] = {
{
.data = sTriangleCursor_Pal,
.tag = PALTAG_TRIANGLE_CURSOR,
},
{
.data = gEasyChatRectangleCursor_Pal,
.tag = PALTAG_RECTANGLE_CURSOR,
},
{
.data = gEasyChatButtonWindow_Pal,
.tag = PALTAG_MISC_UI, // The palette is generated from the button window but used for various parts of the UI
},
{
.data = sUnknown_085979C0,
.tag = PALTAG_3,
},
{0}
};
static const struct CompressedSpriteSheet sCompressedSpriteSheets[] = {
{
.data = sUnknown_085979E0,
.size = 0x800,
.tag = GFXTAG_5,
},
{
.data = gEasyChatRectangleCursor_Gfx,
.size = 0x1000,
.tag = GFXTAG_RECTANGLE_CURSOR,
},
{
.data = gEasyChatButtonWindow_Gfx,
.size = 0x800,
.tag = GFXTAG_BUTTON_WINDOW,
},
{
.data = gEasyChatMode_Gfx,
.size = 0x1000,
.tag = GFXTAG_MODE_WINDOW,
},
};
static const u8 sAlphabetKeyboardColumnOffsets[NUM_ALPHABET_COLUMNS] = {0, 12, 24, 56, 68, 80, 92};
static const struct OamData sOamData_TriangleCursor = {
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(8x8),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(8x8),
.tileNum = 0,
.priority = 3,
.paletteNum = 0,
.affineParam = 0,
};
static const struct SpriteTemplate sSpriteTemplate_TriangleCursor = {
.tileTag = PALTAG_TRIANGLE_CURSOR,
.paletteTag = GFXTAG_TRIANGLE_CURSOR,
.oam = &sOamData_TriangleCursor,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCB_Cursor,
};
static const struct OamData sOamData_RectangleCursor = {
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x32),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(64x32),
.tileNum = 0,
.priority = 1,
.paletteNum = 0,
.affineParam = 0,
};
static const union AnimCmd sAnim_RectangleCursor_OnGroup[] = {
ANIMCMD_FRAME(0, 0),
ANIMCMD_END,
};
static const union AnimCmd sAnim_RectangleCursor_OnButton[] = {
ANIMCMD_FRAME(32, 0),
ANIMCMD_END,
};
static const union AnimCmd sAnim_RectangleCursor_OnOthers[] = {
ANIMCMD_FRAME(64, 0),
ANIMCMD_END,
};
static const union AnimCmd sAnim_RectangleCursor_OnLetter[] = {
ANIMCMD_FRAME(96, 0),
ANIMCMD_END,
};
// Each anim changes the width of the rectangle cursor to fit what it should be selecting
enum {
RECTCURSOR_ANIM_ON_GROUP,
RECTCURSOR_ANIM_ON_BUTTON,
RECTCURSOR_ANIM_ON_OTHERS,
RECTCURSOR_ANIM_ON_LETTER,
};
static const union AnimCmd *const sAnims_RectangleCursor[] = {
[RECTCURSOR_ANIM_ON_GROUP] = sAnim_RectangleCursor_OnGroup,
[RECTCURSOR_ANIM_ON_BUTTON] = sAnim_RectangleCursor_OnButton,
[RECTCURSOR_ANIM_ON_OTHERS] = sAnim_RectangleCursor_OnOthers,
[RECTCURSOR_ANIM_ON_LETTER] = sAnim_RectangleCursor_OnLetter,
};
static const struct SpriteTemplate sSpriteTemplate_RectangleCursor = {
.tileTag = GFXTAG_RECTANGLE_CURSOR,
.paletteTag = PALTAG_RECTANGLE_CURSOR,
.oam = &sOamData_RectangleCursor,
.anims = sAnims_RectangleCursor,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCB_Cursor,
};
static const struct OamData sOamData_ModeWindow = {
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x32),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(64x32),
.tileNum = 0,
.priority = 1,
.paletteNum = 0,
.affineParam = 0,
};
static const union AnimCmd sAnim_ModeWindow_Hidden[] = {
ANIMCMD_FRAME(96, 0),
ANIMCMD_END,
};
static const union AnimCmd sAnim_ModeWindow_ToGroup[] = {
ANIMCMD_FRAME(64, 4), // Transition frame
ANIMCMD_FRAME(32, 4), // 'Group' frame
ANIMCMD_END,
};
static const union AnimCmd sAnim_ModeWindow_ToAlphabet[] = {
ANIMCMD_FRAME(64, 4), // Transition frame
ANIMCMD_FRAME(0, 4), // 'A-Z' frame
ANIMCMD_END,
};
static const union AnimCmd sAnim_ModeWindow_ToHidden[] = {
ANIMCMD_FRAME(64, 4), // Transition frame
ANIMCMD_FRAME(96, 0), // Hidden frame
ANIMCMD_END,
};
static const union AnimCmd sAnim_ModeWindow_Transition[] = {
ANIMCMD_FRAME(64, 4),
ANIMCMD_END,
};
enum {
MODEWINDOW_ANIM_HIDDEN,
MODEWINDOW_ANIM_TO_GROUP,
MODEWINDOW_ANIM_TO_ALPHABET,
MODEWINDOW_ANIM_TO_HIDDEN,
MODEWINDOW_ANIM_TRANSITION,
};
static const union AnimCmd *const sAnims_ModeWindow[] = {
[MODEWINDOW_ANIM_HIDDEN] = sAnim_ModeWindow_Hidden,
[MODEWINDOW_ANIM_TO_GROUP] = sAnim_ModeWindow_ToGroup,
[MODEWINDOW_ANIM_TO_ALPHABET] = sAnim_ModeWindow_ToAlphabet,
[MODEWINDOW_ANIM_TO_HIDDEN] = sAnim_ModeWindow_ToHidden,
[MODEWINDOW_ANIM_TRANSITION] = sAnim_ModeWindow_Transition,
};
static const struct SpriteTemplate sSpriteTemplate_ModeWindow = {
.tileTag = GFXTAG_MODE_WINDOW,
.paletteTag = PALTAG_MISC_UI,
.oam = &sOamData_ModeWindow,
.anims = sAnims_ModeWindow,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
};
static const struct OamData sOamData_ButtonWindow = {
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(64x64),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(64x64),
.tileNum = 0,
.priority = 3,
.paletteNum = 0,
.affineParam = 0,
};
static const struct SpriteTemplate sSpriteTemplate_ButtonWindow = {
.tileTag = GFXTAG_BUTTON_WINDOW,
.paletteTag = PALTAG_MISC_UI,
.oam = &sOamData_ButtonWindow,
.anims = gDummySpriteAnimTable,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
};
static const struct OamData sOamData_StartSelectButton = {
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(32x8),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(32x8),
.tileNum = 0,
.priority = 1,
.paletteNum = 0,
.affineParam = 0,
};
static const struct OamData sOamData_ScrollIndicator = {
.y = 0,
.affineMode = ST_OAM_AFFINE_OFF,
.objMode = ST_OAM_OBJ_NORMAL,
.mosaic = 0,
.bpp = ST_OAM_4BPP,
.shape = SPRITE_SHAPE(16x16),
.x = 0,
.matrixNum = 0,
.size = SPRITE_SIZE(16x16),
.tileNum = 0,
.priority = 1,
.paletteNum = 0,
.affineParam = 0,
};
static const union AnimCmd sAnim_Frame0[] = {
ANIMCMD_FRAME(0, 0),
ANIMCMD_END,
};
static const union AnimCmd sAnim_Frame1[] = {
ANIMCMD_FRAME(4, 0),
ANIMCMD_END,
};
// Frame0 is Start button, Frame1 is Select button, both are identical for the scroll indicators
static const union AnimCmd *const sAnims_TwoFrame[] = {
sAnim_Frame0,
sAnim_Frame1,
};
static const struct SpriteTemplate sSpriteTemplate_StartSelectButton = {
.tileTag = GFXTAG_START_SELECT_BUTTONS,
.paletteTag = PALTAG_MISC_UI,
.oam = &sOamData_StartSelectButton,
.anims = sAnims_TwoFrame,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
};
static const struct SpriteTemplate sSpriteTemplate_ScrollIndicator = {
.tileTag = GFXTAG_SCROLL_INDICATOR,
.paletteTag = PALTAG_MISC_UI,
.oam = &sOamData_ScrollIndicator,
.anims = sAnims_TwoFrame,
.images = NULL,
.affineAnims = gDummySpriteAffineAnimTable,
.callback = SpriteCallbackDummy,
};
static const u8 sFooterOptionXOffsets[NUM_FOOTER_TYPES][4] = {
[FOOTER_NORMAL] = {16, 111, 196, 0},
[FOOTER_QUIZ] = {16, 78, 130, 160},
[FOOTER_ANSWER] = {16, 80, 134, 170},
};
static const u8 *const sFooterTextOptions[NUM_FOOTER_TYPES][4] = {
[FOOTER_NORMAL] = {gText_DelAll, gText_Cancel5, gText_Ok2, NULL},
[FOOTER_QUIZ] = {gText_DelAll, gText_Cancel5, gText_Ok2, gText_Quiz},
[FOOTER_ANSWER] = {gText_DelAll, gText_Cancel5, gText_Ok2, gText_Answer},
};
#include "data/easy_chat/easy_chat_groups.h"
#include "data/easy_chat/easy_chat_words_by_letter.h"
static const u8 *const sEasyChatGroupNamePointers[EC_NUM_GROUPS] = {
[EC_GROUP_POKEMON] = gEasyChatGroupName_Pokemon,
[EC_GROUP_TRAINER] = gEasyChatGroupName_Trainer,
[EC_GROUP_STATUS] = gEasyChatGroupName_Status,
[EC_GROUP_BATTLE] = gEasyChatGroupName_Battle,
[EC_GROUP_GREETINGS] = gEasyChatGroupName_Greetings,
[EC_GROUP_PEOPLE] = gEasyChatGroupName_People,
[EC_GROUP_VOICES] = gEasyChatGroupName_Voices,
[EC_GROUP_SPEECH] = gEasyChatGroupName_Speech,
[EC_GROUP_ENDINGS] = gEasyChatGroupName_Endings,
[EC_GROUP_FEELINGS] = gEasyChatGroupName_Feelings,
[EC_GROUP_CONDITIONS] = gEasyChatGroupName_Conditions,
[EC_GROUP_ACTIONS] = gEasyChatGroupName_Actions,
[EC_GROUP_LIFESTYLE] = gEasyChatGroupName_Lifestyle,
[EC_GROUP_HOBBIES] = gEasyChatGroupName_Hobbies,
[EC_GROUP_TIME] = gEasyChatGroupName_Time,
[EC_GROUP_MISC] = gEasyChatGroupName_Misc,
[EC_GROUP_ADJECTIVES] = gEasyChatGroupName_Adjectives,
[EC_GROUP_EVENTS] = gEasyChatGroupName_Events,
[EC_GROUP_MOVE_1] = gEasyChatGroupName_Move1,
[EC_GROUP_MOVE_2] = gEasyChatGroupName_Move2,
[EC_GROUP_TRENDY_SAYING] = gEasyChatGroupName_TrendySaying,
[EC_GROUP_POKEMON_NATIONAL] = gEasyChatGroupName_Pokemon2,
};
static const u16 sDefaultProfileWords[EASY_CHAT_BATTLE_WORDS_COUNT - 2] = {
EC_WORD_I_AM,
EC_WORD_A,
EC_WORD_POKEMON,
EC_WORD_FRIEND,
};
static const u16 sDefaultBattleStartWords[EASY_CHAT_BATTLE_WORDS_COUNT] = {
EC_WORD_ARE,
EC_WORD_YOU,
EC_WORD_READY,
EC_WORD_QUES,
EC_WORD_HERE_I_COME,
EC_WORD_EXCL,
};
static const u16 sDefaultBattleWonWords[EASY_CHAT_BATTLE_WORDS_COUNT] = {
EC_WORD_YAY,
EC_WORD_YAY,
EC_WORD_EXCL_EXCL,
EC_WORD_I_VE,
EC_WORD_WON,
EC_WORD_EXCL_EXCL,
};
static const u16 sDefaultBattleLostWords[EASY_CHAT_BATTLE_WORDS_COUNT] = {
EC_WORD_TOO,
EC_WORD_BAD,
EC_WORD_ELLIPSIS,
EC_WORD_WE,
EC_WORD_LOST,
EC_WORD_ELLIPSIS,
};
static const u16 sRestrictedWordSpecies[] = {
SPECIES_DEOXYS,
};
// In addition to the task defines below, these two elements
// have their indexes used explicitly because they are 4-byte
// pointers, and occupy the next data element as well.
// SetWordTaskArg/GetWordTaskArg use these defines to
// read the pointer from the two elements
#define TASKIDX_WORDS 2
#define TASKIDX_EXIT_CALLBACK 4
#define tState data[0]
#define tType data[1]
#define tWords data[TASKIDX_WORDS] // Occupies 2 and 3
#define tExitCallback data[TASKIDX_EXIT_CALLBACK] // Occupies 4 and 5
#define tFuncId data[6]
#define tPersonType data[7]
void DoEasyChatScreen(u8 type, u16 *words, MainCallback exitCallback, u8 displayedPersonType)
{
u8 taskId;
ResetTasks();
taskId = CreateTask(Task_InitEasyChatScreen, 0);
gTasks[taskId].tType = type;
gTasks[taskId].tPersonType = displayedPersonType;
SetWordTaskArg(taskId, TASKIDX_WORDS, (u32)words);
SetWordTaskArg(taskId, TASKIDX_EXIT_CALLBACK, (u32)exitCallback);
SetMainCallback2(CB2_EasyChatScreen);
}
static void CB2_EasyChatScreen(void)
{
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
static void VBlankCB_EasyChatScreen(void)
{
TransferPlttBuffer();
LoadOam();
ProcessSpriteCopyRequests();
}
static void StartEasyChatScreen(u8 taskId, TaskFunc taskFunc)
{
gTasks[taskId].func = taskFunc;
gTasks[taskId].tState = MAINSTATE_FADE_IN;
}
static void Task_InitEasyChatScreen(u8 taskId)
{
if (!IsUpdateLinkStateCBActive())
{
while (InitEasyChatScreen(taskId));
}
else
{
if (InitEasyChatScreen(taskId) == TRUE)
return;
}
StartEasyChatScreen(taskId, Task_EasyChatScreen);
}
// After loading, this is the 'main' Easy Chat task
static void Task_EasyChatScreen(u8 taskId)
{
u16 funcId;
s16 *data;
data = gTasks[taskId].data;
switch (tState)
{
case MAINSTATE_FADE_IN:
SetVBlankCallback(VBlankCB_EasyChatScreen);
BlendPalettes(PALETTES_ALL, 16, 0);
BeginNormalPaletteFade(PALETTES_ALL, -1, 16, 0, RGB_BLACK);
tState = MAINSTATE_WAIT_FADE_IN;
break;
case MAINSTATE_HANDLE_INPUT:
funcId = HandleEasyChatInput();
if (IsFuncIdForQuizLadyScreen(funcId))
{
// Fade to Quiz Lady screen
BeginNormalPaletteFade(PALETTES_ALL, -2, 0, 16, RGB_BLACK);
tState = MAINSTATE_TO_QUIZ_LADY;
tFuncId = funcId;
}
else if (funcId == ECFUNC_EXIT)
{
// Fade and exit Easy Chat
BeginNormalPaletteFade(PALETTES_ALL, -1, 0, 16, RGB_BLACK);
tState = MAINSTATE_EXIT;
}
else if (funcId != ECFUNC_NONE)
{
PlaySE(SE_SELECT);
StartEasyChatFunction(funcId);
tState++; // MAINSTATE_RUN_FUNC
}
break;
case MAINSTATE_RUN_FUNC:
if (!RunEasyChatFunction())
tState = MAINSTATE_HANDLE_INPUT;
break;
case MAINSTATE_TO_QUIZ_LADY:
if (!gPaletteFade.active)
EnterQuizLadyScreen(tFuncId);
break;
case MAINSTATE_EXIT:
if (!gPaletteFade.active)
ExitEasyChatScreen((MainCallback)GetWordTaskArg(taskId, TASKIDX_EXIT_CALLBACK));
break;
case MAINSTATE_WAIT_FADE_IN:
if (!gPaletteFade.active)
tState = MAINSTATE_HANDLE_INPUT;
break;
}
}
// Returns TRUE if still initializing, FALSE when finished
// If an allocation fails it will switch to the exit callback
static bool8 InitEasyChatScreen(u8 taskId)
{
s16 *data;
data = gTasks[taskId].data;
switch (tState)
{
case 0:
SetVBlankCallback(NULL);
ResetSpriteData();
FreeAllSpritePalettes();
ResetPaletteFade();
break;
case 1:
if (!InitEasyChatScreenWordData())
{
// Alloc failed, exit
ExitEasyChatScreen((MainCallback)GetWordTaskArg(taskId, TASKIDX_EXIT_CALLBACK));
}
break;
case 2:
if (!InitEasyChatScreenStruct(tType, (u16 *)GetWordTaskArg(taskId, TASKIDX_WORDS), tPersonType))
{
// Alloc failed, exit
ExitEasyChatScreen((MainCallback)GetWordTaskArg(taskId, TASKIDX_EXIT_CALLBACK));
}
break;
case 3:
if (!InitEasyChatScreenControl())
{
// Alloc failed, exit
ExitEasyChatScreen((MainCallback)GetWordTaskArg(taskId, TASKIDX_EXIT_CALLBACK));
}
break;
case 4:
if (LoadEasyChatScreen())
{
return TRUE;
}
break;
default:
return FALSE;
}
tState++;
return TRUE;
}
static void ExitEasyChatScreen(MainCallback callback)
{
FreeEasyChatScreenControl();
FreeEasyChatScreenStruct();
FreeEasyChatScreenWordData();
FreeAllWindowBuffers();
SetMainCallback2(callback);
}
void ShowEasyChatScreen(void)
{
int i;
u16 *words;
struct MauvilleManBard *bard;
u8 displayedPersonType = EASY_CHAT_PERSON_DISPLAY_NONE;
switch (gSpecialVar_0x8004)
{
case EASY_CHAT_TYPE_PROFILE:
words = gSaveBlock1Ptr->easyChatProfile;
break;
case EASY_CHAT_TYPE_BATTLE_START:
words = gSaveBlock1Ptr->easyChatBattleStart;
break;
case EASY_CHAT_TYPE_BATTLE_WON:
words = gSaveBlock1Ptr->easyChatBattleWon;
break;
case EASY_CHAT_TYPE_BATTLE_LOST:
words = gSaveBlock1Ptr->easyChatBattleLost;
break;
case EASY_CHAT_TYPE_MAIL:
words = gSaveBlock1Ptr->mail[gSpecialVar_0x8005].words;
break;
case EASY_CHAT_TYPE_BARD_SONG:
bard = &gSaveBlock1Ptr->oldMan.bard;
for (i = 0; i < BARD_SONG_LENGTH; i ++)
bard->temporaryLyrics[i] = bard->songLyrics[i];
words = bard->temporaryLyrics;
break;
case EASY_CHAT_TYPE_INTERVIEW:
words = gSaveBlock1Ptr->tvShows[gSpecialVar_0x8005].bravoTrainer.words;
displayedPersonType = gSpecialVar_0x8006;
break;
case EASY_CHAT_TYPE_FAN_CLUB:
words = &gSaveBlock1Ptr->tvShows[gSpecialVar_0x8005].fanclubOpinions.words[gSpecialVar_0x8006];
displayedPersonType = EASY_CHAT_PERSON_REPORTER_FEMALE;
break;
case EASY_CHAT_TYPE_UNK_8:
words = gSaveBlock1Ptr->tvShows[gSpecialVar_0x8005].unkShow04.words;
displayedPersonType = EASY_CHAT_PERSON_REPORTER_MALE;
break;
case EASY_CHAT_TYPE_TRENDY_PHRASE:
words = (u16 *)gStringVar3;
words[0] = gSaveBlock1Ptr->easyChatPairs[0].words[0];
words[1] = gSaveBlock1Ptr->easyChatPairs[0].words[1];
break;
case EASY_CHAT_TYPE_GABBY_AND_TY:
words = gSaveBlock1Ptr->gabbyAndTyData.quote;
*words = EC_EMPTY_WORD;
displayedPersonType = EASY_CHAT_PERSON_REPORTER_FEMALE;
break;
case EASY_CHAT_TYPE_CONTEST_INTERVIEW:
words = &gSaveBlock1Ptr->tvShows[gSpecialVar_0x8005].bravoTrainer.words[gSpecialVar_0x8006];
displayedPersonType = EASY_CHAT_PERSON_REPORTER_MALE;
break;
case EASY_CHAT_TYPE_BATTLE_TOWER_INTERVIEW:
words = gSaveBlock1Ptr->tvShows[gSpecialVar_0x8005].fanclubOpinions.words18;
displayedPersonType = EASY_CHAT_PERSON_REPORTER_FEMALE;
break;
case EASY_CHAT_TYPE_GOOD_SAYING:
words = (u16 *)gStringVar3;
InitializeEasyChatWordArray(words, 2);
break;
case EASY_CHAT_TYPE_FAN_QUESTION:
words = gSaveBlock1Ptr->tvShows[gSpecialVar_0x8005].fanClubSpecial.words;
words[0] = EC_EMPTY_WORD;
displayedPersonType = EASY_CHAT_PERSON_BOY;
break;
case EASY_CHAT_TYPE_QUIZ_ANSWER:
words = &gSaveBlock1Ptr->lilycoveLady.quiz.playerAnswer;
break;
case EASY_CHAT_TYPE_QUIZ_QUESTION:
return;
case EASY_CHAT_TYPE_QUIZ_SET_QUESTION:
words = gSaveBlock1Ptr->lilycoveLady.quiz.question;
break;
case EASY_CHAT_TYPE_QUIZ_SET_ANSWER:
words = &gSaveBlock1Ptr->lilycoveLady.quiz.correctAnswer;
break;
case EASY_CHAT_TYPE_APPRENTICE:
words = gSaveBlock2Ptr->apprentices[0].speechWon;
break;
case EASY_CHAT_TYPE_QUESTIONNAIRE:
words = GetQuestionnaireWordsPtr();
break;
default:
return;
}
CleanupOverworldWindowsAndTilemaps();
DoEasyChatScreen(gSpecialVar_0x8004, words, CB2_ReturnToFieldContinueScript, displayedPersonType);
}
static void CB2_QuizLadyQuestion(void)
{
LilycoveLady *lilycoveLady;
UpdatePaletteFade();
switch (gMain.state)
{
case 0:
FadeScreen(FADE_TO_BLACK, 0);
break;
case 1:
if (!gPaletteFade.active)
{
lilycoveLady = &gSaveBlock1Ptr->lilycoveLady;
lilycoveLady->quiz.playerAnswer = EC_EMPTY_WORD;
CleanupOverworldWindowsAndTilemaps();
DoQuizQuestionEasyChatScreen();
}
return;
}
gMain.state ++;
}
void QuizLadyShowQuizQuestion(void)
{
SetMainCallback2(CB2_QuizLadyQuestion);
}
static int GetQuizLadyScreenByFuncId(u16 funcId)
{
int i;
for (i = 0; i < ARRAY_COUNT(sQuizLadyEasyChatScreens); i ++)
{
if (funcId == sQuizLadyEasyChatScreens[i].funcId)
return i;
}
return -1;
}
static bool32 IsFuncIdForQuizLadyScreen(u16 funcId)
{
return GetQuizLadyScreenByFuncId(funcId) == -1 ? FALSE : TRUE;
}
static void EnterQuizLadyScreen(u16 funcId)
{
int i;
i = GetQuizLadyScreenByFuncId(funcId);
ResetTasks();
ExitEasyChatScreen(sQuizLadyEasyChatScreens[i].callback);
}
static void DoQuizAnswerEasyChatScreen(void)
{
DoEasyChatScreen(
EASY_CHAT_TYPE_QUIZ_ANSWER,
&gSaveBlock1Ptr->lilycoveLady.quiz.playerAnswer,
CB2_ReturnToFieldContinueScript,
EASY_CHAT_PERSON_DISPLAY_NONE);
}
static void DoQuizQuestionEasyChatScreen(void)
{
DoEasyChatScreen(EASY_CHAT_TYPE_QUIZ_QUESTION,
gSaveBlock1Ptr->lilycoveLady.quiz.question,
CB2_ReturnToFieldContinueScript,
EASY_CHAT_PERSON_DISPLAY_NONE);
}
static void DoQuizSetAnswerEasyChatScreen(void)
{
DoEasyChatScreen(EASY_CHAT_TYPE_QUIZ_SET_ANSWER,
&gSaveBlock1Ptr->lilycoveLady.quiz.correctAnswer,
CB2_ReturnToFieldContinueScript,
EASY_CHAT_PERSON_DISPLAY_NONE);
}
static void DoQuizSetQuestionEasyChatScreen(void)
{
DoEasyChatScreen(EASY_CHAT_TYPE_QUIZ_SET_QUESTION,
gSaveBlock1Ptr->lilycoveLady.quiz.question,
CB2_ReturnToFieldContinueScript,
EASY_CHAT_PERSON_DISPLAY_NONE);
}
static bool8 InitEasyChatScreenStruct(u8 type, u16 *words, u8 displayedPersonType)
{
u8 templateId;
int i;
sEasyChatScreen = malloc(sizeof(*sEasyChatScreen));
if (sEasyChatScreen == NULL)
return FALSE;
sEasyChatScreen->type = type;
sEasyChatScreen->savedPhrase = words;
sEasyChatScreen->mainCursorColumn = 0;
sEasyChatScreen->mainCursorRow = 0;
sEasyChatScreen->inAlphabetMode = FALSE;
sEasyChatScreen->displayedPersonType = displayedPersonType;
sEasyChatScreen->unused = 0;
templateId = GetEachChatScreenTemplateId(type);
if (type == EASY_CHAT_TYPE_QUIZ_QUESTION)
{
GetQuizTitle(sEasyChatScreen->quizTitle);
sEasyChatScreen->titleText = sEasyChatScreen->quizTitle;
sEasyChatScreen->inputState = INPUTSTATE_QUIZ_QUESTION;
}
else
{
sEasyChatScreen->inputState = INPUTSTATE_PHRASE;
sEasyChatScreen->titleText = sEasyChatScreenTemplates[templateId].titleText;
}
sEasyChatScreen->numColumns = sEasyChatScreenTemplates[templateId].numColumns;
sEasyChatScreen->numRows = sEasyChatScreenTemplates[templateId].numRows;
sEasyChatScreen->maxWords = sEasyChatScreen->numColumns * sEasyChatScreen->numRows;
sEasyChatScreen->templateId = templateId;
if (sEasyChatScreen->maxWords > ARRAY_COUNT(sEasyChatScreen->currentPhrase))
sEasyChatScreen->maxWords = ARRAY_COUNT(sEasyChatScreen->currentPhrase);
if (words != NULL)
{
// Phrase starts with words filled in, copy to current phrase
CpuCopy16(words, sEasyChatScreen->currentPhrase, sEasyChatScreen->maxWords * sizeof(u16));
}
else
{
// Phrase starts with no words, fill with empty words and save
for (i = 0; i < sEasyChatScreen->maxWords; i ++)
sEasyChatScreen->currentPhrase[i] = EC_EMPTY_WORD;
sEasyChatScreen->savedPhrase = sEasyChatScreen->currentPhrase;
}
sEasyChatScreen->keyboardLastRow = (GetNumUnlockedEasyChatGroups() - 1) / 2 + 1;
return TRUE;
}
static void FreeEasyChatScreenStruct(void)
{
if (sEasyChatScreen != NULL)
FREE_AND_SET_NULL(sEasyChatScreen);
}
// Returns the function ID of the action to take as a result of player's input.
// If no action is needed, returns ECFUNC_NONE
static u16 HandleEasyChatInput(void)
{
switch (sEasyChatScreen->inputState)
{
case INPUTSTATE_PHRASE:
return HandleEasyChatInput_Phrase();
case INPUTSTATE_MAIN_SCREEN_BUTTONS:
return HandleEasyChatInput_MainScreenButtons();
case INPUTSTATE_KEYBOARD:
return HandleEasyChatInput_Keyboard();
case INPUTSTATE_WORD_SELECT:
return HandleEasyChatInput_WordSelect();
case INPUTSTATE_EXIT_PROMPT:
return HandleEasyChatInput_ExitPrompt();
case INPUTSTATE_DELETE_ALL_YES_NO:
return HandleEasyChatInput_DeleteAllYesNo();
case INPUTSTATE_CONFIRM_WORDS_YES_NO:
return HandleEasyChatInput_ConfirmWordsYesNo();
case INPUTSTATE_QUIZ_QUESTION:
return HandleEasyChatInput_QuizQuestion();
case INPUTSTATE_WAIT_FOR_MSG:
return HandleEasyChatInput_WaitForMsg();
case INPUTSTATE_START_CONFIRM_LYRICS:
return HandleEasyChatInput_StartConfirmLyrics();
case INPUTSTATE_CONFIRM_LYRICS_YES_NO:
return HandleEasyChatInput_ConfirmLyricsYesNo();
}
return ECFUNC_NONE;
}
static bool32 IsCurrentFrame2x5(void)
{
switch (GetEasyChatScreenFrameId())
{
case FRAMEID_MAIL:
case FRAMEID_QUIZ_QUESTION:
case FRAMEID_QUIZ_SET_QUESTION:
return TRUE;
}
return FALSE;
}
// Handles main screen input while cursor is on a word in the phrase
static u16 HandleEasyChatInput_Phrase(void)
{
do
{
if (JOY_NEW(A_BUTTON))
{
ClearUnusedField();
sEasyChatScreen->inputState = INPUTSTATE_KEYBOARD;
sEasyChatScreen->keyboardColumn = 0;
sEasyChatScreen->keyboardRow = 0;
sEasyChatScreen->keyboardScrollOffset = 0;
return ECFUNC_OPEN_KEYBOARD;
}
else if (JOY_NEW(B_BUTTON))
{
return StartConfirmExitPrompt();
}
else if (JOY_NEW(START_BUTTON))
{
return TryConfirmWords();
}
else if (JOY_NEW(DPAD_UP))
{
sEasyChatScreen->mainCursorRow--;
break;
}
else if (JOY_NEW(DPAD_LEFT))
{
sEasyChatScreen->mainCursorColumn--;
break;
}
else if (JOY_NEW(DPAD_DOWN))
{
sEasyChatScreen->mainCursorRow++;
break;
}
else if (JOY_NEW(DPAD_RIGHT))
{
sEasyChatScreen->mainCursorColumn++;
break;
}
return ECFUNC_NONE;
} while (0);
// Handle D-Pad input
// Wrap row
if (sEasyChatScreen->mainCursorRow < 0)
sEasyChatScreen->mainCursorRow = sEasyChatScreenTemplates[sEasyChatScreen->templateId].numRows;
if (sEasyChatScreen->mainCursorRow > sEasyChatScreenTemplates[sEasyChatScreen->templateId].numRows)
sEasyChatScreen->mainCursorRow = 0;
if (sEasyChatScreen->mainCursorRow == sEasyChatScreenTemplates[sEasyChatScreen->templateId].numRows)
{
// Moved onto bottom row (buttons)
if (sEasyChatScreen->mainCursorColumn > 2)
sEasyChatScreen->mainCursorColumn = 2;
sEasyChatScreen->inputState = INPUTSTATE_MAIN_SCREEN_BUTTONS;
return ECFUNC_UPDATE_MAIN_CURSOR_ON_BUTTONS;
}
// Wrap column
if (sEasyChatScreen->mainCursorColumn < 0)
sEasyChatScreen->mainCursorColumn = sEasyChatScreenTemplates[sEasyChatScreen->templateId].numColumns - 1;
if (sEasyChatScreen->mainCursorColumn >= sEasyChatScreenTemplates[sEasyChatScreen->templateId].numColumns)
sEasyChatScreen->mainCursorColumn = 0;
// All 2x5 phrases are only 9 words long, exclude the bottom right (10th) position
if (IsCurrentFrame2x5() && sEasyChatScreen->mainCursorColumn == 1 && sEasyChatScreen->mainCursorRow == 4)
sEasyChatScreen->mainCursorColumn = 0;
return ECFUNC_UPDATE_MAIN_CURSOR;
}
// Handles main screen input while cursor is below the phrase on one of the buttons, e.g. Del. All or Cancel
static u16 HandleEasyChatInput_MainScreenButtons(void)
{
do
{
if (JOY_NEW(A_BUTTON))
{
switch (sEasyChatScreen->mainCursorColumn)
{
case 0: // Del. All button
return DoDeleteAllButton();
case 1: // Cancel button
return StartConfirmExitPrompt();
case 2: // OK button
return TryConfirmWords();
case 3: // Quiz/Answer button
return DoQuizButton();
}
}
if (JOY_NEW(B_BUTTON))
{
return StartConfirmExitPrompt();
}
else if (JOY_NEW(START_BUTTON))
{
return TryConfirmWords();
}
else if (JOY_NEW(DPAD_UP))
{
sEasyChatScreen->mainCursorRow--;
break;
}
else if (JOY_NEW(DPAD_LEFT))
{
sEasyChatScreen->mainCursorColumn--;
break;
}
else if (JOY_NEW(DPAD_DOWN))
{
sEasyChatScreen->mainCursorRow = 0;
break;
}
else if (JOY_NEW(DPAD_RIGHT))
{
sEasyChatScreen->mainCursorColumn++;
break;
}
return ECFUNC_NONE;
} while (0);
if (sEasyChatScreen->mainCursorRow == sEasyChatScreenTemplates[sEasyChatScreen->templateId].numRows)
{
int numFooterColumns = FooterHasFourOptions() ? 4 : 3;
if (sEasyChatScreen->mainCursorColumn < 0)
sEasyChatScreen->mainCursorColumn = numFooterColumns - 1;
if (sEasyChatScreen->mainCursorColumn >= numFooterColumns)
sEasyChatScreen->mainCursorColumn = 0;
return ECFUNC_UPDATE_MAIN_CURSOR_ON_BUTTONS;
}
if (sEasyChatScreen->mainCursorColumn >= sEasyChatScreenTemplates[sEasyChatScreen->templateId].numColumns)
sEasyChatScreen->mainCursorColumn = sEasyChatScreenTemplates[sEasyChatScreen->templateId].numColumns - 1;
// All 2x5 phrases are only 9 words long, exclude the bottom right (10th) position
if (IsCurrentFrame2x5() && sEasyChatScreen->mainCursorColumn == 1 && sEasyChatScreen->mainCursorRow == 4)
sEasyChatScreen->mainCursorColumn = 0;
sEasyChatScreen->inputState = INPUTSTATE_PHRASE;
return ECFUNC_UPDATE_MAIN_CURSOR;
}
static u16 HandleEasyChatInput_Keyboard(void)
{
if (JOY_NEW(B_BUTTON))
return ExitKeyboardToMainScreen();
if (JOY_NEW(A_BUTTON))
{
if (sEasyChatScreen->keyboardColumn != -1)
return SelectKeyboardGroup();
// Cursor is in button window
switch (sEasyChatScreen->keyboardRow)
{
case 0: // Mode button
return StartSwitchKeyboardMode();
case 1: // Delete button
return DeleteSelectedWord();
case 2: // Cancel button
return ExitKeyboardToMainScreen();
}
}
if (JOY_NEW(SELECT_BUTTON))
return StartSwitchKeyboardMode();
if (JOY_REPEAT(DPAD_UP))
return MoveKeyboardCursor(INPUT_UP);
if (JOY_REPEAT(DPAD_DOWN))
return MoveKeyboardCursor(INPUT_DOWN);
if (JOY_REPEAT(DPAD_LEFT))
return MoveKeyboardCursor(INPUT_LEFT);
if (JOY_REPEAT(DPAD_RIGHT))
return MoveKeyboardCursor(INPUT_RIGHT);
return ECFUNC_NONE;
}
// Input handling for the lower window after a word group has been selected
static u16 HandleEasyChatInput_WordSelect(void)
{
if (JOY_NEW(B_BUTTON))
{
sEasyChatScreen->inputState = INPUTSTATE_KEYBOARD;
return ECFUNC_RETURN_TO_KEYBOARD;
}
if (JOY_NEW(A_BUTTON))
return SelectNewWord();
if (JOY_NEW(START_BUTTON))
return MoveWordSelectCursor(INPUT_START);
if (JOY_NEW(SELECT_BUTTON))
return MoveWordSelectCursor(INPUT_SELECT);
if (JOY_REPEAT(DPAD_UP))
return MoveWordSelectCursor(INPUT_UP);
if (JOY_REPEAT(DPAD_DOWN))
return MoveWordSelectCursor(INPUT_DOWN);
if (JOY_REPEAT(DPAD_LEFT))
return MoveWordSelectCursor(INPUT_LEFT);
if (JOY_REPEAT(DPAD_RIGHT))
return MoveWordSelectCursor(INPUT_RIGHT);
return ECFUNC_NONE;
}
static u16 HandleEasyChatInput_ExitPrompt(void)
{
switch (Menu_ProcessInputNoWrapClearOnChoose())
{
case MENU_B_PRESSED:
case 1: // No (Continue)
sEasyChatScreen->inputState = GetEasyChatBackupState();
return ECFUNC_CLOSE_PROMPT;
case 0: // Yes (Exit)
gSpecialVar_Result = 0;
if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_QUESTION
|| sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_ANSWER)
SaveCurrentPhrase();
return ECFUNC_EXIT;
default:
return ECFUNC_NONE;
}
}
static u16 HandleEasyChatInput_ConfirmWordsYesNo(void)
{
switch (Menu_ProcessInputNoWrapClearOnChoose())
{
case MENU_B_PRESSED:
case 1: // No
sEasyChatScreen->inputState = GetEasyChatBackupState();
return ECFUNC_CLOSE_PROMPT;
case 0: // Yes
SetSpecialEasyChatResult();
gSpecialVar_Result = GetEasyChatCompleted();
SaveCurrentPhrase();
return ECFUNC_EXIT;
default:
return ECFUNC_NONE;
}
}
static u16 HandleEasyChatInput_DeleteAllYesNo(void)
{
switch (Menu_ProcessInputNoWrapClearOnChoose())
{
case MENU_B_PRESSED:
case 1: // No
sEasyChatScreen->inputState = INPUTSTATE_MAIN_SCREEN_BUTTONS;
return ECFUNC_CLOSE_PROMPT;
case 0: // Yes
ResetCurrentPhrase();
sEasyChatScreen->inputState = INPUTSTATE_MAIN_SCREEN_BUTTONS;
return ECFUNC_CLOSE_PROMPT_AFTER_DELETE;
default:
return ECFUNC_NONE;
}
}
static u16 HandleEasyChatInput_QuizQuestion(void)
{
if (JOY_NEW(A_BUTTON))
return ECFUNC_QUIZ_ANSWER;
if (JOY_NEW(B_BUTTON))
return StartConfirmExitPrompt();
return ECFUNC_NONE;
}
// A message has been printed. Wait for player to
// press A or B, then return to previous state
static u16 HandleEasyChatInput_WaitForMsg(void)
{
if (JOY_NEW(A_BUTTON | B_BUTTON))
{
sEasyChatScreen->inputState = GetEasyChatBackupState();
return ECFUNC_CLOSE_PROMPT;
}
return ECFUNC_NONE;
}
// Odd, could have been skipped. Just passes to HandleEasyChatInput_ConfirmLyricsYesNo
static u16 HandleEasyChatInput_StartConfirmLyrics(void)
{
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_LYRICS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
static u16 HandleEasyChatInput_ConfirmLyricsYesNo(void)
{
switch (Menu_ProcessInputNoWrapClearOnChoose())
{
case MENU_B_PRESSED:
case 1: // No
ResetCurrentPhraseToSaved();
sEasyChatScreen->inputStateBackup = INPUTSTATE_PHRASE;
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_SONG_TOO_SHORT;
case 0: // Yes
gSpecialVar_Result = GetEasyChatCompleted();
SaveCurrentPhrase();
return ECFUNC_EXIT;
default:
return ECFUNC_NONE;
}
}
static u16 StartConfirmExitPrompt(void)
{
if (sEasyChatScreen->type == EASY_CHAT_TYPE_APPRENTICE
|| sEasyChatScreen->type == EASY_CHAT_TYPE_CONTEST_INTERVIEW)
{
sEasyChatScreen->inputStateBackup = sEasyChatScreen->inputState;
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_CANT_EXIT;
}
else
{
sEasyChatScreen->inputStateBackup = sEasyChatScreen->inputState;
sEasyChatScreen->inputState = INPUTSTATE_EXIT_PROMPT;
return ECFUNC_PROMPT_EXIT;
}
}
static int DoDeleteAllButton(void)
{
sEasyChatScreen->inputStateBackup = sEasyChatScreen->inputState;
if (sEasyChatScreen->type != EASY_CHAT_TYPE_BARD_SONG)
{
// Show Delete yes/no
sEasyChatScreen->inputState = INPUTSTATE_DELETE_ALL_YES_NO;
return ECFUNC_PROMPT_DELETE_ALL;
}
else
{
// Cannot delete lyrics when setting Bard's song
sEasyChatScreen->inputStateBackup = sEasyChatScreen->inputState;
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_CANT_DELETE_LYRICS;
}
}
static u16 TryConfirmWords(void)
{
sEasyChatScreen->inputStateBackup = sEasyChatScreen->inputState;
if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_QUESTION)
{
if (IsQuizQuestionEmpty())
{
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_CREATE_QUIZ;
}
if (IsQuizAnswerEmpty())
{
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_SELECT_ANSWER;
}
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_WORDS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
else if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_ANSWER)
{
if (IsQuizAnswerEmpty())
{
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_SELECT_ANSWER;
}
if (IsQuizQuestionEmpty())
{
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_CREATE_QUIZ;
}
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_WORDS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
else if (sEasyChatScreen->type == EASY_CHAT_TYPE_TRENDY_PHRASE
|| sEasyChatScreen->type == EASY_CHAT_TYPE_GOOD_SAYING)
{
if (!IsCurrentPhraseFull())
{
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_COMBINE_TWO_WORDS;
}
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_WORDS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
else if (sEasyChatScreen->type == EASY_CHAT_TYPE_APPRENTICE
|| sEasyChatScreen->type == EASY_CHAT_TYPE_CONTEST_INTERVIEW)
{
if (IsCurrentPhraseEmpty())
{
sEasyChatScreen->inputState = INPUTSTATE_WAIT_FOR_MSG;
return ECFUNC_MSG_CANT_EXIT;
}
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_WORDS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
else if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUESTIONNAIRE)
{
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_WORDS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
else
{
if (IsCurrentPhraseEmpty() == TRUE || !GetEasyChatCompleted())
{
sEasyChatScreen->inputState = INPUTSTATE_EXIT_PROMPT;
return ECFUNC_PROMPT_EXIT;
}
sEasyChatScreen->inputState = INPUTSTATE_CONFIRM_WORDS_YES_NO;
return ECFUNC_PROMPT_CONFIRM;
}
}
static int DoQuizButton(void)
{
sEasyChatScreen->inputStateBackup = sEasyChatScreen->inputState;
switch (sEasyChatScreen->type)
{
case EASY_CHAT_TYPE_QUIZ_ANSWER:
return ECFUNC_QUIZ_QUESTION;
case EASY_CHAT_TYPE_QUIZ_SET_QUESTION:
SaveCurrentPhrase();
return ECFUNC_SET_QUIZ_ANSWER;
case EASY_CHAT_TYPE_QUIZ_SET_ANSWER:
SaveCurrentPhrase();
return ECFUNC_SET_QUIZ_QUESTION;
default:
return ECFUNC_NONE;
}
}
static u8 GetEasyChatBackupState(void)
{
return sEasyChatScreen->inputStateBackup;
}
static int SelectKeyboardGroup(void)
{
u16 numWords;
if (!sEasyChatScreen->inAlphabetMode)
{
u8 groupId = GetUnlockedEasyChatGroupId(GetSelectedGroupIndex());
SetSelectedWordGroup(FALSE, groupId);
}
else
{
SetSelectedWordGroup(TRUE, GetSelectedAlphabetGroupId());
}
numWords = GetNumWordsInSelectedGroup();
if (numWords == 0)
return ECFUNC_NONE;
sEasyChatScreen->wordSelectLastRow = (numWords - 1) / 2;
sEasyChatScreen->wordSelectScrollOffset = 0;
sEasyChatScreen->wordSelectColumn = 0;
sEasyChatScreen->wordSelectRow = 0;
sEasyChatScreen->inputState = INPUTSTATE_WORD_SELECT;
return ECFUNC_OPEN_WORD_SELECT;
}
static int ExitKeyboardToMainScreen(void)
{
sEasyChatScreen->inputState = INPUTSTATE_PHRASE;
return ECFUNC_CLOSE_KEYBOARD;
}
static int StartSwitchKeyboardMode(void)
{
sEasyChatScreen->keyboardColumn = 0;
sEasyChatScreen->keyboardRow = 0;
sEasyChatScreen->keyboardScrollOffset = 0;
if (!sEasyChatScreen->inAlphabetMode)
sEasyChatScreen->inAlphabetMode = TRUE;
else
sEasyChatScreen->inAlphabetMode = FALSE;
return ECFUNC_SWITCH_KEYBOARD_MODE;
}
static int DeleteSelectedWord(void)
{
if (sEasyChatScreen->type == EASY_CHAT_TYPE_BARD_SONG)
{
PlaySE(SE_FAILURE);
return ECFUNC_NONE;
}
else
{
SetSelectedWord(EC_EMPTY_WORD);
return ECFUNC_REPRINT_PHRASE;
}
}
static int SelectNewWord(void)
{
u16 easyChatWord = GetWordFromSelectedGroup(GetSelectedWordIndex());
if (DummyWordCheck(easyChatWord))
{
// Never reached. Would disallow selecting certain words
PlaySE(SE_FAILURE);
return ECFUNC_NONE;
}
else
{
SetSelectedWord(easyChatWord);
if (sEasyChatScreen->type != EASY_CHAT_TYPE_BARD_SONG)
{
sEasyChatScreen->inputState = INPUTSTATE_PHRASE;
return ECFUNC_CLOSE_WORD_SELECT;
}
else
{
sEasyChatScreen->inputState = INPUTSTATE_START_CONFIRM_LYRICS;
return ECFUNC_PROMPT_CONFIRM_LYRICS;
}
}
}
static void SaveCurrentPhrase(void)
{
int i;
for (i = 0; i < sEasyChatScreen->maxWords; i++)
sEasyChatScreen->savedPhrase[i] = sEasyChatScreen->currentPhrase[i];
}
static void ResetCurrentPhrase(void)
{
int i;
for (i = 0; i < sEasyChatScreen->maxWords; i++)
sEasyChatScreen->currentPhrase[i] = EC_EMPTY_WORD;
}
static void ResetCurrentPhraseToSaved(void)
{
int i;
for (i = 0; i < sEasyChatScreen->maxWords; i++)
sEasyChatScreen->currentPhrase[i] = sEasyChatScreen->savedPhrase[i];
}
static void SetSelectedWord(u16 easyChatWord)
{
u16 index = GetWordIndexToReplace();
sEasyChatScreen->currentPhrase[index] = easyChatWord;
}
// Compare current phrase to the original saved phrase
static bool8 DidPhraseChange(void)
{
u16 i;
for (i = 0; i < sEasyChatScreen->maxWords; i++)
{
if (sEasyChatScreen->currentPhrase[i] != sEasyChatScreen->savedPhrase[i])
return TRUE;
}
return FALSE;
}
// 'Completed' if the phrase was changed, or in the case of making a quiz, the question and answer were filled out
static bool32 GetEasyChatCompleted(void)
{
if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_QUESTION
|| sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_ANSWER)
{
if (IsQuizQuestionEmpty())
return FALSE;
if (IsQuizAnswerEmpty())
return FALSE;
return TRUE;
}
else
{
return DidPhraseChange();
}
}
static u16 MoveKeyboardCursor(int input)
{
if (sEasyChatScreen->keyboardColumn != -1)
{
if (!sEasyChatScreen->inAlphabetMode)
return MoveKeyboardCursor_GroupNames(input);
else
return MoveKeyboardCursor_Alphabet(input);
}
else
{
return MoveKeyboardCursor_ButtonWindow(input);
}
}
static int MoveKeyboardCursor_GroupNames(u32 input)
{
switch (input)
{
case INPUT_UP:
if (sEasyChatScreen->keyboardRow != -sEasyChatScreen->keyboardScrollOffset)
{
if (sEasyChatScreen->keyboardRow)
{
sEasyChatScreen->keyboardRow--;
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
}
else
{
sEasyChatScreen->keyboardScrollOffset--;
return ECFUNC_GROUP_NAMES_SCROLL_UP;
}
}
break;
case INPUT_DOWN:
if (sEasyChatScreen->keyboardRow + sEasyChatScreen->keyboardScrollOffset < sEasyChatScreen->keyboardLastRow - 1)
{
int funcId;
if (sEasyChatScreen->keyboardRow < NUM_GROUP_NAME_ROWS - 1)
{
sEasyChatScreen->keyboardRow++;
funcId = ECFUNC_UPDATE_KEYBOARD_CURSOR;
}
else
{
sEasyChatScreen->keyboardScrollOffset++;
funcId = ECFUNC_GROUP_NAMES_SCROLL_DOWN;
}
ReduceToValidKeyboardColumn();
return funcId;
}
break;
case INPUT_LEFT:
if (sEasyChatScreen->keyboardColumn)
sEasyChatScreen->keyboardColumn--;
else
SetKeyboardCursorInButtonWindow();
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_RIGHT:
if (sEasyChatScreen->keyboardColumn < 1)
{
sEasyChatScreen->keyboardColumn++;
if (IsSelectedKeyboardIndexInvalid())
SetKeyboardCursorInButtonWindow();
}
else
{
SetKeyboardCursorInButtonWindow();
}
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
}
return ECFUNC_NONE;
}
static int MoveKeyboardCursor_Alphabet(u32 input)
{
switch (input)
{
case INPUT_UP:
if (sEasyChatScreen->keyboardRow > 0)
sEasyChatScreen->keyboardRow--;
else
sEasyChatScreen->keyboardRow = NUM_ALPHABET_ROWS - 1;
ReduceToValidKeyboardColumn();
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_DOWN:
if (sEasyChatScreen->keyboardRow < NUM_ALPHABET_ROWS - 1)
sEasyChatScreen->keyboardRow++;
else
sEasyChatScreen->keyboardRow = 0;
ReduceToValidKeyboardColumn();
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_RIGHT:
sEasyChatScreen->keyboardColumn++;
if (IsSelectedKeyboardIndexInvalid())
SetKeyboardCursorInButtonWindow();
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_LEFT:
sEasyChatScreen->keyboardColumn--;
if (sEasyChatScreen->keyboardColumn < 0)
SetKeyboardCursorInButtonWindow();
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
}
return ECFUNC_NONE;
}
static int MoveKeyboardCursor_ButtonWindow(u32 input)
{
switch (input)
{
case INPUT_UP:
if (sEasyChatScreen->keyboardRow)
sEasyChatScreen->keyboardRow--;
else
sEasyChatScreen->keyboardRow = NUM_BUTTON_ROWS - 1;
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_DOWN:
if (sEasyChatScreen->keyboardRow < NUM_BUTTON_ROWS - 1)
sEasyChatScreen->keyboardRow++;
else
sEasyChatScreen->keyboardRow = 0;
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_LEFT:
sEasyChatScreen->keyboardRow++;
SetKeyboardCursorToLastColumn();
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
case INPUT_RIGHT:
sEasyChatScreen->keyboardColumn = 0;
sEasyChatScreen->keyboardRow++;
return ECFUNC_UPDATE_KEYBOARD_CURSOR;
}
return ECFUNC_NONE;
}
static void SetKeyboardCursorInButtonWindow(void)
{
sEasyChatScreen->keyboardColumn = -1;
if (sEasyChatScreen->keyboardRow)
sEasyChatScreen->keyboardRow--;
}
static void SetKeyboardCursorToLastColumn(void)
{
if (!sEasyChatScreen->inAlphabetMode)
{
sEasyChatScreen->keyboardColumn = 1;
ReduceToValidKeyboardColumn();
}
else
{
sEasyChatScreen->keyboardColumn = GetLastAlphabetColumn(sEasyChatScreen->keyboardRow);
}
}
static u16 MoveWordSelectCursor(u32 input)
{
u16 funcId;
switch (input)
{
case INPUT_UP:
if (sEasyChatScreen->wordSelectRow + sEasyChatScreen->wordSelectScrollOffset > 0)
{
if (sEasyChatScreen->wordSelectRow > 0)
{
sEasyChatScreen->wordSelectRow--;
funcId = ECFUNC_UPDATE_WORD_SELECT_CURSOR;
}
else
{
sEasyChatScreen->wordSelectScrollOffset--;
funcId = ECFUNC_WORD_SELECT_SCROLL_UP;
}
ReduceToValidWordSelectColumn();
return funcId;
}
break;
case INPUT_DOWN:
if (sEasyChatScreen->wordSelectRow + sEasyChatScreen->wordSelectScrollOffset < sEasyChatScreen->wordSelectLastRow)
{
if (sEasyChatScreen->wordSelectRow < NUM_WORD_SELECT_ROWS - 1)
{
sEasyChatScreen->wordSelectRow++;
funcId = ECFUNC_UPDATE_WORD_SELECT_CURSOR;
}
else
{
sEasyChatScreen->wordSelectScrollOffset++;
funcId = ECFUNC_WORD_SELECT_SCROLL_DOWN;
}
ReduceToValidWordSelectColumn();
return funcId;
}
break;
case INPUT_LEFT:
if (sEasyChatScreen->wordSelectColumn > 0)
sEasyChatScreen->wordSelectColumn--;
else
sEasyChatScreen->wordSelectColumn = 1;
ReduceToValidWordSelectColumn();
return ECFUNC_UPDATE_WORD_SELECT_CURSOR;
case INPUT_RIGHT:
if (sEasyChatScreen->wordSelectColumn < 1)
{
sEasyChatScreen->wordSelectColumn++;
if (IsSelectedWordIndexInvalid())
sEasyChatScreen->wordSelectColumn = 0;
}
else
{
sEasyChatScreen->wordSelectColumn = 0;
}
return ECFUNC_UPDATE_WORD_SELECT_CURSOR;
case INPUT_START:
// Page scroll up
if (sEasyChatScreen->wordSelectScrollOffset)
{
if (sEasyChatScreen->wordSelectScrollOffset >= NUM_WORD_SELECT_ROWS)
sEasyChatScreen->wordSelectScrollOffset -= NUM_WORD_SELECT_ROWS;
else
sEasyChatScreen->wordSelectScrollOffset = 0;
return ECFUNC_WORD_SELECT_PAGE_UP;
}
break;
case INPUT_SELECT:
// Page scroll down
if (sEasyChatScreen->wordSelectScrollOffset <= sEasyChatScreen->wordSelectLastRow - NUM_WORD_SELECT_ROWS)
{
sEasyChatScreen->wordSelectScrollOffset += NUM_WORD_SELECT_ROWS;
if (sEasyChatScreen->wordSelectScrollOffset > sEasyChatScreen->wordSelectLastRow - NUM_WORD_SELECT_ROWS + 1)
sEasyChatScreen->wordSelectScrollOffset = sEasyChatScreen->wordSelectLastRow - NUM_WORD_SELECT_ROWS + 1;
ReduceToValidWordSelectColumn();
return ECFUNC_WORD_SELECT_PAGE_DOWN;
}
break;
}
return ECFUNC_NONE;
}
static u16 GetWordIndexToReplace(void)
{
return (sEasyChatScreen->mainCursorRow * sEasyChatScreen->numColumns) + sEasyChatScreen->mainCursorColumn;
}
static u16 GetSelectedGroupIndex(void)
{
return NUM_GROUP_NAME_COLUMNS * (sEasyChatScreen->keyboardRow + sEasyChatScreen->keyboardScrollOffset) + sEasyChatScreen->keyboardColumn;
}
static int GetSelectedAlphabetGroupId(void)
{
int column = (u8)sEasyChatScreen->keyboardColumn < NUM_ALPHABET_COLUMNS ? sEasyChatScreen->keyboardColumn : 0;
int row = (u8)sEasyChatScreen->keyboardRow < NUM_ALPHABET_ROWS ? sEasyChatScreen->keyboardRow : 0;
return sAlphabetGroupIdMap[row][column];
}
static u16 GetSelectedWordIndex(void)
{
return NUM_WORD_SELECT_COLUMNS * (sEasyChatScreen->wordSelectRow + sEasyChatScreen->wordSelectScrollOffset) + sEasyChatScreen->wordSelectColumn;
}
// Get the index of the last column in the alphabet keyboard, depending on current row
static u8 GetLastAlphabetColumn(u8 row)
{
switch (row)
{
case 0:
default:
return NUM_ALPHABET_COLUMNS - 1;
case 1:
return NUM_ALPHABET_COLUMNS - 2; // At 6 letters, only the 2nd row (index 1) has less than the max columns
// The 3rd and 4th row have 7 letters, the 1st row has 6 letters and 'Others'
}
}
static void ReduceToValidKeyboardColumn(void)
{
while (IsSelectedKeyboardIndexInvalid())
{
if (sEasyChatScreen->keyboardColumn)
sEasyChatScreen->keyboardColumn--;
else
break;
}
}
static void ReduceToValidWordSelectColumn(void)
{
while (IsSelectedWordIndexInvalid())
{
if (sEasyChatScreen->wordSelectColumn)
sEasyChatScreen->wordSelectColumn--;
else
break;
}
}
static bool8 IsSelectedKeyboardIndexInvalid(void)
{
if (!sEasyChatScreen->inAlphabetMode)
return GetSelectedGroupIndex() >= GetNumUnlockedEasyChatGroups() ? TRUE : FALSE;
else
return sEasyChatScreen->keyboardColumn > GetLastAlphabetColumn(sEasyChatScreen->keyboardRow) ? TRUE : FALSE;
}
static bool8 IsSelectedWordIndexInvalid(void)
{
return GetSelectedWordIndex() >= GetNumWordsInSelectedGroup() ? TRUE : FALSE;
}
static int FooterHasFourOptions(void)
{
return sEasyChatScreenTemplates[sEasyChatScreen->templateId].fourFooterOptions;
}
static u8 GetEasyChatScreenType(void)
{
return sEasyChatScreen->type;
}
static u8 GetEasyChatScreenFrameId(void)
{
return sEasyChatScreenTemplates[sEasyChatScreen->templateId].frameId;
}
const u8 *GetTitleText(void)
{
return sEasyChatScreen->titleText;
}
static u16 *GetCurrentPhrase(void)
{
return sEasyChatScreen->currentPhrase;
}
static u8 GetNumRows(void)
{
return sEasyChatScreen->numRows;
}
static u8 GetNumColumns(void)
{
return sEasyChatScreen->numColumns;
}
static u8 GetMainCursorColumn(void)
{
return sEasyChatScreen->mainCursorColumn;
}
static u8 GetMainCursorRow(void)
{
return sEasyChatScreen->mainCursorRow;
}
static void GetEasyChatInstructionsText(const u8 **str1, const u8 **str2)
{
*str1 = sEasyChatScreenTemplates[sEasyChatScreen->templateId].instructionsText1;
*str2 = sEasyChatScreenTemplates[sEasyChatScreen->templateId].instructionsText2;
}
static void GetEasyChatConfirmText(const u8 **str1, const u8 **str2)
{
*str1 = sEasyChatScreenTemplates[sEasyChatScreen->templateId].confirmText1;
*str2 = sEasyChatScreenTemplates[sEasyChatScreen->templateId].confirmText2;
}
static void GetEasyChatConfirmExitText(const u8 **str1, const u8 **str2)
{
switch (sEasyChatScreen->type)
{
case EASY_CHAT_TYPE_MAIL:
*str1 = gText_StopGivingPkmnMail;
*str2 = NULL;
break;
case EASY_CHAT_TYPE_QUIZ_ANSWER:
case EASY_CHAT_TYPE_QUIZ_QUESTION:
*str1 = gText_LikeToQuitQuiz;
*str2 = gText_ChallengeQuestionMark;
break;
default:
*str1 = gText_QuitEditing;
*str2 = NULL;
break;
}
}
static void GetEasyChatConfirmDeletionText(const u8 **str1, const u8 **str2)
{
*str1 = gText_AllTextBeingEditedWill;
*str2 = gText_BeDeletedThatOkay;
}
static void GetKeyboardCursorColAndRow(u8 *column, u8 *row)
{
*column = sEasyChatScreen->keyboardColumn;
*row = sEasyChatScreen->keyboardRow;
}
static bool8 GetInAlphabetMode(void)
{
return sEasyChatScreen->inAlphabetMode;
}
static u8 GetKeyboardScrollOffset(void)
{
return sEasyChatScreen->keyboardScrollOffset;
}
static void GetWordSelectColAndRow(u8 *column, u8 *row)
{
*column = sEasyChatScreen->wordSelectColumn;
*row = sEasyChatScreen->wordSelectRow;
}
static u8 GetWordSelectScrollOffset(void)
{
return sEasyChatScreen->wordSelectScrollOffset;
}
static u8 GetWordSelectLastRow(void)
{
return sEasyChatScreen->wordSelectLastRow;
}
static u8 UnusedDummy(void)
{
return FALSE;
}
static bool32 CanScrollUp(void)
{
switch (sEasyChatScreen->inputState)
{
case INPUTSTATE_KEYBOARD:
if (!sEasyChatScreen->inAlphabetMode && sEasyChatScreen->keyboardScrollOffset)
return TRUE;
break;
case INPUTSTATE_WORD_SELECT:
if (sEasyChatScreen->wordSelectScrollOffset)
return TRUE;
break;
}
return FALSE;
}
static bool32 CanScrollDown(void)
{
switch (sEasyChatScreen->inputState)
{
case INPUTSTATE_KEYBOARD:
if (!sEasyChatScreen->inAlphabetMode && sEasyChatScreen->keyboardScrollOffset + NUM_GROUP_NAME_ROWS <= sEasyChatScreen->keyboardLastRow - 1)
return TRUE;
break;
case INPUTSTATE_WORD_SELECT:
if (sEasyChatScreen->wordSelectScrollOffset + NUM_WORD_SELECT_ROWS <= sEasyChatScreen->wordSelectLastRow)
return TRUE;
break;
}
return FALSE;
}
static int FooterHasFourOptions_(void)
{
return FooterHasFourOptions();
}
static bool8 IsPhraseDifferentThanPlayerInput(const u16 *phrase, u8 phraseLength)
{
u8 i;
for (i = 0; i < phraseLength; i++)
{
if (phrase[i] != sEasyChatScreen->currentPhrase[i])
return TRUE;
}
return FALSE;
}
static u8 GetDisplayedPersonType(void)
{
return sEasyChatScreen->displayedPersonType;
}
static u8 GetEachChatScreenTemplateId(u8 type)
{
u32 i;
for (i = 0; i < ARRAY_COUNT(sEasyChatScreenTemplates); i++)
{
if (sEasyChatScreenTemplates[i].type == type)
return i;
}
return 0;
}
static bool32 IsCurrentPhraseEmpty(void)
{
int i;
for (i = 0; i < sEasyChatScreen->maxWords; i++)
{
if (sEasyChatScreen->currentPhrase[i] != EC_EMPTY_WORD)
return FALSE;
}
return TRUE;
}
static bool32 IsCurrentPhraseFull(void)
{
int i;
for (i = 0; i < sEasyChatScreen->maxWords; i++)
{
if (sEasyChatScreen->currentPhrase[i] == EC_EMPTY_WORD)
return FALSE;
}
return TRUE;
}
static int IsQuizQuestionEmpty(void)
{
int i;
struct SaveBlock1 *saveBlock1;
if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_QUESTION)
return IsCurrentPhraseEmpty();
saveBlock1 = gSaveBlock1Ptr;
for (i = 0; i < QUIZ_QUESTION_LEN; i++)
{
if (saveBlock1->lilycoveLady.quiz.question[i] != EC_EMPTY_WORD)
return FALSE;
}
return TRUE;
}
static int IsQuizAnswerEmpty(void)
{
struct LilycoveLadyQuiz *quiz;
if (sEasyChatScreen->type == EASY_CHAT_TYPE_QUIZ_SET_ANSWER)
return IsCurrentPhraseEmpty();
quiz = &gSaveBlock1Ptr->lilycoveLady.quiz;
return quiz->correctAnswer == EC_EMPTY_WORD ? TRUE : FALSE;
}
static void GetQuizTitle(u8 *dst)
{
u8 name[32];
struct SaveBlock1 *saveBlock1 = gSaveBlock1Ptr;
DynamicPlaceholderTextUtil_Reset();
// Buffer author's name
if (StringLength(saveBlock1->lilycoveLady.quiz.playerName) != 0)
{
TVShowConvertInternationalString(name, saveBlock1->lilycoveLady.quiz.playerName, saveBlock1->lilycoveLady.quiz.language);
DynamicPlaceholderTextUtil_SetPlaceholderPtr(0, name);
}
else
{
DynamicPlaceholderTextUtil_SetPlaceholderPtr(0, gText_Lady);
}
// "<author>'s Quiz"
DynamicPlaceholderTextUtil_ExpandPlaceholders(dst, gText_F700sQuiz);
}
static void BufferCurrentPhraseToStringVar2(void)
{
int i;
u16 *phrase;
u8 *str;
phrase = sEasyChatScreen->currentPhrase;
str = gStringVar2;
i = 0;
while (i < sEasyChatScreen->maxWords)
{
str = CopyEasyChatWordPadded(str, *phrase, 0);
*str = 0;
str++;
phrase++;
i++;
}
str--;
str[0] = EOS;
}
static void SetSpecialEasyChatResult(void)
{
switch (sEasyChatScreen->type)
{
case EASY_CHAT_TYPE_PROFILE:
FlagSet(FLAG_SYS_CHAT_USED);
break;
case EASY_CHAT_TYPE_QUESTIONNAIRE:
if (DidPlayerInputMysteryGiftPhrase())
gSpecialVar_0x8004 = 2;
else
gSpecialVar_0x8004 = 0;
break;
case EASY_CHAT_TYPE_TRENDY_PHRASE:
BufferCurrentPhraseToStringVar2();
gSpecialVar_0x8004 = IsPhraseTrendy(sEasyChatScreen->currentPhrase);
break;
case EASY_CHAT_TYPE_GOOD_SAYING:
gSpecialVar_0x8004 = DidPlayerInputABerryMasterWifePhrase();
break;
}
}
static int DidPlayerInputMysteryGiftPhrase(void)
{
return !IsPhraseDifferentThanPlayerInput(sMysteryGiftPhrase, ARRAY_COUNT(sMysteryGiftPhrase));
}
static u16 DidPlayerInputABerryMasterWifePhrase(void)
{
int i;
for (i = 0; i < (int)ARRAY_COUNT(sBerryMasterWifePhrases); i++)
{
if (!IsPhraseDifferentThanPlayerInput(sBerryMasterWifePhrases[i], ARRAY_COUNT(*sBerryMasterWifePhrases)))
return i + 1;
}
return 0;
}
static void ClearUnusedField(void)
{
sEasyChatScreen->unused = 0;
}
static bool32 DummyWordCheck(int easyChatWord)
{
return FALSE;
}
static bool8 InitEasyChatScreenControl(void)
{
if (!InitEasyChatScreenControl_())
return FALSE;
else
return TRUE;
}
static bool8 LoadEasyChatScreen(void)
{
switch (sScreenControl->funcState)
{
case 0:
ResetBgsAndClearDma3BusyFlags(0);
InitBgsFromTemplates(0, sEasyChatBgTemplates, ARRAY_COUNT(sEasyChatBgTemplates));
SetBgTilemapBuffer(3, sScreenControl->bg3TilemapBuffer);
SetBgTilemapBuffer(1, sScreenControl->bg1TilemapBuffer);
InitWindows(sEasyChatWindowTemplates);
DeactivateAllTextPrinters();
LoadEasyChatPalettes();
InitEasyChatBgs();
CpuFastFill(0, (void *)OAM, OAM_SIZE);
break;
case 1:
DecompressAndLoadBgGfxUsingHeap(3, gEasyChatWindow_Gfx, 0, 0, 0);
CopyToBgTilemapBuffer(3, gEasyChatWindow_Tilemap, 0, 0);
AdjustBgTilemapForFooter();
BufferFrameTilemap(sScreenControl->bg1TilemapBuffer);
AddPhraseWindow();
AddMainScreenButtonWindow();
CopyBgTilemapBufferToVram(3);
break;
case 2:
DecompressAndLoadBgGfxUsingHeap(1, sTextInputFrame_Gfx, 0, 0, 0);
CopyBgTilemapBufferToVram(1);
break;
case 3:
PrintTitle();
PrintInitialInstructions();
PrintCurrentPhrase();
DrawLowerWindow();
break;
case 4:
LoadEasyChatGfx();
if (GetEasyChatScreenType() != EASY_CHAT_TYPE_QUIZ_QUESTION)
CreateMainCursorSprite();
break;
case 5:
if (IsDma3ManagerBusyWithBgCopy())
{
return TRUE;
}
else
{
SetWindowDimensions(0, 0, 0, 0);
SetGpuReg(REG_OFFSET_WININ, WININ_WIN0_BG_ALL | WININ_WIN0_OBJ | WININ_WIN0_CLR);
SetGpuReg(REG_OFFSET_WINOUT, WINOUT_WIN01_BG0
| WINOUT_WIN01_BG1
| WINOUT_WIN01_BG3
| WINOUT_WIN01_OBJ
| WINOUT_WIN01_CLR);
ShowBg(3);
ShowBg(1);
ShowBg(2);
ShowBg(0);
CreateScrollIndicatorSprites();
CreateStartSelectButtonSprites();
TryAddInterviewObjectEvents();
}
break;
default:
return FALSE;
}
sScreenControl->funcState++;
return TRUE;
}
static void FreeEasyChatScreenControl(void)
{
if (sScreenControl)
FREE_AND_SET_NULL(sScreenControl);
}
static void StartEasyChatFunction(u16 funcId)
{
sScreenControl->currentFuncId = funcId;
sScreenControl->funcState = 0;
RunEasyChatFunction();
}
// Returns FALSE when called function has finished
static bool8 RunEasyChatFunction(void)
{
switch (sScreenControl->currentFuncId)
{
case ECFUNC_NONE: return FALSE;
case ECFUNC_REPRINT_PHRASE: return ReprintPhrase();
case ECFUNC_UPDATE_MAIN_CURSOR: return UpdateMainCursor();
case ECFUNC_UPDATE_MAIN_CURSOR_ON_BUTTONS: return UpdateMainCursorOnButtons();
case ECFUNC_PROMPT_DELETE_ALL: return ShowConfirmDeleteAllPrompt();
case ECFUNC_PROMPT_EXIT: return ShowConfirmExitPrompt();
case ECFUNC_PROMPT_CONFIRM: return ShowConfirmPrompt();
case ECFUNC_CLOSE_PROMPT: return ClosePrompt();
case ECFUNC_CLOSE_PROMPT_AFTER_DELETE: return ClosePromptAfterDeleteAll();
case ECFUNC_OPEN_KEYBOARD: return OpenKeyboard();
case ECFUNC_CLOSE_KEYBOARD: return CloseKeyboard();
case ECFUNC_OPEN_WORD_SELECT: return OpenWordSelect();
case ECFUNC_CLOSE_WORD_SELECT: return CloseWordSelect();
case ECFUNC_PROMPT_CONFIRM_LYRICS: return ShowConfirmLyricsPrompt();
case ECFUNC_RETURN_TO_KEYBOARD: return ReturnToKeyboard();
case ECFUNC_UPDATE_KEYBOARD_CURSOR: return UpdateKeyboardCursor();
case ECFUNC_GROUP_NAMES_SCROLL_DOWN: return GroupNamesScrollDown();
case ECFUNC_GROUP_NAMES_SCROLL_UP: return GroupNamesScrollUp();
case ECFUNC_UPDATE_WORD_SELECT_CURSOR: return UpdateWordSelectCursor();
case ECFUNC_WORD_SELECT_SCROLL_UP: return WordSelectScrollUp();
case ECFUNC_WORD_SELECT_SCROLL_DOWN: return WordSelectScrollDown();
case ECFUNC_WORD_SELECT_PAGE_UP: return WordSelectPageScrollUp();
case ECFUNC_WORD_SELECT_PAGE_DOWN: return WordSelectPageScrollDown();
case ECFUNC_SWITCH_KEYBOARD_MODE: return SwitchKeyboardMode();
case ECFUNC_EXIT: return FALSE;
case ECFUNC_QUIZ_QUESTION: return FALSE; // The 4 quiz functions
case ECFUNC_QUIZ_ANSWER: return FALSE; // 'finish' automatically
case ECFUNC_SET_QUIZ_QUESTION: return FALSE; // because they switch to a
case ECFUNC_SET_QUIZ_ANSWER: return FALSE; // callback in sQuizLadyEasyChatScreens
case ECFUNC_MSG_CREATE_QUIZ: return ShowCreateQuizMsg();
case ECFUNC_MSG_SELECT_ANSWER: return ShowSelectAnswerMsg();
case ECFUNC_MSG_SONG_TOO_SHORT: return ShowSongTooShortMsg();
case ECFUNC_MSG_CANT_DELETE_LYRICS: return ShowCantDeleteLyricsMsg();
case ECFUNC_MSG_COMBINE_TWO_WORDS: return ShowCombineTwoWordsMsg();
case ECFUNC_MSG_CANT_EXIT: return ShowCantExitMsg();
default: return FALSE;
}
}
// Only used to update the current phrase after a word deletion
static bool8 ReprintPhrase(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintCurrentPhrase();
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 UpdateMainCursor(void)
{
u8 i;
u16 *currentPhrase;
u16 *ecWord;
u8 frameId;
u8 cursorColumn, cursorRow, numColumns;
s16 x;
int stringWidth;
int trueStringWidth;
u8 y;
u8 str[64];
currentPhrase = GetCurrentPhrase();
frameId = GetEasyChatScreenFrameId();
cursorColumn = GetMainCursorColumn();
cursorRow = GetMainCursorRow();
numColumns = GetNumColumns();
ecWord = &currentPhrase[cursorRow * numColumns];
x = 8 * sPhraseFrameDimensions[frameId].left + 13;
for (i = 0; i < cursorColumn; i++)
{
if (*ecWord == EC_EMPTY_WORD)
{
stringWidth = 72;
}
else
{
CopyEasyChatWord(str, *ecWord);
stringWidth = GetStringWidth(1, str, 0);
}
trueStringWidth = stringWidth + 17;
x += trueStringWidth;
ecWord++;
}
y = 8 * (sPhraseFrameDimensions[frameId].top + cursorRow * 2);
SetMainCursorPos(x, y + 8);
return FALSE;
}
static bool8 UpdateMainCursorOnButtons(void)
{
u8 xOffset = GetFooterOptionXOffset(GetMainCursorColumn());
SetMainCursorPos(xOffset, 96);
return FALSE;
}
static bool8 ShowConfirmExitPrompt(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_CONFIRM_EXIT);
CreateEasyChatYesNoMenu(1);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowConfirmPrompt(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_CONFIRM);
CreateEasyChatYesNoMenu(0);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowConfirmDeleteAllPrompt(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_CONFIRM_DELETE);
CreateEasyChatYesNoMenu(1);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ClosePrompt(void)
{
switch (sScreenControl->funcState)
{
case 0:
StartMainCursorAnim();
PrintEasyChatStdMessage(MSG_INSTRUCTIONS);
PrintCurrentPhrase();
ShowBg(0);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ClosePromptAfterDeleteAll(void)
{
switch (sScreenControl->funcState)
{
case 0:
StartMainCursorAnim();
PrintEasyChatStdMessage(MSG_INSTRUCTIONS);
PrintCurrentPhrase();
sScreenControl->funcState++;
// Fall through
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 OpenKeyboard(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
HideBg(0);
SetWindowDimensions(0, 0, 0, 0);
PrintKeyboardText();
sScreenControl->funcState++;
break;
case 1:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowAnim(WINANIM_OPEN_KEYBOARD);
sScreenControl->funcState++;
}
break;
case 2:
if (!IsDma3ManagerBusyWithBgCopy() && !UpdateLowerWindowAnim())
sScreenControl->funcState++;
break;
case 3:
if (!IsDma3ManagerBusyWithBgCopy())
{
CreateSideWindowSprites();
sScreenControl->funcState++;
}
break;
case 4:
if (!ShowSideWindow())
{
CreateRectangleCursorSprites();
SetScrollIndicatorXPos(FALSE);
UpdateScrollIndicatorsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
default:
return FALSE;
}
return TRUE;
}
static bool8 CloseKeyboard(void)
{
switch (sScreenControl->funcState)
{
case 0:
DestroyRectangleCursorSprites();
HideModeWindow();
HideScrollIndicators();
sScreenControl->funcState++;
break;
case 1:
if (DestroySideWindowSprites() == TRUE)
break;
InitLowerWindowAnim(WINANIM_CLOSE_KEYBOARD);
sScreenControl->funcState++;
// Fall through
case 2:
if (!UpdateLowerWindowAnim())
sScreenControl->funcState++;
break;
case 3:
if (!IsDma3ManagerBusyWithBgCopy())
{
StartMainCursorAnim();
ShowBg(0);
sScreenControl->funcState++;
}
break;
case 4:
return FALSE;
}
return TRUE;
}
static bool8 SwitchKeyboardMode(void)
{
switch (sScreenControl->funcState)
{
case 0:
DestroyRectangleCursorSprites();
HideScrollIndicators();
SetModeWindowToTransition();
InitLowerWindowAnim(WINANIM_KEYBOARD_SWITCH_OUT);
sScreenControl->funcState++;
break;
case 1:
if (!UpdateLowerWindowAnim() && !IsModeWindowAnimActive())
{
PrintKeyboardText();
sScreenControl->funcState++;
}
break;
case 2:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowAnim(WINANIM_KEYBOARD_SWITCH_IN);
UpdateModeWindowAnim();
sScreenControl->funcState++;
}
break;
case 3:
if (!UpdateLowerWindowAnim() && !IsModeWindowAnimActive())
{
UpdateScrollIndicatorsVisibility();
CreateRectangleCursorSprites();
sScreenControl->funcState++;
return FALSE;
}
break;
case 4:
return FALSE;
}
return TRUE;
}
static bool8 UpdateKeyboardCursor(void)
{
UpdateRectangleCursorPos();
return FALSE;
}
static bool8 GroupNamesScrollDown(void)
{
switch (sScreenControl->funcState)
{
case 0:
InitLowerWindowScroll(1, 4);
sScreenControl->funcState++;
// Fall through
case 1:
if (!UpdateLowerWindowScroll())
{
UpdateRectangleCursorPos();
UpdateScrollIndicatorsVisibility();
return FALSE;
}
break;
}
return TRUE;
}
static bool8 GroupNamesScrollUp(void)
{
switch (sScreenControl->funcState)
{
case 0:
InitLowerWindowScroll(-1, 4);
sScreenControl->funcState++;
// Fall through
case 1:
if (!UpdateLowerWindowScroll())
{
UpdateScrollIndicatorsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
case 2:
return FALSE;
}
return TRUE;
}
static bool8 OpenWordSelect(void)
{
switch (sScreenControl->funcState)
{
case 0:
DestroyRectangleCursorSprites();
HideModeWindow();
HideScrollIndicators();
sScreenControl->funcState++;
break;
case 1:
if (!DestroySideWindowSprites())
{
ClearWordSelectWindow();
sScreenControl->funcState++;
}
break;
case 2:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowAnim(WINANIM_OPEN_WORD_SELECT);
sScreenControl->funcState++;
}
break;
case 3:
if (!UpdateLowerWindowAnim())
{
InitLowerWindowText(TEXT_WORD_SELECT);
sScreenControl->funcState++;
}
break;
case 4:
if (!IsDma3ManagerBusyWithBgCopy())
{
CreateWordSelectCursorSprite();
SetScrollIndicatorXPos(TRUE);
UpdateScrollIndicatorsVisibility();
UpdateStartSelectButtonsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
case 5:
return FALSE;
}
return TRUE;
}
static bool8 CloseWordSelect(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintCurrentPhrase();
sScreenControl->funcState++;
break;
case 1:
DestroyWordSelectCursorSprite();
HideScrollIndicators();
HideStartSelectButtons();
ClearWordSelectWindow();
sScreenControl->funcState++;
break;
case 2:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowAnim(WINANIM_CLOSE_WORD_SELECT);
sScreenControl->funcState++;
}
break;
case 3:
if (!UpdateLowerWindowAnim())
{
ShowBg(0);
sScreenControl->funcState++;
}
break;
case 4:
if (!IsDma3ManagerBusyWithBgCopy())
{
StartMainCursorAnim();
sScreenControl->funcState++;
return FALSE;
}
break;
case 5:
return FALSE;
}
return TRUE;
}
static bool8 ShowConfirmLyricsPrompt(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintCurrentPhrase();
sScreenControl->funcState++;
break;
case 1:
DestroyWordSelectCursorSprite();
HideScrollIndicators();
HideStartSelectButtons();
ClearWordSelectWindow();
sScreenControl->funcState++;
break;
case 2:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowAnim(WINANIM_CLOSE_WORD_SELECT);
sScreenControl->funcState++;
}
break;
case 3:
if (!UpdateLowerWindowAnim())
{
PrintEasyChatStdMessage(MSG_CONFIRM);
sScreenControl->funcState++;
}
break;
case 4:
if (!IsDma3ManagerBusyWithBgCopy())
{
ShowBg(0);
sScreenControl->funcState++;
}
break;
case 5:
if (!IsDma3ManagerBusyWithBgCopy())
{
StartMainCursorAnim();
sScreenControl->funcState++;
return FALSE;
}
break;
case 6:
return FALSE;
}
return TRUE;
}
static bool8 ReturnToKeyboard(void)
{
switch (sScreenControl->funcState)
{
case 0:
DestroyWordSelectCursorSprite();
HideScrollIndicators();
HideStartSelectButtons();
ClearWordSelectWindow();
sScreenControl->funcState++;
break;
case 1:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowAnim(WINANIM_RETURN_TO_KEYBOARD);
sScreenControl->funcState++;
}
break;
case 2:
if (!UpdateLowerWindowAnim())
{
PrintKeyboardText();
sScreenControl->funcState++;
}
break;
case 3:
if (!IsDma3ManagerBusyWithBgCopy())
{
CreateSideWindowSprites();
sScreenControl->funcState++;
}
break;
case 4:
if (!ShowSideWindow())
{
CreateRectangleCursorSprites();
SetScrollIndicatorXPos(FALSE);
UpdateScrollIndicatorsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
}
return TRUE;
}
static bool8 UpdateWordSelectCursor(void)
{
UpdateWordSelectCursorPos();
return FALSE;
}
static bool8 WordSelectScrollDown(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintWordSelectNextRowDown();
sScreenControl->funcState++;
break;
case 1:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowScroll(1, 4);
sScreenControl->funcState++;
}
break;
case 2:
if (!UpdateLowerWindowScroll())
{
UpdateWordSelectCursorPos();
UpdateScrollIndicatorsVisibility();
UpdateStartSelectButtonsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
case 3:
return FALSE;
}
return TRUE;
}
static bool8 WordSelectScrollUp(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintWordSelectNextRowUp();
sScreenControl->funcState++;
break;
case 1:
if (!IsDma3ManagerBusyWithBgCopy())
{
InitLowerWindowScroll(-1, 4);
sScreenControl->funcState++;
}
break;
case 2:
if (!UpdateLowerWindowScroll())
{
UpdateScrollIndicatorsVisibility();
UpdateStartSelectButtonsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
case 3:
return FALSE;
}
return TRUE;
}
static bool8 WordSelectPageScrollDown(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintWordSelectRowsPageDown();
sScreenControl->funcState++;
break;
case 1:
if (!IsDma3ManagerBusyWithBgCopy())
{
s16 scrollChange = GetWordSelectScrollOffset() - GetLowerWindowScrollOffset();
InitLowerWindowScroll(scrollChange, 8);
sScreenControl->funcState++;
}
break;
case 2:
if (!UpdateLowerWindowScroll())
{
UpdateWordSelectCursorPos();
UpdateScrollIndicatorsVisibility();
UpdateStartSelectButtonsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
case 3:
return FALSE;
}
return TRUE;
}
static bool8 WordSelectPageScrollUp(void)
{
switch (sScreenControl->funcState)
{
case 0:
PrintWordSelectRowsPageUp();
sScreenControl->funcState++;
break;
case 1:
if (!IsDma3ManagerBusyWithBgCopy())
{
s16 scrollChange = GetWordSelectScrollOffset() - GetLowerWindowScrollOffset();
InitLowerWindowScroll(scrollChange, 8);
sScreenControl->funcState++;
}
break;
case 2:
if (!UpdateLowerWindowScroll())
{
UpdateScrollIndicatorsVisibility();
UpdateStartSelectButtonsVisibility();
sScreenControl->funcState++;
return FALSE;
}
break;
case 3:
return FALSE;
}
return TRUE;
}
static bool8 ShowCreateQuizMsg(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_CREATE_QUIZ);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowSelectAnswerMsg(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_SELECT_ANSWER);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowSongTooShortMsg(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_SONG_TOO_SHORT);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowCantDeleteLyricsMsg(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_CANT_DELETE_LYRICS);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowCombineTwoWordsMsg(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_COMBINE_TWO_WORDS);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 ShowCantExitMsg(void)
{
switch (sScreenControl->funcState)
{
case 0:
StopMainCursorAnim();
PrintEasyChatStdMessage(MSG_CANT_QUIT);
sScreenControl->funcState++;
break;
case 1:
return IsDma3ManagerBusyWithBgCopy();
}
return TRUE;
}
static bool8 InitEasyChatScreenControl_(void)
{
sScreenControl = Alloc(sizeof(*sScreenControl));
if (!sScreenControl)
return FALSE;
sScreenControl->funcState = 0;
sScreenControl->mainCursorSprite = NULL;
sScreenControl->rectangleCursorSpriteRight = NULL;
sScreenControl->rectangleCursorSpriteLeft = NULL;
sScreenControl->wordSelectCursorSprite = NULL;
sScreenControl->buttonWindowSprite = NULL;
sScreenControl->modeWindowSprite = NULL;
sScreenControl->scrollIndicatorUpSprite = NULL;
sScreenControl->scrollIndicatorDownSprite = NULL;
sScreenControl->startButtonSprite = NULL;
sScreenControl->selectButtonSprite = NULL;
sScreenControl->fourFooterOptions = FooterHasFourOptions_();
return TRUE;
}
static void InitEasyChatBgs(void)
{
ChangeBgX(3, 0, 0);
ChangeBgY(3, 0, 0);
ChangeBgX(1, 0, 0);
ChangeBgY(1, 0, 0);
ChangeBgX(2, 0, 0);
ChangeBgY(2, 0, 0);
ChangeBgX(0, 0, 0);
ChangeBgY(0, 0, 0);
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 | DISPCNT_OBJ_1D_MAP | DISPCNT_OBJ_ON | DISPCNT_WIN0_ON);
}
static void LoadEasyChatPalettes(void)
{
ResetPaletteFade();
LoadPalette(gEasyChatMode_Pal, 0, 32);
LoadPalette(sTextInputFrameOrange_Pal, 1 * 16, 32);
LoadPalette(sTextInputFrameGreen_Pal, 4 * 16, 32);
LoadPalette(sUnknown_08597C1C, 10 * 16, 8);
LoadPalette(sUnknown_08597C24, 11 * 16, 12);
LoadPalette(sUnknown_08597C24, 15 * 16, 12);
LoadPalette(sUnknown_08597C24, 3 * 16, 12);
}
static void PrintTitle(void)
{
int xOffset;
const u8 *titleText = GetTitleText();
if (!titleText)
return;
xOffset = GetStringCenterAlignXOffset(1, titleText, 144);
FillWindowPixelBuffer(0, PIXEL_FILL(0));
PrintEasyChatTextWithColors(0, 1, titleText, xOffset, 1, TEXT_SPEED_FF, TEXT_COLOR_TRANSPARENT, TEXT_COLOR_DARK_GREY, TEXT_COLOR_LIGHT_GREY);
PutWindowTilemap(0);
CopyWindowToVram(0, 3);
}
static void PrintEasyChatText(u8 windowId, u8 fontId, const u8 *str, u8 x, u8 y, u8 speed, void (*callback)(struct TextPrinterTemplate *, u16))
{
AddTextPrinterParameterized(windowId, fontId, str, x, y, speed, callback);
}
static void PrintEasyChatTextWithColors(u8 windowId, u8 fontId, const u8 *str, u8 left, u8 top, u8 speed, u8 bg, u8 fg, u8 shadow)
{
u8 color[3];
color[0] = bg;
color[1] = fg;
color[2] = shadow;
AddTextPrinterParameterized3(windowId, fontId, left, top, color, speed, str);
}
static void PrintInitialInstructions(void)
{
FillBgTilemapBufferRect(0, 0, 0, 0, 32, 20, 17);
LoadUserWindowBorderGfx(1, 1, 0xE0);
DrawTextBorderOuter(1, 1, 14);
PrintEasyChatStdMessage(MSG_INSTRUCTIONS);
PutWindowTilemap(1);
CopyBgTilemapBufferToVram(0);
}
static void PrintEasyChatStdMessage(u8 msgId)
{
const u8 *text2 = NULL;
const u8 *text1 = NULL;
switch (msgId)
{
case MSG_INSTRUCTIONS:
GetEasyChatInstructionsText(&text1, &text2);
break;
case MSG_CONFIRM_EXIT:
GetEasyChatConfirmExitText(&text1, &text2);
break;
case MSG_CONFIRM:
GetEasyChatConfirmText(&text1, &text2);
break;
case MSG_CONFIRM_DELETE:
GetEasyChatConfirmDeletionText(&text1, &text2);
break;
case MSG_CREATE_QUIZ:
text1 = gText_CreateAQuiz;
break;
case MSG_SELECT_ANSWER:
text1 = gText_SelectTheAnswer;
break;
case MSG_SONG_TOO_SHORT:
text1 = gText_OnlyOnePhrase;
text2 = gText_OriginalSongWillBeUsed;
break;
case MSG_CANT_DELETE_LYRICS:
text1 = gText_LyricsCantBeDeleted;
break;
case MSG_COMBINE_TWO_WORDS:
text1 = gText_CombineTwoWordsOrPhrases3;
break;
case MSG_CANT_QUIT:
text1 = gText_YouCannotQuitHere;
text2 = gText_SectionMustBeCompleted;
break;
}
FillWindowPixelBuffer(1, PIXEL_FILL(1));
if (text1)
PrintEasyChatText(1, 1, text1, 0, 1, TEXT_SPEED_FF, 0);
if (text2)
PrintEasyChatText(1, 1, text2, 0, 17, TEXT_SPEED_FF, 0);
CopyWindowToVram(1, 3);
}
static void CreateEasyChatYesNoMenu(u8 initialCursorPos)
{
CreateYesNoMenu(&sEasyChatYesNoWindowTemplate, 1, 14, initialCursorPos);
}
static void AddPhraseWindow(void)
{
u8 frameId;
struct WindowTemplate template;
frameId = GetEasyChatScreenFrameId();
template.bg = 3;
template.tilemapLeft = sPhraseFrameDimensions[frameId].left;
template.tilemapTop = sPhraseFrameDimensions[frameId].top;
template.width = sPhraseFrameDimensions[frameId].width;
template.height = sPhraseFrameDimensions[frameId].height;
template.paletteNum = 11;
template.baseBlock = 0x6C;
sScreenControl->windowId = AddWindow(&template);
PutWindowTilemap(sScreenControl->windowId);
}
static void PrintCurrentPhrase(void)
{
u8 strClear[4];
u16 *currentPhrase;
u8 numColumns, numRows;
u8 *str;
int frameId;
bool32 isQuizQuestion;
int i, j, k;
currentPhrase = GetCurrentPhrase();
numColumns = GetNumColumns();
numRows = GetNumRows();
frameId = GetEasyChatScreenFrameId();
isQuizQuestion = FALSE;
if (frameId == FRAMEID_QUIZ_QUESTION)
isQuizQuestion = TRUE;
FillWindowPixelBuffer(sScreenControl->windowId, PIXEL_FILL(1));
for (i = 0; i < numRows; i++)
{
memcpy(strClear, sText_Clear17, sizeof(sText_Clear17));
if (isQuizQuestion)
strClear[2] = 6;
str = sScreenControl->phrasePrintBuffer;
sScreenControl->phrasePrintBuffer[0] = EOS;
str = StringAppend(str, strClear);
for (j = 0; j < numColumns; j++)
{
if (*currentPhrase != EC_EMPTY_WORD)
{
str = CopyEasyChatWord(str, *currentPhrase);
currentPhrase++;
}
else
{
currentPhrase++;
if (!isQuizQuestion)
{
str = WriteColorChangeControlCode(str, 0, 4);
for (k = 0; k < 12; k++)
{
*str = CHAR_HYPHEN;
str++;
}
str = WriteColorChangeControlCode(str, 0, 2);
}
}
if (isQuizQuestion)
strClear[2] = 3;
str = StringAppend(str, strClear);
if (frameId == FRAMEID_MAIL || frameId == FRAMEID_QUIZ_QUESTION || frameId == FRAMEID_QUIZ_SET_QUESTION)
{
// Is 2x5 frame, end on 9th word
if (j == 0 && i == 4)
break;
}
}
*str = EOS;
PrintEasyChatText(sScreenControl->windowId, 1, sScreenControl->phrasePrintBuffer, 0, i * 16 + 1, TEXT_SPEED_FF, 0);
}
CopyWindowToVram(sScreenControl->windowId, 3);
}
static void BufferFrameTilemap(u16 *tilemap)
{
u8 frameId;
int right, bottom;
int x, y;
frameId = GetEasyChatScreenFrameId();
CpuFastFill(0, tilemap, BG_SCREEN_SIZE);
if (frameId == FRAMEID_MAIL || frameId == FRAMEID_QUIZ_SET_QUESTION)
{
// These frames fill the screen, no need to draw top/bottom edges
right = sPhraseFrameDimensions[frameId].left + sPhraseFrameDimensions[frameId].width;
bottom = sPhraseFrameDimensions[frameId].top + sPhraseFrameDimensions[frameId].height;
// Draw middle section
for (y = sPhraseFrameDimensions[frameId].top; y < bottom; y++)
{
x = sPhraseFrameDimensions[frameId].left - 1;
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_L_EDGE;
x++;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_TRANSPARENT;
tilemap[y* 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_R_EDGE;
}
}
else
{
y = sPhraseFrameDimensions[frameId].top - 1;
x = sPhraseFrameDimensions[frameId].left - 1;
right = sPhraseFrameDimensions[frameId].left + sPhraseFrameDimensions[frameId].width;
bottom = sPhraseFrameDimensions[frameId].top + sPhraseFrameDimensions[frameId].height;
// Draw top edge
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_TOP_L_CORNER;
x++;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_TOP_EDGE;
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_TOP_R_CORNER;
y++;
// Draw middle section
for (; y < bottom; y++)
{
x = sPhraseFrameDimensions[frameId].left - 1;
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_L_EDGE;
x++;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_TRANSPARENT;
tilemap[y* 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_R_EDGE;
}
// Draw bottom edge
x = sPhraseFrameDimensions[frameId].left - 1;
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_BOTTOM_L_CORNER;
x++;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_BOTTOM_EDGE;
tilemap[y * 32 + x] = FRAME_OFFSET_ORANGE + FRAME_TILE_BOTTOM_R_CORNER;
}
}
static void AdjustBgTilemapForFooter(void)
{
u8 frameId;
u16 *tilemap;
tilemap = GetBgTilemapBuffer(3);
frameId = GetEasyChatScreenFrameId();
switch (sPhraseFrameDimensions[frameId].footerId)
{
case FOOTER_ANSWER:
tilemap += 0x2A0;
CopyToBgTilemapBufferRect(3, tilemap, 0, 11, 32, 2);
break;
case FOOTER_QUIZ:
tilemap += 0x300;
CopyToBgTilemapBufferRect(3, tilemap, 0, 11, 32, 2);
break;
case NUM_FOOTER_TYPES:
CopyToBgTilemapBufferRect(3, tilemap, 0, 10, 32, 4);
break;
}
}
static void DrawLowerWindow(void)
{
PutWindowTilemap(2);
CopyBgTilemapBufferToVram(2);
}
static void InitLowerWindowText(u32 whichText)
{
ResetLowerWindowScroll();
FillWindowPixelBuffer(2, PIXEL_FILL(1));
switch (whichText)
{
case TEXT_GROUPS:
PrintKeyboardGroupNames();
break;
case TEXT_ALPHABET:
PrintKeyboardAlphabet();
break;
case TEXT_WORD_SELECT:
PrintInitialWordSelectText();
break;
}
CopyWindowToVram(2, 2);
}
static void PrintKeyboardText(void)
{
if (!GetInAlphabetMode())
InitLowerWindowText(TEXT_GROUPS);
else
InitLowerWindowText(TEXT_ALPHABET);
}
static void PrintKeyboardGroupNames(void)
{
int i;
int x, y;
i = 0;
y = 97;
while (1)
{
for (x = 0; x < 2; x++)
{
u8 groupId = GetUnlockedEasyChatGroupId(i++);
if (groupId == EC_NUM_GROUPS)
{
InitLowerWindowScroll(GetKeyboardScrollOffset(), 0);
return;
}
PrintEasyChatText(2, 1, GetEasyChatWordGroupName(groupId), x * 84 + 10, y, TEXT_SPEED_FF, NULL);
}
y += 16;
}
}
static void PrintKeyboardAlphabet(void)
{
u32 i;
for (i = 0; i < ARRAY_COUNT(sEasyChatKeyboardAlphabet); i++)
PrintEasyChatText(2, 1, sEasyChatKeyboardAlphabet[i], 10, 97 + i * 16, TEXT_SPEED_FF, NULL);
}
static void PrintInitialWordSelectText(void)
{
PrintWordSelectText(0, NUM_WORD_SELECT_ROWS);
}
static void PrintWordSelectNextRowDown(void)
{
u8 wordScroll = GetWordSelectScrollOffset() + NUM_WORD_SELECT_ROWS - 1;
EraseWordSelectRows(wordScroll, 1);
PrintWordSelectText(wordScroll, 1);
}
static void PrintWordSelectNextRowUp(void)
{
u8 wordScroll = GetWordSelectScrollOffset();
EraseWordSelectRows(wordScroll, 1);
PrintWordSelectText(wordScroll, 1);
}
static void PrintWordSelectRowsPageDown(void)
{
u8 wordScroll = GetWordSelectScrollOffset();
u8 maxScroll = wordScroll + NUM_WORD_SELECT_ROWS;
u8 maxRows = GetWordSelectLastRow() + 1;
if (maxScroll > maxRows)
maxScroll = maxRows;
if (wordScroll < maxScroll)
{
u8 numRows = maxScroll - wordScroll;
EraseWordSelectRows(wordScroll, numRows);
PrintWordSelectText(wordScroll, numRows);
}
}
static void PrintWordSelectRowsPageUp(void)
{
u8 wordScroll = GetWordSelectScrollOffset();
u8 windowScroll = GetLowerWindowScrollOffset();
if (wordScroll < windowScroll)
{
u8 numRows = windowScroll - wordScroll;
EraseWordSelectRows(wordScroll, numRows);
PrintWordSelectText(wordScroll, numRows);
}
}
// Print the easy chat words available for selection in
// the currently selected group and at the given offset and row
static void PrintWordSelectText(u8 scrollOffset, u8 numRows)
{
int i, j;
u16 easyChatWord;
int y;
int wordIndex;
wordIndex = scrollOffset * NUM_WORD_SELECT_COLUMNS;
y = (scrollOffset * 16 + 96) & 0xFF;
y++;
for (i = 0; i < numRows; i++)
{
for (j = 0; j < 2; j++)
{
easyChatWord = GetWordFromSelectedGroup(wordIndex++);
if (easyChatWord != EC_EMPTY_WORD)
{
CopyEasyChatWordPadded(sScreenControl->wordSelectPrintBuffer, easyChatWord, 0);
if (!DummyWordCheck(easyChatWord))
PrintEasyChatText(2, 1, sScreenControl->wordSelectPrintBuffer, (j * 13 + 3) * 8, y, TEXT_SPEED_FF, NULL);
else // Never reached
PrintEasyChatTextWithColors(2, 1, sScreenControl->wordSelectPrintBuffer, (j * 13 + 3) * 8, y, TEXT_SPEED_FF, TEXT_COLOR_WHITE, TEXT_COLOR_LIGHT_RED, TEXT_COLOR_LIGHT_GREY);
}
}
y += 16;
}
CopyWindowToVram(2, 2);
}
static void EraseWordSelectRows(u8 scrollOffset, u8 numRows)
{
int y;
int var0;
int var1;
int var2;
y = (scrollOffset * 16 + 96) & 0xFF;
var2 = numRows * 16;
var0 = y + var2;
if (var0 > 255)
{
var1 = var0 - 256;
var2 = 256 - y;
}
else
{
var1 = 0;
}
FillWindowPixelRect(2, PIXEL_FILL(1), 0, y, 224, var2);
if (var1)
FillWindowPixelRect(2, PIXEL_FILL(1), 0, 0, 224, var1);
}
static void ClearWordSelectWindow(void)
{
FillWindowPixelBuffer(2, PIXEL_FILL(1));
CopyWindowToVram(2, 2);
}
static void InitLowerWindowAnim(int winAnimType)
{
switch (winAnimType)
{
case WINANIM_OPEN_KEYBOARD:
sScreenControl->curWindowAnimState = 0;
sScreenControl->destWindowAnimState = 10;
break;
case WINANIM_CLOSE_KEYBOARD:
sScreenControl->curWindowAnimState = 9;
sScreenControl->destWindowAnimState = 0;
break;
case WINANIM_OPEN_WORD_SELECT:
sScreenControl->curWindowAnimState = 11;
sScreenControl->destWindowAnimState = 17;
break;
case WINANIM_CLOSE_WORD_SELECT:
sScreenControl->curWindowAnimState = 17;
sScreenControl->destWindowAnimState = 0;
break;
case WINANIM_RETURN_TO_KEYBOARD:
sScreenControl->curWindowAnimState = 17;
sScreenControl->destWindowAnimState = 10;
break;
case WINANIM_KEYBOARD_SWITCH_OUT:
sScreenControl->curWindowAnimState = 18;
sScreenControl->destWindowAnimState = 22;
break;
case WINANIM_KEYBOARD_SWITCH_IN:
sScreenControl->curWindowAnimState = 22;
sScreenControl->destWindowAnimState = 18;
break;
}
sScreenControl->windowAnimStateDir = sScreenControl->curWindowAnimState < sScreenControl->destWindowAnimState ? 1 : -1;
}
// Returns FALSE if the anim is finished
static bool8 UpdateLowerWindowAnim(void)
{
u8 curState, destState;
if (sScreenControl->curWindowAnimState == sScreenControl->destWindowAnimState)
return FALSE;
sScreenControl->curWindowAnimState += sScreenControl->windowAnimStateDir;
DrawLowerWindowFrame(sScreenControl->curWindowAnimState);
curState = sScreenControl->curWindowAnimState;
destState = sScreenControl->destWindowAnimState;
return (curState ^ destState) > 0;
}
// States in this function are used incrementally with differing start/end cases
// to draw the lower window and create the appearance that it's opening/closing/animating.
// See InitLowerWindowAnim
static void DrawLowerWindowFrame(u8 type)
{
FillBgTilemapBufferRect_Palette0(1, 0, 0, 10, 30, 10);
switch (type)
{
case 0: // Closed
break;
case 1:
BufferLowerWindowFrame(11, 14, 3, 2);
break;
case 2:
BufferLowerWindowFrame(9, 14, 7, 2);
break;
case 3:
BufferLowerWindowFrame(7, 14, 11, 2);
break;
case 4:
BufferLowerWindowFrame(5, 14, 15, 2);
break;
case 5:
BufferLowerWindowFrame(3, 14, 19, 2);
break;
case 6:
BufferLowerWindowFrame(1, 14, 23, 2);
break;
case 7:
BufferLowerWindowFrame(1, 13, 23, 4);
break;
case 8:
BufferLowerWindowFrame(1, 12, 23, 6);
break;
case 9:
BufferLowerWindowFrame(1, 11, 23, 8);
break;
case 10:
BufferLowerWindowFrame(1, 10, 23, 10);
break;
case 11:
BufferLowerWindowFrame(1, 10, 24, 10);
break;
case 12:
BufferLowerWindowFrame(1, 10, 25, 10);
break;
case 13:
BufferLowerWindowFrame(1, 10, 26, 10);
break;
case 14:
BufferLowerWindowFrame(1, 10, 27, 10);
break;
case 15:
BufferLowerWindowFrame(1, 10, 28, 10);
break;
case 16:
BufferLowerWindowFrame(1, 10, 29, 10);
break;
case 17:
BufferLowerWindowFrame(0, 10, 30, 10);
break;
case 18:
BufferLowerWindowFrame(1, 10, 23, 10);
break;
case 19:
BufferLowerWindowFrame(1, 11, 23, 8);
break;
case 20:
BufferLowerWindowFrame(1, 12, 23, 6);
break;
case 21:
BufferLowerWindowFrame(1, 13, 23, 4);
break;
case 22:
BufferLowerWindowFrame(1, 14, 23, 2);
break;
}
CopyBgTilemapBufferToVram(1);
}
static void BufferLowerWindowFrame(int left, int top, int width, int height)
{
u16 *tilemap;
int right;
int bottom;
int x, y;
tilemap = sScreenControl->bg1TilemapBuffer;
right = left + width - 1;
bottom = top + height - 1;
x = left;
y = top;
// Draw top edge
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_TOP_L_CORNER;
x++;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_TOP_EDGE;
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_TOP_R_CORNER;
y++;
// Draw middle section
for (; y < bottom; y++)
{
tilemap[y * 32 + left] = FRAME_OFFSET_GREEN + FRAME_TILE_L_EDGE;
x = left + 1;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_TRANSPARENT;
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_R_EDGE;
}
// Draw bottom edge
tilemap[y * 32 + left] = FRAME_OFFSET_GREEN + FRAME_TILE_BOTTOM_L_CORNER;
x = left + 1;
for (; x < right; x++)
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_BOTTOM_EDGE;
tilemap[y * 32 + x] = FRAME_OFFSET_GREEN + FRAME_TILE_BOTTOM_R_CORNER;
SetWindowDimensions((left + 1) * 8, (top + 1) * 8, (width - 2) * 8, (height - 2) * 8);
}
static void ResetLowerWindowScroll(void)
{
ChangeBgY(2, 0x800, 0);
sScreenControl->scrollOffset = 0;
}
static void InitLowerWindowScroll(s16 scrollChange, u8 speed)
{
int bgY;
s16 yChange;
bgY = GetBgY(2);
sScreenControl->scrollOffset += scrollChange;
yChange = scrollChange * 16;
bgY += yChange * 256;
if (speed)
{
sScreenControl->scrollDest = bgY;
sScreenControl->scrollSpeed = speed * 256;
if (yChange < 0)
sScreenControl->scrollSpeed = -sScreenControl->scrollSpeed;
}
else
{
ChangeBgY(2, bgY, 0);
}
}
static bool8 UpdateLowerWindowScroll(void)
{
int bgY;
bgY = GetBgY(2);
if (bgY == sScreenControl->scrollDest)
{
return FALSE;
}
else
{
ChangeBgY(2, sScreenControl->scrollSpeed, 1);
return TRUE;
}
}
static int GetLowerWindowScrollOffset(void)
{
return sScreenControl->scrollOffset;
}
static void SetWindowDimensions(u8 left, u8 top, u8 width, u8 height)
{
u16 horizontalDimensions = WIN_RANGE(left, left + width);
u16 verticalDimensions = WIN_RANGE(top, top + height);
SetGpuReg(REG_OFFSET_WIN0H, horizontalDimensions);
SetGpuReg(REG_OFFSET_WIN0V, verticalDimensions);
}
static void LoadEasyChatGfx(void)
{
u32 i;
LoadSpriteSheets(sSpriteSheets);
LoadSpritePalettes(sSpritePalettes);
for (i = 0; i < ARRAY_COUNT(sCompressedSpriteSheets); i++)
LoadCompressedSpriteSheet(&sCompressedSpriteSheets[i]);
}
#define sDelayTimer data[0]
#define sAnimateCursor data[1]
static void CreateMainCursorSprite(void)
{
u8 frameId = GetEasyChatScreenFrameId();
int x = sPhraseFrameDimensions[frameId].left * 8 + 13;
int y = sPhraseFrameDimensions[frameId].top * 8 + 8;
u8 spriteId = CreateSprite(&sSpriteTemplate_TriangleCursor, x, y, 2);
sScreenControl->mainCursorSprite = &gSprites[spriteId];
gSprites[spriteId].sAnimateCursor = TRUE;
}
static void SpriteCB_Cursor(struct Sprite *sprite)
{
if (sprite->sAnimateCursor)
{
if (++sprite->sDelayTimer > 2)
{
sprite->sDelayTimer = 0;
if (++sprite->pos2.x > 0)
sprite->pos2.x = -6;
}
}
}
static void SetMainCursorPos(u8 x, u8 y)
{
sScreenControl->mainCursorSprite->pos1.x = x;
sScreenControl->mainCursorSprite->pos1.y = y;
sScreenControl->mainCursorSprite->pos2.x = 0;
sScreenControl->mainCursorSprite->sDelayTimer = 0;
}
static void StopMainCursorAnim(void)
{
sScreenControl->mainCursorSprite->sDelayTimer = 0;
sScreenControl->mainCursorSprite->sAnimateCursor = FALSE;
sScreenControl->mainCursorSprite->pos2.x = 0;
}
static void StartMainCursorAnim(void)
{
sScreenControl->mainCursorSprite->sAnimateCursor = TRUE;
}
static void CreateRectangleCursorSprites(void)
{
u8 spriteId = CreateSprite(&sSpriteTemplate_RectangleCursor, 0, 0, 3);
sScreenControl->rectangleCursorSpriteRight = &gSprites[spriteId];
sScreenControl->rectangleCursorSpriteRight->pos2.x = 32;
spriteId = CreateSprite(&sSpriteTemplate_RectangleCursor, 0, 0, 3);
sScreenControl->rectangleCursorSpriteLeft = &gSprites[spriteId];
sScreenControl->rectangleCursorSpriteLeft->pos2.x = -32;
sScreenControl->rectangleCursorSpriteRight->hFlip = TRUE;
UpdateRectangleCursorPos();
}
static void DestroyRectangleCursorSprites(void)
{
DestroySprite(sScreenControl->rectangleCursorSpriteRight);
sScreenControl->rectangleCursorSpriteRight = NULL;
DestroySprite(sScreenControl->rectangleCursorSpriteLeft);
sScreenControl->rectangleCursorSpriteLeft = NULL;
}
static void UpdateRectangleCursorPos(void)
{
s8 column;
s8 row;
if (sScreenControl->rectangleCursorSpriteRight
&& sScreenControl->rectangleCursorSpriteLeft)
{
GetKeyboardCursorColAndRow(&column, &row);
if (!GetInAlphabetMode())
SetRectangleCursorPos_GroupMode(column, row);
else
SetRectangleCursorPos_AlphabetMode(column, row);
}
}
static void SetRectangleCursorPos_GroupMode(s8 column, s8 row)
{
if (column != -1)
{
// In group name window
StartSpriteAnim(sScreenControl->rectangleCursorSpriteRight, RECTCURSOR_ANIM_ON_GROUP);
sScreenControl->rectangleCursorSpriteRight->pos1.x = column * 84 + 58;
sScreenControl->rectangleCursorSpriteRight->pos1.y = row * 16 + 96;
StartSpriteAnim(sScreenControl->rectangleCursorSpriteLeft, RECTCURSOR_ANIM_ON_GROUP);
sScreenControl->rectangleCursorSpriteLeft->pos1.x = column * 84 + 58;
sScreenControl->rectangleCursorSpriteLeft->pos1.y = row * 16 + 96;
}
else
{
// In button window
StartSpriteAnim(sScreenControl->rectangleCursorSpriteRight, RECTCURSOR_ANIM_ON_BUTTON);
sScreenControl->rectangleCursorSpriteRight->pos1.x = 216;
sScreenControl->rectangleCursorSpriteRight->pos1.y = row * 16 + 112;
StartSpriteAnim(sScreenControl->rectangleCursorSpriteLeft, RECTCURSOR_ANIM_ON_BUTTON);
sScreenControl->rectangleCursorSpriteLeft->pos1.x = 216;
sScreenControl->rectangleCursorSpriteLeft->pos1.y = row * 16 + 112;
}
}
static void SetRectangleCursorPos_AlphabetMode(s8 column, s8 row)
{
int anim;
int x, y;
if (column != -1)
{
y = row * 16 + 96;
x = 32;
if (column == NUM_ALPHABET_COLUMNS - 1 && row == 0)
{
// Cursor is on 'Others'
x = 158;
anim = RECTCURSOR_ANIM_ON_OTHERS;
}
else
{
// Cursor is on a letter
x += sAlphabetKeyboardColumnOffsets[(u8)column < NUM_ALPHABET_COLUMNS ? column : 0];
anim = RECTCURSOR_ANIM_ON_LETTER;
}
StartSpriteAnim(sScreenControl->rectangleCursorSpriteRight, anim);
sScreenControl->rectangleCursorSpriteRight->pos1.x = x;
sScreenControl->rectangleCursorSpriteRight->pos1.y = y;
StartSpriteAnim(sScreenControl->rectangleCursorSpriteLeft, anim);
sScreenControl->rectangleCursorSpriteLeft->pos1.x = x;
sScreenControl->rectangleCursorSpriteLeft->pos1.y = y;
}
else
{
// In button window
StartSpriteAnim(sScreenControl->rectangleCursorSpriteRight, RECTCURSOR_ANIM_ON_BUTTON);
sScreenControl->rectangleCursorSpriteRight->pos1.x = 216;
sScreenControl->rectangleCursorSpriteRight->pos1.y = row * 16 + 112;
StartSpriteAnim(sScreenControl->rectangleCursorSpriteLeft, RECTCURSOR_ANIM_ON_BUTTON);
sScreenControl->rectangleCursorSpriteLeft->pos1.x = 216;
sScreenControl->rectangleCursorSpriteLeft->pos1.y = row * 16 + 112;
}
}
// Cursor for selecting a new word
// Identical in appearance to the 'main' cursor
static void CreateWordSelectCursorSprite(void)
{
u8 spriteId = CreateSprite(&sSpriteTemplate_TriangleCursor, 0, 0, 4);
sScreenControl->wordSelectCursorSprite = &gSprites[spriteId];
sScreenControl->wordSelectCursorSprite->callback = SpriteCB_WordSelectCursor;
sScreenControl->wordSelectCursorSprite->oam.priority = 2;
UpdateWordSelectCursorPos();
}
static void SpriteCB_WordSelectCursor(struct Sprite *sprite)
{
if (++sprite->sDelayTimer > 2)
{
sprite->sDelayTimer = 0;
if (++sprite->pos2.x > 0)
sprite->pos2.x = -6;
}
}
static void UpdateWordSelectCursorPos(void)
{
s8 column, row, x, y;
GetWordSelectColAndRow(&column, &row);
x = column * 13;
x = x * 8 + 28;
y = row * 16 + 96;
SetWordSelectCursorPos(x, y);
}
static void SetWordSelectCursorPos(u8 x, u8 y)
{
if (sScreenControl->wordSelectCursorSprite)
{
sScreenControl->wordSelectCursorSprite->pos1.x = x;
sScreenControl->wordSelectCursorSprite->pos1.y = y;
sScreenControl->wordSelectCursorSprite->pos2.x = 0;
sScreenControl->wordSelectCursorSprite->sDelayTimer = 0;
}
}
static void DestroyWordSelectCursorSprite(void)
{
if (sScreenControl->wordSelectCursorSprite)
{
DestroySprite(sScreenControl->wordSelectCursorSprite);
sScreenControl->wordSelectCursorSprite = NULL;
}
}
static void CreateSideWindowSprites(void)
{
u8 spriteId = CreateSprite(&sSpriteTemplate_ButtonWindow, 208, 128, 6);
sScreenControl->buttonWindowSprite = &gSprites[spriteId];
sScreenControl->buttonWindowSprite->pos2.x = -64;
spriteId = CreateSprite(&sSpriteTemplate_ModeWindow, 208, 80, 5);
sScreenControl->modeWindowSprite = &gSprites[spriteId];
sScreenControl->modeWindowState = 0;
}
static bool8 ShowSideWindow(void)
{
switch (sScreenControl->modeWindowState)
{
default:
return FALSE;
case 0:
// Slide button window on
sScreenControl->buttonWindowSprite->pos2.x += 8;
if (sScreenControl->buttonWindowSprite->pos2.x >= 0)
{
sScreenControl->buttonWindowSprite->pos2.x = 0;
// Set mode window anim
if (!GetInAlphabetMode())
StartSpriteAnim(sScreenControl->modeWindowSprite, MODEWINDOW_ANIM_TO_GROUP);
else
StartSpriteAnim(sScreenControl->modeWindowSprite, MODEWINDOW_ANIM_TO_ALPHABET);
sScreenControl->modeWindowState++;
}
break;
case 1:
if (sScreenControl->modeWindowSprite->animEnded)
{
sScreenControl->modeWindowState = 2;
return FALSE;
}
}
return TRUE;
}
static void HideModeWindow(void)
{
sScreenControl->modeWindowState = 0;
StartSpriteAnim(sScreenControl->modeWindowSprite, MODEWINDOW_ANIM_TO_HIDDEN);
}
static bool8 DestroySideWindowSprites(void)
{
switch (sScreenControl->modeWindowState)
{
default:
return FALSE;
case 0:
if (sScreenControl->modeWindowSprite->animEnded)
sScreenControl->modeWindowState = 1;
break;
case 1:
sScreenControl->buttonWindowSprite->pos2.x -= 8;
if (sScreenControl->buttonWindowSprite->pos2.x <= -64)
{
DestroySprite(sScreenControl->modeWindowSprite);
DestroySprite(sScreenControl->buttonWindowSprite);
sScreenControl->modeWindowSprite = NULL;
sScreenControl->buttonWindowSprite = NULL;
sScreenControl->modeWindowState++;
return FALSE;
}
}
return TRUE;
}
static void SetModeWindowToTransition(void)
{
StartSpriteAnim(sScreenControl->modeWindowSprite, MODEWINDOW_ANIM_TRANSITION);
}
static void UpdateModeWindowAnim(void)
{
if (!GetInAlphabetMode())
StartSpriteAnim(sScreenControl->modeWindowSprite, MODEWINDOW_ANIM_TO_GROUP);
else
StartSpriteAnim(sScreenControl->modeWindowSprite, MODEWINDOW_ANIM_TO_ALPHABET);
}
static bool8 IsModeWindowAnimActive(void)
{
return !sScreenControl->modeWindowSprite->animEnded;
}
static void CreateScrollIndicatorSprites(void)
{
u8 spriteId = CreateSprite(&sSpriteTemplate_ScrollIndicator, 96, 80, 0);
if (spriteId != MAX_SPRITES)
sScreenControl->scrollIndicatorUpSprite = &gSprites[spriteId];
spriteId = CreateSprite(&sSpriteTemplate_ScrollIndicator, 96, 156, 0);
if (spriteId != MAX_SPRITES)
{
sScreenControl->scrollIndicatorDownSprite = &gSprites[spriteId];
sScreenControl->scrollIndicatorDownSprite->vFlip = TRUE;
}
HideScrollIndicators();
}
static void UpdateScrollIndicatorsVisibility(void)
{
sScreenControl->scrollIndicatorUpSprite->invisible = !CanScrollUp();
sScreenControl->scrollIndicatorDownSprite->invisible = !CanScrollDown();
}
static void HideScrollIndicators(void)
{
sScreenControl->scrollIndicatorUpSprite->invisible = TRUE;
sScreenControl->scrollIndicatorDownSprite->invisible = TRUE;
}
static void SetScrollIndicatorXPos(bool32 inWordSelect)
{
if (!inWordSelect)
{
// Keyboard (only relevant for group mode, can't scroll in alphabet mode)
sScreenControl->scrollIndicatorUpSprite->pos1.x = 96;
sScreenControl->scrollIndicatorDownSprite->pos1.x = 96;
}
else
{
// Word select
sScreenControl->scrollIndicatorUpSprite->pos1.x = 120;
sScreenControl->scrollIndicatorDownSprite->pos1.x = 120;
}
}
// The Start/Select buttons are used as page scroll indicators
static void CreateStartSelectButtonSprites(void)
{
u8 spriteId = CreateSprite(&sSpriteTemplate_StartSelectButton, 220, 84, 1);
if (spriteId != MAX_SPRITES)
sScreenControl->startButtonSprite = &gSprites[spriteId];
spriteId = CreateSprite(&sSpriteTemplate_StartSelectButton, 220, 156, 1);
if (spriteId != MAX_SPRITES)
{
sScreenControl->selectButtonSprite = &gSprites[spriteId];
StartSpriteAnim(sScreenControl->selectButtonSprite, 1);
}
HideStartSelectButtons();
}
static void UpdateStartSelectButtonsVisibility(void)
{
sScreenControl->startButtonSprite->invisible = !CanScrollUp();
sScreenControl->selectButtonSprite->invisible = !CanScrollDown();
}
static void HideStartSelectButtons(void)
{
sScreenControl->startButtonSprite->invisible = TRUE;
sScreenControl->selectButtonSprite->invisible = TRUE;
}
static void TryAddInterviewObjectEvents(void)
{
int graphicsId;
u8 spriteId;
switch (GetDisplayedPersonType())
{
case EASY_CHAT_PERSON_REPORTER_MALE:
graphicsId = OBJ_EVENT_GFX_REPORTER_M;
break;
case EASY_CHAT_PERSON_REPORTER_FEMALE:
graphicsId = OBJ_EVENT_GFX_REPORTER_F;
break;
case EASY_CHAT_PERSON_BOY:
graphicsId = OBJ_EVENT_GFX_BOY_1;
break;
default:
return;
}
if (GetEasyChatScreenFrameId() != FRAMEID_INTERVIEW_SHOW_PERSON)
return;
// Add object for reporter/interviewing fan (facing left)
spriteId = AddPseudoObjectEvent(graphicsId, SpriteCallbackDummy, 76, 40, 0);
if (spriteId != MAX_SPRITES)
{
gSprites[spriteId].oam.priority = 0;
StartSpriteAnim(&gSprites[spriteId], 2);
}
// Add object for player (facing right)
spriteId = AddPseudoObjectEvent(
gSaveBlock2Ptr->playerGender == MALE ? OBJ_EVENT_GFX_RIVAL_BRENDAN_NORMAL : OBJ_EVENT_GFX_RIVAL_MAY_NORMAL,
SpriteCallbackDummy,
52,
40,
0);
if (spriteId != MAX_SPRITES)
{
gSprites[spriteId].oam.priority = 0;
StartSpriteAnim(&gSprites[spriteId], 3);
}
}
int GetFooterIndex(void)
{
u8 frameId = GetEasyChatScreenFrameId();
switch (sPhraseFrameDimensions[frameId].footerId)
{
case FOOTER_QUIZ:
return FOOTER_QUIZ;
case FOOTER_ANSWER:
return FOOTER_ANSWER;
case FOOTER_NORMAL:
return FOOTER_NORMAL;
default:
return NUM_FOOTER_TYPES;
}
}
static int GetFooterOptionXOffset(int option)
{
int footerIndex = GetFooterIndex();
if (footerIndex < NUM_FOOTER_TYPES)
return sFooterOptionXOffsets[footerIndex][option] + 4;
else
return 0;
}
static void AddMainScreenButtonWindow(void)
{
int i;
u16 windowId;
struct WindowTemplate template;
int footerIndex = GetFooterIndex();
if (footerIndex == NUM_FOOTER_TYPES)
return;
template.bg = 3;
template.tilemapLeft = 1;
template.tilemapTop = 11;
template.width = 28;
template.height = 2;
template.paletteNum = 11;
template.baseBlock = 0x34;
windowId = AddWindow(&template);
FillWindowPixelBuffer(windowId, PIXEL_FILL(1));
for (i = 0; i < (int)ARRAY_COUNT(sFooterTextOptions[0]); i++)
{
const u8 *str = sFooterTextOptions[footerIndex][i];
if (str)
{
int x = sFooterOptionXOffsets[footerIndex][i];
PrintEasyChatText(windowId, 1, str, x, 1, 0, NULL);
}
}
PutWindowTilemap(windowId);
}
static bool8 IsEasyChatGroupUnlocked(u8 groupId)
{
switch (groupId)
{
case EC_GROUP_TRENDY_SAYING:
return FlagGet(FLAG_SYS_HIPSTER_MEET);
case EC_GROUP_EVENTS:
case EC_GROUP_MOVE_1:
case EC_GROUP_MOVE_2:
return FlagGet(FLAG_SYS_GAME_CLEAR);
case EC_GROUP_POKEMON_NATIONAL:
return EasyChatIsNationalPokedexEnabled();
default:
return TRUE;
}
}
u16 EasyChat_GetNumWordsInGroup(u8 groupId)
{
if (groupId == EC_GROUP_POKEMON)
return GetNationalPokedexCount(FLAG_GET_SEEN);
if (IsEasyChatGroupUnlocked(groupId))
return gEasyChatGroups[groupId].numEnabledWords;
return 0;
}
static bool8 IsEasyChatWordInvalid(u16 easyChatWord)
{
u16 i;
u8 groupId;
u32 index;
u16 numWords;
const u16 *list;
if (easyChatWord == EC_EMPTY_WORD)
return FALSE;
groupId = EC_GROUP(easyChatWord);
index = EC_INDEX(easyChatWord);
if (groupId >= EC_NUM_GROUPS)
return TRUE;
numWords = gEasyChatGroups[groupId].numWords;
switch (groupId)
{
case EC_GROUP_POKEMON:
case EC_GROUP_POKEMON_NATIONAL:
case EC_GROUP_MOVE_1:
case EC_GROUP_MOVE_2:
list = gEasyChatGroups[groupId].wordData.valueList;
for (i = 0; i < numWords; i++)
{
if (index == list[i])
return FALSE;
}
return TRUE;
}
if (index >= numWords)
return TRUE;
else
return FALSE;
}
bool8 IsBardWordInvalid(u16 easyChatWord)
{
int numWordsInGroup;
u8 groupId = EC_GROUP(easyChatWord);
u32 index = EC_INDEX(easyChatWord);
if (groupId >= EC_NUM_GROUPS)
return TRUE;
switch (groupId)
{
case EC_GROUP_POKEMON:
case EC_GROUP_POKEMON_NATIONAL:
numWordsInGroup = gNumBardWords_Species;
break;
case EC_GROUP_MOVE_1:
case EC_GROUP_MOVE_2:
numWordsInGroup = gNumBardWords_Moves;
break;
default:
numWordsInGroup = gEasyChatGroups[groupId].numWords;
break;
}
if (numWordsInGroup <= index)
return TRUE;
else
return FALSE;
}
static const u8 *GetEasyChatWord(u8 groupId, u16 index)
{
switch (groupId)
{
case EC_GROUP_POKEMON:
case EC_GROUP_POKEMON_NATIONAL:
return gSpeciesNames[index];
case EC_GROUP_MOVE_1:
case EC_GROUP_MOVE_2:
return gMoveNames[index];
default:
return gEasyChatGroups[groupId].wordData.words[index].text;
}
}
u8 *CopyEasyChatWord(u8 *dest, u16 easyChatWord)
{
u8 *resultStr;
if (IsEasyChatWordInvalid(easyChatWord))
{
resultStr = StringCopy(dest, gText_ThreeQuestionMarks);
}
else if (easyChatWord != EC_EMPTY_WORD)
{
u16 index = EC_INDEX(easyChatWord);
u8 groupId = EC_GROUP(easyChatWord);
resultStr = StringCopy(dest, GetEasyChatWord(groupId, index));
}
else
{
*dest = EOS;
resultStr = dest;
}
return resultStr;
}
u8 *ConvertEasyChatWordsToString(u8 *dest, const u16 *src, u16 columns, u16 rows)
{
u16 i, j;
u16 numColumns = columns - 1;
for (i = 0; i < rows; i++)
{
for (j = 0; j < numColumns; j++)
{
dest = CopyEasyChatWord(dest, *src);
if (*src != EC_EMPTY_WORD)
{
*dest = CHAR_SPACE;
dest++;
}
src++;
}
dest = CopyEasyChatWord(dest, *(src++));
*dest = CHAR_NEWLINE;
dest++;
}
dest--;
*dest = EOS;
return dest;
}
static u8 *UnusedConvertEasyChatWordsToString(u8 *dest, const u16 *src, u16 columns, u16 rows)
{
u16 i, j, k;
u16 numColumns;
int notEmpty, lineNumber;
numColumns = columns;
lineNumber = 0;
columns--;
for (i = 0; i < rows; i++)
{
const u16 *str = src;
notEmpty = FALSE;
for (j = 0; j < numColumns; j++)
{
if (str[j] != EC_EMPTY_WORD)
notEmpty = TRUE;
}
if (!notEmpty)
{
src += numColumns;
continue;
}
for (k = 0; k < columns; k++)
{
dest = CopyEasyChatWord(dest, *src);
if (*src != EC_EMPTY_WORD)
{
*dest = CHAR_SPACE;
dest++;
}
src++;
}
dest = CopyEasyChatWord(dest, *(src++));
if (lineNumber == 0)
*dest = CHAR_NEWLINE;
else
*dest = CHAR_PROMPT_SCROLL;
dest++;
lineNumber++;
}
dest--;
*dest = EOS;
return dest;
}
static u16 GetEasyChatWordStringLength(u16 easyChatWord)
{
if (easyChatWord == EC_EMPTY_WORD)
return 0;
if (IsEasyChatWordInvalid(easyChatWord))
{
return StringLength(gText_ThreeQuestionMarks);
}
else
{
u16 index = EC_INDEX(easyChatWord);
u8 groupId = EC_GROUP(easyChatWord);
return StringLength(GetEasyChatWord(groupId, index));
}
}
static bool8 CanPhraseFitInXRowsYCols(const u16 *easyChatWords, u8 numRows, u8 numColumns, u16 maxLength)
{
u8 i, j;
for (i = 0; i < numColumns; i++)
{
u16 totalLength = numRows - 1;
for (j = 0; j < numRows; j++)
totalLength += GetEasyChatWordStringLength(*(easyChatWords++));
if (totalLength > maxLength)
return TRUE;
}
return FALSE;
}
u16 GetRandomEasyChatWordFromGroup(u16 groupId)
{
u16 index = Random() % gEasyChatGroups[groupId].numWords;
if (groupId == EC_GROUP_POKEMON
|| groupId == EC_GROUP_POKEMON_NATIONAL
|| groupId == EC_GROUP_MOVE_1
|| groupId == EC_GROUP_MOVE_2)
{
index = gEasyChatGroups[groupId].wordData.valueList[index];
}
return EC_WORD(groupId, index);
}
u16 GetRandomEasyChatWordFromUnlockedGroup(u16 groupId)
{
if (!IsEasyChatGroupUnlocked(groupId))
return EC_EMPTY_WORD;
if (groupId == EC_GROUP_POKEMON)
return GetRandomUnlockedEasyChatPokemon();
return GetRandomEasyChatWordFromGroup(groupId);
}
void ShowEasyChatProfile(void)
{
u16 *easyChatWords;
int columns, rows;
switch (gSpecialVar_0x8004)
{
case 0:
easyChatWords = gSaveBlock1Ptr->easyChatProfile;
columns = 2;
rows = 2;
break;
case 1:
easyChatWords = gSaveBlock1Ptr->easyChatBattleStart;
if (CanPhraseFitInXRowsYCols(gSaveBlock1Ptr->easyChatBattleStart, 3, 2, 18))
{
columns = 2;
rows = 3;
}
else
{
columns = 3;
rows = 2;
}
break;
case 2:
easyChatWords = gSaveBlock1Ptr->easyChatBattleWon;
columns = 3;
rows = 2;
break;
case 3:
easyChatWords = gSaveBlock1Ptr->easyChatBattleLost;
columns = 3;
rows = 2;
break;
default:
return;
}
ConvertEasyChatWordsToString(gStringVar4, easyChatWords, columns, rows);
ShowFieldAutoScrollMessage(gStringVar4);
}
// The phrase that a man in Dewford Hall suggests has a "deep link" to the current trendy phrase
void BufferDeepLinkPhrase(void)
{
int groupId = Random() & 1 ? EC_GROUP_HOBBIES : EC_GROUP_LIFESTYLE;
u16 easyChatWord = GetRandomEasyChatWordFromUnlockedGroup(groupId);
CopyEasyChatWord(gStringVar2, easyChatWord);
}
static bool8 IsAdditionalPhraseUnlocked(u8 additionalPhraseId)
{
int byteOffset = additionalPhraseId / 8;
int shift = additionalPhraseId % 8;
return (gSaveBlock1Ptr->additionalPhrases[byteOffset] >> shift) & 1;
}
void UnlockAdditionalPhrase(u8 additionalPhraseId)
{
if (additionalPhraseId < NUM_ADDITIONAL_PHRASES)
{
int byteOffset = additionalPhraseId / 8;
int shift = additionalPhraseId % 8;
gSaveBlock1Ptr->additionalPhrases[byteOffset] |= 1 << shift;
}
}
static u8 GetNumAdditionalPhrasesUnlocked(void)
{
u8 i;
u8 numAdditionalPhrasesUnlocked;
for (i = 0, numAdditionalPhrasesUnlocked = 0; i < NUM_ADDITIONAL_PHRASES; i++)
{
if (IsAdditionalPhraseUnlocked(i))
numAdditionalPhrasesUnlocked++;
}
return numAdditionalPhrasesUnlocked;
}
u16 GetNewHipsterPhraseToTeach(void)
{
u16 i;
u16 additionalPhraseId;
u8 numAdditionalPhrasesUnlocked = GetNumAdditionalPhrasesUnlocked();
if (numAdditionalPhrasesUnlocked == NUM_ADDITIONAL_PHRASES)
return EC_EMPTY_WORD;
additionalPhraseId = Random() % (NUM_ADDITIONAL_PHRASES - numAdditionalPhrasesUnlocked);
for (i = 0; i < NUM_ADDITIONAL_PHRASES; i++)
{
if (!IsAdditionalPhraseUnlocked(i))
{
if (additionalPhraseId)
{
additionalPhraseId--;
}
else
{
UnlockAdditionalPhrase(i);
return EC_WORD(EC_GROUP_TRENDY_SAYING, i);
}
}
}
return EC_EMPTY_WORD;
}
// Unused
u16 GetRandomTaughtHipsterPhrase(void)
{
u16 i;
u16 additionalPhraseId = GetNumAdditionalPhrasesUnlocked();
if (additionalPhraseId == 0)
return EC_EMPTY_WORD;
additionalPhraseId = Random() % additionalPhraseId;
for (i = 0; i < NUM_ADDITIONAL_PHRASES; i++)
{
if (IsAdditionalPhraseUnlocked(i))
{
if (additionalPhraseId)
additionalPhraseId--;
else
return EC_WORD(EC_GROUP_TRENDY_SAYING, i);
}
}
return EC_EMPTY_WORD;
}
static bool8 EasyChatIsNationalPokedexEnabled(void)
{
return IsNationalPokedexEnabled();
}
static u16 GetRandomUnlockedEasyChatPokemon(void)
{
u16 i;
u16 numWords;
const u16 *species;
u16 index = EasyChat_GetNumWordsInGroup(EC_GROUP_POKEMON);
if (index == 0)
return EC_EMPTY_WORD;
index = Random() % index;
species = gEasyChatGroups[EC_GROUP_POKEMON].wordData.valueList;
numWords = gEasyChatGroups[EC_GROUP_POKEMON].numWords;
for (i = 0; i < numWords; i++)
{
u16 dexNum = SpeciesToNationalPokedexNum(*species);
if (GetSetPokedexFlag(dexNum, FLAG_GET_SEEN))
{
if (index)
index--;
else
return EC_WORD(EC_GROUP_POKEMON, *species);
}
species++;
}
return EC_EMPTY_WORD;
}
void InitEasyChatPhrases(void)
{
u16 i, j;
for (i = 0; i < ARRAY_COUNT(sDefaultProfileWords); i++)
gSaveBlock1Ptr->easyChatProfile[i] = sDefaultProfileWords[i];
for (i = 0; i < EASY_CHAT_BATTLE_WORDS_COUNT; i++)
gSaveBlock1Ptr->easyChatBattleStart[i] = sDefaultBattleStartWords[i];
for (i = 0; i < EASY_CHAT_BATTLE_WORDS_COUNT; i++)
gSaveBlock1Ptr->easyChatBattleWon[i] = sDefaultBattleWonWords[i];
for (i = 0; i < EASY_CHAT_BATTLE_WORDS_COUNT; i++)
gSaveBlock1Ptr->easyChatBattleLost[i] = sDefaultBattleLostWords[i];
for (i = 0; i < MAIL_COUNT; i++)
{
for (j = 0; j < MAIL_WORDS_COUNT; j++)
gSaveBlock1Ptr->mail[i].words[j] = EC_EMPTY_WORD;
}
#ifndef UBFIX
// BUG: This is supposed to clear 64 bits, but this loop is clearing 64 bytes.
// However, this bug has no resulting effect on gameplay because only the
// Mauville old man data is corrupted, which is initialized directly after
// this function is called when starting a new game.
for (i = 0; i < 64; i++)
gSaveBlock1Ptr->additionalPhrases[i] = 0;
#else
for (i = 0; i < ARRAY_COUNT(gSaveBlock1Ptr->additionalPhrases); i++)
gSaveBlock1Ptr->additionalPhrases[i] = 0;
#endif
}
static bool8 InitEasyChatScreenWordData(void)
{
sWordData = Alloc(sizeof(*sWordData));
if (!sWordData)
return FALSE;
SetUnlockedEasyChatGroups();
SetUnlockedWordsByAlphabet();
return TRUE;
}
static void FreeEasyChatScreenWordData(void)
{
if (sWordData)
FREE_AND_SET_NULL(sWordData);
}
static void SetUnlockedEasyChatGroups(void)
{
int i;
sWordData->numUnlockedGroups = 0;
if (GetNationalPokedexCount(FLAG_GET_SEEN))
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = EC_GROUP_POKEMON;
// These groups are unlocked automatically
for (i = EC_GROUP_TRAINER; i <= EC_GROUP_ADJECTIVES; i++)
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = i;
if (FlagGet(FLAG_SYS_GAME_CLEAR))
{
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = EC_GROUP_EVENTS;
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = EC_GROUP_MOVE_1;
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = EC_GROUP_MOVE_2;
}
if (FlagGet(FLAG_SYS_HIPSTER_MEET))
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = EC_GROUP_TRENDY_SAYING;
if (IsNationalPokedexEnabled())
sWordData->unlockedGroupIds[sWordData->numUnlockedGroups++] = EC_GROUP_POKEMON_NATIONAL;
}
static u8 GetNumUnlockedEasyChatGroups(void)
{
return sWordData->numUnlockedGroups;
}
static u8 GetUnlockedEasyChatGroupId(u8 index)
{
if (index >= sWordData->numUnlockedGroups)
return EC_NUM_GROUPS;
else
return sWordData->unlockedGroupIds[index];
}
// Unused
static u8 *BufferEasyChatWordGroupName(u8 *dest, u8 groupId, u16 totalChars)
{
u16 i;
u8 *str = StringCopy(dest, sEasyChatGroupNamePointers[groupId]);
for (i = str - dest; i < totalChars; i++)
{
*str = CHAR_SPACE;
str++;
}
*str = EOS;
return str;
}
static const u8 *GetEasyChatWordGroupName(u8 groupId)
{
return sEasyChatGroupNamePointers[groupId];
}
static u8 *CopyEasyChatWordPadded(u8 *dest, u16 easyChatWord, u16 totalChars)
{
u16 i;
u8 *str = CopyEasyChatWord(dest, easyChatWord);
for (i = str - dest; i < totalChars; i++)
{
*str = CHAR_SPACE;
str++;
}
*str = EOS;
return str;
}
static void SetUnlockedWordsByAlphabet(void)
{
int i, j, k;
int numWords;
const u16 *words;
u16 numToProcess;
int index;
for (i = 0; i < EC_NUM_ALPHABET_GROUPS; i++)
{
numWords = gEasyChatWordsByLetterPointers[i].numWords;
words = gEasyChatWordsByLetterPointers[i].words;
sWordData->numUnlockedAlphabetWords[i] = 0;
index = 0;
for (j = 0; j < numWords; j++)
{
if (*words == EC_EMPTY_WORD)
{
words++;
numToProcess = *words;
words++;
j += 1 + numToProcess;
}
else
{
numToProcess = 1;
}
for (k = 0; k < numToProcess; k++)
{
if (IsEasyChatWordUnlocked(words[k]))
{
sWordData->unlockedAlphabetWords[i][index++] = words[k];
sWordData->numUnlockedAlphabetWords[i]++;
break;
}
}
words += numToProcess;
}
}
}
static void SetSelectedWordGroup(bool32 inAlphabetMode, u16 groupId)
{
if (!inAlphabetMode)
sWordData->numSelectedGroupWords = SetSelectedWordGroup_GroupMode(groupId);
else
sWordData->numSelectedGroupWords = SetSelectedWordGroup_AlphabetMode(groupId);
}
static u16 GetWordFromSelectedGroup(u16 index)
{
if (index >= sWordData->numSelectedGroupWords)
return EC_EMPTY_WORD;
else
return sWordData->selectedGroupWords[index];
}
static u16 GetNumWordsInSelectedGroup(void)
{
return sWordData->numSelectedGroupWords;
}
static u16 SetSelectedWordGroup_GroupMode(u16 groupId)
{
u32 i;
int totalWords;
const u16 *list;
const struct EasyChatWordInfo *wordInfo;
u16 numWords = gEasyChatGroups[groupId].numWords;
if (groupId == EC_GROUP_POKEMON || groupId == EC_GROUP_POKEMON_NATIONAL
|| groupId == EC_GROUP_MOVE_1 || groupId == EC_GROUP_MOVE_2)
{
list = gEasyChatGroups[groupId].wordData.valueList;
for (i = 0, totalWords = 0; i < numWords; i++)
{
if (IsEasyChatIndexAndGroupUnlocked(list[i], groupId))
sWordData->selectedGroupWords[totalWords++] = EC_WORD(groupId, list[i]);
}
return totalWords;
}
else
{
wordInfo = gEasyChatGroups[groupId].wordData.words;
for (i = 0, totalWords = 0; i < numWords; i++)
{
u16 alphabeticalOrder = wordInfo[i].alphabeticalOrder;
if (IsEasyChatIndexAndGroupUnlocked(alphabeticalOrder, groupId))
sWordData->selectedGroupWords[totalWords++] = EC_WORD(groupId, alphabeticalOrder);
}
return totalWords;
}
}
static u16 SetSelectedWordGroup_AlphabetMode(u16 groupId)
{
u16 i;
u16 totalWords;
for (i = 0, totalWords = 0; i < sWordData->numUnlockedAlphabetWords[groupId]; i++)
sWordData->selectedGroupWords[totalWords++] = sWordData->unlockedAlphabetWords[groupId][i];
return totalWords;
}
static bool8 IsEasyChatGroupUnlocked2(u8 groupId)
{
int i;
for (i = 0; i < sWordData->numUnlockedGroups; i++)
{
if (sWordData->unlockedGroupIds[i] == groupId)
return TRUE;
}
return FALSE;
}
static bool8 IsEasyChatIndexAndGroupUnlocked(u16 wordIndex, u8 groupId)
{
switch (groupId)
{
case EC_GROUP_POKEMON:
return GetSetPokedexFlag(SpeciesToNationalPokedexNum(wordIndex), FLAG_GET_SEEN);
case EC_GROUP_POKEMON_NATIONAL:
if (IsRestrictedWordSpecies(wordIndex))
GetSetPokedexFlag(SpeciesToNationalPokedexNum(wordIndex), FLAG_GET_SEEN);
return TRUE;
case EC_GROUP_MOVE_1:
case EC_GROUP_MOVE_2:
return TRUE;
case EC_GROUP_TRENDY_SAYING:
return IsAdditionalPhraseUnlocked(wordIndex);
default:
return gEasyChatGroups[groupId].wordData.words[wordIndex].enabled;
}
}
// Pokémon words in EC_GROUP_POKEMON_NATIONAL are always allowed (assuming the group is unlocked)
// unless they are in this group. If they are in this group (just Deoxys), they must also have been seen.
static int IsRestrictedWordSpecies(u16 species)
{
u32 i;
for (i = 0; i < ARRAY_COUNT(sRestrictedWordSpecies); i++)
{
if (sRestrictedWordSpecies[i] == species)
return TRUE;
}
return FALSE;
}
static u8 IsEasyChatWordUnlocked(u16 easyChatWord)
{
u8 groupId = EC_GROUP(easyChatWord);
u32 index = EC_INDEX(easyChatWord);
if (!IsEasyChatGroupUnlocked2(groupId))
return FALSE;
else
return IsEasyChatIndexAndGroupUnlocked(index, groupId);
}
void InitializeEasyChatWordArray(u16 *words, u16 length)
{
u16 i;
for (i = length - 1; i != EC_EMPTY_WORD; i--)
*(words++) = EC_EMPTY_WORD;
}
void InitQuestionnaireWords(void)
{
int i;
u16 *words = GetQuestionnaireWordsPtr();
for (i = 0; i < NUM_QUESTIONNAIRE_WORDS; i++)
words[i] = EC_EMPTY_WORD;
}
bool32 IsEasyChatAnswerUnlocked(int easyChatWord)
{
int groupId = EC_GROUP(easyChatWord);
int mask = EC_MASK_GROUP;
int index = EC_INDEX(easyChatWord);
if (!IsEasyChatGroupUnlocked(groupId & mask))
return FALSE;
else
return IsEasyChatIndexAndGroupUnlocked(index, groupId & mask);
}