如何在PIL.ImageFont中使用我自己的位图字体?

时间:2018-10-27 11:34:47

标签: python fonts bitmap python-imaging-library

我创建了一个位图字体,基本上是一个256x256 png图像,其中每个字符占用8x8瓦片。我想将其与Pillow一起用作ImageFont,但Pillow文档中对此没有任何信息。它说我可以加载这样的位图字体

font = ImageFont.load("arial.pil")

但是“ PIL使用其自己的字体文件格式来存储位图字体。”所以我想png文件将无法工作。如何告诉PIL使用上述位图以及每个字符在何处?

1 个答案:

答案 0 :(得分:2)

答案不是一个完整的问题,但评论太多了,这可能会有用或促使其他人解决另外60%的问题:-)

如果其他人提出更好的建议,我可以将其删除...

您可以转到Pillow repository on Github并下载代码的ZIP文件。

如果您到那里走走,您会发现两件事似乎是相辅相成的,即.PIL文件和.PBM文件。


Tests/fonts中有一个名为10x20.pbm的文件,如果您在其中查看它实际上是一个PNG文件。因此,如果将其名称更改为10x20.png,则可以查看它,它看起来像这样:

enter image description here

顺便说一句,如果要将其分成10x20大小的块,每个块一个字母,则可以在Terminal中使用 ImageMagick ,如下所示:

convert 10x20.pbm -crop 10x20 char_%d.png

,您将获得一堆名为char_0.pngchar_1.png等的文件。前四个看起来像这样:

enter image description here


如果您查看src/PIL/FontFile.py,则此代码似乎知道如何访问/生成字体的度量标准:

#
# The Python Imaging Library
# $Id$
#
# base class for raster font file parsers
#
# history:
# 1997-06-05 fl   created
# 1997-08-19 fl   restrict image width
#
# Copyright (c) 1997-1998 by Secret Labs AB
# Copyright (c) 1997-1998 by Fredrik Lundh
#
# See the README file for information on usage and redistribution.
#

from __future__ import print_function

import os
from . import Image, _binary

WIDTH = 800


def puti16(fp, values):
    # write network order (big-endian) 16-bit sequence
    for v in values:
        if v < 0:
            v += 65536
        fp.write(_binary.o16be(v))


##
# Base class for raster font file handlers.

class FontFile(object):

    bitmap = None

    def __init__(self):

        self.info = {}
        self.glyph = [None] * 256

    def __getitem__(self, ix):
        return self.glyph[ix]

    def compile(self):
        "Create metrics and bitmap"

        if self.bitmap:
            return

        # create bitmap large enough to hold all data
        h = w = maxwidth = 0
        lines = 1
        for glyph in self:
            if glyph:
                d, dst, src, im = glyph
                h = max(h, src[3] - src[1])
                w = w + (src[2] - src[0])
                if w > WIDTH:
                    lines += 1
                    w = (src[2] - src[0])
                maxwidth = max(maxwidth, w)

        xsize = maxwidth
        ysize = lines * h

        if xsize == 0 and ysize == 0:
            return ""

        self.ysize = h

        # paste glyphs into bitmap
        self.bitmap = Image.new("1", (xsize, ysize))
        self.metrics = [None] * 256
        x = y = 0
        for i in range(256):
            glyph = self[i]
            if glyph:
                d, dst, src, im = glyph
                xx = src[2] - src[0]
                # yy = src[3] - src[1]
                x0, y0 = x, y
                x = x + xx
                if x > WIDTH:
                    x, y = 0, y + h
                    x0, y0 = x, y
                    x = xx
                s = src[0] + x0, src[1] + y0, src[2] + x0, src[3] + y0
                self.bitmap.paste(im.crop(src), s)
                self.metrics[i] = d, dst, s

    def save(self, filename):
        "Save font"

        self.compile()

        # font data
        self.bitmap.save(os.path.splitext(filename)[0] + ".pbm", "PNG")

        # font metrics
        with open(os.path.splitext(filename)[0] + ".pil", "wb") as fp:
            fp.write(b"PILfont\n")
            fp.write((";;;;;;%d;\n" % self.ysize).encode('ascii'))  # HACK!!!
            fp.write(b"DATA\n")
            for id in range(256):
                m = self.metrics[id]
                if not m:
                    puti16(fp, [0] * 10)
                else:
                    puti16(fp, m[0] + m[1] + m[2])

因此,希望有人有时间/知道如何将这两者放在一起,以使您能够为PNG生成度量标准文件。我认为您只需要为PNG执行该代码的最后10行的代码即可。

您似乎可以简单地复制标题的23个字节,然后有256个“条目” ,即256个字形中的每一个为1。每个条目中都有10个数字,每个数字都是16位大字节序。

让我们看一下标题:

dd if=10x20.pil bs=23 count=1| xxd -c23 | more
00000000: 5049 4c66 6f6e 740a 3b3b 3b3b 3b3b 3230 3b0a 4441 5441 0a  PILfont.;;;;;;20;.DATA.

然后,您可以使用以下命令查看条目,以跳过标题和分组:

dd if=10x20.pil bs=23 iseek=1| xxd -g2 -c20

给出:

enter image description here

第1列似乎是字形的宽度。

第7列是图像中字形左边缘的x偏移,第9列是图像中字形右边缘的x偏移。因此,您会看到每行的第7列与前一行的第9列相同,即字形在图像上彼此邻接。

如果您从文件的更下方查看此摘录,则可以看到它在摘录中间(红色标记)在输出图像中开始了新的字形行。这说明位图的宽度不应超过800像素,第8列是位图文件中字形顶部的y偏移,第10列是位图中字形底部的y偏移。 。您应该看到,当位图文件中一行新的字形行开始时,x变为零,第8列采用第10列中的前一个值。

enter image description here