1#include "KartSuspensionPhysics.hh"
3#include "game/kart/KartBody.hh"
4#include "game/kart/KartCollide.hh"
5#include "game/kart/KartDynamics.hh"
6#include "game/kart/KartState.hh"
7#include "game/kart/KartSub.hh"
8#include "game/kart/KartTire.hh"
10#include <egg/math/Math.hh>
15WheelPhysics::WheelPhysics(
u16 wheelIdx,
u16 bspWheelIdx)
16 : m_wheelIdx(wheelIdx), m_bspWheelIdx(bspWheelIdx), m_bspWheel(nullptr) {}
19WheelPhysics::~WheelPhysics() {
24void WheelPhysics::init() {
25 m_hitboxGroup =
new CollisionGroup;
26 m_hitboxGroup->createSingleHitbox(10.0f, EGG::Vector3f::zero);
30void WheelPhysics::initBsp() {
31 m_bspWheel = &bsp().wheels[m_bspWheelIdx];
35void WheelPhysics::reset() {
38 m_lastPosDiff.setZero();
42 m_wheelEdgePos.setZero();
43 m_effectiveRadius = 0.0f;
44 m_targetEffectiveRadius = 0.0f;
46 m_topmostPos.setZero();
49 m_suspTravel = m_bspWheel->maxTravel;
50 m_effectiveRadius = m_bspWheel->wheelRadius;
56 const EGG::Vector3f topmostPos = m_topmostPos + vehicleMovement;
57 f32 scaledMaxTravel = m_bspWheel->maxTravel * sub()->someScale();
58 f32 suspTravel = bottom.
dot(m_pos - topmostPos);
59 m_suspTravel = std::max(0.0f, std::min(scaledMaxTravel, suspTravel));
60 m_pos = topmostPos + m_suspTravel * bottom;
61 m_speed = m_pos - m_lastPos;
62 m_speed -= dynamics()->intVel();
63 m_speed -= collisionData().movement;
64 m_speed -= collide()->movement();
65 m_hitboxGroup->collisionData().vel += m_speed;
67 m_lastPosDiff = m_pos - topmostPos;
72 m_targetEffectiveRadius = m_bspWheel->wheelRadius;
73 if (!state()->isSkipWheelCalc()) {
74 f32 nextRadius = m_bspWheel->sphereRadius;
75 f32 scalar = m_effectiveRadius * scale().y - nextRadius * move()->totalScale();
78 scalar = 0.3f * (nextRadius * move()->leanRot()) * move()->totalScale();
79 center += scalar * bodyForward();
81 if (state()->isInCannon()) {
82 collisionData().reset();
84 m_hitboxGroup->setHitboxScale(move()->totalScale());
85 if (state()->isUNK2()) {
86 m_hitboxGroup->hitbox(0).setLastPos(dynamics()->pos());
89 collide()->calcWheelCollision(m_wheelIdx, m_hitboxGroup, m_colVel, center, nextRadius);
90 CollisionData &colData = m_hitboxGroup->collisionData();
92 if (colData.bFloor || colData.bWall || colData.bWall3) {
93 m_pos += colData.tangentOff;
94 if (colData.intensity > -1) {
95 f32 sinkDepth = 3.0f *
static_cast<f32
>(colData.intensity);
96 m_targetEffectiveRadius = m_bspWheel->wheelRadius - sinkDepth;
97 body()->trySetTargetSinkDepth(sinkDepth);
101 m_hitboxGroup->hitbox(0).setLastPos(center);
104 m_topmostPos = topmostPos;
105 m_wheelEdgePos = m_pos + m_effectiveRadius * move()->totalScale() * bottom;
106 m_effectiveRadius += (m_targetEffectiveRadius - m_effectiveRadius) * 0.1f;
107 m_suspTravel = bottom.
dot(m_pos - topmostPos);
109 if (m_suspTravel < 0.0f) {
112 sub()->updateSuspOvertravel(suspBottom);
119KartSuspensionPhysics::KartSuspensionPhysics(
u16 wheelIdx, TireType tireType,
u16 bspWheelIdx)
120 : m_tirePhysics(nullptr), m_tireType(tireType), m_bspWheelIdx(bspWheelIdx),
121 m_wheelIdx(wheelIdx) {}
124KartSuspensionPhysics::~KartSuspensionPhysics() =
default;
127void KartSuspensionPhysics::init() {
128 m_tirePhysics = tire(m_wheelIdx)->wheelPhysics();
129 m_bspWheel = &bsp().wheels[m_bspWheelIdx];
133void KartSuspensionPhysics::reset() {
134 m_topmostPos.setZero();
135 m_maxTravelScaled = 0.0f;
136 m_bottomDir.setZero();
140void KartSuspensionPhysics::setInitialState() {
142 if (m_tireType == TireType::KartReflected) {
143 relPos.x = -relPos.x;
146 const EGG::Vector3f rotatedRelPos = dynamics()->fullRot().rotateVector(relPos) + pos();
147 const EGG::Vector3f unitRotated = dynamics()->fullRot().rotateVector(-EGG::Vector3f::ey);
149 m_tirePhysics->setPos(rotatedRelPos + m_bspWheel->maxTravel * unitRotated);
150 m_tirePhysics->setLastPos(rotatedRelPos + m_bspWheel->maxTravel * unitRotated);
151 m_tirePhysics->setLastPosDiff(m_tirePhysics->pos() - rotatedRelPos);
152 m_tirePhysics->setWheelEdgePos(m_tirePhysics->pos() +
153 (m_tirePhysics->effectiveRadius() * move()->totalScale() * unitRotated));
154 m_tirePhysics->hitboxGroup()->hitbox(0).setWorldPos(m_tirePhysics->pos());
155 m_tirePhysics->hitboxGroup()->hitbox(0).setLastPos(pos() + 100 * EGG::Vector3f::ey);
156 m_topmostPos = rotatedRelPos;
160void KartSuspensionPhysics::calcCollision(f32 dt,
const EGG::Vector3f &gravity,
162 m_maxTravelScaled = m_bspWheel->maxTravel * sub()->someScale();
164 EGG::Vector3f scaledRelPos = m_bspWheel->relPosition * scale();
165 if (m_tireType == TireType::KartReflected) {
166 scaledRelPos.x = -scaledRelPos.x;
171 EGG::Vector3f euler_angles(m_bspWheel->xRot * DEG2RAD, 0.0f, 0.0f);
172 mStack_60.
makeR(euler_angles);
176 f32 y_down = m_tirePhysics->suspTravel() + sub()->someScale() * 5.0f;
177 m_tirePhysics->setSuspTravel(std::max(0.0f, std::min(m_maxTravelScaled, y_down)));
178 m_tirePhysics->setColVel(dt * 10.0f * gravity);
179 m_tirePhysics->setPos(topmostPos + m_tirePhysics->suspTravel() * m_bottomDir);
181 if (!state()->isSkipWheelCalc()) {
182 m_tirePhysics->updateCollision(m_bottomDir, topmostPos);
183 m_topmostPos = topmostPos;
194 m_tirePhysics->realign(m_bottomDir, vehicleMovement);
198 if (!collisionData.
bFloor) {
203 f32 yDown = std::max(0.0f, m_bottomDir.dot(topDiff));
205 f32 travel = m_maxTravelScaled - yDown;
206 f32 speedScalar = m_bottomDir.dot(speed);
209 -(m_bspWheel->springStiffness * travel + m_bspWheel->dampingFactor * speedScalar);
214 fRot.y = std::max(-1.0f, std::min(1.0f, fRot.y));
221 rotProj = rotProj.
proj(collisionData.floorNrm);
222 fLinear.y += rotProj.y;
223 fLinear.y = std::min(fLinear.y, param()->stats().maxNormalAcceleration);
225 if (dynamics()->extVel().y > 5.0f || state()->isJumpPadDisableYsusForce()) {
229 dynamics()->applySuspensionWrench(m_topmostPos, fLinear, fRot, state()->isWheelieRot());
231 f32 rate = state()->isSomethingWallCollision() ? 0.01f : collide()->floorMomentRate();
233 collide()->applySomeFloorMoment(0.1f, rate, hitboxGroup, forward, move()->dir(),
234 m_tirePhysics->speed(),
true,
true, !state()->isWheelieRot());
Vector3f multVector33(const Vector3f &vec) const
Multiplies a 3x3 matrix by a vector.
void makeR(const Vector3f &r)
Sets 3x3 rotation matrix from a vector of Euler angles.
Vector3f ps_multVector(const Vector3f &vec) const
Paired-singles impl. of multVector.
Houses hitbox and collision info for an object (body or wheel).
Pertains to kart-related functionality.
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
Vector3f proj(const Vector3f &rhs) const
The projection of this vector onto rhs.
Information about the current collision and its properties.
bool bFloor
Set if colliding with KCL which satisfies KCL_TYPE_FLOOR.