QGIS API Documentation 4.0.0-Norrköping (1ddcee3d0e4)
Loading...
Searching...
No Matches
qgslayerdefinition.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslayerdefinition.cpp
3 ---------------------
4 begin : January 2015
5 copyright : (C) 2015 by Nathan Woodrow
6 email : woodrow dot nathan at gmail dot com
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15#include "qgslayerdefinition.h"
16
17#include "qgsapplication.h"
18#include "qgsfileutils.h"
19#include "qgsgrouplayer.h"
20#include "qgslayertreegroup.h"
21#include "qgslayertreelayer.h"
22#include "qgslogger.h"
23#include "qgsmaplayer.h"
24#include "qgsmaplayerfactory.h"
25#include "qgsmeshlayer.h"
26#include "qgspathresolver.h"
27#include "qgspluginlayer.h"
29#include "qgspointcloudlayer.h"
30#include "qgsproject.h"
31#include "qgsrasterlayer.h"
32#include "qgsreadwritecontext.h"
33#include "qgstiledscenelayer.h"
34#include "qgsvectorlayer.h"
35#include "qgsvectortilelayer.h"
36
37#include <QDir>
38#include <QFile>
39#include <QFileInfo>
40#include <QString>
41#include <QTextStream>
42
43using namespace Qt::StringLiterals;
44
46 const QString &path, QgsProject *project, QgsLayerTreeGroup *rootGroup, QString &errorMessage, Qgis::LayerTreeInsertionMethod insertMethod, const QgsLayerTreeRegistryBridge::InsertionPoint *insertPoint
47)
48{
49 QFile file( path );
50 if ( !file.open( QIODevice::ReadOnly ) )
51 {
52 errorMessage = u"Can not open file"_s;
53 return false;
54 }
55
56 QDomDocument doc;
57 QString message;
58 if ( !doc.setContent( &file, &message ) )
59 {
60 errorMessage = message;
61 return false;
62 }
63
64 const QFileInfo fileinfo( file );
65 QDir::setCurrent( fileinfo.absoluteDir().path() );
66
67 QgsReadWriteContext context;
68 context.setPathResolver( QgsPathResolver( path ) );
69 context.setProjectTranslator( project );
70
71 return loadLayerDefinition( doc, project, rootGroup, errorMessage, context, insertMethod, insertPoint );
72}
73
75 QDomDocument doc,
76 QgsProject *project,
77 QgsLayerTreeGroup *rootGroup,
78 QString &errorMessage,
79 QgsReadWriteContext &context,
82)
83{
84 errorMessage.clear();
85
87
88 // reorder maplayer nodes based on dependencies
89 // dependencies have to be resolved before IDs get changed
90 const DependencySorter depSorter( doc );
91 if ( !depSorter.hasMissingDependency() )
92 {
93 const QVector<QDomNode> sortedLayerNodes = depSorter.sortedLayerNodes();
94 QVector<QDomNode> clonedSorted;
95 const auto constSortedLayerNodes = sortedLayerNodes;
96 for ( const QDomNode &node : constSortedLayerNodes )
97 {
98 clonedSorted << node.cloneNode();
99 }
100 QDomNode layersNode = doc.elementsByTagName( u"maplayers"_s ).at( 0 );
101 // replace old children with new ones
102 QDomNode childNode = layersNode.firstChild();
103 for ( int i = 0; !childNode.isNull(); i++ )
104 {
105 layersNode.replaceChild( clonedSorted.at( i ), childNode );
106 childNode = childNode.nextSibling();
107 }
108 }
109 // if a dependency is missing, we still try to load layers, since dependencies may already be loaded
110
111 // IDs of layers should be changed otherwise we may have more then one layer with the same id
112 // We have to replace the IDs before we load them because it's too late once they are loaded
113 const QDomNodeList treeLayerNodes = doc.elementsByTagName( u"layer-tree-layer"_s );
114 for ( int i = 0; i < treeLayerNodes.length(); ++i )
115 {
116 const QDomNode treeLayerNode = treeLayerNodes.item( i );
117 QDomElement treeLayerElem = treeLayerNode.toElement();
118 const QString oldid = treeLayerElem.attribute( u"id"_s );
119 const QString layername = treeLayerElem.attribute( u"name"_s );
120 const QString newid = QgsMapLayer::generateId( layername );
121 treeLayerElem.setAttribute( u"id"_s, newid );
122
123 // Replace IDs for map layers
124 const QDomNodeList ids = doc.elementsByTagName( u"id"_s );
125 QDomNode idnode = ids.at( 0 );
126 for ( int j = 0; !idnode.isNull(); ++j )
127 {
128 idnode = ids.at( j );
129 const QDomElement idElem = idnode.toElement();
130 if ( idElem.text() == oldid )
131 {
132 idElem.firstChild().setNodeValue( newid );
133 }
134 }
135
136 // change layer IDs for vector joins
137 const QDomNodeList vectorJoinNodes = doc.elementsByTagName( u"join"_s ); // TODO: Find a better way of searching for vectorjoins, there might be other <join> elements within the project.
138 for ( int j = 0; j < vectorJoinNodes.size(); ++j )
139 {
140 const QDomNode joinNode = vectorJoinNodes.at( j );
141 const QDomElement joinElement = joinNode.toElement();
142 if ( joinElement.attribute( u"joinLayerId"_s ) == oldid )
143 {
144 joinNode.toElement().setAttribute( u"joinLayerId"_s, newid );
145 }
146 }
147
148 // change IDs of dependencies
149 const QDomNodeList dataDeps = doc.elementsByTagName( u"dataDependencies"_s );
150 for ( int i = 0; i < dataDeps.size(); i++ )
151 {
152 const QDomNodeList layers = dataDeps.at( i ).childNodes();
153 for ( int j = 0; j < layers.size(); j++ )
154 {
155 QDomElement elt = layers.at( j ).toElement();
156 if ( elt.attribute( u"id"_s ) == oldid )
157 {
158 elt.setAttribute( u"id"_s, newid );
159 }
160 }
161 }
162
163 // Change IDs of widget config values
164 const QDomNodeList widgetConfig = doc.elementsByTagName( u"editWidget"_s );
165 for ( int i = 0; i < widgetConfig.size(); i++ )
166 {
167 const QDomNodeList config = widgetConfig.at( i ).childNodes();
168 for ( int j = 0; j < config.size(); j++ )
169 {
170 const QDomNodeList optMap = config.at( j ).childNodes();
171 for ( int z = 0; z < optMap.size(); z++ )
172 {
173 const QDomNodeList opts = optMap.at( z ).childNodes();
174 for ( int k = 0; k < opts.size(); k++ )
175 {
176 QDomElement opt = opts.at( k ).toElement();
177 if ( opt.attribute( u"value"_s ) == oldid )
178 {
179 opt.setAttribute( u"value"_s, newid );
180 }
181 }
182 }
183 }
184 }
185 }
186
187 QDomElement layerTreeElem = doc.documentElement().firstChildElement( u"layer-tree-group"_s );
188 bool loadInLegend = true;
189 if ( !layerTreeElem.isNull() )
190 {
191 root.readChildrenFromXml( layerTreeElem, context );
192 loadInLegend = false;
193 }
194
195 const QList<QgsMapLayer *> layers = QgsLayerDefinition::loadLayerDefinitionLayersInternal( doc, context, errorMessage );
196
197 project->addMapLayers( layers, loadInLegend );
198
199 // Now that all layers are loaded, refresh the vectorjoins to get the joined fields
200 const auto constLayers = layers;
201 for ( QgsMapLayer *layer : constLayers )
202 {
203 layer->resolveReferences( project );
204 }
205
206 root.resolveReferences( project );
207
208 const QList<QgsLayerTreeNode *> nodes = root.children();
209 root.abandonChildren();
210
211 switch ( insertMethod )
212 {
214 if ( insertPoint )
215 {
216 insertPoint->group->insertChildNodes( insertPoint->position, nodes );
217 }
218 else
219 {
220 rootGroup->insertChildNodes( -1, nodes );
221 }
222 break;
224 rootGroup->insertChildNodes( 0, nodes );
225 break;
226 default: //Keep current behavior for Qgis::LayerTreeInsertionMethod::OptimalInInsertionGroup
227 rootGroup->insertChildNodes( -1, nodes );
228 }
229
230 return true;
231}
232
233bool QgsLayerDefinition::exportLayerDefinition( const QString &path, const QList<QgsLayerTreeNode *> &selectedTreeNodes, QString &errorMessage )
234{
235 return exportLayerDefinition( path, selectedTreeNodes, QgsProject::instance()->filePathStorage(), errorMessage ); // skip-keyword-check
236}
237
238bool QgsLayerDefinition::exportLayerDefinition( const QString &p, const QList<QgsLayerTreeNode *> &selectedTreeNodes, Qgis::FilePathType pathType, QString &errorMessage )
239{
240 const QString path = QgsFileUtils::ensureFileNameHasExtension( p, { u"qlr"_s } );
241
242 QFile file( path );
243 if ( !file.open( QFile::WriteOnly | QFile::Truncate ) )
244 {
245 errorMessage = file.errorString();
246 return false;
247 }
248
249 QgsReadWriteContext context;
250 switch ( pathType )
251 {
253 context.setPathResolver( QgsPathResolver( QString() ) );
254 break;
256 context.setPathResolver( QgsPathResolver( path ) );
257 break;
258 }
259
260 const QDomDocument doc( u"qgis-layer-definition"_s );
261 if ( !exportLayerDefinition( doc, selectedTreeNodes, errorMessage, context ) )
262 return false;
263
264 QTextStream qlayerstream( &file );
265 doc.save( qlayerstream, 2 );
266 return true;
267}
268
269bool QgsLayerDefinition::exportLayerDefinition( QDomDocument doc, const QList<QgsLayerTreeNode *> &selectedTreeNodes, QString &errorMessage, const QgsReadWriteContext &context )
270{
271 Q_UNUSED( errorMessage )
272 QDomElement qgiselm = doc.createElement( u"qlr"_s );
273 doc.appendChild( qgiselm );
275 for ( QgsLayerTreeNode *node : selectedTreeNodes )
276 {
277 QgsLayerTreeNode *newnode = node->clone();
278 root.addChildNode( newnode );
279 }
280 root.writeXml( qgiselm, context );
281
282 QDomElement layerselm = doc.createElement( u"maplayers"_s );
283 const QList<QgsLayerTreeLayer *> layers = root.findLayers();
284 for ( QgsLayerTreeLayer *layer : layers )
285 {
286 if ( !layer->layer() )
287 {
288 QgsDebugMsgLevel( u"Not a valid map layer: skipping %1"_s.arg( layer->name() ), 4 );
289 continue;
290 }
291 QDomElement layerelm = doc.createElement( u"maplayer"_s );
292 layer->layer()->writeLayerXml( layerelm, doc, context );
293 layerselm.appendChild( layerelm );
294 }
295 qgiselm.appendChild( layerselm );
296 return true;
297}
298
299QDomDocument QgsLayerDefinition::exportLayerDefinitionLayers( const QList<QgsMapLayer *> &layers, const QgsReadWriteContext &context )
300{
301 QDomDocument doc( u"qgis-layer-definition"_s );
302 QDomElement qgiselm = doc.createElement( u"qlr"_s );
303 doc.appendChild( qgiselm );
304 QDomElement layerselm = doc.createElement( u"maplayers"_s );
305 const auto constLayers = layers;
306 for ( QgsMapLayer *layer : constLayers )
307 {
308 QDomElement layerelm = doc.createElement( u"maplayer"_s );
309 layer->writeLayerXml( layerelm, doc, context );
310 layerselm.appendChild( layerelm );
311 }
312 qgiselm.appendChild( layerselm );
313 return doc;
314}
315
316QList<QgsMapLayer *> QgsLayerDefinition::loadLayerDefinitionLayers( QDomDocument &document, QgsReadWriteContext &context )
317{
318 QString errorMessage;
319 return loadLayerDefinitionLayersInternal( document, context, errorMessage );
320}
321
322QList<QgsMapLayer *> QgsLayerDefinition::loadLayerDefinitionLayersInternal( QDomDocument &document, QgsReadWriteContext &context, QString &errorMessage )
323{
324 QList<QgsMapLayer *> layers;
325 QDomElement layerElem = document.documentElement().firstChildElement( u"projectlayers"_s ).firstChildElement( u"maplayer"_s );
326 // For QLR:
327 if ( layerElem.isNull() )
328 {
329 layerElem = document.documentElement().firstChildElement( u"maplayers"_s ).firstChildElement( u"maplayer"_s );
330 }
331
332 while ( !layerElem.isNull() )
333 {
334 const QString type = layerElem.attribute( u"type"_s );
335 QgsMapLayer *layer = nullptr;
336
337 bool ok = false;
338 const Qgis::LayerType layerType = QgsMapLayerFactory::typeFromString( type, ok );
339 if ( ok )
340 {
341 switch ( layerType )
342 {
344 layer = new QgsVectorLayer();
345 break;
346
348 layer = new QgsRasterLayer();
349 break;
350
352 {
353 const QString typeName = layerElem.attribute( u"name"_s );
355 break;
356 }
357
359 layer = new QgsMeshLayer();
360 break;
361
363 layer = new QgsVectorTileLayer;
364 break;
365
367 layer = new QgsPointCloudLayer();
368 break;
369
371 layer = new QgsTiledSceneLayer;
372 break;
373
375 layer = new QgsGroupLayer( QString(), QgsGroupLayer::LayerOptions( QgsCoordinateTransformContext() ) );
376 break;
377
379 break;
380 }
381 }
382
383 if ( layer )
384 {
385 // always add the layer, even if the source is invalid -- this allows users to fix the source
386 // at a later stage and still retain all the layer properties intact
387 layer->readLayerXml( layerElem, context );
388 layers << layer;
389 }
390 else
391 {
392 errorMessage = QObject::tr( "Unsupported layer type: %1" ).arg( type );
393 }
394 layerElem = layerElem.nextSiblingElement( u"maplayer"_s );
395 }
396 return layers;
397}
398
399QList<QgsMapLayer *> QgsLayerDefinition::loadLayerDefinitionLayers( const QString &qlrfile )
400{
401 QFile file( qlrfile );
402 if ( !file.open( QIODevice::ReadOnly ) )
403 {
404 QgsDebugError( u"Can't open file"_s );
405 return QList<QgsMapLayer *>();
406 }
407
408 QDomDocument doc;
409 if ( !doc.setContent( &file ) )
410 {
411 QgsDebugError( u"Can't set content"_s );
412 return QList<QgsMapLayer *>();
413 }
414
415 QgsReadWriteContext context;
416 context.setPathResolver( QgsPathResolver( qlrfile ) );
417 //no project translator defined here
419}
420
421void QgsLayerDefinition::DependencySorter::init( const QDomDocument &doc )
422{
423 // Determine a loading order of layers based on a graph of dependencies
424 QMap< QString, QVector< QString > > dependencies;
425 QStringList sortedLayers;
426 QList< QPair<QString, QDomNode> > layersToSort;
427 QStringList layerIds;
428
429 QDomElement layerElem = doc.documentElement().firstChildElement( u"projectlayers"_s ).firstChildElement( u"maplayer"_s );
430 // For QLR:
431 if ( layerElem.isNull() )
432 {
433 layerElem = doc.documentElement().firstChildElement( u"maplayers"_s ).firstChildElement( u"maplayer"_s );
434 }
435 // For tests (I don't know if there is a real use case for such a document except for test_qgslayerdefinition.py)
436 if ( layerElem.isNull() )
437 {
438 layerElem = doc.documentElement().firstChildElement( u"maplayer"_s );
439 }
440
441 const QDomElement &firstElement { layerElem };
442
443 QVector<QString> deps; //avoid expensive allocation for list for every iteration
444 while ( !layerElem.isNull() )
445 {
446 deps.resize( 0 ); // preserve capacity - don't use clear
447
448 const QString id = layerElem.namedItem( u"id"_s ).toElement().text();
449 layerIds << id;
450
451 // dependencies for this layer
452 const QDomElement layerDependenciesElem = layerElem.firstChildElement( u"layerDependencies"_s );
453 if ( !layerDependenciesElem.isNull() )
454 {
455 const QDomNodeList dependencyList = layerDependenciesElem.elementsByTagName( u"layer"_s );
456 for ( int j = 0; j < dependencyList.size(); ++j )
457 {
458 const QDomElement depElem = dependencyList.at( j ).toElement();
459 deps << depElem.attribute( u"id"_s );
460 }
461 }
462 dependencies[id] = deps;
463
464 if ( deps.empty() )
465 {
466 sortedLayers << id;
467 mSortedLayerNodes << layerElem;
468 mSortedLayerIds << id;
469 }
470 else
471 {
472 layersToSort << qMakePair( id, layerElem );
473 mDependentLayerIds.insert( id );
474 }
475 layerElem = layerElem.nextSiblingElement();
476 }
477
478 // check that all dependencies are present
479 const auto constDependencies = dependencies;
480 for ( const QVector< QString > &ids : constDependencies )
481 {
482 const auto constIds = ids;
483 for ( const QString &depId : constIds )
484 {
485 if ( !dependencies.contains( depId ) )
486 {
487 // some dependencies are not satisfied
488 mHasMissingDependency = true;
489 layerElem = firstElement;
490 while ( !layerElem.isNull() )
491 {
492 mSortedLayerNodes << layerElem;
493 layerElem = layerElem.nextSiblingElement();
494 }
495 mSortedLayerIds = layerIds;
496 return;
497 }
498 }
499 }
500
501 // cycles should be very rare, since layers with cyclic dependencies may only be created by
502 // manually modifying the project file
503 mHasCycle = false;
504
505 while ( !layersToSort.empty() && !mHasCycle )
506 {
507 QList< QPair<QString, QDomNode> >::iterator it = layersToSort.begin();
508 while ( it != layersToSort.end() )
509 {
510 const QString idToSort = it->first;
511 const QDomNode node = it->second;
512 mHasCycle = true;
513 bool resolved = true;
514 const auto deps { dependencies.value( idToSort ) };
515 for ( const QString &dep : deps )
516 {
517 if ( !sortedLayers.contains( dep ) )
518 {
519 resolved = false;
520 break;
521 }
522 }
523 if ( resolved ) // dependencies for this layer are resolved
524 {
525 sortedLayers << idToSort;
526 mSortedLayerNodes << node;
527 mSortedLayerIds << idToSort;
528 it = layersToSort.erase( it ); // erase and go to the next
529 mHasCycle = false;
530 }
531 else
532 {
533 ++it;
534 }
535 }
536 }
537}
538
540 : mHasCycle( false )
541 , mHasMissingDependency( false )
542{
543 init( doc );
544}
545
547 : mHasCycle( false )
548 , mHasMissingDependency( false )
549{
550 QString qgsProjectFile = fileName;
551 QgsProjectArchive archive;
552 if ( fileName.endsWith( ".qgz"_L1, Qt::CaseInsensitive ) )
553 {
554 archive.unzip( fileName );
555 qgsProjectFile = archive.projectFile();
556 }
557
558 QDomDocument doc;
559 QFile pFile( qgsProjectFile );
560 ( void ) pFile.open( QIODevice::ReadOnly );
561 ( void ) doc.setContent( &pFile );
562 init( doc );
563}
564
566{
567 return mDependentLayerIds.contains( layerId );
568}
LayerTreeInsertionMethod
Layer tree insertion methods.
Definition qgis.h:4666
@ TopOfTree
Layers are added at the top of the layer tree.
Definition qgis.h:4668
@ AboveInsertionPoint
Layers are added in the tree above the insertion point.
Definition qgis.h:4667
FilePathType
File path types.
Definition qgis.h:1755
@ Relative
Relative path.
Definition qgis.h:1757
@ Absolute
Absolute path.
Definition qgis.h:1756
LayerType
Types of layers that can be added to a map.
Definition qgis.h:206
@ Group
Composite group layer. Added in QGIS 3.24.
Definition qgis.h:214
@ Plugin
Plugin based layer.
Definition qgis.h:209
@ TiledScene
Tiled scene layer. Added in QGIS 3.34.
Definition qgis.h:215
@ Annotation
Contains freeform, georeferenced annotations. Added in QGIS 3.16.
Definition qgis.h:212
@ Vector
Vector layer.
Definition qgis.h:207
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
Definition qgis.h:211
@ Mesh
Mesh layer. Added in QGIS 3.2.
Definition qgis.h:210
@ Raster
Raster layer.
Definition qgis.h:208
@ PointCloud
Point cloud layer. Added in QGIS 3.18.
Definition qgis.h:213
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.
Definition qgsmaplayer.h:83
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.
Definition qgsarchive.h:111
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
Definition qgsproject.h:113
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)
Definition qgslogger.h:63
#define QgsDebugError(str)
Definition qgslogger.h:59
A structure to define the insertion point to the layer tree.