blob: c98c839c776cb98f7aacc4e7d97c8e68c1fddbaf [file] [log] [blame]
"""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