A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KartJump.cc
1#include "KartJump.hh"
2
3#include "game/kart/KartCollide.hh"
4#include "game/kart/KartMove.hh"
5#include "game/kart/KartParam.hh"
6#include "game/kart/KartPhysics.hh"
7#include "game/kart/KartState.hh"
8
9#include "game/system/KPadController.hh"
10
11#include <egg/math/Math.hh>
12
13namespace Kart {
14
16KartJump::KartJump(KartMove *move) : m_move(move) {
17 m_cooldown = 0;
18
19 // The base game doesn't initialize this explicitly, since EGG::Heaps are memset to 0.
20 m_nextAllowTimer = 0;
21}
22
24KartJump::~KartJump() = default;
25
27void KartJump::calcRot() {
28 m_angleDelta *= m_angleDeltaFactor;
29 m_angleDelta = std::max(m_angleDelta, m_properties.angleDeltaMin);
30 m_angleDeltaFactor -= m_angleDeltaFactorDec;
31 m_angleDeltaFactor = std::max(m_angleDeltaFactor, m_properties.angleDeltaFactorMin);
32 m_angle += m_angleDelta;
33 m_angle = std::min(m_angle, m_finalAngle);
34
35 switch (m_type) {
36 case TrickType::KartFlipTrickZ:
37 m_rot.setRPY(EGG::Vector3f(0.0f, 0.0f, -(m_angle * DEG2RAD) * m_rotSign));
38 break;
39 case TrickType::FlipTrickYLeft:
40 case TrickType::FlipTrickYRight:
41 m_rot.setRPY(EGG::Vector3f(0.0f, m_angle * DEG2RAD * m_rotSign, 0.0f));
42 break;
43 default:
44 break;
45 }
46
47 physics()->composeStuntRot(m_rot);
48}
49
51void KartJump::setupProperties() {
52 static constexpr std::array<TrickProperties, 3> TRICK_PROPERTIES = {{
53 {11.0f, 1.5f, 0.9f, 0.0018f},
54 {14.0f, 1.5f, 0.9f, 0.0006f},
55 {7.5f, 2.5f, 0.93f, 0.05f},
56 }};
57
58 static constexpr std::array<f32, 3> FINAL_ANGLES = {{
59 360.0f,
60 720.0f,
61 180.0f,
62 }};
63
64 if (m_variant == SurfaceVariant::SingleFlipTrick) {
65 m_properties = TRICK_PROPERTIES[0];
66 m_finalAngle = FINAL_ANGLES[0];
67 } else if (m_variant == SurfaceVariant::StuntTrick) {
68 m_properties = TRICK_PROPERTIES[1];
69 m_finalAngle = FINAL_ANGLES[1];
70 } else if (m_type == TrickType::BikeSideStuntTrick) {
71 m_properties = TRICK_PROPERTIES[2];
72 m_finalAngle = FINAL_ANGLES[2];
73 }
74
75 m_angleDelta = m_properties.initialAngleDiff;
76 m_angleDeltaFactorDec = m_properties.angleDiffMulDec;
77 m_angle = 0.0f;
78 m_angleDeltaFactor = 1.0f;
79}
80
82void KartJump::reset() {
83 m_cooldown = 0;
84}
85
87void KartJump::tryStart(const EGG::Vector3f &left) {
88 if (!state()->isTrickStart()) {
89 return;
90 }
91
92 if (m_move->speedRatioCapped() > 0.5f) {
93 s32 boostRampType = state()->boostRampType();
94
95 if (boostRampType == 0) {
96 m_variant = SurfaceVariant::StuntTrick;
97 } else if (boostRampType == 1) {
98 m_variant = SurfaceVariant::SingleFlipTrick;
99 } else {
100 m_variant = SurfaceVariant::DoubleFlipTrick;
101 }
102
103 start(left);
104 }
105
106 state()->setTrickStart(false);
107}
108
110void KartJump::calc() {
111 m_cooldown = std::max(0, m_cooldown - 1);
112
113 if (state()->isTrickRot()) {
114 calcRot();
115 }
116
117 calcInput();
118}
119
120bool KartJump::someFlagCheck() {
121 return state()->isInAction() || state()->isTrickStart() || state()->isInATrick() ||
122 state()->isOverZipper();
123}
124
126void KartJump::calcInput() {
127 constexpr s16 TRICK_ALLOW_TIMER = 14;
128
129 System::Trick trick = inputs()->currentState().trick;
130
131 if (!someFlagCheck() && trick != System::Trick::None) {
132 m_nextTrick = trick;
133 m_nextAllowTimer = TRICK_ALLOW_TIMER;
134 }
135
136 u32 airtime = state()->airtime();
137 if (airtime == 0 || m_nextAllowTimer < 1 || airtime > 10 ||
138 (!state()->isTrickable() && state()->boostRampType() < 0) || someFlagCheck()) {
139 m_nextAllowTimer = std::max(0, m_nextAllowTimer - 1);
140 } else {
141 if (airtime > 2) {
142 state()->setTrickStart(true);
143 }
144 if (state()->isRampBoost()) {
145 m_boostRampEnabled = true;
146 }
147 }
148 if (state()->isTouchingGround() &&
149 collide()->surfaceFlags().offBit(KartCollide::eSurfaceFlags::BoostRamp)) {
150 m_boostRampEnabled = false;
151 }
152}
153
155void KartJump::end() {
156 if (state()->isTrickRot()) {
157 physics()->composeDecayingStuntRot(m_rot);
158 }
159
160 state()->setInATrick(false);
161 state()->setTrickRot(false);
162 m_boostRampEnabled = false;
163}
164
166void KartJump::setAngle(const EGG::Vector3f &left) {
167 static constexpr std::array<std::array<AngleProperties, 3>, 3> ANGLE_PROPERTIES = {{
168 {{
169 {40.0f, 15.0f},
170 {45.0f, 20.0f},
171 {45.0f, 20.0f},
172 }},
173 {{
174 {36.0f, 13.0f},
175 {42.0f, 18.0f},
176 {42.0f, 18.0f},
177 }},
178 {{
179 {32.0f, 11.0f},
180 {39.0f, 16.0f},
181 {39.0f, 16.0f},
182 }},
183 }};
184
185 f32 vel1YDot = m_move->vel1Dir().dot(EGG::Vector3f::ey);
186 EGG::Vector3f vel1YCross = m_move->vel1Dir().cross(EGG::Vector3f::ey);
187 f32 vel1YCrossMag = vel1YCross.length();
188 f32 pitch = EGG::Mathf::abs(EGG::Mathf::atan2(vel1YCrossMag, vel1YDot));
189 f32 angle = 90.0f - (pitch * RAD2DEG);
190 u32 weightClass = static_cast<u32>(param()->stats().weightClass);
191 f32 targetAngle = ANGLE_PROPERTIES[weightClass][static_cast<u32>(m_variant)].targetAngle;
192
193 if (state()->isJumpPad() || angle > targetAngle) {
194 return;
195 }
196
197 f32 rotAngle = ANGLE_PROPERTIES[weightClass][static_cast<u32>(m_variant)].rotAngle;
198
199 if (angle + rotAngle > targetAngle) {
200 rotAngle = targetAngle - angle;
201 }
202
203 EGG::Matrix34f nextDir;
204 nextDir.setAxisRotation(-rotAngle * DEG2RAD, left);
205 m_move->setDir(nextDir.ps_multVector(m_move->dir()));
206 m_move->setVel1Dir(m_move->dir());
207}
208
210void KartJump::start(const EGG::Vector3f &left) {
211 init();
212 setAngle(left);
213 state()->setInATrick(true);
214 m_cooldown = 5;
215}
216
218void KartJump::init() {
219 if (m_variant == SurfaceVariant::DoubleFlipTrick) {
220 m_type = TrickType::StuntTrickBasic;
221 return;
222 }
223
224 if (m_nextTrick < System::Trick::Left) {
225 m_type = TrickType::KartFlipTrickZ;
226 m_rotSign = (m_nextTrick == System::Trick::Up) ? -1.0f : 1.0f;
227 } else {
228 m_type = static_cast<TrickType>(m_nextTrick);
229 m_rotSign = (m_type == TrickType::FlipTrickYRight) ? -1.0f : 1.0f;
230 }
231
232 setupProperties();
233 state()->setTrickRot(true);
234}
235
236KartJumpBike::KartJumpBike(KartMove *move) : KartJump(move) {}
237
239KartJumpBike::~KartJumpBike() = default;
240
242void KartJumpBike::calcRot() {
244 constexpr f32 PI_OVER_9 = 0.34906584f;
246 constexpr f32 PI_OVER_3 = 1.0471976f;
247
248 m_angleDelta *= m_angleDeltaFactor;
249 m_angleDelta = std::max(m_angleDelta, m_properties.angleDeltaMin);
250 m_angleDeltaFactor -= m_angleDeltaFactorDec;
251 m_angleDeltaFactor = std::max(m_angleDeltaFactor, m_properties.angleDeltaFactorMin);
252 m_angle += m_angleDelta;
253 m_angle = std::min(m_angle, m_finalAngle);
254
255 switch (m_type) {
256 case TrickType::BikeFlipTrickNose:
257 case TrickType::BikeFlipTrickTail: {
258 EGG::Vector3f angles = EGG::Vector3f(-(m_angle * DEG2RAD) * m_rotSign, 0.0f, 0.0f);
259 m_rot.setRPY(angles);
260 } break;
261 case TrickType::FlipTrickYLeft:
262 case TrickType::FlipTrickYRight: {
263 EGG::Vector3f angles = EGG::Vector3f(0.0f, m_angle * DEG2RAD * m_rotSign, 0.0f);
264 m_rot.setRPY(angles);
265 } break;
266 case TrickType::BikeSideStuntTrick: {
267 f32 sin = EGG::Mathf::SinFIdx(m_angle * DEG2FIDX);
268 EGG::Vector3f angles = EGG::Vector3f(sin * -PI_OVER_9, (sin * m_rotSign) * -PI_OVER_3,
269 (sin * m_rotSign) * PI_OVER_9);
270 m_rot.setRPY(angles);
271 } break;
272 default:
273 break;
274 }
275
276 physics()->composeStuntRot(m_rot);
277}
278
280void KartJumpBike::start(const EGG::Vector3f &left) {
281 KartJump::start(left);
282
283 KartMoveBike *moveBike = static_cast<KartMoveBike *>(m_move);
284 moveBike->cancelWheelie();
285}
286
288void KartJumpBike::init() {
289 constexpr f32 DOUBLE_FLIP_TRICK_FINAL_ANGLE = 180.0f;
290
291 if (m_variant == SurfaceVariant::DoubleFlipTrick) {
292 if (m_nextTrick < System::Trick::Left) {
293 return;
294 }
295
296 m_type = TrickType::BikeSideStuntTrick;
297 m_rotSign = (m_nextTrick == System::Trick::Right) ? -1.0f : 1.0f;
298 setupProperties();
299 m_finalAngle = DOUBLE_FLIP_TRICK_FINAL_ANGLE;
300 } else {
301 m_type = static_cast<TrickType>(m_nextTrick);
302 m_rotSign =
303 (m_type == TrickType::FlipTrickYRight || m_type == TrickType::BikeFlipTrickTail) ?
304 -1.0f :
305 1.0f;
306 setupProperties();
307 }
308
309 state()->setTrickRot(true);
310}
311
312} // namespace Kart
A 3 x 4 matrix.
Definition Matrix.hh:8
void setAxisRotation(f32 angle, const Vector3f &axis)
Rotates the matrix about an axis.
Definition Matrix.cc:175
Vector3f ps_multVector(const Vector3f &vec) const
Paired-singles impl. of multVector.
Definition Matrix.cc:232
Responsible for reacting to player inputs and moving the bike.
Definition KartMove.hh:420
virtual void cancelWheelie()
Clears the wheelie bit flag and resets the rotation decrement.
Definition KartMove.cc:2168
Pertains to kart-related functionality.
A 3D float vector.
Definition Vector.hh:83
f32 length() const
The square root of the vector's dot product.
Definition Vector.hh:187