AbstractPropagatorBuilder.java
- /* Copyright 2002-2018 CS Systèmes d'Information
- * Licensed to CS Systèmes d'Information (CS) under one or more
- * contributor license agreements. See the NOTICE file distributed with
- * this work for additional information regarding copyright ownership.
- * CS licenses this file to You 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.
- */
- package org.orekit.propagation.conversion;
- import java.util.List;
- import org.hipparchus.exception.LocalizedCoreFormats;
- import org.hipparchus.util.FastMath;
- import org.orekit.errors.OrekitException;
- import org.orekit.errors.OrekitIllegalArgumentException;
- import org.orekit.forces.gravity.NewtonianAttraction;
- import org.orekit.frames.Frame;
- import org.orekit.orbits.Orbit;
- import org.orekit.orbits.OrbitType;
- import org.orekit.orbits.PositionAngle;
- import org.orekit.time.AbsoluteDate;
- import org.orekit.utils.ParameterDriver;
- import org.orekit.utils.ParameterDriversList;
- import org.orekit.utils.ParameterDriversList.DelegatingDriver;
- import org.orekit.utils.ParameterObserver;
- /** Base class for propagator builders.
- * @author Pascal Parraud
- * @since 7.1
- */
- public abstract class AbstractPropagatorBuilder implements PropagatorBuilder {
- /** Central attraction scaling factor.
- * <p>
- * We use a power of 2 to avoid numeric noise introduction
- * in the multiplications/divisions sequences.
- * </p>
- */
- private static final double MU_SCALE = FastMath.scalb(1.0, 32);
- /** Date of the initial orbit. */
- private AbsoluteDate initialOrbitDate;
- /** Frame in which the orbit is propagated. */
- private final Frame frame;
- /** Central attraction coefficient (m³/s²). */
- private double mu;
- /** Drivers for orbital parameters. */
- private final ParameterDriversList orbitalDrivers;
- /** List of the supported parameters. */
- private ParameterDriversList propagationDrivers;
- /** Orbit type to use. */
- private final OrbitType orbitType;
- /** Position angle type to use. */
- private final PositionAngle positionAngle;
- /** Position scale to use for the orbital drivers. */
- private final double positionScale;
- /** Build a new instance.
- * <p>
- * The template orbit is used as a model to {@link
- * #createInitialOrbit() create initial orbit}. It defines the
- * inertial frame, the central attraction coefficient, the orbit type, and is also
- * used together with the {@code positionScale} to convert from the {@link
- * ParameterDriver#setNormalizedValue(double) normalized} parameters used by the
- * callers of this builder to the real orbital parameters.
- * </p>
- * <p>
- * By default, all the {@link #getOrbitalParametersDrivers() orbital parameters drivers}
- * are selected, which means that if the builder is used for orbit determination or
- * propagator conversion, all orbital parameters will be estimated. If only a subset
- * of the orbital parameters must be estimated, caller must retrieve the orbital
- * parameters by calling {@link #getOrbitalParametersDrivers()} and then call
- * {@link ParameterDriver#setSelected(boolean) setSelected(false)}.
- * </p>
- * @param templateOrbit reference orbit from which real orbits will be built
- * @param positionAngle position angle type to use
- * @param positionScale scaling factor used for orbital parameters normalization
- * (typically set to the expected standard deviation of the position)
- * @param addDriverForCentralAttraction if true, a {@link ParameterDriver} should
- * be set up for central attraction coefficient
- * @exception OrekitException if parameters drivers cannot be scaled
- * @since 8.0
- */
- protected AbstractPropagatorBuilder(final Orbit templateOrbit, final PositionAngle positionAngle,
- final double positionScale, final boolean addDriverForCentralAttraction)
- throws OrekitException {
- this.initialOrbitDate = templateOrbit.getDate();
- this.frame = templateOrbit.getFrame();
- this.mu = templateOrbit.getMu();
- this.propagationDrivers = new ParameterDriversList();
- this.orbitType = templateOrbit.getType();
- this.positionAngle = positionAngle;
- this.positionScale = positionScale;
- this.orbitalDrivers = orbitType.getDrivers(positionScale, templateOrbit, positionAngle);
- for (final DelegatingDriver driver : orbitalDrivers.getDrivers()) {
- driver.setSelected(true);
- }
- if (addDriverForCentralAttraction) {
- final ParameterDriver muDriver = new ParameterDriver(NewtonianAttraction.CENTRAL_ATTRACTION_COEFFICIENT,
- mu, MU_SCALE, 0, Double.POSITIVE_INFINITY);
- muDriver.addObserver(new ParameterObserver() {
- /** {@inheridDoc} */
- @Override
- public void valueChanged(final double previousValue, final ParameterDriver driver) {
- AbstractPropagatorBuilder.this.mu = driver.getValue();
- }
- });
- propagationDrivers.add(muDriver);
- }
- }
- /** {@inheritDoc} */
- public OrbitType getOrbitType() {
- return orbitType;
- }
- /** {@inheritDoc} */
- public PositionAngle getPositionAngle() {
- return positionAngle;
- }
- /** {@inheritDoc} */
- public AbsoluteDate getInitialOrbitDate() {
- return initialOrbitDate;
- }
- /** {@inheritDoc} */
- public Frame getFrame() {
- return frame;
- }
- /** {@inheritDoc} */
- public ParameterDriversList getOrbitalParametersDrivers() {
- return orbitalDrivers;
- }
- /** {@inheritDoc} */
- public ParameterDriversList getPropagationParametersDrivers() {
- return propagationDrivers;
- }
- /** Get the position scale.
- * @return the position scale used to scale the orbital drivers
- */
- public double getPositionScale() {
- return positionScale;
- }
- /** Get the central attraction coefficient (µ - m³/s²) value.
- * @return the central attraction coefficient (µ - m³/s²) value
- * @since 9.2
- */
- public double getMu() {
- return mu;
- }
- /** Get the number of selected parameters.
- * @return number of selected parameters
- */
- private int getNbSelected() {
- int count = 0;
- // count orbital parameters
- for (final ParameterDriver driver : orbitalDrivers.getDrivers()) {
- if (driver.isSelected()) {
- ++count;
- }
- }
- // count propagation parameters
- for (final ParameterDriver driver : propagationDrivers.getDrivers()) {
- if (driver.isSelected()) {
- ++count;
- }
- }
- return count;
- }
- /** {@inheritDoc} */
- public double[] getSelectedNormalizedParameters() {
- // allocate array
- final double[] selected = new double[getNbSelected()];
- // fill data
- int index = 0;
- for (final ParameterDriver driver : orbitalDrivers.getDrivers()) {
- if (driver.isSelected()) {
- selected[index++] = driver.getNormalizedValue();
- }
- }
- for (final ParameterDriver driver : propagationDrivers.getDrivers()) {
- if (driver.isSelected()) {
- selected[index++] = driver.getNormalizedValue();
- }
- }
- return selected;
- }
- /** Build an initial orbit using the current selected parameters.
- * <p>
- * This method is a stripped down version of {@link #buildPropagator(double[])}
- * that only builds the initial orbit and not the full propagator.
- * </p>
- * @return an initial orbit
- * @since 8.0
- */
- protected Orbit createInitialOrbit() {
- final double[] unNormalized = new double[orbitalDrivers.getNbParams()];
- for (int i = 0; i < unNormalized.length; ++i) {
- unNormalized[i] = orbitalDrivers.getDrivers().get(i).getValue();
- }
- return getOrbitType().mapArrayToOrbit(unNormalized, null, positionAngle, initialOrbitDate, mu, frame);
- }
- /** Set the selected parameters.
- * @param normalizedParameters normalized values for the selected parameters
- * @exception OrekitException if some parameter cannot be set to the specified value
- * @exception OrekitIllegalArgumentException if the number of parameters is not the
- * number of selected parameters (adding orbits and models parameters)
- */
- protected void setParameters(final double[] normalizedParameters)
- throws OrekitException, OrekitIllegalArgumentException {
- if (normalizedParameters.length != getNbSelected()) {
- throw new OrekitIllegalArgumentException(LocalizedCoreFormats.DIMENSIONS_MISMATCH,
- normalizedParameters.length,
- getNbSelected());
- }
- int index = 0;
- // manage orbital parameters
- for (final ParameterDriver driver : orbitalDrivers.getDrivers()) {
- if (driver.isSelected()) {
- driver.setNormalizedValue(normalizedParameters[index++]);
- }
- }
- // manage propagation parameters
- for (final ParameterDriver driver : propagationDrivers.getDrivers()) {
- if (driver.isSelected()) {
- driver.setNormalizedValue(normalizedParameters[index++]);
- }
- }
- }
- /** Add a supported parameter.
- * @param driver driver for the parameter
- * @exception OrekitException if the name is already supported
- */
- protected void addSupportedParameter(final ParameterDriver driver)
- throws OrekitException {
- propagationDrivers.add(driver);
- propagationDrivers.sort();
- }
- /** Reset the orbit in the propagator builder.
- * @param newOrbit New orbit to set in the propagator builder
- * @exception OrekitException if a parameter observer throws an exception during reset
- */
- public void resetOrbit(final Orbit newOrbit)
- throws OrekitException {
- // Map the new orbit in an array of double
- final double[] orbitArray = new double[6];
- orbitType.mapOrbitToArray(newOrbit, getPositionAngle(), orbitArray, null);
- // Update all the orbital drivers, selected or unselected
- final List<DelegatingDriver> orbitalDriversList = getOrbitalParametersDrivers().getDrivers();
- for (int i = 0; i < 6; i++) {
- orbitalDriversList.get(i).setValue(orbitArray[i]);
- }
- // Change the initial orbit date in the builder
- this.initialOrbitDate = newOrbit.getDate();
- }
- }