1058 lines
34 KiB
C++
1058 lines
34 KiB
C++
/*
|
|
* Copyright © 2016 Igalia S.L.
|
|
*
|
|
* 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.
|
|
*
|
|
* Igalia Author(s): Frédéric Wang
|
|
*/
|
|
|
|
#ifndef HB_OT_MATH_TABLE_HH
|
|
#define HB_OT_MATH_TABLE_HH
|
|
|
|
#include "hb-open-type.hh"
|
|
#include "hb-ot-layout-common.hh"
|
|
#include "hb-ot-math.h"
|
|
|
|
namespace OT {
|
|
|
|
|
|
struct MathValueRecord
|
|
{
|
|
hb_position_t get_x_value (hb_font_t *font, const void *base) const
|
|
{ return font->em_scale_x (value) + (base+deviceTable).get_x_delta (font); }
|
|
hb_position_t get_y_value (hb_font_t *font, const void *base) const
|
|
{ return font->em_scale_y (value) + (base+deviceTable).get_y_delta (font); }
|
|
|
|
MathValueRecord* copy (hb_serialize_context_t *c, const void *base) const
|
|
{
|
|
TRACE_SERIALIZE (this);
|
|
auto *out = c->embed (this);
|
|
if (unlikely (!out)) return_trace (nullptr);
|
|
out->deviceTable.serialize_copy (c, deviceTable, base, 0, hb_serialize_context_t::Head);
|
|
|
|
return_trace (out);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c, const void *base) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) && deviceTable.sanitize (c, base));
|
|
}
|
|
|
|
protected:
|
|
HBINT16 value; /* The X or Y value in design units */
|
|
Offset16To<Device> deviceTable; /* Offset to the device table - from the
|
|
* beginning of parent table. May be NULL.
|
|
* Suggested format for device table is 1. */
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (4);
|
|
};
|
|
|
|
struct MathConstants
|
|
{
|
|
MathConstants* copy (hb_serialize_context_t *c) const
|
|
{
|
|
TRACE_SERIALIZE (this);
|
|
auto *out = c->start_embed (this);
|
|
if (unlikely (!out)) return_trace (nullptr);
|
|
|
|
HBINT16 *p = c->allocate_size<HBINT16> (HBINT16::static_size * 2);
|
|
if (unlikely (!p)) return_trace (nullptr);
|
|
memcpy (p, percentScaleDown, HBINT16::static_size * 2);
|
|
|
|
HBUINT16 *m = c->allocate_size<HBUINT16> (HBUINT16::static_size * 2);
|
|
if (unlikely (!m)) return_trace (nullptr);
|
|
memcpy (m, minHeight, HBUINT16::static_size * 2);
|
|
|
|
unsigned count = ARRAY_LENGTH (mathValueRecords);
|
|
for (unsigned i = 0; i < count; i++)
|
|
if (!c->copy (mathValueRecords[i], this))
|
|
return_trace (nullptr);
|
|
|
|
if (!c->embed (radicalDegreeBottomRaisePercent)) return_trace (nullptr);
|
|
return_trace (out);
|
|
}
|
|
|
|
bool sanitize_math_value_records (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
|
|
unsigned int count = ARRAY_LENGTH (mathValueRecords);
|
|
for (unsigned int i = 0; i < count; i++)
|
|
if (!mathValueRecords[i].sanitize (c, this))
|
|
return_trace (false);
|
|
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) && sanitize_math_value_records (c));
|
|
}
|
|
|
|
hb_position_t get_value (hb_ot_math_constant_t constant,
|
|
hb_font_t *font) const
|
|
{
|
|
switch (constant) {
|
|
|
|
case HB_OT_MATH_CONSTANT_SCRIPT_PERCENT_SCALE_DOWN:
|
|
case HB_OT_MATH_CONSTANT_SCRIPT_SCRIPT_PERCENT_SCALE_DOWN:
|
|
return percentScaleDown[constant - HB_OT_MATH_CONSTANT_SCRIPT_PERCENT_SCALE_DOWN];
|
|
|
|
case HB_OT_MATH_CONSTANT_DELIMITED_SUB_FORMULA_MIN_HEIGHT:
|
|
case HB_OT_MATH_CONSTANT_DISPLAY_OPERATOR_MIN_HEIGHT:
|
|
return font->em_scale_y (minHeight[constant - HB_OT_MATH_CONSTANT_DELIMITED_SUB_FORMULA_MIN_HEIGHT]);
|
|
|
|
case HB_OT_MATH_CONSTANT_RADICAL_KERN_AFTER_DEGREE:
|
|
case HB_OT_MATH_CONSTANT_RADICAL_KERN_BEFORE_DEGREE:
|
|
case HB_OT_MATH_CONSTANT_SKEWED_FRACTION_HORIZONTAL_GAP:
|
|
case HB_OT_MATH_CONSTANT_SPACE_AFTER_SCRIPT:
|
|
return mathValueRecords[constant - HB_OT_MATH_CONSTANT_MATH_LEADING].get_x_value (font, this);
|
|
|
|
case HB_OT_MATH_CONSTANT_ACCENT_BASE_HEIGHT:
|
|
case HB_OT_MATH_CONSTANT_AXIS_HEIGHT:
|
|
case HB_OT_MATH_CONSTANT_FLATTENED_ACCENT_BASE_HEIGHT:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_DENOMINATOR_DISPLAY_STYLE_SHIFT_DOWN:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_DENOMINATOR_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_DENOMINATOR_SHIFT_DOWN:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_DENOM_DISPLAY_STYLE_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_NUMERATOR_DISPLAY_STYLE_SHIFT_UP:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_NUMERATOR_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_NUMERATOR_SHIFT_UP:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_NUM_DISPLAY_STYLE_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_FRACTION_RULE_THICKNESS:
|
|
case HB_OT_MATH_CONSTANT_LOWER_LIMIT_BASELINE_DROP_MIN:
|
|
case HB_OT_MATH_CONSTANT_LOWER_LIMIT_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_MATH_LEADING:
|
|
case HB_OT_MATH_CONSTANT_OVERBAR_EXTRA_ASCENDER:
|
|
case HB_OT_MATH_CONSTANT_OVERBAR_RULE_THICKNESS:
|
|
case HB_OT_MATH_CONSTANT_OVERBAR_VERTICAL_GAP:
|
|
case HB_OT_MATH_CONSTANT_RADICAL_DISPLAY_STYLE_VERTICAL_GAP:
|
|
case HB_OT_MATH_CONSTANT_RADICAL_EXTRA_ASCENDER:
|
|
case HB_OT_MATH_CONSTANT_RADICAL_RULE_THICKNESS:
|
|
case HB_OT_MATH_CONSTANT_RADICAL_VERTICAL_GAP:
|
|
case HB_OT_MATH_CONSTANT_SKEWED_FRACTION_VERTICAL_GAP:
|
|
case HB_OT_MATH_CONSTANT_STACK_BOTTOM_DISPLAY_STYLE_SHIFT_DOWN:
|
|
case HB_OT_MATH_CONSTANT_STACK_BOTTOM_SHIFT_DOWN:
|
|
case HB_OT_MATH_CONSTANT_STACK_DISPLAY_STYLE_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_STACK_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_STACK_TOP_DISPLAY_STYLE_SHIFT_UP:
|
|
case HB_OT_MATH_CONSTANT_STACK_TOP_SHIFT_UP:
|
|
case HB_OT_MATH_CONSTANT_STRETCH_STACK_BOTTOM_SHIFT_DOWN:
|
|
case HB_OT_MATH_CONSTANT_STRETCH_STACK_GAP_ABOVE_MIN:
|
|
case HB_OT_MATH_CONSTANT_STRETCH_STACK_GAP_BELOW_MIN:
|
|
case HB_OT_MATH_CONSTANT_STRETCH_STACK_TOP_SHIFT_UP:
|
|
case HB_OT_MATH_CONSTANT_SUBSCRIPT_BASELINE_DROP_MIN:
|
|
case HB_OT_MATH_CONSTANT_SUBSCRIPT_SHIFT_DOWN:
|
|
case HB_OT_MATH_CONSTANT_SUBSCRIPT_TOP_MAX:
|
|
case HB_OT_MATH_CONSTANT_SUB_SUPERSCRIPT_GAP_MIN:
|
|
case HB_OT_MATH_CONSTANT_SUPERSCRIPT_BASELINE_DROP_MAX:
|
|
case HB_OT_MATH_CONSTANT_SUPERSCRIPT_BOTTOM_MAX_WITH_SUBSCRIPT:
|
|
case HB_OT_MATH_CONSTANT_SUPERSCRIPT_BOTTOM_MIN:
|
|
case HB_OT_MATH_CONSTANT_SUPERSCRIPT_SHIFT_UP:
|
|
case HB_OT_MATH_CONSTANT_SUPERSCRIPT_SHIFT_UP_CRAMPED:
|
|
case HB_OT_MATH_CONSTANT_UNDERBAR_EXTRA_DESCENDER:
|
|
case HB_OT_MATH_CONSTANT_UNDERBAR_RULE_THICKNESS:
|
|
case HB_OT_MATH_CONSTANT_UNDERBAR_VERTICAL_GAP:
|
|
case HB_OT_MATH_CONSTANT_UPPER_LIMIT_BASELINE_RISE_MIN:
|
|
case HB_OT_MATH_CONSTANT_UPPER_LIMIT_GAP_MIN:
|
|
return mathValueRecords[constant - HB_OT_MATH_CONSTANT_MATH_LEADING].get_y_value (font, this);
|
|
|
|
case HB_OT_MATH_CONSTANT_RADICAL_DEGREE_BOTTOM_RAISE_PERCENT:
|
|
return radicalDegreeBottomRaisePercent;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
protected:
|
|
HBINT16 percentScaleDown[2];
|
|
HBUINT16 minHeight[2];
|
|
MathValueRecord mathValueRecords[51];
|
|
HBINT16 radicalDegreeBottomRaisePercent;
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (214);
|
|
};
|
|
|
|
struct MathItalicsCorrectionInfo
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
const hb_set_t &glyphset = *c->plan->_glyphset_mathed;
|
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
|
|
|
auto *out = c->serializer->start_embed (*this);
|
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
|
|
|
hb_sorted_vector_t<hb_codepoint_t> new_coverage;
|
|
+ hb_zip (this+coverage, italicsCorrection)
|
|
| hb_filter (glyphset, hb_first)
|
|
| hb_filter (serialize_math_record_array (c->serializer, out->italicsCorrection, this), hb_second)
|
|
| hb_map (hb_first)
|
|
| hb_map (glyph_map)
|
|
| hb_sink (new_coverage)
|
|
;
|
|
|
|
out->coverage.serialize_serialize (c->serializer, new_coverage.iter ());
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
coverage.sanitize (c, this) &&
|
|
italicsCorrection.sanitize (c, this));
|
|
}
|
|
|
|
hb_position_t get_value (hb_codepoint_t glyph,
|
|
hb_font_t *font) const
|
|
{
|
|
unsigned int index = (this+coverage).get_coverage (glyph);
|
|
return italicsCorrection[index].get_x_value (font, this);
|
|
}
|
|
|
|
protected:
|
|
Offset16To<Coverage> coverage; /* Offset to Coverage table -
|
|
* from the beginning of
|
|
* MathItalicsCorrectionInfo
|
|
* table. */
|
|
Array16Of<MathValueRecord> italicsCorrection; /* Array of MathValueRecords
|
|
* defining italics correction
|
|
* values for each
|
|
* covered glyph. */
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (4, italicsCorrection);
|
|
};
|
|
|
|
struct MathTopAccentAttachment
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
const hb_set_t &glyphset = *c->plan->_glyphset_mathed;
|
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
|
|
|
auto *out = c->serializer->start_embed (*this);
|
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
|
|
|
hb_sorted_vector_t<hb_codepoint_t> new_coverage;
|
|
+ hb_zip (this+topAccentCoverage, topAccentAttachment)
|
|
| hb_filter (glyphset, hb_first)
|
|
| hb_filter (serialize_math_record_array (c->serializer, out->topAccentAttachment, this), hb_second)
|
|
| hb_map (hb_first)
|
|
| hb_map (glyph_map)
|
|
| hb_sink (new_coverage)
|
|
;
|
|
|
|
out->topAccentCoverage.serialize_serialize (c->serializer, new_coverage.iter ());
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
topAccentCoverage.sanitize (c, this) &&
|
|
topAccentAttachment.sanitize (c, this));
|
|
}
|
|
|
|
hb_position_t get_value (hb_codepoint_t glyph,
|
|
hb_font_t *font) const
|
|
{
|
|
unsigned int index = (this+topAccentCoverage).get_coverage (glyph);
|
|
if (index == NOT_COVERED)
|
|
return font->get_glyph_h_advance (glyph) / 2;
|
|
return topAccentAttachment[index].get_x_value (font, this);
|
|
}
|
|
|
|
protected:
|
|
Offset16To<Coverage> topAccentCoverage; /* Offset to Coverage table -
|
|
* from the beginning of
|
|
* MathTopAccentAttachment
|
|
* table. */
|
|
Array16Of<MathValueRecord> topAccentAttachment; /* Array of MathValueRecords
|
|
* defining top accent
|
|
* attachment points for each
|
|
* covered glyph. */
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (2 + 2, topAccentAttachment);
|
|
};
|
|
|
|
struct MathKern
|
|
{
|
|
MathKern* copy (hb_serialize_context_t *c) const
|
|
{
|
|
TRACE_SERIALIZE (this);
|
|
auto *out = c->start_embed (this);
|
|
if (unlikely (!out)) return_trace (nullptr);
|
|
|
|
if (unlikely (!c->embed (heightCount))) return_trace (nullptr);
|
|
|
|
unsigned count = 2 * heightCount + 1;
|
|
for (unsigned i = 0; i < count; i++)
|
|
if (!c->copy (mathValueRecordsZ.arrayZ[i], this))
|
|
return_trace (nullptr);
|
|
|
|
return_trace (out);
|
|
}
|
|
|
|
bool sanitize_math_value_records (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
unsigned int count = 2 * heightCount + 1;
|
|
for (unsigned int i = 0; i < count; i++)
|
|
if (!mathValueRecordsZ.arrayZ[i].sanitize (c, this)) return_trace (false);
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
c->check_array (mathValueRecordsZ.arrayZ, 2 * heightCount + 1) &&
|
|
sanitize_math_value_records (c));
|
|
}
|
|
|
|
hb_position_t get_value (hb_position_t correction_height, hb_font_t *font) const
|
|
{
|
|
const MathValueRecord* correctionHeight = mathValueRecordsZ.arrayZ;
|
|
const MathValueRecord* kernValue = mathValueRecordsZ.arrayZ + heightCount;
|
|
int sign = font->y_scale < 0 ? -1 : +1;
|
|
|
|
/* The description of the MathKern table is a ambiguous, but interpreting
|
|
* "between the two heights found at those indexes" for 0 < i < len as
|
|
*
|
|
* correctionHeight[i-1] < correction_height <= correctionHeight[i]
|
|
*
|
|
* makes the result consistent with the limit cases and we can just use the
|
|
* binary search algorithm of std::upper_bound:
|
|
*/
|
|
unsigned int i = 0;
|
|
unsigned int count = heightCount;
|
|
while (count > 0)
|
|
{
|
|
unsigned int half = count / 2;
|
|
hb_position_t height = correctionHeight[i + half].get_y_value (font, this);
|
|
if (sign * height < sign * correction_height)
|
|
{
|
|
i += half + 1;
|
|
count -= half + 1;
|
|
} else
|
|
count = half;
|
|
}
|
|
return kernValue[i].get_x_value (font, this);
|
|
}
|
|
|
|
protected:
|
|
HBUINT16 heightCount;
|
|
UnsizedArrayOf<MathValueRecord>
|
|
mathValueRecordsZ;
|
|
/* Array of correction heights at
|
|
* which the kern value changes.
|
|
* Sorted by the height value in
|
|
* design units (heightCount entries),
|
|
* Followed by:
|
|
* Array of kern values corresponding
|
|
* to heights. (heightCount+1 entries).
|
|
*/
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (2, mathValueRecordsZ);
|
|
};
|
|
|
|
struct MathKernInfoRecord
|
|
{
|
|
MathKernInfoRecord* copy (hb_serialize_context_t *c, const void *base) const
|
|
{
|
|
TRACE_SERIALIZE (this);
|
|
auto *out = c->embed (this);
|
|
if (unlikely (!out)) return_trace (nullptr);
|
|
|
|
unsigned count = ARRAY_LENGTH (mathKern);
|
|
for (unsigned i = 0; i < count; i++)
|
|
out->mathKern[i].serialize_copy (c, mathKern[i], base, 0, hb_serialize_context_t::Head);
|
|
|
|
return_trace (out);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c, const void *base) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
|
|
unsigned int count = ARRAY_LENGTH (mathKern);
|
|
for (unsigned int i = 0; i < count; i++)
|
|
if (unlikely (!mathKern[i].sanitize (c, base)))
|
|
return_trace (false);
|
|
|
|
return_trace (true);
|
|
}
|
|
|
|
hb_position_t get_kerning (hb_ot_math_kern_t kern,
|
|
hb_position_t correction_height,
|
|
hb_font_t *font,
|
|
const void *base) const
|
|
{
|
|
unsigned int idx = kern;
|
|
if (unlikely (idx >= ARRAY_LENGTH (mathKern))) return 0;
|
|
return (base+mathKern[idx]).get_value (correction_height, font);
|
|
}
|
|
|
|
protected:
|
|
/* Offset to MathKern table for each corner -
|
|
* from the beginning of MathKernInfo table. May be NULL. */
|
|
Offset16To<MathKern> mathKern[4];
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (8);
|
|
};
|
|
|
|
struct MathKernInfo
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
const hb_set_t &glyphset = *c->plan->_glyphset_mathed;
|
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
|
|
|
auto *out = c->serializer->start_embed (*this);
|
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
|
|
|
hb_sorted_vector_t<hb_codepoint_t> new_coverage;
|
|
+ hb_zip (this+mathKernCoverage, mathKernInfoRecords)
|
|
| hb_filter (glyphset, hb_first)
|
|
| hb_filter (serialize_math_record_array (c->serializer, out->mathKernInfoRecords, this), hb_second)
|
|
| hb_map (hb_first)
|
|
| hb_map (glyph_map)
|
|
| hb_sink (new_coverage)
|
|
;
|
|
|
|
out->mathKernCoverage.serialize_serialize (c->serializer, new_coverage.iter ());
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
mathKernCoverage.sanitize (c, this) &&
|
|
mathKernInfoRecords.sanitize (c, this));
|
|
}
|
|
|
|
hb_position_t get_kerning (hb_codepoint_t glyph,
|
|
hb_ot_math_kern_t kern,
|
|
hb_position_t correction_height,
|
|
hb_font_t *font) const
|
|
{
|
|
unsigned int index = (this+mathKernCoverage).get_coverage (glyph);
|
|
return mathKernInfoRecords[index].get_kerning (kern, correction_height, font, this);
|
|
}
|
|
|
|
protected:
|
|
Offset16To<Coverage>
|
|
mathKernCoverage;
|
|
/* Offset to Coverage table -
|
|
* from the beginning of the
|
|
* MathKernInfo table. */
|
|
Array16Of<MathKernInfoRecord>
|
|
mathKernInfoRecords;
|
|
/* Array of MathKernInfoRecords,
|
|
* per-glyph information for
|
|
* mathematical positioning
|
|
* of subscripts and
|
|
* superscripts. */
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (4, mathKernInfoRecords);
|
|
};
|
|
|
|
struct MathGlyphInfo
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
auto *out = c->serializer->embed (*this);
|
|
if (unlikely (!out)) return_trace (false);
|
|
|
|
out->mathItalicsCorrectionInfo.serialize_subset (c, mathItalicsCorrectionInfo, this);
|
|
out->mathTopAccentAttachment.serialize_subset (c, mathTopAccentAttachment, this);
|
|
|
|
const hb_set_t &glyphset = *c->plan->_glyphset_mathed;
|
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
|
|
|
auto it =
|
|
+ hb_iter (this+extendedShapeCoverage)
|
|
| hb_filter (glyphset)
|
|
| hb_map_retains_sorting (glyph_map)
|
|
;
|
|
|
|
out->extendedShapeCoverage.serialize_serialize (c->serializer, it);
|
|
|
|
out->mathKernInfo.serialize_subset (c, mathKernInfo, this);
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
mathItalicsCorrectionInfo.sanitize (c, this) &&
|
|
mathTopAccentAttachment.sanitize (c, this) &&
|
|
extendedShapeCoverage.sanitize (c, this) &&
|
|
mathKernInfo.sanitize (c, this));
|
|
}
|
|
|
|
hb_position_t
|
|
get_italics_correction (hb_codepoint_t glyph, hb_font_t *font) const
|
|
{ return (this+mathItalicsCorrectionInfo).get_value (glyph, font); }
|
|
|
|
hb_position_t
|
|
get_top_accent_attachment (hb_codepoint_t glyph, hb_font_t *font) const
|
|
{ return (this+mathTopAccentAttachment).get_value (glyph, font); }
|
|
|
|
bool is_extended_shape (hb_codepoint_t glyph) const
|
|
{ return (this+extendedShapeCoverage).get_coverage (glyph) != NOT_COVERED; }
|
|
|
|
hb_position_t get_kerning (hb_codepoint_t glyph,
|
|
hb_ot_math_kern_t kern,
|
|
hb_position_t correction_height,
|
|
hb_font_t *font) const
|
|
{ return (this+mathKernInfo).get_kerning (glyph, kern, correction_height, font); }
|
|
|
|
protected:
|
|
/* Offset to MathItalicsCorrectionInfo table -
|
|
* from the beginning of MathGlyphInfo table. */
|
|
Offset16To<MathItalicsCorrectionInfo> mathItalicsCorrectionInfo;
|
|
|
|
/* Offset to MathTopAccentAttachment table -
|
|
* from the beginning of MathGlyphInfo table. */
|
|
Offset16To<MathTopAccentAttachment> mathTopAccentAttachment;
|
|
|
|
/* Offset to coverage table for Extended Shape glyphs -
|
|
* from the beginning of MathGlyphInfo table. When the left or right glyph of
|
|
* a box is an extended shape variant, the (ink) box (and not the default
|
|
* position defined by values in MathConstants table) should be used for
|
|
* vertical positioning purposes. May be NULL.. */
|
|
Offset16To<Coverage> extendedShapeCoverage;
|
|
|
|
/* Offset to MathKernInfo table -
|
|
* from the beginning of MathGlyphInfo table. */
|
|
Offset16To<MathKernInfo> mathKernInfo;
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (8);
|
|
};
|
|
|
|
struct MathGlyphVariantRecord
|
|
{
|
|
friend struct MathGlyphConstruction;
|
|
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
auto *out = c->serializer->embed (this);
|
|
if (unlikely (!out)) return_trace (false);
|
|
|
|
const hb_map_t& glyph_map = *c->plan->glyph_map;
|
|
return_trace (c->serializer->check_assign (out->variantGlyph, glyph_map.get (variantGlyph), HB_SERIALIZE_ERROR_INT_OVERFLOW));
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this));
|
|
}
|
|
|
|
void closure_glyphs (hb_set_t *variant_glyphs) const
|
|
{ variant_glyphs->add (variantGlyph); }
|
|
|
|
protected:
|
|
HBGlyphID16 variantGlyph; /* Glyph ID for the variant. */
|
|
HBUINT16 advanceMeasurement; /* Advance width/height, in design units, of the
|
|
* variant, in the direction of requested
|
|
* glyph extension. */
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (4);
|
|
};
|
|
|
|
struct PartFlags : HBUINT16
|
|
{
|
|
enum Flags {
|
|
Extender = 0x0001u, /* If set, the part can be skipped or repeated. */
|
|
|
|
Defined = 0x0001u, /* All defined flags. */
|
|
};
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (2);
|
|
};
|
|
|
|
struct MathGlyphPartRecord
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
auto *out = c->serializer->embed (this);
|
|
if (unlikely (!out)) return_trace (false);
|
|
|
|
const hb_map_t& glyph_map = *c->plan->glyph_map;
|
|
return_trace (c->serializer->check_assign (out->glyph, glyph_map.get (glyph), HB_SERIALIZE_ERROR_INT_OVERFLOW));
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this));
|
|
}
|
|
|
|
void extract (hb_ot_math_glyph_part_t &out,
|
|
int64_t mult,
|
|
hb_font_t *font) const
|
|
{
|
|
out.glyph = glyph;
|
|
|
|
out.start_connector_length = font->em_mult (startConnectorLength, mult);
|
|
out.end_connector_length = font->em_mult (endConnectorLength, mult);
|
|
out.full_advance = font->em_mult (fullAdvance, mult);
|
|
|
|
static_assert ((unsigned int) HB_OT_MATH_GLYPH_PART_FLAG_EXTENDER ==
|
|
(unsigned int) PartFlags::Extender, "");
|
|
|
|
out.flags = (hb_ot_math_glyph_part_flags_t)
|
|
(unsigned int)
|
|
(partFlags & PartFlags::Defined);
|
|
}
|
|
|
|
void closure_glyphs (hb_set_t *variant_glyphs) const
|
|
{ variant_glyphs->add (glyph); }
|
|
|
|
protected:
|
|
HBGlyphID16 glyph; /* Glyph ID for the part. */
|
|
HBUINT16 startConnectorLength;
|
|
/* Advance width/ height of the straight bar
|
|
* connector material, in design units, is at
|
|
* the beginning of the glyph, in the
|
|
* direction of the extension. */
|
|
HBUINT16 endConnectorLength;
|
|
/* Advance width/ height of the straight bar
|
|
* connector material, in design units, is at
|
|
* the end of the glyph, in the direction of
|
|
* the extension. */
|
|
HBUINT16 fullAdvance; /* Full advance width/height for this part,
|
|
* in the direction of the extension.
|
|
* In design units. */
|
|
PartFlags partFlags; /* Part qualifiers. */
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (10);
|
|
};
|
|
|
|
struct MathGlyphAssembly
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
auto *out = c->serializer->start_embed (*this);
|
|
if (unlikely (!out)) return_trace (false);
|
|
|
|
if (!c->serializer->copy (italicsCorrection, this)) return_trace (false);
|
|
if (!c->serializer->copy<HBUINT16> (partRecords.len)) return_trace (false);
|
|
|
|
for (const auto& record : partRecords.iter ())
|
|
if (!record.subset (c)) return_trace (false);
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
italicsCorrection.sanitize (c, this) &&
|
|
partRecords.sanitize (c));
|
|
}
|
|
|
|
unsigned int get_parts (hb_direction_t direction,
|
|
hb_font_t *font,
|
|
unsigned int start_offset,
|
|
unsigned int *parts_count, /* IN/OUT */
|
|
hb_ot_math_glyph_part_t *parts /* OUT */,
|
|
hb_position_t *italics_correction /* OUT */) const
|
|
{
|
|
if (parts_count)
|
|
{
|
|
int64_t mult = font->dir_mult (direction);
|
|
for (auto _ : hb_zip (partRecords.sub_array (start_offset, parts_count),
|
|
hb_array (parts, *parts_count)))
|
|
_.first.extract (_.second, mult, font);
|
|
}
|
|
|
|
if (italics_correction)
|
|
*italics_correction = italicsCorrection.get_x_value (font, this);
|
|
|
|
return partRecords.len;
|
|
}
|
|
|
|
void closure_glyphs (hb_set_t *variant_glyphs) const
|
|
{
|
|
for (const auto& _ : partRecords.iter ())
|
|
_.closure_glyphs (variant_glyphs);
|
|
}
|
|
|
|
protected:
|
|
MathValueRecord
|
|
italicsCorrection;
|
|
/* Italics correction of this
|
|
* MathGlyphAssembly. Should not
|
|
* depend on the assembly size. */
|
|
Array16Of<MathGlyphPartRecord>
|
|
partRecords; /* Array of part records, from
|
|
* left to right and bottom to
|
|
* top. */
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (6, partRecords);
|
|
};
|
|
|
|
struct MathGlyphConstruction
|
|
{
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
auto *out = c->serializer->start_embed (*this);
|
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
|
|
|
out->glyphAssembly.serialize_subset (c, glyphAssembly, this);
|
|
|
|
if (!c->serializer->check_assign (out->mathGlyphVariantRecord.len, mathGlyphVariantRecord.len, HB_SERIALIZE_ERROR_INT_OVERFLOW))
|
|
return_trace (false);
|
|
for (const auto& record : mathGlyphVariantRecord.iter ())
|
|
if (!record.subset (c)) return_trace (false);
|
|
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
glyphAssembly.sanitize (c, this) &&
|
|
mathGlyphVariantRecord.sanitize (c));
|
|
}
|
|
|
|
const MathGlyphAssembly &get_assembly () const { return this+glyphAssembly; }
|
|
|
|
unsigned int get_variants (hb_direction_t direction,
|
|
hb_font_t *font,
|
|
unsigned int start_offset,
|
|
unsigned int *variants_count, /* IN/OUT */
|
|
hb_ot_math_glyph_variant_t *variants /* OUT */) const
|
|
{
|
|
if (variants_count)
|
|
{
|
|
int64_t mult = font->dir_mult (direction);
|
|
for (auto _ : hb_zip (mathGlyphVariantRecord.sub_array (start_offset, variants_count),
|
|
hb_array (variants, *variants_count)))
|
|
_.second = {_.first.variantGlyph, font->em_mult (_.first.advanceMeasurement, mult)};
|
|
}
|
|
return mathGlyphVariantRecord.len;
|
|
}
|
|
|
|
void closure_glyphs (hb_set_t *variant_glyphs) const
|
|
{
|
|
(this+glyphAssembly).closure_glyphs (variant_glyphs);
|
|
|
|
for (const auto& _ : mathGlyphVariantRecord.iter ())
|
|
_.closure_glyphs (variant_glyphs);
|
|
}
|
|
|
|
protected:
|
|
/* Offset to MathGlyphAssembly table for this shape - from the beginning of
|
|
MathGlyphConstruction table. May be NULL. */
|
|
Offset16To<MathGlyphAssembly> glyphAssembly;
|
|
|
|
/* MathGlyphVariantRecords for alternative variants of the glyphs. */
|
|
Array16Of<MathGlyphVariantRecord> mathGlyphVariantRecord;
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (4, mathGlyphVariantRecord);
|
|
};
|
|
|
|
struct MathVariants
|
|
{
|
|
void closure_glyphs (const hb_set_t *glyph_set,
|
|
hb_set_t *variant_glyphs) const
|
|
{
|
|
const hb_array_t<const Offset16To<MathGlyphConstruction>> glyph_construction_offsets = glyphConstruction.as_array (vertGlyphCount + horizGlyphCount);
|
|
|
|
if (vertGlyphCoverage)
|
|
{
|
|
const auto vert_offsets = glyph_construction_offsets.sub_array (0, vertGlyphCount);
|
|
+ hb_zip (this+vertGlyphCoverage, vert_offsets)
|
|
| hb_filter (glyph_set, hb_first)
|
|
| hb_map (hb_second)
|
|
| hb_map (hb_add (this))
|
|
| hb_apply ([=] (const MathGlyphConstruction &_) { _.closure_glyphs (variant_glyphs); })
|
|
;
|
|
}
|
|
|
|
if (horizGlyphCoverage)
|
|
{
|
|
const auto hori_offsets = glyph_construction_offsets.sub_array (vertGlyphCount, horizGlyphCount);
|
|
+ hb_zip (this+horizGlyphCoverage, hori_offsets)
|
|
| hb_filter (glyph_set, hb_first)
|
|
| hb_map (hb_second)
|
|
| hb_map (hb_add (this))
|
|
| hb_apply ([=] (const MathGlyphConstruction &_) { _.closure_glyphs (variant_glyphs); })
|
|
;
|
|
}
|
|
}
|
|
|
|
void collect_coverage_and_indices (hb_sorted_vector_t<hb_codepoint_t>& new_coverage,
|
|
const Offset16To<Coverage>& coverage,
|
|
unsigned i,
|
|
unsigned end_index,
|
|
hb_set_t& indices,
|
|
const hb_set_t& glyphset,
|
|
const hb_map_t& glyph_map) const
|
|
{
|
|
if (!coverage) return;
|
|
|
|
for (const auto _ : (this+coverage).iter ())
|
|
{
|
|
if (i >= end_index) return;
|
|
if (glyphset.has (_))
|
|
{
|
|
unsigned new_gid = glyph_map.get (_);
|
|
new_coverage.push (new_gid);
|
|
indices.add (i);
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
const hb_set_t &glyphset = *c->plan->_glyphset_mathed;
|
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
|
|
|
auto *out = c->serializer->start_embed (*this);
|
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
|
if (!c->serializer->check_assign (out->minConnectorOverlap, minConnectorOverlap, HB_SERIALIZE_ERROR_INT_OVERFLOW))
|
|
return_trace (false);
|
|
|
|
hb_sorted_vector_t<hb_codepoint_t> new_vert_coverage;
|
|
hb_sorted_vector_t<hb_codepoint_t> new_hori_coverage;
|
|
hb_set_t indices;
|
|
collect_coverage_and_indices (new_vert_coverage, vertGlyphCoverage, 0, vertGlyphCount, indices, glyphset, glyph_map);
|
|
collect_coverage_and_indices (new_hori_coverage, horizGlyphCoverage, vertGlyphCount, vertGlyphCount + horizGlyphCount, indices, glyphset, glyph_map);
|
|
|
|
if (!c->serializer->check_assign (out->vertGlyphCount, new_vert_coverage.length, HB_SERIALIZE_ERROR_INT_OVERFLOW))
|
|
return_trace (false);
|
|
if (!c->serializer->check_assign (out->horizGlyphCount, new_hori_coverage.length, HB_SERIALIZE_ERROR_INT_OVERFLOW))
|
|
return_trace (false);
|
|
|
|
for (unsigned i : indices.iter ())
|
|
{
|
|
auto *o = c->serializer->embed (glyphConstruction[i]);
|
|
if (!o) return_trace (false);
|
|
o->serialize_subset (c, glyphConstruction[i], this);
|
|
}
|
|
|
|
out->vertGlyphCoverage.serialize_serialize (c->serializer, new_vert_coverage.iter ());
|
|
out->horizGlyphCoverage.serialize_serialize (c->serializer, new_hori_coverage.iter ());
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize_offsets (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
unsigned int count = vertGlyphCount + horizGlyphCount;
|
|
for (unsigned int i = 0; i < count; i++)
|
|
if (!glyphConstruction.arrayZ[i].sanitize (c, this)) return_trace (false);
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (c->check_struct (this) &&
|
|
vertGlyphCoverage.sanitize (c, this) &&
|
|
horizGlyphCoverage.sanitize (c, this) &&
|
|
c->check_array (glyphConstruction.arrayZ, vertGlyphCount + horizGlyphCount) &&
|
|
sanitize_offsets (c));
|
|
}
|
|
|
|
hb_position_t get_min_connector_overlap (hb_direction_t direction,
|
|
hb_font_t *font) const
|
|
{ return font->em_scale_dir (minConnectorOverlap, direction); }
|
|
|
|
unsigned int get_glyph_variants (hb_codepoint_t glyph,
|
|
hb_direction_t direction,
|
|
hb_font_t *font,
|
|
unsigned int start_offset,
|
|
unsigned int *variants_count, /* IN/OUT */
|
|
hb_ot_math_glyph_variant_t *variants /* OUT */) const
|
|
{ return get_glyph_construction (glyph, direction, font)
|
|
.get_variants (direction, font, start_offset, variants_count, variants); }
|
|
|
|
unsigned int get_glyph_parts (hb_codepoint_t glyph,
|
|
hb_direction_t direction,
|
|
hb_font_t *font,
|
|
unsigned int start_offset,
|
|
unsigned int *parts_count, /* IN/OUT */
|
|
hb_ot_math_glyph_part_t *parts /* OUT */,
|
|
hb_position_t *italics_correction /* OUT */) const
|
|
{ return get_glyph_construction (glyph, direction, font)
|
|
.get_assembly ()
|
|
.get_parts (direction, font,
|
|
start_offset, parts_count, parts,
|
|
italics_correction); }
|
|
|
|
private:
|
|
const MathGlyphConstruction &
|
|
get_glyph_construction (hb_codepoint_t glyph,
|
|
hb_direction_t direction,
|
|
hb_font_t *font HB_UNUSED) const
|
|
{
|
|
bool vertical = HB_DIRECTION_IS_VERTICAL (direction);
|
|
unsigned int count = vertical ? vertGlyphCount : horizGlyphCount;
|
|
const Offset16To<Coverage> &coverage = vertical ? vertGlyphCoverage
|
|
: horizGlyphCoverage;
|
|
|
|
unsigned int index = (this+coverage).get_coverage (glyph);
|
|
if (unlikely (index >= count)) return Null (MathGlyphConstruction);
|
|
|
|
if (!vertical)
|
|
index += vertGlyphCount;
|
|
|
|
return this+glyphConstruction[index];
|
|
}
|
|
|
|
protected:
|
|
HBUINT16 minConnectorOverlap;
|
|
/* Minimum overlap of connecting
|
|
* glyphs during glyph construction,
|
|
* in design units. */
|
|
Offset16To<Coverage> vertGlyphCoverage;
|
|
/* Offset to Coverage table -
|
|
* from the beginning of MathVariants
|
|
* table. */
|
|
Offset16To<Coverage> horizGlyphCoverage;
|
|
/* Offset to Coverage table -
|
|
* from the beginning of MathVariants
|
|
* table. */
|
|
HBUINT16 vertGlyphCount; /* Number of glyphs for which
|
|
* information is provided for
|
|
* vertically growing variants. */
|
|
HBUINT16 horizGlyphCount;/* Number of glyphs for which
|
|
* information is provided for
|
|
* horizontally growing variants. */
|
|
|
|
/* Array of offsets to MathGlyphConstruction tables - from the beginning of
|
|
the MathVariants table, for shapes growing in vertical/horizontal
|
|
direction. */
|
|
UnsizedArrayOf<Offset16To<MathGlyphConstruction>>
|
|
glyphConstruction;
|
|
|
|
public:
|
|
DEFINE_SIZE_ARRAY (10, glyphConstruction);
|
|
};
|
|
|
|
|
|
/*
|
|
* MATH -- Mathematical typesetting
|
|
* https://docs.microsoft.com/en-us/typography/opentype/spec/math
|
|
*/
|
|
|
|
struct MATH
|
|
{
|
|
static constexpr hb_tag_t tableTag = HB_OT_TAG_MATH;
|
|
|
|
bool has_data () const { return version.to_int (); }
|
|
|
|
void closure_glyphs (hb_set_t *glyph_set) const
|
|
{
|
|
if (mathVariants)
|
|
{
|
|
hb_set_t variant_glyphs;
|
|
(this+mathVariants).closure_glyphs (glyph_set, &variant_glyphs);
|
|
hb_set_union (glyph_set, &variant_glyphs);
|
|
}
|
|
}
|
|
|
|
bool subset (hb_subset_context_t *c) const
|
|
{
|
|
TRACE_SUBSET (this);
|
|
auto *out = c->serializer->embed (*this);
|
|
if (unlikely (!out)) return_trace (false);
|
|
|
|
out->mathConstants.serialize_copy (c->serializer, mathConstants, this, 0, hb_serialize_context_t::Head);
|
|
out->mathGlyphInfo.serialize_subset (c, mathGlyphInfo, this);
|
|
out->mathVariants.serialize_subset (c, mathVariants, this);
|
|
return_trace (true);
|
|
}
|
|
|
|
bool sanitize (hb_sanitize_context_t *c) const
|
|
{
|
|
TRACE_SANITIZE (this);
|
|
return_trace (version.sanitize (c) &&
|
|
likely (version.major == 1) &&
|
|
mathConstants.sanitize (c, this) &&
|
|
mathGlyphInfo.sanitize (c, this) &&
|
|
mathVariants.sanitize (c, this));
|
|
}
|
|
|
|
hb_position_t get_constant (hb_ot_math_constant_t constant,
|
|
hb_font_t *font) const
|
|
{ return (this+mathConstants).get_value (constant, font); }
|
|
|
|
const MathGlyphInfo &get_glyph_info () const { return this+mathGlyphInfo; }
|
|
|
|
const MathVariants &get_variants () const { return this+mathVariants; }
|
|
|
|
protected:
|
|
FixedVersion<>version; /* Version of the MATH table
|
|
* initially set to 0x00010000u */
|
|
Offset16To<MathConstants>
|
|
mathConstants; /* MathConstants table */
|
|
Offset16To<MathGlyphInfo>
|
|
mathGlyphInfo; /* MathGlyphInfo table */
|
|
Offset16To<MathVariants>
|
|
mathVariants; /* MathVariants table */
|
|
|
|
public:
|
|
DEFINE_SIZE_STATIC (10);
|
|
};
|
|
|
|
} /* namespace OT */
|
|
|
|
|
|
#endif /* HB_OT_MATH_TABLE_HH */
|