2
0
mirror of https://github.com/AxioDL/metaforce.git synced 2025-10-26 00:50:24 +00:00

Merge branch 'master' of ssh+git://git.axiodl.com:6431/AxioDL/urde

This commit is contained in:
Phillip Stephens 2020-04-17 05:54:17 -07:00
commit acf513c6bf
Signed by: Antidote
GPG Key ID: F8BEE4C83DACA60D
6 changed files with 301 additions and 194 deletions

View File

@ -19,9 +19,9 @@ void CFlameWarp::ModifyParticles(std::vector<CParticle>& particles) {
float maxTransp = 0.f; float maxTransp = 0.f;
u8 idx = 0; u8 idx = 0;
for (CParticle& particle : particles) { for (CParticle& particle : particles) {
float transp = 1.f - particle.x34_color.a(); const float transp = 1.f - particle.x34_color.a();
if (transp > maxTransp) { if (transp > maxTransp) {
float distSq = (particle.x4_pos - x74_warpPoint).magSquared(); const float distSq = (particle.x4_pos - x74_warpPoint).magSquared();
if (distSq > x8c_maxDistSq && distSq < x98_maxInfluenceDistSq) { if (distSq > x8c_maxDistSq && distSq < x98_maxInfluenceDistSq) {
x8c_maxDistSq = distSq; x8c_maxDistSq = distSq;
maxTransp = transp; maxTransp = transp;
@ -29,28 +29,30 @@ void CFlameWarp::ModifyParticles(std::vector<CParticle>& particles) {
} }
} }
if (particle.x2c_lineLengthOrSize < x90_minSize) if (particle.x2c_lineLengthOrSize < x90_minSize) {
x90_minSize = particle.x2c_lineLengthOrSize; x90_minSize = particle.x2c_lineLengthOrSize;
if (particle.x2c_lineLengthOrSize > x94_maxSize) }
if (particle.x2c_lineLengthOrSize > x94_maxSize) {
x94_maxSize = particle.x2c_lineLengthOrSize; x94_maxSize = particle.x2c_lineLengthOrSize;
}
vec.emplace_back(transp, idx); vec.emplace_back(transp, idx);
if (xa0_25_collisionWarp) { if (xa0_25_collisionWarp) {
zeus::CVector3f delta = particle.x4_pos - particle.x10_prevPos; const zeus::CVector3f delta = particle.x4_pos - particle.x10_prevPos;
if (delta.magSquared() >= 0.0011920929f) { if (delta.magSquared() >= 0.0011920929f) {
zeus::CVector3f deltaNorm = delta.normalized(); const zeus::CVector3f deltaNorm = delta.normalized();
zeus::CVector3f behindPos = particle.x10_prevPos - deltaNorm * 5.f; const zeus::CVector3f behindPos = particle.x10_prevPos - deltaNorm * 5.f;
zeus::CVector3f fullDelta = particle.x4_pos - behindPos; const zeus::CVector3f fullDelta = particle.x4_pos - behindPos;
CRayCastResult result = x9c_stateMgr->RayStaticIntersection( const CRayCastResult result = x9c_stateMgr->RayStaticIntersection(
behindPos, deltaNorm, fullDelta.magnitude(), behindPos, deltaNorm, fullDelta.magnitude(),
CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {EMaterialTypes::ProjectilePassthrough})); CMaterialFilter::MakeIncludeExclude({EMaterialTypes::Solid}, {EMaterialTypes::ProjectilePassthrough}));
if (result.IsValid()) { if (result.IsValid()) {
float dist = result.GetPlane().pointToPlaneDist(particle.x4_pos); const float dist = result.GetPlane().pointToPlaneDist(particle.x4_pos);
if (dist <= 0.f) { if (dist <= 0.f) {
particle.x4_pos -= result.GetPlane().normal() * dist; particle.x4_pos -= result.GetPlane().normal() * dist;
if (result.GetPlane().normal().dot(particle.x1c_vel) < 0.f) { if (result.GetPlane().normal().dot(particle.x1c_vel) < 0.f) {
zeus::CVector3f prevStepPos = particle.x4_pos - particle.x1c_vel; const zeus::CVector3f prevStepPos = particle.x4_pos - particle.x1c_vel;
particle.x4_pos += particle.x4_pos +=
(-result.GetPlane().pointToPlaneDist(prevStepPos) / particle.x1c_vel.dot(result.GetPlane().normal()) - (-result.GetPlane().pointToPlaneDist(prevStepPos) / particle.x1c_vel.dot(result.GetPlane().normal()) -
1.f) * 1.f) *
@ -91,8 +93,9 @@ void CFlameWarp::ResetPosition(const zeus::CVector3f& pos) {
zeus::CAABox CFlameWarp::CalculateBounds() const { zeus::CAABox CFlameWarp::CalculateBounds() const {
zeus::CAABox ret; zeus::CAABox ret;
for (const auto& v : x4_collisionPoints) for (const auto& v : x4_collisionPoints) {
ret.accumulateBounds(v); ret.accumulateBounds(v);
}
return ret; return ret;
} }

View File

@ -63,14 +63,18 @@ CFishCloud::CFishCloud(TUniqueId uid, bool active, std::string_view name, const
x1b0_models.emplace_back(std::make_unique<CModelData>(aRes)); x1b0_models.emplace_back(std::make_unique<CModelData>(aRes));
x250_27_validModel = true; x250_27_validModel = true;
} }
if (part1.IsValid()) if (part1.IsValid()) {
x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part1})); x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part1}));
if (part2.IsValid()) }
if (part2.IsValid()) {
x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part2})); x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part2}));
if (part3.IsValid()) }
if (part3.IsValid()) {
x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part3})); x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part3}));
if (part4.IsValid()) }
if (part4.IsValid()) {
x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part4})); x1c4_particleDescs.push_back(g_SimplePool->GetObj({FOURCC('PART'), part4}));
}
for (const auto& p : x1c4_particleDescs) { for (const auto& p : x1c4_particleDescs) {
x1f8_particleGens.emplace_back(std::make_unique<CElementGen>(p)); x1f8_particleGens.emplace_back(std::make_unique<CElementGen>(p));
x1f8_particleGens.back()->SetParticleEmission(false); x1f8_particleGens.back()->SetParticleEmission(false);
@ -79,7 +83,7 @@ CFishCloud::CFishCloud(TUniqueId uid, bool active, std::string_view name, const
x21c_deathParticleCounts.push_back(partCount2); x21c_deathParticleCounts.push_back(partCount2);
x21c_deathParticleCounts.push_back(partCount3); x21c_deathParticleCounts.push_back(partCount3);
x21c_deathParticleCounts.push_back(partCount4); x21c_deathParticleCounts.push_back(partCount4);
zeus::CAABox aabb = GetBoundingBox(); const zeus::CAABox aabb = GetBoundingBox();
x238_partitionPitch = (aabb.max - aabb.min) / 7.f; x238_partitionPitch = (aabb.max - aabb.min) / 7.f;
x244_ooPartitionPitch = 1.f / x238_partitionPitch; x244_ooPartitionPitch = 1.f / x238_partitionPitch;
} }
@ -87,17 +91,18 @@ CFishCloud::CFishCloud(TUniqueId uid, bool active, std::string_view name, const
void CFishCloud::Accept(IVisitor& visitor) { visitor.Visit(this); } void CFishCloud::Accept(IVisitor& visitor) { visitor.Visit(this); }
void CFishCloud::UpdateParticles(float dt) { void CFishCloud::UpdateParticles(float dt) {
for (auto& p : x1f8_particleGens) for (auto& p : x1f8_particleGens) {
p->Update(dt); p->Update(dt);
}
} }
void CFishCloud::UpdatePartitionList() { void CFishCloud::UpdatePartitionList() {
xf8_boidPartitionLists.clear(); xf8_boidPartitionLists.clear();
xf8_boidPartitionLists.resize(xf8_boidPartitionLists.capacity()); xf8_boidPartitionLists.resize(xf8_boidPartitionLists.capacity());
auto aabb = GetBoundingBox(); const auto aabb = GetBoundingBox();
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
zeus::CVector3f idxs = (b.x0_pos - aabb.min) * x244_ooPartitionPitch; const zeus::CVector3f idxs = (b.x0_pos - aabb.min) * x244_ooPartitionPitch;
int idx = int(idxs.x()) + int(idxs.y()) * 7 + int(idxs.z()) * 49; const int idx = int(idxs.x()) + int(idxs.y()) * 7 + int(idxs.z()) * 49;
if (idx >= 0 && idx < 343) { if (idx >= 0 && idx < 343) {
b.x1c_next = xf8_boidPartitionLists[idx]; b.x1c_next = xf8_boidPartitionLists[idx];
xf8_boidPartitionLists[idx] = &b; xf8_boidPartitionLists[idx] = &b;
@ -106,39 +111,41 @@ void CFishCloud::UpdatePartitionList() {
} }
bool CFishCloud::PointInBox(const zeus::CAABox& aabb, const zeus::CVector3f& point) const { bool CFishCloud::PointInBox(const zeus::CAABox& aabb, const zeus::CVector3f& point) const {
if (!x250_25_worldSpace) if (!x250_25_worldSpace) {
return aabb.pointInside(point); return aabb.pointInside(point);
}
return GetUntransformedBoundingBox().pointInside(GetTransform().transposeRotate(point - GetTranslation())); return GetUntransformedBoundingBox().pointInside(GetTransform().transposeRotate(point - GetTranslation()));
} }
zeus::CPlane CFishCloud::FindClosestPlane(const zeus::CAABox& aabb, const zeus::CVector3f& point) const { zeus::CPlane CFishCloud::FindClosestPlane(const zeus::CAABox& aabb, const zeus::CVector3f& point) const {
if (!x250_25_worldSpace) { if (!x250_25_worldSpace) {
float minDist = FLT_MAX; float minDist = FLT_MAX;
zeus::CAABox::EBoxFaceId minFace = zeus::CAABox::EBoxFaceId::YMin; auto minFace = zeus::CAABox::EBoxFaceId::YMin;
for (int i = 0; i < 6; ++i) { for (int i = 0; i < 6; ++i) {
auto tri = aabb.getTri(zeus::CAABox::EBoxFaceId(i), 0); const auto tri = aabb.getTri(zeus::CAABox::EBoxFaceId(i), 0);
float dist = zeus::CPlane(tri.x10_verts[0], tri.x10_verts[2], tri.x10_verts[1]).pointToPlaneDist(point); const float dist = zeus::CPlane(tri.x10_verts[0], tri.x10_verts[2], tri.x10_verts[1]).pointToPlaneDist(point);
if (dist >= 0.f && dist < minDist) { if (dist >= 0.f && dist < minDist) {
minDist = dist; minDist = dist;
minFace = zeus::CAABox::EBoxFaceId(i); minFace = zeus::CAABox::EBoxFaceId(i);
} }
} }
auto tri = aabb.getTri(minFace, 0); const auto tri = aabb.getTri(minFace, 0);
return zeus::CPlane(tri.x10_verts[0], tri.x10_verts[2], tri.x10_verts[1]); return zeus::CPlane(tri.x10_verts[0], tri.x10_verts[2], tri.x10_verts[1]);
} else { } else {
auto unPoint = GetTransform().transposeRotate(point - GetTranslation()); const auto unPoint = GetTransform().transposeRotate(point - GetTranslation());
auto unAabb = GetUntransformedBoundingBox(); const auto unAabb = GetUntransformedBoundingBox();
float minDist = FLT_MAX; float minDist = FLT_MAX;
zeus::CAABox::EBoxFaceId minFace = zeus::CAABox::EBoxFaceId::YMin; auto minFace = zeus::CAABox::EBoxFaceId::YMin;
for (int i = 0; i < 6; ++i) { for (int i = 0; i < 6; ++i) {
auto tri = unAabb.getTri(zeus::CAABox::EBoxFaceId(i), 0); const auto tri = unAabb.getTri(zeus::CAABox::EBoxFaceId(i), 0);
float dist = zeus::CPlane(tri.x10_verts[0], tri.x10_verts[2], tri.x10_verts[1]).pointToPlaneDist(unPoint); const float dist = zeus::CPlane(tri.x10_verts[0], tri.x10_verts[2], tri.x10_verts[1]).pointToPlaneDist(unPoint);
if (dist >= 0.f && dist < minDist) { if (dist >= 0.f && dist < minDist) {
minDist = dist; minDist = dist;
minFace = zeus::CAABox::EBoxFaceId(i); minFace = zeus::CAABox::EBoxFaceId(i);
} }
} }
auto tri = unAabb.getTri(minFace, 0);
const auto tri = unAabb.getTri(minFace, 0);
return zeus::CPlane(GetTransform() * tri.x10_verts[0], return zeus::CPlane(GetTransform() * tri.x10_verts[0],
GetTransform() * tri.x10_verts[2], GetTransform() * tri.x10_verts[2],
GetTransform() * tri.x10_verts[1]); GetTransform() * tri.x10_verts[1]);
@ -146,23 +153,28 @@ zeus::CPlane CFishCloud::FindClosestPlane(const zeus::CAABox& aabb, const zeus::
} }
CFishCloud::CBoid* CFishCloud::GetListAt(const zeus::CVector3f& pos) { CFishCloud::CBoid* CFishCloud::GetListAt(const zeus::CVector3f& pos) {
zeus::CAABox aabb = GetBoundingBox(); const zeus::CAABox aabb = GetBoundingBox();
zeus::CVector3f ints = (pos - aabb.min) * x244_ooPartitionPitch; const zeus::CVector3f ints = (pos - aabb.min) * x244_ooPartitionPitch;
int idx = int(ints.x()) + int(ints.y()) * 7 + int(ints.z()) * 49; const int idx = int(ints.x()) + int(ints.y()) * 7 + int(ints.z()) * 49;
if (idx < 0 || idx >= 343)
if (idx < 0 || idx >= 343) {
return nullptr; return nullptr;
}
return xf8_boidPartitionLists[idx]; return xf8_boidPartitionLists[idx];
} }
void CFishCloud::BuildBoidNearList(const zeus::CVector3f& pos, float radius, void CFishCloud::BuildBoidNearList(const zeus::CVector3f& pos, float radius,
rstl::reserved_vector<CBoid*, 25>& nearList) { rstl::reserved_vector<CBoid*, 25>& nearList) {
float radiusSq = radius * radius; const float radiusSq = radius * radius;
CBoid* b = GetListAt(pos); CBoid* b = GetListAt(pos);
while (b && nearList.size() < 25) {
while (b != nullptr && nearList.size() < 25) {
if (b->x20_active) { if (b->x20_active) {
float distSq = (b->GetTranslation() - pos).magSquared(); const float distSq = (b->GetTranslation() - pos).magSquared();
if (distSq != 0.f && distSq < radiusSq) if (distSq != 0.f && distSq < radiusSq) {
nearList.push_back(b); nearList.push_back(b);
}
} }
b = b->x1c_next; b = b->x1c_next;
} }
@ -170,45 +182,54 @@ void CFishCloud::BuildBoidNearList(const zeus::CVector3f& pos, float radius,
void CFishCloud::BuildBoidNearPartitionList(const zeus::CVector3f& pos, float radius, void CFishCloud::BuildBoidNearPartitionList(const zeus::CVector3f& pos, float radius,
rstl::reserved_vector<CBoid*, 25>& nearList) { rstl::reserved_vector<CBoid*, 25>& nearList) {
float radiusSq = radius * radius; const float radiusSq = radius * radius;
zeus::CAABox aabb = GetBoundingBox(); const zeus::CAABox aabb = GetBoundingBox();
float x = std::max(radius * x244_ooPartitionPitch.x(), float(x238_partitionPitch.x())); const float x = std::max(radius * x244_ooPartitionPitch.x(), float(x238_partitionPitch.x()));
float y = std::max(radius * x244_ooPartitionPitch.y(), float(x238_partitionPitch.y())); const float y = std::max(radius * x244_ooPartitionPitch.y(), float(x238_partitionPitch.y()));
float z = std::max(radius * x244_ooPartitionPitch.z(), float(x238_partitionPitch.z())); const float z = std::max(radius * x244_ooPartitionPitch.z(), float(x238_partitionPitch.z()));
float nx = 0.01f - x; const float nx = 0.01f - x;
float ny = 0.01f - y; const float ny = 0.01f - y;
float nz = 0.01f - z; const float nz = 0.01f - z;
for (float lnx = nx; lnx < x; lnx += x238_partitionPitch.x()) { for (float lnx = nx; lnx < x; lnx += x238_partitionPitch.x()) {
float cx = lnx + pos.x(); const float cx = lnx + pos.x();
if (cx < aabb.min.x()) if (cx < aabb.min.x()) {
continue; continue;
if (cx >= aabb.max.x()) }
if (cx >= aabb.max.x()) {
break; break;
}
for (float lny = ny; lny < y; lny += x238_partitionPitch.y()) { for (float lny = ny; lny < y; lny += x238_partitionPitch.y()) {
float cy = lny + pos.y(); const float cy = lny + pos.y();
if (cy < aabb.min.y()) if (cy < aabb.min.y()) {
continue; continue;
if (cy >= aabb.max.y()) }
if (cy >= aabb.max.y()) {
break; break;
}
for (float lnz = nz; lnz < z; lnz += x238_partitionPitch.z()) { for (float lnz = nz; lnz < z; lnz += x238_partitionPitch.z()) {
float cz = lnz + pos.z(); const float cz = lnz + pos.z();
if (cz < aabb.min.z()) if (cz < aabb.min.z()) {
continue; continue;
if (cz >= aabb.max.z()) }
if (cz >= aabb.max.z()) {
break; break;
zeus::CVector3f ints = (zeus::CVector3f(cx, cy, cz) - aabb.min) * x244_ooPartitionPitch; }
int idx = int(ints.x()) + int(ints.y()) * 7 + int(ints.z()) * 49; const zeus::CVector3f ints = (zeus::CVector3f(cx, cy, cz) - aabb.min) * x244_ooPartitionPitch;
if (idx < 0) const int idx = int(ints.x()) + int(ints.y()) * 7 + int(ints.z()) * 49;
if (idx < 0) {
continue; continue;
}
if (idx < 343) { if (idx < 343) {
CBoid* boid = xf8_boidPartitionLists[idx]; CBoid* boid = xf8_boidPartitionLists[idx];
while (boid) { while (boid != nullptr) {
if (boid->x20_active) { if (boid->x20_active) {
float distSq = (boid->x0_pos - pos).magSquared(); const float distSq = (boid->x0_pos - pos).magSquared();
if (distSq != 0.f && distSq < radiusSq) { if (distSq != 0.f && distSq < radiusSq) {
nearList.push_back(boid); nearList.push_back(boid);
if (nearList.size() == 25) if (nearList.size() == 25) {
return; return;
}
} }
} }
boid = boid->x1c_next; boid = boid->x1c_next;
@ -220,7 +241,7 @@ void CFishCloud::BuildBoidNearPartitionList(const zeus::CVector3f& pos, float ra
} }
void CFishCloud::PlaceBoid(CStateManager& mgr, CBoid& boid, const zeus::CAABox& aabb) const { void CFishCloud::PlaceBoid(CStateManager& mgr, CBoid& boid, const zeus::CAABox& aabb) const {
auto plane = FindClosestPlane(aabb, boid.x0_pos); const auto plane = FindClosestPlane(aabb, boid.x0_pos);
boid.x0_pos -= plane.pointToPlaneDist(boid.x0_pos) * plane.normal() + 0.0001f * plane.normal(); boid.x0_pos -= plane.pointToPlaneDist(boid.x0_pos) * plane.normal() + 0.0001f * plane.normal();
boid.xc_vel.y() = mgr.GetActiveRandom()->Float() - 0.5f; boid.xc_vel.y() = mgr.GetActiveRandom()->Float() - 0.5f;
boid.xc_vel.x() = mgr.GetActiveRandom()->Float() - 0.5f; boid.xc_vel.x() = mgr.GetActiveRandom()->Float() - 0.5f;
@ -233,7 +254,7 @@ void CFishCloud::PlaceBoid(CStateManager& mgr, CBoid& boid, const zeus::CAABox&
} }
} else { } else {
if (!PointInBox(aabb, boid.x0_pos)) { if (!PointInBox(aabb, boid.x0_pos)) {
auto unAabb = GetUntransformedBoundingBox(); const auto unAabb = GetUntransformedBoundingBox();
boid.x0_pos.z() = mgr.GetActiveRandom()->Float() * (unAabb.max.z() - unAabb.min.z()) + unAabb.min.z(); boid.x0_pos.z() = mgr.GetActiveRandom()->Float() * (unAabb.max.z() - unAabb.min.z()) + unAabb.min.z();
boid.x0_pos.y() = mgr.GetActiveRandom()->Float() * (unAabb.max.y() - unAabb.min.y()) + unAabb.min.y(); boid.x0_pos.y() = mgr.GetActiveRandom()->Float() * (unAabb.max.y() - unAabb.min.y()) + unAabb.min.y();
boid.x0_pos.x() = mgr.GetActiveRandom()->Float() * (unAabb.max.x() - unAabb.min.x()) + unAabb.min.x(); boid.x0_pos.x() = mgr.GetActiveRandom()->Float() * (unAabb.max.x() - unAabb.min.x()) + unAabb.min.x();
@ -243,57 +264,75 @@ void CFishCloud::PlaceBoid(CStateManager& mgr, CBoid& boid, const zeus::CAABox&
} }
void CFishCloud::ApplySeparation(CBoid& boid, const rstl::reserved_vector<CBoid*, 25>& nearList) const { void CFishCloud::ApplySeparation(CBoid& boid, const rstl::reserved_vector<CBoid*, 25>& nearList) const {
if (nearList.empty()) if (nearList.empty()) {
return; return;
}
float minDist = FLT_MAX; float minDist = FLT_MAX;
zeus::CVector3f pos; zeus::CVector3f pos;
for (CBoid* b : nearList) { for (const CBoid* b : nearList) {
float dist = (boid.GetTranslation() - b->GetTranslation()).magSquared(); const float dist = (boid.GetTranslation() - b->GetTranslation()).magSquared();
if (dist < minDist) { if (dist < minDist) {
minDist = dist; minDist = dist;
pos = b->GetTranslation(); pos = b->GetTranslation();
} }
} }
ApplySeparation(boid, pos, x138_separationRadius, x144_separationMagnitude); ApplySeparation(boid, pos, x138_separationRadius, x144_separationMagnitude);
} }
void CFishCloud::ApplySeparation(CBoid& boid, const zeus::CVector3f& separateFrom, void CFishCloud::ApplySeparation(CBoid& boid, const zeus::CVector3f& separateFrom,
float separationRadius, float separationMagnitude) const { float separationRadius, float separationMagnitude) const {
zeus::CVector3f delta = boid.GetTranslation() - separateFrom; const zeus::CVector3f delta = boid.GetTranslation() - separateFrom;
if (delta.canBeNormalized()) { if (!delta.canBeNormalized()) {
float deltaDistSq = delta.magSquared(); return;
float capDeltaDistSq = separationRadius * separationRadius;
if (deltaDistSq < capDeltaDistSq)
boid.xc_vel += (1.f - deltaDistSq / capDeltaDistSq) * delta.normalized() * separationMagnitude;
} }
const float deltaDistSq = delta.magSquared();
const float capDeltaDistSq = separationRadius * separationRadius;
if (deltaDistSq >= capDeltaDistSq) {
return;
}
boid.xc_vel += (1.f - deltaDistSq / capDeltaDistSq) * delta.normalized() * separationMagnitude;
} }
void CFishCloud::ApplyCohesion(CBoid& boid, const rstl::reserved_vector<CBoid*, 25>& nearList) const { void CFishCloud::ApplyCohesion(CBoid& boid, const rstl::reserved_vector<CBoid*, 25>& nearList) const {
if (nearList.empty()) if (nearList.empty()) {
return; return;
}
zeus::CVector3f avg; zeus::CVector3f avg;
for (CBoid* b : nearList) for (const CBoid* b : nearList) {
avg += b->GetTranslation(); avg += b->GetTranslation();
}
avg = avg / float(nearList.size()); avg = avg / float(nearList.size());
ApplyCohesion(boid, avg, x138_separationRadius, x13c_cohesionMagnitude); ApplyCohesion(boid, avg, x138_separationRadius, x13c_cohesionMagnitude);
} }
void CFishCloud::ApplyCohesion(CBoid& boid, const zeus::CVector3f& cohesionFrom, void CFishCloud::ApplyCohesion(CBoid& boid, const zeus::CVector3f& cohesionFrom,
float cohesionRadius, float cohesionMagnitude) const { float cohesionRadius, float cohesionMagnitude) const {
zeus::CVector3f delta = cohesionFrom - boid.GetTranslation(); const zeus::CVector3f delta = cohesionFrom - boid.GetTranslation();
if (delta.canBeNormalized()) { if (!delta.canBeNormalized()) {
float distSq = delta.magSquared(); return;
float capDistSq = cohesionRadius * cohesionRadius;
boid.xc_vel += ((distSq > capDistSq) ? 1.f : distSq / capDistSq) * delta.normalized() * cohesionMagnitude;
} }
const float distSq = delta.magSquared();
const float capDistSq = cohesionRadius * cohesionRadius;
boid.xc_vel += ((distSq > capDistSq) ? 1.f : distSq / capDistSq) * delta.normalized() * cohesionMagnitude;
} }
void CFishCloud::ApplyAlignment(CBoid& boid, const rstl::reserved_vector<CBoid*, 25>& nearList) const { void CFishCloud::ApplyAlignment(CBoid& boid, const rstl::reserved_vector<CBoid*, 25>& nearList) const {
if (nearList.empty()) if (nearList.empty()) {
return; return;
}
zeus::CVector3f avg; zeus::CVector3f avg;
for (CBoid* b : nearList) for (const CBoid* b : nearList) {
avg += b->xc_vel; avg += b->xc_vel;
}
avg = avg / float(nearList.size()); avg = avg / float(nearList.size());
boid.xc_vel += zeus::CVector3f::getAngleDiff(boid.xc_vel, avg) / M_PIF * boid.xc_vel += zeus::CVector3f::getAngleDiff(boid.xc_vel, avg) / M_PIF *
(avg * x140_alignmentWeight); (avg * x140_alignmentWeight);
@ -301,12 +340,14 @@ void CFishCloud::ApplyAlignment(CBoid& boid, const rstl::reserved_vector<CBoid*,
void CFishCloud::ApplyAttraction(CBoid& boid, const zeus::CVector3f& attractTo, void CFishCloud::ApplyAttraction(CBoid& boid, const zeus::CVector3f& attractTo,
float attractionRadius, float attractionMagnitude) const { float attractionRadius, float attractionMagnitude) const {
zeus::CVector3f delta = attractTo - boid.GetTranslation(); const zeus::CVector3f delta = attractTo - boid.GetTranslation();
if (delta.canBeNormalized()) { if (!delta.canBeNormalized()) {
float distSq = delta.magSquared(); return;
float capDistSq = attractionRadius * attractionRadius;
boid.xc_vel += ((distSq > capDistSq) ? 0.f : (1.f - distSq / capDistSq)) * delta.normalized() * attractionMagnitude;
} }
const float distSq = delta.magSquared();
const float capDistSq = attractionRadius * attractionRadius;
boid.xc_vel += ((distSq > capDistSq) ? 0.f : (1.f - distSq / capDistSq)) * delta.normalized() * attractionMagnitude;
} }
void CFishCloud::ApplyRepulsion(CBoid& boid, const zeus::CVector3f& attractTo, void CFishCloud::ApplyRepulsion(CBoid& boid, const zeus::CVector3f& attractTo,
@ -316,74 +357,93 @@ void CFishCloud::ApplyRepulsion(CBoid& boid, const zeus::CVector3f& attractTo,
void CFishCloud::ApplySwirl(CBoid& boid, const zeus::CVector3f& swirlPoint, bool clockwise, void CFishCloud::ApplySwirl(CBoid& boid, const zeus::CVector3f& swirlPoint, bool clockwise,
float magnitude, float radius) const { float magnitude, float radius) const {
zeus::CVector3f delta = boid.x0_pos - swirlPoint; const zeus::CVector3f delta = boid.x0_pos - swirlPoint;
float deltaMag = delta.magnitude(); const float deltaMag = delta.magnitude();
zeus::CVector3f alignVec; zeus::CVector3f alignVec;
if (clockwise) if (clockwise) {
alignVec = delta.normalized().cross(zeus::skUp); alignVec = delta.normalized().cross(zeus::skUp);
else } else {
alignVec = zeus::skUp.cross(delta / deltaMag); alignVec = zeus::skUp.cross(delta / deltaMag);
float weight = deltaMag > radius ? 0.f : 1.f - deltaMag / radius; }
const float weight = deltaMag > radius ? 0.f : 1.f - deltaMag / radius;
boid.xc_vel += zeus::CVector3f::getAngleDiff(boid.xc_vel, alignVec) / M_PIF * boid.xc_vel += zeus::CVector3f::getAngleDiff(boid.xc_vel, alignVec) / M_PIF *
weight * (magnitude * alignVec); weight * (magnitude * alignVec);
} }
void CFishCloud::ApplyContainment(CBoid& boid, const zeus::CAABox& aabb) const { void CFishCloud::ApplyContainment(CBoid& boid, const zeus::CAABox& aabb) const {
if (boid.xc_vel.canBeNormalized()) { if (!boid.xc_vel.canBeNormalized()) {
if (!PointInBox(aabb, boid.xc_vel.normalized() * x130_speed * x174_containmentRadius + boid.x0_pos)) { return;
ApplyAttraction(boid, aabb.center(), 100000.f, x158_containmentMagnitude);
}
} }
if (PointInBox(aabb, boid.xc_vel.normalized() * x130_speed * x174_containmentRadius + boid.x0_pos)) {
return;
}
ApplyAttraction(boid, aabb.center(), 100000.f, x158_containmentMagnitude);
} }
void CFishCloud::ScatterBoid(CStateManager& mgr, CBoid& b) const { void CFishCloud::ScatterBoid(CStateManager& mgr, CBoid& b) const {
float angle = (mgr.GetActiveRandom()->Float() - 0.5f) * M_PIF * x154_maxScatterAngle; const float angle = (mgr.GetActiveRandom()->Float() - 0.5f) * M_PIF * x154_maxScatterAngle;
float cosAngle = std::cos(angle); const float cosAngle = std::cos(angle);
float sinAngle = std::sin(angle); const float sinAngle = std::sin(angle);
b.xc_vel.x() += x150_scatterVel * (b.xc_vel.y() * sinAngle + b.xc_vel.x() * cosAngle); b.xc_vel.x() += x150_scatterVel * (b.xc_vel.y() * sinAngle + b.xc_vel.x() * cosAngle);
b.xc_vel.y() += x150_scatterVel * (b.xc_vel.y() * cosAngle + b.xc_vel.x() * sinAngle); b.xc_vel.y() += x150_scatterVel * (b.xc_vel.y() * cosAngle + b.xc_vel.x() * sinAngle);
} }
void CFishCloud::Think(float dt, CStateManager& mgr) { void CFishCloud::Think(float dt, CStateManager& mgr) {
if (!GetActive()) if (!GetActive()) {
return; return;
}
const CGameArea* area = mgr.GetWorld()->GetAreaAlways(x4_areaId); const CGameArea* area = mgr.GetWorld()->GetAreaAlways(x4_areaId);
auto occState = area->IsPostConstructed() ? area->GetOcclusionState() : CGameArea::EOcclusionState::Occluded; const auto occState = area->IsPostConstructed() ? area->GetOcclusionState() : CGameArea::EOcclusionState::Occluded;
if (occState == CGameArea::EOcclusionState::Visible) { if (occState == CGameArea::EOcclusionState::Visible) {
x168_weaponRepelDamping = std::max(0.f, x168_weaponRepelDamping - x160_weaponRepelDampingSpeed * dt * 0.1f); x168_weaponRepelDamping = std::max(0.f, x168_weaponRepelDamping - x160_weaponRepelDampingSpeed * dt * 0.1f);
if (x250_26_enableWeaponRepelDamping) if (x250_26_enableWeaponRepelDamping) {
x168_weaponRepelDamping = std::min(x160_weaponRepelDampingSpeed * dt + x168_weaponRepelDamping, x168_weaponRepelDamping =
x148_weaponRepelMagnitude); std::min(x160_weaponRepelDampingSpeed * dt + x168_weaponRepelDamping, x148_weaponRepelMagnitude);
}
x164_playerRepelDamping = std::max(0.f, x164_playerRepelDamping - x15c_playerRepelDampingSpeed * dt * 0.1f); x164_playerRepelDamping = std::max(0.f, x164_playerRepelDamping - x15c_playerRepelDampingSpeed * dt * 0.1f);
if (x250_30_enablePlayerRepelDamping) if (x250_30_enablePlayerRepelDamping) {
x164_playerRepelDamping = std::min(x15c_playerRepelDampingSpeed * dt + x164_playerRepelDamping, x164_playerRepelDamping =
x14c_playerRepelMagnitude); std::min(x15c_playerRepelDampingSpeed * dt + x164_playerRepelDamping, x14c_playerRepelMagnitude);
}
x250_26_enableWeaponRepelDamping = false; x250_26_enableWeaponRepelDamping = false;
x250_30_enablePlayerRepelDamping = false; x250_30_enablePlayerRepelDamping = false;
++x118_thinkCounter; ++x118_thinkCounter;
UpdateParticles(dt); UpdateParticles(dt);
UpdatePartitionList(); UpdatePartitionList();
zeus::CAABox aabb = GetBoundingBox();
const zeus::CAABox aabb = GetBoundingBox();
int idx = 0; int idx = 0;
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
if (b.x20_active && (idx & x11c_updateMask) == (x118_thinkCounter & x11c_updateMask)) { if (b.x20_active && (idx & x11c_updateMask) == (x118_thinkCounter & x11c_updateMask)) {
rstl::reserved_vector<CBoid*, 25> nearList; rstl::reserved_vector<CBoid*, 25> nearList;
if (x250_31_updateWithoutPartitions) if (x250_31_updateWithoutPartitions) {
BuildBoidNearList(b.x0_pos, x138_separationRadius, nearList); BuildBoidNearList(b.x0_pos, x138_separationRadius, nearList);
else } else {
BuildBoidNearPartitionList(b.x0_pos, x138_separationRadius, nearList); BuildBoidNearPartitionList(b.x0_pos, x138_separationRadius, nearList);
}
for (int i = 0; i < 5; ++i) { for (int i = 0; i < 5; ++i) {
switch (i) { switch (i) {
case 1: case 1:
ApplySeparation(b, nearList); ApplySeparation(b, nearList);
break; break;
case 2: case 2:
if (!x250_24_randomMovement || mgr.GetActiveRandom()->Float() > x12c_randomMovementTimer) if (!x250_24_randomMovement || mgr.GetActiveRandom()->Float() > x12c_randomMovementTimer) {
ApplyCohesion(b, nearList); ApplyCohesion(b, nearList);
}
break; break;
case 3: case 3:
if (!x250_24_randomMovement || mgr.GetActiveRandom()->Float() > x12c_randomMovementTimer) if (!x250_24_randomMovement || mgr.GetActiveRandom()->Float() > x12c_randomMovementTimer) {
ApplyAlignment(b, nearList); ApplyAlignment(b, nearList);
}
break; break;
case 4: case 4:
ScatterBoid(mgr, b); ScatterBoid(mgr, b);
@ -391,12 +451,14 @@ void CFishCloud::Think(float dt, CStateManager& mgr) {
default: default:
break; break;
} }
if (b.xc_vel.magSquared() > 3.2f) if (b.xc_vel.magSquared() > 3.2f) {
break; break;
}
} }
if (!x250_24_randomMovement && b.xc_vel.magSquared() < 3.2f) { if (!x250_24_randomMovement && b.xc_vel.magSquared() < 3.2f) {
for (auto& m : x108_modifierSources) { for (const auto& m : x108_modifierSources) {
if (TCastToPtr<CActor> act = mgr.ObjectById(m.x0_source)) { if (const TCastToConstPtr<CActor> act = mgr.ObjectById(m.x0_source)) {
if (m.xd_isSwirl) { if (m.xd_isSwirl) {
ApplySwirl(b, act->GetTranslation(), m.xc_isRepulsor, m.x8_priority, m.x4_radius); ApplySwirl(b, act->GetTranslation(), m.xc_isRepulsor, m.x8_priority, m.x4_radius);
} else if (m.xc_isRepulsor) { } else if (m.xc_isRepulsor) {
@ -416,28 +478,34 @@ void CFishCloud::Think(float dt, CStateManager& mgr) {
} }
++idx; ++idx;
} }
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
if (b.x20_active) { if (b.x20_active) {
ApplyContainment(b, aabb); ApplyContainment(b, aabb);
float velMag = b.xc_vel.magnitude(); const float velMag = b.xc_vel.magnitude();
if (!zeus::close_enough(velMag, 0.f)) if (!zeus::close_enough(velMag, 0.f)) {
b.xc_vel = b.xc_vel / velMag; b.xc_vel = b.xc_vel / velMag;
}
b.xc_vel.z() *= 0.99f; b.xc_vel.z() *= 0.99f;
} }
} }
if (x12c_randomMovementTimer > 0.f) { if (x12c_randomMovementTimer > 0.f) {
x12c_randomMovementTimer -= dt; x12c_randomMovementTimer -= dt;
} else { } else {
x12c_randomMovementTimer = 0.f; x12c_randomMovementTimer = 0.f;
x250_24_randomMovement = false; x250_24_randomMovement = false;
} }
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
if (b.x20_active) { if (b.x20_active) {
b.x0_pos += b.xc_vel * dt * x130_speed; b.x0_pos += b.xc_vel * dt * x130_speed;
if (!PointInBox(aabb, b.x0_pos)) if (!PointInBox(aabb, b.x0_pos)) {
PlaceBoid(mgr, b, aabb); PlaceBoid(mgr, b, aabb);
}
} }
} }
if (x250_27_validModel) { if (x250_27_validModel) {
for (auto& m : x1b0_models) { for (auto& m : x1b0_models) {
m->GetAnimationData()->SetPlaybackRate(1.f); m->GetAnimationData()->SetPlaybackRate(1.f);
@ -455,8 +523,8 @@ void CFishCloud::AllocateSkinnedModels(CStateManager& mgr, CModelData::EWhichMod
int idx = 0; int idx = 0;
for (auto& m : x1b0_models) { for (auto& m : x1b0_models) {
m->EnableLooping(true); m->EnableLooping(true);
m->AdvanceAnimation( m->AdvanceAnimation(m->GetAnimationData()->GetAnimTimeRemaining("Whole Body"sv) * 0.25f * float(idx), mgr,
m->GetAnimationData()->GetAnimTimeRemaining("Whole Body"sv) * 0.25f * idx, mgr, x4_areaId, true); x4_areaId, true);
++idx; ++idx;
} }
x230_whichModel = which; x230_whichModel = which;
@ -467,10 +535,10 @@ void CFishCloud::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId sender, CSt
switch (msg) { switch (msg) {
case EScriptObjectMessage::Registered: { case EScriptObjectMessage::Registered: {
xe8_boids.reserve(x134_numBoids); xe8_boids.reserve(x134_numBoids);
zeus::CAABox aabb = GetUntransformedBoundingBox(); const zeus::CAABox aabb = GetUntransformedBoundingBox();
zeus::CVector3f extent = aabb.max - aabb.min; const zeus::CVector3f extent = aabb.max - aabb.min;
zeus::CVector3f randPoint; zeus::CVector3f randPoint;
for (int i = 0; i < x134_numBoids; ++i) { for (u32 i = 0; i < x134_numBoids; ++i) {
randPoint.z() = mgr.GetActiveRandom()->Float() * extent.z() + aabb.min.z(); randPoint.z() = mgr.GetActiveRandom()->Float() * extent.z() + aabb.min.z();
randPoint.y() = mgr.GetActiveRandom()->Float() * extent.y() + aabb.min.y(); randPoint.y() = mgr.GetActiveRandom()->Float() * extent.y() + aabb.min.y();
randPoint.x() = mgr.GetActiveRandom()->Float() * extent.x() + aabb.min.x(); randPoint.x() = mgr.GetActiveRandom()->Float() * extent.x() + aabb.min.x();
@ -481,8 +549,9 @@ void CFishCloud::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId sender, CSt
0.2f * std::pow(mgr.GetActiveRandom()->Float(), 7.f) + 0.9f); 0.2f * std::pow(mgr.GetActiveRandom()->Float(), 7.f) + 0.9f);
} }
CreatePartitionList(); CreatePartitionList();
if (x250_27_validModel) if (x250_27_validModel) {
AllocateSkinnedModels(mgr, CModelData::EWhichModel::Normal); AllocateSkinnedModels(mgr, CModelData::EWhichModel::Normal);
}
break; break;
} }
default: default:
@ -501,22 +570,26 @@ void CFishCloud::PreRender(CStateManager& mgr, const zeus::CFrustum& frustum) {
} }
void CFishCloud::AddParticlesToRenderer() const { void CFishCloud::AddParticlesToRenderer() const {
for (const auto& p : x1f8_particleGens) for (const auto& p : x1f8_particleGens) {
g_Renderer->AddParticleGen(*p); g_Renderer->AddParticleGen(*p);
}
} }
void CFishCloud::RenderBoid(int idx, const CBoid& boid, u32& drawMask, void CFishCloud::RenderBoid(int idx, const CBoid& boid, u32& drawMask,
bool thermalHot, const CModelFlags& flags) const { bool thermalHot, const CModelFlags& flags) const {
u32 modelIndex = idx & 0x3; const u32 modelIndex = idx & 0x3;
CModelData& mData = *x1b0_models[modelIndex]; CModelData& mData = *x1b0_models[modelIndex];
CSkinnedModel& model = mData.PickAnimatedModel(CModelData::EWhichModel::Normal); CSkinnedModel& model = mData.PickAnimatedModel(CModelData::EWhichModel::Normal);
if (!model.GetModelInst()->TryLockTextures()) if (!model.GetModelInst()->TryLockTextures()) {
return; return;
u32 thisDrawMask = 1u << modelIndex; }
if (drawMask & thisDrawMask) {
const u32 thisDrawMask = 1u << modelIndex;
if ((drawMask & thisDrawMask) != 0) {
drawMask &= ~thisDrawMask; drawMask &= ~thisDrawMask;
mData.GetAnimationData()->BuildPose(); mData.GetAnimationData()->BuildPose();
} }
model.GetModelInst()->SetAmbientColor(zeus::skWhite); model.GetModelInst()->SetAmbientColor(zeus::skWhite);
CGraphics::SetModelMatrix(zeus::lookAt(boid.x0_pos, boid.x0_pos + boid.xc_vel)); CGraphics::SetModelMatrix(zeus::lookAt(boid.x0_pos, boid.x0_pos + boid.xc_vel));
if (thermalHot) { if (thermalHot) {
@ -528,24 +601,31 @@ void CFishCloud::RenderBoid(int idx, const CBoid& boid, u32& drawMask,
} }
void CFishCloud::Render(CStateManager& mgr) { void CFishCloud::Render(CStateManager& mgr) {
if (!GetActive()) if (!GetActive()) {
return; return;
}
SCOPED_GRAPHICS_DEBUG_GROUP(fmt::format(FMT_STRING("CFishCloud::Render {} {} {}"), SCOPED_GRAPHICS_DEBUG_GROUP(fmt::format(FMT_STRING("CFishCloud::Render {} {} {}"),
x8_uid, xc_editorId, x10_name).c_str(), zeus::skOrange); x8_uid, xc_editorId, x10_name).c_str(), zeus::skOrange);
bool thermalHot = mgr.GetThermalDrawFlag() == EThermalDrawFlag::Hot;
const bool thermalHot = mgr.GetThermalDrawFlag() == EThermalDrawFlag::Hot;
CModelFlags flags(0, 0, 3, zeus::skWhite); CModelFlags flags(0, 0, 3, zeus::skWhite);
if (mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::XRay) if (mgr.GetPlayerState()->GetActiveVisor(mgr) == CPlayerState::EPlayerVisor::XRay) {
flags = CModelFlags(5, 0, 3, x16c_color); flags = CModelFlags(5, 0, 3, x16c_color);
else } else {
flags = CModelFlags(1, 0, 3, x16c_color); flags = CModelFlags(1, 0, 3, x16c_color);
}
AddParticlesToRenderer(); AddParticlesToRenderer();
if (x250_27_validModel) { if (x250_27_validModel) {
// Ambient white // Ambient white
int idx = 0; int idx = 0;
u32 drawMask = 0xffffffff; u32 drawMask = 0xffffffff;
for (const auto& b : xe8_boids) { for (const auto& b : xe8_boids) {
if (b.x20_active) if (b.x20_active) {
RenderBoid(idx, b, drawMask, thermalHot, flags); RenderBoid(idx, b, drawMask, thermalHot, flags);
}
++idx; ++idx;
} }
} else { } else {
@ -595,35 +675,41 @@ void CFishCloud::KillBoid(CBoid& b) const {
void CFishCloud::Touch(CActor& other, CStateManager& mgr) { void CFishCloud::Touch(CActor& other, CStateManager& mgr) {
CActor::Touch(other, mgr); CActor::Touch(other, mgr);
if (TCastToPtr<CWeapon> weap = other) {
if (const TCastToConstPtr<CWeapon> weap = other) {
if (!x250_26_enableWeaponRepelDamping && x250_29_repelFromThreats) { if (!x250_26_enableWeaponRepelDamping && x250_29_repelFromThreats) {
int idx = 0; int idx = 0;
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
if ((idx & 0x3) == (x118_thinkCounter & 0x3)) if ((idx & 0x3) == (x118_thinkCounter & 0x3)) {
ApplyRepulsion(b, weap->GetTranslation(), 8.f, x148_weaponRepelMagnitude - x168_weaponRepelDamping); ApplyRepulsion(b, weap->GetTranslation(), 8.f, x148_weaponRepelMagnitude - x168_weaponRepelDamping);
}
++idx; ++idx;
} }
} }
x250_26_enableWeaponRepelDamping = true; x250_26_enableWeaponRepelDamping = true;
if (x250_28_killable) { if (x250_28_killable) {
if (auto tb = weap->GetTouchBounds()) { if (const auto tb = weap->GetTouchBounds()) {
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
if (b.x20_active && if (b.x20_active && tb->intersects(zeus::CAABox(weap->GetTranslation() - x170_weaponKillRadius,
tb->intersects(zeus::CAABox(weap->GetTranslation() - x170_weaponKillRadius, weap->GetTranslation() + x170_weaponKillRadius))) {
weap->GetTranslation() + x170_weaponKillRadius)))
KillBoid(b); KillBoid(b);
}
} }
} }
} }
} }
if (x250_29_repelFromThreats) { if (x250_29_repelFromThreats) {
if (TCastToPtr<CPlayer> player = other) { if (const TCastToConstPtr<CPlayer> player = other) {
zeus::CVector3f playerPos = player->GetTranslation(); const zeus::CVector3f playerPos = player->GetTranslation();
for (auto& b : xe8_boids) { for (auto& b : xe8_boids) {
zeus::CVector3f adjPlayerPos = playerPos; zeus::CVector3f adjPlayerPos = playerPos;
float zDelta = b.x0_pos.z() - adjPlayerPos.z(); const float zDelta = b.x0_pos.z() - adjPlayerPos.z();
if (zDelta > 0.f && zDelta < 2.3f) if (zDelta > 0.f && zDelta < 2.3f) {
adjPlayerPos.z() = float(b.x0_pos.z()); adjPlayerPos.z() = float(b.x0_pos.z());
}
adjPlayerPos.x() += mgr.GetActiveRandom()->Float() * 0.2f - 0.1f; adjPlayerPos.x() += mgr.GetActiveRandom()->Float() * 0.2f - 0.1f;
adjPlayerPos.y() += mgr.GetActiveRandom()->Float() * 0.2f - 0.1f; adjPlayerPos.y() += mgr.GetActiveRandom()->Float() * 0.2f - 0.1f;
ApplyRepulsion(b, adjPlayerPos, 8.f, x14c_playerRepelMagnitude - x164_playerRepelDamping); ApplyRepulsion(b, adjPlayerPos, 8.f, x14c_playerRepelMagnitude - x164_playerRepelDamping);
@ -634,7 +720,7 @@ void CFishCloud::Touch(CActor& other, CStateManager& mgr) {
} }
zeus::CAABox CFishCloud::GetUntransformedBoundingBox() const { zeus::CAABox CFishCloud::GetUntransformedBoundingBox() const {
zeus::CVector3f extent = x120_scale * 0.75f; const zeus::CVector3f extent = x120_scale * 0.75f;
return zeus::CAABox(-extent, extent); return zeus::CAABox(-extent, extent);
} }
@ -643,46 +729,62 @@ zeus::CAABox CFishCloud::GetBoundingBox() const {
} }
void CFishCloud::RemoveRepulsor(TUniqueId sourceId) { void CFishCloud::RemoveRepulsor(TUniqueId sourceId) {
CModifierSource source(sourceId, true, false, 0.f, 0.f); const CModifierSource source(sourceId, true, false, 0.f, 0.f);
auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source); const auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source);
if (it != x108_modifierSources.end())
x108_modifierSources.erase(it); if (it == x108_modifierSources.end()) {
return;
}
x108_modifierSources.erase(it);
} }
void CFishCloud::RemoveAttractor(TUniqueId sourceId) { void CFishCloud::RemoveAttractor(TUniqueId sourceId) {
CModifierSource source(sourceId, false, false, 0.f, 0.f); const CModifierSource source(sourceId, false, false, 0.f, 0.f);
auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source); const auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source);
if (it != x108_modifierSources.end())
x108_modifierSources.erase(it); if (it == x108_modifierSources.end()) {
return;
}
x108_modifierSources.erase(it);
} }
bool CFishCloud::AddRepulsor(TUniqueId sourceId, bool swirl, float radius, float priority) { bool CFishCloud::AddRepulsor(TUniqueId sourceId, bool swirl, float radius, float priority) {
CModifierSource source(sourceId, true, swirl, radius, priority); const CModifierSource source(sourceId, true, swirl, radius, priority);
auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source); const auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source);
if (it != x108_modifierSources.end()) { if (it != x108_modifierSources.end()) {
it->x4_radius = radius; it->x4_radius = radius;
it->x8_priority = priority; it->x8_priority = priority;
return true; return true;
} else if (x108_modifierSources.size() < x108_modifierSources.capacity()) { }
x108_modifierSources.insert(std::lower_bound(
x108_modifierSources.begin(), x108_modifierSources.end(), source), source); if (x108_modifierSources.size() < x108_modifierSources.capacity()) {
x108_modifierSources.insert(std::lower_bound(x108_modifierSources.begin(), x108_modifierSources.end(), source),
source);
return true; return true;
} }
return false; return false;
} }
bool CFishCloud::AddAttractor(TUniqueId sourceId, bool swirl, float radius, float priority) { bool CFishCloud::AddAttractor(TUniqueId sourceId, bool swirl, float radius, float priority) {
CModifierSource source(sourceId, false, swirl, radius, priority); const CModifierSource source(sourceId, false, swirl, radius, priority);
auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source); const auto it = rstl::binary_find(x108_modifierSources.begin(), x108_modifierSources.end(), source);
if (it != x108_modifierSources.end()) { if (it != x108_modifierSources.end()) {
it->x4_radius = radius; it->x4_radius = radius;
it->x8_priority = priority; it->x8_priority = priority;
return true; return true;
} else if (x108_modifierSources.size() < x108_modifierSources.capacity()) { }
x108_modifierSources.insert(std::lower_bound(
x108_modifierSources.begin(), x108_modifierSources.end(), source), source); if (x108_modifierSources.size() < x108_modifierSources.capacity()) {
x108_modifierSources.insert(std::lower_bound(x108_modifierSources.begin(), x108_modifierSources.end(), source),
source);
return true; return true;
} }
return false; return false;
} }

View File

@ -80,7 +80,7 @@ class CFishCloud : public CActor {
rstl::reserved_vector<TLockedToken<CGenDescription>, 4> x1c4_particleDescs; rstl::reserved_vector<TLockedToken<CGenDescription>, 4> x1c4_particleDescs;
rstl::reserved_vector<std::unique_ptr<CElementGen>, 4> x1f8_particleGens; rstl::reserved_vector<std::unique_ptr<CElementGen>, 4> x1f8_particleGens;
rstl::reserved_vector<u32, 4> x21c_deathParticleCounts; rstl::reserved_vector<u32, 4> x21c_deathParticleCounts;
CModelData::EWhichModel x230_whichModel; CModelData::EWhichModel x230_whichModel{};
u16 x234_deathSfx; u16 x234_deathSfx;
zeus::CVector3f x238_partitionPitch; zeus::CVector3f x238_partitionPitch;
zeus::CVector3f x244_ooPartitionPitch; zeus::CVector3f x244_ooPartitionPitch;

View File

@ -6,15 +6,11 @@
namespace urde { namespace urde {
CFluidUVMotion::CFluidUVMotion(float timeToWrap, float orientation, const CFluidUVMotion::SFluidLayerMotion& colorLayer, CFluidUVMotion::CFluidUVMotion(float timeToWrap, float orientation, const SFluidLayerMotion& colorLayer,
const CFluidUVMotion::SFluidLayerMotion& pattern1Layer, const SFluidLayerMotion& pattern1Layer, const SFluidLayerMotion& pattern2Layer)
const CFluidUVMotion::SFluidLayerMotion& pattern2Layer) : x0_fluidLayers{{colorLayer, pattern1Layer, pattern2Layer}}
: x4c_ooTimeToWrap(1.f / timeToWrap), x50_orientation(orientation) { , x4c_ooTimeToWrap(1.f / timeToWrap)
x0_fluidLayers.resize(3); , x50_orientation(orientation) {}
x0_fluidLayers[0] = colorLayer;
x0_fluidLayers[1] = pattern1Layer;
x0_fluidLayers[2] = pattern2Layer;
}
CFluidUVMotion::CFluidUVMotion(float timeToWrap, float orientation) CFluidUVMotion::CFluidUVMotion(float timeToWrap, float orientation)
: x4c_ooTimeToWrap(1.f / timeToWrap), x50_orientation(orientation) { : x4c_ooTimeToWrap(1.f / timeToWrap), x50_orientation(orientation) {

View File

@ -400,13 +400,14 @@ bool CKnockBackController::TickDeferredTimer(float dt) {
return false; return false;
} }
EKnockBackCharacterState CKnockBackController::GetKnockBackCharacterState(CPatterned& parent) { EKnockBackCharacterState CKnockBackController::GetKnockBackCharacterState(const CPatterned& parent) const {
if (parent.GetBodyController()->IsFrozen()) if (parent.GetBodyController()->IsFrozen()) {
return parent.IsAlive() ? EKnockBackCharacterState::FrozenAlive : EKnockBackCharacterState::FrozenDead; return parent.IsAlive() ? EKnockBackCharacterState::FrozenAlive : EKnockBackCharacterState::FrozenDead;
}
return parent.IsAlive() ? EKnockBackCharacterState::Alive : EKnockBackCharacterState::Dead; return parent.IsAlive() ? EKnockBackCharacterState::Alive : EKnockBackCharacterState::Dead;
} }
void CKnockBackController::ValidateState(CPatterned& parent) { void CKnockBackController::ValidateState(const CPatterned& parent) {
if (x4_activeParms.x0_animState < x18_minAnimState) if (x4_activeParms.x0_animState < x18_minAnimState)
x4_activeParms.x0_animState = x18_minAnimState; x4_activeParms.x0_animState = x18_minAnimState;
else if (x4_activeParms.x0_animState > x1c_maxAnimState) else if (x4_activeParms.x0_animState > x1c_maxAnimState)
@ -472,10 +473,12 @@ void CKnockBackController::ValidateState(CPatterned& parent) {
} }
} }
float CKnockBackController::CalculateExtraHurlVelocity(CStateManager& mgr, float magnitude, float kbResistance) { float CKnockBackController::CalculateExtraHurlVelocity(CStateManager& mgr, float magnitude, float kbResistance) const {
if (magnitude > kbResistance) if (magnitude <= kbResistance) {
return (1.1f - 0.2f * mgr.GetActiveRandom()->Float()) * 2.f * (magnitude - kbResistance); return 0.f;
return 0.f; }
return (1.1f - 0.2f * mgr.GetActiveRandom()->Float()) * 2.f * (magnitude - kbResistance);
} }
void CKnockBackController::DoKnockBackAnimation(const zeus::CVector3f& backVec, CStateManager& mgr, CPatterned& parent, void CKnockBackController::DoKnockBackAnimation(const zeus::CVector3f& backVec, CStateManager& mgr, CPatterned& parent,
@ -517,7 +520,8 @@ void CKnockBackController::DoKnockBackAnimation(const zeus::CVector3f& backVec,
} }
} }
void CKnockBackController::ResetKnockBackImpulse(CPatterned& parent, const zeus::CVector3f& backVec, float magnitude) { void CKnockBackController::ResetKnockBackImpulse(const CPatterned& parent, const zeus::CVector3f& backVec,
float magnitude) {
if (x81_24_autoResetImpulse && x4_activeParms.x0_animState == EKnockBackAnimationState::KnockBack && if (x81_24_autoResetImpulse && x4_activeParms.x0_animState == EKnockBackAnimationState::KnockBack &&
x4_activeParms.x4_animFollowup != EKnockBackAnimationFollowUp::Freeze) { x4_activeParms.x4_animFollowup != EKnockBackAnimationFollowUp::Freeze) {
x50_impulseDir = backVec.canBeNormalized() ? backVec.normalized() : -parent.GetTransform().basis[1]; x50_impulseDir = backVec.canBeNormalized() ? backVec.normalized() : -parent.GetTransform().basis[1];
@ -608,16 +612,18 @@ EKnockBackWeaponType CKnockBackController::GetKnockBackWeaponType(const CDamageI
} }
} }
void CKnockBackController::SelectDamageState(CPatterned& parent, const CDamageInfo& info, EWeaponType wType, void CKnockBackController::SelectDamageState(const CPatterned& parent, const CDamageInfo& info, EWeaponType wType,
EKnockBackType type) { EKnockBackType type) {
x4_activeParms = KnockBackParms(); x4_activeParms = KnockBackParms();
EKnockBackWeaponType weaponType = GetKnockBackWeaponType(info, wType, type);
if (weaponType != EKnockBackWeaponType::Invalid) { const EKnockBackWeaponType weaponType = GetKnockBackWeaponType(info, wType, type);
x4_activeParms = if (weaponType == EKnockBackWeaponType::Invalid) {
KnockBackParmsTable[size_t(x0_variant)][size_t(weaponType)][size_t(GetKnockBackCharacterState(parent))]; return;
ValidateState(parent);
} }
x4_activeParms =
KnockBackParmsTable[size_t(x0_variant)][size_t(weaponType)][size_t(GetKnockBackCharacterState(parent))];
ValidateState(parent);
} }
void CKnockBackController::KnockBack(const zeus::CVector3f& backVec, CStateManager& mgr, CPatterned& parent, void CKnockBackController::KnockBack(const zeus::CVector3f& backVec, CStateManager& mgr, CPatterned& parent,

View File

@ -99,14 +99,14 @@ private:
bool x82_26_locomotionDuringElectrocution : 1; bool x82_26_locomotionDuringElectrocution : 1;
void ApplyImpulse(float dt, CPatterned& parent); void ApplyImpulse(float dt, CPatterned& parent);
bool TickDeferredTimer(float dt); bool TickDeferredTimer(float dt);
EKnockBackCharacterState GetKnockBackCharacterState(CPatterned& parent); EKnockBackCharacterState GetKnockBackCharacterState(const CPatterned& parent) const;
void ValidateState(CPatterned& parent); void ValidateState(const CPatterned& parent);
float CalculateExtraHurlVelocity(CStateManager& mgr, float magnitude, float kbResistance); float CalculateExtraHurlVelocity(CStateManager& mgr, float magnitude, float kbResistance) const;
void DoKnockBackAnimation(const zeus::CVector3f& backVec, CStateManager& mgr, CPatterned& parent, float magnitude); void DoKnockBackAnimation(const zeus::CVector3f& backVec, CStateManager& mgr, CPatterned& parent, float magnitude);
void ResetKnockBackImpulse(CPatterned& parent, const zeus::CVector3f& backVec, float magnitude); void ResetKnockBackImpulse(const CPatterned& parent, const zeus::CVector3f& backVec, float magnitude);
void DoDeferredKnockBack(CStateManager& mgr, CPatterned& parent); void DoDeferredKnockBack(CStateManager& mgr, CPatterned& parent);
EKnockBackWeaponType GetKnockBackWeaponType(const CDamageInfo& info, EWeaponType wType, EKnockBackType type); EKnockBackWeaponType GetKnockBackWeaponType(const CDamageInfo& info, EWeaponType wType, EKnockBackType type);
void SelectDamageState(CPatterned& parent, const CDamageInfo& info, EWeaponType wType, EKnockBackType type); void SelectDamageState(const CPatterned& parent, const CDamageInfo& info, EWeaponType wType, EKnockBackType type);
public: public:
explicit CKnockBackController(EKnockBackVariant variant); explicit CKnockBackController(EKnockBackVariant variant);