2015-01-12 22:38:39 +00:00
|
|
|
/* RetroArch - A frontend for libretro.
|
|
|
|
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
|
|
|
* Copyright (C) 2011-2015 - Daniel De Matteis
|
|
|
|
*
|
|
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "../gl_common.h"
|
|
|
|
#include "../video_shader_driver.h"
|
|
|
|
|
|
|
|
#define emit(c, vx, vy) do { \
|
|
|
|
font_vertex[ 2 * (6 * i + c) + 0] = (x + (delta_x + off_x + vx * width) * scale) * inv_win_width; \
|
|
|
|
font_vertex[ 2 * (6 * i + c) + 1] = (y + (delta_y - off_y - vy * height) * scale) * inv_win_height; \
|
|
|
|
font_tex_coords[ 2 * (6 * i + c) + 0] = (tex_x + vx * width) * inv_tex_size_x; \
|
|
|
|
font_tex_coords[ 2 * (6 * i + c) + 1] = (tex_y + vy * height) * inv_tex_size_y; \
|
|
|
|
font_color[ 4 * (6 * i + c) + 0] = color[0]; \
|
|
|
|
font_color[ 4 * (6 * i + c) + 1] = color[1]; \
|
|
|
|
font_color[ 4 * (6 * i + c) + 2] = color[2]; \
|
|
|
|
font_color[ 4 * (6 * i + c) + 3] = color[3]; \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define MAX_MSG_LEN_CHUNK 64
|
|
|
|
|
2015-03-24 19:43:53 +00:00
|
|
|
typedef struct gl_raster_block {
|
|
|
|
bool active;
|
|
|
|
bool reuse;
|
|
|
|
bool fullscreen;
|
|
|
|
unsigned allocated;
|
|
|
|
struct gl_coords coords;
|
|
|
|
} gl_raster_block_t;
|
|
|
|
|
2015-01-12 22:38:39 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
gl_t *gl;
|
|
|
|
GLuint tex;
|
|
|
|
unsigned tex_width, tex_height;
|
|
|
|
|
|
|
|
const font_renderer_driver_t *font_driver;
|
|
|
|
void *font_data;
|
2015-03-24 19:43:53 +00:00
|
|
|
|
|
|
|
gl_raster_block_t block;
|
2015-01-12 22:38:39 +00:00
|
|
|
} gl_raster_t;
|
|
|
|
|
|
|
|
static void *gl_raster_font_init_font(void *gl_data,
|
|
|
|
const char *font_path, float font_size)
|
|
|
|
{
|
|
|
|
unsigned width, height;
|
|
|
|
uint8_t *tmp_buffer;
|
|
|
|
const struct font_atlas *atlas = NULL;
|
|
|
|
gl_raster_t *font = (gl_raster_t*)calloc(1, sizeof(*font));
|
|
|
|
|
|
|
|
if (!font)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
font->gl = (gl_t*)gl_data;
|
|
|
|
|
|
|
|
if (!font_renderer_create_default(&font->font_driver,
|
|
|
|
&font->font_data, font_path, font_size))
|
|
|
|
{
|
|
|
|
RARCH_WARN("Couldn't init font renderer.\n");
|
|
|
|
free(font);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
glGenTextures(1, &font->tex);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, font->tex);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
|
|
|
|
atlas = font->font_driver->get_atlas(font->font_data);
|
|
|
|
|
|
|
|
width = next_pow2(atlas->width);
|
|
|
|
height = next_pow2(atlas->height);
|
|
|
|
|
|
|
|
/* Ideally, we'd use single component textures, but the
|
|
|
|
* difference in ways to do that between core GL and GLES/legacy GL
|
|
|
|
* is too great to bother going down that route. */
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height,
|
|
|
|
0, GL_RGBA, GL_UNSIGNED_BYTE, NULL);
|
|
|
|
|
|
|
|
tmp_buffer = (uint8_t*)malloc(atlas->width * atlas->height * 4);
|
|
|
|
|
|
|
|
if (tmp_buffer)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
uint8_t *dst = tmp_buffer;
|
|
|
|
const uint8_t *src = atlas->buffer;
|
|
|
|
|
|
|
|
for (i = 0; i < atlas->width * atlas->height; i++)
|
|
|
|
{
|
|
|
|
*dst++ = 0xff;
|
|
|
|
*dst++ = 0xff;
|
|
|
|
*dst++ = 0xff;
|
|
|
|
*dst++ = *src++;
|
|
|
|
}
|
|
|
|
|
|
|
|
glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, atlas->width,
|
|
|
|
atlas->height, GL_RGBA, GL_UNSIGNED_BYTE, tmp_buffer);
|
|
|
|
free(tmp_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
font->tex_width = width;
|
|
|
|
font->tex_height = height;
|
|
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, font->gl->texture[font->gl->tex_index]);
|
|
|
|
return font;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gl_raster_font_free_font(void *data)
|
|
|
|
{
|
|
|
|
gl_raster_t *font = (gl_raster_t*)data;
|
|
|
|
if (!font)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (font->font_driver && font->font_data)
|
|
|
|
font->font_driver->free(font->font_data);
|
|
|
|
|
|
|
|
glDeleteTextures(1, &font->tex);
|
|
|
|
free(font);
|
|
|
|
}
|
|
|
|
|
2015-01-23 00:54:17 +00:00
|
|
|
static int get_message_width(gl_raster_t *font, const char *msg)
|
|
|
|
{
|
|
|
|
int delta_x;
|
|
|
|
unsigned i, msg_len_full, msg_len;
|
|
|
|
|
|
|
|
msg_len_full = strlen(msg);
|
|
|
|
msg_len = min(msg_len_full, MAX_MSG_LEN_CHUNK);
|
|
|
|
|
|
|
|
delta_x = 0;
|
|
|
|
|
|
|
|
while (msg_len_full)
|
|
|
|
{
|
|
|
|
for (i = 0; i < msg_len; i++)
|
|
|
|
{
|
|
|
|
const struct font_glyph *glyph =
|
|
|
|
font->font_driver->get_glyph(font->font_data, (uint8_t)msg[i]);
|
|
|
|
if (!glyph)
|
|
|
|
glyph = font->font_driver->get_glyph(font->font_data, '?'); /* Do something smarter here ... */
|
|
|
|
if (!glyph)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
delta_x += glyph->advance_x;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg_len_full -= msg_len;
|
|
|
|
msg += msg_len;
|
|
|
|
msg_len = min(msg_len_full, MAX_MSG_LEN_CHUNK);
|
|
|
|
}
|
|
|
|
|
|
|
|
return delta_x;
|
|
|
|
}
|
2015-01-12 22:38:39 +00:00
|
|
|
|
2015-03-24 18:00:46 +00:00
|
|
|
static void draw_vertices(gl_t *gl, const struct gl_coords *coords)
|
|
|
|
{
|
|
|
|
gl->shader->set_coords(coords);
|
|
|
|
gl->shader->set_mvp(gl, &gl->mvp_no_rot);
|
2015-03-24 19:43:53 +00:00
|
|
|
|
2015-03-24 18:00:46 +00:00
|
|
|
glDrawArrays(GL_TRIANGLES, 0, coords->vertices);
|
|
|
|
}
|
|
|
|
|
2015-03-24 19:43:53 +00:00
|
|
|
static bool realloc_checked(void **ptr, size_t size)
|
|
|
|
{
|
|
|
|
void *nptr;
|
|
|
|
|
|
|
|
if (size == 0)
|
|
|
|
{
|
|
|
|
if (*ptr)
|
|
|
|
free(*ptr);
|
|
|
|
*ptr = NULL;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*ptr == NULL)
|
|
|
|
nptr = malloc(size);
|
|
|
|
else
|
|
|
|
nptr = realloc(*ptr, size);
|
|
|
|
|
|
|
|
if (nptr)
|
|
|
|
*ptr = nptr;
|
|
|
|
|
|
|
|
return nptr != NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool resize_block(gl_raster_t *font, unsigned new_size)
|
|
|
|
{
|
|
|
|
gl_raster_block_t *block = &font->block;
|
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
if (!font->block.allocated || new_size > font->block.allocated - 1)
|
|
|
|
{
|
|
|
|
unsigned actual_new_size = next_pow2(new_size);
|
|
|
|
|
|
|
|
bool vsucceeded = realloc_checked((void**)&block->coords.vertex, sizeof(GLfloat) * 2 * actual_new_size);
|
|
|
|
bool csuccceeded = realloc_checked((void**)&block->coords.color, sizeof(GLfloat) * 4 * actual_new_size);
|
|
|
|
bool tsuccceeded = realloc_checked((void**)&block->coords.tex_coord, sizeof(GLfloat) * 2 * actual_new_size);
|
|
|
|
bool lsuccceeded = realloc_checked((void**)&block->coords.lut_tex_coord, sizeof(GLfloat) * 2 * actual_new_size);
|
|
|
|
|
|
|
|
if (vsucceeded && csuccceeded && tsuccceeded && lsuccceeded)
|
|
|
|
{
|
|
|
|
font->block.allocated = actual_new_size;
|
|
|
|
block->coords.vertices = new_size;
|
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
block->coords.vertices = new_size;
|
|
|
|
success = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool append_vertices(gl_raster_t *font, const struct gl_coords *coords)
|
|
|
|
{
|
|
|
|
gl_raster_block_t *block = &font->block;
|
|
|
|
unsigned old_size = block->coords.vertices;
|
|
|
|
if (resize_block(font, block->coords.vertices + coords->vertices))
|
|
|
|
{
|
|
|
|
const size_t base_size = coords->vertices * sizeof(GLfloat);
|
|
|
|
memcpy((void*)(block->coords.vertex+old_size*2), coords->vertex, base_size * 2);
|
|
|
|
memcpy((void*)(block->coords.color+old_size*4), coords->color, base_size * 4);
|
|
|
|
memcpy((void*)(block->coords.tex_coord+old_size*2), coords->tex_coord, base_size * 2);
|
|
|
|
memcpy((void*)(block->coords.lut_tex_coord+old_size*2), coords->lut_tex_coord, base_size * 2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
RARCH_WARN("Allocation failed.");
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-01-12 22:38:39 +00:00
|
|
|
static void render_message(gl_raster_t *font, const char *msg, GLfloat scale,
|
2015-01-23 00:54:17 +00:00
|
|
|
const GLfloat color[4], GLfloat pos_x, GLfloat pos_y, bool align_right)
|
2015-01-12 22:38:39 +00:00
|
|
|
{
|
|
|
|
int x, y, delta_x, delta_y;
|
|
|
|
float inv_tex_size_x, inv_tex_size_y, inv_win_width, inv_win_height;
|
|
|
|
unsigned i, msg_len_full, msg_len;
|
|
|
|
GLfloat font_tex_coords[2 * 6 * MAX_MSG_LEN_CHUNK];
|
|
|
|
GLfloat font_vertex[2 * 6 * MAX_MSG_LEN_CHUNK];
|
|
|
|
GLfloat font_color[4 * 6 * MAX_MSG_LEN_CHUNK];
|
|
|
|
gl_t *gl = font->gl;
|
2015-03-24 18:00:46 +00:00
|
|
|
struct gl_coords coords;
|
2015-01-12 22:38:39 +00:00
|
|
|
|
|
|
|
msg_len_full = strlen(msg);
|
|
|
|
msg_len = min(msg_len_full, MAX_MSG_LEN_CHUNK);
|
|
|
|
|
|
|
|
x = roundf(pos_x * gl->vp.width);
|
|
|
|
y = roundf(pos_y * gl->vp.height);
|
|
|
|
delta_x = 0;
|
|
|
|
delta_y = 0;
|
|
|
|
|
2015-01-23 00:54:17 +00:00
|
|
|
if (align_right)
|
|
|
|
x -= get_message_width(font, msg);
|
|
|
|
|
2015-01-12 22:38:39 +00:00
|
|
|
inv_tex_size_x = 1.0f / font->tex_width;
|
|
|
|
inv_tex_size_y = 1.0f / font->tex_height;
|
|
|
|
inv_win_width = 1.0f / font->gl->vp.width;
|
|
|
|
inv_win_height = 1.0f / font->gl->vp.height;
|
|
|
|
|
|
|
|
while (msg_len_full)
|
|
|
|
{
|
|
|
|
for (i = 0; i < msg_len; i++)
|
|
|
|
{
|
|
|
|
int off_x, off_y, tex_x, tex_y, width, height;
|
|
|
|
const struct font_glyph *glyph =
|
|
|
|
font->font_driver->get_glyph(font->font_data, (uint8_t)msg[i]);
|
|
|
|
if (!glyph)
|
|
|
|
glyph = font->font_driver->get_glyph(font->font_data, '?'); /* Do something smarter here ... */
|
|
|
|
if (!glyph)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
off_x = glyph->draw_offset_x;
|
|
|
|
off_y = glyph->draw_offset_y;
|
|
|
|
tex_x = glyph->atlas_offset_x;
|
|
|
|
tex_y = glyph->atlas_offset_y;
|
|
|
|
width = glyph->width;
|
|
|
|
height = glyph->height;
|
|
|
|
|
|
|
|
emit(0, 0, 1); /* Bottom-left */
|
|
|
|
emit(1, 1, 1); /* Bottom-right */
|
|
|
|
emit(2, 0, 0); /* Top-left */
|
|
|
|
|
|
|
|
emit(3, 1, 0); /* Top-right */
|
|
|
|
emit(4, 0, 0); /* Top-left */
|
|
|
|
emit(5, 1, 1); /* Bottom-right */
|
|
|
|
#undef emit
|
|
|
|
|
|
|
|
delta_x += glyph->advance_x;
|
|
|
|
delta_y -= glyph->advance_y;
|
|
|
|
}
|
|
|
|
|
2015-03-24 18:00:46 +00:00
|
|
|
coords.tex_coord = font_tex_coords;
|
|
|
|
coords.vertex = font_vertex;
|
|
|
|
coords.color = font_color;
|
|
|
|
coords.vertices = 6 * msg_len;
|
|
|
|
coords.lut_tex_coord = gl->coords.lut_tex_coord;
|
|
|
|
|
2015-03-24 19:43:53 +00:00
|
|
|
if (font->block.active)
|
|
|
|
append_vertices(font, &coords);
|
|
|
|
else
|
|
|
|
draw_vertices(gl, &coords);
|
2015-01-12 22:38:39 +00:00
|
|
|
|
|
|
|
msg_len_full -= msg_len;
|
|
|
|
msg += msg_len;
|
|
|
|
msg_len = min(msg_len_full, MAX_MSG_LEN_CHUNK);
|
|
|
|
}
|
2015-03-24 18:00:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void setup_viewport(gl_raster_t *font, bool full_screen)
|
|
|
|
{
|
|
|
|
gl_t *gl = font->gl;
|
|
|
|
|
|
|
|
gl_set_viewport(gl, gl->win_width, gl->win_height, full_screen, false);
|
|
|
|
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
glBlendEquation(GL_FUNC_ADD);
|
|
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, font->tex);
|
|
|
|
|
|
|
|
if (gl->shader && gl->shader->use)
|
|
|
|
gl->shader->use(gl, GL_SHADER_STOCK_BLEND);
|
|
|
|
}
|
2015-01-12 22:38:39 +00:00
|
|
|
|
2015-03-24 18:00:46 +00:00
|
|
|
static void restore_viewport(gl_t *gl)
|
|
|
|
{
|
2015-01-12 22:38:39 +00:00
|
|
|
glBindTexture(GL_TEXTURE_2D, gl->texture[gl->tex_index]);
|
2015-03-24 18:00:46 +00:00
|
|
|
|
|
|
|
glDisable(GL_BLEND);
|
|
|
|
gl_set_viewport(gl, gl->win_width, gl->win_height, false, true);
|
2015-01-12 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void gl_raster_font_render_msg(void *data, const char *msg,
|
|
|
|
const struct font_params *params)
|
|
|
|
{
|
|
|
|
GLfloat x, y, scale, drop_mod;
|
|
|
|
GLfloat color[4], color_dark[4];
|
|
|
|
int drop_x, drop_y;
|
|
|
|
bool full_screen;
|
2015-01-23 00:54:17 +00:00
|
|
|
bool align_right;
|
2015-01-12 22:38:39 +00:00
|
|
|
gl_t *gl = NULL;
|
|
|
|
gl_raster_t *font = (gl_raster_t*)data;
|
2015-03-20 21:08:36 +00:00
|
|
|
settings_t *settings = config_get_ptr();
|
2015-01-12 22:38:39 +00:00
|
|
|
|
|
|
|
if (!font)
|
|
|
|
return;
|
|
|
|
|
|
|
|
gl = font->gl;
|
|
|
|
|
|
|
|
if (params)
|
|
|
|
{
|
2015-03-20 21:08:36 +00:00
|
|
|
x = params->x;
|
|
|
|
y = params->y;
|
|
|
|
scale = params->scale;
|
2015-01-12 22:38:39 +00:00
|
|
|
full_screen = params->full_screen;
|
2015-01-23 00:54:17 +00:00
|
|
|
align_right = params->align_right;
|
2015-03-20 21:08:36 +00:00
|
|
|
drop_x = params->drop_x;
|
|
|
|
drop_y = params->drop_y;
|
|
|
|
drop_mod = params->drop_mod;
|
2015-01-12 22:38:39 +00:00
|
|
|
|
2015-03-20 21:08:36 +00:00
|
|
|
color[0] = FONT_COLOR_GET_RED(params->color) / 255.0f;
|
|
|
|
color[1] = FONT_COLOR_GET_GREEN(params->color) / 255.0f;
|
|
|
|
color[2] = FONT_COLOR_GET_BLUE(params->color) / 255.0f;
|
|
|
|
color[3] = FONT_COLOR_GET_ALPHA(params->color) / 255.0f;
|
2015-01-12 22:38:39 +00:00
|
|
|
|
|
|
|
/* If alpha is 0.0f, turn it into default 1.0f */
|
|
|
|
if (color[3] <= 0.0f)
|
|
|
|
color[3] = 1.0f;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-03-20 21:08:36 +00:00
|
|
|
x = settings->video.msg_pos_x;
|
|
|
|
y = settings->video.msg_pos_y;
|
|
|
|
scale = 1.0f;
|
2015-01-12 22:38:39 +00:00
|
|
|
full_screen = false;
|
2015-01-23 00:54:17 +00:00
|
|
|
align_right = false;
|
2015-01-12 22:38:39 +00:00
|
|
|
|
2015-03-20 21:08:36 +00:00
|
|
|
color[0] = settings->video.msg_color_r;
|
|
|
|
color[1] = settings->video.msg_color_g;
|
|
|
|
color[2] = settings->video.msg_color_b;
|
2015-01-12 22:38:39 +00:00
|
|
|
color[3] = 1.0f;
|
|
|
|
|
|
|
|
drop_x = -2;
|
|
|
|
drop_y = -2;
|
|
|
|
drop_mod = 0.3f;
|
|
|
|
}
|
|
|
|
|
2015-03-24 19:43:53 +00:00
|
|
|
if (font->block.active)
|
|
|
|
font->block.fullscreen = true;
|
|
|
|
else
|
|
|
|
setup_viewport(font, full_screen);
|
2015-01-12 22:38:39 +00:00
|
|
|
|
|
|
|
if (drop_x || drop_y)
|
|
|
|
{
|
|
|
|
color_dark[0] = color[0] * drop_mod;
|
|
|
|
color_dark[1] = color[1] * drop_mod;
|
|
|
|
color_dark[2] = color[2] * drop_mod;
|
|
|
|
color_dark[3] = color[3];
|
|
|
|
|
|
|
|
render_message(font, msg, scale, color_dark,
|
|
|
|
x + scale * drop_x / gl->vp.width, y +
|
2015-01-23 00:54:17 +00:00
|
|
|
scale * drop_y / gl->vp.height, align_right);
|
2015-01-12 22:38:39 +00:00
|
|
|
}
|
2015-03-24 18:00:46 +00:00
|
|
|
|
2015-01-23 00:54:17 +00:00
|
|
|
render_message(font, msg, scale, color, x, y, align_right);
|
2015-01-12 22:38:39 +00:00
|
|
|
|
2015-03-24 19:43:53 +00:00
|
|
|
if (!font->block.active)
|
|
|
|
restore_viewport(gl);
|
2015-01-12 22:38:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct font_glyph *gl_raster_font_get_glyph(
|
|
|
|
void *data, uint32_t code)
|
|
|
|
{
|
|
|
|
gl_raster_t *font = (gl_raster_t*)data;
|
|
|
|
|
|
|
|
if (!font)
|
|
|
|
return NULL;
|
|
|
|
return font->font_driver->get_glyph((void*)font->font_driver, code);
|
|
|
|
}
|
|
|
|
|
2015-03-24 19:43:53 +00:00
|
|
|
static void gl_end_block(void *data)
|
|
|
|
{
|
|
|
|
gl_raster_t *font = (gl_raster_t*)data;
|
|
|
|
gl_raster_block_t *block = &font->block;
|
|
|
|
gl_t *gl = font->gl;
|
|
|
|
|
|
|
|
if (block->coords.vertices)
|
|
|
|
{
|
|
|
|
setup_viewport(font, block->fullscreen);
|
|
|
|
|
|
|
|
draw_vertices(gl, &block->coords);
|
|
|
|
|
|
|
|
restore_viewport(gl);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (block->reuse && block->coords.vertices)
|
|
|
|
block->coords.vertices = 0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
block->active = false;
|
|
|
|
block->allocated = 0;
|
|
|
|
resize_block(font, 0);
|
|
|
|
memset(&block->coords, 0, sizeof(block->coords));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gl_begin_block(void *data)
|
|
|
|
{
|
|
|
|
gl_raster_t *font = (gl_raster_t*)data;
|
|
|
|
gl_raster_block_t *block = &font->block;
|
|
|
|
unsigned i = 0;
|
|
|
|
|
|
|
|
if (block->active)
|
|
|
|
{
|
|
|
|
block->reuse = true;
|
|
|
|
gl_end_block(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
block->reuse = false;
|
|
|
|
block->active = true;
|
|
|
|
block->coords.vertices = 0;
|
|
|
|
|
|
|
|
if (!block->allocated)
|
|
|
|
{
|
|
|
|
block->coords.color = block->coords.tex_coord = NULL;
|
|
|
|
block->coords.vertex = block->coords.lut_tex_coord = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-12 22:38:39 +00:00
|
|
|
gl_font_renderer_t gl_raster_font = {
|
|
|
|
gl_raster_font_init_font,
|
|
|
|
gl_raster_font_free_font,
|
|
|
|
gl_raster_font_render_msg,
|
|
|
|
"GL raster",
|
2015-03-24 19:43:53 +00:00
|
|
|
gl_raster_font_get_glyph,
|
|
|
|
gl_begin_block,
|
|
|
|
gl_end_block
|
2015-01-12 22:38:39 +00:00
|
|
|
};
|