mirror of
https://github.com/zeldaret/oot.git
synced 2025-01-15 21:07:15 +00:00
672728455d
* matched ntsc-1.2 fbdemo/fbdemo_fade * remove ifdef for fbdemo * remove macro for 's match
258 lines
9.2 KiB
C
258 lines
9.2 KiB
C
/**
|
|
* @file z_fbdemo.c
|
|
*
|
|
* This file implements an unused transition system that takes the current screen, partitions it into large tiles, and
|
|
* can apply an effect to them.
|
|
*
|
|
* The screen is divided into 7 rows and 10 columns of tiles. (`gScreenWidth`/ 10 = `gScreenHeight` / 7 = 0x20)
|
|
*
|
|
* @note The only coded effect has a visual effect to blend the tiles to a single point, which looks like the screen
|
|
* gets sucked into.
|
|
*/
|
|
#include "global.h"
|
|
|
|
// color framebuffer
|
|
extern u16 D_0F000000[];
|
|
|
|
// Unused background; a blue rectangle with a grey border that fills the screen
|
|
Gfx sTransTileBackgroundDL[] = {
|
|
gsDPPipeSync(),
|
|
gsDPSetCycleType(G_CYC_FILL),
|
|
gsDPSetColorImage(G_IM_FMT_RGBA, G_IM_SIZ_16b, 320, D_0F000000),
|
|
gsDPSetFillColor((GPACK_RGBA5551(65, 65, 65, 1) << 16) | GPACK_RGBA5551(65, 65, 65, 1)),
|
|
gsDPFillRectangle(0, 0, 319, 239),
|
|
gsDPPipeSync(),
|
|
gsDPSetFillColor((GPACK_RGBA5551(65, 65, 255, 1) << 16) | GPACK_RGBA5551(65, 65, 255, 1)),
|
|
gsDPFillRectangle(20, 20, 300, 220),
|
|
gsDPPipeSync(),
|
|
gsSPEndDisplayList(),
|
|
};
|
|
|
|
Gfx sTransTileSetupDL[] = {
|
|
gsDPPipeSync(),
|
|
gsSPTexture(0x8000, 0x8000, 0, G_TX_RENDERTILE, G_ON),
|
|
gsSPClearGeometryMode(G_ZBUFFER | G_SHADE | G_CULL_BOTH | G_FOG | G_LIGHTING | G_TEXTURE_GEN |
|
|
G_TEXTURE_GEN_LINEAR | G_LOD | G_SHADING_SMOOTH),
|
|
gsDPSetCombineMode(G_CC_DECALRGB, G_CC_DECALRGB),
|
|
gsDPSetOtherMode(G_AD_DISABLE | G_CD_MAGICSQ | G_CK_NONE | G_TC_FILT | G_TF_BILERP | G_TT_NONE | G_TL_TILE |
|
|
G_TD_CLAMP | G_TP_PERSP | G_CYC_1CYCLE | G_PM_1PRIMITIVE,
|
|
G_AC_NONE | G_ZS_PIXEL | G_RM_AA_OPA_SURF | G_RM_AA_OPA_SURF2),
|
|
gsSPEndDisplayList(),
|
|
};
|
|
|
|
#define SET_VERTEX(vtx, x, y, z, s, t, nx, ny, nz, alpha) \
|
|
{ \
|
|
Vtx_tn* vtxn = &(vtx)->n; \
|
|
vtxn->ob[0] = (x); \
|
|
vtxn->ob[1] = (y); \
|
|
vtxn->ob[2] = (z); \
|
|
vtxn->flag = 0; \
|
|
vtxn->tc[0] = (s); \
|
|
vtxn->tc[1] = (t); \
|
|
vtxn->n[0] = (nx); \
|
|
vtxn->n[1] = (ny); \
|
|
vtxn->n[2] = (nz); \
|
|
vtxn->a = (alpha); \
|
|
} \
|
|
(void)0
|
|
|
|
void TransitionTile_InitGraphics(TransitionTile* this) {
|
|
s32 frame;
|
|
s32 col;
|
|
s32 col2;
|
|
s32 colTex;
|
|
Vtx* vtx;
|
|
s32 row;
|
|
s32 rowTex;
|
|
Gfx* gfx;
|
|
|
|
guMtxIdent(&this->modelView);
|
|
guMtxIdent(&this->unk_98);
|
|
guOrtho(&this->projection, 0.0f, SCREEN_WIDTH, SCREEN_HEIGHT, 0.0f, -1000.0f, 1000.0f, 1.0f);
|
|
|
|
for (frame = 0; frame < 2; frame++) {
|
|
this->frame = frame;
|
|
vtx = (this->frame == 0) ? this->vtxFrame1 : this->vtxFrame2;
|
|
rowTex = 0;
|
|
for (row = 0; row < this->rows + 1; row++) {
|
|
colTex = 0;
|
|
for (col = 0; col < this->cols + 1; col++) {
|
|
SET_VERTEX(vtx++, col * 0x20, row * 0x20, -5, colTex << 6, rowTex << 6, 0, 0, 120, 255);
|
|
colTex += 0x20;
|
|
}
|
|
|
|
rowTex += 0x20;
|
|
}
|
|
}
|
|
|
|
gfx = this->gfx;
|
|
rowTex = 0;
|
|
for (row = 0; row < this->rows; row++) {
|
|
gSPVertex(gfx++, SEGMENT_ADDR(0xA, (u32)row * (this->cols + 1) * sizeof(Vtx)), 2 * (this->cols + 1), 0);
|
|
|
|
col2 = 0;
|
|
colTex = 0;
|
|
col = 0;
|
|
while (col < this->cols) {
|
|
gDPPipeSync(gfx++);
|
|
|
|
gDPLoadTextureTile(gfx++, SEGMENT_ADDR(0xB, 0), G_IM_FMT_RGBA, G_IM_SIZ_16b, SCREEN_WIDTH, SCREEN_HEIGHT,
|
|
colTex, rowTex, colTex + 0x20, rowTex + 0x20, 0, G_TX_NOMIRROR | G_TX_WRAP,
|
|
G_TX_NOMIRROR | G_TX_WRAP, G_TX_NOMASK, G_TX_NOMASK, G_TX_NOLOD, G_TX_NOLOD);
|
|
|
|
gSP1Quadrangle(gfx++, col, col + 1, col2 + this->cols + 2, this->cols + col2 + 1, 0);
|
|
|
|
colTex += 0x20;
|
|
col2++;
|
|
col++;
|
|
}
|
|
|
|
rowTex += 0x20;
|
|
}
|
|
|
|
gDPPipeSync(gfx++);
|
|
gSPEndDisplayList(gfx++);
|
|
|
|
LOG_NUM("this->col * (1 + this->row * (1 + 7 + 1)) + 1 + 1", this->rows * (1 + this->cols * 9) + 2, "../z_fbdemo.c",
|
|
144);
|
|
LOG_NUM("gp - this->gfxtbl", gfx - this->gfx, "../z_fbdemo.c", 145);
|
|
}
|
|
|
|
void TransitionTile_InitVtxData(TransitionTile* this) {
|
|
s32 row;
|
|
s32 col;
|
|
|
|
for (row = 0; row < this->rows + 1; row++) {
|
|
for (col = 0; col < this->cols + 1; col++) {
|
|
(this->vtxData + col + row * (this->cols + 1))->x = col * 0x20;
|
|
(this->vtxData + col + row * (this->cols + 1))->y = row * 0x20;
|
|
}
|
|
}
|
|
}
|
|
|
|
void TransitionTile_Destroy(TransitionTile* this) {
|
|
PRINTF("fbdemo_cleanup(%08x)\n", this);
|
|
PRINTF("msleep(100);\n");
|
|
Sleep_Msec(100);
|
|
|
|
if (this->vtxData != NULL) {
|
|
SYSTEM_ARENA_FREE(this->vtxData, "../z_fbdemo.c", 180);
|
|
this->vtxData = NULL;
|
|
}
|
|
if (this->vtxFrame1 != NULL) {
|
|
SYSTEM_ARENA_FREE(this->vtxFrame1, "../z_fbdemo.c", 181);
|
|
this->vtxFrame1 = NULL;
|
|
}
|
|
if (this->vtxFrame2 != NULL) {
|
|
SYSTEM_ARENA_FREE(this->vtxFrame2, "../z_fbdemo.c", 182);
|
|
this->vtxFrame2 = NULL;
|
|
}
|
|
if (this->gfx != NULL) {
|
|
SYSTEM_ARENA_FREE(this->gfx, "../z_fbdemo.c", 183);
|
|
this->gfx = NULL;
|
|
}
|
|
}
|
|
|
|
TransitionTile* TransitionTile_Init(TransitionTile* this, s32 cols, s32 rows) {
|
|
PRINTF("fbdemo_init(%08x, %d, %d)\n", this, cols, rows);
|
|
bzero(this, sizeof(TransitionTile));
|
|
this->frame = 0;
|
|
this->cols = cols;
|
|
this->rows = rows;
|
|
this->vtxData = SYSTEM_ARENA_MALLOC((cols + 1) * sizeof(TransitionTileVtxData) * (rows + 1), "../z_fbdemo.c", 195);
|
|
this->vtxFrame1 = SYSTEM_ARENA_MALLOC((cols + 1) * sizeof(Vtx) * (rows + 1), "../z_fbdemo.c", 196);
|
|
this->vtxFrame2 = SYSTEM_ARENA_MALLOC((cols + 1) * sizeof(Vtx) * (rows + 1), "../z_fbdemo.c", 197);
|
|
this->gfx = SYSTEM_ARENA_MALLOC((this->rows * (1 + this->cols * 9) + 2) * sizeof(Gfx), "../z_fbdemo.c", 198);
|
|
|
|
if ((this->vtxData == NULL) || (this->vtxFrame1 == NULL) || (this->vtxFrame2 == NULL) || (this->gfx == NULL)) {
|
|
PRINTF("fbdemo_init allocation error\n");
|
|
if (this->vtxData != NULL) {
|
|
SYSTEM_ARENA_FREE(this->vtxData, "../z_fbdemo.c", 202);
|
|
this->vtxData = NULL;
|
|
}
|
|
if (this->vtxFrame1 != NULL) {
|
|
SYSTEM_ARENA_FREE(this->vtxFrame1, "../z_fbdemo.c", 203);
|
|
this->vtxFrame1 = NULL;
|
|
}
|
|
if (this->vtxFrame2 != NULL) {
|
|
SYSTEM_ARENA_FREE(this->vtxFrame2, "../z_fbdemo.c", 204);
|
|
this->vtxFrame2 = NULL;
|
|
}
|
|
if (this->gfx != NULL) {
|
|
SYSTEM_ARENA_FREE(this->gfx, "../z_fbdemo.c", 205);
|
|
this->gfx = NULL;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
TransitionTile_InitGraphics(this);
|
|
TransitionTile_InitVtxData(this);
|
|
this->frame = 0;
|
|
|
|
return this;
|
|
}
|
|
|
|
void TransitionTile_SetVtx(TransitionTile* this) {
|
|
s32 row;
|
|
s32 col;
|
|
Vtx* vtx;
|
|
|
|
for (row = 0; row < this->rows + 1; row++) {
|
|
for (col = 0; col < this->cols + 1; col++) {
|
|
vtx = (this->frame == 0) ? this->vtxFrame1 : this->vtxFrame2;
|
|
(vtx + col + row * (this->cols + 1))->n.ob[0] = (this->vtxData + col + row * (this->cols + 1))->x;
|
|
|
|
vtx = (this->frame == 0) ? this->vtxFrame1 : this->vtxFrame2;
|
|
(vtx + col + row * (this->cols + 1))->n.ob[1] = (this->vtxData + col + row * (this->cols + 1))->y;
|
|
}
|
|
}
|
|
}
|
|
|
|
void TransitionTile_Draw(TransitionTile* this, Gfx** gfxP) {
|
|
Gfx* gfx = *gfxP;
|
|
|
|
gSPDisplayList(gfx++, sTransTileSetupDL);
|
|
TransitionTile_SetVtx(this);
|
|
gSPMatrix(gfx++, &this->projection, G_MTX_NOPUSH | G_MTX_LOAD | G_MTX_PROJECTION);
|
|
gSPMatrix(gfx++, &this->modelView, G_MTX_NOPUSH | G_MTX_LOAD | G_MTX_MODELVIEW);
|
|
gSPSegment(gfx++, 0xA, this->frame == 0 ? this->vtxFrame1 : this->vtxFrame2);
|
|
gSPSegment(gfx++, 0xB, this->zBuffer);
|
|
gSPDisplayList(gfx++, sTransTileSetupDL);
|
|
gSPDisplayList(gfx++, this->gfx);
|
|
gDPPipeSync(gfx++);
|
|
this->frame ^= 1;
|
|
*gfxP = gfx;
|
|
}
|
|
|
|
/**
|
|
* Blends tiles which has the visual effect of sucking those tiles into a single point
|
|
*/
|
|
void TransitionTile_Suck(TransitionTile* this) {
|
|
s32 row;
|
|
s32 col;
|
|
f32 diffX;
|
|
f32 diffY;
|
|
f32 scale;
|
|
|
|
for (row = 0; row < this->rows + 1; row++) {
|
|
for (col = 0; col < this->cols + 1; col++) {
|
|
diffX = (this->vtxData + col + row * (this->cols + 1))->x - (this->vtxData + 5 + 4 * (this->cols + 1))->x;
|
|
diffY = (this->vtxData + col + row * (this->cols + 1))->y - (this->vtxData + 5 + 4 * (this->cols + 1))->y;
|
|
scale = (SQ(diffX) + SQ(diffY)) / 100.0f;
|
|
if (scale != 0.0f) {
|
|
if (scale < 1.0f) {
|
|
scale = 1.0f;
|
|
}
|
|
(this->vtxData + col + row * (this->cols + 1))->x -= diffX / scale;
|
|
(this->vtxData + col + row * (this->cols + 1))->y -= diffY / scale;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void TransitionTile_Update(TransitionTile* this) {
|
|
}
|
|
|
|
s32 func_800B23F0(TransitionTile* this) {
|
|
return 0;
|
|
}
|