- /* ===========================================================
- * JFreeChart : a free chart library for the Java(tm) platform
- * ===========================================================
- *
- * (C) Copyright 2000-2005, by Object Refinery Limited and Contributors.
- *
- * Project Info: http://www.jfree.org/jfreechart/index.html
- *
- * This library is free software; you can redistribute it and/or modify it under the terms
- * of the GNU Lesser General Public License as published by the Free Software Foundation;
- * either version 2.1 of the License, or (at your option) any later version.
- *
- * This library 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.
- *
- * You should have received a copy of the GNU Lesser General Public License along with this
- * library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
- * Boston, MA 02111-1307, USA.
- *
- * [Java is a trademark or registered trademark of Sun Microsystems, Inc.
- * in the United States and other countries.]
- *
- * -------------------------
- * DefaultKeyedValues2D.java
- * -------------------------
- * (C) Copyright 2002-2005, by Object Refinery Limited.
- *
- * Original Author: David Gilbert (for Object Refinery Limited);
- * Contributor(s): Andreas Schroeder;
- *
- * $Id: DefaultKeyedValues2D.java,v 1.5 2005/01/12 14:54:58 mungady Exp $
- *
- * Changes
- * -------
- * 28-Oct-2002 : Version 1 (DG);
- * 21-Jan-2003 : Updated Javadocs (DG);
- * 13-Mar-2003 : Implemented Serializable (DG);
- * 18-Aug-2003 : Implemented Cloneable (DG);
- * 31-Mar-2004 : Made the rows optionally sortable by a flag (AS);
- * 01-Apr-2004 : Implemented remove method (AS);
- * 05-Apr-2004 : Added clear() method (DG);
- * 15-Sep-2004 : Fixed clone() method (DG);
- * 12-Jan-2005 : Fixed bug in getValue() method (DG);
- *
- */
- package org.jfree.data;
- import java.io.Serializable;
- import java.util.Collections;
- import java.util.Iterator;
- import java.util.List;
- import org.jfree.util.ObjectUtilities;
- /**
- * A data structure that stores zero, one or many values, where each value
- * is associated with two keys (a 'row' key and a 'column' key). The keys
- * should be (a) instances of {@link Comparable} and (b) immutable.
- */
- public class DefaultKeyedValues2D implements KeyedValues2D, Cloneable, Serializable {
- /** The row keys. */
- private List rowKeys;
- /** The column keys. */
- private List columnKeys;
- /** The row data. */
- private List rows;
- /** If the row keys should be sorted by their comparable order. */
- private boolean sortRowKeys;
- /**
- * Creates a new instance (initially empty).
- */
- public DefaultKeyedValues2D() {
- this(false);
- }
- /**
- * Creates a new instance (initially empty).
- *
- * @param sortRowKeys if the row keys should be sorted.
- */
- public DefaultKeyedValues2D(boolean sortRowKeys) {
- this.rowKeys = new java.util.ArrayList();
- this.columnKeys = new java.util.ArrayList();
- this.rows = new java.util.ArrayList();
- this.sortRowKeys = sortRowKeys;
- }
- /**
- * Returns the row count.
- *
- * @return The row count.
- */
- public int getRowCount() {
- return this.rowKeys.size();
- }
- /**
- * Returns the column count.
- *
- * @return The column count.
- */
- public int getColumnCount() {
- return this.columnKeys.size();
- }
- /**
- * Returns the value for a given row and column.
- *
- * @param row the row index.
- * @param column the column index.
- *
- * @return The value.
- */
- public Number getValue(int row, int column) {
- Number result = null;
- DefaultKeyedValues rowData = (DefaultKeyedValues) this.rows.get(row);
- if (rowData != null) {
- Comparable columnKey = (Comparable) this.columnKeys.get(column);
- // the row may not have an entry for this key, in which case the return value is null
- int index = rowData.getIndex(columnKey);
- if (index >= 0) {
- result = rowData.getValue(index);
- }
- }
- return result;
- }
- /**
- * Returns the key for a given row.
- *
- * @param row the row index (zero based).
- *
- * @return The row index.
- */
- public Comparable getRowKey(int row) {
- return (Comparable) this.rowKeys.get(row);
- }
- /**
- * Returns the row index for a given key.
- *
- * @param key the key (<code>null</code> not permitted).
- *
- * @return The row index.
- */
- public int getRowIndex(Comparable key) {
- if (key == null) {
- throw new IllegalArgumentException("Null 'key' argument.");
- }
- if (this.sortRowKeys) {
- return Collections.binarySearch(this.rowKeys, key);
- }
- else {
- return this.rowKeys.indexOf(key);
- }
- }
- /**
- * Returns the row keys.
- *
- * @return The row keys.
- */
- public List getRowKeys() {
- return Collections.unmodifiableList(this.rowKeys);
- }
- /**
- * Returns the key for a given column.
- *
- * @param column the column.
- *
- * @return The key.
- */
- public Comparable getColumnKey(int column) {
- return (Comparable) this.columnKeys.get(column);
- }
- /**
- * Returns the column index for a given key.
- *
- * @param key the key (<code>null</code> not permitted).
- *
- * @return the column index.
- */
- public int getColumnIndex(Comparable key) {
- if (key == null) {
- throw new IllegalArgumentException("Null 'key' argument.");
- }
- return this.columnKeys.indexOf(key);
- }
- /**
- * Returns the column keys.
- *
- * @return The column keys.
- */
- public List getColumnKeys() {
- return Collections.unmodifiableList(this.columnKeys);
- }
- /**
- * Returns the value for the given row and column keys.
- *
- * @param rowKey the row key (<code>null</code> not permitted).
- * @param columnKey the column key (<code>null</code> not permitted).
- *
- * @return The value.
- */
- public Number getValue(Comparable rowKey, Comparable columnKey) {
- if (rowKey == null) {
- throw new IllegalArgumentException("Null 'rowKey' argument.");
- }
- if (columnKey == null) {
- throw new IllegalArgumentException("Null 'columnKey' argument.");
- }
- Number result = null;
- int row = getRowIndex(rowKey);
- if (row >= 0) {
- DefaultKeyedValues rowData = (DefaultKeyedValues) this.rows.get(row);
- result = rowData.getValue(columnKey);
- }
- return result;
- }
- /**
- * Adds a value to the table. Performs the same function as
- * #setValue(Number, Comparable, Comparable).
- *
- * @param value the value (<code>null</code> permitted).
- * @param rowKey the row key (<code>null</code> not permitted).
- * @param columnKey the column key (<code>null</code> not permitted).
- */
- public void addValue(Number value, Comparable rowKey, Comparable columnKey) {
- // defer argument checking
- setValue(value, rowKey, columnKey);
- }
- /**
- * Adds or updates a value.
- *
- * @param value the value (<code>null</code> permitted).
- * @param rowKey the row key (<code>null</code> not permitted).
- * @param columnKey the column key (<code>null</code> not permitted).
- */
- public void setValue(Number value, Comparable rowKey, Comparable columnKey) {
- DefaultKeyedValues row;
- int rowIndex = getRowIndex(rowKey);
- if (rowIndex >= 0) {
- row = (DefaultKeyedValues) this.rows.get(rowIndex);
- }
- else {
- row = new DefaultKeyedValues();
- if (this.sortRowKeys) {
- rowIndex = -rowIndex - 1;
- this.rowKeys.add(rowIndex, rowKey);
- this.rows.add(rowIndex, row);
- }
- else {
- this.rowKeys.add(rowKey);
- this.rows.add(row);
- }
- }
- row.setValue(columnKey, value);
- int columnIndex = this.columnKeys.indexOf(columnKey);
- if (columnIndex < 0) {
- this.columnKeys.add(columnKey);
- }
- }
- /**
- * Removes a value.
- *
- * @param rowKey the row key (<code>null</code> not permitted).
- * @param columnKey the column key (<code>null</code> not permitted).
- */
- public void removeValue(Comparable rowKey, Comparable columnKey) {
- setValue(null, rowKey, columnKey);
- // 1. check whether the row is now empty.
- boolean allNull = true;
- int rowIndex = getRowIndex(rowKey);
- DefaultKeyedValues row = (DefaultKeyedValues) this.rows.get(rowIndex);
- for (int item = 0, itemCount = row.getItemCount(); item < itemCount; item++) {
- if (row.getValue(item) != null) {
- allNull = false;
- break;
- }
- }
- if (allNull) {
- this.rowKeys.remove(rowIndex);
- this.rows.remove(rowIndex);
- }
- // 2. check whether the column is now empty.
- allNull = true;
- int columnIndex = getColumnIndex(columnKey);
- for (int item = 0, itemCount = this.rows.size(); item < itemCount; item++) {
- row = (DefaultKeyedValues) this.rows.get(item);
- if (row.getValue(columnIndex) != null) {
- allNull = false;
- break;
- }
- }
- if (allNull) {
- for (int item = 0, itemCount = this.rows.size(); item < itemCount; item++) {
- row = (DefaultKeyedValues) this.rows.get(item);
- row.removeValue(columnIndex);
- }
- this.columnKeys.remove(columnIndex);
- }
- }
- /**
- * Removes a row.
- *
- * @param rowIndex the row index.
- */
- public void removeRow(int rowIndex) {
- this.rowKeys.remove(rowIndex);
- this.rows.remove(rowIndex);
- }
- /**
- * Removes a row.
- *
- * @param rowKey the row key.
- */
- public void removeRow(Comparable rowKey) {
- removeRow(getRowIndex(rowKey));
- }
- /**
- * Removes a column.
- *
- * @param columnIndex the column index.
- */
- public void removeColumn(int columnIndex) {
- Comparable columnKey = getColumnKey(columnIndex);
- removeColumn(columnKey);
- }
- /**
- * Removes a column.
- *
- * @param columnKey the column key (<code>null</code> not permitted).
- */
- public void removeColumn(Comparable columnKey) {
- Iterator iterator = this.rows.iterator();
- while (iterator.hasNext()) {
- DefaultKeyedValues rowData = (DefaultKeyedValues) iterator.next();
- rowData.removeValue(columnKey);
- }
- this.columnKeys.remove(columnKey);
- }
- /**
- * Clears all the data and associated keys.
- */
- public void clear() {
- this.rowKeys.clear();
- this.columnKeys.clear();
- this.rows.clear();
- }
- /**
- * Tests if this object is equal to another.
- *
- * @param o the other object (<code>null</code> permitted).
- *
- * @return A boolean.
- */
- public boolean equals(Object o) {
- if (o == null) {
- return false;
- }
- if (o == this) {
- return true;
- }
- if (!(o instanceof KeyedValues2D)) {
- return false;
- }
- KeyedValues2D kv2D = (KeyedValues2D) o;
- if (!getRowKeys().equals(kv2D.getRowKeys())) {
- return false;
- }
- if (!getColumnKeys().equals(kv2D.getColumnKeys())) {
- return false;
- }
- int rowCount = getRowCount();
- if (rowCount != kv2D.getRowCount()) {
- return false;
- }
- int colCount = getColumnCount();
- if (colCount != kv2D.getColumnCount()) {
- return false;
- }
- for (int r = 0; r < rowCount; r++) {
- for (int c = 0; c < colCount; c++) {
- Number v1 = getValue(r, c);
- Number v2 = kv2D.getValue(r, c);
- if (v1 == null) {
- if (v2 != null) {
- return false;
- }
- }
- else {
- if (!v1.equals(v2)) {
- return false;
- }
- }
- }
- }
- return true;
- }
- /**
- * Returns a hash code.
- *
- * @return A hash code.
- */
- public int hashCode() {
- int result;
- result = this.rowKeys.hashCode();
- result = 29 * result + this.columnKeys.hashCode();
- result = 29 * result + this.rows.hashCode();
- return result;
- }
- /**
- * Returns a clone.
- *
- * @return A clone.
- *
- * @throws CloneNotSupportedException this class will not throw this exception, but subclasses
- * (if any) might.
- */
- public Object clone() throws CloneNotSupportedException {
- DefaultKeyedValues2D clone = (DefaultKeyedValues2D) super.clone();
- // for the keys, a shallow copy should be fine because keys
- // should be immutable...
- clone.columnKeys = new java.util.ArrayList(this.columnKeys);
- clone.rowKeys = new java.util.ArrayList(this.rowKeys);
- // but the row data requires a deep copy
- clone.rows = (List) ObjectUtilities.deepClone(this.rows);
- return clone;
- }
- }