mirror of
https://github.com/githubuser0xFFFF/Qt-Advanced-Docking-System.git
synced 2025-04-01 02:42:39 +08:00
Added support for make auto hide widget floating via double click or context menu
This commit is contained in:
parent
0b3c3f0123
commit
1c6d86e70f
@ -230,7 +230,7 @@ CAutoHideDockContainer::CAutoHideDockContainer(CDockWidget* DockWidget, SideBarL
|
|||||||
//============================================================================
|
//============================================================================
|
||||||
void CAutoHideDockContainer::updateSize()
|
void CAutoHideDockContainer::updateSize()
|
||||||
{
|
{
|
||||||
std::cout << "CAutoHideDockContainer::updateSize()" << std::endl;
|
qDebug() << "CAutoHideDockContainer::updateSize()";
|
||||||
auto dockContainerParent = dockContainer();
|
auto dockContainerParent = dockContainer();
|
||||||
if (!dockContainerParent)
|
if (!dockContainerParent)
|
||||||
{
|
{
|
||||||
@ -239,6 +239,10 @@ void CAutoHideDockContainer::updateSize()
|
|||||||
|
|
||||||
auto rect = dockContainerParent->contentRect();
|
auto rect = dockContainerParent->contentRect();
|
||||||
qDebug() << "dockContainerParent->contentRect() " << rect;
|
qDebug() << "dockContainerParent->contentRect() " << rect;
|
||||||
|
qDebug() << "dockWidget()->rect()" << dockWidget()->rect();
|
||||||
|
qDebug() << "dockAreaWidget()->rect(): " << dockAreaWidget()->rect();
|
||||||
|
qDebug() << "CAutoHideDockContainer::isVisible " << this->isVisible();
|
||||||
|
qDebug() << "CAutoHideDockContainer::rect " << this->rect();
|
||||||
|
|
||||||
switch (sideBarLocation())
|
switch (sideBarLocation())
|
||||||
{
|
{
|
||||||
@ -273,6 +277,9 @@ void CAutoHideDockContainer::updateSize()
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
qDebug() << "CAutoHideDockContainer::rect (after): " << this->rect();
|
||||||
|
qDebug() << "dockAreaWidget()->rect(): " << dockAreaWidget()->rect();
|
||||||
}
|
}
|
||||||
|
|
||||||
//============================================================================
|
//============================================================================
|
||||||
@ -326,6 +333,7 @@ CDockWidget* CAutoHideDockContainer::dockWidget() const
|
|||||||
//============================================================================
|
//============================================================================
|
||||||
void CAutoHideDockContainer::addDockWidget(CDockWidget* DockWidget)
|
void CAutoHideDockContainer::addDockWidget(CDockWidget* DockWidget)
|
||||||
{
|
{
|
||||||
|
std::cout << "CAutoHideDockContainer::addDockWidget " << std::endl;
|
||||||
if (d->DockWidget)
|
if (d->DockWidget)
|
||||||
{
|
{
|
||||||
// Remove the old dock widget at this area
|
// Remove the old dock widget at this area
|
||||||
@ -346,6 +354,10 @@ void CAutoHideDockContainer::addDockWidget(CDockWidget* DockWidget)
|
|||||||
}
|
}
|
||||||
d->DockArea->addDockWidget(DockWidget);
|
d->DockArea->addDockWidget(DockWidget);
|
||||||
updateSize();
|
updateSize();
|
||||||
|
// The dock area is not visible and will not update the size when updateSize()
|
||||||
|
// is called for this auto hide container. Therefore we explicitely resize
|
||||||
|
// it here. As soon as it will become visible, it will get the right size
|
||||||
|
d->DockArea->resize(size());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -32,6 +32,7 @@
|
|||||||
#include <QBoxLayout>
|
#include <QBoxLayout>
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QElapsedTimer>
|
#include <QElapsedTimer>
|
||||||
|
#include <QMenu>
|
||||||
|
|
||||||
#include "AutoHideDockContainer.h"
|
#include "AutoHideDockContainer.h"
|
||||||
#include "AutoHideSideBar.h"
|
#include "AutoHideSideBar.h"
|
||||||
@ -272,10 +273,65 @@ bool CAutoHideTab::event(QEvent* event)
|
|||||||
return Super::event(event);
|
return Super::event(event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//============================================================================
|
//============================================================================
|
||||||
bool CAutoHideTab::iconOnly() const
|
bool CAutoHideTab::iconOnly() const
|
||||||
{
|
{
|
||||||
return CDockManager::testAutoHideConfigFlag(CDockManager::AutoHideSideBarsIconOnly) && !icon().isNull();
|
return CDockManager::testAutoHideConfigFlag(CDockManager::AutoHideSideBarsIconOnly) && !icon().isNull();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//============================================================================
|
||||||
|
void CAutoHideTab::contextMenuEvent(QContextMenuEvent* ev)
|
||||||
|
{
|
||||||
|
ev->accept();
|
||||||
|
//d->saveDragStartMousePosition(ev->globalPos());
|
||||||
|
|
||||||
|
const bool isFloatable = d->DockWidget->features().testFlag(CDockWidget::DockWidgetFloatable);
|
||||||
|
QAction* Action;
|
||||||
|
QMenu Menu(this);
|
||||||
|
|
||||||
|
|
||||||
|
Action = Menu.addAction(tr("Detach"), this, SLOT(setDockWidgetFloating()));
|
||||||
|
Action->setEnabled(isFloatable);
|
||||||
|
auto IsPinnable = d->DockWidget->features().testFlag(CDockWidget::DockWidgetPinnable);
|
||||||
|
Action->setEnabled(IsPinnable);
|
||||||
|
|
||||||
|
auto menu = Menu.addMenu(tr("Pin To..."));
|
||||||
|
menu->setEnabled(IsPinnable);
|
||||||
|
//d->createAutoHideToAction(tr("Top"), SideBarTop, menu);
|
||||||
|
//d->createAutoHideToAction(tr("Left"), SideBarLeft, menu);
|
||||||
|
//d->createAutoHideToAction(tr("Right"), SideBarRight, menu);
|
||||||
|
//d->createAutoHideToAction(tr("Bottom"), SideBarBottom, menu);
|
||||||
|
|
||||||
|
/*Menu.addSeparator();
|
||||||
|
Action = Menu.addAction(tr("Close"), this, SIGNAL(closeRequested()));
|
||||||
|
Action->setEnabled(isClosable());
|
||||||
|
if (d->DockArea->openDockWidgetsCount() > 1)
|
||||||
|
{
|
||||||
|
Action = Menu.addAction(tr("Close Others"), this, SIGNAL(closeOtherTabsRequested()));
|
||||||
|
}*/
|
||||||
|
Menu.exec(ev->globalPos());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//============================================================================
|
||||||
|
void CAutoHideTab::mouseDoubleClickEvent(QMouseEvent *event)
|
||||||
|
{
|
||||||
|
if (event->button() == Qt::LeftButton)
|
||||||
|
{
|
||||||
|
setDockWidgetFloating();
|
||||||
|
}
|
||||||
|
|
||||||
|
Super::mouseDoubleClickEvent(event);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//============================================================================
|
||||||
|
void CAutoHideTab::setDockWidgetFloating()
|
||||||
|
{
|
||||||
|
dockWidget()->setFloating();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -65,11 +65,12 @@ private:
|
|||||||
friend class CDockContainerWidget;
|
friend class CDockContainerWidget;
|
||||||
friend DockContainerWidgetPrivate;
|
friend DockContainerWidgetPrivate;
|
||||||
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void setSideBar(CAutoHideSideBar *SideTabBar);
|
void setSideBar(CAutoHideSideBar *SideTabBar);
|
||||||
void removeFromSideBar();
|
void removeFromSideBar();
|
||||||
virtual bool event(QEvent* event) override;
|
virtual bool event(QEvent* event) override;
|
||||||
|
virtual void contextMenuEvent(QContextMenuEvent* ev) override;
|
||||||
|
virtual void mouseDoubleClickEvent(QMouseEvent *event) override;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
using Super = CPushButton;
|
using Super = CPushButton;
|
||||||
@ -133,6 +134,12 @@ public:
|
|||||||
* not in a side bar
|
* not in a side bar
|
||||||
*/
|
*/
|
||||||
CAutoHideSideBar* sideBar() const;
|
CAutoHideSideBar* sideBar() const;
|
||||||
|
|
||||||
|
public Q_SLOTS:
|
||||||
|
/**
|
||||||
|
* Set the dock widget floating, if it is floatable
|
||||||
|
*/
|
||||||
|
void setDockWidgetFloating();
|
||||||
}; // class AutoHideTab
|
}; // class AutoHideTab
|
||||||
}
|
}
|
||||||
// namespace ads
|
// namespace ads
|
||||||
|
@ -260,6 +260,7 @@ void DockAreaTitleBarPrivate::createTabBar()
|
|||||||
//============================================================================
|
//============================================================================
|
||||||
IFloatingWidget* DockAreaTitleBarPrivate::makeAreaFloating(const QPoint& Offset, eDragState DragState)
|
IFloatingWidget* DockAreaTitleBarPrivate::makeAreaFloating(const QPoint& Offset, eDragState DragState)
|
||||||
{
|
{
|
||||||
|
qDebug() << "DockAreaTitleBarPrivate::makeAreaFloating " << DockArea->size();
|
||||||
QSize Size = DockArea->size();
|
QSize Size = DockArea->size();
|
||||||
this->DragState = DragState;
|
this->DragState = DragState;
|
||||||
bool CreateFloatingDockContainer = (DraggingFloatingWidget != DragState);
|
bool CreateFloatingDockContainer = (DraggingFloatingWidget != DragState);
|
||||||
@ -660,6 +661,7 @@ void CDockAreaTitleBar::mouseMoveEvent(QMouseEvent* ev)
|
|||||||
//============================================================================
|
//============================================================================
|
||||||
void CDockAreaTitleBar::mouseDoubleClickEvent(QMouseEvent *event)
|
void CDockAreaTitleBar::mouseDoubleClickEvent(QMouseEvent *event)
|
||||||
{
|
{
|
||||||
|
std::cout << "CDockAreaTitleBar::mouseDoubleClickEvent" << std::endl;
|
||||||
// If this is the last dock area in a dock container it does not make
|
// If this is the last dock area in a dock container it does not make
|
||||||
// sense to move it to a new floating widget and leave this one
|
// sense to move it to a new floating widget and leave this one
|
||||||
// empty
|
// empty
|
||||||
@ -677,6 +679,26 @@ void CDockAreaTitleBar::mouseDoubleClickEvent(QMouseEvent *event)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//============================================================================
|
||||||
|
void CDockAreaTitleBar::setAreaFloating()
|
||||||
|
{
|
||||||
|
// If this is the last dock area in a dock container it does not make
|
||||||
|
// sense to move it to a new floating widget and leave this one
|
||||||
|
// empty
|
||||||
|
if (d->DockArea->dockContainer()->isFloating() && d->DockArea->dockContainer()->dockAreaCount() == 1)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!d->DockArea->features().testFlag(CDockWidget::DockWidgetFloatable))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
d->makeAreaFloating(mapFromGlobal(QCursor::pos()), DraggingInactive);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//============================================================================
|
//============================================================================
|
||||||
void CDockAreaTitleBar::contextMenuEvent(QContextMenuEvent* ev)
|
void CDockAreaTitleBar::contextMenuEvent(QContextMenuEvent* ev)
|
||||||
{
|
{
|
||||||
|
@ -163,6 +163,12 @@ public:
|
|||||||
*/
|
*/
|
||||||
QString titleBarButtonToolTip(TitleBarButton Button) const;
|
QString titleBarButtonToolTip(TitleBarButton Button) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Moves the dock area into its own floating widget if the area
|
||||||
|
* DockWidgetFloatable flag is true
|
||||||
|
*/
|
||||||
|
void setAreaFloating();
|
||||||
|
|
||||||
Q_SIGNALS:
|
Q_SIGNALS:
|
||||||
/**
|
/**
|
||||||
* This signal is emitted if a tab in the tab bar is clicked by the user
|
* This signal is emitted if a tab in the tab bar is clicked by the user
|
||||||
|
@ -1442,6 +1442,13 @@ bool CDockAreaWidget::isTopLevelArea() const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//============================================================================
|
||||||
|
void CDockAreaWidget::setFloating()
|
||||||
|
{
|
||||||
|
d->TitleBar->setAreaFloating();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
//============================================================================
|
//============================================================================
|
||||||
bool CDockAreaWidget::event(QEvent *e)
|
bool CDockAreaWidget::event(QEvent *e)
|
||||||
|
@ -414,6 +414,12 @@ public Q_SLOTS:
|
|||||||
*/
|
*/
|
||||||
void closeOtherAreas();
|
void closeOtherAreas();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Moves the dock area into its own floating widget if the area
|
||||||
|
* DockWidgetFloatable flag is true
|
||||||
|
*/
|
||||||
|
void setFloating();
|
||||||
|
|
||||||
Q_SIGNALS:
|
Q_SIGNALS:
|
||||||
/**
|
/**
|
||||||
* This signal is emitted when user clicks on a tab at an index.
|
* This signal is emitted when user clicks on a tab at an index.
|
||||||
|
@ -475,7 +475,6 @@ DockWidgetArea CDockOverlay::dropAreaUnderCursor() const
|
|||||||
auto DockArea = qobject_cast<CDockAreaWidget*>(d->TargetWidget.data());
|
auto DockArea = qobject_cast<CDockAreaWidget*>(d->TargetWidget.data());
|
||||||
if (!DockArea && CDockManager::autoHideConfigFlags().testFlag(CDockManager::AutoHideFeatureEnabled))
|
if (!DockArea && CDockManager::autoHideConfigFlags().testFlag(CDockManager::AutoHideFeatureEnabled))
|
||||||
{
|
{
|
||||||
std::cout << d->Mode << " Find out side bar area " << std::endl;
|
|
||||||
auto Rect = rect();
|
auto Rect = rect();
|
||||||
const QPoint pos = mapFromGlobal(QCursor::pos());
|
const QPoint pos = mapFromGlobal(QCursor::pos());
|
||||||
if (pos.x() < d->sideBarMouseZone(SideBarLeft))
|
if (pos.x() < d->sideBarMouseZone(SideBarLeft))
|
||||||
@ -488,7 +487,6 @@ DockWidgetArea CDockOverlay::dropAreaUnderCursor() const
|
|||||||
}
|
}
|
||||||
else if (pos.y() < d->sideBarMouseZone(SideBarTop))
|
else if (pos.y() < d->sideBarMouseZone(SideBarTop))
|
||||||
{
|
{
|
||||||
std::cout << d->Mode << " TopAutoHideArea " << std::endl;
|
|
||||||
return TopAutoHideArea;
|
return TopAutoHideArea;
|
||||||
}
|
}
|
||||||
else if (pos.y() > (Rect.height() - d->sideBarMouseZone(SideBarBottom)))
|
else if (pos.y() > (Rect.height() - d->sideBarMouseZone(SideBarBottom)))
|
||||||
@ -534,14 +532,12 @@ DockWidgetArea CDockOverlay::visibleDropAreaUnderCursor() const
|
|||||||
//============================================================================
|
//============================================================================
|
||||||
DockWidgetArea CDockOverlay::showOverlay(QWidget* target)
|
DockWidgetArea CDockOverlay::showOverlay(QWidget* target)
|
||||||
{
|
{
|
||||||
std::cout << d->Mode << " CDockOverlay::showOverlay()" << target << " " << target->objectName().toStdString() << std::endl;
|
|
||||||
if (d->TargetWidget == target)
|
if (d->TargetWidget == target)
|
||||||
{
|
{
|
||||||
// Hint: We could update geometry of overlay here.
|
// Hint: We could update geometry of overlay here.
|
||||||
DockWidgetArea da = dropAreaUnderCursor();
|
DockWidgetArea da = dropAreaUnderCursor();
|
||||||
if (da != d->LastLocation)
|
if (da != d->LastLocation)
|
||||||
{
|
{
|
||||||
std::cout << d->Mode << " repaint()" << std::endl;
|
|
||||||
repaint();
|
repaint();
|
||||||
d->LastLocation = da;
|
d->LastLocation = da;
|
||||||
}
|
}
|
||||||
@ -566,7 +562,6 @@ DockWidgetArea CDockOverlay::showOverlay(QWidget* target)
|
|||||||
//============================================================================
|
//============================================================================
|
||||||
void CDockOverlay::hideOverlay()
|
void CDockOverlay::hideOverlay()
|
||||||
{
|
{
|
||||||
std::cout << d->Mode << " CDockOverlay::hideOverlay()" << std::endl;
|
|
||||||
hide();
|
hide();
|
||||||
d->TargetWidget.clear();
|
d->TargetWidget.clear();
|
||||||
d->LastLocation = InvalidDockWidgetArea;
|
d->LastLocation = InvalidDockWidgetArea;
|
||||||
@ -578,7 +573,6 @@ void CDockOverlay::hideOverlay()
|
|||||||
void CDockOverlay::enableDropPreview(bool Enable)
|
void CDockOverlay::enableDropPreview(bool Enable)
|
||||||
{
|
{
|
||||||
d->DropPreviewEnabled = Enable;
|
d->DropPreviewEnabled = Enable;
|
||||||
std::cout << d->Mode << " update() " << Enable << std::endl;
|
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -606,7 +600,6 @@ void CDockOverlay::paintEvent(QPaintEvent* event)
|
|||||||
double Factor = (CDockOverlay::ModeContainerOverlay == d->Mode) ?
|
double Factor = (CDockOverlay::ModeContainerOverlay == d->Mode) ?
|
||||||
3 : 2;
|
3 : 2;
|
||||||
|
|
||||||
std::cout << "CDockOverlay::paintEvent da: " << da << std::endl;
|
|
||||||
switch (da)
|
switch (da)
|
||||||
{
|
{
|
||||||
case TopDockWidgetArea: r.setHeight(r.height() / Factor); break;
|
case TopDockWidgetArea: r.setHeight(r.height() / Factor); break;
|
||||||
|
@ -1021,7 +1021,15 @@ void CDockWidget::setFloating()
|
|||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
d->TabWidget->detachDockWidget();
|
|
||||||
|
if (this->isAutoHide())
|
||||||
|
{
|
||||||
|
dockAreaWidget()->setFloating();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
d->TabWidget->detachDockWidget();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -499,10 +499,8 @@ void FloatingDockContainerPrivate::titleMouseReleaseEvent()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (DockManager->dockAreaOverlay()->dropAreaUnderCursor()
|
if (DockManager->dockAreaOverlay()->dropAreaUnderCursor() != InvalidDockWidgetArea
|
||||||
!= InvalidDockWidgetArea
|
|| DockManager->containerOverlay()->dropAreaUnderCursor() != InvalidDockWidgetArea)
|
||||||
|| DockManager->containerOverlay()->dropAreaUnderCursor()
|
|
||||||
!= InvalidDockWidgetArea)
|
|
||||||
{
|
{
|
||||||
CDockOverlay *Overlay = DockManager->containerOverlay();
|
CDockOverlay *Overlay = DockManager->containerOverlay();
|
||||||
if (!Overlay->dropOverlayRect().isValid())
|
if (!Overlay->dropOverlayRect().isValid())
|
||||||
@ -510,21 +508,26 @@ void FloatingDockContainerPrivate::titleMouseReleaseEvent()
|
|||||||
Overlay = DockManager->dockAreaOverlay();
|
Overlay = DockManager->dockAreaOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Resize the floating widget to the size of the highlighted drop area
|
// Do not resize if we drop into an autohide sidebar area to preserve
|
||||||
// rectangle
|
// the dock area size for the initial size of the auto hide area
|
||||||
QRect Rect = Overlay->dropOverlayRect();
|
if (!ads::internal::isSideBarArea(Overlay->dropAreaUnderCursor()))
|
||||||
int FrameWidth = (_this->frameSize().width() - _this->rect().width())
|
|
||||||
/ 2;
|
|
||||||
int TitleBarHeight = _this->frameSize().height()
|
|
||||||
- _this->rect().height() - FrameWidth;
|
|
||||||
if (Rect.isValid())
|
|
||||||
{
|
{
|
||||||
QPoint TopLeft = Overlay->mapToGlobal(Rect.topLeft());
|
// Resize the floating widget to the size of the highlighted drop area
|
||||||
TopLeft.ry() += TitleBarHeight;
|
// rectangle
|
||||||
_this->setGeometry(
|
QRect Rect = Overlay->dropOverlayRect();
|
||||||
QRect(TopLeft,
|
int FrameWidth = (_this->frameSize().width() - _this->rect().width())
|
||||||
QSize(Rect.width(), Rect.height() - TitleBarHeight)));
|
/ 2;
|
||||||
QApplication::processEvents();
|
int TitleBarHeight = _this->frameSize().height()
|
||||||
|
- _this->rect().height() - FrameWidth;
|
||||||
|
if (Rect.isValid())
|
||||||
|
{
|
||||||
|
QPoint TopLeft = Overlay->mapToGlobal(Rect.topLeft());
|
||||||
|
TopLeft.ry() += TitleBarHeight;
|
||||||
|
_this->setGeometry(
|
||||||
|
QRect(TopLeft,
|
||||||
|
QSize(Rect.width(), Rect.height() - TitleBarHeight)));
|
||||||
|
QApplication::processEvents();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
DropContainer->dropFloatingWidget(_this, QCursor::pos());
|
DropContainer->dropFloatingWidget(_this, QCursor::pos());
|
||||||
}
|
}
|
||||||
@ -533,6 +536,7 @@ void FloatingDockContainerPrivate::titleMouseReleaseEvent()
|
|||||||
DockManager->dockAreaOverlay()->hideOverlay();
|
DockManager->dockAreaOverlay()->hideOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//============================================================================
|
//============================================================================
|
||||||
void FloatingDockContainerPrivate::updateDropOverlays(const QPoint &GlobalPos)
|
void FloatingDockContainerPrivate::updateDropOverlays(const QPoint &GlobalPos)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user