diff options
author | Fabian Bumberger <fbumberger@rim.com> | 2013-06-12 17:05:35 +0200 |
---|---|---|
committer | The Qt Project <gerrit-noreply@qt-project.org> | 2013-07-31 12:15:41 +0200 |
commit | 78e7783a237b8cfc3491aaad04bda5fe9568106e (patch) | |
tree | 65a3083063e1366993a2d9e776be99dd91246d01 | |
parent | d517d39d5491bc95b1dddc3ff5cb880bd3bd0058 (diff) | |
download | qt3d-78e7783a237b8cfc3491aaad04bda5fe9568106e.tar.gz |
Bugfixes and cleanups for the picking code
- Removed the pickRegistry and related code, because it is not needed
- Fixed picking when navigation is turned off
- Introduced a new autotest for picking
Change-Id: Ib43789383760902f42542b45cfe49342f4ffa090
Reviewed-by: Sergio Ahumada <sergio.ahumada@digia.com>
-rw-r--r-- | src/imports/threed/viewport.cpp | 82 | ||||
-rw-r--r-- | src/imports/threed/viewport.h | 10 | ||||
-rw-r--r-- | tests/auto/auto.pro | 4 | ||||
-rw-r--r-- | tests/auto/qml3d_cpp/picking/data/tst_picking.qml | 98 | ||||
-rw-r--r-- | tests/auto/qml3d_cpp/picking/picking.pro | 12 | ||||
-rw-r--r-- | tests/auto/qml3d_cpp/picking/tst_picking.cpp | 134 | ||||
-rw-r--r-- | tests/auto/qml3d_cpp/qml3d_cpp.pro | 2 |
7 files changed, 279 insertions, 63 deletions
diff --git a/src/imports/threed/viewport.cpp b/src/imports/threed/viewport.cpp index 9e506f283..49a3deccd 100644 --- a/src/imports/threed/viewport.cpp +++ b/src/imports/threed/viewport.cpp @@ -236,9 +236,7 @@ public: bool panning; QPointF startPan; QPointF lastPan; - QPointF lastPick; QObject *lastObject; - bool needsPick; QVector3D startEye; QVector3D startCenter; QVector3D startUpVector; @@ -249,15 +247,7 @@ public: bool pickingRenderInitialized; QList<PickEvent *> pickEventQueue; - // INVARIANT: PickEvents are always either in the queue, or in the registry, never - // in both. Here "never" means "not outside of sections guarded by the lock". - // - // Serializing PickEvents across the signal/slot boundary is dicey, since they are - // not a value type. Instead keep our own registry of them here, and forward the - // id values instead. Should also make dispatch faster. - QMap<quint64, PickEvent *> pickEventRegistry; - - // This lock is for the registry and for the pick event queue itself. All accesses + // This lock is for the pick event queue itself. All accesses // to that data structure must be guarded by this lock. QMutexMaybeLocker::Lock pickEventQueueLock; @@ -270,7 +260,6 @@ public: void setDefaults(QGLPainter *painter); void setRenderSettings(QGLPainter *painter); void getOverflow(QMouseEvent *e); - PickEvent *takeFromRegistry(quint64 id); }; ViewportPrivate::ViewportPrivate() @@ -294,9 +283,7 @@ ViewportPrivate::ViewportPrivate() , panning(false) , startPan(-1, -1) , lastPan(-1, -1) - , lastPick(-1, -1) , lastObject(0) - , needsPick(true) , panModifiers(Qt::NoModifier) , renderMode(Viewport::UnknownRender) , directRenderInitialized(false) @@ -312,7 +299,6 @@ ViewportPrivate::~ViewportPrivate() { delete pickFbo; qDeleteAll(pickEventQueue); - qDeleteAll(pickEventRegistry); } void ViewportPrivate::setDefaults(QGLPainter *painter) @@ -372,20 +358,6 @@ void ViewportPrivate::setRenderSettings(QGLPainter *painter) glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); } -PickEvent *ViewportPrivate::takeFromRegistry(quint64 id) -{ - PickEvent *pick = 0; - QMutexMaybeLocker locker(&pickEventQueueLock); - QMap<quint64, PickEvent*>::iterator it = pickEventRegistry.find(id); - if (it != pickEventRegistry.end()) - { - pick = it.value(); - pickEventRegistry.erase(it); - } - locker.unlock(); - return pick; -} - const int Viewport::FBO_SIZE = 8; /*! @@ -939,7 +911,6 @@ void Viewport::render(QGLPainter *painter) // May've been set by early draw glDisable(GL_CULL_FACE); - d->needsPick = true; draw(painter); // May've been set by one of the items @@ -1194,8 +1165,6 @@ void Viewport::objectForPoint() QMutexMaybeLocker locker(&d->pickEventQueueLock); if (d->pickEventQueue.size() > 0) p = d->pickEventQueue.takeFirst(); - if (p) - d->pickEventRegistry.insert(p->id(), p); locker.unlock(); } if (!p) @@ -1205,8 +1174,10 @@ void Viewport::objectForPoint() // Check the viewport boundaries in case a mouse move has // moved the pointer outside the window. QRectF rect = boundingRect(); - if (!rect.contains(pt)) + if (!rect.contains(pt)) { + delete p; continue; + } if (!d->pickFbo) { @@ -1226,8 +1197,6 @@ void Viewport::objectForPoint() painter.setPicking(false); objectId = painter.pickObject(pt.x() / winToFboRatioW, pt.y() / winToFboRatioH); d->setDefaults(&painter); - d->needsPick = false; - d->lastPick = pt; } else { qWarning() << "Warning: unable to paint into fbo, picking will be unavailable"; continue; @@ -1237,7 +1206,7 @@ void Viewport::objectForPoint() d->lastObject = obj; p->setObject(obj); QMetaMethod m = metaObject()->method(p->callback()); - m.invoke(this, Qt::QueuedConnection, Q_ARG(quint64, p->id())); + m.invoke(this, Qt::QueuedConnection, Q_ARG(void*, p)); } } @@ -1277,7 +1246,7 @@ void Viewport::mousePressEvent(QMouseEvent *e) static int processMousePressInvocation = -1; if (processMousePressInvocation == -1) { - processMousePressInvocation = metaObject()->indexOfMethod("processMousePress(quint64)"); + processMousePressInvocation = metaObject()->indexOfMethod("processMousePress(PickEvent*)"); Q_ASSERT(processMousePressInvocation != -1); } if (!d->panning && d->picking) @@ -1285,6 +1254,8 @@ void Viewport::mousePressEvent(QMouseEvent *e) PickEvent * p = initiatePick(e); if (p) p->setCallback(processMousePressInvocation); + e->setAccepted(true); //This is necessary, otherwise we won't get a realese event + return; } if (d->navigation && e->button() == Qt::LeftButton) { @@ -1298,10 +1269,8 @@ void Viewport::mousePressEvent(QMouseEvent *e) e->setAccepted(true); } -void Viewport::processMousePress(quint64 eventId) +void Viewport::processMousePress(PickEvent *pick) { - QScopedPointer<PickEvent> pick(d->takeFromRegistry(eventId)); - Q_ASSERT(pick.data()); QObject *object = pick->object(); QMouseEvent *e = pick->event(); if (d->pressedObject) @@ -1328,10 +1297,12 @@ void Viewport::processMousePress(quint64 eventId) e->modifiers()); QCoreApplication::sendEvent(object, &event); } - else if (d->navigation && e->button() == Qt::LeftButton) + + if (d->navigation && e->button() == Qt::LeftButton) { processNavEvent(e); } + delete pick; } void Viewport::processNavEvent(QMouseEvent *e) @@ -1352,7 +1323,7 @@ void Viewport::mouseReleaseEvent(QMouseEvent *e) static int processMouseReleaseInvocation = -1; if (processMouseReleaseInvocation == -1) { - processMouseReleaseInvocation = metaObject()->indexOfMethod("processMouseRelease(quint64)"); + processMouseReleaseInvocation = metaObject()->indexOfMethod("processMouseRelease(PickEvent*)"); Q_ASSERT(processMouseReleaseInvocation != -1); } if (d->panning && e->button() == Qt::LeftButton) { @@ -1368,10 +1339,8 @@ void Viewport::mouseReleaseEvent(QMouseEvent *e) } } -void Viewport::processMouseRelease(quint64 eventId) +void Viewport::processMouseRelease(PickEvent *pick) { - QScopedPointer<PickEvent> pick(d->takeFromRegistry(eventId)); - Q_ASSERT(pick.data()); Q_ASSERT(d->pressedObject); QObject *object = pick->object(); QMouseEvent *e = pick->event(); @@ -1407,6 +1376,7 @@ void Viewport::processMouseRelease(quint64 eventId) e->screenPos(), e->button(), e->buttons(), e->modifiers()); QCoreApplication::sendEvent(pressed, &event); } + delete pick; } /*! @@ -1417,7 +1387,7 @@ void Viewport::mouseDoubleClickEvent(QMouseEvent *e) static int processMouseDoubleClickInvocation = -1; if (processMouseDoubleClickInvocation == -1) { - processMouseDoubleClickInvocation = metaObject()->indexOfMethod("processMouseDoubleClick(quint64)"); + processMouseDoubleClickInvocation = metaObject()->indexOfMethod("processMouseDoubleClick(PickEvent*)"); Q_ASSERT(processMouseDoubleClickInvocation != -1); } if (d->picking) { @@ -1428,10 +1398,8 @@ void Viewport::mouseDoubleClickEvent(QMouseEvent *e) QQuickItem::mouseDoubleClickEvent(e); } -void Viewport::processMouseDoubleClick(quint64 eventId) +void Viewport::processMouseDoubleClick(PickEvent *pick) { - QScopedPointer<PickEvent> pick(d->takeFromRegistry(eventId)); - Q_ASSERT(pick.data()); QObject *object = pick->object(); QMouseEvent *e = pick->event(); if (object) { @@ -1442,6 +1410,7 @@ void Viewport::processMouseDoubleClick(quint64 eventId) QCoreApplication::sendEvent(object, &event); e->setAccepted(true); } + delete pick; } /*! @@ -1481,7 +1450,7 @@ void Viewport::mouseMoveEvent(QMouseEvent *e) static int processMouseMoveInvocation = -1; if (processMouseMoveInvocation == -1) { - processMouseMoveInvocation = metaObject()->indexOfMethod("processMouseMove(quint64)"); + processMouseMoveInvocation = metaObject()->indexOfMethod("processMouseMove(PickEvent*)"); Q_ASSERT(processMouseMoveInvocation != -1); } if (d->panning) { @@ -1516,10 +1485,8 @@ void Viewport::mouseMoveEvent(QMouseEvent *e) e->setAccepted(true); } -void Viewport::processMouseMove(quint64 eventId) +void Viewport::processMouseMove(PickEvent *pick) { - QScopedPointer<PickEvent> pick(d->takeFromRegistry(eventId)); - Q_ASSERT(pick.data()); QObject *object = pick->object(); QMouseEvent *e = pick->event(); if (d->pressedObject) { @@ -1547,8 +1514,10 @@ void Viewport::processMouseMove(quint64 eventId) d->enteredObject = 0; } else { QQuickItem::mouseMoveEvent(e); + delete pick; return; } + delete pick; } /*! @@ -1683,7 +1652,7 @@ bool Viewport::hoverEvent(QHoverEvent *e) static int processMouseHoverInvocation = -1; if (processMouseHoverInvocation == -1) { - processMouseHoverInvocation = metaObject()->indexOfMethod("processMouseHover(quint64)"); + processMouseHoverInvocation = metaObject()->indexOfMethod("processMouseHover(PickEvent*)"); Q_ASSERT(processMouseHoverInvocation != -1); } if (!d->panning && d->picking) { @@ -1699,10 +1668,8 @@ bool Viewport::hoverEvent(QHoverEvent *e) return false; } -void Viewport::processMouseHover(quint64 eventId) +void Viewport::processMouseHover(PickEvent *pick) { - QScopedPointer<PickEvent> pick(d->takeFromRegistry(eventId)); - Q_ASSERT(pick.data()); QObject *object = pick->object(); QMouseEvent *e = pick->event(); @@ -1730,6 +1697,7 @@ void Viewport::processMouseHover(quint64 eventId) sendLeaveEvent(d->enteredObject); d->enteredObject = 0; } + delete pick; } // Zoom in and out according to the change in wheel delta. diff --git a/src/imports/threed/viewport.h b/src/imports/threed/viewport.h index bc586c3ed..73e4c7d66 100644 --- a/src/imports/threed/viewport.h +++ b/src/imports/threed/viewport.h @@ -163,11 +163,11 @@ private: void setupPickPaint(QGLPainter *painter, const QPointF &pt); bool mouseMoveOverflow(QMouseEvent *e) const; - Q_INVOKABLE void processMousePress(quint64 eventId); - Q_INVOKABLE void processMouseRelease(quint64); - Q_INVOKABLE void processMouseDoubleClick(quint64); - Q_INVOKABLE void processMouseMove(quint64); - Q_INVOKABLE void processMouseHover(quint64); + Q_INVOKABLE void processMousePress(PickEvent *event); + Q_INVOKABLE void processMouseRelease(PickEvent *event); + Q_INVOKABLE void processMouseDoubleClick(PickEvent *event); + Q_INVOKABLE void processMouseMove(PickEvent *event); + Q_INVOKABLE void processMouseHover(PickEvent *event); void processNavEvent(QMouseEvent *event); diff --git a/tests/auto/auto.pro b/tests/auto/auto.pro index 3aac8a00f..46114c1c1 100644 --- a/tests/auto/auto.pro +++ b/tests/auto/auto.pro @@ -5,5 +5,7 @@ SUBDIRS = threed \ qtHaveModule(qml): SUBDIRS += imports qtHaveModule(qmltest) { SUBDIRS += qml3d - !win32 : SUBDIRS += qml3d_visual + + !win32 : SUBDIRS += qml3d_visual \ + qml3d_cpp } diff --git a/tests/auto/qml3d_cpp/picking/data/tst_picking.qml b/tests/auto/qml3d_cpp/picking/data/tst_picking.qml new file mode 100644 index 000000000..c158e857f --- /dev/null +++ b/tests/auto/qml3d_cpp/picking/data/tst_picking.qml @@ -0,0 +1,98 @@ +/*************************************************************************** +** +** Copyright (C) 2011 - 2013 Research In Motion +** Contact: http://www.qt-project.org/legal +** +** This file is part of the Qt3D module of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** Commercial License Usage +** Licensees holding valid commercial Qt licenses may use this file in +** accordance with the commercial license agreement provided with the +** Software or, alternatively, in accordance with the terms contained in +** a written agreement between you and Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +** GNU General Public License Usage +** Alternatively, this file may be used under the terms of the GNU +** General Public License version 3.0 as published by the Free Software +** Foundation and appearing in the file LICENSE.GPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU General Public License version 3.0 requirements will be +** met: http://www.gnu.org/copyleft/gpl.html. +** +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +import Qt3D 2.0 +import Qt3D.Shapes 2.0 +import QtQuick 2.0 + +Rectangle +{ + id: topLevel + width: 480; height: 480 + + property string pickedObjectClicked; + property string pickedObjectPressed; + property string pickedObjectReleased; + property string hoveredObject; + + property alias navigation: viewport.navigation + + Viewport { + id: viewport + width: 480; height: 480 + picking: true + + Quad { + id: fullScreenQuad + + transform: [ + Rotation3D { + axis: Qt.vector3d(1,0,0); + angle: 90 + } + ] + position: Qt.vector3d(0, 1, 0) + + onClicked: topLevel.pickedObjectClicked = "first" + onPressed: topLevel.pickedObjectPressed = "first" + onReleased: topLevel.pickedObjectReleased = "first" + onHoverEnter: hoveredObject= "first" + onHoverLeave: hoveredObject= "" + } + + Quad { + id: smallerQuad + + transform: [ + Rotation3D { + axis: Qt.vector3d(1,0,0); + angle: 90 + } + ] + + position: Qt.vector3d(0, -1, 0) + + onClicked: topLevel.pickedObjectClicked = "second" + onPressed: topLevel.pickedObjectPressed = "second" + onReleased:topLevel.pickedObjectReleased = "second" + } + } +} diff --git a/tests/auto/qml3d_cpp/picking/picking.pro b/tests/auto/qml3d_cpp/picking/picking.pro new file mode 100644 index 000000000..80a81a7e5 --- /dev/null +++ b/tests/auto/qml3d_cpp/picking/picking.pro @@ -0,0 +1,12 @@ +TARGET = tst_qml3d_cpp +CONFIG += testcase +TEMPLATE=app +QT += testlib 3d 3dquick +QT += qml quick gui-private + +SOURCES += tst_picking.cpp +DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0 + +OTHER_FILES += data/tst_picking.qml + +TESTDATA = $$OTHER_FILES diff --git a/tests/auto/qml3d_cpp/picking/tst_picking.cpp b/tests/auto/qml3d_cpp/picking/tst_picking.cpp new file mode 100644 index 000000000..f9ab2ec99 --- /dev/null +++ b/tests/auto/qml3d_cpp/picking/tst_picking.cpp @@ -0,0 +1,134 @@ +/*************************************************************************** +** +** Copyright (C) 2011 - 2013 Research In Motion +** Contact: http://www.qt-project.org/legal +** +** This file is part of the Qt3D module of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** Commercial License Usage +** Licensees holding valid commercial Qt licenses may use this file in +** accordance with the commercial license agreement provided with the +** Software or, alternatively, in accordance with the terms contained in +** a written agreement between you and Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +** GNU General Public License Usage +** Alternatively, this file may be used under the terms of the GNU +** General Public License version 3.0 as published by the Free Software +** Foundation and appearing in the file LICENSE.GPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU General Public License version 3.0 requirements will be +** met: http://www.gnu.org/copyleft/gpl.html. +** +** +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include <QtTest/QtTest> +#include <QtQuick/qquickview.h> +#include <QtQml/qqmlengine.h> +#include <QtQml/qqmlcontext.h> +#include <QQuickItem> +#include <qpa/qwindowsysteminterface.h> + +class tst_Picking : public QObject +{ + Q_OBJECT +public: + tst_Picking() {} + ~tst_Picking() {} + +private slots: + void initTestCase(); + void cleanupTestCase(); + void testWithNavigation(); + void testWithoutNavigation(); + void testMove(); + +private: + void testMouse(); + +private: + QQuickItem *rootObject; + QQuickView *window; +}; + +void tst_Picking::initTestCase() +{ + window = new QQuickView(0); + window->setSource(QUrl::fromLocalFile(QFINDTESTDATA("data/tst_picking.qml"))); + window->setGeometry(0,0,480,480); + window->show(); + QTest::qWaitForWindowExposed(window); + rootObject = window->rootObject(); +} + +void tst_Picking::cleanupTestCase() +{ + delete window; +} + +void tst_Picking::testWithNavigation() +{ + rootObject->setProperty("navigation", true); + testMouse(); +} + +void tst_Picking::testWithoutNavigation() +{ + rootObject->setProperty("navigation", false); + testMouse(); +} + +void tst_Picking::testMouse() +{ + QTest::mousePress(window, Qt::LeftButton, 0, QPoint(240, 90)); + QTest::mouseRelease(window, Qt::LeftButton, 0, QPoint(240, 90), 300); + + QTRY_COMPARE(rootObject->property("pickedObjectPressed").toString(), QString("first")); + QTRY_COMPARE(rootObject->property("pickedObjectClicked").toString(), QString("first")); + QTRY_COMPARE(rootObject->property("pickedObjectReleased").toString(), QString("first")); + + QTest::mousePress(window, Qt::LeftButton, 0, QPoint(240, 320)); + QTest::mouseRelease(window, Qt::LeftButton, 0, QPoint(240, 240),300); + + QTRY_COMPARE(rootObject->property("pickedObjectPressed").toString(), QString("second")); + QTRY_COMPARE(rootObject->property("pickedObjectClicked").toString(), QString("first")); + QTRY_COMPARE(rootObject->property("pickedObjectReleased").toString(), QString("second")); + + QTest::mousePress(window, Qt::LeftButton, 0, QPoint(20, 100)); + QTest::mouseRelease(window, Qt::LeftButton, 0, QPoint(20, 100), 300); + + QTRY_COMPARE(rootObject->property("pickedObjectPressed").toString(), QString("second")); + QTRY_COMPARE(rootObject->property("pickedObjectClicked").toString(), QString("first")); + QTRY_COMPARE(rootObject->property("pickedObjectReleased").toString(), QString("second")); +} + +void tst_Picking::testMove() +{ + rootObject->setProperty("hoveredObject", ""); + QTest::mouseMove(window, QPoint(240, 90)); + QTRY_COMPARE(rootObject->property("hoveredObject").toString(), QString("first")); + + QTest::mouseMove(window, QPoint(240, 200)); + QTRY_COMPARE(rootObject->property("hoveredObject").toString(), QString("")); +} + +QTEST_MAIN(tst_Picking) + +#include "tst_picking.moc" diff --git a/tests/auto/qml3d_cpp/qml3d_cpp.pro b/tests/auto/qml3d_cpp/qml3d_cpp.pro new file mode 100644 index 000000000..baf619016 --- /dev/null +++ b/tests/auto/qml3d_cpp/qml3d_cpp.pro @@ -0,0 +1,2 @@ +TEMPLATE = subdirs +SUBDIRS = picking |