前言
最近在項目中,因客戶要求,將頁面內容(如合同協議)導出成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 import java.io.File;
2 import java.io.FileInputStream;
3 import java.io.FileOutputStream;
4 import java.io.IOException;
5 import java.io.InputStream;
6 import java.io.OutputStreamWriter;
7 import java.io.Writer;
8 import java.net.URLEncoder;
9 import java.util.Date;
10 import java.util.Map;
11
12 import javax.servlet.ServletOutputStream;
13 import javax.servlet.http.HttpServletRequest;
14 import javax.servlet.http.HttpServletResponse;
15
16 import freemarker.template.Configuration;
17 import freemarker.template.Template;
18
19 public class WordUtils {
20 //配置信息,代碼本身寫的還是很可讀的,就不過多注解了
21 private static Configuration configuration = null;
22 //這里注意的是利用WordUtils的類加載器動態獲得模板文件的位置
23 // private static final String templateFolder = WordUtils.class.getClassLoader().getResource("../../").getPath() + "WEB-INF/templetes/";
24 private static final String templateFolder = "H:/我的項目/lm/lm/web/src/main/webapp/WEB-INF/templates";
25 static {
26 configuration = new Configuration();
27 configuration.setDefaultEncoding("utf-8");
28 try {
29 configuration.setDirectoryForTemplateLoading(new File(templateFolder));
30 } catch (IOException e) {
31 e.printStackTrace();
32 }
33 }
34
35 private WordUtils() {
36 throw new AssertionError();
37 }
38
39 public static void exportMillCertificateWord(HttpServletRequest request, HttpServletResponse response, Map map,String title,String ftlFile) throws IOException {
40 Template freemarkerTemplate = configuration.getTemplate(ftlFile);
41 File file = null;
42 InputStream fin = null;
43 ServletOutputStream out = null;
44 try {
45 // 調用工具類的createDoc方法生成Word文檔
46 file = createDoc(map,freemarkerTemplate);
47 fin = new FileInputStream(file);
48
49 response.setCharacterEncoding("utf-8");
50 response.setContentType("application/msword");
51 // 設置瀏覽器以下載的方式處理該文件名
52 String fileName = title+DateUtil.formatDateDetailTime(new Date()) + ".doc";
53 response.setHeader("Content-Disposition", "attachment;filename="
54 .concat(String.valueOf(URLEncoder.encode(fileName, "UTF-8"))));
55
56 out = response.getOutputStream();
57 byte[] buffer = new byte[512]; // 緩沖區
58 int bytesToRead = -1;
59 // 通過循環將讀入的Word文件的內容輸出到瀏覽器中
60 while((bytesToRead = fin.read(buffer)) != -1) {
61 out.write(buffer, 0, bytesToRead);
62 }
63 } finally {
64 if(fin != null) fin.close();
65 if(out != null) out.close();
66 if(file != null) file.delete(); // 刪除臨時文件
67 }
68 }
69
70 private static File createDoc(Map<?, ?> dataMap, Template template) {
71 String name = "sellPlan.doc";
72 File f = new File(name);
73 Template t = template;
74 try {
75 // 這個地方不能使用FileWriter因為需要指定編碼類型否則生成的Word文檔會因為有無法識別的編碼而無法打開
76 Writer w = new OutputStreamWriter(new FileOutputStream(f), "utf-8");
77 t.process(dataMap, w);
78 w.close();
79 } catch (Exception ex) {
80 ex.printStackTrace();
81 throw new RuntimeException(ex);
82 }
83 return f;
84 }
85 }
Action
1 @RequestMapping("/exportSellPlan")
2 public @ResponseBody void exportSellPlan(Long id){
3 Calendar calendar = Calendar.getInstance();// 取當前日期。
4 if(id!=null){
5 SellPlan plan=sellService.getSellPlanInfo(id);
6 //獲得數據
7 Map<String, Object> map = new HashMap<String, Object>();
8 map.put("bYear", plan.getBusinessYear()!=null?plan.getBusinessYear():"");
9 map.put("lYear", plan.getLiveYear()!=null?plan.getLiveYear():"");
10 map.put("leader",plan.getLeader()!=null?plan.getLeader():"");
11 map.put("phone", plan.getPhone()!=null?plan.getPhone():"");
12 map.put("curYear", calendar.get(Calendar.YEAR)+"");
13 map.put("image", getImageBase(plan.getPositionImage()));
14 try {
15 WordUtils.exportMillCertificateWord(getRequest(),getResponse(),map,"方案","sellPlan.ftl");
16 } catch (IOException e) {
17 // TODO Auto-generated catch block
18 e.printStackTrace();
19 }
20 }
21 }
Base64處理
1 //獲得圖片的base64碼
2 @SuppressWarnings("deprecation")
3 public String getImageBase(String src) {
4 if(src==null||src==""){
5 return "";
6 }
7 File file = new File(getRequest().getRealPath("/")+src.replace(getRequest().getContextPath(), ""));
8 if(!file.exists()) {
9 return "";
10 }
11 InputStream in = null;
12 byte[] data = null;
13 try {
14 in = new FileInputStream(file);
15 } catch (FileNotFoundException e1) {
16 e1.printStackTrace();
17 }
18 try {
19 data = new byte[in.available()];
20 in.read(data);
21 in.close();
22 } catch (IOException e) {
23 e.printStackTrace();
24 }
25 BASE64Encoder encoder = new BASE64Encoder();
26 return encoder.encode(data);
27 }
Javascript
1 window.location.href="<%=path%>/exportSellPlan?id=" + id;
pom.xml
<dependency>
<groupId>org.freemarker</groupId>
<artifactId>freemarker</artifactId>
<version>2.3.20</version>
</dependency>
結束語
如果對Freemarker標簽不熟的,可以在網上先學習下,了解文檔結構。
相關鏈接
Firstobject free XML editor下載地址:http://www.firstobject.com/dn_editor.htm
freemarker 官網:http://freemarker.org/

