javafx.beans.value.ObservableValue Maven / Gradle / Ivy
/*
* Copyright (c) 2010, 2024, 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 javafx.beans.value;
import java.util.Objects;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
import java.util.function.Function;
import com.sun.javafx.binding.ConditionalBinding;
import com.sun.javafx.binding.FlatMappedBinding;
import com.sun.javafx.binding.MappedBinding;
import com.sun.javafx.binding.OrElseBinding;
import javafx.beans.InvalidationListener;
import javafx.beans.Observable;
import javafx.util.Subscription;
/**
* An {@code ObservableValue} is an entity that wraps a value and allows to
* observe the value for changes. In general this interface should not be
* implemented directly but one of its sub-interfaces
* ({@code ObservableBooleanValue} etc.).
*
* The value of the {@code ObservableValue} can be requested with
* {@link #getValue()}.
*
* An implementation of {@code ObservableValue} may support lazy evaluation,
* which means that the value is not immediately recomputed after changes, but
* lazily the next time the value is requested (see note 1 in "Implementation Requirements").
*
* An {@code ObservableValue} generates two types of events: change events and
* invalidation events. A change event indicates that the value has changed
* (see note 2 in "Implementation Requirements"). An
* invalidation event is generated if the current value is not valid anymore.
* This distinction becomes important if the {@code ObservableValue} supports
* lazy evaluation, because for a lazily evaluated value one does not know if an
* invalid value really has changed until it is recomputed. For this reason,
* generating change events requires eager evaluation while invalidation events
* can be generated for eager and lazy implementations.
*
* Implementations of this class should strive to generate as few events as
* possible to avoid wasting too much time in event handlers. Implementations in
* this library mark themselves as invalid when the first invalidation event
* occurs. They do not generate any more invalidation events until their value is
* recomputed and valid again.
*
* Two types of listeners can be attached to an {@code ObservableValue}:
* {@link InvalidationListener} to listen to invalidation events and
* {@link ChangeListener} to listen to change events.
*
* Important note: attaching a {@code ChangeListener} enforces eager computation
* even if the implementation of the {@code ObservableValue} supports lazy
* evaluation.
*
* @param
* The type of the wrapped value.
*
* @implSpec
* - All bindings and properties in the JavaFX library support lazy evaluation.
* - All implementing classes in the JavaFX library check for a change using reference
* equality (and not object equality, {@code Object#equals(Object)}) of the value.
*
*
* @see ObservableBooleanValue
* @see ObservableDoubleValue
* @see ObservableFloatValue
* @see ObservableIntegerValue
* @see ObservableLongValue
* @see ObservableNumberValue
* @see ObservableObjectValue
* @see ObservableStringValue
*
*
* @since JavaFX 2.0
*/
public interface ObservableValue extends Observable {
/**
* Adds a {@link ChangeListener} which will be notified whenever the value
* of the {@code ObservableValue} changes. If the same listener is added
* more than once, then it will be notified more than once. That is, no
* check is made to ensure uniqueness.
*
* Note that the same actual {@code ChangeListener} instance may be safely
* registered for different {@code ObservableValues}.
*
* The {@code ObservableValue} stores a strong reference to the listener
* which will prevent the listener from being garbage collected and may
* result in a memory leak. It is recommended to either unregister a
* listener by calling {@link #removeListener(ChangeListener)
* removeListener} after use or to use an instance of
* {@link WeakChangeListener} avoid this situation.
*
* @see #removeListener(ChangeListener)
*
* @param listener
* The listener to register
* @throws NullPointerException
* if the listener is null
*/
void addListener(ChangeListener super T> listener);
/**
* Removes the given listener from the list of listeners that are notified
* whenever the value of the {@code ObservableValue} changes.
*
* If the given listener has not been previously registered (i.e. it was
* never added) then this method call is a no-op. If it had been previously
* added then it will be removed. If it had been added more than once, then
* only the first occurrence will be removed.
*
* @see #addListener(ChangeListener)
*
* @param listener
* The listener to remove
* @throws NullPointerException
* if the listener is null
*/
void removeListener(ChangeListener super T> listener);
/**
* Returns the current value of this {@code ObservableValue}
*
* @return The current value
*/
T getValue();
/**
* Returns an {@code ObservableValue} that holds the result of applying the
* given mapping function on this value. The result is updated when this
* {@code ObservableValue} changes. If this value is {@code null}, no
* mapping is applied and the resulting value is also {@code null}.
*
* For example, mapping a string to an upper case string:
*
{@code
* var text = new SimpleStringProperty("abcd");
* ObservableValue upperCase = text.map(String::toUpperCase);
*
* upperCase.getValue(); // Returns "ABCD"
* text.set("xyz");
* upperCase.getValue(); // Returns "XYZ"
* text.set(null);
* upperCase.getValue(); // Returns null
* }
*
* @param the type of values held by the resulting {@code ObservableValue}
* @param mapper the mapping function to apply to a value, cannot be {@code null}
* @return an {@code ObservableValue} that holds the result of applying the given
* mapping function on this value, or {@code null} when it
* is {@code null}; never returns {@code null}
* @throws NullPointerException if the mapping function is {@code null}
* @since 19
*/
default ObservableValue map(Function super T, ? extends U> mapper) {
return new MappedBinding<>(this, mapper);
}
/**
* Returns an {@code ObservableValue} that holds this value, or the given constant if
* it is {@code null}. The result is updated when this {@code ObservableValue} changes. This
* method, when combined with {@link #map(Function)}, allows handling of all values
* including {@code null} values.
*
* For example, mapping a string to an upper case string, but leaving it blank
* if the input is {@code null}:
*
{@code
* var text = new SimpleStringProperty("abcd");
* ObservableValue upperCase = text.map(String::toUpperCase).orElse("");
*
* upperCase.getValue(); // Returns "ABCD"
* text.set(null);
* upperCase.getValue(); // Returns ""
* }
*
* @param constant the value to use when this {@code ObservableValue}
* holds {@code null}; can be {@code null}
* @return an {@code ObservableValue} that holds this value, or the given constant if
* it is {@code null}; never returns {@code null}
* @since 19
*/
default ObservableValue orElse(T constant) {
return new OrElseBinding<>(this, constant);
}
/**
* Returns an {@code ObservableValue} that holds the value of an {@code ObservableValue}
* produced by applying the given mapping function on this value. The result is updated
* when either this {@code ObservableValue} or the {@code ObservableValue} produced by
* the mapping changes. If this value is {@code null}, no mapping is applied and the
* resulting value is {@code null}. If the mapping resulted in {@code null}, then the
* resulting value is also {@code null}.
*
* This method is similar to {@link #map(Function)}, but the mapping function is
* one whose result is already an {@code ObservableValue}, and if invoked, {@code flatMap} does
* not wrap it within an additional {@code ObservableValue}.
*
* For example, a property that is only {@code true} when a UI element is part of a {@code Scene}
* that is part of a {@code Window} that is currently shown on screen:
*
{@code
* ObservableValue isShowing = listView.sceneProperty()
* .flatMap(Scene::windowProperty)
* .flatMap(Window::showingProperty)
* .orElse(false);
*
* // Assuming the listView is currently shown to the user, then:
*
* isShowing.getValue(); // Returns true
*
* listView.getScene().getWindow().hide();
* isShowing.getValue(); // Returns false
*
* listView.getScene().getWindow().show();
* isShowing.getValue(); // Returns true
*
* listView.getParent().getChildren().remove(listView);
* isShowing.getValue(); // Returns false
* }
* Changes in any of the values of: the scene of {@code listView}, the window of that scene, or
* the showing of that window, will update the boolean value {@code isShowing}.
*
* This method is preferred over {@link javafx.beans.binding.Bindings#select Bindings} methods
* since it is type safe.
*
* @param the type of values held by the resulting {@code ObservableValue}
* @param mapper the mapping function to apply to a value, cannot be {@code null}
* @return an {@code ObservableValue} that holds the value of an {@code ObservableValue}
* produced by applying the given mapping function on this value, or
* {@code null} when the value is {@code null}; never returns {@code null}
* @throws NullPointerException if the mapping function is {@code null}
* @since 19
*/
default ObservableValue flatMap(Function super T, ? extends ObservableValue extends U>> mapper) {
return new FlatMappedBinding<>(this, mapper);
}
/**
* Returns an {@code ObservableValue} that holds this value and is updated only
* when {@code condition} holds {@code true}.
*
* The returned {@code ObservableValue} only observes this value when
* {@code condition} holds {@code true}. This allows this {@code ObservableValue}
* and the conditional {@code ObservableValue} to be garbage collected if neither is
* otherwise strongly referenced when {@code condition} holds {@code false}.
* This is in contrast to the general behavior of bindings, where the binding is
* only eligible for garbage collection when not observed itself.
*
* A {@code condition} holding {@code null} is treated as holding {@code false}.
*
* For example:
*
{@code
* Property condition = new SimpleBooleanProperty(true);
* Property longLivedProperty = new SimpleStringProperty("A");
* ObservableValue whenProperty = longLivedProperty.when(condition);
*
* // observe whenProperty, which will in turn observe longLivedProperty
* whenProperty.addListener((ov, old, current) -> System.out.println(current));
*
* longLivedProperty.setValue("B"); // "B" is printed
*
* condition.setValue(false);
*
* // After condition becomes false, whenProperty stops observing longLivedProperty; condition
* // and whenProperty may now be eligible for GC despite being observed by the ChangeListener
*
* longLivedProperty.setValue("C"); // nothing is printed
* longLivedProperty.setValue("D"); // nothing is printed
*
* condition.setValue(true); // longLivedProperty is observed again, and "D" is printed
* }
*
* @param condition a boolean {@code ObservableValue}, cannot be {@code null}
* @return an {@code ObservableValue} that holds this value whenever the given
* condition evaluates to {@code true}, otherwise holds the last seen value;
* never returns {@code null}
* @since 20
*/
default ObservableValue when(ObservableValue condition) {
return new ConditionalBinding<>(this, condition);
}
/**
* Creates a {@code Subscription} on this {@code ObservableValue} which calls the given
* {@code changeSubscriber} with the old and new value whenever its value changes.
* The provided subscriber is akin to a {@code ChangeListener} without the
* {@code ObservableValue} parameter.
*
* The parameters supplied to the {@link BiConsumer} are the old and new values,
* respectively.
*
* Note that the same subscriber instance may be safely subscribed for
* different {@code Observables}.
*
* Also note that when subscribing on an {@code Observable} with a longer
* lifecycle than the subscriber, the subscriber must be unsubscribed
* when no longer needed as the subscription will otherwise keep the subscriber
* from being garbage collected. Considering creating a derived {@code ObservableValue}
* using {@link #when(ObservableValue)} and subscribing on this derived observable value
* to automatically decouple the lifecycle of the subscriber from this
* {@code ObservableValue} when some condition holds.
*
* @param changeSubscriber a {@code BiConsumer} to supply with the old and new values
* of this {@code ObservableValue}, cannot be {@code null}
* @return a {@code Subscription} which can be used to cancel this
* subscription, never {@code null}
* @throws NullPointerException if the subscriber is {@code null}
* @see #addListener(ChangeListener)
* @since 21
*/
default Subscription subscribe(BiConsumer super T, ? super T> changeSubscriber) {
Objects.requireNonNull(changeSubscriber, "changeSubscriber cannot be null");
ChangeListener listener = (obs, old, current) -> changeSubscriber.accept(old, current);
addListener(listener);
return () -> removeListener(listener);
}
/**
* Creates a {@code Subscription} on this {@code ObservableValue} which immediately
* provides the current value to the given {@code valueSubscriber}, followed by any
* subsequent values whenever its value changes. The {@code valueSubscriber} is called
* immediately for convenience, since usually the user will want to initialize a value
* and then update on changes.
*
* Note that the same subscriber instance may be safely subscribed for
* different {@code Observables}.
*
* Also note that when subscribing on an {@code Observable} with a longer
* lifecycle than the subscriber, the subscriber must be unsubscribed
* when no longer needed as the subscription will otherwise keep the subscriber
* from being garbage collected. Considering creating a derived {@code ObservableValue}
* using {@link #when(ObservableValue)} and subscribing on this derived observable value
* to automatically decouple the lifecycle of the subscriber from this
* {@code ObservableValue} when some condition holds.
*
* @param valueSubscriber a {@code Consumer} to supply with the values of this
* {@code ObservableValue}, cannot be {@code null}
* @return a {@code Subscription} which can be used to cancel this
* subscription, never {@code null}
* @throws NullPointerException if the subscriber is {@code null}
* @since 21
*/
default Subscription subscribe(Consumer super T> valueSubscriber) {
Objects.requireNonNull(valueSubscriber, "valueSubscriber cannot be null");
ChangeListener listener = (obs, old, current) -> valueSubscriber.accept(current);
valueSubscriber.accept(getValue()); // eagerly send current value
addListener(listener);
return () -> removeListener(listener);
}
}