//============================================================================ /// \file DockFocusController.cpp /// \author Uwe Kindler /// \date 05.06.2020 /// \brief Implementation of CDockFocusController class //============================================================================ //============================================================================ // INCLUDES //============================================================================ #include "DockFocusController.h" #include #include #include #include #include #include #include "DockWidget.h" #include "DockAreaWidget.h" #include "DockWidgetTab.h" #include "DockContainerWidget.h" #include "FloatingDockContainer.h" #include "DockManager.h" #include "DockAreaTitleBar.h" #ifdef Q_OS_LINUX #include "linux/FloatingWidgetTitleBar.h" #endif namespace ads { /** * Private data class of CDockFocusController class (pimpl) */ struct DockFocusControllerPrivate { CDockFocusController *_this; QPointer FocusedDockWidget = nullptr; QPointer FocusedArea = nullptr; CDockWidget* OldFocusedDockWidget = nullptr; #ifdef Q_OS_LINUX QPointer FloatingWidget = nullptr; #endif CDockManager* DockManager; bool ForceFocusChangedSignal = false; /** * Private data constructor */ DockFocusControllerPrivate(CDockFocusController *_public); /** * This function updates the focus style of the given dock widget and * the dock area that it belongs to */ void updateDockWidgetFocus(CDockWidget* DockWidget); }; // struct DockFocusControllerPrivate //=========================================================================== static void updateDockWidgetFocusStyle(CDockWidget* DockWidget, bool Focused) { DockWidget->setProperty("focused", Focused); DockWidget->tabWidget()->setProperty("focused", Focused); DockWidget->tabWidget()->updateStyle(); internal::repolishStyle(DockWidget); } //=========================================================================== static void updateDockAreaFocusStyle(CDockAreaWidget* DockArea, bool Focused) { DockArea->setProperty("focused", Focused); internal::repolishStyle(DockArea); internal::repolishStyle(DockArea->titleBar()); } //=========================================================================== #ifdef Q_OS_LINUX static void updateFloatingWidgetFocusStyle(CFloatingDockContainer* FloatingWidget, bool Focused) { if (FloatingWidget->hasNativeTitleBar()) { return; } auto TitleBar = qobject_cast(FloatingWidget->titleBarWidget()); if (!TitleBar) { return; } TitleBar->setProperty("focused", Focused); TitleBar->updateStyle(); } #endif //============================================================================ DockFocusControllerPrivate::DockFocusControllerPrivate( CDockFocusController *_public) : _this(_public) { } //============================================================================ void DockFocusControllerPrivate::updateDockWidgetFocus(CDockWidget* DockWidget) { if (!DockWidget->features().testFlag(CDockWidget::DockWidgetFocusable)) { return; } auto Window = DockWidget->dockContainer()->window()->windowHandle(); if (Window) { Window->setProperty("FocusedDockWidget", QVariant::fromValue(DockWidget)); } CDockAreaWidget* NewFocusedDockArea = nullptr; if (FocusedDockWidget) { updateDockWidgetFocusStyle(FocusedDockWidget, false); } CDockWidget* old = FocusedDockWidget; FocusedDockWidget = DockWidget; updateDockWidgetFocusStyle(FocusedDockWidget, true); NewFocusedDockArea = FocusedDockWidget->dockAreaWidget(); if (NewFocusedDockArea && (FocusedArea != NewFocusedDockArea)) { if (FocusedArea) { QObject::disconnect(FocusedArea, SIGNAL(viewToggled(bool)), _this, SLOT(onFocusedDockAreaViewToggled(bool))); updateDockAreaFocusStyle(FocusedArea, false); } FocusedArea = NewFocusedDockArea; updateDockAreaFocusStyle(FocusedArea, true); QObject::connect(FocusedArea, SIGNAL(viewToggled(bool)), _this, SLOT(onFocusedDockAreaViewToggled(bool))); } auto NewFloatingWidget = FocusedDockWidget->dockContainer()->floatingWidget(); if (NewFloatingWidget) { NewFloatingWidget->setProperty("FocusedDockWidget", QVariant::fromValue(DockWidget)); } #ifdef Q_OS_LINUX // This code is required for styling the floating widget titlebar for linux // depending on the current focus state if (FloatingWidget == NewFloatingWidget) { return; } if (FloatingWidget) { updateFloatingWidgetFocusStyle(FloatingWidget, false); } FloatingWidget = NewFloatingWidget; if (FloatingWidget) { updateFloatingWidgetFocusStyle(FloatingWidget, true); } #endif if (old == DockWidget && !ForceFocusChangedSignal) { return; } ForceFocusChangedSignal = false; if (DockWidget->isVisible()) { Q_EMIT DockManager->focusedDockWidgetChanged(old, DockWidget); } else { OldFocusedDockWidget = old; QObject::connect(DockWidget, SIGNAL(visibilityChanged(bool)), _this, SLOT(onDockWidgetVisibilityChanged(bool))); } } //============================================================================ void CDockFocusController::onDockWidgetVisibilityChanged(bool Visible) { auto Sender = sender(); auto DockWidget = qobject_cast(Sender); disconnect(Sender, SIGNAL(visibilityChanged(bool)), this, SLOT(onDockWidgetVisibilityChanged(bool))); if (DockWidget && Visible) { Q_EMIT d->DockManager->focusedDockWidgetChanged(d->OldFocusedDockWidget, DockWidget); } } //============================================================================ CDockFocusController::CDockFocusController(CDockManager* DockManager) : Super(DockManager), d(new DockFocusControllerPrivate(this)) { d->DockManager = DockManager; connect(QApplication::instance(), SIGNAL(focusChanged(QWidget*, QWidget*)), this, SLOT(onApplicationFocusChanged(QWidget*, QWidget*))); connect(QApplication::instance(), SIGNAL(focusWindowChanged(QWindow*)), this, SLOT(onFocusWindowChanged(QWindow*))); connect(d->DockManager, SIGNAL(stateRestored()), SLOT(onStateRestored())); } //============================================================================ CDockFocusController::~CDockFocusController() { delete d; } //============================================================================ void CDockFocusController::onFocusWindowChanged(QWindow *focusWindow) { if (!focusWindow) { return; } auto vDockWidget = focusWindow->property("FocusedDockWidget"); if (!vDockWidget.isValid()) { return; } auto DockWidget = vDockWidget.value(); if (!DockWidget) { return; } d->updateDockWidgetFocus(DockWidget); } //=========================================================================== void CDockFocusController::onApplicationFocusChanged(QWidget* focusedOld, QWidget* focusedNow) { Q_UNUSED(focusedOld); if (d->DockManager->isRestoringState()) { return; } ADS_PRINT("CDockFocusController::onApplicationFocusChanged " << " old: " << focusedOld << " new: " << focusedNow); if (!focusedNow) { return; } CDockWidget* DockWidget = qobject_cast(focusedNow); if (!DockWidget) { DockWidget = internal::findParent(focusedNow); } #ifdef Q_OS_LINUX if (!DockWidget) { return; } #else if (!DockWidget || DockWidget->tabWidget()->isHidden()) { return; } #endif d->updateDockWidgetFocus(DockWidget); } //=========================================================================== void CDockFocusController::setDockWidgetTabFocused(CDockWidgetTab* Tab) { auto DockWidget = Tab->dockWidget(); if (DockWidget) { d->updateDockWidgetFocus(DockWidget); } } //=========================================================================== void CDockFocusController::setDockWidgetFocused(CDockWidget* focusedNow) { d->updateDockWidgetFocus(focusedNow); } //=========================================================================== void CDockFocusController::onFocusedDockAreaViewToggled(bool Open) { if (d->DockManager->isRestoringState()) { return; } CDockAreaWidget* DockArea = qobject_cast(sender()); if (!DockArea || Open) { return; } auto Container = DockArea->dockContainer(); auto OpenedDockAreas = Container->openedDockAreas(); if (OpenedDockAreas.isEmpty()) { return; } d->updateDockWidgetFocus(OpenedDockAreas[0]->currentDockWidget()); } //=========================================================================== void CDockFocusController::notifyWidgetOrAreaRelocation(QWidget* DroppedWidget) { if (d->DockManager->isRestoringState()) { return; } CDockWidget* DockWidget = qobject_cast(DroppedWidget); if (!DockWidget) { CDockAreaWidget* DockArea = qobject_cast(DroppedWidget); if (DockArea) { DockWidget = DockArea->currentDockWidget(); } } if (!DockWidget) { return; } d->ForceFocusChangedSignal = true; CDockManager::setWidgetFocus(DockWidget); } //=========================================================================== void CDockFocusController::notifyFloatingWidgetDrop(CFloatingDockContainer* FloatingWidget) { if (!FloatingWidget || d->DockManager->isRestoringState()) { return; } auto vDockWidget = FloatingWidget->property("FocusedDockWidget"); if (!vDockWidget.isValid()) { return; } auto DockWidget = vDockWidget.value(); if (DockWidget) { DockWidget->dockAreaWidget()->setCurrentDockWidget(DockWidget); CDockManager::setWidgetFocus(DockWidget); } } //========================================================================== void CDockFocusController::onStateRestored() { if (d->FocusedDockWidget) { updateDockWidgetFocusStyle(d->FocusedDockWidget, false); } } //========================================================================== CDockWidget* CDockFocusController::focusedDockWidget() const { return d->FocusedDockWidget.data(); } } // namespace ads //--------------------------------------------------------------------------- // EOF DockFocusController.cpp