1*2d1272b8SAndroid Build Coastguard Worker /* 2*2d1272b8SAndroid Build Coastguard Worker * Copyright © 2023 Behdad Esfahbod 3*2d1272b8SAndroid Build Coastguard Worker * 4*2d1272b8SAndroid Build Coastguard Worker * This is part of HarfBuzz, a text shaping library. 5*2d1272b8SAndroid Build Coastguard Worker * 6*2d1272b8SAndroid Build Coastguard Worker * Permission is hereby granted, without written agreement and without 7*2d1272b8SAndroid Build Coastguard Worker * license or royalty fees, to use, copy, modify, and distribute this 8*2d1272b8SAndroid Build Coastguard Worker * software and its documentation for any purpose, provided that the 9*2d1272b8SAndroid Build Coastguard Worker * above copyright notice and the following two paragraphs appear in 10*2d1272b8SAndroid Build Coastguard Worker * all copies of this software. 11*2d1272b8SAndroid Build Coastguard Worker * 12*2d1272b8SAndroid Build Coastguard Worker * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR 13*2d1272b8SAndroid Build Coastguard Worker * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES 14*2d1272b8SAndroid Build Coastguard Worker * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN 15*2d1272b8SAndroid Build Coastguard Worker * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH 16*2d1272b8SAndroid Build Coastguard Worker * DAMAGE. 17*2d1272b8SAndroid Build Coastguard Worker * 18*2d1272b8SAndroid Build Coastguard Worker * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, 19*2d1272b8SAndroid Build Coastguard Worker * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND 20*2d1272b8SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS 21*2d1272b8SAndroid Build Coastguard Worker * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO 22*2d1272b8SAndroid Build Coastguard Worker * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. 23*2d1272b8SAndroid Build Coastguard Worker */ 24*2d1272b8SAndroid Build Coastguard Worker 25*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_SUBSET_INSTANCER_SOLVER_HH 26*2d1272b8SAndroid Build Coastguard Worker #define HB_SUBSET_INSTANCER_SOLVER_HH 27*2d1272b8SAndroid Build Coastguard Worker 28*2d1272b8SAndroid Build Coastguard Worker #include "hb.hh" 29*2d1272b8SAndroid Build Coastguard Worker 30*2d1272b8SAndroid Build Coastguard Worker /* pre-normalized distances */ 31*2d1272b8SAndroid Build Coastguard Worker struct TripleDistances 32*2d1272b8SAndroid Build Coastguard Worker { TripleDistancesTripleDistances33*2d1272b8SAndroid Build Coastguard Worker TripleDistances (): negative (1.0), positive (1.0) {} TripleDistancesTripleDistances34*2d1272b8SAndroid Build Coastguard Worker TripleDistances (double neg_, double pos_): negative (neg_), positive (pos_) {} TripleDistancesTripleDistances35*2d1272b8SAndroid Build Coastguard Worker TripleDistances (double min, double default_, double max) 36*2d1272b8SAndroid Build Coastguard Worker { 37*2d1272b8SAndroid Build Coastguard Worker negative = default_ - min; 38*2d1272b8SAndroid Build Coastguard Worker positive = max - default_; 39*2d1272b8SAndroid Build Coastguard Worker } 40*2d1272b8SAndroid Build Coastguard Worker 41*2d1272b8SAndroid Build Coastguard Worker double negative; 42*2d1272b8SAndroid Build Coastguard Worker double positive; 43*2d1272b8SAndroid Build Coastguard Worker }; 44*2d1272b8SAndroid Build Coastguard Worker 45*2d1272b8SAndroid Build Coastguard Worker struct Triple { 46*2d1272b8SAndroid Build Coastguard Worker TripleTriple47*2d1272b8SAndroid Build Coastguard Worker Triple () : 48*2d1272b8SAndroid Build Coastguard Worker minimum (0.0), middle (0.0), maximum (0.0) {} 49*2d1272b8SAndroid Build Coastguard Worker TripleTriple50*2d1272b8SAndroid Build Coastguard Worker Triple (double minimum_, double middle_, double maximum_) : 51*2d1272b8SAndroid Build Coastguard Worker minimum (minimum_), middle (middle_), maximum (maximum_) {} 52*2d1272b8SAndroid Build Coastguard Worker operator ==Triple53*2d1272b8SAndroid Build Coastguard Worker bool operator == (const Triple &o) const 54*2d1272b8SAndroid Build Coastguard Worker { 55*2d1272b8SAndroid Build Coastguard Worker return minimum == o.minimum && 56*2d1272b8SAndroid Build Coastguard Worker middle == o.middle && 57*2d1272b8SAndroid Build Coastguard Worker maximum == o.maximum; 58*2d1272b8SAndroid Build Coastguard Worker } 59*2d1272b8SAndroid Build Coastguard Worker operator !=Triple60*2d1272b8SAndroid Build Coastguard Worker bool operator != (const Triple o) const 61*2d1272b8SAndroid Build Coastguard Worker { return !(*this == o); } 62*2d1272b8SAndroid Build Coastguard Worker is_pointTriple63*2d1272b8SAndroid Build Coastguard Worker bool is_point () const 64*2d1272b8SAndroid Build Coastguard Worker { return minimum == middle && middle == maximum; } 65*2d1272b8SAndroid Build Coastguard Worker containsTriple66*2d1272b8SAndroid Build Coastguard Worker bool contains (double point) const 67*2d1272b8SAndroid Build Coastguard Worker { return minimum <= point && point <= maximum; } 68*2d1272b8SAndroid Build Coastguard Worker 69*2d1272b8SAndroid Build Coastguard Worker /* from hb_array_t hash ()*/ hashTriple70*2d1272b8SAndroid Build Coastguard Worker uint32_t hash () const 71*2d1272b8SAndroid Build Coastguard Worker { 72*2d1272b8SAndroid Build Coastguard Worker uint32_t current = /*cbf29ce4*/0x84222325; 73*2d1272b8SAndroid Build Coastguard Worker current = current ^ hb_hash (minimum); 74*2d1272b8SAndroid Build Coastguard Worker current = current * 16777619; 75*2d1272b8SAndroid Build Coastguard Worker 76*2d1272b8SAndroid Build Coastguard Worker current = current ^ hb_hash (middle); 77*2d1272b8SAndroid Build Coastguard Worker current = current * 16777619; 78*2d1272b8SAndroid Build Coastguard Worker 79*2d1272b8SAndroid Build Coastguard Worker current = current ^ hb_hash (maximum); 80*2d1272b8SAndroid Build Coastguard Worker current = current * 16777619; 81*2d1272b8SAndroid Build Coastguard Worker return current; 82*2d1272b8SAndroid Build Coastguard Worker } 83*2d1272b8SAndroid Build Coastguard Worker 84*2d1272b8SAndroid Build Coastguard Worker 85*2d1272b8SAndroid Build Coastguard Worker double minimum; 86*2d1272b8SAndroid Build Coastguard Worker double middle; 87*2d1272b8SAndroid Build Coastguard Worker double maximum; 88*2d1272b8SAndroid Build Coastguard Worker }; 89*2d1272b8SAndroid Build Coastguard Worker 90*2d1272b8SAndroid Build Coastguard Worker using rebase_tent_result_item_t = hb_pair_t<double, Triple>; 91*2d1272b8SAndroid Build Coastguard Worker using rebase_tent_result_t = hb_vector_t<rebase_tent_result_item_t>; 92*2d1272b8SAndroid Build Coastguard Worker 93*2d1272b8SAndroid Build Coastguard Worker /* renormalize a normalized value v to the range of an axis, 94*2d1272b8SAndroid Build Coastguard Worker * considering the prenormalized distances as well as the new axis limits. 95*2d1272b8SAndroid Build Coastguard Worker * Ported from fonttools */ 96*2d1272b8SAndroid Build Coastguard Worker HB_INTERNAL double renormalizeValue (double v, const Triple &triple, 97*2d1272b8SAndroid Build Coastguard Worker const TripleDistances &triple_distances, 98*2d1272b8SAndroid Build Coastguard Worker bool extrapolate = true); 99*2d1272b8SAndroid Build Coastguard Worker /* Given a tuple (lower,peak,upper) "tent" and new axis limits 100*2d1272b8SAndroid Build Coastguard Worker * (axisMin,axisDefault,axisMax), solves how to represent the tent 101*2d1272b8SAndroid Build Coastguard Worker * under the new axis configuration. All values are in normalized 102*2d1272b8SAndroid Build Coastguard Worker * -1,0,+1 coordinate system. Tent values can be outside this range. 103*2d1272b8SAndroid Build Coastguard Worker * 104*2d1272b8SAndroid Build Coastguard Worker * Return value: a list of tuples. Each tuple is of the form 105*2d1272b8SAndroid Build Coastguard Worker * (scalar,tent), where scalar is a multipler to multiply any 106*2d1272b8SAndroid Build Coastguard Worker * delta-sets by, and tent is a new tent for that output delta-set. 107*2d1272b8SAndroid Build Coastguard Worker * If tent value is Triple{}, that is a special deltaset that should 108*2d1272b8SAndroid Build Coastguard Worker * be always-enabled (called "gain"). 109*2d1272b8SAndroid Build Coastguard Worker */ 110*2d1272b8SAndroid Build Coastguard Worker HB_INTERNAL rebase_tent_result_t rebase_tent (Triple tent, 111*2d1272b8SAndroid Build Coastguard Worker Triple axisLimit, 112*2d1272b8SAndroid Build Coastguard Worker TripleDistances axis_triple_distances); 113*2d1272b8SAndroid Build Coastguard Worker 114*2d1272b8SAndroid Build Coastguard Worker #endif /* HB_SUBSET_INSTANCER_SOLVER_HH */ 115