001/*
002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others
003 *
004 * This program and the accompanying materials are made available under the
005 * terms of the Eclipse Public License 2.0 which is available at
006 * http://www.eclipse.org/legal/epl-2.0.
007 *
008 * SPDX-License-Identifier: EPL-2.0
009 */
010
011package gololang;
012
013import java.util.LinkedList;
014
015/**
016 * Models a thread-safe observable variable.
017 */
018public final class Observable {
019
020  private volatile Object value;
021
022  private final Object lock = new Object();
023  private final LinkedList<Observer> observers = new LinkedList<>();
024
025  /**
026   * Creates a new observable from an initial value.
027   *
028   * @param initialValue the initial value.
029   */
030  public Observable(Object initialValue) {
031    this.value = initialValue;
032  }
033
034  /**
035   * Gets the current value.
036   *
037   * @return the current value.
038   */
039  public Object get() {
040    return value;
041  }
042
043  /**
044   * Changes the current value and notifies all observers.
045   *
046   * @param newValue the new value.
047   */
048  public void set(Object newValue) {
049    synchronized (lock) {
050      this.value = newValue;
051      for (Observer observer : observers) {
052        observer.apply(newValue);
053      }
054    }
055  }
056
057  /**
058   * Registers an observer.
059   *
060   * @param observer an observer.
061   * @return this observable object.
062   */
063  public Observable onChange(Observer observer) {
064    synchronized (lock) {
065      observers.add(observer);
066    }
067    return this;
068  }
069
070  /**
071   * Creates an observer that filters the values of this observable.
072   *
073   * @param predicate a predicate function.
074   * @return an observer whose values filter this observable.
075   */
076  public Observable filter(final Predicate predicate) {
077    final Observable observable = new Observable(null);
078    this.onChange(newValue -> {
079      if (predicate.apply(newValue)) {
080        observable.set(newValue);
081      }
082    });
083    return observable;
084  }
085
086  /**
087   * Creates an observer that maps the values of this observable.
088   *
089   * @param function a mapping function.
090   * @return an observer whose values map this observable.
091   */
092  public Observable map(final Function function) {
093    final Observable observable = new Observable(null);
094    this.onChange(newValue -> observable.set(function.apply(newValue)));
095    return observable;
096  }
097
098  @Override
099  public String toString() {
100    return String.format("Observable{value=%s}", value);
101  }
102
103  @FunctionalInterface
104  public interface Function {
105    Object apply(Object value);
106  }
107
108  @FunctionalInterface
109  public interface Predicate {
110    boolean apply(Object value);
111  }
112
113  @FunctionalInterface
114  public interface Observer {
115    void apply(Object newValue);
116  }
117}