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"
11#include "game/field/CollisionDirector.hh"
14#include "game/item/ItemDirector.hh"
15#include "game/item/KartItem.hh"
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"
22#include <egg/math/Math.hh>
23#include <egg/math/Quat.hh>
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},
41KartMove::KartMove() : m_smoothedUp(
EGG::Vector3f::ey), m_scale(1.0f, 1.0f, 1.0f) {
44 m_padType.makeAllZero();
45 m_flags.makeAllZero();
50KartMove::~KartMove() {
56void KartMove::createSubsystems() {
57 m_jump =
new KartJump(
this);
68 if (state()->isInAction() || state()->isCannonStart() || state()->isInCannon() ||
69 state()->isOverZipper()) {
73 if (state()->isBeforeRespawn()) {
79 if (state()->isAirtimeOver20()) {
87 if (state()->isDrifting()) {
98 if (!state()->isDrifting()) {
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},
119 m_landingDir = m_dir;
121 m_driftingParams = &DRIFTING_PARAMS_ARRAY[
static_cast<u32
>(param()->stats().driftType)];
125void KartMove::init(
bool b1,
bool b2) {
133 m_up = EGG::Vector3f::ey;
135 m_vel1Dir = 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;
142 m_landingAngle = 0.0f;
160 m_standStillBoostRot = 0.0f;
161 m_driftState = DriftState::NotDrifting;
166 m_zipperBoostTimer = 0;
167 m_zipperBoostMax = 0;
173 m_nonZipperAirtime = 0;
180 m_hitboxScale = 1.0f;
185 m_jumpPadMaxSpeed = 0.0f;
186 m_jumpPadProperties =
nullptr;
188 m_autoDriftAngle = 0.0f;
189 m_autoDriftStartFrameCounter = 0;
191 m_cannonEntryOfsLength = 0.0f;
192 m_cannonEntryPos.setZero();
193 m_cannonEntryOfs.setZero();
194 m_cannonOrthog.setZero();
195 m_cannonProgress.setZero();
213void KartMove::clear() {
219 clearOffroadInvincibility();
229 quaternion.
setRPY(angles * DEG2RAD);
232 Field::KCLTypeMask kcl_flags = KCL_NONE;
234 bool bColliding = Field::CollisionDirector::Instance()->checkSphereFullPush(100.0f, newPos,
235 EGG::Vector3f::inf, KCL_ANY, &info, &kcl_flags, 0);
238 newPos = newPos + info.tangentOff + (info.floorNrm * -100.0f);
239 newPos += info.floorNrm * bsp().initialYPos;
245 sub()->initPhysicsValues();
247 physics()->setPos(pos());
248 physics()->setVelocity(dynamics()->velocity());
253 dynamics()->setTop(
m_up);
255 for (
u16 tireIdx = 0; tireIdx < suspCount(); ++tireIdx) {
256 suspension(tireIdx)->setInitialState();
266 if (state()->isInRespawn()) {
271 dynamics()->resetInternalVelocity();
286 if (!state()->isAutoDrift()) {
296 if (state()->isInCannon()) {
307void KartMove::calcRespawnStart() {
308 constexpr float RESPAWN_HEIGHT = 700.0f;
310 const auto *jugemPoint = System::RaceManager::Instance()->jugemPoint();
315 respawnPos.y += RESPAWN_HEIGHT;
320 Item::ItemDirector::Instance()->kartItem(0).clear();
322 state()->setTriggerRespawn(
false);
323 state()->setInRespawn(
true);
327void KartMove::calcInRespawn() {
328 constexpr f32 LAKITU_VELOCITY = 1.5f;
329 constexpr u16 RESPAWN_DURATION = 110;
331 if (!state()->isInRespawn()) {
336 newPos.y -= LAKITU_VELOCITY;
337 dynamics()->setPos(newPos);
338 dynamics()->setNoGravity(
true);
341 state()->setInRespawn(
false);
342 state()->setAfterRespawn(
true);
343 state()->setRespawnKillY(
true);
346 dynamics()->setNoGravity(
false);
351void KartMove::calcRespawnBoost() {
352 constexpr s16 RESPAWN_BOOST_DURATION = 30;
353 constexpr s16 RESPAWN_BOOST_INPUT_LENIENCY = 4;
355 if (state()->isAfterRespawn()) {
356 if (state()->isTouchingGround()) {
358 if (!state()->isBeforeRespawn() && !state()->isInAction()) {
359 activateBoost(KartBoost::Type::AllMt, RESPAWN_BOOST_DURATION);
360 m_respawnTimer = RESPAWN_BOOST_DURATION;
366 state()->setAfterRespawn(
false);
378 if (state()->isAccelerateStart()) {
379 if (!state()->isBeforeRespawn() && !state()->isInAction()) {
380 activateBoost(KartBoost::Type::AllMt, RESPAWN_BOOST_DURATION);
381 m_respawnTimer = RESPAWN_BOOST_DURATION;
389 state()->setRespawnKillY(
false);
393 m_respawnTimer = std::max(0, m_respawnTimer - 1);
397void KartMove::calcTop() {
398 f32 stabilizationFactor = 0.1f;
399 m_hasLandingDir =
false;
402 if (state()->isGroundStart() && m_nonZipperAirtime >= 3) {
406 m_dirDiff = m_landingDir.
proj(m_landingDir);
407 m_hasLandingDir =
true;
409 if (state()->isHop() &&
m_hopPosY > 0.0f) {
411 }
else if (state()->isTouchingGround()) {
413 inputTop.
dot(m_dir) > 0.0f &&
m_speed > 50.0f &&
414 collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::NotTrickable)) {
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));
435 if (bodyDotFront < -0.1f) {
436 stabilizationFactor += std::min(0.2f, EGG::Mathf::abs(bodyDotFront) * 0.5f);
439 if (collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::BoostRamp)) {
440 stabilizationFactor = 0.4f;
447 dynamics()->setStabilizationFactor(stabilizationFactor);
449 m_nonZipperAirtime = state()->isOverZipper() ? 0 : state()->airtime();
450 m_flags.
changeBit(collide()->surfaceFlags().onBit(KartCollide::eSurfaceFlags::Trickable),
457 if (state()->isOverZipper() || !state()->isAirtimeOver20()) {
468 if (
m_up.y <= 0.99f) {
469 m_up += (EGG::Vector3f::ey -
m_up) * 0.03f;
472 m_up = EGG::Vector3f::ey;
480 const auto *raceMgr = System::RaceManager::Instance();
481 if (!raceMgr->isStageReached(System::RaceManager::Stage::Race)) {
485 if (m_padType.
onBit(ePadType::BoostPanel)) {
486 tryStartBoostPanel();
489 if (m_padType.
onBit(ePadType::BoostRamp)) {
493 if (m_padType.
onBit(ePadType::JumpPad)) {
501void KartMove::calcDirs() {
505 m_flags.
setBit(eFlags::LaunchBoost);
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()) {
524 if (dirDiff.
squaredLength() <= std::numeric_limits<f32>::epsilon()) {
535 if (origDirCross.
dot(newDirCross) < 0.0f) {
542 m_flags.
resetBit(eFlags::LaunchBoost);
547 if (!state()->isOverZipper()) {
551 if (m_hasLandingDir) {
552 f32 dot = m_dir.dot(m_landingDir);
554 f32 crossDot = cross.
length();
555 f32 angle = EGG::Mathf::atan2(crossDot, dot);
556 angle = EGG::Mathf::abs(angle);
563 m_landingAngle += (angle * RAD2DEG) * fVar4;
566 if (m_landingAngle <= 0.0f) {
567 if (m_landingAngle < 0.0f) {
568 m_landingAngle = std::min(0.0f, m_landingAngle + 2.0f);
571 m_landingAngle = std::max(0.0f, m_landingAngle - 2.0f);
576void KartMove::calcStickyRoad() {
577 constexpr f32 STICKY_RADIUS = 200.0f;
578 constexpr Field::KCLTypeMask STICKY_MASK =
581 if (state()->isOverZipper()) {
582 state()->setStickyRoad(
false);
586 if ((!state()->isStickyRoad() &&
587 collide()->surfaceFlags().offBit(KartCollide::eSurfaceFlags::Trickable)) ||
588 EGG::Mathf::abs(
m_speed) <= 20.0f) {
596 colInfo.bbox.setZero();
597 Field::KCLTypeMask kcl_flags = KCL_NONE;
598 bool stickyRoad =
false;
600 for (
size_t i = 0; i < 3; ++i) {
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);
610 pos += -STICKY_RADIUS * componentYAxis();
614 state()->setStickyRoad(
false);
622 if (state()->isBoostOffroadInvincibility()) {
626 bool anyWheel = state()->isAnyWheelCollision();
633 if (state()->isVehicleBodyFloorCollision()) {
650void KartMove::calcBoost() {
651 if (m_boost.
calc()) {
652 state()->setAccelerate(
true);
654 state()->setBoost(
false);
661void KartMove::calcRampBoost() {
662 if (!state()->isRampBoost()) {
666 state()->setAccelerate(
true);
667 if (--m_rampBoost < 1) {
669 state()->setRampBoost(
false);
677 if (!state()->isDisableBackwardsAccel()) {
683 state()->setDisableBackwardsAccel(
false);
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;
699 if (state()->isChargingSsmt()) {
720 state()->setDisableBackwardsAccel(
true);
722 if (!state()->isAccelerate() && !state()->isBrake()) {
723 activateBoost(KartBoost::Type::AllMt, SSMT_BOOST_FRAMES);
729 if (state()->isAccelerate() && !state()->isBrake()) {
730 activateBoost(KartBoost::Type::AllMt, SSMT_BOOST_FRAMES);
736 state()->setDisableBackwardsAccel(
true);
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()) {
753 if (state()->isStickRight()) {
755 }
else if (state()->isStickLeft()) {
758 state()->setSlipdriftCharge(
true);
765 if (state()->isHop()) {
767 if (state()->isStickRight()) {
769 }
else if (state()->isStickLeft()) {
776 }
else if (state()->isSlipdriftCharge()) {
780 return state()->isHop() || state()->isSlipdriftCharge();
789 state()->setHop(
false);
790 state()->setDriftManual(
false);
791 m_driftState = DriftState::NotDrifting;
803 m_driftState = DriftState::NotDrifting;
807 state()->setHop(
false);
808 state()->setSlipdriftCharge(
false);
809 state()->setDriftManual(
false);
810 m_autoDriftStartFrameCounter = 0;
814void KartMove::clearJumpPad() {
816 state()->setJumpPad(
false);
820void KartMove::clearRampBoost() {
822 state()->setRampBoost(
false);
826void KartMove::clearZipperBoost() {
827 m_zipperBoostTimer = 0;
828 state()->setZipperBoost(
false);
832void KartMove::clearBoost() {
833 m_boost.resetActive();
834 state()->setBoost(
false);
838void KartMove::clearSsmt() {
846void KartMove::clearOffroadInvincibility() {
848 state()->setBoostOffroadInvincibility(
false);
851void KartMove::clearRejectRoad() {
852 state()->setRejectRoadTrigger(
false);
853 state()->setNoSparkInvisibleWall(
false);
860 constexpr s16 AUTO_DRIFT_START_DELAY = 12;
862 if (!state()->isAutoDrift()) {
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);
871 m_autoDriftStartFrameCounter = 0;
874 if (m_autoDriftStartFrameCounter >= AUTO_DRIFT_START_DELAY) {
875 state()->setDriftAuto(
true);
877 if (state()->isTouchingGround()) {
878 if (state()->stickX() < 0.0f) {
888 f32 halfTarget = 0.5f * param()->stats().driftOutsideTargetAngle;
889 m_autoDriftAngle = std::min(halfTarget, std::max(-halfTarget, m_autoDriftAngle));
891 state()->setDriftAuto(
false);
894 if (m_autoDriftAngle > 0.0f) {
896 std::max(0.0f, m_autoDriftAngle - param()->stats().driftOutsideDecrement);
899 std::min(0.0f, m_autoDriftAngle + param()->stats().driftOutsideDecrement);
905 physics()->composeExtraRot(angleAxis);
914 if (!state()->isOverZipper()) {
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)) {
925 f32 rejCrossDirMag = driftRej.cross(rotZ).
length();
926 f32 angle = EGG::Mathf::atan2(rejCrossDirMag, driftRej.
dot(rotZ));
928 if ((rotZ.z * (rotZ.x - driftRej.x)) - (rotZ.x * (rotZ.z - driftRej.z)) > 0.0f) {
940 if (((!state()->isHop() ||
m_hopFrame < 3) && !state()->isSlipdriftCharge()) ||
941 (state()->isInAction() || !state()->isTouchingGround())) {
953 if (!state()->isDriftManual()) {
954 if (!isHopping && state()->isTouchingGround()) {
957 if (!action()->flags().onBit(KartAction::eFlags::Rotating) ||
m_speed <= 20.0f) {
958 f32 driftAngleDecr = param()->stats().driftOutsideDecrement;
967 if (!state()->isOverZipper() &&
968 (!state()->isDriftInput() || !state()->isAccelerate() || state()->isInAction() ||
969 state()->isRejectRoadTrigger() || state()->isWall3Collision() ||
970 state()->isWallCollision() || !canStartDrift())) {
971 if (canStartDrift()) {
987 constexpr f32 OUTSIDE_DRIFT_BONUS = 0.5f;
989 const auto &stats = param()->stats();
991 if (stats.driftType != KartParam::Stats::DriftType::Inside_Drift_Bike) {
992 f32 driftAngle = 0.0f;
994 if (state()->isHop()) {
1009 state()->setHop(
false);
1010 state()->setSlipdriftCharge(
false);
1012 if (!state()->isDriftInput()) {
1020 state()->setDriftManual(
true);
1021 state()->setHop(
false);
1022 m_driftState = DriftState::ChargingMt;
1030 constexpr f32 SMT_LENGTH_FACTOR = 3.0f;
1032 if (m_driftState < DriftState::ChargedMt || state()->isBrake()) {
1033 m_driftState = DriftState::NotDrifting;
1039 if (m_driftState == DriftState::ChargedSmt) {
1040 mtLength *= SMT_LENGTH_FACTOR;
1043 if (!state()->isBeforeRespawn() && !state()->isInAction()) {
1044 activateBoost(KartBoost::Type::AllMt, mtLength);
1047 m_driftState = DriftState::NotDrifting;
1054 if (state()->airtime() > 5) {
1058 if (param()->stats().driftType != KartParam::Stats::DriftType::Inside_Drift_Bike) {
1061 f32 targetAngle = param()->stats().driftOutsideTargetAngle;
1062 if (angle > targetAngle) {
1064 }
else if (angle < targetAngle) {
1070 f32 targetAngle = -param()->stats().driftOutsideTargetAngle;
1071 if (targetAngle > angle) {
1073 }
else if (targetAngle < angle) {
1088 bool drifting = state()->isDrifting();
1089 bool autoDrift = state()->isAutoDrift();
1090 const auto &stats = param()->stats();
1093 turn = autoDrift ? stats.driftAutomaticTightness : stats.driftManualTightness;
1095 turn = autoDrift ? stats.handlingAutomaticTightness : stats.handlingManualTightness;
1098 if (drifting && stats.driftType != KartParam::Stats::DriftType::Inside_Drift_Bike) {
1103 bool forwards =
true;
1104 if (state()->isBrake() &&
m_speed <= 0.0f) {
1109 if (state()->isChargingSsmt()) {
1112 if (state()->isHop() &&
m_hopPosY > 0.0f) {
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)) {
1125 if (forwards && !noTurn) {
1129 turn += (1.0f - (
m_speed - 20.0f) / 50.0f) * turn;
1132 turn = (turn * 0.4f) + (
m_speed / 20.0f) * (turn * 0.6f);
1141 if (state()->isZipperBoost() && !state()->isDriftManual()) {
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;
1153 if (!state()->isInAction() && !state()->isZipperTrick()) {
1154 if (!state()->isTouchingGround()) {
1155 if (state()->isRampBoost() && m_jump->isBoostRampEnabled()) {
1158 u32 airtime = state()->airtime();
1159 if (airtime >= 70) {
1161 }
else if (airtime >= 30) {
1162 turn = std::max(0.0f, turn * (1.0f - (airtime - 30) * 0.025f));
1168 f32 angle = EGG::Mathf::atan2(forward.cross(m_dir).
length(), forward.
dot(m_dir));
1169 angle = EGG::Mathf::abs(angle) * RAD2DEG;
1171 if (angle > 60.0f) {
1172 turn *= std::max(0.0f, 1.0f - (angle - 60.0f) / 40.0f);
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())) {
1201 if (state()->isInAction()) {
1202 action()->calcVehicleSpeed();
1206 if ((state()->isSomethingWallCollision() && state()->isTouchingGround() &&
1207 !state()->isAnyWheelCollision()) ||
1208 !state()->isTouchingGround() || state()->isChargingSsmt()) {
1209 if (state()->isRampBoost() && state()->airtime() < 4) {
1212 if (state()->isJumpPad() && !state()->isAccelerate()) {
1215 if (state()->isOverZipper()) {
1218 if (state()->airtime() > 5) {
1226 }
else if (state()->isBoost()) {
1229 if (!state()->isJumpPad() && !state()->isRampBoost()) {
1230 if (state()->isAccelerate()) {
1233 if (!state()->isBrake() || state()->isDisableBackwardsAccel() ||
1234 state()->isSomethingWallCollision()) {
1247 if (!state()->isBoost() && !state()->isDriftManual() && !state()->isAutoDrift()) {
1248 const auto &stats = param()->stats();
1251 m_speed *= stats.turningSpeed + (1.0f - stats.turningSpeed) * x;
1265 initialVel = std::min(initialVel * 2.0f, 2.0f);
1267 vel *= std::min(0.5f, std::max(-0.5f, -
bodyFront().y));
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;
1287 as = param()->stats().accelerationStandardA;
1288 ts = param()->stats().accelerationStandardT;
1292 f32 acceleration = 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);
1303 return i < ts.size() ? acceleration : as.back();
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;
1318 dynamics()->setKillExtVelY(state()->isRespawnKillY());
1320 if (state()->isBurnout()) {
1336 if (state()->isBeforeRespawn()) {
1339 if (state()->isChargingSsmt()) {
1350 f32 dVar17 = state()->isJumpPad() ? m_jumpPadMaxSpeed :
m_baseSpeed;
1354 if (state()->isRampBoost()) {
1355 dVar17 = std::max(dVar17, 100.0f);
1360 f32 local_c8 = 1.0f;
1363 if (!state()->isWallCollision() && !state()->isWall3Collision()) {
1373 if (state()->isJumpPad()) {
1384 crossVec = -crossVec;
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;
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)) {
1408 f32 maxSpeedY = state()->isOverZipper() ? KartHalfPipe::TerminalVelocity() : TERMINAL_VELOCITY;
1409 nextSpeed.y = std::min(nextSpeed.y, maxSpeedY);
1411 dynamics()->setIntVel(dynamics()->intVel() + nextSpeed);
1413 if (state()->isTouchingGround() && !state()->isDriftManual() && !state()->isHop()) {
1414 if (state()->isBrake()) {
1417 DrivingDirection::Backwards;
1433 if (!state()->isWallCollision() && !state()->isWall3Collision()) {
1439 if (state()->isZipperInvisibleWall() || state()->isOverZipper()) {
1444 if (wallNrm.y > 0.0f) {
1452 f1 = std::max(0.0f, dot + 1.0f);
1454 return std::min(1.0f, f1 * (state()->isWallCollision() ? 0.4f : 0.7f));
1466 if (!state()->isWallCollisionStart()) {
1471 if (!state()->isInAction()) {
1474 m_landingDir = m_dir;
1477 if (!state()->isOverZipper() && param_2 < 0.9f) {
1480 if (speedDiff > 30.0f) {
1484 f32 dot = -
bodyUp().
dot(colData.relPos) * 0.5f;
1488 speedDiff = std::min(60.0f, speedDiff);
1491 auto [proj, rej] = scaledWallNrm.projAndRej(m_vel1Dir);
1495 if (state()->isBoost()) {
1496 proj = EGG::Vector3f::zero;
1497 rej = EGG::Vector3f::zero;
1500 if (
bodyFront().dot(colData.wallNrm) > 0.0f) {
1501 proj = EGG::Vector3f::zero;
1506 f32 bumpDeviation = 0.0f;
1508 bumpDeviation = param()->stats().bumpDeviationLevel;
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()) {
1529 scalar = std::min(3.0f, std::max(speedDiff, -3.0f));
1531 if (state()->isMushroomBoost()) {
1532 next = (scalar * 0.15f) * 0.25f;
1533 if (state()->isWheelie()) {
1537 next = (scalar * 0.15f) * 0.08f;
1542 constexpr s16 MAX_SSMT_CHARGE = 75;
1543 next = 0.015f * (-
static_cast<f32
>(
m_ssmtCharge) /
static_cast<f32
>(MAX_SSMT_CHARGE));
1548 m_standStillBoostRot = isBike() ? next * 3.0f : next * 10.0f;
1550 m_standStillBoostRot += scalar * (next - m_standStillBoostRot);
1558 constexpr f32 DIVE_LIMIT = 0.8f;
1562 if (state()->isTouchingGround() || state()->isCannonStart() || state()->isInCannon() ||
1563 state()->isInAction() || state()->isOverZipper()) {
1567 f32 stickY = state()->stickY();
1570 stickY = std::min(1.0f, stickY + 0.4f);
1573 u32 airtime = state()->airtime();
1576 if (EGG::Mathf::abs(stickY) < 0.1f) {
1580 stickY *= (airtime / 50.0f);
1587 dynamics()->setAngVel2(angVel2);
1589 if (state()->airtime() < 50) {
1595 f32 upDotTop =
m_up.
dot(topRotated);
1597 f32 crossNorm = upCrossTop.
length();
1598 f32 angle = EGG::Mathf::abs(EGG::Mathf::atan2(crossNorm, upDotTop));
1600 f32 fVar1 = angle * RAD2DEG - 20.0f;
1601 if (fVar1 <= 0.0f) {
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());
1609 dynamics()->setGravity((0.2f * mult + 1.0f) * dynamics()->gravity());
1617 if (EGG::Mathf::abs(
m_speed) >= 10.0f || state()->isBoost() || state()->isRampBoost() ||
1618 !state()->isAccelerate() || !state()->isBrake()) {
1619 state()->setChargingSsmt(
false);
1623 state()->setChargingSsmt(
true);
1624 state()->setHopStart(
false);
1625 state()->setDriftInput(
false);
1629void KartMove::calcHopPhysics() {
1640void KartMove::calcRejectRoad() {
1641 m_reject.calcRejectRoad();
1645bool KartMove::calcZipperCollision(f32 radius, f32 scale,
EGG::Vector3f &pos,
1647 Field::KCLTypeMask *maskOut, Field::KCLTypeMask flags)
const {
1649 pos = dynamics()->pos() + (-scale *
m_scale.y) * upLocal;
1651 auto *colDir = Field::CollisionDirector::Instance();
1652 return colDir->checkSphereFullPush(radius, pos, prevPos, flags, colInfo, maskOut, 0);
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;
1665 f32 tiltMagnitude = 0.0f;
1667 if (!state()->isInAction() && !state()->isSoftWallDrift() && state()->isAnyWheelCollision()) {
1671 f32 magnitude = tiltMagnitude;
1673 if (frontSpeed.
squaredLength() > std::numeric_limits<f32>::epsilon()) {
1674 magnitude = frontSpeed.
length();
1676 if (front.z * frontSpeed.x - front.x * frontSpeed.z > 0.0f) {
1677 magnitude = -magnitude;
1680 tiltMagnitude = -1.0f;
1681 if (-1.0f <= magnitude) {
1682 tiltMagnitude = std::min(1.0f, magnitude);
1685 }
else if (!state()->isHop() ||
m_hopPosY <= 0.0f) {
1688 dynamics()->setAngVel0(angVel0);
1691 f32 lean = EGG::Mathf::abs(
m_weightedTurn) * (tiltMagnitude * param()->stats().tilt);
1696 angVel0.x += m_standStillBoostRot;
1698 dynamics()->setAngVel0(angVel0);
1702 dynamics()->setAngVel2(angVel2);
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;
1719 if (m_driftState == DriftState::ChargedSmt) {
1723 f32 stickX = state()->stickX();
1725 if (m_driftState == DriftState::ChargingMt) {
1728 if (-BONUS_CHARGE_STICK_THRESHOLD <= stickX) {
1729 if (BONUS_CHARGE_STICK_THRESHOLD < stickX &&
m_hopStickX == -1) {
1738 m_driftState = DriftState::ChargingSmt;
1742 if (m_driftState != DriftState::ChargingSmt) {
1748 if (-BONUS_CHARGE_STICK_THRESHOLD <= stickX) {
1749 if (BONUS_CHARGE_STICK_THRESHOLD < stickX &&
m_hopStickX == -1) {
1758 m_driftState = DriftState::ChargedSmt;
1766 state()->setHop(
true);
1767 state()->setDriftManual(
false);
1772 m_driftState = DriftState::NotDrifting;
1784 dynamics()->setExtVel(extVel);
1787 totalForce.y = 0.0f;
1788 dynamics()->setTotalForce(totalForce);
1792void KartMove::tryStartBoostPanel() {
1793 constexpr s16 BOOST_PANEL_DURATION = 60;
1795 if (state()->isBeforeRespawn()) {
1799 activateBoost(KartBoost::Type::MushroomAndBoostPanel, BOOST_PANEL_DURATION);
1807 constexpr s16 BOOST_RAMP_DURATION = 60;
1809 if (state()->isBeforeRespawn()) {
1813 state()->setRampBoost(
true);
1814 m_rampBoost = BOOST_RAMP_DURATION;
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},
1833 if (state()->isBeforeRespawn() || state()->isInAction() || state()->isHalfPipeRamp()) {
1837 state()->setJumpPad(
true);
1838 s32 jumpPadVariant = state()->jumpPadVariant();
1839 m_jumpPadProperties = &JUMP_PAD_PROPERTIES[jumpPadVariant];
1841 if (jumpPadVariant != 4) {
1845 extVel.y = m_jumpPadProperties->velY;
1846 totalForce.y = 0.0f;
1848 dynamics()->setExtVel(extVel);
1849 dynamics()->setTotalForce(totalForce);
1851 if (jumpPadVariant != 3) {
1858 state()->setJumpPadDisableYsusForce(
true);
1863 m_jumpPadMaxSpeed = m_jumpPadProperties->maxSpeed;
1868void KartMove::tryEndJumpPad() {
1869 if (state()->isGroundStart()) {
1875void KartMove::cancelJumpPad() {
1877 state()->setJumpPad(
false);
1881void KartMove::activateBoost(KartBoost::Type type, s16 frames) {
1882 if (m_boost.
activate(type, frames)) {
1883 state()->setBoost(
true);
1888void KartMove::applyStartBoost(s16 frames) {
1889 activateBoost(KartBoost::Type::AllMt, frames);
1893void KartMove::activateMushroom() {
1894 constexpr s16 MUSHROOM_DURATION = 90;
1896 if (state()->isBeforeRespawn() || state()->isInAction()) {
1900 activateBoost(KartBoost::Type::MushroomAndBoostPanel, MUSHROOM_DURATION);
1903 state()->setMushroomBoost(
true);
1908void KartMove::activateZipperBoost() {
1909 constexpr s16 BASE_DURATION = 50;
1910 constexpr s16 TRICK_DURATION = 100;
1912 if (state()->isBeforeRespawn() || state()->isInAction()) {
1916 s16 boostDuration = state()->isZipperTrick() ? TRICK_DURATION : BASE_DURATION;
1917 activateBoost(KartBoost::Type::TrickAndZipper, boostDuration);
1920 m_zipperBoostTimer = 0;
1921 m_zipperBoostMax = boostDuration;
1922 state()->setZipperBoost(
true);
1934 state()->setBoostOffroadInvincibility(
true);
1941 if (!state()->isBoostOffroadInvincibility()) {
1949 state()->setBoostOffroadInvincibility(
false);
1955 if (!state()->isMushroomBoost()) {
1963 state()->setMushroomBoost(
false);
1967void KartMove::calcZipperBoost() {
1968 if (!state()->isZipperBoost()) {
1972 state()->setAccelerate(
true);
1974 if (!state()->isOverZipper() && ++m_zipperBoostTimer >= m_zipperBoostMax) {
1975 m_zipperBoostTimer = 0;
1976 state()->setZipperBoost(
false);
1979 if (m_zipperBoostTimer < 10) {
1982 dynamics()->setAngVel0(angVel);
1987void KartMove::landTrick() {
1988 static constexpr std::array<s16, 3> KART_TRICK_BOOST_DURATION = {{
1993 static constexpr std::array<s16, 3> BIKE_TRICK_BOOST_DURATION = {{
1999 if (state()->isBeforeRespawn() || state()->isInAction()) {
2005 duration = BIKE_TRICK_BOOST_DURATION[
static_cast<u32
>(m_jump->variant())];
2007 duration = KART_TRICK_BOOST_DURATION[
static_cast<u32
>(m_jump->variant())];
2010 activateBoost(KartBoost::Type::TrickAndZipper, duration);
2014void KartMove::enterCannon() {
2016 physics()->clearDecayingRot();
2017 m_boost.resetActive();
2018 state()->setBoost(
false);
2024 clearOffroadInvincibility();
2026 dynamics()->reset();
2029 state()->setHop(
false);
2030 state()->setInCannon(
true);
2031 state()->setSkipWheelCalc(
true);
2032 state()->setCannonStart(
false);
2034 const auto [cannonPos, cannonDir] = getCannonPosRot();
2035 m_cannonEntryPos = pos();
2036 m_cannonEntryOfs = cannonPos - pos();
2037 m_cannonEntryOfsLength = 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();
2046void KartMove::calcCannon() {
2047 auto [cannonPos, cannonDir] = getCannonPosRot();
2048 EGG::Vector3f forwardXZ = cannonPos - m_cannonEntryPos - m_cannonProgress;
2050 f32 forwardLength = forward.
normalise();
2064 if (forwardLength < 30.0f || local94.
dot(forwardXZ) <= 0.0f) {
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);
2078 newSpeed = cannonParams.endDecel;
2079 if (newSpeed <= 0.0f) {
2083 newSpeed += factor * (cannonParams.speed - newSpeed);
2084 if (cannonParams.endDecel > 0.0f) {
2089 m_cannonProgress += m_cannonEntryOfs * newSpeed;
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;
2098 dynamics()->setPos(m_cannonEntryPos + m_cannonProgress + newPos);
2099 m_dir = m_cannonEntryOfs;
2100 m_vel1Dir = m_cannonEntryOfs;
2102 calcRotCannon(forward);
2104 dynamics()->setExtVel(EGG::Vector3f::zero);
2112 EGG::Vector3f local60 = local54 + ((local48 - local54) * 0.3f);
2118 local80 *= dynamics()->fullRot();
2123 dynamics()->setFullRot(newRot);
2124 dynamics()->setMainRot(newRot);
2128void KartMove::exitCannon() {
2129 if (!state()->isInCannon()) {
2133 state()->setInCannon(
false);
2134 state()->setSkipWheelCalc(
false);
2135 state()->setAfterCannon(
true);
2136 dynamics()->setIntVel(m_cannonEntryOfs *
m_speed);
2140void KartMove::triggerRespawn() {
2142 state()->setTriggerRespawn(
true);
2146KartMoveBike::KartMoveBike() : m_leanRot(0.0f) {}
2149KartMoveBike::~KartMoveBike() =
default;
2154 constexpr f32 MAX_WHEELIE_ROTATION = 0.07f;
2155 constexpr u16 WHEELIE_COOLDOWN = 20;
2157 state()->setWheelie(
true);
2162 m_autoHardStickXFrames = 0;
2169 state()->setWheelie(
false);
2171 m_autoHardStickXFrames = 0;
2175void KartMoveBike::createSubsystems() {
2186 const auto *raceManager = System::RaceManager::Instance();
2188 if (!state()->isChargingSsmt()) {
2189 if (!raceManager->isStageReached(System::RaceManager::Stage::Race) ||
2190 EGG::Mathf::abs(
m_speed) < 5.0f) {
2202 f32 stickX = state()->stickX();
2203 f32 extVelXFactor = 0.0f;
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()) {
2213 }
else if (!state()->isDrifting()) {
2214 if (stickX <= 0.2f) {
2215 if (stickX >= -0.2f) {
2230 leanRotMin = -leanRotMax;
2234 if (stickX == 0.0f) {
2240 }
else if (stickX == 0.0f) {
2248 bool capped =
false;
2260 dynamics()->setExtVel(dynamics()->extVel() + componentXAxis() * extVelXFactor);
2263 f32 leanRotScalar = state()->isDrifting() ? 0.065f : 0.05f;
2267 dynamics()->setAngVel2(dynamics()->angVel2() +
2268 EGG::Vector3f(m_standStillBoostRot, turn * wheelieRotFactor(),
2275 if (!state()->isRejectRoad() && !state()->isHalfPipeRamp() && !state()->isOverZipper()) {
2277 scalar = std::min(1.0f, scalar);
2278 top = scalar *
m_up + (1.0f - scalar) * EGG::Vector3f::ey;
2280 if (std::numeric_limits<f32>::epsilon() < top.
squaredLength()) {
2285 dynamics()->setTop_(top);
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},
2298 KartMove::setTurnParams();
2300 if (param()->stats().driftType == KartParam::Stats::DriftType::Outside_Drift_Bike) {
2302 }
else if (param()->stats().driftType == KartParam::Stats::DriftType::Inside_Drift_Bike) {
2306 if (System::RaceManager::Instance()->isStageReached(System::RaceManager::Stage::Race)) {
2316void KartMoveBike::init(
bool b1,
bool b2) {
2317 KartMove::init(b1, b2);
2326 m_autoHardStickXFrames = 0;
2330void KartMoveBike::clear() {
2338 constexpr u32 FAILED_WHEELIE_FRAMES = 15;
2339 constexpr f32 AUTO_WHEELIE_CANCEL_STICK_THRESHOLD = 0.85f;
2344 if (state()->isWheelie()) {
2345 bool cancelAutoWheelie =
false;
2347 if (!state()->isAutoDrift() ||
2348 EGG::Mathf::abs(state()->stickX()) <= AUTO_WHEELIE_CANCEL_STICK_THRESHOLD) {
2349 m_autoHardStickXFrames = 0;
2351 if (++m_autoHardStickXFrames > 15) {
2352 cancelAutoWheelie =
true;
2364 dynamics()->setAngVel0(angVel0);
2374 f32 vel1DirUp = m_vel1Dir.dot(EGG::Vector3f::ey);
2379 angVel2.x -=
m_wheelieRot * (1.0f - EGG::Mathf::abs(vel1DirUp));
2380 dynamics()->setAngVel2(angVel2);
2385 state()->setWheelieRot(
true);
2387 state()->setWheelieRot(
false);
2397 if (state()->isAutoDrift()) {
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;
2419 if (m_driftState != DriftState::ChargingMt) {
2425 f32 stickX = state()->stickX();
2426 if (-BONUS_CHARGE_STICK_THRESHOLD <= stickX) {
2427 if (BONUS_CHARGE_STICK_THRESHOLD < stickX &&
m_hopStickX == -1) {
2436 m_driftState = DriftState::ChargedMt;
2441void KartMoveBike::initOob() {
2447 clearOffroadInvincibility();
2454 constexpr s16 COOLDOWN_FRAMES = 20;
2455 bool dpadUp = inputs()->currentState().trickUp();
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()) {
@ 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_FLOOR
0x20E80FFF - Any KCL that the player or items can drive/land on.
void makeOrthonormalBasis(const Vector3f &v0, const Vector3f &v1)
Sets a 3x3 orthonormal basis for a local coordinate system.
void setAxisRotation(f32 angle, const Vector3f &axis)
Rotates the matrix about an axis.
Vector3f multVector33(const Vector3f &vec) const
Multiplies a 3x3 matrix by a vector.
Vector3f multVector(const Vector3f &vec) const
Multiplies a vector by a matrix.
bool activate(Type type, s16 frames)
Starts/restarts a boost of the given type.
bool calc()
Computes the current frame's boost multiplier, acceleration, and speed limit.
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.
f32 m_leanRotCap
The maximum leaning rotation.
void calcWheelie() override
STAGE 1+ - Every frame, checks player input for wheelies and computes wheelie rotation.
void calcMtCharge() override
Every frame during a drift, calculates MT charge based on player input.
virtual void startWheelie()
STAGE 1+ - Sets the wheelie bit flag and some wheelie-related variables.
f32 m_wheelieRotDec
The wheelie rotation decrementor, used after a wheelie has ended.
u32 m_wheelieFrames
Tracks wheelie duration and cancels the wheelie after 180 frames.
f32 m_wheelieRot
X-axis rotation from wheeling.
const TurningParameters * m_turningParams
Inside/outside drifting bike turn info.
void setTurnParams() override
On init, sets the bike's lean rotation cap and increment.In addition to setting the lean rotation cap...
f32 m_maxWheelieRot
The maximum wheelie rotation.
void tryStartWheelie()
STAGE 1+ - Every frame, checks player input to see if we should start or stop a wheelie.
void onWallCollision() override
Called when you collide with a wall. All it does for bikes is cancel wheelies.
void onHop() override
Virtual function that just cancels wheelies when you hop.
bool canWheelie() const override
Checks if the kart is going fast enough to wheelie.
void calcVehicleRotation(f32) override
Every frame, calculates rotation, EV, and angular velocity for the bike.
s16 m_wheelieCooldown
The number of frames before another wheelie can start.
f32 m_leanRotInc
The incrementor for leaning rotation.
virtual void cancelWheelie()
Clears the wheelie bit flag and resets the rotation decrement.
f32 m_baseSpeed
The speed associated with the current character/vehicle stats.
void calcRotation()
Every frame, calculates kart rotation based on player input.
s16 m_ssmtLeewayTimer
Frames to forgive letting go of A before clearing SSMT charge.
s32 m_hopFrame
A timer that can prevent subsequent hops until reset.
void calcDisableBackwardsAccel()
Computes the current cooldown duration between braking and reversing.
EGG::Vector3f m_hopUp
The up vector when hopping.
u16 m_mushroomBoostTimer
Number of frames until the mushroom boost runs out.
void calcSpecialFloor()
Every frame, calculates any boost resulting from a boost panel.
void calcWallCollisionStart(f32 param_2)
If we started to collide with a wall this frame, applies rotation.
KartHalfPipe * m_halfPipe
Pertains to zipper physics.
f32 m_kclRotFactor
Float between 0-1 that scales the player's turning radius on offroad.
f32 m_outsideDriftBonus
Added to angular velocity when outside drifting.
void tryStartBoostRamp()
Sets offroad invincibility and and enables the ramp boost bitfield flag.
u16 m_smtCharge
A value between 0 and 300 representing current SMT charge.
f32 m_speedRatio
The ratio between current speed and the player's base speed stat.
@ 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.
f32 m_jumpPadMinSpeed
Snaps the player to a minimum speed when first touching a jump pad.
f32 m_hopPosY
Relative position as the result of a hop. Starts at 0.
DrivingDirection m_drivingDirection
Current state of driver's direction.
bool calcPreDrift()
Each frame, checks for hop or slipdrift. Computes drift direction based on player input.
f32 m_speed
Current speed, restricted to the soft speed limit.
s16 m_offroadInvincibility
How many frames until the player is affected by offroad.
void calcAirtimeTop()
Calculates rotation of the bike due to excessive airtime.
void startManualDrift()
Called when the player lands from a drift hop, or to start a slipdrift.
void controlOutsideDriftAngle()
Every frame, handles mini-turbo charging and outside drifting bike rotation.
f32 m_softSpeedLimit
Base speed + boosts + wheelies, restricted to the hard speed limit.
virtual void hop()
Initializes hop information, resets upwards EV and clears upwards force.
void calcStandstillBoostRot()
STAGE Computes the x-component of angular velocity based on the kart's speed.
EGG::Vector3f m_outsideDriftLastDir
Used to compute the next m_outsideDriftAngle.
void calcManualDrift()
Each frame, handles hopping, drifting, and mini-turbos.
virtual void calcMtCharge()
Every frame during a drift, calculates MT/SMT charge based on player input.
void calcSsmt()
Calculates standstill mini-turbo components, if applicable.
void calcAcceleration()
Every frame, applies acceleration to the kart's internal velocity.
f32 m_processedSpeed
Offset 0x28. It's only ever just a copy of m_speed.
void releaseMt()
Stops charging a mini-turbo, and applies boost if charged.
f32 m_kclSpeedFactor
Float between 0-1 that scales the player's speed on offroad.
f32 m_weightedTurn
Magnitude+direction of stick input, factoring in the kart's stats.
void calcVehicleSpeed()
Every frame, computes speed based on acceleration and any active boosts.
f32 m_lastSpeed
Last frame's speed, cached to calculate angular velocity.
s16 m_ssmtDisableAccelTimer
Counter that tracks delay before starting to reverse.
void calcOffroadInvincibility()
Checks a timer to see if we are still ignoring offroad slowdown.
KartBurnout m_burnout
Manages the state of start boost burnout.
f32 calcVehicleAcceleration() const
Every frame, computes acceleration based off the character/vehicle stats.
void calcAutoDrift()
Each frame, handles automatic transmission drifting.
f32 m_realTurn
The "true" turn magnitude. Equal to m_weightedTurn unless drifting.
const DriftingParameters * m_driftingParams
Drift-type-specific parameters.
void calc()
Each frame, calculates the kart's movement.
EGG::Vector3f m_up
Vector perpendicular to the floor, pointing upwards.
s16 m_ssmtCharge
Increments every frame up to 75 when charging stand-still MT.
f32 m_speedDragMultiplier
After 5 frames of airtime, this causes speed to slowly decay.
u16 m_mtCharge
A value between 0 and 270 representing current MT charge.
void calcSsmtStart()
Calculates whether we are starting a standstill mini-turbo.
s16 m_respawnPostLandTimer
Counts up to 4 if not accelerating after respawn landing.
virtual f32 getWheelieSoftSpeedLimitBonus() const
Returns the % speed boost from wheelies. For karts, this is always 0.
f32 m_kclWheelRotFactor
The slowest rotation multiplier of each wheel's floor collision.
void resetDriftManual()
Clears drift state. Called when touching ground and drift is canceled.
f32 m_totalScale
[Unused] Always 1.0f
f32 m_acceleration
Captures the acceleration from player input and boosts.
virtual void calcVehicleRotation(f32 turn)
Every frame, calculates rotation, EV, and angular velocity for the kart.
s16 m_respawnPreLandTimer
Counts down from 4 when pressing A before landing from respawn.
virtual void calcTurn()
Each frame, looks at player input and kart stats. Saves turn-related info.
f32 m_divingRot
Induces x-axis angular velocity based on up/down stick input.
f32 m_outsideDriftAngle
The facing angle of an outward-drifting vehicle.
EGG::Vector3f m_lastDir
m_speed from the previous frame but with signed magnitude.
EGG::Vector3f m_smoothedUp
A smoothed up vector, mostly used after significant airtime.
s32 getAppliedHopStickX() const
Factors in vehicle speed to retrieve our hop direction and magnitude.
u16 m_floorCollisionCount
The number of tires colliding with the floor.
void calcDive()
Responds to player input to handle up/down kart tilt mid-air.
void calcOffroad()
Each frame, computes rotation and speed scalars from the floor KCL.
@ WaitingForBackwards
Holding reverse but waiting on a 15 frame delay.
s16 m_timeInRespawn
The number of frames elapsed after position snap from respawn.
s32 m_hopStickX
A ternary for the direction of our hop, 0 if still neutral hopping.
s16 m_backwardsAllowCounter
Tracks the 15f delay before reversing.
f32 calcWallCollisionSpeedFactor(f32 &f1)
Every frame, computes a speed scalar if we are colliding with a wall.
void calcMushroomBoost()
Checks a timer to see if we are still boosting from a mushroom.
f32 m_hopGravity
Always main gravity (-1.3f).
f32 m_hopVelY
Relative velocity due to a hop. Starts at 10 and decreases with gravity.
f32 m_hardSpeedLimit
Absolute speed cap. It's 120, unless you're in a bullet (140).
void setInitialPhysicsValues(const EGG::Vector3f &position, const EGG::Vector3f &angles)
Initializes the kart's position and rotation. Calls tire suspension initializers.
f32 m_rawTurn
Float in range [-1, 1]. Represents stick magnitude + direction.
void setOffroadInvincibility(s16 timer)
Ignores offroad KCL collision for a set amount of time.
f32 m_speedRatioCapped
m_speedRatio but capped at 1.0f.
EGG::Vector3f m_scale
[Unused] Always 1.0f
f32 m_kclWheelSpeedFactor
The slowest speed multiplier of each wheel's floor collision.
EGG::Vector3f m_hopDir
Used for outward drift. Tracks the forward vector of our rotation.
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.
Pertains to kart-related functionality.
@ BikeSideStuntTrick
A side StuntTrick with a bike.
A quaternion, used to represent 3D rotation.
void normalise()
Scales the quaternion to a unit length.
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
Quatf slerpTo(const Quatf &q2, f32 t) const
Performs spherical linear interpolation.
void setAxisRotation(f32 angle, const Vector3f &axis)
Set the quat given angle and axis.
f32 dot(const Quatf &q) const
Computes .
void setRPY(const Vector3f &rpy)
Sets roll, pitch, and yaw.
void makeVectorRotation(const Vector3f &from, const Vector3f &to)
Captures rotation between two vectors.
constexpr TBitFlag< T, E > & resetBit(Es... es)
Resets the corresponding bits for the provided enum values.
constexpr bool onBit(Es... es) const
Checks if any of the corresponding bits for the provided enum values are on.
constexpr TBitFlag< T, E > & changeBit(bool on, Es... es)
Changes the state of the corresponding bits for the provided enum values.
constexpr bool offBit(Es... es) const
Checks if all of the corresponding bits for the provided enum values are off.
constexpr void makeAllZero()
Resets all the bits to zero.
constexpr TBitFlag< T, E > & setBit(Es... es)
Sets the corresponding bits for the provided enum values.
f32 normalise()
Normalizes the vector and returns the original length.
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
f32 length() const
The square root of the vector's dot product.
f32 squaredLength() const
The dot product between the vector and itself.
Vector3f proj(const Vector3f &rhs) const
The projection of this vector onto rhs.
Vector3f perpInPlane(const EGG::Vector3f &rhs, bool normalise) const
Calculates the orthogonal vector, based on the plane defined by this vector and rhs.
Vector3f rej(const Vector3f &rhs) const
The rejection of this vector onto rhs.
Information about the current collision and its properties.
f32 driftManualTightness
Affects turn radius when manual drifting.
std::array< f32, 32 > kclRot
Rotation scalars, indexed using KCL attributes.
f32 driftAutomaticTightness
Affects turn radius when automatic drifting.
f32 driftReactivity
A weight applied to turn radius when drifting.
f32 speed
Base full speed of the character/vehicle combo.
f32 handlingReactivity
A weight applied to turn radius when not drifting.
u32 miniTurbo
The framecount duration of a charged mini-turbo.