A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KartReject.cc
1#include "KartReject.hh"
2
3#include "game/kart/KartDynamics.hh"
4#include "game/kart/KartMove.hh"
5#include "game/kart/KartParam.hh"
6#include "game/kart/KartState.hh"
7
8#include "game/field/CollisionDirector.hh"
9#include "game/field/CourseColMgr.hh"
10
11#include <egg/math/Math.hh>
12#include <egg/math/Quat.hh>
13
14namespace Kart {
15
17KartReject::KartReject() = default;
18
20KartReject::~KartReject() = default;
21
23void KartReject::reset() {
24 m_rejectSign = 0.0f;
25}
26
28void KartReject::calcRejectRoad() {
29 if (state()->isInAction()) {
30 return;
31 }
32
33 if (state()->isRejectRoadTrigger()) {
34 EGG::Vector3f down = -EGG::Vector3f::ey;
35 down = down.perpInPlane(move()->up(), true);
36 f32 cos = down.dot(move()->lastDir());
37 f32 sin = down.cross(move()->lastDir()).length();
38 f32 angle = EGG::Mathf::atan2(sin, cos);
39 angle = angle > 0.0f ? angle : -angle;
40
41 f32 minAngle = 60.0f;
42 angle *= RAD2DEG;
43 f32 dVar11 = 1.0f;
44
45 if (move()->up().dot(EGG::Vector3f::ey) < -0.7f) {
46 minAngle *= 0.5f;
47 dVar11 *= 2.0f;
48 }
49
50 if (angle > minAngle) {
51 angle = 0.05f * (angle - 60.0f);
52 EGG::Quatf rot;
53 EGG::Vector3f vRot = EGG::Vector3f(0.0f,
54 (1.0f + angle * move()->speedRatio()) * dVar11 * DEG2RAD * m_rejectSign, 0.0f);
55 rot.setRPY(vRot);
56 EGG::Quatf local_78 = mainRot().multSwap(rot);
57 local_78.normalise();
58
59 dynamics()->setAngVel0(EGG::Vector3f::zero);
60 dynamics()->setFullRot(local_78);
61 dynamics()->setMainRot(local_78);
62 }
63
64 state()->setHop(false);
65
66 bool didReject = calcRejection();
67
68 if (!state()->isNoSparkInvisibleWall() && !didReject) {
69 state()->setRejectRoadTrigger(false);
70 }
71
72 return;
73 }
74
75 if (state()->isRejectRoad() && !state()->isZipperInvisibleWall() && !state()->isOverZipper() &&
76 !state()->isHalfPipeRamp()) {
77 EGG::Vector3f upXZ = move()->up();
78 upXZ.y = 0.0f;
79
80 if (upXZ.length() > 0.0f && speed() > 0.0f) {
81 upXZ.normalise();
82 EGG::Vector3f local_88 = move()->lastDir().perpInPlane(upXZ, true);
83
84 if (local_88.y > 0.0f) {
85 EGG::Vector3f upCross = EGG::Vector3f::ey.cross(local_88);
86 m_rejectSign = upCross.dot(move()->up()) > 0.0f ? 1.0f : -1.0f;
87
88 state()->setHop(false);
89 state()->setRejectRoadTrigger(true);
90 }
91 }
92 }
93}
94
96bool KartReject::calcRejection() {
98 Field::KCLTypeMask mask = KCL_NONE;
99 state()->setNoSparkInvisibleWall(false);
100 EGG::Vector3f worldUpPos = dynamics()->pos() + bodyUp() * 100.0f;
101 f32 posScalar = 100.0f;
102 f32 radius = posScalar;
103
104 for (size_t i = 0; i < 2; ++i) {
105 EGG::Vector3f local_d0 = dynamics()->mainRot().rotateVector(EGG::Vector3f::ey);
106 EGG::Vector3f worldPos = pos() + (-posScalar * move()->scale().y) * local_d0;
107
108 auto *colDir = Field::CollisionDirector::Instance();
109 if (!colDir->checkSphereFullPush(radius, worldPos, worldUpPos, KCL_TYPE_B0E82DFF, &colInfo,
110 &mask, 0)) {
111 if (i == 0) {
112 posScalar = 0.0f;
113 }
114
115 continue;
116 }
117
118 bool hasFloorCollision = false;
119 bool hasRejectCollision = false;
120 bool hasInvisibleWallCollision = false;
121 EGG::Vector3f tangentOff = EGG::Vector3f::zero;
122
123 if (mask & KCL_TYPE_INVISIBLE_WALL) {
124 hasInvisibleWallCollision =
125 colDir->findClosestCollisionEntry(&mask, KCL_TYPE_INVISIBLE_WALL);
126 }
127
128 const auto *closestColEntry = colDir->closestCollisionEntry();
129 if (hasInvisibleWallCollision && KCL_VARIANT_TYPE(closestColEntry->attribute) == 0) {
130 hasRejectCollision = true;
131 tangentOff = colInfo.wallNrm;
132 state()->setNoSparkInvisibleWall(true);
133 } else {
134 if (!(mask & KCL_TYPE_DRIVER_FLOOR)) {
135 hasFloorCollision = false;
136 } else {
137 hasFloorCollision = colDir->findClosestCollisionEntry(&mask, KCL_TYPE_DRIVER_FLOOR);
138 }
139
140 closestColEntry = colDir->closestCollisionEntry();
141 if (hasFloorCollision && closestColEntry->attribute & 0x4000) {
142 hasRejectCollision = true;
143 tangentOff = colInfo.floorNrm;
144 }
145 }
146
147 if (!hasRejectCollision) {
148 continue;
149 }
150
151 EGG::Vector3f tangentUp = (tangentOff - move()->up()) * 1.0f;
152 move()->setUp(move()->up() + tangentUp);
153 move()->setSmoothedUp(move()->up());
154
155 bool bVar15 = tangentOff.dot(EGG::Vector3f::ey) < -0.17f;
156 if (bVar15 || extVel().y < 0.0f || state()->isNoSparkInvisibleWall()) {
157 radius = -radius;
158 colInfo.tangentOff += worldPos;
159
160 f32 yOffset = bsp().initialYPos * scale().y;
161 f32 speedScalar = bVar15 ?
162 1.0f :
163 static_cast<f32>(static_cast<f64>(EGG::Mathf::abs(speed()) * 0.01f) - 0.3d);
164 speedScalar = std::min(1.0f, std::max(0.0f, speedScalar));
165
166 EGG::Vector3f posOffset =
167 colInfo.tangentOff + radius * tangentOff + yOffset * tangentOff;
168 posOffset.y += move()->hopPosY();
169 posOffset -= pos();
170 setPos(pos() + posOffset * speedScalar);
171 }
172
173 EGG::Vector3f local_13c = move()->lastDir().perpInPlane(move()->smoothedUp(), true);
174 move()->setDir(local_13c);
175 move()->setVel1Dir(local_13c);
176
177 return true;
178 }
179
180 return false;
181}
182
183} // namespace Kart
#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_DRIVER_FLOOR
0x20E80DFF - Any KCL that the player can drive on.
#define KCL_TYPE_INVISIBLE_WALL
0x80002000
#define KCL_TYPE_B0E82DFF
0xB0E82DFF
EGG::Vector3f bodyUp() const
Returns the second column of the rotation matrix, which is the "up" direction.
Pertains to kart-related functionality.
A quaternion, used to represent 3D rotation.
Definition Quat.hh:12
void normalise()
Scales the quaternion to a unit length.
Definition Quat.cc:23
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
Definition Quat.cc:50
void setRPY(const Vector3f &rpy)
Sets roll, pitch, and yaw.
Definition Quat.cc:7
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 length() const
The square root of the vector's dot product.
Definition Vector.hh:187
Vector3f perpInPlane(const EGG::Vector3f &rhs, bool normalise) const
Calculates the orthogonal vector, based on the plane defined by this vector and rhs.
Definition Vector.cc:96