/*! * This program is free software; you can redistribute it and/or modify it under the * terms of the GNU Lesser General Public License, version 2.1 as published by the Free Software * Foundation. * * You should have received a copy of the GNU Lesser General Public License along with this * program; if not, you can obtain a copy at http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html * or from the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU Lesser General Public License for more details. * * Copyright (c) 2002-2013 Pentaho Corporation.. All rights reserved. */ package org.pentaho.plugin.jfreereport.reportcharts.backport; import org.jfree.data.DefaultKeyedValues2D; import org.jfree.data.DomainInfo; import org.jfree.data.Range; import org.jfree.data.general.DatasetChangeEvent; import org.jfree.data.time.RegularTimePeriod; import org.jfree.data.time.TimePeriod; import org.jfree.data.time.TimePeriodAnchor; import org.jfree.data.xy.AbstractIntervalXYDataset; import org.jfree.data.xy.TableXYDataset; import org.jfree.util.PublicCloneable; import java.util.Calendar; import java.util.List; import java.util.Locale; import java.util.TimeZone; /** * A dataset for regular time periods that implements the {@link TableXYDataset} interface. Note that the {@link * TableXYDataset} interface requires all series to share the same set of x-values. When adding a new item <code>(x, * y)</code> to one series, all other series automatically get a new item <code>(x, null)</code> unless a non-null item * has already been specified. * * @see org.jfree.data.xy.TableXYDataset */ public class ExtTimeTableXYDataset extends AbstractIntervalXYDataset implements Cloneable, PublicCloneable, DomainInfo, TableXYDataset { /** * The data structure to store the values. Each column represents a series (elsewhere in JFreeChart rows are * typically used for series, but it doesn't matter that much since this data structure is private and symmetrical * anyway), each row contains values for the same {@link RegularTimePeriod} (the rows are sorted into ascending * order). */ private DefaultKeyedValues2D values; /** * A flag that indicates that the domain is 'points in time'. If this flag is true, only the x-value (and not the * x-interval) is used to determine the range of values in the domain. */ private boolean domainIsPointsInTime; /** * The point within each time period that is used for the X value when this collection is used as an {@link * org.jfree.data.xy.XYDataset}. This can be the start, middle or end of the time period. */ private TimePeriodAnchor xPosition; /** * A working calendar (to recycle) */ private Calendar workingCalendar; /** * Creates a new dataset. */ public ExtTimeTableXYDataset() { // defer argument checking this( TimeZone.getDefault(), Locale.getDefault() ); } /** * Creates a new dataset with the given time zone. * * @param zone the time zone to use (<code>null</code> not permitted). */ public ExtTimeTableXYDataset( final TimeZone zone ) { // defer argument checking this( zone, Locale.getDefault() ); } /** * Creates a new dataset with the given time zone and locale. * * @param zone the time zone to use (<code>null</code> not permitted). * @param locale the locale to use (<code>null</code> not permitted). */ public ExtTimeTableXYDataset( final TimeZone zone, final Locale locale ) { if ( zone == null ) { throw new IllegalArgumentException( "Null 'zone' argument." ); } if ( locale == null ) { throw new IllegalArgumentException( "Null 'locale' argument." ); } this.values = new DefaultKeyedValues2D( true ); this.workingCalendar = Calendar.getInstance( zone, locale ); this.xPosition = TimePeriodAnchor.START; } /** * Returns a flag that controls whether the domain is treated as 'points in time'. * <p/> * This flag is used when determining the max and min values for the domain. If true, then only the x-values are * considered for the max and min values. If false, then the start and end x-values will also be taken into * consideration. * * @return The flag. * @see #setDomainIsPointsInTime(boolean) */ public boolean getDomainIsPointsInTime() { return this.domainIsPointsInTime; } /** * Sets a flag that controls whether the domain is treated as 'points in time', or time periods. A {@link * DatasetChangeEvent} is sent to all registered listeners. * * @param flag the new value of the flag. * @see #getDomainIsPointsInTime() */ public void setDomainIsPointsInTime( final boolean flag ) { this.domainIsPointsInTime = flag; notifyListeners( new DatasetChangeEvent( this, this ) ); } /** * Returns the position within each time period that is used for the X value. * * @return The anchor position (never <code>null</code>). * @see #setXPosition(TimePeriodAnchor) */ public TimePeriodAnchor getXPosition() { return this.xPosition; } /** * Sets the position within each time period that is used for the X values, then sends a {@link DatasetChangeEvent} to * all registered listeners. * * @param anchor the anchor position (<code>null</code> not permitted). * @see #getXPosition() */ public void setXPosition( final TimePeriodAnchor anchor ) { if ( anchor == null ) { throw new IllegalArgumentException( "Null 'anchor' argument." ); } this.xPosition = anchor; notifyListeners( new DatasetChangeEvent( this, this ) ); } /** * Adds a new data item to the dataset and sends a {@link DatasetChangeEvent} to all registered listeners. * * @param period the time period. * @param y the value for this period. * @param seriesName the name of the series to add the value. * @see #remove(TimePeriod, Comparable) */ public void add( final TimePeriod period, final double y, final Comparable seriesName ) { add( period, new Double( y ), seriesName, true ); } /** * Adds a new data item to the dataset and, if requested, sends a {@link DatasetChangeEvent} to all registered * listeners. * * @param period the time period (<code>null</code> not permitted). * @param y the value for this period (<code>null</code> permitted). * @param seriesName the name of the series to add the value (<code>null</code> not permitted). * @param notify whether dataset listener are notified or not. * @see #remove(TimePeriod, Comparable, boolean) */ public void add( final TimePeriod period, final Number y, final Comparable seriesName, final boolean notify ) { // here's a quirk - the API has been defined in terms of a plain // TimePeriod, which cannot make use of the timezone and locale // specified in the constructor...so we only do the time zone // pegging if the period is an instanceof RegularTimePeriod if ( period instanceof RegularTimePeriod ) { final RegularTimePeriod p = (RegularTimePeriod) period; p.peg( this.workingCalendar ); } this.values.addValue( y, period, seriesName ); if ( notify ) { fireDatasetChanged(); } } /** * Removes an existing data item from the dataset. * * @param period the (existing!) time period of the value to remove (<code>null</code> not permitted). * @param seriesName the (existing!) series name to remove the value (<code>null</code> not permitted). * @see #add(TimePeriod, double, Comparable) */ public void remove( final TimePeriod period, final Comparable seriesName ) { remove( period, seriesName, true ); } /** * Removes an existing data item from the dataset and, if requested, sends a {@link DatasetChangeEvent} to all * registered listeners. * * @param period the (existing!) time period of the value to remove (<code>null</code> not permitted). * @param seriesName the (existing!) series name to remove the value (<code>null</code> not permitted). * @param notify whether dataset listener are notified or not. * @see #add(TimePeriod, double, Comparable) */ public void remove( final TimePeriod period, final Comparable seriesName, final boolean notify ) { this.values.removeValue( period, seriesName ); if ( notify ) { fireDatasetChanged(); } } /** * Removes all data items from the dataset and sends a {@link DatasetChangeEvent} to all registered listeners. * * @since 1.0.7 */ public void clear() { if ( this.values.getRowCount() > 0 ) { this.values.clear(); fireDatasetChanged(); } } /** * Returns the time period for the specified item. Bear in mind that all series share the same set of time periods. * * @param item the item index (0 <= i <= {@link #getItemCount()}). * @return The time period. */ public TimePeriod getTimePeriod( final int item ) { return (TimePeriod) this.values.getRowKey( item ); } /** * Returns the number of items in ALL series. * * @return The item count. */ public int getItemCount() { return this.values.getRowCount(); } /** * Returns the number of items in a series. This is the same value that is returned by {@link #getItemCount()} since * all series share the same x-values (time periods). * * @param series the series (zero-based index, ignored). * @return The number of items within the series. */ public int getItemCount( final int series ) { return getItemCount(); } /** * Returns the number of series in the dataset. * * @return The series count. */ public int getSeriesCount() { return this.values.getColumnCount(); } /** * Returns the key for a series. * * @param series the series (zero-based index). * @return The key for the series. */ public Comparable getSeriesKey( final int series ) { return this.values.getColumnKey( series ); } /** * Returns the x-value for an item within a series. The x-values may or may not be returned in ascending order, that * is up to the class implementing the interface. * * @param series the series (zero-based index). * @param item the item (zero-based index). * @return The x-value. */ public Number getX( final int series, final int item ) { return new Double( getXValue( series, item ) ); } /** * Returns the x-value (as a double primitive) for an item within a series. * * @param series the series index (zero-based). * @param item the item index (zero-based). * @return The value. */ public double getXValue( final int series, final int item ) { final TimePeriod period = (TimePeriod) this.values.getRowKey( item ); return getXValue( period ); } /** * Returns the starting X value for the specified series and item. * * @param series the series (zero-based index). * @param item the item within a series (zero-based index). * @return The starting X value for the specified series and item. * @see #getStartXValue(int, int) */ public Number getStartX( final int series, final int item ) { return new Double( getStartXValue( series, item ) ); } /** * Returns the start x-value (as a double primitive) for an item within a series. * * @param series the series index (zero-based). * @param item the item index (zero-based). * @return The value. */ public double getStartXValue( final int series, final int item ) { final TimePeriod period = (TimePeriod) this.values.getRowKey( item ); return period.getStart().getTime(); } /** * Returns the ending X value for the specified series and item. * * @param series the series (zero-based index). * @param item the item within a series (zero-based index). * @return The ending X value for the specified series and item. * @see #getEndXValue(int, int) */ public Number getEndX( final int series, final int item ) { return new Double( getEndXValue( series, item ) ); } /** * Returns the end x-value (as a double primitive) for an item within a series. * * @param series the series index (zero-based). * @param item the item index (zero-based). * @return The value. */ public double getEndXValue( final int series, final int item ) { final TimePeriod period = (TimePeriod) this.values.getRowKey( item ); return period.getEnd().getTime(); } /** * Returns the y-value for an item within a series. * * @param series the series (zero-based index). * @param item the item (zero-based index). * @return The y-value (possibly <code>null</code>). */ public Number getY( final int series, final int item ) { return this.values.getValue( item, series ); } /** * Returns the starting Y value for the specified series and item. * * @param series the series (zero-based index). * @param item the item within a series (zero-based index). * @return The starting Y value for the specified series and item. */ public Number getStartY( final int series, final int item ) { return getY( series, item ); } /** * Returns the ending Y value for the specified series and item. * * @param series the series (zero-based index). * @param item the item within a series (zero-based index). * @return The ending Y value for the specified series and item. */ public Number getEndY( final int series, final int item ) { return getY( series, item ); } /** * Returns the x-value for a time period. * * @param period the time period. * @return The x-value. */ private long getXValue( final TimePeriod period ) { long result = 0L; if ( this.xPosition == TimePeriodAnchor.START ) { result = period.getStart().getTime(); } else if ( this.xPosition == TimePeriodAnchor.MIDDLE ) { final long t0 = period.getStart().getTime(); final long t1 = period.getEnd().getTime(); result = t0 + ( t1 - t0 ) / 2L; } else if ( this.xPosition == TimePeriodAnchor.END ) { result = period.getEnd().getTime(); } return result; } /** * Returns the minimum x-value in the dataset. * * @param includeInterval a flag that determines whether or not the x-interval is taken into account. * @return The minimum value. */ public double getDomainLowerBound( final boolean includeInterval ) { double result = Double.NaN; final Range r = getDomainBounds( includeInterval ); if ( r != null ) { result = r.getLowerBound(); } return result; } /** * Returns the maximum x-value in the dataset. * * @param includeInterval a flag that determines whether or not the x-interval is taken into account. * @return The maximum value. */ public double getDomainUpperBound( final boolean includeInterval ) { double result = Double.NaN; final Range r = getDomainBounds( includeInterval ); if ( r != null ) { result = r.getUpperBound(); } return result; } /** * Returns the range of the values in this dataset's domain. * * @param includeInterval a flag that controls whether or not the x-intervals are taken into account. * @return The range. */ public Range getDomainBounds( final boolean includeInterval ) { final List keys = this.values.getRowKeys(); if ( keys.isEmpty() ) { return null; } final TimePeriod first = (TimePeriod) keys.get( 0 ); final TimePeriod last = (TimePeriod) keys.get( keys.size() - 1 ); if ( !includeInterval || this.domainIsPointsInTime ) { return new Range( getXValue( first ), getXValue( last ) ); } else { return new Range( first.getStart().getTime(), last.getEnd().getTime() ); } } /** * Tests this dataset for equality with an arbitrary object. * * @param obj the object (<code>null</code> permitted). * @return A boolean. */ public boolean equals( final Object obj ) { if ( obj == this ) { return true; } if ( !( obj instanceof ExtTimeTableXYDataset ) ) { return false; } final ExtTimeTableXYDataset that = (ExtTimeTableXYDataset) obj; if ( this.domainIsPointsInTime != that.domainIsPointsInTime ) { return false; } if ( this.xPosition != that.xPosition ) { return false; } if ( !this.workingCalendar.getTimeZone().equals( that.workingCalendar.getTimeZone() ) ) { return false; } if ( !this.values.equals( that.values ) ) { return false; } return true; } /** * Returns a clone of this dataset. * * @return A clone. * @throws CloneNotSupportedException if the dataset cannot be cloned. */ public Object clone() throws CloneNotSupportedException { final ExtTimeTableXYDataset clone = (ExtTimeTableXYDataset) super.clone(); clone.values = (DefaultKeyedValues2D) this.values.clone(); clone.workingCalendar = (Calendar) this.workingCalendar.clone(); return clone; } }