28#include "moc_qgsmapthemecollection.cpp"
30using namespace Qt::StringLiterals;
40 MapThemeLayerRecord layerRec( nodeLayer->
layer() );
41 layerRec.isVisible = nodeLayer->
isVisible();
42 layerRec.usingCurrentStyle =
true;
44 layerRec.expandedLayerNode = nodeLayer->
isExpanded();
45 const QStringList expandedLegendNodes = nodeLayer->
customProperty( u
"expandedLegendNodes"_s ).toStringList();
46 layerRec.expandedLegendItems = QSet<QString>( expandedLegendNodes.begin(), expandedLegendNodes.end() );
49 bool hasCheckableItems =
false;
50 bool someItemsUnchecked =
false;
51 QSet<QString> checkedItems;
52 const QList<QgsLayerTreeModelLegendNode *> layerLegendNodes = model->
layerLegendNodes( nodeLayer,
true );
55 if ( legendNode->
flags() & Qt::ItemIsUserCheckable )
57 hasCheckableItems =
true;
59 if ( legendNode->
data( Qt::CheckStateRole ).toInt() == Qt::Checked )
62 someItemsUnchecked =
true;
66 if ( hasCheckableItems && someItemsUnchecked )
68 layerRec.usingLegendItems =
true;
69 layerRec.checkedLegendItems = checkedItems;
79 lst.prepend( node->
name() );
82 return lst.join(
'/' );
87 const QList<QgsLayerTreeNode *> constChildren = parent->
children();
88 for ( QgsLayerTreeNode *node : constChildren )
94 rec.mExpandedGroupNodes.insert( _groupId( node ) );
96 rec.mCheckedGroupNodes.insert( _groupId( node ) );
102 rec.mLayerRecords << createThemeLayerRecord( nodeLayer, model );
120 if ( lr.layer() == layer )
132 const bool recordExists = findRecordForLayer( nodeLayer->
layer(), rec, layerRec );
148 if ( layerRec.usingCurrentStyle )
154 if ( layerRec.usingLegendItems )
157 const QList<QgsLayerTreeModelLegendNode *> constLayerLegendNodes = model->
layerLegendNodes( nodeLayer,
true );
158 for ( QgsLayerTreeModelLegendNode *legendNode : constLayerLegendNodes )
161 Qt::CheckState shouldHaveState = layerRec.checkedLegendItems.contains( ruleKey ) ? Qt::Checked : Qt::Unchecked;
169 const QList<QgsLayerTreeModelLegendNode *> constLayerLegendNodes = model->
layerLegendNodes( nodeLayer,
true );
170 for ( QgsLayerTreeModelLegendNode *legendNode : constLayerLegendNodes )
180 nodeLayer->
setExpanded( layerRec.expandedLayerNode );
181 nodeLayer->
setCustomProperty( u
"expandedLegendNodes"_s, QStringList( layerRec.expandedLegendItems.constBegin(), layerRec.expandedLegendItems.constEnd() ) );
188 const QList<QgsLayerTreeNode *> constChildren = parent->
children();
189 for ( QgsLayerTreeNode *node : constChildren )
232 return QList< QgsMapLayer * >();
234 return mProject->layerTreeRoot()->layerOrder();
240 const QList< QgsMapLayer * > visibleLayers = mProject->layerTreeRoot()->checkedLayers();
242 if ( allLayers.isEmpty() )
245 return visibleLayers;
249 QList< QgsMapLayer * > orderedVisibleLayers;
252 if ( visibleLayers.contains( layer ) )
253 orderedVisibleLayers << layer;
255 return orderedVisibleLayers;
262 return mMapThemes.contains( name );
267 mMapThemes.insert( name, state );
269 reconnectToLayersStyleManager();
276 if ( !mMapThemes.contains( name ) )
279 mMapThemes[name] = state;
281 reconnectToLayersStyleManager();
288 if ( !mMapThemes.contains( name ) || mMapThemes.contains( newName ) )
293 insert( newName, newState );
301 if ( !mMapThemes.contains( name ) )
304 mMapThemes.remove( name );
306 reconnectToLayersStyleManager();
314 reconnectToLayersStyleManager();
320 return mMapThemes.keys();
325 QStringList layerIds;
327 for (
QgsMapLayer *layer : constMapThemeVisibleLayers )
329 layerIds << layer->
id();
336 QList<QgsMapLayer *> layers;
337 const QList<MapThemeLayerRecord> recs = mMapThemes.value( name ).mLayerRecords;
339 if ( layerOrder.isEmpty() )
342 const QList<MapThemeLayerRecord> records { mMapThemes.value( name ).mLayerRecords };
345 if ( layerRec.isVisible && layerRec.layer() )
346 layers << layerRec.layer();
355 if ( layerRec.isVisible && layerRec.layer() == layer )
356 layers << layerRec.layer();
365void QgsMapThemeCollection::applyMapThemeCheckedLegendNodesToLayer(
const MapThemeLayerRecord &layerRec,
QgsMapLayer *layer )
368 if ( !vlayer || !vlayer->
renderer() )
375 bool someNodesUnchecked = layerRec.usingLegendItems;
381 bool shouldBeChecked = someNodesUnchecked ? layerRec.checkedLegendItems.contains( item.ruleKey() ) :
true;
382 if ( checked != shouldBeChecked )
390 QMap<QString, QString> styleOverrides;
391 if ( !mMapThemes.contains( presetName ) )
392 return styleOverrides;
394 const QList<MapThemeLayerRecord> records { mMapThemes.value( presetName ).mLayerRecords };
397 if ( !layerRec.layer() )
400 if ( layerRec.usingCurrentStyle )
407 applyMapThemeCheckedLegendNodesToLayer( layerRec, layer );
412 styleOverrides[layer->
id()] = layerStyle.
xmlData();
415 return styleOverrides;
418void QgsMapThemeCollection::reconnectToLayersStyleManager()
420 QSet<QgsMapLayer *> layers;
421 for (
const MapThemeRecord &rec : std::as_const( mMapThemes ) )
423 for (
const MapThemeLayerRecord &layerRec : std::as_const( rec.mLayerRecords ) )
425 if (
auto *lLayer = layerRec.layer() )
430 const QSet<QgsMapLayer *> constLayers = layers;
431 for ( QgsMapLayer *ml : constLayers )
441 QDomElement visPresetsElem = doc.firstChildElement( u
"qgis"_s ).firstChildElement( u
"visibility-presets"_s );
442 if ( visPresetsElem.isNull() )
445 QDomElement visPresetElem = visPresetsElem.firstChildElement( u
"visibility-preset"_s );
446 while ( !visPresetElem.isNull() )
448 QString presetName = visPresetElem.attribute( u
"name"_s );
450 mMapThemes.insert( presetName, rec );
452 visPresetElem = visPresetElem.nextSiblingElement( u
"visibility-preset"_s );
455 reconnectToLayersStyleManager();
461 QDomElement visPresetsElem = doc.createElement( u
"visibility-presets"_s );
463 QList< QString > keys = mMapThemes.keys();
465 std::sort( keys.begin(), keys.end() );
467 for (
const QString &grpName : std::as_const( keys ) )
470 QDomElement visPresetElem = doc.createElement( u
"visibility-preset"_s );
472 visPresetElem.setAttribute( u
"name"_s, grpName );
475 visPresetsElem.appendChild( visPresetElem );
478 doc.firstChildElement( u
"qgis"_s ).appendChild( visPresetsElem );
481void QgsMapThemeCollection::registryLayersRemoved(
const QStringList &layerIDs )
485 QSet< QString > changedThemes;
486 MapThemeRecordMap::iterator it = mMapThemes.begin();
487 for ( ; it != mMapThemes.end(); ++it )
489 MapThemeRecord &rec = it.value();
490 for (
int i = 0; i < rec.mLayerRecords.count(); ++i )
492 MapThemeLayerRecord &layerRec = rec.mLayerRecords[i];
493 if ( layerRec.layer() && layerIDs.contains( layerRec.layer()->id() ) )
495 rec.mLayerRecords.removeAt( i-- );
496 changedThemes << it.key();
501 for (
const QString &theme : std::as_const( changedThemes ) )
508void QgsMapThemeCollection::layerStyleRenamed(
const QString &oldName,
const QString &newName )
510 QgsMapLayerStyleManager *styleMgr = qobject_cast<QgsMapLayerStyleManager *>( sender() );
514 QSet< QString > changedThemes;
516 MapThemeRecordMap::iterator it = mMapThemes.begin();
517 for ( ; it != mMapThemes.end(); ++it )
520 for (
int i = 0; i < rec.mLayerRecords.count(); ++i )
523 if ( layerRec.layer() == styleMgr->
layer() )
525 if ( layerRec.currentStyle == oldName )
527 layerRec.currentStyle = newName;
528 changedThemes << it.key();
534 for (
const QString &theme : std::as_const( changedThemes ) )
543 for (
int i = 0; i < mLayerRecords.length(); ++i )
545 if ( mLayerRecords.at( i ).layer() == layer )
546 mLayerRecords.removeAt( i );
552 mLayerRecords.append( record );
557 QHash<QgsMapLayer *, MapThemeLayerRecord> validSet;
560 if (
auto *lLayer = layerRec.layer() )
561 validSet.insert( lLayer, layerRec );
570 bool expandedStateInfo =
false;
571 if ( element.hasAttribute( u
"has-expanded-info"_s ) )
572 expandedStateInfo = element.attribute( u
"has-expanded-info"_s ).toInt();
574 bool checkedStateInfo =
false;
575 if ( element.hasAttribute( u
"has-checked-group-info"_s ) )
576 checkedStateInfo = element.attribute( u
"has-checked-group-info"_s ).toInt();
578 QDomElement visPresetLayerElem = element.firstChildElement( u
"layer"_s );
579 while ( !visPresetLayerElem.isNull() )
581 QString layerID = visPresetLayerElem.attribute( u
"id"_s );
585 layerRecords[layerID].isVisible = visPresetLayerElem.attribute( u
"visible"_s, u
"1"_s ).toInt();
587 if ( visPresetLayerElem.hasAttribute( u
"style"_s ) )
590 layerRecords[layerID].currentStyle = visPresetLayerElem.attribute( u
"style"_s );
593 if ( visPresetLayerElem.hasAttribute( u
"expanded"_s ) )
594 layerRecords[layerID].expandedLayerNode = visPresetLayerElem.attribute( u
"expanded"_s ).toInt();
596 visPresetLayerElem = visPresetLayerElem.nextSiblingElement( u
"layer"_s );
599 QDomElement checkedLegendNodesElem = element.firstChildElement( u
"checked-legend-nodes"_s );
600 while ( !checkedLegendNodesElem.isNull() )
602 QSet<QString> checkedLegendNodes;
604 QDomElement checkedLegendNodeElem = checkedLegendNodesElem.firstChildElement( u
"checked-legend-node"_s );
605 while ( !checkedLegendNodeElem.isNull() )
607 checkedLegendNodes << checkedLegendNodeElem.attribute( u
"id"_s );
608 checkedLegendNodeElem = checkedLegendNodeElem.nextSiblingElement( u
"checked-legend-node"_s );
611 QString layerID = checkedLegendNodesElem.attribute( u
"id"_s );
612 if (
project->mapLayer( layerID ) )
615 layerRecords[layerID].checkedLegendItems = checkedLegendNodes;
617 checkedLegendNodesElem = checkedLegendNodesElem.nextSiblingElement( u
"checked-legend-nodes"_s );
621 if ( expandedStateInfo )
624 QDomElement expandedLegendNodesElem = element.firstChildElement( u
"expanded-legend-nodes"_s );
625 while ( !expandedLegendNodesElem.isNull() )
627 QSet<QString> expandedLegendNodes;
629 QDomElement expandedLegendNodeElem = expandedLegendNodesElem.firstChildElement( u
"expanded-legend-node"_s );
630 while ( !expandedLegendNodeElem.isNull() )
632 expandedLegendNodes << expandedLegendNodeElem.attribute( u
"id"_s );
633 expandedLegendNodeElem = expandedLegendNodeElem.nextSiblingElement( u
"expanded-legend-node"_s );
636 QString layerID = expandedLegendNodesElem.attribute( u
"id"_s );
637 if (
project->mapLayer( layerID ) )
639 layerRecords[layerID].expandedLegendItems = expandedLegendNodes;
641 expandedLegendNodesElem = expandedLegendNodesElem.nextSiblingElement( u
"expanded-legend-nodes"_s );
645 QDomElement expandedGroupNodesElem = element.firstChildElement( u
"expanded-group-nodes"_s );
646 if ( !expandedGroupNodesElem.isNull() )
648 QDomElement expandedGroupNodeElem = expandedGroupNodesElem.firstChildElement( u
"expanded-group-node"_s );
649 while ( !expandedGroupNodeElem.isNull() )
652 expandedGroupNodeElem = expandedGroupNodeElem.nextSiblingElement( u
"expanded-group-node"_s );
658 if ( checkedStateInfo )
661 QDomElement checkedGroupNodesElem = element.firstChildElement( u
"checked-group-nodes"_s );
662 if ( !checkedGroupNodesElem.isNull() )
664 QDomElement checkedGroupNodeElem = checkedGroupNodesElem.firstChildElement( u
"checked-group-node"_s );
665 while ( !checkedGroupNodeElem.isNull() )
668 checkedGroupNodeElem = checkedGroupNodeElem.nextSiblingElement( u
"checked-group-node"_s );
686 element.setAttribute( u
"has-expanded-info"_s, u
"1"_s );
688 element.setAttribute( u
"has-checked-group-info"_s, u
"1"_s );
691 if ( !layerRec.layer() )
693 QString layerID = layerRec.layer()->id();
694 QDomElement layerElem = document.createElement( u
"layer"_s );
695 layerElem.setAttribute( u
"id"_s, layerID );
696 layerElem.setAttribute( u
"visible"_s, layerRec.isVisible ? u
"1"_s : u
"0"_s );
697 if ( layerRec.usingCurrentStyle )
698 layerElem.setAttribute( u
"style"_s, layerRec.currentStyle );
699 element.appendChild( layerElem );
701 if ( layerRec.usingLegendItems )
703 QDomElement checkedLegendNodesElem = document.createElement( u
"checked-legend-nodes"_s );
704 checkedLegendNodesElem.setAttribute( u
"id"_s, layerID );
705 for (
const QString &checkedLegendNode : std::as_const( layerRec.checkedLegendItems ) )
707 QDomElement checkedLegendNodeElem = document.createElement( u
"checked-legend-node"_s );
708 checkedLegendNodeElem.setAttribute( u
"id"_s, checkedLegendNode );
709 checkedLegendNodesElem.appendChild( checkedLegendNodeElem );
711 element.appendChild( checkedLegendNodesElem );
716 layerElem.setAttribute( u
"expanded"_s, layerRec.expandedLayerNode ? u
"1"_s : u
"0"_s );
718 QDomElement expandedLegendNodesElem = document.createElement( u
"expanded-legend-nodes"_s );
719 expandedLegendNodesElem.setAttribute( u
"id"_s, layerID );
720 for (
const QString &expandedLegendNode : std::as_const( layerRec.expandedLegendItems ) )
722 QDomElement expandedLegendNodeElem = document.createElement( u
"expanded-legend-node"_s );
723 expandedLegendNodeElem.setAttribute( u
"id"_s, expandedLegendNode );
724 expandedLegendNodesElem.appendChild( expandedLegendNodeElem );
726 element.appendChild( expandedLegendNodesElem );
732 QDomElement checkedGroupElems = document.createElement( u
"checked-group-nodes"_s );
734 for (
const QString &groupId : _checkedGroupNodes )
736 QDomElement checkedGroupElem = document.createElement( u
"checked-group-node"_s );
737 checkedGroupElem.setAttribute( u
"id"_s, groupId );
738 checkedGroupElems.appendChild( checkedGroupElem );
740 element.appendChild( checkedGroupElems );
745 QDomElement expandedGroupElems = document.createElement( u
"expanded-group-nodes"_s );
747 for (
const QString &groupId : _expandedGroupNodes )
749 QDomElement expandedGroupElem = document.createElement( u
"expanded-group-node"_s );
750 expandedGroupElem.setAttribute( u
"id"_s, groupId );
751 expandedGroupElems.appendChild( expandedGroupElem );
753 element.appendChild( expandedGroupElems );
Abstract base class for all 2D vector feature renderers.
virtual QgsLegendSymbolList legendSymbolItems() const
Returns a list of symbology items for the legend.
virtual bool legendSymbolItemsCheckable() const
Returns true if symbology items in legend are checkable.
virtual bool legendSymbolItemChecked(const QString &key)
Returns true if the legend symbology item with the specified key is checked.
virtual void checkLegendSymbolItem(const QString &key, bool state=true)
Sets whether the legend symbology item with the specified ley should be checked.
Layer tree group node serves as a container for layers and further groups.
Layer tree node points to a map layer.
QgsMapLayer * layer() const
Returns the map layer associated with this node.
An abstract interface for legend items returned from QgsMapLayerLegend implementation.
virtual QVariant data(int role) const =0
Returns data associated with the item. Must be implemented in derived class.
@ RuleKey
Rule key of the node (QString).
virtual Qt::ItemFlags flags() const
Returns item flags associated with the item. Default implementation returns Qt::ItemIsEnabled.
virtual bool setData(const QVariant &value, int role)
Sets some data associated with the item. Default implementation does nothing and returns false.
A model representing the layer tree, including layers and groups of layers.
QList< QgsLayerTreeModelLegendNode * > layerLegendNodes(QgsLayerTreeLayer *nodeLayer, bool skipNodeEmbeddedInParent=false)
Returns filtered list of active legend nodes attached to a particular layer node (by default it retur...
Base class for nodes in a layer tree.
bool isVisible() const
Returns whether a node is really visible (ie checked and all its ancestors checked as well).
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.
QList< QgsLayerTreeNode * > children()
Gets list of children of the node. Children are owned by the parent.
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 setExpanded(bool expanded)
Sets whether the node should be shown as expanded or collapsed in GUI.
virtual QString name() const =0
Returns name of the node.
QgsLayerTreeNode * parent()
Gets pointer to the parent. If parent is nullptr, the node is a root node.
bool isExpanded() const
Returns whether the node should be shown as expanded or collapsed in GUI.
void setItemVisibilityChecked(bool checked)
Check or uncheck a node (independently of its ancestors or children).
bool itemVisibilityChecked() const
Returns whether a node is checked (independently of its ancestors or children).
void setItemVisibilityCheckedParentRecursive(bool checked)
Check or uncheck a node and all its parents.
static QgsLayerTreeLayer * toLayer(QgsLayerTreeNode *node)
Cast node to a layer.
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.
Stores information about one class/rule of a vector layer renderer in a unified way that can be used ...
QString currentStyle() const
Returns name of the current style.
bool setCurrentStyle(const QString &name)
Set a different style as the current style - will apply it to the layer.
void styleRenamed(const QString &oldName, const QString &newName)
Emitted when a style has been renamed.
QgsMapLayer * layer() const
Gets pointer to the associated map layer.
Restore overridden layer style on destruction.
void setOverrideStyle(const QString &style)
Temporarily apply a different style to the layer.
Stores style information (renderer, opacity, labeling, diagrams etc.) applicable to a map layer.
void readFromLayer(QgsMapLayer *layer)
Store layer's active style information in the instance.
QString xmlData() const
Returns XML content of the style.
Base class for all map layer types.
QgsMapLayerStyleManager * styleManager() const
Gets access to the layer's style manager.
Individual record of a visible layer in a map theme record.
void setLayer(QgsMapLayer *layer)
Sets the map layer for this record.
QgsMapLayer * layer() const
Returns map layer or nullptr if the layer does not exist anymore.
Individual map theme record of visible layers and styles.
void setExpandedGroupNodes(const QSet< QString > &expandedGroupNodes)
Sets a set of group identifiers for group nodes that should have expanded state.
void writeXml(QDomElement element, QDomDocument &document) const
Writes the map theme record to XML.
void removeLayerRecord(QgsMapLayer *layer)
Removes a record for layer if present.
QSet< QString > expandedGroupNodes() const
Returns a set of group identifiers for group nodes that should have expanded state (other group nodes...
void setCheckedGroupNodes(const QSet< QString > &checkedGroupNodes)
Sets a set of group identifiers for group nodes that should have checked state.
void setHasExpandedStateInfo(bool hasInfo)
Sets whether the map theme contains valid expanded/collapsed state of nodes.
bool hasExpandedStateInfo() const
Returns whether information about expanded/collapsed state of nodes has been recorded and thus whethe...
QList< QgsMapThemeCollection::MapThemeLayerRecord > layerRecords() const
Returns a list of records for all visible layer belonging to the theme.
QSet< QString > checkedGroupNodes() const
Returns a set of group identifiers for group nodes that should have checked state (other group nodes ...
void setLayerRecords(const QList< QgsMapThemeCollection::MapThemeLayerRecord > &records)
Sets layer records for the theme.
static MapThemeRecord readXml(const QDomElement &element, const QgsProject *project)
Reads the map theme record from XML.
void setHasCheckedStateInfo(bool hasInfo)
Sets whether the map theme contains valid checked/unchecked state of group nodes.
QHash< QgsMapLayer *, QgsMapThemeCollection::MapThemeLayerRecord > validLayerRecords() const
Returns set with only records for valid layers.
void addLayerRecord(const QgsMapThemeCollection::MapThemeLayerRecord &record)
Add a new record for a layer.
bool hasCheckedStateInfo() const
Returns whether information about checked/unchecked state of groups has been recorded and thus whethe...
void mapThemesChanged()
Emitted when map themes within the collection are changed.
static QgsMapThemeCollection::MapThemeRecord createThemeFromCurrentState(QgsLayerTreeGroup *root, QgsLayerTreeModel *model)
Static method to create theme from the current state of layer visibilities in layer tree,...
void insert(const QString &name, const QgsMapThemeCollection::MapThemeRecord &state)
Inserts a new map theme to the collection.
bool renameMapTheme(const QString &name, const QString &newName)
Renames the existing map theme called name to newName.
void mapThemeRenamed(const QString &name, const QString &newName)
Emitted when a map theme within the collection is renamed.
void clear()
Removes all map themes from the collection.
bool hasMapTheme(const QString &name) const
Returns whether a map theme with a matching name exists.
QList< QgsMapLayer * > mapThemeVisibleLayers(const QString &name) const
Returns the list of layers that are visible for the specified map theme.
void removeMapTheme(const QString &name)
Removes an existing map theme from collection.
QgsMapThemeCollection::MapThemeRecord mapThemeState(const QString &name) const
Returns the recorded state of a map theme.
void setProject(QgsProject *project)
Sets the project on which this map theme collection works.
QList< QgsMapLayer * > masterLayerOrder() const
Returns the master layer order (this will always match the project's QgsProject::layerOrder() ).
void writeXml(QDomDocument &doc) const
Writes the map theme collection state to XML.
void applyTheme(const QString &name, QgsLayerTreeGroup *root, QgsLayerTreeModel *model)
Apply theme given by its name and modify layer tree, current style of layers and checked legend items...
QMap< QString, QString > mapThemeStyleOverrides(const QString &name)
Gets layer style overrides (for QgsMapSettings) of the visible layers for given map theme.
QList< QgsMapLayer * > masterVisibleLayers() const
Returns the master list of visible layers.
QStringList mapThemeVisibleLayerIds(const QString &name) const
Returns the list of layer IDs that are visible for the specified map theme.
void update(const QString &name, const QgsMapThemeCollection::MapThemeRecord &state)
Updates a map theme within the collection.
void mapThemeChanged(const QString &theme)
Emitted when a map theme changes definition.
QgsMapThemeCollection(QgsProject *project=nullptr)
Create map theme collection that handles themes of the given project.
void projectChanged()
Emitted when the project changes.
void readXml(const QDomDocument &doc)
Reads the map theme collection state from XML.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
void layersWillBeRemoved(const QStringList &layerIds)
Emitted when one or more layers are about to be removed from the registry.
Represents a vector layer which manages a vector based dataset.
QgsFeatureRenderer * renderer()
Returns the feature renderer used for rendering the features in the layer in 2D map views.
QgsLayerTreeModelLegendNode * legendNode(const QString &rule, QgsLayerTreeModel &model)