文章来源:http://blog.sina.com.cn/s/blog_46552dd90100g8lw.html
最近一直在研究jasperreport,今天终于搞定了动态报表问题。
所谓动态报表,顾名思义就是指报表的列头、列数和列内容都不是固定的,具体的数据是程序动态生成的。我们知道jasperreport的基本用法是用iReport画出模板jrxml然后编译成jasper,然后在程序里充填数据生成PDF或者其他格式的报表。但如果报表的列数不确定,就无法事先用iReport画好固定的模板。这时候就必须用到jasperdesign在程序中动态生成报表模板。即用jasperdesign生成模板并编译成JasperReport文件,然后充填数据生成jasperprint以供输出。
网上有一些的报表是完全用jasperdesign写成的,但这样的报表往往元素比较简单,不容易添加图片或者特殊效果例如斑马纹。比较好的方法是用iReport画出报表中不变的元素,其他需要变化的元素中画出一个标准的以供clone,然后重新设置位置和大小以及expression_r就可以了,工作量比完全重新生成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_r expression_r = new JRDesignexpression_r();
expression_r.setValueClass(java.lang.String.class);
expression_r.setText(para);
textField.setexpression_r(expression_r);
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_r expression_r1 = new JRDesignexpression_r();
expression_r1.setValueClass(java.lang.String.class);
expression_r1.setText(field);
textField1.setexpression_r(expression_r1);
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行显示底纹,可根据需要调整。