QGIS API Documentation 4.1.0-Master (5bf3c20f3c9)
Loading...
Searching...
No Matches
qgsalgorithmjoinwithlines.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsalgorithmjoinwithlines.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 "qgsdistancearea.h"
21#include "qgslinestring.h"
22#include "qgsmultilinestring.h"
23
24#include <QString>
25
26using namespace Qt::StringLiterals;
27
29
30QString QgsJoinWithLinesAlgorithm::name() const
31{
32 return u"hublines"_s;
33}
34
35QString QgsJoinWithLinesAlgorithm::displayName() const
36{
37 return QObject::tr( "Join by lines (hub lines)" );
38}
39
40QStringList QgsJoinWithLinesAlgorithm::tags() const
41{
42 return QObject::tr( "join,connect,lines,points,hub,spoke,geodesic,great,circle" ).split( ',' );
43}
44
45QString QgsJoinWithLinesAlgorithm::group() const
46{
47 return QObject::tr( "Vector analysis" );
48}
49
50QString QgsJoinWithLinesAlgorithm::groupId() const
51{
52 return u"vectoranalysis"_s;
53}
54
55void QgsJoinWithLinesAlgorithm::initAlgorithm( const QVariantMap & )
56{
57 addParameter( new QgsProcessingParameterFeatureSource( u"HUBS"_s, QObject::tr( "Hub layer" ) ) );
58 addParameter( new QgsProcessingParameterField( u"HUB_FIELD"_s, QObject::tr( "Hub ID field" ), QVariant(), u"HUBS"_s ) );
59
60 addParameter(
61 new QgsProcessingParameterField( u"HUB_FIELDS"_s, QObject::tr( "Hub layer fields to copy (leave empty to copy all fields)" ), QVariant(), u"HUBS"_s, Qgis::ProcessingFieldParameterDataType::Any, true, true )
62 );
63
64 addParameter( new QgsProcessingParameterFeatureSource( u"SPOKES"_s, QObject::tr( "Spoke layer" ) ) );
65 addParameter( new QgsProcessingParameterField( u"SPOKE_FIELD"_s, QObject::tr( "Spoke ID field" ), QVariant(), u"SPOKES"_s ) );
66
67 addParameter(
68 new QgsProcessingParameterField( u"SPOKE_FIELDS"_s, QObject::tr( "Spoke layer fields to copy (leave empty to copy all fields)" ), QVariant(), u"SPOKES"_s, Qgis::ProcessingFieldParameterDataType::Any, true, true )
69 );
70
71 addParameter( new QgsProcessingParameterBoolean( u"GEODESIC"_s, QObject::tr( "Create geodesic lines" ), false ) );
72
73 auto distanceParam = std::make_unique<QgsProcessingParameterDistance>( u"GEODESIC_DISTANCE"_s, QObject::tr( "Distance between vertices (geodesic lines only)" ), 1000 );
74 distanceParam->setFlags( distanceParam->flags() | Qgis::ProcessingParameterFlag::Advanced );
75 distanceParam->setDefaultUnit( Qgis::DistanceUnit::Kilometers );
76 distanceParam->setIsDynamic( true );
77 distanceParam->setDynamicPropertyDefinition( QgsPropertyDefinition( u"Geodesic Distance"_s, QObject::tr( "Distance between vertices" ), QgsPropertyDefinition::DoublePositive ) );
78 distanceParam->setDynamicLayerParameterName( u"HUBS"_s );
79 addParameter( distanceParam.release() );
80
81 auto breakParam = std::make_unique<QgsProcessingParameterBoolean>( u"ANTIMERIDIAN_SPLIT"_s, QObject::tr( "Split lines at antimeridian (±180 degrees longitude)" ), false );
82 breakParam->setFlags( breakParam->flags() | Qgis::ProcessingParameterFlag::Advanced );
83 addParameter( breakParam.release() );
84
85 addParameter( new QgsProcessingParameterFeatureSink( u"OUTPUT"_s, QObject::tr( "Hub lines" ), Qgis::ProcessingSourceType::VectorLine ) );
86}
87
88QString QgsJoinWithLinesAlgorithm::shortHelpString() const
89{
90 return QObject::tr(
91 "This algorithm creates hub and spoke diagrams by connecting lines from points on the Spoke layer to matching points in the Hub layer.\n\n"
92 "Determination of which hub goes with each point is based on a match between the Hub ID field on the hub points and the Spoke ID field on the spoke points.\n\n"
93 "If input layers are not point layers, a point on the surface of the geometries will be taken as the connecting location.\n\n"
94 "Optionally, geodesic lines can be created, which represent the shortest path on the surface of an ellipsoid. When "
95 "geodesic mode is used, it is possible to split the created lines at the antimeridian (±180 degrees longitude), which can improve "
96 "rendering of the lines. Additionally, the distance between vertices can be specified. A smaller distance results in a denser, more "
97 "accurate line."
98 );
99}
100
101QString QgsJoinWithLinesAlgorithm::shortDescription() const
102{
103 return QObject::tr( "Creates lines joining two point layers, based on a common attribute value." );
104}
105
106Qgis::ProcessingAlgorithmDocumentationFlags QgsJoinWithLinesAlgorithm::documentationFlags() const
107{
109}
110
111QgsJoinWithLinesAlgorithm *QgsJoinWithLinesAlgorithm::createInstance() const
112{
113 return new QgsJoinWithLinesAlgorithm();
114}
115
116QVariantMap QgsJoinWithLinesAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
117{
118 if ( parameters.value( u"SPOKES"_s ) == parameters.value( u"HUBS"_s ) )
119 throw QgsProcessingException( QObject::tr( "Same layer given for both hubs and spokes" ) );
120
121 std::unique_ptr<QgsProcessingFeatureSource> hubSource( parameterAsSource( parameters, u"HUBS"_s, context ) );
122 if ( !hubSource )
123 throw QgsProcessingException( invalidSourceError( parameters, u"HUBS"_s ) );
124
125 std::unique_ptr<QgsProcessingFeatureSource> spokeSource( parameterAsSource( parameters, u"SPOKES"_s, context ) );
126 if ( !spokeSource )
127 throw QgsProcessingException( invalidSourceError( parameters, u"SPOKES"_s ) );
128
129 const QString fieldHubName = parameterAsString( parameters, u"HUB_FIELD"_s, context );
130 const int fieldHubIndex = hubSource->fields().lookupField( fieldHubName );
131 const QStringList hubFieldsToCopy = parameterAsStrings( parameters, u"HUB_FIELDS"_s, context );
132
133 const QString fieldSpokeName = parameterAsString( parameters, u"SPOKE_FIELD"_s, context );
134 const int fieldSpokeIndex = spokeSource->fields().lookupField( fieldSpokeName );
135 const QStringList spokeFieldsToCopy = parameterAsStrings( parameters, u"SPOKE_FIELDS"_s, context );
136
137 if ( fieldHubIndex < 0 || fieldSpokeIndex < 0 )
138 throw QgsProcessingException( QObject::tr( "Invalid ID field" ) );
139
140 const bool geodesic = parameterAsBoolean( parameters, u"GEODESIC"_s, context );
141 const double geodesicDistance = parameterAsDouble( parameters, u"GEODESIC_DISTANCE"_s, context ) * 1000;
142 const bool dynamicGeodesicDistance = QgsProcessingParameters::isDynamic( parameters, u"GEODESIC_DISTANCE"_s );
143 QgsExpressionContext expressionContext = createExpressionContext( parameters, context, hubSource.get() );
144 QgsProperty geodesicDistanceProperty;
145 if ( dynamicGeodesicDistance )
146 {
147 geodesicDistanceProperty = parameters.value( u"GEODESIC_DISTANCE"_s ).value<QgsProperty>();
148 }
149
150 const bool splitAntimeridian = parameterAsBoolean( parameters, u"ANTIMERIDIAN_SPLIT"_s, context );
152 da.setSourceCrs( hubSource->sourceCrs(), context.transformContext() );
153 da.setEllipsoid( context.ellipsoid() );
154
155 QgsFields hubOutFields;
156 QgsAttributeList hubFieldIndices;
157 if ( hubFieldsToCopy.empty() )
158 {
159 hubOutFields = hubSource->fields();
160 hubFieldIndices.reserve( hubOutFields.count() );
161 for ( int i = 0; i < hubOutFields.count(); ++i )
162 {
163 hubFieldIndices << i;
164 }
165 }
166 else
167 {
168 hubFieldIndices.reserve( hubOutFields.count() );
169 for ( const QString &field : hubFieldsToCopy )
170 {
171 const int index = hubSource->fields().lookupField( field );
172 if ( index >= 0 )
173 {
174 hubFieldIndices << index;
175 hubOutFields.append( hubSource->fields().at( index ) );
176 }
177 }
178 }
179
180 QgsAttributeList hubFields2Fetch = hubFieldIndices;
181 hubFields2Fetch << fieldHubIndex;
182
183 QgsFields spokeOutFields;
184 QgsAttributeList spokeFieldIndices;
185 if ( spokeFieldsToCopy.empty() )
186 {
187 spokeOutFields = spokeSource->fields();
188 spokeFieldIndices.reserve( spokeOutFields.count() );
189 for ( int i = 0; i < spokeOutFields.count(); ++i )
190 {
191 spokeFieldIndices << i;
192 }
193 }
194 else
195 {
196 for ( const QString &field : spokeFieldsToCopy )
197 {
198 const int index = spokeSource->fields().lookupField( field );
199 if ( index >= 0 )
200 {
201 spokeFieldIndices << index;
202 spokeOutFields.append( spokeSource->fields().at( index ) );
203 }
204 }
205 }
206
207 QgsAttributeList spokeFields2Fetch = spokeFieldIndices;
208 spokeFields2Fetch << fieldSpokeIndex;
209
210
211 const QgsFields fields = QgsProcessingUtils::combineFields( hubOutFields, spokeOutFields );
212
214 bool hasZ = false;
215 if ( !geodesic && ( QgsWkbTypes::hasZ( hubSource->wkbType() ) || QgsWkbTypes::hasZ( spokeSource->wkbType() ) ) )
216 {
217 outType = QgsWkbTypes::addZ( outType );
218 hasZ = true;
219 }
220 bool hasM = false;
221 if ( !geodesic && ( QgsWkbTypes::hasM( hubSource->wkbType() ) || QgsWkbTypes::hasM( spokeSource->wkbType() ) ) )
222 {
223 outType = QgsWkbTypes::addM( outType );
224 hasM = true;
225 }
226
227 QString dest;
228 std::unique_ptr<QgsFeatureSink> sink( parameterAsSink( parameters, u"OUTPUT"_s, context, dest, fields, outType, hubSource->sourceCrs(), QgsFeatureSink::RegeneratePrimaryKey ) );
229 if ( !sink )
230 throw QgsProcessingException( invalidSinkError( parameters, u"OUTPUT"_s ) );
231
232 auto getPointFromFeature = [hasZ, hasM]( const QgsFeature &feature ) -> QgsPoint {
233 QgsPoint p;
234 if ( feature.geometry().type() == Qgis::GeometryType::Point && !feature.geometry().isMultipart() )
235 p = *static_cast<const QgsPoint *>( feature.geometry().constGet() );
236 else
237 p = *static_cast<const QgsPoint *>( feature.geometry().pointOnSurface().constGet() );
238 if ( hasZ && !p.is3D() )
239 p.addZValue( 0 );
240 if ( hasM && !p.isMeasure() )
241 p.addMValue( 0 );
242 return p;
243 };
244
245 QgsFeatureIterator hubFeatures = hubSource->getFeatures( QgsFeatureRequest().setSubsetOfAttributes( hubFields2Fetch ), Qgis::ProcessingFeatureSourceFlag::SkipGeometryValidityChecks );
246 const double step = hubSource->featureCount() > 0 ? 100.0 / hubSource->featureCount() : 1;
247 int i = 0;
248 QgsFeature hubFeature;
249 while ( hubFeatures.nextFeature( hubFeature ) )
250 {
251 i++;
252 if ( feedback->isCanceled() )
253 {
254 break;
255 }
256
257 feedback->setProgress( i * step );
258
259 if ( !hubFeature.hasGeometry() )
260 continue;
261
262 const QgsPoint hubPoint = getPointFromFeature( hubFeature );
263
264 // only keep selected attributes
265 QgsAttributes hubAttributes;
266 const int attributeCount = hubFeature.attributeCount();
267 for ( int j = 0; j < attributeCount; ++j )
268 {
269 if ( !hubFieldIndices.contains( j ) )
270 continue;
271 hubAttributes << hubFeature.attribute( j );
272 }
273
274 QgsFeatureRequest spokeRequest = QgsFeatureRequest().setDestinationCrs( hubSource->sourceCrs(), context.transformContext() );
275 spokeRequest.setSubsetOfAttributes( spokeFields2Fetch );
276 spokeRequest.setFilterExpression( QgsExpression::createFieldEqualityExpression( fieldSpokeName, hubFeature.attribute( fieldHubIndex ) ) );
277
278 QgsFeatureIterator spokeFeatures = spokeSource->getFeatures( spokeRequest, Qgis::ProcessingFeatureSourceFlag::SkipGeometryValidityChecks );
279 QgsFeature spokeFeature;
280 while ( spokeFeatures.nextFeature( spokeFeature ) )
281 {
282 if ( feedback->isCanceled() )
283 {
284 break;
285 }
286 if ( !spokeFeature.hasGeometry() )
287 continue;
288
289 const QgsPoint spokePoint = getPointFromFeature( spokeFeature );
290 QgsGeometry line;
291 if ( !geodesic )
292 {
293 line = QgsGeometry( new QgsLineString( QVector<QgsPoint>() << hubPoint << spokePoint ) );
294 if ( splitAntimeridian )
295 line = da.splitGeometryAtAntimeridian( line );
296 }
297 else
298 {
299 double distance = geodesicDistance;
300 if ( dynamicGeodesicDistance )
301 {
302 expressionContext.setFeature( hubFeature );
303 distance = geodesicDistanceProperty.valueAsDouble( expressionContext, distance );
304 }
305
306 auto ml = std::make_unique<QgsMultiLineString>();
307 auto l = std::make_unique<QgsLineString>( QVector<QgsPoint>() << hubPoint );
308 const QVector<QVector<QgsPointXY>> points = da.geodesicLine( QgsPointXY( hubPoint ), QgsPointXY( spokePoint ), distance, splitAntimeridian );
309 QVector<QgsPointXY> points1 = points.at( 0 );
310 points1.pop_front();
311 if ( points.count() == 1 )
312 points1.pop_back();
313
314 const QgsLineString geodesicPoints( points1 );
315 l->append( &geodesicPoints );
316 if ( points.count() == 1 )
317 l->addVertex( spokePoint );
318
319 ml->addGeometry( l.release() );
320 if ( points.count() > 1 )
321 {
322 QVector<QgsPointXY> points2 = points.at( 1 );
323 points2.pop_back();
324 l = std::make_unique<QgsLineString>( points2 );
325 if ( hasZ )
326 l->addZValue( std::numeric_limits<double>::quiet_NaN() );
327 if ( hasM )
328 l->addMValue( std::numeric_limits<double>::quiet_NaN() );
329
330 l->addVertex( spokePoint );
331 ml->addGeometry( l.release() );
332 }
333 line = QgsGeometry( std::move( ml ) );
334 }
335
336 QgsFeature outFeature;
337 QgsAttributes outAttributes = hubAttributes;
338
339 // only keep selected attributes
340 QgsAttributes spokeAttributes;
341 const int attributeCount = spokeFeature.attributeCount();
342 for ( int j = 0; j < attributeCount; ++j )
343 {
344 if ( !spokeFieldIndices.contains( j ) )
345 continue;
346 spokeAttributes << spokeFeature.attribute( j );
347 }
348
349 outAttributes.append( spokeAttributes );
350 outFeature.setAttributes( outAttributes );
351 outFeature.setGeometry( line );
352 if ( !sink->addFeature( outFeature, QgsFeatureSink::FastInsert ) )
353 throw QgsProcessingException( writeFeatureError( sink.get(), parameters, u"OUTPUT"_s ) );
354 }
355 }
356 sink->finalize();
357
358 QVariantMap outputs;
359 outputs.insert( u"OUTPUT"_s, dest );
360 return outputs;
361}
362
@ VectorLine
Vector line layers.
Definition qgis.h:3649
@ Kilometers
Kilometers.
Definition qgis.h:5172
@ Point
Points.
Definition qgis.h:380
@ RegeneratesPrimaryKey
Algorithm always drops any existing primary keys or FID values and regenerates them in outputs.
Definition qgis.h:3734
QFlags< ProcessingAlgorithmDocumentationFlag > ProcessingAlgorithmDocumentationFlags
Flags describing algorithm behavior for documentation purposes.
Definition qgis.h:3745
@ SkipGeometryValidityChecks
Invalid geometry checks should always be skipped. This flag can be useful for algorithms which always...
Definition qgis.h:3828
WkbType
The WKB type describes the number of dimensions a geometry has.
Definition qgis.h:294
@ LineString
LineString.
Definition qgis.h:297
@ MultiLineString
MultiLineString.
Definition qgis.h:301
@ Advanced
Parameter is an advanced parameter which should be hidden from users by default.
Definition qgis.h:3880
bool isMeasure() const
Returns true if the geometry contains m values.
bool is3D() const
Returns true if the geometry is 3D and contains a z-value.
A vector of attributes.
A general purpose distance and area calculator, capable of performing ellipsoid based calculations.
QVector< QVector< QgsPointXY > > geodesicLine(const QgsPointXY &p1, const QgsPointXY &p2, double interval, bool breakLine=false) const
Calculates the geodesic line between p1 and p2, which represents the shortest path on the ellipsoid b...
void setSourceCrs(const QgsCoordinateReferenceSystem &crs, const QgsCoordinateTransformContext &context)
Sets source spatial reference system crs.
QgsGeometry splitGeometryAtAntimeridian(const QgsGeometry &geometry) const
Splits a (Multi)LineString geometry at the antimeridian (longitude +/- 180 degrees).
bool setEllipsoid(const QString &ellipsoid)
Sets the ellipsoid by its acronym.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
void setFeature(const QgsFeature &feature)
Convenience function for setting a feature for the context.
static QString createFieldEqualityExpression(const QString &fieldName, const QVariant &value, QMetaType::Type fieldType=QMetaType::Type::UnknownType)
Create an expression allowing to evaluate if a field is equal to a value.
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).
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
QgsFeatureRequest & setDestinationCrs(const QgsCoordinateReferenceSystem &crs, const QgsCoordinateTransformContext &context)
Sets the destination crs for feature's geometries.
QgsFeatureRequest & setFilterExpression(const QString &expression)
Set the filter expression.
@ FastInsert
Use faster inserts, at the cost of updating the passed features to reflect changes made at the provid...
@ RegeneratePrimaryKey
This flag indicates, that a primary key field cannot be guaranteed to be unique and the sink should i...
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
int attributeCount() const
Returns the number of attributes attached to the feature.
bool hasGeometry() const
Returns true if the feature has an associated geometry.
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
void setProgress(double progress)
Sets the current progress for the feedback object.
Definition qgsfeedback.h:65
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
int count
Definition qgsfields.h:50
A geometry is the spatial representation of a feature.
Line string geometry type, with support for z-dimension and m-values.
Represents a 2D point.
Definition qgspointxy.h:62
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:53
bool addMValue(double mValue=0) override
Adds a measure to the geometry, initialized to a preset value.
Definition qgspoint.cpp:599
bool addZValue(double zValue=0) override
Adds a z-dimension to the geometry, initialized to a preset value.
Definition qgspoint.cpp:588
Contains information about the context in which a processing algorithm is executed.
QgsCoordinateTransformContext transformContext() const
Returns the coordinate transform context.
QString ellipsoid() const
Returns the ellipsoid to use for distance and area calculations.
Custom exception class for processing related exceptions.
Base class for providing feedback from a processing algorithm.
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.
A vector layer or feature source field parameter for processing algorithms.
static bool isDynamic(const QVariantMap &parameters, const QString &name)
Returns true if the parameter with matching name is a dynamic parameter, and must be evaluated once f...
static QgsFields combineFields(const QgsFields &fieldsA, const QgsFields &fieldsB, const QString &fieldsBPrefix=QString())
Combines two field lists, avoiding duplicate field names (in a case-insensitive manner).
Definition for a property.
Definition qgsproperty.h:47
@ DoublePositive
Positive double value (including 0).
Definition qgsproperty.h:57
A store for object properties.
QVariant value(const QgsExpressionContext &context, const QVariant &defaultValue=QVariant(), bool *ok=nullptr) const
Calculates the current value of the property, including any transforms which are set for the property...
double valueAsDouble(const QgsExpressionContext &context, double defaultValue=0.0, bool *ok=nullptr) const
Calculates the current value of the property and interprets it as a double.
static Qgis::WkbType addM(Qgis::WkbType type)
Adds the m dimension to a WKB type and returns the new type.
static Qgis::WkbType addZ(Qgis::WkbType type)
Adds the z dimension to a WKB type and returns the new type.
static Q_INVOKABLE bool hasZ(Qgis::WkbType type)
Tests whether a WKB type contains the z-dimension.
static Q_INVOKABLE bool hasM(Qgis::WkbType type)
Tests whether a WKB type contains m values.
QList< int > QgsAttributeList
Definition qgsfield.h:30