28#include <QApplication>
29#include <QContextMenuEvent>
35#include "moc_qgslayertreeview.cpp"
37#ifdef ENABLE_MODELTEST
47 , mBlockDoubleClickTimer( new QTimer( this ) )
49 mBlockDoubleClickTimer->setSingleShot(
true );
50 mBlockDoubleClickTimer->setInterval( QApplication::doubleClickInterval() );
52 setHeaderHidden(
true );
54 setDragEnabled(
true );
55 setAcceptDrops(
true );
56 setDropIndicatorShown(
true );
57 setEditTriggers( EditKeyPressed );
58 setExpandsOnDoubleClick(
false );
61 header()->setStretchLastSection(
false );
62 header()->setSectionResizeMode( QHeaderView::ResizeToContents );
65 setVerticalScrollMode( QAbstractItemView::ScrollPerPixel );
67 setSelectionMode( ExtendedSelection );
68 setDefaultDropAction( Qt::MoveAction );
76 delete mBlockDoubleClickTimer;
81 if ( mBlockDoubleClickTimer->isActive() )
84 QTreeView::mouseDoubleClickEvent( event );
89 if ( mLayerTreeModel )
93 disconnect( mLayerTreeModel, &QAbstractItemModel::dataChanged,
this, &QgsLayerTreeViewBase::onDataChanged );
96 mLayerTreeModel = model;
103 connect( mLayerTreeModel, &QAbstractItemModel::dataChanged,
this, &QgsLayerTreeViewBase::onDataChanged );
110 return mLayerTreeModel;
118 const auto constChildren = node->
children();
126 if ( model() == mLayerTreeModel )
129 if ( QAbstractProxyModel *proxy = qobject_cast< QAbstractProxyModel *>( model() ) )
131 return proxy->mapToSource( index );
134 return QModelIndex();
140 if ( model() == mLayerTreeModel )
143 if ( QAbstractProxyModel *proxy = qobject_cast< QAbstractProxyModel *>( model() ) )
145 return proxy->mapFromSource( index );
148 return QModelIndex();
155 node->setExpanded( isExpanded( index ) );
160 QStringList lst = node->layerNode()->customProperty( QStringLiteral(
"expandedLegendNodes" ) ).toStringList();
161 const bool expanded = isExpanded( index );
162 const bool isInList = lst.contains( ruleKey );
163 if ( expanded && !isInList )
165 lst.append( ruleKey );
166 node->layerNode()->setCustomProperty( QStringLiteral(
"expandedLegendNodes" ), lst );
168 else if ( !expanded && isInList )
170 lst.removeAll( ruleKey );
171 node->layerNode()->setCustomProperty( QStringLiteral(
"expandedLegendNodes" ), lst );
179 if ( isExpanded( idx ) != expanded )
180 setExpanded( idx, expanded );
185 if ( !mLayerTreeModel )
191void QgsLayerTreeViewBase::onDataChanged(
const QModelIndex &,
const QModelIndex &,
const QVector<int> & )
193 mBlockDoubleClickTimer->start();
198 if ( mLayerTreeModel )
211 if ( mLayerTreeModel )
212 return mLayerTreeModel->node2index( node );
214 return QModelIndex();
224 if ( QItemSelectionModel *selectModel = selectionModel() )
225 return index2node( selectModel->currentIndex() );
232 if ( !mLayerTreeModel )
235 QModelIndexList mapped;
236 if ( QItemSelectionModel *selectModel = selectionModel() )
238 const QModelIndexList selected = selectModel->selectedIndexes();
239 mapped.reserve( selected.size() );
240 for (
const QModelIndex &index : selected )
244 return mLayerTreeModel->indexes2nodes( mapped, skipInternal );
256 if ( mLayerTreeModel && index.isValid() )
286 if ( QItemSelectionModel *selectModel = selectionModel() )
301 QList<QgsLayerTreeLayer *> layerNodes;
302 const QList<QgsLayerTreeNode *> constSelectedNodes =
selectedNodes();
303 layerNodes.reserve( constSelectedNodes.size() );
314 QList<QgsMapLayer *> list;
316 list.reserve( constSelectedLayerNodes.size() );
320 list << node->layer();
332 if ( mLayerTreeModel )
333 return mLayerTreeModel->legendNode2index( legendNode );
334 return QModelIndex();
341 setCurrentIndex( QModelIndex() );
350 if ( !layer || !mLayerTreeModel )
352 setCurrentIndex( QModelIndex() );
365 if ( QItemSelectionModel *selectModel = selectionModel() )
373 QList<QgsLayerTreeModelLegendNode *> res;
374 QItemSelectionModel *selectModel = selectionModel();
375 if ( !mLayerTreeModel || !selectModel )
378 const QModelIndexList selected = selectModel->selectedIndexes();
379 res.reserve( selected.size() );
380 for (
const QModelIndex &index : selected )
385 res.push_back( node );
394 QModelIndexList mapped;
395 QItemSelectionModel *selectModel = selectionModel();
396 if ( !mLayerTreeModel || !selectModel )
397 return QList<QgsMapLayer *>();
399 const QModelIndexList selected = selectModel->selectedIndexes();
400 mapped.reserve( selected.size() );
401 for (
const QModelIndex &index : selected )
406 const QList<QgsLayerTreeNode *> nodes = mLayerTreeModel->indexes2nodes( mapped,
false );
408 return qgis::setToList( layersSet );
418 const auto constLayerLegendNodes = model->
layerLegendNodes( nodeLayer,
true );
422 if ( !parentKey.isEmpty() && !lst.contains( parentKey ) )
431 const auto constChildren = parent->
children();
434 node->setExpanded( expanded );
478 setStyle(
new QgsLayerTreeViewProxyStyle(
this ) );
482 connect( horizontalScrollBar(), &QScrollBar::valueChanged,
this, &QgsLayerTreeView::onHorizontalScroll );
497 proxyModel->setShowPrivateLayers( mShowPrivateLayers );
498 proxyModel->setHideValidLayers( mHideValidLayers );
508 mMessageBar->pushMessage( message, level );
515#ifdef ENABLE_MODELTEST
516 new ModelTest( mProxyModel,
this );
519 QTreeView::setModel( mProxyModel );
526 connect( treeModel, &QAbstractItemModel::dataChanged,
this, &QgsLayerTreeView::onDataChanged );
540 if ( !layer || !layerModel )
553 const QModelIndex idx = indexAt( event->pos() );
554 if ( !idx.isValid() )
555 setCurrentIndex( QModelIndex() );
562 if ( menu->actions().count() != 0 )
563 menu->exec( mapToGlobal( event->pos() ) );
573 if ( !parentNode || !layerModel )
582 const int widgetsCount = layer->customProperty( QStringLiteral(
"embeddedWidgets/count" ), 0 ).toInt();
583 QList<QgsLayerTreeModelLegendNode *> legendNodes = layerModel->
layerLegendNodes( nodeLayer,
true );
584 for (
int i = 0; i < widgetsCount; ++i )
586 const QString providerId = layer->customProperty( QStringLiteral(
"embeddedWidgets/%1/id" ).arg( i ) ).toString();
590 QWidget *wdgt = provider->createWidget( layer, i );
607 setIndexWidget( index, wdgt );
617 const QStringList expandedNodeKeys = parentNode->
customProperty( QStringLiteral(
"expandedLegendNodes" ) ).toStringList();
618 if ( expandedNodeKeys.isEmpty() )
625 if ( expandedNodeKeys.contains( ruleKey ) )
631 QList<QgsLayerTreeNode *> children = parentNode->
children();
632 for (
int i = start; i <= end && i < children.count(); ++i )
654 const QString layerCurrentID = layerCurrent ? layerCurrent->
id() : QString();
659 QModelIndex proxyModelNodeLayerIndex;
664 proxyModelNodeLayerIndex =
node2index( nodeLayer );
667 if ( !proxyModelNodeLayerIndex.isValid() )
675 layerModel->
setCurrentIndex( mProxyModel->mapToSource( proxyModelNodeLayerIndex ) );
683void QgsLayerTreeView::onCustomPropertyChanged(
QgsLayerTreeNode *node,
const QString &key )
686 if ( key != QLatin1String(
"expandedLegendNodes" ) || !
QgsLayerTree::isLayer( node ) || !layerModel )
689 const QSet<QString> expandedLegendNodes = qgis::listToSet( node->
customProperty( QStringLiteral(
"expandedLegendNodes" ) ).toStringList() );
695 if ( !key.isEmpty() )
696 setExpanded(
legendNode2index( legendNode ), expandedLegendNodes.contains( key ) );
707 viewport()->repaint();
710 viewport()->repaint();
726QStringList QgsLayerTreeView::viewOnlyCustomProperties()
728 return QStringList() << QStringLiteral(
"expandedLegendNodes" );
745 if ( mMessageBar == messageBar )
748 mMessageBar = messageBar;
756 mMessageBar->pushMessage( message, level );
765 mShowPrivateLayers = showPrivate;
766 mProxyModel->setShowPrivateLayers( showPrivate );
774 mHideValidLayers = hideValid;
775 mProxyModel->setHideValidLayers( mHideValidLayers );
780 return mShowPrivateLayers;
785 return mHideValidLayers;
799 if ( event->modifiers() & Qt::ControlModifier )
803 QTreeView::mouseReleaseEvent( event );
809 if ( event->key() == Qt::Key_Space )
811 const QList<QgsLayerTreeNode *> constSelectedNodes =
selectedNodes();
813 if ( !constSelectedNodes.isEmpty() )
815 const bool isFirstNodeChecked = constSelectedNodes[0]->itemVisibilityChecked();
830 if ( event->modifiers() & Qt::ControlModifier )
834 QTreeView::keyPressEvent( event );
840 if ( event->mimeData()->hasUrls() || event->mimeData()->hasFormat( QStringLiteral(
"application/x-vnd.qgis.qgis.uri" ) ) )
843 if ( !event->mimeData()->hasFormat( QStringLiteral(
"application/qgis.layertreemodeldata" ) ) )
849 QTreeView::dragEnterEvent( event );
854 if ( event->mimeData()->hasUrls() || event->mimeData()->hasFormat( QStringLiteral(
"application/x-vnd.qgis.qgis.uri" ) ) )
857 if ( !event->mimeData()->hasFormat( QStringLiteral(
"application/qgis.layertreemodeldata" ) ) )
863 QTreeView::dragMoveEvent( event );
868 if ( event->mimeData()->hasUrls() || event->mimeData()->hasFormat( QStringLiteral(
"application/x-vnd.qgis.qgis.uri" ) ) )
871 if ( !event->mimeData()->hasFormat( QStringLiteral(
"application/qgis.layertreemodeldata" ) ) )
875 QModelIndex index = indexAt( event->pos() );
876 if ( index.isValid() )
878 setCurrentIndex( index );
885 if ( event->keyboardModifiers() & Qt::AltModifier || event->keyboardModifiers() & Qt::ControlModifier )
889 QTreeView::dropEvent( event );
902 header()->setMinimumSectionSize( viewport()->width() );
903 QTreeView::resizeEvent( event );
906void QgsLayerTreeView::onHorizontalScroll(
int value )
909 viewport()->update();
912void QgsLayerTreeView::onDataChanged(
const QModelIndex &topLeft,
const QModelIndex &bottomRight,
const QVector<int> &roles )
915 Q_UNUSED( bottomRight )
921 if ( roles.contains( Qt::SizeHintRole ) )
922 viewport()->update();
929void QgsLayerTreeView::checkModel()
931 std::function<void( QgsLayerTreeNode *,
int )> debug;
932 debug = [ & ]( QgsLayerTreeNode * node,
int depth )
935 qDebug() <<
"----------------------------------------------";
942 for (
int i = 0; i < mProxyModel->rowCount(
node2index( node ) ); i++ )
946 debug( childNode, depth + 1 );
948 qDebug() <<
"Warning no child node!";
961 : QSortFilterProxyModel( parent )
962 , mLayerTreeModel( treeModel )
964 setSourceModel( treeModel );
969 if ( filterText == mFilterText )
972 mFilterText = filterText;
978 QgsLayerTreeNode *node = mLayerTreeModel->index2node( mLayerTreeModel->index( sourceRow, 0, sourceParent ) );
996 if ( !mFilterText.isEmpty() && !layer->
name().contains( mFilterText, Qt::CaseInsensitive ) )
1002 if ( mHideValidLayers && layer->
isValid() )
1011 return mShowPrivateLayers;
1016 if ( showPrivate == mShowPrivateLayers )
1019 mShowPrivateLayers = showPrivate;
1025 return mHideValidLayers;
1030 if ( hideValid == mHideValidLayers )
1033 mHideValidLayers = hideValid;
MessageLevel
Level for messages This will be used both for message log and message bar in application.
@ Info
Information message.
static const double UI_SCALE_FACTOR
UI scaling factor.
static QgsLayerTreeEmbeddedWidgetRegistry * layerTreeEmbeddedWidgetRegistry()
Returns the global layer tree embedded widget registry, used for registering widgets that may be embe...
Layer tree group node serves as a container for layers and further groups.
QgsLayerTreeLayer * findLayer(QgsMapLayer *layer) const
Find layer node representing the map layer.
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.
@ ParentRuleKey
Rule key of the parent legend node - for legends with tree hierarchy (QString). Added in 2....
@ RuleKey
Rule key of the node (QString).
QgsLayerTreeLayer * layerNode() const
Returns pointer to the parent layer node.
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...
Qt::ItemFlags flags(const QModelIndex &index) const override
void setCurrentIndex(const QModelIndex ¤tIndex)
Sets index of the current item. May be used by view. Item marked as current is underlined.
void setFlags(QgsLayerTreeModel::Flags f)
Sets OR-ed combination of model flags.
QgsLayerTree * rootGroup() const
Returns pointer to the root node of the layer tree. Always a non nullptr value.
int rowCount(const QModelIndex &parent=QModelIndex()) const override
void messageEmitted(const QString &message, Qgis::MessageLevel level=Qgis::MessageLevel::Info, int duration=5)
Emits a message than can be displayed to the user in a GUI class.
static QgsLayerTreeModelLegendNode * index2legendNode(const QModelIndex &index)
Returns legend node for given index.
void refreshLayerLegend(QgsLayerTreeLayer *nodeLayer)
Force a refresh of legend nodes of a layer node.
bool testFlag(Flag f) const
Check whether a flag is enabled.
@ ActionHierarchical
Check/uncheck action has consequences on children (or parents for leaf node).
@ UseEmbeddedWidgets
Layer nodes may optionally include extra embedded widgets (if used in QgsLayerTreeView)....
void addTargetScreenProperties(const QgsScreenProperties &properties)
Adds additional target screen properties to use when generating icons for Qt::DecorationRole data.
Base class for nodes in a layer tree.
@ NodeGroup
Container of other groups and layers.
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.
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.
NodeType nodeType() const
Find out about type of the node. It is usually shorter to use convenience functions from QgsLayerTree...
void customPropertyChanged(QgsLayerTreeNode *node, const QString &key)
Emitted when a custom property of a node within the tree has been changed or removed.
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).
void expandedChanged(QgsLayerTreeNode *node, bool expanded)
Emitted when the collapsed/expanded state of a node within the tree has been changed.
A proxy model for QgsLayerTreeModel, supporting private layers and text filtering.
bool showPrivateLayers() const
Returns if private layers are shown.
bool filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const override
void setFilterText(const QString &filterText=QString())
Sets filter to filterText.
bool hideValidLayers() const
Returns if valid layers should be hidden (i.e.
void setHideValidLayers(bool hideValid)
Sets whether valid layers should be hidden (i.e.
virtual bool nodeShown(QgsLayerTreeNode *node) const
Returns true if the specified node should be shown.
void setShowPrivateLayers(bool showPrivate)
Determines if private layers are shown.
QgsLayerTreeProxyModel(QgsLayerTreeModel *treeModel, QObject *parent)
Constructs QgsLayerTreeProxyModel with source model treeModel and a parent.
static QSet< QgsMapLayer * > collectMapLayersRecursive(const QList< QgsLayerTreeNode * > &nodes)
Returns map layers from the given list of layer tree nodes.
void updateExpandedStateToNode(const QModelIndex &index)
Stores the expanded state to a node with matching index.
QList< QgsLayerTreeNode * > selectedNodes(bool skipInternal=false) const
Returns the list of selected layer tree nodes.
QgsLayerTreeNode * currentNode() const
Returns the current node.
QgsLayerTreeGroup * currentGroupNode() const
Returns the current group node.
QList< QgsMapLayer * > selectedLayers() const
Returns the list of selected layers.
QModelIndex node2sourceIndex(QgsLayerTreeNode *node) const
Returns the layer tree source model index for a given node.
QgsLayerTreeModelLegendNode * index2legendNode(const QModelIndex &index) const
Returns legend node for given view index.
void mouseDoubleClickEvent(QMouseEvent *event) override
void setLayerTreeModel(QgsLayerTreeModel *model)
Associates a layer tree model with the view.
QgsMapLayer * layerForIndex(const QModelIndex &index) const
Returns the map layer corresponding to a view index.
QModelIndex legendNode2index(QgsLayerTreeModelLegendNode *legendNode)
Returns the view index for a given legend node.
~QgsLayerTreeViewBase() override
void setCurrentNode(QgsLayerTreeNode *node)
Sets the currently selected node.
void onExpandedChanged(QgsLayerTreeNode *node, bool expanded)
Called when the expanded state changes for a node.
QModelIndex layerTreeModelIndexToViewIndex(const QModelIndex &index) const
Returns the layer tree model index corresponding with a view index.
QModelIndex legendNode2sourceIndex(QgsLayerTreeModelLegendNode *legendNode)
Returns the layer tree source model index for a given legend node.
QList< QgsLayerTreeModelLegendNode * > selectedLegendNodes() const
Returns the list of selected legend nodes.
QModelIndex node2index(QgsLayerTreeNode *node) const
Returns the view model index for a given node.
void onModelReset()
Called when the model is reset.
void updateExpandedStateFromNode(QgsLayerTreeNode *node)
Updates the expanded state from a node.
QList< QgsLayerTreeLayer * > selectedLayerNodes() const
Returns the list of selected nodes filtered to just layer nodes (QgsLayerTreeLayer).
QModelIndex viewIndexToLayerTreeModelIndex(const QModelIndex &index) const
Returns the view index corresponding with a layer tree model index.
QgsLayerTreeViewBase(QWidget *parent=nullptr)
Constructor for QgsLayerTreeViewBase.
QgsLayerTreeViewDefaultActions * defaultActions()
Gets access to the default actions that may be used with the tree view.
void setCurrentLayer(QgsMapLayer *layer)
Sets the currently selected layer.
void collapseAllNodes()
Enhancement of QTreeView::collapseAll() that also records expanded state in layer tree nodes.
QList< QgsMapLayer * > selectedLayersRecursive() const
Gets list of selected layers, including those that are not directly selected, but their ancestor grou...
QgsLayerTreeNode * index2node(const QModelIndex &index) const
Returns the layer tree node for given view index.
QgsLayerTreeViewDefaultActions * mDefaultActions
helper class with default actions. Lazily initialized.
QgsLayerTreeModelLegendNode * currentLegendNode() const
Gets current legend node.
void expandAllNodes()
Enhancement of QTreeView::expandAll() that also records expanded state in layer tree nodes.
QgsMapLayer * currentLayer() const
Returns the currently selected layer, or nullptr if no layers is selected.
QgsLayerTreeModel * layerTreeModel() const
Returns the associated layer tree model.
Serves as a factory of actions that can be used together with a layer tree view.
Indicator that can be used in a layer tree view to display icons next to items of the layer tree.
void changed()
Emitted when the indicator changes state (e.g.
QList< QgsLayerTreeViewIndicator * > indicators(QgsLayerTreeNode *node) const
Returns list of indicators associated with a particular layer tree node.
void currentLayerChanged(QgsMapLayer *layer)
Emitted when a current layer is changed.
void contextMenuAboutToShow(QMenu *menu)
Emitted when the context menu is about to show.
QString mCurrentLayerID
Keeps track of current layer ID (to check when to emit signal about change of current layer).
void datasetsDropped(QDropEvent *event)
Emitted when datasets are dropped onto the layer tree view.
void setModel(QAbstractItemModel *model) override
Overridden setModel() from base class.
void removeIndicator(QgsLayerTreeNode *node, QgsLayerTreeViewIndicator *indicator)
Removes a previously added indicator to a layer tree node.
QHash< QgsLayerTreeNode *, QList< QgsLayerTreeViewIndicator * > > mIndicators
Storage of indicators used with the tree view.
void refreshLayerSymbology(const QString &layerId)
Force refresh of layer symbology. Normally not needed as the changes of layer's renderer are monitore...
void dropEvent(QDropEvent *event) override
QgsLayerTreeView(QWidget *parent=nullptr)
Constructor for QgsLayerTreeView.
QgsLayerTreeViewMenuProvider * menuProvider() const
Returns pointer to the context menu provider. May be nullptr.
void resizeEvent(QResizeEvent *event) override
void mouseReleaseEvent(QMouseEvent *event) override
bool hideValidLayers() const
Returns if valid layers should be hidden (i.e.
void setMenuProvider(QgsLayerTreeViewMenuProvider *menuProvider)
Sets provider for context menu. Takes ownership of the instance.
void setLayerVisible(QgsMapLayer *layer, bool visible)
Convenience methods which sets the visible state of the specified map layer.
void dragMoveEvent(QDragMoveEvent *event) override
QPoint mLastReleaseMousePos
Used by the item delegate for identification of which indicator has been clicked.
friend class QgsLayerTreeViewItemDelegate
QgsLayerTreeProxyModel * proxyModel() const
Returns the proxy model used by the view.
QgsLayerTreeViewMenuProvider * mMenuProvider
Context menu provider. Owned by the view.
void contextMenuEvent(QContextMenuEvent *event) override
void keyPressEvent(QKeyEvent *event) override
void setMessageBar(QgsMessageBar *messageBar)
Set the message bar to display messages from the layer tree.
void addIndicator(QgsLayerTreeNode *node, QgsLayerTreeViewIndicator *indicator)
Adds an indicator to the given layer tree node.
void setHideValidLayers(bool hideValid)
Sets whether valid layers should be hidden (i.e.
void modelRowsInserted(const QModelIndex &index, int start, int end)
void setLayerMarkWidth(int width)
Set width of contextual menu mark, at right of layer node items.
void setShowPrivateLayers(bool showPrivate)
Set the show private layers to showPrivate.
~QgsLayerTreeView() override
void dragEnterEvent(QDragEnterEvent *event) override
bool showPrivateLayers() const
Returns the show private layers status.
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.
Base class for all map layer types.
QgsMapLayer::LayerFlags flags
@ Private
Determines if the layer is meant to be exposed to the GUI, i.e. visible in the layer legend tree.
A bar for displaying non-blocking messages to the user.
Stores properties relating to a screen.