QGIS API Documentation 3.43.0-Master (ac54a16a525)
qgspointlocator.h
Go to the documentation of this file.
1/***************************************************************************
2 qgspointlocator.h
3 --------------------------------------
4 Date : November 2014
5 Copyright : (C) 2014 by Martin Dobias
6 Email : wonder dot sk at gmail dot com
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15
16#ifndef QGSPOINTLOCATOR_H
17#define QGSPOINTLOCATOR_H
18
19class QgsPointXY;
22class QgsRectangle;
24
25#include "qgis_core.h"
26#include "qgspointxy.h"
29#include "qgsfeatureid.h"
30#include "qgsgeometry.h"
31#include "qgsgeometryutils.h"
32#include "qgsvectorlayer.h"
33#include "qgslinestring.h"
35#include <memory>
36
37#include <QPointer>
38
45
53
61
68
75
82
84{
85 class IStorageManager;
86 class ISpatialIndex;
87}
88
102class CORE_EXPORT QgsPointLocator : public QObject
103{
104 Q_OBJECT
105 public:
106
119 const QgsRectangle *extent = nullptr );
120
121 ~QgsPointLocator() override;
122
126 QgsVectorLayer *layer() const { return mLayer; }
127
131 QgsCoordinateReferenceSystem destinationCrs() const;
132
136 const QgsRectangle *extent() const { return mExtent.get(); }
137
141 void setExtent( const QgsRectangle *extent );
142
147 void setRenderContext( const QgsRenderContext *context );
148
152 enum Type SIP_ENUM_BASETYPE( IntFlag )
153 {
154 Invalid = 0,
155 Vertex = 1 << 0,
156 Edge = 1 << 1,
157 Area = 1 << 2,
158 Centroid = 1 << 3,
159 MiddleOfSegment = 1 << 4,
160 LineEndpoint = 1 << 5,
161 All = Vertex | Edge | Area | Centroid | MiddleOfSegment
162 };
163
164 Q_DECLARE_FLAGS( Types, Type )
165
166
179 bool init( int maxFeaturesToIndex = -1, bool relaxed = false );
180
182 bool hasIndex() const;
183
184 struct Match
185 {
187 Match() = default;
188
189 Match( QgsPointLocator::Type t, QgsVectorLayer *vl, QgsFeatureId fid, double dist, const QgsPointXY &pt, int vertexIndex = 0, QgsPointXY *edgePoints = nullptr )
190 : mType( t )
191 , mDist( dist )
192 , mPoint( pt )
193 , mLayer( vl )
194 , mFid( fid )
195 , mVertexIndex( vertexIndex )
196 {
197 if ( edgePoints )
198 {
199 mEdgePoints[0] = edgePoints[0];
200 mEdgePoints[1] = edgePoints[1];
201 }
202 }
203
204 QgsPointLocator::Type type() const { return mType; }
205
206 bool isValid() const { return mType != Invalid; }
208 bool hasVertex() const { return mType == Vertex; }
210 bool hasEdge() const { return mType == Edge; }
212 bool hasCentroid() const { return mType == Centroid; }
214 bool hasArea() const { return mType == Area; }
216 bool hasMiddleSegment() const { return mType == MiddleOfSegment; }
217
223 bool hasLineEndpoint() const { return mType == LineEndpoint; }
224
229 double distance() const { return mDist; }
230
235 QgsPointXY point() const { return mPoint; }
236
238 int vertexIndex() const { return mVertexIndex; }
239
244 QgsVectorLayer *layer() const { return mLayer; }
245
249 QgsFeatureId featureId() const { return mFid; }
250
253 {
254 pt1 = mEdgePoints[0];
255 pt2 = mEdgePoints[1];
256 }
257
265 {
266 QgsPoint point;
267
268 if ( mLayer )
269 {
270 QgsPointXY snappedPoint( mPoint );
271 const QgsGeometry geom = mLayer->getGeometry( mFid );
272 QgsCoordinateTransform transform( destinationCrs, mLayer->crs(), mLayer->transformContext() );
273 if ( transform.isValid() )
274 {
275 try
276 {
277 snappedPoint = transform.transform( snappedPoint );
278 }
279 catch ( QgsCsException & )
280 {
281 QgsDebugError( QStringLiteral( "transformation to layer coordinate failed" ) );
282 }
283 }
284
285 if ( !( geom.isNull() || geom.isEmpty() ) )
286 {
287 // when snapping to a curve we need to use real geometry in order to have correct location
288 // of the snapped point, see https://github.com/qgis/QGIS/issues/53197.
289 // In other cases it is ok to use only a segment to speedup calculations.
290 if ( QgsWkbTypes::isCurvedType( geom.wkbType() ) )
291 {
292 point = QgsGeometryUtils::closestPoint( *geom.constGet(), QgsPoint( snappedPoint ) );
293 }
294 else
295 {
296 const QgsLineString line( geom.vertexAt( mVertexIndex ), geom.vertexAt( mVertexIndex + 1 ) );
297 point = QgsGeometryUtils::closestPoint( line, QgsPoint( snappedPoint ) );
298 }
299 }
300
301 if ( transform.isValid() )
302 {
303 try
304 {
306 }
307 catch ( QgsCsException & )
308 {
309 QgsDebugError( QStringLiteral( "transformation to destination coordinate failed" ) );
310 }
311 }
312 }
313
314 return point;
315 }
316
317 // TODO c++20 - replace with = default
318 bool operator==( const QgsPointLocator::Match &other ) const
319 {
320 return mType == other.mType &&
321 mDist == other.mDist &&
322 mPoint == other.mPoint &&
323 mLayer == other.mLayer &&
324 mFid == other.mFid &&
325 mVertexIndex == other.mVertexIndex &&
326 mEdgePoints[0] == other.mEdgePoints[0] &&
327 mEdgePoints[1] == other.mEdgePoints[1] &&
328 mCentroid == other.mCentroid &&
329 mMiddleOfSegment == other.mMiddleOfSegment;
330 }
331
332 protected:
333 Type mType = Invalid;
334 double mDist = 0;
336 QgsVectorLayer *mLayer = nullptr;
337 QgsFeatureId mFid = 0;
338 int mVertexIndex = 0; // e.g. vertex index
339 QgsPointXY mEdgePoints[2];
342 };
343
344#ifndef SIP_RUN
345 typedef class QList<QgsPointLocator::Match> MatchList;
346#else
347 typedef QList<QgsPointLocator::Match> MatchList;
348#endif
349
356 {
357 virtual ~MatchFilter() = default;
358 virtual bool acceptMatch( const QgsPointLocator::Match &match ) = 0;
359 };
360
361 // intersection queries
362
368 Match nearestVertex( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
369
376 Match nearestCentroid( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
377
384 Match nearestMiddleOfSegment( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
385
392 Match nearestLineEndpoints( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
393
399 Match nearestEdge( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
400
408 Match nearestArea( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
409
415 MatchList edgesInRect( const QgsRectangle &rect, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
416
421 MatchList edgesInRect( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
422
429 MatchList verticesInRect( const QgsRectangle &rect, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
430
436 MatchList verticesInRect( const QgsPointXY &point, double tolerance, QgsPointLocator::MatchFilter *filter = nullptr, bool relaxed = false );
437
438 // point-in-polygon query
439
440 // TODO: function to return just the first match?
441
449 MatchList pointInPolygon( const QgsPointXY &point, bool relaxed = false, QgsPointLocator::MatchFilter *filter = nullptr );
450
454 int cachedGeometryCount() const { return mGeoms.count(); }
455
462 bool isIndexing() const { return mIsIndexing; }
463
468 void waitForIndexingFinished();
469
470 signals:
471
477 void initFinished( bool ok );
478
479 protected:
480 bool rebuildIndex( int maxFeaturesToIndex = -1 );
481
482 protected slots:
483 void destroyIndex();
484 private slots:
485 void onInitTaskFinished();
486 void onFeatureAdded( QgsFeatureId fid );
487 void onFeatureDeleted( QgsFeatureId fid );
488 void onGeometryChanged( QgsFeatureId fid, const QgsGeometry &geom );
489 void onAttributeValueChanged( QgsFeatureId fid, int idx, const QVariant &value );
490
491 private:
492
497 bool prepare( bool relaxed );
498
500 std::unique_ptr< SpatialIndex::IStorageManager > mStorage;
501
502 QHash<QgsFeatureId, QgsGeometry *> mGeoms;
503 std::unique_ptr< SpatialIndex::ISpatialIndex > mRTree;
504
506 bool mIsEmptyLayer = false;
507
508
510 QgsCoordinateTransform mTransform;
511 QgsVectorLayer *mLayer = nullptr;
512 std::unique_ptr< QgsRectangle > mExtent;
513
514 std::unique_ptr<QgsRenderContext> mContext;
515 std::unique_ptr<QgsFeatureRenderer> mRenderer;
516 std::unique_ptr<QgsVectorLayerFeatureSource> mSource;
517 int mMaxFeaturesToIndex = -1;
518 bool mIsIndexing = false;
519 bool mIsDestroying = false;
520 QgsFeatureIds mAddedFeatures;
521 QgsFeatureIds mDeletedFeatures;
522 QPointer<QgsPointLocatorInitTask> mInitTask;
523
531 friend class QgsPointLocatorInitTask;
532 friend class TestQgsPointLocator;
536};
537
538
539#endif // QGSPOINTLOCATOR_H
@ Reverse
Reverse/inverse transform (from destination to source)
Represents a coordinate reference system (CRS).
Contains information about the context in which a coordinate transform is executed.
Handles coordinate transforms between two coordinate systems.
QgsPointXY transform(const QgsPointXY &point, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward) const
Transform the point from the source CRS to the destination CRS.
bool isValid() const
Returns true if the coordinate transform is valid, ie both the source and destination CRS have been s...
Custom exception class for Coordinate Reference System related exceptions.
Abstract base class for all 2D vector feature renderers.
static QgsPoint closestPoint(const QgsAbstractGeometry &geometry, const QgsPoint &point)
Returns the nearest point on a segment of a geometry for the specified point.
A geometry is the spatial representation of a feature.
QgsPoint vertexAt(int atVertex) const
Returns coordinates of a vertex.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
bool isEmpty() const
Returns true if the geometry is empty (eg a linestring with no vertices, or a collection with no geom...
Qgis::WkbType wkbType() const
Returns type of the geometry as a WKB type (point / linestring / polygon etc.)
Line string geometry type, with support for z-dimension and m-values.
Helper class used when traversing the index with areas - builds a list of matches.
Helper class used when traversing the index looking for centroid - builds a list of matches.
Helper class used when traversing the index looking for edges - builds a list of matches.
Helper class used when traversing the index looking for middle segment - builds a list of matches.
Helper class used when traversing the index looking for centroid - builds a list of matches.
Helper class used when traversing the index looking for edges - builds a list of matches.
Helper class used when traversing the index looking for line endpoints (start or end vertex) - builds...
Helper class used when traversing the index looking for middle segment - builds a list of matches.
Helper class used when traversing the index looking for vertices - builds a list of matches.
Helper class used when traversing the index looking for vertices - builds a list of matches.
Defines the interface for querying point locations.
int cachedGeometryCount() const
Returns how many geometries are cached in the index.
QFlags< Type > Types
QgsVectorLayer * layer() const
Gets associated layer.
class QList< QgsPointLocator::Match > MatchList
bool isIndexing() const
Returns true if the point locator is currently indexing the data.
const QgsRectangle * extent() const
Gets extent of the area point locator covers - if nullptr then it caches the whole layer.
void initFinished(bool ok)
Emitted whenever index has been built and initialization is finished.
Type
The type of a snap result or the filter type for a snap request.
Represents a 2D point.
Definition qgspointxy.h:60
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:49
void transform(const QgsCoordinateTransform &ct, Qgis::TransformDirection d=Qgis::TransformDirection::Forward, bool transformZ=false) override
Transforms the geometry using a coordinate transform.
Definition qgspoint.cpp:384
A rectangle specified with double values.
Contains information about the context of a rendering operation.
Partial snapshot of vector layer's state (only the members necessary for access to features).
Represents a vector layer which manages a vector based dataset.
static bool isCurvedType(Qgis::WkbType type)
Returns true if the WKB type is a curved type or can contain curved geometries.
#define SIP_ENUM_BASETYPE(type)
Definition qgis_sip.h:278
#define SIP_SKIP
Definition qgis_sip.h:126
#define SIP_OUT
Definition qgis_sip.h:58
QSet< QgsFeatureId > QgsFeatureIds
qint64 QgsFeatureId
64 bit feature ids negative numbers are used for uncommitted/newly added features
#define QgsDebugError(str)
Definition qgslogger.h:40
Interface that allows rejection of some matches in intersection queries (e.g.
virtual bool acceptMatch(const QgsPointLocator::Match &match)=0
virtual ~MatchFilter()=default
QgsFeatureId featureId() const
The id of the feature to which the snapped geometry belongs.
bool hasCentroid() const
Returns true if the Match is a centroid.
QgsVectorLayer * layer() const
The vector layer where the snap occurred.
QgsPoint interpolatedPoint(const QgsCoordinateReferenceSystem &destinationCrs=QgsCoordinateReferenceSystem()) const
Convenient method to return a point on an edge with linear interpolation of the Z value.
double distance() const
for vertex / edge match units depending on what class returns it (geom.cache: layer units,...
QgsPointXY point() const
for vertex / edge match coords depending on what class returns it (geom.cache: layer coords,...
bool hasEdge() const
Returns true if the Match is an edge.
void edgePoints(QgsPointXY &pt1, QgsPointXY &pt2) const
Only for a valid edge match - obtain endpoints of the edge.
bool hasArea() const
Returns true if the Match is an area.
QgsVectorLayer * mLayer
Match()=default
construct invalid match
Match(QgsPointLocator::Type t, QgsVectorLayer *vl, QgsFeatureId fid, double dist, const QgsPointXY &pt, int vertexIndex=0, QgsPointXY *edgePoints=nullptr)
QgsPointLocator::Type type() const
bool hasLineEndpoint() const
Returns true if the Match is a line endpoint (start or end vertex).
bool hasMiddleSegment() const
Returns true if the Match is the middle of a segment.
int vertexIndex() const
for vertex / edge match (first vertex of the edge)
bool hasVertex() const
Returns true if the Match is a vertex.
bool operator==(const QgsPointLocator::Match &other) const