QGIS API Documentation 3.43.0-Master (3ee7834ace6)
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 geoNone();
989
990 return readEntry( QStringLiteral( "Measure" ), QStringLiteral( "/Ellipsoid" ), 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
1215 mEmbeddedLayers.clear();
1216 mRelationManager->clear();
1217 mAnnotationManager->clear();
1218 mLayoutManager->clear();
1219 m3DViewsManager->clear();
1220 mBookmarkManager->clear();
1221 mSensorManager->clear();
1222 mViewSettings->reset();
1223 mTimeSettings->reset();
1224 mElevationProperties->reset();
1225 mDisplaySettings->reset();
1226 mGpsSettings->reset();
1227 mSnappingConfig.reset();
1228 mAvoidIntersectionsMode = Qgis::AvoidIntersectionsMode::AllowIntersections;
1231
1232 mMapThemeCollection.reset( new QgsMapThemeCollection( this ) );
1234
1235 mLabelingEngineSettings->clear();
1236
1237 // must happen BEFORE archive reset, because we need to release the hold on any files which
1238 // exists within the archive. Otherwise the archive can't be removed.
1239 releaseHandlesToProjectArchive();
1240
1241 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage() );
1242 mArchive.reset( new QgsArchive() );
1243
1244 // must happen AFTER archive reset, as it will populate a new style database within the new archive
1245 mStyleSettings->reset();
1246
1248
1249 if ( !mIsBeingDeleted )
1250 {
1251 // possibly other signals should also not be thrown on destruction -- e.g. labelEngineSettingsChanged, etc.
1252 emit projectColorsChanged();
1253 }
1254
1255 // reset some default project properties
1256 // XXX THESE SHOULD BE MOVED TO STATUSBAR RELATED SOURCE
1257 writeEntry( QStringLiteral( "PositionPrecision" ), QStringLiteral( "/Automatic" ), true );
1258 writeEntry( QStringLiteral( "PositionPrecision" ), QStringLiteral( "/DecimalPlaces" ), 2 );
1259
1260 const bool defaultRelativePaths = mSettings.value( QStringLiteral( "/qgis/defaultProjectPathsRelative" ), true ).toBool();
1262
1263 int red = mSettings.value( QStringLiteral( "qgis/default_canvas_color_red" ), 255 ).toInt();
1264 int green = mSettings.value( QStringLiteral( "qgis/default_canvas_color_green" ), 255 ).toInt();
1265 int blue = mSettings.value( QStringLiteral( "qgis/default_canvas_color_blue" ), 255 ).toInt();
1266 setBackgroundColor( QColor( red, green, blue ) );
1267
1268 red = mSettings.value( QStringLiteral( "qgis/default_selection_color_red" ), 255 ).toInt();
1269 green = mSettings.value( QStringLiteral( "qgis/default_selection_color_green" ), 255 ).toInt();
1270 blue = mSettings.value( QStringLiteral( "qgis/default_selection_color_blue" ), 0 ).toInt();
1271 const int alpha = mSettings.value( QStringLiteral( "qgis/default_selection_color_alpha" ), 255 ).toInt();
1272 setSelectionColor( QColor( red, green, blue, alpha ) );
1273
1274 mSnappingConfig.clearIndividualLayerSettings();
1275
1277 mRootGroup->clear();
1278 if ( mMainAnnotationLayer )
1279 mMainAnnotationLayer->reset();
1280
1281 snapSingleBlocker.release();
1282
1283 if ( !mBlockSnappingUpdates )
1284 emit snappingConfigChanged( mSnappingConfig );
1285
1286 setDirty( false );
1287 emit homePathChanged();
1288 if ( !mBlockChangeSignalsDuringClear )
1289 {
1290 emit verticalCrsChanged();
1291 emit crs3DChanged();
1292 }
1293 emit cleared();
1294}
1295
1296// basically a debugging tool to dump property list values
1297void dump_( const QgsProjectPropertyKey &topQgsPropertyKey )
1298{
1299 QgsDebugMsgLevel( QStringLiteral( "current properties:" ), 3 );
1300 topQgsPropertyKey.dump();
1301}
1302
1331void _getProperties( const QDomDocument &doc, QgsProjectPropertyKey &project_properties )
1332{
1333 const QDomElement propertiesElem = doc.documentElement().firstChildElement( QStringLiteral( "properties" ) );
1334
1335 if ( propertiesElem.isNull() ) // no properties found, so we're done
1336 {
1337 return;
1338 }
1339
1340 const QDomNodeList scopes = propertiesElem.childNodes();
1341
1342 if ( propertiesElem.firstChild().isNull() )
1343 {
1344 QgsDebugError( QStringLiteral( "empty ``properties'' XML tag ... bailing" ) );
1345 return;
1346 }
1347
1348 if ( ! project_properties.readXml( propertiesElem ) )
1349 {
1350 QgsDebugError( QStringLiteral( "Project_properties.readXml() failed" ) );
1351 }
1352}
1353
1360QgsPropertyCollection getDataDefinedServerProperties( const QDomDocument &doc, const QgsPropertiesDefinition &dataDefinedServerPropertyDefinitions )
1361{
1362 QgsPropertyCollection ddServerProperties;
1363 // Read data defined server properties
1364 const QDomElement ddElem = doc.documentElement().firstChildElement( QStringLiteral( "dataDefinedServerProperties" ) );
1365 if ( !ddElem.isNull() )
1366 {
1367 if ( !ddServerProperties.readXml( ddElem, dataDefinedServerPropertyDefinitions ) )
1368 {
1369 QgsDebugError( QStringLiteral( "dataDefinedServerProperties.readXml() failed" ) );
1370 }
1371 }
1372 return ddServerProperties;
1373}
1374
1379static void _getTitle( const QDomDocument &doc, QString &title )
1380{
1381 const QDomElement titleNode = doc.documentElement().firstChildElement( QStringLiteral( "title" ) );
1382
1383 title.clear(); // by default the title will be empty
1384
1385 if ( titleNode.isNull() )
1386 {
1387 QgsDebugMsgLevel( QStringLiteral( "unable to find title element" ), 2 );
1388 return;
1389 }
1390
1391 if ( !titleNode.hasChildNodes() ) // if not, then there's no actual text
1392 {
1393 QgsDebugMsgLevel( QStringLiteral( "unable to find title element" ), 2 );
1394 return;
1395 }
1396
1397 const QDomNode titleTextNode = titleNode.firstChild(); // should only have one child
1398
1399 if ( !titleTextNode.isText() )
1400 {
1401 QgsDebugMsgLevel( QStringLiteral( "unable to find title element" ), 2 );
1402 return;
1403 }
1404
1405 const QDomText titleText = titleTextNode.toText();
1406
1407 title = titleText.data();
1408
1409}
1410
1411static void readProjectFileMetadata( const QDomDocument &doc, QString &lastUser, QString &lastUserFull, QDateTime &lastSaveDateTime )
1412{
1413 const QDomNodeList nl = doc.elementsByTagName( QStringLiteral( "qgis" ) );
1414
1415 if ( !nl.count() )
1416 {
1417 QgsDebugError( QStringLiteral( "unable to find qgis element" ) );
1418 return;
1419 }
1420
1421 const QDomNode qgisNode = nl.item( 0 ); // there should only be one, so zeroth element OK
1422
1423 const QDomElement qgisElement = qgisNode.toElement(); // qgis node should be element
1424 lastUser = qgisElement.attribute( QStringLiteral( "saveUser" ), QString() );
1425 lastUserFull = qgisElement.attribute( QStringLiteral( "saveUserFull" ), QString() );
1426 lastSaveDateTime = QDateTime::fromString( qgisElement.attribute( QStringLiteral( "saveDateTime" ), QString() ), Qt::ISODate );
1427}
1428
1429QgsProjectVersion getVersion( const QDomDocument &doc )
1430{
1431 const QDomNodeList nl = doc.elementsByTagName( QStringLiteral( "qgis" ) );
1432
1433 if ( !nl.count() )
1434 {
1435 QgsDebugError( QStringLiteral( " unable to find qgis element in project file" ) );
1436 return QgsProjectVersion( 0, 0, 0, QString() );
1437 }
1438
1439 const QDomNode qgisNode = nl.item( 0 ); // there should only be one, so zeroth element OK
1440
1441 const QDomElement qgisElement = qgisNode.toElement(); // qgis node should be element
1442 QgsProjectVersion projectVersion( qgisElement.attribute( QStringLiteral( "version" ) ) );
1443 return projectVersion;
1444}
1445
1447{
1449
1450 return mSnappingConfig;
1451}
1452
1454{
1456
1457 if ( mSnappingConfig == snappingConfig )
1458 return;
1459
1460 mSnappingConfig = snappingConfig;
1461 setDirty( true );
1462 emit snappingConfigChanged( mSnappingConfig );
1463}
1464
1466{
1468
1469 if ( mAvoidIntersectionsMode == mode )
1470 return;
1471
1472 mAvoidIntersectionsMode = mode;
1474}
1475
1476static QgsMapLayer::ReadFlags projectFlagsToLayerReadFlags( Qgis::ProjectReadFlags projectReadFlags, Qgis::ProjectFlags projectFlags )
1477{
1479 // Propagate don't resolve layers
1480 if ( projectReadFlags & Qgis::ProjectReadFlag::DontResolveLayers )
1482 // Propagate trust layer metadata flag
1483 // Propagate read extent from XML based trust layer metadata flag
1484 if ( ( projectFlags & Qgis::ProjectFlag::TrustStoredLayerStatistics ) || ( projectReadFlags & Qgis::ProjectReadFlag::TrustLayerMetadata ) )
1485 {
1488 }
1489 // Propagate open layers in read-only mode
1490 if ( ( projectReadFlags & Qgis::ProjectReadFlag::ForceReadOnlyLayers ) )
1491 layerFlags |= QgsMapLayer::FlagForceReadOnly;
1492
1493 return layerFlags;
1494}
1495
1505
1506void QgsProject::preloadProviders( const QVector<QDomNode> &parallelLayerNodes,
1507 const QgsReadWriteContext &context,
1508 QMap<QString, QgsDataProvider *> &loadedProviders,
1509 QgsMapLayer::ReadFlags layerReadFlags,
1510 int totalProviderCount )
1511{
1512 int i = 0;
1513 QEventLoop loop;
1514
1515 QMap<QString, LayerToLoad> layersToLoad;
1516
1517 for ( const QDomNode &node : parallelLayerNodes )
1518 {
1519 LayerToLoad layerToLoad;
1520
1521 const QDomElement layerElement = node.toElement();
1522 layerToLoad.layerElement = layerElement;
1523 layerToLoad.layerId = layerElement.namedItem( QStringLiteral( "id" ) ).toElement().text();
1524 layerToLoad.provider = layerElement.namedItem( QStringLiteral( "provider" ) ).toElement().text();
1525 layerToLoad.dataSource = layerElement.namedItem( QStringLiteral( "datasource" ) ).toElement().text();
1526
1527 layerToLoad.dataSource = QgsProviderRegistry::instance()->relativeToAbsoluteUri( layerToLoad.provider, layerToLoad.dataSource, context );
1528
1529 layerToLoad.options = QgsDataProvider::ProviderOptions( {context.transformContext()} );
1530 layerToLoad.flags = QgsMapLayer::providerReadFlags( node, layerReadFlags );
1531
1532 // Requesting credential from worker thread could lead to deadlocks because the main thread is waiting for worker thread to fininsh
1533 layerToLoad.flags.setFlag( Qgis::DataProviderReadFlag::SkipCredentialsRequest, true );
1534 layerToLoad.flags.setFlag( Qgis::DataProviderReadFlag::ParallelThreadLoading, true );
1535
1536 layersToLoad.insert( layerToLoad.layerId, layerToLoad );
1537 }
1538
1539 while ( !layersToLoad.isEmpty() )
1540 {
1541 const QList<LayerToLoad> layersToAttemptInParallel = layersToLoad.values();
1542 QString layerToAttemptInMainThread;
1543
1544 QHash<QString, QgsRunnableProviderCreator *> runnables;
1545 QThreadPool threadPool;
1546 threadPool.setMaxThreadCount( QgsSettingsRegistryCore::settingsLayerParallelLoadingMaxCount->value() );
1547
1548 for ( const LayerToLoad &lay : layersToAttemptInParallel )
1549 {
1550 QgsRunnableProviderCreator *run = new QgsRunnableProviderCreator( lay.layerId, lay.provider, lay.dataSource, lay.options, lay.flags );
1551 runnables.insert( lay.layerId, run );
1552
1553 QObject::connect( run, &QgsRunnableProviderCreator::providerCreated, run, [&]( bool isValid, const QString & layId )
1554 {
1555 if ( isValid )
1556 {
1557 layersToLoad.remove( layId );
1558 i++;
1559 QgsRunnableProviderCreator *finishedRun = runnables.value( layId, nullptr );
1560 Q_ASSERT( finishedRun );
1561
1562 std::unique_ptr<QgsDataProvider> provider( finishedRun->dataProvider() );
1563 Q_ASSERT( provider && provider->isValid() );
1564
1565 loadedProviders.insert( layId, provider.release() );
1566 emit layerLoaded( i, totalProviderCount );
1567 }
1568 else
1569 {
1570 if ( layerToAttemptInMainThread.isEmpty() )
1571 layerToAttemptInMainThread = layId;
1572 threadPool.clear(); //we have to stop all loading provider to try this layer in main thread and maybe have credentials
1573 }
1574
1575 if ( i == parallelLayerNodes.count() || !isValid )
1576 loop.quit();
1577 } );
1578 threadPool.start( run );
1579 }
1580 loop.exec();
1581
1582 threadPool.waitForDone(); // to be sure all threads are finished
1583
1584 qDeleteAll( runnables );
1585
1586 // 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
1587 auto it = layersToLoad.find( layerToAttemptInMainThread );
1588 if ( it != layersToLoad.end() )
1589 {
1590 std::unique_ptr<QgsDataProvider> provider;
1591 QString layerId;
1592 {
1593 const LayerToLoad &lay = it.value();
1594 Qgis::DataProviderReadFlags providerFlags = lay.flags;
1595 providerFlags.setFlag( Qgis::DataProviderReadFlag::SkipCredentialsRequest, false );
1596 providerFlags.setFlag( Qgis::DataProviderReadFlag::ParallelThreadLoading, false );
1597 QgsScopedRuntimeProfile profile( "Create data providers/" + lay.layerId, QStringLiteral( "projectload" ) );
1598 provider.reset( QgsProviderRegistry::instance()->createProvider( lay.provider, lay.dataSource, lay.options, providerFlags ) );
1599 i++;
1600 if ( provider && provider->isValid() )
1601 {
1602 emit layerLoaded( i, totalProviderCount );
1603 }
1604 layerId = lay.layerId;
1605 layersToLoad.erase( it );
1606 // can't access "lay" anymore -- it's now been freed
1607 }
1608 loadedProviders.insert( layerId, provider.release() );
1609 }
1610
1611 // if there still are some not loaded providers or some invalid in parallel thread we start again
1612 }
1613
1614}
1615
1616void QgsProject::releaseHandlesToProjectArchive()
1617{
1618 mStyleSettings->removeProjectStyle();
1619}
1620
1621bool QgsProject::rebuildCrs3D( QString *error )
1622{
1623 bool res = true;
1624 if ( !mCrs.isValid() )
1625 {
1627 }
1628 else if ( !mVerticalCrs.isValid() )
1629 {
1630 mCrs3D = mCrs;
1631 }
1632 else
1633 {
1634 switch ( mCrs.type() )
1635 {
1639 mCrs3D = mCrs;
1640 break;
1641
1643 {
1644 QString tempError;
1645 mCrs3D = mCrs.hasVerticalAxis() ? mCrs : QgsCoordinateReferenceSystem::createCompoundCrs( mCrs, mVerticalCrs, error ? *error : tempError );
1646 res = mCrs3D.isValid();
1647 break;
1648 }
1649
1651 // nonsense situation
1653 res = false;
1654 break;
1655
1664 {
1665 QString tempError;
1666 mCrs3D = QgsCoordinateReferenceSystem::createCompoundCrs( mCrs, mVerticalCrs, error ? *error : tempError );
1667 res = mCrs3D.isValid();
1668 break;
1669 }
1670 }
1671 }
1672 return res;
1673}
1674
1675bool QgsProject::_getMapLayers( const QDomDocument &doc, QList<QDomNode> &brokenNodes, Qgis::ProjectReadFlags flags )
1676{
1678
1679 // Layer order is set by the restoring the legend settings from project file.
1680 // This is done on the 'readProject( ... )' signal
1681
1682 QDomElement layerElement = doc.documentElement().firstChildElement( QStringLiteral( "projectlayers" ) ).firstChildElement( QStringLiteral( "maplayer" ) );
1683
1684 // process the map layer nodes
1685
1686 if ( layerElement.isNull() ) // if we have no layers to process, bail
1687 {
1688 return true; // Decided to return "true" since it's
1689 // possible for there to be a project with no
1690 // layers; but also, more imporantly, this
1691 // would cause the tests/qgsproject to fail
1692 // since the test suite doesn't currently
1693 // support test layers
1694 }
1695
1696 bool returnStatus = true;
1697 int numLayers = 0;
1698
1699 while ( ! layerElement.isNull() )
1700 {
1701 numLayers++;
1702 layerElement = layerElement.nextSiblingElement( QStringLiteral( "maplayer" ) );
1703 }
1704
1705 // order layers based on their dependencies
1706 QgsScopedRuntimeProfile profile( tr( "Sorting layers" ), QStringLiteral( "projectload" ) );
1707 const QgsLayerDefinition::DependencySorter depSorter( doc );
1708 if ( depSorter.hasCycle() )
1709 return false;
1710
1711 // Missing a dependency? We still load all the layers, otherwise the project is completely broken!
1712 if ( depSorter.hasMissingDependency() )
1713 returnStatus = false;
1714
1715 emit layerLoaded( 0, numLayers );
1716
1717 const QVector<QDomNode> sortedLayerNodes = depSorter.sortedLayerNodes();
1718 const int totalLayerCount = sortedLayerNodes.count();
1719
1720 QVector<QDomNode> parallelLoading;
1721 QMap<QString, QgsDataProvider *> loadedProviders;
1722
1725 {
1726 profile.switchTask( tr( "Load providers in parallel" ) );
1727 for ( const QDomNode &node : sortedLayerNodes )
1728 {
1729 const QDomElement element = node.toElement();
1730 if ( element.attribute( QStringLiteral( "embedded" ) ) != QLatin1String( "1" ) )
1731 {
1732 const QString layerId = node.namedItem( QStringLiteral( "id" ) ).toElement().text();
1733 if ( !depSorter.isLayerDependent( layerId ) )
1734 {
1735 const QDomNode mnl = element.namedItem( QStringLiteral( "provider" ) );
1736 const QDomElement mne = mnl.toElement();
1737 const QString provider = mne.text();
1739 if ( meta && meta->providerCapabilities().testFlag( QgsProviderMetadata::ParallelCreateProvider ) )
1740 {
1741 parallelLoading.append( node );
1742 continue;
1743 }
1744 }
1745 }
1746 }
1747
1748 QgsReadWriteContext context;
1749 context.setPathResolver( pathResolver() );
1750 if ( !parallelLoading.isEmpty() )
1751 preloadProviders( parallelLoading, context, loadedProviders, projectFlagsToLayerReadFlags( flags, mFlags ), sortedLayerNodes.count() );
1752 }
1753
1754 int i = loadedProviders.count();
1755 for ( const QDomNode &node : std::as_const( sortedLayerNodes ) )
1756 {
1757 const QDomElement element = node.toElement();
1758 const QString name = translate( QStringLiteral( "project:layers:%1" ).arg( node.namedItem( QStringLiteral( "id" ) ).toElement().text() ), node.namedItem( QStringLiteral( "layername" ) ).toElement().text() );
1759 if ( !name.isNull() )
1760 emit loadingLayer( tr( "Loading layer %1" ).arg( name ) );
1761
1762 profile.switchTask( name );
1763 if ( element.attribute( QStringLiteral( "embedded" ) ) == QLatin1String( "1" ) )
1764 {
1765 createEmbeddedLayer( element.attribute( QStringLiteral( "id" ) ), readPath( element.attribute( QStringLiteral( "project" ) ) ), brokenNodes, true, flags );
1766 }
1767 else
1768 {
1769 QgsReadWriteContext context;
1770 context.setPathResolver( pathResolver() );
1771 context.setProjectTranslator( this );
1773 QString layerId = element.namedItem( QStringLiteral( "id" ) ).toElement().text();
1774
1775 if ( !addLayer( element, brokenNodes, context, flags, loadedProviders.take( layerId ) ) )
1776 {
1777 returnStatus = false;
1778 }
1779 const auto messages = context.takeMessages();
1780 if ( !messages.isEmpty() )
1781 {
1782 emit loadingLayerMessageReceived( tr( "Loading layer %1" ).arg( name ), messages );
1783 }
1784 }
1785 emit layerLoaded( i + 1, totalLayerCount );
1786 i++;
1787 }
1788
1789 return returnStatus;
1790}
1791
1792bool QgsProject::addLayer( const QDomElement &layerElem,
1793 QList<QDomNode> &brokenNodes,
1794 QgsReadWriteContext &context,
1796 QgsDataProvider *provider )
1797{
1799
1800 const QString type = layerElem.attribute( QStringLiteral( "type" ) );
1801 QgsDebugMsgLevel( "Layer type is " + type, 4 );
1802 std::unique_ptr<QgsMapLayer> mapLayer;
1803
1804 QgsScopedRuntimeProfile profile( tr( "Create layer" ), QStringLiteral( "projectload" ) );
1805
1806 bool ok = false;
1807 const Qgis::LayerType layerType( QgsMapLayerFactory::typeFromString( type, ok ) );
1808 if ( !ok )
1809 {
1810 QgsDebugError( QStringLiteral( "Unknown layer type \"%1\"" ).arg( type ) );
1811 return false;
1812 }
1813
1814 switch ( layerType )
1815 {
1817 mapLayer = std::make_unique<QgsVectorLayer>();
1818 break;
1819
1821 mapLayer = std::make_unique<QgsRasterLayer>();
1822 break;
1823
1825 mapLayer = std::make_unique<QgsMeshLayer>();
1826 break;
1827
1829 mapLayer = std::make_unique<QgsVectorTileLayer>();
1830 break;
1831
1833 mapLayer = std::make_unique<QgsPointCloudLayer>();
1834 break;
1835
1837 mapLayer = std::make_unique<QgsTiledSceneLayer>();
1838 break;
1839
1841 {
1842 const QString typeName = layerElem.attribute( QStringLiteral( "name" ) );
1843 mapLayer.reset( QgsApplication::pluginLayerRegistry()->createLayer( typeName ) );
1844 break;
1845 }
1846
1848 {
1849 const QgsAnnotationLayer::LayerOptions options( mTransformContext );
1850 mapLayer = std::make_unique<QgsAnnotationLayer>( QString(), options );
1851 break;
1852 }
1853
1855 {
1856 const QgsGroupLayer::LayerOptions options( mTransformContext );
1857 mapLayer = std::make_unique<QgsGroupLayer>( QString(), options );
1858 break;
1859 }
1860 }
1861
1862 if ( !mapLayer )
1863 {
1864 QgsDebugError( QStringLiteral( "Unable to create layer" ) );
1865 return false;
1866 }
1867
1868 Q_CHECK_PTR( mapLayer ); // NOLINT
1869
1870 // This is tricky: to avoid a leak we need to check if the layer was already in the store
1871 // because if it was, the newly created layer will not be added to the store and it would leak.
1872 const QString layerId { layerElem.namedItem( QStringLiteral( "id" ) ).toElement().text() };
1873 Q_ASSERT( ! layerId.isEmpty() );
1874 const bool layerWasStored = layerStore()->mapLayer( layerId );
1875
1876 // have the layer restore state that is stored in Dom node
1877 QgsMapLayer::ReadFlags layerFlags = projectFlagsToLayerReadFlags( flags, mFlags );
1878
1879 profile.switchTask( tr( "Load layer source" ) );
1880 const bool layerIsValid = mapLayer->readLayerXml( layerElem, context, layerFlags, provider ) && mapLayer->isValid();
1881
1882 // apply specific settings to vector layer
1883 if ( QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mapLayer.get() ) )
1884 {
1885 vl->setReadExtentFromXml( layerFlags & QgsMapLayer::FlagReadExtentFromXml );
1886 if ( vl->dataProvider() )
1887 {
1889 vl->dataProvider()->setProviderProperty( QgsVectorDataProvider::EvaluateDefaultValues, evaluateDefaultValues );
1890 }
1891 }
1892
1893 profile.switchTask( tr( "Add layer to project" ) );
1894 QList<QgsMapLayer *> newLayers;
1895 newLayers << mapLayer.get();
1896 if ( layerIsValid || flags & Qgis::ProjectReadFlag::DontResolveLayers )
1897 {
1898 emit readMapLayer( mapLayer.get(), layerElem );
1899 addMapLayers( newLayers );
1900 // Try to resolve references here (this is necessary to set up joined fields that will be possibly used by
1901 // virtual layers that point to this layer's joined field in their query otherwise they won't be valid ),
1902 // a second attempt to resolve references will be done after all layers are loaded
1903 // see https://github.com/qgis/QGIS/issues/46834
1904 if ( QgsVectorLayer *vLayer = qobject_cast<QgsVectorLayer *>( mapLayer.get() ) )
1905 {
1906 vLayer->joinBuffer()->resolveReferences( this );
1907 }
1908 }
1909 else
1910 {
1911 // It's a bad layer: do not add to legend (the user will decide if she wants to do so)
1912 addMapLayers( newLayers, false );
1913 newLayers.first();
1914 QgsDebugError( "Unable to load " + type + " layer" );
1915 brokenNodes.push_back( layerElem );
1916 }
1917
1918 const bool wasEditable = layerElem.attribute( QStringLiteral( "editable" ), QStringLiteral( "0" ) ).toInt();
1919 if ( wasEditable )
1920 {
1921 mapLayer->setCustomProperty( QStringLiteral( "_layer_was_editable" ), true );
1922 }
1923 else
1924 {
1925 mapLayer->removeCustomProperty( QStringLiteral( "_layer_was_editable" ) );
1926 }
1927
1928 // It should be safe to delete the layer now if layer was stored, because all the store
1929 // had to to was to reset the data source in case the validity changed.
1930 if ( ! layerWasStored )
1931 {
1932 mapLayer.release();
1933 }
1934
1935 return layerIsValid;
1936}
1937
1938bool QgsProject::read( const QString &filename, Qgis::ProjectReadFlags flags )
1939{
1941
1942 mFile.setFileName( filename );
1943 mCachedHomePath.clear();
1944 mProjectScope.reset();
1945
1946 return read( flags );
1947}
1948
1950{
1952
1953 const QString filename = mFile.fileName();
1954 bool returnValue;
1955
1956 if ( QgsProjectStorage *storage = projectStorage() )
1957 {
1958 QTemporaryFile inDevice;
1959 if ( !inDevice.open() )
1960 {
1961 setError( tr( "Unable to open %1" ).arg( inDevice.fileName() ) );
1962 return false;
1963 }
1964
1965 QgsReadWriteContext context;
1966 context.setProjectTranslator( this );
1967 if ( !storage->readProject( filename, &inDevice, context ) )
1968 {
1969 QString err = tr( "Unable to open %1" ).arg( filename );
1970 QList<QgsReadWriteContext::ReadWriteMessage> messages = context.takeMessages();
1971 if ( !messages.isEmpty() )
1972 err += QStringLiteral( "\n\n" ) + messages.last().message();
1973 setError( err );
1974 return false;
1975 }
1976 returnValue = unzip( inDevice.fileName(), flags ); // calls setError() if returning false
1977 }
1978 else
1979 {
1980 if ( QgsZipUtils::isZipFile( mFile.fileName() ) )
1981 {
1982 returnValue = unzip( mFile.fileName(), flags );
1983 }
1984 else
1985 {
1986 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( *this ) );
1987 const QFileInfo finfo( mFile.fileName() );
1988 const QString attachmentsZip = finfo.absoluteDir().absoluteFilePath( QStringLiteral( "%1_attachments.zip" ).arg( finfo.completeBaseName() ) );
1989 if ( QFile( attachmentsZip ).exists() )
1990 {
1991 auto archive = std::make_unique<QgsArchive>();
1992 if ( archive->unzip( attachmentsZip ) )
1993 {
1994 releaseHandlesToProjectArchive();
1995 mArchive = std::move( archive );
1996 }
1997 }
1998 returnValue = readProjectFile( mFile.fileName(), flags );
1999 }
2000
2001 //on translation we should not change the filename back
2002 if ( !mTranslator )
2003 {
2004 mFile.setFileName( filename );
2005 mCachedHomePath.clear();
2006 mProjectScope.reset();
2007 }
2008 else
2009 {
2010 //but delete the translator
2011 mTranslator.reset( nullptr );
2012 }
2013 }
2014 emit homePathChanged();
2015 return returnValue;
2016}
2017
2018bool QgsProject::readProjectFile( const QString &filename, Qgis::ProjectReadFlags flags )
2019{
2021
2022 // avoid multiple emission of snapping updated signals
2023 ScopedIntIncrementor snapSignalBlock( &mBlockSnappingUpdates );
2024
2025 QFile projectFile( filename );
2026 clearError();
2027
2028 QgsApplication::profiler()->clear( QStringLiteral( "projectload" ) );
2029 QgsScopedRuntimeProfile profile( tr( "Setting up translations" ), QStringLiteral( "projectload" ) );
2030
2031 const QString localeFileName = QStringLiteral( "%1_%2" ).arg( QFileInfo( projectFile.fileName() ).baseName(), QgsApplication::settingsLocaleUserLocale->value() );
2032
2033 if ( QFile( QStringLiteral( "%1/%2.qm" ).arg( QFileInfo( projectFile.fileName() ).absolutePath(), localeFileName ) ).exists() )
2034 {
2035 mTranslator.reset( new QTranslator() );
2036 ( void )mTranslator->load( localeFileName, QFileInfo( projectFile.fileName() ).absolutePath() );
2037 }
2038
2039 profile.switchTask( tr( "Reading project file" ) );
2040 auto doc = std::make_unique<QDomDocument>( QStringLiteral( "qgis" ) );
2041
2042 if ( !projectFile.open( QIODevice::ReadOnly | QIODevice::Text ) )
2043 {
2044 projectFile.close();
2045
2046 setError( tr( "Unable to open %1" ).arg( projectFile.fileName() ) );
2047
2048 return false;
2049 }
2050
2051 QTextStream textStream( &projectFile );
2052#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
2053 textStream.setCodec( "UTF-8" );
2054#endif
2055 QString projectString = textStream.readAll();
2056 projectFile.close();
2057
2058 for ( int i = 0; i < 32; i++ )
2059 {
2060 if ( i == 9 || i == 10 || i == 13 )
2061 {
2062 continue;
2063 }
2064 projectString.replace( QChar( i ), QStringLiteral( "%1%2%1" ).arg( FONTMARKER_CHR_FIX, QString::number( i ) ) );
2065 }
2066
2067 // location of problem associated with errorMsg
2068 int line, column;
2069 QString errorMsg;
2070 if ( !doc->setContent( projectString, &errorMsg, &line, &column ) )
2071 {
2072 const QString errorString = tr( "Project file read error in file %1: %2 at line %3 column %4" )
2073 .arg( projectFile.fileName(), errorMsg ).arg( line ).arg( column );
2074 QgsDebugError( errorString );
2075 setError( errorString );
2076
2077 return false;
2078 }
2079
2080 projectFile.close();
2081
2082 QgsDebugMsgLevel( "Opened document " + projectFile.fileName(), 2 );
2083
2084 // get project version string, if any
2085 const QgsProjectVersion fileVersion = getVersion( *doc );
2086 const QgsProjectVersion thisVersion( Qgis::version() );
2087
2088 profile.switchTask( tr( "Updating project file" ) );
2089 if ( thisVersion > fileVersion )
2090 {
2091 const bool isOlderMajorVersion = fileVersion.majorVersion() < thisVersion.majorVersion();
2092
2093 if ( isOlderMajorVersion )
2094 {
2095 QgsLogger::warning( "Loading a file that was saved with an older "
2096 "version of qgis (saved in " + fileVersion.text() +
2097 ", loaded in " + Qgis::version() +
2098 "). Problems may occur." );
2099 }
2100
2101 QgsProjectFileTransform projectFile( *doc, fileVersion );
2102
2103 // Shows a warning when an old project file is read.
2105 emit oldProjectVersionWarning( fileVersion.text() );
2107 emit readVersionMismatchOccurred( fileVersion.text() );
2108
2109 projectFile.updateRevision( thisVersion );
2110 }
2111 else if ( fileVersion > thisVersion )
2112 {
2113 QgsLogger::warning( "Loading a file that was saved with a newer "
2114 "version of qgis (saved in " + fileVersion.text() +
2115 ", loaded in " + Qgis::version() +
2116 "). Problems may occur." );
2117
2118 emit readVersionMismatchOccurred( fileVersion.text() );
2119 }
2120
2121 // start new project, just keep the file name and auxiliary storage
2122 profile.switchTask( tr( "Creating auxiliary storage" ) );
2123 const QString fileName = mFile.fileName();
2124
2125 const QgsCoordinateReferenceSystem oldVerticalCrs = verticalCrs();
2126 const QgsCoordinateReferenceSystem oldCrs3D = mCrs3D;
2127
2128 // NOTE [ND] -- I suspect this is wrong, as the archive may contain any number of non-auxiliary
2129 // storage related files from the previously loaded project.
2130 std::unique_ptr<QgsAuxiliaryStorage> aStorage = std::move( mAuxiliaryStorage );
2131 std::unique_ptr<QgsArchive> archive = std::move( mArchive );
2132
2133 // don't emit xxxChanged signals during the clear() call, as we'll be emitting
2134 // them again after reading the properties from the project file
2135 mBlockChangeSignalsDuringClear = true;
2136 clear();
2137 mBlockChangeSignalsDuringClear = false;
2138
2139 // this is ugly, but clear() will have created a new archive and started populating it. We
2140 // need to release handles to this archive now as the subsequent call to move will need
2141 // to delete it, and requires free access to do so.
2142 releaseHandlesToProjectArchive();
2143
2144 mAuxiliaryStorage = std::move( aStorage );
2145 mArchive = std::move( archive );
2146
2147 mFile.setFileName( fileName );
2148 mCachedHomePath.clear();
2149 mProjectScope.reset();
2150 mSaveVersion = fileVersion;
2151
2152 // now get any properties
2153 profile.switchTask( tr( "Reading properties" ) );
2154 _getProperties( *doc, mProperties );
2155
2156 // now get the data defined server properties
2157 mDataDefinedServerProperties = getDataDefinedServerProperties( *doc, dataDefinedServerPropertyDefinitions() );
2158
2159 QgsDebugMsgLevel( QString::number( mProperties.count() ) + " properties read", 2 );
2160
2161#if 0
2162 dump_( mProperties );
2163#endif
2164
2165 // get older style project title
2166 QString oldTitle;
2167 _getTitle( *doc, oldTitle );
2168
2169 readProjectFileMetadata( *doc, mSaveUser, mSaveUserFull, mSaveDateTime );
2170
2171 const QDomNodeList homePathNl = doc->elementsByTagName( QStringLiteral( "homePath" ) );
2172 if ( homePathNl.count() > 0 )
2173 {
2174 const QDomElement homePathElement = homePathNl.at( 0 ).toElement();
2175 const QString homePath = homePathElement.attribute( QStringLiteral( "path" ) );
2176 if ( !homePath.isEmpty() )
2178 }
2179 else
2180 {
2181 emit homePathChanged();
2182 }
2183
2184 const QColor backgroundColor( readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorRedPart" ), 255 ),
2185 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorGreenPart" ), 255 ),
2186 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorBluePart" ), 255 ) );
2188 const QColor selectionColor( readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorRedPart" ), 255 ),
2189 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorGreenPart" ), 255 ),
2190 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorBluePart" ), 255 ),
2191 readNumEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorAlphaPart" ), 255 ) );
2193
2194
2195 const QString distanceUnitString = readEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/DistanceUnits" ), QString() );
2196 if ( !distanceUnitString.isEmpty() )
2197 setDistanceUnits( QgsUnitTypes::decodeDistanceUnit( distanceUnitString ) );
2198
2199 const QString areaUnitString = readEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/AreaUnits" ), QString() );
2200 if ( !areaUnitString.isEmpty() )
2201 setAreaUnits( QgsUnitTypes::decodeAreaUnit( areaUnitString ) );
2202
2203 QgsReadWriteContext context;
2204 context.setPathResolver( pathResolver() );
2205 context.setProjectTranslator( this );
2206
2207 //crs
2209 if ( readNumEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectionsEnabled" ), 0 ) )
2210 {
2211 // first preference - dedicated projectCrs node
2212 const QDomNode srsNode = doc->documentElement().namedItem( QStringLiteral( "projectCrs" ) );
2213 if ( !srsNode.isNull() )
2214 {
2215 projectCrs.readXml( srsNode );
2216 }
2217
2218 if ( !projectCrs.isValid() )
2219 {
2220 const QString projCrsString = readEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectCRSProj4String" ) );
2221 const long currentCRS = readNumEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectCRSID" ), -1 );
2222 const QString authid = readEntry( QStringLiteral( "SpatialRefSys" ), QStringLiteral( "/ProjectCrs" ) );
2223
2224 // authid should be prioritized over all
2225 const bool isUserAuthId = authid.startsWith( QLatin1String( "USER:" ), Qt::CaseInsensitive );
2226 if ( !authid.isEmpty() && !isUserAuthId )
2227 projectCrs = QgsCoordinateReferenceSystem( authid );
2228
2229 // try the CRS
2230 if ( !projectCrs.isValid() && currentCRS >= 0 )
2231 {
2232 projectCrs = QgsCoordinateReferenceSystem::fromSrsId( currentCRS );
2233 }
2234
2235 // if that didn't produce a match, try the proj.4 string
2236 if ( !projCrsString.isEmpty() && ( authid.isEmpty() || isUserAuthId ) && ( !projectCrs.isValid() || projectCrs.toProj() != projCrsString ) )
2237 {
2238 projectCrs = QgsCoordinateReferenceSystem::fromProj( projCrsString );
2239 }
2240
2241 // last just take the given id
2242 if ( !projectCrs.isValid() )
2243 {
2244 projectCrs = QgsCoordinateReferenceSystem::fromSrsId( currentCRS );
2245 }
2246 }
2247 }
2248 mCrs = projectCrs;
2249
2250 //vertical CRS
2251 {
2253 const QDomNode verticalCrsNode = doc->documentElement().namedItem( QStringLiteral( "verticalCrs" ) );
2254 if ( !verticalCrsNode.isNull() )
2255 {
2256 verticalCrs.readXml( verticalCrsNode );
2257 }
2258 mVerticalCrs = verticalCrs;
2259 }
2260 rebuildCrs3D();
2261
2262 QStringList datumErrors;
2263 if ( !mTransformContext.readXml( doc->documentElement(), context, datumErrors ) && !datumErrors.empty() )
2264 {
2265 emit missingDatumTransforms( datumErrors );
2266 }
2268
2269 // map shading
2270 const QDomNode elevationShadingNode = doc->documentElement().namedItem( QStringLiteral( "elevation-shading-renderer" ) );
2271 if ( !elevationShadingNode.isNull() )
2272 {
2273 mElevationShadingRenderer.readXml( elevationShadingNode.toElement(), context );
2274 }
2276
2277
2278 //add variables defined in project file - do this early in the reading cycle, as other components
2279 //(e.g. layouts) may depend on these variables
2280 const QStringList variableNames = readListEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableNames" ) );
2281 const QStringList variableValues = readListEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableValues" ) );
2282
2283 mCustomVariables.clear();
2284 if ( variableNames.length() == variableValues.length() )
2285 {
2286 for ( int i = 0; i < variableNames.length(); ++i )
2287 {
2288 mCustomVariables.insert( variableNames.at( i ), variableValues.at( i ) );
2289 }
2290 }
2291 else
2292 {
2293 QgsMessageLog::logMessage( tr( "Project Variables Invalid" ), tr( "The project contains invalid variable settings." ) );
2294 }
2295
2296 // Register expression functions stored in the project.
2297 // They might be using project variables and might be
2298 // in turn being used by other components (e.g., layouts).
2300
2301 QDomElement element = doc->documentElement().firstChildElement( QStringLiteral( "projectMetadata" ) );
2302
2303 if ( !element.isNull() )
2304 {
2305 mMetadata.readMetadataXml( element );
2306 }
2307 else
2308 {
2309 // older project, no metadata => remove auto generated metadata which is populated on QgsProject::clear()
2310 mMetadata = QgsProjectMetadata();
2311 }
2312 if ( mMetadata.title().isEmpty() && !oldTitle.isEmpty() )
2313 {
2314 // upgrade older title storage to storing within project metadata.
2315 mMetadata.setTitle( oldTitle );
2316 }
2317 emit metadataChanged();
2318
2319 // Transaction mode
2320 element = doc->documentElement().firstChildElement( QStringLiteral( "transaction" ) );
2321 if ( !element.isNull() )
2322 {
2323 mTransactionMode = qgsEnumKeyToValue( element.attribute( QStringLiteral( "mode" ) ), Qgis::TransactionMode::Disabled );
2324 }
2325 else
2326 {
2327 // maybe older project => try read autotransaction
2328 element = doc->documentElement().firstChildElement( QStringLiteral( "autotransaction" ) );
2329 if ( ! element.isNull() )
2330 {
2331 mTransactionMode = static_cast<Qgis::TransactionMode>( element.attribute( QStringLiteral( "active" ), QStringLiteral( "0" ) ).toInt() );
2332 }
2333 }
2334
2335 // read the layer tree from project file
2336 profile.switchTask( tr( "Loading layer tree" ) );
2337 mRootGroup->setCustomProperty( QStringLiteral( "loading" ), 1 );
2338
2339 QDomElement layerTreeElem = doc->documentElement().firstChildElement( QStringLiteral( "layer-tree-group" ) );
2340 if ( !layerTreeElem.isNull() )
2341 {
2342 // Use a temporary tree to read the nodes to prevent signals being delivered to the models
2343 QgsLayerTree tempTree;
2344 tempTree.readChildrenFromXml( layerTreeElem, context );
2345 mRootGroup->insertChildNodes( -1, tempTree.abandonChildren() );
2346 }
2347 else
2348 {
2349 QgsLayerTreeUtils::readOldLegend( mRootGroup, doc->documentElement().firstChildElement( QStringLiteral( "legend" ) ) );
2350 }
2351
2352 mLayerTreeRegistryBridge->setEnabled( false );
2353
2354 // get the map layers
2355 profile.switchTask( tr( "Reading map layers" ) );
2356
2357 loadProjectFlags( doc.get() );
2358
2359 QList<QDomNode> brokenNodes;
2360 const bool clean = _getMapLayers( *doc, brokenNodes, flags );
2361
2362 // review the integrity of the retrieved map layers
2363 if ( !clean && !( flags & Qgis::ProjectReadFlag::DontResolveLayers ) )
2364 {
2365 QgsDebugError( QStringLiteral( "Unable to get map layers from project file." ) );
2366
2367 if ( !brokenNodes.isEmpty() )
2368 {
2369 QgsDebugError( "there are " + QString::number( brokenNodes.size() ) + " broken layers" );
2370 }
2371
2372 // we let a custom handler decide what to do with missing layers
2373 // (default implementation ignores them, there's also a GUI handler that lets user choose correct path)
2374 mBadLayerHandler->handleBadLayers( brokenNodes );
2375 }
2376
2377 mMainAnnotationLayer->readLayerXml( doc->documentElement().firstChildElement( QStringLiteral( "main-annotation-layer" ) ), context );
2378 mMainAnnotationLayer->setTransformContext( mTransformContext );
2379
2380 // load embedded groups and layers
2381 profile.switchTask( tr( "Loading embedded layers" ) );
2382 loadEmbeddedNodes( mRootGroup, flags );
2383
2384 // Resolve references to other layers
2385 // Needs to be done here once all dependent layers are loaded
2386 profile.switchTask( tr( "Resolving layer references" ) );
2387 QMap<QString, QgsMapLayer *> layers = mLayerStore->mapLayers();
2388 for ( QMap<QString, QgsMapLayer *>::iterator it = layers.begin(); it != layers.end(); ++it )
2389 {
2390 it.value()->resolveReferences( this );
2391 }
2392 mMainAnnotationLayer->resolveReferences( this );
2393
2394 mLayerTreeRegistryBridge->setEnabled( true );
2395
2396 // now that layers are loaded, we can resolve layer tree's references to the layers
2397 profile.switchTask( tr( "Resolving references" ) );
2398 mRootGroup->resolveReferences( this );
2399
2400 // we need to migrate old fashion designed QgsSymbolLayerReference to new ones
2401 if ( QgsProjectVersion( 3, 28, 0 ) > mSaveVersion )
2402 {
2406 }
2407
2408 if ( !layerTreeElem.isNull() )
2409 {
2410 mRootGroup->readLayerOrderFromXml( layerTreeElem );
2411 }
2412
2413 // Load pre 3.0 configuration
2414 const QDomElement layerTreeCanvasElem = doc->documentElement().firstChildElement( QStringLiteral( "layer-tree-canvas" ) );
2415 if ( !layerTreeCanvasElem.isNull( ) )
2416 {
2417 mRootGroup->readLayerOrderFromXml( layerTreeCanvasElem );
2418 }
2419
2420 // Convert pre 3.4 to create layers flags
2421 if ( QgsProjectVersion( 3, 4, 0 ) > mSaveVersion )
2422 {
2423 const QStringList requiredLayerIds = readListEntry( QStringLiteral( "RequiredLayers" ), QStringLiteral( "Layers" ) );
2424 for ( const QString &layerId : requiredLayerIds )
2425 {
2426 if ( QgsMapLayer *layer = mapLayer( layerId ) )
2427 {
2428 layer->setFlags( layer->flags() & ~QgsMapLayer::Removable );
2429 }
2430 }
2431 const QStringList disabledLayerIds = readListEntry( QStringLiteral( "Identify" ), QStringLiteral( "/disabledLayers" ) );
2432 for ( const QString &layerId : disabledLayerIds )
2433 {
2434 if ( QgsMapLayer *layer = mapLayer( layerId ) )
2435 {
2436 layer->setFlags( layer->flags() & ~QgsMapLayer::Identifiable );
2437 }
2438 }
2439 }
2440
2441 // Convert pre 3.26 default styles
2442 if ( QgsProjectVersion( 3, 26, 0 ) > mSaveVersion )
2443 {
2444 // Convert default symbols
2445 QString styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Marker" ) );
2446 if ( !styleName.isEmpty() )
2447 {
2448 std::unique_ptr<QgsSymbol> symbol( QgsStyle::defaultStyle()->symbol( styleName ) );
2450 }
2451 styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Line" ) );
2452 if ( !styleName.isEmpty() )
2453 {
2454 std::unique_ptr<QgsSymbol> symbol( QgsStyle::defaultStyle()->symbol( styleName ) );
2456 }
2457 styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Fill" ) );
2458 if ( !styleName.isEmpty() )
2459 {
2460 std::unique_ptr<QgsSymbol> symbol( QgsStyle::defaultStyle()->symbol( styleName ) );
2462 }
2463 styleName = readEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/ColorRamp" ) );
2464 if ( !styleName.isEmpty() )
2465 {
2466 std::unique_ptr<QgsColorRamp> colorRamp( QgsStyle::defaultStyle()->colorRamp( styleName ) );
2467 styleSettings()->setDefaultColorRamp( colorRamp.get() );
2468 }
2469
2470 // Convert randomize default symbol fill color
2471 styleSettings()->setRandomizeDefaultSymbolColor( readBoolEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/RandomColors" ), true ) );
2472
2473 // Convert default symbol opacity
2474 double opacity = 1.0;
2475 bool ok = false;
2476 // upgrade old setting
2477 double alpha = readDoubleEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/AlphaInt" ), 255, &ok );
2478 if ( ok )
2479 opacity = alpha / 255.0;
2480 double newOpacity = readDoubleEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Opacity" ), 1.0, &ok );
2481 if ( ok )
2482 opacity = newOpacity;
2484
2485 // Cleanup
2486 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Marker" ) );
2487 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Line" ) );
2488 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Fill" ) );
2489 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/ColorRamp" ) );
2490 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/RandomColors" ) );
2491 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/AlphaInt" ) );
2492 removeEntry( QStringLiteral( "DefaultStyles" ), QStringLiteral( "/Opacity" ) );
2493 }
2494
2495 // After bad layer handling we might still have invalid layers,
2496 // store them in case the user wanted to handle them later
2497 // or wanted to pass them through when saving
2499 {
2500 profile.switchTask( tr( "Storing original layer properties" ) );
2501 QgsLayerTreeUtils::storeOriginalLayersProperties( mRootGroup, doc.get() );
2502 }
2503
2504 mRootGroup->removeCustomProperty( QStringLiteral( "loading" ) );
2505
2506 profile.switchTask( tr( "Loading map themes" ) );
2507 mMapThemeCollection.reset( new QgsMapThemeCollection( this ) );
2509 mMapThemeCollection->readXml( *doc );
2510
2511 profile.switchTask( tr( "Loading label settings" ) );
2512 mLabelingEngineSettings->readSettingsFromProject( this );
2513 {
2514 const QDomElement labelEngineSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "labelEngineSettings" ) );
2515 mLabelingEngineSettings->readXml( labelEngineSettingsElement, context );
2516 }
2517 mLabelingEngineSettings->resolveReferences( this );
2518
2520
2521 profile.switchTask( tr( "Loading annotations" ) );
2523 {
2524 mAnnotationManager->readXml( doc->documentElement(), context );
2525 }
2526 else
2527 {
2528 mAnnotationManager->readXmlAndUpgradeToAnnotationLayerItems( doc->documentElement(), context, mMainAnnotationLayer, mTransformContext );
2529 }
2531 {
2532 profile.switchTask( tr( "Loading layouts" ) );
2533 mLayoutManager->readXml( doc->documentElement(), *doc );
2534 }
2535
2537 {
2538 profile.switchTask( tr( "Loading 3D Views" ) );
2539 m3DViewsManager->readXml( doc->documentElement(), *doc );
2540 }
2541
2542 profile.switchTask( tr( "Loading bookmarks" ) );
2543 mBookmarkManager->readXml( doc->documentElement(), *doc );
2544
2545 profile.switchTask( tr( "Loading sensors" ) );
2546 mSensorManager->readXml( doc->documentElement(), *doc );
2547
2548 // reassign change dependencies now that all layers are loaded
2549 QMap<QString, QgsMapLayer *> existingMaps = mapLayers();
2550 for ( QMap<QString, QgsMapLayer *>::iterator it = existingMaps.begin(); it != existingMaps.end(); ++it )
2551 {
2552 it.value()->setDependencies( it.value()->dependencies() );
2553 }
2554
2555 profile.switchTask( tr( "Loading snapping settings" ) );
2556 mSnappingConfig.readProject( *doc );
2557 mAvoidIntersectionsMode = static_cast<Qgis::AvoidIntersectionsMode>( readNumEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsMode" ), static_cast<int>( Qgis::AvoidIntersectionsMode::AvoidIntersectionsLayers ) ) );
2558
2559 profile.switchTask( tr( "Loading view settings" ) );
2560 // restore older project scales settings
2561 mViewSettings->setUseProjectScales( readBoolEntry( QStringLiteral( "Scales" ), QStringLiteral( "/useProjectScales" ) ) );
2562 const QStringList scales = readListEntry( QStringLiteral( "Scales" ), QStringLiteral( "/ScalesList" ) );
2563 QVector<double> res;
2564 for ( const QString &scale : scales )
2565 {
2566 const QStringList parts = scale.split( ':' );
2567 if ( parts.size() != 2 )
2568 continue;
2569
2570 bool ok = false;
2571 const double denominator = QLocale().toDouble( parts[1], &ok );
2572 if ( ok )
2573 {
2574 res << denominator;
2575 }
2576 }
2577 mViewSettings->setMapScales( res );
2578 const QDomElement viewSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectViewSettings" ) );
2579 if ( !viewSettingsElement.isNull() )
2580 mViewSettings->readXml( viewSettingsElement, context );
2581
2582 // restore style settings
2583 profile.switchTask( tr( "Loading style properties" ) );
2584 const QDomElement styleSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectStyleSettings" ) );
2585 if ( !styleSettingsElement.isNull() )
2586 {
2587 mStyleSettings->removeProjectStyle();
2588 mStyleSettings->readXml( styleSettingsElement, context, flags );
2589 }
2590
2591 // restore time settings
2592 profile.switchTask( tr( "Loading temporal settings" ) );
2593 const QDomElement timeSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectTimeSettings" ) );
2594 if ( !timeSettingsElement.isNull() )
2595 mTimeSettings->readXml( timeSettingsElement, context );
2596
2597
2598 profile.switchTask( tr( "Loading elevation properties" ) );
2599 const QDomElement elevationPropertiesElement = doc->documentElement().firstChildElement( QStringLiteral( "ElevationProperties" ) );
2600 if ( !elevationPropertiesElement.isNull() )
2601 mElevationProperties->readXml( elevationPropertiesElement, context );
2602 mElevationProperties->resolveReferences( this );
2603
2604 profile.switchTask( tr( "Loading display settings" ) );
2605 {
2606 const QDomElement displaySettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectDisplaySettings" ) );
2607 if ( !displaySettingsElement.isNull() )
2608 mDisplaySettings->readXml( displaySettingsElement, context );
2609 }
2610
2611 profile.switchTask( tr( "Loading GPS settings" ) );
2612 {
2613 const QDomElement gpsSettingsElement = doc->documentElement().firstChildElement( QStringLiteral( "ProjectGpsSettings" ) );
2614 if ( !gpsSettingsElement.isNull() )
2615 mGpsSettings->readXml( gpsSettingsElement, context );
2616 mGpsSettings->resolveReferences( this );
2617 }
2618
2619 profile.switchTask( tr( "Updating variables" ) );
2621 profile.switchTask( tr( "Updating CRS" ) );
2622 emit crsChanged();
2623 if ( verticalCrs() != oldVerticalCrs )
2624 emit verticalCrsChanged();
2625 if ( mCrs3D != oldCrs3D )
2626 emit crs3DChanged();
2627 emit ellipsoidChanged( ellipsoid() );
2628
2629 // read the project: used by map canvas and legend
2630 profile.switchTask( tr( "Reading external settings" ) );
2631 emit readProject( *doc );
2632 emit readProjectWithContext( *doc, context );
2633
2634 profile.switchTask( tr( "Updating interface" ) );
2635
2636 snapSignalBlock.release();
2637 if ( !mBlockSnappingUpdates )
2638 emit snappingConfigChanged( mSnappingConfig );
2639
2642 emit projectColorsChanged();
2643
2644 // if all went well, we're allegedly in pristine state
2645 if ( clean )
2646 setDirty( false );
2647
2648 QgsDebugMsgLevel( QStringLiteral( "Project save user: %1" ).arg( mSaveUser ), 2 );
2649 QgsDebugMsgLevel( QStringLiteral( "Project save user: %1" ).arg( mSaveUserFull ), 2 );
2650
2654
2655 if ( mTranslator )
2656 {
2657 //project possibly translated -> rename it with locale postfix
2658 const QString newFileName( QStringLiteral( "%1/%2.qgs" ).arg( QFileInfo( projectFile.fileName() ).absolutePath(), localeFileName ) );
2659 setFileName( newFileName );
2660
2661 if ( write() )
2662 {
2663 setTitle( localeFileName );
2664 QgsMessageLog::logMessage( tr( "Translated project saved with locale prefix %1" ).arg( newFileName ), QObject::tr( "Project translation" ), Qgis::MessageLevel::Success );
2665 }
2666 else
2667 {
2668 QgsMessageLog::logMessage( tr( "Error saving translated project with locale prefix %1" ).arg( newFileName ), QObject::tr( "Project translation" ), Qgis::MessageLevel::Critical );
2669 }
2670 }
2671
2672 // lastly, make any previously editable layers editable
2673 const QMap<QString, QgsMapLayer *> loadedLayers = mapLayers();
2674 for ( auto it = loadedLayers.constBegin(); it != loadedLayers.constEnd(); ++it )
2675 {
2676 if ( it.value()->isValid() && it.value()->customProperty( QStringLiteral( "_layer_was_editable" ) ).toBool() )
2677 {
2678 if ( QgsVectorLayer *vl = qobject_cast< QgsVectorLayer * >( it.value() ) )
2679 vl->startEditing();
2680 it.value()->removeCustomProperty( QStringLiteral( "_layer_was_editable" ) );
2681 }
2682 }
2683
2684 return true;
2685}
2686
2687bool QgsProject::loadEmbeddedNodes( QgsLayerTreeGroup *group, Qgis::ProjectReadFlags flags )
2688{
2690
2691 bool valid = true;
2692 const auto constChildren = group->children();
2693 for ( QgsLayerTreeNode *child : constChildren )
2694 {
2695 if ( QgsLayerTree::isGroup( child ) )
2696 {
2697 QgsLayerTreeGroup *childGroup = QgsLayerTree::toGroup( child );
2698 if ( childGroup->customProperty( QStringLiteral( "embedded" ) ).toInt() )
2699 {
2700 // make sure to convert the path from relative to absolute
2701 const QString projectPath = readPath( childGroup->customProperty( QStringLiteral( "embedded_project" ) ).toString() );
2702 childGroup->setCustomProperty( QStringLiteral( "embedded_project" ), projectPath );
2703 QgsLayerTreeGroup *newGroup = createEmbeddedGroup( childGroup->name(), projectPath, childGroup->customProperty( QStringLiteral( "embedded-invisible-layers" ) ).toStringList(), flags );
2704 if ( newGroup )
2705 {
2706 QList<QgsLayerTreeNode *> clonedChildren;
2707 const QList<QgsLayerTreeNode *> constChildren = newGroup->children();
2708 clonedChildren.reserve( constChildren.size() );
2709 for ( QgsLayerTreeNode *newGroupChild : constChildren )
2710 clonedChildren << newGroupChild->clone();
2711 delete newGroup;
2712
2713 childGroup->insertChildNodes( 0, clonedChildren );
2714 }
2715 }
2716 else
2717 {
2718 loadEmbeddedNodes( childGroup, flags );
2719 }
2720 }
2721 else if ( QgsLayerTree::isLayer( child ) )
2722 {
2723 if ( child->customProperty( QStringLiteral( "embedded" ) ).toInt() )
2724 {
2725 QList<QDomNode> brokenNodes;
2726 if ( ! createEmbeddedLayer( QgsLayerTree::toLayer( child )->layerId(), readPath( child->customProperty( QStringLiteral( "embedded_project" ) ).toString() ), brokenNodes, true, flags ) )
2727 {
2728 valid = valid && false;
2729 }
2730 }
2731 }
2732
2733 }
2734
2735 return valid;
2736}
2737
2739{
2740 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
2742
2743 return mCustomVariables;
2744}
2745
2746void QgsProject::setCustomVariables( const QVariantMap &variables )
2747{
2749
2750 if ( variables == mCustomVariables )
2751 return;
2752
2753 //write variable to project
2754 QStringList variableNames;
2755 QStringList variableValues;
2756
2757 QVariantMap::const_iterator it = variables.constBegin();
2758 for ( ; it != variables.constEnd(); ++it )
2759 {
2760 variableNames << it.key();
2761 variableValues << it.value().toString();
2762 }
2763
2764 writeEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableNames" ), variableNames );
2765 writeEntry( QStringLiteral( "Variables" ), QStringLiteral( "/variableValues" ), variableValues );
2766
2767 mCustomVariables = variables;
2768 mProjectScope.reset();
2769
2771}
2772
2774{
2776
2777 *mLabelingEngineSettings = settings;
2779}
2780
2782{
2784
2785 return *mLabelingEngineSettings;
2786}
2787
2789{
2791
2792 mProjectScope.reset();
2793 return mLayerStore.get();
2794}
2795
2797{
2799
2800 return mLayerStore.get();
2801}
2802
2803QList<QgsVectorLayer *> QgsProject::avoidIntersectionsLayers() const
2804{
2806
2807 QList<QgsVectorLayer *> layers;
2808 const QStringList layerIds = readListEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsList" ), QStringList() );
2809 const auto constLayerIds = layerIds;
2810 for ( const QString &layerId : constLayerIds )
2811 {
2812 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mapLayer( layerId ) ) )
2813 layers << vlayer;
2814 }
2815 return layers;
2816}
2817
2818void QgsProject::setAvoidIntersectionsLayers( const QList<QgsVectorLayer *> &layers )
2819{
2821
2822 QStringList list;
2823 list.reserve( layers.size() );
2824
2825 for ( QgsVectorLayer *layer : layers )
2826 {
2827 if ( layer->geometryType() == Qgis::GeometryType::Polygon )
2828 list << layer->id();
2829 }
2830
2831 writeEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsList" ), list );
2833}
2834
2846
2848{
2849 // this method is called quite extensively using QgsProject::instance() skip-keyword-check
2851
2852 // MUCH cheaper to clone than build
2853 if ( mProjectScope )
2854 {
2855 auto projectScope = std::make_unique< QgsExpressionContextScope >( *mProjectScope );
2856
2857 // we can't cache these variables
2858 projectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_distance_units" ), QgsUnitTypes::toString( distanceUnits() ), true, true ) );
2859 projectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_area_units" ), QgsUnitTypes::toString( areaUnits() ), true, true ) );
2860
2861 // neither this function
2862 projectScope->addFunction( QStringLiteral( "sensor_data" ), new GetSensorData( sensorManager()->sensorsData() ) );
2863
2864 return projectScope.release();
2865 }
2866
2867 mProjectScope = std::make_unique< QgsExpressionContextScope >( QObject::tr( "Project" ) );
2868
2869 const QVariantMap vars = customVariables();
2870
2871 QVariantMap::const_iterator it = vars.constBegin();
2872
2873 for ( ; it != vars.constEnd(); ++it )
2874 {
2875 mProjectScope->setVariable( it.key(), it.value(), true );
2876 }
2877
2878 QString projectPath = projectStorage() ? fileName() : absoluteFilePath();
2879 if ( projectPath.isEmpty() )
2880 projectPath = mOriginalPath;
2881 const QString projectFolder = QFileInfo( projectPath ).path();
2882 const QString projectFilename = QFileInfo( projectPath ).fileName();
2883 const QString projectBasename = baseName();
2884
2885 //add other known project variables
2886 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_title" ), title(), true, true ) );
2887 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_path" ), QDir::toNativeSeparators( projectPath ), true, true ) );
2888 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_folder" ), QDir::toNativeSeparators( projectFolder ), true, true ) );
2889 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_filename" ), projectFilename, true, true ) );
2890 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_basename" ), projectBasename, true, true ) );
2891 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_home" ), QDir::toNativeSeparators( homePath() ), true, true ) );
2892 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_last_saved" ), mSaveDateTime.isNull() ? QVariant() : QVariant( mSaveDateTime ), true, true ) );
2893
2894 const QgsCoordinateReferenceSystem projectCrs = crs();
2895 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs" ), projectCrs.authid(), true, true ) );
2896 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_definition" ), projectCrs.toProj(), true, true ) );
2897 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_description" ), projectCrs.description(), true, true ) );
2898 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_acronym" ), projectCrs.projectionAcronym(), true ) );
2899 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_ellipsoid" ), projectCrs.ellipsoidAcronym(), true ) );
2900 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_proj4" ), projectCrs.toProj(), true ) );
2901 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_crs_wkt" ), projectCrs.toWkt( Qgis::CrsWktVariant::Preferred ), true ) );
2902
2903 const QgsCoordinateReferenceSystem projectVerticalCrs = QgsProject::verticalCrs();
2904 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs" ), projectVerticalCrs.authid(), true, true ) );
2905 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs_definition" ), projectVerticalCrs.toProj(), true, true ) );
2906 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs_description" ), projectVerticalCrs.description(), true, true ) );
2907 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_vertical_crs_wkt" ), projectVerticalCrs.toWkt( Qgis::CrsWktVariant::Preferred ), true ) );
2908
2909 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_ellipsoid" ), ellipsoid(), true, true ) );
2910 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "_project_transform_context" ), QVariant::fromValue<QgsCoordinateTransformContext>( transformContext() ), true, true ) );
2911 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_units" ), QgsUnitTypes::toString( projectCrs.mapUnits() ), true ) );
2912
2913 // metadata
2914 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_author" ), metadata().author(), true, true ) );
2915 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_abstract" ), metadata().abstract(), true, true ) );
2916 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_creation_date" ), metadata().creationDateTime(), true, true ) );
2917 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_identifier" ), metadata().identifier(), true, true ) );
2918
2919 // keywords
2920 QVariantMap keywords;
2921 const QgsAbstractMetadataBase::KeywordMap metadataKeywords = metadata().keywords();
2922 for ( auto it = metadataKeywords.constBegin(); it != metadataKeywords.constEnd(); ++it )
2923 {
2924 keywords.insert( it.key(), it.value() );
2925 }
2926 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "project_keywords" ), keywords, true, true ) );
2927
2928 // layers
2929 QVariantList layersIds;
2930 QVariantList layers;
2931 const QMap<QString, QgsMapLayer *> layersInProject = mLayerStore->mapLayers();
2932 layersIds.reserve( layersInProject.count() );
2933 layers.reserve( layersInProject.count() );
2934 for ( auto it = layersInProject.constBegin(); it != layersInProject.constEnd(); ++it )
2935 {
2936 layersIds << it.value()->id();
2937 layers << QVariant::fromValue<QgsWeakMapLayerPointer>( QgsWeakMapLayerPointer( it.value() ) );
2938 }
2939 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "layer_ids" ), layersIds, true ) );
2940 mProjectScope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "layers" ), layers, true ) );
2941
2942 mProjectScope->addFunction( QStringLiteral( "project_color" ), new GetNamedProjectColor( this ) );
2943 mProjectScope->addFunction( QStringLiteral( "project_color_object" ), new GetNamedProjectColorObject( this ) );
2944
2946}
2947
2948void QgsProject::onMapLayersAdded( const QList<QgsMapLayer *> &layers )
2949{
2951
2952 const QMap<QString, QgsMapLayer *> existingMaps = mapLayers();
2953
2954 const auto constLayers = layers;
2955 for ( QgsMapLayer *layer : constLayers )
2956 {
2957 if ( ! layer->isValid() )
2958 return;
2959
2960 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer ) )
2961 {
2962 vlayer->setReadExtentFromXml( mFlags & Qgis::ProjectFlag::TrustStoredLayerStatistics );
2963 if ( vlayer->dataProvider() )
2964 vlayer->dataProvider()->setProviderProperty( QgsVectorDataProvider::EvaluateDefaultValues,
2966 }
2967
2968 connect( layer, &QgsMapLayer::configChanged, this, [this] { setDirty(); } );
2969
2970 // check if we have to update connections for layers with dependencies
2971 for ( QMap<QString, QgsMapLayer *>::const_iterator it = existingMaps.cbegin(); it != existingMaps.cend(); ++it )
2972 {
2973 const QSet<QgsMapLayerDependency> deps = it.value()->dependencies();
2974 if ( deps.contains( layer->id() ) )
2975 {
2976 // reconnect to change signals
2977 it.value()->setDependencies( deps );
2978 }
2979 }
2980 }
2981
2982 updateTransactionGroups();
2983
2984 if ( !mBlockSnappingUpdates && mSnappingConfig.addLayers( layers ) )
2985 emit snappingConfigChanged( mSnappingConfig );
2986}
2987
2988void QgsProject::onMapLayersRemoved( const QList<QgsMapLayer *> &layers )
2989{
2991
2992 if ( !mBlockSnappingUpdates && mSnappingConfig.removeLayers( layers ) )
2993 emit snappingConfigChanged( mSnappingConfig );
2994
2995 for ( QgsMapLayer *layer : layers )
2996 {
2997 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
2998 if ( ! vlayer )
2999 continue;
3000
3001 mEditBufferGroup.removeLayer( vlayer );
3002 }
3003}
3004
3005void QgsProject::cleanTransactionGroups( bool force )
3006{
3008
3009 bool changed = false;
3010 for ( QMap< QPair< QString, QString>, QgsTransactionGroup *>::Iterator tg = mTransactionGroups.begin(); tg != mTransactionGroups.end(); )
3011 {
3012 if ( tg.value()->isEmpty() || force )
3013 {
3014 delete tg.value();
3015 tg = mTransactionGroups.erase( tg );
3016 changed = true;
3017 }
3018 else
3019 {
3020 ++tg;
3021 }
3022 }
3023 if ( changed )
3025}
3026
3027void QgsProject::updateTransactionGroups()
3028{
3030
3031 mEditBufferGroup.clear();
3032
3033 switch ( mTransactionMode )
3034 {
3036 {
3037 cleanTransactionGroups( true );
3038 return;
3039 }
3040 break;
3042 cleanTransactionGroups( true );
3043 break;
3045 cleanTransactionGroups( false );
3046 break;
3047 }
3048
3049 bool tgChanged = false;
3050 const auto constLayers = mapLayers().values();
3051 for ( QgsMapLayer *layer : constLayers )
3052 {
3053 if ( ! layer->isValid() )
3054 continue;
3055
3056 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
3057 if ( ! vlayer )
3058 continue;
3059
3060 switch ( mTransactionMode )
3061 {
3063 Q_ASSERT( false );
3064 break;
3066 {
3068 {
3069 const QString connString = QgsTransaction::connectionString( vlayer->source() );
3070 const QString key = vlayer->providerType();
3071
3072 QgsTransactionGroup *tg = mTransactionGroups.value( qMakePair( key, connString ) );
3073
3074 if ( !tg )
3075 {
3076 tg = new QgsTransactionGroup();
3077 mTransactionGroups.insert( qMakePair( key, connString ), tg );
3078 tgChanged = true;
3079 }
3080 tg->addLayer( vlayer );
3081 }
3082 }
3083 break;
3085 {
3086 if ( vlayer->supportsEditing() )
3087 mEditBufferGroup.addLayer( vlayer );
3088 }
3089 break;
3090 }
3091 }
3092
3093 if ( tgChanged )
3095}
3096
3097bool QgsProject::readLayer( const QDomNode &layerNode )
3098{
3100
3101 QgsReadWriteContext context;
3102 context.setPathResolver( pathResolver() );
3103 context.setProjectTranslator( this );
3104 context.setTransformContext( transformContext() );
3105 QList<QDomNode> brokenNodes;
3106 if ( addLayer( layerNode.toElement(), brokenNodes, context ) )
3107 {
3108 // have to try to update joins for all layers now - a previously added layer may be dependent on this newly
3109 // added layer for joins
3110 const QVector<QgsVectorLayer *> vectorLayers = layers<QgsVectorLayer *>();
3111 for ( QgsVectorLayer *layer : vectorLayers )
3112 {
3113 // TODO: should be only done later - and with all layers (other layers may have referenced this layer)
3114 layer->resolveReferences( this );
3115
3116 if ( layer->isValid() && layer->customProperty( QStringLiteral( "_layer_was_editable" ) ).toBool() )
3117 {
3118 layer->startEditing();
3119 layer->removeCustomProperty( QStringLiteral( "_layer_was_editable" ) );
3120 }
3121 }
3122 return true;
3123 }
3124 return false;
3125}
3126
3127bool QgsProject::write( const QString &filename )
3128{
3130
3131 mFile.setFileName( filename );
3132 mCachedHomePath.clear();
3133 return write();
3134}
3135
3137{
3139
3140 mProjectScope.reset();
3141 if ( QgsProjectStorage *storage = projectStorage() )
3142 {
3143 QgsReadWriteContext context;
3144 // for projects stored in a custom storage, we have to check for the support
3145 // of relative paths since the storage most likely will not be in a file system
3146 const QString storageFilePath { storage->filePath( mFile.fileName() ) };
3147 if ( storageFilePath.isEmpty() )
3148 {
3150 }
3151 context.setPathResolver( pathResolver() );
3152
3153 const QString tempPath = QStandardPaths::standardLocations( QStandardPaths::TempLocation ).at( 0 );
3154 const QString tmpZipFilename( tempPath + QDir::separator() + QUuid::createUuid().toString() );
3155
3156 if ( !zip( tmpZipFilename ) )
3157 return false; // zip() already calls setError() when returning false
3158
3159 QFile tmpZipFile( tmpZipFilename );
3160 if ( !tmpZipFile.open( QIODevice::ReadOnly ) )
3161 {
3162 setError( tr( "Unable to read file %1" ).arg( tmpZipFilename ) );
3163 return false;
3164 }
3165
3167 if ( !storage->writeProject( mFile.fileName(), &tmpZipFile, context ) )
3168 {
3169 QString err = tr( "Unable to save project to storage %1" ).arg( mFile.fileName() );
3170 QList<QgsReadWriteContext::ReadWriteMessage> messages = context.takeMessages();
3171 if ( !messages.isEmpty() )
3172 err += QStringLiteral( "\n\n" ) + messages.last().message();
3173 setError( err );
3174 return false;
3175 }
3176
3177 tmpZipFile.close();
3178 QFile::remove( tmpZipFilename );
3179
3180 return true;
3181 }
3182
3183 if ( QgsZipUtils::isZipFile( mFile.fileName() ) )
3184 {
3185 return zip( mFile.fileName() );
3186 }
3187 else
3188 {
3189 // write project file even if the auxiliary storage is not correctly
3190 // saved
3191 const bool asOk = saveAuxiliaryStorage();
3192 const bool writeOk = writeProjectFile( mFile.fileName() );
3193 bool attachmentsOk = true;
3194 if ( !mArchive->files().isEmpty() )
3195 {
3196 const QFileInfo finfo( mFile.fileName() );
3197 const QString attachmentsZip = finfo.absoluteDir().absoluteFilePath( QStringLiteral( "%1_attachments.zip" ).arg( finfo.completeBaseName() ) );
3198 attachmentsOk = mArchive->zip( attachmentsZip );
3199 }
3200
3201 // errors raised during writing project file are more important
3202 if ( ( !asOk || !attachmentsOk ) && writeOk )
3203 {
3204 QStringList errorMessage;
3205 if ( !asOk )
3206 {
3207 const QString err = mAuxiliaryStorage->errorString();
3208 errorMessage.append( tr( "Unable to save auxiliary storage ('%1')" ).arg( err ) );
3209 }
3210 if ( !attachmentsOk )
3211 {
3212 errorMessage.append( tr( "Unable to save attachments archive" ) );
3213 }
3214 setError( errorMessage.join( '\n' ) );
3215 }
3216
3217 return asOk && writeOk && attachmentsOk;
3218 }
3219}
3220
3221bool QgsProject::writeProjectFile( const QString &filename )
3222{
3224
3225 QFile projectFile( filename );
3226 clearError();
3227
3228 // if we have problems creating or otherwise writing to the project file,
3229 // let's find out up front before we go through all the hand-waving
3230 // necessary to create all the Dom objects
3231 const QFileInfo myFileInfo( projectFile );
3232 if ( myFileInfo.exists() && !myFileInfo.isWritable() )
3233 {
3234 setError( tr( "%1 is not writable. Please adjust permissions (if possible) and try again." )
3235 .arg( projectFile.fileName() ) );
3236 return false;
3237 }
3238
3239 QgsReadWriteContext context;
3240 context.setPathResolver( pathResolver() );
3242
3243 QDomImplementation::setInvalidDataPolicy( QDomImplementation::DropInvalidChars );
3244
3245 const QDomDocumentType documentType =
3246 QDomImplementation().createDocumentType( QStringLiteral( "qgis" ), QStringLiteral( "http://mrcc.com/qgis.dtd" ),
3247 QStringLiteral( "SYSTEM" ) );
3248 auto doc = std::make_unique<QDomDocument>( documentType );
3249
3250 QDomElement qgisNode = doc->createElement( QStringLiteral( "qgis" ) );
3251 qgisNode.setAttribute( QStringLiteral( "projectname" ), title() );
3252 qgisNode.setAttribute( QStringLiteral( "version" ), Qgis::version() );
3253
3254 if ( !mSettings.value( QStringLiteral( "projects/anonymize_saved_projects" ), false, QgsSettings::Core ).toBool() )
3255 {
3256 const QString newSaveUser = QgsApplication::userLoginName();
3257 const QString newSaveUserFull = QgsApplication::userFullName();
3258 qgisNode.setAttribute( QStringLiteral( "saveUser" ), newSaveUser );
3259 qgisNode.setAttribute( QStringLiteral( "saveUserFull" ), newSaveUserFull );
3260 mSaveUser = newSaveUser;
3261 mSaveUserFull = newSaveUserFull;
3262 mSaveDateTime = QDateTime::currentDateTime();
3263 qgisNode.setAttribute( QStringLiteral( "saveDateTime" ), mSaveDateTime.toString( Qt::ISODate ) );
3264 }
3265 else
3266 {
3267 mSaveUser.clear();
3268 mSaveUserFull.clear();
3269 mSaveDateTime = QDateTime();
3270 }
3271 doc->appendChild( qgisNode );
3272 mSaveVersion = QgsProjectVersion( Qgis::version() );
3273
3274 QDomElement homePathNode = doc->createElement( QStringLiteral( "homePath" ) );
3275 homePathNode.setAttribute( QStringLiteral( "path" ), mHomePath );
3276 qgisNode.appendChild( homePathNode );
3277
3278 // title
3279 QDomElement titleNode = doc->createElement( QStringLiteral( "title" ) );
3280 qgisNode.appendChild( titleNode );
3281
3282 QDomElement transactionNode = doc->createElement( QStringLiteral( "transaction" ) );
3283 transactionNode.setAttribute( QStringLiteral( "mode" ), qgsEnumValueToKey( mTransactionMode ) );
3284 qgisNode.appendChild( transactionNode );
3285
3286 QDomElement flagsNode = doc->createElement( QStringLiteral( "projectFlags" ) );
3287 flagsNode.setAttribute( QStringLiteral( "set" ), qgsFlagValueToKeys( mFlags ) );
3288 qgisNode.appendChild( flagsNode );
3289
3290 const QDomText titleText = doc->createTextNode( title() ); // XXX why have title TWICE?
3291 titleNode.appendChild( titleText );
3292
3293 // write project CRS
3294 {
3295 QDomElement srsNode = doc->createElement( QStringLiteral( "projectCrs" ) );
3296 mCrs.writeXml( srsNode, *doc );
3297 qgisNode.appendChild( srsNode );
3298 }
3299 {
3300 QDomElement verticalSrsNode = doc->createElement( QStringLiteral( "verticalCrs" ) );
3301 mVerticalCrs.writeXml( verticalSrsNode, *doc );
3302 qgisNode.appendChild( verticalSrsNode );
3303 }
3304
3305 QDomElement elevationShadingNode = doc->createElement( QStringLiteral( "elevation-shading-renderer" ) );
3306 mElevationShadingRenderer.writeXml( elevationShadingNode, context );
3307 qgisNode.appendChild( elevationShadingNode );
3308
3309 // write layer tree - make sure it is without embedded subgroups
3310 QgsLayerTreeNode *clonedRoot = mRootGroup->clone();
3312 QgsLayerTreeUtils::updateEmbeddedGroupsProjectPath( QgsLayerTree::toGroup( clonedRoot ), this ); // convert absolute paths to relative paths if required
3313
3314 clonedRoot->writeXml( qgisNode, context );
3315 delete clonedRoot;
3316
3317 mSnappingConfig.writeProject( *doc );
3318 writeEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/AvoidIntersectionsMode" ), static_cast<int>( mAvoidIntersectionsMode ) );
3319
3320 // let map canvas and legend write their information
3321 emit writeProject( *doc );
3322
3323 // within top level node save list of layers
3324 const QMap<QString, QgsMapLayer *> layers = mapLayers();
3325
3326 QDomElement annotationLayerNode = doc->createElement( QStringLiteral( "main-annotation-layer" ) );
3327 mMainAnnotationLayer->writeLayerXml( annotationLayerNode, *doc, context );
3328 qgisNode.appendChild( annotationLayerNode );
3329
3330 // Iterate over layers in zOrder
3331 // Call writeXml() on each
3332 QDomElement projectLayersNode = doc->createElement( QStringLiteral( "projectlayers" ) );
3333
3334 QMap<QString, QgsMapLayer *>::ConstIterator li = layers.constBegin();
3335 while ( li != layers.end() )
3336 {
3337 QgsMapLayer *ml = li.value();
3338
3339 if ( ml )
3340 {
3341 const QHash< QString, QPair< QString, bool> >::const_iterator emIt = mEmbeddedLayers.constFind( ml->id() );
3342 if ( emIt == mEmbeddedLayers.constEnd() )
3343 {
3344 QDomElement maplayerElem;
3345 // If layer is not valid, prefer to restore saved properties from invalidLayerProperties. But if that's
3346 // not available, just write what we DO have
3347 if ( ml->isValid() || ml->originalXmlProperties().isEmpty() )
3348 {
3349 // general layer metadata
3350 maplayerElem = doc->createElement( QStringLiteral( "maplayer" ) );
3351 ml->writeLayerXml( maplayerElem, *doc, context );
3352
3354 maplayerElem.setAttribute( QStringLiteral( "editable" ), QStringLiteral( "1" ) );
3355 }
3356 else if ( ! ml->originalXmlProperties().isEmpty() )
3357 {
3358 QDomDocument document;
3359 if ( document.setContent( ml->originalXmlProperties() ) )
3360 {
3361 maplayerElem = document.firstChildElement();
3362 }
3363 else
3364 {
3365 QgsDebugError( QStringLiteral( "Could not restore layer properties for layer %1" ).arg( ml->id() ) );
3366 }
3367 }
3368
3369 emit writeMapLayer( ml, maplayerElem, *doc );
3370
3371 projectLayersNode.appendChild( maplayerElem );
3372 }
3373 else
3374 {
3375 // layer defined in an external project file
3376 // only save embedded layer if not managed by a legend group
3377 if ( emIt.value().second )
3378 {
3379 QDomElement mapLayerElem = doc->createElement( QStringLiteral( "maplayer" ) );
3380 mapLayerElem.setAttribute( QStringLiteral( "embedded" ), 1 );
3381 mapLayerElem.setAttribute( QStringLiteral( "project" ), writePath( emIt.value().first ) );
3382 mapLayerElem.setAttribute( QStringLiteral( "id" ), ml->id() );
3383 projectLayersNode.appendChild( mapLayerElem );
3384 }
3385 }
3386 }
3387 li++;
3388 }
3389
3390 qgisNode.appendChild( projectLayersNode );
3391
3392 QDomElement layerOrderNode = doc->createElement( QStringLiteral( "layerorder" ) );
3393 const auto constCustomLayerOrder = mRootGroup->customLayerOrder();
3394 for ( QgsMapLayer *layer : constCustomLayerOrder )
3395 {
3396 QDomElement mapLayerElem = doc->createElement( QStringLiteral( "layer" ) );
3397 mapLayerElem.setAttribute( QStringLiteral( "id" ), layer->id() );
3398 layerOrderNode.appendChild( mapLayerElem );
3399 }
3400 qgisNode.appendChild( layerOrderNode );
3401
3402 mLabelingEngineSettings->writeSettingsToProject( this );
3403 {
3404 QDomElement labelEngineSettingsElement = doc->createElement( QStringLiteral( "labelEngineSettings" ) );
3405 mLabelingEngineSettings->writeXml( *doc, labelEngineSettingsElement, context );
3406 qgisNode.appendChild( labelEngineSettingsElement );
3407 }
3408
3409 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorRedPart" ), mBackgroundColor.red() );
3410 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorGreenPart" ), mBackgroundColor.green() );
3411 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/CanvasColorBluePart" ), mBackgroundColor.blue() );
3412
3413 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorRedPart" ), mSelectionColor.red() );
3414 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorGreenPart" ), mSelectionColor.green() );
3415 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorBluePart" ), mSelectionColor.blue() );
3416 writeEntry( QStringLiteral( "Gui" ), QStringLiteral( "/SelectionColorAlphaPart" ), mSelectionColor.alpha() );
3417
3418 writeEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/DistanceUnits" ), QgsUnitTypes::encodeUnit( mDistanceUnits ) );
3419 writeEntry( QStringLiteral( "Measurement" ), QStringLiteral( "/AreaUnits" ), QgsUnitTypes::encodeUnit( mAreaUnits ) );
3420
3421 // now add the optional extra properties
3422#if 0
3423 dump_( mProperties );
3424#endif
3425
3426 QgsDebugMsgLevel( QStringLiteral( "there are %1 property scopes" ).arg( static_cast<int>( mProperties.count() ) ), 2 );
3427
3428 if ( !mProperties.isEmpty() ) // only worry about properties if we
3429 // actually have any properties
3430 {
3431 mProperties.writeXml( QStringLiteral( "properties" ), qgisNode, *doc );
3432 }
3433
3434 QDomElement ddElem = doc->createElement( QStringLiteral( "dataDefinedServerProperties" ) );
3435 mDataDefinedServerProperties.writeXml( ddElem, dataDefinedServerPropertyDefinitions() );
3436 qgisNode.appendChild( ddElem );
3437
3438 mMapThemeCollection->writeXml( *doc );
3439
3440 mTransformContext.writeXml( qgisNode, context );
3441
3442 QDomElement metadataElem = doc->createElement( QStringLiteral( "projectMetadata" ) );
3443 mMetadata.writeMetadataXml( metadataElem, *doc );
3444 qgisNode.appendChild( metadataElem );
3445
3446 {
3447 const QDomElement annotationsElem = mAnnotationManager->writeXml( *doc, context );
3448 qgisNode.appendChild( annotationsElem );
3449 }
3450
3451 {
3452 const QDomElement layoutElem = mLayoutManager->writeXml( *doc );
3453 qgisNode.appendChild( layoutElem );
3454 }
3455
3456 {
3457 const QDomElement views3DElem = m3DViewsManager->writeXml( *doc );
3458 qgisNode.appendChild( views3DElem );
3459 }
3460
3461 {
3462 const QDomElement bookmarkElem = mBookmarkManager->writeXml( *doc );
3463 qgisNode.appendChild( bookmarkElem );
3464 }
3465
3466 {
3467 const QDomElement sensorElem = mSensorManager->writeXml( *doc );
3468 qgisNode.appendChild( sensorElem );
3469 }
3470
3471 {
3472 const QDomElement viewSettingsElem = mViewSettings->writeXml( *doc, context );
3473 qgisNode.appendChild( viewSettingsElem );
3474 }
3475
3476 {
3477 const QDomElement styleSettingsElem = mStyleSettings->writeXml( *doc, context );
3478 qgisNode.appendChild( styleSettingsElem );
3479 }
3480
3481 {
3482 const QDomElement timeSettingsElement = mTimeSettings->writeXml( *doc, context );
3483 qgisNode.appendChild( timeSettingsElement );
3484 }
3485
3486 {
3487 const QDomElement elevationPropertiesElement = mElevationProperties->writeXml( *doc, context );
3488 qgisNode.appendChild( elevationPropertiesElement );
3489 }
3490
3491 {
3492 const QDomElement displaySettingsElem = mDisplaySettings->writeXml( *doc, context );
3493 qgisNode.appendChild( displaySettingsElem );
3494 }
3495
3496 {
3497 const QDomElement gpsSettingsElem = mGpsSettings->writeXml( *doc, context );
3498 qgisNode.appendChild( gpsSettingsElem );
3499 }
3500
3501 // now wrap it up and ship it to the project file
3502 doc->normalize(); // XXX I'm not entirely sure what this does
3503
3504 // Create backup file
3505 if ( QFile::exists( fileName() ) )
3506 {
3507 QFile backupFile( QStringLiteral( "%1~" ).arg( filename ) );
3508 bool ok = true;
3509 ok &= backupFile.open( QIODevice::WriteOnly | QIODevice::Truncate );
3510 ok &= projectFile.open( QIODevice::ReadOnly );
3511
3512 QByteArray ba;
3513 while ( ok && !projectFile.atEnd() )
3514 {
3515 ba = projectFile.read( 10240 );
3516 ok &= backupFile.write( ba ) == ba.size();
3517 }
3518
3519 projectFile.close();
3520 backupFile.close();
3521
3522 if ( !ok )
3523 {
3524 setError( tr( "Unable to create backup file %1" ).arg( backupFile.fileName() ) );
3525 return false;
3526 }
3527
3528 const QFileInfo fi( fileName() );
3529 struct utimbuf tb = { static_cast<time_t>( fi.lastRead().toSecsSinceEpoch() ), static_cast<time_t>( fi.lastModified().toSecsSinceEpoch() ) };
3530 utime( backupFile.fileName().toUtf8().constData(), &tb );
3531 }
3532
3533 if ( !projectFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
3534 {
3535 projectFile.close(); // even though we got an error, let's make
3536 // sure it's closed anyway
3537
3538 setError( tr( "Unable to save to file %1" ).arg( projectFile.fileName() ) );
3539 return false;
3540 }
3541
3542 QTemporaryFile tempFile;
3543 bool ok = tempFile.open();
3544 if ( ok )
3545 {
3546 QTextStream projectFileStream( &tempFile );
3547 doc->save( projectFileStream, 2 ); // save as utf-8
3548 ok &= projectFileStream.pos() > -1;
3549
3550 ok &= tempFile.seek( 0 );
3551
3552 QByteArray ba;
3553 while ( ok && !tempFile.atEnd() )
3554 {
3555 ba = tempFile.read( 10240 );
3556 ok &= projectFile.write( ba ) == ba.size();
3557 }
3558
3559 ok &= projectFile.error() == QFile::NoError;
3560
3561 projectFile.close();
3562 }
3563
3564 tempFile.close();
3565
3566 if ( !ok )
3567 {
3568 setError( tr( "Unable to save to file %1. Your project "
3569 "may be corrupted on disk. Try clearing some space on the volume and "
3570 "check file permissions before pressing save again." )
3571 .arg( projectFile.fileName() ) );
3572 return false;
3573 }
3574
3575 setDirty( false ); // reset to pristine state
3576
3577 emit projectSaved();
3578 return true;
3579}
3580
3581bool QgsProject::writeEntry( const QString &scope, QString const &key, bool value )
3582{
3584
3585 bool propertiesModified;
3586 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3587
3588 if ( propertiesModified )
3589 setDirty( true );
3590
3591 return success;
3592}
3593
3594bool QgsProject::writeEntry( const QString &scope, const QString &key, double value )
3595{
3597
3598 bool propertiesModified;
3599 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3600
3601 if ( propertiesModified )
3602 setDirty( true );
3603
3604 return success;
3605}
3606
3607bool QgsProject::writeEntry( const QString &scope, QString const &key, int value )
3608{
3610
3611 bool propertiesModified;
3612 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3613
3614 if ( propertiesModified )
3615 setDirty( true );
3616
3617 return success;
3618}
3619
3620bool QgsProject::writeEntry( const QString &scope, const QString &key, const QString &value )
3621{
3623
3624 bool propertiesModified;
3625 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3626
3627 if ( propertiesModified )
3628 setDirty( true );
3629
3630 return success;
3631}
3632
3633bool QgsProject::writeEntry( const QString &scope, const QString &key, const QStringList &value )
3634{
3636
3637 bool propertiesModified;
3638 const bool success = addKey_( scope, key, &mProperties, value, propertiesModified );
3639
3640 if ( propertiesModified )
3641 setDirty( true );
3642
3643 return success;
3644}
3645
3646QStringList QgsProject::readListEntry( const QString &scope,
3647 const QString &key,
3648 const QStringList &def,
3649 bool *ok ) const
3650{
3651 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
3653
3654 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3655
3656 QVariant value;
3657
3658 if ( property )
3659 {
3660 value = property->value();
3661
3662 const bool valid = QMetaType::Type::QStringList == value.userType();
3663 if ( ok )
3664 *ok = valid;
3665
3666 if ( valid )
3667 {
3668 return value.toStringList();
3669 }
3670 }
3671 else if ( ok )
3672 *ok = false;
3673
3674
3675 return def;
3676}
3677
3678QString QgsProject::readEntry( const QString &scope,
3679 const QString &key,
3680 const QString &def,
3681 bool *ok ) const
3682{
3684
3685 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3686
3687 QVariant value;
3688
3689 if ( property )
3690 {
3691 value = property->value();
3692
3693 const bool valid = value.canConvert( QMetaType::Type::QString );
3694 if ( ok )
3695 *ok = valid;
3696
3697 if ( valid )
3698 return value.toString();
3699 }
3700 else if ( ok )
3701 *ok = false;
3702
3703 return def;
3704}
3705
3706int QgsProject::readNumEntry( const QString &scope, const QString &key, int def,
3707 bool *ok ) const
3708{
3710
3711 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3712
3713 QVariant value;
3714
3715 if ( property )
3716 {
3717 value = property->value();
3718 }
3719
3720 const bool valid = value.canConvert( QMetaType::Type::Int );
3721
3722 if ( ok )
3723 {
3724 *ok = valid;
3725 }
3726
3727 if ( valid )
3728 {
3729 return value.toInt();
3730 }
3731
3732 return def;
3733}
3734
3735double QgsProject::readDoubleEntry( const QString &scope, const QString &key,
3736 double def,
3737 bool *ok ) const
3738{
3740
3741 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3742 if ( property )
3743 {
3744 const QVariant value = property->value();
3745
3746 const bool valid = value.canConvert( QMetaType::Type::Double );
3747 if ( ok )
3748 *ok = valid;
3749
3750 if ( valid )
3751 return value.toDouble();
3752 }
3753 else if ( ok )
3754 *ok = false;
3755
3756 return def;
3757}
3758
3759bool QgsProject::readBoolEntry( const QString &scope, const QString &key, bool def,
3760 bool *ok ) const
3761{
3763
3764 QgsProjectProperty *property = findKey_( scope, key, mProperties );
3765
3766 if ( property )
3767 {
3768 const QVariant value = property->value();
3769
3770 const bool valid = value.canConvert( QMetaType::Type::Bool );
3771 if ( ok )
3772 *ok = valid;
3773
3774 if ( valid )
3775 return value.toBool();
3776 }
3777 else if ( ok )
3778 *ok = false;
3779
3780 return def;
3781}
3782
3783bool QgsProject::removeEntry( const QString &scope, const QString &key )
3784{
3786
3787 if ( findKey_( scope, key, mProperties ) )
3788 {
3789 removeKey_( scope, key, mProperties );
3790 setDirty( true );
3791 }
3792
3793 return !findKey_( scope, key, mProperties );
3794}
3795
3796QStringList QgsProject::entryList( const QString &scope, const QString &key ) const
3797{
3799
3800 QgsProjectProperty *foundProperty = findKey_( scope, key, mProperties );
3801
3802 QStringList entries;
3803
3804 if ( foundProperty )
3805 {
3806 QgsProjectPropertyKey *propertyKey = dynamic_cast<QgsProjectPropertyKey *>( foundProperty );
3807
3808 if ( propertyKey )
3809 { propertyKey->entryList( entries ); }
3810 }
3811
3812 return entries;
3813}
3814
3815QStringList QgsProject::subkeyList( const QString &scope, const QString &key ) const
3816{
3818
3819 QgsProjectProperty *foundProperty = findKey_( scope, key, mProperties );
3820
3821 QStringList entries;
3822
3823 if ( foundProperty )
3824 {
3825 QgsProjectPropertyKey *propertyKey = dynamic_cast<QgsProjectPropertyKey *>( foundProperty );
3826
3827 if ( propertyKey )
3828 { propertyKey->subkeyList( entries ); }
3829 }
3830
3831 return entries;
3832}
3833
3835{
3837
3838 dump_( mProperties );
3839}
3840
3842{
3844
3845 QString filePath;
3846 switch ( filePathStorage() )
3847 {
3849 break;
3850
3852 {
3853 // for projects stored in a custom storage, we need to ask to the
3854 // storage for the path, if the storage returns an empty path
3855 // relative paths are not supported
3856 if ( QgsProjectStorage *storage = projectStorage() )
3857 {
3858 filePath = storage->filePath( mFile.fileName() );
3859 }
3860 else
3861 {
3862 filePath = fileName();
3863 }
3864 break;
3865 }
3866 }
3867
3868 return QgsPathResolver( filePath, mArchive->dir() );
3869}
3870
3871QString QgsProject::readPath( const QString &src ) const
3872{
3874
3875 return pathResolver().readPath( src );
3876}
3877
3878QString QgsProject::writePath( const QString &src ) const
3879{
3881
3882 return pathResolver().writePath( src );
3883}
3884
3885void QgsProject::setError( const QString &errorMessage )
3886{
3888
3889 mErrorMessage = errorMessage;
3890}
3891
3892QString QgsProject::error() const
3893{
3895
3896 return mErrorMessage;
3897}
3898
3899void QgsProject::clearError()
3900{
3902
3903 setError( QString() );
3904}
3905
3907{
3909
3910 delete mBadLayerHandler;
3911 mBadLayerHandler = handler;
3912}
3913
3914QString QgsProject::layerIsEmbedded( const QString &id ) const
3915{
3917
3918 const QHash< QString, QPair< QString, bool > >::const_iterator it = mEmbeddedLayers.find( id );
3919 if ( it == mEmbeddedLayers.constEnd() )
3920 {
3921 return QString();
3922 }
3923 return it.value().first;
3924}
3925
3926bool QgsProject::createEmbeddedLayer( const QString &layerId, const QString &projectFilePath, QList<QDomNode> &brokenNodes,
3927 bool saveFlag, Qgis::ProjectReadFlags flags )
3928{
3930
3932
3933 static QString sPrevProjectFilePath;
3934 static QDateTime sPrevProjectFileTimestamp;
3935 static QDomDocument sProjectDocument;
3936
3937 QString qgsProjectFile = projectFilePath;
3938 QgsProjectArchive archive;
3939 if ( projectFilePath.endsWith( QLatin1String( ".qgz" ), Qt::CaseInsensitive ) )
3940 {
3941 archive.unzip( projectFilePath );
3942 qgsProjectFile = archive.projectFile();
3943 }
3944
3945 const QDateTime projectFileTimestamp = QFileInfo( projectFilePath ).lastModified();
3946
3947 if ( projectFilePath != sPrevProjectFilePath || projectFileTimestamp != sPrevProjectFileTimestamp )
3948 {
3949 sPrevProjectFilePath.clear();
3950
3951 QFile projectFile( qgsProjectFile );
3952 if ( !projectFile.open( QIODevice::ReadOnly ) )
3953 {
3954 return false;
3955 }
3956
3957 if ( !sProjectDocument.setContent( &projectFile ) )
3958 {
3959 return false;
3960 }
3961
3962 sPrevProjectFilePath = projectFilePath;
3963 sPrevProjectFileTimestamp = projectFileTimestamp;
3964 }
3965
3966 // does project store paths absolute or relative?
3967 bool useAbsolutePaths = true;
3968
3969 const QDomElement propertiesElem = sProjectDocument.documentElement().firstChildElement( QStringLiteral( "properties" ) );
3970 if ( !propertiesElem.isNull() )
3971 {
3972 QDomElement e = propertiesElem.firstChildElement( QStringLiteral( "Paths" ) );
3973 if ( e.isNull() )
3974 {
3975 e = propertiesElem.firstChildElement( QStringLiteral( "properties" ) );
3976 while ( !e.isNull() && e.attribute( QStringLiteral( "name" ) ) != QStringLiteral( "Paths" ) )
3977 e = e.nextSiblingElement( QStringLiteral( "properties" ) );
3978
3979 e = e.firstChildElement( QStringLiteral( "properties" ) );
3980 while ( !e.isNull() && e.attribute( QStringLiteral( "name" ) ) != QStringLiteral( "Absolute" ) )
3981 e = e.nextSiblingElement( QStringLiteral( "properties" ) );
3982 }
3983 else
3984 {
3985 e = e.firstChildElement( QStringLiteral( "Absolute" ) );
3986 }
3987
3988 if ( !e.isNull() )
3989 {
3990 useAbsolutePaths = e.text().compare( QLatin1String( "true" ), Qt::CaseInsensitive ) == 0;
3991 }
3992 }
3993
3994 QgsReadWriteContext embeddedContext;
3995 if ( !useAbsolutePaths )
3996 embeddedContext.setPathResolver( QgsPathResolver( projectFilePath ) );
3997 embeddedContext.setProjectTranslator( this );
3998 embeddedContext.setTransformContext( transformContext() );
3999
4000 const QDomElement projectLayersElem = sProjectDocument.documentElement().firstChildElement( QStringLiteral( "projectlayers" ) );
4001 if ( projectLayersElem.isNull() )
4002 {
4003 return false;
4004 }
4005
4006 QDomElement mapLayerElem = projectLayersElem.firstChildElement( QStringLiteral( "maplayer" ) );
4007 while ( ! mapLayerElem.isNull() )
4008 {
4009 // get layer id
4010 const QString id = mapLayerElem.firstChildElement( QStringLiteral( "id" ) ).text();
4011 if ( id == layerId )
4012 {
4013 // layer can be embedded only once
4014 if ( mapLayerElem.attribute( QStringLiteral( "embedded" ) ) == QLatin1String( "1" ) )
4015 {
4016 return false;
4017 }
4018
4019 mEmbeddedLayers.insert( layerId, qMakePair( projectFilePath, saveFlag ) );
4020
4021 if ( addLayer( mapLayerElem, brokenNodes, embeddedContext, flags ) )
4022 {
4023 return true;
4024 }
4025 else
4026 {
4027 mEmbeddedLayers.remove( layerId );
4028 return false;
4029 }
4030 }
4031 mapLayerElem = mapLayerElem.nextSiblingElement( QStringLiteral( "maplayer" ) );
4032 }
4033
4034 return false;
4035}
4036
4037QgsLayerTreeGroup *QgsProject::createEmbeddedGroup( const QString &groupName, const QString &projectFilePath, const QStringList &invisibleLayers, Qgis::ProjectReadFlags flags )
4038{
4040
4041 QString qgsProjectFile = projectFilePath;
4042 QgsProjectArchive archive;
4043 if ( projectFilePath.endsWith( QLatin1String( ".qgz" ), Qt::CaseInsensitive ) )
4044 {
4045 archive.unzip( projectFilePath );
4046 qgsProjectFile = archive.projectFile();
4047 }
4048
4049 // open project file, get layer ids in group, add the layers
4050 QFile projectFile( qgsProjectFile );
4051 if ( !projectFile.open( QIODevice::ReadOnly ) )
4052 {
4053 return nullptr;
4054 }
4055
4056 QDomDocument projectDocument;
4057 if ( !projectDocument.setContent( &projectFile ) )
4058 {
4059 return nullptr;
4060 }
4061
4062 QgsReadWriteContext context;
4063 context.setPathResolver( pathResolver() );
4064 context.setProjectTranslator( this );
4066
4068
4069 QDomElement layerTreeElem = projectDocument.documentElement().firstChildElement( QStringLiteral( "layer-tree-group" ) );
4070 if ( !layerTreeElem.isNull() )
4071 {
4072 root->readChildrenFromXml( layerTreeElem, context );
4073 }
4074 else
4075 {
4076 QgsLayerTreeUtils::readOldLegend( root, projectDocument.documentElement().firstChildElement( QStringLiteral( "legend" ) ) );
4077 }
4078
4079 QgsLayerTreeGroup *group = root->findGroup( groupName );
4080 if ( !group || group->customProperty( QStringLiteral( "embedded" ) ).toBool() )
4081 {
4082 // embedded groups cannot be embedded again
4083 delete root;
4084 return nullptr;
4085 }
4086
4087 // clone the group sub-tree (it is used already in a tree, we cannot just tear it off)
4088 QgsLayerTreeGroup *newGroup = QgsLayerTree::toGroup( group->clone() );
4089 delete root;
4090 root = nullptr;
4091
4092 newGroup->setCustomProperty( QStringLiteral( "embedded" ), 1 );
4093 newGroup->setCustomProperty( QStringLiteral( "embedded_project" ), projectFilePath );
4094
4095 // set "embedded" to all children + load embedded layers
4096 mLayerTreeRegistryBridge->setEnabled( false );
4097 initializeEmbeddedSubtree( projectFilePath, newGroup, flags );
4098 mLayerTreeRegistryBridge->setEnabled( true );
4099
4100 // consider the layers might be identify disabled in its project
4101 const auto constFindLayerIds = newGroup->findLayerIds();
4102 for ( const QString &layerId : constFindLayerIds )
4103 {
4104 QgsLayerTreeLayer *layer = newGroup->findLayer( layerId );
4105 if ( layer )
4106 {
4107 layer->resolveReferences( this );
4108 layer->setItemVisibilityChecked( !invisibleLayers.contains( layerId ) );
4109 }
4110 }
4111
4112 return newGroup;
4113}
4114
4115void QgsProject::initializeEmbeddedSubtree( const QString &projectFilePath, QgsLayerTreeGroup *group, Qgis::ProjectReadFlags flags )
4116{
4118
4119 const auto constChildren = group->children();
4120 for ( QgsLayerTreeNode *child : constChildren )
4121 {
4122 // all nodes in the subtree will have "embedded" custom property set
4123 child->setCustomProperty( QStringLiteral( "embedded" ), 1 );
4124
4125 if ( QgsLayerTree::isGroup( child ) )
4126 {
4127 initializeEmbeddedSubtree( projectFilePath, QgsLayerTree::toGroup( child ), flags );
4128 }
4129 else if ( QgsLayerTree::isLayer( child ) )
4130 {
4131 // load the layer into our project
4132 QList<QDomNode> brokenNodes;
4133 createEmbeddedLayer( QgsLayerTree::toLayer( child )->layerId(), projectFilePath, brokenNodes, false, flags );
4134 }
4135 }
4136}
4137
4144
4151
4153{
4155
4156 writeEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/TopologicalEditing" ), ( enabled ? 1 : 0 ) );
4158}
4159
4161{
4163
4164 return readNumEntry( QStringLiteral( "Digitizing" ), QStringLiteral( "/TopologicalEditing" ), 0 );
4165}
4166
4168{
4170
4171 if ( mDistanceUnits == unit )
4172 return;
4173
4174 mDistanceUnits = unit;
4175
4176 emit distanceUnitsChanged();
4177}
4178
4180{
4182
4183 if ( mAreaUnits == unit )
4184 return;
4185
4186 mAreaUnits = unit;
4187
4188 emit areaUnitsChanged();
4189}
4190
4192{
4193 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
4195
4196 if ( !mCachedHomePath.isEmpty() )
4197 return mCachedHomePath;
4198
4199 const QFileInfo pfi( fileName() );
4200
4201 if ( !mHomePath.isEmpty() )
4202 {
4203 const QFileInfo homeInfo( mHomePath );
4204 if ( !homeInfo.isRelative() )
4205 {
4206 mCachedHomePath = mHomePath;
4207 return mHomePath;
4208 }
4209 }
4210 else if ( !fileName().isEmpty() )
4211 {
4212
4213 // If it's not stored in the file system, try to get the path from the storage
4214 if ( QgsProjectStorage *storage = projectStorage() )
4215 {
4216 const QString storagePath { storage->filePath( fileName() ) };
4217 if ( ! storagePath.isEmpty() && QFileInfo::exists( storagePath ) )
4218 {
4219 mCachedHomePath = QFileInfo( storagePath ).path();
4220 return mCachedHomePath;
4221 }
4222 }
4223
4224 mCachedHomePath = pfi.path();
4225 return mCachedHomePath;
4226 }
4227
4228 if ( !pfi.exists() )
4229 {
4230 mCachedHomePath = mHomePath;
4231 return mHomePath;
4232 }
4233
4234 if ( !mHomePath.isEmpty() )
4235 {
4236 // path is relative to project file
4237 mCachedHomePath = QDir::cleanPath( pfi.path() + '/' + mHomePath );
4238 }
4239 else
4240 {
4241 mCachedHomePath = pfi.canonicalPath();
4242 }
4243 return mCachedHomePath;
4244}
4245
4247{
4249
4250 return mHomePath;
4251}
4252
4254{
4255 // because relation aggregate functions are not thread safe
4257
4258 return mRelationManager;
4259}
4260
4262{
4264
4265 return mLayoutManager.get();
4266}
4267
4269{
4271
4272 return mLayoutManager.get();
4273}
4274
4276{
4278
4279 return m3DViewsManager.get();
4280}
4281
4283{
4285
4286 return m3DViewsManager.get();
4287}
4288
4290{
4292
4293 return mBookmarkManager;
4294}
4295
4297{
4299
4300 return mBookmarkManager;
4301}
4302
4304{
4306
4307 return mSensorManager;
4308}
4309
4311{
4313
4314 return mSensorManager;
4315}
4316
4318{
4320
4321 return mViewSettings;
4322}
4323
4330
4332{
4334
4335 return mStyleSettings;
4336}
4337
4339{
4340 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
4342
4343 return mStyleSettings;
4344}
4345
4347{
4349
4350 return mTimeSettings;
4351}
4352
4359
4361{
4363
4364 return mElevationProperties;
4365}
4366
4373
4375{
4377
4378 return mDisplaySettings;
4379}
4380
4382{
4384
4385 return mDisplaySettings;
4386}
4387
4389{
4391
4392 return mGpsSettings;
4393}
4394
4401
4403{
4405
4406 return mRootGroup;
4407}
4408
4410{
4412
4413 return mMapThemeCollection.get();
4414}
4415
4417{
4419
4420 return mAnnotationManager.get();
4421}
4422
4424{
4426
4427 return mAnnotationManager.get();
4428}
4429
4430void QgsProject::setNonIdentifiableLayers( const QList<QgsMapLayer *> &layers )
4431{
4433
4434 const QMap<QString, QgsMapLayer *> &projectLayers = mapLayers();
4435 for ( QMap<QString, QgsMapLayer *>::const_iterator it = projectLayers.constBegin(); it != projectLayers.constEnd(); ++it )
4436 {
4437 if ( layers.contains( it.value() ) == !it.value()->flags().testFlag( QgsMapLayer::Identifiable ) )
4438 continue;
4439
4440 if ( layers.contains( it.value() ) )
4441 it.value()->setFlags( it.value()->flags() & ~QgsMapLayer::Identifiable );
4442 else
4443 it.value()->setFlags( it.value()->flags() | QgsMapLayer::Identifiable );
4444 }
4445
4449}
4450
4451void QgsProject::setNonIdentifiableLayers( const QStringList &layerIds )
4452{
4454
4455 QList<QgsMapLayer *> nonIdentifiableLayers;
4456 nonIdentifiableLayers.reserve( layerIds.count() );
4457 for ( const QString &layerId : layerIds )
4458 {
4459 QgsMapLayer *layer = mapLayer( layerId );
4460 if ( layer )
4461 nonIdentifiableLayers << layer;
4462 }
4466}
4467
4469{
4471
4472 QStringList nonIdentifiableLayers;
4473
4474 const QMap<QString, QgsMapLayer *> &layers = mapLayers();
4475 for ( QMap<QString, QgsMapLayer *>::const_iterator it = layers.constBegin(); it != layers.constEnd(); ++it )
4476 {
4477 if ( !it.value()->flags().testFlag( QgsMapLayer::Identifiable ) )
4478 {
4479 nonIdentifiableLayers.append( it.value()->id() );
4480 }
4481 }
4482 return nonIdentifiableLayers;
4483}
4484
4486{
4488
4489 return mTransactionMode == Qgis::TransactionMode::AutomaticGroups;
4490}
4491
4492void QgsProject::setAutoTransaction( bool autoTransaction )
4493{
4495
4496 if ( autoTransaction
4497 && mTransactionMode == Qgis::TransactionMode::AutomaticGroups )
4498 return;
4499
4500 if ( ! autoTransaction
4501 && mTransactionMode == Qgis::TransactionMode::Disabled )
4502 return;
4503
4504 if ( autoTransaction )
4506 else
4508
4509 updateTransactionGroups();
4510}
4511
4513{
4515
4516 return mTransactionMode;
4517}
4518
4520{
4522
4523 if ( transactionMode == mTransactionMode )
4524 return true;
4525
4526 // Check that all layer are not in edit mode
4527 const auto constLayers = mapLayers().values();
4528 for ( QgsMapLayer *layer : constLayers )
4529 {
4530 if ( layer->isEditable() )
4531 {
4532 QgsLogger::warning( tr( "Transaction mode can be changed only if all layers are not editable." ) );
4533 return false;
4534 }
4535 }
4536
4537 mTransactionMode = transactionMode;
4538 updateTransactionGroups();
4540 return true;
4541}
4542
4543QMap<QPair<QString, QString>, QgsTransactionGroup *> QgsProject::transactionGroups()
4544{
4546
4547 return mTransactionGroups;
4548}
4549
4550
4551//
4552// QgsMapLayerStore methods
4553//
4554
4555
4557{
4559
4560 return mLayerStore->count();
4561}
4562
4564{
4566
4567 return mLayerStore->validCount();
4568}
4569
4570QgsMapLayer *QgsProject::mapLayer( const QString &layerId ) const
4571{
4572 // because QgsVirtualLayerProvider is not anywhere NEAR thread safe:
4574
4575 return mLayerStore->mapLayer( layerId );
4576}
4577
4578QList<QgsMapLayer *> QgsProject::mapLayersByName( const QString &layerName ) const
4579{
4581
4582 return mLayerStore->mapLayersByName( layerName );
4583}
4584
4585QList<QgsMapLayer *> QgsProject::mapLayersByShortName( const QString &shortName ) const
4586{
4588
4589 QList<QgsMapLayer *> layers;
4590 const auto constMapLayers { mLayerStore->mapLayers() };
4591 for ( const auto &l : constMapLayers )
4592 {
4593 if ( ! l->serverProperties()->shortName().isEmpty() )
4594 {
4595 if ( l->serverProperties()->shortName() == shortName )
4596 layers << l;
4597 }
4598 else if ( l->name() == shortName )
4599 {
4600 layers << l;
4601 }
4602 }
4603 return layers;
4604}
4605
4606bool QgsProject::unzip( const QString &filename, Qgis::ProjectReadFlags flags )
4607{
4609
4610 clearError();
4611 auto archive = std::make_unique<QgsProjectArchive>();
4612
4613 // unzip the archive
4614 if ( !archive->unzip( filename ) )
4615 {
4616 setError( tr( "Unable to unzip file '%1'" ).arg( filename ) );
4617 return false;
4618 }
4619
4620 // test if zip provides a .qgs file
4621 if ( archive->projectFile().isEmpty() )
4622 {
4623 setError( tr( "Zip archive does not provide a project file" ) );
4624 return false;
4625 }
4626
4627 // Keep the archive
4628 releaseHandlesToProjectArchive();
4629 mArchive = std::move( archive );
4630
4631 // load auxiliary storage
4632 if ( !static_cast<QgsProjectArchive *>( mArchive.get() )->auxiliaryStorageFile().isEmpty() )
4633 {
4634 // database file is already a copy as it's been unzipped. So we don't open
4635 // auxiliary storage in copy mode in this case
4636 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( static_cast<QgsProjectArchive *>( mArchive.get() )->auxiliaryStorageFile(), false ) );
4637 }
4638 else
4639 {
4640 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( *this ) );
4641 }
4642
4643 // read the project file
4644 if ( ! readProjectFile( static_cast<QgsProjectArchive *>( mArchive.get() )->projectFile(), flags ) )
4645 {
4646 setError( tr( "Cannot read unzipped qgs project file" ) + QStringLiteral( ": " ) + error() );
4647 return false;
4648 }
4649
4650 // Remove the temporary .qgs file
4651 static_cast<QgsProjectArchive *>( mArchive.get() )->clearProjectFile();
4652
4653 return true;
4654}
4655
4656bool QgsProject::zip( const QString &filename )
4657{
4659
4660 clearError();
4661
4662 // save the current project in a temporary .qgs file
4663 auto archive = std::make_unique<QgsProjectArchive>();
4664 const QString baseName = QFileInfo( filename ).baseName();
4665 const QString qgsFileName = QStringLiteral( "%1.qgs" ).arg( baseName );
4666 QFile qgsFile( QDir( archive->dir() ).filePath( qgsFileName ) );
4667
4668 bool writeOk = false;
4669 if ( qgsFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
4670 {
4671 writeOk = writeProjectFile( qgsFile.fileName() );
4672 qgsFile.close();
4673 }
4674
4675 // stop here with an error message
4676 if ( ! writeOk )
4677 {
4678 setError( tr( "Unable to write temporary qgs file" ) );
4679 return false;
4680 }
4681
4682 // save auxiliary storage
4683 const QFileInfo info( qgsFile );
4684 const QString asExt = QStringLiteral( ".%1" ).arg( QgsAuxiliaryStorage::extension() );
4685 const QString asFileName = info.path() + QDir::separator() + info.completeBaseName() + asExt;
4686
4687 bool auxiliaryStorageSavedOk = true;
4688 if ( ! saveAuxiliaryStorage( asFileName ) )
4689 {
4690 const QString err = mAuxiliaryStorage->errorString();
4691 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 ) );
4692 auxiliaryStorageSavedOk = false;
4693
4694 // fixes the current archive and keep the previous version of qgd
4695 if ( !mArchive->exists() )
4696 {
4697 releaseHandlesToProjectArchive();
4698 mArchive.reset( new QgsProjectArchive() );
4699 mArchive->unzip( mFile.fileName() );
4700 static_cast<QgsProjectArchive *>( mArchive.get() )->clearProjectFile();
4701
4702 const QString auxiliaryStorageFile = static_cast<QgsProjectArchive *>( mArchive.get() )->auxiliaryStorageFile();
4703 if ( ! auxiliaryStorageFile.isEmpty() )
4704 {
4705 archive->addFile( auxiliaryStorageFile );
4706 mAuxiliaryStorage.reset( new QgsAuxiliaryStorage( auxiliaryStorageFile, false ) );
4707 }
4708 }
4709 }
4710 else
4711 {
4712 // in this case, an empty filename means that the auxiliary database is
4713 // empty, so we don't want to save it
4714 if ( QFile::exists( asFileName ) )
4715 {
4716 archive->addFile( asFileName );
4717 }
4718 }
4719
4720 // create the archive
4721 archive->addFile( qgsFile.fileName() );
4722
4723 // Add all other files
4724 const QStringList &files = mArchive->files();
4725 for ( const QString &file : files )
4726 {
4727 if ( !file.endsWith( QLatin1String( ".qgs" ), Qt::CaseInsensitive ) && !file.endsWith( asExt, Qt::CaseInsensitive ) )
4728 {
4729 archive->addFile( file );
4730 }
4731 }
4732
4733 // zip
4734 bool zipOk = true;
4735 if ( !archive->zip( filename ) )
4736 {
4737 setError( tr( "Unable to perform zip" ) );
4738 zipOk = false;
4739 }
4740
4741 return auxiliaryStorageSavedOk && zipOk;
4742}
4743
4745{
4747
4748 return QgsZipUtils::isZipFile( mFile.fileName() );
4749}
4750
4751QList<QgsMapLayer *> QgsProject::addMapLayers(
4752 const QList<QgsMapLayer *> &layers,
4753 bool addToLegend,
4754 bool takeOwnership )
4755{
4757
4758 const QList<QgsMapLayer *> myResultList { mLayerStore->addMapLayers( layers, takeOwnership ) };
4759 if ( !myResultList.isEmpty() )
4760 {
4761 // Update transform context
4762 for ( auto &l : myResultList )
4763 {
4764 l->setTransformContext( transformContext() );
4765 }
4766 if ( addToLegend )
4767 {
4768 emit legendLayersAdded( myResultList );
4769 }
4770 }
4771
4772 if ( mAuxiliaryStorage )
4773 {
4774 for ( QgsMapLayer *mlayer : myResultList )
4775 {
4776 if ( mlayer->type() != Qgis::LayerType::Vector )
4777 continue;
4778
4779 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mlayer );
4780 if ( vl )
4781 {
4782 vl->loadAuxiliaryLayer( *mAuxiliaryStorage );
4783 }
4784 }
4785 }
4786
4787 mProjectScope.reset();
4788
4789 return myResultList;
4790}
4791
4794 bool addToLegend,
4795 bool takeOwnership )
4796{
4798
4799 QList<QgsMapLayer *> addedLayers;
4800 addedLayers = addMapLayers( QList<QgsMapLayer *>() << layer, addToLegend, takeOwnership );
4801 return addedLayers.isEmpty() ? nullptr : addedLayers[0];
4802}
4803
4804void QgsProject::removeAuxiliaryLayer( const QgsMapLayer *ml )
4805{
4807
4808 if ( ! ml || ml->type() != Qgis::LayerType::Vector )
4809 return;
4810
4811 const QgsVectorLayer *vl = qobject_cast<const QgsVectorLayer *>( ml );
4812 if ( vl && vl->auxiliaryLayer() )
4813 {
4814 const QgsDataSourceUri uri( vl->auxiliaryLayer()->source() );
4816 }
4817}
4818
4819void QgsProject::removeMapLayers( const QStringList &layerIds )
4820{
4822
4823 for ( const auto &layerId : layerIds )
4824 removeAuxiliaryLayer( mLayerStore->mapLayer( layerId ) );
4825
4826 mProjectScope.reset();
4827 mLayerStore->removeMapLayers( layerIds );
4828}
4829
4830void QgsProject::removeMapLayers( const QList<QgsMapLayer *> &layers )
4831{
4833
4834 for ( const auto &layer : layers )
4835 removeAuxiliaryLayer( layer );
4836
4837 mProjectScope.reset();
4838 mLayerStore->removeMapLayers( layers );
4839}
4840
4841void QgsProject::removeMapLayer( const QString &layerId )
4842{
4844
4845 removeAuxiliaryLayer( mLayerStore->mapLayer( layerId ) );
4846 mProjectScope.reset();
4847 mLayerStore->removeMapLayer( layerId );
4848}
4849
4851{
4853
4854 removeAuxiliaryLayer( layer );
4855 mProjectScope.reset();
4856 mLayerStore->removeMapLayer( layer );
4857}
4858
4860{
4862
4863 mProjectScope.reset();
4864 return mLayerStore->takeMapLayer( layer );
4865}
4866
4868{
4870
4871 return mMainAnnotationLayer;
4872}
4873
4875{
4877
4878 if ( mLayerStore->count() == 0 )
4879 return;
4880
4881 ScopedIntIncrementor snapSingleBlocker( &mBlockSnappingUpdates );
4882 mProjectScope.reset();
4883 mLayerStore->removeAllMapLayers();
4884
4885 snapSingleBlocker.release();
4886 mSnappingConfig.clearIndividualLayerSettings();
4887 if ( !mBlockSnappingUpdates )
4888 emit snappingConfigChanged( mSnappingConfig );
4889}
4890
4892{
4894
4895 const QMap<QString, QgsMapLayer *> layers = mLayerStore->mapLayers();
4896 QMap<QString, QgsMapLayer *>::const_iterator it = layers.constBegin();
4897 for ( ; it != layers.constEnd(); ++it )
4898 {
4899 it.value()->reload();
4900 }
4901}
4902
4903QMap<QString, QgsMapLayer *> QgsProject::mapLayers( const bool validOnly ) const
4904{
4905 // because QgsVirtualLayerProvider is not anywhere NEAR thread safe:
4907
4908 return validOnly ? mLayerStore->validMapLayers() : mLayerStore->mapLayers();
4909}
4910
4911QgsTransactionGroup *QgsProject::transactionGroup( const QString &providerKey, const QString &connString )
4912{
4914
4915 return mTransactionGroups.value( qMakePair( providerKey, connString ) );
4916}
4917
4924
4926{
4928
4930
4931 // TODO QGIS 4.0 -- remove this method, and place it somewhere in app (where it belongs)
4932 // 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)
4933 if ( mSettings.value( QStringLiteral( "/projections/unknownCrsBehavior" ), QStringLiteral( "NoAction" ), QgsSettings::App ).toString() == QStringLiteral( "UseProjectCrs" )
4934 || mSettings.value( QStringLiteral( "/projections/unknownCrsBehavior" ), 0, QgsSettings::App ).toString() == QLatin1String( "2" ) )
4935 {
4936 // for new layers if the new layer crs method is set to either prompt or use project, then we use the project crs
4937 defaultCrs = crs();
4938 }
4939 else
4940 {
4941 // global crs
4942 const QString layerDefaultCrs = mSettings.value( QStringLiteral( "/Projections/layerDefaultCrs" ), geoEpsgCrsAuthId() ).toString();
4943 defaultCrs = QgsCoordinateReferenceSystem::fromOgcWmsCrs( layerDefaultCrs );
4944 }
4945
4946 return defaultCrs;
4947}
4948
4955
4962
4963bool QgsProject::saveAuxiliaryStorage( const QString &filename )
4964{
4966
4967 const QMap<QString, QgsMapLayer *> layers = mapLayers();
4968 bool empty = true;
4969 for ( auto it = layers.constBegin(); it != layers.constEnd(); ++it )
4970 {
4971 if ( it.value()->type() != Qgis::LayerType::Vector )
4972 continue;
4973
4974 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( it.value() );
4975 if ( vl && vl->auxiliaryLayer() )
4976 {
4977 vl->auxiliaryLayer()->save();
4978 empty &= vl->auxiliaryLayer()->auxiliaryFields().isEmpty();
4979 }
4980 }
4981
4982 if ( !mAuxiliaryStorage->exists( *this ) && empty )
4983 {
4984 return true; // it's not an error
4985 }
4986 else if ( !filename.isEmpty() )
4987 {
4988 return mAuxiliaryStorage->saveAs( filename );
4989 }
4990 else
4991 {
4992 return mAuxiliaryStorage->saveAs( *this );
4993 }
4994}
4995
4996QgsPropertiesDefinition &QgsProject::dataDefinedServerPropertyDefinitions()
4997{
4998 static QgsPropertiesDefinition sPropertyDefinitions
4999 {
5000 {
5002 QgsPropertyDefinition( "WMSOnlineResource", QObject::tr( "WMS Online Resource" ), QgsPropertyDefinition::String )
5003 },
5004 };
5005 return sPropertyDefinitions;
5006}
5007
5009{
5010 mElevationShadingRenderer = elevationShadingRenderer;
5012}
5013
5015{
5017
5018 return mAuxiliaryStorage.get();
5019}
5020
5022{
5024
5025 return mAuxiliaryStorage.get();
5026}
5027
5028QString QgsProject::createAttachedFile( const QString &nameTemplate )
5029{
5031
5032 const QDir archiveDir( mArchive->dir() );
5033 QTemporaryFile tmpFile( archiveDir.filePath( "XXXXXX_" + nameTemplate ), this );
5034 tmpFile.setAutoRemove( false );
5035 tmpFile.open();
5036 mArchive->addFile( tmpFile.fileName() );
5037 return tmpFile.fileName();
5038}
5039
5040QStringList QgsProject::attachedFiles() const
5041{
5043
5044 QStringList attachments;
5045 const QString baseName = QFileInfo( fileName() ).baseName();
5046 const QStringList files = mArchive->files();
5047 attachments.reserve( files.size() );
5048 for ( const QString &file : files )
5049 {
5050 if ( QFileInfo( file ).baseName() != baseName )
5051 {
5052 attachments.append( file );
5053 }
5054 }
5055 return attachments;
5056}
5057
5058bool QgsProject::removeAttachedFile( const QString &path )
5059{
5061
5062 return mArchive->removeFile( path );
5063}
5064
5065QString QgsProject::attachmentIdentifier( const QString &attachedFile ) const
5066{
5068
5069 return QStringLiteral( "attachment:///%1" ).arg( QFileInfo( attachedFile ).fileName() );
5070}
5071
5072QString QgsProject::resolveAttachmentIdentifier( const QString &identifier ) const
5073{
5075
5076 if ( identifier.startsWith( QLatin1String( "attachment:///" ) ) )
5077 {
5078 return QDir( mArchive->dir() ).absoluteFilePath( identifier.mid( 14 ) );
5079 }
5080 return QString();
5081}
5082
5084{
5085 // this method is called quite extensively from other threads via QgsProject::createExpressionContextScope()
5087
5088 return mMetadata;
5089}
5090
5092{
5094
5095 if ( metadata == mMetadata )
5096 return;
5097
5098 mMetadata = metadata;
5099 mProjectScope.reset();
5100
5101 emit metadataChanged();
5102
5103 setDirty( true );
5104}
5105
5106QSet<QgsMapLayer *> QgsProject::requiredLayers() const
5107{
5109
5110 QSet<QgsMapLayer *> requiredLayers;
5111
5112 const QMap<QString, QgsMapLayer *> &layers = mapLayers();
5113 for ( QMap<QString, QgsMapLayer *>::const_iterator it = layers.constBegin(); it != layers.constEnd(); ++it )
5114 {
5115 if ( !it.value()->flags().testFlag( QgsMapLayer::Removable ) )
5116 {
5117 requiredLayers.insert( it.value() );
5118 }
5119 }
5120 return requiredLayers;
5121}
5122
5123void QgsProject::setRequiredLayers( const QSet<QgsMapLayer *> &layers )
5124{
5126
5127 const QMap<QString, QgsMapLayer *> &projectLayers = mapLayers();
5128 for ( QMap<QString, QgsMapLayer *>::const_iterator it = projectLayers.constBegin(); it != projectLayers.constEnd(); ++it )
5129 {
5130 if ( layers.contains( it.value() ) == !it.value()->flags().testFlag( QgsMapLayer::Removable ) )
5131 continue;
5132
5133 if ( layers.contains( it.value() ) )
5134 it.value()->setFlags( it.value()->flags() & ~QgsMapLayer::Removable );
5135 else
5136 it.value()->setFlags( it.value()->flags() | QgsMapLayer::Removable );
5137 }
5138}
5139
5141{
5143
5144 // save colors to project
5145 QStringList customColors;
5146 QStringList customColorLabels;
5147
5148 QgsNamedColorList::const_iterator colorIt = colors.constBegin();
5149 for ( ; colorIt != colors.constEnd(); ++colorIt )
5150 {
5151 const QString color = QgsColorUtils::colorToString( ( *colorIt ).first );
5152 const QString label = ( *colorIt ).second;
5153 customColors.append( color );
5154 customColorLabels.append( label );
5155 }
5156 writeEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Colors" ), customColors );
5157 writeEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Labels" ), customColorLabels );
5158 mProjectScope.reset();
5159 emit projectColorsChanged();
5160}
5161
5162void QgsProject::setBackgroundColor( const QColor &color )
5163{
5165
5166 if ( mBackgroundColor == color )
5167 return;
5168
5169 mBackgroundColor = color;
5171}
5172
5174{
5176
5177 return mBackgroundColor;
5178}
5179
5180void QgsProject::setSelectionColor( const QColor &color )
5181{
5183
5184 if ( mSelectionColor == color )
5185 return;
5186
5187 mSelectionColor = color;
5188 emit selectionColorChanged();
5189}
5190
5192{
5194
5195 return mSelectionColor;
5196}
5197
5198void QgsProject::setMapScales( const QVector<double> &scales )
5199{
5201
5202 mViewSettings->setMapScales( scales );
5203}
5204
5205QVector<double> QgsProject::mapScales() const
5206{
5208
5209 return mViewSettings->mapScales();
5210}
5211
5213{
5215
5216 mViewSettings->setUseProjectScales( enabled );
5217}
5218
5220{
5222
5223 return mViewSettings->useProjectScales();
5224}
5225
5226void QgsProject::generateTsFile( const QString &locale )
5227{
5229
5230 QgsTranslationContext translationContext;
5231 translationContext.setProject( this );
5232 translationContext.setFileName( QStringLiteral( "%1/%2.ts" ).arg( absolutePath(), baseName() ) );
5233
5234 QgsApplication::instance()->collectTranslatableObjects( &translationContext );
5235
5236 translationContext.writeTsFile( locale );
5237}
5238
5239QString QgsProject::translate( const QString &context, const QString &sourceText, const char *disambiguation, int n ) const
5240{
5242
5243 if ( !mTranslator )
5244 {
5245 return sourceText;
5246 }
5247
5248 QString result = mTranslator->translate( context.toUtf8(), sourceText.toUtf8(), disambiguation, n );
5249
5250 if ( result.isEmpty() )
5251 {
5252 return sourceText;
5253 }
5254 return result;
5255}
5256
5258{
5260
5261 const QMap<QString, QgsMapLayer *> layers = mapLayers( false );
5262 if ( !layers.empty() )
5263 {
5264 for ( auto it = layers.constBegin(); it != layers.constEnd(); ++it )
5265 {
5266 // NOTE: if visitEnter returns false it means "don't visit this layer", not "abort all further visitations"
5267 if ( visitor->visitEnter( QgsStyleEntityVisitorInterface::Node( QgsStyleEntityVisitorInterface::NodeType::Layer, ( *it )->id(), ( *it )->name() ) ) )
5268 {
5269 if ( !( ( *it )->accept( visitor ) ) )
5270 return false;
5271
5272 if ( !visitor->visitExit( QgsStyleEntityVisitorInterface::Node( QgsStyleEntityVisitorInterface::NodeType::Layer, ( *it )->id(), ( *it )->name() ) ) )
5273 return false;
5274 }
5275 }
5276 }
5277
5278 if ( !mLayoutManager->accept( visitor ) )
5279 return false;
5280
5281 if ( !mAnnotationManager->accept( visitor ) )
5282 return false;
5283
5284 return true;
5285}
5286
5288{
5289 return mElevationShadingRenderer;
5290}
5291
5292void QgsProject::loadProjectFlags( const QDomDocument *doc )
5293{
5295
5296 QDomElement element = doc->documentElement().firstChildElement( QStringLiteral( "projectFlags" ) );
5298 if ( !element.isNull() )
5299 {
5300 flags = qgsFlagKeysToValue( element.attribute( QStringLiteral( "set" ) ), Qgis::ProjectFlags() );
5301 }
5302 else
5303 {
5304 // older project compatibility
5305 element = doc->documentElement().firstChildElement( QStringLiteral( "evaluateDefaultValues" ) );
5306 if ( !element.isNull() )
5307 {
5308 if ( element.attribute( QStringLiteral( "active" ), QStringLiteral( "0" ) ).toInt() == 1 )
5310 }
5311
5312 // Read trust layer metadata config in the project
5313 element = doc->documentElement().firstChildElement( QStringLiteral( "trust" ) );
5314 if ( !element.isNull() )
5315 {
5316 if ( element.attribute( QStringLiteral( "active" ), QStringLiteral( "0" ) ).toInt() == 1 )
5318 }
5319 }
5320
5321 setFlags( flags );
5322}
5323
5325{
5327 {
5328 const Qgis::PythonEmbeddedMode pythonEmbeddedMode = QgsSettings().enumValue( QStringLiteral( "qgis/enablePythonEmbedded" ), Qgis::PythonEmbeddedMode::Ask );
5329
5330 if ( force || pythonEmbeddedMode == Qgis::PythonEmbeddedMode::SessionOnly || pythonEmbeddedMode == Qgis::PythonEmbeddedMode::Always )
5331 {
5332 const QString projectFunctions = readEntry( QStringLiteral( "ExpressionFunctions" ), QStringLiteral( "/pythonCode" ), QString() );
5333 if ( !projectFunctions.isEmpty() )
5334 {
5335 QgsPythonRunner::run( projectFunctions );
5336 return true;
5337 }
5338 }
5339 }
5340 return false;
5341}
5342
5344{
5346 {
5347 QgsPythonRunner::run( "qgis.utils.clean_project_expression_functions()" );
5348 }
5349}
5350
5352
5353QHash< QString, QColor > loadColorsFromProject( const QgsProject *project )
5354{
5355 QHash< QString, QColor > colors;
5356
5357 //build up color list from project. Do this in advance for speed
5358 QStringList colorStrings = project->readListEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Colors" ) );
5359 const QStringList colorLabels = project->readListEntry( QStringLiteral( "Palette" ), QStringLiteral( "/Labels" ) );
5360
5361 //generate list from custom colors
5362 int colorIndex = 0;
5363 for ( QStringList::iterator it = colorStrings.begin();
5364 it != colorStrings.end(); ++it )
5365 {
5366 const QColor color = QgsColorUtils::colorFromString( *it );
5367 QString label;
5368 if ( colorLabels.length() > colorIndex )
5369 {
5370 label = colorLabels.at( colorIndex );
5371 }
5372
5373 colors.insert( label.toLower(), color );
5374 colorIndex++;
5375 }
5376
5377 return colors;
5378}
5379
5380
5381GetNamedProjectColor::GetNamedProjectColor( const QgsProject *project )
5382 : QgsScopedExpressionFunction( QStringLiteral( "project_color" ), 1, QStringLiteral( "Color" ) )
5383{
5384 if ( !project )
5385 return;
5386
5387 mColors = loadColorsFromProject( project );
5388}
5389
5390GetNamedProjectColor::GetNamedProjectColor( const QHash<QString, QColor> &colors )
5391 : QgsScopedExpressionFunction( QStringLiteral( "project_color" ), 1, QStringLiteral( "Color" ) )
5392 , mColors( colors )
5393{
5394}
5395
5396QVariant GetNamedProjectColor::func( const QVariantList &values, const QgsExpressionContext *, QgsExpression *, const QgsExpressionNodeFunction * )
5397{
5398 const QString colorName = values.at( 0 ).toString().toLower();
5399 if ( mColors.contains( colorName ) )
5400 {
5401 return QStringLiteral( "%1,%2,%3" ).arg( mColors.value( colorName ).red() ).arg( mColors.value( colorName ).green() ).arg( mColors.value( colorName ).blue() );
5402 }
5403 else
5404 return QVariant();
5405}
5406
5407QgsScopedExpressionFunction *GetNamedProjectColor::clone() const
5408{
5409 return new GetNamedProjectColor( mColors );
5410}
5411
5412GetNamedProjectColorObject::GetNamedProjectColorObject( const QgsProject *project )
5413 : QgsScopedExpressionFunction( QStringLiteral( "project_color_object" ), 1, QStringLiteral( "Color" ) )
5414{
5415 if ( !project )
5416 return;
5417
5418 mColors = loadColorsFromProject( project );
5419}
5420
5421GetNamedProjectColorObject::GetNamedProjectColorObject( const QHash<QString, QColor> &colors )
5422 : QgsScopedExpressionFunction( QStringLiteral( "project_color_object" ), 1, QStringLiteral( "Color" ) )
5423 , mColors( colors )
5424{
5425}
5426
5427QVariant GetNamedProjectColorObject::func( const QVariantList &values, const QgsExpressionContext *, QgsExpression *, const QgsExpressionNodeFunction * )
5428{
5429 const QString colorName = values.at( 0 ).toString().toLower();
5430 if ( mColors.contains( colorName ) )
5431 {
5432 return mColors.value( colorName );
5433 }
5434 else
5435 return QVariant();
5436}
5437
5438QgsScopedExpressionFunction *GetNamedProjectColorObject::clone() const
5439{
5440 return new GetNamedProjectColorObject( mColors );
5441}
5442
5443// ----------------
5444
5445GetSensorData::GetSensorData( const QMap<QString, QgsAbstractSensor::SensorData> &sensorData )
5446 : QgsScopedExpressionFunction( QStringLiteral( "sensor_data" ),
5447 QgsExpressionFunction::ParameterList() << QgsExpressionFunction::Parameter( QStringLiteral( "name" ) ) << QgsExpressionFunction::Parameter( QStringLiteral( "expiration" ), true, 0 ),
5448 QStringLiteral( "Sensors" ) )
5449 , mSensorData( sensorData )
5450{
5451}
5452
5453QVariant GetSensorData::func( const QVariantList &values, const QgsExpressionContext *, QgsExpression *, const QgsExpressionNodeFunction * )
5454{
5455 const QString sensorName = values.at( 0 ).toString();
5456 const int expiration = values.at( 1 ).toInt();
5457 const qint64 timestamp = QDateTime::currentMSecsSinceEpoch();
5458 if ( mSensorData.contains( sensorName ) )
5459 {
5460 if ( expiration <= 0 || ( timestamp - mSensorData[sensorName].lastTimestamp.toMSecsSinceEpoch() ) < expiration )
5461 {
5462 return mSensorData[sensorName].lastValue;
5463 }
5464 }
5465
5466 return QVariant();
5467}
5468
5469QgsScopedExpressionFunction *GetSensorData::clone() const
5470{
5471 return new GetSensorData( mSensorData );
5472}
@ 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:259
QFlags< ProjectCapability > ProjectCapabilities
Flags which control project capabilities.
Definition qgis.h:4189
QFlags< ProjectReadFlag > ProjectReadFlags
Project load flags.
Definition qgis.h:4167
DistanceUnit
Units of distance.
Definition qgis.h:4838
FilePathType
File path types.
Definition qgis.h:1639
@ Relative
Relative path.
@ Absolute
Absolute path.
TransactionMode
Transaction mode.
Definition qgis.h:3815
@ 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:4915
@ 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:4119
@ 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:3938
@ 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.
@ 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:3945
@ Marker
Marker symbol.
@ Line
Line symbol.
@ Fill
Fill symbol.
@ 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.
Class allowing to manage the zip/unzip actions.
Definition qgsarchive.h:35
void addFile(const QString &filename)
Add a new file to this archive.
This is a container for attribute editors, used to group them visually in the attribute form if it is...
QList< QgsAttributeEditorElement * > children() const
Gets a list of the children elements of this container.
This is 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.
Class 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.
This class 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 ...
Class for storing the component parts of a RDBMS data source URI (e.g.
QgsAttributeEditorContainer * invisibleRootContainer()
Gets the invisible root container for the drag and drop designer form (EditorLayout::TabLayout).
This class can render elevation shading on an image with different methods (eye dome lighting,...
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...
A abstract base class for defining QgsExpression functions.
An expression node for expression functions.
Class for 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.
Class used to work with layer 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)
This class is a 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 the updates in map layer registry and does changes in layer tree.
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:76
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:83
QString id
Definition qgsmaplayer.h:79
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:86
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.
Class allowing to manage the zip/unzip actions on project file.
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 file.
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.
Class to 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.
A class to describe 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:208
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...
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 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 a 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)
The class is used as a container of context for various read/write operations on other 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.
This class 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
The QgsRunnableProviderCreator class is used when reading a project to create asynchronously provider...
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.
This class is a composition of two QSettings instances:
Definition qgssettings.h:64
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.
This is a container for configuration of the snapping of 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
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 data sets.
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
QList< QPair< QColor, QString > > QgsNamedColorList
List of colors paired with a friendly display name identifying the color.
CORE_EXPORT bool isZipFile(const QString &filename)
Returns true if the file name is a zipped file ( i.e with a '.qgz' extension, false otherwise.
CORE_EXPORT const QStringList files(const QString &zip)
Returns the list of files within a zip file.
CONSTLATIN1STRING geoNone()
Constant that holds the string representation for "No ellips/No CRS".
Definition qgis.h:6714
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:6429
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:6796
QString qgsEnumValueToKey(const T &value, bool *returnOk=nullptr)
Returns the value for the given key of an enum.
Definition qgis.h:6410
QString qgsFlagValueToKeys(const T &value, bool *returnOk=nullptr)
Returns the value for the given keys of a flag.
Definition qgis.h:6468
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:6490
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:6795
CONSTLATIN1STRING geoEpsgCrsAuthId()
Geographic coord sys from EPSG authority.
Definition qgis.h:6708
#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.