LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/libkml - ogrlibkmlgeometry.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 345 404 85.4 %
Date: 2025-10-24 00:53:13 Functions: 6 8 75.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  KML Translator
       4             :  * Purpose:  Implements OGRLIBKMLDriver
       5             :  * Author:   Brian Case, rush at winkey dot org
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2010, Brian Case
       9             :  * Copyright (c) 2010-2014, Even Rouault <even dot rouault at spatialys.com>
      10             :  *
      11             :  * SPDX-License-Identifier: MIT
      12             :  *****************************************************************************/
      13             : 
      14             : #include "libkml_headers.h"
      15             : 
      16             : #include "ogr_geometry.h"
      17             : #include "ogr_p.h"
      18             : #include "ogrlibkmlgeometry.h"
      19             : 
      20             : using kmlbase::Vec3;
      21             : using kmldom::CoordinatesPtr;
      22             : using kmldom::ElementPtr;
      23             : using kmldom::GeometryPtr;
      24             : using kmldom::GxLatLonQuadPtr;
      25             : using kmldom::GxMultiTrackPtr;
      26             : using kmldom::GxTrackPtr;
      27             : using kmldom::InnerBoundaryIsPtr;
      28             : using kmldom::KmlFactory;
      29             : using kmldom::LatLonBoxPtr;
      30             : using kmldom::LinearRingPtr;
      31             : using kmldom::LineStringPtr;
      32             : using kmldom::MultiGeometryPtr;
      33             : using kmldom::OuterBoundaryIsPtr;
      34             : using kmldom::PointPtr;
      35             : using kmldom::PolygonPtr;
      36             : 
      37         797 : static bool NormalizeLongLat(double &x, double &y)
      38             : {
      39         797 :     if (x >= -180 && x <= 180)
      40             :     {
      41             :         // nominal
      42             :     }
      43           6 :     else if (x > 180 && x <= 180 + 360)
      44           2 :         x -= 360;
      45           4 :     else if (x < -180 && x >= -180 - 360)
      46           2 :         x += 360;
      47             :     else
      48             :     {
      49             :         const bool bStrictCompliance =
      50           2 :             CPLTestBool(CPLGetConfigOption("LIBKML_STRICT_COMPLIANCE", "TRUE"));
      51           2 :         CPLError(bStrictCompliance ? CE_Failure : CE_Warning, CPLE_AppDefined,
      52             :                  "Invalid longitude %g", x);
      53           2 :         if (bStrictCompliance)
      54           2 :             return false;
      55             :     }
      56             : 
      57         795 :     constexpr double EPSILON = 1e-8;
      58         795 :     if (y >= -90 && y <= 90)
      59             :     {
      60             :         // nominal
      61             :     }
      62          11 :     else if (y > 90 && y < 90 + EPSILON)
      63             :     {
      64           1 :         y = 90;
      65             :     }
      66          10 :     else if (y < -90 && y > -90 - EPSILON)
      67             :     {
      68           1 :         y = -90;
      69             :     }
      70             :     else
      71             :     {
      72             :         const bool bStrictCompliance =
      73           9 :             CPLTestBool(CPLGetConfigOption("LIBKML_STRICT_COMPLIANCE", "TRUE"));
      74           9 :         CPLError(bStrictCompliance ? CE_Failure : CE_Warning, CPLE_AppDefined,
      75             :                  "Invalid latitude %g", y);
      76           9 :         if (bStrictCompliance)
      77           9 :             return false;
      78             :     }
      79         786 :     return true;
      80             : }
      81             : 
      82             : /******************************************************************************
      83             :  Function to write out a ogr geometry to kml.
      84             : 
      85             : Args:
      86             :           poOgrGeom     the ogr geometry
      87             :           extra         used in recursion, just pass -1
      88             :           poKmlFactory  pointer to the libkml dom factory
      89             : 
      90             : Returns:
      91             :           ElementPtr to the geometry created
      92             : 
      93             : ******************************************************************************/
      94             : 
      95         323 : ElementPtr geom2kml(OGRGeometry *poOgrGeom, int extra, KmlFactory *poKmlFactory)
      96             : {
      97         323 :     if (!poOgrGeom)
      98             :     {
      99           0 :         return nullptr;
     100             :     }
     101             : 
     102             :     /***** libkml geom vars *****/
     103         646 :     CoordinatesPtr coordinates = nullptr;
     104             : 
     105             :     // This will be the return value.
     106         646 :     ElementPtr poKmlGeometry = nullptr;
     107             : 
     108             :     /***** Other vars *****/
     109         323 :     int numpoints = 0;
     110         323 :     const OGRwkbGeometryType type = poOgrGeom->getGeometryType();
     111             : 
     112         323 :     switch (type)
     113             :     {
     114          64 :         case wkbPoint:
     115             :         {
     116          64 :             const OGRPoint *poOgrPoint = poOgrGeom->toPoint();
     117          64 :             PointPtr poKmlPoint = nullptr;
     118          64 :             if (poOgrPoint->getCoordinateDimension() == 0)
     119             :             {
     120           0 :                 poKmlPoint = poKmlFactory->CreatePoint();
     121           0 :                 poKmlGeometry = poKmlPoint;
     122             :             }
     123             :             else
     124             :             {
     125          64 :                 double x = poOgrPoint->getX();
     126          64 :                 double y = poOgrPoint->getY();
     127             : 
     128          64 :                 if (!NormalizeLongLat(x, y))
     129           6 :                     return nullptr;
     130             : 
     131          58 :                 coordinates = poKmlFactory->CreateCoordinates();
     132          58 :                 coordinates->add_latlng(y, x);
     133          58 :                 poKmlPoint = poKmlFactory->CreatePoint();
     134          58 :                 poKmlGeometry = poKmlPoint;
     135          58 :                 poKmlPoint->set_coordinates(coordinates);
     136             :             }
     137             : 
     138          58 :             break;
     139             :         }
     140          26 :         case wkbPoint25D:
     141             :         {
     142          26 :             const OGRPoint *poOgrPoint = poOgrGeom->toPoint();
     143             : 
     144          26 :             double x = poOgrPoint->getX();
     145          26 :             double y = poOgrPoint->getY();
     146          26 :             const double z = poOgrPoint->getZ();
     147             : 
     148          26 :             if (!NormalizeLongLat(x, y))
     149           1 :                 return nullptr;
     150             : 
     151          25 :             coordinates = poKmlFactory->CreateCoordinates();
     152          25 :             coordinates->add_latlngalt(y, x, z);
     153          50 :             PointPtr poKmlPoint = poKmlFactory->CreatePoint();
     154          25 :             poKmlGeometry = poKmlPoint;
     155          25 :             poKmlPoint->set_coordinates(coordinates);
     156             : 
     157          25 :             break;
     158             :         }
     159          53 :         case wkbLineString:
     160             :         {
     161          53 :             OGRLineString *poOgrLineString = poOgrGeom->toLineString();
     162             : 
     163          53 :             if (extra >= 0)
     164             :             {
     165          23 :                 poOgrGeom->toLinearRing()->closeRings();
     166             :             }
     167             : 
     168          53 :             numpoints = poOgrLineString->getNumPoints();
     169          53 :             if (extra >= 0)
     170             :             {
     171          23 :                 if (numpoints < 4 && CPLTestBool(CPLGetConfigOption(
     172             :                                          "LIBKML_STRICT_COMPLIANCE", "TRUE")))
     173             :                 {
     174           0 :                     CPLError(CE_Failure, CPLE_NotSupported,
     175             :                              "A linearring should have at least 4 points");
     176           2 :                     return nullptr;
     177             :                 }
     178             :             }
     179             :             else
     180             :             {
     181          30 :                 if (numpoints < 2 && CPLTestBool(CPLGetConfigOption(
     182             :                                          "LIBKML_STRICT_COMPLIANCE", "TRUE")))
     183             :                 {
     184           0 :                     CPLError(CE_Failure, CPLE_NotSupported,
     185             :                              "A linestring should have at least 2 points");
     186           0 :                     return nullptr;
     187             :                 }
     188             :             }
     189             : 
     190          53 :             coordinates = poKmlFactory->CreateCoordinates();
     191             : 
     192          53 :             OGRPoint point;
     193             : 
     194         419 :             for (int i = 0; i < numpoints; i++)
     195             :             {
     196         368 :                 poOgrLineString->getPoint(i, &point);
     197             : 
     198         368 :                 double x = point.getX();
     199         368 :                 double y = point.getY();
     200             : 
     201         368 :                 if (!NormalizeLongLat(x, y))
     202           2 :                     return nullptr;
     203             : 
     204         366 :                 coordinates->add_latlng(y, x);
     205             :             }
     206             : 
     207             :             /***** Check if its a wkbLinearRing *****/
     208          51 :             if (extra < 0)
     209             :             {
     210             :                 LineStringPtr poKmlLineString =
     211          58 :                     poKmlFactory->CreateLineString();
     212          29 :                 poKmlGeometry = poKmlLineString;
     213          29 :                 poKmlLineString->set_coordinates(coordinates);
     214             : 
     215          29 :                 break;
     216             :             }
     217             :             [[fallthrough]];
     218             :         }
     219             :             /***** fallthrough *****/
     220             : 
     221             :         case wkbLinearRing:  // This case is for readability only.
     222             :         {
     223          44 :             LinearRingPtr poKmlLinearRing = poKmlFactory->CreateLinearRing();
     224          22 :             poKmlLinearRing->set_coordinates(coordinates);
     225             : 
     226          22 :             if (!extra)
     227             :             {
     228             :                 OuterBoundaryIsPtr poKmlOuterRing =
     229          42 :                     poKmlFactory->CreateOuterBoundaryIs();
     230          21 :                 poKmlOuterRing->set_linearring(poKmlLinearRing);
     231          21 :                 poKmlGeometry = poKmlOuterRing;
     232             :             }
     233             :             else
     234             :             {
     235             :                 InnerBoundaryIsPtr poKmlInnerRing =
     236           2 :                     poKmlFactory->CreateInnerBoundaryIs();
     237           1 :                 poKmlGeometry = poKmlInnerRing;
     238           1 :                 poKmlInnerRing->set_linearring(poKmlLinearRing);
     239             :             }
     240             : 
     241          22 :             break;
     242             :         }
     243          66 :         case wkbLineString25D:
     244             :         {
     245          66 :             const OGRLineString *poOgrLineString = poOgrGeom->toLineString();
     246             : 
     247          66 :             if (extra >= 0)
     248             :             {
     249          49 :                 poOgrGeom->toLinearRing()->closeRings();
     250             :             }
     251             : 
     252          66 :             numpoints = poOgrLineString->getNumPoints();
     253          66 :             if (extra >= 0)
     254             :             {
     255          49 :                 if (numpoints < 4 && CPLTestBool(CPLGetConfigOption(
     256             :                                          "LIBKML_STRICT_COMPLIANCE", "TRUE")))
     257             :                 {
     258           0 :                     CPLError(CE_Failure, CPLE_NotSupported,
     259             :                              "A linearring should have at least 4 points");
     260           2 :                     return nullptr;
     261             :                 }
     262             :             }
     263             :             else
     264             :             {
     265          17 :                 if (numpoints < 2 && CPLTestBool(CPLGetConfigOption(
     266             :                                          "LIBKML_STRICT_COMPLIANCE", "TRUE")))
     267             :                 {
     268           0 :                     CPLError(CE_Failure, CPLE_NotSupported,
     269             :                              "A linestring should have at least 2 points");
     270           0 :                     return nullptr;
     271             :                 }
     272             :             }
     273             : 
     274          66 :             coordinates = poKmlFactory->CreateCoordinates();
     275          66 :             OGRPoint point;
     276             : 
     277         403 :             for (int i = 0; i < numpoints; i++)
     278             :             {
     279         339 :                 poOgrLineString->getPoint(i, &point);
     280             : 
     281         339 :                 double x = point.getX();
     282         339 :                 double y = point.getY();
     283         339 :                 const double z = point.getZ();
     284             : 
     285         339 :                 if (!NormalizeLongLat(x, y))
     286           2 :                     return nullptr;
     287             : 
     288         337 :                 coordinates->add_latlngalt(y, x, z);
     289             :             }
     290             : 
     291             :             /***** Check if its a wkbLinearRing *****/
     292          64 :             if (extra < 0)
     293             :             {
     294             :                 LineStringPtr poKmlLineString =
     295          32 :                     poKmlFactory->CreateLineString();
     296          16 :                 poKmlGeometry = poKmlLineString;
     297          16 :                 poKmlLineString->set_coordinates(coordinates);
     298             : 
     299          16 :                 break;
     300             :             }
     301             :             /***** fallthrough *****/
     302             : 
     303             :             // case wkbLinearRing25D: // This case is for readability only.
     304             : 
     305          96 :             LinearRingPtr poKmlLinearRing = poKmlFactory->CreateLinearRing();
     306          48 :             poKmlLinearRing->set_coordinates(coordinates);
     307             : 
     308          48 :             if (!extra)
     309             :             {
     310             :                 OuterBoundaryIsPtr poKmlOuterRing =
     311          70 :                     poKmlFactory->CreateOuterBoundaryIs();
     312          35 :                 poKmlGeometry = poKmlOuterRing;
     313          35 :                 poKmlOuterRing->set_linearring(poKmlLinearRing);
     314             :             }
     315             :             else
     316             :             {
     317             :                 InnerBoundaryIsPtr poKmlInnerRing =
     318          26 :                     poKmlFactory->CreateInnerBoundaryIs();
     319          13 :                 poKmlGeometry = poKmlInnerRing;
     320          13 :                 poKmlInnerRing->set_linearring(poKmlLinearRing);
     321             :             }
     322             : 
     323          48 :             break;
     324             :         }
     325          23 :         case wkbPolygon:
     326             :         {
     327          23 :             if (CPLTestBool(
     328          46 :                     CPLGetConfigOption("LIBKML_STRICT_COMPLIANCE", "TRUE")) &&
     329          23 :                 OGRGeometryFactory::haveGEOS())
     330             :             {
     331             :                 bool bError;
     332             :                 {
     333          23 :                     CPLErrorStateBackuper oErrorStateBackuper;
     334          23 :                     bError = !poOgrGeom->IsValid();
     335             :                 }
     336          23 :                 if (bError)
     337             :                 {
     338           1 :                     CPLError(CE_Failure, CPLE_NotSupported, "Invalid polygon");
     339           3 :                     return nullptr;
     340             :                 }
     341             :             }
     342             : 
     343          22 :             PolygonPtr poKmlPolygon = poKmlFactory->CreatePolygon();
     344          22 :             poKmlGeometry = poKmlPolygon;
     345             : 
     346          22 :             OGRPolygon *poOgrPolygon = poOgrGeom->toPolygon();
     347             :             ElementPtr poKmlTmpGeometry =
     348          22 :                 geom2kml(poOgrPolygon->getExteriorRing(), 0, poKmlFactory);
     349          22 :             if (!poKmlTmpGeometry)
     350           1 :                 return nullptr;
     351          42 :             poKmlPolygon->set_outerboundaryis(
     352          42 :                 AsOuterBoundaryIs(poKmlTmpGeometry));
     353             : 
     354          21 :             const int nGeom = poOgrPolygon->getNumInteriorRings();
     355          22 :             for (int i = 0; i < nGeom; i++)
     356             :             {
     357           2 :                 poKmlTmpGeometry = geom2kml(poOgrPolygon->getInteriorRing(i),
     358           1 :                                             i + 1, poKmlFactory);
     359           1 :                 if (!poKmlTmpGeometry)
     360           0 :                     return nullptr;
     361           2 :                 poKmlPolygon->add_innerboundaryis(
     362           2 :                     AsInnerBoundaryIs(poKmlTmpGeometry));
     363             :             }
     364             : 
     365          21 :             break;
     366             :         }
     367          37 :         case wkbPolygon25D:
     368             :         {
     369          37 :             if (CPLTestBool(
     370          74 :                     CPLGetConfigOption("LIBKML_STRICT_COMPLIANCE", "TRUE")) &&
     371          37 :                 OGRGeometryFactory::haveGEOS())
     372             :             {
     373             :                 bool bError;
     374             :                 {
     375          37 :                     CPLErrorStateBackuper oErrorStateBackuper;
     376          37 :                     bError = !poOgrGeom->IsValid();
     377             :                 }
     378          37 :                 if (bError)
     379             :                 {
     380           1 :                     CPLError(CE_Failure, CPLE_NotSupported, "Invalid polygon");
     381           3 :                     return nullptr;
     382             :                 }
     383             :             }
     384             : 
     385          36 :             PolygonPtr poKmlPolygon = poKmlFactory->CreatePolygon();
     386          36 :             poKmlGeometry = poKmlPolygon;
     387             : 
     388          36 :             OGRPolygon *poOgrPolygon = poOgrGeom->toPolygon();
     389             :             ElementPtr poKmlTmpGeometry =
     390          36 :                 geom2kml(poOgrPolygon->getExteriorRing(), 0, poKmlFactory);
     391          36 :             if (!poKmlTmpGeometry)
     392           1 :                 return nullptr;
     393          70 :             poKmlPolygon->set_outerboundaryis(
     394          70 :                 AsOuterBoundaryIs(poKmlTmpGeometry));
     395             : 
     396          35 :             const int nGeom = poOgrPolygon->getNumInteriorRings();
     397          48 :             for (int i = 0; i < nGeom; i++)
     398             :             {
     399          26 :                 poKmlTmpGeometry = geom2kml(poOgrPolygon->getInteriorRing(i),
     400          13 :                                             i + 1, poKmlFactory);
     401          13 :                 if (!poKmlTmpGeometry)
     402           0 :                     return nullptr;
     403          26 :                 poKmlPolygon->add_innerboundaryis(
     404          26 :                     AsInnerBoundaryIs(poKmlTmpGeometry));
     405             :             }
     406             : 
     407          35 :             break;
     408             :         }
     409          54 :         case wkbMultiPoint:
     410             :         case wkbMultiLineString:
     411             :         case wkbMultiPolygon:
     412             :         case wkbGeometryCollection:
     413             :         case wkbMultiPoint25D:
     414             :         case wkbMultiLineString25D:
     415             :         case wkbMultiPolygon25D:
     416             :         case wkbGeometryCollection25D:
     417             :         {
     418             :             OGRGeometryCollection *poOgrMultiGeom =
     419          54 :                 poOgrGeom->toGeometryCollection();
     420             : 
     421          54 :             const int nGeom = poOgrMultiGeom->getNumGeometries();
     422             : 
     423          54 :             if (nGeom == 1 && CPLTestBool(CPLGetConfigOption(
     424             :                                   "LIBKML_STRICT_COMPLIANCE", "TRUE")))
     425             :             {
     426          21 :                 CPLDebug("LIBKML",
     427             :                          "Turning multiple geometry into single geometry");
     428          42 :                 poKmlGeometry = geom2kml(poOgrMultiGeom->getGeometryRef(0), -1,
     429          21 :                                          poKmlFactory);
     430          21 :                 if (!poKmlGeometry)
     431           1 :                     return nullptr;
     432             :             }
     433             :             else
     434             :             {
     435          33 :                 if (nGeom == 0 && CPLTestBool(CPLGetConfigOption(
     436             :                                       "LIBKML_STRICT_COMPLIANCE", "TRUE")))
     437             :                 {
     438           1 :                     CPLError(CE_Warning, CPLE_AppDefined,
     439             :                              "Empty multi geometry are not recommended");
     440             :                 }
     441             : 
     442             :                 MultiGeometryPtr poKmlMultiGeometry =
     443          33 :                     poKmlFactory->CreateMultiGeometry();
     444          33 :                 poKmlGeometry = poKmlMultiGeometry;
     445             : 
     446         102 :                 for (int i = 0; i < nGeom; i++)
     447             :                 {
     448             :                     ElementPtr poKmlTmpGeometry = geom2kml(
     449          70 :                         poOgrMultiGeom->getGeometryRef(i), -1, poKmlFactory);
     450          70 :                     if (!poKmlTmpGeometry)
     451           1 :                         return nullptr;
     452         138 :                     poKmlMultiGeometry->add_geometry(
     453         138 :                         AsGeometry(std::move(poKmlTmpGeometry)));
     454             :                 }
     455             :             }
     456             : 
     457          52 :             break;
     458             :         }
     459           0 :         case wkbUnknown:
     460             :         case wkbNone:
     461             :         default:
     462           0 :             break;
     463             :     }
     464             : 
     465         306 :     return poKmlGeometry;
     466             : }
     467             : 
     468             : /******************************************************************************
     469             :  Recursive function to read a kml geometry and translate to ogr.
     470             : 
     471             : Args:
     472             :             poKmlGeometry   pointer to the kml geometry to translate
     473             :             poOgrSRS        pointer to the spatial ref to set on the geometry
     474             : 
     475             : Returns:
     476             :             pointer to the new ogr geometry object
     477             : 
     478             : ******************************************************************************/
     479             : 
     480        2090 : static OGRGeometry *kml2geom_rec(const GeometryPtr &poKmlGeometry,
     481             :                                  OGRSpatialReference *poOgrSRS)
     482             : {
     483             :     /***** ogr geom vars *****/
     484        2090 :     OGRPoint *poOgrPoint = nullptr;
     485        2090 :     OGRLineString *poOgrLineString = nullptr;
     486        2090 :     OGRLinearRing *poOgrLinearRing = nullptr;
     487        2090 :     OGRPolygon *poOgrPolygon = nullptr;
     488        2090 :     OGRGeometryCollection *poOgrMultiGeometry = nullptr;
     489        2090 :     OGRGeometry *poOgrGeometry = nullptr;
     490        2090 :     OGRGeometry *poOgrTmpGeometry = nullptr;
     491             : 
     492        2090 :     switch (poKmlGeometry->Type())
     493             :     {
     494         220 :         case kmldom::Type_Point:
     495             :         {
     496         440 :             PointPtr poKmlPoint = AsPoint(poKmlGeometry);
     497         220 :             if (poKmlPoint->has_coordinates())
     498             :             {
     499         438 :                 CoordinatesPtr poKmlCoordinates = poKmlPoint->get_coordinates();
     500             :                 const size_t nCoords =
     501         219 :                     poKmlCoordinates->get_coordinates_array_size();
     502         219 :                 if (nCoords > 0)
     503             :                 {
     504             :                     const Vec3 oKmlVec =
     505         218 :                         poKmlCoordinates->get_coordinates_array_at(0);
     506             : 
     507         218 :                     if (oKmlVec.has_altitude())
     508         420 :                         poOgrPoint = new OGRPoint(oKmlVec.get_longitude(),
     509         210 :                                                   oKmlVec.get_latitude(),
     510         210 :                                                   oKmlVec.get_altitude());
     511             :                     else
     512          16 :                         poOgrPoint = new OGRPoint(oKmlVec.get_longitude(),
     513           8 :                                                   oKmlVec.get_latitude());
     514             : 
     515         218 :                     poOgrGeometry = poOgrPoint;
     516             :                 }
     517             :                 else
     518             :                 {
     519           1 :                     poOgrGeometry = new OGRPoint();
     520             :                 }
     521             :             }
     522             :             else
     523             :             {
     524           1 :                 poOgrGeometry = new OGRPoint();
     525             :             }
     526             : 
     527         220 :             break;
     528             :         }
     529         247 :         case kmldom::Type_LineString:
     530             :         {
     531         494 :             LineStringPtr poKmlLineString = AsLineString(poKmlGeometry);
     532         247 :             poOgrLineString = new OGRLineString();
     533         247 :             if (poKmlLineString->has_coordinates())
     534             :             {
     535             :                 CoordinatesPtr poKmlCoordinates =
     536         492 :                     poKmlLineString->get_coordinates();
     537             : 
     538             :                 const size_t nCoords =
     539         246 :                     poKmlCoordinates->get_coordinates_array_size();
     540        1922 :                 for (size_t i = 0; i < nCoords; i++)
     541             :                 {
     542             :                     const Vec3 oKmlVec =
     543        1676 :                         poKmlCoordinates->get_coordinates_array_at(i);
     544        1676 :                     if (oKmlVec.has_altitude())
     545        1668 :                         poOgrLineString->addPoint(oKmlVec.get_longitude(),
     546             :                                                   oKmlVec.get_latitude(),
     547             :                                                   oKmlVec.get_altitude());
     548             :                     else
     549           8 :                         poOgrLineString->addPoint(oKmlVec.get_longitude(),
     550             :                                                   oKmlVec.get_latitude());
     551             :                 }
     552             :             }
     553         247 :             poOgrGeometry = poOgrLineString;
     554             : 
     555         247 :             break;
     556             :         }
     557         824 :         case kmldom::Type_LinearRing:
     558             :         {
     559        1648 :             LinearRingPtr poKmlLinearRing = AsLinearRing(poKmlGeometry);
     560         824 :             poOgrLinearRing = new OGRLinearRing();
     561         824 :             if (poKmlLinearRing->has_coordinates())
     562             :             {
     563             :                 CoordinatesPtr poKmlCoordinates =
     564        1642 :                     poKmlLinearRing->get_coordinates();
     565             : 
     566             :                 const size_t nCoords =
     567         821 :                     poKmlCoordinates->get_coordinates_array_size();
     568       16037 :                 for (size_t i = 0; i < nCoords; i++)
     569             :                 {
     570             :                     const Vec3 oKmlVec =
     571       15216 :                         poKmlCoordinates->get_coordinates_array_at(i);
     572       15216 :                     if (oKmlVec.has_altitude())
     573       15174 :                         poOgrLinearRing->addPoint(oKmlVec.get_longitude(),
     574             :                                                   oKmlVec.get_latitude(),
     575             :                                                   oKmlVec.get_altitude());
     576             :                     else
     577          42 :                         poOgrLinearRing->addPoint(oKmlVec.get_longitude(),
     578             :                                                   oKmlVec.get_latitude());
     579             :                 }
     580             :             }
     581         824 :             poOgrGeometry = poOgrLinearRing;
     582             : 
     583         824 :             break;
     584             :         }
     585         768 :         case kmldom::Type_Polygon:
     586             :         {
     587        1536 :             PolygonPtr poKmlPolygon = AsPolygon(poKmlGeometry);
     588             : 
     589         768 :             poOgrPolygon = new OGRPolygon();
     590         768 :             if (poKmlPolygon->has_outerboundaryis())
     591             :             {
     592             :                 OuterBoundaryIsPtr poKmlOuterRing =
     593        1534 :                     poKmlPolygon->get_outerboundaryis();
     594             :                 LinearRingPtr poKmlLinearRing =
     595        1534 :                     poKmlOuterRing->get_linearring();
     596         767 :                 if (poKmlLinearRing)
     597             :                 {
     598         766 :                     poOgrTmpGeometry = kml2geom_rec(poKmlLinearRing, poOgrSRS);
     599             : 
     600        1532 :                     poOgrPolygon->addRingDirectly(
     601         766 :                         poOgrTmpGeometry->toLinearRing());
     602             :                 }
     603             :             }
     604             :             const size_t nRings =
     605         768 :                 poKmlPolygon->get_innerboundaryis_array_size();
     606         824 :             for (size_t i = 0; i < nRings; i++)
     607             :             {
     608             :                 InnerBoundaryIsPtr poKmlInnerRing =
     609         112 :                     poKmlPolygon->get_innerboundaryis_array_at(i);
     610             :                 LinearRingPtr poKmlLinearRing =
     611         112 :                     poKmlInnerRing->get_linearring();
     612          56 :                 if (poKmlLinearRing)
     613             :                 {
     614          55 :                     poOgrTmpGeometry = kml2geom_rec(poKmlLinearRing, poOgrSRS);
     615             : 
     616         110 :                     poOgrPolygon->addRingDirectly(
     617          55 :                         poOgrTmpGeometry->toLinearRing());
     618             :                 }
     619             :             }
     620         768 :             poOgrGeometry = poOgrPolygon;
     621             : 
     622         768 :             break;
     623             :         }
     624          28 :         case kmldom::Type_MultiGeometry:
     625             :         {
     626             :             MultiGeometryPtr poKmlMultiGeometry =
     627          56 :                 AsMultiGeometry(poKmlGeometry);
     628          28 :             const size_t nGeom = poKmlMultiGeometry->get_geometry_array_size();
     629             : 
     630             :             // Detect subgeometry type to instantiate appropriate
     631             :             // multi geometry type.
     632          28 :             kmldom::KmlDomType type = kmldom::Type_Unknown;
     633          67 :             for (size_t i = 0; i < nGeom; i++)
     634             :             {
     635             :                 GeometryPtr poKmlTmpGeometry =
     636          44 :                     poKmlMultiGeometry->get_geometry_array_at(i);
     637          44 :                 if (type == kmldom::Type_Unknown)
     638             :                 {
     639          25 :                     type = poKmlTmpGeometry->Type();
     640             :                 }
     641          19 :                 else if (type != poKmlTmpGeometry->Type())
     642             :                 {
     643           5 :                     type = kmldom::Type_Unknown;
     644           5 :                     break;
     645             :                 }
     646             :             }
     647             : 
     648          28 :             if (type == kmldom::Type_Point)
     649           8 :                 poOgrMultiGeometry = new OGRMultiPoint();
     650          20 :             else if (type == kmldom::Type_LineString)
     651           6 :                 poOgrMultiGeometry = new OGRMultiLineString();
     652          14 :             else if (type == kmldom::Type_Polygon)
     653           6 :                 poOgrMultiGeometry = new OGRMultiPolygon();
     654             :             else
     655           8 :                 poOgrMultiGeometry = new OGRGeometryCollection();
     656             : 
     657          72 :             for (size_t i = 0; i < nGeom; i++)
     658             :             {
     659             :                 GeometryPtr poKmlTmpGeometry =
     660          88 :                     poKmlMultiGeometry->get_geometry_array_at(i);
     661          44 :                 poOgrTmpGeometry = kml2geom_rec(poKmlTmpGeometry, poOgrSRS);
     662             : 
     663          44 :                 poOgrMultiGeometry->addGeometryDirectly(poOgrTmpGeometry);
     664             :             }
     665          28 :             poOgrGeometry = poOgrMultiGeometry;
     666          28 :             break;
     667             :         }
     668           2 :         case kmldom::Type_GxTrack:
     669             :         {
     670           4 :             GxTrackPtr poKmlGxTrack = AsGxTrack(poKmlGeometry);
     671             : #if defined(__GNUC__)
     672             : #pragma GCC diagnostic push
     673             : #pragma GCC diagnostic ignored "-Wnull-dereference"
     674             : #endif
     675           2 :             const size_t nCoords = poKmlGxTrack->get_gx_coord_array_size();
     676             : #if defined(__GNUC__)
     677             : #pragma GCC diagnostic pop
     678             : #endif
     679           2 :             poOgrLineString = new OGRLineString();
     680          11 :             for (size_t i = 0; i < nCoords; i++)
     681             :             {
     682           9 :                 const Vec3 oKmlVec = poKmlGxTrack->get_gx_coord_array_at(i);
     683           9 :                 if (oKmlVec.has_altitude())
     684           7 :                     poOgrLineString->addPoint(oKmlVec.get_longitude(),
     685             :                                               oKmlVec.get_latitude(),
     686             :                                               oKmlVec.get_altitude());
     687             :                 else
     688           2 :                     poOgrLineString->addPoint(oKmlVec.get_longitude(),
     689             :                                               oKmlVec.get_latitude());
     690             :             }
     691           2 :             poOgrGeometry = poOgrLineString;
     692           2 :             break;
     693             :         }
     694           1 :         case kmldom::Type_GxMultiTrack:
     695             :         {
     696           2 :             GxMultiTrackPtr poKmlGxMultiTrack = AsGxMultiTrack(poKmlGeometry);
     697             : #if defined(__GNUC__)
     698             : #pragma GCC diagnostic push
     699             : #pragma GCC diagnostic ignored "-Wnull-dereference"
     700             : #endif
     701           1 :             const size_t nGeom = poKmlGxMultiTrack->get_gx_track_array_size();
     702             : #if defined(__GNUC__)
     703             : #pragma GCC diagnostic pop
     704             : #endif
     705           1 :             poOgrMultiGeometry = new OGRMultiLineString();
     706           2 :             for (size_t j = 0; j < nGeom; j++)
     707             :             {
     708             :                 GxTrackPtr poKmlGxTrack =
     709           2 :                     poKmlGxMultiTrack->get_gx_track_array_at(j);
     710           1 :                 const size_t nCoords = poKmlGxTrack->get_gx_coord_array_size();
     711           1 :                 poOgrLineString = new OGRLineString();
     712           3 :                 for (size_t i = 0; i < nCoords; i++)
     713             :                 {
     714           2 :                     const Vec3 oKmlVec = poKmlGxTrack->get_gx_coord_array_at(i);
     715           2 :                     if (oKmlVec.has_altitude())
     716           0 :                         poOgrLineString->addPoint(oKmlVec.get_longitude(),
     717             :                                                   oKmlVec.get_latitude(),
     718             :                                                   oKmlVec.get_altitude());
     719             :                     else
     720           2 :                         poOgrLineString->addPoint(oKmlVec.get_longitude(),
     721             :                                                   oKmlVec.get_latitude());
     722             :                 }
     723           1 :                 poOgrMultiGeometry->addGeometryDirectly(poOgrLineString);
     724             :             }
     725           1 :             poOgrGeometry = poOgrMultiGeometry;
     726           1 :             break;
     727             :         }
     728             : 
     729           0 :         default:
     730             :         {
     731           0 :             break;
     732             :         }
     733             :     }
     734             : 
     735        2090 :     if (poOgrGeometry)
     736        2090 :         poOgrGeometry->assignSpatialReference(poOgrSRS);
     737             : 
     738        2090 :     return poOgrGeometry;
     739             : }
     740             : 
     741          44 : static OGRGeometry *kml2geom_latlonbox_int(const LatLonBoxPtr &poKmlLatLonBox,
     742             :                                            OGRSpatialReference *poOgrSRS)
     743             : {
     744          88 :     if (!poKmlLatLonBox->has_north() || !poKmlLatLonBox->has_south() ||
     745          88 :         !poKmlLatLonBox->has_east() || !poKmlLatLonBox->has_west())
     746             :     {
     747           0 :         return nullptr;
     748             :     }
     749          44 :     const double north = poKmlLatLonBox->get_north();
     750          44 :     const double south = poKmlLatLonBox->get_south();
     751          44 :     const double east = poKmlLatLonBox->get_east();
     752          44 :     const double west = poKmlLatLonBox->get_west();
     753             : 
     754          44 :     OGRLinearRing *poOgrRing = new OGRLinearRing();
     755          44 :     poOgrRing->addPoint(east, north, 0.0);
     756          44 :     poOgrRing->addPoint(east, south, 0.0);
     757          44 :     poOgrRing->addPoint(west, south, 0.0);
     758          44 :     poOgrRing->addPoint(west, north, 0.0);
     759          44 :     poOgrRing->addPoint(east, north, 0.0);
     760             : 
     761          44 :     OGRPolygon *poOgrPolygon = new OGRPolygon();
     762          44 :     poOgrPolygon->addRingDirectly(poOgrRing);
     763          44 :     poOgrPolygon->assignSpatialReference(poOgrSRS);
     764             : 
     765          44 :     return poOgrPolygon;
     766             : }
     767             : 
     768             : static OGRGeometry *
     769           0 : kml2geom_latlonquad_int(const GxLatLonQuadPtr &poKmlLatLonQuad,
     770             :                         OGRSpatialReference *poOgrSRS)
     771             : {
     772           0 :     if (!poKmlLatLonQuad->has_coordinates())
     773           0 :         return nullptr;
     774             : 
     775           0 :     const CoordinatesPtr &poKmlCoordinates = poKmlLatLonQuad->get_coordinates();
     776             : 
     777           0 :     OGRLinearRing *poOgrLinearRing = new OGRLinearRing();
     778             : 
     779           0 :     size_t nCoords = poKmlCoordinates->get_coordinates_array_size();
     780           0 :     for (size_t i = 0; i < nCoords; i++)
     781             :     {
     782           0 :         Vec3 oKmlVec = poKmlCoordinates->get_coordinates_array_at(i);
     783           0 :         if (oKmlVec.has_altitude())
     784           0 :             poOgrLinearRing->addPoint(oKmlVec.get_longitude(),
     785             :                                       oKmlVec.get_latitude(),
     786             :                                       oKmlVec.get_altitude());
     787             :         else
     788           0 :             poOgrLinearRing->addPoint(oKmlVec.get_longitude(),
     789             :                                       oKmlVec.get_latitude());
     790             :     }
     791           0 :     poOgrLinearRing->closeRings();
     792             : 
     793           0 :     OGRPolygon *poOgrPolygon = new OGRPolygon();
     794           0 :     poOgrPolygon->addRingDirectly(poOgrLinearRing);
     795           0 :     poOgrPolygon->assignSpatialReference(poOgrSRS);
     796             : 
     797           0 :     return poOgrPolygon;
     798             : }
     799             : 
     800             : /******************************************************************************
     801             :  Main function to read a kml geometry and translate to ogr.
     802             : 
     803             : Args:
     804             :             poKmlGeometry   pointer to the kml geometry to translate
     805             :             poOgrSRS        pointer to the spatial ref to set on the geometry
     806             : 
     807             : Returns:
     808             :             pointer to the new ogr geometry object
     809             : 
     810             : ******************************************************************************/
     811             : 
     812        1225 : OGRGeometry *kml2geom(GeometryPtr poKmlGeometry, OGRSpatialReference *poOgrSRS)
     813             : {
     814             :     /***** Get the geometry *****/
     815             :     OGRGeometry *poOgrGeometry =
     816        1225 :         kml2geom_rec(std::move(poKmlGeometry), poOgrSRS);
     817             : 
     818             :     /***** Split the geometry at the dateline? *****/
     819        1225 :     const char *pszWrap = CPLGetConfigOption("LIBKML_WRAPDATELINE", "no");
     820        1225 :     if (!CPLTestBool(pszWrap))
     821        1225 :         return poOgrGeometry;
     822             : 
     823           0 :     char **papszTransformOptions = CSLAddString(nullptr, "WRAPDATELINE=YES");
     824             : 
     825             :     /***** Transform *****/
     826           0 :     OGRGeometry *poOgrDstGeometry = OGRGeometryFactory::transformWithOptions(
     827             :         poOgrGeometry, nullptr, papszTransformOptions);
     828             : 
     829             :     /***** Replace the original geom *****/
     830           0 :     if (poOgrDstGeometry)
     831             :     {
     832           0 :         delete poOgrGeometry;
     833           0 :         poOgrGeometry = poOgrDstGeometry;
     834             :     }
     835             : 
     836           0 :     CSLDestroy(papszTransformOptions);
     837             : 
     838           0 :     return poOgrGeometry;
     839             : }
     840             : 
     841          44 : OGRGeometry *kml2geom_latlonbox(LatLonBoxPtr poKmlLatLonBox,
     842             :                                 OGRSpatialReference *poOgrSRS)
     843             : {
     844             :     /***** Get the geometry *****/
     845             :     OGRGeometry *poOgrGeometry =
     846          44 :         kml2geom_latlonbox_int(poKmlLatLonBox, poOgrSRS);
     847             : 
     848             :     /***** Split the geometry at the dateline? *****/
     849          44 :     const char *pszWrap = CPLGetConfigOption("LIBKML_WRAPDATELINE", "no");
     850             : 
     851          44 :     if (!CPLTestBool(pszWrap))
     852          44 :         return poOgrGeometry;
     853             : 
     854           0 :     char **papszTransformOptions = CSLAddString(nullptr, "WRAPDATELINE=YES");
     855             : 
     856             :     /***** Transform *****/
     857           0 :     OGRGeometry *poOgrDstGeometry = OGRGeometryFactory::transformWithOptions(
     858             :         poOgrGeometry, nullptr, papszTransformOptions);
     859             : 
     860             :     /***** Replace the original geom *****/
     861           0 :     if (poOgrDstGeometry)
     862             :     {
     863           0 :         delete poOgrGeometry;
     864           0 :         poOgrGeometry = poOgrDstGeometry;
     865             :     }
     866             : 
     867           0 :     CSLDestroy(papszTransformOptions);
     868             : 
     869           0 :     return poOgrGeometry;
     870             : }
     871             : 
     872           0 : OGRGeometry *kml2geom_latlonquad(GxLatLonQuadPtr poKmlLatLonQuad,
     873             :                                  OGRSpatialReference *poOgrSRS)
     874             : {
     875             :     /***** Get the geometry *****/
     876             :     OGRGeometry *poOgrGeometry =
     877           0 :         kml2geom_latlonquad_int(poKmlLatLonQuad, poOgrSRS);
     878             : 
     879             :     /***** Split the geometry at the dateline? *****/
     880           0 :     const char *pszWrap = CPLGetConfigOption("LIBKML_WRAPDATELINE", "no");
     881           0 :     if (!CPLTestBool(pszWrap))
     882           0 :         return poOgrGeometry;
     883             : 
     884           0 :     char **papszTransformOptions = CSLAddString(nullptr, "WRAPDATELINE=YES");
     885             : 
     886             :     /***** Transform *****/
     887           0 :     OGRGeometry *poOgrDstGeometry = OGRGeometryFactory::transformWithOptions(
     888             :         poOgrGeometry, nullptr, papszTransformOptions);
     889             : 
     890             :     /***** Replace the original geom *****/
     891           0 :     if (poOgrDstGeometry)
     892             :     {
     893           0 :         delete poOgrGeometry;
     894           0 :         poOgrGeometry = poOgrDstGeometry;
     895             :     }
     896             : 
     897           0 :     CSLDestroy(papszTransformOptions);
     898             : 
     899           0 :     return poOgrGeometry;
     900             : }

Generated by: LCOV version 1.14