QGIS API Documentation 3.43.0-Master (37eec98dbf6)
qgsproject.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsproject.cpp - description
3 -------------------
4 begin : July 23, 2004
5 copyright : (C) 2004 by Mark Coletti
6 email : mcoletti at gmail.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 "qgsproject.h"
19#include "moc_qgsproject.cpp"
20
21#include "qgsdatasourceuri.h"
23#include "qgslayertree.h"
24#include "qgslayertreeutils.h"
26#include "qgslogger.h"
27#include "qgsmessagelog.h"
28#include "qgsmaplayerfactory.h"
31#include "qgssnappingconfig.h"
32#include "qgspathresolver.h"
33#include "qgsprojectstorage.h"
35#include "qgsprojectversion.h"
36#include "qgsrasterlayer.h"
37#include "qgsreadwritecontext.h"
38#include "qgsrelationmanager.h"
42#include "qgslayerdefinition.h"
43#include "qgsunittypes.h"
44#include "qgstransaction.h"
45#include "qgstransactiongroup.h"
48#include "qgsmeshlayer.h"
49#include "qgslayoutmanager.h"
50#include "qgsbookmarkmanager.h"
51#include "qgsmaplayerstore.h"
52#include "qgsziputils.h"
53#include "qgsauxiliarystorage.h"
54#include "qgscolorutils.h"
55#include "qgsapplication.h"
61#include "qgsvectortilelayer.h"
62#include "qgstiledscenelayer.h"
63#include "qgsruntimeprofiler.h"
64#include "qgsannotationlayer.h"
65#include "qgspointcloudlayer.h"
67#include "qgsgrouplayer.h"
68#include "qgsmapviewsmanager.h"
72#include "qgsthreadingutils.h"
73#include "qgssensormanager.h"
74#include "qgsproviderregistry.h"
77#include "qgspluginlayer.h"
78#include "qgspythonrunner.h"
79
80#include <algorithm>
81#include <QApplication>
82#include <QFileInfo>
83#include <QDomNode>
84#include <QObject>
85#include <QTextStream>
86#include <QTemporaryFile>
87#include <QDir>
88#include <QUrl>
89#include <QStandardPaths>
90#include <QUuid>
91#include <QRegularExpression>
92#include <QThreadPool>
93
94#ifdef _MSC_VER
95#include <sys/utime.h>
96#else
97#include <utime.h>
98#endif
99
100// canonical project instance
101QgsProject *QgsProject::sProject = nullptr;
102
111QStringList makeKeyTokens_( const QString &scope, const QString &key )
112{
113 QStringList keyTokens = QStringList( scope );
114 keyTokens += key.split( '/', Qt::SkipEmptyParts );
115
116 // be sure to include the canonical root node
117 keyTokens.push_front( QStringLiteral( "properties" ) );
118
119 return keyTokens;
120}
121
122
123
133QgsProjectProperty *findKey_( const QString &scope,
134 const QString &key,
135 QgsProjectPropertyKey &rootProperty )
136{
137 QgsProjectPropertyKey *currentProperty = &rootProperty;
138 QgsProjectProperty *nextProperty; // link to next property down hierarchy
139
140 QStringList keySequence = makeKeyTokens_( scope, key );
141
142 while ( !keySequence.isEmpty() )
143 {
144 // if the current head of the sequence list matches the property name,
145 // then traverse down the property hierarchy
146 if ( keySequence.first() == currentProperty->name() )
147 {
148 // remove front key since we're traversing down a level
149 keySequence.pop_front();
150
151 if ( 1 == keySequence.count() )
152 {
153 // if we have only one key name left, then return the key found
154 return currentProperty->find( keySequence.front() );
155 }
156 else if ( keySequence.isEmpty() )
157 {
158 // if we're out of keys then the current property is the one we
159 // want; i.e., we're in the rate case of being at the top-most
160 // property node
161 return currentProperty;
162 }
163 else if ( ( nextProperty = currentProperty->find( keySequence.first() ) ) )
164 {
165 if ( nextProperty->isKey() )
166 {
167 currentProperty = static_cast<QgsProjectPropertyKey *>( nextProperty );
168 }
169 else if ( nextProperty->isValue() && 1 == keySequence.count() )
170 {
171 // it may be that this may be one of several property value
172 // nodes keyed by QDict string; if this is the last remaining
173 // key token and the next property is a value node, then
174 // that's the situation, so return the currentProperty
175 return currentProperty;
176 }
177 else
178 {
179 // QgsProjectPropertyValue not Key, so return null
180 return nullptr;
181 }
182 }
183 else
184 {
185 // if the next key down isn't found
186 // then the overall key sequence doesn't exist
187 return nullptr;
188 }
189 }
190 else
191 {
192 return nullptr;
193 }
194 }
195
196 return nullptr;
197}
198
199
200
210QgsProjectProperty *addKey_( const QString &scope,
211 const QString &key,
212 QgsProjectPropertyKey *rootProperty,
213 const QVariant &value,
214 bool &propertiesModified )
215{
216 QStringList keySequence = makeKeyTokens_( scope, key );
217
218 // cursor through property key/value hierarchy
219 QgsProjectPropertyKey *currentProperty = rootProperty;
220 QgsProjectProperty *nextProperty; // link to next property down hierarchy
221 QgsProjectPropertyKey *newPropertyKey = nullptr;
222
223 propertiesModified = false;
224 while ( ! keySequence.isEmpty() )
225 {
226 // if the current head of the sequence list matches the property name,
227 // then traverse down the property hierarchy
228 if ( keySequence.first() == currentProperty->name() )
229 {
230 // remove front key since we're traversing down a level
231 keySequence.pop_front();
232
233 // if key sequence has one last element, then we use that as the
234 // name to store the value
235 if ( 1 == keySequence.count() )
236 {
237 QgsProjectProperty *property = currentProperty->find( keySequence.front() );
238 if ( !property || property->value() != value )
239 {
240 currentProperty->setValue( keySequence.front(), value );
241 propertiesModified = true;
242 }
243
244 return currentProperty;
245 }
246 // we're at the top element if popping the keySequence element
247 // will leave it empty; in that case, just add the key
248 else if ( keySequence.isEmpty() )
249 {
250 if ( currentProperty->value() != value )
251 {
252 currentProperty->setValue( value );
253 propertiesModified = true;
254 }
255
256 return currentProperty;
257 }
258 else if ( ( nextProperty = currentProperty->find( keySequence.first() ) ) )
259 {
260 currentProperty = dynamic_cast<QgsProjectPropertyKey *>( nextProperty );
261
262 if ( currentProperty )
263 {
264 continue;
265 }
266 else // QgsProjectPropertyValue not Key, so return null
267 {
268 return nullptr;
269 }
270 }
271 else // the next subkey doesn't exist, so add it
272 {
273 if ( ( newPropertyKey = currentProperty->addKey( keySequence.first() ) ) )
274 {
275 currentProperty = newPropertyKey;
276 }
277 continue;
278 }
279 }
280 else
281 {
282 return nullptr;
283 }
284 }
285
286 return nullptr;
287}
288
296void removeKey_( const QString &scope,
297 const QString &key,
298 QgsProjectPropertyKey &rootProperty )
299{
300 QgsProjectPropertyKey *currentProperty = &rootProperty;
301
302 QgsProjectProperty *nextProperty = nullptr; // link to next property down hierarchy
303 QgsProjectPropertyKey *previousQgsPropertyKey = nullptr; // link to previous property up hierarchy
304
305 QStringList keySequence = makeKeyTokens_( scope, key );
306
307 while ( ! keySequence.isEmpty() )
308 {
309 // if the current head of the sequence list matches the property name,
310 // then traverse down the property hierarchy
311 if ( keySequence.first() == currentProperty->name() )
312 {
313 // remove front key since we're traversing down a level
314 keySequence.pop_front();
315
316 // if we have only one key name left, then try to remove the key
317 // with that name
318 if ( 1 == keySequence.count() )
319 {
320 currentProperty->removeKey( keySequence.front() );
321 }
322 // if we're out of keys then the current property is the one we
323 // want to remove, but we can't delete it directly; we need to
324 // delete it from the parent property key container
325 else if ( keySequence.isEmpty() )
326 {
327 previousQgsPropertyKey->removeKey( currentProperty->name() );
328 }
329 else if ( ( nextProperty = currentProperty->find( keySequence.first() ) ) )
330 {
331 previousQgsPropertyKey = currentProperty;
332 currentProperty = dynamic_cast<QgsProjectPropertyKey *>( nextProperty );
333
334 if ( currentProperty )
335 {
336 continue;
337 }
338 else // QgsProjectPropertyValue not Key, so return null
339 {
340 return;
341 }
342 }
343 else // if the next key down isn't found
344 {
345 // then the overall key sequence doesn't exist
346 return;
347 }
348 }
349 else
350 {
351 return;
352 }
353 }
354}
355
356QgsProject::QgsProject( QObject *parent, Qgis::ProjectCapabilities capabilities )
357 : QObject( parent )
358 , mCapabilities( capabilities )
359 , mLayerStore( new QgsMapLayerStore( this ) )
360 , mBadLayerHandler( new QgsProjectBadLayerHandler() )
361 , mSnappingConfig( this )
362 , mRelationManager( new QgsRelationManager( this ) )
363 , mAnnotationManager( new QgsAnnotationManager( this ) )
364 , mLayoutManager( new QgsLayoutManager( this ) )
365 , m3DViewsManager( new QgsMapViewsManager( this ) )
366 , mBookmarkManager( QgsBookmarkManager::createProjectBasedManager( this ) )
367 , mSensorManager( new QgsSensorManager( this ) )
368 , mViewSettings( new QgsProjectViewSettings( this ) )
369 , mStyleSettings( new QgsProjectStyleSettings( this ) )
370 , mTimeSettings( new QgsProjectTimeSettings( this ) )
371 , mElevationProperties( new QgsProjectElevationProperties( this ) )
372 , mDisplaySettings( new QgsProjectDisplaySettings( this ) )
373 , mGpsSettings( new QgsProjectGpsSettings( this ) )
374 , mRootGroup( new QgsLayerTree )
375 , mLabelingEngineSettings( new QgsLabelingEngineSettings )
376 , mArchive( new QgsArchive() )
377 , mAuxiliaryStorage( new QgsAuxiliaryStorage() )
378{
379 mProperties.setName( QStringLiteral( "properties" ) );
380
381 mMainAnnotationLayer = new QgsAnnotationLayer( QObject::tr( "Annotations" ), QgsAnnotationLayer::LayerOptions( mTransformContext ) );
382 mMainAnnotationLayer->setParent( this );
383
384 clear();
385
386 // bind the layer tree to the map layer registry.
387 // whenever layers are added to or removed from the registry,
388 // layer tree will be updated
389 mLayerTreeRegistryBridge = new QgsLayerTreeRegistryBridge( mRootGroup, this, this );
390 connect( this, &QgsProject::layersAdded, this, &QgsProject::onMapLayersAdded );
391 connect( this, &QgsProject::layersRemoved, this, [this] { cleanTransactionGroups(); } );
392 connect( this, qOverload< const QList<QgsMapLayer *> & >( &QgsProject::layersWillBeRemoved ), this, &QgsProject::onMapLayersRemoved );
393
394 // proxy map layer store signals to this
395 connect( mLayerStore.get(), qOverload<const QStringList &>( &QgsMapLayerStore::layersWillBeRemoved ),
396 this, [this]( const QStringList & layers ) { mProjectScope.reset(); emit layersWillBeRemoved( layers ); } );
397 connect( mLayerStore.get(), qOverload< const QList<QgsMapLayer *> & >( &QgsMapLayerStore::layersWillBeRemoved ),
398 this, [this]( const QList<QgsMapLayer *> &layers ) { mProjectScope.reset(); emit layersWillBeRemoved( layers ); } );
399 connect( mLayerStore.get(), qOverload< const QString & >( &QgsMapLayerStore::layerWillBeRemoved ),
400 this, [this]( const QString & layer ) { mProjectScope.reset(); emit layerWillBeRemoved( layer ); } );
401 connect( mLayerStore.get(), qOverload< QgsMapLayer * >( &QgsMapLayerStore::layerWillBeRemoved ),
402 this, [this]( QgsMapLayer * layer ) { mProjectScope.reset(); emit layerWillBeRemoved( layer ); } );
403 connect( mLayerStore.get(), qOverload<const QStringList & >( &QgsMapLayerStore::layersRemoved ), this,
404 [this]( const QStringList & layers ) { mProjectScope.reset(); emit layersRemoved( layers ); } );
405 connect( mLayerStore.get(), &QgsMapLayerStore::layerRemoved, this,
406 [this]( const QString & layer ) { mProjectScope.reset(); emit layerRemoved( layer ); } );
407 connect( mLayerStore.get(), &QgsMapLayerStore::allLayersRemoved, this,
408 [this]() { mProjectScope.reset(); emit removeAll(); } );
409 connect( mLayerStore.get(), &QgsMapLayerStore::layersAdded, this,
410 [this]( const QList< QgsMapLayer * > &layers ) { mProjectScope.reset(); emit layersAdded( layers ); } );
411 connect( mLayerStore.get(), &QgsMapLayerStore::layerWasAdded, this,
412 [this]( QgsMapLayer * layer ) { mProjectScope.reset(); emit layerWasAdded( layer ); } );
413
415 {
417 }
418
419 connect( mLayerStore.get(), qOverload< const QList<QgsMapLayer *> & >( &QgsMapLayerStore::layersWillBeRemoved ), this,
420 [this]( const QList<QgsMapLayer *> &layers )
421 {
422 for ( const auto &layer : layers )
423 {
424 disconnect( layer, &QgsMapLayer::dataSourceChanged, mRelationManager, &QgsRelationManager::updateRelationsStatus );
425 }
426 }
427 );
428 connect( mLayerStore.get(), qOverload< const QList<QgsMapLayer *> & >( &QgsMapLayerStore::layersAdded ), this,
429 [this]( const QList<QgsMapLayer *> &layers )
430 {
431 for ( const auto &layer : layers )
432 {
433 connect( layer, &QgsMapLayer::dataSourceChanged, mRelationManager, &QgsRelationManager::updateRelationsStatus );
434 }
435 }
436 );
437
441
442 mStyleSettings->combinedStyleModel()->addDefaultStyle();
443}
444
445
447{
448 mIsBeingDeleted = true;
449
450 clear();
451 releaseHandlesToProjectArchive();
452 delete mBadLayerHandler;
453 delete mRelationManager;
454 delete mLayerTreeRegistryBridge;
455 delete mRootGroup;
456 if ( this == sProject )
457 {
458 sProject = nullptr;
459 }
460}
461
463{
464 sProject = project;
465}
466
467
468QgsProject *QgsProject::instance() // skip-keyword-check
469{
470 if ( !sProject )
471 {
472 sProject = new QgsProject;
473
475 }
476 return sProject;
477}
478
479void QgsProject::setTitle( const QString &title )
480{
482
483 if ( title == mMetadata.title() )
484 return;
485
486 mMetadata.setTitle( title );
487 mProjectScope.reset();
488 emit metadataChanged();
489
490 setDirty( true );
491}
492
493QString QgsProject::title() const
494{
495 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
497
498 return mMetadata.title();
499}
500
502{
504
505 const bool oldEvaluateDefaultValues = mFlags & Qgis::ProjectFlag::EvaluateDefaultValuesOnProviderSide;
506 const bool newEvaluateDefaultValues = flags & Qgis::ProjectFlag::EvaluateDefaultValuesOnProviderSide;
507 if ( oldEvaluateDefaultValues != newEvaluateDefaultValues )
508 {
509 const QMap<QString, QgsMapLayer *> layers = mapLayers();
510 for ( auto layerIt = layers.constBegin(); layerIt != layers.constEnd(); ++layerIt )
511 {
512 if ( QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( layerIt.value() ) )
513 if ( vl->dataProvider() )
514 vl->dataProvider()->setProviderProperty( QgsVectorDataProvider::EvaluateDefaultValues, newEvaluateDefaultValues );
515 }
516 }
517
518 const bool oldTrustLayerMetadata = mFlags & Qgis::ProjectFlag::TrustStoredLayerStatistics;
519 const bool newTrustLayerMetadata = flags & Qgis::ProjectFlag::TrustStoredLayerStatistics;
520 if ( oldTrustLayerMetadata != newTrustLayerMetadata )
521 {
522 const QMap<QString, QgsMapLayer *> layers = mapLayers();
523 for ( auto layerIt = layers.constBegin(); layerIt != layers.constEnd(); ++layerIt )
524 {
525 if ( QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( layerIt.value() ) )
526 {
527 vl->setReadExtentFromXml( newTrustLayerMetadata );
528 }
529 }
530 }
531
532 if ( mFlags != flags )
533 {
534 mFlags = flags;
535 setDirty( true );
536 }
537}
538
539void QgsProject::setFlag( Qgis::ProjectFlag flag, bool enabled )
540{
542
543 Qgis::ProjectFlags newFlags = mFlags;
544 if ( enabled )
545 newFlags |= flag;
546 else
547 newFlags &= ~( static_cast< int >( flag ) );
548 setFlags( newFlags );
549}
550
551QString QgsProject::saveUser() const
552{
554
555 return mSaveUser;
556}
557
559{
561
562 return mSaveUserFull;
563}
564
566{
568
569 return mSaveDateTime;
570}
571
578
580{
582
583 return mDirty;
584}
585
586void QgsProject::setDirty( const bool dirty )
587{
589
590 if ( dirty && mDirtyBlockCount > 0 )
591 return;
592
593 if ( dirty )
594 emit dirtySet();
595
596 if ( mDirty == dirty )
597 return;
598
599 mDirty = dirty;
600 emit isDirtyChanged( mDirty );
601}
602
603void QgsProject::setPresetHomePath( const QString &path )
604{
606
607 if ( path == mHomePath )
608 return;
609
610 mHomePath = path;
611 mCachedHomePath.clear();
612 mProjectScope.reset();
613
614 emit homePathChanged();
615
616 setDirty( true );
617}
618
619void QgsProject::registerTranslatableContainers( QgsTranslationContext *translationContext, QgsAttributeEditorContainer *parent, const QString &layerId )
620{
622
623 const QList<QgsAttributeEditorElement *> elements = parent->children();
624
625 for ( QgsAttributeEditorElement *element : elements )
626 {
627 if ( element->type() == Qgis::AttributeEditorType::Container )
628 {
629 QgsAttributeEditorContainer *container = dynamic_cast<QgsAttributeEditorContainer *>( element );
630
631 translationContext->registerTranslation( QStringLiteral( "project:layers:%1:formcontainers" ).arg( layerId ), container->name() );
632
633 if ( !container->children().empty() )
634 registerTranslatableContainers( translationContext, container, layerId );
635 }
636 }
637}
638
640{
642
643 //register layers
644 const QList<QgsLayerTreeLayer *> layers = mRootGroup->findLayers();
645
646 for ( const QgsLayerTreeLayer *layer : layers )
647 {
648 translationContext->registerTranslation( QStringLiteral( "project:layers:%1" ).arg( layer->layerId() ), layer->name() );
649
650 QgsMapLayer *mapLayer = layer->layer();
652 {
653 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mapLayer );
654
655 //register aliases and widget settings
656 const QgsFields fields = vlayer->fields();
657 for ( const QgsField &field : fields )
658 {
659 QString fieldName;
660 if ( field.alias().isEmpty() )
661 fieldName = field.name();
662 else
663 fieldName = field.alias();
664
665 translationContext->registerTranslation( QStringLiteral( "project:layers:%1:fieldaliases" ).arg( vlayer->id() ), fieldName );
666
667 if ( field.editorWidgetSetup().type() == QStringLiteral( "ValueRelation" ) )
668 {
669 translationContext->registerTranslation( QStringLiteral( "project:layers:%1:fields:%2:valuerelationvalue" ).arg( vlayer->id(), field.name() ), field.editorWidgetSetup().config().value( QStringLiteral( "Value" ) ).toString() );
670 }
671 if ( field.editorWidgetSetup().type() == QStringLiteral( "ValueMap" ) )
672 {
673 if ( field.editorWidgetSetup().config().value( QStringLiteral( "map" ) ).canConvert<QList<QVariant>>() )
674 {
675 const QList<QVariant> valueList = field.editorWidgetSetup().config().value( QStringLiteral( "map" ) ).toList();
676
677 for ( int i = 0, row = 0; i < valueList.count(); i++, row++ )
678 {
679 translationContext->registerTranslation( QStringLiteral( "project:layers:%1:fields:%2:valuemapdescriptions" ).arg( vlayer->id(), field.name() ), valueList[i].toMap().constBegin().key() );
680 }
681 }
682 }
683 }
684
685 //register formcontainers
686 registerTranslatableContainers( translationContext, vlayer->editFormConfig().invisibleRootContainer(), vlayer->id() );
687
688 }
689 }
690
691 //register layergroups
692 const QList<QgsLayerTreeGroup *> groupLayers = mRootGroup->findGroups();
693 for ( const QgsLayerTreeGroup *groupLayer : groupLayers )
694 {
695 translationContext->registerTranslation( QStringLiteral( "project:layergroups" ), groupLayer->name() );
696 }
697
698 //register relations
699 const QList<QgsRelation> &relations = mRelationManager->relations().values();
700 for ( const QgsRelation &relation : relations )
701 {
702 translationContext->registerTranslation( QStringLiteral( "project:relations" ), relation.name() );
703 }
704}
705
707{
709
710 mDataDefinedServerProperties = properties;
711}
712
714{
716
717 return mDataDefinedServerProperties;
718}
719
721{
723
724 switch ( mTransactionMode )
725 {
728 {
729 if ( ! vectorLayer )
730 return false;
731 return vectorLayer->startEditing();
732 }
733
735 return mEditBufferGroup.startEditing();
736 }
737
738 return false;
739}
740
741bool QgsProject::commitChanges( QStringList &commitErrors, bool stopEditing, QgsVectorLayer *vectorLayer )
742{
744
745 switch ( mTransactionMode )
746 {
749 {
750 if ( ! vectorLayer )
751 {
752 commitErrors.append( tr( "Trying to commit changes without a layer specified. This only works if the transaction mode is buffered" ) );
753 return false;
754 }
755 bool success = vectorLayer->commitChanges( stopEditing );
756 commitErrors = vectorLayer->commitErrors();
757 return success;
758 }
759
761 return mEditBufferGroup.commitChanges( commitErrors, stopEditing );
762 }
763
764 return false;
765}
766
767bool QgsProject::rollBack( QStringList &rollbackErrors, bool stopEditing, QgsVectorLayer *vectorLayer )
768{
770
771 switch ( mTransactionMode )
772 {
775 {
776 if ( ! vectorLayer )
777 {
778 rollbackErrors.append( tr( "Trying to roll back changes without a layer specified. This only works if the transaction mode is buffered" ) );
779 return false;
780 }
781 bool success = vectorLayer->rollBack( stopEditing );
782 rollbackErrors = vectorLayer->commitErrors();
783 return success;
784 }
785
787 return mEditBufferGroup.rollBack( rollbackErrors, stopEditing );
788 }
789
790 return false;
791}
792
793void QgsProject::setFileName( const QString &name )
794{
796
797 if ( name == mFile.fileName() )
798 return;
799
800 const QString oldHomePath = homePath();
801
802 mFile.setFileName( name );
803 mCachedHomePath.clear();
804 mProjectScope.reset();
805
806 emit fileNameChanged();
807
808 const QString newHomePath = homePath();
809 if ( newHomePath != oldHomePath )
810 emit homePathChanged();
811
812 setDirty( true );
813}
814
815QString QgsProject::fileName() const
816{
817 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
819
820 return mFile.fileName();
821}
822
823void QgsProject::setOriginalPath( const QString &path )
824{
826
827 mOriginalPath = path;
828}
829
831{
833
834 return mOriginalPath;
835}
836
837QFileInfo QgsProject::fileInfo() const
838{
840
841 return QFileInfo( mFile );
842}
843
845{
846 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
848
850}
851
853{
855
856 if ( QgsProjectStorage *storage = projectStorage() )
857 {
859 storage->readProjectStorageMetadata( mFile.fileName(), metadata );
860 return metadata.lastModified;
861 }
862 else
863 {
864 return QFileInfo( mFile.fileName() ).lastModified();
865 }
866}
867
869{
871
872 if ( projectStorage() )
873 return QString();
874
875 if ( mFile.fileName().isEmpty() )
876 return QString(); // this is to protect ourselves from getting current directory from QFileInfo::absoluteFilePath()
877
878 return QFileInfo( mFile.fileName() ).absolutePath();
879}
880
882{
883 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
885
886 if ( projectStorage() )
887 return QString();
888
889 if ( mFile.fileName().isEmpty() )
890 return QString(); // this is to protect ourselves from getting current directory from QFileInfo::absoluteFilePath()
891
892 return QFileInfo( mFile.fileName() ).absoluteFilePath();
893}
894
895QString QgsProject::baseName() const
896{
897 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
899
900 if ( QgsProjectStorage *storage = projectStorage() )
901 {
903 storage->readProjectStorageMetadata( mFile.fileName(), metadata );
904 return metadata.name;
905 }
906 else
907 {
908 return QFileInfo( mFile.fileName() ).completeBaseName();
909 }
910}
911
913{
915
916 const bool absolutePaths = readBoolEntry( QStringLiteral( "Paths" ), QStringLiteral( "/Absolute" ), false );
918}
919
921{
923
924 switch ( type )
925 {
927 writeEntry( QStringLiteral( "Paths" ), QStringLiteral( "/Absolute" ), true );
928 break;
930 writeEntry( QStringLiteral( "Paths" ), QStringLiteral( "/Absolute" ), false );
931 break;
932 }
933}
934
936{
937 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
939
940 return mCrs;
941}
942
944{
946
947 return mCrs3D.isValid() ? mCrs3D : mCrs;
948}
949
950void QgsProject::setCrs( const QgsCoordinateReferenceSystem &crs, bool adjustEllipsoid )
951{
953
954 if ( crs != mCrs )
955 {
956 const QgsCoordinateReferenceSystem oldVerticalCrs = verticalCrs();
957 const QgsCoordinateReferenceSystem oldCrs3D = mCrs3D;
958 mCrs = crs;
959 writeEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectionsEnabled" ), crs.isValid() ? 1 : 0 );
960 mProjectScope.reset();
961
962 // if annotation layer doesn't have a crs (i.e. in a newly created project), it should
963 // initially inherit the project CRS
964 if ( !mMainAnnotationLayer->crs().isValid() || mMainAnnotationLayer->isEmpty() )
965 mMainAnnotationLayer->setCrs( crs );
966
967 rebuildCrs3D();
968
969 setDirty( true );
970 emit crsChanged();
971 // Did vertical crs also change as a result of this? If so, emit signal
972 if ( oldVerticalCrs != verticalCrs() )
973 emit verticalCrsChanged();
974 if ( oldCrs3D != mCrs3D )
975 emit crs3DChanged();
976 }
977
978 if ( adjustEllipsoid )
980}
981
983{
984 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
986
987 if ( !crs().isValid() )
988 return Qgis::geoNone();
989
990 return readEntry( QStringLiteral( "Measure" ), QStringLiteral( "/Ellipsoid" ), Qgis::geoNone() );
991}
992
993void QgsProject::setEllipsoid( const QString &ellipsoid )
994{
996
997 if ( ellipsoid == readEntry( QStringLiteral( "Measure" ), QStringLiteral( "/Ellipsoid" ) ) )
998 return;
999
1000 mProjectScope.reset();
1001 writeEntry( QStringLiteral( "Measure" ), QStringLiteral( "/Ellipsoid" ), ellipsoid );
1003}
1004
1006{
1007 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
1009
1010 switch ( mCrs.type() )
1011 {
1012 case Qgis::CrsType::Vertical: // would hope this never happens!
1013 QgsDebugError( QStringLiteral( "Project has a vertical CRS set as the horizontal CRS!" ) );
1014 return mCrs;
1015
1017 return mCrs.verticalCrs();
1018
1030 break;
1031 }
1032 return mVerticalCrs;
1033}
1034
1036{
1038 bool res = true;
1039 if ( crs.isValid() )
1040 {
1041 // validate that passed crs is a vertical crs
1042 switch ( crs.type() )
1043 {
1045 break;
1046
1059 if ( errorMessage )
1060 *errorMessage = QObject::tr( "Specified CRS is a %1 CRS, not a Vertical CRS" ).arg( qgsEnumValueToKey( crs.type() ) );
1061 return false;
1062 }
1063 }
1064
1065 if ( crs != mVerticalCrs )
1066 {
1067 const QgsCoordinateReferenceSystem oldVerticalCrs = verticalCrs();
1068 const QgsCoordinateReferenceSystem oldCrs3D = mCrs3D;
1069
1070 switch ( mCrs.type() )
1071 {
1073 if ( crs != oldVerticalCrs )
1074 {
1075 if ( errorMessage )
1076 *errorMessage = QObject::tr( "Project CRS is a Compound CRS, specified Vertical CRS will be ignored" );
1077 return false;
1078 }
1079 break;
1080
1082 if ( crs != oldVerticalCrs )
1083 {
1084 if ( errorMessage )
1085 *errorMessage = QObject::tr( "Project CRS is a Geographic 3D CRS, specified Vertical CRS will be ignored" );
1086 return false;
1087 }
1088 break;
1089
1091 if ( crs != oldVerticalCrs )
1092 {
1093 if ( errorMessage )
1094 *errorMessage = QObject::tr( "Project CRS is a Geocentric CRS, specified Vertical CRS will be ignored" );
1095 return false;
1096 }
1097 break;
1098
1100 if ( mCrs.hasVerticalAxis() && crs != oldVerticalCrs )
1101 {
1102 if ( errorMessage )
1103 *errorMessage = QObject::tr( "Project CRS is a Projected 3D CRS, specified Vertical CRS will be ignored" );
1104 return false;
1105 }
1106 break;
1107
1117 break;
1118 }
1119
1120 mVerticalCrs = crs;
1121 res = rebuildCrs3D( errorMessage );
1122 mProjectScope.reset();
1123
1124 setDirty( true );
1125 // only emit signal if vertical crs was actually changed, so eg if mCrs is compound
1126 // then we haven't actually changed the vertical crs by this call!
1127 if ( verticalCrs() != oldVerticalCrs )
1128 emit verticalCrsChanged();
1129 if ( mCrs3D != oldCrs3D )
1130 emit crs3DChanged();
1131 }
1132 return res;
1133}
1134
1136{
1137 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
1139
1140 return mTransformContext;
1141}
1142
1144{
1146
1147 if ( context == mTransformContext )
1148 return;
1149
1150 mTransformContext = context;
1151 mProjectScope.reset();
1152
1153 mMainAnnotationLayer->setTransformContext( context );
1154 for ( auto &layer : mLayerStore.get()->mapLayers() )
1155 {
1156 layer->setTransformContext( context );
1157 }
1159}
1160
1162{
1164
1165 ScopedIntIncrementor snapSingleBlocker( &mBlockSnappingUpdates );
1166
1167 emit aboutToBeCleared();
1168
1169 if ( !mIsBeingDeleted )
1170 {
1171 // Unregister expression functions stored in the project.
1172 // If we clean on destruction we may end-up with a non-valid
1173 // mPythonUtils, so be safe and only clean when not destroying.
1174 // This should be called before calling mProperties.clearKeys().
1176 }
1177
1178 mProjectScope.reset();
1179 mFile.setFileName( QString() );
1180 mProperties.clearKeys();
1181 mSaveUser.clear();
1182 mSaveUserFull.clear();
1183 mSaveDateTime = QDateTime();
1184 mSaveVersion = QgsProjectVersion();
1185 mHomePath.clear();
1186 mCachedHomePath.clear();
1187 mTransactionMode = Qgis::TransactionMode::Disabled;
1188 mFlags = Qgis::ProjectFlags();
1189 mDirty = false;
1190 mCustomVariables.clear();
1192 mVerticalCrs = QgsCoordinateReferenceSystem();
1194 mMetadata = QgsProjectMetadata();
1195 mElevationShadingRenderer = QgsElevationShadingRenderer();
1196 if ( !mSettings.value( QStringLiteral( "projects/anonymize_new_projects" ), false, QgsSettings::Core ).toBool() )
1197 {
1198 mMetadata.setCreationDateTime( QDateTime::currentDateTime() );
1200 }
1201 emit metadataChanged();
1202
1204 context.readSettings();
1205 setTransformContext( context );
1206
1207 //fallback to QGIS default measurement unit
1208 bool ok = false;
1209 const Qgis::DistanceUnit distanceUnit = QgsUnitTypes::decodeDistanceUnit( mSettings.value( QStringLiteral( "/qgis/measure/displayunits" ) ).toString(), &ok );
1210 setDistanceUnits( ok ? distanceUnit : Qgis::DistanceUnit::Meters );
1211 ok = false;
1212 const Qgis::AreaUnit areaUnits = QgsUnitTypes::decodeAreaUnit( mSettings.value( QStringLiteral( "/qgis/measure/areaunits" ) ).toString(), &ok );
1214
1216
1217 mEmbeddedLayers.clear();
1218 mRelationManager->clear();
1219 mAnnotationManager->clear();
1220 mLayoutManager->clear();
1221 m3DViewsManager->clear();
1222 mBookmarkManager->clear();
1223 mSensorManager->clear();
1224 mViewSettings->reset();
1225 mTimeSettings->reset();
1226 mElevationProperties->reset();
1227 mDisplaySettings->reset();
1228 mGpsSettings->reset();
1229 mSnappingConfig.reset();
1230 mAvoidIntersectionsMode = Qgis::AvoidIntersectionsMode::AllowIntersections;
1233
1234 mMapThemeCollection.reset( new QgsMapThemeCollection( this ) );
1236
1237 mLabelingEngineSettings->clear();
1238
1239 // must happen BEFORE archive reset, because we need to release the hold on any files which
1240 // exists within the archive. Otherwise the archive can't be removed.
1241 releaseHandlesToProjectArchive();
1242
1243 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage() );
1244 mArchive.reset( new QgsArchive() );
1245
1246 // must happen AFTER archive reset, as it will populate a new style database within the new archive
1247 mStyleSettings->reset();
1248
1250
1251 if ( !mIsBeingDeleted )
1252 {
1253 // possibly other signals should also not be thrown on destruction -- e.g. labelEngineSettingsChanged, etc.
1254 emit projectColorsChanged();
1255 }
1256
1257 // reset some default project properties
1258 // XXX THESE SHOULD BE MOVED TO STATUSBAR RELATED SOURCE
1259 writeEntry( QStringLiteral( "PositionPrecision" ), QStringLiteral( "/Automatic" ), true );
1260 writeEntry( QStringLiteral( "PositionPrecision" ), QStringLiteral( "/DecimalPlaces" ), 2 );
1261
1262 const bool defaultRelativePaths = mSettings.value( QStringLiteral( "/qgis/defaultProjectPathsRelative" ), true ).toBool();
1264
1265 int red = mSettings.value( QStringLiteral( "qgis/default_canvas_color_red" ), 255 ).toInt();
1266 int green = mSettings.value( QStringLiteral( "qgis/default_canvas_color_green" ), 255 ).toInt();
1267 int blue = mSettings.value( QStringLiteral( "qgis/default_canvas_color_blue" ), 255 ).toInt();
1268 setBackgroundColor( QColor( red, green, blue ) );
1269
1270 red = mSettings.value( QStringLiteral( "qgis/default_selection_color_red" ), 255 ).toInt();
1271 green = mSettings.value( QStringLiteral( "qgis/default_selection_color_green" ), 255 ).toInt();
1272 blue = mSettings.value( QStringLiteral( "qgis/default_selection_color_blue" ), 0 ).toInt();
1273 const int alpha = mSettings.value( QStringLiteral( "qgis/default_selection_color_alpha" ), 255 ).toInt();
1274 setSelectionColor( QColor( red, green, blue, alpha ) );
1275
1276 mSnappingConfig.clearIndividualLayerSettings();
1277
1279 mRootGroup->clear();
1280 if ( mMainAnnotationLayer )
1281 mMainAnnotationLayer->reset();
1282
1283 snapSingleBlocker.release();
1284
1285 if ( !mBlockSnappingUpdates )
1286 emit snappingConfigChanged( mSnappingConfig );
1287
1288 setDirty( false );
1289 emit homePathChanged();
1290 if ( !mBlockChangeSignalsDuringClear )
1291 {
1292 emit verticalCrsChanged();
1293 emit crs3DChanged();
1294 }
1295 emit cleared();
1296}
1297
1298// basically a debugging tool to dump property list values
1299void dump_( const QgsProjectPropertyKey &topQgsPropertyKey )
1300{
1301 QgsDebugMsgLevel( QStringLiteral( "current properties:" ), 3 );
1302 topQgsPropertyKey.dump();
1303}
1304
1333void _getProperties( const QDomDocument &doc, QgsProjectPropertyKey &project_properties )
1334{
1335 const QDomElement propertiesElem = doc.documentElement().firstChildElement( QStringLiteral( "properties" ) );
1336
1337 if ( propertiesElem.isNull() ) // no properties found, so we're done
1338 {
1339 return;
1340 }
1341
1342 const QDomNodeList scopes = propertiesElem.childNodes();
1343
1344 if ( propertiesElem.firstChild().isNull() )
1345 {
1346 QgsDebugError( QStringLiteral( "empty ``properties'' XML tag ... bailing" ) );
1347 return;
1348 }
1349
1350 if ( ! project_properties.readXml( propertiesElem ) )
1351 {
1352 QgsDebugError( QStringLiteral( "Project_properties.readXml() failed" ) );
1353 }
1354}
1355
1362QgsPropertyCollection getDataDefinedServerProperties( const QDomDocument &doc, const QgsPropertiesDefinition &dataDefinedServerPropertyDefinitions )
1363{
1364 QgsPropertyCollection ddServerProperties;
1365 // Read data defined server properties
1366 const QDomElement ddElem = doc.documentElement().firstChildElement( QStringLiteral( "dataDefinedServerProperties" ) );
1367 if ( !ddElem.isNull() )
1368 {
1369 if ( !ddServerProperties.readXml( ddElem, dataDefinedServerPropertyDefinitions ) )
1370 {
1371 QgsDebugError( QStringLiteral( "dataDefinedServerProperties.readXml() failed" ) );
1372 }
1373 }
1374 return ddServerProperties;
1375}
1376
1381static void _getTitle( const QDomDocument &doc, QString &title )
1382{
1383 const QDomElement titleNode = doc.documentElement().firstChildElement( QStringLiteral( "title" ) );
1384
1385 title.clear(); // by default the title will be empty
1386
1387 if ( titleNode.isNull() )
1388 {
1389 QgsDebugMsgLevel( QStringLiteral( "unable to find title element" ), 2 );
1390 return;
1391 }
1392
1393 if ( !titleNode.hasChildNodes() ) // if not, then there's no actual text
1394 {
1395 QgsDebugMsgLevel( QStringLiteral( "unable to find title element" ), 2 );
1396 return;
1397 }
1398
1399 const QDomNode titleTextNode = titleNode.firstChild(); // should only have one child
1400
1401 if ( !titleTextNode.isText() )
1402 {
1403 QgsDebugMsgLevel( QStringLiteral( "unable to find title element" ), 2 );
1404 return;
1405 }
1406
1407 const QDomText titleText = titleTextNode.toText();
1408
1409 title = titleText.data();
1410
1411}
1412
1413static void readProjectFileMetadata( const QDomDocument &doc, QString &lastUser, QString &lastUserFull, QDateTime &lastSaveDateTime )
1414{
1415 const QDomNodeList nl = doc.elementsByTagName( QStringLiteral( "qgis" ) );
1416
1417 if ( !nl.count() )
1418 {
1419 QgsDebugError( QStringLiteral( "unable to find qgis element" ) );
1420 return;
1421 }
1422
1423 const QDomNode qgisNode = nl.item( 0 ); // there should only be one, so zeroth element OK
1424
1425 const QDomElement qgisElement = qgisNode.toElement(); // qgis node should be element
1426 lastUser = qgisElement.attribute( QStringLiteral( "saveUser" ), QString() );
1427 lastUserFull = qgisElement.attribute( QStringLiteral( "saveUserFull" ), QString() );
1428 lastSaveDateTime = QDateTime::fromString( qgisElement.attribute( QStringLiteral( "saveDateTime" ), QString() ), Qt::ISODate );
1429}
1430
1431QgsProjectVersion getVersion( const QDomDocument &doc )
1432{
1433 const QDomNodeList nl = doc.elementsByTagName( QStringLiteral( "qgis" ) );
1434
1435 if ( !nl.count() )
1436 {
1437 QgsDebugError( QStringLiteral( " unable to find qgis element in project file" ) );
1438 return QgsProjectVersion( 0, 0, 0, QString() );
1439 }
1440
1441 const QDomNode qgisNode = nl.item( 0 ); // there should only be one, so zeroth element OK
1442
1443 const QDomElement qgisElement = qgisNode.toElement(); // qgis node should be element
1444 QgsProjectVersion projectVersion( qgisElement.attribute( QStringLiteral( "version" ) ) );
1445 return projectVersion;
1446}
1447
1449{
1451
1452 return mSnappingConfig;
1453}
1454
1456{
1458
1459 if ( mSnappingConfig == snappingConfig )
1460 return;
1461
1462 mSnappingConfig = snappingConfig;
1463 setDirty( true );
1464 emit snappingConfigChanged( mSnappingConfig );
1465}
1466
1468{
1470
1471 if ( mAvoidIntersectionsMode == mode )
1472 return;
1473
1474 mAvoidIntersectionsMode = mode;
1476}
1477
1478static QgsMapLayer::ReadFlags projectFlagsToLayerReadFlags( Qgis::ProjectReadFlags projectReadFlags, Qgis::ProjectFlags projectFlags )
1479{
1481 // Propagate don't resolve layers
1482 if ( projectReadFlags & Qgis::ProjectReadFlag::DontResolveLayers )
1484 // Propagate trust layer metadata flag
1485 // Propagate read extent from XML based trust layer metadata flag
1486 if ( ( projectFlags & Qgis::ProjectFlag::TrustStoredLayerStatistics ) || ( projectReadFlags & Qgis::ProjectReadFlag::TrustLayerMetadata ) )
1487 {
1490 }
1491 // Propagate open layers in read-only mode
1492 if ( ( projectReadFlags & Qgis::ProjectReadFlag::ForceReadOnlyLayers ) )
1493 layerFlags |= QgsMapLayer::FlagForceReadOnly;
1494
1495 return layerFlags;
1496}
1497
1507
1508void QgsProject::preloadProviders( const QVector<QDomNode> &parallelLayerNodes,
1509 const QgsReadWriteContext &context,
1510 QMap<QString, QgsDataProvider *> &loadedProviders,
1511 QgsMapLayer::ReadFlags layerReadFlags,
1512 int totalProviderCount )
1513{
1514 int i = 0;
1515 QEventLoop loop;
1516
1517 QMap<QString, LayerToLoad> layersToLoad;
1518
1519 for ( const QDomNode &node : parallelLayerNodes )
1520 {
1521 LayerToLoad layerToLoad;
1522
1523 const QDomElement layerElement = node.toElement();
1524 layerToLoad.layerElement = layerElement;
1525 layerToLoad.layerId = layerElement.namedItem( QStringLiteral( "id" ) ).toElement().text();
1526 layerToLoad.provider = layerElement.namedItem( QStringLiteral( "provider" ) ).toElement().text();
1527 layerToLoad.dataSource = layerElement.namedItem( QStringLiteral( "datasource" ) ).toElement().text();
1528
1529 layerToLoad.dataSource = QgsProviderRegistry::instance()->relativeToAbsoluteUri( layerToLoad.provider, layerToLoad.dataSource, context );
1530
1531 layerToLoad.options = QgsDataProvider::ProviderOptions( {context.transformContext()} );
1532 layerToLoad.flags = QgsMapLayer::providerReadFlags( node, layerReadFlags );
1533
1534 // Requesting credential from worker thread could lead to deadlocks because the main thread is waiting for worker thread to fininsh
1535 layerToLoad.flags.setFlag( Qgis::DataProviderReadFlag::SkipCredentialsRequest, true );
1536 layerToLoad.flags.setFlag( Qgis::DataProviderReadFlag::ParallelThreadLoading, true );
1537
1538 layersToLoad.insert( layerToLoad.layerId, layerToLoad );
1539 }
1540
1541 while ( !layersToLoad.isEmpty() )
1542 {
1543 const QList<LayerToLoad> layersToAttemptInParallel = layersToLoad.values();
1544 QString layerToAttemptInMainThread;
1545
1546 QHash<QString, QgsRunnableProviderCreator *> runnables;
1547 QThreadPool threadPool;
1548 threadPool.setMaxThreadCount( QgsSettingsRegistryCore::settingsLayerParallelLoadingMaxCount->value() );
1549
1550 for ( const LayerToLoad &lay : layersToAttemptInParallel )
1551 {
1552 QgsRunnableProviderCreator *run = new QgsRunnableProviderCreator( lay.layerId, lay.provider, lay.dataSource, lay.options, lay.flags );
1553 runnables.insert( lay.layerId, run );
1554
1555 QObject::connect( run, &QgsRunnableProviderCreator::providerCreated, run, [&]( bool isValid, const QString & layId )
1556 {
1557 if ( isValid )
1558 {
1559 layersToLoad.remove( layId );
1560 i++;
1561 QgsRunnableProviderCreator *finishedRun = runnables.value( layId, nullptr );
1562 Q_ASSERT( finishedRun );
1563
1564 std::unique_ptr<QgsDataProvider> provider( finishedRun->dataProvider() );
1565 Q_ASSERT( provider && provider->isValid() );
1566
1567 loadedProviders.insert( layId, provider.release() );
1568 emit layerLoaded( i, totalProviderCount );
1569 }
1570 else
1571 {
1572 if ( layerToAttemptInMainThread.isEmpty() )
1573 layerToAttemptInMainThread = layId;
1574 threadPool.clear(); //we have to stop all loading provider to try this layer in main thread and maybe have credentials
1575 }
1576
1577 if ( i == parallelLayerNodes.count() || !isValid )
1578 loop.quit();
1579 } );
1580 threadPool.start( run );
1581 }
1582 loop.exec();
1583
1584 threadPool.waitForDone(); // to be sure all threads are finished
1585
1586 qDeleteAll( runnables );
1587
1588 // We try with the first layer returned invalid but this time in the main thread to maybe have credentials and continue with others not loaded in parallel
1589 auto it = layersToLoad.find( layerToAttemptInMainThread );
1590 if ( it != layersToLoad.end() )
1591 {
1592 std::unique_ptr<QgsDataProvider> provider;
1593 QString layerId;
1594 {
1595 const LayerToLoad &lay = it.value();
1596 Qgis::DataProviderReadFlags providerFlags = lay.flags;
1597 providerFlags.setFlag( Qgis::DataProviderReadFlag::SkipCredentialsRequest, false );
1598 providerFlags.setFlag( Qgis::DataProviderReadFlag::ParallelThreadLoading, false );
1599 QgsScopedRuntimeProfile profile( "Create data providers/" + lay.layerId, QStringLiteral( "projectload" ) );
1600 provider.reset( QgsProviderRegistry::instance()->createProvider( lay.provider, lay.dataSource, lay.options, providerFlags ) );
1601 i++;
1602 if ( provider && provider->isValid() )
1603 {
1604 emit layerLoaded( i, totalProviderCount );
1605 }
1606 layerId = lay.layerId;
1607 layersToLoad.erase( it );
1608 // can't access "lay" anymore -- it's now been freed
1609 }
1610 loadedProviders.insert( layerId, provider.release() );
1611 }
1612
1613 // if there still are some not loaded providers or some invalid in parallel thread we start again
1614 }
1615
1616}
1617
1618void QgsProject::releaseHandlesToProjectArchive()
1619{
1620 mStyleSettings->removeProjectStyle();
1621}
1622
1623bool QgsProject::rebuildCrs3D( QString *error )
1624{
1625 bool res = true;
1626 if ( !mCrs.isValid() )
1627 {
1629 }
1630 else if ( !mVerticalCrs.isValid() )
1631 {
1632 mCrs3D = mCrs;
1633 }
1634 else
1635 {
1636 switch ( mCrs.type() )
1637 {
1641 mCrs3D = mCrs;
1642 break;
1643
1645 {
1646 QString tempError;
1647 mCrs3D = mCrs.hasVerticalAxis() ? mCrs : QgsCoordinateReferenceSystem::createCompoundCrs( mCrs, mVerticalCrs, error ? *error : tempError );
1648 res = mCrs3D.isValid();
1649 break;
1650 }
1651
1653 // nonsense situation
1655 res = false;
1656 break;
1657
1666 {
1667 QString tempError;
1668 mCrs3D = QgsCoordinateReferenceSystem::createCompoundCrs( mCrs, mVerticalCrs, error ? *error : tempError );
1669 res = mCrs3D.isValid();
1670 break;
1671 }
1672 }
1673 }
1674 return res;
1675}
1676
1677bool QgsProject::_getMapLayers( const QDomDocument &doc, QList<QDomNode> &brokenNodes, Qgis::ProjectReadFlags flags )
1678{
1680
1681 // Layer order is set by the restoring the legend settings from project file.
1682 // This is done on the 'readProject( ... )' signal
1683
1684 QDomElement layerElement = doc.documentElement().firstChildElement( QStringLiteral( "projectlayers" ) ).firstChildElement( QStringLiteral( "maplayer" ) );
1685
1686 // process the map layer nodes
1687
1688 if ( layerElement.isNull() ) // if we have no layers to process, bail
1689 {
1690 return true; // Decided to return "true" since it's
1691 // possible for there to be a project with no
1692 // layers; but also, more imporantly, this
1693 // would cause the tests/qgsproject to fail
1694 // since the test suite doesn't currently
1695 // support test layers
1696 }
1697
1698 bool returnStatus = true;
1699 int numLayers = 0;
1700
1701 while ( ! layerElement.isNull() )
1702 {
1703 numLayers++;
1704 layerElement = layerElement.nextSiblingElement( QStringLiteral( "maplayer" ) );
1705 }
1706
1707 // order layers based on their dependencies
1708 QgsScopedRuntimeProfile profile( tr( "Sorting layers" ), QStringLiteral( "projectload" ) );
1709 const QgsLayerDefinition::DependencySorter depSorter( doc );
1710 if ( depSorter.hasCycle() )
1711 return false;
1712
1713 // Missing a dependency? We still load all the layers, otherwise the project is completely broken!
1714 if ( depSorter.hasMissingDependency() )
1715 returnStatus = false;
1716
1717 emit layerLoaded( 0, numLayers );
1718
1719 const QVector<QDomNode> sortedLayerNodes = depSorter.sortedLayerNodes();
1720 const int totalLayerCount = sortedLayerNodes.count();
1721
1722 QVector<QDomNode> parallelLoading;
1723 QMap<QString, QgsDataProvider *> loadedProviders;
1724
1727 {
1728 profile.switchTask( tr( "Load providers in parallel" ) );
1729 for ( const QDomNode &node : sortedLayerNodes )
1730 {
1731 const QDomElement element = node.toElement();
1732 if ( element.attribute( QStringLiteral( "embedded" ) ) != QLatin1String( "1" ) )
1733 {
1734 const QString layerId = node.namedItem( QStringLiteral( "id" ) ).toElement().text();
1735 if ( !depSorter.isLayerDependent( layerId ) )
1736 {
1737 const QDomNode mnl = element.namedItem( QStringLiteral( "provider" ) );
1738 const QDomElement mne = mnl.toElement();
1739 const QString provider = mne.text();
1741 if ( meta && meta->providerCapabilities().testFlag( QgsProviderMetadata::ParallelCreateProvider ) )
1742 {
1743 parallelLoading.append( node );
1744 continue;
1745 }
1746 }
1747 }
1748 }
1749
1750 QgsReadWriteContext context;
1751 context.setPathResolver( pathResolver() );
1752 if ( !parallelLoading.isEmpty() )
1753 preloadProviders( parallelLoading, context, loadedProviders, projectFlagsToLayerReadFlags( flags, mFlags ), sortedLayerNodes.count() );
1754 }
1755
1756 int i = loadedProviders.count();
1757 for ( const QDomNode &node : std::as_const( sortedLayerNodes ) )
1758 {
1759 const QDomElement element = node.toElement();
1760 const QString name = translate( QStringLiteral( "project:layers:%1" ).arg( node.namedItem( QStringLiteral( "id" ) ).toElement().text() ), node.namedItem( QStringLiteral( "layername" ) ).toElement().text() );
1761 if ( !name.isNull() )
1762 emit loadingLayer( tr( "Loading layer %1" ).arg( name ) );
1763
1764 profile.switchTask( name );
1765 if ( element.attribute( QStringLiteral( "embedded" ) ) == QLatin1String( "1" ) )
1766 {
1767 createEmbeddedLayer( element.attribute( QStringLiteral( "id" ) ), readPath( element.attribute( QStringLiteral( "project" ) ) ), brokenNodes, true, flags );
1768 }
1769 else
1770 {
1771 QgsReadWriteContext context;
1772 context.setPathResolver( pathResolver() );
1773 context.setProjectTranslator( this );
1775 QString layerId = element.namedItem( QStringLiteral( "id" ) ).toElement().text();
1776
1777 if ( !addLayer( element, brokenNodes, context, flags, loadedProviders.take( layerId ) ) )
1778 {
1779 returnStatus = false;
1780 }
1781 const auto messages = context.takeMessages();
1782 if ( !messages.isEmpty() )
1783 {
1784 emit loadingLayerMessageReceived( tr( "Loading layer %1" ).arg( name ), messages );
1785 }
1786 }
1787 emit layerLoaded( i + 1, totalLayerCount );
1788 i++;
1789 }
1790
1791 return returnStatus;
1792}
1793
1794bool QgsProject::addLayer( const QDomElement &layerElem,
1795 QList<QDomNode> &brokenNodes,
1796 QgsReadWriteContext &context,
1798 QgsDataProvider *provider )
1799{
1801
1802 const QString type = layerElem.attribute( QStringLiteral( "type" ) );
1803 QgsDebugMsgLevel( "Layer type is " + type, 4 );
1804 std::unique_ptr<QgsMapLayer> mapLayer;
1805
1806 QgsScopedRuntimeProfile profile( tr( "Create layer" ), QStringLiteral( "projectload" ) );
1807
1808 bool ok = false;
1809 const Qgis::LayerType layerType( QgsMapLayerFactory::typeFromString( type, ok ) );
1810 if ( !ok )
1811 {
1812 QgsDebugError( QStringLiteral( "Unknown layer type \"%1\"" ).arg( type ) );
1813 return false;
1814 }
1815
1816 switch ( layerType )
1817 {
1819 mapLayer = std::make_unique<QgsVectorLayer>();
1820 break;
1821
1823 mapLayer = std::make_unique<QgsRasterLayer>();
1824 break;
1825
1827 mapLayer = std::make_unique<QgsMeshLayer>();
1828 break;
1829
1831 mapLayer = std::make_unique<QgsVectorTileLayer>();
1832 break;
1833
1835 mapLayer = std::make_unique<QgsPointCloudLayer>();
1836 break;
1837
1839 mapLayer = std::make_unique<QgsTiledSceneLayer>();
1840 break;
1841
1843 {
1844 const QString typeName = layerElem.attribute( QStringLiteral( "name" ) );
1845 mapLayer.reset( QgsApplication::pluginLayerRegistry()->createLayer( typeName ) );
1846 break;
1847 }
1848
1850 {
1851 const QgsAnnotationLayer::LayerOptions options( mTransformContext );
1852 mapLayer = std::make_unique<QgsAnnotationLayer>( QString(), options );
1853 break;
1854 }
1855
1857 {
1858 const QgsGroupLayer::LayerOptions options( mTransformContext );
1859 mapLayer = std::make_unique<QgsGroupLayer>( QString(), options );
1860 break;
1861 }
1862 }
1863
1864 if ( !mapLayer )
1865 {
1866 QgsDebugError( QStringLiteral( "Unable to create layer" ) );
1867 return false;
1868 }
1869
1870 Q_CHECK_PTR( mapLayer ); // NOLINT
1871
1872 // This is tricky: to avoid a leak we need to check if the layer was already in the store
1873 // because if it was, the newly created layer will not be added to the store and it would leak.
1874 const QString layerId { layerElem.namedItem( QStringLiteral( "id" ) ).toElement().text() };
1875 Q_ASSERT( ! layerId.isEmpty() );
1876 const bool layerWasStored = layerStore()->mapLayer( layerId );
1877
1878 // have the layer restore state that is stored in Dom node
1879 QgsMapLayer::ReadFlags layerFlags = projectFlagsToLayerReadFlags( flags, mFlags );
1880
1881 profile.switchTask( tr( "Load layer source" ) );
1882 const bool layerIsValid = mapLayer->readLayerXml( layerElem, context, layerFlags, provider ) && mapLayer->isValid();
1883
1884 // apply specific settings to vector layer
1885 if ( QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mapLayer.get() ) )
1886 {
1887 vl->setReadExtentFromXml( layerFlags & QgsMapLayer::FlagReadExtentFromXml );
1888 if ( vl->dataProvider() )
1889 {
1891 vl->dataProvider()->setProviderProperty( QgsVectorDataProvider::EvaluateDefaultValues, evaluateDefaultValues );
1892 }
1893 }
1894
1895 profile.switchTask( tr( "Add layer to project" ) );
1896 QList<QgsMapLayer *> newLayers;
1897 newLayers << mapLayer.get();
1898 if ( layerIsValid || flags & Qgis::ProjectReadFlag::DontResolveLayers )
1899 {
1900 emit readMapLayer( mapLayer.get(), layerElem );
1901 addMapLayers( newLayers );
1902 // Try to resolve references here (this is necessary to set up joined fields that will be possibly used by
1903 // virtual layers that point to this layer's joined field in their query otherwise they won't be valid ),
1904 // a second attempt to resolve references will be done after all layers are loaded
1905 // see https://github.com/qgis/QGIS/issues/46834
1906 if ( QgsVectorLayer *vLayer = qobject_cast<QgsVectorLayer *>( mapLayer.get() ) )
1907 {
1908 vLayer->joinBuffer()->resolveReferences( this );
1909 }
1910 }
1911 else
1912 {
1913 // It's a bad layer: do not add to legend (the user will decide if she wants to do so)
1914 addMapLayers( newLayers, false );
1915 newLayers.first();
1916 QgsDebugError( "Unable to load " + type + " layer" );
1917 brokenNodes.push_back( layerElem );
1918 }
1919
1920 const bool wasEditable = layerElem.attribute( QStringLiteral( "editable" ), QStringLiteral( "0" ) ).toInt();
1921 if ( wasEditable )
1922 {
1923 mapLayer->setCustomProperty( QStringLiteral( "_layer_was_editable" ), true );
1924 }
1925 else
1926 {
1927 mapLayer->removeCustomProperty( QStringLiteral( "_layer_was_editable" ) );
1928 }
1929
1930 // It should be safe to delete the layer now if layer was stored, because all the store
1931 // had to to was to reset the data source in case the validity changed.
1932 if ( ! layerWasStored )
1933 {
1934 mapLayer.release();
1935 }
1936
1937 return layerIsValid;
1938}
1939
1940bool QgsProject::read( const QString &filename, Qgis::ProjectReadFlags flags )
1941{
1943
1944 mFile.setFileName( filename );
1945 mCachedHomePath.clear();
1946 mProjectScope.reset();
1947
1948 return read( flags );
1949}
1950
1952{
1954
1955 const QString filename = mFile.fileName();
1956 bool returnValue;
1957
1958 if ( QgsProjectStorage *storage = projectStorage() )
1959 {
1960 QTemporaryFile inDevice;
1961 if ( !inDevice.open() )
1962 {
1963 setError( tr( "Unable to open %1" ).arg( inDevice.fileName() ) );
1964 return false;
1965 }
1966
1967 QgsReadWriteContext context;
1968 context.setProjectTranslator( this );
1969 if ( !storage->readProject( filename, &inDevice, context ) )
1970 {
1971 QString err = tr( "Unable to open %1" ).arg( filename );
1972 QList<QgsReadWriteContext::ReadWriteMessage> messages = context.takeMessages();
1973 if ( !messages.isEmpty() )
1974 err += QStringLiteral( "\n\n" ) + messages.last().message();
1975 setError( err );
1976 return false;
1977 }
1978 returnValue = unzip( inDevice.fileName(), flags ); // calls setError() if returning false
1979 }
1980 else
1981 {
1982 if ( QgsZipUtils::isZipFile( mFile.fileName() ) )
1983 {
1984 returnValue = unzip( mFile.fileName(), flags );
1985 }
1986 else
1987 {
1988 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( *this ) );
1989 const QFileInfo finfo( mFile.fileName() );
1990 const QString attachmentsZip = finfo.absoluteDir().absoluteFilePath( QStringLiteral( "%1_attachments.zip" ).arg( finfo.completeBaseName() ) );
1991 if ( QFile( attachmentsZip ).exists() )
1992 {
1993 auto archive = std::make_unique<QgsArchive>();
1994 if ( archive->unzip( attachmentsZip ) )
1995 {
1996 releaseHandlesToProjectArchive();
1997 mArchive = std::move( archive );
1998 }
1999 }
2000 returnValue = readProjectFile( mFile.fileName(), flags );
2001 }
2002
2003 //on translation we should not change the filename back
2004 if ( !mTranslator )
2005 {
2006 mFile.setFileName( filename );
2007 mCachedHomePath.clear();
2008 mProjectScope.reset();
2009 }
2010 else
2011 {
2012 //but delete the translator
2013 mTranslator.reset( nullptr );
2014 }
2015 }
2016 emit homePathChanged();
2017 return returnValue;
2018}
2019
2020bool QgsProject::readProjectFile( const QString &filename, Qgis::ProjectReadFlags flags )
2021{
2023
2024 // avoid multiple emission of snapping updated signals
2025 ScopedIntIncrementor snapSignalBlock( &mBlockSnappingUpdates );
2026
2027 QFile projectFile( filename );
2028 clearError();
2029
2030 QgsApplication::profiler()->clear( QStringLiteral( "projectload" ) );
2031 QgsScopedRuntimeProfile profile( tr( "Setting up translations" ), QStringLiteral( "projectload" ) );
2032
2033 const QString localeFileName = QStringLiteral( "%1_%2" ).arg( QFileInfo( projectFile.fileName() ).baseName(), QgsApplication::settingsLocaleUserLocale->value() );
2034
2035 if ( QFile( QStringLiteral( "%1/%2.qm" ).arg( QFileInfo( projectFile.fileName() ).absolutePath(), localeFileName ) ).exists() )
2036 {
2037 mTranslator.reset( new QTranslator() );
2038 ( void )mTranslator->load( localeFileName, QFileInfo( projectFile.fileName() ).absolutePath() );
2039 }
2040
2041 profile.switchTask( tr( "Reading project file" ) );
2042 auto doc = std::make_unique<QDomDocument>( QStringLiteral( "qgis" ) );
2043
2044 if ( !projectFile.open( QIODevice::ReadOnly | QIODevice::Text ) )
2045 {
2046 projectFile.close();
2047
2048 setError( tr( "Unable to open %1" ).arg( projectFile.fileName() ) );
2049
2050 return false;
2051 }
2052
2053 QTextStream textStream( &projectFile );
2054#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
2055 textStream.setCodec( "UTF-8" );
2056#endif
2057 QString projectString = textStream.readAll();
2058 projectFile.close();
2059
2060 for ( int i = 0; i < 32; i++ )
2061 {
2062 if ( i == 9 || i == 10 || i == 13 )
2063 {
2064 continue;
2065 }
2066 projectString.replace( QChar( i ), QStringLiteral( "%1%2%1" ).arg( FONTMARKER_CHR_FIX, QString::number( i ) ) );
2067 }
2068
2069 // location of problem associated with errorMsg
2070 int line, column;
2071 QString errorMsg;
2072 if ( !doc->setContent( projectString, &errorMsg, &line, &column ) )
2073 {
2074 const QString errorString = tr( "Project file read error in file %1: %2 at line %3 column %4" )
2075 .arg( projectFile.fileName(), errorMsg ).arg( line ).arg( column );
2076 QgsDebugError( errorString );
2077 setError( errorString );
2078
2079 return false;
2080 }
2081
2082 projectFile.close();
2083
2084 QgsDebugMsgLevel( "Opened document " + projectFile.fileName(), 2 );
2085
2086 // get project version string, if any
2087 const QgsProjectVersion fileVersion = getVersion( *doc );
2088 const QgsProjectVersion thisVersion( Qgis::version() );
2089
2090 profile.switchTask( tr( "Updating project file" ) );
2091 if ( thisVersion > fileVersion )
2092 {
2093 const bool isOlderMajorVersion = fileVersion.majorVersion() < thisVersion.majorVersion();
2094
2095 if ( isOlderMajorVersion )
2096 {
2097 QgsLogger::warning( "Loading a file that was saved with an older "
2098 "version of qgis (saved in " + fileVersion.text() +
2099 ", loaded in " + Qgis::version() +
2100 "). Problems may occur." );
2101 }
2102
2103 QgsProjectFileTransform projectFile( *doc, fileVersion );
2104
2105 // Shows a warning when an old project file is read.
2107 emit oldProjectVersionWarning( fileVersion.text() );
2109 emit readVersionMismatchOccurred( fileVersion.text() );
2110
2111 projectFile.updateRevision( thisVersion );
2112 }
2113 else if ( fileVersion > thisVersion )
2114 {
2115 QgsLogger::warning( "Loading a file that was saved with a newer "
2116 "version of qgis (saved in " + fileVersion.text() +
2117 ", loaded in " + Qgis::version() +
2118 "). Problems may occur." );
2119
2120 emit readVersionMismatchOccurred( fileVersion.text() );
2121 }
2122
2123 // start new project, just keep the file name and auxiliary storage
2124 profile.switchTask( tr( "Creating auxiliary storage" ) );
2125 const QString fileName = mFile.fileName();
2126
2127 const QgsCoordinateReferenceSystem oldVerticalCrs = verticalCrs();
2128 const QgsCoordinateReferenceSystem oldCrs3D = mCrs3D;
2129
2130 // NOTE [ND] -- I suspect this is wrong, as the archive may contain any number of non-auxiliary
2131 // storage related files from the previously loaded project.
2132 std::unique_ptr<QgsAuxiliaryStorage> aStorage = std::move( mAuxiliaryStorage );
2133 std::unique_ptr<QgsArchive> archive = std::move( mArchive );
2134
2135 // don't emit xxxChanged signals during the clear() call, as we'll be emitting
2136 // them again after reading the properties from the project file
2137 mBlockChangeSignalsDuringClear = true;
2138 clear();
2139 mBlockChangeSignalsDuringClear = false;
2140
2141 // this is ugly, but clear() will have created a new archive and started populating it. We
2142 // need to release handles to this archive now as the subsequent call to move will need
2143 // to delete it, and requires free access to do so.
2144 releaseHandlesToProjectArchive();
2145
2146 mAuxiliaryStorage = std::move( aStorage );
2147 mArchive = std::move( archive );
2148
2149 mFile.setFileName( fileName );
2150 mCachedHomePath.clear();
2151 mProjectScope.reset();
2152 mSaveVersion = fileVersion;
2153
2154 // now get any properties
2155 profile.switchTask( tr( "Reading properties" ) );
2156 _getProperties( *doc, mProperties );
2157
2158 // now get the data defined server properties
2159 mDataDefinedServerProperties = getDataDefinedServerProperties( *doc, dataDefinedServerPropertyDefinitions() );
2160
2161 QgsDebugMsgLevel( QString::number( mProperties.count() ) + " properties read", 2 );
2162
2163#if 0
2164 dump_( mProperties );
2165#endif
2166
2167 // get older style project title
2168 QString oldTitle;
2169 _getTitle( *doc, oldTitle );
2170
2171 readProjectFileMetadata( *doc, mSaveUser, mSaveUserFull, mSaveDateTime );
2172
2173 const QDomNodeList homePathNl = doc->elementsByTagName( QStringLiteral( "homePath" ) );
2174 if ( homePathNl.count() > 0 )
2175 {
2176 const QDomElement homePathElement = homePathNl.at( 0 ).toElement();
2177 const QString homePath = homePathElement.attribute( QStringLiteral( "path" ) );
2178 if ( !homePath.isEmpty() )
2180 }
2181 else
2182 {
2183 emit homePathChanged();
2184 }
2185
2186 const QColor backgroundColor( readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorRedPart" ), 255 ),
2187 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorGreenPart" ), 255 ),
2188 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorBluePart" ), 255 ) );
2190 const QColor selectionColor( readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorRedPart" ), 255 ),
2191 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorGreenPart" ), 255 ),
2192 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorBluePart" ), 255 ),
2193 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorAlphaPart" ), 255 ) );
2195
2196
2197 const QString distanceUnitString = readEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/DistanceUnits" ), QString() );
2198 if ( !distanceUnitString.isEmpty() )
2199 setDistanceUnits( QgsUnitTypes::decodeDistanceUnit( distanceUnitString ) );
2200
2201 const QString areaUnitString = readEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/AreaUnits" ), QString() );
2202 if ( !areaUnitString.isEmpty() )
2203 setAreaUnits( QgsUnitTypes::decodeAreaUnit( areaUnitString ) );
2204
2205 setScaleMethod( qgsEnumKeyToValue( readEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/ScaleMethod" ), QString() ), Qgis::ScaleCalculationMethod::HorizontalMiddle ) );
2206
2207 QgsReadWriteContext context;
2208 context.setPathResolver( pathResolver() );
2209 context.setProjectTranslator( this );
2210
2211 //crs
2213 if ( readNumEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectionsEnabled" ), 0 ) )
2214 {
2215 // first preference - dedicated projectCrs node
2216 const QDomNode srsNode = doc->documentElement().namedItem( QStringLiteral( "projectCrs" ) );
2217 if ( !srsNode.isNull() )
2218 {
2219 projectCrs.readXml( srsNode );
2220 }
2221
2222 if ( !projectCrs.isValid() )
2223 {
2224 const QString projCrsString = readEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectCRSProj4String" ) );
2225 const long currentCRS = readNumEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectCRSID" ), -1 );
2226 const QString authid = readEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectCrs" ) );
2227
2228 // authid should be prioritized over all
2229 const bool isUserAuthId = authid.startsWith( QLatin1String( "USER:" ), Qt::CaseInsensitive );
2230 if ( !authid.isEmpty() && !isUserAuthId )
2231 projectCrs = QgsCoordinateReferenceSystem( authid );
2232
2233 // try the CRS
2234 if ( !projectCrs.isValid() && currentCRS >= 0 )
2235 {
2236 projectCrs = QgsCoordinateReferenceSystem::fromSrsId( currentCRS );
2237 }
2238
2239 // if that didn't produce a match, try the proj.4 string
2240 if ( !projCrsString.isEmpty() && ( authid.isEmpty() || isUserAuthId ) && ( !projectCrs.isValid() || projectCrs.toProj() != projCrsString ) )
2241 {
2242 projectCrs = QgsCoordinateReferenceSystem::fromProj( projCrsString );
2243 }
2244
2245 // last just take the given id
2246 if ( !projectCrs.isValid() )
2247 {
2248 projectCrs = QgsCoordinateReferenceSystem::fromSrsId( currentCRS );
2249 }
2250 }
2251 }
2252 mCrs = projectCrs;
2253
2254 //vertical CRS
2255 {
2257 const QDomNode verticalCrsNode = doc->documentElement().namedItem( QStringLiteral( "verticalCrs" ) );
2258 if ( !verticalCrsNode.isNull() )
2259 {
2260 verticalCrs.readXml( verticalCrsNode );
2261 }
2262 mVerticalCrs = verticalCrs;
2263 }
2264 rebuildCrs3D();
2265
2266 QStringList datumErrors;
2267 if ( !mTransformContext.readXml( doc->documentElement(), context, datumErrors ) && !datumErrors.empty() )
2268 {
2269 emit missingDatumTransforms( datumErrors );
2270 }
2272
2273 // map shading
2274 const QDomNode elevationShadingNode = doc->documentElement().namedItem( QStringLiteral( "elevation-shading-renderer" ) );
2275 if ( !elevationShadingNode.isNull() )
2276 {
2277 mElevationShadingRenderer.readXml( elevationShadingNode.toElement(), context );
2278 }
2280
2281
2282 //add variables defined in project file - do this early in the reading cycle, as other components
2283 //(e.g. layouts) may depend on these variables
2284 const QStringList variableNames = readListEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableNames" ) );
2285 const QStringList variableValues = readListEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableValues" ) );
2286
2287 mCustomVariables.clear();
2288 if ( variableNames.length() == variableValues.length() )
2289 {
2290 for ( int i = 0; i < variableNames.length(); ++i )
2291 {
2292 mCustomVariables.insert( variableNames.at( i ), variableValues.at( i ) );
2293 }
2294 }
2295 else
2296 {
2297 QgsMessageLog::logMessage( tr( "Project Variables Invalid" ), tr( "The project contains invalid variable settings." ) );
2298 }
2299
2300 // Register expression functions stored in the project.
2301 // They might be using project variables and might be
2302 // in turn being used by other components (e.g., layouts).
2304
2305 QDomElement element = doc->documentElement().firstChildElement( QStringLiteral( "projectMetadata" ) );
2306
2307 if ( !element.isNull() )
2308 {
2309 mMetadata.readMetadataXml( element );
2310 }
2311 else
2312 {
2313 // older project, no metadata => remove auto generated metadata which is populated on QgsProject::clear()
2314 mMetadata = QgsProjectMetadata();
2315 }
2316 if ( mMetadata.title().isEmpty() && !oldTitle.isEmpty() )
2317 {
2318 // upgrade older title storage to storing within project metadata.
2319 mMetadata.setTitle( oldTitle );
2320 }
2321 emit metadataChanged();
2322
2323 // Transaction mode
2324 element = doc->documentElement().firstChildElement( QStringLiteral( "transaction" ) );
2325 if ( !element.isNull() )
2326 {
2327 mTransactionMode = qgsEnumKeyToValue( element.attribute( QStringLiteral( "mode" ) ), Qgis::TransactionMode::Disabled );
2328 }
2329 else
2330 {
2331 // maybe older project => try read autotransaction
2332 element = doc->documentElement().firstChildElement( QStringLiteral( "autotransaction" ) );
2333 if ( ! element.isNull() )
2334 {
2335 mTransactionMode = static_cast<Qgis::TransactionMode>( element.attribute( QStringLiteral( "active" ), QStringLiteral( "0" ) ).toInt() );
2336 }
2337 }
2338
2339 // read the layer tree from project file
2340 profile.switchTask( tr( "Loading layer tree" ) );
2341 mRootGroup->setCustomProperty( QStringLiteral( "loading" ), 1 );
2342
2343 QDomElement layerTreeElem = doc->documentElement().firstChildElement( QStringLiteral( "layer-tree-group" ) );
2344 if ( !layerTreeElem.isNull() )
2345 {
2346 // Use a temporary tree to read the nodes to prevent signals being delivered to the models
2347 QgsLayerTree tempTree;
2348 tempTree.readChildrenFromXml( layerTreeElem, context );
2349 mRootGroup->insertChildNodes( -1, tempTree.abandonChildren() );
2350 }
2351 else
2352 {
2353 QgsLayerTreeUtils::readOldLegend( mRootGroup, doc->documentElement().firstChildElement( QStringLiteral( "legend" ) ) );
2354 }
2355
2356 mLayerTreeRegistryBridge->setEnabled( false );
2357
2358 // get the map layers
2359 profile.switchTask( tr( "Reading map layers" ) );
2360
2361 loadProjectFlags( doc.get() );
2362
2363 QList<QDomNode> brokenNodes;
2364 const bool clean = _getMapLayers( *doc, brokenNodes, flags );
2365
2366 // review the integrity of the retrieved map layers
2367 if ( !clean && !( flags & Qgis::ProjectReadFlag::DontResolveLayers ) )
2368 {
2369 QgsDebugError( QStringLiteral( "Unable to get map layers from project file." ) );
2370
2371 if ( !brokenNodes.isEmpty() )
2372 {
2373 QgsDebugError( "there are " + QString::number( brokenNodes.size() ) + " broken layers" );
2374 }
2375
2376 // we let a custom handler decide what to do with missing layers
2377 // (default implementation ignores them, there's also a GUI handler that lets user choose correct path)
2378 mBadLayerHandler->handleBadLayers( brokenNodes );
2379 }
2380
2381 mMainAnnotationLayer->readLayerXml( doc->documentElement().firstChildElement( QStringLiteral( "main-annotation-layer" ) ), context );
2382 mMainAnnotationLayer->setTransformContext( mTransformContext );
2383
2384 // load embedded groups and layers
2385 profile.switchTask( tr( "Loading embedded layers" ) );
2386 loadEmbeddedNodes( mRootGroup, flags );
2387
2388 // Resolve references to other layers
2389 // Needs to be done here once all dependent layers are loaded
2390 profile.switchTask( tr( "Resolving layer references" ) );
2391 QMap<QString, QgsMapLayer *> layers = mLayerStore->mapLayers();
2392 for ( QMap<QString, QgsMapLayer *>::iterator it = layers.begin(); it != layers.end(); ++it )
2393 {
2394 it.value()->resolveReferences( this );
2395 }
2396 mMainAnnotationLayer->resolveReferences( this );
2397
2398 mLayerTreeRegistryBridge->setEnabled( true );
2399
2400 // now that layers are loaded, we can resolve layer tree's references to the layers
2401 profile.switchTask( tr( "Resolving references" ) );
2402 mRootGroup->resolveReferences( this );
2403
2404 // we need to migrate old fashion designed QgsSymbolLayerReference to new ones
2405 if ( QgsProjectVersion( 3, 28, 0 ) > mSaveVersion )
2406 {
2410 }
2411
2412 if ( !layerTreeElem.isNull() )
2413 {
2414 mRootGroup->readLayerOrderFromXml( layerTreeElem );
2415 }
2416
2417 // Load pre 3.0 configuration
2418 const QDomElement layerTreeCanvasElem = doc->documentElement().firstChildElement( QStringLiteral( "layer-tree-canvas" ) );
2419 if ( !layerTreeCanvasElem.isNull( ) )
2420 {
2421 mRootGroup->readLayerOrderFromXml( layerTreeCanvasElem );
2422 }
2423
2424 // Convert pre 3.4 to create layers flags
2425 if ( QgsProjectVersion( 3, 4, 0 ) > mSaveVersion )
2426 {
2427 const QStringList requiredLayerIds = readListEntry( QStringLiteral( "RequiredLayers" ), QStringLiteral( "Layers" ) );
2428 for ( const QString &layerId : requiredLayerIds )
2429 {
2430 if ( QgsMapLayer *layer = mapLayer( layerId ) )
2431 {
2432 layer->setFlags( layer->flags() & ~QgsMapLayer::Removable );
2433 }
2434 }
2435 const QStringList disabledLayerIds = readListEntry( QStringLiteral( "Identify" ), QStringLiteral( "/disabledLayers" ) );
2436 for ( const QString &layerId : disabledLayerIds )
2437 {
2438 if ( QgsMapLayer *layer = mapLayer( layerId ) )
2439 {
2440 layer->setFlags( layer->flags() & ~QgsMapLayer::Identifiable );
2441 }
2442 }
2443 }
2444
2445 // Convert pre 3.26 default styles
2446 if ( QgsProjectVersion( 3, 26, 0 ) > mSaveVersion )
2447 {
2448 // Convert default symbols
2449 QString styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Marker" ) );
2450 if ( !styleName.isEmpty() )
2451 {
2452 std::unique_ptr<QgsSymbol> symbol( QgsStyle::defaultStyle()->symbol( styleName ) );
2454 }
2455 styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Line" ) );
2456 if ( !styleName.isEmpty() )
2457 {
2458 std::unique_ptr<QgsSymbol> symbol( QgsStyle::defaultStyle()->symbol( styleName ) );
2460 }
2461 styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Fill" ) );
2462 if ( !styleName.isEmpty() )
2463 {
2464 std::unique_ptr<QgsSymbol> symbol( QgsStyle::defaultStyle()->symbol( styleName ) );
2466 }
2467 styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/ColorRamp" ) );
2468 if ( !styleName.isEmpty() )
2469 {
2470 std::unique_ptr<QgsColorRamp> colorRamp( QgsStyle::defaultStyle()->colorRamp( styleName ) );
2471 styleSettings()->setDefaultColorRamp( colorRamp.get() );
2472 }
2473
2474 // Convert randomize default symbol fill color
2475 styleSettings()->setRandomizeDefaultSymbolColor( readBoolEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/RandomColors" ), true ) );
2476
2477 // Convert default symbol opacity
2478 double opacity = 1.0;
2479 bool ok = false;
2480 // upgrade old setting
2481 double alpha = readDoubleEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/AlphaInt" ), 255, &ok );
2482 if ( ok )
2483 opacity = alpha / 255.0;
2484 double newOpacity = readDoubleEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Opacity" ), 1.0, &ok );
2485 if ( ok )
2486 opacity = newOpacity;
2488
2489 // Cleanup
2490 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Marker" ) );
2491 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Line" ) );
2492 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Fill" ) );
2493 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/ColorRamp" ) );
2494 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/RandomColors" ) );
2495 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/AlphaInt" ) );
2496 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Opacity" ) );
2497 }
2498
2499 // After bad layer handling we might still have invalid layers,
2500 // store them in case the user wanted to handle them later
2501 // or wanted to pass them through when saving
2503 {
2504 profile.switchTask( tr( "Storing original layer properties" ) );
2505 QgsLayerTreeUtils::storeOriginalLayersProperties( mRootGroup, doc.get() );
2506 }
2507
2508 mRootGroup->removeCustomProperty( QStringLiteral( "loading" ) );
2509
2510 profile.switchTask( tr( "Loading map themes" ) );
2511 mMapThemeCollection.reset( new QgsMapThemeCollection( this ) );
2513 mMapThemeCollection->readXml( *doc );
2514
2515 profile.switchTask( tr( "Loading label settings" ) );
2516 mLabelingEngineSettings->readSettingsFromProject( this );
2517 {
2518 const QDomElement labelEngineSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "labelEngineSettings" ) );
2519 mLabelingEngineSettings->readXml( labelEngineSettingsElement, context );
2520 }
2521 mLabelingEngineSettings->resolveReferences( this );
2522
2524
2525 profile.switchTask( tr( "Loading annotations" ) );
2527 {
2528 mAnnotationManager->readXml( doc->documentElement(), context );
2529 }
2530 else
2531 {
2532 mAnnotationManager->readXmlAndUpgradeToAnnotationLayerItems( doc->documentElement(), context, mMainAnnotationLayer, mTransformContext );
2533 }
2535 {
2536 profile.switchTask( tr( "Loading layouts" ) );
2537 mLayoutManager->readXml( doc->documentElement(), *doc );
2538 }
2539
2541 {
2542 profile.switchTask( tr( "Loading 3D Views" ) );
2543 m3DViewsManager->readXml( doc->documentElement(), *doc );
2544 }
2545
2546 profile.switchTask( tr( "Loading bookmarks" ) );
2547 mBookmarkManager->readXml( doc->documentElement(), *doc );
2548
2549 profile.switchTask( tr( "Loading sensors" ) );
2550 mSensorManager->readXml( doc->documentElement(), *doc );
2551
2552 // reassign change dependencies now that all layers are loaded
2553 QMap<QString, QgsMapLayer *> existingMaps = mapLayers();
2554 for ( QMap<QString, QgsMapLayer *>::iterator it = existingMaps.begin(); it != existingMaps.end(); ++it )
2555 {
2556 it.value()->setDependencies( it.value()->dependencies() );
2557 }
2558
2559 profile.switchTask( tr( "Loading snapping settings" ) );
2560 mSnappingConfig.readProject( *doc );
2561 mAvoidIntersectionsMode = static_cast<Qgis::AvoidIntersectionsMode>( readNumEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsMode" ), static_cast<int>( Qgis::AvoidIntersectionsMode::AvoidIntersectionsLayers ) ) );
2562
2563 profile.switchTask( tr( "Loading view settings" ) );
2564 // restore older project scales settings
2565 mViewSettings->setUseProjectScales( readBoolEntry( QStringLiteral( "Scales" ), QStringLiteral( "/useProjectScales" ) ) );
2566 const QStringList scales = readListEntry( QStringLiteral( "Scales" ), QStringLiteral( "/ScalesList" ) );
2567 QVector<double> res;
2568 for ( const QString &scale : scales )
2569 {
2570 const QStringList parts = scale.split( ':' );
2571 if ( parts.size() != 2 )
2572 continue;
2573
2574 bool ok = false;
2575 const double denominator = QLocale().toDouble( parts[1], &ok );
2576 if ( ok )
2577 {
2578 res << denominator;
2579 }
2580 }
2581 mViewSettings->setMapScales( res );
2582 const QDomElement viewSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectViewSettings" ) );
2583 if ( !viewSettingsElement.isNull() )
2584 mViewSettings->readXml( viewSettingsElement, context );
2585
2586 // restore style settings
2587 profile.switchTask( tr( "Loading style properties" ) );
2588 const QDomElement styleSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectStyleSettings" ) );
2589 if ( !styleSettingsElement.isNull() )
2590 {
2591 mStyleSettings->removeProjectStyle();
2592 mStyleSettings->readXml( styleSettingsElement, context, flags );
2593 }
2594
2595 // restore time settings
2596 profile.switchTask( tr( "Loading temporal settings" ) );
2597 const QDomElement timeSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectTimeSettings" ) );
2598 if ( !timeSettingsElement.isNull() )
2599 mTimeSettings->readXml( timeSettingsElement, context );
2600
2601
2602 profile.switchTask( tr( "Loading elevation properties" ) );
2603 const QDomElement elevationPropertiesElement = doc->documentElement().firstChildElement( QStringLiteral( "ElevationProperties" ) );
2604 if ( !elevationPropertiesElement.isNull() )
2605 mElevationProperties->readXml( elevationPropertiesElement, context );
2606 mElevationProperties->resolveReferences( this );
2607
2608 profile.switchTask( tr( "Loading display settings" ) );
2609 {
2610 const QDomElement displaySettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectDisplaySettings" ) );
2611 if ( !displaySettingsElement.isNull() )
2612 mDisplaySettings->readXml( displaySettingsElement, context );
2613 }
2614
2615 profile.switchTask( tr( "Loading GPS settings" ) );
2616 {
2617 const QDomElement gpsSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectGpsSettings" ) );
2618 if ( !gpsSettingsElement.isNull() )
2619 mGpsSettings->readXml( gpsSettingsElement, context );
2620 mGpsSettings->resolveReferences( this );
2621 }
2622
2623 profile.switchTask( tr( "Updating variables" ) );
2625 profile.switchTask( tr( "Updating CRS" ) );
2626 emit crsChanged();
2627 if ( verticalCrs() != oldVerticalCrs )
2628 emit verticalCrsChanged();
2629 if ( mCrs3D != oldCrs3D )
2630 emit crs3DChanged();
2631 emit ellipsoidChanged( ellipsoid() );
2632
2633 // read the project: used by map canvas and legend
2634 profile.switchTask( tr( "Reading external settings" ) );
2635 emit readProject( *doc );
2636 emit readProjectWithContext( *doc, context );
2637
2638 profile.switchTask( tr( "Updating interface" ) );
2639
2640 snapSignalBlock.release();
2641 if ( !mBlockSnappingUpdates )
2642 emit snappingConfigChanged( mSnappingConfig );
2643
2646 emit projectColorsChanged();
2647
2648 // if all went well, we're allegedly in pristine state
2649 if ( clean )
2650 setDirty( false );
2651
2652 QgsDebugMsgLevel( QStringLiteral( "Project save user: %1" ).arg( mSaveUser ), 2 );
2653 QgsDebugMsgLevel( QStringLiteral( "Project save user: %1" ).arg( mSaveUserFull ), 2 );
2654
2658
2659 if ( mTranslator )
2660 {
2661 //project possibly translated -> rename it with locale postfix
2662 const QString newFileName( QStringLiteral( "%1/%2.qgs" ).arg( QFileInfo( projectFile.fileName() ).absolutePath(), localeFileName ) );
2663 setFileName( newFileName );
2664
2665 if ( write() )
2666 {
2667 setTitle( localeFileName );
2668 QgsMessageLog::logMessage( tr( "Translated project saved with locale prefix %1" ).arg( newFileName ), QObject::tr( "Project translation" ), Qgis::MessageLevel::Success );
2669 }
2670 else
2671 {
2672 QgsMessageLog::logMessage( tr( "Error saving translated project with locale prefix %1" ).arg( newFileName ), QObject::tr( "Project translation" ), Qgis::MessageLevel::Critical );
2673 }
2674 }
2675
2676 // lastly, make any previously editable layers editable
2677 const QMap<QString, QgsMapLayer *> loadedLayers = mapLayers();
2678 for ( auto it = loadedLayers.constBegin(); it != loadedLayers.constEnd(); ++it )
2679 {
2680 if ( it.value()->isValid() && it.value()->customProperty( QStringLiteral( "_layer_was_editable" ) ).toBool() )
2681 {
2682 if ( QgsVectorLayer *vl = qobject_cast< QgsVectorLayer * >( it.value() ) )
2683 vl->startEditing();
2684 it.value()->removeCustomProperty( QStringLiteral( "_layer_was_editable" ) );
2685 }
2686 }
2687
2688 return true;
2689}
2690
2691bool QgsProject::loadEmbeddedNodes( QgsLayerTreeGroup *group, Qgis::ProjectReadFlags flags )
2692{
2694
2695 bool valid = true;
2696 const auto constChildren = group->children();
2697 for ( QgsLayerTreeNode *child : constChildren )
2698 {
2699 if ( QgsLayerTree::isGroup( child ) )
2700 {
2701 QgsLayerTreeGroup *childGroup = QgsLayerTree::toGroup( child );
2702 if ( childGroup->customProperty( QStringLiteral( "embedded" ) ).toInt() )
2703 {
2704 // make sure to convert the path from relative to absolute
2705 const QString projectPath = readPath( childGroup->customProperty( QStringLiteral( "embedded_project" ) ).toString() );
2706 childGroup->setCustomProperty( QStringLiteral( "embedded_project" ), projectPath );
2707 QgsLayerTreeGroup *newGroup = createEmbeddedGroup( childGroup->name(), projectPath, childGroup->customProperty( QStringLiteral( "embedded-invisible-layers" ) ).toStringList(), flags );
2708 if ( newGroup )
2709 {
2710 QList<QgsLayerTreeNode *> clonedChildren;
2711 const QList<QgsLayerTreeNode *> constChildren = newGroup->children();
2712 clonedChildren.reserve( constChildren.size() );
2713 for ( QgsLayerTreeNode *newGroupChild : constChildren )
2714 clonedChildren << newGroupChild->clone();
2715 delete newGroup;
2716
2717 childGroup->insertChildNodes( 0, clonedChildren );
2718 }
2719 }
2720 else
2721 {
2722 loadEmbeddedNodes( childGroup, flags );
2723 }
2724 }
2725 else if ( QgsLayerTree::isLayer( child ) )
2726 {
2727 if ( child->customProperty( QStringLiteral( "embedded" ) ).toInt() )
2728 {
2729 QList<QDomNode> brokenNodes;
2730 if ( ! createEmbeddedLayer( QgsLayerTree::toLayer( child )->layerId(), readPath( child->customProperty( QStringLiteral( "embedded_project" ) ).toString() ), brokenNodes, true, flags ) )
2731 {
2732 valid = valid && false;
2733 }
2734 }
2735 }
2736
2737 }
2738
2739 return valid;
2740}
2741
2743{
2744 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
2746
2747 return mCustomVariables;
2748}
2749
2750void QgsProject::setCustomVariables( const QVariantMap &variables )
2751{
2753
2754 if ( variables == mCustomVariables )
2755 return;
2756
2757 //write variable to project
2758 QStringList variableNames;
2759 QStringList variableValues;
2760
2761 QVariantMap::const_iterator it = variables.constBegin();
2762 for ( ; it != variables.constEnd(); ++it )
2763 {
2764 variableNames << it.key();
2765 variableValues << it.value().toString();
2766 }
2767
2768 writeEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableNames" ), variableNames );
2769 writeEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableValues" ), variableValues );
2770
2771 mCustomVariables = variables;
2772 mProjectScope.reset();
2773
2775}
2776
2778{
2780
2781 *mLabelingEngineSettings = settings;
2783}
2784
2786{
2788
2789 return *mLabelingEngineSettings;
2790}
2791
2793{
2795
2796 mProjectScope.reset();
2797 return mLayerStore.get();
2798}
2799
2801{
2803
2804 return mLayerStore.get();
2805}
2806
2807QList<QgsVectorLayer *> QgsProject::avoidIntersectionsLayers() const
2808{
2810
2811 QList<QgsVectorLayer *> layers;
2812 const QStringList layerIds = readListEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsList" ), QStringList() );
2813 const auto constLayerIds = layerIds;
2814 for ( const QString &layerId : constLayerIds )
2815 {
2816 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mapLayer( layerId ) ) )
2817 layers << vlayer;
2818 }
2819 return layers;
2820}
2821
2822void QgsProject::setAvoidIntersectionsLayers( const QList<QgsVectorLayer *> &layers )
2823{
2825
2826 QStringList list;
2827 list.reserve( layers.size() );
2828
2829 for ( QgsVectorLayer *layer : layers )
2830 {
2831 if ( layer->geometryType() == Qgis::GeometryType::Polygon )
2832 list << layer->id();
2833 }
2834
2835 writeEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsList" ), list );
2837}
2838
2850
2852{
2853 // this method is called quite extensively using QgsProject::instance() skip-keyword-check
2855
2856 // MUCH cheaper to clone than build
2857 if ( mProjectScope )
2858 {
2859 auto projectScope = std::make_unique< QgsExpressionContextScope >( *mProjectScope );
2860
2861 // we can't cache these variables
2862 projectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_distance_units" ), QgsUnitTypes::toString( distanceUnits() ), true, true ) );
2863 projectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_area_units" ), QgsUnitTypes::toString( areaUnits() ), true, true ) );
2864
2865 // neither this function
2866 projectScope->addFunction( QStringLiteral( "sensor_data" ), new GetSensorData( sensorManager()->sensorsData() ) );
2867
2868 return projectScope.release();
2869 }
2870
2871 mProjectScope = std::make_unique< QgsExpressionContextScope >( QObject::tr( "Project" ) );
2872
2873 const QVariantMap vars = customVariables();
2874
2875 QVariantMap::const_iterator it = vars.constBegin();
2876
2877 for ( ; it != vars.constEnd(); ++it )
2878 {
2879 mProjectScope->setVariable( it.key(), it.value(), true );
2880 }
2881
2882 QString projectPath = projectStorage() ? fileName() : absoluteFilePath();
2883 if ( projectPath.isEmpty() )
2884 projectPath = mOriginalPath;
2885 const QString projectFolder = QFileInfo( projectPath ).path();
2886 const QString projectFilename = QFileInfo( projectPath ).fileName();
2887 const QString projectBasename = baseName();
2888
2889 //add other known project variables
2890 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_title" ), title(), true, true ) );
2891 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_path" ), QDir::toNativeSeparators( projectPath ), true, true ) );
2892 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_folder" ), QDir::toNativeSeparators( projectFolder ), true, true ) );
2893 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_filename" ), projectFilename, true, true ) );
2894 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_basename" ), projectBasename, true, true ) );
2895 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_home" ), QDir::toNativeSeparators( homePath() ), true, true ) );
2896 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_last_saved" ), mSaveDateTime.isNull() ? QVariant() : QVariant( mSaveDateTime ), true, true ) );
2897
2898 const QgsCoordinateReferenceSystem projectCrs = crs();
2899 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs" ), projectCrs.authid(), true, true ) );
2900 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_definition" ), projectCrs.toProj(), true, true ) );
2901 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_description" ), projectCrs.description(), true, true ) );
2902 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_acronym" ), projectCrs.projectionAcronym(), true ) );
2903 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_ellipsoid" ), projectCrs.ellipsoidAcronym(), true ) );
2904 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_proj4" ), projectCrs.toProj(), true ) );
2905 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_wkt" ), projectCrs.toWkt( Qgis::CrsWktVariant::Preferred ), true ) );
2906
2907 const QgsCoordinateReferenceSystem projectVerticalCrs = QgsProject::verticalCrs();
2908 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs" ), projectVerticalCrs.authid(), true, true ) );
2909 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs_definition" ), projectVerticalCrs.toProj(), true, true ) );
2910 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs_description" ), projectVerticalCrs.description(), true, true ) );
2911 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs_wkt" ), projectVerticalCrs.toWkt( Qgis::CrsWktVariant::Preferred ), true ) );
2912
2913 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_ellipsoid" ), ellipsoid(), true, true ) );
2914 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "_project_transform_context" ), QVariant::fromValue<QgsCoordinateTransformContext>( transformContext() ), true, true ) );
2915 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_units" ), QgsUnitTypes::toString( projectCrs.mapUnits() ), true ) );
2916
2917 // metadata
2918 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_author" ), metadata().author(), true, true ) );
2919 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_abstract" ), metadata().abstract(), true, true ) );
2920 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_creation_date" ), metadata().creationDateTime(), true, true ) );
2921 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_identifier" ), metadata().identifier(), true, true ) );
2922
2923 // keywords
2924 QVariantMap keywords;
2925 const QgsAbstractMetadataBase::KeywordMap metadataKeywords = metadata().keywords();
2926 for ( auto it = metadataKeywords.constBegin(); it != metadataKeywords.constEnd(); ++it )
2927 {
2928 keywords.insert( it.key(), it.value() );
2929 }
2930 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_keywords" ), keywords, true, true ) );
2931
2932 // layers
2933 QVariantList layersIds;
2934 QVariantList layers;
2935 const QMap<QString, QgsMapLayer *> layersInProject = mLayerStore->mapLayers();
2936 layersIds.reserve( layersInProject.count() );
2937 layers.reserve( layersInProject.count() );
2938 for ( auto it = layersInProject.constBegin(); it != layersInProject.constEnd(); ++it )
2939 {
2940 layersIds << it.value()->id();
2941 layers << QVariant::fromValue<QgsWeakMapLayerPointer>( QgsWeakMapLayerPointer( it.value() ) );
2942 }
2943 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "layer_ids" ), layersIds, true ) );
2944 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "layers" ), layers, true ) );
2945
2946 mProjectScope->addFunction( QStringLiteral( "project_color" ), new GetNamedProjectColor( this ) );
2947 mProjectScope->addFunction( QStringLiteral( "project_color_object" ), new GetNamedProjectColorObject( this ) );
2948
2950}
2951
2952void QgsProject::onMapLayersAdded( const QList<QgsMapLayer *> &layers )
2953{
2955
2956 const QMap<QString, QgsMapLayer *> existingMaps = mapLayers();
2957
2958 const auto constLayers = layers;
2959 for ( QgsMapLayer *layer : constLayers )
2960 {
2961 if ( ! layer->isValid() )
2962 return;
2963
2964 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer ) )
2965 {
2966 vlayer->setReadExtentFromXml( mFlags & Qgis::ProjectFlag::TrustStoredLayerStatistics );
2967 if ( vlayer->dataProvider() )
2968 vlayer->dataProvider()->setProviderProperty( QgsVectorDataProvider::EvaluateDefaultValues,
2970 }
2971
2972 connect( layer, &QgsMapLayer::configChanged, this, [this] { setDirty(); } );
2973
2974 // check if we have to update connections for layers with dependencies
2975 for ( QMap<QString, QgsMapLayer *>::const_iterator it = existingMaps.cbegin(); it != existingMaps.cend(); ++it )
2976 {
2977 const QSet<QgsMapLayerDependency> deps = it.value()->dependencies();
2978 if ( deps.contains( layer->id() ) )
2979 {
2980 // reconnect to change signals
2981 it.value()->setDependencies( deps );
2982 }
2983 }
2984 }
2985
2986 updateTransactionGroups();
2987
2988 if ( !mBlockSnappingUpdates && mSnappingConfig.addLayers( layers ) )
2989 emit snappingConfigChanged( mSnappingConfig );
2990}
2991
2992void QgsProject::onMapLayersRemoved( const QList<QgsMapLayer *> &layers )
2993{
2995
2996 if ( !mBlockSnappingUpdates && mSnappingConfig.removeLayers( layers ) )
2997 emit snappingConfigChanged( mSnappingConfig );
2998
2999 for ( QgsMapLayer *layer : layers )
3000 {
3001 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
3002 if ( ! vlayer )
3003 continue;
3004
3005 mEditBufferGroup.removeLayer( vlayer );
3006 }
3007}
3008
3009void QgsProject::cleanTransactionGroups( bool force )
3010{
3012
3013 bool changed = false;
3014 for ( QMap< QPair< QString, QString>, QgsTransactionGroup *>::Iterator tg = mTransactionGroups.begin(); tg != mTransactionGroups.end(); )
3015 {
3016 if ( tg.value()->isEmpty() || force )
3017 {
3018 delete tg.value();
3019 tg = mTransactionGroups.erase( tg );
3020 changed = true;
3021 }
3022 else
3023 {
3024 ++tg;
3025 }
3026 }
3027 if ( changed )
3029}
3030
3031void QgsProject::updateTransactionGroups()
3032{
3034
3035 mEditBufferGroup.clear();
3036
3037 switch ( mTransactionMode )
3038 {
3040 {
3041 cleanTransactionGroups( true );
3042 return;
3043 }
3044 break;
3046 cleanTransactionGroups( true );
3047 break;
3049 cleanTransactionGroups( false );
3050 break;
3051 }
3052
3053 bool tgChanged = false;
3054 const auto constLayers = mapLayers().values();
3055 for ( QgsMapLayer *layer : constLayers )
3056 {
3057 if ( ! layer->isValid() )
3058 continue;
3059
3060 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
3061 if ( ! vlayer )
3062 continue;
3063
3064 switch ( mTransactionMode )
3065 {
3067 Q_ASSERT( false );
3068 break;
3070 {
3072 {
3073 const QString connString = QgsTransaction::connectionString( vlayer->source() );
3074 const QString key = vlayer->providerType();
3075
3076 QgsTransactionGroup *tg = mTransactionGroups.value( qMakePair( key, connString ) );
3077
3078 if ( !tg )
3079 {
3080 tg = new QgsTransactionGroup();
3081 mTransactionGroups.insert( qMakePair( key, connString ), tg );
3082 tgChanged = true;
3083 }
3084 tg->addLayer( vlayer );
3085 }
3086 }
3087 break;
3089 {
3090 if ( vlayer->supportsEditing() )
3091 mEditBufferGroup.addLayer( vlayer );
3092 }
3093 break;
3094 }
3095 }
3096
3097 if ( tgChanged )
3099}
3100
3101bool QgsProject::readLayer( const QDomNode &layerNode )
3102{
3104
3105 QgsReadWriteContext context;
3106 context.setPathResolver( pathResolver() );
3107 context.setProjectTranslator( this );
3108 context.setTransformContext( transformContext() );
3109 QList<QDomNode> brokenNodes;
3110 if ( addLayer( layerNode.toElement(), brokenNodes, context ) )
3111 {
3112 // have to try to update joins for all layers now - a previously added layer may be dependent on this newly
3113 // added layer for joins
3114 const QVector<QgsVectorLayer *> vectorLayers = layers<QgsVectorLayer *>();
3115 for ( QgsVectorLayer *layer : vectorLayers )
3116 {
3117 // TODO: should be only done later - and with all layers (other layers may have referenced this layer)
3118 layer->resolveReferences( this );
3119
3120 if ( layer->isValid() && layer->customProperty( QStringLiteral( "_layer_was_editable" ) ).toBool() )
3121 {
3122 layer->startEditing();
3123 layer->removeCustomProperty( QStringLiteral( "_layer_was_editable" ) );
3124 }
3125 }
3126 return true;
3127 }
3128 return false;
3129}
3130
3131bool QgsProject::write( const QString &filename )
3132{
3134
3135 mFile.setFileName( filename );
3136 mCachedHomePath.clear();
3137 return write();
3138}
3139
3141{
3143
3144 mProjectScope.reset();
3145 if ( QgsProjectStorage *storage = projectStorage() )
3146 {
3147 QgsReadWriteContext context;
3148 // for projects stored in a custom storage, we have to check for the support
3149 // of relative paths since the storage most likely will not be in a file system
3150 const QString storageFilePath { storage->filePath( mFile.fileName() ) };
3151 if ( storageFilePath.isEmpty() )
3152 {
3154 }
3155 context.setPathResolver( pathResolver() );
3156
3157 const QString tempPath = QStandardPaths::standardLocations( QStandardPaths::TempLocation ).at( 0 );
3158 const QString tmpZipFilename( tempPath + QDir::separator() + QUuid::createUuid().toString() );
3159
3160 if ( !zip( tmpZipFilename ) )
3161 return false; // zip() already calls setError() when returning false
3162
3163 QFile tmpZipFile( tmpZipFilename );
3164 if ( !tmpZipFile.open( QIODevice::ReadOnly ) )
3165 {
3166 setError( tr( "Unable to read file %1" ).arg( tmpZipFilename ) );
3167 return false;
3168 }
3169
3171 if ( !storage->writeProject( mFile.fileName(), &tmpZipFile, context ) )
3172 {
3173 QString err = tr( "Unable to save project to storage %1" ).arg( mFile.fileName() );
3174 QList<QgsReadWriteContext::ReadWriteMessage> messages = context.takeMessages();
3175 if ( !messages.isEmpty() )
3176 err += QStringLiteral( "\n\n" ) + messages.last().message();
3177 setError( err );
3178 return false;
3179 }
3180
3181 tmpZipFile.close();
3182 QFile::remove( tmpZipFilename );
3183
3184 return true;
3185 }
3186
3187 if ( QgsZipUtils::isZipFile( mFile.fileName() ) )
3188 {
3189 return zip( mFile.fileName() );
3190 }
3191 else
3192 {
3193 // write project file even if the auxiliary storage is not correctly
3194 // saved
3195 const bool asOk = saveAuxiliaryStorage();
3196 const bool writeOk = writeProjectFile( mFile.fileName() );
3197 bool attachmentsOk = true;
3198 if ( !mArchive->files().isEmpty() )
3199 {
3200 const QFileInfo finfo( mFile.fileName() );
3201 const QString attachmentsZip = finfo.absoluteDir().absoluteFilePath( QStringLiteral( "%1_attachments.zip" ).arg( finfo.completeBaseName() ) );
3202 attachmentsOk = mArchive->zip( attachmentsZip );
3203 }
3204
3205 // errors raised during writing project file are more important
3206 if ( ( !asOk || !attachmentsOk ) && writeOk )
3207 {
3208 QStringList errorMessage;
3209 if ( !asOk )
3210 {
3211 const QString err = mAuxiliaryStorage->errorString();
3212 errorMessage.append( tr( "Unable to save auxiliary storage ('%1')" ).arg( err ) );
3213 }
3214 if ( !attachmentsOk )
3215 {
3216 errorMessage.append( tr( "Unable to save attachments archive" ) );
3217 }
3218 setError( errorMessage.join( '\n' ) );
3219 }
3220
3221 return asOk && writeOk && attachmentsOk;
3222 }
3223}
3224
3225bool QgsProject::writeProjectFile( const QString &filename )
3226{
3228
3229 QFile projectFile( filename );
3230 clearError();
3231
3232 // if we have problems creating or otherwise writing to the project file,
3233 // let's find out up front before we go through all the hand-waving
3234 // necessary to create all the Dom objects
3235 const QFileInfo myFileInfo( projectFile );
3236 if ( myFileInfo.exists() && !myFileInfo.isWritable() )
3237 {
3238 setError( tr( "%1 is not writable. Please adjust permissions (if possible) and try again." )
3239 .arg( projectFile.fileName() ) );
3240 return false;
3241 }
3242
3243 QgsReadWriteContext context;
3244 context.setPathResolver( pathResolver() );
3246
3247 QDomImplementation::setInvalidDataPolicy( QDomImplementation::DropInvalidChars );
3248
3249 const QDomDocumentType documentType =
3250 QDomImplementation().createDocumentType( QStringLiteral( "qgis" ), QStringLiteral( "http://mrcc.com/qgis.dtd" ),
3251 QStringLiteral( "SYSTEM" ) );
3252 auto doc = std::make_unique<QDomDocument>( documentType );
3253
3254 QDomElement qgisNode = doc->createElement( QStringLiteral( "qgis" ) );
3255 qgisNode.setAttribute( QStringLiteral( "projectname" ), title() );
3256 qgisNode.setAttribute( QStringLiteral( "version" ), Qgis::version() );
3257
3258 if ( !mSettings.value( QStringLiteral( "projects/anonymize_saved_projects" ), false, QgsSettings::Core ).toBool() )
3259 {
3260 const QString newSaveUser = QgsApplication::userLoginName();
3261 const QString newSaveUserFull = QgsApplication::userFullName();
3262 qgisNode.setAttribute( QStringLiteral( "saveUser" ), newSaveUser );
3263 qgisNode.setAttribute( QStringLiteral( "saveUserFull" ), newSaveUserFull );
3264 mSaveUser = newSaveUser;
3265 mSaveUserFull = newSaveUserFull;
3266 mSaveDateTime = QDateTime::currentDateTime();
3267 qgisNode.setAttribute( QStringLiteral( "saveDateTime" ), mSaveDateTime.toString( Qt::ISODate ) );
3268 }
3269 else
3270 {
3271 mSaveUser.clear();
3272 mSaveUserFull.clear();
3273 mSaveDateTime = QDateTime();
3274 }
3275 doc->appendChild( qgisNode );
3276 mSaveVersion = QgsProjectVersion( Qgis::version() );
3277
3278 QDomElement homePathNode = doc->createElement( QStringLiteral( "homePath" ) );
3279 homePathNode.setAttribute( QStringLiteral( "path" ), mHomePath );
3280 qgisNode.appendChild( homePathNode );
3281
3282 // title
3283 QDomElement titleNode = doc->createElement( QStringLiteral( "title" ) );
3284 qgisNode.appendChild( titleNode );
3285
3286 QDomElement transactionNode = doc->createElement( QStringLiteral( "transaction" ) );
3287 transactionNode.setAttribute( QStringLiteral( "mode" ), qgsEnumValueToKey( mTransactionMode ) );
3288 qgisNode.appendChild( transactionNode );
3289
3290 QDomElement flagsNode = doc->createElement( QStringLiteral( "projectFlags" ) );
3291 flagsNode.setAttribute( QStringLiteral( "set" ), qgsFlagValueToKeys( mFlags ) );
3292 qgisNode.appendChild( flagsNode );
3293
3294 const QDomText titleText = doc->createTextNode( title() ); // XXX why have title TWICE?
3295 titleNode.appendChild( titleText );
3296
3297 // write project CRS
3298 {
3299 QDomElement srsNode = doc->createElement( QStringLiteral( "projectCrs" ) );
3300 mCrs.writeXml( srsNode, *doc );
3301 qgisNode.appendChild( srsNode );
3302 }
3303 {
3304 QDomElement verticalSrsNode = doc->createElement( QStringLiteral( "verticalCrs" ) );
3305 mVerticalCrs.writeXml( verticalSrsNode, *doc );
3306 qgisNode.appendChild( verticalSrsNode );
3307 }
3308
3309 QDomElement elevationShadingNode = doc->createElement( QStringLiteral( "elevation-shading-renderer" ) );
3310 mElevationShadingRenderer.writeXml( elevationShadingNode, context );
3311 qgisNode.appendChild( elevationShadingNode );
3312
3313 // write layer tree - make sure it is without embedded subgroups
3314 QgsLayerTreeNode *clonedRoot = mRootGroup->clone();
3316 QgsLayerTreeUtils::updateEmbeddedGroupsProjectPath( QgsLayerTree::toGroup( clonedRoot ), this ); // convert absolute paths to relative paths if required
3317
3318 clonedRoot->writeXml( qgisNode, context );
3319 delete clonedRoot;
3320
3321 mSnappingConfig.writeProject( *doc );
3322 writeEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsMode" ), static_cast<int>( mAvoidIntersectionsMode ) );
3323
3324 // let map canvas and legend write their information
3325 emit writeProject( *doc );
3326
3327 // within top level node save list of layers
3328 const QMap<QString, QgsMapLayer *> layers = mapLayers();
3329
3330 QDomElement annotationLayerNode = doc->createElement( QStringLiteral( "main-annotation-layer" ) );
3331 mMainAnnotationLayer->writeLayerXml( annotationLayerNode, *doc, context );
3332 qgisNode.appendChild( annotationLayerNode );
3333
3334 // Iterate over layers in zOrder
3335 // Call writeXml() on each
3336 QDomElement projectLayersNode = doc->createElement( QStringLiteral( "projectlayers" ) );
3337
3338 QMap<QString, QgsMapLayer *>::ConstIterator li = layers.constBegin();
3339 while ( li != layers.end() )
3340 {
3341 QgsMapLayer *ml = li.value();
3342
3343 if ( ml )
3344 {
3345 const QHash< QString, QPair< QString, bool> >::const_iterator emIt = mEmbeddedLayers.constFind( ml->id() );
3346 if ( emIt == mEmbeddedLayers.constEnd() )
3347 {
3348 QDomElement maplayerElem;
3349 // If layer is not valid, prefer to restore saved properties from invalidLayerProperties. But if that's
3350 // not available, just write what we DO have
3351 if ( ml->isValid() || ml->originalXmlProperties().isEmpty() )
3352 {
3353 // general layer metadata
3354 maplayerElem = doc->createElement( QStringLiteral( "maplayer" ) );
3355 ml->writeLayerXml( maplayerElem, *doc, context );
3356
3358 maplayerElem.setAttribute( QStringLiteral( "editable" ), QStringLiteral( "1" ) );
3359 }
3360 else if ( ! ml->originalXmlProperties().isEmpty() )
3361 {
3362 QDomDocument document;
3363 if ( document.setContent( ml->originalXmlProperties() ) )
3364 {
3365 maplayerElem = document.firstChildElement();
3366 }
3367 else
3368 {
3369 QgsDebugError( QStringLiteral( "Could not restore layer properties for layer %1" ).arg( ml->id() ) );
3370 }
3371 }
3372
3373 emit writeMapLayer( ml, maplayerElem, *doc );
3374
3375 projectLayersNode.appendChild( maplayerElem );
3376 }
3377 else
3378 {
3379 // layer defined in an external project file
3380 // only save embedded layer if not managed by a legend group
3381 if ( emIt.value().second )
3382 {
3383 QDomElement mapLayerElem = doc->createElement( QStringLiteral( "maplayer" ) );
3384 mapLayerElem.setAttribute( QStringLiteral( "embedded" ), 1 );
3385 mapLayerElem.setAttribute( QStringLiteral( "project" ), writePath( emIt.value().first ) );
3386 mapLayerElem.setAttribute( QStringLiteral( "id" ), ml->id() );
3387 projectLayersNode.appendChild( mapLayerElem );
3388 }
3389 }
3390 }
3391 li++;
3392 }
3393
3394 qgisNode.appendChild( projectLayersNode );
3395
3396 QDomElement layerOrderNode = doc->createElement( QStringLiteral( "layerorder" ) );
3397 const auto constCustomLayerOrder = mRootGroup->customLayerOrder();
3398 for ( QgsMapLayer *layer : constCustomLayerOrder )
3399 {
3400 QDomElement mapLayerElem = doc->createElement( QStringLiteral( "layer" ) );
3401 mapLayerElem.setAttribute( QStringLiteral( "id" ), layer->id() );
3402 layerOrderNode.appendChild( mapLayerElem );
3403 }
3404 qgisNode.appendChild( layerOrderNode );
3405
3406 mLabelingEngineSettings->writeSettingsToProject( this );
3407 {
3408 QDomElement labelEngineSettingsElement = doc->createElement( QStringLiteral( "labelEngineSettings" ) );
3409 mLabelingEngineSettings->writeXml( *doc, labelEngineSettingsElement, context );
3410 qgisNode.appendChild( labelEngineSettingsElement );
3411 }
3412
3413 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorRedPart" ), mBackgroundColor.red() );
3414 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorGreenPart" ), mBackgroundColor.green() );
3415 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorBluePart" ), mBackgroundColor.blue() );
3416
3417 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorRedPart" ), mSelectionColor.red() );
3418 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorGreenPart" ), mSelectionColor.green() );
3419 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorBluePart" ), mSelectionColor.blue() );
3420 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorAlphaPart" ), mSelectionColor.alpha() );
3421
3422 writeEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/DistanceUnits" ), QgsUnitTypes::encodeUnit( mDistanceUnits ) );
3423 writeEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/AreaUnits" ), QgsUnitTypes::encodeUnit( mAreaUnits ) );
3424 writeEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/ScaleMethod" ), qgsEnumValueToKey( mScaleMethod ) );
3425
3426 // now add the optional extra properties
3427#if 0
3428 dump_( mProperties );
3429#endif
3430
3431 QgsDebugMsgLevel( QStringLiteral( "there are %1 property scopes" ).arg( static_cast<int>( mProperties.count() ) ), 2 );
3432
3433 if ( !mProperties.isEmpty() ) // only worry about properties if we
3434 // actually have any properties
3435 {
3436 mProperties.writeXml( QStringLiteral( "properties" ), qgisNode, *doc );
3437 }
3438
3439 QDomElement ddElem = doc->createElement( QStringLiteral( "dataDefinedServerProperties" ) );
3440 mDataDefinedServerProperties.writeXml( ddElem, dataDefinedServerPropertyDefinitions() );
3441 qgisNode.appendChild( ddElem );
3442
3443 mMapThemeCollection->writeXml( *doc );
3444
3445 mTransformContext.writeXml( qgisNode, context );
3446
3447 QDomElement metadataElem = doc->createElement( QStringLiteral( "projectMetadata" ) );
3448 mMetadata.writeMetadataXml( metadataElem, *doc );
3449 qgisNode.appendChild( metadataElem );
3450
3451 {
3452 const QDomElement annotationsElem = mAnnotationManager->writeXml( *doc, context );
3453 qgisNode.appendChild( annotationsElem );
3454 }
3455
3456 {
3457 const QDomElement layoutElem = mLayoutManager->writeXml( *doc );
3458 qgisNode.appendChild( layoutElem );
3459 }
3460
3461 {
3462 const QDomElement views3DElem = m3DViewsManager->writeXml( *doc );
3463 qgisNode.appendChild( views3DElem );
3464 }
3465
3466 {
3467 const QDomElement bookmarkElem = mBookmarkManager->writeXml( *doc );
3468 qgisNode.appendChild( bookmarkElem );
3469 }
3470
3471 {
3472 const QDomElement sensorElem = mSensorManager->writeXml( *doc );
3473 qgisNode.appendChild( sensorElem );
3474 }
3475
3476 {
3477 const QDomElement viewSettingsElem = mViewSettings->writeXml( *doc, context );
3478 qgisNode.appendChild( viewSettingsElem );
3479 }
3480
3481 {
3482 const QDomElement styleSettingsElem = mStyleSettings->writeXml( *doc, context );
3483 qgisNode.appendChild( styleSettingsElem );
3484 }
3485
3486 {
3487 const QDomElement timeSettingsElement = mTimeSettings->writeXml( *doc, context );
3488 qgisNode.appendChild( timeSettingsElement );
3489 }
3490
3491 {
3492 const QDomElement elevationPropertiesElement = mElevationProperties->writeXml( *doc, context );
3493 qgisNode.appendChild( elevationPropertiesElement );
3494 }
3495
3496 {
3497 const QDomElement displaySettingsElem = mDisplaySettings->writeXml( *doc, context );
3498 qgisNode.appendChild( displaySettingsElem );
3499 }
3500
3501 {
3502 const QDomElement gpsSettingsElem = mGpsSettings->writeXml( *doc, context );
3503 qgisNode.appendChild( gpsSettingsElem );
3504 }
3505
3506 // now wrap it up and ship it to the project file
3507 doc->normalize(); // XXX I'm not entirely sure what this does
3508
3509 // Create backup file
3510 if ( QFile::exists( fileName() ) )
3511 {
3512 QFile backupFile( QStringLiteral( "%1~" ).arg( filename ) );
3513 bool ok = true;
3514 ok &= backupFile.open( QIODevice::WriteOnly | QIODevice::Truncate );
3515 ok &= projectFile.open( QIODevice::ReadOnly );
3516
3517 QByteArray ba;
3518 while ( ok && !projectFile.atEnd() )
3519 {
3520 ba = projectFile.read( 10240 );
3521 ok &= backupFile.write( ba ) == ba.size();
3522 }
3523
3524 projectFile.close();
3525 backupFile.close();
3526
3527 if ( !ok )
3528 {
3529 setError( tr( "Unable to create backup file %1" ).arg( backupFile.fileName() ) );
3530 return false;
3531 }
3532
3533 const QFileInfo fi( fileName() );
3534 struct utimbuf tb = { static_cast<time_t>( fi.lastRead().toSecsSinceEpoch() ), static_cast<time_t>( fi.lastModified().toSecsSinceEpoch() ) };
3535 utime( backupFile.fileName().toUtf8().constData(), &tb );
3536 }
3537
3538 if ( !projectFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
3539 {
3540 projectFile.close(); // even though we got an error, let's make
3541 // sure it's closed anyway
3542
3543 setError( tr( "Unable to save to file %1" ).arg( projectFile.fileName() ) );
3544 return false;
3545 }
3546
3547 QTemporaryFile tempFile;
3548 bool ok = tempFile.open();
3549 if ( ok )
3550 {
3551 QTextStream projectFileStream( &tempFile );
3552 doc->save( projectFileStream, 2 ); // save as utf-8
3553 ok &= projectFileStream.pos() > -1;
3554
3555 ok &= tempFile.seek( 0 );
3556
3557 QByteArray ba;
3558 while ( ok && !tempFile.atEnd() )
3559 {
3560 ba = tempFile.read( 10240 );
3561 ok &= projectFile.write( ba ) == ba.size();
3562 }
3563
3564 ok &= projectFile.error() == QFile::NoError;
3565
3566 projectFile.close();
3567 }
3568
3569 tempFile.close();
3570
3571 if ( !ok )
3572 {
3573 setError( tr( "Unable to save to file %1. Your project "
3574 "may be corrupted on disk. Try clearing some space on the volume and "
3575 "check file permissions before pressing save again." )
3576 .arg( projectFile.fileName() ) );
3577 return false;
3578 }
3579
3580 setDirty( false ); // reset to pristine state
3581
3582 emit projectSaved();
3583 return true;
3584}
3585
3586bool QgsProject::writeEntry( const QString &scope, QString const &key, bool value )
3587{
3589
3590 bool propertiesModified;
3591 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3592
3593 if ( propertiesModified )
3594 setDirty( true );
3595
3596 return success;
3597}
3598
3599bool QgsProject::writeEntry( const QString &scope, const QString &key, double value )
3600{
3602
3603 bool propertiesModified;
3604 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3605
3606 if ( propertiesModified )
3607 setDirty( true );
3608
3609 return success;
3610}
3611
3612bool QgsProject::writeEntry( const QString &scope, QString const &key, int value )
3613{
3615
3616 bool propertiesModified;
3617 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3618
3619 if ( propertiesModified )
3620 setDirty( true );
3621
3622 return success;
3623}
3624
3625bool QgsProject::writeEntry( const QString &scope, const QString &key, const QString &value )
3626{
3628
3629 bool propertiesModified;
3630 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3631
3632 if ( propertiesModified )
3633 setDirty( true );
3634
3635 return success;
3636}
3637
3638bool QgsProject::writeEntry( const QString &scope, const QString &key, const QStringList &value )
3639{
3641
3642 bool propertiesModified;
3643 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3644
3645 if ( propertiesModified )
3646 setDirty( true );
3647
3648 return success;
3649}
3650
3651QStringList QgsProject::readListEntry( const QString &scope,
3652 const QString &key,
3653 const QStringList &def,
3654 bool *ok ) const
3655{
3656 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
3658
3659 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3660
3661 QVariant value;
3662
3663 if ( property )
3664 {
3665 value = property->value();
3666
3667 const bool valid = QMetaType::Type::QStringList == value.userType();
3668 if ( ok )
3669 *ok = valid;
3670
3671 if ( valid )
3672 {
3673 return value.toStringList();
3674 }
3675 }
3676 else if ( ok )
3677 *ok = false;
3678
3679
3680 return def;
3681}
3682
3683QString QgsProject::readEntry( const QString &scope,
3684 const QString &key,
3685 const QString &def,
3686 bool *ok ) const
3687{
3689
3690 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3691
3692 QVariant value;
3693
3694 if ( property )
3695 {
3696 value = property->value();
3697
3698 const bool valid = value.canConvert( QMetaType::Type::QString );
3699 if ( ok )
3700 *ok = valid;
3701
3702 if ( valid )
3703 return value.toString();
3704 }
3705 else if ( ok )
3706 *ok = false;
3707
3708 return def;
3709}
3710
3711int QgsProject::readNumEntry( const QString &scope, const QString &key, int def,
3712 bool *ok ) const
3713{
3715
3716 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3717
3718 QVariant value;
3719
3720 if ( property )
3721 {
3722 value = property->value();
3723 }
3724
3725 const bool valid = value.canConvert( QMetaType::Type::Int );
3726
3727 if ( ok )
3728 {
3729 *ok = valid;
3730 }
3731
3732 if ( valid )
3733 {
3734 return value.toInt();
3735 }
3736
3737 return def;
3738}
3739
3740double QgsProject::readDoubleEntry( const QString &scope, const QString &key,
3741 double def,
3742 bool *ok ) const
3743{
3745
3746 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3747 if ( property )
3748 {
3749 const QVariant value = property->value();
3750
3751 const bool valid = value.canConvert( QMetaType::Type::Double );
3752 if ( ok )
3753 *ok = valid;
3754
3755 if ( valid )
3756 return value.toDouble();
3757 }
3758 else if ( ok )
3759 *ok = false;
3760
3761 return def;
3762}
3763
3764bool QgsProject::readBoolEntry( const QString &scope, const QString &key, bool def,
3765 bool *ok ) const
3766{
3768
3769 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3770
3771 if ( property )
3772 {
3773 const QVariant value = property->value();
3774
3775 const bool valid = value.canConvert( QMetaType::Type::Bool );
3776 if ( ok )
3777 *ok = valid;
3778
3779 if ( valid )
3780 return value.toBool();
3781 }
3782 else if ( ok )
3783 *ok = false;
3784
3785 return def;
3786}
3787
3788bool QgsProject::removeEntry( const QString &scope, const QString &key )
3789{
3791
3792 if ( findKey_( scope, key, mProperties ) )
3793 {
3794 removeKey_( scope, key, mProperties );
3795 setDirty( true );
3796 }
3797
3798 return !findKey_( scope, key, mProperties );
3799}
3800
3801QStringList QgsProject::entryList( const QString &scope, const QString &key ) const
3802{
3804
3805 QgsProjectProperty *foundProperty = findKey_( scope, key, mProperties );
3806
3807 QStringList entries;
3808
3809 if ( foundProperty )
3810 {
3811 QgsProjectPropertyKey *propertyKey = dynamic_cast<QgsProjectPropertyKey *>( foundProperty );
3812
3813 if ( propertyKey )
3814 { propertyKey->entryList( entries ); }
3815 }
3816
3817 return entries;
3818}
3819
3820QStringList QgsProject::subkeyList( const QString &scope, const QString &key ) const
3821{
3823
3824 QgsProjectProperty *foundProperty = findKey_( scope, key, mProperties );
3825
3826 QStringList entries;
3827
3828 if ( foundProperty )
3829 {
3830 QgsProjectPropertyKey *propertyKey = dynamic_cast<QgsProjectPropertyKey *>( foundProperty );
3831
3832 if ( propertyKey )
3833 { propertyKey->subkeyList( entries ); }
3834 }
3835
3836 return entries;
3837}
3838
3840{
3842
3843 dump_( mProperties );
3844}
3845
3847{
3849
3850 QString filePath;
3851 switch ( filePathStorage() )
3852 {
3854 break;
3855
3857 {
3858 // for projects stored in a custom storage, we need to ask to the
3859 // storage for the path, if the storage returns an empty path
3860 // relative paths are not supported
3861 if ( QgsProjectStorage *storage = projectStorage() )
3862 {
3863 filePath = storage->filePath( mFile.fileName() );
3864 }
3865 else
3866 {
3867 filePath = fileName();
3868 }
3869 break;
3870 }
3871 }
3872
3873 return QgsPathResolver( filePath, mArchive->dir() );
3874}
3875
3876QString QgsProject::readPath( const QString &src ) const
3877{
3879
3880 return pathResolver().readPath( src );
3881}
3882
3883QString QgsProject::writePath( const QString &src ) const
3884{
3886
3887 return pathResolver().writePath( src );
3888}
3889
3890void QgsProject::setError( const QString &errorMessage )
3891{
3893
3894 mErrorMessage = errorMessage;
3895}
3896
3897QString QgsProject::error() const
3898{
3900
3901 return mErrorMessage;
3902}
3903
3904void QgsProject::clearError()
3905{
3907
3908 setError( QString() );
3909}
3910
3912{
3914
3915 delete mBadLayerHandler;
3916 mBadLayerHandler = handler;
3917}
3918
3919QString QgsProject::layerIsEmbedded( const QString &id ) const
3920{
3922
3923 const QHash< QString, QPair< QString, bool > >::const_iterator it = mEmbeddedLayers.find( id );
3924 if ( it == mEmbeddedLayers.constEnd() )
3925 {
3926 return QString();
3927 }
3928 return it.value().first;
3929}
3930
3931bool QgsProject::createEmbeddedLayer( const QString &layerId, const QString &projectFilePath, QList<QDomNode> &brokenNodes,
3932 bool saveFlag, Qgis::ProjectReadFlags flags )
3933{
3935
3937
3938 static QString sPrevProjectFilePath;
3939 static QDateTime sPrevProjectFileTimestamp;
3940 static QDomDocument sProjectDocument;
3941
3942 QString qgsProjectFile = projectFilePath;
3943 QgsProjectArchive archive;
3944 if ( projectFilePath.endsWith( QLatin1String( ".qgz" ), Qt::CaseInsensitive ) )
3945 {
3946 archive.unzip( projectFilePath );
3947 qgsProjectFile = archive.projectFile();
3948 }
3949
3950 const QDateTime projectFileTimestamp = QFileInfo( projectFilePath ).lastModified();
3951
3952 if ( projectFilePath != sPrevProjectFilePath || projectFileTimestamp != sPrevProjectFileTimestamp )
3953 {
3954 sPrevProjectFilePath.clear();
3955
3956 QFile projectFile( qgsProjectFile );
3957 if ( !projectFile.open( QIODevice::ReadOnly ) )
3958 {
3959 return false;
3960 }
3961
3962 if ( !sProjectDocument.setContent( &projectFile ) )
3963 {
3964 return false;
3965 }
3966
3967 sPrevProjectFilePath = projectFilePath;
3968 sPrevProjectFileTimestamp = projectFileTimestamp;
3969 }
3970
3971 // does project store paths absolute or relative?
3972 bool useAbsolutePaths = true;
3973
3974 const QDomElement propertiesElem = sProjectDocument.documentElement().firstChildElement( QStringLiteral( "properties" ) );
3975 if ( !propertiesElem.isNull() )
3976 {
3977 QDomElement e = propertiesElem.firstChildElement( QStringLiteral( "Paths" ) );
3978 if ( e.isNull() )
3979 {
3980 e = propertiesElem.firstChildElement( QStringLiteral( "properties" ) );
3981 while ( !e.isNull() && e.attribute( QStringLiteral( "name" ) ) != QStringLiteral( "Paths" ) )
3982 e = e.nextSiblingElement( QStringLiteral( "properties" ) );
3983
3984 e = e.firstChildElement( QStringLiteral( "properties" ) );
3985 while ( !e.isNull() && e.attribute( QStringLiteral( "name" ) ) != QStringLiteral( "Absolute" ) )
3986 e = e.nextSiblingElement( QStringLiteral( "properties" ) );
3987 }
3988 else
3989 {
3990 e = e.firstChildElement( QStringLiteral( "Absolute" ) );
3991 }
3992
3993 if ( !e.isNull() )
3994 {
3995 useAbsolutePaths = e.text().compare( QLatin1String( "true" ), Qt::CaseInsensitive ) == 0;
3996 }
3997 }
3998
3999 QgsReadWriteContext embeddedContext;
4000 if ( !useAbsolutePaths )
4001 embeddedContext.setPathResolver( QgsPathResolver( projectFilePath ) );
4002 embeddedContext.setProjectTranslator( this );
4003 embeddedContext.setTransformContext( transformContext() );
4004
4005 const QDomElement projectLayersElem = sProjectDocument.documentElement().firstChildElement( QStringLiteral( "projectlayers" ) );
4006 if ( projectLayersElem.isNull() )
4007 {
4008 return false;
4009 }
4010
4011 QDomElement mapLayerElem = projectLayersElem.firstChildElement( QStringLiteral( "maplayer" ) );
4012 while ( ! mapLayerElem.isNull() )
4013 {
4014 // get layer id
4015 const QString id = mapLayerElem.firstChildElement( QStringLiteral( "id" ) ).text();
4016 if ( id == layerId )
4017 {
4018 // layer can be embedded only once
4019 if ( mapLayerElem.attribute( QStringLiteral( "embedded" ) ) == QLatin1String( "1" ) )
4020 {
4021 return false;
4022 }
4023
4024 mEmbeddedLayers.insert( layerId, qMakePair( projectFilePath, saveFlag ) );
4025
4026 if ( addLayer( mapLayerElem, brokenNodes, embeddedContext, flags ) )
4027 {
4028 return true;
4029 }
4030 else
4031 {
4032 mEmbeddedLayers.remove( layerId );
4033 return false;
4034 }
4035 }
4036 mapLayerElem = mapLayerElem.nextSiblingElement( QStringLiteral( "maplayer" ) );
4037 }
4038
4039 return false;
4040}
4041
4042QgsLayerTreeGroup *QgsProject::createEmbeddedGroup( const QString &groupName, const QString &projectFilePath, const QStringList &invisibleLayers, Qgis::ProjectReadFlags flags )
4043{
4045
4046 QString qgsProjectFile = projectFilePath;
4047 QgsProjectArchive archive;
4048 if ( projectFilePath.endsWith( QLatin1String( ".qgz" ), Qt::CaseInsensitive ) )
4049 {
4050 archive.unzip( projectFilePath );
4051 qgsProjectFile = archive.projectFile();
4052 }
4053
4054 // open project file, get layer ids in group, add the layers
4055 QFile projectFile( qgsProjectFile );
4056 if ( !projectFile.open( QIODevice::ReadOnly ) )
4057 {
4058 return nullptr;
4059 }
4060
4061 QDomDocument projectDocument;
4062 if ( !projectDocument.setContent( &projectFile ) )
4063 {
4064 return nullptr;
4065 }
4066
4067 QgsReadWriteContext context;
4068 context.setPathResolver( pathResolver() );
4069 context.setProjectTranslator( this );
4071
4073
4074 QDomElement layerTreeElem = projectDocument.documentElement().firstChildElement( QStringLiteral( "layer-tree-group" ) );
4075 if ( !layerTreeElem.isNull() )
4076 {
4077 root->readChildrenFromXml( layerTreeElem, context );
4078 }
4079 else
4080 {
4081 QgsLayerTreeUtils::readOldLegend( root, projectDocument.documentElement().firstChildElement( QStringLiteral( "legend" ) ) );
4082 }
4083
4084 QgsLayerTreeGroup *group = root->findGroup( groupName );
4085 if ( !group || group->customProperty( QStringLiteral( "embedded" ) ).toBool() )
4086 {
4087 // embedded groups cannot be embedded again
4088 delete root;
4089 return nullptr;
4090 }
4091
4092 // clone the group sub-tree (it is used already in a tree, we cannot just tear it off)
4093 QgsLayerTreeGroup *newGroup = QgsLayerTree::toGroup( group->clone() );
4094 delete root;
4095 root = nullptr;
4096
4097 newGroup->setCustomProperty( QStringLiteral( "embedded" ), 1 );
4098 newGroup->setCustomProperty( QStringLiteral( "embedded_project" ), projectFilePath );
4099
4100 // set "embedded" to all children + load embedded layers
4101 mLayerTreeRegistryBridge->setEnabled( false );
4102 initializeEmbeddedSubtree( projectFilePath, newGroup, flags );
4103 mLayerTreeRegistryBridge->setEnabled( true );
4104
4105 // consider the layers might be identify disabled in its project
4106 const auto constFindLayerIds = newGroup->findLayerIds();
4107 for ( const QString &layerId : constFindLayerIds )
4108 {
4109 QgsLayerTreeLayer *layer = newGroup->findLayer( layerId );
4110 if ( layer )
4111 {
4112 layer->resolveReferences( this );
4113 layer->setItemVisibilityChecked( !invisibleLayers.contains( layerId ) );
4114 }
4115 }
4116
4117 return newGroup;
4118}
4119
4120void QgsProject::initializeEmbeddedSubtree( const QString &projectFilePath, QgsLayerTreeGroup *group, Qgis::ProjectReadFlags flags )
4121{
4123
4124 const auto constChildren = group->children();
4125 for ( QgsLayerTreeNode *child : constChildren )
4126 {
4127 // all nodes in the subtree will have "embedded" custom property set
4128 child->setCustomProperty( QStringLiteral( "embedded" ), 1 );
4129
4130 if ( QgsLayerTree::isGroup( child ) )
4131 {
4132 initializeEmbeddedSubtree( projectFilePath, QgsLayerTree::toGroup( child ), flags );
4133 }
4134 else if ( QgsLayerTree::isLayer( child ) )
4135 {
4136 // load the layer into our project
4137 QList<QDomNode> brokenNodes;
4138 createEmbeddedLayer( QgsLayerTree::toLayer( child )->layerId(), projectFilePath, brokenNodes, false, flags );
4139 }
4140 }
4141}
4142
4149
4156
4158{
4160
4161 writeEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/TopologicalEditing" ), ( enabled ? 1 : 0 ) );
4163}
4164
4166{
4168
4169 return readNumEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/TopologicalEditing" ), 0 );
4170}
4171
4173{
4175
4176 if ( mDistanceUnits == unit )
4177 return;
4178
4179 mDistanceUnits = unit;
4180
4181 emit distanceUnitsChanged();
4182}
4183
4185{
4187
4188 if ( mAreaUnits == unit )
4189 return;
4190
4191 mAreaUnits = unit;
4192
4193 emit areaUnitsChanged();
4194}
4195
4197{
4199
4200 if ( mScaleMethod == method )
4201 return;
4202
4203 mScaleMethod = method;
4204
4205 emit scaleMethodChanged();
4206}
4207
4209{
4210 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
4212
4213 if ( !mCachedHomePath.isEmpty() )
4214 return mCachedHomePath;
4215
4216 const QFileInfo pfi( fileName() );
4217
4218 if ( !mHomePath.isEmpty() )
4219 {
4220 const QFileInfo homeInfo( mHomePath );
4221 if ( !homeInfo.isRelative() )
4222 {
4223 mCachedHomePath = mHomePath;
4224 return mHomePath;
4225 }
4226 }
4227 else if ( !fileName().isEmpty() )
4228 {
4229
4230 // If it's not stored in the file system, try to get the path from the storage
4231 if ( QgsProjectStorage *storage = projectStorage() )
4232 {
4233 const QString storagePath { storage->filePath( fileName() ) };
4234 if ( ! storagePath.isEmpty() && QFileInfo::exists( storagePath ) )
4235 {
4236 mCachedHomePath = QFileInfo( storagePath ).path();
4237 return mCachedHomePath;
4238 }
4239 }
4240
4241 mCachedHomePath = pfi.path();
4242 return mCachedHomePath;
4243 }
4244
4245 if ( !pfi.exists() )
4246 {
4247 mCachedHomePath = mHomePath;
4248 return mHomePath;
4249 }
4250
4251 if ( !mHomePath.isEmpty() )
4252 {
4253 // path is relative to project file
4254 mCachedHomePath = QDir::cleanPath( pfi.path() + '/' + mHomePath );
4255 }
4256 else
4257 {
4258 mCachedHomePath = pfi.canonicalPath();
4259 }
4260 return mCachedHomePath;
4261}
4262
4264{
4266
4267 return mHomePath;
4268}
4269
4271{
4272 // because relation aggregate functions are not thread safe
4274
4275 return mRelationManager;
4276}
4277
4279{
4281
4282 return mLayoutManager.get();
4283}
4284
4286{
4288
4289 return mLayoutManager.get();
4290}
4291
4293{
4295
4296 return m3DViewsManager.get();
4297}
4298
4300{
4302
4303 return m3DViewsManager.get();
4304}
4305
4307{
4309
4310 return mBookmarkManager;
4311}
4312
4314{
4316
4317 return mBookmarkManager;
4318}
4319
4321{
4323
4324 return mSensorManager;
4325}
4326
4328{
4330
4331 return mSensorManager;
4332}
4333
4335{
4337
4338 return mViewSettings;
4339}
4340
4347
4349{
4351
4352 return mStyleSettings;
4353}
4354
4356{
4357 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
4359
4360 return mStyleSettings;
4361}
4362
4364{
4366
4367 return mTimeSettings;
4368}
4369
4376
4378{
4380
4381 return mElevationProperties;
4382}
4383
4390
4392{
4394
4395 return mDisplaySettings;
4396}
4397
4399{
4401
4402 return mDisplaySettings;
4403}
4404
4406{
4408
4409 return mGpsSettings;
4410}
4411
4418
4420{
4422
4423 return mRootGroup;
4424}
4425
4427{
4429
4430 return mMapThemeCollection.get();
4431}
4432
4434{
4436
4437 return mAnnotationManager.get();
4438}
4439
4441{
4443
4444 return mAnnotationManager.get();
4445}
4446
4447void QgsProject::setNonIdentifiableLayers( const QList<QgsMapLayer *> &layers )
4448{
4450
4451 const QMap<QString, QgsMapLayer *> &projectLayers = mapLayers();
4452 for ( QMap<QString, QgsMapLayer *>::const_iterator it = projectLayers.constBegin(); it != projectLayers.constEnd(); ++it )
4453 {
4454 if ( layers.contains( it.value() ) == !it.value()->flags().testFlag( QgsMapLayer::Identifiable ) )
4455 continue;
4456
4457 if ( layers.contains( it.value() ) )
4458 it.value()->setFlags( it.value()->flags() & ~QgsMapLayer::Identifiable );
4459 else
4460 it.value()->setFlags( it.value()->flags() | QgsMapLayer::Identifiable );
4461 }
4462
4466}
4467
4468void QgsProject::setNonIdentifiableLayers( const QStringList &layerIds )
4469{
4471
4472 QList<QgsMapLayer *> nonIdentifiableLayers;
4473 nonIdentifiableLayers.reserve( layerIds.count() );
4474 for ( const QString &layerId : layerIds )
4475 {
4476 QgsMapLayer *layer = mapLayer( layerId );
4477 if ( layer )
4478 nonIdentifiableLayers << layer;
4479 }
4483}
4484
4486{
4488
4489 QStringList nonIdentifiableLayers;
4490
4491 const QMap<QString, QgsMapLayer *> &layers = mapLayers();
4492 for ( QMap<QString, QgsMapLayer *>::const_iterator it = layers.constBegin(); it != layers.constEnd(); ++it )
4493 {
4494 if ( !it.value()->flags().testFlag( QgsMapLayer::Identifiable ) )
4495 {
4496 nonIdentifiableLayers.append( it.value()->id() );
4497 }
4498 }
4499 return nonIdentifiableLayers;
4500}
4501
4503{
4505
4506 return mTransactionMode == Qgis::TransactionMode::AutomaticGroups;
4507}
4508
4509void QgsProject::setAutoTransaction( bool autoTransaction )
4510{
4512
4513 if ( autoTransaction
4514 && mTransactionMode == Qgis::TransactionMode::AutomaticGroups )
4515 return;
4516
4517 if ( ! autoTransaction
4518 && mTransactionMode == Qgis::TransactionMode::Disabled )
4519 return;
4520
4521 if ( autoTransaction )
4523 else
4525
4526 updateTransactionGroups();
4527}
4528
4530{
4532
4533 return mTransactionMode;
4534}
4535
4537{
4539
4540 if ( transactionMode == mTransactionMode )
4541 return true;
4542
4543 // Check that all layer are not in edit mode
4544 const auto constLayers = mapLayers().values();
4545 for ( QgsMapLayer *layer : constLayers )
4546 {
4547 if ( layer->isEditable() )
4548 {
4549 QgsLogger::warning( tr( "Transaction mode can be changed only if all layers are not editable." ) );
4550 return false;
4551 }
4552 }
4553
4554 mTransactionMode = transactionMode;
4555 updateTransactionGroups();
4557 return true;
4558}
4559
4560QMap<QPair<QString, QString>, QgsTransactionGroup *> QgsProject::transactionGroups()
4561{
4563
4564 return mTransactionGroups;
4565}
4566
4567
4568//
4569// QgsMapLayerStore methods
4570//
4571
4572
4574{
4576
4577 return mLayerStore->count();
4578}
4579
4581{
4583
4584 return mLayerStore->validCount();
4585}
4586
4587QgsMapLayer *QgsProject::mapLayer( const QString &layerId ) const
4588{
4589 // because QgsVirtualLayerProvider is not anywhere NEAR thread safe:
4591
4592 return mLayerStore->mapLayer( layerId );
4593}
4594
4595QList<QgsMapLayer *> QgsProject::mapLayersByName( const QString &layerName ) const
4596{
4598
4599 return mLayerStore->mapLayersByName( layerName );
4600}
4601
4602QList<QgsMapLayer *> QgsProject::mapLayersByShortName( const QString &shortName ) const
4603{
4605
4606 QList<QgsMapLayer *> layers;
4607 const auto constMapLayers { mLayerStore->mapLayers() };
4608 for ( const auto &l : constMapLayers )
4609 {
4610 if ( ! l->serverProperties()->shortName().isEmpty() )
4611 {
4612 if ( l->serverProperties()->shortName() == shortName )
4613 layers << l;
4614 }
4615 else if ( l->name() == shortName )
4616 {
4617 layers << l;
4618 }
4619 }
4620 return layers;
4621}
4622
4623bool QgsProject::unzip( const QString &filename, Qgis::ProjectReadFlags flags )
4624{
4626
4627 clearError();
4628 auto archive = std::make_unique<QgsProjectArchive>();
4629
4630 // unzip the archive
4631 if ( !archive->unzip( filename ) )
4632 {
4633 setError( tr( "Unable to unzip file '%1'" ).arg( filename ) );
4634 return false;
4635 }
4636
4637 // test if zip provides a .qgs file
4638 if ( archive->projectFile().isEmpty() )
4639 {
4640 setError( tr( "Zip archive does not provide a project file" ) );
4641 return false;
4642 }
4643
4644 // Keep the archive
4645 releaseHandlesToProjectArchive();
4646 mArchive = std::move( archive );
4647
4648 // load auxiliary storage
4649 if ( !static_cast<QgsProjectArchive *>( mArchive.get() )->auxiliaryStorageFile().isEmpty() )
4650 {
4651 // database file is already a copy as it's been unzipped. So we don't open
4652 // auxiliary storage in copy mode in this case
4653 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( static_cast<QgsProjectArchive *>( mArchive.get() )->auxiliaryStorageFile(), false ) );
4654 }
4655 else
4656 {
4657 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( *this ) );
4658 }
4659
4660 // read the project file
4661 if ( ! readProjectFile( static_cast<QgsProjectArchive *>( mArchive.get() )->projectFile(), flags ) )
4662 {
4663 setError( tr( "Cannot read unzipped qgs project file" ) + QStringLiteral( ": " ) + error() );
4664 return false;
4665 }
4666
4667 // Remove the temporary .qgs file
4668 static_cast<QgsProjectArchive *>( mArchive.get() )->clearProjectFile();
4669
4670 return true;
4671}
4672
4673bool QgsProject::zip( const QString &filename )
4674{
4676
4677 clearError();
4678
4679 // save the current project in a temporary .qgs file
4680 auto archive = std::make_unique<QgsProjectArchive>();
4681 const QString baseName = QFileInfo( filename ).baseName();
4682 const QString qgsFileName = QStringLiteral( "%1.qgs" ).arg( baseName );
4683 QFile qgsFile( QDir( archive->dir() ).filePath( qgsFileName ) );
4684
4685 bool writeOk = false;
4686 if ( qgsFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
4687 {
4688 writeOk = writeProjectFile( qgsFile.fileName() );
4689 qgsFile.close();
4690 }
4691
4692 // stop here with an error message
4693 if ( ! writeOk )
4694 {
4695 setError( tr( "Unable to write temporary qgs file" ) );
4696 return false;
4697 }
4698
4699 // save auxiliary storage
4700 const QFileInfo info( qgsFile );
4701 const QString asExt = QStringLiteral( ".%1" ).arg( QgsAuxiliaryStorage::extension() );
4702 const QString asFileName = info.path() + QDir::separator() + info.completeBaseName() + asExt;
4703
4704 bool auxiliaryStorageSavedOk = true;
4705 if ( ! saveAuxiliaryStorage( asFileName ) )
4706 {
4707 const QString err = mAuxiliaryStorage->errorString();
4708 setError( tr( "Unable to save auxiliary storage file ('%1'). The project has been saved but the latest changes to auxiliary data cannot be recovered. It is recommended to reload the project." ).arg( err ) );
4709 auxiliaryStorageSavedOk = false;
4710
4711 // fixes the current archive and keep the previous version of qgd
4712 if ( !mArchive->exists() )
4713 {
4714 releaseHandlesToProjectArchive();
4715 mArchive.reset( new QgsProjectArchive() );
4716 mArchive->unzip( mFile.fileName() );
4717 static_cast<QgsProjectArchive *>( mArchive.get() )->clearProjectFile();
4718
4719 const QString auxiliaryStorageFile = static_cast<QgsProjectArchive *>( mArchive.get() )->auxiliaryStorageFile();
4720 if ( ! auxiliaryStorageFile.isEmpty() )
4721 {
4722 archive->addFile( auxiliaryStorageFile );
4723 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( auxiliaryStorageFile, false ) );
4724 }
4725 }
4726 }
4727 else
4728 {
4729 // in this case, an empty filename means that the auxiliary database is
4730 // empty, so we don't want to save it
4731 if ( QFile::exists( asFileName ) )
4732 {
4733 archive->addFile( asFileName );
4734 }
4735 }
4736
4737 // create the archive
4738 archive->addFile( qgsFile.fileName() );
4739
4740 // Add all other files
4741 const QStringList &files = mArchive->files();
4742 for ( const QString &file : files )
4743 {
4744 if ( !file.endsWith( QLatin1String( ".qgs" ), Qt::CaseInsensitive ) && !file.endsWith( asExt, Qt::CaseInsensitive ) )
4745 {
4746 archive->addFile( file );
4747 }
4748 }
4749
4750 // zip
4751 bool zipOk = true;
4752 if ( !archive->zip( filename ) )
4753 {
4754 setError( tr( "Unable to perform zip" ) );
4755 zipOk = false;
4756 }
4757
4758 return auxiliaryStorageSavedOk && zipOk;
4759}
4760
4762{
4764
4765 return QgsZipUtils::isZipFile( mFile.fileName() );
4766}
4767
4768QList<QgsMapLayer *> QgsProject::addMapLayers(
4769 const QList<QgsMapLayer *> &layers,
4770 bool addToLegend,
4771 bool takeOwnership )
4772{
4774
4775 const QList<QgsMapLayer *> myResultList { mLayerStore->addMapLayers( layers, takeOwnership ) };
4776 if ( !myResultList.isEmpty() )
4777 {
4778 // Update transform context
4779 for ( auto &l : myResultList )
4780 {
4781 l->setTransformContext( transformContext() );
4782 }
4783 if ( addToLegend )
4784 {
4785 emit legendLayersAdded( myResultList );
4786 }
4787 }
4788
4789 if ( mAuxiliaryStorage )
4790 {
4791 for ( QgsMapLayer *mlayer : myResultList )
4792 {
4793 if ( mlayer->type() != Qgis::LayerType::Vector )
4794 continue;
4795
4796 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mlayer );
4797 if ( vl )
4798 {
4799 vl->loadAuxiliaryLayer( *mAuxiliaryStorage );
4800 }
4801 }
4802 }
4803
4804 mProjectScope.reset();
4805
4806 return myResultList;
4807}
4808
4811 bool addToLegend,
4812 bool takeOwnership )
4813{
4815
4816 QList<QgsMapLayer *> addedLayers;
4817 addedLayers = addMapLayers( QList<QgsMapLayer *>() << layer, addToLegend, takeOwnership );
4818 return addedLayers.isEmpty() ? nullptr : addedLayers[0];
4819}
4820
4821void QgsProject::removeAuxiliaryLayer( const QgsMapLayer *ml )
4822{
4824
4825 if ( ! ml || ml->type() != Qgis::LayerType::Vector )
4826 return;
4827
4828 const QgsVectorLayer *vl = qobject_cast<const QgsVectorLayer *>( ml );
4829 if ( vl && vl->auxiliaryLayer() )
4830 {
4831 const QgsDataSourceUri uri( vl->auxiliaryLayer()->source() );
4833 }
4834}
4835
4836void QgsProject::removeMapLayers( const QStringList &layerIds )
4837{
4839
4840 for ( const auto &layerId : layerIds )
4841 removeAuxiliaryLayer( mLayerStore->mapLayer( layerId ) );
4842
4843 mProjectScope.reset();
4844 mLayerStore->removeMapLayers( layerIds );
4845}
4846
4847void QgsProject::removeMapLayers( const QList<QgsMapLayer *> &layers )
4848{
4850
4851 for ( const auto &layer : layers )
4852 removeAuxiliaryLayer( layer );
4853
4854 mProjectScope.reset();
4855 mLayerStore->removeMapLayers( layers );
4856}
4857
4858void QgsProject::removeMapLayer( const QString &layerId )
4859{
4861
4862 removeAuxiliaryLayer( mLayerStore->mapLayer( layerId ) );
4863 mProjectScope.reset();
4864 mLayerStore->removeMapLayer( layerId );
4865}
4866
4868{
4870
4871 removeAuxiliaryLayer( layer );
4872 mProjectScope.reset();
4873 mLayerStore->removeMapLayer( layer );
4874}
4875
4877{
4879
4880 mProjectScope.reset();
4881 return mLayerStore->takeMapLayer( layer );
4882}
4883
4885{
4887
4888 return mMainAnnotationLayer;
4889}
4890
4892{
4894
4895 if ( mLayerStore->count() == 0 )
4896 return;
4897
4898 ScopedIntIncrementor snapSingleBlocker( &mBlockSnappingUpdates );
4899 mProjectScope.reset();
4900 mLayerStore->removeAllMapLayers();
4901
4902 snapSingleBlocker.release();
4903 mSnappingConfig.clearIndividualLayerSettings();
4904 if ( !mBlockSnappingUpdates )
4905 emit snappingConfigChanged( mSnappingConfig );
4906}
4907
4909{
4911
4912 const QMap<QString, QgsMapLayer *> layers = mLayerStore->mapLayers();
4913 QMap<QString, QgsMapLayer *>::const_iterator it = layers.constBegin();
4914 for ( ; it != layers.constEnd(); ++it )
4915 {
4916 it.value()->reload();
4917 }
4918}
4919
4920QMap<QString, QgsMapLayer *> QgsProject::mapLayers( const bool validOnly ) const
4921{
4922 // because QgsVirtualLayerProvider is not anywhere NEAR thread safe:
4924
4925 return validOnly ? mLayerStore->validMapLayers() : mLayerStore->mapLayers();
4926}
4927
4928QgsTransactionGroup *QgsProject::transactionGroup( const QString &providerKey, const QString &connString )
4929{
4931
4932 return mTransactionGroups.value( qMakePair( providerKey, connString ) );
4933}
4934
4941
4943{
4945
4947
4948 // TODO QGIS 4.0 -- remove this method, and place it somewhere in app (where it belongs)
4949 // in the meantime, we have a slightly hacky way to read the settings key using an enum which isn't available (since it lives in app)
4950 if ( mSettings.value( QStringLiteral( "/projections/unknownCrsBehavior" ), QStringLiteral( "NoAction" ), QgsSettings::App ).toString() == QStringLiteral( "UseProjectCrs" )
4951 || mSettings.value( QStringLiteral( "/projections/unknownCrsBehavior" ), 0, QgsSettings::App ).toString() == QLatin1String( "2" ) )
4952 {
4953 // for new layers if the new layer crs method is set to either prompt or use project, then we use the project crs
4954 defaultCrs = crs();
4955 }
4956 else
4957 {
4958 // global crs
4959 const QString layerDefaultCrs = mSettings.value( QStringLiteral( "/Projections/layerDefaultCrs" ), QStringLiteral( "EPSG:4326" ) ).toString();
4960 defaultCrs = QgsCoordinateReferenceSystem::fromOgcWmsCrs( layerDefaultCrs );
4961 }
4962
4963 return defaultCrs;
4964}
4965
4972
4979
4980bool QgsProject::saveAuxiliaryStorage( const QString &filename )
4981{
4983
4984 const QMap<QString, QgsMapLayer *> layers = mapLayers();
4985 bool empty = true;
4986 for ( auto it = layers.constBegin(); it != layers.constEnd(); ++it )
4987 {
4988 if ( it.value()->type() != Qgis::LayerType::Vector )
4989 continue;
4990
4991 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( it.value() );
4992 if ( vl && vl->auxiliaryLayer() )
4993 {
4994 vl->auxiliaryLayer()->save();
4995 empty &= vl->auxiliaryLayer()->auxiliaryFields().isEmpty();
4996 }
4997 }
4998
4999 if ( !mAuxiliaryStorage->exists( *this ) && empty )
5000 {
5001 return true; // it's not an error
5002 }
5003 else if ( !filename.isEmpty() )
5004 {
5005 return mAuxiliaryStorage->saveAs( filename );
5006 }
5007 else
5008 {
5009 return mAuxiliaryStorage->saveAs( *this );
5010 }
5011}
5012
5013QgsPropertiesDefinition &QgsProject::dataDefinedServerPropertyDefinitions()
5014{
5015 static QgsPropertiesDefinition sPropertyDefinitions
5016 {
5017 {
5019 QgsPropertyDefinition( "WMSOnlineResource", QObject::tr( "WMS Online Resource" ), QgsPropertyDefinition::String )
5020 },
5021 };
5022 return sPropertyDefinitions;
5023}
5024
5026{
5027 mElevationShadingRenderer = elevationShadingRenderer;
5029}
5030
5032{
5034
5035 return mAuxiliaryStorage.get();
5036}
5037
5039{
5041
5042 return mAuxiliaryStorage.get();
5043}
5044
5045QString QgsProject::createAttachedFile( const QString &nameTemplate )
5046{
5048
5049 const QDir archiveDir( mArchive->dir() );
5050 QTemporaryFile tmpFile( archiveDir.filePath( "XXXXXX_" + nameTemplate ), this );
5051 tmpFile.setAutoRemove( false );
5052 tmpFile.open();
5053 mArchive->addFile( tmpFile.fileName() );
5054 return tmpFile.fileName();
5055}
5056
5057QStringList QgsProject::attachedFiles() const
5058{
5060
5061 QStringList attachments;
5062 const QString baseName = QFileInfo( fileName() ).baseName();
5063 const QStringList files = mArchive->files();
5064 attachments.reserve( files.size() );
5065 for ( const QString &file : files )
5066 {
5067 if ( QFileInfo( file ).baseName() != baseName )
5068 {
5069 attachments.append( file );
5070 }
5071 }
5072 return attachments;
5073}
5074
5075bool QgsProject::removeAttachedFile( const QString &path )
5076{
5078
5079 return mArchive->removeFile( path );
5080}
5081
5082QString QgsProject::attachmentIdentifier( const QString &attachedFile ) const
5083{
5085
5086 return QStringLiteral( "attachment:///%1" ).arg( QFileInfo( attachedFile ).fileName() );
5087}
5088
5089QString QgsProject::resolveAttachmentIdentifier( const QString &identifier ) const
5090{
5092
5093 if ( identifier.startsWith( QLatin1String( "attachment:///" ) ) )
5094 {
5095 return QDir( mArchive->dir() ).absoluteFilePath( identifier.mid( 14 ) );
5096 }
5097 return QString();
5098}
5099
5101{
5102 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
5104
5105 return mMetadata;
5106}
5107
5109{
5111
5112 if ( metadata == mMetadata )
5113 return;
5114
5115 mMetadata = metadata;
5116 mProjectScope.reset();
5117
5118 emit metadataChanged();
5119
5120 setDirty( true );
5121}
5122
5123QSet<QgsMapLayer *> QgsProject::requiredLayers() const
5124{
5126
5127 QSet<QgsMapLayer *> requiredLayers;
5128
5129 const QMap<QString, QgsMapLayer *> &layers = mapLayers();
5130 for ( QMap<QString, QgsMapLayer *>::const_iterator it = layers.constBegin(); it != layers.constEnd(); ++it )
5131 {
5132 if ( !it.value()->flags().testFlag( QgsMapLayer::Removable ) )
5133 {
5134 requiredLayers.insert( it.value() );
5135 }
5136 }
5137 return requiredLayers;
5138}
5139
5140void QgsProject::setRequiredLayers( const QSet<QgsMapLayer *> &layers )
5141{
5143
5144 const QMap<QString, QgsMapLayer *> &projectLayers = mapLayers();
5145 for ( QMap<QString, QgsMapLayer *>::const_iterator it = projectLayers.constBegin(); it != projectLayers.constEnd(); ++it )
5146 {
5147 if ( layers.contains( it.value() ) == !it.value()->flags().testFlag( QgsMapLayer::Removable ) )
5148 continue;
5149
5150 if ( layers.contains( it.value() ) )
5151 it.value()->setFlags( it.value()->flags() & ~QgsMapLayer::Removable );
5152 else
5153 it.value()->setFlags( it.value()->flags() | QgsMapLayer::Removable );
5154 }
5155}
5156
5158{
5160
5161 // save colors to project
5162 QStringList customColors;
5163 QStringList customColorLabels;
5164
5165 QgsNamedColorList::const_iterator colorIt = colors.constBegin();
5166 for ( ; colorIt != colors.constEnd(); ++colorIt )
5167 {
5168 const QString color = QgsColorUtils::colorToString( ( *colorIt ).first );
5169 const QString label = ( *colorIt ).second;
5170 customColors.append( color );
5171 customColorLabels.append( label );
5172 }
5173 writeEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Colors" ), customColors );
5174 writeEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Labels" ), customColorLabels );
5175 mProjectScope.reset();
5176 emit projectColorsChanged();
5177}
5178
5179void QgsProject::setBackgroundColor( const QColor &color )
5180{
5182
5183 if ( mBackgroundColor == color )
5184 return;
5185
5186 mBackgroundColor = color;
5188}
5189
5191{
5193
5194 return mBackgroundColor;
5195}
5196
5197void QgsProject::setSelectionColor( const QColor &color )
5198{
5200
5201 if ( mSelectionColor == color )
5202 return;
5203
5204 mSelectionColor = color;
5205 emit selectionColorChanged();
5206}
5207
5209{
5211
5212 return mSelectionColor;
5213}
5214
5215void QgsProject::setMapScales( const QVector<double> &scales )
5216{
5218
5219 mViewSettings->setMapScales( scales );
5220}
5221
5222QVector<double> QgsProject::mapScales() const
5223{
5225
5226 return mViewSettings->mapScales();
5227}
5228
5230{
5232
5233 mViewSettings->setUseProjectScales( enabled );
5234}
5235
5237{
5239
5240 return mViewSettings->useProjectScales();
5241}
5242
5243void QgsProject::generateTsFile( const QString &locale )
5244{
5246
5247 QgsTranslationContext translationContext;
5248 translationContext.setProject( this );
5249 translationContext.setFileName( QStringLiteral( "%1/%2.ts" ).arg( absolutePath(), baseName() ) );
5250
5251 QgsApplication::instance()->collectTranslatableObjects( &translationContext );
5252
5253 translationContext.writeTsFile( locale );
5254}
5255
5256QString QgsProject::translate( const QString &context, const QString &sourceText, const char *disambiguation, int n ) const
5257{
5259
5260 if ( !mTranslator )
5261 {
5262 return sourceText;
5263 }
5264
5265 QString result = mTranslator->translate( context.toUtf8(), sourceText.toUtf8(), disambiguation, n );
5266
5267 if ( result.isEmpty() )
5268 {
5269 return sourceText;
5270 }
5271 return result;
5272}
5273
5275{
5277
5278 const QMap<QString, QgsMapLayer *> layers = mapLayers( false );
5279 if ( !layers.empty() )
5280 {
5281 for ( auto it = layers.constBegin(); it != layers.constEnd(); ++it )
5282 {
5283 // NOTE: if visitEnter returns false it means "don't visit this layer", not "abort all further visitations"
5284 if ( visitor->visitEnter( QgsStyleEntityVisitorInterface::Node( QgsStyleEntityVisitorInterface::NodeType::Layer, ( *it )->id(), ( *it )->name() ) ) )
5285 {
5286 if ( !( ( *it )->accept( visitor ) ) )
5287 return false;
5288
5289 if ( !visitor->visitExit( QgsStyleEntityVisitorInterface::Node( QgsStyleEntityVisitorInterface::NodeType::Layer, ( *it )->id(), ( *it )->name() ) ) )
5290 return false;
5291 }
5292 }
5293 }
5294
5295 if ( !mLayoutManager->accept( visitor ) )
5296 return false;
5297
5298 if ( !mAnnotationManager->accept( visitor ) )
5299 return false;
5300
5301 return true;
5302}
5303
5305{
5306 return mElevationShadingRenderer;
5307}
5308
5309void QgsProject::loadProjectFlags( const QDomDocument *doc )
5310{
5312
5313 QDomElement element = doc->documentElement().firstChildElement( QStringLiteral( "projectFlags" ) );
5315 if ( !element.isNull() )
5316 {
5317 flags = qgsFlagKeysToValue( element.attribute( QStringLiteral( "set" ) ), Qgis::ProjectFlags() );
5318 }
5319 else
5320 {
5321 // older project compatibility
5322 element = doc->documentElement().firstChildElement( QStringLiteral( "evaluateDefaultValues" ) );
5323 if ( !element.isNull() )
5324 {
5325 if ( element.attribute( QStringLiteral( "active" ), QStringLiteral( "0" ) ).toInt() == 1 )
5327 }
5328
5329 // Read trust layer metadata config in the project
5330 element = doc->documentElement().firstChildElement( QStringLiteral( "trust" ) );
5331 if ( !element.isNull() )
5332 {
5333 if ( element.attribute( QStringLiteral( "active" ), QStringLiteral( "0" ) ).toInt() == 1 )
5335 }
5336 }
5337
5338 setFlags( flags );
5339}
5340
5342{
5344 {
5345 const Qgis::PythonEmbeddedMode pythonEmbeddedMode = QgsSettings().enumValue( QStringLiteral( "qgis/enablePythonEmbedded" ), Qgis::PythonEmbeddedMode::Ask );
5346
5347 if ( force || pythonEmbeddedMode == Qgis::PythonEmbeddedMode::SessionOnly || pythonEmbeddedMode == Qgis::PythonEmbeddedMode::Always )
5348 {
5349 const QString projectFunctions = readEntry( QStringLiteral( "ExpressionFunctions" ), QStringLiteral( "/pythonCode" ), QString() );
5350 if ( !projectFunctions.isEmpty() )
5351 {
5352 QgsPythonRunner::run( projectFunctions );
5353 return true;
5354 }
5355 }
5356 }
5357 return false;
5358}
5359
5361{
5363 {
5364 QgsPythonRunner::run( "qgis.utils.clean_project_expression_functions()" );
5365 }
5366}
5367
5369
5370QHash< QString, QColor > loadColorsFromProject( const QgsProject *project )
5371{
5372 QHash< QString, QColor > colors;
5373
5374 //build up color list from project. Do this in advance for speed
5375 QStringList colorStrings = project->readListEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Colors" ) );
5376 const QStringList colorLabels = project->readListEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Labels" ) );
5377
5378 //generate list from custom colors
5379 int colorIndex = 0;
5380 for ( QStringList::iterator it = colorStrings.begin();
5381 it != colorStrings.end(); ++it )
5382 {
5383 const QColor color = QgsColorUtils::colorFromString( *it );
5384 QString label;
5385 if ( colorLabels.length() > colorIndex )
5386 {
5387 label = colorLabels.at( colorIndex );
5388 }
5389
5390 colors.insert( label.toLower(), color );
5391 colorIndex++;
5392 }
5393
5394 return colors;
5395}
5396
5397
5398GetNamedProjectColor::GetNamedProjectColor( const QgsProject *project )
5399 : QgsScopedExpressionFunction( QStringLiteral( "project_color" ), 1, QStringLiteral( "Color" ) )
5400{
5401 if ( !project )
5402 return;
5403
5404 mColors = loadColorsFromProject( project );
5405}
5406
5407GetNamedProjectColor::GetNamedProjectColor( const QHash<QString, QColor> &colors )
5408 : QgsScopedExpressionFunction( QStringLiteral( "project_color" ), 1, QStringLiteral( "Color" ) )
5409 , mColors( colors )
5410{
5411}
5412
5413QVariant GetNamedProjectColor::func( const QVariantList &values, const QgsExpressionContext *, QgsExpression *, const QgsExpressionNodeFunction * )
5414{
5415 const QString colorName = values.at( 0 ).toString().toLower();
5416 if ( mColors.contains( colorName ) )
5417 {
5418 return QStringLiteral( "%1,%2,%3" ).arg( mColors.value( colorName ).red() ).arg( mColors.value( colorName ).green() ).arg( mColors.value( colorName ).blue() );
5419 }
5420 else
5421 return QVariant();
5422}
5423
5424QgsScopedExpressionFunction *GetNamedProjectColor::clone() const
5425{
5426 return new GetNamedProjectColor( mColors );
5427}
5428
5429GetNamedProjectColorObject::GetNamedProjectColorObject( const QgsProject *project )
5430 : QgsScopedExpressionFunction( QStringLiteral( "project_color_object" ), 1, QStringLiteral( "Color" ) )
5431{
5432 if ( !project )
5433 return;
5434
5435 mColors = loadColorsFromProject( project );
5436}
5437
5438GetNamedProjectColorObject::GetNamedProjectColorObject( const QHash<QString, QColor> &colors )
5439 : QgsScopedExpressionFunction( QStringLiteral( "project_color_object" ), 1, QStringLiteral( "Color" ) )
5440 , mColors( colors )
5441{
5442}
5443
5444QVariant GetNamedProjectColorObject::func( const QVariantList &values, const QgsExpressionContext *, QgsExpression *, const QgsExpressionNodeFunction * )
5445{
5446 const QString colorName = values.at( 0 ).toString().toLower();
5447 if ( mColors.contains( colorName ) )
5448 {
5449 return mColors.value( colorName );
5450 }
5451 else
5452 return QVariant();
5453}
5454
5455QgsScopedExpressionFunction *GetNamedProjectColorObject::clone() const
5456{
5457 return new GetNamedProjectColorObject( mColors );
5458}
5459
5460// ----------------
5461
5462GetSensorData::GetSensorData( const QMap<QString, QgsAbstractSensor::SensorData> &sensorData )
5463 : QgsScopedExpressionFunction( QStringLiteral( "sensor_data" ),
5464 QgsExpressionFunction::ParameterList() << QgsExpressionFunction::Parameter( QStringLiteral( "name" ) ) << QgsExpressionFunction::Parameter( QStringLiteral( "expiration" ), true, 0 ),
5465 QStringLiteral( "Sensors" ) )
5466 , mSensorData( sensorData )
5467{
5468}
5469
5470QVariant GetSensorData::func( const QVariantList &values, const QgsExpressionContext *, QgsExpression *, const QgsExpressionNodeFunction * )
5471{
5472 const QString sensorName = values.at( 0 ).toString();
5473 const int expiration = values.at( 1 ).toInt();
5474 const qint64 timestamp = QDateTime::currentMSecsSinceEpoch();
5475 if ( mSensorData.contains( sensorName ) )
5476 {
5477 if ( expiration <= 0 || ( timestamp - mSensorData[sensorName].lastTimestamp.toMSecsSinceEpoch() ) < expiration )
5478 {
5479 return mSensorData[sensorName].lastValue;
5480 }
5481 }
5482
5483 return QVariant();
5484}
5485
5486QgsScopedExpressionFunction *GetSensorData::clone() const
5487{
5488 return new GetSensorData( mSensorData );
5489}
@ DontLoad3DViews
Skip loading 3D views.
@ DontStoreOriginalStyles
Skip the initial XML style storage for layers. Useful for minimising project load times in non-intera...
@ ForceReadOnlyLayers
Open layers in a read-only mode.
@ TrustLayerMetadata
Trust layer metadata. Improves project read time. Do not use it if layers' extent is not fixed during...
@ DontUpgradeAnnotations
Don't upgrade old annotation items to QgsAnnotationItem.
@ DontLoadLayouts
Don't load print layouts. Improves project read time if layouts are not required, and allows projects...
@ DontResolveLayers
Don't resolve layer paths (i.e. don't load any layer content). Dramatically improves project read tim...
static QString version()
Version string.
Definition qgis.cpp:267
QFlags< ProjectCapability > ProjectCapabilities
Flags which control project capabilities.
Definition qgis.h:4194
QFlags< ProjectReadFlag > ProjectReadFlags
Project load flags.
Definition qgis.h:4172
DistanceUnit
Units of distance.
Definition qgis.h:4843
FilePathType
File path types.
Definition qgis.h:1639
@ Relative
Relative path.
@ Absolute
Absolute path.
TransactionMode
Transaction mode.
Definition qgis.h:3820
@ AutomaticGroups
Automatic transactional editing means that on supported datasources (postgres and geopackage database...
@ BufferedGroups
Buffered transactional editing means that all editable layers in the buffered transaction group are t...
@ Disabled
Edits are buffered locally and sent to the provider when toggling layer editing mode.
AreaUnit
Units of area.
Definition qgis.h:4920
@ SquareMeters
Square meters.
@ Critical
Critical/error message.
Definition qgis.h:157
@ Success
Used for reporting a successful operation.
Definition qgis.h:158
PythonEmbeddedMode
Authorisation to run Python Embedded in projects.
Definition qgis.h:403
@ Always
Python embedded is always run.
@ Ask
User is prompt before running.
@ SessionOnly
Only during this session.
@ Vertical
Vertical CRS.
@ Temporal
Temporal CRS.
@ Compound
Compound (horizontal + vertical) CRS.
@ Projected
Projected CRS.
@ Other
Other type.
@ Bound
Bound CRS.
@ DerivedProjected
Derived projected CRS.
@ Unknown
Unknown type.
@ Engineering
Engineering CRS.
@ Geographic3d
3D geopraphic CRS
@ Geodetic
Geodetic CRS.
@ Geographic2d
2D geographic CRS
@ Geocentric
Geocentric CRS.
AvoidIntersectionsMode
Flags which control how intersections of pre-existing feature are handled when digitizing new feature...
Definition qgis.h:4124
@ AvoidIntersectionsLayers
Overlap with features from a specified list of layers when digitizing new features not allowed.
@ AllowIntersections
Overlap with any feature allowed when digitizing new features.
ProjectFlag
Flags which control the behavior of QgsProjects.
Definition qgis.h:3943
@ RememberLayerEditStatusBetweenSessions
If set, then any layers set to be editable will be stored in the project and immediately made editabl...
@ EvaluateDefaultValuesOnProviderSide
If set, default values for fields will be evaluated on the provider side when features from the proje...
@ TrustStoredLayerStatistics
If set, then layer statistics (such as the layer extent) will be read from values stored in the proje...
@ Polygon
Polygons.
QFlags< DataProviderReadFlag > DataProviderReadFlags
Flags which control data provider construction.
Definition qgis.h:450
LayerType
Types of layers that can be added to a map.
Definition qgis.h:169
@ Group
Composite group layer. Added in QGIS 3.24.
@ Plugin
Plugin based layer.
@ TiledScene
Tiled scene layer. Added in QGIS 3.34.
@ Annotation
Contains freeform, georeferenced annotations. Added in QGIS 3.16.
@ Vector
Vector layer.
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
@ Mesh
Mesh layer. Added in QGIS 3.2.
@ Raster
Raster layer.
@ PointCloud
Point cloud layer. Added in QGIS 3.18.
ScaleCalculationMethod
Scale calculation logic.
Definition qgis.h:5080
@ HorizontalMiddle
Calculate horizontally, across midle of map.
@ SkipCredentialsRequest
Skip credentials if the provided one are not valid, let the provider be invalid, avoiding to block th...
@ ParallelThreadLoading
Provider is created in a parallel thread than the one where it will live.
QFlags< ProjectFlag > ProjectFlags
Definition qgis.h:3950
@ Marker
Marker symbol.
@ Line
Line symbol.
@ Fill
Fill symbol.
static QString geoNone()
Constant that holds the string representation for "No ellipse/No CRS".
Definition qgis.h:6011
@ Preferred
Preferred format, matching the most recent WKT ISO standard. Currently an alias to WKT2_2019,...
QMap< QString, QStringList > KeywordMap
Map of vocabulary string to keyword list.
void setTitle(const QString &title)
Sets the human readable title (name) of the resource, typically displayed in search results.
QString title() const
Returns the human readable name of the resource, typically displayed in search results.
QgsAbstractMetadataBase::KeywordMap keywords() const
Returns the keywords map, which is a set of descriptive keywords associated with the resource.
virtual bool readXml(const QDomElement &collectionElem, const QgsPropertiesDefinition &definitions)
Reads property collection state from an XML element.
virtual bool writeXml(QDomElement &collectionElem, const QgsPropertiesDefinition &definitions) const
Writes the current state of the property collection into an XML element.
Represents a map layer containing a set of georeferenced annotations, e.g.
void resolveReferences(QgsProject *project) override
Resolve references to other layers (kept as layer IDs after reading XML) into layer objects.
void setTransformContext(const QgsCoordinateTransformContext &context) override
Sets the coordinate transform context to transformContext.
void reset()
Resets the annotation layer to a default state, and clears all items from it.
bool isEmpty() const
Returns true if the annotation layer is empty and contains no annotations.
Manages storage of a set of QgsAnnotation annotation objects.
static QgsApplication * instance()
Returns the singleton instance of the QgsApplication.
static QgsProjectStorageRegistry * projectStorageRegistry()
Returns registry of available project storage implementations.
static const QgsSettingsEntryString * settingsLocaleUserLocale
Settings entry locale user locale.
static QgsRuntimeProfiler * profiler()
Returns the application runtime profiler.
void collectTranslatableObjects(QgsTranslationContext *translationContext)
Emits the signal to collect all the strings of .qgs to be included in ts file.
static QgsPluginLayerRegistry * pluginLayerRegistry()
Returns the application's plugin layer registry, used for managing plugin layer types.
void requestForTranslatableObjects(QgsTranslationContext *translationContext)
Emitted when project strings which require translation are being collected for inclusion in a ....
static QString userFullName()
Returns the user's operating system login account full display name.
static QString userLoginName()
Returns the user's operating system login account name.
Manages zip/unzip operations for an archive.
Definition qgsarchive.h:35
void addFile(const QString &filename)
Add a new file to this archive.
A container for attribute editors, used to group them visually in the attribute form if it is set to ...
QList< QgsAttributeEditorElement * > children() const
Gets a list of the children elements of this container.
An abstract base class for any elements of a drag and drop form.
QString name() const
Returns the name of this element.
QgsFields auxiliaryFields() const
Returns a list of all auxiliary fields currently managed by the layer.
bool save()
Commits changes and starts editing then.
Providing some utility methods to manage auxiliary storage.
static QString extension()
Returns the extension used for auxiliary databases.
static bool deleteTable(const QgsDataSourceUri &uri)
Removes a table from the auxiliary storage.
Manages storage of a set of bookmarks.
bool readXml(const QDomElement &element, const QDomDocument &doc)
Reads the manager's state from a DOM element, restoring all bookmarks present in the XML document.
void clear()
Removes and deletes all bookmarks from the manager.
QDomElement writeXml(QDomDocument &doc) const
Returns a DOM element representing the state of the manager.
static QColor colorFromString(const QString &string)
Decodes a string into a color value.
static QString colorToString(const QColor &color)
Encodes a color into a string value.
Represents a coordinate reference system (CRS).
static QgsCoordinateReferenceSystem fromOgcWmsCrs(const QString &ogcCrs)
Creates a CRS from a given OGC WMS-format Coordinate Reference System string.
bool isValid() const
Returns whether this CRS is correctly initialized and usable.
bool hasVerticalAxis() const
Returns true if the CRS has a vertical axis.
QString toProj() const
Returns a Proj string representation of this CRS.
bool readXml(const QDomNode &node)
Restores state from the given DOM node.
static QgsCoordinateReferenceSystem createCompoundCrs(const QgsCoordinateReferenceSystem &horizontalCrs, const QgsCoordinateReferenceSystem &verticalCrs, QString &error)
Given a horizontal and vertical CRS, attempts to create a compound CRS from them.
QString ellipsoidAcronym() const
Returns the ellipsoid acronym for the ellipsoid used by the CRS.
QString projectionAcronym() const
Returns the projection acronym for the projection used by the CRS.
QgsCoordinateReferenceSystem verticalCrs() const
Returns the vertical CRS associated with this CRS object.
static QgsCoordinateReferenceSystem fromProj(const QString &proj)
Creates a CRS from a proj style formatted string.
QString toWkt(Qgis::CrsWktVariant variant=Qgis::CrsWktVariant::Wkt1Gdal, bool multiline=false, int indentationWidth=4) const
Returns a WKT representation of this CRS.
bool writeXml(QDomNode &node, QDomDocument &doc) const
Stores state to the given Dom node in the given document.
static QgsCoordinateReferenceSystem fromSrsId(long srsId)
Creates a CRS from a specified QGIS SRS ID.
Qgis::CrsType type() const
Returns the type of the CRS.
Contains information about the context in which a coordinate transform is executed.
void readSettings()
Reads the context's state from application settings.
void writeXml(QDomElement &element, const QgsReadWriteContext &context) const
Writes the context's state to a DOM element.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context, QStringList &missingTransforms)
Reads the context's state from a DOM element.
Abstract base class for spatial data provider implementations.
@ EvaluateDefaultValues
Evaluate default values on provider side when calling QgsVectorDataProvider::defaultValue( int index ...
Stores the component parts of a data source URI (e.g.
QgsAttributeEditorContainer * invisibleRootContainer()
Gets the invisible root container for the drag and drop designer form (EditorLayout::TabLayout).
Renders elevation shading on an image with different methods (eye dome lighting, hillshading,...
void writeXml(QDomElement &elem, const QgsReadWriteContext &context) const
Writes configuration on a DOM element.
void readXml(const QDomElement &element, const QgsReadWriteContext &context)
Reads configuration from a DOM element.
Single scope for storing variables and functions for use within a QgsExpressionContext.
static QgsExpressionContextScope * projectScope(const QgsProject *project)
Creates a new scope which contains variables and functions relating to a QGIS project.
static QgsExpressionContextScope * globalScope()
Creates a new scope which contains variables and functions relating to the global QGIS context.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
An abstract base class for defining QgsExpression functions.
An expression node for expression functions.
Handles parsing and evaluation of expressions (formerly called "search strings").
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:53
Container of fields for a vector layer.
Definition qgsfields.h:46
bool isEmpty
Definition qgsfields.h:49
Stores global configuration for labeling engine.
Handles sorting of dependencies stored in a XML project or layer definition file.
Layer tree group node serves as a container for layers and further groups.
void resolveReferences(const QgsProject *project, bool looseMatching=false) override
Calls resolveReferences() on child tree nodes.
QgsLayerTreeGroup * findGroup(const QString &name)
Find group node with specified name.
QList< QgsLayerTreeGroup * > findGroups(bool recursive=false) const
Find group layer nodes.
QString name() const override
Returns the group's name.
QStringList findLayerIds() const
Find layer IDs used in all layer nodes.
void insertChildNodes(int index, const QList< QgsLayerTreeNode * > &nodes)
Insert existing nodes at specified position.
void readChildrenFromXml(QDomElement &element, const QgsReadWriteContext &context)
Read children from XML and append them to the group.
QgsLayerTreeGroup * clone() const override
Returns a clone of the group.
QList< QgsLayerTreeLayer * > findLayers() const
Find all layer nodes.
QgsLayerTreeLayer * findLayer(QgsMapLayer *layer) const
Find layer node representing the map layer.
Layer tree node points to a map layer.
void resolveReferences(const QgsProject *project, bool looseMatching=false) override
Resolves reference to layer from stored layer ID (if it has not been resolved already)
Base class for nodes in a layer tree.
QList< QgsLayerTreeNode * > abandonChildren()
Removes the children, disconnect all the forwarded and external signals and sets their parent to null...
void setCustomProperty(const QString &key, const QVariant &value)
Sets a custom property for the node. Properties are stored in a map and saved in project file.
virtual void writeXml(QDomElement &parentElement, const QgsReadWriteContext &context)=0
Write layer tree to XML.
QList< QgsLayerTreeNode * > children()
Gets list of children of the node. Children are owned by the parent.
void removeCustomProperty(const QString &key)
Remove a custom property from layer. Properties are stored in a map and saved in project file.
QVariant customProperty(const QString &key, const QVariant &defaultValue=QVariant()) const
Read a custom property from layer. Properties are stored in a map and saved in project file.
void setItemVisibilityChecked(bool checked)
Check or uncheck a node (independently of its ancestors or children)
Listens to layer changes from a QgsProject and applies changes to a QgsLayerTree.
static void replaceChildrenOfEmbeddedGroups(QgsLayerTreeGroup *group)
Remove subtree of embedded groups and replaces it with a custom property embedded-visible-layers.
static void storeOriginalLayersProperties(QgsLayerTreeGroup *group, const QDomDocument *doc)
Stores in a layer's originalXmlProperties the layer properties information.
static void updateEmbeddedGroupsProjectPath(QgsLayerTreeGroup *group, const QgsProject *project)
Updates an embedded group from a project.
static bool readOldLegend(QgsLayerTreeGroup *root, const QDomElement &legendElem)
Try to load layer tree from.
Namespace with helper functions for layer tree operations.
void readLayerOrderFromXml(const QDomElement &doc)
Load the layer order from an XML element.
static QgsLayerTreeLayer * toLayer(QgsLayerTreeNode *node)
Cast node to a layer.
void clear()
Clear any information from this layer tree.
static bool isLayer(const QgsLayerTreeNode *node)
Check whether the node is a valid layer node.
static bool isGroup(QgsLayerTreeNode *node)
Check whether the node is a valid group node.
static QgsLayerTreeGroup * toGroup(QgsLayerTreeNode *node)
Cast node to a group.
QList< QgsMapLayer * > customLayerOrder() const
The order in which layers will be rendered on the canvas.
QgsLayerTree * clone() const override
Create a copy of the node. Returns new instance.
Manages storage of a set of layouts.
static void warning(const QString &msg)
Goes to qWarning.
static Qgis::LayerType typeFromString(const QString &string, bool &ok)
Returns the map layer type corresponding a string value.
A storage object for map layers, in which the layers are owned by the store and have their lifetime b...
void layersWillBeRemoved(const QStringList &layerIds)
Emitted when one or more layers are about to be removed from the store.
void layerWillBeRemoved(const QString &layerId)
Emitted when a layer is about to be removed from the store.
void layersRemoved(const QStringList &layerIds)
Emitted after one or more layers were removed from the store.
void allLayersRemoved()
Emitted when all layers are removed, before layersWillBeRemoved() and layerWillBeRemoved() signals ar...
void layerRemoved(const QString &layerId)
Emitted after a layer was removed from the store.
void layerWasAdded(QgsMapLayer *layer)
Emitted when a layer was added to the store.
QgsMapLayer * mapLayer(const QString &id) const
Retrieve a pointer to a layer by layer id.
void layersAdded(const QList< QgsMapLayer * > &layers)
Emitted when one or more layers were added to the store.
Base class for all map layer types.
Definition qgsmaplayer.h:77
QFlags< ReadFlag > ReadFlags
QString source() const
Returns the source for the layer.
QString providerType() const
Returns the provider type (provider key) for this layer.
void removeCustomProperty(const QString &key)
Remove a custom property from layer.
void configChanged()
Emitted whenever the configuration is changed.
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...
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:84
QString id
Definition qgsmaplayer.h:80
QString originalXmlProperties() const
Returns the XML properties of the original layer as they were when the layer was first read from the ...
Qgis::LayerType type
Definition qgsmaplayer.h:87
Q_INVOKABLE void setCustomProperty(const QString &key, const QVariant &value)
Set a custom property for layer.
virtual bool isEditable() const
Returns true if the layer can be edited.
bool writeLayerXml(QDomElement &layerElement, QDomDocument &document, const QgsReadWriteContext &context) const
Stores state in DOM node.
@ Identifiable
If the layer is identifiable using the identify map tool and as a WMS layer.
@ Removable
If the layer can be removed from the project. The layer will not be removable from the legend menu en...
@ FlagReadExtentFromXml
Read extent from xml and skip get extent from provider.
@ FlagTrustLayerMetadata
Trust layer metadata. Improves layer load time by skipping expensive checks like primary key unicity,...
@ FlagForceReadOnly
Force open as read only.
@ FlagDontResolveLayers
Don't resolve layer paths or create data providers for layers.
bool readLayerXml(const QDomElement &layerElement, QgsReadWriteContext &context, QgsMapLayer::ReadFlags flags=QgsMapLayer::ReadFlags(), QgsDataProvider *preloadedProvider=nullptr)
Sets state from DOM document.
void setCrs(const QgsCoordinateReferenceSystem &srs, bool emitSignal=true)
Sets layer's spatial reference system.
Container class that allows storage of map themes consisting of visible map layers and layer styles.
Manages storage of a set of views.
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).
Resolves relative paths into absolute paths and vice versa.
QString writePath(const QString &filename) const
Prepare a filename to save it to the project file.
QString readPath(const QString &filename) const
Turn filename read from the project file to an absolute path.
Allows managing the zip/unzip actions on project files.
Definition qgsarchive.h:111
QString projectFile() const
Returns the current .qgs project file or an empty string if there's none.
QString auxiliaryStorageFile() const
Returns the current .qgd auxiliary storage file or an empty string if there's none.
bool unzip(const QString &zipFilename) override
Clear the current content of this archive and unzip.
Interface for classes that handle missing layer files when reading project files.
virtual void handleBadLayers(const QList< QDomNode > &layers)
This method will be called whenever the project tries to load layers which cannot be accessed.
Contains settings and properties relating to how a QgsProject should display values such as map coord...
void reset()
Resets the settings to a default state.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context)
Reads the settings's state from a DOM element.
QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Returns a DOM element representing the settings.
Contains elevation properties for a QgsProject.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context)
Reads the property state from a DOM element.
void reset()
Resets the properties to a default state.
QDomElement writeXml(QDomDocument &document, const QgsReadWriteContext &context) const
Returns a DOM element representing the properties.
void resolveReferences(const QgsProject *project)
Resolves reference to layers from stored layer ID.
Convert from older project file versions to newer.
static Q_DECL_DEPRECATED void fixOldSymbolLayerReferences(const QMap< QString, QgsMapLayer * > &mapLayers)
QgsSymbolLayerReference uses QgsSymbolLayer unique uuid identifier since QGIS 3.30,...
Contains settings and properties relating to how a QgsProject should interact with a GPS device.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context)
Reads the settings's state from a DOM element.
void reset()
Resets the settings to a default state.
QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Returns a DOM element representing the settings.
void resolveReferences(const QgsProject *project)
Resolves reference to layers from stored layer ID (if it has not been resolved already)
A structured metadata store for a project.
bool writeMetadataXml(QDomElement &metadataElement, QDomDocument &document) const override
Stores state in a DOM node.
void setCreationDateTime(const QDateTime &creationDateTime)
Sets the project's creation date/timestamp.
bool readMetadataXml(const QDomElement &metadataElement) override
Sets state from DOM document.
void setAuthor(const QString &author)
Sets the project author string.
Project property key node.
QString name() const
The name of the property is used as identifier.
QgsProjectProperty * find(const QString &propertyName) const
Attempts to find a property with a matching sub-key name.
void removeKey(const QString &keyName)
Removes the specified key.
void dump(int tabs=0) const override
Dumps out the keys and values.
bool isEmpty() const
Returns true if this property contains no sub-keys.
virtual void clearKeys()
Deletes any sub-nodes from the property.
bool writeXml(const QString &nodeName, QDomElement &element, QDomDocument &document) override
Writes the property hierarchy to a specified DOM element.
void subkeyList(QStringList &entries) const
Returns any sub-keys contained by this property which themselves contain other keys.
void setName(const QString &name)
The name of the property is used as identifier.
QgsProjectPropertyKey * addKey(const QString &keyName)
Adds the specified property key as a sub-key.
QVariant value() const override
If this key has a value, it will be stored by its name in its properties.
QgsProjectPropertyValue * setValue(const QString &name, const QVariant &value)
Sets the value associated with this key.
void entryList(QStringList &entries) const
Returns any sub-keys contained by this property that do not contain other keys.
int count() const
Returns the number of sub-keys contained by this property.
bool readXml(const QDomNode &keyNode) override
Restores the property hierarchy from a specified DOM node.
An abstract base class for QGIS project property hierarchys.
virtual bool isKey() const =0
Returns true if the property is a QgsProjectPropertyKey.
virtual bool isValue() const =0
Returns true if the property is a QgsProjectPropertyValue.
QgsProjectStorage * projectStorageFromUri(const QString &uri)
Returns storage implementation if the URI matches one. Returns nullptr otherwise (it is a normal file...
Metadata associated with a project.
Abstract interface for project storage - to be implemented by various backends and registered in QgsP...
Contains settings and properties relating to how a QgsProject should handle styling.
QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Returns a DOM element representing the settings.
void setDefaultSymbol(Qgis::SymbolType symbolType, QgsSymbol *symbol)
Sets the project default symbol for a given type.
void reset()
Resets the settings to a default state.
void removeProjectStyle()
Removes and deletes the project style database.
void setRandomizeDefaultSymbolColor(bool randomized)
Sets whether the default symbol fill color is randomized.
void setDefaultColorRamp(QgsColorRamp *colorRamp)
Sets the project default color ramp.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context, Qgis::ProjectReadFlags flags=Qgis::ProjectReadFlags())
Reads the settings's state from a DOM element.
void setDefaultSymbolOpacity(double opacity)
Sets the default symbol opacity.
Contains temporal settings and properties for the project, this may be used when animating maps or sh...
bool readXml(const QDomElement &element, const QgsReadWriteContext &context)
Reads the settings's state from a DOM element.
void reset()
Resets the settings to a default state.
QDomElement writeXml(QDomDocument &document, const QgsReadWriteContext &context) const
Returns a DOM element representing the settings.
Describes the version of a project.
QString text() const
Returns a string representation of the version.
int majorVersion() const
Returns the major version number.
Contains settings and properties relating to how a QgsProject should be displayed inside map canvas,...
bool useProjectScales() const
Returns true if project mapScales() are enabled.
void reset()
Resets the settings to a default state.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context)
Reads the settings's state from a DOM element.
void setMapScales(const QVector< double > &scales)
Sets the list of custom project map scales.
void setUseProjectScales(bool enabled)
Sets whether project mapScales() are enabled.
QVector< double > mapScales() const
Returns the list of custom project map scales.
void mapScalesChanged()
Emitted when the list of custom project map scales changes.
QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Returns a DOM element representing the settings.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
Definition qgsproject.h:107
bool isZipped() const
Returns true if the project comes from a zip archive, false otherwise.
bool removeAttachedFile(const QString &path)
Removes the attached file.
QgsRelationManager * relationManager
Definition qgsproject.h:117
bool write()
Writes the project to its current associated file (see fileName() ).
QgsProject(QObject *parent=nullptr, Qgis::ProjectCapabilities capabilities=Qgis::ProjectCapability::ProjectStyles)
Create a new QgsProject.
void removeMapLayer(const QString &layerId)
Remove a layer from the registry by layer ID.
Q_DECL_DEPRECATED void oldProjectVersionWarning(const QString &warning)
Emitted when an old project file is read.
Q_DECL_DEPRECATED bool evaluateDefaultValues() const
Should default values be evaluated on provider side when requested and not when committed.
Qgis::DistanceUnit distanceUnits
Definition qgsproject.h:124
void layersRemoved(const QStringList &layerIds)
Emitted after one or more layers were removed from the registry.
void clear()
Clears the project, removing all settings and resetting it back to an empty, default state.
~QgsProject() override
QString error() const
Returns error message from previous read/write.
Q_DECL_DEPRECATED void setUseProjectScales(bool enabled)
Sets whether project mapScales() are enabled.
void readProjectWithContext(const QDomDocument &document, QgsReadWriteContext &context)
Emitted when a project is being read.
int readNumEntry(const QString &scope, const QString &key, int def=0, bool *ok=nullptr) const
Reads an integer from the specified scope and key.
Q_DECL_DEPRECATED void setNonIdentifiableLayers(const QList< QgsMapLayer * > &layers)
Set a list of layers which should not be taken into account on map identification.
QList< QgsMapLayer * > addMapLayers(const QList< QgsMapLayer * > &mapLayers, bool addToLegend=true, bool takeOwnership=true)
Add a list of layers to the map of loaded layers.
Qgis::ProjectFlags flags() const
Returns the project's flags, which dictate the behavior of the project.
Definition qgsproject.h:209
Q_DECL_DEPRECATED QFileInfo fileInfo() const
Returns QFileInfo object for the project's associated file.
QString presetHomePath() const
Returns any manual project home path setting, or an empty string if not set.
void setBackgroundColor(const QColor &color)
Sets the default background color used by default map canvases.
void setCrs(const QgsCoordinateReferenceSystem &crs, bool adjustEllipsoid=false)
Sets the project's native coordinate reference system.
QColor selectionColor
Definition qgsproject.h:122
QString title() const
Returns the project's title.
bool commitChanges(QStringList &commitErrors, bool stopEditing=true, QgsVectorLayer *vectorLayer=nullptr)
Attempts to commit to the underlying data provider any buffered changes made since the last to call t...
void mapThemeCollectionChanged()
Emitted when the map theme collection changes.
static QgsProject * instance()
Returns the QgsProject singleton instance.
Qgis::FilePathType filePathStorage() const
Returns the type of paths used when storing file paths in a QGS/QGZ project file.
QString createAttachedFile(const QString &nameTemplate)
Attaches a file to the project.
Q_DECL_DEPRECATED void mapScalesChanged()
Emitted when the list of custom project map scales changes.
void readVersionMismatchOccurred(const QString &fileVersion)
Emitted when a project is read and the version of QGIS used to save the project differs from the curr...
QString ellipsoid
Definition qgsproject.h:114
void fileNameChanged()
Emitted when the file name of the project changes.
Q_INVOKABLE QgsMapLayer * mapLayer(const QString &layerId) const
Retrieve a pointer to a registered layer by layer ID.
void writeMapLayer(QgsMapLayer *mapLayer, QDomElement &layerElem, QDomDocument &doc)
Emitted when a layer is being saved.
const QgsSensorManager * sensorManager() const
Returns the project's sensor manager, which manages sensors within the project.
void setSnappingConfig(const QgsSnappingConfig &snappingConfig)
The snapping configuration for this project.
void areaUnitsChanged()
Emitted when the default area units changes.
QgsPropertyCollection dataDefinedServerProperties() const
Returns the data defined properties used for overrides in user defined server parameters.
Q_DECL_DEPRECATED void nonIdentifiableLayersChanged(QStringList nonIdentifiableLayers)
Emitted when the list of layer which are excluded from map identification changes.
void layersWillBeRemoved(const QStringList &layerIds)
Emitted when one or more layers are about to be removed from the registry.
QString attachmentIdentifier(const QString &attachedFile) const
Returns an identifier for an attachment file path An attachment identifier is a string which does not...
void setScaleMethod(Qgis::ScaleCalculationMethod method)
Sets the method to use for map scale calculations for the project.
QgsVectorLayerEditBufferGroup * editBufferGroup()
Returns the edit buffer group.
void setSelectionColor(const QColor &color)
Sets the color used to highlight selected features.
bool rollBack(QStringList &rollbackErrors, bool stopEditing=true, QgsVectorLayer *vectorLayer=nullptr)
Stops a current editing operation on vectorLayer and discards any uncommitted edits.
void snappingConfigChanged(const QgsSnappingConfig &config)
Emitted whenever the configuration for snapping has changed.
QgsPathResolver pathResolver() const
Returns path resolver object with considering whether the project uses absolute or relative paths and...
void setBadLayerHandler(QgsProjectBadLayerHandler *handler)
Change handler for missing layers.
Q_DECL_DEPRECATED void setEvaluateDefaultValues(bool evaluateDefaultValues)
Defines if default values should be evaluated on provider side when requested and not when committed.
Qgis::AreaUnit areaUnits
Definition qgsproject.h:125
void crsChanged()
Emitted when the crs() of the project has changed.
QString translate(const QString &context, const QString &sourceText, const char *disambiguation=nullptr, int n=-1) const override
Translates a string using the Qt QTranslator mechanism.
const QgsProjectStyleSettings * styleSettings() const
Returns the project's style settings, which contains settings and properties relating to how a QgsPro...
QgsSnappingConfig snappingConfig
Definition qgsproject.h:116
const QgsProjectGpsSettings * gpsSettings() const
Returns the project's GPS settings, which contains settings and properties relating to how a QgsProje...
void setFileName(const QString &name)
Sets the file name associated with the project.
void avoidIntersectionsLayersChanged()
Emitted whenever avoidIntersectionsLayers has changed.
void setDataDefinedServerProperties(const QgsPropertyCollection &properties)
Sets the data defined properties used for overrides in user defined server parameters to properties.
void registerTranslatableObjects(QgsTranslationContext *translationContext)
Registers the objects that require translation into the translationContext.
void distanceUnitsChanged()
Emitted when the default distance units changes.
QgsAnnotationLayer * mainAnnotationLayer()
Returns the main annotation layer associated with the project.
const QgsBookmarkManager * bookmarkManager() const
Returns the project's bookmark manager, which manages bookmarks within the project.
void readMapLayer(QgsMapLayer *mapLayer, const QDomElement &layerNode)
Emitted after the basic initialization of a layer from the project file is done.
Q_DECL_DEPRECATED void setAutoTransaction(bool autoTransaction)
Transactional editing means that on supported datasources (postgres databases) the edit state of all ...
bool startEditing(QgsVectorLayer *vectorLayer=nullptr)
Makes the layer editable.
void aboutToBeCleared()
Emitted when the project is about to be cleared.
Q_DECL_DEPRECATED void setTrustLayerMetadata(bool trust)
Sets the trust option allowing to indicate if the extent has to be read from the XML document when da...
void cleared()
Emitted when the project is cleared (and additionally when an open project is cleared just before a n...
bool setVerticalCrs(const QgsCoordinateReferenceSystem &crs, QString *errorMessage=nullptr)
Sets the project's vertical coordinate reference system.
void setLabelingEngineSettings(const QgsLabelingEngineSettings &settings)
Sets project's global labeling engine settings.
QgsExpressionContext createExpressionContext() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
void metadataChanged()
Emitted when the project's metadata is changed.
QString resolveAttachmentIdentifier(const QString &identifier) const
Resolves an attachment identifier to a attachment file path.
const QgsProjectElevationProperties * elevationProperties() const
Returns the project's elevation properties, which contains the project's elevation related settings.
QString absolutePath() const
Returns full absolute path to the project folder if the project is stored in a file system - derived ...
void crs3DChanged()
Emitted when the crs3D() of the project has changed.
void scaleMethodChanged()
Emitted when the project's scale method is changed.
void removeMapLayers(const QStringList &layerIds)
Remove a set of layers from the registry by layer ID.
Q_DECL_DEPRECATED void setRequiredLayers(const QSet< QgsMapLayer * > &layers)
Configures a set of map layers that are required in the project and therefore they should not get rem...
bool createEmbeddedLayer(const QString &layerId, const QString &projectFilePath, QList< QDomNode > &brokenNodes, bool saveFlag=true, Qgis::ProjectReadFlags flags=Qgis::ProjectReadFlags())
Creates a maplayer instance defined in an arbitrary project file.
QList< QgsVectorLayer * > avoidIntersectionsLayers
Definition qgsproject.h:119
QString readEntry(const QString &scope, const QString &key, const QString &def=QString(), bool *ok=nullptr) const
Reads a string from the specified scope and key.
QgsExpressionContextScope * createExpressionContextScope() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
QString baseName() const
Returns the base name of the project file without the path and without extension - derived from fileN...
void ellipsoidChanged(const QString &ellipsoid)
Emitted when the project ellipsoid is changed.
QgsMapThemeCollection * mapThemeCollection
Definition qgsproject.h:115
void generateTsFile(const QString &locale)
Triggers the collection strings of .qgs to be included in ts file and calls writeTsFile()
QStringList entryList(const QString &scope, const QString &key) const
Returns a list of child keys with values which exist within the the specified scope and key.
Qgis::TransactionMode transactionMode
Definition qgsproject.h:127
QgsAnnotationManager * annotationManager()
Returns pointer to the project's annotation manager.
QgsProjectDisplaySettings * displaySettings
Definition qgsproject.h:126
QgsProjectMetadata metadata
Definition qgsproject.h:120
void projectColorsChanged()
Emitted whenever the project's color scheme has been changed.
QString saveUser() const
Returns the user name that did the last save.
QVector< T > layers() const
Returns a list of registered map layers with a specified layer type.
void setProjectColors(const QgsNamedColorList &colors)
Sets the colors for the project's color scheme (see QgsProjectColorScheme).
bool setTransactionMode(Qgis::TransactionMode transactionMode)
Set transaction mode.
QgsCoordinateTransformContext transformContext
Definition qgsproject.h:113
void transactionModeChanged()
Emitted when the transaction mode has changed.
void labelingEngineSettingsChanged()
Emitted when global configuration of the labeling engine changes.
void customVariablesChanged()
Emitted whenever the expression variables stored in the project have been changed.
QgsLayerTree * layerTreeRoot() const
Returns pointer to the root (invisible) node of the project's layer tree.
bool readBoolEntry(const QString &scope, const QString &key, bool def=false, bool *ok=nullptr) const
Reads a boolean from the specified scope and key.
QgsMapLayerStore * layerStore()
Returns a pointer to the project's internal layer store.
QString originalPath() const
Returns the original path associated with the project.
void setOriginalPath(const QString &path)
Sets the original path associated with the project.
void dumpProperties() const
Dump out current project properties to stderr.
QgsElevationShadingRenderer elevationShadingRenderer() const
Returns the elevation shading renderer used for map shading.
const QgsMapViewsManager * viewsManager() const
Returns the project's views manager, which manages map views (including 3d maps) in the project.
static void setInstance(QgsProject *project)
Set the current project singleton instance to project.
int validCount() const
Returns the number of registered valid layers.
const QgsLayoutManager * layoutManager() const
Returns the project's layout manager, which manages print layouts, atlases and reports within the pro...
void elevationShadingRendererChanged()
Emitted when the map shading renderer changes.
Q_INVOKABLE QList< QgsMapLayer * > mapLayersByName(const QString &layerName) const
Retrieve a list of matching registered layers by layer name.
QString fileName
Definition qgsproject.h:110
QgsCoordinateReferenceSystem crs3D() const
Returns the CRS to use for the project when transforming 3D data, or when z/elevation value handling ...
Q_DECL_DEPRECATED bool autoTransaction() const
Transactional editing means that on supported datasources (postgres databases) the edit state of all ...
bool accept(QgsStyleEntityVisitorInterface *visitor) const
Accepts the specified style entity visitor, causing it to visit all style entities associated with th...
QStringList attachedFiles() const
Returns a map of all attached files with identifier and real paths.
void setMetadata(const QgsProjectMetadata &metadata)
Sets the project's metadata store.
void missingDatumTransforms(const QStringList &missingTransforms)
Emitted when datum transforms stored in the project are not available locally.
QgsTransactionGroup * transactionGroup(const QString &providerKey, const QString &connString)
Returns the matching transaction group from a provider key and connection string.
QgsCoordinateReferenceSystem crs
Definition qgsproject.h:112
QgsMapLayer * addMapLayer(QgsMapLayer *mapLayer, bool addToLegend=true, bool takeOwnership=true)
Add a layer to the map of loaded layers.
QStringList nonIdentifiableLayers
Definition qgsproject.h:109
void setAvoidIntersectionsMode(const Qgis::AvoidIntersectionsMode mode)
Sets the avoid intersections mode.
void transactionGroupsChanged()
Emitted whenever a new transaction group has been created or a transaction group has been removed.
const QgsAuxiliaryStorage * auxiliaryStorage() const
Returns the current const auxiliary storage.
void reloadAllLayers()
Reload all registered layer's provider data caches, synchronising the layer with any changes in the d...
int count() const
Returns the number of registered layers.
void loadingLayerMessageReceived(const QString &layerName, const QList< QgsReadWriteContext::ReadWriteMessage > &messages)
Emitted when loading layers has produced some messages.
void setAreaUnits(Qgis::AreaUnit unit)
Sets the default area measurement units for the project.
void setTitle(const QString &title)
Sets the project's title.
QMap< QPair< QString, QString >, QgsTransactionGroup * > transactionGroups()
Map of transaction groups.
void setFlag(Qgis::ProjectFlag flag, bool enabled=true)
Sets whether a project flag is enabled.
QDateTime lastModified() const
Returns last modified time of the project file as returned by the file system (or other project stora...
bool loadFunctionsFromProject(bool force=false)
Loads python expression functions stored in the currrent project.
bool readLayer(const QDomNode &layerNode)
Reads the layer described in the associated DOM node.
double readDoubleEntry(const QString &scope, const QString &key, double def=0, bool *ok=nullptr) const
Reads a double from the specified scope and key.
bool writeEntry(const QString &scope, const QString &key, bool value)
Write a boolean value to the project file.
QString absoluteFilePath() const
Returns full absolute path to the project file if the project is stored in a file system - derived fr...
QDateTime lastSaveDateTime() const
Returns the date and time when the project was last saved.
void projectSaved()
Emitted when the project file has been written and closed.
Q_DECL_DEPRECATED bool trustLayerMetadata() const
Returns true if the trust option is activated, false otherwise.
QString writePath(const QString &filename) const
Prepare a filename to save it to the project file.
void setEllipsoid(const QString &ellipsoid)
Sets the project's ellipsoid from a proj string representation, e.g., "WGS84".
void readProject(const QDomDocument &document)
Emitted when a project is being read.
void setTransformContext(const QgsCoordinateTransformContext &context)
Sets the project's coordinate transform context, which stores various information regarding which dat...
QColor backgroundColor
Definition qgsproject.h:121
void layerLoaded(int i, int n)
Emitted when a layer from a projects was read.
QStringList subkeyList(const QString &scope, const QString &key) const
Returns a list of child keys which contain other keys that exist within the the specified scope and k...
bool read(const QString &filename, Qgis::ProjectReadFlags flags=Qgis::ProjectReadFlags())
Reads given project file from the given file.
QStringList readListEntry(const QString &scope, const QString &key, const QStringList &def=QStringList(), bool *ok=nullptr) const
Reads a string list from the specified scope and key.
void selectionColorChanged()
Emitted whenever the project's selection color has been changed.
bool topologicalEditing
Definition qgsproject.h:123
const QgsLabelingEngineSettings & labelingEngineSettings() const
Returns project's global labeling engine settings.
void removeAllMapLayers()
Removes all registered layers.
Q_DECL_DEPRECATED QVector< double > mapScales() const
Returns the list of custom project map scales.
void setDirty(bool b=true)
Flag the project as dirty (modified).
void backgroundColorChanged()
Emitted whenever the project's canvas background color has been changed.
QgsLayerTreeGroup * createEmbeddedGroup(const QString &groupName, const QString &projectFilePath, const QStringList &invisibleLayers, Qgis::ProjectReadFlags flags=Qgis::ProjectReadFlags())
Create layer group instance defined in an arbitrary project file.
const QgsProjectViewSettings * viewSettings() const
Returns the project's view settings, which contains settings and properties relating to how a QgsProj...
void cleanFunctionsFromProject()
Unloads python expression functions stored in the current project and reloads local functions from th...
QgsCoordinateReferenceSystem verticalCrs() const
Returns the project's vertical coordinate reference system.
QString readPath(const QString &filename) const
Transforms a filename read from the project file to an absolute path.
void registerTranslatableContainers(QgsTranslationContext *translationContext, QgsAttributeEditorContainer *parent, const QString &layerId)
Registers the containers that require translation into the translationContext.
void setElevationShadingRenderer(const QgsElevationShadingRenderer &elevationShadingRenderer)
Sets the elevation shading renderer used for global map shading.
void setFilePathStorage(Qgis::FilePathType type)
Sets the type of paths used when storing file paths in a QGS/QGZ project file.
Q_DECL_DEPRECATED QSet< QgsMapLayer * > requiredLayers() const
Returns a set of map layers that are required in the project and therefore they should not get remove...
void transformContextChanged()
Emitted when the project transformContext() is changed.
void setTopologicalEditing(bool enabled)
Convenience function to set topological editing.
void legendLayersAdded(const QList< QgsMapLayer * > &layers)
Emitted, when a layer was added to the registry and the legend.
QVariantMap customVariables() const
A map of custom project variables.
void setAvoidIntersectionsLayers(const QList< QgsVectorLayer * > &layers)
Sets the list of layers with which intersections should be avoided.
void homePathChanged()
Emitted when the home path of the project changes.
void dirtySet()
Emitted when setDirty(true) is called.
void setCustomVariables(const QVariantMap &customVariables)
A map of custom project variables.
void writeProject(QDomDocument &document)
Emitted when the project is being written.
QgsCoordinateReferenceSystem defaultCrsForNewLayers() const
Returns the default CRS for new layers based on the settings and the current project CRS.
QString saveUserFullName() const
Returns the full user name that did the last save.
void layersAdded(const QList< QgsMapLayer * > &layers)
Emitted when one or more layers were added to the registry.
QMap< QString, QgsMapLayer * > mapLayers(const bool validOnly=false) const
Returns a map of all registered layers by layer ID.
QString homePath
Definition qgsproject.h:111
bool isDirty() const
Returns true if the project has been modified since the last write()
QgsMapLayer * takeMapLayer(QgsMapLayer *layer)
Takes a layer from the registry.
void isDirtyChanged(bool dirty)
Emitted when the project dirty status changes.
void setDistanceUnits(Qgis::DistanceUnit unit)
Sets the default distance measurement units for the project.
Q_DECL_DEPRECATED bool useProjectScales() const
Returns true if project mapScales() are enabled.
Q_DECL_DEPRECATED void setMapScales(const QVector< double > &scales)
Sets the list of custom project map scales.
void setPresetHomePath(const QString &path)
Sets the project's home path.
void setFlags(Qgis::ProjectFlags flags)
Sets the project's flags, which dictate the behavior of the project.
QList< QgsMapLayer * > mapLayersByShortName(const QString &shortName) const
Retrieves a list of matching registered layers by layer shortName.
QgsProjectStorage * projectStorage() const
Returns pointer to project storage implementation that handles read/write of the project file.
QString layerIsEmbedded(const QString &id) const
Returns the source project file path if the layer with matching id is embedded from other project fil...
const QgsProjectTimeSettings * timeSettings() const
Returns the project's time settings, which contains the project's temporal range and other time based...
void verticalCrsChanged()
Emitted when the verticalCrs() of the project has changed.
void topologicalEditingChanged()
Emitted when the topological editing flag has changed.
bool removeEntry(const QString &scope, const QString &key)
Remove the given key from the specified scope.
QgsProjectVersion lastSaveVersion() const
Returns the QGIS version which the project was last saved using.
void avoidIntersectionsModeChanged()
Emitted whenever the avoid intersections mode has changed.
void loadingLayer(const QString &layerName)
Emitted when a layer is loaded.
A grouped map of multiple QgsProperty objects, each referenced by an integer key value.
Definition for a property.
Definition qgsproperty.h:45
@ String
Any string value.
Definition qgsproperty.h:59
Holds data provider key, description, and associated shared library file or function pointer informat...
virtual QgsProviderMetadata::ProviderCapabilities providerCapabilities() const
Returns the provider's capabilities.
@ ParallelCreateProvider
Indicates that the provider supports parallel creation, that is, can be created on another thread tha...
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.
QgsProviderMetadata * providerMetadata(const QString &providerKey) const
Returns metadata of the provider or nullptr if not found.
static bool run(const QString &command, const QString &messageOnError=QString())
Execute a Python statement.
static bool isValid()
Returns true if the runner has an instance (and thus is able to run commands)
A container for the context for various read/write operations on objects.
void setTransformContext(const QgsCoordinateTransformContext &transformContext)
Sets data coordinate transform context to transformContext.
QgsCoordinateTransformContext transformContext() const
Returns data provider coordinate transform context.
QList< QgsReadWriteContext::ReadWriteMessage > takeMessages()
Returns the stored messages and remove them.
void setProjectTranslator(QgsProjectTranslator *projectTranslator)
Sets the project translator.
void setPathResolver(const QgsPathResolver &resolver)
Sets up path resolver for conversion between relative and absolute paths.
Manages a set of relations between layers.
void clear()
Remove any relation managed by this class.
QMap< QString, QgsRelation > relations() const
Gets access to the relations managed by this class.
Represents a relationship between two vector layers.
Definition qgsrelation.h:44
Used when reading a project to asynchronously create data providers that support asynchronous creatio...
void providerCreated(bool isValid, const QString &layerId)
Emitted when a provider is created with isValid set to True when the provider is valid.
QgsDataProvider * dataProvider()
Returns the created data provider.
void clear(const QString &group="startup")
clear Clear all profile data.
Expression function for use within a QgsExpressionContextScope.
Scoped object for logging of the runtime for a single operation or group of operations.
Manages sensors.
QDomElement writeXml(QDomDocument &document) const
Returns a DOM element representing the state of the manager.
void clear()
Deregisters and removes all sensors from the manager.
bool readXml(const QDomElement &element, const QDomDocument &document)
Reads the manager's state from a DOM element, restoring all sensors present in the XML document.
T value(const QString &dynamicKeyPart=QString()) const
Returns settings value.
static const QgsSettingsEntryInteger * settingsLayerParallelLoadingMaxCount
Settings entry maximum thread count used to load layer in parallel.
static const QgsSettingsEntryBool * settingsLayerParallelLoading
Settings entry whether layer are loading in parallel.
Stores settings for use within QGIS.
Definition qgssettings.h:66
QVariant value(const QString &key, const QVariant &defaultValue=QVariant(), Section section=NoSection) const
Returns the value for setting key.
T enumValue(const QString &key, const T &defaultValue, const Section section=NoSection)
Returns the setting value for a setting based on an enum.
Stores configuration of snapping settings for the project.
bool addLayers(const QList< QgsMapLayer * > &layers)
Adds the specified layers as individual layers to the configuration with standard configuration.
void readProject(const QDomDocument &doc)
Reads the configuration from the specified QGIS project document.
void reset()
reset to default values
void writeProject(QDomDocument &doc)
Writes the configuration to the specified QGIS project document.
void clearIndividualLayerSettings()
Removes all individual layer snapping settings.
bool removeLayers(const QList< QgsMapLayer * > &layers)
Removes the specified layers from the individual layer configuration.
An interface for classes which can visit style entity (e.g.
virtual bool visitExit(const QgsStyleEntityVisitorInterface::Node &node)
Called when the visitor stops visiting a node.
virtual bool visitEnter(const QgsStyleEntityVisitorInterface::Node &node)
Called when the visitor starts visiting a node.
void triggerIconRebuild()
Triggers emission of the rebuildIconPreviews() signal.
static QgsStyle * defaultStyle(bool initialize=true)
Returns the default application-wide style.
Definition qgsstyle.cpp:146
Represents a transaction group.
bool addLayer(QgsVectorLayer *layer)
Add a layer to this transaction group.
static bool supportsTransaction(const QgsVectorLayer *layer)
Checks if the provider of a given layer supports transactions.
QString connectionString() const
Returns the connection string of the transaction.
Used for the collecting of strings from projects for translation and creation of ts files.
void registerTranslation(const QString &context, const QString &source)
Registers the source to be translated.
void setProject(QgsProject *project)
Sets the project being translated.
static Q_INVOKABLE QString toString(Qgis::DistanceUnit unit)
Returns a translated string representing a distance unit.
static Q_INVOKABLE Qgis::AreaUnit decodeAreaUnit(const QString &string, bool *ok=nullptr)
Decodes an areal unit from a string.
static Q_INVOKABLE QString encodeUnit(Qgis::DistanceUnit unit)
Encodes a distance unit to a string.
static Q_INVOKABLE Qgis::DistanceUnit decodeDistanceUnit(const QString &string, bool *ok=nullptr)
Decodes a distance unit from a string.
The edit buffer group manages a group of edit buffers.
bool commitChanges(QStringList &commitErrors, bool stopEditing=true)
Attempts to commit any changes to disk.
void clear()
Remove all layers from this edit buffer group.
bool rollBack(QStringList &rollbackErrors, bool stopEditing=true)
Stop editing and discard the edits.
void removeLayer(QgsVectorLayer *layer)
Remove a layer from this edit buffer group.
void addLayer(QgsVectorLayer *layer)
Add a layer to this edit buffer group.
Represents a vector layer which manages a vector based dataset.
Q_INVOKABLE bool startEditing()
Makes the layer editable.
bool loadAuxiliaryLayer(const QgsAuxiliaryStorage &storage, const QString &key=QString())
Loads the auxiliary layer for this vector layer.
QgsAuxiliaryLayer * auxiliaryLayer()
Returns the current auxiliary layer.
QStringList commitErrors() const
Returns a list containing any error messages generated when attempting to commit changes to the layer...
Q_INVOKABLE bool rollBack(bool deleteBuffer=true)
Stops a current editing operation and discards any uncommitted edits.
Q_INVOKABLE bool commitChanges(bool stopEditing=true)
Attempts to commit to the underlying data provider any buffered changes made since the last to call t...
QgsEditFormConfig editFormConfig
static bool isZipFile(const QString &filename)
Returns true if the file name is a zipped file ( i.e with a '.qgz' extension, false otherwise.
QList< QPair< QColor, QString > > QgsNamedColorList
List of colors paired with a friendly display name identifying the color.
T qgsEnumKeyToValue(const QString &key, const T &defaultValue, bool tryValueAsKey=true, bool *returnOk=nullptr)
Returns the value corresponding to the given key of an enum.
Definition qgis.h:6496
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:6819
QString qgsEnumValueToKey(const T &value, bool *returnOk=nullptr)
Returns the value for the given key of an enum.
Definition qgis.h:6477
QString qgsFlagValueToKeys(const T &value, bool *returnOk=nullptr)
Returns the value for the given keys of a flag.
Definition qgis.h:6535
T qgsFlagKeysToValue(const QString &keys, const T &defaultValue, bool tryValueAsKey=true, bool *returnOk=nullptr)
Returns the value corresponding to the given keys of a flag.
Definition qgis.h:6557
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:6818
#define QgsDebugCall
Definition qgslogger.h:39
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:41
#define QgsDebugError(str)
Definition qgslogger.h:40
QPointer< QgsMapLayer > QgsWeakMapLayerPointer
Weak pointer for QgsMapLayer.
void _getProperties(const QDomDocument &doc, QgsProjectPropertyKey &project_properties)
Restores any optional properties found in "doc" to "properties".
QgsPropertyCollection getDataDefinedServerProperties(const QDomDocument &doc, const QgsPropertiesDefinition &dataDefinedServerPropertyDefinitions)
Returns the data defined server properties collection found in "doc" to "dataDefinedServerProperties"...
void removeKey_(const QString &scope, const QString &key, QgsProjectPropertyKey &rootProperty)
Removes a given key.
QgsProjectVersion getVersion(const QDomDocument &doc)
Returns the version string found in the given DOM document.
QStringList makeKeyTokens_(const QString &scope, const QString &key)
Takes the given scope and key and convert them to a string list of key tokens that will be used to na...
void dump_(const QgsProjectPropertyKey &topQgsPropertyKey)
QgsProjectProperty * findKey_(const QString &scope, const QString &key, QgsProjectPropertyKey &rootProperty)
Returns the property that matches the given key sequence, if any.
QgsProjectProperty * addKey_(const QString &scope, const QString &key, QgsProjectPropertyKey *rootProperty, const QVariant &value, bool &propertiesModified)
Adds the given key and value.
CORE_EXPORT QgsProjectVersion getVersion(QDomDocument const &doc)
Returns the version string found in the given DOM document.
QMap< int, QgsPropertyDefinition > QgsPropertiesDefinition
Definition of available properties.
#define FONTMARKER_CHR_FIX
#define QGIS_PROTECT_QOBJECT_THREAD_ACCESS_NON_FATAL
#define QGIS_PROTECT_QOBJECT_THREAD_ACCESS
const QgsCoordinateReferenceSystem & crs
const QString & typeName
QDomElement layerElement
QString layerId
Qgis::DataProviderReadFlags flags
QgsDataProvider::ProviderOptions options
QString provider
QString dataSource
Setting options for loading annotation layers.
Setting options for creating vector data providers.
Single variable definition for use within a QgsExpressionContextScope.
Setting options for loading group layers.
Contains information relating to a node (i.e.