metaforce/Runtime/Particle/CDecal.cpp

282 lines
9.2 KiB
C++
Raw Normal View History

#include "Runtime/Particle/CDecal.hpp"
#include "Runtime/Graphics/CModel.hpp"
#include "Runtime/Particle/CParticleGlobals.hpp"
2016-08-04 22:24:28 +00:00
2018-12-08 05:30:43 +00:00
namespace urde {
CRandom16 CDecal::sDecalRandom;
2018-03-17 03:41:01 +00:00
bool CDecal::sMoveRedToAlphaBuffer = false;
2016-08-04 22:24:28 +00:00
CDecal::CDecal(const TToken<CDecalDescription>& desc, const zeus::CTransform& xf)
: x0_description(desc), xc_transform(xf) {
2018-12-08 05:30:43 +00:00
CGlobalRandom gr(sDecalRandom);
2016-08-04 22:24:28 +00:00
2018-12-08 05:30:43 +00:00
CDecalDescription& desco = *x0_description;
x5c_31_quad1Invalid = InitQuad(x3c_decalQuads[0], desco.x0_Quads[0]);
x5c_30_quad2Invalid = InitQuad(x3c_decalQuads[1], desco.x0_Quads[1]);
2017-01-22 17:36:25 +00:00
2018-12-08 05:30:43 +00:00
CDecalDescription* d = x0_description.GetObj();
if (d->x38_DMDL) {
2020-09-29 09:54:12 +00:00
if (d->x48_DLFT) {
2018-12-08 05:30:43 +00:00
d->x48_DLFT->GetValue(0, x54_modelLifetime);
2020-09-29 09:54:12 +00:00
} else {
2018-12-08 05:30:43 +00:00
x54_modelLifetime = 0x7FFFFF;
2020-09-29 09:54:12 +00:00
}
2018-12-08 05:30:43 +00:00
2020-09-29 09:54:12 +00:00
if (d->x50_DMRT) {
2018-12-08 05:30:43 +00:00
d->x50_DMRT->GetValue(0, x60_rotation);
2020-09-29 09:54:12 +00:00
}
} else {
2018-12-08 05:30:43 +00:00
x5c_29_modelInvalid = true;
}
for (auto& decal : x3c_decalQuads) {
hsh::texture2d tex{};
if (CUVElement* texDesc = decal.m_desc->x14_TEX.get()) {
tex = texDesc->GetValueTexture(x58_frameIdx)->GetBooTexture();
}
CDecalShaders::BuildShaderDataBinding(decal.m_binding, decal, tex);
}
2017-01-22 17:36:25 +00:00
}
2018-12-08 05:30:43 +00:00
bool CDecal::InitQuad(CQuadDecal& quad, const SQuadDescr& desc) {
quad.m_desc = &desc;
if (desc.x14_TEX) {
2020-09-29 09:54:12 +00:00
quad.m_instBuf = hsh::create_dynamic_vertex_buffer<SParticleInstanceTex>(1);
} else {
quad.m_instBuf = hsh::create_dynamic_vertex_buffer<SParticleInstanceNoTex>(1);
}
quad.m_uniformBuf = hsh::create_dynamic_uniform_buffer<SParticleUniforms>();
if (desc.x14_TEX) {
if (desc.x0_LFT) {
2018-12-08 05:30:43 +00:00
desc.x0_LFT->GetValue(0, quad.x4_lifetime);
2020-09-29 09:54:12 +00:00
} else {
2018-12-08 05:30:43 +00:00
quad.x4_lifetime = 0x7FFFFF;
2020-09-29 09:54:12 +00:00
}
2018-12-08 05:30:43 +00:00
if (desc.x8_ROT) {
desc.x8_ROT->GetValue(0, quad.x8_rotation);
quad.x0_24_invalid = desc.x8_ROT->IsConstant();
2018-03-17 03:41:01 +00:00
}
2018-12-08 05:30:43 +00:00
if (desc.x4_SZE) {
quad.x0_24_invalid = desc.x4_SZE->IsConstant();
2020-09-29 09:54:12 +00:00
if (!quad.x0_24_invalid) {
float size = 1.f;
desc.x4_SZE->GetValue(0, size);
quad.x0_24_invalid = size <= 1.f;
}
2018-03-17 03:41:01 +00:00
}
2020-09-29 09:54:12 +00:00
if (desc.xc_OFF) {
2018-12-08 05:30:43 +00:00
quad.x0_24_invalid = desc.xc_OFF->IsFastConstant();
2020-09-29 09:54:12 +00:00
}
2018-12-08 05:30:43 +00:00
return false;
}
2018-03-17 03:41:01 +00:00
2018-12-08 05:30:43 +00:00
quad.x0_24_invalid = false;
return true;
}
2018-03-17 03:41:01 +00:00
2018-12-08 05:30:43 +00:00
void CDecal::SetGlobalSeed(u16 seed) { sDecalRandom.SetSeed(seed); }
void CDecal::SetMoveRedToAlphaBuffer(bool move) { sMoveRedToAlphaBuffer = move; }
2020-09-29 09:54:12 +00:00
void CDecal::RenderQuad(CQuadDecal& decal, const SQuadDescr& desc) {
zeus::CColor color = zeus::skWhite;
2018-12-08 05:30:43 +00:00
float size = 1.f;
zeus::CVector3f offset;
2020-09-29 09:54:12 +00:00
if (CColorElement* clr = desc.x10_CLR.get()) {
2018-12-08 05:30:43 +00:00
clr->GetValue(x58_frameIdx, color);
2020-09-29 09:54:12 +00:00
}
2018-12-08 05:30:43 +00:00
if (CRealElement* sze = desc.x4_SZE.get()) {
sze->GetValue(x58_frameIdx, size);
size *= 0.5f;
}
2020-09-29 09:54:12 +00:00
if (CRealElement* rot = desc.x8_ROT.get()) {
2018-12-08 05:30:43 +00:00
rot->GetValue(x58_frameIdx, decal.x8_rotation);
2020-09-29 09:54:12 +00:00
}
2018-12-08 05:30:43 +00:00
if (CVectorElement* off = desc.xc_OFF.get()) {
off->GetValue(x58_frameIdx, offset);
offset.y() = 0.f;
}
zeus::CTransform modXf = xc_transform;
modXf.origin += offset;
CGraphics::SetModelMatrix(modXf);
#if !HSH_PROFILE_MODE
2018-12-08 05:30:43 +00:00
SParticleUniforms uniformData = {
2020-09-29 09:54:12 +00:00
CGraphics::GetPerspectiveProjectionMatrix(true) * CGraphics::g_GXModelView.toMatrix4f(),
{1.f, 1.f, 1.f, 1.f},
};
decal.m_uniformBuf.load(uniformData);
#endif
2018-12-08 05:30:43 +00:00
SUVElementSet uvSet = {0.f, 1.f, 0.f, 1.f};
if (CUVElement* tex = desc.x14_TEX.get()) {
tex->GetValueUV(x58_frameIdx, uvSet);
g_instTexData.clear();
g_instTexData.reserve(1);
SParticleInstanceTex& inst = g_instTexData.emplace_back();
2018-12-08 05:30:43 +00:00
if (decal.x8_rotation == 0.f) {
2020-09-29 09:54:12 +00:00
inst.pos[0] = hsh::float4(-size, 0.001f, size, 1.f);
inst.pos[1] = hsh::float4(size, 0.001f, size, 1.f);
inst.pos[2] = hsh::float4(-size, 0.001f, -size, 1.f);
inst.pos[3] = hsh::float4(size, 0.001f, -size, 1.f);
2018-12-08 05:30:43 +00:00
} else {
float ang = zeus::degToRad(decal.x8_rotation);
float sinSize = std::sin(ang) * size;
float cosSize = std::cos(ang) * size;
2020-09-29 09:54:12 +00:00
inst.pos[0] = hsh::float4(sinSize - cosSize, 0.001f, cosSize + sinSize, 1.f);
inst.pos[1] = hsh::float4(cosSize + sinSize, 0.001f, cosSize - sinSize, 1.f);
inst.pos[2] = hsh::float4(-(cosSize + sinSize), 0.001f, -(cosSize - sinSize), 1.f);
inst.pos[3] = hsh::float4(-sinSize + cosSize, 0.001f, -cosSize - sinSize, 1.f);
2018-03-17 03:41:01 +00:00
}
2018-12-08 05:30:43 +00:00
inst.color = color;
2020-09-29 09:54:12 +00:00
inst.uvs[0] = hsh::float2(uvSet.xMin, uvSet.yMin);
inst.uvs[1] = hsh::float2(uvSet.xMax, uvSet.yMin);
inst.uvs[2] = hsh::float2(uvSet.xMin, uvSet.yMax);
inst.uvs[3] = hsh::float2(uvSet.xMax, uvSet.yMax);
#if !HSH_PROFILE_MODE
2020-09-29 09:54:12 +00:00
decal.m_instBuf.load(hsh::detail::ArrayProxy{g_instTexData.data(), g_instTexData.size()});
#endif
decal.m_binding.draw_instanced(0, 4, g_instTexData.size());
2018-12-08 05:30:43 +00:00
} else {
g_instNoTexData.clear();
g_instNoTexData.reserve(1);
SParticleInstanceNoTex& inst = g_instNoTexData.emplace_back();
2018-12-08 05:30:43 +00:00
if (decal.x8_rotation == 0.f) {
2020-09-29 09:54:12 +00:00
inst.pos[0] = hsh::float4(-size, 0.001f, size, 1.f);
inst.pos[1] = hsh::float4(size, 0.001f, size, 1.f);
inst.pos[2] = hsh::float4(-size, 0.001f, -size, 1.f);
inst.pos[3] = hsh::float4(size, 0.001f, -size, 1.f);
2018-12-08 05:30:43 +00:00
} else {
float ang = zeus::degToRad(decal.x8_rotation);
float sinSize = std::sin(ang) * size;
float cosSize = std::cos(ang) * size;
2020-09-29 09:54:12 +00:00
inst.pos[0] = hsh::float4(sinSize - cosSize, 0.001f, cosSize + sinSize, 1.f);
inst.pos[1] = hsh::float4(cosSize + sinSize, 0.001f, cosSize - sinSize, 1.f);
inst.pos[2] = hsh::float4(-(cosSize + sinSize), 0.001f, -(cosSize - sinSize), 1.f);
inst.pos[3] = hsh::float4(-sinSize + cosSize, 0.001f, -cosSize - sinSize, 1.f);
2018-03-17 03:41:01 +00:00
}
2018-12-08 05:30:43 +00:00
inst.color = color;
2018-03-17 03:41:01 +00:00
#if !HSH_PROFILE_MODE
2020-09-29 09:54:12 +00:00
decal.m_instBuf.load(hsh::detail::ArrayProxy{g_instNoTexData.data(), g_instNoTexData.size()});
#endif
decal.m_binding.draw_instanced(0, 4, g_instNoTexData.size());
2018-12-08 05:30:43 +00:00
}
2017-01-22 17:36:25 +00:00
}
void CDecal::RenderMdl() {
2018-12-08 05:30:43 +00:00
const CDecalDescription& desc = *x0_description;
zeus::CColor color = zeus::skWhite;
2018-12-08 05:30:43 +00:00
zeus::CVector3f dmop;
zeus::CTransform rotXf;
if (!desc.x5c_25_DMOO)
rotXf = xc_transform.getRotation();
bool dmrtIsConst = false;
if (CVectorElement* dmrt = desc.x50_DMRT.get())
dmrtIsConst = dmrt->IsFastConstant();
zeus::CTransform dmrtXf;
if (dmrtIsConst) {
desc.x50_DMRT->GetValue(x58_frameIdx, x60_rotation);
2018-12-08 05:30:43 +00:00
dmrtXf = zeus::CTransform::RotateZ(zeus::degToRad(x60_rotation.z()));
dmrtXf.rotateLocalY(zeus::degToRad(x60_rotation.y()));
dmrtXf.rotateLocalX(zeus::degToRad(x60_rotation.x()));
}
dmrtXf = rotXf * dmrtXf;
if (CVectorElement* dmopo = desc.x4c_DMOP.get())
dmopo->GetValue(x58_frameIdx, dmop);
zeus::CTransform worldXf = zeus::CTransform::Translate(rotXf * dmop + xc_transform.origin);
if (dmrtIsConst) {
worldXf = worldXf * dmrtXf;
} else {
if (CVectorElement* dmrt = desc.x50_DMRT.get()) {
zeus::CVector3f dmrtVec;
dmrt->GetValue(x58_frameIdx, dmrtVec);
dmrtXf = zeus::CTransform::RotateZ(zeus::degToRad(dmrtVec.z()));
dmrtXf.rotateLocalY(zeus::degToRad(dmrtVec.y()));
dmrtXf.rotateLocalX(zeus::degToRad(dmrtVec.x()));
worldXf = worldXf * rotXf * dmrtXf;
} else {
worldXf = worldXf * dmrtXf;
2018-03-17 03:41:01 +00:00
}
2018-12-08 05:30:43 +00:00
}
if (CVectorElement* dmsc = desc.x54_DMSC.get()) {
zeus::CVector3f dmscVec;
dmsc->GetValue(x58_frameIdx, dmscVec);
worldXf = worldXf * zeus::CTransform::Scale(dmscVec);
}
if (CColorElement* dmcl = desc.x58_DMCL.get())
dmcl->GetValue(x58_frameIdx, color);
CGraphics::SetModelMatrix(worldXf);
if (desc.x5c_24_DMAB) {
const CModelFlags flags(7, 0, 1, color);
2018-12-08 05:30:43 +00:00
desc.x38_DMDL.m_token->Draw(flags);
} else {
if (color.a() == 1.f) {
constexpr CModelFlags flags(0, 0, 3, zeus::skWhite);
2018-12-08 05:30:43 +00:00
desc.x38_DMDL.m_token->Draw(flags);
} else {
const CModelFlags flags(5, 0, 1, color);
2018-12-08 05:30:43 +00:00
desc.x38_DMDL.m_token->Draw(flags);
2018-03-17 03:41:01 +00:00
}
2018-12-08 05:30:43 +00:00
}
}
2018-03-17 03:41:01 +00:00
void CDecal::Render() {
2019-07-21 08:42:52 +00:00
SCOPED_GRAPHICS_DEBUG_GROUP("CDecal::Render", zeus::skYellow);
2018-12-08 05:30:43 +00:00
CGlobalRandom gr(sDecalRandom);
if (x5c_29_modelInvalid && x5c_30_quad2Invalid && x5c_31_quad1Invalid)
return;
CGraphics::DisableAllLights();
CParticleGlobals::instance()->SetEmitterTime(x58_frameIdx);
2018-12-08 05:30:43 +00:00
const CDecalDescription& desc = *x0_description;
if (desc.x0_Quads[0].x14_TEX && !x5c_31_quad1Invalid) {
CParticleGlobals::instance()->SetParticleLifetime(x3c_decalQuads[0].x4_lifetime);
CParticleGlobals::instance()->UpdateParticleLifetimeTweenValues(x58_frameIdx);
RenderQuad(x3c_decalQuads[0], desc.x0_Quads[0]);
2018-12-08 05:30:43 +00:00
}
if (desc.x0_Quads[1].x14_TEX && !x5c_30_quad2Invalid) {
CParticleGlobals::instance()->SetParticleLifetime(x3c_decalQuads[1].x4_lifetime);
CParticleGlobals::instance()->UpdateParticleLifetimeTweenValues(x58_frameIdx);
RenderQuad(x3c_decalQuads[1], desc.x0_Quads[1]);
2018-12-08 05:30:43 +00:00
}
if (desc.x38_DMDL && !x5c_29_modelInvalid) {
CParticleGlobals::instance()->SetParticleLifetime(x54_modelLifetime);
CParticleGlobals::instance()->UpdateParticleLifetimeTweenValues(x58_frameIdx);
2018-12-08 05:30:43 +00:00
RenderMdl();
}
2018-03-17 03:41:01 +00:00
}
2018-12-08 05:30:43 +00:00
void CDecal::Update(float dt) {
if (x58_frameIdx >= x3c_decalQuads[0].x4_lifetime)
x5c_31_quad1Invalid = true;
if (x58_frameIdx >= x3c_decalQuads[1].x4_lifetime)
x5c_30_quad2Invalid = true;
if (x58_frameIdx >= x54_modelLifetime)
x5c_29_modelInvalid = true;
++x58_frameIdx;
2016-08-04 22:24:28 +00:00
}
2018-12-08 05:30:43 +00:00
} // namespace urde