/* |
|
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. |
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
|
* |
|
* This code is free software; you can redistribute it and/or modify it |
|
* under the terms of the GNU General Public License version 2 only, as |
|
* published by the Free Software Foundation. Oracle designates this |
|
* particular file as subject to the "Classpath" exception as provided |
|
* by Oracle in the LICENSE file that accompanied this code. |
|
* |
|
* This code 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 General Public License |
|
* version 2 for more details (a copy is included in the LICENSE file that |
|
* accompanied this code). |
|
* |
|
* You should have received a copy of the GNU General Public License version |
|
* 2 along with this work; if not, write to the Free Software Foundation, |
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
|
* |
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
|
* or visit www.oracle.com if you need additional information or have any |
|
* questions. |
|
*/ |
|
package javax.swing; |
|
import javax.swing.event.*; |
|
import java.io.Serializable; |
|
import java.util.EventListener; |
|
/** |
|
* A generic implementation of BoundedRangeModel. |
|
* <p> |
|
* <strong>Warning:</strong> |
|
* Serialized objects of this class will not be compatible with |
|
* future Swing releases. The current serialization support is |
|
* appropriate for short term storage or RMI between applications running |
|
* the same version of Swing. As of 1.4, support for long term storage |
|
* of all JavaBeans™ |
|
* has been added to the <code>java.beans</code> package. |
|
* Please see {@link java.beans.XMLEncoder}. |
|
* |
|
* @author David Kloba |
|
* @author Hans Muller |
|
* @see BoundedRangeModel |
|
*/ |
|
public class DefaultBoundedRangeModel implements BoundedRangeModel, Serializable |
|
{ |
|
/** |
|
* Only one <code>ChangeEvent</code> is needed per model instance since the |
|
* event's only (read-only) state is the source property. The source |
|
* of events generated here is always "this". |
|
*/ |
|
protected transient ChangeEvent changeEvent = null; |
|
/** The listeners waiting for model changes. */ |
|
protected EventListenerList listenerList = new EventListenerList(); |
|
private int value = 0; |
|
private int extent = 0; |
|
private int min = 0; |
|
private int max = 100; |
|
private boolean isAdjusting = false; |
|
/** |
|
* Initializes all of the properties with default values. |
|
* Those values are: |
|
* <ul> |
|
* <li><code>value</code> = 0 |
|
* <li><code>extent</code> = 0 |
|
* <li><code>minimum</code> = 0 |
|
* <li><code>maximum</code> = 100 |
|
* <li><code>adjusting</code> = false |
|
* </ul> |
|
*/ |
|
public DefaultBoundedRangeModel() { |
|
} |
|
/** |
|
* Initializes value, extent, minimum and maximum. Adjusting is false. |
|
* Throws an <code>IllegalArgumentException</code> if the following |
|
* constraints aren't satisfied: |
|
* <pre> |
|
* min <= value <= value+extent <= max |
|
* </pre> |
|
*/ |
|
public DefaultBoundedRangeModel(int value, int extent, int min, int max) |
|
{ |
|
if ((max >= min) && |
|
(value >= min) && |
|
((value + extent) >= value) && |
|
((value + extent) <= max)) { |
|
this.value = value; |
|
this.extent = extent; |
|
this.min = min; |
|
this.max = max; |
|
} |
|
else { |
|
throw new IllegalArgumentException("invalid range properties"); |
|
} |
|
} |
|
/** |
|
* Returns the model's current value. |
|
* @return the model's current value |
|
* @see #setValue |
|
* @see BoundedRangeModel#getValue |
|
*/ |
|
public int getValue() { |
|
return value; |
|
} |
|
/** |
|
* Returns the model's extent. |
|
* @return the model's extent |
|
* @see #setExtent |
|
* @see BoundedRangeModel#getExtent |
|
*/ |
|
public int getExtent() { |
|
return extent; |
|
} |
|
/** |
|
* Returns the model's minimum. |
|
* @return the model's minimum |
|
* @see #setMinimum |
|
* @see BoundedRangeModel#getMinimum |
|
*/ |
|
public int getMinimum() { |
|
return min; |
|
} |
|
/** |
|
* Returns the model's maximum. |
|
* @return the model's maximum |
|
* @see #setMaximum |
|
* @see BoundedRangeModel#getMaximum |
|
*/ |
|
public int getMaximum() { |
|
return max; |
|
} |
|
/** |
|
* Sets the current value of the model. For a slider, that |
|
* determines where the knob appears. Ensures that the new |
|
* value, <I>n</I> falls within the model's constraints: |
|
* <pre> |
|
* minimum <= value <= value+extent <= maximum |
|
* </pre> |
|
* |
|
* @see BoundedRangeModel#setValue |
|
*/ |
|
public void setValue(int n) { |
|
n = Math.min(n, Integer.MAX_VALUE - extent); |
|
int newValue = Math.max(n, min); |
|
if (newValue + extent > max) { |
|
newValue = max - extent; |
|
} |
|
setRangeProperties(newValue, extent, min, max, isAdjusting); |
|
} |
|
/** |
|
* Sets the extent to <I>n</I> after ensuring that <I>n</I> |
|
* is greater than or equal to zero and falls within the model's |
|
* constraints: |
|
* <pre> |
|
* minimum <= value <= value+extent <= maximum |
|
* </pre> |
|
* @see BoundedRangeModel#setExtent |
|
*/ |
|
public void setExtent(int n) { |
|
int newExtent = Math.max(0, n); |
|
if(value + newExtent > max) { |
|
newExtent = max - value; |
|
} |
|
setRangeProperties(value, newExtent, min, max, isAdjusting); |
|
} |
|
/** |
|
* Sets the minimum to <I>n</I> after ensuring that <I>n</I> |
|
* that the other three properties obey the model's constraints: |
|
* <pre> |
|
* minimum <= value <= value+extent <= maximum |
|
* </pre> |
|
* @see #getMinimum |
|
* @see BoundedRangeModel#setMinimum |
|
*/ |
|
public void setMinimum(int n) { |
|
int newMax = Math.max(n, max); |
|
int newValue = Math.max(n, value); |
|
int newExtent = Math.min(newMax - newValue, extent); |
|
setRangeProperties(newValue, newExtent, n, newMax, isAdjusting); |
|
} |
|
/** |
|
* Sets the maximum to <I>n</I> after ensuring that <I>n</I> |
|
* that the other three properties obey the model's constraints: |
|
* <pre> |
|
* minimum <= value <= value+extent <= maximum |
|
* </pre> |
|
* @see BoundedRangeModel#setMaximum |
|
*/ |
|
public void setMaximum(int n) { |
|
int newMin = Math.min(n, min); |
|
int newExtent = Math.min(n - newMin, extent); |
|
int newValue = Math.min(n - newExtent, value); |
|
setRangeProperties(newValue, newExtent, newMin, n, isAdjusting); |
|
} |
|
/** |
|
* Sets the <code>valueIsAdjusting</code> property. |
|
* |
|
* @see #getValueIsAdjusting |
|
* @see #setValue |
|
* @see BoundedRangeModel#setValueIsAdjusting |
|
*/ |
|
public void setValueIsAdjusting(boolean b) { |
|
setRangeProperties(value, extent, min, max, b); |
|
} |
|
/** |
|
* Returns true if the value is in the process of changing |
|
* as a result of actions being taken by the user. |
|
* |
|
* @return the value of the <code>valueIsAdjusting</code> property |
|
* @see #setValue |
|
* @see BoundedRangeModel#getValueIsAdjusting |
|
*/ |
|
public boolean getValueIsAdjusting() { |
|
return isAdjusting; |
|
} |
|
/** |
|
* Sets all of the <code>BoundedRangeModel</code> properties after forcing |
|
* the arguments to obey the usual constraints: |
|
* <pre> |
|
* minimum <= value <= value+extent <= maximum |
|
* </pre> |
|
* <p> |
|
* At most, one <code>ChangeEvent</code> is generated. |
|
* |
|
* @see BoundedRangeModel#setRangeProperties |
|
* @see #setValue |
|
* @see #setExtent |
|
* @see #setMinimum |
|
* @see #setMaximum |
|
* @see #setValueIsAdjusting |
|
*/ |
|
public void setRangeProperties(int newValue, int newExtent, int newMin, int newMax, boolean adjusting) |
|
{ |
|
if (newMin > newMax) { |
|
newMin = newMax; |
|
} |
|
if (newValue > newMax) { |
|
newMax = newValue; |
|
} |
|
if (newValue < newMin) { |
|
newMin = newValue; |
|
} |
|
/* Convert the addends to long so that extent can be |
|
* Integer.MAX_VALUE without rolling over the sum. |
|
* A JCK test covers this, see bug 4097718. |
|
*/ |
|
if (((long)newExtent + (long)newValue) > newMax) { |
|
newExtent = newMax - newValue; |
|
} |
|
if (newExtent < 0) { |
|
newExtent = 0; |
|
} |
|
boolean isChange = |
|
(newValue != value) || |
|
(newExtent != extent) || |
|
(newMin != min) || |
|
(newMax != max) || |
|
(adjusting != isAdjusting); |
|
if (isChange) { |
|
value = newValue; |
|
extent = newExtent; |
|
min = newMin; |
|
max = newMax; |
|
isAdjusting = adjusting; |
|
fireStateChanged(); |
|
} |
|
} |
|
/** |
|
* Adds a <code>ChangeListener</code>. The change listeners are run each |
|
* time any one of the Bounded Range model properties changes. |
|
* |
|
* @param l the ChangeListener to add |
|
* @see #removeChangeListener |
|
* @see BoundedRangeModel#addChangeListener |
|
*/ |
|
public void addChangeListener(ChangeListener l) { |
|
listenerList.add(ChangeListener.class, l); |
|
} |
|
/** |
|
* Removes a <code>ChangeListener</code>. |
|
* |
|
* @param l the <code>ChangeListener</code> to remove |
|
* @see #addChangeListener |
|
* @see BoundedRangeModel#removeChangeListener |
|
*/ |
|
public void removeChangeListener(ChangeListener l) { |
|
listenerList.remove(ChangeListener.class, l); |
|
} |
|
/** |
|
* Returns an array of all the change listeners |
|
* registered on this <code>DefaultBoundedRangeModel</code>. |
|
* |
|
* @return all of this model's <code>ChangeListener</code>s |
|
* or an empty |
|
* array if no change listeners are currently registered |
|
* |
|
* @see #addChangeListener |
|
* @see #removeChangeListener |
|
* |
|
* @since 1.4 |
|
*/ |
|
public ChangeListener[] getChangeListeners() { |
|
return listenerList.getListeners(ChangeListener.class); |
|
} |
|
/** |
|
* Runs each <code>ChangeListener</code>'s <code>stateChanged</code> method. |
|
* |
|
* @see #setRangeProperties |
|
* @see EventListenerList |
|
*/ |
|
protected void fireStateChanged() |
|
{ |
|
Object[] listeners = listenerList.getListenerList(); |
|
for (int i = listeners.length - 2; i >= 0; i -=2 ) { |
|
if (listeners[i] == ChangeListener.class) { |
|
if (changeEvent == null) { |
|
changeEvent = new ChangeEvent(this); |
|
} |
|
((ChangeListener)listeners[i+1]).stateChanged(changeEvent); |
|
} |
|
} |
|
} |
|
/** |
|
* Returns a string that displays all of the |
|
* <code>BoundedRangeModel</code> properties. |
|
*/ |
|
public String toString() { |
|
String modelString = |
|
"value=" + getValue() + ", " + |
|
"extent=" + getExtent() + ", " + |
|
"min=" + getMinimum() + ", " + |
|
"max=" + getMaximum() + ", " + |
|
"adj=" + getValueIsAdjusting(); |
|
return getClass().getName() + "[" + modelString + "]"; |
|
} |
|
/** |
|
* Returns an array of all the objects currently registered as |
|
* <code><em>Foo</em>Listener</code>s |
|
* upon this model. |
|
* <code><em>Foo</em>Listener</code>s |
|
* are registered using the <code>add<em>Foo</em>Listener</code> method. |
|
* <p> |
|
* You can specify the <code>listenerType</code> argument |
|
* with a class literal, such as <code><em>Foo</em>Listener.class</code>. |
|
* For example, you can query a <code>DefaultBoundedRangeModel</code> |
|
* instance <code>m</code> |
|
* for its change listeners |
|
* with the following code: |
|
* |
|
* <pre>ChangeListener[] cls = (ChangeListener[])(m.getListeners(ChangeListener.class));</pre> |
|
* |
|
* If no such listeners exist, |
|
* this method returns an empty array. |
|
* |
|
* @param listenerType the type of listeners requested; |
|
* this parameter should specify an interface |
|
* that descends from <code>java.util.EventListener</code> |
|
* @return an array of all objects registered as |
|
* <code><em>Foo</em>Listener</code>s |
|
* on this model, |
|
* or an empty array if no such |
|
* listeners have been added |
|
* @exception ClassCastException if <code>listenerType</code> doesn't |
|
* specify a class or interface that implements |
|
* <code>java.util.EventListener</code> |
|
* |
|
* @see #getChangeListeners |
|
* |
|
* @since 1.3 |
|
*/ |
|
public <T extends EventListener> T[] getListeners(Class<T> listenerType) { |
|
return listenerList.getListeners(listenerType); |
|
} |
|
} |