mirror of
				https://github.com/AxioDL/metaforce.git
				synced 2025-10-25 22:10:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			290 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			290 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Camera/CCameraSpline.hpp"
 | |
| #include "CStateManager.hpp"
 | |
| #include "World/CScriptCameraWaypoint.hpp"
 | |
| #include "TCastTo.hpp"
 | |
| 
 | |
| namespace urde
 | |
| {
 | |
| CCameraSpline::CCameraSpline(bool b) : x48_closedLoop(b) {}
 | |
| 
 | |
| void CCameraSpline::CalculateKnots(TUniqueId cameraId, const std::vector<SConnection>& connections, CStateManager& mgr)
 | |
| {
 | |
|     const SConnection* lastConn = nullptr;
 | |
| 
 | |
|     for (const SConnection& conn : connections)
 | |
|     {
 | |
|         if (conn.x0_state == EScriptObjectState::CameraPath && conn.x4_msg == EScriptObjectMessage::Follow)
 | |
|             lastConn = &conn;
 | |
|     }
 | |
| 
 | |
|     if (lastConn)
 | |
|     {
 | |
|         TCastToPtr<CScriptCameraWaypoint> waypoint = mgr.ObjectById(mgr.GetIdForScript(lastConn->x8_objId));
 | |
|         x14_wpTracker.clear();
 | |
|         x14_wpTracker.reserve(4);
 | |
|         while (waypoint)
 | |
|         {
 | |
|             auto search = std::find_if(x14_wpTracker.begin(), x14_wpTracker.end(),
 | |
|                                        [&waypoint](const auto& a) { return a == waypoint->GetUniqueId(); });
 | |
|             if (search == x14_wpTracker.end())
 | |
|             {
 | |
|                 x14_wpTracker.push_back(waypoint->GetUniqueId());
 | |
|                 waypoint = mgr.ObjectById(waypoint->GetRandomNextWaypointId(mgr));
 | |
|             }
 | |
|         }
 | |
|         Reset(x14_wpTracker.size());
 | |
|         x14_wpTracker.clear();
 | |
| 
 | |
|         waypoint = mgr.ObjectById(mgr.GetIdForScript(lastConn->x8_objId));
 | |
|         while (waypoint)
 | |
|         {
 | |
|             auto search = std::find_if(x14_wpTracker.begin(), x14_wpTracker.end(),
 | |
|                                        [&waypoint](const auto& a) { return a == waypoint->GetUniqueId(); });
 | |
|             if (search == x14_wpTracker.end())
 | |
|             {
 | |
|                 x14_wpTracker.push_back(waypoint->GetUniqueId());
 | |
|                 AddKnot(waypoint->GetTranslation(), waypoint->GetTransform().basis[1]);
 | |
|                 waypoint = mgr.ObjectById(waypoint->GetRandomNextWaypointId(mgr));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CCameraSpline::Reset(int size)
 | |
| {
 | |
|     x4_positions.clear();
 | |
|     x24_t.clear();
 | |
|     x34_directions.clear();
 | |
|     if (size != 0)
 | |
|     {
 | |
|         x4_positions.reserve(size);
 | |
|         x24_t.reserve(size);
 | |
|         x34_directions.reserve(size);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void CCameraSpline::AddKnot(const zeus::CVector3f& pos, const zeus::CVector3f& dir)
 | |
| {
 | |
|     x4_positions.push_back(pos);
 | |
|     x34_directions.push_back(dir);
 | |
| }
 | |
| 
 | |
| void CCameraSpline::SetKnotPosition(int idx, const zeus::CVector3f& pos)
 | |
| {
 | |
|     if (idx >= x4_positions.size())
 | |
|         return;
 | |
|     x4_positions[idx] = pos;
 | |
| }
 | |
| 
 | |
| const zeus::CVector3f& CCameraSpline::GetKnotPosition(int idx) const
 | |
| {
 | |
|     if (idx >= x4_positions.size())
 | |
|         return zeus::CVector3f::skZero;
 | |
|     return x4_positions[idx];
 | |
| }
 | |
| 
 | |
| float CCameraSpline::GetKnotT(int idx) const
 | |
| {
 | |
|     if (idx >= x4_positions.size())
 | |
|         return 0.f;
 | |
|     return x24_t[idx];
 | |
| }
 | |
| 
 | |
| float CCameraSpline::CalculateSplineLength()
 | |
| {
 | |
|     float ret = 0.f;
 | |
|     x24_t.clear();
 | |
|     if (x4_positions.size() > 0)
 | |
|     {
 | |
|         zeus::CVector3f prevPoint = x4_positions[0];
 | |
|         float tDiv = 1.f / float(x4_positions.size() - 1);
 | |
|         for (int i=0 ; i<x4_positions.size() ; ++i)
 | |
|         {
 | |
|             float subT = 0.f;
 | |
|             float baseT = i * tDiv;
 | |
|             x24_t.push_back(ret);
 | |
|             while (subT <= tDiv)
 | |
|             {
 | |
|                 subT += tDiv * 0.03125f;
 | |
|                 zeus::CVector3f nextPoint = GetInterpolatedSplinePointByTime(baseT + subT, 1.f);
 | |
|                 zeus::CVector3f delta = nextPoint - prevPoint;
 | |
|                 if (delta.canBeNormalized())
 | |
|                 {
 | |
|                     prevPoint = nextPoint;
 | |
|                     ret += delta.magnitude();
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         x24_t.push_back(ret);
 | |
|         if (x48_closedLoop)
 | |
|         {
 | |
|             zeus::CVector3f delta = x4_positions[0] - x4_positions[x4_positions.size() - 1];
 | |
|             if (delta.canBeNormalized())
 | |
|                 ret += delta.magnitude();
 | |
|         }
 | |
| 
 | |
|         return ret;
 | |
|     }
 | |
|     return 0.f;
 | |
| }
 | |
| 
 | |
| bool CCameraSpline::GetSurroundingPoints(int idx, rstl::reserved_vector<zeus::CVector3f, 4>& positions,
 | |
|                                          rstl::reserved_vector<zeus::CVector3f, 4>& directions) const
 | |
| {
 | |
|     if (x4_positions.size() <= 3 || idx < 0 || idx >= x4_positions.size())
 | |
|         return false;
 | |
| 
 | |
|     if (idx > 0)
 | |
|     {
 | |
|         positions.push_back(x4_positions[idx - 1]);
 | |
|         directions.push_back(x34_directions[idx - 1]);
 | |
|     }
 | |
|     else if (x48_closedLoop)
 | |
|     {
 | |
|         positions.push_back(x4_positions[x4_positions.size() - 1]);
 | |
|         directions.push_back(x34_directions[x4_positions.size() - 1]);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         positions.push_back(x4_positions[0] - (x4_positions[1] - x4_positions[0]));
 | |
|         directions.push_back(x34_directions[0]);
 | |
|     }
 | |
| 
 | |
|     positions.push_back(x4_positions[idx]);
 | |
|     directions.push_back(x34_directions[idx]);
 | |
| 
 | |
|     if (idx + 1 >= x4_positions.size())
 | |
|     {
 | |
|         if (x48_closedLoop)
 | |
|         {
 | |
|             positions.push_back(x4_positions[idx - x4_positions.size()]);
 | |
|             directions.push_back(x34_directions[idx - x4_positions.size()]);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             positions.push_back(x4_positions[x4_positions.size() - 1] -
 | |
|                 (x4_positions[x4_positions.size() - 2] - x4_positions[x4_positions.size() - 1]));
 | |
|             directions.push_back(x34_directions[x4_positions.size() - 1]);
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         positions.push_back(x4_positions[idx + 1]);
 | |
|         directions.push_back(x34_directions[idx + 1]);
 | |
|     }
 | |
| 
 | |
|     if (idx + 2 >= x4_positions.size())
 | |
|     {
 | |
|         if (x48_closedLoop)
 | |
|         {
 | |
|             positions.push_back(x4_positions[idx + 2 - x4_positions.size()]);
 | |
|             directions.push_back(x34_directions[idx + 2 - x4_positions.size()]);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             positions.push_back(x4_positions[x4_positions.size() - 1] -
 | |
|                 (x4_positions[x4_positions.size() - 2] - x4_positions[x4_positions.size() - 1]));
 | |
|             directions.push_back(x34_directions[x4_positions.size() - 1]);
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         positions.push_back(x4_positions[idx + 2]);
 | |
|         directions.push_back(x34_directions[idx + 2]);
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| zeus::CTransform CCameraSpline::GetInterpolatedSplinePointByLength(float pos) const
 | |
| {
 | |
|     if (x4_positions.empty())
 | |
|         return zeus::CTransform();
 | |
| 
 | |
|     int baseIdx = 0;
 | |
|     int i;
 | |
|     for (i=1 ; i<x4_positions.size() ; ++i)
 | |
|     {
 | |
|         if (x24_t[i] > pos)
 | |
|         {
 | |
|             baseIdx = i - 1;
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (i == x4_positions.size())
 | |
|         baseIdx = i - 1;
 | |
| 
 | |
|     if (pos < 0.f)
 | |
|         baseIdx = 0;
 | |
| 
 | |
|     if (pos >= x44_length)
 | |
|     {
 | |
|         if (x48_closedLoop)
 | |
|         {
 | |
|             pos -= x44_length;
 | |
|             baseIdx = 0;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             baseIdx = x4_positions.size() - 2;
 | |
|             pos = x44_length;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     float range;
 | |
|     if (baseIdx == x4_positions.size() - 1)
 | |
|     {
 | |
|         if (x48_closedLoop)
 | |
|             range = x44_length - x24_t[baseIdx];
 | |
|         else
 | |
|             range = x44_length - x24_t[x4_positions.size() - 2];
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         range = x24_t[baseIdx + 1] - x24_t[baseIdx];
 | |
|     }
 | |
| 
 | |
|     float t = zeus::clamp(0.f, (pos - x24_t[baseIdx]) / range, 1.f);
 | |
| 
 | |
|     rstl::reserved_vector<zeus::CVector3f, 4> positions;
 | |
|     rstl::reserved_vector<zeus::CVector3f, 4> directions;
 | |
|     if (GetSurroundingPoints(baseIdx, positions, directions))
 | |
|     {
 | |
|         float f1 = zeus::clamp(-1.f, directions[1].dot(directions[2]), 1.f);
 | |
|         if (f1 >= 1.f)
 | |
|         {
 | |
|             zeus::CTransform ret = zeus::lookAt(zeus::CVector3f::skZero, directions[2]);
 | |
|             ret.origin = zeus::getCatmullRomSplinePoint(positions[0], positions[1], positions[2], positions[3], t);
 | |
|             return ret;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             zeus::CTransform ret = zeus::lookAt(zeus::CVector3f::skZero,
 | |
|                 zeus::CQuaternion::lookAt(directions[1], directions[2], std::acos(f1) * t).transform(directions[1]));
 | |
|             ret.origin = zeus::getCatmullRomSplinePoint(positions[0], positions[1], positions[2], positions[3], t);
 | |
|             return ret;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return zeus::CTransform();
 | |
| }
 | |
| 
 | |
| zeus::CVector3f CCameraSpline::GetInterpolatedSplinePointByTime(float time, float range) const
 | |
| {
 | |
|     if (x4_positions.empty())
 | |
|         return {};
 | |
| 
 | |
|     rstl::reserved_vector<zeus::CVector3f, 4> positions;
 | |
|     rstl::reserved_vector<zeus::CVector3f, 4> directions;
 | |
|     float rangeFac = range / float(x4_positions.size() - 1);
 | |
|     int baseIdx = std::min(int(x4_positions.size() - 1), int(time / rangeFac));
 | |
|     if (GetSurroundingPoints(baseIdx, positions, directions))
 | |
|         return zeus::getCatmullRomSplinePoint(positions[0], positions[1], positions[2], positions[3],
 | |
|                                               (time - float(baseIdx) * rangeFac) / rangeFac);
 | |
| 
 | |
|     return {};
 | |
| }
 | |
| }
 |