// Ryzom - 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 .
#ifndef RYAI_STATE_INSTANCE_H
#define RYAI_STATE_INSTANCE_H
#include "event_manager.h"
#include "timer.h"
#include "debug_history.h"
#include "keyword_owner.h"
#include "event_manager.h"
#include "script_vm.h"
#include "ai_grp.h"
class CStateMachine;
class CGroup;
class CAIState;
class CPersistentStateInstance;
class CAliasTreeOwner;
extern NLLIGO::CLigoConfig LigoConfig;
//////////////////////////////////////////////////////////////////////////////
// CStateInstance //
//////////////////////////////////////////////////////////////////////////////
class CStateInstance
: public AIVM::IScriptContext
{
public:
inline
CStateInstance(CAIState* startState);
void init(CAIState* startState);
virtual CPersistentStateInstance* getPersistentStateInstance();
//////////////////////////////////////////////////////////////////////////
// State Persistent.
struct CStatePersistentObjEntry
{
CStatePersistentObjEntry();
CStatePersistentObjEntry(CAIState const* state, NLMISC::CSmartPtr obj);
virtual ~CStatePersistentObjEntry();
CAIState const* _State;
NLMISC::CSmartPtr _Obj;
};
typedef std::vector TStatePersistentObjList;
TStatePersistentObjList _StatePersistentObjList;
// Made to allow obj with life time less or equal to state affectation life time.
void addStatePersistentObj(CAIState const* keyState, NLMISC::CSmartPtr anyObj);
void removeExceptForState(CAIState const* keyState);
/** Try to obtain a group interface from the CStateInstance. Can return NULL if the
* CStateInstance if not implemeted by a group related objet.
*/
// Bad, Bad, Bad ..
virtual CGroup* getGroup() = 0;
//////////////////////////////////////////////////////////////////////////
CAITimerExtended& timerStateTimeout() { return _StateTimeout; }
CAITimerExtended& timerPunctTimeout() { return _PunctualStateTimeout; }
CAITimerExtended& timerUser(uint idx);
CAIState* getCAIState();
virtual CAliasTreeOwner* aliasTreeOwner() = 0;
virtual void stateChange(CAIState const* oldState, CAIState const* newState) = 0;
CAIState* getState() const { return _state; }
void setNextState(CAIState*);
CAIState* getPunctualState() const { return _PunctualState; }
CAIState* getNextPunctualState() const { return _NextPunctualState; }
void setNextPunctualState(CAIState* state);
void cancelPunctualState() { _CancelPunctualState = true; }
std::string buidStateInstanceDebugString() const;
void dumpVarsAndFunctions(CStringWriter& sw) const;
virtual CDebugHistory* getDebugHistory () = 0;
CAIState const* getActiveState() const;
sint32 getUserTimer(uint timerId);
void setUserTimer(uint timerId, sint32 time);
void logicVarsToString(std::string& str) const;
float getNelVar(std::string const& varId);
void setNelVar(std::string const& varId, float value);
void delNelVar(std::string const& varId);
std::string getStrNelVar(std::string const& varId);
void setStrNelVar(std::string const& varId, std::string const& value);
void delStrNelVar(std::string const& varId);
static void setGlobalNelVar(std::string const& varId, float value);
static void setGlobalNelVar(std::string const& varId, std::string value);
CAITimerExtended const& userTimer (uint32 index) const;
bool advanceUserTimer(uint32 nbTicks);
void processStateEvent(CAIEvent const& stateEvent, CAIState const* state = NULL);
// callerStateInstance could be NULL;
void interpretCode(AIVM::IScriptContext* callerStateInstance, AIVM::CByteCodeEntry const& codeScriptEntry);
void interpretCode(AIVM::IScriptContext* callerStateInstance, NLMISC::CSmartPtr const& codeScript);
/// @name IScriptContext implementation
//@{
virtual std::string getContextName();
virtual void interpretCodeOnChildren(AIVM::CByteCodeEntry const& codeScriptEntry);
float getLogicVar(NLMISC::TStringId varId);
void setLogicVar(NLMISC::TStringId varId, float value);
std::string getStrLogicVar(NLMISC::TStringId varId);
void setStrLogicVar(NLMISC::TStringId varId, std::string const& value);
AIVM::IScriptContext* getCtxLogicVar(NLMISC::TStringId varId);
void setCtxLogicVar(NLMISC::TStringId varId, AIVM::IScriptContext* value);
void setFirstBotSpawned();
virtual AIVM::IScriptContext* findContext(NLMISC::TStringId const strId);
virtual void setScriptCallBack(NLMISC::TStringId const& eventName, AIVM::CByteCodeEntry const& codeScriptEntry);
virtual AIVM::CByteCodeEntry const* getScriptCallBackPtr(NLMISC::TStringId const& eventName) const;
virtual void callScriptCallBack(AIVM::IScriptContext* caller, NLMISC::TStringId const& funcName, int mode = 0, std::string const& inParamsSig = "", std::string const& outParamsSig = "", AIVM::CScriptStack* stack = NULL);
virtual void callNativeCallBack(AIVM::IScriptContext* caller, std::string const& funcName, int mode = 0, std::string const& inParamsSig = "", std::string const& outParamsSig = "", AIVM::CScriptStack* stack = NULL);
void blockUserEvent(uint32 eventId);
void unblockUserEvent(uint32 eventId);
bool isUserEventBlocked(uint32 eventId) const;
//@}
protected:
/// Logic variables
typedef std::map TLogicVarList;
typedef std::map TStrLogicVarList;
typedef std::map TLogicVarIndex;
typedef std::map TCtxLogicVarList;
TLogicVarList _LogicVar;
TStrLogicVarList _StrLogicVar;
TCtxLogicVarList _CtxLogicVar;
// Nel variables
typedef std::map*> TNelVarList;
typedef std::map*> TStrNelVarList;
TNelVarList _NelVar;
TStrNelVarList _StrNelVar;
// Callbacks (?)
typedef std::map TCallBackList;
TCallBackList _CallBacks;
/// Flag for variable modification
bool _LogicVarChanged;
bool _LogicVarChangedList[4];
//TLogicVarIndex _VarIndex;
// update logic timers ---------------------------------------------
/// 4 timers available for user logic
CAITimerExtended _UserTimer[4];
/// timer for timing positional states
CAITimerExtended _StateTimeout;
/// current state (index into manager's state vector)
CAIState* _state;
/// variable set to request a state change (std::numeric_limits::max() otherwise)
CAIState* _NextState;
/// timer for timing punctual states
CAITimerExtended _PunctualStateTimeout;
CAIState* _PunctualState;
CAIState* _NextPunctualState;
/// Flag for leaving the punctual state, returning to normal behavior
bool _CancelPunctualState;
bool _FirstBotSpawned;
uint32 _UserEventBlocked;
};
//////////////////////////////////////////////////////////////////////////////
// CPersistentStateInstance //
//////////////////////////////////////////////////////////////////////////////
class CPersistentStateInstance
: public NLMISC::CDbgRefCount
, public CKeyWordOwner
, public CStateInstance
{
public:
CPersistentStateInstance(CStateMachine& reactionContainer);
virtual ~CPersistentStateInstance();
typedef std::vector > TChildList;
void setParentStateInstance(CPersistentStateInstance* parentStateInstance);
CPersistentStateInstance* getParentStateInstance() const { return _ParentStateInstance; }
void addChildStateInstance(CPersistentStateInstance* parentStateInstance);
void removeChildStateInstance(CPersistentStateInstance* parentStateInstance);
TChildList& childs() { return _PSIChilds; }
TChildList _PSIChilds;
// Interface to state status variables -----------------------------
CAIState* getStartState() { return _StartState; }
void setStartState(CAIState* state);
CStateMachine& getEventContainer() { return _Container; }
void updateStateInstance();
//////////////////////////////////////////////////////////////////////////
// CStateInstance
CPersistentStateInstance* getPersistentStateInstance() { return this; }
//////////////////////////////////////////////////////////////////////////
private:
/// id of the state to use at startup
NLMISC::CDbgPtr _StartState;
CStateMachine& _Container;
NLMISC::CDbgPtr _ParentStateInstance;
};
/****************************************************************************/
/* Inlined methods */
/****************************************************************************/
//////////////////////////////////////////////////////////////////////////////
// CStateInstance //
//////////////////////////////////////////////////////////////////////////////
inline
CStateInstance::CStateInstance(CAIState* startState)
{
_UserEventBlocked = 0;
_CtxLogicVar[NLMISC::CStringMapper::map("@this")] = this;
init(startState);
}
inline
CPersistentStateInstance* CStateInstance::getPersistentStateInstance()
{
// nlassert(false);
return (CPersistentStateInstance*)NULL;
}
inline
CStateInstance::CStatePersistentObjEntry::CStatePersistentObjEntry()
{
}
inline
CStateInstance::CStatePersistentObjEntry::CStatePersistentObjEntry(CAIState const* state, NLMISC::CSmartPtr obj)
: _State(state)
, _Obj(obj)
{
}
inline
CStateInstance::CStatePersistentObjEntry::~CStatePersistentObjEntry()
{
_Obj = NULL;
_State = NULL;
}
inline
CAITimerExtended& CStateInstance::timerUser(uint idx)
{
nlassert(idx<4); return _UserTimer[idx];
}
inline
void CStateInstance::logicVarsToString(std::string& str) const
{
for (TLogicVarList::const_iterator it=_LogicVar.begin(), itEnd=_LogicVar.end();it!=itEnd;++it)
str+=*(it->first)+"="+NLMISC::toString(it->second)+" ";
}
inline
float CStateInstance::getLogicVar(NLMISC::TStringId varId)
{
TLogicVarList::iterator it=_LogicVar.find(varId);
if (it==_LogicVar.end())
{
_LogicVar[varId]=0.f;
return 0.f;
}
return it->second; //_LogicVar[varId];
}
inline
void CStateInstance::setLogicVar(NLMISC::TStringId varId, float value)
{
_LogicVar[varId] = value;
_LogicVarChanged = true;
if (varId && varId->size() == 2 && (*varId)[0] == 'v')
{
sint32 index = (*varId)[1] - '0';
if (0 <= index && index < 4)
{
//_VarIndex[varId]
_LogicVarChangedList[static_cast(index)] = true;
}
}
}
inline
std::string CStateInstance::getStrLogicVar(NLMISC::TStringId varId)
{
return _StrLogicVar[varId];
}
inline
void CStateInstance::setStrLogicVar(NLMISC::TStringId varId, std::string const& value)
{
_StrLogicVar[varId] = value;
_LogicVarChanged = true;
if (varId && varId->size() == 2 && (*varId)[0] == 'v')
{
sint32 index = (*varId)[1] - '0';
if (0 <= index && index < 4)
{
//_VarIndex[varId]
_LogicVarChangedList[static_cast(index)] = true;
}
}
//_LogicVarChangedList[_VarIndex[varId]] = true;
}
inline
AIVM::IScriptContext* CStateInstance::getCtxLogicVar(NLMISC::TStringId varId)
{
return _CtxLogicVar[varId];
}
inline
void CStateInstance::setCtxLogicVar(NLMISC::TStringId varId, AIVM::IScriptContext* value)
{
_CtxLogicVar[varId] = value;
_LogicVarChanged = true;
}
inline
float CStateInstance::getNelVar(std::string const& varId)
{
TNelVarList::iterator it = _NelVar.find(varId);
if (it != _NelVar.end()) return it->second->get();
if (NLMISC::CVariable::exists(varId))
{
nlwarning("Nel variable \"%s\" exists outside of this state instance", varId.c_str());
return 0.f;
}
_NelVar[varId] = new NLMISC::CVariable("StateInstance", varId.c_str(), "", 0.f);
return _NelVar[varId]->get();
}
inline
void CStateInstance::setNelVar(std::string const& varId, float value)
{
TNelVarList::iterator it = _NelVar.find(varId);
if (it==_NelVar.end())
{
if (NLMISC::CVariable::exists(varId))
{
nlwarning("Nel variable \"%s\" exists outside of this state instance", varId.c_str());
return;
}
_NelVar[varId] = new NLMISC::CVariable("StateInstance", varId.c_str(), "", 0.f);
}
_NelVar[varId]->set(value);
}
inline
void CStateInstance::delNelVar(std::string const& varId)
{
TNelVarList::iterator it = _NelVar.find(varId);
if (it!=_NelVar.end())
{
delete it->second;
_NelVar.erase(it);
}
else
{
nldebug("Trying to delete Nel variable that doesn't exist: \"%s\"", varId.c_str());
}
}
inline
std::string CStateInstance::getStrNelVar(std::string const& varId)
{
TStrNelVarList::iterator it = _StrNelVar.find(varId);
if (it != _StrNelVar.end()) return it->second->get();
if (NLMISC::CVariable::exists(varId))
{
nlwarning("Nel variable \"%s\" exists outside of this state instance", varId.c_str());
return "";
}
_StrNelVar[varId] = new NLMISC::CVariable("StateInstanceStrVar", varId.c_str(), "", std::string());
return _StrNelVar[varId]->get();
}
inline
void CStateInstance::setStrNelVar(std::string const& varId, std::string const& value)
{
TStrNelVarList::iterator it = _StrNelVar.find(varId);
if (it==_StrNelVar.end())
{
if (NLMISC::CVariable::exists(varId))
{
nlwarning("Nel variable \"%s\" exists outside of this state instance", varId.c_str());
return;
}
_StrNelVar[varId] = new NLMISC::CVariable("StateInstanceStrVar", varId.c_str(), "", std::string());
}
_StrNelVar[varId]->set(value);
}
inline
void CStateInstance::delStrNelVar(std::string const& varId)
{
TStrNelVarList::iterator it = _StrNelVar.find(varId);
if (it!=_StrNelVar.end())
{
delete it->second;
_StrNelVar.erase(it);
}
else
{
nldebug("Trying to delete Nel variable that doesn't exist: \"%s\"", varId.c_str());
}
}
inline
CAITimerExtended const& CStateInstance::userTimer (uint32 index) const
{
return _UserTimer[index];
}
inline
bool CStateInstance::advanceUserTimer(uint32 nbTicks)
{
for (uint k=0; k<4; ++k)
{
const uint32 t = getUserTimer(k);
setUserTimer(k, (t>nbTicks)?(t-nbTicks):0);
}
return true;
}
inline
void CStateInstance::processStateEvent(CAIEvent const& stateEvent, CAIState const* state)
{
// NOTE: This is a quick and inefficient implementation of event treatment - needs to be re-worked
// note that it is OK for state to be 'NULL'
if (!state)
{
state=getActiveState();
if (!state)
return;
}
bool foundReaction=false;
// nlassert(_mgr);
for (uint i=0;iaddHistory("STATE: '%s' EVENT: '%s' REACTION: '%s'", state->getAliasNode()->fullName().c_str(),
stateEvent.getName().c_str(), reaction.getAliasNode()->fullName().c_str());
foundReaction=true;
if (!reaction.getAction())
{
nlwarning("Failed to find action for event: %s",reaction.getAliasNode()->fullName().c_str());
continue;
}
if (!reaction.getAction()->executeAction(this, NULL))
{
nlwarning("Failed to execute action for event '%s': for stateInstance:'%s' in state:'%s'", stateEvent.getName().c_str(),
aliasTreeOwner()->getAliasNode()->fullName().c_str(), state->getAliasNode()->fullName().c_str());
continue;
}
}
if (!foundReaction)
{
getDebugHistory()->addHistory("STATE: '%s' EVENT: '%s' NO REACTION", state->getAliasNode()->fullName().c_str(),
stateEvent.getName().c_str());
}
}
inline
void CStateInstance::setNextState(CAIState* state)
{
// we're allowed to set state to 'no state'
if (!state)
{
_NextState = state;
return;
}
// make sure the state is positional (not punctual)
if (!state->isPositional())
{
nlwarning("setNextState(): State should not be punctual '%s'%s - setting state to std::numeric_limits::max()",
state->getAliasNode()->fullName().c_str(),
state->getAliasString().c_str());
_NextState = NULL;
return;
}
// set the next state
_NextState = state;
}
inline
void CStateInstance::setNextPunctualState(CAIState* state)
{
// we're allowed to set state to 'no state'
if (!state)
return;
// make sure the state is not positional (ie punctual)
if (state->isPositional())
{
nlwarning("CStateInstance::setNextPunctualState(): State should be punctual '%s'%s - setting state to std::numeric_limits::max()",
state->getAliasNode()->fullName().c_str(),
state->getAliasString().c_str());
state = NULL;
return;
}
// set the next state
_NextPunctualState = state;
}
//////////////////////////////////////////////////////////////////////////////
// CPersistentStateInstance //
//////////////////////////////////////////////////////////////////////////////
inline
CPersistentStateInstance::CPersistentStateInstance(CStateMachine& reactionContainer)
: CKeyWordOwner()
, CStateInstance(NULL)
, _StartState()
, _Container(reactionContainer)
{
}
inline
CPersistentStateInstance::~CPersistentStateInstance()
{
#if !FINAL_VERSION
if (_PSIChilds.size()!=0)
nlwarning("a Npc group is dying and still have childs, which is not possible !!");
nlassert(_PSIChilds.size()==0);
#endif
if (!_ParentStateInstance.isNULL())
_ParentStateInstance->removeChildStateInstance(this);
}
inline
void CPersistentStateInstance::setParentStateInstance(CPersistentStateInstance* parentStateInstance)
{
if (!_ParentStateInstance.isNULL())
_ParentStateInstance->removeChildStateInstance(this);
_ParentStateInstance=parentStateInstance;
if (parentStateInstance!=NULL)
parentStateInstance->addChildStateInstance(this);
}
inline
void CPersistentStateInstance::addChildStateInstance(CPersistentStateInstance* parentStateInstance)
{
#if !FINAL_VERSION
nlassert(std::find(_PSIChilds.begin(), _PSIChilds.end(), NLMISC::CDbgPtr(parentStateInstance))==_PSIChilds.end());
#endif
_PSIChilds.push_back(parentStateInstance);
}
inline
void CPersistentStateInstance::removeChildStateInstance(CPersistentStateInstance* parentStateInstance)
{
TChildList::iterator it = std::find(_PSIChilds.begin(), _PSIChilds.end(), NLMISC::CDbgPtr(parentStateInstance));
#if !FINAL_VERSION
nlassert(it!=_PSIChilds.end());
#endif
if (it!=_PSIChilds.end())
_PSIChilds.erase(it);
}
inline
void CPersistentStateInstance::setStartState(CAIState* state)
{
_StartState = state;
CStateInstance::init(_StartState);
}
#endif