pokeemerald/src/palette_util.c

505 lines
17 KiB
C
Raw Normal View History

2018-12-08 11:56:59 -06:00
#include "global.h"
#include "palette.h"
2020-08-04 19:24:36 -04:00
#include "palette_util.h"
2018-12-08 11:56:59 -06:00
#include "util.h"
2020-08-04 19:19:36 -04:00
// "RouletteFlash" is more accurately a general flashing/fading util
// this file handles fading the palettes for the color/icon selections on the Roulette wheel
2020-07-31 14:55:42 -04:00
// but it also handles the "pulse blend" effect of Mirage Tower
2020-08-04 19:19:36 -04:00
void RouletteFlash_Reset(struct RouletteFlashUtil *flash)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
flash->enabled = 0;
flash->flags = 0;
memset(&flash->palettes, 0, sizeof(flash->palettes));
2018-12-08 11:56:59 -06:00
}
2020-08-04 19:19:36 -04:00
u8 RouletteFlash_Add(struct RouletteFlashUtil *flash, u8 id, const struct RouletteFlashSettings *settings)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (id >= ARRAY_COUNT(flash->palettes) || flash->palettes[id].available)
2018-12-08 11:56:59 -06:00
return 0xFF;
2020-08-04 19:19:36 -04:00
flash->palettes[id].settings.color = settings->color;
flash->palettes[id].settings.paletteOffset = settings->paletteOffset;
flash->palettes[id].settings.numColors = settings->numColors;
flash->palettes[id].settings.delay = settings->delay;
flash->palettes[id].settings.unk6 = settings->unk6;
flash->palettes[id].settings.numFadeCycles = settings->numFadeCycles;
flash->palettes[id].settings.unk7_5 = settings->unk7_5;
flash->palettes[id].settings.colorDeltaDir = settings->colorDeltaDir;
flash->palettes[id].state = 0;
flash->palettes[id].available = TRUE;
flash->palettes[id].fadeCycleCounter = 0;
flash->palettes[id].delayCounter = 0;
if (flash->palettes[id].settings.colorDeltaDir < 0)
flash->palettes[id].colorDelta = -1;
2018-12-08 11:56:59 -06:00
else
2020-08-04 19:19:36 -04:00
flash->palettes[id].colorDelta = 1;
2018-12-08 11:56:59 -06:00
2020-08-04 19:19:36 -04:00
return id;
2018-12-08 11:56:59 -06:00
}
2020-08-04 19:19:36 -04:00
// Unused
static u8 RouletteFlash_Remove(struct RouletteFlashUtil *flash, u8 id)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (id >= ARRAY_COUNT(flash->palettes))
2018-12-08 11:56:59 -06:00
return 0xFF;
2020-08-04 19:19:36 -04:00
if (!flash->palettes[id].available)
2018-12-08 11:56:59 -06:00
return 0xFF;
2020-08-04 19:19:36 -04:00
memset(&flash->palettes[id], 0, sizeof(flash->palettes[id]));
return id;
2018-12-08 11:56:59 -06:00
}
2020-08-04 19:19:36 -04:00
static u8 RouletteFlash_FadePalette(struct RouletteFlashPalette *pal)
2018-12-08 11:56:59 -06:00
{
u8 i;
u8 returnval;
2020-08-04 19:19:36 -04:00
for (i = 0; i < pal->settings.numColors; i++)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
struct PlttData *faded = (struct PlttData *)&gPlttBufferFaded[pal->settings.paletteOffset + i];
struct PlttData *unfaded = (struct PlttData *)&gPlttBufferUnfaded[pal->settings.paletteOffset + i];
2018-12-08 11:56:59 -06:00
2020-08-04 19:19:36 -04:00
switch (pal->state)
2018-12-08 11:56:59 -06:00
{
case 1:
2020-08-04 19:19:36 -04:00
// Fade color
if (faded->r + pal->colorDelta >= 0 && faded->r + pal->colorDelta < 32)
faded->r += pal->colorDelta;
if (faded->g + pal->colorDelta >= 0 && faded->g + pal->colorDelta < 32)
faded->g += pal->colorDelta;
if (faded->b + pal->colorDelta >= 0 && faded->b + pal->colorDelta < 32)
faded->b += pal->colorDelta;
2018-12-08 11:56:59 -06:00
break;
case 2:
2020-08-04 19:19:36 -04:00
// Fade back to original color
if (pal->colorDelta < 0)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (faded->r + pal->colorDelta >= unfaded->r)
faded->r += pal->colorDelta;
if (faded->g + pal->colorDelta >= unfaded->g)
faded->g += pal->colorDelta;
if (faded->b + pal->colorDelta >= unfaded->b)
faded->b += pal->colorDelta;
2018-12-08 11:56:59 -06:00
}
else
{
2020-08-04 19:19:36 -04:00
if (faded->r + pal->colorDelta <= unfaded->r)
faded->r += pal->colorDelta;
if (faded->g + pal->colorDelta <= unfaded->g)
faded->g += pal->colorDelta;
if (faded->b + pal->colorDelta <= unfaded->b)
faded->b += pal->colorDelta;
2018-12-08 11:56:59 -06:00
}
break;
}
}
2020-08-04 19:19:36 -04:00
if ((u32)pal->fadeCycleCounter++ != pal->settings.numFadeCycles)
2018-12-08 11:56:59 -06:00
{
returnval = 0;
}
else
{
2020-08-04 19:19:36 -04:00
pal->fadeCycleCounter = 0;
pal->colorDelta *= -1;
if (pal->state == 1)
pal->state++;
2018-12-08 11:56:59 -06:00
else
2020-08-04 19:19:36 -04:00
pal->state--;
2018-12-08 11:56:59 -06:00
returnval = 1;
}
return returnval;
}
2020-08-04 19:19:36 -04:00
static u8 RouletteFlash_FlashPalette(struct RouletteFlashPalette *pal)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
u8 i = 0;
switch (pal->state)
2018-12-08 11:56:59 -06:00
{
case 1:
2020-08-04 19:19:36 -04:00
// Flash to color
for (; i < pal->settings.numColors; i++)
gPlttBufferFaded[pal->settings.paletteOffset + i] = pal->settings.color;
pal->state++;
2018-12-08 11:56:59 -06:00
break;
case 2:
2020-08-04 19:19:36 -04:00
// Restore to original color
for (; i < pal->settings.numColors; i++)
gPlttBufferFaded[pal->settings.paletteOffset + i] = gPlttBufferUnfaded[pal->settings.paletteOffset + i];
pal->state--;
2018-12-08 11:56:59 -06:00
break;
}
return 1;
}
2020-08-04 19:19:36 -04:00
void RouletteFlash_Run(struct RouletteFlashUtil *flash)
2018-12-08 11:56:59 -06:00
{
u8 i = 0;
2020-08-04 19:19:36 -04:00
if (flash->enabled)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
for (i = 0; i < ARRAY_COUNT(flash->palettes); i++)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if ((flash->flags >> i) & 1)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (--flash->palettes[i].delayCounter == (u8)-1)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (flash->palettes[i].settings.color & FLASHUTIL_USE_EXISTING_COLOR)
RouletteFlash_FadePalette(&flash->palettes[i]);
2018-12-08 11:56:59 -06:00
else
2020-08-04 19:19:36 -04:00
RouletteFlash_FlashPalette(&flash->palettes[i]);
2018-12-08 11:56:59 -06:00
2020-08-04 19:19:36 -04:00
flash->palettes[i].delayCounter = flash->palettes[i].settings.delay;
2018-12-08 11:56:59 -06:00
}
}
}
}
}
2020-08-04 19:19:36 -04:00
void RouletteFlash_Enable(struct RouletteFlashUtil *flash, u16 flags)
2018-12-08 11:56:59 -06:00
{
u8 i = 0;
2020-08-04 19:19:36 -04:00
flash->enabled++;
for (i = 0; i < ARRAY_COUNT(flash->palettes); i++)
2018-12-08 11:56:59 -06:00
{
2020-08-03 17:31:34 -04:00
if ((flags >> i) & 1)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (flash->palettes[i].available)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
flash->flags |= 1 << i;
flash->palettes[i].state = 1;
2018-12-08 11:56:59 -06:00
}
}
}
}
2020-08-04 19:19:36 -04:00
void RouletteFlash_Stop(struct RouletteFlashUtil *flash, u16 flags)
2018-12-08 11:56:59 -06:00
{
u8 i;
2020-08-04 19:19:36 -04:00
for (i = 0; i < ARRAY_COUNT(flash->palettes); i++)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if ((flash->flags >> i) & 1)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
if (flash->palettes[i].available)
2018-12-08 11:56:59 -06:00
{
2020-08-03 17:31:34 -04:00
if ((flags >> i) & 1)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
u32 offset = flash->palettes[i].settings.paletteOffset;
2018-12-08 11:56:59 -06:00
u16 *faded = &gPlttBufferFaded[offset];
u16 *unfaded = &gPlttBufferUnfaded[offset];
2020-08-04 19:19:36 -04:00
memcpy(faded, unfaded, flash->palettes[i].settings.numColors * 2);
flash->palettes[i].state = 0;
flash->palettes[i].fadeCycleCounter = 0;
flash->palettes[i].delayCounter = 0;
if (flash->palettes[i].settings.colorDeltaDir < 0)
flash->palettes[i].colorDelta = -1;
2018-12-08 11:56:59 -06:00
else
2020-08-04 19:19:36 -04:00
flash->palettes[i].colorDelta = 1;
2018-12-08 11:56:59 -06:00
}
}
}
}
2020-08-04 19:19:36 -04:00
2020-08-03 17:31:34 -04:00
if (flags == 0xFFFF)
2018-12-08 11:56:59 -06:00
{
2020-08-04 19:19:36 -04:00
// Stopped all
flash->enabled = 0;
flash->flags = 0;
2018-12-08 11:56:59 -06:00
}
else
{
2020-08-04 19:19:36 -04:00
flash->flags &= ~flags;
2018-12-08 11:56:59 -06:00
}
}
void InitPulseBlend(struct PulseBlend *pulseBlend)
2018-12-08 11:56:59 -06:00
{
u8 i = 0;
pulseBlend->usedPulseBlendPalettes = 0;
memset(&pulseBlend->pulseBlendPalettes, 0, sizeof(pulseBlend->pulseBlendPalettes));
2018-12-08 11:56:59 -06:00
for (; i < 16; i++)
pulseBlend->pulseBlendPalettes[i].paletteSelector = i;
2018-12-08 11:56:59 -06:00
}
int InitPulseBlendPaletteSettings(struct PulseBlend *pulseBlend, const struct PulseBlendSettings *settings)
2018-12-08 11:56:59 -06:00
{
u8 i = 0;
struct PulseBlendPalette *pulseBlendPalette = NULL;
2018-12-08 11:56:59 -06:00
if (!pulseBlend->pulseBlendPalettes[0].inUse)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette = &pulseBlend->pulseBlendPalettes[0];
2018-12-08 11:56:59 -06:00
}
else
{
while (++i < 16)
{
if (!pulseBlend->pulseBlendPalettes[i].inUse)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette = &pulseBlend->pulseBlendPalettes[i];
2018-12-08 11:56:59 -06:00
break;
}
}
}
if (pulseBlendPalette == NULL)
2018-12-08 11:56:59 -06:00
return 0xFF;
pulseBlendPalette->blendCoeff = 0;
pulseBlendPalette->fadeDirection = 0;
pulseBlendPalette->available = 1;
pulseBlendPalette->inUse = 1;
pulseBlendPalette->delayCounter = 0;
pulseBlendPalette->fadeCycleCounter = 0;
memcpy(&pulseBlendPalette->pulseBlendSettings, settings, sizeof(*settings));
2018-12-08 11:56:59 -06:00
return i;
}
static void ClearPulseBlendPalettesSettings(struct PulseBlendPalette *pulseBlendPalette)
2018-12-08 11:56:59 -06:00
{
u16 i;
if (!pulseBlendPalette->available && pulseBlendPalette->pulseBlendSettings.restorePaletteOnUnload)
2018-12-08 11:56:59 -06:00
{
for (i = pulseBlendPalette->pulseBlendSettings.paletteOffset; i < pulseBlendPalette->pulseBlendSettings.paletteOffset + pulseBlendPalette->pulseBlendSettings.numColors; i++)
2018-12-08 11:56:59 -06:00
gPlttBufferFaded[i] = gPlttBufferUnfaded[i];
}
memset(&pulseBlendPalette->pulseBlendSettings, 0, sizeof(pulseBlendPalette->pulseBlendSettings));
pulseBlendPalette->blendCoeff = 0;
pulseBlendPalette->fadeDirection = 0;
pulseBlendPalette->unk1_5 = 0;
pulseBlendPalette->available = 1;
pulseBlendPalette->inUse = 0;
pulseBlendPalette->fadeCycleCounter = 0;
pulseBlendPalette->delayCounter = 0;
2018-12-08 11:56:59 -06:00
}
void UnloadUsedPulseBlendPalettes(struct PulseBlend *pulseBlend, u16 pulseBlendPaletteSelector, u8 multiSelection)
2018-12-08 11:56:59 -06:00
{
u16 i = 0;
if (!multiSelection)
2018-12-08 11:56:59 -06:00
{
ClearPulseBlendPalettesSettings(&pulseBlend->pulseBlendPalettes[pulseBlendPaletteSelector & 0xF]);
2018-12-08 11:56:59 -06:00
}
else
{
for (i = 0; i < 16; i++)
{
if ((pulseBlendPaletteSelector & 1) && pulseBlend->pulseBlendPalettes[i].inUse)
ClearPulseBlendPalettesSettings(&pulseBlend->pulseBlendPalettes[i]);
2018-12-08 11:56:59 -06:00
pulseBlendPaletteSelector >>= 1;
2018-12-08 11:56:59 -06:00
}
}
}
void MarkUsedPulseBlendPalettes(struct PulseBlend *pulseBlend, u16 pulseBlendPaletteSelector, u8 multiSelection)
2018-12-08 11:56:59 -06:00
{
u8 i = 0;
if (!multiSelection)
2018-12-08 11:56:59 -06:00
{
i = pulseBlendPaletteSelector & 0xF;
pulseBlend->pulseBlendPalettes[i].available = 0;
pulseBlend->usedPulseBlendPalettes |= 1 << i;
2018-12-08 11:56:59 -06:00
}
else
{
for (i = 0; i < 16; i++)
{
if (!(pulseBlendPaletteSelector & 1) || !pulseBlend->pulseBlendPalettes[i].inUse || !pulseBlend->pulseBlendPalettes[i].available)
2018-12-08 11:56:59 -06:00
{
pulseBlendPaletteSelector <<= 1;
2018-12-08 11:56:59 -06:00
}
else
{
pulseBlend->pulseBlendPalettes[i].available = 0;
pulseBlend->usedPulseBlendPalettes |= 1 << i;
2018-12-08 11:56:59 -06:00
}
}
}
2018-12-08 11:56:59 -06:00
}
void UnmarkUsedPulseBlendPalettes(struct PulseBlend *pulseBlend, u16 pulseBlendPaletteSelector, u8 multiSelection)
2018-12-08 11:56:59 -06:00
{
u16 i;
struct PulseBlendPalette *pulseBlendPalette;
2018-12-08 11:56:59 -06:00
u8 j = 0;
if (!multiSelection)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette = &pulseBlend->pulseBlendPalettes[pulseBlendPaletteSelector & 0xF];
if (!pulseBlendPalette->available && pulseBlendPalette->inUse)
2018-12-08 11:56:59 -06:00
{
if (pulseBlendPalette->pulseBlendSettings.restorePaletteOnUnload)
2018-12-08 11:56:59 -06:00
{
for (i = pulseBlendPalette->pulseBlendSettings.paletteOffset; i < pulseBlendPalette->pulseBlendSettings.paletteOffset + pulseBlendPalette->pulseBlendSettings.numColors; i++)
2018-12-08 11:56:59 -06:00
gPlttBufferFaded[i] = gPlttBufferUnfaded[i];
}
pulseBlendPalette->available = 1;
pulseBlend->usedPulseBlendPalettes &= ~(1 << j);
2018-12-08 11:56:59 -06:00
}
}
else
{
for (j = 0; j < 16; j++)
{
pulseBlendPalette = &pulseBlend->pulseBlendPalettes[j];
if (!(pulseBlendPaletteSelector & 1) || pulseBlendPalette->available || !pulseBlendPalette->inUse)
2018-12-08 11:56:59 -06:00
{
pulseBlendPaletteSelector <<= 1;
2018-12-08 11:56:59 -06:00
}
else
{
if (pulseBlendPalette->pulseBlendSettings.restorePaletteOnUnload)
2018-12-08 11:56:59 -06:00
{
for (i = pulseBlendPalette->pulseBlendSettings.paletteOffset; i < pulseBlendPalette->pulseBlendSettings.paletteOffset + pulseBlendPalette->pulseBlendSettings.numColors; i++)
2018-12-08 11:56:59 -06:00
gPlttBufferFaded[i] = gPlttBufferUnfaded[i];
}
pulseBlendPalette->available = 1;
pulseBlend->usedPulseBlendPalettes &= ~(1 << j);
2018-12-08 11:56:59 -06:00
}
}
}
}
void UpdatePulseBlend(struct PulseBlend *pulseBlend)
2018-12-08 11:56:59 -06:00
{
struct PulseBlendPalette *pulseBlendPalette;
2018-12-08 11:56:59 -06:00
u8 i = 0;
if (pulseBlend->usedPulseBlendPalettes)
2018-12-08 11:56:59 -06:00
{
for (i = 0; i < 16; i++)
{
pulseBlendPalette = &pulseBlend->pulseBlendPalettes[i];
if ((!pulseBlendPalette->available && pulseBlendPalette->inUse) && (!gPaletteFade.active || !pulseBlendPalette->pulseBlendSettings.unk7_7))
2018-12-08 11:56:59 -06:00
{
if (--pulseBlendPalette->delayCounter == 0xFF)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette->delayCounter = pulseBlendPalette->pulseBlendSettings.delay;
BlendPalette(pulseBlendPalette->pulseBlendSettings.paletteOffset, pulseBlendPalette->pulseBlendSettings.numColors, pulseBlendPalette->blendCoeff, pulseBlendPalette->pulseBlendSettings.blendColor);
switch (pulseBlendPalette->pulseBlendSettings.fadeType)
2018-12-08 11:56:59 -06:00
{
case 0: // Fade all the way to the max blend amount, then wrap around
// BUG: This comparison will never be true for maxBlendCoeff values that are >= 8. This is because
// maxBlendCoeff is a signed 4-bit field, but blendCoeff is an unsigned 4-bit field. This code is never
// reached, anyway, so the bug is not observable in vanilla gameplay.
if (pulseBlendPalette->blendCoeff++ == pulseBlendPalette->pulseBlendSettings.maxBlendCoeff)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette->fadeCycleCounter++;
pulseBlendPalette->blendCoeff = 0;
2018-12-08 11:56:59 -06:00
}
break;
case 1: // Fade in and out
if (pulseBlendPalette->fadeDirection)
2018-12-08 11:56:59 -06:00
{
if (--pulseBlendPalette->blendCoeff == 0)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette->fadeCycleCounter++;
pulseBlendPalette->fadeDirection ^= 1;
2018-12-08 11:56:59 -06:00
}
}
else
{
u8 max = (pulseBlendPalette->pulseBlendSettings.maxBlendCoeff - 1) & 0xF;
if (pulseBlendPalette->blendCoeff++ == max)
2018-12-08 11:56:59 -06:00
{
pulseBlendPalette->fadeCycleCounter++;
pulseBlendPalette->fadeDirection ^= 1;
2018-12-08 11:56:59 -06:00
}
}
break;
case (MODERN ? -2 : 2): // Flip back and forth
// This code is never reached in vanilla
if (pulseBlendPalette->fadeDirection)
pulseBlendPalette->blendCoeff = 0;
2018-12-08 11:56:59 -06:00
else
pulseBlendPalette->blendCoeff = pulseBlendPalette->pulseBlendSettings.maxBlendCoeff & 0xF;
pulseBlendPalette->fadeDirection ^= 1;
pulseBlendPalette->fadeCycleCounter++;
2018-12-08 11:56:59 -06:00
break;
}
if (pulseBlendPalette->pulseBlendSettings.numFadeCycles != 0xFF
&& pulseBlendPalette->fadeCycleCounter == pulseBlendPalette->pulseBlendSettings.numFadeCycles)
UnmarkUsedPulseBlendPalettes(pulseBlend, pulseBlendPalette->paletteSelector, FALSE);
2018-12-08 11:56:59 -06:00
}
}
}
}
}
2020-07-31 14:55:42 -04:00
// Below used for the Roulette grid
void FillTilemapRect(u16 *dest, u16 value, u8 left, u8 top, u8 width, u8 height)
2018-12-08 11:56:59 -06:00
{
u16 *_dest;
u8 i;
u8 j;
i = 0;
dest = &dest[top * 32 + left];
for (; i < height; i++)
{
_dest = dest + i * 32;
for (j = 0; j < width; j++)
*_dest++ = value;
2018-12-08 11:56:59 -06:00
}
}
2020-07-31 14:55:42 -04:00
void SetTilemapRect(u16 *dest, u16 *src, u8 left, u8 top, u8 width, u8 height)
2018-12-08 11:56:59 -06:00
{
u16 *_dest;
u16 *_src = src;
u8 i;
u8 j;
i = 0;
dest = &dest[top * 32 + left];
for (; i < height; i++)
{
_dest = dest + i * 32;
for (j = 0; j < width; j++)
*_dest++ = *_src++;
}
}
static void FillTilemapRect_Unused(void *dest, u16 value, u8 left, u8 top, u8 width, u8 height)
{
u8 i, j;
u8 x, 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;
}
}
static void SetTilemapRect_Unused(void *dest, const u16 *src, u8 left, u8 top, u8 width, u8 height)
{
u8 i, j;
u8 x, y;
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;
2018-12-08 11:56:59 -06:00
}
y = (y + 1) % 32;
2018-12-08 11:56:59 -06:00
}
}