xref: /aosp_15_r20/external/harfbuzz_ng/src/hb-style.cc (revision 2d1272b857b1f7575e6e246373e1cb218663db8a)
1*2d1272b8SAndroid Build Coastguard Worker /*
2*2d1272b8SAndroid Build Coastguard Worker  * Copyright © 2019  Ebrahim Byagowi
3*2d1272b8SAndroid Build Coastguard Worker  *
4*2d1272b8SAndroid Build Coastguard Worker  *  This is part of HarfBuzz, a text shaping library.
5*2d1272b8SAndroid Build Coastguard Worker  *
6*2d1272b8SAndroid Build Coastguard Worker  * Permission is hereby granted, without written agreement and without
7*2d1272b8SAndroid Build Coastguard Worker  * license or royalty fees, to use, copy, modify, and distribute this
8*2d1272b8SAndroid Build Coastguard Worker  * software and its documentation for any purpose, provided that the
9*2d1272b8SAndroid Build Coastguard Worker  * above copyright notice and the following two paragraphs appear in
10*2d1272b8SAndroid Build Coastguard Worker  * all copies of this software.
11*2d1272b8SAndroid Build Coastguard Worker  *
12*2d1272b8SAndroid Build Coastguard Worker  * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13*2d1272b8SAndroid Build Coastguard Worker  * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14*2d1272b8SAndroid Build Coastguard Worker  * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15*2d1272b8SAndroid Build Coastguard Worker  * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16*2d1272b8SAndroid Build Coastguard Worker  * DAMAGE.
17*2d1272b8SAndroid Build Coastguard Worker  *
18*2d1272b8SAndroid Build Coastguard Worker  * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19*2d1272b8SAndroid Build Coastguard Worker  * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20*2d1272b8SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
21*2d1272b8SAndroid Build Coastguard Worker  * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22*2d1272b8SAndroid Build Coastguard Worker  * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23*2d1272b8SAndroid Build Coastguard Worker  */
24*2d1272b8SAndroid Build Coastguard Worker 
25*2d1272b8SAndroid Build Coastguard Worker #include "hb.hh"
26*2d1272b8SAndroid Build Coastguard Worker 
27*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_STYLE
28*2d1272b8SAndroid Build Coastguard Worker 
29*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-avar-table.hh"
30*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-fvar-table.hh"
31*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-stat-table.hh"
32*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-os2-table.hh"
33*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-head-table.hh"
34*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-post-table.hh"
35*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-face.hh"
36*2d1272b8SAndroid Build Coastguard Worker 
37*2d1272b8SAndroid Build Coastguard Worker /**
38*2d1272b8SAndroid Build Coastguard Worker  * SECTION:hb-style
39*2d1272b8SAndroid Build Coastguard Worker  * @title: hb-style
40*2d1272b8SAndroid Build Coastguard Worker  * @short_description: Font Styles
41*2d1272b8SAndroid Build Coastguard Worker  * @include: hb.h
42*2d1272b8SAndroid Build Coastguard Worker  *
43*2d1272b8SAndroid Build Coastguard Worker  * Functions for fetching style information from fonts.
44*2d1272b8SAndroid Build Coastguard Worker  **/
45*2d1272b8SAndroid Build Coastguard Worker 
46*2d1272b8SAndroid Build Coastguard Worker static inline float
_hb_angle_to_ratio(float a)47*2d1272b8SAndroid Build Coastguard Worker _hb_angle_to_ratio (float a)
48*2d1272b8SAndroid Build Coastguard Worker {
49*2d1272b8SAndroid Build Coastguard Worker   return tanf (a * -HB_PI / 180.f);
50*2d1272b8SAndroid Build Coastguard Worker }
51*2d1272b8SAndroid Build Coastguard Worker 
52*2d1272b8SAndroid Build Coastguard Worker static inline float
_hb_ratio_to_angle(float r)53*2d1272b8SAndroid Build Coastguard Worker _hb_ratio_to_angle (float r)
54*2d1272b8SAndroid Build Coastguard Worker {
55*2d1272b8SAndroid Build Coastguard Worker   return atanf (r) * -180.f / HB_PI;
56*2d1272b8SAndroid Build Coastguard Worker }
57*2d1272b8SAndroid Build Coastguard Worker 
58*2d1272b8SAndroid Build Coastguard Worker /**
59*2d1272b8SAndroid Build Coastguard Worker  * hb_style_get_value:
60*2d1272b8SAndroid Build Coastguard Worker  * @font: a #hb_font_t object.
61*2d1272b8SAndroid Build Coastguard Worker  * @style_tag: a style tag.
62*2d1272b8SAndroid Build Coastguard Worker  *
63*2d1272b8SAndroid Build Coastguard Worker  * Searches variation axes of a #hb_font_t object for a specific axis first,
64*2d1272b8SAndroid Build Coastguard Worker  * if not set, first tries to get default style values in `STAT` table
65*2d1272b8SAndroid Build Coastguard Worker  * then tries to polyfill from different tables of the font.
66*2d1272b8SAndroid Build Coastguard Worker  *
67*2d1272b8SAndroid Build Coastguard Worker  * Returns: Corresponding axis or default value to a style tag.
68*2d1272b8SAndroid Build Coastguard Worker  *
69*2d1272b8SAndroid Build Coastguard Worker  * Since: 3.0.0
70*2d1272b8SAndroid Build Coastguard Worker  **/
71*2d1272b8SAndroid Build Coastguard Worker float
hb_style_get_value(hb_font_t * font,hb_style_tag_t style_tag)72*2d1272b8SAndroid Build Coastguard Worker hb_style_get_value (hb_font_t *font, hb_style_tag_t style_tag)
73*2d1272b8SAndroid Build Coastguard Worker {
74*2d1272b8SAndroid Build Coastguard Worker   if (unlikely (style_tag == HB_STYLE_TAG_SLANT_RATIO))
75*2d1272b8SAndroid Build Coastguard Worker     return _hb_angle_to_ratio (hb_style_get_value (font, HB_STYLE_TAG_SLANT_ANGLE));
76*2d1272b8SAndroid Build Coastguard Worker 
77*2d1272b8SAndroid Build Coastguard Worker   hb_face_t *face = font->face;
78*2d1272b8SAndroid Build Coastguard Worker 
79*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_VAR
80*2d1272b8SAndroid Build Coastguard Worker   hb_ot_var_axis_info_t axis;
81*2d1272b8SAndroid Build Coastguard Worker   if (hb_ot_var_find_axis_info (face, style_tag, &axis))
82*2d1272b8SAndroid Build Coastguard Worker   {
83*2d1272b8SAndroid Build Coastguard Worker     if (axis.axis_index < font->num_coords) return font->design_coords[axis.axis_index];
84*2d1272b8SAndroid Build Coastguard Worker     /* If a face is variable, fvar's default_value is better than STAT records */
85*2d1272b8SAndroid Build Coastguard Worker     return axis.default_value;
86*2d1272b8SAndroid Build Coastguard Worker   }
87*2d1272b8SAndroid Build Coastguard Worker #endif
88*2d1272b8SAndroid Build Coastguard Worker 
89*2d1272b8SAndroid Build Coastguard Worker   if (style_tag == HB_STYLE_TAG_OPTICAL_SIZE && font->ptem)
90*2d1272b8SAndroid Build Coastguard Worker     return font->ptem;
91*2d1272b8SAndroid Build Coastguard Worker 
92*2d1272b8SAndroid Build Coastguard Worker   /* STAT */
93*2d1272b8SAndroid Build Coastguard Worker   float value;
94*2d1272b8SAndroid Build Coastguard Worker   if (face->table.STAT->get_value (style_tag, &value))
95*2d1272b8SAndroid Build Coastguard Worker     return value;
96*2d1272b8SAndroid Build Coastguard Worker 
97*2d1272b8SAndroid Build Coastguard Worker   switch ((unsigned) style_tag)
98*2d1272b8SAndroid Build Coastguard Worker   {
99*2d1272b8SAndroid Build Coastguard Worker   case HB_STYLE_TAG_ITALIC:
100*2d1272b8SAndroid Build Coastguard Worker     return face->table.OS2->is_italic () || face->table.head->is_italic () ? 1 : 0;
101*2d1272b8SAndroid Build Coastguard Worker   case HB_STYLE_TAG_OPTICAL_SIZE:
102*2d1272b8SAndroid Build Coastguard Worker   {
103*2d1272b8SAndroid Build Coastguard Worker     unsigned int lower, design, upper;
104*2d1272b8SAndroid Build Coastguard Worker     return face->table.OS2->v5 ().get_optical_size (&lower, &upper)
105*2d1272b8SAndroid Build Coastguard Worker 	   ? (float) (lower + upper) / 2.f
106*2d1272b8SAndroid Build Coastguard Worker 	   : hb_ot_layout_get_size_params (face, &design, nullptr, nullptr, nullptr, nullptr)
107*2d1272b8SAndroid Build Coastguard Worker 	   ? design / 10.f
108*2d1272b8SAndroid Build Coastguard Worker 	   : 12.f;
109*2d1272b8SAndroid Build Coastguard Worker   }
110*2d1272b8SAndroid Build Coastguard Worker   case HB_STYLE_TAG_SLANT_ANGLE:
111*2d1272b8SAndroid Build Coastguard Worker   {
112*2d1272b8SAndroid Build Coastguard Worker     float angle = face->table.post->table->italicAngle.to_float ();
113*2d1272b8SAndroid Build Coastguard Worker 
114*2d1272b8SAndroid Build Coastguard Worker     if (font->slant)
115*2d1272b8SAndroid Build Coastguard Worker       angle = _hb_ratio_to_angle (font->slant + _hb_angle_to_ratio (angle));
116*2d1272b8SAndroid Build Coastguard Worker 
117*2d1272b8SAndroid Build Coastguard Worker     return angle;
118*2d1272b8SAndroid Build Coastguard Worker   }
119*2d1272b8SAndroid Build Coastguard Worker   case HB_STYLE_TAG_WIDTH:
120*2d1272b8SAndroid Build Coastguard Worker     return face->table.OS2->has_data ()
121*2d1272b8SAndroid Build Coastguard Worker 	   ? face->table.OS2->get_width ()
122*2d1272b8SAndroid Build Coastguard Worker 	   : (face->table.head->is_condensed () ? 75 :
123*2d1272b8SAndroid Build Coastguard Worker 	      face->table.head->is_expanded () ? 125 :
124*2d1272b8SAndroid Build Coastguard Worker 	      100);
125*2d1272b8SAndroid Build Coastguard Worker   case HB_STYLE_TAG_WEIGHT:
126*2d1272b8SAndroid Build Coastguard Worker     return face->table.OS2->has_data ()
127*2d1272b8SAndroid Build Coastguard Worker 	   ? face->table.OS2->usWeightClass
128*2d1272b8SAndroid Build Coastguard Worker 	   : (face->table.head->is_bold () ? 700 : 400);
129*2d1272b8SAndroid Build Coastguard Worker   default:
130*2d1272b8SAndroid Build Coastguard Worker     return 0;
131*2d1272b8SAndroid Build Coastguard Worker   }
132*2d1272b8SAndroid Build Coastguard Worker }
133*2d1272b8SAndroid Build Coastguard Worker 
134*2d1272b8SAndroid Build Coastguard Worker #endif
135