// 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"
// Net
#include "nel/net/service.h"
// Georges
#include "nel/georges/u_form.h"
#include "nel/georges/u_form_elm.h"
#include "nel/georges/u_form_loader.h"
#include "nel/georges/load_form.h"
#include "sheets.h"
#include "nel/misc/o_xml.h"
using namespace MULTI_LINE_FORMATER;
///////////
// USING //
///////////
using namespace NLGEORGES;
using namespace NLMISC;
using namespace NLNET;
using namespace std;
using namespace AITYPES;
//////////////////////////////////////////////////////////////////////////////
// Constants //
//////////////////////////////////////////////////////////////////////////////
#ifdef NL_DEBUG
CVariable debugSheet("ai", "debugSheet", "The sheet to break onto", "", 0, true);
#endif
char const* AISPackedSheetsFilename="ais.packed_sheets";
char const* AISPackedFightConfigSheetsFilename="ais_fight_config.packed_sheets";
char const* AISPackedActionSheetsFilename="ais_action.packed_sheets";
char const* AISPackedRaceStatsSheetsFilename="ais_race_stats.packed_sheets";
static AISHEETS::CCreature EmptySheet;
sint32 AISHEETS::ICreature::InvalidFameForGuardAttack = 0x7FFFFFFF;
//////////////////////////////////////////////////////////////////////////////
// CAIAction //
//////////////////////////////////////////////////////////////////////////////
AISHEETS::CAIAction::CAIAction()
: _SelfAction(false)
{
}
void AISHEETS::CAIAction::readGeorges(NLMISC::CSmartPtr const& form, NLMISC::CSheetId const& sheetId)
{
NLGEORGES::UFormElm const& item = form->getRootNode();
// the form was found so read the true values from George
_SheetId = sheetId;
item.getValueByName(_SelfAction, "SelfAction");
}
uint AISHEETS::CAIAction::getVersion()
{
return 2;
}
void AISHEETS::CAIAction::serial(NLMISC::IStream& s)
{
s.serial(_SheetId);
s.serial(_SelfAction);
}
std::vector AISHEETS::CAIAction::getMultiLineInfoString() const
{
std::vector container;
pushTitle(container, "AISHEETS::CAIAction");
pushEntry(container, "ai_action sheet");
pushFooter(container);
return container;
}
//////////////////////////////////////////////////////////////////////////////
// CActionList //
//////////////////////////////////////////////////////////////////////////////
void AISHEETS::CActionList::computeAbilities()
{
_HasNormalAction = false;
_HasSelfAction = false;
FOREACH(itAction, std::vector, _Actions)
{
IAIActionCPtr const& action = *itAction;
if (!action.isNull() && action->SelfAction())
_HasSelfAction = true;
else
_HasNormalAction = true;
}
}
void AISHEETS::CActionList::readGeorges(NLMISC::CSmartPtr const& form, NLMISC::CSheetId const& sheetId)
{
NLGEORGES::UFormElm const& item = form->getRootNode();
// the form was found so read the true values from George
_SheetId = sheetId;
{
NLGEORGES::UFormElm const* actionListNode = NULL;
item.getNodeByName(&actionListNode, "actions");
if (actionListNode)
{
uint arraySize = 0;
actionListNode->getArraySize(arraySize);
for (uint i=0; igetArrayValue(action, i);
if (!action.empty())
{
addAction(NLMISC::CSheetId(action), action);
}
}
}
}
computeAbilities();
}
uint AISHEETS::CActionList::getVersion()
{
return 4;
}
void AISHEETS::CActionList::serial(NLMISC::IStream& s)
{
s.serial(_SheetId);
if (s.isReading())
{
uint32 nbSheet;
s.serial(nbSheet);
for (uint32 i=0; iSheetId();
s.serial(sheetId);
}
}
computeAbilities();
}
void AISHEETS::CActionList::addAction(NLMISC::CSheetId const& sheetId, std::string const& actionName)
{
IAIActionCPtr action = CSheets::getInstance()->lookupAction(sheetId);
if (!action.isNull())
{
_Actions.push_back(action);
}
else
{
if (!actionName.empty())
nlwarning("action %s doesnt exist", actionName.c_str());
}
}
std::vector AISHEETS::CActionList::getMultiLineInfoString() const
{
std::vector container;
pushTitle(container, "AISHEETS::CActionList");
pushEntry(container, "action_list sheet");
pushFooter(container);
return container;
}
//////////////////////////////////////////////////////////////////////////////
// CGroupProperties //
//////////////////////////////////////////////////////////////////////////////
AISHEETS::CGroupProperties::CGroupProperties()
: _Assist(false)
, _Attack(false)
{
}
//////////////////////////////////////////////////////////////////////////////
// CCreature //
//////////////////////////////////////////////////////////////////////////////
AISHEETS::CCreature::CCreature()
: _Level(1)
, _Radius(0.5f), _Height(2.0f), _Width(1.0f), _Length(1.0f)
, _BoundingRadius(0.5)
, _BonusAggroHungry(0.0), _BonusAggroVeryHungry(0.0)
, _AssistDist(10)
, _MinFightDist(0)
, _FactionIndex(CStaticFames::INVALID_FACTION_INDEX)
, _FameForGuardAttack(ICreature::InvalidFameForGuardAttack)
, _GroupPropertiesIndex(0)
, _DynamicGroupCountMultiplier(1)
{
}
void AISHEETS::CCreature::calcFightAndVisualValues(std::string* left, std::string* right)
{
CVisualSlotManager* visualSlotManager = CVisualSlotManager::getInstance();
#ifdef NL_DEBUG
nlassert(visualSlotManager);
#endif
uint32 leftAsInt = visualSlotManager->leftItem2Index(LeftItem());
uint32 rightAsInt = visualSlotManager->rightItem2Index(RightItem());
if (LeftItem()!=NLMISC::CSheetId::Unknown && leftAsInt==0 && left!=NULL)
{
if (left->size() < 3 || left->at(3) != 'p')
nlwarning("Left item '%s' not allowed, if ammo, this is normal", left->c_str());
}
if (RightItem()!=NLMISC::CSheetId::Unknown && rightAsInt==0 && right!=NULL)
{
if (right->size() < 3 || right->at(3) != 'p')
nlwarning("Right item '%s' not allowed, if ammo, this is normal", right->c_str());
}
_MinFightDist = (!FightConfig(FIGHTCFG_RANGE).isNULL()||!FightConfig(FIGHTCFG_NUKE).isNULL())?20:0;
}
void AISHEETS::CCreature::parseFightConfig(NLGEORGES::UForm const* form, std::string const& fightConfigString, uint32 actionListIndex, NLMISC::CDbgPtr& fightConfig)
{
NLGEORGES::UFormElm const* actionListNode = NULL;
const_cast(form->getRootNode()).getNodeByName(&actionListNode, fightConfigString.c_str());
if (actionListNode)
{
uint arraySize = 0;
actionListNode->getArraySize(arraySize);
if (actionListIndexgetArrayValue(actionListFileName,actionListIndex);
addActionConfig (actionListFileName, fightConfig);
}
}
}
void AISHEETS::CCreature::readFightConfig(NLMISC::IStream& s, NLMISC::CDbgPtr& fightConfig)
{
NLMISC::CSheetId sheetId;
s.serial(sheetId);
if (sheetId!=NLMISC::CSheetId::Unknown)
addActionConfig(sheetId, fightConfig);
}
void AISHEETS::CCreature::saveFightConfig(NLMISC::IStream& s, NLMISC::CDbgPtr& fightConfig)
{
if (!fightConfig.isNULL())
{
s.serial(fightConfig->_SheetId);
}
else
{
NLMISC::CSheetId id;
s.serial(id);
}
}
bool AISHEETS::CCreature::mustAssist(CCreature const& creature) const
{
return getPropertiesCst(creature.GroupPropertiesIndex()).assist();
}
void AISHEETS::CCreature::setAssisGroupIndexs()
{
_GroupPropertiesIndex = CSheets::getInstance()->getGroupPropertiesIndex(GroupIndexStr());
if (_GroupPropertiesIndex==~0)
return;
std::vector groupList;
getGroupStr(groupList, AssistGroupIndexStr());
FOREACH(it, std::vector, groupList)
getProperties(*it).setAssist(true);
}
void AISHEETS::CCreature::setAttackGroupIndexs()
{
_GroupPropertiesIndex = CSheets::getInstance()->getGroupPropertiesIndex(GroupIndexStr());
if (_GroupPropertiesIndex==~0)
return;
std::vector groupList;
getGroupStr(groupList, AttackGroupIndexStr());
FOREACH(it, std::vector, groupList)
getProperties(*it).setAttack(true);
}
void AISHEETS::CCreature::addActionConfig(std::string const& sheetIdName, NLMISC::CDbgPtr& actionConfigList)
{
if (sheetIdName.empty())
{
nlwarning("sheetIdName is empty");
return;
}
if (!addActionConfig(NLMISC::CSheetId(sheetIdName), actionConfigList))
{
#ifdef NL_DEBUG
nlwarning("Error in actionConfig Reference for %s", sheetIdName.c_str());
#endif
}
}
bool AISHEETS::CCreature::addActionConfig(NLMISC::CSheetId const& sheetId, NLMISC::CDbgPtr& actionConfigList)
{
CActionList const* actionConfig = CSheets::getInstance()->lookupActionList(sheetId);
if (actionConfig)
{
actionConfigList = actionConfig; // fightConfigList.push_back(actionConfig);
return true;
}
return false;
}
AISHEETS::CGroupProperties& AISHEETS::CCreature::getProperties(uint32 groupIndex)
{
#if !FINAL_VERSION
nlassert(groupIndex!=~0);
#endif
if (_GroupPropertiesTbl.size()<=groupIndex && groupIndex!=~0)
{
uint32 const resizeSize = std::max((uint32)CSheets::getInstance()->_NameToGroupIndex.size(), (uint32)(groupIndex+1));
_GroupPropertiesTbl.resize(resizeSize);
}
return _GroupPropertiesTbl[groupIndex];
}
AISHEETS::CGroupProperties const& AISHEETS::CCreature::getPropertiesCst(uint32 groupIndex) const
{
if (groupIndex<_GroupPropertiesTbl.size())
return _GroupPropertiesTbl[groupIndex];
else
return CSheets::getInstance()->_DefaultGroupProp;
}
std::vector AISHEETS::CCreature::getMultiLineInfoString() const
{
std::vector container;
pushTitle(container, "AISHEETS::CCreature");
pushEntry(container, "sheet=" + SheetId().toString());
pushEntry(container, "level=" + toString("%d", Level()));
container.back() += "radii=" + toString("%4.1f,%4.1f", Radius(), BoundingRadius());
container.back() += "height=" + toString("%4.1f", Height());
pushFooter(container);
return container;
}
void AISHEETS::CCreature::readGeorges(NLMISC::CSmartPtr const& form, NLMISC::CSheetId const& sheetId)
{
NLGEORGES::UFormElm const& item = form->getRootNode();
// the form was found so read the true values from George
_SheetId = sheetId;
#ifdef NL_DEBUG
nlassert(debugSheet.get().empty() || _SheetId!=NLMISC::CSheetId(debugSheet));
#endif
item.getValueByName(_Level,"Basics.Level");
if (!item.getValueByName(_DynamicGroupCountMultiplier, "Basics.Characteristics.DynGroupCountMultiplier"))
_DynamicGroupCountMultiplier = 1;
item.getValueByName(_ColorHead,"Basics.Equipment.Head.Color");
item.getValueByName(_ColorArms,"Basics.Equipment.Arms.Color");
item.getValueByName(_ColorHands,"Basics.Equipment.Hands.Color");
item.getValueByName(_ColorBody,"Basics.Equipment.Body.Color");
item.getValueByName(_ColorLegs,"Basics.Equipment.Legs.Color");
item.getValueByName(_ColorFeets,"Basics.Equipment.Feet.Color");
item.getValueByName(_Radius,"Collision.CollisionRadius");
item.getValueByName(_Height,"Collision.Height");
item.getValueByName(_Width,"Collision.Width");
item.getValueByName(_Length,"Collision.Length");
item.getValueByName(_BoundingRadius,"Collision.BoundingRadius");
item.getValueByName(_NotTraversable, "Collision.NotTraversable");
item.getValueByName(_BonusAggroHungry,"Combat.BonusAggroHungry");
item.getValueByName(_BonusAggroVeryHungry,"Combat.BonusAggroVeryHungry");
item.getValueByName(_AggroRadiusNotHungry,"Combat.AggroRadiusNotHungry");
item.getValueByName(_AggroRadiusHungry,"Combat.AggroRadiusHungry");
item.getValueByName(_AggroRadiusHunting,"Combat.AggroRadiusHunting");
if (!item.getValueByName(_AggroReturnDistCheck,"Combat.AggroReturnDistCheck"))
_AggroReturnDistCheck = -1.f;
if (!item.getValueByName(_AggroRadiusD1,"Combat.AggroRadiusD1"))
_AggroRadiusD1 = -1.f;
if (!item.getValueByName(_AggroRadiusD2,"Combat.AggroRadiusD2"))
_AggroRadiusD2 = -1.f;
if (!item.getValueByName(_AggroPrimaryGroupDist,"Combat.AggroPrimaryGroupDist"))
_AggroPrimaryGroupDist = -1.f;
if (!item.getValueByName(_AggroPrimaryGroupCoef,"Combat.AggroPrimaryGroupCoef"))
_AggroPrimaryGroupCoef = -1.f;
if (!item.getValueByName(_AggroSecondaryGroupDist,"Combat.AggroSecondaryGroupDist"))
_AggroSecondaryGroupDist = -1.f;
if (!item.getValueByName(_AggroSecondaryGroupCoef,"Combat.AggroSecondaryGroupCoef"))
_AggroSecondaryGroupCoef = -1.f;
if (!item.getValueByName(_AggroPropagationRadius,"Combat.AggroPropagationRadius"))
_AggroPropagationRadius = -1.f;
item.getValueByName(_AssistDist,"Combat.AssistDist");
item.getValueByName(_Scale, "3d data.Scale");
{
std::string faunaTypeStr;
item.getValueByName(faunaTypeStr, "Basics.type");
_FaunaType = getType(faunaTypeStr.c_str());
}
item.getValueByName(_ForceDisplayCreatureName, "3d data.ForceDisplayCreatureName");
// Get the dist fromm Bip to Mid
float tmpBip01ToMid;
if (!item.getValueByName(tmpBip01ToMid, "Collision.Dist Bip01 to mid"))
tmpBip01ToMid = 0.f;
// Get the distance from the bip01 to the front.
if (!item.getValueByName(_DistToFront, "Collision.Dist Bip01 to front"))
_DistToFront = 1.f;
// Get the distance from the bip01 to the front.
if (!item.getValueByName(_DistToBack, "Collision.Dist Bip01 to back"))
_DistToBack = 1.f;
// Get the creature Width.
if (!item.getValueByName(_DistToSide, "Collision.Width"))
_DistToSide = 1.f;
_DistToFront = _DistToFront-tmpBip01ToMid;
_DistToBack = tmpBip01ToMid-_DistToBack;
_DistToSide = _DistToSide/2.f;
_DistToFront *= _Scale;
_DistToBack *= _Scale;
_DistToSide *= _Scale;
if (!item.getValueByName(_DistModulator, "Combat.DistModulator"))
_DistModulator = 0.5f; // (0) - (1) - (n).
if (!item.getValueByName(_TargetModulator, "Combat.TargetModulator"))
_TargetModulator = 1.f; // (0) - (1).
if (!item.getValueByName(_ScoreModulator, "Combat.ScoreModulator"))
_ScoreModulator = 0.01f; // (0) - (1).
if (!item.getValueByName(_FearModulator, "Combat.FearModulator"))
_FearModulator = 0.01f; // (0) - (1).
if (!item.getValueByName(_LifeLevelModulator, "Combat.LifeLevelModulator"))
_LifeLevelModulator = 0.5f; // (0) - (1).
if (!item.getValueByName(_CourageModulator, "Combat.CourageModulator"))
_CourageModulator = 2.f; // (-n) - (0) - (+n).
if (!item.getValueByName(_GroupCohesionModulator, "Combat.GroupCohesionModulator"))
_GroupCohesionModulator = 0.5f; // (0) - (1)
if (!item.getValueByName(_GroupDispersion, "Basics.MovementSpeeds.GroupDispersion"))
_GroupDispersion = 0.5f; // (0) - (1).
if (!item.getValueByName(_XPLevel, "Basics.XPLevel"))
_XPLevel = 1;
if (!item.getValueByName(_NbPlayers, "Basics.NbPlayers"))
_NbPlayers = 1;
nlassert(_DistModulator>=0);
nlassert(_TargetModulator>=0);
nlassert(_ScoreModulator>=0 && _ScoreModulator<=1);
nlassert(_FearModulator>=0 && _FearModulator<=1);
nlassert(_LifeLevelModulator>=0 && _LifeLevelModulator<=1);
nlassert(_GroupCohesionModulator>=0 && _GroupCohesionModulator<=1);
nlassert(_GroupDispersion>=0 && _GroupDispersion<=1);
_EnergyValue = uint32(0.01f * ENERGY_SCALE);
float v;
if (item.getValueByName(v, "Basics.Characteristics.DynamicEnergyValue") && v!=0)
_EnergyValue = uint32(v * ENERGY_SCALE);
if (!item.getValueByName(_CanTurn, "Properties.Turn"))
_CanTurn = true;
uint32 meleeConfigChoice = 0;
uint32 rangeConfigChoice = 0;
uint32 nukeConfigChoice = 0;
uint32 healConfigChoice = 0;
breakable
{
std::string actionConfigStr;
item.getValueByName(actionConfigStr, "action_cfg");
if (actionConfigStr.length()!=5) // 4numbers + "f".
break;
char a[2] = "0";
a[0] = actionConfigStr[0];
meleeConfigChoice = atol(a);
a[0] = actionConfigStr[1];
rangeConfigChoice = atol(a);
a[0] = actionConfigStr[2];
nukeConfigChoice = atol(a);
a[0] = actionConfigStr[3];
healConfigChoice = atol(a);
}
static std::string meleeFightConfigString("melee_cfg");
static std::string rangeFightConfigString("range_cfg");
static std::string nukeFightConfigString("nuke_cfg");
static std::string healFightConfigString("heal_cfg");
if (meleeConfigChoice>0)
parseFightConfig(form, meleeFightConfigString, meleeConfigChoice-1, _FightConfig[FIGHTCFG_MELEE]);
if (rangeConfigChoice>0)
parseFightConfig(form, rangeFightConfigString, rangeConfigChoice-1, _FightConfig[FIGHTCFG_RANGE]);
if (nukeConfigChoice>0)
parseFightConfig(form, nukeFightConfigString, nukeConfigChoice-1, _FightConfig[FIGHTCFG_NUKE]);
if (healConfigChoice>0)
parseFightConfig(form, healFightConfigString, healConfigChoice-1, _FightConfig[FIGHTCFG_HEAL]);
// reads left & right item.
{
std::string left;
item.getValueByName(left, "item_left");
if (!left.empty())
_LeftItem = NLMISC::CSheetId(left);
std::string right;
item.getValueByName(right, "item_right");
if (!right.empty())
_RightItem = NLMISC::CSheetId(right);
calcFightAndVisualValues(&left, &right);
}
std::string s;
if (item.getValueByName(s, "Basics.Fame"))
{
_FactionIndex = CStaticFames::getInstance().getFactionIndex(s);
}
if (item.getValueByName(s, "Basics.FameForGuardAttack") && !s.empty())
{
double tmp;
sscanf(s.c_str(), "%f", &tmp);
_FameForGuardAttack = (sint32)tmp;
}
else
_FameForGuardAttack = ICreature::InvalidFameForGuardAttack;
// Assist Group Indexs.
{
item.getValueByName(_GroupIndexStr,"group_id");
if (_GroupIndexStr.empty())
{
std::string cat;
std::string raceCode;
if (item.getValueByName(cat, "category") && item.getValueByName(raceCode, "race_code"))
_GroupIndexStr = cat + raceCode;
}
item.getValueByName(_AssistGroupIndexStr, "group_assist");
setAssisGroupIndexs();
item.getValueByName(_AttackGroupIndexStr, "group_attack");
setAttackGroupIndexs();
}
// Bot name
item.getValueByName(_BotName, "Basics.BotName");
//////////////////////////////////////////////////////////////////////////
// Reads Script Comps.
breakable
{
NLGEORGES::UFormElm const* scriptCompNode = NULL;
const_cast(form->getRootNode()).getNodeByName(&scriptCompNode, "special_comp");
if (!scriptCompNode)
break;
uint arraySize = 0;
scriptCompNode->getArraySize(arraySize);
for (uint arrayIndex=0; arrayIndexgetArrayValue(scriptCompStr, arrayIndex);
CFightScriptComp* scriptComp;
try
{
scriptComp = CFightScriptCompReader::createScriptComp(scriptCompStr);
registerScriptComp(scriptComp);
}
catch (ReadFightActionException& ex)
{
nlwarning("script read error (ignored): %s", ex.what());
}
}
}
// Creature race
breakable
{
string raceStr;
if(item.getValueByName(raceStr, "Basics.Race") && !raceStr.empty())
_Race = EGSPD::CPeople::fromString(raceStr);
else
_Race = EGSPD::CPeople::Unknown;
}
}
void AISHEETS::CCreature::registerScriptComp(CFightScriptComp* scriptComp)
{
_ScriptCompList.push_back(scriptComp);
CFightSelectFilter* filter = dynamic_cast(scriptComp);
if (!filter)
return;
std::string const& param = filter->getParam();
if (param=="ON_UPDATE")
_UpdateScriptList.push_back(scriptComp);
if (param=="ON_DEATH")
_DeathScriptList.push_back(scriptComp);
if (param=="ON_BIRTH")
_BirthScriptList.push_back(scriptComp);
}
uint AISHEETS::CCreature::getVersion()
{
return 45;
}
void AISHEETS::CCreature::serial(NLMISC::IStream &s)
{
s.serial(_SheetId, _Level);
#ifdef NL_DEBUG
nlassert(debugSheet.get().empty() || _SheetId!=NLMISC::CSheetId(debugSheet));
#endif
s.serial(_DynamicGroupCountMultiplier);
s.serial(_ColorHead, _ColorArms, _ColorHands),
s.serial(_ColorBody, _ColorLegs, _ColorFeets);
s.serial(_Radius, _Height, _Width, _Length);
s.serial(_BoundingRadius);
s.serial(_BonusAggroHungry, _BonusAggroVeryHungry);
s.serial(_AggroRadiusNotHungry, _AggroRadiusHungry, _AggroRadiusHunting);
s.serial(_AggroReturnDistCheck);
s.serial(_AggroRadiusD1, _AggroRadiusD2);
s.serial(_AggroPrimaryGroupDist);
s.serial(_AggroPrimaryGroupCoef);
s.serial(_AggroSecondaryGroupDist);
s.serial(_AggroSecondaryGroupCoef);
s.serial(_AggroPropagationRadius);
s.serial(_Scale);
s.serial(_ForceDisplayCreatureName);
s.serialEnum(_FaunaType);
s.serial(_DistToFront);
s.serial(_DistToBack);
s.serial(_DistToSide);
s.serial(_DistModulator);
s.serial(_TargetModulator);
s.serial(_ScoreModulator);
s.serial(_FearModulator);
s.serial(_LifeLevelModulator);
s.serial(_CourageModulator);
s.serial(_GroupCohesionModulator);
s.serial(_GroupDispersion);
s.serial(_XPLevel);
s.serial(_NbPlayers);
s.serial(_EnergyValue);
s.serial(_CanTurn);
if (s.isReading())
{
readFightConfig(s, _FightConfig[FIGHTCFG_MELEE]);
readFightConfig(s, _FightConfig[FIGHTCFG_RANGE]);
readFightConfig(s, _FightConfig[FIGHTCFG_NUKE]);
readFightConfig(s, _FightConfig[FIGHTCFG_HEAL]);
}
else
{
saveFightConfig(s, _FightConfig[FIGHTCFG_MELEE]);
saveFightConfig(s, _FightConfig[FIGHTCFG_RANGE]);
saveFightConfig(s, _FightConfig[FIGHTCFG_NUKE]);
saveFightConfig(s, _FightConfig[FIGHTCFG_HEAL]);
}
s.serial(_AssistDist);
// serialize left & right item.
s.serial(_LeftItem, _RightItem);
s.serial(_FactionIndex);
s.serial(_FameForGuardAttack);
s.serial(_GroupIndexStr);
s.serial(_AssistGroupIndexStr);
s.serial(_AttackGroupIndexStr);
s.serial(_BotName);
s.serialEnum(_Race);
if (s.isReading())
{
setAssisGroupIndexs();
setAttackGroupIndexs();
}
if (s.isReading())
{
uint32 nbScript;
s.serial(nbScript);
for (uint32 index=0; indextoString();
s.serial(str);
}
}
calcFightAndVisualValues();
}
void AISHEETS::CCreature::getGroupStr(std::vector& groupIndexStrList, std::string const& groupIndexStr)
{
if (groupIndexStr.empty())
return;
size_t firstIndex = 0;
size_t lastIndex = firstIndex - 1;
do
{
firstIndex = lastIndex + 1;
lastIndex = groupIndexStr.find_first_of(',',firstIndex);
std::string str;
if (lastIndex==std::string::npos)
str = groupIndexStr.substr(firstIndex, groupIndexStr.size()-firstIndex);
else
str = groupIndexStr.substr(firstIndex, lastIndex-firstIndex);
uint32 const otherGroupIndex = CSheets::getInstance()->getGroupPropertiesIndex(str);
if (otherGroupIndex!=~0)
groupIndexStrList.push_back(otherGroupIndex);
} while (lastIndex!=std::string::npos);
}
//////////////////////////////////////////////////////////////////////////////
// CRaceStats //
//////////////////////////////////////////////////////////////////////////////
void AISHEETS::CRaceStats::readGeorges(NLMISC::CSmartPtr const& form, NLMISC::CSheetId const& sheetId)
{
NLGEORGES::UFormElm const& item = form->getRootNode();
// the form was found so read the true values from George
_SheetId = sheetId;
#ifdef NL_DEBUG
nlassert(debugSheet.get().empty() || _SheetId!=NLMISC::CSheetId(debugSheet));
#endif
item.getValueByName(_Race, "Race");
}
uint AISHEETS::CRaceStats::getVersion()
{
return 1;
}
void AISHEETS::CRaceStats::serial(NLMISC::IStream &s)
{
s.serial(_SheetId);
#ifdef NL_DEBUG
nlassert(debugSheet.get().empty() || _SheetId!=NLMISC::CSheetId(debugSheet));
#endif
s.serial(_SheetId);
s.serial(_Race);
}
//////////////////////////////////////////////////////////////////////////////
// CSheets //
//////////////////////////////////////////////////////////////////////////////
AISHEETS::CSheets* AISHEETS::CSheets::_Instance = NULL;
AISHEETS::CSheets* AISHEETS::CSheets::getInstance()
{
if (!_Instance)
_Instance = new AISHEETS::CSheets;
return _Instance;
}
void AISHEETS::CSheets::destroyInstance()
{
delete _Instance;
_Instance = NULL;
}
AISHEETS::CSheets::CSheets()
: _Initialised(false)
{
}
void AISHEETS::CSheets::init()
{
if (_Initialised)
return;
_PlayerGroupIndex=getGroupPropertiesIndex("zp");
#if !FINAL_VERSION
nlassert(_PlayerGroupIndex!=~0);
#endif
CConfigFile::CVar *varPtr=IService::getInstance()->ConfigFile.getVarPtr(std::string("GeorgePaths"));
const std::string writeFilesDirectoryName=IService::getInstance()->WriteFilesDirectory.toString();
// if config file variable 'GeorgePaths' exists then only do a minimal loadForms otherwise do the full works
if (varPtr!=NULL)
{
bool addSearchPath=false;
loadForm2("aiaction", writeFilesDirectoryName+AISPackedActionSheetsFilename, _ActionSheets, false, false);
if (_ActionSheets.empty())
{
if (!addSearchPath)
{
addSearchPath=true;
for (uint32 i=0;isize();++i)
CPath::addSearchPath(varPtr->asString(i).c_str(), true, false);
}
loadForm2("aiaction", writeFilesDirectoryName+AISPackedActionSheetsFilename, _ActionSheets, true);
}
loadForm("actionlist", writeFilesDirectoryName+AISPackedFightConfigSheetsFilename, _ActionListSheets, false, false);
if (_ActionListSheets.empty())
{
if (!addSearchPath)
{
addSearchPath=true;
for (uint32 i=0;isize();++i)
CPath::addSearchPath(varPtr->asString(i).c_str(), true, false);
}
loadForm("actionlist", writeFilesDirectoryName+AISPackedFightConfigSheetsFilename, _ActionListSheets, true);
}
loadForm2("creature", writeFilesDirectoryName+AISPackedSheetsFilename, _Sheets, false, false);
if (_Sheets.empty())
{
if (!addSearchPath)
{
addSearchPath=true;
for (uint32 i=0;isize();++i)
CPath::addSearchPath(varPtr->asString(i).c_str(), true, false);
}
loadForm2("creature", writeFilesDirectoryName+AISPackedSheetsFilename, _Sheets, true);
}
loadForm2("race_stats", writeFilesDirectoryName+AISPackedRaceStatsSheetsFilename, _RaceStatsSheets, false, false);
if (_RaceStatsSheets.empty())
{
if (!addSearchPath)
{
addSearchPath=true;
for (uint32 i=0;isize();++i)
CPath::addSearchPath(varPtr->asString(i).c_str(), true, false);
}
loadForm2("race_stats", writeFilesDirectoryName+AISPackedRaceStatsSheetsFilename, _RaceStatsSheets, true);
}
}
else
{
loadForm2("aiaction", writeFilesDirectoryName+AISPackedActionSheetsFilename, _ActionSheets, true);
loadForm("actionlist", writeFilesDirectoryName+AISPackedFightConfigSheetsFilename, _ActionListSheets, true);
loadForm2("creature", writeFilesDirectoryName+AISPackedSheetsFilename, _Sheets, true);
loadForm2("race_stats", writeFilesDirectoryName+AISPackedRaceStatsSheetsFilename, _RaceStatsSheets, true);
}
_Initialised=true;
}
void AISHEETS::CSheets::release()
{
_Sheets.clear();
_ActionListSheets.clear();
_ActionSheets.clear();
_RaceStatsSheets.clear();
}
uint32 AISHEETS::CSheets::getGroupPropertiesIndex(std::string groupIndexName)
{
if (groupIndexName.empty())
return ~0;
NLMISC::strupr(groupIndexName);
std::map::iterator it = _NameToGroupIndex.find(groupIndexName);
if (it==_NameToGroupIndex.end())
{
uint32 groupIndex = (uint32)_NameToGroupIndex.size();
_NameToGroupIndex.insert(make_pair(groupIndexName, groupIndex));
#if !FINAL_VERSION
nldebug("GroupIndex Entry: %s %d", groupIndexName.c_str(), groupIndex);
#endif
it = _NameToGroupIndex.find(groupIndexName);
#ifdef NL_DEBUG
nlassert(it!=_NameToGroupIndex.end());
#endif
// Resize other group table. Better imp should be done with listeners.
}
return it->second;
}
void AISHEETS::CSheets::display(CSmartPtr stringWriter, uint infoSelect)
{
nlassert(_Initialised);
std::map::iterator it;
for(it=_Sheets.begin(); it!=_Sheets.end(); ++it)
{
std::vector strings;
strings = it->second->getMultiLineInfoString();
FOREACHC(itString, std::vector, strings)
stringWriter->append(toString("%04x", it->second->SheetId().asInt()) + " " + *itString);
}
}
AISHEETS::ICreatureCPtr AISHEETS::CSheets::lookup(CSheetId const& id)
{
// setup an iterator and lookup the sheet id in the map
std::map::iterator it=_Sheets.find(id);
// if we found a valid entry return a pointer to the creature record otherwise 0
if (it!=_Sheets.end())
return (AISHEETS::CCreature*)it->second;
else
{
nlwarning("Unknow creature sheet '%s'", id.toString().c_str());
return NULL;
}
}
AISHEETS::IAIActionCPtr AISHEETS::CSheets::lookupAction(CSheetId const& id)
{
// setup an iterator and lookup the sheet id in the map
std::map::iterator it = _ActionSheets.find(id);
// if we found a valid entry return a pointer to the creature record otherwise 0
if (it!=_ActionSheets.end())
return (AISHEETS::CAIAction*)it->second;
else
return NULL;
}
AISHEETS::CActionList const* AISHEETS::CSheets::lookupActionList(CSheetId const& id)
{
// setup an iterator and lookup the sheet id in the map
std::map::iterator it=_ActionListSheets.find(id);
// if we found a valid entry return a pointer to the creature record otherwise 0
if (it!=_ActionListSheets.end())
return &((*it).second);
else
return NULL;
}
AISHEETS::IRaceStatsCPtr AISHEETS::CSheets::lookupRaceStats(CSheetId const& id)
{
// setup an iterator and lookup the sheet id in the map
std::map::iterator it=_RaceStatsSheets.find(id);
if (it!=_RaceStatsSheets.end())
return (AISHEETS::CRaceStats*)it->second;
else
{
nlwarning("Unknow race_stats sheet '%s'", id.toString().c_str());
return NULL;
}
}
//////////////////////////////////////////////////////////////////////////////
// Console commands //
//////////////////////////////////////////////////////////////////////////////
NLMISC_COMMAND(displaySheetNames,"display sheet data for all sheets","")
{
if(args.size() !=0) return false;
AISHEETS::CSheets::getInstance()->display(new CLogStringWriter(&log),0);
return true;
}
NLMISC_COMMAND(displaySheetBasics,"display sheet data for all sheets","")
{
if(args.size() !=0) return false;
AISHEETS::CSheets::getInstance()->display(new CLogStringWriter(&log),1);
return true;
}
NLMISC_COMMAND(displaySheetCombat,"display sheet data for all sheets","")
{
if(args.size() !=0) return false;
AISHEETS::CSheets::getInstance()->display(new CLogStringWriter(&log),2);
return true;
}
NLMISC_COMMAND(displaySheetByName,"display sheet data for given sheets"," [...]")
{
if (args.size() <1)
return false;
for (uint i=0;ilookup(NLMISC::CSheetId(args[i]));
if (!sheet)
{
log.displayNL("Failed to find sheet: %s",args[0].c_str());
continue;
}
std::vector strings = sheet->getMultiLineInfoString();
FOREACHC(it, std::vector, strings)
log.displayNL("%s", it->c_str());
}
return true;
}
/*
NLMISC_COMMAND(setSheetProperty,"change a value read from a sheet"," level|walk|run|radius|bounding|height|aggro|attack|danger|flight|survive|initSurv|crit ")
{
if (args.size() !=3)
return false;
// lookup the sheet id
AISHEETS::CCreature* sheet = const_cast(dynamic_cast(AISHEETS::CSheets::getInstance()->lookup(NLMISC::CSheetId(args[0]))));
if (!sheet)
{
log.displayNL("Failed to find sheet: %s",args[0].c_str());
return false;
}
// get the value
float val = (float)atof(args[2].c_str());
if (val==0 && args[2]!="0" && args[2]!="0.0")
{
log.displayNL("'%s' is not a valid value",args[2].c_str());
return false;
}
breakable
{
if (nlstricmp(args[1].c_str(),"level")==0) { sheet->_Level=(uint32)val; break; }
if (nlstricmp(args[1].c_str(),"radius")==0) { sheet->_Radius=val; break; }
if (nlstricmp(args[1].c_str(),"bounding")==0) { sheet->_BoundingRadius=val; break; }
if (nlstricmp(args[1].c_str(),"height")==0) { sheet->_Height=val; break; }
log.displayNL("variable name not recognised: %s", args[1].c_str());
return false;
}
std::vector strings = sheet->getMultiLineInfoString();
FOREACHC(it, std::vector, strings)
log.displayNL("%s", it->c_str());
return true;
}
*/