Jasper Reports - working with beans and sub report

Jasper Reports

I recently had to research how to use JasperReports (refer. 1) to produce PDF, rich text format (RTF), and Excel reports using an ArrayList collection of Java objects as the source for the data in the report. The main Java class has as one of its attributes an ArrayList collection of another Java object. I had the challenge of figuring out how to include all the data from the nested ArrayList collection in the report. After several frustrating hours researching this issue in the very limited JasperReports documentation and on the JasperReports forums I found a hint that lead me to a solution.

Since there does not appear to be very good documentation on how to use complex collections as the data source for a JasperReport, I decided to write a blog entry that will hopefully help others in the future (and of course remind me how I did this when I have to do it again in six months).


Introduction

I recently had to research how to use JasperReports (refer. 1) to produce PDF, rich text format (RTF), and Excel reports using an ArrayList collection of Java objects as the source for the data in the report. The main Java class has as one of its attributes an ArrayList collection of another Java object. I had the challenge of figuring out how to include all the data from the nested ArrayList collection in the report. After several frustrating hours researching this issue in the very limited JasperReports documentation and on the JasperReports forums I found a hint that lead me to a solution.

Since there does not appear to be very good documentation on how to use complex collections as the data source for a JasperReport, I decided to write a blog entry that will hopefully help others in the future (and of course remind me how I did this when I have to do it again in six months).

JasperReports

If you visit the JasperReports website you can find information about what JasperReports does. Basically it's a comprehensive package of Java classes that enables you to use various sources as data used to create a variety of report types including PDF, Excel, and RTF. There is an open-source community edition of JasperReports that you can download (refer. 2). As of July 6, the version was JasperReports 3.0.0.

After you download and unzip the community edition of JasperReports you'll have a folder named JasperReports-3.0.0. In this folder are sample applications, quick start documentation, and the jar files needed to use JasperReports. The main jar you'll need to include in your Java project is jasperreports-3.0.0.jar, located in the sub-folder dist.

There is some documentation for JasperReports available at reference 3.

iReport

In conjunction with JasperReports you can use iReport to graphically design your reports (instead of writing the report XML manually [the .jrxml file]). iReport gives you the ability to specify a data source and then drag-and-drop the fields from that data source onto the report. You can download iReport at reference 4. iReport is free to download and use. There is a Windows installer version and as of July 6, the iReport version is also 3.0.0.

There is some documentation for iReport available at reference 5.

Learning How To Use JasperReports and iReport

From my research there is limited free documentation available (references 3 and 5) on the JasperReport website. There are some books you can purchase for learning how to use JasperReports and iReport at the JasperReport website. You can also Google JasperReports tutorial for a list of tutorials that may help you get started.

The general steps to creating a report are

  1. In iReport specify a source of the data that will be used to fill the report.
  2. Once iReport has access to this data source, you design the report.
  3. User iReport to compile that report design into a file of type.jasper.
  4. The .jasper file can then be used by JasperReports to be filled with data which creates a .jrprint file type.
  5. The .jrprint file type can then be used by JasperReports to create a PDF, RTF, or Excel version of the file.

An Example of Using An ArrayList As A Data Source For JasperReports

JasperReports and iReport can use data from various sources including databases and collections. For my current project, the data source is a collection of objects. To give you an example of how to use JasperReports and iReport to create a report that uses an ArrayList of objects as the data source I've prepared the following example. Example source code in Eclipse project.

Model Classes:

We have two model classes: Person and Phone. A Person has a firstName and lastName. Since a Person may have multiple phones, a Person also has an ArrayList of Phone objects.

A Phone has a phoneType (for example "work" or "mobile") and a phoneNumber.

You can download an Eclipse Java project with the source code for Person and Phone and a test class from reference 6.

Setup the Data Source in iReport

If you examine the source code in test.TestPerson class (reference 6) you'll see that I've created a static method named getBeanCollection that returns an ArrayList of Person objects. This method is used in iReport to specify the data source and then will be used to fill the report with Person objects.

To create our report in iReport, we need to give iReport access to our class files for Person and Phone so that iReport can find their fields. Then we can drag-and-drop the fields onto a report design. So I created a jar of the model package (which includes the Person and Phone classes) and placed the jar in the iReport lib folder, which on my system is located at: C:\Program Files\JasperSoft\iReport-3.0.0\lib\.

Now after restarting iReport, iReport will have access to the attributes of the Person and Phone class.

After starting up iReport we also need to specify the path to the class that has the getBeanCollection method, in this case that class is test.TestPerson and on my system the path to the test package is C:\coursecatalogtest\TestJasperReport\bin. So I put that path in iReport - Options - ClassPath - Add Folder.

To setup the data source in iReport go to Data - Connections/Data Sources - New. Select JavaBeans set data source. Then specify the following:

Name: PersonDataSource

Factory class: test.TestPerson

The static method to call to retrieve... getBeanCollection

You should be able to click on the Test button and get the message "Connection Test Successful." Click on Save to save this data source.

Create A Report in iReport

Use File - New Document and give the report a name of contacts. The click on OK

You'll now have a blank report with areas for title, pageHeader, columnHeader, detail, columnFooter, pageFooter, lastPageFooter, and summary. For our example we will just be placing static text in the title and columnHeader's area and then our Person fields in the detail area. For each Person object in the ArrayList, there will be a corresponding section in the detail area.

Accessing the Person Class Fields

To access the model class fields (in our example Person and Phone) in the report, click on Data - Report Query. The click on the JavaBean Data Source tab and enter the class name with path: model.Person. Because you created a jar file of the model package and placed it in iReport's lib folder earlier iReport can find the class and its attributes.

Click on the firstName, lastName, and phones attributes and then click on Add Selected Fields. Then click on OK.

Next click on View - Fields and you'll see these attributes. Drag the firstName and lastName fields to the detail area of the report. In the detail area you should see two boxes, one with $F{firstName} and one with $F{lastName}. The firstName and lastName fields (F) from each Person object in the collection will be placed into the detail area.

Save your report.

Accessing the Phone Class Fields

Because the phones attribute of the Person class is an ArrayList containing Phone objects, we will need to handle displaying all the Phone objects for each Person by using a sub-report. Click on Edit - Insert Element - Subreport. Use the cross-hair cursor to drag a rectangle in the detail area below where you placed the firstName and lastName fields.

With the create new report option checked click on Next. The Connection/Data Source should be PersonDataSource and the JavaBean class is model.Phone. Click next.

You should see the Phone class fields, phoneNumber and phoneType. Select both of those and click on the arrow to move them from the left box to the right box. Click Next.

Select either columnar layout or tabular layout and report format (I choose Tabular layout and classicT.xml). Click Next.

Click Finish.

You should see a subreport with the $F{phoneNumber} and $F(phoneType) fields in the detail area and the static text phoneNumber and phoneType in the columnHeader area of the subreport.

Save your reports.

Specify the Data Source for the Sub Report

Double click on the contacts.jrxml in the Files window to return to the main report. In the main report will be a sub-report icon inside the rectangle you drew earlier. Right click on the sub-report icon and select properties. Click on the Subreport tab.

In the Connection/Data Source Expression drop down box select "Use data source expression."

Click on the "open expression editor button" that is just below the drop down box. In the expression editor, delete any text in the window and type in the following: new JRBeanCollectionDataSource($F{phones}). What this means is that for the sub-report you want to use a new bean collection with the phones field of the Person class as the source of the beans.

Click on Check Expression button and you should get the message: "Expression successfully validated."

Click on Apply and then close the sub-report properties window.

Add Title and Column Headings

You can add static text to any area of the report by clicking on Edit - Insert Element - Static Text. Do so and then using the cross-hair cursor draw a rectangle text area in the title area of the report. Then double click on the rectangle and type in a title (eg "Contacts"). You can then use teh font family and size drop downs to style the text. Do the same procedure to add static text to the column headings area (eg "Name and Phones").

Test the Report

To test the report, you need to compile the main and sub-report and then execute the report.

Before compiling the report, set the location of where you want the compiled files to be placed. Click on Options and then settings and then click on the compiler tab. Click on the browse button and select a folder where you want the to save the compiled .jrxml report files (which are the .jasper files). I choose "C:\JasperReports".

Click on Build - Compile to compile the main contacts.jrxml file and then do the same after double-clicking on the subreport file. If you look inside the folder where you told iReport to place the compiled files you should see two files: contacts.jasper and contacts_subreport0.jasper. The .jasper files are the ones JasperReports can use to fill with data from the data source.

Double click on contacts.jrxml in iReport to open that file back up. The click on Build - Execute (with active connection). In the surreport_dir parameter window enter the folder where you told iReport to save the compiled files (in my example "c:\JasperReports\").

The iReport Jasper viewer should open up with all the Person objects displayed in the report. For each Person object there should be Phone objects displayed.

Summary

We created a very simple main and sub reports using an ArrayList of Person objects. The sub-report is tied to each Person object through the phones ArrayList (a collection of Phone objects) that is an attribute of each Person object.

iReport can be used to create very complex reports. Each report element can be styled with a specifc font, size, color, etc.

What's Next?


In part 1, I covered how to create a JasperReport report using iReport. That report's data source is an ArrayList of Person objects. Each Person object also has an ArrayList of Phone objects. In part 2, I explain how to use the JasperReport report in a Java application. You can download all the source code (Eclipse Java project).

Creating A Filled Report

In part 1, I created a main and sub-report and then compiled those reports into two .jasper files stored in c:\jasperreports\. JasperReports uses the .jasper files to fill with data and create a .jrprint file.

To give your Java application access to the JasperReport classes, you'll need to include these jar files:

jasperreports-3.0.0.jar (located in \jasperreports-3.0.0\dist\ folder)

commons-beanutils-1.7.jar (located in \jasperreports-3.0.0\lib\ folder)

commons-collections-2.1.jar (located in \jasperreports-3.0.0\lib\ folder)

commons-logging-1.0.2.jar (located in \jasperreports-3.0.0\lib\ folder)

itext-1.3.1.jar (located in \jasperreports-3.0.0\lib\ folder)

poi-3.0.1-FINAL-20070705.jar (located in \jasperreports-3.0.0\lib\ folder)

So step 1 is to fill the .jasper file with data. These statements do that:

/*
* Setup the parameters and their values
* needed by the .jasper file
*/
Map parameters = new HashMap();
parameters.put("SUBREPORT_DIR", "c:/JasperReports/");
JasperFillManager.fillReportToFile("C:/JasperReports/contacts.jasper", parameters,
new JRBeanCollectionDataSource(TestPerson.getBeanCollection() ) );

The first two statements create a Map containing the parameter name (the key) and the parameter's value (the value). For this JasperReport we need to send it the value of the subreport_dir (the directory where the sub-report is stored).

The we can use the JasperFillManager class's fillReportToFile method to fill the .jasper file with data. This method takes three arguments, the .jasper file (we created this using iReport), the parameters to send to that .jasper file, and the data source.

Note that our data source is using the JRBeanCollectionDataSource class, which has a constructor that takes an ArrayList of bean objects. We create that ArrayList by calling the static getBeanCollection method of our class TestPerson. This was the same data source we used in part 1 to create our report in iReport.

Converting A Filled Report to A PDF

The following code uses the filled report, which is stored in contact.jrprint, to create a PDF.

JasperExportManager.exportReportToPdfFile("C:/JasperReports/contacts.jrprint");

In folder c:\jasperreports\ should now be a PDF named contacts.pdf containing all the data returned by the TestPerson.getBeanCollection method.

Converting A Filled Report to An Rich Text Format (RTF) File

To create an RTF file (which can be opened by Word or other word-processing software) use the following statements.

File sourceFile = new File("C:/JasperReports/contacts.jrprint");
         
JasperPrint jasperPrint = (JasperPrint)JRLoader.loadObject(sourceFile);

File destFile = new File(sourceFile.getParent(), jasperPrint.getName() + ".rtf");

JRRtfExporter exporter = new JRRtfExporter();

exporter.setParameter(JRExporterParameter.JASPER_PRINT, jasperPrint);
exporter.setParameter(JRExporterParameter.OUTPUT_FILE_NAME, destFile.toString());

exporter.exportReport();

The above code creates a JasperPrint object using our .jrprint file (which is filled with data from our data source). Then using a JRRtfExporter object it exports the .jrprint file to a RTF file with the same name but extension of .rtf.

After running the above code you should have a contacts.rtf file in the c:\jasperreports\ folder.

Converting A Filled Report to An Excel File

To create an Excel file use the following statements.

destFile = new File(sourceFile.getParent(), jasperPrint.getName() + ".xls");
         
JRXlsExporter xlsExporter = new JRXlsExporter();

xlsExporter.setParameter(JRExporterParameter.JASPER_PRINT, jasperPrint);
xlsExporter.setParameter(JRExporterParameter.OUTPUT_FILE_NAME, destFile.toString());
xlsExporter.setParameter(JRXlsExporterParameter.IS_ONE_PAGE_PER_SHEET, Boolean.FALSE);

xlsExporter.exportReport();

The code above is similar to how we created the RTF file, except it uses class xlsExporter to export the .jrprint file to an Excel file type.

Summary

The hardest part of using JasperReports is creating the report. Once you have a report created, you can fill that report with data from a data source. In this example the data source is a collection of Person objects. There are other types of data sources including databases.

Once you have filled your .jasper report (which was created by compiling the .jrxml file in iReport) and created the .jrprint file, you can use JasperReport to convert that .jrprint file to various file types including PDF, RTF, and Excel.

Start writing here.

Comments

Problem with example

Hi,
thanks for your article.
i have a problem with your example. When i launch your template with ireports to define a bean datasource the preview is ok i see all phone ... But when i launch with eclipse and the main class, the file generated are empty, they are just static content text that are display. I have reproduce on two computer and two jasper configuration ( one is exactly your configuration ). Have you an idea?
Thanks


Last edited Oct 8, 2009 3:10 AM
Report abusive comment

Problem with JRBeanCollectionDataSource

Platform: linux fc9, OpenJDK 64-Bit Server VM (build 1.6.0-b09), iReport 3.6.0, Jasper 3.6.0

I have some difficulies with the tutorial.
First: I could not find the "Click on Check Expression button".

Compiling the main-Report in iReport I get the following error:

net.sf.jasperreports.engine.JRException: Errors were encountered when compiling report expressions class file: org.codehaus.groovy.control.MultipleCompilationErrorsException:
startup failed, calculator_contacts2_1253173989542_417534: 169:
unable to resolve class JRBeanCollectionDataSource

Executing the TestJasperReport.class I get this errror:

Caused by: org.codehaus.groovy.runtime.typehandling.GroovyCastException: Cannot cast object '[
Phone type: home Phone number: 913-906-6000,
Phone type: work Phone number: 913-906-6001,
Phone type: mobile Phone number: (913-906-6002]' with class 'java.util.ArrayList' to class 'net.sf.jasperreports.engine.JRDataSource'

This message shows: the array does contain data but the constructor

new JRBeanCollectionDataSource($F{phones})

does not work properly.

Last edited Sep 18, 2009 8:21 AM
Report abusive comment

How to print report via client default printer ?

I develop web application. I use jdev 10.1.3.4.0 . I can print report in PDF format.

My Problem
I can't print report to client default printer. when Client run application and print. Report is printed at server default printer
How can I do.

My source code for print to PDF format.
FacesContext context = FacesContext.getCurrentInstance();
response = (HttpServletResponse)context.getExternalContext().getResponse();
String urlSchema = "jdbc:oracle:thin:@localhost:1521:ORCL";
String schemaName = "hr";
String schemaPass = "hr;
reportPath = "D:\\Project\\Reports";
Class.forName("oracle.jdbc.driver.OracleDriver");
conn = DriverManager.getConnection(urlSchema, schemaName, schemaPass);
reportPath = reportPath.endsWith("\\") ? reportPath : (reportPath + "\\");
input = new File(reportPath + reportName + ".jasper");
reportParameters.put("SUBREPORT_DIR", reportPath);
reportParameters.put("P_IMAGE_PATH", reportPath);
jasperPrint = JasperFillManager.fillReport(input.getPath(), reportParameters, conn);
response.setContentType("application/pdf");
response.addHeader("Content-Disposition", "attachment;filename=" + reportNameOutput + ".pdf");
OutputStream outputStream = response.getOutputStream();
JasperExportManager.exportReportToPdfStream(jasperPrint, outputStream);
outputStream.flush();
outputStream.close();
conn.close();


My source code for print to printer.
FacesContext context = FacesContext.getCurrentInstance();
response = (HttpServletResponse)context.getExternalContext().getResponse();
String urlSchema = "jdbc:oracle:thin:@localhost:1521:ORCL";
String schemaName = "hr";
String schemaPass = "hr;
reportPath = "D:\\Project\\Reports";
Class.forName("oracle.jdbc.driver.OracleDriver");
conn = DriverManager.getConnection(urlSchema, schemaName, schemaPass);
reportPath = reportPath.endsWith("\\") ? reportPath : (reportPath + "\\");
input = new File(reportPath + reportName + ".jasper");
reportParameters.put("SUBREPORT_DIR", reportPath);
reportParameters.put("P_IMAGE_PATH", reportPath);
jasperPrint = JasperFillManager.fillReport(input.getPath(), reportParameters, conn);
JasperPrintManager.printReport(jasperPrint, false);
conn.close();

Last edited Jun 17, 2009 3:11 AM
Report abusive comment

Run the sample codes in Web application

Hi, thanks for the guide. I have successful created a similar report based on the above guide. but i can only make it work well in a stand alone java prg, i failed to make it work in a web application (servlet). for the same jasper report, it just display the main report content, and the subreport content is not shown in the web application output. I have made sure the correct subreportdir was pass when calling the jasper report from the servlet.

can you share some idea how to have the above example work in web application.

Thanks.


Last edited Mar 25, 2009 8:53 PM
Report abusive comment

To Generate an Excel sheet

Hi, I've used iReports to create report in PDF format successfully. Now I need to generate reports in .xls format. Can you please help?
The instructions on this site refer to some line of code/instruction, I'm confused as to how and where to use those.

Any help would be appreciated.
Thanks

Last edited Mar 25, 2009 10:23 AM
Report abusive comment

Thanks! Please help with Jasper Assistant

Nasir, this has saved me hours of pain! Thanks for an excellent explanation.

I am running the sample in Jasper Assistant. I get the following error:

net.sf.jasperreports.engine.JRException: Errors were encountered when compiling report expressions class file:
1. The method JRBeanCollectionDataSource(List) is undefined for the type Unnamed_1237278237500_606491
value = (java.lang.String)(JRBeanCollectionDataSource(((java.util.List)field_subRow.getValue()))); //$JR_EXPR_ID=18$
<------------------------>
2. The method JRBeanCollectionDataSource(List) is undefined for the type Unnamed_1237278237500_606491
value = (java.lang.String)(JRBeanCollectionDataSource(((java.util.List)field_subRow.getOldValue()))); //$JR_EXPR_ID=18$
<------------------------>
3. The method JRBeanCollectionDataSource(List) is undefined for the type Unnamed_1237278237500_606491
value = (java.lang.String)(JRBeanCollectionDataSource(((java.util.List)field_subRow.getValue()))); //$JR_EXPR_ID=18$
<------------------------>

Last edited Mar 27, 2009 2:46 AM
Report abusive comment

Great Job

Thanks Nasir. It was really very good description/explanation of using java objects in jasperreports. Am searching for this and got a solution from you. My request, can you please share .jrxml's of the master and sub repots as am not using iReport. I hope this will help many users who are not using iReport.

Last edited Jan 20, 2009 10:47 AM
Report abusive comment

Great Example

Thanks Nasir, great example. I was looking everywhere for something like this, but all sites just showed how to write a report with JDBC Datasource

Last edited Jan 19, 2009 5:29 AM
Report abusive comment
Nasir Qureshi
Nasir Qureshi
IT Consultant
Chicago,Illinois
Article rating:
Your rating:

Reviews

    Similar Content on the Web

    Knol translations

    Activity for this knol

    This week:

    247pageviews
    2comments

    Totals:

    21338pageviews
    30comments