/* * Copyright 2002-2008 the original author or authors. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.springframework.web.servlet.view.jasperreports; import java.io.ByteArrayOutputStream; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import javax.servlet.http.HttpServletResponse; import net.sf.jasperreports.engine.JRExporter; import net.sf.jasperreports.engine.JRExporterParameter; import net.sf.jasperreports.engine.JasperPrint; import org.springframework.ui.jasperreports.JasperReportsUtils; import org.springframework.util.CollectionUtils; import org.springframework.web.util.WebUtils; /** * Extends <code>AbstractJasperReportsView</code> to provide basic rendering logic * for views that use a fixed format, e.g. always PDF or always HTML. * * <p>Subclasses need to implement two template methods: <code>createExporter</code> * to create a JasperReports exporter for a specific output format, and * <code>useWriter</code> to determine whether to write text or binary content. * * @author Rob Harrop * @author Juergen Hoeller * @since 1.1.5 * @see #createExporter() * @see #useWriter() */ public abstract class AbstractJasperReportsSingleFormatView extends AbstractJasperReportsView { protected boolean generatesDownloadContent() { return !useWriter(); } /** * Perform rendering for a single Jasper Reports exporter, that is, * for a pre-defined output format. */ protected void renderReport(JasperPrint populatedReport, Map model, HttpServletResponse response) throws Exception { JRExporter exporter = createExporter(); // Set exporter parameters - overriding with values from the Model. Map mergedExporterParameters = mergeExporterParameters(model); if (!CollectionUtils.isEmpty(mergedExporterParameters)) { exporter.setParameters(mergedExporterParameters); } if (useWriter()) { renderReportUsingWriter(exporter, populatedReport, response); } else { renderReportUsingOutputStream(exporter, populatedReport, response); } } /** * Merges the configured JRExporterParameters with any specified in the supplied model data. * JRExporterParameters in the model override those specified in the configuration. * @see #setExporterParameters(java.util.Map) */ protected Map mergeExporterParameters(Map model) { Map mergedParameters = new HashMap(); Map convertedExporterParameters = getConvertedExporterParameters(); if (!CollectionUtils.isEmpty(convertedExporterParameters)) { mergedParameters.putAll(convertedExporterParameters); } for (Iterator it = model.keySet().iterator(); it.hasNext();) { Object key = it.next(); if (key instanceof JRExporterParameter) { Object value = model.get(key); Object convertedValue = convertParameterValue((JRExporterParameter) key, value); mergedParameters.put(key, convertedValue); } } return mergedParameters; } /** * We need to write text to the response Writer. * @param exporter the JasperReports exporter to use * @param populatedReport the populated <code>JasperPrint</code> to render * @param response the HTTP response the report should be rendered to * @throws Exception if rendering failed */ protected void renderReportUsingWriter( JRExporter exporter, JasperPrint populatedReport, HttpServletResponse response) throws Exception { // Copy the encoding configured for the report into the response. String contentType = getContentType(); String encoding = (String) exporter.getParameter(JRExporterParameter.CHARACTER_ENCODING); if (encoding != null) { // Only apply encoding if content type is specified but does not contain charset clause already. if (contentType != null && contentType.toLowerCase().indexOf(WebUtils.CONTENT_TYPE_CHARSET_PREFIX) == -1) { contentType = contentType + WebUtils.CONTENT_TYPE_CHARSET_PREFIX + encoding; } } response.setContentType(contentType); // Render report into HttpServletResponse's Writer. JasperReportsUtils.render(exporter, populatedReport, response.getWriter()); } /** * We need to write binary output to the response OutputStream. * @param exporter the JasperReports exporter to use * @param populatedReport the populated <code>JasperPrint</code> to render * @param response the HTTP response the report should be rendered to * @throws Exception if rendering failed */ protected void renderReportUsingOutputStream( JRExporter exporter, JasperPrint populatedReport, HttpServletResponse response) throws Exception { // IE workaround: write into byte array first. ByteArrayOutputStream baos = createTemporaryOutputStream(); JasperReportsUtils.render(exporter, populatedReport, baos); writeToResponse(response, baos); } /** * Create a JasperReports exporter for a specific output format, * which will be used to render the report to the HTTP response. * <p>The <code>useWriter</code> method determines whether the * output will be written as text or as binary content. * @see #useWriter() */ protected abstract JRExporter createExporter(); /** * Return whether to use a <code>java.io.Writer</code> to write text content * to the HTTP response. Else, a <code>java.io.OutputStream</code> will be used, * to write binary content to the response. * @see javax.servlet.ServletResponse#getWriter() * @see javax.servlet.ServletResponse#getOutputStream() */ protected abstract boolean useWriter(); }