hb-aat-layout-trak-table.hh 6.4 KB
Newer Older
E
Ebrahim Byagowi 已提交
1 2
/*
 * Copyright © 2018  Ebrahim Byagowi
3
 * Copyright © 2018  Google, Inc.
E
Ebrahim Byagowi 已提交
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
 *
 *  This is part of HarfBuzz, a text shaping library.
 *
 * Permission is hereby granted, without written agreement and without
 * license or royalty fees, to use, copy, modify, and distribute this
 * software and its documentation for any purpose, provided that the
 * above copyright notice and the following two paragraphs appear in
 * all copies of this software.
 *
 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
 * DAMAGE.
 *
 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
 *
 * Google Author(s): Behdad Esfahbod
 */

#ifndef HB_AAT_LAYOUT_TRAK_TABLE_HH
#define HB_AAT_LAYOUT_TRAK_TABLE_HH

31 32 33
#include "hb-aat-layout-common.hh"
#include "hb-ot-layout.hh"
#include "hb-open-type.hh"
E
Ebrahim Byagowi 已提交
34

35 36 37 38
/*
 * trak -- Tracking
 * https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6trak.html
 */
39
#define HB_AAT_TAG_trak HB_TAG('t','r','a','k')
E
Ebrahim Byagowi 已提交
40 41 42 43 44 45 46


namespace AAT {


struct TrackTableEntry
{
47 48 49 50
  friend struct TrackData;

  inline bool sanitize (hb_sanitize_context_t *c, const void *base,
			unsigned int size) const
E
Ebrahim Byagowi 已提交
51 52
  {
    TRACE_SANITIZE (this);
53 54
    return_trace (likely (c->check_struct (this) &&
			  (valuesZ.sanitize (c, base, size))));
E
Ebrahim Byagowi 已提交
55 56
  }

57
  private:
B
Behdad Esfahbod 已提交
58
  inline float get_track_value () const
E
Ebrahim Byagowi 已提交
59
  {
B
Behdad Esfahbod 已提交
60
    return track.to_float ();
E
Ebrahim Byagowi 已提交
61 62
  }

B
Behdad Esfahbod 已提交
63
  inline int get_value (const void *base, unsigned int index) const
64
  {
65
    return (base+valuesZ)[index];
66 67
  }

E
Ebrahim Byagowi 已提交
68
  protected:
B
Behdad Esfahbod 已提交
69 70
  Fixed		track;		/* Track value for this record. */
  NameID	trackNameID;	/* The 'name' table index for this track */
B
Behdad Esfahbod 已提交
71
  OffsetTo<UnsizedArrayOf<FWORD> >
72
		valuesZ;	/* Offset from start of tracking table to
B
Behdad Esfahbod 已提交
73
				 * per-size tracking values for this track. */
E
Ebrahim Byagowi 已提交
74 75

  public:
76
  DEFINE_SIZE_STATIC (8);
E
Ebrahim Byagowi 已提交
77 78 79 80
};

struct TrackData
{
E
Ebrahim Byagowi 已提交
81
  inline bool sanitize (hb_sanitize_context_t *c, const void *base) const
E
Ebrahim Byagowi 已提交
82 83
  {
    TRACE_SANITIZE (this);
B
Behdad Esfahbod 已提交
84 85 86
    return_trace (c->check_struct (this) &&
		  sizeTable.sanitize (c, base, nSizes) &&
		  trackTable.sanitize (c, nTracks, base, nSizes));
E
Ebrahim Byagowi 已提交
87 88
  }

89 90 91 92 93 94 95 96 97 98 99
  inline float get_tracking (const void *base, float ptem) const
  {
    /* CoreText points are CSS pixels (96 per inch),
     * NOT typographic points (72 per inch).
     *
     * https://developer.apple.com/library/content/documentation/GraphicsAnimation/Conceptual/HighResolutionOSX/Explained/Explained.html
     */
    float csspx = ptem * 96.f / 72.f;
    Fixed fixed_size;
    fixed_size.set_float (csspx);

B
Behdad Esfahbod 已提交
100
    /* XXX Clean this up. Make it work with nSizes==1 and 0. */
B
Behdad Esfahbod 已提交
101

B
Behdad Esfahbod 已提交
102
    unsigned int sizes = nSizes;
E
Ebrahim Byagowi 已提交
103 104 105 106

    const TrackTableEntry *trackTableEntry = nullptr;
    for (unsigned int i = 0; i < sizes; ++i)
      // For now we only seek for track entries with zero tracking value
107
      if (trackTable[i].get_track_value () == 0.f)
E
Ebrahim Byagowi 已提交
108 109 110 111
        trackTableEntry = &trackTable[0];

    // We couldn't match any, exit
    if (!trackTableEntry) return 0.;
112

B
Behdad Esfahbod 已提交
113
    /* TODO bfind() */
114
    unsigned int size_index;
E
Ebrahim Byagowi 已提交
115 116 117
    UnsizedArrayOf<Fixed> size_table = base+sizeTable;
    for (size_index = 0; size_index < sizes; ++size_index)
      if (size_table[size_index] >= fixed_size)
118 119
        break;

E
Ebrahim Byagowi 已提交
120 121 122 123 124 125
    // TODO(ebraminio): We don't attempt to extrapolate to larger or
    // smaller values for now but we should do, per spec
    if (size_index == sizes)
      return trackTableEntry->get_value (base, sizes - 1);
    if (size_index == 0 || size_table[size_index] == fixed_size)
      return trackTableEntry->get_value (base, size_index);
126

E
Ebrahim Byagowi 已提交
127 128
    float s0 = size_table[size_index - 1].to_float ();
    float s1 = size_table[size_index].to_float ();
129
    float t = (csspx - s0) / (s1 - s0);
130 131
    return (float) t * trackTableEntry->get_value (base, size_index) +
	   ((float) 1.0 - t) * trackTableEntry->get_value (base, size_index - 1);
132 133
  }

E
Ebrahim Byagowi 已提交
134
  protected:
E
Ebrahim Byagowi 已提交
135 136 137 138
  HBUINT16	nTracks;	/* Number of separate tracks included in this table. */
  HBUINT16	nSizes;		/* Number of point sizes included in this table. */
  LOffsetTo<UnsizedArrayOf<Fixed> >
		sizeTable;	/* Offset to array[nSizes] of size values. */
B
Behdad Esfahbod 已提交
139
  UnsizedArrayOf<TrackTableEntry>
E
Ebrahim Byagowi 已提交
140
		trackTable;	/* Array[nTracks] of TrackTableEntry records. */
E
Ebrahim Byagowi 已提交
141 142

  public:
143
  DEFINE_SIZE_ARRAY (8, trackTable);
E
Ebrahim Byagowi 已提交
144 145 146 147
};

struct trak
{
148
  static const hb_tag_t tableTag = HB_AAT_TAG_trak;
E
Ebrahim Byagowi 已提交
149 150 151 152

  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
E
Ebrahim Byagowi 已提交
153

154 155 156
    return_trace (unlikely (c->check_struct (this) &&
			    horizData.sanitize (c, this, this) &&
			    vertData.sanitize (c, this, this)));
E
Ebrahim Byagowi 已提交
157 158
  }

159 160 161
  inline bool apply (hb_aat_apply_context_t *c) const
  {
    TRACE_APPLY (this);
B
Behdad Esfahbod 已提交
162

163
    const float ptem = c->font->ptem;
164
    if (unlikely (ptem <= 0.f))
B
Behdad Esfahbod 已提交
165 166 167 168
      return_trace (false);

    hb_buffer_t *buffer = c->buffer;
    if (HB_DIRECTION_IS_HORIZONTAL (buffer->props.direction))
169
    {
B
Behdad Esfahbod 已提交
170 171 172 173
      const TrackData &trackData = this+horizData;
      float tracking = trackData.get_tracking (this, ptem);
      hb_position_t advance_to_add = c->font->em_scalef_x (tracking / 2);
      foreach_grapheme (buffer, start, end)
174
      {
E
Ebrahim Byagowi 已提交
175
	buffer->pos[start].x_offset += advance_to_add;
B
Behdad Esfahbod 已提交
176 177
	buffer->pos[start].x_advance += advance_to_add;
	buffer->pos[end].x_advance += advance_to_add;
178
      }
B
Behdad Esfahbod 已提交
179 180 181 182 183 184 185
    }
    else
    {
      const TrackData &trackData = this+vertData;
      float tracking = trackData.get_tracking (this, ptem);
      hb_position_t advance_to_add = c->font->em_scalef_y (tracking / 2);
      foreach_grapheme (buffer, start, end)
186
      {
E
Ebrahim Byagowi 已提交
187
	buffer->pos[start].y_offset += advance_to_add;
B
Behdad Esfahbod 已提交
188 189
	buffer->pos[start].y_advance += advance_to_add;
	buffer->pos[end].y_advance += advance_to_add;
190 191
      }
    }
B
Behdad Esfahbod 已提交
192 193

    return_trace (true);
194 195
  }

E
Ebrahim Byagowi 已提交
196
  protected:
197
  FixedVersion<>	version;	/* Version of the tracking table--currently
E
Ebrahim Byagowi 已提交
198
					 * 0x00010000u for version 1.0. */
199
  HBUINT16		format; 	/* Format of the tracking table */
B
Behdad Esfahbod 已提交
200 201
  OffsetTo<TrackData>	horizData;	/* TrackData for horizontal text */
  OffsetTo<TrackData>	vertData;	/* TrackData for vertical text */
202
  HBUINT16		reserved;	/* Reserved. Set to 0. */
E
Ebrahim Byagowi 已提交
203 204 205 206 207 208 209 210 211

  public:
  DEFINE_SIZE_MIN (12);
};

} /* namespace AAT */


#endif /* HB_AAT_LAYOUT_TRAK_TABLE_HH */