QGIS API Documentation 3.43.0-Master (58029bba303)
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#include "qgsvectorlayer.h"
21
23
24QString QgsCheckValidityAlgorithm::name() const
25{
26 return QStringLiteral( "checkvalidity" );
27}
28
29QString QgsCheckValidityAlgorithm::displayName() const
30{
31 return QObject::tr( "Check validity" );
32}
33
34QStringList QgsCheckValidityAlgorithm::tags() const
35{
36 return QObject::tr( "valid,invalid,detect,error" ).split( ',' );
37}
38
39QString QgsCheckValidityAlgorithm::group() const
40{
41 return QObject::tr( "Vector geometry" );
42}
43
44QString QgsCheckValidityAlgorithm::groupId() const
45{
46 return QStringLiteral( "vectorgeometry" );
47}
48
49QString QgsCheckValidityAlgorithm::shortHelpString() const
50{
51 return QObject::tr( "Performs a validity check on the geometries of a vector layer.\n\n"
52 "The geometries are classified in three groups (valid, invalid and error), and a vector layer "
53 "is generated with the features in each of these categories.\n\n"
54 "By default the algorithm uses the strict OGC definition of polygon validity, where a polygon "
55 "is marked as invalid if a self-intersecting ring causes an interior hole. If the 'Ignore "
56 "ring self intersections' option is checked, then this rule will be ignored and a more "
57 "lenient validity check will be performed.\n\n"
58 "The GEOS method is faster and performs better on larger geometries, but is limited to only "
59 "returning the first error encountered in a geometry. The QGIS method will be slower but "
60 "reports all errors encountered in the geometry, not just the first." );
61}
62
63QgsCheckValidityAlgorithm *QgsCheckValidityAlgorithm::createInstance() const
64{
65 return new QgsCheckValidityAlgorithm();
66}
67
68void QgsCheckValidityAlgorithm::initAlgorithm( const QVariantMap & )
69{
70 addParameter( new QgsProcessingParameterFeatureSource( QStringLiteral( "INPUT_LAYER" ), QObject::tr( "Input layer" ), QList<int>() << static_cast<int>( Qgis::ProcessingSourceType::VectorAnyGeometry ) ) );
71
72 const QStringList options = QStringList()
73 << QObject::tr( "The one selected in digitizing settings" )
74 << QStringLiteral( "QGIS" )
75 << QStringLiteral( "GEOS" );
76 auto methodParam = std::make_unique<QgsProcessingParameterEnum>( QStringLiteral( "METHOD" ), QObject::tr( "Method" ), options, false, 2 );
77 QVariantMap methodParamMetadata;
78 QVariantMap widgetMetadata;
79 widgetMetadata.insert( QStringLiteral( "useCheckBoxes" ), true );
80 widgetMetadata.insert( QStringLiteral( "columns" ), 3 );
81 methodParamMetadata.insert( QStringLiteral( "widget_wrapper" ), widgetMetadata );
82 methodParam->setMetadata( methodParamMetadata );
83 addParameter( methodParam.release() );
84
85 addParameter( new QgsProcessingParameterBoolean( QStringLiteral( "IGNORE_RING_SELF_INTERSECTION" ), QObject::tr( "Ignore ring self intersections" ), false ) );
86 addParameter( new QgsProcessingParameterFeatureSink( QStringLiteral( "VALID_OUTPUT" ), QObject::tr( "Valid output" ), Qgis::ProcessingSourceType::VectorAnyGeometry, QVariant(), true ) );
87 addParameter( new QgsProcessingParameterFeatureSink( QStringLiteral( "INVALID_OUTPUT" ), QObject::tr( "Invalid output" ), Qgis::ProcessingSourceType::VectorAnyGeometry, QVariant(), true ) );
88 addParameter( new QgsProcessingParameterFeatureSink( QStringLiteral( "ERROR_OUTPUT" ), QObject::tr( "Error output" ), Qgis::ProcessingSourceType::VectorAnyGeometry, QVariant(), true ) );
89 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "VALID_COUNT" ), QObject::tr( "Count of valid features" ) ) );
90 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "INVALID_COUNT" ), QObject::tr( "Count of invalid features" ) ) );
91 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "ERROR_COUNT" ), QObject::tr( "Count of errors" ) ) );
92}
93
94QVariantMap QgsCheckValidityAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
95{
96 std::unique_ptr<QgsProcessingFeatureSource> source( parameterAsSource( parameters, QStringLiteral( "INPUT_LAYER" ), context ) );
97 if ( !source )
98 throw QgsProcessingException( invalidSourceError( parameters, QStringLiteral( "INPUT_LAYER" ) ) );
99
100 int method = parameterAsEnum( parameters, QStringLiteral( "METHOD" ), context );
101 if ( method == 0 )
102 {
103 const int methodFromSettings = QgsSettingsRegistryCore::settingsDigitizingValidateGeometries->value() - 1;
104 method = methodFromSettings > 0 ? methodFromSettings : 0;
105 }
106 else
107 {
108 method--;
109 }
110
111 const bool ignoreRingSelfIntersection = parameterAsBool( parameters, QStringLiteral( "IGNORE_RING_SELF_INTERSECTION" ), context );
112
114
115 QString validSinkId;
116 std::unique_ptr<QgsFeatureSink> validSink( parameterAsSink( parameters, QStringLiteral( "VALID_OUTPUT" ), context, validSinkId, source->fields(), source->wkbType(), source->sourceCrs() ) );
117
118 QgsFields invalidFields = source->fields();
119 invalidFields.append( QgsField( QStringLiteral( "_errors" ), QMetaType::Type::QString, QStringLiteral( "string" ), 255 ) );
120 QString invalidSinkId;
121 std::unique_ptr<QgsFeatureSink> invalidSink( parameterAsSink( parameters, QStringLiteral( "INVALID_OUTPUT" ), context, invalidSinkId, invalidFields, source->wkbType(), source->sourceCrs() ) );
122
123 QgsFields errorFields;
124 errorFields.append( QgsField( QStringLiteral( "message" ), QMetaType::Type::QString, QStringLiteral( "string" ), 255 ) );
125 QString errorSinkId;
126 std::unique_ptr<QgsFeatureSink> errorSink( parameterAsSink( parameters, QStringLiteral( "ERROR_OUTPUT" ), context, errorSinkId, errorFields, Qgis::WkbType::Point, source->sourceCrs() ) );
127
128 int validCount = 0;
129 int invalidCount = 0;
130 int errorCount = 0;
131
132 const long count = source->featureCount();
133 const double step = count > 0 ? 100.0 / count : 1;
134 long long current = 0;
135
137 QgsFeature f;
138 while ( it.nextFeature( f ) )
139 {
140 if ( feedback->isCanceled() )
141 {
142 break;
143 }
144
145 const QgsGeometry geom = f.geometry();
146 QgsAttributes attrs = f.attributes();
147
148 bool isValid = true;
149
150 if ( !geom.isNull() && !geom.isEmpty() )
151 {
152 QVector< QgsGeometry::Error > errors;
153 geom.validateGeometry( errors, Qgis::GeometryValidationEngine( method ), flags );
154 if ( errors.count() > 0 )
155 {
156 isValid = false;
157 QStringList reasons;
158 reasons.reserve( errors.count() );
159 for ( const QgsGeometry::Error &error : std::as_const( errors ) )
160 {
161 if ( errorSink )
162 {
163 QgsFeature f;
164 f.setGeometry( QgsGeometry::fromPointXY( error.where() ) );
165 f.setAttributes( QVector< QVariant >() << error.what() );
166 if ( !errorSink->addFeature( f, QgsFeatureSink::FastInsert ) )
167 {
168 throw QgsProcessingException( writeFeatureError( errorSink.get(), parameters, QStringLiteral( "ERROR_OUTPUT" ) ) );
169 }
170 }
171 errorCount++;
172 reasons.append( error.what() );
173 }
174 QString reason = reasons.join( '\n' );
175 if ( reason.size() > 255 )
176 {
177 reason = reason.left( 252 ) + QStringLiteral( "…" );
178 }
179 attrs.append( reason );
180 }
181 }
182
183 QgsFeature f;
184 f.setGeometry( geom );
185 f.setAttributes( attrs );
186
187 if ( isValid )
188 {
189 if ( validSink && !validSink->addFeature( f, QgsFeatureSink::FastInsert ) )
190 {
191 throw QgsProcessingException( writeFeatureError( validSink.get(), parameters, QStringLiteral( "VALID_OUTPUT" ) ) );
192 }
193 validCount++;
194 }
195 else
196 {
197 if ( invalidSink && !invalidSink->addFeature( f, QgsFeatureSink::FastInsert ) )
198 {
199 throw QgsProcessingException( writeFeatureError( invalidSink.get(), parameters, QStringLiteral( "INVALID_OUTPUT" ) ) );
200 }
201 invalidCount++;
202 }
203
204 feedback->setProgress( current * step );
205 current++;
206 }
207
208 if ( validSink )
209 {
210 validSink->finalize();
211 }
212 if ( invalidSink )
213 {
214 invalidSink->finalize();
215 }
216 if ( errorSink )
217 {
218 errorSink->finalize();
219 }
220
221 QVariantMap outputs;
222 outputs.insert( QStringLiteral( "VALID_COUNT" ), validCount );
223 outputs.insert( QStringLiteral( "INVALID_COUNT" ), invalidCount );
224 outputs.insert( QStringLiteral( "ERROR_COUNT" ), errorCount );
225
226 if ( validSink )
227 {
228 outputs.insert( QStringLiteral( "VALID_OUTPUT" ), validSinkId );
229 }
230 if ( invalidSink )
231 {
232 outputs.insert( QStringLiteral( "INVALID_OUTPUT" ), invalidSinkId );
233 }
234 if ( errorSink )
235 {
236 outputs.insert( QStringLiteral( "ERROR_OUTPUT" ), errorSinkId );
237 }
238
239 return outputs;
240}
241
@ VectorAnyGeometry
Any vector layer with geometry.
@ AllowSelfTouchingHoles
Indicates that self-touching holes are permitted. OGC validity states that self-touching holes are NO...
QFlags< GeometryValidityFlag > GeometryValidityFlags
Geometry validity flags.
Definition qgis.h:2038
GeometryValidationEngine
Available engines for validating geometries.
Definition qgis.h:2047
@ SkipGeometryValidityChecks
Invalid geometry checks should always be skipped. This flag can be useful for algorithms which always...
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:58
QgsAttributes attributes
Definition qgsfeature.h:67
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
QgsGeometry geometry
Definition qgsfeature.h:69
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:53
void setProgress(double progress)
Sets the current progress for the feedback object.
Definition qgsfeedback.h:61
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:53
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:70
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.
T value(const QString &dynamicKeyPart=QString()) const
Returns settings value.
static const QgsSettingsEntryInteger * settingsDigitizingValidateGeometries
Settings entry digitizing validate geometries.