improve QgsDockableWidgetHelper API

uses dynamic settings to save the state (docked, area, geometry)
now if the dock is floating, its geometry is also reset
This commit is contained in:
Denis Rouzaud 2025-01-23 14:43:46 +01:00 committed by Nyall Dawson
parent dc64593b19
commit 7ae0fbfa46
4 changed files with 120 additions and 13 deletions

View File

@ -57,6 +57,7 @@
#include "qgsdockwidget.h"
#include "qgssettingsregistrycore.h"
QgsExpressionContext QgsAttributeTableDialog::createExpressionContext() const
{
QgsExpressionContext expContext;
@ -283,9 +284,11 @@ QgsAttributeTableDialog::QgsAttributeTableDialog( QgsVectorLayer *layer, QgsAttr
// info from table to application
connect( this, &QgsAttributeTableDialog::saveEdits, this, [=] { QgisApp::instance()->saveEdits(); } );
const bool isDocked = initiallyDocked ? *initiallyDocked : settings.value( QStringLiteral( "qgis/dockAttributeTable" ), false ).toBool();
toggleShortcuts( !isDocked );
mDockableWidgetHelper = new QgsDockableWidgetHelper( isDocked, windowTitle(), this, QgisApp::instance(), Qt::BottomDockWidgetArea, QStringList(), !initiallyDocked, QStringLiteral( "Windows/BetterAttributeTable/geometry" ) );
QgsDockableWidgetHelper::OpeningMode openingMode = QgsDockableWidgetHelper::OpeningMode::RespectSetting;
if ( initiallyDocked )
openingMode = *initiallyDocked ? QgsDockableWidgetHelper::OpeningMode::ForceDocked : QgsDockableWidgetHelper::OpeningMode::ForceDialog;
mDockableWidgetHelper = new QgsDockableWidgetHelper( windowTitle(), this, QgisApp::instance(), QStringLiteral( "attribute-table" ), QStringList(), openingMode, true, Qt::BottomDockWidgetArea );
toggleShortcuts( !mDockableWidgetHelper->isDocked() );
connect( mDockableWidgetHelper, &QgsDockableWidgetHelper::closed, this, [=]() {
close();
} );

View File

@ -26,6 +26,9 @@
///@cond PRIVATE
const QgsSettingsEntryBool *QgsDockableWidgetHelper::sSettingsIsDocked = new QgsSettingsEntryBool( QStringLiteral( "is-docked" ), QgsDockableWidgetHelper::sTtreeDockConfigs, false );
const QgsSettingsEntryVariant *QgsDockableWidgetHelper::sSettingsWindowGeometry = new QgsSettingsEntryVariant( QStringLiteral( "geometry" ), QgsDockableWidgetHelper::sTtreeDockConfigs );
const QgsSettingsEntryEnumFlag<Qt::DockWidgetArea> *QgsDockableWidgetHelper::sSettingsDockArea = new QgsSettingsEntryEnumFlag<Qt::DockWidgetArea>( QStringLiteral( "area" ), QgsDockableWidgetHelper::sTtreeDockConfigs, Qt::RightDockWidgetArea );
std::function<void( Qt::DockWidgetArea, QDockWidget *, const QStringList &, bool )> QgsDockableWidgetHelper::sAddTabifiedDockWidgetFunction = []( Qt::DockWidgetArea, QDockWidget *, const QStringList &, bool ) {};
std::function<QString()> QgsDockableWidgetHelper::sAppStylesheetFunction = [] { return QString(); };
@ -40,11 +43,33 @@ QgsDockableWidgetHelper::QgsDockableWidgetHelper( bool isDocked, const QString &
, mWindowTitle( windowTitle )
, mOwnerWindow( ownerWindow )
, mTabifyWith( tabifyWith )
, mRaiseTab( raiseTab )
, mWindowGeometrySettingsKey( windowGeometrySettingsKey )
, mUuid( QUuid::createUuid().toString() )
, mUsePersistentWidget( usePersistentWidget )
{
mOptions.setFlag( Option::RaiseTab, raiseTab );
mOptions.setFlag( Option::PermanentWidget, usePersistentWidget );
toggleDockMode( isDocked );
}
QgsDockableWidgetHelper::QgsDockableWidgetHelper( const QString &windowTitle, QWidget *widget, QMainWindow *ownerWindow, const QString &dockId, const QStringList &tabifyWith, OpeningMode openingMode, bool defaultIsDocked, Qt::DockWidgetArea defaultDockArea, Options options )
: QObject( nullptr )
, mWidget( widget )
, mDialogGeometry( 0, 0, 0, 0 )
, mWindowTitle( windowTitle )
, mOwnerWindow( ownerWindow )
, mTabifyWith( tabifyWith )
, mOptions( options )
, mUuid( QUuid::createUuid().toString() )
, mSettingKeyDockId( dockId )
{
bool isDocked = sSettingsIsDocked->valueWithDefaultOverride( defaultIsDocked, mSettingKeyDockId );
if ( openingMode == OpeningMode::ForceDocked )
isDocked = true;
else if ( openingMode == OpeningMode::ForceDialog )
isDocked = false;
mDockArea = sSettingsDockArea->valueWithDefaultOverride( defaultDockArea, mSettingKeyDockId );
mIsDockFloating = mDockArea == Qt::DockWidgetArea::NoDockWidgetArea;
toggleDockMode( isDocked );
}
@ -53,6 +78,8 @@ QgsDockableWidgetHelper::~QgsDockableWidgetHelper()
if ( mDock )
{
mDockGeometry = mDock->geometry();
if ( !mSettingKeyDockId.isEmpty() )
sSettingsWindowGeometry->setValue( mDock->saveGeometry(), mSettingKeyDockId );
mIsDockFloating = mDock->isFloating();
if ( mOwnerWindow )
mDockArea = mOwnerWindow->dockWidgetArea( mDock );
@ -73,6 +100,8 @@ QgsDockableWidgetHelper::~QgsDockableWidgetHelper()
{
QgsSettings().setValue( mWindowGeometrySettingsKey, mDialog->saveGeometry() );
}
if ( !mSettingKeyDockId.isEmpty() )
sSettingsWindowGeometry->setValue( mDialog->saveGeometry(), mSettingKeyDockId );
mDialog->layout()->removeWidget( mWidget );
mDialog->deleteLater();
@ -243,6 +272,8 @@ void QgsDockableWidgetHelper::toggleDockMode( bool docked )
}
mIsDocked = docked;
if ( !mSettingKeyDockId.isEmpty() )
sSettingsIsDocked->setValue( mIsDocked, mSettingKeyDockId );
// If there is no widget set, do not create a dock or a dialog
if ( !mWidget )
@ -258,6 +289,13 @@ void QgsDockableWidgetHelper::toggleDockMode( bool docked )
mDock->setProperty( "dock_uuid", mUuid );
setupDockWidget();
if ( !mSettingKeyDockId.isEmpty() )
{
connect( mDock, &QgsDockWidget::dockLocationChanged, this, [=]( Qt::DockWidgetArea area ) {
sSettingsDockArea->setValue( area, mSettingKeyDockId );
} );
}
connect( mDock, &QgsDockWidget::closed, this, [=]() {
mDockGeometry = mDock->geometry();
mIsDockFloating = mDock->isFloating();
@ -266,7 +304,7 @@ void QgsDockableWidgetHelper::toggleDockMode( bool docked )
emit closed();
} );
if ( mUsePersistentWidget )
if ( mOptions.testFlag( Option::PermanentWidget ) )
mDock->installEventFilter( this );
connect( mDock, &QgsDockWidget::visibilityChanged, this, &QgsDockableWidgetHelper::visibilityChanged );
@ -278,7 +316,7 @@ void QgsDockableWidgetHelper::toggleDockMode( bool docked )
// going from dock -> window
// note -- we explicitly DO NOT set the parent for the dialog, as we want these treated as
// proper top level windows and have their own taskbar entries. See https://github.com/qgis/QGIS/issues/49286
if ( mUsePersistentWidget )
if ( mOptions.testFlag( Option::PermanentWidget ) )
mDialog = new QgsNonRejectableDialog( nullptr, Qt::Window );
else
mDialog = new QDialog( nullptr, Qt::Window );
@ -287,7 +325,7 @@ void QgsDockableWidgetHelper::toggleDockMode( bool docked )
mDialog->setWindowTitle( mWindowTitle );
mDialog->setObjectName( mObjectName );
if ( mUsePersistentWidget )
if ( mOptions.testFlag( Option::PermanentWidget ) )
mDialog->installEventFilter( this );
QVBoxLayout *vl = new QVBoxLayout();
@ -299,6 +337,10 @@ void QgsDockableWidgetHelper::toggleDockMode( bool docked )
QgsSettings settings;
mDialog->restoreGeometry( settings.value( mWindowGeometrySettingsKey ).toByteArray() );
}
else if ( !mSettingKeyDockId.isEmpty() )
{
mDialog->restoreGeometry( sSettingsWindowGeometry->value( mSettingKeyDockId ).toByteArray() );
}
else
{
if ( !mDockGeometry.isEmpty() )
@ -390,6 +432,7 @@ void QgsDockableWidgetHelper::setupDockWidget( const QStringList &tabSiblings )
return;
mDock->setFloating( mIsDockFloating );
// default dock geometry
if ( mDockGeometry.isEmpty() && mOwnerWindow )
{
const QFontMetrics fm( mOwnerWindow->font() );
@ -400,9 +443,9 @@ void QgsDockableWidgetHelper::setupDockWidget( const QStringList &tabSiblings )
{
sAddTabifiedDockWidgetFunction( mDockArea, mDock, tabSiblings, false );
}
else if ( mRaiseTab )
else if ( mOptions.testFlag( Option::RaiseTab ) )
{
sAddTabifiedDockWidgetFunction( mDockArea, mDock, mTabifyWith, mRaiseTab );
sAddTabifiedDockWidgetFunction( mDockArea, mDock, mTabifyWith, true );
}
else if ( mOwnerWindow )
{
@ -411,7 +454,11 @@ void QgsDockableWidgetHelper::setupDockWidget( const QStringList &tabSiblings )
// can only resize properly and set the dock geometry after pending events have been processed,
// so queue the geometry setting on the end of the event loop
QMetaObject::invokeMethod( mDock, [this] { mDock->setGeometry( mDockGeometry ); }, Qt::QueuedConnection );
QMetaObject::invokeMethod( mDock, [this] {
if (mIsDockFloating && sSettingsWindowGeometry->exists(mSettingKeyDockId))
mDock->restoreGeometry( sSettingsWindowGeometry->value(mSettingKeyDockId).toByteArray() );
else
mDock->setGeometry( mDockGeometry ); }, Qt::QueuedConnection );
}
QToolButton *QgsDockableWidgetHelper::createDockUndockToolButton()

View File

@ -24,6 +24,10 @@
#include <QDomElement>
#include <QPointer>
#include "qgssettingsentryimpl.h"
#include "qgssettingsentryenumflag.h"
#include "qgsgui.h"
#define SIP_NO_FILE
class QgsDockWidget;
@ -50,8 +54,29 @@ class GUI_EXPORT QgsNonRejectableDialog : public QDialog
*/
class GUI_EXPORT QgsDockableWidgetHelper : public QObject
{
static inline QgsSettingsTreeNode *sTtreeDockConfigs = QgsGui::sTtreeWidgetGeometry->createNamedListNode( QStringLiteral( "docks" ) ) SIP_SKIP;
static const QgsSettingsEntryBool *sSettingsIsDocked SIP_SKIP;
static const QgsSettingsEntryVariant *sSettingsWindowGeometry SIP_SKIP;
static const QgsSettingsEntryEnumFlag<Qt::DockWidgetArea> *sSettingsDockArea SIP_SKIP;
Q_OBJECT
public:
enum class OpeningMode : int
{
RespectSetting, //! Respect the setting used
ForceDocked, //! Force the widget to be docked, despite its settings
ForceDialog, //! Force the widget to be shown in a dialog, despite its settings
};
enum class Option : int
{
RaiseTab = 1 << 1, //!< Raise Tab
PermanentWidget = 1 << 2, //!< The widget (either as a dock or window) cannot be destroyed and must be hidden instead
};
Q_ENUM( Option )
Q_DECLARE_FLAGS( Options, Option )
/**
* Constructs an object that is responsible of making a docked widget or a window titled \a windowTitle that holds the \a widget
* The ownership of \a widget is returned to \a ownerWindow once the object is destroyed.
@ -59,8 +84,35 @@ class GUI_EXPORT QgsDockableWidgetHelper : public QObject
* If \a usePersistentWidget is TRUE then the \a widget (either as a dock or window) cannot be destroyed and must be hidden instead.
*/
QgsDockableWidgetHelper( bool isDocked, const QString &windowTitle, QWidget *widget, QMainWindow *ownerWindow, Qt::DockWidgetArea defaultDockArea = Qt::NoDockWidgetArea, const QStringList &tabifyWith = QStringList(), bool raiseTab = false, const QString &windowGeometrySettingsKey = QString(), bool usePersistentWidget = false );
/**
* Constructs an object that is responsible of making a docked widget or a window titled \a windowTitle that holds the \a widget
* The ownership of \a widget is returned to \a ownerWindow once the object is destroyed.
*
* With a unique \a dockId, the status (docked, area and geometry) are saved in the settings and re-used on creation.
*
* If \a usePersistentWidget is TRUE then the \a widget (either as a dock or window) cannot be destroyed and must be hidden instead.
*
* \since QGIS 3.42
*/
QgsDockableWidgetHelper(
const QString &windowTitle,
QWidget *widget,
QMainWindow *ownerWindow,
const QString &dockId,
const QStringList &tabifyWith = QStringList(),
OpeningMode openingMode = OpeningMode::RespectSetting,
bool defaultIsDocked = false,
Qt::DockWidgetArea defaultDockArea = Qt::DockWidgetArea::RightDockWidgetArea,
Options options = Options()
);
~QgsDockableWidgetHelper();
//! Returns if the widget is docked
//! \since 3.42
bool isDocked() const { return mIsDocked; }
//! Reads the dimensions of both the dock widget and the top level window
void writeXml( QDomElement &viewDom );
void readXml( const QDomElement &viewDom );
@ -140,16 +192,20 @@ class GUI_EXPORT QgsDockableWidgetHelper : public QObject
QMainWindow *mOwnerWindow = nullptr;
QStringList mTabifyWith;
bool mRaiseTab = false;
Options mOptions;
QString mWindowGeometrySettingsKey;
// Unique identifier of dock
QString mUuid;
bool mUsePersistentWidget = false;
const QString mSettingKeyDockId;
};
Q_DECLARE_OPERATORS_FOR_FLAGS( QgsDockableWidgetHelper::Options )
///@endcond
#endif // QGSDOCKABLEWIDGETHELPER_H

View File

@ -65,6 +65,7 @@ class GUI_EXPORT QgsGui : public QObject
Q_OBJECT
public:
static inline QgsSettingsTreeNode *sTtreeWidgetGeometry = QgsSettingsTree::sTreeApp->createChildNode( QStringLiteral( "widget-geometry" ) ) SIP_SKIP;
static inline QgsSettingsTreeNode *sTtreeWidgetLastUsedValues = QgsSettingsTree::sTreeApp->createChildNode( QStringLiteral( "widget-last-used-values" ) ) SIP_SKIP;
/**