QGIS API Documentation 4.1.0-Master (5bf3c20f3c9)
Loading...
Searching...
No Matches
qgsalgorithmfixgeometrymultipart.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsalgorithmfixgeometrymultipart.cpp
3 ---------------------
4 begin : April 2025
5 copyright : (C) 2025 by Jacky Volpes
6 email : jacky dot volpes at oslandia 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
24#include "qgsvectorfilewriter.h"
25#include "qgsvectorlayer.h"
26
27#include <QString>
28
29using namespace Qt::StringLiterals;
30
32
33QString QgsFixGeometryMultipartAlgorithm::name() const
34{
35 return u"fixgeometrymultipart"_s;
36}
37
38QString QgsFixGeometryMultipartAlgorithm::displayName() const
39{
40 return QObject::tr( "Convert to strictly multipart" );
41}
42
43QString QgsFixGeometryMultipartAlgorithm::shortDescription() const
44{
45 return QObject::tr( "Converts features detected with the \"Strictly multipart\" algorithm from the \"Check geometry\" section to singlepart." );
46}
47
48QStringList QgsFixGeometryMultipartAlgorithm::tags() const
49{
50 return QObject::tr( "fix,multipart,singlepart" ).split( ',' );
51}
52
53QString QgsFixGeometryMultipartAlgorithm::group() const
54{
55 return QObject::tr( "Fix geometry" );
56}
57
58QString QgsFixGeometryMultipartAlgorithm::groupId() const
59{
60 return u"fixgeometry"_s;
61}
62
63QString QgsFixGeometryMultipartAlgorithm::shortHelpString() const
64{
65 return QObject::tr(
66 "This algorithm converts multipart geometries that consists of only one geometry "
67 "into singlepart geometries, based on an error layer from the \"Strict multipart\" algorithm in the \"Check geometry\" section.\n\n"
68 "This algorithm does not change the layer geometry type, which will remain multipart."
69 );
70}
71
72QgsFixGeometryMultipartAlgorithm *QgsFixGeometryMultipartAlgorithm::createInstance() const
73{
74 return new QgsFixGeometryMultipartAlgorithm();
75}
76
77void QgsFixGeometryMultipartAlgorithm::initAlgorithm( const QVariantMap &configuration )
78{
79 Q_UNUSED( configuration )
80
81 addParameter(
82 new QgsProcessingParameterFeatureSource( u"INPUT"_s, QObject::tr( "Input layer" ), QList<int>() << static_cast<int>( Qgis::ProcessingSourceType::VectorPolygon ) << static_cast<int>( Qgis::ProcessingSourceType::VectorLine ) )
83 );
84 addParameter( new QgsProcessingParameterFeatureSource( u"ERRORS"_s, QObject::tr( "Error layer" ), QList<int>() << static_cast<int>( Qgis::ProcessingSourceType::VectorPoint ) ) );
85 addParameter( new QgsProcessingParameterField( u"UNIQUE_ID"_s, QObject::tr( "Field of original feature unique identifier" ), u"id"_s, u"ERRORS"_s ) );
86
87 addParameter( new QgsProcessingParameterFeatureSink( u"OUTPUT"_s, QObject::tr( "Strictly-multipart layer" ), Qgis::ProcessingSourceType::VectorAnyGeometry ) );
88 addParameter( new QgsProcessingParameterFeatureSink( u"REPORT"_s, QObject::tr( "Report layer from fixing multiparts" ), Qgis::ProcessingSourceType::VectorPoint ) );
89
90 auto tolerance = std::make_unique<QgsProcessingParameterNumber>( u"TOLERANCE"_s, QObject::tr( "Tolerance" ), Qgis::ProcessingNumberParameterType::Integer, 8, false, 1, 13 );
91 tolerance->setFlags( tolerance->flags() | Qgis::ProcessingParameterFlag::Advanced );
92 tolerance->setHelp(
93 QObject::tr(
94 "The \"Tolerance\" advanced parameter defines the numerical precision of geometric operations, "
95 "given as an integer n, meaning that any difference smaller than 10⁻ⁿ (in map units) is considered zero."
96 )
97 );
98 addParameter( tolerance.release() );
99}
100
101QVariantMap QgsFixGeometryMultipartAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
102{
103 const std::unique_ptr<QgsProcessingFeatureSource> input( parameterAsSource( parameters, u"INPUT"_s, context ) );
104 if ( !input )
105 throw QgsProcessingException( invalidSourceError( parameters, u"INPUT"_s ) );
106
107 const std::unique_ptr<QgsProcessingFeatureSource> errors( parameterAsSource( parameters, u"ERRORS"_s, context ) );
108 if ( !errors )
109 throw QgsProcessingException( invalidSourceError( parameters, u"ERRORS"_s ) );
110
111 QgsProcessingMultiStepFeedback multiStepFeedback( 2, feedback );
112
113 const QString featIdFieldName = parameterAsString( parameters, u"UNIQUE_ID"_s, context );
114
115 // Verify that input fields exists
116 if ( errors->fields().indexFromName( featIdFieldName ) == -1 )
117 throw QgsProcessingException( QObject::tr( "Field \"%1\" does not exist in the error layer." ).arg( featIdFieldName ) );
118 int inputIdFieldIndex = input->fields().indexFromName( featIdFieldName );
119 if ( inputIdFieldIndex == -1 )
120 throw QgsProcessingException( QObject::tr( "Field \"%1\" does not exist in input layer." ).arg( featIdFieldName ) );
121
122 const QgsField inputFeatIdField = input->fields().at( inputIdFieldIndex );
123 if ( inputFeatIdField.type() != errors->fields().at( errors->fields().indexFromName( featIdFieldName ) ).type() )
124 throw QgsProcessingException( QObject::tr( "Field \"%1\" does not have the same type as in the error layer." ).arg( featIdFieldName ) );
125
126 QString dest_output;
127 const std::unique_ptr<QgsFeatureSink> sink_output( parameterAsSink( parameters, u"OUTPUT"_s, context, dest_output, input->fields(), input->wkbType(), input->sourceCrs() ) );
128 if ( !sink_output )
129 throw QgsProcessingException( invalidSinkError( parameters, u"OUTPUT"_s ) );
130
131 QString dest_report;
132 QgsFields reportFields = errors->fields();
133 reportFields.append( QgsField( u"report"_s, QMetaType::QString ) );
134 reportFields.append( QgsField( u"error_fixed"_s, QMetaType::Bool ) );
135 const std::unique_ptr<QgsFeatureSink> sink_report( parameterAsSink( parameters, u"REPORT"_s, context, dest_report, reportFields, errors->wkbType(), errors->sourceCrs() ) );
136 if ( !sink_report )
137 throw QgsProcessingException( invalidSinkError( parameters, u"REPORT"_s ) );
138
139 QgsGeometryCheckContext checkContext = QgsGeometryCheckContext( mTolerance, input->sourceCrs(), context.transformContext(), context.project() );
140
141 const QgsGeometryMultipartCheck check( &checkContext, QVariantMap() );
142
143 QgsVectorLayer *fixedLayer = input->materialize( QgsFeatureRequest() );
145 QMap<QString, QgsFeaturePool *> featurePools;
146 featurePools.insert( fixedLayer->id(), &featurePool );
147
148 QgsFeature errorFeature, inputFeature, testDuplicateIdFeature;
149 QgsFeatureIterator errorFeaturesIt = errors->getFeatures();
150 QList<QgsGeometryCheck::Changes> changesList;
151 QgsFeature reportFeature;
152 reportFeature.setFields( reportFields );
153 long long progression = 0;
154 long long totalProgression = errors->featureCount();
155 multiStepFeedback.setCurrentStep( 1 );
156 multiStepFeedback.setProgressText( QObject::tr( "Fixing errors..." ) );
157 while ( errorFeaturesIt.nextFeature( errorFeature ) )
158 {
159 if ( feedback->isCanceled() )
160 break;
161
162 progression++;
163 multiStepFeedback.setProgress( static_cast<double>( static_cast<long double>( progression ) / totalProgression ) * 100 );
164 reportFeature.setGeometry( errorFeature.geometry() );
165
166 QString idValue = errorFeature.attribute( featIdFieldName ).toString();
167 if ( inputFeatIdField.type() == QMetaType::QString )
168 idValue = "'" + idValue + "'";
169
170 QgsFeatureIterator it = fixedLayer->getFeatures( QgsFeatureRequest().setFilterExpression( "\"" + featIdFieldName + "\" = " + idValue ) );
171 if ( !it.nextFeature( inputFeature ) || !inputFeature.isValid() )
172 reportFeature.setAttributes( errorFeature.attributes() << QObject::tr( "Source feature not found or invalid" ) << false );
173
174 else if ( it.nextFeature( testDuplicateIdFeature ) )
175 throw QgsProcessingException( QObject::tr( "More than one feature found in input layer with value \"%1\" in unique field \"%2\"" ).arg( idValue, featIdFieldName ) );
176
177 else if ( inputFeature.geometry().isNull() )
178 reportFeature.setAttributes( errorFeature.attributes() << QObject::tr( "Feature geometry is null" ) << false );
179
180 else
181 {
182 QgsGeometryCheckError checkError
183 = QgsGeometryCheckError( &check, QgsGeometryCheckerUtils::LayerFeature( &featurePool, inputFeature, &checkContext, false ), errorFeature.geometry().asPoint(), QgsVertexId() );
184 for ( const QgsGeometryCheck::Changes &changes : std::as_const( changesList ) )
185 checkError.handleChanges( changes );
186
188 check.fixError( featurePools, &checkError, QgsGeometryMultipartCheck::ResolutionMethod::ConvertToSingle, QMap<QString, int>(), changes );
189 changesList << changes;
190 QString resolutionMessage = checkError.resolutionMessage();
191 if ( checkError.status() == QgsGeometryCheckError::StatusObsolete )
192 resolutionMessage = QObject::tr( "Error is obsolete" );
193 reportFeature.setAttributes( errorFeature.attributes() << resolutionMessage << ( checkError.status() == QgsGeometryCheckError::StatusFixed ) );
194 }
195
196 if ( !sink_report->addFeature( reportFeature, QgsFeatureSink::FastInsert ) )
197 throw QgsProcessingException( writeFeatureError( sink_report.get(), parameters, u"REPORT"_s ) );
198 }
199 multiStepFeedback.setProgress( 100 );
200
201 progression = 0;
202 totalProgression = fixedLayer->featureCount();
203 multiStepFeedback.setCurrentStep( 2 );
204 multiStepFeedback.setProgressText( QObject::tr( "Exporting fixed layer..." ) );
205 QgsFeature fixedFeature;
206 QgsFeatureIterator fixedFeaturesIt = fixedLayer->getFeatures();
207 while ( fixedFeaturesIt.nextFeature( fixedFeature ) )
208 {
209 if ( feedback->isCanceled() )
210 break;
211
212 progression++;
213 multiStepFeedback.setProgress( static_cast<double>( static_cast<long double>( progression ) / totalProgression ) * 100 );
214 if ( !sink_output->addFeature( fixedFeature, QgsFeatureSink::FastInsert ) )
215 throw QgsProcessingException( writeFeatureError( sink_output.get(), parameters, u"OUTPUT"_s ) );
216 }
217 multiStepFeedback.setProgress( 100 );
218
219 QVariantMap outputs;
220 outputs.insert( u"OUTPUT"_s, dest_output );
221 outputs.insert( u"REPORT"_s, dest_report );
222
223 return outputs;
224}
225
226bool QgsFixGeometryMultipartAlgorithm::prepareAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback * )
227{
228 mTolerance = parameterAsInt( parameters, u"TOLERANCE"_s, context );
229
230 return true;
231}
232
233Qgis::ProcessingAlgorithmFlags QgsFixGeometryMultipartAlgorithm::flags() const
234{
236}
237
@ VectorAnyGeometry
Any vector layer with geometry.
Definition qgis.h:3647
@ VectorPoint
Vector point layers.
Definition qgis.h:3648
@ VectorPolygon
Vector polygon layers.
Definition qgis.h:3650
@ VectorLine
Vector line layers.
Definition qgis.h:3649
QFlags< ProcessingAlgorithmFlag > ProcessingAlgorithmFlags
Flags indicating how and when an algorithm operates and should be exposed to users.
Definition qgis.h:3724
@ NoThreading
Algorithm is not thread safe and cannot be run in a background thread, e.g. for algorithms which mani...
Definition qgis.h:3703
@ RequiresProject
The algorithm requires that a valid QgsProject is available from the processing context in order to e...
Definition qgis.h:3711
@ Advanced
Parameter is an advanced parameter which should be hidden from users by default.
Definition qgis.h:3880
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...
QgsVectorLayer * materialize(const QgsFeatureRequest &request, QgsFeedback *feedback=nullptr)
Materializes a request (query) made against this feature source, by running it over the source and re...
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.
void setFields(const QgsFields &fields, bool initAttributes=false)
Assigns a field map with the feature to allow attribute access by attribute name.
QgsGeometry geometry
Definition qgsfeature.h:71
bool isValid() const
Returns the validity of this feature.
Q_INVOKABLE QVariant attribute(const QString &name) const
Lookup attribute value by attribute name.
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
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:56
QMetaType::Type type
Definition qgsfield.h:63
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
Base configuration for geometry checks.
This represents an error reported by a geometry check.
@ StatusFixed
The error is fixed.
@ StatusObsolete
The error is obsolete because of other modifications.
Status status() const
The status of the error.
QString resolutionMessage() const
A message with details, how the error has been resolved.
virtual bool handleChanges(const QgsGeometryCheck::Changes &changes)
Apply a list of changes.
QMap< QString, QMap< QgsFeatureId, QList< QgsGeometryCheck::Change > > > Changes
A collection of changes.
A layer feature combination to uniquely identify and access a feature in a set of layers.
QgsPointXY asPoint() const
Returns the contents of the geometry as a 2-dimensional point.
QString id
Definition qgsmaplayer.h:86
virtual Qgis::ProcessingAlgorithmFlags flags() const
Returns the flags indicating how and when the algorithm operates and should be exposed to users.
Contains information about the context in which a processing algorithm is executed.
QgsCoordinateTransformContext transformContext() const
Returns the coordinate transform context.
QgsProject * project() const
Returns the project in which the algorithm is being executed.
Custom exception class for processing related exceptions.
Base class for providing feedback from a processing algorithm.
Processing feedback object for multi-step operations.
A feature sink output for processing algorithms.
An input feature source (such as vector layers) parameter for processing algorithms.
A vector layer or feature source field parameter for processing algorithms.
A feature pool based on a vector data provider.
Represents a vector layer which manages a vector based dataset.
long long featureCount(const QString &legendKey) const
Number of features rendered with specified legend key.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const final
Queries the layer for features specified in request.
Utility class for identifying a unique vertex within a geometry.
Definition qgsvertexid.h:34