1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 23:23:52 +00:00
openmw-tes3mp/apps/openmw/mwrender/camera.cpp

403 lines
11 KiB
C++
Raw Permalink Normal View History

2013-04-29 12:50:40 +00:00
#include "camera.hpp"
#include <OgreSceneNode.h>
#include <OgreCamera.h>
#include <OgreSceneManager.h>
#include <OgreTagPoint.h>
#include "../mwbase/environment.hpp"
#include "../mwbase/windowmanager.hpp"
#include "../mwworld/ptr.hpp"
#include "../mwworld/refdata.hpp"
2012-08-14 16:33:29 +00:00
#include "npcanimation.hpp"
namespace MWRender
{
2013-04-29 12:50:40 +00:00
Camera::Camera (Ogre::Camera *camera)
2012-08-12 11:50:37 +00:00
: mCamera(camera),
mCameraNode(NULL),
mCameraPosNode(NULL),
mAnimation(NULL),
mFirstPersonView(true),
2012-08-13 22:36:18 +00:00
mPreviewMode(false),
mFreeLook(true),
mNearest(30.f),
2013-07-29 14:45:35 +00:00
mFurthest(800.f),
mIsNearest(false),
mHeight(124.f),
mCameraDistance(192.f),
mDistanceAdjusted(false),
mVanityToggleQueued(false),
mViewModeToggleQueued(false)
{
2012-08-14 10:37:48 +00:00
mVanity.enabled = false;
mVanity.allowed = true;
mPreviewCam.pitch = 0.f;
2012-08-13 22:36:18 +00:00
mPreviewCam.yaw = 0.f;
mPreviewCam.offset = 400.f;
mMainCam.pitch = 0.f;
2013-07-31 16:46:32 +00:00
mMainCam.yaw = 0.f;
mMainCam.offset = 400.f;
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
{
}
2013-05-17 20:53:43 +00:00
void Camera::reset()
{
togglePreviewMode(false);
toggleVanityMode(false);
if (!mFirstPersonView)
toggleViewMode();
}
2013-04-29 12:50:40 +00:00
void Camera::rotateCamera(const Ogre::Vector3 &rot, bool adjust)
{
2012-08-12 11:50:37 +00:00
if (adjust) {
2012-08-13 22:36:18 +00:00
setYaw(getYaw() + rot.z);
setPitch(getPitch() + rot.x);
2012-08-12 11:50:37 +00:00
} else {
2012-08-13 22:36:18 +00:00
setYaw(rot.z);
setPitch(rot.x);
2012-08-12 11:50:37 +00:00
}
2013-04-27 08:24:36 +00:00
Ogre::Quaternion xr(Ogre::Radian(getPitch() + Ogre::Math::HALF_PI), Ogre::Vector3::UNIT_X);
Ogre::Quaternion orient = xr;
if (mVanity.enabled || mPreviewMode) {
Ogre::Quaternion zr(Ogre::Radian(getYaw()), Ogre::Vector3::UNIT_Z);
orient = zr * xr;
2012-08-17 08:08:31 +00:00
}
if (isFirstPerson())
mCamera->getParentNode()->setOrientation(orient);
else
mCameraNode->setOrientation(orient);
}
2013-04-29 12:50:40 +00:00
const std::string &Camera::getHandle() const
{
return mTrackingPtr.getRefData().getHandle();
2012-08-12 11:50:37 +00:00
}
Ogre::SceneNode* Camera::attachTo(const MWWorld::Ptr &ptr)
2012-08-12 11:50:37 +00:00
{
mTrackingPtr = ptr;
Ogre::SceneNode *node = mTrackingPtr.getRefData().getBaseNode()->createChildSceneNode(Ogre::Vector3(0.0f, 0.0f, mHeight));
node->setInheritScale(false);
Ogre::SceneNode *posNode = node->createChildSceneNode();
posNode->setInheritScale(false);
if(mCameraNode)
{
node->setOrientation(mCameraNode->getOrientation());
posNode->setPosition(mCameraPosNode->getPosition());
mCameraNode->getCreator()->destroySceneNode(mCameraNode);
mCameraNode->getCreator()->destroySceneNode(mCameraPosNode);
}
mCameraNode = node;
mCameraPosNode = posNode;
if (!isFirstPerson())
{
mCamera->detachFromParent();
mCameraPosNode->attachObject(mCamera);
}
return mCameraPosNode;
}
void Camera::setPosition(const Ogre::Vector3& position)
{
mCameraPosNode->setPosition(position);
}
void Camera::setPosition(float x, float y, float z)
{
setPosition(Ogre::Vector3(x,y,z));
2012-08-12 11:50:37 +00:00
}
void Camera::update(float duration, bool paused)
2012-08-12 11:50:37 +00:00
{
if (mAnimation->upperBodyReady())
{
// Now process the view changes we queued earlier
if (mVanityToggleQueued)
{
toggleVanityMode(!mVanity.enabled);
mVanityToggleQueued = false;
}
if (mViewModeToggleQueued)
{
togglePreviewMode(false);
toggleViewMode();
mViewModeToggleQueued = false;
}
}
if (paused)
return;
2012-10-01 07:24:44 +00:00
2012-08-27 17:18:55 +00:00
// only show the crosshair in game mode and in first person mode.
MWBase::WindowManager *wm = MWBase::Environment::get().getWindowManager();
wm->showCrosshair(!wm->isGuiMode() && (mFirstPersonView && !mVanity.enabled && !mPreviewMode));
2012-08-27 17:18:55 +00:00
if(mVanity.enabled)
{
2012-08-12 14:35:35 +00:00
Ogre::Vector3 rot(0.f, 0.f, 0.f);
rot.z = Ogre::Degree(3.f * duration).valueRadians();
2012-08-13 04:37:32 +00:00
rotateCamera(rot, true);
2012-08-12 11:50:37 +00:00
}
}
void Camera::toggleViewMode(bool force)
2012-08-12 11:50:37 +00:00
{
// Changing the view will stop all playing animations, so if we are playing
// anything important, queue the view change for later
if (!mAnimation->upperBodyReady() && !force)
{
mViewModeToggleQueued = true;
return;
}
else
mViewModeToggleQueued = false;
2012-08-12 11:50:37 +00:00
mFirstPersonView = !mFirstPersonView;
processViewChange();
2012-08-12 11:50:37 +00:00
if (mFirstPersonView) {
setPosition(0.f, 0.f, 0.f);
2012-08-12 11:50:37 +00:00
} else {
setPosition(0.f, 0.f, mCameraDistance);
2012-08-12 11:50:37 +00:00
}
}
2012-08-14 10:37:48 +00:00
2013-04-29 12:50:40 +00:00
void Camera::allowVanityMode(bool allow)
2012-08-14 10:37:48 +00:00
{
2013-04-27 08:24:36 +00:00
if (!allow && mVanity.enabled)
2012-08-14 10:37:48 +00:00
toggleVanityMode(false);
mVanity.allowed = allow;
}
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
{
// Changing the view will stop all playing animations, so if we are playing
// anything important, queue the view change for later
2014-10-02 12:23:13 +00:00
if (isFirstPerson() && !mAnimation->upperBodyReady())
{
mVanityToggleQueued = true;
return false;
}
2013-04-27 08:24:36 +00:00
if(!mVanity.allowed && enable)
2012-08-14 10:37:48 +00:00
return false;
2013-04-27 08:24:36 +00:00
if(mVanity.enabled == enable)
2012-08-14 10:37:48 +00:00
return true;
mVanity.enabled = enable;
2012-08-12 14:35:35 +00:00
processViewChange();
float offset = mPreviewCam.offset;
2012-08-12 14:35:35 +00:00
Ogre::Vector3 rot(0.f, 0.f, 0.f);
2012-08-14 10:37:48 +00:00
if (mVanity.enabled) {
2012-08-12 14:35:35 +00:00
rot.x = Ogre::Degree(-30.f).valueRadians();
mMainCam.offset = mCameraPosNode->getPosition().z;
2012-08-12 14:35:35 +00:00
} else {
2012-08-13 22:36:18 +00:00
rot.x = getPitch();
offset = mMainCam.offset;
}
2012-08-13 22:36:18 +00:00
rot.z = getYaw();
2013-04-27 08:24:36 +00:00
setPosition(0.f, 0.f, offset);
2012-08-13 04:37:32 +00:00
rotateCamera(rot, false);
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
{
if (mFirstPersonView && !mAnimation->upperBodyReady())
return;
2013-04-27 08:24:36 +00:00
if(mPreviewMode == enable)
2012-08-13 22:36:18 +00:00
return;
2013-04-27 08:24:36 +00:00
2012-08-13 22:36:18 +00:00
mPreviewMode = enable;
processViewChange();
2013-04-27 08:24:36 +00:00
float offset = mCameraPosNode->getPosition().z;
2012-08-14 10:37:48 +00:00
if (mPreviewMode) {
mMainCam.offset = offset;
offset = mPreviewCam.offset;
} else {
mPreviewCam.offset = offset;
offset = mMainCam.offset;
}
2013-04-27 08:24:36 +00:00
setPosition(0.f, 0.f, offset);
2012-08-12 11:50:37 +00:00
}
2012-08-12 14:35:35 +00:00
void Camera::setSneakOffset(float offset)
{
if(mAnimation)
mAnimation->addFirstPersonOffset(Ogre::Vector3(0.f, 0.f, -offset));
}
2013-04-29 12:50:40 +00:00
float Camera::getYaw()
2012-08-12 14:35:35 +00:00
{
2013-04-27 08:24:36 +00:00
if(mVanity.enabled || mPreviewMode)
2012-08-13 22:36:18 +00:00
return mPreviewCam.yaw;
return mMainCam.yaw;
}
2012-08-12 14:35:35 +00:00
2013-04-29 12:50:40 +00:00
void Camera::setYaw(float angle)
2012-08-13 22:36:18 +00:00
{
if (angle > Ogre::Math::PI) {
angle -= Ogre::Math::TWO_PI;
} else if (angle < -Ogre::Math::PI) {
angle += Ogre::Math::TWO_PI;
}
2012-08-14 10:37:48 +00:00
if (mVanity.enabled || mPreviewMode) {
2012-08-13 22:36:18 +00:00
mPreviewCam.yaw = angle;
} else {
mMainCam.yaw = angle;
}
2012-08-12 14:35:35 +00:00
}
2013-04-29 12:50:40 +00:00
float Camera::getPitch()
2012-08-12 14:35:35 +00:00
{
2012-08-14 10:37:48 +00:00
if (mVanity.enabled || mPreviewMode) {
2012-08-13 22:36:18 +00:00
return mPreviewCam.pitch;
}
return mMainCam.pitch;
}
2012-08-12 14:35:35 +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;
float limit = Ogre::Math::HALF_PI - epsilon;
2013-04-27 08:24:36 +00:00
if(mPreviewMode)
limit /= 2;
if(angle > limit)
angle = limit;
2013-04-27 08:24:36 +00:00
else if(angle < -limit)
angle = -limit;
2013-04-27 08:24:36 +00:00
2012-08-14 10:37:48 +00:00
if (mVanity.enabled || mPreviewMode) {
2012-08-13 22:36:18 +00:00
mPreviewCam.pitch = angle;
} else {
mMainCam.pitch = angle;
2012-08-12 14:35:35 +00:00
}
}
2012-08-14 10:37:48 +00:00
float Camera::getCameraDistance() const
{
return mCameraPosNode->getPosition().z;
}
2013-04-29 12:50:40 +00:00
void Camera::setCameraDistance(float dist, bool adjust, bool override)
2012-08-14 10:37:48 +00:00
{
2013-04-27 08:24:36 +00:00
if(mFirstPersonView && !mPreviewMode && !mVanity.enabled)
return;
2013-04-27 08:24:36 +00:00
mIsNearest = false;
Ogre::Vector3 v(0.f, 0.f, dist);
if (adjust) {
v += mCameraPosNode->getPosition();
}
if (v.z >= mFurthest) {
v.z = mFurthest;
} else if (!override && v.z < 10.f) {
v.z = 10.f;
} else if (override && v.z <= mNearest) {
v.z = mNearest;
mIsNearest = true;
}
setPosition(v);
2012-08-14 10:37:48 +00:00
if (override) {
if (mVanity.enabled || mPreviewMode) {
mPreviewCam.offset = v.z;
} else if (!mFirstPersonView) {
2012-08-14 10:37:48 +00:00
mCameraDistance = v.z;
}
} else {
mDistanceAdjusted = true;
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()
{
if (mDistanceAdjusted) {
if (mVanity.enabled || mPreviewMode) {
setPosition(0, 0, mPreviewCam.offset);
} else if (!mFirstPersonView) {
setPosition(0, 0, mCameraDistance);
}
}
mDistanceAdjusted = false;
}
2013-04-29 12:50:40 +00:00
void Camera::setAnimation(NpcAnimation *anim)
2012-08-15 11:17:35 +00:00
{
// If we're switching to a new NpcAnimation, ensure the old one is
// using a normal view mode
if(mAnimation && mAnimation != anim)
{
mAnimation->setViewMode(NpcAnimation::VM_Normal);
mAnimation->detachObjectFromBone(mCamera);
}
2012-08-15 11:17:35 +00:00
mAnimation = anim;
processViewChange();
2012-08-15 11:17:35 +00:00
}
void Camera::processViewChange()
2012-08-14 16:33:29 +00:00
{
mAnimation->detachObjectFromBone(mCamera);
mCamera->detachFromParent();
if(isFirstPerson())
{
mAnimation->setViewMode(NpcAnimation::VM_FirstPerson);
Ogre::TagPoint *tag = mAnimation->attachObjectToBone("Head", mCamera);
tag->setInheritOrientation(false);
}
else
{
mAnimation->setViewMode(NpcAnimation::VM_Normal);
mCameraPosNode->attachObject(mCamera);
}
rotateCamera(Ogre::Vector3(getPitch(), 0.f, getYaw()), false);
2012-08-14 16:33:29 +00:00
}
2013-07-23 08:37:41 +00:00
void Camera::getPosition(Ogre::Vector3 &focal, Ogre::Vector3 &camera)
{
mCamera->getParentSceneNode()->needUpdate(true);
2013-07-23 08:37:41 +00:00
camera = mCamera->getRealPosition();
focal = mCameraNode->_getDerivedPosition();
}
2012-08-17 08:08:31 +00:00
2013-04-29 12:50:40 +00:00
void Camera::togglePlayerLooking(bool enable)
{
mFreeLook = enable;
}
2013-04-29 12:50:40 +00:00
bool Camera::isVanityOrPreviewModeEnabled()
{
return mPreviewMode || mVanity.enabled;
}
bool Camera::isNearest()
{
return mIsNearest;
}
}