Files
KDDockWidgets/src/qtwidgets/views/Stack_qtwidgets.cpp
2022-05-19 08:00:20 +01:00

251 lines
7.1 KiB
C++

/*
This file is part of KDDockWidgets.
SPDX-FileCopyrightText: 2019-2022 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.com>
Author: Sérgio Martins <sergio.martins@kdab.com>
SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only
Contact KDAB at <info@kdab.com> for commercial licensing options.
*/
#include "Stack_qtwidgets.h"
#include "Controller.h"
#include "controllers/Stack.h"
#include "controllers/TitleBar.h"
#include "qtwidgets/FrameworkWidgetFactory_qtwidgets.h"
#include "qtwidgets/views/DockWidget_qtwidgets.h"
#include "private/DockRegistry_p.h"
#include <QMouseEvent>
#include <QTabBar>
#include <QHBoxLayout>
#include <QAbstractButton>
#include <QMenu>
using namespace KDDockWidgets;
using namespace KDDockWidgets::Views;
Stack_qtwidgets::Stack_qtwidgets(Controllers::Stack *controller, Controllers::Frame *parent)
: View_qtwidgets<QTabWidget>(controller, Type::Stack, View_qtwidgets::asQWidget(parent))
, m_stack(controller)
{
}
void Stack_qtwidgets::init()
{
setTabBar(tabBar());
setTabsClosable(Config::self().flags() & Config::Flag_TabsHaveCloseButton);
setContextMenuPolicy(Qt::CustomContextMenu);
connect(this, &QTabWidget::customContextMenuRequested, this, &Stack_qtwidgets::showContextMenu);
// In case tabs closable is set by the factory, a tabClosedRequested() is emitted when the user presses [x]
connect(this, &QTabWidget::tabCloseRequested, this, [this](int index) {
if (auto dw = dockwidgetAt(index)) {
if (dw->options() & Controllers::DockWidget::Option_NotClosable) {
qWarning() << "QTabWidget::tabCloseRequested: Refusing to close dock widget with Option_NotClosable option. name=" << dw->uniqueName();
} else {
dw->view()->close();
}
} else {
qWarning() << "QTabWidget::tabCloseRequested Couldn't find dock widget for index" << index << "; count=" << count();
}
});
connect(this, &QTabWidget::currentChanged, this, [this](int index) {
m_stack->onCurrentTabChanged(index);
Q_EMIT m_stack->currentTabChanged(index);
Q_EMIT m_stack->currentDockWidgetChanged(m_stack->currentDockWidget());
});
if (!QTabWidget::tabBar()->isVisible())
setFocusProxy(nullptr);
setupTabBarButtons();
setDocumentMode(m_stack->options() & StackOption_DocumentMode);
}
int Stack_qtwidgets::numDockWidgets() const
{
return count();
}
void Stack_qtwidgets::removeDockWidget(Controllers::DockWidget *dw)
{
removeTab(indexOf(View_qtwidgets::asQWidget(dw)));
}
int Stack_qtwidgets::indexOfDockWidget(const Controllers::DockWidget *dw) const
{
return indexOf(View_qtwidgets::asQWidget(dw->view()));
}
void Stack_qtwidgets::mouseDoubleClickEvent(QMouseEvent *ev)
{
if (m_stack->onMouseDoubleClick(ev->pos())) {
ev->accept();
} else {
ev->ignore();
}
}
void Stack_qtwidgets::mousePressEvent(QMouseEvent *ev)
{
QTabWidget::mousePressEvent(ev);
if ((Config::self().flags() & Config::Flag_TitleBarIsFocusable) && !m_stack->frame()->isFocused()) {
// User clicked on the tab widget itself
m_stack->frame()->FocusScope::focus(Qt::MouseFocusReason);
}
}
void Stack_qtwidgets::tabInserted(int)
{
m_stack->onTabInserted();
}
void Stack_qtwidgets::tabRemoved(int)
{
m_stack->onTabRemoved();
}
void Stack_qtwidgets::setCurrentDockWidget(int index)
{
setCurrentIndex(index);
}
bool Stack_qtwidgets::insertDockWidget(int index, Controllers::DockWidget *dw,
const QIcon &icon, const QString &title)
{
insertTab(index, View_qtwidgets::asQWidget(dw), icon, title);
return true;
}
void Stack_qtwidgets::setTabBarAutoHide(bool b)
{
QTabWidget::setTabBarAutoHide(b);
}
void Stack_qtwidgets::renameTab(int index, const QString &text)
{
setTabText(index, text);
}
void Stack_qtwidgets::changeTabIcon(int index, const QIcon &icon)
{
setTabIcon(index, icon);
}
Controllers::DockWidget *Stack_qtwidgets::dockwidgetAt(int index) const
{
auto view = qobject_cast<DockWidget_qtwidgets *>(widget(index));
return view ? view->dockWidget() : nullptr;
}
int Stack_qtwidgets::currentIndex() const
{
return QTabWidget::currentIndex();
}
void Stack_qtwidgets::setupTabBarButtons()
{
if (!(Config::self().flags() & Config::Flag_ShowButtonsOnTabBarIfTitleBarHidden))
return;
// TODOv2: Make sure people can only inherit from the Default*variants
auto factory = static_cast<ViewFactory_qtwidgets *>(Config::self().viewFactory());
m_closeButton = factory->createTitleBarButton(this, TitleBarButtonType::Close);
m_floatButton = factory->createTitleBarButton(this, TitleBarButtonType::Float);
auto cornerWidget = new QWidget(this);
cornerWidget->setObjectName(QStringLiteral("Corner Widget"));
setCornerWidget(cornerWidget, Qt::TopRightCorner);
m_cornerWidgetLayout = new QHBoxLayout(cornerWidget);
m_cornerWidgetLayout->addWidget(m_floatButton);
m_cornerWidgetLayout->addWidget(m_closeButton);
connect(m_floatButton, &QAbstractButton::clicked, this, [this] {
Controllers::TitleBar *tb = m_stack->frame()->titleBar();
tb->onFloatClicked();
});
connect(m_closeButton, &QAbstractButton::clicked, this, [this] {
Controllers::TitleBar *tb = m_stack->frame()->titleBar();
tb->onCloseClicked();
});
updateMargins();
connect(DockRegistry::self(), &DockRegistry::windowChangedScreen, this, [this](Window::Ptr w) {
if (isInWindow(w))
updateMargins();
});
}
void Stack_qtwidgets::updateMargins()
{
const qreal factor = logicalDpiFactor(this);
m_cornerWidgetLayout->setContentsMargins(QMargins(0, 0, 2, 0) * factor);
m_cornerWidgetLayout->setSpacing(int(2 * factor));
}
void Stack_qtwidgets::showContextMenu(QPoint pos)
{
if (!(Config::self().flags() & Config::Flag_AllowSwitchingTabsViaMenu))
return;
QTabBar *tabBar = QTabWidget::tabBar();
// We don't want context menu if there is only one tab
if (tabBar->count() <= 1)
return;
// Click on a tab => No menu
if (tabBar->tabAt(pos) >= 0)
return;
// Right click is allowed only on the tabs area
QRect tabAreaRect = tabBar->rect();
tabAreaRect.setWidth(this->width());
if (!tabAreaRect.contains(pos))
return;
QMenu menu(this);
for (int i = 0; i < tabBar->count(); ++i) {
QAction *action = menu.addAction(tabText(i), this, [this, i] {
setCurrentIndex(i);
});
if (i == currentIndex())
action->setDisabled(true);
}
menu.exec(mapToGlobal(pos));
}
QTabBar *Stack_qtwidgets::tabBar() const
{
return static_cast<QTabBar *>(View_qtwidgets::asQWidget((m_stack->tabBar())));
}
void Stack_qtwidgets::setDocumentMode(bool is)
{
QTabWidget::setDocumentMode(is);
}
Controllers::Stack *Stack_qtwidgets::stack() const
{
return m_stack;
}
bool Stack_qtwidgets::isPositionDraggable(QPoint p) const
{
if (tabPosition() != QTabWidget::North) {
qWarning() << Q_FUNC_INFO << "Not implemented yet. Only North is supported";
return false;
}
return p.y() >= 0 && p.y() <= tabBar()->height();
}