gpt4 book ai didi

linux - itextpdf 字体未嵌入 Linux

转载 作者:太空宇宙 更新时间:2023-11-04 09:39:26 33 4
gpt4 key购买 nike

我有一个简单的 Java 程序,它使用 iTextPDF 创建一个简单的“Hello World”文件,使用的字体不是 iTextPDF 的原生字体(COOPBL.TTF,直接来自 Windows)。

在 Windows 7-64 上运行它运行良好并创建一个 pdf 文件,文件中嵌入了 Cooper Black 字体的子集,反射(reflect)在文件外观上。

在 Linux 上运行完全相同的类文件(无需重新编译),它不嵌入任何内容并使用 Helvetica。

程序如下:

import java.io.ByteArrayInputStream;
import java.io.FileOutputStream;
import java.io.IOException;

import com.itextpdf.text.FontFactory;
import com.itextpdf.text.Document;
import com.itextpdf.text.PageSize;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.pdf.PdfWriter;

import com.itextpdf.tool.xml.XMLWorkerHelper;


public class iTextTest {
private void doit(String sOut) {
Document doc = new Document(PageSize.LETTER);

try {
FileOutputStream fOut = new FileOutputStream(sOut);

// Register a non-native font.
String sFontDir = ".";
int iResult = FontFactory.registerDirectory(sFontDir);
if(iResult == 0) {
System.out.println("TestPDF(): Could not register font directory " + sFontDir);
} else {
System.out.println("TestPDF(): Registered font directory " + sFontDir);
}
System.out.println(" Fonts registered:");
for(String sFont:FontFactory.getRegisteredFonts()) {
System.out.println(" " + sFont);
}

PdfWriter pdfWriter = PdfWriter.getInstance(doc, fOut);

doc.open();

XMLWorkerHelper helper = XMLWorkerHelper.getInstance();

String htmlContent;
htmlContent = "<HTML><HEAD></HEAD><BODY>";
htmlContent += "<P style=\"font-family: cooperblack;\">";
htmlContent += "Hello World!";
htmlContent += "</P>";
htmlContent += "</BODY></HTML>";
helper.parseXHtml(pdfWriter, doc, new ByteArrayInputStream(htmlContent.getBytes()));

} catch(IOException e) {
e.printStackTrace();
} catch(DocumentException e) {
e.printStackTrace();
}

if(doc != null) {
doc.close();
}
}


public static void main(String[] args) {
if(args.length != 1) {
System.out.println("Usage: iTextTest outfile");
return;
}

iTextTest test = new iTextTest();
test.doit(args[0]);
}
}

我正在使用: itextpdf-5.5.0 xmlworker-5.5.0 Windows Java 1.6.0_21 Linux Java 1.6.0_23

我会包括字体和生成的 pdf 的副本,但看不到这样做的明显方法。对于我的测试,我只是将\Windows\fonts\COOPBL.TTF 复制到测试目录中,用于 Windows 测试和 Linux 测试。

它的值(value):- 调整字体文件中的保护/属性标志没有任何区别。- 我知道这不是公共(public)领域的字体。我仅将其用于测试。

我会洗耳恭听!

谢谢,查克

最佳答案

在 Windows 和 Linux 上并排调试(Windows Ecplise 和控制台 jdb -ack!)后,我发现了问题所在。这似乎是文档缺陷,或者是我的疏忽。我想我会在此处发布修复程序,以防其他人遇到类似问题。

问题是,如果我要使用 XMLWorkerHelper 来解析文件,我就不应该使用默认的 FontFactory 来注册字体。我在第一个代码示例中使用的 XMLWorkerHelper.parseXHtml() 的风格创建了它自己的 FontFactoryImp,即 XMLWorkerFontProvider。该提供程序忽略由默认 FontFactory 执行的任何注册。

修复有两个方面:
1) 创建您自己的 XMLWorkerFontProvider。使用将字体目录作为输入参数的构造函数。
2) 将该提供者传递给 parseXHtml(),而不是使用较短的调用。

我还使用 FontFactory.setFontImp() 设置了默认的 FontFactory 提供者。我没有研究该操作的微妙含义,但跳过或不跳过该步骤很容易。

值得一提的是,原始程序在 Windows 中运行的唯一原因是在我对 parseXHtml() 的原始调用内部提供的默认 XMLWorkerFontProvider 在所有系统目录中注册了所有字体。因为,在 Windows 端,我使用的字体已经在 C:/Windows/fonts 中,这是被拾取的字体,而不是本地目录中的文件。如果我使用了未在 Windows 中注册的新字体,Windows 测试也会失败。

这是可行的解决方案。它在 Windows 和 Linux 上生成相同的 pdf 文件:

注意:如果您使用的是 XMLParser.parse() 而不是 XMLWorkerHelper.parseXHtml(),请查看此代码块之后的附加注释。

import java.io.ByteArrayInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.File;

import com.itextpdf.text.FontFactory;
import com.itextpdf.text.Document;
import com.itextpdf.text.PageSize;
import com.itextpdf.text.DocumentException;
import com.itextpdf.text.pdf.PdfWriter;

import com.itextpdf.tool.xml.XMLWorkerHelper;
import com.itextpdf.tool.xml.XMLWorkerFontProvider;


public class iTextTest {
private void doit(String sOut) {
Document doc = new Document(PageSize.LETTER);

try {
FileOutputStream fOut = new FileOutputStream(sOut);

// Register non-native fonts in a directory.
String sFontDir = ".";
XMLWorkerFontProvider fontImp = new XMLWorkerFontProvider(sFontDir, null);
FontFactory.setFontImp(fontImp);

System.out.println(" Fonts registered before parsing:");
for(String sFont:FontFactory.getRegisteredFonts()) {
System.out.println(" " + sFont);
}

PdfWriter pdfWriter = PdfWriter.getInstance(doc, fOut);

doc.open();

XMLWorkerHelper helper = XMLWorkerHelper.getInstance();

String htmlContent;
htmlContent = "<HTML><HEAD></HEAD><BODY>";
htmlContent += "<P style=\"font-family: cooperblack;\">";
htmlContent += "Hello World!";
htmlContent += "</P>";
htmlContent += "</BODY></HTML>";
helper.parseXHtml(pdfWriter,
doc,
new ByteArrayInputStream(htmlContent.getBytes()),
XMLWorkerHelper.class.getResourceAsStream("/default.css"),
null,
fontImp);

} catch(IOException e) {
e.printStackTrace();
} catch(DocumentException e) {
e.printStackTrace();
}

if(doc != null) {
doc.close();
}
}


public static void main(String[] args) {
if(args.length != 1) {
System.out.println("Usage: iTextTest outfile");
return;
}

iTextTest test = new iTextTest();
test.doit(args[0]);
}
}

在最新的更新中,我还必须调整使用 XMLParser 的代码。问题是相似的。默认的 HtmlPipelineContext 创建它自己的 XMLWorkerFontProvider。与上述情况一样,这将忽略您使用默认 FontFactory.registerDirectory() 注册的任何字体。同样,解决方案是使用您自己的 XMLWorkerFontProvider,然后将其提供给 HtmlPipelineContext。然后,您需要为自己的 XMLWorker 创建一个新的 CssResolverPipeline,然后将其传递给 XMLParser 构造函数。

下面是一小段代码(使用自定义图像处理程序):

  htmlContext = new HtmlPipelineContext(new CssAppliersImpl(fontImp));
htmlContext.setTagFactory(Tags.getHtmlTagProcessorFactory());

pdfImageProvider = new PDFImageProvider(); // Optional
htmlContext.setImageProvider(pdfImageProvider); // Optional

Pipeline<?> pipeline = new CssResolverPipeline(cssResolver, new HtmlPipeline(htmlContext, new PdfWriterPipeline(doc, pdfWriter)));
xmlWorker = new XMLWorker(pipeline, true);
xmlParser = new XMLParser(true, xmlWorker, charsetUTF8);

我希望这可以节省我花了几天时间进行追踪的人!

关于linux - itextpdf 字体未嵌入 Linux,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22976656/

33 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com