1<?xml version="1.0"?> 2<!DOCTYPE book PUBLIC "-//OASIS//DTD DocBook XML V4.3//EN" 3 "http://www.oasis-open.org/docbook/xml/4.3/docbookx.dtd" [ 4 <!ENTITY % local.common.attrib "xmlns:xi CDATA #FIXED 'http://www.w3.org/2003/XInclude'"> 5 <!ENTITY version SYSTEM "version.xml"> 6]> 7<chapter id="fonts-and-faces"> 8 <title>Fonts, faces, and output</title> 9 <para> 10 In the previous chapter, we saw how to set up a buffer and fill 11 it with text as Unicode code points. In order to shape this 12 buffer text with HarfBuzz, you will need also need a font 13 object. 14 </para> 15 <para> 16 HarfBuzz provides abstractions to help you cache and reuse the 17 heavier parts of working with binary fonts, so we will look at 18 how to do that. We will also look at how to work with the 19 FreeType font-rendering library and at how you can customize 20 HarfBuzz to work with other libraries. 21 </para> 22 <para> 23 Finally, we will look at how to work with OpenType variable 24 fonts, the latest update to the OpenType font format, and at 25 some other recent additions to OpenType. 26 </para> 27 28 <section id="fonts-and-faces-objects"> 29 <title>Font and face objects</title> 30 <para> 31 The outcome of shaping a run of text depends on the contents of 32 a specific font file (such as the substitutions and positioning 33 moves in the 'GSUB' and 'GPOS' tables), so HarfBuzz makes 34 accessing those internals fast. 35 </para> 36 <para> 37 An <type>hb_face_t</type> represents a <emphasis>face</emphasis> 38 in HarfBuzz. This data type is a wrapper around an 39 <type>hb_blob_t</type> blob that holds the contents of a binary 40 font file. Since HarfBuzz supports TrueType Collections and 41 OpenType Collections (each of which can include multiple 42 typefaces), a HarfBuzz face also requires an index number 43 specifying which typeface in the file you want to use. Most of 44 the font files you will encounter in the wild include just a 45 single face, however, so most of the time you would pass in 46 <literal>0</literal> as the index when you create a face: 47 </para> 48 <programlisting language="C"> 49 hb_blob_t* blob = hb_blob_create_from_file(file); 50 ... 51 hb_face_t* face = hb_face_create(blob, 0); 52 </programlisting> 53 <para> 54 On its own, a face object is not quite ready to use for 55 shaping. The typeface must be set to a specific point size in 56 order for some details (such as hinting) to work. In addition, 57 if the font file in question is an OpenType Variable Font, then 58 you may need to specify one or variation-axis settings (or a 59 named instance) in order to get the output you need. 60 </para> 61 <para> 62 In HarfBuzz, you do this by creating a <emphasis>font</emphasis> 63 object from your face. 64 </para> 65 <para> 66 Font objects also have the advantage of being considerably 67 lighter-weight than face objects (remember that a face contains 68 the contents of a binary font file mapped into memory). As a 69 result, you can cache and reuse a font object, but you could 70 also create a new one for each additional size you needed. 71 Creating new fonts incurs some additional overhead, of course, 72 but whether or not it is excessive is your call in the end. In 73 contrast, face objects are substantially larger, and you really 74 should cache them and reuse them whenever possible. 75 </para> 76 <para> 77 You can create a font object from a face object: 78 </para> 79 <programlisting language="C"> 80 hb_font_t* hb_font = hb_font_create(hb_face); 81 </programlisting> 82 <para> 83 After creating a font, there are a few properties you should 84 set. Many fonts enable and disable hints based on the size it 85 is used at, so setting this is important for font 86 objects. <function>hb_font_set_ppem(font, x_ppem, 87 y_ppem)</function> sets the pixels-per-EM value of the font. You 88 can also set the point size of the font with 89 <function>hb_font_set_ptem(font, ptem)</function>. HarfBuzz uses the 90 industry standard 72 points per inch. 91 </para> 92 <para> 93 HarfBuzz lets you specify the degree subpixel precision you want 94 through a scaling factor. You can set horizontal and 95 vertical scaling factors on the 96 font by calling <function>hb_font_set_scale(font, x_scale, 97 y_scale)</function>. 98 </para> 99 <para> 100 There may be times when you are handed a font object and need to 101 access the face object that it comes from. For that, you can call 102 </para> 103 <programlisting language="C"> 104 hb_face = hb_font_get_face(hb_font); 105 </programlisting> 106 <para> 107 You can also create a font object from an existing font object 108 using the <function>hb_font_create_sub_font()</function> 109 function. This creates a child font object that is initiated 110 with the same attributes as its parent; it can be used to 111 quickly set up a new font for the purpose of overriding a specific 112 font-functions method. 113 </para> 114 <para> 115 All face objects and font objects are lifecycle-managed by 116 HarfBuzz. After creating a face, you increase its reference 117 count with <function>hb_face_reference(face)</function> and 118 decrease it with 119 <function>hb_face_destroy(face)</function>. Likewise, you 120 increase the reference count on a font with 121 <function>hb_font_reference(font)</function> and decrease it 122 with <function>hb_font_destroy(font)</function>. 123 </para> 124 <para> 125 You can also attach user data to face objects and font objects. 126 </para> 127 </section> 128 129 <section id="fonts-and-faces-custom-functions"> 130 <title>Customizing font functions</title> 131 <para> 132 During shaping, HarfBuzz frequently needs to query font objects 133 to get at the contents and parameters of the glyphs in a font 134 file. It includes a built-in set of functions that is tailored 135 to working with OpenType fonts. However, as was the case with 136 Unicode functions in the buffers chapter, HarfBuzz also wants to 137 make it easy for you to assign a substitute set of font 138 functions if you are developing a program to work with a library 139 or platform that provides its own font functions. 140 </para> 141 <para> 142 Therefore, the HarfBuzz API defines a set of virtual 143 methods for accessing font-object properties, and you can 144 replace the defaults with your own selections without 145 interfering with the shaping process. Each font object in 146 HarfBuzz includes a structure called 147 <literal>font_funcs</literal> that serves as a vtable for the 148 font object. The virtual methods in 149 <literal>font_funcs</literal> are: 150 </para> 151 <itemizedlist> 152 <listitem> 153 <para> 154 <function>hb_font_get_font_h_extents_func_t</function>: returns 155 the extents of the font for horizontal text. 156 </para> 157 </listitem> 158 <listitem> 159 <para> 160 <function>hb_font_get_font_v_extents_func_t</function>: returns 161 the extents of the font for vertical text. 162 </para> 163 </listitem> 164 <listitem> 165 <para> 166 <function>hb_font_get_nominal_glyph_func_t</function>: returns 167 the font's nominal glyph for a given code point. 168 </para> 169 </listitem> 170 <listitem> 171 <para> 172 <function>hb_font_get_variation_glyph_func_t</function>: returns 173 the font's glyph for a given code point when it is followed by a 174 given Variation Selector. 175 </para> 176 </listitem> 177 <listitem> 178 <para> 179 <function>hb_font_get_nominal_glyphs_func_t</function>: returns 180 the font's nominal glyphs for a series of code points. 181 </para> 182 </listitem> 183 <listitem> 184 <para> 185 <function>hb_font_get_glyph_advance_func_t</function>: returns 186 the advance for a glyph. 187 </para> 188 </listitem> 189 <listitem> 190 <para> 191 <function>hb_font_get_glyph_h_advance_func_t</function>: returns 192 the advance for a glyph for horizontal text. 193 </para> 194 </listitem> 195 <listitem> 196 <para> 197 <function>hb_font_get_glyph_v_advance_func_t</function>:returns 198 the advance for a glyph for vertical text. 199 </para> 200 </listitem> 201 <listitem> 202 <para> 203 <function>hb_font_get_glyph_advances_func_t</function>: returns 204 the advances for a series of glyphs. 205 </para> 206 </listitem> 207 <listitem> 208 <para> 209 <function>hb_font_get_glyph_h_advances_func_t</function>: returns 210 the advances for a series of glyphs for horizontal text . 211 </para> 212 </listitem> 213 <listitem> 214 <para> 215 <function>hb_font_get_glyph_v_advances_func_t</function>: returns 216 the advances for a series of glyphs for vertical text. 217 </para> 218 </listitem> 219 <listitem> 220 <para> 221 <function>hb_font_get_glyph_origin_func_t</function>: returns 222 the origin coordinates of a glyph. 223 </para> 224 </listitem> 225 <listitem> 226 <para> 227 <function>hb_font_get_glyph_h_origin_func_t</function>: returns 228 the origin coordinates of a glyph for horizontal text. 229 </para> 230 </listitem> 231 <listitem> 232 <para> 233 <function>hb_font_get_glyph_v_origin_func_t</function>: returns 234 the origin coordinates of a glyph for vertical text. 235 </para> 236 </listitem> 237 <listitem> 238 <para> 239 <function>hb_font_get_glyph_extents_func_t</function>: returns 240 the extents for a glyph. 241 </para> 242 </listitem> 243 <listitem> 244 <para> 245 <function>hb_font_get_glyph_contour_point_func_t</function>: 246 returns the coordinates of a specific contour point from a glyph. 247 </para> 248 </listitem> 249 <listitem> 250 <para> 251 <function>hb_font_get_glyph_name_func_t</function>: returns the 252 name of a glyph (from its glyph index). 253 </para> 254 </listitem> 255 <listitem> 256 <para> 257 <function>hb_font_get_glyph_from_name_func_t</function>: returns 258 the glyph index that corresponds to a given glyph name. 259 </para> 260 </listitem> 261 </itemizedlist> 262 <para> 263 You can create new font-functions by calling 264 <function>hb_font_funcs_create()</function>: 265 </para> 266 <programlisting language="C"> 267 hb_font_funcs_t *ffunctions = hb_font_funcs_create (); 268 hb_font_set_funcs (font, ffunctions, font_data, destroy); 269 </programlisting> 270 <para> 271 The individual methods can each be set with their own setter 272 function, such as 273 <function>hb_font_funcs_set_nominal_glyph_func(ffunctions, 274 func, user_data, destroy)</function>. 275 </para> 276 <para> 277 Font-functions structures can be reused for multiple font 278 objects, and can be reference counted with 279 <function>hb_font_funcs_reference()</function> and 280 <function>hb_font_funcs_destroy()</function>. Just like other 281 objects in HarfBuzz, you can set user-data for each 282 font-functions structure and assign a destroy callback for 283 it. 284 </para> 285 <para> 286 You can also mark a font-functions structure as immutable, 287 with <function>hb_font_funcs_make_immutable()</function>. This 288 is especially useful if your code is a library or framework that 289 will have its own client programs. By marking your 290 font-functions structures as immutable, you prevent your client 291 programs from changing the configuration and introducing 292 inconsistencies and errors downstream. 293 </para> 294 <para> 295 To override only some functions while using the default implementation 296 for the others, you will need to create a sub-font. By default, the 297 sub-font uses the font functions of its parent except for the functions 298 that were explicitly set. The following code will override only the 299 <function>hb_font_get_nominal_glyph_func_t</function> for the sub-font: 300 </para> 301 <programlisting language="C"> 302 hb_font_t *subfont = hb_font_create_sub_font (font) 303 hb_font_funcs_t *ffunctions = hb_font_funcs_create (); 304 hb_font_funcs_set_nominal_glyph_func (ffunctions, func, user_data, destroy); 305 hb_font_set_funcs (subfont, ffunctions, font_data, destroy); 306 hb_font_funcs_destroy (ffunctions); 307 </programlisting> 308 </section> 309 310 <section id="fonts-and-faces-native-opentype"> 311 <title>Font objects and HarfBuzz's native OpenType implementation</title> 312 <para> 313 By default, whenever HarfBuzz creates a font object, it will 314 configure the font to use a built-in set of font functions that 315 supports contemporary OpenType font internals. If you want to 316 work with OpenType or TrueType fonts, you should be able to use 317 these functions without difficulty. 318 </para> 319 <para> 320 Many of the methods in the font-functions structure deal with 321 the fundamental properties of glyphs that are required for 322 shaping text: extents (the maximums and minimums on each axis), 323 origins (the <literal>(0,0)</literal> coordinate point which 324 glyphs are drawn in reference to), and advances (the amount that 325 the cursor needs to be moved after drawing each glyph, including 326 any empty space for the glyph's side bearings). 327 </para> 328 <para> 329 As you can see in the list of functions, there are separate "horizontal" 330 and "vertical" variants depending on whether the text is set in 331 the horizontal or vertical direction. For some scripts, fonts 332 that are designed to support text set horizontally or vertically (for 333 example, in Japanese) may include metrics for both text 334 directions. When fonts don't include this information, HarfBuzz 335 does its best to transform what the font provides. 336 </para> 337 <para> 338 In addition to the direction-specific functions, HarfBuzz 339 provides some higher-level functions for fetching information 340 like extents and advances for a glyph. If you call 341 </para> 342 <programlisting language="C"> 343 hb_font_get_glyph_advance_for_direction(font, direction, extents); 344 </programlisting> 345 <para> 346 then you can provide any <type>hb_direction_t</type> as the 347 <parameter>direction</parameter> parameter, and HarfBuzz will 348 use the correct function variant for the text direction. There 349 are similar higher-level versions of the functions for fetching 350 extents, origin coordinates, and contour-point 351 coordinates. There are also addition and subtraction functions 352 for moving points with respect to the origin. 353 </para> 354 <para> 355 There are also methods for fetching the glyph ID that 356 corresponds to a Unicode code point (possibly when followed by a 357 variation-selector code point), fetching the glyph name from the 358 font, and fetching the glyph ID that corresponds to a glyph name 359 you already have. 360 </para> 361 <para> 362 HarfBuzz also provides functions for converting between glyph 363 names and string 364 variables. <function>hb_font_glyph_to_string(font, glyph, s, 365 size)</function> retrieves the name for the glyph ID 366 <parameter>glyph</parameter> from the font object. It generates a 367 generic name of the form <literal>gidDDD</literal> (where DDD is 368 the glyph index) if there is no name for the glyph in the 369 font. The <function>hb_font_glyph_from_string(font, s, len, 370 glyph)</function> takes an input string <parameter>s</parameter> 371 and looks for a glyph with that name in the font, returning its 372 glyph ID in the <parameter>glyph</parameter> 373 output parameter. It automatically parses 374 <literal>gidDDD</literal> and <literal>uniUUUU</literal> strings. 375 </para> 376 </section> 377 378 379 <!-- Commenting out FreeType integration section-holder for now. May move 380 to the full-blown Integration Chapter. --> 381 382 <!-- <section id="fonts-and-faces-freetype"> 383 <title>Using FreeType</title> 384 <para> 385 386 </para> 387 <para> 388 389 </para> 390 </section> --> 391 392 <section id="fonts-and-faces-variable"> 393 <title>Working with OpenType Variable Fonts</title> 394 <para> 395 If you are working with OpenType Variable Fonts, there are a few 396 additional functions you should use to specify the 397 variation-axis settings of your font object. Without doing so, 398 your variable font's font object can still be used, but only at 399 the default setting for every axis (which, of course, is 400 sometimes what you want, but does not cover general usage). 401 </para> 402 <para> 403 HarfBuzz manages variation settings in the 404 <type>hb_variation_t</type> data type, which holds a <property>tag</property> for the 405 variation-axis identifier tag and a <property>value</property> for its 406 setting. You can retrieve the list of variation axes in a font 407 binary from the face object (not from a font object, notably) by 408 calling <function>hb_ot_var_get_axis_count(face)</function> to 409 find the number of axes, then using 410 <function>hb_ot_var_get_axis_infos()</function> to collect the 411 axis structures: 412 </para> 413 <programlisting language="C"> 414 axes = hb_ot_var_get_axis_count(face); 415 ... 416 hb_ot_var_get_axis_infos(face, 0, axes, axes_array); 417 </programlisting> 418 <para> 419 For each axis returned in the array, you can can access the 420 identifier in its <property>tag</property>. HarfBuzz also has 421 tag definitions predefined for the five standard axes specified 422 in OpenType (<literal>ital</literal> for italic, 423 <literal>opsz</literal> for optical size, 424 <literal>slnt</literal> for slant, <literal>wdth</literal> for 425 width, and <literal>wght</literal> for weight). Each axis also 426 has a <property>min_value</property>, a 427 <property>default_value</property>, and a <property>max_value</property>. 428 </para> 429 <para> 430 To set your font object's variation settings, you call the 431 <function>hb_font_set_variations()</function> function with an 432 array of <type>hb_variation_t</type> variation settings. Let's 433 say our font has weight and width axes. We need to specify each 434 of the axes by tag and assign a value on the axis: 435 </para> 436 <programlisting language="C"> 437 unsigned int variation_count = 2; 438 hb_variation_t variation_data[variation_count]; 439 variation_data[0].tag = HB_OT_TAG_VAR_AXIS_WIDTH; 440 variation_data[1].tag = HB_OT_TAG_VAR_AXIS_WEIGHT; 441 variation_data[0].value = 80; 442 variation_data[1].value = 750; 443 ... 444 hb_font_set_variations(font, variation_data, variation_count); 445 </programlisting> 446 <para> 447 That should give us a slightly condensed font ("normal" on the 448 <literal>wdth</literal> axis is 100) at a noticeably bolder 449 weight ("regular" is 400 on the <literal>wght</literal> axis). 450 </para> 451 <para> 452 In practice, though, you should always check that the value you 453 want to set on the axis is within the 454 [<property>min_value</property>,<property>max_value</property>] 455 range actually implemented in the font's variation axis. After 456 all, a font might only provide lighter-than-regular weights, and 457 setting a heavier value on the <literal>wght</literal> axis will 458 not change that. 459 </para> 460 <para> 461 Once your variation settings are specified on your font object, 462 however, shaping with a variable font is just like shaping a 463 static font. 464 </para> 465 </section> 466 467 </chapter> 468