1 /* 2 * Copyright © 2009 Red Hat, Inc. 3 * Copyright © 2011 Google, Inc. 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 * Google Author(s): Behdad Esfahbod 27 */ 28 29 #ifndef HB_FONT_HH 30 #define HB_FONT_HH 31 32 #include "hb.hh" 33 34 #include "hb-face.hh" 35 #include "hb-shaper.hh" 36 37 38 /* 39 * hb_font_funcs_t 40 */ 41 42 #define HB_FONT_FUNCS_IMPLEMENT_CALLBACKS \ 43 HB_FONT_FUNC_IMPLEMENT (font_h_extents) \ 44 HB_FONT_FUNC_IMPLEMENT (font_v_extents) \ 45 HB_FONT_FUNC_IMPLEMENT (nominal_glyph) \ 46 HB_FONT_FUNC_IMPLEMENT (nominal_glyphs) \ 47 HB_FONT_FUNC_IMPLEMENT (variation_glyph) \ 48 HB_FONT_FUNC_IMPLEMENT (glyph_h_advance) \ 49 HB_FONT_FUNC_IMPLEMENT (glyph_v_advance) \ 50 HB_FONT_FUNC_IMPLEMENT (glyph_h_advances) \ 51 HB_FONT_FUNC_IMPLEMENT (glyph_v_advances) \ 52 HB_FONT_FUNC_IMPLEMENT (glyph_h_origin) \ 53 HB_FONT_FUNC_IMPLEMENT (glyph_v_origin) \ 54 HB_FONT_FUNC_IMPLEMENT (glyph_h_kerning) \ 55 HB_FONT_FUNC_IMPLEMENT (glyph_v_kerning) \ 56 HB_FONT_FUNC_IMPLEMENT (glyph_extents) \ 57 HB_FONT_FUNC_IMPLEMENT (glyph_contour_point) \ 58 HB_FONT_FUNC_IMPLEMENT (glyph_name) \ 59 HB_FONT_FUNC_IMPLEMENT (glyph_from_name) \ 60 /* ^--- Add new callbacks here */ 61 62 struct hb_font_funcs_t 63 { 64 hb_object_header_t header; 65 66 struct { 67 #define HB_FONT_FUNC_IMPLEMENT(name) void *name; 68 HB_FONT_FUNCS_IMPLEMENT_CALLBACKS 69 #undef HB_FONT_FUNC_IMPLEMENT 70 } user_data; 71 72 struct { 73 #define HB_FONT_FUNC_IMPLEMENT(name) hb_destroy_func_t name; 74 HB_FONT_FUNCS_IMPLEMENT_CALLBACKS 75 #undef HB_FONT_FUNC_IMPLEMENT 76 } destroy; 77 78 /* Don't access these directly. Call font->get_*() instead. */ 79 union get_t { 80 struct get_funcs_t { 81 #define HB_FONT_FUNC_IMPLEMENT(name) hb_font_get_##name##_func_t name; 82 HB_FONT_FUNCS_IMPLEMENT_CALLBACKS 83 #undef HB_FONT_FUNC_IMPLEMENT 84 } f; 85 void (*array[0 86 #define HB_FONT_FUNC_IMPLEMENT(name) +1 87 HB_FONT_FUNCS_IMPLEMENT_CALLBACKS 88 #undef HB_FONT_FUNC_IMPLEMENT 89 ]) (); 90 } get; 91 }; 92 DECLARE_NULL_INSTANCE (hb_font_funcs_t); 93 94 95 /* 96 * hb_font_t 97 */ 98 99 #define HB_SHAPER_IMPLEMENT(shaper) HB_SHAPER_DATA_INSTANTIATE_SHAPERS(shaper, font); 100 #include "hb-shaper-list.hh" 101 #undef HB_SHAPER_IMPLEMENT 102 103 struct hb_font_t 104 { 105 hb_object_header_t header; 106 107 hb_font_t *parent; 108 hb_face_t *face; 109 110 int x_scale; 111 int y_scale; 112 113 unsigned int x_ppem; 114 unsigned int y_ppem; 115 116 float ptem; 117 118 /* Font variation coordinates. */ 119 unsigned int num_coords; 120 int *coords; 121 122 hb_font_funcs_t *klass; 123 void *user_data; 124 hb_destroy_func_t destroy; 125 126 hb_shaper_object_dataset_t<hb_font_t> data; /* Various shaper data. */ 127 128 129 /* Convert from font-space to user-space */ dir_scalehb_font_t130 int dir_scale (hb_direction_t direction) 131 { return HB_DIRECTION_IS_VERTICAL(direction) ? y_scale : x_scale; } em_scale_xhb_font_t132 hb_position_t em_scale_x (int16_t v) { return em_scale (v, x_scale); } em_scale_yhb_font_t133 hb_position_t em_scale_y (int16_t v) { return em_scale (v, y_scale); } em_scalef_xhb_font_t134 hb_position_t em_scalef_x (float v) { return em_scalef (v, this->x_scale); } em_scalef_yhb_font_t135 hb_position_t em_scalef_y (float v) { return em_scalef (v, this->y_scale); } em_fscale_xhb_font_t136 float em_fscale_x (int16_t v) { return em_fscale (v, x_scale); } em_fscale_yhb_font_t137 float em_fscale_y (int16_t v) { return em_fscale (v, y_scale); } em_scale_dirhb_font_t138 hb_position_t em_scale_dir (int16_t v, hb_direction_t direction) 139 { return em_scale (v, dir_scale (direction)); } 140 141 /* Convert from parent-font user-space to our user-space */ parent_scale_x_distancehb_font_t142 hb_position_t parent_scale_x_distance (hb_position_t v) 143 { 144 if (unlikely (parent && parent->x_scale != x_scale)) 145 return (hb_position_t) (v * (int64_t) this->x_scale / this->parent->x_scale); 146 return v; 147 } parent_scale_y_distancehb_font_t148 hb_position_t parent_scale_y_distance (hb_position_t v) 149 { 150 if (unlikely (parent && parent->y_scale != y_scale)) 151 return (hb_position_t) (v * (int64_t) this->y_scale / this->parent->y_scale); 152 return v; 153 } parent_scale_x_positionhb_font_t154 hb_position_t parent_scale_x_position (hb_position_t v) 155 { return parent_scale_x_distance (v); } parent_scale_y_positionhb_font_t156 hb_position_t parent_scale_y_position (hb_position_t v) 157 { return parent_scale_y_distance (v); } 158 parent_scale_distancehb_font_t159 void parent_scale_distance (hb_position_t *x, hb_position_t *y) 160 { 161 *x = parent_scale_x_distance (*x); 162 *y = parent_scale_y_distance (*y); 163 } parent_scale_positionhb_font_t164 void parent_scale_position (hb_position_t *x, hb_position_t *y) 165 { 166 *x = parent_scale_x_position (*x); 167 *y = parent_scale_y_position (*y); 168 } 169 170 171 /* Public getters */ 172 173 HB_INTERNAL bool has_func (unsigned int i); 174 HB_INTERNAL bool has_func_set (unsigned int i); 175 176 /* has_* ... */ 177 #define HB_FONT_FUNC_IMPLEMENT(name) \ 178 bool \ 179 has_##name##_func () \ 180 { \ 181 hb_font_funcs_t *funcs = this->klass; \ 182 unsigned int i = offsetof (hb_font_funcs_t::get_t::get_funcs_t, name) / sizeof (funcs->get.array[0]); \ 183 return has_func (i); \ 184 } \ 185 bool \ 186 has_##name##_func_set () \ 187 { \ 188 hb_font_funcs_t *funcs = this->klass; \ 189 unsigned int i = offsetof (hb_font_funcs_t::get_t::get_funcs_t, name) / sizeof (funcs->get.array[0]); \ 190 return has_func_set (i); \ 191 } 192 HB_FONT_FUNCS_IMPLEMENT_CALLBACKS 193 #undef HB_FONT_FUNC_IMPLEMENT 194 get_font_h_extentshb_font_t195 hb_bool_t get_font_h_extents (hb_font_extents_t *extents) 196 { 197 memset (extents, 0, sizeof (*extents)); 198 return klass->get.f.font_h_extents (this, user_data, 199 extents, 200 klass->user_data.font_h_extents); 201 } get_font_v_extentshb_font_t202 hb_bool_t get_font_v_extents (hb_font_extents_t *extents) 203 { 204 memset (extents, 0, sizeof (*extents)); 205 return klass->get.f.font_v_extents (this, user_data, 206 extents, 207 klass->user_data.font_v_extents); 208 } 209 has_glyphhb_font_t210 bool has_glyph (hb_codepoint_t unicode) 211 { 212 hb_codepoint_t glyph; 213 return get_nominal_glyph (unicode, &glyph); 214 } 215 get_nominal_glyphhb_font_t216 hb_bool_t get_nominal_glyph (hb_codepoint_t unicode, 217 hb_codepoint_t *glyph) 218 { 219 *glyph = 0; 220 return klass->get.f.nominal_glyph (this, user_data, 221 unicode, glyph, 222 klass->user_data.nominal_glyph); 223 } get_nominal_glyphshb_font_t224 unsigned int get_nominal_glyphs (unsigned int count, 225 const hb_codepoint_t *first_unicode, 226 unsigned int unicode_stride, 227 hb_codepoint_t *first_glyph, 228 unsigned int glyph_stride) 229 { 230 return klass->get.f.nominal_glyphs (this, user_data, 231 count, 232 first_unicode, unicode_stride, 233 first_glyph, glyph_stride, 234 klass->user_data.nominal_glyphs); 235 } 236 get_variation_glyphhb_font_t237 hb_bool_t get_variation_glyph (hb_codepoint_t unicode, hb_codepoint_t variation_selector, 238 hb_codepoint_t *glyph) 239 { 240 *glyph = 0; 241 return klass->get.f.variation_glyph (this, user_data, 242 unicode, variation_selector, glyph, 243 klass->user_data.variation_glyph); 244 } 245 get_glyph_h_advancehb_font_t246 hb_position_t get_glyph_h_advance (hb_codepoint_t glyph) 247 { 248 return klass->get.f.glyph_h_advance (this, user_data, 249 glyph, 250 klass->user_data.glyph_h_advance); 251 } 252 get_glyph_v_advancehb_font_t253 hb_position_t get_glyph_v_advance (hb_codepoint_t glyph) 254 { 255 return klass->get.f.glyph_v_advance (this, user_data, 256 glyph, 257 klass->user_data.glyph_v_advance); 258 } 259 get_glyph_h_advanceshb_font_t260 void get_glyph_h_advances (unsigned int count, 261 const hb_codepoint_t *first_glyph, 262 unsigned int glyph_stride, 263 hb_position_t *first_advance, 264 unsigned int advance_stride) 265 { 266 return klass->get.f.glyph_h_advances (this, user_data, 267 count, 268 first_glyph, glyph_stride, 269 first_advance, advance_stride, 270 klass->user_data.glyph_h_advances); 271 } 272 get_glyph_v_advanceshb_font_t273 void get_glyph_v_advances (unsigned int count, 274 const hb_codepoint_t *first_glyph, 275 unsigned int glyph_stride, 276 hb_position_t *first_advance, 277 unsigned int advance_stride) 278 { 279 return klass->get.f.glyph_v_advances (this, user_data, 280 count, 281 first_glyph, glyph_stride, 282 first_advance, advance_stride, 283 klass->user_data.glyph_v_advances); 284 } 285 get_glyph_h_originhb_font_t286 hb_bool_t get_glyph_h_origin (hb_codepoint_t glyph, 287 hb_position_t *x, hb_position_t *y) 288 { 289 *x = *y = 0; 290 return klass->get.f.glyph_h_origin (this, user_data, 291 glyph, x, y, 292 klass->user_data.glyph_h_origin); 293 } 294 get_glyph_v_originhb_font_t295 hb_bool_t get_glyph_v_origin (hb_codepoint_t glyph, 296 hb_position_t *x, hb_position_t *y) 297 { 298 *x = *y = 0; 299 return klass->get.f.glyph_v_origin (this, user_data, 300 glyph, x, y, 301 klass->user_data.glyph_v_origin); 302 } 303 get_glyph_h_kerninghb_font_t304 hb_position_t get_glyph_h_kerning (hb_codepoint_t left_glyph, 305 hb_codepoint_t right_glyph) 306 { 307 return klass->get.f.glyph_h_kerning (this, user_data, 308 left_glyph, right_glyph, 309 klass->user_data.glyph_h_kerning); 310 } 311 get_glyph_v_kerninghb_font_t312 hb_position_t get_glyph_v_kerning (hb_codepoint_t top_glyph, 313 hb_codepoint_t bottom_glyph) 314 { 315 return klass->get.f.glyph_v_kerning (this, user_data, 316 top_glyph, bottom_glyph, 317 klass->user_data.glyph_v_kerning); 318 } 319 get_glyph_extentshb_font_t320 hb_bool_t get_glyph_extents (hb_codepoint_t glyph, 321 hb_glyph_extents_t *extents) 322 { 323 memset (extents, 0, sizeof (*extents)); 324 return klass->get.f.glyph_extents (this, user_data, 325 glyph, 326 extents, 327 klass->user_data.glyph_extents); 328 } 329 get_glyph_contour_pointhb_font_t330 hb_bool_t get_glyph_contour_point (hb_codepoint_t glyph, unsigned int point_index, 331 hb_position_t *x, hb_position_t *y) 332 { 333 *x = *y = 0; 334 return klass->get.f.glyph_contour_point (this, user_data, 335 glyph, point_index, 336 x, y, 337 klass->user_data.glyph_contour_point); 338 } 339 get_glyph_namehb_font_t340 hb_bool_t get_glyph_name (hb_codepoint_t glyph, 341 char *name, unsigned int size) 342 { 343 if (size) *name = '\0'; 344 return klass->get.f.glyph_name (this, user_data, 345 glyph, 346 name, size, 347 klass->user_data.glyph_name); 348 } 349 get_glyph_from_namehb_font_t350 hb_bool_t get_glyph_from_name (const char *name, int len, /* -1 means nul-terminated */ 351 hb_codepoint_t *glyph) 352 { 353 *glyph = 0; 354 if (len == -1) len = strlen (name); 355 return klass->get.f.glyph_from_name (this, user_data, 356 name, len, 357 glyph, 358 klass->user_data.glyph_from_name); 359 } 360 361 362 /* A bit higher-level, and with fallback */ 363 get_h_extents_with_fallbackhb_font_t364 void get_h_extents_with_fallback (hb_font_extents_t *extents) 365 { 366 if (!get_font_h_extents (extents)) 367 { 368 extents->ascender = y_scale * .8; 369 extents->descender = extents->ascender - y_scale; 370 extents->line_gap = 0; 371 } 372 } get_v_extents_with_fallbackhb_font_t373 void get_v_extents_with_fallback (hb_font_extents_t *extents) 374 { 375 if (!get_font_v_extents (extents)) 376 { 377 extents->ascender = x_scale / 2; 378 extents->descender = extents->ascender - x_scale; 379 extents->line_gap = 0; 380 } 381 } 382 get_extents_for_directionhb_font_t383 void get_extents_for_direction (hb_direction_t direction, 384 hb_font_extents_t *extents) 385 { 386 if (likely (HB_DIRECTION_IS_HORIZONTAL (direction))) 387 get_h_extents_with_fallback (extents); 388 else 389 get_v_extents_with_fallback (extents); 390 } 391 get_glyph_advance_for_directionhb_font_t392 void get_glyph_advance_for_direction (hb_codepoint_t glyph, 393 hb_direction_t direction, 394 hb_position_t *x, hb_position_t *y) 395 { 396 *x = *y = 0; 397 if (likely (HB_DIRECTION_IS_HORIZONTAL (direction))) 398 *x = get_glyph_h_advance (glyph); 399 else 400 *y = get_glyph_v_advance (glyph); 401 } get_glyph_advances_for_directionhb_font_t402 void get_glyph_advances_for_direction (hb_direction_t direction, 403 unsigned int count, 404 const hb_codepoint_t *first_glyph, 405 unsigned glyph_stride, 406 hb_position_t *first_advance, 407 unsigned advance_stride) 408 { 409 if (likely (HB_DIRECTION_IS_HORIZONTAL (direction))) 410 get_glyph_h_advances (count, first_glyph, glyph_stride, first_advance, advance_stride); 411 else 412 get_glyph_v_advances (count, first_glyph, glyph_stride, first_advance, advance_stride); 413 } 414 guess_v_origin_minus_h_originhb_font_t415 void guess_v_origin_minus_h_origin (hb_codepoint_t glyph, 416 hb_position_t *x, hb_position_t *y) 417 { 418 *x = get_glyph_h_advance (glyph) / 2; 419 420 /* TODO cache this somehow?! */ 421 hb_font_extents_t extents; 422 get_h_extents_with_fallback (&extents); 423 *y = extents.ascender; 424 } 425 get_glyph_h_origin_with_fallbackhb_font_t426 void get_glyph_h_origin_with_fallback (hb_codepoint_t glyph, 427 hb_position_t *x, hb_position_t *y) 428 { 429 if (!get_glyph_h_origin (glyph, x, y) && 430 get_glyph_v_origin (glyph, x, y)) 431 { 432 hb_position_t dx, dy; 433 guess_v_origin_minus_h_origin (glyph, &dx, &dy); 434 *x -= dx; *y -= dy; 435 } 436 } get_glyph_v_origin_with_fallbackhb_font_t437 void get_glyph_v_origin_with_fallback (hb_codepoint_t glyph, 438 hb_position_t *x, hb_position_t *y) 439 { 440 if (!get_glyph_v_origin (glyph, x, y) && 441 get_glyph_h_origin (glyph, x, y)) 442 { 443 hb_position_t dx, dy; 444 guess_v_origin_minus_h_origin (glyph, &dx, &dy); 445 *x += dx; *y += dy; 446 } 447 } 448 get_glyph_origin_for_directionhb_font_t449 void get_glyph_origin_for_direction (hb_codepoint_t glyph, 450 hb_direction_t direction, 451 hb_position_t *x, hb_position_t *y) 452 { 453 if (likely (HB_DIRECTION_IS_HORIZONTAL (direction))) 454 get_glyph_h_origin_with_fallback (glyph, x, y); 455 else 456 get_glyph_v_origin_with_fallback (glyph, x, y); 457 } 458 add_glyph_h_originhb_font_t459 void add_glyph_h_origin (hb_codepoint_t glyph, 460 hb_position_t *x, hb_position_t *y) 461 { 462 hb_position_t origin_x, origin_y; 463 464 get_glyph_h_origin_with_fallback (glyph, &origin_x, &origin_y); 465 466 *x += origin_x; 467 *y += origin_y; 468 } add_glyph_v_originhb_font_t469 void add_glyph_v_origin (hb_codepoint_t glyph, 470 hb_position_t *x, hb_position_t *y) 471 { 472 hb_position_t origin_x, origin_y; 473 474 get_glyph_v_origin_with_fallback (glyph, &origin_x, &origin_y); 475 476 *x += origin_x; 477 *y += origin_y; 478 } add_glyph_origin_for_directionhb_font_t479 void add_glyph_origin_for_direction (hb_codepoint_t glyph, 480 hb_direction_t direction, 481 hb_position_t *x, hb_position_t *y) 482 { 483 hb_position_t origin_x, origin_y; 484 485 get_glyph_origin_for_direction (glyph, direction, &origin_x, &origin_y); 486 487 *x += origin_x; 488 *y += origin_y; 489 } 490 subtract_glyph_h_originhb_font_t491 void subtract_glyph_h_origin (hb_codepoint_t glyph, 492 hb_position_t *x, hb_position_t *y) 493 { 494 hb_position_t origin_x, origin_y; 495 496 get_glyph_h_origin_with_fallback (glyph, &origin_x, &origin_y); 497 498 *x -= origin_x; 499 *y -= origin_y; 500 } subtract_glyph_v_originhb_font_t501 void subtract_glyph_v_origin (hb_codepoint_t glyph, 502 hb_position_t *x, hb_position_t *y) 503 { 504 hb_position_t origin_x, origin_y; 505 506 get_glyph_v_origin_with_fallback (glyph, &origin_x, &origin_y); 507 508 *x -= origin_x; 509 *y -= origin_y; 510 } subtract_glyph_origin_for_directionhb_font_t511 void subtract_glyph_origin_for_direction (hb_codepoint_t glyph, 512 hb_direction_t direction, 513 hb_position_t *x, hb_position_t *y) 514 { 515 hb_position_t origin_x, origin_y; 516 517 get_glyph_origin_for_direction (glyph, direction, &origin_x, &origin_y); 518 519 *x -= origin_x; 520 *y -= origin_y; 521 } 522 get_glyph_kerning_for_directionhb_font_t523 void get_glyph_kerning_for_direction (hb_codepoint_t first_glyph, hb_codepoint_t second_glyph, 524 hb_direction_t direction, 525 hb_position_t *x, hb_position_t *y) 526 { 527 if (likely (HB_DIRECTION_IS_HORIZONTAL (direction))) { 528 *y = 0; 529 *x = get_glyph_h_kerning (first_glyph, second_glyph); 530 } else { 531 *x = 0; 532 *y = get_glyph_v_kerning (first_glyph, second_glyph); 533 } 534 } 535 get_glyph_extents_for_originhb_font_t536 hb_bool_t get_glyph_extents_for_origin (hb_codepoint_t glyph, 537 hb_direction_t direction, 538 hb_glyph_extents_t *extents) 539 { 540 hb_bool_t ret = get_glyph_extents (glyph, extents); 541 542 if (ret) 543 subtract_glyph_origin_for_direction (glyph, direction, &extents->x_bearing, &extents->y_bearing); 544 545 return ret; 546 } 547 get_glyph_contour_point_for_originhb_font_t548 hb_bool_t get_glyph_contour_point_for_origin (hb_codepoint_t glyph, unsigned int point_index, 549 hb_direction_t direction, 550 hb_position_t *x, hb_position_t *y) 551 { 552 hb_bool_t ret = get_glyph_contour_point (glyph, point_index, x, y); 553 554 if (ret) 555 subtract_glyph_origin_for_direction (glyph, direction, x, y); 556 557 return ret; 558 } 559 560 /* Generates gidDDD if glyph has no name. */ 561 void glyph_to_stringhb_font_t562 glyph_to_string (hb_codepoint_t glyph, 563 char *s, unsigned int size) 564 { 565 if (get_glyph_name (glyph, s, size)) return; 566 567 if (size && snprintf (s, size, "gid%u", glyph) < 0) 568 *s = '\0'; 569 } 570 571 /* Parses gidDDD and uniUUUU strings automatically. */ 572 hb_bool_t glyph_from_stringhb_font_t573 glyph_from_string (const char *s, int len, /* -1 means nul-terminated */ 574 hb_codepoint_t *glyph) 575 { 576 if (get_glyph_from_name (s, len, glyph)) return true; 577 578 if (len == -1) len = strlen (s); 579 580 /* Straight glyph index. */ 581 if (hb_codepoint_parse (s, len, 10, glyph)) 582 return true; 583 584 if (len > 3) 585 { 586 /* gidDDD syntax for glyph indices. */ 587 if (0 == strncmp (s, "gid", 3) && 588 hb_codepoint_parse (s + 3, len - 3, 10, glyph)) 589 return true; 590 591 /* uniUUUU and other Unicode character indices. */ 592 hb_codepoint_t unichar; 593 if (0 == strncmp (s, "uni", 3) && 594 hb_codepoint_parse (s + 3, len - 3, 16, &unichar) && 595 get_nominal_glyph (unichar, glyph)) 596 return true; 597 } 598 599 return false; 600 } 601 em_scalehb_font_t602 hb_position_t em_scale (int16_t v, int scale) 603 { 604 int upem = face->get_upem (); 605 int64_t scaled = v * (int64_t) scale; 606 scaled += scaled >= 0 ? upem/2 : -upem/2; /* Round. */ 607 return (hb_position_t) (scaled / upem); 608 } em_scalefhb_font_t609 hb_position_t em_scalef (float v, int scale) 610 { return (hb_position_t) round (v * scale / face->get_upem ()); } em_fscalehb_font_t611 float em_fscale (int16_t v, int scale) 612 { return (float) v * scale / face->get_upem (); } 613 }; 614 DECLARE_NULL_INSTANCE (hb_font_t); 615 616 617 #endif /* HB_FONT_HH */ 618