/* * CRmgTemplateZone.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 "CRmgTemplateZone.h" #include "../mapping/CMapEditManager.h" #include "../mapping/CMap.h" #include "../VCMI_Lib.h" #include "../CTownHandler.h" #include "../CCreatureHandler.h" #include "../CSpellHandler.h" //for choosing random spells #include "../mapObjects/CommonConstructors.h" #include "../mapObjects/MapObjects.h" //needed to resolve templates for CommonConstructors.h #include "../mapObjects/CGPandoraBox.h" #include "../mapObjects/CRewardableObject.h" class CMap; class CMapEditManager; //class CGObjectInstance; CRmgTemplateZone::CTownInfo::CTownInfo() : townCount(0), castleCount(0), townDensity(0), castleDensity(0) { } int CRmgTemplateZone::CTownInfo::getTownCount() const { return townCount; } void CRmgTemplateZone::CTownInfo::setTownCount(int value) { if(value < 0) throw rmgException("Negative value for town count not allowed."); townCount = value; } int CRmgTemplateZone::CTownInfo::getCastleCount() const { return castleCount; } void CRmgTemplateZone::CTownInfo::setCastleCount(int value) { if(value < 0) throw rmgException("Negative value for castle count not allowed."); castleCount = value; } int CRmgTemplateZone::CTownInfo::getTownDensity() const { return townDensity; } void CRmgTemplateZone::CTownInfo::setTownDensity(int value) { if(value < 0) throw rmgException("Negative value for town density not allowed."); townDensity = value; } int CRmgTemplateZone::CTownInfo::getCastleDensity() const { return castleDensity; } void CRmgTemplateZone::CTownInfo::setCastleDensity(int value) { if(value < 0) throw rmgException("Negative value for castle density not allowed."); castleDensity = value; } CTileInfo::CTileInfo():nearestObjectDistance(INT_MAX), terrain(ETerrainType::WRONG) { occupied = ETileType::POSSIBLE; //all tiles are initially possible to place objects or passages } int CTileInfo::getNearestObjectDistance() const { return nearestObjectDistance; } void CTileInfo::setNearestObjectDistance(int value) { nearestObjectDistance = std::max(0, value); //never negative (or unitialized) } bool CTileInfo::shouldBeBlocked() const { return occupied == ETileType::BLOCKED; } bool CTileInfo::isBlocked() const { return occupied == ETileType::BLOCKED || occupied == ETileType::USED; } bool CTileInfo::isPossible() const { return occupied == ETileType::POSSIBLE; } bool CTileInfo::isFree() const { return occupied == ETileType::FREE; } bool CTileInfo::isUsed() const { return occupied == ETileType::USED; } void CTileInfo::setOccupied(ETileType::ETileType value) { occupied = value; } ETerrainType CTileInfo::getTerrainType() const { return terrain; } void CTileInfo::setTerrainType(ETerrainType value) { terrain = value; } CRmgTemplateZone::CRmgTemplateZone() : id(0), type(ETemplateZoneType::PLAYER_START), size(1), townsAreSameType(false), matchTerrainToTown(true), townType(0), terrainType (ETerrainType::GRASS), zoneMonsterStrength(EMonsterStrength::ZONE_NORMAL), totalDensity(0) { townTypes = getDefaultTownTypes(); terrainTypes = getDefaultTerrainTypes(); } TRmgTemplateZoneId CRmgTemplateZone::getId() const { return id; } void CRmgTemplateZone::setId(TRmgTemplateZoneId value) { if(value <= 0) throw rmgException(boost::to_string(boost::format("Zone %d id should be greater than 0.") %id)); id = value; } ETemplateZoneType::ETemplateZoneType CRmgTemplateZone::getType() const { return type; } void CRmgTemplateZone::setType(ETemplateZoneType::ETemplateZoneType value) { type = value; } int CRmgTemplateZone::getSize() const { return size; } void CRmgTemplateZone::setSize(int value) { if(value <= 0) throw rmgException(boost::to_string(boost::format("Zone %d size needs to be greater than 0.") % id)); size = value; } boost::optional CRmgTemplateZone::getOwner() const { return owner; } void CRmgTemplateZone::setOwner(boost::optional value) { if(!(*value >= 0 && *value <= PlayerColor::PLAYER_LIMIT_I)) throw rmgException(boost::to_string(boost::format ("Owner of zone %d has to be in range 0 to max player count.") %id)); owner = value; } const CRmgTemplateZone::CTownInfo & CRmgTemplateZone::getPlayerTowns() const { return playerTowns; } void CRmgTemplateZone::setPlayerTowns(const CTownInfo & value) { playerTowns = value; } const CRmgTemplateZone::CTownInfo & CRmgTemplateZone::getNeutralTowns() const { return neutralTowns; } void CRmgTemplateZone::setNeutralTowns(const CTownInfo & value) { neutralTowns = value; } bool CRmgTemplateZone::getTownsAreSameType() const { return townsAreSameType; } void CRmgTemplateZone::setTownsAreSameType(bool value) { townsAreSameType = value; } const std::set & CRmgTemplateZone::getTownTypes() const { return townTypes; } void CRmgTemplateZone::setTownTypes(const std::set & value) { townTypes = value; } std::set CRmgTemplateZone::getDefaultTownTypes() const { std::set defaultTowns; auto towns = VLC->townh->getDefaultAllowed(); for(int i = 0; i < towns.size(); ++i) { if(towns[i]) defaultTowns.insert(i); } return defaultTowns; } bool CRmgTemplateZone::getMatchTerrainToTown() const { return matchTerrainToTown; } void CRmgTemplateZone::setMatchTerrainToTown(bool value) { matchTerrainToTown = value; } const std::set & CRmgTemplateZone::getTerrainTypes() const { return terrainTypes; } void CRmgTemplateZone::setTerrainTypes(const std::set & value) { assert(value.find(ETerrainType::WRONG) == value.end() && value.find(ETerrainType::BORDER) == value.end() && value.find(ETerrainType::WATER) == value.end() && value.find(ETerrainType::ROCK) == value.end()); terrainTypes = value; } std::set CRmgTemplateZone::getDefaultTerrainTypes() const { std::set terTypes; static const ETerrainType::EETerrainType allowedTerTypes[] = {ETerrainType::DIRT, ETerrainType::SAND, ETerrainType::GRASS, ETerrainType::SNOW, ETerrainType::SWAMP, ETerrainType::ROUGH, ETerrainType::SUBTERRANEAN, ETerrainType::LAVA}; for (auto & allowedTerType : allowedTerTypes) terTypes.insert(allowedTerType); return terTypes; } void CRmgTemplateZone::setMinesAmount (TResource res, ui16 amount) { mines[res] = amount; } std::map CRmgTemplateZone::getMinesInfo() const { return mines; } void CRmgTemplateZone::addConnection(TRmgTemplateZoneId otherZone) { connections.push_back (otherZone); } std::vector CRmgTemplateZone::getConnections() const { return connections; } void CRmgTemplateZone::setMonsterStrength (EMonsterStrength::EMonsterStrength val) { assert (vstd::iswithin(val, EMonsterStrength::ZONE_WEAK, EMonsterStrength::ZONE_STRONG)); zoneMonsterStrength = val; } void CRmgTemplateZone::setTotalDensity (ui16 val) { totalDensity = val; } ui16 CRmgTemplateZone::getTotalDensity () const { return totalDensity; } void CRmgTemplateZone::addTreasureInfo(CTreasureInfo & info) { treasureInfo.push_back(info); } std::vector CRmgTemplateZone::getTreasureInfo() { return treasureInfo; } std::set* CRmgTemplateZone::getFreePaths() { return &freePaths; } float3 CRmgTemplateZone::getCenter() const { return center; } void CRmgTemplateZone::setCenter(const float3 &f) { //limit boundaries to (0,1) square center = float3 (std::min(std::max(f.x, 0.f), 1.f), std::min(std::max(f.y, 0.f), 1.f), f.z); } bool CRmgTemplateZone::pointIsIn(int x, int y) { return true; } int3 CRmgTemplateZone::getPos() const { return pos; } void CRmgTemplateZone::setPos(const int3 &Pos) { pos = Pos; } void CRmgTemplateZone::addTile (const int3 &pos) { tileinfo.insert(pos); } std::set CRmgTemplateZone::getTileInfo () const { return tileinfo; } void CRmgTemplateZone::discardDistantTiles (CMapGenerator* gen, float distance) { //TODO: mark tiles beyond zone as unavailable, but allow to connect with adjacent zones //for (auto tile : tileinfo) //{ // if (tile.dist2d(this->pos) > distance) // { // gen->setOccupied(tile, ETileType::USED); // //gen->setOccupied(tile, ETileType::BLOCKED); //fixme: crash at rendering? // } //} vstd::erase_if (tileinfo, [distance, this](const int3 &tile) -> bool { return tile.dist2d(this->pos) > distance; }); } void CRmgTemplateZone::createBorder(CMapGenerator* gen) { for (auto tile : tileinfo) { gen->foreach_neighbour (tile, [this, gen](int3 &pos) { if (!vstd::contains(this->tileinfo, pos)) { gen->foreach_neighbour (pos, [this, gen](int3 &pos) { if (gen->isPossible(pos)) gen->setOccupied (pos, ETileType::BLOCKED); }); } }); } } void CRmgTemplateZone::fractalize(CMapGenerator* gen) { std::vector clearedTiles (freePaths.begin(), freePaths.end()); std::set possibleTiles; std::set tilesToClear; //will be set clear std::set tilesToIgnore; //will be erased in this iteration //the more treasure density, the greater distance between paths. Scaling is experimental. const float minDistance = std::sqrt(totalDensity * 3); for (auto tile : tileinfo) { if (gen->isFree(tile)) clearedTiles.push_back(tile); else if (gen->isPossible(tile)) possibleTiles.insert(tile); } if (clearedTiles.empty()) //this should come from zone connections { clearedTiles.push_back(pos); //zone center should be always clear } while (possibleTiles.size()) { for (auto tileToMakePath : possibleTiles) { //find closest free tile float currentDistance = 1e10; int3 closestTile (-1,-1,-1); for (auto clearTile : clearedTiles) { float distance = tileToMakePath.dist2d(clearTile); if (distance < currentDistance) { currentDistance = distance; closestTile = clearTile; } if (currentDistance <= minDistance) { //this tile is close enough. Forget about it and check next one tilesToIgnore.insert (tileToMakePath); break; } } //if tiles is not close enough, make path to it if (currentDistance > minDistance) { crunchPath (gen, tileToMakePath, closestTile, id, &tilesToClear); break; //next iteration - use already cleared tiles } } for (auto tileToClear : tilesToClear) { //move cleared tiles from one set to another clearedTiles.push_back(tileToClear); vstd::erase_if_present(possibleTiles, tileToClear); } for (auto tileToClear : tilesToIgnore) { //these tiles are already connected, ignore them vstd::erase_if_present(possibleTiles, tileToClear); } if (tilesToClear.empty()) //nothing else can be done (?) break; tilesToClear.clear(); //empty this container tilesToIgnore.clear(); } for (auto tile : clearedTiles) { freePaths.insert(tile); } if (0) //enable to debug { std::ofstream out(boost::to_string(boost::format("zone %d") % id)); int levels = gen->map->twoLevel ? 2 : 1; int width = gen->map->width; int height = gen->map->height; for (int k = 0; k < levels; k++) { for(int j=0; jinfoStream() << boost::format ("Zone %d subdivided fractally") %id; } bool CRmgTemplateZone::crunchPath (CMapGenerator* gen, const int3 &src, const int3 &dst, TRmgTemplateZoneId zone, std::set* clearedTiles) { /* make shortest path with free tiles, reachning dst or closest already free tile. Avoid blocks. do not leave zone border */ bool result = false; bool end = false; int3 currentPos = src; float distance = currentPos.dist2dSQ (dst); while (!end) { if (currentPos == dst) { result = true; break; } auto lastDistance = distance; gen->foreach_neighbour (currentPos, [this, gen, ¤tPos, dst, &distance, &result, &end, clearedTiles](int3 &pos) { if (!result) //not sure if lambda is worth it... { if (pos == dst) { result = true; end = true; } if (pos.dist2dSQ (dst) < distance) { if (!gen->isBlocked(pos)) { if (vstd::contains (tileinfo, pos)) { if (gen->isPossible(pos)) { gen->setOccupied (pos, ETileType::FREE); if (clearedTiles) clearedTiles->insert(pos); currentPos = pos; distance = currentPos.dist2dSQ (dst); } else if (gen->isFree(pos)) { end = true; result = true; } else throw rmgException(boost::to_string(boost::format("Tile %s of uknown type found on path") % pos())); } } } } }); if (!(result || distance < lastDistance)) //we do not advance, use more avdnaced pathfinding algorithm? { logGlobal->warnStream() << boost::format ("No tile closer than %s found on path from %s to %s") %currentPos %src %dst; break; } } return result; } void CRmgTemplateZone::addRequiredObject(CGObjectInstance * obj, si32 strength) { requiredObjects.push_back(std::make_pair(obj, strength)); } bool CRmgTemplateZone::addMonster(CMapGenerator* gen, int3 &pos, si32 strength) { //precalculate actual (randomized) monster strength based on this post //http://forum.vcmi.eu/viewtopic.php?p=12426#12426 int mapMonsterStrength = gen->mapGenOptions->getMonsterStrength(); int monsterStrength = zoneMonsterStrength + mapMonsterStrength - 1; //array index from 0 to 4 static const int value1[] = {2500, 1500, 1000, 500, 0}; static const int value2[] = {7500, 7500, 7500, 5000, 5000}; static const float multiplier1[] = {0.5, 0.75, 1.0, 1.5, 1.5}; static const float multiplier2[] = {0.5, 0.75, 1.0, 1.0, 1.5}; int strength1 = std::max(0.f, (strength - value1[monsterStrength]) * multiplier1[monsterStrength]); int strength2 = std::max(0.f, (strength - value2[monsterStrength]) * multiplier2[monsterStrength]); strength = strength1 + strength2; if (strength < 2000) return false; //no guard at all CreatureID creId = CreatureID::NONE; int amount = 0; std::vector possibleCreatures; for (auto cre : VLC->creh->creatures) { if (cre->special) continue; if ((cre->AIValue * (cre->ammMin + cre->ammMax) / 2 < strength) && (strength < cre->AIValue * 100)) //at least one full monster. size between minimum size of given stack and 100 { possibleCreatures.push_back(cre->idNumber); } } if (possibleCreatures.size()) { creId = *RandomGeneratorUtil::nextItem(possibleCreatures, gen->rand); amount = strength / VLC->creh->creatures[creId]->AIValue; if (amount >= 4) amount *= gen->rand.nextDouble(0.75, 1.25); } else //just pick any available creature { creId = CreatureID(132); //Azure Dragon amount = strength / VLC->creh->creatures[creId]->AIValue; } auto guard = new CGCreature(); guard->ID = Obj::MONSTER; guard->subID = creId; auto hlp = new CStackInstance(creId, amount); //will be set during initialization guard->putStack(SlotID(0), hlp); placeObject(gen, guard, pos); return true; } bool CRmgTemplateZone::createTreasurePile (CMapGenerator* gen, int3 &pos) { std::map treasures; std::set boundary; int3 guardPos (-1,-1,-1); int3 nextTreasurePos = pos; //default values int maxValue = 5000; int minValue = 1500; if (treasureInfo.size()) { //roulette wheel int r = gen->rand.nextInt (1, totalDensity); for (auto t : treasureInfo) { if (r <= t.threshold) { maxValue = t.max; minValue = t.min; break; } } } int currentValue = 0; CGObjectInstance * object = nullptr; while (currentValue < minValue) { //TODO: this works only for 1-tile objects //make sure our shape is consistent treasures[nextTreasurePos] = nullptr; for (auto treasurePos : treasures) { gen->foreach_neighbour (treasurePos.first, [gen, &boundary](int3 pos) { boundary.insert(pos); }); } for (auto treasurePos : treasures) { //leaving only boundary around objects vstd::erase_if_present (boundary, treasurePos.first); } for (auto tile : boundary) { //we can't extend boundary anymore if (!(gen->isBlocked(tile) || gen->isPossible(tile))) break; } int remaining = maxValue - currentValue; auto oi = getRandomObject(gen, remaining); object = oi.generateObject(); if (!object) break; currentValue += oi.value; treasures[nextTreasurePos] = object; //now find place for next object int3 placeFound(-1,-1,-1); //FIXME: find out why teh following code causes crashes //std::vector boundaryVec(boundary.begin(), boundary.end()); //RandomGeneratorUtil::randomShuffle(boundaryVec, gen->rand); //for (auto tile : boundaryVec) for (auto tile : boundary) { if (gen->isPossible(tile)) //we can place new treasure only on possible tile { bool here = true; gen->foreach_neighbour (tile, [gen, &here](int3 pos) { if (!(gen->isBlocked(pos) || gen->isPossible(pos))) here = false; }); if (here) { placeFound = tile; break; } } } if (placeFound.valid()) nextTreasurePos = placeFound; } if (treasures.size()) { //find object closest to zone center, then con nect it to the middle of the zone int3 zoneCenter = getPos(); int3 closestTile = int3(-1,-1,-1); float minDistance = 1e10; for (auto treasure : treasures) { if (zoneCenter.dist2d(treasure.first) < minDistance) { closestTile = treasure.first; minDistance = zoneCenter.dist2d(treasure.first); } } assert (closestTile.valid()); if (!crunchPath (gen, closestTile, findClosestTile(freePaths, closestTile), id)) //make sure pile is connected to the middle of zone { for (auto treasure : treasures) { if (gen->isPossible(treasure.first)) gen->setOccupied (treasure.first, ETileType::BLOCKED); } return true; } for (auto tile : boundary) //guard must be standing there { if (gen->isFree(tile)) //this tile could be already blocked, don't place a monster here { guardPos = tile; break; } } if (guardPos.valid()) { for (auto treasure : treasures) { bool objectFitsHere = true; //temporary workaround int3 visitableOffset = treasure.second->getVisitableOffset(); std::set blockedOffsets = treasure.second->getBlockedOffsets(); blockedOffsets.insert (visitableOffset); for (auto blockingTile : blockedOffsets) { int3 t = treasure.first + visitableOffset + blockingTile; if (!gen->map->isInTheMap(t)) { objectFitsHere = false; //if at least one tile is not possible, object can't be placed here break; } } if (objectFitsHere) placeObject(gen, treasure.second, treasure.first + visitableOffset); } if (addMonster(gen, guardPos, currentValue)) {//block only if the object is guarded for (auto tile : boundary) { if (gen->isPossible(tile)) gen->setOccupied (tile, ETileType::BLOCKED); } } } else //we couldn't make a connection to this location, block it { for (auto treasure : treasures) { if (gen->isPossible(treasure.first)) gen->setOccupied (treasure.first, ETileType::BLOCKED); } } return true; } else //we did not place eveyrthing successfully return false; } void CRmgTemplateZone::initTownType (CMapGenerator* gen) { //FIXME: handle case that this player is not present -> towns should be set to neutral int totalTowns = 0; auto addNewTowns = [&totalTowns, gen, this](int count, bool hasFort, PlayerColor player) { for (int i = 0; i < count; i++) { auto town = new CGTownInstance(); town->ID = Obj::TOWN; if (this->townsAreSameType) town->subID = townType; else town->subID = *RandomGeneratorUtil::nextItem(VLC->townh->getAllowedFactions(), gen->rand); //TODO: check allowed town types for this zone town->tempOwner = player; if (hasFort) town->builtBuildings.insert(BuildingID::FORT); town->builtBuildings.insert(BuildingID::DEFAULT); if (!totalTowns) { //first town in zone sets the facton of entire zone town->subID = townType; //register MAIN town of zone gen->registerZone(town->subID); //first town in zone goes in the middle placeObject(gen, town, getPos() + town->getVisitableOffset()); } else addRequiredObject (town); totalTowns++; } }; if ((type == ETemplateZoneType::CPU_START) || (type == ETemplateZoneType::PLAYER_START)) { //set zone types to player faction, generate main town logGlobal->infoStream() << "Preparing playing zone"; int player_id = *owner - 1; auto & playerInfo = gen->map->players[player_id]; if (playerInfo.canAnyonePlay()) { PlayerColor player(player_id); townType = gen->mapGenOptions->getPlayersSettings().find(player)->second.getStartingTown(); if(townType == CMapGenOptions::CPlayerSettings::RANDOM_TOWN) townType = *RandomGeneratorUtil::nextItem(VLC->townh->getAllowedFactions(), gen->rand); // all possible towns, skip neutral auto town = new CGTownInstance(); town->ID = Obj::TOWN; town->subID = townType; town->tempOwner = player; town->builtBuildings.insert(BuildingID::FORT); town->builtBuildings.insert(BuildingID::DEFAULT); placeObject(gen, town, getPos() + town->getVisitableOffset()); //towns are big objects and should be centered around visitable position totalTowns++; //register MAIN town of zone only gen->registerZone (town->subID); logGlobal->traceStream() << "Fill player info " << player_id; // Update player info playerInfo.allowedFactions.clear(); playerInfo.allowedFactions.insert (townType); playerInfo.hasMainTown = true; playerInfo.posOfMainTown = town->pos - town->getVisitableOffset(); playerInfo.generateHeroAtMainTown = true; //now create actual towns addNewTowns (playerTowns.getCastleCount() - 1, true, player); addNewTowns (playerTowns.getTownCount(), false, player); //requiredObjects.push_back(town); } else { type = ETemplateZoneType::TREASURE; townType = *RandomGeneratorUtil::nextItem(VLC->townh->getAllowedFactions(), gen->rand); logGlobal->infoStream() << "Skipping this zone cause no player"; } } else //no player { townType = *RandomGeneratorUtil::nextItem(VLC->townh->getAllowedFactions(), gen->rand); } addNewTowns (neutralTowns.getCastleCount(), true, PlayerColor::NEUTRAL); addNewTowns (neutralTowns.getTownCount(), false, PlayerColor::NEUTRAL); } void CRmgTemplateZone::initTerrainType (CMapGenerator* gen) { if (matchTerrainToTown) terrainType = VLC->townh->factions[townType]->nativeTerrain; else terrainType = *RandomGeneratorUtil::nextItem(terrainTypes, gen->rand); //TODO: allow new types of terrain? if (pos.z) { if (terrainType != ETerrainType::LAVA) terrainType = ETerrainType::SUBTERRANEAN; } else { if (terrainType == ETerrainType::SUBTERRANEAN) terrainType = ETerrainType::DIRT; } paintZoneTerrain (gen, terrainType); } void CRmgTemplateZone::paintZoneTerrain (CMapGenerator* gen, ETerrainType terrainType) { std::vector tiles; for (auto tile : tileinfo) { tiles.push_back (tile); } gen->editManager->getTerrainSelection().setSelection(tiles); gen->editManager->drawTerrain(terrainType, &gen->rand); } bool CRmgTemplateZone::placeMines (CMapGenerator* gen) { std::vector required_mines; required_mines.push_back(Res::ERes::WOOD); required_mines.push_back(Res::ERes::ORE); static const Res::ERes woodOre[] = {Res::ERes::WOOD, Res::ERes::ORE}; static const Res::ERes preciousResources[] = {Res::ERes::GEMS, Res::ERes::CRYSTAL, Res::ERes::MERCURY, Res::ERes::SULFUR}; //TODO: factory / copy constructor? for (const auto & res : woodOre) { //TODO: these 2 should be close to town (within 12 tiles radius) for (int i = 0; i < mines[res]; i++) { auto mine = new CGMine(); mine->ID = Obj::MINE; mine->subID = static_cast(res); mine->producedResource = res; mine->producedQuantity = mine->defaultResProduction(); addRequiredObject(mine, 1500); } } for (const auto & res : preciousResources) { for (int i = 0; i < mines[res]; i++) { auto mine = new CGMine(); mine->ID = Obj::MINE; mine->subID = static_cast(res); mine->producedResource = res; mine->producedQuantity = mine->defaultResProduction(); addRequiredObject(mine, 3500); } } for (int i = 0; i < mines[Res::GOLD]; i++) { auto mine = new CGMine(); mine->ID = Obj::MINE; mine->subID = static_cast(Res::GOLD); mine->producedResource = Res::GOLD; mine->producedQuantity = mine->defaultResProduction(); addRequiredObject(mine, 7000); } return true; } bool CRmgTemplateZone::createRequiredObjects(CMapGenerator* gen) { logGlobal->infoStream() << "Creating required objects"; for(const auto &obj : requiredObjects) { int3 pos; logGlobal->traceStream() << "Looking for place"; if ( ! findPlaceForObject(gen, obj.first, 3, pos)) { logGlobal->errorStream() << boost::format("Failed to fill zone %d due to lack of space") %id; //TODO CLEANUP! return false; } logGlobal->traceStream() << "Place found"; placeObject(gen, obj.first, pos); guardObject (gen, obj.first, obj.second); } return true; } void CRmgTemplateZone::createTreasures(CMapGenerator* gen) { const double minDistance = 3; do { int3 pos; if ( ! findPlaceForTreasurePile(gen, minDistance, pos)) { break; } createTreasurePile (gen, pos); } while(true); } void CRmgTemplateZone::createObstacles(CMapGenerator* gen) { if (pos.z) //underground { std::vector rockTiles; for (auto tile : tileinfo) { if (gen->shouldBeBlocked(tile)) { bool placeRock = true; gen->foreach_neighbour (tile, [gen, &placeRock](int3 &pos) { if (!(gen->shouldBeBlocked(pos) || gen->isPossible(pos))) placeRock = false; }); if (placeRock) { rockTiles.push_back(tile); } } } gen->editManager->getTerrainSelection().setSelection(rockTiles); gen->editManager->drawTerrain(ETerrainType::ROCK, &gen->rand); //for (auto tile : rockTiles) //{ // gen->setOccupied (tile, ETileType::USED); // gen->foreach_neighbour (tile, [gen](int3 &pos) // { // if (!gen->isUsed(pos)) // gen->setOccupied (pos, ETileType::BLOCKED); // }); //} } //get all possible obstacles for this terrain for (auto primaryID : VLC->objtypeh->knownObjects()) { for (auto secondaryID : VLC->objtypeh->knownSubObjects(primaryID)) { auto handler = VLC->objtypeh->getHandlerFor(primaryID, secondaryID); if (handler->isStaticObject()) { for (auto temp : handler->getTemplates()) { if (temp.canBePlacedAt(terrainType) && temp.getBlockMapOffset().valid()) possibleObstacles.push_back(temp); } } } } auto sel = gen->editManager->getTerrainSelection(); sel.clearSelection(); auto tryToPlaceObstacleHere = [this, gen](int3& tile)-> bool { auto temp = *RandomGeneratorUtil::nextItem(possibleObstacles, gen->rand); int3 obstaclePos = tile - temp.getBlockMapOffset(); if (canObstacleBePlacedHere(gen, temp, obstaclePos)) //can be placed here { auto obj = VLC->objtypeh->getHandlerFor(temp.id, temp.subid)->create(temp); placeObject(gen, obj, obstaclePos); return true; } return false; }; for (auto tile : tileinfo) { if (gen->shouldBeBlocked(tile)) //fill tiles that should be blocked with obstacles { while (!tryToPlaceObstacleHere(tile)); } else if (gen->isPossible(tile)) { //try to place random obstacle once - if not possible, leave it clear tryToPlaceObstacleHere(tile); } } } bool CRmgTemplateZone::fill(CMapGenerator* gen) { initTownType(gen); initTerrainType(gen); addAllPossibleObjects (gen); placeMines(gen); createRequiredObjects(gen); fractalize(gen); //after required objects are created and linked with their own paths createTreasures(gen); createObstacles(gen); logGlobal->infoStream() << boost::format ("Zone %d filled successfully") %id; return true; } bool CRmgTemplateZone::findPlaceForTreasurePile(CMapGenerator* gen, si32 min_dist, int3 &pos) { //si32 min_dist = sqrt(tileinfo.size()/density); int best_distance = 0; bool result = false; //logGlobal->infoStream() << boost::format("Min dist for density %f is %d") % density % min_dist; for(auto tile : tileinfo) { auto dist = gen->getTile(tile).getNearestObjectDistance(); if (gen->isPossible(tile) && (dist >= min_dist) && (dist > best_distance)) { bool allTilesAvailable = true; gen->foreach_neighbour (tile, [&gen, &allTilesAvailable](int3 neighbour) { if (!(gen->isPossible(neighbour) || gen->isBlocked(neighbour))) { allTilesAvailable = false; //all present tiles must be already blocked or ready for new objects } }); if (allTilesAvailable) { best_distance = dist; pos = tile; result = true; } } } if (result) { gen->setOccupied(pos, ETileType::BLOCKED); //block that tile } return result; } bool CRmgTemplateZone::canObstacleBePlacedHere(CMapGenerator* gen, ObjectTemplate &temp, int3 &pos) { if (!gen->map->isInTheMap(pos)) //blockmap may fit in the map, but botom-right corner does not return false; auto tilesBlockedByObject = temp.getBlockedOffsets(); for (auto blockingTile : tilesBlockedByObject) { int3 t = pos + blockingTile; if (!gen->map->isInTheMap(t) || !(gen->isPossible(t) || gen->shouldBeBlocked(t))) { return false; //if at least one tile is not possible, object can't be placed here } } return true; } bool CRmgTemplateZone::findPlaceForObject(CMapGenerator* gen, CGObjectInstance* obj, si32 min_dist, int3 &pos) { //we need object apperance to deduce free tiles if (obj->appearance.id == Obj::NO_OBJ) { auto templates = VLC->objtypeh->getHandlerFor(obj->ID, obj->subID)->getTemplates(gen->map->getTile(getPos()).terType); if (templates.empty()) throw rmgException(boost::to_string(boost::format("Did not find graphics for object (%d,%d) at %s") %obj->ID %obj->subID %pos)); obj->appearance = templates.front(); } //si32 min_dist = sqrt(tileinfo.size()/density); int best_distance = 0; bool result = false; //si32 w = gen->map->width; //si32 h = gen->map->height; //logGlobal->infoStream() << boost::format("Min dist for density %f is %d") % density % min_dist; auto tilesBlockedByObject = obj->getBlockedOffsets(); for (auto tile : tileinfo) { //object must be accessible from at least one surounding tile bool accessible = false; for (int x = -1; x < 2; x++) for (int y = -1; y <2; y++) { if (x && y) //check only if object is visitable from another tile { int3 offset = obj->getVisitableOffset() + int3(x, y, 0); if (!vstd::contains(tilesBlockedByObject, offset)) { int3 nearbyPos = tile + offset; if (gen->map->isInTheMap(nearbyPos)) { if (obj->appearance.isVisitableFrom(x, y) && !gen->isBlocked(nearbyPos)) accessible = true; } } } }; if (!accessible) continue; auto ti = gen->getTile(tile); auto dist = ti.getNearestObjectDistance(); //avoid borders if (gen->isPossible(tile) && (dist >= min_dist) && (dist > best_distance)) { bool allTilesAvailable = true; for (auto blockingTile : tilesBlockedByObject) { int3 t = tile + blockingTile; if (!gen->map->isInTheMap(t) || !gen->isPossible(t)) { allTilesAvailable = false; //if at least one tile is not possible, object can't be placed here break; } } if (allTilesAvailable) { best_distance = dist; pos = tile; result = true; } } } if (result) { gen->setOccupied(pos, ETileType::BLOCKED); //block that tile } return result; } void CRmgTemplateZone::checkAndPlaceObject(CMapGenerator* gen, CGObjectInstance* object, const int3 &pos) { if (!gen->map->isInTheMap(pos)) throw rmgException(boost::to_string(boost::format("Position of object %d at %s is outside the map") % object->id % pos)); object->pos = pos; if (object->isVisitable() && !gen->map->isInTheMap(object->visitablePos())) throw rmgException(boost::to_string(boost::format("Visitable tile %s of object %d at %s is outside the map") % object->visitablePos() % object->id % object->pos())); for (auto tile : object->getBlockedPos()) { if (!gen->map->isInTheMap(tile)) throw rmgException(boost::to_string(boost::format("Tile %s of object %d at %s is outside the map") % tile() % object->id % object->pos())); } if (object->appearance.id == Obj::NO_OBJ) { auto terrainType = gen->map->getTile(pos).terType; auto templates = VLC->objtypeh->getHandlerFor(object->ID, object->subID)->getTemplates(terrainType); if (templates.empty()) throw rmgException(boost::to_string(boost::format("Did not find graphics for object (%d,%d) at %s (terrain %d)") %object->ID %object->subID %pos %terrainType)); object->appearance = templates.front(); } gen->map->addBlockVisTiles(object); gen->editManager->insertObject(object, pos); //logGlobal->traceStream() << boost::format ("Successfully inserted object (%d,%d) at pos %s") %object->ID %object->subID %pos(); } void CRmgTemplateZone::placeObject(CMapGenerator* gen, CGObjectInstance* object, const int3 &pos) { //logGlobal->traceStream() << boost::format("Inserting object at %d %d") % pos.x % pos.y; checkAndPlaceObject (gen, object, pos); auto points = object->getBlockedPos(); if (object->isVisitable()) points.insert(pos + object->getVisitableOffset()); points.insert(pos); for(auto p : points) { if (gen->map->isInTheMap(p)) { gen->setOccupied(p, ETileType::USED); } } for(auto tile : tileinfo) { si32 d = pos.dist2dSQ(tile); //optimization, only relative distance is interesting gen->setNearestObjectDistance(tile, std::min(d, gen->getNearestObjectDistance(tile))); } } void CRmgTemplateZone::placeAndGuardObject(CMapGenerator* gen, CGObjectInstance* object, const int3 &pos, si32 str) { placeObject(gen, object, pos); guardObject(gen, object, str); } std::vector CRmgTemplateZone::getAccessibleOffsets (CMapGenerator* gen, CGObjectInstance* object) { //get all tiles from which this object can be accessed int3 visitable = object->visitablePos(); std::vector tiles; auto tilesBlockedByObject = object->getBlockedPos(); //absolue value, as object is already placed gen->foreach_neighbour(visitable, [&](int3& pos) { if (gen->isPossible(pos)) { if (!vstd::contains(tilesBlockedByObject, pos)) { if (object->appearance.isVisitableFrom(pos.x - visitable.x, pos.y - visitable.y) && !gen->isBlocked(pos)) //TODO: refactor - info about visitability from absolute coordinates { tiles.push_back(pos); } } }; }); return tiles; } bool CRmgTemplateZone::guardObject(CMapGenerator* gen, CGObjectInstance* object, si32 str) { logGlobal->traceStream() << boost::format("Guard object at %s") % object->pos(); std::vector tiles = getAccessibleOffsets (gen, object); int3 guardTile(-1,-1,-1); for (auto tile : tiles) { //crunching path may fail if center of teh zone is dirrectly over wide object if (crunchPath (gen, tile, findClosestTile(freePaths, tile), id)) //make sure object is accessible before surrounding it with blocked tiles { guardTile = tile; break; } } if (!guardTile.valid()) { logGlobal->errorStream() << boost::format("Failed to crunch path to object at %s") % object->pos(); return false; } if (addMonster (gen, guardTile, str)) //do not place obstacles around unguarded object { for (auto pos : tiles) { if (!gen->isFree(pos)) gen->setOccupied(pos, ETileType::BLOCKED); } gen->setOccupied (guardTile, ETileType::USED); } else //allow no guard or other object in front of this object { for (auto tile : tiles) gen->setOccupied (tile, ETileType::FREE); } return true; } ObjectInfo CRmgTemplateZone::getRandomObject (CMapGenerator* gen, ui32 value) { std::vector> tresholds; ui32 total = 0; ui32 minValue = 0.25f * value; //roulette wheel for (auto oi : possibleObjects) { if (oi.value >= minValue && oi.value <= value) { //TODO: check place for non-removable object //problem: we need at least template for the object that does not yet exist total += oi.probability; tresholds.push_back (std::make_pair (total, oi)); } } //TODO: generate pandora box with gold if the value is very high if (tresholds.empty()) { ObjectInfo oi; if (minValue > 20000) //we don't have object valuable enough { oi.generateObject = [minValue]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; obj->resources[Res::GOLD] = minValue; return obj; }; oi.value = minValue; oi.probability = 0; } else { oi.generateObject = [gen]() -> CGObjectInstance * { return nullptr; }; oi.value = 0; oi.probability = 0; } return oi; } int r = gen->rand.nextInt (1, total); for (auto t : tresholds) { if (r <= t.first) return t.second; } //FIXME: control reaches end of non-void function. Missing return? } void CRmgTemplateZone::addAllPossibleObjects (CMapGenerator* gen) { //TODO: move typical objects to config ObjectInfo oi; for (auto primaryID : VLC->objtypeh->knownObjects()) { for (auto secondaryID : VLC->objtypeh->knownSubObjects(primaryID)) { auto handler = VLC->objtypeh->getHandlerFor(primaryID, secondaryID); if (!handler->isStaticObject() && handler->getRMGInfo().value) { for (auto temp : handler->getTemplates()) { if (temp.canBePlacedAt(terrainType)) { oi.generateObject = [gen, temp]() -> CGObjectInstance * { return VLC->objtypeh->getHandlerFor(temp.id, temp.subid)->create(temp); }; oi.value = handler->getRMGInfo().value; oi.probability = handler->getRMGInfo().rarity; possibleObjects.push_back (oi); } } } } } //dwellings auto subObjects = VLC->objtypeh->knownSubObjects(Obj::CREATURE_GENERATOR1); //don't spawn original "neutral" dwellings that got replaced by Conflux dwellings in AB static int elementalConfluxROE[] = {7, 13, 16, 47}; for (int i = 0; i < 4; i++) vstd::erase_if_present(subObjects, elementalConfluxROE[i]); for (auto secondaryID : subObjects) { auto dwellingHandler = dynamic_cast(VLC->objtypeh->getHandlerFor(Obj::CREATURE_GENERATOR1, secondaryID).get()); auto creatures = dwellingHandler->getProducedCreatures(); if (creatures.empty()) continue; auto cre = creatures.front(); if (cre->faction == townType) { oi.value = cre->AIValue * cre->growth * (1 + (float)(gen->getZoneCount(cre->faction)) / gen->getTotalZoneCount() + (float)(gen->getZoneCount(cre->faction) / 2)); //TODO: include town count in formula oi.probability = 40; for (auto temp : dwellingHandler->getTemplates()) { if (temp.canBePlacedAt(terrainType)) { oi.generateObject = [gen, temp, secondaryID, dwellingHandler]() -> CGObjectInstance * { auto obj = VLC->objtypeh->getHandlerFor(Obj::CREATURE_GENERATOR1, secondaryID)->create(temp); //dwellingHandler->configureObject(obj, gen->rand); obj->tempOwner = PlayerColor::NEUTRAL; return obj; }; possibleObjects.push_back (oi); } } } } static const int scrollValues[] = {500, 2000, 3000, 4000, 5000}; for (int i = 0; i < 5; i++) { oi.generateObject = [i, gen]() -> CGObjectInstance * { auto obj = new CGArtifact(); obj->ID = Obj::SPELL_SCROLL; obj->subID = 0; std::vector out; //TODO: unify with cb->getAllowedSpells? for (ui32 i = 0; i < gen->map->allowedSpell.size(); i++) //spellh size appears to be greater (?) { const CSpell *spell = SpellID(i).toSpell(); if (gen->map->allowedSpell[spell->id] && spell->level == i+1) { out.push_back(spell->id); } } auto a = CArtifactInstance::createScroll(RandomGeneratorUtil::nextItem(out, gen->rand)->toSpell()); gen->map->addNewArtifactInstance(a); obj->storedArtifact = a; return obj; }; oi.value = scrollValues[i]; oi.probability = 30; possibleObjects.push_back (oi); } //pandora box with gold for (int i = 1; i < 5; i++) { oi.generateObject = [i]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; obj->resources[Res::GOLD] = i * 5000; return obj; }; oi.value = i * 5000;; oi.probability = 5; possibleObjects.push_back (oi); } //pandora box with experience for (int i = 1; i < 5; i++) { oi.generateObject = [i]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; obj->gainedExp = i * 5000; return obj; }; oi.value = i * 6000;; oi.probability = 20; possibleObjects.push_back (oi); } //pandora box with creatures static const int tierValues[] = {5000, 7000, 9000, 12000, 16000, 21000, 27000}; for (auto creature : VLC->creh->creatures) { if (!creature->special && VLC->townh->factions[creature->faction]->nativeTerrain == terrainType) { int actualTier = creature->level > 7 ? 6 : creature->level-1; int creaturesAmount = tierValues[actualTier] / creature->AIValue; if (creaturesAmount <= 5) { } else if (creaturesAmount <= 12) { (creaturesAmount /= 2) *= 2; } else if (creaturesAmount <= 50) { creaturesAmount = boost::math::round((float)creaturesAmount / 5) * 5; } else if (creaturesAmount <= 12) { creaturesAmount = boost::math::round((float)creaturesAmount / 10) * 10; } oi.generateObject = [creature, creaturesAmount]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; auto stack = new CStackInstance(creature, creaturesAmount); obj->creatures.putStack(SlotID(0), stack); return obj; }; oi.value = (2 * (creature->AIValue) * creaturesAmount * (1 + (float)(gen->getZoneCount(creature->faction)) / gen->getTotalZoneCount()))/3; //TODO: count number of towns on the map oi.probability = 3; possibleObjects.push_back (oi); } } //Pandora with 12 spells of certain level for (int i = 1; i <= GameConstants::SPELL_LEVELS; i++) { oi.generateObject = [i, gen]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; std::vector spells; for (auto spell : VLC->spellh->objects) { if (!spell->isSpecialSpell() && spell->level == i) spells.push_back(spell); } RandomGeneratorUtil::randomShuffle(spells, gen->rand); for (int j = 0; j < std::min(12, spells.size()); j++) { obj->spells.push_back (spells[j]->id); } return obj; }; oi.value = (i + 1) * 2500; //5000 - 15000 oi.probability = 2; possibleObjects.push_back (oi); } //Pandora with 15 spells of certain school for (int i = 1; i <= 4; i++) { oi.generateObject = [i, gen]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; std::vector spells; for (auto spell : VLC->spellh->objects) { if (!spell->isSpecialSpell()) { bool school = false; //TODO: we could have better interface for iterating schools switch (i) { case 1: school = spell->air; case 2: school = spell->earth; case 3: school = spell->fire; case 4: school = spell->water; } if (school) spells.push_back(spell); } } RandomGeneratorUtil::randomShuffle(spells, gen->rand); for (int j = 0; j < std::min(15, spells.size()); j++) { obj->spells.push_back (spells[j]->id); } return obj; }; oi.value = 15000; oi.probability = 2; possibleObjects.push_back (oi); } // Pandora box with 60 random spells oi.generateObject = [gen]() -> CGObjectInstance * { auto obj = new CGPandoraBox(); obj->ID = Obj::PANDORAS_BOX; obj->subID = 0; std::vector spells; for (auto spell : VLC->spellh->objects) { if (!spell->isSpecialSpell()) spells.push_back(spell); } RandomGeneratorUtil::randomShuffle(spells, gen->rand); for (int j = 0; j < std::min(60, spells.size()); j++) { obj->spells.push_back (spells[j]->id); } return obj; }; oi.value = 3000; oi.probability = 2; possibleObjects.push_back (oi); }