QGIS API Documentation  3.20.0-Odense (decaadbb31)
qgsmultisurface.cpp
Go to the documentation of this file.
1 
2 /***************************************************************************
3  qgsmultisurface.cpp
4  -------------------------------------------------------------------
5 Date : 28 Oct 2014
6 Copyright : (C) 2014 by Marco Hugentobler
7 email : 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 #include "qgsapplication.h"
19 #include "qgsgeometryutils.h"
20 #include "qgssurface.h"
21 #include "qgslinestring.h"
22 #include "qgspolygon.h"
23 #include "qgscurvepolygon.h"
24 #include "qgsmulticurve.h"
25 
26 #include <QJsonObject>
27 #include <nlohmann/json.hpp>
28 
30 {
32 }
33 
35 {
36  return qgsgeometry_cast< QgsSurface * >( geometryN( index ) );
37 }
38 
39 const QgsSurface *QgsMultiSurface::surfaceN( int index ) const
40 {
41  return qgsgeometry_cast< const QgsSurface * >( geometryN( index ) );
42 }
43 
45 {
46  return QStringLiteral( "MultiSurface" );
47 }
48 
50 {
53 }
54 
56 {
57  auto result = std::make_unique< QgsMultiSurface >();
58  result->mWkbType = mWkbType;
59  return result.release();
60 }
61 
63 {
64  return new QgsMultiSurface( *this );
65 }
66 
68 {
69  return clone();
70 }
71 
72 bool QgsMultiSurface::fromWkt( const QString &wkt )
73 {
74  return fromCollectionWkt( wkt,
75  QVector<QgsAbstractGeometry *>() << new QgsPolygon << new QgsCurvePolygon,
76  QStringLiteral( "Polygon" ) );
77 }
78 
79 QDomElement QgsMultiSurface::asGml2( QDomDocument &doc, int precision, const QString &ns, const AxisOrder axisOrder ) const
80 {
81  // GML2 does not support curves
82  QDomElement elemMultiPolygon = doc.createElementNS( ns, QStringLiteral( "MultiPolygon" ) );
83 
84  if ( isEmpty() )
85  return elemMultiPolygon;
86 
87  for ( const QgsAbstractGeometry *geom : mGeometries )
88  {
89  if ( qgsgeometry_cast<const QgsSurface *>( geom ) )
90  {
91  std::unique_ptr< QgsPolygon > polygon( static_cast<const QgsSurface *>( geom )->surfaceToPolygon() );
92 
93  QDomElement elemPolygonMember = doc.createElementNS( ns, QStringLiteral( "polygonMember" ) );
94  elemPolygonMember.appendChild( polygon->asGml2( doc, precision, ns, axisOrder ) );
95  elemMultiPolygon.appendChild( elemPolygonMember );
96  }
97  }
98 
99  return elemMultiPolygon;
100 }
101 
102 QDomElement QgsMultiSurface::asGml3( QDomDocument &doc, int precision, const QString &ns, const AxisOrder axisOrder ) const
103 {
104  QDomElement elemMultiSurface = doc.createElementNS( ns, QStringLiteral( "MultiSurface" ) );
105 
106  if ( isEmpty() )
107  return elemMultiSurface;
108 
109  for ( const QgsAbstractGeometry *geom : mGeometries )
110  {
111  if ( qgsgeometry_cast<const QgsSurface *>( geom ) )
112  {
113  QDomElement elemSurfaceMember = doc.createElementNS( ns, QStringLiteral( "surfaceMember" ) );
114  elemSurfaceMember.appendChild( geom->asGml3( doc, precision, ns, axisOrder ) );
115  elemMultiSurface.appendChild( elemSurfaceMember );
116  }
117  }
118 
119  return elemMultiSurface;
120 }
121 
122 
124 {
125  json polygons( json::array( ) );
126  for ( const QgsAbstractGeometry *geom : std::as_const( mGeometries ) )
127  {
128  if ( qgsgeometry_cast<const QgsSurface *>( geom ) )
129  {
130  json coordinates( json::array( ) );
131  std::unique_ptr< QgsPolygon >polygon( static_cast<const QgsSurface *>( geom )->surfaceToPolygon() );
132  std::unique_ptr< QgsLineString > exteriorLineString( polygon->exteriorRing()->curveToLine() );
133  QgsPointSequence exteriorPts;
134  exteriorLineString->points( exteriorPts );
135  coordinates.push_back( QgsGeometryUtils::pointsToJson( exteriorPts, precision ) );
136 
137  std::unique_ptr< QgsLineString > interiorLineString;
138  for ( int i = 0, n = polygon->numInteriorRings(); i < n; ++i )
139  {
140  interiorLineString.reset( polygon->interiorRing( i )->curveToLine() );
141  QgsPointSequence interiorPts;
142  interiorLineString->points( interiorPts );
143  coordinates.push_back( QgsGeometryUtils::pointsToJson( interiorPts, precision ) );
144  }
145  polygons.push_back( coordinates );
146  }
147  }
148  return
149  {
150  { "type", "MultiPolygon" },
151  { "coordinates", polygons }
152  };
153 }
154 
156 {
157  if ( !qgsgeometry_cast<QgsSurface *>( g ) )
158  {
159  delete g;
160  return false;
161  }
162 
163  if ( mGeometries.empty() )
164  {
166  }
167  if ( is3D() && !g->is3D() )
168  g->addZValue();
169  else if ( !is3D() && g->is3D() )
170  g->dropZValue();
171  if ( isMeasure() && !g->isMeasure() )
172  g->addMValue();
173  else if ( !isMeasure() && g->isMeasure() )
174  g->dropMValue();
175 
177 }
178 
180 {
181  if ( !g || !qgsgeometry_cast< QgsSurface * >( g ) )
182  {
183  delete g;
184  return false;
185  }
186 
187  return QgsGeometryCollection::insertGeometry( g, index );
188 }
189 
191 {
192  std::unique_ptr< QgsMultiCurve > multiCurve( new QgsMultiCurve() );
193  multiCurve->reserve( mGeometries.size() );
194  for ( int i = 0; i < mGeometries.size(); ++i )
195  {
196  if ( QgsSurface *surface = qgsgeometry_cast<QgsSurface *>( mGeometries.at( i ) ) )
197  {
198  multiCurve->addGeometry( surface->boundary() );
199  }
200  }
201  if ( multiCurve->numGeometries() == 0 )
202  {
203  return nullptr;
204  }
205  return multiCurve.release();
206 }
Abstract base class for all geometries.
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 is3D() const SIP_HOLDGIL
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.
virtual bool dropZValue()=0
Drops any z-dimensions which exist in the geometry.
QgsWkbTypes::Type mWkbType
bool isMeasure() const SIP_HOLDGIL
Returns true if the geometry contains m values.
void setZMTypeFromSubGeometry(const QgsAbstractGeometry *subggeom, QgsWkbTypes::Type baseGeomType)
Updates the geometry type based on whether sub geometries contain z or m values.
Curve polygon geometry type.
bool isEmpty() const override SIP_HOLDGIL
Returns true if the geometry is empty.
QVector< QgsAbstractGeometry * > mGeometries
void clear() override
Clears the geometry, ie reset it to a null geometry.
virtual bool insertGeometry(QgsAbstractGeometry *g, int index)
Inserts a geometry before a specified index and takes ownership.
const QgsAbstractGeometry * geometryN(int n) const
Returns a const reference to a geometry from within the collection.
virtual bool addGeometry(QgsAbstractGeometry *g)
Adds a geometry and takes ownership. Returns true in case of success.
bool fromCollectionWkt(const QString &wkt, const QVector< QgsAbstractGeometry * > &subtypes, const QString &defaultChildWkbType=QString())
Reads a collection from a WKT string.
static json pointsToJson(const QgsPointSequence &points, int precision)
Returns coordinates as json object.
Multi curve geometry collection.
Definition: qgsmulticurve.h:30
Multi surface geometry collection.
QgsMultiSurface() SIP_HOLDGIL
Constructor for an empty multisurface geometry.
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.
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.
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.
QString geometryType() const override SIP_HOLDGIL
Returns a unique string representing the geometry type.
Polygon geometry type.
Definition: qgspolygon.h:34
Surface geometry type.
Definition: qgssurface.h:34
QVector< QgsPoint > QgsPointSequence
int precision