diff options
| author | Even Rouault <even.rouault@spatialys.com> | 2020-04-19 15:46:43 +0200 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2020-04-19 15:46:43 +0200 |
| commit | b16b966b7484efd74a7364bd455ed3015d1009d6 (patch) | |
| tree | b3e4b57f04f76e14313be6dd19c3a41a6157b89a /src | |
| parent | dca24574f7c51123cf3d06cf4f33777873bf84e2 (diff) | |
| parent | 2998a46d7449e12d33b1d7f02b04b8ad2a5737da (diff) | |
| download | PROJ-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.cpp | 8 | ||||
| -rw-r--r-- | src/iso19111/io.cpp | 85 |
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"); |
