Branch data Line data Source code
1 : : /***************************************************************************
2 : : * qgsgeometrycheckerutils.cpp *
3 : : * ------------------- *
4 : : * copyright : (C) 2014 by Sandro Mani / Sourcepole AG *
5 : : * email : smani@sourcepole.ch *
6 : : ***************************************************************************/
7 : :
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 "qgsgeometrycheckcontext.h"
18 : : #include "qgsgeometrycheckerutils.h"
19 : : #include "qgsgeometry.h"
20 : : #include "qgsgeometryutils.h"
21 : : #include "qgsfeaturepool.h"
22 : : #include "qgspolygon.h"
23 : : #include "qgsgeos.h"
24 : : #include "qgsgeometrycollection.h"
25 : : #include "qgssurface.h"
26 : : #include "qgsvectorlayer.h"
27 : : #include "qgsgeometrycheck.h"
28 : : #include "qgsfeedback.h"
29 : :
30 : : #include <qmath.h>
31 : :
32 : 1082 : QgsGeometryCheckerUtils::LayerFeature::LayerFeature( const QgsFeaturePool *pool,
33 : : const QgsFeature &feature,
34 : : const QgsGeometryCheckContext *context,
35 : : bool useMapCrs )
36 : 1082 : : mFeaturePool( pool )
37 : 1082 : , mFeature( feature )
38 : 1082 : , mMapCrs( useMapCrs )
39 : : {
40 : 1082 : mGeometry = feature.geometry();
41 : 1082 : const QgsCoordinateTransform transform( pool->crs(), context->mapCrs, context->transformContext );
42 : 1082 : if ( useMapCrs && context->mapCrs.isValid() && !transform.isShortCircuited() )
43 : : {
44 : : try
45 : : {
46 : 0 : mGeometry.transform( transform );
47 : 0 : }
48 : : catch ( const QgsCsException & )
49 : : {
50 : 0 : QgsDebugMsg( QStringLiteral( "Shrug. What shall we do with a geometry that cannot be converted?" ) );
51 : 0 : }
52 : 0 : }
53 : 1082 : }
54 : :
55 : 890 : QgsFeature QgsGeometryCheckerUtils::LayerFeature::feature() const
56 : : {
57 : 890 : return mFeature;
58 : : }
59 : :
60 : 717 : QPointer<QgsVectorLayer> QgsGeometryCheckerUtils::LayerFeature::layer() const
61 : : {
62 : 717 : return mFeaturePool->layerPtr();
63 : : }
64 : :
65 : 746 : QString QgsGeometryCheckerUtils::LayerFeature::layerId() const
66 : : {
67 : 746 : return mFeaturePool->layerId();
68 : : }
69 : :
70 : 1118 : QgsGeometry QgsGeometryCheckerUtils::LayerFeature::geometry() const
71 : : {
72 : 1118 : return mGeometry;
73 : : }
74 : :
75 : 29 : QString QgsGeometryCheckerUtils::LayerFeature::id() const
76 : : {
77 : 58 : return QStringLiteral( "%1:%2" ).arg( mFeaturePool->layerName() ).arg( mFeature.id() );
78 : 0 : }
79 : :
80 : 144 : bool QgsGeometryCheckerUtils::LayerFeature::operator==( const LayerFeature &other ) const
81 : : {
82 : 144 : return layerId() == other.layerId() && mFeature.id() == other.mFeature.id();
83 : 0 : }
84 : :
85 : 0 : bool QgsGeometryCheckerUtils::LayerFeature::operator!=( const LayerFeature &other ) const
86 : : {
87 : 0 : return layerId() != other.layerId() || mFeature.id() != other.mFeature.id();
88 : 0 : }
89 : :
90 : : /////////////////////////////////////////////////////////////////////////////
91 : :
92 : 432 : QgsGeometryCheckerUtils::LayerFeatures::iterator::iterator( const QStringList::const_iterator &layerIt, const LayerFeatures *parent )
93 : 432 : : mLayerIt( layerIt )
94 : 432 : , mFeatureIt( QgsFeatureIds::const_iterator() )
95 : 432 : , mParent( parent )
96 : : {
97 : 432 : nextLayerFeature( true );
98 : 432 : }
99 : :
100 : 0 : QgsGeometryCheckerUtils::LayerFeatures::iterator::iterator( const QgsGeometryCheckerUtils::LayerFeatures::iterator &rh )
101 : : {
102 : 0 : mLayerIt = rh.mLayerIt;
103 : 0 : mFeatureIt = rh.mFeatureIt;
104 : 0 : mParent = rh.mParent;
105 : 0 : mCurrentFeature = std::make_unique<LayerFeature>( *rh.mCurrentFeature.get() );
106 : 0 : }
107 : :
108 : 105 : bool QgsGeometryCheckerUtils::LayerFeature::useMapCrs() const
109 : : {
110 : 105 : return mMapCrs;
111 : : }
112 : 432 : QgsGeometryCheckerUtils::LayerFeatures::iterator::~iterator()
113 : : {
114 : 432 : }
115 : :
116 : 0 : QgsGeometryCheckerUtils::LayerFeatures::iterator QgsGeometryCheckerUtils::LayerFeatures::iterator::operator++( int )
117 : : {
118 : 0 : iterator tmp( *this );
119 : 0 : ++*this;
120 : 0 : return tmp;
121 : 0 : }
122 : :
123 : 1074 : const QgsGeometryCheckerUtils::LayerFeature &QgsGeometryCheckerUtils::LayerFeatures::iterator::operator*() const
124 : : {
125 : : Q_ASSERT( mCurrentFeature );
126 : 1074 : return *mCurrentFeature;
127 : : }
128 : :
129 : 1284 : bool QgsGeometryCheckerUtils::LayerFeatures::iterator::operator!=( const QgsGeometryCheckerUtils::LayerFeatures::iterator &other )
130 : : {
131 : 1284 : return mLayerIt != other.mLayerIt || mFeatureIt != other.mFeatureIt;
132 : : }
133 : :
134 : 1068 : const QgsGeometryCheckerUtils::LayerFeatures::iterator &QgsGeometryCheckerUtils::LayerFeatures::iterator::operator++()
135 : : {
136 : 1068 : nextLayerFeature( false );
137 : 1068 : return *this;
138 : : }
139 : 1500 : bool QgsGeometryCheckerUtils::LayerFeatures::iterator::nextLayerFeature( bool begin )
140 : : {
141 : 1500 : if ( !begin && nextFeature( false ) )
142 : : {
143 : 842 : return true;
144 : : }
145 : 732 : while ( nextLayer( begin ) )
146 : : {
147 : 306 : begin = false;
148 : 306 : if ( nextFeature( true ) )
149 : : {
150 : 232 : return true;
151 : : }
152 : : }
153 : : // End
154 : 426 : mFeatureIt = QgsFeatureIds::const_iterator();
155 : 426 : mCurrentFeature.reset();
156 : 426 : return false;
157 : 1500 : }
158 : :
159 : 732 : bool QgsGeometryCheckerUtils::LayerFeatures::iterator::nextLayer( bool begin )
160 : : {
161 : 732 : if ( !begin )
162 : : {
163 : 300 : ++mLayerIt;
164 : 300 : }
165 : 920 : while ( true )
166 : : {
167 : 920 : if ( mLayerIt == mParent->mLayerIds.end() )
168 : : {
169 : 426 : break;
170 : : }
171 : 494 : if ( mParent->mGeometryTypes.contains( mParent->mFeaturePools[*mLayerIt]->geometryType() ) )
172 : : {
173 : 306 : mFeatureIt = mParent->mFeatureIds[*mLayerIt].constBegin();
174 : 306 : return true;
175 : : }
176 : 188 : ++mLayerIt;
177 : : }
178 : 426 : return false;
179 : 732 : }
180 : :
181 : 1374 : bool QgsGeometryCheckerUtils::LayerFeatures::iterator::nextFeature( bool begin )
182 : : {
183 : 1374 : QgsFeaturePool *featurePool = mParent->mFeaturePools[*mLayerIt];
184 : 1374 : const QgsFeatureIds &featureIds = mParent->mFeatureIds[*mLayerIt];
185 : 1374 : if ( !begin )
186 : : {
187 : 1068 : ++mFeatureIt;
188 : 1068 : }
189 : 1374 : while ( true )
190 : : {
191 : 1374 : if ( mFeatureIt == featureIds.end() )
192 : : {
193 : 300 : break;
194 : : }
195 : 1074 : if ( mParent->mFeedback )
196 : 490 : mParent->mFeedback->setProgress( mParent->mFeedback->progress() + 1.0 );
197 : 1074 : QgsFeature feature;
198 : 1074 : if ( featurePool->getFeature( *mFeatureIt, feature ) && !feature.geometry().isNull() )
199 : : {
200 : 1074 : mCurrentFeature = std::make_unique<LayerFeature>( featurePool, feature, mParent->mContext, mParent->mUseMapCrs );
201 : 1074 : return true;
202 : : }
203 : 0 : ++mFeatureIt;
204 : 1074 : }
205 : 300 : return false;
206 : 1374 : }
207 : :
208 : 189 : /////////////////////////////////////////////////////////////////////////////
209 : :
210 : 54 : QgsGeometryCheckerUtils::LayerFeatures::LayerFeatures( const QMap<QString, QgsFeaturePool *> &featurePools,
211 : : const QMap<QString, QgsFeatureIds> &featureIds,
212 : : const QList<QgsWkbTypes::GeometryType> &geometryTypes,
213 : : QgsFeedback *feedback,
214 : : const QgsGeometryCheckContext *context,
215 : : bool useMapCrs )
216 : 27 : : mFeaturePools( featurePools )
217 : 27 : , mFeatureIds( featureIds )
218 : 27 : , mLayerIds( featurePools.keys() )
219 : 27 : , mGeometryTypes( geometryTypes )
220 : 27 : , mFeedback( feedback )
221 : 27 : , mContext( context )
222 : 27 : , mUseMapCrs( useMapCrs )
223 : 27 : {}
224 : :
225 : 189 : QgsGeometryCheckerUtils::LayerFeatures::LayerFeatures( const QMap<QString, QgsFeaturePool *> &featurePools,
226 : : const QList<QString> &layerIds, const QgsRectangle &extent,
227 : : const QList<QgsWkbTypes::GeometryType> &geometryTypes,
228 : : const QgsGeometryCheckContext *context )
229 : 189 : : mFeaturePools( featurePools )
230 : 189 : , mLayerIds( layerIds )
231 : 189 : , mExtent( extent )
232 : 189 : , mGeometryTypes( geometryTypes )
233 : 189 : , mContext( context )
234 : 189 : , mUseMapCrs( true )
235 : : {
236 : 621 : for ( const QString &layerId : layerIds )
237 : : {
238 : 432 : const QgsFeaturePool *featurePool = featurePools[layerId];
239 : 432 : if ( geometryTypes.contains( featurePool->geometryType() ) )
240 : : {
241 : 267 : QgsCoordinateTransform ct( featurePool->crs(), context->mapCrs, context->transformContext );
242 : 267 : mFeatureIds.insert( layerId, featurePool->getIntersects( ct.transform( extent, QgsCoordinateTransform::ReverseTransform ) ) );
243 : 267 : }
244 : : else
245 : : {
246 : 165 : mFeatureIds.insert( layerId, QgsFeatureIds() );
247 : : }
248 : : }
249 : 189 : }
250 : :
251 : 216 : QgsGeometryCheckerUtils::LayerFeatures::iterator QgsGeometryCheckerUtils::LayerFeatures::begin() const
252 : : {
253 : 216 : return iterator( mLayerIds.constBegin(), this );
254 : : }
255 : :
256 : 216 : QgsGeometryCheckerUtils::LayerFeatures::iterator QgsGeometryCheckerUtils::LayerFeatures::end() const
257 : : {
258 : 216 : return iterator( mLayerIds.end(), this );
259 : : }
260 : :
261 : : /////////////////////////////////////////////////////////////////////////////
262 : :
263 : 339 : std::unique_ptr<QgsGeometryEngine> QgsGeometryCheckerUtils::createGeomEngine( const QgsAbstractGeometry *geometry, double tolerance )
264 : : {
265 : 339 : return std::make_unique<QgsGeos>( geometry, tolerance );
266 : : }
267 : :
268 : 17 : QgsAbstractGeometry *QgsGeometryCheckerUtils::getGeomPart( QgsAbstractGeometry *geom, int partIdx )
269 : : {
270 : 17 : if ( dynamic_cast<QgsGeometryCollection *>( geom ) )
271 : : {
272 : 11 : return static_cast<QgsGeometryCollection *>( geom )->geometryN( partIdx );
273 : : }
274 : 6 : return geom;
275 : 17 : }
276 : :
277 : 252 : const QgsAbstractGeometry *QgsGeometryCheckerUtils::getGeomPart( const QgsAbstractGeometry *geom, int partIdx )
278 : : {
279 : 252 : if ( dynamic_cast<const QgsGeometryCollection *>( geom ) )
280 : : {
281 : 225 : return static_cast<const QgsGeometryCollection *>( geom )->geometryN( partIdx );
282 : : }
283 : 27 : return geom;
284 : 252 : }
285 : :
286 : 9 : QList<const QgsLineString *> QgsGeometryCheckerUtils::polygonRings( const QgsPolygon *polygon )
287 : : {
288 : 9 : QList<const QgsLineString *> rings;
289 : 9 : if ( const QgsLineString *exterior = dynamic_cast<const QgsLineString *>( polygon->exteriorRing() ) )
290 : : {
291 : 9 : rings.append( exterior );
292 : 9 : }
293 : 10 : for ( int iInt = 0, nInt = polygon->numInteriorRings(); iInt < nInt; ++iInt )
294 : : {
295 : 1 : if ( const QgsLineString *interior = dynamic_cast<const QgsLineString *>( polygon->interiorRing( iInt ) ) )
296 : : {
297 : 1 : rings.append( interior );
298 : 1 : }
299 : 1 : }
300 : 9 : return rings;
301 : 9 : }
302 : :
303 : 11 : void QgsGeometryCheckerUtils::filter1DTypes( QgsAbstractGeometry *geom )
304 : : {
305 : 11 : if ( qgsgeometry_cast<QgsGeometryCollection *>( geom ) )
306 : : {
307 : 2 : QgsGeometryCollection *geomCollection = static_cast<QgsGeometryCollection *>( geom );
308 : 6 : for ( int nParts = geom->partCount(), iPart = nParts - 1; iPart >= 0; --iPart )
309 : : {
310 : 4 : if ( !qgsgeometry_cast<QgsSurface *>( geomCollection->geometryN( iPart ) ) )
311 : : {
312 : 0 : geomCollection->removeGeometry( iPart );
313 : 0 : }
314 : 4 : }
315 : 2 : }
316 : 11 : }
317 : :
318 : 622 : double pointLineDist( const QgsPoint &p1, const QgsPoint &p2, const QgsPoint &q )
319 : : {
320 : 622 : double nom = std::fabs( ( p2.y() - p1.y() ) * q.x() - ( p2.x() - p1.x() ) * q.y() + p2.x() * p1.y() - p2.y() * p1.x() );
321 : 622 : double dx = p2.x() - p1.x();
322 : 622 : double dy = p2.y() - p1.y();
323 : 622 : return nom / std::sqrt( dx * dx + dy * dy );
324 : : }
325 : :
326 : 45 : bool QgsGeometryCheckerUtils::pointOnLine( const QgsPoint &p, const QgsLineString *line, double tol, bool excludeExtremities )
327 : : {
328 : 45 : int nVerts = line->vertexCount();
329 : 143 : for ( int i = 0 + excludeExtremities; i < nVerts - 1 - excludeExtremities; ++i )
330 : : {
331 : 114 : QgsPoint p1 = line->vertexAt( QgsVertexId( 0, 0, i ) );
332 : 114 : QgsPoint p2 = line->vertexAt( QgsVertexId( 0, 0, i + 1 ) );
333 : 114 : double dist = pointLineDist( p1, p2, p );
334 : 114 : if ( dist < tol )
335 : : {
336 : 16 : return true;
337 : : }
338 : 114 : }
339 : 29 : return false;
340 : 45 : }
341 : :
342 : 19 : QList<QgsPoint> QgsGeometryCheckerUtils::lineIntersections( const QgsLineString *line1, const QgsLineString *line2, double tol )
343 : : {
344 : 19 : QList<QgsPoint> intersections;
345 : 19 : QgsPoint inter;
346 : 19 : bool intersection = false;
347 : 97 : for ( int i = 0, n = line1->vertexCount() - 1; i < n; ++i )
348 : : {
349 : 426 : for ( int j = 0, m = line2->vertexCount() - 1; j < m; ++j )
350 : : {
351 : 348 : QgsPoint p1 = line1->vertexAt( QgsVertexId( 0, 0, i ) );
352 : 348 : QgsPoint p2 = line1->vertexAt( QgsVertexId( 0, 0, i + 1 ) );
353 : 348 : QgsPoint q1 = line2->vertexAt( QgsVertexId( 0, 0, j ) );
354 : 348 : QgsPoint q2 = line2->vertexAt( QgsVertexId( 0, 0, j + 1 ) );
355 : 348 : if ( QgsGeometryUtils::segmentIntersection( p1, p2, q1, q2, inter, intersection, tol ) )
356 : : {
357 : 6 : intersections.append( inter );
358 : 6 : }
359 : 348 : }
360 : 78 : }
361 : 19 : return intersections;
362 : 19 : }
363 : :
364 : 19 : double QgsGeometryCheckerUtils::sharedEdgeLength( const QgsAbstractGeometry *geom1, const QgsAbstractGeometry *geom2, double tol )
365 : : {
366 : 19 : double len = 0;
367 : :
368 : : // Test every pair of segments for shared edges
369 : 38 : for ( int iPart1 = 0, nParts1 = geom1->partCount(); iPart1 < nParts1; ++iPart1 )
370 : : {
371 : 39 : for ( int iRing1 = 0, nRings1 = geom1->ringCount( iPart1 ); iRing1 < nRings1; ++iRing1 )
372 : : {
373 : 105 : for ( int iVert1 = 0, jVert1 = 1, nVerts1 = geom1->vertexCount( iPart1, iRing1 ); jVert1 < nVerts1; iVert1 = jVert1++ )
374 : : {
375 : 85 : QgsPoint p1 = geom1->vertexAt( QgsVertexId( iPart1, iRing1, iVert1 ) );
376 : 85 : QgsPoint p2 = geom1->vertexAt( QgsVertexId( iPart1, iRing1, jVert1 ) );
377 : 85 : double lambdap1 = 0.;
378 : 85 : double lambdap2 = std::sqrt( QgsGeometryUtils::sqrDistance2D( p1, p2 ) );
379 : 85 : QgsVector d;
380 : : try
381 : : {
382 : 85 : d = QgsVector( p2.x() - p1.x(), p2.y() - p1.y() ).normalized();
383 : 85 : }
384 : : catch ( const QgsException & )
385 : : {
386 : : // Edge has zero length, skip
387 : : continue;
388 : 0 : }
389 : :
390 : 170 : for ( int iPart2 = 0, nParts2 = geom2->partCount(); iPart2 < nParts2; ++iPart2 )
391 : : {
392 : 170 : for ( int iRing2 = 0, nRings2 = geom2->ringCount( iPart2 ); iRing2 < nRings2; ++iRing2 )
393 : : {
394 : 482 : for ( int iVert2 = 0, jVert2 = 1, nVerts2 = geom2->vertexCount( iPart2, iRing2 ); jVert2 < nVerts2; iVert2 = jVert2++ )
395 : : {
396 : 397 : QgsPoint q1 = geom2->vertexAt( QgsVertexId( iPart2, iRing2, iVert2 ) );
397 : 397 : QgsPoint q2 = geom2->vertexAt( QgsVertexId( iPart2, iRing2, jVert2 ) );
398 : :
399 : : // Check whether q1 and q2 are on the line p1, p
400 : 397 : if ( pointLineDist( p1, p2, q1 ) <= tol && pointLineDist( p1, p2, q2 ) <= tol )
401 : : {
402 : : // Get length common edge
403 : 43 : double lambdaq1 = QgsVector( q1.x() - p1.x(), q1.y() - p1.y() ) * d;
404 : 43 : double lambdaq2 = QgsVector( q2.x() - p1.x(), q2.y() - p1.y() ) * d;
405 : 43 : if ( lambdaq1 > lambdaq2 )
406 : : {
407 : 32 : std::swap( lambdaq1, lambdaq2 );
408 : 32 : }
409 : 43 : double lambda1 = std::max( lambdaq1, lambdap1 );
410 : 43 : double lambda2 = std::min( lambdaq2, lambdap2 );
411 : 43 : len += std::max( 0., lambda2 - lambda1 );
412 : 43 : }
413 : 397 : }
414 : 85 : }
415 : 85 : }
416 : 85 : }
417 : 20 : }
418 : 19 : }
419 : 19 : return len;
420 : 0 : }
|