(5) 确实看起来很棘手,没有在字符串中插入虚拟窄空格(这会破坏字距调整)或使用更高级别的东西,如 SVG 或 HTML/CSS 渲染器。
然而,如果你不介意弄脏你的手,看起来很容易破解 PIL 的 freetype 渲染器来添加水平空间。见_imagingft.c;在 font_getsize 和 font_render 中的以下代码之后:
if (kerning && last_index && index) {
FT_Vector delta;
FT_Get_Kerning(self->face, last_index, index, ft_kerning_default,
&delta);
x += delta.x >> 6;
}
添加:
if (last_index && index) {
x += tracking;
}
首先尝试使用纯整数进行跟踪(根据'>>6'判断可能相当大);编译并查看它是否有效。下一步是将跟踪值从 Python 获取到 C 函数中,为此您必须将 font_render 中的 ParseTuple 调用更改为:
long tracking;
if (!PyArg_ParseTuple(args, "Ol|il:render", &string, &id, &mask, &tracking))
return NULL;
在 font_getsize 中:
long tracking;
if (!PyArg_ParseTuple(args, "O|l:getsize", &string, &tracking))
return NULL;
然后看你想要什么Python接口。这是通过界面的每一层添加额外的“跟踪”参数的一个微不足道但相当乏味的案例,例如:
def truetype(filename, size, index=0, encoding="", tracking= 0): # added optional tracking
"Load a truetype font file."
try:
return FreeTypeFont(filename, size, index, encoding, tracking) # added tracking
...
class FreeTypeFont:
"FreeType font wrapper (requires _imagingft service)"
def __init__(self, file, size, index=0, encoding="", tracking= 0): # added tracking
import _imagingft
self.font = _imagingft.getfont(file, size, index, encoding)
self.tracking= tracking # add this line
...
def getmask2(self, text, mode="", fill=Image.core.fill):
size, offset = self.font.getsize(text, self.tracking) # use tracking
im = fill("L", size, 0)
self.font.render(text, im.id, mode=="1", self.tracking) # use tracking
return im, offset
我没有测试过这些!如果它有效,可能值得作为补丁提交。