Branch data Line data Source code
1 : : /*************************************************************************** 2 : : qgsgeometrysegmentlengthcheck.cpp 3 : : --------------------- 4 : : begin : September 2015 5 : : copyright : (C) 2014 by Sandro Mani / Sourcepole AG 6 : : email : smani at sourcepole dot ch 7 : : *************************************************************************** 8 : : * * 9 : : * This program is free software; you can redistribute it and/or modify * 10 : : * it under the terms of the GNU General Public License as published by * 11 : : * the Free Software Foundation; either version 2 of the License, or * 12 : : * (at your option) any later version. * 13 : : * * 14 : : ***************************************************************************/ 15 : : 16 : : #include "qgsgeometrycheckcontext.h" 17 : : #include "qgsgeometrysegmentlengthcheck.h" 18 : : #include "qgsgeometryutils.h" 19 : : #include "qgsfeaturepool.h" 20 : : #include "qgsgeometrycheckerror.h" 21 : : 22 : 1 : void QgsGeometrySegmentLengthCheck::collectErrors( const QMap<QString, QgsFeaturePool *> &featurePools, QList<QgsGeometryCheckError *> &errors, QStringList &messages, QgsFeedback *feedback, const LayerFeatureIds &ids ) const 23 : : { 24 : 1 : Q_UNUSED( messages ) 25 : : 26 : 1 : QMap<QString, QgsFeatureIds> featureIds = ids.isEmpty() ? allLayerFeatureIds( featurePools ) : ids.toMap(); 27 : 1 : QgsGeometryCheckerUtils::LayerFeatures layerFeatures( featurePools, featureIds, compatibleGeometryTypes(), feedback, mContext ); 28 : 35 : for ( const QgsGeometryCheckerUtils::LayerFeature &layerFeature : layerFeatures ) 29 : : { 30 : 34 : double layerToMapUnits = scaleFactor( layerFeature.layer() ); 31 : 34 : double minLength = mMinLengthMapUnits / layerToMapUnits; 32 : : 33 : 34 : const QgsAbstractGeometry *geom = layerFeature.geometry().constGet(); 34 : 70 : for ( int iPart = 0, nParts = geom->partCount(); iPart < nParts; ++iPart ) 35 : : { 36 : 73 : for ( int iRing = 0, nRings = geom->ringCount( iPart ); iRing < nRings; ++iRing ) 37 : : { 38 : 37 : bool isClosed = false; 39 : 37 : int nVerts = QgsGeometryCheckerUtils::polyLineSize( geom, iPart, iRing, &isClosed ); 40 : 37 : if ( nVerts < 2 ) 41 : : { 42 : 0 : continue; 43 : : } 44 : 206 : for ( int iVert = isClosed ? 0 : 1, jVert = isClosed ? nVerts - 1 : 0; iVert < nVerts; jVert = iVert++ ) 45 : : { 46 : 169 : QgsPoint pi = geom->vertexAt( QgsVertexId( iPart, iRing, iVert ) ); 47 : 169 : QgsPoint pj = geom->vertexAt( QgsVertexId( iPart, iRing, jVert ) ); 48 : 169 : double dist = pi.distance( pj ); 49 : : // Don't report very small lengths, they are either duplicate nodes or degenerate geometries 50 : 169 : if ( dist < minLength && dist > mContext->tolerance ) 51 : : { 52 : 4 : QgsPointXY pos( 0.5 * ( pi.x() + pj.x() ), 0.5 * ( pi.y() + pj.y() ) ); 53 : 4 : errors.append( new QgsGeometryCheckError( this, layerFeature, pos, QgsVertexId( iPart, iRing, iVert ), dist * layerToMapUnits, QgsGeometryCheckError::ValueLength ) ); 54 : 4 : } 55 : 169 : } 56 : 37 : } 57 : 36 : } 58 : : } 59 : 1 : } 60 : : 61 : 0 : void QgsGeometrySegmentLengthCheck::fixError( const QMap<QString, QgsFeaturePool *> &featurePools, QgsGeometryCheckError *error, int method, const QMap<QString, int> & /*mergeAttributeIndices*/, Changes &/*changes*/ ) const 62 : : { 63 : 0 : QgsFeaturePool *featurePool = featurePools[ error->layerId() ]; 64 : 0 : QgsFeature feature; 65 : 0 : if ( !featurePool->getFeature( error->featureId(), feature ) ) 66 : : { 67 : 0 : error->setObsolete(); 68 : 0 : return; 69 : : } 70 : : 71 : 0 : QgsGeometry featureGeom = feature.geometry(); 72 : 0 : const QgsAbstractGeometry *geom = featureGeom.constGet(); 73 : 0 : QgsVertexId vidx = error->vidx(); 74 : : 75 : : // Check if point still exists 76 : 0 : if ( !vidx.isValid( geom ) ) 77 : : { 78 : 0 : error->setObsolete(); 79 : 0 : return; 80 : : } 81 : : 82 : : // Check if error still applies 83 : 0 : int nVerts = QgsGeometryCheckerUtils::polyLineSize( geom, vidx.part, vidx.ring ); 84 : 0 : if ( nVerts == 0 ) 85 : : { 86 : 0 : error->setObsolete(); 87 : 0 : return; 88 : : } 89 : : 90 : 0 : QgsPoint pi = geom->vertexAt( error->vidx() ); 91 : 0 : QgsPoint pj = geom->vertexAt( QgsVertexId( vidx.part, vidx.ring, ( vidx.vertex - 1 + nVerts ) % nVerts ) ); 92 : 0 : double dist = pi.distance( pj ); 93 : 0 : double layerToMapUnits = scaleFactor( featurePool->layer() ); 94 : 0 : double minLength = mMinLengthMapUnits / layerToMapUnits; 95 : 0 : if ( dist >= minLength ) 96 : : { 97 : 0 : error->setObsolete(); 98 : 0 : return; 99 : : } 100 : : 101 : : // Fix error 102 : 0 : if ( method == NoChange ) 103 : : { 104 : 0 : error->setFixed( method ); 105 : 0 : } 106 : : else 107 : : { 108 : 0 : error->setFixFailed( tr( "Unknown method" ) ); 109 : : } 110 : 0 : } 111 : : 112 : 0 : QStringList QgsGeometrySegmentLengthCheck::resolutionMethods() const 113 : : { 114 : 0 : static QStringList methods = QStringList() << tr( "No action" ); 115 : 0 : return methods; 116 : 0 : } 117 : : 118 : 0 : QgsGeometryCheck::CheckType QgsGeometrySegmentLengthCheck::factoryCheckType() 119 : : { 120 : 0 : return QgsGeometryCheck::FeatureNodeCheck; 121 : : }