// 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 NL_CONTINENT_CONTAINER_H
#define NL_CONTINENT_CONTAINER_H
// Nel Misc
#include "nel/misc/types_nl.h"
#include "nel/misc/smart_ptr.h"
#include "nel/misc/vectord.h"
#include "nel/misc/sheet_id.h"
#include "nel/misc/log.h"
#include "nel/misc/entity_id.h"
// Nel Georges
#include "nel/georges/u_form.h"
#include "nel/georges/u_form_elm.h"
// Nel Pacs
#include "nel/pacs/u_move_container.h"
#include "nel/pacs/u_move_primitive.h"
#include "nel/pacs/u_retriever_bank.h"
#include "nel/pacs/u_global_retriever.h"
#include "nel/pacs/u_global_position.h"
#include "nel/pacs/u_primitive_block.h"
/**
*
* \author Benjamin Legros
* \author Nevrax France
* \date 2001
*/
class CContinentContainer
{
protected:
/// Container of continent
class CContinentMoveContainer
{
public:
CContinentMoveContainer() : GlobalRetriever(NULL), RetrieverBank(NULL), MoveContainer(NULL), AllowAutoSpawn(false) {}
/// Then name of the continent
std::string Name;
/// The PACS global retriever for this continent
NLPACS::UGlobalRetriever *GlobalRetriever; // Global retriever retrieved surfaces of all territory
/// The PACS retriever bank for this continent
NLPACS::URetrieverBank *RetrieverBank; // Bank of global retriever
/// The PACS move container for this continent
NLPACS::UMoveContainer *MoveContainer; // Move container
/// True if entities are allowed to try spawning in this continent if no continent index is provided
bool AllowAutoSpawn;
};
///
class CSheet
{
public:
CSheet() {}
std::string Name;
std::string PacsRBank;
std::string PacsGR;
std::string LandscapeIG;
std::vector ListIG;
void readGeorges (const NLMISC::CSmartPtr &form, const NLMISC::CSheetId &sheetId)
{
Name = sheetId.toString();
// the form was found so read the true values from George
form->getRootNode ().getValueByName (PacsRBank, "PacsRBank");
form->getRootNode ().getValueByName (PacsGR, "PacsGR");
form->getRootNode ().getValueByName (LandscapeIG, "LandscapeIG");
NLGEORGES::UFormElm *node;
uint numVillages, numIgs;
if (form->getRootNode().getNodeByName(&node, "Villages") && node &&
node->getArraySize(numVillages))
{
for (uint village=0; villagegetRootNode ().getNodeByName(&node, NLMISC::toString("Villages[%d].IgList", village).c_str()) && node &&
node->getArraySize(numIgs))
{
for (uint ig=0; iggetRootNode ().getValueByName(igName, NLMISC::toString("Villages[%d].IgList[%d].IgName", village, ig).c_str()))
ListIG.push_back(igName);
}
}
}
}
}
void serial (NLMISC::IStream &s)
{
s.serial(Name);
s.serial(PacsRBank, PacsGR, LandscapeIG);
s.serialCont(ListIG);
}
static uint getVersion () { return 1; }
void removed() {}
};
/// The global continent container
typedef std::vector TContinentContainer;
///
typedef std::map TPacsPrimMap;
///
typedef std::map TTriggerMap;
/// the continents
TContinentContainer _Continents;
/// pacs primitives
TPacsPrimMap _PacsPrimMap;
/// Sheet map type
typedef std::map TSheetMap;
///
TSheetMap _SheetMap;
///
TTriggerMap _TriggerMap;
///
uint _GridWidth;
uint _GridHeight;
double _CellSize;
double _PrimitiveMaxSize;
uint _NbWorldImages;
/// Should load dynamic collision (e.g. trees)
bool _LoadPacsPrims;
public:
/// Constructor
CContinentContainer();
/// Init whole continent container
void init(uint gridWidth, uint gridHeight, double primitiveMaxSize, uint nbWorldImages, const std::string packedSheetsDirectory, double cellSize=0.0, bool loadPacsPrims = true);
/// Init pacs prims
void initPacsPrim(const std::string &path = std::string("landscape_col_prim_pacs_list.txt"));
/// Load continent
void loadContinent(std::string name, std::string file, sint index, bool allowAutoSpawn = true);
/// Remove continent
void removeContinent(sint index);
/// Get move container for continent
NLPACS::UMoveContainer *getMoveContainer(sint index)
{
if (index < 0 || index >= (sint)_Continents.size())
{
nlwarning("getMoveContainer(): invalid index %d", index);
return NULL;
}
return _Continents[index].MoveContainer;
}
/// Get move container for continent
NLPACS::UGlobalRetriever *getRetriever(sint index)
{
if (index < 0 || index >= (sint)_Continents.size())
{
nlwarning("getRetriever(): invalid index %d", index);
return NULL;
}
return _Continents[index].GlobalRetriever;
}
/// Get move container for continent
NLPACS::URetrieverBank *getRetrieverBank(sint index)
{
if (index < 0 || index >= (sint)_Continents.size())
{
nlwarning("getRetriverBank(): invalid index %d", index);
return NULL;
}
return _Continents[index].RetrieverBank;
}
/// Find continent to spawn in
sint findContinent(const NLMISC::CVectorD &worldPosition, const NLMISC::CEntityId& id = NLMISC::CEntityId::Unknown)
{
sint continent = -1;
// finds the continent that fits given position
uint i;
for (i=0; i<_Continents.size(); ++i)
{
if (_Continents[i].GlobalRetriever == NULL ||
!_Continents[i].AllowAutoSpawn)
continue;
NLPACS::UGlobalPosition globalPosition = _Continents[i].GlobalRetriever->retrievePosition(worldPosition);
if (globalPosition.InstanceId != -1 && globalPosition.LocalPosition.Surface != -1)
{
if (continent != -1)
{
nlwarning("findContinent%s: continent not provided, and sole position (%.3f, %.3f, %.3f) refers to several continents (%d, %d) -- use first found",
id.toString().c_str(),
worldPosition.x, worldPosition.y, worldPosition.z,
i, continent);
continue;
}
continent = i;
}
}
if (continent == -1)
{
if (worldPosition.x != 0.0 || worldPosition.y != 0.0 || worldPosition.z != 0.0)
nlwarning("findContinent%s: can't find best fitting continent for position (%.3f, %.3f, %.3f) -- use first available continent", id.toString().c_str(), worldPosition.x, worldPosition.y, worldPosition.z);
for (uint i=0; i<_Continents.size(); ++i)
{
if (_Continents[i].MoveContainer != NULL)
{
continent = i;
break;
}
}
}
return continent;
}
/// Get container size
sint size() const { return (sint)_Continents.size(); }
/// clear the container -- WARNING
void clear()
{
uint i;
for (i=0; i<_Continents.size(); ++i)
removeContinent(i);
_Continents.clear();
}
/// get trigger position
NLMISC::CVectorD getTriggerPosition(uint i) const
{
TTriggerMap::const_iterator it = _TriggerMap.find(i);
return (it == _TriggerMap.end()) ? NLMISC::CVectorD::Null : (*it).second;
}
/// Display triggers
void displayTriggers(NLMISC::CLog *log = NLMISC::InfoLog) const
{
TTriggerMap::const_iterator it;
for (it=_TriggerMap.begin(); it!=_TriggerMap.end(); ++it)
log->displayNL("Trigger %d: (%.3f, %.3f, %.3f)", (*it).first, (*it).second.x, (*it).second.y, (*it).second.z);
}
protected:
void loadPacsPrims(const CSheet &sheet, NLPACS::UMoveContainer *moveContainer);
};
#endif // NL_CONTINENT_CONTAINER_H
/* End of continent_container.h */