如何用Python获得以1200分之一英寸为单位的trueetype字体字符的宽度?

问题描述 投票:3回答:2

我可以用 PIL 获得一个字符的高度和宽度,单位是像素 (见下图),但是 (除非我弄错了)像素大小取决于屏幕的 DPI,而 DPI 可以变化。我想做的是计算一个字符的宽度,用绝对单位,比如英寸,或者1200分之一英寸("wordperfect units")。

>>> # Getting pixels width with PIL
>>> font = ImageFont.truetype('/blah/Fonts/times.ttf' , 12)
>>> font.getsize('a')
(5, 14)

我想这样做的原因是为了创建一个用于编写二进制Word Perfect文档的包字功能。Word Perfect要求在整个文本中的有效点插入软换行代码,否则文件会损坏,无法打开。问题是,对于可变宽度的字体,应该在哪里添加它们。

我意识到我没有完全理解像素和屏幕分辨率与字体大小之间的关系。我的做法是不是都错了?

python true-type-fonts
2个回答
7
投票

原始文本宽度通常以 打字员的观点但由于字体定义中的点被定义为172英寸,你可以很容易地将其转换为任何其他单位。

为了得到 设计宽度 字的 鸸鹋 单位),你需要访问字体的低级数据。最简单的方法是 pip install fonttools它的一切工作都在尽可能低的字体定义水平上。

安装了fontTools之后,你可以

  1. 加载字体数据 - 这需要实际字体文件的路径。

  2. 字符宽度被存储为 字形 宽度,这意味着你必须检索 "字符到图形 "的映射;这是在 cmap 字体表。

    A. 载入 cmap 的字体。最有用的是Unicode地图 - 一种字体可能包含其他字体。字形集 的字体。这是一个列表,其中包括 姓名 为该字体中的字形。

  3. 然后,对于每个Unicode字符,首先查找它的名称,然后用名称来检索它的设计单位宽度。

  4. 不要忘记,"设计单位 "是基于字体的整体 "设计宽度"。这可以是一个标准值,即 1000 (典型的1类字体)。2048 (典型的TrueType字体),或任何其他值。

这就导致了这个函数。

from fontTools.ttLib import TTFont
from fontTools.ttLib.tables._c_m_a_p import CmapSubtable

font = TTFont('/Library/Fonts/Arial.ttf')
cmap = font['cmap']
t = cmap.getcmap(3,1).cmap
s = font.getGlyphSet()
units_per_em = font['head'].unitsPerEm

def getTextWidth(text,pointSize):
    total = 0
    for c in text:
        if ord(c) in t and t[ord(c)] in s:
            total += s[t[ord(c)]].width
        else:
            total += s['.notdef'].width
    total = total*float(pointSize)/units_per_em;
    return total

text = 'This is a test'

width = getTextWidth(text,12)

print ('Text: "%s"' % text)
print ('Width in points: %f' % width)
print ('Width in inches: %f' % (width/72))
print ('Width in cm: %f' % (width*2.54/72))
print ('Width in WP Units: %f' % (width*1200/72))

结果是:

Text: "This is a test"
Width in points: 67.353516
Width in inches: 0.935465
Width in cm: 2.376082
Width in WP Units: 1122.558594

并与Adobe InDesign的报告进行比较,结果是正确的。请注意,每个字符 去核 在这里不适用!这需要更多的代码。这将需要更多的代码)。)

没有在字体中定义的字符会被忽略,而且,通常情况下,字体的宽度会被忽略。.notdef 字形被使用。如果你想将此报告为错误,请删除掉下面的 if 函数中的测试。

投向 float 在函数中 getTextWidth 是所以这个函数在Python 2.7和3.5下都能工作,但是请注意,如果你使用Python 2.7和更大值的Unicode字符(不是纯ASCII),你需要重写函数来正确使用UTF8字符。


0
投票

这对我来说效果更好。

def pixel_width(unicode_text): 
    width=len(unicode_text)*50 
    height=100 
    back_ground_color=(0,0,0) 
    font_size=64 
    font_color=(255,255,255) 

    im  =  Image.new ( "RGB", (width,height), back_ground_color ) 
    draw  =  ImageDraw.Draw (im) 
    unicode_font = ImageFont.truetype("./usr/share/fonts/truetype/dejavu/DejaVuSansMono.ttf", font_size) 
    draw.text ( (0,0), unicode_text, font=unicode_font, fill=font_color ) 
    im.save("/dev/shm/text.png") 
    box = Image.open("/dev/shm/text.png").getbbox() 
    return box[2] - box[0] 
© www.soinside.com 2019 - 2024. All rights reserved.