ingo@1115: /* ingo@1115: * Copyright (c) 2010 by Intevation GmbH ingo@1115: * ingo@1115: * This program is free software under the LGPL (>=v2.1) ingo@1115: * Read the file LGPL.txt coming with the software for details ingo@1115: * or visit http://www.gnu.org/licenses/ if it does not exist. ingo@1115: */ ingo@1115: ingo@297: package de.intevation.gnv.chart; ingo@297: sascha@779: import de.intevation.gnv.geobackend.base.Result; sascha@779: sascha@779: import de.intevation.gnv.state.describedata.KeyValueDescibeData; sascha@779: ingo@297: import java.awt.Color; sascha@779: ingo@327: import java.awt.geom.Ellipse2D; sascha@779: ingo@315: import java.text.NumberFormat; sascha@779: ingo@297: import java.util.Collection; ingo@297: import java.util.Iterator; ingo@315: import java.util.Locale; ingo@334: import java.util.Map; ingo@297: ingo@297: import org.apache.log4j.Logger; ingo@297: sascha@779: import org.jfree.chart.ChartFactory; ingo@297: import org.jfree.chart.JFreeChart; sascha@779: ingo@315: import org.jfree.chart.axis.Axis; sascha@779: import org.jfree.chart.axis.AxisLocation; ingo@297: import org.jfree.chart.axis.NumberAxis; ingo@297: import org.jfree.chart.axis.NumberTickUnit; sascha@779: ingo@297: import org.jfree.chart.plot.PlotOrientation; ingo@297: import org.jfree.chart.plot.XYPlot; sascha@779: ingo@327: import org.jfree.chart.renderer.xy.XYLineAndShapeRenderer; sascha@779: ingo@331: import org.jfree.chart.title.TextTitle; sascha@779: ingo@364: import org.jfree.data.Range; sascha@779: ingo@297: import org.jfree.data.general.Series; ingo@297: sascha@779: import org.jfree.data.xy.XYDataset; ingo@333: ingo@783: import org.jfree.ui.RectangleInsets; ingo@783: ingo@297: /** ingo@767: * This abstract class defines some methods to adjust chart settings after its ingo@767: * creation. ingo@767: * ingo@767: * @author Ingo Weinzierl ingo@297: */ ingo@297: public abstract class AbstractXYLineChart ingo@297: extends AbstractChart ingo@297: { ingo@767: /** ingo@767: * Constant field used to expand the area between data and chart border. Its ingo@767: * value is {@value}.
ingo@767: * A value of 0.05 equals 5 percent. ingo@767: */ ingo@364: public static final double LOWER_MARGIN = 0.05D; ingo@767: ingo@767: /** ingo@767: * Constant field used to expand the area between data and chart border. Its ingo@767: * value is {@value}.
ingo@767: * A value of 0.05 equals 5 percent. ingo@767: */ ingo@364: public static final double UPPER_MARGIN = 0.05D; ingo@364: ingo@767: /** ingo@767: * Logger used to log with log4j. ingo@767: */ ingo@327: private static Logger log = Logger.getLogger(AbstractXYLineChart.class); ingo@297: ingo@767: /** ingo@767: * Field of supported colors used for lines and data points in charts. ingo@767: */ ingo@297: protected static Color[] COLOR = { ingo@334: Color.black, Color.red, Color.green, Color.blue, Color.yellow, ingo@334: Color.gray, Color.orange, Color.pink, Color.cyan ingo@297: }; ingo@297: ingo@767: /** ingo@767: * Static field to remember the index of the previously used color. ingo@767: */ ingo@334: protected static int nextColor = 0; ingo@334: ingo@767: /** ingo@815: * Default PlotOrientation. ingo@767: */ ingo@297: protected PlotOrientation PLOT_ORIENTATION = PlotOrientation.VERTICAL; ingo@297: ingo@767: /** ingo@767: * Map to store datasets for each parameter. ingo@767: */ ingo@334: protected Map datasets; ingo@334: ingo@767: /** ingo@767: * Map to store max ranges of each parameter (axis.setAutoRange(true) ingo@364: * doesn't seem to work */ ingo@364: protected Map ranges; ingo@364: ingo@767: /** ingo@767: * This method is called by Chart to bring the data into the ingo@767: * right form fitting to JFreeChart objects. ingo@767: */ ingo@297: protected abstract void initData(); ingo@767: ingo@767: /** ingo@767: * Add a value of row to series. ingo@767: * ingo@767: * @param row Result Object returned from database. Contains ingo@767: * a value used to add to series ingo@767: * @param series A JFreeChart Series object. ingo@767: */ ingo@297: protected abstract void addValue(Result row, Series series); ingo@767: ingo@767: /** ingo@767: * Add series to JFreeChart's Dataset object currently which is ingo@767: * processing. ingo@767: * ingo@767: * @param series Series to add. ingo@767: * @param label Label used show in legend. ingo@767: * @param idx Currently not used. ingo@767: */ ingo@334: protected abstract void addSeries(Series series, String label, int idx); ingo@767: ingo@767: /** ingo@767: * Abstract method which is called by Chart interface after ingo@767: * chart creation. It turns an axis' label into a locale specific format. ingo@767: * ingo@767: * @param axis Axis to adjust. ingo@788: * @param locale java.util.Locale object used specify the format. ingo@767: */ ingo@315: protected abstract void localizeDomainAxis(Axis axis, Locale locale); ingo@767: ingo@767: /** ingo@767: * Abstract method to create a label for a series of parameters. ingo@767: * ingo@767: * @param breakPoint1 Identifier returned from database. These identifier ingo@767: * are used to identify the results from database which are all stored in ingo@767: * one big java.util.Collection. ingo@767: * @param breakPoint2 Identifier returned from database. ingo@767: * @param breakPoint3 Identifier returned from database. ingo@767: * ingo@767: * @return Concatinated string of parameter name and measurement. ingo@767: */ ingo@297: protected abstract String createSeriesName( ingo@297: String breakPoint1, ingo@297: String breakPoint2, ingo@297: String breakPoint3 ingo@297: ); ingo@297: ingo@297: ingo@767: /** ingo@767: * @see de.intevation.gnv.chart.Chart#generateChart() ingo@767: */ ingo@297: public JFreeChart generateChart() { ingo@297: log.debug("generate XYLineChart"); ingo@351: nextColor = 0; ingo@297: ingo@297: if (chart != null) ingo@297: return chart; ingo@297: ingo@333: initChart(); ingo@333: ingo@333: chart.addSubtitle(new TextTitle(labels.getSubtitle())); ingo@333: ingo@333: theme.apply(chart); ingo@333: initData(); ingo@333: ingo@344: adjustPlot((XYPlot)chart.getPlot()); ingo@344: ingo@333: return chart; ingo@333: } ingo@333: ingo@333: ingo@333: protected void initChart() { ingo@297: chart = ChartFactory.createXYLineChart( ingo@297: labels.getTitle(), ingo@297: labels.getDomainAxisLabel(), ingo@297: null, ingo@297: null, ingo@297: PLOT_ORIENTATION, ingo@297: true, ingo@297: false, ingo@297: false ingo@297: ); ingo@297: } ingo@297: ingo@297: ingo@767: /** ingo@767: * Method used to adjust the axes after chart generation. Methods for i18n ingo@815: * support ({@link #localizeDomainAxis} and {@link #localizeRangeAxis}) are ingo@815: * called and axes of this series are expanded. ingo@767: * ingo@767: * @param seriesKey Identifier of an axis which have to be adjusted. ingo@767: * @param idx Set the axis identified by seriesKey to position ingo@767: * idx. ingo@767: */ ingo@297: protected void prepareAxis(String seriesKey, int idx) { ingo@297: log.debug("prepare axis of xychart"); ingo@315: ingo@315: XYPlot plot = chart.getXYPlot(); ingo@315: Axis xAxis = plot.getDomainAxis(); ingo@315: NumberAxis yAxis = new NumberAxis(seriesKey); ingo@315: ingo@315: localizeDomainAxis(xAxis, locale); ingo@315: localizeRangeAxis(yAxis, locale); ingo@297: ingo@364: // litte workarround to adjust the max range of axes. ingo@364: // NumberAxis.setAutoRange(true) doesn't seem to work properly. ingo@656: Range yRange = (Range) ranges.get(seriesKey); ingo@1104: double lo = yRange.getLowerBound(); ingo@1104: double hi = yRange.getUpperBound(); ingo@1104: ingo@1104: if (lo == hi) { ingo@1104: yRange = new Range( ingo@1104: lo - (lo / 100 * LOWER_MARGIN), ingo@1104: hi + (hi / 100 * UPPER_MARGIN)); ingo@1104: } ingo@1104: else { ingo@1104: yRange = Range.expand(yRange, LOWER_MARGIN, UPPER_MARGIN); ingo@1104: } ingo@1104: yAxis.setRange(yRange); ingo@364: log.debug("Max Range of dataset is: " + yRange.toString()); ingo@364: ingo@297: if (seriesKey.contains("richtung")) { ingo@315: yAxis.setTickUnit(new NumberTickUnit(30.0)); ingo@315: yAxis.setUpperBound(360.0); ingo@315: yAxis.setLowerBound(0.0); ingo@297: } ingo@297: else { ingo@315: yAxis.setFixedDimension(10.0); ingo@315: yAxis.setAutoRangeIncludesZero(false); ingo@297: } ingo@297: ingo@364: plot.setRangeAxis(idx, yAxis); ingo@364: yAxis.configure(); ingo@364: ingo@297: if (idx % 2 != 0) ingo@297: plot.setRangeAxisLocation(idx, AxisLocation.BOTTOM_OR_RIGHT); ingo@297: else ingo@297: plot.setRangeAxisLocation(idx, AxisLocation.BOTTOM_OR_LEFT); ingo@506: ingo@506: plot.mapDatasetToRangeAxis(idx, idx); ingo@327: } ingo@297: ingo@327: ingo@767: /** ingo@767: * Method to adjust the rendering of a series in a chart. Line color and ingo@767: * symbols of vertices are configured here. ingo@767: * ingo@767: * @param idx Position of the renderer. ingo@767: * @param seriesCount Maximum number of series in this chart. ingo@767: * @param renderLines Lines are displayed if true, otherwise they are not. ingo@767: * @param renderShapes Vertices are displayed if true, otherwise they are ingo@767: * not. ingo@767: */ ingo@327: protected void adjustRenderer( ingo@327: int idx, ingo@334: int seriesCount, ingo@327: boolean renderLines, ingo@327: boolean renderShapes ingo@327: ) { ingo@334: log.debug("Adjust render of series"); ingo@327: XYLineAndShapeRenderer renderer = null; ingo@327: XYPlot plot = chart.getXYPlot(); ingo@327: ingo@327: try { ingo@327: renderer = (XYLineAndShapeRenderer)((XYLineAndShapeRenderer) ingo@327: (plot.getRenderer())).clone(); ingo@327: } ingo@327: catch (CloneNotSupportedException cnse) { ingo@327: log.warn("Error while cloning renderer.", cnse); ingo@327: renderer = new XYLineAndShapeRenderer(renderLines, renderShapes); ingo@327: renderer.setBaseShape(new Ellipse2D.Double(-2,-2,4,4)); ingo@327: } ingo@327: ingo@334: for (int i = 0; i < seriesCount; i++) { ingo@334: renderer.setSeriesShape(i, renderer.getSeriesShape(0)); ingo@774: renderer.setSeriesPaint(i, COLOR[nextColor() % COLOR.length]); ingo@334: renderer.setSeriesShapesVisible(i, renderShapes); ingo@334: renderer.setSeriesLinesVisible(i, renderLines); ingo@334: } ingo@297: plot.setRenderer(idx, renderer); ingo@297: } ingo@297: ingo@788: ingo@788: /** ingo@788: * @return Index of the next color ingo@788: */ ingo@774: protected static synchronized int nextColor() { ingo@774: return nextColor++; ingo@774: } ingo@774: ingo@297: ingo@767: /** sascha@778: * Method to adjust the plot rendering. Disable horizontal grid lines if ingo@767: * plot contains only a single y-axis. ingo@767: * ingo@767: * @param plot JFreeChart Plot object to be adjusted. ingo@767: */ ingo@344: protected void adjustPlot(XYPlot plot) { ingo@344: if (plot.getRangeAxisCount() > 1) ingo@344: plot.setRangeGridlinesVisible(false); ingo@783: ingo@783: plot.setAxisOffset(new RectangleInsets(0, 0, 0, 15)); ingo@344: } ingo@344: ingo@344: ingo@767: /** sascha@778: * Abstract method which is called after chart creation. It turns an ingo@767: * axis' label into a locale specific format. ingo@767: * ingo@767: * @param axis Axis to adjust. ingo@788: * @param locale java.util.Locale object used specify the format. ingo@767: * ingo@767: */ ingo@315: protected void localizeRangeAxis(Axis axis, Locale locale) { ingo@315: if (locale == null) ingo@315: return; ingo@315: ingo@315: log.debug( ingo@315: "Set language of axis [" + axis.getLabel() + "] " + ingo@315: "to " + locale.toString() ingo@315: ); ingo@315: ingo@315: NumberFormat format = NumberFormat.getInstance(locale); ingo@315: ((NumberAxis) axis).setNumberFormatOverride(format); ingo@315: } ingo@315: ingo@315: ingo@767: /** ingo@767: * Return the maximum y-range of dataset. ingo@767: * ingo@767: * @param dataset Dataset to be scaned. ingo@767: * ingo@767: * @return JFreeChart Range object containing min and max y-value. ingo@767: */ ingo@364: public Range getMaxRangeOfDataset(XYDataset dataset) { ingo@364: int seriesCount = dataset.getSeriesCount(); ingo@364: double upper = Double.NEGATIVE_INFINITY; ingo@364: double lower = Double.POSITIVE_INFINITY; ingo@364: ingo@364: for (int i = 0; i < seriesCount; i++) { ingo@364: int itemCount = dataset.getItemCount(i); ingo@364: ingo@364: for (int j = 0; j < itemCount; j++) { ingo@364: Number num = dataset.getY(i, j); ingo@364: ingo@364: if (num != null) { ingo@364: double y = num.doubleValue(); ingo@364: lower = y < lower ? y : lower; ingo@364: upper = y > upper ? y : upper; ingo@364: } ingo@364: } ingo@364: } ingo@364: ingo@364: return new Range(lower, upper); ingo@364: } ingo@364: ingo@364: ingo@767: /** ingo@767: * Return the maximum y-range of dataset with a margin of ingo@767: * percent percent. ingo@767: * ingo@767: * @param dataset Dataset to be scaned. ingo@788: * @param percent Percent used to expand the range. ingo@767: * @return JFreeChart Range object containing min and max y-value with a ingo@767: * margin. ingo@767: */ ingo@364: public Range getMaxRangeOfDatasetWithMargin( ingo@364: XYDataset dataset, ingo@364: double percent ingo@364: ) { ingo@364: Range range = getMaxRangeOfDataset(dataset); ingo@364: double length = range.getLength(); ingo@364: double upper = range.getUpperBound() + length /100 * percent; ingo@364: double lower = range.getLowerBound() - length /100 * percent; ingo@364: ingo@364: return new Range(lower, upper); ingo@364: } ingo@364: ingo@364: ingo@767: /** ingo@767: * Method to find a parameter specified by its value. ingo@767: * ingo@767: * @param label Search string. ingo@767: * ingo@767: * @return Value of a parameter with the given label. ingo@767: */ ingo@334: protected String findParameter(String label) { ingo@334: Iterator iter = parameters.iterator(); ingo@334: ingo@334: while (iter.hasNext()) { ingo@334: KeyValueDescibeData data = (KeyValueDescibeData) iter.next(); ingo@334: String key = data.getValue(); ingo@334: ingo@334: if (label.indexOf(key) > -1) ingo@334: return key; ingo@334: } ingo@334: ingo@334: return label; ingo@334: } ingo@334: ingo@334: ingo@767: /** ingo@767: * Method to find a description of a given collection of values. ingo@767: * ingo@767: * @param values Collection to be scaned. ingo@767: * @param id Identifier and search string of the searched value. ingo@767: * ingo@767: * @return title ingo@767: */ ingo@297: protected String findValueTitle(Collection values, String id) { ingo@297: log.debug("find description of dataset"); ingo@297: tim@307: if (values != null){ tim@307: Iterator it = values.iterator(); tim@307: while (it.hasNext()) { tim@307: KeyValueDescibeData data = (KeyValueDescibeData) it.next(); ingo@315: tim@307: if (id.equals(data.getKey())) tim@307: return data.getValue(); tim@307: } ingo@297: } ingo@297: return ""; ingo@297: } ingo@364: ingo@364: ingo@767: /** ingo@767: * Method to store the maximum range. Since we need to adjust the range of ingo@767: * each range axis, we have to memorize its range - each parameter uses an ingo@767: * own range axis. ingo@767: * ingo@767: * @param ranges Map where ranges of each axis will be stored in. ingo@767: * @param value A given value on an axis. ingo@767: * @param parameter Parameter name which belongs to value. ingo@767: */ ingo@656: protected void storeMaxRange(Map ranges, double value, String parameter) { ingo@364: Range range = null; ingo@364: ingo@364: range = ranges.containsKey(parameter) ingo@364: ? (Range) ranges.get(parameter) ingo@364: : new Range(value, value); ingo@364: ingo@364: double lower = range.getLowerBound(); ingo@364: double upper = range.getUpperBound(); ingo@364: ingo@364: lower = value < lower ? value : lower; ingo@364: upper = value > upper ? value : upper; ingo@364: ingo@364: ranges.put(parameter, new Range(lower, upper)); ingo@364: } ingo@297: } ingo@315: // vim:set ts=4 sw=4 si et sta sts=4 fenc=utf-8 :