2

这个问题是关于在Java中恢复字形字体信息,它与此处发布的问题有关。有关更多详细信息,请查看问题和答案。

那里建议使用Apache FOP库直接从 Truetype 文件中恢复字距调整对,因为 Java 不提供此信息。然后我将库移植到 Windows 并使用以下代码恢复字距调整对:

TTFFile file;
File ttf = new File("C:\\Windows\\Fonts\\calibri.ttf" );
try { file = TTFFile.open(ttf); }
catch (IOException e) {e.printStackTrace(); }
Map<Integer, Map<Integer, Integer>> kerning = file.getKerning();

最后,该库可以工作,但返回的字距调整对不适用于使用下面的函数在Path2D.Float中检索到的字形以及紧随其后显示的代码片段:

void vectorize(Path2D.Float path, String s) {
    PathIterator pIter;
    FontRenderContext frc = new FontRenderContext(null,true,true);
    GlyphVector gv;
    Shape glyph;
    gv = font.createGlyphVector(frc, s);
    glyph = gv.getGlyphOutline(0);
    pIter = glyph.getPathIterator(null);
    while (!pIter.isDone()) {
        switch(pIter.currentSegment(points)) {
        case PathIterator.SEG_MOVETO:
            path.moveTo(points[0], points[1]);
            break;
        case PathIterator.SEG_LINETO :
            path.lineTo(points[0], points[1]);
            break;
        case PathIterator.SEG_QUADTO :
            path.quadTo(points[0], points[1], points[2], points[3]);
            break;
        case PathIterator.SEG_CUBICTO :
            path.curveTo(points[0], points[1], points[2], points[3], points[4], points[5]);
            break;
        case PathIterator.SEG_CLOSE :
            path.closePath();
        }
        pIter.next();
    } 
}

字形长度被检索到阵列lens中:

Font font = new Font("Calibri", Font.PLAIN, 1000);
double interchar = 1000. * 0.075;
int size = '}' - ' ' + 1;
Path2D.Float[] glyphs = new Path2D.Float[size];
double[] lens = new double[size];
String chars[] = new String[size];
int i; char c; 
char[] s = { '0' };
for (i = 0, c = ' '; c <= '}'; c++, i++) { s[0] = c; chars[i] = new String(s); }
for (i = 0; i < size; i++) {
    vectorize(glyphs[i] = new Path2D.Float(), chars[i]); // function shown above
    lens[i] = glyphs[i].getBounds2D().getWidth() + interchar;
}

为了清楚起见,我使用 Graphics2D 的填充显示字形,并使用上面添加到 Apache FOP 库返回的字距偏移的长度进行翻译,但结果很糟糕。字体大小为标准 1000,如该讨论中所建议的那样,而interchar结果为 75。所有这些似乎都是正确的,但我的手动字距调整对看起来比使用 TTF 文件中的字距调整对好得多。

在这个库或 Truetype 字体中是否有知识渊博的人能够告诉我们应该如何使用这些字距调整对?

是否有必要直接从 TTF 文件访问字形,而不是如上所示使用 Java 字体管理?如果是,如何?

4

2 回答 2

1

问题解决了!

回想一下,使用Apache FOP库打开文件并获取字距调整对需要此代码:

TTFFile file;
File ttf = new File("C:\\Windows\\Fonts\\calibri.ttf" );
try { file = TTFFile.open(ttf); }
catch (IOException e) {e.printStackTrace(); }
Map<Integer, Map<Integer, Integer>> kerning = file.getKerning();

以下用于矢量化字形的代码现在是正确的:

Font font = new Font("Calibri", Font.PLAIN, 2048);
int size = '}' - ' ' + 1;
Path2D.Float[] glyphs = new Path2D.Float[size];
//double[] lens = new double[size];
String chars[] = new String[size];
int i; char c; 
char[] s = { '0' };
for (i = 0, c = ' '; c <= '}'; c++, i++) { s[0] = c; chars[i] = new String(s); }
for (i = 0; i < size; i++) {
    vectorize(glyphs[i] = new Path2D.Float(), chars[i]);
    //lens[i] = glyphs[i].getBounds2D().getWidth();
}

请注意,现在字体大小为 2048,这是此特定字体的 unitsPerEm。该值由字体文件中的 HEAD 标记给出,如此处所述

lens请注意,上面注释掉的数组和代码不能给出宽度。它必须从文件中读取。使用int width = getCharWidthRaw(prev)from Apache FOP,其中prev前一个字符width是文件中写入的字符的原始宽度。必须将此值添加到可以在 map 中获得的字距调整对值中kerning

该映射以这种方式使用:kerning.get(prev)它返回另一个包含要添加的字符和字距调整值的映射。如果在此映射中找到下一个要显示的字符,则将相应的值添加到width。如果未找到,或者如果null返回,则该对没有字距调整值。

这是一个显示字距调整现在有效的文本。

具有正确字距和正确字符宽度的文本

于 2020-11-11T07:48:58.963 回答
0

GNU Classpath包含一个示例, gnu.classpath.examples.awt。HintingDemo.java,这可能有助于解决这个问题。此示例允许您可视化字形。它读取字体并解释语言以获取其中给出的提示。您可以选择显示或不显示提示(提示字形适用于小字体,但不推荐用于大字体)。如果您不习惯 Truetype 提示,您将通过此演示了解它们在整数边界内对齐路径。该程序不是很花哨,但它具有读取字形和解释提示的所有必要工具,并具有可视化结果的优势。

你不需要整个包来编译和运行这个演示。如果您使用的是 Eclipse,很容易为它创建一个项目。首先创建包gnu.classpath.examples.awt并在其中导入HintingDemo.java。然后,您只需一次导入其所有依赖项,一个文件一个文件或整个包。例如,您可以导入整个包gnu.java.awt.font并擦除OpenTypeFontPeer.java(演示不需要它,如果您离开它会导致错误)。

这提供了一种直接从字体文件读取和显示字形的独立方法。有趣的是,它不使用任何字距调整信息。这必须与Apache FOP库一起添加。如果两次读取文件是一个问题,您将需要一个解决方法,要么深入 GNU Classpath 以获取相同的信息,要么尝试让 Apache FOP “说话”使用 GNU 类路径。目前我无法说这是多么困难。我只是将它用作复制信息并在其他地方使用的工具,而不是在实际程序中真正读取字体文件的一种方式。字体非常紧凑,但不是显示文本的最有效方式,尤其是在需要对字体语言进行解释的情况下,例如 Type 1 和 Truetype 字体。如果您愿意高质量和高速度,摆脱这种解释似乎是个好主意。

于 2020-11-10T21:31:26.563 回答