/* * 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_LCAR_TABLE_HH #define HB_AAT_LAYOUT_LCAR_TABLE_HH #include "hb-open-type.hh" #include "hb-aat-layout-common.hh" /* * lcar -- Ligature caret * https://developer.apple.com/fonts/TrueType-Reference-Manual/RM06/Chap6lcar.html */ #define HB_AAT_TAG_lcar HB_TAG('l','c','a','r') namespace AAT { typedef ArrayOf LigCaretClassEntry; struct lcarFormat0 { unsigned int get_lig_carets (hb_font_t *font, hb_direction_t direction, hb_codepoint_t glyph, unsigned int start_offset, unsigned int *caret_count /* IN/OUT */, hb_position_t *caret_array /* OUT */, const void *base) const { const OffsetTo* entry_offset = lookupTable.get_value (glyph, font->face->get_num_glyphs ()); const LigCaretClassEntry& array = entry_offset ? base+*entry_offset : Null (LigCaretClassEntry); if (caret_count) { hb_array_t arr = array.sub_array (start_offset, caret_count); for (unsigned int i = 0; i < arr.length; ++i) caret_array[i] = font->em_scale_dir (arr[i], direction); } return array.len; } bool sanitize (hb_sanitize_context_t *c, const void *base) const { TRACE_SANITIZE (this); return_trace (likely (c->check_struct (this) && lookupTable.sanitize (c, base))); } protected: Lookup> lookupTable; /* data Lookup table associating glyphs */ public: DEFINE_SIZE_MIN (2); }; struct lcarFormat1 { unsigned int get_lig_carets (hb_font_t *font, hb_direction_t direction, hb_codepoint_t glyph, unsigned int start_offset, unsigned int *caret_count /* IN/OUT */, hb_position_t *caret_array /* OUT */, const void *base) const { const OffsetTo* entry_offset = lookupTable.get_value (glyph, font->face->get_num_glyphs ()); const LigCaretClassEntry& array = entry_offset ? base+*entry_offset : Null (LigCaretClassEntry); if (caret_count) { hb_array_t arr = array.sub_array (start_offset, caret_count); for (unsigned int i = 0; i < arr.length; ++i) { hb_position_t x = 0, y = 0; font->get_glyph_contour_point_for_origin (glyph, arr[i], direction, &x, &y); caret_array[i] = HB_DIRECTION_IS_HORIZONTAL (direction) ? x : y; } } return array.len; } bool sanitize (hb_sanitize_context_t *c, const void *base) const { TRACE_SANITIZE (this); return_trace (likely (c->check_struct (this) && lookupTable.sanitize (c, base))); } protected: Lookup> lookupTable; /* data Lookup table associating glyphs */ public: DEFINE_SIZE_MIN (2); }; struct lcar { static constexpr hb_tag_t tableTag = HB_AAT_TAG_lcar; bool has_data () const { return version.major; } unsigned int get_lig_carets (hb_font_t *font, hb_direction_t direction, hb_codepoint_t glyph, unsigned int start_offset, unsigned int *caret_count /* IN/OUT */, hb_position_t *caret_array /* OUT */) const { if (!has_data ()) { if (caret_count) *caret_count = 0; return 0; } switch (format) { case 0: return u.format0.get_lig_carets (font, direction, glyph, start_offset, caret_count, caret_array, this); case 1: return u.format1.get_lig_carets (font, direction, glyph, start_offset, caret_count, caret_array, this); default:if (caret_count) *caret_count = 0; return 0; } } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); if (unlikely (!c->check_struct (this) || version.major != 1)) return_trace (false); switch (format) { case 0: return_trace (u.format0.sanitize (c, this)); case 1: return_trace (u.format1.sanitize (c, this)); default:return_trace (true); } } protected: FixedVersion<>version; /* Version number of the ligature caret table */ HBUINT16 format; /* Format of the ligature caret table. */ union { lcarFormat0 format0; lcarFormat0 format1; } u; public: DEFINE_SIZE_MIN (8); }; } /* namespace AAT */ #endif /* HB_AAT_LAYOUT_LCAR_TABLE_HH */