| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508 | //-----------------------------------------------------------------------------// 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.//-----------------------------------------------------------------------------#include "platform/platform.h"#include "T3D/gameFunctions.h"#include "T3D/gameBase/gameConnection.h"#include "T3D/camera.h"#include "T3D/sfx/sfx3DWorld.h"#include "console/consoleTypes.h"#include "gui/3d/guiTSControl.h"#include "core/util/journal/process.h"#include "materials/materialManager.h"#include "math/mEase.h"#include "core/module.h"#include "console/engineAPI.h"#include "platform/output/IDisplayDevice.h"static void RegisterGameFunctions();static void Process3D();MODULE_BEGIN( 3D )   MODULE_INIT_AFTER( Process )   MODULE_INIT_AFTER( Scene )      MODULE_SHUTDOWN_BEFORE( Process )   MODULE_SHUTDOWN_BEFORE( Sim )   MODULE_SHUTDOWN_AFTER( Scene )      MODULE_INIT   {      Process::notify(Process3D, PROCESS_TIME_ORDER);      GameConnection::smFovUpdate.notify(GameSetCameraFov);      RegisterGameFunctions();   }      MODULE_SHUTDOWN   {      GameConnection::smFovUpdate.remove(GameSetCameraFov);      Process::remove(Process3D);   }MODULE_END;static S32 gEaseInOut = Ease::InOut;static S32 gEaseIn = Ease::In;static S32 gEaseOut = Ease::Out;static S32 gEaseLinear = Ease::Linear;static S32 gEaseQuadratic= Ease::Quadratic;static S32 gEaseCubic= Ease::Cubic;static S32 gEaseQuartic = Ease::Quartic;static S32 gEaseQuintic = Ease::Quintic;static S32 gEaseSinusoidal= Ease::Sinusoidal;static S32 gEaseExponential = Ease::Exponential;static S32 gEaseCircular = Ease::Circular;static S32 gEaseElastic = Ease::Elastic;static S32 gEaseBack = Ease::Back;static S32 gEaseBounce = Ease::Bounce;	extern bool gEditingMission;extern void ShowInit();//------------------------------------------------------------------------------/// Camera and FOV infonamespace {   const  U32 MaxZoomSpeed             = 2000;     ///< max number of ms to reach target FOV   static F32 sConsoleCameraFov        = 90.f;     ///< updated to camera FOV each frame   static F32 sDefaultFov              = 90.f;     ///< normal FOV   static F32 sCameraFov               = 90.f;     ///< current camera FOV   static F32 sTargetFov               = 90.f;     ///< the desired FOV   static F32 sLastCameraUpdateTime    = 0;        ///< last time camera was updated   static S32 sZoomSpeed               = 500;      ///< ms per 90deg fov change   /// A scale to apply to the normal visible distance   /// typically used for tuning performance.   static F32 sVisDistanceScale = 1.0f;} // namespace {}// querystatic SimpleQueryList sgServerQueryList;static U32 sgServerQueryIndex = 0;//SERVER FUNCTIONS ONLYConsoleFunctionGroupBegin( Containers, "Spatial query functions. <b>Server side only!</b>");ConsoleFunction(containerFindFirst, const char*, 6, 6, "(int mask, Point3F point, float x, float y, float z)"   "@brief Find objects matching the bitmask type within a box centered at point, with extents x, y, z.\n\n"   "@returns The first object found, or an empty string if nothing was found.  Thereafter, you can get more "   "results using containerFindNext()."   "@see containerFindNext\n"   "@ingroup Game"){   //find out what we're looking for   U32 typeMask = U32(dAtoi(argv[1]));   //find the center of the container volume   Point3F origin(0.0f, 0.0f, 0.0f);   dSscanf(argv[2], "%g %g %g", &origin.x, &origin.y, &origin.z);   //find the box dimensions   Point3F size(0.0f, 0.0f, 0.0f);   size.x = mFabs(dAtof(argv[3]));   size.y = mFabs(dAtof(argv[4]));   size.z = mFabs(dAtof(argv[5]));   //build the container volume   Box3F queryBox;   queryBox.minExtents = origin;   queryBox.maxExtents = origin;   queryBox.minExtents -= size;   queryBox.maxExtents += size;   //initialize the list, and do the query   sgServerQueryList.mList.clear();   gServerContainer.findObjects(queryBox, typeMask, SimpleQueryList::insertionCallback, &sgServerQueryList);   //return the first element   sgServerQueryIndex = 0;   static const U32 bufSize = 100;   char *buff = Con::getReturnBuffer(bufSize);   if (sgServerQueryList.mList.size())      dSprintf(buff, bufSize, "%d", sgServerQueryList.mList[sgServerQueryIndex++]->getId());   else      buff[0] = '\0';   return buff;}ConsoleFunction( containerFindNext, const char*, 1, 1, "()"   "@brief Get more results from a previous call to containerFindFirst().\n\n"   "@note You must call containerFindFirst() to begin the search.\n"   "@returns The next object found, or an empty string if nothing else was found.\n"   "@see containerFindFirst()\n"	"@ingroup Game"){   //return the next element   static const U32 bufSize = 100;   char *buff = Con::getReturnBuffer(bufSize);   if (sgServerQueryIndex < sgServerQueryList.mList.size())      dSprintf(buff, bufSize, "%d", sgServerQueryList.mList[sgServerQueryIndex++]->getId());   else      buff[0] = '\0';   return buff;}ConsoleFunctionGroupEnd( Containers );//------------------------------------------------------------------------------bool GameGetCameraTransform(MatrixF *mat, Point3F *velocity){   // Return the position and velocity of the control object   GameConnection* connection = GameConnection::getConnectionToServer();   return connection && connection->getControlCameraTransform(0, mat) &&      connection->getControlCameraVelocity(velocity);}//------------------------------------------------------------------------------DefineEngineFunction( setDefaultFov, void, ( F32 defaultFOV ),,				"@brief Set the default FOV for a camera.\n"            "@param defaultFOV The default field of view in degrees\n"				"@ingroup CameraSystem"){   sDefaultFov = mClampF(defaultFOV, MinCameraFov, MaxCameraFov);   if(sCameraFov == sTargetFov)      sTargetFov = sDefaultFov;}DefineEngineFunction( setZoomSpeed, void, ( S32 speed ),,				"@brief Set the zoom speed of the camera.\n"            "This affects how quickly the camera changes from one field of view "            "to another.\n"            "@param speed The camera's zoom speed in ms per 90deg FOV change\n"				"@ingroup CameraSystem"){   sZoomSpeed = mClamp(speed, 0, MaxZoomSpeed);}DefineEngineFunction( setFov, void, ( F32 FOV ),,				"@brief Set the FOV of the camera.\n"            "@param FOV The camera's new FOV in degrees\n"				"@ingroup CameraSystem"){   sTargetFov = mClampF(FOV, MinCameraFov, MaxCameraFov);}F32 GameGetCameraFov(){   return(sCameraFov);}void GameSetCameraFov(F32 fov){   sTargetFov = sCameraFov = fov;}void GameSetCameraTargetFov(F32 fov){   sTargetFov = fov;}void GameUpdateCameraFov(){   F32 time = F32(Platform::getVirtualMilliseconds());   // need to update fov?   if(sTargetFov != sCameraFov)   {      F32 delta = time - sLastCameraUpdateTime;      // snap zoom?      if((sZoomSpeed == 0) || (delta <= 0.0f))         sCameraFov = sTargetFov;      else      {         // gZoomSpeed is time in ms to zoom 90deg         F32 step = 90.f * (delta / F32(sZoomSpeed));         if(sCameraFov > sTargetFov)         {            sCameraFov -= step;            if(sCameraFov < sTargetFov)               sCameraFov = sTargetFov;         }         else         {            sCameraFov += step;            if(sCameraFov > sTargetFov)               sCameraFov = sTargetFov;         }      }   }   // the game connection controls the vertical and the horizontal   GameConnection * connection = GameConnection::getConnectionToServer();   if(connection)   {      // check if fov is valid on control object      if(connection->isValidControlCameraFov(sCameraFov))         connection->setControlCameraFov(sCameraFov);      else      {         // will set to the closest fov (fails only on invalid control object)         if(connection->setControlCameraFov(sCameraFov))         {            F32 setFov = sCameraFov;            connection->getControlCameraFov(&setFov);            sTargetFov = sCameraFov = setFov;         }      }   }   // update the console variable   sConsoleCameraFov = sCameraFov;   sLastCameraUpdateTime = time;}//--------------------------------------------------------------------------#ifdef TORQUE_DEBUG// ConsoleFunction(dumpTSShapes, void, 1, 1, "dumpTSShapes();")// {//    argc, argv;//    FindMatch match("*.dts", 4096);//    gResourceManager->findMatches(&match);//    for (U32 i = 0; i < match.numMatches(); i++)//    {//       U32 j;//       Resource<TSShape> shape = ResourceManager::get().load(match.matchList[i]);//       if (bool(shape) == false)//          Con::errorf(" aaa Couldn't load: %s", match.matchList[i]);//       U32 numMeshes = 0, numSkins = 0;//       for (j = 0; j < shape->meshes.size(); j++)//          if (shape->meshes[j])//             numMeshes++;//       for (j = 0; j < shape->skins.size(); j++)//          if (shape->skins[j])//             numSkins++;//      Con::printf(" aaa Shape: %s (%d meshes, %d skins)", match.matchList[i], numMeshes, numSkins);//       Con::printf(" aaa   Meshes");//       for (j = 0; j < shape->meshes.size(); j++)//       {//          if (shape->meshes[j])//             Con::printf(" aaa     %d -> nf: %d, nmf: %d, nvpf: %d (%d, %d, %d, %d, %d)",//                         shape->meshes[j]->meshType & TSMesh::TypeMask,//                         shape->meshes[j]->numFrames,//                         shape->meshes[j]->numMatFrames,//                         shape->meshes[j]->vertsPerFrame,//                         shape->meshes[j]->verts.size(),//                         shape->meshes[j]->norms.size(),//                         shape->meshes[j]->tverts.size(),//                         shape->meshes[j]->primitives.size(),//                         shape->meshes[j]->indices.size());//       }//       Con::printf(" aaa   Skins");//       for (j = 0; j < shape->skins.size(); j++)//       {//          if (shape->skins[j])//             Con::printf(" aaa     %d -> nf: %d, nmf: %d, nvpf: %d (%d, %d, %d, %d, %d)",//                         shape->skins[j]->meshType & TSMesh::TypeMask,//                         shape->skins[j]->numFrames,//                         shape->skins[j]->numMatFrames,//                         shape->skins[j]->vertsPerFrame,//                         shape->skins[j]->verts.size(),//                         shape->skins[j]->norms.size(),//                         shape->skins[j]->tverts.size(),//                         shape->skins[j]->primitives.size(),//                         shape->skins[j]->indices.size());//       }//    }// }#endifbool GameProcessCameraQuery(CameraQuery *query){   GameConnection* connection = GameConnection::getConnectionToServer();   if (connection && connection->getControlCameraTransform(0.032f, &query->cameraMatrix))   {      query->object = dynamic_cast<ShapeBase*>(connection->getControlObject());      query->nearPlane = gClientSceneGraph->getNearClip();      // Scale the normal visible distance by the performance       // tuning scale which we never let over 1.      sVisDistanceScale = mClampF( sVisDistanceScale, 0.01f, 1.0f );      query->farPlane = gClientSceneGraph->getVisibleDistance() * sVisDistanceScale;      // Provide some default values      query->projectionOffset = Point2F::Zero;      query->eyeOffset = Point3F::Zero;      F32 cameraFov = 0.0f;      bool fovSet = false;      // Try to use the connection's display deivce, if any, but only if the editor      // is not open      if(!gEditingMission && connection->hasDisplayDevice())      {         const IDisplayDevice* display = connection->getDisplayDevice();         // The connection's display device may want to set the FOV         if(display->providesYFOV())         {            cameraFov = mRadToDeg(display->getYFOV());            fovSet = true;         }         // The connection's display device may want to set the projection offset         if(display->providesProjectionOffset())         {            query->projectionOffset = display->getProjectionOffset();         }         // The connection's display device may want to set the eye offset         if(display->providesEyeOffset())         {            query->eyeOffset = display->getEyeOffset();         }      }      // Use the connection's FOV settings if requried      if(!fovSet && !connection->getControlCameraFov(&cameraFov))      {         return false;      }      query->fov = mDegToRad(cameraFov);      return true;   }   return false;}void GameRenderWorld(){   PROFILE_START(GameRenderWorld);   FrameAllocator::setWaterMark(0);   gClientSceneGraph->renderScene( SPT_Diffuse );   // renderScene leaves some states dirty, which causes problems if GameTSCtrl is the last Gui object rendered   GFX->updateStates();   AssertFatal(FrameAllocator::getWaterMark() == 0,      "Error, someone didn't reset the water mark on the frame allocator!");   FrameAllocator::setWaterMark(0);   PROFILE_END();}static void Process3D(){   MATMGR->updateTime();      // Update the SFX world, if there is one.      if( gSFX3DWorld )      gSFX3DWorld->update();}static void RegisterGameFunctions(){   Con::addVariable( "$pref::Camera::distanceScale", TypeF32, &sVisDistanceScale,       "A scale to apply to the normal visible distance, typically used for tuning performance.\n"	   "@ingroup Game");   Con::addVariable( "$cameraFov", TypeF32, &sConsoleCameraFov,       "The camera's Field of View.\n\n"	   "@ingroup Game" );   // Stuff game types into the console   Con::setIntVariable("$TypeMasks::StaticObjectType",         StaticObjectType);   Con::setIntVariable("$TypeMasks::EnvironmentObjectType",    EnvironmentObjectType);   Con::setIntVariable("$TypeMasks::TerrainObjectType",        TerrainObjectType);   Con::setIntVariable("$TypeMasks::WaterObjectType",          WaterObjectType);   Con::setIntVariable("$TypeMasks::TriggerObjectType",        TriggerObjectType);   Con::setIntVariable("$TypeMasks::MarkerObjectType",         MarkerObjectType);   Con::setIntVariable("$TypeMasks::GameBaseObjectType",       GameBaseObjectType);   Con::setIntVariable("$TypeMasks::ShapeBaseObjectType",      ShapeBaseObjectType);   Con::setIntVariable("$TypeMasks::CameraObjectType",         CameraObjectType);   Con::setIntVariable("$TypeMasks::StaticShapeObjectType",    StaticShapeObjectType);   Con::setIntVariable("$TypeMasks::DynamicShapeObjectType",   DynamicShapeObjectType);   Con::setIntVariable("$TypeMasks::PlayerObjectType",         PlayerObjectType);   Con::setIntVariable("$TypeMasks::ItemObjectType",           ItemObjectType);   Con::setIntVariable("$TypeMasks::VehicleObjectType",        VehicleObjectType);   Con::setIntVariable("$TypeMasks::VehicleBlockerObjectType", VehicleBlockerObjectType);   Con::setIntVariable("$TypeMasks::ProjectileObjectType",     ProjectileObjectType);   Con::setIntVariable("$TypeMasks::ExplosionObjectType",      ExplosionObjectType);   Con::setIntVariable("$TypeMasks::CorpseObjectType",         CorpseObjectType);   Con::setIntVariable("$TypeMasks::DebrisObjectType",         DebrisObjectType);   Con::setIntVariable("$TypeMasks::PhysicalZoneObjectType",   PhysicalZoneObjectType);   Con::setIntVariable("$TypeMasks::LightObjectType",          LightObjectType);   Con::addVariable("Ease::InOut", TypeS32, &gEaseInOut,       "InOut ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::In", TypeS32, &gEaseIn,       "In ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Out", TypeS32, &gEaseOut,       "Out ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Linear", TypeS32, &gEaseLinear,       "Linear ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Quadratic", TypeS32, &gEaseQuadratic,       "Quadratic ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Cubic", TypeS32, &gEaseCubic,       "Cubic ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Quartic", TypeS32, &gEaseQuartic,       "Quartic ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Quintic", TypeS32, &gEaseQuintic,       "Quintic ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Sinusoidal", TypeS32, &gEaseSinusoidal,       "Sinusoidal ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Exponential", TypeS32, &gEaseExponential,       "Exponential ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Circular", TypeS32, &gEaseCircular,       "Circular ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Elastic", TypeS32, &gEaseElastic,       "Elastic ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Back", TypeS32, &gEaseBack,       "Backwards ease for curve movement.\n"	   "@ingroup Game");   Con::addVariable("Ease::Bounce", TypeS32, &gEaseBounce,       "Bounce ease for curve movement.\n"	   "@ingroup Game");}
 |