metaforce/Runtime/World/CFluidPlaneManager.cpp

144 lines
4.2 KiB
C++
Raw Permalink Normal View History

#include "Runtime/World/CFluidPlaneManager.hpp"
#include "Runtime/CStateManager.hpp"
#include "Runtime/World/CExplosion.hpp"
#include "Runtime/World/CScriptWater.hpp"
2021-04-10 01:42:06 -07:00
namespace metaforce {
CFluidPlaneManager::CFluidProfile CFluidPlaneManager::sProfile = {};
2018-12-07 21:30:43 -08:00
CFluidPlaneManager::CFluidPlaneManager() : x0_rippleManager(20, 0.5f) {
sProfile.Clear();
SetupRippleMap();
}
2018-12-07 21:30:43 -08:00
void CFluidPlaneManager::CFluidProfile::Clear() {
x0_ = 0.f;
x4_ = 0.f;
x8_ = 0.f;
xc_ = 0.f;
x10_ = 0.f;
}
2018-12-07 21:30:43 -08:00
void CFluidPlaneManager::StartFrame(bool b) {
x121_ = b;
sProfile.Clear();
}
2018-12-07 21:30:43 -08:00
void CFluidPlaneManager::Update(float dt) {
x11c_uvT += dt;
x0_rippleManager.Update(dt);
for (CSplashRecord& record : x18_splashes) {
record.SetTime(record.GetTime() + dt);
if (record.GetTime() > 9999.f)
record.SetTime(9999.f);
}
}
2018-12-07 21:30:43 -08:00
float CFluidPlaneManager::GetLastRippleDeltaTime(TUniqueId rippler) const {
return x0_rippleManager.GetLastRippleDeltaTime(rippler);
}
2018-12-07 21:30:43 -08:00
float CFluidPlaneManager::GetLastSplashDeltaTime(TUniqueId splasher) const {
float newestTime = 9999.f;
for (const CSplashRecord& record : x18_splashes)
if (record.GetUniqueId() == splasher && newestTime > record.GetTime())
newestTime = record.GetTime();
return newestTime;
}
void CFluidPlaneManager::CreateSplash(TUniqueId splasher, CStateManager& mgr, const CScriptWater& water,
2018-12-07 21:30:43 -08:00
const zeus::CVector3f& pos, float factor, bool sfx) {
if (water.CanRippleAtPoint(pos)) {
float oldestTime = 0.f;
CSplashRecord* oldestRecord = nullptr;
for (CSplashRecord& record : x18_splashes) {
if (record.GetTime() > oldestTime) {
oldestRecord = &record;
oldestTime = record.GetTime();
}
}
if (oldestRecord)
*oldestRecord = CSplashRecord(0.f, splasher);
else
x18_splashes.emplace_back(0.f, splasher);
float splashScale = water.GetSplashEffectScale(factor);
if (water.GetSplashEffect(factor)) {
CExplosion* expl = new CExplosion(*water.GetSplashEffect(factor), mgr.AllocateUniqueId(), true,
CEntityInfo(water.GetAreaIdAlways(), CEntity::NullConnectionList), "Splash",
2021-06-07 12:29:18 -07:00
zeus::CTransform(zeus::CMatrix3f(), pos), 1, zeus::CVector3f{splashScale},
water.GetSplashColor());
2018-12-07 21:30:43 -08:00
mgr.AddObject(expl);
}
if (sfx) {
2021-06-07 12:29:18 -07:00
CSfxManager::AddEmitter(water.GetSplashSound(factor), pos, zeus::skUp, true, false, 0x7f, kInvalidAreaId);
2018-02-01 15:19:34 -08:00
}
2018-12-07 21:30:43 -08:00
}
}
static bool g_RippleMapSetup = false;
std::array<std::array<u8, 64>, 64> CFluidPlaneManager::RippleValues{};
std::array<u8, 64> CFluidPlaneManager::RippleMins{};
std::array<u8, 64> CFluidPlaneManager::RippleMaxs{};
2022-07-29 13:16:55 -07:00
GXTexObj CFluidPlaneManager::RippleMapTex;
2017-08-07 23:03:57 -07:00
2018-12-07 21:30:43 -08:00
void CFluidPlaneManager::SetupRippleMap() {
if (g_RippleMapSetup) {
2018-12-07 21:30:43 -08:00
return;
}
2018-12-07 21:30:43 -08:00
g_RippleMapSetup = true;
float curX = 0.f;
for (size_t i = 0; i < 64; ++i) {
2018-12-07 21:30:43 -08:00
float curY = 0.f;
float minY = 1.f;
float maxY = 0.f;
for (size_t j = 0; j < 64; ++j) {
const float rVal = 1.f - curY;
2018-12-07 21:30:43 -08:00
float minX = curY;
float maxX = 1.25f * (0.25f * rVal + 0.1f) + curY;
if (curY < 0.f) {
2018-12-07 21:30:43 -08:00
minX = 0.f;
} else if (maxX > 1.f) {
2018-12-07 21:30:43 -08:00
maxX = 1.f;
}
2018-12-07 21:30:43 -08:00
float val = 0.f;
if (curX >= minX && curX <= maxX) {
const float t = (curX - minX) / (maxX - minX);
if (t < 0.4f) {
2018-12-07 21:30:43 -08:00
val = 2.5f * t;
} else if (t > 0.75f) {
2018-12-07 21:30:43 -08:00
val = 4.f * (1.f - t);
} else {
2018-12-07 21:30:43 -08:00
val = 1.f;
}
2018-12-07 21:30:43 -08:00
}
const auto valA = u8(std::max(int(255.f * val * rVal * rVal) - 1, 0));
2018-12-07 21:30:43 -08:00
RippleValues[i][j] = valA;
if (valA != 0 && curY < minY) {
2018-12-07 21:30:43 -08:00
minY = curY;
}
if (valA != 0 && curY > maxY) {
2018-12-07 21:30:43 -08:00
maxY = curY;
}
2018-12-07 21:30:43 -08:00
curY += (1.f / 63.f);
2017-08-07 23:03:57 -07:00
}
const auto valB = u8(std::max(int(255.f * minY) - 1, 0));
const auto valC = u8(std::min(int(255.f * maxY) + 1, 255));
2018-12-07 21:30:43 -08:00
RippleMins[i] = valB;
RippleMaxs[i] = valC;
curX += (1.f / 63.f);
}
2022-07-29 13:16:55 -07:00
GXInitTexObj(&RippleMapTex, RippleValues.data(), 64, 64, GX_TF_R8_PC, GX_REPEAT, GX_REPEAT, false);
2017-08-07 23:03:57 -07:00
}
2022-07-29 13:16:55 -07:00
void CFluidPlaneManager::Shutdown() { GXDestroyTexObj(&RippleMapTex); }
2022-03-08 15:35:40 -08:00
2021-04-10 01:42:06 -07:00
} // namespace metaforce