2018-09-15 13:50:23 +02:00
|
|
|
#include "global.h"
|
2020-06-19 00:14:30 +02:00
|
|
|
#include "confetti_util.h"
|
2019-09-09 03:07:54 +02:00
|
|
|
#include "malloc.h"
|
2018-09-15 13:50:23 +02:00
|
|
|
#include "main.h"
|
2020-04-08 22:24:30 +02:00
|
|
|
#include "digit_obj_util.h"
|
2018-09-15 13:50:23 +02:00
|
|
|
|
|
|
|
static EWRAM_DATA struct
|
|
|
|
{
|
|
|
|
u8 count;
|
2020-06-19 00:14:30 +02:00
|
|
|
struct ConfettiUtil *array;
|
2020-06-17 09:39:03 +02:00
|
|
|
} *sWork = NULL;
|
2018-09-15 13:50:23 +02:00
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
static void sub_81520A8(void *dest, u16 value, u8 left, u8 top, u8 width, u8 height) // Unused.
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
|
|
|
u8 i;
|
|
|
|
u8 j;
|
|
|
|
u8 x;
|
|
|
|
u8 y;
|
|
|
|
|
|
|
|
for (i = 0, y = top; i < height; i++)
|
|
|
|
{
|
|
|
|
for (x = left, j = 0; j < width; j++)
|
|
|
|
{
|
|
|
|
*(u16 *)((dest) + (y * 64 + x * 2)) = value;
|
|
|
|
x = (x + 1) % 32;
|
|
|
|
}
|
|
|
|
y = (y + 1) % 32;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
static void sub_8152134(void *dest, const u16 *src, u8 left, u8 top, u8 width, u8 height) // Unused.
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
Undo PokeCodec's PRs
This commit undoes most of PokeCodec's PRs after the debate in chat. Some
harmless or completely superseded PRs have been left alone, as there is not
much benefit in attempting to undo them.
Reverts #1104, #1108, #1115, #1118, #1119, #1124, #1126, #1127, #1132, #1136,
#1137, #1139, #1140, #1144, #1148, #1149, #1150, #1153, #1155, #1177, #1179,
#1180, #1181, #1182 and #1183.
2020-09-13 09:22:50 +02:00
|
|
|
u8 i;
|
|
|
|
u8 j;
|
|
|
|
u8 x;
|
|
|
|
u8 y;
|
2018-09-15 13:50:23 +02:00
|
|
|
const u16 *_src;
|
|
|
|
|
|
|
|
for (i = 0, _src = src, y = top; i < height; i++)
|
|
|
|
{
|
|
|
|
for (x = left, j = 0; j < width; j++)
|
|
|
|
{
|
|
|
|
*(u16 *)((dest) + (y * 64 + x * 2)) = *(_src++);
|
|
|
|
x = (x + 1) % 32;
|
|
|
|
}
|
|
|
|
y = (y + 1) % 32;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
bool32 ConfettiUtil_Init(u8 count)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
|
|
|
u8 i = 0;
|
|
|
|
|
|
|
|
if (count == 0)
|
|
|
|
return FALSE;
|
|
|
|
if (count > 64)
|
|
|
|
count = 64;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
sWork = AllocZeroed(sizeof(*sWork));
|
|
|
|
if (sWork == NULL)
|
2018-09-15 13:50:23 +02:00
|
|
|
return FALSE;
|
2020-06-19 00:14:30 +02:00
|
|
|
sWork->array = AllocZeroed(count * sizeof(struct ConfettiUtil));
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork->array == NULL)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
FREE_AND_SET_NULL(sWork);
|
2018-09-15 13:50:23 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
sWork->count = count;
|
2018-09-15 13:50:23 +02:00
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
memcpy(&sWork->array[i].oam, &gDummyOamData, sizeof(struct OamData));
|
|
|
|
sWork->array[i].dummied = TRUE;
|
2018-09-15 13:50:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
bool32 ConfettiUtil_Free(void)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
|
|
|
u8 i = 0;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork == NULL)
|
2018-09-15 13:50:23 +02:00
|
|
|
return FALSE;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
for (i = 0; i < sWork->count; i++)
|
2018-09-15 13:50:23 +02:00
|
|
|
memcpy(&gMain.oamBuffer[i + 64], &gDummyOamData, sizeof(struct OamData));
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
memset(sWork->array, 0, sWork->count * sizeof(struct ConfettiUtil));
|
2020-06-17 09:39:03 +02:00
|
|
|
FREE_AND_SET_NULL(sWork->array);
|
|
|
|
memset(sWork, 0, sizeof(*sWork));
|
|
|
|
FREE_AND_SET_NULL(sWork);
|
2018-09-15 13:50:23 +02:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
bool32 ConfettiUtil_Update(void)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
|
|
|
u8 i = 0;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork == NULL || sWork->array == NULL)
|
2018-09-15 13:50:23 +02:00
|
|
|
return FALSE;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
for (i = 0; i < sWork->count; i++)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork->array[i].active && sWork->array[i].allowUpdates)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork->array[i].callback != NULL)
|
|
|
|
sWork->array[i].callback(&sWork->array[i]);
|
2018-09-15 13:50:23 +02:00
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork->array[i].dummied)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
|
|
|
memcpy(&gMain.oamBuffer[i + 64], &gDummyOamData, sizeof(struct OamData));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
sWork->array[i].oam.y = sWork->array[i].y + sWork->array[i].yDelta;
|
|
|
|
sWork->array[i].oam.x = sWork->array[i].x + sWork->array[i].xDelta;
|
|
|
|
sWork->array[i].oam.priority = sWork->array[i].priority;
|
|
|
|
sWork->array[i].oam.tileNum = sWork->array[i].tileNum;
|
|
|
|
memcpy(&gMain.oamBuffer[i + 64], &sWork->array[i], sizeof(struct OamData));
|
2018-09-15 13:50:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
static bool32 SetAnimAndTileNum(struct ConfettiUtil *structPtr, u8 animNum)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
|
|
|
u16 tileStart;
|
|
|
|
|
|
|
|
if (structPtr == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
tileStart = GetSpriteTileStartByTag(structPtr->tileTag);
|
|
|
|
if (tileStart == 0xFFFF)
|
|
|
|
return FALSE;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
structPtr->animNum = animNum;
|
|
|
|
structPtr->tileNum = (GetTilesPerImage(structPtr->oam.shape, structPtr->oam.size) * animNum) + tileStart;
|
2018-09-15 13:50:23 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
u8 ConfettiUtil_SetCallback(u8 id, void (*func)(struct ConfettiUtil *))
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork == NULL || id >= sWork->count)
|
2018-09-15 13:50:23 +02:00
|
|
|
return 0xFF;
|
2020-06-17 09:39:03 +02:00
|
|
|
else if (!sWork->array[id].active)
|
2018-09-15 13:50:23 +02:00
|
|
|
return 0xFF;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
sWork->array[id].callback = func;
|
2018-09-15 13:50:23 +02:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
u8 ConfettiUtil_SetData(u8 id, u8 dataArrayId, s16 dataValue)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork == NULL || id >= sWork->count)
|
2018-09-15 13:50:23 +02:00
|
|
|
return 0xFF;
|
2020-06-17 09:39:03 +02:00
|
|
|
else if (!sWork->array[id].active || dataArrayId > ARRAY_COUNT(sWork->array[id].data) - 1) // - 1 b/c last slot is reserved for taskId
|
2018-09-15 13:50:23 +02:00
|
|
|
return 0xFF;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
sWork->array[id].data[dataArrayId] = dataValue;
|
2018-09-15 13:50:23 +02:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
u8 ConfettiUtil_AddNew(const struct OamData *oam, u16 tileTag, u16 palTag, s16 x, s16 y, u8 animNum, u8 priority)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-19 00:14:30 +02:00
|
|
|
struct ConfettiUtil *structPtr = NULL;
|
2018-09-15 13:50:23 +02:00
|
|
|
u8 i;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork == NULL || oam == NULL)
|
2018-09-15 13:50:23 +02:00
|
|
|
return 0xFF;
|
|
|
|
|
2020-06-17 09:39:03 +02:00
|
|
|
for (i = 0; i < sWork->count; i++)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
if (!sWork->array[i].active)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
structPtr = &sWork->array[i];
|
2018-09-15 13:50:23 +02:00
|
|
|
memset(structPtr, 0, sizeof(*structPtr));
|
|
|
|
structPtr->id = i;
|
2020-06-17 09:39:03 +02:00
|
|
|
structPtr->active = TRUE;
|
|
|
|
structPtr->allowUpdates = TRUE;
|
2018-09-15 13:50:23 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (structPtr == NULL)
|
|
|
|
return 0xFF;
|
|
|
|
|
|
|
|
memcpy(&structPtr->oam, oam, sizeof(*oam));
|
|
|
|
structPtr->tileTag = tileTag;
|
|
|
|
structPtr->palTag = palTag;
|
|
|
|
structPtr->x = x;
|
|
|
|
structPtr->y = y;
|
|
|
|
structPtr->oam.paletteNum = IndexOfSpritePaletteTag(palTag);
|
|
|
|
if (priority < 4)
|
|
|
|
{
|
|
|
|
structPtr->priority = priority;
|
|
|
|
structPtr->oam.priority = priority;
|
|
|
|
}
|
2020-06-17 09:39:03 +02:00
|
|
|
SetAnimAndTileNum(structPtr, animNum);
|
2018-09-15 13:50:23 +02:00
|
|
|
|
|
|
|
return structPtr->id;
|
|
|
|
}
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
u8 ConfettiUtil_Remove(u8 id)
|
2018-09-15 13:50:23 +02:00
|
|
|
{
|
2020-06-17 09:39:03 +02:00
|
|
|
if (sWork == NULL || !sWork->array[id].active)
|
2018-09-15 13:50:23 +02:00
|
|
|
return 0xFF;
|
|
|
|
|
2020-06-19 00:14:30 +02:00
|
|
|
memset(&sWork->array[id], 0, sizeof(struct ConfettiUtil));
|
2020-06-17 09:39:03 +02:00
|
|
|
sWork->array[id].oam.y = 160;
|
|
|
|
sWork->array[id].oam.x = 240;
|
|
|
|
sWork->array[id].dummied = TRUE;
|
2018-09-15 13:50:23 +02:00
|
|
|
memcpy(&gMain.oamBuffer[id + 64], &gDummyOamData, sizeof(struct OamData));
|
|
|
|
return id;
|
|
|
|
}
|