1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-16 02:19:55 +00:00
openmw-tes3mp/apps/openmw/mwvr/vrcamera.cpp

192 lines
5 KiB
C++

#include "vrcamera.hpp"
#include "vrgui.hpp"
#include "vrinputmanager.hpp"
#include "vrenvironment.hpp"
#include "vranimation.hpp"
#include <components/sceneutil/visitor.hpp>
#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.
if (frameMeta && frameMeta->mShouldRender)
{
auto currentHeadPose = frameMeta->mPredictedPoses.head;
currentHeadPose.position *= Environment::get().unitsPerMeter();
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));
}
}