All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.reactfx.value.Val Maven / Gradle / Ivy

package org.reactfx.value;

import java.time.Duration;
import java.util.NoSuchElementException;
import java.util.Objects;
import java.util.Optional;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;

import javafx.animation.Interpolatable;
import javafx.beans.InvalidationListener;
import javafx.beans.property.Property;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.scene.Node;
import javafx.scene.Scene;
import javafx.stage.Window;

import org.reactfx.Change;
import org.reactfx.EventStream;
import org.reactfx.EventStreamBase;
import org.reactfx.EventStreams;
import org.reactfx.Observable;
import org.reactfx.Subscription;
import org.reactfx.collection.LiveList;
import org.reactfx.util.HexaFunction;
import org.reactfx.util.Interpolator;
import org.reactfx.util.PentaFunction;
import org.reactfx.util.TetraFunction;
import org.reactfx.util.TriFunction;
import org.reactfx.util.WrapperBase;

/**
 * Adds more operations to {@link ObservableValue}.
 *
 * 

Canonical observer of {@code Val} is an invalidation observer * of type {@code Consumer}, which accepts the _invalidated_ value. * This is different from {@linkplain InvalidationListener}, which does not * accept the invalidated value. */ public interface Val extends ObservableValue, Observable> { /* *************** * * Default methods * * *************** */ default void addInvalidationObserver(Consumer observer) { addObserver(observer); } default void removeInvalidationObserver(Consumer observer) { removeObserver(observer); } default Subscription observeInvalidations( Consumer oldValueObserver) { return observe(oldValueObserver); } default Subscription pin() { return observeInvalidations(oldVal -> {}); } @Override default void addListener(InvalidationListener listener) { addInvalidationObserver(new InvalidationListenerWrapper<>(this, listener)); } @Override default void removeListener(InvalidationListener listener) { removeInvalidationObserver(new InvalidationListenerWrapper<>(this, listener)); } @Override default void addListener(ChangeListener listener) { addInvalidationObserver(new ChangeListenerWrapper<>(this, listener)); } @Override default void removeListener(ChangeListener listener) { removeInvalidationObserver(new ChangeListenerWrapper<>(this, listener)); } /** * Adds a change listener and returns a Subscription that can be * used to remove that listener. */ default Subscription observeChanges(ChangeListener listener) { return observeInvalidations(new ChangeListenerWrapper<>(this, listener)); } /** * Returns a stream of invalidated values, which emits the invalidated value * (i.e. the old value) on each invalidation of this observable value. */ default EventStream invalidations() { return new EventStreamBase() { @Override protected Subscription observeInputs() { return observeInvalidations(this::emit); } }; } /** * Returns a stream of changed values, which emits the changed value * (i.e. the old and the new value) on each change of this observable value. */ default EventStream> changes() { return EventStreams.changesOf(this); } /** * Returns a stream of values of this {@linkplain Val}. The returned stream * emits the current value of this {@linkplain Val} for each new subscriber * and then the new value whenever the value changes. */ default EventStream values() { return EventStreams.valuesOf(this); } /** * Checks whether this {@linkplain Val} holds a (non-null) value. * @return {@code true} if this {@linkplain Val} holds a (non-null) value, * {@code false} otherwise. */ default boolean isPresent() { return getValue() != null; } /** * Inverse of {@link #isPresent()}. */ default boolean isEmpty() { return getValue() == null; } /** * Invokes the given function if this {@linkplain Val} holds a (non-null) * value. * @param f function to invoke on the value currently held by this * {@linkplain Val}. */ default void ifPresent(Consumer f) { T val = getValue(); if(val != null) { f.accept(val); } } /** * Returns the value currently held by this {@linkplain Val}. * @throws NoSuchElementException if there is no value present. */ default T getOrThrow() { T res = getValue(); if(res != null) { return res; } else { throw new NoSuchElementException(); } } /** * Returns the value currently held by this {@linkplain Val}. If this * {@linkplain Val} is empty, {@code defaultValue} is returned instead. * @param defaultValue value to return if there is no value present in * this {@linkplain Val}. */ default T getOrElse(T defaultValue) { T res = getValue(); if(res != null) { return res; } else { return defaultValue; } } /** * Like {@link #getOrElse(Object)}, except the default value is computed * by {@code defaultSupplier} only when necessary. * @param defaultSupplier computation to produce default value, if this * {@linkplain Val} is empty. */ default T getOrSupply(Supplier defaultSupplier) { T res = getValue(); if(res != null) { return res; } else { return defaultSupplier.get(); } } /** * Returns an {@code Optional} describing the value currently held by this * {@linkplain Val}, or and empty {@code Optional} if this {@linkplain Val} * is empty. */ default Optional getOpt() { return Optional.ofNullable(getValue()); } /** * Returns a new {@linkplain Val} that holds the value held by this * {@linkplain Val}, or {@code other} when this {@linkplain Val} is empty. */ default Val orElseConst(T other) { return orElseConst(this, other); } /** * Returns a new {@linkplain Val} that holds the value held by this * {@linkplain Val}, or the value held by {@code other} when this * {@linkplain Val} is empty. */ default Val orElse(ObservableValue other) { return orElse(this, other); } /** * Returns a new {@linkplain Val} that holds the same value * as this {@linkplain Val} when the value satisfies the predicate * and is empty when this {@linkplain Val} is empty or its value * does not satisfy the given predicate. */ default Val filter(Predicate p) { return filter(this, p); } /** * Returns a new {@linkplain Val} that holds a mapping of the value held by * this {@linkplain Val}, and is empty when this {@linkplain Val} is empty. * @param f function to map the value held by this {@linkplain Val}. */ default Val map(Function f) { return map(this, f); } /** * Like {@link #map(Function)}, but also allows dynamically changing * map function. */ default Val mapDynamic( ObservableValue> f) { return mapDynamic(this, f); } /** * Returns a new {@linkplain Val} that, when this {@linkplain Val} holds * value {@code x}, holds the value held by {@code f(x)}, and is empty * when this {@linkplain Val} is empty. */ default Val flatMap( Function> f) { return flatMap(this, f); } /** * Similar to {@link #flatMap(Function)}, except the returned Val is also * a Var. This means you can call {@code setValue()} and {@code bind()} * methods on the returned value, which delegate to the currently selected * Property. * *

As the value of this {@linkplain Val} changes, so does the selected * Property. When the Var returned from this method is bound, as the * selected Property changes, the previously selected Property is unbound * and the newly selected Property is bound. * *

Note that if the currently selected Property is {@code null}, then * calling {@code getValue()} on the returned value will return {@code null} * regardless of any prior call to {@code setValue()} or {@code bind()}. */ default Var selectVar( Function> f) { return selectVar(this, f); } default Var selectVar( Function> f, U resetToOnUnbind) { return selectVar(this, f, resetToOnUnbind); } /** * Returns a new {@linkplain Val} that only observes this {@linkplain Val} * when {@code condition} is {@code true}. More precisely, the returned * {@linkplain Val} observes {@code condition} whenever it itself has at * least one observer and observes {@code this} {@linkplain Val} whenever * it itself has at least one observer and the value of * {@code condition} is {@code true}. When {@code condition} is * {@code true}, the returned {@linkplain Val} has the same value as this * {@linkplain Val}. When {@code condition} is {@code false}, the returned * {@linkplain Val} has the value that was held by this {@linkplain Val} at * the time when {@code condition} changed to {@code false}. */ default Val conditionOn(ObservableValue condition) { return conditionOn(this, condition); } /** * Equivalent to {@link #conditionOn(ObservableValue)} where the condition * is that {@code node} is showing: it is part of a scene graph * ({@link Node#sceneProperty()} is not {@code null}), its scene is part of * a window ({@link Scene#windowProperty()} is not {@code null}) and the * window is showing ({@link Window#showingProperty()} is {@code true}). */ default Val conditionOnShowing(Node node) { return conditionOnShowing(this, node); } default SuspendableVal suspendable() { return suspendable(this); } /** * Returns a new {@linkplain Val} that gradually transitions to the value * of this {@linkplain Val} every time this {@linkplain Val} changes. * *

When the returned {@linkplain Val} has no observer, there is no * gradual transition taking place. This means that there is no animation * running in the background that would consume system resources. This also * means that in that case {@link #getValue()} always returns the target * value (i.e. the current value of this {@linkplain Val}), instead of any * intermediate interpolated value. * * @param duration function that calculates the desired duration of the * transition for two boundary values. * @param interpolator calculates the interpolated value between two * boundary values, given a fraction. */ default Val animate( BiFunction duration, Interpolator interpolator) { return animate(this, duration, interpolator); } /** * Returns a new {@linkplain Val} that gradually transitions to the value * of this {@linkplain Val} every time this {@linkplain Val} changes. * *

When the returned {@linkplain Val} has no observer, there is no * gradual transition taking place. This means that there is no animation * running in the background that would consume system resources. This also * means that in that case {@link #getValue()} always returns the target * value (i.e. the current value of this {@linkplain Val}), instead of any * intermediate interpolated value. * * @param duration the desired duration of the transition * @param interpolator calculates the interpolated value between two * boundary values, given a fraction. */ default Val animate( Duration duration, Interpolator interpolator) { return animate(this, duration, interpolator); } /** * Let's this {@linkplain Val} be viewed as a {@linkplain Var}, with the * given {@code setValue} function serving the purpose of * {@link Var#setValue(Object)}. * @see Var#fromVal(ObservableValue, Consumer) */ default Var asVar(Consumer setValue) { return new VarFromVal<>(this, setValue); } /** * Returns a {@linkplain LiveList} view of this {@linkplain Val}. The * returned list will have size 1 when this {@linkplain Val} is present * (i.e. not {@code null}) and size 0 otherwise. */ default LiveList asList() { return LiveList.wrapVal(this); } /* ************** * * Static methods * * ************** */ /** * Returns a {@linkplain Val} wrapper around {@linkplain ObservableValue}. * If the argument is already a {@code Val}, no wrapping occurs and the * argument is returned as is. * Note that one rarely needs to use this method, because most of the time * one can use the appropriate static method directly to get the desired * result. For example, instead of * *

     * {@code
     * Val.wrap(obs).orElse(other)
     * }
     * 
* * one can write * *
     * {@code
     * Val.orElse(obs, other)
     * }
     * 
* * However, an explicit wrapper is necessary if access to * {@link #observeInvalidations(Consumer)}, or {@link #invalidations()} * is needed, since there is no direct static method equivalent for them. */ static Val wrap(ObservableValue obs) { return obs instanceof Val ? (Val) obs : new ValWrapper<>(obs); } static Subscription observeChanges( ObservableValue obs, ChangeListener listener) { if(obs instanceof Val) { return ((Val) obs).observeChanges(listener); } else { obs.addListener(listener); return () -> obs.removeListener(listener); } } static Subscription observeInvalidations( ObservableValue obs, InvalidationListener listener) { obs.addListener(listener); return () -> obs.removeListener(listener); } static Val orElseConst(ObservableValue src, T other) { return new OrElseConst<>(src, other); } static Val orElse( ObservableValue src, ObservableValue other) { return new OrElse<>(src, other); } static Val filter( ObservableValue src, Predicate p) { return map(src, t -> p.test(t) ? t : null); } static Val map( ObservableValue src, Function f) { return new MappedVal<>(src, f); } static Val mapDynamic( ObservableValue src, ObservableValue> f) { return combine( src, f, (t, fn) -> t == null || fn == null ? null : fn.apply(t)); } static Val flatMap( ObservableValue src, Function> f) { return new FlatMappedVal<>(src, f); } static Var selectVar( ObservableValue src, Function> f) { return new FlatMappedVar<>(src, f); } static Var selectVar( ObservableValue src, Function> f, U resetToOnUnbind) { return new FlatMappedVar<>(src, f, resetToOnUnbind); } static Val conditionOn( ObservableValue obs, ObservableValue condition) { return flatMap(condition, con -> con ? obs : constant(obs.getValue())); } static Val conditionOnShowing(ObservableValue obs, Node node) { return conditionOn(obs, showingProperty(node)); } static SuspendableVal suspendable(ObservableValue obs) { if(obs instanceof SuspendableVal) { return (SuspendableVal) obs; } else { Val val = obs instanceof Val ? (Val) obs : new ValWrapper<>(obs); return new SuspendableValWrapper<>(val); } } /** * Creates a new {@linkplain Val} that gradually transitions to the value * of the given {@linkplain ObservableValue} {@code obs} every time * {@code obs} changes. * *

When the returned {@linkplain Val} has no observer, there is no * gradual transition taking place. This means that there is no animation * running in the background that would consume system resources. This also * means that in that case {@link #getValue()} always returns the target * value (i.e. the current value of {@code obs}), instead of any intermediate * interpolated value. * * @param obs observable value to animate * @param duration function that calculates the desired duration of the * transition for two boundary values. * @param interpolator calculates the interpolated value between two * boundary values, given a fraction. */ static Val animate( ObservableValue obs, BiFunction duration, Interpolator interpolator) { return new AnimatedVal<>(obs, duration, interpolator); } /** * Creates a new {@linkplain Val} that gradually transitions to the value * of the given {@linkplain ObservableValue} {@code obs} every time * {@code obs} changes. * *

When the returned {@linkplain Val} has no observer, there is no * gradual transition taking place. This means that there is no animation * running in the background that would consume system resources. This also * means that in that case {@link #getValue()} always returns the target * value (i.e. the current value of {@code obs}), instead of any intermediate * interpolated value. * * @param obs observable value to animate * @param duration the desired duration of the transition * @param interpolator calculates the interpolated value between two * boundary values, given a fraction. */ static Val animate( ObservableValue obs, Duration duration, Interpolator interpolator) { return animate(obs, (a, b) -> duration, interpolator); } /** * Like {@link #animate(ObservableValue, BiFunction, Interpolator)}, but * uses the interpolation defined by the {@linkplain Interpolatable} type * {@code T}. */ static > Val animate( ObservableValue obs, BiFunction duration) { return animate(obs, duration, Interpolator.get()); } /** * Like {@link #animate(ObservableValue, Duration, Interpolator)}, but * uses the interpolation defined by the {@linkplain Interpolatable} type * {@code T}. */ static > Val animate( ObservableValue obs, Duration duration) { return animate(obs, duration, Interpolator.get()); } static Val combine( ObservableValue src1, ObservableValue src2, BiFunction f) { return create( () -> { if(src1.getValue() != null && src2.getValue() != null) { return f.apply(src1.getValue(), src2.getValue()); } else { return null; } }, src1, src2); } static Val combine( ObservableValue src1, ObservableValue src2, ObservableValue src3, TriFunction f) { return create( () -> { if(src1.getValue() != null && src2.getValue() != null && src3.getValue() != null) { return f.apply( src1.getValue(), src2.getValue(), src3.getValue()); } else { return null; } }, src1, src2, src3); } static Val combine( ObservableValue src1, ObservableValue src2, ObservableValue src3, ObservableValue src4, TetraFunction f) { return create( () -> { if(src1.getValue() != null && src2.getValue() != null && src3.getValue() != null && src4.getValue() != null) { return f.apply( src1.getValue(), src2.getValue(), src3.getValue(), src4.getValue()); } else { return null; } }, src1, src2, src3, src4); } static Val combine( ObservableValue src1, ObservableValue src2, ObservableValue src3, ObservableValue src4, ObservableValue src5, PentaFunction f) { return create( () -> { if(src1.getValue() != null && src2.getValue() != null && src3.getValue() != null && src4.getValue() != null && src5.getValue() != null) { return f.apply( src1.getValue(), src2.getValue(), src3.getValue(), src4.getValue(), src5.getValue()); } else { return null; } }, src1, src2, src3, src4, src5); } static Val combine( ObservableValue src1, ObservableValue src2, ObservableValue src3, ObservableValue src4, ObservableValue src5, ObservableValue src6, HexaFunction f) { return create( () -> { if(src1.getValue() != null && src2.getValue() != null && src3.getValue() != null && src4.getValue() != null && src5.getValue() != null && src6.getValue() != null) { return f.apply( src1.getValue(), src2.getValue(), src3.getValue(), src4.getValue(), src5.getValue(), src6.getValue()); } else { return null; } }, src1, src2, src3, src4, src5, src6); } static Val create( Supplier computeValue, javafx.beans.Observable... dependencies) { return new ValBase() { @Override protected Subscription connect() { InvalidationListener listener = obs -> invalidate(); for(javafx.beans.Observable dep: dependencies) { dep.addListener(listener); } return () -> { for(javafx.beans.Observable dep: dependencies) { dep.removeListener(listener); } }; } @Override protected T computeValue() { return computeValue.get(); } }; } static Val create( Supplier computeValue, EventStream invalidations) { return new ValBase() { @Override protected Subscription connect() { return invalidations.subscribe(x -> invalidate()); } @Override protected T computeValue() { return computeValue.get(); } }; } /** * Returns a constant {@linkplain Val} that holds the given value. * The value never changes and no notifications are ever produced. */ static Val constant(T value) { return new ConstVal<>(value); } /** * Returns a {@linkplain Val} whose value is {@code true} when {@code node} * is showing: it is part of a scene graph * ({@link Node#sceneProperty()} is not {@code null}), its scene is part of * a window ({@link Scene#windowProperty()} is not {@code null}) and the * window is showing ({@link Window#showingProperty()} is {@code true}). */ static Val showingProperty(Node node) { return Val .flatMap(node.sceneProperty(), Scene::windowProperty) .flatMap(Window::showingProperty) .orElseConst(false); } } class InvalidationListenerWrapper extends WrapperBase implements Consumer { private final ObservableValue obs; public InvalidationListenerWrapper( ObservableValue obs, InvalidationListener listener) { super(listener); this.obs = obs; } @Override public void accept(T oldValue) { getWrappedValue().invalidated(obs); } } class ChangeListenerWrapper extends WrapperBase> implements Consumer { private final ObservableValue obs; public ChangeListenerWrapper( ObservableValue obs, ChangeListener listener) { super(listener); this.obs = obs; } @Override public void accept(T oldValue) { T newValue = obs.getValue(); if(!Objects.equals(oldValue, newValue)) { getWrappedValue().changed(obs, oldValue, newValue); } } }