|
|
|
/*==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==*/
|
|
|
|
#include "plAvCallbackAction.h" // havok-contaminated file: must go first
|
|
|
|
|
|
|
|
// singular
|
|
|
|
#include "plAvBrainGeneric.h"
|
|
|
|
|
|
|
|
// local
|
|
|
|
#include "plAnimStage.h"
|
|
|
|
#include "plArmatureMod.h"
|
|
|
|
// #include "plAvatarTasks.h"
|
|
|
|
#include "plAvTask.h"
|
|
|
|
#include "plAvTaskBrain.h"
|
|
|
|
#include "plAvBrainHuman.h"
|
|
|
|
#include "plAGAnimInstance.h"
|
|
|
|
#include "plMatrixChannel.h"
|
|
|
|
|
|
|
|
// global
|
|
|
|
#include "hsTimer.h"
|
|
|
|
#include "plgDispatch.h"
|
|
|
|
|
|
|
|
// other
|
|
|
|
#include "pnNetCommon/plSDLTypes.h"
|
|
|
|
#include "pnMessage/plCameraMsg.h"
|
|
|
|
#include "pnMessage/plNotifyMsg.h"
|
|
|
|
#include "plMessage/plAvatarMsg.h"
|
|
|
|
#include "plMessage/plInputEventMsg.h"
|
|
|
|
#include "plMessage/plSimStateMsg.h"
|
|
|
|
#include "plMessage/plConsoleMsg.h"
|
|
|
|
#include "plPipeline/plDebugText.h"
|
|
|
|
#include "plInputCore/plAvatarInputInterface.h"
|
|
|
|
#include "plMessage/plInputIfaceMgrMsg.h"
|
|
|
|
|
|
|
|
#ifdef DEBUG_MULTISTAGE
|
|
|
|
#include "plAvatarMgr.h"
|
|
|
|
#include "plStatusLog/plStatusLog.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
hsBool plAvBrainGeneric::fForce3rdPerson = true;
|
|
|
|
const float plAvBrainGeneric::kDefaultFadeIn = 6.f; // 1/6th of a second to fade in
|
|
|
|
const float plAvBrainGeneric::kDefaultFadeOut = 0.f; // instant fade out.
|
|
|
|
|
|
|
|
// plAvBrainGeneric ----------------
|
|
|
|
// -----------------
|
|
|
|
plAvBrainGeneric::plAvBrainGeneric()
|
|
|
|
: fRecipient(nil),
|
|
|
|
fStages(TRACKED_NEW plAnimStageVec),
|
|
|
|
fCurStage(0),
|
|
|
|
fType(kGeneric),
|
|
|
|
fExitFlags(kExitNormal),
|
|
|
|
fMode(kEntering),
|
|
|
|
fForward(true),
|
|
|
|
fStartMessage(nil),
|
|
|
|
fEndMessage(nil),
|
|
|
|
fFadeIn(0.0f),
|
|
|
|
fFadeOut(0.0f),
|
|
|
|
fMoveMode(kMoveRelative),
|
|
|
|
fCallbackAction(nil),
|
|
|
|
fBodyUsage(plAGAnim::kBodyUnknown)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// plAvBrainGeneric --------------------------------------
|
|
|
|
// -----------------
|
|
|
|
plAvBrainGeneric::plAvBrainGeneric(plAnimStageVec *stages,
|
|
|
|
plMessage *startMessage,
|
|
|
|
plMessage *endMessage,
|
|
|
|
plKey recipient,
|
|
|
|
uint32_t exitFlags,
|
|
|
|
float fadeIn,
|
|
|
|
float fadeOut,
|
|
|
|
MoveMode moveMode)
|
|
|
|
: plArmatureBrain(),
|
|
|
|
fRecipient(recipient),
|
|
|
|
fStages(stages),
|
|
|
|
fCurStage(0),
|
|
|
|
fType(kGeneric),
|
|
|
|
fExitFlags(exitFlags),
|
|
|
|
fMode(kEntering),
|
|
|
|
fForward(true),
|
|
|
|
fStartMessage(startMessage),
|
|
|
|
fEndMessage(endMessage),
|
|
|
|
fFadeIn(fadeIn),
|
|
|
|
fFadeOut(fadeOut),
|
|
|
|
fMoveMode(moveMode),
|
|
|
|
fCallbackAction(nil),
|
|
|
|
fBodyUsage(plAGAnim::kBodyUnknown)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// plAvBrainGeneric
|
|
|
|
plAvBrainGeneric::plAvBrainGeneric(uint32_t exitFlags, float fadeIn, float fadeOut, MoveMode moveMode)
|
|
|
|
: fRecipient(nil),
|
|
|
|
fStages(nil),
|
|
|
|
fCurStage(0),
|
|
|
|
fType(kGeneric),
|
|
|
|
fExitFlags(exitFlags),
|
|
|
|
fMode(kEntering),
|
|
|
|
fForward(true),
|
|
|
|
fStartMessage(nil),
|
|
|
|
fEndMessage(nil),
|
|
|
|
fFadeIn(fadeIn),
|
|
|
|
fFadeOut(fadeOut),
|
|
|
|
fMoveMode(moveMode),
|
|
|
|
fCallbackAction(nil),
|
|
|
|
fBodyUsage(plAGAnim::kBodyUnknown)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ~plAvBrainGeneric ----------------
|
|
|
|
// ------------------
|
|
|
|
plAvBrainGeneric::~plAvBrainGeneric()
|
|
|
|
{
|
|
|
|
int fNumStages = fStages->size();
|
|
|
|
|
|
|
|
for(int i = 0; i < fNumStages; i++)
|
|
|
|
{
|
|
|
|
plAnimStage *stage = (*fStages)[i];
|
|
|
|
(*fStages)[i] = nil;
|
|
|
|
stage->Detach(fAvMod);
|
|
|
|
delete stage;
|
|
|
|
}
|
|
|
|
delete fStages;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Activate -------------------------------------------
|
|
|
|
// ---------
|
|
|
|
void plAvBrainGeneric::Activate(plArmatureModBase *avMod)
|
|
|
|
{
|
|
|
|
plArmatureBrain::Activate(avMod);
|
|
|
|
|
|
|
|
if ((GetType() == kEmote || GetType() == kAFK || GetType() == kSitOnGround) && fAvMod->IsLocalAvatar())
|
|
|
|
{
|
|
|
|
plInputIfaceMgrMsg* msg = TRACKED_NEW plInputIfaceMgrMsg(plInputIfaceMgrMsg::kDisableClickables );
|
|
|
|
plgDispatch::MsgSend(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
int numStages = fStages->size();
|
|
|
|
if (!numStages)
|
|
|
|
return;
|
|
|
|
plAnimStage *stage = (*fStages)[fCurStage];
|
|
|
|
|
|
|
|
bool useFadeIn = fFadeIn > 0.0f;
|
|
|
|
float initialBlend = useFadeIn ? 0.0f : 1.0f;
|
|
|
|
|
|
|
|
if (GetType() == kEmote)
|
|
|
|
((plArmatureMod*)avMod)->SendBehaviorNotify(plHBehavior::kBehaviorTypeEmote,true);
|
|
|
|
double worldTime = hsTimer::GetSysSeconds();
|
|
|
|
|
|
|
|
if (fMoveMode == kMoveRelative || fMoveMode == kMoveAbsolute)
|
|
|
|
{
|
|
|
|
// enable kinematic... ignore outside forces... but still collide with detector regions
|
|
|
|
fAvMod->EnablePhysicsKinematic( true );
|
|
|
|
}
|
|
|
|
else if(fMoveMode == kMoveStandstill)
|
|
|
|
{
|
|
|
|
// Avatar stands still automatically now, so we do nothing here
|
|
|
|
}
|
|
|
|
if (stage->Attach(fAvMod, this, initialBlend, worldTime))
|
|
|
|
{
|
|
|
|
if(fStartMessage)
|
|
|
|
{
|
|
|
|
fStartMessage->Send();
|
|
|
|
fStartMessage = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (plAvBrainGeneric::fForce3rdPerson && fAvMod->IsLocalAvatar())
|
|
|
|
{
|
|
|
|
// create message to force 3rd person mode
|
|
|
|
plCameraMsg* pMsg = TRACKED_NEW plCameraMsg;
|
|
|
|
pMsg->SetBCastFlag(plMessage::kBCastByExactType);
|
|
|
|
pMsg->SetCmd(plCameraMsg::kResponderSetThirdPerson);
|
|
|
|
pMsg->SetBCastFlag(plMessage::kNetPropagate, false);
|
|
|
|
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fType == kLadder && fAvMod->IsLocalAvatar())
|
|
|
|
plAvatarInputInterface::GetInstance()->SetLadderMode();
|
|
|
|
|
|
|
|
if (fReverseFBControlsOnRelease)
|
|
|
|
fAvMod->SetReverseFBOnIdle(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plAvBrainGeneric::IsRunningTask()
|
|
|
|
{
|
|
|
|
if ( fStages->size() > 0 )
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool plAvBrainGeneric::MatchAnimNames(const char *names[], int count)
|
|
|
|
{
|
|
|
|
if (count != GetStageCount())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
if (strcmp(names[i], GetStage(i)->GetAnimName()))
|
|
|
|
return false; // different names.
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply ----------------------------------------------------
|
|
|
|
// ------
|
|
|
|
hsBool plAvBrainGeneric::Apply(double time, float elapsed)
|
|
|
|
{
|
|
|
|
hsBool result = false;
|
|
|
|
|
|
|
|
switch(fMode)
|
|
|
|
{
|
|
|
|
case kAbort:
|
|
|
|
break;
|
|
|
|
case kEntering:
|
|
|
|
case kFadingIn:
|
|
|
|
result = IProcessFadeIn(time, elapsed);
|
|
|
|
break;
|
|
|
|
case kExit:
|
|
|
|
case kFadingOut:
|
|
|
|
// go through the fade logic whether or not we actually need a fade;
|
|
|
|
// centralizes some exit conditions.
|
|
|
|
result = IProcessFadeOut(time, elapsed);
|
|
|
|
break;
|
|
|
|
case kNormal:
|
|
|
|
result = IProcessNormal(time, elapsed);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
plArmatureBrain::Apply(time, elapsed);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deactivate -----------------------
|
|
|
|
// -----------
|
|
|
|
void plAvBrainGeneric::Deactivate()
|
|
|
|
{
|
|
|
|
if (fEndMessage)
|
|
|
|
{
|
|
|
|
fEndMessage->Send();
|
|
|
|
fEndMessage = nil;
|
|
|
|
}
|
|
|
|
if (fMode != kAbort) // we're being forcibly removed...
|
|
|
|
IExitMoveMode();
|
|
|
|
|
|
|
|
if (fMoveMode == kMoveRelative || fMoveMode == kMoveAbsolute)
|
|
|
|
{
|
|
|
|
// re-enable normal physics... outside forces affect us
|
|
|
|
fAvMod->EnablePhysicsKinematic( false );
|
|
|
|
}
|
|
|
|
else if (fMoveMode == kMoveStandstill)
|
|
|
|
{
|
|
|
|
// Avatar stands still automaticaly now, so we do nothing here
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fType == plAvBrainGeneric::kLadder && fAvMod->IsLocalAvatar())
|
|
|
|
{
|
|
|
|
plAvatarInputInterface::GetInstance()->ClearLadderMode();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fReverseFBControlsOnRelease)
|
|
|
|
fAvMod->SetReverseFBOnIdle(false);
|
|
|
|
|
|
|
|
if (plAvBrainGeneric::fForce3rdPerson && fAvMod->IsLocalAvatar())
|
|
|
|
{
|
|
|
|
// create message to force 3rd person mode
|
|
|
|
plCameraMsg* pMsg = TRACKED_NEW plCameraMsg;
|
|
|
|
pMsg->SetBCastFlag(plMessage::kBCastByExactType);
|
|
|
|
pMsg->SetBCastFlag(plMessage::kNetPropagate, false);
|
|
|
|
pMsg->SetCmd(plCameraMsg::kResponderUndoThirdPerson);
|
|
|
|
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
|
|
|
|
}
|
|
|
|
|
|
|
|
plArmatureBrain::Deactivate();
|
|
|
|
|
|
|
|
if ((GetType() == kEmote || GetType() == kAFK || GetType() == kSitOnGround) && fAvMod->IsLocalAvatar())
|
|
|
|
{
|
|
|
|
plInputIfaceMgrMsg* msg = TRACKED_NEW plInputIfaceMgrMsg(plInputIfaceMgrMsg::kEnableClickables );
|
|
|
|
plgDispatch::MsgSend(msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GETRECIPIENT
|
|
|
|
plKey plAvBrainGeneric::GetRecipient()
|
|
|
|
{
|
|
|
|
return fRecipient;
|
|
|
|
}
|
|
|
|
|
|
|
|
// SETRECIPIENT
|
|
|
|
void plAvBrainGeneric::SetRecipient(const plKey &recipient)
|
|
|
|
{
|
|
|
|
fRecipient = recipient;
|
|
|
|
}
|
|
|
|
|
|
|
|
// RELAYNOTIFYMSG
|
|
|
|
bool plAvBrainGeneric::RelayNotifyMsg(plNotifyMsg *msg)
|
|
|
|
{
|
|
|
|
if(fRecipient)
|
|
|
|
{
|
|
|
|
msg->AddReceiver(fRecipient);
|
|
|
|
msg->Send();
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IGetAnimDelta ------------------------------------------------
|
|
|
|
// --------------
|
|
|
|
float plAvBrainGeneric::IGetAnimDelta(double time, float elapsed)
|
|
|
|
{
|
|
|
|
float delta = 0.0f;
|
|
|
|
plAnimStage *curStage = (*fStages)[fCurStage];
|
|
|
|
plAnimStage::ForwardType forward = curStage->GetForwardType();
|
|
|
|
plAnimStage::BackType back = curStage->GetBackType();
|
|
|
|
bool fwdIsDown = (fReverseFBControlsOnRelease && fAvMod->IsFBReversed()) ? fAvMod->BackwardKeyDown() : fAvMod->ForwardKeyDown();
|
|
|
|
hsBool backIsDown = (fReverseFBControlsOnRelease && fAvMod->IsFBReversed()) ? fAvMod->ForwardKeyDown() : fAvMod->BackwardKeyDown();
|
|
|
|
|
|
|
|
// forward with a key down gets top priority
|
|
|
|
if(forward == plAnimStage::kForwardKey && fwdIsDown)
|
|
|
|
{
|
|
|
|
// key drive forward, forward key is down
|
|
|
|
delta = elapsed;
|
|
|
|
fForward = true;
|
|
|
|
} else if(back == plAnimStage::kBackKey && backIsDown)
|
|
|
|
{
|
|
|
|
// key drive back, back key is down
|
|
|
|
delta = -elapsed;
|
|
|
|
fForward = false;
|
|
|
|
} else if (forward == plAnimStage::kForwardAuto && fForward)
|
|
|
|
{
|
|
|
|
// auto drive forward
|
|
|
|
delta = elapsed;
|
|
|
|
} else if (back == plAnimStage::kBackAuto && ! fForward)
|
|
|
|
{
|
|
|
|
// auto drive backward
|
|
|
|
delta = -elapsed;
|
|
|
|
}
|
|
|
|
return delta;
|
|
|
|
}
|
|
|
|
|
|
|
|
// IProcessNormal -------------------------------------------------
|
|
|
|
// ---------------
|
|
|
|
hsBool plAvBrainGeneric::IProcessNormal(double time, float elapsed)
|
|
|
|
{
|
|
|
|
plAnimStage *curStage = (*fStages)[fCurStage];
|
|
|
|
if(curStage)
|
|
|
|
{
|
|
|
|
float animDelta = IGetAnimDelta(time, elapsed); // how far to move the anim (may be negative)
|
|
|
|
float overage;
|
|
|
|
hsBool done = curStage->MoveRelative(time, animDelta, overage, fAvMod);
|
|
|
|
|
|
|
|
if(done)
|
|
|
|
{
|
|
|
|
bool forward = animDelta > 0.0f;
|
|
|
|
int nextStage = forward ? curStage->GetNextStage(fCurStage) : curStage->GetPrevStage(fCurStage);
|
|
|
|
|
|
|
|
if((nextStage == -1) || (nextStage >= fStages->size()))
|
|
|
|
{
|
|
|
|
// ran off one end; we're done.
|
|
|
|
fMode = kExit;
|
|
|
|
} else {
|
|
|
|
ISwitchStages(fCurStage, nextStage, overage, false, 0.0f, 1.0f, -1.0f, time);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
// current stage is missing; abort
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IProcessFadeIn -------------------------------------------------
|
|
|
|
// ---------------
|
|
|
|
hsBool plAvBrainGeneric::IProcessFadeIn(double time, float elapsed)
|
|
|
|
{
|
|
|
|
plAnimStage *curStage = (*fStages)[fCurStage];
|
|
|
|
|
|
|
|
if(fMode != kFadingIn)
|
|
|
|
{
|
|
|
|
bool needFade = fFadeIn != 0.0f;
|
|
|
|
|
|
|
|
if(fFadeIn == 0.0f)
|
|
|
|
{
|
|
|
|
IEnterMoveMode(time); // if fadeIn's not zero, we have to wait until fade's done
|
|
|
|
// before animating
|
|
|
|
} else {
|
|
|
|
plAGAnimInstance *curAnim = curStage->GetAnimInstance();
|
|
|
|
if(curAnim)
|
|
|
|
curAnim->Fade(1.0f, fFadeIn);
|
|
|
|
}
|
|
|
|
fMode = kFadingIn;
|
|
|
|
} else {
|
|
|
|
plAGAnimInstance *curAnim = curStage->GetAnimInstance();
|
|
|
|
float curBlend = 1.0f;
|
|
|
|
if(curAnim)
|
|
|
|
curBlend = curAnim->GetBlend();
|
|
|
|
if(curBlend == 1.0f)
|
|
|
|
{
|
|
|
|
IEnterMoveMode(time);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// IProcessFadeOut -------------------------------------------------
|
|
|
|
// ----------------
|
|
|
|
hsBool plAvBrainGeneric::IProcessFadeOut(double time, float elapsed)
|
|
|
|
{
|
|
|
|
plAnimStage *curStage = (*fStages)[fCurStage];
|
|
|
|
|
|
|
|
if(fMode != kFadingOut)
|
|
|
|
{
|
|
|
|
// haven't actually started fading; see if we need to
|
|
|
|
if(fFadeOut > 0.0f)
|
|
|
|
{
|
|
|
|
plAGAnimInstance *curAnim = curStage->GetAnimInstance();
|
|
|
|
if(curAnim)
|
|
|
|
{
|
|
|
|
curAnim->Fade(0.0f, fFadeOut);
|
|
|
|
IExitMoveMode();
|
|
|
|
fMode = kFadingOut;
|
|
|
|
} else {
|
|
|
|
fMode = kAbort;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
curStage->Detach(fAvMod);
|
|
|
|
IExitMoveMode();
|
|
|
|
fMode = kAbort;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// already fading; just keeping looking for the anim to zero out.
|
|
|
|
plAGAnimInstance *curAnim = curStage->GetAnimInstance();
|
|
|
|
float curBlend = 0.0f;
|
|
|
|
if(curAnim)
|
|
|
|
curBlend = curAnim->GetBlend();
|
|
|
|
if(curBlend == 0.0f)
|
|
|
|
{
|
|
|
|
curStage->Detach(fAvMod);
|
|
|
|
fMode = kAbort;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ISwitchStages ---------------------------------------------------------------------------------------------------
|
|
|
|
// --------------
|
|
|
|
hsBool plAvBrainGeneric::ISwitchStages(int oldStageNum, int newStageNum, float delta, hsBool setTime, float newTime,
|
|
|
|
float fadeNew, float fadeOld, double worldTime)
|
|
|
|
{
|
|
|
|
#ifdef DEBUG_MULTISTAGE
|
|
|
|
char sbuf[256];
|
|
|
|
sprintf(sbuf,"ISwitchStage - old=%d new=%d (fCurStage=%d)",oldStageNum,newStageNum,fCurStage);
|
|
|
|
plAvatarMgr::GetInstance()->GetLog()->AddLine(sbuf);
|
|
|
|
#endif
|
|
|
|
if(oldStageNum != newStageNum) {
|
|
|
|
plAnimStage *newStage = fStages->at(newStageNum);
|
|
|
|
plAnimStage *oldStage = fStages->at(oldStageNum);
|
|
|
|
if(setTime)
|
|
|
|
newStage->SetLocalTime(newTime);
|
|
|
|
|
|
|
|
hsAssert(oldStageNum < fStages->size(), "PLAVBRAINGENERIC: Stage out of range.");
|
|
|
|
|
|
|
|
oldStage->Detach(fAvMod);
|
|
|
|
newStage->Attach(fAvMod, this, 1.0f, worldTime);
|
|
|
|
|
|
|
|
fCurStage = newStageNum;
|
|
|
|
fAvMod->DirtySynchState(kSDLAvatar, 0); // write our new stage to the server
|
|
|
|
}
|
|
|
|
if(setTime) {
|
|
|
|
plAnimStage *curStage = fStages->at(fCurStage);
|
|
|
|
curStage->SetLocalTime(newTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(fMoveMode == kMoveRelative)
|
|
|
|
fAvMod->GetRootAnimator()->Reset(worldTime);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plAvBrainGeneric::IEnterMoveMode(double time)
|
|
|
|
{
|
|
|
|
if(fMoveMode == kMoveRelative)
|
|
|
|
{
|
|
|
|
fAvMod->GetRootAnimator()->Enable(true);
|
|
|
|
fAvMod->GetRootAnimator()->Reset(time);
|
|
|
|
}
|
|
|
|
fMode = kNormal;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plAvBrainGeneric::IExitMoveMode()
|
|
|
|
{
|
|
|
|
if(fAvMod)
|
|
|
|
{
|
|
|
|
if(fMoveMode == kMoveRelative)
|
|
|
|
{
|
|
|
|
if(fAvMod->GetRootAnimator())
|
|
|
|
fAvMod->GetRootAnimator()->Enable(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fFadeOut == 0.f)
|
|
|
|
{
|
|
|
|
// if we're exiting instantly (no fade out) then the end of the animation expects to line up with
|
|
|
|
// the first frame of the idle animation, so we need to reset it.
|
|
|
|
plAvBrainHuman *brain = plAvBrainHuman::ConvertNoRef(fAvMod->FindBrainByClass(plAvBrainHuman::Index()));
|
|
|
|
if (brain)
|
|
|
|
brain->ResetIdle();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// MESSAGE HANDLING
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// MsgReceive -------------------------------------
|
|
|
|
// -----------
|
|
|
|
hsBool plAvBrainGeneric::MsgReceive(plMessage *msg)
|
|
|
|
{
|
|
|
|
hsBool result = false;
|
|
|
|
|
|
|
|
plAvBrainGenericMsg *genMsg = plAvBrainGenericMsg::ConvertNoRef(msg);
|
|
|
|
//plAvExitModeMsg *exitMsg = plAvExitModeMsg::ConvertNoRef(msg);
|
|
|
|
plAvTaskMsg *taskMsg = plAvTaskMsg::ConvertNoRef(msg);
|
|
|
|
plControlEventMsg *ctrlMsg = plControlEventMsg::ConvertNoRef(msg);
|
|
|
|
|
|
|
|
|
|
|
|
if(genMsg)
|
|
|
|
{
|
|
|
|
result = IHandleGenBrainMsg(genMsg);
|
|
|
|
}
|
|
|
|
// else if(exitMsg) {
|
|
|
|
// fMode = kExit;
|
|
|
|
// result = true;
|
|
|
|
// }
|
|
|
|
else if (taskMsg) {
|
|
|
|
result = IHandleTaskMsg(taskMsg);
|
|
|
|
}
|
|
|
|
else if (ctrlMsg && (fExitFlags & kExitAnyInput) ) {
|
|
|
|
fMode = kExit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(result == false) // if still haven't handled msg
|
|
|
|
{
|
|
|
|
if(fMode == kExit) // if we're exiting
|
|
|
|
{
|
|
|
|
result = fAvMod->GetNextBrain(this)->MsgReceive(msg); // pass msg to next brain
|
|
|
|
} else { // otherwise
|
|
|
|
result = plArmatureBrain::MsgReceive(msg); // pass msg to base class
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// IHandleGenBrainMsg -----------------------------------------------------
|
|
|
|
// -------------------
|
|
|
|
hsBool plAvBrainGeneric::IHandleGenBrainMsg(const plAvBrainGenericMsg *msg)
|
|
|
|
{
|
|
|
|
hsBool setTime = msg->fSetTime;
|
|
|
|
float newTime = msg->fNewTime;
|
|
|
|
hsBool setDirection = msg->fSetDirection;
|
|
|
|
bool newDirection = msg->fNewDirection ? true : false;
|
|
|
|
double worldTime = hsTimer::GetSysSeconds();
|
|
|
|
|
|
|
|
switch(msg->fType)
|
|
|
|
{
|
|
|
|
case plAvBrainGenericMsg::kGotoStage:
|
|
|
|
{
|
|
|
|
int wantStage = msg->fWhichStage;
|
|
|
|
#ifdef DEBUG_MULTISTAGE
|
|
|
|
char sbuf[256];
|
|
|
|
sprintf(sbuf,"GenericMsg - Goto Stage %d (oldstage %d)",wantStage,fCurStage);
|
|
|
|
plAvatarMgr::GetInstance()->GetLog()->AddLine(sbuf);
|
|
|
|
#endif
|
|
|
|
if(wantStage == -1) {
|
|
|
|
fMode = kExit;
|
|
|
|
} else {
|
|
|
|
int count = fStages->size();
|
|
|
|
if(wantStage < count && wantStage >= 0)
|
|
|
|
{
|
|
|
|
ISwitchStages(fCurStage, wantStage, 0.0f, setTime, newTime, 1.0f, -1.0f, worldTime);
|
|
|
|
// direction is set within the brain, not the stage
|
|
|
|
if(setDirection)
|
|
|
|
fForward = newDirection;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case plAvBrainGenericMsg::kNextStage:
|
|
|
|
{
|
|
|
|
int wantStage = fCurStage + 1;
|
|
|
|
#ifdef DEBUG_MULTISTAGE
|
|
|
|
char sbuf[256];
|
|
|
|
sprintf(sbuf,"GenericMsg - Next Stage %d (oldstage %d)",wantStage,fCurStage);
|
|
|
|
plAvatarMgr::GetInstance()->GetLog()->AddLine(sbuf);
|
|
|
|
#endif
|
|
|
|
if(wantStage == fStages->size())
|
|
|
|
{
|
|
|
|
fMode = kExit; // walked off the end of the brain
|
|
|
|
} else {
|
|
|
|
ISwitchStages(fCurStage, wantStage, 0.0f, setTime, newTime, 1.0f, -1.0f, worldTime);
|
|
|
|
if(setDirection)
|
|
|
|
fForward = newDirection;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case plAvBrainGenericMsg::kPrevStage:
|
|
|
|
{
|
|
|
|
int wantStage = fCurStage - 1;
|
|
|
|
#ifdef DEBUG_MULTISTAGE
|
|
|
|
char sbuf[256];
|
|
|
|
sprintf(sbuf,"GenericMsg - PrevStage %d (oldstage %d)",wantStage,fCurStage);
|
|
|
|
plAvatarMgr::GetInstance()->GetLog()->AddLine(sbuf);
|
|
|
|
#endif
|
|
|
|
if(wantStage < 0)
|
|
|
|
{
|
|
|
|
fMode = kExit; // walked off the beginning of the brain
|
|
|
|
} else {
|
|
|
|
ISwitchStages(fCurStage, wantStage, 0.0f, setTime, 0.0f, 1.0f, -1.0f, worldTime);
|
|
|
|
if(setDirection)
|
|
|
|
fForward = newDirection;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#ifdef DEBUG_MULTISTAGE
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
char sbuf[256];
|
|
|
|
sprintf(sbuf,"GenericMsg - Unknown command %d ",msg->fType);
|
|
|
|
plAvatarMgr::GetInstance()->GetLog()->AddLine(sbuf);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plAvBrainGeneric::IHandleTaskMsg(plAvTaskMsg *msg)
|
|
|
|
{
|
|
|
|
plAvTask *task = msg->GetTask();
|
|
|
|
plAvTaskBrain *brainTask = plAvTaskBrain::ConvertNoRef(task);
|
|
|
|
|
|
|
|
if(brainTask)
|
|
|
|
{
|
|
|
|
plArmatureBrain * brain = brainTask->GetBrain();
|
|
|
|
|
|
|
|
if(brain)
|
|
|
|
{
|
|
|
|
if(fExitFlags & kExitNewBrain)
|
|
|
|
{
|
|
|
|
// RULE 1: if kExitNewBrain, exit on any new brain
|
|
|
|
fMode = kExit;
|
|
|
|
return false; // we didn't consume the message
|
|
|
|
} else {
|
|
|
|
plAvBrainGeneric * gBrain = plAvBrainGeneric::ConvertNoRef(brain);
|
|
|
|
|
|
|
|
if(gBrain && IBrainIsCompatible(gBrain))
|
|
|
|
{
|
|
|
|
// RULE 2: if not kExitNewBrain and brain is compatible, apply it
|
|
|
|
QueueTask(brainTask);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
if(fMode == kExit || fMode == kFadingOut)
|
|
|
|
{
|
|
|
|
// RULE 3: if brain is incompatible and we're exiting anyway,
|
|
|
|
// queue it to be next
|
|
|
|
fAvMod->GetNextBrain(this)->QueueTask(brainTask);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// RULE 4: if brain is incompatible and we're still running, ignore.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// no brain; it's an exit task, exit and CONSUME it.
|
|
|
|
fMode = kExit;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// note that this check has to come after the brain task check; if it's a brain
|
|
|
|
// task we need to examine it so we can say whether we consumed it or not.
|
|
|
|
// popbrain messages get consumed, even if we exit on any task.
|
|
|
|
if(fExitFlags & kExitAnyTask)
|
|
|
|
{
|
|
|
|
// RULE 4: if kExitAnyTask, exit on any task (but if it was an exit brain task,
|
|
|
|
// make sure to consume it )
|
|
|
|
fMode = kExit;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// RULE 4: if brain is incompatible and we're still running, ignore.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool plAvBrainGeneric::IBrainIsCompatible(plAvBrainGeneric *otherBrain)
|
|
|
|
{
|
|
|
|
plAGAnim::BodyUsage otherUsage = otherBrain->GetBodyUsage();
|
|
|
|
|
|
|
|
switch(fBodyUsage)
|
|
|
|
{
|
|
|
|
case plAGAnim::kBodyUnknown:
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
case plAGAnim::kBodyFull:
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
case plAGAnim::kBodyUpper:
|
|
|
|
if(otherUsage == plAGAnim::kBodyLower)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
case plAGAnim::kBodyLower:
|
|
|
|
if(otherUsage == plAGAnim::kBodyUpper)
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// READ/WRITE
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Write ----------------------------------------------------
|
|
|
|
// ------
|
|
|
|
void plAvBrainGeneric::Write(hsStream *stream, hsResMgr *mgr)
|
|
|
|
{
|
|
|
|
plArmatureBrain::Write(stream, mgr);
|
|
|
|
int numStages = fStages->size();
|
|
|
|
stream->WriteLE32(numStages);
|
|
|
|
|
|
|
|
for(int i = 0; i < numStages; i++)
|
|
|
|
{
|
|
|
|
plAnimStage *stage = (*fStages)[i];
|
|
|
|
plCreatable *cre = reinterpret_cast<plCreatable *>(stage);
|
|
|
|
mgr->WriteCreatable(stream, cre); // save base state
|
|
|
|
// ** replace this with Write(..)
|
|
|
|
stage->SaveAux(stream, mgr); // save ephemeral state
|
|
|
|
}
|
|
|
|
|
|
|
|
stream->WriteLE32(fCurStage);
|
|
|
|
stream->WriteLE32(fType);
|
|
|
|
stream->WriteLE32(fExitFlags);
|
|
|
|
stream->WriteByte(fMode);
|
|
|
|
stream->Writebool(fForward);
|
|
|
|
|
|
|
|
if(fStartMessage) {
|
|
|
|
stream->WriteBool(true);
|
|
|
|
mgr->WriteCreatable(stream, fStartMessage);
|
|
|
|
} else {
|
|
|
|
stream->WriteBool(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(fEndMessage) {
|
|
|
|
stream->WriteBool(true);
|
|
|
|
mgr->WriteCreatable(stream, fEndMessage);
|
|
|
|
} else {
|
|
|
|
stream->WriteBool(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
stream->WriteLEScalar(fFadeIn);
|
|
|
|
stream->WriteLEScalar(fFadeOut);
|
|
|
|
stream->WriteByte(fMoveMode);
|
|
|
|
stream->WriteByte(fBodyUsage);
|
|
|
|
mgr->WriteKey(stream, fRecipient);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read ----------------------------------------------------
|
|
|
|
// -----
|
|
|
|
void plAvBrainGeneric::Read(hsStream *stream, hsResMgr *mgr)
|
|
|
|
{
|
|
|
|
plArmatureBrain::Read(stream, mgr);
|
|
|
|
int numStages = stream->ReadLE32();
|
|
|
|
|
|
|
|
for(int i = 0; i < numStages; i++)
|
|
|
|
{
|
|
|
|
plCreatable *created = mgr->ReadCreatable(stream); // load base state
|
|
|
|
plAnimStage *stage = reinterpret_cast<plAnimStage *>(created);
|
|
|
|
// Replace this with Read(..)
|
|
|
|
stage->LoadAux(stream, mgr, 0.0); // load ephemeral state
|
|
|
|
|
|
|
|
fStages->push_back(stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
fCurStage = stream->ReadLE32();
|
|
|
|
fType = static_cast<plAvBrainGeneric::BrainType>(stream->ReadLE32());
|
|
|
|
fExitFlags = stream->ReadLE32();
|
|
|
|
fMode = static_cast<Mode>(stream->ReadByte());
|
|
|
|
fForward = stream->Readbool();
|
|
|
|
|
|
|
|
if(stream->ReadBool()) {
|
|
|
|
fStartMessage = plMessage::ConvertNoRef(mgr->ReadCreatable(stream));
|
|
|
|
} else {
|
|
|
|
fStartMessage = nil;
|
|
|
|
}
|
|
|
|
if(stream->ReadBool()) {
|
|
|
|
fEndMessage = plMessage::ConvertNoRef(mgr->ReadCreatable(stream));
|
|
|
|
} else {
|
|
|
|
fEndMessage = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
fFadeIn = stream->ReadLEScalar();
|
|
|
|
fFadeOut = stream->ReadLEScalar();
|
|
|
|
fMoveMode = static_cast<MoveMode>(stream->ReadByte());
|
|
|
|
fBodyUsage = static_cast<plAGAnim::BodyUsage>(stream->ReadByte());
|
|
|
|
fRecipient = mgr->ReadKey(stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// MINOR FNs / GETTERS & SETTERS
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// LeaveAge ---------------------
|
|
|
|
// ---------
|
|
|
|
hsBool plAvBrainGeneric::LeaveAge()
|
|
|
|
{
|
|
|
|
IExitMoveMode();
|
|
|
|
|
|
|
|
fMode = kAbort;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddStage --------------------------------------
|
|
|
|
// ---------
|
|
|
|
int plAvBrainGeneric::AddStage(plAnimStage *stage)
|
|
|
|
{
|
|
|
|
if(!fStages)
|
|
|
|
fStages = TRACKED_NEW plAnimStageVec;
|
|
|
|
fStages->push_back(stage);
|
|
|
|
return fStages->size() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetStageNum --------------------------------------
|
|
|
|
// ------------
|
|
|
|
int plAvBrainGeneric::GetStageNum(plAnimStage *stage)
|
|
|
|
{
|
|
|
|
int count = fStages->size();
|
|
|
|
for(int i = 0; i < count; i++)
|
|
|
|
{
|
|
|
|
plAnimStage *any = (*fStages)[i];
|
|
|
|
if(any == stage)
|
|
|
|
{
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCurStageNum --------------------
|
|
|
|
// ---------------
|
|
|
|
int plAvBrainGeneric::GetCurStageNum()
|
|
|
|
{
|
|
|
|
return fCurStage;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetStageCount --------------------
|
|
|
|
// --------------
|
|
|
|
int plAvBrainGeneric::GetStageCount()
|
|
|
|
{
|
|
|
|
return fStages->size();
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetStage ---------------------------------------
|
|
|
|
// ---------
|
|
|
|
plAnimStage * plAvBrainGeneric::GetStage(int which)
|
|
|
|
{
|
|
|
|
return fStages->at(which);
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCurStage ------------------------------
|
|
|
|
// ------------
|
|
|
|
plAnimStage * plAvBrainGeneric::GetCurStage()
|
|
|
|
{
|
|
|
|
return fStages->at(fCurStage);
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetType -------------------------------------------------------------------------------
|
|
|
|
// --------
|
|
|
|
plAvBrainGeneric::BrainType plAvBrainGeneric::SetType(plAvBrainGeneric::BrainType newType)
|
|
|
|
{
|
|
|
|
BrainType oldType = fType;
|
|
|
|
fType = newType;
|
|
|
|
return oldType;
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetType --------------------------------------------
|
|
|
|
// --------
|
|
|
|
plAvBrainGeneric::BrainType plAvBrainGeneric::GetType()
|
|
|
|
{
|
|
|
|
return fType;
|
|
|
|
}
|
|
|
|
|
|
|
|
plAGAnim::BodyUsage plAvBrainGeneric::GetBodyUsage()
|
|
|
|
{
|
|
|
|
return fBodyUsage;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plAvBrainGeneric::SetBodyUsage(plAGAnim::BodyUsage bodyUsage)
|
|
|
|
{
|
|
|
|
fBodyUsage = bodyUsage;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// DEBUGGING
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// DumpToDebugDisplay ----------------------------------------------------------------------------------------
|
|
|
|
// -------------------
|
|
|
|
void plAvBrainGeneric::DumpToDebugDisplay(int &x, int &y, int lineHeight, char *strBuf, plDebugText &debugTxt)
|
|
|
|
{
|
|
|
|
debugTxt.DrawString(x, y, "Brain type: Generic AKA Multistage");
|
|
|
|
y += lineHeight;
|
|
|
|
|
|
|
|
int stageCount = fStages->size();
|
|
|
|
for(int i = 0; i < stageCount; i++)
|
|
|
|
{
|
|
|
|
plAnimStage *stage = (*fStages)[i];
|
|
|
|
stage->DumpDebug(i == fCurStage, x, y, lineHeight, strBuf, debugTxt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|