mirror of
https://github.com/HifiExperiments/overte.git
synced 2025-08-06 07:23:08 +02:00
253 lines
No EOL
10 KiB
C++
253 lines
No EOL
10 KiB
C++
//
|
|
// RayPickManager.cpp
|
|
// interface/src/raypick
|
|
//
|
|
// Created by Sam Gondelman 7/11/2017
|
|
// Copyright 2017 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
|
|
//
|
|
#include "RayPickManager.h"
|
|
|
|
#include "Application.h"
|
|
#include "EntityScriptingInterface.h"
|
|
#include "ui/overlays/Overlays.h"
|
|
#include "avatar/AvatarManager.h"
|
|
#include "scripting/HMDScriptingInterface.h"
|
|
#include "DependencyManager.h"
|
|
|
|
#include "JointRayPick.h"
|
|
#include "StaticRayPick.h"
|
|
#include "MouseRayPick.h"
|
|
|
|
bool RayPickManager::checkAndCompareCachedResults(QPair<glm::vec3, glm::vec3>& ray, RayPickCache& cache, RayPickResult& res, const RayPickFilter::Flags& mask) {
|
|
if (cache.contains(ray) && cache[ray].find(mask) != cache[ray].end()) {
|
|
if (cache[ray][mask].distance < res.distance) {
|
|
res = cache[ray][mask];
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void RayPickManager::cacheResult(const bool intersects, const RayPickResult& resTemp, const RayPickFilter::Flags& mask, RayPickResult& res, QPair<glm::vec3, glm::vec3>& ray, RayPickCache& cache) {
|
|
if (intersects) {
|
|
cache[ray][mask] = resTemp;
|
|
if (resTemp.distance < res.distance) {
|
|
res = resTemp;
|
|
}
|
|
} else {
|
|
cache[ray][mask] = RayPickResult();
|
|
}
|
|
}
|
|
|
|
void RayPickManager::update() {
|
|
RayPickCache results;
|
|
for (auto& uid : _rayPicks.keys()) {
|
|
std::shared_ptr<RayPick> rayPick = _rayPicks[uid];
|
|
if (!rayPick->isEnabled() || rayPick->getFilter().doesPickNothing() || rayPick->getMaxDistance() < 0.0f) {
|
|
continue;
|
|
}
|
|
|
|
bool valid;
|
|
PickRay ray = rayPick->getPickRay(valid);
|
|
|
|
if (!valid) {
|
|
continue;
|
|
}
|
|
|
|
QPair<glm::vec3, glm::vec3> rayKey = QPair<glm::vec3, glm::vec3>(ray.origin, ray.direction);
|
|
RayPickResult res;
|
|
|
|
if (rayPick->getFilter().doesPickEntities()) {
|
|
RayToEntityIntersectionResult entityRes;
|
|
bool fromCache = true;
|
|
bool invisible = rayPick->getFilter().doesPickInvisible();
|
|
bool noncollidable = rayPick->getFilter().doesPickNonCollidable();
|
|
RayPickFilter::Flags entityMask = rayPick->getFilter().getEntityFlags();
|
|
if (!checkAndCompareCachedResults(rayKey, results, res, entityMask)) {
|
|
entityRes = DependencyManager::get<EntityScriptingInterface>()->findRayIntersection(ray, true, rayPick->getIncludeEntites(), rayPick->getIgnoreEntites(), !invisible, !noncollidable);
|
|
fromCache = false;
|
|
}
|
|
|
|
if (!fromCache) {
|
|
cacheResult(entityRes.intersects, RayPickResult(IntersectionType::ENTITY, entityRes.entityID, entityRes.distance, entityRes.intersection, entityRes.surfaceNormal),
|
|
entityMask, res, rayKey, results);
|
|
}
|
|
}
|
|
|
|
if (rayPick->getFilter().doesPickOverlays()) {
|
|
RayToOverlayIntersectionResult overlayRes;
|
|
bool fromCache = true;
|
|
bool invisible = rayPick->getFilter().doesPickInvisible();
|
|
bool noncollidable = rayPick->getFilter().doesPickNonCollidable();
|
|
RayPickFilter::Flags overlayMask = rayPick->getFilter().getOverlayFlags();
|
|
if (!checkAndCompareCachedResults(rayKey, results, res, overlayMask)) {
|
|
overlayRes = qApp->getOverlays().findRayIntersection(ray, true, rayPick->getIncludeOverlays(), rayPick->getIgnoreOverlays(), !invisible, !noncollidable);
|
|
fromCache = false;
|
|
}
|
|
|
|
if (!fromCache) {
|
|
cacheResult(overlayRes.intersects, RayPickResult(IntersectionType::OVERLAY, overlayRes.overlayID, overlayRes.distance, overlayRes.intersection, overlayRes.surfaceNormal),
|
|
overlayMask, res, rayKey, results);
|
|
}
|
|
}
|
|
|
|
if (rayPick->getFilter().doesPickAvatars()) {
|
|
RayPickFilter::Flags avatarMask = rayPick->getFilter().getAvatarFlags();
|
|
if (!checkAndCompareCachedResults(rayKey, results, res, avatarMask)) {
|
|
RayToAvatarIntersectionResult avatarRes = DependencyManager::get<AvatarManager>()->findRayIntersection(ray, rayPick->getIncludeAvatars(), rayPick->getIgnoreAvatars());
|
|
cacheResult(avatarRes.intersects, RayPickResult(IntersectionType::AVATAR, avatarRes.avatarID, avatarRes.distance, avatarRes.intersection), avatarMask, res, rayKey, results);
|
|
}
|
|
}
|
|
|
|
// Can't intersect with HUD in desktop mode
|
|
if (rayPick->getFilter().doesPickHUD() && DependencyManager::get<HMDScriptingInterface>()->isHMDMode()) {
|
|
RayPickFilter::Flags hudMask = rayPick->getFilter().getHUDFlags();
|
|
if (!checkAndCompareCachedResults(rayKey, results, res, hudMask)) {
|
|
glm::vec3 hudRes = DependencyManager::get<HMDScriptingInterface>()->calculateRayUICollisionPoint(ray.origin, ray.direction);
|
|
cacheResult(true, RayPickResult(IntersectionType::HUD, 0, glm::distance(ray.origin, hudRes), hudRes), hudMask, res, rayKey, results);
|
|
}
|
|
}
|
|
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
if (rayPick->getMaxDistance() == 0.0f || (rayPick->getMaxDistance() > 0.0f && res.distance < rayPick->getMaxDistance())) {
|
|
rayPick->setRayPickResult(res);
|
|
} else {
|
|
rayPick->setRayPickResult(RayPickResult());
|
|
}
|
|
}
|
|
|
|
QWriteLocker containsLock(&_containsLock);
|
|
{
|
|
QWriteLocker lock(&_addLock);
|
|
while (!_rayPicksToAdd.empty()) {
|
|
std::pair<QUuid, std::shared_ptr<RayPick>> rayPickToAdd = _rayPicksToAdd.front();
|
|
_rayPicksToAdd.pop();
|
|
_rayPicks[rayPickToAdd.first] = rayPickToAdd.second;
|
|
_rayPickLocks[rayPickToAdd.first] = std::make_shared<QReadWriteLock>();
|
|
}
|
|
}
|
|
|
|
{
|
|
QWriteLocker lock(&_removeLock);
|
|
while (!_rayPicksToRemove.empty()) {
|
|
QUuid uid = _rayPicksToRemove.front();
|
|
_rayPicksToRemove.pop();
|
|
_rayPicks.remove(uid);
|
|
_rayPickLocks.remove(uid);
|
|
}
|
|
}
|
|
}
|
|
|
|
QUuid RayPickManager::createRayPick(const std::string& jointName, const glm::vec3& posOffset, const glm::vec3& dirOffset, const RayPickFilter& filter, const float maxDistance, const bool enabled) {
|
|
QWriteLocker lock(&_addLock);
|
|
QUuid id = QUuid::createUuid();
|
|
_rayPicksToAdd.push(std::pair<QUuid, std::shared_ptr<RayPick>>(id, std::make_shared<JointRayPick>(jointName, posOffset, dirOffset, filter, maxDistance, enabled)));
|
|
return id;
|
|
}
|
|
|
|
QUuid RayPickManager::createRayPick(const RayPickFilter& filter, const float maxDistance, const bool enabled) {
|
|
QWriteLocker lock(&_addLock);
|
|
QUuid id = QUuid::createUuid();
|
|
_rayPicksToAdd.push(std::pair<QUuid, std::shared_ptr<RayPick>>(id, std::make_shared<MouseRayPick>(filter, maxDistance, enabled)));
|
|
return id;
|
|
}
|
|
|
|
QUuid RayPickManager::createRayPick(const glm::vec3& position, const glm::vec3& direction, const RayPickFilter& filter, const float maxDistance, const bool enabled) {
|
|
QWriteLocker lock(&_addLock);
|
|
QUuid id = QUuid::createUuid();
|
|
_rayPicksToAdd.push(std::pair<QUuid, std::shared_ptr<RayPick>>(id, std::make_shared<StaticRayPick>(position, direction, filter, maxDistance, enabled)));
|
|
return id;
|
|
}
|
|
|
|
void RayPickManager::removeRayPick(const QUuid uid) {
|
|
QWriteLocker lock(&_removeLock);
|
|
_rayPicksToRemove.push(uid);
|
|
}
|
|
|
|
void RayPickManager::enableRayPick(const QUuid uid) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker rayPickLock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->enable();
|
|
}
|
|
}
|
|
|
|
void RayPickManager::disableRayPick(const QUuid uid) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker rayPickLock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->disable();
|
|
}
|
|
}
|
|
|
|
const PickRay RayPickManager::getPickRay(const QUuid uid) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
bool valid;
|
|
PickRay pickRay = _rayPicks[uid]->getPickRay(valid);
|
|
if (valid) {
|
|
return pickRay;
|
|
}
|
|
}
|
|
return PickRay();
|
|
}
|
|
|
|
const RayPickResult RayPickManager::getPrevRayPickResult(const QUuid uid) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QReadLocker lock(_rayPickLocks[uid].get());
|
|
return _rayPicks[uid]->getPrevRayPickResult();
|
|
}
|
|
return RayPickResult();
|
|
}
|
|
|
|
void RayPickManager::setIgnoreEntities(QUuid uid, const QScriptValue& ignoreEntities) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->setIgnoreEntities(ignoreEntities);
|
|
}
|
|
}
|
|
|
|
void RayPickManager::setIncludeEntities(QUuid uid, const QScriptValue& includeEntities) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->setIncludeEntities(includeEntities);
|
|
}
|
|
}
|
|
|
|
void RayPickManager::setIgnoreOverlays(QUuid uid, const QScriptValue& ignoreOverlays) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->setIgnoreOverlays(ignoreOverlays);
|
|
}
|
|
}
|
|
|
|
void RayPickManager::setIncludeOverlays(QUuid uid, const QScriptValue& includeOverlays) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->setIncludeOverlays(includeOverlays);
|
|
}
|
|
}
|
|
|
|
void RayPickManager::setIgnoreAvatars(QUuid uid, const QScriptValue& ignoreAvatars) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->setIgnoreAvatars(ignoreAvatars);
|
|
}
|
|
}
|
|
|
|
void RayPickManager::setIncludeAvatars(QUuid uid, const QScriptValue& includeAvatars) {
|
|
QReadLocker containsLock(&_containsLock);
|
|
if (_rayPicks.contains(uid)) {
|
|
QWriteLocker lock(_rayPickLocks[uid].get());
|
|
_rayPicks[uid]->setIncludeAvatars(includeAvatars);
|
|
}
|
|
} |