-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathhelpers.py
93 lines (76 loc) · 2.97 KB
/
helpers.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
from functools import cache
from fontTools.pens.basePen import AbstractPen, BasePen
from fontTools.pens.basePen import BasePen
from fontTools.pens.momentsPen import MomentsPen
from fontTools.pens.ttGlyphPen import TTGlyphPen
from fontTools.pens.recordingPen import DecomposingRecordingPen
from fontTools.ttLib import getTableModule, TTFont
from picosvg.geometric_types import Rect
from picosvg.svg_types import SVGPath
def color(hex: str):
return getTableModule('CPAL').Color.fromHex(hex)
def make_glyph_name(glyph_name, nth):
return f"{glyph_name}_{nth}"
@cache
def bbox(path: str) -> Rect:
return SVGPath(d=path).bounding_box()
def draw(svg_path: SVGPath, pen: AbstractPen):
for cmd in svg_path.as_cmd_seq():
match cmd:
case ("M", pt):
pen.moveTo(pt)
case ("L", pt):
pen.lineTo(pt)
case ("Q", (x1, y1, x, y)):
pen.qCurveTo((x1, y1), (x, y))
case ("Z", ()):
pen.closePath()
case _:
raise ValueError(f"TODO {cmd}")
def make_glyph(font: TTFont, original_glyph_name: str, new_glyph_name: str, path: SVGPath):
assert new_glyph_name not in font.getGlyphOrder(), new_glyph_name
glyphs = font.getGlyphSet()
glyph_pen = TTGlyphPen(glyphs)
draw(path, glyph_pen)
new_glyph = glyph_pen.glyph()
# Adding a glyph is surprisingly tricksy
font.getGlyphOrder().append(new_glyph_name)
glyf = font["glyf"]
glyf.glyphs[new_glyph_name] = new_glyph
new_glyph.recalcBounds(glyf)
# Copy advance, use our own xMin
font["hmtx"].metrics[new_glyph_name] = (font["hmtx"].metrics[original_glyph_name][0], new_glyph.xMin)
# Records each subpath separately
#
# This is handy if you want to know things like which subpaths
# are "cut out" rather than filled.
class SubpathPen(BasePen):
def __init__(self, glyphset=None):
BasePen.__init__(self, glyphset)
self.glyphset = glyphset
self.recordings = []
self.active_recording = False
def _recording(self) -> DecomposingRecordingPen:
if not self.active_recording:
self.recordings.append(DecomposingRecordingPen(self.glyphset))
self.active_recording = True
return self.recordings[-1]
def moveTo(self, p0):
self._recording().moveTo(p0)
def lineTo(self, p1):
self._recording().lineTo(p1)
def qCurveTo(self, *points):
self._recording().qCurveTo(*points)
def curveTo(self, *points):
self._recording().curveTo(*points)
def closePath(self):
self._recording().closePath()
self.active_recording = False
def endPath(self):
self._recording().endPath()
self.active_recording = False
def addComponent(self, glyphName, transformation):
self.recordings.append(DecomposingRecordingPen(self.glyphset))
self.active_recording = True
self._recording().addComponent(glyphName, transformation)
self.active_recording = False