`

iReport学习笔记——动态报表

阅读更多
iReport学习笔记——动态报表

最近一直在研究jasperreport,今天终于搞定了动态报表问题。

所谓动态报表,顾名思义就是指报表的列头、列数和列内容都不是固定的,具体的数据是程序动态生成的。我们知道jasperreport的基本用法是用iReport画出模板jrxml然后编译成jasper,然后在程序里充填数据生成PDF或者其他格式的报表。但如果报表的列数不确定,就无法事先用 iReport画好固定的模板。这时候就必须用到jasperdesign在程序中动态生成报表模板。即用jasperdesign生成模板并编译成 JasperReport文件,然后充填数据生成jasperprint以供输出。

网上有一些的报表是完全用jasperdesign写成的,但这样的报表往往元素比较简单,不容易添加图片或者特殊效果例如斑马纹。比较好的方法是用iReport画出报表中不变的元素,其他需要变化的元素中画出一个标准的以供clone,然后重新设置位置和大小以及expression就可以了,工作量比完全重新生成jasperdesign模板小得多,以下是一个例子:

File file = new File(fileDir);
   JasperDesign jasperDesign = new JasperDesign();
   try {
    jasperDesign = JRXmlLoader.load(file);
    int restWidth = jasperDesign.getColumnWidth()-89;
    int columnNum = strArray.length-3;
    int columnWidth = restWidth/columnNum;
    JRDesignBand columnHeader = (JRDesignBand)jasperDesign.getColumnHeader();
    JRDesignBand detail = (JRDesignBand)jasperDesign.getDetail();
    for (int i=3;i<strArray.length;i++){
     //生成列头
     String para = "$P{para"+i+"}";
     JRDesignTextField textField = (JRDesignTextField)(((JRDesignBand)jasperDesign.getColumnHeader()).getElementByKey("textField-20")).clone();
     textField.setX(89+columnWidth*(i-3));
     textField.setY(89);
     textField.setWidth(columnWidth);
     textField.setHeight(41);
     JRDesignExpression expression = new JRDesignExpression();
     expression.setValueClass(java.lang.String.class);
     expression.setText(para);
     textField.setExpression(expression);
     columnHeader.addElement(textField);
     //生成矩形
     JRDesignRectangle rectangle = (JRDesignRectangle)(((JRDesignBand)jasperDesign.getDetail()).getElementByKey("rectangle-24")).clone();
     rectangle.setX(89+columnWidth*(i-3));
     rectangle.setY(0);
     rectangle.setWidth(columnWidth);
     rectangle.setHeight(12);
     detail.addElement(rectangle);
     //生成表内容
     String field = "$F{field"+(i+1)+"}";
     JRDesignTextField textField1 = (JRDesignTextField)(((JRDesignBand)jasperDesign.getDetail()).getElementByKey("textField-24")).clone();
     textField1.setX(89+columnWidth*(i-3));
     textField1.setY(0);
     textField1.setWidth(columnWidth);
     textField1.setHeight(12);
     JRDesignExpression expression1 = new JRDesignExpression();
     expression1.setValueClass(java.lang.String.class);
     expression1.setText(field);
     textField1.setExpression(expression1);
     detail.addElement(textField1);
    }
    //删除原有的内容
    JRDesignTextField textFieldDel = (JRDesignTextField)(((JRDesignBand)jasperDesign.getColumnHeader()).getElementByKey("textField-20"));
    JRDesignTextField textField1Del = (JRDesignTextField)(((JRDesignBand)jasperDesign.getDetail()).getElementByKey("textField-24"));
    JRDesignRectangle rectangleDel = (JRDesignRectangle)(((JRDesignBand)jasperDesign.getDetail()).getElementByKey("rectangle-24"));
    columnHeader.removeElement(textFieldDel);
    detail.removeElement(textField1Del);
    detail.removeElement(rectangleDel);
  
    return JasperCompileManager.compileReport(jasperDesign);

需要注意的是,最后要删除事先画好的那几个元素以免重叠。

PS:斑马纹效果——添加一个矩形设成所需底纹的颜色,大小和单元格一样大,顺序置于文本框后面,文本框设成透明,在矩形的表达式中加入 Boolean.valueOf((($V{COLUMN_COUNT}.intValue()-1)/5)%2!=0)即可,我这里是每隔5行显示底纹,可根据需要调整。
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics