Как использовать свой собственный растровый шрифт в PIL.ImageFont? - PullRequest
0 голосов
/ 27 октября 2018

Я создал растровый шрифт, в основном изображение 256x256 png, где каждый символ занимает плитку 8x8. Я хочу использовать его с Подушкой как ImageFont, но в документации по Подушке об этом нет никакой информации. Он говорит, что я могу загрузить растровые шрифты, как это

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

но "PIL использует свой собственный формат файла шрифта для хранения растровых шрифтов". так что я думаю, png файл не будет работать Как я могу сказать PIL использовать указанное растровое изображение и где каждый символ находится на нем?

1 Ответ

0 голосов
/ 27 октября 2018

Не полный ответ, но слишком много для комментария, и это может быть полезно или побудить кого-то еще отработать другие 60%: -)

Я могу удалить его, если кто-нибудь придумает что-нибудь получше ...

Вы можете перейти в репозиторий Pillow на Github и загрузить ZIP-файл с кодом.

Если вы зайдете туда-сюда, вы обнаружите две вещи, которые работают рука об руку: файл .PIL и файл .PBM.


В Tests/fonts есть файл с именем 10x20.pbm, который на самом деле является PNG файлом, если вы загляните внутрь него. Итак, если вы измените его имя на 10x20.png, вы можете просмотреть его, и оно будет выглядеть так:

enter image description here

Кстати, если вы хотите разделить это на куски размером 10x20 с одной буквой в каждой, вы можете использовать ImageMagick в терминале следующим образом:

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

и вы получите набор файлов с именами char_0.png, char_1.png и т. Д. Первые 4 выглядят так:

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. Я думаю, вам просто нужно что-то, что делает последние 10 строк этого кода для вашего PNG.

Кажется, что есть 23 байта заголовка, которые вы можете просто скопировать, а затем есть 256 "записей" , то есть 1 для каждого из 256 символов. Каждая запись содержит 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

...