"""Calculate the area of a glyph.""" from __future__ import print_function, division, absolute_import from fontTools.misc.py23 import * from fontTools.pens.basePen import BasePen class AreaPen(BasePen): def __init__(self, glyphset=None): BasePen.__init__(self, glyphset) self.value = 0 def _moveTo(self, p0): """Remember the first point in this contour, in case it's closed. Also set the initial value for p0 in this contour, which will always refer to the most recent point. """ self._p0 = self._startPoint = p0 def _lineTo(self, p1): """Add the signed area beneath the line from the latest point to this one. Signed areas cancel each other based on the horizontal direction of the line. """ x0, y0 = self._p0 x1, y1 = p1 self.value -= (x1 - x0) * (y1 + y0) * .5 self._p0 = p1 def _qCurveToOne(self, p1, p2): """Add the signed area of this quadratic curve. https://github.com/Pomax/bezierinfo/issues/44 """ p0 = self._p0 x0, y0 = p0[0], p0[1] x1, y1 = p1[0] - x0, p1[1] - y0 x2, y2 = p2[0] - x0, p2[1] - y0 self.value -= (x2 * y1 - x1 * y2) / 3 self._lineTo(p2) self._p0 = p2 def _curveToOne(self, p1, p2, p3): """Add the signed area of this cubic curve. https://github.com/Pomax/bezierinfo/issues/44 """ p0 = self._p0 x0, y0 = p0[0], p0[1] x1, y1 = p1[0] - x0, p1[1] - y0 x2, y2 = p2[0] - x0, p2[1] - y0 x3, y3 = p3[0] - x0, p3[1] - y0 self.value -= ( x1 * ( - y2 - y3) + x2 * (y1 - 2*y3) + x3 * (y1 + 2*y2 ) ) * 0.15 self._lineTo(p3) self._p0 = p3 def _closePath(self): """Add the area beneath this contour's closing line.""" self._lineTo(self._startPoint) del self._p0, self._startPoint def _endPath(self): """Area is not defined for open contours. Single-point open contours, which often represent anchors, are allowed. """ if self._p0 != self.startPoint: raise NotImplementedError del self._p0, self._startPoint