/** * $Id$ * * ***** BEGIN GPL LICENSE BLOCK ***** * * 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. * * * Contributor(s): Blender Foundation * * ***** END GPL LICENSE BLOCK ***** */ #include #include #include #include #include #include FT_FREETYPE_H #include FT_GLYPH_H #include "MEM_guardedalloc.h" #include "DNA_vec_types.h" #include "BLI_blenlib.h" #include "BLI_linklist.h" /* linknode */ #include "BLI_math.h" #include "BIF_gl.h" #include "BLF_api.h" #include "blf_internal_types.h" #include "blf_internal.h" /* freetype2 handle ONLY for this file!. */ static FT_Library ft_lib; int blf_font_init(void) { return(FT_Init_FreeType(&ft_lib)); } void blf_font_exit(void) { FT_Done_FreeType(ft_lib); } void blf_font_size(FontBLF *font, int size, int dpi) { GlyphCacheBLF *gc; FT_Error err; err= FT_Set_Char_Size(font->face, 0, (size * 64), 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, %d and dpi, %d\n", size, dpi); return; } font->size= size; font->dpi= dpi; gc= blf_glyph_cache_find(font, size, dpi); if (gc) font->glyph_cache= gc; else { gc= blf_glyph_cache_new(font); if (gc) font->glyph_cache= gc; else font->glyph_cache= NULL; } } void blf_font_draw(FontBLF *font, const char *str, unsigned int len) { unsigned int c; GlyphBLF *g, *g_prev; FT_Vector delta; FT_UInt glyph_index; int pen_x, pen_y; int i, has_kerning, st; if (!font->glyph_cache) return; i= 0; pen_x= 0; pen_y= 0; has_kerning= FT_HAS_KERNING(font->face); g_prev= NULL; while (str[i] && i < len) { c= blf_utf8_next((unsigned char *)str, &i); if (c == 0) break; g= blf_glyph_search(font->glyph_cache, c); if (!g) { glyph_index= FT_Get_Char_Index(font->face, c); g= blf_glyph_add(font, glyph_index, c); } /* if we don't found a glyph, skip it. */ if (!g) continue; if (has_kerning && g_prev) { delta.x= 0; delta.y= 0; if (font->flags & BLF_KERNING_DEFAULT) st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, ft_kerning_default, &delta); else st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, FT_KERNING_UNFITTED, &delta); if (st == 0) pen_x += delta.x >> 6; } /* do not return this loop if clipped, we want every character tested */ blf_glyph_render(font, g, (float)pen_x, (float)pen_y); pen_x += g->advance; g_prev= g; } } /* faster version of blf_font_draw, ascii only for view dimensions */ void blf_font_draw_ascii(FontBLF *font, const char *str, unsigned int len) { char c; GlyphBLF *g, *g_prev; FT_Vector delta; FT_UInt glyph_index; int pen_x, pen_y; int i, has_kerning, st; if (!font->glyph_cache) return; pen_x= 0; pen_y= 0; has_kerning= FT_HAS_KERNING(font->face); g_prev= NULL; /* build ascii on demand */ if(font->glyph_ascii_table['0']==NULL) { for(i=0; i<256; i++) { g= blf_glyph_search(font->glyph_cache, i); if (!g) { glyph_index= FT_Get_Char_Index(font->face, i); g= blf_glyph_add(font, glyph_index, i); } font->glyph_ascii_table[i]= g; } } while ((c= *(str++)) && len--) { g= font->glyph_ascii_table[c]; /* if we don't found a glyph, skip it. */ if (!g) continue; if (has_kerning && g_prev) { delta.x= 0; delta.y= 0; if (font->flags & BLF_KERNING_DEFAULT) st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, ft_kerning_default, &delta); else st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, FT_KERNING_UNFITTED, &delta); if (st == 0) pen_x += delta.x >> 6; } /* do not return this loop if clipped, we want every character tested */ blf_glyph_render(font, g, (float)pen_x, (float)pen_y); pen_x += g->advance; g_prev= g; } } void blf_font_buffer(FontBLF *font, const char *str) { unsigned char *cbuf; unsigned int c; unsigned char b_col_char[3]; GlyphBLF *g, *g_prev; FT_Vector delta; FT_UInt glyph_index; float a, *fbuf; int pen_x, y, x, yb; int i, has_kerning, st, chx, chy; if (!font->glyph_cache || (!font->b_fbuf && !font->b_cbuf)) return; i= 0; pen_x= (int)font->pos[0]; has_kerning= FT_HAS_KERNING(font->face); g_prev= NULL; b_col_char[0]= font->b_col[0] * 255; b_col_char[1]= font->b_col[1] * 255; b_col_char[2]= font->b_col[2] * 255; while (str[i]) { int pen_y; c= blf_utf8_next((unsigned char *)str, &i); if (c == 0) break; g= blf_glyph_search(font->glyph_cache, c); if (!g) { glyph_index= FT_Get_Char_Index(font->face, c); g= blf_glyph_add(font, glyph_index, c); } /* if we don't found a glyph, skip it. */ if (!g) continue; if (has_kerning && g_prev) { delta.x= 0; delta.y= 0; if (font->flags & BLF_KERNING_DEFAULT) st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, ft_kerning_default, &delta); else st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, FT_KERNING_UNFITTED, &delta); if (st == 0) pen_x += delta.x >> 6; } chx= pen_x + ((int)g->pos_x); chy= (int)font->pos[1] + g->height; if (g->pitch < 0) { pen_y = (int)font->pos[1] + (g->height - (int)g->pos_y); } else { pen_y = (int)font->pos[1] - (g->height - (int)g->pos_y); } if ((chx + g->width) >= 0 && chx < font->bw && (pen_y + g->height) >= 0 && pen_y < font->bh) { /* dont draw beyond the buffer bounds */ int width_clip= g->width; int height_clip= g->height; if (width_clip + chx > font->bw) width_clip -= chx + width_clip - font->bw; if (height_clip + pen_y > font->bh) height_clip -= pen_y + height_clip - font->bh; yb= g->pitch < 0 ? 0 : g->height-1; if (font->b_fbuf) { for (y=(chy >= 0 ? 0:-chy); y < height_clip; y++) { for (x=(chx >= 0 ? 0:-chx); x < width_clip; x++) { a= *(g->bitmap + x + (yb * g->pitch)) / 255.0f; if(a > 0.0f) { fbuf= font->b_fbuf + font->bch * ((chx + x) + ((pen_y + y)*font->bw)); if (a >= 1.0f) { fbuf[0]= font->b_col[0]; fbuf[1]= font->b_col[1]; fbuf[2]= font->b_col[2]; } else { fbuf[0]= (font->b_col[0]*a) + (fbuf[0] * (1-a)); fbuf[1]= (font->b_col[1]*a) + (fbuf[1] * (1-a)); fbuf[2]= (font->b_col[2]*a) + (fbuf[2] * (1-a)); } } } if (g->pitch < 0) yb++; else yb--; } } if (font->b_cbuf) { for (y= 0; y < height_clip; y++) { for (x= 0; x < width_clip; x++) { a= *(g->bitmap + x + (yb * g->pitch)) / 255.0f; if(a > 0.0f) { cbuf= font->b_cbuf + font->bch * ((chx + x) + ((pen_y + y)*font->bw)); if (a >= 1.0f) { cbuf[0]= b_col_char[0]; cbuf[1]= b_col_char[1]; cbuf[2]= b_col_char[2]; } else { cbuf[0]= (b_col_char[0]*a) + (cbuf[0] * (1-a)); cbuf[1]= (b_col_char[1]*a) + (cbuf[1] * (1-a)); cbuf[2]= (b_col_char[2]*a) + (cbuf[2] * (1-a)); } } } if (g->pitch < 0) yb++; else yb--; } } } pen_x += g->advance; g_prev= g; } } void blf_font_boundbox(FontBLF *font, const char *str, rctf *box) { unsigned int c; GlyphBLF *g, *g_prev; FT_Vector delta; FT_UInt glyph_index; rctf gbox; int pen_x, pen_y; int i, has_kerning, st; if (!font->glyph_cache) return; box->xmin= 32000.0f; box->xmax= -32000.0f; box->ymin= 32000.0f; box->ymax= -32000.0f; i= 0; pen_x= 0; pen_y= 0; has_kerning= FT_HAS_KERNING(font->face); g_prev= NULL; while (str[i]) { c= blf_utf8_next((unsigned char *)str, &i); if (c == 0) break; g= blf_glyph_search(font->glyph_cache, c); if (!g) { glyph_index= FT_Get_Char_Index(font->face, c); g= blf_glyph_add(font, glyph_index, c); } /* if we don't found a glyph, skip it. */ if (!g) continue; if (has_kerning && g_prev) { delta.x= 0; delta.y= 0; if (font->flags & BLF_KERNING_DEFAULT) st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, ft_kerning_default, &delta); else st= FT_Get_Kerning(font->face, g_prev->idx, g->idx, FT_KERNING_UNFITTED, &delta); if (st == 0) pen_x += delta.x >> 6; } gbox.xmin= pen_x; gbox.xmax= pen_x + g->advance; gbox.ymin= g->box.ymin + pen_y; gbox.ymax= g->box.ymax + 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; g_prev= g; } if (box->xmin > box->xmax) { box->xmin= 0.0f; box->ymin= 0.0f; box->xmax= 0.0f; box->ymax= 0.0f; } } void blf_font_width_and_height(FontBLF *font, const char *str, float *width, float *height) { float xa, ya; rctf box; if (font->glyph_cache) { if (font->flags & BLF_ASPECT) { xa= font->aspect[0]; ya= font->aspect[1]; } else { xa= 1.0f; ya= 1.0f; } blf_font_boundbox(font, str, &box); *width= ((box.xmax - box.xmin) * xa); *height= ((box.ymax - box.ymin) * ya); } } float blf_font_width(FontBLF *font, const char *str) { float xa; rctf box; if (!font->glyph_cache) return(0.0f); if (font->flags & BLF_ASPECT) xa= font->aspect[0]; else xa= 1.0f; blf_font_boundbox(font, str, &box); return((box.xmax - box.xmin) * xa); } float blf_font_height(FontBLF *font, const char *str) { float ya; rctf box; if (!font->glyph_cache) return(0.0f); if (font->flags & BLF_ASPECT) ya= font->aspect[1]; else ya= 1.0f; blf_font_boundbox(font, str, &box); return((box.ymax - box.ymin) * ya); } float blf_font_fixed_width(FontBLF *font) { GlyphBLF *g; FT_UInt glyph_index; unsigned int c = ' '; if (!font->glyph_cache) return 0.0f; glyph_index= FT_Get_Char_Index(font->face, c); g= blf_glyph_search(font->glyph_cache, c); if (!g) g= blf_glyph_add(font, glyph_index, c); /* if we don't find the glyph. */ if (!g) return 0.0f; return g->advance; } void blf_font_free(FontBLF *font) { GlyphCacheBLF *gc; font->glyph_cache= NULL; while (font->cache.first) { gc= font->cache.first; BLI_remlink(&font->cache, gc); blf_glyph_cache_free(gc); } FT_Done_Face(font->face); if (font->filename) MEM_freeN(font->filename); if (font->name) MEM_freeN(font->name); MEM_freeN(font); } static void blf_font_fill(FontBLF *font) { int i; 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 (i= 0; i < 16; i++) font->m[i]= 0; 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; font->cache.first= NULL; font->cache.last= NULL; font->glyph_cache= NULL; font->blur= 0; font->max_tex_size= -1; font->b_fbuf= NULL; font->b_cbuf= NULL; font->bw= 0; font->bh= 0; font->bch= 0; font->b_col[0]= 0; font->b_col[1]= 0; font->b_col[2]= 0; font->b_col[3]= 0; font->ft_lib= ft_lib; memset(font->glyph_ascii_table, 0, sizeof(font->glyph_ascii_table)); } FontBLF *blf_font_new(const char *name, const char *filename) { FontBLF *font; FT_Error err; char *mfile; font= (FontBLF *)MEM_mallocN(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) { printf("Can't set the unicode 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); MEM_freeN(mfile); } font->name= BLI_strdup(name); font->filename= BLI_strdup(filename); blf_font_fill(font); 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= (FT_Byte *)mem; open.memory_size= mem_size; FT_Attach_Stream(font->face, &open); } FontBLF *blf_font_new_from_mem(const char *name, unsigned char *mem, int mem_size) { FontBLF *font; FT_Error err; font= (FontBLF *)MEM_mallocN(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); }