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<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
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
@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处理
//获得图片的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下载地址:
freemarker 官网:
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
上一篇: 雾霾天气如何护肤 八招远离肌肤衰老
下一篇: Android实现短信验证功能