2018-03-02 15:34:31 +00:00
|
|
|
#include "global.h"
|
|
|
|
#include "menu.h"
|
|
|
|
#include "list_menu.h"
|
|
|
|
#include "window.h"
|
|
|
|
#include "text_window.h"
|
|
|
|
#include "main.h"
|
|
|
|
#include "task.h"
|
2018-05-19 10:36:31 +01:00
|
|
|
#include "trig.h"
|
|
|
|
#include "decompress.h"
|
|
|
|
#include "palette.h"
|
|
|
|
#include "malloc.h"
|
2018-03-03 13:58:41 +00:00
|
|
|
#include "strings.h"
|
|
|
|
#include "sound.h"
|
|
|
|
#include "constants/songs.h"
|
2018-03-02 15:34:31 +00:00
|
|
|
|
2018-05-19 10:53:22 +01:00
|
|
|
// This file needs better labels and function names.
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
struct UnknownMysteryGiftLinkMenuStruct
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
s32 field_0;
|
|
|
|
u8 field_4;
|
|
|
|
u8 field_5;
|
|
|
|
u8 field_6;
|
|
|
|
};
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
struct UnknownListMenuPals
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
2018-03-03 13:58:41 +00:00
|
|
|
u8 cursorPal:4;
|
|
|
|
u8 fillValue:4;
|
|
|
|
u8 cursorShadowPal:4;
|
|
|
|
u8 lettersSpacing:6;
|
|
|
|
u8 field_2_2:6; // unused
|
|
|
|
u8 fontId:7;
|
|
|
|
u8 field_3_7:1;
|
2018-03-02 15:34:31 +00:00
|
|
|
};
|
|
|
|
|
2018-05-19 10:36:31 +01:00
|
|
|
struct UnkIndicatorsStruct
|
|
|
|
{
|
|
|
|
u8 field_0;
|
|
|
|
u16 *field_4;
|
|
|
|
u16 field_8;
|
|
|
|
u16 field_A;
|
|
|
|
u16 field_C;
|
|
|
|
u16 field_E;
|
|
|
|
u8 field_10;
|
|
|
|
u8 field_11;
|
|
|
|
u8 field_12;
|
|
|
|
u8 field_13;
|
|
|
|
u8 field_14_0:4;
|
|
|
|
u8 field_14_1:4;
|
|
|
|
u8 field_15_0:4;
|
|
|
|
u8 field_15_1:4;
|
|
|
|
u8 field_16_0:3;
|
|
|
|
u8 field_16_1:3;
|
|
|
|
u8 field_16_2:2;
|
|
|
|
u8 field_17_0:6;
|
|
|
|
u8 field_17_1:2;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct UnkIndicatorsStruct2
|
|
|
|
{
|
|
|
|
u8 field_0;
|
|
|
|
u16 *field_4;
|
|
|
|
u16 field_8;
|
|
|
|
u16 field_A;
|
|
|
|
u8 field_C;
|
|
|
|
u8 field_D;
|
|
|
|
u16 tileTag;
|
|
|
|
u16 palTag;
|
|
|
|
u16 field_12;
|
|
|
|
u16 field_14;
|
|
|
|
u16 field_16;
|
|
|
|
u16 field_18;
|
|
|
|
u16 field_1A;
|
|
|
|
u16 field_1C;
|
|
|
|
s16 field_1E;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct UnkIndicatorsStruct3
|
|
|
|
{
|
|
|
|
struct SubspriteTable subspriteTable;
|
|
|
|
struct Subsprite *subspritesPtr; // not a const pointer
|
|
|
|
u8 spriteId;
|
|
|
|
u16 tileTag;
|
|
|
|
u16 palTag;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct UnkIndicatorsStruct4
|
|
|
|
{
|
|
|
|
u8 field_0;
|
|
|
|
u16 tileTag;
|
|
|
|
u16 palTag;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct UnkSmallIndicatorsStruct
|
|
|
|
{
|
|
|
|
u8 field_0_0:4;
|
|
|
|
u8 field_0_1:4;
|
|
|
|
u8 field_1;
|
|
|
|
u16 field_2;
|
|
|
|
};
|
2018-03-02 15:34:31 +00:00
|
|
|
|
|
|
|
// this file's functions
|
2018-03-03 13:58:41 +00:00
|
|
|
static u8 ListMenuInitInternal(struct ListMenuTemplate *listMenuTemplate, u16 scrollOffset, u16 selectedRow);
|
|
|
|
static bool8 ListMenuChangeSelection(struct ListMenu *list, bool8 updateCursorAndCallCallback, u8 count, bool8 movingDown);
|
|
|
|
static void ListMenuPrintEntries(struct ListMenu *list, u16 startIndex, u16 yOffset, u16 count);
|
|
|
|
static void ListMenuDrawCursor(struct ListMenu *list);
|
|
|
|
static void ListMenuCallSelectionChangedCallback(struct ListMenu *list, u8 a2);
|
|
|
|
static u8 ListMenuAddCursorObject(struct ListMenu *list, u32 cursorKind);
|
2018-05-19 10:36:31 +01:00
|
|
|
static void Task_ScrollIndicatorArrowPair(u8 taskId);
|
|
|
|
static u8 ListMenuAddRedOutlineCursorObject(struct CursorStruct *cursor);
|
|
|
|
static u8 ListMenuAddRedArrowCursorObject(struct CursorStruct *cursor);
|
|
|
|
static void ListMenuUpdateRedOutlineCursorObject(u8 taskId, u16 x, u16 y);
|
|
|
|
static void ListMenuUpdateRedArrowCursorObject(u8 taskId, u16 x, u16 y);
|
|
|
|
static void ListMenuRemoveRedOutlineCursorObject(u8 taskId);
|
|
|
|
static void ListMenuRemoveRedArrowCursorObject(u8 taskId);
|
|
|
|
static u8 ListMenuAddCursorObjectInternal(struct CursorStruct *cursor, u32 cursorKind);
|
|
|
|
static void ListMenuUpdateCursorObject(u8 taskId, u16 x, u16 y, u32 cursorKind);
|
|
|
|
static void ListMenuRemoveCursorObject(u8 taskId, u32 cursorKind);
|
|
|
|
static void sub_81AF264(struct Sprite *sprite);
|
|
|
|
static void ObjectCB_RedArrowCursor(struct Sprite *sprite);
|
|
|
|
|
|
|
|
// EWRAM vars
|
|
|
|
EWRAM_DATA struct UnknownMysteryGiftLinkMenuStruct gUnknown_0203CE84 = {0};
|
|
|
|
EWRAM_DATA struct ArrowStruct gUnknown_0203CE8C = {0};
|
2018-03-03 13:58:41 +00:00
|
|
|
|
|
|
|
// IWRAM common
|
|
|
|
struct UnknownListMenuPals gUnknown_03006300;
|
|
|
|
struct ListMenuTemplate gMultiuseListMenuTemplate;
|
2018-03-02 15:34:31 +00:00
|
|
|
|
2018-05-19 10:36:31 +01:00
|
|
|
// const rom data
|
|
|
|
static const struct UnkSmallIndicatorsStruct gUnknown_086141FC[] =
|
|
|
|
{
|
|
|
|
{0, 0, 2, 8},
|
|
|
|
{1, 0, 2, -8},
|
|
|
|
{2, 1, 2, 8},
|
|
|
|
{3, 1, 2, -8},
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct OamData sOamData_861420C =
|
|
|
|
{
|
|
|
|
.y = 0,
|
|
|
|
.affineMode = 0,
|
|
|
|
.objMode = 0,
|
|
|
|
.mosaic = 0,
|
|
|
|
.bpp = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.x = 0,
|
|
|
|
.matrixNum = 0,
|
|
|
|
.size = 1,
|
|
|
|
.tileNum = 0,
|
|
|
|
.priority = 0,
|
|
|
|
.paletteNum = 0,
|
|
|
|
.affineParam = 0
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd sSpriteAnim_8614214[] =
|
|
|
|
{
|
|
|
|
ANIMCMD_FRAME(0, 30),
|
|
|
|
ANIMCMD_END
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd sSpriteAnim_861421C[] =
|
|
|
|
{
|
|
|
|
ANIMCMD_FRAME(0, 30, 1, 0),
|
|
|
|
ANIMCMD_END
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd sSpriteAnim_8614224[] =
|
|
|
|
{
|
|
|
|
ANIMCMD_FRAME(4, 30),
|
|
|
|
ANIMCMD_END
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd sSpriteAnim_861422C[] =
|
|
|
|
{
|
|
|
|
ANIMCMD_FRAME(4, 30, 0, 1),
|
|
|
|
ANIMCMD_END
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd *const sSpriteAnimTable_8614234[] =
|
|
|
|
{
|
|
|
|
sSpriteAnim_8614214,
|
|
|
|
sSpriteAnim_861421C,
|
|
|
|
sSpriteAnim_8614224,
|
|
|
|
sSpriteAnim_861422C
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct SpriteTemplate gUnknown_08614244 =
|
|
|
|
{
|
|
|
|
.tileTag = 0,
|
|
|
|
.paletteTag = 0,
|
|
|
|
.oam = &sOamData_861420C,
|
|
|
|
.anims = sSpriteAnimTable_8614234,
|
|
|
|
.images = NULL,
|
|
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
|
|
.callback = sub_81AF264,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_861425C =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 0,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_8614260 =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 1,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_8614264 =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 2,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_8614268 =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 3,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_861426C =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 4,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_8614270 =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 5,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_8614274 =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 6,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct Subsprite sSubsprite_8614278 =
|
|
|
|
{
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.size = 0,
|
|
|
|
.tileOffset = 7,
|
|
|
|
.priority = 0,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct OamData sOamData_861427C =
|
|
|
|
{
|
|
|
|
.y = 0,
|
|
|
|
.affineMode = 0,
|
|
|
|
.objMode = 0,
|
|
|
|
.mosaic = 0,
|
|
|
|
.bpp = 0,
|
|
|
|
.shape = 0,
|
|
|
|
.x = 0,
|
|
|
|
.matrixNum = 0,
|
|
|
|
.size = 1,
|
|
|
|
.tileNum = 0,
|
|
|
|
.priority = 0,
|
|
|
|
.paletteNum = 0,
|
|
|
|
.affineParam = 0
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd sSpriteAnim_8614284[] =
|
|
|
|
{
|
|
|
|
ANIMCMD_FRAME(0, 30),
|
|
|
|
ANIMCMD_END
|
|
|
|
};
|
|
|
|
|
|
|
|
static const union AnimCmd *const sSpriteAnimTable_861428C[] =
|
|
|
|
{
|
|
|
|
sSpriteAnim_8614284
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct SpriteTemplate gUnknown_08614290 =
|
|
|
|
{
|
|
|
|
.tileTag = 0,
|
|
|
|
.paletteTag = 0,
|
|
|
|
.oam = &sOamData_861427C,
|
|
|
|
.anims = sSpriteAnimTable_861428C,
|
|
|
|
.images = NULL,
|
|
|
|
.affineAnims = gDummySpriteAffineAnimTable,
|
|
|
|
.callback = ObjectCB_RedArrowCursor,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const u16 gUnknown_086142A8[] = INCBIN_U16("graphics/interface/red_arrow.gbapal");
|
|
|
|
static const u8 gUnknown_086142C8[] = INCBIN_U8("graphics/interface/red_arrow_other.4bpp.lz");
|
|
|
|
static const u8 gUnknown_08614338[] = INCBIN_U8("graphics/interface/selector_outline.4bpp.lz");
|
|
|
|
static const u8 gUnknown_08614378[] = INCBIN_U8("graphics/interface/red_arrow.4bpp.lz");
|
|
|
|
|
2018-03-02 15:34:31 +00:00
|
|
|
// code
|
2018-03-03 13:58:41 +00:00
|
|
|
static void ListMenuDummyTask(u8 taskId)
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
s32 DoMysteryGiftListMenu(struct WindowTemplate *windowTemplate, struct ListMenuTemplate *listMenuTemplate, u8 arg2, u16 tileNum, u16 palNum)
|
|
|
|
{
|
|
|
|
switch (gUnknown_0203CE84.field_4)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
default:
|
|
|
|
gUnknown_0203CE84.field_5 = AddWindow(windowTemplate);
|
|
|
|
switch (arg2)
|
|
|
|
{
|
|
|
|
case 2:
|
|
|
|
sub_809882C(gUnknown_0203CE84.field_5, tileNum, palNum);
|
|
|
|
case 1:
|
|
|
|
sub_8098858(gUnknown_0203CE84.field_5, tileNum, palNum / 16);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
gMultiuseListMenuTemplate = *listMenuTemplate;
|
|
|
|
gMultiuseListMenuTemplate.windowId = gUnknown_0203CE84.field_5;
|
|
|
|
gUnknown_0203CE84.field_6 = ListMenuInit(&gMultiuseListMenuTemplate, 0, 0);
|
|
|
|
CopyWindowToVram(gUnknown_0203CE84.field_5, 1);
|
|
|
|
gUnknown_0203CE84.field_4 = 1;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
gUnknown_0203CE84.field_0 = ListMenuHandleInputGetItemId(gUnknown_0203CE84.field_6);
|
|
|
|
if (gMain.newKeys & A_BUTTON)
|
2018-03-09 21:18:58 +00:00
|
|
|
{
|
2018-03-02 15:34:31 +00:00
|
|
|
gUnknown_0203CE84.field_4 = 2;
|
2018-03-09 21:18:58 +00:00
|
|
|
}
|
2018-03-02 15:34:31 +00:00
|
|
|
if (gMain.newKeys & B_BUTTON)
|
|
|
|
{
|
|
|
|
gUnknown_0203CE84.field_0 = LIST_B_PRESSED;
|
|
|
|
gUnknown_0203CE84.field_4 = 2;
|
|
|
|
}
|
|
|
|
if (gUnknown_0203CE84.field_4 == 2)
|
|
|
|
{
|
|
|
|
if (arg2 == 0)
|
2018-03-09 21:18:58 +00:00
|
|
|
{
|
2018-03-02 15:34:31 +00:00
|
|
|
ClearWindowTilemap(gUnknown_0203CE84.field_5);
|
2018-03-09 21:18:58 +00:00
|
|
|
}
|
2018-03-02 15:34:31 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (arg2)
|
|
|
|
{
|
|
|
|
case 0: // can never be reached, because of the if statement above
|
|
|
|
sub_819746C(gUnknown_0203CE84.field_5, FALSE);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
case 1:
|
|
|
|
sub_819746C(gUnknown_0203CE84.field_5, FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyWindowToVram(gUnknown_0203CE84.field_5, 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
DestroyListMenuTask(gUnknown_0203CE84.field_6, NULL, NULL);
|
|
|
|
RemoveWindow(gUnknown_0203CE84.field_5);
|
|
|
|
gUnknown_0203CE84.field_4 = 0;
|
|
|
|
return gUnknown_0203CE84.field_0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 ListMenuInit(struct ListMenuTemplate *listMenuTemplate, u16 scrollOffset, u16 selectedRow)
|
|
|
|
{
|
|
|
|
u8 taskId = ListMenuInitInternal(listMenuTemplate, scrollOffset, selectedRow);
|
|
|
|
PutWindowTilemap(listMenuTemplate->windowId);
|
|
|
|
CopyWindowToVram(listMenuTemplate->windowId, 2);
|
|
|
|
|
|
|
|
return taskId;
|
|
|
|
}
|
|
|
|
|
|
|
|
// unused
|
2018-03-03 13:58:41 +00:00
|
|
|
u8 ListMenuInitWithWindows(struct ListMenuTemplate *listMenuTemplate, struct UnknownListMenuWindowStruct *arg1, u16 scrollOffset, u16 selectedRow)
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
s32 i;
|
|
|
|
|
|
|
|
u8 taskId = ListMenuInitInternal(listMenuTemplate, scrollOffset, selectedRow);
|
|
|
|
for (i = 0; arg1[i].palNum != 0xFF; i++)
|
|
|
|
{
|
|
|
|
PutWindowRectTilemapOverridePalette(listMenuTemplate->windowId,
|
|
|
|
arg1[i].x,
|
|
|
|
arg1[i].y,
|
|
|
|
arg1[i].width,
|
|
|
|
arg1[i].height,
|
|
|
|
arg1[i].palNum);
|
|
|
|
}
|
|
|
|
CopyWindowToVram(listMenuTemplate->windowId, 2);
|
|
|
|
|
|
|
|
return taskId;
|
|
|
|
}
|
|
|
|
|
|
|
|
s32 ListMenuHandleInputGetItemId(u8 listTaskId)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
if (gMain.newKeys & A_BUTTON)
|
|
|
|
{
|
|
|
|
return list->template.items[list->scrollOffset + list->selectedRow].id;
|
|
|
|
}
|
|
|
|
else if (gMain.newKeys & B_BUTTON)
|
|
|
|
{
|
|
|
|
return LIST_B_PRESSED;
|
|
|
|
}
|
|
|
|
else if (gMain.newAndRepeatedKeys & DPAD_UP)
|
|
|
|
{
|
|
|
|
ListMenuChangeSelection(list, TRUE, 1, FALSE);
|
|
|
|
return LIST_NOTHING_CHOSEN;
|
|
|
|
}
|
|
|
|
else if (gMain.newAndRepeatedKeys & DPAD_DOWN)
|
|
|
|
{
|
|
|
|
ListMenuChangeSelection(list, TRUE, 1, TRUE);
|
|
|
|
return LIST_NOTHING_CHOSEN;
|
|
|
|
}
|
|
|
|
else // try to move by one window scroll
|
|
|
|
{
|
|
|
|
bool16 rightButton, leftButton;
|
|
|
|
switch (list->template.scrollMultiple)
|
|
|
|
{
|
|
|
|
case LIST_NO_MULTIPLE_SCROLL:
|
|
|
|
default:
|
|
|
|
leftButton = FALSE;
|
|
|
|
rightButton = FALSE;
|
|
|
|
break;
|
|
|
|
case LIST_MULTIPLE_SCROLL_DPAD:
|
|
|
|
leftButton = gMain.newAndRepeatedKeys & DPAD_LEFT;
|
|
|
|
rightButton = gMain.newAndRepeatedKeys & DPAD_RIGHT;
|
|
|
|
break;
|
|
|
|
case LIST_MULTIPLE_SCROLL_L_R:
|
|
|
|
leftButton = gMain.newAndRepeatedKeys & L_BUTTON;
|
|
|
|
rightButton = gMain.newAndRepeatedKeys & R_BUTTON;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (leftButton)
|
|
|
|
{
|
|
|
|
ListMenuChangeSelection(list, TRUE, list->template.maxShowed, FALSE);
|
|
|
|
return LIST_NOTHING_CHOSEN;
|
|
|
|
}
|
|
|
|
else if (rightButton)
|
|
|
|
{
|
|
|
|
ListMenuChangeSelection(list, TRUE, list->template.maxShowed, TRUE);
|
|
|
|
return LIST_NOTHING_CHOSEN;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return LIST_NOTHING_CHOSEN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DestroyListMenuTask(u8 listTaskId, u16 *scrollOffset, u16 *selectedRow)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
if (scrollOffset != NULL)
|
|
|
|
*scrollOffset = list->scrollOffset;
|
|
|
|
if (selectedRow != NULL)
|
|
|
|
*selectedRow = list->selectedRow;
|
|
|
|
|
|
|
|
if (list->unk_1E != 0xFF)
|
|
|
|
ListMenuRemoveCursorObject(list->unk_1E, list->template.cursorKind - 2);
|
|
|
|
|
|
|
|
DestroyTask(listTaskId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sub_81AE70C(u8 listTaskId)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
FillWindowPixelBuffer(list->template.windowId, (list->template.fillValue << 4) | (list->template.fillValue));
|
2018-03-02 15:34:31 +00:00
|
|
|
ListMenuPrintEntries(list, list->scrollOffset, 0, list->template.maxShowed);
|
|
|
|
ListMenuDrawCursor(list);
|
|
|
|
CopyWindowToVram(list->template.windowId, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unused
|
2018-03-03 13:58:41 +00:00
|
|
|
void ChangeListMenuPals(u8 listTaskId, u8 cursorPal, u8 fillValue, u8 cursorShadowPal)
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
list->template.cursorPal = cursorPal;
|
2018-03-03 13:58:41 +00:00
|
|
|
list->template.fillValue = fillValue;
|
2018-03-02 15:34:31 +00:00
|
|
|
list->template.cursorShadowPal = cursorShadowPal;
|
|
|
|
}
|
|
|
|
|
|
|
|
// unused
|
|
|
|
void ChangeListMenuCoords(u8 listTaskId, u8 x, u8 y)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
SetWindowAttribute(list->template.windowId, WINDOW_TILEMAP_LEFT, x);
|
|
|
|
SetWindowAttribute(list->template.windowId, WINDOW_TILEMAP_TOP, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unused
|
|
|
|
s32 ListMenuTestInput(struct ListMenuTemplate *template, u32 scrollOffset, u32 selectedRow, u16 keys, u16 *newScrollOffset, u16 *newSelectedRow)
|
|
|
|
{
|
|
|
|
struct ListMenu list;
|
|
|
|
|
|
|
|
list.template = *template;
|
|
|
|
list.scrollOffset = scrollOffset;
|
|
|
|
list.selectedRow = selectedRow;
|
|
|
|
list.unk_1C = 0;
|
|
|
|
list.unk_1D = 0;
|
|
|
|
|
|
|
|
if (keys == DPAD_UP)
|
|
|
|
ListMenuChangeSelection(&list, FALSE, 1, FALSE);
|
|
|
|
if (keys == DPAD_DOWN)
|
|
|
|
ListMenuChangeSelection(&list, FALSE, 1, TRUE);
|
|
|
|
|
|
|
|
if (newScrollOffset != NULL)
|
|
|
|
*newScrollOffset = list.scrollOffset;
|
|
|
|
if (newSelectedRow != NULL)
|
|
|
|
*newSelectedRow = list.selectedRow;
|
|
|
|
|
|
|
|
return LIST_NOTHING_CHOSEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ListMenuGetCurrentItemArrayId(u8 listTaskId, u16 *arrayId)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
if (arrayId != NULL)
|
|
|
|
*arrayId = list->scrollOffset + list->selectedRow;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ListMenuGetScrollAndRow(u8 listTaskId, u16 *scrollOffset, u16 *selectedRow)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
if (scrollOffset != NULL)
|
|
|
|
*scrollOffset = list->scrollOffset;
|
|
|
|
if (selectedRow != NULL)
|
|
|
|
*selectedRow = list->selectedRow;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16 ListMenuGetYCoordForPrintingArrowCursor(u8 listTaskId)
|
|
|
|
{
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
u8 yMultiplier = GetFontAttribute(list->template.fontId, 1) + list->template.unk_16_3;
|
|
|
|
|
|
|
|
return list->selectedRow * yMultiplier + list->template.upText_Y;
|
|
|
|
}
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
static u8 ListMenuInitInternal(struct ListMenuTemplate *listMenuTemplate, u16 scrollOffset, u16 selectedRow)
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
u8 listTaskId = CreateTask(ListMenuDummyTask, 0);
|
|
|
|
struct ListMenu *list = (void*) gTasks[listTaskId].data;
|
|
|
|
|
|
|
|
list->template = *listMenuTemplate;
|
|
|
|
list->scrollOffset = scrollOffset;
|
|
|
|
list->selectedRow = selectedRow;
|
|
|
|
list->unk_1C = 0;
|
|
|
|
list->unk_1D = 0;
|
|
|
|
list->unk_1E = 0xFF;
|
|
|
|
list->unk_1F = 0;
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
gUnknown_03006300.cursorPal = list->template.cursorPal;
|
|
|
|
gUnknown_03006300.fillValue = list->template.fillValue;
|
|
|
|
gUnknown_03006300.cursorShadowPal = list->template.cursorShadowPal;
|
2018-03-02 15:34:31 +00:00
|
|
|
gUnknown_03006300.lettersSpacing = list->template.lettersSpacing;
|
|
|
|
gUnknown_03006300.fontId = list->template.fontId;
|
|
|
|
gUnknown_03006300.field_3_7 = 0;
|
|
|
|
|
|
|
|
if (list->template.totalItems < list->template.maxShowed)
|
|
|
|
list->template.maxShowed = list->template.totalItems;
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
FillWindowPixelBuffer(list->template.windowId, (list->template.fillValue << 4) | (list->template.fillValue));
|
2018-03-02 15:34:31 +00:00
|
|
|
ListMenuPrintEntries(list, list->scrollOffset, 0, list->template.maxShowed);
|
|
|
|
ListMenuDrawCursor(list);
|
|
|
|
ListMenuCallSelectionChangedCallback(list, 1);
|
|
|
|
|
|
|
|
return listTaskId;
|
|
|
|
}
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
static void ListMenuPrint(struct ListMenu *list, const u8 *str, u8 x, u8 y)
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
u8 colors[3];
|
|
|
|
if (gUnknown_03006300.field_3_7)
|
|
|
|
{
|
2018-03-03 13:58:41 +00:00
|
|
|
colors[0] = gUnknown_03006300.fillValue;
|
|
|
|
colors[1] = gUnknown_03006300.cursorPal;
|
|
|
|
colors[2] = gUnknown_03006300.cursorShadowPal;
|
2018-03-02 15:34:31 +00:00
|
|
|
AddTextPrinterParameterized2(list->template.windowId,
|
|
|
|
gUnknown_03006300.fontId,
|
|
|
|
x, y,
|
|
|
|
gUnknown_03006300.lettersSpacing,
|
|
|
|
0, colors, TEXT_SPEED_FF, str);
|
|
|
|
|
|
|
|
gUnknown_03006300.field_3_7 = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-03-03 13:58:41 +00:00
|
|
|
colors[0] = list->template.fillValue;
|
2018-03-02 15:34:31 +00:00
|
|
|
colors[1] = list->template.cursorPal;
|
|
|
|
colors[2] = list->template.cursorShadowPal;
|
|
|
|
AddTextPrinterParameterized2(list->template.windowId,
|
|
|
|
list->template.fontId,
|
|
|
|
x, y,
|
|
|
|
list->template.lettersSpacing,
|
|
|
|
0, colors, TEXT_SPEED_FF, str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
static void ListMenuPrintEntries(struct ListMenu *list, u16 startIndex, u16 yOffset, u16 count)
|
2018-03-02 15:34:31 +00:00
|
|
|
{
|
|
|
|
s32 i;
|
|
|
|
u8 x, y;
|
|
|
|
u8 yMultiplier = GetFontAttribute(list->template.fontId, 1) + list->template.unk_16_3;
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
if (list->template.items[startIndex].id != -3)
|
|
|
|
x = list->template.unk_12;
|
|
|
|
else
|
|
|
|
x = list->template.unk_11;
|
|
|
|
|
|
|
|
y = (yOffset + i) * yMultiplier + list->template.upText_Y;
|
|
|
|
if (list->template.unk_08 != NULL)
|
|
|
|
list->template.unk_08(list->template.windowId, list->template.items[startIndex].id, y);
|
|
|
|
|
|
|
|
ListMenuPrint(list, list->template.items[startIndex].name, x, y);
|
|
|
|
startIndex++;
|
|
|
|
}
|
|
|
|
}
|
2018-03-03 00:02:07 +00:00
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
static void ListMenuDrawCursor(struct ListMenu *list)
|
2018-03-03 00:02:07 +00:00
|
|
|
{
|
|
|
|
u8 yMultiplier = GetFontAttribute(list->template.fontId, 1) + list->template.unk_16_3;
|
2018-03-03 13:58:41 +00:00
|
|
|
u8 x = list->template.cursor_X;
|
2018-03-03 00:02:07 +00:00
|
|
|
u8 y = list->selectedRow * yMultiplier + list->template.upText_Y;
|
|
|
|
switch (list->template.cursorKind)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
ListMenuPrint(list, gText_SelectorArrow2, x, y);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
if (list->unk_1E == 0xFF)
|
|
|
|
list->unk_1E = ListMenuAddCursorObject(list, 0);
|
|
|
|
ListMenuUpdateCursorObject(list->unk_1E,
|
|
|
|
GetWindowAttribute(list->template.windowId, WINDOW_TILEMAP_LEFT) * 8 - 1,
|
|
|
|
GetWindowAttribute(list->template.windowId, WINDOW_TILEMAP_TOP) * 8 + y - 1, 0);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
if (list->unk_1E == 0xFF)
|
|
|
|
list->unk_1E = ListMenuAddCursorObject(list, 1);
|
|
|
|
ListMenuUpdateCursorObject(list->unk_1E,
|
|
|
|
GetWindowAttribute(list->template.windowId, WINDOW_TILEMAP_LEFT) * 8 + x,
|
|
|
|
GetWindowAttribute(list->template.windowId, WINDOW_TILEMAP_TOP) * 8 + y, 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-03 13:58:41 +00:00
|
|
|
static u8 ListMenuAddCursorObject(struct ListMenu *list, u32 cursorKind)
|
2018-03-03 00:02:07 +00:00
|
|
|
{
|
|
|
|
struct CursorStruct cursor;
|
|
|
|
|
|
|
|
cursor.unk0 = 0;
|
|
|
|
cursor.unk1 = 0xA0;
|
|
|
|
cursor.unk2 = GetWindowAttribute(list->template.windowId, WINDOW_WIDTH) * 8 + 2;
|
|
|
|
cursor.unk4 = GetFontAttribute(list->template.fontId, 1) + 2;
|
2018-05-19 10:36:31 +01:00
|
|
|
cursor.tileTag = 0x4000;
|
|
|
|
cursor.palTag = 0xFFFF;
|
|
|
|
cursor.palNum = 0xF;
|
2018-03-03 00:02:07 +00:00
|
|
|
|
|
|
|
return ListMenuAddCursorObjectInternal(&cursor, cursorKind);
|
|
|
|
}
|
2018-03-03 13:58:41 +00:00
|
|
|
|
|
|
|
static void ListMenuErasePrintedCursor(struct ListMenu *list, u16 selectedRow)
|
|
|
|
{
|
|
|
|
u8 cursorKind = list->template.cursorKind;
|
|
|
|
if (cursorKind == 0)
|
|
|
|
{
|
|
|
|
u8 yMultiplier = GetFontAttribute(list->template.fontId, 1) + list->template.unk_16_3;
|
|
|
|
u8 width = GetMenuCursorDimensionByFont(list->template.fontId, 0);
|
|
|
|
u8 height = GetMenuCursorDimensionByFont(list->template.fontId, 1);
|
|
|
|
FillWindowPixelRect(list->template.windowId,
|
|
|
|
(list->template.fillValue << 4) | (list->template.fillValue),
|
|
|
|
list->template.cursor_X,
|
|
|
|
selectedRow * yMultiplier + list->template.upText_Y,
|
|
|
|
width,
|
|
|
|
height);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static u8 ListMenuUpdateSelectedRowIndexAndScrollOffset(struct ListMenu *list, bool8 movingDown)
|
|
|
|
{
|
|
|
|
u16 selectedRow = list->selectedRow;
|
|
|
|
u16 scrollOffset = list->scrollOffset;
|
|
|
|
u16 newRow;
|
|
|
|
u32 newScroll;
|
|
|
|
|
|
|
|
if (!movingDown)
|
|
|
|
{
|
|
|
|
if (list->template.maxShowed == 1)
|
|
|
|
newRow = 0;
|
|
|
|
else
|
|
|
|
newRow = list->template.maxShowed - ((list->template.maxShowed / 2) + (list->template.maxShowed % 2)) - 1;
|
|
|
|
|
|
|
|
if (scrollOffset == 0)
|
|
|
|
{
|
|
|
|
while (selectedRow != 0)
|
|
|
|
{
|
|
|
|
selectedRow--;
|
|
|
|
if (list->template.items[scrollOffset + selectedRow].id != -3)
|
|
|
|
{
|
|
|
|
list->selectedRow = selectedRow;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (selectedRow > newRow)
|
|
|
|
{
|
|
|
|
selectedRow--;
|
|
|
|
if (list->template.items[scrollOffset + selectedRow].id != -3)
|
|
|
|
{
|
|
|
|
list->selectedRow = selectedRow;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
newScroll = scrollOffset - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (list->template.maxShowed == 1)
|
|
|
|
newRow = 0;
|
|
|
|
else
|
|
|
|
newRow = ((list->template.maxShowed / 2) + (list->template.maxShowed % 2));
|
|
|
|
|
|
|
|
if (scrollOffset == list->template.totalItems - list->template.maxShowed)
|
|
|
|
{
|
|
|
|
while (selectedRow < list->template.maxShowed - 1)
|
|
|
|
{
|
|
|
|
selectedRow++;
|
|
|
|
if (list->template.items[scrollOffset + selectedRow].id != -3)
|
|
|
|
{
|
|
|
|
list->selectedRow = selectedRow;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (selectedRow < newRow)
|
|
|
|
{
|
|
|
|
selectedRow++;
|
|
|
|
if (list->template.items[scrollOffset + selectedRow].id != -3)
|
|
|
|
{
|
|
|
|
list->selectedRow = selectedRow;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
newScroll = scrollOffset + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
list->selectedRow = newRow;
|
|
|
|
list->scrollOffset = newScroll;
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuScroll(struct ListMenu *list, u8 count, bool8 movingDown)
|
|
|
|
{
|
|
|
|
if (count >= list->template.maxShowed)
|
|
|
|
{
|
|
|
|
FillWindowPixelBuffer(list->template.windowId, (list->template.fillValue << 4) | (list->template.fillValue));
|
|
|
|
ListMenuPrintEntries(list, list->scrollOffset, 0, list->template.maxShowed);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
u8 yMultiplier = GetFontAttribute(list->template.fontId, 1) + list->template.unk_16_3;
|
|
|
|
|
|
|
|
if (!movingDown)
|
|
|
|
{
|
|
|
|
u16 y, width, height;
|
|
|
|
|
|
|
|
ScrollWindow(list->template.windowId, 1, count * yMultiplier, (list->template.fillValue << 4) | (list->template.fillValue));
|
|
|
|
ListMenuPrintEntries(list, list->scrollOffset, 0, count);
|
|
|
|
|
|
|
|
y = (list->template.maxShowed * yMultiplier) + list->template.upText_Y;
|
|
|
|
width = GetWindowAttribute(list->template.windowId, WINDOW_WIDTH) * 8;
|
|
|
|
height = (GetWindowAttribute(list->template.windowId, WINDOW_HEIGHT) * 8) - y;
|
|
|
|
FillWindowPixelRect(list->template.windowId,
|
|
|
|
(list->template.fillValue << 4) | (list->template.fillValue),
|
|
|
|
0, y, width, height);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
u16 width;
|
|
|
|
|
|
|
|
ScrollWindow(list->template.windowId, 0, count * yMultiplier, (list->template.fillValue << 4) | (list->template.fillValue));
|
|
|
|
ListMenuPrintEntries(list, list->scrollOffset + (list->template.maxShowed - count), list->template.maxShowed - count, count);
|
|
|
|
|
|
|
|
width = GetWindowAttribute(list->template.windowId, WINDOW_WIDTH) * 8;
|
|
|
|
FillWindowPixelRect(list->template.windowId,
|
|
|
|
(list->template.fillValue << 4) | (list->template.fillValue),
|
|
|
|
0, 0, width, list->template.upText_Y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool8 ListMenuChangeSelection(struct ListMenu *list, bool8 updateCursorAndCallCallback, u8 count, bool8 movingDown)
|
|
|
|
{
|
|
|
|
u16 oldSelectedRow;
|
|
|
|
u8 selectionChange, i, cursorCount;
|
|
|
|
|
|
|
|
oldSelectedRow = list->selectedRow;
|
|
|
|
cursorCount = 0;
|
|
|
|
selectionChange = 0;
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
u8 ret = ListMenuUpdateSelectedRowIndexAndScrollOffset(list, movingDown);
|
|
|
|
selectionChange |= ret;
|
|
|
|
if (ret != 2)
|
|
|
|
break;
|
|
|
|
cursorCount++;
|
|
|
|
} while (list->template.items[list->scrollOffset + list->selectedRow].id == -3);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (updateCursorAndCallCallback)
|
|
|
|
{
|
|
|
|
switch (selectionChange)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
default:
|
|
|
|
return TRUE;
|
|
|
|
case 1:
|
|
|
|
ListMenuErasePrintedCursor(list, oldSelectedRow);
|
|
|
|
ListMenuDrawCursor(list);
|
|
|
|
ListMenuCallSelectionChangedCallback(list, 0);
|
|
|
|
CopyWindowToVram(list->template.windowId, 2);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
ListMenuErasePrintedCursor(list, oldSelectedRow);
|
|
|
|
ListMenuScroll(list, cursorCount, movingDown);
|
|
|
|
ListMenuDrawCursor(list);
|
|
|
|
ListMenuCallSelectionChangedCallback(list, 0);
|
|
|
|
CopyWindowToVram(list->template.windowId, 2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuCallSelectionChangedCallback(struct ListMenu *list, u8 a2)
|
|
|
|
{
|
|
|
|
if (list->template.moveCursorFunc != NULL)
|
|
|
|
list->template.moveCursorFunc(list->template.items[list->scrollOffset + list->selectedRow].id, a2, list);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unused
|
|
|
|
void sub_81AF028(u8 cursorPal, u8 fillValue, u8 cursorShadowPal)
|
|
|
|
{
|
|
|
|
gUnknown_03006300.cursorPal = cursorPal;
|
|
|
|
gUnknown_03006300.fillValue = fillValue;
|
|
|
|
gUnknown_03006300.cursorShadowPal = cursorShadowPal;
|
|
|
|
gUnknown_03006300.field_3_7 = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sub_81AF078(s32 arg0, u8 arg1, struct ListMenu *list)
|
|
|
|
{
|
|
|
|
if (!arg1)
|
|
|
|
PlaySE(SE_SELECT);
|
|
|
|
}
|
2018-05-19 10:36:31 +01:00
|
|
|
|
|
|
|
// unused
|
|
|
|
s32 sub_81AF08C(u8 taskId, u8 field)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
switch (field)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
return (s32)(data->field_4);
|
|
|
|
case 2:
|
|
|
|
return data->field_C;
|
|
|
|
case 3:
|
|
|
|
return data->field_E;
|
|
|
|
case 4:
|
|
|
|
return data->field_10;
|
|
|
|
case 5:
|
|
|
|
return data->field_11;
|
|
|
|
case 6:
|
|
|
|
return data->field_12;
|
|
|
|
case 7:
|
|
|
|
return data->field_13;
|
|
|
|
case 8:
|
|
|
|
return data->field_14_0;
|
|
|
|
case 9:
|
|
|
|
return data->field_14_1;
|
|
|
|
case 10:
|
|
|
|
return data->field_15_0;
|
|
|
|
case 11:
|
|
|
|
return data->field_15_1;
|
|
|
|
case 12:
|
|
|
|
return data->field_16_0;
|
|
|
|
case 13:
|
|
|
|
return data->field_16_1;
|
|
|
|
case 14:
|
|
|
|
return data->field_16_2;
|
|
|
|
case 15:
|
|
|
|
return data->field_17_0;
|
|
|
|
case 16:
|
|
|
|
return data->field_17_1;
|
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// unused
|
|
|
|
void sub_81AF15C(u8 taskId, u8 field, s32 value)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct *data = (void*) &gTasks[taskId].data;
|
|
|
|
|
|
|
|
switch (field)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
data->field_4 = (void*)(value);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
data->field_C = value;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
data->field_E = value;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
data->field_10 = value;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
data->field_11 = value;
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
data->field_12 = value;
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
data->field_13 = value;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
data->field_14_0 = value;
|
|
|
|
break;
|
|
|
|
case 9:
|
|
|
|
data->field_14_1 = value;
|
|
|
|
break;
|
|
|
|
case 10:
|
|
|
|
data->field_15_0 = value;
|
|
|
|
break;
|
|
|
|
case 11:
|
|
|
|
data->field_15_1 = value;
|
|
|
|
break;
|
|
|
|
case 12:
|
|
|
|
data->field_16_0 = value;
|
|
|
|
break;
|
|
|
|
case 13:
|
|
|
|
data->field_16_1 = value;
|
|
|
|
break;
|
|
|
|
case 14:
|
|
|
|
data->field_16_2 = value;
|
|
|
|
break;
|
|
|
|
case 15:
|
|
|
|
data->field_17_0 = value;
|
|
|
|
break;
|
|
|
|
case 16:
|
|
|
|
data->field_17_1 = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sub_81AF264(struct Sprite *sprite)
|
|
|
|
{
|
|
|
|
s32 multiplier;
|
|
|
|
|
|
|
|
switch (sprite->data[0])
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
StartSpriteAnim(sprite, sprite->data[1]);
|
|
|
|
sprite->data[0]++;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
switch (sprite->data[2])
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
multiplier = sprite->data[3];
|
|
|
|
sprite->pos2.x = (gSineTable[(u8)(sprite->data[5])] * multiplier) / 256;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
multiplier = sprite->data[3];
|
|
|
|
sprite->pos2.y = (gSineTable[(u8)(sprite->data[5])] * multiplier) / 256;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sprite->data[5] += sprite->data[4];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static u8 AddScrollIndicatorArrowObject(u8 arg0, u8 x, u8 y, u16 tileTag, u16 palTag)
|
|
|
|
{
|
|
|
|
u8 spriteId;
|
|
|
|
struct SpriteTemplate spriteTemplate;
|
|
|
|
|
|
|
|
spriteTemplate = gUnknown_08614244;
|
|
|
|
spriteTemplate.tileTag = tileTag;
|
|
|
|
spriteTemplate.paletteTag = palTag;
|
|
|
|
|
|
|
|
spriteId = CreateSprite(&spriteTemplate, x, y, 0);
|
|
|
|
gSprites[spriteId].invisible = 1;
|
|
|
|
gSprites[spriteId].data[0] = 0;
|
|
|
|
gSprites[spriteId].data[1] = gUnknown_086141FC[arg0].field_0_0;
|
|
|
|
gSprites[spriteId].data[2] = gUnknown_086141FC[arg0].field_0_1;
|
|
|
|
gSprites[spriteId].data[3] = gUnknown_086141FC[arg0].field_1;
|
|
|
|
gSprites[spriteId].data[4] = gUnknown_086141FC[arg0].field_2;
|
|
|
|
gSprites[spriteId].data[5] = 0;
|
|
|
|
|
|
|
|
return spriteId;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 AddScrollIndicatorArrowPair(const struct ArrowStruct *arrowInfo, u16 *arg1)
|
|
|
|
{
|
|
|
|
struct CompressedSpriteSheet spriteSheet;
|
|
|
|
struct SpritePalette spritePal;
|
|
|
|
struct UnkIndicatorsStruct2 *data;
|
|
|
|
u8 taskId;
|
|
|
|
|
|
|
|
spriteSheet.data = gUnknown_086142C8;
|
|
|
|
spriteSheet.size = 0x100;
|
|
|
|
spriteSheet.tag = arrowInfo->tileTag;
|
|
|
|
LoadCompressedObjectPic(&spriteSheet);
|
|
|
|
|
|
|
|
if (arrowInfo->palTag == 0xFFFF)
|
|
|
|
{
|
|
|
|
LoadPalette(gUnknown_086142A8, (16 * arrowInfo->palNum) + 0x100, 0x20);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
spritePal.data = gUnknown_086142A8;
|
|
|
|
spritePal.tag = arrowInfo->palTag;
|
|
|
|
LoadSpritePalette(&spritePal);
|
|
|
|
}
|
|
|
|
|
|
|
|
taskId = CreateTask(Task_ScrollIndicatorArrowPair, 0);
|
|
|
|
data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
data->field_0 = 0;
|
|
|
|
data->field_4 = arg1;
|
|
|
|
data->field_8 = arrowInfo->unk6;
|
|
|
|
data->field_A = arrowInfo->unk8;
|
|
|
|
data->tileTag = arrowInfo->tileTag;
|
|
|
|
data->palTag = arrowInfo->palTag;
|
|
|
|
data->field_C = AddScrollIndicatorArrowObject(arrowInfo->unk0, arrowInfo->unk1, arrowInfo->unk2, arrowInfo->tileTag, arrowInfo->palTag);
|
|
|
|
data->field_D = AddScrollIndicatorArrowObject(arrowInfo->unk3, arrowInfo->unk4, arrowInfo->unk5, arrowInfo->tileTag, arrowInfo->palTag);
|
|
|
|
|
|
|
|
if (arrowInfo->palTag == 0xFFFF)
|
|
|
|
{
|
|
|
|
gSprites[data->field_C].oam.paletteNum = arrowInfo->palNum;
|
|
|
|
gSprites[data->field_D].oam.paletteNum = arrowInfo->palNum;
|
|
|
|
}
|
|
|
|
|
|
|
|
return taskId;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 AddScrollIndicatorArrowPairParametrized(u32 arg0, s32 arg1, s32 arg2, s32 arg3, s32 arg4, s32 tileTag, s32 palTag, void *arg7)
|
|
|
|
{
|
|
|
|
if (arg0 == 2 || arg0 == 3)
|
|
|
|
{
|
|
|
|
gUnknown_0203CE8C.unk0 = 2;
|
|
|
|
gUnknown_0203CE8C.unk1 = arg1;
|
|
|
|
gUnknown_0203CE8C.unk2 = arg2;
|
|
|
|
gUnknown_0203CE8C.unk3 = 3;
|
|
|
|
gUnknown_0203CE8C.unk4 = arg1;
|
|
|
|
gUnknown_0203CE8C.unk5 = arg3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gUnknown_0203CE8C.unk0 = 0;
|
|
|
|
gUnknown_0203CE8C.unk1 = arg2;
|
|
|
|
gUnknown_0203CE8C.unk2 = arg1;
|
|
|
|
gUnknown_0203CE8C.unk3 = 1;
|
|
|
|
gUnknown_0203CE8C.unk4 = arg3;
|
|
|
|
gUnknown_0203CE8C.unk5 = arg1;
|
|
|
|
}
|
|
|
|
|
|
|
|
gUnknown_0203CE8C.unk6 = 0;
|
|
|
|
gUnknown_0203CE8C.unk8 = arg4;
|
|
|
|
gUnknown_0203CE8C.tileTag = tileTag;
|
|
|
|
gUnknown_0203CE8C.palTag = palTag;
|
|
|
|
gUnknown_0203CE8C.palNum = 0;
|
|
|
|
|
|
|
|
return AddScrollIndicatorArrowPair(&gUnknown_0203CE8C, arg7);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void Task_ScrollIndicatorArrowPair(u8 taskId)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct2 *data = (void*) gTasks[taskId].data;
|
|
|
|
u16 var = (*data->field_4);
|
|
|
|
|
|
|
|
if (var == data->field_8 && var != 0xFFFF)
|
|
|
|
gSprites[data->field_C].invisible = 1;
|
|
|
|
else
|
|
|
|
gSprites[data->field_C].invisible = 0;
|
|
|
|
|
|
|
|
if (var == data->field_A)
|
|
|
|
gSprites[data->field_D].invisible = 1;
|
|
|
|
else
|
|
|
|
gSprites[data->field_D].invisible = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Task_ScrollIndicatorArrowPairOnMainMenu(u8 taskId)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct2 *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
if (data->field_1E != 0)
|
|
|
|
{
|
|
|
|
gSprites[data->field_C].invisible = 0;
|
|
|
|
gSprites[data->field_D].invisible = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gSprites[data->field_C].invisible = 1;
|
|
|
|
gSprites[data->field_D].invisible = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveScrollIndicatorArrowPair(u8 taskId)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct2 *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
if (data->tileTag != 0xFFFF)
|
|
|
|
FreeSpriteTilesByTag(data->tileTag);
|
|
|
|
if (data->palTag != 0xFFFF)
|
|
|
|
FreeSpritePaletteByTag(data->palTag);
|
|
|
|
|
|
|
|
DestroySprite(&gSprites[data->field_C]);
|
|
|
|
DestroySprite(&gSprites[data->field_D]);
|
|
|
|
|
|
|
|
DestroyTask(taskId);
|
|
|
|
}
|
|
|
|
|
|
|
|
static u8 ListMenuAddCursorObjectInternal(struct CursorStruct *cursor, u32 cursorKind)
|
|
|
|
{
|
|
|
|
switch (cursorKind)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
default:
|
|
|
|
return ListMenuAddRedOutlineCursorObject(cursor);
|
|
|
|
case 1:
|
|
|
|
return ListMenuAddRedArrowCursorObject(cursor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuUpdateCursorObject(u8 taskId, u16 x, u16 y, u32 cursorKind)
|
|
|
|
{
|
|
|
|
switch (cursorKind)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
ListMenuUpdateRedOutlineCursorObject(taskId, x, y);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
ListMenuUpdateRedArrowCursorObject(taskId, x, y);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuRemoveCursorObject(u8 taskId, u32 cursorKind)
|
|
|
|
{
|
|
|
|
switch (cursorKind)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
ListMenuRemoveRedOutlineCursorObject(taskId);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
ListMenuRemoveRedArrowCursorObject(taskId);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void Task_RedOutlineCursor(u8 taskId)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 ListMenuGetRedOutlineCursorSpriteCount(u16 arg0, u16 arg1)
|
|
|
|
{
|
|
|
|
s32 i;
|
|
|
|
s32 count = 4;
|
|
|
|
|
|
|
|
if (arg0 > 16)
|
|
|
|
{
|
|
|
|
for (i = 8; i < (arg0 - 8); i += 8)
|
|
|
|
count += 2;
|
|
|
|
}
|
|
|
|
if (arg1 > 16)
|
|
|
|
{
|
|
|
|
for (i = 8; i < (arg1 - 8); i += 8)
|
|
|
|
count += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ListMenuSetUpRedOutlineCursorSpriteOamTable(u16 arg0, u16 arg1, struct Subsprite *subsprites)
|
|
|
|
{
|
|
|
|
s32 i, j, id = 0;
|
|
|
|
|
|
|
|
subsprites[id] = sSubsprite_861425C;
|
|
|
|
subsprites[id].x = 136;
|
|
|
|
subsprites[id].y = 136;
|
|
|
|
id++;
|
|
|
|
|
|
|
|
subsprites[id] = sSubsprite_8614260;
|
|
|
|
subsprites[id].x = arg0 + 128;
|
|
|
|
subsprites[id].y = 136;
|
|
|
|
id++;
|
|
|
|
|
|
|
|
subsprites[id] = sSubsprite_8614274;
|
|
|
|
subsprites[id].x = 136;
|
|
|
|
subsprites[id].y = arg1 + 128;
|
|
|
|
id++;
|
|
|
|
|
|
|
|
subsprites[id] = sSubsprite_8614278;
|
|
|
|
subsprites[id].x = arg0 + 128;
|
|
|
|
subsprites[id].y = arg1 + 128;
|
|
|
|
id++;
|
|
|
|
|
|
|
|
if (arg0 > 16)
|
|
|
|
{
|
|
|
|
for (i = 8; i < arg0 - 8; i += 8)
|
|
|
|
{
|
|
|
|
subsprites[id] = sSubsprite_8614264;
|
|
|
|
subsprites[id].x = i - 120;
|
|
|
|
subsprites[id].y = 136;
|
|
|
|
id++;
|
|
|
|
|
|
|
|
subsprites[id] = sSubsprite_8614270;
|
|
|
|
subsprites[id].x = i - 120;
|
|
|
|
subsprites[id].y = arg1 + 128;
|
|
|
|
id++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (arg1 > 16)
|
|
|
|
{
|
|
|
|
for (j = 8; j < arg1 - 8; j += 8)
|
|
|
|
{
|
|
|
|
subsprites[id] = sSubsprite_8614268;
|
|
|
|
subsprites[id].x = 136;
|
|
|
|
subsprites[id].y = j - 120;
|
|
|
|
id++;
|
|
|
|
|
|
|
|
subsprites[id] = sSubsprite_861426C;
|
|
|
|
subsprites[id].x = arg0 + 128;
|
|
|
|
subsprites[id].y = j - 120;
|
|
|
|
id++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static u8 ListMenuAddRedOutlineCursorObject(struct CursorStruct *cursor)
|
|
|
|
{
|
|
|
|
struct CompressedSpriteSheet spriteSheet;
|
|
|
|
struct SpritePalette spritePal;
|
|
|
|
struct UnkIndicatorsStruct3 *data;
|
|
|
|
struct SpriteTemplate spriteTemplate;
|
|
|
|
u8 taskId;
|
|
|
|
|
|
|
|
spriteSheet.data = gUnknown_08614338;
|
|
|
|
spriteSheet.size = 0x100;
|
|
|
|
spriteSheet.tag = cursor->tileTag;
|
|
|
|
LoadCompressedObjectPic(&spriteSheet);
|
|
|
|
|
|
|
|
if (cursor->palTag == 0xFFFF)
|
|
|
|
{
|
|
|
|
LoadPalette(gUnknown_086142A8, (16 * cursor->palNum) + 0x100, 0x20);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
spritePal.data = gUnknown_086142A8;
|
|
|
|
spritePal.tag = cursor->palTag;
|
|
|
|
LoadSpritePalette(&spritePal);
|
|
|
|
}
|
|
|
|
|
|
|
|
taskId = CreateTask(Task_RedOutlineCursor, 0);
|
|
|
|
data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
data->tileTag = cursor->tileTag;
|
|
|
|
data->palTag = cursor->palTag;
|
|
|
|
data->subspriteTable.subspriteCount = ListMenuGetRedOutlineCursorSpriteCount(cursor->unk2, cursor->unk4);
|
|
|
|
data->subspriteTable.subsprites = data->subspritesPtr = Alloc(data->subspriteTable.subspriteCount * 4);
|
|
|
|
ListMenuSetUpRedOutlineCursorSpriteOamTable(cursor->unk2, cursor->unk4, data->subspritesPtr);
|
|
|
|
|
|
|
|
spriteTemplate = gDummySpriteTemplate;
|
|
|
|
spriteTemplate.tileTag = cursor->tileTag;
|
|
|
|
spriteTemplate.paletteTag = cursor->palTag;
|
|
|
|
|
|
|
|
data->spriteId = CreateSprite(&spriteTemplate, cursor->unk0 + 120, cursor->unk1 + 120, 0);
|
|
|
|
SetSubspriteTables(&gSprites[data->spriteId], &data->subspriteTable);
|
|
|
|
gSprites[data->spriteId].oam.priority = 0;
|
|
|
|
gSprites[data->spriteId].subpriority = 0;
|
|
|
|
gSprites[data->spriteId].subspriteTableNum = 0;
|
|
|
|
|
|
|
|
if (cursor->palTag == 0xFFFF)
|
|
|
|
{
|
|
|
|
gSprites[data->spriteId].oam.paletteNum = cursor->palNum;
|
|
|
|
}
|
|
|
|
|
|
|
|
return taskId;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuUpdateRedOutlineCursorObject(u8 taskId, u16 x, u16 y)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct3 *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
gSprites[data->spriteId].pos1.x = x + 120;
|
|
|
|
gSprites[data->spriteId].pos1.y = y + 120;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuRemoveRedOutlineCursorObject(u8 taskId)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct3 *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
Free(data->subspritesPtr);
|
|
|
|
|
|
|
|
if (data->tileTag != 0xFFFF)
|
|
|
|
FreeSpriteTilesByTag(data->tileTag);
|
|
|
|
if (data->palTag != 0xFFFF)
|
|
|
|
FreeSpritePaletteByTag(data->palTag);
|
|
|
|
|
|
|
|
DestroySprite(&gSprites[data->spriteId]);
|
|
|
|
DestroyTask(taskId);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ObjectCB_RedArrowCursor(struct Sprite *sprite)
|
|
|
|
{
|
|
|
|
sprite->pos2.x = gSineTable[(u8)(sprite->data[0])] / 64;
|
|
|
|
sprite->data[0] += 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void Task_RedArrowCursor(u8 taskId)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static u8 ListMenuAddRedArrowCursorObject(struct CursorStruct *cursor)
|
|
|
|
{
|
|
|
|
struct CompressedSpriteSheet spriteSheet;
|
|
|
|
struct SpritePalette spritePal;
|
|
|
|
struct UnkIndicatorsStruct4 *data;
|
|
|
|
struct SpriteTemplate spriteTemplate;
|
|
|
|
u8 taskId;
|
|
|
|
|
|
|
|
spriteSheet.data = gUnknown_08614378;
|
|
|
|
spriteSheet.size = 0x80;
|
|
|
|
spriteSheet.tag = cursor->tileTag;
|
|
|
|
LoadCompressedObjectPic(&spriteSheet);
|
|
|
|
|
|
|
|
if (cursor->palTag == 0xFFFF)
|
|
|
|
{
|
|
|
|
LoadPalette(gUnknown_086142A8, (16 * cursor->palNum) + 0x100, 0x20);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
spritePal.data = gUnknown_086142A8;
|
|
|
|
spritePal.tag = cursor->palTag;
|
|
|
|
LoadSpritePalette(&spritePal);
|
|
|
|
}
|
|
|
|
|
|
|
|
taskId = CreateTask(Task_RedArrowCursor, 0);
|
|
|
|
data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
data->tileTag = cursor->tileTag;
|
|
|
|
data->palTag = cursor->palTag;
|
|
|
|
|
|
|
|
spriteTemplate = gUnknown_08614290;
|
|
|
|
spriteTemplate.tileTag = cursor->tileTag;
|
|
|
|
spriteTemplate.paletteTag = cursor->palTag;
|
|
|
|
|
|
|
|
data->field_0 = CreateSprite(&spriteTemplate, cursor->unk0, cursor->unk1, 0);
|
|
|
|
gSprites[data->field_0].pos2.x = 8;
|
|
|
|
gSprites[data->field_0].pos2.y = 8;
|
|
|
|
|
|
|
|
if (cursor->palTag == 0xFFFF)
|
|
|
|
{
|
|
|
|
gSprites[data->field_0].oam.paletteNum = cursor->palNum;
|
|
|
|
}
|
|
|
|
|
|
|
|
return taskId;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuUpdateRedArrowCursorObject(u8 taskId, u16 x, u16 y)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct4 *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
gSprites[data->field_0].pos1.x = x;
|
|
|
|
gSprites[data->field_0].pos1.y = y;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ListMenuRemoveRedArrowCursorObject(u8 taskId)
|
|
|
|
{
|
|
|
|
struct UnkIndicatorsStruct4 *data = (void*) gTasks[taskId].data;
|
|
|
|
|
|
|
|
if (data->tileTag != 0xFFFF)
|
|
|
|
FreeSpriteTilesByTag(data->tileTag);
|
|
|
|
if (data->palTag != 0xFFFF)
|
|
|
|
FreeSpritePaletteByTag(data->palTag);
|
|
|
|
|
|
|
|
DestroySprite(&gSprites[data->field_0]);
|
|
|
|
DestroyTask(taskId);
|
|
|
|
}
|