All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ro.nextreports.engine.chart.JFreeChartExporter Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 ro.nextreports.engine.chart;

import java.awt.Color;
import java.awt.Shape;
import java.awt.geom.Ellipse2D;
import java.awt.geom.GeneralPath;
import java.io.File;
import java.io.IOException;
import java.io.Serializable;
import java.sql.SQLException;
import java.text.DecimalFormat;
import java.text.FieldPosition;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;

import org.jfree.chart.ChartFactory;
import org.jfree.chart.ChartUtilities;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.axis.CategoryAxis;
import org.jfree.chart.axis.CategoryLabelPositions;
import org.jfree.chart.axis.NumberAxis;
import org.jfree.chart.axis.NumberTickUnit;
import org.jfree.chart.axis.ValueAxis;
import org.jfree.chart.plot.CategoryPlot;
import org.jfree.chart.plot.DatasetRenderingOrder;
import org.jfree.chart.plot.PiePlot;
import org.jfree.chart.plot.PlotOrientation;
import org.jfree.chart.plot.XYPlot;
import org.jfree.chart.renderer.category.AreaRenderer;
import org.jfree.chart.renderer.category.BarRenderer;
import org.jfree.chart.renderer.category.LineAndShapeRenderer;
import org.jfree.chart.renderer.category.StandardBarPainter;
import org.jfree.chart.renderer.xy.XYItemRenderer;
import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer;
import org.jfree.chart.title.TextTitle;
import org.jfree.data.Range;
import org.jfree.data.category.DefaultCategoryDataset;
import org.jfree.data.general.DefaultPieDataset;
import org.jfree.data.xy.DefaultXYZDataset;
import org.jfree.data.xy.XYDataset;
import org.jfree.data.xy.XYSeries;
import org.jfree.data.xy.XYSeriesCollection;
import org.jfree.ui.HorizontalAlignment;
import org.jfree.util.PublicCloneable;
import org.jfree.chart.labels.StandardCategoryItemLabelGenerator;
import org.jfree.chart.labels.StandardXYItemLabelGenerator;
import org.jfree.chart.labels.StandardPieSectionLabelGenerator;
import org.jfree.chart.labels.XYItemLabelGenerator;

import ro.nextreports.engine.exporter.exception.NoDataFoundException;
import ro.nextreports.engine.exporter.util.function.AbstractGFunction;
import ro.nextreports.engine.exporter.util.function.FunctionFactory;
import ro.nextreports.engine.exporter.util.function.FunctionUtil;
import ro.nextreports.engine.exporter.util.function.GFunction;
import ro.nextreports.engine.i18n.I18nUtil;
import ro.nextreports.engine.queryexec.QueryException;
import ro.nextreports.engine.queryexec.QueryResult;
import ro.nextreports.engine.util.StringUtil;
import ro.nextreports.engine.util.chart.CylinderRenderer;
import ro.nextreports.engine.util.chart.Star;

public class JFreeChartExporter implements ChartExporter {
	
	private QueryResult result;
    private Chart chart;    
    private String chartImageName;
    private Object lastXObjValue = null;    
    private boolean integerXValue = true;
    private DefaultCategoryDataset barDataset;
    private DefaultPieDataset pieDataset;
    private DefaultCategoryDataset lineBarDataset;
    private DefaultXYZDataset bubbleDataset;    
    private Map bubbleData = new LinkedHashMap();
    private String path;    
    private int width;
    private int height;
    private Map parameterValues;
    private final String DEFAULT_LEGEND_PREFIX = "_L_";
    private static final int DEFAULT_WIDTH = 500;
    private static final int DEFAULT_HEIGHT = 300;
    private Map xValueSerie = new HashMap();
    private float transparency = 0.7f;
    private boolean isLineCombo = false;
    private String language;
    
    public JFreeChartExporter(Map parameterValues, QueryResult result, Chart chart, String language) {
    	this(parameterValues, result, chart, ".", DEFAULT_WIDTH, DEFAULT_HEIGHT, language);
    }
    
    public JFreeChartExporter(Map parameterValues, QueryResult result, Chart chart, String path, String language) {
    	this(parameterValues, result, chart, path, DEFAULT_WIDTH, DEFAULT_HEIGHT, language);
    }
    
    public JFreeChartExporter(Map parameterValues, QueryResult result, Chart chart, String path, 
    		int width, int height, String language) {
    	
    	this(parameterValues, result, chart, path, null, width, height, language);
    }
    
    public JFreeChartExporter(Map parameterValues, QueryResult result, Chart chart, String path, 
    		String imageName, int width, int height, String language) {
    	
    	if (width <= 0) {
    		width = DEFAULT_WIDTH;
    	}
    	if (height <= 0) {
    		height = DEFAULT_HEIGHT;
    	}
    	this.parameterValues = parameterValues;
        this.result = result;
        this.chart = chart;
        this.path = path;
        this.chartImageName = imageName;
        this.width = width;
        this.height = height;
        this.language = language;
    }

	public boolean export() throws QueryException, NoDataFoundException {
		testForData();
        createImage();
        return true;
	}
	
	private void testForData() throws QueryException, NoDataFoundException {
        // for procedure call we do not know the row count (is -1)
        if (result == null || result.getColumnCount() <= 0 || result.getRowCount() == 0) {
            throw new NoDataFoundException();
        }
    }
	
	private void createImage() throws QueryException {
		byte type = chart.getType().getType();  
		JFreeChart jfreechart = null;
        if (ChartType.LINE == type) {
        	jfreechart = createLineChart();
        } else if ((ChartType.BAR == type) || (ChartType.NEGATIVE_BAR == type)) {
        	jfreechart = createBarChart(false, false);
        } else if (ChartType.BAR_COMBO == type) {
        	jfreechart = createBarChart(false, false, true);	
        } else if (ChartType.HORIZONTAL_BAR == type) {
        	jfreechart = createBarChart(true, false);
        } else if (ChartType.STACKED_BAR == type) {
        	jfreechart = createBarChart(false, true);
        } else if (ChartType.STACKED_BAR_COMBO == type) {
        	jfreechart = createBarChart(false, true, true);	
        } else if (ChartType.HORIZONTAL_STACKED_BAR == type) {
        	jfreechart = createBarChart(true, true);
        } else if (ChartType.PIE == type) {
        	jfreechart = createPieChart();
        } else if (ChartType.BUBBLE == type) {
        	jfreechart = createBubbleChart();	
        } else if (ChartType.AREA == type) {
        	jfreechart = createAreaChart();
        } 
        try {        	
        	if ((chartImageName == null) || "".equals(chartImageName.trim())) {
        		chartImageName = "chart_" + UUID.randomUUID() + ".jpg";
        	} 
            ChartUtilities.saveChartAsJPEG(new File(path + File.separator + chartImageName), jfreechart, width, height);
        } catch (IOException e) {
            System.err.println("Problem occurred creating chart.");
        }
	}
	
	private JFreeChart createLineChart() throws QueryException {
		XYSeriesCollection dataset = new XYSeriesCollection(); 
		String chartTitle = replaceParameters(chart.getTitle().getTitle());		
		chartTitle = StringUtil.getI18nString(chartTitle, I18nUtil.getLanguageByName(chart, language));
		Object[] charts = new Object[chart.getYColumns().size()];
		List legends = chart.getYColumnsLegends();
		boolean hasLegend = false;
		for (int i = 0; i < charts.length; i++) {
			String legend = "";
			try {
				legend = replaceParameters(legends.get(i));
				legend = StringUtil.getI18nString(legend, I18nUtil.getLanguageByName(chart, language));
			} catch (IndexOutOfBoundsException ex){
				// no legend set
			}
			if ((legend != null) && !"".equals(legend.trim())) {
				hasLegend = true;
			}
			XYSeries lineChart = new XYSeries(legend);			
			charts[i] = lineChart;
			dataset.addSeries(lineChart);
		}
		
		String xLegend = StringUtil.getI18nString(replaceParameters(chart.getXLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		String yLegend = StringUtil.getI18nString(replaceParameters(chart.getYLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		
		JFreeChart jfreechart = ChartFactory.createXYLineChart(
				chartTitle, // Title
				replaceParameters(xLegend), // x-axis Label
				replaceParameters(yLegend), // y-axis Label
                dataset, // Dataset
                PlotOrientation.VERTICAL, // Plot Orientation
                true, // Show Legend
                true, // Use tooltips
                false // Configure chart to generate URLs?
            );
		
		// hide legend if necessary
		if (!hasLegend) {
			jfreechart.removeLegend();
		}
		
		// hide border
		jfreechart.setBorderVisible(false);
		
		// title
		setTitle(jfreechart);
		
		// charts colors & values 
		boolean showValues = (chart.getShowYValuesOnChart() == null) ? false : chart.getShowYValuesOnChart();
		DecimalFormat decimalFormat;
		DecimalFormat percentageFormat;
		if (chart.getYTooltipPattern() == null) {
			decimalFormat = new DecimalFormat("#");
			percentageFormat = new DecimalFormat("0.00%");
		} else {
			decimalFormat = new DecimalFormat(chart.getYTooltipPattern());
			percentageFormat = decimalFormat;
		}
		XYPlot plot = (XYPlot) jfreechart.getPlot();		
		for (int i = 0; i < charts.length; i++) {
			plot.getRenderer().setSeriesPaint(i, chart.getForegrounds().get(i));
			if (showValues) {
				plot.getRenderer().setSeriesItemLabelsVisible(i, true); 
				plot.getRenderer().setSeriesItemLabelGenerator(i, new StandardXYItemLabelGenerator("{2}", decimalFormat, percentageFormat));
			}
		}
		
		if (showValues) {
			// increase a little bit the range axis to view all item label values over points
			plot.getRangeAxis().setUpperMargin(0.2);
		}
				
		// grid axis visibility & colors 
		if ((chart.getXShowGrid() != null) && !chart.getXShowGrid()) {			
			plot.setDomainGridlinesVisible(false);                           
        } else {        	
        	if (chart.getXGridColor() != null) {        		
        		plot.setDomainGridlinePaint(chart.getXGridColor());
        	} else {        		
        		plot.setDomainGridlinePaint(Color.BLACK);
        	}
        }
        if ((chart.getYShowGrid() != null) && !chart.getYShowGrid()) {
        	plot.setRangeGridlinesVisible(false);
        } else {        	
        	if (chart.getYGridColor() != null) {
        		plot.setRangeGridlinePaint(chart.getYGridColor());
        	} else {
        		plot.setRangeGridlinePaint(Color.BLACK);
        	}
        }
       
        // chart background
        plot.setBackgroundPaint(chart.getBackground());                
        
        // labels color
        plot.getDomainAxis().setTickLabelPaint(chart.getXColor());
        plot.getRangeAxis().setTickLabelPaint(chart.getYColor());
        
        //legend color
        plot.getDomainAxis().setLabelPaint(chart.getXLegend().getColor());
        plot.getRangeAxis().setLabelPaint(chart.getYLegend().getColor());
        
        // legend font
        plot.getDomainAxis().setLabelFont(chart.getXLegend().getFont());
        plot.getRangeAxis().setLabelFont(chart.getYLegend().getFont());
        
        // hide labels
        if ((chart.getXShowLabel() != null) && !chart.getXShowLabel()) {        	
        	plot.getDomainAxis().setTickLabelsVisible(false);
        	plot.getDomainAxis().setTickMarksVisible(false);
        }
        if ((chart.getYShowLabel() != null) && !chart.getYShowLabel()) {
        	plot.getRangeAxis().setTickLabelsVisible(false);
        	plot.getRangeAxis().setTickMarksVisible(false);
        }       
        
        // label orientation 
        if (chart.getXorientation() == Chart.VERTICAL) {
        	plot.getDomainAxis().setVerticalTickLabels(true);
        }      
        
        // labels fonts
        plot.getDomainAxis().setTickLabelFont(chart.getXLabelFont());
        plot.getRangeAxis().setTickLabelFont(chart.getYLabelFont());        
        
        // point style
        Shape pointShape = null;
        byte style = chart.getType().getStyle();
        switch (style) {
            case ChartType.STYLE_LINE_DOT_SOLID:               
            case ChartType.STYLE_LINE_DOT_HOLLOW:
            	pointShape = new Ellipse2D.Float(-3.0f, -3.0f, 6.0f, 6.0f);
                break;
            case ChartType.STYLE_LINE_DOT_ANCHOR:  // triangle
            	GeneralPath s5 = new GeneralPath();
            	s5.moveTo(0.0f, -3.0f);
            	s5.lineTo(3.0f, 3.0f);
            	s5.lineTo(-3.0f, 3.0f);
            	s5.closePath(); 
            	pointShape = s5;
                break;
            case ChartType.STYLE_LINE_DOT_BOW:
            	GeneralPath s4 = new GeneralPath();
            	s4.moveTo(-3.0f, -3.0f);
            	s4.lineTo(3.0f, -3.0f);
            	s4.lineTo(-3.0f, 3.0f);
            	s4.lineTo(3.0f, 3.0f);
            	s4.closePath(); 
            	pointShape = s4;
                break;
            case ChartType.STYLE_LINE_DOT_STAR:
                pointShape = new Star(-3.0f, 0f).getShape();
                break;
            default:
            	// no shape
                break;
        }

		if (pointShape != null) {
			XYLineAndShapeRenderer renderer = (XYLineAndShapeRenderer) plot.getRenderer();	
			renderer.setUseFillPaint(true);
			for (int i = 0; i < charts.length; i++) {
				renderer.setSeriesShapesVisible(i, true);				
				if (style != ChartType.STYLE_LINE_DOT_SOLID) {
					renderer.setSeriesFillPaint(i, chart.getBackground());					
				} else {
					renderer.setSeriesFillPaint(i, chart.getForegrounds().get(i));		
				}
				renderer.setSeriesShape(i, pointShape);
			}
		}
        
        final HashMap formatValues = createChart(plot.getRangeAxis(), charts);
                
        // in x axis does not contain number values , values are strings representing one unit 
        if (!integerXValue) {        	
        	((NumberAxis)plot.getDomainAxis()).setTickUnit(new NumberTickUnit(1));
        	((NumberAxis)plot.getDomainAxis()).setNumberFormatOverride(new DecimalFormat(){
    			@Override
    			public StringBuffer format(double number, StringBuffer result, FieldPosition fieldPosition) {						
    				String s = formatValues.get(String.valueOf(Math.round(number)));	    				
    				if (s == null) {
    					s = "";
    				}
    				return result.append(s);
    			}        	
            });
        }    
        
        // need integer tick units, otherwise values will look like doubles with thousands separator and decimal point
        final NumberAxis rangeAxis = (NumberAxis) plot.getDomainAxis();
		rangeAxis.setStandardTickUnits(NumberAxis.createIntegerTickUnits());
        
        return jfreechart;
               
	}
	
	private JFreeChart createBarChart(boolean horizontal, boolean stacked) throws QueryException {
		return createBarChart(horizontal, stacked, false);
	}
	
	private JFreeChart createBarChart(boolean horizontal, boolean stacked, boolean isCombo) throws QueryException {
		barDataset = new DefaultCategoryDataset();
		String chartTitle = replaceParameters(chart.getTitle().getTitle());
		chartTitle = StringUtil.getI18nString(chartTitle, I18nUtil.getLanguageByName(chart, language));
		Object[] charts;
		List legends;
		Object[] lineCharts = null;
		String lineLegend = null;
		if (isCombo) {			 
			 lineCharts = new Object[1];
			 if (chart.getYColumnsLegends().size() < chart.getYColumns().size()) {
				 lineLegend = "";
			 } else {
				 lineLegend = chart.getYColumnsLegends().get(chart.getYColumns().size()-1);
			 }
			 charts = new Object[chart.getYColumns().size()-1];
			 legends = chart.getYColumnsLegends().subList(0, chart.getYColumns().size()-1);			 			 
		} else {
			 charts = new Object[chart.getYColumns().size()];
			 legends = chart.getYColumnsLegends();
		}
		boolean hasLegend = false;
		for (int i = 0; i < charts.length; i++) {			
			String legend = "";
			try {
				legend = replaceParameters(legends.get(i));
				legend = StringUtil.getI18nString(legend, I18nUtil.getLanguageByName(chart, language));
			} catch (IndexOutOfBoundsException ex){
				// no legend set
			}			
			// Important : must have default different legends used in barDataset.addValue
			if ((legend == null) || "".equals(legend.trim())) {
				legend = DEFAULT_LEGEND_PREFIX + String.valueOf(i+1);
			} else {
				hasLegend = true;
			}
			charts[i] = legend;			
		}		
		if (isCombo) {
			String leg = "";
			if (lineLegend != null) {
				leg = replaceParameters(lineLegend);
			}
			lineCharts[0] = leg;
		}
						
		byte style = chart.getType().getStyle();
		JFreeChart jfreechart;
						
		String xLegend = StringUtil.getI18nString(replaceParameters(chart.getXLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		String yLegend = StringUtil.getI18nString(replaceParameters(chart.getYLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		PlotOrientation plotOrientation = horizontal ? PlotOrientation.HORIZONTAL : PlotOrientation.VERTICAL;
		if (stacked) {
			jfreechart = ChartFactory.createStackedBarChart(
    				chartTitle,                    // chart title
    				xLegend,                       // x-axis Label
    				yLegend,                       // y-axis Label
    	            barDataset,                    // data
    	            plotOrientation,               // orientation
    	            true,                          // include legend
    	            true,                          // tooltips
    	            false                          // URLs
    	        );
		} else {
			switch (style) {
			case ChartType.STYLE_BAR_PARALLELIPIPED:
			case ChartType.STYLE_BAR_CYLINDER:
				jfreechart = ChartFactory.createBarChart3D(
						chartTitle,                    // chart title
						xLegend,                       // x-axis Label
						yLegend,                       // y-axis Label
						barDataset,                    // data
						plotOrientation,               // orientation
						true,                          // include legend
						true,                          // tooltips
						false                          // URLs
						);
				break;
			default:
				jfreechart = ChartFactory.createBarChart(
						chartTitle,                    // chart title
						xLegend,                       // x-axis Label
						yLegend,                       // y-axis Label
						barDataset,                    // data
						plotOrientation,               // orientation
						true,                          // include legend
						true,                          // tooltips
						false                          // URLs
						);
				break;
			}
		}
		
		if (style == ChartType.STYLE_BAR_CYLINDER) {
			((CategoryPlot)jfreechart.getPlot()).setRenderer(new CylinderRenderer());
		}
						
		// hide legend if necessary
		if (!hasLegend) {
			jfreechart.removeLegend();
		}
		
		// hide border
		jfreechart.setBorderVisible(false);
		
		// title
		setTitle(jfreechart);
		
		// chart colors & values shown on bars
		boolean showValues = (chart.getShowYValuesOnChart() == null) ? false : chart.getShowYValuesOnChart();
		CategoryPlot plot = (CategoryPlot)jfreechart.getPlot();		
		plot.setForegroundAlpha(transparency);
		BarRenderer renderer = (BarRenderer) plot.getRenderer();
        renderer.setDrawBarOutline(false);
        renderer.setItemMargin(0); // no gap between bars in same category
        DecimalFormat decimalformat;
        DecimalFormat percentageFormat;
        if (chart.getYTooltipPattern() == null) {
        	decimalformat = new DecimalFormat("#");
        	percentageFormat = new DecimalFormat("0.00%");
		} else {
			decimalformat = new DecimalFormat(chart.getYTooltipPattern());
			percentageFormat = decimalformat;
		}
        for (int i = 0; i < charts.length; i++) {
			renderer.setSeriesPaint(i, chart.getForegrounds().get(i));
			if (showValues) {
				renderer.setSeriesItemLabelsVisible(i, true); 
				renderer.setSeriesItemLabelGenerator(i, new StandardCategoryItemLabelGenerator("{2}", decimalformat, percentageFormat));
			}
		}   
        
        if (showValues) {
        	// increase a little bit the range axis to view all item label values over bars
        	plot.getRangeAxis().setUpperMargin(0.2);
        }
        
        // grid axis visibility & colors 
        if ((chart.getXShowGrid() != null) && !chart.getXShowGrid()) {			
			plot.setDomainGridlinesVisible(false);                           
        } else {        	
        	if (chart.getXGridColor() != null) {        		
        		plot.setDomainGridlinePaint(chart.getXGridColor());
        	} else {        		
        		plot.setDomainGridlinePaint(Color.BLACK);
        	}
        }
        if ((chart.getYShowGrid() != null) && !chart.getYShowGrid()) {
        	plot.setRangeGridlinesVisible(false);
        } else {        	
        	if (chart.getYGridColor() != null) {
        		plot.setRangeGridlinePaint(chart.getYGridColor());
        	} else {
        		plot.setRangeGridlinePaint(Color.BLACK);
        	}
        }
       
        // chart background
        plot.setBackgroundPaint(chart.getBackground());                
        
        // labels color
        plot.getDomainAxis().setTickLabelPaint(chart.getXColor());
        plot.getRangeAxis().setTickLabelPaint(chart.getYColor());
        
        // legend color
        plot.getDomainAxis().setLabelPaint(chart.getXLegend().getColor());
        plot.getRangeAxis().setLabelPaint(chart.getYLegend().getColor());
        
        // legend font
        plot.getDomainAxis().setLabelFont(chart.getXLegend().getFont());
        plot.getRangeAxis().setLabelFont(chart.getYLegend().getFont());
        
        // axis color
        plot.getDomainAxis().setAxisLinePaint(chart.getxAxisColor());
        plot.getRangeAxis().setAxisLinePaint(chart.getyAxisColor());
        
        // hide labels
        if ((chart.getXShowLabel() != null) && !chart.getXShowLabel()) {        	
        	plot.getDomainAxis().setTickLabelsVisible(false);
        	plot.getDomainAxis().setTickMarksVisible(false);
        }
        if ((chart.getYShowLabel() != null) && !chart.getYShowLabel()) {
        	plot.getRangeAxis().setTickLabelsVisible(false);
        	plot.getRangeAxis().setTickMarksVisible(false);
        }       
        
        if (chart.getType().getStyle() == ChartType.STYLE_NORMAL) {
            // no shadow
        	renderer.setShadowVisible(false);
        	// no gradient
        	renderer.setBarPainter(new StandardBarPainter());
        }
        
        // label orientation
        CategoryAxis domainAxis = plot.getDomainAxis();        
        if (chart.getXorientation() == Chart.VERTICAL) {
        	domainAxis.setCategoryLabelPositions(CategoryLabelPositions.createUpRotationLabelPositions(Math.PI / 2));
        } else if (chart.getXorientation() == Chart.DIAGONAL) {
        	domainAxis.setCategoryLabelPositions(CategoryLabelPositions.createUpRotationLabelPositions(Math.PI / 4));
        } else if (chart.getXorientation() == Chart.HALF_DIAGONAL) {
        	domainAxis.setCategoryLabelPositions(CategoryLabelPositions.createUpRotationLabelPositions(Math.PI / 8));
        }            
        
        // labels fonts
        domainAxis.setTickLabelFont(chart.getXLabelFont());
        plot.getRangeAxis().setTickLabelFont(chart.getYLabelFont());        
		
		createChart(plot.getRangeAxis(), charts);           
		
		if (isCombo) {			
			addLineChartOverBar(jfreechart, lineCharts, lineLegend);
		}
		
		return jfreechart;
	}
	
	// see http://www.java2s.com/Code/Java/Chart/JFreeChartOverlaidBarChartDemo.htm
	private JFreeChart addLineChartOverBar(JFreeChart jfreechart, Object[] lineCharts, String lineLegend) throws QueryException {
		// first we read data for bar series, so we have to go back at the start of the result set
		try {
			result.getResultSet().beforeFirst();
		} catch (SQLException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		isLineCombo = true;
		lineBarDataset =  new DefaultCategoryDataset();						
		boolean hasLegend = false;
		for (int i = 0; i < lineCharts.length; i++) {
			String legend = "";
			try {
				legend = replaceParameters(lineLegend);
			} catch (IndexOutOfBoundsException ex){
				// no legend set
			}
			if ((legend != null) && !"".equals(legend.trim())) {
				hasLegend = true;
			}						
			lineCharts[i] = legend;			
		}
									
		int index = chart.getYColumns().size()-1;
		CategoryPlot plot = jfreechart.getCategoryPlot();  		
        final LineAndShapeRenderer renderer2 = new LineAndShapeRenderer();       
        plot.setRenderer(1, renderer2);
        renderer2.setSeriesPaint(0, chart.getForegrounds().get(index));
                  		     		     		
	    final ValueAxis axis2 = new NumberAxis("");	    
	    plot.setRangeAxis(1, axis2);
	    plot.setDataset(1, lineBarDataset);
	    plot.mapDatasetToRangeAxis(1, 1);
	    plot.setDatasetRenderingOrder(DatasetRenderingOrder.FORWARD);
	    
	    boolean showValues = (chart.getShowYValuesOnChart() == null) ? false : chart.getShowYValuesOnChart();
 		DecimalFormat decimalFormat;
 		DecimalFormat percentageFormat;
 		if (chart.getYTooltipPattern() == null) {
 			decimalFormat = new DecimalFormat("#");
 			percentageFormat = new DecimalFormat("0.00%");
 		} else {
 			decimalFormat = new DecimalFormat(chart.getYTooltipPattern());
 			percentageFormat = decimalFormat;
 		}
 		if (showValues) {
 			renderer2.setSeriesItemLabelsVisible(0, true); 
 			renderer2.setSeriesItemLabelGenerator(0, new StandardCategoryItemLabelGenerator("{2}", decimalFormat, percentageFormat));
			// increase a little bit the range axis to view all item label values over points
			plot.getRangeAxis(1).setUpperMargin(0.2);
		}
					                
        final HashMap formatValues = createChart(chart.getYColumns().subList(index, index+1), plot.getRangeAxis(1), lineCharts);                        
        
        isLineCombo = false;
        return jfreechart;               
	}
		
	private JFreeChart createAreaChart() throws QueryException {
		barDataset = new DefaultCategoryDataset();
		String chartTitle = replaceParameters(chart.getTitle().getTitle());
		chartTitle = StringUtil.getI18nString(chartTitle, I18nUtil.getLanguageByName(chart, language));
		Object[] charts = new Object[chart.getYColumns().size()];
		List legends = chart.getYColumnsLegends();
		boolean hasLegend = false;
		for (int i = 0; i < charts.length; i++) {			
			String legend = "";
			try {
				legend = replaceParameters(legends.get(i));
				legend = StringUtil.getI18nString(legend, I18nUtil.getLanguageByName(chart, language));
			} catch (IndexOutOfBoundsException ex){
				// no legend set
			}			
			// Important : must have default different legends used in barDataset.addValue
			if ((legend == null) || "".equals(legend.trim())) {
				legend = DEFAULT_LEGEND_PREFIX + String.valueOf(i+1);
			} else {
				hasLegend = true;
			}			
			charts[i] = legend;			
		}		
						
		String xLegend = StringUtil.getI18nString(replaceParameters(chart.getXLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		String yLegend = StringUtil.getI18nString(replaceParameters(chart.getYLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		byte style = chart.getType().getStyle();
		JFreeChart jfreechart = ChartFactory.createAreaChart(
	            "Area Chart",             // chart title
	            xLegend,                  // x-axis Label
				yLegend,                  // y-axis Label
	            barDataset,               // data
	            PlotOrientation.VERTICAL, // orientation
	            true,                     // include legend
	            true,                     // tooltips
	            false                     // urls
	        );
					
		// hide legend if necessary
		if (!hasLegend) {			
			jfreechart.removeLegend();
		}
		
		// hide border
		jfreechart.setBorderVisible(false);
		
		// title
		setTitle(jfreechart);
		
		// chart colors & values shown on bars
		boolean showValues = (chart.getShowYValuesOnChart() == null) ? false : chart.getShowYValuesOnChart();
		CategoryPlot plot = (CategoryPlot)jfreechart.getPlot();		
		plot.setForegroundAlpha(transparency);
		AreaRenderer renderer = (AreaRenderer) plot.getRenderer();        
        DecimalFormat decimalformat;   
        DecimalFormat percentageFormat;
        if (chart.getYTooltipPattern() == null) {
        	decimalformat = new DecimalFormat("#");
        	percentageFormat = new DecimalFormat("0.00%");
		} else {
			decimalformat = new DecimalFormat(chart.getYTooltipPattern());
			percentageFormat = decimalformat;
		}
        for (int i = 0; i < charts.length; i++) {        	
			renderer.setSeriesPaint(i, chart.getForegrounds().get(i));
			if (showValues) {
				renderer.setSeriesItemLabelsVisible(i, true); 
				renderer.setSeriesItemLabelGenerator(i, new StandardCategoryItemLabelGenerator("{2}", decimalformat, percentageFormat));
			}
		}   
        
        if (showValues) {
        	// increase a little bit the range axis to view all item label values over bars
        	plot.getRangeAxis().setUpperMargin(0.2);
        }
        
        // grid axis visibility & colors 
        if ((chart.getXShowGrid() != null) && !chart.getXShowGrid()) {			
			plot.setDomainGridlinesVisible(false);                           
        } else {        	
        	if (chart.getXGridColor() != null) {        		
        		plot.setDomainGridlinePaint(chart.getXGridColor());
        	} else {        		
        		plot.setDomainGridlinePaint(Color.BLACK);
        	}
        }
        if ((chart.getYShowGrid() != null) && !chart.getYShowGrid()) {
        	plot.setRangeGridlinesVisible(false);
        } else {        	
        	if (chart.getYGridColor() != null) {
        		plot.setRangeGridlinePaint(chart.getYGridColor());
        	} else {
        		plot.setRangeGridlinePaint(Color.BLACK);
        	}
        }
       
        // chart background
        plot.setBackgroundPaint(chart.getBackground());                
        
        // labels color
        plot.getDomainAxis().setTickLabelPaint(chart.getXColor());
        plot.getRangeAxis().setTickLabelPaint(chart.getYColor());
        
        // legend color
        plot.getDomainAxis().setLabelPaint(chart.getXLegend().getColor());
        plot.getRangeAxis().setLabelPaint(chart.getYLegend().getColor());
        
        // legend font
        plot.getDomainAxis().setLabelFont(chart.getXLegend().getFont());
        plot.getRangeAxis().setLabelFont(chart.getYLegend().getFont());
        
        // axis color
        plot.getDomainAxis().setAxisLinePaint(chart.getxAxisColor());
        plot.getRangeAxis().setAxisLinePaint(chart.getyAxisColor());
        
        // hide labels
        if ((chart.getXShowLabel() != null) && !chart.getXShowLabel()) {        	
        	plot.getDomainAxis().setTickLabelsVisible(false);
        	plot.getDomainAxis().setTickMarksVisible(false);
        }
        if ((chart.getYShowLabel() != null) && !chart.getYShowLabel()) {
        	plot.getRangeAxis().setTickLabelsVisible(false);
        	plot.getRangeAxis().setTickMarksVisible(false);
        }       
        
        // label orientation
        CategoryAxis domainAxis = plot.getDomainAxis();        
        if (chart.getXorientation() == Chart.VERTICAL) {
        	domainAxis.setCategoryLabelPositions(CategoryLabelPositions.createUpRotationLabelPositions(Math.PI / 2));
        } else if (chart.getXorientation() == Chart.DIAGONAL) {
        	domainAxis.setCategoryLabelPositions(CategoryLabelPositions.createUpRotationLabelPositions(Math.PI / 4));
        } else if (chart.getXorientation() == Chart.HALF_DIAGONAL) {
        	domainAxis.setCategoryLabelPositions(CategoryLabelPositions.createUpRotationLabelPositions(Math.PI / 8));
        }     
        
        // labels fonts
        plot.getDomainAxis().setTickLabelFont(chart.getXLabelFont());
        plot.getRangeAxis().setTickLabelFont(chart.getYLabelFont());      
		
		createChart(plot.getRangeAxis(), charts);              
		
		return jfreechart;
	}
	
	
	private JFreeChart createPieChart() throws QueryException {		
		pieDataset = new DefaultPieDataset();
		String chartTitle = replaceParameters(chart.getTitle().getTitle());
		chartTitle = StringUtil.getI18nString(chartTitle, I18nUtil.getLanguageByName(chart, language));
		JFreeChart jfreechart = ChartFactory.createPieChart(
				    chartTitle,
	                pieDataset, 
	                true, 
	                true, 
	                false);
		
		// hide border
		jfreechart.setBorderVisible(false);
		
		// title
		setTitle(jfreechart);
		
		PiePlot plot = (PiePlot)jfreechart.getPlot();
		plot.setForegroundAlpha(transparency);
		// a start angle used to create similarities between flash chart and this jfreechart
		plot.setStartAngle(330);	
		// legend label will contain the text and the value
		plot.setLegendLabelGenerator(new StandardPieSectionLabelGenerator("{0} = {1}"));
		
		// no shadow
		plot.setShadowXOffset(0);
		plot.setShadowYOffset(0);
		
		DecimalFormat decimalformat;
		DecimalFormat percentageFormat;
		if (chart.getYTooltipPattern() == null) {
        	decimalformat = new DecimalFormat("#");
        	percentageFormat = new DecimalFormat("0.00%");
		} else {
			decimalformat = new DecimalFormat(chart.getYTooltipPattern());
			percentageFormat = decimalformat;
		}
		boolean showValues = (chart.getShowYValuesOnChart() == null) ? false : chart.getShowYValuesOnChart();
		if (showValues) {
			// label will contain also the percentage formatted with two decimals
			plot.setLabelGenerator(new StandardPieSectionLabelGenerator("{0} ({2})", decimalformat, percentageFormat));
		}
		
		// chart background
        plot.setBackgroundPaint(chart.getBackground());
                                                      
        createChart(null, new Object[1]);   
        
        // after chart creation we can set slices colors
        List  keys = pieDataset.getKeys();
        List colors = chart.getForegrounds();
        for (int i = 0, size = colors.size(); i < keys.size(); i++) {            	
            plot.setSectionPaint(keys.get(i), colors.get(i % size));
            plot.setLabelFont(chart.getFont());
        }
		
		return jfreechart;
	}
	
	private JFreeChart createBubbleChart() throws QueryException {		
		bubbleDataset = new DefaultXYZDataset();
				
		// x, y are inverted for jfree bubble chart!
		// that's why we use minX & maxX values to compute Tu (tickUnit)
		String chartTitle = replaceParameters(chart.getTitle().getTitle());
		chartTitle = StringUtil.getI18nString(chartTitle, I18nUtil.getLanguageByName(chart, language));
		String xLegend = StringUtil.getI18nString(replaceParameters(chart.getYLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		String yLegend = StringUtil.getI18nString(replaceParameters(chart.getXLegend().getTitle()), I18nUtil.getLanguageByName(chart, language));
		JFreeChart jfreechart = ChartFactory.createBubbleChart(
				    chartTitle,
				    xLegend, // x-axis Label
					yLegend, // y-axis Label
				    bubbleDataset, 
				    PlotOrientation.HORIZONTAL,
	                true, 
	                true, 
	                false);				
		
		// hide border
		jfreechart.setBorderVisible(false);
		
		// title
		setTitle(jfreechart);
		
		boolean showValues = (chart.getShowYValuesOnChart() == null) ? false : chart.getShowYValuesOnChart();
		
		XYPlot xyplot = (XYPlot) jfreechart.getPlot();
		xyplot.setForegroundAlpha(transparency);
		XYItemRenderer xyitemrenderer = xyplot.getRenderer();
								
		DecimalFormat decimalformat;   
        DecimalFormat percentageFormat;
        if (chart.getYTooltipPattern() == null) {
        	decimalformat = new DecimalFormat("#");
        	percentageFormat = new DecimalFormat("0.00%");
		} else {
			decimalformat = new DecimalFormat(chart.getYTooltipPattern());
			percentageFormat = decimalformat;
		}
        
        
        if (showValues) {
        	// increase a little bit the range axis to view all item label values over bars
        	xyplot.getRangeAxis().setUpperMargin(0.2);
        }
        
        // grid axis visibility & colors 
        if ((chart.getXShowGrid() != null) && !chart.getXShowGrid()) {			
        	xyplot.setDomainGridlinesVisible(false);                           
        } else {        	
        	if (chart.getXGridColor() != null) {        		
        		xyplot.setDomainGridlinePaint(chart.getXGridColor());
        	} else {        		
        		xyplot.setDomainGridlinePaint(Color.BLACK);
        	}
        }
        if ((chart.getYShowGrid() != null) && !chart.getYShowGrid()) {
        	xyplot.setRangeGridlinesVisible(false);
        } else {        	
        	if (chart.getYGridColor() != null) {
        		xyplot.setRangeGridlinePaint(chart.getYGridColor());
        	} else {
        		xyplot.setRangeGridlinePaint(Color.BLACK);
        	}
        }
       
        // chart background
        xyplot.setBackgroundPaint(chart.getBackground());                
        
        // labels color
        xyplot.getDomainAxis().setTickLabelPaint(chart.getXColor());
        xyplot.getRangeAxis().setTickLabelPaint(chart.getYColor());
        
        // legend color
        xyplot.getDomainAxis().setLabelPaint(chart.getXLegend().getColor());
        xyplot.getRangeAxis().setLabelPaint(chart.getYLegend().getColor());
        
        // legend font
        xyplot.getDomainAxis().setLabelFont(chart.getXLegend().getFont());
        xyplot.getRangeAxis().setLabelFont(chart.getYLegend().getFont());
        
        // axis color
        xyplot.getDomainAxis().setAxisLinePaint(chart.getxAxisColor());
        xyplot.getRangeAxis().setAxisLinePaint(chart.getyAxisColor());
        
        // hide labels
        if ((chart.getXShowLabel() != null) && !chart.getXShowLabel()) {        	
        	xyplot.getDomainAxis().setTickLabelsVisible(false);
        	xyplot.getDomainAxis().setTickMarksVisible(false);
        }
        if ((chart.getYShowLabel() != null) && !chart.getYShowLabel()) {
        	xyplot.getRangeAxis().setTickLabelsVisible(false);
        	xyplot.getRangeAxis().setTickMarksVisible(false);
        }                              
        
        // label orientation
        ValueAxis domainAxis = xyplot.getDomainAxis();        
        if (chart.getXorientation() == Chart.VERTICAL) {
        	domainAxis.setLabelAngle(Math.PI / 2);
        } else if (chart.getXorientation() == Chart.DIAGONAL) {
        	domainAxis.setLabelAngle(Math.PI / 4);
        } else if (chart.getXorientation() == Chart.HALF_DIAGONAL) {
        	domainAxis.setLabelAngle(Math.PI / 8);
        }     
        
        // labels fonts
        xyplot.getDomainAxis().setTickLabelFont(chart.getXLabelFont());
        xyplot.getRangeAxis().setTickLabelFont(chart.getYLabelFont());
				                                                      
        createChart(xyplot.getRangeAxis(), new Object[4]);     
        
        double minX = Double.MAX_VALUE;
        double maxX = Double.MIN_VALUE;
        double minY = Double.MAX_VALUE;
        double maxY = Double.MIN_VALUE;
        double maxZ = Double.MIN_VALUE;
        for (String serie : bubbleData.keySet()) {
        	XYZList xyzList = bubbleData.get(serie);	
        	List yList = xyzList.getyList();
        	for (Number n : yList) {
        		minY = Math.min(minY, n.doubleValue());
        		maxY = Math.max(maxY, n.doubleValue());
        	}    
        	List xList = xyzList.getxList();
        	for (Number n : xList) {
        		minX = Math.min(minX, n.doubleValue());
        		maxX = Math.max(maxX, n.doubleValue());
        	}    
        	List zList = xyzList.getzList();
        	for (Number n : zList) {        		
        		maxZ = Math.max(maxZ, n.doubleValue());
        	}
        }
        
        double tu = Math.floor((maxX-minX)/5 +0.5d);      
        NumberTickUnit rUnit = new NumberTickUnit(tu);
        ((NumberAxis) xyplot.getRangeAxis()).setTickUnit(rUnit);
        
        // make the bubble with text fit on X axis (which is shown vertically!)
        xyplot.getDomainAxis().setUpperMargin(0.2);   
        xyplot.getDomainAxis().setLowerMargin(0.2);
        
        for (String serie : bubbleData.keySet()) {
			XYZList xyzList = bubbleData.get(serie);				
			double[][] data = { getDoubleArray(xyzList.getyList()), getDoubleArray(xyzList.getxList()), getZDoubleArray(xyzList.getzList(), tu, maxZ) };
			bubbleDataset.addSeries(serie, data);
		}
        
        int series = bubbleData.keySet().size();
        for (int i = 0; i < series; i++) {        	
        	xyitemrenderer.setSeriesPaint(i, chart.getForegrounds().get(i));
			if (showValues) {
				xyitemrenderer.setSeriesItemLabelsVisible(i, true); 
				xyitemrenderer.setSeriesItemLabelGenerator(i, new StandardXYItemLabelGenerator("{2}", decimalformat, percentageFormat));
			}
		}  

        // show labels on bubbles
        xyitemrenderer.setBaseItemLabelsVisible(true);
		xyitemrenderer.setBaseItemLabelGenerator(new LegendXYItemLabelGenerator());
		return jfreechart;
	}
	
	public class LegendXYItemLabelGenerator extends StandardXYItemLabelGenerator implements XYItemLabelGenerator, Cloneable, PublicCloneable, Serializable {
		
		public LegendXYItemLabelGenerator() {
		    super();		    
		}
		
		@Override
		public String generateLabel(XYDataset dataset, int series, int item) {					
			String[] keys = new String[bubbleData.size()]; 
			bubbleData.keySet().toArray(keys);
			String legend = bubbleData.get(keys[series]).getLabels().get(item);
			legend = StringUtil.getI18nString(legend, I18nUtil.getLanguageByName(chart, language));			
			return legend;
		}
	}
	
	private void setTitle(JFreeChart jfreechart) {
		TextTitle title = new TextTitle(StringUtil.getI18nString(replaceParameters(chart.getTitle().getTitle()), I18nUtil.getLanguageByName(chart, language)));
		title.setFont(chart.getTitle().getFont());
		title.setPaint(chart.getTitle().getColor());
		if  (chart.getTitle().getAlignment() == ChartTitle.LEFT_ALIGNMENT) {
			title.setHorizontalAlignment(HorizontalAlignment.LEFT);
		} else if  (chart.getTitle().getAlignment() == ChartTitle.RIGHT_ALIGNMENT) {
			title.setHorizontalAlignment(HorizontalAlignment.RIGHT);
		} else {
			title.setHorizontalAlignment(HorizontalAlignment.CENTER);
		}
		jfreechart.setTitle(title);
	}
	
	private HashMap createChart(ValueAxis rangeAxis, Object[] charts) throws QueryException {   
		return createChart(chart.getYColumns(), rangeAxis, charts);
	}
	
	private HashMap createChart(List yColumns, ValueAxis rangeAxis, Object[] charts) throws QueryException {       
        int row = 0;
        Object previous = null;
        String xColumn = chart.getXColumn();
        String xPattern = chart.getXPattern();          
        String lastXValue = "";
        HashMap formatValues = new HashMap();      
        Number min = Double.MAX_VALUE;
        Number max = Double.MIN_VALUE;
        
        int chartsNo = charts.length;        
        int[] index = new int[chartsNo];
        GFunction[] functions = new GFunction[chartsNo];   
        for (int i = 0; i < chartsNo; i++) {
            functions[i] = FunctionFactory.getFunction(chart.getYFunction());
            index[i] = 1;
        }
        boolean isStacked = chart.getType().isStacked();        
        while (result.hasNext()) {
        	
            Object[] objects = new Object[chartsNo];
            Number[] computedValues = new Number[chartsNo];    
            String currentCategory = null;
            for (int i = 0; i < chartsNo; i++) {
                if (yColumns.get(i) != null) {
                    objects[i] = result.nextValue(yColumns.get(i));
                    Number value = null;
                    String sv = null;
                    if (objects[i] instanceof Number) {
                        value = (Number) objects[i];
                    } else if (objects[i] != null){
                    	if (ChartType.BUBBLE == chart.getType().getType()) {
                    		sv = (String)objects[i];
                    	} 
                        value = 1;
                        integerXValue = false;
                    } else {
                    	value = 0;
                    }
                    computedValues[i] = value;
                    if (sv != null) {
                    	currentCategory = sv;                      	
                    }
                }
            }

            Object xValue = null;                
            if (row == 0) {
                xValue = result.nextValue(xColumn);          
                lastXObjValue = xValue;
                lastXValue = getStringValue(xColumn, xPattern);
            } else {
                xValue = previous;
            }            
            Object newXValue = result.nextValue(xColumn);
            

            boolean add = false;
            int position = 0;
            // no function : add the value
            if (AbstractGFunction.NOOP.equals(functions[0].getName())) {        
            	lastXValue = getStringValue(xColumn, xPattern);            	
                add = true;
                // compute function
            } else {
                boolean equals = FunctionUtil.parameterEquals(xValue, newXValue);                
                if (equals) {
                    for (int i = 0; i < chartsNo; i++) {
                        functions[i].compute(objects[i]);
                    }
                } else {
                    for (int i = 0; i < chartsNo; i++) {
                    	position = i;
                        add = true;
                        computedValues[i] = (Number) functions[i].getComputedValue();
                        functions[i].reset();
                        functions[i].compute(objects[i]);
                    }
                }
            }
            
            if (add) {   
            	Number n;
            	Number sum = 0;            	
            	if (xValue instanceof Number) {
            		n = (Number)newXValue;
            	} else {
            		integerXValue = false;
            		n = index[position]++;
            	}
            	if (chart.getType().isBubble()) {            		
            		XYZList xyzList = bubbleData.get(currentCategory);
            		if (xyzList == null) {
            			xyzList = new XYZList();
            			bubbleData.put(currentCategory, xyzList);            			
            		}
            		xyzList.getxList().add(computedValues[0]);
            		xyzList.getyList().add(computedValues[1]);
            		xyzList.getzList().add(computedValues[2]);            	    
            		min = Math.min(min.doubleValue(), computedValues[0].doubleValue());
                	max = Math.max(max.doubleValue(), computedValues[0].doubleValue());
            	} else {   
	                for (int i = 0; i < chartsNo; i++) {                  	
	                    addValue(charts[i], n, lastXValue, computedValues[i], formatValues, chartsNo);   
	                    if (!isStacked) {
	                    	min = Math.min(min.doubleValue(), computedValues[i].doubleValue());
	                    	max = Math.max(max.doubleValue(), computedValues[i].doubleValue());
	                    } else {
	                    	sum = sum.doubleValue() + computedValues[i].doubleValue();                    	
	                    }
	                }       
            	}
            	if ((chart.getStartingFromZero() !=  null) && chart.getStartingFromZero() && (min.intValue() > 0)) {
            		min = 0;
            	}
                if (isStacked) {
                    min = 0;
                    max = Math.max(max.doubleValue(), sum.doubleValue());                    
                }
                lastXValue = getStringValue(xColumn, xPattern);   
                if (ChartType.BUBBLE == chart.getType().getType()) {
                	XYZList xyzList = bubbleData.get(currentCategory);
                	xyzList.getLabels().add(lastXValue);                
                }
            }
            row++;
            previous = newXValue;            
        }

        // last group
        if (!AbstractGFunction.NOOP.equals(functions[0].getName())) {   
        	Number n;
        	Number sum = 0;
        	if (lastXObjValue instanceof Number) {
        		n = (Number)lastXObjValue;
        	} else {
        		integerXValue = false;
        		n = index[chartsNo-1]++;
        	}
            for (int i = 0; i < chartsNo; i++) {
                Number value = (Number) functions[i].getComputedValue();                
                addValue(charts[i], n, lastXValue, value, formatValues, chartsNo);   
                if (!isStacked) {
                	min = Math.min(min.doubleValue(), value.doubleValue());
                	max = Math.max(max.doubleValue(), value.doubleValue());
                } else {
                	sum = sum.doubleValue() + value.doubleValue();
                }
            } 
            if ((chart.getStartingFromZero() !=  null) && chart.getStartingFromZero() && (min.intValue() > 0)) {
        		min = 0;
        	}
            if (isStacked) {
                min = 0;
                max = Math.max(max.doubleValue(), sum.doubleValue());
            }
        }   
        
        
        setAxisRange(rangeAxis, min, max);

        return formatValues;       
    }
	
	// take care if no function is set : every value is in a differnet category (we use an incremental integer : serie)
	private void addValue(Object chartSerie, Number x, String lastXValue, Number y, HashMap formatValues, int chartsNo) {			
		if (chart.getType().isLine()) {					
		   ((XYSeries)chartSerie).add(x, y);
		} else if (isLineCombo) {			
			lineBarDataset.addValue(y, (String)chartSerie, lastXValue);
		} else if (!chart.getType().isPie()) {
						
			GFunction function = FunctionFactory.getFunction(chart.getYFunction());
			if (!AbstractGFunction.NOOP.equals(function.getName()) || (chartsNo > 1)) {
				barDataset.setValue(y, (String)chartSerie, lastXValue);
			} else {			
				int serie = 0;
				Integer i = xValueSerie.get(lastXValue);
				if (i != null) {
					serie = i+1;
					xValueSerie.put(lastXValue, serie);
					barDataset.setValue(y, (String)chartSerie, lastXValue + " (" + serie + ")");		
				} else {
					barDataset.setValue(y, (String)chartSerie, lastXValue);
					xValueSerie.put(lastXValue, 0);
				}	
			}
			
		} else if (chart.getType().isPie()) {
			if (AbstractGFunction.NOOP.equals(chart.getYFunction())) {
				int serie = 0;
				Integer i = xValueSerie.get(lastXValue);
				if (i != null) {
					serie = i+1;
					xValueSerie.put(lastXValue, serie);
					pieDataset.setValue(lastXValue + " (" + serie + ")", y);					
				} else {
					pieDataset.setValue(lastXValue, y);
					xValueSerie.put(lastXValue, 0);
				}				
			} else {
				pieDataset.setValue(lastXValue, y);
			}
		}
		formatValues.put(x.toString(), lastXValue);
	}
	
	private String getStringValue(String column, String pattern) throws QueryException {
        Object xObject = result.nextValue(column);
        lastXObjValue = xObject;
        return StringUtil.getValueAsString(xObject, pattern);
    }

	public String getChartImageName() {
		return chartImageName;
	}	
	
	public String getChartImageAbsolutePath() {
		File file = new File(path + File.separator+ chartImageName);
		return file.getAbsolutePath();
	}
	
	public int getWidth() {
		return width;
	}
	
	public int getHeight() {
		return height;
	}
	
	// replace $P{...} parameters (used in title and x,y legends
    private String replaceParameters(String text) {
    	if (text == null) {
    		return null;
    	}
        for  (String param : parameterValues.keySet()) {
             text = StringUtil.replace(text, "\\$P\\{" + param + "\\}", StringUtil.getValueAsString(parameterValues.get(param), null,I18nUtil.getLanguageByName(chart, language)));
        }
        return text;
    }
    
    private void setAxisRange(ValueAxis rangeAxis, Number min, Number max) {
    	if (rangeAxis == null) {
    		return;
    	}
    	
    	YRange yRange = new YRange(min, max);
    	yRange = yRange.update();    	
    	rangeAxis.setRange(new Range(yRange.getMin().doubleValue(), yRange.getMax().doubleValue()));
    }
    
    private double[] getDoubleArray(List list) {
    	double[] result = new double[list.size()];
    	for (int i=0, size=list.size(); i zList, double tu, double maxZ) {
    	double[] result = new double[zList.size()];    	    	
    	for (int i=0, size=zList.size(); i




© 2015 - 2025 Weber Informatics LLC | Privacy Policy