28using namespace Qt::StringLiterals;
32QString QgsPointsToPathsAlgorithm::name()
const
34 return u
"pointstopath"_s;
37QString QgsPointsToPathsAlgorithm::displayName()
const
39 return QObject::tr(
"Points to path" );
42QString QgsPointsToPathsAlgorithm::shortHelpString()
const
44 return QObject::tr(
"This algorithm takes a point layer and connects its features to create a new line layer.\n\n"
45 "An attribute or expression may be specified to define the order the points should be connected. "
46 "If no order expression is specified, the feature ID is used.\n\n"
47 "A natural sort can be used when sorting by a string attribute "
48 "or expression (ie. place 'a9' before 'a10').\n\n"
49 "An attribute or expression can be selected to group points having the same value into the same resulting line." );
52QString QgsPointsToPathsAlgorithm::shortDescription()
const
54 return QObject::tr(
"Takes a point layer and connects its features to create a new line layer." );
62QStringList QgsPointsToPathsAlgorithm::tags()
const
64 return QObject::tr(
"create,lines,points,connect,convert,join,path" ).split(
',' );
67QString QgsPointsToPathsAlgorithm::group()
const
69 return QObject::tr(
"Vector creation" );
72QString QgsPointsToPathsAlgorithm::groupId()
const
74 return u
"vectorcreation"_s;
77void QgsPointsToPathsAlgorithm::initAlgorithm(
const QVariantMap & )
80 addParameter( std::make_unique<QgsProcessingParameterBoolean>( u
"CLOSE_PATH"_s, QObject::tr(
"Create closed paths" ),
false ) );
81 addParameter( std::make_unique<QgsProcessingParameterExpression>( u
"ORDER_EXPRESSION"_s, QObject::tr(
"Order expression" ), QVariant(), u
"INPUT"_s,
true ) );
82 addParameter( std::make_unique<QgsProcessingParameterBoolean>( u
"NATURAL_SORT"_s, QObject::tr(
"Sort text containing numbers naturally" ),
false ) );
83 addParameter( std::make_unique<QgsProcessingParameterExpression>( u
"GROUP_EXPRESSION"_s, QObject::tr(
"Path group expression" ), QVariant(), u
"INPUT"_s,
true ) );
86 addParameter( std::make_unique<QgsProcessingParameterFolderDestination>( u
"OUTPUT_TEXT_DIR"_s, QObject::tr(
"Directory for text output" ), QVariant(),
true,
false ) );
87 addOutput( std::make_unique<QgsProcessingOutputNumber>( u
"NUM_PATHS"_s, QObject::tr(
"Number of paths" ) ) );
93 addParameter( std::move( orderField ) );
97 addParameter( std::move( groupField ) );
99 auto dateFormat = std::make_unique<QgsProcessingParameterString>( u
"DATE_FORMAT"_s, QObject::tr(
"Date format (if order field is DateTime)" ), QVariant(),
false,
true );
101 addParameter( std::move( dateFormat ) );
104QgsPointsToPathsAlgorithm *QgsPointsToPathsAlgorithm::createInstance()
const
106 return new QgsPointsToPathsAlgorithm();
111 std::unique_ptr<QgsProcessingFeatureSource> source( parameterAsSource( parameters, u
"INPUT"_s, context ) );
115 const bool closePaths = parameterAsBool( parameters, u
"CLOSE_PATH"_s, context );
117 QString orderExpressionString = parameterAsString( parameters, u
"ORDER_EXPRESSION"_s, context );
118 const QString orderFieldString = parameterAsString( parameters, u
"ORDER_FIELD"_s, context );
119 if ( !orderFieldString.isEmpty() )
124 QString dateFormat = parameterAsString( parameters, u
"DATE_FORMAT"_s, context );
125 if ( !dateFormat.isEmpty() )
127 QVector<QPair<QString, QString>> codeMap;
128 codeMap << QPair<QString, QString>(
"%%",
"%" )
129 << QPair<QString, QString>(
"%a",
"ddd" )
130 << QPair<QString, QString>(
"%A",
"dddd" )
131 << QPair<QString, QString>(
"%w",
"" )
132 << QPair<QString, QString>(
"%d",
"dd" )
133 << QPair<QString, QString>(
"%b",
"MMM" )
134 << QPair<QString, QString>(
"%B",
"MMMM" )
135 << QPair<QString, QString>(
"%m",
"MM" )
136 << QPair<QString, QString>(
"%y",
"yy" )
137 << QPair<QString, QString>(
"%Y",
"yyyy" )
138 << QPair<QString, QString>(
"%H",
"hh" )
139 << QPair<QString, QString>(
"%I",
"hh" )
140 << QPair<QString, QString>(
"%p",
"AP" )
141 << QPair<QString, QString>(
"%M",
"mm" )
142 << QPair<QString, QString>(
"%S",
"ss" )
143 << QPair<QString, QString>(
"%f",
"zzz" )
144 << QPair<QString, QString>(
"%z",
"" )
145 << QPair<QString, QString>(
"%Z",
"" )
146 << QPair<QString, QString>(
"%j",
"" )
147 << QPair<QString, QString>(
"%U",
"" )
148 << QPair<QString, QString>(
"%W",
"" )
149 << QPair<QString, QString>(
"%c",
"" )
150 << QPair<QString, QString>(
"%x",
"" )
151 << QPair<QString, QString>(
"%X",
"" )
152 << QPair<QString, QString>(
"%G",
"yyyy" )
153 << QPair<QString, QString>(
"%u",
"" )
154 << QPair<QString, QString>(
"%V",
"" );
155 for (
const auto &pair : codeMap )
157 dateFormat.replace( pair.first, pair.second );
159 orderExpressionString = QString(
"to_datetime(%1, '%2')" ).arg( orderExpressionString ).arg( dateFormat );
162 else if ( orderExpressionString.isEmpty() )
165 orderExpressionString = QString(
"$id" );
167 QgsExpressionContext expressionContext = createExpressionContext( parameters, context, source.get() );
172 QStringList requiredFields = QStringList( orderExpression.
referencedColumns().values() );
173 orderExpression.
prepare( &expressionContext );
175 QMetaType::Type orderFieldType = QMetaType::Type::QString;
176 if ( orderExpression.
isField() )
178 const int orderFieldIndex = source->fields().indexFromName( orderExpression.
referencedColumns().values().first() );
179 orderFieldType = source->fields().field( orderFieldIndex ).type();
183 QString groupExpressionString = parameterAsString( parameters, u
"GROUP_EXPRESSION"_s, context );
185 const QString groupFieldString = parameterAsString( parameters, u
"GROUP_FIELD"_s, context );
186 if ( !groupFieldString.isEmpty() )
194 if ( !groupExpressionString.isEmpty() )
196 requiredFields.append( groupExpression.referencedColumns().values() );
197 const QgsField field = groupExpression.isField() ? source->fields().field( requiredFields.last() ) : u
"group"_s;
198 outputFields.append( field );
201 outputFields.append(
QgsField(
"end", orderFieldType ) );
203 const bool naturalSort = parameterAsBool( parameters, u
"NATURAL_SORT"_s, context );
205 collator.setNumericMode(
true );
214 std::unique_ptr<QgsFeatureSink> sink( parameterAsSink( parameters, u
"OUTPUT"_s, context, dest, outputFields, wkbType, source->sourceCrs() ) );
218 const QString textDir = parameterAsString( parameters, u
"OUTPUT_TEXT_DIR"_s, context );
219 if ( !textDir.isEmpty() && !QDir().mkpath( textDir ) )
229 QHash<QVariant, QVector<QPair<QVariant, QgsPoint>>> allPoints;
234 const double totalPoints = source->featureCount() > 0 ? 100.0 / source->featureCount() : 0;
235 long currentPoint = 0;
243 feedback->
setProgress( 0.5 * currentPoint * totalPoints );
248 const QVariant orderValue = orderExpression.
evaluate( &expressionContext );
249 const QVariant groupValue = groupExpressionString.isEmpty() ? QVariant() : groupExpression.evaluate( &expressionContext );
251 if ( !allPoints.contains( groupValue ) )
252 allPoints[groupValue] = QVector<QPair<QVariant, QgsPoint>>();
256 const QgsMultiPoint mp( *qgsgeometry_cast<const QgsMultiPoint *>( geom ) );
257 for ( auto pit = mp.const_parts_begin(); pit != mp.const_parts_end(); ++pit )
259 if ( const QgsPoint *point = qgsgeometry_cast<const QgsPoint *>( *pit ) )
261 allPoints[groupValue] << qMakePair( orderValue, *point );
269 allPoints[groupValue] << qMakePair( orderValue, *point );
278 QHashIterator<QVariant, QVector<QPair<QVariant, QgsPoint>>> hit( allPoints );
280 while ( hit.hasNext() )
287 QVector<QPair<QVariant, QgsPoint>> pairs = hit.value();
291 std::stable_sort( pairs.begin(), pairs.end(), [&collator](
const QPair<const QVariant, QgsPoint> &pair1,
const QPair<const QVariant, QgsPoint> &pair2 ) {
292 return collator.compare( pair1.first.toString(), pair2.first.toString() ) < 0;
297 std::stable_sort( pairs.begin(), pairs.end(), [](
const QPair<const QVariant, QgsPoint> &pair1,
const QPair<const QVariant, QgsPoint> &pair2 ) {
298 return qgsVariantLessThan( pair1.first, pair2.first );
303 QVector<QgsPoint> pathPoints;
304 for (
auto pit = pairs.constBegin(); pit != pairs.constEnd(); ++pit )
310 feedback->
setProgress( 50 + 0.5 * currentPoint * totalPoints );
311 pathPoints.append( pit->second );
314 if ( pathPoints.size() < 2 )
316 feedback->
pushInfo( QObject::tr(
"Skipping path with group %1 : insufficient vertices" ).arg( hit.key().toString() ) );
319 if ( closePaths && pathPoints.size() > 2 && pathPoints.constFirst() != pathPoints.constLast() )
320 pathPoints.append( pathPoints.constFirst() );
324 if ( !groupExpressionString.isEmpty() )
325 attrs.append( hit.key() );
326 attrs.append( pairs.first().first );
327 attrs.append( pairs.last().first );
333 if ( !textDir.isEmpty() )
335 const QString filename = QDir( textDir ).filePath( hit.key().toString() + QString(
".txt" ) );
336 QFile textFile( filename );
337 if ( !textFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
340 QTextStream out( &textFile );
341 out << QString(
"angle=Azimuth\n"
342 "heading=Coordinate_System\n"
343 "dist_units=Default\n"
347 .arg( pathPoints.at( 0 ).x() )
348 .arg( pathPoints.at( 0 ).y() );
350 for (
int i = 1; i < pathPoints.size(); ++i )
352 const double angle = pathPoints.at( i - 1 ).azimuth( pathPoints.at( i ) );
356 distance = da.
measureLine( pathPoints.at( i - 1 ), pathPoints.at( i ) );
362 out << QString(
"%1;%2;90\n" ).arg( angle ).arg( distance );
372 outputs.insert( u
"OUTPUT"_s, dest );
373 outputs.insert( u
"NUM_PATHS"_s, pathCount );
374 if ( !textDir.isEmpty() )
376 outputs.insert( u
"OUTPUT_TEXT_DIR"_s, textDir );
@ VectorPoint
Vector point layers.
@ VectorLine
Vector line layers.
@ RespectsEllipsoid
Algorithm respects the context's ellipsoid settings, and uses ellipsoidal based measurements.
QFlags< ProcessingAlgorithmDocumentationFlag > ProcessingAlgorithmDocumentationFlags
Flags describing algorithm behavior for documentation purposes.
@ SkipGeometryValidityChecks
Invalid geometry checks should always be skipped. This flag can be useful for algorithms which always...
WkbType
The WKB type describes the number of dimensions a geometry has.
@ Hidden
Parameter is hidden and should not be shown to users.
Abstract base class for all geometries.
Custom exception class for Coordinate Reference System related exceptions.
A general purpose distance and area calculator, capable of performing ellipsoid based calculations.
double measureLine(const QVector< QgsPointXY > &points) const
Measures the length of a line with multiple segments.
void setSourceCrs(const QgsCoordinateReferenceSystem &crs, const QgsCoordinateTransformContext &context)
Sets source spatial reference system crs.
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.
Handles parsing and evaluation of expressions (formerly called "search strings").
bool prepare(const QgsExpressionContext *context)
Gets the expression ready for evaluation - find out column indexes.
bool hasParserError() const
Returns true if an error occurred when parsing the input expression.
QString parserErrorString() const
Returns parser error.
bool isField() const
Checks whether an expression consists only of a single field reference.
QSet< QString > referencedColumns() const
Gets list of columns referenced by the expression.
static QString quotedColumnRef(QString name)
Returns a quoted column reference (in double quotes).
QVariant evaluate()
Evaluate the feature and return the result.
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.
@ 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...
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
bool hasGeometry() const
Returns true if the feature has an associated geometry.
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
bool isCanceled() const
Tells whether the operation has been canceled already.
void setProgress(double progress)
Sets the current progress for the feedback object.
Encapsulate a field in an attribute table or data source.
Container of fields for a vector layer.
bool append(const QgsField &field, Qgis::FieldOrigin origin=Qgis::FieldOrigin::Provider, int originIndex=-1)
Appends a field.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
static QgsGeometry fromPolyline(const QgsPolyline &polyline)
Creates a new LineString geometry from a list of QgsPoint points.
Point geometry type, with support for z-dimension and m-values.
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.
virtual void pushInfo(const QString &info)
Pushes a general informational message from the algorithm.
virtual void setProgressText(const QString &text)
Sets a progress report text string.
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.
static Q_INVOKABLE bool isMultiType(Qgis::WkbType type)
Returns true if the WKB type is a multi type.
double ANALYSIS_EXPORT angle(QgsPoint *p1, QgsPoint *p2, QgsPoint *p3, QgsPoint *p4)
Calculates the angle between two segments (in 2 dimension, z-values are ignored).
T qgsgeometry_cast(QgsAbstractGeometry *geom)