| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623 | //-----------------------------------------------------------------------------// Copyright (c) 2012 GarageGames, LLC//// 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.//-----------------------------------------------------------------------------//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//// Arcane-FX for MIT Licensed Open Source version of Torque 3D from GarageGames// Copyright (C) 2015 Faust Logic, Inc.//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~//~~~~~~~~~~~~~~~~~~~~~//#include "T3D/physicalZone.h"#include "core/stream/bitStream.h"#include "collision/boxConvex.h"#include "collision/clippedPolyList.h"#include "console/consoleTypes.h"#include "math/mathIO.h"#include "scene/sceneRenderState.h"#include "T3D/trigger.h"#include "gfx/gfxTransformSaver.h"#include "renderInstance/renderPassManager.h"#include "gfx/gfxDrawUtil.h"#include "console/engineAPI.h"//#include "console/engineTypes.h"#include "sim/netConnection.h"IMPLEMENT_CO_NETOBJECT_V1(PhysicalZone);ConsoleDocClass( PhysicalZone,   "@brief Physical Zones are areas that modify the player's gravity and/or velocity and/or applied force.\n\n"   "The datablock properties determine how the physics, velocity and applied forces affect a player who enters this zone.\n"   "@tsexample\n"      "new PhysicalZone(Team1JumpPad) {\n"         "velocityMod = \"1\";"         "gravityMod = \"0\";\n"         "appliedForce = \"0 0 20000\";\n"         "polyhedron = \"0.0000000 0.0000000 0.0000000 1.0000000 0.0000000 0.0000000 0.0000000 -1.0000000 0.0000000 0.0000000 0.0000000 1.0000000\";\n"         "position = \"273.559 -166.371 249.856\";\n"         "rotation = \"0 0 1 13.0216\";\n"         "scale = \"8 4.95 28.31\";\n"         "isRenderEnabled = \"true\";\n"         "canSaveDynamicFields = \"1\";\n"         "enabled = \"1\";\n"      "};\n"   "@endtsexample\n\n"   "@ingroup enviroMisc\n");bool PhysicalZone::smRenderPZones = false;DefineEngineMethod(PhysicalZone, activate, void, (),, "Activate the physical zone's effects.\n"													"@tsexample\n"														"// Activate effects for a specific physical zone.\n"														"%thisPhysicalZone.activate();\n"													"@endtsexample\n"													"@ingroup Datablocks\n"				  ){   if (object->isClientObject())      return;   object->activate();}DefineEngineMethod(PhysicalZone, deactivate, void, (),, "Deactivate the physical zone's effects.\n"													"@tsexample\n"														"// Deactivate effects for a specific physical zone.\n"														"%thisPhysicalZone.deactivate();\n"													"@endtsexample\n"													"@ingroup Datablocks\n"				  ){   if (object->isClientObject())      return;   object->deactivate();}//--------------------------------------------------------------------------//--------------------------------------//PhysicalZone::PhysicalZone(){   mNetFlags.set(Ghostable | ScopeAlways);   mTypeMask |= PhysicalZoneObjectType;   mVelocityMod = 1.0f;   mGravityMod  = 1.0f;   mAppliedForce.set(0, 0, 0);   mConvexList = new Convex;   mActive = true;   force_type = VECTOR;   force_mag = 0.0f;   orient_force = false;   fade_amt = 1.0f;}PhysicalZone::~PhysicalZone(){   delete mConvexList;   mConvexList = NULL;}ImplementEnumType( PhysicalZone_ForceType, "Possible physical zone force types.\n" "@ingroup PhysicalZone\n\n" )   { PhysicalZone::VECTOR,          "vector",        "..." },   { PhysicalZone::SPHERICAL,       "spherical",     "..." },   { PhysicalZone::CYLINDRICAL,     "cylindrical",   "..." },   // aliases   { PhysicalZone::SPHERICAL,       "sphere",        "..." },   { PhysicalZone::CYLINDRICAL,     "cylinder",      "..." },EndImplementEnumType;//--------------------------------------------------------------------------void PhysicalZone::consoleInit(){   Con::addVariable( "$PhysicalZone::renderZones", TypeBool, &smRenderPZones, "If true, a box will render around the location of all PhysicalZones.\n"	   "@ingroup EnviroMisc\n");}void PhysicalZone::initPersistFields(){   addGroup("Misc");   addField("velocityMod",  TypeF32,               Offset(mVelocityMod,  PhysicalZone), "Multiply velocity of objects entering zone by this value every tick.");   addField("gravityMod",   TypeF32,               Offset(mGravityMod,   PhysicalZone), "Gravity in PhysicalZone. Multiplies against standard gravity.");   addField("appliedForce", TypePoint3F,           Offset(mAppliedForce, PhysicalZone), "Three-element floating point value representing forces in three axes to apply to objects entering PhysicalZone.");   addField("polyhedron",   TypeTriggerPolyhedron, Offset(mPolyhedron,   PhysicalZone),      "The polyhedron type is really a quadrilateral and consists of a corner"      "point followed by three vectors representing the edges extending from the corner." );   endGroup("Misc");   addGroup("AFX");   addField("forceType", TYPEID<PhysicalZone::ForceType>(), Offset(force_type, PhysicalZone));   addField("orientForce", TypeBool, Offset(orient_force, PhysicalZone));   endGroup("AFX");   Parent::initPersistFields();}//--------------------------------------------------------------------------bool PhysicalZone::onAdd(){   if(!Parent::onAdd())      return false;   if (mVelocityMod < -40.0f || mVelocityMod > 40.0f) {      Con::errorf("PhysicalZone: velocity mod out of range.  [-40, 40]");      mVelocityMod = mVelocityMod < -40.0f ? -40.0f : 40.0f;   }   if (mGravityMod < -40.0f || mGravityMod > 40.0f) {      Con::errorf("PhysicalZone: GravityMod out of range.  [-40, 40]");      mGravityMod = mGravityMod < -40.0f ? -40.0f : 40.0f;   }   static const char* coordString[] = { "x", "y", "z" };   F32* p = mAppliedForce;   for (U32 i = 0; i < 3; i++) {      if (p[i] < -40000.0f || p[i] > 40000.0f) {         Con::errorf("PhysicalZone: applied force: %s out of range.  [-40000, 40000]", coordString[i]);         p[i] = p[i] < -40000.0f ? -40000.0f : 40000.0f;      }   }   Polyhedron temp = mPolyhedron;   setPolyhedron(temp);   switch (force_type)   {   case SPHERICAL:      force_mag = mAppliedForce.magnitudeSafe();      break;   case CYLINDRICAL:      {         Point3F force_vec = mAppliedForce;         force_vec.z = 0.0;         force_mag = force_vec.magnitudeSafe();      }      break;   }   addToScene();   return true;}void PhysicalZone::onRemove(){   mConvexList->nukeList();   removeFromScene();   Parent::onRemove();}void PhysicalZone::inspectPostApply(){   setPolyhedron(mPolyhedron);   Parent::inspectPostApply();}//------------------------------------------------------------------------------void PhysicalZone::setTransform(const MatrixF & mat){   Parent::setTransform(mat);      MatrixF base(true);      base.scale(Point3F(1.0/mObjScale.x,                         1.0/mObjScale.y,                         1.0/mObjScale.z));      base.mul(mWorldToObj);      mClippedList.setBaseTransform(base);   if (isServerObject())      setMaskBits(MoveMask);}void PhysicalZone::prepRenderImage( SceneRenderState *state ){   // only render if selected or render flag is set   if ( !smRenderPZones && !isSelected() )      return;   ObjectRenderInst *ri = state->getRenderPass()->allocInst<ObjectRenderInst>();   ri->renderDelegate.bind( this, &PhysicalZone::renderObject );   ri->type = RenderPassManager::RIT_Editor;   ri->defaultKey = 0;   ri->defaultKey2 = 0;   state->getRenderPass()->addInst( ri );}void PhysicalZone::renderObject( ObjectRenderInst *ri,                                 SceneRenderState *state,                                 BaseMatInstance *overrideMat ){   if (overrideMat)      return;   GFXStateBlockDesc desc;   desc.setZReadWrite( true, false );   desc.setBlend( true );   desc.setCullMode( GFXCullNone );   GFXTransformSaver saver;   MatrixF mat = getRenderTransform();   mat.scale( getScale() );   GFX->multWorld( mat );   GFXDrawUtil *drawer = GFX->getDrawUtil();   drawer->drawPolyhedron( desc, mPolyhedron, ColorI( 0, 255, 0, 45 ) );   desc.setFillModeWireframe();   drawer->drawPolyhedron( desc, mPolyhedron, ColorI::BLACK );}//--------------------------------------------------------------------------U32 PhysicalZone::packUpdate(NetConnection* con, U32 mask, BitStream* stream){   U32 i;   U32 retMask = Parent::packUpdate(con, mask, stream);   if (stream->writeFlag(mask & PolyhedronMask))    {      // Write the polyhedron      stream->write(mPolyhedron.pointList.size());      for (i = 0; i < mPolyhedron.pointList.size(); i++)         mathWrite(*stream, mPolyhedron.pointList[i]);      stream->write(mPolyhedron.planeList.size());      for (i = 0; i < mPolyhedron.planeList.size(); i++)         mathWrite(*stream, mPolyhedron.planeList[i]);      stream->write(mPolyhedron.edgeList.size());      for (i = 0; i < mPolyhedron.edgeList.size(); i++) {         const Polyhedron::Edge& rEdge = mPolyhedron.edgeList[i];         stream->write(rEdge.face[0]);         stream->write(rEdge.face[1]);         stream->write(rEdge.vertex[0]);         stream->write(rEdge.vertex[1]);      }   }   if (stream->writeFlag(mask & MoveMask))   {      stream->writeAffineTransform(mObjToWorld);      mathWrite(*stream, mObjScale);   }   if (stream->writeFlag(mask & SettingsMask))   {      stream->write(mVelocityMod);      stream->write(mGravityMod);      mathWrite(*stream, mAppliedForce);      stream->writeInt(force_type, FORCE_TYPE_BITS);      stream->writeFlag(orient_force);   }   if (stream->writeFlag(mask & FadeMask))   {      U8 fade_byte = (U8)(fade_amt*255.0f);      stream->write(fade_byte);   }   stream->writeFlag(mActive);   // AFX CODE BLOCK (enhanced-physical-zone)(pz-opt) >>   return retMask;}void PhysicalZone::unpackUpdate(NetConnection* con, BitStream* stream){   Parent::unpackUpdate(con, stream);   bool new_ph = false;   if (stream->readFlag()) // PolyhedronMask   {      U32 i, size;      Polyhedron tempPH;      // Read the polyhedron      stream->read(&size);      tempPH.pointList.setSize(size);      for (i = 0; i < tempPH.pointList.size(); i++)         mathRead(*stream, &tempPH.pointList[i]);      stream->read(&size);      tempPH.planeList.setSize(size);      for (i = 0; i < tempPH.planeList.size(); i++)         mathRead(*stream, &tempPH.planeList[i]);      stream->read(&size);      tempPH.edgeList.setSize(size);      for (i = 0; i < tempPH.edgeList.size(); i++) {         Polyhedron::Edge& rEdge = tempPH.edgeList[i];         stream->read(&rEdge.face[0]);         stream->read(&rEdge.face[1]);         stream->read(&rEdge.vertex[0]);         stream->read(&rEdge.vertex[1]);      }      setPolyhedron(tempPH);      new_ph = true;   }   if (stream->readFlag()) // MoveMask   {      MatrixF temp;      stream->readAffineTransform(&temp);      Point3F tempScale;      mathRead(*stream, &tempScale);      //if (!new_ph)      //{      //  Polyhedron rPolyhedron = mPolyhedron;      //  setPolyhedron(rPolyhedron);      //}      setScale(tempScale);      setTransform(temp);   }   if (stream->readFlag()) //SettingsMask   {      stream->read(&mVelocityMod);      stream->read(&mGravityMod);      mathRead(*stream, &mAppliedForce);      force_type = stream->readInt(FORCE_TYPE_BITS); // AFX      orient_force = stream->readFlag(); // AFX   }   if (stream->readFlag()) //FadeMask   {      U8 fade_byte;      stream->read(&fade_byte);      fade_amt = ((F32)fade_byte)/255.0f;   }   else      fade_amt = 1.0f;   mActive = stream->readFlag();}//--------------------------------------------------------------------------void PhysicalZone::setPolyhedron(const Polyhedron& rPolyhedron){   mPolyhedron = rPolyhedron;   if (mPolyhedron.pointList.size() != 0) {      mObjBox.minExtents.set(1e10, 1e10, 1e10);      mObjBox.maxExtents.set(-1e10, -1e10, -1e10);      for (U32 i = 0; i < mPolyhedron.pointList.size(); i++) {         mObjBox.minExtents.setMin(mPolyhedron.pointList[i]);         mObjBox.maxExtents.setMax(mPolyhedron.pointList[i]);      }   } else {      mObjBox.minExtents.set(-0.5, -0.5, -0.5);      mObjBox.maxExtents.set( 0.5,  0.5,  0.5);   }   MatrixF xform = getTransform();   setTransform(xform);   mClippedList.clear();   mClippedList.mPlaneList = mPolyhedron.planeList;   MatrixF base(true);   base.scale(Point3F(1.0/mObjScale.x,                      1.0/mObjScale.y,                      1.0/mObjScale.z));   base.mul(mWorldToObj);   mClippedList.setBaseTransform(base);}//--------------------------------------------------------------------------void PhysicalZone::buildConvex(const Box3F& box, Convex* convex){   // These should really come out of a pool   mConvexList->collectGarbage();   Box3F realBox = box;   mWorldToObj.mul(realBox);   realBox.minExtents.convolveInverse(mObjScale);   realBox.maxExtents.convolveInverse(mObjScale);   if (realBox.isOverlapped(getObjBox()) == false)      return;   // Just return a box convex for the entire shape...   Convex* cc = 0;   CollisionWorkingList& wl = convex->getWorkingList();   for (CollisionWorkingList* itr = wl.wLink.mNext; itr != &wl; itr = itr->wLink.mNext) {      if (itr->mConvex->getType() == BoxConvexType &&          itr->mConvex->getObject() == this) {         cc = itr->mConvex;         break;      }   }   if (cc)      return;   // Create a new convex.   BoxConvex* cp = new BoxConvex;   mConvexList->registerObject(cp);   convex->addToWorkingList(cp);   cp->init(this);   mObjBox.getCenter(&cp->mCenter);   cp->mSize.x = mObjBox.len_x() / 2.0f;   cp->mSize.y = mObjBox.len_y() / 2.0f;   cp->mSize.z = mObjBox.len_z() / 2.0f;}bool PhysicalZone::testObject(SceneObject* enter){   // TODO: This doesn't look like it's testing against the polyhedron at   // all.  And whats the point of building a convex if no collision methods   // are implemented?   if (mPolyhedron.pointList.size() == 0)      return false;   mClippedList.clear();   SphereF sphere;   sphere.center = (mWorldBox.minExtents + mWorldBox.maxExtents) * 0.5;   VectorF bv = mWorldBox.maxExtents - sphere.center;   sphere.radius = bv.len();   enter->buildPolyList(PLC_Collision, &mClippedList, mWorldBox, sphere);   return mClippedList.isEmpty() == false;}bool PhysicalZone::testBox( const Box3F &box ) const{   return mWorldBox.isOverlapped( box );}void PhysicalZone::activate(){   AssertFatal(isServerObject(), "Client objects not allowed in ForceFieldInstance::open()");   if (mActive != true)      setMaskBits(ActiveMask);   mActive = true;}void PhysicalZone::deactivate(){   AssertFatal(isServerObject(), "Client objects not allowed in ForceFieldInstance::close()");   if (mActive != false)      setMaskBits(ActiveMask);   mActive = false;}void PhysicalZone::onStaticModified(const char* slotName, const char*newValue){   if (dStricmp(slotName, "appliedForce") == 0 || dStricmp(slotName, "forceType") == 0)   {      switch (force_type)      {      case SPHERICAL:         force_mag = mAppliedForce.magnitudeSafe();         break;      case CYLINDRICAL:         {            Point3F force_vec = mAppliedForce;            force_vec.z = 0.0;            force_mag = force_vec.magnitudeSafe();         }         break;      }   }}const Point3F& PhysicalZone::getForce(const Point3F* center) const {    static Point3F force_vec;   if (force_type == VECTOR)   {      if (orient_force)      {         getTransform().mulV(mAppliedForce, &force_vec);         force_vec *= fade_amt;         return force_vec;       }      force_vec = mAppliedForce;      force_vec *= fade_amt;      return force_vec;   }   if (!center)   {      force_vec.zero();      return force_vec;    }   if (force_type == SPHERICAL)   {      force_vec = *center - getPosition();      force_vec.normalizeSafe();      force_vec *= force_mag*fade_amt;      return force_vec;   }   if (orient_force)   {      force_vec = *center - getPosition();      getWorldTransform().mulV(force_vec);      force_vec.z = 0.0f;      force_vec.normalizeSafe();      force_vec *= force_mag;      force_vec.z = mAppliedForce.z;      getTransform().mulV(force_vec);      force_vec *= fade_amt;      return force_vec;   }   force_vec = *center - getPosition();   force_vec.z = 0.0f;   force_vec.normalizeSafe();   force_vec *= force_mag;   force_vec *= fade_amt;   return force_vec;}bool PhysicalZone::isExcludedObject(SceneObject* obj) const{   for (S32 i = 0; i < excluded_objects.size(); i++)      if (excluded_objects[i] == obj)         return true;   return false;}void PhysicalZone::registerExcludedObject(SceneObject* obj){   if (isExcludedObject(obj))      return;   excluded_objects.push_back(obj);   setMaskBits(FadeMask);}void PhysicalZone::unregisterExcludedObject(SceneObject* obj){   for (S32 i = 0; i < excluded_objects.size(); i++)      if (excluded_objects[i] == obj)      {         excluded_objects.erase(i);         setMaskBits(FadeMask);         return;      }}
 |