1398 lines
38 KiB
C
1398 lines
38 KiB
C
/*
|
|
* This program 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 Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program 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 this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* The Original Code is Copyright (C) 2009 Blender Foundation.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
/** \file
|
|
* \ingroup blf
|
|
*
|
|
* Deals with drawing text to OpenGL or bitmap buffers.
|
|
*
|
|
* Also low level functions for managing \a FontBLF.
|
|
*/
|
|
|
|
#include <math.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <ft2build.h>
|
|
|
|
#include FT_FREETYPE_H
|
|
#include FT_GLYPH_H
|
|
#include FT_ADVANCES_H /* For FT_Get_Advance */
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "DNA_vec_types.h"
|
|
|
|
#include "BLI_listbase.h"
|
|
#include "BLI_math.h"
|
|
#include "BLI_math_color_blend.h"
|
|
#include "BLI_rect.h"
|
|
#include "BLI_string.h"
|
|
#include "BLI_string_utf8.h"
|
|
#include "BLI_threads.h"
|
|
|
|
#include "BLF_api.h"
|
|
|
|
#include "GPU_batch.h"
|
|
#include "GPU_matrix.h"
|
|
|
|
#include "blf_internal.h"
|
|
#include "blf_internal_types.h"
|
|
|
|
#include "BLI_strict_flags.h"
|
|
|
|
#ifdef WIN32
|
|
# define FT_New_Face FT_New_Face__win32_compat
|
|
#endif
|
|
|
|
/* Batching buffer for drawing. */
|
|
|
|
BatchBLF g_batch;
|
|
|
|
/* freetype2 handle ONLY for this file! */
|
|
static FT_Library ft_lib;
|
|
static SpinLock ft_lib_mutex;
|
|
static SpinLock blf_glyph_cache_mutex;
|
|
|
|
/* May be set to #UI_widgetbase_draw_cache_flush. */
|
|
static void (*blf_draw_cache_flush)(void) = NULL;
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name FreeType Utilities (Internal)
|
|
* \{ */
|
|
|
|
/**
|
|
* Convert a FreeType 26.6 value representing an unscaled design size to pixels.
|
|
* This is an exact copy of the scaling done inside FT_Get_Kerning when called
|
|
* with #FT_KERNING_DEFAULT, including arbitrary resizing for small fonts.
|
|
*/
|
|
static int blf_unscaled_F26Dot6_to_pixels(FontBLF *font, FT_Pos value)
|
|
{
|
|
/* Scale value by font size using integer-optimized multiplication. */
|
|
FT_Long scaled = FT_MulFix(value, font->face->size->metrics.x_scale);
|
|
|
|
/* FreeType states that this '25' has been determined heuristically. */
|
|
if (font->face->size->metrics.x_ppem < 25) {
|
|
scaled = FT_MulDiv(scaled, font->face->size->metrics.x_ppem, 25);
|
|
}
|
|
|
|
/* Copies of internal FreeType macros needed here. */
|
|
#define FT_PIX_FLOOR(x) ((x) & ~63)
|
|
#define FT_PIX_ROUND(x) FT_PIX_FLOOR((x) + 32)
|
|
|
|
/* Round to even 64ths, then divide by 64. */
|
|
return (int)FT_PIX_ROUND(scaled) >> 6;
|
|
|
|
#undef FT_PIX_FLOOR
|
|
#undef FT_PIX_ROUND
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Glyph Batching
|
|
* \{ */
|
|
|
|
/**
|
|
* Draw-calls are precious! make them count!
|
|
* Since most of the Text elements are not covered by other UI elements, we can
|
|
* group some strings together and render them in one draw-call. This behavior
|
|
* is on demand only, between #BLF_batch_draw_begin() and #BLF_batch_draw_end().
|
|
*/
|
|
static void blf_batch_draw_init(void)
|
|
{
|
|
GPUVertFormat format = {0};
|
|
g_batch.pos_loc = GPU_vertformat_attr_add(&format, "pos", GPU_COMP_F32, 4, GPU_FETCH_FLOAT);
|
|
g_batch.col_loc = GPU_vertformat_attr_add(
|
|
&format, "col", GPU_COMP_U8, 4, GPU_FETCH_INT_TO_FLOAT_UNIT);
|
|
g_batch.offset_loc = GPU_vertformat_attr_add(&format, "offset", GPU_COMP_I32, 1, GPU_FETCH_INT);
|
|
g_batch.glyph_size_loc = GPU_vertformat_attr_add(
|
|
&format, "glyph_size", GPU_COMP_I32, 2, GPU_FETCH_INT);
|
|
|
|
g_batch.verts = GPU_vertbuf_create_with_format_ex(&format, GPU_USAGE_STREAM);
|
|
GPU_vertbuf_data_alloc(g_batch.verts, BLF_BATCH_DRAW_LEN_MAX);
|
|
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.pos_loc, &g_batch.pos_step);
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.col_loc, &g_batch.col_step);
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.offset_loc, &g_batch.offset_step);
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.glyph_size_loc, &g_batch.glyph_size_step);
|
|
g_batch.glyph_len = 0;
|
|
|
|
/* A dummy VBO containing 4 points, attributes are not used. */
|
|
GPUVertBuf *vbo = GPU_vertbuf_create_with_format(&format);
|
|
GPU_vertbuf_data_alloc(vbo, 4);
|
|
|
|
/* We render a quad as a triangle strip and instance it for each glyph. */
|
|
g_batch.batch = GPU_batch_create_ex(GPU_PRIM_TRI_STRIP, vbo, NULL, GPU_BATCH_OWNS_VBO);
|
|
GPU_batch_instbuf_set(g_batch.batch, g_batch.verts, true);
|
|
}
|
|
|
|
static void blf_batch_draw_exit(void)
|
|
{
|
|
GPU_BATCH_DISCARD_SAFE(g_batch.batch);
|
|
}
|
|
|
|
void blf_batch_draw_begin(FontBLF *font)
|
|
{
|
|
if (g_batch.batch == NULL) {
|
|
blf_batch_draw_init();
|
|
}
|
|
|
|
const bool font_changed = (g_batch.font != font);
|
|
const bool simple_shader = ((font->flags & (BLF_ROTATION | BLF_MATRIX | BLF_ASPECT)) == 0);
|
|
const bool shader_changed = (simple_shader != g_batch.simple_shader);
|
|
|
|
g_batch.active = g_batch.enabled && simple_shader;
|
|
|
|
if (simple_shader) {
|
|
/* Offset is applied to each glyph. */
|
|
g_batch.ofs[0] = floorf(font->pos[0]);
|
|
g_batch.ofs[1] = floorf(font->pos[1]);
|
|
}
|
|
else {
|
|
/* Offset is baked in modelview mat. */
|
|
zero_v2(g_batch.ofs);
|
|
}
|
|
|
|
if (g_batch.active) {
|
|
float gpumat[4][4];
|
|
GPU_matrix_model_view_get(gpumat);
|
|
|
|
bool mat_changed = (memcmp(gpumat, g_batch.mat, sizeof(g_batch.mat)) != 0);
|
|
|
|
if (mat_changed) {
|
|
/* Modelviewmat is no longer the same.
|
|
* Flush cache but with the previous mat. */
|
|
GPU_matrix_push();
|
|
GPU_matrix_set(g_batch.mat);
|
|
}
|
|
|
|
/* flush cache if config is not the same. */
|
|
if (mat_changed || font_changed || shader_changed) {
|
|
blf_batch_draw();
|
|
g_batch.simple_shader = simple_shader;
|
|
g_batch.font = font;
|
|
}
|
|
else {
|
|
/* Nothing changed continue batching. */
|
|
return;
|
|
}
|
|
|
|
if (mat_changed) {
|
|
GPU_matrix_pop();
|
|
/* Save for next memcmp. */
|
|
memcpy(g_batch.mat, gpumat, sizeof(g_batch.mat));
|
|
}
|
|
}
|
|
else {
|
|
/* flush cache */
|
|
blf_batch_draw();
|
|
g_batch.font = font;
|
|
g_batch.simple_shader = simple_shader;
|
|
}
|
|
}
|
|
|
|
static GPUTexture *blf_batch_cache_texture_load(void)
|
|
{
|
|
GlyphCacheBLF *gc = g_batch.glyph_cache;
|
|
BLI_assert(gc);
|
|
BLI_assert(gc->bitmap_len > 0);
|
|
|
|
if (gc->bitmap_len > gc->bitmap_len_landed) {
|
|
const int tex_width = GPU_texture_width(gc->texture);
|
|
|
|
int bitmap_len_landed = gc->bitmap_len_landed;
|
|
int remain = gc->bitmap_len - bitmap_len_landed;
|
|
int offset_x = bitmap_len_landed % tex_width;
|
|
int offset_y = bitmap_len_landed / tex_width;
|
|
|
|
/* TODO(germano): Update more than one row in a single call. */
|
|
while (remain) {
|
|
int remain_row = tex_width - offset_x;
|
|
int width = remain > remain_row ? remain_row : remain;
|
|
GPU_texture_update_sub(gc->texture,
|
|
GPU_DATA_UBYTE,
|
|
&gc->bitmap_result[bitmap_len_landed],
|
|
offset_x,
|
|
offset_y,
|
|
0,
|
|
width,
|
|
1,
|
|
0);
|
|
|
|
bitmap_len_landed += width;
|
|
remain -= width;
|
|
offset_x = 0;
|
|
offset_y += 1;
|
|
}
|
|
|
|
gc->bitmap_len_landed = bitmap_len_landed;
|
|
}
|
|
|
|
return gc->texture;
|
|
}
|
|
|
|
void blf_batch_draw(void)
|
|
{
|
|
if (g_batch.glyph_len == 0) {
|
|
return;
|
|
}
|
|
|
|
GPU_blend(GPU_BLEND_ALPHA);
|
|
|
|
/* We need to flush widget base first to ensure correct ordering. */
|
|
if (blf_draw_cache_flush != NULL) {
|
|
blf_draw_cache_flush();
|
|
}
|
|
|
|
GPUTexture *texture = blf_batch_cache_texture_load();
|
|
GPU_vertbuf_data_len_set(g_batch.verts, g_batch.glyph_len);
|
|
GPU_vertbuf_use(g_batch.verts); /* send data */
|
|
|
|
GPU_batch_program_set_builtin(g_batch.batch, GPU_SHADER_TEXT);
|
|
GPU_batch_texture_bind(g_batch.batch, "glyph", texture);
|
|
GPU_batch_draw(g_batch.batch);
|
|
|
|
GPU_blend(GPU_BLEND_NONE);
|
|
|
|
GPU_texture_unbind(texture);
|
|
|
|
/* restart to 1st vertex data pointers */
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.pos_loc, &g_batch.pos_step);
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.col_loc, &g_batch.col_step);
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.offset_loc, &g_batch.offset_step);
|
|
GPU_vertbuf_attr_get_raw_data(g_batch.verts, g_batch.glyph_size_loc, &g_batch.glyph_size_step);
|
|
g_batch.glyph_len = 0;
|
|
}
|
|
|
|
static void blf_batch_draw_end(void)
|
|
{
|
|
if (!g_batch.active) {
|
|
blf_batch_draw();
|
|
}
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Glyph Stepping Utilities (Internal)
|
|
* \{ */
|
|
|
|
/* Fast path for runs of ASCII characters. Given that common UTF-8
|
|
* input will consist of an overwhelming majority of ASCII
|
|
* characters.
|
|
*/
|
|
|
|
BLI_INLINE GlyphBLF *blf_glyph_from_utf8_and_step(
|
|
FontBLF *font, GlyphCacheBLF *gc, const char *str, size_t str_len, size_t *i_p)
|
|
{
|
|
uint charcode = BLI_str_utf8_as_unicode_step(str, str_len, i_p);
|
|
/* Invalid unicode sequences return the byte value, stepping forward one.
|
|
* This allows `latin1` to display (which is sometimes used for file-paths). */
|
|
BLI_assert(charcode != BLI_UTF8_ERR);
|
|
return blf_glyph_ensure(font, gc, charcode);
|
|
}
|
|
|
|
BLI_INLINE int blf_kerning(FontBLF *font, const GlyphBLF *g_prev, const GlyphBLF *g)
|
|
{
|
|
if (!FT_HAS_KERNING(font->face) || g_prev == NULL) {
|
|
return 0;
|
|
}
|
|
|
|
FT_Vector delta = {KERNING_ENTRY_UNSET};
|
|
|
|
/* Get unscaled kerning value from our cache if ASCII. */
|
|
if ((g_prev->c < KERNING_CACHE_TABLE_SIZE) && (g->c < GLYPH_ASCII_TABLE_SIZE)) {
|
|
delta.x = font->kerning_cache->ascii_table[g->c][g_prev->c];
|
|
}
|
|
|
|
/* If not ASCII or not found in cache, ask FreeType for kerning. */
|
|
if (UNLIKELY(delta.x == KERNING_ENTRY_UNSET)) {
|
|
/* Note that this function sets delta values to zero on any error. */
|
|
FT_Get_Kerning(font->face, g_prev->idx, g->idx, FT_KERNING_UNSCALED, &delta);
|
|
}
|
|
|
|
/* If ASCII we save this value to our cache for quicker access next time. */
|
|
if ((g_prev->c < KERNING_CACHE_TABLE_SIZE) && (g->c < GLYPH_ASCII_TABLE_SIZE)) {
|
|
font->kerning_cache->ascii_table[g->c][g_prev->c] = (int)delta.x;
|
|
}
|
|
|
|
if (delta.x != 0) {
|
|
/* Convert unscaled design units to pixels and move pen. */
|
|
return blf_unscaled_F26Dot6_to_pixels(font, delta.x);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Text Drawing: GPU
|
|
* \{ */
|
|
|
|
static void blf_font_draw_ex(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info,
|
|
int pen_y)
|
|
{
|
|
GlyphBLF *g, *g_prev = NULL;
|
|
int pen_x = 0;
|
|
size_t i = 0;
|
|
|
|
if (str_len == 0) {
|
|
/* early output, don't do any IMM OpenGL. */
|
|
return;
|
|
}
|
|
|
|
blf_batch_draw_begin(font);
|
|
|
|
while ((i < str_len) && str[i]) {
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (UNLIKELY(g == NULL)) {
|
|
continue;
|
|
}
|
|
pen_x += blf_kerning(font, g_prev, g);
|
|
|
|
/* do not return this loop if clipped, we want every character tested */
|
|
blf_glyph_draw(font, gc, g, (float)pen_x, (float)pen_y);
|
|
|
|
pen_x += g->advance_i;
|
|
g_prev = g;
|
|
}
|
|
|
|
blf_batch_draw_end();
|
|
|
|
if (r_info) {
|
|
r_info->lines = 1;
|
|
r_info->width = pen_x;
|
|
}
|
|
}
|
|
void blf_font_draw(FontBLF *font, const char *str, const size_t str_len, struct ResultBLF *r_info)
|
|
{
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
blf_font_draw_ex(font, gc, str, str_len, r_info, 0);
|
|
blf_glyph_cache_release(font);
|
|
}
|
|
|
|
int blf_font_draw_mono(FontBLF *font, const char *str, const size_t str_len, int cwidth)
|
|
{
|
|
GlyphBLF *g;
|
|
int col, columns = 0;
|
|
int pen_x = 0, pen_y = 0;
|
|
size_t i = 0;
|
|
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
|
|
blf_batch_draw_begin(font);
|
|
|
|
while ((i < str_len) && str[i]) {
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (UNLIKELY(g == NULL)) {
|
|
continue;
|
|
}
|
|
/* do not return this loop if clipped, we want every character tested */
|
|
blf_glyph_draw(font, gc, g, (float)pen_x, (float)pen_y);
|
|
|
|
col = BLI_wcwidth((char32_t)g->c);
|
|
if (col < 0) {
|
|
col = 1;
|
|
}
|
|
|
|
columns += col;
|
|
pen_x += cwidth * col;
|
|
}
|
|
|
|
blf_batch_draw_end();
|
|
|
|
blf_glyph_cache_release(font);
|
|
return columns;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Text Drawing: Buffer
|
|
* \{ */
|
|
|
|
/* Sanity checks are done by BLF_draw_buffer() */
|
|
static void blf_font_draw_buffer_ex(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info,
|
|
int pen_y)
|
|
{
|
|
GlyphBLF *g, *g_prev = NULL;
|
|
int pen_x = (int)font->pos[0];
|
|
int pen_y_basis = (int)font->pos[1] + pen_y;
|
|
size_t i = 0;
|
|
|
|
/* buffer specific vars */
|
|
FontBufInfoBLF *buf_info = &font->buf_info;
|
|
const float *b_col_float = buf_info->col_float;
|
|
const unsigned char *b_col_char = buf_info->col_char;
|
|
int chx, chy;
|
|
int y, x;
|
|
|
|
/* another buffer specific call for color conversion */
|
|
|
|
while ((i < str_len) && str[i]) {
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (UNLIKELY(g == NULL)) {
|
|
continue;
|
|
}
|
|
pen_x += blf_kerning(font, g_prev, g);
|
|
|
|
chx = pen_x + ((int)g->pos[0]);
|
|
chy = pen_y_basis + g->dims[1];
|
|
|
|
if (g->pitch < 0) {
|
|
pen_y = pen_y_basis + (g->dims[1] - g->pos[1]);
|
|
}
|
|
else {
|
|
pen_y = pen_y_basis - (g->dims[1] - g->pos[1]);
|
|
}
|
|
|
|
if ((chx + g->dims[0]) >= 0 && chx < buf_info->dims[0] && (pen_y + g->dims[1]) >= 0 &&
|
|
pen_y < buf_info->dims[1]) {
|
|
/* don't draw beyond the buffer bounds */
|
|
int width_clip = g->dims[0];
|
|
int height_clip = g->dims[1];
|
|
int yb_start = g->pitch < 0 ? 0 : g->dims[1] - 1;
|
|
|
|
if (width_clip + chx > buf_info->dims[0]) {
|
|
width_clip -= chx + width_clip - buf_info->dims[0];
|
|
}
|
|
if (height_clip + pen_y > buf_info->dims[1]) {
|
|
height_clip -= pen_y + height_clip - buf_info->dims[1];
|
|
}
|
|
|
|
/* drawing below the image? */
|
|
if (pen_y < 0) {
|
|
yb_start += (g->pitch < 0) ? -pen_y : pen_y;
|
|
height_clip += pen_y;
|
|
pen_y = 0;
|
|
}
|
|
|
|
if (buf_info->fbuf) {
|
|
int yb = yb_start;
|
|
for (y = ((chy >= 0) ? 0 : -chy); y < height_clip; y++) {
|
|
for (x = ((chx >= 0) ? 0 : -chx); x < width_clip; x++) {
|
|
const char a_byte = *(g->bitmap + x + (yb * g->pitch));
|
|
if (a_byte) {
|
|
const float a = (a_byte / 255.0f) * b_col_float[3];
|
|
const size_t buf_ofs = (((size_t)(chx + x) +
|
|
((size_t)(pen_y + y) * (size_t)buf_info->dims[0])) *
|
|
(size_t)buf_info->ch);
|
|
float *fbuf = buf_info->fbuf + buf_ofs;
|
|
|
|
float font_pixel[4];
|
|
font_pixel[0] = b_col_float[0] * a;
|
|
font_pixel[1] = b_col_float[1] * a;
|
|
font_pixel[2] = b_col_float[2] * a;
|
|
font_pixel[3] = a;
|
|
blend_color_mix_float(fbuf, fbuf, font_pixel);
|
|
}
|
|
}
|
|
|
|
if (g->pitch < 0) {
|
|
yb++;
|
|
}
|
|
else {
|
|
yb--;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (buf_info->cbuf) {
|
|
int yb = yb_start;
|
|
for (y = ((chy >= 0) ? 0 : -chy); y < height_clip; y++) {
|
|
for (x = ((chx >= 0) ? 0 : -chx); x < width_clip; x++) {
|
|
const char a_byte = *(g->bitmap + x + (yb * g->pitch));
|
|
|
|
if (a_byte) {
|
|
const float a = (a_byte / 255.0f) * b_col_float[3];
|
|
const size_t buf_ofs = (((size_t)(chx + x) +
|
|
((size_t)(pen_y + y) * (size_t)buf_info->dims[0])) *
|
|
(size_t)buf_info->ch);
|
|
unsigned char *cbuf = buf_info->cbuf + buf_ofs;
|
|
|
|
uchar font_pixel[4];
|
|
font_pixel[0] = b_col_char[0];
|
|
font_pixel[1] = b_col_char[1];
|
|
font_pixel[2] = b_col_char[2];
|
|
font_pixel[3] = unit_float_to_uchar_clamp(a);
|
|
blend_color_mix_byte(cbuf, cbuf, font_pixel);
|
|
}
|
|
}
|
|
|
|
if (g->pitch < 0) {
|
|
yb++;
|
|
}
|
|
else {
|
|
yb--;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pen_x += g->advance_i;
|
|
g_prev = g;
|
|
}
|
|
|
|
if (r_info) {
|
|
r_info->lines = 1;
|
|
r_info->width = pen_x;
|
|
}
|
|
}
|
|
|
|
void blf_font_draw_buffer(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
blf_font_draw_buffer_ex(font, gc, str, str_len, r_info, 0);
|
|
blf_glyph_cache_release(font);
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Text Evaluation: Width to String Length
|
|
*
|
|
* Use to implement exported functions:
|
|
* - #BLF_width_to_strlen
|
|
* - #BLF_width_to_rstrlen
|
|
* \{ */
|
|
|
|
static bool blf_font_width_to_strlen_glyph_process(
|
|
FontBLF *font, GlyphBLF *g_prev, GlyphBLF *g, int *pen_x, const int width_i)
|
|
{
|
|
if (UNLIKELY(g == NULL)) {
|
|
return false; /* continue the calling loop. */
|
|
}
|
|
*pen_x += blf_kerning(font, g_prev, g);
|
|
*pen_x += g->advance_i;
|
|
|
|
/* When true, break the calling loop. */
|
|
return (*pen_x >= width_i);
|
|
}
|
|
|
|
size_t blf_font_width_to_strlen(
|
|
FontBLF *font, const char *str, const size_t str_len, float width, float *r_width)
|
|
{
|
|
GlyphBLF *g, *g_prev;
|
|
int pen_x, width_new;
|
|
size_t i, i_prev;
|
|
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
const int width_i = (int)width;
|
|
|
|
for (i_prev = i = 0, width_new = pen_x = 0, g_prev = NULL; (i < str_len) && str[i];
|
|
i_prev = i, width_new = pen_x, g_prev = g) {
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (blf_font_width_to_strlen_glyph_process(font, g_prev, g, &pen_x, width_i)) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (r_width) {
|
|
*r_width = (float)width_new;
|
|
}
|
|
|
|
blf_glyph_cache_release(font);
|
|
return i_prev;
|
|
}
|
|
|
|
size_t blf_font_width_to_rstrlen(
|
|
FontBLF *font, const char *str, const size_t str_len, float width, float *r_width)
|
|
{
|
|
GlyphBLF *g, *g_prev;
|
|
int pen_x, width_new;
|
|
size_t i, i_prev, i_tmp;
|
|
const char *s, *s_prev;
|
|
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
const int width_i = (int)width;
|
|
|
|
i = BLI_strnlen(str, str_len);
|
|
s = BLI_str_find_prev_char_utf8(&str[i], str);
|
|
i = (size_t)(s - str);
|
|
s_prev = BLI_str_find_prev_char_utf8(s, str);
|
|
i_prev = (size_t)(s_prev - str);
|
|
|
|
i_tmp = i;
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i_tmp);
|
|
for (width_new = pen_x = 0; (s != NULL);
|
|
i = i_prev, s = s_prev, g = g_prev, g_prev = NULL, width_new = pen_x) {
|
|
s_prev = BLI_str_find_prev_char_utf8(s, str);
|
|
i_prev = (size_t)(s_prev - str);
|
|
|
|
if (s_prev != NULL) {
|
|
i_tmp = i_prev;
|
|
g_prev = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i_tmp);
|
|
BLI_assert(i_tmp == i);
|
|
}
|
|
|
|
if (blf_font_width_to_strlen_glyph_process(font, g_prev, g, &pen_x, width_i)) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (r_width) {
|
|
*r_width = (float)width_new;
|
|
}
|
|
|
|
blf_glyph_cache_release(font);
|
|
return i;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Text Evaluation: Glyph Bound Box with Callback
|
|
* \{ */
|
|
|
|
static void blf_font_boundbox_ex(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
rctf *box,
|
|
struct ResultBLF *r_info,
|
|
int pen_y)
|
|
{
|
|
GlyphBLF *g, *g_prev = NULL;
|
|
int pen_x = 0;
|
|
size_t i = 0;
|
|
rctf gbox;
|
|
|
|
box->xmin = 32000.0f;
|
|
box->xmax = -32000.0f;
|
|
box->ymin = 32000.0f;
|
|
box->ymax = -32000.0f;
|
|
|
|
while ((i < str_len) && str[i]) {
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (UNLIKELY(g == NULL)) {
|
|
continue;
|
|
}
|
|
pen_x += blf_kerning(font, g_prev, g);
|
|
|
|
gbox.xmin = (float)pen_x;
|
|
gbox.xmax = (float)pen_x + g->advance;
|
|
gbox.ymin = g->box.ymin + (float)pen_y;
|
|
gbox.ymax = g->box.ymax + (float)pen_y;
|
|
|
|
if (gbox.xmin < box->xmin) {
|
|
box->xmin = gbox.xmin;
|
|
}
|
|
if (gbox.ymin < box->ymin) {
|
|
box->ymin = gbox.ymin;
|
|
}
|
|
|
|
if (gbox.xmax > box->xmax) {
|
|
box->xmax = gbox.xmax;
|
|
}
|
|
if (gbox.ymax > box->ymax) {
|
|
box->ymax = gbox.ymax;
|
|
}
|
|
|
|
pen_x += g->advance_i;
|
|
g_prev = g;
|
|
}
|
|
|
|
if (box->xmin > box->xmax) {
|
|
box->xmin = 0.0f;
|
|
box->ymin = 0.0f;
|
|
box->xmax = 0.0f;
|
|
box->ymax = 0.0f;
|
|
}
|
|
|
|
if (r_info) {
|
|
r_info->lines = 1;
|
|
r_info->width = pen_x;
|
|
}
|
|
}
|
|
void blf_font_boundbox(
|
|
FontBLF *font, const char *str, const size_t str_len, rctf *r_box, struct ResultBLF *r_info)
|
|
{
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
blf_font_boundbox_ex(font, gc, str, str_len, r_box, r_info, 0);
|
|
blf_glyph_cache_release(font);
|
|
}
|
|
|
|
void blf_font_width_and_height(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
float *r_width,
|
|
float *r_height,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
float xa, ya;
|
|
rctf box;
|
|
|
|
if (font->flags & BLF_ASPECT) {
|
|
xa = font->aspect[0];
|
|
ya = font->aspect[1];
|
|
}
|
|
else {
|
|
xa = 1.0f;
|
|
ya = 1.0f;
|
|
}
|
|
|
|
if (font->flags & BLF_WORD_WRAP) {
|
|
blf_font_boundbox__wrap(font, str, str_len, &box, r_info);
|
|
}
|
|
else {
|
|
blf_font_boundbox(font, str, str_len, &box, r_info);
|
|
}
|
|
*r_width = (BLI_rctf_size_x(&box) * xa);
|
|
*r_height = (BLI_rctf_size_y(&box) * ya);
|
|
}
|
|
|
|
float blf_font_width(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
float xa;
|
|
rctf box;
|
|
|
|
if (font->flags & BLF_ASPECT) {
|
|
xa = font->aspect[0];
|
|
}
|
|
else {
|
|
xa = 1.0f;
|
|
}
|
|
|
|
if (font->flags & BLF_WORD_WRAP) {
|
|
blf_font_boundbox__wrap(font, str, str_len, &box, r_info);
|
|
}
|
|
else {
|
|
blf_font_boundbox(font, str, str_len, &box, r_info);
|
|
}
|
|
return BLI_rctf_size_x(&box) * xa;
|
|
}
|
|
|
|
float blf_font_height(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
float ya;
|
|
rctf box;
|
|
|
|
if (font->flags & BLF_ASPECT) {
|
|
ya = font->aspect[1];
|
|
}
|
|
else {
|
|
ya = 1.0f;
|
|
}
|
|
|
|
if (font->flags & BLF_WORD_WRAP) {
|
|
blf_font_boundbox__wrap(font, str, str_len, &box, r_info);
|
|
}
|
|
else {
|
|
blf_font_boundbox(font, str, str_len, &box, r_info);
|
|
}
|
|
return BLI_rctf_size_y(&box) * ya;
|
|
}
|
|
|
|
float blf_font_fixed_width(FontBLF *font)
|
|
{
|
|
return (float)font->fixed_width;
|
|
}
|
|
|
|
static void blf_font_boundbox_foreach_glyph_ex(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
BLF_GlyphBoundsFn user_fn,
|
|
void *user_data,
|
|
struct ResultBLF *r_info,
|
|
int pen_y)
|
|
{
|
|
GlyphBLF *g, *g_prev = NULL;
|
|
int pen_x = 0;
|
|
size_t i = 0, i_curr;
|
|
rcti gbox;
|
|
|
|
if (str_len == 0) {
|
|
/* early output. */
|
|
return;
|
|
}
|
|
|
|
while ((i < str_len) && str[i]) {
|
|
i_curr = i;
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (UNLIKELY(g == NULL)) {
|
|
continue;
|
|
}
|
|
pen_x += blf_kerning(font, g_prev, g);
|
|
|
|
gbox.xmin = pen_x;
|
|
gbox.xmax = gbox.xmin + MIN2(g->advance_i, g->dims[0]);
|
|
gbox.ymin = pen_y;
|
|
gbox.ymax = gbox.ymin - g->dims[1];
|
|
|
|
pen_x += g->advance_i;
|
|
|
|
if (user_fn(str, i_curr, &gbox, g->advance_i, &g->box, g->pos, user_data) == false) {
|
|
break;
|
|
}
|
|
|
|
g_prev = g;
|
|
}
|
|
|
|
if (r_info) {
|
|
r_info->lines = 1;
|
|
r_info->width = pen_x;
|
|
}
|
|
}
|
|
void blf_font_boundbox_foreach_glyph(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
BLF_GlyphBoundsFn user_fn,
|
|
void *user_data,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
blf_font_boundbox_foreach_glyph_ex(font, gc, str, str_len, user_fn, user_data, r_info, 0);
|
|
blf_glyph_cache_release(font);
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Text Evaluation: Word-Wrap with Callback
|
|
* \{ */
|
|
|
|
/**
|
|
* Generic function to add word-wrap support for other existing functions.
|
|
*
|
|
* Wraps on spaces and respects newlines.
|
|
* Intentionally ignores non-unix newlines, tabs and more advanced text formatting.
|
|
*
|
|
* \note If we want rich text - we better have a higher level API to handle that
|
|
* (color, bold, switching fonts... etc).
|
|
*/
|
|
static void blf_font_wrap_apply(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info,
|
|
void (*callback)(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
int pen_y,
|
|
void *userdata),
|
|
void *userdata)
|
|
{
|
|
GlyphBLF *g, *g_prev = NULL;
|
|
int pen_x = 0, pen_y = 0;
|
|
size_t i = 0;
|
|
int lines = 0;
|
|
int pen_x_next = 0;
|
|
|
|
GlyphCacheBLF *gc = blf_glyph_cache_acquire(font);
|
|
|
|
struct WordWrapVars {
|
|
int wrap_width;
|
|
size_t start, last[2];
|
|
} wrap = {font->wrap_width != -1 ? font->wrap_width : INT_MAX, 0, {0, 0}};
|
|
|
|
// printf("%s wrapping (%d, %d) `%s`:\n", __func__, str_len, strlen(str), str);
|
|
while ((i < str_len) && str[i]) {
|
|
|
|
/* wrap vars */
|
|
size_t i_curr = i;
|
|
bool do_draw = false;
|
|
|
|
g = blf_glyph_from_utf8_and_step(font, gc, str, str_len, &i);
|
|
|
|
if (UNLIKELY(g == NULL)) {
|
|
continue;
|
|
}
|
|
pen_x += blf_kerning(font, g_prev, g);
|
|
|
|
/**
|
|
* Implementation Detail (utf8).
|
|
*
|
|
* Take care with single byte offsets here,
|
|
* since this is utf8 we can't be sure a single byte is a single character.
|
|
*
|
|
* This is _only_ done when we know for sure the character is ascii (newline or a space).
|
|
*/
|
|
pen_x_next = pen_x + g->advance_i;
|
|
if (UNLIKELY((pen_x_next >= wrap.wrap_width) && (wrap.start != wrap.last[0]))) {
|
|
do_draw = true;
|
|
}
|
|
else if (UNLIKELY(((i < str_len) && str[i]) == 0)) {
|
|
/* need check here for trailing newline, else we draw it */
|
|
wrap.last[0] = i + ((g->c != '\n') ? 1 : 0);
|
|
wrap.last[1] = i;
|
|
do_draw = true;
|
|
}
|
|
else if (UNLIKELY(g->c == '\n')) {
|
|
wrap.last[0] = i_curr + 1;
|
|
wrap.last[1] = i;
|
|
do_draw = true;
|
|
}
|
|
else if (UNLIKELY(g->c != ' ' && (g_prev ? g_prev->c == ' ' : false))) {
|
|
wrap.last[0] = i_curr;
|
|
wrap.last[1] = i_curr;
|
|
}
|
|
|
|
if (UNLIKELY(do_draw)) {
|
|
// printf("(%03d..%03d) `%.*s`\n",
|
|
// wrap.start, wrap.last[0], (wrap.last[0] - wrap.start) - 1, &str[wrap.start]);
|
|
|
|
callback(font, gc, &str[wrap.start], (wrap.last[0] - wrap.start) - 1, pen_y, userdata);
|
|
wrap.start = wrap.last[0];
|
|
i = wrap.last[1];
|
|
pen_x = 0;
|
|
pen_y -= blf_font_height_max(font);
|
|
g_prev = NULL;
|
|
lines += 1;
|
|
continue;
|
|
}
|
|
|
|
pen_x = pen_x_next;
|
|
g_prev = g;
|
|
}
|
|
|
|
// printf("done! lines: %d, width, %d\n", lines, pen_x_next);
|
|
|
|
if (r_info) {
|
|
r_info->lines = lines;
|
|
/* width of last line only (with wrapped lines) */
|
|
r_info->width = pen_x_next;
|
|
}
|
|
|
|
blf_glyph_cache_release(font);
|
|
}
|
|
|
|
/* blf_font_draw__wrap */
|
|
static void blf_font_draw__wrap_cb(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
int pen_y,
|
|
void *UNUSED(userdata))
|
|
{
|
|
blf_font_draw_ex(font, gc, str, str_len, NULL, pen_y);
|
|
}
|
|
void blf_font_draw__wrap(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
blf_font_wrap_apply(font, str, str_len, r_info, blf_font_draw__wrap_cb, NULL);
|
|
}
|
|
|
|
/* blf_font_boundbox__wrap */
|
|
static void blf_font_boundbox_wrap_cb(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
int pen_y,
|
|
void *userdata)
|
|
{
|
|
rctf *box = userdata;
|
|
rctf box_single;
|
|
|
|
blf_font_boundbox_ex(font, gc, str, str_len, &box_single, NULL, pen_y);
|
|
BLI_rctf_union(box, &box_single);
|
|
}
|
|
void blf_font_boundbox__wrap(
|
|
FontBLF *font, const char *str, const size_t str_len, rctf *box, struct ResultBLF *r_info)
|
|
{
|
|
box->xmin = 32000.0f;
|
|
box->xmax = -32000.0f;
|
|
box->ymin = 32000.0f;
|
|
box->ymax = -32000.0f;
|
|
|
|
blf_font_wrap_apply(font, str, str_len, r_info, blf_font_boundbox_wrap_cb, box);
|
|
}
|
|
|
|
/* blf_font_draw_buffer__wrap */
|
|
static void blf_font_draw_buffer__wrap_cb(FontBLF *font,
|
|
GlyphCacheBLF *gc,
|
|
const char *str,
|
|
const size_t str_len,
|
|
int pen_y,
|
|
void *UNUSED(userdata))
|
|
{
|
|
blf_font_draw_buffer_ex(font, gc, str, str_len, NULL, pen_y);
|
|
}
|
|
void blf_font_draw_buffer__wrap(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
struct ResultBLF *r_info)
|
|
{
|
|
blf_font_wrap_apply(font, str, str_len, r_info, blf_font_draw_buffer__wrap_cb, NULL);
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Text Evaluation: Count Missing Characters
|
|
* \{ */
|
|
|
|
int blf_font_count_missing_chars(FontBLF *font,
|
|
const char *str,
|
|
const size_t str_len,
|
|
int *r_tot_chars)
|
|
{
|
|
int missing = 0;
|
|
size_t i = 0;
|
|
|
|
*r_tot_chars = 0;
|
|
while (i < str_len) {
|
|
unsigned int c;
|
|
|
|
if ((c = str[i]) < GLYPH_ASCII_TABLE_SIZE) {
|
|
i++;
|
|
}
|
|
else {
|
|
c = BLI_str_utf8_as_unicode_step(str, str_len, &i);
|
|
if (FT_Get_Char_Index((font)->face, c) == 0) {
|
|
missing++;
|
|
}
|
|
}
|
|
(*r_tot_chars)++;
|
|
}
|
|
return missing;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Font Query: Attributes
|
|
* \{ */
|
|
|
|
int blf_font_height_max(FontBLF *font)
|
|
{
|
|
int height_max;
|
|
if (FT_IS_SCALABLE(font->face)) {
|
|
height_max = (int)((float)(font->face->ascender - font->face->descender) *
|
|
(((float)font->face->size->metrics.y_ppem) /
|
|
((float)font->face->units_per_EM)));
|
|
}
|
|
else {
|
|
height_max = (int)(((float)font->face->size->metrics.height) / 64.0f);
|
|
}
|
|
/* can happen with size 1 fonts */
|
|
return MAX2(height_max, 1);
|
|
}
|
|
|
|
int blf_font_width_max(FontBLF *font)
|
|
{
|
|
int width_max;
|
|
if (FT_IS_SCALABLE(font->face)) {
|
|
width_max = (int)((float)(font->face->bbox.xMax - font->face->bbox.xMin) *
|
|
(((float)font->face->size->metrics.x_ppem) /
|
|
((float)font->face->units_per_EM)));
|
|
}
|
|
else {
|
|
width_max = (int)(((float)font->face->size->metrics.max_advance) / 64.0f);
|
|
}
|
|
/* can happen with size 1 fonts */
|
|
return MAX2(width_max, 1);
|
|
}
|
|
|
|
float blf_font_descender(FontBLF *font)
|
|
{
|
|
return ((float)font->face->size->metrics.descender) / 64.0f;
|
|
}
|
|
|
|
float blf_font_ascender(FontBLF *font)
|
|
{
|
|
return ((float)font->face->size->metrics.ascender) / 64.0f;
|
|
}
|
|
|
|
char *blf_display_name(FontBLF *font)
|
|
{
|
|
if (!font->face->family_name) {
|
|
return NULL;
|
|
}
|
|
return BLI_sprintfN("%s %s", font->face->family_name, font->face->style_name);
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Font Subsystem Init/Exit
|
|
* \{ */
|
|
|
|
int blf_font_init(void)
|
|
{
|
|
memset(&g_batch, 0, sizeof(g_batch));
|
|
BLI_spin_init(&ft_lib_mutex);
|
|
BLI_spin_init(&blf_glyph_cache_mutex);
|
|
return FT_Init_FreeType(&ft_lib);
|
|
}
|
|
|
|
void blf_font_exit(void)
|
|
{
|
|
FT_Done_FreeType(ft_lib);
|
|
BLI_spin_end(&ft_lib_mutex);
|
|
BLI_spin_end(&blf_glyph_cache_mutex);
|
|
blf_batch_draw_exit();
|
|
}
|
|
|
|
void BLF_cache_flush_set_fn(void (*cache_flush_fn)(void))
|
|
{
|
|
blf_draw_cache_flush = cache_flush_fn;
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Font New/Free
|
|
* \{ */
|
|
|
|
static void blf_font_fill(FontBLF *font)
|
|
{
|
|
font->aspect[0] = 1.0f;
|
|
font->aspect[1] = 1.0f;
|
|
font->aspect[2] = 1.0f;
|
|
font->pos[0] = 0.0f;
|
|
font->pos[1] = 0.0f;
|
|
font->angle = 0.0f;
|
|
|
|
for (int i = 0; i < 16; i++) {
|
|
font->m[i] = 0;
|
|
}
|
|
|
|
/* annoying bright color so we can see where to add BLF_color calls */
|
|
font->color[0] = 255;
|
|
font->color[1] = 255;
|
|
font->color[2] = 0;
|
|
font->color[3] = 255;
|
|
|
|
font->clip_rec.xmin = 0.0f;
|
|
font->clip_rec.xmax = 0.0f;
|
|
font->clip_rec.ymin = 0.0f;
|
|
font->clip_rec.ymax = 0.0f;
|
|
font->flags = 0;
|
|
font->dpi = 0;
|
|
font->size = 0;
|
|
BLI_listbase_clear(&font->cache);
|
|
font->kerning_cache = NULL;
|
|
#if BLF_BLUR_ENABLE
|
|
font->blur = 0;
|
|
#endif
|
|
font->tex_size_max = -1;
|
|
|
|
font->buf_info.fbuf = NULL;
|
|
font->buf_info.cbuf = NULL;
|
|
font->buf_info.dims[0] = 0;
|
|
font->buf_info.dims[1] = 0;
|
|
font->buf_info.ch = 0;
|
|
font->buf_info.col_init[0] = 0;
|
|
font->buf_info.col_init[1] = 0;
|
|
font->buf_info.col_init[2] = 0;
|
|
font->buf_info.col_init[3] = 0;
|
|
|
|
font->ft_lib = ft_lib;
|
|
font->ft_lib_mutex = &ft_lib_mutex;
|
|
font->glyph_cache_mutex = &blf_glyph_cache_mutex;
|
|
}
|
|
|
|
FontBLF *blf_font_new(const char *name, const char *filename)
|
|
{
|
|
FontBLF *font;
|
|
FT_Error err;
|
|
char *mfile;
|
|
|
|
font = (FontBLF *)MEM_callocN(sizeof(FontBLF), "blf_font_new");
|
|
err = FT_New_Face(ft_lib, filename, 0, &font->face);
|
|
if (err) {
|
|
MEM_freeN(font);
|
|
return NULL;
|
|
}
|
|
|
|
err = FT_Select_Charmap(font->face, FT_ENCODING_UNICODE);
|
|
if (err) {
|
|
err = FT_Select_Charmap(font->face, FT_ENCODING_APPLE_ROMAN);
|
|
}
|
|
if (err && font->face->num_charmaps > 0) {
|
|
err = FT_Select_Charmap(font->face, font->face->charmaps[0]->encoding);
|
|
}
|
|
if (err) {
|
|
printf("Can't set a character map!\n");
|
|
FT_Done_Face(font->face);
|
|
MEM_freeN(font);
|
|
return NULL;
|
|
}
|
|
|
|
mfile = blf_dir_metrics_search(filename);
|
|
if (mfile) {
|
|
err = FT_Attach_File(font->face, mfile);
|
|
if (err) {
|
|
fprintf(stderr, "FT_Attach_File failed to load '%s' with error %d\n", filename, (int)err);
|
|
}
|
|
MEM_freeN(mfile);
|
|
}
|
|
|
|
font->name = BLI_strdup(name);
|
|
font->filename = BLI_strdup(filename);
|
|
blf_font_fill(font);
|
|
|
|
if (FT_HAS_KERNING(font->face)) {
|
|
/* Create kerning cache table and fill with value indicating "unset". */
|
|
font->kerning_cache = MEM_mallocN(sizeof(KerningCacheBLF), __func__);
|
|
for (uint i = 0; i < KERNING_CACHE_TABLE_SIZE; i++) {
|
|
for (uint j = 0; j < KERNING_CACHE_TABLE_SIZE; j++) {
|
|
font->kerning_cache->ascii_table[i][j] = KERNING_ENTRY_UNSET;
|
|
}
|
|
}
|
|
}
|
|
|
|
return font;
|
|
}
|
|
|
|
void blf_font_attach_from_mem(FontBLF *font, const unsigned char *mem, int mem_size)
|
|
{
|
|
FT_Open_Args open;
|
|
|
|
open.flags = FT_OPEN_MEMORY;
|
|
open.memory_base = (const FT_Byte *)mem;
|
|
open.memory_size = mem_size;
|
|
FT_Attach_Stream(font->face, &open);
|
|
}
|
|
|
|
FontBLF *blf_font_new_from_mem(const char *name, const unsigned char *mem, int mem_size)
|
|
{
|
|
FontBLF *font;
|
|
FT_Error err;
|
|
|
|
font = (FontBLF *)MEM_callocN(sizeof(FontBLF), "blf_font_new_from_mem");
|
|
err = FT_New_Memory_Face(ft_lib, mem, mem_size, 0, &font->face);
|
|
if (err) {
|
|
MEM_freeN(font);
|
|
return NULL;
|
|
}
|
|
|
|
err = FT_Select_Charmap(font->face, ft_encoding_unicode);
|
|
if (err) {
|
|
printf("Can't set the unicode character map!\n");
|
|
FT_Done_Face(font->face);
|
|
MEM_freeN(font);
|
|
return NULL;
|
|
}
|
|
|
|
font->name = BLI_strdup(name);
|
|
font->filename = NULL;
|
|
blf_font_fill(font);
|
|
return font;
|
|
}
|
|
|
|
void blf_font_free(FontBLF *font)
|
|
{
|
|
BLI_spin_lock(&blf_glyph_cache_mutex);
|
|
GlyphCacheBLF *gc;
|
|
|
|
while ((gc = BLI_pophead(&font->cache))) {
|
|
blf_glyph_cache_free(gc);
|
|
}
|
|
|
|
if (font->kerning_cache) {
|
|
MEM_freeN(font->kerning_cache);
|
|
}
|
|
|
|
FT_Done_Face(font->face);
|
|
if (font->filename) {
|
|
MEM_freeN(font->filename);
|
|
}
|
|
if (font->name) {
|
|
MEM_freeN(font->name);
|
|
}
|
|
MEM_freeN(font);
|
|
|
|
BLI_spin_unlock(&blf_glyph_cache_mutex);
|
|
}
|
|
|
|
/** \} */
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
/** \name Font Configure
|
|
* \{ */
|
|
|
|
void blf_font_size(FontBLF *font, float size, unsigned int dpi)
|
|
{
|
|
blf_glyph_cache_acquire(font);
|
|
|
|
/* FreeType uses fixed-point integers in 64ths. */
|
|
FT_F26Dot6 ft_size = lroundf(size * 64.0f);
|
|
/* Adjust our size to be on even 64ths. */
|
|
size = (float)ft_size / 64.0f;
|
|
|
|
GlyphCacheBLF *gc = blf_glyph_cache_find(font, size, dpi);
|
|
if (gc && (font->size == size && font->dpi == dpi)) {
|
|
/* Optimization: do not call FT_Set_Char_Size if size did not change. */
|
|
}
|
|
else {
|
|
const FT_Error err = FT_Set_Char_Size(font->face, 0, ft_size, dpi, dpi);
|
|
if (err) {
|
|
/* FIXME: here we can go through the fixed size and choice a close one */
|
|
printf("The current font don't support the size, %f and dpi, %u\n", size, dpi);
|
|
}
|
|
else {
|
|
font->size = size;
|
|
font->dpi = dpi;
|
|
if (gc == NULL) {
|
|
blf_glyph_cache_new(font);
|
|
}
|
|
}
|
|
}
|
|
|
|
blf_glyph_cache_release(font);
|
|
|
|
/* Set fixed-width size for monospaced output. */
|
|
FT_UInt gindex = FT_Get_Char_Index(font->face, U'0');
|
|
if (gindex) {
|
|
FT_Fixed advance = 0;
|
|
FT_Get_Advance(font->face, gindex, FT_LOAD_NO_HINTING, &advance);
|
|
/* Use CSS 'ch unit' width, advance of zero character. */
|
|
font->fixed_width = (int)(advance >> 16);
|
|
}
|
|
else {
|
|
/* Font does not contain "0" so use CSS fallback of 1/2 of em. */
|
|
font->fixed_width = (int)((font->face->size->metrics.height / 2) >> 6);
|
|
}
|
|
if (font->fixed_width < 1) {
|
|
font->fixed_width = 1;
|
|
}
|
|
}
|
|
|
|
/** \} */
|