1#include "RailInterpolator.hh"
3#include "game/field/RailManager.hh"
8RailInterpolator::RailInterpolator(f32 speed, u32 idx) {
10 auto *rail = RailManager::Instance()->rail(idx);
11 m_pointCount = rail->pointCount();
12 m_points = rail->points();
13 m_isOscillating = rail->isOscillating();
18RailInterpolator::~RailInterpolator() =
default;
21const EGG::Vector3f &RailInterpolator::floorNrm(
size_t idx)
const {
22 return RailManager::Instance()->rail(idx)->floorNrm(idx);
26f32 RailInterpolator::railLength()
const {
27 return RailManager::Instance()->rail(m_railIdx)->getPathLength();
31void RailInterpolator::updateVel() {
33 setCurrVel((1.0f - t) * m_prevPointVel + t * m_nextPointVel);
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]);
41 if (m_prevPointVel == 0.0f) {
42 m_prevPointVel = m_speed;
45 if (m_nextPointVel == 0.0f) {
46 m_nextPointVel = m_speed;
51bool RailInterpolator::shouldChangeDirection()
const {
52 if (!m_isOscillating) {
53 return m_pointCount == m_nextPointIdx;
56 return m_movementDirectionForward ? m_nextPointIdx == m_pointCount : m_nextPointIdx == -1;
60void RailInterpolator::calcDirectionChange() {
61 if (!m_isOscillating) {
65 if (m_movementDirectionForward) {
71 m_movementDirectionForward = !m_movementDirectionForward;
74void RailInterpolator::calcNextIndices() {
75 if (m_movementDirectionForward) {
83 if (!m_isOscillating) {
84 if (m_nextPointIdx == m_pointCount) {
88 if (m_currPointIdx == m_pointCount) {
95RailLinearInterpolator::RailLinearInterpolator(f32 speed, u32 idx) : RailInterpolator(speed, idx) {
96 m_transitions = RailManager::Instance()->rail(m_railIdx)->getLinearTransitions();
101RailLinearInterpolator::~RailLinearInterpolator() =
default;
104void RailLinearInterpolator::init(f32 t, u32 idx) {
106 m_currPointIdx = idx;
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;
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();
117 m_prevPointVel = m_speed;
118 m_nextPointVel = m_speed;
120 m_usePerPointVelocities =
false;
121 m_currSegmentVel = m_speed / m_currentDirection.length();
125RailInterpolator::Status RailLinearInterpolator::calc() {
127 m_curPos = m_points[m_pointCount - 1].pos;
129 return Status::ChangingDirection;
132 if (m_usePerPointVelocities) {
136 m_segmentT += m_currSegmentVel;
137 m_curPos = lerp(m_segmentT, m_currPointIdx, m_nextPointIdx);
139 if (m_segmentT <= 1.0f) {
140 return Status::InProgress;
143 Status status = Status::SegmentEnd;
147 if (shouldChangeDirection()) {
148 status = Status::ChangingDirection;
150 calcDirectionChange();
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();
162void RailLinearInterpolator::setCurrVel(f32 speed) {
164 m_currSegmentVel = m_currVel / m_currentDirection.length();
168void RailLinearInterpolator::evalCubicBezierOnPath(f32 t,
EGG::Vector3f &currDir,
173 getPathLocation(t, currIdx, len);
175 s16 nextIdx = currIdx + 1;
176 if (nextIdx == m_pointCount) {
180 currDir = m_points[currIdx].pos * (1.0f - len) + m_points[nextIdx].pos * len;
181 curTangentDir = m_transitions[currIdx].m_dir;
185void RailLinearInterpolator::getPathLocation(f32 t, s16 &idx, f32 &len) {
186 if (!m_movementDirectionForward) {
190 f32 dist = m_segmentT * m_transitions[m_currPointIdx].m_length;
192 idx = m_currPointIdx;
193 len = (dist - t) * m_transitions[m_currPointIdx].m_lengthInv;
197 for (s32 i = 0; i < m_pointCount; ++i) {
198 s32 currIdx = m_currPointIdx - 1;
200 currIdx = m_pointCount - 1;
205 currIdx += m_pointCount;
208 dist += m_transitions[currIdx].m_length;
211 len = (dist - t) * m_transitions[currIdx].m_lengthInv;
218void RailLinearInterpolator::calcNextSegment() {
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();
225 if (shouldChangeDirection()) {
229 if (m_segmentT > 1.0f) {
233 if (m_usePerPointVelocities) {
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;
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();
255RailSmoothInterpolator::~RailSmoothInterpolator() =
default;
258void RailSmoothInterpolator::init(f32 t, u32 idx) {
261 m_currPointIdx = idx;
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;
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;
275 m_curPos = m_points[m_currPointIdx].pos;
276 m_prevPos = m_points[m_currPointIdx].pos;
278 m_prevPointVel = m_speed;
279 m_nextPointVel = m_speed;
282 m_usePerPointVelocities =
false;
286RailInterpolator::Status RailSmoothInterpolator::calc() {
288 m_curPos = m_transitions[m_pointCount - 2].m_p3;
290 return Status::ChangingDirection;
293 if (m_usePerPointVelocities) {
297 m_prevPos = m_curPos;
299 f32 t = m_movementDirectionForward ? calcT(m_segmentT) : calcT(1.0f - m_segmentT);
301 calcCubicBezier(t, m_currPointIdx, m_nextPointIdx, m_curPos, m_curTangentDir);
304 m_velocity = deltaPos.
length();
305 m_segmentT += m_currSegmentVel;
307 if (m_segmentT <= 1.0f) {
308 return Status::InProgress;
311 Status status = Status::SegmentEnd;
315 if (shouldChangeDirection()) {
316 status = Status::ChangingDirection;
318 calcDirectionChange();
321 if (m_movementDirectionForward) {
322 m_currSegmentVel = m_currVel * m_transitions[m_currPointIdx].m_lengthInv;
324 m_currSegmentVel = m_currVel * m_transitions[m_nextPointIdx].m_lengthInv;
331void RailSmoothInterpolator::setCurrVel(f32 speed) {
334 if (m_movementDirectionForward) {
335 m_currSegmentVel = speed * m_transitions[m_currPointIdx].m_lengthInv;
337 m_currSegmentVel = speed * m_transitions[m_nextPointIdx].m_lengthInv;
342void RailSmoothInterpolator::evalCubicBezierOnPath(f32 t,
EGG::Vector3f &currDir,
347 getPathLocation(t, currIdx, len);
349 s16 nextIdx = currIdx + 1;
350 if (nextIdx == m_pointCount) {
354 len = m_movementDirectionForward ? calcT(len) : calcT(1.0f - len);
356 calcCubicBezier(len, currIdx, nextIdx, currDir, curTangentDir);
360void RailSmoothInterpolator::getPathLocation(f32 t, s16 &idx, f32 &len) {
361 if (!m_movementDirectionForward) {
365 f32 dist = m_segmentT * m_transitions[m_currPointIdx].m_length;
367 idx = m_currPointIdx;
368 len = (dist - t) * m_transitions[m_currPointIdx].m_lengthInv;
372 for (s32 i = 0; i < m_pointCount; ++i) {
373 s32 currIdx = m_currPointIdx - 1;
375 currIdx = m_pointCount - 1;
380 currIdx += m_pointCount;
383 dist += m_transitions[currIdx].m_length;
386 len = (dist - t) * m_transitions[currIdx].m_lengthInv;
393void RailSmoothInterpolator::calcCubicBezier(f32 t, u32 currIdx, u32 nextIdx,
EGG::Vector3f &pos,
395 auto &transition = m_movementDirectionForward ? m_transitions[currIdx] : m_transitions[nextIdx];
397 pos = calcCubicBezierPos(t, transition);
398 dir = calcCubicBezierTangentDir(t, transition);
402EGG::Vector3f RailSmoothInterpolator::calcCubicBezierPos(f32 t,
403 const RailSplineTransition &trans)
const {
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);
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;
420 EGG::Vector3f ret = c1 * 3.0f * (t * t) + c2 * 2.0f * t + c3;
424 if (!m_movementDirectionForward) {
432f32 RailSmoothInterpolator::calcT(f32 t)
const {
433 u16 sampleIdx = m_movementDirectionForward ? m_currPointIdx * m_estimatorSampleCount :
434 m_nextPointIdx * m_estimatorSampleCount;
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];
443 if (currPercent <= t && nextPercent > t) {
444 delta = (t - currPercent) / (nextPercent - currPercent);
449 f32 lastPercent = m_pathPercentages[sampleIdx + m_estimatorSampleCount - 1];
451 if (lastPercent <= t && 1.0f >= t) {
452 idx = m_estimatorSampleCount - 1;
453 delta = (t - lastPercent) / (1.0f - lastPercent);
456 return m_estimatorStep *
static_cast<f32
>(idx) + m_estimatorStep * delta;
460void RailSmoothInterpolator::calcNextSegment() {
461 f32 nextT = m_segmentT - 1.0f;
463 if (m_isOscillating) {
464 if (m_nextPointIdx == 0 || m_nextPointIdx == m_pointCount - 1) {
466 }
else if (m_movementDirectionForward) {
467 m_segmentT = nextT * m_transitions[m_currPointIdx].m_length *
468 m_transitions[m_nextPointIdx].m_lengthInv;
470 m_segmentT = nextT * m_transitions[m_currPointIdx - 1].m_length *
471 m_transitions[m_nextPointIdx - 1].m_lengthInv;
474 m_segmentT = nextT * m_transitions[m_currPointIdx].m_length *
475 m_transitions[m_nextPointIdx].m_lengthInv;
478 if (m_segmentT > 1.0f) {
484 if (m_usePerPointVelocities) {
f32 length() const
The square root of the vector's dot product.