19#include "moc_qgsproject.cpp"
81#include <QApplication>
86#include <QTemporaryFile>
89#include <QStandardPaths>
91#include <QRegularExpression>
113 QStringList keyTokens = QStringList( scope );
114 keyTokens += key.split(
'/', Qt::SkipEmptyParts );
117 keyTokens.push_front( QStringLiteral(
"properties" ) );
142 while ( !keySequence.isEmpty() )
146 if ( keySequence.first() == currentProperty->
name() )
149 keySequence.pop_front();
151 if ( 1 == keySequence.count() )
154 return currentProperty->
find( keySequence.front() );
156 else if ( keySequence.isEmpty() )
161 return currentProperty;
163 else if ( ( nextProperty = currentProperty->
find( keySequence.first() ) ) )
165 if ( nextProperty->
isKey() )
169 else if ( nextProperty->
isValue() && 1 == keySequence.count() )
175 return currentProperty;
213 const QVariant &value,
214 bool &propertiesModified )
223 propertiesModified =
false;
224 while ( ! keySequence.isEmpty() )
228 if ( keySequence.first() == currentProperty->
name() )
231 keySequence.pop_front();
235 if ( 1 == keySequence.count() )
238 if ( !property || property->value() != value )
240 currentProperty->
setValue( keySequence.front(), value );
241 propertiesModified =
true;
244 return currentProperty;
248 else if ( keySequence.isEmpty() )
250 if ( currentProperty->
value() != value )
253 propertiesModified =
true;
256 return currentProperty;
258 else if ( ( nextProperty = currentProperty->
find( keySequence.first() ) ) )
262 if ( currentProperty )
273 if ( ( newPropertyKey = currentProperty->
addKey( keySequence.first() ) ) )
275 currentProperty = newPropertyKey;
307 while ( ! keySequence.isEmpty() )
311 if ( keySequence.first() == currentProperty->
name() )
314 keySequence.pop_front();
318 if ( 1 == keySequence.count() )
320 currentProperty->
removeKey( keySequence.front() );
325 else if ( keySequence.isEmpty() )
327 previousQgsPropertyKey->
removeKey( currentProperty->
name() );
329 else if ( ( nextProperty = currentProperty->
find( keySequence.first() ) ) )
331 previousQgsPropertyKey = currentProperty;
334 if ( currentProperty )
358 , mCapabilities( capabilities )
361 , mSnappingConfig( this )
379 mProperties.
setName( QStringLiteral(
"properties" ) );
382 mMainAnnotationLayer->setParent(
this );
396 this, [
this](
const QStringList &
layers ) { mProjectScope.reset(); emit layersWillBeRemoved( layers ); } );
398 this, [
this](
const QList<QgsMapLayer *> &
layers ) { mProjectScope.reset(); emit layersWillBeRemoved( layers ); } );
400 this, [
this](
const QString & layer ) { mProjectScope.reset(); emit layerWillBeRemoved( layer ); } );
402 this, [
this](
QgsMapLayer * layer ) { mProjectScope.reset(); emit layerWillBeRemoved( layer ); } );
404 [
this](
const QStringList &
layers ) { mProjectScope.reset(); emit layersRemoved( layers ); } );
406 [
this](
const QString & layer ) { mProjectScope.reset(); emit layerRemoved( layer ); } );
408 [
this]() { mProjectScope.reset(); emit removeAll(); } );
410 [
this](
const QList< QgsMapLayer * > &
layers ) { mProjectScope.reset(); emit layersAdded( layers ); } );
412 [
this](
QgsMapLayer * layer ) { mProjectScope.reset(); emit layerWasAdded( layer ); } );
420 [
this](
const QList<QgsMapLayer *> &
layers )
422 for ( const auto &layer : layers )
424 disconnect( layer, &QgsMapLayer::dataSourceChanged, mRelationManager, &QgsRelationManager::updateRelationsStatus );
429 [
this](
const QList<QgsMapLayer *> &layers )
431 for ( const auto &layer : layers )
433 connect( layer, &QgsMapLayer::dataSourceChanged, mRelationManager, &QgsRelationManager::updateRelationsStatus );
442 mStyleSettings->combinedStyleModel()->addDefaultStyle();
448 mIsBeingDeleted =
true;
451 releaseHandlesToProjectArchive();
452 delete mBadLayerHandler;
453 delete mRelationManager;
454 delete mLayerTreeRegistryBridge;
456 if (
this == sProject )
487 mProjectScope.reset();
498 return mMetadata.
title();
507 if ( oldEvaluateDefaultValues != newEvaluateDefaultValues )
510 for (
auto layerIt =
layers.constBegin(); layerIt !=
layers.constEnd(); ++layerIt )
512 if (
QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( layerIt.value() ) )
513 if ( vl->dataProvider() )
520 if ( oldTrustLayerMetadata != newTrustLayerMetadata )
523 for (
auto layerIt =
layers.constBegin(); layerIt !=
layers.constEnd(); ++layerIt )
525 if (
QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( layerIt.value() ) )
527 vl->setReadExtentFromXml( newTrustLayerMetadata );
532 if ( mFlags !=
flags )
547 newFlags &= ~(
static_cast< int >( flag ) );
562 return mSaveUserFull;
569 return mSaveDateTime;
590 if ( dirty && mDirtyBlockCount > 0 )
596 if ( mDirty == dirty )
607 if ( path == mHomePath )
611 mCachedHomePath.clear();
612 mProjectScope.reset();
623 const QList<QgsAttributeEditorElement *> elements = parent->
children();
631 translationContext->
registerTranslation( QStringLiteral(
"project:layers:%1:formcontainers" ).arg( layerId ), container->
name() );
633 if ( !container->
children().empty() )
648 translationContext->
registerTranslation( QStringLiteral(
"project:layers:%1" ).arg( layer->layerId() ), layer->name() );
657 for (
const QgsField &field : fields )
660 if ( field.alias().isEmpty() )
661 fieldName = field.name();
663 fieldName = field.alias();
665 translationContext->
registerTranslation( QStringLiteral(
"project:layers:%1:fieldaliases" ).arg( vlayer->
id() ), fieldName );
667 if ( field.editorWidgetSetup().type() == QStringLiteral(
"ValueRelation" ) )
669 translationContext->
registerTranslation( QStringLiteral(
"project:layers:%1:fields:%2:valuerelationvalue" ).arg( vlayer->
id(), field.name() ), field.editorWidgetSetup().config().value( QStringLiteral(
"Value" ) ).toString() );
671 if ( field.editorWidgetSetup().type() == QStringLiteral(
"ValueMap" ) )
673 if ( field.editorWidgetSetup().config().value( QStringLiteral(
"map" ) ).canConvert<QList<QVariant>>() )
675 const QList<QVariant> valueList = field.editorWidgetSetup().config().value( QStringLiteral(
"map" ) ).toList();
677 for (
int i = 0, row = 0; i < valueList.count(); i++, row++ )
679 translationContext->
registerTranslation( QStringLiteral(
"project:layers:%1:fields:%2:valuemapdescriptions" ).arg( vlayer->
id(), field.name() ), valueList[i].toMap().constBegin().key() );
692 const QList<QgsLayerTreeGroup *> groupLayers = mRootGroup->
findGroups();
695 translationContext->
registerTranslation( QStringLiteral(
"project:layergroups" ), groupLayer->name() );
699 const QList<QgsRelation> &relations = mRelationManager->
relations().values();
702 translationContext->
registerTranslation( QStringLiteral(
"project:relations" ), relation.name() );
710 mDataDefinedServerProperties = properties;
717 return mDataDefinedServerProperties;
724 switch ( mTransactionMode )
745 switch ( mTransactionMode )
752 commitErrors.append( tr(
"Trying to commit changes without a layer specified. This only works if the transaction mode is buffered" ) );
761 return mEditBufferGroup.
commitChanges( commitErrors, stopEditing );
771 switch ( mTransactionMode )
778 rollbackErrors.append( tr(
"Trying to roll back changes without a layer specified. This only works if the transaction mode is buffered" ) );
781 bool success = vectorLayer->
rollBack( stopEditing );
787 return mEditBufferGroup.
rollBack( rollbackErrors, stopEditing );
797 if ( name == mFile.fileName() )
800 const QString oldHomePath =
homePath();
802 mFile.setFileName( name );
803 mCachedHomePath.clear();
804 mProjectScope.reset();
808 const QString newHomePath =
homePath();
809 if ( newHomePath != oldHomePath )
820 return mFile.fileName();
827 mOriginalPath = path;
834 return mOriginalPath;
841 return QFileInfo( mFile );
859 storage->readProjectStorageMetadata( mFile.fileName(),
metadata );
864 return QFileInfo( mFile.fileName() ).lastModified();
875 if ( mFile.fileName().isEmpty() )
878 return QFileInfo( mFile.fileName() ).absolutePath();
889 if ( mFile.fileName().isEmpty() )
892 return QFileInfo( mFile.fileName() ).absoluteFilePath();
903 storage->readProjectStorageMetadata( mFile.fileName(),
metadata );
908 return QFileInfo( mFile.fileName() ).completeBaseName();
916 const bool absolutePaths =
readBoolEntry( QStringLiteral(
"Paths" ), QStringLiteral(
"/Absolute" ),
false );
927 writeEntry( QStringLiteral(
"Paths" ), QStringLiteral(
"/Absolute" ),
true );
930 writeEntry( QStringLiteral(
"Paths" ), QStringLiteral(
"/Absolute" ),
false );
947 return mCrs3D.
isValid() ? mCrs3D : mCrs;
959 writeEntry( QStringLiteral(
"SpatialRefSys" ), QStringLiteral(
"/ProjectionsEnabled" ),
crs.
isValid() ? 1 : 0 );
960 mProjectScope.reset();
974 if ( oldCrs3D != mCrs3D )
978 if ( adjustEllipsoid )
987 if ( !
crs().isValid() )
990 return readEntry( QStringLiteral(
"Measure" ), QStringLiteral(
"/Ellipsoid" ),
geoNone() );
997 if (
ellipsoid ==
readEntry( QStringLiteral(
"Measure" ), QStringLiteral(
"/Ellipsoid" ) ) )
1000 mProjectScope.reset();
1010 switch ( mCrs.
type() )
1013 QgsDebugError( QStringLiteral(
"Project has a vertical CRS set as the horizontal CRS!" ) );
1032 return mVerticalCrs;
1060 *errorMessage = QObject::tr(
"Specified CRS is a %1 CRS, not a Vertical CRS" ).arg(
qgsEnumValueToKey(
crs.
type() ) );
1065 if (
crs != mVerticalCrs )
1070 switch ( mCrs.
type() )
1073 if (
crs != oldVerticalCrs )
1076 *errorMessage = QObject::tr(
"Project CRS is a Compound CRS, specified Vertical CRS will be ignored" );
1082 if (
crs != oldVerticalCrs )
1085 *errorMessage = QObject::tr(
"Project CRS is a Geographic 3D CRS, specified Vertical CRS will be ignored" );
1091 if (
crs != oldVerticalCrs )
1094 *errorMessage = QObject::tr(
"Project CRS is a Geocentric CRS, specified Vertical CRS will be ignored" );
1103 *errorMessage = QObject::tr(
"Project CRS is a Projected 3D CRS, specified Vertical CRS will be ignored" );
1121 res = rebuildCrs3D( errorMessage );
1122 mProjectScope.reset();
1129 if ( mCrs3D != oldCrs3D )
1140 return mTransformContext;
1147 if ( context == mTransformContext )
1150 mTransformContext = context;
1151 mProjectScope.reset();
1154 for (
auto &layer : mLayerStore.get()->mapLayers() )
1156 layer->setTransformContext( context );
1165 ScopedIntIncrementor snapSingleBlocker( &mBlockSnappingUpdates );
1169 if ( !mIsBeingDeleted )
1178 mProjectScope.reset();
1179 mFile.setFileName( QString() );
1182 mSaveUserFull.clear();
1183 mSaveDateTime = QDateTime();
1186 mCachedHomePath.clear();
1190 mCustomVariables.clear();
1196 if ( !mSettings.
value( QStringLiteral(
"projects/anonymize_new_projects" ),
false,
QgsSettings::Core ).toBool() )
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();
1235 mLabelingEngineSettings->clear();
1239 releaseHandlesToProjectArchive();
1245 mStyleSettings->
reset();
1249 if ( !mIsBeingDeleted )
1257 writeEntry( QStringLiteral(
"PositionPrecision" ), QStringLiteral(
"/Automatic" ),
true );
1258 writeEntry( QStringLiteral(
"PositionPrecision" ), QStringLiteral(
"/DecimalPlaces" ), 2 );
1260 const bool defaultRelativePaths = mSettings.
value( QStringLiteral(
"/qgis/defaultProjectPathsRelative" ),
true ).toBool();
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();
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();
1277 mRootGroup->
clear();
1278 if ( mMainAnnotationLayer )
1279 mMainAnnotationLayer->
reset();
1281 snapSingleBlocker.release();
1283 if ( !mBlockSnappingUpdates )
1288 if ( !mBlockChangeSignalsDuringClear )
1300 topQgsPropertyKey.
dump();
1333 const QDomElement propertiesElem = doc.documentElement().firstChildElement( QStringLiteral(
"properties" ) );
1335 if ( propertiesElem.isNull() )
1340 const QDomNodeList scopes = propertiesElem.childNodes();
1342 if ( propertiesElem.firstChild().isNull() )
1344 QgsDebugError( QStringLiteral(
"empty ``properties'' XML tag ... bailing" ) );
1348 if ( ! project_properties.
readXml( propertiesElem ) )
1350 QgsDebugError( QStringLiteral(
"Project_properties.readXml() failed" ) );
1364 const QDomElement ddElem = doc.documentElement().firstChildElement( QStringLiteral(
"dataDefinedServerProperties" ) );
1365 if ( !ddElem.isNull() )
1367 if ( !ddServerProperties.
readXml( ddElem, dataDefinedServerPropertyDefinitions ) )
1369 QgsDebugError( QStringLiteral(
"dataDefinedServerProperties.readXml() failed" ) );
1372 return ddServerProperties;
1379static void _getTitle(
const QDomDocument &doc, QString &title )
1381 const QDomElement titleNode = doc.documentElement().firstChildElement( QStringLiteral(
"title" ) );
1385 if ( titleNode.isNull() )
1391 if ( !titleNode.hasChildNodes() )
1397 const QDomNode titleTextNode = titleNode.firstChild();
1399 if ( !titleTextNode.isText() )
1405 const QDomText titleText = titleTextNode.toText();
1407 title = titleText.data();
1411static void readProjectFileMetadata(
const QDomDocument &doc, QString &lastUser, QString &lastUserFull, QDateTime &lastSaveDateTime )
1413 const QDomNodeList nl = doc.elementsByTagName( QStringLiteral(
"qgis" ) );
1417 QgsDebugError( QStringLiteral(
"unable to find qgis element" ) );
1421 const QDomNode qgisNode = nl.item( 0 );
1423 const QDomElement qgisElement = qgisNode.toElement();
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 );
1431 const QDomNodeList nl = doc.elementsByTagName( QStringLiteral(
"qgis" ) );
1435 QgsDebugError( QStringLiteral(
" unable to find qgis element in project file" ) );
1439 const QDomNode qgisNode = nl.item( 0 );
1441 const QDomElement qgisElement = qgisNode.toElement();
1442 QgsProjectVersion projectVersion( qgisElement.attribute( QStringLiteral(
"version" ) ) );
1443 return projectVersion;
1450 return mSnappingConfig;
1469 if ( mAvoidIntersectionsMode == mode )
1472 mAvoidIntersectionsMode = mode;
1506void QgsProject::preloadProviders(
const QVector<QDomNode> ¶llelLayerNodes,
1508 QMap<QString, QgsDataProvider *> &loadedProviders,
1510 int totalProviderCount )
1515 QMap<QString, LayerToLoad> layersToLoad;
1517 for (
const QDomNode &node : parallelLayerNodes )
1521 const QDomElement layerElement = node.toElement();
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();
1536 layersToLoad.insert( layerToLoad.
layerId, layerToLoad );
1539 while ( !layersToLoad.isEmpty() )
1541 const QList<LayerToLoad> layersToAttemptInParallel = layersToLoad.values();
1542 QString layerToAttemptInMainThread;
1544 QHash<QString, QgsRunnableProviderCreator *> runnables;
1545 QThreadPool threadPool;
1548 for (
const LayerToLoad &lay : layersToAttemptInParallel )
1551 runnables.insert( lay.layerId, run );
1557 layersToLoad.remove( layId );
1560 Q_ASSERT( finishedRun );
1562 std::unique_ptr<QgsDataProvider> provider( finishedRun->
dataProvider() );
1563 Q_ASSERT( provider && provider->isValid() );
1565 loadedProviders.insert( layId, provider.release() );
1570 if ( layerToAttemptInMainThread.isEmpty() )
1571 layerToAttemptInMainThread = layId;
1575 if ( i == parallelLayerNodes.count() || !isValid )
1578 threadPool.start( run );
1582 threadPool.waitForDone();
1584 qDeleteAll( runnables );
1587 auto it = layersToLoad.find( layerToAttemptInMainThread );
1588 if ( it != layersToLoad.end() )
1590 std::unique_ptr<QgsDataProvider> provider;
1600 if ( provider && provider->isValid() )
1605 layersToLoad.erase( it );
1608 loadedProviders.insert( layerId, provider.release() );
1616void QgsProject::releaseHandlesToProjectArchive()
1621bool QgsProject::rebuildCrs3D( QString *error )
1628 else if ( !mVerticalCrs.
isValid() )
1634 switch ( mCrs.
type() )
1675bool QgsProject::_getMapLayers(
const QDomDocument &doc, QList<QDomNode> &brokenNodes,
Qgis::ProjectReadFlags flags )
1682 QDomElement layerElement = doc.documentElement().firstChildElement( QStringLiteral(
"projectlayers" ) ).firstChildElement( QStringLiteral(
"maplayer" ) );
1686 if ( layerElement.isNull() )
1696 bool returnStatus =
true;
1699 while ( ! layerElement.isNull() )
1702 layerElement = layerElement.nextSiblingElement( QStringLiteral(
"maplayer" ) );
1708 if ( depSorter.hasCycle() )
1712 if ( depSorter.hasMissingDependency() )
1713 returnStatus =
false;
1717 const QVector<QDomNode> sortedLayerNodes = depSorter.sortedLayerNodes();
1718 const int totalLayerCount = sortedLayerNodes.count();
1720 QVector<QDomNode> parallelLoading;
1721 QMap<QString, QgsDataProvider *> loadedProviders;
1726 profile.switchTask( tr(
"Load providers in parallel" ) );
1727 for (
const QDomNode &node : sortedLayerNodes )
1729 const QDomElement element = node.toElement();
1730 if ( element.attribute( QStringLiteral(
"embedded" ) ) != QLatin1String(
"1" ) )
1732 const QString layerId = node.namedItem( QStringLiteral(
"id" ) ).toElement().text();
1733 if ( !depSorter.isLayerDependent( layerId ) )
1735 const QDomNode mnl = element.namedItem( QStringLiteral(
"provider" ) );
1736 const QDomElement mne = mnl.toElement();
1737 const QString provider = mne.text();
1741 parallelLoading.append( node );
1750 if ( !parallelLoading.isEmpty() )
1751 preloadProviders( parallelLoading, context, loadedProviders, projectFlagsToLayerReadFlags(
flags, mFlags ), sortedLayerNodes.count() );
1754 int i = loadedProviders.count();
1755 for (
const QDomNode &node : std::as_const( sortedLayerNodes ) )
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 ) );
1762 profile.switchTask( name );
1763 if ( element.attribute( QStringLiteral(
"embedded" ) ) == QLatin1String(
"1" ) )
1765 createEmbeddedLayer( element.attribute( QStringLiteral(
"id" ) ),
readPath( element.attribute( QStringLiteral(
"project" ) ) ), brokenNodes,
true,
flags );
1773 QString layerId = element.namedItem( QStringLiteral(
"id" ) ).toElement().text();
1775 if ( !addLayer( element, brokenNodes, context,
flags, loadedProviders.take( layerId ) ) )
1777 returnStatus =
false;
1780 if ( !messages.isEmpty() )
1789 return returnStatus;
1792bool QgsProject::addLayer(
const QDomElement &layerElem,
1793 QList<QDomNode> &brokenNodes,
1800 const QString type = layerElem.attribute( QStringLiteral(
"type" ) );
1802 std::unique_ptr<QgsMapLayer>
mapLayer;
1810 QgsDebugError( QStringLiteral(
"Unknown layer type \"%1\"" ).arg( type ) );
1814 switch ( layerType )
1817 mapLayer = std::make_unique<QgsVectorLayer>();
1821 mapLayer = std::make_unique<QgsRasterLayer>();
1825 mapLayer = std::make_unique<QgsMeshLayer>();
1829 mapLayer = std::make_unique<QgsVectorTileLayer>();
1833 mapLayer = std::make_unique<QgsPointCloudLayer>();
1837 mapLayer = std::make_unique<QgsTiledSceneLayer>();
1842 const QString
typeName = layerElem.attribute( QStringLiteral(
"name" ) );
1850 mapLayer = std::make_unique<QgsAnnotationLayer>( QString(), options );
1857 mapLayer = std::make_unique<QgsGroupLayer>( QString(), options );
1864 QgsDebugError( QStringLiteral(
"Unable to create layer" ) );
1872 const QString layerId { layerElem.namedItem( QStringLiteral(
"id" ) ).toElement().text() };
1873 Q_ASSERT( ! layerId.isEmpty() );
1879 profile.switchTask( tr(
"Load layer source" ) );
1886 if ( vl->dataProvider() )
1893 profile.switchTask( tr(
"Add layer to project" ) );
1894 QList<QgsMapLayer *> newLayers;
1906 vLayer->joinBuffer()->resolveReferences(
this );
1915 brokenNodes.push_back( layerElem );
1918 const bool wasEditable = layerElem.attribute( QStringLiteral(
"editable" ), QStringLiteral(
"0" ) ).toInt();
1930 if ( ! layerWasStored )
1935 return layerIsValid;
1942 mFile.setFileName( filename );
1943 mCachedHomePath.clear();
1944 mProjectScope.reset();
1953 const QString filename = mFile.fileName();
1958 QTemporaryFile inDevice;
1959 if ( !inDevice.open() )
1961 setError( tr(
"Unable to open %1" ).arg( inDevice.fileName() ) );
1967 if ( !storage->readProject( filename, &inDevice, context ) )
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();
1976 returnValue = unzip( inDevice.fileName(),
flags );
1982 returnValue = unzip( mFile.fileName(),
flags );
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() )
1991 auto archive = std::make_unique<QgsArchive>();
1992 if ( archive->unzip( attachmentsZip ) )
1994 releaseHandlesToProjectArchive();
1995 mArchive = std::move( archive );
1998 returnValue = readProjectFile( mFile.fileName(),
flags );
2004 mFile.setFileName( filename );
2005 mCachedHomePath.clear();
2006 mProjectScope.reset();
2011 mTranslator.reset(
nullptr );
2023 ScopedIntIncrementor snapSignalBlock( &mBlockSnappingUpdates );
2025 QFile projectFile( filename );
2033 if ( QFile( QStringLiteral(
"%1/%2.qm" ).arg( QFileInfo( projectFile.fileName() ).absolutePath(), localeFileName ) ).exists() )
2035 mTranslator.reset(
new QTranslator() );
2036 ( void )mTranslator->load( localeFileName, QFileInfo( projectFile.fileName() ).absolutePath() );
2039 profile.switchTask( tr(
"Reading project file" ) );
2040 auto doc = std::make_unique<QDomDocument>( QStringLiteral(
"qgis" ) );
2042 if ( !projectFile.open( QIODevice::ReadOnly | QIODevice::Text ) )
2044 projectFile.close();
2046 setError( tr(
"Unable to open %1" ).arg( projectFile.fileName() ) );
2051 QTextStream textStream( &projectFile );
2052#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
2053 textStream.setCodec(
"UTF-8" );
2055 QString projectString = textStream.readAll();
2056 projectFile.close();
2058 for (
int i = 0; i < 32; i++ )
2060 if ( i == 9 || i == 10 || i == 13 )
2064 projectString.replace( QChar( i ), QStringLiteral(
"%1%2%1" ).arg(
FONTMARKER_CHR_FIX, QString::number( i ) ) );
2070 if ( !doc->setContent( projectString, &errorMsg, &line, &column ) )
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 );
2075 setError( errorString );
2080 projectFile.close();
2088 profile.switchTask( tr(
"Updating project file" ) );
2089 if ( thisVersion > fileVersion )
2091 const bool isOlderMajorVersion = fileVersion.
majorVersion() < thisVersion.majorVersion();
2093 if ( isOlderMajorVersion )
2096 "version of qgis (saved in " + fileVersion.
text() +
2098 "). Problems may occur." );
2109 projectFile.updateRevision( thisVersion );
2111 else if ( fileVersion > thisVersion )
2114 "version of qgis (saved in " + fileVersion.
text() +
2116 "). Problems may occur." );
2122 profile.switchTask( tr(
"Creating auxiliary storage" ) );
2123 const QString
fileName = mFile.fileName();
2130 std::unique_ptr<QgsAuxiliaryStorage> aStorage = std::move( mAuxiliaryStorage );
2131 std::unique_ptr<QgsArchive> archive = std::move( mArchive );
2135 mBlockChangeSignalsDuringClear =
true;
2137 mBlockChangeSignalsDuringClear =
false;
2142 releaseHandlesToProjectArchive();
2144 mAuxiliaryStorage = std::move( aStorage );
2145 mArchive = std::move( archive );
2148 mCachedHomePath.clear();
2149 mProjectScope.reset();
2150 mSaveVersion = fileVersion;
2153 profile.switchTask( tr(
"Reading properties" ) );
2162 dump_( mProperties );
2167 _getTitle( *doc, oldTitle );
2169 readProjectFileMetadata( *doc, mSaveUser, mSaveUserFull, mSaveDateTime );
2171 const QDomNodeList homePathNl = doc->elementsByTagName( QStringLiteral(
"homePath" ) );
2172 if ( homePathNl.count() > 0 )
2174 const QDomElement homePathElement = homePathNl.at( 0 ).toElement();
2175 const QString
homePath = homePathElement.attribute( QStringLiteral(
"path" ) );
2185 readNumEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/CanvasColorGreenPart" ), 255 ),
2186 readNumEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/CanvasColorBluePart" ), 255 ) );
2189 readNumEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/SelectionColorGreenPart" ), 255 ),
2190 readNumEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/SelectionColorBluePart" ), 255 ),
2191 readNumEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/SelectionColorAlphaPart" ), 255 ) );
2195 const QString distanceUnitString =
readEntry( QStringLiteral(
"Measurement" ), QStringLiteral(
"/DistanceUnits" ), QString() );
2196 if ( !distanceUnitString.isEmpty() )
2199 const QString areaUnitString =
readEntry( QStringLiteral(
"Measurement" ), QStringLiteral(
"/AreaUnits" ), QString() );
2200 if ( !areaUnitString.isEmpty() )
2209 if (
readNumEntry( QStringLiteral(
"SpatialRefSys" ), QStringLiteral(
"/ProjectionsEnabled" ), 0 ) )
2212 const QDomNode srsNode = doc->documentElement().namedItem( QStringLiteral(
"projectCrs" ) );
2213 if ( !srsNode.isNull() )
2215 projectCrs.
readXml( srsNode );
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" ) );
2225 const bool isUserAuthId = authid.startsWith( QLatin1String(
"USER:" ), Qt::CaseInsensitive );
2226 if ( !authid.isEmpty() && !isUserAuthId )
2230 if ( !projectCrs.
isValid() && currentCRS >= 0 )
2236 if ( !projCrsString.isEmpty() && ( authid.isEmpty() || isUserAuthId ) && ( !projectCrs.
isValid() || projectCrs.
toProj() != projCrsString ) )
2253 const QDomNode verticalCrsNode = doc->documentElement().namedItem( QStringLiteral(
"verticalCrs" ) );
2254 if ( !verticalCrsNode.isNull() )
2262 QStringList datumErrors;
2263 if ( !mTransformContext.
readXml( doc->documentElement(), context, datumErrors ) && !datumErrors.empty() )
2270 const QDomNode elevationShadingNode = doc->documentElement().namedItem( QStringLiteral(
"elevation-shading-renderer" ) );
2271 if ( !elevationShadingNode.isNull() )
2273 mElevationShadingRenderer.
readXml( elevationShadingNode.toElement(), context );
2280 const QStringList variableNames =
readListEntry( QStringLiteral(
"Variables" ), QStringLiteral(
"/variableNames" ) );
2281 const QStringList variableValues =
readListEntry( QStringLiteral(
"Variables" ), QStringLiteral(
"/variableValues" ) );
2283 mCustomVariables.clear();
2284 if ( variableNames.length() == variableValues.length() )
2286 for (
int i = 0; i < variableNames.length(); ++i )
2288 mCustomVariables.insert( variableNames.at( i ), variableValues.at( i ) );
2293 QgsMessageLog::logMessage( tr(
"Project Variables Invalid" ), tr(
"The project contains invalid variable settings." ) );
2301 QDomElement element = doc->documentElement().firstChildElement( QStringLiteral(
"projectMetadata" ) );
2303 if ( !element.isNull() )
2312 if ( mMetadata.
title().isEmpty() && !oldTitle.isEmpty() )
2320 element = doc->documentElement().firstChildElement( QStringLiteral(
"transaction" ) );
2321 if ( !element.isNull() )
2328 element = doc->documentElement().firstChildElement( QStringLiteral(
"autotransaction" ) );
2329 if ( ! element.isNull() )
2331 mTransactionMode =
static_cast<Qgis::TransactionMode>( element.attribute( QStringLiteral(
"active" ), QStringLiteral(
"0" ) ).toInt() );
2336 profile.switchTask( tr(
"Loading layer tree" ) );
2339 QDomElement layerTreeElem = doc->documentElement().firstChildElement( QStringLiteral(
"layer-tree-group" ) );
2340 if ( !layerTreeElem.isNull() )
2352 mLayerTreeRegistryBridge->
setEnabled(
false );
2355 profile.switchTask( tr(
"Reading map layers" ) );
2357 loadProjectFlags( doc.get() );
2359 QList<QDomNode> brokenNodes;
2360 const bool clean = _getMapLayers( *doc, brokenNodes,
flags );
2365 QgsDebugError( QStringLiteral(
"Unable to get map layers from project file." ) );
2367 if ( !brokenNodes.isEmpty() )
2369 QgsDebugError(
"there are " + QString::number( brokenNodes.size() ) +
" broken layers" );
2377 mMainAnnotationLayer->
readLayerXml( doc->documentElement().firstChildElement( QStringLiteral(
"main-annotation-layer" ) ), context );
2381 profile.switchTask( tr(
"Loading embedded layers" ) );
2382 loadEmbeddedNodes( mRootGroup,
flags );
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 )
2390 it.value()->resolveReferences(
this );
2394 mLayerTreeRegistryBridge->
setEnabled(
true );
2397 profile.switchTask( tr(
"Resolving references" ) );
2408 if ( !layerTreeElem.isNull() )
2414 const QDomElement layerTreeCanvasElem = doc->documentElement().firstChildElement( QStringLiteral(
"layer-tree-canvas" ) );
2415 if ( !layerTreeCanvasElem.isNull( ) )
2423 const QStringList requiredLayerIds =
readListEntry( QStringLiteral(
"RequiredLayers" ), QStringLiteral(
"Layers" ) );
2424 for (
const QString &layerId : requiredLayerIds )
2431 const QStringList disabledLayerIds =
readListEntry( QStringLiteral(
"Identify" ), QStringLiteral(
"/disabledLayers" ) );
2432 for (
const QString &layerId : disabledLayerIds )
2445 QString styleName =
readEntry( QStringLiteral(
"DefaultStyles" ), QStringLiteral(
"/Marker" ) );
2446 if ( !styleName.isEmpty() )
2451 styleName =
readEntry( QStringLiteral(
"DefaultStyles" ), QStringLiteral(
"/Line" ) );
2452 if ( !styleName.isEmpty() )
2457 styleName =
readEntry( QStringLiteral(
"DefaultStyles" ), QStringLiteral(
"/Fill" ) );
2458 if ( !styleName.isEmpty() )
2463 styleName =
readEntry( QStringLiteral(
"DefaultStyles" ), QStringLiteral(
"/ColorRamp" ) );
2464 if ( !styleName.isEmpty() )
2474 double opacity = 1.0;
2477 double alpha =
readDoubleEntry( QStringLiteral(
"DefaultStyles" ), QStringLiteral(
"/AlphaInt" ), 255, &ok );
2479 opacity = alpha / 255.0;
2480 double newOpacity =
readDoubleEntry( QStringLiteral(
"DefaultStyles" ), QStringLiteral(
"/Opacity" ), 1.0, &ok );
2482 opacity = newOpacity;
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" ) );
2500 profile.switchTask( tr(
"Storing original layer properties" ) );
2506 profile.switchTask( tr(
"Loading map themes" ) );
2509 mMapThemeCollection->readXml( *doc );
2511 profile.switchTask( tr(
"Loading label settings" ) );
2512 mLabelingEngineSettings->readSettingsFromProject(
this );
2514 const QDomElement labelEngineSettingsElement = doc->documentElement().firstChildElement( QStringLiteral(
"labelEngineSettings" ) );
2515 mLabelingEngineSettings->readXml( labelEngineSettingsElement, context );
2517 mLabelingEngineSettings->resolveReferences(
this );
2521 profile.switchTask( tr(
"Loading annotations" ) );
2524 mAnnotationManager->readXml( doc->documentElement(), context );
2528 mAnnotationManager->readXmlAndUpgradeToAnnotationLayerItems( doc->documentElement(), context, mMainAnnotationLayer, mTransformContext );
2532 profile.switchTask( tr(
"Loading layouts" ) );
2533 mLayoutManager->readXml( doc->documentElement(), *doc );
2538 profile.switchTask( tr(
"Loading 3D Views" ) );
2539 m3DViewsManager->readXml( doc->documentElement(), *doc );
2542 profile.switchTask( tr(
"Loading bookmarks" ) );
2543 mBookmarkManager->
readXml( doc->documentElement(), *doc );
2545 profile.switchTask( tr(
"Loading sensors" ) );
2546 mSensorManager->
readXml( doc->documentElement(), *doc );
2549 QMap<QString, QgsMapLayer *> existingMaps =
mapLayers();
2550 for ( QMap<QString, QgsMapLayer *>::iterator it = existingMaps.begin(); it != existingMaps.end(); ++it )
2552 it.value()->setDependencies( it.value()->dependencies() );
2555 profile.switchTask( tr(
"Loading snapping settings" ) );
2559 profile.switchTask( tr(
"Loading view settings" ) );
2562 const QStringList scales =
readListEntry( QStringLiteral(
"Scales" ), QStringLiteral(
"/ScalesList" ) );
2563 QVector<double> res;
2564 for (
const QString &scale : scales )
2566 const QStringList parts = scale.split(
':' );
2567 if ( parts.size() != 2 )
2571 const double denominator = QLocale().toDouble( parts[1], &ok );
2578 const QDomElement viewSettingsElement = doc->documentElement().firstChildElement( QStringLiteral(
"ProjectViewSettings" ) );
2579 if ( !viewSettingsElement.isNull() )
2580 mViewSettings->
readXml( viewSettingsElement, context );
2583 profile.switchTask( tr(
"Loading style properties" ) );
2584 const QDomElement styleSettingsElement = doc->documentElement().firstChildElement( QStringLiteral(
"ProjectStyleSettings" ) );
2585 if ( !styleSettingsElement.isNull() )
2588 mStyleSettings->
readXml( styleSettingsElement, context,
flags );
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 );
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 );
2604 profile.switchTask( tr(
"Loading display settings" ) );
2606 const QDomElement displaySettingsElement = doc->documentElement().firstChildElement( QStringLiteral(
"ProjectDisplaySettings" ) );
2607 if ( !displaySettingsElement.isNull() )
2608 mDisplaySettings->
readXml( displaySettingsElement, context );
2611 profile.switchTask( tr(
"Loading GPS settings" ) );
2613 const QDomElement gpsSettingsElement = doc->documentElement().firstChildElement( QStringLiteral(
"ProjectGpsSettings" ) );
2614 if ( !gpsSettingsElement.isNull() )
2615 mGpsSettings->
readXml( gpsSettingsElement, context );
2619 profile.switchTask( tr(
"Updating variables" ) );
2621 profile.switchTask( tr(
"Updating CRS" ) );
2625 if ( mCrs3D != oldCrs3D )
2630 profile.switchTask( tr(
"Reading external settings" ) );
2634 profile.switchTask( tr(
"Updating interface" ) );
2636 snapSignalBlock.release();
2637 if ( !mBlockSnappingUpdates )
2648 QgsDebugMsgLevel( QStringLiteral(
"Project save user: %1" ).arg( mSaveUser ), 2 );
2649 QgsDebugMsgLevel( QStringLiteral(
"Project save user: %1" ).arg( mSaveUserFull ), 2 );
2658 const QString newFileName( QStringLiteral(
"%1/%2.qgs" ).arg( QFileInfo( projectFile.fileName() ).absolutePath(), localeFileName ) );
2673 const QMap<QString, QgsMapLayer *> loadedLayers =
mapLayers();
2674 for (
auto it = loadedLayers.constBegin(); it != loadedLayers.constEnd(); ++it )
2676 if ( it.value()->isValid() && it.value()->customProperty( QStringLiteral(
"_layer_was_editable" ) ).toBool() )
2678 if (
QgsVectorLayer *vl = qobject_cast< QgsVectorLayer * >( it.value() ) )
2680 it.value()->removeCustomProperty( QStringLiteral(
"_layer_was_editable" ) );
2692 const auto constChildren = group->
children();
2698 if ( childGroup->
customProperty( QStringLiteral(
"embedded" ) ).toInt() )
2701 const QString projectPath =
readPath( childGroup->
customProperty( QStringLiteral(
"embedded_project" ) ).toString() );
2702 childGroup->
setCustomProperty( QStringLiteral(
"embedded_project" ), projectPath );
2706 QList<QgsLayerTreeNode *> clonedChildren;
2707 const QList<QgsLayerTreeNode *> constChildren = newGroup->
children();
2708 clonedChildren.reserve( constChildren.size() );
2710 clonedChildren << newGroupChild->clone();
2718 loadEmbeddedNodes( childGroup,
flags );
2723 if ( child->customProperty( QStringLiteral(
"embedded" ) ).toInt() )
2725 QList<QDomNode> brokenNodes;
2728 valid = valid &&
false;
2743 return mCustomVariables;
2750 if ( variables == mCustomVariables )
2754 QStringList variableNames;
2755 QStringList variableValues;
2757 QVariantMap::const_iterator it = variables.constBegin();
2758 for ( ; it != variables.constEnd(); ++it )
2760 variableNames << it.key();
2761 variableValues << it.value().toString();
2764 writeEntry( QStringLiteral(
"Variables" ), QStringLiteral(
"/variableNames" ), variableNames );
2765 writeEntry( QStringLiteral(
"Variables" ), QStringLiteral(
"/variableValues" ), variableValues );
2767 mCustomVariables = variables;
2768 mProjectScope.reset();
2777 *mLabelingEngineSettings = settings;
2785 return *mLabelingEngineSettings;
2792 mProjectScope.reset();
2793 return mLayerStore.get();
2800 return mLayerStore.get();
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 )
2823 list.reserve(
layers.size() );
2828 list << layer->id();
2831 writeEntry( QStringLiteral(
"Digitizing" ), QStringLiteral(
"/AvoidIntersectionsList" ), list );
2853 if ( mProjectScope )
2855 auto projectScope = std::make_unique< QgsExpressionContextScope >( *mProjectScope );
2862 projectScope->addFunction( QStringLiteral(
"sensor_data" ),
new GetSensorData(
sensorManager()->sensorsData() ) );
2864 return projectScope.release();
2867 mProjectScope = std::make_unique< QgsExpressionContextScope >( QObject::tr(
"Project" ) );
2871 QVariantMap::const_iterator it = vars.constBegin();
2873 for ( ; it != vars.constEnd(); ++it )
2875 mProjectScope->setVariable( it.key(), it.value(),
true );
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();
2920 QVariantMap keywords;
2922 for (
auto it = metadataKeywords.constBegin(); it != metadataKeywords.constEnd(); ++it )
2924 keywords.insert( it.key(), it.value() );
2929 QVariantList layersIds;
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 )
2936 layersIds << it.value()->id();
2942 mProjectScope->addFunction( QStringLiteral(
"project_color" ),
new GetNamedProjectColor(
this ) );
2943 mProjectScope->addFunction( QStringLiteral(
"project_color_object" ),
new GetNamedProjectColorObject(
this ) );
2948void QgsProject::onMapLayersAdded(
const QList<QgsMapLayer *> &layers )
2952 const QMap<QString, QgsMapLayer *> existingMaps =
mapLayers();
2954 const auto constLayers =
layers;
2957 if ( ! layer->isValid() )
2960 if (
QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer ) )
2963 if ( vlayer->dataProvider() )
2971 for ( QMap<QString, QgsMapLayer *>::const_iterator it = existingMaps.cbegin(); it != existingMaps.cend(); ++it )
2973 const QSet<QgsMapLayerDependency> deps = it.value()->dependencies();
2974 if ( deps.contains( layer->id() ) )
2977 it.value()->setDependencies( deps );
2982 updateTransactionGroups();
2988void QgsProject::onMapLayersRemoved(
const QList<QgsMapLayer *> &layers )
2997 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
3005void QgsProject::cleanTransactionGroups(
bool force )
3009 bool changed =
false;
3010 for ( QMap< QPair< QString, QString>,
QgsTransactionGroup *>::Iterator tg = mTransactionGroups.begin(); tg != mTransactionGroups.end(); )
3012 if ( tg.value()->isEmpty() || force )
3015 tg = mTransactionGroups.erase( tg );
3027void QgsProject::updateTransactionGroups()
3031 mEditBufferGroup.
clear();
3033 switch ( mTransactionMode )
3037 cleanTransactionGroups(
true );
3042 cleanTransactionGroups(
true );
3045 cleanTransactionGroups(
false );
3049 bool tgChanged =
false;
3050 const auto constLayers =
mapLayers().values();
3053 if ( ! layer->isValid() )
3056 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
3060 switch ( mTransactionMode )
3077 mTransactionGroups.insert( qMakePair( key, connString ), tg );
3087 mEditBufferGroup.
addLayer( vlayer );
3103 context.setProjectTranslator(
this );
3105 QList<QDomNode> brokenNodes;
3106 if ( addLayer( layerNode.toElement(), brokenNodes, context ) )
3110 const QVector<QgsVectorLayer *> vectorLayers = layers<QgsVectorLayer *>();
3114 layer->resolveReferences(
this );
3116 if ( layer->isValid() && layer->customProperty( QStringLiteral(
"_layer_was_editable" ) ).toBool() )
3118 layer->startEditing();
3119 layer->removeCustomProperty( QStringLiteral(
"_layer_was_editable" ) );
3131 mFile.setFileName( filename );
3132 mCachedHomePath.clear();
3140 mProjectScope.reset();
3146 const QString storageFilePath { storage->filePath( mFile.fileName() ) };
3147 if ( storageFilePath.isEmpty() )
3153 const QString tempPath = QStandardPaths::standardLocations( QStandardPaths::TempLocation ).at( 0 );
3154 const QString tmpZipFilename( tempPath + QDir::separator() + QUuid::createUuid().toString() );
3156 if ( !zip( tmpZipFilename ) )
3159 QFile tmpZipFile( tmpZipFilename );
3160 if ( !tmpZipFile.open( QIODevice::ReadOnly ) )
3162 setError( tr(
"Unable to read file %1" ).arg( tmpZipFilename ) );
3167 if ( !storage->writeProject( mFile.fileName(), &tmpZipFile, context ) )
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();
3178 QFile::remove( tmpZipFilename );
3185 return zip( mFile.fileName() );
3191 const bool asOk = saveAuxiliaryStorage();
3192 const bool writeOk = writeProjectFile( mFile.fileName() );
3193 bool attachmentsOk =
true;
3194 if ( !mArchive->files().isEmpty() )
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 );
3202 if ( ( !asOk || !attachmentsOk ) && writeOk )
3204 QStringList errorMessage;
3207 const QString err = mAuxiliaryStorage->errorString();
3208 errorMessage.append( tr(
"Unable to save auxiliary storage ('%1')" ).arg( err ) );
3210 if ( !attachmentsOk )
3212 errorMessage.append( tr(
"Unable to save attachments archive" ) );
3214 setError( errorMessage.join(
'\n' ) );
3217 return asOk && writeOk && attachmentsOk;
3221bool QgsProject::writeProjectFile(
const QString &filename )
3225 QFile projectFile( filename );
3231 const QFileInfo myFileInfo( projectFile );
3232 if ( myFileInfo.exists() && !myFileInfo.isWritable() )
3234 setError( tr(
"%1 is not writable. Please adjust permissions (if possible) and try again." )
3235 .arg( projectFile.fileName() ) );
3243 QDomImplementation::setInvalidDataPolicy( QDomImplementation::DropInvalidChars );
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 );
3250 QDomElement qgisNode = doc->createElement( QStringLiteral(
"qgis" ) );
3251 qgisNode.setAttribute( QStringLiteral(
"projectname" ),
title() );
3252 qgisNode.setAttribute( QStringLiteral(
"version" ),
Qgis::version() );
3254 if ( !mSettings.
value( QStringLiteral(
"projects/anonymize_saved_projects" ),
false,
QgsSettings::Core ).toBool() )
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 ) );
3268 mSaveUserFull.clear();
3269 mSaveDateTime = QDateTime();
3271 doc->appendChild( qgisNode );
3274 QDomElement homePathNode = doc->createElement( QStringLiteral(
"homePath" ) );
3275 homePathNode.setAttribute( QStringLiteral(
"path" ), mHomePath );
3276 qgisNode.appendChild( homePathNode );
3279 QDomElement titleNode = doc->createElement( QStringLiteral(
"title" ) );
3280 qgisNode.appendChild( titleNode );
3282 QDomElement transactionNode = doc->createElement( QStringLiteral(
"transaction" ) );
3283 transactionNode.setAttribute( QStringLiteral(
"mode" ),
qgsEnumValueToKey( mTransactionMode ) );
3284 qgisNode.appendChild( transactionNode );
3286 QDomElement flagsNode = doc->createElement( QStringLiteral(
"projectFlags" ) );
3288 qgisNode.appendChild( flagsNode );
3290 const QDomText titleText = doc->createTextNode(
title() );
3291 titleNode.appendChild( titleText );
3295 QDomElement srsNode = doc->createElement( QStringLiteral(
"projectCrs" ) );
3297 qgisNode.appendChild( srsNode );
3300 QDomElement verticalSrsNode = doc->createElement( QStringLiteral(
"verticalCrs" ) );
3301 mVerticalCrs.
writeXml( verticalSrsNode, *doc );
3302 qgisNode.appendChild( verticalSrsNode );
3305 QDomElement elevationShadingNode = doc->createElement( QStringLiteral(
"elevation-shading-renderer" ) );
3306 mElevationShadingRenderer.
writeXml( elevationShadingNode, context );
3307 qgisNode.appendChild( elevationShadingNode );
3314 clonedRoot->
writeXml( qgisNode, context );
3318 writeEntry( QStringLiteral(
"Digitizing" ), QStringLiteral(
"/AvoidIntersectionsMode" ),
static_cast<int>( mAvoidIntersectionsMode ) );
3326 QDomElement annotationLayerNode = doc->createElement( QStringLiteral(
"main-annotation-layer" ) );
3327 mMainAnnotationLayer->
writeLayerXml( annotationLayerNode, *doc, context );
3328 qgisNode.appendChild( annotationLayerNode );
3332 QDomElement projectLayersNode = doc->createElement( QStringLiteral(
"projectlayers" ) );
3334 QMap<QString, QgsMapLayer *>::ConstIterator li =
layers.constBegin();
3335 while ( li !=
layers.end() )
3341 const QHash< QString, QPair< QString, bool> >::const_iterator emIt = mEmbeddedLayers.constFind( ml->
id() );
3342 if ( emIt == mEmbeddedLayers.constEnd() )
3344 QDomElement maplayerElem;
3350 maplayerElem = doc->createElement( QStringLiteral(
"maplayer" ) );
3354 maplayerElem.setAttribute( QStringLiteral(
"editable" ), QStringLiteral(
"1" ) );
3358 QDomDocument document;
3361 maplayerElem = document.firstChildElement();
3365 QgsDebugError( QStringLiteral(
"Could not restore layer properties for layer %1" ).arg( ml->
id() ) );
3371 projectLayersNode.appendChild( maplayerElem );
3377 if ( emIt.value().second )
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 );
3390 qgisNode.appendChild( projectLayersNode );
3392 QDomElement layerOrderNode = doc->createElement( QStringLiteral(
"layerorder" ) );
3394 for (
QgsMapLayer *layer : constCustomLayerOrder )
3396 QDomElement mapLayerElem = doc->createElement( QStringLiteral(
"layer" ) );
3397 mapLayerElem.setAttribute( QStringLiteral(
"id" ), layer->id() );
3398 layerOrderNode.appendChild( mapLayerElem );
3400 qgisNode.appendChild( layerOrderNode );
3402 mLabelingEngineSettings->writeSettingsToProject(
this );
3404 QDomElement labelEngineSettingsElement = doc->createElement( QStringLiteral(
"labelEngineSettings" ) );
3405 mLabelingEngineSettings->writeXml( *doc, labelEngineSettingsElement, context );
3406 qgisNode.appendChild( labelEngineSettingsElement );
3409 writeEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/CanvasColorRedPart" ), mBackgroundColor.red() );
3410 writeEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/CanvasColorGreenPart" ), mBackgroundColor.green() );
3411 writeEntry( QStringLiteral(
"Gui" ), QStringLiteral(
"/CanvasColorBluePart" ), mBackgroundColor.blue() );
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() );
3423 dump_( mProperties );
3426 QgsDebugMsgLevel( QStringLiteral(
"there are %1 property scopes" ).arg(
static_cast<int>( mProperties.
count() ) ), 2 );
3431 mProperties.
writeXml( QStringLiteral(
"properties" ), qgisNode, *doc );
3434 QDomElement ddElem = doc->createElement( QStringLiteral(
"dataDefinedServerProperties" ) );
3435 mDataDefinedServerProperties.
writeXml( ddElem, dataDefinedServerPropertyDefinitions() );
3436 qgisNode.appendChild( ddElem );
3438 mMapThemeCollection->writeXml( *doc );
3440 mTransformContext.
writeXml( qgisNode, context );
3442 QDomElement metadataElem = doc->createElement( QStringLiteral(
"projectMetadata" ) );
3444 qgisNode.appendChild( metadataElem );
3447 const QDomElement annotationsElem = mAnnotationManager->writeXml( *doc, context );
3448 qgisNode.appendChild( annotationsElem );
3452 const QDomElement layoutElem = mLayoutManager->writeXml( *doc );
3453 qgisNode.appendChild( layoutElem );
3457 const QDomElement views3DElem = m3DViewsManager->writeXml( *doc );
3458 qgisNode.appendChild( views3DElem );
3462 const QDomElement bookmarkElem = mBookmarkManager->
writeXml( *doc );
3463 qgisNode.appendChild( bookmarkElem );
3467 const QDomElement sensorElem = mSensorManager->
writeXml( *doc );
3468 qgisNode.appendChild( sensorElem );
3472 const QDomElement viewSettingsElem = mViewSettings->
writeXml( *doc, context );
3473 qgisNode.appendChild( viewSettingsElem );
3477 const QDomElement styleSettingsElem = mStyleSettings->
writeXml( *doc, context );
3478 qgisNode.appendChild( styleSettingsElem );
3482 const QDomElement timeSettingsElement = mTimeSettings->
writeXml( *doc, context );
3483 qgisNode.appendChild( timeSettingsElement );
3487 const QDomElement elevationPropertiesElement = mElevationProperties->
writeXml( *doc, context );
3488 qgisNode.appendChild( elevationPropertiesElement );
3492 const QDomElement displaySettingsElem = mDisplaySettings->
writeXml( *doc, context );
3493 qgisNode.appendChild( displaySettingsElem );
3497 const QDomElement gpsSettingsElem = mGpsSettings->
writeXml( *doc, context );
3498 qgisNode.appendChild( gpsSettingsElem );
3507 QFile backupFile( QStringLiteral(
"%1~" ).arg( filename ) );
3509 ok &= backupFile.open( QIODevice::WriteOnly | QIODevice::Truncate );
3510 ok &= projectFile.open( QIODevice::ReadOnly );
3513 while ( ok && !projectFile.atEnd() )
3515 ba = projectFile.read( 10240 );
3516 ok &= backupFile.write( ba ) == ba.size();
3519 projectFile.close();
3524 setError( tr(
"Unable to create backup file %1" ).arg( backupFile.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 );
3533 if ( !projectFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
3535 projectFile.close();
3538 setError( tr(
"Unable to save to file %1" ).arg( projectFile.fileName() ) );
3542 QTemporaryFile tempFile;
3543 bool ok = tempFile.open();
3546 QTextStream projectFileStream( &tempFile );
3547 doc->save( projectFileStream, 2 );
3548 ok &= projectFileStream.pos() > -1;
3550 ok &= tempFile.seek( 0 );
3553 while ( ok && !tempFile.atEnd() )
3555 ba = tempFile.read( 10240 );
3556 ok &= projectFile.write( ba ) == ba.size();
3559 ok &= projectFile.error() == QFile::NoError;
3561 projectFile.close();
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() ) );
3585 bool propertiesModified;
3586 const bool success =
addKey_( scope, key, &mProperties, value, propertiesModified );
3588 if ( propertiesModified )
3598 bool propertiesModified;
3599 const bool success =
addKey_( scope, key, &mProperties, value, propertiesModified );
3601 if ( propertiesModified )
3611 bool propertiesModified;
3612 const bool success =
addKey_( scope, key, &mProperties, value, propertiesModified );
3614 if ( propertiesModified )
3624 bool propertiesModified;
3625 const bool success =
addKey_( scope, key, &mProperties, value, propertiesModified );
3627 if ( propertiesModified )
3637 bool propertiesModified;
3638 const bool success =
addKey_( scope, key, &mProperties, value, propertiesModified );
3640 if ( propertiesModified )
3648 const QStringList &def,
3660 value =
property->value();
3662 const bool valid = QMetaType::Type::QStringList == value.userType();
3668 return value.toStringList();
3691 value =
property->value();
3693 const bool valid = value.canConvert( QMetaType::Type::QString );
3698 return value.toString();
3717 value =
property->value();
3720 const bool valid = value.canConvert( QMetaType::Type::Int );
3729 return value.toInt();
3744 const QVariant value =
property->value();
3746 const bool valid = value.canConvert( QMetaType::Type::Double );
3751 return value.toDouble();
3768 const QVariant value =
property->value();
3770 const bool valid = value.canConvert( QMetaType::Type::Bool );
3775 return value.toBool();
3787 if (
findKey_( scope, key, mProperties ) )
3793 return !
findKey_( scope, key, mProperties );
3802 QStringList entries;
3804 if ( foundProperty )
3821 QStringList entries;
3823 if ( foundProperty )
3838 dump_( mProperties );
3858 filePath = storage->filePath( mFile.fileName() );
3885void QgsProject::setError(
const QString &errorMessage )
3889 mErrorMessage = errorMessage;
3896 return mErrorMessage;
3899void QgsProject::clearError()
3903 setError( QString() );
3910 delete mBadLayerHandler;
3911 mBadLayerHandler = handler;
3918 const QHash< QString, QPair< QString, bool > >::const_iterator it = mEmbeddedLayers.find(
id );
3919 if ( it == mEmbeddedLayers.constEnd() )
3923 return it.value().first;
3933 static QString sPrevProjectFilePath;
3934 static QDateTime sPrevProjectFileTimestamp;
3935 static QDomDocument sProjectDocument;
3937 QString qgsProjectFile = projectFilePath;
3939 if ( projectFilePath.endsWith( QLatin1String(
".qgz" ), Qt::CaseInsensitive ) )
3941 archive.
unzip( projectFilePath );
3945 const QDateTime projectFileTimestamp = QFileInfo( projectFilePath ).lastModified();
3947 if ( projectFilePath != sPrevProjectFilePath || projectFileTimestamp != sPrevProjectFileTimestamp )
3949 sPrevProjectFilePath.clear();
3951 QFile projectFile( qgsProjectFile );
3952 if ( !projectFile.open( QIODevice::ReadOnly ) )
3957 if ( !sProjectDocument.setContent( &projectFile ) )
3962 sPrevProjectFilePath = projectFilePath;
3963 sPrevProjectFileTimestamp = projectFileTimestamp;
3967 bool useAbsolutePaths =
true;
3969 const QDomElement propertiesElem = sProjectDocument.documentElement().firstChildElement( QStringLiteral(
"properties" ) );
3970 if ( !propertiesElem.isNull() )
3972 QDomElement e = propertiesElem.firstChildElement( QStringLiteral(
"Paths" ) );
3975 e = propertiesElem.firstChildElement( QStringLiteral(
"properties" ) );
3976 while ( !e.isNull() && e.attribute( QStringLiteral(
"name" ) ) != QStringLiteral(
"Paths" ) )
3977 e = e.nextSiblingElement( QStringLiteral(
"properties" ) );
3979 e = e.firstChildElement( QStringLiteral(
"properties" ) );
3980 while ( !e.isNull() && e.attribute( QStringLiteral(
"name" ) ) != QStringLiteral(
"Absolute" ) )
3981 e = e.nextSiblingElement( QStringLiteral(
"properties" ) );
3985 e = e.firstChildElement( QStringLiteral(
"Absolute" ) );
3990 useAbsolutePaths = e.text().compare( QLatin1String(
"true" ), Qt::CaseInsensitive ) == 0;
3995 if ( !useAbsolutePaths )
4000 const QDomElement projectLayersElem = sProjectDocument.documentElement().firstChildElement( QStringLiteral(
"projectlayers" ) );
4001 if ( projectLayersElem.isNull() )
4006 QDomElement mapLayerElem = projectLayersElem.firstChildElement( QStringLiteral(
"maplayer" ) );
4007 while ( ! mapLayerElem.isNull() )
4010 const QString
id = mapLayerElem.firstChildElement( QStringLiteral(
"id" ) ).text();
4011 if (
id == layerId )
4014 if ( mapLayerElem.attribute( QStringLiteral(
"embedded" ) ) == QLatin1String(
"1" ) )
4019 mEmbeddedLayers.insert( layerId, qMakePair( projectFilePath, saveFlag ) );
4021 if ( addLayer( mapLayerElem, brokenNodes, embeddedContext,
flags ) )
4027 mEmbeddedLayers.remove( layerId );
4031 mapLayerElem = mapLayerElem.nextSiblingElement( QStringLiteral(
"maplayer" ) );
4041 QString qgsProjectFile = projectFilePath;
4043 if ( projectFilePath.endsWith( QLatin1String(
".qgz" ), Qt::CaseInsensitive ) )
4045 archive.
unzip( projectFilePath );
4050 QFile projectFile( qgsProjectFile );
4051 if ( !projectFile.open( QIODevice::ReadOnly ) )
4056 QDomDocument projectDocument;
4057 if ( !projectDocument.setContent( &projectFile ) )
4069 QDomElement layerTreeElem = projectDocument.documentElement().firstChildElement( QStringLiteral(
"layer-tree-group" ) );
4070 if ( !layerTreeElem.isNull() )
4080 if ( !group || group->
customProperty( QStringLiteral(
"embedded" ) ).toBool() )
4093 newGroup->
setCustomProperty( QStringLiteral(
"embedded_project" ), projectFilePath );
4096 mLayerTreeRegistryBridge->
setEnabled(
false );
4097 initializeEmbeddedSubtree( projectFilePath, newGroup,
flags );
4098 mLayerTreeRegistryBridge->
setEnabled(
true );
4101 const auto constFindLayerIds = newGroup->
findLayerIds();
4102 for (
const QString &layerId : constFindLayerIds )
4119 const auto constChildren = group->
children();
4123 child->setCustomProperty( QStringLiteral(
"embedded" ), 1 );
4132 QList<QDomNode> brokenNodes;
4156 writeEntry( QStringLiteral(
"Digitizing" ), QStringLiteral(
"/TopologicalEditing" ), ( enabled ? 1 : 0 ) );
4164 return readNumEntry( QStringLiteral(
"Digitizing" ), QStringLiteral(
"/TopologicalEditing" ), 0 );
4171 if ( mDistanceUnits == unit )
4174 mDistanceUnits = unit;
4183 if ( mAreaUnits == unit )
4196 if ( !mCachedHomePath.isEmpty() )
4197 return mCachedHomePath;
4201 if ( !mHomePath.isEmpty() )
4203 const QFileInfo homeInfo( mHomePath );
4204 if ( !homeInfo.isRelative() )
4206 mCachedHomePath = mHomePath;
4216 const QString storagePath { storage->filePath(
fileName() ) };
4217 if ( ! storagePath.isEmpty() && QFileInfo::exists( storagePath ) )
4219 mCachedHomePath = QFileInfo( storagePath ).path();
4220 return mCachedHomePath;
4224 mCachedHomePath = pfi.path();
4225 return mCachedHomePath;
4228 if ( !pfi.exists() )
4230 mCachedHomePath = mHomePath;
4234 if ( !mHomePath.isEmpty() )
4237 mCachedHomePath = QDir::cleanPath( pfi.path() +
'/' + mHomePath );
4241 mCachedHomePath = pfi.canonicalPath();
4243 return mCachedHomePath;
4258 return mRelationManager;
4265 return mLayoutManager.get();
4272 return mLayoutManager.get();
4279 return m3DViewsManager.get();
4286 return m3DViewsManager.get();
4293 return mBookmarkManager;
4300 return mBookmarkManager;
4307 return mSensorManager;
4314 return mSensorManager;
4321 return mViewSettings;
4328 return mViewSettings;
4335 return mStyleSettings;
4343 return mStyleSettings;
4350 return mTimeSettings;
4357 return mTimeSettings;
4364 return mElevationProperties;
4371 return mElevationProperties;
4378 return mDisplaySettings;
4385 return mDisplaySettings;
4392 return mGpsSettings;
4399 return mGpsSettings;
4413 return mMapThemeCollection.get();
4420 return mAnnotationManager.get();
4427 return mAnnotationManager.get();
4434 const QMap<QString, QgsMapLayer *> &projectLayers =
mapLayers();
4435 for ( QMap<QString, QgsMapLayer *>::const_iterator it = projectLayers.constBegin(); it != projectLayers.constEnd(); ++it )
4440 if (
layers.contains( it.value() ) )
4441 it.value()->setFlags( it.value()->flags() &
~QgsMapLayer::Identifiable );
4457 for (
const QString &layerId : layerIds )
4475 for ( QMap<QString, QgsMapLayer *>::const_iterator it =
layers.constBegin(); it !=
layers.constEnd(); ++it )
4509 updateTransactionGroups();
4516 return mTransactionMode;
4527 const auto constLayers =
mapLayers().values();
4530 if ( layer->isEditable() )
4532 QgsLogger::warning( tr(
"Transaction mode can be changed only if all layers are not editable." ) );
4538 updateTransactionGroups();
4547 return mTransactionGroups;
4560 return mLayerStore->count();
4567 return mLayerStore->validCount();
4575 return mLayerStore->mapLayer( layerId );
4582 return mLayerStore->mapLayersByName( layerName );
4589 QList<QgsMapLayer *>
layers;
4590 const auto constMapLayers { mLayerStore->mapLayers() };
4591 for (
const auto &l : constMapLayers )
4593 if ( ! l->serverProperties()->shortName().isEmpty() )
4595 if ( l->serverProperties()->shortName() == shortName )
4598 else if ( l->name() == shortName )
4611 auto archive = std::make_unique<QgsProjectArchive>();
4614 if ( !archive->unzip( filename ) )
4616 setError( tr(
"Unable to unzip file '%1'" ).arg( filename ) );
4621 if ( archive->projectFile().isEmpty() )
4623 setError( tr(
"Zip archive does not provide a project file" ) );
4628 releaseHandlesToProjectArchive();
4629 mArchive = std::move( archive );
4646 setError( tr(
"Cannot read unzipped qgs project file" ) + QStringLiteral(
": " ) +
error() );
4656bool QgsProject::zip(
const QString &filename )
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 ) );
4668 bool writeOk =
false;
4669 if ( qgsFile.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
4671 writeOk = writeProjectFile( qgsFile.fileName() );
4678 setError( tr(
"Unable to write temporary qgs file" ) );
4683 const QFileInfo info( qgsFile );
4685 const QString asFileName = info.path() + QDir::separator() + info.completeBaseName() + asExt;
4687 bool auxiliaryStorageSavedOk =
true;
4688 if ( ! saveAuxiliaryStorage( asFileName ) )
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;
4695 if ( !mArchive->exists() )
4697 releaseHandlesToProjectArchive();
4699 mArchive->unzip( mFile.fileName() );
4702 const QString auxiliaryStorageFile =
static_cast<QgsProjectArchive *
>( mArchive.get() )->auxiliaryStorageFile();
4703 if ( ! auxiliaryStorageFile.isEmpty() )
4705 archive->
addFile( auxiliaryStorageFile );
4714 if ( QFile::exists( asFileName ) )
4716 archive->addFile( asFileName );
4721 archive->addFile( qgsFile.fileName() );
4724 const QStringList &
files = mArchive->files();
4725 for (
const QString &file :
files )
4727 if ( !file.endsWith( QLatin1String(
".qgs" ), Qt::CaseInsensitive ) && !file.endsWith( asExt, Qt::CaseInsensitive ) )
4729 archive->addFile( file );
4735 if ( !archive->zip( filename ) )
4737 setError( tr(
"Unable to perform zip" ) );
4741 return auxiliaryStorageSavedOk && zipOk;
4752 const QList<QgsMapLayer *> &layers,
4754 bool takeOwnership )
4758 const QList<QgsMapLayer *> myResultList { mLayerStore->addMapLayers(
layers, takeOwnership ) };
4759 if ( !myResultList.isEmpty() )
4762 for (
auto &l : myResultList )
4772 if ( mAuxiliaryStorage )
4787 mProjectScope.reset();
4789 return myResultList;
4795 bool takeOwnership )
4799 QList<QgsMapLayer *> addedLayers;
4800 addedLayers =
addMapLayers( QList<QgsMapLayer *>() << layer, addToLegend, takeOwnership );
4801 return addedLayers.isEmpty() ? nullptr : addedLayers[0];
4804void QgsProject::removeAuxiliaryLayer(
const QgsMapLayer *ml )
4811 const QgsVectorLayer *vl = qobject_cast<const QgsVectorLayer *>( ml );
4823 for (
const auto &layerId : layerIds )
4824 removeAuxiliaryLayer( mLayerStore->mapLayer( layerId ) );
4826 mProjectScope.reset();
4827 mLayerStore->removeMapLayers( layerIds );
4834 for (
const auto &layer :
layers )
4835 removeAuxiliaryLayer( layer );
4837 mProjectScope.reset();
4838 mLayerStore->removeMapLayers(
layers );
4845 removeAuxiliaryLayer( mLayerStore->mapLayer( layerId ) );
4846 mProjectScope.reset();
4847 mLayerStore->removeMapLayer( layerId );
4854 removeAuxiliaryLayer( layer );
4855 mProjectScope.reset();
4856 mLayerStore->removeMapLayer( layer );
4863 mProjectScope.reset();
4864 return mLayerStore->takeMapLayer( layer );
4871 return mMainAnnotationLayer;
4878 if ( mLayerStore->count() == 0 )
4881 ScopedIntIncrementor snapSingleBlocker( &mBlockSnappingUpdates );
4882 mProjectScope.reset();
4883 mLayerStore->removeAllMapLayers();
4885 snapSingleBlocker.release();
4887 if ( !mBlockSnappingUpdates )
4895 const QMap<QString, QgsMapLayer *>
layers = mLayerStore->mapLayers();
4896 QMap<QString, QgsMapLayer *>::const_iterator it =
layers.constBegin();
4897 for ( ; it !=
layers.constEnd(); ++it )
4899 it.value()->reload();
4908 return validOnly ? mLayerStore->validMapLayers() : mLayerStore->mapLayers();
4915 return mTransactionGroups.value( qMakePair( providerKey, connString ) );
4922 return &mEditBufferGroup;
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" ) )
4942 const QString layerDefaultCrs = mSettings.
value( QStringLiteral(
"/Projections/layerDefaultCrs" ),
geoEpsgCrsAuthId() ).toString();
4963bool QgsProject::saveAuxiliaryStorage(
const QString &filename )
4969 for (
auto it =
layers.constBegin(); it !=
layers.constEnd(); ++it )
4974 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( it.value() );
4982 if ( !mAuxiliaryStorage->exists( *
this ) && empty )
4986 else if ( !filename.isEmpty() )
4988 return mAuxiliaryStorage->saveAs( filename );
4992 return mAuxiliaryStorage->saveAs( *
this );
5005 return sPropertyDefinitions;
5018 return mAuxiliaryStorage.get();
5025 return mAuxiliaryStorage.get();
5032 const QDir archiveDir( mArchive->dir() );
5033 QTemporaryFile tmpFile( archiveDir.filePath(
"XXXXXX_" + nameTemplate ),
this );
5034 tmpFile.setAutoRemove(
false );
5036 mArchive->addFile( tmpFile.fileName() );
5037 return tmpFile.fileName();
5044 QStringList attachments;
5046 const QStringList files = mArchive->files();
5047 attachments.reserve( files.size() );
5048 for (
const QString &file : files )
5050 if ( QFileInfo( file ).baseName() !=
baseName )
5052 attachments.append( file );
5062 return mArchive->removeFile( path );
5069 return QStringLiteral(
"attachment:///%1" ).arg( QFileInfo( attachedFile ).
fileName() );
5076 if ( identifier.startsWith( QLatin1String(
"attachment:///" ) ) )
5078 return QDir( mArchive->dir() ).absoluteFilePath( identifier.mid( 14 ) );
5099 mProjectScope.reset();
5113 for ( QMap<QString, QgsMapLayer *>::const_iterator it =
layers.constBegin(); it !=
layers.constEnd(); ++it )
5127 const QMap<QString, QgsMapLayer *> &projectLayers =
mapLayers();
5128 for ( QMap<QString, QgsMapLayer *>::const_iterator it = projectLayers.constBegin(); it != projectLayers.constEnd(); ++it )
5133 if (
layers.contains( it.value() ) )
5134 it.value()->setFlags( it.value()->flags() &
~QgsMapLayer::Removable );
5145 QStringList customColors;
5146 QStringList customColorLabels;
5148 QgsNamedColorList::const_iterator colorIt = colors.constBegin();
5149 for ( ; colorIt != colors.constEnd(); ++colorIt )
5152 const QString label = ( *colorIt ).second;
5153 customColors.append( color );
5154 customColorLabels.append( label );
5156 writeEntry( QStringLiteral(
"Palette" ), QStringLiteral(
"/Colors" ), customColors );
5157 writeEntry( QStringLiteral(
"Palette" ), QStringLiteral(
"/Labels" ), customColorLabels );
5158 mProjectScope.reset();
5166 if ( mBackgroundColor == color )
5169 mBackgroundColor = color;
5177 return mBackgroundColor;
5184 if ( mSelectionColor == color )
5187 mSelectionColor = color;
5195 return mSelectionColor;
5232 translationContext.setFileName( QStringLiteral(
"%1/%2.ts" ).arg(
absolutePath(),
baseName() ) );
5236 translationContext.writeTsFile( locale );
5239QString
QgsProject::translate(
const QString &context,
const QString &sourceText,
const char *disambiguation,
int n )
const
5248 QString result = mTranslator->translate( context.toUtf8(), sourceText.toUtf8(), disambiguation, n );
5250 if ( result.isEmpty() )
5264 for (
auto it =
layers.constBegin(); it !=
layers.constEnd(); ++it )
5269 if ( !( ( *it )->accept( visitor ) ) )
5278 if ( !mLayoutManager->accept( visitor ) )
5281 if ( !mAnnotationManager->accept( visitor ) )
5289 return mElevationShadingRenderer;
5292void QgsProject::loadProjectFlags(
const QDomDocument *doc )
5296 QDomElement element = doc->documentElement().firstChildElement( QStringLiteral(
"projectFlags" ) );
5298 if ( !element.isNull() )
5305 element = doc->documentElement().firstChildElement( QStringLiteral(
"evaluateDefaultValues" ) );
5306 if ( !element.isNull() )
5308 if ( element.attribute( QStringLiteral(
"active" ), QStringLiteral(
"0" ) ).toInt() == 1 )
5313 element = doc->documentElement().firstChildElement( QStringLiteral(
"trust" ) );
5314 if ( !element.isNull() )
5316 if ( element.attribute( QStringLiteral(
"active" ), QStringLiteral(
"0" ) ).toInt() == 1 )
5332 const QString projectFunctions =
readEntry( QStringLiteral(
"ExpressionFunctions" ), QStringLiteral(
"/pythonCode" ), QString() );
5333 if ( !projectFunctions.isEmpty() )
5353QHash< QString, QColor > loadColorsFromProject(
const QgsProject *project )
5355 QHash< QString, QColor > colors;
5358 QStringList colorStrings = project->
readListEntry( QStringLiteral(
"Palette" ), QStringLiteral(
"/Colors" ) );
5359 const QStringList colorLabels = project->
readListEntry( QStringLiteral(
"Palette" ), QStringLiteral(
"/Labels" ) );
5363 for ( QStringList::iterator it = colorStrings.begin();
5364 it != colorStrings.end(); ++it )
5368 if ( colorLabels.length() > colorIndex )
5370 label = colorLabels.at( colorIndex );
5373 colors.insert( label.toLower(), color );
5381GetNamedProjectColor::GetNamedProjectColor(
const QgsProject *project )
5387 mColors = loadColorsFromProject( project );
5390GetNamedProjectColor::GetNamedProjectColor(
const QHash<QString, QColor> &colors )
5398 const QString colorName = values.at( 0 ).toString().toLower();
5399 if ( mColors.contains( colorName ) )
5401 return QStringLiteral(
"%1,%2,%3" ).arg( mColors.value( colorName ).red() ).arg( mColors.value( colorName ).green() ).arg( mColors.value( colorName ).blue() );
5409 return new GetNamedProjectColor( mColors );
5412GetNamedProjectColorObject::GetNamedProjectColorObject(
const QgsProject *project )
5418 mColors = loadColorsFromProject( project );
5421GetNamedProjectColorObject::GetNamedProjectColorObject(
const QHash<QString, QColor> &colors )
5429 const QString colorName = values.at( 0 ).toString().toLower();
5430 if ( mColors.contains( colorName ) )
5432 return mColors.value( colorName );
5440 return new GetNamedProjectColorObject( mColors );
5445GetSensorData::GetSensorData(
const QMap<QString, QgsAbstractSensor::SensorData> &sensorData )
5448 QStringLiteral(
"Sensors" ) )
5449 , mSensorData( sensorData )
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 ) )
5460 if ( expiration <= 0 || ( timestamp - mSensorData[sensorName].lastTimestamp.toMSecsSinceEpoch() ) < expiration )
5462 return mSensorData[sensorName].lastValue;
5471 return new GetSensorData( mSensorData );
@ 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.
QFlags< ProjectCapability > ProjectCapabilities
Flags which control project capabilities.
QFlags< ProjectReadFlag > ProjectReadFlags
Project load flags.
DistanceUnit
Units of distance.
FilePathType
File path types.
TransactionMode
Transaction mode.
@ 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.
@ SquareMeters
Square meters.
@ Critical
Critical/error message.
@ Success
Used for reporting a successful operation.
PythonEmbeddedMode
Authorisation to run Python Embedded in projects.
@ Always
Python embedded is always run.
@ Ask
User is prompt before running.
@ SessionOnly
Only during this session.
@ Compound
Compound (horizontal + vertical) CRS.
@ Projected
Projected CRS.
@ DerivedProjected
Derived projected CRS.
@ Engineering
Engineering CRS.
@ Geographic3d
3D geopraphic CRS
@ Geographic2d
2D geographic CRS
@ Geocentric
Geocentric CRS.
AvoidIntersectionsMode
Flags which control how intersections of pre-existing feature are handled when digitizing new feature...
@ 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.
@ 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...
QFlags< DataProviderReadFlag > DataProviderReadFlags
Flags which control data provider construction.
LayerType
Types of layers that can be added to a map.
@ 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.
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
@ Mesh
Mesh layer. Added in QGIS 3.2.
@ 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
@ Preferred
Preferred format, matching the most recent WKT ISO standard. Currently an alias to WKT2_2019,...
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.
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.
Qgis::DistanceUnit mapUnits
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.
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.
Container of fields for a vector layer.
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.
void setEnabled(bool enabled)
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.
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
QString originalXmlProperties() const
Returns the XML properties of the original layer as they were when the layer was first read from the ...
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.
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.
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)
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...
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,...
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
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
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.
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.
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.
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...
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.
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
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 ...
@ WMSOnlineResource
Alias.
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
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
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
QgsAnnotationManager * annotationManager()
Returns pointer to the project's annotation manager.
QgsProjectDisplaySettings * displaySettings
QgsProjectMetadata metadata
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
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.
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
QgsMapLayer * addMapLayer(QgsMapLayer *mapLayer, bool addToLegend=true, bool takeOwnership=true)
Add a layer to the map of loaded layers.
QStringList nonIdentifiableLayers
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...
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.
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.
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.
@ String
Any string value.
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.
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.
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:
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.
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.
bool startEditing()
Start editing.
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".
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.
#define Q_NOWARN_DEPRECATED_POP
QString qgsEnumValueToKey(const T &value, bool *returnOk=nullptr)
Returns the value for the given key of an enum.
QString qgsFlagValueToKeys(const T &value, bool *returnOk=nullptr)
Returns the value for the given keys of a flag.
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.
#define Q_NOWARN_DEPRECATED_PUSH
CONSTLATIN1STRING geoEpsgCrsAuthId()
Geographic coord sys from EPSG authority.
#define QgsDebugMsgLevel(str, level)
#define QgsDebugError(str)
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
Qgis::DataProviderReadFlags flags
QgsDataProvider::ProviderOptions options
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.