// 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 . // Nel misc #include "nel/misc/command.h" #include "nel/misc/variable.h" #include "nel/misc/common.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 #include 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 StartPoints; multimap PrimFiles; CVectorD DefaultStartPoint = CVectorD::Null; /// The output path string OutputPath = string("./"); vector PacsPrimPath; vector LookupPath; vector 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 _Positions1; deque _Positions3; deque _Positions5; CVectorD _BMin; CVectorD _BMax; CVectorD _BSize; uint32 _RetrieverWidth; uint32 _RetrieverHeight; uint32 _RetrieverArea; typedef std::map 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(_Retriever))->testPosition(newPos); bool test=true; NLPACS::CGlobalRetriever* gRetriever = static_cast(_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(primitive) != NULL) { CPrimZone *prim = static_cast(primitive); uint i; for (i=0; iVPoints.size(); ++i) { StartPoints.insert(multimap::value_type(insertAt, prim->VPoints[i])); } } else if (dynamic_cast(primitive) != NULL) { CPrimPoint *prim = static_cast(primitive); StartPoints.insert(multimap::value_type(insertAt, prim->Point)); } else if (dynamic_cast(primitive) != NULL) { CPrimPath *prim = static_cast(primitive); uint i; for (i=0; iVPoints.size(); ++i) { StartPoints.insert(multimap::value_type(insertAt, prim->VPoints[i])); } } // parse children uint i; for (i=0; igetNumChildren(); ++i) { IPrimitive *child; if (!primitive->getChild(child, i)) continue; readPrimitive(child, insertAt); } } void readStartupPrims(const std::string &continent) { pair::iterator, multimap::iterator> rng; rng = PrimFiles.equal_range(continent); multimap::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; igetBBox(); 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 startPoints; pair::iterator, multimap::iterator> range = StartPoints.equal_range(name); multimap::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; icgetInstances().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(_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()(_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() 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 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 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 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 stack; multimap 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(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(ndist, tmp)); } else { stacks[tmpflags + 1].insert(std::pair(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::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 &tops = cell->getTopologiesNodes(); uint i, j; // for all topologies in cell for (i=0; i::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::SHdr imageHdr(imageWidth, imageHeight); CBMP4Image<2,2>::SPalette imagePalette; FILE *outf = nlfopen(OutputPath+name+".bmp", "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; ygetRootCellCst(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.empty()) 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 = nlfopen(OutputPath+name+".bmp", "wb"); FILE *outfh = nlfopen(OutputPath+name+"_hm.bmp", "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; ygetRootCellCst(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.empty()) 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 heightMap; heightMap.init(scanWidth, scanHeight); sint16 *dest = &heightMap(0, 0); for (y=0; ygetRootCellCst(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::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::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"," [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::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", " ") { 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"," ") { if (args.size() < 2) return false; string continent = args[0]; string primFile = args[1]; PrimFiles.insert(multimap::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","") { if (args.size() < 1) return false; PacsPrimPath.push_back(args[0]); return true; } NLMISC_COMMAND(setDefaultStart,"Set the default start point for all continents"," [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","") { if (args.size() < 1) return false; OutputPath = CPath::standardizePath(args[0]); return true; } NLMISC_COMMAND(pacsCrunch,"Run a test of pacs crunch"," [ ]") { 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"," [ ]") { 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"," [=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") { 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","") { 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","") { if(args.size()<1) return false; CPacsCruncher pc; pc.buildCrunchedMap(args[0]); return true; } NLMISC_COMMAND(pacsBuildGabarit,"build gabarit maps from a world map","") { 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","") { 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", "") { 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", " ") { /* 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; igetMetricHeight(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", " ") { 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");