mirror of
https://gitlab.com/Theopse/fbi-i18n-zh.git
synced 2025-04-06 03:58:02 +08:00
797 lines
26 KiB
C
797 lines
26 KiB
C
#include <errno.h>
|
|
#include <malloc.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <3ds.h>
|
|
#include <citro3d.h>
|
|
|
|
#include "../stb_image/stb_image.h"
|
|
#include "screen.h"
|
|
#include "util.h"
|
|
|
|
#include "default_shbin.h"
|
|
|
|
GX_TRANSFER_FORMAT gpu_to_gx_format[13] = {
|
|
GX_TRANSFER_FMT_RGBA8,
|
|
GX_TRANSFER_FMT_RGB8,
|
|
GX_TRANSFER_FMT_RGB5A1,
|
|
GX_TRANSFER_FMT_RGB565,
|
|
GX_TRANSFER_FMT_RGBA4,
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8, // Unsupported
|
|
GX_TRANSFER_FMT_RGBA8 // Unsupported
|
|
};
|
|
|
|
static u32 color_config[NUM_COLORS] = {0xFF000000};
|
|
|
|
static bool c3d_initialized;
|
|
|
|
static bool shader_initialized;
|
|
static DVLB_s* dvlb;
|
|
static shaderProgram_s program;
|
|
|
|
static C3D_RenderTarget* target_top;
|
|
static C3D_RenderTarget* target_bottom;
|
|
static C3D_Mtx projection_top;
|
|
static C3D_Mtx projection_bottom;
|
|
|
|
static struct {
|
|
bool initialized;
|
|
C3D_Tex tex;
|
|
u32 width;
|
|
u32 height;
|
|
u32 pow2Width;
|
|
u32 pow2Height;
|
|
} textures[MAX_TEXTURES];
|
|
|
|
static C3D_Tex* glyph_sheets;
|
|
|
|
static u8 base_alpha = 0xFF;
|
|
|
|
static FILE* screen_open_resource(const char* path) {
|
|
u32 realPathSize = strlen(path) + 17;
|
|
char realPath[realPathSize];
|
|
|
|
snprintf(realPath, realPathSize, "sdmc:/fbi/theme/%s", path);
|
|
FILE* fd = fopen(realPath, "rb");
|
|
|
|
if(fd != NULL) {
|
|
return fd;
|
|
} else {
|
|
snprintf(realPath, realPathSize, "romfs:/%s", path);
|
|
|
|
return fopen(realPath, "rb");
|
|
}
|
|
}
|
|
|
|
void screen_init() {
|
|
if(!C3D_Init(C3D_DEFAULT_CMDBUF_SIZE * 4)) {
|
|
util_panic("Failed to initialize the GPU.");
|
|
return;
|
|
}
|
|
|
|
c3d_initialized = true;
|
|
|
|
target_top = C3D_RenderTargetCreate(240, 400, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8);
|
|
if(target_top == NULL) {
|
|
util_panic("Failed to initialize the top screen target.");
|
|
return;
|
|
}
|
|
|
|
u32 displayFlags = GX_TRANSFER_FLIP_VERT(0) | GX_TRANSFER_OUT_TILED(0) | GX_TRANSFER_RAW_COPY(0) | GX_TRANSFER_IN_FORMAT(GX_TRANSFER_FMT_RGBA8) | GX_TRANSFER_OUT_FORMAT(GX_TRANSFER_FMT_RGB8) | GX_TRANSFER_SCALING(GX_TRANSFER_SCALE_NO);
|
|
|
|
C3D_RenderTargetSetClear(target_top, C3D_CLEAR_ALL, 0, 0);
|
|
C3D_RenderTargetSetOutput(target_top, GFX_TOP, GFX_LEFT, displayFlags);
|
|
|
|
target_bottom = C3D_RenderTargetCreate(240, 320, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8);
|
|
if(target_bottom == NULL) {
|
|
util_panic("Failed to initialize the bottom screen target.");
|
|
return;
|
|
}
|
|
|
|
C3D_RenderTargetSetClear(target_bottom, C3D_CLEAR_ALL, 0, 0);
|
|
C3D_RenderTargetSetOutput(target_bottom, GFX_BOTTOM, GFX_LEFT, displayFlags);
|
|
|
|
dvlb = DVLB_ParseFile((u32*) default_shbin, default_shbin_len);
|
|
if(dvlb == NULL) {
|
|
util_panic("Failed to parse shader.");
|
|
return;
|
|
}
|
|
|
|
Result progInitRes = shaderProgramInit(&program);
|
|
if(R_FAILED(progInitRes)) {
|
|
util_panic("Failed to initialize shader program: 0x%08lX", progInitRes);
|
|
return;
|
|
}
|
|
|
|
shader_initialized = true;
|
|
|
|
Result progSetVshRes = shaderProgramSetVsh(&program, &dvlb->DVLE[0]);
|
|
if(R_FAILED(progSetVshRes)) {
|
|
util_panic("Failed to set up vertex shader: 0x%08lX", progInitRes);
|
|
return;
|
|
}
|
|
|
|
C3D_BindProgram(&program);
|
|
|
|
C3D_AttrInfo* attrInfo = C3D_GetAttrInfo();
|
|
if(attrInfo == NULL) {
|
|
util_panic("Failed to retrieve attribute info.");
|
|
return;
|
|
}
|
|
|
|
AttrInfo_Init(attrInfo);
|
|
AttrInfo_AddLoader(attrInfo, 0, GPU_FLOAT, 3);
|
|
AttrInfo_AddLoader(attrInfo, 1, GPU_FLOAT, 2);
|
|
|
|
C3D_TexEnv* env = C3D_GetTexEnv(0);
|
|
if(env == NULL) {
|
|
util_panic("Failed to retrieve combiner settings.");
|
|
return;
|
|
}
|
|
|
|
C3D_TexEnvSrc(env, C3D_Both, GPU_TEXTURE0, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
|
C3D_TexEnvOp(env, C3D_Both, GPU_TEVOP_RGB_SRC_COLOR, GPU_TEVOP_RGB_SRC_COLOR, GPU_TEVOP_RGB_SRC_COLOR);
|
|
C3D_TexEnvFunc(env, C3D_Both, GPU_REPLACE);
|
|
|
|
C3D_DepthTest(true, GPU_GEQUAL, GPU_WRITE_ALL);
|
|
|
|
Mtx_OrthoTilt(&projection_top, 0.0, 400.0, 240.0, 0.0, 0.0, 1.0, true);
|
|
Mtx_OrthoTilt(&projection_bottom, 0.0, 320.0, 240.0, 0.0, 0.0, 1.0, true);
|
|
|
|
Result fontMapRes = fontEnsureMapped();
|
|
if(R_FAILED(fontMapRes)) {
|
|
util_panic("Failed to map system font: 0x%08lX", fontMapRes);
|
|
return;
|
|
}
|
|
|
|
TGLP_s* glyphInfo = fontGetGlyphInfo();
|
|
glyph_sheets = calloc(glyphInfo->nSheets, sizeof(C3D_Tex));
|
|
if(glyph_sheets == NULL) {
|
|
util_panic("Failed to allocate font glyph texture data.");
|
|
return;
|
|
}
|
|
|
|
for(int i = 0; i < glyphInfo->nSheets; i++) {
|
|
C3D_Tex* tex = &glyph_sheets[i];
|
|
tex->data = fontGetGlyphSheetTex(i);
|
|
tex->fmt = (GPU_TEXCOLOR) glyphInfo->sheetFmt;
|
|
tex->size = glyphInfo->sheetSize;
|
|
tex->width = glyphInfo->sheetWidth;
|
|
tex->height = glyphInfo->sheetHeight;
|
|
tex->param = GPU_TEXTURE_MAG_FILTER(GPU_LINEAR) | GPU_TEXTURE_MIN_FILTER(GPU_LINEAR) | GPU_TEXTURE_WRAP_S(GPU_CLAMP_TO_EDGE) | GPU_TEXTURE_WRAP_T(GPU_CLAMP_TO_EDGE);
|
|
}
|
|
|
|
FILE* fd = screen_open_resource("textcolor.cfg");
|
|
if(fd == NULL) {
|
|
util_panic("Failed to open text color config: %s\n", strerror(errno));
|
|
return;
|
|
}
|
|
|
|
char line[128];
|
|
while(fgets(line, sizeof(line), fd) != NULL) {
|
|
char* newline = strchr(line, '\n');
|
|
if(newline != NULL) {
|
|
*newline = '\0';
|
|
}
|
|
|
|
char* equals = strchr(line, '=');
|
|
if(equals != NULL) {
|
|
char key[64] = {'\0'};
|
|
char value[64] = {'\0'};
|
|
|
|
strncpy(key, line, equals - line);
|
|
strncpy(value, equals + 1, strlen(equals) - 1);
|
|
|
|
u32 color = strtoul(value, NULL, 16);
|
|
|
|
if(strcasecmp(key, "text") == 0) {
|
|
color_config[COLOR_TEXT] = color;
|
|
} else if(strcasecmp(key, "nand") == 0) {
|
|
color_config[COLOR_NAND] = color;
|
|
} else if(strcasecmp(key, "sd") == 0) {
|
|
color_config[COLOR_SD] = color;
|
|
} else if(strcasecmp(key, "gamecard") == 0) {
|
|
color_config[COLOR_GAME_CARD] = color;
|
|
} else if(strcasecmp(key, "dstitle") == 0) {
|
|
color_config[COLOR_DS_TITLE] = color;
|
|
} else if(strcasecmp(key, "file") == 0) {
|
|
color_config[COLOR_FILE] = color;
|
|
} else if(strcasecmp(key, "directory") == 0) {
|
|
color_config[COLOR_DIRECTORY] = color;
|
|
} else if(strcasecmp(key, "enabled") == 0) {
|
|
color_config[COLOR_ENABLED] = color;
|
|
} else if(strcasecmp(key, "disabled") == 0) {
|
|
color_config[COLOR_DISABLED] = color;
|
|
} else if(strcasecmp(key, "installed") == 0) {
|
|
color_config[COLOR_INSTALLED] = color;
|
|
} else if(strcasecmp(key, "notinstalled") == 0) {
|
|
color_config[COLOR_NOT_INSTALLED] = color;
|
|
} else if(strcasecmp(key, "ticketinuse") == 0) {
|
|
color_config[COLOR_TICKET_IN_USE] = color;
|
|
} else if(strcasecmp(key, "ticketnotinuse") == 0) {
|
|
color_config[COLOR_TICKET_NOT_IN_USE] = color;
|
|
}
|
|
}
|
|
}
|
|
|
|
fclose(fd);
|
|
|
|
screen_load_texture_file(TEXTURE_BOTTOM_SCREEN_BG, "bottom_screen_bg.png", true);
|
|
screen_load_texture_file(TEXTURE_BOTTOM_SCREEN_TOP_BAR, "bottom_screen_top_bar.png", true);
|
|
screen_load_texture_file(TEXTURE_BOTTOM_SCREEN_TOP_BAR_SHADOW, "bottom_screen_top_bar_shadow.png", true);
|
|
screen_load_texture_file(TEXTURE_BOTTOM_SCREEN_BOTTOM_BAR, "bottom_screen_bottom_bar.png", true);
|
|
screen_load_texture_file(TEXTURE_BOTTOM_SCREEN_BOTTOM_BAR_SHADOW, "bottom_screen_bottom_bar_shadow.png", true);
|
|
screen_load_texture_file(TEXTURE_TOP_SCREEN_BG, "top_screen_bg.png", true);
|
|
screen_load_texture_file(TEXTURE_TOP_SCREEN_TOP_BAR, "top_screen_top_bar.png", true);
|
|
screen_load_texture_file(TEXTURE_TOP_SCREEN_TOP_BAR_SHADOW, "top_screen_top_bar_shadow.png", true);
|
|
screen_load_texture_file(TEXTURE_TOP_SCREEN_BOTTOM_BAR, "top_screen_bottom_bar.png", true);
|
|
screen_load_texture_file(TEXTURE_TOP_SCREEN_BOTTOM_BAR_SHADOW, "top_screen_bottom_bar_shadow.png", true);
|
|
screen_load_texture_file(TEXTURE_LOGO, "logo.png", true);
|
|
screen_load_texture_file(TEXTURE_SELECTION_OVERLAY, "selection_overlay.png", true);
|
|
screen_load_texture_file(TEXTURE_SCROLL_BAR, "scroll_bar.png", true);
|
|
screen_load_texture_file(TEXTURE_BUTTON_SMALL, "button_small.png", true);
|
|
screen_load_texture_file(TEXTURE_BUTTON_LARGE, "button_large.png", true);
|
|
screen_load_texture_file(TEXTURE_PROGRESS_BAR_BG, "progress_bar_bg.png", true);
|
|
screen_load_texture_file(TEXTURE_PROGRESS_BAR_CONTENT, "progress_bar_content.png", true);
|
|
screen_load_texture_file(TEXTURE_META_INFO_BOX, "meta_info_box.png", true);
|
|
screen_load_texture_file(TEXTURE_META_INFO_BOX_SHADOW, "meta_info_box_shadow.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_CHARGING, "battery_charging.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_0, "battery0.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_1, "battery1.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_2, "battery2.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_3, "battery3.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_4, "battery4.png", true);
|
|
screen_load_texture_file(TEXTURE_BATTERY_5, "battery5.png", true);
|
|
screen_load_texture_file(TEXTURE_WIFI_DISCONNECTED, "wifi_disconnected.png", true);
|
|
screen_load_texture_file(TEXTURE_WIFI_0, "wifi0.png", true);
|
|
screen_load_texture_file(TEXTURE_WIFI_1, "wifi1.png", true);
|
|
screen_load_texture_file(TEXTURE_WIFI_2, "wifi2.png", true);
|
|
screen_load_texture_file(TEXTURE_WIFI_3, "wifi3.png", true);
|
|
}
|
|
|
|
void screen_exit() {
|
|
for(u32 id = 0; id < MAX_TEXTURES; id++) {
|
|
screen_unload_texture(id);
|
|
}
|
|
|
|
if(glyph_sheets != NULL) {
|
|
free(glyph_sheets);
|
|
glyph_sheets = NULL;
|
|
}
|
|
|
|
if(shader_initialized) {
|
|
shaderProgramFree(&program);
|
|
shader_initialized = false;
|
|
}
|
|
|
|
if(dvlb != NULL) {
|
|
DVLB_Free(dvlb);
|
|
dvlb = NULL;
|
|
}
|
|
|
|
if(target_top != NULL) {
|
|
C3D_RenderTargetDelete(target_top);
|
|
target_top = NULL;
|
|
}
|
|
|
|
if(target_bottom != NULL) {
|
|
C3D_RenderTargetDelete(target_bottom);
|
|
target_bottom = NULL;
|
|
}
|
|
|
|
if(c3d_initialized) {
|
|
C3D_Fini();
|
|
c3d_initialized = false;
|
|
}
|
|
}
|
|
|
|
void screen_set_base_alpha(u8 alpha) {
|
|
base_alpha = alpha;
|
|
}
|
|
|
|
static u32 screen_next_pow_2(u32 i) {
|
|
i--;
|
|
i |= i >> 1;
|
|
i |= i >> 2;
|
|
i |= i >> 4;
|
|
i |= i >> 8;
|
|
i |= i >> 16;
|
|
i++;
|
|
|
|
return i;
|
|
}
|
|
|
|
void screen_load_texture(u32 id, void* data, u32 size, u32 width, u32 height, GPU_TEXCOLOR format, bool linearFilter) {
|
|
if(id >= MAX_TEXTURES) {
|
|
util_panic("Attempted to load buffer to invalid texture ID \"%lu\".", id);
|
|
return;
|
|
}
|
|
|
|
u32 pow2Width = screen_next_pow_2(width);
|
|
if(pow2Width < 64) {
|
|
pow2Width = 64;
|
|
}
|
|
|
|
u32 pow2Height = screen_next_pow_2(height);
|
|
if(pow2Height < 64) {
|
|
pow2Height = 64;
|
|
}
|
|
|
|
u32 pixelSize = size / width / height;
|
|
|
|
u8* pow2Tex = linearAlloc(pow2Width * pow2Height * pixelSize);
|
|
if(pow2Tex == NULL) {
|
|
util_panic("Failed to allocate temporary texture buffer.");
|
|
return;
|
|
}
|
|
|
|
memset(pow2Tex, 0, pow2Width * pow2Height * pixelSize);
|
|
|
|
for(u32 x = 0; x < width; x++) {
|
|
for(u32 y = 0; y < height; y++) {
|
|
u32 dataPos = (y * width + x) * pixelSize;
|
|
u32 pow2TexPos = (y * pow2Width + x) * pixelSize;
|
|
|
|
for(u32 i = 0; i < pixelSize; i++) {
|
|
pow2Tex[pow2TexPos + i] = ((u8*) data)[dataPos + i];
|
|
}
|
|
}
|
|
}
|
|
|
|
textures[id].initialized = true;
|
|
textures[id].width = width;
|
|
textures[id].height = height;
|
|
textures[id].pow2Width = pow2Width;
|
|
textures[id].pow2Height = pow2Height;
|
|
|
|
if(!C3D_TexInit(&textures[id].tex, (int) pow2Width, (int) pow2Height, format)) {
|
|
util_panic("Failed to initialize texture.");
|
|
return;
|
|
}
|
|
|
|
C3D_TexSetFilter(&textures[id].tex, linearFilter ? GPU_LINEAR : GPU_NEAREST, GPU_NEAREST);
|
|
|
|
Result flushRes = GSPGPU_FlushDataCache(pow2Tex, pow2Width * pow2Height * 4);
|
|
if(R_FAILED(flushRes)) {
|
|
util_panic("Failed to flush texture buffer: 0x%08lX", flushRes);
|
|
return;
|
|
}
|
|
|
|
C3D_SafeDisplayTransfer((u32*) pow2Tex, GX_BUFFER_DIM(pow2Width, pow2Height), (u32*) textures[id].tex.data, GX_BUFFER_DIM(pow2Width, pow2Height), GX_TRANSFER_FLIP_VERT(1) | GX_TRANSFER_OUT_TILED(1) | GX_TRANSFER_RAW_COPY(0) | GX_TRANSFER_IN_FORMAT((u32) gpu_to_gx_format[format]) | GX_TRANSFER_OUT_FORMAT((u32) gpu_to_gx_format[format]) | GX_TRANSFER_SCALING(GX_TRANSFER_SCALE_NO));
|
|
gspWaitForPPF();
|
|
|
|
linearFree(pow2Tex);
|
|
}
|
|
|
|
u32 screen_load_texture_auto(void* tiledData, u32 size, u32 width, u32 height, GPU_TEXCOLOR format, bool linearFilter) {
|
|
int id = -1;
|
|
for(int i = TEXTURE_AUTO_START; i < MAX_TEXTURES; i++) {
|
|
if(!textures[i].initialized) {
|
|
id = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(id == -1) {
|
|
util_panic("Attempted to load auto texture from buffer without free textures.");
|
|
return 0;
|
|
}
|
|
|
|
screen_load_texture((u32) id, tiledData, size, width, height, format, linearFilter);
|
|
return (u32) id;
|
|
}
|
|
|
|
void screen_load_texture_file(u32 id, const char* path, bool linearFilter) {
|
|
if(id >= MAX_TEXTURES) {
|
|
util_panic("Attempted to load path \"%s\" to invalid texture ID \"%lu\".", path, id);
|
|
return;
|
|
}
|
|
|
|
FILE* fd = screen_open_resource(path);
|
|
if(fd == NULL) {
|
|
util_panic("Failed to load PNG file \"%s\": %s", path, strerror(errno));
|
|
return;
|
|
}
|
|
|
|
int width;
|
|
int height;
|
|
int depth;
|
|
u8* image = stbi_load_from_file(fd, &width, &height, &depth, STBI_rgb_alpha);
|
|
fclose(fd);
|
|
|
|
if(image == NULL || depth != STBI_rgb_alpha) {
|
|
util_panic("Failed to load PNG file \"%s\".", path);
|
|
return;
|
|
}
|
|
|
|
for(u32 x = 0; x < width; x++) {
|
|
for(u32 y = 0; y < height; y++) {
|
|
u32 pos = (y * width + x) * 4;
|
|
|
|
u8 c1 = image[pos + 0];
|
|
u8 c2 = image[pos + 1];
|
|
u8 c3 = image[pos + 2];
|
|
u8 c4 = image[pos + 3];
|
|
|
|
image[pos + 0] = c4;
|
|
image[pos + 1] = c3;
|
|
image[pos + 2] = c2;
|
|
image[pos + 3] = c1;
|
|
}
|
|
}
|
|
|
|
screen_load_texture(id, image, (u32) (width * height * 4), (u32) width, (u32) height, GPU_RGBA8, linearFilter);
|
|
|
|
free(image);
|
|
}
|
|
|
|
u32 screen_load_texture_file_auto(const char* path, bool linearFilter) {
|
|
int id = -1;
|
|
for(int i = TEXTURE_AUTO_START; i < MAX_TEXTURES; i++) {
|
|
if(!textures[i].initialized) {
|
|
id = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(id == -1) {
|
|
util_panic("Attempted to load auto texture from path \"%s\" without free textures.", path);
|
|
return 0;
|
|
}
|
|
|
|
screen_load_texture_file((u32) id, path, linearFilter);
|
|
return (u32) id;
|
|
}
|
|
|
|
static u32 screen_tiled_texture_index(u32 x, u32 y, u32 w, u32 h) {
|
|
return (((y >> 3) * (w >> 3) + (x >> 3)) << 6) + ((x & 1) | ((y & 1) << 1) | ((x & 2) << 1) | ((y & 2) << 2) | ((x & 4) << 2) | ((y & 4) << 3));
|
|
}
|
|
|
|
void screen_load_texture_tiled(u32 id, void* tiledData, u32 size, u32 width, u32 height, GPU_TEXCOLOR format, bool linearFilter) {
|
|
if(id >= MAX_TEXTURES) {
|
|
util_panic("Attempted to load tiled data to invalid texture ID \"%lu\".", id);
|
|
return;
|
|
}
|
|
|
|
u8* untiledData = (u8*) calloc(1, size);
|
|
if(untiledData == NULL) {
|
|
util_panic("Failed to allocate buffer for texture untiling.");
|
|
return;
|
|
}
|
|
|
|
u32 pixelSize = size / width / height;
|
|
|
|
for(u32 x = 0; x < width; x++) {
|
|
for(u32 y = 0; y < height; y++) {
|
|
u32 tiledDataPos = screen_tiled_texture_index(x, y, width, height) * pixelSize;
|
|
u32 untiledDataPos = (y * width + x) * pixelSize;
|
|
|
|
for(u32 i = 0; i < pixelSize; i++) {
|
|
untiledData[untiledDataPos + i] = ((u8*) tiledData)[tiledDataPos + i];
|
|
}
|
|
}
|
|
}
|
|
|
|
screen_load_texture(id, untiledData, size, width, height, format, linearFilter);
|
|
|
|
free(untiledData);
|
|
}
|
|
|
|
u32 screen_load_texture_tiled_auto(void* tiledData, u32 size, u32 width, u32 height, GPU_TEXCOLOR format, bool linearFilter) {
|
|
int id = -1;
|
|
for(int i = TEXTURE_AUTO_START; i < MAX_TEXTURES; i++) {
|
|
if(!textures[i].initialized) {
|
|
id = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(id == -1) {
|
|
util_panic("Attempted to load auto texture from tiled data without free textures.");
|
|
return 0;
|
|
}
|
|
|
|
screen_load_texture_tiled((u32) id, tiledData, size, width, height, format, linearFilter);
|
|
return (u32) id;
|
|
}
|
|
|
|
void screen_unload_texture(u32 id) {
|
|
if(id >= MAX_TEXTURES) {
|
|
util_panic("Attempted to unload invalid texture ID \"%lu\".", id);
|
|
return;
|
|
}
|
|
|
|
if(textures[id].initialized) {
|
|
C3D_TexDelete(&textures[id].tex);
|
|
|
|
textures[id].initialized = false;
|
|
textures[id].width = 0;
|
|
textures[id].height = 0;
|
|
textures[id].pow2Width = 0;
|
|
textures[id].pow2Height = 0;
|
|
}
|
|
}
|
|
|
|
void screen_get_texture_size(u32* width, u32* height, u32 id) {
|
|
if(id >= MAX_TEXTURES || !textures[id].initialized) {
|
|
util_panic("Attempted to get size of invalid texture ID \"%lu\".", id);
|
|
return;
|
|
}
|
|
|
|
if(width) {
|
|
*width = textures[id].width;
|
|
}
|
|
|
|
if(height) {
|
|
*height = textures[id].height;
|
|
}
|
|
}
|
|
|
|
void screen_begin_frame() {
|
|
if(!C3D_FrameBegin(C3D_FRAME_SYNCDRAW)) {
|
|
util_panic("Failed to begin frame.");
|
|
return;
|
|
}
|
|
}
|
|
|
|
void screen_end_frame() {
|
|
C3D_FrameEnd(0);
|
|
}
|
|
|
|
void screen_select(gfxScreen_t screen) {
|
|
if(!C3D_FrameDrawOn(screen == GFX_TOP ? target_top : target_bottom)) {
|
|
util_panic("Failed to select render target.");
|
|
return;
|
|
}
|
|
|
|
C3D_FVUnifMtx4x4(GPU_VERTEX_SHADER, shaderInstanceGetUniformLocation(program.vertexShader, "projection"), screen == GFX_TOP ? &projection_top : &projection_bottom);
|
|
}
|
|
|
|
void screen_set_scissor(bool enabled, u32 x, u32 y, u32 width, u32 height) {
|
|
C3D_SetScissor(enabled ? GPU_SCISSOR_NORMAL : GPU_SCISSOR_DISABLE, 240 - (y + height), x, 240 - y, x + width);
|
|
}
|
|
|
|
static void screen_set_blend(u32 color, bool rgb, bool alpha) {
|
|
C3D_TexEnv* env = C3D_GetTexEnv(0);
|
|
if(env == NULL) {
|
|
util_panic("Failed to retrieve combiner settings.");
|
|
return;
|
|
}
|
|
|
|
if(rgb) {
|
|
C3D_TexEnvSrc(env, C3D_RGB, GPU_CONSTANT, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
|
C3D_TexEnvFunc(env, C3D_RGB, GPU_REPLACE);
|
|
} else {
|
|
C3D_TexEnvSrc(env, C3D_RGB, GPU_TEXTURE0, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
|
C3D_TexEnvFunc(env, C3D_RGB, GPU_REPLACE);
|
|
}
|
|
|
|
if(alpha) {
|
|
C3D_TexEnvSrc(env, C3D_Alpha, GPU_TEXTURE0, GPU_CONSTANT, GPU_PRIMARY_COLOR);
|
|
C3D_TexEnvFunc(env, C3D_Alpha, GPU_MODULATE);
|
|
} else {
|
|
C3D_TexEnvSrc(env, C3D_Alpha, GPU_TEXTURE0, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
|
C3D_TexEnvFunc(env, C3D_Alpha, GPU_REPLACE);
|
|
}
|
|
|
|
C3D_TexEnvColor(env, color);
|
|
}
|
|
|
|
static void screen_draw_quad(float x1, float y1, float x2, float y2, float tx1, float ty1, float tx2, float ty2) {
|
|
C3D_ImmDrawBegin(GPU_TRIANGLES);
|
|
|
|
C3D_ImmSendAttrib(x1, y1, 0.5f, 0.0f);
|
|
C3D_ImmSendAttrib(tx1, ty1, 0.0f, 0.0f);
|
|
|
|
C3D_ImmSendAttrib(x2, y2, 0.5f, 0.0f);
|
|
C3D_ImmSendAttrib(tx2, ty2, 0.0f, 0.0f);
|
|
|
|
C3D_ImmSendAttrib(x2, y1, 0.5f, 0.0f);
|
|
C3D_ImmSendAttrib(tx2, ty1, 0.0f, 0.0f);
|
|
|
|
C3D_ImmSendAttrib(x1, y1, 0.5f, 0.0f);
|
|
C3D_ImmSendAttrib(tx1, ty1, 0.0f, 0.0f);
|
|
|
|
C3D_ImmSendAttrib(x1, y2, 0.5f, 0.0f);
|
|
C3D_ImmSendAttrib(tx1, ty2, 0.0f, 0.0f);
|
|
|
|
C3D_ImmSendAttrib(x2, y2, 0.5f, 0.0f);
|
|
C3D_ImmSendAttrib(tx2, ty2, 0.0f, 0.0f);
|
|
|
|
C3D_ImmDrawEnd();
|
|
}
|
|
|
|
void screen_draw_texture(u32 id, float x, float y, float width, float height) {
|
|
if(id >= MAX_TEXTURES || !textures[id].initialized) {
|
|
util_panic("Attempted to draw invalid texture ID \"%lu\".", id);
|
|
return;
|
|
}
|
|
|
|
if(base_alpha != 0xFF) {
|
|
screen_set_blend(base_alpha << 24, false, true);
|
|
}
|
|
|
|
C3D_TexBind(0, &textures[id].tex);
|
|
screen_draw_quad(x, y, x + width, y + height, 0, 0, (float) textures[id].width / (float) textures[id].pow2Width, (float) textures[id].height / (float) textures[id].pow2Height);
|
|
|
|
if(base_alpha != 0xFF) {
|
|
screen_set_blend(0, false, false);
|
|
}
|
|
}
|
|
|
|
void screen_draw_texture_crop(u32 id, float x, float y, float width, float height) {
|
|
if(id >= MAX_TEXTURES || !textures[id].initialized) {
|
|
util_panic("Attempted to draw invalid texture ID \"%lu\".", id);
|
|
return;
|
|
}
|
|
|
|
if(base_alpha != 0xFF) {
|
|
screen_set_blend(base_alpha << 24, false, true);
|
|
}
|
|
|
|
C3D_TexBind(0, &textures[id].tex);
|
|
screen_draw_quad(x, y, x + width, y + height, 0, 0, width / (float) textures[id].pow2Width, height / (float) textures[id].pow2Height);
|
|
|
|
if(base_alpha != 0xFF) {
|
|
screen_set_blend(0, false, false);
|
|
}
|
|
}
|
|
|
|
float screen_get_font_height(float scaleY) {
|
|
return scaleY * fontGetInfo()->lineFeed;
|
|
}
|
|
|
|
static void screen_get_string_size_internal(float* width, float* height, const char* text, float scaleX, float scaleY, bool oneLine, bool wrap, float wrapX) {
|
|
float w = 0;
|
|
float h = 0;
|
|
float lineWidth = 0;
|
|
|
|
if(text != NULL) {
|
|
h = scaleY * fontGetInfo()->lineFeed;
|
|
|
|
const uint8_t* p = (const uint8_t*) text;
|
|
const uint8_t* lastAlign = p;
|
|
u32 code = 0;
|
|
ssize_t units = -1;
|
|
while(*p && (units = decode_utf8(&code, p)) != -1 && code > 0) {
|
|
p += units;
|
|
|
|
if(code == '\n' || (wrap && lineWidth + scaleX * fontGetCharWidthInfo(fontGlyphIndexFromCodePoint(code))->charWidth >= wrapX)) {
|
|
lastAlign = p;
|
|
|
|
if(lineWidth > w) {
|
|
w = lineWidth;
|
|
}
|
|
|
|
lineWidth = 0;
|
|
|
|
if(oneLine) {
|
|
break;
|
|
}
|
|
|
|
h += scaleY * fontGetInfo()->lineFeed;
|
|
}
|
|
|
|
if(code != '\n') {
|
|
u32 num = 1;
|
|
if(code == '\t') {
|
|
code = ' ';
|
|
num = 4 - (p - units - lastAlign) % 4;
|
|
|
|
lastAlign = p;
|
|
}
|
|
|
|
lineWidth += (scaleX * fontGetCharWidthInfo(fontGlyphIndexFromCodePoint(code))->charWidth) * num;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(width) {
|
|
*width = lineWidth > w ? lineWidth : w;
|
|
}
|
|
|
|
if(height) {
|
|
*height = h;
|
|
}
|
|
}
|
|
|
|
void screen_get_string_size(float* width, float* height, const char* text, float scaleX, float scaleY) {
|
|
screen_get_string_size_internal(width, height, text, scaleX, scaleY, false, false, 0);
|
|
}
|
|
|
|
void screen_get_string_size_wrap(float* width, float* height, const char* text, float scaleX, float scaleY, float wrapX) {
|
|
screen_get_string_size_internal(width, height, text, scaleX, scaleY, false, true, wrapX);
|
|
}
|
|
|
|
static void screen_draw_string_internal(const char* text, float x, float y, float scaleX, float scaleY, u32 colorId, bool centerLines, bool wrap, float wrapX) {
|
|
if(text == NULL) {
|
|
return;
|
|
}
|
|
|
|
u32 blendColor = color_config[colorId];
|
|
if(base_alpha != 0xFF) {
|
|
float alpha1 = ((blendColor >> 24) & 0xFF) / 255.0f;
|
|
float alpha2 = base_alpha / 255.0f;
|
|
float blendedAlpha = alpha1 * alpha2;
|
|
|
|
blendColor = (((u32) (blendedAlpha * 0xFF)) << 24) | (blendColor & 0x00FFFFFF);
|
|
}
|
|
|
|
screen_set_blend(blendColor, true, true);
|
|
|
|
float stringWidth;
|
|
screen_get_string_size_internal(&stringWidth, NULL, text, scaleX, scaleY, false, wrap, wrapX);
|
|
|
|
float lineWidth;
|
|
screen_get_string_size_internal(&lineWidth, NULL, text, scaleX, scaleY, true, wrap, wrapX);
|
|
|
|
float currX = x;
|
|
if(centerLines) {
|
|
currX += (stringWidth - lineWidth) / 2;
|
|
}
|
|
|
|
int lastSheet = -1;
|
|
|
|
const uint8_t* p = (const uint8_t*) text;
|
|
const uint8_t* lastAlign = p;
|
|
u32 code = 0;
|
|
ssize_t units = -1;
|
|
while(*p && (units = decode_utf8(&code, p)) != -1 && code > 0) {
|
|
p += units;
|
|
|
|
if(code == '\n' || (wrap && currX + scaleX * fontGetCharWidthInfo(fontGlyphIndexFromCodePoint(code))->charWidth >= wrapX)) {
|
|
lastAlign = p;
|
|
|
|
screen_get_string_size_internal(&lineWidth, NULL, (const char*) p, scaleX, scaleY, true, wrap, wrapX);
|
|
|
|
currX = x;
|
|
if(centerLines) {
|
|
currX += (stringWidth - lineWidth) / 2;
|
|
}
|
|
|
|
y += scaleY * fontGetInfo()->lineFeed;
|
|
}
|
|
|
|
if(code != '\n') {
|
|
u32 num = 1;
|
|
if(code == '\t') {
|
|
code = ' ';
|
|
num = 4 - (p - units - lastAlign) % 4;
|
|
|
|
lastAlign = p;
|
|
}
|
|
|
|
fontGlyphPos_s data;
|
|
fontCalcGlyphPos(&data, fontGlyphIndexFromCodePoint(code), GLYPH_POS_CALC_VTXCOORD, scaleX, scaleY);
|
|
|
|
if(data.sheetIndex != lastSheet) {
|
|
lastSheet = data.sheetIndex;
|
|
C3D_TexBind(0, &glyph_sheets[lastSheet]);
|
|
}
|
|
|
|
for(u32 i = 0; i < num; i++) {
|
|
screen_draw_quad(currX + data.vtxcoord.left, y + data.vtxcoord.top, currX + data.vtxcoord.right, y + data.vtxcoord.bottom, data.texcoord.left, data.texcoord.top, data.texcoord.right, data.texcoord.bottom);
|
|
|
|
currX += data.xAdvance;
|
|
}
|
|
}
|
|
}
|
|
|
|
screen_set_blend(0, false, false);
|
|
}
|
|
|
|
void screen_draw_string(const char* text, float x, float y, float scaleX, float scaleY, u32 colorId, bool centerLines) {
|
|
screen_draw_string_internal(text, x, y, scaleX, scaleY, colorId, centerLines, false, 0);
|
|
}
|
|
|
|
void screen_draw_string_wrap(const char* text, float x, float y, float scaleX, float scaleY, u32 colorId, bool centerLines, float wrapX) {
|
|
screen_draw_string_internal(text, x, y, scaleX, scaleY, colorId, centerLines, true, wrapX);
|
|
}
|