mirror of
https://github.com/githubuser0xFFFF/Qt-Advanced-Docking-System.git
synced 2025-01-12 16:20:25 +08:00
Improved handling of sizes when dropping or removing content
This commit is contained in:
parent
48382ccd82
commit
32e5d599f7
@ -139,6 +139,14 @@ public:
|
|||||||
void dropIntoSection(CFloatingDockContainer* FloatingWidget,
|
void dropIntoSection(CFloatingDockContainer* FloatingWidget,
|
||||||
CDockAreaWidget* TargetArea, DockWidgetArea area);
|
CDockAreaWidget* TargetArea, DockWidgetArea area);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new tab for a widget dropped into the center of a section
|
||||||
|
*/
|
||||||
|
void dropIntoCenterOfSection(CFloatingDockContainer* FloatingWidget,
|
||||||
|
CDockAreaWidget* TargetArea);
|
||||||
|
|
||||||
|
void dropIntoSectionWithSameOrientation();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds new dock areas to the internal dock area list
|
* Adds new dock areas to the internal dock area list
|
||||||
*/
|
*/
|
||||||
@ -343,45 +351,55 @@ void DockContainerWidgetPrivate::dropIntoContainer(CFloatingDockContainer* Float
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//============================================================================
|
||||||
|
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);
|
||||||
|
FloatingWidget->deleteLater();
|
||||||
|
TargetArea->updateTitleBarVisibility();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//============================================================================
|
//============================================================================
|
||||||
void DockContainerWidgetPrivate::dropIntoSection(CFloatingDockContainer* FloatingWidget,
|
void DockContainerWidgetPrivate::dropIntoSection(CFloatingDockContainer* FloatingWidget,
|
||||||
CDockAreaWidget* TargetArea, DockWidgetArea area)
|
CDockAreaWidget* TargetArea, DockWidgetArea area)
|
||||||
{
|
{
|
||||||
CDockContainerWidget* FloatingContainer = FloatingWidget->dockContainer();
|
|
||||||
// Dropping into center means all dock widgets in the dropped floating
|
// Dropping into center means all dock widgets in the dropped floating
|
||||||
// widget will become tabs of the drop area
|
// widget will become tabs of the drop area
|
||||||
if (CenterDockWidgetArea == area)
|
if (CenterDockWidgetArea == area)
|
||||||
{
|
{
|
||||||
auto NewDockWidgets = FloatingContainer->dockWidgets();
|
dropIntoCenterOfSection(FloatingWidget, TargetArea);
|
||||||
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);
|
|
||||||
FloatingWidget->deleteLater();
|
|
||||||
TargetArea->updateTitleBarVisibility();
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CDockContainerWidget* FloatingContainer = FloatingWidget->dockContainer();
|
||||||
auto InsertParam = internal::dockAreaInsertParameters(area);
|
auto InsertParam = internal::dockAreaInsertParameters(area);
|
||||||
auto NewDockAreas = FloatingWidget->dockContainer()->findChildren<CDockAreaWidget*>(
|
auto NewDockAreas = FloatingWidget->dockContainer()->findChildren<CDockAreaWidget*>(
|
||||||
QString(), Qt::FindChildrenRecursively);
|
QString(), Qt::FindChildrenRecursively);
|
||||||
@ -396,40 +414,65 @@ void DockContainerWidgetPrivate::dropIntoSection(CFloatingDockContainer* Floatin
|
|||||||
}
|
}
|
||||||
int AreaIndex = TargetAreaSplitter->indexOf(TargetArea);
|
int AreaIndex = TargetAreaSplitter->indexOf(TargetArea);
|
||||||
auto Widget = FloatingWidget->dockContainer()->findChild<QWidget*>(QString(), Qt::FindDirectChildrenOnly);
|
auto Widget = FloatingWidget->dockContainer()->findChild<QWidget*>(QString(), Qt::FindDirectChildrenOnly);
|
||||||
auto FloatingSplitter = dynamic_cast<QSplitter*>(Widget);
|
auto FloatingSplitter = qobject_cast<QSplitter*>(Widget);
|
||||||
|
|
||||||
if (TargetAreaSplitter->orientation() == InsertParam.orientation())
|
if (TargetAreaSplitter->orientation() == InsertParam.orientation())
|
||||||
{
|
{
|
||||||
|
auto Sizes = TargetAreaSplitter->sizes();
|
||||||
|
int TargetAreaSize = (InsertParam.orientation() == Qt::Horizontal) ? TargetArea->width() : TargetArea->height();
|
||||||
|
bool AdjustSplitterSizes = true;
|
||||||
if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1)
|
if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1)
|
||||||
{
|
{
|
||||||
TargetAreaSplitter->insertWidget(AreaIndex + InsertParam.insertOffset(), Widget);
|
TargetAreaSplitter->insertWidget(AreaIndex + InsertParam.insertOffset(), Widget);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
AdjustSplitterSizes = (FloatingSplitter->count() == 1);
|
||||||
int InsertIndex = AreaIndex + InsertParam.insertOffset();
|
int InsertIndex = AreaIndex + InsertParam.insertOffset();
|
||||||
while (FloatingSplitter->count())
|
while (FloatingSplitter->count())
|
||||||
{
|
{
|
||||||
TargetAreaSplitter->insertWidget(InsertIndex++, FloatingSplitter->widget(0));
|
TargetAreaSplitter->insertWidget(InsertIndex++, FloatingSplitter->widget(0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (AdjustSplitterSizes)
|
||||||
|
{
|
||||||
|
int Size = (TargetAreaSize - TargetAreaSplitter->handleWidth()) / 2;
|
||||||
|
Sizes[AreaIndex] = Size;
|
||||||
|
Sizes.insert(AreaIndex, Size);
|
||||||
|
TargetAreaSplitter->setSizes(Sizes);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
QList<int> NewSplitterSizes;
|
||||||
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
QSplitter* NewSplitter = newSplitter(InsertParam.orientation());
|
||||||
|
int TargetAreaSize = (InsertParam.orientation() == Qt::Horizontal) ? TargetArea->width() : TargetArea->height();
|
||||||
|
bool AdjustSplitterSizes = true;
|
||||||
if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1)
|
if ((FloatingSplitter->orientation() != InsertParam.orientation()) && FloatingSplitter->count() > 1)
|
||||||
{
|
{
|
||||||
NewSplitter->addWidget(Widget);
|
NewSplitter->addWidget(Widget);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
AdjustSplitterSizes = (FloatingSplitter->count() == 1);
|
||||||
while (FloatingSplitter->count())
|
while (FloatingSplitter->count())
|
||||||
{
|
{
|
||||||
NewSplitter->addWidget(FloatingSplitter->widget(0));
|
NewSplitter->addWidget(FloatingSplitter->widget(0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TargetAreaSplitter->insertWidget(AreaIndex, NewSplitter);
|
// Save the sizes before insertion and restore it later to prevent
|
||||||
|
// shrinking of existing area
|
||||||
|
auto Sizes = TargetAreaSplitter->sizes();
|
||||||
insertWidgetIntoSplitter(NewSplitter, TargetArea, !InsertParam.append());
|
insertWidgetIntoSplitter(NewSplitter, TargetArea, !InsertParam.append());
|
||||||
|
if (AdjustSplitterSizes)
|
||||||
|
{
|
||||||
|
int Size = TargetAreaSize / 2;
|
||||||
|
NewSplitter->setSizes({Size, Size});
|
||||||
|
}
|
||||||
|
TargetAreaSplitter->insertWidget(AreaIndex, NewSplitter);
|
||||||
|
TargetAreaSplitter->setSizes(Sizes);
|
||||||
}
|
}
|
||||||
|
|
||||||
FloatingWidget->deleteLater();
|
FloatingWidget->deleteLater();
|
||||||
@ -1026,10 +1069,12 @@ void CDockContainerWidget::removeDockArea(CDockAreaWidget* area)
|
|||||||
else if (Splitter->count() == 1)
|
else if (Splitter->count() == 1)
|
||||||
{
|
{
|
||||||
qDebug() << "Replacing splitter with content";
|
qDebug() << "Replacing splitter with content";
|
||||||
|
QSplitter* ParentSplitter = internal::findParent<QSplitter*>(Splitter);
|
||||||
|
auto Sizes = ParentSplitter->sizes();
|
||||||
QWidget* widget = Splitter->widget(0);
|
QWidget* widget = Splitter->widget(0);
|
||||||
widget->setParent(this);
|
widget->setParent(this);
|
||||||
QSplitter* ParentSplitter = internal::findParent<QSplitter*>(Splitter);
|
|
||||||
internal::replaceSplitterWidget(ParentSplitter, Splitter, widget);
|
internal::replaceSplitterWidget(ParentSplitter, Splitter, widget);
|
||||||
|
ParentSplitter->setSizes(Sizes);
|
||||||
}
|
}
|
||||||
|
|
||||||
delete Splitter;
|
delete Splitter;
|
||||||
|
@ -346,7 +346,6 @@ void CDockWidgetTab::mouseMoveEvent(QMouseEvent* ev)
|
|||||||
void CDockWidgetTab::contextMenuEvent(QContextMenuEvent* ev)
|
void CDockWidgetTab::contextMenuEvent(QContextMenuEvent* ev)
|
||||||
{
|
{
|
||||||
ev->accept();
|
ev->accept();
|
||||||
std::cout << "CDockAreaTabBar::onTabContextMenuRequested" << std::endl;
|
|
||||||
|
|
||||||
d->DragStartMousePosition = ev->pos();
|
d->DragStartMousePosition = ev->pos();
|
||||||
QMenu Menu(this);
|
QMenu Menu(this);
|
||||||
|
Loading…
Reference in New Issue
Block a user