// NeL - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
//
// Includes
//
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "animation.h"
#include "entities.h"
#include "snowballs_client.h"
//
// Namespaces
//
using namespace std;
using namespace NLMISC;
using namespace NL3D;
namespace SBCLIENT {
//
// Constantes
//
// Amount of time for the transistion between 2 animations
CAnimationTime TransitionTime = 0.25f;
//
// Variables
//
UAnimationSet *AnimationSet = NULL;
UPlayListManager *PlayListManager = NULL;
struct Anim
{
char *Name;
bool Loop;
uint Id;
UAnimation *Animation;
};
Anim AnimIdArray[][2] =
{
{ { "patterfeet.anim", false, 0, NULL }, { "", false, 0, NULL } },
{ { "marche.anim", true, 0, NULL }, { "", false, 0, NULL } },
{ { "idle.anim", true, 0, NULL }, { "", false, 0, NULL } },
{ { "log_on.anim", false, 0, NULL }, { "", false, 0, NULL } },
{ { "log_off.anim", false, 0, NULL }, { "", false, 0, NULL } },
{ { "lancelaboule.anim", false, 0, NULL }, { "", false, 0, NULL } },
{ { "prepaboule.anim", false, 0, NULL }, { "", false, 0, NULL } },
{ { "prepaboulecycle.anim", true, 0, NULL }, { "", false, 0, NULL } },
{ { "impact.anim", false, 0, NULL }, { "", false, 0, NULL } },
};
//
// Functions
//
void computeAnimation (CEntity &entity, EAnim anim)
{
// Get the current time
double currentTime = double (CTime::getLocalTime ())/1000.0f;
// nlinfo ("%d playing animation", anim);
// nlinfo ("%d playing animation %s ct%f st%f et%f", anim, AnimIdArray[anim][0].Name, currentTime, AnimIdArray[anim][0].Animation->getBeginTime (), AnimIdArray[anim][0].Animation->getEndTime ());
// Find the new slot for the full animation (0 or 1)
uint newSlot = entity.NextEmptySlot;
uint oldSlot = 1 - entity.NextEmptySlot;
entity.NextEmptySlot = 1 - entity.NextEmptySlot;
UPlayList::TWrapMode wrapMode = AnimIdArray[anim][0].Loop ? UPlayList::Repeat : UPlayList::Clamp;
entity.PlayList->setAnimation (newSlot, AnimIdArray[anim][0].Id);
entity.PlayList->setTimeOrigin (newSlot, currentTime);
entity.PlayList->setWeightSmoothness (newSlot, 1.0f);
entity.PlayList->setWrapMode (newSlot, wrapMode);
double OldStartTime, OldEndTime;
double NewStartTime, NewEndTime;
// Get the starting time of the old animation slot
entity.PlayList->getStartWeight (oldSlot, OldStartTime);
// Compute the time delta between start of the old animation and now
double dt = currentTime - OldStartTime;
// Compute the new transition value depending of the current time
if (dt > TransitionTime)
dt = TransitionTime;
OldStartTime = currentTime - (TransitionTime - dt);
OldEndTime = currentTime + dt;
NewStartTime = currentTime;
NewEndTime = currentTime + dt;
// Set new weights on the old and the new animation slot
entity.PlayList->setStartWeight (oldSlot, 1.0f, OldStartTime);
entity.PlayList->setEndWeight (oldSlot, 0.0f, OldEndTime);
entity.PlayList->setStartWeight (newSlot, 0.0f, NewStartTime);
entity.PlayList->setEndWeight (newSlot, 1.0f, OldEndTime);
// Keep in mind what is the last animation id we set
entity.StartAnimationTime = (float)currentTime;
}
void playAnimation (CEntity &entity, EAnim anim, bool force)
{
nlassert (anim > -2 && anim < 20);
// nlinfo ("playAnimation() %d", anim);
// Get the current time
CAnimationTime currentTime = CAnimationTime(CTime::getLocalTime ())/1000.0f;
// Can't do animation without skeleton
if (entity.Skeleton.empty())
return;
// If the first time we play an animation, creates the animation class
if (entity.PlayList == NULL)
createAnimation (entity);
if (force || entity.AnimQueue.empty())
{
computeAnimation (entity, anim);
// clear the animation queue
//nlinfo ("clearing animation queue");
while (!entity.AnimQueue.empty())
entity.AnimQueue.pop ();
}
// nlinfo ("pushing animation %d", anim);
// nlinfo ("pushing animation %s", AnimIdArray[anim][0].Name);
entity.AnimQueue.push (anim);
}
void createAnimation (CEntity &entity)
{
nlassert (!entity.Instance.empty() && !entity.Skeleton.empty() && AnimationSet != NULL);
entity.PlayList = PlayListManager->createPlayList (AnimationSet);
entity.PlayList->registerTransform (entity.Instance);
entity.PlayList->registerTransform (entity.Skeleton);
}
void deleteAnimation (CEntity &entity)
{
if (entity.PlayList == NULL)
return;
PlayListManager->deletePlayList (entity.PlayList);
entity.PlayList= NULL;
}
void initAnimation()
{
AnimationSet = Driver->createAnimationSet ();
// Add all animations in the animation set
for (uint i = 0; i < sizeof (AnimIdArray) / sizeof (AnimIdArray[0]); i++)
{
if (AnimIdArray[i][0].Name[0] != '\0')
{
AnimIdArray[i][0].Id = AnimationSet->addAnimation (AnimIdArray[i][0].Name, AnimIdArray[i][0].Name);
AnimIdArray[i][0].Animation = AnimationSet->getAnimation (AnimIdArray[i][0].Id);
}
if (AnimIdArray[i][1].Name[0] != '\0')
{
AnimIdArray[i][1].Id = AnimationSet->addAnimation (AnimIdArray[i][1].Name, AnimIdArray[i][1].Name);
AnimIdArray[i][1].Animation = AnimationSet->getAnimation (AnimIdArray[i][1].Id);
}
}
AnimationSet->build ();
PlayListManager = Scene->createPlayListManager ();
}
void updateAnimation()
{
// Get the current time
CAnimationTime currentTime = CAnimationTime(CTime::getLocalTime ())/1000.0f;
for (EIT eit = Entities.begin (); eit != Entities.end (); eit++)
{
CEntity &entity = (*eit).second;
if (entity.AnimQueue.empty ())
{
// nlwarning ("empty queue update!!!");
continue;
}
EAnim currentAnim = entity.AnimQueue.front ();
if (!AnimIdArray[currentAnim][0].Loop && currentTime >= entity.StartAnimationTime + AnimIdArray[currentAnim][0].Animation->getEndTime () - TransitionTime/2)
{
// remove the current anim
entity.AnimQueue.pop ();
if (entity.AnimQueue.empty ())
{
// nlwarning ("empty queue!!!!!!");
continue;
}
EAnim newAnim = entity.AnimQueue.front ();
computeAnimation (entity, newAnim);
/*
nlinfo ("playing animation %s ct%f st%f et%f", AnimIdArray[newAnim][0].Name, currentTime, AnimIdArray[newAnim][0].Animation->getBeginTime (), AnimIdArray[newAnim][0].Animation->getEndTime ());
// setup the new anim
entity.PlayList->setAnimation (0, AnimIdArray[newAnim][0].Id);
entity.PlayList->setTimeOrigin (0, currentTime);
entity.PlayList->setStartWeight (0, 1.0f, currentTime);
entity.PlayList->setEndWeight (0, 1.0f, currentTime+TransitionTime);
entity.PlayList->setWeightSmoothness (0, 1.0f);
if (AnimIdArray[newAnim][0].Loop)
entity.PlayList->setWrapMode (0, UPlayList::Repeat);
else
entity.PlayList->setWrapMode (0, UPlayList::Clamp);
entity.StartAnimationTime = currentTime;
*/ }
}
// compute new animation position depending of the current time
PlayListManager->animate (double(CTime::getLocalTime ())/1000.0f);
}
void releaseAnimation()
{
Scene->deletePlayListManager (PlayListManager);
// The next line doesn t work (say that AnimationSet is not a valid AnimationSet Ptr) so we comment it.
// Scene->deleteAnimationSet (AnimationSet);
}
} /* namespace SBCLIENT */
/* end of file */