// 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 .
#include "stdpch.h"
#include "ai_mgr.h"
#include "ai_mgr_fauna.h"
#include "ai_mgr_npc.h"
#include "ai_mgr_pet.h"
#include "owners.h"
using namespace MULTI_LINE_FORMATER;
CManager::CManager(IManagerParent* parent, uint32 alias, std::string const& name, std::string const& filename)
: CAliasChild(parent, alias, name)
, CAliasTreeRoot(filename)
{
_CurSpawnRing = 0;
}
CManager::~CManager()
{
release();
_Groups.setChildSize(0);
}
CManager* CManager::createManager(AITYPES::TMgrType type, IManagerParent* managerParent, uint32 alias, std::string const& name, std::string const& mapName, std::string const& filename)
{
CManager* manager = NULL;
switch (type)
{
case AITYPES::MgrTypeFauna: manager = new CMgrFauna(managerParent, alias,name, filename); break;
case AITYPES::MgrTypeKami: manager = new CMgrNpc(managerParent, alias, name, filename); break;
case AITYPES::MgrTypeNpc: manager = new CMgrNpc(managerParent, alias, name, filename); break;
case AITYPES::MgrTypeKaravan: manager = new CMgrNpc(managerParent, alias,name, filename); break;
case AITYPES::MgrTypePet: manager = new CMgrPet(managerParent, alias,name, filename); break;
default:
nlwarning("CManager::createManager(): FAILED to instantiate manager due unmanaged type: %s", AITYPES::getName(type));
break;
}
return manager;
}
void CManager::update()
{
if (_SpawnList.size()>0)
{
H_AUTO(MgrUpdateSpawnList)
// deal with the spawn list
// if there are bot groups waiting to be spawned then treat a few of them
for (uint count=0;!_SpawnList.empty() && count<_MaxSpawns;count++)
{
while (_SpawnList.size()>0)
{
_CurSpawnRing++;
if (_CurSpawnRing>=_SpawnList.size())
_CurSpawnRing=0;
if (!_SpawnList[_CurSpawnRing]->isSpawned())
break;
_SpawnList.erase(_SpawnList.begin()+_CurSpawnRing); // remove the group.
}
if (_SpawnList.size()>0)
{
if (_SpawnList[_CurSpawnRing]->spawn())
_SpawnList.erase(_SpawnList.begin()+_CurSpawnRing); // remove the group.
}
}
}
if (groups().size()==0)
return;
// we're going to run through the groups updating each one once every n ticks where n is
// determined by the 'getUpdatePriority()' mask
// note that 'getUpdatePriority()' needs to be renamed but it's late, I'm tired and I just need
// an unused variable to work with that already exists so that I don't have to go through a complete
// project recompile.
uint32 timeOffset = CTimeInterface::gameCycle();
{
H_AUTO(MgrUpdateSpawned);
int index = -1;
FOREACH(it, CCont, groups())
{
++index;
NLMISC::CSmartPtr persistent = *it; // To avoid destruction while executing ..
if (!persistent->isSpawned())
continue;
CSpawnGroup* const spawned = persistent->getSpawnObj();
if (spawned->getUpdatePriority()==1)
++AISStat::GrpFastUpdCtr;
if (spawned->getUpdatePriority()==31)
++AISStat::GrpSlowUpdCtr;
if (((index+timeOffset)&spawned->getUpdatePriority())!=0)
continue;
spawned->update();
}
}
}
CGroup* CManager::getNextValidGroupChild(CGroup* group)
{
return _Groups.getNextValidChild(static_cast(group));
}
void CManager::serviceEvent(CServiceEvent const& info)
{
FOREACH(it, CCont, groups())
it->serviceEvent(info);
}
void CManager::spawn()
{
FOREACH(it, CCont, groups())
_SpawnList.push_back(*it);
_MaxSpawns = (groups().size()+127)/(127+1); // the 127 and 128 are arbitrary ( D M ).
}
void CManager::despawnMgr()
{
FOREACH(it, CCont, groups())
it->despawnGrp();
}
CGroup* CManager::getGroup(uint32 index)
{
if (index>=groups().size())
return NULL;
return groups()[index];
}
bool CManager::isEmpty() const
{
for(uint k = 0; k < _Groups.size(); ++k)
{
if (_Groups[k] && _Groups[k]->getNextValidBotChild() != NULL) return false;
}
return true;
}
std::string CManager::getIndexString() const
{
return getOwner()->getManagerIndexString(this);
}
std::string CManager::getOneLineInfoString() const
{
return std::string("Manager '") + getName() + "'";
}
std::string CManager::getFullName() const
{
return getName();
}
std::vector CManager::getMultiLineInfoString() const
{
std::vector container;
pushTitle(container, "CManager");
pushEntry(container, "id=" + getIndexString());
container.back() += " type=" + std::string(AITYPES::getName(type()));
container.back() += " alias=" + getAliasString();
container.back() += " name=" + getName();
pushFooter(container);
return container;
}
void CManager::release()
{
}
void CManager::addToSpawn(CGroup* group)
{
for (size_t i=0; i<_SpawnList.size(); ++i)
{
if (_SpawnList[i].ptr()==group)
return;
}
_SpawnList.push_back(group);
}
void CManager::removeFromSpawnList(CGroup const* group)
{
for (size_t i=0; i<_SpawnList.size(); ++i)
{
if (_SpawnList[i]==group)
{
_SpawnList.erase(_SpawnList.begin()+i); // remove the group.
return;
}
}
}
CAIInstance* CManager::getAIInstance() const
{
return getOwner()->getAIInstance();
}