From 097d2d9503da50d972fc53b07bec53a7e3e90799 Mon Sep 17 00:00:00 2001 From: Sam Gondelman Date: Thu, 20 Dec 2018 16:53:33 -0800 Subject: [PATCH] Revert "Case 20200: Update EntityScriptingInterface to not find local entities" --- interface/src/Application.cpp | 12 +- interface/src/Application.h | 2 +- interface/src/avatar/MyAvatar.cpp | 12 +- interface/src/raypick/LaserPointer.cpp | 6 +- .../raypick/LaserPointerScriptingInterface.h | 2 +- interface/src/raypick/ParabolaPick.cpp | 14 +- interface/src/raypick/PathPointer.cpp | 12 +- .../src/raypick/PickScriptingInterface.cpp | 22 +- .../src/raypick/PickScriptingInterface.h | 131 +++--- .../src/raypick/PointerScriptingInterface.h | 9 +- interface/src/raypick/RayPick.cpp | 13 +- .../src/raypick/RayPickScriptingInterface.h | 10 +- interface/src/raypick/StylusPointer.cpp | 2 +- interface/src/ui/Keyboard.cpp | 2 +- .../src/EntityTreeRenderer.cpp | 13 +- .../entities/src/EntityScriptingInterface.cpp | 130 +++--- .../entities/src/EntityScriptingInterface.h | 95 +++-- libraries/entities/src/EntityTree.cpp | 374 ++++++++---------- libraries/entities/src/EntityTree.h | 57 ++- libraries/entities/src/EntityTreeElement.cpp | 256 +++--------- libraries/entities/src/EntityTreeElement.h | 52 ++- libraries/pointers/src/Pick.cpp | 5 +- libraries/pointers/src/Pick.h | 79 +++- libraries/pointers/src/PickCacheOptimizer.h | 6 +- libraries/shared/src/PickFilter.h | 99 ----- .../utilities/tests/entityPerfTest.js | 8 + .../DomainContent/Toybox/pistol/pistol.js | 2 +- 27 files changed, 647 insertions(+), 778 deletions(-) delete mode 100644 libraries/shared/src/PickFilter.h diff --git a/interface/src/Application.cpp b/interface/src/Application.cpp index a3a8478b0d..5e41530d93 100644 --- a/interface/src/Application.cpp +++ b/interface/src/Application.cpp @@ -4911,7 +4911,7 @@ void Application::calibrateEyeTracker5Points() { #endif bool Application::exportEntities(const QString& filename, - const QVector& entityIDs, + const QVector& entityIDs, const glm::vec3* givenOffset) { QHash entities; @@ -4986,12 +4986,16 @@ bool Application::exportEntities(const QString& filename, float x, float y, floa glm::vec3 minCorner = center - vec3(scale); float cubeSize = scale * 2; AACube boundingCube(minCorner, cubeSize); - QVector entities; + QVector entities; + QVector ids; auto entityTree = getEntities()->getTree(); entityTree->withReadLock([&] { - entityTree->evalEntitiesInCube(boundingCube, PickFilter(), entities); + entityTree->findEntities(boundingCube, entities); + foreach(EntityItemPointer entity, entities) { + ids << entity->getEntityItemID(); + } }); - return exportEntities(filename, entities, ¢er); + return exportEntities(filename, ids, ¢er); } void Application::loadSettings() { diff --git a/interface/src/Application.h b/interface/src/Application.h index a78df6afdd..fd45a594b5 100644 --- a/interface/src/Application.h +++ b/interface/src/Application.h @@ -351,7 +351,7 @@ signals: public slots: QVector pasteEntities(float x, float y, float z); - bool exportEntities(const QString& filename, const QVector& entityIDs, const glm::vec3* givenOffset = nullptr); + bool exportEntities(const QString& filename, const QVector& entityIDs, const glm::vec3* givenOffset = nullptr); bool exportEntities(const QString& filename, float x, float y, float z, float scale); bool importEntities(const QString& url, const bool isObservable = true, const qint64 callerId = -1); void updateThreadPoolCount() const; diff --git a/interface/src/avatar/MyAvatar.cpp b/interface/src/avatar/MyAvatar.cpp index 46c1f3dda1..397817cf60 100755 --- a/interface/src/avatar/MyAvatar.cpp +++ b/interface/src/avatar/MyAvatar.cpp @@ -3187,15 +3187,17 @@ bool MyAvatar::requiresSafeLanding(const glm::vec3& positionIn, glm::vec3& bette OctreeElementPointer element; float distance; BoxFace face; - const auto lockType = Octree::Lock; // Should we refactor to take a lock just once? - bool* accurateResult = NULL; - + const bool visibleOnly = false; // This isn't quite what we really want here. findRayIntersection always works on mesh, skipping entirely based on collidable. // What we really want is to use the collision hull! // See https://highfidelity.fogbugz.com/f/cases/5003/findRayIntersection-has-option-to-use-collidableOnly-but-doesn-t-actually-use-colliders + const bool collidableOnly = true; + const bool precisionPicking = true; + const auto lockType = Octree::Lock; // Should we refactor to take a lock just once? + bool* accurateResult = NULL; + QVariantMap extraInfo; - EntityItemID entityID = entityTree->evalRayIntersection(startPointIn, directionIn, include, ignore, - PickFilter(PickFilter::getBitMask(PickFilter::FlagBit::COLLIDABLE) | PickFilter::getBitMask(PickFilter::FlagBit::PRECISE)), + EntityItemID entityID = entityTree->findRayIntersection(startPointIn, directionIn, include, ignore, visibleOnly, collidableOnly, precisionPicking, element, distance, face, normalOut, extraInfo, lockType, accurateResult); if (entityID.isNull()) { return false; diff --git a/interface/src/raypick/LaserPointer.cpp b/interface/src/raypick/LaserPointer.cpp index 236512f2fe..1b53f9ab30 100644 --- a/interface/src/raypick/LaserPointer.cpp +++ b/interface/src/raypick/LaserPointer.cpp @@ -125,7 +125,7 @@ LaserPointer::RenderState::RenderState(const OverlayID& startID, const OverlayID StartEndRenderState(startID, endID), _pathID(pathID) { if (!_pathID.isNull()) { - _pathIgnoreRays = qApp->getOverlays().getProperty(_pathID, "ignorePickIntersection").value.toBool(); + _pathIgnoreRays = qApp->getOverlays().getProperty(_pathID, "ignoreRayIntersection").value.toBool(); _lineWidth = qApp->getOverlays().getProperty(_pathID, "lineWidth").value.toFloat(); } } @@ -142,7 +142,7 @@ void LaserPointer::RenderState::disable() { if (!getPathID().isNull()) { QVariantMap pathProps; pathProps.insert("visible", false); - pathProps.insert("ignorePickIntersection", true); + pathProps.insert("ignoreRayIntersection", true); qApp->getOverlays().editOverlay(getPathID(), pathProps); } } @@ -156,7 +156,7 @@ void LaserPointer::RenderState::update(const glm::vec3& origin, const glm::vec3& pathProps.insert("start", vec3toVariant(origin)); pathProps.insert("end", endVariant); pathProps.insert("visible", true); - pathProps.insert("ignorePickIntersection", doesPathIgnoreRays()); + pathProps.insert("ignoreRayIntersection", doesPathIgnoreRays()); pathProps.insert("lineWidth", getLineWidth() * parentScale); qApp->getOverlays().editOverlay(getPathID(), pathProps); } diff --git a/interface/src/raypick/LaserPointerScriptingInterface.h b/interface/src/raypick/LaserPointerScriptingInterface.h index d85e329e9a..5aaacd7960 100644 --- a/interface/src/raypick/LaserPointerScriptingInterface.h +++ b/interface/src/raypick/LaserPointerScriptingInterface.h @@ -21,7 +21,7 @@ class LaserPointerScriptingInterface : public QObject, public Dependency { SINGLETON_DEPENDENCY /**jsdoc - * Synonym for {@link Pointers} as used for laser pointers. Deprecated. + * Synonym for {@link Pointers} as used for laser pointers. * * @namespace LaserPointers * diff --git a/interface/src/raypick/ParabolaPick.cpp b/interface/src/raypick/ParabolaPick.cpp index b93ced17c6..571f4a6ea6 100644 --- a/interface/src/raypick/ParabolaPick.cpp +++ b/interface/src/raypick/ParabolaPick.cpp @@ -9,7 +9,6 @@ #include "Application.h" #include "EntityScriptingInterface.h" -#include "PickScriptingInterface.h" #include "ui/overlays/Overlays.h" #include "avatar/AvatarManager.h" #include "scripting/HMDScriptingInterface.h" @@ -58,15 +57,10 @@ PickParabola ParabolaPick::getMathematicalPick() const { PickResultPointer ParabolaPick::getEntityIntersection(const PickParabola& pick) { if (glm::length2(pick.acceleration) > EPSILON && glm::length2(pick.velocity) > EPSILON) { - PickFilter searchFilter = getFilter(); - if (DependencyManager::get()->getForceCoarsePicking()) { - searchFilter.setFlag(PickFilter::COARSE, true); - searchFilter.setFlag(PickFilter::PRECISE, false); - } - + bool precisionPicking = !(getFilter().doesPickCoarse() || DependencyManager::get()->getForceCoarsePicking()); ParabolaToEntityIntersectionResult entityRes = - DependencyManager::get()->evalParabolaIntersectionVector(pick, searchFilter, - getIncludeItemsAs(), getIgnoreItemsAs()); + DependencyManager::get()->findParabolaIntersectionVector(pick, precisionPicking, + getIncludeItemsAs(), getIgnoreItemsAs(), !getFilter().doesPickInvisible(), !getFilter().doesPickNonCollidable()); if (entityRes.intersects) { return std::make_shared(IntersectionType::ENTITY, entityRes.entityID, entityRes.distance, entityRes.parabolicDistance, entityRes.intersection, pick, entityRes.surfaceNormal, entityRes.extraInfo); } @@ -76,7 +70,7 @@ PickResultPointer ParabolaPick::getEntityIntersection(const PickParabola& pick) PickResultPointer ParabolaPick::getOverlayIntersection(const PickParabola& pick) { if (glm::length2(pick.acceleration) > EPSILON && glm::length2(pick.velocity) > EPSILON) { - bool precisionPicking = !(getFilter().isCoarse() || DependencyManager::get()->getForceCoarsePicking()); + bool precisionPicking = !(getFilter().doesPickCoarse() || DependencyManager::get()->getForceCoarsePicking()); ParabolaToOverlayIntersectionResult overlayRes = qApp->getOverlays().findParabolaIntersectionVector(pick, precisionPicking, getIncludeItemsAs(), getIgnoreItemsAs(), !getFilter().doesPickInvisible(), !getFilter().doesPickNonCollidable()); diff --git a/interface/src/raypick/PathPointer.cpp b/interface/src/raypick/PathPointer.cpp index 8cee6134c8..00ab32bde4 100644 --- a/interface/src/raypick/PathPointer.cpp +++ b/interface/src/raypick/PathPointer.cpp @@ -253,12 +253,12 @@ StartEndRenderState::StartEndRenderState(const OverlayID& startID, const Overlay _startID(startID), _endID(endID) { if (!_startID.isNull()) { _startDim = vec3FromVariant(qApp->getOverlays().getProperty(_startID, "dimensions").value); - _startIgnoreRays = qApp->getOverlays().getProperty(_startID, "ignorePickIntersection").value.toBool(); + _startIgnoreRays = qApp->getOverlays().getProperty(_startID, "ignoreRayIntersection").value.toBool(); } if (!_endID.isNull()) { _endDim = vec3FromVariant(qApp->getOverlays().getProperty(_endID, "dimensions").value); _endRot = quatFromVariant(qApp->getOverlays().getProperty(_endID, "rotation").value); - _endIgnoreRays = qApp->getOverlays().getProperty(_endID, "ignorePickIntersection").value.toBool(); + _endIgnoreRays = qApp->getOverlays().getProperty(_endID, "ignoreRayIntersection").value.toBool(); } } @@ -275,13 +275,13 @@ void StartEndRenderState::disable() { if (!getStartID().isNull()) { QVariantMap startProps; startProps.insert("visible", false); - startProps.insert("ignorePickIntersection", true); + startProps.insert("ignoreRayIntersection", true); qApp->getOverlays().editOverlay(getStartID(), startProps); } if (!getEndID().isNull()) { QVariantMap endProps; endProps.insert("visible", false); - endProps.insert("ignorePickIntersection", true); + endProps.insert("ignoreRayIntersection", true); qApp->getOverlays().editOverlay(getEndID(), endProps); } _enabled = false; @@ -294,7 +294,7 @@ void StartEndRenderState::update(const glm::vec3& origin, const glm::vec3& end, startProps.insert("position", vec3toVariant(origin)); startProps.insert("visible", true); startProps.insert("dimensions", vec3toVariant(getStartDim() * parentScale)); - startProps.insert("ignorePickIntersection", doesStartIgnoreRays()); + startProps.insert("ignoreRayIntersection", doesStartIgnoreRays()); qApp->getOverlays().editOverlay(getStartID(), startProps); } @@ -346,7 +346,7 @@ void StartEndRenderState::update(const glm::vec3& origin, const glm::vec3& end, endProps.insert("position", vec3toVariant(position)); endProps.insert("rotation", quatToVariant(rotation)); endProps.insert("visible", true); - endProps.insert("ignorePickIntersection", doesEndIgnoreRays()); + endProps.insert("ignoreRayIntersection", doesEndIgnoreRays()); qApp->getOverlays().editOverlay(getEndID(), endProps); } _enabled = true; diff --git a/interface/src/raypick/PickScriptingInterface.cpp b/interface/src/raypick/PickScriptingInterface.cpp index e8f84e63fe..6e979d2d91 100644 --- a/interface/src/raypick/PickScriptingInterface.cpp +++ b/interface/src/raypick/PickScriptingInterface.cpp @@ -49,17 +49,11 @@ unsigned int PickScriptingInterface::createPick(const PickQuery::PickType type, } } -PickFilter getPickFilter(unsigned int filter) { - // FIXME: Picks always intersect visible and collidable things right now - filter = filter | (PickScriptingInterface::PICK_INCLUDE_VISIBLE() | PickScriptingInterface::PICK_INCLUDE_COLLIDABLE()); - return PickFilter(filter); -} - /**jsdoc * A set of properties that can be passed to {@link Picks.createPick} to create a new Ray Pick. * @typedef {object} Picks.RayPickProperties * @property {boolean} [enabled=false] If this Pick should start enabled or not. Disabled Picks do not updated their pick results. - * @property {number} [filter=0] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. + * @property {number} [filter=Picks.PICK_NOTHING] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. * @property {number} [maxDistance=0.0] The max distance at which this Pick will intersect. 0.0 = no max. < 0.0 is invalid. * @property {Uuid} parentID - The ID of the parent, either an avatar, an entity, an overlay, or a pick. * @property {number} [parentJointIndex=0] - The joint of the parent to parent to, for example, the joints on the model of an avatar. (default = 0, no joint) @@ -79,7 +73,7 @@ unsigned int PickScriptingInterface::createRayPick(const QVariant& properties) { PickFilter filter = PickFilter(); if (propMap["filter"].isValid()) { - filter = getPickFilter(propMap["filter"].toUInt()); + filter = PickFilter(propMap["filter"].toUInt()); } float maxDistance = 0.0f; @@ -117,7 +111,7 @@ unsigned int PickScriptingInterface::createRayPick(const QVariant& properties) { * @typedef {object} Picks.StylusPickProperties * @property {number} [hand=-1] An integer. 0 == left, 1 == right. Invalid otherwise. * @property {boolean} [enabled=false] If this Pick should start enabled or not. Disabled Picks do not updated their pick results. - * @property {number} [filter=0] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. + * @property {number} [filter=Picks.PICK_NOTHING] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. * @property {number} [maxDistance=0.0] The max distance at which this Pick will intersect. 0.0 = no max. < 0.0 is invalid. */ unsigned int PickScriptingInterface::createStylusPick(const QVariant& properties) { @@ -138,7 +132,7 @@ unsigned int PickScriptingInterface::createStylusPick(const QVariant& properties PickFilter filter = PickFilter(); if (propMap["filter"].isValid()) { - filter = getPickFilter(propMap["filter"].toUInt()); + filter = PickFilter(propMap["filter"].toUInt()); } float maxDistance = 0.0f; @@ -159,7 +153,7 @@ unsigned int PickScriptingInterface::createStylusPick(const QVariant& properties * A set of properties that can be passed to {@link Picks.createPick} to create a new Parabola Pick. * @typedef {object} Picks.ParabolaPickProperties * @property {boolean} [enabled=false] If this Pick should start enabled or not. Disabled Picks do not updated their pick results. - * @property {number} [filter=0] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. + * @property {number} [filter=Picks.PICK_NOTHING] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. * @property {number} [maxDistance=0.0] The max distance at which this Pick will intersect. 0.0 = no max. < 0.0 is invalid. * @property {Uuid} parentID - The ID of the parent, either an avatar, an entity, an overlay, or a pick. * @property {number} [parentJointIndex=0] - The joint of the parent to parent to, for example, the joints on the model of an avatar. (default = 0, no joint) @@ -184,7 +178,7 @@ unsigned int PickScriptingInterface::createParabolaPick(const QVariant& properti PickFilter filter = PickFilter(); if (propMap["filter"].isValid()) { - filter = getPickFilter(propMap["filter"].toUInt()); + filter = PickFilter(propMap["filter"].toUInt()); } float maxDistance = 0.0f; @@ -256,7 +250,7 @@ unsigned int PickScriptingInterface::createParabolaPick(const QVariant& properti * @typedef {object} Picks.CollisionPickProperties * @property {boolean} [enabled=false] If this Pick should start enabled or not. Disabled Picks do not updated their pick results. -* @property {number} [filter=0] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. +* @property {number} [filter=Picks.PICK_NOTHING] The filter for this Pick to use, constructed using filter flags combined using bitwise OR. * @property {Shape} shape - The information about the collision region's size and shape. Dimensions are in world space, but will scale with the parent if defined. * @property {Vec3} position - The position of the collision region, relative to a parent if defined. * @property {Quat} orientation - The orientation of the collision region, relative to a parent if defined. @@ -279,7 +273,7 @@ unsigned int PickScriptingInterface::createCollisionPick(const QVariant& propert PickFilter filter = PickFilter(); if (propMap["filter"].isValid()) { - filter = getPickFilter(propMap["filter"].toUInt()); + filter = PickFilter(propMap["filter"].toUInt()); } float maxDistance = 0.0f; diff --git a/interface/src/raypick/PickScriptingInterface.h b/interface/src/raypick/PickScriptingInterface.h index e795068cd3..94112d5fae 100644 --- a/interface/src/raypick/PickScriptingInterface.h +++ b/interface/src/raypick/PickScriptingInterface.h @@ -14,7 +14,6 @@ #include #include #include -#include /**jsdoc * The Picks API lets you create and manage objects for repeatedly calculating intersections in different ways. @@ -24,62 +23,41 @@ * @hifi-interface * @hifi-client-entity * - * @property {number} PICK_ENTITIES A filter flag. Include domain and avatar entities when intersecting. Read-only.. Deprecated. - * @property {number} PICK_OVERLAYS A filter flag. Include local entities when intersecting. Read-only.. Deprecated. - * - * @property {number} PICK_DOMAIN_ENTITIES A filter flag. Include domain entities when intersecting. Read-only.. - * @property {number} PICK_AVATAR_ENTITIES A filter flag. Include avatar entities when intersecting. Read-only.. - * @property {number} PICK_LOCAL_ENTITIES A filter flag. Include local entities when intersecting. Read-only.. - * @property {number} PICK_AVATARS A filter flag. Include avatars when intersecting. Read-only.. - * @property {number} PICK_HUD A filter flag. Include the HUD sphere when intersecting in HMD mode. Read-only.. - * - * @property {number} PICK_INCLUDE_VISIBLE A filter flag. Include visible objects when intersecting. Read-only.. - * @property {number} PICK_INCLUDE_INVISIBLE A filter flag. Include invisible objects when intersecting. Read-only.. - * - * @property {number} PICK_INCLUDE_COLLIDABLE A filter flag. Include collidable objects when intersecting. Read-only.. - * @property {number} PICK_INCLUDE_NONCOLLIDABLE A filter flag. Include non-collidable objects when intersecting. Read-only.. - * - * @property {number} PICK_PRECISE A filter flag. Pick against exact meshes. Read-only.. - * @property {number} PICK_COARSE A filter flag. Pick against coarse meshes. Read-only.. - * - * @property {number} PICK_ALL_INTERSECTIONS Read-only.. - * - * @property {number} INTERSECTED_NONE An intersection type. Intersected nothing with the given filter flags. Read-only. + * @property {number} PICK_NOTHING A filter flag. Don't intersect with anything. Read-only. + * @property {number} PICK_ENTITIES A filter flag. Include entities when intersecting. Read-only. + * @property {number} PICK_OVERLAYS A filter flag. Include overlays when intersecting. Read-only. + * @property {number} PICK_AVATARS A filter flag. Include avatars when intersecting. Read-only. + * @property {number} PICK_HUD A filter flag. Include the HUD sphere when intersecting in HMD mode. Read-only. + * @property {number} PICK_COARSE A filter flag. Pick against coarse meshes, instead of exact meshes. Read-only. + * @property {number} PICK_INCLUDE_INVISIBLE A filter flag. Include invisible objects when intersecting. Read-only. + * @property {number} PICK_INCLUDE_NONCOLLIDABLE A filter flag. Include non-collidable objects when intersecting. + * Read-only. + * @property {number} PICK_ALL_INTERSECTIONS Read-only. + * @property {number} INTERSECTED_NONE An intersection type. Intersected nothing with the given filter flags. + * Read-only. * @property {number} INTERSECTED_ENTITY An intersection type. Intersected an entity. Read-only. * @property {number} INTERSECTED_OVERLAY An intersection type. Intersected an overlay. Read-only. * @property {number} INTERSECTED_AVATAR An intersection type. Intersected an avatar. Read-only. * @property {number} INTERSECTED_HUD An intersection type. Intersected the HUD sphere. Read-only. - * @property {number} perFrameTimeBudget - The max number of usec to spend per frame updating Pick results. + * @property {number} perFrameTimeBudget - The max number of usec to spend per frame updating Pick results. Read-only. */ class PickScriptingInterface : public QObject, public Dependency { Q_OBJECT + Q_PROPERTY(unsigned int PICK_NOTHING READ PICK_NOTHING CONSTANT) Q_PROPERTY(unsigned int PICK_ENTITIES READ PICK_ENTITIES CONSTANT) Q_PROPERTY(unsigned int PICK_OVERLAYS READ PICK_OVERLAYS CONSTANT) - - Q_PROPERTY(unsigned int PICK_DOMAIN_ENTITIES READ PICK_DOMAIN_ENTITIES CONSTANT) - Q_PROPERTY(unsigned int PICK_AVATAR_ENTITIES READ PICK_AVATAR_ENTITIES CONSTANT) - Q_PROPERTY(unsigned int PICK_LOCAL_ENTITIES READ PICK_LOCAL_ENTITIES CONSTANT) Q_PROPERTY(unsigned int PICK_AVATARS READ PICK_AVATARS CONSTANT) Q_PROPERTY(unsigned int PICK_HUD READ PICK_HUD CONSTANT) - - Q_PROPERTY(unsigned int PICK_INCLUDE_VISIBLE READ PICK_INCLUDE_VISIBLE CONSTANT) - Q_PROPERTY(unsigned int PICK_INCLUDE_INVISIBLE READ PICK_INCLUDE_INVISIBLE CONSTANT) - - Q_PROPERTY(unsigned int PICK_INCLUDE_COLLIDABLE READ PICK_INCLUDE_COLLIDABLE CONSTANT) - Q_PROPERTY(unsigned int PICK_INCLUDE_NONCOLLIDABLE READ PICK_INCLUDE_NONCOLLIDABLE CONSTANT) - - Q_PROPERTY(unsigned int PICK_PRECISE READ PICK_PRECISE CONSTANT) Q_PROPERTY(unsigned int PICK_COARSE READ PICK_COARSE CONSTANT) - + Q_PROPERTY(unsigned int PICK_INCLUDE_INVISIBLE READ PICK_INCLUDE_INVISIBLE CONSTANT) + Q_PROPERTY(unsigned int PICK_INCLUDE_NONCOLLIDABLE READ PICK_INCLUDE_NONCOLLIDABLE CONSTANT) Q_PROPERTY(unsigned int PICK_ALL_INTERSECTIONS READ PICK_ALL_INTERSECTIONS CONSTANT) - Q_PROPERTY(unsigned int INTERSECTED_NONE READ INTERSECTED_NONE CONSTANT) Q_PROPERTY(unsigned int INTERSECTED_ENTITY READ INTERSECTED_ENTITY CONSTANT) Q_PROPERTY(unsigned int INTERSECTED_OVERLAY READ INTERSECTED_OVERLAY CONSTANT) Q_PROPERTY(unsigned int INTERSECTED_AVATAR READ INTERSECTED_AVATAR CONSTANT) Q_PROPERTY(unsigned int INTERSECTED_HUD READ INTERSECTED_HUD CONSTANT) - Q_PROPERTY(unsigned int perFrameTimeBudget READ getPerFrameTimeBudget WRITE setPerFrameTimeBudget) SINGLETON_DEPENDENCY public: @@ -94,13 +72,11 @@ public: * Adds a new Pick. * Different {@link PickType}s use different properties, and within one PickType, the properties you choose can lead to a wide range of behaviors. For example, * with PickType.Ray, depending on which optional parameters you pass, you could create a Static Ray Pick, a Mouse Ray Pick, or a Joint Ray Pick. - * Picks created with this method always intersect at least visible and collidable things * @function Picks.createPick * @param {PickType} type A PickType that specifies the method of picking to use * @param {Picks.RayPickProperties|Picks.StylusPickProperties|Picks.ParabolaPickProperties|Picks.CollisionPickProperties} properties A PickProperties object, containing all the properties for initializing this Pick * @returns {number} The ID of the created Pick. Used for managing the Pick. 0 if invalid. */ - // TODO: expand Pointers to be able to be fully configurable with PickFilters Q_INVOKABLE unsigned int createPick(const PickQuery::PickType type, const QVariant& properties); /**jsdoc @@ -251,80 +227,61 @@ public: */ Q_INVOKABLE bool isMouse(unsigned int uid); + // FIXME: Move to other property definitions. + Q_PROPERTY(unsigned int perFrameTimeBudget READ getPerFrameTimeBudget WRITE setPerFrameTimeBudget) + unsigned int getPerFrameTimeBudget() const; void setPerFrameTimeBudget(unsigned int numUsecs); public slots: + /**jsdoc + * @function Picks.PICK_NOTHING + * @returns {number} + */ + static constexpr unsigned int PICK_NOTHING() { return 0; } + /**jsdoc * @function Picks.PICK_ENTITIES * @returns {number} */ - static constexpr unsigned int PICK_ENTITIES() { return PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); } + static constexpr unsigned int PICK_ENTITIES() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_ENTITIES); } + /**jsdoc * @function Picks.PICK_OVERLAYS * @returns {number} */ - static constexpr unsigned int PICK_OVERLAYS() { return PickFilter::getBitMask(PickFilter::FlagBit::LOCAL_ENTITIES); } + static constexpr unsigned int PICK_OVERLAYS() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_OVERLAYS); } - /**jsdoc - * @function Picks.PICK_DOMAIN_ENTITIES - * @returns {number} - */ - static constexpr unsigned int PICK_DOMAIN_ENTITIES() { return PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES); } - /**jsdoc - * @function Picks.PICK_AVATAR_ENTITIES - * @returns {number} - */ - static constexpr unsigned int PICK_AVATAR_ENTITIES() { return PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); } - /**jsdoc - * @function Picks.PICK_LOCAL_ENTITIES - * @returns {number} - */ - static constexpr unsigned int PICK_LOCAL_ENTITIES() { return PickFilter::getBitMask(PickFilter::FlagBit::LOCAL_ENTITIES); } /**jsdoc * @function Picks.PICK_AVATARS * @returns {number} */ - static constexpr unsigned int PICK_AVATARS() { return PickFilter::getBitMask(PickFilter::FlagBit::AVATARS); } + static constexpr unsigned int PICK_AVATARS() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_AVATARS); } + /**jsdoc * @function Picks.PICK_HUD * @returns {number} */ - static constexpr unsigned int PICK_HUD() { return PickFilter::getBitMask(PickFilter::FlagBit::HUD); } + static constexpr unsigned int PICK_HUD() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_HUD); } - /**jsdoc - * @function Picks.PICK_INCLUDE_VISIBLE - * @returns {number} - */ - static constexpr unsigned int PICK_INCLUDE_VISIBLE() { return PickFilter::getBitMask(PickFilter::FlagBit::VISIBLE); } - /**jsdoc - * @function Picks.PICK_INCLUDE_INVISIBLE - * @returns {number} - */ - static constexpr unsigned int PICK_INCLUDE_INVISIBLE() { return PickFilter::getBitMask(PickFilter::FlagBit::INVISIBLE); } - - /**jsdoc - * @function Picks.PICK_INCLUDE_COLLIDABLE - * @returns {number} - */ - static constexpr unsigned int PICK_INCLUDE_COLLIDABLE() { return PickFilter::getBitMask(PickFilter::FlagBit::COLLIDABLE); } - /**jsdoc - * @function Picks.PICK_INCLUDE_NONCOLLIDABLE - * @returns {number} - */ - static constexpr unsigned int PICK_INCLUDE_NONCOLLIDABLE() { return PickFilter::getBitMask(PickFilter::FlagBit::NONCOLLIDABLE); } - - /**jsdoc - * @function Picks.PICK_PRECISE - * @returns {number} - */ - static constexpr unsigned int PICK_PRECISE() { return PickFilter::getBitMask(PickFilter::FlagBit::PRECISE); } /**jsdoc * @function Picks.PICK_COARSE * @returns {number} */ - static constexpr unsigned int PICK_COARSE() { return PickFilter::getBitMask(PickFilter::FlagBit::COARSE); } + static constexpr unsigned int PICK_COARSE() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_COARSE); } + + /**jsdoc + * @function Picks.PICK_INCLUDE_INVISIBLE + * @returns {number} + */ + static constexpr unsigned int PICK_INCLUDE_INVISIBLE() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_INCLUDE_INVISIBLE); } + + /**jsdoc + * @function Picks.PICK_INCLUDE_NONCOLLIDABLE + * @returns {number} + */ + static constexpr unsigned int PICK_INCLUDE_NONCOLLIDABLE() { return PickFilter::getBitMask(PickFilter::FlagBit::PICK_INCLUDE_NONCOLLIDABLE); } /**jsdoc * @function Picks.PICK_ALL_INTERSECTIONS diff --git a/interface/src/raypick/PointerScriptingInterface.h b/interface/src/raypick/PointerScriptingInterface.h index a21c1f2470..2677f37fae 100644 --- a/interface/src/raypick/PointerScriptingInterface.h +++ b/interface/src/raypick/PointerScriptingInterface.h @@ -41,12 +41,10 @@ public: * @property {string} button Which button to trigger. "Primary", "Secondary", "Tertiary", and "Focus" are currently supported. Only "Primary" will trigger clicks on web surfaces. If "Focus" is triggered, * it will try to set the entity or overlay focus to the object at which the Pointer is aimed. Buttons besides the first three will still trigger events, but event.button will be "None". */ - /**jsdoc * Adds a new Pointer * Different {@link PickType}s use different properties, and within one PickType, the properties you choose can lead to a wide range of behaviors. For example, * with PickType.Ray, depending on which optional parameters you pass, you could create a Static Ray Pointer, a Mouse Ray Pointer, or a Joint Ray Pointer. - * Pointers created with this method always intersect at least visible and collidable things * @function Pointers.createPointer * @param {PickType} type A PickType that specifies the method of picking to use * @param {Pointers.LaserPointerProperties|Pointers.StylusPointerProperties|Pointers.ParabolaPointerProperties} properties A PointerProperties object, containing all the properties for initializing this Pointer and the {@link Picks.PickProperties} for the Pick that @@ -60,21 +58,21 @@ public: * dimensions: {x:0.5, y:0.5, z:0.5}, * solid: true, * color: {red:0, green:255, blue:0}, - * ignorePickIntersection: true + * ignoreRayIntersection: true * }; * var end2 = { * type: "sphere", * dimensions: {x:0.5, y:0.5, z:0.5}, * solid: true, * color: {red:255, green:0, blue:0}, - * ignorePickIntersection: true + * ignoreRayIntersection: true * }; * * var renderStates = [ {name: "test", end: end} ]; * var defaultRenderStates = [ {name: "test", distance: 10.0, end: end2} ]; * var pointer = Pointers.createPointer(PickType.Ray, { * joint: "_CAMERA_RELATIVE_CONTROLLER_LEFTHAND", - * filter: Picks.PICK_LOCAL_ENTITIES | Picks.PICK_DOMAIN_ENTITIES | Picks.PICK_INCLUDE_NONCOLLIDABLE, + * filter: Picks.PICK_OVERLAYS | Picks.PICK_ENTITIES | Picks.PICK_INCLUDE_NONCOLLIDABLE, * renderStates: renderStates, * defaultRenderStates: defaultRenderStates, * distanceScaleEnd: true, @@ -84,7 +82,6 @@ public: * }); * Pointers.setRenderState(pointer, "test"); */ - // TODO: expand Pointers to be able to be fully configurable with PickFilters Q_INVOKABLE unsigned int createPointer(const PickQuery::PickType& type, const QVariant& properties); /**jsdoc diff --git a/interface/src/raypick/RayPick.cpp b/interface/src/raypick/RayPick.cpp index 507e45b470..a48d858504 100644 --- a/interface/src/raypick/RayPick.cpp +++ b/interface/src/raypick/RayPick.cpp @@ -27,15 +27,10 @@ PickRay RayPick::getMathematicalPick() const { } PickResultPointer RayPick::getEntityIntersection(const PickRay& pick) { - PickFilter searchFilter = getFilter(); - if (DependencyManager::get()->getForceCoarsePicking()) { - searchFilter.setFlag(PickFilter::COARSE, true); - searchFilter.setFlag(PickFilter::PRECISE, false); - } - + bool precisionPicking = !(getFilter().doesPickCoarse() || DependencyManager::get()->getForceCoarsePicking()); RayToEntityIntersectionResult entityRes = - DependencyManager::get()->evalRayIntersectionVector(pick, searchFilter, - getIncludeItemsAs(), getIgnoreItemsAs()); + DependencyManager::get()->findRayIntersectionVector(pick, precisionPicking, + getIncludeItemsAs(), getIgnoreItemsAs(), !getFilter().doesPickInvisible(), !getFilter().doesPickNonCollidable()); if (entityRes.intersects) { return std::make_shared(IntersectionType::ENTITY, entityRes.entityID, entityRes.distance, entityRes.intersection, pick, entityRes.surfaceNormal, entityRes.extraInfo); } else { @@ -44,7 +39,7 @@ PickResultPointer RayPick::getEntityIntersection(const PickRay& pick) { } PickResultPointer RayPick::getOverlayIntersection(const PickRay& pick) { - bool precisionPicking = !(getFilter().isCoarse() || DependencyManager::get()->getForceCoarsePicking()); + bool precisionPicking = !(getFilter().doesPickCoarse() || DependencyManager::get()->getForceCoarsePicking()); RayToOverlayIntersectionResult overlayRes = qApp->getOverlays().findRayIntersectionVector(pick, precisionPicking, getIncludeItemsAs(), getIgnoreItemsAs(), !getFilter().doesPickInvisible(), !getFilter().doesPickNonCollidable()); diff --git a/interface/src/raypick/RayPickScriptingInterface.h b/interface/src/raypick/RayPickScriptingInterface.h index 3ad0efd439..d5e224018e 100644 --- a/interface/src/raypick/RayPickScriptingInterface.h +++ b/interface/src/raypick/RayPickScriptingInterface.h @@ -19,13 +19,14 @@ #include "PickScriptingInterface.h" /**jsdoc - * Synonym for {@link Picks} as used for ray picks. Deprecated. + * Synonym for {@link Picks} as used for ray picks. * * @namespace RayPick * * @hifi-interface * @hifi-client-entity * + * @property {number} PICK_NOTHING Read-only. * @property {number} PICK_ENTITIES Read-only. * @property {number} PICK_OVERLAYS Read-only. * @property {number} PICK_AVATARS Read-only. @@ -43,6 +44,7 @@ class RayPickScriptingInterface : public QObject, public Dependency { Q_OBJECT + Q_PROPERTY(unsigned int PICK_NOTHING READ PICK_NOTHING CONSTANT) Q_PROPERTY(unsigned int PICK_ENTITIES READ PICK_ENTITIES CONSTANT) Q_PROPERTY(unsigned int PICK_OVERLAYS READ PICK_OVERLAYS CONSTANT) Q_PROPERTY(unsigned int PICK_AVATARS READ PICK_AVATARS CONSTANT) @@ -138,6 +140,12 @@ public: public slots: + /**jsdoc + * @function RayPick.PICK_NOTHING + * @returns {number} + */ + static unsigned int PICK_NOTHING() { return PickScriptingInterface::PICK_NOTHING(); } + /**jsdoc * @function RayPick.PICK_ENTITIES * @returns {number} diff --git a/interface/src/raypick/StylusPointer.cpp b/interface/src/raypick/StylusPointer.cpp index 867f896763..5595c54b71 100644 --- a/interface/src/raypick/StylusPointer.cpp +++ b/interface/src/raypick/StylusPointer.cpp @@ -61,7 +61,7 @@ OverlayID StylusPointer::buildStylusOverlay(const QVariantMap& properties) { overlayProperties["loadPriority"] = 10.0f; overlayProperties["solid"] = true; overlayProperties["visible"] = false; - overlayProperties["ignorePickIntersection"] = true; + overlayProperties["ignoreRayIntersection"] = true; overlayProperties["drawInFront"] = false; return qApp->getOverlays().addOverlay("model", overlayProperties); diff --git a/interface/src/ui/Keyboard.cpp b/interface/src/ui/Keyboard.cpp index 9e9a319802..9e3a3ce961 100644 --- a/interface/src/ui/Keyboard.cpp +++ b/interface/src/ui/Keyboard.cpp @@ -781,7 +781,7 @@ void Keyboard::loadKeyboardFile(const QString& keyboardFile) { { "isSolid", true }, { "visible", false }, { "grabbable", true }, - { "ignorePickIntersection", false }, + { "ignoreRayIntersection", false }, { "dimensions", anchorObject["dimensions"].toVariant() }, { "position", anchorObject["position"].toVariant() }, { "orientation", anchorObject["rotation"].toVariant() } diff --git a/libraries/entities-renderer/src/EntityTreeRenderer.cpp b/libraries/entities-renderer/src/EntityTreeRenderer.cpp index a88a8de308..4fe77c90ad 100644 --- a/libraries/entities-renderer/src/EntityTreeRenderer.cpp +++ b/libraries/entities-renderer/src/EntityTreeRenderer.cpp @@ -498,27 +498,20 @@ void EntityTreeRenderer::handleSpaceUpdate(std::pair proxyUp bool EntityTreeRenderer::findBestZoneAndMaybeContainingEntities(QVector* entitiesContainingAvatar) { bool didUpdate = false; float radius = 0.01f; // for now, assume 0.01 meter radius, because we actually check the point inside later - QVector entityIDs; + QVector foundEntities; // find the entities near us // don't let someone else change our tree while we search _tree->withReadLock([&] { - auto entityTree = std::static_pointer_cast(_tree); // FIXME - if EntityTree had a findEntitiesContainingPoint() this could theoretically be a little faster - entityTree->evalEntitiesInSphere(_avatarPosition, radius, - PickFilter(PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES)), entityIDs); + std::static_pointer_cast(_tree)->findEntities(_avatarPosition, radius, foundEntities); LayeredZones oldLayeredZones(std::move(_layeredZones)); _layeredZones.clear(); // create a list of entities that actually contain the avatar's position - for (auto& entityID : entityIDs) { - auto entity = entityTree->findEntityByID(entityID); - if (!entity) { - continue; - } - + for (auto& entity : foundEntities) { auto isZone = entity->getType() == EntityTypes::Zone; auto hasScript = !entity->getScript().isEmpty(); diff --git a/libraries/entities/src/EntityScriptingInterface.cpp b/libraries/entities/src/EntityScriptingInterface.cpp index 303e6d54c4..33ec92b8de 100644 --- a/libraries/entities/src/EntityScriptingInterface.cpp +++ b/libraries/entities/src/EntityScriptingInterface.cpp @@ -1085,10 +1085,13 @@ QUuid EntityScriptingInterface::findClosestEntity(const glm::vec3& center, float EntityItemID result; if (_entityTree) { - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); + EntityItemPointer closestEntity; _entityTree->withReadLock([&] { - result = _entityTree->evalClosestEntity(center, radius, PickFilter(searchFilter)); + closestEntity = _entityTree->findClosestEntity(center, radius); }); + if (closestEntity) { + result = closestEntity->getEntityItemID(); + } } return result; } @@ -1107,10 +1110,14 @@ QVector EntityScriptingInterface::findEntities(const glm::vec3& center, f QVector result; if (_entityTree) { - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); + QVector entities; _entityTree->withReadLock([&] { - _entityTree->evalEntitiesInSphere(center, radius, PickFilter(searchFilter), result); + _entityTree->findEntities(center, radius, entities); }); + + foreach (EntityItemPointer entity, entities) { + result << entity->getEntityItemID(); + } } return result; } @@ -1120,11 +1127,15 @@ QVector EntityScriptingInterface::findEntitiesInBox(const glm::vec3& corn QVector result; if (_entityTree) { - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); + QVector entities; _entityTree->withReadLock([&] { AABox box(corner, dimensions); - _entityTree->evalEntitiesInBox(box, PickFilter(searchFilter), result); + _entityTree->findEntities(box, entities); }); + + foreach (EntityItemPointer entity, entities) { + result << entity->getEntityItemID(); + } } return result; } @@ -1159,10 +1170,14 @@ QVector EntityScriptingInterface::findEntitiesInFrustum(QVariantMap frust viewFrustum.calculate(); if (_entityTree) { - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); + QVector entities; _entityTree->withReadLock([&] { - _entityTree->evalEntitiesInFrustum(viewFrustum, PickFilter(searchFilter), result); + _entityTree->findEntities(viewFrustum, entities); }); + + foreach(EntityItemPointer entity, entities) { + result << entity->getEntityItemID(); + } } } @@ -1174,64 +1189,86 @@ QVector EntityScriptingInterface::findEntitiesByType(const QString entity QVector result; if (_entityTree) { - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); + QVector entities; _entityTree->withReadLock([&] { - _entityTree->evalEntitiesInSphereWithType(center, radius, type, PickFilter(searchFilter), result); + _entityTree->findEntities(center, radius, entities); }); + + foreach(EntityItemPointer entity, entities) { + if (entity->getType() == type) { + result << entity->getEntityItemID().toString(); + } + } } return result; } QVector EntityScriptingInterface::findEntitiesByName(const QString entityName, const glm::vec3& center, float radius, bool caseSensitiveSearch) const { + QVector result; if (_entityTree) { + QVector entities; _entityTree->withReadLock([&] { - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); - _entityTree->evalEntitiesInSphereWithName(center, radius, entityName, caseSensitiveSearch, PickFilter(searchFilter), result); + _entityTree->findEntities(center, radius, entities); }); + + if (caseSensitiveSearch) { + foreach(EntityItemPointer entity, entities) { + if (entity->getName() == entityName) { + result << entity->getEntityItemID(); + } + } + + } else { + QString entityNameLowerCase = entityName.toLower(); + + foreach(EntityItemPointer entity, entities) { + QString entityItemLowerCase = entity->getName().toLower(); + if (entityItemLowerCase == entityNameLowerCase) { + result << entity->getEntityItemID(); + } + } + } } return result; } -RayToEntityIntersectionResult EntityScriptingInterface::findRayIntersection(const PickRay& ray, bool precisionPicking, - const QScriptValue& entityIdsToInclude, const QScriptValue& entityIdsToDiscard, bool visibleOnly, bool collidableOnly) const { - PROFILE_RANGE(script_entities, __FUNCTION__); +RayToEntityIntersectionResult EntityScriptingInterface::findRayIntersection(const PickRay& ray, bool precisionPicking, + const QScriptValue& entityIdsToInclude, const QScriptValue& entityIdsToDiscard, bool visibleOnly, bool collidableOnly) { QVector entitiesToInclude = qVectorEntityItemIDFromScriptValue(entityIdsToInclude); QVector entitiesToDiscard = qVectorEntityItemIDFromScriptValue(entityIdsToDiscard); - unsigned int searchFilter = PickFilter::getBitMask(PickFilter::FlagBit::DOMAIN_ENTITIES) | PickFilter::getBitMask(PickFilter::FlagBit::AVATAR_ENTITIES); - - if (!precisionPicking) { - searchFilter = searchFilter | PickFilter::getBitMask(PickFilter::FlagBit::COARSE); - } - - if (visibleOnly) { - searchFilter = searchFilter | PickFilter::getBitMask(PickFilter::FlagBit::VISIBLE); - } - - if (collidableOnly) { - searchFilter = searchFilter | PickFilter::getBitMask(PickFilter::FlagBit::COLLIDABLE); - } - - return evalRayIntersectionWorker(ray, Octree::Lock, PickFilter(searchFilter), entitiesToInclude, entitiesToDiscard); + return findRayIntersectionVector(ray, precisionPicking, entitiesToInclude, entitiesToDiscard, visibleOnly, collidableOnly); } -RayToEntityIntersectionResult EntityScriptingInterface::evalRayIntersectionVector(const PickRay& ray, PickFilter searchFilter, - const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard) { +RayToEntityIntersectionResult EntityScriptingInterface::findRayIntersectionVector(const PickRay& ray, bool precisionPicking, + const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly) { PROFILE_RANGE(script_entities, __FUNCTION__); - return evalRayIntersectionWorker(ray, Octree::Lock, searchFilter, entityIdsToInclude, entityIdsToDiscard); + return findRayIntersectionWorker(ray, Octree::Lock, precisionPicking, entityIdsToInclude, entityIdsToDiscard, visibleOnly, collidableOnly); } -RayToEntityIntersectionResult EntityScriptingInterface::evalRayIntersectionWorker(const PickRay& ray, - Octree::lockType lockType, PickFilter searchFilter, const QVector& entityIdsToInclude, - const QVector& entityIdsToDiscard) const { +// FIXME - we should remove this API and encourage all users to use findRayIntersection() instead. We've changed +// findRayIntersection() to be blocking because it never makes sense for a script to get back a non-answer +RayToEntityIntersectionResult EntityScriptingInterface::findRayIntersectionBlocking(const PickRay& ray, bool precisionPicking, + const QScriptValue& entityIdsToInclude, const QScriptValue& entityIdsToDiscard) { + + qWarning() << "Entities.findRayIntersectionBlocking() is obsolete, use Entities.findRayIntersection() instead."; + const QVector& entitiesToInclude = qVectorEntityItemIDFromScriptValue(entityIdsToInclude); + const QVector entitiesToDiscard = qVectorEntityItemIDFromScriptValue(entityIdsToDiscard); + return findRayIntersectionWorker(ray, Octree::Lock, precisionPicking, entitiesToInclude, entitiesToDiscard); +} + +RayToEntityIntersectionResult EntityScriptingInterface::findRayIntersectionWorker(const PickRay& ray, + Octree::lockType lockType, bool precisionPicking, const QVector& entityIdsToInclude, + const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly) { + RayToEntityIntersectionResult result; if (_entityTree) { OctreeElementPointer element; - result.entityID = _entityTree->evalRayIntersection(ray.origin, ray.direction, - entityIdsToInclude, entityIdsToDiscard, searchFilter, + result.entityID = _entityTree->findRayIntersection(ray.origin, ray.direction, + entityIdsToInclude, entityIdsToDiscard, visibleOnly, collidableOnly, precisionPicking, element, result.distance, result.face, result.surfaceNormal, result.extraInfo, lockType, &result.accurate); result.intersects = !result.entityID.isNull(); @@ -1242,22 +1279,23 @@ RayToEntityIntersectionResult EntityScriptingInterface::evalRayIntersectionWorke return result; } -ParabolaToEntityIntersectionResult EntityScriptingInterface::evalParabolaIntersectionVector(const PickParabola& parabola, PickFilter searchFilter, - const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard) { +ParabolaToEntityIntersectionResult EntityScriptingInterface::findParabolaIntersectionVector(const PickParabola& parabola, bool precisionPicking, + const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly) { PROFILE_RANGE(script_entities, __FUNCTION__); - return evalParabolaIntersectionWorker(parabola, Octree::Lock, searchFilter, entityIdsToInclude, entityIdsToDiscard); + return findParabolaIntersectionWorker(parabola, Octree::Lock, precisionPicking, entityIdsToInclude, entityIdsToDiscard, visibleOnly, collidableOnly); } -ParabolaToEntityIntersectionResult EntityScriptingInterface::evalParabolaIntersectionWorker(const PickParabola& parabola, - Octree::lockType lockType, PickFilter searchFilter, const QVector& entityIdsToInclude, - const QVector& entityIdsToDiscard) const { +ParabolaToEntityIntersectionResult EntityScriptingInterface::findParabolaIntersectionWorker(const PickParabola& parabola, + Octree::lockType lockType, bool precisionPicking, const QVector& entityIdsToInclude, + const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly) { + ParabolaToEntityIntersectionResult result; if (_entityTree) { OctreeElementPointer element; - result.entityID = _entityTree->evalParabolaIntersection(parabola, - entityIdsToInclude, entityIdsToDiscard, searchFilter, + result.entityID = _entityTree->findParabolaIntersection(parabola, + entityIdsToInclude, entityIdsToDiscard, visibleOnly, collidableOnly, precisionPicking, element, result.intersection, result.distance, result.parabolicDistance, result.face, result.surfaceNormal, result.extraInfo, lockType, &result.accurate); result.intersects = !result.entityID.isNull(); diff --git a/libraries/entities/src/EntityScriptingInterface.h b/libraries/entities/src/EntityScriptingInterface.h index 1398c2ad1c..890c666010 100644 --- a/libraries/entities/src/EntityScriptingInterface.h +++ b/libraries/entities/src/EntityScriptingInterface.h @@ -24,7 +24,6 @@ #include #include #include -#include #include "PolyVoxEntityItem.h" #include "LineEntityItem.h" @@ -57,7 +56,8 @@ private: }; /**jsdoc - * The result of a {@link PickRay} search using {@link Entities.findRayIntersection|findRayIntersection}. + * The result of a {@link PickRay} search using {@link Entities.findRayIntersection|findRayIntersection} or + * {@link Entities.findRayIntersectionBlocking|findRayIntersectionBlocking}. * @typedef {object} Entities.RayToEntityIntersectionResult * @property {boolean} intersects - true if the {@link PickRay} intersected an entity, otherwise * false. @@ -119,6 +119,7 @@ public: /// handles scripting of Entity commands from JS passed to assigned clients class EntityScriptingInterface : public OctreeScriptingInterface, public Dependency { Q_OBJECT + Q_PROPERTY(QUuid keyboardFocusEntity READ getKeyboardFocusEntity WRITE setKeyboardFocusEntity) friend EntityPropertyMetadataRequest; @@ -143,10 +144,10 @@ public: void resetActivityTracking(); ActivityTracking getActivityTracking() const { return _activityTracking; } - RayToEntityIntersectionResult evalRayIntersectionVector(const PickRay& ray, PickFilter searchFilter, - const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard); - ParabolaToEntityIntersectionResult evalParabolaIntersectionVector(const PickParabola& parabola, PickFilter searchFilter, - const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard); + // TODO: expose to script? + ParabolaToEntityIntersectionResult findParabolaIntersectionVector(const PickParabola& parabola, bool precisionPicking, + const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, + bool visibleOnly, bool collidableOnly); /**jsdoc * Get the properties of multiple entities. @@ -393,8 +394,9 @@ public slots: Q_INVOKABLE void callEntityClientMethod(QUuid clientSessionID, QUuid entityID, const QString& method, const QStringList& params = QStringList()); + /**jsdoc - * Find the non-local entity with a position closest to a specified point and within a specified radius. + * Find the entity with a position closest to a specified point and within a specified radius. * @function Entities.findClosestEntity * @param {Vec3} center - The point about which to search. * @param {number} radius - The radius within which to search. @@ -408,7 +410,7 @@ public slots: Q_INVOKABLE QUuid findClosestEntity(const glm::vec3& center, float radius) const; /**jsdoc - * Find all non-local entities that intersect a sphere defined by a center point and radius. + * Find all entities that intersect a sphere defined by a center point and radius. * @function Entities.findEntities * @param {Vec3} center - The point about which to search. * @param {number} radius - The radius within which to search. @@ -422,23 +424,23 @@ public slots: Q_INVOKABLE QVector findEntities(const glm::vec3& center, float radius) const; /**jsdoc - * Find all non-local entities whose axis-aligned boxes intersect a search axis-aligned box defined by its minimum coordinates corner + * Find all entities whose axis-aligned boxes intersect a search axis-aligned box defined by its minimum coordinates corner * and dimensions. * @function Entities.findEntitiesInBox * @param {Vec3} corner - The corner of the search AA box with minimum co-ordinate values. * @param {Vec3} dimensions - The dimensions of the search AA box. - * @returns {Uuid[]} An array of entity IDs whose AA boxes intersect the search AA box. The array is empty if no entities + * @returns {Uuid[]} An array of entity IDs whose AA boxes intersect the search AA box. The array is empty if no entities * could be found. */ /// this function will not find any models in script engine contexts which don't have access to models Q_INVOKABLE QVector findEntitiesInBox(const glm::vec3& corner, const glm::vec3& dimensions) const; /**jsdoc - * Find all non-local entities whose axis-aligned boxes intersect a search frustum. + * Find all entities whose axis-aligned boxes intersect a search frustum. * @function Entities.findEntitiesInFrustum * @param {ViewFrustum} frustum - The frustum to search in. The position, orientation, * projection, and centerRadius properties must be specified. - * @returns {Uuid[]} An array of entity IDs axis-aligned boxes intersect the frustum. The array is empty if no entities + * @returns {Uuid[]} An array of entity IDs axis-aligned boxes intersect the frustum. The array is empty if no entities * could be found. * @example Report the number of entities in view. * var entityIDs = Entities.findEntitiesInFrustum(Camera.frustum); @@ -448,12 +450,12 @@ public slots: Q_INVOKABLE QVector findEntitiesInFrustum(QVariantMap frustum) const; /**jsdoc - * Find all non-local entities of a particular type that intersect a sphere defined by a center point and radius. + * Find all entities of a particular type that intersect a sphere defined by a center point and radius. * @function Entities.findEntitiesByType * @param {Entities.EntityType} entityType - The type of entity to search for. * @param {Vec3} center - The point about which to search. * @param {number} radius - The radius within which to search. - * @returns {Uuid[]} An array of entity IDs of the specified type that intersect the search sphere. The array is empty if + * @returns {Uuid[]} An array of entity IDs of the specified type that intersect the search sphere. The array is empty if * no entities could be found. * @example Report the number of Model entities within 10m of your avatar. * var entityIDs = Entities.findEntitiesByType("Model", MyAvatar.position, 10); @@ -463,7 +465,7 @@ public slots: Q_INVOKABLE QVector findEntitiesByType(const QString entityType, const glm::vec3& center, float radius) const; /**jsdoc - * Find all non-local entities with a particular name that intersect a sphere defined by a center point and radius. + * Find all entities of a particular name that intersect a sphere defined by a center point and radius. * @function Entities.findEntitiesByName * @param {string} entityName - The name of the entity to search for. * @param {Vec3} center - The point about which to search. @@ -473,13 +475,13 @@ public slots: * if no entities could be found. * @example Report the number of entities with the name, "Light-Target". * var entityIDs = Entities.findEntitiesByName("Light-Target", MyAvatar.position, 10, false); - * print("Number of entities with the name Light-Target: " + entityIDs.length); + * print("Number of entities with the name "Light-Target": " + entityIDs.length); */ - Q_INVOKABLE QVector findEntitiesByName(const QString entityName, const glm::vec3& center, float radius, - bool caseSensitiveSearch = false) const; + Q_INVOKABLE QVector findEntitiesByName(const QString entityName, const glm::vec3& center, float radius, + bool caseSensitiveSearch = false ) const; /**jsdoc - * Find the first non-local entity intersected by a {@link PickRay}. Light and Zone entities are not + * Find the first entity intersected by a {@link PickRay}. Light and Zone entities are not * intersected unless they've been configured as pickable using {@link Entities.setLightsArePickable|setLightsArePickable} * and {@link Entities.setZonesArePickable|setZonesArePickable}, respectively.
* @function Entities.findRayIntersection @@ -510,8 +512,33 @@ public slots: /// may be inaccurate if the engine is unable to access the visible entities, in which case result.accurate /// will be false. Q_INVOKABLE RayToEntityIntersectionResult findRayIntersection(const PickRay& ray, bool precisionPicking = false, - const QScriptValue& entityIdsToInclude = QScriptValue(), const QScriptValue& entityIdsToDiscard = QScriptValue(), - bool visibleOnly = false, bool collidableOnly = false) const; + const QScriptValue& entityIdsToInclude = QScriptValue(), const QScriptValue& entityIdsToDiscard = QScriptValue(), + bool visibleOnly = false, bool collidableOnly = false); + + /// Same as above but with QVectors + RayToEntityIntersectionResult findRayIntersectionVector(const PickRay& ray, bool precisionPicking, + const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, + bool visibleOnly, bool collidableOnly); + + /**jsdoc + * Find the first entity intersected by a {@link PickRay}. Light and Zone entities are not + * intersected unless they've been configured as pickable using {@link Entities.setLightsArePickable|setLightsArePickable} + * and {@link Entities.setZonesArePickable|setZonesArePickable}, respectively.
+ * This is a synonym for {@link Entities.findRayIntersection|findRayIntersection}. + * @function Entities.findRayIntersectionBlocking + * @param {PickRay} pickRay - The PickRay to use for finding entities. + * @param {boolean} [precisionPicking=false] - If true and the intersected entity is a Model + * entity, the result's extraInfo property includes more information than it otherwise would. + * @param {Uuid[]} [entitiesToInclude=[]] - If not empty then the search is restricted to these entities. + * @param {Uuid[]} [entitiesToDiscard=[]] - Entities to ignore during the search. + * @deprecated This function is deprecated and will soon be removed. Use + * {@link Entities.findRayIntersection|findRayIntersection} instead; it blocks and performs the same function. + */ + /// If the scripting context has visible entities, this will determine a ray intersection, and will block in + /// order to return an accurate result + Q_INVOKABLE RayToEntityIntersectionResult findRayIntersectionBlocking(const PickRay& ray, bool precisionPicking = false, + const QScriptValue& entityIdsToInclude = QScriptValue(), const QScriptValue& entityIdsToDiscard = QScriptValue()); + /**jsdoc * Reloads an entity's server entity script such that the latest version re-downloaded. @@ -576,7 +603,9 @@ public slots: /**jsdoc * Set whether or not ray picks intersect the bounding box of {@link Entities.EntityType|Light} entities. By default, Light * entities are not intersected. The setting lasts for the Interface session. Ray picks are done using - * {@link Entities.findRayIntersection|findRayIntersection}, or the {@link Picks} API. + * {@link Entities.findRayIntersection|findRayIntersection} or + * {@link Entities.findRayIntersectionBlocking|findRayIntersectionBlocking}, or the {@link Picks} and {@link RayPick} + * APIs. * @function Entities.setLightsArePickable * @param {boolean} value - Set true to make ray picks intersect the bounding box of * {@link Entities.EntityType|Light} entities, otherwise false. @@ -586,7 +615,9 @@ public slots: /**jsdoc * Get whether or not ray picks intersect the bounding box of {@link Entities.EntityType|Light} entities. Ray picks are - * done using {@link Entities.findRayIntersection|findRayIntersection}, or the {@link Picks} API. + * done using {@link Entities.findRayIntersection|findRayIntersection} or + * {@link Entities.findRayIntersectionBlocking|findRayIntersectionBlocking}, or the {@link Picks} and {@link RayPick} + * APIs. * @function Entities.getLightsArePickable * @returns {boolean} true if ray picks intersect the bounding box of {@link Entities.EntityType|Light} * entities, otherwise false. @@ -597,7 +628,9 @@ public slots: /**jsdoc * Set whether or not ray picks intersect the bounding box of {@link Entities.EntityType|Zone} entities. By default, Light * entities are not intersected. The setting lasts for the Interface session. Ray picks are done using - * {@link Entities.findRayIntersection|findRayIntersection}, or the {@link Picks} API. + * {@link Entities.findRayIntersection|findRayIntersection} or + * {@link Entities.findRayIntersectionBlocking|findRayIntersectionBlocking}, or the {@link Picks} and {@link RayPick} + * APIs. * @function Entities.setZonesArePickable * @param {boolean} value - Set true to make ray picks intersect the bounding box of * {@link Entities.EntityType|Zone} entities, otherwise false. @@ -607,7 +640,9 @@ public slots: /**jsdoc * Get whether or not ray picks intersect the bounding box of {@link Entities.EntityType|Zone} entities. Ray picks are - * done using {@link Entities.findRayIntersection|findRayIntersection}, or the {@link Picks} API. + * done using {@link Entities.findRayIntersection|findRayIntersection} or + * {@link Entities.findRayIntersectionBlocking|findRayIntersectionBlocking}, or the {@link Picks} and {@link RayPick} + * APIs. * @function Entities.getZonesArePickable * @returns {boolean} true if ray picks intersect the bounding box of {@link Entities.EntityType|Zone} * entities, otherwise false. @@ -1952,12 +1987,14 @@ private: /// actually does the work of finding the ray intersection, can be called in locking mode or tryLock mode - RayToEntityIntersectionResult evalRayIntersectionWorker(const PickRay& ray, Octree::lockType lockType, - PickFilter searchFilter, const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard) const; + RayToEntityIntersectionResult findRayIntersectionWorker(const PickRay& ray, Octree::lockType lockType, + bool precisionPicking, const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, + bool visibleOnly = false, bool collidableOnly = false); /// actually does the work of finding the parabola intersection, can be called in locking mode or tryLock mode - ParabolaToEntityIntersectionResult evalParabolaIntersectionWorker(const PickParabola& parabola, Octree::lockType lockType, - PickFilter searchFilter, const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard) const; + ParabolaToEntityIntersectionResult findParabolaIntersectionWorker(const PickParabola& parabola, Octree::lockType lockType, + bool precisionPicking, const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, + bool visibleOnly = false, bool collidableOnly = false); EntityTreePointer _entityTree; diff --git a/libraries/entities/src/EntityTree.cpp b/libraries/entities/src/EntityTree.cpp index b23fbca6d1..dd020da5a0 100644 --- a/libraries/entities/src/EntityTree.cpp +++ b/libraries/entities/src/EntityTree.cpp @@ -43,6 +43,50 @@ static const quint64 DELETED_ENTITIES_EXTRA_USECS_TO_CONSIDER = USECS_PER_MSEC * 50; const float EntityTree::DEFAULT_MAX_TMP_ENTITY_LIFETIME = 60 * 60; // 1 hour +// combines the ray cast arguments into a single object +class RayArgs { +public: + // Inputs + glm::vec3 origin; + glm::vec3 direction; + glm::vec3 invDirection; + const QVector& entityIdsToInclude; + const QVector& entityIdsToDiscard; + bool visibleOnly; + bool collidableOnly; + bool precisionPicking; + + // Outputs + OctreeElementPointer& element; + float& distance; + BoxFace& face; + glm::vec3& surfaceNormal; + QVariantMap& extraInfo; + EntityItemID entityID; +}; + +class ParabolaArgs { +public: + // Inputs + glm::vec3 origin; + glm::vec3 velocity; + glm::vec3 acceleration; + const QVector& entityIdsToInclude; + const QVector& entityIdsToDiscard; + bool visibleOnly; + bool collidableOnly; + bool precisionPicking; + + // Outputs + OctreeElementPointer& element; + float& parabolicDistance; + BoxFace& face; + glm::vec3& surfaceNormal; + QVariantMap& extraInfo; + EntityItemID entityID; +}; + + EntityTree::EntityTree(bool shouldReaverage) : Octree(shouldReaverage) { @@ -734,32 +778,59 @@ void EntityTree::processRemovedEntities(const DeleteEntityOperator& theOperator) } } -class RayArgs { -public: - // Inputs - glm::vec3 origin; - glm::vec3 direction; - glm::vec3 invDirection; - const QVector& entityIdsToInclude; - const QVector& entityIdsToDiscard; - PickFilter searchFilter; - // Outputs - OctreeElementPointer& element; - float& distance; - BoxFace& face; - glm::vec3& surfaceNormal; - QVariantMap& extraInfo; - EntityItemID entityID; +class FindNearPointArgs { +public: + glm::vec3 position; + float targetRadius; + bool found; + EntityItemPointer closestEntity; + float closestEntityDistance; }; -bool evalRayIntersectionOp(const OctreeElementPointer& element, void* extraData) { + +bool EntityTree::findNearPointOperation(const OctreeElementPointer& element, void* extraData) { + FindNearPointArgs* args = static_cast(extraData); + EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); + + glm::vec3 penetration; + bool sphereIntersection = entityTreeElement->getAACube().findSpherePenetration(args->position, args->targetRadius, penetration); + + // If this entityTreeElement contains the point, then search it... + if (sphereIntersection) { + EntityItemPointer thisClosestEntity = entityTreeElement->getClosestEntity(args->position); + + // we may have gotten NULL back, meaning no entity was available + if (thisClosestEntity) { + glm::vec3 entityPosition = thisClosestEntity->getWorldPosition(); + float distanceFromPointToEntity = glm::distance(entityPosition, args->position); + + // If we're within our target radius + if (distanceFromPointToEntity <= args->targetRadius) { + // we are closer than anything else we've found + if (distanceFromPointToEntity < args->closestEntityDistance) { + args->closestEntity = thisClosestEntity; + args->closestEntityDistance = distanceFromPointToEntity; + args->found = true; + } + } + } + + // we should be able to optimize this... + return true; // keep searching in case children have closer entities + } + + // if this element doesn't contain the point, then none of its children can contain the point, so stop searching + return false; +} + +bool findRayIntersectionOp(const OctreeElementPointer& element, void* extraData) { RayArgs* args = static_cast(extraData); bool keepSearching = true; EntityTreeElementPointer entityTreeElementPointer = std::static_pointer_cast(element); - EntityItemID entityID = entityTreeElementPointer->evalRayIntersection(args->origin, args->direction, + EntityItemID entityID = entityTreeElementPointer->findRayIntersection(args->origin, args->direction, args->element, args->distance, args->face, args->surfaceNormal, args->entityIdsToInclude, - args->entityIdsToDiscard, args->searchFilter, args->extraInfo); + args->entityIdsToDiscard, args->visibleOnly, args->collidableOnly, args->extraInfo, args->precisionPicking); if (!entityID.isNull()) { args->entityID = entityID; // We recurse OctreeElements in order, so if we hit something, we can stop immediately @@ -768,7 +839,7 @@ bool evalRayIntersectionOp(const OctreeElementPointer& element, void* extraData) return keepSearching; } -float evalRayIntersectionSortingOp(const OctreeElementPointer& element, void* extraData) { +float findRayIntersectionSortingOp(const OctreeElementPointer& element, void* extraData) { RayArgs* args = static_cast(extraData); EntityTreeElementPointer entityTreeElementPointer = std::static_pointer_cast(element); float distance = FLT_MAX; @@ -789,18 +860,19 @@ float evalRayIntersectionSortingOp(const OctreeElementPointer& element, void* ex return distance; } -EntityItemID EntityTree::evalRayIntersection(const glm::vec3& origin, const glm::vec3& direction, +EntityItemID EntityTree::findRayIntersection(const glm::vec3& origin, const glm::vec3& direction, QVector entityIdsToInclude, QVector entityIdsToDiscard, - PickFilter searchFilter, OctreeElementPointer& element, float& distance, + bool visibleOnly, bool collidableOnly, bool precisionPicking, + OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo, Octree::lockType lockType, bool* accurateResult) { RayArgs args = { origin, direction, 1.0f / direction, entityIdsToInclude, entityIdsToDiscard, - searchFilter, element, distance, face, surfaceNormal, extraInfo, EntityItemID() }; + visibleOnly, collidableOnly, precisionPicking, element, distance, face, surfaceNormal, extraInfo, EntityItemID() }; distance = FLT_MAX; bool requireLock = lockType == Octree::Lock; bool lockResult = withReadLock([&]{ - recurseTreeWithOperationSorted(evalRayIntersectionOp, evalRayIntersectionSortingOp, &args); + recurseTreeWithOperationSorted(findRayIntersectionOp, findRayIntersectionSortingOp, &args); }, requireLock); if (accurateResult) { @@ -810,32 +882,13 @@ EntityItemID EntityTree::evalRayIntersection(const glm::vec3& origin, const glm: return args.entityID; } -class ParabolaArgs { -public: - // Inputs - glm::vec3 origin; - glm::vec3 velocity; - glm::vec3 acceleration; - const QVector& entityIdsToInclude; - const QVector& entityIdsToDiscard; - PickFilter searchFilter; - - // Outputs - OctreeElementPointer& element; - float& parabolicDistance; - BoxFace& face; - glm::vec3& surfaceNormal; - QVariantMap& extraInfo; - EntityItemID entityID; -}; - -bool evalParabolaIntersectionOp(const OctreeElementPointer& element, void* extraData) { +bool findParabolaIntersectionOp(const OctreeElementPointer& element, void* extraData) { ParabolaArgs* args = static_cast(extraData); bool keepSearching = true; EntityTreeElementPointer entityTreeElementPointer = std::static_pointer_cast(element); - EntityItemID entityID = entityTreeElementPointer->evalParabolaIntersection(args->origin, args->velocity, args->acceleration, + EntityItemID entityID = entityTreeElementPointer->findParabolaIntersection(args->origin, args->velocity, args->acceleration, args->element, args->parabolicDistance, args->face, args->surfaceNormal, args->entityIdsToInclude, - args->entityIdsToDiscard, args->searchFilter, args->extraInfo); + args->entityIdsToDiscard, args->visibleOnly, args->collidableOnly, args->extraInfo, args->precisionPicking); if (!entityID.isNull()) { args->entityID = entityID; // We recurse OctreeElements in order, so if we hit something, we can stop immediately @@ -844,7 +897,7 @@ bool evalParabolaIntersectionOp(const OctreeElementPointer& element, void* extra return keepSearching; } -float evalParabolaIntersectionSortingOp(const OctreeElementPointer& element, void* extraData) { +float findParabolaIntersectionSortingOp(const OctreeElementPointer& element, void* extraData) { ParabolaArgs* args = static_cast(extraData); EntityTreeElementPointer entityTreeElementPointer = std::static_pointer_cast(element); float distance = FLT_MAX; @@ -865,20 +918,20 @@ float evalParabolaIntersectionSortingOp(const OctreeElementPointer& element, voi return distance; } -EntityItemID EntityTree::evalParabolaIntersection(const PickParabola& parabola, +EntityItemID EntityTree::findParabolaIntersection(const PickParabola& parabola, QVector entityIdsToInclude, QVector entityIdsToDiscard, - PickFilter searchFilter, + bool visibleOnly, bool collidableOnly, bool precisionPicking, OctreeElementPointer& element, glm::vec3& intersection, float& distance, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo, Octree::lockType lockType, bool* accurateResult) { ParabolaArgs args = { parabola.origin, parabola.velocity, parabola.acceleration, entityIdsToInclude, entityIdsToDiscard, - searchFilter, element, parabolicDistance, face, surfaceNormal, extraInfo, EntityItemID() }; + visibleOnly, collidableOnly, precisionPicking, element, parabolicDistance, face, surfaceNormal, extraInfo, EntityItemID() }; parabolicDistance = FLT_MAX; distance = FLT_MAX; bool requireLock = lockType == Octree::Lock; bool lockResult = withReadLock([&] { - recurseTreeWithOperationSorted(evalParabolaIntersectionOp, evalParabolaIntersectionSortingOp, &args); + recurseTreeWithOperationSorted(findParabolaIntersectionOp, findParabolaIntersectionSortingOp, &args); }, requireLock); if (accurateResult) { @@ -893,80 +946,33 @@ EntityItemID EntityTree::evalParabolaIntersection(const PickParabola& parabola, return args.entityID; } -class FindClosestEntityArgs { -public: - // Inputs - glm::vec3 position; - float targetRadius; - PickFilter searchFilter; - // Outputs - QUuid closestEntity; - float closestEntityDistance; -}; - - -bool evalClosestEntityOperation(const OctreeElementPointer& element, void* extraData) { - FindClosestEntityArgs* args = static_cast(extraData); - EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - - glm::vec3 penetration; - bool sphereIntersection = entityTreeElement->getAACube().findSpherePenetration(args->position, args->targetRadius, penetration); - - // If this entityTreeElement contains the point, then search it... - if (sphereIntersection) { - float closestDistanceSquared = FLT_MAX; - QUuid thisClosestEntity = entityTreeElement->evalClosetEntity(args->position, args->searchFilter, closestDistanceSquared); - - // we may have gotten NULL back, meaning no entity was available - if (!thisClosestEntity.isNull()) { - float distanceFromPointToEntity = glm::sqrt(closestDistanceSquared); - - // If we're within our target radius - if (distanceFromPointToEntity <= args->targetRadius) { - // we are closer than anything else we've found - if (distanceFromPointToEntity < args->closestEntityDistance) { - args->closestEntity = thisClosestEntity; - args->closestEntityDistance = distanceFromPointToEntity; - } - } - } - - // we should be able to optimize this... - return true; // keep searching in case children have closer entities - } - - // if this element doesn't contain the point, then none of its children can contain the point, so stop searching - return false; -} - -// NOTE: assumes caller has handled locking -QUuid EntityTree::evalClosestEntity(const glm::vec3& position, float targetRadius, PickFilter searchFilter) { - FindClosestEntityArgs args = { position, targetRadius, searchFilter, QUuid(), FLT_MAX }; - recurseTreeWithOperation(evalClosestEntityOperation, &args); +EntityItemPointer EntityTree::findClosestEntity(const glm::vec3& position, float targetRadius) { + FindNearPointArgs args = { position, targetRadius, false, NULL, FLT_MAX }; + withReadLock([&] { + // NOTE: This should use recursion, since this is a spatial operation + recurseTreeWithOperation(findNearPointOperation, &args); + }); return args.closestEntity; } -class FindEntitiesInSphereArgs { +class FindAllNearPointArgs { public: - // Inputs glm::vec3 position; float targetRadius; - PickFilter searchFilter; - - // Outputs - QVector entities; + QVector entities; }; -bool evalInSphereOperation(const OctreeElementPointer& element, void* extraData) { - FindEntitiesInSphereArgs* args = static_cast(extraData); + +bool EntityTree::findInSphereOperation(const OctreeElementPointer& element, void* extraData) { + FindAllNearPointArgs* args = static_cast(extraData); glm::vec3 penetration; bool sphereIntersection = element->getAACube().findSpherePenetration(args->position, args->targetRadius, penetration); // If this element contains the point, then search it... if (sphereIntersection) { EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - entityTreeElement->evalEntitiesInSphere(args->position, args->targetRadius, args->searchFilter, args->entities); + entityTreeElement->getEntities(args->position, args->targetRadius, args->entities); return true; // keep searching in case children have closer entities } @@ -975,168 +981,104 @@ bool evalInSphereOperation(const OctreeElementPointer& element, void* extraData) } // NOTE: assumes caller has handled locking -void EntityTree::evalEntitiesInSphere(const glm::vec3& center, float radius, PickFilter searchFilter, QVector& foundEntities) { - FindEntitiesInSphereArgs args = { center, radius, searchFilter, QVector() }; - recurseTreeWithOperation(evalInSphereOperation, &args); - foundEntities.swap(args.entities); -} +void EntityTree::findEntities(const glm::vec3& center, float radius, QVector& foundEntities) { + FindAllNearPointArgs args = { center, radius, QVector() }; + // NOTE: This should use recursion, since this is a spatial operation + recurseTreeWithOperation(findInSphereOperation, &args); -class FindEntitiesInSphereWithTypeArgs { -public: - // Inputs - glm::vec3 position; - float targetRadius; - EntityTypes::EntityType type; - PickFilter searchFilter; - - // Outputs - QVector entities; -}; - -bool evalInSphereWithTypeOperation(const OctreeElementPointer& element, void* extraData) { - FindEntitiesInSphereWithTypeArgs* args = static_cast(extraData); - glm::vec3 penetration; - bool sphereIntersection = element->getAACube().findSpherePenetration(args->position, args->targetRadius, penetration); - - // If this element contains the point, then search it... - if (sphereIntersection) { - EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - entityTreeElement->evalEntitiesInSphereWithType(args->position, args->targetRadius, args->type, args->searchFilter, args->entities); - return true; // keep searching in case children have closer entities - } - - // if this element doesn't contain the point, then none of it's children can contain the point, so stop searching - return false; -} - -// NOTE: assumes caller has handled locking -void EntityTree::evalEntitiesInSphereWithType(const glm::vec3& center, float radius, EntityTypes::EntityType type, PickFilter searchFilter, QVector& foundEntities) { - FindEntitiesInSphereWithTypeArgs args = { center, radius, type, searchFilter, QVector() }; - recurseTreeWithOperation(evalInSphereWithTypeOperation, &args); - foundEntities.swap(args.entities); -} - -class FindEntitiesInSphereWithNameArgs { -public: - // Inputs - glm::vec3 position; - float targetRadius; - QString name; - bool caseSensitive; - PickFilter searchFilter; - - // Outputs - QVector entities; -}; - -bool evalInSphereWithNameOperation(const OctreeElementPointer& element, void* extraData) { - FindEntitiesInSphereWithNameArgs* args = static_cast(extraData); - glm::vec3 penetration; - bool sphereIntersection = element->getAACube().findSpherePenetration(args->position, args->targetRadius, penetration); - - // If this element contains the point, then search it... - if (sphereIntersection) { - EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - entityTreeElement->evalEntitiesInSphereWithName(args->position, args->targetRadius, args->name, args->caseSensitive, args->searchFilter, args->entities); - return true; // keep searching in case children have closer entities - } - - // if this element doesn't contain the point, then none of it's children can contain the point, so stop searching - return false; -} - -// NOTE: assumes caller has handled locking -void EntityTree::evalEntitiesInSphereWithName(const glm::vec3& center, float radius, const QString& name, bool caseSensitive, PickFilter searchFilter, QVector& foundEntities) { - FindEntitiesInSphereWithNameArgs args = { center, radius, name, caseSensitive, searchFilter, QVector() }; - recurseTreeWithOperation(evalInSphereWithNameOperation, &args); + // swap the two lists of entity pointers instead of copy foundEntities.swap(args.entities); } class FindEntitiesInCubeArgs { public: - // Inputs - AACube cube; - PickFilter searchFilter; + FindEntitiesInCubeArgs(const AACube& cube) + : _cube(cube), _foundEntities() { + } - // Outputs - QVector entities; + AACube _cube; + QVector _foundEntities; }; -bool findInCubeOperation(const OctreeElementPointer& element, void* extraData) { +bool EntityTree::findInCubeOperation(const OctreeElementPointer& element, void* extraData) { FindEntitiesInCubeArgs* args = static_cast(extraData); - if (element->getAACube().touches(args->cube)) { + if (element->getAACube().touches(args->_cube)) { EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - entityTreeElement->evalEntitiesInCube(args->cube, args->searchFilter, args->entities); + entityTreeElement->getEntities(args->_cube, args->_foundEntities); return true; } return false; } // NOTE: assumes caller has handled locking -void EntityTree::evalEntitiesInCube(const AACube& cube, PickFilter searchFilter, QVector& foundEntities) { - FindEntitiesInCubeArgs args { cube, searchFilter, QVector() }; +void EntityTree::findEntities(const AACube& cube, QVector& foundEntities) { + FindEntitiesInCubeArgs args(cube); + // NOTE: This should use recursion, since this is a spatial operation recurseTreeWithOperation(findInCubeOperation, &args); - foundEntities.swap(args.entities); + // swap the two lists of entity pointers instead of copy + foundEntities.swap(args._foundEntities); } class FindEntitiesInBoxArgs { public: - // Inputs - AABox box; - PickFilter searchFilter; + FindEntitiesInBoxArgs(const AABox& box) + : _box(box), _foundEntities() { + } - // Outputs - QVector entities; + AABox _box; + QVector _foundEntities; }; -bool findInBoxOperation(const OctreeElementPointer& element, void* extraData) { +bool EntityTree::findInBoxOperation(const OctreeElementPointer& element, void* extraData) { FindEntitiesInBoxArgs* args = static_cast(extraData); - if (element->getAACube().touches(args->box)) { + if (element->getAACube().touches(args->_box)) { EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - entityTreeElement->evalEntitiesInBox(args->box, args->searchFilter, args->entities); + entityTreeElement->getEntities(args->_box, args->_foundEntities); return true; } return false; } // NOTE: assumes caller has handled locking -void EntityTree::evalEntitiesInBox(const AABox& box, PickFilter searchFilter, QVector& foundEntities) { - FindEntitiesInBoxArgs args { box, searchFilter, QVector() }; +void EntityTree::findEntities(const AABox& box, QVector& foundEntities) { + FindEntitiesInBoxArgs args(box); // NOTE: This should use recursion, since this is a spatial operation recurseTreeWithOperation(findInBoxOperation, &args); // swap the two lists of entity pointers instead of copy - foundEntities.swap(args.entities); + foundEntities.swap(args._foundEntities); } -class FindEntitiesInFrustumArgs { +class FindInFrustumArgs { public: - // Inputs ViewFrustum frustum; - PickFilter searchFilter; - - // Outputs - QVector entities; + QVector entities; }; -bool findInFrustumOperation(const OctreeElementPointer& element, void* extraData) { - FindEntitiesInFrustumArgs* args = static_cast(extraData); +bool EntityTree::findInFrustumOperation(const OctreeElementPointer& element, void* extraData) { + FindInFrustumArgs* args = static_cast(extraData); if (element->isInView(args->frustum)) { EntityTreeElementPointer entityTreeElement = std::static_pointer_cast(element); - entityTreeElement->evalEntitiesInFrustum(args->frustum, args->searchFilter, args->entities); + entityTreeElement->getEntities(args->frustum, args->entities); return true; } return false; } // NOTE: assumes caller has handled locking -void EntityTree::evalEntitiesInFrustum(const ViewFrustum& frustum, PickFilter searchFilter, QVector& foundEntities) { - FindEntitiesInFrustumArgs args = { frustum, searchFilter, QVector() }; +void EntityTree::findEntities(const ViewFrustum& frustum, QVector& foundEntities) { + FindInFrustumArgs args = { frustum, QVector() }; // NOTE: This should use recursion, since this is a spatial operation recurseTreeWithOperation(findInFrustumOperation, &args); // swap the two lists of entity pointers instead of copy foundEntities.swap(args.entities); } +// NOTE: assumes caller has handled locking +void EntityTree::findEntities(RecurseOctreeOperation& elementFilter, + QVector& foundEntities) { + recurseTreeWithOperation(elementFilter, nullptr); +} + EntityItemPointer EntityTree::findEntityByID(const QUuid& id) const { EntityItemID entityID(id); return findEntityByEntityItemID(entityID); diff --git a/libraries/entities/src/EntityTree.h b/libraries/entities/src/EntityTree.h index d08e21e4f0..d7f93b1eb2 100644 --- a/libraries/entities/src/EntityTree.h +++ b/libraries/entities/src/EntityTree.h @@ -45,6 +45,7 @@ public: QHash* map; }; + class EntityTree : public Octree, public SpatialParentTree { Q_OBJECT public: @@ -91,16 +92,18 @@ public: virtual void processChallengeOwnershipReplyPacket(ReceivedMessage& message, const SharedNodePointer& sourceNode) override; virtual void processChallengeOwnershipPacket(ReceivedMessage& message, const SharedNodePointer& sourceNode) override; - virtual EntityItemID evalRayIntersection(const glm::vec3& origin, const glm::vec3& direction, + virtual EntityItemID findRayIntersection(const glm::vec3& origin, const glm::vec3& direction, QVector entityIdsToInclude, QVector entityIdsToDiscard, - PickFilter searchFilter, OctreeElementPointer& element, float& distance, + bool visibleOnly, bool collidableOnly, bool precisionPicking, + OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo, Octree::lockType lockType = Octree::TryLock, bool* accurateResult = NULL); - virtual EntityItemID evalParabolaIntersection(const PickParabola& parabola, + virtual EntityItemID findParabolaIntersection(const PickParabola& parabola, QVector entityIdsToInclude, QVector entityIdsToDiscard, - PickFilter searchFilter, OctreeElementPointer& element, glm::vec3& intersection, - float& distance, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo, + bool visibleOnly, bool collidableOnly, bool precisionPicking, + OctreeElementPointer& element, glm::vec3& intersection, float& distance, float& parabolicDistance, + BoxFace& face, glm::vec3& surfaceNormal, QVariantMap& extraInfo, Octree::lockType lockType = Octree::TryLock, bool* accurateResult = NULL); virtual bool rootElementHasData() const override { return true; } @@ -125,19 +128,44 @@ public: void deleteEntity(const EntityItemID& entityID, bool force = false, bool ignoreWarnings = true); void deleteEntities(QSet entityIDs, bool force = false, bool ignoreWarnings = true); + /// \param position point of query in world-frame (meters) + /// \param targetRadius radius of query (meters) + EntityItemPointer findClosestEntity(const glm::vec3& position, float targetRadius); EntityItemPointer findEntityByID(const QUuid& id) const; EntityItemPointer findEntityByEntityItemID(const EntityItemID& entityID) const; virtual SpatiallyNestablePointer findByID(const QUuid& id) const override { return findEntityByID(id); } EntityItemID assignEntityID(const EntityItemID& entityItemID); /// Assigns a known ID for a creator token ID - QUuid evalClosestEntity(const glm::vec3& position, float targetRadius, PickFilter searchFilter); - void evalEntitiesInSphere(const glm::vec3& center, float radius, PickFilter searchFilter, QVector& foundEntities); - void evalEntitiesInSphereWithType(const glm::vec3& center, float radius, EntityTypes::EntityType type, PickFilter searchFilter, QVector& foundEntities); - void evalEntitiesInSphereWithName(const glm::vec3& center, float radius, const QString& name, bool caseSensitive, PickFilter searchFilter, QVector& foundEntities); - void evalEntitiesInCube(const AACube& cube, PickFilter searchFilter, QVector& foundEntities); - void evalEntitiesInBox(const AABox& box, PickFilter searchFilter, QVector& foundEntities); - void evalEntitiesInFrustum(const ViewFrustum& frustum, PickFilter searchFilter, QVector& foundEntities); + + /// finds all entities that touch a sphere + /// \param center the center of the sphere in world-frame (meters) + /// \param radius the radius of the sphere in world-frame (meters) + /// \param foundEntities[out] vector of EntityItemPointer + /// \remark Side effect: any initial contents in foundEntities will be lost + void findEntities(const glm::vec3& center, float radius, QVector& foundEntities); + + /// finds all entities that touch a cube + /// \param cube the query cube in world-frame (meters) + /// \param foundEntities[out] vector of non-EntityItemPointer + /// \remark Side effect: any initial contents in entities will be lost + void findEntities(const AACube& cube, QVector& foundEntities); + + /// finds all entities that touch a box + /// \param box the query box in world-frame (meters) + /// \param foundEntities[out] vector of non-EntityItemPointer + /// \remark Side effect: any initial contents in entities will be lost + void findEntities(const AABox& box, QVector& foundEntities); + + /// finds all entities within a frustum + /// \parameter frustum the query frustum + /// \param foundEntities[out] vector of EntityItemPointer + void findEntities(const ViewFrustum& frustum, QVector& foundEntities); + + /// finds all entities that match scanOperator + /// \parameter scanOperator function that scans entities that match criteria + /// \parameter foundEntities[out] vector of EntityItemPointer + void findEntities(RecurseOctreeOperation& scanOperator, QVector& foundEntities); void addNewlyCreatedHook(NewlyCreatedEntityHook* hook); void removeNewlyCreatedHook(NewlyCreatedEntityHook* hook); @@ -294,6 +322,11 @@ protected: void processRemovedEntities(const DeleteEntityOperator& theOperator); bool updateEntity(EntityItemPointer entity, const EntityItemProperties& properties, const SharedNodePointer& senderNode = SharedNodePointer(nullptr)); + static bool findNearPointOperation(const OctreeElementPointer& element, void* extraData); + static bool findInSphereOperation(const OctreeElementPointer& element, void* extraData); + static bool findInCubeOperation(const OctreeElementPointer& element, void* extraData); + static bool findInBoxOperation(const OctreeElementPointer& element, void* extraData); + static bool findInFrustumOperation(const OctreeElementPointer& element, void* extraData); static bool sendEntitiesOperation(const OctreeElementPointer& element, void* extraData); static void bumpTimestamp(EntityItemProperties& properties); diff --git a/libraries/entities/src/EntityTreeElement.cpp b/libraries/entities/src/EntityTreeElement.cpp index 2ece6835ea..5b71010f75 100644 --- a/libraries/entities/src/EntityTreeElement.cpp +++ b/libraries/entities/src/EntityTreeElement.cpp @@ -139,23 +139,10 @@ bool EntityTreeElement::bestFitBounds(const glm::vec3& minPoint, const glm::vec3 return false; } -bool checkFilterSettings(const EntityItemPointer& entity, PickFilter searchFilter) { - bool visible = entity->isVisible(); - bool collidable = !entity->getCollisionless() && (entity->getShapeType() != SHAPE_TYPE_NONE); - if ((!searchFilter.doesPickVisible() && visible) || (!searchFilter.doesPickInvisible() && !visible) || - (!searchFilter.doesPickCollidable() && collidable) || (!searchFilter.doesPickNonCollidable() && !collidable) || - (!searchFilter.doesPickDomainEntities() && entity->isDomainEntity()) || - (!searchFilter.doesPickAvatarEntities() && entity->isAvatarEntity()) || - (!searchFilter.doesPickLocalEntities() && entity->isLocalEntity())) { - return false; - } - return true; -} - -EntityItemID EntityTreeElement::evalRayIntersection(const glm::vec3& origin, const glm::vec3& direction, - OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, - const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, - PickFilter searchFilter, QVariantMap& extraInfo) { +EntityItemID EntityTreeElement::findRayIntersection(const glm::vec3& origin, const glm::vec3& direction, + OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, + const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, + bool visibleOnly, bool collidableOnly, QVariantMap& extraInfo, bool precisionPicking) { EntityItemID result; BoxFace localFace { UNKNOWN_FACE }; @@ -167,8 +154,9 @@ EntityItemID EntityTreeElement::evalRayIntersection(const glm::vec3& origin, con QVariantMap localExtraInfo; float distanceToElementDetails = distance; - EntityItemID entityID = evalDetailedRayIntersection(origin, direction, element, distanceToElementDetails, - localFace, localSurfaceNormal, entityIdsToInclude, entityIdsToDiscard, searchFilter, localExtraInfo); + EntityItemID entityID = findDetailedRayIntersection(origin, direction, element, distanceToElementDetails, + localFace, localSurfaceNormal, entityIdsToInclude, entityIdsToDiscard, visibleOnly, collidableOnly, + localExtraInfo, precisionPicking); if (!entityID.isNull() && distanceToElementDetails < distance) { distance = distanceToElementDetails; face = localFace; @@ -179,12 +167,13 @@ EntityItemID EntityTreeElement::evalRayIntersection(const glm::vec3& origin, con return result; } -EntityItemID EntityTreeElement::evalDetailedRayIntersection(const glm::vec3& origin, const glm::vec3& direction, +EntityItemID EntityTreeElement::findDetailedRayIntersection(const glm::vec3& origin, const glm::vec3& direction, OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, const QVector& entityIDsToDiscard, - PickFilter searchFilter, QVariantMap& extraInfo) { + bool visibleOnly, bool collidableOnly, QVariantMap& extraInfo, bool precisionPicking) { // only called if we do intersect our bounding cube, but find if we actually intersect with entities... + int entityNumber = 0; EntityItemID entityID; forEachEntity([&](EntityItemPointer entity) { // use simple line-sphere for broadphase check @@ -198,9 +187,11 @@ EntityItemID EntityTreeElement::evalDetailedRayIntersection(const glm::vec3& ori return; } - if (!checkFilterSettings(entity, searchFilter) || - (entityIdsToInclude.size() > 0 && !entityIdsToInclude.contains(entity->getID())) || - (entityIDsToDiscard.size() > 0 && entityIDsToDiscard.contains(entity->getID())) ) { + // check RayPick filter settings + if ((visibleOnly && !entity->isVisible()) + || (collidableOnly && (entity->getCollisionless() || entity->getShapeType() == SHAPE_TYPE_NONE)) + || (entityIdsToInclude.size() > 0 && !entityIdsToInclude.contains(entity->getID())) + || (entityIDsToDiscard.size() > 0 && entityIDsToDiscard.contains(entity->getID())) ) { return; } @@ -231,7 +222,7 @@ EntityItemID EntityTreeElement::evalDetailedRayIntersection(const glm::vec3& ori if (entity->supportsDetailedIntersection()) { QVariantMap localExtraInfo; if (entity->findDetailedRayIntersection(origin, direction, element, localDistance, - localFace, localSurfaceNormal, localExtraInfo, searchFilter.isPrecise())) { + localFace, localSurfaceNormal, localExtraInfo, precisionPicking)) { if (localDistance < distance) { distance = localDistance; face = localFace; @@ -253,6 +244,7 @@ EntityItemID EntityTreeElement::evalDetailedRayIntersection(const glm::vec3& ori } } } + entityNumber++; }); return entityID; } @@ -283,10 +275,11 @@ bool EntityTreeElement::findSpherePenetration(const glm::vec3& center, float rad return result; } -EntityItemID EntityTreeElement::evalParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, +EntityItemID EntityTreeElement::findParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, const glm::vec3& acceleration, OctreeElementPointer& element, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, - const QVector& entityIdsToDiscard, PickFilter searchFilter, QVariantMap& extraInfo) { + const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly, + QVariantMap& extraInfo, bool precisionPicking) { EntityItemID result; BoxFace localFace; @@ -307,8 +300,9 @@ EntityItemID EntityTreeElement::evalParabolaIntersection(const glm::vec3& origin } // Get the normal of the plane, the cross product of two vectors on the plane glm::vec3 normal = glm::normalize(glm::cross(vectorOnPlane, acceleration)); - EntityItemID entityID = evalDetailedParabolaIntersection(origin, velocity, acceleration, normal, element, distanceToElementDetails, - localFace, localSurfaceNormal, entityIdsToInclude, entityIdsToDiscard, searchFilter, localExtraInfo); + EntityItemID entityID = findDetailedParabolaIntersection(origin, velocity, acceleration, normal, element, distanceToElementDetails, + localFace, localSurfaceNormal, entityIdsToInclude, entityIdsToDiscard, visibleOnly, collidableOnly, + localExtraInfo, precisionPicking); if (!entityID.isNull() && distanceToElementDetails < parabolicDistance) { parabolicDistance = distanceToElementDetails; face = localFace; @@ -319,12 +313,13 @@ EntityItemID EntityTreeElement::evalParabolaIntersection(const glm::vec3& origin return result; } -EntityItemID EntityTreeElement::evalDetailedParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, const glm::vec3& acceleration, +EntityItemID EntityTreeElement::findDetailedParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, const glm::vec3& acceleration, const glm::vec3& normal, OctreeElementPointer& element, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, const QVector& entityIDsToDiscard, - PickFilter searchFilter, QVariantMap& extraInfo) { + bool visibleOnly, bool collidableOnly, QVariantMap& extraInfo, bool precisionPicking) { // only called if we do intersect our bounding cube, but find if we actually intersect with entities... + int entityNumber = 0; EntityItemID entityID; forEachEntity([&](EntityItemPointer entity) { // use simple line-sphere for broadphase check @@ -343,9 +338,11 @@ EntityItemID EntityTreeElement::evalDetailedParabolaIntersection(const glm::vec3 return; } - if (!checkFilterSettings(entity, searchFilter) || - (entityIdsToInclude.size() > 0 && !entityIdsToInclude.contains(entity->getID())) || - (entityIDsToDiscard.size() > 0 && entityIDsToDiscard.contains(entity->getID()))) { + // check RayPick filter settings + if ((visibleOnly && !entity->isVisible()) + || (collidableOnly && (entity->getCollisionless() || entity->getShapeType() == SHAPE_TYPE_NONE)) + || (entityIdsToInclude.size() > 0 && !entityIdsToInclude.contains(entity->getID())) + || (entityIDsToDiscard.size() > 0 && entityIDsToDiscard.contains(entity->getID())) ) { return; } @@ -377,7 +374,7 @@ EntityItemID EntityTreeElement::evalDetailedParabolaIntersection(const glm::vec3 if (entity->supportsDetailedIntersection()) { QVariantMap localExtraInfo; if (entity->findDetailedParabolaIntersection(origin, velocity, acceleration, element, localDistance, - localFace, localSurfaceNormal, localExtraInfo, searchFilter.isPrecise())) { + localFace, localSurfaceNormal, localExtraInfo, precisionPicking)) { if (localDistance < parabolicDistance) { parabolicDistance = localDistance; face = localFace; @@ -399,56 +396,55 @@ EntityItemID EntityTreeElement::evalDetailedParabolaIntersection(const glm::vec3 } } } + entityNumber++; }); return entityID; } -QUuid EntityTreeElement::evalClosetEntity(const glm::vec3& position, PickFilter searchFilter, float& closestDistanceSquared) const { - QUuid closestEntity; - forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter)) { - return; - } - - float distanceToEntity = glm::distance2(position, entity->getWorldPosition()); - if (distanceToEntity < closestDistanceSquared) { - closestEntity = entity->getID(); - closestDistanceSquared = distanceToEntity; +EntityItemPointer EntityTreeElement::getClosestEntity(glm::vec3 position) const { + EntityItemPointer closestEntity = NULL; + float closestEntityDistance = FLT_MAX; + withReadLock([&] { + foreach(EntityItemPointer entity, _entityItems) { + float distanceToEntity = glm::distance2(position, entity->getWorldPosition()); + if (distanceToEntity < closestEntityDistance) { + closestEntity = entity; + } } }); return closestEntity; } -void EntityTreeElement::evalEntitiesInSphere(const glm::vec3& position, float radius, PickFilter searchFilter, QVector& foundEntities) const { +// TODO: change this to use better bounding shape for entity than sphere +void EntityTreeElement::getEntities(const glm::vec3& searchPosition, float searchRadius, QVector& foundEntities) const { forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter)) { - return; - } bool success; AABox entityBox = entity->getAABox(success); // if the sphere doesn't intersect with our world frame AABox, we don't need to consider the more complex case glm::vec3 penetration; - if (success && entityBox.findSpherePenetration(position, radius, penetration)) { + if (!success || entityBox.findSpherePenetration(searchPosition, searchRadius, penetration)) { glm::vec3 dimensions = entity->getRaycastDimensions(); // FIXME - consider allowing the entity to determine penetration so that - // entities could presumably do actual hull testing if they wanted to + // entities could presumably dull actuall hull testing if they wanted to // FIXME - handle entity->getShapeType() == SHAPE_TYPE_SPHERE case better in particular // can we handle the ellipsoid case better? We only currently handle perfect spheres // with centered registration points - if (entity->getShapeType() == SHAPE_TYPE_SPHERE && (dimensions.x == dimensions.y && dimensions.y == dimensions.z)) { + if (entity->getShapeType() == SHAPE_TYPE_SPHERE && + (dimensions.x == dimensions.y && dimensions.y == dimensions.z)) { // NOTE: entity->getRadius() doesn't return the true radius, it returns the radius of the // maximum bounding sphere, which is actually larger than our actual radius float entityTrueRadius = dimensions.x / 2.0f; bool success; - if (findSphereSpherePenetration(position, radius, entity->getCenterPosition(success), entityTrueRadius, penetration)) { + if (findSphereSpherePenetration(searchPosition, searchRadius, + entity->getCenterPosition(success), entityTrueRadius, penetration)) { if (success) { - foundEntities.push_back(entity->getID()); + foundEntities.push_back(entity); } } } else { @@ -464,134 +460,17 @@ void EntityTreeElement::evalEntitiesInSphere(const glm::vec3& position, float ra AABox entityFrameBox(corner, dimensions); - glm::vec3 entityFrameSearchPosition = glm::vec3(worldToEntityMatrix * glm::vec4(position, 1.0f)); - if (entityFrameBox.findSpherePenetration(entityFrameSearchPosition, radius, penetration)) { - foundEntities.push_back(entity->getID()); + glm::vec3 entityFrameSearchPosition = glm::vec3(worldToEntityMatrix * glm::vec4(searchPosition, 1.0f)); + if (entityFrameBox.findSpherePenetration(entityFrameSearchPosition, searchRadius, penetration)) { + foundEntities.push_back(entity); } } } }); } -void EntityTreeElement::evalEntitiesInSphereWithType(const glm::vec3& position, float radius, EntityTypes::EntityType type, PickFilter searchFilter, QVector& foundEntities) const { +void EntityTreeElement::getEntities(const AACube& cube, QVector& foundEntities) { forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter) || type != entity->getType()) { - return; - } - - bool success; - AABox entityBox = entity->getAABox(success); - - // if the sphere doesn't intersect with our world frame AABox, we don't need to consider the more complex case - glm::vec3 penetration; - if (success && entityBox.findSpherePenetration(position, radius, penetration)) { - - glm::vec3 dimensions = entity->getRaycastDimensions(); - - // FIXME - consider allowing the entity to determine penetration so that - // entities could presumably do actual hull testing if they wanted to - // FIXME - handle entity->getShapeType() == SHAPE_TYPE_SPHERE case better in particular - // can we handle the ellipsoid case better? We only currently handle perfect spheres - // with centered registration points - if (entity->getShapeType() == SHAPE_TYPE_SPHERE && (dimensions.x == dimensions.y && dimensions.y == dimensions.z)) { - - // NOTE: entity->getRadius() doesn't return the true radius, it returns the radius of the - // maximum bounding sphere, which is actually larger than our actual radius - float entityTrueRadius = dimensions.x / 2.0f; - - bool success; - if (findSphereSpherePenetration(position, radius, entity->getCenterPosition(success), entityTrueRadius, penetration)) { - if (success) { - foundEntities.push_back(entity->getID()); - } - } - } else { - // determine the worldToEntityMatrix that doesn't include scale because - // we're going to use the registration aware aa box in the entity frame - glm::mat4 rotation = glm::mat4_cast(entity->getWorldOrientation()); - glm::mat4 translation = glm::translate(entity->getWorldPosition()); - glm::mat4 entityToWorldMatrix = translation * rotation; - glm::mat4 worldToEntityMatrix = glm::inverse(entityToWorldMatrix); - - glm::vec3 registrationPoint = entity->getRegistrationPoint(); - glm::vec3 corner = -(dimensions * registrationPoint); - - AABox entityFrameBox(corner, dimensions); - - glm::vec3 entityFrameSearchPosition = glm::vec3(worldToEntityMatrix * glm::vec4(position, 1.0f)); - if (entityFrameBox.findSpherePenetration(entityFrameSearchPosition, radius, penetration)) { - foundEntities.push_back(entity->getID()); - } - } - } - }); -} - -void EntityTreeElement::evalEntitiesInSphereWithName(const glm::vec3& position, float radius, const QString& name, bool caseSensitive, PickFilter searchFilter, QVector& foundEntities) const { - forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter)) { - return; - } - - QString entityName = entity->getName(); - if ((caseSensitive && name != entityName) || (!caseSensitive && name.toLower() != entityName.toLower())) { - return; - } - - bool success; - AABox entityBox = entity->getAABox(success); - - // if the sphere doesn't intersect with our world frame AABox, we don't need to consider the more complex case - glm::vec3 penetration; - if (success && entityBox.findSpherePenetration(position, radius, penetration)) { - - glm::vec3 dimensions = entity->getRaycastDimensions(); - - // FIXME - consider allowing the entity to determine penetration so that - // entities could presumably do actual hull testing if they wanted to - // FIXME - handle entity->getShapeType() == SHAPE_TYPE_SPHERE case better in particular - // can we handle the ellipsoid case better? We only currently handle perfect spheres - // with centered registration points - if (entity->getShapeType() == SHAPE_TYPE_SPHERE && (dimensions.x == dimensions.y && dimensions.y == dimensions.z)) { - - // NOTE: entity->getRadius() doesn't return the true radius, it returns the radius of the - // maximum bounding sphere, which is actually larger than our actual radius - float entityTrueRadius = dimensions.x / 2.0f; - - bool success; - if (findSphereSpherePenetration(position, radius, entity->getCenterPosition(success), entityTrueRadius, penetration)) { - if (success) { - foundEntities.push_back(entity->getID()); - } - } - } else { - // determine the worldToEntityMatrix that doesn't include scale because - // we're going to use the registration aware aa box in the entity frame - glm::mat4 rotation = glm::mat4_cast(entity->getWorldOrientation()); - glm::mat4 translation = glm::translate(entity->getWorldPosition()); - glm::mat4 entityToWorldMatrix = translation * rotation; - glm::mat4 worldToEntityMatrix = glm::inverse(entityToWorldMatrix); - - glm::vec3 registrationPoint = entity->getRegistrationPoint(); - glm::vec3 corner = -(dimensions * registrationPoint); - - AABox entityFrameBox(corner, dimensions); - - glm::vec3 entityFrameSearchPosition = glm::vec3(worldToEntityMatrix * glm::vec4(position, 1.0f)); - if (entityFrameBox.findSpherePenetration(entityFrameSearchPosition, radius, penetration)) { - foundEntities.push_back(entity->getID()); - } - } - } - }); -} - -void EntityTreeElement::evalEntitiesInCube(const AACube& cube, PickFilter searchFilter, QVector& foundEntities) const { - forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter)) { - return; - } - bool success; AABox entityBox = entity->getAABox(success); // FIXME - handle entity->getShapeType() == SHAPE_TYPE_SPHERE case better @@ -610,18 +489,14 @@ void EntityTreeElement::evalEntitiesInCube(const AACube& cube, PickFilter search // // If the entities AABox touches the search cube then consider it to be found - if (success && entityBox.touches(cube)) { - foundEntities.push_back(entity->getID()); + if (!success || entityBox.touches(cube)) { + foundEntities.push_back(entity); } }); } -void EntityTreeElement::evalEntitiesInBox(const AABox& box, PickFilter searchFilter, QVector& foundEntities) const { +void EntityTreeElement::getEntities(const AABox& box, QVector& foundEntities) { forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter)) { - return; - } - bool success; AABox entityBox = entity->getAABox(success); // FIXME - handle entity->getShapeType() == SHAPE_TYPE_SPHERE case better @@ -640,24 +515,19 @@ void EntityTreeElement::evalEntitiesInBox(const AABox& box, PickFilter searchFil // // If the entities AABox touches the search cube then consider it to be found - if (success && entityBox.touches(box)) { - foundEntities.push_back(entity->getID()); + if (!success || entityBox.touches(box)) { + foundEntities.push_back(entity); } }); } -void EntityTreeElement::evalEntitiesInFrustum(const ViewFrustum& frustum, PickFilter searchFilter, QVector& foundEntities) const { +void EntityTreeElement::getEntities(const ViewFrustum& frustum, QVector& foundEntities) { forEachEntity([&](EntityItemPointer entity) { - if (!checkFilterSettings(entity, searchFilter)) { - return; - } - bool success; AABox entityBox = entity->getAABox(success); - // FIXME - See FIXMEs for similar methods above. - if (success && (frustum.boxIntersectsFrustum(entityBox) || frustum.boxIntersectsKeyhole(entityBox))) { - foundEntities.push_back(entity->getID()); + if (!success || frustum.boxIntersectsFrustum(entityBox) || frustum.boxIntersectsKeyhole(entityBox)) { + foundEntities.push_back(entity); } }); } diff --git a/libraries/entities/src/EntityTreeElement.h b/libraries/entities/src/EntityTreeElement.h index aed19eed15..793340c9a4 100644 --- a/libraries/entities/src/EntityTreeElement.h +++ b/libraries/entities/src/EntityTreeElement.h @@ -19,8 +19,7 @@ #include "EntityEditPacketSender.h" #include "EntityItem.h" - -#include +#include "EntityTree.h" class EntityTree; class EntityTreeElement; @@ -78,6 +77,7 @@ public: EntityEditPacketSender* packetSender; }; + class EntityTreeElement : public OctreeElement, ReadWriteLockable { friend class EntityTree; // to allow createElement to new us... @@ -135,25 +135,28 @@ public: virtual bool deleteApproved() const override { return !hasEntities(); } virtual bool canPickIntersect() const override { return hasEntities(); } - virtual EntityItemID evalRayIntersection(const glm::vec3& origin, const glm::vec3& direction, + virtual EntityItemID findRayIntersection(const glm::vec3& origin, const glm::vec3& direction, OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, const QVector& entityIdsToDiscard, - PickFilter searchFilter, QVariantMap& extraInfo); - virtual EntityItemID evalDetailedRayIntersection(const glm::vec3& origin, const glm::vec3& direction, + bool visibleOnly, bool collidableOnly, QVariantMap& extraInfo, bool precisionPicking = false); + virtual EntityItemID findDetailedRayIntersection(const glm::vec3& origin, const glm::vec3& direction, OctreeElementPointer& element, float& distance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, - const QVector& entityIdsToDiscard, PickFilter searchFilter, QVariantMap& extraInfo); + const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly, + QVariantMap& extraInfo, bool precisionPicking); virtual bool findSpherePenetration(const glm::vec3& center, float radius, glm::vec3& penetration, void** penetratedObject) const override; - virtual EntityItemID evalParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, + virtual EntityItemID findParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, const glm::vec3& acceleration, OctreeElementPointer& element, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, - const QVector& entityIdsToDiscard, PickFilter searchFilter, QVariantMap& extraInfo); - virtual EntityItemID evalDetailedParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, + const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly, + QVariantMap& extraInfo, bool precisionPicking = false); + virtual EntityItemID findDetailedParabolaIntersection(const glm::vec3& origin, const glm::vec3& velocity, const glm::vec3& normal, const glm::vec3& acceleration, OctreeElementPointer& element, float& parabolicDistance, BoxFace& face, glm::vec3& surfaceNormal, const QVector& entityIdsToInclude, - const QVector& entityIdsToDiscard, PickFilter searchFilter, QVariantMap& extraInfo); + const QVector& entityIdsToDiscard, bool visibleOnly, bool collidableOnly, + QVariantMap& extraInfo, bool precisionPicking); template void forEachEntity(F f) const { @@ -172,13 +175,28 @@ public: void addEntityItem(EntityItemPointer entity); - QUuid evalClosetEntity(const glm::vec3& position, PickFilter searchFilter, float& closestDistanceSquared) const; - void evalEntitiesInSphere(const glm::vec3& position, float radius, PickFilter searchFilter, QVector& foundEntities) const; - void evalEntitiesInSphereWithType(const glm::vec3& position, float radius, EntityTypes::EntityType type, PickFilter searchFilter, QVector& foundEntities) const; - void evalEntitiesInSphereWithName(const glm::vec3& position, float radius, const QString& name, bool caseSensitive, PickFilter searchFilter, QVector& foundEntities) const; - void evalEntitiesInCube(const AACube& cube, PickFilter searchFilter, QVector& foundEntities) const; - void evalEntitiesInBox(const AABox& box, PickFilter searchFilter, QVector& foundEntities) const; - void evalEntitiesInFrustum(const ViewFrustum& frustum, PickFilter searchFilter, QVector& foundEntities) const; + EntityItemPointer getClosestEntity(glm::vec3 position) const; + + /// finds all entities that touch a sphere + /// \param position the center of the query sphere + /// \param radius the radius of the query sphere + /// \param entities[out] vector of const EntityItemPointer + void getEntities(const glm::vec3& position, float radius, QVector& foundEntities) const; + + /// finds all entities that touch a box + /// \param box the query box + /// \param entities[out] vector of non-const EntityItemPointer + void getEntities(const AACube& cube, QVector& foundEntities); + + /// finds all entities that touch a box + /// \param box the query box + /// \param entities[out] vector of non-const EntityItemPointer + void getEntities(const AABox& box, QVector& foundEntities); + + /// finds all entities that touch a frustum + /// \param frustum the query frustum + /// \param entities[out] vector of non-const EntityItemPointer + void getEntities(const ViewFrustum& frustum, QVector& foundEntities); /// finds all entities that match filter /// \param filter function that adds matching entities to foundEntities diff --git a/libraries/pointers/src/Pick.cpp b/libraries/pointers/src/Pick.cpp index e95721c04b..7ac53a2643 100644 --- a/libraries/pointers/src/Pick.cpp +++ b/libraries/pointers/src/Pick.cpp @@ -7,6 +7,8 @@ // #include "Pick.h" +const PickFilter PickFilter::NOTHING; + int pickTypeMetaTypeId = qRegisterMetaType("PickType"); PickQuery::PickQuery(const PickFilter& filter, const float maxDistance, const bool enabled) : @@ -39,8 +41,7 @@ bool PickQuery::isEnabled() const { void PickQuery::setPrecisionPicking(bool precisionPicking) { withWriteLock([&] { - _filter.setFlag(PickFilter::PRECISE, precisionPicking); - _filter.setFlag(PickFilter::COARSE, !precisionPicking); + _filter.setFlag(PickFilter::PICK_COARSE, !precisionPicking); }); } diff --git a/libraries/pointers/src/Pick.h b/libraries/pointers/src/Pick.h index 857a72caa8..9ca0f14c5f 100644 --- a/libraries/pointers/src/Pick.h +++ b/libraries/pointers/src/Pick.h @@ -18,7 +18,6 @@ #include #include -#include enum IntersectionType { NONE = 0, @@ -28,6 +27,84 @@ enum IntersectionType { HUD }; +class PickFilter { +public: + enum FlagBit { + PICK_ENTITIES = 0, + PICK_OVERLAYS, + PICK_AVATARS, + PICK_HUD, + + PICK_COARSE, // if not set, does precise intersection, otherwise, doesn't + + PICK_INCLUDE_INVISIBLE, // if not set, will not intersect invisible elements, otherwise, intersects both visible and invisible elements + PICK_INCLUDE_NONCOLLIDABLE, // if not set, will not intersect noncollidable elements, otherwise, intersects both collidable and noncollidable elements + + // NOT YET IMPLEMENTED + PICK_ALL_INTERSECTIONS, // if not set, returns closest intersection, otherwise, returns list of all intersections + + NUM_FLAGS, // Not a valid flag + }; + typedef std::bitset Flags; + + // The key is the Flags + Flags _flags; + + PickFilter() {} + PickFilter(const Flags& flags) : _flags(flags) {} + + bool operator== (const PickFilter& rhs) const { return _flags == rhs._flags; } + bool operator!= (const PickFilter& rhs) const { return _flags != rhs._flags; } + + void setFlag(FlagBit flag, bool value) { _flags[flag] = value; } + + bool doesPickNothing() const { return _flags == NOTHING._flags; } + bool doesPickEntities() const { return _flags[PICK_ENTITIES]; } + bool doesPickOverlays() const { return _flags[PICK_OVERLAYS]; } + bool doesPickAvatars() const { return _flags[PICK_AVATARS]; } + bool doesPickHUD() const { return _flags[PICK_HUD]; } + + bool doesPickCoarse() const { return _flags[PICK_COARSE]; } + bool doesPickInvisible() const { return _flags[PICK_INCLUDE_INVISIBLE]; } + bool doesPickNonCollidable() const { return _flags[PICK_INCLUDE_NONCOLLIDABLE]; } + + bool doesWantAllIntersections() const { return _flags[PICK_ALL_INTERSECTIONS]; } + + // Helpers for RayPickManager + Flags getEntityFlags() const { + unsigned int toReturn = getBitMask(PICK_ENTITIES); + if (doesPickInvisible()) { + toReturn |= getBitMask(PICK_INCLUDE_INVISIBLE); + } + if (doesPickNonCollidable()) { + toReturn |= getBitMask(PICK_INCLUDE_NONCOLLIDABLE); + } + if (doesPickCoarse()) { + toReturn |= getBitMask(PICK_COARSE); + } + return Flags(toReturn); + } + Flags getOverlayFlags() const { + unsigned int toReturn = getBitMask(PICK_OVERLAYS); + if (doesPickInvisible()) { + toReturn |= getBitMask(PICK_INCLUDE_INVISIBLE); + } + if (doesPickNonCollidable()) { + toReturn |= getBitMask(PICK_INCLUDE_NONCOLLIDABLE); + } + if (doesPickCoarse()) { + toReturn |= getBitMask(PICK_COARSE); + } + return Flags(toReturn); + } + Flags getAvatarFlags() const { return Flags(getBitMask(PICK_AVATARS)); } + Flags getHUDFlags() const { return Flags(getBitMask(PICK_HUD)); } + + static constexpr unsigned int getBitMask(FlagBit bit) { return 1 << bit; } + + static const PickFilter NOTHING; +}; + class PickResult { public: PickResult() {} diff --git a/libraries/pointers/src/PickCacheOptimizer.h b/libraries/pointers/src/PickCacheOptimizer.h index e91283f02c..c74e84937b 100644 --- a/libraries/pointers/src/PickCacheOptimizer.h +++ b/libraries/pointers/src/PickCacheOptimizer.h @@ -85,10 +85,10 @@ QVector4D PickCacheOptimizer::update(std::unordered_mapgetMathematicalPick(); PickResultPointer res = pick->getDefaultResult(mathematicalPick.toVariantMap()); - if (!pick->isEnabled() || pick->getMaxDistance() < 0.0f || !mathematicalPick) { + if (!pick->isEnabled() || pick->getFilter().doesPickNothing() || pick->getMaxDistance() < 0.0f || !mathematicalPick) { pick->setPickResult(res); } else { - if (pick->getFilter().doesPickDomainEntities() || pick->getFilter().doesPickAvatarEntities()) { + if (pick->getFilter().doesPickEntities()) { PickCacheKey entityKey = { pick->getFilter().getEntityFlags(), pick->getIncludeItems(), pick->getIgnoreItems() }; if (!checkAndCompareCachedResults(mathematicalPick, results, res, entityKey)) { PickResultPointer entityRes = pick->getEntityIntersection(mathematicalPick); @@ -99,7 +99,7 @@ QVector4D PickCacheOptimizer::update(std::unordered_mapgetFilter().doesPickLocalEntities()) { + if (pick->getFilter().doesPickOverlays()) { PickCacheKey overlayKey = { pick->getFilter().getOverlayFlags(), pick->getIncludeItems(), pick->getIgnoreItems() }; if (!checkAndCompareCachedResults(mathematicalPick, results, res, overlayKey)) { PickResultPointer overlayRes = pick->getOverlayIntersection(mathematicalPick); diff --git a/libraries/shared/src/PickFilter.h b/libraries/shared/src/PickFilter.h deleted file mode 100644 index 96fbadf117..0000000000 --- a/libraries/shared/src/PickFilter.h +++ /dev/null @@ -1,99 +0,0 @@ -// -// Created by Sam Gondelman on 12/7/18. -// Copyright 2018 High Fidelity, Inc. -// -// Distributed under the Apache License, Version 2.0. -// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html -// - -#ifndef hifi_PickFilter_h -#define hifi_PickFilter_h - -#include - -class PickFilter { -public: - enum FlagBit { - DOMAIN_ENTITIES = 0, - AVATAR_ENTITIES, - LOCAL_ENTITIES, - AVATARS, - HUD, - - VISIBLE, - INVISIBLE, - - COLLIDABLE, - NONCOLLIDABLE, - - PRECISE, - COARSE, - - // NOT YET IMPLEMENTED - PICK_ALL_INTERSECTIONS, // if not set, returns closest intersection, otherwise, returns list of all intersections - - NUM_FLAGS, // Not a valid flag - }; - typedef std::bitset Flags; - - // The key is the Flags - Flags _flags; - - PickFilter() {} - PickFilter(const Flags& flags) : _flags(flags) {} - - bool operator==(const PickFilter& rhs) const { return _flags == rhs._flags; } - bool operator!=(const PickFilter& rhs) const { return _flags != rhs._flags; } - - void setFlag(FlagBit flag, bool value) { _flags[flag] = value; } - - // There are different groups of related flags. If none of the flags in a group are set, the search filter includes them all. - bool doesPickDomainEntities() const { return _flags[DOMAIN_ENTITIES] || !(_flags[AVATAR_ENTITIES] || _flags[LOCAL_ENTITIES] || _flags[AVATARS] || _flags[HUD]); } - bool doesPickAvatarEntities() const { return _flags[AVATAR_ENTITIES] || !(_flags[DOMAIN_ENTITIES] || _flags[LOCAL_ENTITIES] || _flags[AVATARS] || _flags[HUD]); } - bool doesPickLocalEntities() const { return _flags[LOCAL_ENTITIES] || !(_flags[DOMAIN_ENTITIES] || _flags[AVATAR_ENTITIES] || _flags[AVATARS] || _flags[HUD]); } - bool doesPickAvatars() const { return _flags[AVATARS] || !(_flags[DOMAIN_ENTITIES] || _flags[AVATAR_ENTITIES] || _flags[LOCAL_ENTITIES] || _flags[HUD]); } - bool doesPickHUD() const { return _flags[HUD] || !(_flags[DOMAIN_ENTITIES] || _flags[AVATAR_ENTITIES] || _flags[LOCAL_ENTITIES] || _flags[AVATARS]); } - - bool doesPickVisible() const { return _flags[VISIBLE] || !_flags[INVISIBLE]; } - bool doesPickInvisible() const { return _flags[INVISIBLE] || !_flags[VISIBLE]; } - - bool doesPickCollidable() const { return _flags[COLLIDABLE] || !_flags[NONCOLLIDABLE]; } - bool doesPickNonCollidable() const { return _flags[NONCOLLIDABLE] || !_flags[COLLIDABLE]; } - - bool isPrecise() const { return _flags[PRECISE] || !_flags[COARSE]; } - bool isCoarse() const { return _flags[COARSE] || !_flags[PRECISE]; } - - bool doesWantAllIntersections() const { return _flags[PICK_ALL_INTERSECTIONS]; } - - // Helpers for RayPickManager - Flags getEntityFlags() const { - unsigned int toReturn = 0; - for (int i = DOMAIN_ENTITIES; i < LOCAL_ENTITIES; i++) { - if (_flags[i]) { - toReturn |= getBitMask(FlagBit(i)); - } - } - for (int i = HUD + 1; i < NUM_FLAGS; i++) { - if (_flags[i]) { - toReturn |= getBitMask(FlagBit(i)); - } - } - return Flags(toReturn); - } - Flags getOverlayFlags() const { - unsigned int toReturn = getBitMask(LOCAL_ENTITIES); - for (int i = HUD + 1; i < NUM_FLAGS; i++) { - if (_flags[i]) { - toReturn |= getBitMask(FlagBit(i)); - } - } - return Flags(toReturn); - } - Flags getAvatarFlags() const { return Flags(getBitMask(AVATARS)); } - Flags getHUDFlags() const { return Flags(getBitMask(HUD)); } - - static constexpr unsigned int getBitMask(FlagBit bit) { return 1 << bit; } -}; - -#endif // hifi_PickFilter_h - diff --git a/scripts/developer/utilities/tests/entityPerfTest.js b/scripts/developer/utilities/tests/entityPerfTest.js index d3846baba9..181afcf360 100644 --- a/scripts/developer/utilities/tests/entityPerfTest.js +++ b/scripts/developer/utilities/tests/entityPerfTest.js @@ -87,6 +87,14 @@ function makeEntity () { .withRay(function(){ Entities.findRayIntersection(this.ray, false); }); + entityTests.addTestCase('findRayIntersectionBlocking, precisionPicking=true') + .withRay(function(){ + Entities.findRayIntersectionBlocking(this.ray, true); + }) + entityTests.addTestCase('findRayIntersectionBlocking, precisionPicking=false') + .withRay(function(){ + Entities.findRayIntersectionBlocking(this.ray, false); + }) entityTests.addTestCase('no-op') .run(function(){}); diff --git a/unpublishedScripts/DomainContent/Toybox/pistol/pistol.js b/unpublishedScripts/DomainContent/Toybox/pistol/pistol.js index 038c53054c..b408e4f464 100644 --- a/unpublishedScripts/DomainContent/Toybox/pistol/pistol.js +++ b/unpublishedScripts/DomainContent/Toybox/pistol/pistol.js @@ -137,7 +137,7 @@ direction: this.firingDirection }; this.createGunFireEffect(this.barrelPoint) - var intersection = Entities.findRayIntersection(pickRay, true); + var intersection = Entities.findRayIntersectionBlocking(pickRay, true); if (intersection.intersects) { this.createEntityHitEffect(intersection.intersection); if (Math.random() < this.playRichochetSoundChance) {