ParameterDriversList.java

  1. /* Copyright 2002-2025 CS GROUP
  2.  * Licensed to CS GROUP (CS) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * CS licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *   http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.orekit.utils;

  18. import java.util.ArrayList;
  19. import java.util.Collections;
  20. import java.util.Comparator;
  21. import java.util.Iterator;
  22. import java.util.List;

  23. import org.orekit.time.AbsoluteDate;
  24. import org.orekit.utils.TimeSpanMap.Span;


  25. /** Class managing several {@link ParameterDriver parameter drivers},
  26.  * taking care of duplicated names.
  27.  * <p>
  28.  * Once parameter drivers sharing the same name have been added to
  29.  * an instance of this class, they are permanently bound together and
  30.  * also bound to the {@link #getDrivers() delegating driver} that
  31.  * manages them. This means that if drivers {@code d1}, {@code d2}...
  32.  * {@code dn} are added to the list and both correspond to parameter
  33.  * name "P", then {@link #getDrivers()} will return a list containing
  34.  * a delegating driver {@code delegateD} for the same name "P".
  35.  * Afterwards, whenever either {@link ParameterDriver#setValue(double)}
  36.  * or {@link ParameterDriver#setReferenceDate(AbsoluteDate)} is called
  37.  * on any of the {@code n+1} instances {@code d1}, {@code d2}... {@code dn}
  38.  * or {@code delegateD}, the call will be automatically forwarded to the
  39.  * {@code n} remaining instances, hence ensuring they remain consistent
  40.  * with each other.
  41.  * </p>
  42.  * @author Luc Maisonobe
  43.  * @author Mélina Vanel
  44.  * @since 8.0
  45.  */
  46. public class ParameterDriversList {

  47.     /** Managed drivers. */
  48.     private final List<DelegatingDriver> delegating;

  49.     /** Creates an empty list.
  50.      */
  51.     public ParameterDriversList() {
  52.         this.delegating = new ArrayList<>();
  53.     }

  54.     /** Add a driver.
  55.      * <p>
  56.      * If the driver is already present, it will not be added.
  57.      * If another driver managing the same parameter is present,
  58.      * both drivers will be managed together, existing drivers
  59.      * being set to the value of the last driver added (i.e.
  60.      * each addition overrides the parameter value).
  61.      * </p>
  62.      * <p>
  63.      * Warning if a driver is added and a driver with the same name
  64.      * was already added before, they should have the same validity
  65.      * periods to avoid surprises. Whatever, all driver having
  66.      * same name will have their valueSpanMap, nameSpanMap and validity period
  67.      * overwritten with the last driver added attributes.
  68.      * </p>
  69.      * @param driver driver to add
  70.      */
  71.     public void add(final ParameterDriver driver) {

  72.         final DelegatingDriver existingHere = findByName(driver.getName());
  73.         final DelegatingDriver alreadyBound = getAssociatedDelegatingDriver(driver);

  74.         if (existingHere != null) {
  75.             if (alreadyBound != null) {
  76.                 // merge the two delegating drivers
  77.                 existingHere.merge(alreadyBound);
  78.             } else {
  79.                 // this is a new driver for an already managed parameter
  80.                 existingHere.add(driver);
  81.             }
  82.         } else {
  83.             if (alreadyBound != null) {
  84.                 // the driver is new here, but already bound to other drivers in other lists
  85.                 delegating.add(alreadyBound);
  86.                 alreadyBound.addOwner(this);
  87.             } else {
  88.                 // this is the first driver we have for this parameter name
  89.                 delegating.add(new DelegatingDriver(this, driver));
  90.             }
  91.         }

  92.     }

  93.     /** Get a {@link DelegatingDriver delegating driver} bound to a driver.
  94.      * @param driver driver to check
  95.      * @return a {@link DelegatingDriver delegating driver} bound to a driver, or
  96.      * null if this driver is not associated with any {@link DelegatingDriver delegating driver}
  97.      * @since 9.1
  98.      */
  99.     private DelegatingDriver getAssociatedDelegatingDriver(final ParameterDriver driver) {
  100.         for (final ParameterObserver observer : driver.getObservers()) {
  101.             if (observer instanceof ChangesForwarder) {
  102.                 return ((ChangesForwarder) observer).getDelegatingDriver();
  103.             }
  104.         }
  105.         return null;
  106.     }

  107.     /** Replace a {@link DelegatingDriver delegating driver}.
  108.      * @param oldDelegating delegating driver to replace
  109.      * @param newDelegating new delegating driver to use
  110.      * @since 10.1
  111.      */
  112.     private void replaceDelegating(final DelegatingDriver oldDelegating, final DelegatingDriver newDelegating) {
  113.         for (int i = 0; i < delegating.size(); ++i) {
  114.             if (delegating.get(i) == oldDelegating) {
  115.                 delegating.set(i, newDelegating);
  116.             }
  117.         }
  118.     }

  119.     /** Find  a {@link DelegatingDriver delegating driver} by name.
  120.      * @param name name to check
  121.      * @return a {@link DelegatingDriver delegating driver} managing this parameter name
  122.      * @since 9.1
  123.      */
  124.     public DelegatingDriver findByName(final String name) {
  125.         for (final DelegatingDriver d : delegating) {
  126.             if (d.getName().equals(name)) {
  127.                 return d;
  128.             }
  129.         }
  130.         return null;
  131.     }

  132.     /** Find  a {@link DelegatingDriver delegating driver} by name.
  133.      * @param name name to check
  134.      * @return a {@link DelegatingDriver delegating driver} managing this parameter name
  135.      * @since 9.1
  136.      */
  137.     public String findDelegatingSpanNameBySpanName(final String name) {
  138.         for (final DelegatingDriver d : delegating) {
  139.             for (Span<String> span = d.getNamesSpanMap().getFirstSpan(); span != null; span = span.next()) {
  140.                 if (span.getData().equals(name)) {
  141.                     return span.getData();
  142.                 }
  143.             }
  144.         }
  145.         return null;
  146.     }


  147.     /** Sort the parameters lexicographically.
  148.      */
  149.     public void sort() {
  150.         delegating.sort(Comparator.comparing(ParameterDriver::getName));
  151.     }

  152.     /** Filter parameters to keep only one type of selection status.
  153.      * @param selected if true, only {@link ParameterDriver#isSelected()
  154.      * selected} parameters will be kept, the other ones will be removed
  155.      */
  156.     public void filter(final boolean selected) {
  157.         for (final Iterator<DelegatingDriver> iterator = delegating.iterator(); iterator.hasNext();) {
  158.             final DelegatingDriver delegatingDriver = iterator.next();
  159.             if (delegatingDriver.isSelected() != selected) {
  160.                 iterator.remove();
  161.                 delegatingDriver.removeOwner(this);
  162.             }
  163.         }
  164.     }

  165.     /** Get the number of parameters with different names.
  166.      * @return number of parameters with different names
  167.      */
  168.     public int getNbParams() {
  169.         return delegating.size();
  170.     }

  171.     /** Get the number of values to estimate for parameters with different names.
  172.      * @return number of values to estimate for parameters with different names
  173.      */
  174.     public int getNbValuesToEstimate() {
  175.         int nbValuesToEstimate = 0;
  176.         for (DelegatingDriver driver : delegating) {
  177.             nbValuesToEstimate += driver.getNbOfValues();
  178.         }
  179.         return nbValuesToEstimate;
  180.     }

  181.     /** Get delegating drivers for all parameters.
  182.      * <p>
  183.      * The delegating drivers are <em>not</em> the same as
  184.      * the drivers added to the list, but they delegate to them.
  185.      * </p>
  186.      * <p>
  187.      * All delegating drivers manage parameters with different names.
  188.      * </p>
  189.      * @return unmodifiable view of the list of delegating drivers
  190.      */
  191.     public List<DelegatingDriver> getDrivers() {
  192.         return Collections.unmodifiableList(delegating);
  193.     }

  194.     /** Specialized driver delegating to several other managing
  195.      * the same parameter name.
  196.      */
  197.     public static class DelegatingDriver extends ParameterDriver {

  198.         /** Lists owning this delegating driver. */
  199.         private final List<ParameterDriversList> owners;

  200.         /** Observer for propagating changes between all drivers. */
  201.         private ChangesForwarder forwarder;

  202.         /** Simple constructor.
  203.          * @param owner list owning this delegating driver
  204.          * @param driver first driver in the series
  205.          */
  206.         DelegatingDriver(final ParameterDriversList owner, final ParameterDriver driver) {
  207.             super(driver.getName(), driver.getNamesSpanMap(),
  208.                   driver.getValueSpanMap(), driver.getReferenceValue(),
  209.                   driver.getScale(), driver.getMinValue(), driver.getMaxValue());

  210.             owners = new ArrayList<>();
  211.             addOwner(owner);

  212.             setValueSpanMap(driver);
  213.             setReferenceDate(driver.getReferenceDate());
  214.             setSelected(driver.isSelected());

  215.             // set up a change forwarder observing both the raw driver and the delegating driver
  216.             this.forwarder = new ChangesForwarder(this, driver);
  217.             addObserver(forwarder);
  218.             driver.addObserver(forwarder);

  219.         }

  220.         /** Add an owner for this delegating driver.
  221.          * @param owner owner to add
  222.          */
  223.         void addOwner(final ParameterDriversList owner) {
  224.             owners.add(owner);
  225.         }

  226.         /** Remove one owner of this driver.
  227.          * @param owner owner to remove delegating driver from
  228.          * @since 10.1
  229.          */
  230.         private void removeOwner(final ParameterDriversList owner) {
  231.             for (final Iterator<ParameterDriversList> iterator = owners.iterator(); iterator.hasNext();) {
  232.                 if (iterator.next() == owner) {
  233.                     iterator.remove();
  234.                 }
  235.             }
  236.         }

  237.         /** Add a driver. Warning, by doing this operation
  238.          * all the delegated drivers present in the parameterDriverList
  239.          * will be overwritten with the attributes of the driver given
  240.          * in argument.
  241.          * <p>
  242.          * </p>
  243.          * Warning if a driver is added and a driver with the same name
  244.          * was already added before, they should have the same validity
  245.          * Period (that is to say that the {@link
  246.          * ParameterDriver#addSpans(AbsoluteDate, AbsoluteDate, double)}
  247.          * and {@link ParameterDriver#addSpanAtDate(AbsoluteDate)} methods
  248.          * should have been called with the same arguments for all drivers
  249.          * having the same name) to avoid surprises. Whatever, all driver having
  250.          * same name will have their valueSpanMap, nameSpanMap and validity period
  251.          * overwritten with the last driver added attributes.
  252.          * @param driver driver to add
  253.          */
  254.         private void add(final ParameterDriver driver) {

  255.             setValueSpanMap(driver);
  256.             setReferenceDate(driver.getReferenceDate());

  257.             // if any of the drivers is selected, all must be selected
  258.             if (isSelected()) {
  259.                 driver.setSelected(true);
  260.             } else {
  261.                 setSelected(driver.isSelected());
  262.             }

  263.             driver.addObserver(forwarder);
  264.             forwarder.add(driver);

  265.         }

  266.         /** Merge another instance.
  267.          * <p>
  268.          * After merging, the other instance is merely empty and preserved
  269.          * only as a child of the current instance. Changes are therefore
  270.          * still forwarded to it, but it is itself not responsible anymore
  271.          * for forwarding change.
  272.          * <p>
  273.          * </p>
  274.          * Warning if a driver is added and a driver with the same name
  275.          * was already added before, they should have the same validity
  276.          * periods (that is to say that the {@link
  277.          * ParameterDriver#addSpans(AbsoluteDate, AbsoluteDate, double)}
  278.          * and {@link ParameterDriver#addSpanAtDate(AbsoluteDate)} methods
  279.          * should have been called with same arguments for all drivers
  280.          * having the same name) to avoid surprises. Whatever, all driver having
  281.          * same name will have their valueSpanMap, nameSpanMap and validity period
  282.          * overwritten with the last driver added attributes.
  283.          * </p>
  284.          * @param other instance to merge
  285.          */
  286.         private void merge(final DelegatingDriver other) {

  287.             if (other.forwarder == forwarder) {
  288.                 // we are attempting to merge an instance with either itself
  289.                 // or an already embedded one, just ignore the request
  290.                 return;
  291.             }

  292.             // synchronize parameter
  293.             setValueSpanMap(other);
  294.             //setValue(other.getValue());
  295.             setReferenceDate(other.getReferenceDate());
  296.             if (isSelected()) {
  297.                 other.setSelected(true);
  298.             } else {
  299.                 setSelected(other.isSelected());
  300.             }

  301.             // move around drivers
  302.             for (final ParameterDriver otherDriver : other.forwarder.getDrivers()) {
  303.                 // as drivers are added one at a time and always refer back to a single
  304.                 // DelegatingDriver (through the ChangesForwarder), they cannot be
  305.                 // referenced by two different DelegatingDriver. We can blindly move
  306.                 // around all drivers, there cannot be any duplicates
  307.                 forwarder.add(otherDriver);
  308.                 otherDriver.replaceObserver(other.forwarder, forwarder);
  309.             }

  310.             // forwarding is now delegated to current instance
  311.             other.replaceObserver(other.forwarder, forwarder);
  312.             other.forwarder = forwarder;

  313.             // replace merged instance with current instance in former owners
  314.             for (final ParameterDriversList otherOwner : other.owners) {
  315.                 owners.add(otherOwner);
  316.                 otherOwner.replaceDelegating(other, this);
  317.             }

  318.         }

  319.         /** Get the raw drivers to which this one delegates.
  320.          * <p>
  321.          * These raw drivers all manage the same parameter name.
  322.          * </p>
  323.          * @return raw drivers to which this one delegates
  324.          */
  325.         public List<ParameterDriver> getRawDrivers() {
  326.             return Collections.unmodifiableList(forwarder.getDrivers());
  327.         }

  328.     }

  329.     /** Local observer for propagating changes, avoiding infinite recursion. */
  330.     private static class ChangesForwarder implements ParameterObserver {

  331.         /** DelegatingDriver we are associated with. */
  332.         private final DelegatingDriver delegating;

  333.         /** Drivers synchronized together by the instance. */
  334.         private final List<ParameterDriver> drivers;

  335.         /** Root of the current update chain. */
  336.         private ParameterDriver root;

  337.         /** Depth of the current update chain. */
  338.         private int depth;

  339.         /** Simple constructor.
  340.          * @param delegating delegatingDriver we are associated with
  341.          * @param driver first driver in the series
  342.          */
  343.         ChangesForwarder(final DelegatingDriver delegating, final ParameterDriver driver) {
  344.             this.delegating = delegating;
  345.             this.drivers    = new ArrayList<>();
  346.             drivers.add(driver);
  347.         }

  348.         /** Get the {@link DelegatingDriver} associated with this instance.
  349.          * @return {@link DelegatingDriver} associated with this instance
  350.          * @since 9.1
  351.          */
  352.         DelegatingDriver getDelegatingDriver() {
  353.             return delegating;
  354.         }

  355.         /** Add a driver to the list synchronized together by the instance.
  356.          * @param driver driver to add
  357.          * @since 10.1
  358.          */
  359.         void add(final ParameterDriver driver) {
  360.             drivers.add(driver);
  361.         }

  362.         /** Get the drivers synchronized together by the instance.
  363.          * @return drivers synchronized together by the instance.
  364.          * @since 10.1
  365.          */
  366.         public List<ParameterDriver> getDrivers() {
  367.             return drivers;
  368.         }

  369.         /** {@inheritDoc} */
  370.         @Override
  371.         public void valueSpanMapChanged(final TimeSpanMap<Double> previousValueSpanMap, final ParameterDriver driver) {
  372.             updateAll(driver, d -> d.setValueSpanMap(driver));
  373.         }

  374.         /** {@inheritDoc} */
  375.         @Override
  376.         public void valueChanged(final double previousValue, final ParameterDriver driver, final AbsoluteDate date) {
  377.             updateAll(driver, d -> d.setValue(driver.getValue(date), date));
  378.         }

  379.         /** {@inheritDoc} */
  380.         @Override
  381.         public void referenceDateChanged(final AbsoluteDate previousReferenceDate, final ParameterDriver driver) {
  382.             updateAll(driver, d -> d.setReferenceDate(driver.getReferenceDate()));
  383.         }

  384.         /** {@inheritDoc} */
  385.         @Override
  386.         public void nameChanged(final String previousName, final ParameterDriver driver) {
  387.             updateAll(driver, d -> d.setName(driver.getName()));
  388.         }

  389.         /** {@inheritDoc} */
  390.         @Override
  391.         public void selectionChanged(final boolean previousSelection, final ParameterDriver driver) {
  392.             updateAll(driver, d -> d.setSelected(driver.isSelected()));
  393.         }

  394.         /** {@inheritDoc} */
  395.         @Override
  396.         public void estimationTypeChanged(final boolean previousSelection, final ParameterDriver driver) {
  397.             updateAll(driver, d -> d.setContinuousEstimation(driver.isContinuousEstimation()));
  398.         }

  399.         /** {@inheritDoc} */
  400.         @Override
  401.         public void referenceValueChanged(final double previousReferenceValue, final ParameterDriver driver) {
  402.             updateAll(driver, d -> d.setReferenceValue(driver.getReferenceValue()));
  403.         }

  404.         /** {@inheritDoc} */
  405.         @Override
  406.         public void minValueChanged(final double previousMinValue, final ParameterDriver driver) {
  407.             updateAll(driver, d -> d.setMinValue(driver.getMinValue()));
  408.         }

  409.         /** {@inheritDoc} */
  410.         @Override
  411.         public void maxValueChanged(final double previousMaxValue, final ParameterDriver driver) {
  412.             updateAll(driver, d -> d.setMaxValue(driver.getMaxValue()));
  413.         }

  414.         /** {@inheritDoc} */
  415.         @Override
  416.         public void scaleChanged(final double previousScale, final ParameterDriver driver) {
  417.             updateAll(driver, d -> d.setScale(driver.getScale()));
  418.         }

  419.         /** Update all bound parameters.
  420.          * @param driver driver triggering the update
  421.          * @param updater updater to use
  422.          */
  423.         private void updateAll(final ParameterDriver driver, final Updater updater) {

  424.             final boolean firstCall = depth++ == 0;
  425.             if (firstCall) {
  426.                 root = driver;
  427.             }

  428.             if (driver == getDelegatingDriver()) {
  429.                 // propagate change downwards, which will trigger recursive calls
  430.                 for (final ParameterDriver d : drivers) {
  431.                     if (d != root) {
  432.                         updater.update(d);
  433.                     }
  434.                 }
  435.             } else if (firstCall) {
  436.                 // first call started from an underlying driver, propagate change upwards
  437.                 updater.update(getDelegatingDriver());
  438.             }

  439.             if (--depth == 0) {
  440.                 // this is the end of the root call
  441.                 root = null;
  442.             }

  443.         }

  444.     }

  445.     /** Interface for updating parameters. */
  446.     @FunctionalInterface
  447.     private interface Updater {
  448.         /** Update a driver.
  449.          * @param driver driver to update
  450.          */
  451.         void update(ParameterDriver driver);
  452.     }

  453. }