You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

301 lines
8.8 KiB

/*==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/>.
Additional permissions under GNU GPL version 3 section 7
If you modify this Program, or any covered work, by linking or
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
(or a modified version of those libraries),
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
licensors of this Program grant you additional
permission to convey the resulting work. Corresponding Source for a
non-source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work.
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==*/
#ifndef plCollisionDetector_inc
#define plCollisionDetector_inc
#include "plDetectorModifier.h"
#include "hsGeometry3.h"
#include <list>
#include <set>
class plMessage;
class plCameraMsg;
class plArmatureMod;
class plActivatorMsg;
class plEvalMsg;
#define USE_PHYSX_COLLISION_FLUTTER_WORKAROUND
class plCollisionDetector : public plDetectorModifier
{
protected:
Int8 fType;
hsBool fBumped, fTriggered;
plArmatureMod* IGetAvatarModifier(plKey key);
bool IIsDisabledAvatar(plKey key);
public:
enum
{
kTypeEnter = 0x01,
kTypeExit = 0x02,
kTypeAny = 0x04,
kTypeUnEnter = 0x08,
kTypeUnExit = 0x10,
kTypeBump = 0x20,
};
plCollisionDetector() : fType(0), fTriggered(false), fBumped(false){;}
virtual ~plCollisionDetector(){;}
virtual hsBool MsgReceive(plMessage* msg);
CLASSNAME_REGISTER( plCollisionDetector );
GETINTERFACE_ANY( plCollisionDetector, plDetectorModifier );
virtual void SetType(Int8 i) { fType |= i; }
void Read(hsStream* stream, hsResMgr* mgr);
void Write(hsStream* stream, hsResMgr* mgr);
};
// sub type for object-in-volume detectors
class plObjectInVolumeDetector : public plCollisionDetector
{
protected:
class plCollisionBookKeepingInfo
{
public:
plCollisionBookKeepingInfo(plKey& key, bool entering)
: fHitter(key), fEntering(entering) { }
plKey fHitter;
#ifdef USE_PHYSX_COLLISION_FLUTTER_WORKAROUND
UInt32 fLastStep;
#endif // USE_PHYSX_COLLISION_FLUTTER_WORKAROUND
bool fEntering;
};
virtual void ITrigger(plKey hitter, bool entering);
virtual void ISendTriggerMsg(plKey hitter, bool entering);
virtual void IRegisterForEval();
virtual void IHandleEval(plEvalMsg* pEval);
bool fWaitingForEval;
typedef std::list<plCollisionBookKeepingInfo*> bookKeepingList;
bookKeepingList fCollisionList;
typedef std::set<plKey> ResidentSet;
ResidentSet fCurrentResidents;
public:
plObjectInVolumeDetector()
: plCollisionDetector(), fWaitingForEval(false) { }
plObjectInVolumeDetector(Int8 i)
: plCollisionDetector(), fWaitingForEval(false) { fType = i; }
virtual ~plObjectInVolumeDetector() { }
virtual hsBool MsgReceive(plMessage* msg);
CLASSNAME_REGISTER(plObjectInVolumeDetector);
GETINTERFACE_ANY(plObjectInVolumeDetector, plCollisionDetector);
virtual void SetTarget(plSceneObject* so);
void Read(hsStream* stream, hsResMgr* mgr);
void Write(hsStream* stream, hsResMgr* mgr);
};
class plObjectInVolumeAndFacingDetector : public plObjectInVolumeDetector
{
protected:
hsScalar fFacingTolerance;
bool fNeedWalkingForward;
bool fAvatarInVolume;
bool fTriggered;
void ICheckForTrigger();
public:
plObjectInVolumeAndFacingDetector();
virtual ~plObjectInVolumeAndFacingDetector();
virtual hsBool MsgReceive(plMessage* msg);
CLASSNAME_REGISTER(plObjectInVolumeAndFacingDetector);
GETINTERFACE_ANY(plObjectInVolumeAndFacingDetector, plObjectInVolumeDetector);
void Read(hsStream* stream, hsResMgr* mgr);
void Write(hsStream* stream, hsResMgr* mgr);
// Export only
void SetFacingTolerance(int degrees);
void SetNeedWalkingForward(bool v) { fNeedWalkingForward = v; }
};
// sub-type for camera command regions
class plCameraRegionDetector : public plObjectInVolumeDetector
{
protected:
hsTArray<plCameraMsg*> fMessages;
#ifdef USE_PHYSX_COLLISION_FLUTTER_WORKAROUND
UInt32 fLastStep;
#endif // USE_PHYSX_COLLISION_FLUTTER_WORKAROUND
bool fIsInside;
bool fEntering;
virtual void ISendTriggerMsg();
virtual void IHandleEval(plEvalMsg* pEval);
public:
plCameraRegionDetector()
: plObjectInVolumeDetector(), fIsInside(false) { }
~plCameraRegionDetector();
virtual hsBool MsgReceive(plMessage* msg);
void AddMessage(plCameraMsg* pMsg) { fMessages.Append(pMsg); }
CLASSNAME_REGISTER( plCameraRegionDetector );
GETINTERFACE_ANY( plCameraRegionDetector, plCollisionDetector );
void Read(hsStream* stream, hsResMgr* mgr);
void Write(hsStream* stream, hsResMgr* mgr);
};
// sub-type for subworld regions
class plSubworldRegionDetector : public plCollisionDetector
{
protected:
plKey fSub;
hsBool fOnExit;
public:
enum
{
kSubworld = 0,
};
plSubworldRegionDetector() : fSub(nil), fOnExit(false){;}
~plSubworldRegionDetector();
virtual hsBool MsgReceive(plMessage* msg);
void SetSubworldKey(plKey pKey) { fSub = pKey; }
void SetTriggerOnExit(hsBool b) { fOnExit = b; }
CLASSNAME_REGISTER( plSubworldRegionDetector );
GETINTERFACE_ANY( plSubworldRegionDetector, plCollisionDetector );
void Read(hsStream* stream, hsResMgr* mgr);
void Write(hsStream* stream, hsResMgr* mgr);
};
// sub-type for panic link regions
class plPanicLinkRegion : public plCollisionDetector
{
public:
hsBool fPlayLinkOutAnim;
plPanicLinkRegion() : fPlayLinkOutAnim(true) {;}
~plPanicLinkRegion(){;}
virtual hsBool MsgReceive(plMessage* msg);
CLASSNAME_REGISTER( plPanicLinkRegion );
GETINTERFACE_ANY( plPanicLinkRegion, plCollisionDetector );
void Read(hsStream* stream, hsResMgr* mgr);
void Write(hsStream* stream, hsResMgr* mgr);
};
/** \Class plSimpleRegionSensor
A dead-simple interface for a collision region. Holds one message that it
sends to anyone who enters, and another message that it sends to anyone
who exits.
We may want to tie this into the plCollisionDetector so that it could be
integrated with responders.
*/
class plSimpleRegionSensor : public plSingleModifier
{
public:
plSimpleRegionSensor();
plSimpleRegionSensor(plMessage *enterMsg, plMessage *exitMsg);
virtual ~plSimpleRegionSensor();
virtual hsBool MsgReceive(plMessage *msg);
CLASSNAME_REGISTER( plSimpleRegionSensor );
GETINTERFACE_ANY( plSimpleRegionSensor, plSingleModifier);
virtual void Write(hsStream *stream, hsResMgr *mgr);
virtual void Read(hsStream *stream, hsResMgr *mgr);
virtual hsBool IEval(double secs, hsScalar del, UInt32 dirty);
protected:
plMessage *fEnterMsg;
plMessage *fExitMsg;
};
// This class really just exists so that I can hunt for it specifically by index
// (and not accidentally get some other SimpleRegionSensor).
class plSwimDetector : public plSimpleRegionSensor
{
public:
plSwimDetector() {}
plSwimDetector(plMessage *enterMsg, plMessage *exitMsg) : plSimpleRegionSensor(enterMsg, exitMsg) {}
virtual ~plSwimDetector() {}
CLASSNAME_REGISTER( plSwimDetector );
GETINTERFACE_ANY( plSwimDetector, plSimpleRegionSensor);
virtual void Write(hsStream *stream, hsResMgr *mgr);
virtual void Read(hsStream *stream, hsResMgr *mgr);
hsBool MsgReceive(plMessage *msg);
};
class plRidingAnimatedPhysicalDetector: public plSimpleRegionSensor
{
public:
plRidingAnimatedPhysicalDetector(){}
plRidingAnimatedPhysicalDetector(plMessage *enterMsg, plMessage *exitMsg) : plSimpleRegionSensor(enterMsg, exitMsg) {}
virtual ~plRidingAnimatedPhysicalDetector(){}
virtual hsBool MsgReceive(plMessage *msg);
CLASSNAME_REGISTER( plRidingAnimatedPhysicalDetector );
GETINTERFACE_ANY( plRidingAnimatedPhysicalDetector, plSimpleRegionSensor);
};
#endif plCollisionDetector_inc