2022-08-09 09:54:32 +04:00
|
|
|
/*
|
|
|
|
* RockPlacer.cpp, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "StdInc.h"
|
|
|
|
#include "RockPlacer.h"
|
|
|
|
#include "TreasurePlacer.h"
|
|
|
|
#include "ObjectManager.h"
|
|
|
|
#include "RoadPlacer.h"
|
|
|
|
#include "RiverPlacer.h"
|
|
|
|
#include "RmgMap.h"
|
|
|
|
#include "CMapGenerator.h"
|
|
|
|
#include "Functions.h"
|
2023-01-09 01:17:37 +02:00
|
|
|
#include "../TerrainHandler.h"
|
2022-08-09 09:54:32 +04:00
|
|
|
#include "../CRandomGenerator.h"
|
|
|
|
#include "../mapping/CMapEditManager.h"
|
2023-02-24 22:38:12 +02:00
|
|
|
#include "../mapping/CMap.h"
|
2022-08-09 09:54:32 +04:00
|
|
|
|
2022-07-26 16:07:42 +03:00
|
|
|
VCMI_LIB_NAMESPACE_BEGIN
|
|
|
|
|
2022-08-09 09:54:32 +04:00
|
|
|
void RockPlacer::process()
|
|
|
|
{
|
2022-12-20 16:14:06 +02:00
|
|
|
rockTerrain = VLC->terrainTypeHandler->getById(zone.getTerrainType())->rockTerrain;
|
|
|
|
assert(!VLC->terrainTypeHandler->getById(rockTerrain)->isPassable());
|
2022-08-09 09:54:32 +04:00
|
|
|
|
|
|
|
accessibleArea = zone.freePaths() + zone.areaUsed();
|
|
|
|
if(auto * m = zone.getModificator<ObjectManager>())
|
|
|
|
accessibleArea.unite(m->getVisitableArea());
|
|
|
|
|
|
|
|
//negative approach - create rock tiles first, then make sure all accessible tiles have no rock
|
|
|
|
rockArea = zone.area().getSubarea([this](const int3 & t)
|
|
|
|
{
|
|
|
|
return map.shouldBeBlocked(t);
|
|
|
|
});
|
|
|
|
|
|
|
|
for(auto & z : map.getZones())
|
|
|
|
{
|
|
|
|
if(auto * m = z.second->getModificator<RockPlacer>())
|
|
|
|
{
|
|
|
|
if(m != this && !m->isFinished())
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
processMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RockPlacer::processMap()
|
|
|
|
{
|
|
|
|
//merge all areas
|
|
|
|
for(auto & z : map.getZones())
|
|
|
|
{
|
|
|
|
if(auto * m = z.second->getModificator<RockPlacer>())
|
|
|
|
{
|
|
|
|
map.getEditManager()->getTerrainSelection().setSelection(m->rockArea.getTilesVector());
|
|
|
|
map.getEditManager()->drawTerrain(m->rockTerrain, &generator.rand);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for(auto & z : map.getZones())
|
|
|
|
{
|
|
|
|
if(auto * m = z.second->getModificator<RockPlacer>())
|
|
|
|
{
|
|
|
|
//now make sure all accessible tiles have no additional rock on them
|
|
|
|
map.getEditManager()->getTerrainSelection().setSelection(m->accessibleArea.getTilesVector());
|
|
|
|
map.getEditManager()->drawTerrain(z.second->getTerrainType(), &generator.rand);
|
|
|
|
m->postProcess();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RockPlacer::postProcess()
|
|
|
|
{
|
|
|
|
//finally mark rock tiles as occupied, spawn no obstacles there
|
|
|
|
rockArea = zone.area().getSubarea([this](const int3 & t)
|
|
|
|
{
|
2022-09-21 11:34:23 +02:00
|
|
|
return !map.map().getTile(t).terType->isPassable();
|
2022-08-09 09:54:32 +04:00
|
|
|
});
|
|
|
|
|
|
|
|
zone.areaUsed().unite(rockArea);
|
|
|
|
zone.areaPossible().subtract(rockArea);
|
|
|
|
if(auto * m = zone.getModificator<RiverPlacer>())
|
|
|
|
m->riverProhibit().unite(rockArea);
|
|
|
|
if(auto * m = zone.getModificator<RoadPlacer>())
|
|
|
|
m->areaIsolated().unite(rockArea);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RockPlacer::init()
|
|
|
|
{
|
|
|
|
POSTFUNCTION_ALL(RoadPlacer);
|
|
|
|
DEPENDENCY(TreasurePlacer);
|
|
|
|
}
|
|
|
|
|
|
|
|
char RockPlacer::dump(const int3 & t)
|
|
|
|
{
|
2022-09-21 11:34:23 +02:00
|
|
|
if(!map.map().getTile(t).terType->isPassable())
|
2022-08-09 09:54:32 +04:00
|
|
|
{
|
|
|
|
return zone.area().contains(t) ? 'R' : 'E';
|
|
|
|
}
|
|
|
|
return Modificator::dump(t);
|
|
|
|
}
|
2022-07-26 16:07:42 +03:00
|
|
|
|
|
|
|
VCMI_LIB_NAMESPACE_END
|