fonttools 4.60.2__cp311-cp311-win32.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- fontTools/__init__.py +8 -0
- fontTools/__main__.py +35 -0
- fontTools/afmLib.py +439 -0
- fontTools/agl.py +5233 -0
- fontTools/annotations.py +30 -0
- fontTools/cffLib/CFF2ToCFF.py +258 -0
- fontTools/cffLib/CFFToCFF2.py +305 -0
- fontTools/cffLib/__init__.py +3694 -0
- fontTools/cffLib/specializer.py +927 -0
- fontTools/cffLib/transforms.py +495 -0
- fontTools/cffLib/width.py +210 -0
- fontTools/colorLib/__init__.py +0 -0
- fontTools/colorLib/builder.py +664 -0
- fontTools/colorLib/errors.py +2 -0
- fontTools/colorLib/geometry.py +143 -0
- fontTools/colorLib/table_builder.py +223 -0
- fontTools/colorLib/unbuilder.py +81 -0
- fontTools/config/__init__.py +90 -0
- fontTools/cu2qu/__init__.py +15 -0
- fontTools/cu2qu/__main__.py +6 -0
- fontTools/cu2qu/benchmark.py +54 -0
- fontTools/cu2qu/cli.py +198 -0
- fontTools/cu2qu/cu2qu.c +15817 -0
- fontTools/cu2qu/cu2qu.cp311-win32.pyd +0 -0
- fontTools/cu2qu/cu2qu.py +563 -0
- fontTools/cu2qu/errors.py +77 -0
- fontTools/cu2qu/ufo.py +363 -0
- fontTools/designspaceLib/__init__.py +3343 -0
- fontTools/designspaceLib/__main__.py +6 -0
- fontTools/designspaceLib/split.py +475 -0
- fontTools/designspaceLib/statNames.py +260 -0
- fontTools/designspaceLib/types.py +147 -0
- fontTools/encodings/MacRoman.py +258 -0
- fontTools/encodings/StandardEncoding.py +258 -0
- fontTools/encodings/__init__.py +1 -0
- fontTools/encodings/codecs.py +135 -0
- fontTools/feaLib/__init__.py +4 -0
- fontTools/feaLib/__main__.py +78 -0
- fontTools/feaLib/ast.py +2143 -0
- fontTools/feaLib/builder.py +1814 -0
- fontTools/feaLib/error.py +22 -0
- fontTools/feaLib/lexer.c +17029 -0
- fontTools/feaLib/lexer.cp311-win32.pyd +0 -0
- fontTools/feaLib/lexer.py +287 -0
- fontTools/feaLib/location.py +12 -0
- fontTools/feaLib/lookupDebugInfo.py +12 -0
- fontTools/feaLib/parser.py +2394 -0
- fontTools/feaLib/variableScalar.py +118 -0
- fontTools/fontBuilder.py +1014 -0
- fontTools/help.py +36 -0
- fontTools/merge/__init__.py +248 -0
- fontTools/merge/__main__.py +6 -0
- fontTools/merge/base.py +81 -0
- fontTools/merge/cmap.py +173 -0
- fontTools/merge/layout.py +526 -0
- fontTools/merge/options.py +85 -0
- fontTools/merge/tables.py +352 -0
- fontTools/merge/unicode.py +78 -0
- fontTools/merge/util.py +143 -0
- fontTools/misc/__init__.py +1 -0
- fontTools/misc/arrayTools.py +424 -0
- fontTools/misc/bezierTools.c +39731 -0
- fontTools/misc/bezierTools.cp311-win32.pyd +0 -0
- fontTools/misc/bezierTools.py +1500 -0
- fontTools/misc/classifyTools.py +170 -0
- fontTools/misc/cliTools.py +53 -0
- fontTools/misc/configTools.py +349 -0
- fontTools/misc/cython.py +27 -0
- fontTools/misc/dictTools.py +83 -0
- fontTools/misc/eexec.py +119 -0
- fontTools/misc/encodingTools.py +72 -0
- fontTools/misc/enumTools.py +23 -0
- fontTools/misc/etree.py +456 -0
- fontTools/misc/filenames.py +245 -0
- fontTools/misc/filesystem/__init__.py +68 -0
- fontTools/misc/filesystem/_base.py +134 -0
- fontTools/misc/filesystem/_copy.py +45 -0
- fontTools/misc/filesystem/_errors.py +54 -0
- fontTools/misc/filesystem/_info.py +75 -0
- fontTools/misc/filesystem/_osfs.py +164 -0
- fontTools/misc/filesystem/_path.py +67 -0
- fontTools/misc/filesystem/_subfs.py +92 -0
- fontTools/misc/filesystem/_tempfs.py +34 -0
- fontTools/misc/filesystem/_tools.py +34 -0
- fontTools/misc/filesystem/_walk.py +55 -0
- fontTools/misc/filesystem/_zipfs.py +204 -0
- fontTools/misc/fixedTools.py +253 -0
- fontTools/misc/intTools.py +25 -0
- fontTools/misc/iterTools.py +12 -0
- fontTools/misc/lazyTools.py +42 -0
- fontTools/misc/loggingTools.py +543 -0
- fontTools/misc/macCreatorType.py +56 -0
- fontTools/misc/macRes.py +261 -0
- fontTools/misc/plistlib/__init__.py +681 -0
- fontTools/misc/plistlib/py.typed +0 -0
- fontTools/misc/psCharStrings.py +1511 -0
- fontTools/misc/psLib.py +398 -0
- fontTools/misc/psOperators.py +572 -0
- fontTools/misc/py23.py +96 -0
- fontTools/misc/roundTools.py +110 -0
- fontTools/misc/sstruct.py +227 -0
- fontTools/misc/symfont.py +242 -0
- fontTools/misc/testTools.py +233 -0
- fontTools/misc/textTools.py +156 -0
- fontTools/misc/timeTools.py +88 -0
- fontTools/misc/transform.py +516 -0
- fontTools/misc/treeTools.py +45 -0
- fontTools/misc/vector.py +147 -0
- fontTools/misc/visitor.py +158 -0
- fontTools/misc/xmlReader.py +188 -0
- fontTools/misc/xmlWriter.py +231 -0
- fontTools/mtiLib/__init__.py +1400 -0
- fontTools/mtiLib/__main__.py +5 -0
- fontTools/otlLib/__init__.py +1 -0
- fontTools/otlLib/builder.py +3465 -0
- fontTools/otlLib/error.py +11 -0
- fontTools/otlLib/maxContextCalc.py +96 -0
- fontTools/otlLib/optimize/__init__.py +53 -0
- fontTools/otlLib/optimize/__main__.py +6 -0
- fontTools/otlLib/optimize/gpos.py +439 -0
- fontTools/pens/__init__.py +1 -0
- fontTools/pens/areaPen.py +52 -0
- fontTools/pens/basePen.py +475 -0
- fontTools/pens/boundsPen.py +98 -0
- fontTools/pens/cairoPen.py +26 -0
- fontTools/pens/cocoaPen.py +26 -0
- fontTools/pens/cu2quPen.py +325 -0
- fontTools/pens/explicitClosingLinePen.py +101 -0
- fontTools/pens/filterPen.py +433 -0
- fontTools/pens/freetypePen.py +462 -0
- fontTools/pens/hashPointPen.py +89 -0
- fontTools/pens/momentsPen.c +13378 -0
- fontTools/pens/momentsPen.cp311-win32.pyd +0 -0
- fontTools/pens/momentsPen.py +879 -0
- fontTools/pens/perimeterPen.py +69 -0
- fontTools/pens/pointInsidePen.py +192 -0
- fontTools/pens/pointPen.py +643 -0
- fontTools/pens/qtPen.py +29 -0
- fontTools/pens/qu2cuPen.py +105 -0
- fontTools/pens/quartzPen.py +43 -0
- fontTools/pens/recordingPen.py +335 -0
- fontTools/pens/reportLabPen.py +79 -0
- fontTools/pens/reverseContourPen.py +96 -0
- fontTools/pens/roundingPen.py +130 -0
- fontTools/pens/statisticsPen.py +312 -0
- fontTools/pens/svgPathPen.py +310 -0
- fontTools/pens/t2CharStringPen.py +88 -0
- fontTools/pens/teePen.py +55 -0
- fontTools/pens/transformPen.py +115 -0
- fontTools/pens/ttGlyphPen.py +335 -0
- fontTools/pens/wxPen.py +29 -0
- fontTools/qu2cu/__init__.py +15 -0
- fontTools/qu2cu/__main__.py +7 -0
- fontTools/qu2cu/benchmark.py +56 -0
- fontTools/qu2cu/cli.py +125 -0
- fontTools/qu2cu/qu2cu.c +16682 -0
- fontTools/qu2cu/qu2cu.cp311-win32.pyd +0 -0
- fontTools/qu2cu/qu2cu.py +405 -0
- fontTools/subset/__init__.py +4096 -0
- fontTools/subset/__main__.py +6 -0
- fontTools/subset/cff.py +184 -0
- fontTools/subset/svg.py +253 -0
- fontTools/subset/util.py +25 -0
- fontTools/svgLib/__init__.py +3 -0
- fontTools/svgLib/path/__init__.py +65 -0
- fontTools/svgLib/path/arc.py +154 -0
- fontTools/svgLib/path/parser.py +322 -0
- fontTools/svgLib/path/shapes.py +183 -0
- fontTools/t1Lib/__init__.py +648 -0
- fontTools/tfmLib.py +460 -0
- fontTools/ttLib/__init__.py +30 -0
- fontTools/ttLib/__main__.py +148 -0
- fontTools/ttLib/macUtils.py +54 -0
- fontTools/ttLib/removeOverlaps.py +395 -0
- fontTools/ttLib/reorderGlyphs.py +285 -0
- fontTools/ttLib/scaleUpem.py +436 -0
- fontTools/ttLib/sfnt.py +661 -0
- fontTools/ttLib/standardGlyphOrder.py +271 -0
- fontTools/ttLib/tables/B_A_S_E_.py +14 -0
- fontTools/ttLib/tables/BitmapGlyphMetrics.py +64 -0
- fontTools/ttLib/tables/C_B_D_T_.py +113 -0
- fontTools/ttLib/tables/C_B_L_C_.py +19 -0
- fontTools/ttLib/tables/C_F_F_.py +61 -0
- fontTools/ttLib/tables/C_F_F__2.py +26 -0
- fontTools/ttLib/tables/C_O_L_R_.py +165 -0
- fontTools/ttLib/tables/C_P_A_L_.py +305 -0
- fontTools/ttLib/tables/D_S_I_G_.py +158 -0
- fontTools/ttLib/tables/D__e_b_g.py +35 -0
- fontTools/ttLib/tables/DefaultTable.py +49 -0
- fontTools/ttLib/tables/E_B_D_T_.py +835 -0
- fontTools/ttLib/tables/E_B_L_C_.py +718 -0
- fontTools/ttLib/tables/F_F_T_M_.py +52 -0
- fontTools/ttLib/tables/F__e_a_t.py +149 -0
- fontTools/ttLib/tables/G_D_E_F_.py +13 -0
- fontTools/ttLib/tables/G_M_A_P_.py +148 -0
- fontTools/ttLib/tables/G_P_K_G_.py +133 -0
- fontTools/ttLib/tables/G_P_O_S_.py +14 -0
- fontTools/ttLib/tables/G_S_U_B_.py +13 -0
- fontTools/ttLib/tables/G_V_A_R_.py +5 -0
- fontTools/ttLib/tables/G__l_a_t.py +235 -0
- fontTools/ttLib/tables/G__l_o_c.py +85 -0
- fontTools/ttLib/tables/H_V_A_R_.py +13 -0
- fontTools/ttLib/tables/J_S_T_F_.py +13 -0
- fontTools/ttLib/tables/L_T_S_H_.py +58 -0
- fontTools/ttLib/tables/M_A_T_H_.py +13 -0
- fontTools/ttLib/tables/M_E_T_A_.py +352 -0
- fontTools/ttLib/tables/M_V_A_R_.py +13 -0
- fontTools/ttLib/tables/O_S_2f_2.py +752 -0
- fontTools/ttLib/tables/S_I_N_G_.py +99 -0
- fontTools/ttLib/tables/S_T_A_T_.py +15 -0
- fontTools/ttLib/tables/S_V_G_.py +223 -0
- fontTools/ttLib/tables/S__i_l_f.py +1040 -0
- fontTools/ttLib/tables/S__i_l_l.py +92 -0
- fontTools/ttLib/tables/T_S_I_B_.py +13 -0
- fontTools/ttLib/tables/T_S_I_C_.py +14 -0
- fontTools/ttLib/tables/T_S_I_D_.py +13 -0
- fontTools/ttLib/tables/T_S_I_J_.py +13 -0
- fontTools/ttLib/tables/T_S_I_P_.py +13 -0
- fontTools/ttLib/tables/T_S_I_S_.py +13 -0
- fontTools/ttLib/tables/T_S_I_V_.py +26 -0
- fontTools/ttLib/tables/T_S_I__0.py +70 -0
- fontTools/ttLib/tables/T_S_I__1.py +163 -0
- fontTools/ttLib/tables/T_S_I__2.py +17 -0
- fontTools/ttLib/tables/T_S_I__3.py +22 -0
- fontTools/ttLib/tables/T_S_I__5.py +60 -0
- fontTools/ttLib/tables/T_T_F_A_.py +14 -0
- fontTools/ttLib/tables/TupleVariation.py +884 -0
- fontTools/ttLib/tables/V_A_R_C_.py +12 -0
- fontTools/ttLib/tables/V_D_M_X_.py +249 -0
- fontTools/ttLib/tables/V_O_R_G_.py +165 -0
- fontTools/ttLib/tables/V_V_A_R_.py +13 -0
- fontTools/ttLib/tables/__init__.py +98 -0
- fontTools/ttLib/tables/_a_n_k_r.py +15 -0
- fontTools/ttLib/tables/_a_v_a_r.py +193 -0
- fontTools/ttLib/tables/_b_s_l_n.py +15 -0
- fontTools/ttLib/tables/_c_i_d_g.py +24 -0
- fontTools/ttLib/tables/_c_m_a_p.py +1591 -0
- fontTools/ttLib/tables/_c_v_a_r.py +94 -0
- fontTools/ttLib/tables/_c_v_t.py +56 -0
- fontTools/ttLib/tables/_f_e_a_t.py +15 -0
- fontTools/ttLib/tables/_f_p_g_m.py +62 -0
- fontTools/ttLib/tables/_f_v_a_r.py +261 -0
- fontTools/ttLib/tables/_g_a_s_p.py +63 -0
- fontTools/ttLib/tables/_g_c_i_d.py +13 -0
- fontTools/ttLib/tables/_g_l_y_f.py +2311 -0
- fontTools/ttLib/tables/_g_v_a_r.py +340 -0
- fontTools/ttLib/tables/_h_d_m_x.py +127 -0
- fontTools/ttLib/tables/_h_e_a_d.py +130 -0
- fontTools/ttLib/tables/_h_h_e_a.py +147 -0
- fontTools/ttLib/tables/_h_m_t_x.py +164 -0
- fontTools/ttLib/tables/_k_e_r_n.py +289 -0
- fontTools/ttLib/tables/_l_c_a_r.py +13 -0
- fontTools/ttLib/tables/_l_o_c_a.py +70 -0
- fontTools/ttLib/tables/_l_t_a_g.py +72 -0
- fontTools/ttLib/tables/_m_a_x_p.py +147 -0
- fontTools/ttLib/tables/_m_e_t_a.py +112 -0
- fontTools/ttLib/tables/_m_o_r_t.py +14 -0
- fontTools/ttLib/tables/_m_o_r_x.py +15 -0
- fontTools/ttLib/tables/_n_a_m_e.py +1242 -0
- fontTools/ttLib/tables/_o_p_b_d.py +14 -0
- fontTools/ttLib/tables/_p_o_s_t.py +319 -0
- fontTools/ttLib/tables/_p_r_e_p.py +16 -0
- fontTools/ttLib/tables/_p_r_o_p.py +12 -0
- fontTools/ttLib/tables/_s_b_i_x.py +129 -0
- fontTools/ttLib/tables/_t_r_a_k.py +332 -0
- fontTools/ttLib/tables/_v_h_e_a.py +139 -0
- fontTools/ttLib/tables/_v_m_t_x.py +19 -0
- fontTools/ttLib/tables/asciiTable.py +20 -0
- fontTools/ttLib/tables/grUtils.py +92 -0
- fontTools/ttLib/tables/otBase.py +1458 -0
- fontTools/ttLib/tables/otConverters.py +2068 -0
- fontTools/ttLib/tables/otData.py +6400 -0
- fontTools/ttLib/tables/otTables.py +2703 -0
- fontTools/ttLib/tables/otTraverse.py +163 -0
- fontTools/ttLib/tables/sbixGlyph.py +149 -0
- fontTools/ttLib/tables/sbixStrike.py +177 -0
- fontTools/ttLib/tables/table_API_readme.txt +91 -0
- fontTools/ttLib/tables/ttProgram.py +594 -0
- fontTools/ttLib/ttCollection.py +125 -0
- fontTools/ttLib/ttFont.py +1148 -0
- fontTools/ttLib/ttGlyphSet.py +490 -0
- fontTools/ttLib/ttVisitor.py +32 -0
- fontTools/ttLib/woff2.py +1680 -0
- fontTools/ttx.py +479 -0
- fontTools/ufoLib/__init__.py +2575 -0
- fontTools/ufoLib/converters.py +407 -0
- fontTools/ufoLib/errors.py +30 -0
- fontTools/ufoLib/etree.py +6 -0
- fontTools/ufoLib/filenames.py +356 -0
- fontTools/ufoLib/glifLib.py +2120 -0
- fontTools/ufoLib/kerning.py +141 -0
- fontTools/ufoLib/plistlib.py +47 -0
- fontTools/ufoLib/pointPen.py +6 -0
- fontTools/ufoLib/utils.py +107 -0
- fontTools/ufoLib/validators.py +1208 -0
- fontTools/unicode.py +50 -0
- fontTools/unicodedata/Blocks.py +817 -0
- fontTools/unicodedata/Mirrored.py +446 -0
- fontTools/unicodedata/OTTags.py +50 -0
- fontTools/unicodedata/ScriptExtensions.py +832 -0
- fontTools/unicodedata/Scripts.py +3639 -0
- fontTools/unicodedata/__init__.py +306 -0
- fontTools/varLib/__init__.py +1600 -0
- fontTools/varLib/__main__.py +6 -0
- fontTools/varLib/avar/__init__.py +0 -0
- fontTools/varLib/avar/__main__.py +72 -0
- fontTools/varLib/avar/build.py +79 -0
- fontTools/varLib/avar/map.py +108 -0
- fontTools/varLib/avar/plan.py +1004 -0
- fontTools/varLib/avar/unbuild.py +271 -0
- fontTools/varLib/avarPlanner.py +8 -0
- fontTools/varLib/builder.py +215 -0
- fontTools/varLib/cff.py +631 -0
- fontTools/varLib/errors.py +219 -0
- fontTools/varLib/featureVars.py +703 -0
- fontTools/varLib/hvar.py +113 -0
- fontTools/varLib/instancer/__init__.py +2052 -0
- fontTools/varLib/instancer/__main__.py +5 -0
- fontTools/varLib/instancer/featureVars.py +190 -0
- fontTools/varLib/instancer/names.py +388 -0
- fontTools/varLib/instancer/solver.py +309 -0
- fontTools/varLib/interpolatable.py +1209 -0
- fontTools/varLib/interpolatableHelpers.py +399 -0
- fontTools/varLib/interpolatablePlot.py +1269 -0
- fontTools/varLib/interpolatableTestContourOrder.py +82 -0
- fontTools/varLib/interpolatableTestStartingPoint.py +107 -0
- fontTools/varLib/interpolate_layout.py +124 -0
- fontTools/varLib/iup.c +19815 -0
- fontTools/varLib/iup.cp311-win32.pyd +0 -0
- fontTools/varLib/iup.py +490 -0
- fontTools/varLib/merger.py +1717 -0
- fontTools/varLib/models.py +642 -0
- fontTools/varLib/multiVarStore.py +253 -0
- fontTools/varLib/mutator.py +529 -0
- fontTools/varLib/mvar.py +40 -0
- fontTools/varLib/plot.py +238 -0
- fontTools/varLib/stat.py +149 -0
- fontTools/varLib/varStore.py +739 -0
- fontTools/voltLib/__init__.py +5 -0
- fontTools/voltLib/__main__.py +206 -0
- fontTools/voltLib/ast.py +452 -0
- fontTools/voltLib/error.py +12 -0
- fontTools/voltLib/lexer.py +99 -0
- fontTools/voltLib/parser.py +664 -0
- fontTools/voltLib/voltToFea.py +911 -0
- fonttools-4.60.2.data/data/share/man/man1/ttx.1 +225 -0
- fonttools-4.60.2.dist-info/METADATA +2250 -0
- fonttools-4.60.2.dist-info/RECORD +353 -0
- fonttools-4.60.2.dist-info/WHEEL +5 -0
- fonttools-4.60.2.dist-info/entry_points.txt +5 -0
- fonttools-4.60.2.dist-info/licenses/LICENSE +21 -0
- fonttools-4.60.2.dist-info/licenses/LICENSE.external +388 -0
- fonttools-4.60.2.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,399 @@
|
|
|
1
|
+
from fontTools.ttLib.ttGlyphSet import LerpGlyphSet
|
|
2
|
+
from fontTools.pens.basePen import AbstractPen, BasePen, DecomposingPen
|
|
3
|
+
from fontTools.pens.pointPen import AbstractPointPen, SegmentToPointPen
|
|
4
|
+
from fontTools.pens.recordingPen import RecordingPen, DecomposingRecordingPen
|
|
5
|
+
from fontTools.misc.transform import Transform
|
|
6
|
+
from collections import defaultdict, deque
|
|
7
|
+
from math import sqrt, copysign, atan2, pi
|
|
8
|
+
from enum import Enum
|
|
9
|
+
import itertools
|
|
10
|
+
|
|
11
|
+
import logging
|
|
12
|
+
|
|
13
|
+
log = logging.getLogger("fontTools.varLib.interpolatable")
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
class InterpolatableProblem:
|
|
17
|
+
NOTHING = "nothing"
|
|
18
|
+
MISSING = "missing"
|
|
19
|
+
OPEN_PATH = "open_path"
|
|
20
|
+
PATH_COUNT = "path_count"
|
|
21
|
+
NODE_COUNT = "node_count"
|
|
22
|
+
NODE_INCOMPATIBILITY = "node_incompatibility"
|
|
23
|
+
CONTOUR_ORDER = "contour_order"
|
|
24
|
+
WRONG_START_POINT = "wrong_start_point"
|
|
25
|
+
KINK = "kink"
|
|
26
|
+
UNDERWEIGHT = "underweight"
|
|
27
|
+
OVERWEIGHT = "overweight"
|
|
28
|
+
|
|
29
|
+
severity = {
|
|
30
|
+
MISSING: 1,
|
|
31
|
+
OPEN_PATH: 2,
|
|
32
|
+
PATH_COUNT: 3,
|
|
33
|
+
NODE_COUNT: 4,
|
|
34
|
+
NODE_INCOMPATIBILITY: 5,
|
|
35
|
+
CONTOUR_ORDER: 6,
|
|
36
|
+
WRONG_START_POINT: 7,
|
|
37
|
+
KINK: 8,
|
|
38
|
+
UNDERWEIGHT: 9,
|
|
39
|
+
OVERWEIGHT: 10,
|
|
40
|
+
NOTHING: 11,
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
def sort_problems(problems):
|
|
45
|
+
"""Sort problems by severity, then by glyph name, then by problem message."""
|
|
46
|
+
return dict(
|
|
47
|
+
sorted(
|
|
48
|
+
problems.items(),
|
|
49
|
+
key=lambda _: -min(
|
|
50
|
+
(
|
|
51
|
+
(InterpolatableProblem.severity[p["type"]] + p.get("tolerance", 0))
|
|
52
|
+
for p in _[1]
|
|
53
|
+
),
|
|
54
|
+
),
|
|
55
|
+
reverse=True,
|
|
56
|
+
)
|
|
57
|
+
)
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
def rot_list(l, k):
|
|
61
|
+
"""Rotate list by k items forward. Ie. item at position 0 will be
|
|
62
|
+
at position k in returned list. Negative k is allowed."""
|
|
63
|
+
return l[-k:] + l[:-k]
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
class PerContourPen(BasePen):
|
|
67
|
+
def __init__(self, Pen, glyphset=None):
|
|
68
|
+
BasePen.__init__(self, glyphset)
|
|
69
|
+
self._glyphset = glyphset
|
|
70
|
+
self._Pen = Pen
|
|
71
|
+
self._pen = None
|
|
72
|
+
self.value = []
|
|
73
|
+
|
|
74
|
+
def _moveTo(self, p0):
|
|
75
|
+
self._newItem()
|
|
76
|
+
self._pen.moveTo(p0)
|
|
77
|
+
|
|
78
|
+
def _lineTo(self, p1):
|
|
79
|
+
self._pen.lineTo(p1)
|
|
80
|
+
|
|
81
|
+
def _qCurveToOne(self, p1, p2):
|
|
82
|
+
self._pen.qCurveTo(p1, p2)
|
|
83
|
+
|
|
84
|
+
def _curveToOne(self, p1, p2, p3):
|
|
85
|
+
self._pen.curveTo(p1, p2, p3)
|
|
86
|
+
|
|
87
|
+
def _closePath(self):
|
|
88
|
+
self._pen.closePath()
|
|
89
|
+
self._pen = None
|
|
90
|
+
|
|
91
|
+
def _endPath(self):
|
|
92
|
+
self._pen.endPath()
|
|
93
|
+
self._pen = None
|
|
94
|
+
|
|
95
|
+
def _newItem(self):
|
|
96
|
+
self._pen = pen = self._Pen()
|
|
97
|
+
self.value.append(pen)
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
class PerContourOrComponentPen(PerContourPen):
|
|
101
|
+
def addComponent(self, glyphName, transformation):
|
|
102
|
+
self._newItem()
|
|
103
|
+
self.value[-1].addComponent(glyphName, transformation)
|
|
104
|
+
|
|
105
|
+
|
|
106
|
+
class SimpleRecordingPointPen(AbstractPointPen):
|
|
107
|
+
def __init__(self):
|
|
108
|
+
self.value = []
|
|
109
|
+
|
|
110
|
+
def beginPath(self, identifier=None, **kwargs):
|
|
111
|
+
pass
|
|
112
|
+
|
|
113
|
+
def endPath(self) -> None:
|
|
114
|
+
pass
|
|
115
|
+
|
|
116
|
+
def addPoint(self, pt, segmentType=None):
|
|
117
|
+
self.value.append((pt, False if segmentType is None else True))
|
|
118
|
+
|
|
119
|
+
|
|
120
|
+
def vdiff_hypot2(v0, v1):
|
|
121
|
+
s = 0
|
|
122
|
+
for x0, x1 in zip(v0, v1):
|
|
123
|
+
d = x1 - x0
|
|
124
|
+
s += d * d
|
|
125
|
+
return s
|
|
126
|
+
|
|
127
|
+
|
|
128
|
+
def vdiff_hypot2_complex(v0, v1):
|
|
129
|
+
s = 0
|
|
130
|
+
for x0, x1 in zip(v0, v1):
|
|
131
|
+
d = x1 - x0
|
|
132
|
+
s += d.real * d.real + d.imag * d.imag
|
|
133
|
+
# This does the same but seems to be slower:
|
|
134
|
+
# s += (d * d.conjugate()).real
|
|
135
|
+
return s
|
|
136
|
+
|
|
137
|
+
|
|
138
|
+
def matching_cost(G, matching):
|
|
139
|
+
return sum(G[i][j] for i, j in enumerate(matching))
|
|
140
|
+
|
|
141
|
+
|
|
142
|
+
def min_cost_perfect_bipartite_matching_scipy(G):
|
|
143
|
+
n = len(G)
|
|
144
|
+
rows, cols = linear_sum_assignment(G)
|
|
145
|
+
assert (rows == list(range(n))).all()
|
|
146
|
+
# Convert numpy array and integer to Python types,
|
|
147
|
+
# to ensure that this is JSON-serializable.
|
|
148
|
+
cols = list(int(e) for e in cols)
|
|
149
|
+
return list(cols), matching_cost(G, cols)
|
|
150
|
+
|
|
151
|
+
|
|
152
|
+
def min_cost_perfect_bipartite_matching_munkres(G):
|
|
153
|
+
n = len(G)
|
|
154
|
+
cols = [None] * n
|
|
155
|
+
for row, col in Munkres().compute(G):
|
|
156
|
+
cols[row] = col
|
|
157
|
+
return cols, matching_cost(G, cols)
|
|
158
|
+
|
|
159
|
+
|
|
160
|
+
def min_cost_perfect_bipartite_matching_bruteforce(G):
|
|
161
|
+
n = len(G)
|
|
162
|
+
|
|
163
|
+
if n > 6:
|
|
164
|
+
raise Exception("Install Python module 'munkres' or 'scipy >= 0.17.0'")
|
|
165
|
+
|
|
166
|
+
# Otherwise just brute-force
|
|
167
|
+
permutations = itertools.permutations(range(n))
|
|
168
|
+
best = list(next(permutations))
|
|
169
|
+
best_cost = matching_cost(G, best)
|
|
170
|
+
for p in permutations:
|
|
171
|
+
cost = matching_cost(G, p)
|
|
172
|
+
if cost < best_cost:
|
|
173
|
+
best, best_cost = list(p), cost
|
|
174
|
+
return best, best_cost
|
|
175
|
+
|
|
176
|
+
|
|
177
|
+
# Prefer `scipy.optimize.linear_sum_assignment` for performance.
|
|
178
|
+
# `Munkres` is also supported as a fallback for minimalistic systems
|
|
179
|
+
# where installing SciPy is not feasible.
|
|
180
|
+
try:
|
|
181
|
+
from scipy.optimize import linear_sum_assignment
|
|
182
|
+
|
|
183
|
+
min_cost_perfect_bipartite_matching = min_cost_perfect_bipartite_matching_scipy
|
|
184
|
+
except ImportError:
|
|
185
|
+
try:
|
|
186
|
+
from munkres import Munkres
|
|
187
|
+
|
|
188
|
+
min_cost_perfect_bipartite_matching = (
|
|
189
|
+
min_cost_perfect_bipartite_matching_munkres
|
|
190
|
+
)
|
|
191
|
+
except ImportError:
|
|
192
|
+
min_cost_perfect_bipartite_matching = (
|
|
193
|
+
min_cost_perfect_bipartite_matching_bruteforce
|
|
194
|
+
)
|
|
195
|
+
|
|
196
|
+
|
|
197
|
+
def contour_vector_from_stats(stats):
|
|
198
|
+
# Don't change the order of items here.
|
|
199
|
+
# It's okay to add to the end, but otherwise, other
|
|
200
|
+
# code depends on it. Search for "covariance".
|
|
201
|
+
size = sqrt(abs(stats.area))
|
|
202
|
+
return (
|
|
203
|
+
copysign((size), stats.area),
|
|
204
|
+
stats.meanX,
|
|
205
|
+
stats.meanY,
|
|
206
|
+
stats.stddevX * 2,
|
|
207
|
+
stats.stddevY * 2,
|
|
208
|
+
stats.correlation * size,
|
|
209
|
+
)
|
|
210
|
+
|
|
211
|
+
|
|
212
|
+
def matching_for_vectors(m0, m1):
|
|
213
|
+
n = len(m0)
|
|
214
|
+
|
|
215
|
+
identity_matching = list(range(n))
|
|
216
|
+
|
|
217
|
+
costs = [[vdiff_hypot2(v0, v1) for v1 in m1] for v0 in m0]
|
|
218
|
+
(
|
|
219
|
+
matching,
|
|
220
|
+
matching_cost,
|
|
221
|
+
) = min_cost_perfect_bipartite_matching(costs)
|
|
222
|
+
identity_cost = sum(costs[i][i] for i in range(n))
|
|
223
|
+
return matching, matching_cost, identity_cost
|
|
224
|
+
|
|
225
|
+
|
|
226
|
+
def points_characteristic_bits(points):
|
|
227
|
+
bits = 0
|
|
228
|
+
for pt, b in reversed(points):
|
|
229
|
+
bits = (bits << 1) | b
|
|
230
|
+
return bits
|
|
231
|
+
|
|
232
|
+
|
|
233
|
+
_NUM_ITEMS_PER_POINTS_COMPLEX_VECTOR = 4
|
|
234
|
+
|
|
235
|
+
|
|
236
|
+
def points_complex_vector(points):
|
|
237
|
+
vector = []
|
|
238
|
+
if not points:
|
|
239
|
+
return vector
|
|
240
|
+
points = [complex(*pt) for pt, _ in points]
|
|
241
|
+
n = len(points)
|
|
242
|
+
assert _NUM_ITEMS_PER_POINTS_COMPLEX_VECTOR == 4
|
|
243
|
+
points.extend(points[: _NUM_ITEMS_PER_POINTS_COMPLEX_VECTOR - 1])
|
|
244
|
+
while len(points) < _NUM_ITEMS_PER_POINTS_COMPLEX_VECTOR:
|
|
245
|
+
points.extend(points[: _NUM_ITEMS_PER_POINTS_COMPLEX_VECTOR - 1])
|
|
246
|
+
for i in range(n):
|
|
247
|
+
# The weights are magic numbers.
|
|
248
|
+
|
|
249
|
+
# The point itself
|
|
250
|
+
p0 = points[i]
|
|
251
|
+
vector.append(p0)
|
|
252
|
+
|
|
253
|
+
# The vector to the next point
|
|
254
|
+
p1 = points[i + 1]
|
|
255
|
+
d0 = p1 - p0
|
|
256
|
+
vector.append(d0 * 3)
|
|
257
|
+
|
|
258
|
+
# The turn vector
|
|
259
|
+
p2 = points[i + 2]
|
|
260
|
+
d1 = p2 - p1
|
|
261
|
+
vector.append(d1 - d0)
|
|
262
|
+
|
|
263
|
+
# The angle to the next point, as a cross product;
|
|
264
|
+
# Square root of, to match dimentionality of distance.
|
|
265
|
+
cross = d0.real * d1.imag - d0.imag * d1.real
|
|
266
|
+
cross = copysign(sqrt(abs(cross)), cross)
|
|
267
|
+
vector.append(cross * 4)
|
|
268
|
+
|
|
269
|
+
return vector
|
|
270
|
+
|
|
271
|
+
|
|
272
|
+
def add_isomorphisms(points, isomorphisms, reverse):
|
|
273
|
+
reference_bits = points_characteristic_bits(points)
|
|
274
|
+
n = len(points)
|
|
275
|
+
|
|
276
|
+
# if points[0][0] == points[-1][0]:
|
|
277
|
+
# abort
|
|
278
|
+
|
|
279
|
+
if reverse:
|
|
280
|
+
points = points[::-1]
|
|
281
|
+
bits = points_characteristic_bits(points)
|
|
282
|
+
else:
|
|
283
|
+
bits = reference_bits
|
|
284
|
+
|
|
285
|
+
vector = points_complex_vector(points)
|
|
286
|
+
|
|
287
|
+
assert len(vector) % n == 0
|
|
288
|
+
mult = len(vector) // n
|
|
289
|
+
mask = (1 << n) - 1
|
|
290
|
+
|
|
291
|
+
for i in range(n):
|
|
292
|
+
b = ((bits << (n - i)) & mask) | (bits >> i)
|
|
293
|
+
if b == reference_bits:
|
|
294
|
+
isomorphisms.append(
|
|
295
|
+
(rot_list(vector, -i * mult), n - 1 - i if reverse else i, reverse)
|
|
296
|
+
)
|
|
297
|
+
|
|
298
|
+
|
|
299
|
+
def find_parents_and_order(glyphsets, locations, *, discrete_axes=set()):
|
|
300
|
+
parents = [None] + list(range(len(glyphsets) - 1))
|
|
301
|
+
order = list(range(len(glyphsets)))
|
|
302
|
+
if locations:
|
|
303
|
+
# Order base master first
|
|
304
|
+
bases = [
|
|
305
|
+
i
|
|
306
|
+
for i, l in enumerate(locations)
|
|
307
|
+
if all(v == 0 for k, v in l.items() if k not in discrete_axes)
|
|
308
|
+
]
|
|
309
|
+
if bases:
|
|
310
|
+
logging.info("Found %s base masters: %s", len(bases), bases)
|
|
311
|
+
else:
|
|
312
|
+
logging.warning("No base master location found")
|
|
313
|
+
|
|
314
|
+
# Form a minimum spanning tree of the locations
|
|
315
|
+
try:
|
|
316
|
+
from scipy.sparse.csgraph import minimum_spanning_tree
|
|
317
|
+
|
|
318
|
+
graph = [[0] * len(locations) for _ in range(len(locations))]
|
|
319
|
+
axes = set()
|
|
320
|
+
for l in locations:
|
|
321
|
+
axes.update(l.keys())
|
|
322
|
+
axes = sorted(axes)
|
|
323
|
+
vectors = [tuple(l.get(k, 0) for k in axes) for l in locations]
|
|
324
|
+
for i, j in itertools.combinations(range(len(locations)), 2):
|
|
325
|
+
i_discrete_location = {
|
|
326
|
+
k: v for k, v in zip(axes, vectors[i]) if k in discrete_axes
|
|
327
|
+
}
|
|
328
|
+
j_discrete_location = {
|
|
329
|
+
k: v for k, v in zip(axes, vectors[j]) if k in discrete_axes
|
|
330
|
+
}
|
|
331
|
+
if i_discrete_location != j_discrete_location:
|
|
332
|
+
continue
|
|
333
|
+
graph[i][j] = vdiff_hypot2(vectors[i], vectors[j])
|
|
334
|
+
|
|
335
|
+
tree = minimum_spanning_tree(graph, overwrite=True)
|
|
336
|
+
rows, cols = tree.nonzero()
|
|
337
|
+
graph = defaultdict(set)
|
|
338
|
+
for row, col in zip(rows, cols):
|
|
339
|
+
graph[row].add(col)
|
|
340
|
+
graph[col].add(row)
|
|
341
|
+
|
|
342
|
+
# Traverse graph from the base and assign parents
|
|
343
|
+
parents = [None] * len(locations)
|
|
344
|
+
order = []
|
|
345
|
+
visited = set()
|
|
346
|
+
queue = deque(bases)
|
|
347
|
+
while queue:
|
|
348
|
+
i = queue.popleft()
|
|
349
|
+
visited.add(i)
|
|
350
|
+
order.append(i)
|
|
351
|
+
for j in sorted(graph[i]):
|
|
352
|
+
if j not in visited:
|
|
353
|
+
parents[j] = i
|
|
354
|
+
queue.append(j)
|
|
355
|
+
assert len(order) == len(
|
|
356
|
+
parents
|
|
357
|
+
), "Not all masters are reachable; report an issue"
|
|
358
|
+
|
|
359
|
+
except ImportError:
|
|
360
|
+
pass
|
|
361
|
+
|
|
362
|
+
log.info("Parents: %s", parents)
|
|
363
|
+
log.info("Order: %s", order)
|
|
364
|
+
return parents, order
|
|
365
|
+
|
|
366
|
+
|
|
367
|
+
def transform_from_stats(stats, inverse=False):
|
|
368
|
+
# https://cookierobotics.com/007/
|
|
369
|
+
a = stats.varianceX
|
|
370
|
+
b = stats.covariance
|
|
371
|
+
c = stats.varianceY
|
|
372
|
+
|
|
373
|
+
delta = (((a - c) * 0.5) ** 2 + b * b) ** 0.5
|
|
374
|
+
lambda1 = (a + c) * 0.5 + delta # Major eigenvalue
|
|
375
|
+
lambda2 = (a + c) * 0.5 - delta # Minor eigenvalue
|
|
376
|
+
theta = atan2(lambda1 - a, b) if b != 0 else (pi * 0.5 if a < c else 0)
|
|
377
|
+
trans = Transform()
|
|
378
|
+
|
|
379
|
+
if lambda2 < 0:
|
|
380
|
+
# XXX This is a hack.
|
|
381
|
+
# The problem is that the covariance matrix is singular.
|
|
382
|
+
# This happens when the contour is a line, or a circle.
|
|
383
|
+
# In that case, the covariance matrix is not a good
|
|
384
|
+
# representation of the contour.
|
|
385
|
+
# We should probably detect this earlier and avoid
|
|
386
|
+
# computing the covariance matrix in the first place.
|
|
387
|
+
# But for now, we just avoid the division by zero.
|
|
388
|
+
lambda2 = 0
|
|
389
|
+
|
|
390
|
+
if inverse:
|
|
391
|
+
trans = trans.translate(-stats.meanX, -stats.meanY)
|
|
392
|
+
trans = trans.rotate(-theta)
|
|
393
|
+
trans = trans.scale(1 / sqrt(lambda1), 1 / sqrt(lambda2))
|
|
394
|
+
else:
|
|
395
|
+
trans = trans.scale(sqrt(lambda1), sqrt(lambda2))
|
|
396
|
+
trans = trans.rotate(theta)
|
|
397
|
+
trans = trans.translate(stats.meanX, stats.meanY)
|
|
398
|
+
|
|
399
|
+
return trans
|