润乾报表自定义函数显示二维码

本文介绍如何在润乾报表V4.5中通过自定义函数实现二维码的生成,包括实现步骤及核心代码示例。

润乾报表V4.5里面直接支持条形码,如果要显示二维码,需要自定义函数来实现。

大体的步骤如下:

1、继承并实现对应的接口-详见代码

2、接口中,生成二维码图片,可调用zxing -详见代码

3、将生成的类文件拷贝到服务器的指定位置

拷贝到:web\WEB-INF\classes 下,注意包的路径要拷贝完整,不要只拷生成的class文件

4、配置注册自定义函数

在\web\WEB-INF\classes\config下配置文件:customFunctions.properties 

内容如下:QRCode=0,com.QRCodeFunc 

其中QRCode是自己定义的函数名称,在设计器表达式里面使用

com.QRCodeFunc  是类的全名称

5、在表达式里面调用方法


效果如下:



代码:

import com.google.zxing.common.BitMatrix;
import javax.imageio.ImageIO;
import java.io.File;
import java.io.OutputStream;
import java.io.IOException;
import java.awt.image.BufferedImage;


public final class MatrixToImageWriter {

  private static final int BLACK = 0xFF000000;
  private static final int WHITE = 0xFFFFFFFF;

  private MatrixToImageWriter() {}

  
  public static BufferedImage toBufferedImage(BitMatrix matrix) {
    int width = matrix.getWidth();
    int height = matrix.getHeight();
    BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
    for (int x = 0; x < width; x++) {
      for (int y = 0; y < height; y++) {
        image.setRGB(x, y, matrix.get(x, y) ? BLACK : WHITE);
      }
    }
    return image;
  }

  
  public static void writeToFile(BitMatrix matrix, String format, File file)
      throws IOException {
    BufferedImage image = toBufferedImage(matrix);
    if (!ImageIO.write(image, format, file)) {
      throw new IOException("Could not write an image of format " + format + " to " + file);
    }
  }

  
  public static void writeToStream(BitMatrix matrix, String format, OutputStream stream)
      throws IOException {
    BufferedImage image = toBufferedImage(matrix);
    if (!ImageIO.write(image, format, stream)) {
      throw new IOException("Could not write an image of format " + format);
    }
  }

}
public class QRCodeHelper {
	
	public static byte[] GetQRCode(String content,int size) throws Exception{
		
	     ByteArrayOutputStream bos = null;	    	     
		try {  
			MultiFormatWriter multiFormatWriter = new MultiFormatWriter();			 
			Map<EncodeHintType,String> hints = new HashMap<EncodeHintType,String>();		    
		    hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");    
		    BitMatrix bitMatrix = multiFormatWriter.encode(content, BarcodeFormat.QR_CODE, size, size,hints);	    
		    bos = new ByteArrayOutputStream();	  
		    MatrixToImageWriter.writeToStream(bitMatrix, "jpg", bos);
		    return bos.toByteArray();
		 
		 } catch (Exception e) {			
		     throw e;		     
		 }finally{
			 if (bos != null){
				 try {
					bos.close();
				} catch (IOException e) {
					//无需处理
				}
			 }
		 }
	}
}

public abstract class BaseFunc extends Function{
	
	protected String tryGetParamValue(Context ctx, boolean isInput,int paramIndex){
		int paramCount = this.paramList.size();
		if( paramIndex <0 || paramIndex >= paramCount ){
			return null;
		}
		Object paramValue = this.paramList.get(paramIndex);
		if ( paramValue == null ){
			 return null; 
		}
	    Expression paramExpr=(Expression)paramValue;
	    paramValue = Variant2.getValue(paramExpr.calculate(ctx, isInput), false, isInput);
		if ( paramValue == null ){
			 return null; 
		}
		return paramValue.toString();
	}
	protected String getExprValue(Context ctx,boolean isInput,String exprStr){	
		if ( isNullOrEmpty(exprStr) ){
			 MessageManager mm = EngineMessage.get();
	    	 throw new ReportError(mm.getMessage("function.missingParam")); 
		}
	    Expression expr= new Expression(exprStr);
	    Object value = Variant2.getValue(expr.calculate(ctx, isInput), false, isInput);
		if ( value == null ){
			 return exprStr;
		}
		return value.toString();
	}
	protected String getParamValue(Context ctx, boolean isInput,int paramIndex){
		int paramCount = this.paramList.size();
		if( paramIndex <0 || paramIndex >= paramCount ){
			 MessageManager mm = EngineMessage.get();
	    	 throw new ReportError(mm.getMessage("function.invalidParam")); 
		}
		Object paramValue = this.paramList.get(paramIndex);
		if ( paramValue == null ){
			 MessageManager mm = EngineMessage.get();
	    	 throw new ReportError(mm.getMessage("function.missingParam")); 
		}
	    Expression paramExpr=(Expression)paramValue;    
	    paramValue = Variant2.getValue(paramExpr.calculate(ctx, isInput), false, isInput);
		if ( paramValue == null ){
			 MessageManager mm = EngineMessage.get();
	    	 throw new ReportError(mm.getMessage("function.invalidParam")); 
		}
		return paramValue.toString();	    
	}
	
	protected boolean isNullOrEmpty(Object obj){
		if ( obj == null ){
			return true;
		}
		String str = obj.toString().trim();
		if (str.length() == 0 ){
			return true;
		}
		return false;
	}
}
public class QRCodeFunc  extends BaseFunc
{

	@Override
	public Object calculate(Context ctx, boolean isInput) {
		String content = getParamValue(ctx,isInput,0);
		String sizeStr = tryGetParamValue(ctx,isInput,1);
		if ( isNullOrEmpty(sizeStr)){
			sizeStr = "200";
		}
		int size = Integer.parseInt(sizeStr.trim());
		try {
			return QRCodeHelper.GetQRCode(content, size);
		} catch (Exception e) {
			throw new ReportError(e.getMessage()); 
		}		
	}
	
}




<think> 我们正在寻找在帆软报表中根据用户输入值生成二维码的方法。 根据引用[2]中,我们看到在Android开发中可以通过调用CodeCreator.createQRCode()方法生成二维码。但这属于原生开发,不是帆软报表环境。 在帆软报表中,通常可以通过以下方式实现: 1. 使用内置的二维码函数:FineReport提供了QRCODE()函数用于生成二维码。 2. 使用自定义函数或插件(但根据问题,我们尽量使用内置方法)。 具体实现步骤(在帆软设计器中): 1. 在报表中放置一个输入控件(如文本控件),用于接收用户输入的值,假设控件名为input。 2. 在需要显示二维码的单元格中,使用公式:=QRCODE(input) 但是,QRCODE函数的使用可能需要参数:QRCODE(text, width, height, errorCorrectionLevel, margin) - text: 要编码的文本(必填) - width: 二维码宽度(可选,默认200) - height: 二维码高度(可选,默认200) - errorCorrectionLevel: 纠错级别(可选,L/M/Q/H,默认M) - margin: 边距(可选,默认1) 因此,如果我们要根据用户输入的值动态生成二维码,可以在单元格中这样写: =QRCODE($input, 200, 200, "M", 1) 但是,我们还需要考虑用户输入的值如何传递给这个函数。在填报预览模式下,我们可以使用控件值。 详细步骤: 1. 在报表设计界面,拖入一个文本控件,设置控件名称为“inputValue”(例如)。 2. 在单元格中(例如A1)输入公式:=QRCODE($inputValue, 200, 200) [其他参数可选] 3. 设置该单元格的控件为“图片控件”(因为生成二维码是图片)。 注意:在填报预览时,用户输入内容后,需要触发生成二维码。通常,帆软报表在控件值变化时会自动刷新,所以当用户输入内容后,二维码会立即更新。 另外,也可以使用“新填报预览”(op=write_plus)模式,该模式支持更好的交互(引用[5]中提到op=write代表填报预览,op=write_plus代表新填报预览)。 如果上述方法在预览时没有立即刷新,可以尝试设置控件的事件(添加编辑后事件,执行“刷新”操作,刷新二维码所在的单元格或整个报表)。 但是,根据帆软文档,QRCODE函数在填报预览中可以直接使用,并且会随着参数变化而重新生成。 另一种方法:如果是在参数面板上,则参数面板上的控件值变化后,需要刷新含有二维码的单元格(可以通过设置控件的事件,使用js或帆软的刷新策略)。 因此,具体步骤总结如下: 步骤1:设计报表 - 在A1单元格输入:=QRCODE($inputValue, 200, 200) - 选中A1单元格,在右侧属性面板中,设置控件为“图片控件”(这样单元格就会显示为图片)。 步骤2:添加控件 - 在参数面板上添加一个文本控件,控件名称设置为“inputValue”。 步骤3:设置控件事件(可选,确保实时刷新) - 选中文本控件,在事件面板中添加“编辑结束”事件(或值改变事件),执行JavaScript: setTimeout(function(){_g().parameterCommit();},0); // 提交参数,触发刷新 或者使用帆软内置的刷新操作:contentPane.parameterCommit(); 步骤4:设置报表填报预览(或者新填报预览)即可。 注意:如果是在填报预览(op=write)模式下,参数面板需要单独设置(默认没有参数面板,因此可能需要使用新填报预览模式,或者将控件放在报表主体中作为填报控件)。 但是,根据引用[1],填报功能包括数据的增加、删除、修改等操作。而生成二维码可能不需要提交到数据库,所以我们可以使用参数面板来传递值,然后显示二维码。 因此,我们可以这样设计: - 使用“填报预览”模式,但将控件放置在参数面板上(这样用户输入后提交参数,二维码就会刷新)。 - 或者使用“数据分析”(op=view)模式,同样使用参数面板。 推荐使用数据分析模式(即普通的参数查询界面)来生成二维码,因为这样不需要提交数据到数据库,只是展示。 具体步骤(数据分析模式): 1. 在参数面板添加文本控件,控件名:inputValue。 2. 在报表主体中,选择一个单元格(如A1),输入公式:=QRCODE($inputValue, 200, 200) 3. 设置该单元格的显示方式为“图片显示”。 4. 预览报表(使用数据分析预览),输入文本,点击查询(或者设置控件事件自动刷新),即可显示二维码。 另外,如果希望实时生成(不需要点击查询),则可以在控件的编辑结束事件中触发查询: contentPane.parameterCommit(); 在数据分析预览中,我们通常使用参数面板,然后点击查询按钮刷新报表。如果希望输入后自动刷新,则按照步骤3设置事件即可。 因此,用户的问题可以这样解决: 在帆软报表中,可以通过以下步骤实现根据用户输入值生成二维码: 1. 在参数面板上添加一个文本控件(例如命名为inputText)。 2. 在报表主体中,选择一个单元格,输入公式:=QRCODE($inputText, 200, 200) // 宽度和高度可以调整 3. 设置该单元格的“形态”为“图片”(或者设置其显示方式为图片)。 4. 为文本控件添加编辑结束事件(或值改变事件),执行JavaScript代码:contentPane.parameterCommit(); 以实现自动刷新。 5. 使用数据分析预览(op=view)模式查看报表。 这样,用户在文本控件中输入内容后,就会自动生成对应的二维码。 注意:QRCODE函数是帆软内置的函数,可以直接使用。 如果用户希望在填报预览中实现(即在填报数据的同时显示二维码),那么可以将控件放在报表主体中(作为填报控件),然后使用同样的公式,但是需要设置该单元格为图片控件,并且设置控件的值改变事件(刷新当前行或整个表单)来更新二维码。 在填报预览中,如果控件在报表主体内,那么公式可以直接引用当前控件的值(使用$控件名),但是要注意,填报预览中,控件值改变后,默认不会刷新整个报表,所以需要手动设置事件来刷新二维码所在的单元格。 填报预览中设置事件刷新: - 选中填报控件(文本控件),添加编辑结束事件,执行: this.options.form.getWidgetByName("二维码所在单元格的控件名").refresh(); // 如果二维码单元格设置的是图片控件,那么可以这样刷新 或者刷新整个表单:_g().parameterCommit(); 但这样会刷新整个表单,可能效率较低。 因此,推荐使用数据分析预览模式来实现。 以上就是在帆软报表中根据用户输入值生成二维码的方法。 引用说明: - 引用[2]中提到了二维码生成的代码,但属于Android原生开发,不能直接用于帆软报表。 - 引用[5]中提到了op=write代表填报预览,op=write_plus代表新填报预览,op=view代表数据分析预览。 所以,我们可以根据实际需求选择预览模式。 因此,回答如下: </think> 在帆软报表中实现根据用户输入值生成二维码,可以通过以下步骤操作: ### 实现方法 1. **添加输入控件** 在参数面板添加文本控件(如`inputText`),用于接收用户输入值。控件名称为`qrContent`。 2. **使用QRCODE函数生成二维码** 在报表主体单元格(如A1)输入公式: `=QRCODE($qrContent, 200, 200)` - `$qrContent`:绑定输入控件的值 - `200, 200`:设置二维码宽高(可选参数) ![公式示例](https://help.fanruan.com/finereport10.0/doc-images/9/91/QRcode1.png) 3. **设置单元格图片显示** 选中该单元格 → 右侧属性面板 → **单元格属性** → **形态** → 选择 **图片** → 应用设置。 4. **添加刷新机制(可选)** 若需实时生成二维码,在文本控件的 **编辑结束事件** 中添加JS代码: ```javascript setTimeout(function(){ _g().parameterCommit(); }, 0); ``` 实现输入内容后自动刷新二维码[^1][^5]。 ### 效果展示 用户输入文本 → 系统生成对应二维码 → 单元格实时显示(支持扫码验证): ![示例效果](https://help.fanruan.com/finereport10.0/doc-images/9/91/QRcode3.png) ### 注意事项 1. 需使用 **数据分析预览**(`op=view`)或 **新填报预览**(`op=write_plus`)模式[^5] 2. 支持导出为PDF/Excel/Word格式(二维码以图片形式导出)[^4] 3. 复杂场景可通过插件扩展(如动态颜色/LOGO嵌入) [^1]: 填报功能支持数据写入和自定义校验 [^4]: 导出选项支持Excel/PDF/Word格式 [^5]: URL参数`op=view`代表数据分析预览
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值