废话不多说直接进入正题,首先分析生成pdf场景及生成内容,考虑复用性和维护难度是我们当前开发工作的第一要务!
下面是调研的几个主要方案:
使用方式:
方案优缺点:
使用方式:
方案优缺点:
这可能是由于doc4J迭代问题无法保证新元素的支持,导出结果比较奔放。。。
使用方式:
方案优缺点:
受限于调试前期需要的修修改改,模板能给人整吐了,所以才有了下一个方案。
使用方式:
方案优缺点:
这个方案是综合以上多次踩坑的结果,结果是显而易见的。
浅浅来一点代码,省的大家到处找
<dependencies>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>itextpdf</artifactId>
<version>5.5.13</version>
</dependency>
<dependency>
<groupId>com.itextpdf</groupId>
<artifactId>html2pdf</artifactId>
<version>3.0.3</version>
</dependency>
<dependency>
<groupId>org.xhtmlrenderer</groupId>
<artifactId>flying-saucer-pdf-itext5</artifactId>
<version>9.0.3</version>
</dependency>
<dependency>
<groupId>binarta.oss</groupId>
<artifactId>groovy-template-enginex-freemarker</artifactId>
<version>0.1.3</version>
</dependency>
<dependencies>
个人以为自己的代码会自解释,就不贴太多注释了
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.map.MapUtil;
import com.google.common.collect.Lists;
import com.itextpdf.text.pdf.BaseFont;
import freemarker.cache.ByteArrayTemplateLoader;
import freemarker.template.Configuration;
import freemarker.template.Template;
import lombok.SneakyThrows;
import org.xhtmlrenderer.pdf.ITextFontResolver;
import org.xhtmlrenderer.pdf.ITextRenderer;
import java.io.BufferedWriter;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.StringWriter;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.function.Supplier;
public class HtmlToPdfUtil {
private static final ByteArrayTemplateLoader TEMPLATE_LOADER = new ByteArrayTemplateLoader();
// 导入需要字体库的位置哦;simsun 为 宋体
public static final String FRONT_PATH = "/usr/share/fonts/simsun.ttc";
/**
* 看明白的话只用这个方法就够
*/
public static ByteArrayOutputStream htmlToPdf(String templateName, Supplier<byte[]> loadTemplateSupplier, Map<String, Object> modeViewMap) {
String html = xmlFormat(templateName, loadTemplateSupplier, modeViewMap);
return htmlToPdf(html);
}
@SneakyThrows
public static ByteArrayOutputStream htmlToPdf(String htmlStr) {
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
ITextRenderer renderer = new ITextRenderer();
renderer.setDocumentFromString(htmlStr);
ITextFontResolver resolver = renderer.getFontResolver();
//添加字体,解决中文不显示的问题
resolver.addFont(FRONT_PATH, BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
renderer.layout();
renderer.createPDF(outputStream);
return outputStream;
}
public static String xmlFormat(String templateName, Supplier<byte[]> loadTemplateSupplier, Map<String, Object> modeViewMap) {
if (Objects.isNull(TEMPLATE_LOADER.findTemplateSource(templateName))) {
synchronized (TEMPLATE_LOADER) {
if (Objects.isNull(TEMPLATE_LOADER.findTemplateSource(templateName))) {
TEMPLATE_LOADER.putTemplate(templateName, loadTemplateSupplier.get());
}
}
}
return xmlFormat(templateName, modeViewMap);
}
@SneakyThrows
public static String xmlFormat(String templateName, Map<String, Object> modeViewMap) {
Configuration cfg = new Configuration(Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS);
// 指定FreeMarker模板文件的位置
cfg.setTemplateLoader(TEMPLATE_LOADER);
// 设置模板的编码格式
cfg.setEncoding(Locale.CHINA, Charset.defaultCharset().name());
// 获取模板文件 template
Template template = cfg.getTemplate(templateName, Charset.defaultCharset().name());
StringWriter stringWriter = new StringWriter();
BufferedWriter writer = new BufferedWriter(stringWriter);
template.process(modeViewMap, writer);
return stringWriter.toString();
}
}
解决这个问题的核心思路方案其实一直没变,变化的只是工具,一定要思路清晰!
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8