43using namespace Qt::StringLiterals;
48 if ( !file.open( QIODevice::ReadOnly ) )
50 errorMessage = u
"Can not open file"_s;
56 if ( !doc.setContent( &file, &message ) )
58 errorMessage = message;
62 const QFileInfo fileinfo( file );
63 QDir::setCurrent( fileinfo.absoluteDir().path() );
69 return loadLayerDefinition( doc, project, rootGroup, errorMessage, context, insertMethod, insertPoint );
84 QVector<QDomNode> clonedSorted;
85 const auto constSortedLayerNodes = sortedLayerNodes;
86 for (
const QDomNode &node : constSortedLayerNodes )
88 clonedSorted << node.cloneNode();
90 QDomNode layersNode = doc.elementsByTagName( u
"maplayers"_s ).at( 0 );
92 QDomNode childNode = layersNode.firstChild();
93 for (
int i = 0; ! childNode.isNull(); i++ )
95 layersNode.replaceChild( clonedSorted.at( i ), childNode );
96 childNode = childNode.nextSibling();
103 const QDomNodeList treeLayerNodes = doc.elementsByTagName( u
"layer-tree-layer"_s );
104 for (
int i = 0; i < treeLayerNodes.length(); ++i )
106 const QDomNode treeLayerNode = treeLayerNodes.item( i );
107 QDomElement treeLayerElem = treeLayerNode.toElement();
108 const QString oldid = treeLayerElem.attribute( u
"id"_s );
109 const QString layername = treeLayerElem.attribute( u
"name"_s );
111 treeLayerElem.setAttribute( u
"id"_s, newid );
114 const QDomNodeList ids = doc.elementsByTagName( u
"id"_s );
115 QDomNode idnode = ids.at( 0 );
116 for (
int j = 0; ! idnode.isNull() ; ++j )
118 idnode = ids.at( j );
119 const QDomElement idElem = idnode.toElement();
120 if ( idElem.text() == oldid )
122 idElem.firstChild().setNodeValue( newid );
127 const QDomNodeList vectorJoinNodes = doc.elementsByTagName( u
"join"_s );
128 for (
int j = 0; j < vectorJoinNodes.size(); ++j )
130 const QDomNode joinNode = vectorJoinNodes.at( j );
131 const QDomElement joinElement = joinNode.toElement();
132 if ( joinElement.attribute( u
"joinLayerId"_s ) == oldid )
134 joinNode.toElement().setAttribute( u
"joinLayerId"_s, newid );
139 const QDomNodeList dataDeps = doc.elementsByTagName( u
"dataDependencies"_s );
140 for (
int i = 0; i < dataDeps.size(); i++ )
142 const QDomNodeList layers = dataDeps.at( i ).childNodes();
143 for (
int j = 0; j < layers.size(); j++ )
145 QDomElement elt = layers.at( j ).toElement();
146 if ( elt.attribute( u
"id"_s ) == oldid )
148 elt.setAttribute( u
"id"_s, newid );
154 const QDomNodeList widgetConfig = doc.elementsByTagName( u
"editWidget"_s );
155 for (
int i = 0; i < widgetConfig.size(); i++ )
157 const QDomNodeList config = widgetConfig.at( i ).childNodes();
158 for (
int j = 0; j < config.size(); j++ )
160 const QDomNodeList optMap = config.at( j ).childNodes();
161 for (
int z = 0; z < optMap.size(); z++ )
163 const QDomNodeList opts = optMap.at( z ).childNodes();
164 for (
int k = 0; k < opts.size(); k++ )
166 QDomElement opt = opts.at( k ).toElement();
167 if ( opt.attribute( u
"value"_s ) == oldid )
169 opt.setAttribute( u
"value"_s, newid );
177 QDomElement layerTreeElem = doc.documentElement().firstChildElement( u
"layer-tree-group"_s );
178 bool loadInLegend =
true;
179 if ( !layerTreeElem.isNull() )
182 loadInLegend =
false;
185 const QList<QgsMapLayer *> layers = QgsLayerDefinition::loadLayerDefinitionLayersInternal( doc, context, errorMessage );
190 const auto constLayers = layers;
193 layer->resolveReferences( project );
198 const QList<QgsLayerTreeNode *> nodes = root.
children();
201 switch ( insertMethod )
233 if ( !file.open( QFile::WriteOnly | QFile::Truncate ) )
235 errorMessage = file.errorString();
250 const QDomDocument doc( u
"qgis-layer-definition"_s );
254 QTextStream qlayerstream( &file );
255 doc.save( qlayerstream, 2 );
261 Q_UNUSED( errorMessage )
262 QDomElement qgiselm = doc.createElement( u
"qlr"_s );
263 doc.appendChild( qgiselm );
272 QDomElement layerselm = doc.createElement( u
"maplayers"_s );
273 const QList<QgsLayerTreeLayer *> layers = root.
findLayers();
276 if ( ! layer->layer() )
278 QgsDebugMsgLevel( u
"Not a valid map layer: skipping %1"_s.arg( layer->name( ) ), 4 );
281 QDomElement layerelm = doc.createElement( u
"maplayer"_s );
282 layer->layer()->writeLayerXml( layerelm, doc, context );
283 layerselm.appendChild( layerelm );
285 qgiselm.appendChild( layerselm );
291 QDomDocument doc( u
"qgis-layer-definition"_s );
292 QDomElement qgiselm = doc.createElement( u
"qlr"_s );
293 doc.appendChild( qgiselm );
294 QDomElement layerselm = doc.createElement( u
"maplayers"_s );
295 const auto constLayers = layers;
298 QDomElement layerelm = doc.createElement( u
"maplayer"_s );
299 layer->writeLayerXml( layerelm, doc, context );
300 layerselm.appendChild( layerelm );
302 qgiselm.appendChild( layerselm );
308 QString errorMessage;
309 return loadLayerDefinitionLayersInternal( document, context, errorMessage );
312QList<QgsMapLayer *> QgsLayerDefinition::loadLayerDefinitionLayersInternal( QDomDocument &document,
QgsReadWriteContext &context, QString &errorMessage )
314 QList<QgsMapLayer *> layers;
315 QDomElement layerElem = document.documentElement().firstChildElement( u
"projectlayers"_s ).firstChildElement( u
"maplayer"_s );
317 if ( layerElem.isNull() )
319 layerElem = document.documentElement().firstChildElement( u
"maplayers"_s ).firstChildElement( u
"maplayer"_s );
322 while ( ! layerElem.isNull() )
324 const QString type = layerElem.attribute( u
"type"_s );
325 QgsMapLayer *layer =
nullptr;
334 layer =
new QgsVectorLayer();
338 layer =
new QgsRasterLayer();
343 const QString typeName = layerElem.attribute( u
"name"_s );
349 layer =
new QgsMeshLayer();
353 layer =
new QgsVectorTileLayer;
357 layer =
new QgsPointCloudLayer();
361 layer =
new QgsTiledSceneLayer;
365 layer =
new QgsGroupLayer( QString(), QgsGroupLayer::LayerOptions( QgsCoordinateTransformContext() ) );
382 errorMessage = QObject::tr(
"Unsupported layer type: %1" ).arg( type );
384 layerElem = layerElem.nextSiblingElement( u
"maplayer"_s );
391 QFile file( qlrfile );
392 if ( !file.open( QIODevice::ReadOnly ) )
395 return QList<QgsMapLayer *>();
399 if ( !doc.setContent( &file ) )
402 return QList<QgsMapLayer *>();
411void QgsLayerDefinition::DependencySorter::init(
const QDomDocument &doc )
414 QMap< QString, QVector< QString > > dependencies;
415 QStringList sortedLayers;
416 QList< QPair<QString, QDomNode> > layersToSort;
417 QStringList layerIds;
419 QDomElement layerElem = doc.documentElement().firstChildElement( u
"projectlayers"_s ).firstChildElement( u
"maplayer"_s );
421 if ( layerElem.isNull() )
423 layerElem = doc.documentElement().firstChildElement( u
"maplayers"_s ).firstChildElement( u
"maplayer"_s );
426 if ( layerElem.isNull() )
428 layerElem = doc.documentElement().firstChildElement( u
"maplayer"_s );
431 const QDomElement &firstElement { layerElem };
433 QVector<QString> deps;
434 while ( !layerElem.isNull() )
438 const QString
id = layerElem.namedItem( u
"id"_s ).toElement().text();
442 const QDomElement layerDependenciesElem = layerElem.firstChildElement( u
"layerDependencies"_s );
443 if ( !layerDependenciesElem.isNull() )
445 const QDomNodeList dependencyList = layerDependenciesElem.elementsByTagName( u
"layer"_s );
446 for (
int j = 0; j < dependencyList.size(); ++j )
448 const QDomElement depElem = dependencyList.at( j ).toElement();
449 deps << depElem.attribute( u
"id"_s );
452 dependencies[id] = deps;
457 mSortedLayerNodes << layerElem;
458 mSortedLayerIds << id;
462 layersToSort << qMakePair(
id, layerElem );
463 mDependentLayerIds.insert(
id );
465 layerElem = layerElem.nextSiblingElement( );
469 const auto constDependencies = dependencies;
470 for (
const QVector< QString > &ids : constDependencies )
472 const auto constIds = ids;
473 for (
const QString &depId : constIds )
475 if ( !dependencies.contains( depId ) )
478 mHasMissingDependency =
true;
479 layerElem = firstElement;
480 while ( ! layerElem.isNull() )
482 mSortedLayerNodes << layerElem;
483 layerElem = layerElem.nextSiblingElement( );
485 mSortedLayerIds = layerIds;
495 while ( !layersToSort.empty() && !mHasCycle )
497 QList< QPair<QString, QDomNode> >::iterator it = layersToSort.begin();
498 while ( it != layersToSort.end() )
500 const QString idToSort = it->first;
501 const QDomNode node = it->second;
503 bool resolved =
true;
504 const auto deps { dependencies.value( idToSort ) };
505 for (
const QString &dep : deps )
507 if ( !sortedLayers.contains( dep ) )
515 sortedLayers << idToSort;
516 mSortedLayerNodes << node;
517 mSortedLayerIds << idToSort;
518 it = layersToSort.erase( it );
531 , mHasMissingDependency( false )
538 , mHasMissingDependency( false )
540 QString qgsProjectFile = fileName;
542 if ( fileName.endsWith(
".qgz"_L1, Qt::CaseInsensitive ) )
544 archive.unzip( fileName );
545 qgsProjectFile = archive.projectFile();
549 QFile pFile( qgsProjectFile );
550 ( void )pFile.open( QIODevice::ReadOnly );
551 ( void )doc.setContent( &pFile );
557 return mDependentLayerIds.contains( layerId );
LayerTreeInsertionMethod
Layer tree insertion methods.
@ TopOfTree
Layers are added at the top of the layer tree.
@ AboveInsertionPoint
Layers are added in the tree above the insertion point.
FilePathType
File path types.
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.
static QgsPluginLayerRegistry * pluginLayerRegistry()
Returns the application's plugin layer registry, used for managing plugin layer types.
static QString ensureFileNameHasExtension(const QString &fileName, const QStringList &extensions)
Ensures that a fileName ends with an extension from the provided list of extensions.
Handles sorting of dependencies stored in a XML project or layer definition file.
bool hasMissingDependency() const
Whether some dependency is missing.
bool isLayerDependent(const QString &layerId) const
Returns whether the layer associated with thelayerId is dependent from another layer.
DependencySorter(const QDomDocument &doc)
Constructor.
QVector< QDomNode > sortedLayerNodes() const
Gets the layer nodes in an order where they can be loaded incrementally without dependency break.
static QDomDocument exportLayerDefinitionLayers(const QList< QgsMapLayer * > &layers, const QgsReadWriteContext &context)
Returns the given layer as a layer definition document Layer definitions store the data source as wel...
static QList< QgsMapLayer * > loadLayerDefinitionLayers(QDomDocument &document, QgsReadWriteContext &context)
Creates new layers from a layer definition document.
static bool loadLayerDefinition(const QString &path, QgsProject *project, QgsLayerTreeGroup *rootGroup, QString &errorMessage, Qgis::LayerTreeInsertionMethod insertMethod=Qgis::LayerTreeInsertionMethod::OptimalInInsertionGroup, const QgsLayerTreeRegistryBridge::InsertionPoint *insertPoint=nullptr)
Loads the QLR at path into QGIS.
static bool exportLayerDefinition(const QString &path, const QList< QgsLayerTreeNode * > &selectedTreeNodes, QString &errorMessage)
Exports the selected layer tree nodes to a QLR 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.
void readChildrenFromXml(const QDomElement &element, const QgsReadWriteContext &context)
Read children from XML and append them to the group.
void writeXml(QDomElement &parentElement, const QgsReadWriteContext &context) override
Write group (tree) as XML element <layer-tree-group> and add it to the given parent element.
void addChildNode(QgsLayerTreeNode *node)
Append an existing node.
void insertChildNodes(int index, const QList< QgsLayerTreeNode * > &nodes)
Insert existing nodes at specified position.
QList< QgsLayerTreeLayer * > findLayers() const
Find all layer nodes.
Layer tree node points to a map layer.
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...
QList< QgsLayerTreeNode * > children()
Gets list of children of the node. Children are owned by the parent.
virtual QgsLayerTreeNode * clone() const =0
Create a copy of the node. Returns new instance.
static Qgis::LayerType typeFromString(const QString &string, bool &ok)
Returns the map layer type corresponding a string value.
Base class for all map layer types.
static QString generateId(const QString &layerName)
Generates an unique identifier for this layer, the generate ID is prefixed by layerName.
bool readLayerXml(const QDomElement &layerElement, QgsReadWriteContext &context, QgsMapLayer::ReadFlags flags=QgsMapLayer::ReadFlags(), QgsDataProvider *preloadedProvider=nullptr)
Sets state from DOM document.
Resolves relative paths into absolute paths and vice versa.
QgsPluginLayer * createLayer(const QString &typeName, const QString &uri=QString())
Returns new layer if corresponding plugin has been found else returns nullptr.
Allows managing the zip/unzip actions on project files.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
QList< QgsMapLayer * > addMapLayers(const QList< QgsMapLayer * > &mapLayers, bool addToLegend=true, bool takeOwnership=true)
Add a list of layers to the map of loaded layers.
static QgsProject * instance()
Returns the QgsProject singleton instance.
A container for the context for various read/write operations on objects.
void setProjectTranslator(QgsProjectTranslator *projectTranslator)
Sets the project translator.
void setPathResolver(const QgsPathResolver &resolver)
Sets up path resolver for conversion between relative and absolute paths.
#define QgsDebugMsgLevel(str, level)
#define QgsDebugError(str)
A structure to define the insertion point to the layer tree.
QgsLayerTreeGroup * group