// // Created by Sam Gondelman 10/19/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 "PickManager.h" #include "PerfStat.h" #include "Profile.h" PickManager::PickManager() { setShouldPickHUDOperator([]() { return false; }); setCalculatePos2DFromHUDOperator([](const glm::vec3& intersection) { return glm::vec2(NAN); }); } unsigned int PickManager::addPick(PickQuery::PickType type, const std::shared_ptr pick) { unsigned int id = INVALID_PICK_ID; withWriteLock([&] { // Don't let the pick IDs overflow if (_nextPickID < UINT32_MAX) { id = _nextPickID++; _picks[type][id] = pick; _typeMap[id] = type; _totalPickCounts[type]++; } }); return id; } std::shared_ptr PickManager::findPick(unsigned int uid) const { return resultWithReadLock>([&] { auto type = _typeMap.find(uid); if (type != _typeMap.end()) { return _picks.find(type->second)->second.find(uid)->second; } return std::shared_ptr(); }); } void PickManager::removePick(unsigned int uid) { withWriteLock([&] { auto typeIt = _typeMap.find(uid); if (typeIt != _typeMap.end()) { _picks[typeIt->second].erase(uid); _totalPickCounts[typeIt->second]--; _typeMap.erase(typeIt); } }); } QVariantMap PickManager::getPickProperties(unsigned int uid) const { auto pick = findPick(uid); if (pick) { return pick->toVariantMap(); } return QVariantMap(); } QVariantMap PickManager::getPickScriptParameters(unsigned int uid) const { auto pick = findPick(uid); if (pick) { return pick->getScriptParameters(); } return QVariantMap(); } QVector PickManager::getCreatedPicks() const { QVector picks; withReadLock([&] { for (auto typeIt = _picks.cbegin(); typeIt != _picks.cend(); ++typeIt) { auto& picksForType = typeIt->second; for (auto pickIt = picksForType.cbegin(); pickIt != picksForType.cend(); ++pickIt) { picks.push_back(pickIt->first); } } }); return picks; } PickResultPointer PickManager::getPrevPickResult(unsigned int uid) const { auto pick = findPick(uid); if (pick) { return pick->getPrevPickResult(); } return PickResultPointer(); } void PickManager::enablePick(unsigned int uid) const { auto pick = findPick(uid); if (pick) { pick->enable(); } } void PickManager::disablePick(unsigned int uid) const { auto pick = findPick(uid); if (pick) { pick->disable(); } } bool PickManager::isPickEnabled(unsigned int uid) const { auto pick = findPick(uid); if (pick) { return pick->isEnabled(); } return false; } void PickManager::setPrecisionPicking(unsigned int uid, bool precisionPicking) const { auto pick = findPick(uid); if (pick) { pick->setPrecisionPicking(precisionPicking); } } void PickManager::setIgnoreItems(unsigned int uid, const QVector& ignore) const { auto pick = findPick(uid); if (pick) { pick->setIgnoreItems(ignore); } } void PickManager::setIncludeItems(unsigned int uid, const QVector& include) const { auto pick = findPick(uid); if (pick) { pick->setIncludeItems(include); } } Transform PickManager::getParentTransform(unsigned int uid) const { auto pick = findPick(uid); if (pick) { auto parentTransform = pick->parentTransform; if (parentTransform) { return parentTransform->getTransform(); } } return Transform(); } Transform PickManager::getResultTransform(unsigned int uid) const { auto pick = findPick(uid); if (pick) { return pick->getResultTransform(); } return Transform(); } void PickManager::update() { uint64_t expiry = usecTimestampNow() + _perFrameTimeBudget; std::unordered_map>> cachedPicks; withReadLock([&] { cachedPicks = _picks; }); bool shouldPickHUD = _shouldPickHUDOperator(); // FIXME: give each type its own expiry // Each type will update at least one pick, regardless of the expiry { PROFILE_RANGE_EX(picks, "StylusPicks", 0xffff0000, (uint64_t)_totalPickCounts[PickQuery::Stylus]); PerformanceTimer perfTimer("StylusPicks"); _updatedPickCounts[PickQuery::Stylus] = _stylusPickCacheOptimizer.update(cachedPicks[PickQuery::Stylus], _nextPickToUpdate[PickQuery::Stylus], expiry, false); } { PROFILE_RANGE_EX(picks, "RayPicks", 0xffff0000, (uint64_t)_totalPickCounts[PickQuery::Ray]); PerformanceTimer perfTimer("RayPicks"); _updatedPickCounts[PickQuery::Ray] = _rayPickCacheOptimizer.update(cachedPicks[PickQuery::Ray], _nextPickToUpdate[PickQuery::Ray], expiry, shouldPickHUD); } { PROFILE_RANGE_EX(picks, "ParabolaPicks", 0xffff0000, (uint64_t)_totalPickCounts[PickQuery::Parabola]); PerformanceTimer perfTimer("ParabolaPicks"); _updatedPickCounts[PickQuery::Parabola] = _parabolaPickCacheOptimizer.update(cachedPicks[PickQuery::Parabola], _nextPickToUpdate[PickQuery::Parabola], expiry, shouldPickHUD); } { PROFILE_RANGE_EX(picks, "CollisionPicks", 0xffff0000, (uint64_t)_totalPickCounts[PickQuery::Collision]); PerformanceTimer perfTimer("CollisionPicks"); _updatedPickCounts[PickQuery::Collision] = _collisionPickCacheOptimizer.update(cachedPicks[PickQuery::Collision], _nextPickToUpdate[PickQuery::Collision], expiry, false); } } bool PickManager::isLeftHand(unsigned int uid) { auto pick = findPick(uid); if (pick) { return pick->isLeftHand(); } return false; } bool PickManager::isRightHand(unsigned int uid) { auto pick = findPick(uid); if (pick) { return pick->isRightHand(); } return false; } bool PickManager::isMouse(unsigned int uid) { auto pick = findPick(uid); if (pick) { return pick->isMouse(); } return false; }