8void CourseMap::init() {
9 void *buffer = LoadFile(
"course.kmp");
13 constexpr u32 CANNON_POINT_SIGNATURE = 0x434e5054;
14 constexpr u32 CHECK_PATH_SIGNATURE = 0x434b5048;
15 constexpr u32 CHECK_POINT_SIGNATURE = 0x434b5054;
16 constexpr u32 GEO_OBJ_SIGNATURE = 0x474f424a;
17 constexpr u32 JUGEM_POINT_SIGNATURE = 0x4a475054;
18 constexpr u32 START_POINT_SIGNATURE = 0x4b545054;
19 constexpr u32 POINT_INFO_SIGNATURE = 0x504f5449;
20 constexpr u32 STAGE_INFO_SIGNATURE = 0x53544749;
22 m_startPoint = parseMapdata<MapdataStartPointAccessor>(START_POINT_SIGNATURE);
23 m_checkPath = parseMapdata<MapdataCheckPathAccessor>(CHECK_PATH_SIGNATURE);
24 m_checkPoint = parseMapdata<MapdataCheckPointAccessor>(CHECK_POINT_SIGNATURE);
25 m_geoObj = parseMapdata<MapdataGeoObjAccessor>(GEO_OBJ_SIGNATURE);
26 m_pointInfo = parseMapdata<MapdataPointInfoAccessor>(POINT_INFO_SIGNATURE);
27 m_jugemPoint = parseMapdata<MapdataJugemPointAccessor>(JUGEM_POINT_SIGNATURE);
28 m_cannonPoint = parseMapdata<MapdataCannonPointAccessor>(CANNON_POINT_SIGNATURE);
29 m_stageInfo = parseMapdata<MapdataStageInfoAccessor>(STAGE_INFO_SIGNATURE);
32 constexpr u8 TRANSLATION_MODE_NARROW = 1;
33 if (stageInfo && stageInfo->translationMode() == TRANSLATION_MODE_NARROW) {
34 m_startTmpAngle = 25.0f;
38 m_startTmpAngle = 30.0f;
44 m_startTmp1 = 1200.0f;
48s16 CourseMap::findSector(
const EGG::Vector3f &pos, u16 checkpointIdx, f32 &distanceRatio) {
55 checkpoint->checkSectorAndDistanceRatio(pos, distanceRatio);
56 checkpoint->setSearched();
61 id = checkpoint->id();
67 id = findSectorBetweenSides(pos, checkpoint, distanceRatio);
73 id = findSectorOutsideSector(pos, checkpoint, distanceRatio);
80 return id > -1 ? id : findSectorRegional(pos, checkpoint, distanceRatio);
84s16 CourseMap::findRecursiveSector(
const EGG::Vector3f &pos, s16 depth,
bool searchBackwardsFirst,
86 constexpr s16 MAX_DEPTH = 6;
88 if (depth >= 0 && depth > MAX_DEPTH) {
95 if (!checkpoint->searched()) {
96 completion = checkpoint->checkSectorAndDistanceRatio(pos, distanceRatio);
97 checkpoint->setSearched();
102 return checkpoint->id();
106 if (!searchBackwardsFirst) {
110 distanceRatio < 0.0f) {
111 distanceRatio = 0.0f;
112 return checkpoint->id();
116 if (checkpoint->checkArea() >= 0) {
123 distanceRatio > 0.0f;
126 s16
id = searchNextCheckpoint(pos, depth, checkpoint, distanceRatio, forward,
false);
130 searchPrevCheckpoint(pos, depth, checkpoint, distanceRatio, forward,
true) :
139 distanceRatio > 1.0f) {
140 distanceRatio = 1.0f;
141 return checkpoint->id();
145 if (checkpoint->checkArea() >= 0) {
155 s16
id = searchPrevCheckpoint(pos, depth, checkpoint, distanceRatio, forward,
false);
158 return id == -1 ? searchNextCheckpoint(pos, depth, checkpoint, distanceRatio, forward,
true) :
163u16 CourseMap::getCheckPointEntryOffsetMs(u16 i,
const EGG::Vector3f &pos,
173f32 CourseMap::getCheckPointEntryOffsetExact(u16 i,
const EGG::Vector3f &pos,
191void CourseMap::DestroyInstance() {
193 auto *instance = s_instance;
194 s_instance =
nullptr;
199CourseMap::CourseMap()
200 : m_course(
nullptr), m_startPoint(
nullptr), m_stageInfo(
nullptr), m_startTmpAngle(0.0f),
201 m_startTmp0(0.0f), m_startTmp1(0.0f), m_startTmp2(0.0f), m_startTmp3(0.0f) {}
204CourseMap::~CourseMap() {
206 s_instance =
nullptr;
207 WARN(
"CourseMap instance not explicitly handled!");
217 delete m_cannonPoint;
222 f32 &distanceRatio) {
226 if (distanceRatio > 0.5f) {
228 id = searchNextCheckpoint(pos, 0, checkpoint, distanceRatio,
false,
false);
235 for (
size_t i = 0; i < checkpoint->nextCount(); ++i) {
238 for (
size_t j = 0; j < next->prevCount(); ++j) {
241 if (prev == checkpoint) {
245 id = findRecursiveSector(pos, 1,
true, prev, distanceRatio,
false);
253 for (
size_t i = 0; i < checkpoint->prevCount(); ++i) {
256 for (
size_t j = 0; j < prev->nextCount(); ++j) {
259 if (next == checkpoint) {
263 id = findRecursiveSector(pos, 1,
false, next, distanceRatio,
false);
271 return searchPrevCheckpoint(pos, 0, checkpoint, distanceRatio,
false,
false);
274 id = searchPrevCheckpoint(pos, 0, checkpoint, distanceRatio,
false,
false);
281 for (
size_t i = 0; i < checkpoint->prevCount(); ++i) {
284 for (
size_t j = 0; j < prev->nextCount(); ++j) {
287 if (next == checkpoint) {
291 id = findRecursiveSector(pos, 1,
false, next, distanceRatio,
false);
300 for (
size_t i = 0; i < checkpoint->nextCount(); ++i) {
303 for (
size_t j = 0; j < next->prevCount(); ++j) {
306 if (prev == checkpoint) {
310 id = findRecursiveSector(pos, 1,
true, prev, distanceRatio,
false);
319 return searchNextCheckpoint(pos, 0, checkpoint, distanceRatio,
false,
false);
326 f32 &distanceRatio) {
330 for (
size_t i = 0; i < checkpoint->nextCount(); ++i) {
333 for (
size_t j = 0; j < next->prevCount(); ++j) {
336 if (prev == checkpoint) {
340 id = findRecursiveSector(pos, 1,
true, prev, distanceRatio,
false);
349 for (
size_t i = 0; i < checkpoint->prevCount(); ++i) {
352 for (
size_t j = 0; j < prev->nextCount(); ++j) {
355 if (next == checkpoint) {
359 id = findRecursiveSector(pos, 1,
false, next, distanceRatio,
false);
368 for (
size_t i = 0; i < checkpoint->nextCount(); ++i) {
369 id = findRecursiveSector(pos, 1,
false, checkpoint->nextPoint(i), distanceRatio,
false);
377 for (
size_t i = 0; i < checkpoint->prevCount(); ++i) {
378 id = findRecursiveSector(pos, 1,
true, checkpoint->prevPoint(i), distanceRatio,
false);
390 f32 &distanceRatio) {
394 for (
size_t i = 0; i < checkpoint->nextCount(); ++i) {
395 id = findRecursiveSector(pos, -1,
false, checkpoint->nextPoint(i), distanceRatio,
false);
403 for (
size_t i = 0; i < checkpoint->prevCount(); ++i) {
404 id = findRecursiveSector(pos, -1,
true, checkpoint->prevPoint(i), distanceRatio,
false);
415s16 CourseMap::searchNextCheckpoint(
const EGG::Vector3f &pos, s16 depth,
417 bool useCache)
const {
419 depth = depth >= 0 ? depth + 1 : -1;
421 for (
size_t i = 0; i < checkpoint->nextCount(); ++i) {
424 if (!useCache || !next->searched()) {
425 id = findRecursiveSector(pos, depth,
false, next, completion, playerIsForwards);
437s16 CourseMap::searchPrevCheckpoint(
const EGG::Vector3f &pos, s16 depth,
439 bool useCache)
const {
441 depth = depth >= 0 ? depth + 1 : -1;
443 for (
size_t i = 0; i < checkpoint->prevCount(); ++i) {
446 if (!useCache || !prev->searched()) {
447 id = findRecursiveSector(pos, depth,
true, prev, completion, playerIsForwards);
459void CourseMap::clearSectorChecked() {
460 for (
size_t i = 0; i < m_checkPoint->size(); ++i) {
461 getCheckPoint(i)->clearSearched();
466void *CourseMap::LoadFile(
const char *filename) {
467 return ResourceManager::Instance()->getFile(filename,
nullptr, ArchiveId::Course);
470CourseMap *CourseMap::s_instance =
nullptr;