| /* |
| * Copyright © 2018 Ebrahim Byagowi |
| * |
| * 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. |
| */ |
| |
| #ifndef HB_AAT_LAYOUT_JUST_TABLE_HH |
| #define HB_AAT_LAYOUT_JUST_TABLE_HH |
| |
| #include "hb-aat-layout-common.hh" |
| #include "hb-ot-layout.hh" |
| #include "hb-open-type.hh" |
| |
| #include "hb-aat-layout-morx-table.hh" |
| |
| /* |
| * just -- Justification |
| * https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6just.html |
| */ |
| #define HB_AAT_TAG_just HB_TAG('j','u','s','t') |
| |
| |
| namespace AAT { |
| |
| using namespace OT; |
| |
| |
| struct ActionSubrecordHeader |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this))); |
| } |
| |
| HBUINT16 actionClass; /* The JustClass value associated with this |
| * ActionSubrecord. */ |
| HBUINT16 actionType; /* The type of postcompensation action. */ |
| HBUINT16 actionLength; /* Length of this ActionSubrecord record, which |
| * must be a multiple of 4. */ |
| public: |
| DEFINE_SIZE_STATIC (6); |
| }; |
| |
| struct DecompositionAction |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this))); |
| } |
| |
| ActionSubrecordHeader |
| header; |
| HBFixed lowerLimit; /* If the distance factor is less than this value, |
| * then the ligature is decomposed. */ |
| HBFixed upperLimit; /* If the distance factor is greater than this value, |
| * then the ligature is decomposed. */ |
| HBUINT16 order; /* Numerical order in which this ligature will |
| * be decomposed; you may want infrequent ligatures |
| * to decompose before more frequent ones. The ligatures |
| * on the line of text will decompose in increasing |
| * value of this field. */ |
| ArrayOf<HBUINT16> |
| decomposedglyphs; |
| /* Number of 16-bit glyph indexes that follow; |
| * the ligature will be decomposed into these glyphs. |
| * |
| * Array of decomposed glyphs. */ |
| public: |
| DEFINE_SIZE_ARRAY (18, decomposedglyphs); |
| }; |
| |
| struct UnconditionalAddGlyphAction |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (c->check_struct (this)); |
| } |
| |
| protected: |
| ActionSubrecordHeader |
| header; |
| HBGlyphID addGlyph; /* Glyph that should be added if the distance factor |
| * is growing. */ |
| |
| public: |
| DEFINE_SIZE_STATIC (8); |
| }; |
| |
| struct ConditionalAddGlyphAction |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this))); |
| } |
| |
| protected: |
| ActionSubrecordHeader |
| header; |
| HBFixed substThreshold; /* Distance growth factor (in ems) at which |
| * this glyph is replaced and the growth factor |
| * recalculated. */ |
| HBGlyphID addGlyph; /* Glyph to be added as kashida. If this value is |
| * 0xFFFF, no extra glyph will be added. Note that |
| * generally when a glyph is added, justification |
| * will need to be redone. */ |
| HBGlyphID substGlyph; /* Glyph to be substituted for this glyph if the |
| * growth factor equals or exceeds the value of |
| * substThreshold. */ |
| public: |
| DEFINE_SIZE_STATIC (14); |
| }; |
| |
| struct DuctileGlyphAction |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this))); |
| } |
| |
| protected: |
| ActionSubrecordHeader |
| header; |
| HBUINT32 variationAxis; /* The 4-byte tag identifying the ductile axis. |
| * This would normally be 0x64756374 ('duct'), |
| * but you may use any axis the font contains. */ |
| HBFixed minimumLimit; /* The lowest value for the ductility axis tha |
| * still yields an acceptable appearance. Normally |
| * this will be 1.0. */ |
| HBFixed noStretchValue; /* This is the default value that corresponds to |
| * no change in appearance. Normally, this will |
| * be 1.0. */ |
| HBFixed maximumLimit; /* The highest value for the ductility axis that |
| * still yields an acceptable appearance. */ |
| public: |
| DEFINE_SIZE_STATIC (22); |
| }; |
| |
| struct RepeatedAddGlyphAction |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this))); |
| } |
| |
| protected: |
| ActionSubrecordHeader |
| header; |
| HBUINT16 flags; /* Currently unused; set to 0. */ |
| HBGlyphID glyph; /* Glyph that should be added if the distance factor |
| * is growing. */ |
| public: |
| DEFINE_SIZE_STATIC (10); |
| }; |
| |
| struct ActionSubrecord |
| { |
| unsigned int get_length () const { return u.header.actionLength; } |
| |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| if (unlikely (!c->check_struct (this))) |
| return_trace (false); |
| |
| switch (u.header.actionType) |
| { |
| case 0: return_trace (u.decompositionAction.sanitize (c)); |
| case 1: return_trace (u.unconditionalAddGlyphAction.sanitize (c)); |
| case 2: return_trace (u.conditionalAddGlyphAction.sanitize (c)); |
| // case 3: return_trace (u.stretchGlyphAction.sanitize (c)); |
| case 4: return_trace (u.decompositionAction.sanitize (c)); |
| case 5: return_trace (u.decompositionAction.sanitize (c)); |
| default: return_trace (true); |
| } |
| } |
| |
| protected: |
| union { |
| ActionSubrecordHeader header; |
| DecompositionAction decompositionAction; |
| UnconditionalAddGlyphAction unconditionalAddGlyphAction; |
| ConditionalAddGlyphAction conditionalAddGlyphAction; |
| /* StretchGlyphAction stretchGlyphAction; -- Not supported by CoreText */ |
| DuctileGlyphAction ductileGlyphAction; |
| RepeatedAddGlyphAction repeatedAddGlyphAction; |
| } u; /* Data. The format of this data depends on |
| * the value of the actionType field. */ |
| public: |
| DEFINE_SIZE_UNION (6, header); |
| }; |
| |
| struct PostcompensationActionChain |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| if (unlikely (!c->check_struct (this))) |
| return_trace (false); |
| |
| unsigned int offset = min_size; |
| for (unsigned int i = 0; i < count; i++) |
| { |
| const ActionSubrecord& subrecord = StructAtOffset<ActionSubrecord> (this, offset); |
| if (unlikely (!subrecord.sanitize (c))) return_trace (false); |
| offset += subrecord.get_length (); |
| } |
| |
| return_trace (true); |
| } |
| |
| protected: |
| HBUINT32 count; |
| |
| public: |
| DEFINE_SIZE_STATIC (4); |
| }; |
| |
| struct JustWidthDeltaEntry |
| { |
| enum Flags |
| { |
| Reserved1 =0xE000,/* Reserved. You should set these bits to zero. */ |
| UnlimiteGap =0x1000,/* The glyph can take unlimited gap. When this |
| * glyph participates in the justification process, |
| * it and any other glyphs on the line having this |
| * bit set absorb all the remaining gap. */ |
| Reserved2 =0x0FF0,/* Reserved. You should set these bits to zero. */ |
| Priority =0x000F /* The justification priority of the glyph. */ |
| }; |
| |
| enum Priority |
| { |
| Kashida = 0, /* Kashida priority. This is the highest priority |
| * during justification. */ |
| Whitespace = 1, /* Whitespace priority. Any whitespace glyphs (as |
| * identified in the glyph properties table) will |
| * get this priority. */ |
| InterCharacter = 2, /* Inter-character priority. Give this to any |
| * remaining glyphs. */ |
| NullPriority = 3 /* Null priority. You should set this priority for |
| * glyphs that only participate in justification |
| * after the above priorities. Normally all glyphs |
| * have one of the previous three values. If you |
| * don't want a glyph to participate in justification, |
| * and you don't want to set its factors to zero, |
| * you may instead assign it to the null priority. */ |
| }; |
| |
| protected: |
| HBFixed beforeGrowLimit;/* The ratio by which the advance width of the |
| * glyph is permitted to grow on the left or top side. */ |
| HBFixed beforeShrinkLimit; |
| /* The ratio by which the advance width of the |
| * glyph is permitted to shrink on the left or top side. */ |
| HBFixed afterGrowLimit; /* The ratio by which the advance width of the glyph |
| * is permitted to shrink on the left or top side. */ |
| HBFixed afterShrinkLimit; |
| /* The ratio by which the advance width of the glyph |
| * is at most permitted to shrink on the right or |
| * bottom side. */ |
| HBUINT16 growFlags; /* Flags controlling the grow case. */ |
| HBUINT16 shrinkFlags; /* Flags controlling the shrink case. */ |
| |
| public: |
| DEFINE_SIZE_STATIC (20); |
| }; |
| |
| struct WidthDeltaPair |
| { |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this))); |
| } |
| |
| protected: |
| HBUINT32 justClass; /* The justification category associated |
| * with the wdRecord field. Only 7 bits of |
| * this field are used. (The other bits are |
| * used as padding to guarantee longword |
| * alignment of the following record). */ |
| JustWidthDeltaEntry |
| wdRecord; /* The actual width delta record. */ |
| |
| public: |
| DEFINE_SIZE_STATIC (24); |
| }; |
| |
| typedef OT::LArrayOf<WidthDeltaPair> WidthDeltaCluster; |
| |
| struct JustificationCategory |
| { |
| typedef void EntryData; |
| |
| enum Flags |
| { |
| SetMark =0x8000,/* If set, make the current glyph the marked |
| * glyph. */ |
| DontAdvance =0x4000,/* If set, don't advance to the next glyph before |
| * going to the new state. */ |
| MarkCategory =0x3F80,/* The justification category for the marked |
| * glyph if nonzero. */ |
| CurrentCategory =0x007F /* The justification category for the current |
| * glyph if nonzero. */ |
| }; |
| |
| bool sanitize (hb_sanitize_context_t *c, const void *base) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this) && |
| morphHeader.sanitize (c) && |
| stHeader.sanitize (c))); |
| } |
| |
| protected: |
| ChainSubtable<ObsoleteTypes> |
| morphHeader; /* Metamorphosis-style subtable header. */ |
| StateTable<ObsoleteTypes, EntryData> |
| stHeader; /* The justification insertion state table header */ |
| public: |
| DEFINE_SIZE_STATIC (30); |
| }; |
| |
| struct JustificationHeader |
| { |
| bool sanitize (hb_sanitize_context_t *c, const void *base) const |
| { |
| TRACE_SANITIZE (this); |
| return_trace (likely (c->check_struct (this) && |
| justClassTable.sanitize (c, base, base) && |
| wdcTable.sanitize (c, base) && |
| pcTable.sanitize (c, base) && |
| lookupTable.sanitize (c, base))); |
| } |
| |
| protected: |
| OffsetTo<JustificationCategory> |
| justClassTable; /* Offset to the justification category state table. */ |
| OffsetTo<WidthDeltaCluster> |
| wdcTable; /* Offset from start of justification table to start |
| * of the subtable containing the width delta factors |
| * for the glyphs in your font. |
| * |
| * The width delta clusters table. */ |
| OffsetTo<PostcompensationActionChain> |
| pcTable; /* Offset from start of justification table to start |
| * of postcompensation subtable (set to zero if none). |
| * |
| * The postcompensation subtable, if present in the font. */ |
| Lookup<OffsetTo<WidthDeltaCluster>> |
| lookupTable; /* Lookup table associating glyphs with width delta |
| * clusters. See the description of Width Delta Clusters |
| * table for details on how to interpret the lookup values. */ |
| |
| public: |
| DEFINE_SIZE_MIN (8); |
| }; |
| |
| struct just |
| { |
| static constexpr hb_tag_t tableTag = HB_AAT_TAG_just; |
| |
| bool sanitize (hb_sanitize_context_t *c) const |
| { |
| TRACE_SANITIZE (this); |
| |
| return_trace (likely (c->check_struct (this) && |
| version.major == 1 && |
| horizData.sanitize (c, this, this) && |
| vertData.sanitize (c, this, this))); |
| } |
| |
| protected: |
| FixedVersion<>version; /* Version of the justification table |
| * (0x00010000u for version 1.0). */ |
| HBUINT16 format; /* Format of the justification table (set to 0). */ |
| OffsetTo<JustificationHeader> |
| horizData; /* Byte offset from the start of the justification table |
| * to the header for tables that contain justification |
| * information for horizontal text. |
| * If you are not including this information, |
| * store 0. */ |
| OffsetTo<JustificationHeader> |
| vertData; /* ditto, vertical */ |
| |
| public: |
| DEFINE_SIZE_STATIC (10); |
| }; |
| |
| } /* namespace AAT */ |
| |
| |
| #endif /* HB_AAT_LAYOUT_JUST_TABLE_HH */ |