Java实现用Freemarker完美导出word文档(带图片)

网友投稿 904 2023-04-27


Java实现用Freemarker完美导出word文档(带图片)

前言

最近在项目中,因客户要求,将页面内容(如合同协议)导出成word,在网上翻了好多,感觉太乱了,不过最后还是较好解决了这个问题。

准备材料

1.word原件 2.编辑器(推荐Firstobject free XML editor)

实现步骤

1.用Microsoft Office Word打开word原件;

2.把需要动态修改的内容替换成***,如果有图片,尽量选择较小的图片几十K左右,并调整好位置;

3.另存为,选择保存类型Word 2003 XML 文档(*.xml)【这里说一下为什么用Microsoft Office Word打开且要保存为Word 2003XML,本人亲测,用WPS找不到Word 2003XML选项,如果保存为Word XML,会有兼容问题,避免出现导出的word文档不能用Word 2003打开的问题】;

4.用Firstobject free XML editor打开文件,选择Tools下的Indent【或者按快捷键F8】格式化文件内容。左边是文档结构,右边是文档内容;

5. 将文档内容中需要动态修改内容的地方,换成freemarker的标识。其实就是Map中key,如${landName};

6.在加入了图片占位的地方,会看到一片base64编码后的代码,把base64替换成${image},也就是Map中key,值必须要处理成base64;

代码如:${image}

注意:“>${image}<”这尖括号中间不能加任何其他的诸如空格,tab,换行等符号。

如果需要循环,则使用:<#list maps as map>#list>  maps是Map中key,值为数组,map为自定义;

7. 标识替换完之后,模板就弄完了,另存为.ftl后缀文件即可。注意:一定不要用word打开ftl模板文件,否则xml内容会发生变化,导致前面的工作白做了。

代码实现

工具类WordUtils.java

import java.io.File;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.InputStream;

import java.io.OutputStreamWriter;

import java.io.Writer;

import java.net.URLEncoder;

import java.util.Date;

import java.util.Map;

import javax.servlet.ServletOutputStream;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import freemarker.template.Configuration;

import freemarker.template.Template;

public class WordUtils {

//配置信息,代码本身写的还是很可读的,就不过多注解了

private static Configuration configuration = null;

//这里注意的是利用WordUtils的类加载器动态获得模板文件的位置

// private static final String templateFolder = WordUtils.class.getClassLoader().getResource("../../").getPath() + "WEB-INF/templetes/";

private static final String templateFolder = "H:/我的项目/lm/lm/web/src/main/webapp/WEB-INF/templates";

static {

configuration = new Configuration();

configuration.shttp://etDefaultEncoding("utf-8");

try {

configuration.setDirectoryForTemplateLoading(new File(templateFolder));

} catch (IOException e) {

e.printStackTrace();

}

}

private WordUtils() {

throw new AssertionError();

}

public static void exportMillCertificateWord(HttpServletRequest request, HttpServletResponse response, Map map,String title,String ftlFile) throws IOException {

Template freemarkerTemplate = configuration.getTemplate(ftlFile);

File file = null;

InputStVaGvUnZMOream fin = null;

ServletOutputStream out = null;

try {

// 调用工具类的createDoc方法生成Word文档

file = createDoc(map,freemarkerTemplate);

fin = new FileInputStream(file);

response.setCharacterEncoding("utf-8");

response.setContentType("application/msword");

// 设置浏览器以下载的方式处理该文件名

String fileName = title+DateUtil.formatDateDetailTime(new Date()) + ".doc";

response.setHeader("Content-Disposition", "attachment;filename="

.concat(String.valueOf(URLEncoder.encode(fileName, "UTF-8"))));

out = response.getOutputStream();

byte[] buffer = new byte[512]; // 缓冲区

int bytesToRead = -1;

// 通过循环将读入的Word文件的内容输出到浏览器中

while((bytesToRead = fin.read(buffer)) != -1) {

out.write(buffer, 0, bytesToRead);

}

} finally {

if(fin != null) fin.close();

if(out != null) out.close();

if(file != null) file.delete(); // 删除临时文件

}

}

private static File createDoc(Map, ?> dataMap, Template template) {

String name = "sellPlan.doc";

File f = new File(name);

Template t = template;

try {

// 这个地方不能使用FileWriter因为需要指定编码类型否则生成的Word文档会因为有无法识别的编码而无法打开

Writer w = new OutputStreamWriter(new FileOutputStream(f), "utf-8");

t.process(dataMap, w);

w.close();

} catch (Exception ex) {

ex.printStackTrace();

throw new RuntimeException(ex);

}

return f;

}

}

Action

@RequestMapping("/exportSellPlan")

public @ResponseBody void exportSellPlan(Long id){

Calendar calendar = Calendar.getInstance();// 取当前日期。

if(id!=null){

SellPlan plan=sellService.getSellPlanInfo(id);

//获得数据

Map map = new HashMap();

map.put("bYear", plan.getBusinessYear()!=null?plan.getBusinessYear():"");

map.put("lYear", plan.getLiveYear()!=null?plan.getLiveYear():"");

map.put("leader",plan.getLeader()!=null?plan.getLeader():"");

map.put("phone", plan.getPhone()!=null?plan.getPhone():"");

map.put("curYear", calendar.get(Calendar.YEAR)+"");

map.put("image", getImageBase(plan.getPositionImage()));

try {

WordUtils.exportMillCertificateWord(getRequest(),getResponse(),map,"方案","sellPlan.ftl");

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

Base64处理

//获得图片的base64码

@SuppressWarnings("deprecation")

public String getImageBase(String src) {

if(src==null||src==""){

return "";

}

File file = new File(getRequest().getRealPath("/")+src.replace(getRequest().getContextPath(), ""));

if(!file.exists()) {

return "";

}

InputStream in = null;

byte[] data = null;

try {

in = new FileInputStream(file);

} catch (FileNotFoundException e1) {

e1.printStackTrace();

}

try {

data = new byte[in.available()];

in.read(data);

in.close();

} catch (IOException e) {

e.printStackTrace();

}

BASE64Encoder encoder = new BASE64Encoder();

return encoder.encode(data);

}

javascript

window.location.href="<%=path%>/exportSellPlan?id=" rel="external nofollow" + id;

结束语

如果对Freemarker标签不熟的,可以在网上先学习下,了解文档结构。

相关链接

Firstobject free XML editor下载地址:http://firstobject.com/dn_editor.htm

freemarker 官网:http://freemarker.org/


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Spring实战之XML与JavaConfig的混合配置详解
下一篇:springmvc的validator数据校验的实现示例代码
相关文章

 发表评论

暂时没有评论,来抢沙发吧~