/** * @file llinterp.h * * Copyright (c) 2001-2007, Linden Research, Inc. * * The source code in this file ("Source Code") is provided by Linden Lab * to you under the terms of the GNU General Public License, version 2.0 * ("GPL"), unless you have obtained a separate licensing agreement * ("Other License"), formally executed by you and Linden Lab. Terms of * the GPL can be found in doc/GPL-license.txt in this distribution, or * online at http://secondlife.com/developers/opensource/gplv2 * * There are special exceptions to the terms and conditions of the GPL as * it is applied to this Source Code. View the full text of the exception * in the file doc/FLOSS-exception.txt in this software distribution, or * online at http://secondlife.com/developers/opensource/flossexception * * By copying, modifying or distributing this software, you acknowledge * that you have read and understood your obligations described above, * and agree to abide by those obligations. * * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, * COMPLETENESS OR PERFORMANCE. */ #ifndef LL_LLINTERP_H #define LL_LLINTERP_H #include "math.h" // Class from which different types of interpolators can be derived class LLInterpVal { public: virtual ~LLInterpVal() {} virtual void interp(LLInterpVal &target, const F32 frac); // Linear interpolation for each type }; template class LLInterp { public: LLInterp(); virtual ~LLInterp() {} virtual void start(); void update(const F32 time); const Type &getCurVal() const; void setStartVal(const Type &start_val); const Type &getStartVal() const; void setEndVal(const Type &target_val); const Type &getEndVal() const; void setStartTime(const F32 time); F32 getStartTime() const; void setEndTime(const F32 time); F32 getEndTime() const; BOOL isActive() const; BOOL isDone() const; protected: F32 mStartTime; F32 mEndTime; F32 mDuration; BOOL mActive; BOOL mDone; Type mStartVal; Type mEndVal; F32 mCurTime; Type mCurVal; }; template class LLInterpLinear : public LLInterp { public: /*virtual*/ void start(); void update(const F32 time); F32 getCurFrac() const; protected: F32 mCurFrac; }; template class LLInterpExp : public LLInterpLinear { public: void update(const F32 time); protected: }; template class LLInterpAttractor : public LLInterp { public: LLInterpAttractor(); /*virtual*/ void start(); void setStartVel(const Type &vel); void setForce(const F32 force); void update(const F32 time); protected: F32 mForce; Type mStartVel; Type mVelocity; }; template class LLInterpFunc : public LLInterp { public: LLInterpFunc(); void update(const F32 time); void setFunc(Type (*)(const F32, void *data), void *data); protected: Type (*mFunc)(const F32 time, void *data); void *mData; }; /////////////////////////////////// // // Implementation // // ///////////////////////////////// // // LLInterp base class implementation // template LLInterp::LLInterp() { mStartTime = 0.f; mEndTime = 1.f; mDuration = 1.f; mCurTime = 0.f; mDone = FALSE; mActive = FALSE; } template void LLInterp::setStartVal(const Type &start_val) { mStartVal = start_val; } template void LLInterp::start() { mCurVal = mStartVal; mCurTime = mStartTime; mDone = FALSE; mActive = FALSE; } template const Type &LLInterp::getStartVal() const { return mStartVal; } template void LLInterp::setEndVal(const Type &end_val) { mEndVal = end_val; } template const Type &LLInterp::getEndVal() const { return mEndVal; } template const Type &LLInterp::getCurVal() const { return mCurVal; } template void LLInterp::setStartTime(const F32 start_time) { mStartTime = start_time; mDuration = mEndTime - mStartTime; } template F32 LLInterp::getStartTime() const { return mStartTime; } template void LLInterp::setEndTime(const F32 end_time) { mEndTime = end_time; mDuration = mEndTime - mStartTime; } template F32 LLInterp::getEndTime() const { return mEndTime; } template BOOL LLInterp::isDone() const { return mDone; } template BOOL LLInterp::isActive() const { return mActive; } ////////////////////////////// // // LLInterpLinear derived class implementation. // template void LLInterpLinear::start() { LLInterp::start(); mCurFrac = 0.f; } template void LLInterpLinear::update(const F32 time) { F32 target_frac = (time - this->mStartTime) / this->mDuration; F32 dfrac = target_frac - this->mCurFrac; if (target_frac >= 0.f) { this->mActive = TRUE; } if (target_frac > 1.f) { this->mCurVal = this->mEndVal; this->mCurFrac = 1.f; this->mCurTime = time; this->mDone = TRUE; return; } target_frac = llmin(1.f, target_frac); target_frac = llmax(0.f, target_frac); if (dfrac >= 0.f) { F32 total_frac = 1.f - this->mCurFrac; F32 inc_frac = dfrac / total_frac; this->mCurVal = inc_frac * this->mEndVal + (1.f - inc_frac) * this->mCurVal; this->mCurTime = time; } else { F32 total_frac = this->mCurFrac - 1.f; F32 inc_frac = dfrac / total_frac; this->mCurVal = inc_frac * this->mStartVal + (1.f - inc_frac) * this->mCurVal; this->mCurTime = time; } mCurFrac = target_frac; } template F32 LLInterpLinear::getCurFrac() const { return mCurFrac; } ////////////////////////////// // // LLInterpAttractor derived class implementation. // template LLInterpAttractor::LLInterpAttractor() : LLInterp() { mForce = 0.1f; mVelocity *= 0.f; mStartVel *= 0.f; } template void LLInterpAttractor::start() { LLInterp::start(); mVelocity = mStartVel; } template void LLInterpAttractor::setStartVel(const Type &vel) { mStartVel = vel; } template void LLInterpAttractor::setForce(const F32 force) { mForce = force; } template void LLInterpAttractor::update(const F32 time) { if (time > this->mStartTime) { this->mActive = TRUE; } else { return; } if (time > this->mEndTime) { this->mDone = TRUE; return; } F32 dt = time - this->mCurTime; Type dist_val = this->mEndVal - this->mCurVal; Type dv = 0.5*dt*dt*this->mForce*dist_val; this->mVelocity += dv; this->mCurVal += this->mVelocity * dt; this->mCurTime = time; } ////////////////////////////// // // LLInterpFucn derived class implementation. // template LLInterpFunc::LLInterpFunc() : LLInterp() { mFunc = NULL; mData = NULL; } template void LLInterpFunc::setFunc(Type (*func)(const F32, void *data), void *data) { mFunc = func; mData = data; } template void LLInterpFunc::update(const F32 time) { if (time > this->mStartTime) { this->mActive = TRUE; } else { return; } if (time > this->mEndTime) { this->mDone = TRUE; return; } this->mCurVal = (*mFunc)(time - this->mStartTime, mData); this->mCurTime = time; } ////////////////////////////// // // LLInterpExp derived class implementation. // template void LLInterpExp::update(const F32 time) { F32 target_frac = (time - this->mStartTime) / this->mDuration; if (target_frac >= 0.f) { this->mActive = TRUE; } if (target_frac > 1.f) { this->mCurVal = this->mEndVal; this->mCurFrac = 1.f; this->mCurTime = time; this->mDone = TRUE; return; } this->mCurFrac = 1.f - (F32)(exp(-2.f*target_frac)); this->mCurVal = this->mStartVal + this->mCurFrac * (this->mEndVal - this->mStartVal); this->mCurTime = time; } #endif // LL_LLINTERP_H