Java后台报表尝试了很多,最终发现了一款,而且是开源的,简表地址:http://www.jatools.com/jor/。问题的引入:该报表支持嵌套,钻去,应对excel类似的报表,足够了。但是,报表的图表特别难看,所以想改一下,结合现在流行的图表Echart, 访问地址:http://echarts.baidu.com/, 开源了这个解决方案,和大家一起完善这个报表。

Jor 报表的设计:使用awt 展示报表,最终使用的是graphics 对象,然后使用jreport 等开源组件,实现导出pdf,导出word ,导出excel 等。

Echrt 的引入,Echart 是javascript 组件,然后执行javascript,使用echart 的getDataURL可以得到Echart 展示的图片的base64 编码,这样,结合读取图片,展示grahics ,就可以展示出来,然后顺便导出等等。Js这部分,可以根据绑定的数据源,动态写js 方法,修改html 文件,然后展示即可。

方案1:

使用HtmlUnit组件,即http://htmlunit.sourceforge.net/。 展示html ,加载js,然后导出base64编码。大致源码如下:

final WebClient webClient = new WebClient(BrowserVersion.CHROME);

            final HtmlPage page = webClient.getPage("http://localhost:8000/line2.html");

            System.out.println(" // 1 启动JS ");
webClient.getOptions().setJavaScriptEnabled(true);
System.out.println("// 2 禁用Css,可避免自动二次请求CSS进行渲染 ");
webClient.getOptions().setCssEnabled(false);
System.out.println("// 3 启动客户端重定向 ");
webClient.getOptions().setRedirectEnabled(true);
System.out.println("// 4 js运行错误时,是否抛出异常");
webClient.getOptions().setThrowExceptionOnScriptError(false);
System.out.println("// 5 设置超时 ");
webClient.getOptions().setTimeout(50000);
System.out.println(" 允许绕过SSL认证 ");
webClient.getOptions().setUseInsecureSSL(true);
System.out.println(" 允许启动注册组件 ");
webClient.getOptions().setActiveXNative(true); System.out.println(" //等待JS驱动dom完成获得还原后的网页 ");
webClient.waitForBackgroundJavaScript(5000); webClient.getOptions().setThrowExceptionOnFailingStatusCode(false);
webClient.getCookieManager().setCookiesEnabled(true);
webClient.setAjaxController(new NicelyResynchronizingAjaxController()); webClient.addWebWindowListener( new WebWindowListener() {
@Override
public void webWindowOpened(WebWindowEvent webWindowEvent) {
System.out.println("windows opened");
} @Override
public void webWindowContentChanged(WebWindowEvent webWindowEvent) {
System.out.println("windows changed");
} @Override
public void webWindowClosed(WebWindowEvent webWindowEvent) {
System.out.println("windows closed");
}
}); try {
Thread.sleep(10000);
}catch (Exception exp) { }
final HtmlDivision div = page.getHtmlElementById("text2"); //执行按钮出发的js事件
ScriptResult sr = page.executeJavaScript("javascript:getData();"); try {
String fileStr = "";
String test = sr.getJavaScriptResult().toString();
byte[] b = new BASE64Decoder().decodeBuffer(test); // 生成图片
OutputStream out = new FileOutputStream(new File(fileStr + "\\test.png"));
out.write(b);
out.flush();
out.close();
}catch (Exception exp) {
exp.printStackTrace();
} }catch (Exception exp) {
exp.printStackTrace();
}

结果: 不行,执行不了js,原因没有深入研究。

方案2:

JDIC,https://www.ibm.com/developerworks/cn/java/j-jdic/ ,JDIC 是java 组件,可以展示html,执行javascript。

代码如下:

//BrowserEngineManager bem = BrowserEngineManager.instance();
// bem.setActiveEngine(BrowserEngineManager.IE);
//IBrowserEngine be = bem.getActiveEngine(); //URL url = new URL("http://www.hao123.com");
URL url = new File("http://localhost.:8000/line2.html").toURI().toURL();
final WebBrowser browser = new WebBrowser();
//browser = be.getWebBrowser();//new WebBrowser();
browser.addWebBrowserListener(new WebBrowserListener() {
public void downloadStarted(WebBrowserEvent event) {
System.out.println("27");
}
public void downloadCompleted(WebBrowserEvent event) {
System.out.println("30");
}
public void downloadProgress(WebBrowserEvent event) {
System.out.println("33");
}
public void downloadError(WebBrowserEvent event) {
System.out.println("36");
}
public void documentCompleted(WebBrowserEvent event) {
System.out.println("39");
browser.executeScript("alert('文档下载完毕!')");
String res = browser.executeScript("getData");
System.out.println(res);
}
public void titleChange(WebBrowserEvent event) {
System.out.println("43");
}
public void statusTextChange(WebBrowserEvent event) {
System.out.println("46");
}
public void windowClose(WebBrowserEvent webBrowserEvent) {
System.out.println("49");
}
public void initializationCompleted(WebBrowserEvent arg0) {
System.out.println("52");
} });
browser.setURL(url); JFrame f = new JFrame();
f.setTitle("浏览器");
f.setSize(800,600);
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
//f.getContentPane().add(browser1);
f.getContentPane().add(browser);
f.setVisible(true);
}

  

结果: 代码跑不起来,dll 都是32 位,没有办法使用,所以放弃。

方案3:

使用c# winform 的webbrowser 控件,加载html, 然后执行js脚本。

代码:

  private void init()
{
InitializeComponent();
CleanTempFiles();
webBrowser1.AllowWebBrowserDrop = false;
webBrowser1.WebBrowserShortcutsEnabled = false;
webBrowser1.IsWebBrowserContextMenuEnabled = false;
webBrowser1.Navigate(htmlurl + "?random=" + DateTime.Now.ToString("yyyyMMddHHmmss"), null, null, null);
}
private void webBrowser1_DocumentCompleted(object sender, WebBrowserDocumentCompletedEventArgs e)
{
try
{
var doc = this.webBrowser1.Document;
var ele = doc.GetElementById("text2");
Console.WriteLine(ele.InnerText);
// 读取base64 , 然后转换为图片,保存
string base64 = ele.InnerText.Split(',')[];
byte[] arr = Convert.FromBase64String(base64);
MemoryStream ms = new MemoryStream(arr);
Bitmap bmp = new Bitmap(ms); //bmp.Save(txtFileName + ".jpg", System.Drawing.Imaging.ImageFormat.Jpeg);
//bmp.Save(txtFileName + ".bmp", ImageFormat.Bmp);
//bmp.Save(txtFileName + ".gif", ImageFormat.Gif);
//bmp.Save(txtFileName + ".png", ImageFormat.Png);
// bmp.Save(imageGuid + ".jpg", System.Drawing.Imaging.ImageFormat.Jpeg);
bmp.Save(imageGuid + ".png", System.Drawing.Imaging.ImageFormat.Png);
ms.Close(); }catch(Exception exp)
{
Console.WriteLine(exp.ToString());
}
finally
{
Application.Exit();
}
}

结果: 可行, 导出图片成功, 所以采用。

接下来要做:

1,    实现所有图表,代码中只实现了linechart,其他,饼图,什么的都需要更改。

2,    增加脚步选项,直接在界面中设置js代码,高度定制。

开源地址:

https://github.com/gitsteven/jorreport-echart

简表-Java-Echart报表介绍的更多相关文章

  1. 葡萄城报表介绍:Java 报表

    一.Java 报表定义 Java 是一门面向对象编程语言,不仅吸收了 C++ 语言的各种优点,还摒弃了 C++ 里难以理解的多继承.指针等概念,因此 Java 语言具有功能强大和简单易用两个特征.Ja ...

  2. JAVA基本类库介绍

    我们曾经讲过,Java已经为编程者编制了许多类,这些类已经经过测试,基本上不存在错误,这些类都是我们编程的基础.如果不利用这些已存在的类,我们的 编程工作将变得异常复杂,所以我们应尽可能多的掌握Jav ...

  3. java.util.concurrent介绍【转】

    java.util.concurrent介绍   java.util.concurrent 包含许多线程安全.测试良好.高性能的并发构建块.不客气地说,创建 java.util.concurrent ...

  4. JDK框架简析--java.lang包中的基础类库、基础数据类型

    题记 JDK.Java Development Kit. 我们必须先认识到,JDK不过,不过一套Java基础类库而已,是Sun公司开发的基础类库,仅此而已,JDK本身和我们自行书写总结的类库,从技术含 ...

  5. EChart报表插件使用笔记(1)

    报表插件Echart java类 package com.spring.controller; import java.io.IOException; import java.util.Arrays; ...

  6. echart报表插件使用笔记(二)--按月统计

    按月统计注冊人数 java类: package com.spring.controller; import java.io.IOException; import java.sql.Connectio ...

  7. 记录面试龙腾简合-java开发工程师经历

    /** * ############ * 变强是会掉光头发的!现在的头发还是很茂盛,是该开心还是难过呢.. * ############ * / 总结下近期面试龙腾简合-java开发岗的经历.附上笔试 ...

  8. Java语言的介绍

    1. 计算机语言 语言:沟通交流的方式 计算机语言:人与计算机之间的交流方式 java是一门计算机编程语言,也是意大利自行车品牌 软件工程师,java开发工程师 <--------------- ...

  9. ionic3引用外部插件--百度地图及echart报表的使用

    前言 ionic3提供的组件已经相当丰富咯,但是事实上有些特殊的需求,比如使用百度地图,或者第三方插件echart报表插件是,就不能用传统的方式去使用第三方插件咯,如何在Ionic3项目中使用第三方J ...

随机推荐

  1. Java获取时间,将当前时间减一年,减一天,减一个月

    在Java中操作时间的时候,需要计算某段时间开始到结束的区间日期,常用的时间工具 Date date = new Date();//获取当前时间 Calendar calendar = Calenda ...

  2. CSS图片下面产生间隙的6种解决方案

    CSS图片下面产生间隙的6种解决方案 在进行页面的DIV+CSS排版时,遇到IE6(当然有时Firefox下也会偶遇)浏览器中的图片元素img下出现多余空白的问题绝对是常见的对於 该问题的解决方法也是 ...

  3. NOI2001 方程的解数(双向搜索)

    solution 一道非常经典的双向搜索题目,先将前3个未知数枚举一遍得到方程的前半部分所有可能的值,取负存入第一个队列中再将后3个未知数枚举一遍,存入第二个队列中.这样我们只要匹配两个队列中相同的元 ...

  4. mybatis多对多关联查询——(十)

    1.需求 查询用户及用户购买商品信息. 2     sql语句 查询主表是:用户表 关联表:由于用户和商品没有直接关联,通过订单和订单明细进行关联,所以关联表: orders.orderdetail. ...

  5. python3之模块io使用流的核心工具

    1.io概叙 io模块提供了python用于处理各种类型I/O的主要工具,主要有三种类型的I/O:文本I/O,二进制I/O和原始I/O:这些都是通用类型,各种后备存储可使用其中的每一种类型,所以这些类 ...

  6. vim助手

    移动光标 hjkl 2w 向前移动两个单词 3e 向前移动到第 3 个单词的末尾 0 移动到行首 $ 当前行的末尾 gg 文件第一行 G 文件最后一行 行号+G 指定行 <ctrl>+o ...

  7. TCP长连接和短连接的区别【转】

    转自:https://www.cnblogs.com/onlysun/p/4520553.html 当网络通信时采用TCP协议时,在真正的读写操作之前,server与client之间必须建立一个连接, ...

  8. ubuntu12.04安装maven

    step: 1,确认已经安装jdk, java --version 2,下载apache-maven-3.3.9 下载地址:http://maven.apache.org/download.cgi 3 ...

  9. XSS代码注入框架

    首先需要了解一下几点: 1.浏览器中Javascript变量的生命周期 Javascript变量的生命周期并不是你声明这个变量个窗口闭就被回收,只要有引用就会一直持续到浏览器关闭. 2.window对 ...

  10. mapreduce的组件介绍

    第一部分:重要的组件 Combiner •什么是Combiner •combine函数把一个map函数产生的<key,value>对(多个key, value)合并成一个新的<key ...