metaforce/Runtime/Camera/CGameCamera.cpp

160 lines
4.3 KiB
C++
Raw Normal View History

2016-04-24 04:03:30 +00:00
#include "CGameCamera.hpp"
2016-09-14 05:45:46 +00:00
#include "CStateManager.hpp"
#include "Camera/CCameraManager.hpp"
2016-04-24 04:03:30 +00:00
#include "World/CActorParameters.hpp"
namespace urde
{
2017-11-13 06:19:18 +00:00
CGameCamera::CGameCamera(TUniqueId uid, bool active, std::string_view name, const CEntityInfo& info,
2017-10-08 02:58:13 +00:00
const zeus::CTransform& xf, float fovy, float znear, float zfar, float aspect,
TUniqueId watchedId, bool disableInput, u32 controllerIdx)
2016-12-29 21:38:59 +00:00
: CActor(uid, active, name, info, xf, CModelData::CModelDataNull(), CMaterialList(EMaterialTypes::Unknown),
2016-09-14 05:45:46 +00:00
CActorParameters::None(), kInvalidUniqueId)
2017-10-08 02:58:13 +00:00
, xe8_watchedObject(watchedId)
2016-09-14 05:45:46 +00:00
, x12c_(xf)
2016-10-31 22:56:44 +00:00
, x15c_currentFov(fovy)
2016-09-14 05:45:46 +00:00
, x160_znear(znear)
, x164_zfar(zfar)
, x168_aspect(aspect)
2017-03-24 05:30:16 +00:00
, x16c_controllerIdx(controllerIdx)
2016-09-14 05:45:46 +00:00
, x170_24_perspDirty(true)
, x170_25_disablesInput(disableInput)
2016-09-14 05:45:46 +00:00
, x180_(fovy)
2016-10-31 22:56:44 +00:00
, x184_fov(fovy)
2016-04-24 04:03:30 +00:00
{
2016-09-14 05:45:46 +00:00
xe7_29_ = false;
}
void CGameCamera::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId uid, CStateManager& mgr)
{
2017-06-12 04:23:34 +00:00
if (msg == EScriptObjectMessage::UpdateSplashInhabitant)
2016-09-14 05:45:46 +00:00
{
mgr.GetCameraManager()->SetInsideFluid(true, uid);
return;
}
2017-06-12 04:23:34 +00:00
else if (msg == EScriptObjectMessage::RemoveSplashInhabitant)
2016-09-14 05:45:46 +00:00
{
mgr.GetCameraManager()->SetInsideFluid(false, kInvalidUniqueId);
return;
}
CActor::AcceptScriptMsg(msg, uid, mgr);
}
void CGameCamera::SetActive(bool active)
{
CActor::SetActive(active);
xe7_29_ = false;
}
zeus::CMatrix4f CGameCamera::GetPerspectiveMatrix() const
{
if (x170_24_perspDirty)
{
const_cast<CGameCamera*>(this)->xec_perspectiveMatrix =
2016-10-31 22:56:44 +00:00
CGraphics::CalculatePerspectiveMatrix(x15c_currentFov, x168_aspect, x160_znear, x164_zfar, false);
2016-09-14 05:45:46 +00:00
const_cast<CGameCamera*>(this)->x170_24_perspDirty = false;
}
return xec_perspectiveMatrix;
}
zeus::CVector3f CGameCamera::ConvertToScreenSpace(const zeus::CVector3f& v) const
{
zeus::CVector3f rVec = x34_transform.transposeRotate(v - x34_transform.origin);
if (rVec.x == 0.f && rVec.y == 0.f && rVec.z == 0.f)
return {-1.f, -1.f, 1.f};
zeus::CMatrix4f mtx = GetPerspectiveMatrix();
return mtx.multiplyOneOverW(rVec);
}
zeus::CTransform CGameCamera::ValidateCameraTransform(const zeus::CTransform& a, const zeus::CTransform& b)
{
zeus::CTransform xfCpy(a);
constexpr double epsilon = FLT_EPSILON * 1000.f;
if ((a.rightVector().magnitude() - 1.f) >= epsilon || (a.frontVector().magnitude() - 1.f) >= epsilon ||
(a.upVector().magnitude() - 1.f) >= epsilon)
xfCpy.orthonormalize();
2017-02-12 03:51:59 +00:00
float f2 = zeus::CVector3f::skUp.x + a.upVector().x * a.upVector().y * a.upVector().z *
zeus::CVector3f::skUp.y + zeus::CVector3f::skUp.z;
2016-09-14 05:45:46 +00:00
if (std::fabs(f2) > 1.0f)
f2 = (f2 >= -0.f ? -1.0f : 1.0f);
if (std::fabs(f2) > 0.999f)
xfCpy = b;
if (xfCpy.upVector().z < -0.2f)
xfCpy = zeus::CQuaternion::fromAxisAngle(xfCpy.frontVector(), M_PIF).toTransform() * xfCpy;
if (std::fabs(xfCpy.rightVector().z - 0.f) >= 0.000009f && std::fabs(xfCpy.upVector().z - 0.f) > 0.000009f)
{
if (xfCpy.frontVector().canBeNormalized())
xfCpy = zeus::lookAt(zeus::CUnitVector3f(xfCpy.frontVector(), true), zeus::CVector3f::skZero);
else
xfCpy = b;
}
xfCpy.origin = a.origin;
return xfCpy;
2016-04-24 04:03:30 +00:00
}
2016-10-31 22:56:44 +00:00
void CGameCamera::UpdatePerspective(float dt)
{
if (x174_ > 0.f)
{
2016-10-31 22:56:44 +00:00
x174_ -= dt;
return;
}
if (x178_ <= 0.f)
return;
2016-10-31 22:56:44 +00:00
x178_ -= dt;
if (x178_ > 0.f)
{
2016-10-31 22:56:44 +00:00
x15c_currentFov = x184_fov;
x170_24_perspDirty = true;
}
else
{
2016-10-31 22:56:44 +00:00
x15c_currentFov = zeus::clamp(0.f, (dt / x17c_), 1.f) + ((x180_ - x184_fov) * x184_fov);
x170_24_perspDirty = true;
}
}
2017-08-18 01:30:15 +00:00
void CGameCamera::SetFovInterpolation(float start, float fov, float time, float f4)
{
2017-08-18 01:30:15 +00:00
if (time < 0.f)
{
2016-10-31 22:56:44 +00:00
x15c_currentFov = fov;
x170_24_perspDirty = true;
2016-10-31 22:56:44 +00:00
x184_fov = fov;
x178_ = x174_ = 0.f;
}
else
{
x174_ = std::max(0.f, f4);
2017-08-18 01:30:15 +00:00
x17c_ = time;
x178_ = time;
x180_ = start;
2016-10-31 22:56:44 +00:00
x184_fov = fov;
2017-08-18 01:30:15 +00:00
x15c_currentFov = start;
x170_24_perspDirty = true;
}
}
2017-10-07 05:32:11 +00:00
void CGameCamera::SkipFovInterpolation()
{
if (x178_ > 0)
{
2016-10-31 22:56:44 +00:00
x15c_currentFov = x184_fov;
x170_24_perspDirty = true;
}
x178_ = x174_ = 0.f;
}
2016-04-24 04:03:30 +00:00
}