/******************************************************************************* ** Qt Advanced Docking System ** Copyright (C) 2017 Uwe Kindler ** ** This library is free software; you can redistribute it and/or ** modify it under the terms of the GNU Lesser General Public ** License as published by the Free Software Foundation; either ** version 2.1 of the License, or (at your option) any later version. ** ** This library is distributed in the hope that it will be useful, ** but WITHOUT ANY WARRANTY; without even the implied warranty of ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU ** Lesser General Public License for more details. ** ** You should have received a copy of the GNU Lesser General Public ** License along with this library; If not, see . ******************************************************************************/ //============================================================================ /// \file DockContainerWidget.cpp /// \author Uwe Kindler /// \date 24.02.2017 /// \brief Implementation of CDockContainerWidget class //============================================================================ //============================================================================ // INCLUDES //============================================================================ #include "DockContainerWidget.h" #include #include #include #include #include #include #include #include "DockManager.h" #include "DockAreaWidget.h" #include "DockWidget.h" #include "FloatingDockContainer.h" #include "DockOverlay.h" #include "DockStateSerialization.h" #include "ads_globals.h" #include "DockSplitter.h" #include #include namespace ads { static unsigned int zOrderCounter = 0; /** * Converts dock area ID to an index for array access */ static int areaIdToIndex(DockWidgetArea area) { switch (area) { case LeftDockWidgetArea: return 0; case RightDockWidgetArea: return 1; case TopDockWidgetArea: return 2; case BottomDockWidgetArea: return 3; case CenterDockWidgetArea: return 4; default: return 4; } return 4; } /** * Helper function to ease insertion of dock area into splitter */ static void insertWidgetIntoSplitter(QSplitter* Splitter, QWidget* widget, bool Append) { if (Append) { Splitter->addWidget(widget); } else { Splitter->insertWidget(0, widget); } } /** * Private data class of CDockContainerWidget class (pimpl) */ class DockContainerWidgetPrivate { public: CDockContainerWidget* _this; QPointer DockManager; unsigned int zOrderIndex = 0; QList DockAreas; QGridLayout* Layout = nullptr; QSplitter* RootSplitter; bool isFloating = false; CDockAreaWidget* LastAddedAreaCache[5]{0, 0, 0, 0, 0}; int VisibleDockAreaCount = -1; /** * Private data constructor */ DockContainerWidgetPrivate(CDockContainerWidget* _public); /** * Adds dock widget to container and returns the dock area that contains * the inserted dock widget */ CDockAreaWidget* dockWidgetIntoContainer(DockWidgetArea area, CDockWidget* Dockwidget); /** * Adds dock widget to a existing DockWidgetArea */ CDockAreaWidget* dockWidgetIntoDockArea(DockWidgetArea area, CDockWidget* Dockwidget, CDockAreaWidget* TargetDockArea); /** * Add dock area to this container */ void addDockArea(CDockAreaWidget* NewDockWidget, DockWidgetArea area = CenterDockWidgetArea); /** * Drop floating widget into container */ void dropIntoContainer(CFloatingDockContainer* FloatingWidget, DockWidgetArea area); /** * Drop floating widget into dock area */ void dropIntoSection(CFloatingDockContainer* FloatingWidget, CDockAreaWidget* TargetArea, DockWidgetArea area); /** * Adds new dock areas to the internal dock area list */ void addDockAreasToList(const QList NewDockAreas); /** * Save state of child nodes */ void saveChildNodesState(QXmlStreamWriter& Stream, QWidget* Widget); /** * Restore state of child nodes. * \param[in] Stream The data stream that contains the serialized state * \param[out] CreatedWidget The widget created from parsed data or 0 if * the parsed widget was an empty splitter * \param[in] Testing If Testing is true, only the stream data is * parsed without modifiying anything. */ bool restoreChildNodes(QXmlStreamReader& Stream, QWidget*& CreatedWidget, bool Testing); /** * Restores a splitter. * \see restoreChildNodes() for details */ bool restoreSplitter(QXmlStreamReader& Stream, QWidget*& CreatedWidget, bool Testing); /** * Restores a dock area. * \see restoreChildNodes() for details */ bool restoreDockArea(QXmlStreamReader& Stream, QWidget*& CreatedWidget, bool Testing); /** * Helper function for recursive dumping of layout */ void dumpRecursive(int level, QWidget* widget); /** * Initializes the visible dock area count variable if it is not initialized * yet */ void initVisibleDockAreaCount() { if (VisibleDockAreaCount > -1) { return; } VisibleDockAreaCount = 0; for (auto DockArea : DockAreas) { VisibleDockAreaCount += DockArea->isHidden() ? 0 : 1; } } /** * Access function for the visible dock area counter */ int& visibleDockAreaCount() { // Lazy initialisation - we initialize the VisibleDockAreaCount variable // on first use initVisibleDockAreaCount(); return VisibleDockAreaCount; } // private slots: ------------------------------------------------------------ void onDockAreaViewToggled(bool Visible) { std::cout << "onDockAreaViewToggled " << Visible << std::endl; VisibleDockAreaCount += Visible ? 1 : -1; } }; // struct DockContainerWidgetPrivate //============================================================================ DockContainerWidgetPrivate::DockContainerWidgetPrivate(CDockContainerWidget* _public) : _this(_public) { } //============================================================================ void DockContainerWidgetPrivate::dropIntoContainer(CFloatingDockContainer* FloatingWidget, DockWidgetArea area) { auto InsertParam = internal::dockAreaInsertParameters(area); CDockContainerWidget* FloatingDockContainer = FloatingWidget->dockContainer(); auto NewDockAreas = FloatingDockContainer->findChildren( QString(), Qt::FindChildrenRecursively); CDockWidget* SingleDroppedDockWidget = FloatingDockContainer->topLevelDockWidget(); CDockWidget* SingleDockWidget = _this->topLevelDockWidget(); QSplitter* Splitter = RootSplitter; if (DockAreas.count() <= 1) { Splitter->setOrientation(InsertParam.orientation()); } else if (Splitter->orientation() != InsertParam.orientation()) { QSplitter* NewSplitter = internal::newSplitter(InsertParam.orientation()); QLayoutItem* li = Layout->replaceWidget(Splitter, NewSplitter); NewSplitter->addWidget(Splitter); Splitter = NewSplitter; delete li; } // Now we can insert the floating widget content into this container auto FloatingSplitter = FloatingDockContainer->rootSplitter(); if (FloatingSplitter->count() == 1) { insertWidgetIntoSplitter(Splitter, FloatingSplitter->widget(0), InsertParam.append()); } else if (FloatingSplitter->orientation() == InsertParam.orientation()) { while (FloatingSplitter->count()) { insertWidgetIntoSplitter(Splitter, FloatingSplitter->widget(0), InsertParam.append()); } } else { insertWidgetIntoSplitter(Splitter, FloatingSplitter, InsertParam.append()); } RootSplitter = Splitter; addDockAreasToList(NewDockAreas); FloatingWidget->deleteLater(); CDockWidget::emitTopLevelEventForWidget(SingleDroppedDockWidget, false); CDockWidget::emitTopLevelEventForWidget(SingleDockWidget, false); // If we dropped the floating widget into the main dock container that does // not contain any dock widgets, then splitter is invisible and we need to // show it to display the docked widgets if (!Splitter->isVisible()) { Splitter->show(); } _this->dumpLayout(); } //============================================================================ void DockContainerWidgetPrivate::dropIntoSection(CFloatingDockContainer* FloatingWidget, CDockAreaWidget* TargetArea, DockWidgetArea area) { CDockContainerWidget* FloatingContainer = FloatingWidget->dockContainer(); if (area == CenterDockWidgetArea) { auto NewDockWidgets = FloatingContainer->dockWidgets(); for (int i = 0; i < NewDockWidgets.count(); ++i) { TargetArea->insertDockWidget(i, NewDockWidgets[i], false); } TargetArea->setCurrentIndex(0); // make the topmost widget active FloatingWidget->deleteLater(); TargetArea->updateTabBarVisibility(); return; } auto InsertParam = internal::dockAreaInsertParameters(area); auto NewDockAreas = FloatingWidget->dockContainer()->findChildren( QString(), Qt::FindChildrenRecursively); QSplitter* TargetAreaSplitter = internal::findParent(TargetArea); if (!TargetAreaSplitter) { QSplitter* Splitter = internal::newSplitter(InsertParam.orientation()); Layout->replaceWidget(TargetArea, Splitter); Splitter->addWidget(TargetArea); TargetAreaSplitter = Splitter; } int AreaIndex = TargetAreaSplitter->indexOf(TargetArea); auto Widget = FloatingWidget->dockContainer()->findChild(QString(), Qt::FindDirectChildrenOnly); auto FloatingSplitter = dynamic_cast(Widget); if (TargetAreaSplitter->orientation() == InsertParam.orientation()) { if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1) { TargetAreaSplitter->insertWidget(AreaIndex + InsertParam.insertOffset(), Widget); } else { int InsertIndex = AreaIndex + InsertParam.insertOffset(); while (FloatingSplitter->count()) { TargetAreaSplitter->insertWidget(InsertIndex++, FloatingSplitter->widget(0)); } } } else { QSplitter* NewSplitter = internal::newSplitter(InsertParam.orientation()); if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1) { NewSplitter->addWidget(Widget); } else { while (FloatingSplitter->count()) { NewSplitter->addWidget(FloatingSplitter->widget(0)); } } TargetAreaSplitter->insertWidget(AreaIndex, NewSplitter); insertWidgetIntoSplitter(NewSplitter, TargetArea, !InsertParam.append()); } FloatingWidget->deleteLater(); addDockAreasToList(NewDockAreas); _this->dumpLayout(); } //============================================================================ void DockContainerWidgetPrivate::addDockAreasToList(const QList NewDockAreas) { int CountBefore = DockAreas.count(); int NewAreaCount = NewDockAreas.count(); DockAreas.append(NewDockAreas); // We need to ensure, that the dock area title bar is visible. The title bar // is invisible, if the dock are is a single dock area in a floating widget. if (1 == CountBefore) { DockAreas.at(0)->updateTabBarVisibility(); } if (1 == NewAreaCount) { DockAreas.last()->updateTabBarVisibility(); } emit _this->dockAreasAdded(); } //============================================================================ void DockContainerWidgetPrivate::saveChildNodesState(QXmlStreamWriter& s, QWidget* Widget) { QSplitter* Splitter = dynamic_cast(Widget); if (Splitter) { s.writeStartElement("Splitter"); s.writeAttribute("Orientation", QString::number(Splitter->orientation())); s.writeAttribute("Count", QString::number(Splitter->count())); qDebug() << "NodeSplitter orient: " << Splitter->orientation() << " WidgetCont: " << Splitter->count(); for (int i = 0; i < Splitter->count(); ++i) { saveChildNodesState(s, Splitter->widget(i)); } s.writeStartElement("Sizes"); for (auto Size : Splitter->sizes()) { s.writeCharacters(QString::number(Size) + " "); } s.writeEndElement(); s.writeEndElement(); } else { CDockAreaWidget* DockArea = dynamic_cast(Widget); if (DockArea) { DockArea->saveState(s); } } } //============================================================================ bool DockContainerWidgetPrivate::restoreSplitter(QXmlStreamReader& s, QWidget*& CreatedWidget, bool Testing) { bool Ok; int Orientation = s.attributes().value("Orientation").toInt(&Ok); if (!Ok) { return false; } int WidgetCount = s.attributes().value("Count").toInt(&Ok); if (!Ok) { return false; } qDebug() << "Restore NodeSplitter Orientation: " << Orientation << " WidgetCount: " << WidgetCount; QSplitter* Splitter = nullptr; if (!Testing) { Splitter = internal::newSplitter((Qt::Orientation)Orientation); } bool Visible = false; QList Sizes; while (s.readNextStartElement()) { QWidget* ChildNode = nullptr; bool Result = true; if (s.name() == "Splitter") { Result = restoreSplitter(s, ChildNode, Testing); } else if (s.name() == "DockAreaWidget") { Result = restoreDockArea(s, ChildNode, Testing); } else if (s.name() == "Sizes") { QString sSizes = s.readElementText().trimmed(); qDebug() << "Sizes: " << sSizes; QTextStream TextStream(&sSizes); while (!TextStream.atEnd()) { int value; TextStream >> value; Sizes.append(value); } } else { s.skipCurrentElement(); } if (!Result) { return false; } if (Testing || !ChildNode) { continue; } qDebug() << "ChildNode isVisible " << ChildNode->isVisible() << " isVisibleTo " << ChildNode->isVisibleTo(Splitter); Splitter->addWidget(ChildNode); Visible |= ChildNode->isVisibleTo(Splitter); } if (Sizes.count() != WidgetCount) { return false; } if (!Testing) { if (!Splitter->count()) { delete Splitter; Splitter = nullptr; } else { Splitter->setSizes(Sizes); Splitter->setVisible(Visible); } CreatedWidget = Splitter; } else { CreatedWidget = nullptr; } return true; } //============================================================================ bool DockContainerWidgetPrivate::restoreDockArea(QXmlStreamReader& s, QWidget*& CreatedWidget, bool Testing) { bool Ok; int Tabs = s.attributes().value("Tabs").toInt(&Ok); if (!Ok) { return false; } QString CurrentDockWidget = s.attributes().value("CurrentDockWidget").toString(); qDebug() << "Restore NodeDockArea Tabs: " << Tabs << " CurrentDockWidget: " << CurrentDockWidget; CDockAreaWidget* DockArea = nullptr; if (!Testing) { DockArea = new CDockAreaWidget(DockManager, _this); } while (s.readNextStartElement()) { if (s.name() != "DockWidget") { continue; } auto ObjectName = s.attributes().value("ObjectName"); if (ObjectName.isEmpty()) { return false; } bool Closed = s.attributes().value("Closed").toInt(&Ok); if (!Ok) { return false; } s.skipCurrentElement(); CDockWidget* DockWidget = DockManager->findDockWidget(ObjectName.toString()); if (!DockWidget || Testing) { continue; } qDebug() << "Dock Widget found - parent " << DockWidget->parent(); DockArea->addDockWidget(DockWidget); // We hide the DockArea here to prevent the short display (the flashing) // of the dock areas during application startup DockArea->hide(); DockWidget->setToggleViewActionChecked(!Closed); DockWidget->setClosedState(Closed); DockWidget->setProperty("closed", Closed); DockWidget->setProperty("dirty", false); } if (Testing) { return true; } if (!DockArea->dockWidgetsCount()) { delete DockArea; DockArea = nullptr; } else { DockArea->setProperty("currentDockWidget", CurrentDockWidget); DockAreas.append(DockArea); } CreatedWidget = DockArea; return true; } //============================================================================ bool DockContainerWidgetPrivate::restoreChildNodes(QXmlStreamReader& s, QWidget*& CreatedWidget, bool Testing) { bool Result = true; while (s.readNextStartElement()) { if (s.name() == "Splitter") { Result = restoreSplitter(s, CreatedWidget, Testing); qDebug() << "Splitter"; } else if (s.name() == "DockAreaWidget") { Result = restoreDockArea(s, CreatedWidget, Testing); qDebug() << "DockAreaWidget"; } else { s.skipCurrentElement(); qDebug() << "Unknown element"; } } return Result; } //============================================================================ CDockAreaWidget* DockContainerWidgetPrivate::dockWidgetIntoContainer(DockWidgetArea area, CDockWidget* Dockwidget) { CDockAreaWidget* NewDockArea = new CDockAreaWidget(DockManager, _this); NewDockArea->addDockWidget(Dockwidget); addDockArea(NewDockArea, area); NewDockArea->updateTabBarVisibility(); LastAddedAreaCache[areaIdToIndex(area)] = NewDockArea; return NewDockArea; } //============================================================================ void DockContainerWidgetPrivate::addDockArea(CDockAreaWidget* NewDockArea, DockWidgetArea area) { auto InsertParam = internal::dockAreaInsertParameters(area); // As long as we have only one dock area in the splitter we can adjust // its orientation if (DockAreas.count() <= 1) { RootSplitter->setOrientation(InsertParam.orientation()); } QSplitter* Splitter = RootSplitter; if (Splitter->orientation() == InsertParam.orientation()) { insertWidgetIntoSplitter(Splitter, NewDockArea, InsertParam.append()); } else { QSplitter* NewSplitter = internal::newSplitter(InsertParam.orientation()); if (InsertParam.append()) { QLayoutItem* li = Layout->replaceWidget(Splitter, NewSplitter); NewSplitter->addWidget(Splitter); NewSplitter->addWidget(NewDockArea); delete li; } else { NewSplitter->addWidget(NewDockArea); QLayoutItem* li = Layout->replaceWidget(Splitter, NewSplitter); NewSplitter->addWidget(Splitter); delete li; } RootSplitter = NewSplitter; } DockAreas.append(NewDockArea); NewDockArea->updateTabBarVisibility(); emit _this->dockAreasAdded(); } //============================================================================ void DockContainerWidgetPrivate::dumpRecursive(int level, QWidget* widget) { #if defined(QT_DEBUG) QSplitter* Splitter = dynamic_cast(widget); QByteArray buf; buf.fill(' ', level * 4); if (Splitter) { qDebug("%sSplitter %s v: %s c: %s", (const char*)buf, (Splitter->orientation() == Qt::Vertical) ? "-" : "|", Splitter->isVisibleTo(Splitter->parentWidget()) ? "1" : "0", QString::number(Splitter->count()).toStdString().c_str()); for (int i = 0; i < Splitter->count(); ++i) { dumpRecursive(level + 1, Splitter->widget(i)); } } else { CDockAreaWidget* DockArea = dynamic_cast(widget); if (!DockArea) { return; } qDebug("%sDockArea", (const char*)buf); } #else Q_UNUSED(level); Q_UNUSED(widget); #endif } //============================================================================ CDockAreaWidget* DockContainerWidgetPrivate::dockWidgetIntoDockArea(DockWidgetArea area, CDockWidget* Dockwidget, CDockAreaWidget* TargetDockArea) { if (CenterDockWidgetArea == area) { TargetDockArea->addDockWidget(Dockwidget); return TargetDockArea; } CDockAreaWidget* NewDockArea = new CDockAreaWidget(DockManager, _this); NewDockArea->addDockWidget(Dockwidget); auto InsertParam = internal::dockAreaInsertParameters(area); QSplitter* TargetAreaSplitter = internal::findParent(TargetDockArea); int index = TargetAreaSplitter ->indexOf(TargetDockArea); if (TargetAreaSplitter->orientation() == InsertParam.orientation()) { qDebug() << "TargetAreaSplitter->orientation() == InsertParam.orientation()"; TargetAreaSplitter->insertWidget(index + InsertParam.insertOffset(), NewDockArea); } else { qDebug() << "TargetAreaSplitter->orientation() != InsertParam.orientation()"; QSplitter* NewSplitter = internal::newSplitter(InsertParam.orientation()); NewSplitter->addWidget(TargetDockArea); insertWidgetIntoSplitter(NewSplitter, NewDockArea, InsertParam.append()); TargetAreaSplitter->insertWidget(index, NewSplitter); } DockAreas.append(NewDockArea); emit _this->dockAreasAdded(); return NewDockArea; } //============================================================================ CDockContainerWidget::CDockContainerWidget(CDockManager* DockManager, QWidget *parent) : QFrame(parent), d(new DockContainerWidgetPrivate(this)) { d->isFloating = floatingWidget() != nullptr; d->DockManager = DockManager; if (DockManager != this) { d->DockManager->registerDockContainer(this); } d->Layout = new QGridLayout(); d->Layout->setContentsMargins(0, 1, 0, 1); d->Layout->setSpacing(0); setLayout(d->Layout); d->RootSplitter = internal::newSplitter(Qt::Horizontal); d->Layout->addWidget(d->RootSplitter); } //============================================================================ CDockContainerWidget::~CDockContainerWidget() { if (d->DockManager) { d->DockManager->removeDockContainer(this); } delete d; } //============================================================================ CDockAreaWidget* CDockContainerWidget::addDockWidget(DockWidgetArea area, CDockWidget* Dockwidget, CDockAreaWidget* DockAreaWidget) { CDockAreaWidget* OldDockArea = Dockwidget->dockAreaWidget(); if (OldDockArea) { OldDockArea->removeDockWidget(Dockwidget); } Dockwidget->setDockManager(d->DockManager); if (DockAreaWidget) { return d->dockWidgetIntoDockArea(area, Dockwidget, DockAreaWidget); } else { return d->dockWidgetIntoContainer(area, Dockwidget); } } //============================================================================ unsigned int CDockContainerWidget::zOrderIndex() const { return d->zOrderIndex; } //============================================================================ bool CDockContainerWidget::isInFrontOf(CDockContainerWidget* Other) const { return this->zOrderIndex() > Other->zOrderIndex(); } //============================================================================ bool CDockContainerWidget::event(QEvent *e) { bool Result = QWidget::event(e); if (e->type() == QEvent::WindowActivate) { d->zOrderIndex = ++zOrderCounter; } else if (e->type() == QEvent::Show && !d->zOrderIndex) { d->zOrderIndex = ++zOrderCounter; } return Result; } //============================================================================ void CDockContainerWidget::addDockArea(CDockAreaWidget* DockAreaWidget, DockWidgetArea area) { CDockContainerWidget* Container = DockAreaWidget->dockContainer(); if (Container && Container != this) { Container->removeDockArea(DockAreaWidget); } d->addDockArea(DockAreaWidget, area); } //============================================================================ void CDockContainerWidget::removeDockArea(CDockAreaWidget* area) { qDebug() << "CDockContainerWidget::removeDockArea"; d->DockAreas.removeAll(area); CDockSplitter* Splitter = internal::findParent(area); // Remove are from parent splitter and hide splitter if it has no visible // content area->setParent(0); Splitter->setVisible(Splitter->hasVisibleContent()); // If splitter has more than 1 widgets, we are finished and can leave if (Splitter->count() > 1) { goto emitAndExit; } // If this is the RootSplitter we need to remove empty splitters to // avoid too many empty splitters if (Splitter == d->RootSplitter) { qDebug() << "Removed from RootSplitter"; // If splitter is empty, we are finished if (!Splitter->count()) { Splitter->hide(); goto emitAndExit; } QWidget* widget = Splitter->widget(0); QSplitter* ChildSplitter = dynamic_cast(widget); // If the one and only content widget of the splitter is not a splitter // then we are finished if (!ChildSplitter) { goto emitAndExit; } // We replace the superfluous RootSplitter with the ChildSplitter ChildSplitter->setParent(0); QLayoutItem* li = d->Layout->replaceWidget(Splitter, ChildSplitter); d->RootSplitter = ChildSplitter; delete li; qDebug() << "RootSplitter replaced by child splitter"; } else if (Splitter->count() == 1) { qDebug() << "Replacing splitter with content"; QWidget* widget = Splitter->widget(0); widget->setParent(this); QSplitter* ParentSplitter = internal::findParent(Splitter); internal::replaceSplitterWidget(ParentSplitter, Splitter, widget); } delete Splitter; emitAndExit: CDockWidget* TopLevelWidget = topLevelDockWidget(); // Updated the title bar visibility of the dock widget if there is only // one single visible dock widget CDockWidget::emitTopLevelEventForWidget(TopLevelWidget, true); dumpLayout(); emit dockAreasRemoved(); } //============================================================================ CDockAreaWidget* CDockContainerWidget::dockAreaAt(const QPoint& GlobalPos) const { for (const auto& DockArea : d->DockAreas) { if (DockArea->isVisible() && DockArea->rect().contains(DockArea->mapFromGlobal(GlobalPos))) { return DockArea; } } return 0; } //============================================================================ CDockAreaWidget* CDockContainerWidget::dockArea(int Index) const { return (Index < dockAreaCount()) ? d->DockAreas[Index] : 0; } //============================================================================ bool CDockContainerWidget::isFloating() const { return d->isFloating; } //============================================================================ int CDockContainerWidget::dockAreaCount() const { return d->DockAreas.count(); } //============================================================================ int CDockContainerWidget::visibleDockAreaCount() const { int Result = 0; for (auto DockArea : d->DockAreas) { Result += DockArea->isHidden() ? 0 : 1; } return Result; // TODO Cache or precalculate this to speed it up because it is used during // movement of floating widget //return d->visibleDockAreaCount(); } //============================================================================ void CDockContainerWidget::dropFloatingWidget(CFloatingDockContainer* FloatingWidget, const QPoint& TargetPos) { QElapsedTimer Timer; Timer.start(); qDebug() << "CDockContainerWidget::dropFloatingWidget"; CDockAreaWidget* DockArea = dockAreaAt(TargetPos); auto dropArea = InvalidDockWidgetArea; auto ContainerDropArea = d->DockManager->containerOverlay()->dropAreaUnderCursor(); CDockWidget* FloatingTopLevelDockWidget = FloatingWidget->topLevelDockWidget(); CDockWidget* TopLevelDockWidget = topLevelDockWidget(); if (DockArea) { auto dropOverlay = d->DockManager->dockAreaOverlay(); dropOverlay->setAllowedAreas(AllDockAreas); dropArea = dropOverlay->showOverlay(DockArea); if (ContainerDropArea != InvalidDockWidgetArea && ContainerDropArea != dropArea) { dropArea = InvalidDockWidgetArea; } if (dropArea != InvalidDockWidgetArea) { qDebug() << "Dock Area Drop Content: " << dropArea; d->dropIntoSection(FloatingWidget, DockArea, dropArea); } } // mouse is over container if (InvalidDockWidgetArea == dropArea) { dropArea = ContainerDropArea; qDebug() << "Container Drop Content: " << dropArea; if (dropArea != InvalidDockWidgetArea) { d->dropIntoContainer(FloatingWidget, dropArea); } } // If there was a top level widget before the drop, then it is not top // level widget anymore if (TopLevelDockWidget) { TopLevelDockWidget->emitTopLevelChanged(false); } // If we drop a floating widget with only one single dock widget, then we // drop a top level widget that changes from floating to docked now if (FloatingTopLevelDockWidget) { FloatingTopLevelDockWidget->emitTopLevelChanged(false); } } //============================================================================ QList CDockContainerWidget::openedDockAreas() const { QList Result; for (auto DockArea : d->DockAreas) { if (!DockArea->isHidden()) { Result.append(DockArea); } } return Result; } //============================================================================ void CDockContainerWidget::saveState(QXmlStreamWriter& s) const { qDebug() << "CDockContainerWidget::saveState isFloating " << isFloating(); s.writeStartElement("DockContainerWidget"); s.writeAttribute("Floating", QString::number(isFloating() ? 1 : 0)); if (isFloating()) { CFloatingDockContainer* FloatingWidget = floatingWidget(); QByteArray Geometry = FloatingWidget->saveGeometry(); s.writeTextElement("Geometry", Geometry.toHex(' ')); } d->saveChildNodesState(s, d->RootSplitter); s.writeEndElement(); } //============================================================================ bool CDockContainerWidget::restoreState(QXmlStreamReader& s, bool Testing) { bool IsFloating = s.attributes().value("Floating").toInt(); qDebug() << "Restore CDockContainerWidget Floating" << IsFloating; QWidget*NewRootSplitter {}; if (!Testing) { d->VisibleDockAreaCount = -1;// invalidate the dock area count d->DockAreas.clear(); } if (IsFloating) { qDebug() << "Restore floating widget"; if (!s.readNextStartElement() || s.name() != "Geometry") { return false; } QByteArray GeometryString = s.readElementText(QXmlStreamReader::ErrorOnUnexpectedElement).toLocal8Bit(); QByteArray Geometry = QByteArray::fromHex(GeometryString); if (Geometry.isEmpty()) { return false; } if (!Testing) { CFloatingDockContainer* FloatingWidget = floatingWidget(); FloatingWidget->restoreGeometry(Geometry); } } if (!d->restoreChildNodes(s, NewRootSplitter, Testing)) { return false; } if (Testing) { return true; } // If the root splitter is empty, rostoreChildNodes returns a 0 pointer // and we need to create a new empty root splitter if (!NewRootSplitter) { NewRootSplitter = internal::newSplitter(Qt::Horizontal); } d->Layout->replaceWidget(d->RootSplitter, NewRootSplitter); QSplitter* OldRoot = d->RootSplitter; d->RootSplitter = dynamic_cast(NewRootSplitter); OldRoot->deleteLater(); // All dock widgets, that have not been processed in the restore state // function are invisible to the user now and have no assigned dock area // They do not belong to any dock container, until the user toggles the // toggle view action the next time for (auto DockWidget : dockWidgets()) { if (DockWidget->property("dirty").toBool()) { DockWidget->flagAsUnassigned(); } else { DockWidget->toggleViewInternal(!DockWidget->property("closed").toBool()); } } // Finally we need to send the topLevelChanged() signals for all dock // widgets if top level changed CDockWidget* TopLevelDockWidget = topLevelDockWidget(); if (TopLevelDockWidget) { TopLevelDockWidget->emitTopLevelChanged(true); } else { for (auto DockArea : d->DockAreas) { for (auto DockWidget : DockArea->dockWidgets()) { DockWidget->emitTopLevelChanged(false); } } } return true; } //============================================================================ QSplitter* CDockContainerWidget::rootSplitter() const { return d->RootSplitter; } //============================================================================ void CDockContainerWidget::dumpLayout() { #if (ADS_DEBUG_LEVEL > 0) qDebug("\n\nDumping layout --------------------------"); d->dumpRecursive(0, d->RootSplitter); qDebug("--------------------------\n\n"); #endif } //============================================================================ CDockAreaWidget* CDockContainerWidget::lastAddedDockAreaWidget(DockWidgetArea area) const { return d->LastAddedAreaCache[areaIdToIndex(area)]; } //============================================================================ bool CDockContainerWidget::hasTopLevelDockWidget() const { if (!isFloating()) { return false; } auto DockAreas = openedDockAreas(); if (DockAreas.count() != 1) { return false; } return DockAreas[0]->openDockWidgetsCount() == 1; } //============================================================================ CDockWidget* CDockContainerWidget::topLevelDockWidget() const { auto TopLevelDockArea = topLevelDockArea(); if (!TopLevelDockArea) { return nullptr; } auto DockWidgets = TopLevelDockArea->openedDockWidgets(); if (DockWidgets.count() != 1) { return nullptr; } return DockWidgets[0]; } //============================================================================ CDockAreaWidget* CDockContainerWidget::topLevelDockArea() const { if (!isFloating()) { return nullptr; } auto DockAreas = openedDockAreas(); if (DockAreas.count() != 1) { return nullptr; } return DockAreas[0]; } //============================================================================ QList CDockContainerWidget::dockWidgets() const { QList Result; for (const auto DockArea : d->DockAreas) { Result.append(DockArea->dockWidgets()); } return Result; } //============================================================================ CDockWidget::DockWidgetFeatures CDockContainerWidget::features() const { CDockWidget::DockWidgetFeatures Features(CDockWidget::AllDockWidgetFeatures); for (const auto DockArea : d->DockAreas) { Features &= DockArea->features(); } return Features; } //============================================================================ CFloatingDockContainer* CDockContainerWidget::floatingWidget() const { return internal::findParent(this); } } // namespace ads #include "moc_DockContainerWidget.cpp" //--------------------------------------------------------------------------- // EOF DockContainerWidget.cpp