mirror of
				https://github.com/AxioDL/metaforce.git
				synced 2025-10-26 18:10:23 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			278 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			278 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #pragma once
 | |
| 
 | |
| #include <cmath>
 | |
| #include <memory>
 | |
| #include <vector>
 | |
| 
 | |
| #include "Runtime/Camera/CCameraSpline.hpp"
 | |
| #include "Runtime/Camera/CGameCamera.hpp"
 | |
| 
 | |
| #include <zeus/CAABox.hpp>
 | |
| #include <zeus/CTransform.hpp>
 | |
| #include <zeus/CVector3f.hpp>
 | |
| 
 | |
| namespace metaforce {
 | |
| class CPlayer;
 | |
| 
 | |
| class CCameraSpring {
 | |
|   float x0_k;
 | |
|   float x4_k2Sqrt;
 | |
|   float x8_max;
 | |
|   float xc_tardis;
 | |
|   float x10_dx = 0.f;
 | |
| 
 | |
| public:
 | |
|   CCameraSpring(float k, float max, float tardis)
 | |
|   : x0_k(k), x4_k2Sqrt(2.f * std::sqrt(k)), x8_max(max), xc_tardis(tardis) {}
 | |
|   void Reset();
 | |
|   float ApplyDistanceSpringNoMax(float targetX, float curX, float dt);
 | |
|   float ApplyDistanceSpring(float targetX, float curX, float dt);
 | |
| };
 | |
| 
 | |
| class CCameraCollider {
 | |
|   friend class CBallCamera;
 | |
|   float x4_radius;
 | |
|   zeus::CVector3f x8_lastLocalPos;
 | |
|   zeus::CVector3f x14_localPos;
 | |
|   zeus::CVector3f x20_scaledWorldPos;
 | |
|   zeus::CVector3f x2c_lastWorldPos;
 | |
|   CCameraSpring x38_spring;
 | |
|   u32 x4c_occlusionCount = 0;
 | |
|   float x50_scale;
 | |
| 
 | |
| public:
 | |
|   CCameraCollider(float radius, const zeus::CVector3f& vec, const CCameraSpring& spring, float scale)
 | |
|   : x4_radius(radius)
 | |
|   , x8_lastLocalPos(vec)
 | |
|   , x14_localPos(vec)
 | |
|   , x20_scaledWorldPos(vec)
 | |
|   , x2c_lastWorldPos(vec)
 | |
|   , x38_spring(spring)
 | |
|   , x50_scale(scale) {}
 | |
| };
 | |
| 
 | |
| class CBallCamera : public CGameCamera {
 | |
| public:
 | |
|   DEFINE_ENTITY
 | |
|   enum class EBallCameraState { Default, One, Chase, Boost, ToBall, FromBall };
 | |
|   enum class EBallCameraBehaviour {
 | |
|     Default,
 | |
|     FreezeLookPosition, // Unused
 | |
|     HintBallToCam,
 | |
|     HintInitializePosition,
 | |
|     HintFixedPosition,
 | |
|     HintFixedTransform,
 | |
|     PathCameraDesiredPos, // Unused
 | |
|     PathCamera,
 | |
|     SpindleCamera
 | |
|   };
 | |
|   enum class ESplineState { Invalid, Nav, Arc };
 | |
| 
 | |
| private:
 | |
|   struct SFailsafeState {
 | |
|     zeus::CTransform x0_playerXf;
 | |
|     zeus::CTransform x30_camXf;
 | |
|     zeus::CVector3f x60_lookPos;
 | |
|     zeus::CVector3f x6c_behindPos;
 | |
|     zeus::CVector3f x78_;
 | |
|     zeus::CVector3f x84_playerPos;
 | |
|     std::vector<zeus::CVector3f> x90_splinePoints;
 | |
|   };
 | |
| 
 | |
|   EBallCameraBehaviour x188_behaviour = EBallCameraBehaviour::Default;
 | |
|   bool x18c_24_ : 1 = true;
 | |
|   bool x18c_25_chaseAllowed : 1 = true;
 | |
|   bool x18c_26_boostAllowed : 1 = true;
 | |
|   bool x18c_27_obscureAvoidance : 1 = true;
 | |
|   bool x18c_28_volumeCollider : 1 = true;
 | |
|   bool x18c_29_clampAttitude : 1 = false;
 | |
|   bool x18c_30_clampAzimuth : 1 = false;
 | |
|   bool x18c_31_clearLOS : 1 = true;
 | |
|   bool x18d_24_prevClearLOS : 1 = true;
 | |
|   bool x18d_25_avoidGeometryFull : 1 = false;
 | |
|   bool x18d_26_lookAtBall : 1 = false;
 | |
|   bool x18d_27_forceProcessing : 1 = false;
 | |
|   bool x18d_28_obtuseDirection : 1 = false;
 | |
|   bool x18d_29_noElevationInterp : 1 = false;
 | |
|   bool x18d_30_directElevation : 1 = false;
 | |
|   bool x18d_31_overrideLookDir : 1 = false;
 | |
|   bool x18e_24_noElevationVelClamp : 1 = false;
 | |
|   bool x18e_25_noSpline : 1 = false;
 | |
|   bool x18e_26_ : 1 = false;
 | |
|   bool x18e_27_nearbyDoorClosed : 1 = false;
 | |
|   bool x18e_28_nearbyDoorClosing : 1 = false;
 | |
|   float x190_curMinDistance;
 | |
|   float x194_targetMinDistance;
 | |
|   float x198_maxDistance;
 | |
|   float x19c_backwardsDistance;
 | |
|   float x1a0_elevation;
 | |
|   float x1a4_curAnglePerSecond;
 | |
|   float x1a8_targetAnglePerSecond;
 | |
|   float x1ac_attitudeRange = zeus::degToRad(89.f);
 | |
|   float x1b0_azimuthRange = zeus::degToRad(89.f);
 | |
|   zeus::CVector3f x1b4_lookAtOffset;
 | |
|   zeus::CVector3f x1c0_lookPosAhead;
 | |
|   zeus::CVector3f x1cc_fixedLookPos;
 | |
|   zeus::CVector3f x1d8_lookPos;
 | |
|   zeus::CTransform x1e4_nextLookXf;
 | |
|   CCameraSpring x214_ballCameraSpring;
 | |
|   CCameraSpring x228_ballCameraCentroidSpring;
 | |
|   CCameraSpring x23c_ballCameraLookAtSpring;
 | |
|   CCameraSpring x250_ballCameraCentroidDistanceSpring;
 | |
|   std::vector<CCameraCollider> x264_smallColliders;
 | |
|   std::vector<CCameraCollider> x274_mediumColliders;
 | |
|   std::vector<CCameraCollider> x284_largeColliders;
 | |
|   zeus::CVector3f x294_dampedPos;
 | |
|   zeus::CVector3f x2a0_smallCentroid = zeus::skUp;
 | |
|   zeus::CVector3f x2ac_mediumCentroid = zeus::skUp;
 | |
|   zeus::CVector3f x2b8_largeCentroid = zeus::skUp;
 | |
|   int x2c4_smallCollidersObsCount = 0;
 | |
|   int x2c8_mediumCollidersObsCount = 0;
 | |
|   int x2cc_largeCollidersObsCount = 0;
 | |
|   int x2d0_smallColliderIt = 0;
 | |
|   int x2d4_mediumColliderIt = 0;
 | |
|   int x2d8_largeColliderIt = 0;
 | |
|   zeus::CVector3f x2dc_prevBallPos;
 | |
|   float x2e8_ballVelFlat = 0.f;
 | |
|   float x2ec_maxBallVel = 0.f;
 | |
|   zeus::CVector3f x2f0_ballDelta;
 | |
|   zeus::CVector3f x2fc_ballDeltaFlat;
 | |
|   float x308_speedFactor = 0.f;
 | |
|   float x30c_speedingTime = 0.f;
 | |
|   zeus::CVector3f x310_idealLookVec;
 | |
|   zeus::CVector3f x31c_predictedLookPos;
 | |
|   u32 x328_avoidGeomCycle = 0;
 | |
|   float x32c_colliderMag = 1.f;
 | |
|   float x330_clearColliderThreshold = 0.2f;
 | |
|   zeus::CAABox x334_collidersAABB = zeus::skNullBox;
 | |
|   float x34c_obscuredTime = 0.f;
 | |
|   CMaterialList x350_obscuringMaterial = {EMaterialTypes::NoStepLogic};
 | |
|   float x358_unobscureMag = 0.f;
 | |
|   zeus::CVector3f x35c_splineIntermediatePos;
 | |
|   TUniqueId x368_obscuringObjectId = kInvalidUniqueId;
 | |
|   ESplineState x36c_splineState = ESplineState::Invalid;
 | |
|   bool x370_24_reevalSplineEnd : 1 = false;
 | |
|   float x374_splineCtrl = 0.f;
 | |
|   float x378_splineCtrlRange;
 | |
|   CCameraSpline x37c_camSpline{false};
 | |
|   CMaterialList x3c8_collisionExcludeList = {EMaterialTypes::NoStepLogic};
 | |
|   bool x3d0_24_camBehindFloorOrWall : 1 = false;
 | |
|   float x3d4_elevInterpTimer = 0.f;
 | |
|   float x3d8_elevInterpStart = 0.f;
 | |
|   TUniqueId x3dc_tooCloseActorId = kInvalidUniqueId;
 | |
|   float x3e0_tooCloseActorDist = 10000.f;
 | |
|   bool x3e4_pendingFailsafe = false;
 | |
|   float x3e8_ = 0.f;
 | |
|   float x3ec_ = 0.f;
 | |
|   float x3f0_ = 0.f;
 | |
|   float x3f4_ = 2.f;
 | |
|   float x3f8_ = 0.f;
 | |
|   float x3fc_ = 0.f;
 | |
|   EBallCameraState x400_state = EBallCameraState::Default;
 | |
|   float x404_chaseElevation;
 | |
|   float x408_chaseDistance;
 | |
|   float x40c_chaseAnglePerSecond;
 | |
|   zeus::CVector3f x410_chaseLookAtOffset;
 | |
|   CCameraSpring x41c_ballCameraChaseSpring;
 | |
|   float x430_boostElevation;
 | |
|   float x434_boostDistance;
 | |
|   float x438_boostAnglePerSecond;
 | |
|   zeus::CVector3f x43c_boostLookAtOffset;
 | |
|   CCameraSpring x448_ballCameraBoostSpring;
 | |
|   zeus::CVector3f x45c_overrideBallToCam;
 | |
|   float x468_conservativeDoorCamDistance;
 | |
|   TUniqueId x46c_collisionActorId = kInvalidUniqueId;
 | |
|   float x470_clampVelTimer = 0.f;
 | |
|   float x474_clampVelRange = 0.f;
 | |
|   u32 x478_shortMoveCount = 0;
 | |
|   std::unique_ptr<SFailsafeState> x47c_failsafeState;
 | |
|   std::unique_ptr<u32> x480_;
 | |
| 
 | |
|   void SetupColliders(std::vector<CCameraCollider>& out, float xMag, float zMag, float radius, int count, float k,
 | |
|                       float max, float startAngle);
 | |
|   void BuildSplineNav(CStateManager& mgr);
 | |
|   void BuildSplineArc(CStateManager& mgr);
 | |
|   bool ShouldResetSpline(CStateManager& mgr) const;
 | |
|   void UpdatePlayerMovement(float dt, CStateManager& mgr);
 | |
|   void UpdateTransform(const zeus::CVector3f& lookDir, const zeus::CVector3f& pos, float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f ConstrainYawAngle(const CPlayer& player, float distance, float yawSpeed, float dt,
 | |
|                                     CStateManager& mgr) const;
 | |
|   void CheckFailsafe(float dt, CStateManager& mgr);
 | |
|   void UpdateObjectTooCloseId(CStateManager& mgr);
 | |
|   void UpdateAnglePerSecond(float dt);
 | |
|   void UpdateUsingPathCameras(float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f GetFixedLookTarget(const zeus::CVector3f& hintToLookDir, CStateManager& mgr) const;
 | |
|   void UpdateUsingFixedCameras(float dt, CStateManager& mgr);
 | |
|   [[nodiscard]] zeus::CVector3f ComputeVelocity(const zeus::CVector3f& curVel, const zeus::CVector3f& posDelta) const;
 | |
|   zeus::CVector3f TweenVelocity(const zeus::CVector3f& curVel, const zeus::CVector3f& newVel, float rate, float dt);
 | |
|   zeus::CVector3f MoveCollisionActor(const zeus::CVector3f& pos, float dt, CStateManager& mgr);
 | |
|   void UpdateUsingFreeLook(float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f InterpolateCameraElevation(const zeus::CVector3f& camPos, float dt);
 | |
|   [[nodiscard]] zeus::CVector3f CalculateCollidersCentroid(const std::vector<CCameraCollider>& colliderList,
 | |
|                                                            int numObscured) const;
 | |
|   zeus::CVector3f ApplyColliders();
 | |
|   void UpdateColliders(const zeus::CTransform& xf, std::vector<CCameraCollider>& colliderList, int& it, int count,
 | |
|                        float tolerance, const EntityList& nearList, float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f AvoidGeometry(const zeus::CTransform& xf, const EntityList& nearList, float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f AvoidGeometryFull(const zeus::CTransform& xf, const EntityList& nearList, float dt,
 | |
|                                     CStateManager& mgr);
 | |
|   zeus::CAABox CalculateCollidersBoundingBox(const std::vector<CCameraCollider>& colliderList,
 | |
|                                              CStateManager& mgr) const;
 | |
|   [[nodiscard]] int CountObscuredColliders(const std::vector<CCameraCollider>& colliderList) const;
 | |
|   void UpdateCollidersDistances(std::vector<CCameraCollider>& colliderList, float xMag, float zMag, float angOffset);
 | |
|   void UpdateUsingColliders(float dt, CStateManager& mgr);
 | |
|   void UpdateUsingSpindleCameras(float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f ClampElevationToWater(zeus::CVector3f& pos, CStateManager& mgr) const;
 | |
|   void UpdateTransitionFromBallCamera(CStateManager& mgr);
 | |
|   void UpdateUsingTransitions(float dt, CStateManager& mgr);
 | |
|   zeus::CTransform UpdateCameraPositions(float dt, const zeus::CTransform& oldXf, const zeus::CTransform& newXf);
 | |
|   static zeus::CVector3f GetFailsafeSplinePoint(const std::vector<zeus::CVector3f>& points, float t);
 | |
|   bool CheckFailsafeFromMorphBallState(CStateManager& mgr) const;
 | |
|   bool SplineIntersectTest(CMaterialList& intersectMat, CStateManager& mgr) const;
 | |
|   void ActivateFailsafe(float dt, CStateManager& mgr);
 | |
|   bool ConstrainElevationAndDistance(float& elevation, float& distance, float dt, CStateManager& mgr);
 | |
|   zeus::CVector3f FindDesiredPosition(float distance, float elevation, const zeus::CVector3f& dir, CStateManager& mgr,
 | |
|                                       bool fullTest);
 | |
|   static bool DetectCollision(const zeus::CVector3f& from, const zeus::CVector3f& to, float radius, float& d,
 | |
|                               CStateManager& mgr);
 | |
|   void TeleportColliders(std::vector<CCameraCollider>& colliderList, const zeus::CVector3f& pos);
 | |
|   static bool CheckTransitionLineOfSight(const zeus::CVector3f& eyePos, const zeus::CVector3f& behindPos,
 | |
|                                          float& eyeToOccDist, float colRadius, CStateManager& mgr);
 | |
| 
 | |
| public:
 | |
|   CBallCamera(TUniqueId uid, TUniqueId watchedId, const zeus::CTransform& xf, float fovy, float znear, float zfar,
 | |
|               float aspect);
 | |
| 
 | |
|   void Accept(IVisitor& visitor) override;
 | |
|   void AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId objId, CStateManager& stateMgr) override;
 | |
|   void ProcessInput(const CFinalInput& input, CStateManager& mgr) override;
 | |
|   void Reset(const zeus::CTransform&, CStateManager& mgr) override;
 | |
|   void Render(CStateManager& mgr) override;
 | |
|   [[nodiscard]] EBallCameraBehaviour GetBehaviour() const { return x188_behaviour; }
 | |
|   [[nodiscard]] EBallCameraState GetState() const { return x400_state; }
 | |
|   void SetState(EBallCameraState state, CStateManager& mgr);
 | |
|   void Think(float dt, CStateManager& mgr) override;
 | |
|   bool TransitionFromMorphBallState(CStateManager& mgr);
 | |
|   [[nodiscard]] TUniqueId GetTooCloseActorId() const { return x3dc_tooCloseActorId; }
 | |
|   [[nodiscard]] float GetTooCloseActorDistance() const { return x3e0_tooCloseActorDist; }
 | |
|   void TeleportCamera(const zeus::CVector3f& pos, CStateManager& mgr);
 | |
|   void TeleportCamera(const zeus::CTransform& xf, CStateManager& mgr);
 | |
|   [[nodiscard]] const zeus::CVector3f& GetLookPos() const { return x1d8_lookPos; }
 | |
|   void ResetToTweaks(CStateManager& mgr);
 | |
|   void UpdateLookAtPosition(float dt, CStateManager& mgr);
 | |
|   zeus::CTransform UpdateLookDirection(const zeus::CVector3f& dir, CStateManager& mgr);
 | |
|   void SetClampVelTimer(float f) { x470_clampVelTimer = f; }
 | |
|   void SetClampVelRange(float f) { x474_clampVelRange = f; }
 | |
|   void ApplyCameraHint(CStateManager& mgr);
 | |
|   void ResetPosition(CStateManager& mgr);
 | |
|   void DoorClosed(TUniqueId doorId);
 | |
|   void DoorClosing(TUniqueId doorId);
 | |
|   [[nodiscard]] const zeus::CVector3f& GetLookPosAhead() const { return x1c0_lookPosAhead; }
 | |
|   [[nodiscard]] const zeus::CVector3f& GetFixedLookPos() const { return x1cc_fixedLookPos; }
 | |
|   const TUniqueId GetCollisionActorId() const { return x46c_collisionActorId; }
 | |
| 
 | |
|   static bool IsBallNearDoor(const zeus::CVector3f& pos, CStateManager& mgr);
 | |
| };
 | |
| 
 | |
| } // namespace metaforce
 |