QGIS API Documentation 3.99.0-Master (d270888f95f)
Loading...
Searching...
No Matches
qgspointcloudlayer.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgspointcloudlayer.cpp
3 --------------------
4 begin : October 2020
5 copyright : (C) 2020 by Peter Petrik
6 email : zilolv 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
18#include "qgspointcloudlayer.h"
19
21#include "qgsapplication.h"
22#include "qgseventtracing.h"
24#include "qgslogger.h"
25#include "qgsmaplayerfactory.h"
26#include "qgsmaplayerlegend.h"
27#include "qgsmessagelog.h"
28#include "qgspainting.h"
31#include "qgspointcloudindex.h"
42#include "qgsproviderregistry.h"
43#include "qgsrectangle.h"
44#include "qgsruntimeprofiler.h"
45#include "qgstaskmanager.h"
46#include "qgsthreadingutils.h"
47#include "qgsvirtualpointcloudprovider.h"
48#include "qgsxmlutils.h"
49
50#include <QString>
51#include <QUrl>
52
53#include "moc_qgspointcloudlayer.cpp"
54
55using namespace Qt::StringLiterals;
56
58 const QString &baseName,
59 const QString &providerLib,
61 : QgsMapLayer( Qgis::LayerType::PointCloud, baseName, uri )
62 , mElevationProperties( new QgsPointCloudLayerElevationProperties( this ) )
63 , mLayerOptions( options )
64{
65 if ( !uri.isEmpty() && !providerLib.isEmpty() )
66 {
67 const QgsDataProvider::ProviderOptions providerOptions { options.transformContext };
68 Qgis::DataProviderReadFlags providerFlags;
69 if ( options.loadDefaultStyle )
70 {
71 providerFlags |= Qgis::DataProviderReadFlag::LoadDefaultStyle;
72 }
73 setDataSource( uri, baseName, providerLib, providerOptions, providerFlags );
74 }
75
78 connect( undoStack(), &QUndoStack::indexChanged, this, &QgsMapLayer::layerModified );
79 connect( this, &QgsMapLayer::layerModified, this, [this] { triggerRepaint(); } );
80}
81
83{
84 if ( QgsTask *task = QgsApplication::taskManager()->task( mStatsCalculationTask ) )
85 {
86 mStatsCalculationTask = 0;
87 task->cancel();
88 task->waitForFinished();
89 }
90}
91
93{
95
96 QgsPointCloudLayer *layer = new QgsPointCloudLayer( source(), name(), mProviderKey, mLayerOptions );
97 QgsMapLayer::clone( layer );
98
99 if ( mRenderer )
100 layer->setRenderer( mRenderer->clone() );
101
102 layer->mElevationProperties = mElevationProperties->clone();
103 layer->mElevationProperties->setParent( layer );
104
105 layer->mLayerOptions = mLayerOptions;
106 layer->mSync3DRendererTo2DRenderer = mSync3DRendererTo2DRenderer;
107
108 return layer;
109}
110
112{
114
115 if ( !mDataProvider )
116 return QgsRectangle();
117
118 return mDataProvider->extent();
119}
120
122{
124
125 if ( mRenderer->type() != "extent"_L1 )
126 loadIndexesForRenderContext( rendererContext );
127
128 return new QgsPointCloudLayerRenderer( this, rendererContext );
129}
130
137
144
146{
148
149 return mDataProvider.get();
150}
151
152bool QgsPointCloudLayer::readXml( const QDomNode &layerNode, QgsReadWriteContext &context )
153{
155
156 // create provider
157 const QDomNode pkeyNode = layerNode.namedItem( u"provider"_s );
158 mProviderKey = pkeyNode.toElement().text();
159
161 {
162 const QgsDataProvider::ProviderOptions providerOptions { context.transformContext() };
164 // read extent
166 {
167 const QDomNode extentNode = layerNode.namedItem( u"extent"_s );
168 if ( !extentNode.isNull() )
169 {
170 // get the extent
171 const QgsRectangle mbr = QgsXmlUtils::readRectangle( extentNode.toElement() );
172
173 // store the extent
174 setExtent( mbr );
175 }
176 }
177
179 const QDomNode subset = layerNode.namedItem( u"subset"_s );
180 const QString subsetText = subset.toElement().text();
181 if ( !subsetText.isEmpty() )
182 setSubsetString( subsetText );
183 }
184
185 if ( !isValid() )
186 {
187 return false;
188 }
189
190 QString errorMsg;
191 if ( !readSymbology( layerNode, errorMsg, context ) )
192 return false;
193
194 readStyleManager( layerNode );
195 return true;
196}
197
198bool QgsPointCloudLayer::writeXml( QDomNode &layerNode, QDomDocument &doc, const QgsReadWriteContext &context ) const
199{
201
202 QDomElement mapLayerNode = layerNode.toElement();
203 mapLayerNode.setAttribute( u"type"_s, QgsMapLayerFactory::typeToString( Qgis::LayerType::PointCloud ) );
204
205 if ( !subsetString().isEmpty() )
206 {
207 QDomElement subset = doc.createElement( u"subset"_s );
208 const QDomText subsetText = doc.createTextNode( subsetString() );
209 subset.appendChild( subsetText );
210 layerNode.appendChild( subset );
211 }
212 if ( mDataProvider )
213 {
214 QDomElement provider = doc.createElement( u"provider"_s );
215 const QDomText providerText = doc.createTextNode( providerType() );
216 provider.appendChild( providerText );
217 layerNode.appendChild( provider );
218 }
219
220 writeStyleManager( layerNode, doc );
221
222 QString errorMsg;
223 return writeSymbology( layerNode, doc, errorMsg, context );
224}
225
226bool QgsPointCloudLayer::readSymbology( const QDomNode &node, QString &errorMessage, QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories )
227{
229
230 const QDomElement elem = node.toElement();
231
232 readCommonStyle( elem, context, categories );
233
234 readStyle( node, errorMessage, context, categories );
235
236 if ( categories.testFlag( CustomProperties ) )
237 readCustomProperties( node, u"variable"_s );
238
239 if ( categories.testFlag( Legend ) )
240 {
241 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Legend" ) );
242
243 const QDomElement legendElem = node.firstChildElement( u"legend"_s );
244 if ( QgsMapLayerLegend *l = legend(); !legendElem.isNull() )
245 {
246 l->readXml( legendElem, context );
247 }
248 }
249
250 return true;
251}
252
253bool QgsPointCloudLayer::readStyle( const QDomNode &node, QString &, QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories )
254{
256
257 bool result = true;
258
259 if ( categories.testFlag( Symbology3D ) )
260 {
261 bool ok;
262 bool sync = node.attributes().namedItem( u"sync3DRendererTo2DRenderer"_s ).nodeValue().toInt( &ok );
263 if ( ok )
265 }
266
267 if ( categories.testFlag( Symbology ) )
268 {
269 QDomElement rendererElement = node.firstChildElement( u"renderer"_s );
270 if ( !rendererElement.isNull() )
271 {
272 std::unique_ptr< QgsPointCloudRenderer > r( QgsPointCloudRenderer::load( rendererElement, context ) );
273 if ( r )
274 {
275 setRenderer( r.release() );
276 }
277 else
278 {
279 result = false;
280 }
281 }
282 // make sure layer has a renderer - if none exists, fallback to a default renderer
283 if ( !mRenderer )
284 {
286 }
287 }
288
289 if ( categories.testFlag( Symbology ) )
290 {
291 // get and set the blend mode if it exists
292 const QDomNode blendModeNode = node.namedItem( u"blendMode"_s );
293 if ( !blendModeNode.isNull() )
294 {
295 const QDomElement e = blendModeNode.toElement();
296 setBlendMode( QgsPainting::getCompositionMode( static_cast< Qgis::BlendMode >( e.text().toInt() ) ) );
297 }
298 }
299
300 // get and set the layer transparency and scale visibility if they exists
301 if ( categories.testFlag( Rendering ) )
302 {
303 const QDomNode layerOpacityNode = node.namedItem( u"layerOpacity"_s );
304 if ( !layerOpacityNode.isNull() )
305 {
306 const QDomElement e = layerOpacityNode.toElement();
307 setOpacity( e.text().toDouble() );
308 }
309
310 const bool hasScaleBasedVisibiliy { node.attributes().namedItem( u"hasScaleBasedVisibilityFlag"_s ).nodeValue() == '1' };
311 setScaleBasedVisibility( hasScaleBasedVisibiliy );
312 bool ok;
313 const double maxScale { node.attributes().namedItem( u"maxScale"_s ).nodeValue().toDouble( &ok ) };
314 if ( ok )
315 {
316 setMaximumScale( maxScale );
317 }
318 const double minScale { node.attributes().namedItem( u"minScale"_s ).nodeValue().toDouble( &ok ) };
319 if ( ok )
320 {
321 setMinimumScale( minScale );
322 }
323 }
324 return result;
325}
326
327bool QgsPointCloudLayer::writeSymbology( QDomNode &node, QDomDocument &doc, QString &errorMessage,
328 const QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories ) const
329{
331
332 Q_UNUSED( errorMessage )
333
334 QDomElement elem = node.toElement();
335 writeCommonStyle( elem, doc, context, categories );
336
337 ( void )writeStyle( node, doc, errorMessage, context, categories );
338
339 if ( categories.testFlag( Legend ) && legend() )
340 {
341 QDomElement legendElement = legend()->writeXml( doc, context );
342 if ( !legendElement.isNull() )
343 node.appendChild( legendElement );
344 }
345
346 return true;
347}
348
349bool QgsPointCloudLayer::writeStyle( QDomNode &node, QDomDocument &doc, QString &, const QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories ) const
350{
352
353 QDomElement mapLayerNode = node.toElement();
354
355 if ( categories.testFlag( Symbology3D ) )
356 {
357 mapLayerNode.setAttribute( u"sync3DRendererTo2DRenderer"_s, mSync3DRendererTo2DRenderer ? 1 : 0 );
358 }
359
360 if ( categories.testFlag( Symbology ) )
361 {
362 if ( mRenderer )
363 {
364 const QDomElement rendererElement = mRenderer->save( doc, context );
365 node.appendChild( rendererElement );
366 }
367 }
368
369 //save customproperties
370 if ( categories.testFlag( CustomProperties ) )
371 {
372 writeCustomProperties( node, doc );
373 }
374
375 if ( categories.testFlag( Symbology ) )
376 {
377 // add the blend mode field
378 QDomElement blendModeElem = doc.createElement( u"blendMode"_s );
379 const QDomText blendModeText = doc.createTextNode( QString::number( static_cast< int >( QgsPainting::getBlendModeEnum( blendMode() ) ) ) );
380 blendModeElem.appendChild( blendModeText );
381 node.appendChild( blendModeElem );
382 }
383
384 // add the layer opacity and scale visibility
385 if ( categories.testFlag( Rendering ) )
386 {
387 QDomElement layerOpacityElem = doc.createElement( u"layerOpacity"_s );
388 const QDomText layerOpacityText = doc.createTextNode( QString::number( opacity() ) );
389 layerOpacityElem.appendChild( layerOpacityText );
390 node.appendChild( layerOpacityElem );
391
392 mapLayerNode.setAttribute( u"hasScaleBasedVisibilityFlag"_s, hasScaleBasedVisibility() ? 1 : 0 );
393 mapLayerNode.setAttribute( u"maxScale"_s, maximumScale() );
394 mapLayerNode.setAttribute( u"minScale"_s, minimumScale() );
395 }
396 return true;
397}
398
400{
402
403 if ( mDataProvider )
404 mDataProvider->setTransformContext( transformContext );
406}
407
408void QgsPointCloudLayer::setDataSourcePrivate( const QString &dataSource, const QString &baseName, const QString &provider,
410{
412
413 if ( mDataProvider )
414 {
415 disconnect( mDataProvider.get(), &QgsPointCloudDataProvider::dataChanged, this, &QgsPointCloudLayer::dataChanged );
416 disconnect( mDataProvider.get(), &QgsPointCloudDataProvider::indexGenerationStateChanged, this, &QgsPointCloudLayer::onPointCloudIndexGenerationStateChanged );
417 }
418
419 setName( baseName );
420 mProviderKey = provider;
421 mDataSource = dataSource;
422
423 if ( mPreloadedProvider )
424 {
425 mDataProvider.reset( qobject_cast< QgsPointCloudDataProvider * >( mPreloadedProvider.release() ) );
426 }
427 else
428 {
429 std::unique_ptr< QgsScopedRuntimeProfile > profile;
430 if ( QgsApplication::profiler()->groupIsActive( u"projectload"_s ) )
431 profile = std::make_unique< QgsScopedRuntimeProfile >( tr( "Create %1 provider" ).arg( provider ), u"projectload"_s );
432 mDataProvider.reset( qobject_cast<QgsPointCloudDataProvider *>( QgsProviderRegistry::instance()->createProvider( provider, dataSource, options, flags ) ) );
433 }
434
435 if ( !mDataProvider )
436 {
437 QgsDebugError( u"Unable to get point cloud data provider"_s );
438 setValid( false );
439 return;
440 }
441
442 mDataProvider->setParent( this );
443 QgsDebugMsgLevel( u"Instantiated the point cloud data provider plugin"_s, 2 );
444
445 setValid( mDataProvider->isValid() );
446 if ( !isValid() )
447 {
448 QgsDebugError( u"Invalid point cloud provider plugin %1"_s.arg( QString( mDataSource.toUtf8() ) ) );
449 setError( mDataProvider->error() );
450 return;
451 }
452
453 connect( mDataProvider.get(), &QgsPointCloudDataProvider::indexGenerationStateChanged, this, &QgsPointCloudLayer::onPointCloudIndexGenerationStateChanged );
454 connect( mDataProvider.get(), &QgsPointCloudDataProvider::dataChanged, this, &QgsPointCloudLayer::dataChanged );
455
456 // Load initial extent, crs and renderer
457 setCrs( mDataProvider->crs() );
459 {
460 setExtent3D( mDataProvider->extent3D() );
461 }
462
463 bool loadDefaultStyleFlag = false;
465 {
466 loadDefaultStyleFlag = true;
467 }
468
469 if ( !mLayerOptions.skipIndexGeneration &&
470 mDataProvider &&
472 mDataProvider->pointCount() > 0 )
473 {
474 mDataProvider->generateIndex();
475 }
476
477 if ( !mLayerOptions.skipStatisticsCalculation &&
478 mDataProvider &&
480 mDataProvider->pointCount() > 0 )
481 {
482 calculateStatistics();
483 }
484
485 if ( !mRenderer || loadDefaultStyleFlag )
486 {
487 std::unique_ptr< QgsScopedRuntimeProfile > profile;
488 if ( QgsApplication::profiler()->groupIsActive( u"projectload"_s ) )
489 profile = std::make_unique< QgsScopedRuntimeProfile >( tr( "Load layer style" ), u"projectload"_s );
490
491 bool defaultLoadedFlag = false;
492
493 if ( loadDefaultStyleFlag && isSpatial() && mDataProvider->capabilities() & QgsPointCloudDataProvider::CreateRenderer )
494 {
495 // first try to create a renderer directly from the data provider
496 std::unique_ptr< QgsPointCloudRenderer > defaultRenderer( mDataProvider->createRenderer() );
497 if ( defaultRenderer )
498 {
499 defaultLoadedFlag = true;
500 setRenderer( defaultRenderer.release() );
501 }
502 }
503
504 if ( !defaultLoadedFlag && loadDefaultStyleFlag )
505 {
506 loadDefaultStyle( defaultLoadedFlag );
507 }
508
509 if ( !defaultLoadedFlag )
510 {
511 // all else failed, create default renderer
513 }
514 }
515}
516
523
524QString QgsPointCloudLayer::decodedSource( const QString &source, const QString &dataProvider, const QgsReadWriteContext &context ) const
525{
527
529}
530
531void QgsPointCloudLayer::onPointCloudIndexGenerationStateChanged( QgsPointCloudDataProvider::PointCloudIndexGenerationState state )
532{
534
535 switch ( state )
536 {
538 {
539 resetRenderer();
540 break;
541 }
543 {
544 QgsError providerError = mDataProvider->error();
545 if ( !providerError.isEmpty() )
546 {
547 setError( providerError );
548 emit raiseError( providerError.summary() );
549 }
550 break;
551 }
553 break;
554 }
555}
556
557
558QString QgsPointCloudLayer::loadDefaultStyle( bool &resultFlag )
559{
561
562 if ( mDataProvider->capabilities() & QgsPointCloudDataProvider::CreateRenderer )
563 {
564 // first try to create a renderer directly from the data provider
565 std::unique_ptr< QgsPointCloudRenderer > defaultRenderer( mDataProvider->createRenderer() );
566 if ( defaultRenderer )
567 {
568 resultFlag = true;
569 setRenderer( defaultRenderer.release() );
570 return QString();
571 }
572 }
573
574 return QgsMapLayer::loadDefaultStyle( resultFlag );
575}
576
578{
580
581 const QgsLayerMetadataFormatter htmlFormatter( metadata() );
582 QString myMetadata = u"<html>\n<body>\n"_s;
583
584 myMetadata += generalHtmlMetadata();
585
586 // Begin Provider section
587 myMetadata += u"<h1>"_s + tr( "Information from provider" ) + u"</h1>\n<hr>\n"_s;
588 myMetadata += "<table class=\"list-view\">\n"_L1;
589
590 // Extent
591 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Extent" ) + u"</td><td>"_s + extent().toString() + u"</td></tr>\n"_s;
592
593 // feature count
594 QLocale locale = QLocale();
595 locale.setNumberOptions( locale.numberOptions() &= ~QLocale::NumberOption::OmitGroupSeparator );
596 const qint64 pointCount = mDataProvider ? mDataProvider->pointCount() : -1;
597 myMetadata += u"<tr><td class=\"highlight\">"_s
598 + tr( "Point count" ) + u"</td><td>"_s
599 + ( pointCount < 0 ? tr( "unknown" ) : locale.toString( static_cast<qlonglong>( pointCount ) ) )
600 + u"</td></tr>\n"_s;
601
602 if ( const QgsPointCloudDataProvider *provider = dataProvider() )
603 {
604 myMetadata += provider->htmlMetadata();
605 }
606
607 myMetadata += "</table>\n<br><br>"_L1;
608
609 // CRS
610 myMetadata += crsHtmlMetadata();
611
612 // provider metadata section
613 myMetadata += u"<h1>"_s + tr( "Metadata" ) + u"</h1>\n<hr>\n"_s + u"<table class=\"list-view\">\n"_s;
614 const QVariantMap originalMetadata = mDataProvider ? mDataProvider->originalMetadata() : QVariantMap();
615
616 if ( originalMetadata.value( u"creation_year"_s ).toInt() > 0 && originalMetadata.contains( u"creation_doy"_s ) )
617 {
618 QDate creationDate( originalMetadata.value( u"creation_year"_s ).toInt(), 1, 1 );
619 creationDate = creationDate.addDays( originalMetadata.value( u"creation_doy"_s ).toInt() );
620
621 myMetadata += u"<tr><td class=\"highlight\">"_s
622 + tr( "Creation date" ) + u"</td><td>"_s
623 + creationDate.toString( Qt::ISODate )
624 + u"</td></tr>\n"_s;
625 }
626 if ( originalMetadata.contains( u"major_version"_s ) && originalMetadata.contains( u"minor_version"_s ) )
627 {
628 myMetadata += u"<tr><td class=\"highlight\">"_s
629 + tr( "Version" ) + u"</td><td>"_s
630 + u"%1.%2"_s.arg( originalMetadata.value( u"major_version"_s ).toString(),
631 originalMetadata.value( u"minor_version"_s ).toString() )
632 + u"</td></tr>\n"_s;
633 }
634
635 if ( !originalMetadata.value( u"dataformat_id"_s ).toString().isEmpty() )
636 {
637 myMetadata += u"<tr><td class=\"highlight\">"_s
638 + tr( "Data format" ) + u"</td><td>"_s
639 + u"%1 (%2)"_s.arg( QgsPointCloudDataProvider::translatedDataFormatIds().value( originalMetadata.value( u"dataformat_id"_s ).toInt() ),
640 originalMetadata.value( u"dataformat_id"_s ).toString() ).trimmed()
641 + u"</td></tr>\n"_s;
642 }
643
644 myMetadata += u"<tr><td class=\"highlight\">"_s
645 + tr( "Scale X" ) + u"</td><td>"_s
646 + QString::number( originalMetadata.value( u"scale_x"_s ).toDouble() )
647 + u"</td></tr>\n"_s;
648 myMetadata += u"<tr><td class=\"highlight\">"_s
649 + tr( "Scale Y" ) + u"</td><td>"_s
650 + QString::number( originalMetadata.value( u"scale_y"_s ).toDouble() )
651 + u"</td></tr>\n"_s;
652 myMetadata += u"<tr><td class=\"highlight\">"_s
653 + tr( "Scale Z" ) + u"</td><td>"_s
654 + QString::number( originalMetadata.value( u"scale_z"_s ).toDouble() )
655 + u"</td></tr>\n"_s;
656
657 myMetadata += u"<tr><td class=\"highlight\">"_s
658 + tr( "Offset X" ) + u"</td><td>"_s
659 + QString::number( originalMetadata.value( u"offset_x"_s ).toDouble() )
660 + u"</td></tr>\n"_s;
661 myMetadata += u"<tr><td class=\"highlight\">"_s
662 + tr( "Offset Y" ) + u"</td><td>"_s
663 + QString::number( originalMetadata.value( u"offset_y"_s ).toDouble() )
664 + u"</td></tr>\n"_s;
665 myMetadata += u"<tr><td class=\"highlight\">"_s
666 + tr( "Offset Z" ) + u"</td><td>"_s
667 + QString::number( originalMetadata.value( u"offset_z"_s ).toDouble() )
668 + u"</td></tr>\n"_s;
669
670 if ( !originalMetadata.value( u"project_id"_s ).toString().isEmpty() )
671 {
672 myMetadata += u"<tr><td class=\"highlight\">"_s
673 + tr( "Project ID" ) + u"</td><td>"_s
674 + originalMetadata.value( u"project_id"_s ).toString()
675 + u"</td></tr>\n"_s;
676 }
677
678 if ( !originalMetadata.value( u"system_id"_s ).toString().isEmpty() )
679 {
680 myMetadata += u"<tr><td class=\"highlight\">"_s
681 + tr( "System ID" ) + u"</td><td>"_s
682 + originalMetadata.value( u"system_id"_s ).toString()
683 + u"</td></tr>\n"_s;
684 }
685
686 if ( !originalMetadata.value( u"software_id"_s ).toString().isEmpty() )
687 {
688 myMetadata += u"<tr><td class=\"highlight\">"_s
689 + tr( "Software ID" ) + u"</td><td>"_s
690 + originalMetadata.value( u"software_id"_s ).toString()
691 + u"</td></tr>\n"_s;
692 }
693
694 // End Provider section
695 myMetadata += "</table>\n<br><br>"_L1;
696
697 // identification section
698 myMetadata += u"<h1>"_s + tr( "Identification" ) + u"</h1>\n<hr>\n"_s;
699 myMetadata += htmlFormatter.identificationSectionHtml( );
700 myMetadata += "<br><br>\n"_L1;
701
702 // extent section
703 myMetadata += u"<h1>"_s + tr( "Extent" ) + u"</h1>\n<hr>\n"_s;
704 myMetadata += htmlFormatter.extentSectionHtml( isSpatial() );
705 myMetadata += "<br><br>\n"_L1;
706
707 // Start the Access section
708 myMetadata += u"<h1>"_s + tr( "Access" ) + u"</h1>\n<hr>\n"_s;
709 myMetadata += htmlFormatter.accessSectionHtml( );
710 myMetadata += "<br><br>\n"_L1;
711
712 // Attributes section
713 myMetadata += u"<h1>"_s + tr( "Attributes" ) + u"</h1>\n<hr>\n<table class=\"list-view\">\n"_s;
714
716
717 // count attributes
718 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Count" ) + u"</td><td>"_s + QString::number( attrs.count() ) + u"</td></tr>\n"_s;
719
720 myMetadata += "</table>\n<br><table width=\"100%\" class=\"tabular-view\">\n"_L1;
721 myMetadata += "<tr><th>"_L1 + tr( "Attribute" ) + "</th><th>"_L1 + tr( "Type" ) + "</th></tr>\n"_L1;
722
723 for ( int i = 0; i < attrs.count(); ++i )
724 {
725 const QgsPointCloudAttribute attribute = attrs.at( i );
726 QString rowClass;
727 if ( i % 2 )
728 rowClass = u"class=\"odd-row\""_s;
729 myMetadata += "<tr "_L1 + rowClass + "><td>"_L1 + attribute.name() + "</td><td>"_L1 + attribute.displayType() + "</td></tr>\n"_L1;
730 }
731
732 //close field list
733 myMetadata += "</table>\n<br><br>"_L1;
734
735
736 // Start the contacts section
737 myMetadata += u"<h1>"_s + tr( "Contacts" ) + u"</h1>\n<hr>\n"_s;
738 myMetadata += htmlFormatter.contactsSectionHtml( );
739 myMetadata += "<br><br>\n"_L1;
740
741 // Start the links section
742 myMetadata += u"<h1>"_s + tr( "Links" ) + u"</h1>\n<hr>\n"_s;
743 myMetadata += htmlFormatter.linksSectionHtml( );
744 myMetadata += "<br><br>\n"_L1;
745
746 // Start the history section
747 myMetadata += u"<h1>"_s + tr( "History" ) + u"</h1>\n<hr>\n"_s;
748 myMetadata += htmlFormatter.historySectionHtml( );
749 myMetadata += "<br><br>\n"_L1;
750
751 myMetadata += customPropertyHtmlMetadata();
752
753 myMetadata += "\n</body>\n</html>\n"_L1;
754 return myMetadata;
755}
756
763
765{
767
768 return mDataProvider ? mDataProvider->attributes() : QgsPointCloudAttributeCollection();
769}
770
772{
774
775 return mDataProvider ? mDataProvider->pointCount() : 0;
776}
777
784
786{
788
789 return mRenderer.get();
790}
791
793{
795
796 if ( renderer == mRenderer.get() )
797 return;
798
799 mRenderer.reset( renderer );
800 emit rendererChanged();
802
803 if ( mSync3DRendererTo2DRenderer )
805}
806
807bool QgsPointCloudLayer::setSubsetString( const QString &subset )
808{
810
811 if ( !isValid() || !mDataProvider )
812 {
813 QgsDebugMsgLevel( u"invoked with invalid layer or null mDataProvider"_s, 3 );
814 setCustomProperty( u"storedSubsetString"_s, subset );
815 return false;
816 }
817 else if ( subset == mDataProvider->subsetString() )
818 return true;
819
820 bool res = mDataProvider->setSubsetString( subset );
821 if ( res )
822 {
823 emit subsetStringChanged();
825 }
826 return res;
827}
828
830{
832
833 if ( !isValid() || !mDataProvider )
834 {
835 QgsDebugMsgLevel( u"invoked with invalid layer or null mDataProvider"_s, 3 );
836 return customProperty( u"storedSubsetString"_s ).toString();
837 }
838 return mDataProvider->subsetString();
839}
840
842{
844
845 bool result = false;
847 {
848 std::unique_ptr< QgsAbstractPointCloud3DRenderer > newRenderer( static_cast< QgsAbstractPointCloud3DRenderer * >( r->clone() ) );
849 result = newRenderer->convertFrom2DRenderer( renderer() );
850 setRenderer3D( newRenderer.release() );
852 }
853 return result;
854}
855
857{
859
860 return mSync3DRendererTo2DRenderer;
861}
862
864{
866
867 mSync3DRendererTo2DRenderer = sync;
868 if ( sync )
870}
871
872void QgsPointCloudLayer::calculateStatistics()
873{
875
876 if ( !mDataProvider.get() || !mDataProvider->hasValidIndex() )
877 {
878 QgsMessageLog::logMessage( QObject::tr( "Failed to calculate statistics of the point cloud %1" ).arg( this->name() ) );
879 return;
880 }
881 if ( mStatsCalculationTask )
882 {
883 QgsMessageLog::logMessage( QObject::tr( "A statistics calculation task for the point cloud %1 is already in progress" ).arg( this->name() ) );
884 return;
885 }
886
887 QgsPointCloudStatistics indexStats = mDataProvider->metadataStatistics();
888 QList<QString> indexStatsAttributes = indexStats.statisticsMap().keys();
889 QVector<QgsPointCloudAttribute> attributes = mDataProvider->attributes().attributes();
890 // Do not calculate stats for attributes that the index gives us stats for
891 for ( int i = 0; i < attributes.size(); ++i )
892 {
893 if ( indexStatsAttributes.contains( attributes[i].name() ) )
894 {
895 attributes.remove( i );
896 --i;
897 }
898 }
899
900 // Use the layer statistics for now, until we can calculate complete ones
901 mStatistics = indexStats;
902 if ( attributes.empty() && indexStats.sampledPointsCount() > 0 )
903 {
904 // All attributes are covered by the saved stats, skip calculating anything
906 emit statisticsCalculationStateChanged( mStatisticsCalculationState );
907 resetRenderer();
908 return;
909 }
910
911 QgsPointCloudStatsCalculationTask *task = new QgsPointCloudStatsCalculationTask( mDataProvider->index(), attributes, 1000000 );
912 connect( task, &QgsTask::taskCompleted, this, [this, task, indexStats, indexStatsAttributes]()
913 {
914 mStatistics = task->calculationResults();
915
916 // Fetch what we can directly from the index
917 QMap<QString, QgsPointCloudAttributeStatistics> statsMap = mStatistics.statisticsMap();
918 for ( const QString &attribute : indexStatsAttributes )
919 {
920 statsMap[ attribute ] = indexStats.statisticsOf( attribute );
921 }
922 mStatistics = QgsPointCloudStatistics( mStatistics.sampledPointsCount(), statsMap );
923
925 emit statisticsCalculationStateChanged( mStatisticsCalculationState );
926 resetRenderer();
927 mStatsCalculationTask = 0;
928#ifdef HAVE_COPC
929 if ( mDataProvider && mDataProvider->index() && mDataProvider->index().isValid() && mDataProvider->name() == "pdal"_L1 && mStatistics.sampledPointsCount() != 0 )
930 {
931 mDataProvider->index().writeStatistics( mStatistics );
932 }
933#endif
934 } );
935
936 // In case the statistics calculation fails, QgsTask::taskTerminated will be called
937 connect( task, &QgsTask::taskTerminated, this, [this]()
938 {
939 if ( mStatsCalculationTask )
940 {
941 QgsMessageLog::logMessage( QObject::tr( "Failed to calculate statistics of the point cloud %1" ).arg( this->name() ) );
942 mStatsCalculationTask = 0;
943 }
944 } );
945
946 mStatsCalculationTask = QgsApplication::taskManager()->addTask( task );
947
949 emit statisticsCalculationStateChanged( mStatisticsCalculationState );
950}
951
952void QgsPointCloudLayer::resetRenderer()
953{
955
956 mDataProvider->loadIndex();
958 {
959 calculateStatistics();
960 }
961 if ( !mRenderer || mRenderer->type() == "extent"_L1 )
962 {
964 }
966
967 emit rendererChanged();
968}
969
970void QgsPointCloudLayer::loadIndexesForRenderContext( QgsRenderContext &rendererContext ) const
971{
972 if ( mDataProvider->capabilities() & QgsPointCloudDataProvider::ContainSubIndexes )
973 {
974 QgsRectangle renderExtent;
975 try
976 {
977 renderExtent = rendererContext.coordinateTransform().transformBoundingBox( rendererContext.mapExtent(), Qgis::TransformDirection::Reverse );
978 }
979 catch ( QgsCsException & )
980 {
981 QgsDebugError( u"Transformation of extent failed!"_s );
982 }
983
984 const QVector<QgsPointCloudSubIndex> subIndex = mDataProvider->subIndexes();
985 if ( const QgsVirtualPointCloudProvider *vpcProvider = dynamic_cast<QgsVirtualPointCloudProvider *>( mDataProvider.get() ) )
986 {
987 for ( int i = 0; i < subIndex.size(); ++i )
988 {
989 // no need to load as it's there
990 if ( subIndex.at( i ).index() )
991 continue;
992
993 if ( subIndex.at( i ).extent().intersects( renderExtent ) &&
994 ( renderExtent.width() < vpcProvider->averageSubIndexWidth() ||
995 renderExtent.height() < vpcProvider->averageSubIndexHeight() ) )
996 {
997 mDataProvider->loadSubIndex( i );
998 }
999 }
1000 }
1001 }
1002}
1003
1005{
1007 if ( mEditIndex )
1008 return false;
1009
1010 mEditIndex = QgsPointCloudIndex( new QgsPointCloudEditingIndex( this ) );
1011
1012 if ( !mEditIndex.isValid() )
1013 {
1014 mEditIndex = QgsPointCloudIndex();
1015 return false;
1016 }
1017
1018 emit editingStarted();
1019 return true;
1020}
1021
1023{
1025 if ( !mEditIndex )
1026 return false;
1027
1028 if ( mEditIndex.isModified() )
1029 {
1030 if ( !mEditIndex.commitChanges( &mCommitError ) )
1031 return false;
1032
1033 // emitting layerModified() is not required as that's done automatically
1034 // when undo stack index changes
1035 }
1036
1037 undoStack()->clear();
1038
1039 if ( stopEditing )
1040 {
1041 mEditIndex = QgsPointCloudIndex();
1042 emit editingStopped();
1043 }
1044
1045 return true;
1046}
1047
1049{
1051 return mCommitError;
1052}
1053
1055{
1057 if ( !mEditIndex )
1058 return false;
1059
1060 const QList<QgsPointCloudNodeId> updatedNodes = mEditIndex.updatedNodes();
1061
1062 undoStack()->clear();
1063
1064 mEditIndex = QgsPointCloudIndex();
1065 emit editingStopped();
1066
1067 if ( !updatedNodes.isEmpty() )
1068 {
1069 for ( const QgsPointCloudNodeId &n : updatedNodes )
1071
1072 // emitting layerModified() is not required as that's done automatically
1073 // when undo stack index changes
1074 }
1075
1076 return true;
1077}
1078
1080{
1082 return mDataProvider && mDataProvider->capabilities() & QgsPointCloudDataProvider::Capability::ChangeAttributeValues;
1083}
1084
1086{
1088 if ( mEditIndex )
1089 return true;
1090
1091 return false;
1092}
1093
1095{
1097 if ( !mEditIndex )
1098 return false;
1099
1100 return mEditIndex.isModified();
1101}
1102
1103bool QgsPointCloudLayer::changeAttributeValue( const QgsPointCloudNodeId &n, const QVector<int> &points, const QgsPointCloudAttribute &attribute, double value )
1104{
1105 return this->changeAttributeValue( { { n, points } }, attribute, value );
1106}
1107
1108bool QgsPointCloudLayer::changeAttributeValue( const QHash<QgsPointCloudNodeId, QVector<int>> &nodesAndPoints, const QgsPointCloudAttribute &attribute, double value )
1109{
1111
1112 QgsEventTracing::ScopedEvent _trace( u"PointCloud"_s, u"QgsPointCloudLayer::changeAttributeValue"_s );
1113
1114 if ( !mEditIndex )
1115 return false;
1116
1117 // Cannot allow x,y,z editing as points may get moved outside the node extents
1118 if ( attribute.name().compare( 'X'_L1, Qt::CaseInsensitive ) == 0 ||
1119 attribute.name().compare( 'Y'_L1, Qt::CaseInsensitive ) == 0 ||
1120 attribute.name().compare( 'Z'_L1, Qt::CaseInsensitive ) == 0 )
1121 return false;
1122
1123 const QgsPointCloudAttributeCollection attributeCollection = mEditIndex.attributes();
1124
1125 int attributeOffset;
1126 const QgsPointCloudAttribute *at = attributeCollection.find( attribute.name(), attributeOffset );
1127
1128 if ( !at ||
1129 at->size() != attribute.size() ||
1130 at->type() != attribute.type() )
1131 {
1132 return false;
1133 }
1134
1135 if ( !QgsPointCloudLayerEditUtils::isAttributeValueValid( attribute, value ) )
1136 {
1137 return false;
1138 }
1139
1140 for ( auto it = nodesAndPoints.constBegin(); it != nodesAndPoints.constEnd(); it++ )
1141 {
1142 QgsPointCloudNodeId n = it.key();
1143 QVector<int> points = it.value();
1144
1145 if ( !n.isValid() || !mEditIndex.hasNode( n ) ) // todo: should not have to check if n.isValid
1146 return false;
1147
1148 if ( points.isEmpty() )
1149 continue;
1150
1151 int pointsMin = std::numeric_limits<int>::max();
1152 int pointsMax = std::numeric_limits<int>::min();
1153 for ( int pt : std::as_const( points ) )
1154 {
1155 if ( pt < pointsMin )
1156 pointsMin = pt;
1157 if ( pt > pointsMax )
1158 pointsMax = pt;
1159 }
1160
1161 if ( pointsMin < 0 || pointsMax >= mEditIndex.getNode( n ).pointCount() )
1162 return false;
1163
1164 }
1165
1166 undoStack()->push( new QgsPointCloudLayerUndoCommandChangeAttribute( this, nodesAndPoints, attribute, value ) );
1167
1168 return true;
1169}
1170
1172{
1174 if ( mEditIndex )
1175 return mEditIndex;
1176
1177 if ( mDataProvider )
1178 return mDataProvider->index();
1179
1180 return QgsPointCloudIndex();
1181}
Provides global constants and enumerations for use throughout the application.
Definition qgis.h:59
BlendMode
Blending modes defining the available composition modes that can be used when painting.
Definition qgis.h:5002
QFlags< DataProviderReadFlag > DataProviderReadFlags
Flags which control data provider construction.
Definition qgis.h:505
@ PointCloud
Point cloud layer. Added in QGIS 3.18.
Definition qgis.h:200
@ LoadDefaultStyle
Reset the layer's style to the default for the datasource.
Definition qgis.h:489
@ SkipGetExtent
Skip the extent from provider.
Definition qgis.h:490
@ Reverse
Reverse/inverse transform (from destination to source).
Definition qgis.h:2731
Base class for point cloud 3D renderers.
Abstract base class for objects which generate elevation profiles.
static QgsRuntimeProfiler * profiler()
Returns the application runtime profiler.
static QgsTaskManager * taskManager()
Returns the application's task manager, used for managing application wide background task handling.
Contains information about the context in which a coordinate transform is executed.
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.
void dataChanged()
Emitted whenever a change is made to the data provider which may have caused changes in the provider'...
bool isEmpty() const
Test if no error is set.
Definition qgserror.h:112
QString summary() const
Short error description, usually the first error in chain, the real error.
Definition qgserror.cpp:133
Formats layer metadata into HTML.
Base class for storage of map layer elevation properties.
static QString typeToString(Qgis::LayerType type)
Converts a map layer type to a string value.
An abstract interface for implementations of legends for one map layer.
static QgsMapLayerLegend * defaultPointCloudLegend(QgsPointCloudLayer *layer)
Create new legend implementation for a point cloud layer.
virtual QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Writes configuration to a DOM element, to be used later with readXml().
Base class for utility classes that encapsulate information necessary for rendering of map layers.
QString name
Definition qgsmaplayer.h:87
void readStyleManager(const QDomNode &layerNode)
Read style manager's configuration (if any). To be called by subclasses.
virtual bool isSpatial() const
Returns true if the layer is considered a spatial layer, ie it has some form of geometry associated w...
QgsAbstract3DRenderer * renderer3D() const
Returns 3D renderer associated with the layer.
void setError(const QgsError &error)
Sets error message.
void writeStyleManager(QDomNode &layerNode, QDomDocument &doc) const
Write style manager's configuration (if exists). To be called by subclasses.
QgsMapLayerLegend * legend() const
Can be nullptr.
void editingStopped()
Emitted when edited changes have been successfully written to the data provider.
QString source() const
Returns the source for the layer.
Q_INVOKABLE QVariant customProperty(const QString &value, const QVariant &defaultValue=QVariant()) const
Read a custom property from layer.
QString providerType() const
Returns the provider type (provider key) for this layer.
virtual void setExtent3D(const QgsBox3D &box)
Sets the extent.
void setBlendMode(QPainter::CompositionMode blendMode)
Set the blending mode used for rendering a layer.
void configChanged()
Emitted whenever the configuration is changed.
void trigger3DUpdate()
Will advise any 3D maps that this layer requires to be updated in the scene.
void setMinimumScale(double scale)
Sets the minimum map scale (i.e.
static Qgis::DataProviderReadFlags providerReadFlags(const QDomNode &layerNode, QgsMapLayer::ReadFlags layerReadFlags)
Returns provider read flag deduced from layer read flags layerReadFlags and a dom node layerNode that...
void editingStarted()
Emitted when editing on this layer has started.
void writeCustomProperties(QDomNode &layerNode, QDomDocument &doc) const
Write custom properties to project file.
virtual QString loadDefaultStyle(bool &resultFlag)
Retrieve the default style for this layer if one exists (either as a .qml file on disk or as a record...
void setDataSource(const QString &dataSource, const QString &baseName=QString(), const QString &provider=QString(), bool loadDefaultStyleFlag=false)
Updates the data source of the layer.
QString mLayerName
Name of the layer - used for display.
void triggerRepaint(bool deferredUpdate=false)
Will advise the map canvas (and any other interested party) that this layer requires to be repainted.
QString crsHtmlMetadata() const
Returns a HTML fragment containing the layer's CRS metadata, for use in the htmlMetadata() method.
void setMaximumScale(double scale)
Sets the maximum map scale (i.e.
QgsLayerMetadata metadata
Definition qgsmaplayer.h:89
QgsMapLayer(Qgis::LayerType type=Qgis::LayerType::Vector, const QString &name=QString(), const QString &source=QString())
Constructor for QgsMapLayer.
QPainter::CompositionMode blendMode() const
Returns the current blending mode for a layer.
virtual void setOpacity(double opacity)
Sets the opacity for the layer, where opacity is a value between 0 (totally transparent) and 1....
void readCustomProperties(const QDomNode &layerNode, const QString &keyStartsWith=QString())
Read custom properties from project file.
QFlags< StyleCategory > StyleCategories
Q_INVOKABLE void setCustomProperty(const QString &key, const QVariant &value)
Set a custom property for layer.
QString mProviderKey
Data provider key (name of the data provider).
QgsCoordinateTransformContext transformContext() const
Returns the layer data provider coordinate transform context or a default transform context if the la...
QUndoStack * undoStack()
Returns pointer to layer's undo stack.
std::unique_ptr< QgsDataProvider > mPreloadedProvider
Optionally used when loading a project, it is released when the layer is effectively created.
void rendererChanged()
Signal emitted when renderer is changed.
void setScaleBasedVisibility(bool enabled)
Sets whether scale based visibility is enabled for the layer.
QgsMapLayer::LayerFlags flags
Definition qgsmaplayer.h:99
bool hasScaleBasedVisibility() const
Returns whether scale based visibility is enabled for the layer.
void emitStyleChanged()
Triggers an emission of the styleChanged() signal.
void dataChanged()
Data of layer changed.
virtual QgsMapLayer * clone() const =0
Returns a new instance equivalent to this one except for the id which is still unique.
void setName(const QString &name)
Set the display name of the layer.
virtual void setExtent(const QgsRectangle &rect)
Sets the extent.
QString mDataSource
Data source description string, varies by layer type.
@ FlagReadExtentFromXml
Read extent from xml and skip get extent from provider.
@ FlagDontResolveLayers
Don't resolve layer paths or create data providers for layers.
void setValid(bool valid)
Sets whether layer is valid or not.
void readCommonStyle(const QDomElement &layerElement, const QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories)
Read style data common to all layer types.
QgsMapLayer::ReadFlags mReadFlags
Read flags. It's up to the subclass to respect these when restoring state from XML.
double minimumScale() const
Returns the minimum map scale (i.e.
double opacity
Definition qgsmaplayer.h:95
@ Symbology
Symbology.
@ Rendering
Rendering: scale visibility, simplify method, opacity.
@ Symbology3D
3D symbology
@ CustomProperties
Custom properties (by plugins for instance).
@ Legend
Legend settings.
void layerModified()
Emitted when modifications has been done on layer.
void setRenderer3D(QgsAbstract3DRenderer *renderer)
Sets 3D renderer for the layer.
QString customPropertyHtmlMetadata() const
Returns an HTML fragment containing custom property information, for use in the htmlMetadata() method...
QString generalHtmlMetadata() const
Returns an HTML fragment containing general metadata information, for use in the htmlMetadata() metho...
void writeCommonStyle(QDomElement &layerElement, QDomDocument &document, const QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories) const
Write style data common to all layer types.
double maximumScale() const
Returns the maximum map scale (i.e.
void invalidateWgs84Extent()
Invalidates the WGS84 extent.
void setCrs(const QgsCoordinateReferenceSystem &srs, bool emitSignal=true)
Sets layer's spatial reference system.
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).
static Qgis::BlendMode getBlendModeEnum(QPainter::CompositionMode blendMode)
Returns a Qgis::BlendMode corresponding to a QPainter::CompositionMode.
static QPainter::CompositionMode getCompositionMode(Qgis::BlendMode blendMode)
Returns a QPainter::CompositionMode corresponding to a Qgis::BlendMode.
A collection of point cloud attributes.
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 count() const
Returns the number of attributes present in the collection.
QVector< QgsPointCloudAttribute > attributes() const
Returns all attributes.
Attribute for point cloud data pair of name and size in bytes.
QString displayType() const
Returns the type to use when displaying this field.
int size() const
Returns size of the attribute in bytes.
QString name() const
Returns name of the attribute.
DataType type() const
Returns the data type.
Base class for providing data for QgsPointCloudLayer.
@ CreateRenderer
Provider can create 2D renderers using backend-specific formatting information. See QgsPointCloudData...
@ ChangeAttributeValues
Provider can modify the values of point attributes.
@ ContainSubIndexes
Provider can contain multiple indexes. Virtual point cloud files for example.
static QMap< int, QString > translatedDataFormatIds()
Returns the map of LAS data format ID to translated string value.
void indexGenerationStateChanged(QgsPointCloudDataProvider::PointCloudIndexGenerationState state)
Emitted when point cloud generation state is changed.
PointCloudIndexGenerationState
Point cloud index state.
@ NotIndexed
Provider has no index available.
@ Indexing
Provider try to index the source data.
@ Indexed
The index is ready to be used.
A QgsPointCloudIndex that is used as an editing buffer when editing point cloud data.
Smart pointer for QgsAbstractPointCloudIndex.
static bool isAttributeValueValid(const QgsPointCloudAttribute &attribute, double value)
Check if value is within proper range for the attribute.
Point cloud layer specific subclass of QgsMapLayerElevationProperties.
Implementation of QgsAbstractProfileGenerator for point cloud layers.
Implementation of threaded rendering for point cloud layers.
An undo command subclass for changing point attribute values in a point cloud index.
QString decodedSource(const QString &source, const QString &dataProvider, const QgsReadWriteContext &context) const override
Called by readLayerXML(), used by derived classes to decode provider's specific data source from proj...
QString htmlMetadata() const override
Obtain a formatted HTML string containing assorted metadata for this layer.
void setSync3DRendererTo2DRenderer(bool sync)
Sets whether this layer's 3D renderer should be automatically updated with changes applied to the lay...
QgsMapLayerElevationProperties * elevationProperties() override
Returns the layer's elevation properties.
@ Calculated
The statistics calculation task is done and statistics are available.
@ NotStarted
The statistics calculation task has not been started.
@ Calculating
The statistics calculation task is running.
bool sync3DRendererTo2DRenderer() const
Returns whether this layer's 3D renderer should be automatically updated with changes applied to the ...
bool writeXml(QDomNode &layerNode, QDomDocument &doc, const QgsReadWriteContext &context) const override
Called by writeLayerXML(), used by children to write state specific to them to project files.
QgsRectangle extent() const override
Returns the extent of the layer.
bool isModified() const override
Returns true if the layer has been modified since last commit/save.
QString encodedSource(const QString &source, const QgsReadWriteContext &context) const override
Called by writeLayerXML(), used by derived classes to encode provider's specific data source to proje...
bool changeAttributeValue(const QgsPointCloudNodeId &n, const QVector< int > &points, const QgsPointCloudAttribute &attribute, double value)
Attempts to modify attribute values for specific points in the editing buffer.
QString loadDefaultStyle(bool &resultFlag) final
Retrieve the default style for this layer if one exists (either as a .qml file on disk or as a record...
QgsPointCloudRenderer * renderer()
Returns the 2D renderer for the point cloud.
bool convertRenderer3DFromRenderer2D()
Updates the layer's 3D renderer's symbol to match that of the layer's 2D renderer.
qint64 pointCount() const
Returns the total number of points available in the layer.
QgsPointCloudIndex index() const
Returns the point cloud index associated with the layer.
bool commitChanges(bool stopEditing=true)
Attempts to commit to the underlying data provider any buffered changes made since the last to call t...
bool writeSymbology(QDomNode &node, QDomDocument &doc, QString &errorMessage, const QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories) const override
Write the style for the layer into the document provided.
void raiseError(const QString &msg)
Signals an error related to this point cloud layer.
PointCloudStatisticsCalculationState statisticsCalculationState() const
Returns the status of point cloud statistics calculation.
QgsPointCloudDataProvider * dataProvider() override
Returns the layer's data provider, it may be nullptr.
bool isEditable() const override
Returns true if the layer can be edited.
void statisticsCalculationStateChanged(QgsPointCloudLayer::PointCloudStatisticsCalculationState state)
Emitted when statistics calculation state has changed.
bool readSymbology(const QDomNode &node, QString &errorMessage, QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories) override
Read the symbology for the current layer from the DOM node supplied.
bool rollBack()
Stops a current editing operation and discards any uncommitted edits.
bool writeStyle(QDomNode &node, QDomDocument &doc, QString &errorMessage, const QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories) const final
Write just the symbology information for the layer into the document.
QgsPointCloudLayer(const QString &uri=QString(), const QString &baseName=QString(), const QString &providerLib=u"pointcloud"_s, const QgsPointCloudLayer::LayerOptions &options=QgsPointCloudLayer::LayerOptions())
Constructor - creates a point cloud layer.
bool readXml(const QDomNode &layerNode, QgsReadWriteContext &context) override
Called by readLayerXML(), used by children to read state specific to them from project files.
QString subsetString() const
Returns the string used to define a subset of the layer.
void chunkAttributeValuesChanged(const QgsPointCloudNodeId &n)
Emitted when a node gets some attribute values of some points changed.
bool setSubsetString(const QString &subset)
Sets the string used to define a subset of the layer.
QString commitError() const
Returns the last error message generated when attempting to commit changes to the layer.
QgsPointCloudLayer * clone() const override
Returns a new instance equivalent to this one except for the id which is still unique.
void setRenderer(QgsPointCloudRenderer *renderer)
Sets the 2D renderer for the point cloud.
QgsPointCloudAttributeCollection attributes() const
Returns the attributes available from the layer.
void subsetStringChanged()
Emitted when the layer's subset string has changed.
void setTransformContext(const QgsCoordinateTransformContext &transformContext) override
Sets the coordinate transform context to transformContext.
bool startEditing()
Makes the layer editable.
bool supportsEditing() const override
Returns whether the layer supports editing or not.
QgsMapLayerRenderer * createMapRenderer(QgsRenderContext &rendererContext) override
Returns new instance of QgsMapLayerRenderer that will be used for rendering of given context.
bool readStyle(const QDomNode &node, QString &errorMessage, QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories) final
Read the style for the current layer from the DOM node supplied.
QgsAbstractProfileGenerator * createProfileGenerator(const QgsProfileRequest &request) override
Given a profile request, returns a new profile generator ready for generating elevation profiles.
Represents an indexed point cloud node's position in octree.
bool isValid() const
Returns whether node is valid.
static QgsPointCloudRenderer * defaultRenderer(const QgsPointCloudLayer *layer)
Returns a new default point cloud renderer for a specified layer.
Abstract base class for 2d point cloud renderers.
static QgsPointCloudRenderer * load(QDomElement &element, const QgsReadWriteContext &context)
Creates a renderer from an XML element.
int sampledPointsCount() const
Returns the number of points used to calculate the statistics.
QMap< QString, QgsPointCloudAttributeStatistics > statisticsMap() const
Returns a map object containing all the statistics.
QgsPointCloudAttributeStatistics statisticsOf(const QString &attribute) const
Returns the calculated statistics of attribute attribute.
Encapsulates properties and constraints relating to fetching elevation profiles from different source...
QString absoluteToRelativeUri(const QString &providerKey, const QString &uri, const QgsReadWriteContext &context) const
Converts absolute path(s) to relative path(s) in the given provider-specific URI.
static QgsProviderRegistry * instance(const QString &pluginPath=QString())
Means of accessing canonical single instance.
QString relativeToAbsoluteUri(const QString &providerKey, const QString &uri, const QgsReadWriteContext &context) const
Converts relative path(s) to absolute path(s) in the given provider-specific URI.
Allows entering a context category and takes care of leaving this category on deletion of the class.
A container for the context for various read/write operations on objects.
QgsReadWriteContextCategoryPopper enterCategory(const QString &category, const QString &details=QString()) const
Push a category to the stack.
QgsCoordinateTransformContext transformContext() const
Returns data provider coordinate transform context.
A rectangle specified with double values.
Q_INVOKABLE QString toString(int precision=16) const
Returns a string representation of form xmin,ymin : xmax,ymax Coordinates will be truncated to the sp...
Contains information about the context of a rendering operation.
QgsRectangle mapExtent() const
Returns the original extent of the map being rendered.
QgsCoordinateTransform coordinateTransform() const
Returns the current coordinate transform for the context.
long addTask(QgsTask *task, int priority=0)
Adds a task to the manager.
Abstract base class for long running background tasks.
void taskCompleted()
Will be emitted by task to indicate its successful completion.
void taskTerminated()
Will be emitted by task if it has terminated for any reason other then completion (e....
static QgsRectangle readRectangle(const QDomElement &element)
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63
#define QgsDebugError(str)
Definition qgslogger.h:59
#define QGIS_PROTECT_QOBJECT_THREAD_ACCESS
Setting options for creating vector data providers.
Setting options for loading point cloud layers.
bool loadDefaultStyle
Set to true if the default layer style should be loaded.