A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KartHalfPipe.cc
1#include "KartHalfPipe.hh"
2
3#include "game/kart/KartCollide.hh"
4#include "game/kart/KartDynamics.hh"
5#include "game/kart/KartMove.hh"
6#include "game/kart/KartParam.hh"
7#include "game/kart/KartPhysics.hh"
8#include "game/kart/KartState.hh"
9
10#include "game/field/CourseColMgr.hh"
11
12#include <egg/math/Math.hh>
13
14namespace Kart {
15
17KartHalfPipe::KartHalfPipe() = default;
18
20KartHalfPipe::~KartHalfPipe() = default;
21
23void KartHalfPipe::reset() {
24 m_stunt = StuntType::None;
25 m_touchingZipper = false;
26 m_timer = 0;
27}
28
30void KartHalfPipe::calc() {
31 constexpr s16 LANDING_BOOST_DELAY = 3;
32
33 if (state()->airtime() > 15 && state()->isOverZipper()) {
34 m_timer = LANDING_BOOST_DELAY;
35 }
36
37 bool isLanding = state()->isHalfPipeRamp() && m_timer <= 0;
38
39 calcTrick();
40
41 if (collide()->surfaceFlags().offBit(KartCollide::eSurfaceFlags::StopHalfPipeState) &&
42 m_touchingZipper && state()->isAirStart()) {
43 dynamics()->setExtVel(EGG::Vector3f::zero);
44 state()->setOverZipper(true);
45
46 EGG::Vector3f upXZ = move()->up();
47 upXZ.y = 0.0f;
48 upXZ.normalise();
49 EGG::Vector3f up = move()->dir().perpInPlane(upXZ, true);
50
51 EGG::Vector3f local_64 = up.cross(bodyUp().perpInPlane(up, true));
52 m_nextSign = local_64.dot(EGG::Vector3f::ey) > 0.0f ? 1.0f : -1.0f;
53
54 EGG::Vector3f velNorm = velocity();
55 velNorm.normalise();
56 EGG::Vector3f rot = dynamics()->mainRot().rotateVectorInv(velNorm);
57
58 m_rot.makeVectorRotation(rot, EGG::Vector3f::ez);
59 m_prevPos = prevPos();
60
61 calcLanding(false);
62
63 f32 scaledDir = std::min(65.0f, move()->dir().y * move()->speed());
64 m_attemptedTrickTimer = std::max<s32>(0, scaledDir * 2.0f / 1.3f - 1.0f);
65 } else {
66 if (state()->isOverZipper()) {
67 dynamics()->setGravity(-1.3f);
68
69 EGG::Vector3f side = mainRot().rotateVector(EGG::Vector3f::ez);
70 EGG::Vector3f velNorm = velocity();
71 velNorm.normalise();
72
73 EGG::Quatf sideRot;
74 sideRot.makeVectorRotation(side, velNorm);
75 sideRot = sideRot.multSwap(mainRot()).multSwap(m_rot);
76
77 f32 t = move()->calcSlerpRate(DEG2RAD360, mainRot(), sideRot);
78 EGG::Quatf slerp = mainRot().slerpTo(sideRot, t);
79 dynamics()->setFullRot(slerp);
80 dynamics()->setMainRot(slerp);
81
83
84 calcRot();
85 calcLanding(false);
86 } else {
87 if (state()->isHalfPipeRamp()) {
88 calcLanding(true);
89 }
90 }
91 }
92
93 m_timer = std::max(0, m_timer - 1);
94 m_touchingZipper = isLanding;
95}
96
98void KartHalfPipe::calcTrick() {
99 constexpr s16 TRICK_COOLDOWN = 10;
100
101 auto &trick = inputs()->currentState().trick;
102
103 if (trick != System::Trick::None) {
104 m_nextTimer = TRICK_COOLDOWN;
105 m_trick = trick;
106 }
107
108 if (state()->isOverZipper()) {
109 if (!state()->isZipperTrick() && m_nextTimer > 0 && state()->airtime() > 3 &&
110 state()->airtime() < 10) {
111 activateTrick(m_attemptedTrickTimer, m_trick);
112 }
113 }
114
115 m_nextTimer = std::max(0, m_nextTimer - 1);
116}
117
119void KartHalfPipe::calcRot() {
120 if (m_stunt == StuntType::None) {
121 return;
122 }
123
124 m_stuntManager.calcAngle();
125
126 f32 angle = m_rotSign * (DEG2RAD * m_stuntManager.angle);
127
128 switch (m_stunt) {
129 case StuntType::Side360:
130 case StuntType::Side720:
131 m_stuntRot.setRPY(EGG::Vector3f(0.0f, angle, 0.0f));
132 break;
133 case StuntType::Backside: {
134 EGG::Quatf rpy;
135 rpy.setRPY(
136 EGG::Vector3f(0.0f, DEG2RAD * (0.25f * -m_rotSign * m_stuntManager.angle), 0.0f));
137 EGG::Vector3f rot = rpy.rotateVector(EGG::Vector3f::ez);
138 m_stuntRot.setAxisRotation(angle, rot);
139 } break;
140 case StuntType::Frontside: {
141 EGG::Quatf rpy;
142 rpy.setRPY(EGG::Vector3f(0.0f, 0.0f, DEG2RAD * (0.2f * -m_rotSign * m_stuntManager.angle)));
143 EGG::Vector3f rot = rpy.rotateVector(EGG::Vector3f::ey);
144 m_stuntRot.setAxisRotation(angle, rot);
145 } break;
146 case StuntType::Frontflip:
147 m_stuntRot.setRPY(EGG::Vector3f(m_rotSign * angle, 0.0f, 0.0f));
148 break;
149 case StuntType::Backflip:
150 m_stuntRot.setRPY(EGG::Vector3f(-m_rotSign * angle, 0.0f, 0.0f));
151 break;
152 default:
153 break;
154 }
155
156 physics()->composeStuntRot(m_stuntRot);
157}
158
160void KartHalfPipe::calcLanding(bool) {
161 constexpr f32 LANDING_RADIUS = 150.0f;
162 constexpr f32 PREVIOUS_RADIUS = 200.0f;
163 constexpr f32 MIDAIR_RADIUS = 50.0f;
164 constexpr f32 WALL_RADIUS = 100.0f;
165
166 constexpr f32 COS_PI_OVER_4 = 0.707f;
167
168 Field::CollisionInfo colInfo;
169 Field::CollisionInfo colInfo2;
170 Field::KCLTypeMask maskOut;
171 EGG::Vector3f pos;
172 EGG::Vector3f upLocal;
173
174 Field::KCLTypeMask mask = state()->isOverZipper() ?
177 EGG::Vector3f prevPos = m_prevPos + EGG::Vector3f::ey * PREVIOUS_RADIUS;
178
179 bool hasDriverFloorCollision = move()->calcZipperCollision(LANDING_RADIUS, bsp().initialYPos,
180 pos, upLocal, prevPos, &colInfo, &maskOut, KCL_TYPE_DRIVER_FLOOR);
181
182 prevPos = hasDriverFloorCollision ? EGG::Vector3f::inf : prevPos;
183
184 if (state()->isOverZipper()) {
185 if (!move()->calcZipperCollision(MIDAIR_RADIUS, bsp().initialYPos, pos, upLocal, prevPos,
186 &colInfo2, &maskOut, mask)) {
187 mask |= KCL_TYPE_DRIVER_WALL;
188 }
189 }
190
191 if (move()->calcZipperCollision(WALL_RADIUS, bsp().initialYPos, pos, upLocal, prevPos,
192 &colInfo2, &maskOut, mask)) {
193 EGG::Vector3f up = move()->up();
194 move()->setUp(up + (colInfo2.wallNrm - up) * 0.2f);
195 move()->setSmoothedUp(move()->up());
196
197 f32 yScale = bsp().initialYPos * scale().y;
198 EGG::Vector3f newPos =
199 pos + colInfo2.tangentOff + -WALL_RADIUS * colInfo2.wallNrm + yScale * upLocal;
200 newPos.y += move()->hopPosY();
201
202 dynamics()->setPos(newPos);
203 move()->setDir(move()->dir().perpInPlane(move()->up(), true));
204 move()->setVel1Dir(move()->dir());
205
206 if (state()->isOverZipper()) {
207 state()->setZipperStick(true);
208 }
209
210 m_prevPos = newPos;
211 } else {
212 if (state()->isOverZipper()) {
213 state()->setZipperStick(false);
214 }
215 }
216
217 if (!hasDriverFloorCollision || state()->airtime() <= 5) {
218 return;
219 }
220
221 if (colInfo.floorNrm.dot(EGG::Vector3f::ey) <= COS_PI_OVER_4) {
222 return;
223 }
224
225 if (state()->isOverZipper()) {
226 state()->setZipperStick(false);
227 }
228}
229
231void KartHalfPipe::activateTrick(s32 duration, System::Trick trick) {
232 if (duration < 51 || trick == System::Trick::None) {
233 m_stunt = StuntType::None;
234 } else {
235 m_rotSign = m_nextSign;
236 bool timerThreshold = duration > 70;
237
238 switch (trick) {
239 case System::Trick::Up:
240 m_stunt = timerThreshold ? StuntType::Backside : StuntType::Backflip;
241 break;
242 case System::Trick::Down:
243 m_stunt = timerThreshold ? StuntType::Frontside : StuntType::Frontflip;
244 break;
245 case System::Trick::Left:
246 case System::Trick::Right:
247 m_stunt = timerThreshold ? StuntType::Side720 : StuntType::Side360;
248 m_rotSign = trick == System::Trick::Left ? 1.0f : -1.0f;
249 break;
250 default:
251 break;
252 }
253
254 m_stuntManager.setProperties(static_cast<size_t>(m_stunt));
255
256 state()->setZipperTrick(true);
257 }
258
259 m_stuntRot = EGG::Quatf::ident;
260}
261
263void KartHalfPipe::end(bool boost) {
264 if (state()->isOverZipper() && state()->airtime() > 5 && boost) {
265 move()->activateZipperBoost();
266 }
267
268 if (state()->isZipperTrick()) {
269 physics()->composeDecayingStuntRot(m_stuntRot);
270 }
271
272 if (state()->isOverZipper()) {
273 move()->setDir(mainRot().rotateVector(EGG::Vector3f::ez));
274 move()->setVel1Dir(move()->dir());
275 }
276
277 state()->setOverZipper(false);
278 state()->setZipperTrick(false);
279 state()->setZipperStick(false);
280
281 m_stunt = StuntType::None;
282}
283
284void KartHalfPipe::StuntManager::calcAngle() {
285 if (finalAngle * properties.finalAngleScalar < angle) {
286 angleDelta = std::max(properties.angleDeltaMin, angleDelta * angleDeltaFactor);
287 angleDeltaFactor = std::max(properties.angleDeltaFactorMin,
288 angleDeltaFactor - properties.angleDeltaFactorDecr);
289 }
290
291 angle = std::min(finalAngle, angle + angleDelta);
292}
293
294void KartHalfPipe::StuntManager::setProperties(size_t idx) {
295 static constexpr std::array<StuntProperties, 6> STUNT_PROPERTIES = {{
296 {6.0f, 2.5f, 0.955f, 0.01f, 0.7f, 360.0f},
297 {7.0f, 3.0f, 0.955f, 0.01f, 0.7f, 360.0f},
298 {7.0f, 3.0f, 0.95f, 0.01f, 0.7f, 360.0f},
299 {12.0f, 2.5f, 0.955f, 0.01f, 0.0f, 360.0f},
300 {4.0f, 4.0f, 0.98f, 0.01f, 0.0f, 360.0f},
301 {9.0f, 3.0f, 0.92f, 0.01f, 0.8f, 720.0f},
302 }};
303
304 ASSERT(idx < STUNT_PROPERTIES.size());
305
306 properties = STUNT_PROPERTIES[idx];
307 finalAngle = properties.finalAngle;
308 angleDelta = properties.angleDelta;
309 angleDeltaFactorDecr = properties.angleDeltaFactorDecr;
310 angle = 0.0f;
311 angleDeltaFactor = 1.0f;
312}
313
314} // namespace Kart
#define KCL_TYPE_DRIVER_WALL
0xC010B000
@ COL_TYPE_HALFPIPE_INVISIBLE_WALL
Invisible wall after a half-pipe jump, like in BC.
#define KCL_TYPE_DRIVER_FLOOR
0x20E80DFF - Any KCL that the player can drive on.
#define KCL_TYPE_BIT(x)
#define KCL_TYPE_ANY_INVISIBLE_WALL
0x90002000
s32 m_attemptedTrickTimer
When attempting a trick, tracks how long the animation would be.
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
Vector3f rotateVector(const Vector3f &vec) const
Rotates a vector based on the quat.
Definition Quat.cc:50
Quatf slerpTo(const Quatf &q2, f32 t) const
Performs spherical linear interpolation.
Definition Quat.cc:79
void setAxisRotation(f32 angle, const Vector3f &axis)
Set the quat given angle and axis.
Definition Quat.cc:106
Vector3f rotateVectorInv(const Vector3f &vec) const
Rotates a vector on the inverse quat.
Definition Quat.cc:64
void setRPY(const Vector3f &rpy)
Sets roll, pitch, and yaw.
Definition Quat.cc:7
void makeVectorRotation(const Vector3f &from, const Vector3f &to)
Captures rotation between two vectors.
Definition Quat.cc:35
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
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