如何使用嵌入式字体在iText中调用Graphics2D.drawString(...)(使用DefaultFontMapper?)

6
生成有效的PDF/X文档,必须嵌入所有字体。但在Graphics2D上下文中使用这些字体似乎是不可能的。
以下是展示此问题的单元测试(已注释的行是我进行的一些测试):
import java.awt.Font;
import java.awt.Graphics2D;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.PrintStream;
import java.util.Map.Entry;

import org.junit.Test;

import com.itextpdf.awt.DefaultFontMapper;
import com.itextpdf.awt.DefaultFontMapper.BaseFontParameters;
import com.itextpdf.awt.PdfGraphics2D;
import com.itextpdf.text.Document;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.Rectangle;
import com.itextpdf.text.pdf.PdfWriter;

public class TestFont
{

    @Test
    public void shouldBeAbleToAddFountsAndDrawOnCanvas() throws FileNotFoundException, DocumentException
    {
        final DefaultFontMapper mapper = new DefaultFontMapper();
        mapper.insertDirectory(".");

        final PrintStream out2 = new PrintStream(System.out);
        for (final Entry<String, BaseFontParameters> entry : mapper.getMapper().entrySet())
        {
            out2.println(String.format("%s: %s", entry.getKey(), entry.getValue().fontName));
        }
        out2.flush();

        final float width = 150;
        final float height = 150;

        final Document document = new Document(new Rectangle(width, height));
        final PdfWriter writer = PdfWriter.getInstance(document, new FileOutputStream("fonts.pdf"));
        writer.setPDFXConformance(PdfWriter.PDFX32002);

        document.open();
        final Graphics2D g2d = new PdfGraphics2D(writer.getDirectContent(), width, height, mapper);

        g2d.setFont(new Font("Comicate", Font.PLAIN, 12));

        g2d.drawString("Hello world", 5, 24);

        g2d.dispose();

        document.close();
    }

}

它将抛出一个带有信息“所有字体必须嵌入,但此字体没有:Helvetica”的PdfXConformanceException异常。
我已经浏览了PdfGraphics2D类以检查setFont()的实现,并发现将使用FontMapper。我已经将其添加到上面的单元测试中。
public void setFont(Font f) {
    if (f == null)
        return;
    if (onlyShapes) {
        font = f;
        return;
    }
    if (f == font)
        return;
    font = f;
    fontSize = f.getSize2D();
    baseFont = getCachedBaseFont(f);
}

private BaseFont getCachedBaseFont(Font f) {
    synchronized (baseFonts) {
        BaseFont bf = (BaseFont)baseFonts.get(f.getFontName());
        if (bf == null) {
            bf = fontMapper.awtToPdf(f);
            baseFonts.put(f.getFontName(), bf);
        }
        return bf;
    }
}

Unittest基于《iText in Action》一书中的此示例。这里还有一些关于FontMapper的其他示例
要运行Unittest,您需要这个依赖项:
<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>itextpdf</artifactId>
    <version>5.3.2</version>
</dependency>

你可以在这里找到位于"."中的自定义字体:此处

控制台输出给我展示了这个(以识别fontName):

Comicate: ./COMICATE.TTF
2个回答

2

我不确定如何在您的代码中准确地纠正错误,但有一些简单的解决方法:

解决方法1)创建一个BufferedImage来进行所有的图形绘制。然后,您可以使用所有常规的java.awt.Graphics函数,如drawStringsetColor,而不受iText的影响。当您完成后,只需将图像绘制到PDF上即可。警告:缩放时会失去文本质量,但这里是一个示例:

//create doccument and writer    
Rectangle pagesize = new Rectangle(200, 100);
Document document= new Document(pagesize);
PdfWriter writer = PdfWriter.getInstance(document, new FileOutputStream("C:\\sample.pdf"));

BufferedImage bf = new BufferedImage(BorderWidth, BorderHeight, BorderWidth);
//Do all graphics code here, draw strings and images etc
    //Some code to set font (java.awt.Font)
    //Some code to draw string
    //Some code to draw image?

//Convert BufferedImage to Image
Image img = (Image)bf;
//draw image to PDF using writer
writer.getDirectContentUnder().addImage(img);

解决方法2) 这种方法使用iText特性来绘制字符串,无需创建任何图形对象,通过使用BaseFont即可处理字体,具体操作如下:

//create doccument and writer    
Rectangle pagesize = new Rectangle(200, 100);
Document document= new Document(pagesize);
PdfWriter writer = PdfWriter.getInstance(document, new FileOutputStream("C:\\sample.pdf"));

document.open();
//This sample uses the "GOTHIC.TTF" font file located in the "Template" package
BaseFont bf = BaseFont.createFont(GUI.class.getClass().getResource("/Template/GOTHIC.TTF") + "", BaseFont.WINANSI, BaseFont.EMBEDDED);

//set font type, size and color
Font font = new Font(bf, 13.5f);

PdfContentByte canvas = writer.getDirectContent();

canvas.beginText();
canvas.setFontAndSize(bf, 10);
//Method Usage: showTextAligned(Align, String, x, y, rotation);
canvas.showTextAligned(Element.ALIGN_TOP, "My Text Here", 75, 40, 0);
canvas.endText();

document.close();

我知道这并不是你想要的答案,但是如果你只是绘制少量的文本,那么解决方法2非常有效,我以前也使用过类似于解决方法2的东西。如果这不能帮助你,我相信Bruno会有答案。


嗨,感谢您的建议。遗憾的是我不能使用您的解决方法 :(. 调用g2d.drawText(...)的不是我,而是JFreechart。 - d0x
你能看一下这里的问题吗?https://stackoverflow.com/q/52736441/3169868 - S_S

1
通过定义新的BaseFont(s)并实现FontMapper()接口 >>> public BaseFont awtToPdf(java.awt.Font font),可以将awt.font(s)嵌入到pdf中。
在下面的示例中,我将在外部类中绘制的g2D(包括“drawString”方法)“打印”出来。结果是只嵌入了“ArialMT”和“Arial-BoldMT”字体的导出矢量pdf。 preview img
    PdfWriter pdfWriter = null;
    try {
        pdfWriter = PdfWriter.getInstance(document, new FileOutputStream(file));
        document.open();
        BaseFont fontRegular = BaseFont.createFont("C:\\Windows\\Fonts\\arial_0.ttf", "Cp1251", BaseFont.EMBEDDED);
        BaseFont fontBold = BaseFont.createFont("C:\\Windows\\Fonts\\arialbd_0.ttf", "Cp1251", BaseFont.EMBEDDED);
        FontMapper fontMapper = new FontMapper() {

            @Override
            public java.awt.Font pdfToAwt(BaseFont arg0, int arg1) {
                // TODO Auto-generated method stub
                return null;
            }

            @Override
            public BaseFont awtToPdf(java.awt.Font font) {
                if (font.equals(Fonts.getFontRegular10()) || font.equals(Fonts.getFontRegular12())){
                    return fontRegular;
                }
                else {
                    return fontBold;
                }
            }
        };
        PdfContentByte cb = pdfWriter.getDirectContent();
        PdfTemplate template = cb.createTemplate(MainFrame.getFRAME_WIDTH(), MainFrame.getFRAME_HEIGHT());
        Graphics2D g2D = new PdfGraphics2D(template, MainFrame.getFRAME_WIDTH(), MainFrame.getFRAME_HEIGHT(), fontMapper);
        MainFrame.getPanel().print(g2D);
        g2D.dispose();
        cb.addTemplate(template, 0, 0);
    }
    catch (Exception e1) {
        e1.printStackTrace();
    }
    finally {
        document.close();
    }

将.ttf文件放置在可运行的Jar包内,是使用itext创建PDF的唯一可行方案。 - AntJavaDev

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接