Source code for bdflib.writer

# bdflib, a library for working with BDF font files
# Copyright (C) 2009-2022, Timothy Allen
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.

import math


def _quote_property_value(val):
    if isinstance(val, int):
        return b"%d" % val
    else:
        return b'"' + bytes(val).replace(b'"', b'""') + b'"'


[docs]def write_bdf(font, stream): """ Write a BDF-format font to the given stream. :param Font font: The font to write to the given stream. :param stream: The stream that will receive the font. ``stream`` must be an object with at ``.write()`` method that takes a :class:`bytes`. If you want to write to an actual file, make sure you use the 'b' flag:: bdflib.writer.write_bdf(font, open(path, 'wb')) """ # The font bounding box is the union of glyph bounding boxes. font_bbX = 0 font_bbY = 0 font_bbW = 0 font_bbH = 0 for g in font.glyphs: new_bbX = min(font_bbX, g.bbX) new_bbY = min(font_bbY, g.bbY) new_bbW = max(font_bbX + font_bbW, g.bbX + g.bbW) - new_bbX new_bbH = max(font_bbY + font_bbH, g.bbY + g.bbH) - new_bbY (font_bbX, font_bbY, font_bbW, font_bbH) = ( new_bbX, new_bbY, new_bbW, new_bbH, ) font_pixel_size = math.ceil(font.ydpi * font.ptSize / 72.0) # Write the basic header. stream.write(b"STARTFONT 2.1\n") stream.write(b"FONT ") stream.write(font.name) stream.write(b"\n") stream.write(b"SIZE %g %d %d\n" % (font.ptSize, font.xdpi, font.ydpi)) stream.write( b"FONTBOUNDINGBOX %d %d %d %d\n" % (font_bbW, font_bbH, font_bbX, font_bbY) ) # Write the properties stream.write(b"STARTPROPERTIES %d\n" % (len(font.properties),)) keys = sorted(font.properties.keys()) for key in keys: stream.write(key) stream.write(b" ") stream.write(_quote_property_value(font.properties[key])) stream.write(b"\n") stream.write(b"ENDPROPERTIES\n") # Write out the glyphs stream.write(b"CHARS %d\n" % (len(font.glyphs),)) for glyph in font.glyphs: scalable_width = int(1000.0 * glyph.advance / font_pixel_size) stream.write(b"STARTCHAR ") stream.write(glyph.name) stream.write(b"\n") stream.write(b"ENCODING %d\n" % (glyph.codepoint,)) stream.write(b"SWIDTH %d 0\n" % (scalable_width,)) stream.write(b"DWIDTH %d 0\n" % (glyph.advance,)) stream.write( b"BBX %d %d %d %d\n" % (glyph.bbW, glyph.bbH, glyph.bbX, glyph.bbY) ) stream.write(b"BITMAP\n") # How many bytes do we need to represent the bits in each row? rowWidth, extraBits = divmod(glyph.bbW, 8) # How many bits of padding do we need to round up to a full byte? if extraBits > 0: rowWidth += 1 paddingBits = 8 - extraBits else: paddingBits = 0 # glyph.data goes bottom-to-top # like any proper coordinate system does, # but rows wants to be top-to-bottom # like any proper stream-output. for row in reversed(glyph.data): # rowWidth is the number of bytes, # but Python wants the number of nybbles, # so multiply by 2. stream.write(b"%0*X\n" % (rowWidth * 2, row << paddingBits)) stream.write(b"ENDCHAR\n") stream.write(b"ENDFONT\n")