2017-03-23 17:23:53 +08:00
|
|
|
/*******************************************************************************
|
2017-06-10 04:04:02 +08:00
|
|
|
** Qt Advanced Docking System
|
2017-03-23 17:23:53 +08:00
|
|
|
** Copyright (C) 2017 Uwe Kindler
|
2017-06-10 04:04:02 +08:00
|
|
|
**
|
|
|
|
** 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,
|
2017-03-23 17:23:53 +08:00
|
|
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2017-06-10 04:04:02 +08:00
|
|
|
** 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 <http://www.gnu.org/licenses/>.
|
2017-03-23 17:23:53 +08:00
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
/// \file DockContainerWidget.cpp
|
|
|
|
/// \author Uwe Kindler
|
|
|
|
/// \date 24.02.2017
|
|
|
|
/// \brief Implementation of CDockContainerWidget class
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
// INCLUDES
|
|
|
|
//============================================================================
|
|
|
|
#include "DockContainerWidget.h"
|
|
|
|
|
|
|
|
#include <QEvent>
|
|
|
|
#include <QList>
|
|
|
|
#include <QGridLayout>
|
|
|
|
#include <QPointer>
|
2017-03-28 16:57:03 +08:00
|
|
|
#include <QVariant>
|
2017-03-28 18:01:27 +08:00
|
|
|
#include <QDebug>
|
2017-12-30 01:18:16 +08:00
|
|
|
#include <QXmlStreamWriter>
|
2018-11-05 16:07:18 +08:00
|
|
|
#include <QAbstractButton>
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
#include "DockManager.h"
|
|
|
|
#include "DockAreaWidget.h"
|
|
|
|
#include "DockWidget.h"
|
2019-11-29 22:56:57 +08:00
|
|
|
#include "DockingStateReader.h"
|
2017-03-23 17:23:53 +08:00
|
|
|
#include "FloatingDockContainer.h"
|
|
|
|
#include "DockOverlay.h"
|
|
|
|
#include "ads_globals.h"
|
2017-03-24 17:18:25 +08:00
|
|
|
#include "DockSplitter.h"
|
2017-03-23 17:23:53 +08:00
|
|
|
|
2019-01-17 01:52:53 +08:00
|
|
|
#include <functional>
|
2018-12-03 19:52:57 +08:00
|
|
|
#include <iostream>
|
2018-10-10 21:15:59 +08:00
|
|
|
|
2019-01-14 15:59:37 +08:00
|
|
|
#if QT_VERSION < 0x050900
|
|
|
|
|
|
|
|
inline char toHexLower(uint value)
|
|
|
|
{
|
|
|
|
return "0123456789abcdef"[value & 0xF];
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray qByteArrayToHex(const QByteArray& src, char separator)
|
|
|
|
{
|
|
|
|
if(src.size() == 0)
|
|
|
|
return QByteArray();
|
|
|
|
|
|
|
|
const int length = separator ? (src.size() * 3 - 1) : (src.size() * 2);
|
|
|
|
QByteArray hex(length, Qt::Uninitialized);
|
|
|
|
char *hexData = hex.data();
|
2019-01-17 01:52:53 +08:00
|
|
|
const uchar *data = reinterpret_cast<const uchar *>(src.data());
|
2019-01-14 15:59:37 +08:00
|
|
|
for (int i = 0, o = 0; i < src.size(); ++i) {
|
|
|
|
hexData[o++] = toHexLower(data[i] >> 4);
|
|
|
|
hexData[o++] = toHexLower(data[i] & 0xf);
|
|
|
|
|
|
|
|
if ((separator) && (o < length))
|
|
|
|
hexData[o++] = separator;
|
|
|
|
}
|
|
|
|
return hex;
|
|
|
|
}
|
|
|
|
#endif
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
namespace ads
|
|
|
|
{
|
|
|
|
static unsigned int zOrderCounter = 0;
|
|
|
|
|
2019-11-27 22:50:18 +08:00
|
|
|
enum eDropMode
|
|
|
|
{
|
|
|
|
DropModeIntoArea,///< drop widget into a dock area
|
|
|
|
DropModeIntoContainer,///< drop into container
|
|
|
|
DropModeInvalid///< invalid mode - do not drop
|
|
|
|
};
|
2018-09-07 17:10:14 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
*/
|
2018-11-01 14:53:54 +08:00
|
|
|
class DockContainerWidgetPrivate
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2018-11-01 14:53:54 +08:00
|
|
|
public:
|
2017-03-23 17:23:53 +08:00
|
|
|
CDockContainerWidget* _this;
|
|
|
|
QPointer<CDockManager> DockManager;
|
|
|
|
unsigned int zOrderIndex = 0;
|
|
|
|
QList<CDockAreaWidget*> DockAreas;
|
|
|
|
QGridLayout* Layout = nullptr;
|
2018-12-11 22:19:59 +08:00
|
|
|
QSplitter* RootSplitter = nullptr;
|
2017-03-23 17:23:53 +08:00
|
|
|
bool isFloating = false;
|
2019-01-26 00:28:36 +08:00
|
|
|
CDockAreaWidget* LastAddedAreaCache[5];
|
2018-11-01 14:53:54 +08:00
|
|
|
int VisibleDockAreaCount = -1;
|
2018-11-05 16:07:18 +08:00
|
|
|
CDockAreaWidget* TopLevelDockArea = nullptr;
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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);
|
|
|
|
|
2019-11-26 21:40:56 +08:00
|
|
|
/**
|
|
|
|
* Moves the dock widget or dock area given in Widget parameter to a
|
|
|
|
* new dock widget area
|
|
|
|
*/
|
|
|
|
void moveToNewSection(QWidget* Widget, CDockAreaWidget* TargetArea, DockWidgetArea area);
|
|
|
|
|
2019-11-27 19:00:04 +08:00
|
|
|
/**
|
|
|
|
* Moves the dock widget or dock area given in Widget parameter to a
|
|
|
|
* a dock area in container
|
|
|
|
*/
|
|
|
|
void moveToContainer(QWidget* Widgett, DockWidgetArea area);
|
|
|
|
|
2019-01-14 20:58:40 +08:00
|
|
|
/**
|
|
|
|
* Creates a new tab for a widget dropped into the center of a section
|
|
|
|
*/
|
|
|
|
void dropIntoCenterOfSection(CFloatingDockContainer* FloatingWidget,
|
|
|
|
CDockAreaWidget* TargetArea);
|
|
|
|
|
2019-11-26 21:40:56 +08:00
|
|
|
/**
|
|
|
|
* Creates a new tab for a widget dropped into the center of a section
|
|
|
|
*/
|
|
|
|
void moveIntoCenterOfSection(QWidget* Widget, CDockAreaWidget* TargetArea);
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
/**
|
|
|
|
* Adds new dock areas to the internal dock area list
|
|
|
|
*/
|
|
|
|
void addDockAreasToList(const QList<CDockAreaWidget*> NewDockAreas);
|
2017-03-23 22:57:15 +08:00
|
|
|
|
2018-11-05 16:07:18 +08:00
|
|
|
/**
|
|
|
|
* Wrapper function for DockAreas append, that ensures that dock area signals
|
|
|
|
* are properly connected to dock container slots
|
|
|
|
*/
|
|
|
|
void appendDockAreas(const QList<CDockAreaWidget*> NewDockAreas);
|
|
|
|
|
2017-03-23 22:57:15 +08:00
|
|
|
/**
|
|
|
|
* Save state of child nodes
|
|
|
|
*/
|
2017-12-30 01:18:16 +08:00
|
|
|
void saveChildNodesState(QXmlStreamWriter& Stream, QWidget* Widget);
|
2017-03-23 22:57:15 +08:00
|
|
|
|
|
|
|
/**
|
2017-03-27 16:41:27 +08:00
|
|
|
* Restore state of child nodes.
|
|
|
|
* \param[in] Stream The data stream that contains the serialized state
|
2017-03-27 19:18:16 +08:00
|
|
|
* \param[out] CreatedWidget The widget created from parsed data or 0 if
|
|
|
|
* the parsed widget was an empty splitter
|
2017-03-27 16:41:27 +08:00
|
|
|
* \param[in] Testing If Testing is true, only the stream data is
|
|
|
|
* parsed without modifiying anything.
|
2017-03-23 22:57:15 +08:00
|
|
|
*/
|
2019-11-29 22:56:57 +08:00
|
|
|
bool restoreChildNodes(CDockingStateReader& Stream, QWidget*& CreatedWidget,
|
2017-03-27 16:41:27 +08:00
|
|
|
bool Testing);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Restores a splitter.
|
|
|
|
* \see restoreChildNodes() for details
|
|
|
|
*/
|
2019-11-29 22:56:57 +08:00
|
|
|
bool restoreSplitter(CDockingStateReader& Stream, QWidget*& CreatedWidget,
|
2017-03-27 16:41:27 +08:00
|
|
|
bool Testing);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Restores a dock area.
|
|
|
|
* \see restoreChildNodes() for details
|
|
|
|
*/
|
2019-11-29 22:56:57 +08:00
|
|
|
bool restoreDockArea(CDockingStateReader& Stream, QWidget*& CreatedWidget,
|
2017-03-27 16:41:27 +08:00
|
|
|
bool Testing);
|
2017-03-28 14:48:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function for recursive dumping of layout
|
|
|
|
*/
|
|
|
|
void dumpRecursive(int level, QWidget* widget);
|
2018-11-01 14:53:54 +08:00
|
|
|
|
2019-11-27 22:50:18 +08:00
|
|
|
/**
|
|
|
|
* Calculate the drop mode from the given target position
|
|
|
|
*/
|
|
|
|
eDropMode getDropMode(const QPoint& TargetPos);
|
|
|
|
|
2018-11-01 14:53:54 +08:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
|
|
|
|
2018-11-05 16:07:18 +08:00
|
|
|
/**
|
|
|
|
* The visible dock area count changes, if dock areas are remove, added or
|
|
|
|
* when its view is toggled
|
|
|
|
*/
|
|
|
|
void onVisibleDockAreaCountChanged();
|
|
|
|
|
|
|
|
void emitDockAreasRemoved()
|
|
|
|
{
|
|
|
|
onVisibleDockAreaCountChanged();
|
|
|
|
emit _this->dockAreasRemoved();
|
|
|
|
}
|
|
|
|
|
|
|
|
void emitDockAreasAdded()
|
|
|
|
{
|
|
|
|
onVisibleDockAreaCountChanged();
|
|
|
|
emit _this->dockAreasAdded();
|
|
|
|
}
|
|
|
|
|
2018-12-11 22:19:59 +08:00
|
|
|
/**
|
|
|
|
* Helper function for creation of new splitter
|
|
|
|
*/
|
2019-01-26 00:28:36 +08:00
|
|
|
CDockSplitter* newSplitter(Qt::Orientation orientation, QWidget* parent = nullptr)
|
2018-12-11 22:19:59 +08:00
|
|
|
{
|
|
|
|
CDockSplitter* s = new CDockSplitter(orientation, parent);
|
2019-09-13 03:15:35 +08:00
|
|
|
s->setOpaqueResize(CDockManager::configFlags().testFlag(CDockManager::OpaqueSplitterResize));
|
2018-12-11 22:19:59 +08:00
|
|
|
s->setChildrenCollapsible(false);
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-01 14:53:54 +08:00
|
|
|
// private slots: ------------------------------------------------------------
|
|
|
|
void onDockAreaViewToggled(bool Visible)
|
|
|
|
{
|
2018-11-05 16:07:18 +08:00
|
|
|
CDockAreaWidget* DockArea = qobject_cast<CDockAreaWidget*>(_this->sender());
|
2018-11-01 14:53:54 +08:00
|
|
|
VisibleDockAreaCount += Visible ? 1 : -1;
|
2018-11-05 16:07:18 +08:00
|
|
|
onVisibleDockAreaCountChanged();
|
|
|
|
emit _this->dockAreaViewToggled(DockArea, Visible);
|
2018-11-01 14:53:54 +08:00
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
}; // struct DockContainerWidgetPrivate
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
DockContainerWidgetPrivate::DockContainerWidgetPrivate(CDockContainerWidget* _public) :
|
|
|
|
_this(_public)
|
|
|
|
{
|
2019-01-26 00:28:36 +08:00
|
|
|
std::fill(std::begin(LastAddedAreaCache),std::end(LastAddedAreaCache), nullptr);
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-27 22:50:18 +08:00
|
|
|
//============================================================================
|
|
|
|
eDropMode DockContainerWidgetPrivate::getDropMode(const QPoint& TargetPos)
|
|
|
|
{
|
|
|
|
CDockAreaWidget* DockArea = _this->dockAreaAt(TargetPos);
|
|
|
|
auto dropArea = InvalidDockWidgetArea;
|
|
|
|
auto ContainerDropArea = DockManager->containerOverlay()->dropAreaUnderCursor();
|
|
|
|
|
|
|
|
if (DockArea)
|
|
|
|
{
|
|
|
|
auto dropOverlay = DockManager->dockAreaOverlay();
|
2020-02-02 22:56:31 +08:00
|
|
|
dropOverlay->setAllowedAreas(DockArea->allowedAreas());
|
2019-11-27 22:50:18 +08:00
|
|
|
dropArea = dropOverlay->showOverlay(DockArea);
|
|
|
|
if (ContainerDropArea != InvalidDockWidgetArea &&
|
|
|
|
ContainerDropArea != dropArea)
|
|
|
|
{
|
|
|
|
dropArea = InvalidDockWidgetArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dropArea != InvalidDockWidgetArea)
|
|
|
|
{
|
|
|
|
ADS_PRINT("Dock Area Drop Content: " << dropArea);
|
|
|
|
return DropModeIntoArea;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// mouse is over container
|
|
|
|
if (InvalidDockWidgetArea == dropArea)
|
|
|
|
{
|
|
|
|
dropArea = ContainerDropArea;
|
|
|
|
ADS_PRINT("Container Drop Content: " << dropArea);
|
|
|
|
if (dropArea != InvalidDockWidgetArea)
|
|
|
|
{
|
|
|
|
return DropModeIntoContainer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return DropModeInvalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-05 16:07:18 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::onVisibleDockAreaCountChanged()
|
|
|
|
{
|
|
|
|
auto TopLevelDockArea = _this->topLevelDockArea();
|
|
|
|
|
|
|
|
if (TopLevelDockArea)
|
|
|
|
{
|
|
|
|
this->TopLevelDockArea = TopLevelDockArea;
|
|
|
|
TopLevelDockArea->titleBarButton(TitleBarButtonUndock)->setVisible(false || !_this->isFloating());
|
2018-12-02 19:09:31 +08:00
|
|
|
TopLevelDockArea->titleBarButton(TitleBarButtonClose)->setVisible(false || !_this->isFloating());
|
2018-11-05 16:07:18 +08:00
|
|
|
}
|
|
|
|
else if (this->TopLevelDockArea)
|
|
|
|
{
|
|
|
|
this->TopLevelDockArea->titleBarButton(TitleBarButtonUndock)->setVisible(true);
|
2018-12-02 19:09:31 +08:00
|
|
|
this->TopLevelDockArea->titleBarButton(TitleBarButtonClose)->setVisible(true);
|
2018-11-05 16:07:18 +08:00
|
|
|
this->TopLevelDockArea = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::dropIntoContainer(CFloatingDockContainer* FloatingWidget,
|
|
|
|
DockWidgetArea area)
|
|
|
|
{
|
|
|
|
auto InsertParam = internal::dockAreaInsertParameters(area);
|
2018-09-14 19:21:29 +08:00
|
|
|
CDockContainerWidget* FloatingDockContainer = FloatingWidget->dockContainer();
|
|
|
|
auto NewDockAreas = FloatingDockContainer->findChildren<CDockAreaWidget*>(
|
2017-03-23 17:23:53 +08:00
|
|
|
QString(), Qt::FindChildrenRecursively);
|
2017-03-24 19:54:43 +08:00
|
|
|
QSplitter* Splitter = RootSplitter;
|
2017-03-23 17:23:53 +08:00
|
|
|
|
2017-03-24 19:54:43 +08:00
|
|
|
if (DockAreas.count() <= 1)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
Splitter->setOrientation(InsertParam.orientation());
|
|
|
|
}
|
2017-03-28 14:48:44 +08:00
|
|
|
else if (Splitter->orientation() != InsertParam.orientation())
|
2017-03-24 19:54:43 +08:00
|
|
|
{
|
2018-12-11 22:19:59 +08:00
|
|
|
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
2017-03-24 19:54:43 +08:00
|
|
|
QLayoutItem* li = Layout->replaceWidget(Splitter, NewSplitter);
|
|
|
|
NewSplitter->addWidget(Splitter);
|
|
|
|
Splitter = NewSplitter;
|
2017-03-29 18:18:49 +08:00
|
|
|
delete li;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now we can insert the floating widget content into this container
|
2018-09-14 19:21:29 +08:00
|
|
|
auto FloatingSplitter = FloatingDockContainer->rootSplitter();
|
2017-03-28 14:48:44 +08:00
|
|
|
if (FloatingSplitter->count() == 1)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
insertWidgetIntoSplitter(Splitter, FloatingSplitter->widget(0), InsertParam.append());
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
else if (FloatingSplitter->orientation() == InsertParam.orientation())
|
|
|
|
{
|
|
|
|
while (FloatingSplitter->count())
|
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
insertWidgetIntoSplitter(Splitter, FloatingSplitter->widget(0), InsertParam.append());
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
insertWidgetIntoSplitter(Splitter, FloatingSplitter, InsertParam.append());
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
2017-03-24 19:54:43 +08:00
|
|
|
RootSplitter = Splitter;
|
2017-03-23 17:23:53 +08:00
|
|
|
addDockAreasToList(NewDockAreas);
|
2018-09-14 19:21:29 +08:00
|
|
|
|
2018-09-07 17:10:14 +08:00
|
|
|
// 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())
|
2017-09-06 21:45:22 +08:00
|
|
|
{
|
2018-09-07 17:10:14 +08:00
|
|
|
Splitter->show();
|
2017-09-06 21:45:22 +08:00
|
|
|
}
|
2017-03-28 14:48:44 +08:00
|
|
|
_this->dumpLayout();
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-01-14 20:58:40 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::dropIntoCenterOfSection(
|
|
|
|
CFloatingDockContainer* FloatingWidget, CDockAreaWidget* TargetArea)
|
|
|
|
{
|
|
|
|
CDockContainerWidget* FloatingContainer = FloatingWidget->dockContainer();
|
|
|
|
auto NewDockWidgets = FloatingContainer->dockWidgets();
|
|
|
|
auto TopLevelDockArea = FloatingContainer->topLevelDockArea();
|
|
|
|
int NewCurrentIndex = -1;
|
|
|
|
|
|
|
|
// If the floating widget contains only one single dock are, then the
|
|
|
|
// current dock widget of the dock area will also be the future current
|
|
|
|
// dock widget in the drop area.
|
|
|
|
if (TopLevelDockArea)
|
|
|
|
{
|
|
|
|
NewCurrentIndex = TopLevelDockArea->currentIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < NewDockWidgets.count(); ++i)
|
|
|
|
{
|
|
|
|
CDockWidget* DockWidget = NewDockWidgets[i];
|
|
|
|
TargetArea->insertDockWidget(i, DockWidget, false);
|
|
|
|
// If the floating widget contains multiple visible dock areas, then we
|
|
|
|
// simply pick the first visible open dock widget and make it
|
|
|
|
// the current one.
|
|
|
|
if (NewCurrentIndex < 0 && !DockWidget->isClosed())
|
|
|
|
{
|
|
|
|
NewCurrentIndex = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TargetArea->setCurrentIndex(NewCurrentIndex);
|
|
|
|
TargetArea->updateTitleBarVisibility();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::dropIntoSection(CFloatingDockContainer* FloatingWidget,
|
|
|
|
CDockAreaWidget* TargetArea, DockWidgetArea area)
|
|
|
|
{
|
2018-12-11 22:19:59 +08:00
|
|
|
// Dropping into center means all dock widgets in the dropped floating
|
|
|
|
// widget will become tabs of the drop area
|
|
|
|
if (CenterDockWidgetArea == area)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2019-01-14 20:58:40 +08:00
|
|
|
dropIntoCenterOfSection(FloatingWidget, TargetArea);
|
2017-03-23 17:23:53 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto InsertParam = internal::dockAreaInsertParameters(area);
|
|
|
|
auto NewDockAreas = FloatingWidget->dockContainer()->findChildren<CDockAreaWidget*>(
|
|
|
|
QString(), Qt::FindChildrenRecursively);
|
|
|
|
QSplitter* TargetAreaSplitter = internal::findParent<QSplitter*>(TargetArea);
|
|
|
|
|
|
|
|
if (!TargetAreaSplitter)
|
|
|
|
{
|
2018-12-11 22:19:59 +08:00
|
|
|
QSplitter* Splitter = newSplitter(InsertParam.orientation());
|
2017-03-23 17:23:53 +08:00
|
|
|
Layout->replaceWidget(TargetArea, Splitter);
|
|
|
|
Splitter->addWidget(TargetArea);
|
|
|
|
TargetAreaSplitter = Splitter;
|
|
|
|
}
|
|
|
|
int AreaIndex = TargetAreaSplitter->indexOf(TargetArea);
|
|
|
|
auto Widget = FloatingWidget->dockContainer()->findChild<QWidget*>(QString(), Qt::FindDirectChildrenOnly);
|
2019-01-14 20:58:40 +08:00
|
|
|
auto FloatingSplitter = qobject_cast<QSplitter*>(Widget);
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
if (TargetAreaSplitter->orientation() == InsertParam.orientation())
|
|
|
|
{
|
2019-01-14 20:58:40 +08:00
|
|
|
auto Sizes = TargetAreaSplitter->sizes();
|
|
|
|
int TargetAreaSize = (InsertParam.orientation() == Qt::Horizontal) ? TargetArea->width() : TargetArea->height();
|
|
|
|
bool AdjustSplitterSizes = true;
|
2017-03-24 19:54:43 +08:00
|
|
|
if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
|
|
|
TargetAreaSplitter->insertWidget(AreaIndex + InsertParam.insertOffset(), Widget);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-14 20:58:40 +08:00
|
|
|
AdjustSplitterSizes = (FloatingSplitter->count() == 1);
|
2017-03-23 17:23:53 +08:00
|
|
|
int InsertIndex = AreaIndex + InsertParam.insertOffset();
|
|
|
|
while (FloatingSplitter->count())
|
|
|
|
{
|
|
|
|
TargetAreaSplitter->insertWidget(InsertIndex++, FloatingSplitter->widget(0));
|
|
|
|
}
|
|
|
|
}
|
2019-01-14 20:58:40 +08:00
|
|
|
|
|
|
|
if (AdjustSplitterSizes)
|
|
|
|
{
|
|
|
|
int Size = (TargetAreaSize - TargetAreaSplitter->handleWidth()) / 2;
|
|
|
|
Sizes[AreaIndex] = Size;
|
|
|
|
Sizes.insert(AreaIndex, Size);
|
|
|
|
TargetAreaSplitter->setSizes(Sizes);
|
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-14 20:58:40 +08:00
|
|
|
QList<int> NewSplitterSizes;
|
2018-12-11 22:19:59 +08:00
|
|
|
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
2019-01-14 20:58:40 +08:00
|
|
|
int TargetAreaSize = (InsertParam.orientation() == Qt::Horizontal) ? TargetArea->width() : TargetArea->height();
|
|
|
|
bool AdjustSplitterSizes = true;
|
2017-03-24 19:54:43 +08:00
|
|
|
if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
|
|
|
NewSplitter->addWidget(Widget);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-14 20:58:40 +08:00
|
|
|
AdjustSplitterSizes = (FloatingSplitter->count() == 1);
|
2017-03-23 17:23:53 +08:00
|
|
|
while (FloatingSplitter->count())
|
|
|
|
{
|
|
|
|
NewSplitter->addWidget(FloatingSplitter->widget(0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-14 20:58:40 +08:00
|
|
|
// Save the sizes before insertion and restore it later to prevent
|
|
|
|
// shrinking of existing area
|
|
|
|
auto Sizes = TargetAreaSplitter->sizes();
|
2018-09-14 21:02:47 +08:00
|
|
|
insertWidgetIntoSplitter(NewSplitter, TargetArea, !InsertParam.append());
|
2019-01-14 20:58:40 +08:00
|
|
|
if (AdjustSplitterSizes)
|
|
|
|
{
|
|
|
|
int Size = TargetAreaSize / 2;
|
|
|
|
NewSplitter->setSizes({Size, Size});
|
|
|
|
}
|
|
|
|
TargetAreaSplitter->insertWidget(AreaIndex, NewSplitter);
|
|
|
|
TargetAreaSplitter->setSizes(Sizes);
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
addDockAreasToList(NewDockAreas);
|
2017-03-28 14:48:44 +08:00
|
|
|
_this->dumpLayout();
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-26 21:40:56 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::moveIntoCenterOfSection(QWidget* Widget, CDockAreaWidget* TargetArea)
|
|
|
|
{
|
|
|
|
auto DroppedDockWidget = qobject_cast<CDockWidget*>(Widget);
|
|
|
|
auto DroppedArea = qobject_cast<CDockAreaWidget*>(Widget);
|
|
|
|
|
|
|
|
if (DroppedDockWidget)
|
|
|
|
{
|
|
|
|
CDockAreaWidget* OldDockArea = DroppedDockWidget->dockAreaWidget();
|
|
|
|
if (OldDockArea)
|
|
|
|
{
|
|
|
|
OldDockArea->removeDockWidget(DroppedDockWidget);
|
|
|
|
}
|
2019-12-17 20:45:33 +08:00
|
|
|
TargetArea->insertDockWidget(0, DroppedDockWidget, true);
|
2019-11-26 21:40:56 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QList<CDockWidget*> NewDockWidgets = DroppedArea->dockWidgets();
|
|
|
|
int NewCurrentIndex = DroppedArea->currentIndex();
|
|
|
|
for (int i = 0; i < NewDockWidgets.count(); ++i)
|
|
|
|
{
|
|
|
|
CDockWidget* DockWidget = NewDockWidgets[i];
|
|
|
|
TargetArea->insertDockWidget(i, DockWidget, false);
|
|
|
|
}
|
|
|
|
TargetArea->setCurrentIndex(NewCurrentIndex);
|
|
|
|
DroppedArea->dockContainer()->removeDockArea(DroppedArea);
|
|
|
|
DroppedArea->deleteLater();
|
|
|
|
}
|
|
|
|
|
|
|
|
TargetArea->updateTitleBarVisibility();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
2019-11-28 04:43:36 +08:00
|
|
|
void DockContainerWidgetPrivate::moveToNewSection(QWidget* Widget, CDockAreaWidget* TargetArea, DockWidgetArea area)
|
2019-11-26 21:40:56 +08:00
|
|
|
{
|
|
|
|
// Dropping into center means all dock widgets in the dropped floating
|
|
|
|
// widget will become tabs of the drop area
|
|
|
|
if (CenterDockWidgetArea == area)
|
|
|
|
{
|
2019-11-28 04:43:36 +08:00
|
|
|
moveIntoCenterOfSection(Widget, TargetArea);
|
2019-11-26 21:40:56 +08:00
|
|
|
return;
|
|
|
|
}
|
2019-11-27 19:00:04 +08:00
|
|
|
|
|
|
|
|
|
|
|
CDockWidget* DroppedDockWidget = qobject_cast<CDockWidget*>(Widget);
|
|
|
|
CDockAreaWidget* DroppedDockArea = qobject_cast<CDockAreaWidget*>(Widget);
|
|
|
|
CDockAreaWidget* NewDockArea;
|
|
|
|
if (DroppedDockWidget)
|
|
|
|
{
|
|
|
|
NewDockArea = new CDockAreaWidget(DockManager, _this);
|
|
|
|
CDockAreaWidget* OldDockArea = DroppedDockWidget->dockAreaWidget();
|
|
|
|
if (OldDockArea)
|
|
|
|
{
|
|
|
|
OldDockArea->removeDockWidget(DroppedDockWidget);
|
|
|
|
}
|
|
|
|
NewDockArea->addDockWidget(DroppedDockWidget);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DroppedDockArea->dockContainer()->removeDockArea(DroppedDockArea);
|
|
|
|
NewDockArea = DroppedDockArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto InsertParam = internal::dockAreaInsertParameters(area);
|
2019-11-28 04:43:36 +08:00
|
|
|
QSplitter* TargetAreaSplitter = internal::findParent<QSplitter*>(TargetArea);
|
|
|
|
int AreaIndex = TargetAreaSplitter->indexOf(TargetArea);
|
|
|
|
auto Sizes = TargetAreaSplitter->sizes();
|
2019-11-27 19:00:04 +08:00
|
|
|
if (TargetAreaSplitter->orientation() == InsertParam.orientation())
|
|
|
|
{
|
2019-11-28 04:43:36 +08:00
|
|
|
int TargetAreaSize = (InsertParam.orientation() == Qt::Horizontal) ? TargetArea->width() : TargetArea->height();
|
|
|
|
TargetAreaSplitter->insertWidget(AreaIndex + InsertParam.insertOffset(), NewDockArea);
|
|
|
|
int Size = (TargetAreaSize - TargetAreaSplitter->handleWidth()) / 2;
|
|
|
|
Sizes[AreaIndex] = Size;
|
|
|
|
Sizes.insert(AreaIndex, Size);
|
2019-11-27 19:00:04 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-28 04:43:36 +08:00
|
|
|
auto Sizes = TargetAreaSplitter->sizes();
|
|
|
|
int TargetAreaSize = (InsertParam.orientation() == Qt::Horizontal) ? TargetArea->width() : TargetArea->height();
|
2019-11-27 19:00:04 +08:00
|
|
|
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
2019-11-28 04:43:36 +08:00
|
|
|
NewSplitter->addWidget(TargetArea);
|
2019-11-27 19:00:04 +08:00
|
|
|
insertWidgetIntoSplitter(NewSplitter, NewDockArea, InsertParam.append());
|
2019-11-28 04:43:36 +08:00
|
|
|
int Size = TargetAreaSize / 2;
|
|
|
|
NewSplitter->setSizes({Size, Size});
|
|
|
|
TargetAreaSplitter->insertWidget(AreaIndex, NewSplitter);
|
2019-11-27 19:00:04 +08:00
|
|
|
}
|
2019-11-28 04:43:36 +08:00
|
|
|
TargetAreaSplitter->setSizes(Sizes);
|
2019-11-27 19:00:04 +08:00
|
|
|
|
|
|
|
addDockAreasToList({NewDockArea});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::moveToContainer(QWidget* Widget, DockWidgetArea area)
|
|
|
|
{
|
|
|
|
CDockWidget* DroppedDockWidget = qobject_cast<CDockWidget*>(Widget);
|
|
|
|
CDockAreaWidget* DroppedDockArea = qobject_cast<CDockAreaWidget*>(Widget);
|
|
|
|
CDockAreaWidget* NewDockArea;
|
2019-11-27 22:50:18 +08:00
|
|
|
|
2019-11-27 19:00:04 +08:00
|
|
|
if (DroppedDockWidget)
|
|
|
|
{
|
|
|
|
NewDockArea = new CDockAreaWidget(DockManager, _this);
|
|
|
|
CDockAreaWidget* OldDockArea = DroppedDockWidget->dockAreaWidget();
|
|
|
|
if (OldDockArea)
|
|
|
|
{
|
|
|
|
OldDockArea->removeDockWidget(DroppedDockWidget);
|
|
|
|
}
|
|
|
|
NewDockArea->addDockWidget(DroppedDockWidget);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DroppedDockArea->dockContainer()->removeDockArea(DroppedDockArea);
|
|
|
|
NewDockArea = DroppedDockArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
addDockArea(NewDockArea, area);
|
|
|
|
LastAddedAreaCache[areaIdToIndex(area)] = NewDockArea;
|
2019-11-26 21:40:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::addDockAreasToList(const QList<CDockAreaWidget*> NewDockAreas)
|
|
|
|
{
|
|
|
|
int CountBefore = DockAreas.count();
|
|
|
|
int NewAreaCount = NewDockAreas.count();
|
2018-11-05 16:07:18 +08:00
|
|
|
appendDockAreas(NewDockAreas);
|
|
|
|
// If the user dropped a floating widget that contains only one single
|
|
|
|
// visible dock area, then its title bar button TitleBarButtonUndock is
|
|
|
|
// likely hidden. We need to ensure, that it is visible
|
|
|
|
for (auto DockArea : NewDockAreas)
|
|
|
|
{
|
|
|
|
DockArea->titleBarButton(TitleBarButtonUndock)->setVisible(true);
|
2018-12-02 19:09:31 +08:00
|
|
|
DockArea->titleBarButton(TitleBarButtonClose)->setVisible(true);
|
2018-11-05 16:07:18 +08:00
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
// 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)
|
|
|
|
{
|
2018-11-05 16:07:18 +08:00
|
|
|
DockAreas.at(0)->updateTitleBarVisibility();
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (1 == NewAreaCount)
|
|
|
|
{
|
2018-11-05 16:07:18 +08:00
|
|
|
DockAreas.last()->updateTitleBarVisibility();
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
2018-11-05 16:07:18 +08:00
|
|
|
emitDockAreasAdded();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::appendDockAreas(const QList<CDockAreaWidget*> NewDockAreas)
|
|
|
|
{
|
|
|
|
DockAreas.append(NewDockAreas);
|
|
|
|
for (auto DockArea : NewDockAreas)
|
|
|
|
{
|
2019-01-26 00:28:36 +08:00
|
|
|
QObject::connect(DockArea,
|
|
|
|
&CDockAreaWidget::viewToggled,
|
|
|
|
_this,
|
|
|
|
std::bind(&DockContainerWidgetPrivate::onDockAreaViewToggled, this, std::placeholders::_1));
|
2018-11-05 16:07:18 +08:00
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 22:57:15 +08:00
|
|
|
//============================================================================
|
2017-12-30 01:18:16 +08:00
|
|
|
void DockContainerWidgetPrivate::saveChildNodesState(QXmlStreamWriter& s, QWidget* Widget)
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
|
|
|
QSplitter* Splitter = dynamic_cast<QSplitter*>(Widget);
|
|
|
|
if (Splitter)
|
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
s.writeStartElement("Splitter");
|
2019-11-25 22:24:44 +08:00
|
|
|
s.writeAttribute("Orientation", (Splitter->orientation() == Qt::Horizontal) ? "|" : "-");
|
2017-12-30 01:18:16 +08:00
|
|
|
s.writeAttribute("Count", QString::number(Splitter->count()));
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("NodeSplitter orient: " << Splitter->orientation()
|
|
|
|
<< " WidgetCont: " << Splitter->count());
|
2017-12-30 01:18:16 +08:00
|
|
|
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();
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CDockAreaWidget* DockArea = dynamic_cast<CDockAreaWidget*>(Widget);
|
|
|
|
if (DockArea)
|
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
DockArea->saveState(s);
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
2019-11-29 22:56:57 +08:00
|
|
|
bool DockContainerWidgetPrivate::restoreSplitter(CDockingStateReader& s,
|
2017-03-27 16:41:27 +08:00
|
|
|
QWidget*& CreatedWidget, bool Testing)
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
bool Ok;
|
2018-12-11 22:19:59 +08:00
|
|
|
QString OrientationStr = s.attributes().value("Orientation").toString();
|
2019-11-29 22:56:57 +08:00
|
|
|
|
|
|
|
// Check if the orientation string is right
|
|
|
|
if (!OrientationStr.startsWith("|") && !OrientationStr.startsWith("-"))
|
2018-12-11 22:19:59 +08:00
|
|
|
{
|
2019-11-29 22:56:57 +08:00
|
|
|
return false;
|
2018-12-11 22:19:59 +08:00
|
|
|
}
|
2019-11-29 22:56:57 +08:00
|
|
|
|
|
|
|
// The "|" shall indicate a vertical splitter handle which in turn means
|
|
|
|
// a Horizontal orientation of the splitter layout.
|
|
|
|
bool HorizontalSplitter = OrientationStr.startsWith("|");
|
|
|
|
// In version 0 we had a small bug. The "|" indicated a vertical orientation,
|
|
|
|
// but this is wrong, because only the splitter handle is vertical, the
|
|
|
|
// layout of the splitter is a horizontal layout. We fix this here
|
|
|
|
if (s.fileVersion() == 0)
|
2017-12-30 01:18:16 +08:00
|
|
|
{
|
2019-11-29 22:56:57 +08:00
|
|
|
HorizontalSplitter = !HorizontalSplitter;
|
2017-12-30 01:18:16 +08:00
|
|
|
}
|
|
|
|
|
2019-11-29 22:56:57 +08:00
|
|
|
int Orientation = HorizontalSplitter ? Qt::Horizontal : Qt::Vertical;
|
2017-12-30 01:18:16 +08:00
|
|
|
int WidgetCount = s.attributes().value("Count").toInt(&Ok);
|
|
|
|
if (!Ok)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Restore NodeSplitter Orientation: " << Orientation <<
|
|
|
|
" WidgetCount: " << WidgetCount);
|
2017-03-27 16:41:27 +08:00
|
|
|
QSplitter* Splitter = nullptr;
|
|
|
|
if (!Testing)
|
|
|
|
{
|
2019-01-26 00:28:36 +08:00
|
|
|
Splitter = newSplitter(static_cast<Qt::Orientation>(Orientation));
|
2017-03-27 16:41:27 +08:00
|
|
|
}
|
|
|
|
bool Visible = false;
|
2017-12-30 01:18:16 +08:00
|
|
|
QList<int> Sizes;
|
|
|
|
while (s.readNextStartElement())
|
2017-03-27 16:41:27 +08:00
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
QWidget* ChildNode = nullptr;
|
|
|
|
bool Result = true;
|
|
|
|
if (s.name() == "Splitter")
|
|
|
|
{
|
|
|
|
Result = restoreSplitter(s, ChildNode, Testing);
|
|
|
|
}
|
2018-12-11 22:19:59 +08:00
|
|
|
else if (s.name() == "Area")
|
2017-12-30 01:18:16 +08:00
|
|
|
{
|
|
|
|
Result = restoreDockArea(s, ChildNode, Testing);
|
|
|
|
}
|
|
|
|
else if (s.name() == "Sizes")
|
|
|
|
{
|
|
|
|
QString sSizes = s.readElementText().trimmed();
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Sizes: " << sSizes);
|
2017-12-30 01:18:16 +08:00
|
|
|
QTextStream TextStream(&sSizes);
|
|
|
|
while (!TextStream.atEnd())
|
|
|
|
{
|
|
|
|
int value;
|
|
|
|
TextStream >> value;
|
|
|
|
Sizes.append(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s.skipCurrentElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Result)
|
2017-03-24 17:18:25 +08:00
|
|
|
{
|
2017-03-27 16:41:27 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
if (Testing || !ChildNode)
|
2017-03-27 16:41:27 +08:00
|
|
|
{
|
|
|
|
continue;
|
2017-03-24 17:18:25 +08:00
|
|
|
}
|
|
|
|
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("ChildNode isVisible " << ChildNode->isVisible()
|
|
|
|
<< " isVisibleTo " << ChildNode->isVisibleTo(Splitter));
|
2017-03-27 16:41:27 +08:00
|
|
|
Splitter->addWidget(ChildNode);
|
|
|
|
Visible |= ChildNode->isVisibleTo(Splitter);
|
|
|
|
}
|
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
if (Sizes.count() != WidgetCount)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-27 16:41:27 +08:00
|
|
|
if (!Testing)
|
|
|
|
{
|
2017-03-24 23:17:55 +08:00
|
|
|
if (!Splitter->count())
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
2017-03-24 23:17:55 +08:00
|
|
|
delete Splitter;
|
|
|
|
Splitter = nullptr;
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-24 23:17:55 +08:00
|
|
|
Splitter->setSizes(Sizes);
|
|
|
|
Splitter->setVisible(Visible);
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
2017-03-24 23:17:55 +08:00
|
|
|
CreatedWidget = Splitter;
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-27 16:41:27 +08:00
|
|
|
CreatedWidget = nullptr;
|
|
|
|
}
|
2017-12-30 01:18:16 +08:00
|
|
|
|
2017-03-27 16:41:27 +08:00
|
|
|
return true;
|
|
|
|
}
|
2017-03-24 23:17:55 +08:00
|
|
|
|
2017-03-27 16:41:27 +08:00
|
|
|
|
|
|
|
//============================================================================
|
2019-11-29 22:56:57 +08:00
|
|
|
bool DockContainerWidgetPrivate::restoreDockArea(CDockingStateReader& s,
|
2017-03-27 16:41:27 +08:00
|
|
|
QWidget*& CreatedWidget, bool Testing)
|
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
bool Ok;
|
2019-10-01 17:48:53 +08:00
|
|
|
#ifdef ADS_DEBUG_PRINT
|
2017-12-30 01:18:16 +08:00
|
|
|
int Tabs = s.attributes().value("Tabs").toInt(&Ok);
|
|
|
|
if (!Ok)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2019-10-01 17:48:53 +08:00
|
|
|
#endif
|
2018-11-01 15:52:14 +08:00
|
|
|
|
2018-12-11 22:19:59 +08:00
|
|
|
QString CurrentDockWidget = s.attributes().value("Current").toString();
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Restore NodeDockArea Tabs: " << Tabs << " Current: "
|
|
|
|
<< CurrentDockWidget);
|
2017-03-27 16:41:27 +08:00
|
|
|
|
|
|
|
CDockAreaWidget* DockArea = nullptr;
|
|
|
|
if (!Testing)
|
|
|
|
{
|
|
|
|
DockArea = new CDockAreaWidget(DockManager, _this);
|
|
|
|
}
|
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
while (s.readNextStartElement())
|
2017-03-27 16:41:27 +08:00
|
|
|
{
|
2018-12-11 22:19:59 +08:00
|
|
|
if (s.name() != "Widget")
|
2017-12-30 01:18:16 +08:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-12-11 22:19:59 +08:00
|
|
|
auto ObjectName = s.attributes().value("Name");
|
2017-12-30 01:18:16 +08:00
|
|
|
if (ObjectName.isEmpty())
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
2017-03-27 16:41:27 +08:00
|
|
|
return false;
|
2017-03-24 23:17:55 +08:00
|
|
|
}
|
2017-03-24 17:18:25 +08:00
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
bool Closed = s.attributes().value("Closed").toInt(&Ok);
|
|
|
|
if (!Ok)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-27 16:41:27 +08:00
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
s.skipCurrentElement();
|
|
|
|
CDockWidget* DockWidget = DockManager->findDockWidget(ObjectName.toString());
|
2017-03-27 16:41:27 +08:00
|
|
|
if (!DockWidget || Testing)
|
2017-03-24 23:17:55 +08:00
|
|
|
{
|
2017-03-27 16:41:27 +08:00
|
|
|
continue;
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
2017-03-27 16:41:27 +08:00
|
|
|
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Dock Widget found - parent " << DockWidget->parent());
|
2018-09-07 17:10:14 +08:00
|
|
|
// We hide the DockArea here to prevent the short display (the flashing)
|
|
|
|
// of the dock areas during application startup
|
2017-03-29 04:38:47 +08:00
|
|
|
DockArea->hide();
|
2018-11-08 19:22:15 +08:00
|
|
|
DockArea->addDockWidget(DockWidget);
|
2017-03-29 21:43:18 +08:00
|
|
|
DockWidget->setToggleViewActionChecked(!Closed);
|
2018-09-26 15:57:36 +08:00
|
|
|
DockWidget->setClosedState(Closed);
|
2019-10-01 17:48:53 +08:00
|
|
|
DockWidget->setProperty(internal::ClosedProperty, Closed);
|
|
|
|
DockWidget->setProperty(internal::DirtyProperty, false);
|
2017-03-27 16:41:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Testing)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-09-14 14:46:10 +08:00
|
|
|
if (!DockArea->dockWidgetsCount())
|
2017-03-27 16:41:27 +08:00
|
|
|
{
|
|
|
|
delete DockArea;
|
|
|
|
DockArea = nullptr;
|
|
|
|
}
|
2017-03-29 04:38:47 +08:00
|
|
|
else
|
|
|
|
{
|
2018-11-01 15:52:14 +08:00
|
|
|
DockArea->setProperty("currentDockWidget", CurrentDockWidget);
|
2018-11-05 16:07:18 +08:00
|
|
|
appendDockAreas({DockArea});
|
2017-03-29 04:38:47 +08:00
|
|
|
}
|
2018-02-13 19:00:58 +08:00
|
|
|
|
2017-03-27 16:41:27 +08:00
|
|
|
CreatedWidget = DockArea;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
2019-11-29 22:56:57 +08:00
|
|
|
bool DockContainerWidgetPrivate::restoreChildNodes(CDockingStateReader& s,
|
2017-03-27 16:41:27 +08:00
|
|
|
QWidget*& CreatedWidget, bool Testing)
|
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
bool Result = true;
|
|
|
|
while (s.readNextStartElement())
|
2017-03-27 16:41:27 +08:00
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
if (s.name() == "Splitter")
|
|
|
|
{
|
|
|
|
Result = restoreSplitter(s, CreatedWidget, Testing);
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Splitter");
|
2017-12-30 01:18:16 +08:00
|
|
|
}
|
2018-12-11 22:19:59 +08:00
|
|
|
else if (s.name() == "Area")
|
2017-12-30 01:18:16 +08:00
|
|
|
{
|
|
|
|
Result = restoreDockArea(s, CreatedWidget, Testing);
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("DockAreaWidget");
|
2017-12-30 01:18:16 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s.skipCurrentElement();
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Unknown element");
|
2017-12-30 01:18:16 +08:00
|
|
|
}
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
2017-12-30 01:18:16 +08:00
|
|
|
|
|
|
|
return Result;
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
CDockAreaWidget* DockContainerWidgetPrivate::dockWidgetIntoContainer(DockWidgetArea area,
|
|
|
|
CDockWidget* Dockwidget)
|
|
|
|
{
|
|
|
|
CDockAreaWidget* NewDockArea = new CDockAreaWidget(DockManager, _this);
|
|
|
|
NewDockArea->addDockWidget(Dockwidget);
|
|
|
|
addDockArea(NewDockArea, area);
|
2018-11-05 16:07:18 +08:00
|
|
|
NewDockArea->updateTitleBarVisibility();
|
2018-09-07 17:10:14 +08:00
|
|
|
LastAddedAreaCache[areaIdToIndex(area)] = NewDockArea;
|
2017-03-23 17:23:53 +08:00
|
|
|
return NewDockArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::addDockArea(CDockAreaWidget* NewDockArea, DockWidgetArea area)
|
|
|
|
{
|
|
|
|
auto InsertParam = internal::dockAreaInsertParameters(area);
|
2017-03-24 19:54:43 +08:00
|
|
|
// As long as we have only one dock area in the splitter we can adjust
|
|
|
|
// its orientation
|
|
|
|
if (DockAreas.count() <= 1)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
RootSplitter->setOrientation(InsertParam.orientation());
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
2017-03-24 19:54:43 +08:00
|
|
|
|
|
|
|
QSplitter* Splitter = RootSplitter;
|
|
|
|
if (Splitter->orientation() == InsertParam.orientation())
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
|
|
|
insertWidgetIntoSplitter(Splitter, NewDockArea, InsertParam.append());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-12-11 22:19:59 +08:00
|
|
|
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
2017-03-24 19:54:43 +08:00
|
|
|
if (InsertParam.append())
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
QLayoutItem* li = Layout->replaceWidget(Splitter, NewSplitter);
|
|
|
|
NewSplitter->addWidget(Splitter);
|
|
|
|
NewSplitter->addWidget(NewDockArea);
|
|
|
|
delete li;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-24 19:54:43 +08:00
|
|
|
NewSplitter->addWidget(NewDockArea);
|
|
|
|
QLayoutItem* li = Layout->replaceWidget(Splitter, NewSplitter);
|
|
|
|
NewSplitter->addWidget(Splitter);
|
|
|
|
delete li;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
2017-03-24 19:54:43 +08:00
|
|
|
RootSplitter = NewSplitter;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
2019-11-27 19:00:04 +08:00
|
|
|
addDockAreasToList({NewDockArea});
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-28 14:48:44 +08:00
|
|
|
//============================================================================
|
|
|
|
void DockContainerWidgetPrivate::dumpRecursive(int level, QWidget* widget)
|
|
|
|
{
|
2017-03-28 16:57:03 +08:00
|
|
|
#if defined(QT_DEBUG)
|
2017-03-28 14:48:44 +08:00
|
|
|
QSplitter* Splitter = dynamic_cast<QSplitter*>(widget);
|
|
|
|
QByteArray buf;
|
|
|
|
buf.fill(' ', level * 4);
|
|
|
|
if (Splitter)
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
#ifdef ADS_DEBUG_PRINT
|
2017-09-06 21:45:22 +08:00
|
|
|
qDebug("%sSplitter %s v: %s c: %s",
|
|
|
|
(const char*)buf,
|
2018-12-03 19:52:57 +08:00
|
|
|
(Splitter->orientation() == Qt::Vertical) ? "--" : "|",
|
|
|
|
Splitter->isHidden() ? " " : "v",
|
2017-09-06 21:45:22 +08:00
|
|
|
QString::number(Splitter->count()).toStdString().c_str());
|
2019-07-21 15:53:24 +08:00
|
|
|
std::cout << (const char*)buf << "Splitter "
|
|
|
|
<< ((Splitter->orientation() == Qt::Vertical) ? "--" : "|") << " "
|
|
|
|
<< (Splitter->isHidden() ? " " : "v") << " "
|
|
|
|
<< QString::number(Splitter->count()).toStdString() << std::endl;
|
|
|
|
#endif
|
2017-03-28 14:48:44 +08:00
|
|
|
for (int i = 0; i < Splitter->count(); ++i)
|
|
|
|
{
|
|
|
|
dumpRecursive(level + 1, Splitter->widget(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CDockAreaWidget* DockArea = dynamic_cast<CDockAreaWidget*>(widget);
|
|
|
|
if (!DockArea)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2019-07-21 15:53:24 +08:00
|
|
|
#ifdef ADS_DEBUG_PRINT
|
2017-03-28 18:01:27 +08:00
|
|
|
qDebug("%sDockArea", (const char*)buf);
|
2018-12-03 19:52:57 +08:00
|
|
|
std::cout << (const char*)buf
|
|
|
|
<< (DockArea->isHidden() ? " " : "v")
|
|
|
|
<< (DockArea->openDockWidgetsCount() > 0 ? " " : "c")
|
|
|
|
<< " DockArea" << std::endl;
|
|
|
|
buf.fill(' ', (level + 1) * 4);
|
|
|
|
for (int i = 0; i < DockArea->dockWidgetsCount(); ++i)
|
|
|
|
{
|
|
|
|
std::cout << (const char*)buf << (i == DockArea->currentIndex() ? "*" : " ");
|
|
|
|
CDockWidget* DockWidget = DockArea->dockWidget(i);
|
|
|
|
std::cout << (DockWidget->isHidden() ? " " : "v");
|
|
|
|
std::cout << (DockWidget->isClosed() ? "c" : " ") << " ";
|
|
|
|
std::cout << DockWidget->windowTitle().toStdString() << std::endl;
|
2019-07-21 15:53:24 +08:00
|
|
|
}
|
|
|
|
#endif
|
2017-03-28 14:48:44 +08:00
|
|
|
}
|
2017-03-29 18:18:49 +08:00
|
|
|
#else
|
|
|
|
Q_UNUSED(level);
|
|
|
|
Q_UNUSED(widget);
|
2017-03-28 16:57:03 +08:00
|
|
|
#endif
|
2017-03-28 14:48:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
CDockAreaWidget* DockContainerWidgetPrivate::dockWidgetIntoDockArea(DockWidgetArea area,
|
|
|
|
CDockWidget* Dockwidget, CDockAreaWidget* TargetDockArea)
|
|
|
|
{
|
|
|
|
if (CenterDockWidgetArea == area)
|
|
|
|
{
|
|
|
|
TargetDockArea->addDockWidget(Dockwidget);
|
2019-09-30 23:33:40 +08:00
|
|
|
TargetDockArea->updateTitleBarVisibility();
|
2017-03-23 17:23:53 +08:00
|
|
|
return TargetDockArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
CDockAreaWidget* NewDockArea = new CDockAreaWidget(DockManager, _this);
|
|
|
|
NewDockArea->addDockWidget(Dockwidget);
|
|
|
|
auto InsertParam = internal::dockAreaInsertParameters(area);
|
2017-03-24 19:54:43 +08:00
|
|
|
|
|
|
|
QSplitter* TargetAreaSplitter = internal::findParent<QSplitter*>(TargetDockArea);
|
|
|
|
int index = TargetAreaSplitter ->indexOf(TargetDockArea);
|
|
|
|
if (TargetAreaSplitter->orientation() == InsertParam.orientation())
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("TargetAreaSplitter->orientation() == InsertParam.orientation()");
|
2017-03-24 19:54:43 +08:00
|
|
|
TargetAreaSplitter->insertWidget(index + InsertParam.insertOffset(), NewDockArea);
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("TargetAreaSplitter->orientation() != InsertParam.orientation()");
|
2018-12-11 22:19:59 +08:00
|
|
|
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
2017-03-24 19:54:43 +08:00
|
|
|
NewSplitter->addWidget(TargetDockArea);
|
|
|
|
insertWidgetIntoSplitter(NewSplitter, NewDockArea, InsertParam.append());
|
|
|
|
TargetAreaSplitter->insertWidget(index, NewSplitter);
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
2018-11-05 16:07:18 +08:00
|
|
|
appendDockAreas({NewDockArea});
|
|
|
|
emitDockAreasAdded();
|
2017-03-23 17:23:53 +08:00
|
|
|
return NewDockArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CDockContainerWidget::CDockContainerWidget(CDockManager* DockManager, QWidget *parent) :
|
|
|
|
QFrame(parent),
|
|
|
|
d(new DockContainerWidgetPrivate(this))
|
|
|
|
{
|
|
|
|
d->DockManager = DockManager;
|
2018-12-11 22:19:59 +08:00
|
|
|
d->isFloating = floatingWidget() != nullptr;
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
d->Layout = new QGridLayout();
|
|
|
|
d->Layout->setContentsMargins(0, 1, 0, 1);
|
|
|
|
d->Layout->setSpacing(0);
|
|
|
|
setLayout(d->Layout);
|
2017-03-24 19:54:43 +08:00
|
|
|
|
2018-12-11 22:19:59 +08:00
|
|
|
// The function d->newSplitter() accesses the config flags from dock
|
|
|
|
// manager which in turn requires a properly constructed dock manager.
|
|
|
|
// If this dock container is the dock manager, then it is not properly
|
|
|
|
// constructed yet because this base class constructor is called before
|
|
|
|
// the constructor of the DockManager private class
|
|
|
|
if (DockManager != this)
|
|
|
|
{
|
|
|
|
d->DockManager->registerDockContainer(this);
|
|
|
|
createRootSplitter();
|
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-15 20:47:58 +08:00
|
|
|
//============================================================================
|
|
|
|
void CDockContainerWidget::removeDockWidget(CDockWidget* Dockwidget)
|
|
|
|
{
|
|
|
|
CDockAreaWidget* Area = Dockwidget->dockAreaWidget();
|
|
|
|
if (Area)
|
|
|
|
{
|
|
|
|
Area->removeDockWidget(Dockwidget);
|
|
|
|
}
|
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2017-03-29 18:18:49 +08:00
|
|
|
d->addDockArea(DockAreaWidget, area);
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
void CDockContainerWidget::removeDockArea(CDockAreaWidget* area)
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("CDockContainerWidget::removeDockArea");
|
2018-11-05 16:07:18 +08:00
|
|
|
area->disconnect(this);
|
2017-03-23 17:23:53 +08:00
|
|
|
d->DockAreas.removeAll(area);
|
2017-09-06 21:45:22 +08:00
|
|
|
CDockSplitter* Splitter = internal::findParent<CDockSplitter*>(area);
|
|
|
|
|
2018-12-03 19:52:57 +08:00
|
|
|
// Remove are from parent splitter and recursively hide tree of parent
|
|
|
|
// splitters if it has no visible content
|
2019-01-26 00:28:36 +08:00
|
|
|
area->setParent(nullptr);
|
2018-12-03 19:52:57 +08:00
|
|
|
internal::hideEmptyParentSplitters(Splitter);
|
2017-09-06 21:45:22 +08:00
|
|
|
|
2019-05-15 21:20:08 +08:00
|
|
|
// Remove this area from cached areas
|
|
|
|
const auto& cache = d->LastAddedAreaCache;
|
|
|
|
if (auto p = std::find(cache, cache+sizeof(cache)/sizeof(cache[0]), area)) {
|
|
|
|
d->LastAddedAreaCache[std::distance(cache, p)] = nullptr;
|
|
|
|
}
|
|
|
|
|
2017-09-06 21:45:22 +08:00
|
|
|
// If splitter has more than 1 widgets, we are finished and can leave
|
|
|
|
if (Splitter->count() > 1)
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
2017-09-06 21:45:22 +08:00
|
|
|
goto emitAndExit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this is the RootSplitter we need to remove empty splitters to
|
|
|
|
// avoid too many empty splitters
|
|
|
|
if (Splitter == d->RootSplitter)
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Removed from RootSplitter");
|
2017-09-06 21:45:22 +08:00
|
|
|
// If splitter is empty, we are finished
|
|
|
|
if (!Splitter->count())
|
|
|
|
{
|
|
|
|
Splitter->hide();
|
|
|
|
goto emitAndExit;
|
|
|
|
}
|
|
|
|
|
|
|
|
QWidget* widget = Splitter->widget(0);
|
|
|
|
QSplitter* ChildSplitter = dynamic_cast<QSplitter*>(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
|
2019-01-26 00:28:36 +08:00
|
|
|
ChildSplitter->setParent(nullptr);
|
2017-09-06 21:45:22 +08:00
|
|
|
QLayoutItem* li = d->Layout->replaceWidget(Splitter, ChildSplitter);
|
|
|
|
d->RootSplitter = ChildSplitter;
|
|
|
|
delete li;
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("RootSplitter replaced by child splitter");
|
2017-09-06 21:45:22 +08:00
|
|
|
}
|
|
|
|
else if (Splitter->count() == 1)
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Replacing splitter with content");
|
2019-01-14 20:58:40 +08:00
|
|
|
QSplitter* ParentSplitter = internal::findParent<QSplitter*>(Splitter);
|
|
|
|
auto Sizes = ParentSplitter->sizes();
|
2017-09-06 21:45:22 +08:00
|
|
|
QWidget* widget = Splitter->widget(0);
|
|
|
|
widget->setParent(this);
|
|
|
|
internal::replaceSplitterWidget(ParentSplitter, Splitter, widget);
|
2019-01-14 20:58:40 +08:00
|
|
|
ParentSplitter->setSizes(Sizes);
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
delete Splitter;
|
2017-09-06 21:45:22 +08:00
|
|
|
|
|
|
|
emitAndExit:
|
2018-09-26 15:57:36 +08:00
|
|
|
CDockWidget* TopLevelWidget = topLevelDockWidget();
|
|
|
|
|
|
|
|
// Updated the title bar visibility of the dock widget if there is only
|
2018-09-14 19:21:29 +08:00
|
|
|
// one single visible dock widget
|
2018-09-26 15:57:36 +08:00
|
|
|
CDockWidget::emitTopLevelEventForWidget(TopLevelWidget, true);
|
2017-03-28 14:48:44 +08:00
|
|
|
dumpLayout();
|
2018-11-05 16:07:18 +08:00
|
|
|
d->emitDockAreasRemoved();
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CDockAreaWidget* CDockContainerWidget::dockAreaAt(const QPoint& GlobalPos) const
|
|
|
|
{
|
|
|
|
for (const auto& DockArea : d->DockAreas)
|
|
|
|
{
|
2017-09-06 21:45:22 +08:00
|
|
|
if (DockArea->isVisible() && DockArea->rect().contains(DockArea->mapFromGlobal(GlobalPos)))
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
|
|
|
return DockArea;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-26 00:28:36 +08:00
|
|
|
return nullptr;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CDockAreaWidget* CDockContainerWidget::dockArea(int Index) const
|
|
|
|
{
|
2019-01-26 00:28:36 +08:00
|
|
|
return (Index < dockAreaCount()) ? d->DockAreas[Index] : nullptr;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
bool CDockContainerWidget::isFloating() const
|
|
|
|
{
|
|
|
|
return d->isFloating;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
int CDockContainerWidget::dockAreaCount() const
|
|
|
|
{
|
|
|
|
return d->DockAreas.count();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-29 21:43:18 +08:00
|
|
|
//============================================================================
|
|
|
|
int CDockContainerWidget::visibleDockAreaCount() const
|
|
|
|
{
|
|
|
|
int Result = 0;
|
|
|
|
for (auto DockArea : d->DockAreas)
|
|
|
|
{
|
2018-10-31 06:45:59 +08:00
|
|
|
Result += DockArea->isHidden() ? 0 : 1;
|
2017-03-29 21:43:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
2018-11-01 14:53:54 +08:00
|
|
|
|
|
|
|
// TODO Cache or precalculate this to speed it up because it is used during
|
|
|
|
// movement of floating widget
|
|
|
|
//return d->visibleDockAreaCount();
|
2017-03-29 21:43:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
void CDockContainerWidget::dropFloatingWidget(CFloatingDockContainer* FloatingWidget,
|
|
|
|
const QPoint& TargetPos)
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("CDockContainerWidget::dropFloatingWidget");
|
2019-11-27 22:50:18 +08:00
|
|
|
CDockWidget* SingleDroppedDockWidget = FloatingWidget->topLevelDockWidget();
|
|
|
|
CDockWidget* SingleDockWidget = topLevelDockWidget();
|
2017-03-23 17:23:53 +08:00
|
|
|
CDockAreaWidget* DockArea = dockAreaAt(TargetPos);
|
|
|
|
auto dropArea = InvalidDockWidgetArea;
|
2017-09-01 22:14:43 +08:00
|
|
|
auto ContainerDropArea = d->DockManager->containerOverlay()->dropAreaUnderCursor();
|
2019-12-10 19:47:55 +08:00
|
|
|
bool Dropped = false;
|
2018-11-02 16:19:53 +08:00
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
if (DockArea)
|
|
|
|
{
|
|
|
|
auto dropOverlay = d->DockManager->dockAreaOverlay();
|
2020-02-02 22:56:31 +08:00
|
|
|
dropOverlay->setAllowedAreas(DockArea->allowedAreas());
|
2017-03-23 17:23:53 +08:00
|
|
|
dropArea = dropOverlay->showOverlay(DockArea);
|
2017-09-06 21:45:22 +08:00
|
|
|
if (ContainerDropArea != InvalidDockWidgetArea &&
|
|
|
|
ContainerDropArea != dropArea)
|
2017-09-01 22:14:43 +08:00
|
|
|
{
|
|
|
|
dropArea = InvalidDockWidgetArea;
|
|
|
|
}
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
if (dropArea != InvalidDockWidgetArea)
|
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Dock Area Drop Content: " << dropArea);
|
2017-03-23 17:23:53 +08:00
|
|
|
d->dropIntoSection(FloatingWidget, DockArea, dropArea);
|
2019-12-10 19:47:55 +08:00
|
|
|
Dropped = true;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// mouse is over container
|
|
|
|
if (InvalidDockWidgetArea == dropArea)
|
|
|
|
{
|
2017-09-01 22:14:43 +08:00
|
|
|
dropArea = ContainerDropArea;
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Container Drop Content: " << dropArea);
|
2017-03-23 17:23:53 +08:00
|
|
|
if (dropArea != InvalidDockWidgetArea)
|
|
|
|
{
|
|
|
|
d->dropIntoContainer(FloatingWidget, dropArea);
|
2019-12-10 19:47:55 +08:00
|
|
|
Dropped = true;
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
}
|
2018-09-07 17:10:14 +08:00
|
|
|
|
2019-12-10 19:47:55 +08:00
|
|
|
if (Dropped)
|
|
|
|
{
|
|
|
|
FloatingWidget->deleteLater();
|
|
|
|
|
|
|
|
// If we dropped a floating widget with only one single dock widget, then we
|
|
|
|
// drop a top level widget that changes from floating to docked now
|
|
|
|
CDockWidget::emitTopLevelEventForWidget(SingleDroppedDockWidget, false);
|
2019-11-27 22:50:18 +08:00
|
|
|
|
2019-12-10 19:47:55 +08:00
|
|
|
// If there was a top level widget before the drop, then it is not top
|
|
|
|
// level widget anymore
|
|
|
|
CDockWidget::emitTopLevelEventForWidget(SingleDockWidget, false);
|
|
|
|
}
|
2017-03-23 17:23:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-26 21:40:56 +08:00
|
|
|
//============================================================================
|
2019-11-27 19:00:04 +08:00
|
|
|
void CDockContainerWidget::dropWidget(QWidget* Widget, const QPoint& TargetPos)
|
2019-11-26 21:40:56 +08:00
|
|
|
{
|
|
|
|
ADS_PRINT("CDockContainerWidget::dropFloatingWidget");
|
2019-11-27 22:50:18 +08:00
|
|
|
CDockWidget* SingleDockWidget = topLevelDockWidget();
|
2019-11-26 21:40:56 +08:00
|
|
|
CDockAreaWidget* DockArea = dockAreaAt(TargetPos);
|
|
|
|
auto dropArea = InvalidDockWidgetArea;
|
|
|
|
auto ContainerDropArea = d->DockManager->containerOverlay()->dropAreaUnderCursor();
|
|
|
|
|
|
|
|
if (DockArea)
|
|
|
|
{
|
|
|
|
auto dropOverlay = d->DockManager->dockAreaOverlay();
|
2020-02-02 22:56:31 +08:00
|
|
|
dropOverlay->setAllowedAreas(DockArea->allowedAreas());
|
2019-11-26 21:40:56 +08:00
|
|
|
dropArea = dropOverlay->showOverlay(DockArea);
|
|
|
|
if (ContainerDropArea != InvalidDockWidgetArea &&
|
|
|
|
ContainerDropArea != dropArea)
|
|
|
|
{
|
|
|
|
dropArea = InvalidDockWidgetArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dropArea != InvalidDockWidgetArea)
|
|
|
|
{
|
|
|
|
ADS_PRINT("Dock Area Drop Content: " << dropArea);
|
2019-11-27 19:00:04 +08:00
|
|
|
d->moveToNewSection(Widget, DockArea, dropArea);
|
2019-11-26 21:40:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// mouse is over container
|
|
|
|
if (InvalidDockWidgetArea == dropArea)
|
|
|
|
{
|
|
|
|
dropArea = ContainerDropArea;
|
|
|
|
ADS_PRINT("Container Drop Content: " << dropArea);
|
|
|
|
if (dropArea != InvalidDockWidgetArea)
|
|
|
|
{
|
2019-11-27 19:00:04 +08:00
|
|
|
d->moveToContainer(Widget, dropArea);
|
2019-11-26 21:40:56 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there was a top level widget before the drop, then it is not top
|
|
|
|
// level widget anymore
|
2019-11-27 22:50:18 +08:00
|
|
|
CDockWidget::emitTopLevelEventForWidget(SingleDockWidget, false);
|
2019-11-26 21:40:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
//============================================================================
|
|
|
|
QList<CDockAreaWidget*> CDockContainerWidget::openedDockAreas() const
|
|
|
|
{
|
|
|
|
QList<CDockAreaWidget*> Result;
|
|
|
|
for (auto DockArea : d->DockAreas)
|
|
|
|
{
|
2018-10-31 06:45:59 +08:00
|
|
|
if (!DockArea->isHidden())
|
2017-03-23 17:23:53 +08:00
|
|
|
{
|
|
|
|
Result.append(DockArea);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 22:57:15 +08:00
|
|
|
//============================================================================
|
2017-12-30 01:18:16 +08:00
|
|
|
void CDockContainerWidget::saveState(QXmlStreamWriter& s) const
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("CDockContainerWidget::saveState isFloating "
|
|
|
|
<< isFloating());
|
2017-12-30 01:18:16 +08:00
|
|
|
|
2018-12-11 22:19:59 +08:00
|
|
|
s.writeStartElement("Container");
|
2017-12-30 01:18:16 +08:00
|
|
|
s.writeAttribute("Floating", QString::number(isFloating() ? 1 : 0));
|
2017-03-23 22:57:15 +08:00
|
|
|
if (isFloating())
|
|
|
|
{
|
2018-11-02 16:19:53 +08:00
|
|
|
CFloatingDockContainer* FloatingWidget = floatingWidget();
|
2017-12-30 01:18:16 +08:00
|
|
|
QByteArray Geometry = FloatingWidget->saveGeometry();
|
2019-01-14 15:59:37 +08:00
|
|
|
#if QT_VERSION < 0x050900
|
|
|
|
s.writeTextElement("Geometry", qByteArrayToHex(Geometry, ' '));
|
|
|
|
#else
|
2017-12-30 01:18:16 +08:00
|
|
|
s.writeTextElement("Geometry", Geometry.toHex(' '));
|
2019-01-14 15:59:37 +08:00
|
|
|
#endif
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
2017-12-30 01:18:16 +08:00
|
|
|
d->saveChildNodesState(s, d->RootSplitter);
|
|
|
|
s.writeEndElement();
|
2017-03-23 22:57:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
2019-11-29 22:56:57 +08:00
|
|
|
bool CDockContainerWidget::restoreState(CDockingStateReader& s, bool Testing)
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
2017-12-30 01:18:16 +08:00
|
|
|
bool IsFloating = s.attributes().value("Floating").toInt();
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Restore CDockContainerWidget Floating" << IsFloating);
|
2017-03-24 17:18:25 +08:00
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
QWidget*NewRootSplitter {};
|
2017-03-27 16:41:27 +08:00
|
|
|
if (!Testing)
|
|
|
|
{
|
2018-11-01 14:53:54 +08:00
|
|
|
d->VisibleDockAreaCount = -1;// invalidate the dock area count
|
2017-03-27 16:41:27 +08:00
|
|
|
d->DockAreas.clear();
|
2019-05-15 23:13:55 +08:00
|
|
|
std::fill(std::begin(d->LastAddedAreaCache),std::end(d->LastAddedAreaCache), nullptr);
|
2017-03-27 16:41:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (IsFloating)
|
2017-03-23 22:57:15 +08:00
|
|
|
{
|
2019-07-21 15:53:24 +08:00
|
|
|
ADS_PRINT("Restore floating widget");
|
2017-12-30 01:18:16 +08:00
|
|
|
if (!s.readNextStartElement() || s.name() != "Geometry")
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-11-29 22:56:57 +08:00
|
|
|
QByteArray GeometryString = s.readElementText(CDockingStateReader::ErrorOnUnexpectedElement).toLocal8Bit();
|
2017-12-30 01:18:16 +08:00
|
|
|
QByteArray Geometry = QByteArray::fromHex(GeometryString);
|
|
|
|
if (Geometry.isEmpty())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-27 16:41:27 +08:00
|
|
|
if (!Testing)
|
|
|
|
{
|
2018-11-02 16:19:53 +08:00
|
|
|
CFloatingDockContainer* FloatingWidget = floatingWidget();
|
2017-03-27 16:41:27 +08:00
|
|
|
FloatingWidget->restoreGeometry(Geometry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-30 01:18:16 +08:00
|
|
|
if (!d->restoreChildNodes(s, NewRootSplitter, Testing))
|
2017-03-27 16:41:27 +08:00
|
|
|
{
|
|
|
|
return false;
|
2017-03-24 23:17:55 +08:00
|
|
|
}
|
|
|
|
|
2017-03-27 16:41:27 +08:00
|
|
|
if (Testing)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2017-03-23 22:57:15 +08:00
|
|
|
|
2017-03-27 19:18:16 +08:00
|
|
|
// If the root splitter is empty, rostoreChildNodes returns a 0 pointer
|
|
|
|
// and we need to create a new empty root splitter
|
|
|
|
if (!NewRootSplitter)
|
|
|
|
{
|
2018-12-11 22:19:59 +08:00
|
|
|
NewRootSplitter = d->newSplitter(Qt::Horizontal);
|
2017-03-27 19:18:16 +08:00
|
|
|
}
|
|
|
|
|
2017-03-24 23:17:55 +08:00
|
|
|
d->Layout->replaceWidget(d->RootSplitter, NewRootSplitter);
|
|
|
|
QSplitter* OldRoot = d->RootSplitter;
|
|
|
|
d->RootSplitter = dynamic_cast<QSplitter*>(NewRootSplitter);
|
2017-03-27 16:41:27 +08:00
|
|
|
OldRoot->deleteLater();
|
2017-03-27 19:18:16 +08:00
|
|
|
|
2017-03-23 22:57:15 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-28 14:48:44 +08:00
|
|
|
//============================================================================
|
|
|
|
QSplitter* CDockContainerWidget::rootSplitter() const
|
|
|
|
{
|
|
|
|
return d->RootSplitter;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-12-11 22:19:59 +08:00
|
|
|
//============================================================================
|
|
|
|
void CDockContainerWidget::createRootSplitter()
|
|
|
|
{
|
|
|
|
if (d->RootSplitter)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
d->RootSplitter = d->newSplitter(Qt::Horizontal);
|
|
|
|
d->Layout->addWidget(d->RootSplitter);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-28 14:48:44 +08:00
|
|
|
//============================================================================
|
|
|
|
void CDockContainerWidget::dumpLayout()
|
|
|
|
{
|
2018-09-14 14:46:10 +08:00
|
|
|
#if (ADS_DEBUG_LEVEL > 0)
|
2017-09-06 21:45:22 +08:00
|
|
|
qDebug("\n\nDumping layout --------------------------");
|
2018-12-03 19:52:57 +08:00
|
|
|
std::cout << "\n\nDumping layout --------------------------" << std::endl;
|
2017-03-28 14:48:44 +08:00
|
|
|
d->dumpRecursive(0, d->RootSplitter);
|
2017-09-06 21:45:22 +08:00
|
|
|
qDebug("--------------------------\n\n");
|
2018-12-03 19:52:57 +08:00
|
|
|
std::cout << "--------------------------\n\n" << std::endl;
|
2018-09-14 14:46:10 +08:00
|
|
|
#endif
|
2017-03-28 14:48:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-07 17:10:14 +08:00
|
|
|
//============================================================================
|
|
|
|
CDockAreaWidget* CDockContainerWidget::lastAddedDockAreaWidget(DockWidgetArea area) const
|
|
|
|
{
|
|
|
|
return d->LastAddedAreaCache[areaIdToIndex(area)];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-14 19:21:29 +08:00
|
|
|
//============================================================================
|
2018-09-26 15:57:36 +08:00
|
|
|
bool CDockContainerWidget::hasTopLevelDockWidget() const
|
2018-09-14 19:21:29 +08:00
|
|
|
{
|
2018-11-02 16:19:53 +08:00
|
|
|
if (!isFloating())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-09-14 19:21:29 +08:00
|
|
|
auto DockAreas = openedDockAreas();
|
|
|
|
if (DockAreas.count() != 1)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DockAreas[0]->openDockWidgetsCount() == 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
2018-09-26 15:57:36 +08:00
|
|
|
CDockWidget* CDockContainerWidget::topLevelDockWidget() const
|
2018-09-14 19:21:29 +08:00
|
|
|
{
|
2018-11-02 16:19:53 +08:00
|
|
|
auto TopLevelDockArea = topLevelDockArea();
|
|
|
|
if (!TopLevelDockArea)
|
2018-09-14 19:21:29 +08:00
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-11-02 16:19:53 +08:00
|
|
|
auto DockWidgets = TopLevelDockArea->openedDockWidgets();
|
2018-09-14 19:21:29 +08:00
|
|
|
if (DockWidgets.count() != 1)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DockWidgets[0];
|
2018-11-02 16:19:53 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CDockAreaWidget* CDockContainerWidget::topLevelDockArea() const
|
|
|
|
{
|
|
|
|
if (!isFloating())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto DockAreas = openedDockAreas();
|
|
|
|
if (DockAreas.count() != 1)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DockAreas[0];
|
2018-09-14 19:21:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-11 14:54:32 +08:00
|
|
|
//============================================================================
|
|
|
|
QList<CDockWidget*> CDockContainerWidget::dockWidgets() const
|
|
|
|
{
|
|
|
|
QList<CDockWidget*> Result;
|
|
|
|
for (const auto DockArea : d->DockAreas)
|
|
|
|
{
|
|
|
|
Result.append(DockArea->dockWidgets());
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-12 21:18:05 +08:00
|
|
|
//============================================================================
|
|
|
|
CDockWidget::DockWidgetFeatures CDockContainerWidget::features() const
|
|
|
|
{
|
2018-10-15 14:29:30 +08:00
|
|
|
CDockWidget::DockWidgetFeatures Features(CDockWidget::AllDockWidgetFeatures);
|
2018-10-12 21:18:05 +08:00
|
|
|
for (const auto DockArea : d->DockAreas)
|
|
|
|
{
|
|
|
|
Features &= DockArea->features();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Features;
|
|
|
|
}
|
|
|
|
|
2018-11-02 16:19:53 +08:00
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
CFloatingDockContainer* CDockContainerWidget::floatingWidget() const
|
|
|
|
{
|
|
|
|
return internal::findParent<CFloatingDockContainer*>(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-12-20 22:29:38 +08:00
|
|
|
//============================================================================
|
|
|
|
void CDockContainerWidget::closeOtherAreas(CDockAreaWidget* KeepOpenArea)
|
|
|
|
{
|
|
|
|
for (const auto DockArea : d->DockAreas)
|
|
|
|
{
|
2020-01-06 18:42:36 +08:00
|
|
|
if (DockArea == KeepOpenArea)
|
2018-12-20 22:29:38 +08:00
|
|
|
{
|
2020-01-06 18:42:36 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!DockArea->features(BitwiseAnd).testFlag(CDockWidget::DockWidgetClosable))
|
|
|
|
{
|
|
|
|
continue;
|
2018-12-20 22:29:38 +08:00
|
|
|
}
|
2020-01-06 18:42:36 +08:00
|
|
|
|
|
|
|
// We do not close areas with widgets with custom close handling
|
|
|
|
if (DockArea->features(BitwiseOr).testFlag(CDockWidget::CustomCloseHandling))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
DockArea->closeArea();
|
2018-12-20 22:29:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-23 17:23:53 +08:00
|
|
|
} // namespace ads
|
|
|
|
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
// EOF DockContainerWidget.cpp
|