2020-10-21 19:22:38 +00:00
|
|
|
#include "vrcamera.hpp"
|
|
|
|
#include "vrgui.hpp"
|
|
|
|
#include "vrinputmanager.hpp"
|
|
|
|
#include "vrenvironment.hpp"
|
|
|
|
#include "vranimation.hpp"
|
|
|
|
|
|
|
|
#include <components/sceneutil/visitor.hpp>
|
|
|
|
|
2020-11-01 14:45:14 +00:00
|
|
|
#include <components/misc/constants.hpp>
|
|
|
|
|
2020-10-21 19:22:38 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
|
|
|
#include "../mwworld/player.hpp"
|
|
|
|
|
|
|
|
#include <osg/Quat>
|
|
|
|
|
|
|
|
namespace MWVR
|
|
|
|
{
|
|
|
|
|
|
|
|
VRCamera::VRCamera(osg::Camera* camera)
|
|
|
|
: MWRender::Camera(camera)
|
|
|
|
, mRoll(0.f)
|
|
|
|
{
|
|
|
|
mVanityAllowed = false;
|
|
|
|
mFirstPersonView = true;
|
|
|
|
|
|
|
|
auto* vrGuiManager = MWVR::Environment::get().getGUIManager();
|
|
|
|
vrGuiManager->setCamera(camera);
|
|
|
|
}
|
|
|
|
|
|
|
|
VRCamera::~VRCamera()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::recenter()
|
|
|
|
{
|
|
|
|
// Move position of head to center of character
|
|
|
|
// Z should not be affected
|
|
|
|
mHeadOffset = osg::Vec3(0, 0, 0);
|
|
|
|
mHeadOffset.z() = mHeadPose.position.z();
|
|
|
|
|
|
|
|
// Adjust orientation to zero yaw
|
|
|
|
float yaw = 0.f;
|
|
|
|
float pitch = 0.f;
|
|
|
|
float roll = 0.f;
|
|
|
|
getEulerAngles(mHeadPose.orientation, yaw, pitch, roll);
|
|
|
|
mYawOffset = -yaw;
|
|
|
|
|
|
|
|
mShouldRecenter = false;
|
|
|
|
Log(Debug::Verbose) << "Recentered";
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::applyTracking()
|
|
|
|
{
|
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
|
|
|
if (!world)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto& player = world->getPlayer();
|
|
|
|
auto playerPtr = player.getPlayer();
|
|
|
|
|
|
|
|
float yaw = 0.f;
|
|
|
|
float pitch = 0.f;
|
|
|
|
float roll = 0.f;
|
|
|
|
getEulerAngles(mHeadPose.orientation, yaw, pitch, roll);
|
|
|
|
|
|
|
|
yaw += mYawOffset;
|
|
|
|
|
|
|
|
if (player.isDisabled() || !mTrackingNode)
|
|
|
|
{
|
|
|
|
rotateCamera(-pitch, -roll, -yaw, false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
world->rotateObject(playerPtr, pitch, roll, yaw, MWBase::RotationFlag_none);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::updateTracking()
|
|
|
|
{
|
|
|
|
auto* session = Environment::get().getSession();
|
|
|
|
auto& frameMeta = session->getFrame(VRSession::FramePhase::Update);
|
|
|
|
// Only update tracking if rendering.
|
|
|
|
// OpenXR does not provide tracking information while not rendering.
|
2020-10-24 20:42:34 +00:00
|
|
|
if (frameMeta && frameMeta->mShouldRender)
|
2020-10-21 19:22:38 +00:00
|
|
|
{
|
|
|
|
auto currentHeadPose = frameMeta->mPredictedPoses.head;
|
2020-11-01 14:45:14 +00:00
|
|
|
currentHeadPose.position *= Constants::UnitsPerMeter;
|
2020-10-21 19:22:38 +00:00
|
|
|
osg::Vec3 vrMovement = currentHeadPose.position - mHeadPose.position;
|
|
|
|
mHeadPose = currentHeadPose;
|
|
|
|
mHeadOffset += stageRotation() * vrMovement;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::updateCamera(osg::Camera* cam)
|
|
|
|
{
|
|
|
|
updateTracking();
|
|
|
|
|
|
|
|
if (mShouldRecenter)
|
|
|
|
{
|
|
|
|
recenter();
|
|
|
|
Camera::updateCamera(cam);
|
|
|
|
auto* vrGuiManager = MWVR::Environment::get().getGUIManager();
|
|
|
|
vrGuiManager->updateTracking();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
applyTracking();
|
|
|
|
Camera::updateCamera(cam);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::updateCamera()
|
|
|
|
{
|
|
|
|
Camera::updateCamera();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::reset()
|
|
|
|
{
|
|
|
|
Camera::reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::rotateCamera(float pitch, float roll, float yaw, bool adjust)
|
|
|
|
{
|
|
|
|
if (adjust)
|
|
|
|
{
|
|
|
|
pitch += getPitch();
|
|
|
|
yaw += getYaw();
|
|
|
|
roll += getRoll();
|
|
|
|
}
|
|
|
|
setYaw(yaw);
|
|
|
|
setPitch(pitch);
|
|
|
|
setRoll(roll);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::setRoll(float angle)
|
|
|
|
{
|
|
|
|
if (angle > osg::PI) {
|
|
|
|
angle -= osg::PI * 2;
|
|
|
|
}
|
|
|
|
else if (angle < -osg::PI) {
|
|
|
|
angle += osg::PI * 2;
|
|
|
|
}
|
|
|
|
mRoll = angle;
|
|
|
|
}
|
|
|
|
void VRCamera::toggleViewMode(bool force)
|
|
|
|
{
|
|
|
|
mFirstPersonView = true;
|
|
|
|
}
|
|
|
|
bool VRCamera::toggleVanityMode(bool enable)
|
|
|
|
{
|
|
|
|
// Vanity mode makes no sense in VR
|
|
|
|
return Camera::toggleVanityMode(false);
|
|
|
|
}
|
|
|
|
void VRCamera::allowVanityMode(bool allow)
|
|
|
|
{
|
|
|
|
// Vanity mode makes no sense in VR
|
|
|
|
mVanityAllowed = false;
|
|
|
|
}
|
|
|
|
void VRCamera::getPosition(osg::Vec3d& focal, osg::Vec3d& camera) const
|
|
|
|
{
|
|
|
|
Camera::getPosition(focal, camera);
|
|
|
|
camera += mHeadOffset;
|
|
|
|
}
|
|
|
|
void VRCamera::getOrientation(osg::Quat& orientation) const
|
|
|
|
{
|
|
|
|
orientation = osg::Quat(getPitch(), osg::Vec3d(1, 0, 0))
|
|
|
|
* osg::Quat(getRoll(), osg::Vec3d(0, 1, 0))
|
|
|
|
* osg::Quat(getYaw(), osg::Vec3d(0, 0, 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::processViewChange()
|
|
|
|
{
|
|
|
|
SceneUtil::FindByNameVisitor findRootVisitor("Player Root", osg::NodeVisitor::TRAVERSE_PARENTS);
|
|
|
|
mAnimation->getObjectRoot()->accept(findRootVisitor);
|
|
|
|
mTrackingNode = findRootVisitor.mFoundNode;
|
|
|
|
|
|
|
|
if (!mTrackingNode)
|
|
|
|
throw std::logic_error("Unable to find tracking node for VR camera");
|
|
|
|
mHeightScale = 1.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRCamera::rotateCameraToTrackingPtr()
|
|
|
|
{
|
|
|
|
Camera::rotateCameraToTrackingPtr();
|
|
|
|
setRoll(-mTrackingPtr.getRefData().getPosition().rot[1] - mDeferredRotation.y());
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::Quat VRCamera::stageRotation()
|
|
|
|
{
|
|
|
|
return osg::Quat(mYawOffset, osg::Vec3(0, 0, -1));
|
|
|
|
}
|
|
|
|
}
|