#pragma once #include #include #include "Runtime/CToken.hpp" #include "Runtime/RetroTypes.hpp" #include "Runtime/rstl.hpp" #include "Runtime/Audio/CSfxManager.hpp" #include "Runtime/Particle/CGenDescription.hpp" #include "Runtime/Graphics/CLineRenderer.hpp" #include "Runtime/Graphics/Shaders/CEnvFxShaders.hpp" #include "zeus/CAABox.hpp" #include "zeus/CTransform.hpp" #include "zeus/CVector2i.hpp" namespace metaforce { class CActor; class CStateManager; class CTexture; enum class EEnvFxType { None, Snow, Rain, UnderwaterFlake }; enum class EPhazonType { None, Blue, Orange }; class CVectorFixed8_8 { public: union { struct { s16 x, y, z; }; std::array v{}; }; CVectorFixed8_8() = default; CVectorFixed8_8(s16 xi, s16 yi, s16 zi) : v{xi, yi, zi} {} CVectorFixed8_8(const zeus::CVector3f& vec) { zeus::simd_floats f(vec.mSimd); x = s16(f[0] * 256.f); y = s16(f[1] * 256.f); z = s16(f[2] * 256.f); } CVectorFixed8_8 operator+(const CVectorFixed8_8& other) const { return {s16(x + other.x), s16(y + other.y), s16(z + other.z)}; } CVectorFixed8_8 operator-(const CVectorFixed8_8& other) const { return {s16(x - other.x), s16(y - other.y), s16(z - other.z)}; } zeus::CVector3f toVec3f() const { return {x / 256.f, y / 256.f, z / 256.f}; } }; class CEnvFxManagerGrid { friend class CEnvFxManager; friend class CEnvFxShaders; bool x0_24_blockDirty = true; zeus::CVector2i x4_position; /* 8.8 fixed point */ zeus::CVector2i xc_extent; /* 8.8 fixed point */ std::pair x14_block = {false, FLT_MAX}; /* Blocked-bool, Z-coordinate */ std::vector x1c_particles; hsh::dynamic_owner> m_instBuf; hsh::dynamic_owner> m_uniformBuf; CLineRenderer m_lineRenderer; hsh::binding m_snowBinding; hsh::binding m_underwaterBinding; float m_uvyOffset = 0.f; void RenderSnowParticles(const zeus::CTransform& camXf); void RenderRainParticles(const zeus::CTransform& camXf); void RenderUnderwaterParticles(const zeus::CTransform& camXf); public: CEnvFxManagerGrid(const zeus::CVector2i& position, const zeus::CVector2i& extent, std::vector initialParticles, int reserve, CEnvFxManager& parent); void Render(const zeus::CTransform& xf, const zeus::CTransform& invXf, const zeus::CTransform& camXf, float fxDensity, EEnvFxType fxType, CEnvFxManager& parent); }; class CEnvFxManager { friend class CEnvFxManagerGrid; friend class CEnvFxShaders; CEnvFxShaders::Uniform m_uniformData; hsh::dynamic_owner> m_fogUniformBuf; zeus::CAABox x0_particleBounds = zeus::CAABox(-63.5f, 63.5f); zeus::CVector3f x18_focusCellPosition = zeus::skZero3f; bool x24_enableSplash = false; float x28_firstSnowForce = 0.f; s32 x2c_lastBlockedGridIdx = -1; float x30_fxDensity = 0.f; float x34_targetFxDensity = 0.f; float x38_maxDensityDeltaSpeed = 0.f; // bool x3c_snowflakeTextureMipBlanked = false; /* Shader simulates this texture mod */ TLockedToken x40_txtrEnvGradient; rstl::reserved_vector x50_grids; float xb54_baseSplashRate = 0.f; TLockedToken xb58_envRainSplash; bool xb64_ = true; TUniqueId xb68_envRainSplashId = kInvalidUniqueId; bool xb6a_rainSoundActive = false; CSfxHandle xb6c_leftRainSound; CSfxHandle xb70_rightRainSound; TLockedToken xb74_txtrSnowFlake; bool xb80_ = true; rstl::reserved_vector xb84_snowZDeltas; TLockedToken xc48_underwaterFlake; bool xc54_ = true; void SetSplashEffectRate(float f, const CStateManager& mgr); void UpdateRainSounds(const CStateManager& mgr); zeus::CVector3f GetParticleBoundsToWorldScale() const; zeus::CTransform GetParticleBoundsToWorldTransform() const; void UpdateVisorSplash(CStateManager& mgr, float dt, const zeus::CTransform& camXf); void MoveWrapCells(s32, s32); void CalculateSnowForces(const CVectorFixed8_8& zVec, rstl::reserved_vector& snowForces, EEnvFxType type, const zeus::CVector3f& oopbtws, float dt); static void BuildBlockObjectList(rstl::reserved_vector& list, CStateManager& mgr); void UpdateBlockedGrids(CStateManager& mgr, EEnvFxType type, const zeus::CTransform& camXf, const zeus::CTransform& xf, const zeus::CTransform& invXf); void CreateNewParticles(EEnvFxType type); void UpdateSnowParticles(const rstl::reserved_vector& snowForces); void UpdateRainParticles(const CVectorFixed8_8& zVec, const zeus::CVector3f& oopbtws, float dt); void UpdateUnderwaterParticles(const CVectorFixed8_8& zVec); void SetupSnowTevs(const CStateManager& mgr); void SetupRainTevs() const; void SetupUnderwaterTevs(const zeus::CTransform& invXf, const CStateManager& mgr) const; public: CEnvFxManager(); void AsyncLoadResources(CStateManager& mgr); void Update(float, CStateManager& mgr); void Render(const CStateManager& mgr); void SetFxDensity(s32, float); void AreaLoaded(); void SetSplashRate(float f) { xb54_baseSplashRate = f; } bool IsSplashActive() const { return x24_enableSplash; } float GetRainMagnitude() const { return x30_fxDensity; } void Cleanup(); static void Initialize(); }; } // namespace metaforce