QGIS API Documentation 3.41.0-Master (cea29feecf2)
Loading...
Searching...
No Matches
qgshistorywidget.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgshistorywidget.cpp
3 ------------------
4 Date : April 2023
5 Copyright : (C) 2023 Nyall Dawson
6 Email : nyall dot dawson 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
16#include "qgshistorywidget.h"
17#include "moc_qgshistorywidget.cpp"
18#include "qgsgui.h"
20#include "qgshistoryentrynode.h"
21#include "qgssettings.h"
22#include "qgsnative.h"
23
24#include <QTextBrowser>
25#include <QtGlobal>
26#include <QMenu>
27#include <QFileInfo>
28#include <QDesktopServices>
29
30QgsHistoryWidget::QgsHistoryWidget( const QString &providerId, Qgis::HistoryProviderBackends backends, QgsHistoryProviderRegistry *registry, const QgsHistoryWidgetContext &context, QWidget *parent )
31 : QgsPanelWidget( parent )
32 , mContext( context )
33{
34 setupUi( this );
35
36 mModel = new QgsHistoryEntryModel( providerId, backends, registry, mContext, this );
37 mProxyModel = new QgsHistoryEntryProxyModel( this );
38 mProxyModel->setSourceModel( mModel );
39
40 mTreeView->setModel( mProxyModel );
41
42 mFilterEdit->setShowClearButton( true );
43 mFilterEdit->setShowSearchIcon( true );
44 connect( mFilterEdit, &QLineEdit::textChanged, mProxyModel, &QgsHistoryEntryProxyModel::setFilter );
45 connect( mTreeView->selectionModel(), &QItemSelectionModel::currentChanged, this, &QgsHistoryWidget::currentItemChanged );
46 connect( mTreeView, &QTreeView::doubleClicked, this, &QgsHistoryWidget::nodeDoubleClicked );
47 mTreeView->setExpandsOnDoubleClick( false );
48
49 mTreeView->setContextMenuPolicy( Qt::CustomContextMenu );
50 connect( mTreeView, &QWidget::customContextMenuRequested, this, &QgsHistoryWidget::showNodeContextMenu );
51
52 // expand first group (usually most recent date group)
53 const QModelIndex firstGroup = mProxyModel->index( 0, 0, QModelIndex() );
54 mTreeView->expand( firstGroup );
55
56 QgsSettings settings;
57 mSplitter->restoreState( settings.value( QStringLiteral( "history/splitterState%1" ).arg( providerId ) ).toByteArray() );
58
59 connect( mSplitter, &QSplitter::splitterMoved, this, [providerId, this] {
60 QgsSettings settings;
61 settings.setValue( QStringLiteral( "history/splitterState%1" ).arg( providerId ), mSplitter->saveState() );
62 } );
63}
64
65void QgsHistoryWidget::currentItemChanged( const QModelIndex &selected, const QModelIndex & )
66{
67 QWidget *newWidget = nullptr;
68 if ( QgsHistoryEntryNode *node = mModel->index2node( mProxyModel->mapToSource( selected ) ) )
69 {
70 newWidget = node->createWidget( mContext );
71 if ( !newWidget )
72 {
73 const QString html = node->html( mContext );
74 if ( !html.isEmpty() )
75 {
76 QTextBrowser *htmlBrowser = new QTextBrowser();
77 htmlBrowser->setOpenLinks( false );
78 htmlBrowser->setHtml( html );
79 connect( htmlBrowser, &QTextBrowser::anchorClicked, this, &QgsHistoryWidget::urlClicked );
80
81 newWidget = htmlBrowser;
82 }
83 }
84 if ( newWidget )
85 {
86 mContainerStackedWidget->addWidget( newWidget );
87 mContainerStackedWidget->setCurrentWidget( newWidget );
88 }
89 }
90
91 if ( !newWidget )
92 {
93 //remove current widget, if any
94 if ( mContainerStackedWidget->count() > 1 )
95 {
96 mContainerStackedWidget->removeWidget( mContainerStackedWidget->widget( 1 ) );
97 mContainerStackedWidget->setCurrentIndex( 0 );
98 }
99 }
100}
101
102void QgsHistoryWidget::nodeDoubleClicked( const QModelIndex &index )
103{
104 if ( QgsHistoryEntryNode *node = mModel->index2node( mProxyModel->mapToSource( index ) ) )
105 {
106 if ( node->doubleClicked( mContext ) )
107 return; // double-click handled
108 }
109
110 // otherwise double-clicks expands/collapses the node
111 if ( mTreeView->isExpanded( index ) )
112 mTreeView->collapse( index );
113 else
114 mTreeView->expand( index );
115}
116
117void QgsHistoryWidget::showNodeContextMenu( const QPoint &pos )
118{
119 if ( QgsHistoryEntryNode *node = mModel->index2node( mProxyModel->mapToSource( mTreeView->currentIndex() ) ) )
120 {
121 QMenu *menu = new QMenu();
122
123 node->populateContextMenu( menu, mContext );
124 if ( !menu->isEmpty() )
125 {
126 menu->exec( mTreeView->mapToGlobal( pos ) );
127 }
128 delete menu;
129 }
130}
131
132void QgsHistoryWidget::urlClicked( const QUrl &url )
133{
134 const QFileInfo file( url.toLocalFile() );
135 if ( file.exists() && !file.isDir() )
136 QgsGui::nativePlatformInterface()->openFileExplorerAndSelectFile( url.toLocalFile() );
137 else
138 QDesktopServices::openUrl( url );
139}
140
141//
142// QgsHistoryEntryProxyModel
143//
144
146QgsHistoryEntryProxyModel::QgsHistoryEntryProxyModel( QObject *parent )
147 : QSortFilterProxyModel( parent )
148{
149 setDynamicSortFilter( true );
150 setRecursiveFilteringEnabled( true );
151}
152
153void QgsHistoryEntryProxyModel::setFilter( const QString &filter )
154{
155 if ( filter == mFilter )
156 return;
157
158 mFilter = filter;
159 invalidateFilter();
160}
161
162bool QgsHistoryEntryProxyModel::filterAcceptsRow( int source_row, const QModelIndex &source_parent ) const
163{
164 if ( mFilter.isEmpty() )
165 return true;
166
167 const QModelIndex sourceIndex = sourceModel()->index( source_row, 0, source_parent );
168 if ( QgsHistoryEntryNode *node = qobject_cast<QgsHistoryEntryModel *>( sourceModel() )->index2node( sourceIndex ) )
169 {
170 if ( !node->matchesString( mFilter ) )
171 {
172 return false;
173 }
174 }
175 return true;
176}
QFlags< HistoryProviderBackend > HistoryProviderBackends
Definition qgis.h:3304
static QgsNative * nativePlatformInterface()
Returns the global native interface, which offers abstraction to the host OS's underlying public inte...
Definition qgsgui.cpp:85
An item model representing history entries in a hierarchical tree structure.
QgsHistoryEntryNode * index2node(const QModelIndex &index) const
Returns node for given index.
Base class for nodes representing a QgsHistoryEntry.
The QgsHistoryProviderRegistry is a registry for objects which track user history (i....
Contains settings which reflect the context in which a history widget is shown, e....
QgsHistoryWidget(const QString &providerId=QString(), Qgis::HistoryProviderBackends backends=Qgis::HistoryProviderBackend::LocalProfile, QgsHistoryProviderRegistry *registry=nullptr, const QgsHistoryWidgetContext &context=QgsHistoryWidgetContext(), QWidget *parent=nullptr)
Constructor for QgsHistoryWidget, with the specified parent widget.
Base class for any widget that can be shown as a inline panel.
This class is a composition of two QSettings instances:
Definition qgssettings.h:64
QVariant value(const QString &key, const QVariant &defaultValue=QVariant(), Section section=NoSection) const
Returns the value for setting key.
void setValue(const QString &key, const QVariant &value, QgsSettings::Section section=QgsSettings::NoSection)
Sets the value of setting key to value.