21#include <nlohmann/json.hpp>
33#include <QPainterPath>
36using namespace Qt::StringLiterals;
50 auto result = std::make_unique< QgsCompoundCurve >();
52 return result.release();
63 while ( i < mCurves.size() && j < otherCurve->mCurves.size() )
67 const int comparison = aGeom->
compareTo( bGeom );
68 if ( comparison != 0 )
75 if ( i < mCurves.size() )
79 if ( j < otherCurve->mCurves.size() )
88 return u
"CompoundCurve"_s;
100 mCurves.reserve( curve.mCurves.size() );
101 for (
const QgsCurve *
c : curve.mCurves )
103 mCurves.append( c->clone() );
110 if ( &curve !=
this )
113 for (
const QgsCurve *
c : curve.mCurves )
115 mCurves.append(
c->clone() );
129 qDeleteAll( mCurves );
136 if ( mCurves.empty() )
141 QgsBox3D bbox = mCurves.at( 0 )->boundingBox3D();
142 for (
int i = 1; i < mCurves.size(); ++i )
144 QgsBox3D curveBox = mCurves.at( i )->boundingBox3D();
153 if ( index < 1 || index >= size - 1 )
162 mCurves = std::move( curve2->mCurves );
167 mCurves.append( curve1->mCurves );
168 curve1->mCurves.clear();
190 for (
int i = 0; i <
nCurves; ++i )
193 wkbPtr -= 1 +
sizeof( int );
206 currentCurve->
fromWkb( wkbPtr );
207 mCurves.append( currentCurve );
222 QString secondWithoutParentheses =
parts.second;
223 secondWithoutParentheses = secondWithoutParentheses.remove(
'(' ).remove(
')' ).simplified().remove(
' ' );
224 if ( (
parts.second.compare(
"EMPTY"_L1, Qt::CaseInsensitive ) == 0 ) || secondWithoutParentheses.isEmpty() )
227 QString defaultChildWkbType = u
"LineString%1%2"_s.arg(
is3D() ? u
"Z"_s : QString(),
isMeasure() ? u
"M"_s : QString() );
230 for (
const QString &childWkt : blocks )
243 if ( !mCurves.back()->fromWkt( childWkt ) )
254 for (
const QgsCurve *curve : std::as_const( mCurves ) )
256 hasZ = hasZ || curve->is3D();
257 hasM = hasM || curve->isMeasure();
271 int binarySize =
sizeof( char ) +
sizeof( quint32 ) +
sizeof( quint32 );
272 for (
const QgsCurve *curve : mCurves )
274 binarySize += curve->wkbSize( flags );
285 wkb << static_cast<quint32>(
wkbType() );
286 wkb << static_cast<quint32>( mCurves.size() );
287 for (
const QgsCurve *curve : mCurves )
289 wkb << curve->asWkb( flags );
302 for (
const QgsCurve *curve : mCurves )
304 QString childWkt = curve->asWkt( precision );
308 childWkt = childWkt.mid( childWkt.indexOf(
'(' ) );
310 wkt += childWkt +
',';
312 if ( wkt.endsWith(
',' ) )
324 std::unique_ptr< QgsLineString > line(
curveToLine() );
325 QDomElement gml = line->asGml2( doc, precision, ns, axisOrder );
331 QDomElement compoundCurveElem = doc.createElementNS( ns, u
"CompositeCurve"_s );
334 return compoundCurveElem;
336 for (
const QgsCurve *curve : mCurves )
338 QDomElement curveMemberElem = doc.createElementNS( ns, u
"curveMember"_s );
339 QDomElement curveElem = curve->asGml3( doc, precision, ns, axisOrder );
340 curveMemberElem.appendChild( curveElem );
341 compoundCurveElem.appendChild( curveMemberElem );
344 return compoundCurveElem;
350 std::unique_ptr< QgsLineString > line(
curveToLine() );
351 return line->asJsonObject( precision );
357 for (
const QgsCurve *curve : mCurves )
359 length += curve->length();
366 if ( mCurves.empty() )
370 return mCurves.at( 0 )->startPoint();
375 if ( mCurves.empty() )
379 return mCurves.at( mCurves.size() - 1 )->endPoint();
385 if ( mCurves.empty() )
390 mCurves[0]->points( pts );
391 for (
int i = 1; i < mCurves.size(); ++i )
394 mCurves[i]->points( pList );
409 for (
int i = 0; i <
nCurves; ++i )
411 nPoints += mCurves.at( i )->numPoints() - 1;
419 if ( mCurves.isEmpty() )
424 if ( !curve->isEmpty() )
432 if ( mCurves.isEmpty() )
435 for (
int i = 0; i < mCurves.size(); ++i )
437 if ( !mCurves[i]->
isValid( error, flags ) )
439 error = QObject::tr(
"Curve[%1]: %2" ).arg( i + 1 ).arg( error );
449 for (
const QgsCurve *curve : mCurves )
451 const int curveIndex = curve->indexOf( point );
452 if ( curveIndex >= 0 )
453 return curveStart + curveIndex;
456 curveStart += curve->numPoints() - 1;
464 std::unique_ptr< QgsLineString > currentLine;
465 for (
const QgsCurve *curve : mCurves )
467 currentLine.reset( curve->curveToLine( tolerance, toleranceType ) );
468 line->
append( currentLine.get() );
479 std::unique_ptr<QgsCurve> gridified(
static_cast< QgsCurve *
>( curve->snappedToGrid( hSpacing, vSpacing, dSpacing, mSpacing, removeRedundantPoints ) ) );
482 result->mCurves.append( gridified.release() );
486 if ( result->mCurves.empty() )
489 return result.release();
494 std::unique_ptr< QgsLineString > line(
curveToLine() );
495 return line->simplifyByDistance( tolerance );
501 const QVector< QgsCurve * > curves = mCurves;
506 result = curve->removeDuplicateNodes( epsilon, useZValues ) || result;
507 if ( curve->numPoints() == 0 ||
qgsDoubleNear( curve->length(), 0.0, epsilon ) )
510 delete mCurves.takeAt( i );
518 curve->moveVertex(
QgsVertexId( -1, -1, 0 ), lastEnd );
529 if ( mCurves.empty() )
543 for (
const QgsCurve *curve : mCurves )
545 if ( curve->boundingBoxIntersects( box3d ) )
558 if ( mCurves.size() == 1 )
559 return mCurves.at( 0 );
566 if ( i < 0 || i >= mCurves.size() )
570 return mCurves.at( i );
578 if ( mCurves.empty() )
602 const bool canExtendPrevious = extendPrevious && previousLineString && newLineString;
603 if ( canExtendPrevious )
605 previousLineString->
append( newLineString );
620 if ( i < 0 || i >= mCurves.size() )
625 delete mCurves.takeAt( i );
638 if ( !mCurves.isEmpty() )
640 lastCurve = mCurves.at( mCurves.size() - 1 );
647 mCurves.append( line );
665 QVector< QgsCurve * > newCurves;
666 newCurves.reserve( mCurves.size() );
667 for (
QgsCurve *curve : std::as_const( mCurves ) )
669 if ( lastCurve && lastCurve->
wkbType() == curve->wkbType() )
693 for (
const QgsCurve *curve : mCurves )
701 for (
QgsCurve *curve : std::as_const( mCurves ) )
703 curve->transform( ct, d, transformZ );
710 for (
QgsCurve *curve : std::as_const( mCurves ) )
712 curve->transform( t, zTranslate, zScale, mTranslate, mScale );
721 for (
const QgsCurve *curve : mCurves )
723 if ( curve != mCurves.at( 0 ) && pp.currentPosition() != curve->startPoint().toQPointF() )
725 pp.lineTo( curve->startPoint().toQPointF() );
727 curve->addToPainterPath( pp );
735 for (
const QgsCurve *curve : mCurves )
737 if ( curve != mCurves.at( 0 ) && pp.currentPosition() != curve->startPoint().toQPointF() )
739 pp.lineTo( curve->startPoint().toQPointF() );
741 curve->addToPainterPath( pp );
748 QVector< QPair<int, QgsVertexId> > curveIds = curveVertexId( position );
749 if ( curveIds.empty() )
753 int curveId = curveIds.at( 0 ).first;
754 if ( curveId >= mCurves.size() )
759 bool success = mCurves.at( curveId )->insertVertex( curveIds.at( 0 ).second, vertex );
769 QVector< QPair<int, QgsVertexId> > curveIds = curveVertexId( position );
770 QVector< QPair<int, QgsVertexId> >::const_iterator idIt = curveIds.constBegin();
771 for ( ; idIt != curveIds.constEnd(); ++idIt )
773 mCurves.at( idIt->first )->moveVertex( idIt->second, newPos );
776 bool success = !curveIds.isEmpty();
786 const QVector< QPair<int, QgsVertexId> > curveIds = curveVertexId( position );
787 if ( curveIds.isEmpty() )
790 const int curveId = curveIds.at( 0 ).first;
791 QgsCurve *curve = mCurves.at( curveId );
792 const QgsVertexId subVertexId = curveIds.at( 0 ).second;
795 if ( curveIds.size() == 1 )
800 if ( circularString && subVertexId.
vertex % 2 == 1 )
810 auto curveC = std::make_unique<QgsCircularString>();
811 curveC->setPoints(
points.mid( subVertexId.
vertex + 1 ) );
812 mCurves.insert( curveId, curveC.release() );
816 auto curveB = std::make_unique<QgsLineString>();
817 curveB->setPoints( partB );
818 mCurves.insert( curveId, curveB.release() );
819 curve = mCurves.at( curveId );
821 if ( subVertexId.
vertex > 1 )
823 auto curveA = std::make_unique<QgsCircularString>();
824 curveA->setPoints(
points.mid( 0, subVertexId.
vertex ) );
825 mCurves.insert( curveId, curveA.release() );
840 else if ( curveIds.size() == 2 )
842 const int nextCurveId = curveIds.at( 1 ).first;
843 QgsCurve *nextCurve = mCurves.at( nextCurveId );
844 const QgsVertexId nextSubVertexId = curveIds.at( 1 ).second;
846 Q_ASSERT( nextCurveId == curveId + 1 );
848 Q_ASSERT( nextSubVertexId.
vertex == 0 );
878 mCurves.insert( curveId, line );
890 mCurves.insert( nextCurveId, line );
902 mCurves.insert( curveId, line );
910 if ( endPointOfFirst != startPointOfSecond )
915 mCurves.insert( nextCurveId, line );
921 bool success = !curveIds.isEmpty();
927QVector< QPair<int, QgsVertexId> > QgsCompoundCurve::curveVertexId(
QgsVertexId id )
const
929 QVector< QPair<int, QgsVertexId> > curveIds;
931 int currentVertexIndex = 0;
932 for (
int i = 0; i < mCurves.size(); ++i )
934 int increment = mCurves.at( i )->numPoints() - 1;
935 if (
id.vertex >= currentVertexIndex &&
id.vertex <= currentVertexIndex + increment )
937 int curveVertexId =
id.vertex - currentVertexIndex;
941 vid.
vertex = curveVertexId;
942 curveIds.append( qMakePair( i, vid ) );
943 if ( curveVertexId == increment && i < ( mCurves.size() - 1 ) )
946 curveIds.append( qMakePair( i + 1, vid ) );
950 else if (
id.vertex >= currentVertexIndex &&
id.vertex == currentVertexIndex + increment + 1 && i == ( mCurves.size() - 1 ) )
952 int curveVertexId =
id.vertex - currentVertexIndex;
956 vid.
vertex = curveVertexId;
957 curveIds.append( qMakePair( i, vid ) );
960 currentVertexIndex += increment;
981 QVector< QPair<int, QgsVertexId> > curveIds = curveVertexId( position );
984 if ( curveIds.length() != 1 )
987 int curveId = curveIds[0].first;
1000 if ( subVertexId.
vertex % 2 == 0 )
1004 circularString->points(
points );
1010 auto curveA = std::make_unique<QgsCircularString>();
1011 curveA->setPoints( partA );
1012 auto curveB = std::make_unique<QgsLineString>();
1013 curveB->setPoints( partB );
1014 auto curveC = std::make_unique<QgsCircularString>();
1015 curveC->setPoints( partC );
1019 mCurves.insert( curveId, curveC.release() );
1020 mCurves.insert( curveId, curveB.release() );
1021 if ( subVertexId.
vertex > 1 )
1022 mCurves.insert( curveId, curveA.release() );
1029 lineString->points(
points );
1035 auto curveA = std::make_unique<QgsLineString>();
1036 curveA->setPoints( partA );
1037 auto curveB = std::make_unique<QgsCircularString>();
1038 curveB->setPoints( partB );
1039 auto curveC = std::make_unique<QgsLineString>();
1040 curveC->setPoints( partC );
1044 mCurves.insert( curveId, curveC.release() );
1045 mCurves.insert( curveId, curveB.release() );
1046 if ( subVertexId.
vertex > 1 )
1047 mCurves.insert( curveId, curveA.release() );
1065 int currentVertexId = 0;
1066 for (
int j = 0; j < mCurves.size(); ++j )
1068 int nCurvePoints = mCurves.at( j )->numPoints();
1069 if ( ( node - currentVertexId ) < nCurvePoints )
1071 return ( mCurves.at( j )->pointAt( node - currentVertexId, point, type ) );
1073 currentVertexId += ( nCurvePoints - 1 );
1080 int currentVertexId = 0;
1081 for (
int j = 0; j < mCurves.size(); ++j )
1083 int nCurvePoints = mCurves.at( j )->numPoints();
1084 if ( ( index - currentVertexId ) < nCurvePoints )
1086 return mCurves.at( j )->xAt( index - currentVertexId );
1088 currentVertexId += ( nCurvePoints - 1 );
1095 int currentVertexId = 0;
1096 for (
int j = 0; j < mCurves.size(); ++j )
1098 int nCurvePoints = mCurves.at( j )->numPoints();
1099 if ( ( index - currentVertexId ) < nCurvePoints )
1101 return mCurves.at( j )->yAt( index - currentVertexId );
1103 currentVertexId += ( nCurvePoints - 1 );
1110 int currentVertexId = 0;
1111 for (
int j = 0; j < mCurves.size(); ++j )
1113 int nCurvePoints = mCurves.at( j )->numPoints();
1114 if ( ( index - currentVertexId ) < nCurvePoints )
1116 return mCurves.at( j )->zAt( index - currentVertexId );
1118 currentVertexId += ( nCurvePoints - 1 );
1125 int currentVertexId = 0;
1126 for (
int j = 0; j < mCurves.size(); ++j )
1128 int nCurvePoints = mCurves.at( j )->numPoints();
1129 if ( ( index - currentVertexId ) < nCurvePoints )
1131 return mCurves.at( j )->mAt( index - currentVertexId );
1133 currentVertexId += ( nCurvePoints - 1 );
1141 for (
QgsCurve *curve : std::as_const( mCurves ) )
1143 if ( !curve->transform( transformer ) )
1161 for (
QgsCurve *curve : std::as_const( mCurves ) )
1163 curve->filterVertices( filter );
1170 for (
QgsCurve *curve : std::as_const( mCurves ) )
1179 if ( mCurves.empty() )
1180 return std::make_tuple( std::make_unique< QgsCompoundCurve >(), std::make_unique< QgsCompoundCurve >() );
1184 auto curve1 = std::make_unique< QgsCompoundCurve >();
1185 std::unique_ptr< QgsCompoundCurve > curve2;
1187 for (
const QgsCurve *curve : mCurves )
1189 const int curveSize = curve->numPoints();
1190 if ( !curve2 && index < curveStart + curveSize )
1193 auto [p1, p2] = curve->splitCurveAtVertex( index - curveStart );
1194 if ( !p1->isEmpty() )
1195 curve1->addCurve( p1.release() );
1197 curve2 = std::make_unique< QgsCompoundCurve >();
1198 if ( !p2->isEmpty() )
1199 curve2->addCurve( p2.release() );
1204 curve2->addCurve( curve->clone() );
1206 curve1->addCurve( curve->clone() );
1211 curveStart += curve->numPoints() - 1;
1214 return std::make_tuple( std::move( curve1 ), curve2 ? std::move( curve2 ) : std::make_unique< QgsCompoundCurve >() );
1226 for (
const QgsCurve *curve : mCurves )
1243 for (
const QgsCurve *curve : mCurves )
1262 for (
const QgsCurve *curve : mCurves )
1264 if ( curve->hasCurvedSegments() )
1274 QVector< QPair<int, QgsVertexId> > curveIds = curveVertexId( vertex );
1275 if ( curveIds.size() == 1 )
1277 QgsCurve *curve = mCurves[curveIds.at( 0 ).first];
1278 return curve->
vertexAngle( curveIds.at( 0 ).second );
1280 else if ( curveIds.size() > 1 )
1282 QgsCurve *curve1 = mCurves[curveIds.at( 0 ).first];
1283 QgsCurve *curve2 = mCurves[curveIds.at( 1 ).first];
1284 double angle1 = curve1->
vertexAngle( curveIds.at( 0 ).second );
1285 double angle2 = curve2->
vertexAngle( curveIds.at( 1 ).second );
1296 QVector< QPair<int, QgsVertexId> > curveIds = curveVertexId( startVertex );
1298 for (
auto it = curveIds.constBegin(); it != curveIds.constEnd(); ++it )
1300 length += mCurves.at( it->first )->segmentLength( it->second );
1308 for (
int i = mCurves.count() - 1; i >= 0; --i )
1310 QgsCurve *reversedCurve = mCurves.at( i )->reversed();
1311 clone->addCurve( reversedCurve );
1321 double distanceTraversed = 0;
1322 for (
const QgsCurve *curve : mCurves )
1324 const double thisCurveLength = curve->length();
1325 if ( distanceTraversed + thisCurveLength > distance ||
qgsDoubleNear( distanceTraversed + thisCurveLength, distance ) )
1328 const double distanceToPoint = std::min( distance - distanceTraversed, thisCurveLength );
1331 return curve->interpolatePoint( distanceToPoint );
1334 distanceTraversed += thisCurveLength;
1342 if ( startDistance < 0 && endDistance < 0 )
1345 endDistance = std::max( startDistance, endDistance );
1346 auto substring = std::make_unique< QgsCompoundCurve >();
1348 double distanceTraversed = 0;
1349 for (
const QgsCurve *curve : mCurves )
1351 const double thisCurveLength = curve->length();
1352 if ( distanceTraversed + thisCurveLength < startDistance )
1358 std::unique_ptr< QgsCurve > part( curve->curveSubstring( startDistance - distanceTraversed, endDistance - distanceTraversed ) );
1360 substring->addCurve( part.release() );
1363 distanceTraversed += thisCurveLength;
1364 if ( distanceTraversed > endDistance )
1368 return substring.release();
1378 for (
QgsCurve *curve : std::as_const( mCurves ) )
1380 curve->addZValue( zValue );
1393 for (
QgsCurve *curve : std::as_const( mCurves ) )
1395 curve->addMValue( mValue );
1407 for (
QgsCurve *curve : std::as_const( mCurves ) )
1409 curve->dropZValue();
1421 for (
QgsCurve *curve : std::as_const( mCurves ) )
1423 curve->dropMValue();
1431 for (
QgsCurve *curve : std::as_const( mCurves ) )
1447 if ( fromVertex.
part != 0 || fromVertex.
ring != 0 || toVertex.
part != 0 || toVertex.
ring != 0 )
1450 const int fromVertexNumber = fromVertex.
vertex;
1451 const int toVertexNumber = toVertex.
vertex;
1454 if ( fromVertexNumber < 0 || fromVertexNumber >= totalVertices || toVertexNumber < 0 || toVertexNumber >= totalVertices )
1457 if ( fromVertexNumber == toVertexNumber )
1460 double totalDistance = 0.0;
1463 int currentVertexId = 0;
1464 int fromCurve = -1, toCurve = -1;
1465 int fromCurveVertex = -1, toCurveVertex = -1;
1468 for (
int j = 0; j < mCurves.size(); ++j )
1470 int nCurvePoints = mCurves.at( j )->numPoints();
1473 if ( fromCurve == -1 && fromVertexNumber >= currentVertexId && fromVertexNumber < currentVertexId + nCurvePoints )
1476 fromCurveVertex = fromVertexNumber - currentVertexId;
1480 if ( toCurve == -1 && toVertexNumber >= currentVertexId && toVertexNumber < currentVertexId + nCurvePoints )
1483 toCurveVertex = toVertexNumber - currentVertexId;
1487 currentVertexId += ( nCurvePoints - 1 );
1490 if ( fromCurve == -1 || toCurve == -1 )
1493 if ( fromCurve == toCurve )
1498 return mCurves.at( fromCurve )->distanceBetweenVertices( fromId, toId );
1505 if ( fromCurveVertex < mCurves.at( fromCurve )->numPoints() - 1 )
1508 QgsVertexId endId( 0, 0, mCurves.at( fromCurve )->numPoints() - 1 );
1509 totalDistance += mCurves.at( fromCurve )->distanceBetweenVertices( fromId, endId );
1513 for (
int j = fromCurve + 1; j < toCurve; ++j )
1515 totalDistance += mCurves.at( j )->length();
1519 if ( toCurveVertex > 0 )
1523 totalDistance += mCurves.at( toCurve )->distanceBetweenVertices( startId, toId );
1527 return totalDistance;
QFlags< GeometryValidityFlag > GeometryValidityFlags
Geometry validity flags.
VertexType
Types of vertex.
WkbType
The WKB type describes the number of dimensions a geometry has.
@ CompoundCurve
CompoundCurve.
@ CircularString
CircularString.
TransformDirection
Indicates the direction (forward or inverse) of a transform.
virtual bool fromWkb(QgsConstWkbPtr &wkb)=0
Sets the geometry from a WKB string.
SegmentationToleranceType
Segmentation tolerance as maximum angle or maximum difference between approximation and circle.
virtual double vertexAngle(QgsVertexId vertex) const =0
Returns approximate angle at a vertex.
bool isMeasure() const
Returns true if the geometry contains m values.
QFlags< WkbFlag > WkbFlags
bool is3D() const
Returns true if the geometry is 3D and contains a z-value.
AxisOrder
Axis order for GML generation.
QString wktTypeStr() const
Returns the WKT type string of the geometry.
QgsAbstractGeometry & operator=(const QgsAbstractGeometry &geom)
Qgis::WkbType wkbType() const
Returns the WKB type of the geometry.
void setZMTypeFromSubGeometry(const QgsAbstractGeometry *subggeom, Qgis::WkbType baseGeomType)
Updates the geometry type based on whether sub geometries contain z or m values.
virtual bool boundingBoxIntersects(const QgsRectangle &rectangle) const
Returns true if the bounding box of this geometry intersects with a rectangle.
virtual bool deleteVertex(QgsVertexId position)=0
Deletes a vertex within the geometry.
virtual int compareTo(const QgsAbstractGeometry *other) const
Comparator for sorting of geometry.
QgsAbstractGeometry()=default
QgsGeometryConstPartIterator parts() const
Returns Java-style iterator for traversal of parts of the geometry.
static endian_t endian()
Returns whether this machine uses big or little endian.
A 3-dimensional box composed of x, y, z coordinates.
void combineWith(const QgsBox3D &box)
Expands the bbox so that it covers both the original rectangle and the given rectangle.
Circular string geometry type.
void points(QgsPointSequence &pts) const override
Returns a list of points within the curve.
bool fromWkt(const QString &wkt) override
Sets the geometry from a WKT string.
void draw(QPainter &p) const override
Draws the geometry using the specified QPainter.
void sumUpArea(double &sum) const override
Sums up the area of the curve by iterating over the vertices (shoelace formula).
QgsLineString * curveToLine(double tolerance=M_PI_2/90, SegmentationToleranceType toleranceType=MaximumAngle) const override
Returns a new line string geometry corresponding to a segmentized approximation of the curve.
bool insertVertex(QgsVertexId position, const QgsPoint &vertex) override
Inserts a vertex into the geometry.
bool removeDuplicateNodes(double epsilon=4 *std::numeric_limits< double >::epsilon(), bool useZValues=false) override
Removes duplicate nodes from the geometry, wherever removing the nodes does not result in a degenerat...
QgsAbstractGeometry * simplifyByDistance(double tolerance) const override
Simplifies the geometry by applying the Douglas Peucker simplification by distance algorithm.
int compareToSameClass(const QgsAbstractGeometry *other) const final
Compares to an other geometry of the same class, and returns a integer for sorting of the two geometr...
bool fromWkb(QgsConstWkbPtr &wkb) override
Sets the geometry from a WKB string.
QgsCompoundCurve * reversed() const override
Returns a reversed copy of the curve, where the direction of the curve has been flipped.
void condenseCurves()
Condenses the curves in this geometry by combining adjacent linestrings a to a single continuous line...
void close()
Appends first point if not already closed.
bool addMValue(double mValue=0) override
Adds a measure to the geometry, initialized to a preset value.
void drawAsPolygon(QPainter &p) const override
Draws the curve as a polygon on the specified QPainter.
int dimension() const override
Returns the inherent dimension of the geometry.
void filterVertices(const std::function< bool(const QgsPoint &) > &filter) override
Filters the vertices from the geometry in place, removing any which do not return true for the filter...
std::tuple< std::unique_ptr< QgsCurve >, std::unique_ptr< QgsCurve > > splitCurveAtVertex(int index) const final
Splits the curve at the specified vertex index, returning two curves which represent the portion of t...
bool boundingBoxIntersects(const QgsBox3D &box3d) const override
Returns true if the bounding box of this geometry intersects with a box3d.
json asJsonObject(int precision=17) const override
Returns a json object representation of the geometry.
QString geometryType() const override
Returns a unique string representing the geometry type.
double mAt(int index) const override
Returns the m-coordinate of the specified node in the line string.
double distanceBetweenVertices(QgsVertexId fromVertex, QgsVertexId toVertex) const override
Returns the distance along the curve between two vertices.
bool isEmpty() const override
Returns true if the geometry is empty.
double vertexAngle(QgsVertexId vertex) const override
Returns approximate angle at a vertex.
double segmentLength(QgsVertexId startVertex) const override
Returns the length of the segment of the geometry which begins at startVertex.
void transformVertices(const std::function< QgsPoint(const QgsPoint &) > &transform) override
Transforms the vertices from the geometry in place, applying the transform function to every vertex.
int nCurves() const
Returns the number of curves in the geometry.
bool toggleCircularAtVertex(QgsVertexId position)
Converts the vertex at the given position from/to circular.
QgsBox3D calculateBoundingBox3D() const override
Calculates the minimal 3D bounding box for the geometry.
QString asWkt(int precision=17) const override
Returns a WKT representation of the geometry.
const QgsAbstractGeometry * simplifiedTypeRef() const override
Returns a reference to the simplest lossless representation of this geometry, e.g.
int wkbSize(QgsAbstractGeometry::WkbFlags flags=QgsAbstractGeometry::WkbFlags()) const override
Returns the length of the QByteArray returned by asWkb().
void sumUpArea3D(double &sum) const override
Sums up the 3d area of the curve by iterating over the vertices (shoelace formula).
void swapXy() override
Swaps the x and y coordinates from the geometry.
void removeCurve(int i)
Removes a curve from the geometry.
void addCurve(QgsCurve *c, bool extendPrevious=false)
Adds a curve to the geometry (takes ownership).
bool moveVertex(QgsVertexId position, const QgsPoint &newPos) override
Moves a vertex within the geometry.
bool deleteVertex(QgsVertexId position) override
Deletes a vertex within the geometry.
double closestSegment(const QgsPoint &pt, QgsPoint &segmentPt, QgsVertexId &vertexAfter, int *leftOf=nullptr, double epsilon=4 *std::numeric_limits< double >::epsilon()) const override
Searches for the closest segment of the geometry to a given point.
QDomElement asGml2(QDomDocument &doc, int precision=17, const QString &ns="gml", QgsAbstractGeometry::AxisOrder axisOrder=QgsAbstractGeometry::AxisOrder::XY) const override
Returns a GML2 representation of the geometry.
bool dropZValue() override
Drops any z-dimensions which exist in the geometry.
double xAt(int index) const override
Returns the x-coordinate of the specified node in the line string.
double yAt(int index) const override
Returns the y-coordinate of the specified node in the line string.
bool dropMValue() override
Drops any measure values which exist in the geometry.
QgsCompoundCurve & operator=(const QgsCompoundCurve &curve)
QgsCompoundCurve * clone() const override
Clones the geometry by performing a deep copy.
const QgsCurve * curveAt(int i) const
Returns the curve at the specified index.
void points(QgsPointSequence &pts) const override
Returns a list of points within the curve.
bool isValid(QString &error, Qgis::GeometryValidityFlags flags=Qgis::GeometryValidityFlags()) const override
Checks validity of the geometry, and returns true if the geometry is valid.
~QgsCompoundCurve() override
QByteArray asWkb(QgsAbstractGeometry::WkbFlags flags=QgsAbstractGeometry::WkbFlags()) const override
Returns a WKB representation of the geometry.
void clear() override
Clears the geometry, ie reset it to a null geometry.
int indexOf(const QgsPoint &point) const final
Returns the index of the first vertex matching the given point, or -1 if a matching vertex is not fou...
void transform(const QgsCoordinateTransform &ct, Qgis::TransformDirection d=Qgis::TransformDirection::Forward, bool transformZ=false) override
Transforms the geometry using a coordinate transform.
bool hasCurvedSegments() const override
Returns true if the geometry contains curved segments.
QgsCompoundCurve * curveSubstring(double startDistance, double endDistance) const override
Returns a new curve representing a substring of this curve.
void scroll(int firstVertexIndex) final
Scrolls the curve vertices so that they start with the vertex at the given index.
double length() const override
Returns the planar, 2-dimensional length of the geometry.
void addToPainterPath(QPainterPath &path) const override
Adds a curve to a painter path.
QgsPoint * interpolatePoint(double distance) const override
Returns an interpolated point on the curve at the specified distance.
QgsCompoundCurve * createEmptyWithSameType() const override
Creates a new geometry with the same class and same WKB type as the original and transfers ownership.
bool pointAt(int node, QgsPoint &point, Qgis::VertexType &type) const override
Returns the point and vertex id of a point within the curve.
QgsPoint startPoint() const override
Returns the starting point of the curve.
QgsCompoundCurve * snappedToGrid(double hSpacing, double vSpacing, double dSpacing=0, double mSpacing=0, bool removeRedundantPoints=false) const override
Makes a new geometry with all the points or vertices snapped to the closest point of the grid.
QgsPoint endPoint() const override
Returns the end point of the curve.
int numPoints() const override
Returns the number of points in the curve.
double zAt(int index) const override
Returns the z-coordinate of the specified node in the line string.
void addVertex(const QgsPoint &pt)
Adds a vertex to the end of the geometry.
bool addZValue(double zValue=0) override
Adds a z-dimension to the geometry, initialized to a preset value.
QDomElement asGml3(QDomDocument &doc, int precision=17, const QString &ns="gml", QgsAbstractGeometry::AxisOrder axisOrder=QgsAbstractGeometry::AxisOrder::XY) const override
Returns a GML3 representation of the geometry.
Qgis::WkbType readHeader() const
readHeader
virtual int numPoints() const =0
Returns the number of points in the curve.
void clearCache() const override
Clears any cached parameters associated with the geometry, e.g., bounding boxes.
bool mHasCachedSummedUpArea
bool mHasCachedSummedUpArea3D
virtual bool isClosed() const
Returns true if the curve is closed.
bool isValid(QString &error, Qgis::GeometryValidityFlags flags=Qgis::GeometryValidityFlags()) const override
Checks validity of the geometry, and returns true if the geometry is valid.
QgsBox3D mBoundingBox
Cached bounding box.
virtual QgsPoint startPoint() const =0
Returns the starting point of the curve.
virtual QgsPoint endPoint() const =0
Returns the end point of the curve.
Base class for feedback objects to be used for cancellation of something running in a worker thread.
bool isCanceled() const
Tells whether the operation has been canceled already.
static double averageAngle(double x1, double y1, double x2, double y2, double x3, double y3)
Calculates the average angle (in radians) between the two linear segments from (x1,...
static QStringList wktGetChildBlocks(const QString &wkt, const QString &defaultType=QString())
Parses a WKT string and returns of list of blocks contained in the WKT.
static QPair< Qgis::WkbType, QString > wktReadBlock(const QString &wkt)
Parses a WKT block of the format "TYPE( contents )" and returns a pair of geometry type to contents (...
static double closestSegmentFromComponents(T &container, ComponentType ctype, const QgsPoint &pt, QgsPoint &segmentPt, QgsVertexId &vertexAfter, int *leftOf, double epsilon)
Line string geometry type, with support for z-dimension and m-values.
void append(const QgsLineString *line)
Appends the contents of another line string to the end of this line string.
bool insertVertex(QgsVertexId position, const QgsPoint &vertex) override
Inserts a vertex into the geometry.
void addVertex(const QgsPoint &pt)
Adds a new vertex to the end of the line string.
Point geometry type, with support for z-dimension and m-values.
QgsPoint vertexAt(QgsVertexId) const override
Returns the point corresponding to a specified vertex id.
static Qgis::WkbType dropM(Qgis::WkbType type)
Drops the m dimension (if present) for a WKB type and returns the new type.
static Qgis::WkbType dropZ(Qgis::WkbType type)
Drops the z dimension (if present) for a WKB type and returns the new type.
static Qgis::WkbType addM(Qgis::WkbType type)
Adds the m dimension to a WKB type and returns the new type.
static Qgis::WkbType addZ(Qgis::WkbType type)
Adds the z dimension to a WKB type and returns the new type.
static Q_INVOKABLE bool hasZ(Qgis::WkbType type)
Tests whether a WKB type contains the z-dimension.
static Q_INVOKABLE bool hasM(Qgis::WkbType type)
Tests whether a WKB type contains m values.
static Qgis::WkbType flatType(Qgis::WkbType type)
Returns the flat type for a WKB type.
As part of the API refactoring and improvements which landed in the Processing API was substantially reworked from the x version This was done in order to allow much of the underlying Processing framework to be ported into c
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference).
T qgsgeometry_cast(QgsAbstractGeometry *geom)
QVector< QgsPoint > QgsPointSequence
Utility class for identifying a unique vertex within a geometry.