1#include "KartCollide.hh"
3#include "game/kart/KartBody.hh"
4#include "game/kart/KartMove.hh"
5#include "game/kart/KartPhysics.hh"
6#include "game/kart/KartState.hh"
8#include "game/field/CollisionDirector.hh"
9#include "game/field/ObjectCollisionKart.hh"
10#include "game/field/ObjectDirector.hh"
12#include <egg/math/BoundBox.hh>
13#include <egg/math/Math.hh>
18KartCollide::KartCollide() {
19 m_boundingRadius = 100.0f;
24KartCollide::~KartCollide() =
default;
27void KartCollide::init() {
29 m_floorMomentRate = 0.8f;
33 m_smoothedBack = 0.0f;
34 m_suspBottomHeightNonSoftWall = 0.0f;
35 m_suspBottomHeightSoftWall = 0.0f;
36 m_someNonSoftWallTimer = 0;
37 m_someSoftWallTimer = 0;
38 m_poleAngVelTimer = 0;
40 m_colPerpendicularity = 0.0f;
44void KartCollide::resetHitboxes() {
45 CollisionGroup *hitboxGroup = physics()->hitboxGroup();
46 for (
u16 idx = 0; idx < hitboxGroup->hitboxCount(); ++idx) {
47 hitboxGroup->hitbox(idx).setLastPos(scale(), pose());
56 for (
u16 idx = 0; idx < hitboxGroup->hitboxCount(); ++idx) {
57 hitboxGroup->hitbox(idx).
calc(move()->totalScale(), body()->sinkDepth(), scale(), fullRot(),
65 const EGG::Quatf &rot = state()->isEndHalfPipe() ? mainRot() : fullRot();
68 auto &colData = collisionData();
69 bool existingWallCollision = colData.bWall || colData.bWall3;
70 bool newWallCollision =
71 m_surfaceFlags.
onBit(eSurfaceFlags::ObjectWall, eSurfaceFlags::ObjectWall3);
72 if (existingWallCollision || newWallCollision) {
73 if (!existingWallCollision) {
74 colData.wallNrm = m_totalReactionWallNrm;
75 if (m_surfaceFlags.
onBit(eSurfaceFlags::ObjectWall)) {
77 }
else if (m_surfaceFlags.
onBit(eSurfaceFlags::ObjectWall3)) {
78 colData.bWall3 =
true;
80 }
else if (newWallCollision) {
81 colData.wallNrm += m_totalReactionWallNrm;
82 if (m_surfaceFlags.
onBit(eSurfaceFlags::ObjectWall)) {
84 }
else if (m_surfaceFlags.
onBit(eSurfaceFlags::ObjectWall3)) {
85 colData.bWall3 =
true;
89 colData.wallNrm.normalise();
101 if (isInRespawn() || state()->isBoost() || state()->isOverZipper() ||
102 state()->isZipperInvisibleWall() || state()->isNoSparkInvisibleWall() ||
103 state()->isHalfPipeRamp()) {
109 bool resetXZ = fVar1 > 0.0f && state()->isAirtimeOver20() && dynamics()->velocity().y < -50.0f;
111 FUN_805B72B8(state()->isInAction() ? 0.3f : 0.01f, fVar1, resetXZ,
112 !state()->isJumpPadDisableYsusForce());
123 const auto &colData = collisionData();
125 if (!colData.bFloor && !colData.bWall && !colData.bWall3) {
129 EGG::Vector3f collisionDir = colData.floorNrm + colData.wallNrm;
132 f32 directionalVelocity = colData.vel.dot(collisionDir);
133 if (directionalVelocity >= 0.0f) {
140 rotMat.
makeQ(dynamics()->mainRot());
153 f32 val = (-directionalVelocity * (param_2 + 1.0f)) / (1.0f + collisionDir.
dot(step3));
158 f32 fVar1 = param_1 * EGG::Mathf::abs(val);
159 f32 otherVal = (val * colData.vel.dot(step5)) / directionalVelocity;
161 f32 fVar3 = otherVal;
162 if (fVar1 < EGG::Mathf::abs(otherVal)) {
164 if (otherVal < 0.0f) {
165 fVar3 = -param_1 * EGG::Mathf::abs(val);
171 f32 local_1d0 = step6.y;
174 }
else if (colData.bFloor) {
175 f32 velY = intVel().y;
181 dynamics()->setExtVel(newExtVel);
186 f32 prevExtVelY = extVel().y;
188 extVelAdd.y = local_1d0;
189 dynamics()->setExtVel(extVel() + extVelAdd);
191 if (prevExtVelY < 0.0f && extVel().y > 0.0f && extVel().y < 10.0f) {
194 dynamics()->setExtVel(extVelNoY);
201 dynamics()->setAngVel0(dynamics()->angVel0() + step9);
214 collisionData.reset();
219 colInfo.bbox.setDirect(EGG::Vector3f::zero, EGG::Vector3f::zero);
220 Field::KCLTypeMask maskOut;
226 for (
u16 hitboxIdx = 0; hitboxIdx < hitboxGroup->hitboxCount(); ++hitboxIdx) {
228 Hitbox &hitbox = hitboxGroup->hitbox(hitboxIdx);
230 if (hitbox.bspHitbox()->wallsOnly != 0) {
232 Field::CourseColMgr::Instance()->setNoBounceWallInfo(&noBounceWallInfo);
235 hitbox.
calc(totalScale, sinkDepth, scale, rot, pos());
237 if (Field::CollisionDirector::Instance()->checkSphereCachedFullPush(hitbox.radius(),
238 hitbox.worldPos(), hitbox.lastPos(), flags, &colInfo, &maskOut, 0)) {
244 if (!
FUN_805B6A9C(collisionData, hitbox, minMax, posRel, count, maskOut, colInfo)) {
246 processBody(collisionData, hitbox, &colInfo, &maskOut);
255 collisionData.speedFactor = 1.0f;
256 collisionData.rotFactor = 1.0f;
261void KartCollide::calcFloorEffect() {
262 if (state()->isTouchingGround()) {
266 m_suspBottomHeightNonSoftWall = 0.0f;
267 m_surfaceFlags.
resetBit(eSurfaceFlags::Wall, eSurfaceFlags::SolidOOB, eSurfaceFlags::BoostRamp,
269 eSurfaceFlags::StopHalfPipeState);
270 m_suspBottomHeightSoftWall = 0.0f;
271 m_someNonSoftWallTimer = 0;
272 m_someSoftWallTimer = 0;
274 Field::KCLTypeMask mask = KCL_NONE;
275 calcTriggers(&mask, pos(),
false);
277 if (m_solidOobTimer >= 3 && m_surfaceFlags.
onBit(eSurfaceFlags::SolidOOB) &&
278 m_surfaceFlags.
offBit(eSurfaceFlags::Wall)) {
284 activateOob(
true, &mask,
false,
false);
288 calcTriggers(&mask, pos(),
true);
291 m_surfaceFlags.
onBit(eSurfaceFlags::SolidOOB) ? std::min(3, m_solidOobTimer + 1) : 0;
295void KartCollide::calcTriggers(Field::KCLTypeMask *mask,
const EGG::Vector3f &pos,
bool twoPoint) {
296 EGG::Vector3f v1 = twoPoint ? physics()->pos() : EGG::Vector3f::inf;
298 f32 radius = twoPoint ? 80.0f : 100.0f * move()->totalScale();
299 f32 scalar = -bsp().initialYPos * move()->totalScale() * 0.3f;
303 m_smoothedBack += (back.
dot(move()->smoothedUp()) - m_smoothedBack) * 0.3f;
305 scalar = m_smoothedBack * -physics()->fc() * 1.8f * move()->totalScale();
306 scaledPos += scalar * back;
308 bool collide = Field::CollisionDirector::Instance()->checkSphereCachedPartialPush(radius,
309 scaledPos, v1, typeMask,
nullptr, mask, 0);
316 handleTriggers(mask);
323 m_surfaceFlags.
setBit(eSurfaceFlags::Wall);
327 m_surfaceFlags.
setBit(eSurfaceFlags::SolidOOB);
333void KartCollide::handleTriggers(Field::KCLTypeMask *mask) {
334 calcFallBoundary(mask,
false);
338 auto *colDir = Field::CollisionDirector::Instance();
341 halfPipe()->end(
true);
342 state()->setEndHalfPipe(
true);
343 m_surfaceFlags.
setBit(eSurfaceFlags::StopHalfPipeState);
350void KartCollide::calcFallBoundary(Field::KCLTypeMask *mask,
bool ) {
355 auto *colDir = Field::CollisionDirector::Instance();
360 activateOob(
false, mask,
false,
false);
364void KartCollide::calcBeforeRespawn() {
365 if (pos().y < 0.0f) {
366 activateOob(
true,
nullptr,
false,
false);
369 if (!state()->isBeforeRespawn()) {
373 if (--m_respawnTimer > 0) {
377 state()->setBeforeRespawn(
false);
379 move()->triggerRespawn();
383void KartCollide::activateOob(
bool , Field::KCLTypeMask * ,
385 constexpr s16 RESPAWN_TIME = 130;
387 if (state()->isBeforeRespawn()) {
393 m_respawnTimer = RESPAWN_TIME;
394 state()->setBeforeRespawn(
true);
407 Hitbox &firstHitbox = hitboxGroup->hitbox(0);
409 bspHitbox->radius = radius;
410 hitboxGroup->resetCollision();
411 firstHitbox.setWorldPos(center);
414 colInfo.bbox.setZero();
415 Field::KCLTypeMask kclOut;
417 Field::CourseColMgr::Instance()->setNoBounceWallInfo(&noBounceWallInfo);
419 bool collided = Field::CollisionDirector::Instance()->checkSphereCachedFullPush(
420 firstHitbox.radius(), firstHitbox.worldPos(), firstHitbox.lastPos(),
426 collisionData.speedFactor = 1.0f;
427 collisionData.rotFactor = 1.0f;
431 collisionData.tangentOff = colInfo.tangentOff;
433 if (noBounceWallInfo.dist > std::numeric_limits<f32>::min()) {
434 collisionData.tangentOff += noBounceWallInfo.tangentOff;
435 collisionData.noBounceWallNrm = noBounceWallInfo.fnrm;
436 collisionData.bSoftWall =
true;
440 collisionData.
bFloor =
true;
441 collisionData.floorNrm = colInfo.floorNrm;
444 collisionData.relPos = firstHitbox.worldPos() - pos();
445 collisionData.vel = colVel;
447 processWheel(collisionData, firstHitbox, &colInfo, &kclOut);
461 if (colInfo->perpendicularity <= 0.0f) {
465 m_colPerpendicularity = std::max(m_colPerpendicularity, colInfo->perpendicularity);
467 if (collisionData.bWallAtLeftCloser || collisionData.bWallAtRightCloser) {
471 f32 bspPosX = hitbox.bspHitbox()->
position.x;
472 if (EGG::Mathf::abs(bspPosX) > 10.0f) {
473 if (bspPosX > 0.0f) {
474 collisionData.bWallAtLeftCloser =
true;
476 collisionData.bWallAtRightCloser =
true;
479 collisionData.colPerpendicularity = colInfo->perpendicularity;
485 std::array<f32, 2> tangents = {0.0f, 0.0f};
488 for (
size_t i = 0; i < tangents.size(); ++i) {
489 f32 sign = i == 1 ? -1.0f : 1.0f;
490 f32 effectiveRadius = sign * hitbox.radius();
491 EGG::Vector3f effectivePos = hitbox.worldPos() + effectiveRadius * right;
494 if (Field::CollisionDirector::Instance()->checkSphereCachedPartial(hitbox.radius(),
501 if (tangents[0] > tangents[1]) {
502 collisionData.bWallAtLeftCloser =
true;
503 collisionData.colPerpendicularity = colInfo->perpendicularity;
504 }
else if (tangents[1] > tangents[0]) {
505 collisionData.bWallAtRightCloser =
true;
506 collisionData.colPerpendicularity = colInfo->perpendicularity;
511void KartCollide::calcBoundingRadius() {
512 m_boundingRadius = collisionGroup()->boundingRadius() * move()->hitboxScale();
516void KartCollide::calcObjectCollision() {
517 constexpr f32 COS_PI_OVER_4 = 0.707f;
518 constexpr s32 DUMMY_POLE_ANG_VEL_TIME = 3;
519 constexpr f32 DUMMY_POLE_ANG_VEL = 0.005f;
521 m_totalReactionWallNrm = EGG::Vector3f::zero;
522 m_surfaceFlags.
resetBit(eSurfaceFlags::ObjectWall, eSurfaceFlags::ObjectWall3);
524 size_t collisionCount = objectCollisionKart()->checkCollision(pose(), velocity());
526 const auto *objectDirector = Field::ObjectDirector::Instance();
528 for (
size_t i = 0; i < collisionCount; ++i) {
529 Reaction reaction = objectDirector->reaction(i);
530 if (reaction != Reaction::None && reaction != Reaction::UNK_7) {
531 size_t handlerIdx =
static_cast<std::underlying_type_t<Reaction>
>(reaction);
532 Action newAction = (this->*s_objectCollisionHandlers[handlerIdx])(i);
533 if (newAction != Action::None) {
534 action()->setHitDepth(objectDirector->hitDepth(i));
535 action()->
start(newAction);
538 if (reaction != Reaction::SmallBump && reaction != Reaction::BigBump) {
540 m_tangentOff += hitDepth;
541 m_movement += hitDepth;
545 if (objectDirector->collidingObject(i)->id() == Field::ObjectId::DummyPole) {
546 EGG::Vector3f hitDirection = objectCollisionKart()->GetHitDirection(i);
549 if (lastDir.
dot(hitDirection) < -COS_PI_OVER_4) {
551 f32 sign = angVel.y > 0.0f ? -1.0f : 1.0f;
553 m_poleAngVelTimer = DUMMY_POLE_ANG_VEL_TIME;
554 m_poleYaw = DUMMY_POLE_ANG_VEL * sign;
563void KartCollide::calcPoleTimer() {
564 if (m_poleAngVelTimer > 0 && (state()->isAccelerate() || state()->isBrake())) {
566 angVel2.y += m_poleYaw;
567 dynamics()->setAngVel2(angVel2);
570 m_poleAngVelTimer = std::max(0, m_poleAngVelTimer - 1);
578 processFloor(collisionData, hitbox, colInfo, maskOut,
true);
584 bool hasWallCollision = processWall(collisionData, maskOut);
586 processFloor(collisionData, hitbox, colInfo, maskOut,
false);
588 if (hasWallCollision) {
596bool KartCollide::processWall(CollisionData &collisionData, Field::KCLTypeMask *maskOut) {
601 auto *colDirector = Field::CollisionDirector::Instance();
607 colDirector->findClosestCollisionEntry(maskOut,
609 auto *entry = colDirector->closestCollisionEntry();
611 collisionData.bSoftWall =
true;
628 if (collisionData.bSoftWall) {
629 ++m_someSoftWallTimer;
630 m_suspBottomHeightSoftWall += hitbox.worldPos().y - hitbox.radius();
637 auto *colDirector = Field::CollisionDirector::Instance();
639 if (!colDirector->findClosestCollisionEntry(maskOut,
KCL_TYPE_FLOOR)) {
643 const auto *closestColEntry = colDirector->closestCollisionEntry();
645 u16 attribute = closestColEntry->attribute;
646 if (!(attribute & 0x2000)) {
647 m_surfaceFlags.
setBit(eSurfaceFlags::NotTrickable);
649 collisionData.bTrickable =
true;
650 m_surfaceFlags.
setBit(eSurfaceFlags::Trickable);
653 collisionData.speedFactor = std::min(collisionData.speedFactor,
656 collisionData.
intensity = (attribute >> 0xb) & 3;
659 if (attribute & 0x4000) {
660 state()->setRejectRoad(
true);
667 move()->padType().
setBit(KartMove::ePadType::BoostPanel);
670 if (!!(*maskOut & BOOST_RAMP_MASK) &&
671 colDirector->findClosestCollisionEntry(maskOut, BOOST_RAMP_MASK)) {
672 closestColEntry = colDirector->closestCollisionEntry();
673 move()->padType().
setBit(KartMove::ePadType::BoostRamp);
675 m_surfaceFlags.
setBit(eSurfaceFlags::BoostRamp, eSurfaceFlags::Trickable);
677 state()->setBoostRampType(-1);
678 m_surfaceFlags.
setBit(eSurfaceFlags::NotTrickable);
681 if (!collisionData.bSoftWall) {
682 ++m_someNonSoftWallTimer;
683 m_suspBottomHeightNonSoftWall += hitbox.worldPos().y - hitbox.radius();
687 state()->setStickyRoad(
true);
691 if ((*maskOut & halfPipeRampMask) &&
692 colDirector->findClosestCollisionEntry(maskOut, halfPipeRampMask)) {
693 state()->setHalfPipeRamp(
true);
694 state()->setHalfPipeInvisibilityTimer(2);
695 if (
KCL_VARIANT_TYPE(colDirector->closestCollisionEntry()->attribute) == 1) {
696 move()->padType().
setBit(KartMove::ePadType::BoostPanel);
701 if (*maskOut & jumpPadMask && colDirector->findClosestCollisionEntry(maskOut, jumpPadMask)) {
702 if (!state()->isTouchingGround() || !state()->isJumpPad()) {
703 move()->padType().
setBit(KartMove::ePadType::JumpPad);
704 closestColEntry = colDirector->closestCollisionEntry();
707 collisionData.bTrickable =
true;
714 auto *colDirector = Field::CollisionDirector::Instance();
716 state()->setCannonPointId(
718 state()->setCannonStart(
true);
733 bool b1,
bool b2,
bool b3) {
739 f32 velDotFloorNrm = colData.vel.
dot(colData.floorNrm);
741 if (velDotFloorNrm >= 0.0f) {
746 rotMat.
makeQ(dynamics()->mainRot());
751 EGG::Vector3f crossVec = colData.relPos.cross(colData.floorNrm);
753 crossVec = crossVec.cross(colData.relPos);
755 f32 scalar = -velDotFloorNrm / (1.0f + colData.floorNrm.
dot(crossVec));
757 crossVec = colData.floorNrm.cross(negSpeed);
758 crossVec = crossVec.cross(colData.floorNrm);
760 if (std::numeric_limits<f32>::epsilon() >= crossVec.
squaredLength()) {
765 f32 speedDot = std::min(0.0f, speed.
dot(crossVec));
766 crossVec *= ((scalar * speedDot) / velDotFloorNrm);
768 auto [proj, rej] = crossVec.projAndRej(forward);
770 f32 projNorm = proj.length();
771 f32 rejNorm = rej.length();
772 f32 projNorm_ = projNorm;
773 f32 rejNorm_ = rejNorm;
775 f32 dVar7 = down * EGG::Mathf::abs(scalar);
776 if (dVar7 < EGG::Mathf::abs(projNorm)) {
778 if (projNorm < 0.0f) {
779 projNorm_ = -down * EGG::Mathf::abs(scalar);
783 f32 dVar5 = rate * EGG::Mathf::abs(scalar);
784 if (dVar5 < EGG::Mathf::abs(rejNorm)) {
786 if (rejNorm < 0.0f) {
787 rejNorm_ = -rate * EGG::Mathf::abs(scalar);
808 projRejSum = projRejSum.
rej(nextDir);
810 dynamics()->setExtVel(dynamics()->extVel() + projRejSum);
813 EGG::Vector3f rotation = colData.relPos.cross(projRejSumOrig);
821 dynamics()->setAngVel0(dynamics()->angVel0() + angVel);
834 state()->softWallSpeed().dot(colInfo.wallNrm) < 0.3f) {
838 bool skipWalls =
false;
840 collisionData.wallNrm += colInfo.wallNrm;
843 collisionData.bInvisibleWall =
true;
846 collisionData.bInvisibleWallOnly =
true;
856 collisionData.
bWall3 =
true;
858 collisionData.
bWall =
true;
864 collisionData.floorNrm += colInfo.floorNrm;
865 collisionData.
bFloor =
true;
869 minMax.min = minMax.min.
minimize(tangentOff);
870 minMax.max = minMax.max.
maximize(tangentOff);
873 relPos += hitbox.relPos();
874 relPos += -hitbox.radius() * tangentOff;
886 setPos(pos() + movement);
889 collisionData.movement = movement;
892 f32 rotFactor = 1.0f /
static_cast<f32
>(count);
894 collisionData.rotFactor *= rotFactor;
896 EGG::Vector3f scaledAngVel0 = dynamics()->angVel0Factor() * dynamics()->angVel0();
900 local_30 += extVel();
902 collisionData.vel = local_30;
903 collisionData.relPos = scaledRelPos;
905 if (collisionData.
bFloor) {
906 f32 intVelY = dynamics()->intVel().y;
907 if (intVelY > 0.0f) {
908 collisionData.vel.y += intVelY;
915void KartCollide::startFloorMomentRate() {
916 m_floorMomentRate = 0.01f;
920void KartCollide::calcFloorMomentRate() {
921 m_floorMomentRate = state()->isInAction() ? 0.01f : std::min(m_floorMomentRate + 0.01f, 0.8f);
930Action KartCollide::handleReactWallAllSpeed(
size_t idx) {
931 m_totalReactionWallNrm += Field::ObjectCollisionKart::GetHitDirection(idx);
932 m_surfaceFlags.
setBit(eSurfaceFlags::ObjectWall);
938Action KartCollide::handleReactSpinAllSpeed(
size_t ) {
939 return Action::UNK_0;
943Action KartCollide::handleReactSpinSomeSpeed(
size_t ) {
944 return Action::UNK_1;
948Action KartCollide::handleReactFireSpin(
size_t ) {
949 return Action::UNK_9;
953Action KartCollide::handleReactSmallLaunch(
size_t ) {
954 return Action::UNK_2;
958Action KartCollide::handleReactKnockbackSomeSpeedLoseItem(
size_t ) {
959 return Action::UNK_3;
963Action KartCollide::handleReactLaunchSpinLoseItem(
size_t ) {
964 return Action::UNK_6;
968Action KartCollide::handleReactKnockbackBumpLoseItem(
size_t ) {
969 return Action::UNK_4;
973Action KartCollide::handleReactLongCrushLoseItem(
size_t ) {
974 return Action::UNK_12;
978Action KartCollide::handleReactHighLaunchLoseItem(
size_t ) {
979 return Action::UNK_8;
983Action KartCollide::handleReactWeakWall(
size_t ) {
984 move()->setSpeed(move()->speed() * 0.82f);
989Action KartCollide::handleReactLaunchSpin(
size_t ) {
990 return Action::UNK_5;
994Action KartCollide::handleReactWallSpark(
size_t idx) {
995 m_totalReactionWallNrm += Field::ObjectCollisionKart::GetHitDirection(idx);
996 m_surfaceFlags.
setBit(eSurfaceFlags::ObjectWall3);
1002Action KartCollide::handleReactShortCrushLoseItem(
size_t ) {
1003 return Action::UNK_14;
1007Action KartCollide::handleReactCrushRespawn(
size_t ) {
1008 return Action::UNK_16;
1012Action KartCollide::handleReactExplosionLoseItem(
size_t ) {
1013 return Action::UNK_7;
1016std::array<KartCollide::ObjectCollisionHandler, 33> KartCollide::s_objectCollisionHandlers = {{
1025 &KartCollide::handleReactWallAllSpeed,
1026 &KartCollide::handleReactSpinAllSpeed,
1027 &KartCollide::handleReactSpinSomeSpeed,
1028 &KartCollide::handleReactFireSpin,
1030 &KartCollide::handleReactSmallLaunch,
1031 &KartCollide::handleReactKnockbackSomeSpeedLoseItem,
1032 &KartCollide::handleReactLaunchSpinLoseItem,
1033 &KartCollide::handleReactKnockbackBumpLoseItem,
1034 &KartCollide::handleReactLongCrushLoseItem,
1038 &KartCollide::handleReactHighLaunchLoseItem,
1040 &KartCollide::handleReactWeakWall,
1042 &KartCollide::handleReactLaunchSpin,
1043 &KartCollide::handleReactWallSpark,
1047 &KartCollide::handleReactShortCrushLoseItem,
1048 &KartCollide::handleReactCrushRespawn,
1049 &KartCollide::handleReactExplosionLoseItem,
#define KCL_TYPE_DRIVER_SOLID_SURFACE
0xEAFABDFF
#define KCL_TYPE_DRIVER_WALL_NO_INVISIBLE_WALL2
0x4010B000
#define KCL_TYPE_DRIVER_WALL
0xC010B000
@ COL_TYPE_STICKY_ROAD
Player sticks if within 200 units (rBC stairs).
@ COL_TYPE_BOOST_RAMP
Trickable. Variant affects boost duration.
@ COL_TYPE_WALL_2
Difference to COL_TYPE_WALL is unknown.
@ COL_TYPE_HALFPIPE_RAMP
Bowser's Castle half-pipe ramps.
@ COL_TYPE_HALFPIPE_INVISIBLE_WALL
Invisible wall after a half-pipe jump, like in BC.
@ COL_TYPE_FALL_BOUNDARY
Non-solid out-of-bounds trigger.
@ COL_TYPE_BOOST_PAD
Boost panel.
@ COL_TYPE_CANNON_TRIGGER
Launches player to a destination "cannon point".
@ COL_TYPE_SOLID_OOB
Solid out-of-bounds trigger.
@ COL_TYPE_JUMP_PAD
Like GBA Shy Guy Beach.
@ COL_TYPE_EFFECT_TRIGGER
Activates various effects based on variant.
#define KCL_ATTRIBUTE_TYPE(x)
Computes the "Base Type" portion of the KCL flags. It's the lower 5 bits of the flag.
#define KCL_VARIANT_TYPE(x)
Extracts the "Variant" portion of the 2 byte KCL flag. It's the 3 bits before the "Bast Type".
#define KCL_TYPE_NON_DIRECTIONAL
0xE0F8BDFF
#define KCL_TYPE_4010D000
0x4010D000
#define KCL_TYPE_DRIVER_WALL_NO_INVISIBLE_WALL
0xC0109000
#define KCL_TYPE_FLOOR
0x20E80FFF - Any KCL that the player or items can drive/land on.
#define KCL_TYPE_ANY_INVISIBLE_WALL
0x90002000
#define KCL_TYPE_VEHICLE_COLLIDEABLE
0xEAF8BDFF
#define KCL_TYPE_DIRECTIONAL
0x05070000
#define KCL_TYPE_WALL
0xD010F000
Matrix34f multiplyTo(const Matrix34f &rhs) const
Multiplies two matrices.
void makeQ(const Quatf &q)
Sets rotation matrix from quaternion.
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.
Matrix34f transpose() const
Transposes the 3x3 portion of the matrix.
bool findClosestCollisionEntry(KCLTypeMask *typeMask, KCLTypeMask type)
Finds the closest KCL triangle out of the list of tris we are colliding with.
Houses hitbox and collision info for an object (body or wheel).
Represents a hitbox for the kart body or a wheel.
void calc(f32 totalScale, f32 sinkDepth, const EGG::Vector3f &scale, const EGG::Quatf &rot, const EGG::Vector3f &pos)
Calculates the position of a given hitbox, both relative to the player and world.
bool start(Action action)
Starts an action.
void applyBodyCollision(CollisionData &collisionData, const EGG::Vector3f &movement, const EGG::Vector3f &posRel, s32 count)
Saves collision info when vehicle body collision occurs.
void calcWheelCollision(u16 wheelIdx, CollisionGroup *hitboxGroup, const EGG::Vector3f &colVel, const EGG::Vector3f ¢er, f32 radius)
Checks wheel hitbox collision and stores position/velocity info.
void applySomeFloorMoment(f32 down, f32 rate, CollisionGroup *hitboxGroup, const EGG::Vector3f &forward, const EGG::Vector3f &nextDir, const EGG::Vector3f &speed, bool b1, bool b2, bool b3)
Applies external and angular velocity based on the collision with the floor.
void processCannon(Field::KCLTypeMask *maskOut)
Checks if we are colliding with a cannon trigger and sets the state flag if so.
Action handleReactNone(size_t idx)
Object collision functions.
bool FUN_805B6A9C(CollisionData &collisionData, const Hitbox &hitbox, EGG::BoundBox3f &minMax, EGG::Vector3f &relPos, s32 &count, const Field::KCLTypeMask &maskOut, const Field::CollisionInfo &colInfo)
Called on collision of a new KCL type??? This only happens after airtime so far.
void processWheel(CollisionData &collisionData, Hitbox &hitbox, Field::CollisionInfo *colInfo, Field::KCLTypeMask *maskOut)
Processes moving water and floor collision effects.
void calcHitboxes()
On each frame, calculates the positions for each hitbox.
void calcSideCollision(CollisionData &collisionData, Hitbox &hitbox, Field::CollisionInfo *colInfo)
void processFloor(CollisionData &collisionData, Hitbox &hitbox, Field::CollisionInfo *colInfo, Field::KCLTypeMask *maskOut, bool wheel)
Processes the floor triangles' attributes.
void FUN_805B72B8(f32 param_1, f32 param_2, bool lockXZ, bool addExtVelY)
Affects velocity when landing from airtime.
void calcBodyCollision(f32 totalScale, f32 sinkDepth, const EGG::Quatf &rot, const EGG::Vector3f &scale)
Checks and acts on collision for each kart hitbox.
Pertains to kart-related functionality.
A representation of a bounding cuboid.
A quaternion, used to represent 3D rotation.
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
Vector3f rotateVectorInv(const Vector3f &vec) const
Rotates a vector on the inverse quat.
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 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 squaredLength() const
The dot product between the vector and itself.
Vector3f rej(const Vector3f &rhs) const
The rejection of this vector onto rhs.
Vector3f maximize(const Vector3f &rhs) const
Returns a vector whose elements are the max of the elements of both vectors.
Vector3f minimize(const Vector3f &rhs) const
Returns a vector whose elements are the min of the elements of both vectors.
Represents one of the many hitboxes that make up a vehicle.
EGG::Vector3f position
The relative position of the hitbox.
Information about the current collision and its properties.
u32 closestFloorSettings
The KCL flag's "variant".
bool bFloor
Set if colliding with KCL which satisfies KCL_TYPE_FLOOR.
bool bWall3
Set if colliding with COL_TYPE_WALL_2.
bool bWall
Set if colliding with KCL which satisfies KCL_TYPE_WALL.
Field::KCLTypeMask closestFloorFlags
The KCL flag's KColType.
s32 intensity
The KCL flag's "wheel depth".
std::array< f32, 32 > kclRot
Rotation scalars, indexed using KCL attributes.