Line data Source code
1 : /******************************************************************************
2 : *
3 : * Project: OpenGIS Simple Features Reference Implementation
4 : * Purpose: The OGRCurveCollection class.
5 : * Author: Even Rouault, even dot rouault at spatialys dot com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2014, Even Rouault <even dot rouault at spatialys dot com>
9 : *
10 : * SPDX-License-Identifier: MIT
11 : ****************************************************************************/
12 :
13 : #include "cpl_port.h"
14 : #include "ogr_geometry.h"
15 :
16 : #include <cstddef>
17 : #include <cstring>
18 : #include <limits>
19 : #include <new>
20 :
21 : #include "ogr_core.h"
22 : #include "ogr_p.h"
23 : #include "ogr_spatialref.h"
24 : #include "cpl_conv.h"
25 : #include "cpl_error.h"
26 : #include "cpl_string.h"
27 : #include "cpl_vsi.h"
28 :
29 : //! @cond Doxygen_Suppress
30 :
31 : /************************************************************************/
32 : /* OGRCurveCollection( const OGRCurveCollection& ) */
33 : /************************************************************************/
34 :
35 : /**
36 : * \brief Copy constructor.
37 : *
38 : * Note: before GDAL 2.1, only the default implementation of the constructor
39 : * existed, which could be unsafe to use.
40 : *
41 : * @since GDAL 2.1
42 : */
43 :
44 1548740 : OGRCurveCollection::OGRCurveCollection(const OGRCurveCollection &other)
45 : {
46 1548740 : if (other.nCurveCount > 0)
47 : {
48 1548420 : nCurveCount = other.nCurveCount;
49 1548420 : papoCurves = static_cast<OGRCurve **>(
50 1548420 : VSI_CALLOC_VERBOSE(sizeof(void *), nCurveCount));
51 :
52 1548420 : if (papoCurves)
53 : {
54 3097990 : for (int i = 0; i < nCurveCount; i++)
55 : {
56 1549570 : papoCurves[i] = other.papoCurves[i]->clone();
57 : }
58 : }
59 : }
60 1548740 : }
61 :
62 : /************************************************************************/
63 : /* OGRCurveCollection( OGRCurveCollection&& ) */
64 : /************************************************************************/
65 :
66 : /**
67 : * \brief Move constructor.
68 : *
69 : * @since GDAL 3.11
70 : */
71 :
72 4 : OGRCurveCollection::OGRCurveCollection(OGRCurveCollection &&other)
73 4 : : nCurveCount(other.nCurveCount), papoCurves(other.papoCurves)
74 : {
75 4 : other.nCurveCount = 0;
76 4 : other.papoCurves = nullptr;
77 4 : }
78 :
79 : /************************************************************************/
80 : /* ~OGRCurveCollection() */
81 : /************************************************************************/
82 :
83 6117890 : OGRCurveCollection::~OGRCurveCollection()
84 :
85 : {
86 3058950 : empty(nullptr);
87 3058940 : }
88 :
89 : /************************************************************************/
90 : /* operator=( const OGRCurveCollection& ) */
91 : /************************************************************************/
92 :
93 : /**
94 : * \brief Assignment operator.
95 : *
96 : * Note: before GDAL 2.1, only the default implementation of the operator
97 : * existed, which could be unsafe to use.
98 : *
99 : * @since GDAL 2.1
100 : */
101 :
102 : OGRCurveCollection &
103 16 : OGRCurveCollection::operator=(const OGRCurveCollection &other)
104 : {
105 16 : if (this != &other)
106 : {
107 16 : empty(nullptr);
108 :
109 16 : if (other.nCurveCount > 0)
110 : {
111 8 : nCurveCount = other.nCurveCount;
112 8 : papoCurves = static_cast<OGRCurve **>(
113 8 : VSI_MALLOC2_VERBOSE(sizeof(void *), nCurveCount));
114 :
115 8 : if (papoCurves)
116 : {
117 22 : for (int i = 0; i < nCurveCount; i++)
118 : {
119 14 : papoCurves[i] = other.papoCurves[i]->clone();
120 : }
121 : }
122 : }
123 : }
124 16 : return *this;
125 : }
126 :
127 : /************************************************************************/
128 : /* operator=( OGRCurveCollection&& ) */
129 : /************************************************************************/
130 :
131 : /**
132 : * \brief Move assignment operator.
133 : *
134 : * @since GDAL 3.11
135 : */
136 :
137 8 : OGRCurveCollection &OGRCurveCollection::operator=(OGRCurveCollection &&other)
138 : {
139 8 : if (this != &other)
140 : {
141 8 : empty(nullptr);
142 8 : std::swap(nCurveCount, other.nCurveCount);
143 8 : std::swap(papoCurves, other.papoCurves);
144 : }
145 8 : return *this;
146 : }
147 :
148 : /************************************************************************/
149 : /* WkbSize() */
150 : /************************************************************************/
151 :
152 1690 : size_t OGRCurveCollection::WkbSize() const
153 : {
154 1690 : size_t nSize = 9;
155 :
156 3596 : for (auto &&poSubGeom : *this)
157 : {
158 1906 : nSize += poSubGeom->WkbSize();
159 : }
160 :
161 1690 : return nSize;
162 : }
163 :
164 : /************************************************************************/
165 : /* addCurveDirectly() */
166 : /************************************************************************/
167 :
168 1451730 : OGRErr OGRCurveCollection::addCurveDirectly(OGRGeometry *poGeom,
169 : OGRCurve *poCurve, int bNeedRealloc)
170 : {
171 1451730 : poGeom->HomogenizeDimensionalityWith(poCurve);
172 :
173 1451730 : if (bNeedRealloc)
174 : {
175 : #if SIZEOF_VOIDP < 8
176 : if (nCurveCount == std::numeric_limits<int>::max() /
177 : static_cast<int>(sizeof(OGRCurve *)))
178 : {
179 : CPLError(CE_Failure, CPLE_OutOfMemory, "Too many subgeometries");
180 : return OGRERR_FAILURE;
181 : }
182 : #else
183 1449980 : if (nCurveCount == std::numeric_limits<int>::max())
184 : {
185 0 : CPLError(CE_Failure, CPLE_AppDefined, "Too many subgeometries");
186 0 : return OGRERR_FAILURE;
187 : }
188 : #endif
189 :
190 1449980 : OGRCurve **papoNewCurves = static_cast<OGRCurve **>(VSI_REALLOC_VERBOSE(
191 : papoCurves, sizeof(OGRCurve *) * (nCurveCount + 1)));
192 1449980 : if (papoNewCurves == nullptr)
193 0 : return OGRERR_FAILURE;
194 1449980 : papoCurves = papoNewCurves;
195 : }
196 :
197 1451730 : papoCurves[nCurveCount] = poCurve;
198 :
199 1451730 : nCurveCount++;
200 :
201 1451730 : return OGRERR_NONE;
202 : }
203 :
204 : /************************************************************************/
205 : /* importPreambleFromWkb() */
206 : /************************************************************************/
207 :
208 43415 : OGRErr OGRCurveCollection::importPreambleFromWkb(
209 : OGRGeometry *poGeom, const unsigned char *pabyData, size_t &nSize,
210 : size_t &nDataOffset, OGRwkbByteOrder &eByteOrder, size_t nMinSubGeomSize,
211 : OGRwkbVariant eWkbVariant)
212 : {
213 43415 : int nCurveCountNew = 0;
214 :
215 43415 : OGRErr eErr = poGeom->importPreambleOfCollectionFromWkb(
216 : pabyData, nSize, nDataOffset, eByteOrder, nMinSubGeomSize,
217 : nCurveCountNew, eWkbVariant);
218 43415 : if (eErr != OGRERR_NONE)
219 391 : return eErr;
220 :
221 43024 : CPLAssert(nCurveCount == 0);
222 43024 : nCurveCount = nCurveCountNew;
223 :
224 : // coverity[tainted_data]
225 43024 : papoCurves = static_cast<OGRCurve **>(
226 43024 : VSI_CALLOC_VERBOSE(sizeof(void *), nCurveCount));
227 43024 : if (nCurveCount != 0 && papoCurves == nullptr)
228 : {
229 0 : nCurveCount = 0;
230 0 : return OGRERR_NOT_ENOUGH_MEMORY;
231 : }
232 :
233 43024 : return OGRERR_NONE;
234 : }
235 :
236 : /************************************************************************/
237 : /* importBodyFromWkb() */
238 : /************************************************************************/
239 :
240 1924 : OGRErr OGRCurveCollection::importBodyFromWkb(
241 : OGRGeometry *poGeom, const unsigned char *pabyData, size_t nSize,
242 : bool bAcceptCompoundCurve,
243 : OGRErr (*pfnAddCurveDirectlyFromWkb)(OGRGeometry *poGeom,
244 : OGRCurve *poCurve),
245 : OGRwkbVariant eWkbVariant, size_t &nBytesConsumedOut)
246 : {
247 1924 : nBytesConsumedOut = 0;
248 : /* -------------------------------------------------------------------- */
249 : /* Get the Geoms. */
250 : /* -------------------------------------------------------------------- */
251 1924 : const int nIter = nCurveCount;
252 1924 : nCurveCount = 0;
253 1924 : size_t nDataOffset = 0;
254 3669 : for (int iGeom = 0; iGeom < nIter; iGeom++)
255 : {
256 1925 : OGRGeometry *poSubGeom = nullptr;
257 :
258 : // Parses sub-geometry.
259 1925 : const unsigned char *pabySubData = pabyData + nDataOffset;
260 1925 : if (nSize < 9 && nSize != static_cast<size_t>(-1))
261 180 : return OGRERR_NOT_ENOUGH_DATA;
262 :
263 1923 : OGRwkbGeometryType eFlattenSubGeomType = wkbUnknown;
264 1923 : if (OGRReadWKBGeometryType(pabySubData, eWkbVariant,
265 1923 : &eFlattenSubGeomType) != OGRERR_NONE)
266 63 : return OGRERR_FAILURE;
267 1860 : eFlattenSubGeomType = wkbFlatten(eFlattenSubGeomType);
268 :
269 1860 : OGRErr eErr = OGRERR_NONE;
270 1860 : size_t nSubGeomBytesConsumedOut = 0;
271 5576 : if ((eFlattenSubGeomType != wkbCompoundCurve &&
272 1863 : OGR_GT_IsCurve(eFlattenSubGeomType)) ||
273 3 : (bAcceptCompoundCurve && eFlattenSubGeomType == wkbCompoundCurve))
274 : {
275 1858 : eErr = OGRGeometryFactory::createFromWkb(
276 : pabySubData, nullptr, &poSubGeom, nSize, eWkbVariant,
277 : nSubGeomBytesConsumedOut);
278 : }
279 : else
280 : {
281 2 : CPLDebug(
282 : "OGR",
283 : "Cannot add geometry of type (%d) to geometry of type (%d)",
284 2 : eFlattenSubGeomType, poGeom->getGeometryType());
285 2 : return OGRERR_UNSUPPORTED_GEOMETRY_TYPE;
286 : }
287 :
288 1858 : if (eErr == OGRERR_NONE)
289 : {
290 1747 : CPLAssert(nSubGeomBytesConsumedOut > 0);
291 1747 : if (nSize != static_cast<size_t>(-1))
292 : {
293 1747 : CPLAssert(nSize >= nSubGeomBytesConsumedOut);
294 1747 : nSize -= nSubGeomBytesConsumedOut;
295 : }
296 :
297 1747 : nDataOffset += nSubGeomBytesConsumedOut;
298 :
299 1747 : OGRCurve *poCurve = poSubGeom->toCurve();
300 1747 : eErr = pfnAddCurveDirectlyFromWkb(poGeom, poCurve);
301 : }
302 1858 : if (eErr != OGRERR_NONE)
303 : {
304 113 : delete poSubGeom;
305 113 : return eErr;
306 : }
307 : }
308 1744 : nBytesConsumedOut = nDataOffset;
309 :
310 1744 : return OGRERR_NONE;
311 : }
312 :
313 : /************************************************************************/
314 : /* exportToWkt() */
315 : /************************************************************************/
316 :
317 220 : std::string OGRCurveCollection::exportToWkt(const OGRGeometry *baseGeom,
318 : const OGRWktOptions &opts,
319 : OGRErr *err) const
320 : {
321 : try
322 : {
323 220 : bool first = true;
324 440 : std::string wkt(baseGeom->getGeometryName());
325 :
326 220 : OGRWktOptions optsModified(opts);
327 220 : optsModified.variant = wkbVariantIso;
328 220 : wkt += baseGeom->wktTypeString(optsModified.variant);
329 :
330 483 : for (int i = 0; i < nCurveCount; ++i)
331 : {
332 263 : OGRGeometry *geom = papoCurves[i];
333 :
334 263 : OGRErr subgeomErr = OGRERR_NONE;
335 263 : std::string tempWkt = geom->exportToWkt(optsModified, &subgeomErr);
336 263 : if (subgeomErr != OGRERR_NONE)
337 : {
338 0 : if (err)
339 0 : *err = subgeomErr;
340 0 : return std::string();
341 : }
342 :
343 : // A curve collection has a list of linestrings (OGRCompoundCurve),
344 : // which should have their leader removed, or a OGRCurvePolygon,
345 : // which has leaders for each of its sub-geometries that aren't
346 : // linestrings.
347 263 : if (tempWkt.compare(0, strlen("LINESTRING"), "LINESTRING") == 0)
348 : {
349 126 : auto pos = tempWkt.find('(');
350 126 : if (pos != std::string::npos)
351 126 : tempWkt = tempWkt.substr(pos);
352 : }
353 :
354 263 : if (tempWkt.find("EMPTY") != std::string::npos)
355 0 : continue;
356 :
357 263 : if (first)
358 189 : wkt += '(';
359 : else
360 74 : wkt += ',';
361 263 : first = false;
362 263 : wkt += tempWkt;
363 : }
364 :
365 220 : if (err)
366 219 : *err = OGRERR_NONE;
367 220 : if (first)
368 31 : wkt += "EMPTY";
369 : else
370 189 : wkt += ')';
371 220 : return wkt;
372 : }
373 0 : catch (const std::bad_alloc &e)
374 : {
375 0 : CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
376 0 : if (err)
377 0 : *err = OGRERR_FAILURE;
378 0 : return std::string();
379 : }
380 : }
381 :
382 : /************************************************************************/
383 : /* exportToWkb() */
384 : /************************************************************************/
385 :
386 : OGRErr
387 913 : OGRCurveCollection::exportToWkb(const OGRGeometry *poGeom,
388 : unsigned char *pabyData,
389 : const OGRwkbExportOptions *psOptions) const
390 : {
391 913 : if (psOptions == nullptr)
392 : {
393 : static const OGRwkbExportOptions defaultOptions;
394 0 : psOptions = &defaultOptions;
395 : }
396 :
397 : /* -------------------------------------------------------------------- */
398 : /* Set the byte order. */
399 : /* -------------------------------------------------------------------- */
400 913 : pabyData[0] = DB2_V72_UNFIX_BYTE_ORDER(
401 : static_cast<unsigned char>(psOptions->eByteOrder));
402 :
403 : /* -------------------------------------------------------------------- */
404 : /* Set the geometry feature type, ensuring that 3D flag is */
405 : /* preserved. */
406 : /* -------------------------------------------------------------------- */
407 913 : GUInt32 nGType = poGeom->getIsoGeometryType();
408 913 : if (psOptions->eWkbVariant == wkbVariantPostGIS1)
409 : {
410 0 : const bool bIs3D = wkbHasZ(static_cast<OGRwkbGeometryType>(nGType));
411 0 : nGType = wkbFlatten(nGType);
412 0 : if (nGType == wkbCurvePolygon)
413 0 : nGType = POSTGIS15_CURVEPOLYGON;
414 0 : if (bIs3D)
415 : // Explicitly set wkb25DBit.
416 0 : nGType =
417 0 : static_cast<OGRwkbGeometryType>(nGType | wkb25DBitInternalUse);
418 : }
419 :
420 913 : if (OGR_SWAP(psOptions->eByteOrder))
421 : {
422 0 : nGType = CPL_SWAP32(nGType);
423 : }
424 :
425 913 : memcpy(pabyData + 1, &nGType, 4);
426 :
427 : /* -------------------------------------------------------------------- */
428 : /* Copy in the raw data. */
429 : /* -------------------------------------------------------------------- */
430 913 : if (OGR_SWAP(psOptions->eByteOrder))
431 : {
432 0 : const int nCount = CPL_SWAP32(nCurveCount);
433 0 : memcpy(pabyData + 5, &nCount, 4);
434 : }
435 : else
436 : {
437 913 : memcpy(pabyData + 5, &nCurveCount, 4);
438 : }
439 :
440 : // TODO(schwehr): Where do these 9 values come from?
441 913 : size_t nOffset = 9;
442 :
443 : /* ==================================================================== */
444 : /* Serialize each of the Geoms. */
445 : /* ==================================================================== */
446 1825 : for (auto &&poSubGeom : *this)
447 : {
448 912 : poSubGeom->exportToWkb(pabyData + nOffset, psOptions);
449 :
450 912 : nOffset += poSubGeom->WkbSize();
451 : }
452 :
453 913 : return OGRERR_NONE;
454 : }
455 :
456 : /************************************************************************/
457 : /* empty() */
458 : /************************************************************************/
459 :
460 3121060 : void OGRCurveCollection::empty(OGRGeometry *poGeom)
461 : {
462 3121060 : if (papoCurves != nullptr)
463 : {
464 6115230 : for (auto &&poSubGeom : *this)
465 : {
466 3059740 : delete poSubGeom;
467 : }
468 3055490 : CPLFree(papoCurves);
469 : }
470 :
471 3121060 : nCurveCount = 0;
472 3121060 : papoCurves = nullptr;
473 3121060 : if (poGeom)
474 62065 : poGeom->setCoordinateDimension(2);
475 3121060 : }
476 :
477 : /************************************************************************/
478 : /* getEnvelope() */
479 : /************************************************************************/
480 :
481 251107 : void OGRCurveCollection::getEnvelope(OGREnvelope *psEnvelope) const
482 : {
483 251107 : OGREnvelope3D oEnv3D;
484 251107 : getEnvelope(&oEnv3D);
485 251107 : psEnvelope->MinX = oEnv3D.MinX;
486 251107 : psEnvelope->MinY = oEnv3D.MinY;
487 251107 : psEnvelope->MaxX = oEnv3D.MaxX;
488 251107 : psEnvelope->MaxY = oEnv3D.MaxY;
489 251107 : }
490 :
491 : /************************************************************************/
492 : /* getEnvelope() */
493 : /************************************************************************/
494 :
495 632388 : void OGRCurveCollection::getEnvelope(OGREnvelope3D *psEnvelope) const
496 : {
497 632388 : OGREnvelope3D oGeomEnv;
498 632388 : bool bExtentSet = false;
499 :
500 632388 : *psEnvelope = OGREnvelope3D();
501 1266160 : for (int iGeom = 0; iGeom < nCurveCount; iGeom++)
502 : {
503 633772 : if (!papoCurves[iGeom]->IsEmpty())
504 : {
505 633772 : bExtentSet = true;
506 633772 : papoCurves[iGeom]->getEnvelope(&oGeomEnv);
507 633772 : psEnvelope->Merge(oGeomEnv);
508 : }
509 : }
510 :
511 632388 : if (!bExtentSet)
512 : {
513 : // To be backward compatible when called on empty geom
514 8 : psEnvelope->MinX = 0.0;
515 8 : psEnvelope->MinY = 0.0;
516 8 : psEnvelope->MinZ = 0.0;
517 8 : psEnvelope->MaxX = 0.0;
518 8 : psEnvelope->MaxY = 0.0;
519 8 : psEnvelope->MaxZ = 0.0;
520 : }
521 632388 : }
522 :
523 : /************************************************************************/
524 : /* IsEmpty() */
525 : /************************************************************************/
526 :
527 562550 : OGRBoolean OGRCurveCollection::IsEmpty() const
528 : {
529 562575 : for (auto &&poSubGeom : *this)
530 : {
531 562018 : if (!poSubGeom->IsEmpty())
532 561993 : return FALSE;
533 : }
534 557 : return TRUE;
535 : }
536 :
537 : /************************************************************************/
538 : /* Equals() */
539 : /************************************************************************/
540 :
541 43427 : OGRBoolean OGRCurveCollection::Equals(const OGRCurveCollection *poOCC) const
542 : {
543 43427 : if (getNumCurves() != poOCC->getNumCurves())
544 2 : return FALSE;
545 :
546 : // Should eventually test the SRS.
547 :
548 72995 : for (int iGeom = 0; iGeom < nCurveCount; iGeom++)
549 : {
550 43650 : if (!getCurve(iGeom)->Equals(poOCC->getCurve(iGeom)))
551 14080 : return FALSE;
552 : }
553 :
554 29345 : return TRUE;
555 : }
556 :
557 : /************************************************************************/
558 : /* setCoordinateDimension() */
559 : /************************************************************************/
560 :
561 62932 : bool OGRCurveCollection::setCoordinateDimension(OGRGeometry *poGeom,
562 : int nNewDimension)
563 : {
564 63829 : for (auto &&poSubGeom : *this)
565 : {
566 897 : if (!poSubGeom->setCoordinateDimension(nNewDimension))
567 0 : return false;
568 : }
569 :
570 62932 : return poGeom->OGRGeometry::setCoordinateDimension(nNewDimension);
571 : }
572 :
573 1429420 : bool OGRCurveCollection::set3D(OGRGeometry *poGeom, OGRBoolean bIs3D)
574 : {
575 1536860 : for (auto &&poSubGeom : *this)
576 : {
577 107447 : if (!poSubGeom->set3D(bIs3D))
578 0 : return false;
579 : }
580 :
581 1429420 : return poGeom->OGRGeometry::set3D(bIs3D);
582 : }
583 :
584 235242 : bool OGRCurveCollection::setMeasured(OGRGeometry *poGeom,
585 : OGRBoolean bIsMeasured)
586 : {
587 342674 : for (auto &&poSubGeom : *this)
588 : {
589 107432 : if (!poSubGeom->setMeasured(bIsMeasured))
590 0 : return false;
591 : }
592 :
593 235242 : return poGeom->OGRGeometry::setMeasured(bIsMeasured);
594 : }
595 :
596 : /************************************************************************/
597 : /* assignSpatialReference() */
598 : /************************************************************************/
599 :
600 1463430 : void OGRCurveCollection::assignSpatialReference(OGRGeometry *poGeom,
601 : const OGRSpatialReference *poSR)
602 : {
603 2928960 : for (auto &&poSubGeom : *this)
604 : {
605 1465520 : poSubGeom->assignSpatialReference(poSR);
606 : }
607 1463430 : poGeom->OGRGeometry::assignSpatialReference(poSR);
608 1463430 : }
609 :
610 : /************************************************************************/
611 : /* getNumCurves() */
612 : /************************************************************************/
613 :
614 86854 : int OGRCurveCollection::getNumCurves() const
615 : {
616 86854 : return nCurveCount;
617 : }
618 :
619 : /************************************************************************/
620 : /* getCurve() */
621 : /************************************************************************/
622 :
623 124892 : OGRCurve *OGRCurveCollection::getCurve(int i)
624 : {
625 124892 : if (i < 0 || i >= nCurveCount)
626 184 : return nullptr;
627 124708 : return papoCurves[i];
628 : }
629 :
630 : /************************************************************************/
631 : /* getCurve() */
632 : /************************************************************************/
633 :
634 211113 : const OGRCurve *OGRCurveCollection::getCurve(int i) const
635 : {
636 211113 : if (i < 0 || i >= nCurveCount)
637 3 : return nullptr;
638 211110 : return papoCurves[i];
639 : }
640 :
641 : /************************************************************************/
642 : /* stealCurve() */
643 : /************************************************************************/
644 :
645 5 : OGRCurve *OGRCurveCollection::stealCurve(int i)
646 : {
647 5 : if (i < 0 || i >= nCurveCount)
648 0 : return nullptr;
649 5 : OGRCurve *poRet = papoCurves[i];
650 5 : if (i < nCurveCount - 1)
651 : {
652 2 : memmove(papoCurves + i, papoCurves + i + 1,
653 2 : (nCurveCount - i - 1) * sizeof(OGRCurve *));
654 : }
655 5 : nCurveCount--;
656 5 : return poRet;
657 : }
658 :
659 : /************************************************************************/
660 : /* transform() */
661 : /************************************************************************/
662 :
663 519 : OGRErr OGRCurveCollection::transform(OGRGeometry *poGeom,
664 : OGRCoordinateTransformation *poCT)
665 : {
666 1137 : for (int iGeom = 0; iGeom < nCurveCount; iGeom++)
667 : {
668 618 : const OGRErr eErr = papoCurves[iGeom]->transform(poCT);
669 618 : if (eErr != OGRERR_NONE)
670 : {
671 0 : if (iGeom != 0)
672 : {
673 0 : CPLDebug("OGR", "OGRCurveCollection::transform() failed for a "
674 : "geometry other than the first, meaning some "
675 : "geometries are transformed and some are not!");
676 :
677 0 : return OGRERR_FAILURE;
678 : }
679 :
680 0 : return eErr;
681 : }
682 : }
683 :
684 519 : poGeom->assignSpatialReference(poCT->GetTargetCS());
685 :
686 519 : return OGRERR_NONE;
687 : }
688 :
689 : /************************************************************************/
690 : /* flattenTo2D() */
691 : /************************************************************************/
692 :
693 760 : void OGRCurveCollection::flattenTo2D(OGRGeometry *poGeom)
694 : {
695 1586 : for (auto &&poSubGeom : *this)
696 : {
697 826 : poSubGeom->flattenTo2D();
698 : }
699 :
700 760 : poGeom->setCoordinateDimension(2);
701 760 : }
702 :
703 : /************************************************************************/
704 : /* segmentize() */
705 : /************************************************************************/
706 :
707 37 : bool OGRCurveCollection::segmentize(double dfMaxLength)
708 : {
709 74 : for (auto &&poSubGeom : *this)
710 : {
711 37 : if (!poSubGeom->segmentize(dfMaxLength))
712 0 : return false;
713 : }
714 37 : return true;
715 : }
716 :
717 : /************************************************************************/
718 : /* swapXY() */
719 : /************************************************************************/
720 :
721 75 : void OGRCurveCollection::swapXY()
722 : {
723 150 : for (auto &&poSubGeom : *this)
724 : {
725 75 : poSubGeom->swapXY();
726 : }
727 75 : }
728 :
729 : /************************************************************************/
730 : /* hasCurveGeometry() */
731 : /************************************************************************/
732 :
733 94 : OGRBoolean OGRCurveCollection::hasCurveGeometry(int bLookForNonLinear) const
734 : {
735 133 : for (auto &&poSubGeom : *this)
736 : {
737 105 : if (poSubGeom->hasCurveGeometry(bLookForNonLinear))
738 66 : return TRUE;
739 : }
740 28 : return FALSE;
741 : }
742 :
743 : /************************************************************************/
744 : /* removeCurve() */
745 : /************************************************************************/
746 :
747 : /**
748 : * \brief Remove a geometry from the container.
749 : *
750 : * Removing a geometry will cause the geometry count to drop by one, and all
751 : * "higher" geometries will shuffle down one in index.
752 : *
753 : * @param iIndex the index of the geometry to delete. A value of -1 is a
754 : * special flag meaning that all geometries should be removed.
755 : *
756 : * @param bDelete if true the geometry will be deallocated, otherwise it will
757 : * not. The default is true as the container is considered to own the
758 : * geometries in it.
759 : *
760 : * @return OGRERR_NONE if successful, or OGRERR_FAILURE if the index is
761 : * out of range.
762 : */
763 :
764 26 : OGRErr OGRCurveCollection::removeCurve(int iIndex, bool bDelete)
765 :
766 : {
767 26 : if (iIndex < -1 || iIndex >= nCurveCount)
768 2 : return OGRERR_FAILURE;
769 :
770 : // Special case.
771 24 : if (iIndex == -1)
772 : {
773 3 : while (nCurveCount > 0)
774 2 : removeCurve(nCurveCount - 1, bDelete);
775 1 : return OGRERR_NONE;
776 : }
777 :
778 23 : if (bDelete)
779 19 : delete papoCurves[iIndex];
780 :
781 23 : memmove(papoCurves + iIndex, papoCurves + iIndex + 1,
782 23 : sizeof(void *) * (nCurveCount - iIndex - 1));
783 :
784 23 : nCurveCount--;
785 :
786 23 : return OGRERR_NONE;
787 : }
788 :
789 : /************************************************************************/
790 : /* hasEmptyParts() */
791 : /************************************************************************/
792 :
793 : /**
794 : * \brief Returns whether a geometry has empty parts/rings.
795 : *
796 : * Returns true if removeEmptyParts() will modify the geometry.
797 : *
798 : * This is different from IsEmpty().
799 : *
800 : * @since GDAL 3.10
801 : */
802 16 : bool OGRCurveCollection::hasEmptyParts() const
803 : {
804 32 : for (int i = 0; i < nCurveCount; ++i)
805 : {
806 20 : if (papoCurves[i]->IsEmpty() || papoCurves[i]->hasEmptyParts())
807 4 : return true;
808 : }
809 12 : return false;
810 : }
811 :
812 : /************************************************************************/
813 : /* removeEmptyParts() */
814 : /************************************************************************/
815 :
816 : /**
817 : * \brief Remove empty parts/rings from this geometry.
818 : *
819 : * @since GDAL 3.10
820 : */
821 9 : void OGRCurveCollection::removeEmptyParts()
822 : {
823 21 : for (int i = nCurveCount - 1; i >= 0; --i)
824 : {
825 12 : papoCurves[i]->removeEmptyParts();
826 12 : if (papoCurves[i]->IsEmpty())
827 4 : removeCurve(i, true);
828 : }
829 9 : }
830 :
831 : /************************************************************************/
832 : /* reversePoints() */
833 : /************************************************************************/
834 :
835 : /**
836 : * \brief Reverse point order.
837 : *
838 : * This method updates the points in this curve in place
839 : * reversing the point ordering (first for last, etc) and component ordering.
840 : *
841 : * @since 3.10
842 : */
843 1 : void OGRCurveCollection::reversePoints()
844 :
845 : {
846 2 : for (int i = 0; i < nCurveCount / 2; ++i)
847 : {
848 1 : std::swap(papoCurves[i], papoCurves[nCurveCount - 1 - i]);
849 : }
850 3 : for (int i = 0; i < nCurveCount; ++i)
851 : {
852 2 : papoCurves[i]->reversePoints();
853 : }
854 1 : }
855 :
856 : //! @endcond
|