QGIS API Documentation 3.41.0-Master (02257426e5a)
Loading...
Searching...
No Matches
qgspointcloudlayerprofilegenerator.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgspointcloudlayerprofilegenerator.cpp
3 ---------------
4 begin : April 2022
5 copyright : (C) 2022 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 ***************************************************************************/
18#include "qgsprofilerequest.h"
19#include "qgscurve.h"
20#include "qgspointcloudlayer.h"
22#include "qgsgeos.h"
24#include "qgsprofilesnapping.h"
25#include "qgsprofilepoint.h"
29#include "qgsmessagelog.h"
30#include "qgsproject.h"
31
32//
33// QgsPointCloudLayerProfileGenerator
34//
35
37{
38 mPointIndex = GEOSSTRtree_create_r( QgsGeosContext::get(), ( size_t )10 );
39}
40
42{
43 GEOSSTRtree_destroy_r( QgsGeosContext::get(), mPointIndex );
44 mPointIndex = nullptr;
45}
46
48{
49 GEOSContextHandle_t geosctxt = QgsGeosContext::get();
50
51 const std::size_t size = results.size();
52 PointResult *pointData = results.data();
53 for ( std::size_t i = 0; i < size; ++i, ++pointData )
54 {
55 if ( feedback->isCanceled() )
56 break;
57
58 geos::unique_ptr geosPoint( GEOSGeom_createPointFromXY_r( geosctxt, pointData->distanceAlongCurve, pointData->z ) );
59
60 GEOSSTRtree_insert_r( geosctxt, mPointIndex, geosPoint.get(), pointData );
61 // only required for GEOS < 3.9
62 }
63}
64
66{
67 return QStringLiteral( "pointcloud" );
68}
69
71{
72 // TODO -- cache?
73 QMap< double, double > res;
74 for ( const PointResult &point : results )
75 {
76 res.insert( point.distanceAlongCurve, point.z );
77 }
78 return res;
79}
80
82{
83 // TODO -- cache?
85 res.reserve( results.size() );
86 for ( const PointResult &point : results )
87 {
88 res.append( QgsPoint( point.x, point.y, point.z ) );
89 }
90 return res;
91}
92
94{
95 // TODO -- cache?
96 QVector< QgsGeometry > res;
97 res.reserve( results.size() );
98 for ( const PointResult &point : results )
99 {
100 res.append( QgsGeometry( new QgsPoint( point.x, point.y, point.z ) ) );
101 }
102 return res;
103}
104
105QVector<QgsAbstractProfileResults::Feature> QgsPointCloudLayerProfileResults::asFeatures( Qgis::ProfileExportType type, QgsFeedback *feedback ) const
106{
107 QVector< QgsAbstractProfileResults::Feature > res;
108 res.reserve( static_cast< int >( results.size() ) );
109 switch ( type )
110 {
112 {
113 for ( const PointResult &point : results )
114 {
115 if ( feedback && feedback->isCanceled() )
116 break;
118 f.layerIdentifier = mLayerId;
119 f.geometry = QgsGeometry( std::make_unique< QgsPoint >( point.x, point.y, point.z ) );
120 res.append( f );
121 }
122 break;
123 }
124
126 {
127 for ( const PointResult &point : results )
128 {
129 if ( feedback && feedback->isCanceled() )
130 break;
132 f.layerIdentifier = mLayerId;
133 f.geometry = QgsGeometry( std::make_unique< QgsPoint >( point.distanceAlongCurve, point.z ) );
134 res.append( f );
135 }
136 break;
137 }
138
140 {
141 for ( const PointResult &point : results )
142 {
143 if ( feedback && feedback->isCanceled() )
144 break;
145
147 f.layerIdentifier = mLayerId;
148 f.attributes =
149 {
150 { QStringLiteral( "distance" ), point.distanceAlongCurve },
151 { QStringLiteral( "elevation" ), point.z }
152 };
153 f.geometry = QgsGeometry( std::make_unique< QgsPoint >( point.x, point.y, point.z ) );
154 res << f;
155 }
156 break;
157 }
158 }
159
160 return res;
161}
162
167
169{
170 QPainter *painter = context.renderContext().painter();
171 if ( !painter )
172 return;
173
174 const QgsScopedQPainterState painterState( painter );
175
176 painter->setBrush( Qt::NoBrush );
177 painter->setPen( Qt::NoPen );
178
179 switch ( pointSymbol )
180 {
182 // for square point we always disable antialiasing -- it's not critical here and we benefit from the performance boost disabling it gives
183 context.renderContext().painter()->setRenderHint( QPainter::Antialiasing, false );
184 break;
185
187 break;
188 }
189
190 const double minDistance = context.distanceRange().lower();
191 const double maxDistance = context.distanceRange().upper();
192 const double minZ = context.elevationRange().lower();
193 const double maxZ = context.elevationRange().upper();
194
195 const QRectF visibleRegion( minDistance, minZ, maxDistance - minDistance, maxZ - minZ );
196 QPainterPath clipPath;
197 clipPath.addPolygon( context.worldTransform().map( visibleRegion ) );
198 painter->setClipPath( clipPath, Qt::ClipOperation::IntersectClip );
199
200 const double penWidth = context.renderContext().convertToPainterUnits( pointSize, pointSizeUnit );
201
202 for ( const PointResult &point : std::as_const( results ) )
203 {
204 QPointF p = context.worldTransform().map( QPointF( point.distanceAlongCurve, point.z ) );
205 QColor color = respectLayerColors ? point.color : pointColor;
207 color.setAlphaF( color.alphaF() * ( 1.0 - std::pow( point.distanceFromCurve / tolerance, 0.5 ) ) );
208
209 switch ( pointSymbol )
210 {
212 painter->fillRect( QRectF( p.x() - penWidth * 0.5,
213 p.y() - penWidth * 0.5,
214 penWidth, penWidth ), color );
215 break;
216
218 painter->setBrush( QBrush( color ) );
219 painter->setPen( Qt::NoPen );
220 painter->drawEllipse( QRectF( p.x() - penWidth * 0.5,
221 p.y() - penWidth * 0.5,
222 penWidth, penWidth ) );
223 break;
224 }
225 }
226}
227
229{
230 QList< const QgsPointCloudLayerProfileResults::PointResult * > *list;
231};
232void _GEOSQueryCallback( void *item, void *userdata )
233{
234 reinterpret_cast<_GEOSQueryCallbackData *>( userdata )->list->append( reinterpret_cast<const QgsPointCloudLayerProfileResults::PointResult *>( item ) );
235}
236
238{
240
241 GEOSContextHandle_t geosctxt = QgsGeosContext::get();
242
243 const double minDistance = point.distance() - context.maximumPointDistanceDelta;
244 const double maxDistance = point.distance() + context.maximumPointDistanceDelta;
245 const double minElevation = point.elevation() - context.maximumPointElevationDelta;
246 const double maxElevation = point.elevation() + context.maximumPointElevationDelta;
247
248 GEOSCoordSequence *coord = GEOSCoordSeq_create_r( geosctxt, 2, 2 );
249 GEOSCoordSeq_setXY_r( geosctxt, coord, 0, minDistance, minElevation );
250 GEOSCoordSeq_setXY_r( geosctxt, coord, 1, maxDistance, maxElevation );
251 geos::unique_ptr searchDiagonal( GEOSGeom_createLineString_r( geosctxt, coord ) );
252
253 QList<const PointResult *> items;
254 struct _GEOSQueryCallbackData callbackData;
255 callbackData.list = &items;
256 GEOSSTRtree_query_r( geosctxt, mPointIndex, searchDiagonal.get(), _GEOSQueryCallback, &callbackData );
257 if ( items.empty() )
258 return result;
259
260 double bestMatchDistance = std::numeric_limits< double >::max();
261 const PointResult *bestMatch = nullptr;
262 for ( const PointResult *candidate : std::as_const( items ) )
263 {
264 const double distance = std::sqrt( std::pow( candidate->distanceAlongCurve - point.distance(), 2 )
265 + std::pow( ( candidate->z - point.elevation() ) / context.displayRatioElevationVsDistance, 2 ) );
266 if ( distance < bestMatchDistance )
267 {
268 bestMatchDistance = distance;
269 bestMatch = candidate;
270 }
271 }
272 if ( !bestMatch )
273 return result;
274
275 result.snappedPoint = QgsProfilePoint( bestMatch->distanceAlongCurve, bestMatch->z );
276 return result;
277}
278
279QVector<QgsProfileIdentifyResults> QgsPointCloudLayerProfileResults::identify( const QgsProfilePoint &point, const QgsProfileIdentifyContext &context )
280{
281 return identify( QgsDoubleRange( point.distance() - context.maximumPointDistanceDelta, point.distance() + context.maximumPointDistanceDelta ),
282 QgsDoubleRange( point.elevation() - context.maximumPointElevationDelta, point.elevation() + context.maximumPointElevationDelta ), context );
283}
284
285QVector<QgsProfileIdentifyResults> QgsPointCloudLayerProfileResults::identify( const QgsDoubleRange &distanceRange, const QgsDoubleRange &elevationRange, const QgsProfileIdentifyContext &context )
286{
287 if ( !mLayer )
288 return {};
289
290 std::unique_ptr< QgsCurve > substring( mProfileCurve->curveSubstring( distanceRange.lower(), distanceRange.upper() ) );
291 QgsGeos substringGeos( substring.get() );
292 std::unique_ptr< QgsAbstractGeometry > searchGeometry( substringGeos.buffer( mTolerance, 8, Qgis::EndCapStyle::Flat, Qgis::JoinStyle::Round, 2 ) );
293 if ( !searchGeometry )
294 return {};
295
296 const QgsCoordinateTransform curveToLayerTransform = QgsCoordinateTransform( mCurveCrs, mLayer->crs(), context.project ? context.project->transformContext() : QgsCoordinateTransformContext() );
297 try
298 {
299 searchGeometry->transform( curveToLayerTransform );
300 }
301 catch ( QgsCsException & )
302 {
303 return {};
304 }
305
306 // we have to adjust the elevation range to "undo" the z offset/z scale before we can hand to the provider
307 const QgsDoubleRange providerElevationRange( ( elevationRange.lower() - mZOffset ) / mZScale, ( elevationRange.upper() - mZOffset ) / mZScale );
308
309 const QgsGeometry pointCloudSearchGeometry( std::move( searchGeometry ) );
310 const QVector<QVariantMap> pointAttributes = mLayer->dataProvider()->identify( mMaxErrorInLayerCoordinates, pointCloudSearchGeometry, providerElevationRange );
311 if ( pointAttributes.empty() )
312 return {};
313
314 return { QgsProfileIdentifyResults( mLayer, pointAttributes )};
315}
316
318{
319 const QgsPointCloudLayerProfileGenerator *pcGenerator = qgis::down_cast< const QgsPointCloudLayerProfileGenerator *>( generator );
320 tolerance = pcGenerator->mTolerance;
321 pointSize = pcGenerator->mPointSize;
322 pointSizeUnit = pcGenerator->mPointSizeUnit;
323 pointSymbol = pcGenerator->mPointSymbol;
324 pointColor = pcGenerator->mPointColor;
325 respectLayerColors = static_cast< bool >( pcGenerator->mRenderer );
326 opacityByDistanceEffect = pcGenerator->mOpacityByDistanceEffect;
327
328 mLayer = pcGenerator->mLayer;
329 mLayerId = pcGenerator->mId;
330 mCurveCrs = pcGenerator->mTargetCrs;
331 mProfileCurve.reset( pcGenerator->mProfileCurve->clone() );
332 mTolerance = pcGenerator->mTolerance;
333
334 mZOffset = pcGenerator->mZOffset;
335 mZScale = pcGenerator->mZScale;
336}
337
338//
339// QgsPointCloudLayerProfileGenerator
340//
341
343 : mLayer( layer )
344 , mIndex( layer->index() )
345 , mSubIndexes( layer->dataProvider()->subIndexes() )
346 , mLayerAttributes( layer->attributes() )
347 , mRenderer( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->respectLayerColors() && mLayer->renderer() ? mLayer->renderer()->clone() : nullptr )
348 , mMaximumScreenError( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->maximumScreenError() )
349 , mMaximumScreenErrorUnit( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->maximumScreenErrorUnit() )
350 , mPointSize( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->pointSize() )
351 , mPointSizeUnit( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->pointSizeUnit() )
352 , mPointSymbol( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->pointSymbol() )
353 , mPointColor( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->pointColor() )
354 , mOpacityByDistanceEffect( qgis::down_cast< QgsPointCloudLayerElevationProperties* >( layer->elevationProperties() )->applyOpacityByDistanceEffect() )
355 , mId( layer->id() )
356 , mFeedback( std::make_unique< QgsFeedback >() )
357 , mProfileCurve( request.profileCurve() ? request.profileCurve()->clone() : nullptr )
358 , mTolerance( request.tolerance() )
359 , mSourceCrs( layer->crs3D() )
360 , mTargetCrs( request.crs() )
361 , mTransformContext( request.transformContext() )
362 , mZOffset( layer->elevationProperties()->zOffset() )
363 , mZScale( layer->elevationProperties()->zScale() )
364 , mStepDistance( request.stepDistance() )
365{
366 if ( mIndex )
367 {
368 mScale = mIndex.scale();
369 mOffset = mIndex.offset();
370 }
371}
372
374{
375 return mId;
376}
377
382
384
386{
387 mGatheredPoints.clear();
388 if ( !mLayer || !mProfileCurve || mFeedback->isCanceled() )
389 return false;
390
391 QVector<QgsPointCloudIndex> indexes;
392 if ( mIndex && mIndex.isValid() )
393 indexes.append( mIndex );
394
395 // Gather all relevant sub-indexes
396 const QgsRectangle profileCurveBbox = mProfileCurve->boundingBox();
397 for ( const QgsPointCloudSubIndex &subidx : mSubIndexes )
398 {
399 QgsPointCloudIndex index = subidx.index();
400 if ( index && index.isValid() && subidx.polygonBounds().intersects( profileCurveBbox ) )
401 indexes.append( subidx.index() );
402 }
403
404 if ( indexes.empty() )
405 return false;
406
407 const double startDistanceOffset = std::max( !context.distanceRange().isInfinite() ? context.distanceRange().lower() : 0, 0.0 );
408 const double endDistance = context.distanceRange().upper();
409
410 std::unique_ptr< QgsCurve > trimmedCurve;
411 QgsCurve *sourceCurve = nullptr;
412 if ( startDistanceOffset > 0 || endDistance < mProfileCurve->length() )
413 {
414 trimmedCurve.reset( mProfileCurve->curveSubstring( startDistanceOffset, endDistance ) );
415 sourceCurve = trimmedCurve.get();
416 }
417 else
418 {
419 sourceCurve = mProfileCurve.get();
420 }
421
422 // we need to transform the profile curve and max search extent to the layer's CRS
423 QgsGeos originalCurveGeos( sourceCurve );
424 originalCurveGeos.prepareGeometry();
425 mSearchGeometryInLayerCrs.reset( originalCurveGeos.buffer( mTolerance, 8, Qgis::EndCapStyle::Flat, Qgis::JoinStyle::Round, 2 ) );
426 mLayerToTargetTransform = QgsCoordinateTransform( mSourceCrs, mTargetCrs, mTransformContext );
427
428 try
429 {
430 mSearchGeometryInLayerCrs->transform( mLayerToTargetTransform, Qgis::TransformDirection::Reverse );
431 }
432 catch ( QgsCsException & )
433 {
434 QgsDebugError( QStringLiteral( "Error transforming profile line to layer CRS" ) );
435 return false;
436 }
437
438 if ( mFeedback->isCanceled() )
439 return false;
440
441 mSearchGeometryInLayerCrsGeometryEngine = std::make_unique< QgsGeos >( mSearchGeometryInLayerCrs.get() );
442 mSearchGeometryInLayerCrsGeometryEngine->prepareGeometry();
443 mMaxSearchExtentInLayerCrs = mSearchGeometryInLayerCrs->boundingBox();
444
445 double maximumErrorPixels = context.convertDistanceToPixels( mMaximumScreenError, mMaximumScreenErrorUnit );
446 if ( maximumErrorPixels < 0.0 )
447 {
448 QgsDebugError( QStringLiteral( "Invalid maximum error in pixels" ) );
449 return false;
450 }
451
452 const double toleranceInPixels = context.convertDistanceToPixels( mTolerance, Qgis::RenderUnit::MapUnits );
453 // ensure that the maximum error is compatible with the tolerance size -- otherwise if the tolerance size
454 // is much smaller than the maximum error, we don't dig deep enough into the point cloud nodes to find
455 // points which are inside the tolerance.
456 // "4" is a magic number here, based purely on what "looks good" in the profile results!
457 if ( toleranceInPixels / 4 < maximumErrorPixels )
458 maximumErrorPixels = toleranceInPixels / 4;
459
460 QgsPointCloudRequest request;
462 attributes.push_back( QgsPointCloudAttribute( QStringLiteral( "X" ), QgsPointCloudAttribute::Int32 ) );
463 attributes.push_back( QgsPointCloudAttribute( QStringLiteral( "Y" ), QgsPointCloudAttribute::Int32 ) );
464 attributes.push_back( QgsPointCloudAttribute( QStringLiteral( "Z" ), QgsPointCloudAttribute::Int32 ) );
465
466 if ( mRenderer )
467 {
468 mPreparedRendererData = mRenderer->prepare();
469 if ( mPreparedRendererData )
470 {
471 const QSet< QString > rendererAttributes = mPreparedRendererData->usedAttributes();
472 for ( const QString &attribute : std::as_const( rendererAttributes ) )
473 {
474 if ( attributes.indexOf( attribute ) >= 0 )
475 continue; // don't re-add attributes we are already going to fetch
476
477 const int layerIndex = mLayerAttributes.indexOf( attribute );
478 if ( layerIndex < 0 )
479 {
480 QgsMessageLog::logMessage( QObject::tr( "Required attribute %1 not found in layer" ).arg( attribute ), QObject::tr( "Point Cloud" ) );
481 continue;
482 }
483
484 attributes.push_back( mLayerAttributes.at( layerIndex ) );
485 }
486 }
487 }
488 else
489 {
490 mPreparedRendererData.reset();
491 }
492
493 request.setAttributes( attributes );
494
495 mResults = std::make_unique< QgsPointCloudLayerProfileResults >();
496 mResults->copyPropertiesFromGenerator( this );
497 mResults->mMaxErrorInLayerCoordinates = 0;
498
499 QgsCoordinateTransform extentTransform = mLayerToTargetTransform;
500 extentTransform.setBallparkTransformsAreAppropriate( true );
501
502 const double mapUnitsPerPixel = context.mapUnitsPerDistancePixel();
503 if ( mapUnitsPerPixel < 0.0 )
504 {
505 QgsDebugError( QStringLiteral( "Invalid map units per pixel ratio" ) );
506 return false;
507 }
508
509 for ( QgsPointCloudIndex pc : std::as_const( indexes ) )
510 {
511 const QgsPointCloudNode root = pc.getNode( pc.root() );
512 const QgsRectangle rootNodeExtentLayerCoords = root.bounds().toRectangle();
513 QgsRectangle rootNodeExtentInCurveCrs;
514 try
515 {
516 rootNodeExtentInCurveCrs = extentTransform.transformBoundingBox( rootNodeExtentLayerCoords );
517 }
518 catch ( QgsCsException & )
519 {
520 QgsDebugError( QStringLiteral( "Could not transform node extent to curve CRS" ) );
521 rootNodeExtentInCurveCrs = rootNodeExtentLayerCoords;
522 }
523
524 const double rootErrorInMapCoordinates = rootNodeExtentInCurveCrs.width() / pc.span(); // in curve coords
525 if ( rootErrorInMapCoordinates < 0.0 )
526 {
527 QgsDebugError( QStringLiteral( "Invalid root node error" ) );
528 return false;
529 }
530
531 double rootErrorPixels = rootErrorInMapCoordinates / mapUnitsPerPixel; // in pixels
532 const QVector<QgsPointCloudNodeId> nodes = traverseTree( pc, pc.root(), maximumErrorPixels, rootErrorPixels, context.elevationRange() );
533
534 const double rootErrorInLayerCoordinates = rootNodeExtentLayerCoords.width() / pc.span();
535 const double maxErrorInMapCoordinates = maximumErrorPixels * mapUnitsPerPixel;
536
537 mResults->mMaxErrorInLayerCoordinates = std::max(
538 mResults->mMaxErrorInLayerCoordinates,
539 maxErrorInMapCoordinates * rootErrorInLayerCoordinates / rootErrorInMapCoordinates );
540
541 switch ( pc.accessType() )
542 {
544 {
545 visitNodesSync( nodes, pc, request, context.elevationRange() );
546 break;
547 }
549 {
550 visitNodesAsync( nodes, pc, request, context.elevationRange() );
551 break;
552 }
553 }
554
555 if ( mFeedback->isCanceled() )
556 return false;
557 }
558
559 if ( mGatheredPoints.empty() )
560 {
561 mResults = nullptr;
562 return false;
563 }
564
565 // convert x/y values back to distance/height values
566
567 QString lastError;
568 const QgsPointCloudLayerProfileResults::PointResult *pointData = mGatheredPoints.constData();
569 const int size = mGatheredPoints.size();
570 mResults->results.resize( size );
571 QgsPointCloudLayerProfileResults::PointResult *destData = mResults->results.data();
572 for ( int i = 0; i < size; ++i, ++pointData, ++destData )
573 {
574 if ( mFeedback->isCanceled() )
575 return false;
576
577 *destData = *pointData;
578 destData->distanceAlongCurve = startDistanceOffset + originalCurveGeos.lineLocatePoint( destData->x, destData->y, &lastError );
579 if ( mOpacityByDistanceEffect ) // don't calculate this if we don't need it
580 destData->distanceFromCurve = originalCurveGeos.distance( destData->x, destData->y );
581
582 mResults->minZ = std::min( destData->z, mResults->minZ );
583 mResults->maxZ = std::max( destData->z, mResults->maxZ );
584 }
585 mResults->finalize( mFeedback.get() );
586
587 return true;
588}
589
594
596{
597 return mFeedback.get();
598}
599
600QVector<QgsPointCloudNodeId> QgsPointCloudLayerProfileGenerator::traverseTree( const QgsPointCloudIndex &pc, QgsPointCloudNodeId n, double maxErrorPixels, double nodeErrorPixels, const QgsDoubleRange &zRange )
601{
602 QVector<QgsPointCloudNodeId> nodes;
603
604 if ( mFeedback->isCanceled() )
605 {
606 return nodes;
607 }
608
609 QgsPointCloudNode node = pc.getNode( n );
610 QgsBox3D nodeBounds = node.bounds();
611 const QgsDoubleRange nodeZRange( nodeBounds.zMinimum(), nodeBounds.zMaximum() );
612 if ( !zRange.isInfinite() && !zRange.overlaps( nodeZRange ) )
613 return nodes;
614
615 if ( !mMaxSearchExtentInLayerCrs.intersects( nodeBounds.toRectangle() ) )
616 return nodes;
617
618 const QgsGeometry nodeMapGeometry = QgsGeometry::fromRect( nodeBounds.toRectangle() );
619 if ( !mSearchGeometryInLayerCrsGeometryEngine->intersects( nodeMapGeometry.constGet() ) )
620 return nodes;
621
622 if ( node.pointCount() > 0 )
623 nodes.append( n );
624
625 double childrenErrorPixels = nodeErrorPixels / 2.0;
626 if ( childrenErrorPixels < maxErrorPixels )
627 return nodes;
628
629 for ( const QgsPointCloudNodeId &nn : node.children() )
630 {
631 nodes += traverseTree( pc, nn, maxErrorPixels, childrenErrorPixels, zRange );
632 }
633
634 return nodes;
635}
636
637int QgsPointCloudLayerProfileGenerator::visitNodesSync( const QVector<QgsPointCloudNodeId> &nodes, QgsPointCloudIndex &pc, QgsPointCloudRequest &request, const QgsDoubleRange &zRange )
638{
639 int nodesDrawn = 0;
640 for ( const QgsPointCloudNodeId &n : nodes )
641 {
642 if ( mFeedback->isCanceled() )
643 break;
644
645 std::unique_ptr<QgsPointCloudBlock> block( pc.nodeData( n, request ) );
646
647 if ( !block )
648 continue;
649
650 visitBlock( block.get(), zRange );
651
652 ++nodesDrawn;
653 }
654 return nodesDrawn;
655}
656
657int QgsPointCloudLayerProfileGenerator::visitNodesAsync( const QVector<QgsPointCloudNodeId> &nodes, QgsPointCloudIndex &pc, QgsPointCloudRequest &request, const QgsDoubleRange &zRange )
658{
659 int nodesDrawn = 0;
660
661 // see notes about this logic in QgsPointCloudLayerRenderer::renderNodesAsync
662
663 // Async loading of nodes
664 QVector<QgsPointCloudBlockRequest *> blockRequests;
665 QEventLoop loop;
666 QObject::connect( mFeedback.get(), &QgsFeedback::canceled, &loop, &QEventLoop::quit );
667
668 for ( int i = 0; i < nodes.size(); ++i )
669 {
670 const QgsPointCloudNodeId &n = nodes[i];
671 const QString nStr = n.toString();
672 QgsPointCloudBlockRequest *blockRequest = pc.asyncNodeData( n, request );
673 blockRequests.append( blockRequest );
674 QObject::connect( blockRequest, &QgsPointCloudBlockRequest::finished, &loop,
675 [ this, &nodesDrawn, &loop, &blockRequests, &zRange, nStr, blockRequest ]()
676 {
677 blockRequests.removeOne( blockRequest );
678
679 // If all blocks are loaded, exit the event loop
680 if ( blockRequests.isEmpty() )
681 loop.exit();
682
683 std::unique_ptr<QgsPointCloudBlock> block = blockRequest->takeBlock();
684
685 blockRequest->deleteLater();
686
687 if ( mFeedback->isCanceled() )
688 {
689 return;
690 }
691
692 if ( !block )
693 {
694 QgsDebugError( QStringLiteral( "Unable to load node %1, error: %2" ).arg( nStr, blockRequest->errorStr() ) );
695 return;
696 }
697
698 visitBlock( block.get(), zRange );
699 ++nodesDrawn;
700 } );
701 }
702
703 // Wait for all point cloud nodes to finish loading
704 loop.exec();
705
706 // Generation may have got canceled and the event loop exited before finished()
707 // was called for all blocks, so let's clean up anything that is left
708 for ( QgsPointCloudBlockRequest *blockRequest : std::as_const( blockRequests ) )
709 {
710 std::unique_ptr<QgsPointCloudBlock> block = blockRequest->takeBlock();
711 block.reset();
712 blockRequest->deleteLater();
713 }
714
715 return nodesDrawn;
716}
717
718void QgsPointCloudLayerProfileGenerator::visitBlock( const QgsPointCloudBlock *block, const QgsDoubleRange &zRange )
719{
720 const char *ptr = block->data();
721 int count = block->pointCount();
722
723 const QgsPointCloudAttributeCollection request = block->attributes();
724
725 const std::size_t recordSize = request.pointRecordSize();
726
727 const QgsPointCloudAttributeCollection blockAttributes = block->attributes();
728 int xOffset = 0, yOffset = 0, zOffset = 0;
729 const QgsPointCloudAttribute::DataType xType = blockAttributes.find( QStringLiteral( "X" ), xOffset )->type();
730 const QgsPointCloudAttribute::DataType yType = blockAttributes.find( QStringLiteral( "Y" ), yOffset )->type();
731 const QgsPointCloudAttribute::DataType zType = blockAttributes.find( QStringLiteral( "Z" ), zOffset )->type();
732
733 bool useRenderer = false;
734 if ( mPreparedRendererData )
735 {
736 useRenderer = mPreparedRendererData->prepareBlock( block );
737 }
738
739 QColor color;
740 const bool reproject = !mLayerToTargetTransform.isShortCircuited();
741 for ( int i = 0; i < count; ++i )
742 {
743 if ( mFeedback->isCanceled() )
744 {
745 break;
746 }
747
749 QgsPointCloudAttribute::getPointXYZ( ptr, i, recordSize, xOffset, xType, yOffset, yType, zOffset, zType, block->scale(), block->offset(), res.x, res.y, res.z );
750
751 res.z = res.z * mZScale + mZOffset;
752 if ( !zRange.contains( res.z ) )
753 continue;
754
755 if ( useRenderer )
756 {
757 color = mPreparedRendererData->pointColor( block, i, res.z );
758 if ( !color.isValid() )
759 continue;
760
761 res.color = color.rgba();
762 }
763 else
764 {
765 res.color = mPointColor.rgba();
766 }
767
768 if ( mSearchGeometryInLayerCrsGeometryEngine->contains( res.x, res.y ) )
769 {
770 if ( reproject )
771 {
772 try
773 {
774 mLayerToTargetTransform.transformInPlace( res.x, res.y, res.z );
775 }
776 catch ( QgsCsException & )
777 {
778 continue;
779 }
780 }
781
782 mGatheredPoints.append( res );
783 }
784 }
785}
786
787
@ RespectsMaximumErrorMapUnit
Generated profile respects the QgsProfileGenerationContext::maximumErrorMapUnits() property.
@ RespectsDistanceRange
Generated profile respects the QgsProfileGenerationContext::distanceRange() property.
@ Circle
Renders points as circles.
@ Square
Renders points as squares.
QFlags< ProfileGeneratorFlag > ProfileGeneratorFlags
Definition qgis.h:3991
@ Round
Use rounded joins.
@ MapUnits
Map units.
@ Flat
Flat cap (in line with start/end of line)
@ Local
Local means the source is a local file on the machine.
@ Remote
Remote means it's loaded through a protocol like HTTP.
ProfileExportType
Types of export for elevation profiles.
Definition qgis.h:4000
@ Profile2D
Export profiles as 2D profile lines, with elevation stored in exported geometry Y dimension and dista...
@ Features3D
Export profiles as 3D features, with elevation values stored in exported geometry Z values.
@ DistanceVsElevationTable
Export profiles as a table of sampled distance vs elevation values.
@ Reverse
Reverse/inverse transform (from destination to source)
Abstract base class for objects which generate elevation profiles.
Abstract base class for storage of elevation profiles.
A 3-dimensional box composed of x, y, z coordinates.
Definition qgsbox3d.h:43
double zMaximum() const
Returns the maximum z value.
Definition qgsbox3d.h:274
QgsRectangle toRectangle() const
Converts the box to a 2D rectangle.
Definition qgsbox3d.h:394
double zMinimum() const
Returns the minimum z value.
Definition qgsbox3d.h:267
Contains information about the context in which a coordinate transform is executed.
Class for doing transforms between two map coordinate systems.
void setBallparkTransformsAreAppropriate(bool appropriate)
Sets whether approximate "ballpark" results are appropriate for this coordinate transform.
QgsPointXY transform(const QgsPointXY &point, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward) const
Transform the point from the source CRS to the destination CRS.
void transformInPlace(double &x, double &y, double &z, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward) const
Transforms an array of x, y and z double coordinates in place, from the source CRS to the destination...
bool isShortCircuited() const
Returns true if the transform short circuits because the source and destination are equivalent.
QgsRectangle transformBoundingBox(const QgsRectangle &rectangle, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward, bool handle180Crossover=false) const
Transforms a rectangle from the source CRS to the destination CRS.
Custom exception class for Coordinate Reference System related exceptions.
Abstract base class for curved geometry type.
Definition qgscurve.h:35
QgsRange which stores a range of double values.
Definition qgsrange.h:237
bool isInfinite() const
Returns true if the range consists of all possible values.
Definition qgsrange.h:291
Base class for feedback objects to be used for cancellation of something running in a worker thread.
Definition qgsfeedback.h:44
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:53
void canceled()
Internal routines can connect to this signal if they use event loop.
A geometry is the spatial representation of a feature.
static QgsGeometry fromRect(const QgsRectangle &rect)
Creates a new geometry from a QgsRectangle.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
static GEOSContextHandle_t get()
Returns a thread local instance of a GEOS context, safe for use in the current thread.
Does vector analysis using the GEOS library and handles import, export, and exception handling.
Definition qgsgeos.h:139
QgsAbstractGeometry * buffer(double distance, int segments, QString *errorMsg=nullptr) const override
Definition qgsgeos.cpp:2079
double distance(const QgsAbstractGeometry *geom, QString *errorMsg=nullptr) const override
Calculates the distance between this and geom.
Definition qgsgeos.cpp:581
void prepareGeometry() override
Prepares the geometry, so that subsequent calls to spatial relation methods are much faster.
Definition qgsgeos.cpp:289
double lineLocatePoint(const QgsPoint &point, QString *errorMsg=nullptr) const
Returns a distance representing the location along this linestring of the closest point on this lines...
Definition qgsgeos.cpp:3153
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true, const char *file=__builtin_FILE(), const char *function=__builtin_FUNCTION(), int line=__builtin_LINE())
Adds a message to the log instance (and creates it if necessary).
Collection of point cloud attributes.
void push_back(const QgsPointCloudAttribute &attribute)
Adds extra attribute.
int pointRecordSize() const
Returns total size of record.
const QgsPointCloudAttribute & at(int index) const
Returns the attribute at the specified index.
const QgsPointCloudAttribute * find(const QString &attributeName, int &offset) const
Finds the attribute with the name.
int indexOf(const QString &name) const
Returns the index of the attribute with the specified name.
Attribute for point cloud data pair of name and size in bytes.
DataType
Systems of unit measurement.
static void getPointXYZ(const char *ptr, int i, std::size_t pointRecordSize, int xOffset, QgsPointCloudAttribute::DataType xType, int yOffset, QgsPointCloudAttribute::DataType yType, int zOffset, QgsPointCloudAttribute::DataType zType, const QgsVector3D &indexScale, const QgsVector3D &indexOffset, double &x, double &y, double &z)
Retrieves the x, y, z values for the point at index i.
DataType type() const
Returns the data type.
Base class for handling loading QgsPointCloudBlock asynchronously.
QString errorStr()
Returns the error message string of the request.
void finished()
Emitted when the request processing has finished.
std::unique_ptr< QgsPointCloudBlock > takeBlock()
Returns the requested block.
Base class for storing raw data from point cloud nodes.
QgsVector3D scale() const
Returns the custom scale of the block.
const char * data() const
Returns raw pointer to data.
QgsPointCloudAttributeCollection attributes() const
Returns the attributes that are stored in the data block, along with their size.
int pointCount() const
Returns number of points that are stored in the block.
QgsVector3D offset() const
Returns the custom offset of the block.
Smart pointer for QgsAbstractPointCloudIndex.
QgsVector3D offset() const
Returns offset of data from CRS.
QgsVector3D scale() const
Returns scale of data relative to CRS.
QgsPointCloudBlockRequest * asyncNodeData(const QgsPointCloudNodeId &n, const QgsPointCloudRequest &request)
Returns a handle responsible for loading a node data block.
bool isValid() const
Returns whether index is loaded and valid.
std::unique_ptr< QgsPointCloudBlock > nodeData(const QgsPointCloudNodeId &n, const QgsPointCloudRequest &request)
Returns node data block.
QgsPointCloudNode getNode(const QgsPointCloudNodeId &id) const
Returns object for a given node.
Point cloud layer specific subclass of QgsMapLayerElevationProperties.
Implementation of QgsAbstractProfileGenerator for point cloud layers.
Qgis::ProfileGeneratorFlags flags() const override
Returns flags which reflect how the profile generator operates.
QgsPointCloudLayerProfileGenerator(QgsPointCloudLayer *layer, const QgsProfileRequest &request)
Constructor for QgsPointCloudLayerProfileGenerator.
QString sourceId() const override
Returns a unique identifier representing the source of the profile.
QgsFeedback * feedback() const override
Access to feedback object of the generator (may be nullptr)
QgsAbstractProfileResults * takeResults() override
Takes results from the generator.
bool generateProfile(const QgsProfileGenerationContext &context=QgsProfileGenerationContext()) override
Generate the profile (based on data stored in the class).
QgsProfileSnapResult snapPoint(const QgsProfilePoint &point, const QgsProfileSnapContext &context) override
Snaps a point to the generated elevation profile.
void finalize(QgsFeedback *feedback)
Finalizes results – should be called after last point is added.
QgsPointSequence sampledPoints() const override
Returns a list of sampled points, with their calculated elevation as the point z value.
QgsDoubleRange zRange() const override
Returns the range of the retrieved elevation values.
QMap< double, double > distanceToHeightMap() const override
Returns the map of distance (chainage) to height.
QVector< QgsGeometry > asGeometries() const override
Returns a list of geometries representing the calculated elevation results.
void copyPropertiesFromGenerator(const QgsAbstractProfileGenerator *generator) override
Copies properties from specified generator to the results object.
QVector< QgsProfileIdentifyResults > identify(const QgsProfilePoint &point, const QgsProfileIdentifyContext &context) override
Identify results visible at the specified profile point.
QString type() const override
Returns the unique string identifier for the results type.
void renderResults(QgsProfileRenderContext &context) override
Renders the results to the specified context.
QVector< QgsAbstractProfileResults::Feature > asFeatures(Qgis::ProfileExportType type, QgsFeedback *feedback=nullptr) const override
Returns a list of features representing the calculated elevation results.
Represents a map layer supporting display of point clouds.
Represents a indexed point cloud node's position in octree.
QString toString() const
Encode node to string.
Keeps metadata for indexed point cloud node.
qint64 pointCount() const
Returns number of points contained in node data.
QgsBox3D bounds() const
Returns node's bounding cube in CRS coords.
Point cloud data request.
void setAttributes(const QgsPointCloudAttributeCollection &attributes)
Set attributes filter in the request.
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:49
Encapsulates the context in which an elevation profile is to be generated.
double mapUnitsPerDistancePixel() const
Returns the number of map units per pixel in the distance dimension.
QgsDoubleRange elevationRange() const
Returns the range of elevations to include in the generation.
double convertDistanceToPixels(double size, Qgis::RenderUnit unit) const
Converts a distance size from the specified units to pixels.
QgsDoubleRange distanceRange() const
Returns the range of distances to include in the generation.
Encapsulates the context of identifying profile results.
double maximumPointElevationDelta
Maximum allowed snapping delta for the elevation values when identifying a point.
double maximumPointDistanceDelta
Maximum allowed snapping delta for the distance values when identifying a point.
QgsProject * project
Associated project.
Stores identify results generated by a QgsAbstractProfileResults object.
Encapsulates a point on a distance-elevation profile.
double elevation() const
Returns the elevation of the point.
double distance() const
Returns the distance of the point.
Abstract base class for storage of elevation profiles.
const QTransform & worldTransform() const
Returns the transform from world coordinates to painter coordinates.
QgsDoubleRange elevationRange() const
Returns the range of elevations to include in the render.
QgsDoubleRange distanceRange() const
Returns the range of distances to include in the render.
QgsRenderContext & renderContext()
Returns a reference to the component QgsRenderContext.
Encapsulates properties and constraints relating to fetching elevation profiles from different source...
Encapsulates the context of snapping a profile point.
double maximumPointDistanceDelta
Maximum allowed snapping delta for the distance values when snapping to a point.
double maximumPointElevationDelta
Maximum allowed snapping delta for the elevation values when snapping to a point.
double displayRatioElevationVsDistance
Display ratio of elevation vs distance units.
Encapsulates results of snapping a profile point.
QgsProfilePoint snappedPoint
Snapped point.
QgsCoordinateTransformContext transformContext
Definition qgsproject.h:113
bool overlaps(const QgsRange< T > &other) const
Returns true if this range overlaps another range.
Definition qgsrange.h:176
bool contains(const QgsRange< T > &other) const
Returns true if this range contains another range.
Definition qgsrange.h:137
T lower() const
Returns the lower bound of the range.
Definition qgsrange.h:78
T upper() const
Returns the upper bound of the range.
Definition qgsrange.h:85
A rectangle specified with double values.
bool intersects(const QgsRectangle &rect) const
Returns true when rectangle intersects with other rectangle.
double convertToPainterUnits(double size, Qgis::RenderUnit unit, const QgsMapUnitScale &scale=QgsMapUnitScale(), Qgis::RenderSubcomponentProperty property=Qgis::RenderSubcomponentProperty::Generic) const
Converts a size from the specified units to painter units (pixels).
QPainter * painter()
Returns the destination QPainter for the render operation.
Scoped object for saving and restoring a QPainter object's state.
QVector< QgsPoint > QgsPointSequence
#define QgsDebugError(str)
Definition qgslogger.h:40
void _GEOSQueryCallback(void *item, void *userdata)
const QgsCoordinateReferenceSystem & crs
Encapsulates information about a feature exported from the profile results.
QString layerIdentifier
Identifier for grouping output features.
QVariantMap attributes
Exported attributes.
QList< const QgsPointCloudLayerProfileResults::PointResult * > * list