metaforce/Runtime/Particle/CVectorElement.cpp

300 lines
9.1 KiB
C++
Raw Permalink Normal View History

#include "Runtime/Particle/CVectorElement.hpp"
#include "Runtime/CRandom16.hpp"
#include "Runtime/Particle/CElementGen.hpp"
#include "Runtime/Particle/CGenDescription.hpp"
#include "Runtime/Particle/CParticleGlobals.hpp"
#include <zeus/Math.hpp>
2016-02-06 16:19:59 -08:00
/* Documentation at: https://wiki.axiodl.com/w/Particle_Script#Vector_Elements */
2016-02-27 22:55:05 -08:00
2021-04-10 01:42:06 -07:00
namespace metaforce {
2018-12-07 21:30:43 -08:00
CVEKeyframeEmitter::CVEKeyframeEmitter(CInputStream& in) {
x4_percent = in.ReadLong();
x8_unk1 = in.ReadLong();
xc_loop = in.ReadBool();
xd_unk2 = in.ReadBool();
x10_loopEnd = in.ReadLong();
x14_loopStart = in.ReadLong();
2018-12-07 21:30:43 -08:00
const u32 count = in.ReadLong();
2018-12-07 21:30:43 -08:00
x18_keys.reserve(count);
for (u32 i = 0; i < count; ++i) {
x18_keys.emplace_back(in.Get<zeus::CVector3f>());
}
2016-02-06 16:19:59 -08:00
}
bool CVEKeyframeEmitter::GetValue([[maybe_unused]] int frame, zeus::CVector3f& valOut) const {
2018-12-07 21:30:43 -08:00
if (!x4_percent) {
int emitterTime = CParticleGlobals::instance()->m_EmitterTime;
2018-12-07 21:30:43 -08:00
int calcKey = emitterTime;
if (xc_loop) {
if (emitterTime >= x10_loopEnd) {
int v1 = emitterTime - x14_loopStart;
int v2 = x10_loopEnd - x14_loopStart;
calcKey = v1 % v2;
calcKey += x14_loopStart;
}
} else {
int v1 = x10_loopEnd - 1;
if (v1 < emitterTime)
calcKey = v1;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
valOut = x18_keys[calcKey];
} else {
int ltPerc = CParticleGlobals::instance()->m_ParticleLifetimePercentage;
float ltPercRem = CParticleGlobals::instance()->m_ParticleLifetimePercentageRemainder;
2018-12-07 21:30:43 -08:00
if (ltPerc == 100)
valOut = x18_keys[100];
2016-02-06 16:19:59 -08:00
else
2018-12-07 21:30:43 -08:00
valOut = ltPercRem * x18_keys[ltPerc + 1] + (1.0f - ltPercRem) * x18_keys[ltPerc];
}
return false;
2016-02-06 16:19:59 -08:00
}
CVECone::CVECone(std::unique_ptr<CVectorElement>&& a, std::unique_ptr<CRealElement>&& b)
2018-12-07 21:30:43 -08:00
: x4_direction(std::move(a)), x8_magnitude(std::move(b)) {
zeus::CVector3f av;
x4_direction->GetValue(0, av);
zeus::CVector3f avNorm = av.normalized();
if (avNorm.x() > 0.8f)
xc_xVec = av.cross(zeus::CVector3f(0.f, 1.f, 0.f));
else
xc_xVec = av.cross(zeus::CVector3f(1.f, 0.f, 0.f));
x18_yVec = avNorm.cross(xc_xVec);
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVECone::GetValue(int frame, zeus::CVector3f& valOut) const {
float b;
x8_magnitude->GetValue(frame, b);
zeus::CVector3f dir;
x4_direction->GetValue(frame, dir);
float b2 = std::min(1.f, b);
float randX, randY;
do {
float rand1 = CRandom16::GetRandomNumber()->Float() - 0.5f;
randX = 2.f * b2 * rand1;
float rand2 = CRandom16::GetRandomNumber()->Float() - 0.5f;
randY = 2.f * b2 * rand2;
} while (randX * randX + randY * randY > 1.f);
valOut = xc_xVec * randX + x18_yVec * randY + dir;
return false;
2016-02-06 23:25:34 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVETimeChain::GetValue(int frame, zeus::CVector3f& valOut) const {
int v;
xc_swFrame->GetValue(frame, v);
if (frame >= v)
return x8_b->GetValue(frame, valOut);
else
return x4_a->GetValue(frame, valOut);
2016-02-06 23:25:34 -08:00
}
2016-02-06 16:19:59 -08:00
2018-12-07 21:30:43 -08:00
bool CVEAngleCone::GetValue(int frame, zeus::CVector3f& valOut) const {
float xc, yc, xr, yr;
x4_angleXConstant->GetValue(frame, xc);
x8_angleYConstant->GetValue(frame, yc);
xc_angleXRange->GetValue(frame, xr);
x10_angleYRange->GetValue(frame, yr);
2016-02-06 16:19:59 -08:00
2018-12-07 21:30:43 -08:00
float xtmp = CRandom16::GetRandomNumber()->Float() * xr;
float xang = zeus::degToRad(0.5f * xr - xtmp + xc);
2016-02-06 16:19:59 -08:00
2018-12-07 21:30:43 -08:00
float ytmp = CRandom16::GetRandomNumber()->Float() * yr;
float yang = zeus::degToRad(0.5f * yr - ytmp + yc);
2016-02-06 23:25:34 -08:00
2018-12-07 21:30:43 -08:00
float mag;
x14_magnitude->GetValue(frame, mag);
2016-02-06 23:25:34 -08:00
2018-12-07 21:30:43 -08:00
/* This takes a +Z vector and rotates it around X and Y axis (like a rotation matrix would) */
valOut = zeus::CVector3f(std::cos(xang) * -std::sin(yang), std::sin(xang), std::cos(xang) * std::cos(yang)) *
zeus::CVector3f(mag);
return false;
2016-02-06 23:25:34 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEAdd::GetValue(int frame, zeus::CVector3f& valOut) const {
zeus::CVector3f a, b;
x4_a->GetValue(frame, a);
x8_b->GetValue(frame, b);
valOut = a + b;
return false;
2016-02-06 23:25:34 -08:00
}
CVECircleCluster::CVECircleCluster(std::unique_ptr<CVectorElement>&& a, std::unique_ptr<CVectorElement>&& b,
std::unique_ptr<CIntElement>&& c, std::unique_ptr<CRealElement>&& d)
2018-12-07 21:30:43 -08:00
: x4_a(std::move(a)), x24_magnitude(std::move(d)) {
int cv;
c->GetValue(0, cv);
x20_deltaAngle = zeus::degToRad(360.f / float(cv));
zeus::CVector3f bv;
b->GetValue(0, bv);
bv.normalize();
if (bv[0] > 0.8f)
x8_xVec = bv.cross(zeus::CVector3f(0.f, 1.f, 0.f));
else
x8_xVec = bv.cross(zeus::CVector3f(1.f, 0.f, 0.f));
x14_yVec = bv.cross(x8_xVec);
2016-02-07 15:59:05 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVECircleCluster::GetValue(int frame, zeus::CVector3f& valOut) const {
zeus::CVector3f av;
x4_a->GetValue(frame, av);
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
float curAngle = frame * x20_deltaAngle;
zeus::CVector3f x = x8_xVec * std::cos(curAngle);
zeus::CVector3f y = x14_yVec * std::sin(curAngle);
zeus::CVector3f tv = x + y + av;
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
float dv;
x24_magnitude->GetValue(frame, dv);
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
zeus::CVector3f magVec(dv * tv.magnitude());
zeus::CVector3f rv =
magVec * zeus::CVector3f(CRandom16::GetRandomNumber()->Float(), CRandom16::GetRandomNumber()->Float(),
CRandom16::GetRandomNumber()->Float());
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
valOut = tv + rv;
return false;
2016-02-07 15:59:05 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEConstant::GetValue(int frame, zeus::CVector3f& valOut) const {
float a, b, c;
x4_a->GetValue(frame, a);
x8_b->GetValue(frame, b);
xc_c->GetValue(frame, c);
valOut = zeus::CVector3f(a, b, c);
return false;
2016-02-07 15:59:05 -08:00
}
bool CVEFastConstant::GetValue([[maybe_unused]] int frame, zeus::CVector3f& valOut) const {
2018-12-07 21:30:43 -08:00
valOut = x4_val;
return false;
2016-02-07 15:59:05 -08:00
}
CVECircle::CVECircle(std::unique_ptr<CVectorElement>&& a, std::unique_ptr<CVectorElement>&& b,
std::unique_ptr<CRealElement>&& c, std::unique_ptr<CRealElement>&& d,
std::unique_ptr<CRealElement>&& e)
2018-12-07 21:30:43 -08:00
: x4_direction(std::move(a)), x20_angleConstant(std::move(c)), x24_angleLinear(std::move(d)), x28_radius(std::move(e)) {
zeus::CVector3f bv;
b->GetValue(0, bv);
bv.normalize();
if (bv[0] > 0.8f)
x8_xVec = bv.cross(zeus::CVector3f(0.f, 1.f, 0.f));
else
x8_xVec = bv.cross(zeus::CVector3f(1.f, 0.f, 0.f));
x14_yVec = bv.cross(x8_xVec);
2016-02-07 15:59:05 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVECircle::GetValue(int frame, zeus::CVector3f& valOut) const {
float c, d, e;
x20_angleConstant->GetValue(frame, c);
x24_angleLinear->GetValue(frame, d);
x28_radius->GetValue(frame, e);
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
float curAngle = zeus::degToRad(d * frame + c);
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
zeus::CVector3f av;
x4_direction->GetValue(frame, av);
2016-02-07 15:59:05 -08:00
2018-12-07 21:30:43 -08:00
zeus::CVector3f x = x8_xVec * e * std::cos(curAngle);
zeus::CVector3f y = x14_yVec * e * std::sin(curAngle);
2016-02-06 23:25:34 -08:00
2018-12-07 21:30:43 -08:00
valOut = x + y + av;
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEMultiply::GetValue(int frame, zeus::CVector3f& valOut) const {
zeus::CVector3f a, b;
x4_a->GetValue(frame, a);
x8_b->GetValue(frame, b);
valOut = a * b;
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVERealToVector::GetValue(int frame, zeus::CVector3f& valOut) const {
float a;
x4_a->GetValue(frame, a);
valOut = zeus::CVector3f(a);
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEPulse::GetValue(int frame, zeus::CVector3f& valOut) const {
int a, b;
x4_aDuration->GetValue(frame, a);
x8_bDuration->GetValue(frame, b);
2020-04-12 02:31:00 -07:00
int cv = a + b + 1;
if (cv < 0) {
cv = 1;
}
2018-12-07 21:30:43 -08:00
2020-04-12 02:31:00 -07:00
if (b < 1 || frame % cv <= a) {
2018-12-07 21:30:43 -08:00
xc_aVal->GetValue(frame, valOut);
2020-04-12 02:31:00 -07:00
} else {
x10_bVal->GetValue(frame, valOut);
}
2018-12-07 21:30:43 -08:00
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleVelocity::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
valOut = CElementGen::g_currentParticle->x1c_vel;
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleColor::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
valOut = CElementGen::g_currentParticle->x10_prevPos;
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleLocation::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
valOut = CElementGen::g_currentParticle->x4_pos;
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleSystemOrientationFront::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
zeus::CMatrix4f trans =
CParticleGlobals::instance()->m_currentParticleSystem->x4_system->GetOrientation().toMatrix4f().transposed();
2018-12-07 21:30:43 -08:00
valOut.assign(trans.m[0].y(), trans.m[1].y(), trans.m[2].y());
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleSystemOrientationUp::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
zeus::CMatrix4f trans =
CParticleGlobals::instance()->m_currentParticleSystem->x4_system->GetOrientation().toMatrix4f().transposed();
2018-12-07 21:30:43 -08:00
valOut.assign(trans.m[0].z(), trans.m[1].z(), trans.m[2].z());
return false;
2016-02-11 22:06:17 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleSystemOrientationRight::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
zeus::CMatrix4f trans =
CParticleGlobals::instance()->m_currentParticleSystem->x4_system->GetOrientation().toMatrix4f().transposed();
2018-12-07 21:30:43 -08:00
valOut.assign(trans.m[0].x(), trans.m[1].x(), trans.m[2].x());
return false;
2016-02-11 22:06:17 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEParticleSystemTranslation::GetValue(int /*frame*/, zeus::CVector3f& valOut) const {
valOut = CParticleGlobals::instance()->m_currentParticleSystem->x4_system->GetTranslation();
2018-12-07 21:30:43 -08:00
return false;
2016-02-06 16:19:59 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVESubtract::GetValue(int frame, zeus::CVector3f& valOut) const {
zeus::CVector3f a, b;
x4_a->GetValue(frame, a);
x8_b->GetValue(frame, b);
valOut = a - b;
return false;
2016-02-11 22:06:17 -08:00
}
2018-12-07 21:30:43 -08:00
bool CVEColorToVector::GetValue(int frame, zeus::CVector3f& valOut) const {
zeus::CColor val = {0.0f, 0.0f, 0.0f, 1.0f};
x4_a->GetValue(frame, val);
valOut = zeus::CVector3f{val.mSimd};
return false;
2016-02-17 00:07:32 -08:00
}
2021-04-10 01:42:06 -07:00
} // namespace metaforce