/* SPDX-FileCopyrightText: 2020 Michail Vourlakos SPDX-License-Identifier: GPL-2.0-or-later */ #include "eventssink.h" // local #include "view.h" #include "positioner.h" // Qt #include #include #include #include #include #include namespace Latte { namespace ViewPart { EventsSink::EventsSink(Latte::View *parent) : QObject(parent), m_view(parent) { } EventsSink::~EventsSink() { } QQuickItem *EventsSink::originParentItem() const { return m_originParentItem; } QQuickItem *EventsSink::destinationItem() const { return m_destinationItem; } void EventsSink::setSink(QQuickItem *originParent, QQuickItem *destination) { if ((m_originParentItem == originParent) && (m_destinationItem == destination)) { return; } m_originParentItem = originParent; m_destinationItem = destination; emit itemsChanged(); } bool EventsSink::isActive() { return ((m_originParentItem != nullptr) && (m_destinationItem != nullptr)); } void EventsSink::release() { setSink(nullptr, nullptr); } QEvent *EventsSink::onEvent(QEvent *e) { if (!e) { return nullptr; } if (!isActive()) { return e; } QEvent *sunkevent = e; switch (e->type()) { case QEvent::Leave: release(); break; case QEvent::DragEnter: if (auto de = static_cast(e)) { QPointF point = de->posF(); if (originSinksContain(point)) { auto de2 = new QDragEnterEvent(positionAdjustedForDestination(point).toPoint(), de->possibleActions(), de->mimeData(), de->mouseButtons(), de->keyboardModifiers()); sunkevent = de2; } else if (!destinationContains(point)) { release(); } } break; case QEvent::DragMove: if (auto de = static_cast(e)) { QPointF point = de->posF(); if (originSinksContain(point)) { auto de2 = new QDragMoveEvent(positionAdjustedForDestination(point).toPoint(), de->possibleActions(), de->mimeData(), de->mouseButtons(), de->keyboardModifiers()); sunkevent = de2; } else if (!destinationContains(point)) { release(); } } break; case QEvent::Drop: if (auto de = static_cast(e)) { QPointF point = de->posF(); if (originSinksContain(point)) { auto de2 = new QDropEvent(positionAdjustedForDestination(point).toPoint(), de->possibleActions(), de->mimeData(), de->mouseButtons(), de->keyboardModifiers()); sunkevent = de2; } else if (!destinationContains(point)) { release(); } } break; case QEvent::MouseMove: if (auto me = dynamic_cast(e)) { if (m_view->positioner() && m_view->positioner()->isCursorInsideView() && originSinksContain(me->windowPos())) { auto positionadjusted = positionAdjustedForDestination(me->windowPos()); auto me2 = new QMouseEvent(me->type(), positionadjusted, positionadjusted, positionadjusted + m_view->position(), me->button(), me->buttons(), me->modifiers()); sunkevent = me2; } else if (!destinationContains(me->windowPos())) { release(); } } break; case QEvent::MouseButtonPress: if (auto me = dynamic_cast(e)) { if (originSinksContain(me->windowPos())) { auto positionadjusted = positionAdjustedForDestination(me->windowPos()); auto me2 = new QMouseEvent(me->type(), positionadjusted, positionadjusted, positionadjusted + m_view->position(), me->button(), me->buttons(), me->modifiers()); qDebug() << "Sunk Event:: sunk event pressed..."; sunkevent = me2; } else if (!destinationContains(me->windowPos())) { release(); } } break; case QEvent::MouseButtonRelease: if (auto me = dynamic_cast(e)) { if (originSinksContain(me->windowPos())) { auto positionadjusted = positionAdjustedForDestination(me->windowPos()); auto me2 = new QMouseEvent(me->type(), positionadjusted, positionadjusted, positionadjusted + m_view->position(), me->button(), me->buttons(), me->modifiers()); sunkevent = me2; } else if (!destinationContains(me->windowPos())) { release(); } } break; case QEvent::Wheel: if (auto we = dynamic_cast(e)) { #if QT_VERSION < QT_VERSION_CHECK(5, 14, 0) QPoint pos = QPoint(we->x(), we->y()); #else QPoint pos = we->position().toPoint(); #endif if (originSinksContain(pos)) { auto positionadjusted = positionAdjustedForDestination(pos); auto we2 = new QWheelEvent(positionadjusted, positionadjusted + m_view->position(), we->pixelDelta(), we->angleDelta(), we->angleDelta().y(), we->orientation(), we->buttons(), we->modifiers(), we->phase()); sunkevent = we2; } else if (!destinationContains(pos)) { release(); } } break; default: break; } return sunkevent; } QPointF EventsSink::positionAdjustedForDestination(const QPointF &point) const { QRectF destinationRectToScene = m_destinationItem->mapRectToScene(QRectF(0, 0, m_destinationItem->width() - 1, m_destinationItem->height() - 1)); return QPointF(qBound(destinationRectToScene.left(), point.x(), destinationRectToScene.right()), qBound(destinationRectToScene.top(), point.y(), destinationRectToScene.bottom())); } bool EventsSink::destinationContains(const QPointF &point) const { QRectF destinationRectToScene = m_destinationItem->mapRectToScene(QRectF(0, 0, m_destinationItem->width() - 1, m_destinationItem->height() - 1)); return destinationRectToScene.contains(point); } bool EventsSink::originSinksContain(const QPointF &point) const { QRegion originsRegion; for(const auto currentOrigin: m_originParentItem->childItems()) { QRectF currentOriginGeometry = currentOrigin->mapRectToScene(QRectF(0, 0, currentOrigin->width(), currentOrigin->height())); originsRegion = originsRegion.united(currentOriginGeometry.toRect()); } return originsRegion.contains(point.toPoint()); } } }