QGIS API Documentation 4.1.0-Master (5bf3c20f3c9)
Loading...
Searching...
No Matches
qgsalgorithmcheckvalidity.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsalgorithmcheckvalidity.cpp
3 ---------------------
4 begin : February 2025
5 copyright : (C) 2025 by Alexander Bruy
6 email : alexander dot bruy at gmail dot com
7 ***************************************************************************/
8
9/***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
19
21#include "qgsvectorlayer.h"
22
23#include <QString>
24
25using namespace Qt::StringLiterals;
26
28
29QString QgsCheckValidityAlgorithm::name() const
30{
31 return u"checkvalidity"_s;
32}
33
34QString QgsCheckValidityAlgorithm::displayName() const
35{
36 return QObject::tr( "Check validity" );
37}
38
39QStringList QgsCheckValidityAlgorithm::tags() const
40{
41 return QObject::tr( "valid,invalid,detect,error" ).split( ',' );
42}
43
44QString QgsCheckValidityAlgorithm::group() const
45{
46 return QObject::tr( "Vector geometry" );
47}
48
49QString QgsCheckValidityAlgorithm::groupId() const
50{
51 return u"vectorgeometry"_s;
52}
53
54QString QgsCheckValidityAlgorithm::shortHelpString() const
55{
56 return QObject::tr(
57 "This algorithm performs a validity check on the geometries of a vector layer.\n\n"
58 "The geometries are classified in three groups (valid, invalid and error), and a vector layer "
59 "is generated with the features in each of these categories.\n\n"
60 "By default the algorithm uses the strict OGC definition of polygon validity, where a polygon "
61 "is marked as invalid if a self-intersecting ring causes an interior hole. If the 'Ignore "
62 "ring self intersections' option is checked, then this rule will be ignored and a more "
63 "lenient validity check will be performed.\n\n"
64 "The GEOS method is faster and performs better on larger geometries, but is limited to only "
65 "returning the first error encountered in a geometry. The QGIS method will be slower but "
66 "reports all errors encountered in the geometry, not just the first."
67 );
68}
69
70QString QgsCheckValidityAlgorithm::shortDescription() const
71{
72 return QObject::tr(
73 "Performs a validity check on the geometries of a vector layer "
74 "and classifies them in three groups (valid, invalid and error)."
75 );
76}
77
78QgsCheckValidityAlgorithm *QgsCheckValidityAlgorithm::createInstance() const
79{
80 return new QgsCheckValidityAlgorithm();
81}
82
83void QgsCheckValidityAlgorithm::initAlgorithm( const QVariantMap & )
84{
85 addParameter( new QgsProcessingParameterFeatureSource( u"INPUT_LAYER"_s, QObject::tr( "Input layer" ), QList<int>() << static_cast<int>( Qgis::ProcessingSourceType::VectorAnyGeometry ) ) );
86
87 const QStringList options = QStringList() << QObject::tr( "The one selected in digitizing settings" ) << u"QGIS"_s << u"GEOS"_s;
88 auto methodParam = std::make_unique<QgsProcessingParameterEnum>( u"METHOD"_s, QObject::tr( "Method" ), options, false, 2 );
89 QVariantMap methodParamMetadata;
90 QVariantMap widgetMetadata;
91 widgetMetadata.insert( u"useCheckBoxes"_s, true );
92 widgetMetadata.insert( u"columns"_s, 3 );
93 methodParamMetadata.insert( u"widget_wrapper"_s, widgetMetadata );
94 methodParam->setMetadata( methodParamMetadata );
95 addParameter( methodParam.release() );
96
97 addParameter( new QgsProcessingParameterBoolean( u"IGNORE_RING_SELF_INTERSECTION"_s, QObject::tr( "Ignore ring self intersections" ), false ) );
98 addParameter( new QgsProcessingParameterFeatureSink( u"VALID_OUTPUT"_s, QObject::tr( "Valid output" ), Qgis::ProcessingSourceType::VectorAnyGeometry, QVariant(), true ) );
99 addParameter( new QgsProcessingParameterFeatureSink( u"INVALID_OUTPUT"_s, QObject::tr( "Invalid output" ), Qgis::ProcessingSourceType::VectorAnyGeometry, QVariant(), true ) );
100 addParameter( new QgsProcessingParameterFeatureSink( u"ERROR_OUTPUT"_s, QObject::tr( "Error output" ), Qgis::ProcessingSourceType::VectorAnyGeometry, QVariant(), true ) );
101 addOutput( new QgsProcessingOutputNumber( u"VALID_COUNT"_s, QObject::tr( "Count of valid features" ) ) );
102 addOutput( new QgsProcessingOutputNumber( u"INVALID_COUNT"_s, QObject::tr( "Count of invalid features" ) ) );
103 addOutput( new QgsProcessingOutputNumber( u"ERROR_COUNT"_s, QObject::tr( "Count of errors" ) ) );
104}
105
106QVariantMap QgsCheckValidityAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
107{
108 std::unique_ptr<QgsProcessingFeatureSource> source( parameterAsSource( parameters, u"INPUT_LAYER"_s, context ) );
109 if ( !source )
110 throw QgsProcessingException( invalidSourceError( parameters, u"INPUT_LAYER"_s ) );
111
112 int method = parameterAsEnum( parameters, u"METHOD"_s, context );
113 if ( method == 0 )
114 {
115 const int methodFromSettings = QgsSettingsRegistryCore::settingsDigitizingValidateGeometries->value() - 1;
116 method = methodFromSettings > 0 ? methodFromSettings : 0;
117 }
118 else
119 {
120 method--;
121 }
122
123 const bool ignoreRingSelfIntersection = parameterAsBool( parameters, u"IGNORE_RING_SELF_INTERSECTION"_s, context );
124
126
127 QString validSinkId;
128 std::unique_ptr<QgsFeatureSink> validSink( parameterAsSink( parameters, u"VALID_OUTPUT"_s, context, validSinkId, source->fields(), source->wkbType(), source->sourceCrs() ) );
129
130 QgsFields invalidFields = source->fields();
131 invalidFields.append( QgsField( u"_errors"_s, QMetaType::Type::QString, u"string"_s, 255 ) );
132 QString invalidSinkId;
133 std::unique_ptr<QgsFeatureSink> invalidSink( parameterAsSink( parameters, u"INVALID_OUTPUT"_s, context, invalidSinkId, invalidFields, source->wkbType(), source->sourceCrs() ) );
134
135 QgsFields errorFields;
136 errorFields.append( QgsField( u"message"_s, QMetaType::Type::QString, u"string"_s, 255 ) );
137 QString errorSinkId;
138 std::unique_ptr<QgsFeatureSink> errorSink( parameterAsSink( parameters, u"ERROR_OUTPUT"_s, context, errorSinkId, errorFields, Qgis::WkbType::Point, source->sourceCrs() ) );
139
140 int validCount = 0;
141 int invalidCount = 0;
142 int errorCount = 0;
143
144 const long count = source->featureCount();
145 const double step = count > 0 ? 100.0 / count : 1;
146 long long current = 0;
147
149 QgsFeature f;
150 while ( it.nextFeature( f ) )
151 {
152 if ( feedback->isCanceled() )
153 {
154 break;
155 }
156
157 const QgsGeometry geom = f.geometry();
158 QgsAttributes attrs = f.attributes();
159
160 bool isValid = true;
161
162 if ( !geom.isNull() && !geom.isEmpty() )
163 {
164 QVector< QgsGeometry::Error > errors;
165 geom.validateGeometry( errors, Qgis::GeometryValidationEngine( method ), flags );
166 if ( errors.count() > 0 )
167 {
168 isValid = false;
169 QStringList reasons;
170 reasons.reserve( errors.count() );
171 for ( const QgsGeometry::Error &error : std::as_const( errors ) )
172 {
173 if ( errorSink )
174 {
175 QgsFeature f;
176 f.setGeometry( QgsGeometry::fromPointXY( error.where() ) );
177 f.setAttributes( QVector< QVariant >() << error.what() );
178 if ( !errorSink->addFeature( f, QgsFeatureSink::FastInsert ) )
179 {
180 throw QgsProcessingException( writeFeatureError( errorSink.get(), parameters, u"ERROR_OUTPUT"_s ) );
181 }
182 }
183 errorCount++;
184 reasons.append( error.what() );
185 }
186 QString reason = reasons.join( '\n' );
187 if ( reason.size() > 255 )
188 {
189 reason = reason.left( 252 ) + u"…"_s;
190 }
191 attrs.append( reason );
192 }
193 }
194
195 QgsFeature f;
196 f.setGeometry( geom );
197 f.setAttributes( attrs );
198
199 if ( isValid )
200 {
201 if ( validSink && !validSink->addFeature( f, QgsFeatureSink::FastInsert ) )
202 {
203 throw QgsProcessingException( writeFeatureError( validSink.get(), parameters, u"VALID_OUTPUT"_s ) );
204 }
205 validCount++;
206 }
207 else
208 {
209 if ( invalidSink && !invalidSink->addFeature( f, QgsFeatureSink::FastInsert ) )
210 {
211 throw QgsProcessingException( writeFeatureError( invalidSink.get(), parameters, u"INVALID_OUTPUT"_s ) );
212 }
213 invalidCount++;
214 }
215
216 feedback->setProgress( current * step );
217 current++;
218 }
219
220 if ( validSink )
221 {
222 validSink->finalize();
223 }
224 if ( invalidSink )
225 {
226 invalidSink->finalize();
227 }
228 if ( errorSink )
229 {
230 errorSink->finalize();
231 }
232
233 QVariantMap outputs;
234 outputs.insert( u"VALID_COUNT"_s, validCount );
235 outputs.insert( u"INVALID_COUNT"_s, invalidCount );
236 outputs.insert( u"ERROR_COUNT"_s, errorCount );
237
238 if ( validSink )
239 {
240 outputs.insert( u"VALID_OUTPUT"_s, validSinkId );
241 }
242 if ( invalidSink )
243 {
244 outputs.insert( u"INVALID_OUTPUT"_s, invalidSinkId );
245 }
246 if ( errorSink )
247 {
248 outputs.insert( u"ERROR_OUTPUT"_s, errorSinkId );
249 }
250
251 return outputs;
252}
253
@ VectorAnyGeometry
Any vector layer with geometry.
Definition qgis.h:3647
@ AllowSelfTouchingHoles
Indicates that self-touching holes are permitted. OGC validity states that self-touching holes are NO...
Definition qgis.h:2151
QFlags< GeometryValidityFlag > GeometryValidityFlags
Geometry validity flags.
Definition qgis.h:2155
GeometryValidationEngine
Available engines for validating geometries.
Definition qgis.h:2164
@ SkipGeometryValidityChecks
Invalid geometry checks should always be skipped. This flag can be useful for algorithms which always...
Definition qgis.h:3828
@ Point
Point.
Definition qgis.h:296
A vector of attributes.
Wrapper for iterator of features from vector data provider or vector layer.
bool nextFeature(QgsFeature &f)
Fetch next feature and stores in f, returns true on success.
Wraps a request for features to a vector layer (or directly its vector data provider).
@ FastInsert
Use faster inserts, at the cost of updating the passed features to reflect changes made at the provid...
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
QgsAttributes attributes
Definition qgsfeature.h:69
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
QgsGeometry geometry
Definition qgsfeature.h:71
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:56
void setProgress(double progress)
Sets the current progress for the feedback object.
Definition qgsfeedback.h:65
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:56
Container of fields for a vector layer.
Definition qgsfields.h:46
bool append(const QgsField &field, Qgis::FieldOrigin origin=Qgis::FieldOrigin::Provider, int originIndex=-1)
Appends a field.
Definition qgsfields.cpp:75
A geometry error.
A geometry is the spatial representation of a feature.
static QgsGeometry fromPointXY(const QgsPointXY &point)
Creates a new geometry from a QgsPointXY object.
void validateGeometry(QVector< QgsGeometry::Error > &errors, Qgis::GeometryValidationEngine method=Qgis::GeometryValidationEngine::QgisInternal, Qgis::GeometryValidityFlags flags=Qgis::GeometryValidityFlags()) const
Validates geometry and produces a list of geometry errors.
bool isEmpty() const
Returns true if the geometry is empty (eg a linestring with no vertices, or a collection with no geom...
Contains information about the context in which a processing algorithm is executed.
Custom exception class for processing related exceptions.
Base class for providing feedback from a processing algorithm.
A numeric output for processing algorithms.
A boolean parameter for processing algorithms.
A feature sink output for processing algorithms.
An input feature source (such as vector layers) parameter for processing algorithms.
static const QgsSettingsEntryInteger * settingsDigitizingValidateGeometries
Settings entry digitizing validate geometries.