khanat-opennel-code/code/ryzom/server/src/ai_service/continent_inline.h

1352 lines
39 KiB
C++

// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
// 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 <http://www.gnu.org/licenses/>.
#ifndef CONTINENT_INLINE_H
#define CONTINENT_INLINE_H
#include "ai_bot.h"
#include "ai_grp_fauna.h"
#include "ai_grp_npc.h"
#include "ai_bot_npc.h"
#include "ais_actions.h"
extern NLMISC::CVariable<bool> LogAcceptablePos;
extern NLMISC::CVariable<bool> LogGroupCreationFailure;
extern NLMISC::CVariable<bool> LogOutpostDebug;
//////////////////////////////////////////////////////////////////////////////
// CPopulationRecord //
//////////////////////////////////////////////////////////////////////////////
inline
CPopulationRecord::CPopulationRecord(AISHEETS::ICreatureCPtr const& sheetData, uint count)
: _SheetData(sheetData)
, _Count(count)
{
}
inline
bool CPopulationRecord::operator==(CPopulationRecord const& other) const
{
return (_Count==other._Count && _SheetData==other._SheetData);
}
inline
AISHEETS::ICreatureCPtr CPopulationRecord::getCreatureSheet() const
{
return _SheetData;
}
inline
uint CPopulationRecord::getBotCount(bool useCreatureMultiplier) const
{
if (useCreatureMultiplier && _SheetData)
return _Count * _SheetData->DynamicGroupCountMultiplier();
else
return _Count;
}
inline
uint32 CPopulationRecord::getEnergyValue(bool useCreatureMultiplier) const
{
if (_SheetData)
{
return getBotCount(useCreatureMultiplier)*_SheetData->EnergyValue();
}
else
{
#ifdef NL_DEBUG
nlassert(_SheetData);
#endif
return 0;
}
}
//////////////////////////////////////////////////////////////////////////////
// CPopulation //
//////////////////////////////////////////////////////////////////////////////
inline
CPopulation::CPopulation(CPopulationOwner* owner, CAIAliasDescriptionNode* aliasDescription)
: CAliasChild<CPopulationOwner>(owner, aliasDescription)
, _Weight(0)
, _SpawnType(AITYPES::SpawnTypeBadType)
{
}
inline
CPopulation::CPopulation(CPopulationOwner* owner, uint32 alias, std::string name)
: CAliasChild<CPopulationOwner>(owner, alias, name)
, _Weight(0)
, _SpawnType(AITYPES::SpawnTypeBadType)
{
}
inline
void CPopulation::addPopRecord(CPopulationRecord popRecord)
{
_PopRecords.push_back(popRecord);
}
//////////////////////////////////////////////////////////////////////////////
// CAICircle //
//////////////////////////////////////////////////////////////////////////////
/*
// Template members must be inside class definitions for VC++6
template <class V, class R>
CAICircle::CAICircle(V const& center, R radius)
: _Center(center)
, _Radius(radius)
{
}
// Template members must be inside class definitions for VC++6
template <class V>
bool CAICircle::isInside(V const& pos)
{
return (pos - _Center).sqrnorm() <= (_Radius+1)*(_Radius+1);
}
*/
//////////////////////////////////////////////////////////////////////////////
// CAabb //
//////////////////////////////////////////////////////////////////////////////
inline
CAabb::CAabb()
: _VMin(INT_MAX/CAICoord::UNITS_PER_METER, INT_MAX/CAICoord::UNITS_PER_METER)
, _VMax(INT_MIN/CAICoord::UNITS_PER_METER, INT_MIN/CAICoord::UNITS_PER_METER)
{
}
/*
template <class V>
CAabb::CAabb(std::vector<V> const& coords)
: _VMax(INT_MIN/CAICoord::UNITS_PER_METER, INT_MIN/CAICoord::UNITS_PER_METER)
, _VMin(INT_MAX/CAICoord::UNITS_PER_METER, INT_MAX/CAICoord::UNITS_PER_METER)
{
for (uint k=0; k<coords.size(); ++k)
includePoint(coords[k]);
}
template <class V>
void CAabb::includePoint(V const& point)
{
if (point.x() < _VMin.x())
_VMin.setX(point.x());
if (point.x() > _VMax.x())
_VMax.setX(point.x());
if (point.y() < _VMin.y())
_VMin.setY(point.y());
if (point.y() > _VMax.y())
_VMax.setY(point.y());
}
*/
inline
void CAabb::includeAabb(CAabb const& box)
{
if (box._VMin.x() < _VMin.x())
_VMin.setX(box._VMin.x());
if (box._VMax.x() > _VMax.x())
_VMax.setX(box._VMax.x());
if (box._VMin.y() < _VMin.y())
_VMin.setY(box._VMin.y());
if (box._VMax.y() > _VMax.y())
_VMax.setY(box._VMax.y());
}
inline
void CAabb::init()
{
*this = CAabb();
}
/*
// Template members must be inside class definitions for VC++6
template<class V>
bool CAabb::isInside(V const& v)
{
return v.x() >= _VMin.x() && v.y() >= _VMin.y() && v.x() <= _VMax.x() && v.y() <= _VMax.y();
}
*/
//////////////////////////////////////////////////////////////////////////////
// CFaunaZone //
//////////////////////////////////////////////////////////////////////////////
inline
bool CFaunaZone::haveActivity(AITYPES::CPropertyId const& activity) const
{
if (!_AdditionalActivities.empty())
return _AdditionalActivities.have(activity);
return _InitialActivities.have(activity);
}
inline
bool CFaunaZone::haveActivity(AITYPES::CPropertySet const& activities) const
{
if (!_AdditionalActivities.empty())
return _AdditionalActivities.containsPartOfStrict(activities);
return _InitialActivities.containsPartOfStrict(activities);
}
inline
float CFaunaZone::getFreeAreaScore() const
{
float radius = getRadius();
return (float)((radius*radius)/((float)getNbUse()+0.001));
}
inline
uint32 CFaunaZone::getNbUse() const
{
sint nbUse = getRefCount()-1; // -1 for the container
#ifdef NL_DEBUG
nlassert(nbUse>=0);
#endif
return nbUse;
}
//////////////////////////////////////////////////////////////////////////////
// CAIRefPlaceXYR //
//////////////////////////////////////////////////////////////////////////////
inline
CAIRefPlaceXYR::CAIRefPlaceXYR(CPlaceOwner* owner, CAIPlace const* zone)
: CAIPlace(owner, NULL)
{
#ifdef NL_DEBUG
nlassert(zone!=NULL);
#endif
_Zone = zone;
}
inline
CAIRefPlaceXYR::operator CAIPlace const*() const
{
return _Zone;
}
//////////////////////////////////////////////////////////////////////////////
// CAIRefPlaceXYRFauna //
//////////////////////////////////////////////////////////////////////////////
inline
CAIRefPlaceXYRFauna::CAIRefPlaceXYRFauna(CPlaceOwner* owner, CAIPlace const* zone)
: CAIRefPlaceXYR(owner, zone)
{
}
//////////////////////////////////////////////////////////////////////////////
// CRebuildContinentAndOutPost //
//////////////////////////////////////////////////////////////////////////////
inline
CRebuildContinentAndOutPost::CRebuildContinentAndOutPost(CContinent* continent)
: _Continent(continent)
{
}
inline
bool CRebuildContinentAndOutPost::absorb(CLazyProcess const& lazyProcess) const
{
CRebuildContinentAndOutPost const* other = dynamic_cast<CRebuildContinentAndOutPost const*>(&lazyProcess);
if (!other)
return false;
return *other==*this;
}
inline
bool CRebuildContinentAndOutPost::operator==(CRebuildContinentAndOutPost const& other) const
{
return other._Continent==_Continent;
}
//////////////////////////////////////////////////////////////////////////////
// CContinent //
//////////////////////////////////////////////////////////////////////////////
inline
CContinent::CContinent(CAIInstance* owner)
: CChild<CAIInstance>(owner)
{
}
//////////////////////////////////////////////////////////////////////////////
// CRoad //
//////////////////////////////////////////////////////////////////////////////
inline
void CRoad::calcLength()
{
double length = 0.0f;
for (sint j=((sint)_Coords.size())-1; j>0; j--)
length += (_Coords[j] - _Coords[j-1]).toAIVector().norm();
_Length = (float)length;
calCost();
}
inline
void CRoad::setDifficulty(float const& difficulty)
{
_Difficulty = difficulty;
calCost();
}
inline
void CRoad::calCost()
{
_CostCoef = _Length*_Difficulty;
}
inline
float CRoad::getCost() const
{
return _CostCoef*getRefCount(); // takes account of use.
}
//////////////////////////////////////////////////////////////////////////////
// CRoadTrigger //
//////////////////////////////////////////////////////////////////////////////
inline
CRoadTrigger::CRoadTrigger(CRoad* owner, uint32 alias, std::string const& name)
: CAliasChild<CRoad>(owner, alias, name)
{
}
//////////////////////////////////////////////////////////////////////////////
// CCell //
//////////////////////////////////////////////////////////////////////////////
inline
size_t CCell::npcZoneCount()
{
return _NpcZonePlaces.size() + _NpcZoneShapes.size();
}
inline
CNpcZone* CCell::npcZone(size_t index)
{
if (index<_NpcZonePlaces.size())
return _NpcZonePlaces[(uint32)index];
else
return _NpcZoneShapes[(uint32)index];
}
inline
void CCell::getNeighBourgCellList(std::vector<CCell*>& cells) const
{
cells.reserve(_NeighbourCells.size()+1);
// build a list of candidate cell to look for the rest zone : ether the current zone or a beighbour
std::set<NLMISC::CDbgPtr<CCell> >::const_iterator first(_NeighbourCells.begin()), last(_NeighbourCells.end());
for (; first != last; ++first)
cells.push_back(*first);
}
//////////////////////////////////////////////////////////////////////////////
// CGroupFamily //
//////////////////////////////////////////////////////////////////////////////
inline
CGroupFamily::~CGroupFamily()
{
_GroupDescs.clear();
}
inline
void CGroupFamily::setProfileParams(std::string const& str, NLMISC::CVirtualRefCount* objet)
{
#if !FINAL_VERSION
nlassert(objet);
#endif
_Params.insert(std::make_pair(NLMISC::CStringMapper::map(str),objet));
}
inline
NLMISC::CVirtualRefCount const* CGroupFamily::getProfileParams(std::string const& str) const
{
TParamsList::const_iterator const it = _Params.find(NLMISC::CStringMapper::map(str));
if (it==_Params.end())
return NULL;
return it->second;
}
inline
void CGroupFamily::addProfileProperty(std::string const& propertyName, AITYPES::CPropertySet const& property)
{
_Properties[NLMISC::CStringMapper::map(propertyName)] = property;
}
inline
AITYPES::CPropertySet const& CGroupFamily::getProfileProperty(std::string const& propertyName) const
{
static AITYPES::CPropertySet emptyActivities;
TActivityList::const_iterator it = _Properties.find(NLMISC::CStringMapper::map(propertyName));
if (it==_Properties.end())
return emptyActivities;
return it->second;
}
//////////////////////////////////////////////////////////////////////////////
// CGroupDesc //
//////////////////////////////////////////////////////////////////////////////
template <class FamilyT>
CGroupDesc<FamilyT>::CGroupDesc(FamilyT* owner, uint32 alias, std::string const& name)
: CAliasChild<FamilyT>(owner, alias, name)
, _BotCount(0)
, _CountMultiplier(false)
, _GroupEnergyValue((uint32)(0.01*AITYPES::ENERGY_SCALE))
, _SpawnType(AITYPES::SpawnTypeAlways)
, _EnergyCoef(1.f)
, _MultiLevel(false)
, _Sheet(NULL)
, _MultiLevelSheets(_MultiLevelSheetCount)
, _LevelDelta(0)
, _PlayerAttackable(true)
, _BotAttackable(true)
{
for (size_t j=0; j<_MultiLevelSheetCount; ++j)
_MultiLevelSheets[j] = NULL;
for (uint32 i=0; i<4; ++i)
{
_SeasonFlags[i] = false;
_WeightLevel[i] = 0;
}
}
template <class FamilyT>
bool CGroupDesc<FamilyT>::isValidForDayOrNight(bool const& isDay) const
{
if (_SpawnType == AITYPES::SpawnTypeAlways)
return true;
return isDay?(_SpawnType == AITYPES::SpawnTypeDay):(_SpawnType == AITYPES::SpawnTypeNight);
}
template <class FamilyT>
void CGroupDesc<FamilyT>::setSheet(AISHEETS::ICreatureCPtr const& sheetPtr)
{
#ifdef NL_DEBUG
nlassert(sheetPtr);
#endif
_Sheet = sheetPtr;
}
template <class FamilyT>
bool CGroupDesc<FamilyT>::setSheet(std::string const& sheetName)
{
if (!sheetName.empty())
{
if (_MultiLevel)
{
for (size_t i=0; i<_MultiLevelSheetCount; ++i)
{
char letter = char(i/4) + 'b';
char number = (i%4) + '1';
std::string sheetNameLevel = sheetName+letter+number;
// Compute sheet id
NLMISC::CSheetId sheetId(sheetNameLevel+".creature");
// Find the sheet
AISHEETS::ICreatureCPtr const sheet = AISHEETS::CSheets::getInstance()->lookup(sheetId);
// If the sheet doesn't exist
if (sheetId==NLMISC::CSheetId::Unknown || !sheet)
{
nlwarning("Sheet '%s' for group '%s'%s is unknown !",
sheetNameLevel.c_str(),
this->getFullName().c_str(),
this->getAliasString().c_str());
return false;
}
_MultiLevelSheets[i] = sheet;
}
}
else
{
// Compute sheet id
NLMISC::CSheetId sheetId(sheetName+".creature");
// Find the sheet
AISHEETS::ICreatureCPtr const sheet = AISHEETS::CSheets::getInstance()->lookup(sheetId);
// If the sheet doesn't exist
if (sheetId==NLMISC::CSheetId::Unknown || !sheet)
{
nlwarning("Sheet '%s' for group '%s'%s is unknown !",
sheetName.c_str(),
this->getFullName().c_str(),
this->getAliasString().c_str());
return false;
}
_Sheet=sheet;
}
}
return true;
}
template <class FamilyT>
AISHEETS::ICreatureCPtr CGroupDesc<FamilyT>::sheet(sint32 baseLevel) const
{
if (_MultiLevel && baseLevel!=-1)
{
sint32 level = baseLevel + getLevelDelta();
// Clamp to [0;_MultiLevelSheetCount]
level = std::min(level, (sint32)(_MultiLevelSheetCount-1));
level = std::max(level, (sint32)0);
return _MultiLevelSheets[level];
}
else
return _Sheet;
}
template <class FamilyT>
uint32 CGroupDesc<FamilyT>::getRealBotCount() const
{
if (_Sheet && getCountMultiplierFlag() && !_MultiLevel)
return _BotCount * _Sheet->DynamicGroupCountMultiplier();
else
return _BotCount;
}
template <class FamilyT>
void CGroupDesc<FamilyT>::setSeasonFlags(bool const seasonFlags[4])
{
for (uint32 i=0;i<4;i++)
_SeasonFlags[i] = seasonFlags[i];
}
template <class FamilyT>
void CGroupDesc<FamilyT>::setWeightLevels(uint32 const weights[4])
{
for (uint32 i=0;i<4;i++)
_WeightLevel[i] = weights[i];
}
template <class FamilyT>
CGrpFauna* CGroupDesc<FamilyT>::createFaunaGroup(CFamilyBehavior* familyBehavior) const
{
H_AUTO(createFaunaGroup)
uint32 energyLevel = familyBehavior->effectiveLevel();
AISHEETS::ICreatureCPtr const ls = sheet();
const RYAI_MAP_CRUNCH::TAStarFlag AStarFlag=RYAI_MAP_CRUNCH::WaterAndNogo;
if (!ls)
{
#if !FINAL_VERSION
nlwarning("CRegion::createGroup Can't retreive creature info for sheet '%s' from group desc '%s'%s to spawn in region '%s'%s",
(sheet()?sheet()->SheetId().toString().c_str():NLMISC::CSheetId::Unknown.toString().c_str()),
this->getAliasFullName().c_str(),
this->getAliasString().c_str(),
this->getOwner()->getAliasFullName().c_str(),
this->getOwner()->getAliasString().c_str()); // CaracSheet
#endif
return NULL;
}
AITYPES::CPropertySet food, rest;
familyBehavior->getActivities (food, rest);
/*
nlinfo("Num food activities = %d", (int) food.size());
nlinfo("Num rest activities = %d", (int) rest.size());
*/
const CFaunaZone *faunaZone=familyBehavior->getOwner()->lookupFaunaZone(rest, AStarFlag, familyBehavior->grpFamily()->getSubstitutionId());
if (!faunaZone)
return NULL;
const CFaunaZone *fzFood=NULL;
const CFaunaZone *fzRest=NULL;
// select a random spawn zone into the vector
{
std::vector<CCell*> cells;
faunaZone->getOwner()->getNeighBourgCellList(cells);
cells.push_back(faunaZone->getOwner());
if (!familyBehavior->getOwner()->findRestAndFoodFaunaZoneInCellList(fzRest, rest, fzFood, food, cells, AStarFlag))
{
#if !FINAL_VERSION
nlwarning("CRegion::createGroup can't find zone pair with properties food: <%s> rest: <%s>, for family %s, group '%s'%s",
food.toString().c_str(),
rest.toString().c_str(),
familyBehavior->getName().c_str(),
this->getAliasFullName().c_str(),
this->getAliasString().c_str());
#endif
return NULL;
}
if ( !fzFood
|| !fzRest)
{
#if !FINAL_VERSION
nlwarning("CRegion::createGroup can't find zone pair with properties food: <%s> rest: <%s>, for family %s, group '%s'%s",
food.toString().c_str(),
rest.toString().c_str(),
familyBehavior->getName().c_str(),
this->getAliasFullName().c_str(),
this->getAliasString().c_str());
#endif
return NULL;
}
}
if ( !fzRest
&& !rest.empty() )
{
#if !FINAL_VERSION
nlwarning("Fauna dynamic: CRegion::createGroup can't find fauna zone to rest '%s' in region '%s'%s",
familyBehavior->getName().c_str(),
this->getAliasFullName().c_str(),
this->getAliasString().c_str());
#endif
return NULL;
}
CMgrFauna *mf = familyBehavior->mgrFauna();
CGrpFauna *grp = new CGrpFauna(mf, NULL, AStarFlag);
mf->groups().addAliasChild(grp);
grp->initDynGrp (this, familyBehavior);
/// Fill the fauna group data
CAIRefPlaceXYRFauna *refPlace = new CAIRefPlaceXYRFauna(grp,fzRest);
refPlace->setupFromOldName("spawn");
grp->places().addChild (refPlace, CGrpFauna::SPAWN_PLACE );
refPlace = new CAIRefPlaceXYRFauna(grp,fzFood);
refPlace->setupFromOldName("food");
grp->places().addChild (refPlace, CGrpFauna::EAT_PLACE );
refPlace = new CAIRefPlaceXYRFauna(grp,fzRest);
refPlace->setupFromOldName("rest");
grp->places().addChild (refPlace, CGrpFauna::REST_PLACE );
if ( !grp->places()[CGrpFauna::SPAWN_PLACE]->worldValidPos().isValid()
|| !grp->places()[CGrpFauna::EAT_PLACE]->worldValidPos().isValid()
|| !grp->places()[CGrpFauna::REST_PLACE]->worldValidPos().isValid() )
{
if (LogGroupCreationFailure)
{
#if !FINAL_VERSION
nlwarning("Invalid place to spawn dynamic group '%s'%s",
this->getAliasFullName().c_str(),
this->getAliasString().c_str());
if (!grp->places()[CGrpFauna::SPAWN_PLACE]->worldValidPos().isValid())
nlwarning(" Invalid spawn place %s", grp->places()[CGrpFauna::SPAWN_PLACE]->midPos().toString().c_str());
if (!grp->places()[CGrpFauna::EAT_PLACE]->worldValidPos().isValid())
nlwarning(" Invalid eat place %s", grp->places()[CGrpFauna::EAT_PLACE]->midPos().toString().c_str());
if (!grp->places()[CGrpFauna::REST_PLACE]->worldValidPos().isValid())
nlwarning(" Invalid rest place %s", grp->places()[CGrpFauna::REST_PLACE]->midPos().toString().c_str());
#endif
}
mf->groups().removeChildByIndex(grp->getChildIndex());
return NULL;
}
CPopulation *pop = new CPopulation(grp);
{
pop->setWeight(1);
pop->setSpawnType(spawnType());
pop->addPopRecord(CPopulationRecord(ls, getRealBotCount()));
for (size_t i=0; i<_PopulationRecords.size(); ++i)
pop->addPopRecord(_PopulationRecords[i]);
}
grp->populations().addAliasChild(pop);
grp->setAutoSpawn(false);
grp->spawn();
if (!grp->getSpawnObj())
{
// the spawning has failed, delete the useless object
#if !FINAL_VERSION
if (!grp->getSpawnCounter().remainToMax())
nldebug("Cannot spawn the dynamic group: maximum reached");
else
nlwarning("Failed to spawn the dynamic group");
#endif
mf->groups().removeChildByIndex(grp->getChildIndex());
return NULL;
}
return grp;
}
inline
static CAIVector randomPos(double dispersionRadius)
{
if (dispersionRadius<=0.)
{
return CAIVector(0., 0.);
}
uint32 const maxLimit=((uint32)~0U)>>1;
double rval = (double)CAIS::rand32(maxLimit)/(double)maxLimit; // [0-1[
double r = dispersionRadius*sqrt(rval);
rval = (double)CAIS::rand32(maxLimit)/(double)maxLimit; // [0-1[
double t = 2.0*NLMISC::Pi*rval;
double dx = cos(t)*r;
double dy = sin(t)*r;
return CAIVector(dx, dy);
}
template <class FamilyT>
CGroupNpc* CGroupDesc<FamilyT>::createNpcGroup(CMgrNpc* mgr, CAIVector const& pos, double dispersionRadius, sint32 baseLevel, bool spawnBots) const
{
H_AUTO(createNpcGroup)
// Keep base level positive or -1 (single level)
baseLevel = std::max(baseLevel, (sint32)-1);
// Verify we have all the sheets
// :TODO: Add verification for named bots sheets
if (!sheet(baseLevel) && getBaseBotCount()>0)
{
nlwarning("CGroupDesc::createNpcGroup can't retrieve sheet from group '%s'%s in region '%s'%s", this->getAliasFullName().c_str(), this->getAliasString().c_str(), this->getOwner()->getAliasFullName().c_str(), this->getOwner()->getAliasString().c_str());
return NULL;
}
FOREACHC (itBotDesc, typename CCont<CBotDesc<FamilyT> >, botDescs())
{
if (!itBotDesc->sheet(baseLevel))
{
nlwarning("CGroupDesc::createNpcGroup can't retrieve sheet from bot '%s'%s in group '%s'%s in region '%s'%s", itBotDesc->getAliasFullName().c_str(), itBotDesc->getAliasString().c_str(), this->getAliasFullName().c_str(), this->getAliasString().c_str(), this->getOwner()->getAliasFullName().c_str(), this->getOwner()->getAliasString().c_str());
return NULL;
}
}
// Create a group
CGroupNpc *grp = new CGroupNpc(mgr, NULL, /*AStarFlag*/RYAI_MAP_CRUNCH::Nothing);
// Register it in the manager
mgr->groups().addAliasChild(grp);
// Set the group parameters
grp->setAutoSpawn(false);
grp->setName(this->getName());
grp->clearParameters();
for (uint i=0; i<grpParameters().size(); ++i)
grp->addParameter(grpParameters()[i]);
grp->setPlayerAttackable(_PlayerAttackable);
grp->setBotAttackable(_BotAttackable);
// Save whether we have named or unnamed bots
if (getRealBotCount() == 0)
grp->setBotsAreNamedFlag();
else
grp->clrBotsAreNamedFlag();
{
uint i=0;
// build the specific bots data
for (; i<botDescs().size(); ++i)
{
const CBotDesc<FamilyT> *const bd = botDescs()[i];
uint nbClone = 1;
if (getCountMultiplierFlag())
{
// the group use the multiplier from the creature sheet
nbClone *= bd->sheet(baseLevel)->DynamicGroupCountMultiplier();
}
// loop for the requested clones
for (uint j=0; j<nbClone; ++j)
{
grp->bots().addChild(new CBotNpc(grp, 0, bd->getBotName()), i); // Doub: 0 instead of bd->getAlias() otherwise all bots will have the same non-zero alias
CBotNpc *const bot = static_cast<CBotNpc*>(grp->bots()[i]);
bot->setSheet (bd->sheet(baseLevel));
bot->equipmentInit ();
bot->initEnergy (groupEnergyCoef());
CAIVector rpos(pos);
if (i!=0)
{
RYAI_MAP_CRUNCH::CWorldMap const& worldMap = CWorldContainer::getWorldMap();
RYAI_MAP_CRUNCH::CWorldPosition wp;
uint32 maxTries = 100;
do
{
rpos = pos;
rpos += randomPos(dispersionRadius);
--maxTries;
}
while (!worldMap.setWorldPosition(AITYPES::vp_auto, wp, rpos) && maxTries>0);
if (maxTries<=0)
rpos = pos;
}
bot->setStartPos (rpos.x().asDouble(),rpos.y().asDouble(), 0, AITYPES::vp_auto);
}
}
// build un-named bot
uint nbClone = getRealBotCount();
for (uint j=0; j<nbClone; ++i,++j)
{
grp->bots().addChild(new CBotNpc(grp, 0, grp->getName()), i); // Doub: 0 instead of getAlias()+i otherwise aliases are wrong
CBotNpc *const bot = NLMISC::safe_cast<CBotNpc*>(grp->bots()[i]);
bot->setSheet (sheet(baseLevel));
bot->equipmentInit ();
bot->initEnergy (groupEnergyCoef());
CAIVector rpos(pos);
if (i!=0)
{
RYAI_MAP_CRUNCH::CWorldMap const& worldMap = CWorldContainer::getWorldMap();
RYAI_MAP_CRUNCH::CWorldPosition wp;
uint32 maxTries = 100;
do
{
rpos = pos;
rpos += randomPos(dispersionRadius);
--maxTries;
}
while (!worldMap.setWorldPosition(AITYPES::vp_auto, wp, rpos) && maxTries>0);
if (maxTries<=0)
rpos = pos;
}
bot->setStartPos (rpos.x().asDouble(),rpos.y().asDouble(), 0, AITYPES::vp_auto);
}
}
grp->spawn();
if (!grp->getSpawnObj())
{
// the spawning has failed, delete the useless object
nlwarning("Failed to spawn the dynamic group");
mgr->groups().removeChildByIndex(grp->getChildIndex());
return NULL;
}
if (spawnBots)
grp->getSpawnObj()->spawnBots();
return grp;
}
template <class FamilyT>
uint32 CGroupDesc<FamilyT>::calcTotalEnergyValue () const
{
uint32 totalEnergyValue=0;
{
typename CCont<CBotDesc<FamilyT> >::const_iterator it=botDescs().begin(), itEnd=botDescs().end();
// add specified bots ..
while (it!=itEnd)
{
totalEnergyValue+=it->energyValue();
++it;
}
}
// add botcount ..
if (sheet())
totalEnergyValue += getRealBotCount()*sheet()->EnergyValue();
{
std::vector<CPopulationRecord>::const_iterator it=_PopulationRecords.begin(), itEnd=_PopulationRecords.end();
while (it!=itEnd)
{
const CPopulationRecord &pr = *it;
totalEnergyValue += pr.getEnergyValue(getCountMultiplierFlag());
++it;
}
}
return totalEnergyValue;
}
template <class FamilyT>
IAliasCont *CGroupDesc<FamilyT>::getAliasCont(AITYPES::TAIType type)
{
switch(type)
{
case AITYPES::AITypeSquadTemplateMember:
case AITYPES::AITypeBotTemplate:
case AITYPES::AITypeBotTemplateMultiLevel:
return &_BotDescs;
default:
return NULL;
}
}
template <class FamilyT>
sint CGroupDesc<FamilyT>::getNbUse() const
{
return getRefCount()-1; // less one because its also referenced by aliascont.
}
template <class FamilyT>
CAliasTreeOwner *CGroupDesc<FamilyT>::createChild(IAliasCont *cont, CAIAliasDescriptionNode *aliasTree)
{
if (!cont)
return NULL;
CAliasTreeOwner* child = NULL;
switch(aliasTree->getType())
{
// create the child and adds it to the corresponding position.
case AITYPES::AITypeSquadTemplateMember:
case AITYPES::AITypeBotTemplate:
case AITYPES::AITypeBotTemplateMultiLevel:
child = new CBotDesc<FamilyT>(this, aliasTree->getAlias(), aliasTree->getName());
break;
default:
break;
}
if (child)
cont->addAliasChild(child);
return child;
}
template <class FamilyT>
std::string CGroupDesc<FamilyT>::getIndexString() const
{
return this->getOwner()->getIndexString()+NLMISC::toString(":%u", this->getChildIndex());
}
//////////////////////////////////////////////////////////////////////////////
// ContextGroupDesc actions //
//////////////////////////////////////////////////////////////////////////////
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_SHEE,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
std::string lookSheet;
if (!getArgs(args,name(), lookSheet))
return;
if (!groupDesc->setSheet(lookSheet))
{
groupDesc->getOwner()->groupDescs().removeChildByIndex(groupDesc->getChildIndex());
CWorkPtr::groupDesc(NULL);
return;
}
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_LVLD,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
sint32 levelDelta;
if (!getArgs(args,name(), levelDelta))
return;
groupDesc->setLevelDelta(levelDelta);
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_SEAS,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
bool seasons[4];
if (!getArgs(args,name(), seasons[0], seasons[1], seasons[2], seasons[3]))
return;
groupDesc->setSeasonFlags(seasons);
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_ACT,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
uint32 spawnType;
if (!getArgs(args, name(), spawnType))
return;
groupDesc->setSpawnType((AITYPES::TSpawnType)spawnType);
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_APRM,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
for (size_t i=0; i<args.size(); ++i)
{
std::string property;
args[i].get(property);
groupDesc->properties().addProperty(AITYPES::CPropertyId::create(property));
}
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_NRG,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
uint32 weight[4];
if (!getArgs(args,name(), weight[0], weight[1], weight[2], weight[3]))
return;
groupDesc->setWeightLevels(weight);
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_EQUI,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
groupDesc->botEquipment().clear();
for (size_t i=0; i<args.size(); ++i)
{
std::string equip;
args[i].get(equip);
groupDesc->botEquipment().push_back(equip);
}
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_GPRM,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
for (size_t i=0; i<args.size(); ++i)
{
std::string param;
args[i].get(param);
param = NLMISC::toLower(param);
if ( param == "contact camp"
|| param == "contact outpost"
|| param == "contact city"
|| param == "boss" )
groupDesc->properties().addProperty(param);
else // unreconized param, leace it for the group instance
groupDesc->grpParameters().push_back(param);
}
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,BOTTMPL,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
std::string lookSheet;
bool multiLevel;
// read the alias tree from the argument list
CAIAliasDescriptionNode* aliasTree;
if (!getArgs(args, name(), aliasTree, lookSheet, multiLevel))
return;
// see whether the region is already loaded
CBotDesc<FamilyT>* botDesc = groupDesc->botDescs().getChildByAlias(aliasTree->getAlias());
if (!botDesc)
return;
botDesc->setMultiLevel(multiLevel);
botDesc->setSheet(lookSheet);
CWorkPtr::botDesc(botDesc);
CContextStack::setContext(CAISActionEnums::ContextBotDesc);
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextBotDesc,BT_EQUI,FamilyT)
{
CBotDesc<FamilyT>* botDesc = static_cast<CBotDesc<FamilyT>*>(CWorkPtr::botDesc());
if (!botDesc)
return;
for (size_t i=0; i<args.size(); ++i)
{
std::string equip;
args[i].get(equip);
botDesc->equipement().push_back(equip);
}
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextBotDesc,BT_LVLD,FamilyT)
{
CBotDesc<FamilyT>* botDesc = static_cast<CBotDesc<FamilyT>*>(CWorkPtr::botDesc());
if (!botDesc)
return;
sint32 levelDelta;
if (!getArgs(args,name(), levelDelta))
return;
botDesc->setLevelDelta(levelDelta);
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_GNRJ,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
uint32 energyValue;
if (!getArgs(args,name(), energyValue))
return;
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,POPVER,FamilyT)
{
// add a population version for a group
// args: uint32 alias, string spawn_type, uint weight, (string sheet, uint32 count)+
if(!CWorkPtr::groupDesc())
return;
const uint32 fixedArgsCount = 0;
if (args.size()<fixedArgsCount+2 || ((args.size()-fixedArgsCount)&1)==1)
{
nlwarning("POPVER action FAILED due to bad number of arguments (%d)", args.size());
return;
}
// get hold of the parameters and check their validity
for (size_t i=fixedArgsCount; i+1<args.size(); i+=2)
{
std::string sheet;
uint32 count;
if ( !args[i].get(sheet)
|| !args[i+1].get(count))
{
nlwarning("POPVER Add Record FAILED due to bad arguments");
continue;
}
NLMISC::CSheetId sheetId(sheet);
if (sheetId==NLMISC::CSheetId::Unknown)
{
nlwarning("POPVER Add Record Invalid sheet: %s", sheet.c_str());
continue;
}
AISHEETS::ICreatureCPtr sheetPtr = AISHEETS::CSheets::getInstance()->lookup(sheetId);
if (!sheetPtr)
{
nlwarning("POPVER Add Record Invalid sheet: %s", sheet.c_str());
continue;
}
static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc())->populationRecords().push_back(CPopulationRecord(sheetPtr, count));
}
}
/// :KLUDGE: This code is copied in ai_outpost_actions.h. Update both if you
/// make a modification.
// scales bot energy .. to match with group's one.
DEFINE_ACTION_TEMPLATE1(ContextGroupDesc,GT_END,FamilyT)
{
CGroupDesc<FamilyT>* groupDesc = static_cast<CGroupDesc<FamilyT>*>(CWorkPtr::groupDesc());
if (!groupDesc)
return;
if (!groupDesc->isMultiLevel())
{
uint32 totalEnergyValue = groupDesc->calcTotalEnergyValue();
if (totalEnergyValue)
{
double coef = (double)groupDesc->groupEnergyValue()/(double)totalEnergyValue;
groupDesc->setGroupEnergyCoef((float)coef);
}
else
{
nlwarning("Retrieved total energy value of 0 for group: %s",groupDesc->getFullName().c_str());
}
}
}
//////////////////////////////////////////////////////////////////////////////
// CBotDesc //
//////////////////////////////////////////////////////////////////////////////
template <class FamilyT>
CBotDesc<FamilyT>::CBotDesc(CGroupDesc<FamilyT>* owner, uint32 alias, std::string const& name)
: CAliasChild<CGroupDesc<FamilyT> >(owner, alias, name)
, _MultiLevel(false)
, _Sheet(NULL)
, _MultiLevelSheets(_MultiLevelSheetCount)
, _LevelDelta(0)
, _UseSheetBotName(false)
{
for (size_t i=0; i<_MultiLevelSheetCount; ++i)
_MultiLevelSheets[i] = NULL;
}
template <class FamilyT>
std::string CBotDesc<FamilyT>::getIndexString() const
{
return this->getOwner()->getIndexString() + NLMISC::toString(":%u", this->getChildIndex());
}
template <class FamilyT>
void CBotDesc<FamilyT>::setSheet(std::string const& sheetName)
{
if (!sheetName.empty())
{
if (_MultiLevel)
{
for (size_t i=0; i<_MultiLevelSheetCount; ++i)
{
char letter = char(i/4) + 'b';
char number = (i%4) + '1';
std::string sheetNameLevel = sheetName+letter+number;
// Compute sheet id
NLMISC::CSheetId sheetId(sheetNameLevel+".creature");
// Find the sheet
AISHEETS::ICreatureCPtr const sheet = AISHEETS::CSheets::getInstance()->lookup(sheetId);
// If the sheet doesn't exist
if (sheetId==NLMISC::CSheetId::Unknown || !sheet)
{
nlwarning("Sheet '%s' for bot '%s' is unknown !", sheetNameLevel.c_str(), this->getAliasFullName().c_str());
}
_MultiLevelSheets[i] = sheet;
}
}
else
{
// Compute sheet id
NLMISC::CSheetId sheetId(sheetName+".creature");
// Find the sheet
AISHEETS::ICreatureCPtr const sheet = AISHEETS::CSheets::getInstance()->lookup(sheetId);
// If the sheet doesn't exist
if (sheetId==NLMISC::CSheetId::Unknown || !sheet)
{
nlwarning("Sheet '%s' for bot '%s' is unknown !", sheetName.c_str(), this->getAliasFullName().c_str());
}
_Sheet = sheet;
}
}
}
template <class FamilyT>
AISHEETS::ICreatureCPtr CBotDesc<FamilyT>::sheet(sint32 baseLevel) const
{
if (_MultiLevel && baseLevel!=-1)
{
CGroupDesc<FamilyT>* parent = this->getOwner();
sint32 level = baseLevel + getLevelDelta() + parent->getLevelDelta();
// Clamp to [0;_MultiLevelSheetCount]
level = std::min(level, (sint32)(_MultiLevelSheetCount-1));
level = std::max(level, (sint32)0);
return _MultiLevelSheets[level];
}
else
return _Sheet;
}
template <class FamilyT>
uint32 CBotDesc<FamilyT>::energyValue() const
{
if (!_Sheet && !_MultiLevel)
nlwarning("Bot descriptor has no sheet and is not multilevel, correct above warnings!");
if (_Sheet)
return _Sheet->EnergyValue() * _Sheet->DynamicGroupCountMultiplier();
return 0;
}
template <class FamilyT>
std::string const& CBotDesc<FamilyT>::getBotName() const
{
if (_UseSheetBotName && _Sheet && !_Sheet->BotName().empty())
return _Sheet->BotName();
else
return this->getName();
}
//////////////////////////////////////////////////////////////////////////////
// CCell //
//////////////////////////////////////////////////////////////////////////////
inline
void CCell::unrefZoneInRoads()
{
FOREACH(it, TAliasZonePlaceList, _NpcZonePlaces)
it->unrefZoneInRoads ();
FOREACH(it, TAliasZoneShapeList, _NpcZoneShapes)
it->unrefZoneInRoads ();
_NeighbourCells.clear ();
}
//////////////////////////////////////////////////////////////////////////////
// CNpcZone //
//////////////////////////////////////////////////////////////////////////////
// :TODO: check if that inlining is necessary
inline
void CNpcZone::unrefZoneInRoads()
{
while (!_Roads.empty())
_Roads.back()->unlinkRoad();
}
//////////////////////////////////////////////////////////////////////////////
// CCellZone //
//////////////////////////////////////////////////////////////////////////////
inline
void CCellZone::unrefZoneInRoads()
{
FOREACH(it, CCont<CCell>, _Cells)
it->unrefZoneInRoads();
}
#endif