| OLD | NEW |
| (Empty) |
| 1 /* | |
| 2 * Copyright (C) 2009 Red Hat, Inc. | |
| 3 * Copyright (C) 2009 Keith Stribley | |
| 4 * | |
| 5 * This is part of HarfBuzz, a text shaping library. | |
| 6 * | |
| 7 * Permission is hereby granted, without written agreement and without | |
| 8 * license or royalty fees, to use, copy, modify, and distribute this | |
| 9 * software and its documentation for any purpose, provided that the | |
| 10 * above copyright notice and the following two paragraphs appear in | |
| 11 * all copies of this software. | |
| 12 * | |
| 13 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR | |
| 14 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES | |
| 15 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN | |
| 16 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH | |
| 17 * DAMAGE. | |
| 18 * | |
| 19 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, | |
| 20 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND | |
| 21 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS | |
| 22 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO | |
| 23 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. | |
| 24 * | |
| 25 * Red Hat Author(s): Behdad Esfahbod | |
| 26 */ | |
| 27 | |
| 28 #include "hb-private.h" | |
| 29 | |
| 30 #include "hb-ft.h" | |
| 31 | |
| 32 #include "hb-font-private.h" | |
| 33 | |
| 34 #include FT_TRUETYPE_TABLES_H | |
| 35 | |
| 36 HB_BEGIN_DECLS | |
| 37 | |
| 38 | |
| 39 static hb_codepoint_t | |
| 40 hb_ft_get_glyph (hb_font_t *font HB_UNUSED, | |
| 41 hb_face_t *face HB_UNUSED, | |
| 42 const void *user_data, | |
| 43 hb_codepoint_t unicode, | |
| 44 hb_codepoint_t variation_selector) | |
| 45 { | |
| 46 FT_Face ft_face = (FT_Face) user_data; | |
| 47 | |
| 48 #ifdef HAVE_FT_FACE_GETCHARVARIANTINDEX | |
| 49 if (unlikely (variation_selector)) { | |
| 50 hb_codepoint_t glyph = FT_Face_GetCharVariantIndex (ft_face, unicode, variat
ion_selector); | |
| 51 if (glyph) | |
| 52 return glyph; | |
| 53 } | |
| 54 #endif | |
| 55 | |
| 56 return FT_Get_Char_Index (ft_face, unicode); | |
| 57 } | |
| 58 | |
| 59 static void | |
| 60 hb_ft_get_glyph_advance (hb_font_t *font HB_UNUSED, | |
| 61 hb_face_t *face HB_UNUSED, | |
| 62 const void *user_data, | |
| 63 hb_codepoint_t glyph, | |
| 64 hb_position_t *x_advance, | |
| 65 hb_position_t *y_advance) | |
| 66 { | |
| 67 FT_Face ft_face = (FT_Face) user_data; | |
| 68 int load_flags = FT_LOAD_DEFAULT; | |
| 69 | |
| 70 /* TODO: load_flags, embolden, etc */ | |
| 71 | |
| 72 if (likely (!FT_Load_Glyph (ft_face, glyph, load_flags))) | |
| 73 { | |
| 74 *x_advance = ft_face->glyph->advance.x; | |
| 75 *y_advance = ft_face->glyph->advance.y; | |
| 76 } | |
| 77 } | |
| 78 | |
| 79 static void | |
| 80 hb_ft_get_glyph_extents (hb_font_t *font HB_UNUSED, | |
| 81 hb_face_t *face HB_UNUSED, | |
| 82 const void *user_data, | |
| 83 hb_codepoint_t glyph, | |
| 84 hb_glyph_extents_t *extents) | |
| 85 { | |
| 86 FT_Face ft_face = (FT_Face) user_data; | |
| 87 int load_flags = FT_LOAD_DEFAULT; | |
| 88 | |
| 89 /* TODO: load_flags, embolden, etc */ | |
| 90 | |
| 91 if (likely (!FT_Load_Glyph (ft_face, glyph, load_flags))) | |
| 92 { | |
| 93 /* XXX: A few negations should be in order here, not sure. */ | |
| 94 extents->x_bearing = ft_face->glyph->metrics.horiBearingX; | |
| 95 extents->y_bearing = ft_face->glyph->metrics.horiBearingY; | |
| 96 extents->width = ft_face->glyph->metrics.width; | |
| 97 extents->height = ft_face->glyph->metrics.height; | |
| 98 } | |
| 99 } | |
| 100 | |
| 101 static hb_bool_t | |
| 102 hb_ft_get_contour_point (hb_font_t *font HB_UNUSED, | |
| 103 hb_face_t *face HB_UNUSED, | |
| 104 const void *user_data, | |
| 105 unsigned int point_index, | |
| 106 hb_codepoint_t glyph, | |
| 107 hb_position_t *x, | |
| 108 hb_position_t *y) | |
| 109 { | |
| 110 FT_Face ft_face = (FT_Face) user_data; | |
| 111 int load_flags = FT_LOAD_DEFAULT; | |
| 112 | |
| 113 /* TODO: load_flags, embolden, etc */ | |
| 114 | |
| 115 if (unlikely (FT_Load_Glyph (ft_face, glyph, load_flags))) | |
| 116 return FALSE; | |
| 117 | |
| 118 if (unlikely (ft_face->glyph->format != FT_GLYPH_FORMAT_OUTLINE)) | |
| 119 return FALSE; | |
| 120 | |
| 121 if (unlikely (point_index >= (unsigned int) ft_face->glyph->outline.n_points)) | |
| 122 return FALSE; | |
| 123 | |
| 124 *x = ft_face->glyph->outline.points[point_index].x; | |
| 125 *y = ft_face->glyph->outline.points[point_index].y; | |
| 126 | |
| 127 return TRUE; | |
| 128 } | |
| 129 | |
| 130 static hb_position_t | |
| 131 hb_ft_get_kerning (hb_font_t *font HB_UNUSED, | |
| 132 hb_face_t *face HB_UNUSED, | |
| 133 const void *user_data, | |
| 134 hb_codepoint_t first_glyph, | |
| 135 hb_codepoint_t second_glyph) | |
| 136 { | |
| 137 FT_Face ft_face = (FT_Face) user_data; | |
| 138 FT_Vector kerning; | |
| 139 | |
| 140 /* TODO: Kern type? */ | |
| 141 if (FT_Get_Kerning (ft_face, first_glyph, second_glyph, FT_KERNING_DEFAULT, &k
erning)) | |
| 142 return 0; | |
| 143 | |
| 144 return kerning.x; | |
| 145 } | |
| 146 | |
| 147 static hb_font_funcs_t ft_ffuncs = { | |
| 148 HB_REFERENCE_COUNT_INVALID, /* ref_count */ | |
| 149 TRUE, /* immutable */ | |
| 150 { | |
| 151 hb_ft_get_glyph, | |
| 152 hb_ft_get_glyph_advance, | |
| 153 hb_ft_get_glyph_extents, | |
| 154 hb_ft_get_contour_point, | |
| 155 hb_ft_get_kerning | |
| 156 } | |
| 157 }; | |
| 158 | |
| 159 hb_font_funcs_t * | |
| 160 hb_ft_get_font_funcs (void) | |
| 161 { | |
| 162 return &ft_ffuncs; | |
| 163 } | |
| 164 | |
| 165 | |
| 166 static hb_blob_t * | |
| 167 get_table (hb_tag_t tag, void *user_data) | |
| 168 { | |
| 169 FT_Face ft_face = (FT_Face) user_data; | |
| 170 FT_Byte *buffer; | |
| 171 FT_ULong length = 0; | |
| 172 FT_Error error; | |
| 173 | |
| 174 if (unlikely (tag == HB_TAG_NONE)) | |
| 175 return NULL; | |
| 176 | |
| 177 error = FT_Load_Sfnt_Table (ft_face, tag, 0, NULL, &length); | |
| 178 if (error) | |
| 179 return NULL; | |
| 180 | |
| 181 /* TODO Use FT_Memory? */ | |
| 182 buffer = (FT_Byte *) malloc (length); | |
| 183 if (buffer == NULL) | |
| 184 return NULL; | |
| 185 | |
| 186 error = FT_Load_Sfnt_Table (ft_face, tag, 0, buffer, &length); | |
| 187 if (error) | |
| 188 return NULL; | |
| 189 | |
| 190 return hb_blob_create ((const char *) buffer, length, | |
| 191 HB_MEMORY_MODE_WRITABLE, | |
| 192 free, buffer); | |
| 193 } | |
| 194 | |
| 195 | |
| 196 hb_face_t * | |
| 197 hb_ft_face_create (FT_Face ft_face, | |
| 198 hb_destroy_func_t destroy) | |
| 199 { | |
| 200 hb_face_t *face; | |
| 201 | |
| 202 if (ft_face->stream->read == NULL) { | |
| 203 hb_blob_t *blob; | |
| 204 | |
| 205 blob = hb_blob_create ((const char *) ft_face->stream->base, | |
| 206 (unsigned int) ft_face->stream->size, | |
| 207 /* TODO: Check FT_FACE_FLAG_EXTERNAL_STREAM? */ | |
| 208 HB_MEMORY_MODE_READONLY_MAY_MAKE_WRITABLE, | |
| 209 destroy, ft_face); | |
| 210 face = hb_face_create_for_data (blob, ft_face->face_index); | |
| 211 hb_blob_destroy (blob); | |
| 212 } else { | |
| 213 face = hb_face_create_for_tables (get_table, destroy, ft_face); | |
| 214 } | |
| 215 | |
| 216 return face; | |
| 217 } | |
| 218 | |
| 219 static void | |
| 220 hb_ft_face_finalize (FT_Face ft_face) | |
| 221 { | |
| 222 hb_face_destroy ((hb_face_t *) ft_face->generic.data); | |
| 223 } | |
| 224 | |
| 225 hb_face_t * | |
| 226 hb_ft_face_create_cached (FT_Face ft_face) | |
| 227 { | |
| 228 if (unlikely (!ft_face->generic.data || ft_face->generic.finalizer != (FT_Gene
ric_Finalizer) hb_ft_face_finalize)) | |
| 229 { | |
| 230 if (ft_face->generic.finalizer) | |
| 231 ft_face->generic.finalizer (ft_face); | |
| 232 | |
| 233 ft_face->generic.data = hb_ft_face_create (ft_face, NULL); | |
| 234 ft_face->generic.finalizer = (FT_Generic_Finalizer) hb_ft_face_finalize; | |
| 235 } | |
| 236 | |
| 237 return hb_face_reference ((hb_face_t *) ft_face->generic.data); | |
| 238 } | |
| 239 | |
| 240 | |
| 241 hb_font_t * | |
| 242 hb_ft_font_create (FT_Face ft_face, | |
| 243 hb_destroy_func_t destroy) | |
| 244 { | |
| 245 hb_font_t *font; | |
| 246 | |
| 247 font = hb_font_create (); | |
| 248 hb_font_set_funcs (font, | |
| 249 hb_ft_get_font_funcs (), | |
| 250 destroy, ft_face); | |
| 251 hb_font_set_scale (font, | |
| 252 ((uint64_t) ft_face->size->metrics.x_scale * (uint64_t) ft_
face->units_per_EM) >> 16, | |
| 253 ((uint64_t) ft_face->size->metrics.y_scale * (uint64_t) ft_
face->units_per_EM) >> 16); | |
| 254 hb_font_set_ppem (font, | |
| 255 ft_face->size->metrics.x_ppem, | |
| 256 ft_face->size->metrics.y_ppem); | |
| 257 | |
| 258 return font; | |
| 259 } | |
| 260 | |
| 261 | |
| 262 HB_END_DECLS | |
| OLD | NEW |