【Java】使用xdocreport、FreeMarker按模板样式导出word

一、场景

根据模板样式进行word导出。

二、模板制作

通过xdocreport、FreeMarker来实现该场景。
首先,当然是要有一个word模板,接下来简单讲解如何制作word模板。

2.1、文本替换

1.选中要替换的文本。
2.键盘使用 Ctrl + F9 组合键将其设置为域,此时文本会被{}包围。
3.鼠标右键选择“编辑域”。
4.域名选择“邮件合并”,域代码填写 ${XXX!},${}是标识符,XXX表示变量名,!表示该变量允许为空(不加!则反之)。

在这里插入图片描述
在这里插入图片描述

2.2、表格循环

1.在第一个单元格设置域 @before-row[#list XXX as item],XXX为数组对象。
2.紧接着在 @before-row域后面继续设置域 @after-row[/#list]。
3.在 @before-row域与 @after-row域之间设置内容域 ${item.index!}。
4.后续单元格不用设置 @before-row域和@after-row域,直接设置内容域 ${item.detail!}就行。

在这里插入图片描述

2.3、普通循环

1.设置域 [#list XXX as item],XXX为数组对象。
2.设置域 [/#list]。
3.将内容域 ${item.index!}放到中间即可。

在这里插入图片描述

2.4、图片替换

1.正常在word插入图片。
2.选中图片,工具栏选择插入,选择书签,新增书签。

在这里插入图片描述

三、代码实现

先引入依赖:

<dependency>
    <groupId>org.apache.poi</groupId>
    <artifactId>poi</artifactId>
    <version>4.1.1</version>
</dependency>
<dependency>
    <groupId>org.apache.poi</groupId>
    <artifactId>poi-ooxml</artifactId>
    <version>4.1.1</version>
</dependency>
<dependency>
    <groupId>org.jxls</groupId>
    <artifactId>jxls</artifactId>
    <version>2.6.0</version>
    <exclusions>
        <exclusion>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-core</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<dependency>
    <groupId>org.jxls</groupId>
    <artifactId>jxls-poi</artifactId>
    <version>1.2.0</version>
</dependency>
<dependency>
    <groupId>fr.opensagres.xdocreport</groupId>
    <artifactId>fr.opensagres.xdocreport.core</artifactId>
    <version>2.0.2</version>
</dependency>
<dependency>
    <groupId>fr.opensagres.xdocreport</groupId>
    <artifactId>fr.opensagres.xdocreport.document</artifactId>
    <version>2.0.2</version>
</dependency>
<dependency>
    <groupId>fr.opensagres.xdocreport</groupId>
    <artifactId>fr.opensagres.xdocreport.template</artifactId>
    <version>2.0.2</version>
</dependency>
<dependency>
    <groupId>fr.opensagres.xdocreport</groupId>
    <artifactId>fr.opensagres.xdocreport.document.docx</artifactId>
    <version>2.0.2</version>
</dependency>
<dependency>
    <groupId>fr.opensagres.xdocreport</groupId>
    <artifactId>fr.opensagres.xdocreport.template.freemarker</artifactId>
    <version>2.0.2</version>
</dependency>
<dependency>
    <groupId>org.freemarker</groupId>
    <artifactId>freemarker</artifactId>
    <version>2.3.23</version>
</dependency>
<dependency>
    <groupId>commons-io</groupId>
    <artifactId>commons-io</artifactId>
    <version>2.5</version>
</dependency>

接下来进行接口代码编写:

3.1、文本替换

public void createReport() throws IOException, XDocReportException {
	// 获取Word模板,模板存放路径在项目的resources目录下
	Resource resource = new ClassPathResource("templates/模板.docx");
	@Cleanup InputStream inputStream = resource.getInputStream();
	HttpServletResponse response = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getResponse();
	// 设置响应类型
	response.setContentType("application/msword");
	// 设置字符编码
	response.setCharacterEncoding("utf-8");
	// 设置响应头信息
	response.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode("报告.docx", "utf-8"));
	ServletOutputStream outputStream = response.getOutputStream();
	
	try {
		// 注册xdocreport实例并加载FreeMarker模板引擎
		IXDocReport report = XDocReportRegistry.getRegistry().loadReport(inputStream, TemplateEngineKind.Freemarker);
		// 创建xdocreport上下文对象
		IContext context = report.createContext();
		// 创建xdocreport元数据
		FieldsMetadata fm = report.createFieldsMetadata();

		// 创建要替换的文本变量,key要与word模板设置的key对应
		context.put("createTime", new SimpleDateFormat("yyyy-MM-dd").format(new Date()));
		context.put("organization", "码农保护协会");
		
		// 输出
		report.setFieldsMetadata(fm);
		report.process(context, outputStream);
	} catch (IOException e) {
		e.printStackTrace();
	} finally {
		outputStream.flush();
		outputStream.close();
	}
}

3.2、图片替换

// 获取图片给 new ByteArrayImageProvider()构造函数使用
URL url = new URL("https://e12b7d3.png");
URLConnection conn = url.openConnection();
InputStream inputStream = conn.getInputStream();
// 与word模板中图片的书签名一致
fm.addFieldAsImage("wayline_img");
// 也可将图片转为byte[],再使用 new ByteArrayImageProvider()构造函数
context.put("wayline_img", new ByteArrayImageProvider(inputStream, true)); // true:使用原图宽高,false:使用word模板插入的图片宽高

3.3、循环替换文本及图片

// 加载集合元数据
fm.load("XXX", ReportVO.class, true); // XXX为模板中设置的循环域数组对象名,ReportVO为循环类
// 集合图片字段与模板图片书签映射
fm.addFieldAsImage("detail_img", "item.imageByte", NullImageBehaviour.RemoveImageTemplate); // detail_img为模板循环图片的书签名,imageByte为ReportVO的图片字段
context.put("XXX", value); // XXX为模板中设置的循环域数组对象名,value为List<ReportVO>对象,ReportVO的字段要与模板中循环域内容的key对应
  • 4
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您可以使用XDocReportFreeMarker模板生成带页数的Word文档。以下是一个示例代码: 1. 添加以下依赖项到您的项目中: ```xml <dependency> <groupId>fr.opensagres.xdocreport</groupId> <artifactId>xdocreport</artifactId> <version>1.0.6</version> </dependency> <dependency> <groupId>fr.opensagres.xdocreport</groupId> <artifactId>xdocreport-template-freemarker</artifactId> <version>1.0.6</version> </dependency> ``` 2. 创建一个FreeMarker模板,包含一个名为“page”的变量,用于显示当前页数。例如,以下是一个简单的模板: ``` <html> <head> <title>My Document</title> </head> <body> <p>Page ${page} of ${nbPages}</p> <p>This is the content of my document.</p> </body> </html> ``` 3. 在Java代码中使用XDocReportFreeMarker模板来填充Word文档。以下是一个示例代码: ```java import java.io.FileOutputStream; import java.io.InputStream; import java.io.OutputStream; import java.util.HashMap; import java.util.Map; import org.apache.poi.xwpf.usermodel.XWPFDocument; import fr.opensagres.xdocreport.core.XDocReportException; import fr.opensagres.xdocreport.core.io.XOutputStream; import fr.opensagres.xdocreport.document.registry.XDocReportRegistry; import fr.opensagres.xdocreport.template.TemplateEngineKind; import fr.opensagres.xdocreport.template.formatter.FieldsMetadata; public class GenerateWordDoc { public static void main(String[] args) { try { // Load the template InputStream in = GenerateWordDoc.class.getResourceAsStream("template.docx"); XWPFDocument document = new XWPFDocument(in); FieldsMetadata metadata = new FieldsMetadata(); metadata.addFieldAsTextStyling("page"); XDocReportRegistry.getRegistry().register(document, metadata, TemplateEngineKind.Freemarker); // Populate the template Map<String, Object> context = new HashMap<>(); context.put("page", "${page}"); context.put("nbPages", "${nbPages}"); OutputStream out = new FileOutputStream("output.docx"); XDocReportRegistry.getRegistry().getReport(document, null).process(context, new XOutputStream(out)); // Close the streams in.close(); out.close(); } catch (Exception e) { e.printStackTrace(); } } } ``` 此代码将使用名为“template.docx”的模板创建一个新的Word文档,并将“page”变量替换为当前页数。请注意,您需要使用FieldsMetadata对象来标识文档中的字段,并将其注册到XDocReportRegistry中。最后,代码将输出生成Word文档到名为“output.docx”的文件中。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值