CCollisionPrimitive: Simplify the InitAdd* member functions

We can extract the duplicated behavior out into its own member function
and then reuse it in order to deduplicate repeated behavior.

This allows simplifying the member functions in a manner that allows
declaring most of the variables const. The lack of mutable behavior
within these functions makes it much nicer to read.
This commit is contained in:
Lioncash 2019-10-11 23:19:48 -04:00
parent c4ecf972f5
commit 5f4aba60e1
2 changed files with 47 additions and 60 deletions

View File

@ -1,8 +1,14 @@
#include "CCollisionPrimitive.hpp" #include "Runtime/Collision/CCollisionPrimitive.hpp"
#include "CInternalRayCastStructure.hpp"
#include "CMaterialFilter.hpp" #include <algorithm>
#include "InternalColliders.hpp" #include <climits>
#include "CCollisionInfoList.hpp" #include <cstring>
#include <iterator>
#include "Runtime/Collision/CCollisionInfoList.hpp"
#include "Runtime/Collision/CInternalRayCastStructure.hpp"
#include "Runtime/Collision/CMaterialFilter.hpp"
#include "Runtime/Collision/InternalColliders.hpp"
namespace urde { namespace urde {
s32 CCollisionPrimitive::sNumTypes = 0; s32 CCollisionPrimitive::sNumTypes = 0;
@ -29,6 +35,11 @@ CRayCastResult CCollisionPrimitive::CastRay(const zeus::CVector3f& start, const
return CastRayInternal(CInternalRayCastStructure(start, dir, length, xf, filter)); return CastRayInternal(CInternalRayCastStructure(start, dir, length, xf, filter));
} }
std::vector<CCollisionPrimitive::Type>::const_iterator CCollisionPrimitive::FindCollisionType(const char* name) {
return std::find_if(sCollisionTypeList->cbegin(), sCollisionTypeList->cend(),
[name](const auto& type) { return std::strcmp(name, type.GetInfo()) == 0; });
}
bool CCollisionPrimitive::InternalCollide(const CInternalCollisionStructure& collision, CCollisionInfoList& list) { bool CCollisionPrimitive::InternalCollide(const CInternalCollisionStructure& collision, CCollisionInfoList& list) {
u32 idx0 = collision.GetLeft().GetPrim().GetTableIndex(); u32 idx0 = collision.GetLeft().GetPrim().GetTableIndex();
u32 idx1 = collision.GetRight().GetPrim().GetTableIndex(); u32 idx1 = collision.GetRight().GetPrim().GetTableIndex();
@ -176,28 +187,19 @@ void CCollisionPrimitive::InitBeginColliders() {
InternalColliders::AddColliders(); InternalColliders::AddColliders();
} }
void CCollisionPrimitive::InitAddBooleanCollider(const CCollisionPrimitive::BooleanComparison& cmp) { void CCollisionPrimitive::InitAddBooleanCollider(const BooleanComparison& cmp) {
int idx0 = -1; const auto iter1 = FindCollisionType(cmp.GetType1());
for (int i = 0; i < sCollisionTypeList->size(); ++i) { const auto iter2 = FindCollisionType(cmp.GetType2());
if (!strcmp(cmp.GetType1(), (*sCollisionTypeList)[i].GetInfo())) { const auto index1 = std::distance(sCollisionTypeList->cbegin(), iter1);
idx0 = i; const auto index2 = std::distance(sCollisionTypeList->cbegin(), iter2);
break; const bool hasReachedEnd = iter1 == sCollisionTypeList->cend() || iter2 == sCollisionTypeList->cend();
}
}
int idx1 = -1; if (index1 >= sNumTypes || index2 >= sNumTypes || hasReachedEnd) {
for (int i = 0; i < sCollisionTypeList->size(); ++i) {
if (!strcmp(cmp.GetType2(), (*sCollisionTypeList)[i].GetInfo())) {
idx1 = i;
break;
}
}
if (idx0 < 0 || idx1 < 0 || idx0 >= sNumTypes || idx1 >= sNumTypes)
return; return;
}
BooleanComparisonFunc& funcOut = BooleanComparisonFunc& funcOut =
(idx0 == -1 || idx1 == -1) ? sNullBooleanCollider : (*sTableOfBooleanCollidables)[idx1 * sNumTypes + idx0]; hasReachedEnd ? sNullBooleanCollider : (*sTableOfBooleanCollidables)[index2 * sNumTypes + index1];
funcOut = cmp.GetCollider(); funcOut = cmp.GetCollider();
} }
@ -205,28 +207,19 @@ void CCollisionPrimitive::InitAddBooleanCollider(BooleanComparisonFunc cmp, cons
InitAddBooleanCollider({cmp, a, b}); InitAddBooleanCollider({cmp, a, b});
} }
void CCollisionPrimitive::InitAddMovingCollider(const CCollisionPrimitive::MovingComparison& cmp) { void CCollisionPrimitive::InitAddMovingCollider(const MovingComparison& cmp) {
int idx0 = -1; const auto iter1 = FindCollisionType(cmp.GetType1());
for (int i = 0; i < sCollisionTypeList->size(); ++i) { const auto iter2 = FindCollisionType(cmp.GetType2());
if (!strcmp(cmp.GetType1(), (*sCollisionTypeList)[i].GetInfo())) { const auto index1 = std::distance(sCollisionTypeList->cbegin(), iter1);
idx0 = i; const auto index2 = std::distance(sCollisionTypeList->cbegin(), iter2);
break; const bool hasReachedEnd = iter1 == sCollisionTypeList->cend() || iter2 == sCollisionTypeList->cend();
}
}
int idx1 = -1; if (index1 >= sNumTypes || index2 >= sNumTypes || hasReachedEnd) {
for (int i = 0; i < sCollisionTypeList->size(); ++i) {
if (!strcmp(cmp.GetType2(), (*sCollisionTypeList)[i].GetInfo())) {
idx1 = i;
break;
}
}
if (idx0 < 0 || idx1 < 0 || idx0 >= sNumTypes || idx1 >= sNumTypes)
return; return;
}
MovingComparisonFunc& funcOut = MovingComparisonFunc& funcOut =
(idx0 == -1 || idx1 == -1) ? sNullMovingCollider : (*sTableOfMovingCollidables)[idx1 * sNumTypes + idx0]; hasReachedEnd ? sNullMovingCollider : (*sTableOfMovingCollidables)[index2 * sNumTypes + index1];
funcOut = cmp.GetCollider(); funcOut = cmp.GetCollider();
} }
@ -234,28 +227,18 @@ void CCollisionPrimitive::InitAddMovingCollider(MovingComparisonFunc cmp, const
InitAddMovingCollider({cmp, a, b}); InitAddMovingCollider({cmp, a, b});
} }
void CCollisionPrimitive::InitAddCollider(const CCollisionPrimitive::Comparison& cmp) { void CCollisionPrimitive::InitAddCollider(const Comparison& cmp) {
int idx0 = -1; const auto iter1 = FindCollisionType(cmp.GetType1());
for (int i = 0; i < sCollisionTypeList->size(); ++i) { const auto iter2 = FindCollisionType(cmp.GetType2());
if (!strcmp(cmp.GetType1(), (*sCollisionTypeList)[i].GetInfo())) { const auto index1 = std::distance(sCollisionTypeList->cbegin(), iter1);
idx0 = i; const auto index2 = std::distance(sCollisionTypeList->cbegin(), iter2);
break; const bool hasReachedEnd = iter1 == sCollisionTypeList->cend() || iter2 == sCollisionTypeList->cend();
}
}
int idx1 = -1; if (index1 >= sNumTypes || index2 >= sNumTypes || hasReachedEnd) {
for (int i = 0; i < sCollisionTypeList->size(); ++i) {
if (!strcmp(cmp.GetType2(), (*sCollisionTypeList)[i].GetInfo())) {
idx1 = i;
break;
}
}
if (idx0 < 0 || idx1 < 0 || idx0 >= sNumTypes || idx1 >= sNumTypes)
return; return;
}
ComparisonFunc& funcOut = ComparisonFunc& funcOut = hasReachedEnd ? sNullCollider : (*sTableOfCollidables)[index2 * sNumTypes + index1];
(idx0 == -1 || idx1 == -1) ? sNullCollider : (*sTableOfCollidables)[idx1 * sNumTypes + idx0];
funcOut = cmp.GetCollider(); funcOut = cmp.GetCollider();
} }

View File

@ -122,6 +122,10 @@ private:
static BooleanComparisonFunc sNullBooleanCollider; static BooleanComparisonFunc sNullBooleanCollider;
static MovingComparisonFunc sNullMovingCollider; static MovingComparisonFunc sNullMovingCollider;
// Attempts to locate an entry within the collision type list that matches the supplied name.
// Returns the end iterator in the event of no matches.
static std::vector<Type>::const_iterator FindCollisionType(const char* name);
static bool InternalCollide(const CInternalCollisionStructure& collision, CCollisionInfoList& list); static bool InternalCollide(const CInternalCollisionStructure& collision, CCollisionInfoList& list);
static bool InternalCollideBoolean(const CInternalCollisionStructure& collision); static bool InternalCollideBoolean(const CInternalCollisionStructure& collision);
static bool InternalCollideMoving(const CInternalCollisionStructure& collision, const zeus::CVector3f& dir, static bool InternalCollideMoving(const CInternalCollisionStructure& collision, const zeus::CVector3f& dir,