A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
RailInterpolator.cc
1#include "RailInterpolator.hh"
2
3#include "game/field/RailManager.hh"
4
5namespace Field {
6
8RailInterpolator::RailInterpolator(f32 speed, u32 idx) {
9 m_railIdx = idx;
10 auto *rail = RailManager::Instance()->rail(idx);
11 m_pointCount = rail->pointCount();
12 m_points = rail->points();
13 m_isOscillating = rail->isOscillating();
14 m_speed = speed;
15}
16
18RailInterpolator::~RailInterpolator() = default;
19
21const EGG::Vector3f &RailInterpolator::floorNrm(size_t idx) const {
22 return RailManager::Instance()->rail(idx)->floorNrm(idx);
23}
24
26f32 RailInterpolator::railLength() const {
27 return RailManager::Instance()->rail(m_railIdx)->getPathLength();
28}
29
31void RailInterpolator::updateVel() {
32 f32 t = m_segmentT;
33 setCurrVel((1.0f - t) * m_prevPointVel + t * m_nextPointVel);
34}
35
37void RailInterpolator::calcVelocities() {
38 m_prevPointVel = static_cast<f32>(m_points[m_currPointIdx].setting[0]);
39 m_nextPointVel = static_cast<f32>(m_points[m_nextPointIdx].setting[0]);
40
41 if (m_prevPointVel == 0.0f) {
42 m_prevPointVel = m_speed;
43 }
44
45 if (m_nextPointVel == 0.0f) {
46 m_nextPointVel = m_speed;
47 }
48}
49
51bool RailInterpolator::shouldChangeDirection() const {
52 if (!m_isOscillating) {
53 return m_pointCount == m_nextPointIdx;
54 }
55
56 return m_movementDirectionForward ? m_nextPointIdx == m_pointCount : m_nextPointIdx == -1;
57}
58
60void RailInterpolator::calcDirectionChange() {
61 if (!m_isOscillating) {
62 return;
63 }
64
65 if (m_movementDirectionForward) {
66 m_nextPointIdx -= 2;
67 } else {
68 m_nextPointIdx += 2;
69 }
70
71 m_movementDirectionForward = !m_movementDirectionForward;
72}
73
74void RailInterpolator::calcNextIndices() {
75 if (m_movementDirectionForward) {
76 ++m_currPointIdx;
77 ++m_nextPointIdx;
78 } else {
79 --m_currPointIdx;
80 --m_nextPointIdx;
81 }
82
83 if (!m_isOscillating) {
84 if (m_nextPointIdx == m_pointCount) {
85 m_nextPointIdx = 0;
86 }
87
88 if (m_currPointIdx == m_pointCount) {
89 m_currPointIdx = 0;
90 }
91 }
92}
93
95RailLinearInterpolator::RailLinearInterpolator(f32 speed, u32 idx) : RailInterpolator(speed, idx) {
96 m_transitions = RailManager::Instance()->rail(m_railIdx)->getLinearTransitions();
97 init(0.0f, 0);
98}
99
101RailLinearInterpolator::~RailLinearInterpolator() = default;
102
104void RailLinearInterpolator::init(f32 t, u32 idx) {
105 m_segmentT = t;
106 m_currPointIdx = idx;
107
108 bool isLastPoint = (static_cast<u16>(idx) == m_pointCount - 1);
109 m_nextPointIdx = isLastPoint ? idx - 1 : idx + 1;
110 m_movementDirectionForward = isLastPoint ? !m_isOscillating : true;
111
112 m_curPos = m_points[m_currPointIdx].pos;
113 m_currentDirection = m_points[m_nextPointIdx].pos - m_curPos;
114 m_curTangentDir = m_currentDirection;
115 m_curTangentDir.normalise2();
116 m_currVel = m_speed;
117 m_prevPointVel = m_speed;
118 m_nextPointVel = m_speed;
119 m_4a = false;
120 m_usePerPointVelocities = false;
121 m_currSegmentVel = m_speed / m_currentDirection.length();
122}
123
125RailInterpolator::Status RailLinearInterpolator::calc() {
126 if (m_4a) {
127 m_curPos = m_points[m_pointCount - 1].pos;
128
129 return Status::ChangingDirection;
130 }
131
132 if (m_usePerPointVelocities) {
133 updateVel();
134 }
135
136 m_segmentT += m_currSegmentVel;
137 m_curPos = lerp(m_segmentT, m_currPointIdx, m_nextPointIdx);
138
139 if (m_segmentT <= 1.0f) {
140 return Status::InProgress;
141 }
142
143 Status status = Status::SegmentEnd;
144
145 calcNextSegment();
146
147 if (shouldChangeDirection()) {
148 status = Status::ChangingDirection;
149
150 calcDirectionChange();
151 }
152
153 m_currentDirection = m_points[m_nextPointIdx].pos - m_points[m_currPointIdx].pos;
154 m_curTangentDir = m_currentDirection;
155 m_currSegmentVel = m_currVel / m_currentDirection.length();
156 m_curTangentDir.normalise2();
157
158 return status;
159}
160
162void RailLinearInterpolator::setCurrVel(f32 speed) {
163 m_currVel = speed;
164 m_currSegmentVel = m_currVel / m_currentDirection.length();
165}
166
168void RailLinearInterpolator::evalCubicBezierOnPath(f32 t, EGG::Vector3f &currDir,
169 EGG::Vector3f &curTangentDir) {
170 s16 currIdx = 0;
171 f32 len = 0.0f;
172
173 getPathLocation(t, currIdx, len);
174
175 s16 nextIdx = currIdx + 1;
176 if (nextIdx == m_pointCount) {
177 nextIdx = 0;
178 }
179
180 currDir = m_points[currIdx].pos * (1.0f - len) + m_points[nextIdx].pos * len;
181 curTangentDir = m_transitions[currIdx].m_dir;
182}
183
185void RailLinearInterpolator::getPathLocation(f32 t, s16 &idx, f32 &len) {
186 if (!m_movementDirectionForward) {
187 return;
188 }
189
190 f32 dist = m_segmentT * m_transitions[m_currPointIdx].m_length;
191 if (t <= dist) {
192 idx = m_currPointIdx;
193 len = (dist - t) * m_transitions[m_currPointIdx].m_lengthInv;
194 return;
195 }
196
197 for (s32 i = 0; i < m_pointCount; ++i) {
198 s32 currIdx = m_currPointIdx - 1;
199 if (currIdx == -1) {
200 currIdx = m_pointCount - 1;
201 }
202
203 currIdx -= i;
204 if (currIdx < 0) {
205 currIdx += m_pointCount;
206 }
207
208 dist += m_transitions[currIdx].m_length;
209 if (t <= dist) {
210 idx = currIdx;
211 len = (dist - t) * m_transitions[currIdx].m_lengthInv;
212 return;
213 }
214 }
215}
216
218void RailLinearInterpolator::calcNextSegment() {
219 calcNextIndices();
220
221 f32 prevDirLength = m_currentDirection.length();
222 m_currentDirection = m_points[m_nextPointIdx].pos - m_points[m_currPointIdx].pos;
223 m_segmentT = ((m_segmentT - 1.0f) * prevDirLength) / m_currentDirection.length();
224
225 if (shouldChangeDirection()) {
226 m_segmentT = 0.0f;
227 }
228
229 if (m_segmentT > 1.0f) {
230 m_segmentT = 0.99f;
231 }
232
233 if (m_usePerPointVelocities) {
234 calcVelocities();
235 }
236}
237
239EGG::Vector3f RailLinearInterpolator::lerp(f32 t, u32 currIdx, u32 nextIdx) const {
240 return m_points[currIdx].pos * (1.0f - t) + m_points[nextIdx].pos * t;
241}
242
244RailSmoothInterpolator::RailSmoothInterpolator(f32 speed, u32 idx) : RailInterpolator(speed, idx) {
245 auto *rail = RailManager::Instance()->rail(m_railIdx);
246 m_transitions = rail->getSplineTransitions();
247 m_estimatorSampleCount = static_cast<u32>(rail->getEstimatorSampleCount());
248 m_estimatorStep = rail->getEstimatorStep();
249 m_pathPercentages = rail->getPathPercentages();
250
251 init(0.0f, 0);
252}
253
255RailSmoothInterpolator::~RailSmoothInterpolator() = default;
256
258void RailSmoothInterpolator::init(f32 t, u32 idx) {
259 m_segmentT = t;
260
261 m_currPointIdx = idx;
262
263 if (idx == static_cast<u32>(m_pointCount - 1) && m_isOscillating) {
264 m_nextPointIdx = m_currPointIdx - 1;
265 m_movementDirectionForward = false;
266 m_curTangentDir = calcCubicBezierTangentDir(m_segmentT, m_transitions[m_currPointIdx]);
267 m_currSegmentVel = m_currVel * m_transitions[m_nextPointIdx].m_lengthInv;
268 } else {
269 m_nextPointIdx = idx + 1 == m_pointCount ? 0 : idx + 1;
270 m_movementDirectionForward = true;
271 m_curTangentDir = calcCubicBezierTangentDir(m_segmentT, m_transitions[m_currPointIdx]);
272 m_currSegmentVel = m_currVel * m_transitions[m_currPointIdx].m_lengthInv;
273 }
274
275 m_curPos = m_points[m_currPointIdx].pos;
276 m_prevPos = m_points[m_currPointIdx].pos;
277 m_currVel = m_speed;
278 m_prevPointVel = m_speed;
279 m_nextPointVel = m_speed;
280 m_velocity = 0.0f;
281 m_4a = false;
282 m_usePerPointVelocities = false;
283}
284
286RailInterpolator::Status RailSmoothInterpolator::calc() {
287 if (m_4a) {
288 m_curPos = m_transitions[m_pointCount - 2].m_p3;
289
290 return Status::ChangingDirection;
291 }
292
293 if (m_usePerPointVelocities) {
294 updateVel();
295 }
296
297 m_prevPos = m_curPos;
298
299 f32 t = m_movementDirectionForward ? calcT(m_segmentT) : calcT(1.0f - m_segmentT);
300
301 calcCubicBezier(t, m_currPointIdx, m_nextPointIdx, m_curPos, m_curTangentDir);
302
303 EGG::Vector3f deltaPos = m_curPos - m_prevPos;
304 m_velocity = deltaPos.length();
305 m_segmentT += m_currSegmentVel;
306
307 if (m_segmentT <= 1.0f) {
308 return Status::InProgress;
309 }
310
311 Status status = Status::SegmentEnd;
312
313 calcNextSegment();
314
315 if (shouldChangeDirection()) {
316 status = Status::ChangingDirection;
317
318 calcDirectionChange();
319 }
320
321 if (m_movementDirectionForward) {
322 m_currSegmentVel = m_currVel * m_transitions[m_currPointIdx].m_lengthInv;
323 } else {
324 m_currSegmentVel = m_currVel * m_transitions[m_nextPointIdx].m_lengthInv;
325 }
326
327 return status;
328}
329
331void RailSmoothInterpolator::setCurrVel(f32 speed) {
332 m_currVel = speed;
333
334 if (m_movementDirectionForward) {
335 m_currSegmentVel = speed * m_transitions[m_currPointIdx].m_lengthInv;
336 } else {
337 m_currSegmentVel = speed * m_transitions[m_nextPointIdx].m_lengthInv;
338 }
339}
340
342void RailSmoothInterpolator::evalCubicBezierOnPath(f32 t, EGG::Vector3f &currDir,
343 EGG::Vector3f &curTangentDir) {
344 s16 currIdx = 0;
345 f32 len = 0.0f;
346
347 getPathLocation(t, currIdx, len);
348
349 s16 nextIdx = currIdx + 1;
350 if (nextIdx == m_pointCount) {
351 nextIdx = 0;
352 }
353
354 len = m_movementDirectionForward ? calcT(len) : calcT(1.0f - len);
355
356 calcCubicBezier(len, currIdx, nextIdx, currDir, curTangentDir);
357}
358
360void RailSmoothInterpolator::getPathLocation(f32 t, s16 &idx, f32 &len) {
361 if (!m_movementDirectionForward) {
362 return;
363 }
364
365 f32 dist = m_segmentT * m_transitions[m_currPointIdx].m_length;
366 if (t <= dist) {
367 idx = m_currPointIdx;
368 len = (dist - t) * m_transitions[m_currPointIdx].m_lengthInv;
369 return;
370 }
371
372 for (s32 i = 0; i < m_pointCount; ++i) {
373 s32 currIdx = m_currPointIdx - 1;
374 if (currIdx == -1) {
375 currIdx = m_pointCount - 1;
376 }
377
378 currIdx -= i;
379 if (currIdx < 0) {
380 currIdx += m_pointCount;
381 }
382
383 dist += m_transitions[currIdx].m_length;
384 if (t <= dist) {
385 idx = currIdx;
386 len = (dist - t) * m_transitions[currIdx].m_lengthInv;
387 return;
388 }
389 }
390}
391
393void RailSmoothInterpolator::calcCubicBezier(f32 t, u32 currIdx, u32 nextIdx, EGG::Vector3f &pos,
394 EGG::Vector3f &dir) const {
395 auto &transition = m_movementDirectionForward ? m_transitions[currIdx] : m_transitions[nextIdx];
396
397 pos = calcCubicBezierPos(t, transition);
398 dir = calcCubicBezierTangentDir(t, transition);
399}
400
402EGG::Vector3f RailSmoothInterpolator::calcCubicBezierPos(f32 t,
403 const RailSplineTransition &trans) const {
404 f32 dt = 1.0f - t;
405
406 EGG::Vector3f res = trans.m_p0 * (dt * dt * dt);
407 res += trans.m_p1 * (3.0f * t * (dt * dt));
408 res += trans.m_p2 * (3.0f * (t * t) * dt);
409 res += trans.m_p3 * (t * t * t);
410
411 return res;
412}
413
415EGG::Vector3f RailSmoothInterpolator::calcCubicBezierTangentDir(f32 t,
416 const RailSplineTransition &trans) const {
417 EGG::Vector3f c1 = trans.m_p0 * -1.0f + trans.m_p1 * 3.0f - (trans.m_p2 * 3.0f) + trans.m_p3;
418 EGG::Vector3f c2 = trans.m_p0 * 3.0f - trans.m_p1 * 6.0f + trans.m_p2 * 3.0f;
419 EGG::Vector3f c3 = trans.m_p0 * -3.0f + trans.m_p1 * 3.0f;
420 EGG::Vector3f ret = c1 * 3.0f * (t * t) + c2 * 2.0f * t + c3;
421
422 ret.normalise2();
423
424 if (!m_movementDirectionForward) {
425 ret *= -1.0f;
426 }
427
428 return ret;
429}
430
432f32 RailSmoothInterpolator::calcT(f32 t) const {
433 u16 sampleIdx = m_movementDirectionForward ? m_currPointIdx * m_estimatorSampleCount :
434 m_nextPointIdx * m_estimatorSampleCount;
435
436 f32 delta = 0.0f;
437 u16 idx = 0;
438
439 for (u16 i = 0; i < m_estimatorSampleCount - 1; ++i) {
440 f32 currPercent = m_pathPercentages[sampleIdx + i];
441 f32 nextPercent = m_pathPercentages[sampleIdx + i + 1];
442
443 if (currPercent <= t && nextPercent > t) {
444 delta = (t - currPercent) / (nextPercent - currPercent);
445 idx = i;
446 }
447 }
448
449 f32 lastPercent = m_pathPercentages[sampleIdx + m_estimatorSampleCount - 1];
450
451 if (lastPercent <= t && 1.0f >= t) {
452 idx = m_estimatorSampleCount - 1;
453 delta = (t - lastPercent) / (1.0f - lastPercent);
454 }
455
456 return m_estimatorStep * static_cast<f32>(idx) + m_estimatorStep * delta;
457}
458
460void RailSmoothInterpolator::calcNextSegment() {
461 f32 nextT = m_segmentT - 1.0f;
462
463 if (m_isOscillating) {
464 if (m_nextPointIdx == 0 || m_nextPointIdx == m_pointCount - 1) {
465 m_segmentT = 0.0f;
466 } else if (m_movementDirectionForward) {
467 m_segmentT = nextT * m_transitions[m_currPointIdx].m_length *
468 m_transitions[m_nextPointIdx].m_lengthInv;
469 } else {
470 m_segmentT = nextT * m_transitions[m_currPointIdx - 1].m_length *
471 m_transitions[m_nextPointIdx - 1].m_lengthInv;
472 }
473 } else {
474 m_segmentT = nextT * m_transitions[m_currPointIdx].m_length *
475 m_transitions[m_nextPointIdx].m_lengthInv;
476 }
477
478 if (m_segmentT > 1.0f) {
479 m_segmentT = 0.99f;
480 }
481
482 calcNextIndices();
483
484 if (m_usePerPointVelocities) {
485 calcVelocities();
486 }
487}
488
489} // namespace Field
Pertains to collision.
A 3D float vector.
Definition Vector.hh:83
f32 length() const
The square root of the vector's dot product.
Definition Vector.hh:187