A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
RailInterpolator.hh
1#pragma once
2
3#include <Common.hh>
4
5#include "game/field/Rail.hh"
6
7#include "game/system/map/MapdataPointInfo.hh"
8
9#include <egg/math/Vector.hh>
10
11namespace Field {
12
14public:
15 enum class Status {
16 InProgress = 0,
17 SegmentEnd = 1,
18 ChangingDirection = 2,
19 };
20
21 RailInterpolator(f32 speed, u32 idx);
22 virtual ~RailInterpolator();
23
24 virtual void init(f32 t, u32 idx) = 0;
25 virtual Status calc() = 0;
26 virtual void setCurrVel(f32 speed) = 0;
27
28 virtual f32 getCurrVel() = 0;
29 virtual void evalCubicBezierOnPath(f32 t, EGG::Vector3f &currDir,
30 EGG::Vector3f &curTangentDir) = 0;
31 virtual void getPathLocation(f32 t, s16 &idx, f32 &len) = 0;
32
34 void setT(f32 t) {
35 m_segmentT = t;
36 }
37
38 [[nodiscard]] const EGG::Vector3f &floorNrm(size_t idx) const;
39 [[nodiscard]] f32 railLength() const;
40
41 [[nodiscard]] const System::MapdataPointInfo::Point &curPoint() const {
42 ASSERT(static_cast<size_t>(m_currPointIdx) < m_points.size());
43 return m_points[m_currPointIdx];
44 }
45
46 [[nodiscard]] const EGG::Vector3f &curPos() const {
47 return m_curPos;
48 }
49
50 [[nodiscard]] const EGG::Vector3f &curTangentDir() const {
51 return m_curTangentDir;
52 }
53
54 [[nodiscard]] bool isMovementDirectionForward() const {
55 return m_movementDirectionForward;
56 }
57
58protected:
59 void updateVel();
60 void calcVelocities();
61 [[nodiscard]] bool shouldChangeDirection() const;
62 void calcDirectionChange();
63 void calcNextIndices();
64
65 s16 m_railIdx;
66 u16 m_pointCount;
67 std::span<System::MapdataPointInfo::Point> m_points;
68 bool m_isOscillating;
69 f32 m_speed;
70 bool m_usePerPointVelocities;
71 EGG::Vector3f m_curPos;
72 EGG::Vector3f m_curTangentDir;
73 f32 m_currVel;
74 f32 m_prevPointVel;
75 f32 m_nextPointVel;
76 f32 m_currSegmentVel;
77 f32 m_segmentT;
78 bool m_movementDirectionForward;
79 s16 m_currPointIdx;
80 s16 m_nextPointIdx;
81 bool m_4a;
82};
83
85public:
86 RailLinearInterpolator(f32 speed, u32 idx);
87 ~RailLinearInterpolator() override;
88
89 void init(f32 t, u32 idx) override;
90 Status calc() override;
91 void setCurrVel(f32 speed) override;
92
94 [[nodiscard]] f32 getCurrVel() override {
95 return m_currVel;
96 }
97
98 void evalCubicBezierOnPath(f32 t, EGG::Vector3f &currDir,
99 EGG::Vector3f &curTangentDir) override;
100 void getPathLocation(f32 t, s16 &idx, f32 &len) override;
101
102private:
103 void calcNextSegment();
104 EGG::Vector3f lerp(f32 t, u32 currIdx, u32 nextIdx) const;
105
106 EGG::Vector3f m_currentDirection;
107 std::span<RailLineTransition> m_transitions;
108};
109
111public:
112 RailSmoothInterpolator(f32 speed, u32 idx);
113 ~RailSmoothInterpolator() override;
114
115 void init(f32 t, u32 idx) override;
116 Status calc() override;
117 void setCurrVel(f32 speed) override;
118
120 [[nodiscard]] f32 getCurrVel() override {
121 return m_velocity;
122 }
123
124 void evalCubicBezierOnPath(f32 t, EGG::Vector3f &currDir,
125 EGG::Vector3f &curTangentDir) override;
126 void getPathLocation(f32 t, s16 &idx, f32 &len) override;
127
128private:
129 void calcCubicBezier(f32 t, u32 currIdx, u32 nextIdx, EGG::Vector3f &pos,
130 EGG::Vector3f &dir) const;
131 [[nodiscard]] EGG::Vector3f calcCubicBezierPos(f32 t, const RailSplineTransition &trans) const;
132 [[nodiscard]] EGG::Vector3f calcCubicBezierTangentDir(f32 t,
133 const RailSplineTransition &trans) const;
134 [[nodiscard]] f32 calcT(f32 t) const;
135 void calcNextSegment();
136
137 std::span<RailSplineTransition> m_transitions;
138 u32 m_estimatorSampleCount;
139 f32 m_estimatorStep;
140 std::span<f32> m_pathPercentages;
141 EGG::Vector3f m_prevPos;
142 f32 m_velocity;
143};
144
145} // namespace Field
This header houses common data types such as our integral types and enums.
Pertains to collision.
A 3D float vector.
Definition Vector.hh:83