A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KartMove.cc
1#include "KartMove.hh"
2
3#include "game/kart/KartCollide.hh"
4#include "game/kart/KartDynamics.hh"
5#include "game/kart/KartJump.hh"
6#include "game/kart/KartParam.hh"
7#include "game/kart/KartPhysics.hh"
8#include "game/kart/KartSub.hh"
9#include "game/kart/KartSuspension.hh"
10
11#include "game/field/CollisionDirector.hh"
13
14#include "game/item/ItemDirector.hh"
15#include "game/item/KartItem.hh"
16
17#include "game/system/CourseMap.hh"
18#include "game/system/RaceManager.hh"
19#include "game/system/map/MapdataCannonPoint.hh"
20#include "game/system/map/MapdataJugemPoint.hh"
21
22#include <egg/math/Math.hh>
23#include <egg/math/Quat.hh>
24
25namespace Kart {
26
28 f32 speed;
29 f32 height;
30 f32 decelFactor;
31 f32 endDecel;
32};
33
34static constexpr std::array<CannonParameter, 3> CANNON_PARAMETERS = {{
35 {500.0f, 0.0f, 6000.0f, -1.0f},
36 {500.0f, 5000.0f, 6000.0f, -1.0f},
37 {120.0f, 2000.0f, 1000.0f, 45.0f},
38}};
39
41KartMove::KartMove() : m_smoothedUp(EGG::Vector3f::ey), m_scale(1.0f, 1.0f, 1.0f) {
42 m_totalScale = 1.0f;
43 m_hitboxScale = 1.0f;
44 m_padType.makeAllZero();
45 m_flags.makeAllZero();
46 m_jump = nullptr;
47}
48
50KartMove::~KartMove() {
51 delete m_jump;
52 delete m_halfPipe;
53}
54
56void KartMove::createSubsystems() {
57 m_jump = new KartJump(this);
58 m_halfPipe = new KartHalfPipe();
59}
60
65 m_realTurn = 0.0f;
66 m_rawTurn = 0.0f;
67
68 if (state()->isInAction() || state()->isCannonStart() || state()->isInCannon() ||
69 state()->isOverZipper()) {
70 return;
71 }
72
73 if (state()->isBeforeRespawn()) {
74 return;
75 }
76
77 if (!state()->isHop() || m_hopStickX == 0) {
78 m_rawTurn = -state()->stickX();
79 if (state()->isAirtimeOver20()) {
80 m_rawTurn *= 0.01f;
81 }
82 } else {
83 m_rawTurn = static_cast<f32>(m_hopStickX);
84 }
85
86 f32 reactivity;
87 if (state()->isDrifting()) {
88 reactivity = param()->stats().driftReactivity;
89 } else {
90 reactivity = param()->stats().handlingReactivity;
91 }
92
93 m_weightedTurn = m_rawTurn * reactivity + m_weightedTurn * (1.0f - reactivity);
94 m_weightedTurn = std::max(-1.0f, std::min(1.0f, m_weightedTurn));
95
97
98 if (!state()->isDrifting()) {
99 return;
100 }
101
102 m_realTurn = (m_weightedTurn + static_cast<f32>(m_hopStickX)) * 0.5f;
103 m_realTurn = m_realTurn * 0.8f + 0.2f * static_cast<f32>(m_hopStickX);
104 m_realTurn = std::max(-1.0f, std::min(1.0f, m_realTurn));
105}
106
108void KartMove::setTurnParams() {
109 static constexpr std::array<DriftingParameters, 3> DRIFTING_PARAMS_ARRAY = {{
110 {10.0f, 0.5f, 0.5f, 1.0f},
111 {10.0f, 0.5f, 0.5f, 0.2f},
112 {10.0f, 0.22f, 0.5f, 0.2f},
113 }};
114
115 init(false, false);
116 m_dir = bodyFront();
117 m_lastDir = m_dir;
118 m_vel1Dir = m_dir;
119 m_landingDir = m_dir;
120 m_outsideDriftLastDir = m_dir;
121 m_driftingParams = &DRIFTING_PARAMS_ARRAY[static_cast<u32>(param()->stats().driftType)];
122}
123
125void KartMove::init(bool b1, bool b2) {
126 m_lastSpeed = 0.0f;
127 m_baseSpeed = param()->stats().speed;
128 m_softSpeedLimit = param()->stats().speed;
129 m_speed = 0.0f;
130 setKartSpeedLimit();
131 m_acceleration = 0.0f;
133 m_up = EGG::Vector3f::ey;
134 m_smoothedUp = EGG::Vector3f::ey;
135 m_vel1Dir = EGG::Vector3f::ez;
136 m_lastDir = EGG::Vector3f::ez;
137 m_dir = EGG::Vector3f::ez;
138 m_landingDir = EGG::Vector3f::ez;
139 m_dirDiff = EGG::Vector3f::zero;
140 m_hasLandingDir = false;
141 m_outsideDriftAngle = 0.0f;
142 m_landingAngle = 0.0f;
143 m_outsideDriftLastDir = EGG::Vector3f::ez;
144 m_speedRatio = 0.0f;
145 m_speedRatioCapped = 0.0f;
146 m_kclSpeedFactor = 1.0f;
147 m_kclRotFactor = 1.0f;
149 m_kclWheelRotFactor = 1.0f;
150
151 if (!b2) {
153 }
154
155 m_hopStickX = 0;
156 m_hopFrame = 0;
157 m_hopUp = EGG::Vector3f::ey;
158 m_hopDir = EGG::Vector3f::ez;
159 m_divingRot = 0.0f;
160 m_standStillBoostRot = 0.0f;
161 m_driftState = DriftState::NotDrifting;
162 m_smtCharge = 0;
163 m_mtCharge = 0;
164 m_outsideDriftBonus = 0.0f;
165 m_boost.reset();
166 m_zipperBoostTimer = 0;
167 m_zipperBoostMax = 0;
168 m_reject.reset();
170 m_ssmtCharge = 0;
173 m_nonZipperAirtime = 0;
174 m_realTurn = 0.0f;
175 m_weightedTurn = 0.0f;
176
177 if (!b1) {
178 m_scale.set(1.0f);
179 m_totalScale = 1.0f;
180 m_hitboxScale = 1.0f;
182 }
183
184 m_jumpPadMinSpeed = 0.0f;
185 m_jumpPadMaxSpeed = 0.0f;
186 m_jumpPadProperties = nullptr;
187 m_rampBoost = 0;
188 m_autoDriftAngle = 0.0f;
189 m_autoDriftStartFrameCounter = 0;
190
191 m_cannonEntryOfsLength = 0.0f;
192 m_cannonEntryPos.setZero();
193 m_cannonEntryOfs.setZero();
194 m_cannonOrthog.setZero();
195 m_cannonProgress.setZero();
196
197 m_hopVelY = 0.0f;
198 m_hopPosY = 0.0f;
199 m_hopGravity = 0.0f;
200 m_timeInRespawn = 0;
203 m_respawnTimer = 0;
204 m_drivingDirection = DrivingDirection::Forwards;
205 m_padType.makeAllZero();
206 m_flags.makeAllZero();
207 m_jump->reset();
208 m_halfPipe->reset();
209 m_rawTurn = 0.0f;
210}
211
213void KartMove::clear() {
214 clearBoost();
215 clearJumpPad();
216 clearRampBoost();
217 clearZipperBoost();
218 clearSsmt();
219 clearOffroadInvincibility();
220 m_halfPipe->end(false);
221 m_jump->end();
222 clearRejectRoad();
223}
224
228 EGG::Quatf quaternion;
229 quaternion.setRPY(angles * DEG2RAD);
230 EGG::Vector3f newPos = position;
232 Field::KCLTypeMask kcl_flags = KCL_NONE;
233
234 bool bColliding = Field::CollisionDirector::Instance()->checkSphereFullPush(100.0f, newPos,
235 EGG::Vector3f::inf, KCL_ANY, &info, &kcl_flags, 0);
236
237 if (bColliding && (kcl_flags & KCL_TYPE_FLOOR)) {
238 newPos = newPos + info.tangentOff + (info.floorNrm * -100.0f);
239 newPos += info.floorNrm * bsp().initialYPos;
240 }
241
242 setPos(newPos);
243 setRot(quaternion);
244
245 sub()->initPhysicsValues();
246
247 physics()->setPos(pos());
248 physics()->setVelocity(dynamics()->velocity());
249
250 m_landingDir = bodyFront();
251 m_dir = bodyFront();
252 m_up = bodyUp();
253 dynamics()->setTop(m_up);
254
255 for (u16 tireIdx = 0; tireIdx < suspCount(); ++tireIdx) {
256 suspension(tireIdx)->setInitialState();
257 }
258}
259
266 if (state()->isInRespawn()) {
267 calcInRespawn();
268 return;
269 }
270
271 dynamics()->resetInternalVelocity();
272 m_burnout.calc();
274 m_halfPipe->calc();
275 calcTop();
276 tryEndJumpPad();
277 calcRespawnBoost();
279 m_jump->calc();
281 calcDirs();
282 calcStickyRoad();
283 calcOffroad();
284 calcTurn();
285
286 if (!state()->isAutoDrift()) {
288 }
289
290 calcWheelie();
291 calcSsmt();
292 calcBoost();
294 calcZipperBoost();
295
296 if (state()->isInCannon()) {
297 calcCannon();
298 }
299
303 calcRotation();
304}
305
307void KartMove::calcRespawnStart() {
308 constexpr float RESPAWN_HEIGHT = 700.0f;
309
310 const auto *jugemPoint = System::RaceManager::Instance()->jugemPoint();
311 const EGG::Vector3f &jugemPos = jugemPoint->pos();
312 const EGG::Vector3f &jugemRot = jugemPoint->rot();
313
314 EGG::Vector3f respawnPos = jugemPos;
315 respawnPos.y += RESPAWN_HEIGHT;
316 EGG::Vector3f respawnRot = EGG::Vector3f(0.0f, jugemRot.y, 0.0f);
317
318 setInitialPhysicsValues(respawnPos, respawnRot);
319
320 Item::ItemDirector::Instance()->kartItem(0).clear();
321
322 state()->setTriggerRespawn(false);
323 state()->setInRespawn(true);
324}
325
327void KartMove::calcInRespawn() {
328 constexpr f32 LAKITU_VELOCITY = 1.5f;
329 constexpr u16 RESPAWN_DURATION = 110;
330
331 if (!state()->isInRespawn()) {
332 return;
333 }
334
335 EGG::Vector3f newPos = pos();
336 newPos.y -= LAKITU_VELOCITY;
337 dynamics()->setPos(newPos);
338 dynamics()->setNoGravity(true);
339
340 if (++m_timeInRespawn > RESPAWN_DURATION) {
341 state()->setInRespawn(false);
342 state()->setAfterRespawn(true);
343 state()->setRespawnKillY(true);
344 m_timeInRespawn = 0;
345 m_flags.setBit(eFlags::Respawned);
346 dynamics()->setNoGravity(false);
347 }
348}
349
351void KartMove::calcRespawnBoost() {
352 constexpr s16 RESPAWN_BOOST_DURATION = 30;
353 constexpr s16 RESPAWN_BOOST_INPUT_LENIENCY = 4;
354
355 if (state()->isAfterRespawn()) {
356 if (state()->isTouchingGround()) {
357 if (m_respawnPreLandTimer > 0) {
358 if (!state()->isBeforeRespawn() && !state()->isInAction()) {
359 activateBoost(KartBoost::Type::AllMt, RESPAWN_BOOST_DURATION);
360 m_respawnTimer = RESPAWN_BOOST_DURATION;
361 }
362 } else {
363 m_respawnPostLandTimer = RESPAWN_BOOST_INPUT_LENIENCY;
364 }
365
366 state()->setAfterRespawn(false);
368 }
369
371
372 if (m_flags.onBit(eFlags::Respawned) && state()->isAccelerateStart()) {
373 m_respawnPreLandTimer = RESPAWN_BOOST_INPUT_LENIENCY;
375 }
376 } else {
377 if (m_respawnPostLandTimer > 0) {
378 if (state()->isAccelerateStart()) {
379 if (!state()->isBeforeRespawn() && !state()->isInAction()) {
380 activateBoost(KartBoost::Type::AllMt, RESPAWN_BOOST_DURATION);
381 m_respawnTimer = RESPAWN_BOOST_DURATION;
382 }
383
385 }
386
388 } else {
389 state()->setRespawnKillY(false);
390 }
391 }
392
393 m_respawnTimer = std::max(0, m_respawnTimer - 1);
394}
395
397void KartMove::calcTop() {
398 f32 stabilizationFactor = 0.1f;
399 m_hasLandingDir = false;
400 EGG::Vector3f inputTop = state()->top();
401
402 if (state()->isGroundStart() && m_nonZipperAirtime >= 3) {
403 m_smoothedUp = inputTop;
404 m_up = inputTop;
405 m_landingDir = m_dir.perpInPlane(m_smoothedUp, true);
406 m_dirDiff = m_landingDir.proj(m_landingDir);
407 m_hasLandingDir = true;
408 } else {
409 if (state()->isHop() && m_hopPosY > 0.0f) {
410 stabilizationFactor = m_driftingParams->stabilizationFactor;
411 } else if (state()->isTouchingGround()) {
412 if ((m_flags.onBit(eFlags::TrickableSurface) || state()->trickableTimer() > 0) &&
413 inputTop.dot(m_dir) > 0.0f && m_speed > 50.0f &&
414 collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::NotTrickable)) {
415 inputTop = m_up;
416 } else {
417 m_up = inputTop;
418 }
419
420 f32 scalar = 0.8f;
421
422 if (state()->isHalfPipeRamp() ||
423 (!state()->isBoost() && !state()->isRampBoost() && !state()->isWheelie() &&
424 !state()->isOverZipper() &&
425 (!state()->isZipperBoost() || m_zipperBoostTimer > 15))) {
426 f32 topDotZ = 0.8f - 6.0f * (EGG::Mathf::abs(inputTop.dot(componentZAxis())));
427 scalar = std::min(0.8f, std::max(0.3f, topDotZ));
428 }
429
430 m_smoothedUp += (inputTop - m_smoothedUp) * scalar;
432
433 f32 bodyDotFront = bodyFront().dot(m_smoothedUp);
434
435 if (bodyDotFront < -0.1f) {
436 stabilizationFactor += std::min(0.2f, EGG::Mathf::abs(bodyDotFront) * 0.5f);
437 }
438
439 if (collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::BoostRamp)) {
440 stabilizationFactor = 0.4f;
441 }
442 } else {
444 }
445 }
446
447 dynamics()->setStabilizationFactor(stabilizationFactor);
448
449 m_nonZipperAirtime = state()->isOverZipper() ? 0 : state()->airtime();
450 m_flags.changeBit(collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::Trickable),
452}
453
457 if (state()->isOverZipper() || !state()->isAirtimeOver20()) {
458 return;
459 }
460
461 if (m_smoothedUp.y <= 0.99f) {
462 m_smoothedUp += (EGG::Vector3f::ey - m_smoothedUp) * 0.03f;
464 } else {
465 m_smoothedUp = EGG::Vector3f::ey;
466 }
467
468 if (m_up.y <= 0.99f) {
469 m_up += (EGG::Vector3f::ey - m_up) * 0.03f;
470 m_up.normalise();
471 } else {
472 m_up = EGG::Vector3f::ey;
473 }
474}
475
480 const auto *raceMgr = System::RaceManager::Instance();
481 if (!raceMgr->isStageReached(System::RaceManager::Stage::Race)) {
482 return;
483 }
484
485 if (m_padType.onBit(ePadType::BoostPanel)) {
486 tryStartBoostPanel();
487 }
488
489 if (m_padType.onBit(ePadType::BoostRamp)) {
491 }
492
493 if (m_padType.onBit(ePadType::JumpPad)) {
495 }
496
497 m_padType.makeAllZero();
498}
499
501void KartMove::calcDirs() {
502 EGG::Vector3f right = dynamics()->mainRot().rotateVector(EGG::Vector3f::ex);
503 EGG::Vector3f local_88 = right.cross(m_smoothedUp);
504 local_88.normalise();
505 m_flags.setBit(eFlags::LaunchBoost);
506
507 if (!state()->isInATrick() && !state()->isOverZipper() &&
508 (((state()->isTouchingGround() || !state()->isRampBoost() ||
509 !m_jump->isBoostRampEnabled()) &&
510 !state()->isJumpPad() && state()->airtime() <= 5) ||
511 state()->isNoSparkInvisibleWall())) {
512 if (state()->isHop()) {
513 local_88 = m_hopDir;
514 }
515
516 EGG::Matrix34f mat;
517 mat.setAxisRotation(DEG2RAD * (m_autoDriftAngle + m_outsideDriftAngle + m_landingAngle),
519 EGG::Vector3f local_b8 = mat.multVector(local_88);
520 local_b8 = local_b8.perpInPlane(m_smoothedUp, true);
521
522 EGG::Vector3f dirDiff = local_b8 - m_dir;
523
524 if (dirDiff.squaredLength() <= std::numeric_limits<f32>::epsilon()) {
525 m_dir = local_b8;
526 m_dirDiff.setZero();
527 } else {
528 EGG::Vector3f origDirCross = m_dir.cross(local_b8);
529 m_dirDiff += m_kclRotFactor * dirDiff;
530 m_dir += m_dirDiff;
531 m_dir.normalise();
532 m_dirDiff *= 0.1f;
533 EGG::Vector3f newDirCross = m_dir.cross(local_b8);
534
535 if (origDirCross.dot(newDirCross) < 0.0f) {
536 m_dir = local_b8;
537 m_dirDiff.setZero();
538 }
539 }
540
541 m_vel1Dir = m_dir.perpInPlane(m_smoothedUp, true);
542 m_flags.resetBit(eFlags::LaunchBoost);
543 } else {
544 m_vel1Dir = m_dir;
545 }
546
547 if (!state()->isOverZipper()) {
548 m_jump->tryStart(m_smoothedUp.cross(m_dir));
549 }
550
551 if (m_hasLandingDir) {
552 f32 dot = m_dir.dot(m_landingDir);
553 EGG::Vector3f cross = m_dir.cross(m_landingDir);
554 f32 crossDot = cross.length();
555 f32 angle = EGG::Mathf::atan2(crossDot, dot);
556 angle = EGG::Mathf::abs(angle);
557
558 f32 fVar4 = 1.0f;
559 if (cross.dot(m_smoothedUp) < 0.0f) {
560 fVar4 = -1.0f;
561 }
562
563 m_landingAngle += (angle * RAD2DEG) * fVar4;
564 }
565
566 if (m_landingAngle <= 0.0f) {
567 if (m_landingAngle < 0.0f) {
568 m_landingAngle = std::min(0.0f, m_landingAngle + 2.0f);
569 }
570 } else {
571 m_landingAngle = std::max(0.0f, m_landingAngle - 2.0f);
572 }
573}
574
576void KartMove::calcStickyRoad() {
577 constexpr f32 STICKY_RADIUS = 200.0f;
578 constexpr Field::KCLTypeMask STICKY_MASK =
580
581 if (state()->isOverZipper()) {
582 state()->setStickyRoad(false);
583 return;
584 }
585
586 if ((!state()->isStickyRoad() &&
587 collide()->surfaceFlags().offBit(KartCollide::eSurfaceFlags::Trickable)) ||
588 EGG::Mathf::abs(m_speed) <= 20.0f) {
589 return;
590 }
591
592 EGG::Vector3f pos = dynamics()->pos();
593 EGG::Vector3f vel = m_speed * m_vel1Dir;
594 EGG::Vector3f down = -STICKY_RADIUS * componentYAxis();
595 Field::CollisionInfo colInfo;
596 colInfo.bbox.setZero();
597 Field::KCLTypeMask kcl_flags = KCL_NONE;
598 bool stickyRoad = false;
599
600 for (size_t i = 0; i < 3; ++i) {
601 EGG::Vector3f newPos = pos + vel;
602 if (Field::CollisionDirector::Instance()->checkSphereFull(STICKY_RADIUS, newPos,
603 EGG::Vector3f::inf, STICKY_MASK, &colInfo, &kcl_flags, 0)) {
604 m_vel1Dir = m_vel1Dir.perpInPlane(colInfo.floorNrm, true);
605 stickyRoad = true;
606
607 break;
608 }
609 vel *= 0.5f;
610 pos += -STICKY_RADIUS * componentYAxis();
611 }
612
613 if (!stickyRoad) {
614 state()->setStickyRoad(false);
615 }
616}
617
622 if (state()->isBoostOffroadInvincibility()) {
623 m_kclSpeedFactor = 1.0f;
624 m_kclRotFactor = param()->stats().kclRot[0];
625 } else {
626 bool anyWheel = state()->isAnyWheelCollision();
627 if (anyWheel) {
631 }
632
633 if (state()->isVehicleBodyFloorCollision()) {
634 const CollisionData &colData = collisionData();
635 if (anyWheel) {
636 if (colData.speedFactor < m_kclWheelSpeedFactor) {
637 m_kclSpeedFactor = colData.speedFactor;
638 }
639 m_kclRotFactor = (m_kclWheelRotFactor + colData.rotFactor) /
640 static_cast<f32>(m_floorCollisionCount + 1);
641 } else {
642 m_kclSpeedFactor = colData.speedFactor;
643 m_kclRotFactor = colData.rotFactor;
644 }
645 }
646 }
647}
648
650void KartMove::calcBoost() {
651 if (m_boost.calc()) {
652 state()->setAccelerate(true);
653 } else {
654 state()->setBoost(false);
655 }
656
657 calcRampBoost();
658}
659
661void KartMove::calcRampBoost() {
662 if (!state()->isRampBoost()) {
663 return;
664 }
665
666 state()->setAccelerate(true);
667 if (--m_rampBoost < 1) {
668 m_rampBoost = 0;
669 state()->setRampBoost(false);
670 }
671}
672
677 if (!state()->isDisableBackwardsAccel()) {
678 return;
679 }
680
681 if (--m_ssmtDisableAccelTimer < 0 ||
682 (m_flags.offBit(eFlags::SsmtLeeway) && !state()->isBrake())) {
683 state()->setDisableBackwardsAccel(false);
685 }
686}
687
692 constexpr s16 MAX_SSMT_CHARGE = 75;
693 constexpr s16 SSMT_BOOST_FRAMES = 30;
694 constexpr s16 LEEWAY_FRAMES = 1;
695 constexpr s16 DISABLE_ACCEL_FRAMES = 20;
696
698
699 if (state()->isChargingSsmt()) {
700 if (++m_ssmtCharge > MAX_SSMT_CHARGE) {
701 m_ssmtCharge = MAX_SSMT_CHARGE;
704 }
705
706 return;
707 }
708
709 m_ssmtCharge = 0;
710
711 if (m_flags.offBit(eFlags::SsmtCharged)) {
712 return;
713 }
714
715 if (m_flags.onBit(eFlags::SsmtLeeway)) {
716 if (--m_ssmtLeewayTimer < 0) {
719 m_ssmtDisableAccelTimer = DISABLE_ACCEL_FRAMES;
720 state()->setDisableBackwardsAccel(true);
721 } else {
722 if (!state()->isAccelerate() && !state()->isBrake()) {
723 activateBoost(KartBoost::Type::AllMt, SSMT_BOOST_FRAMES);
726 }
727 }
728 } else {
729 if (state()->isAccelerate() && !state()->isBrake()) {
730 activateBoost(KartBoost::Type::AllMt, SSMT_BOOST_FRAMES);
733 } else {
734 m_ssmtLeewayTimer = LEEWAY_FRAMES;
735 m_flags.setBit(eFlags::SsmtLeeway);
736 state()->setDisableBackwardsAccel(true);
737 m_ssmtDisableAccelTimer = LEEWAY_FRAMES;
738 }
739 }
740}
741
747 if (!state()->isTouchingGround() && !state()->isHop() && !state()->isDriftManual()) {
748 if (state()->isStickLeft() || state()->isStickRight()) {
749 if (!state()->isDriftInput()) {
750 state()->setSlipdriftCharge(false);
751 } else if (!state()->isSlipdriftCharge()) {
752 if (m_hopStickX == 0) {
753 if (state()->isStickRight()) {
754 m_hopStickX = -1;
755 } else if (state()->isStickLeft()) {
756 m_hopStickX = 1;
757 }
758 state()->setSlipdriftCharge(true);
759 onHop();
760 }
761 }
762 }
763 }
764
765 if (state()->isHop()) {
766 if (m_hopStickX == 0) {
767 if (state()->isStickRight()) {
768 m_hopStickX = -1;
769 } else if (state()->isStickLeft()) {
770 m_hopStickX = 1;
771 }
772 }
773 if (m_hopFrame < 3) {
774 ++m_hopFrame;
775 }
776 } else if (state()->isSlipdriftCharge()) {
777 m_hopFrame = 0;
778 }
779
780 return state()->isHop() || state()->isSlipdriftCharge();
781}
782
787 m_hopStickX = 0;
788 m_hopFrame = 0;
789 state()->setHop(false);
790 state()->setDriftManual(false);
791 m_driftState = DriftState::NotDrifting;
792 m_smtCharge = 0;
793 m_mtCharge = 0;
794}
795
800 m_outsideDriftAngle = 0.0f;
801 m_hopStickX = 0;
802 m_hopFrame = 0;
803 m_driftState = DriftState::NotDrifting;
804 m_smtCharge = 0;
805 m_mtCharge = 0;
806 m_outsideDriftBonus = 0.0f;
807 state()->setHop(false);
808 state()->setSlipdriftCharge(false);
809 state()->setDriftManual(false);
810 m_autoDriftStartFrameCounter = 0;
811}
812
814void KartMove::clearJumpPad() {
815 m_jumpPadMinSpeed = 0.0f;
816 state()->setJumpPad(false);
817}
818
820void KartMove::clearRampBoost() {
821 m_rampBoost = 0;
822 state()->setRampBoost(false);
823}
824
826void KartMove::clearZipperBoost() {
827 m_zipperBoostTimer = 0;
828 state()->setZipperBoost(false);
829}
830
832void KartMove::clearBoost() {
833 m_boost.resetActive();
834 state()->setBoost(false);
835}
836
838void KartMove::clearSsmt() {
839 m_ssmtCharge = 0;
843}
844
846void KartMove::clearOffroadInvincibility() {
848 state()->setBoostOffroadInvincibility(false);
849}
850
851void KartMove::clearRejectRoad() {
852 state()->setRejectRoadTrigger(false);
853 state()->setNoSparkInvisibleWall(false);
854}
855
860 constexpr s16 AUTO_DRIFT_START_DELAY = 12;
861
862 if (!state()->isAutoDrift()) {
863 return;
864 }
865
866 if (canStartDrift() && !state()->isOverZipper() && !state()->isRejectRoadTrigger() &&
867 !state()->isWheelie() && EGG::Mathf::abs(state()->stickX()) > 0.85f) {
868 m_autoDriftStartFrameCounter =
869 std::min<s16>(AUTO_DRIFT_START_DELAY, m_autoDriftStartFrameCounter + 1);
870 } else {
871 m_autoDriftStartFrameCounter = 0;
872 }
873
874 if (m_autoDriftStartFrameCounter >= AUTO_DRIFT_START_DELAY) {
875 state()->setDriftAuto(true);
876
877 if (state()->isTouchingGround()) {
878 if (state()->stickX() < 0.0f) {
879 m_hopStickX = 1;
880 m_autoDriftAngle -= 30.0f * param()->stats().driftAutomaticTightness;
881
882 } else {
883 m_hopStickX = -1;
884 m_autoDriftAngle += 30.0f * param()->stats().driftAutomaticTightness;
885 }
886 }
887
888 f32 halfTarget = 0.5f * param()->stats().driftOutsideTargetAngle;
889 m_autoDriftAngle = std::min(halfTarget, std::max(-halfTarget, m_autoDriftAngle));
890 } else {
891 state()->setDriftAuto(false);
892 m_hopStickX = 0;
893
894 if (m_autoDriftAngle > 0.0f) {
895 m_autoDriftAngle =
896 std::max(0.0f, m_autoDriftAngle - param()->stats().driftOutsideDecrement);
897 } else {
898 m_autoDriftAngle =
899 std::min(0.0f, m_autoDriftAngle + param()->stats().driftOutsideDecrement);
900 }
901 }
902
903 EGG::Quatf angleAxis;
904 angleAxis.setAxisRotation(-m_autoDriftAngle * DEG2RAD, m_up);
905 physics()->composeExtraRot(angleAxis);
906}
907
912 bool isHopping = calcPreDrift();
913
914 if (!state()->isOverZipper()) {
915 const EGG::Vector3f rotZ = dynamics()->mainRot().rotateVector(EGG::Vector3f::ez);
916
917 if (!state()->isTouchingGround() &&
918 param()->stats().driftType != KartParam::Stats::DriftType::Inside_Drift_Bike &&
919 (state()->isDriftManual() || state()->isSlipdriftCharge()) &&
920 m_flags.onBit(eFlags::LaunchBoost)) {
921 const EGG::Vector3f up = dynamics()->mainRot().rotateVector(EGG::Vector3f::ey);
923
924 if (driftRej.normalise() != 0.0f) {
925 f32 rejCrossDirMag = driftRej.cross(rotZ).length();
926 f32 angle = EGG::Mathf::atan2(rejCrossDirMag, driftRej.dot(rotZ));
927 f32 sign = 1.0f;
928 if ((rotZ.z * (rotZ.x - driftRej.x)) - (rotZ.x * (rotZ.z - driftRej.z)) > 0.0f) {
929 sign = -1.0f;
930 }
931
932 m_outsideDriftAngle += angle * RAD2DEG * sign;
933 }
934 }
935
937 }
938
939 // TODO: Is this backwards/inverted?
940 if (((!state()->isHop() || m_hopFrame < 3) && !state()->isSlipdriftCharge()) ||
941 (state()->isInAction() || !state()->isTouchingGround())) {
942 if (canHop()) {
943 hop();
944 isHopping = true;
945 }
946 } else {
948 isHopping = false;
949 }
950
952
953 if (!state()->isDriftManual()) {
954 if (!isHopping && state()->isTouchingGround()) {
956
957 if (!action()->flags().onBit(KartAction::eFlags::Rotating) || m_speed <= 20.0f) {
958 f32 driftAngleDecr = param()->stats().driftOutsideDecrement;
959 if (m_outsideDriftAngle > 0.0f) {
960 m_outsideDriftAngle = std::max(0.0f, m_outsideDriftAngle - driftAngleDecr);
961 } else if (m_outsideDriftAngle < 0.0f) {
962 m_outsideDriftAngle = std::min(0.0f, m_outsideDriftAngle + driftAngleDecr);
963 }
964 }
965 }
966 } else {
967 if (!state()->isOverZipper() &&
968 (!state()->isDriftInput() || !state()->isAccelerate() || state()->isInAction() ||
969 state()->isRejectRoadTrigger() || state()->isWall3Collision() ||
970 state()->isWallCollision() || !canStartDrift())) {
971 if (canStartDrift()) {
972 releaseMt();
973 }
974
976 m_flags.setBit(eFlags::DriftReset);
977 } else {
979 }
980 }
981}
982
987 constexpr f32 OUTSIDE_DRIFT_BONUS = 0.5f;
988
989 const auto &stats = param()->stats();
990
991 if (stats.driftType != KartParam::Stats::DriftType::Inside_Drift_Bike) {
992 f32 driftAngle = 0.0f;
993
994 if (state()->isHop()) {
995 const EGG::Vector3f rotZ = dynamics()->mainRot().rotateVector(EGG::Vector3f::ez);
996 EGG::Vector3f rotRej = rotZ.rej(m_hopUp);
997
998 if (rotRej.normalise() != 0.0f) {
999 const EGG::Vector3f hopCrossRot = m_hopDir.cross(rotRej);
1000 driftAngle =
1001 EGG::Mathf::atan2(hopCrossRot.length(), m_hopDir.dot(rotRej)) * RAD2DEG;
1002 }
1003 }
1004
1005 m_outsideDriftAngle += driftAngle * static_cast<f32>(-m_hopStickX);
1006 m_outsideDriftAngle = std::max(-60.0f, std::min(60.0f, m_outsideDriftAngle));
1007 }
1008
1009 state()->setHop(false);
1010 state()->setSlipdriftCharge(false);
1011
1012 if (!state()->isDriftInput()) {
1013 return;
1014 }
1015
1016 if (getAppliedHopStickX() == 0) {
1017 return;
1018 }
1019
1020 state()->setDriftManual(true);
1021 state()->setHop(false);
1022 m_driftState = DriftState::ChargingMt;
1023 m_outsideDriftBonus = OUTSIDE_DRIFT_BONUS * (m_speedRatioCapped * stats.driftManualTightness);
1024}
1025
1030 constexpr f32 SMT_LENGTH_FACTOR = 3.0f;
1031
1032 if (m_driftState < DriftState::ChargedMt || state()->isBrake()) {
1033 m_driftState = DriftState::NotDrifting;
1034 return;
1035 }
1036
1037 u16 mtLength = param()->stats().miniTurbo;
1038
1039 if (m_driftState == DriftState::ChargedSmt) {
1040 mtLength *= SMT_LENGTH_FACTOR;
1041 }
1042
1043 if (!state()->isBeforeRespawn() && !state()->isInAction()) {
1044 activateBoost(KartBoost::Type::AllMt, mtLength);
1045 }
1046
1047 m_driftState = DriftState::NotDrifting;
1048}
1049
1054 if (state()->airtime() > 5) {
1055 return;
1056 }
1057
1058 if (param()->stats().driftType != KartParam::Stats::DriftType::Inside_Drift_Bike) {
1059 if (m_hopStickX == -1) {
1060 f32 angle = m_outsideDriftAngle;
1061 f32 targetAngle = param()->stats().driftOutsideTargetAngle;
1062 if (angle > targetAngle) {
1063 m_outsideDriftAngle = std::max(m_outsideDriftAngle - 2.0f, targetAngle);
1064 } else if (angle < targetAngle) {
1065 m_outsideDriftAngle += 150.0f * param()->stats().driftManualTightness;
1066 m_outsideDriftAngle = std::min(m_outsideDriftAngle, targetAngle);
1067 }
1068 } else if (m_hopStickX == 1) {
1069 f32 angle = m_outsideDriftAngle;
1070 f32 targetAngle = -param()->stats().driftOutsideTargetAngle;
1071 if (targetAngle > angle) {
1072 m_outsideDriftAngle = std::min(m_outsideDriftAngle + 2.0f, targetAngle);
1073 } else if (targetAngle < angle) {
1074 m_outsideDriftAngle -= 150.0f * param()->stats().driftManualTightness;
1075 m_outsideDriftAngle = std::max(m_outsideDriftAngle, targetAngle);
1076 }
1077 }
1078 }
1079
1080 calcMtCharge();
1081}
1082
1087 f32 turn;
1088 bool drifting = state()->isDrifting();
1089 bool autoDrift = state()->isAutoDrift();
1090 const auto &stats = param()->stats();
1091
1092 if (drifting) {
1093 turn = autoDrift ? stats.driftAutomaticTightness : stats.driftManualTightness;
1094 } else {
1095 turn = autoDrift ? stats.handlingAutomaticTightness : stats.handlingManualTightness;
1096 }
1097
1098 if (drifting && stats.driftType != KartParam::Stats::DriftType::Inside_Drift_Bike) {
1099 m_outsideDriftBonus *= 0.99f;
1100 turn += m_outsideDriftBonus;
1101 }
1102
1103 bool forwards = true;
1104 if (state()->isBrake() && m_speed <= 0.0f) {
1105 forwards = false;
1106 }
1107
1108 turn *= m_realTurn;
1109 if (state()->isChargingSsmt()) {
1110 turn = m_realTurn * 0.04f;
1111 } else {
1112 if (state()->isHop() && m_hopPosY > 0.0f) {
1113 turn *= 1.4f;
1114 }
1115
1116 if (!drifting) {
1117 bool noTurn = false;
1118 if (!state()->isWallCollision() && !state()->isWall3Collision() &&
1119 EGG::Mathf::abs(m_speed) < 1.0f) {
1120 if (!(state()->isHop() && m_hopPosY > 0.0f)) {
1121 turn = 0.0f;
1122 noTurn = true;
1123 }
1124 }
1125 if (forwards && !noTurn) {
1126 if (m_speed >= 20.0f) {
1127 turn *= 0.5f;
1128 if (m_speed < 70.0f) {
1129 turn += (1.0f - (m_speed - 20.0f) / 50.0f) * turn;
1130 }
1131 } else {
1132 turn = (turn * 0.4f) + (m_speed / 20.0f) * (turn * 0.6f);
1133 }
1134 }
1135 }
1136
1137 if (!forwards) {
1138 turn = -turn;
1139 }
1140
1141 if (state()->isZipperBoost() && !state()->isDriftManual()) {
1142 turn *= 2.0f;
1143 }
1144
1145 f32 stickX = EGG::Mathf::abs(state()->stickX());
1146 if (autoDrift && stickX > 0.3f) {
1147 f32 stickScalar = (stickX - 0.3f) / 0.7f;
1148 stickX = drifting ? 0.2f : 0.5f;
1149 turn += stickScalar * (turn * stickX * m_speedRatioCapped);
1150 }
1151 }
1152
1153 if (!state()->isInAction() && !state()->isZipperTrick()) {
1154 if (!state()->isTouchingGround()) {
1155 if (state()->isRampBoost() && m_jump->isBoostRampEnabled()) {
1156 turn = 0.0f;
1157 } else {
1158 u32 airtime = state()->airtime();
1159 if (airtime >= 70) {
1160 turn = 0.0f;
1161 } else if (airtime >= 30) {
1162 turn = std::max(0.0f, turn * (1.0f - (airtime - 30) * 0.025f));
1163 }
1164 }
1165 }
1166
1167 const EGG::Vector3f forward = dynamics()->mainRot().rotateVector(EGG::Vector3f::ez);
1168 f32 angle = EGG::Mathf::atan2(forward.cross(m_dir).length(), forward.dot(m_dir));
1169 angle = EGG::Mathf::abs(angle) * RAD2DEG;
1170
1171 if (angle > 60.0f) {
1172 turn *= std::max(0.0f, 1.0f - (angle - 60.0f) / 40.0f);
1173 }
1174 }
1175
1176 calcVehicleRotation(turn);
1177}
1178
1183 const auto *raceMgr = System::RaceManager::Instance();
1184 if (raceMgr->isStageReached(System::RaceManager::Stage::Race)) {
1185 f32 speedFix = dynamics()->speedFix();
1186 if (state()->isInAction() ||
1187 ((state()->isWallCollisionStart() || state()->wallBonkTimer() == 0 ||
1188 EGG::Mathf::abs(speedFix) >= 3.0f) &&
1189 !state()->isDriftManual())) {
1190 m_speed += speedFix;
1191 }
1192 }
1193
1194 if (m_speed < -20.0f) {
1195 m_speed += 0.5f;
1196 }
1197
1198 m_acceleration = 0.0f;
1199 m_speedDragMultiplier = 1.0f;
1200
1201 if (state()->isInAction()) {
1202 action()->calcVehicleSpeed();
1203 return;
1204 }
1205
1206 if ((state()->isSomethingWallCollision() && state()->isTouchingGround() &&
1207 !state()->isAnyWheelCollision()) ||
1208 !state()->isTouchingGround() || state()->isChargingSsmt()) {
1209 if (state()->isRampBoost() && state()->airtime() < 4) {
1210 m_acceleration = 7.0f;
1211 } else {
1212 if (state()->isJumpPad() && !state()->isAccelerate()) {
1213 m_speedDragMultiplier = 0.99f;
1214 } else {
1215 if (state()->isOverZipper()) {
1216 m_speedDragMultiplier = 0.999f;
1217 } else {
1218 if (state()->airtime() > 5) {
1219 m_speedDragMultiplier = 0.999f;
1220 }
1221 }
1222 }
1224 }
1225
1226 } else if (state()->isBoost()) {
1227 m_acceleration = m_boost.acceleration();
1228 } else {
1229 if (!state()->isJumpPad() && !state()->isRampBoost()) {
1230 if (state()->isAccelerate()) {
1231 m_acceleration = state()->isHalfPipeRamp() ? 5.0f : calcVehicleAcceleration();
1232 } else {
1233 if (!state()->isBrake() || state()->isDisableBackwardsAccel() ||
1234 state()->isSomethingWallCollision()) {
1235 m_speed *= m_speed > 0.0f ? 0.98f : 0.95f;
1236 } else if (m_drivingDirection == DrivingDirection::Braking) {
1237 m_acceleration = -1.5f;
1239 if (++m_backwardsAllowCounter > 15) {
1240 m_drivingDirection = DrivingDirection::Backwards;
1241 }
1242 } else if (m_drivingDirection == DrivingDirection::Backwards) {
1243 m_acceleration = -2.0f;
1244 }
1245 }
1246
1247 if (!state()->isBoost() && !state()->isDriftManual() && !state()->isAutoDrift()) {
1248 const auto &stats = param()->stats();
1249
1250 f32 x = 1.0f - EGG::Mathf::abs(m_weightedTurn) * m_speedRatioCapped;
1251 m_speed *= stats.turningSpeed + (1.0f - stats.turningSpeed) * x;
1252 }
1253 } else {
1254 m_acceleration = 7.0f;
1255 }
1256 }
1257}
1258
1262 f32 vel = 0.0f;
1263 f32 initialVel = 1.0f - m_smoothedUp.y;
1264 if (EGG::Mathf::abs(m_speed) < 30.0f && m_smoothedUp.y > 0.0f && initialVel > 0.0f) {
1265 initialVel = std::min(initialVel * 2.0f, 2.0f);
1266 vel += initialVel;
1267 vel *= std::min(0.5f, std::max(-0.5f, -bodyFront().y));
1268 }
1269 m_speed += vel;
1270}
1271
1276 f32 ratio = m_speed / m_softSpeedLimit;
1277 if (ratio < 0.0f) {
1278 return 1.0f;
1279 }
1280
1281 std::span<const f32> as;
1282 std::span<const f32> ts;
1283 if (state()->isDrifting()) {
1284 as = param()->stats().accelerationDriftA;
1285 ts = param()->stats().accelerationDriftT;
1286 } else {
1287 as = param()->stats().accelerationStandardA;
1288 ts = param()->stats().accelerationStandardT;
1289 }
1290
1291 size_t i = 0;
1292 f32 acceleration = 0.0f;
1293 f32 t_curr = 0.0f;
1294 for (; i < ts.size(); ++i) {
1295 if (ratio < ts[i]) {
1296 acceleration = as[i] + ((as[i + 1] - as[i]) / (ts[i] - t_curr)) * (ratio - t_curr);
1297 break;
1298 }
1299
1300 t_curr = ts[i];
1301 }
1302
1303 return i < ts.size() ? acceleration : as.back();
1304}
1305
1310 constexpr f32 ROTATION_SCALAR_NORMAL = 0.5f;
1311 constexpr f32 ROTATION_SCALAR_MIDAIR = 0.2f;
1312 constexpr f32 ROTATION_SCALAR_BOOST_RAMP = 4.0f;
1313 constexpr f32 OOB_SLOWDOWN_RATE = 0.95f;
1314 constexpr f32 TERMINAL_VELOCITY = 90.0f;
1315
1317
1318 dynamics()->setKillExtVelY(state()->isRespawnKillY());
1319
1320 if (state()->isBurnout()) {
1321 m_speed = 0.0f;
1322 } else {
1323 if (m_acceleration < 0.0f) {
1324 if (m_speed < -20.0f) {
1325 m_acceleration = 0.0f;
1326 } else {
1327 if (m_speed + m_acceleration <= -20.0f) {
1328 m_acceleration = -20.0f - m_speed;
1329 }
1330 }
1331 }
1332
1334 }
1335
1336 if (state()->isBeforeRespawn()) {
1337 m_speed *= OOB_SLOWDOWN_RATE;
1338 } else {
1339 if (state()->isChargingSsmt()) {
1340 m_speed *= 0.8f;
1341 } else {
1342 if (m_drivingDirection == DrivingDirection::Braking && m_speed < 0.0f) {
1343 m_speed = 0.0f;
1346 }
1347 }
1348 }
1349
1350 f32 dVar17 = state()->isJumpPad() ? m_jumpPadMaxSpeed : m_baseSpeed;
1351 dVar17 *= (m_boost.multiplier() + getWheelieSoftSpeedLimitBonus()) * m_kclSpeedFactor;
1352 dVar17 = std::max(dVar17, m_boost.speedLimit() * m_kclSpeedFactor);
1353
1354 if (state()->isRampBoost()) {
1355 dVar17 = std::max(dVar17, 100.0f);
1356 }
1357
1358 m_lastDir = (m_speed > 0.0f) ? 1.0f * m_dir : -1.0f * m_dir;
1359
1360 f32 local_c8 = 1.0f;
1361 dVar17 *= calcWallCollisionSpeedFactor(local_c8);
1362
1363 if (!state()->isWallCollision() && !state()->isWall3Collision()) {
1364 m_softSpeedLimit = std::max(m_softSpeedLimit - 3.0f, dVar17);
1365 } else {
1366 m_softSpeedLimit = dVar17;
1367 }
1368
1370
1371 m_speed = std::min(m_softSpeedLimit, std::max(-m_softSpeedLimit, m_speed));
1372
1373 if (state()->isJumpPad()) {
1374 m_speed = std::max(m_speed, m_jumpPadMinSpeed);
1375 }
1376
1377 calcWallCollisionStart(local_c8);
1378
1379 m_speedRatio = EGG::Mathf::abs(m_speed / m_baseSpeed);
1380 m_speedRatioCapped = std::min(1.0f, m_speedRatio);
1381
1382 EGG::Vector3f crossVec = m_smoothedUp.cross(m_dir);
1383 if (m_speed < 0.0f) {
1384 crossVec = -crossVec;
1385 }
1386
1387 f32 rotationScalar = ROTATION_SCALAR_NORMAL;
1388 if (collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::BoostRamp)) {
1389 rotationScalar = ROTATION_SCALAR_BOOST_RAMP;
1390 } else if (!state()->isTouchingGround()) {
1391 rotationScalar = ROTATION_SCALAR_MIDAIR;
1392 }
1393
1394 EGG::Matrix34f local_90;
1395 local_90.setAxisRotation(rotationScalar * DEG2RAD, crossVec);
1396 m_vel1Dir = local_90.multVector33(m_vel1Dir);
1397
1398 const auto *raceMgr = System::RaceManager::Instance();
1399 if (!state()->isInAction() && !state()->isDisableBackwardsAccel() &&
1400 state()->isTouchingGround() && !state()->isAccelerate() &&
1401 raceMgr->isStageReached(System::RaceManager::Stage::Race)) {
1403 }
1404
1406 EGG::Vector3f nextSpeed = m_speed * m_vel1Dir;
1407
1408 f32 maxSpeedY = state()->isOverZipper() ? KartHalfPipe::TerminalVelocity() : TERMINAL_VELOCITY;
1409 nextSpeed.y = std::min(nextSpeed.y, maxSpeedY);
1410
1411 dynamics()->setIntVel(dynamics()->intVel() + nextSpeed);
1412
1413 if (state()->isTouchingGround() && !state()->isDriftManual() && !state()->isHop()) {
1414 if (state()->isBrake()) {
1415 if (m_drivingDirection == DrivingDirection::Forwards) {
1416 m_drivingDirection = m_processedSpeed > 5.0f ? DrivingDirection::Braking :
1417 DrivingDirection::Backwards;
1418 }
1419 } else {
1420 if (m_processedSpeed >= 0.0f) {
1421 m_drivingDirection = DrivingDirection::Forwards;
1422 }
1423 }
1424 } else {
1425 m_drivingDirection = DrivingDirection::Forwards;
1426 }
1427}
1428
1433 if (!state()->isWallCollision() && !state()->isWall3Collision()) {
1434 return 1.0f;
1435 }
1436
1437 onWallCollision();
1438
1439 if (state()->isZipperInvisibleWall() || state()->isOverZipper()) {
1440 return 1.0f;
1441 }
1442
1443 EGG::Vector3f wallNrm = collisionData().wallNrm;
1444 if (wallNrm.y > 0.0f) {
1445 wallNrm.y = 0.0f;
1446 wallNrm.normalise();
1447 }
1448
1449 f32 dot = m_lastDir.dot(wallNrm);
1450
1451 if (dot < 0.0f) {
1452 f1 = std::max(0.0f, dot + 1.0f);
1453
1454 return std::min(1.0f, f1 * (state()->isWallCollision() ? 0.4f : 0.7f));
1455 }
1456
1457 return 1.0f;
1458}
1459
1465
1466 if (!state()->isWallCollisionStart()) {
1467 return;
1468 }
1469
1470 m_outsideDriftAngle = 0.0f;
1471 if (!state()->isInAction()) {
1472 m_dir = bodyFront();
1473 m_vel1Dir = m_dir;
1474 m_landingDir = m_dir;
1475 }
1476
1477 if (!state()->isOverZipper() && param_2 < 0.9f) {
1478 f32 speedDiff = m_lastSpeed - m_speed;
1479
1480 if (speedDiff > 30.0f) {
1481 m_flags.setBit(eFlags::WallBounce);
1482 const CollisionData &colData = collisionData();
1483 EGG::Vector3f newPos = colData.relPos + pos();
1484 f32 dot = -bodyUp().dot(colData.relPos) * 0.5f;
1485 EGG::Vector3f scaledUp = dot * bodyUp();
1486 newPos -= scaledUp;
1487
1488 speedDiff = std::min(60.0f, speedDiff);
1489 EGG::Vector3f scaledWallNrm = speedDiff * colData.wallNrm;
1490
1491 auto [proj, rej] = scaledWallNrm.projAndRej(m_vel1Dir);
1492 proj *= 0.3f;
1493 rej *= 0.9f;
1494
1495 if (state()->isBoost()) {
1496 proj = EGG::Vector3f::zero;
1497 rej = EGG::Vector3f::zero;
1498 }
1499
1500 if (bodyFront().dot(colData.wallNrm) > 0.0f) {
1501 proj = EGG::Vector3f::zero;
1502 }
1503 rej *= 0.9f;
1504
1505 EGG::Vector3f projRejSum = proj + rej;
1506 f32 bumpDeviation = 0.0f;
1507 if (m_flags.offBit(eFlags::DriftReset) && state()->isTouchingGround()) {
1508 bumpDeviation = param()->stats().bumpDeviationLevel;
1509 }
1510
1511 dynamics()->applyWrenchScaled(newPos, projRejSum, bumpDeviation);
1512 }
1513 }
1514}
1515
1520 f32 next = 0.0f;
1521 f32 scalar = 1.0f;
1522
1523 if (state()->isTouchingGround()) {
1524 if (System::RaceManager::Instance()->stage() == System::RaceManager::Stage::Countdown) {
1525 next = 0.015f * -state()->startBoostCharge();
1526 } else if (!state()->isChargingSsmt()) {
1527 if (!state()->isJumpPad() && !state()->isRampBoost() && !state()->isSoftWallDrift()) {
1528 f32 speedDiff = m_lastSpeed - m_speed;
1529 scalar = std::min(3.0f, std::max(speedDiff, -3.0f));
1530
1531 if (state()->isMushroomBoost()) {
1532 next = (scalar * 0.15f) * 0.25f;
1533 if (state()->isWheelie()) {
1534 next *= 0.5f;
1535 }
1536 } else {
1537 next = (scalar * 0.15f) * 0.08f;
1538 }
1539 scalar = m_driftingParams->boostRotFactor;
1540 }
1541 } else {
1542 constexpr s16 MAX_SSMT_CHARGE = 75;
1543 next = 0.015f * (-static_cast<f32>(m_ssmtCharge) / static_cast<f32>(MAX_SSMT_CHARGE));
1544 }
1545 }
1546
1547 if (m_flags.onBit(eFlags::WallBounce)) {
1548 m_standStillBoostRot = isBike() ? next * 3.0f : next * 10.0f;
1549 } else {
1550 m_standStillBoostRot += scalar * (next - m_standStillBoostRot);
1551 }
1552}
1553
1558 constexpr f32 DIVE_LIMIT = 0.8f;
1559
1560 m_divingRot *= 0.96f;
1561
1562 if (state()->isTouchingGround() || state()->isCannonStart() || state()->isInCannon() ||
1563 state()->isInAction() || state()->isOverZipper()) {
1564 return;
1565 }
1566
1567 f32 stickY = state()->stickY();
1568
1569 if (state()->isInATrick() && m_jump->type() == TrickType::BikeSideStuntTrick) {
1570 stickY = std::min(1.0f, stickY + 0.4f);
1571 }
1572
1573 u32 airtime = state()->airtime();
1574
1575 if (airtime > 50) {
1576 if (EGG::Mathf::abs(stickY) < 0.1f) {
1577 m_divingRot += 0.05f * (-0.025f - m_divingRot);
1578 }
1579 } else {
1580 stickY *= (airtime / 50.0f);
1581 }
1582
1583 m_divingRot = std::max(-DIVE_LIMIT, std::min(DIVE_LIMIT, m_divingRot + stickY * 0.005f));
1584
1585 EGG::Vector3f angVel2 = dynamics()->angVel2();
1586 angVel2.x += m_divingRot;
1587 dynamics()->setAngVel2(angVel2);
1588
1589 if (state()->airtime() < 50) {
1590 return;
1591 }
1592
1593 EGG::Vector3f topRotated = dynamics()->mainRot().rotateVector(EGG::Vector3f::ey);
1594 EGG::Vector3f forwardRotated = dynamics()->mainRot().rotateVector(EGG::Vector3f::ez);
1595 f32 upDotTop = m_up.dot(topRotated);
1596 EGG::Vector3f upCrossTop = m_up.cross(topRotated);
1597 f32 crossNorm = upCrossTop.length();
1598 f32 angle = EGG::Mathf::abs(EGG::Mathf::atan2(crossNorm, upDotTop));
1599
1600 f32 fVar1 = angle * RAD2DEG - 20.0f;
1601 if (fVar1 <= 0.0f) {
1602 return;
1603 }
1604
1605 f32 mult = std::min(1.0f, fVar1 / 20.0f);
1606 if (forwardRotated.y > 0.0f) {
1607 dynamics()->setGravity((1.0f - 0.2f * mult) * dynamics()->gravity());
1608 } else {
1609 dynamics()->setGravity((0.2f * mult + 1.0f) * dynamics()->gravity());
1610 }
1611}
1612
1617 if (EGG::Mathf::abs(m_speed) >= 10.0f || state()->isBoost() || state()->isRampBoost() ||
1618 !state()->isAccelerate() || !state()->isBrake()) {
1619 state()->setChargingSsmt(false);
1620 return;
1621 }
1622
1623 state()->setChargingSsmt(true);
1624 state()->setHopStart(false);
1625 state()->setDriftInput(false);
1626}
1627
1629void KartMove::calcHopPhysics() {
1630 m_hopVelY = m_hopVelY * 0.998f + m_hopGravity;
1632
1633 if (m_hopPosY < 0.0f) {
1634 m_hopPosY = 0.0f;
1635 m_hopVelY = 0.0f;
1636 }
1637}
1638
1640void KartMove::calcRejectRoad() {
1641 m_reject.calcRejectRoad();
1642}
1643
1645bool KartMove::calcZipperCollision(f32 radius, f32 scale, EGG::Vector3f &pos,
1646 EGG::Vector3f &upLocal, const EGG::Vector3f &prevPos, Field::CollisionInfo *colInfo,
1647 Field::KCLTypeMask *maskOut, Field::KCLTypeMask flags) const {
1648 upLocal = mainRot().rotateVector(EGG::Vector3f::ey);
1649 pos = dynamics()->pos() + (-scale * m_scale.y) * upLocal;
1650
1651 auto *colDir = Field::CollisionDirector::Instance();
1652 return colDir->checkSphereFullPush(radius, pos, prevPos, flags, colInfo, maskOut, 0);
1653}
1654
1656f32 KartMove::calcSlerpRate(f32 scale, const EGG::Quatf &from, const EGG::Quatf &to) const {
1657 f32 dotNorm = std::max(-1.0f, std::min(1.0f, from.dot(to)));
1658 f32 acos = EGG::Mathf::acos(dotNorm);
1659 return acos > 0.0f ? std::min(0.1f, scale / acos) : 0.1f;
1660}
1661
1665 f32 tiltMagnitude = 0.0f;
1666
1667 if (!state()->isInAction() && !state()->isSoftWallDrift() && state()->isAnyWheelCollision()) {
1668 EGG::Vector3f front = componentZAxis();
1669 front = front.perpInPlane(m_up, true);
1670 EGG::Vector3f frontSpeed = velocity().rej(front).perpInPlane(m_up, false);
1671 f32 magnitude = tiltMagnitude;
1672
1673 if (frontSpeed.squaredLength() > std::numeric_limits<f32>::epsilon()) {
1674 magnitude = frontSpeed.length();
1675
1676 if (front.z * frontSpeed.x - front.x * frontSpeed.z > 0.0f) {
1677 magnitude = -magnitude;
1678 }
1679
1680 tiltMagnitude = -1.0f;
1681 if (-1.0f <= magnitude) {
1682 tiltMagnitude = std::min(1.0f, magnitude);
1683 }
1684 }
1685 } else if (!state()->isHop() || m_hopPosY <= 0.0f) {
1686 EGG::Vector3f angVel0 = dynamics()->angVel0();
1687 angVel0.z *= 0.98f;
1688 dynamics()->setAngVel0(angVel0);
1689 }
1690
1691 f32 lean = EGG::Mathf::abs(m_weightedTurn) * (tiltMagnitude * param()->stats().tilt);
1692
1694
1695 EGG::Vector3f angVel0 = dynamics()->angVel0();
1696 angVel0.x += m_standStillBoostRot;
1697 angVel0.z += lean;
1698 dynamics()->setAngVel0(angVel0);
1699
1700 EGG::Vector3f angVel2 = dynamics()->angVel2();
1701 angVel2.y += turn;
1702 dynamics()->setAngVel2(angVel2);
1703
1704 calcDive();
1705}
1706
1711 // TODO: Some of these are shared between the base and derived class implementations.
1712 constexpr u16 MAX_MT_CHARGE = 270;
1713 constexpr u16 MAX_SMT_CHARGE = 300;
1714 constexpr u16 BASE_MT_CHARGE = 2;
1715 constexpr u16 BASE_SMT_CHARGE = 2;
1716 constexpr f32 BONUS_CHARGE_STICK_THRESHOLD = 0.4f;
1717 constexpr u16 EXTRA_MT_CHARGE = 3;
1718
1719 if (m_driftState == DriftState::ChargedSmt) {
1720 return;
1721 }
1722
1723 f32 stickX = state()->stickX();
1724
1725 if (m_driftState == DriftState::ChargingMt) {
1726 m_mtCharge += BASE_MT_CHARGE;
1727
1728 if (-BONUS_CHARGE_STICK_THRESHOLD <= stickX) {
1729 if (BONUS_CHARGE_STICK_THRESHOLD < stickX && m_hopStickX == -1) {
1730 m_mtCharge += EXTRA_MT_CHARGE;
1731 }
1732 } else if (m_hopStickX != -1) {
1733 m_mtCharge += EXTRA_MT_CHARGE;
1734 }
1735
1736 if (m_mtCharge > MAX_MT_CHARGE) {
1737 m_mtCharge = MAX_MT_CHARGE;
1738 m_driftState = DriftState::ChargingSmt;
1739 }
1740 }
1741
1742 if (m_driftState != DriftState::ChargingSmt) {
1743 return;
1744 }
1745
1746 m_smtCharge += BASE_SMT_CHARGE;
1747
1748 if (-BONUS_CHARGE_STICK_THRESHOLD <= stickX) {
1749 if (BONUS_CHARGE_STICK_THRESHOLD < stickX && m_hopStickX == -1) {
1750 m_smtCharge += EXTRA_MT_CHARGE;
1751 }
1752 } else if (m_hopStickX != -1) {
1753 m_smtCharge += EXTRA_MT_CHARGE;
1754 }
1755
1756 if (m_smtCharge > MAX_SMT_CHARGE) {
1757 m_smtCharge = MAX_SMT_CHARGE;
1758 m_driftState = DriftState::ChargedSmt;
1759 }
1760}
1761
1766 state()->setHop(true);
1767 state()->setDriftManual(false);
1768 onHop();
1769
1770 m_hopUp = dynamics()->mainRot().rotateVector(EGG::Vector3f::ey);
1771 m_hopDir = dynamics()->mainRot().rotateVector(EGG::Vector3f::ez);
1772 m_driftState = DriftState::NotDrifting;
1773 m_smtCharge = 0;
1774 m_mtCharge = 0;
1775 m_hopStickX = 0;
1776 m_hopFrame = 0;
1777 m_hopPosY = 0.0f;
1778 m_hopGravity = dynamics()->gravity();
1779 m_hopVelY = m_driftingParams->hopVelY;
1780 m_outsideDriftBonus = 0.0f;
1781
1782 EGG::Vector3f extVel = dynamics()->extVel();
1783 extVel.y = 0.0f + m_hopVelY;
1784 dynamics()->setExtVel(extVel);
1785
1786 EGG::Vector3f totalForce = dynamics()->totalForce();
1787 totalForce.y = 0.0f;
1788 dynamics()->setTotalForce(totalForce);
1789}
1790
1792void KartMove::tryStartBoostPanel() {
1793 constexpr s16 BOOST_PANEL_DURATION = 60;
1794
1795 if (state()->isBeforeRespawn()) {
1796 return;
1797 }
1798
1799 activateBoost(KartBoost::Type::MushroomAndBoostPanel, BOOST_PANEL_DURATION);
1800 setOffroadInvincibility(BOOST_PANEL_DURATION);
1801}
1802
1807 constexpr s16 BOOST_RAMP_DURATION = 60;
1808
1809 if (state()->isBeforeRespawn()) {
1810 return;
1811 }
1812
1813 state()->setRampBoost(true);
1814 m_rampBoost = BOOST_RAMP_DURATION;
1815 setOffroadInvincibility(BOOST_RAMP_DURATION);
1816}
1817
1823 static constexpr std::array<JumpPadProperties, 8> JUMP_PAD_PROPERTIES = {{
1824 {50.0f, 50.0f, 35.0f},
1825 {50.0f, 50.0f, 47.0f},
1826 {59.0f, 59.0f, 30.0f},
1827 {73.0f, 73.0f, 45.0f},
1828 {73.0f, 73.0f, 53.0f},
1829 {55.0f, 55.0f, 35.0f},
1830 {56.0f, 56.0f, 50.0f},
1831 }};
1832
1833 if (state()->isBeforeRespawn() || state()->isInAction() || state()->isHalfPipeRamp()) {
1834 return;
1835 }
1836
1837 state()->setJumpPad(true);
1838 s32 jumpPadVariant = state()->jumpPadVariant();
1839 m_jumpPadProperties = &JUMP_PAD_PROPERTIES[jumpPadVariant];
1840
1841 if (jumpPadVariant != 4) {
1842 EGG::Vector3f extVel = dynamics()->extVel();
1843 EGG::Vector3f totalForce = dynamics()->totalForce();
1844
1845 extVel.y = m_jumpPadProperties->velY;
1846 totalForce.y = 0.0f;
1847
1848 dynamics()->setExtVel(extVel);
1849 dynamics()->setTotalForce(totalForce);
1850
1851 if (jumpPadVariant != 3) {
1852 EGG::Vector3f dir = m_dir;
1853 dir.y = 0.0f;
1854 dir.normalise();
1855 m_speed *= m_dir.dot(dir);
1856 m_dir = dir;
1857 m_vel1Dir = dir;
1858 state()->setJumpPadDisableYsusForce(true);
1859 }
1860 }
1861
1862 m_jumpPadMinSpeed = m_jumpPadProperties->minSpeed;
1863 m_jumpPadMaxSpeed = m_jumpPadProperties->maxSpeed;
1864 m_speed = std::max(m_speed, m_jumpPadMinSpeed);
1865}
1866
1868void KartMove::tryEndJumpPad() {
1869 if (state()->isGroundStart()) {
1870 cancelJumpPad();
1871 }
1872}
1873
1875void KartMove::cancelJumpPad() {
1876 m_jumpPadMinSpeed = 0.0f;
1877 state()->setJumpPad(false);
1878}
1879
1881void KartMove::activateBoost(KartBoost::Type type, s16 frames) {
1882 if (m_boost.activate(type, frames)) {
1883 state()->setBoost(true);
1884 }
1885}
1886
1888void KartMove::applyStartBoost(s16 frames) {
1889 activateBoost(KartBoost::Type::AllMt, frames);
1890}
1891
1893void KartMove::activateMushroom() {
1894 constexpr s16 MUSHROOM_DURATION = 90;
1895
1896 if (state()->isBeforeRespawn() || state()->isInAction()) {
1897 return;
1898 }
1899
1900 activateBoost(KartBoost::Type::MushroomAndBoostPanel, MUSHROOM_DURATION);
1901
1902 m_mushroomBoostTimer = MUSHROOM_DURATION;
1903 state()->setMushroomBoost(true);
1904 setOffroadInvincibility(MUSHROOM_DURATION);
1905}
1906
1908void KartMove::activateZipperBoost() {
1909 constexpr s16 BASE_DURATION = 50;
1910 constexpr s16 TRICK_DURATION = 100;
1911
1912 if (state()->isBeforeRespawn() || state()->isInAction()) {
1913 return;
1914 }
1915
1916 s16 boostDuration = state()->isZipperTrick() ? TRICK_DURATION : BASE_DURATION;
1917 activateBoost(KartBoost::Type::TrickAndZipper, boostDuration);
1918
1919 setOffroadInvincibility(boostDuration);
1920 m_zipperBoostTimer = 0;
1921 m_zipperBoostMax = boostDuration;
1922 state()->setZipperBoost(true);
1923}
1924
1930 if (timer > m_offroadInvincibility) {
1931 m_offroadInvincibility = timer;
1932 }
1933
1934 state()->setBoostOffroadInvincibility(true);
1935}
1936
1941 if (!state()->isBoostOffroadInvincibility()) {
1942 return;
1943 }
1944
1945 if (--m_offroadInvincibility > 0) {
1946 return;
1947 }
1948
1949 state()->setBoostOffroadInvincibility(false);
1950}
1951
1955 if (!state()->isMushroomBoost()) {
1956 return;
1957 }
1958
1959 if (--m_mushroomBoostTimer > 0) {
1960 return;
1961 }
1962
1963 state()->setMushroomBoost(false);
1964}
1965
1967void KartMove::calcZipperBoost() {
1968 if (!state()->isZipperBoost()) {
1969 return;
1970 }
1971
1972 state()->setAccelerate(true);
1973
1974 if (!state()->isOverZipper() && ++m_zipperBoostTimer >= m_zipperBoostMax) {
1975 m_zipperBoostTimer = 0;
1976 state()->setZipperBoost(false);
1977 }
1978
1979 if (m_zipperBoostTimer < 10) {
1980 EGG::Vector3f angVel = dynamics()->angVel0();
1981 angVel.y = 0.0f;
1982 dynamics()->setAngVel0(angVel);
1983 }
1984}
1985
1987void KartMove::landTrick() {
1988 static constexpr std::array<s16, 3> KART_TRICK_BOOST_DURATION = {{
1989 40,
1990 70,
1991 85,
1992 }};
1993 static constexpr std::array<s16, 3> BIKE_TRICK_BOOST_DURATION = {{
1994 45,
1995 80,
1996 95,
1997 }};
1998
1999 if (state()->isBeforeRespawn() || state()->isInAction()) {
2000 return;
2001 }
2002
2003 s16 duration;
2004 if (isBike()) {
2005 duration = BIKE_TRICK_BOOST_DURATION[static_cast<u32>(m_jump->variant())];
2006 } else {
2007 duration = KART_TRICK_BOOST_DURATION[static_cast<u32>(m_jump->variant())];
2008 }
2009
2010 activateBoost(KartBoost::Type::TrickAndZipper, duration);
2011}
2012
2014void KartMove::enterCannon() {
2015 init(true, true);
2016 physics()->clearDecayingRot();
2017 m_boost.resetActive();
2018 state()->setBoost(false);
2019
2020 cancelJumpPad();
2021 clearRampBoost();
2022 clearZipperBoost();
2023 clearSsmt();
2024 clearOffroadInvincibility();
2025
2026 dynamics()->reset();
2027
2028 clearDrift();
2029 state()->setHop(false);
2030 state()->setInCannon(true);
2031 state()->setSkipWheelCalc(true);
2032 state()->setCannonStart(false);
2033
2034 const auto [cannonPos, cannonDir] = getCannonPosRot();
2035 m_cannonEntryPos = pos();
2036 m_cannonEntryOfs = cannonPos - pos();
2037 m_cannonEntryOfsLength = m_cannonEntryOfs.normalise();
2038 m_cannonEntryOfs.normalise();
2039 m_dir = m_cannonEntryOfs;
2040 m_vel1Dir = m_cannonEntryOfs;
2041 m_cannonOrthog = EGG::Vector3f::ey.perpInPlane(m_cannonEntryOfs, true);
2042 m_cannonProgress.setZero();
2043}
2044
2046void KartMove::calcCannon() {
2047 auto [cannonPos, cannonDir] = getCannonPosRot();
2048 EGG::Vector3f forwardXZ = cannonPos - m_cannonEntryPos - m_cannonProgress;
2049 EGG::Vector3f forward = forwardXZ;
2050 f32 forwardLength = forward.normalise();
2051 forwardXZ.y = 0;
2052 forwardXZ.normalise();
2053 EGG::Vector3f local94 = m_cannonEntryOfs;
2054 local94.y = 0;
2055 local94.normalise();
2056 m_speedRatioCapped = 1.0f;
2057 m_speedRatio = 1.5f;
2058 EGG::Matrix34f cannonOrientation;
2059 cannonOrientation.makeOrthonormalBasis(forward, EGG::Vector3f::ey);
2060 EGG::Vector3f up = cannonOrientation.multVector33(EGG::Vector3f::ey);
2061 m_smoothedUp = up;
2062 m_up = up;
2063
2064 if (forwardLength < 30.0f || local94.dot(forwardXZ) <= 0.0f) {
2065 exitCannon();
2066 return;
2067 }
2069 const auto *cannonPoint =
2070 System::CourseMap::Instance()->getCannonPoint(state()->cannonPointId());
2071 size_t cannonParameterIdx = std::max<s16>(0, cannonPoint->parameterIdx());
2072 ASSERT(cannonParameterIdx < CANNON_PARAMETERS.size());
2073 const auto &cannonParams = CANNON_PARAMETERS[cannonParameterIdx];
2074 f32 newSpeed = cannonParams.speed;
2075 if (forwardLength < cannonParams.decelFactor) {
2076 f32 factor = std::max(0.0f, forwardLength / cannonParams.decelFactor);
2077
2078 newSpeed = cannonParams.endDecel;
2079 if (newSpeed <= 0.0f) {
2080 newSpeed = m_baseSpeed;
2081 }
2082
2083 newSpeed += factor * (cannonParams.speed - newSpeed);
2084 if (cannonParams.endDecel > 0.0f) {
2085 m_speed = std::min(newSpeed, m_speed);
2086 }
2087 }
2088
2089 m_cannonProgress += m_cannonEntryOfs * newSpeed;
2090
2091 EGG::Vector3f newPos = EGG::Vector3f::zero;
2092 if (cannonParams.height > 0.0f) {
2093 f32 fVar9 = EGG::Mathf::SinFIdx(
2094 (1.0f - (forwardLength / m_cannonEntryOfsLength)) * 180.0f * DEG2FIDX);
2095 newPos = fVar9 * cannonParams.height * m_cannonOrthog;
2096 }
2097
2098 dynamics()->setPos(m_cannonEntryPos + m_cannonProgress + newPos);
2099 m_dir = m_cannonEntryOfs;
2100 m_vel1Dir = m_cannonEntryOfs;
2101
2102 calcRotCannon(forward);
2103
2104 dynamics()->setExtVel(EGG::Vector3f::zero);
2105}
2106
2108void KartMove::calcRotCannon(const EGG::Vector3f &forward) {
2109 EGG::Vector3f local48 = forward;
2110 local48.normalise();
2111 EGG::Vector3f local54 = bodyFront();
2112 EGG::Vector3f local60 = local54 + ((local48 - local54) * 0.3f);
2113 local54.normalise();
2114 local60.normalise();
2115 // also local70, localA8
2116 EGG::Quatf local80;
2117 local80.makeVectorRotation(local54, local60);
2118 local80 *= dynamics()->fullRot();
2119 local80.normalise();
2120 EGG::Quatf localB8;
2121 localB8.makeVectorRotation(local80.rotateVector(EGG::Vector3f::ey), smoothedUp());
2122 EGG::Quatf newRot = local80.slerpTo(localB8.multSwap(local80), 0.3f);
2123 dynamics()->setFullRot(newRot);
2124 dynamics()->setMainRot(newRot);
2125}
2126
2128void KartMove::exitCannon() {
2129 if (!state()->isInCannon()) {
2130 return;
2131 }
2132
2133 state()->setInCannon(false);
2134 state()->setSkipWheelCalc(false);
2135 state()->setAfterCannon(true);
2136 dynamics()->setIntVel(m_cannonEntryOfs * m_speed);
2137}
2138
2140void KartMove::triggerRespawn() {
2141 m_timeInRespawn = 0;
2142 state()->setTriggerRespawn(true);
2143}
2144
2146KartMoveBike::KartMoveBike() : m_leanRot(0.0f) {}
2147
2149KartMoveBike::~KartMoveBike() = default;
2150
2154 constexpr f32 MAX_WHEELIE_ROTATION = 0.07f;
2155 constexpr u16 WHEELIE_COOLDOWN = 20;
2156
2157 state()->setWheelie(true);
2158 m_wheelieFrames = 0;
2159 m_maxWheelieRot = MAX_WHEELIE_ROTATION;
2160 m_wheelieCooldown = WHEELIE_COOLDOWN;
2161 m_wheelieRotDec = 0.0f;
2162 m_autoHardStickXFrames = 0;
2163}
2164
2169 state()->setWheelie(false);
2170 m_wheelieRotDec = 0.0f;
2171 m_autoHardStickXFrames = 0;
2172}
2173
2175void KartMoveBike::createSubsystems() {
2176 m_jump = new KartJumpBike(this);
2177 m_halfPipe = new KartHalfPipe();
2178}
2179
2184 f32 leanRotInc = m_turningParams->leanRotIncRace;
2185 f32 leanRotCap = m_turningParams->leanRotCapRace;
2186 const auto *raceManager = System::RaceManager::Instance();
2187
2188 if (!state()->isChargingSsmt()) {
2189 if (!raceManager->isStageReached(System::RaceManager::Stage::Race) ||
2190 EGG::Mathf::abs(m_speed) < 5.0f) {
2191 leanRotInc = m_turningParams->leanRotIncCountdown;
2192 leanRotCap = m_turningParams->leanRotCapCountdown;
2193 }
2194 } else {
2195 leanRotInc = m_turningParams->leanRotIncSSMT;
2196 leanRotCap = m_turningParams->leanRotCapSSMT;
2197 }
2198
2199 m_leanRotCap += 0.3f * (leanRotCap - m_leanRotCap);
2200 m_leanRotInc += 0.3f * (leanRotInc - m_leanRotInc);
2201
2202 f32 stickX = state()->stickX();
2203 f32 extVelXFactor = 0.0f;
2204 f32 leanRotMin = -m_leanRotCap;
2205 f32 leanRotMax = m_leanRotCap;
2206
2207 if (state()->isBeforeRespawn() || state()->isInAction() || state()->isWheelie() ||
2208 state()->isOverZipper() || state()->isRejectRoadTrigger() ||
2209 state()->isAirtimeOver20() || state()->isSoftWallDrift() ||
2210 state()->isSomethingWallCollision() || state()->isHWG() || state()->isCannonStart() ||
2211 state()->isInCannon()) {
2212 m_leanRot *= m_turningParams->leanRotDecayFactor;
2213 } else if (!state()->isDrifting()) {
2214 if (stickX <= 0.2f) {
2215 if (stickX >= -0.2f) {
2216 m_leanRot *= m_turningParams->leanRotDecayFactor;
2217 } else {
2219 extVelXFactor = m_turningParams->leanRotShallowFactor;
2220 }
2221 } else {
2223 extVelXFactor = -m_turningParams->leanRotShallowFactor;
2224 }
2225 } else {
2226 leanRotMax = m_turningParams->leanRotMaxDrift;
2227 leanRotMin = m_turningParams->leanRotMinDrift;
2228
2229 if (m_hopStickX == 1) {
2230 leanRotMin = -leanRotMax;
2231 leanRotMax = -m_turningParams->leanRotMinDrift;
2232 }
2233 if (m_hopStickX == -1) {
2234 if (stickX == 0.0f) {
2235 m_leanRot += (0.5f - m_leanRot) * 0.05f;
2236 } else {
2237 m_leanRot += m_turningParams->driftStickXFactor * stickX;
2238 extVelXFactor = -m_turningParams->leanRotShallowFactor * stickX;
2239 }
2240 } else if (stickX == 0.0f) {
2241 m_leanRot += (-0.5f - m_leanRot) * 0.05f;
2242 } else {
2243 m_leanRot += m_turningParams->driftStickXFactor * stickX;
2244 extVelXFactor = -m_turningParams->leanRotShallowFactor * stickX;
2245 }
2246 }
2247
2248 bool capped = false;
2249 if (leanRotMin <= m_leanRot) {
2250 if (leanRotMax < m_leanRot) {
2251 m_leanRot = leanRotMax;
2252 capped = true;
2253 }
2254 } else {
2255 m_leanRot = leanRotMin;
2256 capped = true;
2257 }
2258
2259 if (!capped) {
2260 dynamics()->setExtVel(dynamics()->extVel() + componentXAxis() * extVelXFactor);
2261 }
2262
2263 f32 leanRotScalar = state()->isDrifting() ? 0.065f : 0.05f;
2264
2266
2267 dynamics()->setAngVel2(dynamics()->angVel2() +
2268 EGG::Vector3f(m_standStillBoostRot, turn * wheelieRotFactor(),
2269 m_leanRot * leanRotScalar));
2270
2271 calcDive();
2272
2273 EGG::Vector3f top = m_up;
2274
2275 if (!state()->isRejectRoad() && !state()->isHalfPipeRamp() && !state()->isOverZipper()) {
2276 f32 scalar = (m_speed >= 0.0f) ? m_speedRatioCapped * 2.0f : 0.0f;
2277 scalar = std::min(1.0f, scalar);
2278 top = scalar * m_up + (1.0f - scalar) * EGG::Vector3f::ey;
2279
2280 if (std::numeric_limits<f32>::epsilon() < top.squaredLength()) {
2281 top.normalise();
2282 }
2283 }
2284
2285 dynamics()->setTop_(top);
2286}
2287
2293 static constexpr std::array<TurningParameters, 2> TURNING_PARAMS_ARRAY = {{
2294 {0.8f, 0.08f, 1.0f, 0.1f, 1.2f, 0.8f, 0.08f, 0.6f, 0.15f, 1.6f, 0.9f, 180},
2295 {1.0f, 0.1f, 1.0f, 0.05f, 1.5f, 0.7f, 0.08f, 0.6f, 0.15f, 1.3f, 0.9f, 180},
2296 }};
2297
2298 KartMove::setTurnParams();
2299
2300 if (param()->stats().driftType == KartParam::Stats::DriftType::Outside_Drift_Bike) {
2301 m_turningParams = &TURNING_PARAMS_ARRAY[0];
2302 } else if (param()->stats().driftType == KartParam::Stats::DriftType::Inside_Drift_Bike) {
2303 m_turningParams = &TURNING_PARAMS_ARRAY[1];
2304 }
2305
2306 if (System::RaceManager::Instance()->isStageReached(System::RaceManager::Stage::Race)) {
2307 m_leanRotInc = m_turningParams->leanRotIncRace;
2308 m_leanRotCap = m_turningParams->leanRotCapRace;
2309 } else {
2310 m_leanRotInc = m_turningParams->leanRotIncCountdown;
2311 m_leanRotCap = m_turningParams->leanRotCapCountdown;
2312 }
2313}
2314
2316void KartMoveBike::init(bool b1, bool b2) {
2317 KartMove::init(b1, b2);
2318
2319 m_leanRot = 0.0f;
2320 m_leanRotCap = 0.0f;
2321 m_leanRotInc = 0.0f;
2322 m_wheelieRot = 0.0f;
2323 m_maxWheelieRot = 0.0f;
2324 m_wheelieFrames = 0;
2326 m_autoHardStickXFrames = 0;
2327}
2328
2330void KartMoveBike::clear() {
2331 KartMove::clear();
2332 cancelWheelie();
2333}
2334
2338 constexpr u32 FAILED_WHEELIE_FRAMES = 15;
2339 constexpr f32 AUTO_WHEELIE_CANCEL_STICK_THRESHOLD = 0.85f;
2340
2342 m_wheelieCooldown = std::max(0, m_wheelieCooldown - 1);
2343
2344 if (state()->isWheelie()) {
2345 bool cancelAutoWheelie = false;
2346
2347 if (!state()->isAutoDrift() ||
2348 EGG::Mathf::abs(state()->stickX()) <= AUTO_WHEELIE_CANCEL_STICK_THRESHOLD) {
2349 m_autoHardStickXFrames = 0;
2350 } else {
2351 if (++m_autoHardStickXFrames > 15) {
2352 cancelAutoWheelie = true;
2353 }
2354 }
2355
2357 if (m_turningParams->maxWheelieFrames < m_wheelieFrames || cancelAutoWheelie ||
2358 (!canWheelie() && FAILED_WHEELIE_FRAMES <= m_wheelieFrames)) {
2359 cancelWheelie();
2360 } else {
2361 m_wheelieRot += 0.01f;
2362 EGG::Vector3f angVel0 = dynamics()->angVel0();
2363 angVel0.x *= 0.9f;
2364 dynamics()->setAngVel0(angVel0);
2365 }
2366 } else if (0.0f < m_wheelieRot) {
2367 m_wheelieRotDec -= 0.001f;
2368 m_wheelieRotDec = std::max(-0.03f, m_wheelieRotDec);
2370 }
2371
2372 m_wheelieRot = std::max(0.0f, std::min(m_wheelieRot, m_maxWheelieRot));
2373
2374 f32 vel1DirUp = m_vel1Dir.dot(EGG::Vector3f::ey);
2375
2376 if (m_wheelieRot > 0.0f) {
2377 if (vel1DirUp <= 0.5f || m_wheelieFrames < FAILED_WHEELIE_FRAMES) {
2378 EGG::Vector3f angVel2 = dynamics()->angVel2();
2379 angVel2.x -= m_wheelieRot * (1.0f - EGG::Mathf::abs(vel1DirUp));
2380 dynamics()->setAngVel2(angVel2);
2381 } else {
2382 cancelWheelie();
2383 }
2384
2385 state()->setWheelieRot(true);
2386 } else {
2387 state()->setWheelieRot(false);
2388 }
2389}
2390
2397 if (state()->isAutoDrift()) {
2398 return;
2399 }
2400
2401 cancelWheelie();
2402}
2403
2409
2414 constexpr u16 MAX_MT_CHARGE = 270;
2415 constexpr u16 BASE_MT_CHARGE = 2;
2416 constexpr f32 BONUS_CHARGE_STICK_THRESHOLD = 0.4f;
2417 constexpr u16 EXTRA_MT_CHARGE = 3;
2418
2419 if (m_driftState != DriftState::ChargingMt) {
2420 return;
2421 }
2422
2423 m_mtCharge += BASE_MT_CHARGE;
2424
2425 f32 stickX = state()->stickX();
2426 if (-BONUS_CHARGE_STICK_THRESHOLD <= stickX) {
2427 if (BONUS_CHARGE_STICK_THRESHOLD < stickX && m_hopStickX == -1) {
2428 m_mtCharge += EXTRA_MT_CHARGE;
2429 }
2430 } else if (m_hopStickX != -1) {
2431 m_mtCharge += EXTRA_MT_CHARGE;
2432 }
2433
2434 if (m_mtCharge > MAX_MT_CHARGE) {
2435 m_mtCharge = MAX_MT_CHARGE;
2436 m_driftState = DriftState::ChargedMt;
2437 }
2438}
2439
2441void KartMoveBike::initOob() {
2442 clearBoost();
2443 clearJumpPad();
2444 clearRampBoost();
2445 clearZipperBoost();
2446 clearSsmt();
2447 clearOffroadInvincibility();
2448 cancelWheelie();
2449}
2450
2454 constexpr s16 COOLDOWN_FRAMES = 20;
2455 bool dpadUp = inputs()->currentState().trickUp();
2456
2457 if (!state()->isWheelie()) {
2458 if (dpadUp && state()->isTouchingGround()) {
2459 if (state()->isDriftManual() || state()->isWallCollision() ||
2460 state()->isWall3Collision() || state()->isHop() || state()->isDriftAuto() ||
2461 state()->isInAction()) {
2462 return;
2463 }
2464
2465 if (m_wheelieCooldown > 0) {
2466 return;
2467 }
2468
2469 startWheelie();
2470 }
2471 } else if (inputs()->currentState().trickDown() && m_wheelieCooldown <= 0) {
2472 cancelWheelie();
2473 m_wheelieCooldown = COOLDOWN_FRAMES;
2474 }
2475}
2476
2477} // namespace Kart
@ COL_TYPE_MOVING_WATER
Koopa Cape and DS Yoshi Falls.
@ COL_TYPE_STICKY_ROAD
Player sticks if within 200 units (rBC stairs).
#define KCL_TYPE_BIT(x)
#define KCL_TYPE_FLOOR
0x20E80FFF - Any KCL that the player or items can drive/land on.
A 3 x 4 matrix.
Definition Matrix.hh:8
void makeOrthonormalBasis(const Vector3f &v0, const Vector3f &v1)
Sets a 3x3 orthonormal basis for a local coordinate system.
Definition Matrix.cc:162
void setAxisRotation(f32 angle, const Vector3f &axis)
Rotates the matrix about an axis.
Definition Matrix.cc:175
Vector3f multVector33(const Vector3f &vec) const
Multiplies a 3x3 matrix by a vector.
Definition Matrix.cc:243
Vector3f multVector(const Vector3f &vec) const
Multiplies a vector by a matrix.
Definition Matrix.cc:220
bool activate(Type type, s16 frames)
Starts/restarts a boost of the given type.
Definition KartBoost.cc:21
bool calc()
Computes the current frame's boost multiplier, acceleration, and speed limit.
Definition KartBoost.cc:38
void applyWrenchScaled(const EGG::Vector3f &p, const EGG::Vector3f &f, f32 scale)
Applies a force linearly and rotationally to the kart.
Handles the physics and boosts associated with zippers.
f32 m_leanRot
Z-axis rotation of the bike from leaning.
Definition KartMove.hh:489
f32 m_leanRotCap
The maximum leaning rotation.
Definition KartMove.hh:490
void calcWheelie() override
STAGE 1+ - Every frame, checks player input for wheelies and computes wheelie rotation.
Definition KartMove.cc:2337
void calcMtCharge() override
Every frame during a drift, calculates MT charge based on player input.
Definition KartMove.cc:2413
virtual void startWheelie()
STAGE 1+ - Sets the wheelie bit flag and some wheelie-related variables.
Definition KartMove.cc:2153
f32 m_wheelieRotDec
The wheelie rotation decrementor, used after a wheelie has ended.
Definition KartMove.hh:496
u32 m_wheelieFrames
Tracks wheelie duration and cancels the wheelie after 180 frames.
Definition KartMove.hh:494
f32 m_wheelieRot
X-axis rotation from wheeling.
Definition KartMove.hh:492
const TurningParameters * m_turningParams
Inside/outside drifting bike turn info.
Definition KartMove.hh:498
void setTurnParams() override
On init, sets the bike's lean rotation cap and increment.In addition to setting the lean rotation cap...
Definition KartMove.cc:2292
f32 m_maxWheelieRot
The maximum wheelie rotation.
Definition KartMove.hh:493
void tryStartWheelie()
STAGE 1+ - Every frame, checks player input to see if we should start or stop a wheelie.
Definition KartMove.cc:2453
void onWallCollision() override
Called when you collide with a wall. All it does for bikes is cancel wheelies.
Definition KartMove.cc:2406
void onHop() override
Virtual function that just cancels wheelies when you hop.
Definition KartMove.cc:2396
bool canWheelie() const override
Checks if the kart is going fast enough to wheelie.
Definition KartMove.hh:480
void calcVehicleRotation(f32) override
Every frame, calculates rotation, EV, and angular velocity for the bike.
Definition KartMove.cc:2183
s16 m_wheelieCooldown
The number of frames before another wheelie can start.
Definition KartMove.hh:495
f32 m_leanRotInc
The incrementor for leaning rotation.
Definition KartMove.hh:491
virtual void cancelWheelie()
Clears the wheelie bit flag and resets the rotation decrement.
Definition KartMove.cc:2168
f32 m_baseSpeed
The speed associated with the current character/vehicle stats.
Definition KartMove.hh:336
void calcRotation()
Every frame, calculates kart rotation based on player input.
Definition KartMove.cc:1086
s16 m_ssmtLeewayTimer
Frames to forgive letting go of A before clearing SSMT charge.
Definition KartMove.hh:378
s32 m_hopFrame
A timer that can prevent subsequent hops until reset.
Definition KartMove.hh:363
void calcDisableBackwardsAccel()
Computes the current cooldown duration between braking and reversing.
Definition KartMove.cc:676
EGG::Vector3f m_hopUp
The up vector when hopping.
Definition KartMove.hh:364
u16 m_mushroomBoostTimer
Number of frames until the mushroom boost runs out.
Definition KartMove.hh:385
void calcSpecialFloor()
Every frame, calculates any boost resulting from a boost panel.
Definition KartMove.cc:479
void calcWallCollisionStart(f32 param_2)
If we started to collide with a wall this frame, applies rotation.
Definition KartMove.cc:1463
KartHalfPipe * m_halfPipe
Pertains to zipper physics.
Definition KartMove.hh:410
f32 m_kclRotFactor
Float between 0-1 that scales the player's turning radius on offroad.
Definition KartMove.hh:358
f32 m_outsideDriftBonus
Added to angular velocity when outside drifting.
Definition KartMove.hh:371
void tryStartBoostRamp()
Sets offroad invincibility and and enables the ramp boost bitfield flag.
Definition KartMove.cc:1806
void calcDeceleration()
Definition KartMove.cc:1261
u16 m_smtCharge
A value between 0 and 300 representing current SMT charge.
Definition KartMove.hh:370
f32 m_speedRatio
The ratio between current speed and the player's base speed stat.
Definition KartMove.hh:356
@ DriftReset
Set when a wall bonk should cancel your drift.
@ SsmtCharged
Set after holding a stand-still mini-turbo for 75 frames.
@ TrickableSurface
Set when driving on a trickable surface.
@ SsmtLeeway
If set, activates SSMT when not pressing A or B.
@ WallBounce
Set when our speed loss from wall collision is > 30.0f.
@ Respawned
Set when Lakitu lets go of the player, cleared when landing.
void tryStartJumpPad()
Applies calculations to start interacting with a jump pad.
Definition KartMove.cc:1822
f32 m_jumpPadMinSpeed
Snaps the player to a minimum speed when first touching a jump pad.
Definition KartMove.hh:387
f32 m_hopPosY
Relative position as the result of a hop. Starts at 0.
Definition KartMove.hh:399
DrivingDirection m_drivingDirection
Current state of driver's direction.
Definition KartMove.hh:405
bool calcPreDrift()
Each frame, checks for hop or slipdrift. Computes drift direction based on player input.
Definition KartMove.cc:746
f32 m_speed
Current speed, restricted to the soft speed limit.
Definition KartMove.hh:338
s16 m_offroadInvincibility
How many frames until the player is affected by offroad.
Definition KartMove.hh:376
void calcAirtimeTop()
Calculates rotation of the bike due to excessive airtime.
Definition KartMove.cc:456
void startManualDrift()
Called when the player lands from a drift hop, or to start a slipdrift.
Definition KartMove.cc:986
void controlOutsideDriftAngle()
Every frame, handles mini-turbo charging and outside drifting bike rotation.
Definition KartMove.cc:1053
f32 m_softSpeedLimit
Base speed + boosts + wheelies, restricted to the hard speed limit.
Definition KartMove.hh:337
virtual void hop()
Initializes hop information, resets upwards EV and clears upwards force.
Definition KartMove.cc:1765
void calcStandstillBoostRot()
STAGE Computes the x-component of angular velocity based on the kart's speed.
Definition KartMove.cc:1519
EGG::Vector3f m_outsideDriftLastDir
Used to compute the next m_outsideDriftAngle.
Definition KartMove.hh:354
void calcManualDrift()
Each frame, handles hopping, drifting, and mini-turbos.
Definition KartMove.cc:911
virtual void calcMtCharge()
Every frame during a drift, calculates MT/SMT charge based on player input.
Definition KartMove.cc:1710
void calcSsmt()
Calculates standstill mini-turbo components, if applicable.
Definition KartMove.cc:691
void calcAcceleration()
Every frame, applies acceleration to the kart's internal velocity.
Definition KartMove.cc:1309
f32 m_processedSpeed
Offset 0x28. It's only ever just a copy of m_speed.
Definition KartMove.hh:340
void releaseMt()
Stops charging a mini-turbo, and applies boost if charged.
Definition KartMove.cc:1029
f32 m_kclSpeedFactor
Float between 0-1 that scales the player's speed on offroad.
Definition KartMove.hh:357
f32 m_weightedTurn
Magnitude+direction of stick input, factoring in the kart's stats.
Definition KartMove.hh:381
void calcVehicleSpeed()
Every frame, computes speed based on acceleration and any active boosts.
Definition KartMove.cc:1182
f32 m_lastSpeed
Last frame's speed, cached to calculate angular velocity.
Definition KartMove.hh:339
s16 m_ssmtDisableAccelTimer
Counter that tracks delay before starting to reverse.
Definition KartMove.hh:379
void calcOffroadInvincibility()
Checks a timer to see if we are still ignoring offroad slowdown.
Definition KartMove.cc:1940
KartBurnout m_burnout
Manages the state of start boost burnout.
Definition KartMove.hh:411
f32 calcVehicleAcceleration() const
Every frame, computes acceleration based off the character/vehicle stats.
Definition KartMove.cc:1275
void calcAutoDrift()
Each frame, handles automatic transmission drifting.
Definition KartMove.cc:859
f32 m_realTurn
The "true" turn magnitude. Equal to m_weightedTurn unless drifting.
Definition KartMove.hh:380
const DriftingParameters * m_driftingParams
Drift-type-specific parameters.
Definition KartMove.hh:412
void clearDrift()
Definition KartMove.cc:798
void calc()
Each frame, calculates the kart's movement.
Definition KartMove.cc:265
EGG::Vector3f m_up
Vector perpendicular to the floor, pointing upwards.
Definition KartMove.hh:345
s16 m_ssmtCharge
Increments every frame up to 75 when charging stand-still MT.
Definition KartMove.hh:377
f32 m_speedDragMultiplier
After 5 frames of airtime, this causes speed to slowly decay.
Definition KartMove.hh:343
u16 m_mtCharge
A value between 0 and 270 representing current MT charge.
Definition KartMove.hh:369
void calcSsmtStart()
Calculates whether we are starting a standstill mini-turbo.
Definition KartMove.cc:1616
s16 m_respawnPostLandTimer
Counts up to 4 if not accelerating after respawn landing.
Definition KartMove.hh:403
virtual f32 getWheelieSoftSpeedLimitBonus() const
Returns the % speed boost from wheelies. For karts, this is always 0.
Definition KartMove.hh:100
f32 m_kclWheelRotFactor
The slowest rotation multiplier of each wheel's floor collision.
Definition KartMove.hh:360
void resetDriftManual()
Clears drift state. Called when touching ground and drift is canceled.
Definition KartMove.cc:786
f32 m_totalScale
[Unused] Always 1.0f
Definition KartMove.hh:383
f32 m_acceleration
Captures the acceleration from player input and boosts.
Definition KartMove.hh:342
virtual void calcVehicleRotation(f32 turn)
Every frame, calculates rotation, EV, and angular velocity for the kart.
Definition KartMove.cc:1664
s16 m_respawnPreLandTimer
Counts down from 4 when pressing A before landing from respawn.
Definition KartMove.hh:402
virtual void calcTurn()
Each frame, looks at player input and kart stats. Saves turn-related info.
Definition KartMove.cc:64
f32 m_divingRot
Induces x-axis angular velocity based on up/down stick input.
Definition KartMove.hh:366
f32 m_outsideDriftAngle
The facing angle of an outward-drifting vehicle.
Definition KartMove.hh:352
EGG::Vector3f m_lastDir
m_speed from the previous frame but with signed magnitude.
Definition KartMove.hh:348
EGG::Vector3f m_smoothedUp
A smoothed up vector, mostly used after significant airtime.
Definition KartMove.hh:344
s32 getAppliedHopStickX() const
Factors in vehicle speed to retrieve our hop direction and magnitude.
Definition KartMove.hh:197
u16 m_floorCollisionCount
The number of tires colliding with the floor.
Definition KartMove.hh:361
void calcDive()
Responds to player input to handle up/down kart tilt mid-air.
Definition KartMove.cc:1557
void calcOffroad()
Each frame, computes rotation and speed scalars from the floor KCL.
Definition KartMove.cc:621
@ WaitingForBackwards
Holding reverse but waiting on a 15 frame delay.
s16 m_timeInRespawn
The number of frames elapsed after position snap from respawn.
Definition KartMove.hh:401
s32 m_hopStickX
A ternary for the direction of our hop, 0 if still neutral hopping.
Definition KartMove.hh:362
s16 m_backwardsAllowCounter
Tracks the 15f delay before reversing.
Definition KartMove.hh:406
f32 calcWallCollisionSpeedFactor(f32 &f1)
Every frame, computes a speed scalar if we are colliding with a wall.
Definition KartMove.cc:1432
void calcMushroomBoost()
Checks a timer to see if we are still boosting from a mushroom.
Definition KartMove.cc:1954
f32 m_hopGravity
Always main gravity (-1.3f).
Definition KartMove.hh:400
f32 m_hopVelY
Relative velocity due to a hop. Starts at 10 and decreases with gravity.
Definition KartMove.hh:398
f32 m_hardSpeedLimit
Absolute speed cap. It's 120, unless you're in a bullet (140).
Definition KartMove.hh:341
void setInitialPhysicsValues(const EGG::Vector3f &position, const EGG::Vector3f &angles)
Initializes the kart's position and rotation. Calls tire suspension initializers.
Definition KartMove.cc:227
f32 m_rawTurn
Float in range [-1, 1]. Represents stick magnitude + direction.
Definition KartMove.hh:413
void setOffroadInvincibility(s16 timer)
Ignores offroad KCL collision for a set amount of time.
Definition KartMove.cc:1929
f32 m_speedRatioCapped
m_speedRatio but capped at 1.0f.
Definition KartMove.hh:355
EGG::Vector3f m_scale
[Unused] Always 1.0f
Definition KartMove.hh:382
f32 m_kclWheelSpeedFactor
The slowest speed multiplier of each wheel's floor collision.
Definition KartMove.hh:359
EGG::Vector3f m_hopDir
Used for outward drift. Tracks the forward vector of our rotation.
Definition KartMove.hh:365
EGG::Vector3f bodyUp() const
Returns the second column of the rotation matrix, which is the "up" direction.
EGG::Vector3f bodyFront() const
Returns the third column of the rotation matrix, which is the facing vector.
EGG core library.
Definition Archive.cc:6
Pertains to kart-related functionality.
@ BikeSideStuntTrick
A side StuntTrick with a bike.
A quaternion, used to represent 3D rotation.
Definition Quat.hh:12
void normalise()
Scales the quaternion to a unit length.
Definition Quat.cc:23
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
Definition Quat.cc:50
Quatf slerpTo(const Quatf &q2, f32 t) const
Performs spherical linear interpolation.
Definition Quat.cc:79
void setAxisRotation(f32 angle, const Vector3f &axis)
Set the quat given angle and axis.
Definition Quat.cc:106
f32 dot(const Quatf &q) const
Computes .
Definition Quat.hh:105
void setRPY(const Vector3f &rpy)
Sets roll, pitch, and yaw.
Definition Quat.cc:7
void makeVectorRotation(const Vector3f &from, const Vector3f &to)
Captures rotation between two vectors.
Definition Quat.cc:35
constexpr TBitFlag< T, E > & resetBit(Es... es)
Resets the corresponding bits for the provided enum values.
Definition BitFlag.hh:68
constexpr bool onBit(Es... es) const
Checks if any of the corresponding bits for the provided enum values are on.
Definition BitFlag.hh:103
constexpr TBitFlag< T, E > & changeBit(bool on, Es... es)
Changes the state of the corresponding bits for the provided enum values.
Definition BitFlag.hh:80
constexpr bool offBit(Es... es) const
Checks if all of the corresponding bits for the provided enum values are off.
Definition BitFlag.hh:114
constexpr void makeAllZero()
Resets all the bits to zero.
Definition BitFlag.hh:185
constexpr TBitFlag< T, E > & setBit(Es... es)
Sets the corresponding bits for the provided enum values.
Definition BitFlag.hh:57
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
f32 squaredLength() const
The dot product between the vector and itself.
Definition Vector.hh:177
Vector3f proj(const Vector3f &rhs) const
The projection of this vector onto rhs.
Definition Vector.hh:193
Vector3f perpInPlane(const EGG::Vector3f &rhs, bool normalise) const
Calculates the orthogonal vector, based on the plane defined by this vector and rhs.
Definition Vector.cc:96
Vector3f rej(const Vector3f &rhs) const
The rejection of this vector onto rhs.
Definition Vector.hh:199
Information about the current collision and its properties.
f32 driftManualTightness
Affects turn radius when manual drifting.
Definition KartParam.hh:106
std::array< f32, 32 > kclRot
Rotation scalars, indexed using KCL attributes.
Definition KartParam.hh:113
f32 driftAutomaticTightness
Affects turn radius when automatic drifting.
Definition KartParam.hh:107
f32 driftReactivity
A weight applied to turn radius when drifting.
Definition KartParam.hh:108
f32 speed
Base full speed of the character/vehicle combo.
Definition KartParam.hh:96
f32 handlingReactivity
A weight applied to turn radius when not drifting.
Definition KartParam.hh:105
u32 miniTurbo
The framecount duration of a charged mini-turbo.
Definition KartParam.hh:111