java web ireport 3.7 打印二维码

2024-06-20 15:08

本文主要是介绍java web ireport 3.7 打印二维码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

       前几天做项目的时候遇到一个需求,客户想在医院的诊间结算单上打印出二维码,供病人去扫描。由于现有的版本项目报表显示和打印都是用ireport 3.7来做的,还没转化成lodop打印(新版本是lodop 在那上面打印二维码轻而易举)。 原本以前打印过条形码,想着应该差不多。于是就着手做了,但是做的时候发现,ireport 的组件面板上根本没有 二维码的样式,只有条形码。

     这下懵逼了,总不能因为这样一个小需求还要把打印格式全部转化为lodop,而且还得去每个诊间医生的电脑上一个个去安装lodop吧。那只能在原有的方式上想办法了。

     看到组件模板上有image ,突然有个想法。能不能在后台生成好图片,把这个图片给给到ireport呢?由于用的是java,和报表之间是通过jasperReports 去实现,于是从网上搜索了一堆发现都是在ireport 中引入javase.jar 和core.jar  包 然后再

新建一个【Image】属性的文本,选择图片路径时选择“取消”,这时候在“image Expression”框中添加“com.google.zxing.client.j2se.MatrixToImageWriter.toBufferedImage(new com.google.zxing.qrcode.QRCodeWriter().encode($P{QrCode},com.google.zxing.BarcodeFormat.QR_CODE,1000,1000))”字段。 
其中:$P{QrCode}是二维码的内容,这个参数可以根据自己的实际情况来定 

 

       但是其实这种方法是有缺陷的!!!比如我现在遇到的情况:客户又要求在二维码中间加入自己医院的logo (没办法,需求一直变是经常有的事情。。。。在这里就不吐槽了)。本着客户是上帝的原则 咱必须得咬咬牙满足到底啊。

    下面,正式进入解决方法:

  1.   在ireport的Parameters里面新增一个参数,这里我们取名:QRCODEURL。(这一步很重要!!)·

 2.image 拖入要放二维码的地方

3.设置 image的 expression 属性,选中新加的QRCODEURL (这个一定要注意,不能在Image Expression里面直接设置$P{QRCODEURL},否则会报net.sf.jasperreports.engine.design.JRValidationException: Report design not valid : 
     1. Parameter not found QRCODEURL的错误
) ,并且在 is Lazy 打上勾(懒汉模式加载)

 

4.开始写后台。

如果是一张固定的二维码图片,建议直接引入生成好的二维码图片的路径就可以了,我这边是根目录下。

5.如果是动态的二维码,可以有两种方式去实现。

     5.1  先生成一张图片 保存到服务器上。然后传给前台地址去调用。 

             关于如何生成图片的代码我贴在下面。需要core-2.2.jar,javase-2.2.jar 这两个jar 包

package com.bsoft.bsphis.msxzz;import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;import javax.imageio.ImageIO;
import org.apache.commons.codec.binary.Base64;import com.bsoft.bsphis.mobilePay.util.SecurityUtil;
import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.MultiFormatWriter;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;/**  * @Description: (二维码)     * @author:luoguohui  * @date:2015-10-29 下午05:27:13     */
public class ZXingCode
{private static final int QRCOLOR = 0xFF000000;   //默认是黑色private static final int BGWHITE = 0xFFFFFFFF;   //背景颜色public static void main(String[] args) throws WriterException  {  try  {  getLogoQRCode("https://puser.zjzwfw.gov.cn/sso/mobile.do?action=oauth&scope=1&servicecode=hzczaxqp&goto=330105004", "浙江政务服务");}  catch (Exception e)  {  e.printStackTrace();  }  }  /*** 生成带logo的二维码图片** @param qrPic* @param logoPic*/public static String getLogoQRCode(String qrUrl,String productName){
//      String filePath = (javax.servlet.http.HttpServletRequest)request.getSession().getServletContext().getRealPath("/") + "resources/images/logoImages/llhlogo.png";//filePath是二维码logo的路径,但是实际中我们是放在项目的某个路径下面的,所以路径用上面的,把下面的注释就好String filePath = "C:/Users/Administrator/Desktop/tupian/touxiang.png";  //TODO  String content = qrUrl;try{  ZXingCode zp = new ZXingCode();BufferedImage bim = zp.getQR_CODEBufferedImage(content, BarcodeFormat.QR_CODE, 120, 120, zp.getDecodeHintType());return zp.addLogo_QRCode(bim, new File(filePath), new LogoConfig(), productName);}catch (Exception e){e.printStackTrace();}return null;}/*** 给二维码图片添加Logo** @param qrPic* @param logoPic*/public String addLogo_QRCode(BufferedImage bim, File logoPic, LogoConfig logoConfig, String productName){try{/*** 读取二维码图片,并构建绘图对象*/BufferedImage image = bim;Graphics2D g = image.createGraphics();/*** 读取Logo图片*/BufferedImage logo = ImageIO.read(logoPic);/*** 设置logo的大小,本人设置为二维码图片的20%,因为过大会盖掉二维码*/int widthLogo = logo.getWidth(null)>image.getWidth()*3/10?(image.getWidth()*3/10):logo.getWidth(null), heightLogo = logo.getHeight(null)>image.getHeight()*3/10?(image.getHeight()*3/10):logo.getWidth(null);/*** logo放在中心*/int x = (image.getWidth() - widthLogo) / 2;int y = (image.getHeight() - heightLogo) / 2;/*** logo放在右下角*  int x = (image.getWidth() - widthLogo);*  int y = (image.getHeight() - heightLogo);*///开始绘制图片g.drawImage(logo, x, y, widthLogo, heightLogo, null);
//            g.drawRoundRect(x, y, widthLogo, heightLogo, 15, 15);
//            g.setStroke(new BasicStroke(logoConfig.getBorder()));
//            g.setColor(logoConfig.getBorderColor());
//            g.drawRect(x, y, widthLogo, heightLogo);g.dispose();//把商品名称添加上去,商品名称不要太长哦,这里最多支持两行。太长就会自动截取啦if (productName != null && !productName.equals("")) {//新的图片,把带logo的二维码下面加上文字BufferedImage outImage = new BufferedImage(150, 150, BufferedImage.TYPE_4BYTE_ABGR);Graphics2D outg = outImage.createGraphics();//画二维码到新的面板outg.drawImage(image, 0, 0, image.getWidth(), image.getHeight(), null);//画文字到新的面板outg.setColor(Color.BLACK); outg.setFont(new Font("宋体",Font.BOLD,20)); //字体、字型、字号 int strWidth = outg.getFontMetrics().stringWidth(productName);if (strWidth > 399) {
//                  //长度过长就截取前面部分
//                  outg.drawString(productName, 0, image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 5 ); //画文字//长度过长就换行String productName1 = productName.substring(0, productName.length()/2);String productName2 = productName.substring(productName.length()/2, productName.length());int strWidth1 = outg.getFontMetrics().stringWidth(productName1);int strWidth2 = outg.getFontMetrics().stringWidth(productName2);outg.drawString(productName1, 70- strWidth1/2, image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 12 );BufferedImage outImage2 = new BufferedImage(400, 485, BufferedImage.TYPE_4BYTE_ABGR);Graphics2D outg2 = outImage2.createGraphics();outg2.drawImage(outImage, 0, 0, outImage.getWidth(), outImage.getHeight(), null);outg2.setColor(Color.BLACK); outg2.setFont(new Font("宋体",Font.BOLD,20)); //字体、字型、字号 outg2.drawString(productName2,  70-strWidth2/2, outImage.getHeight() + (outImage2.getHeight() - outImage.getHeight())/2 + 5 );outg2.dispose(); outImage2.flush();outImage = outImage2;}else {outg.drawString(productName, 70-strWidth/2 , image.getHeight() + (outImage.getHeight() - image.getHeight())/2 + 12 ); //画文字 }outg.dispose(); outImage.flush();image = outImage;}logo.flush();image.flush();ByteArrayOutputStream baos = new ByteArrayOutputStream();baos.flush();ImageIO.write(image, "png", baos);//二维码生成的路径,但是实际项目中,我们是把这生成的二维码显示到界面上的,因此下面的折行代码可以注释掉//可以看到这个方法最终返回的是这个二维码的imageBase64字符串//前端用 <img src="https://img-blog.csdnimg.cn/2022010709290228980.png"/>  其中${imageBase64QRCode}对应二维码的imageBase64字符串ImageIO.write(image, "png", new File("F:/homeCode.jpg")); //TODO  String imageBase64QRCode = SecurityUtil.encryptBASE64(baos.toByteArray());
//            		Base64.encodeBase64URLSafeString(baos.toByteArray());baos.close();return imageBase64QRCode;}catch (Exception e){e.printStackTrace();}return null;}/*** 构建初始化二维码** @param bm* @return*/public BufferedImage fileToBufferedImage(BitMatrix bm){BufferedImage image = null;try{int w = bm.getWidth(), h = bm.getHeight();image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);for (int x = 0; x < w; x++){for (int y = 0; y < h; y++){image.setRGB(x, y, bm.get(x, y) ? 0xFF000000 : 0xFFCCDDEE);}}}catch (Exception e){e.printStackTrace();}return image;}/*** 生成二维码bufferedImage图片** @param content*            编码内容* @param barcodeFormat*            编码类型* @param width*            图片宽度* @param height*            图片高度* @param hints*            设置参数* @return*/public BufferedImage getQR_CODEBufferedImage(String content, BarcodeFormat barcodeFormat, int width, int height, Map<EncodeHintType, ?> hints){MultiFormatWriter multiFormatWriter = null;BitMatrix bm = null;BufferedImage image = null;try{multiFormatWriter = new MultiFormatWriter();// 参数顺序分别为:编码内容,编码类型,生成图片宽度,生成图片高度,设置参数bm = multiFormatWriter.encode(content, barcodeFormat, width, height, hints);int w = bm.getWidth();int h = bm.getHeight();image = new BufferedImage(w, h, BufferedImage.TYPE_INT_RGB);// 开始利用二维码数据创建Bitmap图片,分别设为黑(0xFFFFFFFF)白(0xFF000000)两色for (int x = 0; x < w; x++){for (int y = 0; y < h; y++){image.setRGB(x, y, bm.get(x, y) ? QRCOLOR : BGWHITE);}}}catch (WriterException e){e.printStackTrace();}return image;}/*** 设置二维码的格式参数** @return*/public Map<EncodeHintType, Object> getDecodeHintType(){// 用于设置QR二维码参数Map<EncodeHintType, Object> hints = new HashMap<EncodeHintType, Object>();// 设置QR二维码的纠错级别(H为最高级别)具体级别信息hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.H);// 设置编码方式hints.put(EncodeHintType.CHARACTER_SET, "utf-8");hints.put(EncodeHintType.MARGIN, 0);hints.put(EncodeHintType.MAX_SIZE, 350);hints.put(EncodeHintType.MIN_SIZE, 100);return hints;}
}class LogoConfig{// logo默认边框颜色public static final Color DEFAULT_BORDERCOLOR = Color.WHITE;// logo默认边框宽度public static final int DEFAULT_BORDER = 2;// logo大小默认为照片的1/5public static final int DEFAULT_LOGOPART = 5;private final int border = DEFAULT_BORDER;private final Color borderColor;private final int logoPart;/*** Creates a default config with on color {@link #BLACK} and off color* {@link #WHITE}, generating normal black-on-white barcodes.*/public LogoConfig(){this(DEFAULT_BORDERCOLOR, DEFAULT_LOGOPART);}public LogoConfig(Color borderColor, int logoPart){this.borderColor = borderColor;this.logoPart = logoPart;}public Color getBorderColor(){return borderColor;}public int getBorder(){return border;}public int getLogoPart(){return logoPart;}}

其中base64的方法我这边就补贴出来了。自己百度搜,一搜一大把。

这种方式有一个弊端:当图片量很大的时候,就整个项目会变的很大。如果这些照片不需要存储,建议用第二种方案。

方案二:

把图片转成base64输出到QRCODEURL 中去就可以了。至于怎么转,看上面传的addLogo_QRCode 返回的String类型就行啦。

 

response.put("QRCODEURL", imageBase64QRCode);

 

总结:

   代码在上面,注释应该已经写的很详细了。遇到问题,百度后一定要自己思考才行。不能原模原样照搬照抄

这篇关于java web ireport 3.7 打印二维码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1078479

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定