2013-04-29 12:50:40 +00:00
|
|
|
#include "camera.hpp"
|
2011-01-08 14:11:37 +00:00
|
|
|
|
2015-06-14 21:13:26 +00:00
|
|
|
#include <osg/Camera>
|
2012-08-08 20:15:52 +00:00
|
|
|
|
2015-11-20 20:57:04 +00:00
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
2020-06-23 18:05:22 +00:00
|
|
|
#include <components/settings/settings.hpp>
|
2020-02-14 21:11:19 +00:00
|
|
|
#include <components/sceneutil/visitor.hpp>
|
2015-11-20 20:57:04 +00:00
|
|
|
|
2012-08-09 13:01:03 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-08-17 14:55:21 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2012-08-09 13:01:03 +00:00
|
|
|
|
2020-06-21 23:54:08 +00:00
|
|
|
#include "../mwworld/class.hpp"
|
2012-08-08 20:15:52 +00:00
|
|
|
#include "../mwworld/ptr.hpp"
|
|
|
|
#include "../mwworld/refdata.hpp"
|
2012-07-03 13:32:38 +00:00
|
|
|
|
2020-02-23 10:02:38 +00:00
|
|
|
#ifdef USE_OPENXR
|
2020-06-21 21:40:07 +00:00
|
|
|
#include "../mwvr/vrinputmanager.hpp"
|
2020-03-15 14:31:38 +00:00
|
|
|
#include "../mwvr/vrenvironment.hpp"
|
2020-02-23 10:02:38 +00:00
|
|
|
#endif
|
|
|
|
|
2020-06-21 23:54:08 +00:00
|
|
|
#include "../mwmechanics/drawstate.hpp"
|
2020-07-18 09:48:46 +00:00
|
|
|
#include "../mwmechanics/movement.hpp"
|
2020-06-21 23:54:08 +00:00
|
|
|
#include "../mwmechanics/npcstats.hpp"
|
|
|
|
|
2012-08-14 16:33:29 +00:00
|
|
|
#include "npcanimation.hpp"
|
2020-02-14 21:11:19 +00:00
|
|
|
#include <components/debug/debuglog.hpp>
|
2012-08-14 16:33:29 +00:00
|
|
|
|
2015-05-21 21:54:39 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
2015-09-24 13:21:42 +00:00
|
|
|
class UpdateRenderCameraCallback : public osg::NodeCallback
|
2015-05-21 21:54:39 +00:00
|
|
|
{
|
|
|
|
public:
|
2015-09-24 13:21:42 +00:00
|
|
|
UpdateRenderCameraCallback(MWRender::Camera* cam)
|
2015-05-21 21:54:39 +00:00
|
|
|
: mCamera(cam)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv)
|
|
|
|
{
|
|
|
|
osg::Camera* cam = static_cast<osg::Camera*>(node);
|
|
|
|
|
|
|
|
// traverse first to update animations, in case the camera is attached to an animated node
|
|
|
|
traverse(node, nv);
|
|
|
|
|
|
|
|
mCamera->updateCamera(cam);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
MWRender::Camera* mCamera;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-01-08 14:11:37 +00:00
|
|
|
namespace MWRender
|
|
|
|
{
|
2015-05-21 21:54:39 +00:00
|
|
|
|
|
|
|
Camera::Camera (osg::Camera* camera)
|
2015-11-09 16:30:11 +00:00
|
|
|
: mHeightScale(1.f),
|
|
|
|
mCamera(camera),
|
2018-10-09 06:21:12 +00:00
|
|
|
mAnimation(nullptr),
|
2012-08-08 20:15:52 +00:00
|
|
|
mFirstPersonView(true),
|
2020-07-18 08:27:13 +00:00
|
|
|
mMode(Mode::Normal),
|
|
|
|
mVanityAllowed(true),
|
2020-07-18 11:27:53 +00:00
|
|
|
mStandingPreviewAllowed(Settings::Manager::getBool("preview if stand still", "Camera")),
|
|
|
|
mDeferredRotationAllowed(Settings::Manager::getBool("deferred preview rotation", "Camera")),
|
2013-07-29 14:43:16 +00:00
|
|
|
mNearest(30.f),
|
2013-07-29 14:45:35 +00:00
|
|
|
mFurthest(800.f),
|
|
|
|
mIsNearest(false),
|
2014-10-01 15:45:20 +00:00
|
|
|
mHeight(124.f),
|
2020-06-23 18:05:22 +00:00
|
|
|
mBaseCameraDistance(Settings::Manager::getFloat("third person camera distance", "Camera")),
|
2013-12-26 23:36:06 +00:00
|
|
|
mVanityToggleQueued(false),
|
2016-06-09 23:39:37 +00:00
|
|
|
mVanityToggleQueuedValue(false),
|
2015-05-21 21:54:39 +00:00
|
|
|
mViewModeToggleQueued(false),
|
2020-06-21 23:54:08 +00:00
|
|
|
mCameraDistance(0.f),
|
2020-07-16 21:32:59 +00:00
|
|
|
mMaxNextCameraDistance(800.f),
|
2020-06-19 20:03:11 +00:00
|
|
|
mFocalPointCurrentOffset(osg::Vec2d()),
|
2020-07-11 12:21:18 +00:00
|
|
|
mFocalPointTargetOffset(osg::Vec2d()),
|
2020-07-11 23:55:20 +00:00
|
|
|
mFocalPointTransitionSpeedCoef(1.f),
|
2020-07-16 20:21:50 +00:00
|
|
|
mSkipFocalPointTransition(true),
|
2020-07-11 23:55:20 +00:00
|
|
|
mPreviousTransitionInfluence(0.f),
|
2020-07-11 12:21:18 +00:00
|
|
|
mSmoothedSpeed(0.f),
|
2020-07-12 19:03:18 +00:00
|
|
|
mZoomOutWhenMoveCoef(Settings::Manager::getFloat("zoom out when move coef", "Camera")),
|
2020-07-11 12:21:18 +00:00
|
|
|
mDynamicCameraDistanceEnabled(false),
|
2020-07-18 09:48:46 +00:00
|
|
|
mShowCrosshairInThirdPersonMode(false),
|
|
|
|
mDeferredRotation(osg::Vec3f()),
|
|
|
|
mDeferredRotationDisabled(false)
|
2012-05-30 13:52:39 +00:00
|
|
|
{
|
2020-06-21 23:54:08 +00:00
|
|
|
mCameraDistance = mBaseCameraDistance;
|
2016-06-09 23:05:43 +00:00
|
|
|
|
2015-09-24 13:21:42 +00:00
|
|
|
mUpdateCallback = new UpdateRenderCameraCallback(this);
|
2015-05-21 21:54:39 +00:00
|
|
|
mCamera->addUpdateCallback(mUpdateCallback);
|
2012-08-12 11:50:37 +00:00
|
|
|
}
|
2012-09-13 17:03:31 +00:00
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
Camera::~Camera()
|
2012-09-13 17:03:31 +00:00
|
|
|
{
|
2015-05-21 21:54:39 +00:00
|
|
|
mCamera->removeUpdateCallback(mUpdateCallback);
|
2012-09-13 17:03:31 +00:00
|
|
|
}
|
2012-07-03 13:32:38 +00:00
|
|
|
|
2020-06-21 23:54:08 +00:00
|
|
|
osg::Vec3d Camera::getFocalPoint() const
|
2012-07-03 13:32:38 +00:00
|
|
|
{
|
2020-07-16 23:11:09 +00:00
|
|
|
if (!mTrackingNode)
|
2020-02-14 21:11:19 +00:00
|
|
|
{
|
2015-06-01 13:34:46 +00:00
|
|
|
return osg::Vec3d();
|
2020-02-14 21:11:19 +00:00
|
|
|
}
|
2020-07-22 16:56:25 +00:00
|
|
|
|
2020-07-16 23:11:09 +00:00
|
|
|
osg::NodePathList nodepaths = mTrackingNode->getParentalNodePaths();
|
2016-02-22 17:58:19 +00:00
|
|
|
if (nodepaths.empty())
|
2015-06-01 13:34:46 +00:00
|
|
|
return osg::Vec3d();
|
2016-02-22 17:58:19 +00:00
|
|
|
osg::Matrix worldMat = osg::computeLocalToWorld(nodepaths[0]);
|
2013-04-27 08:24:36 +00:00
|
|
|
|
2015-05-31 21:09:37 +00:00
|
|
|
osg::Vec3d position = worldMat.getTrans();
|
2015-05-21 21:54:39 +00:00
|
|
|
if (!isFirstPerson())
|
2020-06-21 23:54:08 +00:00
|
|
|
{
|
2015-11-09 16:30:11 +00:00
|
|
|
position.z() += mHeight * mHeightScale;
|
2020-06-21 23:54:08 +00:00
|
|
|
|
|
|
|
// We subtract 10.f here and add it within focalPointOffset in order to avoid camera clipping through ceiling.
|
|
|
|
// Needed because character's head can be a bit higher than collision area.
|
|
|
|
position.z() -= 10.f;
|
|
|
|
|
|
|
|
position += getFocalPointOffset() + mFocalPointAdjustment;
|
|
|
|
}
|
2015-06-01 13:34:46 +00:00
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
osg::Camera* Camera::getOsgCamera()
|
|
|
|
{
|
|
|
|
return mCamera;
|
|
|
|
}
|
|
|
|
|
2020-02-29 13:12:38 +00:00
|
|
|
void Camera::updateCamera()
|
|
|
|
{
|
|
|
|
updateCamera(mCamera);
|
|
|
|
}
|
2020-06-28 09:59:07 +00:00
|
|
|
|
2020-06-21 23:54:08 +00:00
|
|
|
osg::Vec3d Camera::getFocalPointOffset() const
|
|
|
|
{
|
|
|
|
osg::Vec3d offset(0, 0, 10.f);
|
2020-07-16 23:11:09 +00:00
|
|
|
offset.x() += mFocalPointCurrentOffset.x() * cos(getYaw());
|
|
|
|
offset.y() += mFocalPointCurrentOffset.x() * sin(getYaw());
|
|
|
|
offset.z() += mFocalPointCurrentOffset.y();
|
2020-06-21 23:54:08 +00:00
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Camera::getPosition(osg::Vec3d &focal, osg::Vec3d &camera) const
|
|
|
|
{
|
|
|
|
focal = getFocalPoint();
|
|
|
|
osg::Vec3d offset(0,0,0);
|
|
|
|
if (!isFirstPerson())
|
|
|
|
{
|
|
|
|
osg::Quat orient = osg::Quat(getPitch(), osg::Vec3d(1,0,0)) * osg::Quat(getYaw(), osg::Vec3d(0,0,1));
|
|
|
|
offset = orient * osg::Vec3d(0.f, -mCameraDistance, 0.f);
|
|
|
|
}
|
|
|
|
camera = focal + offset;
|
|
|
|
}
|
2020-02-29 13:12:38 +00:00
|
|
|
|
2015-06-01 13:34:46 +00:00
|
|
|
void Camera::updateCamera(osg::Camera *cam)
|
|
|
|
{
|
2020-02-23 11:56:48 +00:00
|
|
|
osg::Quat orient = osg::Quat(getPitch(), osg::Vec3d(1,0,0)) * osg::Quat(getRoll(), osg::Vec3d(0, 1, 0)) * osg::Quat(getYaw(), osg::Vec3d(0,0,1));
|
2020-06-21 23:54:08 +00:00
|
|
|
osg::Vec3d focal, position;
|
|
|
|
getPosition(focal, position);
|
2014-10-01 16:55:35 +00:00
|
|
|
|
2020-02-23 10:02:38 +00:00
|
|
|
#ifdef USE_OPENXR
|
2020-03-15 14:31:38 +00:00
|
|
|
auto* inputManager = MWVR::Environment::get().getInputManager();
|
2020-02-23 10:02:38 +00:00
|
|
|
if (inputManager)
|
|
|
|
{
|
2020-06-21 21:40:07 +00:00
|
|
|
position += inputManager->headOffset();
|
2020-02-23 10:02:38 +00:00
|
|
|
}
|
2020-02-23 11:56:48 +00:00
|
|
|
#endif
|
|
|
|
|
2015-05-31 21:09:37 +00:00
|
|
|
osg::Vec3d forward = orient * osg::Vec3d(0,1,0);
|
|
|
|
osg::Vec3d up = orient * osg::Vec3d(0,0,1);
|
2015-05-21 21:54:39 +00:00
|
|
|
|
2020-02-29 13:12:38 +00:00
|
|
|
cam->setViewMatrixAsLookAt(position, position + forward, up);
|
2012-08-08 20:15:52 +00:00
|
|
|
}
|
2012-08-09 06:24:18 +00:00
|
|
|
|
2015-05-21 21:54:39 +00:00
|
|
|
void Camera::reset()
|
2012-08-09 06:24:18 +00:00
|
|
|
{
|
2015-05-21 21:54:39 +00:00
|
|
|
togglePreviewMode(false);
|
|
|
|
toggleVanityMode(false);
|
|
|
|
if (!mFirstPersonView)
|
|
|
|
toggleViewMode();
|
2012-08-12 11:50:37 +00:00
|
|
|
}
|
|
|
|
|
2020-02-23 10:02:38 +00:00
|
|
|
void Camera::rotateCamera(float pitch, float roll, float yaw, bool adjust)
|
2012-08-12 11:50:37 +00:00
|
|
|
{
|
2015-05-21 21:54:39 +00:00
|
|
|
if (adjust)
|
2013-04-29 06:44:44 +00:00
|
|
|
{
|
2015-05-21 21:54:39 +00:00
|
|
|
pitch += getPitch();
|
|
|
|
yaw += getYaw();
|
2020-02-23 10:02:38 +00:00
|
|
|
roll += getRoll();
|
2013-04-29 06:44:44 +00:00
|
|
|
}
|
2015-05-21 21:54:39 +00:00
|
|
|
setYaw(yaw);
|
|
|
|
setPitch(pitch);
|
2020-02-23 10:02:38 +00:00
|
|
|
setRoll(roll);
|
2014-10-01 16:55:35 +00:00
|
|
|
}
|
|
|
|
|
2013-07-30 21:24:18 +00:00
|
|
|
void Camera::update(float duration, bool paused)
|
2012-08-12 11:50:37 +00:00
|
|
|
{
|
2014-08-27 22:41:52 +00:00
|
|
|
if (mAnimation->upperBodyReady())
|
2013-12-26 23:36:06 +00:00
|
|
|
{
|
|
|
|
// Now process the view changes we queued earlier
|
|
|
|
if (mVanityToggleQueued)
|
|
|
|
{
|
2016-06-09 23:39:37 +00:00
|
|
|
toggleVanityMode(mVanityToggleQueuedValue);
|
2013-12-26 23:36:06 +00:00
|
|
|
mVanityToggleQueued = false;
|
|
|
|
}
|
|
|
|
if (mViewModeToggleQueued)
|
|
|
|
{
|
|
|
|
togglePreviewMode(false);
|
|
|
|
toggleViewMode();
|
|
|
|
mViewModeToggleQueued = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-30 21:24:18 +00:00
|
|
|
if (paused)
|
|
|
|
return;
|
2012-10-01 07:24:44 +00:00
|
|
|
|
2020-06-21 23:54:08 +00:00
|
|
|
// only show the crosshair in game mode
|
2013-04-29 06:44:44 +00:00
|
|
|
MWBase::WindowManager *wm = MWBase::Environment::get().getWindowManager();
|
2020-07-18 08:27:13 +00:00
|
|
|
wm->showCrosshair(!wm->isGuiMode() && mMode != Mode::Preview && mMode != Mode::Vanity
|
2020-07-11 12:21:18 +00:00
|
|
|
&& (mFirstPersonView || mShowCrosshairInThirdPersonMode));
|
2012-08-27 17:18:55 +00:00
|
|
|
|
2020-07-18 08:27:13 +00:00
|
|
|
if(mMode == Mode::Vanity)
|
2020-02-23 10:02:38 +00:00
|
|
|
rotateCamera(0.f, 0.f, osg::DegreesToRadians(3.f * duration), true);
|
2020-06-21 23:54:08 +00:00
|
|
|
|
2020-06-19 20:03:11 +00:00
|
|
|
updateFocalPointOffset(duration);
|
2020-06-22 17:26:37 +00:00
|
|
|
|
|
|
|
float speed = mTrackingPtr.getClass().getSpeed(mTrackingPtr);
|
2020-07-16 20:21:50 +00:00
|
|
|
speed /= (1.f + speed / 500.f);
|
2020-06-22 17:26:37 +00:00
|
|
|
float maxDelta = 300.f * duration;
|
|
|
|
mSmoothedSpeed += osg::clampBetween(speed - mSmoothedSpeed, -maxDelta, maxDelta);
|
2020-07-16 21:32:59 +00:00
|
|
|
|
|
|
|
mMaxNextCameraDistance = mCameraDistance + duration * (100.f + mBaseCameraDistance);
|
2020-07-18 11:27:53 +00:00
|
|
|
updateStandingPreviewMode();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Camera::updateStandingPreviewMode()
|
|
|
|
{
|
|
|
|
if (!mStandingPreviewAllowed)
|
|
|
|
return;
|
|
|
|
float speed = mTrackingPtr.getClass().getSpeed(mTrackingPtr);
|
|
|
|
bool combat = mTrackingPtr.getClass().isActor() &&
|
|
|
|
mTrackingPtr.getClass().getCreatureStats(mTrackingPtr).getDrawState() != MWMechanics::DrawState_Nothing;
|
|
|
|
bool standingStill = speed == 0 && !combat && !mFirstPersonView;
|
|
|
|
if (!standingStill && mMode == Mode::StandingPreview)
|
|
|
|
{
|
|
|
|
mMode = Mode::Normal;
|
|
|
|
calculateDeferredRotation();
|
|
|
|
}
|
|
|
|
else if (standingStill && mMode == Mode::Normal)
|
|
|
|
mMode = Mode::StandingPreview;
|
2020-06-21 23:54:08 +00:00
|
|
|
}
|
|
|
|
|
2020-07-11 23:55:20 +00:00
|
|
|
void Camera::setFocalPointTargetOffset(osg::Vec2d v)
|
2020-06-24 18:07:41 +00:00
|
|
|
{
|
2020-07-11 23:55:20 +00:00
|
|
|
mFocalPointTargetOffset = v;
|
|
|
|
mPreviousTransitionSpeed = mFocalPointTransitionSpeed;
|
|
|
|
mPreviousTransitionInfluence = 1.0f;
|
2020-06-24 18:07:41 +00:00
|
|
|
}
|
|
|
|
|
2020-06-19 20:03:11 +00:00
|
|
|
void Camera::updateFocalPointOffset(float duration)
|
2020-06-21 23:54:08 +00:00
|
|
|
{
|
2020-07-11 23:55:20 +00:00
|
|
|
if (duration <= 0)
|
|
|
|
return;
|
2020-06-21 23:54:08 +00:00
|
|
|
|
2020-07-16 20:21:50 +00:00
|
|
|
if (mSkipFocalPointTransition)
|
|
|
|
{
|
|
|
|
mSkipFocalPointTransition = false;
|
|
|
|
mPreviousExtraOffset = osg::Vec2d();
|
|
|
|
mPreviousTransitionInfluence = 0.f;
|
|
|
|
mFocalPointCurrentOffset = mFocalPointTargetOffset;
|
|
|
|
}
|
|
|
|
|
2020-07-11 23:55:20 +00:00
|
|
|
osg::Vec2d oldOffset = mFocalPointCurrentOffset;
|
|
|
|
|
|
|
|
if (mPreviousTransitionInfluence > 0)
|
|
|
|
{
|
|
|
|
mFocalPointCurrentOffset -= mPreviousExtraOffset;
|
|
|
|
mPreviousExtraOffset = mPreviousExtraOffset / mPreviousTransitionInfluence + mPreviousTransitionSpeed * duration;
|
|
|
|
mPreviousTransitionInfluence =
|
|
|
|
std::max(0.f, mPreviousTransitionInfluence - duration * mFocalPointTransitionSpeedCoef);
|
|
|
|
mPreviousExtraOffset *= mPreviousTransitionInfluence;
|
|
|
|
mFocalPointCurrentOffset += mPreviousExtraOffset;
|
|
|
|
}
|
|
|
|
|
2020-07-11 12:21:18 +00:00
|
|
|
osg::Vec2d delta = mFocalPointTargetOffset - mFocalPointCurrentOffset;
|
2020-06-19 20:03:11 +00:00
|
|
|
if (delta.length2() > 0)
|
|
|
|
{
|
2020-07-11 23:55:20 +00:00
|
|
|
float coef = duration * (1.0 + 5.0 / delta.length()) *
|
|
|
|
mFocalPointTransitionSpeedCoef * (1.0f - mPreviousTransitionInfluence);
|
2020-06-19 20:03:11 +00:00
|
|
|
mFocalPointCurrentOffset += delta * std::min(coef, 1.0f);
|
|
|
|
}
|
|
|
|
else
|
2020-07-11 23:55:20 +00:00
|
|
|
{
|
|
|
|
mPreviousExtraOffset = osg::Vec2d();
|
|
|
|
mPreviousTransitionInfluence = 0.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
mFocalPointTransitionSpeed = (mFocalPointCurrentOffset - oldOffset) / duration;
|
2012-08-12 11:50:37 +00:00
|
|
|
}
|
|
|
|
|
2014-06-05 15:21:02 +00:00
|
|
|
void Camera::toggleViewMode(bool force)
|
2012-08-12 11:50:37 +00:00
|
|
|
{
|
2013-12-26 23:36:06 +00:00
|
|
|
// Changing the view will stop all playing animations, so if we are playing
|
|
|
|
// anything important, queue the view change for later
|
2014-08-27 22:41:52 +00:00
|
|
|
if (!mAnimation->upperBodyReady() && !force)
|
2013-12-26 23:36:06 +00:00
|
|
|
{
|
|
|
|
mViewModeToggleQueued = true;
|
|
|
|
return;
|
|
|
|
}
|
2014-07-03 17:37:12 +00:00
|
|
|
else
|
|
|
|
mViewModeToggleQueued = false;
|
2013-12-26 23:36:06 +00:00
|
|
|
|
2020-06-22 00:03:38 +00:00
|
|
|
if (mTrackingPtr.getClass().isActor())
|
|
|
|
mTrackingPtr.getClass().getCreatureStats(mTrackingPtr).setSideMovementAngle(0);
|
|
|
|
|
2012-08-12 11:50:37 +00:00
|
|
|
mFirstPersonView = !mFirstPersonView;
|
2020-07-18 11:27:53 +00:00
|
|
|
updateStandingPreviewMode();
|
2020-07-19 20:12:17 +00:00
|
|
|
instantTransition();
|
2013-07-13 23:12:38 +00:00
|
|
|
processViewChange();
|
2012-08-12 11:50:37 +00:00
|
|
|
}
|
2020-07-16 23:11:09 +00:00
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
void Camera::allowVanityMode(bool allow)
|
2012-08-14 10:37:48 +00:00
|
|
|
{
|
2020-07-18 08:27:13 +00:00
|
|
|
if (!allow && mMode == Mode::Vanity)
|
2020-07-18 09:48:46 +00:00
|
|
|
{
|
|
|
|
disableDeferredPreviewRotation();
|
2012-08-14 10:37:48 +00:00
|
|
|
toggleVanityMode(false);
|
2020-07-18 09:48:46 +00:00
|
|
|
}
|
2020-07-18 08:27:13 +00:00
|
|
|
mVanityAllowed = allow;
|
2012-08-14 10:37:48 +00:00
|
|
|
}
|
2012-08-12 11:50:37 +00:00
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
bool Camera::toggleVanityMode(bool enable)
|
2012-08-12 11:50:37 +00:00
|
|
|
{
|
2020-02-14 21:11:19 +00:00
|
|
|
#ifdef USE_OPENXR
|
|
|
|
// Vanity mode makes no sense in VR
|
|
|
|
enable = false;
|
|
|
|
#endif
|
2013-12-26 23:36:06 +00:00
|
|
|
// Changing the view will stop all playing animations, so if we are playing
|
|
|
|
// anything important, queue the view change for later
|
2016-06-09 23:44:32 +00:00
|
|
|
if (mFirstPersonView && !mAnimation->upperBodyReady())
|
2013-12-26 23:36:06 +00:00
|
|
|
{
|
|
|
|
mVanityToggleQueued = true;
|
2016-06-09 23:39:37 +00:00
|
|
|
mVanityToggleQueuedValue = enable;
|
2013-12-26 23:36:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-07-18 08:27:13 +00:00
|
|
|
if (!mVanityAllowed && enable)
|
2012-08-14 10:37:48 +00:00
|
|
|
return false;
|
2013-04-27 08:24:36 +00:00
|
|
|
|
2020-07-18 08:27:13 +00:00
|
|
|
if ((mMode == Mode::Vanity) == enable)
|
2012-08-14 10:37:48 +00:00
|
|
|
return true;
|
2020-07-18 08:27:13 +00:00
|
|
|
mMode = enable ? Mode::Vanity : Mode::Normal;
|
2020-07-18 11:27:53 +00:00
|
|
|
if (!mDeferredRotationAllowed)
|
|
|
|
disableDeferredPreviewRotation();
|
2020-07-18 09:48:46 +00:00
|
|
|
if (!enable)
|
|
|
|
calculateDeferredRotation();
|
2012-08-12 14:35:35 +00:00
|
|
|
|
2013-07-13 23:12:38 +00:00
|
|
|
processViewChange();
|
2012-08-14 10:37:48 +00:00
|
|
|
return true;
|
2012-08-12 11:50:37 +00:00
|
|
|
}
|
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
void Camera::togglePreviewMode(bool enable)
|
2012-08-12 11:50:37 +00:00
|
|
|
{
|
2014-08-27 22:41:52 +00:00
|
|
|
if (mFirstPersonView && !mAnimation->upperBodyReady())
|
2013-12-26 23:36:06 +00:00
|
|
|
return;
|
|
|
|
|
2020-07-18 08:27:13 +00:00
|
|
|
if((mMode == Mode::Preview) == enable)
|
2012-08-13 22:36:18 +00:00
|
|
|
return;
|
2013-04-27 08:24:36 +00:00
|
|
|
|
2020-07-18 08:27:13 +00:00
|
|
|
mMode = enable ? Mode::Preview : Mode::Normal;
|
2020-07-18 11:27:53 +00:00
|
|
|
if (mMode == Mode::Normal)
|
|
|
|
updateStandingPreviewMode();
|
2020-07-19 20:12:17 +00:00
|
|
|
else if (mFirstPersonView)
|
|
|
|
instantTransition();
|
2020-07-18 11:27:53 +00:00
|
|
|
if (mMode == Mode::Normal)
|
|
|
|
{
|
|
|
|
if (!mDeferredRotationAllowed)
|
|
|
|
disableDeferredPreviewRotation();
|
2020-07-18 09:48:46 +00:00
|
|
|
calculateDeferredRotation();
|
2012-08-14 18:39:42 +00:00
|
|
|
}
|
2013-07-13 23:12:38 +00:00
|
|
|
processViewChange();
|
2012-08-12 11:50:37 +00:00
|
|
|
}
|
2012-08-12 14:35:35 +00:00
|
|
|
|
2014-01-10 21:39:01 +00:00
|
|
|
void Camera::setSneakOffset(float offset)
|
2013-10-02 09:16:52 +00:00
|
|
|
{
|
2015-05-31 00:26:31 +00:00
|
|
|
mAnimation->setFirstPersonOffset(osg::Vec3f(0,0,-offset));
|
2013-10-02 09:16:52 +00:00
|
|
|
}
|
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
void Camera::setYaw(float angle)
|
2012-08-13 22:36:18 +00:00
|
|
|
{
|
2015-05-21 21:54:39 +00:00
|
|
|
if (angle > osg::PI) {
|
|
|
|
angle -= osg::PI*2;
|
|
|
|
} else if (angle < -osg::PI) {
|
|
|
|
angle += osg::PI*2;
|
2012-08-13 22:36:18 +00:00
|
|
|
}
|
2020-07-16 23:11:09 +00:00
|
|
|
mYaw = angle;
|
2020-02-23 10:02:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Camera::setRoll(float angle)
|
|
|
|
{
|
2020-03-31 18:28:43 +00:00
|
|
|
#ifdef USE_OPENXR
|
2020-02-23 10:02:38 +00:00
|
|
|
if (angle > osg::PI) {
|
|
|
|
angle -= osg::PI * 2;
|
|
|
|
}
|
|
|
|
else if (angle < -osg::PI) {
|
|
|
|
angle += osg::PI * 2;
|
|
|
|
}
|
2020-07-22 16:56:25 +00:00
|
|
|
mRoll = angle;
|
2020-03-31 18:28:43 +00:00
|
|
|
#else
|
|
|
|
// It seems OpenMW saves roll data, causing the camera to get tilted
|
|
|
|
// when loading a VR save in non-VR.
|
2020-07-22 16:56:25 +00:00
|
|
|
mRoll = 0.f;
|
2020-03-31 18:28:43 +00:00
|
|
|
#endif
|
2020-02-23 10:02:38 +00:00
|
|
|
}
|
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
void Camera::setPitch(float angle)
|
2012-08-14 10:37:48 +00:00
|
|
|
{
|
2013-04-27 08:24:36 +00:00
|
|
|
const float epsilon = 0.000001f;
|
2015-05-21 21:54:39 +00:00
|
|
|
float limit = osg::PI_2 - epsilon;
|
2020-07-16 23:11:09 +00:00
|
|
|
mPitch = osg::clampBetween(angle, -limit, limit);
|
2012-08-12 14:35:35 +00:00
|
|
|
}
|
2012-08-14 10:37:48 +00:00
|
|
|
|
2014-01-01 22:59:17 +00:00
|
|
|
float Camera::getCameraDistance() const
|
|
|
|
{
|
2016-06-09 23:05:43 +00:00
|
|
|
if (isFirstPerson())
|
|
|
|
return 0.f;
|
2015-05-21 21:54:39 +00:00
|
|
|
return mCameraDistance;
|
2014-01-01 22:59:17 +00:00
|
|
|
}
|
|
|
|
|
2020-06-23 18:05:22 +00:00
|
|
|
void Camera::updateBaseCameraDistance(float dist, bool adjust)
|
2012-08-14 10:37:48 +00:00
|
|
|
{
|
2020-07-18 08:27:13 +00:00
|
|
|
if (isFirstPerson())
|
2012-08-16 09:15:38 +00:00
|
|
|
return;
|
2013-04-27 08:24:36 +00:00
|
|
|
|
2015-05-21 21:54:39 +00:00
|
|
|
if (adjust)
|
2020-07-16 23:11:09 +00:00
|
|
|
dist += std::min(mCameraDistance - getCameraDistanceCorrection(), mBaseCameraDistance);
|
2015-05-21 21:54:39 +00:00
|
|
|
|
2020-07-16 23:11:09 +00:00
|
|
|
mIsNearest = dist <= mNearest;
|
2020-07-18 08:27:13 +00:00
|
|
|
mBaseCameraDistance = osg::clampBetween(dist, mNearest, mFurthest);
|
|
|
|
Settings::Manager::setFloat("third person camera distance", "Camera", mBaseCameraDistance);
|
2020-06-21 23:54:08 +00:00
|
|
|
setCameraDistance();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Camera::setCameraDistance(float dist, bool adjust)
|
|
|
|
{
|
2020-07-18 08:27:13 +00:00
|
|
|
if (isFirstPerson())
|
2020-06-21 23:54:08 +00:00
|
|
|
return;
|
2020-07-16 23:11:09 +00:00
|
|
|
if (adjust)
|
|
|
|
dist += mCameraDistance;
|
|
|
|
mCameraDistance = osg::clampBetween(dist, 10.f, mFurthest);
|
2020-06-21 23:54:08 +00:00
|
|
|
}
|
2012-08-14 10:37:48 +00:00
|
|
|
|
2020-06-21 23:54:08 +00:00
|
|
|
float Camera::getCameraDistanceCorrection() const
|
|
|
|
{
|
2020-07-11 12:21:18 +00:00
|
|
|
if (!mDynamicCameraDistanceEnabled)
|
2020-06-22 17:26:37 +00:00
|
|
|
return 0;
|
|
|
|
|
2020-07-11 12:21:18 +00:00
|
|
|
float pitchCorrection = std::max(-getPitch(), 0.f) * 50.f;
|
2020-06-22 17:26:37 +00:00
|
|
|
|
2020-07-11 12:21:18 +00:00
|
|
|
float smoothedSpeedSqr = mSmoothedSpeed * mSmoothedSpeed;
|
2020-07-12 19:03:18 +00:00
|
|
|
float speedCorrection = smoothedSpeedSqr / (smoothedSpeedSqr + 300.f*300.f) * mZoomOutWhenMoveCoef;
|
2020-07-11 12:21:18 +00:00
|
|
|
|
|
|
|
return pitchCorrection + speedCorrection;
|
2012-08-14 10:37:48 +00:00
|
|
|
}
|
2012-08-14 16:33:29 +00:00
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
void Camera::setCameraDistance()
|
2012-08-16 09:15:38 +00:00
|
|
|
{
|
2020-06-21 23:54:08 +00:00
|
|
|
mFocalPointAdjustment = osg::Vec3d();
|
2020-07-18 08:27:13 +00:00
|
|
|
if (isFirstPerson())
|
2020-07-16 23:11:09 +00:00
|
|
|
return;
|
|
|
|
mCameraDistance = mBaseCameraDistance + getCameraDistanceCorrection();
|
|
|
|
if (mDynamicCameraDistanceEnabled)
|
|
|
|
mCameraDistance = std::min(mCameraDistance, mMaxNextCameraDistance);
|
2012-08-16 09:15:38 +00:00
|
|
|
}
|
|
|
|
|
2013-04-29 12:50:40 +00:00
|
|
|
void Camera::setAnimation(NpcAnimation *anim)
|
2012-08-15 11:17:35 +00:00
|
|
|
{
|
|
|
|
mAnimation = anim;
|
2013-07-13 23:12:38 +00:00
|
|
|
processViewChange();
|
2012-08-15 11:17:35 +00:00
|
|
|
}
|
|
|
|
|
2013-07-13 23:12:38 +00:00
|
|
|
void Camera::processViewChange()
|
2012-08-14 16:33:29 +00:00
|
|
|
{
|
2020-02-14 21:11:19 +00:00
|
|
|
#ifdef USE_OPENXR
|
2020-03-28 15:30:56 +00:00
|
|
|
//mAnimation->setViewMode(NpcAnimation::VM_VRFirstPerson);
|
2020-02-14 21:11:19 +00:00
|
|
|
|
|
|
|
SceneUtil::FindByNameVisitor findRootVisitor("Player Root", osg::NodeVisitor::TRAVERSE_PARENTS);
|
|
|
|
mAnimation->getObjectRoot()->accept(findRootVisitor);
|
|
|
|
mTrackingNode = findRootVisitor.mFoundNode;
|
|
|
|
|
|
|
|
if (!mTrackingNode)
|
2020-03-23 22:32:47 +00:00
|
|
|
throw std::logic_error("Unable to find tracking node for VR camera");
|
2020-02-14 21:11:19 +00:00
|
|
|
mHeightScale = 1.f;
|
|
|
|
#else
|
2013-07-13 23:12:38 +00:00
|
|
|
if(isFirstPerson())
|
|
|
|
{
|
|
|
|
mAnimation->setViewMode(NpcAnimation::VM_FirstPerson);
|
2015-05-30 23:07:43 +00:00
|
|
|
mTrackingNode = mAnimation->getNode("Camera");
|
2015-06-01 02:41:03 +00:00
|
|
|
if (!mTrackingNode)
|
|
|
|
mTrackingNode = mAnimation->getNode("Head");
|
2015-11-09 16:30:11 +00:00
|
|
|
mHeightScale = 1.f;
|
2013-07-13 23:12:38 +00:00
|
|
|
}
|
2014-10-01 16:55:35 +00:00
|
|
|
else
|
2013-07-13 23:12:38 +00:00
|
|
|
{
|
|
|
|
mAnimation->setViewMode(NpcAnimation::VM_Normal);
|
2015-11-20 20:57:04 +00:00
|
|
|
SceneUtil::PositionAttitudeTransform* transform = mTrackingPtr.getRefData().getBaseNode();
|
2015-11-09 16:30:11 +00:00
|
|
|
mTrackingNode = transform;
|
|
|
|
if (transform)
|
2015-11-11 16:23:47 +00:00
|
|
|
mHeightScale = transform->getScale().z();
|
2015-11-09 16:30:11 +00:00
|
|
|
else
|
|
|
|
mHeightScale = 1.f;
|
2013-07-13 23:12:38 +00:00
|
|
|
}
|
2020-03-23 22:32:47 +00:00
|
|
|
rotateCamera(getPitch(), 0.f, getYaw(), false);
|
2020-02-14 21:11:19 +00:00
|
|
|
#endif
|
2012-08-14 16:33:29 +00:00
|
|
|
}
|
|
|
|
|
2020-07-18 09:48:46 +00:00
|
|
|
void Camera::applyDeferredPreviewRotationToPlayer(float dt)
|
|
|
|
{
|
|
|
|
if (isVanityOrPreviewModeEnabled() || mTrackingPtr.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
osg::Vec3f rot = mDeferredRotation;
|
|
|
|
float delta = rot.normalize();
|
|
|
|
delta = std::min(delta, (delta + 1.f) * 3 * dt);
|
|
|
|
rot *= delta;
|
|
|
|
mDeferredRotation -= rot;
|
|
|
|
|
2020-07-18 11:27:53 +00:00
|
|
|
if (mDeferredRotationDisabled)
|
|
|
|
{
|
|
|
|
mDeferredRotationDisabled = delta > 0.0001;
|
|
|
|
rotateCameraToTrackingPtr();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-18 09:48:46 +00:00
|
|
|
auto& movement = mTrackingPtr.getClass().getMovementSettings(mTrackingPtr);
|
|
|
|
movement.mRotation[0] += rot.x();
|
|
|
|
movement.mRotation[1] += rot.y();
|
|
|
|
movement.mRotation[2] += rot.z();
|
|
|
|
if (std::abs(mDeferredRotation.z()) > 0.0001)
|
|
|
|
{
|
|
|
|
float s = std::sin(mDeferredRotation.z());
|
|
|
|
float c = std::cos(mDeferredRotation.z());
|
|
|
|
float x = movement.mPosition[0];
|
|
|
|
float y = movement.mPosition[1];
|
|
|
|
movement.mPosition[0] = x * c + y * s;
|
|
|
|
movement.mPosition[1] = x * -s + y * c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Camera::rotateCameraToTrackingPtr()
|
|
|
|
{
|
|
|
|
setPitch(-mTrackingPtr.getRefData().getPosition().rot[0] - mDeferredRotation.x());
|
|
|
|
setYaw(-mTrackingPtr.getRefData().getPosition().rot[2] - mDeferredRotation.z());
|
|
|
|
}
|
|
|
|
|
2020-07-18 11:27:53 +00:00
|
|
|
void Camera::instantTransition()
|
2013-03-07 23:12:56 +00:00
|
|
|
{
|
2020-07-18 11:27:53 +00:00
|
|
|
mSkipFocalPointTransition = true;
|
|
|
|
mDeferredRotationDisabled = false;
|
|
|
|
mDeferredRotation = osg::Vec3f();
|
|
|
|
rotateCameraToTrackingPtr();
|
2013-03-07 23:12:56 +00:00
|
|
|
}
|
2013-07-29 14:43:16 +00:00
|
|
|
|
2020-07-18 09:48:46 +00:00
|
|
|
void Camera::calculateDeferredRotation()
|
2013-07-29 14:43:16 +00:00
|
|
|
{
|
2020-07-18 09:48:46 +00:00
|
|
|
MWWorld::Ptr ptr = mTrackingPtr;
|
|
|
|
if (isVanityOrPreviewModeEnabled() || ptr.isEmpty())
|
|
|
|
return;
|
2020-07-18 11:27:53 +00:00
|
|
|
if (mFirstPersonView)
|
2020-07-18 09:48:46 +00:00
|
|
|
{
|
2020-07-18 11:27:53 +00:00
|
|
|
instantTransition();
|
2020-07-18 09:48:46 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
mDeferredRotation.x() = -ptr.getRefData().getPosition().rot[0] - mPitch;
|
|
|
|
mDeferredRotation.z() = -ptr.getRefData().getPosition().rot[2] - mYaw;
|
|
|
|
if (mDeferredRotation.x() > osg::PI)
|
|
|
|
mDeferredRotation.x() -= 2 * osg::PI;
|
|
|
|
if (mDeferredRotation.x() < -osg::PI)
|
|
|
|
mDeferredRotation.x() += 2 * osg::PI;
|
|
|
|
if (mDeferredRotation.z() > osg::PI)
|
|
|
|
mDeferredRotation.z() -= 2 * osg::PI;
|
|
|
|
if (mDeferredRotation.z() < -osg::PI)
|
|
|
|
mDeferredRotation.z() += 2 * osg::PI;
|
2013-07-29 14:43:16 +00:00
|
|
|
}
|
2020-07-18 09:48:46 +00:00
|
|
|
|
2011-01-08 14:11:37 +00:00
|
|
|
}
|