001    /* ===========================================================
002     * JFreeChart : a free chart library for the Java(tm) platform
003     * ===========================================================
004     *
005     * (C) Copyright 2000-2007, by Object Refinery Limited and Contributors.
006     *
007     * Project Info:  http://www.jfree.org/jfreechart/index.html
008     *
009     * This library is free software; you can redistribute it and/or modify it 
010     * under the terms of the GNU Lesser General Public License as published by 
011     * the Free Software Foundation; either version 2.1 of the License, or 
012     * (at your option) any later version.
013     *
014     * This library is distributed in the hope that it will be useful, but 
015     * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY 
016     * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public 
017     * License for more details.
018     *
019     * You should have received a copy of the GNU Lesser General Public
020     * License along with this library; if not, write to the Free Software
021     * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, 
022     * USA.  
023     *
024     * [Java is a trademark or registered trademark of Sun Microsystems, Inc. 
025     * in the United States and other countries.]
026     *
027     * ---------------------------
028     * ComparableObjectSeries.java
029     * ---------------------------
030     * (C) Copyright 2006, 2007, by Object Refinery Limited.
031     *
032     * Original Author:  David Gilbert (for Object Refinery Limited);
033     * Contributor(s):   -;
034     *
035     * Changes
036     * -------
037     * 19-Oct-2006 : New class (DG);
038     * 31-Oct-2007 : Implemented faster hashCode() (DG);
039     *
040     */
041    
042    package org.jfree.data;
043    
044    import java.io.Serializable;
045    import java.util.Collections;
046    import java.util.List;
047    
048    import org.jfree.data.general.Series;
049    import org.jfree.data.general.SeriesChangeEvent;
050    import org.jfree.data.general.SeriesException;
051    import org.jfree.util.ObjectUtilities;
052    
053    /**
054     * A (possibly ordered) list of (Comparable, Object) data items.
055     *
056     * @since 1.0.3
057     */
058    public class ComparableObjectSeries extends Series 
059            implements Cloneable, Serializable {
060        
061        /** Storage for the data items in the series. */
062        protected List data;
063    
064        /** The maximum number of items for the series. */
065        private int maximumItemCount = Integer.MAX_VALUE;
066    
067        /** A flag that controls whether the items are automatically sorted. */
068        private boolean autoSort;
069        
070        /** A flag that controls whether or not duplicate x-values are allowed. */
071        private boolean allowDuplicateXValues;
072    
073        /**
074         * Creates a new empty series.  By default, items added to the series will 
075         * be sorted into ascending order by x-value, and duplicate x-values will 
076         * be allowed (these defaults can be modified with another constructor.
077         *
078         * @param key  the series key (<code>null</code> not permitted).
079         */
080        public ComparableObjectSeries(Comparable key) {
081            this(key, true, true);
082        }
083        
084        /**
085         * Constructs a new series that contains no data.  You can specify 
086         * whether or not duplicate x-values are allowed for the series.
087         *
088         * @param key  the series key (<code>null</code> not permitted).
089         * @param autoSort  a flag that controls whether or not the items in the 
090         *                  series are sorted.
091         * @param allowDuplicateXValues  a flag that controls whether duplicate 
092         *                               x-values are allowed.
093         */
094        public ComparableObjectSeries(Comparable key, boolean autoSort, 
095                boolean allowDuplicateXValues) {
096            super(key);
097            this.data = new java.util.ArrayList();
098            this.autoSort = autoSort;
099            this.allowDuplicateXValues = allowDuplicateXValues;
100        }
101    
102        /**
103         * Returns the flag that controls whether the items in the series are 
104         * automatically sorted.  There is no setter for this flag, it must be 
105         * defined in the series constructor.
106         * 
107         * @return A boolean.
108         */
109        public boolean getAutoSort() {
110            return this.autoSort;
111        }
112        
113        /**
114         * Returns a flag that controls whether duplicate x-values are allowed.  
115         * This flag can only be set in the constructor.
116         *
117         * @return A boolean.
118         */
119        public boolean getAllowDuplicateXValues() {
120            return this.allowDuplicateXValues;
121        }
122    
123        /**
124         * Returns the number of items in the series.
125         *
126         * @return The item count.
127         */
128        public int getItemCount() {
129            return this.data.size();
130        }
131    
132        /**
133         * Returns the maximum number of items that will be retained in the series.
134         * The default value is <code>Integer.MAX_VALUE</code>.
135         *
136         * @return The maximum item count.
137         * @see #setMaximumItemCount(int)
138         */
139        public int getMaximumItemCount() {
140            return this.maximumItemCount;
141        }
142    
143        /**
144         * Sets the maximum number of items that will be retained in the series.  
145         * If you add a new item to the series such that the number of items will 
146         * exceed the maximum item count, then the first element in the series is 
147         * automatically removed, ensuring that the maximum item count is not 
148         * exceeded.
149         * <p>
150         * Typically this value is set before the series is populated with data,
151         * but if it is applied later, it may cause some items to be removed from
152         * the series (in which case a {@link SeriesChangeEvent} will be sent to
153         * all registered listeners.
154         *
155         * @param maximum  the maximum number of items for the series.
156         */
157        public void setMaximumItemCount(int maximum) {
158            this.maximumItemCount = maximum;
159            boolean dataRemoved = false;
160            while (this.data.size() > maximum) {
161                this.data.remove(0);   
162                dataRemoved = true;
163            }
164            if (dataRemoved) {
165                fireSeriesChanged();
166            }
167        }
168        
169        /**
170         * Adds new data to the series and sends a {@link SeriesChangeEvent} to 
171         * all registered listeners.
172         * <P>
173         * Throws an exception if the x-value is a duplicate AND the 
174         * allowDuplicateXValues flag is false.
175         *
176         * @param x  the x-value (<code>null</code> not permitted).
177         * @param y  the y-value (<code>null</code> permitted).
178         */
179        protected void add(Comparable x, Object y) {
180            // argument checking delegated...
181            add(x, y, true);
182        }
183        
184        /**
185         * Adds new data to the series and, if requested, sends a 
186         * {@link SeriesChangeEvent} to all registered listeners.
187         * <P>
188         * Throws an exception if the x-value is a duplicate AND the 
189         * allowDuplicateXValues flag is false.
190         *
191         * @param x  the x-value (<code>null</code> not permitted).
192         * @param y  the y-value (<code>null</code> permitted).
193         * @param notify  a flag the controls whether or not a 
194         *                {@link SeriesChangeEvent} is sent to all registered 
195         *                listeners.
196         */
197        protected void add(Comparable x, Object y, boolean notify) {
198            // delegate argument checking to XYDataItem...
199            ComparableObjectItem item = new ComparableObjectItem(x, y);
200            add(item, notify);
201        }
202    
203        /**
204         * Adds a data item to the series and, if requested, sends a 
205         * {@link SeriesChangeEvent} to all registered listeners.
206         *
207         * @param item  the (x, y) item (<code>null</code> not permitted).
208         * @param notify  a flag that controls whether or not a 
209         *                {@link SeriesChangeEvent} is sent to all registered 
210         *                listeners.
211         */
212        protected void add(ComparableObjectItem item, boolean notify) {
213    
214            if (item == null) {
215                throw new IllegalArgumentException("Null 'item' argument.");
216            }
217    
218            if (this.autoSort) {
219                int index = Collections.binarySearch(this.data, item);
220                if (index < 0) {
221                    this.data.add(-index - 1, item);
222                }
223                else {
224                    if (this.allowDuplicateXValues) {
225                        // need to make sure we are adding *after* any duplicates
226                        int size = this.data.size();
227                        while (index < size 
228                               && item.compareTo(this.data.get(index)) == 0) {
229                            index++;
230                        }
231                        if (index < this.data.size()) {
232                            this.data.add(index, item);
233                        }
234                        else {
235                            this.data.add(item);
236                        }
237                    }
238                    else {
239                        throw new SeriesException("X-value already exists.");
240                    }
241                }
242            }
243            else {
244                if (!this.allowDuplicateXValues) {
245                    // can't allow duplicate values, so we need to check whether
246                    // there is an item with the given x-value already
247                    int index = indexOf(item.getComparable());
248                    if (index >= 0) {
249                        throw new SeriesException("X-value already exists.");      
250                    }
251                }
252                this.data.add(item);
253            }
254            if (getItemCount() > this.maximumItemCount) {
255                this.data.remove(0);
256            }                    
257            if (notify) {
258                fireSeriesChanged();
259            }
260        }
261        
262        /**
263         * Returns the index of the item with the specified x-value, or a negative 
264         * index if the series does not contain an item with that x-value.  Be 
265         * aware that for an unsorted series, the index is found by iterating 
266         * through all items in the series.
267         * 
268         * @param x  the x-value (<code>null</code> not permitted).
269         * 
270         * @return The index.
271         */
272        public int indexOf(Comparable x) {
273            if (this.autoSort) {
274                return Collections.binarySearch(this.data, new ComparableObjectItem(
275                        x, null));   
276            }
277            else {
278                for (int i = 0; i < this.data.size(); i++) {
279                    ComparableObjectItem item = (ComparableObjectItem) 
280                            this.data.get(i);
281                    if (item.getComparable().equals(x)) {
282                        return i;   
283                    }
284                }
285                return -1;
286            }
287        } 
288    
289        /**
290         * Updates an item in the series.
291         * 
292         * @param x  the x-value (<code>null</code> not permitted).
293         * @param y  the y-value (<code>null</code> permitted).
294         * 
295         * @throws SeriesException if there is no existing item with the specified
296         *         x-value.
297         */
298        protected void update(Comparable x, Object y) {
299            int index = indexOf(x);
300            if (index < 0) {
301                throw new SeriesException("No observation for x = " + x);
302            }
303            else {
304                ComparableObjectItem item = getDataItem(index);
305                item.setObject(y);
306                fireSeriesChanged();
307            }
308        }
309    
310        /**
311         * Updates the value of an item in the series and sends a 
312         * {@link SeriesChangeEvent} to all registered listeners.
313         * 
314         * @param index  the item (zero based index).
315         * @param y  the new value (<code>null</code> permitted).
316         */
317        protected void updateByIndex(int index, Object y) {
318            ComparableObjectItem item = getDataItem(index);
319            item.setObject(y);
320            fireSeriesChanged();
321        }
322        
323        /**
324         * Return the data item with the specified index.
325         *
326         * @param index  the index.
327         *
328         * @return The data item with the specified index.
329         */
330        protected ComparableObjectItem getDataItem(int index) {
331            return (ComparableObjectItem) this.data.get(index);
332        }
333    
334        /**
335         * Deletes a range of items from the series and sends a 
336         * {@link SeriesChangeEvent} to all registered listeners.
337         *
338         * @param start  the start index (zero-based).
339         * @param end  the end index (zero-based).
340         */
341        protected void delete(int start, int end) {
342            for (int i = start; i <= end; i++) {
343                this.data.remove(start);
344            }
345            fireSeriesChanged();
346        }
347        
348        /**
349         * Removes all data items from the series.
350         */
351        protected void clear() {
352            if (this.data.size() > 0) {
353                this.data.clear();
354                fireSeriesChanged();
355            }
356        }
357    
358        /**
359         * Removes the item at the specified index and sends a 
360         * {@link SeriesChangeEvent} to all registered listeners.
361         * 
362         * @param index  the index.
363         * 
364         * @return The item removed.
365         */
366        protected ComparableObjectItem remove(int index) {
367            ComparableObjectItem result = (ComparableObjectItem) this.data.remove(
368                    index);
369            fireSeriesChanged();
370            return result;
371        }
372        
373        /**
374         * Removes the item with the specified x-value and sends a 
375         * {@link SeriesChangeEvent} to all registered listeners.
376         * 
377         * @param x  the x-value.
378    
379         * @return The item removed.
380         */
381        public ComparableObjectItem remove(Comparable x) {
382            return remove(indexOf(x));
383        }
384        
385        /**
386         * Tests this series for equality with an arbitrary object.
387         *
388         * @param obj  the object to test against for equality 
389         *             (<code>null</code> permitted).
390         *
391         * @return A boolean.
392         */
393        public boolean equals(Object obj) {
394            if (obj == this) {
395                return true;
396            }
397            if (!(obj instanceof ComparableObjectSeries)) {
398                return false;
399            }
400            if (!super.equals(obj)) {
401                return false;
402            }
403            ComparableObjectSeries that = (ComparableObjectSeries) obj;
404            if (this.maximumItemCount != that.maximumItemCount) {
405                return false;
406            }
407            if (this.autoSort != that.autoSort) {
408                return false;
409            }
410            if (this.allowDuplicateXValues != that.allowDuplicateXValues) {
411                return false;
412            }
413            if (!ObjectUtilities.equal(this.data, that.data)) {
414                return false;
415            }
416            return true;
417        }
418        
419        /**
420         * Returns a hash code.
421         * 
422         * @return A hash code.
423         */
424        public int hashCode() {
425            int result = super.hashCode();
426            // it is too slow to look at every data item, so let's just look at
427            // the first, middle and last items...
428            int count = getItemCount();
429            if (count > 0) {
430                ComparableObjectItem item = getDataItem(0);
431                result = 29 * result + item.hashCode();
432            }
433            if (count > 1) {
434                ComparableObjectItem item = getDataItem(count - 1);
435                result = 29 * result + item.hashCode();
436            }
437            if (count > 2) {
438                ComparableObjectItem item = getDataItem(count / 2);
439                result = 29 * result + item.hashCode();
440            }
441            result = 29 * result + this.maximumItemCount;
442            result = 29 * result + (this.autoSort ? 1 : 0);
443            result = 29 * result + (this.allowDuplicateXValues ? 1 : 0);
444            return result;
445        }
446        
447    }