freetypePen
Pen to rasterize paths with FreeType.
- class fontTools.pens.freetypePen.FreeTypePen(glyphSet)[source]
Pen to rasterize paths with FreeType. Requires freetype-py module.
Constructs
FT_Outline
from the paths, and renders it within a bitmap buffer.For
array()
andshow()
, numpy and matplotlib must be installed. Forimage()
, Pillow is required. Each module is lazily loaded when the corresponding method is called.- Parameters
glyphSet – a dictionary of drawable glyph objects keyed by name used to resolve component references in composite glyphs.
- Examples
If numpy and matplotlib is available, the following code will show the glyph image of fi in a new window:
from fontTools.ttLib import TTFont from fontTools.pens.freetypePen import FreeTypePen from fontTools.misc.transform import Offset pen = FreeTypePen(None) font = TTFont('SourceSansPro-Regular.otf') glyph = font.getGlyphSet()['fi'] glyph.draw(pen) width, ascender, descender = glyph.width, font['OS/2'].usWinAscent, -font['OS/2'].usWinDescent height = ascender - descender pen.show(width=width, height=height, transform=Offset(0, -descender))
Combining with uharfbuzz, you can typeset a chunk of glyphs in a pen:
import uharfbuzz as hb from fontTools.pens.freetypePen import FreeTypePen from fontTools.pens.transformPen import TransformPen from fontTools.misc.transform import Offset en1, en2, ar, ja = 'Typesetting', 'Jeff', 'صف الحروف', 'たいぷせっと' for text, font_path, direction, typo_ascender, typo_descender, vhea_ascender, vhea_descender, contain, features in ( (en1, 'NotoSans-Regular.ttf', 'ltr', 2189, -600, None, None, False, {"kern": True, "liga": True}), (en2, 'NotoSans-Regular.ttf', 'ltr', 2189, -600, None, None, True, {"kern": True, "liga": True}), (ar, 'NotoSansArabic-Regular.ttf', 'rtl', 1374, -738, None, None, False, {"kern": True, "liga": True}), (ja, 'NotoSansJP-Regular.otf', 'ltr', 880, -120, 500, -500, False, {"palt": True, "kern": True}), (ja, 'NotoSansJP-Regular.otf', 'ttb', 880, -120, 500, -500, False, {"vert": True, "vpal": True, "vkrn": True}) ): blob = hb.Blob.from_file_path(font_path) face = hb.Face(blob) font = hb.Font(face) buf = hb.Buffer() buf.direction = direction buf.add_str(text) buf.guess_segment_properties() hb.shape(font, buf, features) x, y = 0, 0 pen = FreeTypePen(None) for info, pos in zip(buf.glyph_infos, buf.glyph_positions): gid = info.codepoint transformed = TransformPen(pen, Offset(x + pos.x_offset, y + pos.y_offset)) font.draw_glyph_with_pen(gid, transformed) x += pos.x_advance y += pos.y_advance offset, width, height = None, None, None if direction in ('ltr', 'rtl'): offset = (0, -typo_descender) width = x height = typo_ascender - typo_descender else: offset = (-vhea_descender, -y) width = vhea_ascender - vhea_descender height = -y pen.show(width=width, height=height, transform=Offset(*offset), contain=contain)
For Jupyter Notebook, the rendered image will be displayed in a cell if you replace
show()
withimage()
in the examples.
- addComponent(glyphName, transformation)
Transform the points of the base glyph and draw it onto self.
- addVarComponent(glyphName, transformation, location)
Add a VarComponent sub glyph. The ‘transformation’ argument must be a DecomposedTransform from the fontTools.misc.transform module, and the ‘location’ argument must be a dictionary mapping axis tags to their locations.
- array(width=None, height=None, transform=None, contain=False, evenOdd=False)[source]
Returns the rendered contours as a numpy array. Requires numpy.
- Parameters
width – Image width of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
height – Image height of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
transform – An optional 6-tuple containing an affine transformation, or a
Transform
object from thefontTools.misc.transform
module. The bitmap size is not affected by this matrix.contain – If
True
, the image size will be automatically expanded so that it fits to the bounding box of the paths. Useful for rendering glyphs with negative sidebearings without clipping.evenOdd – Pass
True
for even-odd fill instead of non-zero.
- Returns
A
numpy.ndarray
object with a shape of(height, width)
. Each element takes a value in the range of[0.0, 1.0]
.- Notes
The image size should always be given explicitly if you need to get a proper glyph image. When
width
andheight
are omitted, it forcifully fits to the bounding box and the side bearings get cropped. If you pass0
to bothwidth
andheight
and setcontain
toTrue
, it expands to the bounding box while maintaining the origin of the contours, meaning that LSB will be maintained but RSB won’t. The difference between the two becomes more obvious when rotate or skew transformation is applied.- Example
>> pen = FreeTypePen(None) >> glyph.draw(pen) >> arr = pen.array(width=500, height=1000) >> type(a), a.shape (<class 'numpy.ndarray'>, (1000, 500))
- property bbox
Computes the exact bounding box of an outline.
- Returns
A tuple of
(xMin, yMin, xMax, yMax)
.
- buffer(width=None, height=None, transform=None, contain=False, evenOdd=False)[source]
Renders the current contours within a bitmap buffer.
- Parameters
width – Image width of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
height – Image height of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
transform – An optional 6-tuple containing an affine transformation, or a
Transform
object from thefontTools.misc.transform
module. The bitmap size is not affected by this matrix.contain – If
True
, the image size will be automatically expanded so that it fits to the bounding box of the paths. Useful for rendering glyphs with negative sidebearings without clipping.evenOdd – Pass
True
for even-odd fill instead of non-zero.
- Returns
A tuple of
(buffer, size)
, wherebuffer
is abytes
object of the resulted bitmap andsize
is a 2-tuple of its dimension.- Notes
The image size should always be given explicitly if you need to get a proper glyph image. When
width
andheight
are omitted, it forcifully fits to the bounding box and the side bearings get cropped. If you pass0
to bothwidth
andheight
and setcontain
toTrue
, it expands to the bounding box while maintaining the origin of the contours, meaning that LSB will be maintained but RSB won’t. The difference between the two becomes more obvious when rotate or skew transformation is applied.- Example
>> pen = FreeTypePen(None) >> glyph.draw(pen) >> buf, size = pen.buffer(width=500, height=1000) >> type(buf), len(buf), size (<class 'bytes'>, 500000, (500, 1000))
- property cbox
Returns an outline’s ‘control box’.
- Returns
A tuple of
(xMin, yMin, xMax, yMax)
.
- closePath()
Close the current sub path. You must call either pen.closePath() or pen.endPath() after each sub path.
- curveTo(*points)
Draw a cubic bezier with an arbitrary number of control points.
The last point specified is on-curve, all others are off-curve (control) points. If the number of control points is > 2, the segment is split into multiple bezier segments. This works like this:
Let n be the number of control points (which is the number of arguments to this call minus 1). If n==2, a plain vanilla cubic bezier is drawn. If n==1, we fall back to a quadratic segment and if n==0 we draw a straight line. It gets interesting when n>2: n-1 PostScript-style cubic segments will be drawn as if it were one curve. See decomposeSuperBezierSegment().
The conversion algorithm used for n>2 is inspired by NURB splines, and is conceptually equivalent to the TrueType “implied points” principle. See also decomposeQuadraticSegment().
- endPath()
End the current sub path, but don’t close it. You must call either pen.closePath() or pen.endPath() after each sub path.
- image(width=None, height=None, transform=None, contain=False, evenOdd=False)[source]
Returns the rendered contours as a PIL image. Requires Pillow. Can be used to display a glyph image in Jupyter Notebook.
- Parameters
width – Image width of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
height – Image height of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
transform – An optional 6-tuple containing an affine transformation, or a
Transform
object from thefontTools.misc.transform
module. The bitmap size is not affected by this matrix.contain – If
True
, the image size will be automatically expanded so that it fits to the bounding box of the paths. Useful for rendering glyphs with negative sidebearings without clipping.evenOdd – Pass
True
for even-odd fill instead of non-zero.
- Returns
A
PIL.image
object. The image is filled in black with alpha channel obtained from the rendered bitmap.- Notes
The image size should always be given explicitly if you need to get a proper glyph image. When
width
andheight
are omitted, it forcifully fits to the bounding box and the side bearings get cropped. If you pass0
to bothwidth
andheight
and setcontain
toTrue
, it expands to the bounding box while maintaining the origin of the contours, meaning that LSB will be maintained but RSB won’t. The difference between the two becomes more obvious when rotate or skew transformation is applied.- Example
>> pen = FreeTypePen(None) >> glyph.draw(pen) >> img = pen.image(width=500, height=1000) >> type(img), img.size (<class 'PIL.Image.Image'>, (500, 1000))
- lineTo(pt)
Draw a straight line from the current point to ‘pt’.
- property log
- moveTo(pt)
Begin a new sub path, set the current point to ‘pt’. You must end each sub path with a call to pen.closePath() or pen.endPath().
- outline(transform=None, evenOdd=False)[source]
Converts the current contours to
FT_Outline
.- Parameters
transform – An optional 6-tuple containing an affine transformation, or a
Transform
object from thefontTools.misc.transform
module.evenOdd – Pass
True
for even-odd fill instead of non-zero.
- qCurveTo(*points)
Draw a whole string of quadratic curve segments.
The last point specified is on-curve, all others are off-curve points.
This method implements TrueType-style curves, breaking up curves using ‘implied points’: between each two consequtive off-curve points, there is one implied point exactly in the middle between them. See also decomposeQuadraticSegment().
The last argument (normally the on-curve point) may be None. This is to support contours that have NO on-curve points (a rarely seen feature of TrueType outlines).
- show(width=None, height=None, transform=None, contain=False, evenOdd=False)[source]
Plots the rendered contours with pyplot. Requires numpy and matplotlib.
- Parameters
width – Image width of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
height – Image height of the bitmap in pixels. If omitted, it automatically fits to the bounding box of the contours.
transform – An optional 6-tuple containing an affine transformation, or a
Transform
object from thefontTools.misc.transform
module. The bitmap size is not affected by this matrix.contain – If
True
, the image size will be automatically expanded so that it fits to the bounding box of the paths. Useful for rendering glyphs with negative sidebearings without clipping.evenOdd – Pass
True
for even-odd fill instead of non-zero.
- Notes
The image size should always be given explicitly if you need to get a proper glyph image. When
width
andheight
are omitted, it forcifully fits to the bounding box and the side bearings get cropped. If you pass0
to bothwidth
andheight
and setcontain
toTrue
, it expands to the bounding box while maintaining the origin of the contours, meaning that LSB will be maintained but RSB won’t. The difference between the two becomes more obvious when rotate or skew transformation is applied.- Example
>> pen = FreeTypePen(None) >> glyph.draw(pen) >> pen.show(width=500, height=1000)
- skipMissingComponents = True