2013-08-17 15:46:48 +03:00
/*
* 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"
2014-05-22 20:25:17 +03:00
# include "../mapping/CMapEditManager.h"
# include "../mapping/CMap.h"
2013-08-17 15:46:48 +03:00
# include "../VCMI_Lib.h"
# include "../CTownHandler.h"
2014-06-01 15:10:44 +03:00
# include "../CCreatureHandler.h"
2014-06-05 21:19:42 +03:00
# include "../CSpellHandler.h" //for choosing random spells
2013-08-17 15:46:48 +03:00
2014-06-05 19:52:14 +03:00
# include "../mapObjects/CObjectClassesHandler.h"
2014-06-22 17:26:08 +03:00
# include "../mapObjects/CGPandoraBox.h"
# include "../mapObjects/CRewardableObject.h"
2014-06-05 19:52:14 +03:00
2014-05-22 20:25:17 +03:00
class CMap ;
class CMapEditManager ;
2013-08-17 15:46:48 +03:00
CRmgTemplateZone : : CTownInfo : : CTownInfo ( ) : townCount ( 0 ) , castleCount ( 0 ) , townDensity ( 0 ) , castleDensity ( 0 )
{
}
int CRmgTemplateZone : : CTownInfo : : getTownCount ( ) const
{
return townCount ;
}
void CRmgTemplateZone : : CTownInfo : : setTownCount ( int value )
{
2014-05-23 18:12:31 +03:00
if ( value < 0 )
throw rmgException ( " Negative value for town count not allowed. " ) ;
2013-08-17 15:46:48 +03:00
townCount = value ;
}
int CRmgTemplateZone : : CTownInfo : : getCastleCount ( ) const
{
return castleCount ;
}
void CRmgTemplateZone : : CTownInfo : : setCastleCount ( int value )
{
2014-05-23 18:12:31 +03:00
if ( value < 0 )
throw rmgException ( " Negative value for castle count not allowed. " ) ;
2013-08-17 15:46:48 +03:00
castleCount = value ;
}
int CRmgTemplateZone : : CTownInfo : : getTownDensity ( ) const
{
return townDensity ;
}
void CRmgTemplateZone : : CTownInfo : : setTownDensity ( int value )
{
2014-05-23 18:12:31 +03:00
if ( value < 0 )
throw rmgException ( " Negative value for town density not allowed. " ) ;
2013-08-17 15:46:48 +03:00
townDensity = value ;
}
int CRmgTemplateZone : : CTownInfo : : getCastleDensity ( ) const
{
return castleDensity ;
}
void CRmgTemplateZone : : CTownInfo : : setCastleDensity ( int value )
{
2014-05-23 18:12:31 +03:00
if ( value < 0 )
throw rmgException ( " Negative value for castle density not allowed. " ) ;
2013-08-17 15:46:48 +03:00
castleDensity = value ;
}
2014-05-30 17:50:06 +03:00
CTileInfo : : CTileInfo ( ) : nearestObjectDistance ( INT_MAX ) , terrain ( ETerrainType : : WRONG )
2014-05-22 20:25:17 +03:00
{
2014-05-30 17:50:06 +03:00
occupied = ETileType : : POSSIBLE ; //all tiles are initially possible to place objects or passages
2014-05-22 20:25:17 +03:00
}
2014-05-30 17:50:06 +03:00
int CTileInfo : : getNearestObjectDistance ( ) const
2014-05-22 20:25:17 +03:00
{
return nearestObjectDistance ;
}
2014-05-30 17:50:06 +03:00
void CTileInfo : : setNearestObjectDistance ( int value )
2014-05-22 20:25:17 +03:00
{
2014-05-23 20:14:20 +03:00
nearestObjectDistance = std : : max ( 0 , value ) ; //never negative (or unitialized)
2014-05-22 20:25:17 +03:00
}
2014-05-30 17:50:06 +03:00
bool CTileInfo : : shouldBeBlocked ( ) const
2014-05-22 20:25:17 +03:00
{
2014-05-30 17:50:06 +03:00
return occupied = = ETileType : : BLOCKED ;
2014-05-22 20:25:17 +03:00
}
2014-05-30 17:50:06 +03:00
bool CTileInfo : : isBlocked ( ) const
2014-05-22 20:25:17 +03:00
{
2014-05-30 17:50:06 +03:00
return occupied = = ETileType : : BLOCKED | | occupied = = ETileType : : USED ;
2014-05-22 20:25:17 +03:00
}
2014-05-30 17:50:06 +03:00
bool CTileInfo : : isPossible ( ) const
2014-05-22 20:25:17 +03:00
{
2014-05-30 17:50:06 +03:00
return occupied = = ETileType : : POSSIBLE ;
2014-05-22 20:25:17 +03:00
}
2014-05-30 17:50:06 +03:00
bool CTileInfo : : isFree ( ) const
{
return occupied = = ETileType : : FREE ;
}
2014-07-04 00:11:24 +03:00
bool CTileInfo : : isUsed ( ) const
{
return occupied = = ETileType : : USED ;
}
2014-05-30 17:50:06 +03:00
void CTileInfo : : setOccupied ( ETileType : : ETileType value )
2014-05-22 20:25:17 +03:00
{
occupied = value ;
}
2014-05-30 17:50:06 +03:00
ETerrainType CTileInfo : : getTerrainType ( ) const
2014-05-22 20:25:17 +03:00
{
return terrain ;
}
2014-05-30 17:50:06 +03:00
void CTileInfo : : setTerrainType ( ETerrainType value )
2014-05-22 20:25:17 +03:00
{
terrain = value ;
}
2014-06-22 17:26:08 +03:00
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 )
2013-08-17 15:46:48 +03:00
{
townTypes = getDefaultTownTypes ( ) ;
terrainTypes = getDefaultTerrainTypes ( ) ;
}
TRmgTemplateZoneId CRmgTemplateZone : : getId ( ) const
{
return id ;
}
void CRmgTemplateZone : : setId ( TRmgTemplateZoneId value )
{
2014-05-23 18:12:31 +03:00
if ( value < = 0 )
throw rmgException ( boost : : to_string ( boost : : format ( " Zone %d id should be greater than 0. " ) % id ) ) ;
2013-08-17 15:46:48 +03:00
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 )
{
2014-05-23 18:12:31 +03:00
if ( value < = 0 )
throw rmgException ( boost : : to_string ( boost : : format ( " Zone %d size needs to be greater than 0. " ) % id ) ) ;
2013-08-17 15:46:48 +03:00
size = value ;
}
boost : : optional < int > CRmgTemplateZone : : getOwner ( ) const
{
return owner ;
}
void CRmgTemplateZone : : setOwner ( boost : : optional < int > value )
{
2014-05-23 18:12:31 +03:00
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 ) ) ;
2013-08-17 15:46:48 +03:00
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 < TFaction > & CRmgTemplateZone : : getTownTypes ( ) const
{
return townTypes ;
}
void CRmgTemplateZone : : setTownTypes ( const std : : set < TFaction > & value )
{
townTypes = value ;
}
std : : set < TFaction > CRmgTemplateZone : : getDefaultTownTypes ( ) const
{
std : : set < TFaction > 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 < ETerrainType > & CRmgTemplateZone : : getTerrainTypes ( ) const
{
return terrainTypes ;
}
void CRmgTemplateZone : : setTerrainTypes ( const std : : set < ETerrainType > & 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 < ETerrainType > CRmgTemplateZone : : getDefaultTerrainTypes ( ) const
{
std : : set < ETerrainType > terTypes ;
2014-06-14 18:14:59 +03:00
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 ) ;
2013-08-17 15:46:48 +03:00
2014-06-14 18:14:59 +03:00
return terTypes ;
2013-08-17 15:46:48 +03:00
}
2014-05-22 20:25:17 +03:00
2014-06-14 21:05:19 +03:00
void CRmgTemplateZone : : setMinesAmount ( TResource res , ui16 amount )
{
mines [ res ] = amount ;
}
std : : map < TResource , ui16 > CRmgTemplateZone : : getMinesInfo ( ) const
{
return mines ;
}
2014-05-24 13:42:06 +03:00
void CRmgTemplateZone : : addConnection ( TRmgTemplateZoneId otherZone )
{
connections . push_back ( otherZone ) ;
}
std : : vector < TRmgTemplateZoneId > CRmgTemplateZone : : getConnections ( ) const
{
return connections ;
}
2014-06-04 16:16:23 +03:00
2014-06-15 12:08:06 +03:00
void CRmgTemplateZone : : setMonsterStrength ( EMonsterStrength : : EMonsterStrength val )
{
assert ( vstd : : iswithin ( val , EMonsterStrength : : ZONE_WEAK , EMonsterStrength : : ZONE_STRONG ) ) ;
zoneMonsterStrength = val ;
}
2014-06-08 15:35:41 +03:00
void CRmgTemplateZone : : setTotalDensity ( ui16 val )
{
totalDensity = val ;
}
ui16 CRmgTemplateZone : : getTotalDensity ( ) const
{
return totalDensity ;
}
2014-06-04 16:16:23 +03:00
void CRmgTemplateZone : : addTreasureInfo ( CTreasureInfo & info )
{
treasureInfo . push_back ( info ) ;
}
std : : vector < CTreasureInfo > CRmgTemplateZone : : getTreasureInfo ( )
{
return treasureInfo ;
}
2014-06-26 21:12:37 +03:00
std : : set < int3 > * CRmgTemplateZone : : getFreePaths ( )
{
return & freePaths ;
}
2014-05-24 13:42:06 +03:00
float3 CRmgTemplateZone : : getCenter ( ) const
{
return center ;
}
2014-05-30 17:50:06 +03:00
void CRmgTemplateZone : : setCenter ( const float3 & f )
2014-05-24 13:42:06 +03:00
{
//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 ) ;
}
2014-05-22 20:25:17 +03:00
bool CRmgTemplateZone : : pointIsIn ( int x , int y )
{
2014-05-24 15:06:08 +03:00
return true ;
2014-05-22 20:25:17 +03:00
}
2014-05-30 17:50:06 +03:00
int3 CRmgTemplateZone : : getPos ( ) const
2014-05-22 20:25:17 +03:00
{
2014-05-24 13:42:06 +03:00
return pos ;
}
2014-05-25 12:02:15 +03:00
void CRmgTemplateZone : : setPos ( const int3 & Pos )
2014-05-24 13:42:06 +03:00
{
pos = Pos ;
2014-05-22 20:25:17 +03:00
}
2014-05-25 12:02:15 +03:00
void CRmgTemplateZone : : addTile ( const int3 & pos )
2014-05-24 15:06:08 +03:00
{
2014-05-30 22:23:41 +03:00
tileinfo . insert ( pos ) ;
}
2014-05-31 11:56:14 +03:00
std : : set < int3 > CRmgTemplateZone : : getTileInfo ( ) const
{
return tileinfo ;
}
2014-07-03 18:24:28 +03:00
void CRmgTemplateZone : : discardDistantTiles ( CMapGenerator * gen , int distance )
2014-07-03 13:28:51 +03:00
{
2014-07-03 18:24:28 +03:00
for ( auto tile : tileinfo )
{
if ( tile . dist2d ( this - > pos ) > distance )
2014-07-04 00:11:24 +03:00
{
2014-07-03 18:24:28 +03:00
gen - > setOccupied ( tile , ETileType : : USED ) ;
2014-07-04 00:11:24 +03:00
//gen->setOccupied(tile, ETileType::BLOCKED); //fixme: crash at rendering?
}
2014-07-03 18:24:28 +03:00
}
2014-07-03 13:28:51 +03:00
vstd : : erase_if ( tileinfo , [ distance , this ] ( const int3 & tile ) - > bool
{
return tile . dist2d ( this - > pos ) > distance ;
} ) ;
}
2014-05-30 22:23:41 +03:00
void CRmgTemplateZone : : createBorder ( CMapGenerator * gen )
{
for ( auto tile : tileinfo )
{
gen - > foreach_neighbour ( tile , [ this , gen ] ( int3 & pos )
{
if ( ! vstd : : contains ( this - > tileinfo , pos ) )
2014-05-31 11:56:14 +03:00
{
gen - > foreach_neighbour ( pos , [ this , gen ] ( int3 & pos )
{
if ( gen - > isPossible ( pos ) )
gen - > setOccupied ( pos , ETileType : : BLOCKED ) ;
} ) ;
}
} ) ;
}
}
2014-06-12 22:10:43 +03:00
void CRmgTemplateZone : : fractalize ( CMapGenerator * gen )
{
std : : vector < int3 > clearedTiles ;
std : : set < int3 > possibleTiles ;
std : : set < int3 > tilesToClear ; //will be set clear
std : : set < int3 > tilesToIgnore ; //will be erased in this iteration
const float minDistance = std : : sqrt ( totalDensity ) ;
for ( auto tile : tileinfo )
{
if ( gen - > isFree ( tile ) )
clearedTiles . push_back ( tile ) ;
2014-06-22 17:26:08 +03:00
else if ( gen - > isPossible ( tile ) )
2014-06-12 22:10:43 +03:00
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 ( ) ;
}
logGlobal - > infoStream ( ) < < boost : : format ( " Zone %d subdivided fractally " ) % id ;
}
bool CRmgTemplateZone : : crunchPath ( CMapGenerator * gen , const int3 & src , const int3 & dst , TRmgTemplateZoneId zone , std : : set < int3 > * clearedTiles )
2014-05-31 11:56:14 +03:00
{
/*
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 )
2014-06-13 13:04:17 +03:00
{
result = true ;
2014-05-31 11:56:14 +03:00
break ;
2014-06-13 13:04:17 +03:00
}
2014-05-31 11:56:14 +03:00
auto lastDistance = distance ;
2014-06-12 22:10:43 +03:00
gen - > foreach_neighbour ( currentPos , [ this , gen , & currentPos , dst , & distance , & result , & end , clearedTiles ] ( int3 & pos )
2014-05-31 11:56:14 +03:00
{
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 ) ;
2014-06-12 22:10:43 +03:00
if ( clearedTiles )
clearedTiles - > insert ( pos ) ;
2014-05-31 11:56:14 +03:00
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 ( ) ) ) ;
}
}
}
}
2014-05-30 22:23:41 +03:00
} ) ;
2014-05-31 11:56:14 +03:00
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 ;
}
2014-05-30 22:23:41 +03:00
}
2014-05-31 11:56:14 +03:00
return result ;
2014-05-24 15:06:08 +03:00
}
2014-06-01 17:31:15 +03:00
void CRmgTemplateZone : : addRequiredObject ( CGObjectInstance * obj , si32 strength )
2014-05-22 20:25:17 +03:00
{
2014-06-01 17:31:15 +03:00
requiredObjects . push_back ( std : : make_pair ( obj , strength ) ) ;
2014-06-01 13:02:43 +03:00
}
2014-05-24 15:33:22 +03:00
2014-06-04 21:59:01 +03:00
bool CRmgTemplateZone : : addMonster ( CMapGenerator * gen , int3 & pos , si32 strength )
2014-06-01 15:10:44 +03:00
{
2014-06-04 11:16:08 +03:00
//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 )
2014-06-04 21:59:01 +03:00
return false ; //no guard at all
2014-06-04 11:16:08 +03:00
2014-06-01 15:10:44 +03:00
CreatureID creId = CreatureID : : NONE ;
int amount = 0 ;
2014-06-07 23:27:36 +03:00
std : : vector < CreatureID > possibleCreatures ;
for ( auto cre : VLC - > creh - > creatures )
2014-06-01 15:10:44 +03:00
{
2014-06-07 23:27:36 +03:00
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
2014-06-04 11:16:08 +03:00
{
2014-06-07 23:27:36 +03:00
possibleCreatures . push_back ( cre - > idNumber ) ;
2014-06-04 11:16:08 +03:00
}
2014-06-01 15:10:44 +03:00
}
2014-06-07 23:27:36 +03:00
if ( possibleCreatures . size ( ) )
2014-06-08 09:42:29 +03:00
{
2014-06-07 23:27:36 +03:00
creId = * RandomGeneratorUtil : : nextItem ( possibleCreatures , gen - > rand ) ;
2014-06-08 09:42:29 +03:00
amount = strength / VLC - > creh - > creatures [ creId ] - > AIValue ;
if ( amount > = 4 )
amount * = gen - > rand . nextDouble ( 0.75 , 1.25 ) ;
}
2014-06-07 23:27:36 +03:00
else //just pick any available creature
{
creId = CreatureID ( 132 ) ; //Azure Dragon
amount = strength / VLC - > creh - > creatures [ creId ] - > AIValue ;
}
2014-06-01 15:10:44 +03:00
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 ) ;
2014-06-04 21:59:01 +03:00
return true ;
2014-06-01 15:10:44 +03:00
}
2014-06-04 11:16:08 +03:00
bool CRmgTemplateZone : : createTreasurePile ( CMapGenerator * gen , int3 & pos )
{
2014-06-05 18:19:11 +03:00
std : : map < int3 , CGObjectInstance * > treasures ;
std : : set < int3 > boundary ;
2014-06-13 13:04:17 +03:00
int3 guardPos ( - 1 , - 1 , - 1 ) ;
2014-06-05 18:19:11 +03:00
int3 nextTreasurePos = pos ;
2014-06-04 16:16:23 +03:00
//default values
int maxValue = 5000 ;
int minValue = 1500 ;
if ( treasureInfo . size ( ) )
{
2014-06-07 16:51:03 +03:00
//roulette wheel
2014-06-08 15:35:41 +03:00
int r = gen - > rand . nextInt ( 1 , totalDensity ) ;
2014-06-07 16:51:03 +03:00
2014-06-08 15:35:41 +03:00
for ( auto t : treasureInfo )
2014-06-07 16:51:03 +03:00
{
2014-06-08 15:35:41 +03:00
if ( r < = t . threshold )
2014-06-07 16:51:03 +03:00
{
2014-06-08 15:35:41 +03:00
maxValue = t . max ;
minValue = t . min ;
2014-06-07 16:51:03 +03:00
break ;
}
}
2014-06-04 16:16:23 +03:00
}
2014-06-04 11:16:08 +03:00
int currentValue = 0 ;
CGObjectInstance * object = nullptr ;
while ( currentValue < minValue )
{
2014-06-05 18:19:11 +03:00
//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 ;
}
2014-06-04 11:16:08 +03:00
int remaining = maxValue - currentValue ;
2014-06-04 20:08:04 +03:00
auto oi = getRandomObject ( gen , remaining ) ;
object = oi . generateObject ( ) ;
if ( ! object )
break ;
currentValue + = oi . value ;
2014-06-05 18:19:11 +03:00
treasures [ nextTreasurePos ] = object ;
//now find place for next object
int3 placeFound ( - 1 , - 1 , - 1 ) ;
2014-06-07 15:02:57 +03:00
//FIXME: find out why teh following code causes crashes
//std::vector<int3> boundaryVec(boundary.begin(), boundary.end());
//RandomGeneratorUtil::randomShuffle(boundaryVec, gen->rand);
//for (auto tile : boundaryVec)
2014-06-05 18:19:11 +03:00
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 ;
2014-06-04 20:08:04 +03:00
}
2014-06-05 18:19:11 +03:00
if ( treasures . size ( ) )
2014-06-04 20:08:04 +03:00
{
2014-06-13 09:00:26 +03:00
//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 ( ) ) ;
2014-06-13 13:04:17 +03:00
if ( ! crunchPath ( gen , closestTile , getPos ( ) , 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 ;
}
2014-06-13 09:00:26 +03:00
2014-06-12 22:51:16 +03:00
for ( auto tile : boundary ) //guard must be standing there
2014-06-05 22:37:39 +03:00
{
2014-06-12 22:51:16 +03:00
if ( gen - > isFree ( tile ) ) //this tile could be already blocked, don't place a monster here
2014-06-05 22:37:39 +03:00
{
2014-06-12 22:51:16 +03:00
guardPos = tile ;
break ;
}
2014-06-05 22:37:39 +03:00
}
2014-06-13 13:04:17 +03:00
if ( guardPos . valid ( ) )
{
for ( auto treasure : treasures )
{
placeObject ( gen , treasure . second , treasure . first - treasure . second - > getVisitableOffset ( ) ) ;
}
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 )
2014-06-05 18:19:11 +03:00
{
2014-06-13 13:04:17 +03:00
if ( gen - > isPossible ( treasure . first ) )
gen - > setOccupied ( treasure . first , ETileType : : BLOCKED ) ;
2014-06-05 18:19:11 +03:00
}
}
2014-06-13 13:04:17 +03:00
2014-06-04 20:08:04 +03:00
return true ;
}
else //we did not place eveyrthing successfully
return false ;
}
2014-06-14 18:14:59 +03:00
void CRmgTemplateZone : : initTownType ( CMapGenerator * gen )
2014-06-01 13:02:43 +03:00
{
2014-06-15 22:23:32 +03:00
//FIXME: handle case that this player is not present -> towns should be set to neutral
int totalTowns = 0 ;
2014-07-03 10:39:26 +03:00
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 goes in the middle
placeObject ( gen , town , getPos ( ) + town - > getVisitableOffset ( ) ) ;
else
addRequiredObject ( town ) ;
totalTowns + + ;
}
} ;
2014-05-22 20:25:17 +03:00
if ( ( type = = ETemplateZoneType : : CPU_START ) | | ( type = = ETemplateZoneType : : PLAYER_START ) )
{
2014-06-15 22:23:32 +03:00
//set zone types to player faction, generate main town
2014-05-22 20:25:17 +03:00
logGlobal - > infoStream ( ) < < " Preparing playing zone " ;
int player_id = * owner - 1 ;
auto & playerInfo = gen - > map - > players [ player_id ] ;
if ( playerInfo . canAnyonePlay ( ) )
{
PlayerColor player ( player_id ) ;
2014-06-14 18:14:59 +03:00
townType = gen - > mapGenOptions - > getPlayersSettings ( ) . find ( player ) - > second . getStartingTown ( ) ;
2014-05-22 20:25:17 +03:00
2014-06-14 18:14:59 +03:00
if ( townType = = CMapGenOptions : : CPlayerSettings : : RANDOM_TOWN )
townType = * RandomGeneratorUtil : : nextItem ( VLC - > townh - > getAllowedFactions ( ) , gen - > rand ) ; // all possible towns, skip neutral
2014-06-15 22:23:32 +03:00
auto town = new CGTownInstance ( ) ;
town - > ID = Obj : : TOWN ;
2014-05-22 20:25:17 +03:00
2014-06-14 18:14:59 +03:00
town - > subID = townType ;
2014-05-22 20:25:17 +03:00
town - > tempOwner = player ;
town - > builtBuildings . insert ( BuildingID : : FORT ) ;
town - > builtBuildings . insert ( BuildingID : : DEFAULT ) ;
2014-06-03 09:57:20 +03:00
placeObject ( gen , town , getPos ( ) + town - > getVisitableOffset ( ) ) ; //towns are big objects and should be centered around visitable position
2014-05-22 20:25:17 +03:00
2014-06-15 22:23:32 +03:00
totalTowns + + ;
2014-05-24 19:39:58 +03:00
logGlobal - > traceStream ( ) < < " Fill player info " < < player_id ;
2014-06-05 18:19:11 +03:00
2014-05-22 20:25:17 +03:00
// Update player info
playerInfo . allowedFactions . clear ( ) ;
2014-06-14 18:14:59 +03:00
playerInfo . allowedFactions . insert ( townType ) ;
2014-05-22 20:25:17 +03:00
playerInfo . hasMainTown = true ;
2014-07-01 08:07:40 +03:00
playerInfo . posOfMainTown = town - > pos - town - > getVisitableOffset ( ) ;
2014-05-22 20:25:17 +03:00
playerInfo . generateHeroAtMainTown = true ;
2014-06-15 22:23:32 +03:00
//now create actual towns
2014-07-03 10:39:26 +03:00
addNewTowns ( playerTowns . getCastleCount ( ) - 1 , true , player ) ;
addNewTowns ( playerTowns . getTownCount ( ) , false , player ) ;
2014-06-15 22:23:32 +03:00
2014-06-01 13:02:43 +03:00
//requiredObjects.push_back(town);
2014-05-22 20:25:17 +03:00
}
else
{
type = ETemplateZoneType : : TREASURE ;
2014-06-14 18:14:59 +03:00
townType = * RandomGeneratorUtil : : nextItem ( VLC - > townh - > getAllowedFactions ( ) , gen - > rand ) ;
2014-05-22 20:25:17 +03:00
logGlobal - > infoStream ( ) < < " Skipping this zone cause no player " ;
}
}
2014-05-24 15:33:22 +03:00
else //no player
{
2014-06-14 18:14:59 +03:00
townType = * RandomGeneratorUtil : : nextItem ( VLC - > townh - > getAllowedFactions ( ) , gen - > rand ) ;
2014-05-24 15:33:22 +03:00
}
2014-06-15 22:23:32 +03:00
2014-07-03 10:39:26 +03:00
addNewTowns ( neutralTowns . getCastleCount ( ) , true , PlayerColor : : NEUTRAL ) ;
addNewTowns ( neutralTowns . getTownCount ( ) , false , PlayerColor : : NEUTRAL ) ;
2014-06-14 18:14:59 +03:00
}
void CRmgTemplateZone : : initTerrainType ( CMapGenerator * gen )
{
if ( matchTerrainToTown )
terrainType = VLC - > townh - > factions [ townType ] - > nativeTerrain ;
else
terrainType = * RandomGeneratorUtil : : nextItem ( terrainTypes , gen - > rand ) ;
2014-07-04 11:45:00 +03:00
//TODO: allow new types of terrain?
if ( pos . z )
{
if ( terrainType ! = ETerrainType : : LAVA )
terrainType = ETerrainType : : SUBTERRANEAN ;
}
else
{
if ( terrainType = = ETerrainType : : SUBTERRANEAN )
terrainType = ETerrainType : : DIRT ;
}
2014-07-03 18:24:28 +03:00
paintZoneTerrain ( gen , terrainType ) ;
}
void CRmgTemplateZone : : paintZoneTerrain ( CMapGenerator * gen , ETerrainType terrainType )
{
2014-05-24 15:33:22 +03:00
std : : vector < int3 > tiles ;
for ( auto tile : tileinfo )
{
2014-05-30 22:23:41 +03:00
tiles . push_back ( tile ) ;
2014-05-24 15:33:22 +03:00
}
gen - > editManager - > getTerrainSelection ( ) . setSelection ( tiles ) ;
2014-06-14 18:14:59 +03:00
gen - > editManager - > drawTerrain ( terrainType , & gen - > rand ) ;
}
2014-06-14 21:05:19 +03:00
bool CRmgTemplateZone : : placeMines ( CMapGenerator * gen )
{
std : : vector < Res : : ERes > 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 < si32 > ( 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 < si32 > ( 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 < si32 > ( Res : : GOLD ) ;
mine - > producedResource = Res : : GOLD ;
mine - > producedQuantity = mine - > defaultResProduction ( ) ;
addRequiredObject ( mine , 7000 ) ;
}
return true ;
}
2014-06-15 09:48:46 +03:00
bool CRmgTemplateZone : : createRequiredObjects ( CMapGenerator * gen )
2014-06-14 18:14:59 +03:00
{
2014-05-22 20:25:17 +03:00
logGlobal - > infoStream ( ) < < " Creating required objects " ;
2014-06-01 13:02:43 +03:00
for ( const auto & obj : requiredObjects )
2014-05-22 20:25:17 +03:00
{
int3 pos ;
2014-05-24 19:39:58 +03:00
logGlobal - > traceStream ( ) < < " Looking for place " ;
2014-06-01 17:31:15 +03:00
if ( ! findPlaceForObject ( gen , obj . first , 3 , pos ) )
2014-05-22 20:25:17 +03:00
{
2014-05-23 18:12:31 +03:00
logGlobal - > errorStream ( ) < < boost : : format ( " Failed to fill zone %d due to lack of space " ) % id ;
2014-05-22 20:25:17 +03:00
//TODO CLEANUP!
return false ;
}
2014-05-24 19:39:58 +03:00
logGlobal - > traceStream ( ) < < " Place found " ;
2014-05-22 20:25:17 +03:00
2014-06-01 17:31:15 +03:00
placeObject ( gen , obj . first , pos ) ;
2014-06-15 20:56:58 +03:00
guardObject ( gen , obj . first , obj . second ) ;
2014-05-22 20:25:17 +03:00
}
2014-06-15 09:48:46 +03:00
return true ;
}
void CRmgTemplateZone : : createTreasures ( CMapGenerator * gen )
{
const double minDistance = 3 ;
2014-06-04 11:16:08 +03:00
2014-05-22 20:25:17 +03:00
do {
int3 pos ;
2014-06-15 09:48:46 +03:00
if ( ! findPlaceForTreasurePile ( gen , minDistance , pos ) )
2014-05-22 20:25:17 +03:00
{
break ;
}
2014-06-04 11:16:08 +03:00
createTreasurePile ( gen , pos ) ;
2014-05-23 20:14:20 +03:00
2014-05-22 20:25:17 +03:00
} while ( true ) ;
2014-06-15 09:48:46 +03:00
}
2014-05-22 20:25:17 +03:00
2014-06-15 09:48:46 +03:00
void CRmgTemplateZone : : createObstacles ( CMapGenerator * gen )
2014-07-04 00:11:24 +03:00
{
if ( pos . z ) //underground
{
std : : vector < int3 > rockTiles ;
for ( auto tile : tileinfo )
2014-07-04 10:54:55 +03:00
{
2014-07-04 00:11:24 +03:00
if ( gen - > shouldBeBlocked ( tile ) )
{
2014-07-04 10:54:55 +03:00
bool placeRock = true ;
2014-07-04 00:11:24 +03:00
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);
// });
//}
}
2014-06-28 10:46:32 +03:00
//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 ) ;
}
}
}
}
2014-05-22 20:25:17 +03:00
auto sel = gen - > editManager - > getTerrainSelection ( ) ;
sel . clearSelection ( ) ;
2014-06-28 10:46:32 +03:00
auto tryToPlaceObstacleHere = [ this , gen ] ( int3 & tile ) - > bool
2014-05-22 20:25:17 +03:00
{
2014-06-28 10:46:32 +03:00
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 ;
} ;
2014-05-31 13:26:37 +03:00
2014-06-28 10:46:32 +03:00
for ( auto tile : tileinfo )
{
2014-05-30 22:23:41 +03:00
if ( gen - > shouldBeBlocked ( tile ) ) //fill tiles that should be blocked with obstacles
2014-05-22 20:25:17 +03:00
{
2014-06-28 10:46:32 +03:00
while ( ! tryToPlaceObstacleHere ( tile ) ) ;
}
else if ( gen - > isPossible ( tile ) )
{
//try to place random obstacle once - if not possible, leave it clear
tryToPlaceObstacleHere ( tile ) ;
2014-05-22 20:25:17 +03:00
}
}
2014-06-15 09:48:46 +03:00
}
bool CRmgTemplateZone : : fill ( CMapGenerator * gen )
{
addAllPossibleObjects ( gen ) ;
initTownType ( gen ) ;
initTerrainType ( gen ) ;
placeMines ( gen ) ;
createRequiredObjects ( gen ) ;
fractalize ( gen ) ; //after required objects are created and linked with their own paths
createTreasures ( gen ) ;
createObstacles ( gen ) ;
2014-05-23 18:12:31 +03:00
logGlobal - > infoStream ( ) < < boost : : format ( " Zone %d filled successfully " ) % id ;
2014-05-22 20:25:17 +03:00
return true ;
}
2014-06-07 10:09:50 +03:00
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 ;
}
2014-06-28 10:46:32 +03:00
bool CRmgTemplateZone : : canObstacleBePlacedHere ( CMapGenerator * gen , ObjectTemplate & temp , int3 & pos )
{
auto tilesBlockedByObject = temp . getBlockedOffsets ( ) ;
bool allTilesAvailable = true ;
for ( auto blockingTile : tilesBlockedByObject )
{
int3 t = pos + blockingTile ;
if ( ! gen - > map - > isInTheMap ( t ) | | ! ( gen - > isPossible ( t ) | | gen - > shouldBeBlocked ( t ) ) )
{
allTilesAvailable = false ; //if at least one tile is not possible, object can't be placed here
break ;
}
}
return allTilesAvailable ;
}
2014-05-22 20:25:17 +03:00
bool CRmgTemplateZone : : findPlaceForObject ( CMapGenerator * gen , CGObjectInstance * obj , si32 min_dist , int3 & pos )
{
2014-05-31 13:26:37 +03:00
//we need object apperance to deduce free tiles
if ( obj - > appearance . id = = Obj : : NO_OBJ )
{
2014-06-03 22:45:18 +03:00
auto templates = VLC - > objtypeh - > getHandlerFor ( obj - > ID , obj - > subID ) - > getTemplates ( gen - > map - > getTile ( getPos ( ) ) . terType ) ;
2014-05-31 13:26:37 +03:00
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 ( ) ;
}
2014-05-22 20:25:17 +03:00
//si32 min_dist = sqrt(tileinfo.size()/density);
int best_distance = 0 ;
bool result = false ;
2014-06-22 17:26:08 +03:00
//si32 w = gen->map->width;
//si32 h = gen->map->height;
2014-05-31 15:11:20 +03:00
2014-05-22 20:25:17 +03:00
//logGlobal->infoStream() << boost::format("Min dist for density %f is %d") % density % min_dist;
2014-06-07 11:13:59 +03:00
auto tilesBlockedByObject = obj - > getBlockedOffsets ( ) ;
for ( auto tile : tileinfo )
2014-05-22 20:25:17 +03:00
{
2014-06-07 11:13:59 +03:00
//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 ;
2014-05-31 15:11:20 +03:00
auto ti = gen - > getTile ( tile ) ;
2014-05-22 20:25:17 +03:00
auto dist = ti . getNearestObjectDistance ( ) ;
//avoid borders
2014-05-30 22:23:41 +03:00
if ( gen - > isPossible ( tile ) & & ( dist > = min_dist ) & & ( dist > best_distance ) )
2014-05-22 20:25:17 +03:00
{
2014-05-31 11:56:14 +03:00
bool allTilesAvailable = true ;
2014-06-07 11:13:59 +03:00
for ( auto blockingTile : tilesBlockedByObject )
2014-05-31 11:56:14 +03:00
{
2014-05-31 13:26:37 +03:00
int3 t = tile + blockingTile ;
if ( ! gen - > map - > isInTheMap ( t ) | | ! gen - > isPossible ( t ) )
2014-05-31 11:56:14 +03:00
{
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 ;
}
2014-05-22 20:25:17 +03:00
}
}
2014-05-31 11:56:14 +03:00
if ( result )
{
gen - > setOccupied ( pos , ETileType : : BLOCKED ) ; //block that tile
}
2014-05-22 20:25:17 +03:00
return result ;
}
2014-05-23 18:12:31 +03:00
void CRmgTemplateZone : : checkAndPlaceObject ( CMapGenerator * gen , CGObjectInstance * object , const int3 & pos )
2014-05-22 20:25:17 +03:00
{
2014-05-23 20:14:20 +03:00
if ( ! gen - > map - > isInTheMap ( pos ) )
throw rmgException ( boost : : to_string ( boost : : format ( " Position of object %d at %s is outside the map " ) % object - > id % object - > pos ( ) ) ) ;
2014-05-22 20:25:17 +03:00
object - > pos = pos ;
2014-05-23 18:12:31 +03:00
2014-05-23 20:14:20 +03:00
if ( object - > isVisitable ( ) & & ! gen - > map - > isInTheMap ( object - > visitablePos ( ) ) )
2014-05-23 18:12:31 +03:00
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 ( ) ) ) ;
}
2014-05-31 13:26:37 +03:00
if ( object - > appearance . id = = Obj : : NO_OBJ )
{
2014-07-03 18:24:28 +03:00
auto terrainType = gen - > map - > getTile ( pos ) . terType ;
auto templates = VLC - > objtypeh - > getHandlerFor ( object - > ID , object - > subID ) - > getTemplates ( terrainType ) ;
2014-05-31 13:26:37 +03:00
if ( templates . empty ( ) )
2014-07-03 18:24:28 +03:00
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)) ;
2014-05-23 20:14:20 +03:00
2014-05-31 13:26:37 +03:00
object - > appearance = templates . front ( ) ;
}
2014-05-23 20:14:20 +03:00
gen - > map - > addBlockVisTiles ( object ) ;
2014-05-22 20:25:17 +03:00
gen - > editManager - > insertObject ( object , pos ) ;
2014-06-15 09:48:46 +03:00
//logGlobal->traceStream() << boost::format ("Successfully inserted object (%d,%d) at pos %s") %object->ID %object->subID %pos();
2014-05-23 18:12:31 +03:00
}
void CRmgTemplateZone : : placeObject ( CMapGenerator * gen , CGObjectInstance * object , const int3 & pos )
{
2014-06-15 09:48:46 +03:00
//logGlobal->traceStream() << boost::format("Inserting object at %d %d") % pos.x % pos.y;
2014-05-23 18:12:31 +03:00
checkAndPlaceObject ( gen , object , pos ) ;
2014-05-22 20:25:17 +03:00
auto points = object - > getBlockedPos ( ) ;
if ( object - > isVisitable ( ) )
2014-05-29 13:42:05 +03:00
points . insert ( pos + object - > getVisitableOffset ( ) ) ;
points . insert ( pos ) ;
2014-06-01 15:10:44 +03:00
for ( auto p : points )
2014-05-22 20:25:17 +03:00
{
2014-06-01 15:10:44 +03:00
if ( gen - > map - > isInTheMap ( p ) )
2014-05-22 20:25:17 +03:00
{
2014-06-01 15:10:44 +03:00
gen - > setOccupied ( p , ETileType : : USED ) ;
2014-05-22 20:25:17 +03:00
}
}
2014-05-30 22:23:41 +03:00
for ( auto tile : tileinfo )
2014-05-22 20:25:17 +03:00
{
2014-07-03 10:39:26 +03:00
si32 d = pos . dist2dSQ ( tile ) ; //optimization, only relative distance is interesting
2014-05-30 22:23:41 +03:00
gen - > setNearestObjectDistance ( tile , std : : min ( d , gen - > getNearestObjectDistance ( tile ) ) ) ;
2014-05-22 20:25:17 +03:00
}
}
2014-07-03 18:24:28 +03:00
void CRmgTemplateZone : : placeAndGuardObject ( CMapGenerator * gen , CGObjectInstance * object , const int3 & pos , si32 str )
{
placeObject ( gen , object , pos ) ;
guardObject ( gen , object , str ) ;
}
2014-06-15 20:56:58 +03:00
std : : vector < int3 > CRmgTemplateZone : : getAccessibleOffsets ( CMapGenerator * gen , CGObjectInstance * object )
2014-05-22 20:25:17 +03:00
{
2014-06-15 20:56:58 +03:00
//get all tiles from which this object can be accessed
2014-05-23 20:14:20 +03:00
int3 visitable = object - > visitablePos ( ) ;
2014-05-22 20:25:17 +03:00
std : : vector < int3 > tiles ;
2014-06-15 10:39:30 +03:00
auto tilesBlockedByObject = object - > getBlockedPos ( ) ; //absolue value, as object is already placed
2014-05-30 22:23:41 +03:00
gen - > foreach_neighbour ( visitable , [ & ] ( int3 & pos )
2014-05-22 20:25:17 +03:00
{
2014-05-30 22:23:41 +03:00
if ( gen - > isPossible ( pos ) )
2014-05-22 20:25:17 +03:00
{
2014-06-15 10:39:30 +03:00
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 ) ;
}
}
2014-05-30 22:23:41 +03:00
} ;
} ) ;
2014-06-15 20:56:58 +03:00
return tiles ;
}
bool CRmgTemplateZone : : guardObject ( CMapGenerator * gen , CGObjectInstance * object , si32 str )
{
logGlobal - > traceStream ( ) < < boost : : format ( " Guard object at %s " ) % object - > pos ( ) ;
std : : vector < int3 > 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 , getPos ( ) , 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 ( ) ;
2014-05-22 20:25:17 +03:00
return false ;
}
2014-06-15 20:56:58 +03:00
if ( addMonster ( gen , guardTile , str ) ) //do not place obstacles around unguarded object
2014-06-04 21:59:01 +03:00
{
for ( auto pos : tiles )
2014-06-15 20:56:58 +03:00
{
if ( ! gen - > isFree ( pos ) )
gen - > setOccupied ( pos , ETileType : : BLOCKED ) ;
}
2014-06-04 21:59:01 +03:00
2014-06-15 20:56:58 +03:00
gen - > setOccupied ( guardTile , ETileType : : USED ) ;
2014-06-04 21:59:01 +03:00
}
2014-07-01 08:07:40 +03:00
else //allow no guard or other object in front of this object
{
for ( auto tile : tiles )
gen - > setOccupied ( tile , ETileType : : FREE ) ;
}
2014-06-01 15:10:44 +03:00
2014-05-22 20:25:17 +03:00
return true ;
}
2014-06-05 21:19:42 +03:00
ObjectInfo CRmgTemplateZone : : getRandomObject ( CMapGenerator * gen , ui32 value )
{
std : : vector < std : : pair < ui32 , ObjectInfo > > tresholds ;
ui32 total = 0 ;
ui32 minValue = 0.25f * value ;
//roulette wheel
for ( auto oi : possibleObjects )
{
if ( oi . value > = minValue & & oi . value < = value )
{
2014-06-07 15:02:57 +03:00
//TODO: check place for non-removable object
//problem: we need at least template for the object that does not yet exist
2014-06-05 21:19:42 +03:00
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 ;
2014-06-07 23:27:36 +03:00
if ( minValue > 20000 ) //we don't have object valuable enough
2014-06-05 21:19:42 +03:00
{
2014-06-07 23:27:36 +03:00
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 ;
2014-06-05 21:19:42 +03:00
}
int r = gen - > rand . nextInt ( 1 , total ) ;
for ( auto t : tresholds )
{
if ( r < = t . first )
return t . second ;
}
2014-06-22 17:26:08 +03:00
//FIXME: control reaches end of non-void function. Missing return?
2014-06-05 21:19:42 +03:00
}
void CRmgTemplateZone : : addAllPossibleObjects ( CMapGenerator * gen )
{
//TODO: move typical objects to config
ObjectInfo oi ;
static const Res : : ERes preciousRes [ ] = { Res : : ERes : : CRYSTAL , Res : : ERes : : GEMS , Res : : ERes : : MERCURY , Res : : ERes : : SULFUR } ;
for ( int i = 0 ; i < 4 ; i + + )
{
oi . generateObject = [ i , gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGResource ( ) ;
obj - > ID = Obj : : RESOURCE ;
obj - > subID = static_cast < si32 > ( preciousRes [ i ] ) ;
obj - > amount = 0 ;
return obj ;
} ;
oi . value = 1400 ;
oi . probability = 300 ;
possibleObjects . push_back ( oi ) ;
}
static const Res : : ERes woodOre [ ] = { Res : : ERes : : WOOD , Res : : ERes : : ORE } ;
for ( int i = 0 ; i < 2 ; i + + )
{
oi . generateObject = [ i , gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGResource ( ) ;
obj - > ID = Obj : : RESOURCE ;
obj - > subID = static_cast < si32 > ( woodOre [ i ] ) ;
obj - > amount = 0 ;
return obj ;
} ;
oi . value = 1400 ;
oi . probability = 300 ;
possibleObjects . push_back ( oi ) ;
}
oi . generateObject = [ gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGResource ( ) ;
obj - > ID = Obj : : RESOURCE ;
obj - > subID = static_cast < si32 > ( Res : : ERes : : GOLD ) ;
obj - > amount = 0 ;
return obj ;
} ;
oi . value = 750 ;
oi . probability = 300 ;
possibleObjects . push_back ( oi ) ;
oi . generateObject = [ gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGPickable ( ) ;
obj - > ID = Obj : : TREASURE_CHEST ;
obj - > subID = 0 ;
return obj ;
} ;
oi . value = 1500 ;
oi . probability = 1000 ;
possibleObjects . push_back ( oi ) ;
oi . generateObject = [ gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGArtifact ( ) ;
obj - > ID = Obj : : RANDOM_TREASURE_ART ;
obj - > subID = 0 ;
auto a = new CArtifactInstance ( ) ;
gen - > map - > addNewArtifactInstance ( a ) ;
obj - > storedArtifact = a ;
return obj ;
} ;
oi . value = 2000 ;
oi . probability = 150 ;
possibleObjects . push_back ( oi ) ;
oi . generateObject = [ gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGArtifact ( ) ;
obj - > ID = Obj : : RANDOM_MINOR_ART ;
obj - > subID = 0 ;
auto a = new CArtifactInstance ( ) ;
gen - > map - > addNewArtifactInstance ( a ) ;
obj - > storedArtifact = a ;
return obj ;
} ;
oi . value = 5000 ;
oi . probability = 150 ;
possibleObjects . push_back ( oi ) ;
oi . generateObject = [ gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGArtifact ( ) ;
obj - > ID = Obj : : RANDOM_MAJOR_ART ;
obj - > subID = 0 ;
auto a = new CArtifactInstance ( ) ;
gen - > map - > addNewArtifactInstance ( a ) ;
obj - > storedArtifact = a ;
return obj ;
} ;
oi . value = 10000 ;
oi . probability = 150 ;
possibleObjects . push_back ( oi ) ;
oi . generateObject = [ gen ] ( ) - > CGObjectInstance *
{
auto obj = new CGArtifact ( ) ;
obj - > ID = Obj : : RANDOM_RELIC_ART ;
obj - > subID = 0 ;
auto a = new CArtifactInstance ( ) ;
gen - > map - > addNewArtifactInstance ( a ) ;
obj - > storedArtifact = a ;
return obj ;
} ;
oi . value = 20000 ;
oi . probability = 150 ;
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 < SpellID > 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 ) ;
}
2014-06-07 15:02:57 +03:00
//non-removable object for test
2014-06-07 16:51:03 +03:00
//oi.generateObject = [gen]() -> CGObjectInstance *
//{
// auto obj = new CGMagicWell();
// obj->ID = Obj::MAGIC_WELL;
// obj->subID = 0;
// return obj;
//};
//oi.value = 250;
//oi.probability = 100;
//possibleObjects.push_back (oi);
//oi.generateObject = [gen]() -> CGObjectInstance *
//{
// auto obj = new CGObelisk();
// obj->ID = Obj::OBELISK;
// obj->subID = 0;
// return obj;
//};
//oi.value = 3500;
//oi.probability = 200;
//possibleObjects.push_back (oi);
//oi.generateObject = [gen]() -> CGObjectInstance *
//{
// auto obj = new CBank();
// obj->ID = Obj::CREATURE_BANK;
// obj->subID = 5; //naga bank
// return obj;
//};
//oi.value = 3000;
//oi.probability = 100;
//possibleObjects.push_back (oi);
2014-06-07 15:02:57 +03:00
2014-06-05 21:19:42 +03:00
}