5

我在将 TrueType 字体(例如 Arial)的字符转换java.awt.Shape为进一步手动渲染(作为 EPS 但没关系)时遇到问题。

我已将执行此操作的程序分解为小进程,以找出问题出在哪里,在我看来,问题出在从字体加载字形的过程中。

我正在使用以下代码片段来加载字体(Arial,来自 msttcore 包)并将字符转换ÖShape我以后可以使用的字符:

Font font = new Font("Arial", Font.PLAIN, 24);
AttributedString attributedString = new AttributedString("Ö");
attributedString.addAttribute(TextAttribute.FONT, font, 0, "Ö".length());
FontRenderContext fontRenderContext = new FontRenderContext(null, false, false);
TextLayout layout = new TextLayout(attributedString.getIterator(), fontRenderContext);
Shape shape = layout.getOutline(null);

我也尝试过使用以下代码片段,但它给了我相同的结果:

Font font = new Font("Arial", Font.PLAIN, 24);
FontRenderContext fontRenderContext = new FontRenderContext(null, false, false);
GlyphVector glyphVector = font.createGlyphVector(fontRenderContext, text);
Shape shape = glyphVector.getOutline();

之后,我使用shape.getPathIterator(null)and 遍历它给我的段来打印点坐标。我在三个不同的系统上这样做:

  • Mac OS X Mountain Lion (10.8) 我负责使用 Font() 构造函数,允许我指向正确的 arial.ttf 文件,以避免使用系统内置 Arial 字体的代码段。
  • 看起来基于 Fedora 的 Amazon AWS linux 发行版
  • 基于 Ubuntu 服务器的 Amazon AWS linux 发行版

java.awt.Shape我的 Mac 上生成时,生成的 EPS 文件看起来正确。在 Linux 机器上生成时java.awt.Shape,似乎某些点坐标与在我的 Mac 上生成的坐标不同。

  • 'Ö'的O部分坐标不同,但只是在某种程度上它看起来像舍入误差并且我的眼睛无法察觉:

  • 然而,¨部分看起来很奇怪,点坐标的差异远大于舍入误差。

见下图:

同一张图片上的两个字形

绿色是在我的 Mac 上生成的 Shape 的路径,红色是在类似 Fedora 的计算机上生成的 Shape 的路径。

因为两者都在O部分很好地重叠,所以看起来有点深绿色。但是你可以看到¨部分是非常不同的。它甚至没有以红色路径为中心...

我的实验总结:

  • Java 版本/发行版看起来不会影响该问题。
  • 它看起来像是所有重音字符的常见问题(到目前为止:ÖÄÅÜ)。
  • 我对 Times New Roman 字体也有同样的问题。

我想我已经尝试了很多没有成功的事情,我真的不明白为什么会发生这种情况,并且会感谢任何提示。


这是我能给你的显示问题的最小的完整代码片段:

package Experiments;

import java.awt.Font;
import java.awt.Shape;
import java.awt.font.FontRenderContext;
import java.awt.font.GlyphVector;
import java.awt.font.TextAttribute;
import java.awt.font.TextLayout;
import java.awt.geom.PathIterator;
import java.io.File;
import java.text.AttributedString;

public class MyClass
{
    public static void main(String[] args) throws Exception
    {
        Font font = new Font("Arial", Font.PLAIN, 24);
        AttributedString attributedString = new AttributedString("Ö");
        attributedString.addAttribute(TextAttribute.FONT, font, 0, "Ö".length());
        FontRenderContext fontRenderContext = new FontRenderContext(null, false, false);
        TextLayout layout = new TextLayout(attributedString.getIterator(), fontRenderContext);
        Shape shape = layout.getOutline(null);
        PathIterator it = shape.getPathIterator(null);
        double[] points = new double[6];
        double x = 0, y = 0;

        while (!it.isDone())
        {
            double x1 = points[0], y1 = points[1];
            double x2 = points[2], y2 = points[3];
            double x3 = points[4], y3 = points[5];

            switch (it.currentSegment(points))
            {
                case PathIterator.SEG_CLOSE:
                    System.out.println("close");
                    break;
                case PathIterator.SEG_QUADTO:
                    // Convert to cubic curve
                    x3 = x2;
                    y3 = y2;
                    x2 = x1 + 1 / 3f * (x2 - x1);
                    y2 = y1 + 1 / 3f * (y2 - y1);
                    x1 = x + 2 / 3f * (x1 - x);
                    y1 = y + 2 / 3f * (y1 - y);
                case PathIterator.SEG_CUBICTO:
                    System.out.println("curve: " + x1 + "," + y1 + "," + x2 + "," + y2 + "," + x3 + "," + y3);
                    x = x3;
                    y = y3;
                    break;
                case PathIterator.SEG_LINETO:
                    System.out.println("lineto: " + x1 + "," + y1);
                    x = x1;
                    y = y1;
                    break;
                case PathIterator.SEG_MOVETO:
                    System.out.println("moveto: " + x1 + "," + y1);
                    x = x1;
                    y = y1;
                    break;
            }
            it.next();
        }
    }
}

我的 Mac 上的输出是:

moveto: 0.0,0.0
curve: 0.7734375230502337,-5.57812516624108,0.7734374884748831,-5.57812491687946,0.0,0.0
curve: 0.7734375230502337,-8.429687751224265,1.925781272817403,-13.451171899039764,3.45703125,-15.064453125
curve: 4.988281295634806,-16.677734423079528,6.96484378608875,-17.484375,9.38671875,-17.484375
curve: 10.97265629726462,-17.484375,12.402343768975697,-17.105468738707714,13.67578125,-16.34765625
curve: 14.949218787951395,-15.589843727415428,15.91992188495351,-14.533203104801942,16.587890625,-13.177734375
curve: 17.25585939490702,-11.822265584603883,17.58984375,-10.28515622438863,17.58984375,-8.56640625
curve: 17.58984375,-6.8242186980787665,17.23828123952262,-5.265624979510903,16.53515625,-3.890625
curve: 15.832031229045242,-2.5156249590218067,14.835937480791472,-1.4746093644644134,13.546875,-0.767578125
curve: 12.257812461582944,-0.06054685392882675,10.867187477764674,0.29296875,9.375,0.29296875
curve: 7.757812451804057,0.29296875,6.3124999810243025,-0.09765626164153218,5.0390625,-0.87890625
curve: 3.765624962048605,-1.6601562732830644,2.800781240221113,-2.7265625201398507,2.14453125,-4.078125
close
moveto: 1.16015625,-6.10546875
curve: 2.7226562965661287,-7.589843794237822,2.7226562267169356,-8.343750000349246,1.16015625,-8.3671875
curve: 2.7226562965661287,-6.2734374376013875,4.060546891589183,-4.630859357246663,5.173828125,-3.439453125
curve: 6.287109408178367,-2.248046839493327,7.683593775029294,-1.65234375,9.36328125,-1.65234375
curve: 11.074218800989911,-1.65234375,12.482421891472768,-2.2539062679279596,13.587890625,-3.45703125
curve: 14.693359407945536,-4.660156285855919,15.24609375,-6.367187532945536,15.24609375,-8.578125
curve: 15.24609375,-9.976562541676685,15.009765617956873,-11.197265640541445,14.537109375,-12.240234375
curve: 14.064453110913746,-13.283203156082891,13.373046861437615,-14.091796883556526,12.462890625,-14.666015625
curve: 11.55273434787523,-15.240234392113052,10.531249983119778,-15.52734375,9.3984375,-15.52734375
curve: 7.789062452036887,-15.52734375,6.404296857712325,-14.974609358527232,5.244140625,-13.869140625
close
moveto: 3.50390625,-12.2109375
lineto: 6.046875,-18.234375
lineto: 6.046875,-20.63671875
lineto: 8.25,-20.63671875
close
moveto: 8.25,-18.234375
lineto: 10.41796875,-18.234375
lineto: 10.41796875,-20.63671875
lineto: 12.62109375,-20.63671875
close

在类似 Fedora 的计算机上:

moveto: 0.0,0.0
curve: 0.7708333563059568,-5.583333499729633,0.7708333218470216,-5.583333250135183,0.0,0.0
curve: 0.7708333563059568,-8.427083584479988,1.921875022817403,-13.447916690725833,3.453125,-15.0625
curve: 4.984375045634806,-16.677083381451666,6.963541702833027,-17.484375,9.390625,-17.484375
curve: 10.973958380520344,-17.484375,12.403645852347836,-17.106770822079852,13.6796875,-16.3515625
curve: 14.955729204695672,-15.596354144159704,15.92708334326744,-14.539062479743734,16.59375,-13.1796875
curve: 17.26041668653488,-11.820312459487468,17.59375,-10.28124997438863,17.59375,-8.5625
curve: 17.59375,-6.822916614823043,17.24218748952262,-5.265624979510903,16.5390625,-3.890625
curve: 15.835937479045242,-2.5156249590218067,14.838541647419333,-1.4739583227783442,13.546875,-0.765625
curve: 12.255208294838667,-0.05729164555668831,10.864583311136812,0.296875,9.375,0.296875
curve: 7.760416618548334,0.296875,6.315104147652164,-0.09375001164153218,5.0390625,-0.875
curve: 3.763020795304328,-1.6562500232830644,2.796874990221113,-2.723958353511989,2.140625,-4.078125
close
moveto: 1.15625,-6.109375
curve: 2.7187500465661287,-7.5885417107492685,2.7187499767169356,-8.343750000465661,1.15625,-8.375
curve: 2.7187500465661287,-6.2812499376013875,4.057291683275253,-4.638020815560594,5.171875,-3.4453125
curve: 6.286458366550505,-2.2526041311211884,7.682291691657156,-1.65625,9.359375,-1.65625
curve: 11.078125051222742,-1.65625,12.489583349786699,-2.2578125179279596,13.59375,-3.4609375
curve: 14.697916699573398,-4.664062535855919,15.25,-6.369791699573398,15.25,-8.578125
curve: 15.25,-9.973958374932408,15.013020826270804,-11.195312515599653,14.5390625,-12.2421875
curve: 14.065104152541608,-13.289062531199306,13.372395819751546,-14.098958341870457,12.4609375,-14.671875
curve: 11.549479139503092,-15.244791683740914,10.531249983236194,-15.53125,9.40625,-15.53125
curve: 7.791666618548334,-15.53125,6.4036458160262555,-14.979166650213301,5.2421875,-13.875
close
moveto: 3.5,-12.21875
lineto: 6.625,-18.0
lineto: 6.625,-20.390625
lineto: 8.828125,-20.390625
lineto: 8.828125,-18.0
close
moveto: 6.625,-18.0
lineto: 11.0,-18.0
lineto: 11.0,-20.390625
lineto: 13.203125,-20.390625
lineto: 13.203125,-18.0
close

如果你读到这里,我已经感谢你了 :)

4

2 回答 2

2

你到底为什么要打扰 Arial?“免费”的 msttcorefonts 版本是多年未得到修复的废弃软件。它针对不再存在的字体渲染系统进行了优化(现在众所周知,微软在这些文件中使用了错误的元数据值来解决其当时最先进的系统中的错误)。

使用旨在用于跨平台使用的现代字体,并使用实际上允许重新分发和衍生的许可(请参阅 DejaVu 或 Google 字体库)。你的形状衍生物。

OSX 上的旧 java 由 Apple 维护,可能使用 Apple 特定的字体引擎。其他系统上的旧 java(如 Oracle 下载)使用专有字体引擎。OpenJDK 使用 freetype(Oracle 不敢更改官方 jdk,因为害怕破坏依赖其旧引擎缺陷的应用程序。可惜,freetype维护得很好)

期望他们都给出完全相同的塑造结果是徒劳的。而且字体越复杂和越旧,它们就越有可能出现分歧(旧字体包含旧元数据,新字体不包含,并且取决于字体引擎,它会尝试对这些旧数据做些什么,或者不。字体格式甚至包括 Apple 和 Windows 特定的元数据,这些元数据将根据所使用的字体引擎被读取或忽略)。

智能字体格式很棒……</p>

于 2013-08-26T14:01:49.480 回答
1

nim 的回答引导我们上路,所以我接受了。

我只会在这里记录我们在调查中发现的内容。

通过一些黑盒测试,我们意识到当使用更大的字体大小时问题就消失了。然后,使用非常大的字体大小(25000 像素),我们得到字形的上点得到负坐标,这看起来像是整数溢出的副作用。

我没有尝试找到字体引擎的源代码,但是通过黑盒测试,我可以怀疑我们的 java 版本中使用的字体引擎在从字体中读取字形时使用整数,使用时会出现很大的舍入错误使用大字体时出现小字体和溢出错误。

为了解决我们的问题并尽可能少地更改我们的生产代码,我们决定始终加载字体大小为 100px 的字体,然后使用 AffineTransform 将 Shape 缩放到所需的大小。

我们没有按照 nim 的回答所建议的那样更改字体,因为我们正在尝试尽可能少地进行修复,但我们始终牢记,msttcore 字体有更好的替代品(Google Fonts lib 就是其中之一(顺便说一句) ,我们也有使用谷歌字体的舍入错误,尽管比 msttcore 字体少一点))。

于 2013-08-28T09:41:08.557 回答