afxXM_PathConform.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  2. // Arcane-FX for MIT Licensed Open Source version of Torque 3D from GarageGames
  3. // Copyright (C) 2015 Faust Logic, Inc.
  4. //
  5. // Permission is hereby granted, free of charge, to any person obtaining a copy
  6. // of this software and associated documentation files (the "Software"), to
  7. // deal in the Software without restriction, including without limitation the
  8. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  9. // sell copies of the Software, and to permit persons to whom the Software is
  10. // furnished to do so, subject to the following conditions:
  11. //
  12. // The above copyright notice and this permission notice shall be included in
  13. // all copies or substantial portions of the Software.
  14. //
  15. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  16. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  17. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  18. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  19. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  20. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  21. // IN THE SOFTWARE.
  22. //
  23. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  24. #include "afx/arcaneFX.h"
  25. #include "math/mathIO.h"
  26. #include "math/mathUtils.h"
  27. #include "afx/afxEffectWrapper.h"
  28. #include "afx/afxChoreographer.h"
  29. #include "afx/xm/afxXfmMod.h"
  30. #include "afx/util/afxPath3D.h"
  31. #include "afx/util/afxPath.h"
  32. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  33. // PATH CONFORM
  34. class afxPathData;
  35. class afxXM_PathConformData : public afxXM_WeightedBaseData
  36. {
  37. typedef afxXM_WeightedBaseData Parent;
  38. public:
  39. StringTableEntry paths_string; //
  40. Vector<afxPathData*> pathDataBlocks; // datablocks for paths
  41. Vector<U32> pathDataBlockIds; // datablock IDs which correspond to the pathDataBlocks
  42. F32 path_mult;
  43. F32 path_time_offset;
  44. bool orient_to_path;
  45. public:
  46. /*C*/ afxXM_PathConformData();
  47. /*C*/ afxXM_PathConformData(const afxXM_PathConformData&, bool = false);
  48. bool onAdd();
  49. bool preload(bool server, String &errorStr);
  50. void packData(BitStream* stream);
  51. void unpackData(BitStream* stream);
  52. virtual bool allowSubstitutions() const { return true; }
  53. static void initPersistFields();
  54. afxXM_Base* create(afxEffectWrapper* fx, bool on_server);
  55. DECLARE_CONOBJECT(afxXM_PathConformData);
  56. DECLARE_CATEGORY("AFX");
  57. };
  58. class afxPath3D;
  59. class afxAnimCurve;
  60. class afxXM_PathConform : public afxXM_WeightedBase
  61. {
  62. typedef afxXM_WeightedBase Parent;
  63. afxXM_PathConformData* db;
  64. Vector<afxPath3D*> paths;
  65. Vector<F32> path_mults;
  66. Vector<F32> path_time_offsets;
  67. Vector<afxAnimCurve*> rolls;
  68. void init_paths(F32 lifetime);
  69. public:
  70. /*C*/ afxXM_PathConform(afxXM_PathConformData*, afxEffectWrapper*);
  71. /*D*/ ~afxXM_PathConform();
  72. virtual void start(F32 timestamp);
  73. virtual void updateParams(F32 dt, F32 elapsed, afxXM_Params& params);
  74. };
  75. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  76. IMPLEMENT_CO_DATABLOCK_V1(afxXM_PathConformData);
  77. ConsoleDocClass( afxXM_PathConformData,
  78. "@brief An xmod datablock.\n\n"
  79. "@ingroup afxXMods\n"
  80. "@ingroup AFX\n"
  81. "@ingroup Datablocks\n"
  82. );
  83. afxXM_PathConformData::afxXM_PathConformData()
  84. {
  85. paths_string = ST_NULLSTRING;
  86. pathDataBlocks.clear();
  87. pathDataBlockIds.clear();
  88. path_mult = 1.0f;
  89. path_time_offset = 0.0f;
  90. orient_to_path = false;
  91. }
  92. afxXM_PathConformData::afxXM_PathConformData(const afxXM_PathConformData& other, bool temp_clone) : afxXM_WeightedBaseData(other, temp_clone)
  93. {
  94. paths_string = other.paths_string;
  95. pathDataBlocks = other.pathDataBlocks;
  96. pathDataBlockIds = other.pathDataBlockIds;
  97. path_mult = other.path_mult;
  98. path_time_offset = other.path_time_offset;
  99. orient_to_path = other.orient_to_path;
  100. }
  101. void afxXM_PathConformData::initPersistFields()
  102. {
  103. docsURL;
  104. addField("paths", TypeString, Offset(paths_string, afxXM_PathConformData),
  105. "...");
  106. addField("pathMult", TypeF32, Offset(path_mult, afxXM_PathConformData),
  107. "...");
  108. addField("pathTimeOffset", TypeF32, Offset(path_time_offset, afxXM_PathConformData),
  109. "...");
  110. addField("orientToPath", TypeBool, Offset(orient_to_path, afxXM_PathConformData),
  111. "...");
  112. Parent::initPersistFields();
  113. // disallow some field substitutions
  114. disableFieldSubstitutions("paths");
  115. }
  116. void afxXM_PathConformData::packData(BitStream* stream)
  117. {
  118. Parent::packData(stream);
  119. stream->write(pathDataBlockIds.size());
  120. for (int i = 0; i < pathDataBlockIds.size(); i++)
  121. stream->write(pathDataBlockIds[i]);
  122. stream->write(path_mult);
  123. stream->write(path_time_offset);
  124. stream->write(orient_to_path);
  125. }
  126. void afxXM_PathConformData::unpackData(BitStream* stream)
  127. {
  128. Parent::unpackData(stream);
  129. U32 n_db;
  130. stream->read(&n_db);
  131. pathDataBlockIds.setSize(n_db);
  132. for (U32 i = 0; i < n_db; i++)
  133. stream->read(&pathDataBlockIds[i]);
  134. stream->read(&path_mult);
  135. stream->read(&path_time_offset);
  136. stream->read(&orient_to_path);
  137. }
  138. bool afxXM_PathConformData::onAdd()
  139. {
  140. if (Parent::onAdd() == false)
  141. return false;
  142. if (paths_string != ST_NULLSTRING)
  143. {
  144. //Con::printf("afxEffectWrapperData: Path string found: %s", paths_string);
  145. }
  146. if (paths_string != ST_NULLSTRING && paths_string[0] == '\0')
  147. {
  148. Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) empty paths string, invalid datablock", getName());
  149. return false;
  150. }
  151. if (paths_string != ST_NULLSTRING && dStrlen(paths_string) > 255)
  152. {
  153. Con::errorf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) paths string too long [> 255 chars]", getName());
  154. return false;
  155. }
  156. if (paths_string != ST_NULLSTRING)
  157. {
  158. Vector<char*> dataBlocks(__FILE__, __LINE__);
  159. dsize_t tokCopyLen = dStrlen(paths_string) + 1;
  160. char* tokCopy = new char[tokCopyLen];
  161. dStrcpy(tokCopy, paths_string, tokCopyLen);
  162. char* currTok = dStrtok(tokCopy, " \t");
  163. while (currTok != NULL)
  164. {
  165. dataBlocks.push_back(currTok);
  166. currTok = dStrtok(NULL, " \t");
  167. }
  168. if (dataBlocks.size() == 0)
  169. {
  170. Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) invalid paths string. No datablocks found", getName());
  171. delete [] tokCopy;
  172. return false;
  173. }
  174. pathDataBlocks.clear();
  175. pathDataBlockIds.clear();
  176. for (U32 i = 0; i < dataBlocks.size(); i++)
  177. {
  178. afxPathData* pData = NULL;
  179. if (Sim::findObject(dataBlocks[i], pData) == false)
  180. {
  181. Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) unable to find path datablock: %s", getName(), dataBlocks[i]);
  182. }
  183. else
  184. {
  185. pathDataBlocks.push_back(pData);
  186. pathDataBlockIds.push_back(pData->getId());
  187. }
  188. }
  189. delete [] tokCopy;
  190. if (pathDataBlocks.size() == 0)
  191. {
  192. Con::warnf(ConsoleLogEntry::General, "afxEffectWrapperData(%s) unable to find any path datablocks", getName());
  193. return false;
  194. }
  195. }
  196. return true;
  197. }
  198. bool afxXM_PathConformData::preload(bool server, String &errorStr)
  199. {
  200. if (!Parent::preload(server, errorStr))
  201. return false;
  202. pathDataBlocks.clear();
  203. for (U32 i = 0; i < pathDataBlockIds.size(); i++)
  204. {
  205. afxPathData* pData = NULL;
  206. if (Sim::findObject(pathDataBlockIds[i], pData) == false)
  207. {
  208. Con::warnf(ConsoleLogEntry::General,
  209. "afxEffectWrapperData(%s) unable to find path datablock: %d",
  210. getName(), pathDataBlockIds[i]);
  211. }
  212. else
  213. pathDataBlocks.push_back(pData);
  214. }
  215. return true;
  216. }
  217. afxXM_Base* afxXM_PathConformData::create(afxEffectWrapper* fx, bool on_server)
  218. {
  219. afxXM_PathConformData* datablock = this;
  220. if (getSubstitutionCount() > 0)
  221. {
  222. datablock = new afxXM_PathConformData(*this, true);
  223. this->performSubstitutions(datablock, fx->getChoreographer(), fx->getGroupIndex());
  224. }
  225. return new afxXM_PathConform(datablock, fx);
  226. }
  227. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//
  228. afxXM_PathConform::afxXM_PathConform(afxXM_PathConformData* db, afxEffectWrapper* fxw)
  229. : afxXM_WeightedBase(db, fxw)
  230. {
  231. this->db = db;
  232. }
  233. afxXM_PathConform::~afxXM_PathConform()
  234. {
  235. for (S32 i = 0; i < paths.size(); i++)
  236. if (paths[i])
  237. delete paths[i];
  238. for (S32 j = 0; j < paths.size(); j++)
  239. if (rolls[j])
  240. delete rolls[j];
  241. }
  242. void afxXM_PathConform::start(F32 timestamp)
  243. {
  244. init_paths(fx_wrapper->getFullLifetime());
  245. }
  246. void afxXM_PathConform::init_paths(F32 lifetime)
  247. {
  248. for( U32 i=0; i < db->pathDataBlocks.size(); i++ )
  249. {
  250. afxPathData* pd = db->pathDataBlocks[i];
  251. if (!pd)
  252. continue;
  253. if (pd->getSubstitutionCount() > 0)
  254. {
  255. afxPathData* orig_db = pd;
  256. pd = new afxPathData(*orig_db, true);
  257. orig_db->performSubstitutions(pd, fx_wrapper->getChoreographer(), fx_wrapper->getGroupIndex());
  258. }
  259. if (pd->num_points > 0)
  260. {
  261. if (pd->lifetime == 0)
  262. {
  263. if (lifetime <= 0)
  264. {
  265. // Is this possible or is the lifetime always inherited properly???
  266. }
  267. else
  268. {
  269. pd->lifetime = lifetime;
  270. }
  271. }
  272. F32 pd_delay = pd->delay*time_factor;
  273. F32 pd_lifetime = pd->lifetime*time_factor;
  274. F32 pd_time_offset = pd->time_offset*time_factor;
  275. afxPath3D* path = new afxPath3D();
  276. if (pd->times)
  277. path->buildPath( pd->num_points, pd->points, pd->times, pd_delay, time_factor );
  278. else
  279. path->buildPath( pd->num_points, pd->points, pd_delay, pd_delay+pd_lifetime );
  280. path->setLoopType( pd->loop_type );
  281. paths.push_back(path);
  282. // path->print();
  283. path_mults.push_back( db->path_mult * pd->mult );
  284. path_time_offsets.push_back( db->path_time_offset + pd_time_offset );
  285. if (pd->roll_string != ST_NULLSTRING && pd->rolls != NULL)
  286. {
  287. afxAnimCurve* roll_curve = new afxAnimCurve();
  288. for (U32 j=0; j<pd->num_points; j++ )
  289. {
  290. roll_curve->addKey( path->getPointTime(j), pd->rolls[j] );
  291. }
  292. roll_curve->sort();
  293. // roll_curve->print();
  294. rolls.push_back(roll_curve);
  295. }
  296. else
  297. {
  298. rolls.push_back( NULL );
  299. }
  300. }
  301. else
  302. {
  303. Con::warnf("afxXM_PathConform::init_paths() -- paths datablock (%d) has no points.", i);
  304. }
  305. if (pd->isTempClone())
  306. delete pd;
  307. }
  308. }
  309. void afxXM_PathConform::updateParams(F32 dt, F32 elapsed, afxXM_Params& params)
  310. {
  311. F32 wt_factor = calc_weight_factor(elapsed);
  312. // compute path offset
  313. VectorF path_offset(0,0,0);
  314. for( U32 i=0; i < paths.size(); i++ )
  315. path_offset += path_mults[i]*paths[i]->evaluateAtTime(elapsed + path_time_offsets[i])*wt_factor;
  316. params.ori.mulV(path_offset);
  317. params.pos += path_offset;
  318. if (db->orient_to_path && paths.size())
  319. {
  320. VectorF path_v = paths[0]->evaluateTangentAtTime(elapsed+path_time_offsets[0]);
  321. path_v.normalize();
  322. MatrixF mat(true);
  323. if( rolls[0] )
  324. {
  325. F32 roll_angle = rolls[0]->evaluate(elapsed+path_time_offsets[0]);
  326. roll_angle = mDegToRad( roll_angle );
  327. //Con::printf( "roll: %f", roll_angle );
  328. Point3F roll_axis = path_v;
  329. AngAxisF rollRot(roll_axis, roll_angle);
  330. MatrixF roll_mat(true);
  331. rollRot.setMatrix(&roll_mat);
  332. mat.mul(roll_mat);
  333. }
  334. mat.mul(MathUtils::createOrientFromDir(path_v));
  335. params.ori.mul(mat);
  336. }
  337. }
  338. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//