aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorEven Rouault <even.rouault@spatialys.com>2020-04-19 15:46:43 +0200
committerGitHub <noreply@github.com>2020-04-19 15:46:43 +0200
commitb16b966b7484efd74a7364bd455ed3015d1009d6 (patch)
treeb3e4b57f04f76e14313be6dd19c3a41a6157b89a /src
parentdca24574f7c51123cf3d06cf4f33777873bf84e2 (diff)
parent2998a46d7449e12d33b1d7f02b04b8ad2a5737da (diff)
downloadPROJ-b16b966b7484efd74a7364bd455ed3015d1009d6.tar.gz
PROJ-b16b966b7484efd74a7364bd455ed3015d1009d6.zip
Merge pull request #2166 from rouault/lcea_ellipsoidal_wkt1_ingestion
Ingestion of WKT1_GDAL: correctly map 'Cylindrical_Equal_Area'
Diffstat (limited to 'src')
-rw-r--r--src/iso19111/coordinateoperation.cpp8
-rw-r--r--src/iso19111/io.cpp85
2 files changed, 66 insertions, 27 deletions
diff --git a/src/iso19111/coordinateoperation.cpp b/src/iso19111/coordinateoperation.cpp
index 03eea4ee..4b174636 100644
--- a/src/iso19111/coordinateoperation.cpp
+++ b/src/iso19111/coordinateoperation.cpp
@@ -1797,6 +1797,14 @@ bool SingleOperation::_isEquivalentTo(const util::IComparable *other,
EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA &&
methodEPSGCode ==
EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA_SPHERICAL) ||
+ (methodEPSGCode ==
+ EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA &&
+ otherMethodEPSGCode ==
+ EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA_SPHERICAL) ||
+ (otherMethodEPSGCode ==
+ EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA &&
+ methodEPSGCode ==
+ EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA_SPHERICAL) ||
(methodEPSGCode == EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL &&
otherMethodEPSGCode ==
EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL_SPHERICAL) ||
diff --git a/src/iso19111/io.cpp b/src/iso19111/io.cpp
index 8968b70e..e516e4b5 100644
--- a/src/iso19111/io.cpp
+++ b/src/iso19111/io.cpp
@@ -1298,19 +1298,22 @@ struct WKTParser::Private {
static std::string projectionGetParameter(const WKTNodeNNPtr &projCRSNode,
const char *paramName);
- ConversionNNPtr buildProjection(const WKTNodeNNPtr &projCRSNode,
+ ConversionNNPtr buildProjection(const GeodeticCRSNNPtr &baseGeodCRS,
+ const WKTNodeNNPtr &projCRSNode,
const WKTNodeNNPtr &projectionNode,
const UnitOfMeasure &defaultLinearUnit,
const UnitOfMeasure &defaultAngularUnit);
ConversionNNPtr
- buildProjectionStandard(const WKTNodeNNPtr &projCRSNode,
+ buildProjectionStandard(const GeodeticCRSNNPtr &baseGeodCRS,
+ const WKTNodeNNPtr &projCRSNode,
const WKTNodeNNPtr &projectionNode,
const UnitOfMeasure &defaultLinearUnit,
const UnitOfMeasure &defaultAngularUnit);
ConversionNNPtr
- buildProjectionFromESRI(const WKTNodeNNPtr &projCRSNode,
+ buildProjectionFromESRI(const GeodeticCRSNNPtr &baseGeodCRS,
+ const WKTNodeNNPtr &projCRSNode,
const WKTNodeNNPtr &projectionNode,
const UnitOfMeasure &defaultLinearUnit,
const UnitOfMeasure &defaultAngularUnit);
@@ -3178,9 +3181,40 @@ bool WKTParser::Private::hasWebMercPROJ4String(
// ---------------------------------------------------------------------------
+static const MethodMapping *
+selectSphericalOrEllipsoidal(const MethodMapping *mapping,
+ const GeodeticCRSNNPtr &baseGeodCRS) {
+ if (mapping->epsg_code ==
+ EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA_SPHERICAL ||
+ mapping->epsg_code == EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA) {
+ mapping = getMapping(
+ baseGeodCRS->ellipsoid()->isSphere()
+ ? EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA_SPHERICAL
+ : EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA);
+ } else if (mapping->epsg_code ==
+ EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA_SPHERICAL ||
+ mapping->epsg_code ==
+ EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA) {
+ mapping = getMapping(
+ baseGeodCRS->ellipsoid()->isSphere()
+ ? EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA_SPHERICAL
+ : EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA);
+ } else if (mapping->epsg_code ==
+ EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL_SPHERICAL ||
+ mapping->epsg_code == EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL) {
+ mapping =
+ getMapping(baseGeodCRS->ellipsoid()->isSphere()
+ ? EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL_SPHERICAL
+ : EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL);
+ }
+ return mapping;
+}
+
+// ---------------------------------------------------------------------------
+
ConversionNNPtr WKTParser::Private::buildProjectionFromESRI(
- const WKTNodeNNPtr &projCRSNode, const WKTNodeNNPtr &projectionNode,
- const UnitOfMeasure &defaultLinearUnit,
+ const GeodeticCRSNNPtr &baseGeodCRS, const WKTNodeNNPtr &projCRSNode,
+ const WKTNodeNNPtr &projectionNode, const UnitOfMeasure &defaultLinearUnit,
const UnitOfMeasure &defaultAngularUnit) {
const std::string esriProjectionName =
stripQuotes(projectionNode->GP()->children()[0]);
@@ -3190,7 +3224,7 @@ ConversionNNPtr WKTParser::Private::buildProjectionFromESRI(
// on the parameters / their values
const auto esriMappings = getMappingsFromESRI(esriProjectionName);
if (esriMappings.empty()) {
- return buildProjectionStandard(projCRSNode, projectionNode,
+ return buildProjectionStandard(baseGeodCRS, projCRSNode, projectionNode,
defaultLinearUnit, defaultAngularUnit);
}
@@ -3273,6 +3307,8 @@ ConversionNNPtr WKTParser::Private::buildProjectionFromESRI(
}
assert(wkt2_mapping);
+ wkt2_mapping = selectSphericalOrEllipsoidal(wkt2_mapping, baseGeodCRS);
+
PropertyMap propertiesMethod;
propertiesMethod.set(IdentifiedObject::NAME_KEY, wkt2_mapping->wkt2_name);
if (wkt2_mapping->epsg_code != 0) {
@@ -3360,19 +3396,18 @@ ConversionNNPtr WKTParser::Private::buildProjectionFromESRI(
// ---------------------------------------------------------------------------
-ConversionNNPtr
-WKTParser::Private::buildProjection(const WKTNodeNNPtr &projCRSNode,
- const WKTNodeNNPtr &projectionNode,
- const UnitOfMeasure &defaultLinearUnit,
- const UnitOfMeasure &defaultAngularUnit) {
+ConversionNNPtr WKTParser::Private::buildProjection(
+ const GeodeticCRSNNPtr &baseGeodCRS, const WKTNodeNNPtr &projCRSNode,
+ const WKTNodeNNPtr &projectionNode, const UnitOfMeasure &defaultLinearUnit,
+ const UnitOfMeasure &defaultAngularUnit) {
if (projectionNode->GP()->childrenSize() == 0) {
ThrowNotEnoughChildren(WKTConstants::PROJECTION);
}
if (esriStyle_) {
- return buildProjectionFromESRI(projCRSNode, projectionNode,
+ return buildProjectionFromESRI(baseGeodCRS, projCRSNode, projectionNode,
defaultLinearUnit, defaultAngularUnit);
}
- return buildProjectionStandard(projCRSNode, projectionNode,
+ return buildProjectionStandard(baseGeodCRS, projCRSNode, projectionNode,
defaultLinearUnit, defaultAngularUnit);
}
@@ -3397,8 +3432,8 @@ WKTParser::Private::projectionGetParameter(const WKTNodeNNPtr &projCRSNode,
// ---------------------------------------------------------------------------
ConversionNNPtr WKTParser::Private::buildProjectionStandard(
- const WKTNodeNNPtr &projCRSNode, const WKTNodeNNPtr &projectionNode,
- const UnitOfMeasure &defaultLinearUnit,
+ const GeodeticCRSNNPtr &baseGeodCRS, const WKTNodeNNPtr &projCRSNode,
+ const WKTNodeNNPtr &projectionNode, const UnitOfMeasure &defaultLinearUnit,
const UnitOfMeasure &defaultAngularUnit) {
std::string wkt1ProjectionName =
stripQuotes(projectionNode->GP()->children()[0]);
@@ -3516,6 +3551,9 @@ ConversionNNPtr WKTParser::Private::buildProjectionStandard(
std::string projectionName(wkt1ProjectionName);
const MethodMapping *mapping =
tryToIdentifyWKT1Method ? getMappingFromWKT1(projectionName) : nullptr;
+ if (mapping) {
+ mapping = selectSphericalOrEllipsoidal(mapping, baseGeodCRS);
+ }
// For Krovak, we need to look at axis to decide between the Krovak and
// Krovak East-North Oriented methods
@@ -3768,7 +3806,8 @@ WKTParser::Private::buildProjectedCRS(const WKTNodeNNPtr &node) {
auto conversion =
!isNull(conversionNode)
? buildConversion(conversionNode, linearUnit, angularUnit)
- : buildProjection(node, projectionNode, linearUnit, angularUnit);
+ : buildProjection(baseGeodCRS, node, projectionNode, linearUnit,
+ angularUnit);
// No explicit AXIS node ? (WKT1)
if (isNull(nodeP->lookForChild(WKTConstants::AXIS))) {
@@ -8677,6 +8716,9 @@ CRSNNPtr PROJStringParser::Private::buildProjectedCRS(
auto &step = steps_[iStep];
auto mappings = getMappingsFromPROJName(step.name);
const MethodMapping *mapping = mappings.empty() ? nullptr : mappings[0];
+ if (mapping) {
+ mapping = selectSphericalOrEllipsoidal(mapping, geogCRS);
+ }
assert(isProjectedStep(step.name));
assert(iUnitConvert < 0 ||
@@ -8722,8 +8764,6 @@ CRSNNPtr PROJStringParser::Private::buildProjectedCRS(
}
} else if (step.name == "aeqd" && hasParamValue(step, "guam")) {
mapping = getMapping(EPSG_CODE_METHOD_GUAM_PROJECTION);
- } else if (step.name == "cea" && !geogCRS->ellipsoid()->isSphere()) {
- mapping = getMapping(EPSG_CODE_METHOD_LAMBERT_CYLINDRICAL_EQUAL_AREA);
} else if (step.name == "geos" && getParamValue(step, "sweep") == "x") {
mapping =
getMapping(PROJ_WKT2_NAME_METHOD_GEOSTATIONARY_SATELLITE_SWEEP_X);
@@ -8829,15 +8869,6 @@ CRSNNPtr PROJStringParser::Private::buildProjectedCRS(
axisType = AxisType::SOUTH_POLE;
}
}
- if (geogCRS->ellipsoid()->isSphere()) {
- mapping = getMapping(
- EPSG_CODE_METHOD_LAMBERT_AZIMUTHAL_EQUAL_AREA_SPHERICAL);
- }
- } else if (step.name == "eqc") {
- if (geogCRS->ellipsoid()->isSphere()) {
- mapping =
- getMapping(EPSG_CODE_METHOD_EQUIDISTANT_CYLINDRICAL_SPHERICAL);
- }
}
UnitOfMeasure unit = buildUnit(step, "units", "to_meter");