All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.geotoolkit.referencing.operation.AbstractCoordinateOperationFactory Maven / Gradle / Ivy

/*
 *    Geotoolkit.org - An Open Source Java GIS Toolkit
 *    http://www.geotoolkit.org
 *
 *    (C) 2001-2012, Open Source Geospatial Foundation (OSGeo)
 *    (C) 2009-2012, Geomatys
 *
 *    This library is free software; you can redistribute it and/or
 *    modify it under the terms of the GNU Lesser General Public
 *    License as published by the Free Software Foundation;
 *    version 2.1 of the License.
 *
 *    This library is distributed in the hope that it will be useful,
 *    but WITHOUT ANY WARRANTY; without even the implied warranty of
 *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *    Lesser General Public License for more details.
 *
 *    This package contains documentation from OpenGIS specifications.
 *    OpenGIS consortium's work is fully acknowledged here.
 */
package org.geotoolkit.referencing.operation;

import java.util.Map;
import java.util.HashMap;
import java.util.LinkedList;
import java.awt.RenderingHints;
import javax.measure.converter.ConversionException;
import net.jcip.annotations.ThreadSafe;

import org.opengis.util.FactoryException;
import org.opengis.util.NoSuchIdentifierException;
import org.opengis.metadata.quality.PositionalAccuracy;
import org.opengis.parameter.ParameterValueGroup;
import org.opengis.parameter.ParameterDescriptorGroup;
import org.opengis.referencing.IdentifiedObject;
import org.opengis.referencing.ReferenceIdentifier;
import org.opengis.referencing.cs.CoordinateSystem;
import org.opengis.referencing.crs.CoordinateReferenceSystem;
import org.opengis.referencing.operation.*;

import org.geotoolkit.factory.Hints;
import org.geotoolkit.util.Utilities;
import org.geotoolkit.util.converter.Classes;
import org.geotoolkit.util.collection.WeakHashSet;
import org.geotoolkit.referencing.NamedIdentifier;
import org.geotoolkit.referencing.IdentifiedObjects;
import org.geotoolkit.referencing.factory.ReferencingFactory;
import org.geotoolkit.referencing.factory.ReferencingFactoryContainer;
import org.geotoolkit.referencing.operation.provider.Affine;
import org.geotoolkit.referencing.datum.BursaWolfParameters;
import org.geotoolkit.referencing.cs.AbstractCS;
import org.geotoolkit.resources.Vocabulary;
import org.geotoolkit.resources.Errors;

import static java.util.Collections.singletonMap;
import static org.opengis.referencing.IdentifiedObject.NAME_KEY;
import static org.opengis.referencing.operation.CoordinateOperation.COORDINATE_OPERATION_ACCURACY_KEY;
import static org.geotoolkit.metadata.iso.quality.AbstractPositionalAccuracy.DATUM_SHIFT_APPLIED;
import static org.geotoolkit.metadata.iso.quality.AbstractPositionalAccuracy.DATUM_SHIFT_OMITTED;
import static org.geotoolkit.metadata.iso.citation.Citations.GEOTOOLKIT;
import static org.geotoolkit.resources.Vocabulary.formatInternational;
import static org.geotoolkit.internal.InternalUtilities.debugEquals;


/**
 * Base class for coordinate operation factories. This class provides helper methods for the
 * construction of building blocks. It doesn't figure out any operation path by itself. This
 * more "intelligent" job is left to subclasses.
 *
 * @author Martin Desruisseaux (IRD, Geomatys)
 * @version 3.19
 *
 * @since 2.1
 * @level advanced
 * @module
 */
@ThreadSafe
public abstract class AbstractCoordinateOperationFactory extends ReferencingFactory
        implements CoordinateOperationFactory
{
    /**
     * The identifier for an identity operation.
     */
    protected static final ReferenceIdentifier IDENTITY =
            new NamedIdentifier(GEOTOOLKIT, formatInternational(Vocabulary.Keys.IDENTITY));

    /**
     * The identifier for conversion using an affine transform for axis swapping and/or
     * unit conversions.
     */
    protected static final ReferenceIdentifier AXIS_CHANGES =
            new NamedIdentifier(GEOTOOLKIT, formatInternational(Vocabulary.Keys.AXIS_CHANGES));

    /**
     * The identifier for a transformation which is a datum shift.
     *
     * @see org.geotoolkit.metadata.iso.quality.AbstractPositionalAccuracy#DATUM_SHIFT_APPLIED
     */
    protected static final ReferenceIdentifier DATUM_SHIFT =
            new NamedIdentifier(GEOTOOLKIT, formatInternational(Vocabulary.Keys.DATUM_SHIFT));

    /**
     * The identifier for a transformation which is a datum shift without
     * {@linkplain BursaWolfParameters Bursa Wolf parameters}. Only the changes in ellipsoid
     * axis-length are taken in account. Such ellipsoid shifts are approximative and may have
     * 1 kilometre error. This transformation is allowed only if the factory was created with
     * {@link Hints#LENIENT_DATUM_SHIFT} set to {@link Boolean#TRUE}.
     *
     * @see org.geotoolkit.metadata.iso.quality.AbstractPositionalAccuracy#DATUM_SHIFT_OMITTED
     */
    protected static final ReferenceIdentifier ELLIPSOID_SHIFT =
            new NamedIdentifier(GEOTOOLKIT, formatInternational(Vocabulary.Keys.ELLIPSOID_SHIFT));

    /**
     * The identifier for a geocentric conversion.
     */
    protected static final ReferenceIdentifier GEOCENTRIC_CONVERSION =
            new NamedIdentifier(GEOTOOLKIT, formatInternational(Vocabulary.Keys.GEOCENTRIC_TRANSFORM));

    /**
     * The identifier for an inverse operation.
     */
    protected static final ReferenceIdentifier INVERSE_OPERATION =
            new NamedIdentifier(GEOTOOLKIT, formatInternational(Vocabulary.Keys.INVERSE_OPERATION));

    /**
     * The set of helper methods on factories.
     */
    final ReferencingFactoryContainer factories;

    /**
     * A pool of coordinate operation. This pool is used in order
     * to returns instance of existing operations when possible.
     */
    private final WeakHashSet pool =
            WeakHashSet.newInstance(CoordinateOperation.class);

    /**
     * Tells if {@link FactoryGroup#hints} has been invoked. It must be invoked exactly once,
     * but can't be invoked in the constructor because it causes a {@link StackOverflowError}
     * in some situations.
     */
    private volatile boolean hintsInitialized;

    /**
     * Constructs a coordinate operation factory using the specified hints.
     * This constructor recognizes the {@link Hints#CRS_FACTORY CRS}, {@link Hints#CS_FACTORY CS},
     * {@link Hints#DATUM_FACTORY DATUM} and {@link Hints#MATH_TRANSFORM_FACTORY MATH_TRANSFORM}
     * {@code FACTORY} hints.
     *
     * @param userHints The hints, or {@code null} if none.
     */
    public AbstractCoordinateOperationFactory(final Hints userHints) {
        factories = ReferencingFactoryContainer.instance(userHints);
    }

    /**
     * If the specified factory is an instance of {@code AbstractCoordinateOperationFactory},
     * fetch the {@link ReferencingFactoryContainer} from this instance instead of from the
     * hints. This constructor is strictly reserved for factory subclasses that are wrapper
     * around an other factory, like {@link CachingCoordinateOperationFactory}.
     *
     * @param factory   The factory to wrap.
     * @param userHints The hints, or {@code null} if none.
     */
    AbstractCoordinateOperationFactory(final CoordinateOperationFactory factory, final Hints hints) {
        if (factory instanceof AbstractCoordinateOperationFactory) {
            factories = ((AbstractCoordinateOperationFactory) factory).factories;
        } else {
            factories = ReferencingFactoryContainer.instance(hints);
        }
    }

    /**
     * To be overridden by {@link CachingAuthorityFactory} only.
     */
    CoordinateOperationFactory getBackingFactory() {
        return null;
    }

    /**
     * Returns the implementation hints for this factory. The returned map contains values for
     * {@link Hints#CRS_FACTORY CRS}, {@link Hints#CS_FACTORY CS}, {@link Hints#DATUM_FACTORY DATUM}
     * and {@link Hints#MATH_TRANSFORM_FACTORY MATH_TRANSFORM} {@code FACTORY} hints. Other values
     * may be provided as well, at implementation choice.
     */
    @Override
    public Map getImplementationHints() {
        if (!hintsInitialized) {
            final Map toAdd = factories.getImplementationHints();
            final CoordinateOperationFactory back = getBackingFactory();
            /*
             * Double-check locking: was a deprecated practice before Java 5, but is okay since
             * Java 5 providing that the variable is volatile. In this particular case, we want
             * the above lines to be executed outside the synchronization block in order to reduce
             * the risk of deadlock. This is not a big deal if those values are computed twice.
             */
            synchronized (this) {
                if (!hintsInitialized) {
                    hintsInitialized = true;
                    hints.putAll(toAdd);
                    if (back != null) {
                        hints.put(Hints.COORDINATE_OPERATION_FACTORY, back);
                    }
                }
            }
        }
        return super.getImplementationHints();
    }

    /**
     * Invoked when the {@link #hints} map should be initialized. This method may
     * be overridden by subclasses like {@link CachingCoordinateOperationFactory}.
     *
     * @return The hints to add to {@link #hints}.
     */
    Map initializeHints() {
        return factories.getImplementationHints();
    }

    /**
     * Returns the operation method of the given name. The default implementation returns the first
     * method from the set returned by {@link MathTransformFactory#getAvailableMethods(Class)}
     * which have a {@linkplain IdentifiedObjects#nameMatches matching name}.
     *
     * @param  name The name of the operation method to fetch.
     * @return The operation method of the given name.
     * @throws FactoryException if the requested operation method can not be fetched.
     *
     * @see #createOperationMethod(Map, Integer, Integer, ParameterDescriptorGroup)
     *
     * @since 3.19
     */
    public OperationMethod getOperationMethod(final String name) throws FactoryException {
        final MathTransformFactory mtFactory = getMathTransformFactory();
        if (mtFactory instanceof DefaultMathTransformFactory) {
            return ((DefaultMathTransformFactory) mtFactory).getOperationMethod(name);
        }
        for (final OperationMethod method : mtFactory.getAvailableMethods(SingleOperation.class)) {
            if (IdentifiedObjects.nameMatches(method, name)) {
                return method;
            }
        }
        throw new NoSuchIdentifierException(Errors.format(
                Errors.Keys.NO_TRANSFORM_FOR_CLASSIFICATION_$1, name), name);
    }

    /**
     * Returns the underlying math transform factory. This factory
     * is used for constructing {@link MathTransform} objects for
     * all {@linkplain CoordinateOperation coordinate operations}.
     *
     * @return The underlying math transform factory.
     */
    public final MathTransformFactory getMathTransformFactory() {
        return factories.getMathTransformFactory();
    }

    /**
     * Returns an affine transform between two coordinate systems. Only units and
     * axis order (e.g. transforming from (NORTH,WEST) to (EAST,NORTH)) are taken
     * in account.
     * 

* Example: If coordinates in {@code sourceCS} are (x,y) pairs in metres and * coordinates in {@code targetCS} are (-y,x) pairs in centimetres, then the * transformation can be performed as below: * * {@preformat text * ┌ ┐ ┌ ┐ ┌ ┐ * │-y(cm)│ │ 0 -100 0 │ │ x(m)│ * │ x(cm)│ = │ 100 0 0 │ │ y(m)│ * │ 1 │ │ 0 0 1 │ │ 1 │ * └ ┘ └ ┘ └ ┘ * } * * The default implementation performs the same work than the static method in the * {@link AbstractCS#swapAndScaleAxis(CoordinateSystem, CoordinateSystem) AbstractCS} * class, except that the unchecked exceptions are wrapped into the checked * {@link OperationNotFoundException}. * * @param sourceCS The source coordinate system. * @param targetCS The target coordinate system. * @return The transformation from {@code sourceCS} to {@code targetCS} as * an affine transform. Only axis orientation and units are taken in account. * @throws OperationNotFoundException If the affine transform can't be constructed. * * @see AbstractCS#swapAndScaleAxis(CoordinateSystem, CoordinateSystem) */ protected Matrix swapAndScaleAxis(final CoordinateSystem sourceCS, final CoordinateSystem targetCS) throws OperationNotFoundException { try { return AbstractCS.swapAndScaleAxis(sourceCS,targetCS); } catch (IllegalArgumentException exception) { throw new OperationNotFoundException(getErrorMessage(sourceCS, targetCS), exception); } catch (ConversionException exception) { throw new OperationNotFoundException(getErrorMessage(sourceCS, targetCS), exception); } // No attempt to catch ClassCastException since such // exception would indicates a programming error. } /** * Returns the specified identifier in a map to be given to coordinate operation constructors. * In the special case where the {@code name} identifier is {@link #DATUM_SHIFT} or * {@link #ELLIPSOID_SHIFT}, the map will contains extra informations like positional * accuracy. * * {@note In the datum shift case, an operation version is mandatory but unknown at this time. * However, we noticed that the EPSG database do not always defines a version neither. * Consequently, the Geotk implementation relax the rule requirying an operation * version and we do not try to provide this information here for now.} */ private static Map getProperties(final ReferenceIdentifier name) { final Map properties; if ((name == DATUM_SHIFT) || (name == ELLIPSOID_SHIFT)) { properties = new HashMap(4); properties.put(NAME_KEY, name); properties.put(COORDINATE_OPERATION_ACCURACY_KEY, new PositionalAccuracy[] { name == DATUM_SHIFT ? DATUM_SHIFT_APPLIED : DATUM_SHIFT_OMITTED }); } else { properties = singletonMap(NAME_KEY, (Object) name); } return properties; } /** * 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_SHIFT}, * the operation will be an instance of {@link Transformation} instead of the usual * {@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't be created. */ protected CoordinateOperation createFromAffineTransform( final ReferenceIdentifier name, final CoordinateReferenceSystem sourceCRS, final CoordinateReferenceSystem targetCRS, final Matrix matrix) throws FactoryException { final MathTransformFactory mtFactory = getMathTransformFactory(); final MathTransform transform = mtFactory.createAffineTransform(matrix); final Map properties = getProperties(name); final Class type = properties.containsKey(COORDINATE_OPERATION_ACCURACY_KEY) ? Transformation.class : Conversion.class; return createFromMathTransform(properties, sourceCRS, targetCRS, transform, Affine.getProvider(transform.getSourceDimensions(), transform.getTargetDimensions()), type); } /** * Creates a coordinate operation from a set of parameters. * The {@linkplain OperationMethod operation method} is inferred automatically, * if possible. * * @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 parameters The parameters. * @return The conversion or transformation. * @throws FactoryException if the operation can't be created. */ protected CoordinateOperation createFromParameters( final ReferenceIdentifier name, final CoordinateReferenceSystem sourceCRS, final CoordinateReferenceSystem targetCRS, final ParameterValueGroup parameters) throws FactoryException { final Map properties = getProperties(name); final MathTransformFactory mtFactory = getMathTransformFactory(); final MathTransform transform = mtFactory.createParameterizedTransform(parameters); final OperationMethod method = mtFactory.getLastMethodUsed(); return createFromMathTransform(properties, sourceCRS, targetCRS, transform, method, SingleOperation.class); } /** * Creates a coordinate operation from a math transform. * * @param name The identifier for the operation to be created. * @param sourceCRS The source coordinate reference system. * @param targetCRS The destination coordinate reference system. * @param transform The math transform. * @return A coordinate operation using the specified math transform. * @throws FactoryException if the operation can't be constructed. */ protected CoordinateOperation createFromMathTransform( final ReferenceIdentifier name, final CoordinateReferenceSystem sourceCRS, final CoordinateReferenceSystem targetCRS, final MathTransform transform) throws FactoryException { final Map properties = singletonMap(NAME_KEY, name); return createFromMathTransform(properties, sourceCRS, targetCRS, transform, createOperationMethod(properties, sourceCRS.getCoordinateSystem().getDimension(), targetCRS.getCoordinateSystem().getDimension(), null), CoordinateOperation.class); } /** * Creates a coordinate operation from a math transform. * If the specified math transform is already a coordinate operation, and if source * and target CRS match, then {@code transform} is returned with no change. * Otherwise, a new coordinate operation is created. * * @param properties The properties to give to the operation. * @param sourceCRS The source coordinate reference system. * @param targetCRS The destination coordinate reference system. * @param transform The math transform. * @param method The operation method, or {@code null}. * @param type The required super-class (e.g. {@linkplain Transformation}.class). * @return A coordinate operation using the specified math transform. * @throws FactoryException if the operation can't be constructed. */ protected CoordinateOperation createFromMathTransform( final Map properties, final CoordinateReferenceSystem sourceCRS, final CoordinateReferenceSystem targetCRS, final MathTransform transform, final OperationMethod method, final Class type) throws FactoryException { CoordinateOperation operation; if (transform instanceof CoordinateOperation) { operation = (CoordinateOperation) transform; if (Utilities.equals(operation.getSourceCRS(), sourceCRS) && Utilities.equals(operation.getTargetCRS(), targetCRS) && Utilities.equals(operation.getMathTransform(), transform)) { if (operation instanceof SingleOperation) { if (Utilities.equals(((SingleOperation) operation).getMethod(), method)) { return operation; } } else { return operation; } } } operation = DefaultSingleOperation.create(properties, sourceCRS, targetCRS, transform, method, type); operation = pool.unique(operation); return operation; } /** * Constructs a defining conversion from a set of properties. * * @param properties Set of properties. Should contains at least {@code "name"}. * @param method The operation method. * @param parameters The parameter values. * @return The defining conversion. * @throws FactoryException if the object creation failed. * * @see DefiningConversion * * @since 2.5 */ @Override public Conversion createDefiningConversion( final Map properties, final OperationMethod method, final ParameterValueGroup parameters) throws FactoryException { Conversion conversion = new DefiningConversion(properties, method, parameters); conversion = pool.unique(conversion); return conversion; } /** * Creates an operation method from a set of properties and a descriptor group. The default * implementation delegates to the {@link DefaultOperationMethod#DefaultOperationMethod(Map, * Integer, Integer, ParameterDescriptorGroup) DefaultOperationMethod} constructor. * * @param properties Set of properties. Shall contains at least {@code "name"}. * @param sourceDimension Number of dimensions in the source CRS of this operation method. * @param targetDimension Number of dimensions in the target CRS of this operation method. * @param parameters The set of parameters, or {@code null} if none. * * @see #getOperationMethod(String) * * @since 3.19 */ public OperationMethod createOperationMethod(final Map properties, final Integer sourceDimension, final Integer targetDimension, final ParameterDescriptorGroup parameters) throws FactoryException { return new DefaultOperationMethod(properties, sourceDimension, targetDimension, parameters); } /** * Creates a concatenated operation from a sequence of operations. * * @param properties Set of properties. Should contains at least {@code "name"}. * @param operations The sequence of operations. * @return The concatenated operation. * @throws FactoryException if the object creation failed. */ @Override public CoordinateOperation createConcatenatedOperation(final Map properties, final CoordinateOperation... operations) throws FactoryException { CoordinateOperation operation; operation = new DefaultConcatenatedOperation(properties, operations, getMathTransformFactory()); operation = pool.unique(operation); return operation; } /** * Concatenates two operation steps. If an operation is an {@link #AXIS_CHANGES}, * it will be included as part of the second operation instead of creating an * {@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. * @return A concatenated operation, or {@code null} if all arguments was nul. * @throws FactoryException if the operation can't be constructed. */ protected CoordinateOperation concatenate(final CoordinateOperation step1, final CoordinateOperation step2) throws FactoryException { if (step1 == null) return step2; if (step2 == null) return step1; // Note: we sometime get this assertion failure if the user provided CRS with two // different ellipsoids but an identical TOWGS84 conversion infos (which is // usually wrong, but still happen). assert debugEquals(step1.getTargetCRS(), step2.getSourceCRS()); 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(); CoordinateOperation step = null; if (step1.getName() == AXIS_CHANGES && mt1.getSourceDimensions() == mt1.getTargetDimensions()) step = step2; if (step2.getName() == AXIS_CHANGES && mt2.getSourceDimensions() == mt2.getTargetDimensions()) step = step1; if (step instanceof SingleOperation) { /* * Applies only on operation in order to avoid merging with PassThroughOperation. * Also applies only if the transform to hide has identical source and target * dimensions in order to avoid mismatch with the method's dimensions. */ final MathTransformFactory mtFactory = getMathTransformFactory(); return createFromMathTransform(IdentifiedObjects.getProperties(step), sourceCRS, targetCRS, mtFactory.createConcatenatedTransform(mt1, mt2), ((SingleOperation) step).getMethod(), CoordinateOperation.class); } return createConcatenatedOperation(getTemporaryName(sourceCRS, targetCRS), step1, step2); } /** * Concatenates three transformation steps. If the first and/or the last operation is an * {@link #AXIS_CHANGES}, it will be included as part of the second operation instead of * creating an {@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't be constructed. */ protected CoordinateOperation concatenate(final CoordinateOperation step1, final CoordinateOperation step2, final CoordinateOperation step3) throws FactoryException { if (step1 == null) return concatenate(step2, step3); if (step2 == null) return concatenate(step1, step3); if (step3 == null) return concatenate(step1, step2); // Note: we use approximative equality check because the CRS objects may be slightly // different if one CRS has been created from the EPSG database while the other // CRS has been created from WKT parsing. There is rounding error in the second // defining parameter of the ellipsoid, because the WKT parser always use the // createFlattenedSphere(...) constructor while the EPSG database will select // createFlattenedSphere(...) or createEllipsoid(...) depending on the Ellipsoid // definition. assert debugEquals(step1.getTargetCRS(), step2.getSourceCRS()) : step1; assert debugEquals(step2.getTargetCRS(), step3.getSourceCRS()) : step3; if (isIdentity(step1)) return concatenate(step2, step3); if (isIdentity(step2)) return concatenate(step1, step3); if (isIdentity(step3)) return concatenate(step1, step2); if (step1.getName() == AXIS_CHANGES) return concatenate(concatenate(step1, step2), step3); if (step3.getName() == AXIS_CHANGES) return concatenate(step1, concatenate(step2, step3)); final CoordinateReferenceSystem sourceCRS = step1.getSourceCRS(); final CoordinateReferenceSystem targetCRS = step3.getTargetCRS(); return createConcatenatedOperation(getTemporaryName(sourceCRS, targetCRS), step1, step2, step3); } /** * 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) { return (operation instanceof Conversion) && operation.getMathTransform().isIdentity(); } /** * Returns the inverse of the specified operation. * * @param operation The operation to invert. * @return The inverse of {@code operation}. * @throws NoninvertibleTransformException if the operation is not invertible. * @throws FactoryException if the operation creation failed for an other reason. * * @since 2.3 */ protected CoordinateOperation inverse(final CoordinateOperation operation) throws NoninvertibleTransformException, FactoryException { final CoordinateReferenceSystem sourceCRS = operation.getSourceCRS(); final CoordinateReferenceSystem targetCRS = operation.getTargetCRS(); final Map properties = IdentifiedObjects.getProperties(operation, null); properties.putAll(getTemporaryName(targetCRS, sourceCRS)); if (operation instanceof ConcatenatedOperation) { final LinkedList inverted = new LinkedList(); for (final CoordinateOperation op : ((ConcatenatedOperation) operation).getOperations()) { inverted.addFirst(inverse(op)); } return createConcatenatedOperation(properties, inverted.toArray(new CoordinateOperation[inverted.size()])); } final MathTransform transform = operation.getMathTransform().inverse(); final Class type = AbstractCoordinateOperation.getType(operation); final OperationMethod method = (operation instanceof SingleOperation) ? ((SingleOperation) operation).getMethod() : null; return createFromMathTransform(properties, targetCRS, sourceCRS, transform, method, type); } ///////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////// //////////// //////////// //////////// M I S C E L L A N E O U S //////////// //////////// //////////// ///////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////// /** * Returns the dimension of the specified coordinate system, * or {@code 0} if the coordinate system is null. */ static int getDimension(final CoordinateReferenceSystem crs) { return (crs!=null) ? crs.getCoordinateSystem().getDimension() : 0; } /** * An identifier for temporary objects. This identifier manage a count of temporary * identifiers. The count is appended to the identifier name (e.g. "WGS84 (step 1)"). *

* This class is used only for names formatting. This class doesn't have any impact * on the numerical values to be computed. */ private static final class TemporaryIdentifier extends NamedIdentifier { /** For cross-version compatibility. */ private static final long serialVersionUID = -2784354058026177076L; /** The parent identifier. */ private final ReferenceIdentifier parent; /** The temporary object count. */ private final int count; /** Constructs an identifier derived from the specified one. */ public TemporaryIdentifier(final ReferenceIdentifier parent) { this(parent, ((parent instanceof TemporaryIdentifier) ? ((TemporaryIdentifier) parent).count : 0) + 1); } /** Work around for RFE #4093999 in Sun's bug database */ private TemporaryIdentifier(final ReferenceIdentifier parent, final int count) { super(GEOTOOLKIT, unwrap(parent).getCode() + " (step " + count + ')'); this.parent = parent; this.count = count; } /** Returns the parent identifier for the specified identifier, if any. */ public static ReferenceIdentifier unwrap(ReferenceIdentifier identifier) { while (identifier instanceof TemporaryIdentifier) { identifier = ((TemporaryIdentifier) identifier).parent; } return identifier; } } /** * Returns the name of the GeoAPI interface implemented by the specified object. * In addition, the name may be added between brackets. */ private static String getClassName(final IdentifiedObject object) { if (object != null) { Class type = object.getClass(); final Class[] interfaces = type.getInterfaces(); for (int i=0; i candidate = interfaces[i]; if (candidate.getName().startsWith("org.opengis.referencing.")) { type = candidate; break; } } String name = Classes.getShortName(type); final ReferenceIdentifier id = object.getName(); if (id != null) { name = name + '[' + id.getCode() + ']'; } return name; } return null; } /** * Returns a temporary name for object derived from the specified one. * * @param source The CRS to base name on, or {@code null} if none. */ static Map getTemporaryName(final IdentifiedObject source) { final Map properties = new HashMap(4); properties.put(NAME_KEY, new TemporaryIdentifier(source.getName())); properties.put(IdentifiedObject.REMARKS_KEY, formatInternational( Vocabulary.Keys.DERIVED_FROM_$1, getClassName(source))); return properties; } /** * Returns a temporary name for object derived from a concatenation. * * @param source The CRS to base name on, or {@code null} if none. */ static Map getTemporaryName(final CoordinateReferenceSystem source, final CoordinateReferenceSystem target) { final String name = getClassName(source) + " \u21E8 " + getClassName(target); return singletonMap(NAME_KEY, name); } /** * 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. */ protected static String getErrorMessage(final IdentifiedObject source, final IdentifiedObject target) { return Errors.format(Errors.Keys.NO_TRANSFORMATION_PATH_$2, getClassName(source), getClassName(target)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy