QGIS API Documentation 3.99.0-Master (357b655ed83)
Loading...
Searching...
No Matches
qgsalgorithmfixgeometries.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsalgorithmfixgeometries.cpp
3 -----------------------------
4 begin : April 2017
5 copyright : (C) 2017 by Nyall Dawson
6 email : nyall dot dawson 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
20#include <geos_c.h>
21
22#include "qgsvectorlayer.h"
23
24#include <QString>
25
26using namespace Qt::StringLiterals;
27
29
30QString QgsFixGeometriesAlgorithm::name() const
31{
32 return u"fixgeometries"_s;
33}
34
35QString QgsFixGeometriesAlgorithm::displayName() const
36{
37 return QObject::tr( "Fix geometries" );
38}
39
40QStringList QgsFixGeometriesAlgorithm::tags() const
41{
42 return QObject::tr( "repair,invalid,geometry,make,valid,error" ).split( ',' );
43}
44
45QString QgsFixGeometriesAlgorithm::group() const
46{
47 return QObject::tr( "Vector geometry" );
48}
49
50QString QgsFixGeometriesAlgorithm::groupId() const
51{
52 return u"vectorgeometry"_s;
53}
54
55Qgis::ProcessingFeatureSourceFlags QgsFixGeometriesAlgorithm::sourceFlags() const
56{
58}
59
60QString QgsFixGeometriesAlgorithm::outputName() const
61{
62 return QObject::tr( "Fixed geometries" );
63}
64
65Qgis::WkbType QgsFixGeometriesAlgorithm::outputWkbType( Qgis::WkbType type ) const
66{
68}
69
70QString QgsFixGeometriesAlgorithm::shortHelpString() const
71{
72 return QObject::tr( "This algorithm attempts to create a valid representation of a given invalid geometry without "
73 "losing any of the input vertices. Already-valid geometries are returned without further intervention. "
74 "Always outputs multi-geometry layer.\n\n"
75 "NOTE: M values will be dropped from the output." );
76}
77
78QString QgsFixGeometriesAlgorithm::shortDescription() const
79{
80 return QObject::tr( "Attempts to create a valid representation of a given invalid geometry without losing any of the input vertices." );
81}
82
83QgsFixGeometriesAlgorithm *QgsFixGeometriesAlgorithm::createInstance() const
84{
85 return new QgsFixGeometriesAlgorithm();
86}
87
88bool QgsFixGeometriesAlgorithm::supportInPlaceEdit( const QgsMapLayer *l ) const
89{
90 const QgsVectorLayer *layer = qobject_cast<const QgsVectorLayer *>( l );
91 if ( !layer )
92 return false;
93
95 return false;
96 // The algorithm would drop M, so disable it if the layer has M
97 return !QgsWkbTypes::hasM( layer->wkbType() );
98}
99
100void QgsFixGeometriesAlgorithm::initParameters( const QVariantMap & )
101{
102 auto methodParameter = std::make_unique<QgsProcessingParameterEnum>(
103 u"METHOD"_s,
104 QObject::tr( "Repair method" ),
105 QStringList { QObject::tr( "Linework" ), QObject::tr( "Structure" ) },
106 0,
107 false
108 );
109#if GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 10
110 methodParameter->setDefaultValue( 0 );
111#else
112 methodParameter->setDefaultValue( 1 );
113#endif
114 addParameter( methodParameter.release() );
115}
116
117bool QgsFixGeometriesAlgorithm::prepareAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback * )
118{
119 mMethod = static_cast<Qgis::MakeValidMethod>( parameterAsInt( parameters, u"METHOD"_s, context ) );
120#if GEOS_VERSION_MAJOR == 3 && GEOS_VERSION_MINOR < 10
121 if ( mMethod == Qgis::MakeValidMethod::Structure )
122 {
123 throw QgsProcessingException( QObject::tr( "The structured method to make geometries valid requires a QGIS build based on GEOS 3.10 or later" ) );
124 }
125#endif
126 return true;
127}
128
129QgsFeatureList QgsFixGeometriesAlgorithm::processFeature( const QgsFeature &feature, QgsProcessingContext &, QgsProcessingFeedback *feedback )
130{
131 if ( !feature.hasGeometry() )
132 return QgsFeatureList() << feature;
133
134 QgsFeature outputFeature = feature;
135
136 QgsGeometry outputGeometry = outputFeature.geometry().makeValid( mMethod );
137 if ( outputGeometry.isNull() )
138 {
139 feedback->pushInfo( QObject::tr( "makeValid failed for feature %1 " ).arg( feature.id() ) );
140 outputFeature.clearGeometry();
141 return QgsFeatureList() << outputFeature;
142 }
143
144 if ( outputGeometry.wkbType() == Qgis::WkbType::Unknown || QgsWkbTypes::flatType( outputGeometry.wkbType() ) == Qgis::WkbType::GeometryCollection )
145 {
146 // keep only the parts of the geometry collection with correct type
147 const QVector<QgsGeometry> tmpGeometries = outputGeometry.asGeometryCollection();
148 QVector<QgsGeometry> matchingParts;
149 for ( const QgsGeometry &g : tmpGeometries )
150 {
151 if ( g.type() == feature.geometry().type() )
152 matchingParts << g;
153 }
154 if ( !matchingParts.empty() )
155 outputGeometry = QgsGeometry::collectGeometry( matchingParts );
156 else
157 outputGeometry = QgsGeometry();
158 }
159
160 if ( outputGeometry.type() != Qgis::GeometryType::Point )
161 {
162 // some data providers are picky about the geometries we pass to them: we can't add single-part geometries
163 // when we promised multi-part geometries, so ensure we have the right type
164 outputGeometry.convertToMultiType();
165 }
166
167 if ( QgsWkbTypes::geometryType( outputGeometry.wkbType() ) != QgsWkbTypes::geometryType( feature.geometry().wkbType() ) )
168 {
169 // don't keep geometries which have different types - e.g. lines converted to points
170 feedback->pushInfo( QObject::tr( "Fixing geometry for feature %1 resulted in %2, geometry has been dropped." ).arg( feature.id() ).arg( QgsWkbTypes::displayString( outputGeometry.wkbType() ) ) );
171 outputFeature.clearGeometry();
172 }
173 else
174 {
175 outputFeature.setGeometry( outputGeometry );
176 }
177 return QgsFeatureList() << outputFeature;
178}
179
@ Point
Points.
Definition qgis.h:366
MakeValidMethod
Algorithms to use when repairing invalid geometries.
Definition qgis.h:2238
@ Structure
Structured method, first makes all rings valid and then merges shells and subtracts holes from shells...
Definition qgis.h:2240
@ SkipGeometryValidityChecks
Invalid geometry checks should always be skipped. This flag can be useful for algorithms which always...
Definition qgis.h:3782
WkbType
The WKB type describes the number of dimensions a geometry has.
Definition qgis.h:280
@ Unknown
Unknown.
Definition qgis.h:281
@ GeometryCollection
GeometryCollection.
Definition qgis.h:289
QFlags< ProcessingFeatureSourceFlag > ProcessingFeatureSourceFlags
Flags which control how QgsProcessingFeatureSource fetches features.
Definition qgis.h:3793
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
QgsFeatureId id
Definition qgsfeature.h:68
QgsGeometry geometry
Definition qgsfeature.h:71
void clearGeometry()
Removes any geometry associated with the feature.
bool hasGeometry() const
Returns true if the feature has an associated geometry.
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
A geometry is the spatial representation of a feature.
static QgsGeometry collectGeometry(const QVector< QgsGeometry > &geometries)
Creates a new multipart geometry from a list of QgsGeometry objects.
QgsGeometry makeValid(Qgis::MakeValidMethod method=Qgis::MakeValidMethod::Linework, bool keepCollapsed=false) const
Attempts to make an invalid geometry valid without losing vertices.
QVector< QgsGeometry > asGeometryCollection() const
Returns contents of the geometry as a list of geometries.
Qgis::GeometryType type
bool convertToMultiType()
Converts single type geometry into multitype geometry e.g.
Qgis::WkbType wkbType() const
Returns type of the geometry as a WKB type (point / linestring / polygon etc.).
Base class for all map layer types.
Definition qgsmaplayer.h:83
Contains information about the context in which a processing algorithm is executed.
Custom exception class for processing related exceptions.
bool supportInPlaceEdit(const QgsMapLayer *layer) const override
Checks whether this algorithm supports in-place editing on the given layer Default implementation for...
Base class for providing feedback from a processing algorithm.
virtual void pushInfo(const QString &info)
Pushes a general informational message from the algorithm.
Represents a vector layer which manages a vector based dataset.
bool isSpatial() const final
Returns true if this is a geometry layer and false in case of NoGeometry (table only) or UnknownGeome...
Q_INVOKABLE Qgis::WkbType wkbType() const final
Returns the WKBType or WKBUnknown in case of error.
static Qgis::GeometryType geometryType(Qgis::WkbType type)
Returns the geometry type for a WKB type, e.g., both MultiPolygon and CurvePolygon would have a Polyg...
static Q_INVOKABLE QString displayString(Qgis::WkbType type)
Returns a non-translated display string type for a WKB type, e.g., the geometry name used in WKT geom...
static Qgis::WkbType promoteNonPointTypesToMulti(Qgis::WkbType type)
Promotes a WKB geometry type to its multi-type equivalent, with the exception of point geometry types...
static Q_INVOKABLE bool hasM(Qgis::WkbType type)
Tests whether a WKB type contains m values.
static Qgis::WkbType flatType(Qgis::WkbType type)
Returns the flat type for a WKB type.
QList< QgsFeature > QgsFeatureList