基于one2team框架的Highcharts图表图片导出方案
这篇文章已经没有什么意义了,新版的HIghcharts提供Java图片导出解决方案,你需要做的就是下个Maven,bulid一个war就Ok了。---addedy on 2012-11-15
多说一句废话:我觉得这个功能其实对于大多数应用场景来说是多余的。
Highcharts是一个纯JSWeb图表绘制解决方案,它的功能之丰富,使用之简单可能是目前开源领域排名比较考前的优秀解决方案,它对个人使用是免费的。
它的默认版本也有图片导出功能,不过导出服务器是Highcharts官方服务器,我开发的过程试了一下,好像特别慢,图片导出服务用自己的对商业用户来说也有“便利之处”。
官方的下载页面推荐的Java图片导出方案是one2team/highcharts-serverside-export,这个方案是基于apache的batik包的,也有人直接采用batik包开发了图片导出Servlet代码 (这个代码没好像没有解决中文问题)。我是比较轴的那种人,本来one2team和上面这个代码核心是一样,我还是研究了一下怎么使用这个官方推荐的导出框架。说实话,官方推荐的这个框架不是很好用,它采用JDK6的泛型特性,项目的编译器兼容性必须提高到1.6,否则编译会出错。其次这个框架的主HighchartsExporter类的功能是转换以Json数据或者Java语言对象为数据源的导出功能,而Highchart导出服务器是要转换Highchart图表post的SVG数据。所以需要对One2Team的框架稍加改造才能使用。One2team的框架的OO设计比较复杂,我就不画具体的UML图了。只列出需要引进的几个类:
首先继承SVGRendererInternal抽象类,重写callJavascript方法,这个方法的实际作用是把其他格式的数据源转换成SVG数据,我不是很明白为什么SVG数据生成非要在Java里面调用JavaScript来做,难道为了和浏览器的高度一致?,这个地方我们直接返回chartoption即可,这个chartoption在SVG源数据导出情况下就是SVG数据本书,所以无需处理,直接返回即可。
/**
*
*/
package org.one2team.highcharts.server.export.util; import org.mozilla.javascript.ScriptableObject;
import org.one2team.highcharts.server.export.util.SVGRendererInternal;
import org.one2team.highcharts.shared.ChartOptions; /**
* @author Dipolar
*
*/
public class SVGRendererInternalSVG extends SVGRendererInternal<String> {
@Override
protected Object callJavascript (final String generalOptions, final String chartOptions) {
//return ScriptableObject.callMethod (null, SCRIPTABLE, "renderSVGFromObject", new Object [] {'(' + generalOptions + ')', chartOptions});
return chartOptions;
}
}
本来加这么一个类,导出图片就够了,我为了方便Servlet操作,我又模仿HighchartsExporter类重写了一个HttpHighchartsExporter类,代码如下:上面这两个类都放在了one2team的包类,因为涉及到的一些类是protected的,为了不改写人家的代码,所以只好放在人家的包里了。还有代码中的globalOptions参数可以省略不计,直接赋null即可,因为这个参数在实际转换中并没有用,实际转换中的globalOptions用的是一个框架本身预定义的常量。
package org.one2team.highcharts.server.export;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.OutputStream; import org.apache.commons.io.IOUtils;
import org.one2team.highcharts.server.export.ExportType;
import org.one2team.highcharts.server.export.util.*; public class HttpHighchartsExporter<T> { public HttpHighchartsExporter(ExportType type, SVGRendererInternal<T> internalRenderer) {
this.type = type;
this.renderer =
new SVGStreamRenderer<T> (new SVGRenderer<T> (internalRenderer),
type.getTranscoder ());
}
public HttpHighchartsExporter(ExportType type) {
SVGRendererInternal svgRender=new SVGRendererInternalSVG();
this.type = type;
this.renderer =
new SVGStreamRenderer<T> (new SVGRenderer<T> (svgRender),
type.getTranscoder ());
} public void export (T chartOptions,
T globalOptions,
OutputStream out) { OutputStream fos = null;
try {
fos = render (chartOptions, globalOptions, out); } catch (Exception e) {
e.printStackTrace ();
throw (new RuntimeException (e));
} finally {
if (fos != null)
IOUtils.closeQuietly (fos);
}
} private OutputStream render (T chartOptions,
T globalOptions,
OutputStream out) throws FileNotFoundException { renderer.setChartOptions (chartOptions)
.setGlobalOptions (globalOptions)
.setOutputStream (out)
.render ();
return out;
} public SVGStreamRenderer<T> getRenderer () {
return renderer;
} public ExportType getType () {
return type;
} private final SVGStreamRenderer<T> renderer; private final ExportType type;
}
下面来一下真正的Servlet的 doPost的代码,这个代码虽然解决了jpg和png的中文乱码问题,但是没有解决SVG的导出的中文乱码问题,SVG导出采用svg2svgTranscoder的话,在框架内部的字体调用上出bug了,直接输出svg,不加OutputStreamWriter包装的情况下,chrome可以正常打开这个xml文件,但是中文是乱码),IE8下在中文位置出问题,加了OutputStreamWriter包装器以后,Chrome下图片可以正常渲染但是报某元素属性值有问题,IE下报了另外一个错误。PDF的导出需要扩充下一下ExportType枚举,PDFTranscoder类也不在batik的包里在apache的fop项目里面,我尝试了一下报出Java堆空间不够的错误,我的生产服务器资源也有限,PDF导出功能就省略把。
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
try{
request.setCharacterEncoding("utf-8");//这一行解决了PNG、JPG的中文乱码问题。
String filename=request.getParameter("filename");
String type=request.getParameter("type");
type=type.replace("svg+", "");
MimeType mtype=new MimeType(type);
response.addHeader("Content-Disposition", "attachment; filename="+ filename + "."+mtype.getSubType());
response.addHeader("Content-Type", mtype.getBaseType()+"; charset=UTF-8");
ServletOutputStream out=response.getOutputStream();
String svg=request.getParameter("svg");
if (mtype.getSubType().equals("xml")){
//OutputStreamWriter writer = new OutputStreamWriter(out, "utf-8");
//writer.write(svg);
}else{
HttpHighchartsExporter<String> httpExporter = new HttpHighchartsExporter<String> (Enum.valueOf(ExportType.class,mtype.getSubType()));
httpExporter.export(svg,null,out);
}
out.flush();
out.close();
}catch(Exception e){
e.printStackTrace();
} }
结语:最后提示一下one2team虽然在做图片导出并不一定好,但是JSM系列类可以用来很方便地为前端生成图表绘制所需要的Json数据,当然这个Json数据可以你自己硬代码生成Json数据,但是既然有了与HighChart对象一一对象的Java类,为什么不用一下哪?
基于one2team框架的Highcharts图表图片导出方案的更多相关文章
- 基于ssh框架的highcharts前后台数据交互实例
Highcharts 是一个用纯JavaScript编写的一个图表库, 能够很简单便捷的在web网站或是web应用程序添加有交互性的图表,并且免费提供给个人学习.个人网站和非商业用途使用.HighCh ...
- Highcharts结合PhantomJS在服务端生成高质量的图表图片
项目背景 最近忙着给部门开发一套交互式的报表系统,来替换原有的静态报表系统. 老系统是基于dotnetCHARTING开发的,dotnetCHARTING的优势是图表类型丰富,接口调用简单,使用时只需 ...
- 在asp.net中如何自己编写highcharts图表导出到自己的服务器上来
1.准备工作:网上下载highcharts导出的关键dll. 1).Svg.dll:因为highcharts的格式其实就是一个xml,采用svg的方式画图: 2).itextsha ...
- highCharts图表入门简介
一.Highcharts简介 Highcharts:功能强大.开源.美观.图表丰富.兼容绝大多数浏览器的纯js图表库 Highcharts是一款纯javascript编写的图表库,能够很简单便捷的在W ...
- 分享我基于NPOI+ExcelReport实现的导入与导出EXCEL类库:ExcelUtility (续2篇-模板导出综合示例)
自ExcelUtility类推出以来,经过项目中的实际使用与不断完善,现在又做了许多的优化并增加了许多的功能,本篇不再讲述原理,直接贴出示例代码以及相关的模板.结果图,以便大家快速掌握,另外这些示例说 ...
- 基于ADB框架Robotium跨进程操作
转自:http://blog.csdn.net/qingchunjun/article/details/42580937 2015年2月3日更新: 有些朋友在用真机尝试本方法时,抛出了InputStr ...
- rtvue-lowcode:一款基于uniapp框架和uview组件库的开源低代码开发平台
rtvue-lowcode低代码开发平台 rtvue-lowcode一款基于uniapp框架和uview组件库的低代码开发平台,项目提供可视化拖拽编辑器,采用MIT开源协议,适用于app.小程序等项目 ...
- 基于BootStrap框架构建快速响应的GPS部标监控平台
最近一个客户要求将gps部标平台移植到bootStrap框架作为前端框架,符合交通部796部标只是他们的一个基本要求,重点是要和他们的冷链云物流平台进行适配.我自己先浏览了客户的云物流平台的界面,采用 ...
- 转:Highcharts图表控件的使用
摘要 Highcharts图表控件是目前使用最为广泛的图表控件.本文将从零开始逐步为你介绍Highcharts图表控件.通过本文,你将学会如何配置Highcharts以及动态生成Highchart图表 ...
随机推荐
- 初次见面 你好EF
EF(yif),第一次听到这个名字的时候,以为是一个帅帅的魔术师,在小编的傻傻的梦想里,就是有一天,有一个魔术师站在小编面前,变出一大捧的玫瑰花,然后,然后不要钱`(*∩_∩*)′,然而在我们的编程世 ...
- 精通CSS+DIV网页样式与布局--初探CSS
CSS英文名Cascading Style Sheet,中文名字叫层叠样式表,是用于控制页面样式并允许将样式信息与网页内容分离的一种标记性语言,DIV+CSS是WEB设计标准,它是一种网页的布局方法. ...
- 学生信息管理小系统(以XML为存储方式)
为了更好地应用XML,就写了这个小项目. 下面是我的项目的目录结构 项目思路 dao是Date Access Object 数据访问层,主要是负责操作数据 domain是实体层,类似于bean层,放置 ...
- Ext JS 6应用程序Build后出现“c is not a constructor return new c(a[0])”的处理
概述 在对Ext JS 6的应用程序打包后,时不时会出现以下错误: 由于是压缩后出现的错误,要进行调试也无从下手,因而这个错误会令新手手足无措,不知道是怎么回事. 错误原因 造成该错误的主要原因是要创 ...
- 并发编程(三): 使用C++11实现无锁stack(lock-free stack)
前几篇文章,我们讨论了如何使用mutex保护数据及使用使用condition variable在多线程中进行同步.然而,使用mutex将会导致一下问题: 等待互斥锁会消耗宝贵的时间 - 有时候是很多时 ...
- Java将网络地址对应的图片转成本地的图片
只知道浏览器使用的是HTTP协议,那么如何将网络资源使用JavaHTTP下载下来呢! 这只是一个非常简单的小示例,只是不想每次碰到关于此方面的内容忘了就无从下手! 示例创建HttpURLConn ...
- Leetcode_83_Remove Duplicates from Sorted List
本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/41728739 Given a sorted linked ...
- Device tree customization
Step 1: OEMs can create their own device tree by adding "qcom,msm-id/qcom,board-id" entry ...
- 详解EBS接口开发之物料导入API
create_item inv_item_grp.create_item(p_commit => fnd_api.g_true, -- p_item_rec => l_item_rec, ...
- 关于Android自定义view 你所需要知道的基本函数
开始时之前想吐槽一句..iphone的闹钟,12小时制.我成功的把闹钟订到了下午5:00 导致错过一班飞机.心疼改签费. 候机ing,没有事做,来写一下学习自定义view必须掌握的基本函数.这里只挑一 ...