harfbuzz/src/hb-aat-layout-trak-table.hh

237 lines
6.8 KiB
C++
Raw Normal View History

2018-02-19 20:57:08 +01:00
/*
* Copyright © 2018 Ebrahim Byagowi
2018-02-24 10:19:42 +01:00
* Copyright © 2018 Google, Inc.
2018-02-19 20:57:08 +01:00
*
* 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
#include "hb-aat-layout-common-private.hh"
#include "hb-open-type-private.hh"
#define HB_AAT_TAG_trak HB_TAG('t','r','a','k')
namespace AAT {
struct TrackTableEntry
{
2018-02-25 10:00:33 +01:00
inline bool sanitize (hb_sanitize_context_t *c, const void *base, uint16_t size) const
2018-02-19 20:57:08 +01:00
{
TRACE_SANITIZE (this);
2018-02-25 10:00:33 +01:00
return_trace (c->check_struct (this) && ((base+values).sanitize (c, size)));
}
inline Fixed get_track_value () const
{
return track;
2018-02-19 20:57:08 +01:00
}
2018-02-24 10:19:42 +01:00
inline float get_value (const void *base, unsigned int index) const
{
return (base+values)[index];
}
2018-02-19 20:57:08 +01:00
protected:
2018-02-21 00:02:25 +01:00
Fixed track; /* Track value for this record. */
HBUINT16 trackNameID; /* The 'name' table index for this track */
2018-02-24 10:19:42 +01:00
OffsetTo<UnsizedArrayOf<HBINT16> >
2018-02-21 00:02:25 +01:00
values; /* Offset from start of tracking table to
* per-size tracking values for this track. */
2018-02-19 20:57:08 +01:00
public:
2018-02-21 00:02:25 +01:00
DEFINE_SIZE_STATIC (8);
2018-02-19 20:57:08 +01:00
};
struct TrackData
{
2018-02-25 10:00:33 +01:00
inline bool sanitize (hb_sanitize_context_t *c, const void *base) const
2018-02-19 20:57:08 +01:00
{
TRACE_SANITIZE (this);
2018-02-25 10:00:33 +01:00
if (!(c->check_struct (this)))
return_trace (false);
uint16_t tracks = (uint16_t) nTracks;
uint16_t sizes = (uint16_t) nSizes;
// It should have at least one track
if (tracks < 1) return_trace (false);
// We can not do interpolation with less than two
if (sizes < 2) return_trace (false);
if (!((base+sizeTable).sanitize (c, sizes)))
return_trace (false);
for (uint16_t i = 0; i < tracks; ++i)
if (!(trackTable[i].sanitize (c, base, sizes)))
return_trace (false);
return_trace (true);
2018-02-19 20:57:08 +01:00
}
2018-02-24 10:19:42 +01:00
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);
/* TODO Clean this up. */
2018-02-25 10:00:33 +01:00
uint16_t tracks = (uint16_t) nTracks;
uint16_t sizes = (uint16_t) nSizes;
const TrackTableEntry *trackTableEntry = nullptr;
for (unsigned int i = 0; i < sizes; ++i)
// For now we only seek for track entries with zero tracking value
if (trackTable[i].get_track_value () == 0)
trackTableEntry = &trackTable[0];
// We couldn't match any, exit
if (!trackTableEntry) return 0.;
2018-02-24 10:19:42 +01:00
/* TODO bfind() */
2018-02-24 10:19:42 +01:00
unsigned int size_index;
2018-02-25 10:00:33 +01:00
UnsizedArrayOf<Fixed> size_table = base+sizeTable;
for (size_index = 0; size_index < sizes; ++size_index)
if (size_table[size_index] >= fixed_size)
2018-02-24 10:19:42 +01:00
break;
2018-02-25 10:00:33 +01:00
// 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);
2018-02-24 10:19:42 +01:00
2018-02-25 10:00:33 +01:00
float s0 = size_table[size_index - 1].to_float ();
float s1 = size_table[size_index].to_float ();
2018-02-24 10:19:42 +01:00
float t = (csspx - s0) / (s1 - s0);
2018-02-25 10:00:33 +01:00
return t * trackTableEntry->get_value (base, size_index) +
(1.0 - t) * trackTableEntry->get_value (base, size_index - 1);
2018-02-24 10:19:42 +01:00
}
2018-02-19 20:57:08 +01:00
protected:
2018-02-21 00:02:25 +01:00
HBUINT16 nTracks; /* Number of separate tracks included in this table. */
HBUINT16 nSizes; /* Number of point sizes included in this table. */
LOffsetTo<UnsizedArrayOf<Fixed> >
sizeTable;
2018-02-24 10:19:42 +01:00
UnsizedArrayOf<TrackTableEntry> trackTable;/* Array[nSizes] of size values. */
2018-02-19 20:57:08 +01:00
public:
2018-02-21 00:02:25 +01:00
DEFINE_SIZE_ARRAY (8, trackTable);
2018-02-19 20:57:08 +01:00
};
struct trak
{
static const hb_tag_t tableTag = HB_AAT_TAG_trak;
inline bool sanitize (hb_sanitize_context_t *c) const
{
TRACE_SANITIZE (this);
2018-02-25 10:00:33 +01:00
if (!(c->check_struct (this)))
return_trace (false);
if ((format != 0) || (reserved != 0))
return_trace (false);
if (horizData)
{
const TrackData &trackData = this+horizData;
if (!trackData.sanitize (c, this))
return_trace (false);
}
if (vertData)
{
const TrackData &trackData = this+horizData;
if (!trackData.sanitize (c, this))
return_trace (false);
}
return_trace (true);
2018-02-19 20:57:08 +01:00
}
2018-02-24 10:19:42 +01:00
inline bool apply (hb_aat_apply_context_t *c) const
{
TRACE_APPLY (this);
2018-02-24 10:19:42 +01:00
const float ptem = c->font->ptem;
if (ptem <= 0.f)
return_trace (false);
hb_buffer_t *buffer = c->buffer;
if (HB_DIRECTION_IS_HORIZONTAL (buffer->props.direction))
2018-02-24 10:19:42 +01:00
{
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)
2018-02-24 10:19:42 +01:00
{
/* TODO This is wrong. */
buffer->pos[start].x_advance += advance_to_add;
buffer->pos[end].x_advance += advance_to_add;
2018-02-24 10:19:42 +01:00
}
}
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)
2018-02-24 10:19:42 +01:00
{
/* TODO This is wrong. */
buffer->pos[start].y_advance += advance_to_add;
buffer->pos[end].y_advance += advance_to_add;
2018-02-24 10:19:42 +01:00
}
}
return_trace (true);
2018-02-24 10:19:42 +01:00
}
2018-02-19 20:57:08 +01:00
protected:
2018-02-21 00:02:25 +01:00
FixedVersion<> version; /* Version of the tracking table--currently
2018-02-19 20:57:08 +01:00
* 0x00010000u for version 1.0. */
2018-02-21 00:02:25 +01:00
HBUINT16 format; /* Format of the tracking table */
OffsetTo<TrackData> horizData; /* TrackData for horizontal text */
OffsetTo<TrackData> vertData; /* TrackData for vertical text */
2018-02-21 00:02:25 +01:00
HBUINT16 reserved; /* Reserved. Set to 0. */
2018-02-19 20:57:08 +01:00
public:
DEFINE_SIZE_MIN (12);
};
} /* namespace AAT */
#endif /* HB_AAT_LAYOUT_TRAK_TABLE_HH */