614 lines
18 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/>.
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==*/
/////////////////////////////////////////////////////////////////////////////
//
// NAME: cyAnimation
//
// PURPOSE: Class wrapper to map animation functions to plasma2 message
//
#include "plgDispatch.h"
#include "plMessage/plAnimCmdMsg.h"
#include "pnMessage/plEventCallbackMsg.h"
#include "cyAnimation.h"
cyAnimation::cyAnimation()
{
fSender = nil;
fAnimName = nil;
fNetForce = false;
}
cyAnimation::cyAnimation(pyKey& sender)
{
SetSender(sender);
fAnimName = nil;
fNetForce = false;
}
// copy constructor
cyAnimation::cyAnimation(const cyAnimation& anim)
{
fSender = anim.fSender;
fRecvr = anim.fRecvr;
// here is why we needed the copy constructor
fAnimName = hsStrcpy(anim.fAnimName); // make our own copy of this string
fNetForce = anim.fNetForce;
}
// clean up on the way out
cyAnimation::~cyAnimation()
{
if (fAnimName != nil )
{
delete [] fAnimName;
fAnimName = nil;
}
}
// setters
void cyAnimation::SetSender(pyKey& sender)
{
fSender = sender.getKey();
}
void cyAnimation::AddRecvr(pyKey& recvr)
{
fRecvr.Append(recvr.getKey());
}
PyObject* cyAnimation::GetFirstRecvr()
{
if ( fRecvr.Count() > 0 )
return pyKey::New(fRecvr[0]);
return nil;
}
void cyAnimation::SetAnimName(const char* name)
{
if ( fAnimName != nil )
delete [] fAnimName;
fAnimName = hsStrcpy(name);
}
void cyAnimation::SetNetForce(hsBool state)
{
// set our flag
fNetForce = state;
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : Play
// PARAMETERS :
//
// PURPOSE : Play animation from start to end (whatever is already set)
//
void cyAnimation::Play()
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
// NOTE: The animation modifier will set the animation back to the starting point automatically
// then continue from there
pMsg->SetCmd(plAnimCmdMsg::kGoToBegin);
pMsg->SetCmd(plAnimCmdMsg::kContinue);
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : Stop
// PARAMETERS :
//
// PURPOSE : Stop an animation
//
void cyAnimation::Stop()
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kStop);
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : Resume
// PARAMETERS :
//
// PURPOSE : Continue playing animation from wherever it last stopped
//
void cyAnimation::Resume()
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kContinue);
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : PlayRange
// PARAMETERS : start - start time for the range
// : end - end time of the range
//
// PURPOSE : Play an animation only from specific time start to end
//
void cyAnimation::PlayRange(hsScalar start, hsScalar end)
{
SkipToTime(start);
PlayToTime(end);
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : PlayToTime
// PARAMETERS : time - where to stop playing animation
//
// PURPOSE : Play (continue) an animation until the specified time is reached
//
void cyAnimation::PlayToTime(hsScalar time)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kPlayToTime);
pMsg->fTime = time;
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : PlayToPercentage
// PARAMETERS : zeroToOne - How far (scale of 0 to 1) to play into the anim
//
// PURPOSE : Play (continue) an animation until the specified point is reached
//
void cyAnimation::PlayToPercentage(hsScalar zeroToOne)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kPlayToPercentage);
pMsg->fTime = zeroToOne;
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : SkipToTime
// PARAMETERS :
//
// PURPOSE : Jump the animation to the specified time
// : Doesn't start or stop playing of animation
//
void cyAnimation::SkipToTime(hsScalar time)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kGoToTime);
pMsg->fTime = time;
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : Looped
// PARAMETERS : looped - state to change to
//
// PURPOSE : Set whether the animation is to be looped or not
//
void cyAnimation::Looped(hsBool looped)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
if ( looped )
pMsg->SetCmd(plAnimCmdMsg::kSetLooping);
else
pMsg->SetCmd(plAnimCmdMsg::kUnSetLooping);
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : Backwards
// PARAMETERS : backwards - state of the backwards flag
//
// PURPOSE : Sets the backwards state for the animation
//
void cyAnimation::Backwards(hsBool backwards)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
if ( backwards )
pMsg->SetCmd(plAnimCmdMsg::kSetBackwards);
else
pMsg->SetCmd(plAnimCmdMsg::kSetForewards);
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : SetLoopStart and SetLoopEnd
// PARAMETERS : value - sets the start or the end of the animation
//
void cyAnimation::SetLoopStart(hsScalar start)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kSetLoopBegin);
pMsg->fLoopBegin = start;
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
void cyAnimation::SetLoopEnd(hsScalar end)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kSetLoopEnd);
pMsg->fLoopEnd = end;
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
/////////////////////////////////////////////////////////////////////////////
//
// Function : Speed
// PARAMETERS : speed - speed to set the animation to
//
// PURPOSE : Sets the speed of the animation
// : Doesn't start or stop playing animation
//
void cyAnimation::Speed(hsScalar speed)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(plAnimCmdMsg::kSetSpeed);
pMsg->fSpeed = speed;
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
void cyAnimation::IRunOneCmd(int cmd)
{
// must have a receiver!
if ( fRecvr.Count() > 0 )
{
// create message
plAnimCmdMsg* pMsg = TRACKED_NEW plAnimCmdMsg;
// check if this needs to be network forced to all clients
if (fNetForce )
{
// set the network propagate flag to make sure it gets to the other clients
pMsg->SetBCastFlag(plMessage::kNetPropagate);
pMsg->SetBCastFlag(plMessage::kNetForce);
}
if ( fSender )
pMsg->SetSender(fSender);
// add all our receivers to the message receiver list
int i;
for ( i=0; i<fRecvr.Count(); i++ )
{
pMsg->AddReceiver(fRecvr[i]);
}
// set the notetrack name (if there is one)
if ( fAnimName != nil )
pMsg->SetAnimName(fAnimName);
pMsg->SetCmd(cmd);
plgDispatch::MsgSend( pMsg ); // whoosh... off it goes
}
}
void cyAnimation::SkipToBegin()
{
IRunOneCmd(plAnimCmdMsg::kGoToBegin);
}
void cyAnimation::SkipToEnd()
{
IRunOneCmd(plAnimCmdMsg::kGoToEnd);
}
void cyAnimation::SkipToLoopBegin()
{
IRunOneCmd(plAnimCmdMsg::kGoToLoopBegin);
}
void cyAnimation::SkipToLoopEnd()
{
IRunOneCmd(plAnimCmdMsg::kGoToLoopEnd);
}
// Bump the animation ahead one frame (whatever deltime is)
//
void cyAnimation::IncrementForward()
{
IRunOneCmd(plAnimCmdMsg::kIncrementForward);
}
// Bump the animation back one frame (whatever deltime is)
//
void cyAnimation::IncrementBackward()
{
IRunOneCmd(plAnimCmdMsg::kIncrementBackward);
}