You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
910 lines
27 KiB
910 lines
27 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
GNU General Public License for more details. |
|
|
|
You should have received a copy of the GNU General Public License |
|
along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
|
|
Additional permissions under GNU GPL version 3 section 7 |
|
|
|
If you modify this Program, or any covered work, by linking or |
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, |
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent |
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK |
|
(or a modified version of those libraries), |
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, |
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG |
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the |
|
licensors of this Program grant you additional |
|
permission to convey the resulting work. Corresponding Source for a |
|
non-source form of such a combination shall include the source code for |
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered |
|
work. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
#include "HeadSpin.h" |
|
#include "hsGeometry3.h" |
|
#include "plParticle.h" |
|
#include "plParticleEffect.h" |
|
#include "plEffectTargetInfo.h" |
|
#include "plConvexVolume.h" |
|
#include "plBoundInterface.h" |
|
#include "hsResMgr.h" |
|
#include "plPipeline.h" |
|
#include "hsFastMath.h" |
|
#include "pnEncryption/plRandom.h" |
|
#include "plParticleSystem.h" |
|
#include "plMessage/plParticleUpdateMsg.h" |
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
plParticleCollisionEffect::plParticleCollisionEffect() |
|
{ |
|
fBounds = nil; |
|
fSceneObj = nil; |
|
} |
|
|
|
plParticleCollisionEffect::~plParticleCollisionEffect() |
|
{ |
|
} |
|
|
|
void plParticleCollisionEffect::PrepareEffect(const plEffectTargetInfo &target) |
|
{ |
|
if (fBounds == nil) |
|
{ |
|
plBoundInterface *bi = plBoundInterface::ConvertNoRef(fSceneObj->GetGenericInterface(plBoundInterface::Index())); |
|
if (bi == nil) |
|
return; |
|
fBounds = bi->GetVolume(); |
|
} |
|
} |
|
|
|
bool plParticleCollisionEffect::MsgReceive(plMessage* msg) |
|
{ |
|
plRefMsg* refMsg = plRefMsg::ConvertNoRef(msg); |
|
plSceneObject *so; |
|
if (refMsg) |
|
{ |
|
if ((so = plSceneObject::ConvertNoRef(refMsg->GetRef()))) |
|
{ |
|
if( refMsg->GetContext() & (plRefMsg::kOnCreate|plRefMsg::kOnRequest|plRefMsg::kOnReplace) ) |
|
fSceneObj = so; |
|
else |
|
fSceneObj = nil; |
|
return true; |
|
} |
|
} |
|
return hsKeyedObject::MsgReceive(msg); |
|
} |
|
|
|
void plParticleCollisionEffect::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
hsKeyedObject::Read(s, mgr); |
|
|
|
plGenRefMsg* msg; |
|
msg = new plGenRefMsg(GetKey(), plRefMsg::kOnCreate, 0, 0); // SceneObject |
|
mgr->ReadKeyNotifyMe(s, msg, plRefFlags::kActiveRef); |
|
fBounds = nil; |
|
} |
|
|
|
void plParticleCollisionEffect::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
hsKeyedObject::Write(s, mgr); |
|
|
|
mgr->WriteKey(s, fSceneObj); |
|
} |
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
// Some permutations on the CollisionEffect follow |
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
|
|
plParticleCollisionEffectBeat::plParticleCollisionEffectBeat() |
|
{ |
|
} |
|
|
|
bool plParticleCollisionEffectBeat::ApplyEffect(const plEffectTargetInfo &target, int32_t i) |
|
{ |
|
hsAssert(i >= 0, "Use of default argument doesn't make sense for plParticleCollisionEffect"); |
|
|
|
if( !fBounds ) |
|
return false; |
|
|
|
hsPoint3 *currPos = (hsPoint3 *)(target.fPos + i * target.fPosStride); |
|
fBounds->ResolvePoint(*currPos); |
|
|
|
return false; |
|
} |
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
|
|
plParticleCollisionEffectDie::plParticleCollisionEffectDie() |
|
{ |
|
} |
|
|
|
bool plParticleCollisionEffectDie::ApplyEffect(const plEffectTargetInfo &target, int32_t i) |
|
{ |
|
hsAssert(i >= 0, "Use of default argument doesn't make sense for plParticleCollisionEffect"); |
|
|
|
if( !fBounds ) |
|
return false; |
|
|
|
hsPoint3 *currPos = (hsPoint3 *)(target.fPos + i * target.fPosStride); |
|
return fBounds->IsInside(*currPos); |
|
} |
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
|
|
plParticleCollisionEffectBounce::plParticleCollisionEffectBounce() |
|
: fBounce(1.f), |
|
fFriction(0.f) |
|
{ |
|
} |
|
|
|
bool plParticleCollisionEffectBounce::ApplyEffect(const plEffectTargetInfo &target, int32_t i) |
|
{ |
|
hsAssert(i >= 0, "Use of default argument doesn't make sense for plParticleCollisionEffect"); |
|
|
|
if( !fBounds ) |
|
return false; |
|
|
|
hsPoint3* currPos = (hsPoint3 *)(target.fPos + i * target.fPosStride); |
|
hsVector3* currVel = (hsVector3*)(target.fVelocity + i * target.fVelocityStride); |
|
fBounds->BouncePoint(*currPos, *currVel, fBounce, fFriction); |
|
|
|
return false; |
|
} |
|
|
|
void plParticleCollisionEffectBounce::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleCollisionEffect::Read(s, mgr); |
|
|
|
fBounce = s->ReadLEScalar(); |
|
fFriction = s->ReadLEScalar(); |
|
} |
|
|
|
void plParticleCollisionEffectBounce::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleCollisionEffect::Write(s, mgr); |
|
|
|
s->WriteLEScalar(fBounce); |
|
s->WriteLEScalar(fFriction); |
|
} |
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
/////////////////////////////////////////////////////////////////////////////////////////// |
|
|
|
plParticleFadeVolumeEffect::plParticleFadeVolumeEffect() : fLength(100.0f), fIgnoreZ(true) |
|
{ |
|
} |
|
|
|
plParticleFadeVolumeEffect::~plParticleFadeVolumeEffect() |
|
{ |
|
} |
|
|
|
// |
|
// It's not really clear looking at the math here what's actually going on, |
|
// but once you visualize it, it's pretty easy to follow. So the camera position, |
|
// view direction, and length of the fade volume define a sphere, where the camera |
|
// position is a point on the sphere, the view direction points from that surface |
|
// point toward the center, and the length is the sphere's radius. Since the view |
|
// direction points straight through the sphere, that sphere is the sweet zone for |
|
// putting our particles to pile them up in front of the camera. But we'd like to |
|
// do this independently for each axis (for efficiency, rather than true 3D calculations), |
|
// so we put an axis aligned box around the sphere, and that's the volume we restrict |
|
// our particles to. |
|
// Now we could fade all around the box, but that's not really what we want, because |
|
// that means fading particles that are behind us. And in the case where we're looking |
|
// along an axis, the camera is pushed up against a face of the box (where the axis |
|
// aligned box is tangent to the inscribed sphere), so we'd actually be fading |
|
// particles just in front of the camera. Because of this non-symmetry, we're going to |
|
// define the Max in a given dimension as the world space value for that dimension |
|
// FARTHEST from the camera (NOT largest in value). So if the camera is looking |
|
// in a negative direction in one dimension, the Max will be less than the Min for |
|
// that dimension. |
|
// So we set up our Max's and Min's for each dimension in PrepareEffect(), and then |
|
// at runtime we calculate the parameter value of the particle ranging from 0 where |
|
// particleLoc == Min to 1 where particleLoc == Max. If the parameter is outside |
|
// [0..1], then we can move it into the box using the fractional part of the parameter. |
|
// Finally, if the (possibly relocated) parameter value says the particle is approaching |
|
// the Max value, we can calclulate its faded opacity from the parameter. |
|
// |
|
|
|
// Need to experiment to minimize this fade distance. The greater |
|
// the fade distance, the more faded out (wasted) particles we're drawing. |
|
// The shorter the distance, the more noticable the fade out. |
|
// Note the wierdness between the fractions, because kFadeFrac is fraction |
|
// of fLength, but kFadeParm and kInvFadeFrac are fraction of 2.f*fLength. Sorry. |
|
const float kFadeFrac = 0.5f; |
|
const float kFadeParm = 1.f - kFadeFrac * 0.5f; |
|
const float kInvFadeFrac = 1.f / (kFadeFrac * 0.5f); |
|
|
|
void plParticleFadeVolumeEffect::PrepareEffect(const plEffectTargetInfo &target) |
|
{ |
|
hsPoint3 viewLoc = target.fContext.fPipeline->GetViewPositionWorld(); |
|
hsVector3 viewDir = target.fContext.fPipeline->GetViewDirWorld(); |
|
|
|
// Nuking out the setting of viewDir.fZ to 0 when we aren't centering |
|
// about Z (fIgnoreZ == true), because we still want to center our |
|
// volume about the camera (rather than push the camera to the edge of |
|
// the cylinder) in that case, so we don't get artifacts when we look |
|
// straight up or down. mf |
|
|
|
hsPoint3 signs(viewDir.fX >= 0 ? 1.f : -1.f, viewDir.fY >= 0 ? 1.f : -1.f, viewDir.fZ >= 0 ? 1.f : -1.f); |
|
|
|
fMax.fX = viewLoc.fX + (viewDir.fX + signs.fX) * fLength; |
|
fMin.fX = fMax.fX - 2.f * signs.fX * fLength; |
|
|
|
fMax.fY = viewLoc.fY + (viewDir.fY + signs.fY) * fLength; |
|
fMin.fY = fMax.fY - 2.f * signs.fY * fLength; |
|
|
|
fMax.fZ = viewLoc.fZ + (viewDir.fZ + signs.fZ) * fLength; |
|
fMin.fZ = fMax.fZ - 2.f * signs.fZ * fLength; |
|
|
|
fNorm.fX = 1.f / (fMax.fX - fMin.fX); |
|
fNorm.fY = 1.f / (fMax.fY - fMin.fY); |
|
fNorm.fZ = 1.f / (fMax.fZ - fMin.fZ); |
|
} |
|
|
|
bool plParticleFadeVolumeEffect::ApplyEffect(const plEffectTargetInfo& target, int32_t i) |
|
{ |
|
hsPoint3 *currPos = (hsPoint3 *)(target.fPos + i * target.fPosStride); |
|
|
|
float parm; |
|
|
|
float fade = 1.f; |
|
|
|
parm = (currPos->fX - fMin.fX) * fNorm.fX; |
|
if( parm < 0 ) |
|
{ |
|
parm -= int(parm); |
|
currPos->fX = fMax.fX + parm * (fMax.fX - fMin.fX); |
|
parm += 1.f; |
|
} |
|
else if( parm > 1.f ) |
|
{ |
|
parm -= int(parm); |
|
currPos->fX = fMin.fX + parm * (fMax.fX - fMin.fX); |
|
} |
|
if( parm > kFadeParm ) |
|
{ |
|
parm = 1.f - parm; |
|
parm *= kInvFadeFrac; |
|
if( parm < fade ) |
|
fade = parm; |
|
} |
|
|
|
parm = (currPos->fY - fMin.fY) * fNorm.fY; |
|
if( parm < 0 ) |
|
{ |
|
parm -= int(parm); |
|
currPos->fY = fMax.fY + parm * (fMax.fY - fMin.fY); |
|
parm += 1.f; |
|
} |
|
else if( parm > 1.f ) |
|
{ |
|
parm -= int(parm); |
|
currPos->fY = fMin.fY + parm * (fMax.fY - fMin.fY); |
|
} |
|
if( parm > kFadeParm ) |
|
{ |
|
parm = 1.f - parm; |
|
parm *= kInvFadeFrac; |
|
if( parm < fade ) |
|
fade = parm; |
|
} |
|
|
|
if( !fIgnoreZ ) |
|
{ |
|
parm = (currPos->fZ - fMin.fZ) * fNorm.fZ; |
|
if( parm < 0 ) |
|
{ |
|
parm -= int(parm); |
|
currPos->fZ = fMax.fZ + parm * (fMax.fZ - fMin.fZ); |
|
parm += 1.f; |
|
} |
|
else if( parm > 1.f ) |
|
{ |
|
parm -= int(parm); |
|
currPos->fZ = fMin.fZ + parm * (fMax.fZ - fMin.fZ); |
|
} |
|
if( parm > kFadeParm ) |
|
{ |
|
parm = 1.f - parm; |
|
parm *= kInvFadeFrac; |
|
if( parm < fade ) |
|
fade = parm; |
|
} |
|
} |
|
|
|
if( fade < 1.f ) |
|
{ |
|
uint32_t *color = (uint32_t *)(target.fColor + i * target.fColorStride); |
|
uint32_t alpha = (uint32_t)((*color >> 24) * fade); |
|
*color = (*color & 0x00ffffff) | (alpha << 24); |
|
} |
|
|
|
return false; |
|
} |
|
|
|
void plParticleFadeVolumeEffect::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
hsKeyedObject::Read(s, mgr); |
|
|
|
fLength = s->ReadLEScalar(); |
|
fIgnoreZ = s->ReadBool(); |
|
} |
|
|
|
void plParticleFadeVolumeEffect::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
hsKeyedObject::Write(s, mgr); |
|
|
|
s->WriteLEScalar(fLength); |
|
s->WriteBool(fIgnoreZ); |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////////// |
|
//////////////////////////////////////////////////////////////////////// |
|
// Particle wind - Base class first |
|
plParticleWindEffect::plParticleWindEffect() |
|
: fWindVec(0,0,0), |
|
fDir(1.f,0,0), |
|
fSwirl(0.1f), |
|
fConstancy(0), |
|
fHorizontal(0), |
|
fLastDirSecs(-1.f), |
|
fRefDir(0.f,0.f,0.f), |
|
fRandDir(1.f,0.f,0.f) |
|
{ |
|
} |
|
|
|
plParticleWindEffect::~plParticleWindEffect() |
|
{ |
|
} |
|
|
|
void plParticleWindEffect::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
hsKeyedObject::Read(s, mgr); |
|
|
|
fStrength = s->ReadLEScalar(); |
|
fConstancy = s->ReadLEScalar(); |
|
fSwirl = s->ReadLEScalar(); |
|
fHorizontal = s->ReadBool(); |
|
fRefDir.Read(s); |
|
fDir.Read(s); |
|
fRandDir = fDir; |
|
} |
|
|
|
void plParticleWindEffect::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
hsKeyedObject::Write(s, mgr); |
|
|
|
s->WriteLEScalar(fStrength); |
|
s->WriteLEScalar(fConstancy); |
|
s->WriteLEScalar(fSwirl); |
|
s->WriteBool(fHorizontal); |
|
fRefDir.Write(s); |
|
fDir.Write(s); |
|
} |
|
|
|
void plParticleWindEffect::SetRefDirection(const hsVector3& v) |
|
{ |
|
fRefDir = v; |
|
float lenSq = fRefDir.MagnitudeSquared(); |
|
if( lenSq > 1.e-1f ) |
|
{ |
|
fDir = fRefDir * hsFastMath::InvSqrtAppr(lenSq); |
|
fRandDir = fDir; |
|
} |
|
} |
|
|
|
void plParticleWindEffect::PrepareEffect(const plEffectTargetInfo& target) |
|
{ |
|
if( fLastDirSecs != target.fContext.fSecs ) |
|
{ |
|
static plRandom random; |
|
fRandDir.fX += random.RandMinusOneToOne() * fSwirl; |
|
fRandDir.fY += random.RandMinusOneToOne() * fSwirl; |
|
if( !GetHorizontal() ) |
|
fRandDir.fZ += random.RandMinusOneToOne() * fSwirl; |
|
hsFastMath::NormalizeAppr(fRandDir); |
|
fDir = fRandDir + fRefDir; |
|
hsFastMath::NormalizeAppr(fDir); |
|
|
|
fWindVec = fDir * (fStrength * target.fContext.fSystem->GetWindMult() * target.fContext.fDelSecs); |
|
|
|
fLastDirSecs = target.fContext.fSecs; |
|
} |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////////// |
|
// Localized wind (how much wind you're getting depends on where you are |
|
plParticleLocalWind::plParticleLocalWind() |
|
: fScale(0, 0, 0), |
|
fSpeed(0), |
|
fPhase(0,0,0), |
|
fInvScale(0,0,0), |
|
fLastPhaseSecs(-1.f) |
|
{ |
|
} |
|
|
|
plParticleLocalWind::~plParticleLocalWind() |
|
{ |
|
} |
|
|
|
void plParticleLocalWind::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleWindEffect::Read(s, mgr); |
|
|
|
fScale.Read(s); |
|
fSpeed = s->ReadLEScalar(); |
|
} |
|
|
|
void plParticleLocalWind::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleWindEffect::Write(s, mgr); |
|
|
|
fScale.Write(s); |
|
s->WriteLEScalar(fSpeed); |
|
} |
|
|
|
void plParticleLocalWind::PrepareEffect(const plEffectTargetInfo& target) |
|
{ |
|
if( fLastPhaseSecs != target.fContext.fSecs ) |
|
{ |
|
plParticleWindEffect::PrepareEffect(target); |
|
|
|
fPhase += fDir * fSpeed * target.fContext.fDelSecs; |
|
|
|
fInvScale.fX = fScale.fX > 0 ? 1.f / fScale.fX : 0; |
|
fInvScale.fY = fScale.fY > 0 ? 1.f / fScale.fY : 0; |
|
fInvScale.fZ = fScale.fZ > 0 ? 1.f / fScale.fZ : 0; |
|
|
|
fLastPhaseSecs = target.fContext.fSecs; |
|
} |
|
} |
|
|
|
|
|
bool plParticleLocalWind::ApplyEffect(const plEffectTargetInfo& target, int32_t i) |
|
{ |
|
const hsPoint3& pos = *(hsPoint3 *)(target.fPos + i * target.fPosStride); |
|
hsVector3& vel = *(hsVector3*)(target.fVelocity + i * target.fVelocityStride); |
|
|
|
const float kMinToBother = 0; |
|
|
|
float strength = 1.f / ( (1.f + fConstancy) * (1.f + fConstancy) ); |
|
float s, c, t; |
|
t = (pos[0] - fPhase[0]) * fInvScale[0]; |
|
hsFastMath::SinCosAppr(t, s, c); |
|
c += fConstancy; |
|
if( c <= kMinToBother ) |
|
return false; |
|
strength *= c; |
|
|
|
t = (pos[1] - fPhase[1]) * fInvScale[1]; |
|
hsFastMath::SinCosAppr(t, s, c); |
|
c += fConstancy; |
|
if( c <= kMinToBother ) |
|
return false; |
|
strength *= c; |
|
|
|
#if 0 // if you turn this back on, strength needs to drop by another factor of (1.f + fConstancy) |
|
t = (pos[2] - fPhase[2]) * fInvScale[2]; |
|
hsFastMath::SinCosAppr(t, s, c); |
|
c += fConstancy; |
|
if( c <= kMinToBother ) |
|
return false; |
|
strength *= c; |
|
#endif |
|
|
|
const float& invMass = *(float*)(target.fInvMass + i * target.fInvMassStride); |
|
strength *= invMass; |
|
|
|
vel += fWindVec * strength; |
|
|
|
return false; |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////////// |
|
// Uniform wind - wind changes over time, but not space |
|
plParticleUniformWind::plParticleUniformWind() |
|
: fFreqMin(0.1f), |
|
fFreqMax(0.2f), |
|
fFreqCurr(0.1f), |
|
fFreqRate(0.05f), |
|
|
|
fCurrPhase(0), |
|
fLastFreqSecs(-1.f), |
|
fCurrentStrength(0) |
|
{ |
|
} |
|
|
|
plParticleUniformWind::~plParticleUniformWind() |
|
{ |
|
} |
|
|
|
void plParticleUniformWind::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleWindEffect::Read(s, mgr); |
|
|
|
fFreqMin = s->ReadLEScalar(); |
|
fFreqMax = s->ReadLEScalar(); |
|
fFreqRate = s->ReadLEScalar(); |
|
|
|
#if 0 |
|
fFreqMin = 1.f / 6.f; |
|
fFreqMax = 1.f / 1.f; |
|
fConstancy = -0.5f; |
|
fSwirl = 0.05f; |
|
#endif |
|
|
|
fFreqCurr = fFreqMin; |
|
} |
|
|
|
void plParticleUniformWind::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleWindEffect::Write(s, mgr); |
|
|
|
s->WriteLEScalar(fFreqMin); |
|
s->WriteLEScalar(fFreqMax); |
|
s->WriteLEScalar(fFreqRate); |
|
} |
|
|
|
void plParticleUniformWind::SetFrequencyRange(float minSecsPerCycle, float maxSecsPerCycle) |
|
{ |
|
const float kMinSecsPerCycle = 1.f; |
|
if( minSecsPerCycle < kMinSecsPerCycle ) |
|
minSecsPerCycle = kMinSecsPerCycle; |
|
if( minSecsPerCycle < kMinSecsPerCycle ) |
|
minSecsPerCycle = kMinSecsPerCycle; |
|
|
|
if( minSecsPerCycle < maxSecsPerCycle ) |
|
{ |
|
fFreqMin = 1.f / maxSecsPerCycle; |
|
fFreqMax = 1.f / minSecsPerCycle; |
|
} |
|
else |
|
{ |
|
fFreqMin = 1.f / minSecsPerCycle; |
|
fFreqMax = 1.f / maxSecsPerCycle; |
|
} |
|
} |
|
|
|
void plParticleUniformWind::SetFrequencyRate(float secsPerCycle) |
|
{ |
|
const float kMinSecsPerCycle = 1.f; |
|
if( secsPerCycle < kMinSecsPerCycle ) |
|
secsPerCycle = kMinSecsPerCycle; |
|
fFreqRate = 1.f / secsPerCycle; |
|
} |
|
|
|
|
|
void plParticleUniformWind::PrepareEffect(const plEffectTargetInfo& target) |
|
{ |
|
plParticleWindEffect::PrepareEffect(target); |
|
|
|
if( fLastFreqSecs != target.fContext.fSecs ) |
|
{ |
|
static plRandom random; |
|
|
|
const double kTwoPi = M_PI * 2.0; |
|
double t0 = fFreqCurr * fLastFreqSecs + fCurrPhase; |
|
float t1 = (float)fmod(t0, kTwoPi); |
|
fCurrPhase -= t0 - t1; |
|
|
|
fFreqCurr += fFreqRate * target.fContext.fDelSecs * random.RandZeroToOne(); |
|
|
|
if( fFreqCurr > fFreqMax ) |
|
{ |
|
fFreqCurr = fFreqMax; |
|
fFreqRate = -fFreqRate; |
|
} |
|
else if( fFreqCurr < fFreqMin ) |
|
{ |
|
fFreqCurr = fFreqMin; |
|
fFreqRate = -fFreqRate; |
|
} |
|
|
|
float phaseDel = (float)(t1 - (fFreqCurr * fLastFreqSecs + fCurrPhase)); |
|
fCurrPhase += phaseDel; |
|
|
|
float t = float(fFreqCurr * target.fContext.fSecs + fCurrPhase); |
|
float s; |
|
hsFastMath::SinCosAppr(t, s, fCurrentStrength); |
|
fCurrentStrength += fConstancy; |
|
fCurrentStrength /= (1.f + fConstancy); |
|
|
|
if( fCurrentStrength < 0 ) |
|
fCurrentStrength = 0; |
|
|
|
fLastFreqSecs = target.fContext.fSecs; |
|
} |
|
} |
|
|
|
|
|
bool plParticleUniformWind::ApplyEffect(const plEffectTargetInfo& target, int32_t i) |
|
{ |
|
hsVector3& vel = *(hsVector3*)(target.fVelocity + i * target.fVelocityStride); |
|
|
|
const float& invMass = *(float*)(target.fInvMass + i * target.fInvMassStride); |
|
|
|
vel += fWindVec * (invMass * fCurrentStrength); |
|
|
|
return false; |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////////// |
|
// Simplified flocking. |
|
|
|
plParticleFlockEffect::plParticleFlockEffect() : |
|
fInfAvgRadSq(1), |
|
fInfRepRadSq(1), |
|
fAvgVelStr(1), |
|
fRepDirStr(1), |
|
fGoalOrbitStr(1), |
|
fGoalChaseStr(1), |
|
fGoalDistSq(1), |
|
fFullChaseDistSq(1), |
|
fMaxOrbitSpeed(1), |
|
fMaxChaseSpeed(1), |
|
fMaxParticles(0), |
|
fDistSq(nil), |
|
fInfluences(nil) |
|
{ |
|
fTargetOffset.Set(0.f, 0.f, 0.f); |
|
fDissenterTarget.Set(0.f, 0.f, 0.f); |
|
} |
|
|
|
plParticleFlockEffect::~plParticleFlockEffect() |
|
{ |
|
SetMaxParticles(0); |
|
} |
|
|
|
void plParticleFlockEffect::IUpdateDistances(const plEffectTargetInfo& target) |
|
{ |
|
uint32_t numParticles = std::min(static_cast<uint32_t>(fMaxParticles), target.fNumValidParticles); |
|
|
|
for (uint32_t i = 0; i < numParticles; i++) |
|
{ |
|
for (uint32_t j = i + 1; j < numParticles; j++) |
|
{ |
|
hsVector3 diff((hsPoint3*)(target.fPos + i * target.fPosStride), (hsPoint3*)(target.fPos + j * target.fPosStride)); |
|
fDistSq[i * fMaxParticles + j] = fDistSq[j * fMaxParticles + i] = diff.MagnitudeSquared(); |
|
} |
|
} |
|
} |
|
|
|
void plParticleFlockEffect::IUpdateInfluences(const plEffectTargetInfo &target) |
|
{ |
|
uint32_t numParticles = std::min(static_cast<uint32_t>(fMaxParticles), target.fNumValidParticles); |
|
|
|
for (uint32_t i = 0; i < numParticles; i++) |
|
{ |
|
int numAvg = 0; |
|
int numRep = 0; |
|
fInfluences[i].fAvgVel.Set(0.f, 0.f, 0.f); |
|
fInfluences[i].fRepDir.Set(0.f, 0.f, 0.f); |
|
|
|
for (uint32_t j = 0; j < numParticles; j++) |
|
{ |
|
if (i == j) |
|
continue; |
|
|
|
const int distIdx = i * fMaxParticles + j; |
|
if (fDistSq[distIdx] > fInfAvgRadSq) |
|
{ |
|
numAvg++; |
|
fInfluences[i].fAvgVel += *(hsVector3*)(target.fVelocity + j * target.fVelocityStride); |
|
} |
|
|
|
if (fDistSq[distIdx] > fInfRepRadSq) |
|
{ |
|
numRep++; |
|
hsVector3 repDir((hsPoint3*)(target.fPos + i * target.fPosStride), (hsPoint3*)(target.fPos + j * target.fPosStride)); |
|
repDir.Normalize(); |
|
fInfluences[i].fRepDir += repDir; |
|
} |
|
|
|
} |
|
|
|
if (numAvg > 0) |
|
fInfluences[i].fAvgVel /= (float)numAvg; |
|
if (numRep > 0) |
|
fInfluences[i].fRepDir /= (float)numRep; |
|
} |
|
} |
|
|
|
void plParticleFlockEffect::PrepareEffect(const plEffectTargetInfo& target) |
|
{ |
|
IUpdateDistances(target); |
|
IUpdateInfluences(target); |
|
} |
|
|
|
// Some of this is the same for every particle and should be cached in PrepareEffect(). |
|
// Holding off on that until I like the behavior. |
|
bool plParticleFlockEffect::ApplyEffect(const plEffectTargetInfo& target, int32_t i) |
|
{ |
|
if (i >= fMaxParticles) |
|
return false; // Don't have the memory to deal with you. Good luck kid... |
|
|
|
const hsPoint3 &pos = *(hsPoint3*)(target.fPos + i * target.fPosStride); |
|
hsVector3 &vel = *(hsVector3*)(target.fVelocity + i * target.fVelocityStride); |
|
|
|
float curSpeed = vel.Magnitude(); |
|
hsPoint3 goal; |
|
if (*(uint32_t*)(target.fMiscFlags + i * target.fMiscFlagsStride) & plParticleExt::kImmortal) |
|
goal = target.fContext.fSystem->GetTarget(0)->GetLocalToWorld().GetTranslate() + fTargetOffset; |
|
else |
|
goal = fDissenterTarget; |
|
|
|
hsVector3 goalDir; |
|
hsPoint3 tmp = goal - pos; |
|
goalDir.Set(&tmp); |
|
float distSq = goalDir.MagnitudeSquared(); |
|
|
|
goalDir.Normalize(); |
|
|
|
float goalStr; |
|
float maxSpeed; |
|
float maxSpeedSq; |
|
if (distSq <= fGoalDistSq) |
|
{ |
|
goalStr = fGoalOrbitStr; |
|
if (i & 0x1) |
|
goalDir.Set(goalDir.fY, -goalDir.fX, goalDir.fZ); |
|
else |
|
goalDir.Set(-goalDir.fY, goalDir.fX, goalDir.fZ); |
|
|
|
maxSpeed = fMaxOrbitSpeed; |
|
} |
|
else if (distSq >= fFullChaseDistSq) |
|
{ |
|
goalStr = fGoalChaseStr; |
|
maxSpeed = fMaxChaseSpeed; |
|
} |
|
else |
|
{ |
|
float pct = (distSq - fGoalDistSq) / (fFullChaseDistSq - fGoalDistSq); |
|
goalStr = fGoalOrbitStr + (fGoalChaseStr - fGoalOrbitStr) * pct; |
|
maxSpeed = fMaxOrbitSpeed + (fMaxChaseSpeed - fMaxOrbitSpeed) * pct; |
|
} |
|
maxSpeedSq = maxSpeed * maxSpeed; |
|
|
|
vel += (fInfluences[i].fAvgVel - vel) * (fAvgVelStr * target.fContext.fDelSecs); |
|
vel += goalDir * (curSpeed * goalStr * target.fContext.fDelSecs); |
|
vel += fInfluences[i].fRepDir * (curSpeed * fRepDirStr * target.fContext.fDelSecs); |
|
|
|
if (vel.MagnitudeSquared() > maxSpeedSq) |
|
{ |
|
vel.Normalize(); |
|
vel *= maxSpeed; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
void plParticleFlockEffect::SetMaxParticles(const uint16_t num) |
|
{ |
|
delete [] fDistSq; |
|
delete [] fInfluences; |
|
fMaxParticles = num; |
|
|
|
if (num > 0) |
|
{ |
|
fDistSq = new float[num * num]; |
|
fInfluences = new plParticleInfluenceInfo[num]; |
|
} |
|
} |
|
|
|
void plParticleFlockEffect::Read(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleEffect::Read(s, mgr); |
|
|
|
fTargetOffset.Read(s); |
|
fDissenterTarget.Read(s); |
|
fInfAvgRadSq = s->ReadLEScalar(); |
|
fInfRepRadSq = s->ReadLEScalar(); |
|
fGoalDistSq = s->ReadLEScalar(); |
|
fFullChaseDistSq = s->ReadLEScalar(); |
|
fAvgVelStr = s->ReadLEScalar(); |
|
fRepDirStr = s->ReadLEScalar(); |
|
fGoalOrbitStr = s->ReadLEScalar(); |
|
fGoalChaseStr = s->ReadLEScalar(); |
|
SetMaxOrbitSpeed(s->ReadLEScalar()); |
|
SetMaxChaseSpeed(s->ReadLEScalar()); |
|
SetMaxParticles((uint16_t)s->ReadLEScalar()); |
|
} |
|
|
|
void plParticleFlockEffect::Write(hsStream *s, hsResMgr *mgr) |
|
{ |
|
plParticleEffect::Write(s, mgr); |
|
|
|
fTargetOffset.Write(s); |
|
fDissenterTarget.Write(s); |
|
s->WriteLEScalar(fInfAvgRadSq); |
|
s->WriteLEScalar(fInfRepRadSq); |
|
s->WriteLEScalar(fGoalDistSq); |
|
s->WriteLEScalar(fFullChaseDistSq); |
|
s->WriteLEScalar(fAvgVelStr); |
|
s->WriteLEScalar(fRepDirStr); |
|
s->WriteLEScalar(fGoalOrbitStr); |
|
s->WriteLEScalar(fGoalChaseStr); |
|
s->WriteLEScalar(fMaxOrbitSpeed); |
|
s->WriteLEScalar(fMaxChaseSpeed); |
|
s->WriteLEScalar(fMaxParticles); |
|
} |
|
|
|
bool plParticleFlockEffect::MsgReceive(plMessage *msg) |
|
{ |
|
plParticleFlockMsg *flockMsg = plParticleFlockMsg::ConvertNoRef(msg); |
|
if (flockMsg) |
|
{ |
|
switch (flockMsg->fCmd) |
|
{ |
|
case plParticleFlockMsg::kFlockCmdSetDissentPoint: |
|
fDissenterTarget.Set(flockMsg->fX, flockMsg->fY, flockMsg->fZ); |
|
break; |
|
case plParticleFlockMsg::kFlockCmdSetOffset: |
|
fTargetOffset.Set(flockMsg->fX, flockMsg->fY, flockMsg->fZ); |
|
break; |
|
default: |
|
break; |
|
} |
|
return true; |
|
} |
|
|
|
return plParticleEffect::MsgReceive(msg); |
|
} |
|
|
|
/////////////////////////////////////////////////////////////////////////////////////// |
|
|
|
plParticleFollowSystemEffect::plParticleFollowSystemEffect() : fEvalThisFrame(true) |
|
{ |
|
fOldW2L = hsMatrix44::IdentityMatrix(); |
|
} |
|
|
|
void plParticleFollowSystemEffect::PrepareEffect(const plEffectTargetInfo& target) |
|
{ |
|
fEvalThisFrame = (fOldW2L != target.fContext.fSystem->GetTarget(0)->GetWorldToLocal()); |
|
} |
|
|
|
bool plParticleFollowSystemEffect::ApplyEffect(const plEffectTargetInfo& target, int32_t i) |
|
{ |
|
if (fEvalThisFrame) |
|
{ |
|
if (i < target.fFirstNewParticle && !fOldW2L.IsIdentity()) |
|
{ |
|
hsPoint3 &pos = *(hsPoint3*)(target.fPos + i * target.fPosStride); |
|
pos = target.fContext.fSystem->GetTarget(0)->GetLocalToWorld() * fOldW2L * pos; |
|
} |
|
} |
|
return true; |
|
} |
|
|
|
void plParticleFollowSystemEffect::EndEffect(const plEffectTargetInfo& target) |
|
{ |
|
if (fEvalThisFrame) |
|
fOldW2L = target.fContext.fSystem->GetTarget(0)->GetWorldToLocal(); |
|
} |
|
|
|
|
|
|
|
|