前言
最近在项目中,因客户要求,将页面内容(如合同协议)导出成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<String, Object>中key,如${landName};
6.在加入了图片占位的地方,会看到一片base64编码后的代码,把base64替换成${image},也就是Map<String, Object>中key,值必须要处理成base64;
代码如:<w:binData w:name="wordml://自定义.png" xml:space="preserve">${image}</w:binData>
注意:“>${image}<”这尖括号中间不能加任何其他的诸如空格,tab,换行等符号。
如果需要循环,则使用:<#list maps as map></#list> maps是Map<String, Object>中key,值为数组,map为自定义;
7. 标识替换完之后,模板就弄完了,另存为.ftl后缀文件即可。注意:一定不要用word打开ftl模板文件,否则xml内容会发生变化,导致前面的工作白做了。
代码实现
工具类WordUtils.Java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
|
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.setDefaultEncoding( "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 ;
InputStream 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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
@RequestMapping ( "/exportSellPlan" )
public @ResponseBody void exportSellPlan(Long id){
Calendar calendar = Calendar.getInstance(); // 取当前日期。
if (id!= null ){
SellPlan plan=sellService.getSellPlanInfo(id);
//获得数据
Map<String, Object> map = new HashMap<String, Object>();
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处理
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
//获得图片的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
1
|
window.location.href= "<%=path%>/exportSellPlan?id=" rel= "external nofollow" + id;
|
结束语
如果对Freemarker标签不熟的,可以在网上先学习下,了解文档结构。
相关链接
Firstobject free XML editor下载地址:http://www.firstobject.com/dn_editor.htm
freemarker 官网:http://freemarker.org/
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:http://www.cnblogs.com/zwqh/p/7182953.html?utm_source=tuicool&utm_medium=referral