123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413 |
- //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
- // Arcane-FX for MIT Licensed Open Source version of Torque 3D from GarageGames
- // Copyright (C) 2015 Faust Logic, Inc.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to
- // deal in the Software without restriction, including without limitation the
- // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
- // sell copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
- // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
- // IN THE SOFTWARE.
- //
- //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
- #include "afx/arcaneFX.h"
- #include "math/mathIO.h"
- #include "math/mathUtils.h"
- #include "afx/afxEffectWrapper.h"
- #include "afx/afxChoreographer.h"
- #include "afx/xm/afxXfmMod.h"
- #include "afx/util/afxPath3D.h"
- #include "afx/util/afxPath.h"
- //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
- // PATH CONFORM
- class afxPathData;
- class afxXM_PathConformData : public afxXM_WeightedBaseData
- {
- typedef afxXM_WeightedBaseData Parent;
- public:
- StringTableEntry paths_string; //
- Vector<afxPathData*> pathDataBlocks; // datablocks for paths
- Vector<U32> pathDataBlockIds; // datablock IDs which correspond to the pathDataBlocks
- F32 path_mult;
- F32 path_time_offset;
- bool orient_to_path;
- public:
- /*C*/ afxXM_PathConformData();
- /*C*/ afxXM_PathConformData(const afxXM_PathConformData&, bool = false);
- bool onAdd();
- bool preload(bool server, String &errorStr);
- void packData(BitStream* stream);
- void unpackData(BitStream* stream);
- virtual bool allowSubstitutions() const { return true; }
- static void initPersistFields();
- afxXM_Base* create(afxEffectWrapper* fx, bool on_server);
- DECLARE_CONOBJECT(afxXM_PathConformData);
- DECLARE_CATEGORY("AFX");
- };
- class afxPath3D;
- class afxAnimCurve;
- class afxXM_PathConform : public afxXM_WeightedBase
- {
- typedef afxXM_WeightedBase Parent;
- afxXM_PathConformData* db;
- Vector<afxPath3D*> paths;
- Vector<F32> path_mults;
- Vector<F32> path_time_offsets;
- Vector<afxAnimCurve*> rolls;
- void init_paths(F32 lifetime);
- public:
- /*C*/ afxXM_PathConform(afxXM_PathConformData*, afxEffectWrapper*);
- /*D*/ ~afxXM_PathConform();
- virtual void start(F32 timestamp);
- virtual void updateParams(F32 dt, F32 elapsed, afxXM_Params& params);
- };
- //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
- IMPLEMENT_CO_DATABLOCK_V1(afxXM_PathConformData);
- ConsoleDocClass( afxXM_PathConformData,
- "@brief An xmod datablock.\n\n"
- "@ingroup afxXMods\n"
- "@ingroup AFX\n"
- "@ingroup Datablocks\n"
- );
- afxXM_PathConformData::afxXM_PathConformData()
- {
- paths_string = ST_NULLSTRING;
- pathDataBlocks.clear();
- pathDataBlockIds.clear();
- path_mult = 1.0f;
- path_time_offset = 0.0f;
- orient_to_path = false;
- }
- afxXM_PathConformData::afxXM_PathConformData(const afxXM_PathConformData& other, bool temp_clone) : afxXM_WeightedBaseData(other, temp_clone)
- {
- paths_string = other.paths_string;
- pathDataBlocks = other.pathDataBlocks;
- pathDataBlockIds = other.pathDataBlockIds;
- path_mult = other.path_mult;
- path_time_offset = other.path_time_offset;
- orient_to_path = other.orient_to_path;
- }
- void afxXM_PathConformData::initPersistFields()
- {
- addField("paths", TypeString, Offset(paths_string, afxXM_PathConformData),
- "...");
- addField("pathMult", TypeF32, Offset(path_mult, afxXM_PathConformData),
- "...");
- addField("pathTimeOffset", TypeF32, Offset(path_time_offset, afxXM_PathConformData),
- "...");
- addField("orientToPath", TypeBool, Offset(orient_to_path, afxXM_PathConformData),
- "...");
- Parent::initPersistFields();
- // disallow some field substitutions
- disableFieldSubstitutions("paths");
- }
- void afxXM_PathConformData::packData(BitStream* stream)
- {
- Parent::packData(stream);
- stream->write(pathDataBlockIds.size());
- for (int i = 0; i < pathDataBlockIds.size(); i++)
- stream->write(pathDataBlockIds[i]);
- stream->write(path_mult);
- stream->write(path_time_offset);
- stream->write(orient_to_path);
- }
- void afxXM_PathConformData::unpackData(BitStream* stream)
- {
- Parent::unpackData(stream);
- U32 n_db;
- stream->read(&n_db);
- pathDataBlockIds.setSize(n_db);
- for (U32 i = 0; i < n_db; i++)
- stream->read(&pathDataBlockIds[i]);
- stream->read(&path_mult);
- stream->read(&path_time_offset);
- stream->read(&orient_to_path);
- }
- bool afxXM_PathConformData::onAdd()
- {
- if (Parent::onAdd() == false)
- return false;
- if (paths_string != ST_NULLSTRING)
- {
- //Con::printf("afxEffectWrapperData: Path string found: %s", paths_string);
- }
-
- if (paths_string != ST_NULLSTRING && paths_string[0] == '\0')
- {
- Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) empty paths string, invalid datablock", getName());
- return false;
- }
- if (paths_string != ST_NULLSTRING && dStrlen(paths_string) > 255)
- {
- Con::errorf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) paths string too long [> 255 chars]", getName());
- return false;
- }
- if (paths_string != ST_NULLSTRING)
- {
- Vector<char*> dataBlocks(__FILE__, __LINE__);
- dsize_t tokCopyLen = dStrlen(paths_string) + 1;
- char* tokCopy = new char[tokCopyLen];
- dStrcpy(tokCopy, paths_string, tokCopyLen);
-
- char* currTok = dStrtok(tokCopy, " \t");
- while (currTok != NULL)
- {
- dataBlocks.push_back(currTok);
- currTok = dStrtok(NULL, " \t");
- }
- if (dataBlocks.size() == 0)
- {
- Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) invalid paths string. No datablocks found", getName());
- delete [] tokCopy;
- return false;
- }
- pathDataBlocks.clear();
- pathDataBlockIds.clear();
-
- for (U32 i = 0; i < dataBlocks.size(); i++)
- {
- afxPathData* pData = NULL;
- if (Sim::findObject(dataBlocks[i], pData) == false)
- {
- Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) unable to find path datablock: %s", getName(), dataBlocks[i]);
- }
- else
- {
- pathDataBlocks.push_back(pData);
- pathDataBlockIds.push_back(pData->getId());
- }
- }
- delete [] tokCopy;
- if (pathDataBlocks.size() == 0)
- {
- Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) unable to find any path datablocks", getName());
- return false;
- }
- }
- return true;
- }
- bool afxXM_PathConformData::preload(bool server, String &errorStr)
- {
- if (!Parent::preload(server, errorStr))
- return false;
-
- pathDataBlocks.clear();
- for (U32 i = 0; i < pathDataBlockIds.size(); i++)
- {
- afxPathData* pData = NULL;
- if (Sim::findObject(pathDataBlockIds[i], pData) == false)
- {
- Con::warnf(ConsoleLogEntry::General,
- "afxEffectWrapperData(%s) unable to find path datablock: %d",
- getName(), pathDataBlockIds[i]);
- }
- else
- pathDataBlocks.push_back(pData);
- }
- return true;
- }
- afxXM_Base* afxXM_PathConformData::create(afxEffectWrapper* fx, bool on_server)
- {
- afxXM_PathConformData* datablock = this;
- if (getSubstitutionCount() > 0)
- {
- datablock = new afxXM_PathConformData(*this, true);
- this->performSubstitutions(datablock, fx->getChoreographer(), fx->getGroupIndex());
- }
- return new afxXM_PathConform(datablock, fx);
- }
- //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//
- afxXM_PathConform::afxXM_PathConform(afxXM_PathConformData* db, afxEffectWrapper* fxw)
- : afxXM_WeightedBase(db, fxw)
- {
- this->db = db;
- }
- afxXM_PathConform::~afxXM_PathConform()
- {
- for (S32 i = 0; i < paths.size(); i++)
- if (paths[i])
- delete paths[i];
- for (S32 j = 0; j < paths.size(); j++)
- if (rolls[j])
- delete rolls[j];
- }
- void afxXM_PathConform::start(F32 timestamp)
- {
- init_paths(fx_wrapper->getFullLifetime());
- }
- void afxXM_PathConform::init_paths(F32 lifetime)
- {
- for( U32 i=0; i < db->pathDataBlocks.size(); i++ )
- {
- afxPathData* pd = db->pathDataBlocks[i];
- if (!pd)
- continue;
- if (pd->getSubstitutionCount() > 0)
- {
- afxPathData* orig_db = pd;
- pd = new afxPathData(*orig_db, true);
- orig_db->performSubstitutions(pd, fx_wrapper->getChoreographer(), fx_wrapper->getGroupIndex());
- }
- if (pd->num_points > 0)
- {
- if (pd->lifetime == 0)
- {
- if (lifetime <= 0)
- {
- // Is this possible or is the lifetime always inherited properly???
- }
- else
- {
- pd->lifetime = lifetime;
- }
- }
- F32 pd_delay = pd->delay*time_factor;
- F32 pd_lifetime = pd->lifetime*time_factor;
- F32 pd_time_offset = pd->time_offset*time_factor;
- afxPath3D* path = new afxPath3D();
- if (pd->times)
- path->buildPath( pd->num_points, pd->points, pd->times, pd_delay, time_factor );
- else
- path->buildPath( pd->num_points, pd->points, pd_delay, pd_delay+pd_lifetime );
- path->setLoopType( pd->loop_type );
- paths.push_back(path);
- // path->print();
- path_mults.push_back( db->path_mult * pd->mult );
- path_time_offsets.push_back( db->path_time_offset + pd_time_offset );
- if (pd->roll_string != ST_NULLSTRING && pd->rolls != NULL)
- {
- afxAnimCurve* roll_curve = new afxAnimCurve();
- for (U32 j=0; j<pd->num_points; j++ )
- {
- roll_curve->addKey( path->getPointTime(j), pd->rolls[j] );
- }
- roll_curve->sort();
- // roll_curve->print();
- rolls.push_back(roll_curve);
- }
- else
- {
- rolls.push_back( NULL );
- }
- }
- else
- {
- Con::warnf("afxXM_PathConform::init_paths() -- paths datablock (%d) has no points.", i);
- }
- if (pd->isTempClone())
- delete pd;
- }
- }
- void afxXM_PathConform::updateParams(F32 dt, F32 elapsed, afxXM_Params& params)
- {
- F32 wt_factor = calc_weight_factor(elapsed);
- // compute path offset
- VectorF path_offset(0,0,0);
- for( U32 i=0; i < paths.size(); i++ )
- path_offset += path_mults[i]*paths[i]->evaluateAtTime(elapsed + path_time_offsets[i])*wt_factor;
- params.ori.mulV(path_offset);
- params.pos += path_offset;
- if (db->orient_to_path && paths.size())
- {
- VectorF path_v = paths[0]->evaluateTangentAtTime(elapsed+path_time_offsets[0]);
- path_v.normalize();
- MatrixF mat(true);
- if( rolls[0] )
- {
- F32 roll_angle = rolls[0]->evaluate(elapsed+path_time_offsets[0]);
- roll_angle = mDegToRad( roll_angle );
- //Con::printf( "roll: %f", roll_angle );
-
- Point3F roll_axis = path_v;
- AngAxisF rollRot(roll_axis, roll_angle);
- MatrixF roll_mat(true);
- rollRot.setMatrix(&roll_mat);
-
- mat.mul(roll_mat);
- }
- mat.mul(MathUtils::createOrientFromDir(path_v));
- params.ori.mul(mat);
- }
- }
- //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
|