【java】Echarts导出为word文档

最近在使用Echarts做图表分析,同时要求有导出分析报告的功能,这时Echarts自带的保存图片功能就不能满足自己的需求,因此在网上找了许多前辈的资料,在此拾人牙慧,小作总结。

主要思路:

前台echarts生成图片后,获取base64码,传给后台解析,然后写入freemarker模板,进行下载。


第一步:

echarts的生成就不再多做赘述,可参考:echarts官网

图表生成后,使用 myChart.getDataURL("png")获取base64码,其格式为:"data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAABI4AAAEsCAYAAAClh/jbAAA ..."

定义一个简单的测试点击按钮:<button type="button" id="button1">导出报告</button>,考虑到后台要接收base64码,同时要下载,因此选择使用ajax传递


document.getElementById("button1").onclick = function(){

//alert(myChart.getDataURL("png"))

var image = myChart.getDataURL("png");

$.ajax({

          type: 'post',

          url: "/****Servlet?method=getImage",

          data : {'image':image},

         dataType : "text",

         success : function(data) {

                  window.location.href="/*****Servlet?method=download"

         },

         error : function(error, status) {

         alert("机构数据错误!");

}

});

}


第二步:
生成word文档的freemarker模板。
首先创建一个word文档,里面加上自己所需要的内容
如:

image.png

其中${name}的内容,即为所需动态写入的内容。
然后选择另存为Word 2003 XML 文档(*.xml),注意名称不要包含中文。用xml编译器打开文件,推荐使用Firstobject free XML editor,按F8格式化文档,左边是文档结构,右边就是文档内容。找到原图片的base64码部分,替换成自己的${image}。代码如:
<w:binData w:name="wordml://自定义.png" xml:space="preserve">${image}</w:binData>

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

同时需注意文档中:

<w:t>${</w:t>
</w:r>
<w:r>
<w:t>name</w:t>
</w:r>
<w:r>
<w:rPr>
<w:rFonts w:hint="fareast"/>
</w:rPr>
<w:t>}</w:t>

之前修改的动态内容,由于word的xml格式化,也需要再次调整,改为<w:t>${name}</w:t>即可,一定注意删除和修改的标签。都调整完之后,就可以另存为.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.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 WordUtil {
     //配置信息,代码本身写的还是很可读的,就不过多注解了  
    private static Configuration configuration = null;  
    //这里注意的是利用WordUtils的类加载器动态获得模板文件的位置  
   // private static final String templateFolder = WordUtils.class.getClassLoader().getResource("../../").getPath() + "WEB-INF/templetes/";  
    private static final String templateFolder = "D:/***/js2";  
    static {  
        configuration = new Configuration();  
        configuration.setDefaultEncoding("utf-8");  
        try {  
            configuration.setDirectoryForTemplateLoading(new File(templateFolder));  
        } catch (IOException e) {  
            e.printStackTrace();  
        }  
   }  
  
    private WordUtil() {  
        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+ ".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;  
    }  
}

servlet中
设置全局变量private static String image
doGet()中

          if(method.equals("download")){
            try {
                download(request, response);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        if(method.equals("getImage")){
            try {
                getImage(request, response);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

处理前台传递的base64码

    public void getImage(HttpServletRequest request, HttpServletResponse response){
        String src = request.getParameter("image");
        System.out.println(src);
        String base64 = src.replaceAll(" ", "+");
        // 数据中:data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAABI4AAAEsCAYAAAClh/jbAAA ...  在"base64,"之后的才是图片信息
        String[] arr = base64.split("base64,"); 
        image = arr[1];
        System.out.println(image);

    }

使用map,写入模板

    /**
     * echarts导出
     * @throws IOException 
     */
    public void download(HttpServletRequest request, HttpServletResponse response){
        Map<String, String> map = new HashMap<String, String>();
        map.put("name", "机构预警");
        map.put("title", "预警玫瑰图");
        map.put("image", image);
        map.put("bz", "12");
        map.put("jl", "12");
        map.put("hb", "12");
        map.put("jk", "12");
        map.put("lj", "12");
        map.put("dw", "12");
        map.put("jx", "12");
        map.put("gw", "12");
        map.put("qp", "12");
        
        try {
            WordUtil.exportMillCertificateWord(request, response, map, "分析报告", "test.ftl");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

最后就是测试了
效果图:


image.png

结语:
第一次写,轻喷。由于多为测试代码,故细节部分未做处理,主要注重于功能的实现。
完结撒花

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容