2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2020 Uwe Kindler
|
2023-01-04 08:19:47 +01:00
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-2.1-or-later OR GPL-3.0-or-later
|
2020-06-22 16:46:25 +02:00
|
|
|
|
|
|
|
|
#include "dockfocuscontroller.h"
|
|
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
#include "ads_globals_p.h"
|
2020-06-22 16:46:25 +02:00
|
|
|
#include "dockareatitlebar.h"
|
2023-06-23 19:58:25 +02:00
|
|
|
#include "dockareawidget.h"
|
2020-06-22 16:46:25 +02:00
|
|
|
#include "dockcontainerwidget.h"
|
|
|
|
|
#include "dockmanager.h"
|
|
|
|
|
#include "dockwidget.h"
|
|
|
|
|
#include "dockwidgettab.h"
|
|
|
|
|
#include "floatingdockcontainer.h"
|
|
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
|
2020-06-22 16:46:25 +02:00
|
|
|
#include "linux/floatingwidgettitlebar.h"
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <QApplication>
|
|
|
|
|
#include <QPointer>
|
2023-06-23 19:58:25 +02:00
|
|
|
#include <QWindow>
|
2020-06-22 16:46:25 +02:00
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
#include <iostream>
|
|
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
namespace ADS {
|
|
|
|
|
|
|
|
|
|
static const char *const g_focusedDockWidgetProperty = "FocusedDockWidget";
|
|
|
|
|
|
|
|
|
|
class DockFocusControllerPrivate
|
2020-06-22 16:46:25 +02:00
|
|
|
{
|
2023-06-23 19:58:25 +02:00
|
|
|
public:
|
|
|
|
|
DockFocusController *q;
|
|
|
|
|
QPointer<DockWidget> m_focusedDockWidget = nullptr;
|
|
|
|
|
QPointer<DockAreaWidget> m_focusedArea = nullptr;
|
|
|
|
|
QPointer<DockWidget> m_oldFocusedDockWidget = nullptr;
|
|
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
|
|
|
|
|
QPointer<FloatingDockContainer> m_floatingWidget = nullptr;
|
|
|
|
|
#endif
|
|
|
|
|
DockManager *m_dockManager = nullptr;
|
|
|
|
|
bool m_forceFocusChangedSignal = false;
|
|
|
|
|
bool m_tabPressed = false;
|
|
|
|
|
|
2020-06-22 16:46:25 +02:00
|
|
|
/**
|
2023-06-23 19:58:25 +02:00
|
|
|
* Private data constructor
|
2020-06-22 16:46:25 +02:00
|
|
|
*/
|
2023-06-23 19:58:25 +02:00
|
|
|
DockFocusControllerPrivate(DockFocusController *parent);
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
/**
|
|
|
|
|
* This function updates the focus style of the given dock widget and
|
|
|
|
|
* the dock area that it belongs to
|
|
|
|
|
*/
|
|
|
|
|
void updateDockWidgetFocus(DockWidget *dockWidget);
|
|
|
|
|
}; // class DockFocusControllerPrivate
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
static void updateDockWidgetFocusStyle(DockWidget *dockWidget, bool focused)
|
|
|
|
|
{
|
|
|
|
|
dockWidget->setProperty("focused", focused);
|
|
|
|
|
dockWidget->tabWidget()->setProperty("focused", focused);
|
|
|
|
|
dockWidget->tabWidget()->updateStyle();
|
|
|
|
|
internal::repolishStyle(dockWidget);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
static void updateDockAreaFocusStyle(DockAreaWidget *dockArea, bool focused)
|
|
|
|
|
{
|
|
|
|
|
dockArea->setProperty("focused", focused);
|
|
|
|
|
internal::repolishStyle(dockArea);
|
|
|
|
|
internal::repolishStyle(dockArea->titleBar());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
|
|
|
|
|
static void updateFloatingWidgetFocusStyle(FloatingDockContainer *floatingWidget, bool focused)
|
|
|
|
|
{
|
|
|
|
|
if (floatingWidget->hasNativeTitleBar())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
auto titleBar = qobject_cast<FloatingWidgetTitleBar *>(floatingWidget->titleBarWidget());
|
|
|
|
|
if (!titleBar)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
titleBar->setProperty("focused", focused);
|
|
|
|
|
titleBar->updateStyle();
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
#endif
|
|
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
DockFocusControllerPrivate::DockFocusControllerPrivate(DockFocusController *parent)
|
|
|
|
|
: q(parent)
|
|
|
|
|
{}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusControllerPrivate::updateDockWidgetFocus(DockWidget *dockWidget)
|
|
|
|
|
{
|
|
|
|
|
if (!dockWidget->features().testFlag(DockWidget::DockWidgetFocusable))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
QWindow *window = nullptr;
|
|
|
|
|
auto dockContainer = dockWidget->dockContainer();
|
|
|
|
|
if (dockContainer)
|
|
|
|
|
window = dockContainer->window()->windowHandle();
|
|
|
|
|
|
|
|
|
|
if (window)
|
|
|
|
|
window->setProperty(g_focusedDockWidgetProperty,
|
|
|
|
|
QVariant::fromValue(QPointer<DockWidget>(dockWidget)));
|
|
|
|
|
|
|
|
|
|
DockAreaWidget *newFocusedDockArea = nullptr;
|
|
|
|
|
if (m_focusedDockWidget)
|
|
|
|
|
updateDockWidgetFocusStyle(m_focusedDockWidget, false);
|
|
|
|
|
|
|
|
|
|
DockWidget *old = m_focusedDockWidget;
|
|
|
|
|
m_focusedDockWidget = dockWidget;
|
|
|
|
|
updateDockWidgetFocusStyle(m_focusedDockWidget, true);
|
|
|
|
|
newFocusedDockArea = m_focusedDockWidget->dockAreaWidget();
|
|
|
|
|
if (newFocusedDockArea && (m_focusedArea != newFocusedDockArea)) {
|
|
|
|
|
if (m_focusedArea) {
|
|
|
|
|
QObject::disconnect(m_focusedArea,
|
|
|
|
|
&DockAreaWidget::viewToggled,
|
|
|
|
|
q,
|
|
|
|
|
&DockFocusController::onFocusedDockAreaViewToggled);
|
|
|
|
|
updateDockAreaFocusStyle(m_focusedArea, false);
|
|
|
|
|
}
|
2023-02-20 11:09:25 +01:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
m_focusedArea = newFocusedDockArea;
|
|
|
|
|
updateDockAreaFocusStyle(m_focusedArea, true);
|
|
|
|
|
QObject::connect(m_focusedArea,
|
|
|
|
|
&DockAreaWidget::viewToggled,
|
|
|
|
|
q,
|
|
|
|
|
&DockFocusController::onFocusedDockAreaViewToggled);
|
|
|
|
|
}
|
2023-02-20 11:09:25 +01:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
FloatingDockContainer *newFloatingWidget = nullptr;
|
|
|
|
|
dockContainer = m_focusedDockWidget->dockContainer();
|
|
|
|
|
if (dockContainer)
|
|
|
|
|
newFloatingWidget = dockContainer->floatingWidget();
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
if (newFloatingWidget)
|
|
|
|
|
newFloatingWidget->setProperty(g_focusedDockWidgetProperty,
|
|
|
|
|
QVariant::fromValue(QPointer<DockWidget>(dockWidget)));
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
|
|
|
|
|
// This code is required for styling the floating widget titlebar for linux
|
|
|
|
|
// depending on the current focus state
|
|
|
|
|
if (m_floatingWidget != newFloatingWidget) {
|
|
|
|
|
if (m_floatingWidget) {
|
2020-06-22 16:46:25 +02:00
|
|
|
updateFloatingWidgetFocusStyle(m_floatingWidget, false);
|
2023-06-23 19:58:25 +02:00
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
m_floatingWidget = newFloatingWidget;
|
|
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
if (m_floatingWidget) {
|
2020-06-22 16:46:25 +02:00
|
|
|
updateFloatingWidgetFocusStyle(m_floatingWidget, true);
|
2023-06-23 19:58:25 +02:00
|
|
|
}
|
|
|
|
|
}
|
2023-02-20 11:09:25 +01:00
|
|
|
#endif
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
if (old == dockWidget && !m_forceFocusChangedSignal)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
m_forceFocusChangedSignal = false;
|
|
|
|
|
if (dockWidget->isVisible()) {
|
|
|
|
|
emit m_dockManager->focusedDockWidgetChanged(old, dockWidget);
|
|
|
|
|
} else {
|
|
|
|
|
m_oldFocusedDockWidget = old;
|
|
|
|
|
QObject::connect(dockWidget,
|
|
|
|
|
&DockWidget::visibilityChanged,
|
|
|
|
|
q,
|
|
|
|
|
&DockFocusController::onDockWidgetVisibilityChanged);
|
2020-06-22 16:46:25 +02:00
|
|
|
}
|
2023-06-23 19:58:25 +02:00
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::onDockWidgetVisibilityChanged(bool visible)
|
|
|
|
|
{
|
|
|
|
|
auto dockWidget = qobject_cast<DockWidget *>(sender());
|
|
|
|
|
QObject::disconnect(dockWidget,
|
|
|
|
|
&DockWidget::visibilityChanged,
|
|
|
|
|
this,
|
|
|
|
|
&DockFocusController::onDockWidgetVisibilityChanged);
|
|
|
|
|
if (dockWidget && visible)
|
|
|
|
|
emit d->m_dockManager->focusedDockWidgetChanged(d->m_oldFocusedDockWidget, dockWidget);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DockFocusController::DockFocusController(DockManager *dockManager)
|
|
|
|
|
: QObject(dockManager)
|
|
|
|
|
, d(new DockFocusControllerPrivate(this))
|
|
|
|
|
{
|
|
|
|
|
d->m_dockManager = dockManager;
|
|
|
|
|
connect(qApp,
|
|
|
|
|
&QApplication::focusChanged,
|
|
|
|
|
this,
|
|
|
|
|
&DockFocusController::onApplicationFocusChanged);
|
|
|
|
|
connect(qApp,
|
|
|
|
|
&QApplication::focusWindowChanged,
|
|
|
|
|
this,
|
|
|
|
|
&DockFocusController::onFocusWindowChanged);
|
|
|
|
|
connect(d->m_dockManager,
|
|
|
|
|
&DockManager::stateRestored,
|
|
|
|
|
this,
|
|
|
|
|
&DockFocusController::onStateRestored);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DockFocusController::~DockFocusController()
|
|
|
|
|
{
|
|
|
|
|
delete d;
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::onFocusWindowChanged(QWindow *focusWindow)
|
|
|
|
|
{
|
|
|
|
|
if (!focusWindow)
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
auto dockWidgetVar = focusWindow->property(g_focusedDockWidgetProperty);
|
|
|
|
|
if (!dockWidgetVar.isValid())
|
|
|
|
|
return;
|
2020-08-05 13:28:02 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
auto dockWidget = dockWidgetVar.value<QPointer<DockWidget>>();
|
|
|
|
|
if (!dockWidget)
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
d->updateDockWidgetFocus(dockWidget);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::onApplicationFocusChanged(QWidget *focusedOld, QWidget *focusedNow)
|
|
|
|
|
{
|
|
|
|
|
Q_UNUSED(focusedOld);
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
// Ignore focus changes if we are restoring state, or if user clicked a tab which in turn
|
|
|
|
|
// caused the focus change.
|
|
|
|
|
if (d->m_dockManager->isRestoringState() || d->m_tabPressed)
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
qCInfo(adsLog) << Q_FUNC_INFO << "old: " << focusedOld << "new:" << focusedNow;
|
2023-02-16 16:57:34 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
if (!focusedNow)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
DockWidget *dockWidget = qobject_cast<DockWidget *>(focusedNow);
|
|
|
|
|
if (!dockWidget)
|
|
|
|
|
dockWidget = internal::findParent<DockWidget *>(focusedNow);
|
|
|
|
|
|
|
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
|
|
|
|
|
if (!dockWidget)
|
|
|
|
|
return;
|
|
|
|
|
#else
|
|
|
|
|
if (!dockWidget || dockWidget->tabWidget()->isHidden())
|
|
|
|
|
return;
|
|
|
|
|
#endif
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
d->updateDockWidgetFocus(dockWidget);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::setDockWidgetTabFocused(DockWidgetTab *tab)
|
|
|
|
|
{
|
|
|
|
|
auto dockWidget = tab->dockWidget();
|
|
|
|
|
if (dockWidget)
|
2020-06-22 16:46:25 +02:00
|
|
|
d->updateDockWidgetFocus(dockWidget);
|
2023-06-23 19:58:25 +02:00
|
|
|
}
|
2023-02-16 16:57:34 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::clearDockWidgetFocus(DockWidget *dockWidget)
|
|
|
|
|
{
|
|
|
|
|
dockWidget->clearFocus();
|
|
|
|
|
updateDockWidgetFocusStyle(dockWidget, false);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::setDockWidgetFocused(DockWidget *focusedNow)
|
|
|
|
|
{
|
|
|
|
|
d->updateDockWidgetFocus(focusedNow);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::onFocusedDockAreaViewToggled(bool open)
|
|
|
|
|
{
|
|
|
|
|
if (d->m_dockManager->isRestoringState() || !d->m_focusedArea || open)
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
DockAreaWidget *dockArea = qobject_cast<DockAreaWidget *>(sender());
|
|
|
|
|
if (!dockArea || open)
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
auto container = dockArea->dockContainer();
|
|
|
|
|
auto openedDockAreas = container->openedDockAreas();
|
|
|
|
|
if (openedDockAreas.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
d->updateDockWidgetFocus(openedDockAreas[0]->currentDockWidget());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DockFocusController::notifyWidgetOrAreaRelocation(QWidget *droppedWidget)
|
|
|
|
|
{
|
|
|
|
|
if (d->m_dockManager->isRestoringState())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
DockWidget *dockWidget = qobject_cast<DockWidget *>(droppedWidget);
|
|
|
|
|
if (!dockWidget) {
|
|
|
|
|
DockAreaWidget *dockArea = qobject_cast<DockAreaWidget *>(droppedWidget);
|
|
|
|
|
if (dockArea)
|
|
|
|
|
dockWidget = dockArea->currentDockWidget();
|
2020-06-22 16:46:25 +02:00
|
|
|
}
|
|
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
if (!dockWidget)
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
d->m_forceFocusChangedSignal = true;
|
|
|
|
|
DockManager::setWidgetFocus(dockWidget);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::notifyFloatingWidgetDrop(FloatingDockContainer *floatingWidget)
|
|
|
|
|
{
|
|
|
|
|
if (!floatingWidget || d->m_dockManager->isRestoringState())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
auto dockWidgetVar = floatingWidget->property(g_focusedDockWidgetProperty);
|
|
|
|
|
if (!dockWidgetVar.isValid())
|
|
|
|
|
return;
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
auto dockWidget = dockWidgetVar.value<DockWidget *>();
|
|
|
|
|
if (dockWidget) {
|
|
|
|
|
dockWidget->dockAreaWidget()->setCurrentDockWidget(dockWidget);
|
2020-06-22 16:46:25 +02:00
|
|
|
DockManager::setWidgetFocus(dockWidget->tabWidget());
|
|
|
|
|
}
|
2023-06-23 19:58:25 +02:00
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::onStateRestored()
|
|
|
|
|
{
|
|
|
|
|
if (d->m_focusedDockWidget)
|
|
|
|
|
updateDockWidgetFocusStyle(d->m_focusedDockWidget, false);
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
DockWidget *DockFocusController::focusedDockWidget() const
|
|
|
|
|
{
|
|
|
|
|
return d->m_focusedDockWidget.data();
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
2023-06-23 19:58:25 +02:00
|
|
|
void DockFocusController::setDockWidgetTabPressed(bool value)
|
|
|
|
|
{
|
|
|
|
|
d->m_tabPressed = value;
|
|
|
|
|
}
|
2020-06-22 16:46:25 +02:00
|
|
|
|
|
|
|
|
} // namespace ADS
|