Files
KDDockWidgets/src/private/quick/QWidgetAdapter_quick.cpp
2020-11-27 13:59:27 +00:00

617 lines
13 KiB
C++

/*
This file is part of KDDockWidgets.
SPDX-FileCopyrightText: 2019-2020 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.
*/
/**
* @file
* @brief A class that is QWidget when building for QtWidgets, and QObject when building for QtQuick.
*
* Allows to have the same code base supporting both stacks.
*
* @author Sérgio Martins \<sergio.martins@kdab.com\>
*/
#include "QWidgetAdapter.h"
#include "FloatingWindow_p.h"
#include "MainWindowBase.h"
#include "DockRegistry_p.h"
#include <QResizeEvent>
#include <QMouseEvent>
#include <QQmlComponent>
#include <QQuickItem>
#include <QQmlEngine>
#include <QQuickView>
using namespace KDDockWidgets;
static bool flagsAreTopLevelFlags(Qt::WindowFlags flags)
{
return flags & (Qt::Window | Qt::Tool);
}
static QQuickItem* actualParentItem(QQuickItem *candidateParentItem, Qt::WindowFlags flags)
{
// When we have a top-level, such as FloatingWindow, we only want to set QObject parentship
// and not parentItem.
return flagsAreTopLevelFlags(flags) ? nullptr
: candidateParentItem;
}
QWidgetAdapter::QWidgetAdapter(QQuickItem *parent, Qt::WindowFlags flags)
: QQuickItem(actualParentItem(parent, flags))
, m_windowFlags(flags)
{
if (parent && flagsAreTopLevelFlags(flags)) {
// See comment in actualParentItem(). We set only the QObject parent. Mimics QWidget behaviour
QObject::setParent(parent);
}
connect(this, &QQuickItem::widthChanged, this, [this] {
onResize(size());
updateGeometry();
});
connect(this, &QQuickItem::heightChanged, this, [this] {
if (!m_windowIsBeingDestroyed) { // If Window is being destroyed we don't bother
onResize(size());
updateGeometry();
}
});
setSize(QSize(800, 800));
}
QWidgetAdapter::~QWidgetAdapter()
{
}
void QWidgetAdapter::raiseAndActivate()
{
if (QWindow *w = windowHandle()) {
w->raise();
w->requestActivate();
}
}
void QWidgetAdapter::setWindowOpacity(qreal level)
{
if (QWindow *w = windowHandle())
w->setOpacity(level);
}
bool QWidgetAdapter::onResize(QSize) { return false; }
void QWidgetAdapter::onLayoutRequest() {}
void QWidgetAdapter::onMousePress() {}
void QWidgetAdapter::onMouseMove(QPoint) {}
void QWidgetAdapter::onMouseRelease() {}
void QWidgetAdapter::onCloseEvent(QCloseEvent *) {}
void QWidgetAdapter::itemChange(QQuickItem::ItemChange change, const QQuickItem::ItemChangeData &data)
{
QQuickItem::itemChange(change, data);
// Emulate the QWidget behaviour as QQuickItem doesn't receive some QEvents.
switch (change) {
case QQuickItem::ItemParentHasChanged: {
QEvent ev(QEvent::ParentChange);
qApp->sendEvent(this, &ev); // Not calling event() directly, otherwise it would skip event filters
Q_EMIT parentChanged();
break;
}
case QQuickItem::ItemVisibleHasChanged: {
QEvent ev(isVisible() ? QEvent::Show : QEvent::Hide);
event(&ev);
break;
}
default:
break;
}
}
void QWidgetAdapter::geometryChanged(const QRectF &newGeometry, const QRectF &oldGeometry)
{
// Send a few events manually, since QQuickItem doesn't do it for us.
QQuickItem::geometryChanged(newGeometry, oldGeometry);
// Not calling event() directly, otherwise it would skip event filters
if (newGeometry.size() != oldGeometry.size()) {
QEvent ev(QEvent::Resize);
qApp->sendEvent(this, &ev);
}
if (newGeometry.topLeft() != oldGeometry.topLeft()) {
QEvent ev(QEvent::Move);
qApp->sendEvent(this, &ev);
}
}
void QWidgetAdapter::raise()
{
if (QWindow *w = windowHandle())
w->raise();
}
QSize QWidgetAdapter::minimumSize() const
{
const QSize min = property("kddockwidgets_min_size").toSize();
return min.expandedTo({KDDOCKWIDGETS_MIN_WIDTH, KDDOCKWIDGETS_MIN_HEIGHT});
}
QSize QWidgetAdapter::maximumSize() const
{
const QSize max = property("kddockwidgets_max_size").toSize();
const QSize defaultMax(KDDOCKWIDGETS_MAX_WIDTH, KDDOCKWIDGETS_MAX_HEIGHT);
return max.isEmpty() ? defaultMax
: max.boundedTo(defaultMax);
}
WId QWidgetAdapter::winId() const
{
if (QWindow *w = windowHandle())
return w->winId();
return WId(-1);
}
FloatingWindow * QWidgetAdapter::floatingWindow() const
{
if (auto fw = qobject_cast<FloatingWindow*>(window()))
return fw;
return nullptr;
}
QRect QWidgetAdapter::geometry() const
{
if (isTopLevel()) {
if (QWindow *w = windowHandle()) {
return w->geometry();
}
}
return KDDockWidgets::Private::geometry(this);
}
QRect QWidgetAdapter::rect() const
{
return QRectF(0, 0, width(), height()).toRect();
}
QPoint QWidgetAdapter::pos() const
{
return geometry().topLeft();
}
void QWidgetAdapter::show()
{
setVisible(true);
}
void QWidgetAdapter::setFixedHeight(int height)
{
setHeight(height);
}
void QWidgetAdapter::setFixedWidth(int width)
{
setWidth(width);
}
void QWidgetAdapter::setGeometry(QRect rect)
{
setWidth(rect.width());
setHeight(rect.height());
move(rect.topLeft());
}
QRect QWidgetAdapter::frameGeometry() const
{
if (QWindow *w = windowHandle())
return w->frameGeometry();
return geometry();
}
void QWidgetAdapter::grabMouse()
{
QQuickItem::grabMouse();
}
void QWidgetAdapter::releaseMouse()
{
QQuickItem::ungrabMouse();
}
void QWidgetAdapter::releaseKeyboard()
{
qWarning() << Q_FUNC_INFO << "Implement me";
}
void QWidgetAdapter::setMinimumSize(QSize sz)
{
if (minimumSize() != sz) {
setProperty("kddockwidgets_min_size", sz);
updateGeometry();
}
}
void QWidgetAdapter::setMaximumSize(QSize sz)
{
if (minimumSize() != sz) {
setProperty("kddockwidgets_max_size", sz);
updateGeometry();
}
}
void QWidgetAdapter::setMaximumSize(int w, int h)
{
QWidgetAdapter::setMaximumSize(QSize(w, h));
}
void QWidgetAdapter::setMinimumSize(int w, int h)
{
QWidgetAdapter::setMinimumSize(QSize(w, h));
}
void QWidgetAdapter::updateGeometry()
{
Q_EMIT geometryUpdated();
}
void QWidgetAdapter::resize(QSize sz)
{
setWidth(sz.width());
setHeight(sz.height());
}
void QWidgetAdapter::resize(int w, int h)
{
resize({w, h});
}
bool QWidgetAdapter::isWindow() const
{
QQuickItem *parent = parentItem();
if (!parent)
return true;
if (QQuickView *w = quickView()) {
if (parent == w->contentItem() || parent == w->rootObject())
return true;
}
return false;
}
bool QWidgetAdapter::isMaximized() const
{
if (QWindow *w = windowHandle())
return w->windowStates() & Qt::WindowMaximized;
return false;
}
bool KDDockWidgets::QWidgetAdapter::isActiveWindow() const
{
if (QWindow *w = windowHandle())
return w->isActive();
return false;
}
void QWidgetAdapter::showMaximized()
{
if (QWindow *w = windowHandle())
w->showMaximized();
}
void QWidgetAdapter::showMinimized()
{
if (QWindow *w = windowHandle())
w->showMinimized();
}
void QWidgetAdapter::showNormal()
{
if (QWindow *w = windowHandle())
w->showNormal();
}
QQuickView *QWidgetAdapter::quickView() const
{
return qobject_cast<QQuickView *>(QQuickItem::window());
}
QWindow *QWidgetAdapter::windowHandle() const
{
return QQuickItem::window();
}
QWidgetAdapter *QWidgetAdapter::window() const
{
// We return the top-most QWidgetAdapter
if (QWidgetAdapter *w = parentWidget(/*includeTransient =*/ false))
return w->window();
return const_cast<QWidgetAdapter *>(this);
}
QWidgetAdapter *QWidgetAdapter::parentWidget(bool includeTransient) const
{
QQuickItem *p = parentItem();
while (p) {
if (auto qa = qobject_cast<QWidgetAdapter*>(p))
return qa;
p = p->parentItem();
}
if (includeTransient) {
if (QQuickView *w = quickView()) {
// Here we're mimicing QWidget::parentWidget(), which can return the transient parent of the QWindow.
MainWindowBase *mw = DockRegistry::self()->mainWindowForHandle(w->transientParent());
if (mw)
return mw;
}
}
return nullptr;
}
QPoint QWidgetAdapter::mapToGlobal(QPoint pt) const
{
return QQuickItem::mapToGlobal(pt).toPoint();
}
QPoint QWidgetAdapter::mapFromGlobal(QPoint pt) const
{
return QQuickItem::mapFromGlobal(pt).toPoint();
}
QPoint QWidgetAdapter::mapTo(const QQuickItem *parent, const QPoint &pos) const
{
if (!parent)
return {};
return parent->mapFromGlobal(QQuickItem::mapToGlobal(pos)).toPoint();
}
bool QWidgetAdapter::testAttribute(Qt::WidgetAttribute attr) const
{
return m_widgetAttributes & attr;
}
void QWidgetAdapter::setAttribute(Qt::WidgetAttribute attr, bool enable)
{
if (enable)
m_widgetAttributes |= attr;
else
m_widgetAttributes &= ~attr;
}
void QWidgetAdapter::setWindowTitle(const QString &title)
{
if (QWindow *window = windowHandle())
window->setTitle(title);
}
void QWidgetAdapter::setWindowIcon(const QIcon &icon)
{
if (QWindow *window = windowHandle())
window->setIcon(icon);
}
bool QWidgetAdapter::close()
{
QCloseEvent ev;
onCloseEvent(&ev);
if (ev.isAccepted()) {
setVisible(false);
return true;
}
return false;
}
QQuickItem *QWidgetAdapter::childAt(QPoint p) const
{
return QQuickItem::childAt(p.x(), p.y());
}
void QWidgetAdapter::move(QPoint pt)
{
move(pt.x(), pt.y());
}
void QWidgetAdapter::move(int x, int y)
{
if (isTopLevel()) {
if (QWindow *w = windowHandle()) {
w->setPosition(x, y);
return;
}
}
setX(x);
setY(y);
setAttribute(Qt::WA_Moved);
}
void QWidgetAdapter::setParent(QQuickItem *p)
{
QQuickItem::setParent(p);
QQuickItem::setParentItem(p);
}
void QWidgetAdapter::activateWindow()
{
if (QWindow *w = windowHandle())
w->requestActivate();
}
void QWidgetAdapter::setSizePolicy(QSizePolicy sp)
{
m_sizePolicy = sp;
}
QSizePolicy QWidgetAdapter::sizePolicy() const
{
return m_sizePolicy;
}
QSize QWidgetAdapter::sizeHint() const
{
return m_sizeHint;
}
bool QWidgetAdapter::hasFocus() const
{
return hasActiveFocus();
}
void QWidgetAdapter::setWindowFlag(int flag, bool enable)
{
Q_UNUSED(flag);
Q_UNUSED(enable);
}
Qt::WindowFlags QWidgetAdapter::windowFlags() const
{
return m_windowFlags;
}
/** static */
QQuickItem *QWidgetAdapter::createItem(QQmlEngine *engine, const QString &filename)
{
QQmlComponent component(engine, filename);
QObject *obj = component.create();
if (!obj) {
qWarning() << Q_FUNC_INFO << component.errorString();
return nullptr;
}
return qobject_cast<QQuickItem*>(obj);
}
void QWidgetAdapter::makeItemFillParent(QQuickItem *item)
{
// This is equivalent to "anchors.fill: parent
if (!item) {
qWarning() << Q_FUNC_INFO << "Invalid item";
return;
}
QQuickItem *parentItem = item->parentItem();
if (!parentItem) {
qWarning() << Q_FUNC_INFO << "Invalid parentItem for" << item;
return;
}
QObject *anchors = item->property("anchors").value<QObject*>();
if (!anchors) {
qWarning() << Q_FUNC_INFO << "Invalid anchors for" << item;
return;
}
anchors->setProperty("fill", QVariant::fromValue(parentItem));
}
void QWidgetAdapter::setFlag(Qt::WindowType f, bool on)
{
if (on) {
m_windowFlags |= f;
} else {
m_windowFlags &= ~f;
}
}
Qt::FocusPolicy QWidgetAdapter::focusPolicy() const
{
return m_focusPolicy;
}
void QWidgetAdapter::setFocusPolicy(Qt::FocusPolicy policy)
{
m_focusPolicy = policy;
}
void QWidgetAdapter::setFocus(Qt::FocusReason reason)
{
QQuickItem::setFocus(true, reason);
forceActiveFocus(reason);
}
void QWidgetAdapter::render(QPainter *)
{
qWarning() << Q_FUNC_INFO << "Implement me";
}
void QWidgetAdapter::setMouseTracking(bool enabled)
{
m_mouseTrackingEnabled = enabled;
}
bool QWidgetAdapter::event(QEvent *ev)
{
if (ev->type() == QEvent::Close)
onCloseEvent(static_cast<QCloseEvent*>(ev));
return QQuickItem::event(ev);
}
bool QWidgetAdapter::eventFilter(QObject *watched, QEvent *ev)
{
if (qobject_cast<QWindow*>(watched)) {
if (m_mouseTrackingEnabled) {
switch (ev->type()) {
case QEvent::MouseMove:
case QEvent::MouseButtonPress:
case QEvent::MouseButtonRelease:
ev->ignore();
qApp->sendEvent(this, ev);
//qDebug() << "Mouse event" << ev;
if (ev->isAccepted())
return true;
break;
default:
break;
}
}
}
return QQuickItem::eventFilter(watched, ev);
}
void QWidgetAdapter::setWindowIsBeingDestroyed(bool is)
{
m_windowIsBeingDestroyed = is;
}
void QWidgetAdapter::create()
{
// Nothing to do, for QtQuick ?
}
QQuickItem* KDDockWidgets::Private::widgetForWindow(QWindow *window)
{
if (!window)
return nullptr;
return window->property("kddockwidgets_qwidget").value<QQuickItem*>();
}
LayoutGuestWidget::~LayoutGuestWidget() = default;