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

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





© 2015 - 2025 Weber Informatics LLC | Privacy Policy