1#include "MapdataCheckPoint.hh"
3#include "game/system/CourseMap.hh"
4#include "game/system/map/MapdataCheckPath.hh"
11MapdataCheckPoint::MapdataCheckPoint(
const SData *data)
12 : m_rawData(data), m_nextCount(0), m_prevCount(0) {
15 m_midpoint = 0.5f * (m_left + m_right);
16 m_dir =
EGG::Vector2f(m_right.y - m_left.y, m_left.x - m_right.x);
23 m_jugemIndex = stream.read_s8();
24 m_checkArea = stream.read_s8();
25 m_prevPt = stream.read_u8();
26 m_nextPt = stream.read_u8();
35 const auto *checkPathAccessor = CourseMap::Instance()->checkPath();
39 if (m_prevPt == 0xFF) {
40 MapdataCheckPath *checkpath = checkPathAccessor->findCheckpathForCheckpoint(
id);
44 for (
auto [i, prevID] : std::views::enumerate(checkpath->prev())) {
49 m_prevPoints[i] = accessor.get(checkPathAccessor->get(prevID)->end());
54 m_prevPoints[0] = accessor.get(m_prevPt);
61 if (m_nextPt == 0xFF) {
62 MapdataCheckPath *checkpath = checkPathAccessor->findCheckpathForCheckpoint(
id);
66 for (
auto [i, nextID] : std::views::enumerate(checkpath->next())) {
71 m_nextPoints[i].checkpoint = accessor.get(checkPathAccessor->get(nextID)->start());
76 m_nextPoints[0].checkpoint = accessor.get(m_nextPt);
81 for (
auto [i, next] : std::views::enumerate(m_nextPoints)) {
82 if (i < m_nextCount) {
83 auto &nextLinked = m_nextPoints[i];
84 auto *nextPoint = nextLinked.checkpoint;
86 next.distance = (nextPoint->m_midpoint - m_midpoint).normalise();
87 next.p0diff = nextPoint->m_left - m_left;
88 next.p1diff = nextPoint->m_right - m_right;
91 next.p0diff = EGG::Vector2f::zero;
92 next.p1diff = EGG::Vector2f::zero;
100 bool betweenSides =
false;
105 for (
size_t i = 0; i < m_nextCount; ++i) {
111 checkSectorAndDistanceRatio(next, p0, p1, distanceRatio);
113 if (result == SectorOccupancy::InsideSector) {
114 return SectorOccupancy::InsideSector;
115 }
else if (result == SectorOccupancy::BetweenSides) {
120 return betweenSides ? SectorOccupancy::BetweenSides : SectorOccupancy::OutsideSector;
131 constexpr f32 REFRESH_PERIOD = 1000.0f / 59.94f;
134 velocity *= 1.0f / REFRESH_PERIOD;
139 EGG::Vector2f displacement = prevPos - m_midpoint + velocity;
142 for (;
static_cast<f32
>(k) < REFRESH_PERIOD && displacement.dot(m_dir) < 0.0f; ++k) {
143 displacement += velocity;
157 constexpr f32 REFRESH_PERIOD = 1000.0f / 59.94f;
160 velocity *= 1.0f / REFRESH_PERIOD;
166 f32 x = (m_midpoint - prevPos).dot(m_dir);
167 f32 y = velocity.dot(m_dir);
170 return y != 0.0f ? x / y : 0.0f;
176 f32 &distanceRatio)
const {
177 if (!checkSector(next, p0, p1)) {
178 return SectorOccupancy::OutsideSector;
181 return checkDistanceRatio(next, p0, p1, distanceRatio) ? SectorOccupancy::InsideSector :
182 SectorOccupancy::BetweenSides;
189 if (-(next.p0diff.y) * p0.x + next.p0diff.x * p0.y < 0.0f) {
193 if (next.p1diff.y * p1.x - next.p1diff.x * p1.y < 0.0f) {
206 f32 d1 = m_dir.dot(p1);
207 f32 d2 = -(next.checkpoint->m_dir.dot(p0));
208 distanceRatio = d1 / (d1 + d2);
209 return distanceRatio >= 0.0f && distanceRatio <= 1.0f;
212MapdataCheckPointAccessor::MapdataCheckPointAccessor(
const MapSectionHeader *header)
214 MapdataAccessorBase::init(
216 parse<u16>(m_sectionHeader->count));
220MapdataCheckPointAccessor::~MapdataCheckPointAccessor() =
default;
226 s16 finishLineCheckpointId = -1;
228 for (
size_t ckptId = 0; ckptId < size(); ckptId++) {
232 if (checkpoint->isFinishLine()) {
233 finishLineCheckpointId = ckptId;
236 lastKcpType = std::max(lastKcpType, checkpoint->checkArea());
239 m_lastKcpType = lastKcpType;
240 m_finishLineCheckpointId = finishLineCheckpointId;
A stream of data stored in memory.
A stream of data, abstracted to allow for continuous seeking.
void init()
Initializes all checkpoint links, and finds the finish line and last key checkpoint.
void initCheckpointLinks(MapdataCheckPointAccessor &accessor, int id)
Calculates m_nextPoints and m_prevPoints from m_nextPt and m_prevPt.
High-level handling for generic system operations, such as input reading, race configuration,...