`
aijnecJay
  • 浏览: 58420 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Jasperreport的客户端打印

阅读更多
Jasperreport客户端打印


Jasperreport以及配合Ireport工具作为模板报表的开源解决方案,想必大家都非常熟悉了。
Jasperreport可以根据Ireport制作的模板,方便地填入业务数据,去生成包括PDF,HTML等各种格式的报表文件,基本能满足常见的各种需求。而当用在BS框架的应用里如电子商务,我们可能就要面对一个客户端打印的问题。

由于在BS框架中,Jasperreport处理和生成报表的代码都是部署在服务端,势必只能在服务端生成报表文件。在业务上要求客户端打印的时候,我们可以有几种方案,一种就是把报表文件在服务端生成好,通过下载数据流等方式下载到本地,进行打印。这种方式开发模式最简单,但是客户体验度很差。另一种方式考虑把报表文件转成图片或者HTMl等浏览器直接能解析的格是,通过浏览器打开,浏览并打印。这个方案的问题在于,如果进入浏览器菜单触发打印,客户的体验度还是较差;而如果通过页面按钮来实现web打印,要兼容性好的话还得使用ActiveX类的web打印控件,也不够简练。

这里要和大家介绍第三种方案,通过JAVA Applet控件方式来实现Japserreport客户端打印。
JAVA的Applet在BS框架中用的较少,好在Jasperreport的包里已经为我们写好了例子,但是实际开发中还是要面临各种问题,本文对此会一一讲述。

我们来看例子:
Applet方式的原理是本地下载Applet以及Jasperreport applet的JAR包,然后服务端准备好被打印对象,applet通过如servlet的方式获得服务端Jasperreport的打印对象,调用本地的打印接口。

首先applet要放在web访问能看到的目录下,可以直接在webapp的根目录下建一个applets目录,里面放你的applet类。
构建和编译applet的时候,建议单独启一个project,编译好class然后贴到项目里的applets目录里,以免和项目的build path混淆。

参考官方的applet例子:
public class PrinterApplet extends javax.swing.JApplet
{


	/**
	 *
	 */
	private URL url;
	private JasperPrint jasperPrint;


	/** Creates new form AppletViewer */
	public PrinterApplet()
	{
		initComponents();
	}


	/**
	*
	*/
	public void init()
	{
		String strUrl = getParameter("REPORT_URL");
		if (strUrl != null)
		{
			try
			{
				url = new URL(getCodeBase(), strUrl);
			}
			catch (Exception e)
			{
				StringWriter swriter = new StringWriter();
				PrintWriter pwriter = new PrintWriter(swriter);
				e.printStackTrace(pwriter);
				JOptionPane.showMessageDialog(this, swriter.toString());
			}
		}
		else
		{
		JOptionPane.showMessageDialog(this, "Source URL not specified");
		}
	}


	/** This method is called from within the constructor to
	 * initialize the form.
	 * WARNING: Do NOT modify this code. The content of this method is
	 * always regenerated by the Form Editor.
	 */
	private void initComponents() {//GEN-BEGIN:initComponents
		pnlMain = new javax.swing.JPanel();
		btnPrint = new javax.swing.JButton();
		btnView = new javax.swing.JButton();

		btnPrint.setText("Print the report");
		btnPrint.addActionListener(new java.awt.event.ActionListener() {
			public void actionPerformed(java.awt.event.ActionEvent evt) {
				btnPrintActionPerformed(evt);
			}
		});

		pnlMain.add(btnPrint);

		btnView.setText("View the report");
		btnView.addActionListener(new java.awt.event.ActionListener() {
			public void actionPerformed(java.awt.event.ActionEvent evt) {
				btnViewActionPerformed(evt);
			}
		});

		pnlMain.add(btnView);

		getContentPane().add(pnlMain, java.awt.BorderLayout.WEST);

	}//GEN-END:initComponents

	protected void btnViewActionPerformed(java.awt.event.ActionEvent evt) {//GEN-FIRST:event_btnViewActionPerformed
		// Add your handling code here:
		if (url != null)
		{
			try
			{
				if (jasperPrint == null)
				{
					jasperPrint = (JasperPrint)JRLoader.loadObject(url);
				}
				if (jasperPrint != null)
				{
					ViewerFrame viewerFrame = new ViewerFrame(this.getAppletContext(), jasperPrint);
					viewerFrame.show();
				}
				else
				{
					JOptionPane.showMessageDialog(this, "Empty report.");
				}
			}
			catch (Exception e)
			{
				StringWriter swriter = new StringWriter();
				PrintWriter pwriter = new PrintWriter(swriter);
				e.printStackTrace(pwriter);
				JOptionPane.showMessageDialog(this, swriter.toString());
			}
		}
		else
		{
			JOptionPane.showMessageDialog(this, "Source URL not specified");
		}
	}//GEN-LAST:event_btnViewActionPerformed

	protected void btnPrintActionPerformed(java.awt.event.ActionEvent evt) {//GEN-FIRST:event_btnPrintActionPerformed
		// Add your handling code here:
		if (url != null)
		{
			if (jasperPrint == null)
			{
				try
				{
					jasperPrint = (JasperPrint)JRLoader.loadObject(url);
				}
				catch (Exception e)
				{
					StringWriter swriter = new StringWriter();
					PrintWriter pwriter = new PrintWriter(swriter);
					e.printStackTrace(pwriter);
					JOptionPane.showMessageDialog(this, swriter.toString());
				}
			}
			
			if (jasperPrint != null)
			{				
				final JasperPrint print = jasperPrint;
				
				Thread thread = new Thread
					(
						new Runnable()
						{
							public void run()
							{
								try 
								{
									JasperPrintManager.printReport(print, true);
								}
								catch (Exception e) 
								{
									StringWriter swriter = new StringWriter();
									PrintWriter pwriter = new PrintWriter(swriter);
									e.printStackTrace(pwriter);
									JOptionPane.showMessageDialog(null, swriter.toString());
								}
							}
						}
					);
				
				thread.start();
				
				 // JSObject win = JSObject.getWindow(this);
				 // JSObject doc =(JSObject)win.getMember("document");
				  
                 // doc.eval("alert('123')");
			}
			else
			{
				JOptionPane.showMessageDialog(this, "Empty report.");
			}
		}
		else
		{
			JOptionPane.showMessageDialog(this, "Source URL not specified");
		}
	}//GEN-LAST:event_btnPrintActionPerformed
	
	
	// Variables declaration - do not modify//GEN-BEGIN:variables
	private javax.swing.JPanel pnlMain;
	private javax.swing.JButton btnView;
	private javax.swing.JButton btnPrint;
	// End of variables declaration//GEN-END:variables
	
}

我们看到这个applet加入了两个按钮以及事件Print the report和View the report,applet的代码参考官方的例子已经很清楚了,不用做什么大改。
然后我们要注意,因为PrinterApplet需要调用本地打印的接口,所以牵涉到applet的权限问题,有两种方案,一种是通过更改jre对applet的授权。
如使用appletviewer url方式调试APPLET的话,需要在对应的系统JRE如
C:\Program Files\Java\jre7\lib\security\java.policy 文件中加入
permission java.security.AllPermission;
来进行授权。(系统或用户的JRE具体环境配置可以通过控制面板里的JAVA控制台进行察看,同时APPLET使用了缓存机制,也可以在JAVA控制台对缓存进行清理)。

当然,对于大规模的客户端来说,对每个客户端进行文件修改这样的方式是基本不可行的,
这里我们用了第二种方案,就是对JAR包进行签名。
把编译好的applet class进行打包
jar -cvf print.jar *.class  (我打了一个print.jar的包)

创建一个证书
keytool -genkey -validity 1800 -keystore applet.store -alias applet  
导出证书文件
keytool -export -keystore applet.store -alias applet -file applet.cer
对jar包进行签名
jarsigner -keystore applet.store jasperreports-applet-4.6.0 .jar applet
jarsigner -keystore applet.store print.jar applet

注意print.jar和jasperreports-applet 的jar 包必须都要签名。

签名以后,jar包里的applet就拥有了操作本地IO设备的权限,当然,这个例子里创建的证书并未获得CA认证,在实际使用中,客户端会出现未受信证书的提示和风险提示,当然这并不影响功能的使用,如果有条件的话可以使用受信证书。

另外要提一点的是,用官方自带例子构建出的jasperreports-applet-4.6.0.jar (测试了4.1的版本,同样的问题)是有BUG的,在net\sf\jasperreports\engine\print 这个目录下少一个JRPrintAWT$1.class 这个内部类,这个类可以在官方例子中同版本的jasperreports.jar包中同名目录中找到,我们需要把它补进去,否则打印时会报一个找不到类的错。


下面我们来看页面。
页面因为各个浏览器对APPLET标签以及OBJECT标签支持程度各异,在开发中还是会碰到一些困难。
  我们这里主要针对IE以及FIREFOX做了测试和定制。
  如果在客户端环境已经安装了JRE的前提下,直接使用
<APPLET CODE = "PrinterApplet.class" CODEBASE = "applets" ARCHIVE = "print.jar,jasperreports-applet-4.6.0.jar,commons-logging-1.1.1.jar,commons-collections-2.1.1.jar" WIDTH = "300" HEIGHT = "40">
<PARAM NAME = "REPORT_URL" VALUE ="../servlets/jasperprint">
</APPLET>
这样的写法IE和FF都能认(FF试了最新版本,IE试了6,8,9),但问题是需要对没有安装JRE的客户端指名安装包。
这里又会有一个问题,就是目前JAVA的官网(ORACLE)默认安装的JRE版本是7,而很不幸的是在我的测试中发现Jasperreport 4 (测试了4.1 和4.6 两个版本)的applet打印,在JRE7的环境下,打印中英文混合字符串的时候,会出现叠行错位的BUG,而在JRE6的环境下一切OK,因此我们必须自备一个JRE6的安装包以供客户端下载,来解决JRE7的问题。
IE下的写法:
<OBJECT classid="clsid:8AD9C840-044E-11D1-B3E9-00805F499D93" WIDTH = "300" HEIGHT = "40"  codebase="applets/jre-6u35-windows-i586.exe" MAYSCRIPT>
    <PARAM NAME = CODE VALUE = "PrinterApplet.class" >
<PARAM NAME = CODEBASE VALUE = "applets" >
<PARAM NAME = ARCHIVE VALUE = "print.jar,jasperreports-applet-4.6.0.jar,commons-logging-1.1.1.jar,commons-collections-2.1.1.jar" >

    <PARAM NAME="type" VALUE="application/x-java-applet;version=1.6.0">
    <PARAM NAME="scriptable" VALUE="false">
    <PARAM NAME = "REPORT_URL" VALUE ="../servlets/jasperprint">
<comment>
<embed type="application/x-java-applet;version=1.6.0" CODE="PrinterApplet.class" JAVA_CODEBASE="applets" ARCHIVE="print.jar,jasperreports-applet-4.6.0.jar,commons-logging-1.1.1.jar,commons-collections-2.1.1.jar" scriptable=false pluginspage="applets/jre-6u35-windows-i586.exe" >

<noembed></noembed>
</embed>
</comment>
</OBJECT>

FF下的写法:
<EMBED type="application/x-java-applet;version=1.1.2" CODE = "PrinterApplet.class" CODEBASE = "applets" ARCHIVE = "print.jar,jasperreports-applet-4.6.0.jar,commons-logging-1.1.1.jar,commons-collections-2.1.1.jar" WIDTH = "300" HEIGHT = "40" REPORT_URL = "../servlets/jasperprint" scriptable=false pluginspage="applets/jre-6u35-windows-i586.exe">
<NOEMBED><XMP>
<APPLET  CODE = "PrinterApplet.class" CODEBASE = "applets" ARCHIVE = "print.jar,jasperreports-applet-4.6.0.jar,commons-logging-1.1.1.jar,commons-collections-2.1.1.jar" WIDTH = "300" HEIGHT = "40" MAYSCRIPT></XMP>
    <PARAM NAME = CODE VALUE = "PrinterApplet.class" >
<PARAM NAME = CODEBASE VALUE = "applets" >
<PARAM NAME = ARCHIVE VALUE = "print.jar,jasperreports-applet-4.6.0.jar,commons-logging-1.1.1.jar,commons-collections-2.1.1.jar" >

    <PARAM NAME="type" VALUE="application/x-java-applet;version=1.6.0">
    <PARAM NAME="scriptable" VALUE="false">
    <PARAM NAME = "REPORT_URL" VALUE ="../servlets/jasperprint">


</APPLET>
</NOEMBED>
</EMBED>

在上述标签中,applets/jre-6u35-windows-i586.exe 是我们自己JRE安装包的位置,记得把安装包放好。其他需要注意的是对应的applet的类和jar包位置的问题,本例里该JSP页面和applets 的目录同在根目录下,其他位置的话请自行调整相对路径。REPORT_URL是我们调用的servlet的地址,我们来看例子:
public class JasperPrintServlet extends HttpServlet
{


	/**
	 *
	 */
	public void service(
		HttpServletRequest request,
		HttpServletResponse response
		) throws IOException, ServletException
	{
		ServletContext context = this.getServletConfig().getServletContext();

		File reportFile = new File(context.getRealPath("/reports/WebappReport.jasper"));
		if (!reportFile.exists())
			throw new JRRuntimeException("File WebappReport.jasper not found. The report design must be compiled first.");

		Map parameters = new HashMap();
		parameters.put("ReportTitle", "Address Report");
		parameters.put("BaseDir", reportFile.getParentFile());
					
		JasperPrint jasperPrint = null;

		try
		{
			JasperReport jasperReport = (JasperReport)JRLoader.loadObject(reportFile.getPath());
			
			jasperPrint = 
				JasperFillManager.fillReport(
					jasperReport,
					parameters, 
					new WebappDataSource()
					);
		}
		catch (JRException e)
		{
			response.setContentType("text/html");
			PrintWriter out = response.getWriter();
			out.println("<html>");
			out.println("<head>");
			out.println("<title>JasperReports - Web Application Sample</title>");
			out.println("<link rel=\"stylesheet\" type=\"text/css\" href=\"../stylesheet.css\" title=\"Style\">");
			out.println("</head>");
			
			out.println("<body bgcolor=\"white\">");

			out.println("<span class=\"bnew\">JasperReports encountered this error :</span>");
			out.println("<pre>");

			e.printStackTrace(out);

			out.println("</pre>");

			out.println("</body>");
			out.println("</html>");

			return;
		}

		if (jasperPrint != null)
		{
			response.setContentType("application/octet-stream");
			ServletOutputStream ouputStream = response.getOutputStream();
			
			ObjectOutputStream oos = new ObjectOutputStream(ouputStream);
			oos.writeObject(jasperPrint);
			oos.flush();
			oos.close();

			ouputStream.flush();
			ouputStream.close();
		}
		else
		{
			response.setContentType("text/html");
			PrintWriter out = response.getWriter();
			out.println("<html>");
			out.println("<head>");
			out.println("<title>JasperReports - Web Application Sample</title>");
			out.println("<link rel=\"stylesheet\" type=\"text/css\" href=\"../stylesheet.css\" title=\"Style\">");
			out.println("</head>");
			
			out.println("<body bgcolor=\"white\">");
	
			out.println("<span class=\"bold\">Empty response.</span>");
	
			out.println("</body>");
			out.println("</html>");
		}
	}


}

可以看出servlet的例子非常简单,就是返回一个JasperPrint的被序列化对象。

到这里整个流程基本就串联起来了,然后对于页面的业务需求来说,有时候我们在页面打印动作的同时,可能还要调用一些其他的JS方法,这就涉及到在applet中调用JS,写法也是非常简单,在之前的applet里我注释了该部分:
                 JSObject win = JSObject.getWindow(this);
    JSObject doc =(JSObject)win.getMember("document");
 
                     doc.eval("alert('123')");
eval方法中可以调用页面上写好的js方法。
这里要注意的是需要在applets目录中加入一个jar包:plugin.jar,这个包也不用去别处找,就在JRE自带的lib里 (\jre\lib),贴过来即可。

还有一点就是applet里使用JS的话,在页面控件的标签里一定要加入“MAYSCRIPT”,IE的写法加在OBJECT标签里,FF写在ACTIVE的标签里(前面的例子中已加入,请参考)。

至此,一个完整的Jasperreport客户端打印的例子就完成了,一路的大坑小坑已被我们一一填平,我们可以跑一下。
大功告成.

  • 大小: 47.3 KB
分享到:
评论
2 楼 jitash 2016-03-04  
iris_1992 写道
2005年以前,国外开原报表完全碾压国产软件,但是现在国内软件,像帆软、FineBI,都比较牛掰了,设计模式和数据处理方面优于开源报表,真犯不着再用Jasperreport了,

怎么哪里都有你啊
1 楼 iris_1992 2015-04-30  
2005年以前,国外开原报表完全碾压国产软件,但是现在国内软件,像帆软、FineBI,都比较牛掰了,设计模式和数据处理方面优于开源报表,真犯不着再用Jasperreport了,

相关推荐

    Java-美妆神域_3rm1m18i_221-wx.zip

    Java-美妆神域_3rm1m18i_221-wx.zip

    51单片机的温度监测与控制(温控风扇)

    51单片机的温度监测与控制(温控风扇)

    电赛案例,C++简单的智能家居系统,其中包含了温度监测、光照控制和报警系

    电赛案例,C++简单的智能家居系统,其中包含了温度监测、光照控制和报警系统。该系统可以: 监控室内温度:当温度超过设定阈值时,触发警报。 自动调节光照:根据光线传感器的值自动调节LED灯的亮度。 入侵检测:通过红外传感器检测入侵,并触发警报。

    圣诞树 html版 可修改祝福语

    圣诞树 html版 可修改祝福语。 记事本或vscode编辑html文件:ctrl+F寻找”myLabels“关键词,定位到该处即可修改祝福语

    基于python编写的selenium自动化测试框架,采用PO模式,页面元素采用yaml进行管理资料齐全+详细文档+高分项目+源码.zip

    【资源说明】 基于python编写的selenium自动化测试框架,采用PO模式,页面元素采用yaml进行管理资料齐全+详细文档+高分项目+源码.zip 【备注】 1、该项目是个人高分项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(人工智能、通信工程、自动化、电子信息、物联网等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    屏幕截图 2024-12-21 170434.png

    屏幕截图 2024-12-21 170434

    基于SpringBoot的学生信息管理系统源码

    基于SpringBoot的学生信息管理系统(前后端源码+数据库+文档+运行截图) 学生信息管理 班级信息管理 教师信息管理 课程信息管理 选课信息管理 考勤信息管理 请假信息管理 成绩信息管理 基于SpringBoot的学生信息管理系统(前后端源码+数据库+文档+运行截图) 学生信息管理 班级信息管理 教师信息管理 课程信息管理 选课信息管理 考勤信息管理 请假信息管理 成绩信息管理基于SpringBoot的学生信息管理系统(前后端源码+数据库+文档+运行截图) 学生信息管理 班级信息管理 教师信息管理 课程信息管理 选课信息管理 考勤信息管理 请假信息管理 成绩信息管理基于SpringBoot的学生信息管理系统(前后端源码+数据库+文档+运行截图) 学生信息管理 班级信息管理 教师信息管理 课程信息管理 选课信息管理 考勤信息管理 请假信息管理 成绩信息管理基于SpringBoot的学生信息管理系统(前后端源码+数据库+文档+运行截图) 学生信息管理 班级信息管理 教师信息管理 课程信息管理 选课信息管理 考勤信息管理

    径向基函数内核 – 机器学习python案例脚本,内核在将数据转换为更高维空间方面发挥着重要作用

    径向基函数内核 – 机器学习 内核在将数据转换为更高维空间方面发挥着重要作用,使算法能够学习复杂的模式和关系。在众多的内核函数中,径向基函数(RBF)内核作为一种多功能且强大的工具脱颖而出。在本文中,我们深入探讨了RBF内核的复杂性,探讨了它的数学公式、直观理解、实际应用及其在各种机器学习算法中的重要性。

    工具变量-中国省级数字经济发展水平面板数据(2012-2022).xlsx

    详细介绍及样例数据:https://blog.csdn.net/samLi0620/article/details/144636765

    51单片机控制的智能小车.7z

    51单片机控制的智能小车.7z

    基于卷积神经网络的数字手势识别安卓APP,识别数字手势0-10详细文档+全部资料+优秀项目+源码.zip

    【资源说明】 基于卷积神经网络的数字手势识别安卓APP,识别数字手势0-10详细文档+全部资料+优秀项目+源码.zip 【备注】 1、该项目是个人高分项目源码,已获导师指导认可通过,答辩评审分达到95分 2、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 3、本项目适合计算机相关专业(人工智能、通信工程、自动化、电子信息、物联网等)的在校学生、老师或者企业员工下载使用,也可作为毕业设计、课程设计、作业、项目初期立项演示等,当然也适合小白学习进阶。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 欢迎下载,沟通交流,互相学习,共同进步!

    pymssql-2.1.4.dev5-cp37-cp37m-win-amd64.whl pymssql-2.1.4.dev5-cp37-cp37m-win32.whl

    python 使用sqlserver必须要这个问题,没办法,只能满世界的找地方下载,终于让我下载到了,现在分享给大家使用

    四川采矿场生产安全事故管理制度.docx

    四川采矿场生产安全事故管理制度

    简约灰粉共存版_8.0.53.apk

    简约灰粉共存版_8.0.53.apk

    ECharts散点图-全国主要城市空气质量(百度地图).rar

    ECharts散点图-全国主要城市空气质量(百度地图)

    四川采矿场安全检查管理规定.docx

    四川采矿场安全检查管理规定

    JSP基于WEB网上论坛设计与实现(源代码+论文+开题报告+答辩PPT+外文翻译)(2024kt).7z

    1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于计算机科学与技术等相关专业,更为适合;

    空中俯视物体检测9-YOLOv5数据集合集.rar

    空中俯视物体检测9-YOLOv5数据集合集.rar使用YOLO算法从图像中检测对象-V2 2023-05-11 2:51 PM ============================= *与您的团队在计算机视觉项目上合作 *收集和组织图像 *了解和搜索非结构化图像数据 *注释,创建数据集 *导出,训练和部署计算机视觉模型 *使用主动学习随着时间的推移改善数据集 对于最先进的计算机视觉培训笔记本,您可以与此数据集一起使用 该数据集包括1015张图像。 以YOLO V5 PYTORCH格式注释检测对象 - 图像。 将以下预处理应用于每个图像: *像素数据的自动取向(带有Exif-Arientation剥离) *调整大小为640x640(拉伸) 没有应用图像增强技术。

    会使用到的js文件词云图

    词云图

    Python&OpenCV手势识别系统(完整源码&自定义UI操作界面&视频教程)

    Python高分毕设——Python&Opencv手势识别系统(完整源码&自定义UI操作界面&视频教程) Python高分毕设——Python&Opencv手势识别系统(完整源码&自定义UI操作界面&视频教程) 使用了OpenCV的视频采集, 图像色域转换, 颜色通道分割, 高斯滤波, OSTU自动阈值, 凸点检测, 边缘检测, 余弦定理计算手势等功能. 准备工作 安装 Python-OpenCV 库 pip install opencv-python -i https://mirrors.ustc.edu.cn/pypi/web/simple 利用 -i 为pip指令镜像源, 这里使用电子科技大学的源, 速度比官方源更快. 安装 Numpy 科学计算库 pip install numpy -i https://mirrors.ustc.edu.cn/pypi/web/simple 安装 PyAutogui 库 pip install pyautogui -i https://mirrors.ustc.edu.cn/pypi/web/simple 代码实现 import nu

Global site tag (gtag.js) - Google Analytics