A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KartCollide.cc
1#include "KartCollide.hh"
2
3#include "game/kart/KartBody.hh"
4#include "game/kart/KartMove.hh"
5#include "game/kart/KartPhysics.hh"
6#include "game/kart/KartState.hh"
7
8#include "game/field/CollisionDirector.hh"
9#include "game/field/ObjectCollisionKart.hh"
10#include "game/field/ObjectDirector.hh"
11
12#include <egg/math/BoundBox.hh>
13#include <egg/math/Math.hh>
14
15namespace Kart {
16
18KartCollide::KartCollide() {
19 m_boundingRadius = 100.0f;
20 m_surfaceFlags.makeAllZero();
21}
22
24KartCollide::~KartCollide() = default;
25
27void KartCollide::init() {
28 calcBoundingRadius();
29 m_floorMomentRate = 0.8f;
30 m_surfaceFlags.makeAllZero();
31 m_respawnTimer = 0;
32 m_solidOobTimer = 0;
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;
39 m_poleYaw = 0.0f;
40 m_colPerpendicularity = 0.0f;
41}
42
44void KartCollide::resetHitboxes() {
45 CollisionGroup *hitboxGroup = physics()->hitboxGroup();
46 for (u16 idx = 0; idx < hitboxGroup->hitboxCount(); ++idx) {
47 hitboxGroup->hitbox(idx).setLastPos(scale(), pose());
48 }
49}
50
55 CollisionGroup *hitboxGroup = physics()->hitboxGroup();
56 for (u16 idx = 0; idx < hitboxGroup->hitboxCount(); ++idx) {
57 hitboxGroup->hitbox(idx).calc(move()->totalScale(), body()->sinkDepth(), scale(), fullRot(),
58 pos());
59 }
60}
61
65 const EGG::Quatf &rot = state()->isEndHalfPipe() ? mainRot() : fullRot();
66 calcBodyCollision(move()->totalScale(), body()->sinkDepth(), rot, scale());
67
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)) {
76 colData.bWall = true;
77 } else if (m_surfaceFlags.onBit(eSurfaceFlags::ObjectWall3)) {
78 colData.bWall3 = true;
79 }
80 } else if (newWallCollision) {
81 colData.wallNrm += m_totalReactionWallNrm;
82 if (m_surfaceFlags.onBit(eSurfaceFlags::ObjectWall)) {
83 colData.bWall = true;
84 } else if (m_surfaceFlags.onBit(eSurfaceFlags::ObjectWall3)) {
85 colData.bWall3 = true;
86 }
87 }
88
89 colData.wallNrm.normalise();
90 }
91
93}
94
99 f32 fVar1;
100
101 if (isInRespawn() || state()->isBoost() || state()->isOverZipper() ||
102 state()->isZipperInvisibleWall() || state()->isNoSparkInvisibleWall() ||
103 state()->isHalfPipeRamp()) {
104 fVar1 = 0.0f;
105 } else {
106 fVar1 = 0.05f;
107 }
108
109 bool resetXZ = fVar1 > 0.0f && state()->isAirtimeOver20() && dynamics()->velocity().y < -50.0f;
110
111 FUN_805B72B8(state()->isInAction() ? 0.3f : 0.01f, fVar1, resetXZ,
112 !state()->isJumpPadDisableYsusForce());
113}
114
122void KartCollide::FUN_805B72B8(f32 param_1, f32 param_2, bool lockXZ, bool addExtVelY) {
123 const auto &colData = collisionData();
124
125 if (!colData.bFloor && !colData.bWall && !colData.bWall3) {
126 return;
127 }
128
129 EGG::Vector3f collisionDir = colData.floorNrm + colData.wallNrm;
130 collisionDir.normalise();
131
132 f32 directionalVelocity = colData.vel.dot(collisionDir);
133 if (directionalVelocity >= 0.0f) {
134 return;
135 }
136
137 EGG::Matrix34f rotMat;
138 EGG::Matrix34f rotMatTrans;
139
140 rotMat.makeQ(dynamics()->mainRot());
141 rotMatTrans = rotMat.transpose();
142 rotMat = rotMat.multiplyTo(dynamics()->invInertiaTensor()).multiplyTo(rotMatTrans);
143
144 EGG::Vector3f relPos = colData.relPos;
145 if (lockXZ) {
146 relPos.x = 0.0f;
147 relPos.z = 0.0f;
148 }
149
150 EGG::Vector3f step1 = relPos.cross(collisionDir);
151 EGG::Vector3f step2 = rotMat.multVector33(step1);
152 EGG::Vector3f step3 = step2.cross(relPos);
153 f32 val = (-directionalVelocity * (param_2 + 1.0f)) / (1.0f + collisionDir.dot(step3));
154 EGG::Vector3f step4 = collisionDir.cross(-colData.vel);
155 EGG::Vector3f step5 = step4.cross(collisionDir);
156 step5.normalise();
157
158 f32 fVar1 = param_1 * EGG::Mathf::abs(val);
159 f32 otherVal = (val * colData.vel.dot(step5)) / directionalVelocity;
160
161 f32 fVar3 = otherVal;
162 if (fVar1 < EGG::Mathf::abs(otherVal)) {
163 fVar3 = fVar1;
164 if (otherVal < 0.0f) {
165 fVar3 = -param_1 * EGG::Mathf::abs(val);
166 }
167 }
168
169 EGG::Vector3f step6 = val * collisionDir + fVar3 * step5;
170
171 f32 local_1d0 = step6.y;
172 if (!addExtVelY) {
173 local_1d0 = 0.0f;
174 } else if (colData.bFloor) {
175 f32 velY = intVel().y;
176 if (velY > 0.0f) {
177 velY += extVel().y;
178 if (velY < 0.0f) {
179 EGG::Vector3f newExtVel = extVel();
180 newExtVel.y = velY;
181 dynamics()->setExtVel(newExtVel);
182 }
183 }
184 }
185
186 f32 prevExtVelY = extVel().y;
187 EGG::Vector3f extVelAdd = step6;
188 extVelAdd.y = local_1d0;
189 dynamics()->setExtVel(extVel() + extVelAdd);
190
191 if (prevExtVelY < 0.0f && extVel().y > 0.0f && extVel().y < 10.0f) {
192 EGG::Vector3f extVelNoY = extVel();
193 extVelNoY.y = 0.0f;
194 dynamics()->setExtVel(extVelNoY);
195 }
196
197 EGG::Vector3f step7 = relPos.cross(step6);
198 EGG::Vector3f step8 = rotMat.multVector33(step7);
199 EGG::Vector3f step9 = mainRot().rotateVectorInv(step8);
200 step9.y = 0.0f;
201 dynamics()->setAngVel0(dynamics()->angVel0() + step9);
202}
203
210void KartCollide::calcBodyCollision(f32 totalScale, f32 sinkDepth, const EGG::Quatf &rot,
211 const EGG::Vector3f &scale) {
212 CollisionGroup *hitboxGroup = physics()->hitboxGroup();
213 CollisionData &collisionData = hitboxGroup->collisionData();
214 collisionData.reset();
215
216 EGG::Vector3f posRel = EGG::Vector3f::zero;
217 s32 count = 0;
218 Field::CollisionInfo colInfo;
219 colInfo.bbox.setDirect(EGG::Vector3f::zero, EGG::Vector3f::zero);
220 Field::KCLTypeMask maskOut;
222 EGG::BoundBox3f minMax;
223 minMax.setZero();
224 bool bVar1 = false;
225
226 for (u16 hitboxIdx = 0; hitboxIdx < hitboxGroup->hitboxCount(); ++hitboxIdx) {
227 Field::KCLTypeMask flags = KCL_TYPE_DRIVER_SOLID_SURFACE;
228 Hitbox &hitbox = hitboxGroup->hitbox(hitboxIdx);
229
230 if (hitbox.bspHitbox()->wallsOnly != 0) {
231 flags = 0x4A109000;
232 Field::CourseColMgr::Instance()->setNoBounceWallInfo(&noBounceWallInfo);
233 }
234
235 hitbox.calc(totalScale, sinkDepth, scale, rot, pos());
236
237 if (Field::CollisionDirector::Instance()->checkSphereCachedFullPush(hitbox.radius(),
238 hitbox.worldPos(), hitbox.lastPos(), flags, &colInfo, &maskOut, 0)) {
239 if (!!(maskOut & KCL_TYPE_VEHICLE_COLLIDEABLE)) {
240 Field::CollisionDirector::Instance()->findClosestCollisionEntry(&maskOut,
242 }
243
244 if (!FUN_805B6A9C(collisionData, hitbox, minMax, posRel, count, maskOut, colInfo)) {
245 bVar1 = true;
246 processBody(collisionData, hitbox, &colInfo, &maskOut);
247 }
248 }
249 }
250
251 if (bVar1) {
252 EGG::Vector3f movement = minMax.min + minMax.max;
253 applyBodyCollision(collisionData, movement, posRel, count);
254 } else {
255 collisionData.speedFactor = 1.0f;
256 collisionData.rotFactor = 1.0f;
257 }
258}
259
261void KartCollide::calcFloorEffect() {
262 if (state()->isTouchingGround()) {
263 m_surfaceFlags.setBit(eSurfaceFlags::Offroad, eSurfaceFlags::GroundBoostPanelOrRamp);
264 }
265
266 m_suspBottomHeightNonSoftWall = 0.0f;
267 m_surfaceFlags.resetBit(eSurfaceFlags::Wall, eSurfaceFlags::SolidOOB, eSurfaceFlags::BoostRamp,
268 eSurfaceFlags::Offroad, eSurfaceFlags::Trickable, eSurfaceFlags::NotTrickable,
269 eSurfaceFlags::StopHalfPipeState);
270 m_suspBottomHeightSoftWall = 0.0f;
271 m_someNonSoftWallTimer = 0;
272 m_someSoftWallTimer = 0;
273
274 Field::KCLTypeMask mask = KCL_NONE;
275 calcTriggers(&mask, pos(), false);
276
277 if (m_solidOobTimer >= 3 && m_surfaceFlags.onBit(eSurfaceFlags::SolidOOB) &&
278 m_surfaceFlags.offBit(eSurfaceFlags::Wall)) {
279 if (mask & KCL_TYPE_BIT(COL_TYPE_SOLID_OOB)) {
280 Field::CollisionDirector::Instance()->findClosestCollisionEntry(&mask,
282 }
283
284 activateOob(true, &mask, false, false);
285 }
286
287 mask = KCL_NONE;
288 calcTriggers(&mask, pos(), true);
289
290 m_solidOobTimer =
291 m_surfaceFlags.onBit(eSurfaceFlags::SolidOOB) ? std::min(3, m_solidOobTimer + 1) : 0;
292}
293
295void KartCollide::calcTriggers(Field::KCLTypeMask *mask, const EGG::Vector3f &pos, bool twoPoint) {
296 EGG::Vector3f v1 = twoPoint ? physics()->pos() : EGG::Vector3f::inf;
297 Field::KCLTypeMask typeMask = twoPoint ? KCL_TYPE_DIRECTIONAL : KCL_TYPE_NON_DIRECTIONAL;
298 f32 radius = twoPoint ? 80.0f : 100.0f * move()->totalScale();
299 f32 scalar = -bsp().initialYPos * move()->totalScale() * 0.3f;
300 EGG::Vector3f scaledPos = pos + scalar * componentYAxis();
301 EGG::Vector3f back = dynamics()->mainRot().rotateVector(EGG::Vector3f::ez);
302
303 m_smoothedBack += (back.dot(move()->smoothedUp()) - m_smoothedBack) * 0.3f;
304
305 scalar = m_smoothedBack * -physics()->fc() * 1.8f * move()->totalScale();
306 scaledPos += scalar * back;
307
308 bool collide = Field::CollisionDirector::Instance()->checkSphereCachedPartialPush(radius,
309 scaledPos, v1, typeMask, nullptr, mask, 0);
310
311 if (!collide) {
312 return;
313 }
314
315 if (twoPoint) {
316 handleTriggers(mask);
317 } else {
318 if (*mask & KCL_TYPE_FLOOR) {
319 Field::CollisionDirector::Instance()->findClosestCollisionEntry(mask, KCL_TYPE_FLOOR);
320 }
321
322 if (*mask & KCL_TYPE_WALL) {
323 m_surfaceFlags.setBit(eSurfaceFlags::Wall);
324 }
325
326 if (*mask & KCL_TYPE_BIT(COL_TYPE_SOLID_OOB)) {
327 m_surfaceFlags.setBit(eSurfaceFlags::SolidOOB);
328 }
329 }
330}
331
333void KartCollide::handleTriggers(Field::KCLTypeMask *mask) {
334 calcFallBoundary(mask, false);
335 processCannon(mask);
336
338 auto *colDir = Field::CollisionDirector::Instance();
339 if (colDir->findClosestCollisionEntry(mask, KCL_TYPE_BIT(COL_TYPE_EFFECT_TRIGGER))) {
340 if (KCL_VARIANT_TYPE(colDir->closestCollisionEntry()->attribute) == 4) {
341 halfPipe()->end(true);
342 state()->setEndHalfPipe(true);
343 m_surfaceFlags.setBit(eSurfaceFlags::StopHalfPipeState);
344 }
345 }
346 }
347}
348
350void KartCollide::calcFallBoundary(Field::KCLTypeMask *mask, bool /*shortBoundary*/) {
351 if (!(*mask & KCL_TYPE_BIT(COL_TYPE_FALL_BOUNDARY))) {
352 return;
353 }
354
355 auto *colDir = Field::CollisionDirector::Instance();
356 if (!colDir->findClosestCollisionEntry(mask, KCL_TYPE_BIT(COL_TYPE_FALL_BOUNDARY))) {
357 return;
358 }
359
360 activateOob(false, mask, false, false);
361}
362
364void KartCollide::calcBeforeRespawn() {
365 if (pos().y < 0.0f) {
366 activateOob(true, nullptr, false, false);
367 }
368
369 if (!state()->isBeforeRespawn()) {
370 return;
371 }
372
373 if (--m_respawnTimer > 0) {
374 return;
375 }
376
377 state()->setBeforeRespawn(false);
378 m_respawnTimer = 0;
379 move()->triggerRespawn();
380}
381
383void KartCollide::activateOob(bool /*detachCamera*/, Field::KCLTypeMask * /*mask*/,
384 bool /*somethingCPU*/, bool /*somethingBullet*/) {
385 constexpr s16 RESPAWN_TIME = 130;
386
387 if (state()->isBeforeRespawn()) {
388 return;
389 }
390
391 move()->initOob();
392
393 m_respawnTimer = RESPAWN_TIME;
394 state()->setBeforeRespawn(true);
395}
396
405void KartCollide::calcWheelCollision(u16 /*wheelIdx*/, CollisionGroup *hitboxGroup,
406 const EGG::Vector3f &colVel, const EGG::Vector3f &center, f32 radius) {
407 Hitbox &firstHitbox = hitboxGroup->hitbox(0);
408 BSP::Hitbox *bspHitbox = const_cast<BSP::Hitbox *>(firstHitbox.bspHitbox());
409 bspHitbox->radius = radius;
410 hitboxGroup->resetCollision();
411 firstHitbox.setWorldPos(center);
412
413 Field::CollisionInfo colInfo;
414 colInfo.bbox.setZero();
415 Field::KCLTypeMask kclOut;
417 Field::CourseColMgr::Instance()->setNoBounceWallInfo(&noBounceWallInfo);
418
419 bool collided = Field::CollisionDirector::Instance()->checkSphereCachedFullPush(
420 firstHitbox.radius(), firstHitbox.worldPos(), firstHitbox.lastPos(),
421 KCL_TYPE_VEHICLE_COLLIDEABLE, &colInfo, &kclOut, 0);
422
423 CollisionData &collisionData = hitboxGroup->collisionData();
424
425 if (!collided) {
426 collisionData.speedFactor = 1.0f;
427 collisionData.rotFactor = 1.0f;
428 return;
429 }
430
431 collisionData.tangentOff = colInfo.tangentOff;
432
433 if (noBounceWallInfo.dist > std::numeric_limits<f32>::min()) {
434 collisionData.tangentOff += noBounceWallInfo.tangentOff;
435 collisionData.noBounceWallNrm = noBounceWallInfo.fnrm;
436 collisionData.bSoftWall = true;
437 }
438
439 if (kclOut & KCL_TYPE_FLOOR) {
440 collisionData.bFloor = true;
441 collisionData.floorNrm = colInfo.floorNrm;
442 }
443
444 collisionData.relPos = firstHitbox.worldPos() - pos();
445 collisionData.vel = colVel;
446
447 processWheel(collisionData, firstHitbox, &colInfo, &kclOut);
448
449 if (!(kclOut & KCL_TYPE_VEHICLE_COLLIDEABLE)) {
450 return;
451 }
452
453 Field::CollisionDirector::Instance()->findClosestCollisionEntry(&kclOut,
455}
456
460 Field::CollisionInfo *colInfo) {
461 if (colInfo->perpendicularity <= 0.0f) {
462 return;
463 }
464
465 m_colPerpendicularity = std::max(m_colPerpendicularity, colInfo->perpendicularity);
466
467 if (collisionData.bWallAtLeftCloser || collisionData.bWallAtRightCloser) {
468 return;
469 }
470
471 f32 bspPosX = hitbox.bspHitbox()->position.x;
472 if (EGG::Mathf::abs(bspPosX) > 10.0f) {
473 if (bspPosX > 0.0f) {
474 collisionData.bWallAtLeftCloser = true;
475 } else {
476 collisionData.bWallAtRightCloser = true;
477 }
478
479 collisionData.colPerpendicularity = colInfo->perpendicularity;
480
481 return;
482 }
483
484 EGG::Vector3f right = dynamics()->mainRot().rotateVector(EGG::Vector3f::ex);
485 std::array<f32, 2> tangents = {0.0f, 0.0f};
486
487 // The loop is just to do left/right wall
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;
492 Field::CollisionInfoPartial tempColInfo;
493
494 if (Field::CollisionDirector::Instance()->checkSphereCachedPartial(hitbox.radius(),
495 effectivePos, hitbox.lastPos(), KCL_TYPE_DRIVER_WALL, &tempColInfo, nullptr,
496 0)) {
497 tangents[i] = colInfo->tangentOff.squaredLength();
498 }
499 }
500
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;
507 }
508}
509
511void KartCollide::calcBoundingRadius() {
512 m_boundingRadius = collisionGroup()->boundingRadius() * move()->hitboxScale();
513}
514
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;
520
521 m_totalReactionWallNrm = EGG::Vector3f::zero;
522 m_surfaceFlags.resetBit(eSurfaceFlags::ObjectWall, eSurfaceFlags::ObjectWall3);
523
524 size_t collisionCount = objectCollisionKart()->checkCollision(pose(), velocity());
525
526 const auto *objectDirector = Field::ObjectDirector::Instance();
527
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);
536 }
537
538 if (reaction != Reaction::SmallBump && reaction != Reaction::BigBump) {
539 const EGG::Vector3f &hitDepth = objectDirector->hitDepth(i);
540 m_tangentOff += hitDepth;
541 m_movement += hitDepth;
542 }
543 }
544
545 if (objectDirector->collidingObject(i)->id() == Field::ObjectId::DummyPole) {
546 EGG::Vector3f hitDirection = objectCollisionKart()->GetHitDirection(i);
547 const EGG::Vector3f &lastDir = move()->lastDir();
548
549 if (lastDir.dot(hitDirection) < -COS_PI_OVER_4) {
550 EGG::Vector3f angVel = hitDirection.cross(lastDir);
551 f32 sign = angVel.y > 0.0f ? -1.0f : 1.0f;
552
553 m_poleAngVelTimer = DUMMY_POLE_ANG_VEL_TIME;
554 m_poleYaw = DUMMY_POLE_ANG_VEL * sign;
555 }
556 }
557 }
558
559 calcPoleTimer();
560}
561
563void KartCollide::calcPoleTimer() {
564 if (m_poleAngVelTimer > 0 && (state()->isAccelerate() || state()->isBrake())) {
565 EGG::Vector3f angVel2 = dynamics()->angVel2();
566 angVel2.y += m_poleYaw;
567 dynamics()->setAngVel2(angVel2);
568 }
569
570 m_poleAngVelTimer = std::max(0, m_poleAngVelTimer - 1);
571}
572
576void KartCollide::processWheel(CollisionData &collisionData, Hitbox &hitbox,
577 Field::CollisionInfo *colInfo, Field::KCLTypeMask *maskOut) {
578 processFloor(collisionData, hitbox, colInfo, maskOut, true);
579}
580
582void KartCollide::processBody(CollisionData &collisionData, Hitbox &hitbox,
583 Field::CollisionInfo *colInfo, Field::KCLTypeMask *maskOut) {
584 bool hasWallCollision = processWall(collisionData, maskOut);
585
586 processFloor(collisionData, hitbox, colInfo, maskOut, false);
587
588 if (hasWallCollision) {
589 calcSideCollision(collisionData, hitbox, colInfo);
590 }
591
592 processCannon(maskOut);
593}
594
596bool KartCollide::processWall(CollisionData &collisionData, Field::KCLTypeMask *maskOut) {
597 if (!(*maskOut & KCL_TYPE_DRIVER_WALL_NO_INVISIBLE_WALL2)) {
598 return false;
599 }
600
601 auto *colDirector = Field::CollisionDirector::Instance();
602 if (!colDirector->findClosestCollisionEntry(maskOut, KCL_TYPE_DRIVER_WALL_NO_INVISIBLE_WALL2)) {
603 return false;
604 }
605
607 colDirector->findClosestCollisionEntry(maskOut,
609 auto *entry = colDirector->closestCollisionEntry();
610 if (entry->attribute & KCL_TYPE_BIT(COL_TYPE_WALL_2)) {
611 collisionData.bSoftWall = true;
612 }
613 }
614
615 return true;
616}
617
624void KartCollide::processFloor(CollisionData &collisionData, Hitbox &hitbox,
625 Field::CollisionInfo * /*colInfo*/, Field::KCLTypeMask *maskOut, bool wheel) {
626 constexpr Field::KCLTypeMask BOOST_RAMP_MASK = KCL_TYPE_BIT(COL_TYPE_BOOST_RAMP);
627
628 if (collisionData.bSoftWall) {
629 ++m_someSoftWallTimer;
630 m_suspBottomHeightSoftWall += hitbox.worldPos().y - hitbox.radius();
631 }
632
633 if (!(*maskOut & KCL_TYPE_VEHICLE_COLLIDEABLE)) {
634 return;
635 }
636
637 auto *colDirector = Field::CollisionDirector::Instance();
638
639 if (!colDirector->findClosestCollisionEntry(maskOut, KCL_TYPE_FLOOR)) {
640 return;
641 }
642
643 const auto *closestColEntry = colDirector->closestCollisionEntry();
644
645 u16 attribute = closestColEntry->attribute;
646 if (!(attribute & 0x2000)) {
647 m_surfaceFlags.setBit(eSurfaceFlags::NotTrickable);
648 } else {
649 collisionData.bTrickable = true;
650 m_surfaceFlags.setBit(eSurfaceFlags::Trickable);
651 }
652
653 collisionData.speedFactor = std::min(collisionData.speedFactor,
654 param()->stats().kclSpeed[KCL_ATTRIBUTE_TYPE(attribute)]);
655
656 collisionData.intensity = (attribute >> 0xb) & 3;
657 collisionData.rotFactor += param()->stats().kclRot[KCL_ATTRIBUTE_TYPE(attribute)];
658
659 if (attribute & 0x4000) {
660 state()->setRejectRoad(true);
661 }
662
663 collisionData.closestFloorFlags = closestColEntry->typeMask;
664 collisionData.closestFloorSettings = KCL_VARIANT_TYPE(attribute);
665
666 if (wheel && !!(*maskOut & KCL_TYPE_BIT(COL_TYPE_BOOST_PAD))) {
667 move()->padType().setBit(KartMove::ePadType::BoostPanel);
668 }
669
670 if (!!(*maskOut & BOOST_RAMP_MASK) &&
671 colDirector->findClosestCollisionEntry(maskOut, BOOST_RAMP_MASK)) {
672 closestColEntry = colDirector->closestCollisionEntry();
673 move()->padType().setBit(KartMove::ePadType::BoostRamp);
674 state()->setBoostRampType(KCL_VARIANT_TYPE(closestColEntry->attribute));
675 m_surfaceFlags.setBit(eSurfaceFlags::BoostRamp, eSurfaceFlags::Trickable);
676 } else {
677 state()->setBoostRampType(-1);
678 m_surfaceFlags.setBit(eSurfaceFlags::NotTrickable);
679 }
680
681 if (!collisionData.bSoftWall) {
682 ++m_someNonSoftWallTimer;
683 m_suspBottomHeightNonSoftWall += hitbox.worldPos().y - hitbox.radius();
684 }
685
686 if (*maskOut & KCL_TYPE_BIT(COL_TYPE_STICKY_ROAD)) {
687 state()->setStickyRoad(true);
688 }
689
690 Field::KCLTypeMask halfPipeRampMask = KCL_TYPE_BIT(COL_TYPE_HALFPIPE_RAMP);
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);
697 }
698 }
699
700 Field::KCLTypeMask jumpPadMask = KCL_TYPE_BIT(COL_TYPE_JUMP_PAD);
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();
705 state()->setJumpPadVariant(KCL_VARIANT_TYPE(closestColEntry->attribute));
706 }
707 collisionData.bTrickable = true;
708 }
709}
710
713void KartCollide::processCannon(Field::KCLTypeMask *maskOut) {
714 auto *colDirector = Field::CollisionDirector::Instance();
715 if (colDirector->findClosestCollisionEntry(maskOut, KCL_TYPE_BIT(COL_TYPE_CANNON_TRIGGER))) {
716 state()->setCannonPointId(
717 KCL_VARIANT_TYPE(colDirector->closestCollisionEntry()->attribute));
718 state()->setCannonStart(true);
719 }
720}
721
731void KartCollide::applySomeFloorMoment(f32 down, f32 rate, CollisionGroup *hitboxGroup,
732 const EGG::Vector3f &forward, const EGG::Vector3f &nextDir, const EGG::Vector3f &speed,
733 bool b1, bool b2, bool b3) {
734 CollisionData &colData = hitboxGroup->collisionData();
735 if (!colData.bFloor) {
736 return;
737 }
738
739 f32 velDotFloorNrm = colData.vel.dot(colData.floorNrm);
740
741 if (velDotFloorNrm >= 0.0f) {
742 return;
743 }
744
745 EGG::Matrix34f rotMat;
746 rotMat.makeQ(dynamics()->mainRot());
747 EGG::Matrix34f tmp = rotMat.multiplyTo(dynamics()->invInertiaTensor());
748 EGG::Matrix34f rotMatTrans = rotMat.transpose();
749 tmp = tmp.multiplyTo(rotMatTrans);
750
751 EGG::Vector3f crossVec = colData.relPos.cross(colData.floorNrm);
752 crossVec = tmp.multVector(crossVec);
753 crossVec = crossVec.cross(colData.relPos);
754
755 f32 scalar = -velDotFloorNrm / (1.0f + colData.floorNrm.dot(crossVec));
756 EGG::Vector3f negSpeed = -speed;
757 crossVec = colData.floorNrm.cross(negSpeed);
758 crossVec = crossVec.cross(colData.floorNrm);
759
760 if (std::numeric_limits<f32>::epsilon() >= crossVec.squaredLength()) {
761 return;
762 }
763
764 crossVec.normalise();
765 f32 speedDot = std::min(0.0f, speed.dot(crossVec));
766 crossVec *= ((scalar * speedDot) / velDotFloorNrm);
767
768 auto [proj, rej] = crossVec.projAndRej(forward);
769
770 f32 projNorm = proj.length();
771 f32 rejNorm = rej.length();
772 f32 projNorm_ = projNorm;
773 f32 rejNorm_ = rejNorm;
774
775 f32 dVar7 = down * EGG::Mathf::abs(scalar);
776 if (dVar7 < EGG::Mathf::abs(projNorm)) {
777 projNorm_ = dVar7;
778 if (projNorm < 0.0f) {
779 projNorm_ = -down * EGG::Mathf::abs(scalar);
780 }
781 }
782
783 f32 dVar5 = rate * EGG::Mathf::abs(scalar);
784 if (dVar5 < EGG::Mathf::abs(rejNorm)) {
785 rejNorm_ = dVar5;
786 if (rejNorm < 0.0f) {
787 rejNorm_ = -rate * EGG::Mathf::abs(scalar);
788 }
789 }
790
791 proj.normalise();
792 rej.normalise();
793
794 proj *= projNorm_;
795 rej *= rejNorm_;
796
797 EGG::Vector3f projRejSum = proj + rej;
798 EGG::Vector3f projRejSumOrig = projRejSum;
799
800 if (!b1) {
801 projRejSum.x = 0.0f;
802 projRejSum.z = 0.0f;
803 }
804 if (!b2) {
805 projRejSum.y = 0.0f;
806 }
807
808 projRejSum = projRejSum.rej(nextDir);
809
810 dynamics()->setExtVel(dynamics()->extVel() + projRejSum);
811
812 if (b3) {
813 EGG::Vector3f rotation = colData.relPos.cross(projRejSumOrig);
814 EGG::Vector3f rotation2 = dynamics()->mainRot().rotateVectorInv(tmp.multVector(rotation));
815
816 EGG::Vector3f angVel = rotation2;
817 angVel.y = 0.0f;
818 if (!b1) {
819 angVel.x = 0.0f;
820 }
821 dynamics()->setAngVel0(dynamics()->angVel0() + angVel);
822 }
823}
824
829bool KartCollide::FUN_805B6A9C(CollisionData &collisionData, const Hitbox &hitbox,
830 EGG::BoundBox3f &minMax, EGG::Vector3f &relPos, s32 &count,
831 const Field::KCLTypeMask &maskOut, const Field::CollisionInfo &colInfo) {
832 if (maskOut & KCL_TYPE_WALL) {
833 if (!(maskOut & KCL_TYPE_FLOOR) && state()->isHWG() &&
834 state()->softWallSpeed().dot(colInfo.wallNrm) < 0.3f) {
835 return true;
836 }
837
838 bool skipWalls = false;
839
840 collisionData.wallNrm += colInfo.wallNrm;
841
842 if (maskOut & KCL_TYPE_ANY_INVISIBLE_WALL) {
843 collisionData.bInvisibleWall = true;
844
845 if (!(maskOut & KCL_TYPE_4010D000)) {
846 collisionData.bInvisibleWallOnly = true;
847
849 skipWalls = true;
850 }
851 }
852 }
853
854 if (!skipWalls) {
855 if (maskOut & KCL_TYPE_BIT(COL_TYPE_WALL_2)) {
856 collisionData.bWall3 = true;
857 } else {
858 collisionData.bWall = true;
859 }
860 }
861 }
862
863 if (maskOut & KCL_TYPE_FLOOR) {
864 collisionData.floorNrm += colInfo.floorNrm;
865 collisionData.bFloor = true;
866 }
867
868 EGG::Vector3f tangentOff = colInfo.tangentOff;
869 minMax.min = minMax.min.minimize(tangentOff);
870 minMax.max = minMax.max.maximize(tangentOff);
871 tangentOff.normalise();
872
873 relPos += hitbox.relPos();
874 relPos += -hitbox.radius() * tangentOff;
875 ++count;
876
877 return false;
878}
879
885 const EGG::Vector3f &posRel, s32 count) {
886 setPos(pos() + movement);
887
888 if (!collisionData.bFloor && (collisionData.bWall || collisionData.bWall3)) {
889 collisionData.movement = movement;
890 }
891
892 f32 rotFactor = 1.0f / static_cast<f32>(count);
893 EGG::Vector3f scaledRelPos = rotFactor * posRel;
894 collisionData.rotFactor *= rotFactor;
895
896 EGG::Vector3f scaledAngVel0 = dynamics()->angVel0Factor() * dynamics()->angVel0();
897 EGG::Vector3f local_48 = mainRot().rotateVectorInv(scaledRelPos);
898 EGG::Vector3f local_30 = scaledAngVel0.cross(local_48);
899 local_30 = mainRot().rotateVector(local_30);
900 local_30 += extVel();
901
902 collisionData.vel = local_30;
903 collisionData.relPos = scaledRelPos;
904
905 if (collisionData.bFloor) {
906 f32 intVelY = dynamics()->intVel().y;
907 if (intVelY > 0.0f) {
908 collisionData.vel.y += intVelY;
909 }
910 collisionData.floorNrm.normalise();
911 }
912}
913
915void KartCollide::startFloorMomentRate() {
916 m_floorMomentRate = 0.01f;
917}
918
920void KartCollide::calcFloorMomentRate() {
921 m_floorMomentRate = state()->isInAction() ? 0.01f : std::min(m_floorMomentRate + 0.01f, 0.8f);
922}
923
925Action KartCollide::handleReactNone(size_t /*idx*/) {
926 return Action::None;
927}
928
930Action KartCollide::handleReactWallAllSpeed(size_t idx) {
931 m_totalReactionWallNrm += Field::ObjectCollisionKart::GetHitDirection(idx);
932 m_surfaceFlags.setBit(eSurfaceFlags::ObjectWall);
933
934 return Action::None;
935}
936
938Action KartCollide::handleReactSpinAllSpeed(size_t /*idx*/) {
939 return Action::UNK_0;
940}
941
943Action KartCollide::handleReactSpinSomeSpeed(size_t /*idx*/) {
944 return Action::UNK_1;
945}
946
948Action KartCollide::handleReactFireSpin(size_t /*idx*/) {
949 return Action::UNK_9;
950}
951
953Action KartCollide::handleReactSmallLaunch(size_t /*idx*/) {
954 return Action::UNK_2;
955}
956
958Action KartCollide::handleReactKnockbackSomeSpeedLoseItem(size_t /*idx*/) {
959 return Action::UNK_3;
960}
961
963Action KartCollide::handleReactLaunchSpinLoseItem(size_t /*idx*/) {
964 return Action::UNK_6;
965}
966
968Action KartCollide::handleReactKnockbackBumpLoseItem(size_t /*idx*/) {
969 return Action::UNK_4;
970}
971
973Action KartCollide::handleReactLongCrushLoseItem(size_t /*idx*/) {
974 return Action::UNK_12;
975}
976
978Action KartCollide::handleReactHighLaunchLoseItem(size_t /*idx*/) {
979 return Action::UNK_8;
980}
981
983Action KartCollide::handleReactWeakWall(size_t /*idx*/) {
984 move()->setSpeed(move()->speed() * 0.82f);
985 return Action::None;
986}
987
989Action KartCollide::handleReactLaunchSpin(size_t /*idx*/) {
990 return Action::UNK_5;
991}
992
994Action KartCollide::handleReactWallSpark(size_t idx) {
995 m_totalReactionWallNrm += Field::ObjectCollisionKart::GetHitDirection(idx);
996 m_surfaceFlags.setBit(eSurfaceFlags::ObjectWall3);
997
998 return Action::None;
999}
1000
1002Action KartCollide::handleReactShortCrushLoseItem(size_t /*idx*/) {
1003 return Action::UNK_14;
1004}
1005
1007Action KartCollide::handleReactCrushRespawn(size_t /*idx*/) {
1008 return Action::UNK_16;
1009}
1010
1012Action KartCollide::handleReactExplosionLoseItem(size_t /*idx*/) {
1013 return Action::UNK_7;
1014}
1015
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,
1050}};
1051
1052} // namespace Kart
#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_BIT(x)
#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
A 3 x 4 matrix.
Definition Matrix.hh:8
Matrix34f multiplyTo(const Matrix34f &rhs) const
Multiplies two matrices.
Definition Matrix.cc:197
void makeQ(const Quatf &q)
Sets rotation matrix from quaternion.
Definition Matrix.cc:65
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
Matrix34f transpose() const
Transposes the 3x3 portion of the matrix.
Definition Matrix.cc:328
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.
Definition KartAction.cc:46
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 &center, 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.
Definition BoundBox.hh:30
A quaternion, used to represent 3D rotation.
Definition Quat.hh:12
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
Definition Quat.cc:50
Vector3f rotateVectorInv(const Vector3f &vec) const
Rotates a vector on the inverse quat.
Definition Quat.cc:64
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 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 squaredLength() const
The dot product between the vector and itself.
Definition Vector.hh:177
Vector3f rej(const Vector3f &rhs) const
The rejection of this vector onto rhs.
Definition Vector.hh:199
Vector3f maximize(const Vector3f &rhs) const
Returns a vector whose elements are the max of the elements of both vectors.
Definition Vector.cc:65
Vector3f minimize(const Vector3f &rhs) const
Returns a vector whose elements are the min of the elements of both vectors.
Definition Vector.cc:77
Represents one of the many hitboxes that make up a vehicle.
Definition KartParam.hh:10
EGG::Vector3f position
The relative position of the hitbox.
Definition KartParam.hh:12
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.
Definition KartParam.hh:113