tsStatic.cpp 57 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2012 GarageGames, LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  23. // Arcane-FX for MIT Licensed Open Source version of Torque 3D from GarageGames
  24. // Copyright (C) 2015 Faust Logic, Inc.
  25. //~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//
  26. #include "platform/platform.h"
  27. #include "T3D/tsStatic.h"
  28. #include "core/resourceManager.h"
  29. #include "core/stream/bitStream.h"
  30. #include "scene/sceneRenderState.h"
  31. #include "scene/sceneManager.h"
  32. #include "scene/sceneObjectLightingPlugin.h"
  33. #include "lighting/lightManager.h"
  34. #include "math/mathIO.h"
  35. #include "ts/tsShapeInstance.h"
  36. #include "ts/tsMaterialList.h"
  37. #include "console/consoleTypes.h"
  38. #include "T3D/shapeBase.h"
  39. #include "sim/netConnection.h"
  40. #include "gfx/gfxDevice.h"
  41. #include "gfx/gfxTransformSaver.h"
  42. #include "ts/tsRenderState.h"
  43. #include "collision/boxConvex.h"
  44. #include "T3D/physics/physicsPlugin.h"
  45. #include "T3D/physics/physicsBody.h"
  46. #include "T3D/physics/physicsCollision.h"
  47. #include "materials/materialDefinition.h"
  48. #include "materials/materialManager.h"
  49. #include "materials/matInstance.h"
  50. #include "materials/materialFeatureData.h"
  51. #include "materials/materialFeatureTypes.h"
  52. #include "console/engineAPI.h"
  53. #include "T3D/accumulationVolume.h"
  54. #include "math/mTransform.h"
  55. #include "gui/editor/inspector/group.h"
  56. #include "console/typeValidators.h"
  57. using namespace Torque;
  58. extern bool gEditingMission;
  59. #ifdef TORQUE_AFX_ENABLED
  60. #include "afx/ce/afxZodiacMgr.h"
  61. #endif
  62. IMPLEMENT_CO_NETOBJECT_V1(TSStatic);
  63. ConsoleDocClass(TSStatic,
  64. "@brief A static object derived from a 3D model file and placed within the game world.\n\n"
  65. "TSStatic is the most basic 3D shape in Torque. Unlike StaticShape it doesn't make use of "
  66. "a datablock. It derrives directly from SceneObject. This makes TSStatic extremely light "
  67. "weight, which is why the Tools use this class when you want to drop in a DTS or DAE object.\n\n"
  68. "While a TSStatic doesn't provide any motion -- it stays were you initally put it -- it does allow for "
  69. "a single ambient animation sequence to play when the object is first added to the scene.\n\n"
  70. "@tsexample\n"
  71. "new TSStatic(Team1Base) {\n"
  72. " shapeName = \"art/shapes/desertStructures/station01.dts\";\n"
  73. " playAmbient = \"1\";\n"
  74. " receiveSunLight = \"1\";\n"
  75. " receiveLMLighting = \"1\";\n"
  76. " useCustomAmbientLighting = \"0\";\n"
  77. " customAmbientLighting = \"0 0 0 1\";\n"
  78. " collisionType = \"Visible Mesh\";\n"
  79. " decalType = \"Collision Mesh\";\n"
  80. " allowPlayerStep = \"1\";\n"
  81. " renderNormals = \"0\";\n"
  82. " forceDetail = \"-1\";\n"
  83. " position = \"315.18 -180.418 244.313\";\n"
  84. " rotation = \"0 0 1 195.952\";\n"
  85. " scale = \"1 1 1\";\n"
  86. " isRenderEnabled = \"true\";\n"
  87. " canSaveDynamicFields = \"1\";\n"
  88. "};\n"
  89. "@endtsexample\n"
  90. "@ingroup gameObjects\n"
  91. );
  92. bool TSStatic::smUseStaticObjectFade = false;
  93. F32 TSStatic::smStaticObjectFadeStart = 50;
  94. F32 TSStatic::smStaticObjectFadeEnd = 75;
  95. F32 TSStatic::smStaticObjectUnfadeableSize = 75;
  96. TSStatic::TSStatic()
  97. :
  98. cubeDescId(0),
  99. reflectorDesc(NULL)
  100. {
  101. mNetFlags.set(Ghostable | ScopeAlways);
  102. mTypeMask |= StaticObjectType | StaticShapeObjectType;
  103. mShapeInstance = NULL;
  104. mPlayAmbient = true;
  105. mAmbientThread = NULL;
  106. mAllowPlayerStep = false;
  107. mConvexList = new Convex;
  108. mRenderNormalScalar = 0;
  109. mForceDetail = -1;
  110. mMeshCulling = false;
  111. mUseOriginSort = false;
  112. mUseAlphaFade = false;
  113. mAlphaFadeStart = 100.0f;
  114. mAlphaFadeEnd = 150.0f;
  115. mInvertAlphaFade = false;
  116. mAlphaFade = 1.0f;
  117. mPhysicsRep = NULL;
  118. mCollisionLOD = 0;
  119. mCollisionType = CollisionMesh;
  120. mDecalType = CollisionMesh;
  121. mIgnoreZodiacs = false;
  122. mHasGradients = false;
  123. mInvertGradientRange = false;
  124. mGradientRangeUser.set(0.0f, 180.0f);
  125. #ifdef TORQUE_AFX_ENABLED
  126. afxZodiacData::convertGradientRangeFromDegrees(mGradientRange, mGradientRangeUser);
  127. #endif
  128. mAnimOffset = 0.0f;
  129. mAnimSpeed = 1.0f;
  130. mShapeAsset.registerRefreshNotify(this);
  131. }
  132. TSStatic::~TSStatic()
  133. {
  134. delete mConvexList;
  135. mConvexList = NULL;
  136. mShapeAsset.unregisterRefreshNotify();
  137. }
  138. ImplementEnumType(TSMeshType,
  139. "Type of mesh data available in a shape.\n"
  140. "@ingroup gameObjects")
  141. {
  142. TSStatic::None, "None", "No mesh data."
  143. },
  144. { TSStatic::Bounds, "Bounds", "Bounding box of the shape." },
  145. { TSStatic::CollisionMesh, "Collision Mesh", "Specifically desingated \"collision\" meshes." },
  146. { TSStatic::VisibleMesh, "Visible Mesh", "Rendered mesh polygons." },
  147. EndImplementEnumType;
  148. FRangeValidator percentValidator(0.0f, 1.0f);
  149. F32 AnimSpeedMax = 4.0f;
  150. FRangeValidator speedValidator(0.0f, AnimSpeedMax);
  151. void TSStatic::initPersistFields()
  152. {
  153. docsURL;
  154. addGroup("Shape");
  155. INITPERSISTFIELD_SHAPEASSET_REFACTOR(Shape, TSStatic, "Model to use for this TSStatic");
  156. endGroup("Shape");
  157. addGroup("Materials");
  158. addProtectedField("skin", TypeRealString, Offset(mAppliedSkinName, TSStatic), &_setFieldSkin, &_getFieldSkin,
  159. "@brief The skin applied to the shape.\n\n"
  160. "'Skinning' the shape effectively renames the material targets, allowing "
  161. "different materials to be used on different instances of the same model.\n\n"
  162. "Any material targets that start with the old skin name have that part "
  163. "of the name replaced with the new skin name. The initial old skin name is "
  164. "\"base\". For example, if a new skin of \"blue\" was applied to a model "
  165. "that had material targets <i>base_body</i> and <i>face</i>, the new targets "
  166. "would be <i>blue_body</i> and <i>face</i>. Note that <i>face</i> was not "
  167. "renamed since it did not start with the old skin name of \"base\".\n\n"
  168. "To support models that do not use the default \"base\" naming convention, "
  169. "you can also specify the part of the name to replace in the skin field "
  170. "itself. For example, if a model had a material target called <i>shapemat</i>, "
  171. "we could apply a new skin \"shape=blue\", and the material target would be "
  172. "renamed to <i>bluemat</i> (note \"shape\" has been replaced with \"blue\").\n\n"
  173. "Multiple skin updates can also be applied at the same time by separating "
  174. "them with a semicolon. For example: \"base=blue;face=happy_face\".\n\n"
  175. "Material targets are only renamed if an existing Material maps to that "
  176. "name, or if there is a diffuse texture in the model folder with the same "
  177. "name as the new target.\n\n");
  178. endGroup("Materials");
  179. addGroup("Animation");
  180. addField("playAmbient", TypeBool, Offset(mPlayAmbient, TSStatic),
  181. "Enables automatic playing of the animation sequence named \"ambient\" (if it exists) when the TSStatic is loaded.");
  182. addFieldV("AnimOffset", TypeRangedF32, Offset(mAnimOffset, TSStatic), &percentValidator,
  183. "Percent Animation Offset.");
  184. addFieldV("AnimSpeed", TypeRangedF32, Offset(mAnimSpeed, TSStatic), &speedValidator,
  185. "Percent Animation Speed.");
  186. endGroup("Animation");
  187. addGroup("Rendering");
  188. addField("meshCulling", TypeBool, Offset(mMeshCulling, TSStatic),
  189. "Enables detailed culling of meshes within the TSStatic. Should only be used "
  190. "with large complex shapes like buildings which contain many submeshes.");
  191. addField("originSort", TypeBool, Offset(mUseOriginSort, TSStatic),
  192. "Enables translucent sorting of the TSStatic by its origin instead of the bounds.");
  193. endGroup("Rendering");
  194. addGroup("Reflection");
  195. addField("cubeReflectorDesc", TypeRealString, Offset(cubeDescName, TSStatic),
  196. "References a ReflectorDesc datablock that defines performance and quality properties for dynamic reflections.\n");
  197. endGroup("Reflection");
  198. addGroup("Collision");
  199. addField("collisionLOD", TypeS16, Offset(mCollisionLOD, TSStatic),
  200. "The level of detail to use for 'Visible Mesh' collision queries.");
  201. addField("collisionType", TypeTSMeshType, Offset(mCollisionType, TSStatic),
  202. "The type of mesh data to use for collision queries.");
  203. addField("decalType", TypeTSMeshType, Offset(mDecalType, TSStatic),
  204. "The type of mesh data used to clip decal polygons against.");
  205. addField("allowPlayerStep", TypeBool, Offset(mAllowPlayerStep, TSStatic),
  206. "@brief Allow a Player to walk up sloping polygons in the TSStatic (based on the collisionType).\n\n"
  207. "When set to false, the slightest bump will stop the player from walking on top of the object.\n");
  208. endGroup("Collision");
  209. addGroup("AlphaFade");
  210. addField("alphaFadeEnable", TypeBool, Offset(mUseAlphaFade, TSStatic), "Turn on/off Alpha Fade");
  211. addFieldV("alphaFadeStart", TypeRangedF32, Offset(mAlphaFadeStart, TSStatic), &CommonValidators::PositiveFloat, "Distance of start Alpha Fade");
  212. addFieldV("alphaFadeEnd", TypeRangedF32, Offset(mAlphaFadeEnd, TSStatic), &CommonValidators::PositiveFloat, "Distance of end Alpha Fade");
  213. addField("alphaFadeInverse", TypeBool, Offset(mInvertAlphaFade, TSStatic), "Invert Alpha Fade's Start & End Distance");
  214. endGroup("AlphaFade");
  215. addGroup("Debug");
  216. addFieldV("renderNormals", TypeRangedF32, Offset(mRenderNormalScalar, TSStatic), &CommonValidators::PositiveFloat,
  217. "Debug rendering mode shows the normals for each point in the TSStatic's mesh.");
  218. addFieldV("forceDetail", TypeRangedS32, Offset(mForceDetail, TSStatic), &CommonValidators::S32Range,
  219. "Forces rendering to a particular detail level.");
  220. endGroup("Debug");
  221. addGroup("AFX");
  222. addField("ignoreZodiacs", TypeBool, Offset(mIgnoreZodiacs, TSStatic));
  223. addField("useGradientRange", TypeBool, Offset(mHasGradients, TSStatic));
  224. addField("gradientRange", TypePoint2F, Offset(mGradientRangeUser, TSStatic));
  225. addField("invertGradientRange", TypeBool, Offset(mInvertGradientRange, TSStatic));
  226. endGroup("AFX");
  227. Parent::initPersistFields();
  228. }
  229. void TSStatic::consoleInit()
  230. {
  231. Parent::consoleInit();
  232. // Vars for debug rendering while the RoadEditor is open, only used if smEditorOpen is true.
  233. Con::addVariable("$pref::useStaticObjectFade", TypeBool, &TSStatic::smUseStaticObjectFade, "Indicates if all statics should utilize the distance-based object fadeout logic.\n");
  234. Con::addVariable("$pref::staticObjectFadeStart", TypeF32, &TSStatic::smStaticObjectFadeStart, "Distance at which static object fading begins if $pref::useStaticObjectFade is on.\n");
  235. Con::addVariable("$pref::staticObjectFadeEnd", TypeF32, &TSStatic::smStaticObjectFadeEnd, "Distance at which static object fading should have fully faded if $pref::useStaticObjectFade is on.\n");
  236. Con::addVariable("$pref::staticObjectUnfadeableSize", TypeF32, &TSStatic::smStaticObjectUnfadeableSize, "Size of object where if the bounds is at or bigger than this, it will be ignored in the $pref::useStaticObjectFade logic. Useful for very large, distance-important objects.\n");
  237. }
  238. bool TSStatic::_setFieldSkin(void* object, const char* index, const char* data)
  239. {
  240. TSStatic* ts = static_cast<TSStatic*>(object);
  241. if (ts)
  242. ts->setSkinName(data);
  243. return false;
  244. }
  245. const char* TSStatic::_getFieldSkin(void* object, const char* data)
  246. {
  247. TSStatic* ts = static_cast<TSStatic*>(object);
  248. return ts ? ts->mSkinNameHandle.getString() : "";
  249. }
  250. void TSStatic::inspectPostApply()
  251. {
  252. // Apply any transformations set in the editor
  253. Parent::inspectPostApply();
  254. if (isServerObject())
  255. {
  256. setMaskBits(-1);
  257. prepCollision();
  258. }
  259. _updateShouldTick();
  260. }
  261. bool TSStatic::onAdd()
  262. {
  263. PROFILE_SCOPE(TSStatic_onAdd);
  264. if (isServerObject())
  265. {
  266. // Handle the old "usePolysoup" field
  267. SimFieldDictionary* fieldDict = getFieldDictionary();
  268. if (fieldDict)
  269. {
  270. StringTableEntry slotName = StringTable->insert("usePolysoup");
  271. SimFieldDictionary::Entry* entry = fieldDict->findDynamicField(slotName);
  272. if (entry)
  273. {
  274. // Was "usePolysoup" set?
  275. bool usePolysoup = dAtob(entry->value);
  276. // "usePolysoup" maps to the new VisibleMesh type
  277. if (usePolysoup)
  278. mCollisionType = VisibleMesh;
  279. // Remove the field in favor on the new "collisionType" field
  280. fieldDict->setFieldValue(slotName, "");
  281. }
  282. }
  283. }
  284. if (!Parent::onAdd())
  285. return false;
  286. // Setup the shape.
  287. if (!_createShape())
  288. {
  289. Con::errorf("TSStatic::onAdd() - Shape creation failed!");
  290. return false;
  291. }
  292. setRenderTransform(mObjToWorld);
  293. // Register for the resource change signal.
  294. //ResourceManager::get().getChangedSignal().notify(this, &TSStatic::_onResourceChanged);
  295. addToScene();
  296. if (isClientObject())
  297. {
  298. mCubeReflector.unregisterReflector();
  299. if (reflectorDesc)
  300. mCubeReflector.registerReflector(this, reflectorDesc);
  301. }
  302. _updateShouldTick();
  303. // Accumulation and environment mapping
  304. if (isClientObject() && mShapeInstance)
  305. {
  306. AccumulationVolume::addObject(this);
  307. }
  308. return true;
  309. }
  310. bool TSStatic::_createShape()
  311. {
  312. // Cleanup before we create.
  313. mCollisionDetails.clear();
  314. mDecalDetails.clear();
  315. mDecalDetailsPtr = 0;
  316. mLOSDetails.clear();
  317. SAFE_DELETE(mPhysicsRep);
  318. SAFE_DELETE(mShapeInstance);
  319. mAmbientThread = NULL;
  320. //mShape = NULL;
  321. if (getShape())
  322. {
  323. if (isClientObject() &&
  324. !getShape()->preloadMaterialList(getShapeFile()) &&
  325. NetConnection::filesWereDownloaded())
  326. return false;
  327. mObjBox = getShape()->mBounds;
  328. resetWorldBox();
  329. mShapeInstance = new TSShapeInstance(getShape(), true);
  330. mShapeInstance->resetMaterialList();
  331. mShapeInstance->cloneMaterialList();
  332. if (isGhost())
  333. {
  334. // Reapply the current skin
  335. mAppliedSkinName = "";
  336. reSkin();
  337. updateMaterials();
  338. }
  339. prepCollision();
  340. // Find the "ambient" animation if it exists
  341. S32 ambientSeq = getShape()->findSequence("ambient");
  342. if (ambientSeq > -1 && !mAmbientThread)
  343. mAmbientThread = mShapeInstance->addThread();
  344. if ( mAmbientThread )
  345. mShapeInstance->setSequence(mAmbientThread, ambientSeq, mAnimOffset);
  346. // Resolve CubeReflectorDesc.
  347. if (cubeDescName.isNotEmpty())
  348. {
  349. Sim::findObject(cubeDescName, reflectorDesc);
  350. }
  351. else if (cubeDescId > 0)
  352. {
  353. Sim::findObject(cubeDescId, reflectorDesc);
  354. }
  355. }
  356. else
  357. {
  358. Con::errorf("TSStatic::_createShape() - Shape Asset %s had no valid shape!", mShapeAsset.getAssetId());
  359. return false;
  360. }
  361. //Set up the material slot vars for easy manipulation
  362. /*S32 materialCount = mShape->materialList->getMaterialNameList().size(); //mMeshAsset->getMaterialCount();
  363. //Temporarily disabled until fixup of materialName->assetId lookup logic is sorted for easy persistance
  364. if (isServerObject())
  365. {
  366. char matFieldName[128];
  367. for (U32 i = 0; i < materialCount; i++)
  368. {
  369. StringTableEntry materialname = StringTable->insert(mShape->materialList->getMaterialName(i).c_str());
  370. dSprintf(matFieldName, 128, "MaterialSlot%d", i);
  371. StringTableEntry matFld = StringTable->insert(matFieldName);
  372. setDataField(matFld, NULL, materialname);
  373. }
  374. }*/
  375. return true;
  376. }
  377. void TSStatic::onDynamicModified(const char* slotName, const char* newValue)
  378. {
  379. if (FindMatch::isMatch("materialslot*", slotName, false))
  380. {
  381. if (!getShape())
  382. return;
  383. S32 slot = -1;
  384. String outStr(String::GetTrailingNumber(slotName, slot));
  385. if (slot == -1)
  386. return;
  387. //Safe to assume the inbound value for the material will be a MaterialAsset, so lets do a lookup on the name
  388. MaterialAsset* matAsset = AssetDatabase.acquireAsset<MaterialAsset>(newValue);
  389. if (!matAsset)
  390. return;
  391. bool found = false;
  392. for (U32 i = 0; i < mChangingMaterials.size(); i++)
  393. {
  394. if (mChangingMaterials[i].slot == slot)
  395. {
  396. mChangingMaterials[i].matAsset = matAsset;
  397. mChangingMaterials[i].assetId = newValue;
  398. found = true;
  399. }
  400. }
  401. if (!found)
  402. {
  403. matMap newMatMap;
  404. newMatMap.slot = slot;
  405. newMatMap.matAsset = matAsset;
  406. newMatMap.assetId = newValue;
  407. mChangingMaterials.push_back(newMatMap);
  408. }
  409. setMaskBits(MaterialMask);
  410. }
  411. Parent::onDynamicModified(slotName, newValue);
  412. }
  413. void TSStatic::prepCollision()
  414. {
  415. // Let the client know that the collision was updated
  416. setMaskBits(UpdateCollisionMask);
  417. // Allow the ShapeInstance to prep its collision if it hasn't already
  418. if (mShapeInstance)
  419. mShapeInstance->prepCollision();
  420. // Cleanup any old collision data
  421. mCollisionDetails.clear();
  422. mDecalDetails.clear();
  423. mDecalDetailsPtr = 0;
  424. mLOSDetails.clear();
  425. mConvexList->nukeList();
  426. if (mCollisionType == CollisionMesh || mCollisionType == VisibleMesh)
  427. {
  428. getShape()->findColDetails(mCollisionType == VisibleMesh, &mCollisionDetails, &mLOSDetails, mCollisionLOD);
  429. if (mDecalType == mCollisionType)
  430. {
  431. mDecalDetailsPtr = &mCollisionDetails;
  432. }
  433. else if (mDecalType == CollisionMesh || mDecalType == VisibleMesh)
  434. {
  435. getShape()->findColDetails(mDecalType == VisibleMesh, &mDecalDetails, 0, mCollisionLOD);
  436. mDecalDetailsPtr = &mDecalDetails;
  437. }
  438. }
  439. else if (mDecalType == CollisionMesh || mDecalType == VisibleMesh)
  440. {
  441. getShape()->findColDetails(mDecalType == VisibleMesh, &mDecalDetails, 0, mCollisionLOD);
  442. mDecalDetailsPtr = &mDecalDetails;
  443. }
  444. _updatePhysics();
  445. }
  446. void TSStatic::_updatePhysics()
  447. {
  448. SAFE_DELETE(mPhysicsRep);
  449. if (!PHYSICSMGR || mCollisionType == None)
  450. return;
  451. PhysicsCollision* colShape = NULL;
  452. if (mCollisionType == Bounds)
  453. {
  454. MatrixF offset(true);
  455. offset.setPosition(getShape()->center);
  456. colShape = PHYSICSMGR->createCollision();
  457. colShape->addBox(getObjBox().getExtents() * 0.5f * mObjScale, offset);
  458. }
  459. else
  460. colShape = getShape()->buildColShape(mCollisionType == VisibleMesh, getScale());
  461. if (colShape)
  462. {
  463. PhysicsWorld* world = PHYSICSMGR->getWorld(isServerObject() ? "server" : "client");
  464. mPhysicsRep = PHYSICSMGR->createBody();
  465. mPhysicsRep->init(colShape, 0, 0, this, world);
  466. mPhysicsRep->setTransform(getTransform());
  467. }
  468. }
  469. void TSStatic::onRemove()
  470. {
  471. SAFE_DELETE(mPhysicsRep);
  472. // Accumulation
  473. if (isClientObject() && mShapeInstance)
  474. {
  475. if (mShapeInstance->hasAccumulation())
  476. AccumulationVolume::removeObject(this);
  477. }
  478. mConvexList->nukeList();
  479. removeFromScene();
  480. // Remove the resource change signal.
  481. //ResourceManager::get().getChangedSignal().remove(this, &TSStatic::_onResourceChanged);
  482. delete mShapeInstance;
  483. mShapeInstance = NULL;
  484. mAmbientThread = NULL;
  485. if (isClientObject())
  486. mCubeReflector.unregisterReflector();
  487. Parent::onRemove();
  488. }
  489. void TSStatic::onShapeChanged()
  490. {
  491. _createShape();
  492. _updateShouldTick();
  493. }
  494. void TSStatic::setSkinName(const char* name)
  495. {
  496. if (!isGhost())
  497. {
  498. if (name[0] != '\0')
  499. {
  500. // Use tags for better network performance
  501. // Should be a tag, but we'll convert to one if it isn't.
  502. if (name[0] == StringTagPrefixByte)
  503. mSkinNameHandle = NetStringHandle(U32(dAtoi(name + 1)));
  504. else
  505. mSkinNameHandle = NetStringHandle(name);
  506. }
  507. else
  508. mSkinNameHandle = NetStringHandle();
  509. setMaskBits(SkinMask);
  510. }
  511. }
  512. void TSStatic::reSkin()
  513. {
  514. if (isGhost() && mShapeInstance)
  515. {
  516. if (mSkinNameHandle.isValidString())
  517. {
  518. mShapeInstance->resetMaterialList();
  519. Vector<String> skins;
  520. String(mSkinNameHandle.getString()).split(";", skins);
  521. for (S32 i = 0; i < skins.size(); i++)
  522. {
  523. String oldSkin(mAppliedSkinName.c_str());
  524. String newSkin(skins[i]);
  525. // Check if the skin handle contains an explicit "old" base string. This
  526. // allows all models to support skinning, even if they don't follow the
  527. // "base_xxx" material naming convention.
  528. S32 split = newSkin.find('='); // "old=new" format skin?
  529. if (split != String::NPos)
  530. {
  531. oldSkin = newSkin.substr(0, split);
  532. newSkin = newSkin.erase(0, split + 1);
  533. }
  534. else
  535. {
  536. oldSkin = "";
  537. }
  538. mShapeInstance->reSkin(newSkin, oldSkin);
  539. mAppliedSkinName = newSkin;
  540. }
  541. }
  542. else
  543. {
  544. mShapeInstance->reSkin("", mAppliedSkinName);
  545. mAppliedSkinName = "";
  546. }
  547. }
  548. }
  549. void TSStatic::processTick(const Move* move)
  550. {
  551. if ( isServerObject() && mPlayAmbient && mAmbientThread )
  552. {
  553. mShapeInstance->setTimeScale(mAmbientThread, mAnimSpeed);
  554. mShapeInstance->advanceTime( TickSec, mAmbientThread );
  555. }
  556. if (isMounted())
  557. {
  558. MatrixF mat(true);
  559. mMount.object->getMountTransform(mMount.node, mMount.xfm, &mat);
  560. setTransform(mat);
  561. }
  562. }
  563. void TSStatic::interpolateTick(F32 delta)
  564. {
  565. }
  566. void TSStatic::advanceTime(F32 dt)
  567. {
  568. if ( mPlayAmbient && mAmbientThread )
  569. {
  570. mShapeInstance->setTimeScale(mAmbientThread, mAnimSpeed);
  571. mShapeInstance->advanceTime( dt, mAmbientThread );
  572. }
  573. if (isMounted())
  574. {
  575. MatrixF mat(true);
  576. mMount.object->getRenderMountTransform(dt, mMount.node, mMount.xfm, &mat);
  577. setRenderTransform(mat);
  578. }
  579. }
  580. void TSStatic::_updateShouldTick()
  581. {
  582. bool shouldTick = (mPlayAmbient && mAmbientThread) || isMounted();
  583. if (isTicking() != shouldTick)
  584. setProcessTick(shouldTick);
  585. }
  586. void TSStatic::prepRenderImage(SceneRenderState* state)
  587. {
  588. if (!mShapeAsset.isValid() || !mShapeInstance)
  589. return;
  590. Point3F cameraOffset;
  591. getRenderTransform().getColumn(3, &cameraOffset);
  592. cameraOffset -= state->getDiffuseCameraPosition();
  593. F32 dist = cameraOffset.len();
  594. if (dist < 0.01f)
  595. dist = 0.01f;
  596. if (mUseAlphaFade)
  597. {
  598. mAlphaFade = 1.0f;
  599. if ((mAlphaFadeStart < mAlphaFadeEnd) && mAlphaFadeStart > 0.1f)
  600. {
  601. if (mInvertAlphaFade)
  602. {
  603. if (dist <= mAlphaFadeStart)
  604. {
  605. return;
  606. }
  607. if (dist < mAlphaFadeEnd)
  608. {
  609. mAlphaFade = ((dist - mAlphaFadeStart) / (mAlphaFadeEnd - mAlphaFadeStart));
  610. }
  611. }
  612. else
  613. {
  614. if (dist >= mAlphaFadeEnd)
  615. {
  616. return;
  617. }
  618. if (dist > mAlphaFadeStart)
  619. {
  620. mAlphaFade -= ((dist - mAlphaFadeStart) / (mAlphaFadeEnd - mAlphaFadeStart));
  621. }
  622. }
  623. }
  624. }
  625. else if (smUseStaticObjectFade)
  626. {
  627. F32 boundsLen = getWorldSphere().radius;
  628. if (boundsLen < smStaticObjectUnfadeableSize)
  629. {
  630. F32 distAdjust = (boundsLen) / (smStaticObjectUnfadeableSize);
  631. distAdjust = 1 - distAdjust;
  632. dist *= distAdjust;
  633. mAlphaFade = 1.0f;
  634. if ((smStaticObjectFadeStart < smStaticObjectFadeEnd) && smStaticObjectFadeStart > 0.1f)
  635. {
  636. if (dist >= smStaticObjectFadeEnd)
  637. {
  638. return;
  639. }
  640. if (dist > smStaticObjectFadeStart)
  641. {
  642. mAlphaFade -= ((dist - smStaticObjectFadeStart) / (smStaticObjectFadeEnd - smStaticObjectFadeStart));
  643. }
  644. }
  645. }
  646. }
  647. F32 invScale = (1.0f / getMax(getMax(mObjScale.x, mObjScale.y), mObjScale.z));
  648. // If we're currently rendering our own reflection we
  649. // don't want to render ourselves into it.
  650. if (mCubeReflector.isRendering())
  651. return;
  652. if (mForceDetail == -1)
  653. mShapeInstance->setDetailFromDistance(state, dist * invScale);
  654. else
  655. mShapeInstance->setCurrentDetail(mForceDetail);
  656. if (mShapeInstance->getCurrentDetail() < 0)
  657. return;
  658. GFXTransformSaver saver;
  659. // Set up our TS render state.
  660. TSRenderState rdata;
  661. rdata.setSceneState(state);
  662. rdata.setFadeOverride(1.0f);
  663. rdata.setOriginSort(mUseOriginSort);
  664. if (mCubeReflector.isEnabled())
  665. rdata.setCubemap(mCubeReflector.getCubemap());
  666. // Acculumation
  667. rdata.setAccuTex(mAccuTex);
  668. // If we have submesh culling enabled then prepare
  669. // the object space frustum to pass to the shape.
  670. Frustum culler;
  671. if (mMeshCulling)
  672. {
  673. culler = state->getCullingFrustum();
  674. MatrixF xfm(true);
  675. xfm.scale(Point3F::One / getScale());
  676. xfm.mul(getRenderWorldTransform());
  677. xfm.mul(culler.getTransform());
  678. culler.setTransform(xfm);
  679. rdata.setCuller(&culler);
  680. }
  681. // We might have some forward lit materials
  682. // so pass down a query to gather lights.
  683. LightQuery query;
  684. query.init(getWorldSphere());
  685. rdata.setLightQuery(&query);
  686. MatrixF mat = getRenderTransform();
  687. mat.scale(mObjScale);
  688. GFX->setWorldMatrix(mat);
  689. if (state->isDiffusePass() && mCubeReflector.isEnabled() && mCubeReflector.getOcclusionQuery())
  690. {
  691. RenderPassManager* pass = state->getRenderPass();
  692. OccluderRenderInst* ri = pass->allocInst<OccluderRenderInst>();
  693. ri->type = RenderPassManager::RIT_Occluder;
  694. ri->query = mCubeReflector.getOcclusionQuery();
  695. mObjToWorld.mulP(mObjBox.getCenter(), &ri->position);
  696. ri->scale.set(mObjBox.getExtents());
  697. ri->orientation = pass->allocUniqueXform(mObjToWorld);
  698. ri->isSphere = false;
  699. state->getRenderPass()->addInst(ri);
  700. }
  701. if (mShapeInstance)
  702. {
  703. mShapeInstance->animate();
  704. if (mUseAlphaFade || smUseStaticObjectFade)
  705. {
  706. mShapeInstance->setAlphaAlways(mAlphaFade);
  707. S32 s = mShapeInstance->mMeshObjects.size();
  708. for (S32 x = 0; x < s; x++)
  709. {
  710. mShapeInstance->mMeshObjects[x].visible = mAlphaFade;
  711. }
  712. }
  713. }
  714. mShapeInstance->render(rdata);
  715. #ifdef TORQUE_AFX_ENABLED
  716. if (!mIgnoreZodiacs && mDecalDetailsPtr != 0)
  717. afxZodiacMgr::renderPolysoupZodiacs(state, this);
  718. #endif
  719. if (mRenderNormalScalar > 0)
  720. {
  721. ObjectRenderInst* ri = state->getRenderPass()->allocInst<ObjectRenderInst>();
  722. ri->renderDelegate.bind(this, &TSStatic::_renderNormals);
  723. ri->type = RenderPassManager::RIT_Editor;
  724. state->getRenderPass()->addInst(ri);
  725. }
  726. }
  727. void TSStatic::_renderNormals(ObjectRenderInst* ri, SceneRenderState* state, BaseMatInstance* overrideMat)
  728. {
  729. PROFILE_SCOPE(TSStatic_RenderNormals);
  730. GFXTransformSaver saver;
  731. MatrixF mat = getRenderTransform();
  732. mat.scale(mObjScale);
  733. GFX->multWorld(mat);
  734. S32 dl = mShapeInstance->getCurrentDetail();
  735. mShapeInstance->renderDebugNormals(mRenderNormalScalar, dl);
  736. }
  737. void TSStatic::onScaleChanged()
  738. {
  739. Parent::onScaleChanged();
  740. if (mPhysicsRep)
  741. {
  742. // If the editor is enabled delay the scale operation
  743. // by a few milliseconds so that we're not rebuilding
  744. // during an active scale drag operation.
  745. if (gEditingMission)
  746. mPhysicsRep->queueCallback(500, Delegate<void()>(this, &TSStatic::_updatePhysics));
  747. else
  748. _updatePhysics();
  749. }
  750. setMaskBits(ScaleMask);
  751. }
  752. void TSStatic::setTransform(const MatrixF& mat)
  753. {
  754. Parent::setTransform(mat);
  755. if (!isMounted())
  756. setMaskBits(TransformMask);
  757. if (mPhysicsRep)
  758. mPhysicsRep->setTransform(mat);
  759. // Accumulation
  760. if (isClientObject() && mShapeInstance)
  761. {
  762. if (mShapeInstance->hasAccumulation())
  763. AccumulationVolume::updateObject(this);
  764. }
  765. // Since this is a static it's render transform changes 1
  766. // to 1 with it's collision transform... no interpolation.
  767. setRenderTransform(mat);
  768. }
  769. U32 TSStatic::packUpdate(NetConnection* con, U32 mask, BitStream* stream)
  770. {
  771. U32 retMask = Parent::packUpdate(con, mask, stream);
  772. if (stream->writeFlag(mask & TransformMask))
  773. mathWrite(*stream, getTransform());
  774. if (stream->writeFlag(mask & ScaleMask))
  775. {
  776. // Only write one bit if the scale is one.
  777. if (stream->writeFlag(mObjScale != Point3F::One))
  778. mathWrite(*stream, mObjScale);
  779. }
  780. if (stream->writeFlag(mask & UpdateCollisionMask))
  781. {
  782. if (stream->writeFlag(mCollisionLOD>0))
  783. stream->writeInt(mCollisionLOD,12);
  784. stream->writeInt(mCollisionType,4);
  785. }
  786. if (stream->writeFlag(mask & SkinMask))
  787. con->packNetStringHandleU(stream, mSkinNameHandle);
  788. if (stream->writeFlag(mask & AdvancedStaticOptionsMask))
  789. {
  790. PACK_ASSET_REFACTOR(con, Shape);
  791. stream->writeInt(mDecalType,4);
  792. stream->writeFlag(mAllowPlayerStep);
  793. stream->writeFlag(mMeshCulling);
  794. stream->writeFlag(mUseOriginSort);
  795. stream->write(mRenderNormalScalar);
  796. stream->write(mForceDetail);
  797. if (stream->writeFlag(mPlayAmbient && hasAnim()))
  798. {
  799. if (stream->writeFlag(mAnimOffset != 0.0f))
  800. stream->writeFloat(mAnimOffset, 7);
  801. if (stream->writeFlag(mAnimSpeed != 1.0f))
  802. stream->writeSignedFloat(mAnimSpeed / AnimSpeedMax, 7);
  803. }
  804. }
  805. if (stream->writeFlag(mUseAlphaFade))
  806. {
  807. stream->write(mAlphaFadeStart);
  808. stream->write(mAlphaFadeEnd);
  809. stream->write(mInvertAlphaFade);
  810. }
  811. stream->writeFlag(mIgnoreZodiacs);
  812. if (stream->writeFlag(mHasGradients))
  813. {
  814. stream->writeFlag(mInvertGradientRange);
  815. stream->write(mGradientRange.x);
  816. stream->write(mGradientRange.y);
  817. }
  818. if (mLightPlugin)
  819. retMask |= mLightPlugin->packUpdate(this, AdvancedStaticOptionsMask, con, mask, stream);
  820. if (stream->writeFlag(reflectorDesc != NULL))
  821. {
  822. stream->writeRangedU32(reflectorDesc->getId(), DataBlockObjectIdFirst, DataBlockObjectIdLast);
  823. }
  824. stream->write(mOverrideColor);
  825. if (stream->writeFlag(mask & MaterialMask))
  826. {
  827. if (stream->writeFlag(mChangingMaterials.size() > 0))
  828. {
  829. stream->writeInt(mChangingMaterials.size(), 16);
  830. for (U32 i = 0; i < mChangingMaterials.size(); i++)
  831. {
  832. stream->writeInt(mChangingMaterials[i].slot, 16);
  833. NetStringHandle matNameStr = mChangingMaterials[i].assetId.c_str();
  834. con->packNetStringHandleU(stream, matNameStr);
  835. }
  836. }
  837. }
  838. return retMask;
  839. }
  840. void TSStatic::unpackUpdate(NetConnection* con, BitStream* stream)
  841. {
  842. Parent::unpackUpdate(con, stream);
  843. if (stream->readFlag()) // TransformMask
  844. {
  845. MatrixF mat;
  846. mathRead(*stream, &mat);
  847. setTransform(mat);
  848. setRenderTransform(mat);
  849. }
  850. if (stream->readFlag()) // ScaleMask
  851. {
  852. if (stream->readFlag())
  853. {
  854. VectorF scale;
  855. mathRead(*stream, &scale);
  856. setScale(scale);
  857. }
  858. else
  859. setScale(Point3F::One);
  860. }
  861. if (stream->readFlag()) // UpdateCollisionMask
  862. {
  863. U32 collisionType = CollisionMesh;
  864. if (stream->readFlag())
  865. mCollisionLOD = stream->readInt(12);
  866. collisionType = stream->readInt(4);
  867. // Handle it if we have changed CollisionType's
  868. if ((MeshType)collisionType != mCollisionType)
  869. {
  870. mCollisionType = (MeshType)collisionType;
  871. if (isProperlyAdded() && mShapeInstance)
  872. prepCollision();
  873. }
  874. }
  875. if (stream->readFlag()) // SkinMask
  876. {
  877. NetStringHandle skinDesiredNameHandle = con->unpackNetStringHandleU(stream);;
  878. if (mSkinNameHandle != skinDesiredNameHandle)
  879. {
  880. mSkinNameHandle = skinDesiredNameHandle;
  881. reSkin();
  882. }
  883. }
  884. if (stream->readFlag()) // AdvancedStaticOptionsMask
  885. {
  886. UNPACK_ASSET_REFACTOR(con, Shape);
  887. mDecalType = (MeshType)stream->readInt(4);
  888. mAllowPlayerStep = stream->readFlag();
  889. mMeshCulling = stream->readFlag();
  890. mUseOriginSort = stream->readFlag();
  891. stream->read(&mRenderNormalScalar);
  892. stream->read(&mForceDetail);
  893. mPlayAmbient = stream->readFlag();
  894. if (mPlayAmbient)
  895. {
  896. if (stream->readFlag())
  897. mAnimOffset = stream->readFloat(7);
  898. if (stream->readFlag())
  899. mAnimSpeed = stream->readSignedFloat(7) * AnimSpeedMax;
  900. }
  901. //update our shape, figuring that it likely changed
  902. _createShape();
  903. }
  904. mUseAlphaFade = stream->readFlag();
  905. if (mUseAlphaFade)
  906. {
  907. stream->read(&mAlphaFadeStart);
  908. stream->read(&mAlphaFadeEnd);
  909. stream->read(&mInvertAlphaFade);
  910. }
  911. mIgnoreZodiacs = stream->readFlag();
  912. mHasGradients = stream->readFlag();
  913. if (mHasGradients)
  914. {
  915. mInvertGradientRange = stream->readFlag();
  916. stream->read(&mGradientRange.x);
  917. stream->read(&mGradientRange.y);
  918. }
  919. if (mLightPlugin)
  920. {
  921. mLightPlugin->unpackUpdate(this, con, stream);
  922. }
  923. if (stream->readFlag())
  924. {
  925. cubeDescId = stream->readRangedU32(DataBlockObjectIdFirst, DataBlockObjectIdLast);
  926. }
  927. stream->read(&mOverrideColor);
  928. if (stream->readFlag())
  929. {
  930. mChangingMaterials.clear();
  931. if (stream->readFlag())
  932. {
  933. U32 materialCount = stream->readInt(16);
  934. for (U32 i = 0; i < materialCount; i++)
  935. {
  936. matMap newMatMap;
  937. newMatMap.slot = stream->readInt(16);
  938. newMatMap.assetId = String(con->unpackNetStringHandleU(stream).getString());
  939. //do the lookup, now
  940. newMatMap.matAsset = AssetDatabase.acquireAsset<MaterialAsset>(newMatMap.assetId);
  941. mChangingMaterials.push_back(newMatMap);
  942. }
  943. }
  944. updateMaterials();
  945. }
  946. if (isProperlyAdded())
  947. _updateShouldTick();
  948. set_special_typing();
  949. }
  950. //----------------------------------------------------------------------------
  951. bool TSStatic::castRay(const Point3F& start, const Point3F& end, RayInfo* info)
  952. {
  953. if (mCollisionType == None)
  954. return false;
  955. if (!mShapeInstance)
  956. return false;
  957. if (mCollisionType == Bounds)
  958. {
  959. F32 fst;
  960. if (!mObjBox.collideLine(start, end, &fst, &info->normal))
  961. return false;
  962. info->t = fst;
  963. info->object = this;
  964. info->point.interpolate(start, end, fst);
  965. info->material = NULL;
  966. return true;
  967. }
  968. else
  969. {
  970. RayInfo shortest = *info;
  971. RayInfo localInfo;
  972. shortest.t = 1e8f;
  973. localInfo.generateTexCoord = info->generateTexCoord;
  974. for (U32 i = 0; i < mLOSDetails.size(); i++)
  975. {
  976. mShapeInstance->animate(mLOSDetails[i]);
  977. if (mShapeInstance->castRayOpcode(mLOSDetails[i], start, end, &localInfo))
  978. {
  979. localInfo.object = this;
  980. if (localInfo.t < shortest.t)
  981. shortest = localInfo;
  982. }
  983. }
  984. if (shortest.object == this)
  985. {
  986. // Copy out the shortest time...
  987. *info = shortest;
  988. return true;
  989. }
  990. }
  991. return false;
  992. }
  993. bool TSStatic::castRayRendered(const Point3F& start, const Point3F& end, RayInfo* info)
  994. {
  995. if (!mShapeInstance)
  996. return false;
  997. // Cast the ray against the currently visible detail
  998. RayInfo localInfo;
  999. if (info && info->generateTexCoord)
  1000. localInfo.generateTexCoord = true;
  1001. bool res = mShapeInstance->castRayOpcode(mShapeInstance->getCurrentDetail(), start, end, &localInfo);
  1002. if (res)
  1003. {
  1004. *info = localInfo;
  1005. info->object = this;
  1006. return true;
  1007. }
  1008. return false;
  1009. }
  1010. bool TSStatic::buildPolyList(PolyListContext context, AbstractPolyList* polyList, const Box3F& box, const SphereF&)
  1011. {
  1012. if (!mShapeInstance)
  1013. return false;
  1014. // This is safe to set even if we're not outputing
  1015. polyList->setTransform(&mObjToWorld, mObjScale);
  1016. polyList->setObject(this);
  1017. if (context == PLC_Export)
  1018. {
  1019. // Use highest detail level
  1020. S32 dl = 0;
  1021. // Try to call on the client so we can export materials
  1022. if (isServerObject() && getClientObject())
  1023. dynamic_cast<TSStatic*>(getClientObject())->mShapeInstance->buildPolyList(polyList, dl);
  1024. else
  1025. mShapeInstance->buildPolyList(polyList, dl);
  1026. }
  1027. else if (context == PLC_Selection)
  1028. {
  1029. // Use the last rendered detail level
  1030. S32 dl = mShapeInstance->getCurrentDetail();
  1031. mShapeInstance->buildPolyListOpcode(dl, polyList, box);
  1032. }
  1033. else
  1034. {
  1035. // Figure out the mesh type we're looking for.
  1036. MeshType meshType = (context == PLC_Decal) ? mDecalType : mCollisionType;
  1037. if (meshType == None)
  1038. return false;
  1039. else if (meshType == Bounds)
  1040. polyList->addBox(mObjBox);
  1041. else if (meshType == VisibleMesh)
  1042. mShapeInstance->buildPolyListOpcode(0, polyList, box);
  1043. else if (context == PLC_Decal && mDecalDetailsPtr != 0)
  1044. {
  1045. for (U32 i = 0; i < mDecalDetailsPtr->size(); i++)
  1046. mShapeInstance->buildPolyListOpcode((*mDecalDetailsPtr)[i], polyList, box);
  1047. }
  1048. else
  1049. {
  1050. // Everything else is done from the collision meshes
  1051. // which may be built from either the visual mesh or
  1052. // special collision geometry.
  1053. for (U32 i = 0; i < mCollisionDetails.size(); i++)
  1054. mShapeInstance->buildPolyListOpcode(mCollisionDetails[i], polyList, box);
  1055. }
  1056. }
  1057. return true;
  1058. }
  1059. bool TSStatic::buildExportPolyList(ColladaUtils::ExportData* exportData, const Box3F& box, const SphereF&)
  1060. {
  1061. if (!mShapeInstance)
  1062. return false;
  1063. if (mCollisionType == Bounds)
  1064. {
  1065. ColladaUtils::ExportData::colMesh* colMesh;
  1066. exportData->colMeshes.increment();
  1067. colMesh = &exportData->colMeshes.last();
  1068. colMesh->mesh.setTransform(&mObjToWorld, mObjScale);
  1069. colMesh->mesh.setObject(this);
  1070. colMesh->mesh.addBox(mObjBox);
  1071. colMesh->colMeshName = String::ToString("ColBox%d-1", exportData->colMeshes.size());
  1072. }
  1073. else if (mCollisionType == VisibleMesh)
  1074. {
  1075. ColladaUtils::ExportData::colMesh* colMesh;
  1076. exportData->colMeshes.increment();
  1077. colMesh = &exportData->colMeshes.last();
  1078. colMesh->mesh.setTransform(&mObjToWorld, mObjScale);
  1079. colMesh->mesh.setObject(this);
  1080. mShapeInstance->buildPolyList(&colMesh->mesh, 0);
  1081. colMesh->colMeshName = String::ToString("ColMesh%d-1", exportData->colMeshes.size());
  1082. }
  1083. else if (mCollisionType == CollisionMesh)
  1084. {
  1085. // Everything else is done from the collision meshes
  1086. // which may be built from either the visual mesh or
  1087. // special collision geometry.
  1088. for (U32 i = 0; i < mCollisionDetails.size(); i++)
  1089. {
  1090. ColladaUtils::ExportData::colMesh* colMesh;
  1091. exportData->colMeshes.increment();
  1092. colMesh = &exportData->colMeshes.last();
  1093. colMesh->mesh.setTransform(&mObjToWorld, mObjScale);
  1094. colMesh->mesh.setObject(this);
  1095. mShapeInstance->buildPolyListOpcode(mCollisionDetails[i], &colMesh->mesh, box);
  1096. colMesh->colMeshName = String::ToString("ColMesh%d-1", exportData->colMeshes.size());
  1097. }
  1098. }
  1099. //Next, process the LOD levels and materials.
  1100. if (isServerObject())
  1101. {
  1102. exportData->meshData.increment();
  1103. //Prep a meshData for this shape in particular
  1104. ColladaUtils::ExportData::meshLODData* meshData = &exportData->meshData.last();
  1105. //Fill out the info we'll need later to actually append our mesh data for the detail levels during the processing phase
  1106. meshData->shapeInst = mShapeInstance;
  1107. meshData->originatingObject = this;
  1108. meshData->meshTransform = mObjToWorld;
  1109. meshData->scale = mObjScale;
  1110. //Iterate over all our detail levels
  1111. for (U32 i = 0; i < mShapeInstance->getNumDetails(); i++)
  1112. {
  1113. TSShape::Detail detail = mShapeInstance->getShape()->details[i];
  1114. String detailName = String::ToLower(mShapeInstance->getShape()->getName(detail.nameIndex));
  1115. //Skip it if it's a collision or line of sight element
  1116. if (detailName.startsWith("col") || detailName.startsWith("los"))
  1117. continue;
  1118. meshData->meshDetailLevels.increment();
  1119. ColladaUtils::ExportData::detailLevel* curDetail = &meshData->meshDetailLevels.last();
  1120. //Make sure we denote the size this detail level has
  1121. curDetail->size = getNextPow2(detail.size);
  1122. }
  1123. }
  1124. return true;
  1125. }
  1126. void TSStatic::buildConvex(const Box3F& box, Convex* convex)
  1127. {
  1128. if (mCollisionType == None)
  1129. return;
  1130. if (mShapeInstance == NULL)
  1131. return;
  1132. // These should really come out of a pool
  1133. mConvexList->collectGarbage();
  1134. if (mCollisionType == Bounds)
  1135. {
  1136. // Just return a box convex for the entire shape...
  1137. Convex* cc = 0;
  1138. CollisionWorkingList& wl = convex->getWorkingList();
  1139. for (CollisionWorkingList* itr = wl.wLink.mNext; itr != &wl; itr = itr->wLink.mNext)
  1140. {
  1141. if (itr->mConvex->getType() == BoxConvexType &&
  1142. itr->mConvex->getObject() == this)
  1143. {
  1144. cc = itr->mConvex;
  1145. break;
  1146. }
  1147. }
  1148. if (cc)
  1149. return;
  1150. // Create a new convex.
  1151. BoxConvex* cp = new BoxConvex;
  1152. mConvexList->registerObject(cp);
  1153. convex->addToWorkingList(cp);
  1154. cp->init(this);
  1155. mObjBox.getCenter(&cp->mCenter);
  1156. cp->mSize.x = mObjBox.len_x() / 2.0f;
  1157. cp->mSize.y = mObjBox.len_y() / 2.0f;
  1158. cp->mSize.z = mObjBox.len_z() / 2.0f;
  1159. }
  1160. else // CollisionMesh || VisibleMesh
  1161. {
  1162. TSStaticPolysoupConvex::smCurObject = this;
  1163. for (U32 i = 0; i < mCollisionDetails.size(); i++)
  1164. mShapeInstance->buildConvexOpcode(mObjToWorld, mObjScale, mCollisionDetails[i], box, convex, mConvexList);
  1165. TSStaticPolysoupConvex::smCurObject = NULL;
  1166. }
  1167. }
  1168. SceneObject* TSStaticPolysoupConvex::smCurObject = NULL;
  1169. TSStaticPolysoupConvex::TSStaticPolysoupConvex()
  1170. : box(0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f),
  1171. normal(0.0f, 0.0f, 0.0f, 0.0f),
  1172. idx(0),
  1173. mesh(NULL)
  1174. {
  1175. mType = TSPolysoupConvexType;
  1176. for (U32 i = 0; i < 4; ++i)
  1177. {
  1178. verts[i].set(0.0f, 0.0f, 0.0f);
  1179. }
  1180. }
  1181. Point3F TSStaticPolysoupConvex::support(const VectorF& vec) const
  1182. {
  1183. F32 bestDot = mDot(verts[0], vec);
  1184. const Point3F* bestP = &verts[0];
  1185. for (S32 i = 1; i < 4; i++)
  1186. {
  1187. F32 newD = mDot(verts[i], vec);
  1188. if (newD > bestDot)
  1189. {
  1190. bestDot = newD;
  1191. bestP = &verts[i];
  1192. }
  1193. }
  1194. return *bestP;
  1195. }
  1196. Box3F TSStaticPolysoupConvex::getBoundingBox() const
  1197. {
  1198. Box3F wbox = box;
  1199. wbox.minExtents.convolve(mObject->getScale());
  1200. wbox.maxExtents.convolve(mObject->getScale());
  1201. mObject->getTransform().mul(wbox);
  1202. return wbox;
  1203. }
  1204. Box3F TSStaticPolysoupConvex::getBoundingBox(const MatrixF& mat, const Point3F& scale) const
  1205. {
  1206. AssertISV(false, "TSStaticPolysoupConvex::getBoundingBox(m,p) - Not implemented. -- XEA");
  1207. return box;
  1208. }
  1209. void TSStaticPolysoupConvex::getPolyList(AbstractPolyList* list)
  1210. {
  1211. // Transform the list into object space and set the pointer to the object
  1212. MatrixF i(mObject->getTransform());
  1213. Point3F iS(mObject->getScale());
  1214. list->setTransform(&i, iS);
  1215. list->setObject(mObject);
  1216. // Add only the original collision triangle
  1217. S32 base = list->addPoint(verts[0]);
  1218. list->addPoint(verts[2]);
  1219. list->addPoint(verts[1]);
  1220. list->begin(0, (U32)idx ^ (uintptr_t)mesh);
  1221. list->vertex(base + 2);
  1222. list->vertex(base + 1);
  1223. list->vertex(base + 0);
  1224. list->plane(base + 0, base + 1, base + 2);
  1225. list->end();
  1226. }
  1227. void TSStaticPolysoupConvex::getFeatures(const MatrixF& mat, const VectorF& n, ConvexFeature* cf)
  1228. {
  1229. cf->material = 0;
  1230. cf->mObject = mObject;
  1231. // For a tetrahedron this is pretty easy... first
  1232. // convert everything into world space.
  1233. Point3F tverts[4];
  1234. mat.mulP(verts[0], &tverts[0]);
  1235. mat.mulP(verts[1], &tverts[1]);
  1236. mat.mulP(verts[2], &tverts[2]);
  1237. mat.mulP(verts[3], &tverts[3]);
  1238. // points...
  1239. S32 firstVert = cf->mVertexList.size();
  1240. cf->mVertexList.increment(); cf->mVertexList.last() = tverts[0];
  1241. cf->mVertexList.increment(); cf->mVertexList.last() = tverts[1];
  1242. cf->mVertexList.increment(); cf->mVertexList.last() = tverts[2];
  1243. cf->mVertexList.increment(); cf->mVertexList.last() = tverts[3];
  1244. // edges...
  1245. cf->mEdgeList.increment();
  1246. cf->mEdgeList.last().vertex[0] = firstVert + 0;
  1247. cf->mEdgeList.last().vertex[1] = firstVert + 1;
  1248. cf->mEdgeList.increment();
  1249. cf->mEdgeList.last().vertex[0] = firstVert + 1;
  1250. cf->mEdgeList.last().vertex[1] = firstVert + 2;
  1251. cf->mEdgeList.increment();
  1252. cf->mEdgeList.last().vertex[0] = firstVert + 2;
  1253. cf->mEdgeList.last().vertex[1] = firstVert + 0;
  1254. cf->mEdgeList.increment();
  1255. cf->mEdgeList.last().vertex[0] = firstVert + 3;
  1256. cf->mEdgeList.last().vertex[1] = firstVert + 0;
  1257. cf->mEdgeList.increment();
  1258. cf->mEdgeList.last().vertex[0] = firstVert + 3;
  1259. cf->mEdgeList.last().vertex[1] = firstVert + 1;
  1260. cf->mEdgeList.increment();
  1261. cf->mEdgeList.last().vertex[0] = firstVert + 3;
  1262. cf->mEdgeList.last().vertex[1] = firstVert + 2;
  1263. // triangles...
  1264. cf->mFaceList.increment();
  1265. cf->mFaceList.last().normal = PlaneF(tverts[2], tverts[1], tverts[0]);
  1266. cf->mFaceList.last().vertex[0] = firstVert + 2;
  1267. cf->mFaceList.last().vertex[1] = firstVert + 1;
  1268. cf->mFaceList.last().vertex[2] = firstVert + 0;
  1269. cf->mFaceList.increment();
  1270. cf->mFaceList.last().normal = PlaneF(tverts[1], tverts[0], tverts[3]);
  1271. cf->mFaceList.last().vertex[0] = firstVert + 1;
  1272. cf->mFaceList.last().vertex[1] = firstVert + 0;
  1273. cf->mFaceList.last().vertex[2] = firstVert + 3;
  1274. cf->mFaceList.increment();
  1275. cf->mFaceList.last().normal = PlaneF(tverts[2], tverts[1], tverts[3]);
  1276. cf->mFaceList.last().vertex[0] = firstVert + 2;
  1277. cf->mFaceList.last().vertex[1] = firstVert + 1;
  1278. cf->mFaceList.last().vertex[2] = firstVert + 3;
  1279. cf->mFaceList.increment();
  1280. cf->mFaceList.last().normal = PlaneF(tverts[0], tverts[2], tverts[3]);
  1281. cf->mFaceList.last().vertex[0] = firstVert + 0;
  1282. cf->mFaceList.last().vertex[1] = firstVert + 2;
  1283. cf->mFaceList.last().vertex[2] = firstVert + 3;
  1284. // All done!
  1285. }
  1286. void TSStatic::onMount(SceneObject* obj, S32 node)
  1287. {
  1288. Parent::onMount(obj, node);
  1289. _updateShouldTick();
  1290. }
  1291. void TSStatic::onUnmount(SceneObject* obj, S32 node)
  1292. {
  1293. Parent::onUnmount(obj, node);
  1294. setMaskBits(TransformMask);
  1295. _updateShouldTick();
  1296. }
  1297. U32 TSStatic::getNumDetails()
  1298. {
  1299. if (isServerObject() && getClientObject())
  1300. {
  1301. TSStatic* clientShape = dynamic_cast<TSStatic*>(getClientObject());
  1302. return clientShape->mShapeInstance->getNumDetails();
  1303. }
  1304. return 0;
  1305. };
  1306. void TSStatic::updateMaterials()
  1307. {
  1308. if (mChangingMaterials.empty() || !mShapeInstance)
  1309. return;
  1310. TSMaterialList* pMatList = mShapeInstance->getMaterialList();
  1311. String path;
  1312. if (mShapeAsset->isAssetValid())
  1313. path = mShapeAsset->getShapeFile();
  1314. else
  1315. path = mShapeFile;
  1316. pMatList->setTextureLookupPath(path);
  1317. bool found = false;
  1318. const Vector<String>& materialNames = pMatList->getMaterialNameList();
  1319. for (S32 i = 0; i < materialNames.size(); i++)
  1320. {
  1321. if (found)
  1322. break;
  1323. for (U32 m = 0; m < mChangingMaterials.size(); m++)
  1324. {
  1325. if (mChangingMaterials[m].slot == i)
  1326. {
  1327. //Fetch the actual material asset
  1328. pMatList->renameMaterial(i, mChangingMaterials[m].matAsset->getMaterialDefinitionName());
  1329. found = true;
  1330. break;
  1331. }
  1332. }
  1333. }
  1334. // Initialize the material instances
  1335. mShapeInstance->initMaterialList();
  1336. }
  1337. void TSStatic::getUtilizedAssets(Vector<StringTableEntry>* usedAssetsList)
  1338. {
  1339. U32 assetStatus = ShapeAsset::getAssetErrCode(mShapeAsset);
  1340. if (assetStatus == AssetBase::Ok)
  1341. {
  1342. usedAssetsList->push_back_unique(mShapeAsset->getAssetId());
  1343. }
  1344. }
  1345. //------------------------------------------------------------------------
  1346. //These functions are duplicated in tsStatic and shapeBase.
  1347. //They each function a little differently; but achieve the same purpose of gathering
  1348. //target names/counts without polluting simObject.
  1349. #ifdef TORQUE_TOOLS
  1350. void TSStatic::onInspect(GuiInspector* inspector)
  1351. {
  1352. //if (mShapeAsset == nullptr)
  1353. return;
  1354. /*
  1355. //Put the GameObject group before everything that'd be gameobject-effecting, for orginazational purposes
  1356. GuiInspectorGroup* materialGroup = inspector->findExistentGroup(StringTable->insert("Materials"));
  1357. if (!materialGroup)
  1358. return;
  1359. GuiControl* stack = dynamic_cast<GuiControl*>(materialGroup->findObjectByInternalName(StringTable->insert("Stack")));
  1360. //Do this on both the server and client
  1361. TSMaterialList* matList = mShapeInstance->getMaterialList();
  1362. Vector<String> matListNames = matList->getMaterialNameList();
  1363. S32 materialCount = matListNames.size();
  1364. if (isServerObject())
  1365. {
  1366. //next, get a listing of our materials in the shape, and build our field list for them
  1367. char matFieldName[128];
  1368. for (U32 i = 0; i < materialCount; i++)
  1369. {
  1370. StringTableEntry materialname = StringTable->insert(mShapeInstance->getMaterialList()->getMaterialName(i).c_str());
  1371. AssetPtr<MaterialAsset> matAsset;
  1372. if(MaterialAsset::getAssetByMaterialName(materialname, &matAsset) == MaterialAsset::Ok)
  1373. {
  1374. dSprintf(matFieldName, 128, "MaterialSlot%d", i);
  1375. GuiInspectorField* fieldGui = materialGroup->constructField(TypeMaterialAssetPtr);
  1376. fieldGui->init(inspector, materialGroup);
  1377. fieldGui->setSpecialEditField(true);
  1378. fieldGui->setTargetObject(this);
  1379. StringTableEntry fldnm = StringTable->insert(matFieldName);
  1380. fieldGui->setSpecialEditVariableName(fldnm);
  1381. fieldGui->setInspectorField(NULL, fldnm);
  1382. fieldGui->setDocs("");
  1383. if (fieldGui->registerObject())
  1384. {
  1385. StringTableEntry fieldValue = matAsset->getAssetId();
  1386. GuiInspectorTypeMaterialAssetPtr* matFieldPtr = dynamic_cast<GuiInspectorTypeMaterialAssetPtr*>(fieldGui);
  1387. matFieldPtr->setPreviewImage(fieldValue);
  1388. //Check if we'd already actually changed it, and display the modified value
  1389. for (U32 c = 0; c < mChangingMaterials.size(); c++)
  1390. {
  1391. if (mChangingMaterials[c].slot == i)
  1392. {
  1393. fieldValue = StringTable->insert(mChangingMaterials[i].assetId.c_str());
  1394. break;
  1395. }
  1396. }
  1397. fieldGui->setValue(fieldValue);
  1398. stack->addObject(fieldGui);
  1399. }
  1400. else
  1401. {
  1402. SAFE_DELETE(fieldGui);
  1403. }
  1404. }
  1405. }
  1406. }
  1407. */
  1408. }
  1409. #endif
  1410. DefineEngineMethod(TSStatic, getTargetName, const char*, (S32 index), (0),
  1411. "Get the name of the indexed shape material.\n"
  1412. "@param index index of the material to get (valid range is 0 - getTargetCount()-1).\n"
  1413. "@return the name of the indexed material.\n"
  1414. "@see getTargetCount()\n")
  1415. {
  1416. TSStatic* obj = dynamic_cast<TSStatic*> (object);
  1417. if (obj)
  1418. {
  1419. // Try to use the client object (so we get the reskinned targets in the Material Editor)
  1420. if ((TSStatic*)obj->getClientObject())
  1421. obj = (TSStatic*)obj->getClientObject();
  1422. return obj->getShapeInstance()->getTargetName(index);
  1423. }
  1424. return "";
  1425. }
  1426. DefineEngineMethod(TSStatic, getTargetCount, S32, (), ,
  1427. "Get the number of materials in the shape.\n"
  1428. "@return the number of materials in the shape.\n"
  1429. "@see getTargetName()\n")
  1430. {
  1431. TSStatic* obj = dynamic_cast<TSStatic*> (object);
  1432. if (obj)
  1433. {
  1434. // Try to use the client object (so we get the reskinned targets in the Material Editor)
  1435. if ((TSStatic*)obj->getClientObject())
  1436. obj = (TSStatic*)obj->getClientObject();
  1437. return obj->getShapeInstance()->getTargetCount();
  1438. }
  1439. return -1;
  1440. }
  1441. // This method is able to change materials per map to with others. The material that is being replaced is being mapped to
  1442. // unmapped_mat as a part of this transition
  1443. DefineEngineMethod(TSStatic, changeMaterial, void, (const char* mapTo, Material* oldMat, Material* newMat), ("", nullAsType<Material*>(), nullAsType<Material*>()),
  1444. "@brief Change one of the materials on the shape.\n\n"
  1445. "This method changes materials per mapTo with others. The material that "
  1446. "is being replaced is mapped to unmapped_mat as a part of this transition.\n"
  1447. "@note Warning, right now this only sort of works. It doesn't do a live "
  1448. "update like it should.\n"
  1449. "@param mapTo the name of the material target to remap (from getTargetName)\n"
  1450. "@param oldMat the old Material that was mapped \n"
  1451. "@param newMat the new Material to map\n\n"
  1452. "@tsexample\n"
  1453. "// remap the first material in the shape\n"
  1454. "%mapTo = %obj.getTargetName( 0 );\n"
  1455. "%obj.changeMaterial( %mapTo, 0, MyMaterial );\n"
  1456. "@endtsexample\n")
  1457. {
  1458. // if no valid new material, theres no reason for doing this
  1459. if (!newMat)
  1460. {
  1461. Con::errorf("TSShape::changeMaterial failed: New material does not exist!");
  1462. return;
  1463. }
  1464. TSMaterialList* shapeMaterialList = object->getShape()->materialList;
  1465. // Check the mapTo name exists for this shape
  1466. S32 matIndex = shapeMaterialList->getMaterialNameList().find_next(String(mapTo));
  1467. if (matIndex < 0)
  1468. {
  1469. Con::errorf("TSShape::changeMaterial failed: Invalid mapTo name '%s'", mapTo);
  1470. return;
  1471. }
  1472. // Lets remap the old material off, so as to let room for our current material room to claim its spot
  1473. if (oldMat)
  1474. oldMat->mMapTo = String("unmapped_mat");
  1475. newMat->mMapTo = mapTo;
  1476. // Map the material by name in the matmgr
  1477. MATMGR->mapMaterial(mapTo, newMat->getName());
  1478. // Replace instances with the new material being traded in. Lets make sure that we only
  1479. // target the specific targets per inst, this is actually doing more than we thought
  1480. delete shapeMaterialList->mMatInstList[matIndex];
  1481. shapeMaterialList->mMatInstList[matIndex] = newMat->createMatInstance();
  1482. // Finish up preparing the material instances for rendering
  1483. const GFXVertexFormat* flags = getGFXVertexFormat<GFXVertexPNTTB>();
  1484. FeatureSet features = MATMGR->getDefaultFeatures();
  1485. shapeMaterialList->getMaterialInst(matIndex)->init(features, flags);
  1486. }
  1487. DefineEngineMethod(TSStatic, getModelFile, const char*, (), ,
  1488. "@brief Get the model filename used by this shape.\n\n"
  1489. "@return the shape filename\n\n"
  1490. "@tsexample\n"
  1491. "// Acquire the model filename used on this shape.\n"
  1492. "%modelFilename = %obj.getModelFile();\n"
  1493. "@endtsexample\n"
  1494. )
  1495. {
  1496. return object->getShapeFile();
  1497. }
  1498. void TSStatic::set_special_typing()
  1499. {
  1500. if (mCollisionType == VisibleMesh || mCollisionType == CollisionMesh)
  1501. mTypeMask |= InteriorLikeObjectType;
  1502. else
  1503. mTypeMask &= ~InteriorLikeObjectType;
  1504. }
  1505. void TSStatic::onStaticModified(const char* slotName, const char* newValue)
  1506. {
  1507. #ifdef TORQUE_AFX_ENABLED
  1508. if (slotName == afxZodiacData::GradientRangeSlot)
  1509. {
  1510. afxZodiacData::convertGradientRangeFromDegrees(mGradientRange, mGradientRangeUser);
  1511. return;
  1512. }
  1513. #endif
  1514. set_special_typing();
  1515. }
  1516. void TSStatic::setSelectionFlags(U8 flags)
  1517. {
  1518. Parent::setSelectionFlags(flags);
  1519. if (!mShapeInstance || !isClientObject())
  1520. return;
  1521. if (!mShapeInstance->ownMaterialList())
  1522. return;
  1523. TSMaterialList* pMatList = mShapeInstance->getMaterialList();
  1524. for (S32 j = 0; j < pMatList->size(); j++)
  1525. {
  1526. BaseMatInstance* bmi = pMatList->getMaterialInst(j);
  1527. bmi->setSelectionHighlighting(needsSelectionHighlighting());
  1528. }
  1529. }
  1530. bool TSStatic::hasNode(const char* nodeName)
  1531. {
  1532. S32 nodeIDx = getShape()->findNode(nodeName);
  1533. return nodeIDx >= 0;
  1534. }
  1535. void TSStatic::getNodeTransform(const char *nodeName, const MatrixF &xfm, MatrixF *outMat)
  1536. {
  1537. S32 nodeIDx = getShape()->findNode(nodeName);
  1538. MatrixF nodeTransform(xfm);
  1539. const Point3F& scale = getScale();
  1540. if (nodeIDx != -1)
  1541. {
  1542. nodeTransform = mShapeInstance->mNodeTransforms[nodeIDx];
  1543. nodeTransform.mul(xfm);
  1544. }
  1545. // The position of the mount point needs to be scaled.
  1546. Point3F position = nodeTransform.getPosition();
  1547. position.convolve(scale);
  1548. nodeTransform.setPosition(position);
  1549. // Also we would like the object to be scaled to the model.
  1550. outMat->mul(mObjToWorld, nodeTransform);
  1551. return;
  1552. }
  1553. DefineEngineMethod(TSStatic, hasNode, bool, (const char* nodeName), ,
  1554. "@brief Get if this model has this node name.\n\n")
  1555. {
  1556. return object->hasNode(nodeName);
  1557. }
  1558. DefineEngineMethod(TSStatic, getNodeTransform, TransformF, (const char *nodeName), ,
  1559. "@brief Get the world transform of the specified node name.\n\n"
  1560. "@param node name query\n"
  1561. "@return the mount transform\n\n")
  1562. {
  1563. MatrixF xf(true);
  1564. object->getNodeTransform(nodeName, MatrixF::Identity, &xf);
  1565. return xf;
  1566. }
  1567. DefineEngineMethod(TSStatic, setSkinName, void, (const char* name), ,
  1568. "@brief Apply a new skin to this shape.\n\n"
  1569. "'Skinning' the shape effectively renames the material targets, allowing "
  1570. "different materials to be used on different instances of the same model.\n\n"
  1571. "@param name name of the skin to apply\n\n"
  1572. "@see skin\n"
  1573. "@see getSkinName()\n")
  1574. {
  1575. object->setSkinName(name);
  1576. }