A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KartState.cc
1#include "KartState.hh"
2
3#include "game/kart/CollisionGroup.hh"
4#include "game/kart/KartCollide.hh"
5#include "game/kart/KartDynamics.hh"
6#include "game/kart/KartJump.hh"
7#include "game/kart/KartMove.hh"
8
9#include "game/system/RaceManager.hh"
10
11namespace Kart {
12
14 f32 range;
15 s16 frames;
16};
17
20static constexpr std::array<StartBoostEntry, 6> START_BOOST_ENTRIES = {{
21 {0.85f, 0},
22 {0.88f, 10},
23 {0.905f, 20},
24 {0.925f, 30},
25 {0.94f, 45},
26 {0.95f, 70},
27}};
28
30KartState::KartState() {
35
36 m_bAutoDrift = inputs()->driftIsAuto();
37
38 m_airtime = 0;
39 m_cannonPointId = 0;
41}
42
44void KartState::init() {
45 reset();
46}
47
49void KartState::reset() {
54
55 m_airtime = 0;
56 m_top.setZero();
57 m_hwgTimer = 0;
58 m_boostRampType = -1;
59 m_jumpPadVariant = -1;
60 m_halfPipeInvisibilityTimer = 0;
61 m_startBoostCharge = 0.0f;
62 m_stickX = 0.0f;
64 m_trickableTimer = 0;
65}
66
71 const auto *raceMgr = System::RaceManager::Instance();
72 if (raceMgr->isStageReached(System::RaceManager::Stage::Race)) {
73 if (!m_bInAction && !m_bBeforeRespawn && !m_bCannonStart && !m_bInCannon &&
75 const auto &currentState = inputs()->currentState();
76 const auto &lastState = inputs()->lastState();
77 m_stickX = currentState.stick.x;
78 m_stickY = currentState.stick.y;
79
81 if (m_stickX < 0.0f) {
82 m_bStickLeft = true;
83 } else if (m_stickX > 0.0f) {
84 m_bStickRight = true;
85 }
86 }
87
88 if (!m_bBurnout) {
89 m_bAccelerate = currentState.accelerate();
90 m_bAccelerateStart = m_bAccelerate && !lastState.accelerate();
91 m_bBrake = currentState.brake();
92
93 if (!m_bAutoDrift) {
94 m_bDriftInput = currentState.drift();
95 m_bHopStart = m_bDriftInput && !lastState.drift();
96 }
97 }
98 }
99
101 } else {
102 if (!raceMgr->isStageReached(System::RaceManager::Stage::Countdown)) {
103 return;
104 }
105
106 const auto &currentState = inputs()->currentState();
107 m_stickX = currentState.stick.x;
108 m_bChargeStartBoost = currentState.accelerate();
109
111 }
112}
113
118 resetFlags();
119
120 collide()->calcBeforeRespawn();
121
123 collide()->calcBoundingRadius();
124}
125
127void KartState::resetFlags() {
128 m_bAccelerate = false;
129 m_bBrake = false;
130 m_bDriftInput = false;
131 m_bHopStart = false;
132 m_bAccelerateStart = false;
133 m_bGroundStart = false;
134 m_bStickLeft = false;
135 m_bWallCollisionStart = false;
136 m_bAirStart = false;
137 m_bStickRight = false;
138
140
141 m_bJumpPadDisableYsusForce = false;
142
143 m_stickY = 0.0f;
144 m_stickX = 0.0f;
145}
146
155 bool wasTouchingGround = m_bTouchingGround;
156 bool wasWallCollision = m_bWallCollision || m_bWall3Collision;
157
158 m_bWall3Collision = false;
159 m_bWallCollision = false;
161 m_bAnyWheelCollision = false;
162 m_bAllWheelsCollision = false;
163 m_bTouchingGround = false;
164
165 if (m_hwgTimer > 0) {
166 if (--m_hwgTimer == 0) {
167 m_bUNK2 = false;
168 m_bSomethingWallCollision = false;
169 }
170 }
171
172 m_top.setZero();
173 bool softWallCollision = false;
174
175 if (collide()->someSoftWallTimer() > 0) {
176 if (collide()->someNonSoftWallTimer() == 0) {
177 softWallCollision = true;
178 } else {
179 f32 softSusp = collide()->suspBottomHeightSoftWall() /
180 static_cast<f32>(collide()->someSoftWallTimer());
181 f32 nonSusp = collide()->suspBottomHeightNonSoftWall() /
182 static_cast<f32>(collide()->someNonSoftWallTimer());
183
184 if (softSusp - nonSusp >= 40.0f) {
185 m_bSoftWallDrift = false;
186 } else {
187 softWallCollision = true;
188 }
189 }
190 }
191
192 u16 wheelCollisions = 0;
193 u16 softWallCount = 0;
194 EGG::Vector3f wallNrm = EGG::Vector3f::zero;
195 bool trickable = false;
196
197 for (u16 tireIdx = 0; tireIdx < tireCount(); ++tireIdx) {
198 const auto &colData = collisionData(tireIdx);
199 if (hasFloorCollision(tirePhysics(tireIdx))) {
200 m_top += colData.floorNrm;
201 trickable = trickable || colData.bTrickable;
202 ++wheelCollisions;
203 }
204
205 if (softWallCollision && colData.bSoftWall) {
206 ++softWallCount;
207 wallNrm += colData.noBounceWallNrm;
208 }
209 }
210
211 if (wheelCollisions > 0) {
213 if (wheelCollisions == tireCount()) {
215 }
216 }
217
218 CollisionData &colData = collisionData();
219 if (colData.bFloor) {
221 m_top += colData.floorNrm;
222 trickable = trickable || colData.bTrickable;
223
224 if (m_bOverZipper) {
225 halfPipe()->end(true);
226 }
227 }
228
229 bool hitboxGroupSoftWallCollision = false;
230 if (softWallCollision && colData.bSoftWall) {
231 hitboxGroupSoftWallCollision = true;
232 ++softWallCount;
233 wallNrm += colData.wallNrm;
234 }
235
236 bool bVar3 = colData.bInvisibleWallOnly && m_halfPipeInvisibilityTimer > 0;
237 m_halfPipeInvisibilityTimer = std::max(0, m_halfPipeInvisibilityTimer - 1);
238
239 m_wallBonkTimer = std::max(0, m_wallBonkTimer - 1);
240
241 bool hwg = false;
242
243 if ((colData.bWall || colData.bWall3) && !bVar3) {
244 if (colData.bWall) {
245 m_bWallCollision = true;
246 }
247
248 if (colData.bWall3) {
249 m_bWall3Collision = true;
250 }
251
252 if (!wasWallCollision) {
254 }
255
256 m_wallBonkTimer = 2;
257
258 if (m_hwgTimer == 0 && colData.movement.y > 1.0f) {
259 EGG::Vector3f movement = colData.movement;
260 movement.normalise();
261
262 if (movement.dot(EGG::Vector3f::ey) > 0.8f &&
263 colData.wallNrm.dot(EGG::Vector3f::ey) > 0.85f &&
264 (movement.x * colData.wallNrm.x + movement.z * colData.wallNrm.z < 0.0f ||
265 collide()->colPerpendicularity() >= 1.0f)) {
266 colData.wallNrm.y = 0.0f;
267 colData.wallNrm.normalise();
268 wallNrm = colData.wallNrm;
269
270 if (wallNrm.length() < 0.05f) {
271 wallNrm = movement;
272 wallNrm.y = 0.0f;
273 }
274
275 hwg = true;
276 }
277 }
278 }
279
280 if (colData.bInvisibleWall && m_bHalfPipeRamp &&
281 collide()->surfaceFlags().offBit(KartCollide::eSurfaceFlags::StopHalfPipeState)) {
283 }
284
285 if (softWallCount > 0 || hwg) {
286 m_bUNK2 = true;
287 m_softWallSpeed = wallNrm;
288 m_softWallSpeed.normalise();
289 if (softWallCount > 0 && !m_bHop) {
290 m_bSoftWallDrift = true;
291 }
292
293 if (hwg) {
294 m_bHWG = true;
295 }
296
297 if (hitboxGroupSoftWallCollision || hwg || isBike()) {
298 m_bSomethingWallCollision = true;
299 m_hwgTimer = 10;
300
301 if (hwg) {
302 m_hwgTimer *= 2;
303 }
304 }
305 }
306
307 m_bAirtimeOver20 = false;
308 m_trickableTimer = std::max(0, m_trickableTimer - 1);
309
310 if (wheelCollisions < 1 && !colData.bFloor) {
311 if (wasTouchingGround) {
312 m_bAirStart = true;
313 }
314
315 if (++m_airtime > 20) {
316 m_bAirtimeOver20 = true;
317 }
318 } else {
319 m_top.normalise();
320
321 m_bTouchingGround = true;
322 m_bAfterCannon = false;
323
324 if (!m_bInAction) {
325 m_bEndHalfPipe = false;
326 }
327
328 if (m_bOverZipper) {
329 halfPipe()->end(true);
330 }
331
332 if (trickable) {
333 m_trickableTimer = 3;
334 }
335
336 m_bTrickable = m_trickableTimer > 0;
337
338 if (!wasTouchingGround) {
339 m_bGroundStart = true;
340 }
341
342 if (m_bInATrick && jump()->cooldown() == 0) {
343 move()->landTrick();
344 dynamics()->setForceUpright(true);
345 jump()->end();
346 }
347
348 m_airtime = 0;
349 }
350}
351
357 constexpr f32 START_BOOST_DELTA_ONE = 0.02f;
358 constexpr f32 START_BOOST_DELTA_TWO = 0.002f;
359 constexpr f32 START_BOOST_FALLOFF = 0.96f;
360
362 m_startBoostCharge += START_BOOST_DELTA_ONE -
363 (START_BOOST_DELTA_ONE - START_BOOST_DELTA_TWO) * m_startBoostCharge;
364 } else {
365 m_startBoostCharge *= START_BOOST_FALLOFF;
366 }
367
368 m_startBoostCharge = std::max(0.0f, std::min(1.0f, m_startBoostCharge));
369}
370
375 if (System::RaceManager::Instance()->getCountdownTimer() != 0) {
376 return;
377 }
378
379 if (m_bAccelerate) {
380 if (m_startBoostCharge > START_BOOST_ENTRIES.back().range) {
381 m_startBoostIdx = std::numeric_limits<size_t>::max();
382 } else if (m_startBoostCharge > START_BOOST_ENTRIES.front().range) {
383 // Ranges are exclusive on the lower bound and inclusive on the upper bound
384 for (size_t i = 1; i < START_BOOST_ENTRIES.size(); ++i) {
385 if (m_startBoostCharge > START_BOOST_ENTRIES[i - 1].range &&
386 m_startBoostCharge <= START_BOOST_ENTRIES[i].range) {
387 m_startBoostIdx = i;
388 break;
389 }
390 }
391 }
392 }
393
394 if (m_startBoostIdx <= 0) {
395 return;
396 }
397
399 m_bChargeStartBoost = false;
400}
401
406 if (m_startBoostIdx == std::numeric_limits<size_t>::max()) {
407 move()->burnout().start();
408 } else {
409 move()->applyStartBoost(START_BOOST_ENTRIES[idx].frames);
410 }
411}
412
416 m_bHalfPipeRamp = false;
417 m_bRejectRoad = false;
418}
419
422 m_bAccelerate = false;
423 m_bBrake = false;
424 m_bDriftInput = false;
425 m_bDriftManual = false;
426 m_bBeforeRespawn = false;
427 m_bWall3Collision = false;
428 m_bWallCollision = false;
429 m_bHopStart = false;
430 m_bAccelerateStart = false;
431 m_bGroundStart = false;
433 m_bAnyWheelCollision = false;
434 m_bAllWheelsCollision = false;
435 m_bStickLeft = false;
436 m_bWallCollisionStart = false;
437 m_bAirtimeOver20 = false;
438 m_bStickyRoad = false;
439 m_bTouchingGround = false;
440 m_bHop = false;
441 m_bBoost = false;
442 m_bAirStart = false;
443 m_bStickRight = false;
444 m_bMushroomBoost = false;
445 m_bDriftAuto = false;
446 m_bSlipdriftCharge = false;
447 m_bWheelie = false;
448 m_bJumpPad = false;
449 m_bRampBoost = false;
450}
451
454 m_bInAction = false;
455 m_bTriggerRespawn = false;
456 m_bCannonStart = false;
457 m_bInCannon = false;
458 m_bTrickStart = false;
459 m_bInATrick = false;
461 m_bHalfPipeRamp = false;
462 m_bOverZipper = false;
465 m_bZipperBoost = false;
466 m_bZipperStick = false;
467 m_bZipperTrick = false;
468 m_bRespawnKillY = false;
469 m_bBurnout = false;
470 m_bTrickRot = false;
471 m_bChargingSsmt = false;
472 m_bRejectRoad = false;
473 m_bRejectRoadTrigger = false;
474 m_bTrickable = false;
475}
476
479 m_bWheelieRot = false;
480 m_bSkipWheelCalc = false;
481 m_bNoSparkInvisibleWall = false;
482 m_bInRespawn = false;
483 m_bAfterRespawn = false;
484 m_bJumpPadDisableYsusForce = false;
485}
486
489 m_bUNK2 = false;
490 m_bSomethingWallCollision = false;
491 m_bSoftWallDrift = false;
492 m_bHWG = false;
493 m_bAfterCannon = false;
494 m_bChargeStartBoost = false;
495 m_bEndHalfPipe = false;
496}
497
498} // namespace Kart
bool m_bZipperBoost
Set when boosting after landing from a zipper.
Definition KartState.hh:618
bool m_bRespawnKillY
Set while respawning to cap external velocity at 0.
Definition KartState.hh:622
bool m_bBoost
Set while in a boost.
Definition KartState.hh:594
bool m_bChargeStartBoost
Like m_bAccelerate but during countdown.
Definition KartState.hh:650
bool m_bHop
Set while we are in a drift hop. Clears when we land.
Definition KartState.hh:593
bool m_bBeforeRespawn
Set on respawn collision, cleared on position snap.
Definition KartState.hh:579
void clearBitfield1()
Helper function to clear all bit flags at 0x8-0xB in KartState.
Definition KartState.cc:453
size_t m_startBoostIdx
Used to map m_startBoostCharge to a start boost duration.
Definition KartState.hh:671
void clearBitfield0()
Helper function to clear all bit flags at 0x4-0x7 in KartState.
Definition KartState.cc:421
bool m_bHopStart
Set if m_bDriftInput was toggled on this frame.
Definition KartState.hh:582
bool m_bHWG
Set when "Horizontal Wall Glitch" is active.
Definition KartState.hh:648
bool m_bBurnout
Set during a burnout on race start.
Definition KartState.hh:623
bool m_bDriftAuto
Currently in a drift w/ automatic.
Definition KartState.hh:598
bool m_bStickRight
Set on right stick input. Mutually exclusive to m_bStickLeft.
Definition KartState.hh:596
void clearBitfield2()
Helper function to clear all bit flags at 0xC-0xF in KartState.
Definition KartState.cc:478
void calcStartBoost()
STAGE 1 - Each frame, calculates the start boost charge.
Definition KartState.cc:356
bool m_bRejectRoad
Collision which causes a change in the player's pos and rot.
Definition KartState.hh:626
bool m_bWallCollision
Set if we are colliding with a wall.
Definition KartState.hh:581
bool m_bZipperTrick
Set while tricking mid-air from a zipper.
Definition KartState.hh:620
s16 m_wallBonkTimer
2f counter that stunts your speed after hitting a wall.
Definition KartState.hh:672
void calcCollisions()
Each frame, checks for collision and saves relevant bit flags.
Definition KartState.cc:154
bool m_bAllWheelsCollision
Set when all wheels are touching floor collision.
Definition KartState.hh:587
bool m_bDriftInput
A "fake" button, normally set if you meet the speed requirement to hop.
Definition KartState.hh:577
void clearBitfield3()
Helper function to clear all bit flags at 0x10-0x13 in KartState.
Definition KartState.cc:488
void calcHandleStartBoost()
On countdown end, calculates and applies our start boost charge.
Definition KartState.cc:374
f32 m_stickY
One of 15 discrete stick values from [-1.0, 1.0].
Definition KartState.hh:669
bool m_bChargingSsmt
Tracks whether we are charging a stand-still mini-turbo.
Definition KartState.hh:625
bool m_bAccelerateStart
Set if m_bAccelerate was toggled on this frame.
Definition KartState.hh:583
bool m_bDriftManual
Currently in a drift w/ manual.
Definition KartState.hh:578
bool m_bAutoDrift
True if auto transmission, false if manual.
Definition KartState.hh:657
void handleStartBoost(size_t idx)
Applies the relevant start boost duration.
Definition KartState.cc:405
void calc()
Every frame, resets the input state and saves collision-related bit flags.
Definition KartState.cc:117
bool m_bZipperStick
Set while mid-air and still influenced by the zipper.
Definition KartState.hh:619
bool m_bMushroomBoost
Set while we are in a mushroom boost.
Definition KartState.hh:597
bool m_bDisableBackwardsAccel
Enforces a 20f delay when reversing after charging SSMT.
Definition KartState.hh:621
bool m_bHalfPipeRamp
Set while colliding with zipper KCL.
Definition KartState.hh:615
f32 m_stickX
One of 15 discrete stick values from [-1.0, 1.0].
Definition KartState.hh:668
bool m_bVehicleBodyFloorCollision
Set if the vehicle body is colliding with the floor.
Definition KartState.hh:585
bool m_bBoostOffroadInvincibility
Set if we should ignore offroad slowdown this frame.
Definition KartState.hh:614
bool m_bAnyWheelCollision
Set when any wheel is touching floor collision.
Definition KartState.hh:586
bool m_bStickyRoad
Like the rBC stairs.
Definition KartState.hh:591
bool m_bWall3Collision
Set when colliding with wall KCL COL_TYPE_WALL_2.
Definition KartState.hh:580
f32 m_startBoostCharge
0-1 representation of start boost charge. Burnout if >0.95f.
Definition KartState.hh:670
bool m_bWheelie
Set while we are in a wheelie (even during the countdown).
Definition KartState.hh:600
bool m_bZipperInvisibleWall
Set when colliding with invisible wall above a zipper.
Definition KartState.hh:617
bool m_bTouchingGround
Set when any part of the vehicle is colliding with floor KCL.
Definition KartState.hh:592
bool m_bRejectRoadTrigger
e.g. DK Summit ending, and Maple Treeway side walls.
Definition KartState.hh:627
bool m_bStickLeft
Set on left stick input. Mutually exclusive to m_bStickRight.
Definition KartState.hh:588
void calcInput()
Each frame, read input and save related bit flags. Also handles start boosts.
Definition KartState.cc:70
bool m_bWallCollisionStart
Set if we have just started colliding with a wall.
Definition KartState.hh:589
bool m_bOverZipper
Set while mid-air from a zipper.
Definition KartState.hh:616
bool m_bAccelerate
Accel button is pressed.
Definition KartState.hh:571
void resetEjection()
Resets certain bitfields pertaining to ejections (reject road, half pipe zippers, etc....
Definition KartState.cc:415
bool m_bGroundStart
Set first frame landing from airtime.
Definition KartState.hh:584
bool m_bAirtimeOver20
Set after 20 frames of airtime, resets on landing.
Definition KartState.hh:590
Pertains to kart-related functionality.
A 3D float vector.
Definition Vector.hh:83
f32 normalise()
Normalizes the vector and returns the original length.
Definition Vector.cc:44
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
Definition Vector.hh:182
f32 length() const
The square root of the vector's dot product.
Definition Vector.hh:187
Information about the current collision and its properties.
bool bFloor
Set if colliding with KCL which satisfies KCL_TYPE_FLOOR.
bool bWall3
Set if colliding with COL_TYPE_WALL_2.
bool bWall
Set if colliding with KCL which satisfies KCL_TYPE_WALL.