210 lines
4.5 KiB
C
Raw Normal View History

2016-08-01 00:43:38 +02:00
#include <psp2/pgf.h>
#include <psp2/kernel/sysmem.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <stdarg.h>
#include <malloc.h>
#include <math.h>
#include "vita2d.h"
#include "texture_atlas.h"
#include "bin_packing_2d.h"
#include "utils.h"
#include "shared.h"
#define ATLAS_DEFAULT_W 256
#define ATLAS_DEFAULT_H 256
typedef struct vita2d_pgf {
SceFontLibHandle lib_handle;
SceFontHandle font_handle;
texture_atlas *tex_atlas;
} vita2d_pgf;
static void *pgf_alloc_func(void *userdata, unsigned int size)
{
return memalign(sizeof(int), size + sizeof(int));
}
static void pgf_free_func(void *userdata, void *p)
{
free(p);
}
vita2d_pgf *vita2d_load_default_pgf()
{
unsigned int error;
vita2d_pgf *font = malloc(sizeof(*font));
if (!font)
return NULL;
SceFontNewLibParams params = {
font,
1,
NULL,
pgf_alloc_func,
pgf_free_func,
NULL,
NULL,
NULL,
NULL,
NULL,
NULL
};
font->lib_handle = sceFontNewLib(&params, &error);
if (error != 0) {
free(font);
return NULL;
}
font->font_handle = sceFontOpen(font->lib_handle, 1, 0, &error);
if (error != 0) {
sceFontDoneLib(font->lib_handle);
free(font);
return NULL;
}
font->tex_atlas = texture_atlas_create(ATLAS_DEFAULT_W, ATLAS_DEFAULT_H,
SCE_GXM_TEXTURE_FORMAT_U8_R111);
return font;
}
void vita2d_free_pgf(vita2d_pgf *font)
{
if (font) {
sceFontClose(font->font_handle);
sceFontDoneLib(font->lib_handle);
texture_atlas_free(font->tex_atlas);
free(font);
}
}
2016-08-01 01:02:26 +02:00
static int atlas_add_glyph_pgf(vita2d_pgf *font, unsigned int character)
2016-08-01 00:43:38 +02:00
{
SceFontCharInfo char_info;
if (sceFontGetCharInfo(font->font_handle, character, &char_info) < 0)
return 0;
int pos_x;
int pos_y;
if (!texture_atlas_insert(font->tex_atlas, character,
char_info.bitmapWidth, char_info.bitmapHeight,
char_info.bitmapLeft, char_info.bitmapTop,
char_info.sfp26AdvanceH,
char_info.sfp26AdvanceV,
0, &pos_x, &pos_y))
return 0;
vita2d_texture *tex = font->tex_atlas->tex;
SceFontGlyphImage glyph_image;
glyph_image.pixelFormat = SCE_FONT_PIXELFORMAT_8;
glyph_image.xPos64 = pos_x << 6;
glyph_image.yPos64 = pos_y << 6;
glyph_image.bufWidth = vita2d_texture_get_width(tex);
glyph_image.bufHeight = vita2d_texture_get_height(tex);
glyph_image.bytesPerLine = vita2d_texture_get_stride(tex);
glyph_image.pad = 0;
glyph_image.bufferPtr = (unsigned int)vita2d_texture_get_datap(tex);
return sceFontGetCharGlyphImage(font->font_handle, character, &glyph_image) == 0;
}
int vita2d_pgf_draw_text(vita2d_pgf *font, int x, int y, unsigned int color, float scale, const char *text)
{
int pen_x = x;
while (*text) {
unsigned int character = *text++;
if (!texture_atlas_exists(font->tex_atlas, character)) {
2016-08-01 01:02:26 +02:00
if (!atlas_add_glyph_pgf(font, character)) {
2016-08-01 00:43:38 +02:00
continue;
}
}
bp2d_rectangle rect;
int bitmap_left, bitmap_top;
int advance_x, advance_y;
if (!texture_atlas_get(font->tex_atlas, character,
&rect, &bitmap_left, &bitmap_top,
&advance_x, &advance_y, NULL))
continue;
vita2d_draw_texture_tint_part_scale(font->tex_atlas->tex,
pen_x + bitmap_left * scale,
y - bitmap_top * scale,
rect.x, rect.y, rect.w, rect.h,
scale,
scale,
color);
pen_x += (advance_x >> 6) * scale;
}
return pen_x - x;
}
int vita2d_pgf_draw_textf(vita2d_pgf *font, int x, int y, unsigned int color, float scale, const char *text, ...)
{
char buf[1024];
va_list argptr;
va_start(argptr, text);
vsnprintf(buf, sizeof(buf), text, argptr);
va_end(argptr);
return vita2d_pgf_draw_text(font, x, y, color, scale, buf);
}
void vita2d_pgf_text_dimensions(vita2d_pgf *font, float scale, const char *text, int *width, int *height)
{
int pen_x = 0;
int max_h = 0;
while (*text) {
unsigned int character = *text++;
if (!texture_atlas_exists(font->tex_atlas, character)) {
2016-08-01 01:02:26 +02:00
if (!atlas_add_glyph_pgf(font, character)) {
2016-08-01 00:43:38 +02:00
continue;
}
}
bp2d_rectangle rect;
int bitmap_left, bitmap_top;
int advance_x, advance_y;
if (!texture_atlas_get(font->tex_atlas, character,
&rect, &bitmap_left, &bitmap_top,
&advance_x, &advance_y, NULL))
continue;
if (rect.h > max_h)
max_h = rect.h;
pen_x += (advance_x >> 6) * scale;
}
if (width)
*width = pen_x;
if (height)
*height = max_h;
}
int vita2d_pgf_text_width(vita2d_pgf *font, float scale, const char *text)
{
int width;
vita2d_pgf_text_dimensions(font, scale, text, &width, NULL);
return width;
}
int vita2d_pgf_text_height(vita2d_pgf *font, float scale, const char *text)
{
int height;
vita2d_pgf_text_dimensions(font, scale, text, NULL, &height);
return height;
}