11BoxColUnit::BoxColUnit() : m_pos(
nullptr), m_radius(0.0f), m_range(0.0f), m_userData(
nullptr) {}
14BoxColUnit::~BoxColUnit() =
default;
21 m_range = radius + maxSpeed;
23 m_flag.
setBit(eBoxColFlag::Active);
24 m_userData = userData;
25 m_xMax = pos->x + m_range;
26 m_xMin = pos->x - m_range;
30void BoxColUnit::makeInactive() {
31 m_flag.
resetBit(eBoxColFlag::Active);
35void BoxColUnit::resize(f32 radius, f32 maxSpeed) {
37 m_range = radius + maxSpeed;
42void BoxColUnit::reinsert() {
43 BoxColManager::Instance()->reinsertUnit(
this);
47void BoxColUnit::search(
const BoxColFlag &flag) {
48 BoxColManager::Instance()->search(
this, flag);
54 constexpr f32 SPATIAL_BOUND = 999999.9f;
56 static const EGG::Vector3f upperBound(SPATIAL_BOUND, SPATIAL_BOUND, SPATIAL_BOUND);
57 static const EGG::Vector3f lowerBound(-SPATIAL_BOUND, -SPATIAL_BOUND, -SPATIAL_BOUND);
72BoxColManager::~BoxColManager() {
75 WARN(
"BoxColManager instance not explicitly handled!");
80void BoxColManager::clear() {
81 m_nextObjectID = MAX_UNIT_COUNT;
82 m_nextDrivableID = MAX_UNIT_COUNT;
84 m_cacheQueryUnit =
nullptr;
85 m_cacheRadius = -1.0f;
96 int activeUnitIdx = 0;
99 if (unit.m_flag.offBit(eBoxColFlag::Active)) {
105 unit.m_xMax = unit.m_pos->x + unit.m_range;
106 unit.m_xMin = unit.m_pos->x - unit.m_range;
107 m_highPoints[unit.m_highPointIdx].z = unit.m_pos->z + unit.m_range;
108 m_lowPoints[unit.m_lowPointIdx].z = unit.m_pos->z - unit.m_range;
115 if (++activeUnitIdx >= m_unitCount) {
125 for (
size_t i = 1; i < static_cast<size_t>(m_unitCount); ++i) {
130 std::swap(upper, lower);
134 ++upperLow.highPoint;
135 --lowerLow.highPoint;
140 u8 &nextMinLowPoint = upper.minLowPoint;
141 if (nextMinLowPoint == lower.lowPoint) {
144 }
while (
m_lowPoints[nextMinLowPoint].highPoint < j);
147 lower.minLowPoint = std::min(lower.minLowPoint, upper.lowPoint);
152 for (
size_t i = 1; i < static_cast<size_t>(m_unitCount); ++i) {
157 std::swap(upper, lower);
165 if (lower.highPoint > upper.highPoint) {
166 int k = upper.highPoint;
168 while (k > lower.highPoint &&
m_highPoints[k].minLowPoint == j - 1) {
172 int k = lower.highPoint;
174 while (k > upper.highPoint &&
m_highPoints[k].minLowPoint == j) {
188ObjectDrivable *BoxColManager::getNextDrivable() {
189 return reinterpret_cast<ObjectDrivable *
>(
getNextImpl(m_nextDrivableID, eBoxColFlag::Drivable));
193void BoxColManager::resetIterators() {
195 iterate(m_nextObjectID, eBoxColFlag::Object);
197 m_nextDrivableID = -1;
198 iterate(m_nextDrivableID, eBoxColFlag::Drivable);
202BoxColUnit *BoxColManager::insertDriver(f32 radius, f32 maxSpeed,
const EGG::Vector3f *pos,
204 BoxColFlag flag = BoxColFlag(eBoxColFlag::Driver);
210 return insert(radius, maxSpeed, pos, flag, kartObject);
214BoxColUnit *BoxColManager::insertObject(f32 radius, f32 maxSpeed,
const EGG::Vector3f *pos,
215 bool alwaysRecalc,
void *userData) {
216 BoxColFlag flag = BoxColFlag(eBoxColFlag::Object);
222 return insert(radius, maxSpeed, pos, flag, userData);
226BoxColUnit *BoxColManager::insertDrivable(f32 radius, f32 maxSpeed,
const EGG::Vector3f *pos,
227 bool alwaysRecalc,
void *userData) {
228 BoxColFlag flag = BoxColFlag(eBoxColFlag::Drivable);
234 return insert(radius, maxSpeed, pos, flag, userData);
238void BoxColManager::reinsertUnit(BoxColUnit *unit) {
239 f32 radius = unit->m_radius;
240 f32 maxSpeed = unit->m_range - radius;
242 BoxColFlag flag = unit->m_flag;
243 void *userData = unit->m_userData;
246 insert(radius, maxSpeed, pos, BoxColFlag(), userData)->m_flag = flag;
250void BoxColManager::search(BoxColUnit *unit,
const BoxColFlag &flag) {
251 searchImpl(unit, flag);
256void BoxColManager::search(f32 radius,
const EGG::Vector3f &pos,
const BoxColFlag &flag) {
257 searchImpl(radius, pos, flag);
262bool BoxColManager::isSphereInSpatialCache(f32 radius,
const EGG::Vector3f &pos,
263 const BoxColFlag &flag)
const {
264 if (m_cacheRadius == -1.0f) {
268 if (!m_cacheFlag.
onAll(flag)) {
272 f32 radiusDiff = m_cacheRadius - radius;
275 return EGG::Mathf::abs(posDiff.x) <= radiusDiff && EGG::Mathf::abs(posDiff.z) <= radiusDiff;
279BoxColManager *BoxColManager::CreateInstance() {
286void BoxColManager::DestroyInstance() {
288 auto *instance = s_instance;
289 s_instance =
nullptr;
293BoxColManager *BoxColManager::Instance() {
299 if (
id == MAX_UNIT_COUNT) {
306 return unit->m_userData;
310void BoxColManager::iterate(s32 &iter,
const BoxColFlag &flag) {
311 while (++iter < m_maxID) {
312 if (
m_units[iter]->m_flag.on(flag)) {
317 iter = MAX_UNIT_COUNT;
321BoxColUnit *BoxColManager::insert(f32 radius, f32 maxSpeed,
const EGG::Vector3f *pos,
322 const BoxColFlag &flag,
void *userData) {
323 if (m_unitCount >=
static_cast<s32
>(MAX_UNIT_COUNT)) {
327 s32 unitID = m_nextUnitID;
329 unit.init(radius, maxSpeed, pos, flag, userData);
331 f32 range = radius + maxSpeed;
332 f32 zHigh = pos->z + range;
333 f32 zLow = pos->z - range;
335 if (m_unitCount == 0) {
351 int highPointIdx = 0;
356 int highSearch = highPointIdx + i;
357 int lowSearch = lowPointIdx + i;
359 if (highSearch <= m_unitCount && zHigh >
m_highPoints[highSearch - 1].z) {
360 highPointIdx = highSearch;
363 if (lowSearch <= m_unitCount && zLow >
m_lowPoints[lowSearch - 1].z) {
364 lowPointIdx = lowSearch;
370 unit.m_highPointIdx = highPointIdx;
371 unit.m_lowPointIdx = lowPointIdx;
374 for (
int i = m_unitCount; i > highPointIdx; --i) {
382 if (high.minLowPoint >= lowPointIdx) {
391 if (highPointIdx == m_unitCount ||
m_highPoints[highPointIdx + 1].minLowPoint > lowPointIdx) {
394 for (
int i = highPointIdx - 1; i >= 0 &&
m_highPoints[i].minLowPoint > lowPointIdx; --i) {
402 for (
int i = m_unitCount; i > lowPointIdx; --i) {
419void BoxColManager::remove(BoxColUnit *&pUnit) {
420 if (!pUnit || pUnit->m_flag.offBit(eBoxColFlag::Active)) {
424 int highPointIdx = pUnit->m_highPointIdx;
425 int lowPointIdx = pUnit->m_lowPointIdx;
428 for (
int i = highPointIdx; i < m_unitCount - 1; ++i) {
434 if (high.minLowPoint > lowPointIdx) {
440 for (
int i = lowPointIdx; i < m_unitCount - 1; ++i) {
447 if (low.highPoint >= highPointIdx) {
451 int minLowPoint =
m_highPoints[low.highPoint].minLowPoint;
453 if (minLowPoint != lowPointIdx) {
457 for (BoxColLowPoint *pLowPoint = &
m_lowPoints[minLowPoint];
458 pLowPoint->highPoint < low.highPoint; ++minLowPoint) {
465 pUnit->makeInactive();
468 m_nextUnitID = nextID;
474void BoxColManager::searchImpl(BoxColUnit *unit,
const BoxColFlag &flag) {
475 if (unit->m_flag.offBit(eBoxColFlag::Active)) {
479 int highPointIdx = unit->m_highPointIdx;
480 int lowPointIdx = unit->m_lowPointIdx;
481 int origLowPointIdx = unit->m_lowPointIdx;
486 f32 xMax = unit->m_xMax;
487 f32 xMin = unit->m_xMin;
490 f32 radius = unit->m_radius;
492 f32 zHigh = pos->z + radius;
493 f32 zLow = pos->z - radius;
494 f32 xHigh = pos->x + radius;
495 f32 xLow = pos->x - radius;
497 int maxIdx = m_unitCount - 1;
500 m_cacheQueryUnit = unit;
501 m_cacheRadius = -1.0f;
504 for (; highPointIdx > 7 &&
m_highPoints[highPointIdx - 8].z >= lowZPos;) {
508 for (; highPointIdx > 0 &&
m_highPoints[highPointIdx - 1].z >= lowZPos;) {
512 for (; lowPointIdx < maxIdx - 7 &&
m_lowPoints[lowPointIdx + 8].z <= highZPos;) {
516 for (; lowPointIdx < maxIdx &&
m_lowPoints[lowPointIdx + 1].z <= highZPos;) {
520 u8 minLowPoint =
m_highPoints[highPointIdx].minLowPoint;
522 for (
int i = lowPointIdx; i >= minLowPoint; --i, --lowPointIdx) {
525 if (low.highPoint >= highPointIdx && lowPointIdx != origLowPointIdx) {
528 if (lowUnit.m_xMax < xMin || lowUnit.m_xMin > xMax) {
536 f32 radius = lowUnit.m_radius;
537 if (lowUnit.m_pos->z + radius < zLow || lowUnit.m_pos->z - radius > zHigh) {
541 if (lowUnit.m_pos->x + radius < xLow || lowUnit.m_pos->x - radius > xHigh) {
547 if (m_maxID == MAX_UNIT_COUNT) {
552 if (lowPointIdx == 0) {
559void BoxColManager::searchImpl(f32 radius,
const EGG::Vector3f &pos,
const BoxColFlag &flag) {
561 int highPointIdx = 0;
563 f32 zHigh = pos.z + radius;
564 f32 zLow = pos.z - radius;
565 f32 xHigh = pos.x + radius;
566 f32 xLow = pos.x - radius;
569 m_cacheQueryUnit =
nullptr;
571 m_cacheRadius = radius;
574 int i = m_unitCount - 1;
576 int highSearch = highPointIdx + i;
577 int lowSearch = lowPointIdx + i;
578 if (highSearch <= m_unitCount && zLow >
m_highPoints[highSearch - 1].z) {
579 highPointIdx = highSearch;
582 if (lowSearch <= m_unitCount && zHigh >=
m_lowPoints[lowSearch].z) {
583 lowPointIdx = lowSearch;
589 u8 minLowPoint =
m_highPoints[highPointIdx].minLowPoint;
591 for (i = lowPointIdx; i >= minLowPoint; --i, --lowPointIdx) {
593 if (low.highPoint >= highPointIdx) {
596 if (unit.m_xMax < xLow || unit.m_xMin > xHigh) {
606 if (m_maxID == MAX_UNIT_COUNT) {
611 if (lowPointIdx == 0) {
617BoxColManager *BoxColManager::s_instance =
nullptr;