|
|
|
/*==LICENSE==*
|
|
|
|
|
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools
|
|
|
|
Copyright (C) 2011 Cyan Worlds, Inc.
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
|
|
|
or by snail mail at:
|
|
|
|
Cyan Worlds, Inc.
|
|
|
|
14617 N Newport Hwy
|
|
|
|
Mead, WA 99021
|
|
|
|
|
|
|
|
*==LICENSE==*/
|
|
|
|
#include "plPhysicalSDLModifier.h"
|
|
|
|
|
|
|
|
#include "hsGeometry3.h"
|
|
|
|
#include "plPhysical.h"
|
|
|
|
#include "plSDL/plSDL.h"
|
|
|
|
#include "pnSceneObject/plSceneObject.h"
|
|
|
|
#include "pnSceneObject/plSimulationInterface.h"
|
|
|
|
#include "pnNetCommon/plNetApp.h"
|
|
|
|
#include "hsQuat.h"
|
|
|
|
//#include "plHavok1/plSimulationMgr.h"
|
|
|
|
#include "plStatusLog/plStatusLog.h"
|
|
|
|
|
|
|
|
// static vars
|
|
|
|
static const char* kStrLinear = "linear";
|
|
|
|
static const char* kStrAngular = "angular";
|
|
|
|
static const char* kStrPosition = "position";
|
|
|
|
static const char* kStrOrientation = "orientation";
|
|
|
|
|
|
|
|
int plPhysicalSDLModifier::fLogLevel = 0;
|
|
|
|
|
|
|
|
static void IGetVars(plStateDataRecord::SimpleVarsList& vars,
|
|
|
|
hsPoint3& pos, bool& isPosSet,
|
|
|
|
hsQuat& rot, bool& isRotSet,
|
|
|
|
hsVector3& linV, bool& isLinVSet,
|
|
|
|
hsVector3& angV, bool& isAngVSet);
|
|
|
|
|
|
|
|
//
|
|
|
|
// get current state from physical
|
|
|
|
// fill out state data rec
|
|
|
|
//
|
|
|
|
void plPhysicalSDLModifier::IPutCurrentStateIn(plStateDataRecord* dstState)
|
|
|
|
{
|
|
|
|
plPhysical* phys = IGetPhysical();
|
|
|
|
|
|
|
|
// get latest state
|
|
|
|
hsPoint3 curPos;
|
|
|
|
hsQuat curOrientation;
|
|
|
|
hsVector3 curLinear, curAngular;
|
|
|
|
phys->GetSyncState(curPos, curOrientation, curLinear, curAngular);
|
|
|
|
|
|
|
|
// put it in sdl state record
|
|
|
|
dstState->FindVar(kStrPosition)->Set(&curPos.fX);
|
|
|
|
dstState->FindVar(kStrOrientation)->Set(&curOrientation.fX);
|
|
|
|
dstState->FindVar(kStrLinear)->Set(&curLinear.fX);
|
|
|
|
dstState->FindVar(kStrAngular)->Set(&curAngular.fX);
|
|
|
|
|
|
|
|
if (fLogLevel > 1)
|
|
|
|
ILogState(dstState, false, "PUT", plStatusLog::kWhite);
|
|
|
|
}
|
|
|
|
|
|
|
|
void plPhysicalSDLModifier::ISetCurrentStateFrom(const plStateDataRecord* srcState)
|
|
|
|
{
|
|
|
|
plPhysical* phys = IGetPhysical();
|
|
|
|
|
|
|
|
// FIXME PHYSX
|
|
|
|
|
|
|
|
// if(phys->GetBody()->isFixed())
|
|
|
|
// {
|
|
|
|
// plSimulationMgr::Log("Received synch for fixed body %s", phys->GetKey()->GetName());
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
// else if (phys->GetProperty(plSimulationInterface::kPinned))
|
|
|
|
// {
|
|
|
|
// // This is mainly intended for avatars. When pinning them (like in a multistage),
|
|
|
|
// // we don't want physical updates to sneak in due to network lag. If necessary,
|
|
|
|
// // this could be made a separate property on the physical, orthagonal to kPinned.
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
// else
|
|
|
|
{
|
|
|
|
hsPoint3 pos;
|
|
|
|
bool isPosSet;
|
|
|
|
hsQuat rot;
|
|
|
|
bool isRotSet;
|
|
|
|
hsVector3 linV;
|
|
|
|
bool isLinVSet;
|
|
|
|
hsVector3 angV;
|
|
|
|
bool isAngVSet;
|
|
|
|
|
|
|
|
plStateDataRecord::SimpleVarsList vars;
|
|
|
|
srcState->GetUsedVars(&vars);
|
|
|
|
IGetVars(vars, pos, isPosSet, rot, isRotSet, linV, isLinVSet, angV, isAngVSet);
|
|
|
|
|
|
|
|
if (fLogLevel > 0)
|
|
|
|
ILogState(srcState, false, "RCV", plStatusLog::kGreen);
|
|
|
|
|
|
|
|
phys->SetSyncState(
|
|
|
|
isPosSet ? &pos : nil,
|
|
|
|
isRotSet ? &rot : nil,
|
|
|
|
isLinVSet ? &linV : nil,
|
|
|
|
isAngVSet ? &angV : nil);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void plPhysicalSDLModifier::ISentState(const plStateDataRecord* sentState)
|
|
|
|
{
|
|
|
|
if (fLogLevel > 0)
|
|
|
|
{
|
|
|
|
ILogState(sentState, true, "SND", plStatusLog::kYellow);
|
|
|
|
|
|
|
|
// plPhysical* phys = IGetPhysical();
|
|
|
|
// if (!phys->GetBody()->isActive())
|
|
|
|
// IGetLog()->AddLineF("Phys %s sent state because it deactivated", phys->GetKeyName());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void IGetVars(plStateDataRecord::SimpleVarsList& vars,
|
|
|
|
hsPoint3& pos, bool& isPosSet,
|
|
|
|
hsQuat& rot, bool& isRotSet,
|
|
|
|
hsVector3& linV, bool& isLinVSet,
|
|
|
|
hsVector3& angV, bool& isAngVSet)
|
|
|
|
{
|
|
|
|
isPosSet = false;
|
|
|
|
isRotSet = false;
|
|
|
|
isLinVSet = false;
|
|
|
|
isAngVSet = false;
|
|
|
|
|
|
|
|
int num = vars.size();
|
|
|
|
for (int i = 0; i < num; i++)
|
|
|
|
{
|
|
|
|
if (vars[i]->IsNamed(kStrPosition))
|
|
|
|
{
|
|
|
|
vars[i]->Get(&pos.fX);
|
|
|
|
isPosSet= true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if (vars[i]->IsNamed(kStrOrientation))
|
|
|
|
{
|
|
|
|
vars[i]->Get(&rot.fX);
|
|
|
|
isRotSet = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if (vars[i]->IsNamed(kStrLinear))
|
|
|
|
{
|
|
|
|
vars[i]->Get(&linV.fX);
|
|
|
|
isLinVSet = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if (vars[i]->IsNamed(kStrAngular))
|
|
|
|
{
|
|
|
|
vars[i]->Get(&angV.fX);
|
|
|
|
isAngVSet = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if (vars[i]->IsNamed("subworld"))
|
|
|
|
{
|
|
|
|
// Unused
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
hsAssert(false, "Unknown var name");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void plPhysicalSDLModifier::ILogState(const plStateDataRecord* state, bool useDirty, const char* prefix, UInt32 color)
|
|
|
|
{
|
|
|
|
hsPoint3 pos;
|
|
|
|
bool isPosSet;
|
|
|
|
hsQuat rot;
|
|
|
|
bool isRotSet;
|
|
|
|
hsVector3 linV;
|
|
|
|
bool isLinVSet;
|
|
|
|
hsVector3 angV;
|
|
|
|
bool isAngVSet;
|
|
|
|
|
|
|
|
plStateDataRecord::SimpleVarsList vars;
|
|
|
|
if (useDirty)
|
|
|
|
state->GetDirtyVars(&vars);
|
|
|
|
else
|
|
|
|
state->GetUsedVars(&vars);
|
|
|
|
|
|
|
|
IGetVars(vars, pos, isPosSet, rot, isRotSet, linV, isLinVSet, angV, isAngVSet);
|
|
|
|
|
|
|
|
plPhysical* phys = IGetPhysical();
|
|
|
|
|
|
|
|
std::string log = xtl::format("%s: %s", phys->GetKeyName(), prefix);
|
|
|
|
|
|
|
|
if (isPosSet)
|
|
|
|
log += xtl::format(" Pos=%.1f %.1f %.1f", pos.fX, pos.fY, pos.fZ);
|
|
|
|
else
|
|
|
|
log += " Pos=None";
|
|
|
|
|
|
|
|
if (isLinVSet)
|
|
|
|
log += xtl::format(" LinV=%.1f %.1f %.1f", linV.fX, linV.fY, linV.fZ);
|
|
|
|
else
|
|
|
|
log += " LinV=None";
|
|
|
|
|
|
|
|
if (isAngVSet)
|
|
|
|
log += xtl::format(" AngV=%.1f %.1f %.1f", angV.fX, angV.fY, angV.fZ);
|
|
|
|
else
|
|
|
|
log += " AngV=None";
|
|
|
|
|
|
|
|
if (isRotSet)
|
|
|
|
log += xtl::format(" Rot=%.1f %.1f %.1f %.1f", rot.fX, rot.fY, rot.fZ, rot.fW);
|
|
|
|
else
|
|
|
|
log += " Rot=None";
|
|
|
|
|
|
|
|
IGetLog()->AddLine(log.c_str(), color);
|
|
|
|
}
|
|
|
|
|
|
|
|
plStatusLog* plPhysicalSDLModifier::IGetLog()
|
|
|
|
{
|
|
|
|
static plStatusLog* gLog = nil;
|
|
|
|
if (!gLog)
|
|
|
|
{
|
|
|
|
gLog = plStatusLogMgr::GetInstance().CreateStatusLog(20, "PhysicsSDL.log",
|
|
|
|
plStatusLog::kFilledBackground |
|
|
|
|
plStatusLog::kTimestamp |
|
|
|
|
plStatusLog::kDeleteForMe |
|
|
|
|
plStatusLog::kAlignToTop);
|
|
|
|
}
|
|
|
|
|
|
|
|
return gLog;
|
|
|
|
}
|
|
|
|
|
|
|
|
plPhysical* plPhysicalSDLModifier::IGetPhysical()
|
|
|
|
{
|
|
|
|
plPhysical* phys = nil;
|
|
|
|
|
|
|
|
plSceneObject* sobj = GetTarget();
|
|
|
|
if (sobj)
|
|
|
|
{
|
|
|
|
const plSimulationInterface* si = sobj->GetSimulationInterface();
|
|
|
|
if (si)
|
|
|
|
phys = si->GetPhysical();
|
|
|
|
}
|
|
|
|
|
|
|
|
hsAssert(phys, "nil hkPhysical");
|
|
|
|
return phys;
|
|
|
|
}
|