khanat-opennel-code/code/ryzom/server/src/ai_data_service/pacs_scan.cpp
2015-12-15 15:08:54 +01:00

2975 lines
77 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/>.
// Nel misc
#include "nel/misc/command.h"
#include "nel/misc/variable.h"
#include "nel/misc/aabbox.h"
#include "nel/misc/vector.h"
#include "nel/misc/vectord.h"
#include "nel/misc/i_xml.h"
#include "nel/misc/array_2d.h"
// TMP TMP
#include "nel/misc/bitmap.h"
#include "nel/misc/algo.h"
// Nel pacs
#include "nel/pacs/u_collision_desc.h"
#include "nel/pacs/u_global_position.h"
#include "nel/pacs/u_global_retriever.h"
#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_primitive_block.h"
#include "nel/pacs/global_retriever.h"
// Nel Ligo
#include "nel/ligo/ligo_config.h"
#include "nel/ligo/primitive.h"
// Server share
#include "game_share/bmp4image.h"
#include "server_share/continent_container.h"
// AI share
#include "ai_share/world_map.h"
#include "ai_share/ai_spawn_commands.h"
// STL
#include <vector>
#include <map>
using namespace std;
using namespace NLMISC;
using namespace NLPACS;
using namespace RYAI_MAP_CRUNCH;
using namespace NLLIGO;
extern CLigoConfig LigoConfig;
//CAISpawnCtrl *CAISpawnCtrl::_instance=NULL;
string EvaluatedPos;
/// The start point for each continent
multimap<string, CVectorD> StartPoints;
multimap<string, string> PrimFiles;
CVectorD DefaultStartPoint = CVectorD::Null;
/// The output path
string OutputPath = string("./");
vector<string> PacsPrimPath;
vector<string> LookupPath;
vector<string> LookupNoRecursePath;
bool PathInitialized = false;
CWorldMap StaticWorldMap;
uint Verbose = 1;
CVectorD BoxMin, BoxMax;
namespace RYPACSCRUNCH
{
class CPacsCruncher
{
public:
CContinentContainer _Continents;
URetrieverBank *_Bank;
UGlobalRetriever *_Retriever;
UMoveContainer *_Container;
UMovePrimitive *_Primitive;
UMovePrimitive *_Primitive3;
UMovePrimitive *_Primitive5;
CWorldMap _WorldMap;
deque<UGlobalPosition> _Positions1;
deque<UGlobalPosition> _Positions3;
deque<UGlobalPosition> _Positions5;
CVectorD _BMin;
CVectorD _BMax;
CVectorD _BSize;
uint32 _RetrieverWidth;
uint32 _RetrieverHeight;
uint32 _RetrieverArea;
typedef std::map<std::string, NLPACS::UPrimitiveBlock*> TPacsPrimMap;
static TPacsPrimMap _PacsPrimMap;
protected:
CSlot getSurfaceAfterMove(UGlobalPosition &pos, const CMapPosition &newPosition, uint maxGabarit)
{
CVectorD target = newPosition.toVectorD()+CVectorD(0.1, 0.1, 0.0);
CVectorD motion = target - _Retriever->getDoubleGlobalPosition(pos);
motion.z = 0.0;
UMovePrimitive *prims[3] = { _Primitive, _Primitive3, _Primitive5 };
sint gabarit;
CSlot slot;
for (gabarit=maxGabarit; gabarit>=0; --gabarit)
{
UMovePrimitive *prim = prims[gabarit];
prim->setGlobalPosition(pos, 0);
prim->move(motion, 0);
//_Container->evalNCPrimitiveCollision(1.0, prim, 0);
_Container->evalCollision(1.0, 0);
UGlobalPosition newPos;
prim->getGlobalPosition(newPos, 0);
// DEBUG HERE
//
CVectorD dpos = _Retriever->getDoubleGlobalPosition(pos);
CVectorD dnpos = _Retriever->getDoubleGlobalPosition(newPos);
//bool test = (static_cast<NLPACS::CGlobalRetriever*>(_Retriever))->testPosition(newPos);
bool test=true;
NLPACS::CGlobalRetriever* gRetriever = static_cast<NLPACS::CGlobalRetriever*>(_Retriever);
if (newPos.InstanceId < 0 || newPos.InstanceId >= (sint)gRetriever->getInstances().size())
test=false;
const CRetrieverInstance & instance = gRetriever->getInstances()[newPos.InstanceId];
if (!instance.getBBox().include(newPos.LocalPosition.Estimation + instance.getOrigin()))
test=false;
else
{
const CLocalRetriever & lRetriever = gRetriever->getRetriever(instance.getRetrieverId());
if (!lRetriever.isLoaded())
test=false;
else if (newPos.LocalPosition.Surface < 0 || newPos.LocalPosition.Surface >= (sint)lRetriever.getSurfaces().size())
{
nlwarning("can't test inexistant surface %d", newPos.LocalPosition.Surface);
test=false;
}
else if (fabs(newPos.LocalPosition.Estimation.x) >= 256.0 || fabs(newPos.LocalPosition.Estimation.y) >= 256.0)
test=false;
}
if (!test)
{
if (Verbose)
nlinfo("Move from pos(%d,%d,%f,%f/%f,%f) motion(%f,%f) -> pos(%d,%d,%f,%f/%f,%f) %s", pos.InstanceId, pos.LocalPosition.Surface, pos.LocalPosition.Estimation.x, pos.LocalPosition.Estimation.y, dpos.x, dpos.y, motion.x, motion.y, newPos.InstanceId, newPos.LocalPosition.Surface, newPos.LocalPosition.Estimation.x, newPos.LocalPosition.Estimation.y, dnpos.x, dnpos.y, test ? "succeded" : "failed");
return slot;
}
//
//
CVectorD dmotion = dpos+motion-dnpos;
dmotion.z = 0.0;
// if the move was not accomplished successfully return '3' as "direction blocked"
if ( dmotion.norm() > 1.0e-2 )
continue;
CMapPosition checkPosition(_Retriever->getDoubleGlobalPosition(newPos));
nlassert(newPosition == checkPosition);
CUnitSlot surfId(newPos);
TCellUnit &slots = _WorldMap.getCellUnit(newPosition);
// locate a slot to stick the surface in
bool newSlot = false;
for (slot=CSlot(0); slot.isValid(); ++slot)
{
CUnitSlot& unitSlot=slots[slot.slot()];
if (!unitSlot.used())
{
unitSlot = surfId;
unitSlot.setHeight((uint)floor(dnpos.z/2.0 + 0.5));
unitSlot.setInterior(_Retriever->isInterior(newPos));
float waterHeight;
unitSlot.setWater(_Retriever->isWaterPosition(newPos, waterHeight));
unitSlot.setGabarit(gabarit);
if (gabarit==0) _Positions1.push_back(newPos);
else if (gabarit==1) _Positions3.push_back(newPos);
else _Positions5.push_back(newPos);
newSlot = true;
break;
}
else if (unitSlot.hasSameSurface(surfId))
{
break;
}
}
break;
}
return slot;
}
void readPrimitive(IPrimitive *primitive, const std::string &insertAt)
{
if (dynamic_cast<CPrimZone*>(primitive) != NULL)
{
CPrimZone *prim = static_cast<CPrimZone*>(primitive);
uint i;
for (i=0; i<prim->VPoints.size(); ++i)
{
StartPoints.insert(multimap<string, CVectorD>::value_type(insertAt, prim->VPoints[i]));
}
}
else if (dynamic_cast<CPrimPoint*>(primitive) != NULL)
{
CPrimPoint *prim = static_cast<CPrimPoint*>(primitive);
StartPoints.insert(multimap<string, CVectorD>::value_type(insertAt, prim->Point));
}
else if (dynamic_cast<CPrimPath*>(primitive) != NULL)
{
CPrimPath *prim = static_cast<CPrimPath*>(primitive);
uint i;
for (i=0; i<prim->VPoints.size(); ++i)
{
StartPoints.insert(multimap<string, CVectorD>::value_type(insertAt, prim->VPoints[i]));
}
}
// parse children
uint i;
for (i=0; i<primitive->getNumChildren(); ++i)
{
IPrimitive *child;
if (!primitive->getChild(child, i))
continue;
readPrimitive(child, insertAt);
}
}
void readStartupPrims(const std::string &continent)
{
pair<multimap<string, string>::iterator, multimap<string, string>::iterator> rng;
rng = PrimFiles.equal_range(continent);
multimap<string, string>::iterator it;
for (it=rng.first; it!=rng.second; ++it)
{
string prim = (*it).second;
CIFile f(CPath::lookup(prim));
CIXml xml;
CPrimitives prims;
// load xml file
xml.init(f);
nlinfo("Loaded prim file '%s'", prim.c_str());
// read nodes
if (!prims.read(xml.getRootNode(), prim.c_str(), LigoConfig))
{
nlwarning("Can't use primitive file '%s', xml parse error", prim.c_str());
continue;
}
}
}
public:
CPacsCruncher() : _Bank(NULL), _Retriever(NULL), _Container(NULL)
{
}
void initPackedSheets()
{
if (!PathInitialized)
{
uint i;
for (i=0; i<PacsPrimPath.size(); ++i)
_Continents.initPacsPrim(PacsPrimPath[i]);
for (i=0; i<LookupPath.size(); ++i)
CPath::addSearchPath(LookupPath[i], true, false);
for (i=0; i<LookupNoRecursePath.size(); ++i)
CPath::addSearchPath(LookupNoRecursePath[i], false, false);
PathInitialized = true;
}
_Continents.init(0, 0, 32.0f, 2, "./", 32.0);
}
void init(const std::string &name)
{
initPackedSheets();
_Continents.loadContinent(name, name, 0, true);
_Bank = _Continents.getRetrieverBank(0);
_Retriever = _Continents.getRetriever(0);
_Container = _Continents.getMoveContainer(0);
}
void release()
{
_Continents.removeContinent(0);
_Container = NULL;
_Retriever = NULL;
_Bank = NULL;
}
string secToString(sint sec)
{
return toString(sec/60)+"m"+toString(sec%60)+"s";
}
//
void crunch(const std::string &name, const CAABBox *constraintBox = NULL, const string &outputsuffix = "", const CVector *startPoint=NULL)
{
// nlassert(false);
// setup pacs
bool keep = true;
if (_Bank == NULL && _Retriever == NULL && _Container == NULL)
{
init(name);
keep = false;
}
CAABBox rBox = _Retriever->getBBox();
nlinfo("%s rbox: %.1f,%.1f - %.1f,%.1f", name.c_str(), rBox.getMin().x, rBox.getMin().y, rBox.getMax().x, rBox.getMax().y);
// compute new box
CAABBox useBox;
if (constraintBox)
{
CVector vmin, vmax;
vmin.maxof(constraintBox->getMin(), rBox.getMin());
vmax.minof(constraintBox->getMax(), rBox.getMax());
useBox.setMinMax(vmin, vmax);
}
else
{
useBox = rBox;
}
useBox.setMinMax(CVector((float)floor(useBox.getMin().x), (float)floor(useBox.getMin().y), (float)floor(useBox.getMin().z)),
CVector((float)ceil(useBox.getMax().x), (float)ceil(useBox.getMax().y), (float)ceil(useBox.getMax().z)));
nlinfo("Use Box: (%.3f,%.3f)-(%.3f,%.3f)", useBox.getMin().x, useBox.getMin().y, useBox.getMax().x, useBox.getMax().y);
// setup the grid of surface ids per location
_RetrieverWidth = (uint32)(useBox.getSize().x+1.0);
_RetrieverHeight = (uint32)(useBox.getSize().y+1.0);
_RetrieverArea = _RetrieverWidth*_RetrieverHeight;
// clear the world map object
_WorldMap.clear();
// setup the UMovePrimitive
_Primitive = _Container->addCollisionablePrimitive(0, 1);
_Primitive->setPrimitiveType( UMovePrimitive::_2DOrientedCylinder );
_Primitive->setReactionType( UMovePrimitive::Stop );
_Primitive->setTriggerType((UMovePrimitive::TTrigger)UMovePrimitive::NotATrigger);
_Primitive->setCollisionMask( 0xffffffff );
_Primitive->setOcclusionMask( 0x00000000 );
_Primitive->setObstacle( true );
_Primitive->setAbsorbtion( 0 );
_Primitive->setHeight( 6.0f );
_Primitive->setRadius( 0.5f );
_Primitive3 = _Container->addCollisionablePrimitive(0, 1);
_Primitive3->setPrimitiveType( UMovePrimitive::_2DOrientedCylinder );
_Primitive3->setReactionType( UMovePrimitive::Stop );
_Primitive3->setTriggerType((UMovePrimitive::TTrigger)UMovePrimitive::NotATrigger);
_Primitive3->setCollisionMask( 0xffffffff );
_Primitive3->setOcclusionMask( 0x00000000 );
_Primitive3->setObstacle( true );
_Primitive3->setAbsorbtion( 0 );
_Primitive3->setHeight( 6.0f );
_Primitive3->setRadius( 1.5f );
_Primitive5 = _Container->addCollisionablePrimitive(0, 1);
_Primitive5->setPrimitiveType( UMovePrimitive::_2DOrientedCylinder );
_Primitive5->setReactionType( UMovePrimitive::Stop );
_Primitive5->setTriggerType((UMovePrimitive::TTrigger)UMovePrimitive::NotATrigger);
_Primitive5->setCollisionMask( 0xffffffff );
_Primitive5->setOcclusionMask( 0x00000000 );
_Primitive5->setObstacle( true );
_Primitive5->setAbsorbtion( 0 );
_Primitive5->setHeight( 6.0f );
_Primitive5->setRadius( 2.5f );
_BMin = useBox.getMin();
_BMax = useBox.getMax();
if (BoxMin != CVectorD::Null)
{
if (BoxMin.x > _BMin.x) _BMin.x = BoxMin.x;
if (BoxMin.x > _BMin.x) _BMin.x = BoxMin.x;
}
if (BoxMax != CVectorD::Null)
{
if (BoxMax.x < _BMax.x) _BMax.x = BoxMax.x;
if (BoxMax.x < _BMax.x) _BMax.x = BoxMax.x;
}
_BSize = _BMax-_BMin;
// setup a start position for the primitive
CVectorD zoneCentre = useBox.getCenter();
zoneCentre.z = 0.0;
vector<CVectorD> startPoints;
pair<multimap<string, CVectorD>::iterator, multimap<string, CVectorD>::iterator> range = StartPoints.equal_range(name);
multimap<string, CVectorD>::iterator itsp;
for (itsp=range.first; itsp!=range.second; ++itsp)
startPoints.push_back((*itsp).second);
if (startPoint != NULL)
{
startPoints.push_back(*startPoint);
startPoints.back().z = 0.0;
}
if (DefaultStartPoint != CVectorD::Null)
{
startPoints.push_back(DefaultStartPoint);
startPoints.back().z = 0.0;
}
if (startPoints.empty())
{
startPoints.push_back(zoneCentre);
}
// init time counter
CGlobalRetriever *cgr = (CGlobalRetriever*)_Retriever;
uint sz = 0;
uint ic;
for (ic = 0; ic<cgr->getInstances().size(); ++ic)
{
const CRetrieverInstance &inst = cgr->getInstance(ic);
sz += (uint)(inst.getBBox().getHalfSize().x*inst.getBBox().getHalfSize().y*0.6f)*4;
}
nlinfo("Estimated %d iterations", sz);
uint count=0;
while (!startPoints.empty())
{
zoneCentre = startPoints.back();
startPoints.pop_back();
CMapPosition firstIndex = CMapPosition(zoneCentre);
zoneCentre = firstIndex.toVectorD();
nlinfo("Start scan at %.1f,%.1f", zoneCentre.x, zoneCentre.y);
_Positions5.clear();
_Positions3.clear();
_Positions1.clear();
_Primitive->setGlobalPosition(zoneCentre, 0);
_Container->evalCollision(1.0, 0);
_Primitive3->setGlobalPosition(zoneCentre, 0);
_Container->evalCollision(1.0, 0);
_Primitive5->setGlobalPosition(zoneCentre, 0);
_Container->evalCollision(1.0, 0);
// setup and intialise the positions to visit stack
_Positions5.push_back(UGlobalPosition());
_Primitive->getGlobalPosition(_Positions5[0], 0);
{
TCellUnit &slots = _WorldMap.getCellUnit(firstIndex);
CUnitSlot surfId(_Positions5[0]);
// locate a slot to stick the surface in
bool newSlot = true;
CSlot slot;
for (slot=CSlot(0); slot.isValid(); ++slot)
{
CUnitSlot& unitSlot=slots[slot.slot()];
if (unitSlot.hasSameSurface(surfId))
{
newSlot = false;
break;
}
}
if (!newSlot)
continue;
CWorldPosition worldPosition(_WorldMap.getWorldPositionGeneration(firstIndex,CSlot(0)));
_WorldMap.getUnitSlot(worldPosition) = CUnitSlot(_Positions5[0]);
_WorldMap.getUnitSlot(worldPosition).setGabarit(2);
}
uint xmaxreached = 0,
xminreached = 0,
ymaxreached = 0,
yminreached = 0;
TTime startTime = CTime::getLocalTime();
uint countGabarits[3] = { 0, 0, 0 };
while (!_Positions5.empty() || !_Positions3.empty() || !_Positions1.empty())
{
// pop the next postion to test off the stack
UGlobalPosition refPos;
if (!_Positions5.empty())
{
refPos = _Positions5.front();
_Positions5.pop_front();
}
else if (!_Positions3.empty())
{
refPos = _Positions3.front();
_Positions3.pop_front();
}
else
{
refPos = _Positions1.front();
_Positions1.pop_front();
}
if (!(count&0xfff))
{
TTime itime = CTime::getLocalTime();
UGlobalPosition nextPos = refPos;
CVectorD dpos = _Retriever->getDoubleGlobalPosition(nextPos);
double persec = (double)count*1000.0 / (double)(itime-startTime);
sint toGo = sz-count;
double ttoGo = (double)toGo/persec;
sint elapsed = (sint)((itime-startTime)/1000);
nlinfo("crunchPacsMap: %d iterations (%d,%d,%d) (%dpct, estimated %s, %s to go) - start at (%d,%d,%f,%f-%f,%f)", count, countGabarits[0], countGabarits[1], countGabarits[2], count*100/sz, secToString(elapsed+(sint)ttoGo).c_str(), secToString((sint)ttoGo).c_str(), nextPos.InstanceId, nextPos.LocalPosition.Surface, nextPos.LocalPosition.Estimation.x, nextPos.LocalPosition.Estimation.y, dpos.x, dpos.y);
}
++count;
if (!(count&0xffff) && count>0)
{
//buildBMP("temp_"+name+toString(count));
}
CVectorD pos = _Retriever->getDoubleGlobalPosition(refPos);
// EvaluatedPos = toString(refPos.InstanceId)+","+toString(refPos.LocalPosition.Surface)+":"+toString(pos.x)+","+toString(pos.y)+","+toString(pos.z);
/*
CVectorD d = pos - CVectorD(1045.0, -6086.0, 0.0);
d.z = 0.0f;
if (d.norm() < 0.5f)
nlinfo("Bla");
*/
CMapPosition refIndex(pos);
TCellUnit &slots = _WorldMap.getCellUnit(refIndex);
uint slot;
CUnitSlot refSlot(refPos);
for (slot=0; slot<3 && !slots[slot].hasSameSurface(refSlot); ++slot)
;
if (slot >= 3)
{
nlwarning("Failed to find point in surface slots");
continue;
}
uint gabarit = slots[slot].gabarit();
nlassert(gabarit <= 2);
++countGabarits[gabarit];
// check out eastern cell and OR east move into neighbour grid
if (pos.x + 1.0 <= _BMax.x)
slots[slot].cellLink().setESlot(getSurfaceAfterMove(refPos, refIndex.getStepE(), gabarit));
else
++xmaxreached;
// check out westtern cell and OR west move into neighbour grid
if (pos.x - 1.0 >= _BMin.x)
slots[slot].cellLink().setWSlot(getSurfaceAfterMove(refPos, refIndex.getStepW(), gabarit));
else
++xminreached;
// check out southern cell and OR south move into neighbour grid
if (pos.y + 1.0 <= _BMax.y)
slots[slot].cellLink().setNSlot(getSurfaceAfterMove(refPos, refIndex.getStepN(), gabarit));
else
++ymaxreached;
// check out northern cell and OR north move into neighbour grid
if (pos.y - 1.0 >= _BMin.y)
slots[slot].cellLink().setSSlot(getSurfaceAfterMove(refPos, refIndex.getStepS(), gabarit));
else
++yminreached;
}
nlinfo("crunchPacsMap: performed %d iterations",count);
}
if (count > 1)
{
//filter();
//checkMap(_WorldMap, true);
COFile f(OutputPath+name+".wmap");
f.serial(_WorldMap);
}
// HOUSEKEEPING
_Container->removePrimitive(_Primitive);
_Container->removePrimitive(_Primitive3);
_Container->removePrimitive(_Primitive5);
if (!keep)
{
release();
}
_WorldMap.clear();
}
void buildGabarit(const string &name, uint gabarit)
{
nlinfo("building gabarit %d map...", gabarit);
_WorldMap.clear();
CIFile f(OutputPath+name+".wmap");
f.serial(_WorldMap);
checkMap(_WorldMap);
filter();
checkMap(_WorldMap, true);
CMapPosition min, max;
_WorldMap.getBounds(min, max);
CMapPosition scan, scanline;
for (scan = min; scan.y() != max.y(); scan = scan.stepCell(0, 1))
{
for (scanline = scan; scanline.x() != max.x(); scanline = scanline.stepCell(1, 0))
{
//scanline = CMapPosition(0x0380, 0x2650);
if (!_WorldMap.exist(scanline))
continue;
const CComputeCell *cc = const_cast<const CWorldMap&>(_WorldMap).getComputeCellCst(scanline);
uint slot, i, j;
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
CMapPosition pos=scanline;
pos.setUnitId(j,i);
TCellUnit &slots = _WorldMap.getCellUnit(pos);
for (slot=0; slot<3; ++slot)
{
CUnitSlot &unitSlot=slots[slot];
// if slot is used but not accessible for my gabarit, reset it
if (unitSlot.getCellLink().used() && unitSlot.gabarit()<gabarit)
{
CSlot currentSlot=unitSlot.getCellLink().ESlot();
if (currentSlot.isValid())
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepE(), currentSlot)).cellLink().setWSlot(CSlot());
currentSlot=unitSlot.getCellLink().WSlot();
if (currentSlot.isValid())
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepW(), currentSlot)).cellLink().setESlot(CSlot());
currentSlot=unitSlot.getCellLink().NSlot();
if (currentSlot.isValid())
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepN(), currentSlot)).cellLink().setSSlot(CSlot());
currentSlot=unitSlot.getCellLink().SSlot();
if (currentSlot.isValid())
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepS(), currentSlot)).cellLink().setNSlot(CSlot());
unitSlot.reset();
}
}
}
}
}
}
checkMap(_WorldMap);
COFile of(OutputPath+name+"_"+toString(gabarit)+".wmap");
of.serial(_WorldMap);
}
void buildCrunchedMap(const string &name)
{
nlinfo("building crunched map...");
_WorldMap.clear();
CIFile f(OutputPath+name+".wmap");
f.serial(_WorldMap);
checkMap(_WorldMap);
buildTopologies();
checkMap(_WorldMap);
nlinfo("Build map...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
CMapPosition scan, scanline;
for (scan = min; scan.y() != max.y(); scan = scan.stepCell(0, 1))
{
for (scanline = scan; scanline.x() != max.x(); scanline = scanline.stepCell(1, 0))
{
//scanline = CMapPosition(0x1731, 0xe2f8);
const CComputeCell *cell = const_cast<const CWorldMap&>(_WorldMap).getComputeCellCst(scanline);
CRootCell *newCell = NULL;
if (cell == NULL)
continue;
CMapPosition pos;
uint i, j;
bool failed = false;
bool white = true;
for (i=0; i<16 && !failed; ++i)
{
// char buff[17];
for (j=0; j<16 && !failed; ++j)
{
bool sl0 = cell->isSlotUsed(pos, CSlot(0));
bool sl1 = cell->isSlotUsed(pos, CSlot(1));
bool sl2 = cell->isSlotUsed(pos, CSlot(2));
pos = scanline;
pos.setUnitId(j,i);
const TCellUnit &cunit = cell->getCellUnitCst(pos);
uint32 used = cell->maxUsedSlot(pos);
if (used>=1)
failed = true;
else if (used>0)
white = false;
if (white && cunit[0].getCellLink().getLinks() != 0)
white = false;
if (!failed && !white && cunit[0].getCellLink().used())
{
if ((!cunit[0].getCellLink().isNSlotValid() && _WorldMap.isSlotUsed(pos.getStepN(), CSlot(0))) ||
(!cunit[0].getCellLink().isSSlotValid() && _WorldMap.isSlotUsed(pos.getStepS(), CSlot(0))) ||
(!cunit[0].getCellLink().isESlotValid() && _WorldMap.isSlotUsed(pos.getStepE(), CSlot(0))) ||
(!cunit[0].getCellLink().isWSlotValid() && _WorldMap.isSlotUsed(pos.getStepW(), CSlot(0))))
failed = true;
}
if (sl0 && (cunit[0].interior() /*|| cunit[0].water()*/ || cunit[0].topology() != 0))
white = false;
}
// buff[16] = '\0';
// nlinfo("Dump: %s", buff);
}
uint32 used;
pos = scanline.getStepS();
for (i=0; i<16 && !failed; ++i, pos = pos.getStepE())
if ((used = _WorldMap.maxUsedSlot(pos)) >= 1)
failed = true;
else if (used != 0)
white = false;
pos = scanline.getStepW();
for (i=0; i<16 && !failed; ++i, pos = pos.getStepN())
if ((used = _WorldMap.maxUsedSlot(pos)) >= 1)
failed = true;
else if (used != 0)
white = false;
pos = scanline.stepCell(0, 1);
for (i=0; i<16 && !failed; ++i, pos = pos.getStepE())
if ((used = _WorldMap.maxUsedSlot(pos)) >= 1)
failed = true;
else if (used != 0)
white = false;
pos = scanline.stepCell(1, 0);
for (i=0; i<16 && !failed; ++i, pos = pos.getStepN())
if ((used = _WorldMap.maxUsedSlot(pos)) >= 1)
failed = true;
else if (used != 0)
white = false;
if (!failed && white)
{
// replace the compute cell by a white cell
CWhiteCell *wcell = new CWhiteCell(_WorldMap);
wcell->setTopologiesNodes(cell->getTopologiesNodes());
// build height map
CFull16x16Layer *heightMap = new CFull16x16Layer();
for (i=0; i<16; ++i)
for (j=0; j<16; ++j)
heightMap->set(i, j, cell->getHeight(_WorldMap.getWorldPositionGeneration(CMapPosition(j, i),CSlot(0))));
wcell->setHeightMap(I16x16Layer::compress(heightMap, 0x7fffffff));
newCell = wcell;
// _WorldMap.setRootCell(scanline, wcell);
}
else if (!failed)
{
// replace the compute cell by a single layer cell
CSingleLayerCell *slcell = new CSingleLayerCell(_WorldMap);
uint maxtopo = 0;
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
pos = scanline;
pos.setUnitId(j,i);
sint maxs = cell->maxUsedSlot(pos);
nlassert(maxs == 0 || maxs == -1);
slcell->setPos(pos, cell->isSlotUsed(pos, CSlot(0)));
uint topology=cell->getTopology(_WorldMap.getWorldPositionGeneration(pos,CSlot(0)));
if (topology>maxtopo)
maxtopo=topology;
}
}
if (maxtopo == 0)
{
slcell->setTopologies(NULL);
}
else if (maxtopo == 1)
{
C1Bit16x16Layer *layer = new C1Bit16x16Layer();
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
pos = scanline;
pos.setUnitId(j,i);
layer->set(i, j, cell->getTopology(_WorldMap.getWorldPositionGeneration(pos,CSlot(0))));
}
}
slcell->setTopologies(layer);
}
else
{
C8Bits16x16Layer *layer = new C8Bits16x16Layer();
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
pos = scanline;
pos.setUnitId(j,i);
layer->set(i, j, cell->getTopology(_WorldMap.getWorldPositionGeneration(pos,CSlot(0))));
}
}
slcell->setTopologies(layer);
}
pos = scanline.getStepS();
for (i=0; i<16; ++i, pos = pos.getStepE())
{
sint maxs = cell->maxUsedSlot(pos);
nlassert(maxs == 0 || maxs == -1);
slcell->setSLink(i, _WorldMap.maxUsedSlot(pos) == 0);
}
pos = scanline.getStepW();
for (i=0; i<16 && !failed; ++i, pos = pos.getStepN())
{
sint maxs = cell->maxUsedSlot(pos);
nlassert(maxs == 0 || maxs == -1);
slcell->setWLink(i, _WorldMap.maxUsedSlot(pos) == 0);
}
pos = scanline.stepCell(0, 1);
for (i=0; i<16 && !failed; ++i, pos = pos.getStepE())
{
sint maxs = cell->maxUsedSlot(pos);
nlassert(maxs == 0 || maxs == -1);
slcell->setNLink(i, _WorldMap.maxUsedSlot(pos) == 0);
}
pos = scanline.stepCell(1, 0);
for (i=0; i<16 && !failed; ++i, pos = pos.getStepN())
{
sint maxs = cell->maxUsedSlot(pos);
nlassert(maxs == 0 || maxs == -1);
slcell->setELink(i, _WorldMap.maxUsedSlot(pos) == 0);
}
slcell->setTopologiesNodes(cell->getTopologiesNodes());
// build height map
CFull16x16Layer *heightMap = new CFull16x16Layer();
for (i=0; i<16; ++i)
for (j=0; j<16; ++j)
heightMap->set(i, j, slcell->isSlotUsed(CMapPosition(j, i), CSlot(0)) ? cell->getHeight(_WorldMap.getWorldPositionGeneration(CMapPosition(j,i),CSlot(0))) :
0x7fffffff);
slcell->setHeightMap(I16x16Layer::compress(heightMap, 0x7fffffff));
newCell = slcell;
// _WorldMap.setRootCell(scanline, slcell);
}
else
{
// build multi layer
CMultiLayerCell *mlcell = new CMultiLayerCell(_WorldMap);
uint32 slot;
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
pos = scanline;
pos.setUnitId(j,i);
for (slot=0; slot<3; ++slot)
{
CWorldPosition wp = _WorldMap.getWorldPositionGeneration(pos,CSlot(slot));
const CUnitSlot &uslot = cell->getUnitSlotCst(wp);
if (uslot.getCellLink().used())
{
mlcell->setLinks(wp, uslot.getCellLink());
mlcell->setTopology(wp, uslot.topology());
}
}
}
}
// build height map
CFull16x16Layer *heightMap[3] = { NULL, NULL, NULL };
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
for (slot=0; slot<3; ++slot)
{
CFull16x16Layer* heightMapPt=heightMap[slot];
if (mlcell->isSlotUsed(CMapPosition(j, i), CSlot(slot)))
{
if (!heightMapPt)
{
heightMapPt=new CFull16x16Layer();
heightMap[slot]=heightMapPt;
}
nlassert(heightMapPt);
heightMapPt->set(i, j, cell->getHeight(_WorldMap.getWorldPositionGeneration(CMapPosition(j, i), CSlot(slot))));
}
else
{
if (heightMapPt)
heightMapPt->set(i, j, 0x7fffffff);
}
}
}
}
for (slot=0; slot<3; ++slot)
if (heightMap[slot] != NULL)
mlcell->setHeightMap(CSlot(slot), I16x16Layer::compress(heightMap[slot], 0x7fffffff));
mlcell->setTopologiesNodes(cell->getTopologiesNodes());
newCell = mlcell;
}
nlassert(newCell != NULL);
pos = scanline;
uint32 slot;
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
pos.setUnitId(j, i);
for (slot=0; slot<3; ++slot)
{
CWorldPosition wpos = _WorldMap.getWorldPositionGeneration(pos, CSlot(slot));
bool sused = cell->isSlotUsed(pos, CSlot(slot)),
nsused = newCell->isSlotUsed(pos, CSlot(slot));
nlassert(sused == nsused);
if (cell->isSlotUsed(pos, CSlot(slot)))
{
CCellLinkage lnk=cell->getCellLink(wpos),
nlnk=newCell->getCellLink(wpos);
uint top=cell->getTopology(wpos),
ntop=newCell->getTopology(wpos);
if (failed || !white)
nlassert(cell->getCellLink(wpos).getLinks() == newCell->getCellLink(wpos).getLinks());
nlassert(cell->getHeight(wpos) == newCell->getHeight(wpos));
nlassert(cell->getTopology(wpos) == newCell->getTopology(wpos));
}
}
}
}
_WorldMap.setRootCell(scanline, newCell);
}
}
checkMap(_WorldMap);
buildMotionLayers();
// nlassert(false);
COFile out(OutputPath+name+".cwmap2");
out.serial(_WorldMap);
}
void checkMap(CWorldMap &wmap, bool fix=false)
{
nlinfo("Checking Wmap link integrity...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
CMapPosition posy;
for (posy=min; posy.y()<max.y(); posy=posy.step(0, 1))
{
CMapPosition posx;
for (posx=posy; posx.x()<max.x(); posx=posx.step(1, 0))
{
uint s;
for (s=0; s<3; ++s)
{
CWorldPosition wpos = wmap.getSafeWorldPosition(posx, CSlot(s));
if (!wpos.isValid())
continue;
CWorldPosition test;
//
if (wpos.getCellLinkage().isNSlotValid())
{
CSlot slot = wpos.getCellLinkage().NSlot();
test = wmap.getSafeWorldPosition(CMapPosition(wpos).getStepN(), slot);
bool failed = true;
if (!test.isValid())
{
//nlwarning("Check: invalid N link at (%04X,%04X,%d)", wpos.x(), wpos.y(), s);
}
else if (!test.getCellLinkage().isSSlotValid())
{
//nlwarning("Check: N slot of (%04X,%04X,%d) has invalid S link", wpos.x(), wpos.y(), s);
if (fix)
{
wmap.getUnitSlot(test).cellLink().setSSlot(CSlot(s));
failed = false;
}
}
else if (test.getCellLinkage().SSlot().slot() != s)
{
//nlwarning("Check: N slot of (%04X,%04X,%d) points to another slot at S", wpos.x(), wpos.y(), s);
}
else
{
failed = false;
}
if (failed && fix)
wmap.resetUnitSlotNLink(wpos);
}
//
if (wpos.getCellLinkage().isSSlotValid())
{
CSlot slot = wpos.getCellLinkage().SSlot();
test = wmap.getSafeWorldPosition(CMapPosition(wpos).getStepS(), slot);
bool failed = true;
if (!test.isValid())
{
//nlwarning("Check: invalid S link at (%04X,%04X,%d)", wpos.x(), wpos.y(), s);
}
else if (!test.getCellLinkage().isNSlotValid())
{
//nlwarning("Check: S slot of (%04X,%04X,%d) has invalid N link", wpos.x(), wpos.y(), s);
if (fix)
{
wmap.getUnitSlot(test).cellLink().setNSlot(CSlot(s));
failed = false;
}
}
else if (test.getCellLinkage().NSlot().slot() != s)
{
//nlwarning("Check: S slot of (%04X,%04X,%d) points to another slot at N", wpos.x(), wpos.y(), s);
}
else
{
failed = false;
}
if (failed && fix)
wmap.resetUnitSlotSLink(wpos);
}
//
if (wpos.getCellLinkage().isESlotValid())
{
CSlot slot = wpos.getCellLinkage().ESlot();
test = wmap.getSafeWorldPosition(CMapPosition(wpos).getStepE(), slot);
bool failed = true;
if (!test.isValid())
{
//nlwarning("Check: invalid E link at (%04X,%04X,%d)", wpos.x(), wpos.y(), s);
}
else if (!test.getCellLinkage().isWSlotValid())
{
//nlwarning("Check: E slot of (%04X,%04X,%d) has invalid W link", wpos.x(), wpos.y(), s);
if (fix)
{
wmap.getUnitSlot(test).cellLink().setWSlot(CSlot(s));
failed = false;
}
}
else if (test.getCellLinkage().WSlot().slot() != s)
{
//nlwarning("Check: E slot of (%04X,%04X,%d) points to another slot at W", wpos.x(), wpos.y(), s);
}
else
{
failed = false;
}
if (failed && fix)
wmap.resetUnitSlotELink(wpos);
}
//
if (wpos.getCellLinkage().isWSlotValid())
{
CSlot slot = wpos.getCellLinkage().WSlot();
test = wmap.getSafeWorldPosition(CMapPosition(wpos).getStepW(), slot);
bool failed = true;
if (!test.isValid())
{
//nlwarning("Check: invalid W link at (%04X,%04X,%d)", wpos.x(), wpos.y(), s);
}
else if (!test.getCellLinkage().isESlotValid())
{
//nlwarning("Check: W slot of (%04X,%04X,%d) has invalid E link", wpos.x(), wpos.y(), s);
if (fix)
{
wmap.getUnitSlot(test).cellLink().setESlot(CSlot(s));
failed = false;
}
}
else if (test.getCellLinkage().ESlot().slot() != s)
{
//nlwarning("Check: W slot of (%04X,%04X,%d) points to another slot at E", wpos.x(), wpos.y(), s);
}
else
{
failed = false;
}
if (failed && fix)
wmap.resetUnitSlotWLink(wpos);
}
}
}
}
}
//
void clearHeightMap(const string &name, uint gabarit)
{
nlinfo("clearing height map for '%s'", (name+"_"+toString(gabarit)+".cwmap2").c_str());
// nlassert(false);
_WorldMap.clear();
CIFile fi(OutputPath+name+"_"+toString(gabarit)+".cwmap2");
fi.serial(_WorldMap);
fi.close();
_WorldMap.clearHeightMap();
COFile fo(OutputPath+name+"_"+toString(gabarit)+".cwmap2");
fo.serial(_WorldMap);
fo.close();
}
//
void filter()
{
nlinfo("filtering...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
CMapPosition scanpos = min;
uint x, y;
uint scanWidth = max.x()-min.x();
uint scanHeight = max.y()-min.y();
for (y=0; y<scanHeight; ++y)
{
CMapPosition pos(scanpos);
for (x=0; x<scanWidth; ++x)
{
if (_WorldMap.exist(pos))
{
TCellUnit &slots = _WorldMap.getCellUnit(pos);
if (slots[0].used() && slots[1].used() && !slots[2].used() && abs(sint(slots[0].height())-sint(slots[1].height()))<2)
{
CMapPosition np;
if ((!_WorldMap.exist(np = pos.getStepS()) || _WorldMap.nbUsedSlots(np) <= 1) &&
(!_WorldMap.exist(np = pos.getStepN()) || _WorldMap.nbUsedSlots(np) <= 1) &&
(!_WorldMap.exist(np = pos.getStepE()) || _WorldMap.nbUsedSlots(np) <= 1) &&
(!_WorldMap.exist(np = pos.getStepW()) || _WorldMap.nbUsedSlots(np) <= 1))
{
slots[1].reset();
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepW(),CSlot(0))).cellLink().setESlot(CSlot(0));
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepE(),CSlot(0))).cellLink().setWSlot(CSlot(0));
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepS(),CSlot(0))).cellLink().setNSlot(CSlot(0));
_WorldMap.getUnitSlot(_WorldMap.getWorldPosition(pos.getStepN(),CSlot(0))).cellLink().setSSlot(CSlot(0));
}
}
}
pos = pos.getStepE();
}
scanpos = scanpos.getStepN();
}
}
//
void buildTopologies()
{
// nlassert(false);
nlinfo("building topologies...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
CMapPosition scan, scanline;
uint maxtopo = 0;
vector<uint> toposcount;
struct CTopoGrid
{
sint getTopo(const CSlot &slot) const
{
return topos[slot.slot()];
}
sint topos[3];
bool testGrid;
};
CTopoGrid toposGridList[16][16];
for (scan = min; scan.y() != max.y(); scan = scan.stepCell(0, 1))
{
for (scanline = scan; scanline.x() != max.x(); scanline = scanline.stepCell(1, 0))
{
if (!_WorldMap.exist(scanline))
continue;
//scanline = CMapPosition(0x02e0, 0xd440);
CComputeCell *cell = _WorldMap.getComputeCell(scanline);
{
for (uint i = 0; i < 16; ++i)
{
for (uint j = 0; j < 16; ++j)
{
toposGridList[i][j].topos[0] =
toposGridList[i][j].topos[1] =
toposGridList[i][j].topos[2] = -1;
}
}
}
uint i = 0; // current topo
CMapPosition sp = scanline;
sp.setUnitId(0,0);
uint spx, spy, spslot;
for (spy=0; spy<16; ++spy)
{
for (spx=0; spx<16; ++spx)
{
sp.setUnitId(spx, spy);
for (spslot=0; spslot<3; ++spslot)
{
if (!cell->isSlotUsed(sp, CSlot(spslot)))
continue;
CTopoGrid &topoGrid=toposGridList[sp.yCoord().getUnitId()][sp.xCoord().getUnitId()];
sint &curTopos=topoGrid.topos[spslot];
if (curTopos<0)
{
{ // -- clean flood fill table --
uint gi, gj;
for (gi=0; gi<16; ++gi)
for (gj=0; gj<16; ++gj)
toposGridList[gi][gj].testGrid=false;
} // ----------------------------
topoGrid.testGrid=true;
curTopos=i;
CUnitSlot &unitslot = _WorldMap.getUnitSlot(_WorldMap.getWorldPosition(sp,CSlot(spslot)));
unitslot.setTopology(i);
bool interior = unitslot.interior();
bool water = unitslot.water();
bool nogo = unitslot.nogo();
CVector totalPos(0.0f, 0.0f, 0.0f);
uint totalElm = 0;
vector<CWorldPosition> stack;
stack.push_back(_WorldMap.getWorldPosition(sp, CSlot(spslot)));
bool topoHasNeighb = false;
while (!stack.empty())
{
CWorldPosition wp(stack.back());
stack.pop_back();
totalPos+=CVector(wp.toVectorD());
++totalElm;
static const CDirection::TDirection dirs[] =
{
CDirection::W,
CDirection::E,
CDirection::S,
CDirection::N
};
for (uint dir=0; dir<4; ++dir)
{
const CDirection direction=CDirection(dirs[dir]);
CWorldPosition tm(wp);
uint x = tm.xCoord().getUnitId()+direction.dx();
uint y = tm.yCoord().getUnitId()+direction.dy();
//InfoLog->displayRaw("Test move from %4X,%4X,%d to %4X,%4X", tm.x(), tm.y(), tm.slot(), tm.x()+direction.dx(), tm.y()+direction.dy());
// store if move succeded
bool mvres = _WorldMap.move(tm, direction);
/*
bool test1 = false;
bool test2 = false;
bool test3 = false;
bool test4 = false;
bool test5 = false;
bool test6 = false;
bool test7 = false;
// check move is on same cell, and position hasn't been visited (nor in same topo or in another)
// and moves in same kind of floor
if ( (test1 = ((x&0xf0)==0))
&& (test2 = ((y&0xf0)==0))
&& (test3 = (!toposGridList[y][x].testGrid))
&& (test4 = (mvres))
&& (test5 = (toposGridList[y][x].topos[tm.slot()]<0))
&& (test6 = (_WorldMap.getUnitSlot(tm).interior() == interior))
&& (test7 = (_WorldMap.getUnitSlot(tm).water() == water)))
*/
if ( ((x&0xf0)==0)
&& ((y&0xf0)==0)
&& (!toposGridList[y][x].testGrid)
&& (mvres)
&& (toposGridList[y][x].topos[tm.slot()]<0)
&& (_WorldMap.getUnitSlot(tm).interior() == interior)
&& (_WorldMap.getUnitSlot(tm).water() == water)
&& (_WorldMap.getUnitSlot(tm).nogo() == nogo) )
{
toposGridList[y][x].testGrid=true;
toposGridList[y][x].topos[tm.slot()] = i;
_WorldMap.getUnitSlot(tm).setTopology(i);
stack.push_back(tm);
// fakes move result so if on same topo, doesn't add a neighbour
mvres = false;
//InfoLog->displayRawNL(" SUCCESS");
}
else
{
//InfoLog->displayRawNL(" FAILED (test1=%d test2=%d test3=%d test4=%d test5=%d test6=%d test7=%d)", test1, test2, test3, test4, test5, test6, test7);
}
if (mvres)
topoHasNeighb = true;
}
}
if (topoHasNeighb)
{
CTopology &topology = cell->getTopologyNode(i);
topology.Id = CTopology::TTopologyId(scanline, i);
topology.Position = totalPos/(float)totalElm;
topology.Flags = (interior ? Interior : 0) | (water ? Water : 0) | (nogo ? NoGo : 0);
//nlinfo("Topology %08X %04X", topology.Id.getVal(), topology.Flags);
++i;
}
else
{
nlinfo("Unactivated topo %d in cell (%4X,%4X)",i, scanline.x()&0xffff, scanline.y()&0xffff);
uint i, j, s;
for (i=0; i<16; ++i)
{
for (j=0; j<16; ++j)
{
for (s=0; s<3; ++s)
{
if (toposGridList[i][j].topos[s] == (sint)i)
{
CMapPosition errp = scanline;
errp.setUnitId(j, i);
_WorldMap.resetUnitSlot(_WorldMap.getWorldPosition(errp, CSlot(s)));
toposGridList[i][j].topos[s] = 0xffff;
}
}
}
}
}
}
}
}
}
if (i>50)
nlstop;
if (i>maxtopo)
maxtopo = i;
if (i >= toposcount.size())
toposcount.resize(i+1);
toposcount[i]++;
}
}
nlinfo("Found %d topologies maximum", maxtopo);
uint i;
for (i=0; i<toposcount.size(); ++i)
nlinfo("%d cells of %d topologies", toposcount[i], i);
}
class CMotionTrace
{
public:
CMotionTrace() : dx(0), dy(0), distance(0xffffffff), flags(0)
{
}
sint8 dx;
sint8 dy;
uint32 distance;
uint8 flags;
};
//
void buildMotionLayers()
{
nlinfo("building motion layers...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
CMapPosition scan, scanline;
uint compute = 0, white = 0, simple = 0, multi = 0, other = 0;
_WorldMap.countCells(compute, white, simple, multi, other);
uint total = compute+white+simple+multi+other;
uint compCells = 0;
TTime startTime = CTime::getLocalTime();
//
//min = CMapPosition(0x3a40, 0x8330);
//max = CMapPosition(0x3a50, 0x8340);
for (scan = min; scan.y() != max.y(); scan = scan.stepCell(0, 1))
{
for (scanline = scan; scanline.x() != max.x(); scanline = scanline.stepCell(1, 0))
{
if (CTime::getLocalTime() - startTime > 5000)
{
startTime = CTime::getLocalTime();
nlinfo("buildMotionLayers: %d pct done...", compCells*100/total);
}
//
//scanline = CMapPosition(0x3A40, 0x8330);
if (!_WorldMap.exist(scanline))
continue;
//if (scanline.x() == 4436 && scanline.y() == -4240)
// nlstop;
++compCells;
CRootCell *cell = _WorldMap.getRootCell(scanline);
uint topo = 0;
uint maxtopo = 0;
const CMapPosition mincell = scanline.stepCell(-1, -1),
maxcell = scanline.stepCell( 2, 2);
while (true)
{
CMotionTrace fillGrid[16*3][16*3][3];
bool found = false;
uint stx = 0, sty = 0;
uint32 stsl=0;
CMapPosition sp = scanline;
const CMapPosition soffset = sp.stepCell(-1, -1);
set<CTopology::TTopologyRef> visited;
const CTopology::TTopologyId startTId(scanline, topo);
for (sty=0; sty<16; ++sty)
{
for (stx=0; stx<16; ++stx)
{
sp.setUnitId(stx,sty);
for (stsl=0; stsl<3; ++stsl)
{
if (!cell->isSlotUsed(sp, CSlot(stsl)))
continue;
CWorldPosition stwp = _WorldMap.getWorldPosition(sp, CSlot(stsl));
const uint topology=cell->getTopology(stwp);
if (topology>maxtopo && topology<0x80)
maxtopo=topology;
// find a point in the current topo
if (topology==topo && fillGrid[sty+16][stx+16][stsl].distance == 0xffffffff)
{
// this point exists, is not marked in fill grid and belongs to the current topo
found = true;
//
// HERE: use multimap instead of vector so the position stack is always sorted, and thus
// the flood fill should be minimal (Dijkstra like graph route)
//
//vector<CWorldPosition> stack;
multimap<sint, CWorldPosition> stacks[5];
// mark start point
CMotionTrace &first = fillGrid[sp.y()-soffset.y()][sp.x()-soffset.x()][stsl];
first.dx = 0;
first.dy = 0;
first.distance = 0;
const CTopology& topNode = stwp.getTopologyRef().getCstTopologyNode();
first.flags = (topNode.isInWater() ? 1 : 0) + (topNode.isInNogo() ? 2 : 0);
// push first position
stacks[first.flags].insert(std::pair<sint, CWorldPosition>(first.flags, stwp));
while (true)
{
uint stack;
for (stack=0; stack<5; ++stack)
if (!stacks[stack].empty())
break;
if (stack == 5)
break;
CWorldPosition next((*(stacks[stack].begin())).second);
stacks[stack].erase(stacks[stack].begin());
const CTopology& nextNode = next.getTopologyRef().getCstTopologyNode();
uint8 nextflags = (nextNode.isInWater() ? 1 : 0) + (nextNode.isInNogo() ? 2 : 0);
CTopology::TTopologyRef nextTId(next);
sint cdist = fillGrid[next.y()-soffset.y()][next.x()-soffset.x()][next.slot()].distance;
if ( nextTId != startTId
&& cdist<=CDirection::MAX_COST) // check if it is a touching topology
visited.insert(nextTId);
uint dir;
for (dir=0; dir<8; ++dir)
{
CWorldPosition tmp(next);
CDirection direction((CDirection::TDirection)dir);
if (_WorldMap.moveSecure(tmp,direction))
{
// don't move more than 1 cell
if ( tmp.x() < mincell.x()
|| tmp.x() >= maxcell.x()
|| tmp.y() < mincell.y()
|| tmp.y() >= maxcell.y())
continue;
// check we can move back
CDirection opp = direction;
opp.addStep(CDirection::HALF_TURN);
CWorldPosition checkBackMove(tmp);
if (!_WorldMap.move(checkBackMove, opp))
{
nlwarning("Can't move back from pos (%d,%d,%d) to pos (%d,%d,%d)",
next.x(), next.y(), next.slot(),
tmp.x(), tmp.y(), tmp.slot());
continue;
}
else if (checkBackMove != next)
{
if (Verbose)
nlwarning("Reverse move from pos (%d,%d,%d) to pos (%d,%d,%d) gave a different path",
next.x(), next.y(), next.slot(),
tmp.x(), tmp.y(), tmp.slot());
continue;
}
const CTopology& tmpNode = tmp.getTopologyRef().getCstTopologyNode();
uint8 tmpflags = (tmpNode.isInWater() ? 1 : 0) + (tmpNode.isInNogo() ? 2 : 0);
/*
if (tmpNode.Id != nextNode.Id)
nlinfo("tamere");
if (tmpflags != 0)
nlinfo("blabla");
*/
uint ndist = (_WorldMap.getTopology(tmp)==topo && tmp.hasSameFullCellId(sp) ) ? 0 : cdist+direction.getWeight();
CMotionTrace& motionTrace = fillGrid[tmp.y()-soffset.y()][tmp.x()-soffset.x()][tmp.slot()];
uint forbid = nextflags & (~tmpflags);
if ((motionTrace.distance == 0xffffffff) ||
(forbid == 0 && motionTrace.distance > ndist))
{
/*nlinfo("stack=%d ndist=%d move from (%04X,%04X,%d - topo=%08X,flags=%d) to (%04X,%04X,%d - topo=%08X,flags=%d) - %d %d %d %d",
stack, ndist,
next.x(), next.y(), next.slot(), nextNode.Id.getVal(), nextflags,
tmp.x(), tmp.y(), tmp.slot(), tmpNode.Id.getVal(), tmpflags,
stacks[0].size(), stacks[1].size(), stacks[2].size(), stacks[3].size());*/
motionTrace.distance = ndist;
if (ndist == 0)
{
motionTrace.dx =
motionTrace.dy = 0;
}
else
{
motionTrace.dx=-direction.dx();
motionTrace.dy=-direction.dy();
}
if (tmp.getTopologyNode().Id == topNode.Id)
{
stacks[0].insert(std::pair<sint, CWorldPosition>(ndist, tmp));
}
else
{
stacks[tmpflags + 1].insert(std::pair<sint, CWorldPosition>(ndist, tmp));
}
}
}
}
}
}
}
}
}
if (found)
{
// generate direction map for this topo
CDirectionMap *dmap = new CDirectionMap();
uint x, y, s;
for (s=0; s<3; ++s)
{
CDirectionLayer* directionLayer=dmap->Layers[s];
for (y=0; y<16*3; ++y)
{
// char output[256];
for (x=0; x<16*3; ++x)
{
// static const char ht[]= "0123456789ABCDEF";
CMotionTrace &motionTrace=fillGrid[y][x][s];
// output[x] = ht[motionTrace.distance&15];
if (motionTrace.distance != 0xffffffff)
{
// create CDirectionLayer if it do not exists
if (!directionLayer)
{
directionLayer = new CDirectionLayer();
nlassert(directionLayer!=NULL);
dmap->Layers[s] = directionLayer;
}
CGridDirectionLayer* dirLayer=directionLayer->getGridLayer(y>>4,x>>4);
// create CGridDirectionLayer if it do not exists
if (!dirLayer)
{
directionLayer->setGridLayer(y>>4,x>>4, new C4Bits16x16Layer());
dirLayer=directionLayer->getGridLayer(y>>4,x>>4);
nlassert(dirLayer);
}
dirLayer->setDirection(y&15, x&15, (motionTrace.dx == 0 && motionTrace.dy == 0 ? CDirection(CDirection::UNDEFINED) : CDirection(motionTrace.dx,motionTrace.dy)));
}
}
// output[x] = '\0';
// nlinfo("%s", output);
}
}
for (s=0; s<3; ++s)
{
CDirectionLayer* directionLayer=dmap->Layers[s];
if (!directionLayer)
continue;
for (y=0; y<3; ++y)
{
for (x=0; x<3; ++x)
{
CGridDirectionLayer *layer=directionLayer->getGridLayer(y,x);
if (layer==NULL)
continue;
uint i, j;
uint cmotion = 15;
bool different = false;
for (i=0; i<16 && !different; ++i)
{
for (j=0; j<16 && !different; ++j)
{
uint nm = layer->get(i, j);
if (cmotion != 15 && nm != 15 && nm != cmotion)
{
different = true;
}
else if (nm != 15)
{
cmotion = layer->get(i, j);
}
}
}
if (!different)
{
CWhite16x16Layer *nlayer = new CWhite16x16Layer();
nlayer->set (0, 0, cmotion);
delete directionLayer->getGridLayer(y,x);
directionLayer->setGridLayer(y,x,nlayer);
}
}
}
}
// dmap->dump();
cell->setDirectionMap(dmap, topo);
CTopology &topology = cell->getTopologyNode(topo);
topology.Id = CTopology::TTopologyId(scanline, topo);
topology.Neighbours.clear();
set<CTopology::TTopologyRef>::iterator it;
for (it=visited.begin(); it!=visited.end(); ++it)
{
CTopology::TTopologyRef topId(*it);
const CTopology &node = topId.getCstTopologyNode(); // _WorldMap.getTopologyNode(topId)
float Norm=(topology.Position-node.Position).norm();
topology.Neighbours.push_back(CTopology::CNeighbourLink(topId, Norm));
}
}
else if (Verbose)
{
nlwarning("Topo %d not found, probably not accessible, left unchecked", topo);
}
++topo;
if (topo > maxtopo)
break;
}
}
}
for (scan = min; scan.y() != max.y(); scan = scan.stepCell(0, 1))
{
for (scanline = scan; scanline.x() != max.x(); scanline = scanline.stepCell(1, 0))
{
if (!_WorldMap.exist(scanline))
continue;
CRootCell *cell = _WorldMap.getRootCell(scanline);
if (cell == NULL)
continue;
vector<CTopology> &tops = cell->getTopologiesNodes();
uint i, j;
// for all topologies in cell
for (i=0; i<tops.size(); ++i)
{
CTopology &node = tops[i];
vector<CTopology::CNeighbourLink>::iterator it;
// look for all neighbours
for (it=node.Neighbours.begin(); it!=node.Neighbours.end();)
{
// and check that it is a neighbour of the neighbour
const CTopology &neighb = (*it).getTopologyRef().getCstTopologyNode();
for (j=0; j<neighb.Neighbours.size(); ++j)
if (neighb.Neighbours[j].getTopologyRef() == node.Id)
break;
if (j == neighb.Neighbours.size())
{
if (Verbose)
nlwarning("Non bijective link between topology %08X and topology %08X, fixed", node.Id.getVal(), neighb.Id.getVal());
it = node.Neighbours.erase(it);
}
else
++it;
}
}
}
}
_WorldMap.checkMotionLayer();
_WorldMap.countSuperTopo();
_WorldMap.buildMasterTopo(false, false);
_WorldMap.buildMasterTopo(true, false);
_WorldMap.buildMasterTopo(false, true);
_WorldMap.buildMasterTopo(true, true);
}
//
void buildBMP(const string &name)
{
// nlassert(false);
nlinfo("building bitmap...");
uint compute = 0, white = 0, simple = 0, multi = 0, other = 0;
_WorldMap.countCells(compute, white, simple, multi, other);
uint total = compute+white+simple+multi+other;
nlinfo("%d cells : compute=%d, white=%d, simple=%d, multi=%d, other=%d", total, compute, white, simple, multi, other);
nlinfo("Build bmp...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
uint scanWidth = max.x()-min.x();
uint scanHeight = max.y()-min.y();
uint imageWidth = (scanWidth+15)&~15;
uint imageHeight = (scanHeight);
CBMP4Image<2,2>::SHdr imageHdr(imageWidth, imageHeight);
CBMP4Image<2,2>::SPalette imagePalette;
FILE *outf = fopen((OutputPath+name+".bmp").c_str(),"wb");
if (outf == NULL)
return;
uint16 BM = 0x4D42;
imagePalette.setupForCol();
fwrite((void *)&BM, 1, sizeof(BM), outf);
fwrite((void *)&imageHdr, 1, sizeof(imageHdr), outf);
fwrite((void *)&imagePalette, 1, sizeof(imagePalette), outf);
uint8 *lineBuffer = new uint8[imageWidth/2];
memset(lineBuffer, 255, imageWidth/2);
CMapPosition scanpos(min.x(),min.y());
uint x, y;
const CWorldMap *wmap = &_WorldMap;
for (y=0; y<scanHeight; ++y)
{
uint8 *linePtr = lineBuffer;
uint8 pointBuffer;
CMapPosition pos(scanpos);
for (x=0; x<scanWidth; ++x)
{
uint8 color = 0;
sint32 colorHM = -32768;
uint32 slot=0;
const CRootCell *cell = wmap->getRootCellCst(pos);
if (cell != NULL)
{
uint ns;
for (ns=0; ns<3; ++ns)
{
CWorldPosition wp = _WorldMap.getSafeWorldPosition(pos, CSlot(ns));
if (!wp.isValid())
continue;
++slot;
CCellLinkage links = cell->getCellLink(wp);
sint32 height = cell->getHeight(wp);
uint8 cl = 0;
if (!links.isESlotValid()) ++cl;
if (!links.isWSlotValid()) ++cl;
if (!links.isSSlotValid()) ++cl;
if (!links.isNSlotValid()) ++cl;
if (cl > color)
color = cl;
if (height > colorHM)
colorHM = height;
}
}
pointBuffer = ((pointBuffer<<4) | (color+((uint8)slot<<2)));
if (x & 1)
{
*(linePtr++) = pointBuffer;
}
pos = pos.getStepE();
}
scanpos = scanpos.getStepN();
fwrite((void*)lineBuffer, 1, imageWidth/2, outf);
}
fclose(outf);
delete [] lineBuffer;
}
//
void buildBitMap(const string &name)
{
// nlassert(false);
nlinfo("building bitmap...");
_WorldMap.clear();
string ext = CFile::getExtension(name);
if (ext == "")
ext = "cwmap2";
CIFile f(OutputPath+CFile::getFilenameWithoutExtension(name)+"."+ext);
f.serial(_WorldMap);
uint compute = 0, white = 0, simple = 0, multi = 0, other = 0;
_WorldMap.countCells(compute, white, simple, multi, other);
uint total = compute+white+simple+multi+other;
nlinfo("%d cells : compute=%d, white=%d, simple=%d, multi=%d, other=%d", total, compute, white, simple, multi, other);
nlinfo("Build bmp...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
uint scanWidth = max.x()-min.x();
uint scanHeight = max.y()-min.y();
uint imageWidth = (scanWidth+15)&~15;
uint imageHeight = (scanHeight);
CBMP4Image<2,2>::SHdr imageHdr(imageWidth, imageHeight);
CBMP4Image<2,2>::SPalette imagePalette;
FILE *outf = fopen((OutputPath+name+".bmp").c_str(),"wb");
FILE *outfh = fopen((OutputPath+name+"_hm.bmp").c_str(),"wb");
if (outf == NULL)
return;
uint16 BM = 0x4D42;
imagePalette.setupForCol();
fwrite((void *)&BM, 1, sizeof(BM), outf);
fwrite((void *)&imageHdr, 1, sizeof(imageHdr), outf);
fwrite((void *)&imagePalette, 1, sizeof(imagePalette), outf);
imagePalette.setupHueCircle();
fwrite((void *)&BM, 1, sizeof(BM), outfh);
fwrite((void *)&imageHdr, 1, sizeof(imageHdr), outfh);
fwrite((void *)&imagePalette, 1, sizeof(imagePalette), outfh);
uint8 *lineBuffer = new uint8[imageWidth/2];
memset(lineBuffer, 255, imageWidth/2);
uint8 *lineBufferHM = new uint8[imageWidth/2];
memset(lineBufferHM, 255, imageWidth/2);
CMapPosition scanpos(min.x(),min.y());
uint x, y;
const CWorldMap *wmap = &_WorldMap;
CTGAImage tgaImage;
tgaImage.setup((uint16)imageWidth, (uint16)imageHeight, OutputPath+name+".tga", (uint16)min.x(), (uint16)-max.y());
tgaImage.setupForCol();
for (y=0; y<scanHeight; ++y)
{
uint8 *linePtr = lineBuffer;
uint8 *linePtrHM = lineBufferHM;
uint8 pointBuffer = 0;
uint8 pointBufferHM = 0;
CMapPosition pos(scanpos);
for (x=0; x<scanWidth; ++x)
{
uint8 color = 0;
sint32 colorHM = -32768;
uint32 slot=0;
const CRootCell *cell = wmap->getRootCellCst(pos);
if (cell != NULL)
{
uint ns;
for (ns=0; ns<3; ++ns)
{
CWorldPosition wp = _WorldMap.getSafeWorldPosition(pos, CSlot(ns));
if (!wp.isValid())
continue;
++slot;
CCellLinkage links = cell->getCellLink(wp);
sint32 height = cell->getHeight(wp);
uint8 cl = 0;
if (!links.isESlotValid()) ++cl;
if (!links.isWSlotValid()) ++cl;
if (!links.isSSlotValid()) ++cl;
if (!links.isNSlotValid()) ++cl;
if (cl > color)
color = cl;
if (height > colorHM)
colorHM = height;
}
}
tgaImage.set(x, color+((uint8)slot<<2));
pointBuffer = ((pointBuffer<<4) | (color+((uint8)slot<<2)));
pointBufferHM = ((pointBufferHM<<4) | (colorHM & 0xf));
if (x & 1)
{
*(linePtr++) = pointBuffer;
*(linePtrHM++) = pointBufferHM;
}
pos = pos.getStepE();
}
scanpos = scanpos.getStepN();
fwrite((void*)lineBuffer, 1, imageWidth/2, outf);
fwrite((void*)lineBufferHM, 1, imageWidth/2, outfh);
tgaImage.writeLine();
}
fclose(outf);
fclose(outfh);
delete [] lineBuffer;
delete [] lineBufferHM;
_WorldMap.clear();
}
// build a height map of sint16. Full path should be given for the input cw_map2
void buildHeightMap16(const string &name)
{
// nlassert(false);
nlinfo("building heightmap for %s", name.c_str());
_WorldMap.clear();
string ext = CFile::getExtension(name);
if (ext == "")
ext = "cw_map2";
CIFile f(CFile::getPath(name) + CFile::getFilenameWithoutExtension(name)+"."+ext);
f.serial(_WorldMap);
uint compute = 0, white = 0, simple = 0, multi = 0, other = 0;
_WorldMap.countCells(compute, white, simple, multi, other);
uint total = compute+white+simple+multi+other;
nlinfo("%d cells : compute=%d, white=%d, simple=%d, multi=%d, other=%d", total, compute, white, simple, multi, other);
nlinfo("Build bmp...");
CMapPosition min, max;
_WorldMap.getBounds(min, max);
uint scanWidth = max.x()-min.x();
uint scanHeight = max.y()-min.y();
CMapPosition scanpos(min.x(),min.y());
uint x, y;
const CWorldMap *wmap = &_WorldMap;
CArray2D<sint16> heightMap;
heightMap.init(scanWidth, scanHeight);
sint16 *dest = &heightMap(0, 0);
for (y=0; y<scanHeight; ++y)
{
CMapPosition pos(scanpos);
for (x=0; x<scanWidth; ++x)
{
sint16 height = 0x7fff;
const CRootCell *cell = wmap->getRootCellCst(pos);
if (cell != NULL)
{
uint ns;
for (ns=0; ns<3; ++ns)
{
CWorldPosition wp = _WorldMap.getSafeWorldPosition(pos, CSlot(ns));
if (!wp.isValid())
continue;
sint32 newHeight = cell->getHeight(wp);
if (newHeight < (sint16) height)
height = (sint16) newHeight;
}
}
*dest ++ = height;
pos = pos.getStepE();
}
scanpos = scanpos.getStepN();
}
// write result
COFile output(OutputPath + CFile::getFilenameWithoutExtension(name)+".cw_height");
sint32 xmin = (sint32) min.x();
sint32 xmax = (sint32) max.x();
sint32 ymin = (sint32) (sint16) min.y();
sint32 ymax = (sint32) (sint16) max.y();
output.serialCheck(NELID("OBSI"));
output.serial(xmin);
output.serial(xmax);
output.serial(ymin);
output.serial(ymax);
output.serial(heightMap);
// TMP TMP
/*
CBitmap tgaHM;
tgaHM.resize(heightMap.getWidth(), heightMap.getHeight());
sint16 hMax = -32768;
sint16 hMin= 32767;
uint numPix = heightMap.getWidth() * heightMap.getHeight();
for(CArray2D<sint16>::iterator it = heightMap.begin(); it != heightMap.end(); ++it)
{
if (*it != 32767)
{
hMax = std::max(hMax, *it);
hMin = std::min(hMin, *it);
}
}
float scale = 255.f / favoid0((float) (hMax - hMin));
float bias = (float) - hMin;
CRGBA *destCol = (CRGBA *) &tgaHM.getPixels()[0];
for(CArray2D<sint16>::iterator it = heightMap.begin(); it != heightMap.end(); ++it)
{
if (*it == 0x7fff)
{
*destCol++ = CRGBA::Magenta;
}
else
{
float height = scale * ((*it) + bias);
clamp(height, 0.f, 255.f);
*destCol++ = CRGBA((uint8) height, (uint8) height, (uint8) height);
}
}
//
COFile tgaOut("d:\\tmp\\whole_world_height_map.tga");
tgaHM.writeTGA(tgaOut, 0, true);
*/
//
_WorldMap.clear();
}
//
void findPath(const string &name, CVectorD start, CVectorD end)
{
_WorldMap.clear();
CIFile f(OutputPath+name+".cwmap2");
f.serial(_WorldMap);
CAStarPath path;
path.setStartPos(_WorldMap.getWorldPosition(start));
path.setEndPos(_WorldMap.getWorldPosition(end));
_WorldMap.findAStarPath(path.getStartPos(), path.getEndPos(), path.topologiesPathForCalc());
uint step=0;
CWorldPosition cpos(_WorldMap.getWorldPosition(start));
while (_WorldMap.move(cpos, path, step))
{
CVectorD dpos = cpos.getPosition(); // _WorldMap.getPosition(cpos)
CMapPosition mpos = cpos;
nldebug("Move: (%.1f,%.1f,%.1f)/(%04x,%04x)", dpos.x, dpos.y, dpos.z, mpos.x(), mpos.y());
}
_WorldMap.clear();
}
//
void findInsidePath(const string &name, CVectorD start, CVectorD end)
{
_WorldMap.clear();
CIFile f(OutputPath+name+".cwmap2");
f.serial(_WorldMap);
CInsideAStarPath path;
path.setStartPos(_WorldMap.getWorldPosition(start));
path.setEndPos(_WorldMap.getWorldPosition(end));
_WorldMap.findInsideAStarPath(path.getStartPos(), path.getEndPos(), path.getStepPathForCalc());
uint step=0;
/*
CWorldPosition cpos = _WorldMap.getWorldPosition(start);
while (_WorldMap.move(cpos, path, step))
{
CVectorD dpos = _WorldMap.getWorldPosition(cpos);
CMapPosition mpos = cpos.getMapPosition();
nldebug("Move: (%.1f,%.1f,%.1f)/(%04x,%04x)", dpos.x, dpos.y, dpos.z, mpos.x.c, mpos.y.c);
}
*/
_WorldMap.clear();
}
//
void testMove(const string &name, CVectorD start, CVectorD end)
{
_WorldMap.clear();
CIFile f(OutputPath+name+".cwmap2");
f.serial(_WorldMap);
CWorldPosition pos(_WorldMap.getWorldPosition(start));
CMapPosition towards = _WorldMap.getWorldPosition(end);
_WorldMap.move (pos,towards, Nothing);
_WorldMap.clear ();
}
};
CPacsCruncher::TPacsPrimMap CPacsCruncher::_PacsPrimMap;
NLMISC_COMMAND(setStartPoint,"Set the start point for a continent","<continent> <startx> <starty> [startz]")
{
if (args.size() < 3)
return false;
CVectorD startPoint;
NLMISC::fromString(args[1], startPoint.x);
NLMISC::fromString(args[2], startPoint.y);
if (args.size() < 4)
{
startPoint.z = 0.0;
}
else
{
NLMISC::fromString(args[3], startPoint.z);
}
StartPoints.insert(multimap<string, CVectorD>::value_type(args[0], startPoint));
nlinfo("Set start point (%.1f,%.1f,%.1f) for continent '%s'", startPoint.x, startPoint.y, startPoint.z, args[0].c_str());
return true;
}
NLMISC_COMMAND(setBoundingBox, "Set the working bounding box", "<minx> <miny> <maxx> <maxy>")
{
if (args.size() < 4)
return false;
NLMISC::fromString(args[0], BoxMin.x);
NLMISC::fromString(args[1], BoxMin.y);
NLMISC::fromString(args[2], BoxMax.x);
NLMISC::fromString(args[3], BoxMax.y);
return true;
}
NLMISC_COMMAND(setStartPrimFile,"Adds a .primitive file for a continent","<continent> <primitive file>")
{
if (args.size() < 2)
return false;
string continent = args[0];
string primFile = args[1];
PrimFiles.insert(multimap<string, string>::value_type(continent, primFile));
nlinfo("Added primitive file %s to continent '%s'", primFile.c_str(), continent.c_str());
return true;
}
NLMISC_COMMAND(setPacsPrimPath,"the pacs prim path","<path>")
{
if (args.size() < 1)
return false;
PacsPrimPath.push_back(args[0]);
return true;
}
NLMISC_COMMAND(setDefaultStart,"Set the default start point for all continents","<startx> <starty> [startz]")
{
if (args.size() < 2)
return false;
CVectorD startPoint;
NLMISC::fromString(args[0], startPoint.x);
NLMISC::fromString(args[1], startPoint.y);
if (args.size() > 2)
{
NLMISC::fromString(args[2], startPoint.z);
}
else
{
startPoint.z = 0.0;
}
DefaultStartPoint = startPoint;
nlinfo("Set default start point (%.1f,%.1f,%.1f)", startPoint.x, startPoint.y, startPoint.z);
return true;
}
NLMISC_COMMAND(setOutputPath,"Set the output path","<path>")
{
if (args.size() < 1)
return false;
OutputPath = CPath::standardizePath(args[0]);
return true;
}
NLMISC_COMMAND(pacsCrunch,"Run a test of pacs crunch","<file name root> [<minx> <miny> <maxx> <maxy>]")
{
if(args.size()<1)
return false;
//crunchPacsMap(args[0]);
CPacsCruncher pc;
if (args.size() == 5)
{
float bxmin, bymin, bxmax, bymax;
bxmin = (float)atof(args[1].c_str());
bymin = (float)atof(args[2].c_str());
bxmax = (float)atof(args[3].c_str());
bymax = (float)atof(args[4].c_str());
CAABBox box;
box.setMinMax(CVector(bxmin, bymin, -10000.0f), CVector(bxmax, bymax, +10000.0f));
pc.crunch(args[0], &box);
}
else
{
pc.crunch(args[0]);
}
return true;
}
NLMISC_COMMAND(pacsCrunchStart,"Run a test of pacs crunch","<file name root> <startx> <starty> [<minx> <miny> <maxx> <maxy>]")
{
if(args.size()<3)
return false;
//crunchPacsMap(args[0]);
float startx, starty;
startx = (float)atof(args[1].c_str());
starty = (float)atof(args[2].c_str());
CVector start(startx, starty, 0.0f);
CPacsCruncher pc;
if (args.size() == 7)
{
float bxmin, bymin, bxmax, bymax;
bxmin = (float)atof(args[3].c_str());
bymin = (float)atof(args[4].c_str());
bxmax = (float)atof(args[5].c_str());
bymax = (float)atof(args[6].c_str());
CAABBox box;
box.setMinMax(CVector(bxmin, bymin, -10000.0f), CVector(bxmax, bymax, +10000.0f));
pc.crunch(args[0], &box, "", &start);
}
else
{
pc.crunch(args[0], NULL, "", &start);
}
return true;
}
NLMISC_COMMAND(pacsCrunchLoop,"Run a serie of tests of pacs crunch","<file name root> <startx> <starty> <loopx> <loopy> [<size>=160]")
{
if(args.size()<5 || args.size()>6)
return false;
CPacsCruncher pc;
float startx, starty;
uint loopx, loopy;
uint x, y;
float size = 160.0f;
startx = (float)atof(args[1].c_str());
starty = (float)atof(args[2].c_str());
NLMISC::fromString(args[3], loopx);
NLMISC::fromString(args[4], loopy);
if (args.size() >= 6)
NLMISC::fromString(args[5], size);
pc.init(args[0]);
for (y=0; y<loopy; ++y)
{
for (x=0; x<loopx; ++x)
{
CAABBox box;
box.setMinMax(CVector(startx+x*size, starty+y*size, -10000.0f), CVector(startx+(x+1)*size, starty+(y+1)*size, +10000.0f));
pc.crunch(args[0], &box, toString(x)+"_"+toString(y));
}
}
pc.release();
return true;
}
NLMISC_COMMAND(pacsBuildBitmap,"build a bitmap from a world map","<file name root>")
{
if(args.size()<1)
return false;
CPacsCruncher pc;
pc.buildBitMap(args[0]);
return true;
}
NLMISC_COMMAND(pacsBuildHeightMap16, "build a sint16 heightmap from a world map","<file name root>")
{
if(args.size()<1)
return false;
CPacsCruncher pc;
pc.buildHeightMap16(args[0]);
return true;
}
NLMISC_COMMAND(pacsBuildWmap,"build crunched world map from a world map","<file name root>")
{
if(args.size()<1)
return false;
CPacsCruncher pc;
pc.buildCrunchedMap(args[0]);
return true;
}
NLMISC_COMMAND(pacsBuildGabarit,"build gabarit maps from a world map","<file name root>")
{
if(args.size()<1)
return false;
CPacsCruncher pc;
uint i;
for (i=0; i<3; ++i)
pc.buildGabarit(args[0], i);
return true;
}
NLMISC_COMMAND(pacsClearHeightmap,"build gabarit maps from a world map","<file name root>")
{
if(args.size()<1)
return false;
CPacsCruncher pc;
uint i;
for (i=1; i<3; ++i)
pc.clearHeightMap(args[0], i);
return true;
}
NLMISC_COMMAND(testAstar, "test astar", "file startx starty endx endy (m)")
{
if (args.size() < 4)
return false;
CPacsCruncher pc;
pc.findPath(args[0], CVectorD(atof(args[1].c_str()), atof(args[2].c_str()), 0.0), CVectorD(atof(args[3].c_str()), atof(args[4].c_str()), 0.0));
return true;
}
NLMISC_COMMAND(testInsideAstar, "test inside astar", "file startx starty endx endy (m)")
{
if (args.size() < 4)
return false;
CPacsCruncher pc;
pc.findInsidePath(args[0], CVectorD(atof(args[1].c_str()), atof(args[2].c_str()), 0.0), CVectorD(atof(args[3].c_str()), atof(args[4].c_str()), 0.0));
return true;
}
NLMISC_COMMAND(testLine, "test linear movement", "file startx starty endx endy (m)")
{
if (args.size() < 4)
return false;
CPacsCruncher pc;
pc.testMove(args[0], CVectorD(atof(args[1].c_str()), atof(args[2].c_str()), 0.0), CVectorD(atof(args[3].c_str()), atof(args[4].c_str()), 0.0));
return true;
}
//
NLMISC_COMMAND(checkPackedSheets, "checks continents.packed_sheets file", "")
{
// a simple pc will automatically checks for continents.packed_sheets file
CPacsCruncher pc;
pc.initPackedSheets();
return true;
}
//
NLMISC_COMMAND(loadWmap, "Loads a worldmap in static world map", "<file>")
{
if (args.size() < 1)
return false;
StaticWorldMap.clear();
CIFile f(OutputPath+args[0]);
f.serial(StaticWorldMap);
return true;
}
NLMISC_COMMAND(clearWmap, "Clears static world map", "")
{
StaticWorldMap.clear();
return true;
}
NLMISC_COMMAND(dumpMotionLayer, "Dumps motion layer around a position", "<x> <y> <z>")
{
/*
if (args.size() < 3)
return false;
CVectorD vpos;
CWorldPosition pos;
vpos.x = atof(args[0].c_str());
vpos.y = atof(args[1].c_str());
vpos.z = atof(args[2].c_str());
StaticWorldMap.setWorldPosition(pos, CMapPosition(vpos));
if (!pos.isValid())
return false;
CTopology::TTopologyId topoId = StaticWorldMap.getTopologyId(pos);
const CTopology &topo = StaticWorldMap.getTopologyNode(topoId);
topo.DirectionMap->dump();
*/
return true;
}
NLMISC_COMMAND(dumpTopo, "Dumps motion layer around a position", "topoId in hexa")
{
if (args.size() < 1)
return false;
uint32 id;
sscanf(args[0].c_str(), "%x", &id);
CTopology::TTopologyId topoId(id);
const CTopology &topo = StaticWorldMap.getTopologyNode(topoId);
topo.DirectionMap->dump();
uint i;
nlinfo("%08X neighbours (%d neighbours)", id, topo.Neighbours.size());
for (i=0; i<topo.Neighbours.size(); ++i)
nlinfo("%d: %08X", i, topo.Neighbours[i].getTopologyRef().getVal());
return true;
}
NLMISC_COMMAND(findTopoPath, "Find path between 2 topologies", "startTopoId endTopoId")
{
if (args.size() < 2)
return false;
uint32 startId, endId;
sscanf(args[0].c_str(), "%x", &startId);
sscanf(args[1].c_str(), "%x", &endId);
CTopology::TTopologyId startTopoId(startId);
CTopology::TTopologyId endTopoId(endId);
CAStarPath path;
StaticWorldMap.findAStarPath(startTopoId, endTopoId, path);
return true;
}
NLMISC_COMMAND(findPath, "Find path between 2 world positions", "startx starty startslot endx endy endslot")
{
if (args.size() < 6)
return false;
sint32 startx, starty, startslot;
sint32 endx, endy, endslot;
NLMISC::fromString(args[0], startx);
NLMISC::fromString(args[1], starty);
NLMISC::fromString(args[2], startslot);
NLMISC::fromString(args[3], endx);
NLMISC::fromString(args[4], endy);
NLMISC::fromString(args[5], endslot);
CWorldPosition start = StaticWorldMap.getWorldPosition(CMapPosition(startx, starty), CSlot(startslot));
CWorldPosition end = StaticWorldMap.getWorldPosition(CMapPosition(endx, endy), CSlot(endslot));
CAStarPath path;
nlinfo("Start: topo %d, i0:%d i1:%d i2:%d i3:%d",
start.getTopologyRef().getCstTopologyNode().Id.getVal(),
start.getTopologyRef().getCstTopologyNode().MasterTopL,
start.getTopologyRef().getCstTopologyNode().MasterTopLN,
start.getTopologyRef().getCstTopologyNode().MasterTopLW,
start.getTopologyRef().getCstTopologyNode().MasterTopLNW);
nlinfo("End: topo %d, i0:%d i1:%d i2:%d i3:%d",
end.getTopologyRef().getCstTopologyNode().Id.getVal(),
end.getTopologyRef().getCstTopologyNode().MasterTopL,
end.getTopologyRef().getCstTopologyNode().MasterTopLN,
end.getTopologyRef().getCstTopologyNode().MasterTopLW,
end.getTopologyRef().getCstTopologyNode().MasterTopLNW);
path.setStartPos(start);
path.setEndPos(end);
StaticWorldMap.findAStarPath(start, end, path.topologiesPathForCalc(), Water);
CWorldPosition current = start;
uint currentStep = 0;
while (StaticWorldMap.move(current, path, currentStep))
nlinfo("current: x:%-10d y:%-10d topoid:%08X topoflags:%d", current.x(), current.y(), current.getTopologyRef().getCstTopologyNode().Id.getVal(), current.getTopologyRef().getCstTopologyNode().Flags);
return true;
}
NLMISC_COMMAND(testLinks, "test links at a position", "x y slot")
{
if (args.size() < 3)
return false;
CWorldPosition pos, mv, back;
CDirection dir(CDirection::E);
sint posx, posy;
uint slot;
NLMISC::fromString(args[0], posx);
NLMISC::fromString(args[1], posy);
NLMISC::fromString(args[2], slot);
pos = StaticWorldMap.getWorldPosition(CMapPosition(posx, posy), CSlot(slot));
uint i;
for (i=0; i<8; ++i)
{
CDirection opp(dir);
opp.addStep(CDirection::HALF_TURN);
if (StaticWorldMap.move(mv=pos, dir))
{
if (!StaticWorldMap.move(back=mv, opp))
nlwarning("Link failure at direction %d of (%04X,%04X,%d)", dir.getVal(), pos.x(), pos.y(), pos.slot());
else if (back != pos)
nlwarning("Reverse link failure at direction %d of (%04X,%04X,%d)", dir.getVal(), pos.x(), pos.y(), pos.slot());
}
dir.addStep(CDirection::HALF_TURN_LEFT);
}
return true;
}
NLMISC_COMMAND(checkMotionLayer, "checks motion layer", "")
{
// StaticWorldMap.checkMotionLayer();
StaticWorldMap.countSuperTopo();
return true;
}
NLMISC_COMMAND(getH, "get H", "")
{
if (args.size() < 3)
return false;
sint32 startx, starty, startslot;
NLMISC::fromString(args[0], startx);
NLMISC::fromString(args[1], starty);
NLMISC::fromString(args[2], startslot);
CWorldPosition start = StaticWorldMap.getWorldPosition(CMapPosition(startx, starty), CSlot(startslot));
sint32 h = start.getRootCell()->getMetricHeight(start);
nlinfo("Pos at (%d,%d,%d), z=%d", start.x(), start.y(), start.slot(), h);
return true;
}
NLMISC_COMMAND(testPacsMove, "test a pacs move", "<continent> <x> <y> <dx> <dy>")
{
if (args.size() != 5)
return false;
CPacsCruncher pc;
string name = args[0];
double x, y, dx, dy;
NLMISC::fromString(args[1], x);
NLMISC::fromString(args[2], y);
NLMISC::fromString(args[3], dx);
NLMISC::fromString(args[4], dy);
pc.init(name);
UMovePrimitive *primitive = pc._Container->addNonCollisionablePrimitive();
primitive->setPrimitiveType( UMovePrimitive::_2DOrientedCylinder );
primitive->setReactionType( UMovePrimitive::Stop );
primitive->setTriggerType((UMovePrimitive::TTrigger)(UMovePrimitive::EnterTrigger+
UMovePrimitive::ExitTrigger) );
primitive->setCollisionMask( 0xffffffff );
primitive->setOcclusionMask( 0x00000000 );
primitive->setObstacle( true );
primitive->setAbsorbtion( 0 );
primitive->setHeight( 6.0f );
primitive->setRadius( 0.5f );
CVectorD startPos(x, y, 0);
primitive->setGlobalPosition(startPos, 0);
pc._Container->evalCollision(1.0, 0);
primitive->move(CVectorD(dx, dy, 0.0), 0);
pc._Container->evalNCPrimitiveCollision(1.0, primitive, 0);
UGlobalPosition newPos;
primitive->getGlobalPosition(newPos, 0);
return true;
}
//
}
NLMISC_VARIABLE(string, EvaluatedPos, "Last evaluated pacs position");
NLMISC_VARIABLE(uint, Verbose, "Verbosity");