QGIS API Documentation 4.0.0-Norrköping (1ddcee3d0e4)
Loading...
Searching...
No Matches
qgsmultisurface.cpp
Go to the documentation of this file.
1
2/***************************************************************************
3 qgsmultisurface.cpp
4 -------------------------------------------------------------------
5Date : 28 Oct 2014
6Copyright : (C) 2014 by Marco Hugentobler
7email : marco.hugentobler at sourcepole dot com
8 ***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16
17#include "qgsmultisurface.h"
18
19#include <nlohmann/json.hpp>
20
21#include "qgscurvepolygon.h"
22#include "qgsgeometryutils.h"
23#include "qgslinestring.h"
24#include "qgsmulticurve.h"
25#include "qgspolygon.h"
26#include "qgssurface.h"
27
28#include <QJsonObject>
29#include <QString>
30
31using namespace Qt::StringLiterals;
32
37
42
43const QgsSurface *QgsMultiSurface::surfaceN( int index ) const
44{
46}
47
49{
50 return u"MultiSurface"_s;
51}
52
58
60{
61 auto result = std::make_unique< QgsMultiSurface >();
62 result->mWkbType = mWkbType;
63 return result.release();
64}
65
67{
68 return new QgsMultiSurface( *this );
69}
70
75
76bool QgsMultiSurface::fromWkt( const QString &wkt )
77{
79}
80
81QDomElement QgsMultiSurface::asGml2( QDomDocument &doc, int precision, const QString &ns, const AxisOrder axisOrder ) const
82{
83 // GML2 does not support curves
84 QDomElement elemMultiPolygon = doc.createElementNS( ns, u"MultiPolygon"_s );
85
86 if ( isEmpty() )
87 return elemMultiPolygon;
88
89 for ( const QgsAbstractGeometry *geom : mGeometries )
90 {
92 {
93 std::unique_ptr< QgsPolygon > polygon( static_cast<const QgsCurvePolygon *>( geom )->surfaceToPolygon() );
94
95 QDomElement elemPolygonMember = doc.createElementNS( ns, u"polygonMember"_s );
96 elemPolygonMember.appendChild( polygon->asGml2( doc, precision, ns, axisOrder ) );
97 elemMultiPolygon.appendChild( elemPolygonMember );
98 }
99 }
100
101 return elemMultiPolygon;
102}
103
104QDomElement QgsMultiSurface::asGml3( QDomDocument &doc, int precision, const QString &ns, const AxisOrder axisOrder ) const
105{
106 QDomElement elemMultiSurface = doc.createElementNS( ns, u"MultiSurface"_s );
107
108 if ( isEmpty() )
109 return elemMultiSurface;
110
111 for ( const QgsAbstractGeometry *geom : mGeometries )
112 {
114 {
115 QDomElement elemSurfaceMember = doc.createElementNS( ns, u"surfaceMember"_s );
116 elemSurfaceMember.appendChild( geom->asGml3( doc, precision, ns, axisOrder ) );
117 elemMultiSurface.appendChild( elemSurfaceMember );
118 }
119 }
120
121 return elemMultiSurface;
122}
123
124
125json QgsMultiSurface::asJsonObject( int precision ) const
126{
127 json polygons( json::array() );
128 for ( const QgsAbstractGeometry *geom : std::as_const( mGeometries ) )
129 {
131 {
132 json coordinates( json::array() );
133 std::unique_ptr< QgsPolygon > polygon( static_cast<const QgsCurvePolygon *>( geom )->surfaceToPolygon() );
134 std::unique_ptr< QgsLineString > exteriorLineString( polygon->exteriorRing()->curveToLine() );
135 QgsPointSequence exteriorPts;
136 exteriorLineString->points( exteriorPts );
137 coordinates.push_back( QgsGeometryUtils::pointsToJson( exteriorPts, precision ) );
138
139 std::unique_ptr< QgsLineString > interiorLineString;
140 for ( int i = 0, n = polygon->numInteriorRings(); i < n; ++i )
141 {
142 interiorLineString.reset( polygon->interiorRing( i )->curveToLine() );
143 QgsPointSequence interiorPts;
144 interiorLineString->points( interiorPts );
145 coordinates.push_back( QgsGeometryUtils::pointsToJson( interiorPts, precision ) );
146 }
147 polygons.push_back( coordinates );
148 }
149 }
150 return { { "type", "MultiPolygon" }, { "coordinates", polygons } };
151}
152
154{
156 {
157 delete g;
158 return false;
159 }
160
161 if ( mGeometries.empty() )
162 {
164 }
165 if ( is3D() && !g->is3D() )
166 g->addZValue();
167 else if ( !is3D() && g->is3D() )
168 g->dropZValue();
169 if ( isMeasure() && !g->isMeasure() )
170 g->addMValue();
171 else if ( !isMeasure() && g->isMeasure() )
172 g->dropMValue();
173
175}
176
177bool QgsMultiSurface::addGeometries( const QVector<QgsAbstractGeometry *> &geometries )
178{
179 for ( QgsAbstractGeometry *g : geometries )
180 {
182 {
183 qDeleteAll( geometries );
184 return false;
185 }
186 }
187
188 if ( mGeometries.empty() && !geometries.empty() )
189 {
191 }
192 mGeometries.reserve( mGeometries.size() + geometries.size() );
193 for ( QgsAbstractGeometry *g : geometries )
194 {
195 if ( is3D() && !g->is3D() )
196 g->addZValue();
197 else if ( !is3D() && g->is3D() )
198 g->dropZValue();
199 if ( isMeasure() && !g->isMeasure() )
200 g->addMValue();
201 else if ( !isMeasure() && g->isMeasure() )
202 g->dropMValue();
203 mGeometries.append( g );
204 }
205
206 clearCache();
207 return true;
208}
209
211{
212 if ( !g || !qgsgeometry_cast< QgsSurface * >( g ) )
213 {
214 delete g;
215 return false;
216 }
217
218 return QgsGeometryCollection::insertGeometry( g, index );
219}
220
222{
223 auto multiCurve = std::make_unique<QgsMultiCurve>();
224 multiCurve->reserve( mGeometries.size() );
225 for ( int i = 0; i < mGeometries.size(); ++i )
226 {
227 if ( QgsSurface *surface = qgsgeometry_cast<QgsSurface *>( mGeometries.at( i ) ) )
228 {
229 multiCurve->addGeometry( surface->boundary() );
230 }
231 }
232 if ( multiCurve->numGeometries() == 0 )
233 {
234 return nullptr;
235 }
236 return multiCurve.release();
237}
238
240{
241 auto res = std::make_unique< QgsMultiSurface >();
242 res->reserve( mGeometries.size() );
243 for ( int i = 0; i < mGeometries.size(); ++i )
244 {
245 res->addGeometry( mGeometries.at( i )->simplifyByDistance( tolerance ) );
246 }
247 return res.release();
248}
@ Polygon
Polygon.
Definition qgis.h:298
@ CurvePolygon
CurvePolygon.
Definition qgis.h:306
@ MultiSurface
MultiSurface.
Definition qgis.h:308
virtual bool addZValue(double zValue=0)=0
Adds a z-dimension to the geometry, initialized to a preset value.
virtual bool dropMValue()=0
Drops any measure values which exist in the geometry.
bool isMeasure() const
Returns true if the geometry contains m values.
bool is3D() const
Returns true if the geometry is 3D and contains a z-value.
AxisOrder
Axis order for GML generation.
virtual bool addMValue(double mValue=0)=0
Adds a measure to the geometry, initialized to a preset value.
void setZMTypeFromSubGeometry(const QgsAbstractGeometry *subggeom, Qgis::WkbType baseGeomType)
Updates the geometry type based on whether sub geometries contain z or m values.
virtual bool dropZValue()=0
Drops any z-dimensions which exist in the geometry.
QgsAbstractGeometry()=default
Curve polygon geometry type.
QVector< QgsAbstractGeometry * > mGeometries
bool fromCollectionWkt(const QString &wkt, const QVector< Qgis::WkbType > &subtypes, const QString &defaultChildWkbType=QString())
Reads a collection from a WKT string.
void clear() override
Clears the geometry, ie reset it to a null geometry.
void clearCache() const override
Clears any cached parameters associated with the geometry, e.g., bounding boxes.
virtual bool insertGeometry(QgsAbstractGeometry *g, int index)
Inserts a geometry before a specified index and takes ownership.
bool isEmpty() const override
Returns true if the geometry is empty.
virtual bool addGeometry(QgsAbstractGeometry *g)
Adds a geometry and takes ownership. Returns true in case of success.
const QgsAbstractGeometry * geometryN(int n) const
Returns a const reference to a geometry from within the collection.
static json pointsToJson(const QgsPointSequence &points, int precision)
Returns coordinates as json object.
QgsMultiSurface()
Constructor for an empty multisurface geometry.
QgsMultiSurface * simplifyByDistance(double tolerance) const override
Simplifies the geometry by applying the Douglas Peucker simplification by distance algorithm.
QgsSurface * surfaceN(int index)
Returns the surface with the specified index.
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.
QString geometryType() const override
Returns a unique string representing the geometry type.
bool addGeometry(QgsAbstractGeometry *g) override
Adds a geometry and takes ownership. Returns true in case of success.
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.
bool fromWkt(const QString &wkt) override
Sets the geometry from a WKT string.
QgsMultiSurface * clone() const override
Clones the geometry by performing a deep copy.
bool addGeometries(const QVector< QgsAbstractGeometry * > &geometries) override
Adds a list of geometries to the collection, transferring ownership to the collection.
json asJsonObject(int precision=17) const override
Returns a json object representation of the geometry.
QgsMultiSurface * createEmptyWithSameType() const override
Creates a new geometry with the same class and same WKB type as the original and transfers ownership.
QgsAbstractGeometry * boundary() const override
Returns the closure of the combinatorial boundary of the geometry (ie the topological boundary of the...
QgsMultiSurface * toCurveType() const override
Returns the geometry converted to the more generic curve type.
bool insertGeometry(QgsAbstractGeometry *g, int index) override
Inserts a geometry before a specified index and takes ownership.
void clear() override
Clears the geometry, ie reset it to a null geometry.
Surface geometry type.
Definition qgssurface.h:34
T qgsgeometry_cast(QgsAbstractGeometry *geom)
QVector< QgsPoint > QgsPointSequence