我可以使用 PIL 获取字符的高度和宽度(以像素为单位)(见下文),但是(除非我弄错了)像素大小取决于屏幕的 DPI,它可能会有所不同。相反,我想做的是以绝对单位(例如英寸或 1200 英寸)(“wordperfect 单位”)计算字符的宽度。
>>> # Getting pixels width with PIL
>>> font = ImageFont.truetype('/blah/Fonts/times.ttf' , 12)
>>> font.getsize('a')
(5, 14)
我想要这样做的原因是创建一个用于编写二进制 Word Perfect 文档的自动换行功能。 Word Perfect 要求在整个文本的有效点插入软换行代码,否则文件将被损坏且无法打开。问题是在哪里为可变宽度字体添加它们。
我意识到我并不完全理解像素与屏幕分辨率和字体大小之间的关系。我这一切都错了吗?
原始文本宽度通常计算为印刷师的要点,但由于用于字体定义的点被定义为 1/72 英寸,因此您可以轻松地将其转换为任何其他单位。
为了得到设计宽度一个字符(表示为em单位),您需要访问字体的低级数据。最简单的方法是pip install fonttools,它具有在尽可能最低的字体定义级别上工作的一切。
安装 fontTools 后,您可以:
-
加载字体数据——这需要实际字体文件的路径;
-
字符宽度存储为glyph宽度,这意味着您必须检索“字符到字形”映射;这是在cmap
字体表:
A。加载cmap
为你的字体。最有用的是 Unicode 映射 - 一种字体可能包含其他字体。
b.加载字形集为你的字体。这是一个列表names对于该字体的字形。
-
然后,对于每个 Unicode 字符,首先查找其名称,然后使用该名称检索其设计单位的宽度。
-
不要忘记“设计单位”是基于字体的整体“设计宽度”。这可以是一个标准值1000
(典型的 Type 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 报告的内容进行比较时是正确的。 (请注意,每个字符kerning这里不适用!这将需要更多的代码。)
字体中未定义的字符将被默默地忽略,并且像通常所做的那样,字体的宽度.notdef
字形被使用。如果您希望将此报告为错误,请删除if
在函数中进行测试。
演员阵容float
在函数中getTextWidth
所以这在 Python 2.7 和 3.5 下都有效,但请注意,如果您使用 Python 2.7 和更大值的 Unicode 字符(不是纯 ASCII),则需要重写该函数以正确使用 UTF8 字符。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)