PropertyAdapter.java

/*
 * *********************************************************************************************************************
 *
 * SteelBlue: DCI User Interfaces
 * http://tidalwave.it/projects/steelblue
 *
 * Copyright (C) 2015 - 2023 by Tidalwave s.a.s. (http://tidalwave.it)
 *
 * *********************************************************************************************************************
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
 * the License. You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
 * an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations under the License.
 *
 * *********************************************************************************************************************
 *
 * git clone https://bitbucket.org/tidalwave/steelblue-src
 * git clone https://github.com/tidalwave-it/steelblue-src
 *
 * *********************************************************************************************************************
 */
package it.tidalwave.role.ui.javafx.impl.common;

import javax.annotation.Nonnull;
import java.beans.PropertyChangeListener;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.Executor;
import javafx.beans.InvalidationListener;
import javafx.beans.property.Property;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.application.Platform;
import it.tidalwave.role.ui.BoundProperty;
import lombok.extern.slf4j.Slf4j;

/***********************************************************************************************************************
 *
 * Adapts a {@link BoundProperty} to a JavaFX {@link Property}. It also takes care of threaring issues, making sure that
 * the JavaFX {@code Property} is updated in the JavaFX UI thread. Conversely, updates on the JavaFX 
 * {@code BoundProperty} are executed in a separated thread provided by an {@link Executor}.
 *
 * TODO: javafx.beans.binding.BooleanExpression.booleanExpression(source)? Does it do threading?
 * 
 * @author  Fabrizio Giudici
 *
 **********************************************************************************************************************/
@Slf4j
public class PropertyAdapter<T> implements Property<T>
  {
    @Nonnull
    private final Executor executor;

    @Nonnull
    private final BoundProperty<T> delegate;

    // FIXME: WEAK LISTENER!!
    private final List<ChangeListener<? super T>> changeListeners = new ArrayList<>();
    
    // FIXME: WEAK LISTENER!!
    private final List<InvalidationListener> invalidationListeners = new ArrayList<>();

    private T boundValue;

    private final PropertyChangeListener propertyChangeListener = (event) -> 
      {
        log.trace("propertyChange({}) - bound value: {}", event, boundValue);
        
        if (!Objects.equals(boundValue, event.getNewValue()))
          {
            boundValue = (T)event.getNewValue();
            Platform.runLater(() ->
              {
                new ArrayList<>(invalidationListeners)
                        .forEach(listener -> listener.invalidated(PropertyAdapter.this));
                new ArrayList<>(changeListeners)
                        .forEach(listener -> listener.changed(PropertyAdapter.this,
                                (T)event.getOldValue(), (T)event.getNewValue()));
              });
          }
    };

    public PropertyAdapter (@Nonnull final Executor executor, @Nonnull final BoundProperty<T> delegate)
      {
        this.executor = executor;
        this.delegate = delegate;
        this.boundValue = delegate.get();
        delegate.addPropertyChangeListener(propertyChangeListener);
      }

    @Override
    public T getValue()
      {
        return delegate.get();
      }

    @Override
    public void setValue (final T value)
      {
        log.debug("setValue({})", value);
        boundValue = value;

        if (!Objects.equals(value, delegate.get()))
          {
            executor.execute(() -> delegate.set(value));
          }
      }

    @Override
    public void addListener (@Nonnull final ChangeListener<? super T> listener)
      {
        changeListeners.add(listener);
      }

    @Override
    public void removeListener (@Nonnull final ChangeListener<? super T> listener)
      {
        changeListeners.remove(listener);
      }

    @Override
    public void addListener (@Nonnull final InvalidationListener listener)
      {
        invalidationListeners.add(listener);
      }

    @Override
    public void removeListener (@Nonnull final InvalidationListener listener)
      {
        invalidationListeners.remove(listener);
      }

    @Override
    public void bind (final ObservableValue<? extends T> observable)
      {
        log.warn("bind({})", observable);
        throw new UnsupportedOperationException("Not supported yet.");
      }

    @Override
    public void unbind()
      {
        log.warn("unbind()");
        throw new UnsupportedOperationException("Not supported yet.");
      }

    @Override
    public boolean isBound()
      {
        log.warn("isBound()");
        throw new UnsupportedOperationException("Not supported yet.");
      }

    @Override
    public void bindBidirectional (final Property<T> other)
      {
        log.warn("bindBidirectional({})", other);
        throw new UnsupportedOperationException("Not supported yet.");
      }

    @Override
    public void unbindBidirectional (final Property<T> other)
      {
        log.warn("unbindBidirectional({})", other);
        throw new UnsupportedOperationException("Not supported yet.");
      }

    @Override
    public Object getBean()
      {
        log.warn("getBean()");
        throw new UnsupportedOperationException("Not supported yet.");
      }

    @Override
    public String getName()
      {
        log.warn("getName()");
        throw new UnsupportedOperationException("Not supported yet.");
      }
  }