blob: d0c7d77f4a2cf1073fd4e71ecdf2a8a6134cb67d [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF 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.apache.sis.referencing.operation;
import java.util.Map;
import java.util.HashMap;
import java.util.List;
import java.util.ArrayList;
import java.util.Collections;
import java.util.ListIterator;
import javax.measure.Unit;
import javax.measure.quantity.Time;
import javax.measure.IncommensurableException;
import org.opengis.util.FactoryException;
import org.opengis.referencing.IdentifiedObject;
import org.opengis.referencing.cs.*;
import org.opengis.referencing.crs.*;
import org.opengis.referencing.datum.*;
import org.opengis.referencing.operation.*;
import org.opengis.metadata.Identifier;
import org.opengis.metadata.quality.PositionalAccuracy;
import org.opengis.metadata.extent.GeographicBoundingBox;
import org.opengis.parameter.ParameterValueGroup;
import org.opengis.parameter.ParameterDescriptorGroup;
import org.apache.sis.internal.referencing.AxisDirections;
import org.apache.sis.internal.referencing.AnnotatedMatrix;
import org.apache.sis.internal.referencing.CoordinateOperations;
import org.apache.sis.internal.referencing.EllipsoidalHeightCombiner;
import org.apache.sis.internal.referencing.ReferencingUtilities;
import org.apache.sis.internal.referencing.provider.Affine;
import org.apache.sis.internal.referencing.provider.DatumShiftMethod;
import org.apache.sis.internal.referencing.provider.Geographic2Dto3D;
import org.apache.sis.internal.referencing.provider.Geographic3Dto2D;
import org.apache.sis.internal.referencing.provider.GeographicToGeocentric;
import org.apache.sis.internal.referencing.provider.GeocentricToGeographic;
import org.apache.sis.internal.referencing.provider.GeocentricAffine;
import org.apache.sis.internal.referencing.SpecializedOperationFactory;
import org.apache.sis.internal.referencing.Resources;
import org.apache.sis.internal.util.Constants;
import org.apache.sis.measure.Units;
import org.apache.sis.metadata.iso.citation.Citations;
import org.apache.sis.metadata.iso.extent.Extents;
import org.apache.sis.parameter.TensorParameters;
import org.apache.sis.referencing.CRS;
import org.apache.sis.referencing.CommonCRS;
import org.apache.sis.referencing.IdentifiedObjects;
import org.apache.sis.referencing.NamedIdentifier;
import org.apache.sis.referencing.cs.CoordinateSystems;
import org.apache.sis.referencing.datum.BursaWolfParameters;
import org.apache.sis.referencing.datum.DefaultGeodeticDatum;
import org.apache.sis.referencing.operation.matrix.Matrices;
import org.apache.sis.referencing.operation.transform.DefaultMathTransformFactory;
import org.apache.sis.util.ArgumentChecks;
import org.apache.sis.util.resources.Vocabulary;
import static org.apache.sis.util.Utilities.equalsIgnoreMetadata;
/**
* Finds a conversion or transformation path from a source CRS to a target CRS.
* This class implements two strategies for searching the coordinate operation:
*
* <ol class="verbose">
* <li>When <code>{@linkplain #createOperation createOperation}(sourceCRS, targetCRS)</code> is invoked,
* this class first {@linkplain org.apache.sis.referencing.factory.IdentifiedObjectFinder tries to
* find the authority codes} for the given source and target CRS. If such codes are found, they are
* {@linkplain org.apache.sis.referencing.factory.GeodeticAuthorityFactory#createFromCoordinateReferenceSystemCodes
* submitted to a registry of coordinate operations}. If an operation is found, it will be returned.
*
* <div class="note"><b>Note:</b> the above is known as the <cite>late-binding</cite> approach.
* The late-binding approach allows the authority to define better suited operations than what
* we would get if we were transforming everything from and to a pivot system (e.g. WGS84).
* In addition, this approach provides useful information like the coordinate operation
* {@linkplain AbstractCoordinateOperation#getScope() scope} and
* {@linkplain AbstractCoordinateOperation#getDomainOfValidity() domain of validity},
* {@linkplain AbstractCoordinateOperation#getCoordinateOperationAccuracy() accuracy}.</div>
* </li>
* <li>If the above authority factory does not know about the specified CRS, then this class tries to
* infer the coordinate operation by itself. The CRS type is examined and the work is dispatched
* to one or many of the {@code createOperationStep(…)} protected methods defined in this class.
* Those methods use properties associated to the CRS, including {@code BOUNDCRS} or {@code TOWGS84}
* elements found in <cite>Well Known Text</cite> (WKT).
*
* <div class="note"><b>Note:</b> the use of elements like {@code TOWGS84} is known as the
* <cite>early-binding</cite> approach. The operation found by this approach may be sub-optimal.
* The early-binding approach is used only as a fallback when the late-binding approach gave no result.</div>
* </li>
* </ol>
*
* <h2>Customization</h2>
* Instances of this class are created by {@link DefaultCoordinateOperationFactory}.
* The only public method is {@link #createOperation(CoordinateReferenceSystem, CoordinateReferenceSystem)},
* which dispatches its work to the {@code createOperationStep(…)} protected methods.
* Developers can override those protected methods if they want to alter the way some operations are created.
*
* <h2>Limitations</h2>
* <ul>
* <li>Each instance of this class shall be used only once.</li>
* <li>This class is not thread-safe. A new instance shall be created for each coordinate operation to infer.</li>
* </ul>
*
* @author Martin Desruisseaux (Geomatys)
* @version 1.0
*
* @see DefaultCoordinateOperationFactory#createOperation(CoordinateReferenceSystem, CoordinateReferenceSystem, CoordinateOperationContext)
*
* @since 0.7
* @module
*/
public class CoordinateOperationFinder extends CoordinateOperationRegistry {
/**
* Identifiers used as the basis for identifier of CRS used as an intermediate step.
* The values can be of two kinds:
*
* <ul>
* <li>If the value is an instance of {@link Integer}, then this is the number
* of identifiers derived from the identifier associated to the key.</li>
* <li>Otherwise the key is itself an {@link Identifier} derived from another
* identifier, and the value is that identifier.</li>
* </ul>
*
* @see #derivedFrom(IdentifiedObject)
*/
private final Map<Identifier,Object> identifierOfStepCRS;
/**
* The pair of source and target CRS for which we already searched a coordinate operation.
* This is used as a safety against infinite recursivity.
*/
private final Map<CRSPair,Boolean> previousSearches;
/**
* Whether this finder instance is allowed to use {@link DefaultCoordinateOperationFactory#cache}.
*/
private final boolean useCache;
/**
* Creates a new instance for the given factory and context.
*
* @param registry the factory to use for creating operations as defined by authority, or {@code null} if none.
* @param factory the factory to use for creating operations not found in the registry.
* @param context the area of interest and desired accuracy, or {@code null} if none.
* @throws FactoryException if an error occurred while initializing this {@code CoordinateOperationFinder}.
*
* @see DefaultCoordinateOperationFactory#createOperationFinder(CoordinateOperationAuthorityFactory, CoordinateOperationContext)
*/
public CoordinateOperationFinder(final CoordinateOperationAuthorityFactory registry,
final CoordinateOperationFactory factory,
final CoordinateOperationContext context) throws FactoryException
{
super(registry, factory, context);
identifierOfStepCRS = new HashMap<>(8);
previousSearches = new HashMap<>(8);
useCache = (context == null) && (factory == factorySIS);
}
/**
* Infers an operation for conversion or transformation between two coordinate reference systems.
* If a non-null authority factory – the <cite>registry</cite> – has been specified at construction time,
* then this method will first query that factory (<cite>late-binding</cite> approach – see class javadoc).
* If no operation has been found in the registry or if no registry has been specified to the constructor,
* this method inspects the given CRS and delegates the work to one or many {@code createOperationStep(…)}
* methods (<cite>early-binding</cite> approach).
*
* <p>The default implementation invokes <code>{@linkplain #createOperations createOperations}(sourceCRS,
* targetCRS)</code>, then returns the first operation in the returned list or throws an exception if the
* list is empty.</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws OperationNotFoundException if no operation path was found from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation creation failed for some other reason.
*/
public CoordinateOperation createOperation(final CoordinateReferenceSystem sourceCRS,
final CoordinateReferenceSystem targetCRS)
throws OperationNotFoundException, FactoryException
{
final boolean oldState = stopAtFirst;
stopAtFirst = true;
final List<CoordinateOperation> operations = createOperations(sourceCRS, targetCRS);
stopAtFirst = oldState;
if (!operations.isEmpty()) {
return operations.get(0);
}
throw new OperationNotFoundException(notFoundMessage(sourceCRS, targetCRS));
}
/**
* Infers operations for conversions or transformations between two coordinate reference systems.
* If a non-null authority factory – the <cite>registry</cite> – has been specified at construction time,
* then this method will first query that factory (<cite>late-binding</cite> approach – see class javadoc).
* If no operation has been found in the registry or if no registry has been specified to the constructor,
* this method inspects the given CRS and delegates the work to one or many {@code createOperationStep(…)}
* methods (<cite>early-binding</cite> approach).
*
* <p>At first, this method is invoked with the {@code sourceCRS} and {@code targetCRS} arguments given to the
* {@link DefaultCoordinateOperationFactory#createOperation(CoordinateReferenceSystem, CoordinateReferenceSystem,
* CoordinateOperationContext) CoordinateOperationFactory.createOperation(…)} method. But then, this method may
* be invoked recursively by some {@code createOperationStep(…)} methods with different source or target CRS,
* for example in order to process the {@linkplain org.apache.sis.referencing.crs.DefaultProjectedCRS#getBaseCRS()
* base geographic CRS} of a projected CRS.</p>
*
* <p>Coordinate operations are returned in preference order: best operations for the area of interest should be first.
* The returned list is modifiable: callers can add, remove or set elements without impact on this
* {@code CoordinateOperationFinder} instance.</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return coordinate operations from {@code sourceCRS} to {@code targetCRS}.
* @throws OperationNotFoundException if no operation path was found from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation creation failed for some other reason.
*
* @since 1.0
*/
@Override
public List<CoordinateOperation> createOperations(final CoordinateReferenceSystem sourceCRS,
final CoordinateReferenceSystem targetCRS)
throws FactoryException
{
ArgumentChecks.ensureNonNull("sourceCRS", sourceCRS);
ArgumentChecks.ensureNonNull("targetCRS", targetCRS);
if (equalsIgnoreMetadata(sourceCRS, targetCRS)) try {
return asList(createFromAffineTransform(AXIS_CHANGES, sourceCRS, targetCRS,
CoordinateSystems.swapAndScaleAxes(sourceCRS.getCoordinateSystem(),
targetCRS.getCoordinateSystem())));
} catch (IllegalArgumentException | IncommensurableException e) {
throw new FactoryException(Resources.format(Resources.Keys.CanNotInstantiateGeodeticObject_1, new CRSPair(sourceCRS, targetCRS)), e);
}
/*
* If this method is invoked recursively, verify if the requested operation is already in the cache.
* We do not perform this verification on the first invocation because it was already verified by
* DefaultCoordinateOperationFactory.createOperation(…). We do not block if the operation is in
* process of being computed in another thread because of the risk of deadlock. If the operation
* is not in the cache, store the key in our internal map for preventing infinite recursivity.
*/
final CRSPair key = new CRSPair(sourceCRS, targetCRS);
if (useCache && stopAtFirst && !previousSearches.isEmpty()) {
final CoordinateOperation op = factorySIS.cache.peek(key);
if (op != null) return asList(op); // Must be a modifiable list as per this method contract.
}
if (previousSearches.put(key, Boolean.TRUE) != null) {
throw new FactoryException(Resources.format(Resources.Keys.RecursiveCreateCallForCode_2, CoordinateOperation.class, key));
}
/*
* If the user did not specified an area of interest, use the domain of validity of the CRS.
*/
GeographicBoundingBox bbox = Extents.getGeographicBoundingBox(areaOfInterest);
if (bbox == null) {
bbox = Extents.intersection(CRS.getGeographicBoundingBox(sourceCRS),
CRS.getGeographicBoundingBox(targetCRS));
areaOfInterest = CoordinateOperationContext.setGeographicBoundingBox(areaOfInterest, bbox);
}
/*
* Verify if some extension module handles this pair of CRS in a special way. For example it may
* be the "sis-gdal" module checking if the given CRS are wrappers around Proj.4 data structure.
*/
{ // For keeping 'operations' list locale.
final List<CoordinateOperation> operations = new ArrayList<>();
for (final SpecializedOperationFactory sp : factorySIS.getSpecializedFactories()) {
for (final CoordinateOperation op : sp.findOperations(sourceCRS, targetCRS)) {
if (filter(op)) {
operations.add(op);
}
}
}
if (!operations.isEmpty()) {
CoordinateOperationSorter.sort(operations, bbox);
return operations;
}
}
/*
* Verify in the EPSG dataset if the operation is explicitly defined by an authority.
*/
if (registry != null) {
final List<CoordinateOperation> authoritatives = super.createOperations(sourceCRS, targetCRS);
if (!authoritatives.isEmpty()) return authoritatives;
}
////////////////////////////////////////////////////////////////////////////////
//// ////
//// Derived → any Single CRS ////
//// ////
////////////////////////////////////////////////////////////////////////////////
if (sourceCRS instanceof GeneralDerivedCRS) {
final GeneralDerivedCRS source = (GeneralDerivedCRS) sourceCRS;
if (targetCRS instanceof GeneralDerivedCRS) {
return createOperationStep(source, (GeneralDerivedCRS) targetCRS);
}
if (targetCRS instanceof SingleCRS) {
return createOperationStep(source, (SingleCRS) targetCRS);
}
}
////////////////////////////////////////////////////////////////////////////////
//// ////
//// any Single CRS → Derived ////
//// ////
////////////////////////////////////////////////////////////////////////////////
if (targetCRS instanceof GeneralDerivedCRS) {
final GeneralDerivedCRS target = (GeneralDerivedCRS) targetCRS;
if (sourceCRS instanceof SingleCRS) {
return createOperationStep((SingleCRS) sourceCRS, target);
}
}
////////////////////////////////////////////////////////////////////////////////
//// ////
//// Geodetic → Geocetric, Geographic or Projected ////
//// ////
////////////////////////////////////////////////////////////////////////////////
if (sourceCRS instanceof GeodeticCRS) {
final GeodeticCRS source = (GeodeticCRS) sourceCRS;
if (targetCRS instanceof GeodeticCRS) {
return createOperationStep(source, (GeodeticCRS) targetCRS);
}
if (targetCRS instanceof VerticalCRS) {
return createOperationStep(source, (VerticalCRS) targetCRS);
}
}
////////////////////////////////////////////////////////////////////////////////
//// ////
//// Vertical → Vertical ////
//// ////
////////////////////////////////////////////////////////////////////////////////
if (sourceCRS instanceof VerticalCRS) {
final VerticalCRS source = (VerticalCRS) sourceCRS;
if (targetCRS instanceof VerticalCRS) {
return createOperationStep(source, (VerticalCRS) targetCRS);
}
}
////////////////////////////////////////////////////////////////////////////////
//// ////
//// Temporal → Temporal ////
//// ////
////////////////////////////////////////////////////////////////////////////////
if (sourceCRS instanceof TemporalCRS) {
final TemporalCRS source = (TemporalCRS) sourceCRS;
if (targetCRS instanceof TemporalCRS) {
return createOperationStep(source, (TemporalCRS) targetCRS);
}
}
////////////////////////////////////////////////////////////////////////////////
//// ////
//// Compound ↔ various CRS ////
//// ////
////////////////////////////////////////////////////////////////////////////////
if (sourceCRS instanceof CompoundCRS || targetCRS instanceof CompoundCRS) {
return createOperationStep(sourceCRS, CRS.getSingleComponents(sourceCRS),
targetCRS, CRS.getSingleComponents(targetCRS));
}
throw new OperationNotFoundException(notFoundMessage(sourceCRS, targetCRS));
}
/**
* Creates operations from an arbitrary single CRS to a derived coordinate reference system.
* Conversions from {@code GeographicCRS} to {@code ProjectedCRS} are also handled by this method,
* since projected CRS are a special kind of {@code GeneralDerivedCRS}.
*
* <p>The default implementation constructs the following operation chain:</p>
* <blockquote><code>sourceCRS → {@linkplain GeneralDerivedCRS#getBaseCRS() baseCRS} → targetCRS</code></blockquote>
*
* where the conversion from {@code baseCRS} to {@code targetCRS} is obtained from
* <code>targetCRS.{@linkplain GeneralDerivedCRS#getConversionFromBase() getConversionFromBase()}</code>.
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return coordinate operations from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
protected List<CoordinateOperation> createOperationStep(final SingleCRS sourceCRS,
final GeneralDerivedCRS targetCRS)
throws FactoryException
{
final List<CoordinateOperation> operations = createOperations(sourceCRS, targetCRS.getBaseCRS());
final ListIterator<CoordinateOperation> it = operations.listIterator();
if (it.hasNext()) {
final CoordinateOperation step2 = targetCRS.getConversionFromBase();
do {
final CoordinateOperation step1 = it.next();
it.set(concatenate(step1, step2));
} while (it.hasNext());
}
return operations;
}
/**
* Creates an operation from a derived CRS to an arbitrary single coordinate reference system.
* Conversions from {@code ProjectedCRS} to {@code GeographicCRS} are also handled by this method,
* since projected CRS are a special kind of {@code GeneralDerivedCRS}.
*
* <p>The default implementation constructs the following operation chain:</p>
* <blockquote><code>sourceCRS → {@linkplain GeneralDerivedCRS#getBaseCRS() baseCRS} → targetCRS</code></blockquote>
*
* where the conversion from {@code sourceCRS} to {@code baseCRS} is obtained from the inverse of
* <code>sourceCRS.{@linkplain GeneralDerivedCRS#getConversionFromBase() getConversionFromBase()}</code>.
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
protected List<CoordinateOperation> createOperationStep(final GeneralDerivedCRS sourceCRS,
final SingleCRS targetCRS)
throws FactoryException
{
final List<CoordinateOperation> operations = createOperations(sourceCRS.getBaseCRS(), targetCRS);
final ListIterator<CoordinateOperation> it = operations.listIterator();
if (it.hasNext()) {
final CoordinateOperation step1;
try {
step1 = inverse(sourceCRS.getConversionFromBase());
} catch (OperationNotFoundException exception) {
throw exception;
} catch (FactoryException | NoninvertibleTransformException exception) {
throw new OperationNotFoundException(canNotInvert(sourceCRS), exception);
}
do {
final CoordinateOperation step2 = it.next();
it.set(concatenate(step1, step2));
} while (it.hasNext());
}
return operations;
}
/**
* Creates an operation between two derived coordinate reference systems.
* The default implementation performs three steps:
*
* <ol>
* <li>Convert from {@code sourceCRS} to its base CRS.</li>
* <li>Convert the source base CRS to target base CRS.</li>
* <li>Convert from the target base CRS to the {@code targetCRS}.</li>
* </ol>
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
protected List<CoordinateOperation> createOperationStep(final GeneralDerivedCRS sourceCRS,
final GeneralDerivedCRS targetCRS)
throws FactoryException
{
final List<CoordinateOperation> operations = createOperations(sourceCRS.getBaseCRS(), targetCRS.getBaseCRS());
final ListIterator<CoordinateOperation> it = operations.listIterator();
if (it.hasNext()) {
final CoordinateOperation step3 = targetCRS.getConversionFromBase();
final CoordinateOperation step1;
try {
step1 = inverse(sourceCRS.getConversionFromBase());
} catch (OperationNotFoundException exception) {
throw exception;
} catch (FactoryException | NoninvertibleTransformException exception) {
throw new OperationNotFoundException(canNotInvert(sourceCRS), exception);
}
do {
final CoordinateOperation step2 = it.next();
it.set(concatenate(step1, step2, step3));
} while (it.hasNext());
}
return operations;
}
/**
* Creates an operation between two geodetic (geographic or geocentric) coordinate reference systems.
* The default implementation can:
*
* <ul>
* <li>adjust axis order and orientation, for example converting from (<cite>North</cite>, <cite>West</cite>)
* axes to (<cite>East</cite>, <cite>North</cite>) axes,</li>
* <li>apply units conversion if needed,</li>
* <li>perform longitude rotation if needed,</li>
* <li>perform datum shift if {@linkplain BursaWolfParameters Bursa-Wolf parameters} are available
* for the area of interest.</li>
* </ul>
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
@SuppressWarnings("null")
protected List<CoordinateOperation> createOperationStep(final GeodeticCRS sourceCRS,
final GeodeticCRS targetCRS)
throws FactoryException
{
final GeodeticDatum sourceDatum = sourceCRS.getDatum();
final GeodeticDatum targetDatum = targetCRS.getDatum();
Matrix datumShift = null;
/*
* If the prime meridian is not the same, we will concatenate a longitude rotation before or after datum shift
* (that concatenation will be performed by the customized DefaultMathTransformFactory.Context created below).
* Actually we do not know if the longitude rotation should be before or after datum shift. But this ambiguity
* can usually be ignored because Bursa-Wolf parameters are always used with source and target prime meridians
* set to Greenwich in EPSG dataset 8.9. For safety, the SIS's DefaultGeodeticDatum class ensures that if the
* prime meridians are not the same, then the target meridian must be Greenwich.
*/
final DefaultMathTransformFactory.Context context = ReferencingUtilities.createTransformContext(
sourceCRS, targetCRS, new MathTransformContext(sourceDatum, targetDatum));
/*
* If both CRS use the same datum and the same prime meridian, then the coordinate operation is only axis
* swapping, unit conversion or change of coordinate system type (Ellipsoidal ↔ Cartesian ↔ Spherical).
* Otherwise (if the datum are not the same), we will need to perform a scale, translation and rotation
* in Cartesian space using the Bursa-Wolf parameters. If the user does not require the best accuracy,
* then the Molodensky approximation may be used for avoiding the conversion step to geocentric CRS.
*/
Identifier identifier;
boolean isGeographicToGeocentric = false;
final CoordinateSystem sourceCS = context.getSourceCS();
final CoordinateSystem targetCS = context.getTargetCS();
if (equalsIgnoreMetadata(sourceDatum, targetDatum)) {
final boolean isGeocentricToGeographic;
isGeographicToGeocentric = (sourceCS instanceof EllipsoidalCS && targetCS instanceof CartesianCS);
isGeocentricToGeographic = (sourceCS instanceof CartesianCS && targetCS instanceof EllipsoidalCS);
/*
* Above booleans should never be true in same time. If it nevertheless happen (we are paranoiac;
* maybe a lazy user implemented all interfaces in a single class), do not apply any geographic ↔
* geocentric conversion. Instead do as if the coordinate system types were the same.
*/
if (isGeocentricToGeographic ^ isGeographicToGeocentric) {
identifier = GEOCENTRIC_CONVERSION;
} else {
identifier = AXIS_CHANGES;
}
} else {
identifier = ELLIPSOID_CHANGE;
if (sourceDatum instanceof DefaultGeodeticDatum) {
datumShift = ((DefaultGeodeticDatum) sourceDatum).getPositionVectorTransformation(targetDatum, areaOfInterest);
if (datumShift != null) {
identifier = DATUM_SHIFT;
}
}
}
/*
* Conceptually, all transformations below could done by first converting from the source coordinate
* system to geocentric Cartesian coordinates (X,Y,Z), apply an affine transform represented by the
* datum shift matrix, then convert from the (X′,Y′,Z′) coordinates to the target coordinate system.
* However there is two exceptions to this path:
*
* 1) In the particular where both the source and target CS are ellipsoidal, we may use the
* Molodensky approximation as a shortcut (if the desired accuracy allows).
*
* 2) Even if we really go through the XYZ coordinates without Molodensky approximation, there is
* at least 9 different ways to name this operation depending on whether the source and target
* CRS are geocentric or geographic, 2- or 3-dimensional, whether there is a translation or not,
* the rotation sign, etc. We try to use the most specific name if we can find one, and fallback
* on an arbitrary name only in last resort.
*/
final DefaultMathTransformFactory mtFactory = factorySIS.getDefaultMathTransformFactory();
MathTransform before = null, after = null;
ParameterValueGroup parameters;
if (identifier == DATUM_SHIFT || identifier == ELLIPSOID_CHANGE) {
/*
* If the transform can be represented by a single coordinate operation, returns that operation.
* Possible operations are:
*
* - Position Vector transformation (in geocentric, geographic-2D or geographic-3D domains)
* - Geocentric translation (in geocentric, geographic-2D or geographic-3D domains)
* - [Abridged] Molodensky (as an approximation of geocentric translation)
* - Identity (if the desired accuracy is so large than we can skip datum shift)
*
* TODO: if both CS are ellipsoidal but with different number of dimensions, then we should use
* an intermediate 3D geographic CRS in order to enable the use of Molodensky method if desired.
*/
final DatumShiftMethod preferredMethod = DatumShiftMethod.forAccuracy(desiredAccuracy);
parameters = GeocentricAffine.createParameters(sourceCS, targetCS, datumShift, preferredMethod);
if (parameters == null) {
/*
* Failed to select a coordinate operation. Maybe because the coordinate system types are not the same.
* Convert unconditionally to XYZ geocentric coordinates and apply the datum shift in that CS space.
*
* TODO: operation name should not be "Affine" if 'before' or 'after' transforms are not identity.
* Reminder: the parameter group name here determines the OperationMethod later in this method.
*
* See https://issues.apache.org/jira/browse/SIS-462
*/
if (datumShift != null) {
parameters = TensorParameters.WKT1.createValueGroup(properties(Constants.AFFINE), datumShift);
} else {
parameters = Affine.identity(3); // Dimension of geocentric CRS.
}
final CoordinateSystem normalized = CommonCRS.WGS84.geocentric().getCoordinateSystem();
before = mtFactory.createCoordinateSystemChange(sourceCS, normalized, sourceDatum.getEllipsoid());
after = mtFactory.createCoordinateSystemChange(normalized, targetCS, targetDatum.getEllipsoid());
context.setSource(normalized);
context.setTarget(normalized);
}
} else if (identifier == GEOCENTRIC_CONVERSION) {
/*
* Geographic ↔︎ Geocentric conversion. The "dim" parameter is Apache SIS specific but is guaranteed
* to be present since we use SIS parameter descriptors directly. The default number of dimension is 3,
* but we specify the value unconditionally anyway as a safety.
*/
final ParameterDescriptorGroup descriptor;
final GeodeticCRS geographic;
if (isGeographicToGeocentric) {
geographic = sourceCRS;
descriptor = GeographicToGeocentric.PARAMETERS;
} else {
geographic = targetCRS;
descriptor = GeocentricToGeographic.PARAMETERS;
}
parameters = descriptor.createValue();
parameters.parameter("dim").setValue(geographic.getCoordinateSystem().getDimension());
} else {
/*
* Coordinate system change (including change in the number of dimensions) without datum shift.
*/
final int sourceDim = sourceCS.getDimension();
final int targetDim = targetCS.getDimension();
if ((sourceDim & ~1) == 2 // sourceDim == 2 or 3.
&& (sourceDim ^ targetDim) == 1 // abs(sourceDim - targetDim) == 1.
&& (sourceCS instanceof EllipsoidalCS)
&& (targetCS instanceof EllipsoidalCS))
{
parameters = (sourceDim == 2 ? Geographic2Dto3D.PARAMETERS
: Geographic3Dto2D.PARAMETERS).createValue();
} else {
/*
* TODO: instead than creating parameters for an identity operation, we should create the
* CoordinateOperation directly from the MathTransform created by mtFactory below.
* The intent if to get the correct OperationMethod, which should not be "Affine"
* if there is a CS type change.
*/
parameters = Affine.identity(targetDim);
/*
* createCoordinateSystemChange(…) needs the ellipsoid associated to the ellipsoidal coordinate system,
* if any. If none or both coordinate systems are ellipsoidal, then the ellipsoid will be ignored (see
* createCoordinateSystemChange(…) javadoc for the rational) so it does not matter which one we pick.
*/
before = mtFactory.createCoordinateSystemChange(sourceCS, targetCS,
(sourceCS instanceof EllipsoidalCS ? sourceDatum : targetDatum).getEllipsoid());
context.setSource(targetCS);
}
}
/*
* Transform between differents datums using Bursa Wolf parameters. The Bursa Wolf parameters are used
* with "standard" geocentric CS, i.e. with X axis towards the prime meridian, Y axis towards East and
* Z axis toward North, unless the Molodensky approximation is used. The following steps are applied:
*
* source CRS →
* normalized CRS with source datum →
* normalized CRS with target datum →
* target CRS
*
* Those steps may be either explicit with the 'before' and 'after' transform, or implicit with the
* Context parameter.
*/
MathTransform transform = mtFactory.createParameterizedTransform(parameters, context);
final OperationMethod method = mtFactory.getLastMethodUsed();
if (before != null) {
transform = mtFactory.createConcatenatedTransform(before, transform);
if (after != null) {
transform = mtFactory.createConcatenatedTransform(transform, after);
}
}
/*
* Adjust the accuracy information if the datum shift has been computed by an indirect path.
* The indirect path uses a third datum (typically WGS84) as an intermediate between the two
* specified datum.
*/
final Map<String, Object> properties = properties(identifier);
if (datumShift instanceof AnnotatedMatrix) {
properties.put(CoordinateOperation.COORDINATE_OPERATION_ACCURACY_KEY, new PositionalAccuracy[] {
((AnnotatedMatrix) datumShift).accuracy
});
}
return asList(createFromMathTransform(properties, sourceCRS, targetCRS, transform, method, parameters, null));
}
/**
* Creates an operation between a geodetic and a vertical coordinate reference systems.
* The height returned by this method will usually be part of a
* {@linkplain DefaultPassThroughOperation pass-through operation}.
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
protected List<CoordinateOperation> createOperationStep(final GeodeticCRS sourceCRS,
final VerticalCRS targetCRS)
throws FactoryException
{
/*
* We will perform the conversion or transformation as a 3 steps process:
*
* source CRS →
* interpolation CRS →
* ellipsoidal height →
* target height
*/
CoordinateOperation step1 = null;
CoordinateOperation step2;
CoordinateOperation step3 = null;
/*
* Convert the source CRS to the CRS needed for transforming the heights.
* For now this step is fixed to a three-dimensional geographic CRS, but
* a future version should use a plugin-mechanism, with the code below
* as the last fallback.
*/
CoordinateReferenceSystem interpolationCRS = sourceCRS;
CoordinateSystem interpolationCS = interpolationCRS.getCoordinateSystem();
if (!(interpolationCS instanceof EllipsoidalCS)) {
final EllipsoidalCS cs = CommonCRS.WGS84.geographic3D().getCoordinateSystem();
if (!equalsIgnoreMetadata(interpolationCS, cs)) {
final GeographicCRS stepCRS = factorySIS.getCRSFactory()
.createGeographicCRS(derivedFrom(sourceCRS), sourceCRS.getDatum(), cs);
step1 = createOperation(sourceCRS, toAuthorityDefinition(GeographicCRS.class, stepCRS));
interpolationCRS = step1.getTargetCRS();
interpolationCS = interpolationCRS.getCoordinateSystem();
}
}
/*
* Transform from ellipsoidal height to the height requested by the caller.
* This operation requires the horizontal components (φ,λ) of source CRS,
* unless the user asked for ellipsooidal height (which strictly speaking
* is not allowed by ISO 19111). Those horizontal components are given by
* the interpolation CRS.
*
* TODO: store the interpolationCRS in some field for allowing other methods to use it.
*/
final int i = AxisDirections.indexOfColinear(interpolationCS, AxisDirection.UP);
if (i < 0) {
throw new OperationNotFoundException(notFoundMessage(sourceCRS, targetCRS));
}
final CoordinateSystemAxis expectedAxis = interpolationCS.getAxis(i);
final boolean isEllipsoidalHeight; // Whether heightCRS is okay or need to be recreated.
VerticalCRS heightCRS = targetCRS; // First candidate, will be replaced if it doesn't fit.
VerticalCS heightCS = heightCRS.getCoordinateSystem();
if (equalsIgnoreMetadata(heightCS.getAxis(0), expectedAxis)) {
isEllipsoidalHeight = ReferencingUtilities.isEllipsoidalHeight(heightCRS.getDatum());
} else {
heightCRS = CommonCRS.Vertical.ELLIPSOIDAL.crs();
heightCS = heightCRS.getCoordinateSystem();
isEllipsoidalHeight = equalsIgnoreMetadata(heightCS.getAxis(0), expectedAxis);
if (!isEllipsoidalHeight) {
heightCS = toAuthorityDefinition(VerticalCS.class, factorySIS.getCSFactory()
.createVerticalCS(derivedFrom(heightCS), expectedAxis));
}
}
if (!isEllipsoidalHeight) { // 'false' if we need to change datum, unit or axis direction.
heightCRS = toAuthorityDefinition(VerticalCRS.class, factorySIS.getCRSFactory()
.createVerticalCRS(derivedFrom(heightCRS), CommonCRS.Vertical.ELLIPSOIDAL.datum(), heightCS));
}
if (heightCRS != targetCRS) {
step3 = createOperation(heightCRS, targetCRS); // May need interpolationCRS for performing datum change.
heightCRS = (VerticalCRS) step3.getSourceCRS();
heightCS = heightCRS.getCoordinateSystem();
}
/*
* Conversion from three-dimensional geographic CRS to ellipsoidal height.
* This part does nothing more than dropping the horizontal components,
* like the "Geographic3D to 2D conversion" (EPSG:9659).
* It is not the job of this block to perform unit conversions.
* Unit conversions, if needed, are done by 'step3' computed in above block.
*
* The "Geographic3DtoVertical.txt" file in the provider package is a reminder.
* If this policy is changed, that file should be edited accordingly.
*/
final int srcDim = interpolationCS.getDimension(); // Should always be 3.
final int tgtDim = heightCS.getDimension(); // Should always be 1.
final Matrix matrix = Matrices.createZero(tgtDim + 1, srcDim + 1);
matrix.setElement(0, i, 1); // Scale factor for height.
matrix.setElement(tgtDim, srcDim, 1); // Always 1 for affine transform.
step2 = createFromAffineTransform(AXIS_CHANGES, interpolationCRS, heightCRS, matrix);
return asList(concatenate(step1, step2, step3));
}
/**
* Creates an operation between two vertical coordinate reference systems.
* The default implementation checks if both CRS use the same datum, then
* adjusts for axis direction and units.
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*
* @todo Needs to implement vertical datum shift.
*/
protected List<CoordinateOperation> createOperationStep(final VerticalCRS sourceCRS,
final VerticalCRS targetCRS)
throws FactoryException
{
final VerticalDatum sourceDatum = sourceCRS.getDatum();
final VerticalDatum targetDatum = targetCRS.getDatum();
if (!equalsIgnoreMetadata(sourceDatum, targetDatum)) {
throw new OperationNotFoundException(notFoundMessage(sourceDatum, targetDatum));
}
final VerticalCS sourceCS = sourceCRS.getCoordinateSystem();
final VerticalCS targetCS = targetCRS.getCoordinateSystem();
final Matrix matrix;
try {
matrix = CoordinateSystems.swapAndScaleAxes(sourceCS, targetCS);
} catch (IllegalArgumentException | IncommensurableException exception) {
throw new OperationNotFoundException(notFoundMessage(sourceCRS, targetCRS), exception);
}
return asList(createFromAffineTransform(AXIS_CHANGES, sourceCRS, targetCRS, matrix));
}
/**
* Creates an operation between two temporal coordinate reference systems.
* The default implementation checks if both CRS use the same datum, then
* adjusts for axis direction, units and epoch.
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param targetCRS output coordinate reference system.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
protected List<CoordinateOperation> createOperationStep(final TemporalCRS sourceCRS,
final TemporalCRS targetCRS)
throws FactoryException
{
final TemporalDatum sourceDatum = sourceCRS.getDatum();
final TemporalDatum targetDatum = targetCRS.getDatum();
final TimeCS sourceCS = sourceCRS.getCoordinateSystem();
final TimeCS targetCS = targetCRS.getCoordinateSystem();
/*
* Compute the epoch shift. The epoch is the time "0" in a particular coordinate reference system.
* For example, the epoch for java.util.Date object is january 1, 1970 at 00:00 UTC. We compute how
* much to add to a time in 'sourceCRS' in order to get a time in 'targetCRS'. This "epoch shift" is
* in units of 'targetCRS'.
*/
final Unit<Time> targetUnit = targetCS.getAxis(0).getUnit().asType(Time.class);
double epochShift = sourceDatum.getOrigin().getTime() -
targetDatum.getOrigin().getTime();
epochShift = Units.MILLISECOND.getConverterTo(targetUnit).convert(epochShift);
/*
* Check axis directions. The method 'swapAndScaleAxes' should returns a matrix of size 2×2.
* The element at index (0,0) may be +1 if source and target axes are in the same direction,
* or -1 if there are in opposite direction ("PAST" vs "FUTURE"). The value may be something
* else than ±1 if a unit conversion is applied too. For example the value is 60 if time in
* sourceCRS is in hours while time in targetCRS is in minutes.
*
* The "epoch shift" previously computed is a translation. Consequently, it is added to element (0,1).
*/
final Matrix matrix;
try {
matrix = CoordinateSystems.swapAndScaleAxes(sourceCS, targetCS);
} catch (IllegalArgumentException | IncommensurableException exception) {
throw new OperationNotFoundException(notFoundMessage(sourceCRS, targetCRS), exception);
}
final int translationColumn = matrix.getNumCol() - 1; // Paranoiac check: should always be 1.
final double translation = matrix.getElement(0, translationColumn);
matrix.setElement(0, translationColumn, translation + epochShift);
return asList(createFromAffineTransform(AXIS_CHANGES, sourceCRS, targetCRS, matrix));
}
/**
* Creates an operation between at least one {@code CompoundCRS} (usually the source) and an arbitrary CRS.
* The default implementation tries to invoke the {@link #createOperation createOperation(…)} method with
* various combinations of source and target components. A preference is given for components of the same
* type (e.g. source {@link GeodeticCRS} with target {@code GeodeticCRS}, <i>etc.</i>).
*
* <p>This method returns only <em>one</em> step for a chain of concatenated operations (to be built by the caller).
* But a list is returned because the same step may be implemented by different operation methods. Only one element
* in the returned list should be selected (usually the first one).</p>
*
* @param sourceCRS input coordinate reference system.
* @param sourceComponents components of the source CRS.
* @param targetCRS output coordinate reference system.
* @param targetComponents components of the target CRS.
* @return a coordinate operation from {@code sourceCRS} to {@code targetCRS}.
* @throws FactoryException if the operation can not be constructed.
*/
protected List<CoordinateOperation> createOperationStep(
final CoordinateReferenceSystem sourceCRS, final List<? extends SingleCRS> sourceComponents,
final CoordinateReferenceSystem targetCRS, final List<? extends SingleCRS> targetComponents)
throws FactoryException
{
final SubOperationInfo[] infos = new SubOperationInfo[targetComponents.size()];
final boolean[] sourceIsUsed = new boolean[sourceComponents.size()];
final CoordinateReferenceSystem[] stepComponents = new CoordinateReferenceSystem[infos.length];
/*
* Operations found are stored in 'infos', but are not yet wrapped in PassThroughOperations.
* We need to know first if some coordinate values need reordering for matching the target CRS
* order. We also need to know if any source coordinates should be dropped.
*/
for (int i=0; i<infos.length; i++) {
if ((infos[i] = SubOperationInfo.create(this, sourceIsUsed, sourceComponents, targetComponents.get(i))) == null) {
throw new OperationNotFoundException(notFoundMessage(sourceCRS, targetCRS));
}
stepComponents[i] = infos[i].operation.getSourceCRS();
}
/*
* At this point, a coordinate operation has been found for all components of the target CRS.
* However the CoordinateOperation.getSourceCRS() values are not necessarily in the same order
* than the components of the source CRS given to this method, and some dimensions may be dropped.
* The matrix computed by sourceToSelected(…) gives us the rearrangement needed for the coordinate
* operations that we just found.
*/
int remainingSourceDimensions = 0;
for (final SubOperationInfo component : infos) {
remainingSourceDimensions += component.endAtDimension - component.startAtDimension;
}
final Matrix select = SubOperationInfo.sourceToSelected(
sourceCRS.getCoordinateSystem().getDimension(), remainingSourceDimensions, infos);
/*
* First, we need a CRS matching the above-cited rearrangement. That CRS will be named 'stepSourceCRS'
* and its components will be named 'stepComponents'. Then we will execute a loop in which each component
* is progressively (one by one) updated from a source component to a target component. A new step CRS is
* recreated each time, since it will be needed for each PassThroughOperation.
*/
CoordinateReferenceSystem stepSourceCRS;
CoordinateOperation operation;
if (select.isIdentity()) {
stepSourceCRS = sourceCRS; // No rearrangement - we can use source CRS as-is.
operation = null;
} else {
if (stepComponents.length == 1) {
stepSourceCRS = stepComponents[0]; // Slight optimization of the next block (in the 'else' case).
} else {
stepSourceCRS = toAuthorityDefinition(CoordinateReferenceSystem.class,
factorySIS.getCRSFactory().createCompoundCRS(derivedFrom(sourceCRS), stepComponents));
}
operation = createFromAffineTransform(AXIS_CHANGES, sourceCRS, stepSourceCRS, select);
}
/*
* For each sub-operation, create a PassThroughOperation for the (stepSourceCRS → stepTargetCRS) operation.
* Each source CRS inside this loop will be for dimensions at indices [startAtDimension … endAtDimension-1].
* Note that those indices are not necessarily the same than the indices in the fields of the same name in
* SubOperationInfo, because those indices are not relative to the same CompoundCRS.
*/
int endAtDimension = 0;
final int startOfIdentity = SubOperationInfo.startOfIdentity(infos);
for (int i=0; i<stepComponents.length; i++) {
final CoordinateReferenceSystem source = stepComponents[i];
final CoordinateReferenceSystem target = targetComponents.get(i);
CoordinateOperation subOperation = infos[i].operation;
final MathTransform subTransform = subOperation.getMathTransform();
/*
* In order to compute 'stepTargetCRS', replace in-place a single element in 'stepComponents'.
* For each step except the last one, 'stepTargetCRS' is a mix of target and source CRS. Only
* after the loop finished, 'stepTargetCRS' will become the complete targetCRS definition.
*/
final CoordinateReferenceSystem stepTargetCRS;
stepComponents[i] = target;
if (i >= startOfIdentity) {
stepTargetCRS = targetCRS; // If all remaining transforms are identity, we reached the final CRS.
} else if (subTransform.isIdentity()) {
stepTargetCRS = stepSourceCRS; // In any identity transform, the source and target CRS are equal.
} else if (stepComponents.length == 1) {
stepTargetCRS = target; // Slight optimization of the next block.
} else {
final EllipsoidalHeightCombiner c = new EllipsoidalHeightCombiner(factorySIS.getCRSFactory(), factorySIS.getCSFactory(), factory);
stepTargetCRS = toAuthorityDefinition(CoordinateReferenceSystem.class, c.createCompoundCRS(derivedFrom(target), stepComponents));
}
int delta = source.getCoordinateSystem().getDimension();
final int startAtDimension = endAtDimension;
endAtDimension += delta;
/*
* Constructs the pass through transform only if there is at least one coordinate to pass.
* Actually the code below would work inconditionally, but we perform this check anyway
* for avoiding the creation of intermediate objects.
*/
if (!(startAtDimension == 0 && endAtDimension == remainingSourceDimensions)) {
final Map<String,?> properties = IdentifiedObjects.getProperties(subOperation);
/*
* The DefaultPassThroughOperation constuctor expect a SingleOperation.
* In most case, the 'subOperation' is already of this kind. However if
* it is not, try to copy it in such object.
*/
final SingleOperation op;
if (SubTypes.isSingleOperation(subOperation)) {
op = (SingleOperation) subOperation;
} else {
op = factorySIS.createSingleOperation(properties,
subOperation.getSourceCRS(), subOperation.getTargetCRS(), null,
new DefaultOperationMethod(subTransform), subTransform);
}
subOperation = new DefaultPassThroughOperation(properties, stepSourceCRS, stepTargetCRS,
op, startAtDimension, remainingSourceDimensions - endAtDimension);
}
/*
* Concatenate the operation with the ones we have found so far, and use the current 'stepTargetCRS'
* as the source CRS for the next operation step. We also need to adjust the dimension indices,
* since the previous operations may have removed some dimensions. Note that the delta may also
* be negative in a few occasions.
*/
operation = concatenate(operation, subOperation);
stepSourceCRS = stepTargetCRS;
delta -= target.getCoordinateSystem().getDimension();
endAtDimension -= delta;
remainingSourceDimensions -= delta;
}
return asList(operation);
}
/////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////
//////////// ////////////
//////////// M I S C E L L A N E O U S ////////////
//////////// ////////////
/////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////
/**
* Creates a coordinate operation from a matrix, which usually describes an affine transform.
* A default {@link OperationMethod} object is given to this transform. In the special case
* where the {@code name} identifier is {@link #DATUM_SHIFT} or {@link #ELLIPSOID_CHANGE},
* the operation will be a {@link Transformation} instance instead of {@link Conversion}.
*
* @param name the identifier for the operation to be created.
* @param sourceCRS the source coordinate reference system.
* @param targetCRS the target coordinate reference system.
* @param matrix the matrix which describe an affine transform operation.
* @return the conversion or transformation.
* @throws FactoryException if the operation can not be created.
*/
private CoordinateOperation createFromAffineTransform(final Identifier name,
final CoordinateReferenceSystem sourceCRS,
final CoordinateReferenceSystem targetCRS,
final Matrix matrix)
throws FactoryException
{
final MathTransform transform = factorySIS.getMathTransformFactory().createAffineTransform(matrix);
return createFromMathTransform(properties(name), sourceCRS, targetCRS, transform, null, null, null);
}
/**
* Concatenates two operation steps.
* The new concatenated operation gets an automatically generated name.
*
* <h4>Special case</h4>
* If one of the given operation steps performs a change of axis order or units,
* then that change will be merged with the other operation instead of creating an {@link ConcatenatedOperation}.
*
* @param step1 the first step, or {@code null} for the identity operation.
* @param step2 the second step, or {@code null} for the identity operation.
* @return a concatenated operation, or {@code null} if all arguments were null.
* @throws FactoryException if the operation can't be constructed.
*/
private CoordinateOperation concatenate(final CoordinateOperation step1,
final CoordinateOperation step2)
throws FactoryException
{
if (isIdentity(step1)) return step2;
if (isIdentity(step2)) return step1;
final MathTransform mt1 = step1.getMathTransform();
final MathTransform mt2 = step2.getMathTransform();
final CoordinateReferenceSystem sourceCRS = step1.getSourceCRS();
final CoordinateReferenceSystem targetCRS = step2.getTargetCRS();
/*
* If one of the transform performs nothing more than a change of axis order or units, do
* not expose that conversion in a ConcatenatedTransform. Instead, merge that conversion
* with the "main" operation. The intent is to simplify the operation chain by hidding
* trivial operations.
*/
CoordinateOperation main = null;
final boolean isAxisChange1 = canHide(step1.getName());
final boolean isAxisChange2 = canHide(step2.getName());
if (isAxisChange1 && isAxisChange2 && isAffine(step1) && isAffine(step2)) {
main = step2; // Arbitrarily take the last step.
if (main.getName() == IDENTITY && step1.getName() != IDENTITY) {
main = step1;
}
} else {
if (isAxisChange1 && mt1.getSourceDimensions() == mt1.getTargetDimensions()) main = step2;
if (isAxisChange2 && mt2.getSourceDimensions() == mt2.getTargetDimensions()) main = step1;
}
if (SubTypes.isSingleOperation(main)) {
final SingleOperation op = (SingleOperation) main;
final MathTransform mt = factorySIS.getMathTransformFactory().createConcatenatedTransform(mt1, mt2);
main = createFromMathTransform(new HashMap<>(IdentifiedObjects.getProperties(main)),
sourceCRS, targetCRS, mt, op.getMethod(), op.getParameterValues(),
(main instanceof Transformation) ? Transformation.class :
(main instanceof Conversion) ? Conversion.class : SingleOperation.class);
} else {
main = factory.createConcatenatedOperation(defaultName(sourceCRS, targetCRS), step1, step2);
}
/*
* Sometime we get a concatenated operation made of an operation followed by its inverse.
* We can identify thoses case when the associated MathTransform is the identity transform.
* In such case, simplify by replacing the ConcatenatedTransform by a SingleTransform.
*/
if (main instanceof ConcatenatedOperation && main.getMathTransform().isIdentity()) {
Class<? extends CoordinateOperation> type = null;
for (final CoordinateOperation component : ((ConcatenatedOperation) main).getOperations()) {
if (component instanceof Transformation) {
type = Transformation.class;
break;
}
}
main = createFromMathTransform(new HashMap<>(IdentifiedObjects.getProperties(main)),
main.getSourceCRS(), main.getTargetCRS(), main.getMathTransform(), null, null, type);
}
return main;
}
/**
* Concatenates three transformation steps. If the first and/or the last operation is an {@link #AXIS_CHANGES},
* then it will be included as part of the second operation instead of creating a {@link ConcatenatedOperation}.
* If a concatenated operation is created, it will get an automatically generated name.
*
* @param step1 the first step, or {@code null} for the identity operation.
* @param step2 the second step, or {@code null} for the identity operation.
* @param step3 the third step, or {@code null} for the identity operation.
* @return a concatenated operation, or {@code null} if all arguments were null.
* @throws FactoryException if the operation can not be constructed.
*/
private CoordinateOperation concatenate(final CoordinateOperation step1,
final CoordinateOperation step2,
final CoordinateOperation step3)
throws FactoryException
{
if (isIdentity(step1)) return concatenate(step2, step3);
if (isIdentity(step2)) return concatenate(step1, step3);
if (isIdentity(step3)) return concatenate(step1, step2);
if (canHide(step1.getName())) return concatenate(concatenate(step1, step2), step3);
if (canHide(step3.getName())) return concatenate(step1, concatenate(step2, step3));
final Map<String,?> properties = defaultName(step1.getSourceCRS(), step3.getTargetCRS());
return factory.createConcatenatedOperation(properties, step1, step2, step3);
}
/**
* Returns {@code true} if the given operation is non-null and use the affine operation method.
*/
private static boolean isAffine(final CoordinateOperation operation) {
if (operation instanceof SingleOperation) {
if (IdentifiedObjects.isHeuristicMatchForName(((SingleOperation) operation).getMethod(), Constants.AFFINE)) {
return true;
}
}
return false;
}
/**
* Returns {@code true} if the specified operation is an identity conversion.
* This method always returns {@code false} for transformations even if their
* associated math transform is an identity one, because such transformations
* are usually datum shift and must be visible.
*/
private static boolean isIdentity(final CoordinateOperation operation) {
if (operation == null) {
return true;
}
if ((operation instanceof Conversion) && operation.getMathTransform().isIdentity()) {
return CoordinateOperations.wrapAroundChanges(operation).isEmpty();
}
return false;
}
/**
* Returns {@code true} if a coordinate operation of the given name can be hidden
* in the list of operations. Note that the {@code MathTransform} will still take
* the operation in account however.
*/
private static boolean canHide(final Identifier id) {
return (id == AXIS_CHANGES) || (id == IDENTITY);
}
/**
* Returns the given name in a singleton map.
*/
private static Map<String,?> properties(final String name) {
return Collections.singletonMap(IdentifiedObject.NAME_KEY, name);
}
/**
* Returns a name for an object derived from the specified one.
* This method builds a name of the form "{@literal <original identifier>} (step 1)"
* where "(step 1)" may be replaced by "(step 2)", "(step 3)", <i>etc.</i> if this
* method has already been invoked for the same identifier (directly or indirectly).
*/
private Map<String,?> derivedFrom(final IdentifiedObject object) {
Identifier oldID = object.getName();
Object p = identifierOfStepCRS.get(oldID);
if (p instanceof Identifier) {
oldID = (Identifier) p;
p = identifierOfStepCRS.get(oldID);
}
final int count = (p != null) ? (Integer) p + 1 : 1;
final Identifier newID = new NamedIdentifier(Citations.SIS, oldID.getCode() + " (step " + count + ')');
identifierOfStepCRS.put(newID, oldID);
identifierOfStepCRS.put(oldID, count);
final Map<String,Object> properties = new HashMap<>(4);
properties.put(IdentifiedObject.NAME_KEY, newID);
properties.put(IdentifiedObject.REMARKS_KEY, Vocabulary.formatInternational(
Vocabulary.Keys.DerivedFrom_1, CRSPair.label(object)));
return properties;
}
/**
* Returns a name for a transformation between two CRS.
*/
private static Map<String,?> defaultName(CoordinateReferenceSystem source, CoordinateReferenceSystem target) {
return properties(new CRSPair(source, target).toString());
}
/**
* Returns the given operation as a list of one element. We can not use {@link Collections#singletonList(Object)}
* because the list needs to be modifiable, as required by {@link #createOperations(CoordinateReferenceSystem,
* CoordinateReferenceSystem)} method contract.
*/
private static List<CoordinateOperation> asList(final CoordinateOperation operation) {
final List<CoordinateOperation> operations = new ArrayList<>(1);
operations.add(operation);
return operations;
}
/**
* Returns an error message for "No path found from sourceCRS to targetCRS".
* This is used for the construction of {@link OperationNotFoundException}.
*
* @param source the source CRS.
* @param target the target CRS.
* @return a default error message.
*/
private static String notFoundMessage(final IdentifiedObject source, final IdentifiedObject target) {
return Resources.format(Resources.Keys.CoordinateOperationNotFound_2, CRSPair.label(source), CRSPair.label(target));
}
/**
* Returns an error message for "Can not invert operation XYZ.".
* This is used for the construction of {@link OperationNotFoundException}.
*
* @param crs the CRS having a conversion that can not be inverted.
* @return a default error message.
*/
private static String canNotInvert(final GeneralDerivedCRS crs) {
return Resources.format(Resources.Keys.NonInvertibleOperation_1, crs.getConversionFromBase().getName().getCode());
}
}