3#include "game/kart/KartAction.hh"
4#include "game/kart/KartBody.hh"
5#include "game/kart/KartCollide.hh"
6#include "game/kart/KartMove.hh"
7#include "game/kart/KartObject.hh"
8#include "game/kart/KartState.hh"
9#include "game/kart/KartSuspensionPhysics.hh"
11#include "game/field/BoxColManager.hh"
12#include "game/field/CollisionDirector.hh"
14#include "game/system/RaceManager.hh"
16#include <egg/math/Math.hh>
20KartSub::KartSub() =
default;
31void KartSub::createSubsystems(
bool isBike) {
32 m_move = isBike ?
new KartMoveBike :
new KartMove;
33 m_action =
new KartAction;
34 m_move->createSubsystems();
35 m_state =
new KartState;
36 m_collide =
new KartCollide;
42 pointers.collide = m_collide;
43 pointers.state = m_state;
44 pointers.move = m_move;
45 pointers.action = m_action;
53 move()->setTurnParams();
59void KartSub::initAABB(KartAccessor &accessor, KartObject *
object) {
60 f32 radius = 25.0f + collide()->boundingRadius();
61 f32 hardSpeedLimit = move()->hardSpeedLimit();
63 accessor.boxColUnit = Field::BoxColManager::Instance()->insertDriver(radius, hardSpeedLimit,
64 &pos(),
true,
object);
68void KartSub::initPhysicsValues() {
70 collide()->resetHitboxes();
74void KartSub::resetPhysics() {
77 collide()->resetHitboxes();
79 for (
u16 wheelIdx = 0; wheelIdx < suspCount(); ++wheelIdx) {
80 suspensionPhysics(wheelIdx)->reset();
82 for (
u16 tireIdx = 0; tireIdx < tireCount(); ++tireIdx) {
83 tirePhysics(tireIdx)->reset();
85 m_move->setKartSpeedLimit();
91 m_maxSuspOvertravel.setZero();
92 m_minSuspOvertravel.setZero();
101 if (state()->isCannonStart()) {
102 physics()->hitboxGroup()->reset();
103 for (
size_t i = 0; i < tireCount(); ++i) {
104 tirePhysics(i)->hitboxGroup()->reset();
106 move()->enterCannon();
111 if (state()->isTriggerRespawn()) {
115 move()->setTurnParams();
116 move()->calcRespawnStart();
119 physics()->setPos(dynamics()->pos());
120 physics()->setVelocity(dynamics()->velocity());
121 dynamics()->setGravity(-1.3f);
122 dynamics()->setAngVel0YFactor(0.9f);
128 if (state()->isSkipWheelCalc()) {
129 for (
size_t tireIdx = 0; tireIdx < tireCount(); ++tireIdx) {
130 tirePhysics(tireIdx)->setLastPos(pos());
137 dynamics()->setTop(move()->up());
140 const auto *raceManager = System::RaceManager::Instance();
141 if (!raceManager->isStageReached(System::RaceManager::Stage::Race)) {
142 dynamics()->setIntVel(EGG::Vector3f::zero);
146 killExtVel = killExtVel.
rej(move()->smoothedUp());
152 dynamics()->setExtVel(killExtVel);
155 f32 maxSpeed = move()->hardSpeedLimit();
156 physics()->
calc(
DT, maxSpeed, scale(), !state()->isTouchingGround());
158 move()->calcRejectRoad();
160 if (!state()->isInCannon()) {
162 collisionGroup()->setHitboxScale(move()->totalScale());
172 constexpr s16 SIDE_COLLISION_TIME = 5;
176 m_floorCollisionCount = 0;
177 m_maxSuspOvertravel.setZero();
178 m_minSuspOvertravel.setZero();
182 flags.
setBit(Field::eBoxColFlag::Drivable, Field::eBoxColFlag::Object);
183 boxColUnit()->search(flags);
185 collide()->calcObjectCollision();
186 dynamics()->setPos(pos() + collide()->tangentOff());
188 if (state()->isSomethingWallCollision()) {
189 const EGG::Vector3f &softWallSpeed = state()->softWallSpeed();
190 f32 speedFactor = 5.0f;
193 if (state()->isHWG()) {
195 effectiveSpeed = softWallSpeed;
197 effectiveSpeed = softWallSpeed.
perpInPlane(move()->smoothedUp(),
true);
198 f32 speedDotUp = softWallSpeed.
dot(move()->smoothedUp());
199 if (speedDotUp < 0.0f) {
200 speedFactor += -speedDotUp * 10.0f;
204 effectiveSpeed *= speedFactor * scale().y;
205 setPos(pos() + effectiveSpeed);
206 collide()->setMovement(collide()->movement() + effectiveSpeed);
209 auto &colData = collisionData();
213 if (colData.bWallAtLeftCloser || colData.bWallAtRightCloser) {
214 f32 sign = colData.bWallAtRightCloser ? 1.0f : -1.0f;
220 colPerpBounceDir.y = 0.0f;
221 setPos(pos() + colPerpBounceDir);
222 collide()->setMovement(collide()->movement() + colPerpBounceDir);
227 body()->calcSinkDepth();
229 Field::CollisionDirector::Instance()->checkCourseColNarrScLocal(250.0f, pos(),
232 if (!state()->isInCannon()) {
233 if (!state()->isZipperStick()) {
235 body()->calcTargetSinkDepth();
237 if (colData.bWall || colData.bWall3) {
238 collide()->setMovement(collide()->movement() + colData.movement);
244 collide()->calcFloorEffect();
245 collide()->calcFloorMomentRate();
247 if (colData.bFloor) {
249 addFloor(colData,
false);
254 m_someScale = scale().y;
257 f32 speedFactor = 1.0f;
258 f32 handlingFactor = 0.0f;
259 for (
u16 i = 0; i < suspCount(); ++i) {
261 suspensionPhysics(i)->calcCollision(
DT, gravity, wheelMatrix);
263 const CollisionData &colData = tirePhysics(i)->hitboxGroup()->collisionData();
265 speedFactor = std::min(speedFactor, colData.speedFactor);
268 handlingFactor += colData.rotFactor;
269 addFloor(colData,
false);
273 if (!state()->isSkipWheelCalc()) {
274 EGG::Vector3f vehicleCompensation = m_maxSuspOvertravel + m_minSuspOvertravel;
275 dynamics()->setPos(dynamics()->pos() + vehicleCompensation);
277 if (!collisionData().bFloor) {
283 for (
u16 wheelIdx = 0; wheelIdx < tireCount(); ++wheelIdx) {
284 const WheelPhysics *wheelPhysics = tirePhysics(wheelIdx);
285 if (wheelPhysics->_74() == 0.0f) {
289 const CollisionData &colData = wheelPhysics->hitboxGroup()->collisionData();
290 relPos += colData.relPos;
292 floorNrm += colData.floorNrm;
297 f32 scalar = (1.0f /
static_cast<f32
>(count));
300 collide()->setFloorColInfo(collisionData(), relPos * scalar, vel * scalar,
307 for (
u16 wheelIdx = 0; wheelIdx < suspCount(); ++wheelIdx) {
308 suspensionPhysics(wheelIdx)->
calcSuspension(forward, vehicleCompensation);
311 move()->calcHopPhysics();
314 move()->setKCLWheelSpeedFactor(speedFactor);
315 move()->setKCLWheelRotFactor(handlingFactor);
317 move()->setFloorCollisionCount(m_floorCollisionCount);
321 collide()->resetHitboxes();
327void KartSub::resizeAABB(f32 radiusScale) {
328 f32 radius = radiusScale * collisionGroup()->boundingRadius();
329 boxColUnit()->resize(radius + 25.0f, move()->hardSpeedLimit());
333void KartSub::addFloor(
const CollisionData &,
bool) {
334 ++m_floorCollisionCount;
338void KartSub::updateSuspOvertravel(
const EGG::Vector3f &suspOvertravel) {
339 m_maxSuspOvertravel = m_maxSuspOvertravel.
minimize(suspOvertravel);
340 m_minSuspOvertravel = m_minSuspOvertravel.
maximize(suspOvertravel);
344void KartSub::tryEndHWG() {
345 if (state()->isSoftWallDrift()) {
346 if (EGG::Mathf::abs(move()->speed()) > 15.0f || state()->isAirtimeOver20() ||
347 state()->isAllWheelsCollision()) {
348 state()->setSoftWallDrift(
false);
349 }
else if (state()->isTouchingGround()) {
350 if (EGG::Mathf::abs(componentXAxis().dot(EGG::Vector3f::ey)) > 0.8f) {
351 state()->setSoftWallDrift(
false);
356 if (state()->isHWG() && !state()->isSomethingWallCollision()) {
357 if (!state()->isWallCollision() || state()->isAllWheelsCollision()) {
358 state()->setHWG(
false);
362 if (!state()->isInAction()) {
363 dynamics()->setForceUpright(!state()->isSoftWallDrift());
#define KCL_TYPE_VEHICLE_INTERACTABLE
0xEFFFBDFF
virtual EGG::Matrix34f wheelMatrix(u16)
Computes a matrix to represent wheel rotation. For Karts, this is wheel-agnostic.
void calcHitboxes()
On each frame, calculates the positions for each hitbox.
void calc()
Each frame, calculates the kart's movement.
void updatePose()
Constructs a transformation matrix from rotation and position.
void calc(f32 dt, f32 maxSpeed, const EGG::Vector3f &scale, bool air)
Computes trick rotation and calls to KartDynamics::calc().
void calc()
Every frame, resets the input state and saves collision-related bit flags.
void calcInput()
Each frame, read input and save related bit flags. Also handles start boosts.
void resetEjection()
Resets certain bitfields pertaining to ejections (reject road, half pipe zippers, etc....
void calcPass0()
The first phase of physics computations on each frame.
static constexpr f32 DT
Delta time.
void copyPointers(KartAccessor &pointers)
Called during static construction of KartObject to synchronize the pointers.
f32 m_colPerpendicularity
Dot product between floor and colliding wall normals.
void calcPass1()
The second phase of physics computations on each frame.Handles the second-half of physics calculation...
s16 m_sideCollisionTimer
Number of frames to apply movement from wall collision.
void calcSuspension(const EGG::Vector3f &forward, const EGG::Vector3f &vehicleMovement)
Calculates linear force and rotation from the kart's suspension.
Manages wheel physics and collision checks.
Pertains to kart-related functionality.
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
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.
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.
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.
Information about the current collision and its properties.
bool bFloor
Set if colliding with KCL which satisfies KCL_TYPE_FLOOR.
Shared between classes who inherit KartObjectProxy so they can access one another.