/*==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/>.

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==*/

#ifndef hsOscillator_inc
#define hsOscillator_inc

#include "hsPerterber.h"
#include "hsTemplates.h"

#include "hsGeometry3.h"
#include "../plIntersect/hsBounds.h"

class hsStream;
class plPipeline;

class hsWave
{
protected:
	hsPoint3			fWorldCenter;
	hsPoint3			fLocalCenter;

	hsScalar			fWorldFrequency; // 1.0 / Period
	hsScalar			fLocalFrequency;

	hsScalar			fWorldAmplitude;
	hsScalar			fLocalAmplitude;

	hsScalar			fPhase;
	hsScalar			fRate; // how long a crest takes to reach next crest

	hsScalar			fStartSecs;
	hsScalar			fSecsToLive;

	hsScalar			fInnerRadius;
	hsScalar			fOuterRadius;
	hsScalar			fAttenuateOutScale;

	hsScalar			AgeScale(hsScalar secs) const;

public:
	void				Accumulate(const hsPoint3& pos, const hsVector3& localZ, hsVector3& accum, hsVector3& accumNorm) const;
	hsScalar			ScaledAmplitude(hsScalar secs) const;

	void				Init(hsScalar secs, hsPoint3& center, hsScalar per, hsScalar amp, hsScalar rate, hsScalar life, hsBool32 attenOut=false);
	
	void				Update(hsScalar secs, const hsMatrix44& l2w, const hsMatrix44& w2l);
	hsBool32			IsSpent(hsScalar secs) const;
	void				Kill() { fStartSecs = fSecsToLive = 0; }
	void				AttenuateOut(hsBool32 on) { fAttenuateOutScale = (on ? 1.f : 0); }
	hsBool32			GetAttenuateOut() { return fAttenuateOutScale > 0; }

	void				Save(hsStream* s, hsScalar secs);
	void				Load(hsStream* s, hsScalar secs);
};

class hsOscillator : public hsPerterber
{
protected:
	hsTArray<hsWave>	fWaves;
	hsTArray<hsWave>	fTempWaves;

	hsMatrix44		fLocalToWorld;
	hsMatrix44		fWorldToLocal;

	hsPoint3		fWorldCenter;
	hsPoint3		fLocalCenter;

	hsVector3		fWorldAttenScale;
	hsVector3		fLocalAttenScale;
	
	hsBounds3Ext	fWorldCenterBounds;

	hsScalar		fMinPeriod;
	hsScalar		fMaxPeriod;

	hsScalar		fMinAmplitude;
	hsScalar		fMaxAmplitude;

	hsScalar		fMinRate;
	hsScalar		fMaxRate;

	hsScalar		fMinLife;
	hsScalar		fMaxLife;

	hsVector3		fLocalX;
	hsVector3		fLocalY;
	hsVector3		fLocalZ;

	hsScalar			IAttenuate(const hsPoint3& in) const;
	void			ISpawnWave(hsScalar secs, int i);

	virtual void IUpdate(hsScalar secs, plPipeline* pipe, const hsMatrix44& l2w, const hsMatrix44& w2l);

	virtual void IPerterb(const hsPoint3& in, hsGVertex3& out) const;
public:
	hsOscillator();
	virtual ~hsOscillator();

	virtual void AdjustWorldBounds(const hsMatrix44& l2w, const hsMatrix44& w2l, hsBounds3Ext& bnd) const;

	virtual UInt32 GetType() const { return kTypeOscillator; }

	// Don't call these, use base class LabelAndWrite() and CreateAndRead()
	virtual void Read(hsStream* s);
	virtual void Write(hsStream* s);

	virtual void Load(hsStream* s, hsScalar secs);
	virtual void Save(hsStream* s, hsScalar secs);

	void SetPeriodRange(hsScalar lo, hsScalar hi) { fMinPeriod = lo; fMaxPeriod = hi; }
	void SetAmplitudeRange(hsScalar lo, hsScalar hi) { fMinAmplitude = lo; fMaxAmplitude = hi; }
	void SetRateRange(hsScalar lo, hsScalar hi) { fMinRate = lo; fMaxRate = hi; }
	void SetLifeRange(hsScalar lo, hsScalar hi) { fMinLife = lo; fMaxLife = hi; }

	hsScalar GetMinPeriod() const { return fMinPeriod; }
	hsScalar GetMaxPeriod() const { return fMaxPeriod; }
	hsScalar GetMinAmplitude() const { return fMinAmplitude; }
	hsScalar GetMaxAmplitude() const { return fMaxAmplitude; }
	hsScalar GetMinRate() const { return fMinRate; }
	hsScalar GetMaxRate() const { return fMaxRate; }
	hsScalar GetMinLife() const { return fMinLife; }
	hsScalar GetMaxLife() const { return fMaxLife; }

	void SetWorldAttenScale(const hsVector3& s) { fWorldAttenScale = s; }
	void SetWorldCenterBounds(const hsBounds3Ext& bnd) { fWorldCenterBounds = bnd; }

	const hsVector3& GetWorldAttenScale() const { return fWorldAttenScale; }
	const hsBounds3Ext& GetWorldCenterBounds() const { return fWorldCenterBounds; }

	void	SetNumWaves(int n);
	UInt32	GetNumWaves() const { return fWaves.GetCount(); }
	hsWave& GetWeakestWave(hsScalar secs);
	hsWave& GetTempWave(hsScalar secs);

	virtual void Init(Int32 nParams, hsScalar* params);

	static hsGTriMesh* MakeWaveMesh(int nSpokes, const hsPoint3& center, hsScalar minRad, hsScalar maxRad, hsScalar uRange, hsScalar vRange, hsScalar attenStartFrac, hsBool32 stitch);

};

#endif // hsOscillator_inc