org.geotoolkit.referencing.crs.DefaultGeographicCRS 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.crs;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import javax.measure.unit.Unit;
import javax.measure.unit.NonSI;
import javax.measure.quantity.Angle;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;
import net.jcip.annotations.Immutable;
import org.opengis.referencing.cs.AxisDirection;
import org.opengis.referencing.cs.CoordinateSystem;
import org.opengis.referencing.cs.CoordinateSystemAxis;
import org.opengis.referencing.cs.EllipsoidalCS;
import org.opengis.referencing.datum.Ellipsoid;
import org.opengis.referencing.datum.GeodeticDatum;
import org.opengis.referencing.crs.GeographicCRS;
import org.opengis.geometry.MismatchedDimensionException;
import org.geotoolkit.measure.Measure;
import org.geotoolkit.metadata.iso.extent.DefaultExtent;
import org.geotoolkit.referencing.IdentifiedObjects;
import org.geotoolkit.referencing.AbstractReferenceSystem;
import org.geotoolkit.referencing.cs.AxisRangeType;
import org.geotoolkit.referencing.cs.DefaultEllipsoidalCS;
import org.geotoolkit.referencing.datum.DefaultEllipsoid;
import org.geotoolkit.referencing.datum.DefaultGeodeticDatum;
import org.geotoolkit.internal.referencing.AxisDirections;
import org.geotoolkit.internal.referencing.CRSUtilities;
import org.geotoolkit.util.UnsupportedImplementationException;
import org.geotoolkit.io.wkt.Formatter;
import org.geotoolkit.measure.Units;
/**
* A coordinate reference system based on an ellipsoidal approximation of the geoid; this provides
* an accurate representation of the geometry of geographic features for a large portion of the
* earth's surface.
*
*
* Used with CS type(s)
*
* {@link EllipsoidalCS Ellipsoidal}
*
*
* @author Martin Desruisseaux (IRD, Geomatys)
* @version 3.20
*
* @since 1.2
* @module
*/
@Immutable
@XmlRootElement(name = "GeographicCRS")
public class DefaultGeographicCRS extends AbstractSingleCRS implements GeographicCRS {
/**
* Serial number for inter-operability with different versions.
*/
private static final long serialVersionUID = 861224913438092335L;
/**
* A two-dimensional geographic coordinate reference system using the WGS84 datum.
* This CRS uses (longitude, latitude) ordinates with longitude values
* increasing towards the East and latitude values increasing towards the North. The angular
* units are decimal degrees and the prime meridian is Greenwich.
*
* This CRS is equivalent to {@code EPSG:4326} except for axis order,
* since EPSG puts latitude before longitude.
*
* @see DefaultGeodeticDatum#WGS84
*/
public static final DefaultGeographicCRS WGS84;
/**
* A three-dimensional geographic coordinate reference system using the WGS84 datum.
* This CRS uses (longitude, latitude, height)
* ordinates with longitude values increasing towards the East, latitude values
* increasing towards the North and height positive above the ellipsoid. The angular
* units are decimal degrees, the height unit is the metre, and the prime meridian
* is Greenwich.
*
* This CRS is equivalent to {@code EPSG:4979} (the successor to
* {@code EPSG:4329}, itself the successor to {@code EPSG:4327}) except for
* axis order, since EPSG puts latitude before longitude.
*
* @see DefaultGeodeticDatum#WGS84
*/
public static final DefaultGeographicCRS WGS84_3D;
static {
final Map properties = new HashMap(4);
properties.put(NAME_KEY, "WGS84(DD)"); // Name used in WCS 1.0.
final String[] alias = {
"WGS84",
"WGS 84", // EPSG:4979 name.
"WGS 84 (3D)", // EPSG:4329 name.
"WGS 84 (geographic 3D)" // EPSG:4327 name.
};
properties.put(ALIAS_KEY, alias);
properties.put(DOMAIN_OF_VALIDITY_KEY, DefaultExtent.WORLD);
// Do not declare EPSG identifiers, because axis order are not the same.
WGS84 = new DefaultGeographicCRS(properties, DefaultGeodeticDatum.WGS84,
DefaultEllipsoidalCS.GEODETIC_2D);
alias[1] = "WGS 84 (geographic 3D)"; // Replaces the EPSG name.
WGS84_3D = new DefaultGeographicCRS(properties, DefaultGeodeticDatum.WGS84,
DefaultEllipsoidalCS.GEODETIC_3D);
}
/**
* A two-dimensional geographic coordinate reference system using a spherical datum.
* This CRS uses (longitude, latitude) ordinates with longitude values
* increasing towards the East and latitude values increasing towards the North. The angular
* units are decimal degrees and the prime meridian is Greenwich.
*
* {@note This CRS is close, but not identical, to the geographic CRS based on the GRS
* 1980 Authalic Sphere (EPSG:4047). This CRS uses a sphere radius of 6371000
* metres, while the GRS 1980 Authalic Sphere uses a sphere radius of 6371007 metres.}
*
* @see DefaultGeodeticDatum#SPHERE
*
* @since 3.15
*/
public static final DefaultGeographicCRS SPHERE = new DefaultGeographicCRS(
IdentifiedObjects.getProperties(DefaultGeodeticDatum.SPHERE),
DefaultGeodeticDatum.SPHERE, DefaultEllipsoidalCS.GEODETIC_2D);
/**
* Coordinate reference systems equivalent to this one, except for a shift in the range of
* longitude values. This field is computed by {@link #shiftAxisRange(AxisRangeType)}
* when first needed.
*
* @since 3.20
*/
private transient DefaultGeographicCRS[] shifted;
/**
* Constructs a new object in which every attributes are set to a default value.
* This is not a valid object. This constructor is strictly
* reserved to JAXB, which will assign values to the fields using reflexion.
*/
private DefaultGeographicCRS() {
this(org.geotoolkit.internal.referencing.NilReferencingObject.INSTANCE);
}
/**
* Constructs a new geographic CRS with the same values than the specified one.
* This copy constructor provides a way to convert an arbitrary implementation into a
* Geotk one or a user-defined one (as a subclass), usually in order to leverage
* some implementation-specific API. This constructor performs a shallow copy,
* i.e. the properties are not cloned.
*
* @param crs The coordinate reference system to copy.
*
* @since 2.2
*/
public DefaultGeographicCRS(final GeographicCRS crs) {
super(crs);
}
/**
* Constructs a geographic CRS with the same properties than the given datum.
* The inherited properties include the {@linkplain #getName name} and aliases.
*
* @param datum The datum.
* @param cs The coordinate system.
*
* @since 2.5
*/
public DefaultGeographicCRS(final GeodeticDatum datum, final EllipsoidalCS cs) {
this(IdentifiedObjects.getProperties(datum), datum, cs);
}
/**
* Constructs a geographic CRS from a name.
*
* @param name The name.
* @param datum The datum.
* @param cs The coordinate system.
*/
public DefaultGeographicCRS(final String name,
final GeodeticDatum datum,
final EllipsoidalCS cs)
{
this(Collections.singletonMap(NAME_KEY, name), datum, cs);
}
/**
* Constructs a geographic CRS from a set of properties. The properties are given unchanged to
* the {@linkplain AbstractReferenceSystem#AbstractReferenceSystem(Map) super-class constructor}.
*
* @param properties Set of properties. Should contains at least {@code "name"}.
* @param datum The datum.
* @param cs The coordinate system.
*/
public DefaultGeographicCRS(final Map properties,
final GeodeticDatum datum,
final EllipsoidalCS cs)
{
super(properties, datum, cs);
}
/**
* Returns a Geotk CRS implementation with the same values than the given arbitrary
* implementation. If the given object is {@code null}, then this method returns {@code null}.
* Otherwise if the given object is already a Geotk implementation, then the given object is
* returned unchanged. Otherwise a new Geotk implementation is created and initialized to the
* attribute values of the given object.
*
* @param object The object to get as a Geotk implementation, or {@code null} if none.
* @return A Geotk implementation containing the values of the given object (may be the
* given object itself), or {@code null} if the argument was null.
*
* @since 3.18
*/
public static DefaultGeographicCRS castOrCopy(final GeographicCRS object) {
return (object == null) || (object instanceof DefaultGeographicCRS)
? (DefaultGeographicCRS) object : new DefaultGeographicCRS(object);
}
/**
* Returns the coordinate system.
*/
@Override
@XmlElement(name="ellipsoidalCS")
public EllipsoidalCS getCoordinateSystem() {
return (EllipsoidalCS) super.getCoordinateSystem();
}
/**
* Used by JAXB only (invoked by reflection).
*/
final void setCoordinateSystem(final EllipsoidalCS cs) {
super.setCoordinateSystem(cs);
}
/**
* Returns the datum.
*/
@Override
@XmlElement(name="geodeticDatum")
public GeodeticDatum getDatum() {
return (GeodeticDatum) super.getDatum();
}
/**
* Used by JAXB only (invoked by reflection).
*/
final void setDatum(final GeodeticDatum datum) {
super.setDatum(datum);
}
/**
* Computes the orthodromic distance between two points. This convenience method delegates
* the work to the underlying {@linkplain DefaultEllipsoid ellipsoid}, if possible.
*
* @param coord1 Coordinates of the first point.
* @param coord2 Coordinates of the second point.
* @return The distance between {@code coord1} and {@code coord2}.
* @throws UnsupportedOperationException if this coordinate reference system can't compute
* distances.
* @throws MismatchedDimensionException if a coordinate doesn't have the expected dimension.
*/
@Override
public Measure distance(final double[] coord1, final double[] coord2)
throws UnsupportedOperationException, MismatchedDimensionException
{
final CoordinateSystem coordinateSystem = getCoordinateSystem();
if (!(coordinateSystem instanceof DefaultEllipsoidalCS)) {
throw new UnsupportedImplementationException(coordinateSystem.getClass());
}
final Ellipsoid ellipsoid = getDatum().getEllipsoid();
if (!(ellipsoid instanceof DefaultEllipsoid)) {
throw new UnsupportedImplementationException(ellipsoid.getClass());
}
final DefaultEllipsoidalCS cs = (DefaultEllipsoidalCS) coordinateSystem;
final DefaultEllipsoid e = (DefaultEllipsoid) ellipsoid;
if (coord1.length!=2 || coord2.length!=2 || cs.getDimension()!=2) {
/*
* Not yet implemented (an exception will be thrown later).
* We should probably revisit the way we compute distances.
*/
return super.distance(coord1, coord2);
}
return new Measure(e.orthodromicDistance(cs.getLongitude(coord1),
cs.getLatitude (coord1),
cs.getLongitude(coord2),
cs.getLatitude (coord2)), e.getAxisUnit());
}
/**
* Returns the angular unit of the specified coordinate system.
* The preference will be given to the longitude axis, if found.
*/
static Unit getAngularUnit(final CoordinateSystem coordinateSystem) {
Unit unit = NonSI.DEGREE_ANGLE;
for (int i=coordinateSystem.getDimension(); --i>=0;) {
final CoordinateSystemAxis axis = coordinateSystem.getAxis(i);
final Unit> candidate = axis.getUnit();
if (Units.isAngular(candidate)) {
unit = candidate.asType(Angle.class);
if (AxisDirection.EAST.equals(AxisDirections.absolute(axis.getDirection()))) {
break; // Found the longitude axis.
}
}
}
return unit;
}
/**
* Returns a coordinate reference system with the same axes than this CRS, except that the
* longitude axis is shifted to a positive or negative range. This method can be used in
* order to shift between the [-180 … +180]° and [0 … 360]° ranges.
*
* This method shifts the axis {@linkplain CoordinateSystemAxis#getMinimumValue() minimum}
* and {@linkplain CoordinateSystemAxis#getMaximumValue() maximum} values by a multiple of
* 180°, converted to the units of the axis.
*
* This method does not change the meaning of ordinate values. For example a longitude of
* -60° still locate the same longitude in the old and the new coordinate system. But the
* preferred way to locate that longitude may become the 300° value if the range has been
* shifted to positive values.
*
* @param range {@link AxisRangeType#POSITIVE_LONGITUDE POSITIVE_LONGITUDE} for a range
* of positive longitude values, or {@link AxisRangeType#SPANNING_ZERO_LONGITUDE
* SPANNING_ZERO_LONGITUDE} for a range of positive and negative longitude values.
* @return A coordinate reference system using the given kind of longitude range
* (may be {@code this}).
*
* @see org.geotoolkit.referencing.cs.DefaultEllipsoidalCS#shiftAxisRange(AxisRangeType)
* @see DefaultCompoundCRS#shiftAxisRange(AxisRangeType)
*
* @since 3.20
*/
public DefaultGeographicCRS shiftAxisRange(final AxisRangeType range) {
DefaultGeographicCRS[] shifted;
synchronized (this) {
shifted = this.shifted;
if (shifted == null) {
this.shifted = shifted = new DefaultGeographicCRS[CRSUtilities.AXIS_RANGE_COUNT];
}
}
final int ordinal = range.ordinal();
DefaultGeographicCRS crs;
synchronized (shifted) {
crs = shifted[ordinal];
if (crs == null) {
final DefaultEllipsoidalCS oldCS = DefaultEllipsoidalCS.castOrCopy(getCoordinateSystem());
final DefaultEllipsoidalCS newCS = oldCS.shiftAxisRange(range);
if (oldCS == newCS) {
crs = this;
} else {
crs = new DefaultGeographicCRS(IdentifiedObjects.getProperties(this, null), getDatum(), newCS);
crs.shifted = shifted;
shifted[ordinal ^ CRSUtilities.AXIS_RANGE_RECIPROCAL_MASK] = this;
}
shifted[ordinal] = crs;
}
}
return crs;
}
/**
* Formats the inner part of a
* Well
* Known Text (WKT) element.
*
* @param formatter The formatter to use.
* @return The name of the WKT element type, which is {@code "GEOGCS"}.
*/
@Override
public String formatWKT(final Formatter formatter) {
final Unit oldUnit = formatter.getAngularUnit();
final Unit unit = getAngularUnit(getCoordinateSystem());
final GeodeticDatum datum = getDatum();
formatter.setAngularUnit(unit);
formatter.append(datum);
formatter.append(datum.getPrimeMeridian());
formatter.append(unit);
final EllipsoidalCS cs = getCoordinateSystem();
final int dimension = cs.getDimension();
for (int i=0; i