blob: d8ad8305ca746f715166b7c00ceccbea5721cb21 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2005, 2009 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
* Brad Reynolds - bug 164653
* Matthew Hall - bug 263691
*******************************************************************************/
package org.eclipse.core.databinding.observable.value;
import org.eclipse.core.databinding.observable.Diffs;
import org.eclipse.core.databinding.observable.ListenerList;
import org.eclipse.core.databinding.observable.Realm;
import org.eclipse.core.internal.databinding.observable.Util;
/**
*
* <p>
* This class is thread safe. All state accessing methods must be invoked from
* the {@link Realm#isCurrent() current realm}. Methods for adding and removing
* listeners may be invoked from any thread.
* </p>
*
* @param <T>
* @since 1.0
*
*/
public abstract class AbstractVetoableValue<T> extends
AbstractObservableValue<T> implements IVetoableValue<T> {
/**
* In addition to the listener/event types supported by
* AbstractObservableValue, we add support for one more type.
*/
protected ListenerList<IValueChangingListener<T>> valueChangingListenerList = null;
/**
* Creates a new vetoable value.
*/
public AbstractVetoableValue() {
this(Realm.getDefault());
}
/**
* @param realm
*/
public AbstractVetoableValue(Realm realm) {
super(realm);
}
final protected void doSetValue(T value) {
T currentValue = doGetValue();
ValueDiff<T> diff = Diffs.createValueDiff(currentValue, value);
boolean okToProceed = fireValueChanging(diff);
if (!okToProceed) {
throw new ChangeVetoException("Change not permitted"); //$NON-NLS-1$
}
doSetApprovedValue(value);
if (!Util.equals(diff.getOldValue(), diff.getNewValue())) {
fireValueChange(diff);
}
}
/**
* Sets the value. Invoked after performing veto checks. Should not fire
* change events.
*
* @param value
*/
protected abstract void doSetApprovedValue(T value);
public synchronized void addValueChangingListener(
IValueChangingListener<T> listener) {
if (valueChangingListenerList == null) {
valueChangingListenerList = new ListenerList<IValueChangingListener<T>>();
}
addListener(valueChangingListenerList, listener);
}
public synchronized void removeValueChangingListener(
IValueChangingListener<T> listener) {
if (valueChangingListenerList != null) {
removeListener(valueChangingListenerList, listener);
}
}
@Override
protected boolean hasListeners() {
return ((valueChangingListenerList != null && valueChangingListenerList
.hasListeners()) || super.hasListeners());
}
/**
* Notifies listeners about a pending change, and returns true if no
* listener vetoed the change.
*
* @param diff
* @return false if the change was vetoed, true otherwise
*/
protected boolean fireValueChanging(ValueDiff<T> diff) {
checkRealm();
ValueChangingEvent<T> event = new ValueChangingEvent<T>(this, diff);
if (valueChangingListenerList != null) {
valueChangingListenerList.fireEvent(event);
}
return !event.veto;
}
}