使用 poi-tl 根据模板生成 word 文件。
 使用 xdocreport 将 docx 文件转换为 pdf 文件。
 xdocreport 也支持根据模板导出 word ,但是 poi-tl 的功能更齐全,操作更简单,文档清晰。
  poi-tl 、xdocreport 内部均依赖了 poi ,要注意两者中 poi 和 自身项目引用的 poi 的版本是否存在冲突。
 
文章目录
Pom 依赖
使用 poi 5.2.2 ,poi-tl 1.12.1 ,xdocreport 2.0.3
<!--        poi依赖-->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>5.2.2</version>
        </dependency>
<!--        poi-tl依赖-->
        <dependency>
            <groupId>com.deepoove</groupId>
            <artifactId>poi-tl</artifactId>
            <version>1.12.1</version>
        </dependency>
<!--        xdocreport依赖-->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml-full</artifactId>
            <version>5.2.2</version>
        </dependency>
        <dependency>
            <groupId>fr.opensagres.xdocreport</groupId>
            <artifactId>xdocreport</artifactId>
            <version>2.0.3</version>
        </dependency>
        <dependency>
            <groupId>fr.opensagres.xdocreport</groupId>
            <artifactId>org.apache.poi.xwpf.converter.pdf</artifactId>
            <version>1.0.6</version>
        </dependency>
生成 DOCX 文件
创建 DOCX 模板
根据官方文档按要求创建模板,并放在resources文件夹下。官方文档 http://deepoove.com/poi-tl/。
 
 
生成 DOCX 文档
创建模板填充类
其实也可以在生成文件时使用 Map 类型的方式填充文件内容,如下
XWPFTemplate.compile(inputStream).render(
  new HashMap<String, Object>(){{
    put("title", "Hi, poi-tl Word模板引擎");
}});
但是使用实体类更规范一些。
@Data
public class ChildRoundsProvalDocxEntity {
    private String childName;//孩子姓名
    private String identify;//身份证号
    private String gender;//性别
    private String childRounds;//孩次
    private PictureRenderData avatar;//头像地址
    private String entryDate;//入院时间
    private String gardenMonths;//孩次
    private String charge;//每月收费
    private String fatherName;//父亲姓名
    private String fatherIdentify;//父亲身份证号
    private String motherName;//母亲姓名
    private String motherIdentify;//母亲身份证号
    private String address;//住址及联系方式
    private String firstChildName;//第一个子女姓名
    private String firstChildIdentify;//第一个子女身份证号
    private String secondChildName;//第二个子女姓名
    private String secondChildIdentify;//第二个子女身份证号
    private String signDate;//签署日期
}
生成文件(多种方式)
1、生成一个文件输入流,方便用于再次编辑
  不要在方法内部将输入流关闭
/**
 * 生成文件到输入流中
 * @param templatePath
 * @param data
 * @return
 */
public InputStream createWordFile1(Object data){
    Resource templateFile = resourceLoader.getResource("classpath:wordtemplate/childRoundsProval.docx");
    XWPFTemplate template = null;
    InputStream resultStream = null;
    try {
        // word模板填充
        InputStream inputStream = templateFile.getInputStream();
        template = XWPFTemplate.compile(inputStream).render(data);
        resultStream = PoitlIOUtils.templateToInputStream(template);
        PoitlIOUtils.closeQuietlyMulti(template);
    } catch (Exception e) {
        log.error("导出失败,异常原因:" + e.getMessage());
    } finally {
        try {
            if (template != null) {
                template.close();
            }
        } catch (Exception e) {
            log.error("流关闭失败,异常原因:" + e.getMessage());
        }
    }
    return resultStream;
}
2、生成docx到输出流中,一般是在网络响应中直接输出,浏览器去下载
public void createWordFile2(Object data, HttpServletResponse httpServletResponse){
	//获取模板信息
	Resource templateFile = resourceLoader.getResource("classpath:wordtemplate/childRoundsProval.docx");
	XWPFTemplate template = null;
	docName = URLEncoder.encode(docName, StandardCharsets.UTF_8);
	try {
		httpServletResponse.setContentType("application/octet-stream");
        httpServletResponse.addHeader("Content-Disposition", "attachment;filename=" + docName + ".docx");
        httpServletResponse.addHeader("filename", docName);
	    // word模板内容填充
	    InputStream inputStream = templateFile.getInputStream();
	    template = XWPFTemplate.compile(inputStream).render(data);
	    OutputStream out = httpServletResponse.getOutputStream();//要记得关闭
	    BufferedOutputStream bos = new BufferedOutputStream(out);//要记得关闭
	    template.write(bos);
	    bos.flush();
	    out.flush();
	    PoitlIOUtils.closeQuietlyMulti(template, bos, out);
	} catch (Exception e) {
	    log.error("导出失败,异常原因:" + e.getMessage());
	    throw new BaseException("Word文档生成失败");
	} finally {
	    try {
	        if (template != null) {
	            template.close();
	        }
	    } catch (Exception e) {
	        log.error("流关闭失败,异常原因:" + e.getMessage());
	    }
	}
}
3、直接生成文件到指定路径
public void createWordFile3(Object data, String path){
	//获取模板信息
	Resource templateFile = resourceLoader.getResource("classpath:wordtemplate/childRoundsProval.docx");
	XWPFTemplate template = null;
	try {
	    // word模板内容填充
	    InputStream inputStream = templateFile.getInputStream();
	    template = XWPFTemplate.compile(inputStream).render(data);
	    template.writeToFile(path);//文件夹路径必须存在 可以提前创建
	    PoitlIOUtils.closeQuietlyMulti(template);
	} catch (Exception e) {
	    log.error("导出失败,异常原因:" + e.getMessage());
	    throw new BaseException("Word文档生成失败");
	} finally {
	    try {
	        if (template != null) {
	            template.close();
	        }
	    } catch (Exception e) {
	        log.error("流关闭失败,异常原因:" + e.getMessage());
	    }
	}
}
生成效果

DOCX 转 PDF
初始化 XWPFDocument 需要一个输入流,以下是直接使用文件输入流去初始化。
FileInputStream inputStream1 = new FileInputStream("docx文件位置.docx");
XWPFDocument xwpfDocument = new XWPFDocument(inputStream1);
PdfOptions options = PdfOptions.create();
try (OutputStream outPDF = Files.newOutputStream(Paths.get("要生成的pdf位置.pdf"))) {
    PdfConverter.getInstance().convert(xwpfDocument.getXWPFDocument(), outPDF, options);
} catch (IOException e) {
    log.error("PDF转换失败",e);
}
PDF 生成效果

注意
DOCX 模板中如果表格元素中放了图片(如上图中的头像),要确保生成的文件中的图片大小不超过模板中单元格大小。
 即图片所在单元格不能被图片撑大,否则图片在转换成PDF时无法展示。










