2023-10-03 17:23:56 +03:00

264 lines
8.6 KiB
C++

//============ Copyright (c) Valve Corporation, All rights reserved. ============
//
//
//
//===============================================================================
#include "asw_npcs.h"
#include "asw_mission_chooser.h"
#include "TileSource/MapLayout.h"
#include "TileSource/Room.h"
#include "TileSource/RoomTemplate.h"
#include "KeyValues.h"
#include "asw_spawn_selection.h"
#include "layout_system/tilegen_layout_system.h"
ConVar asw_encounters_distance_min( "asw_encounters_distance_min", "900", FCVAR_CHEAT, "Min distance between edges of encounter circles" );
ConVar asw_encounter_radius_min( "asw_encounter_radius_min", "384.0f", FCVAR_CHEAT );
ConVar asw_encounter_radius_max( "asw_encounter_radius_max", "512.0f", FCVAR_CHEAT );
// TODO: Parameters exposed to level designer?
void CASWMissionChooserNPCs::InitFixedSpawns( CLayoutSystem *pLayoutSystem, CMapLayout *pLayout )
{
// init the spawn set for this mission
KeyValues *pGenerationOptions = pLayout->GetGenerationOptions();
if ( !pGenerationOptions )
{
Warning( "Error placed fixed alien spawns, no generation options in this layout." );
return;
}
bool bChosenSpawnSet = false;
const char *szNamedSpawnSet = pGenerationOptions->GetString( "AlienSpawnSet" );
if ( szNamedSpawnSet && szNamedSpawnSet[0] )
{
bChosenSpawnSet = SpawnSelection()->SetCurrentSpawnSet( szNamedSpawnSet );
}
if ( !bChosenSpawnSet )
{
SpawnSelection()->SetCurrentSpawnSet( pGenerationOptions->GetInt( "Difficulty", 5 ) );
}
// if we have any rooms with the alien encounter tag, then just use those for fixed spawn locations
bool bAlienEncounterTag = false;
int iRooms = pLayout->m_PlacedRooms.Count();
for ( int i = 0; i < iRooms; i++ )
{
CRoom *pRoom = pLayout->m_PlacedRooms[i];
if ( pRoom && pRoom->m_pRoomTemplate && pRoom->m_pRoomTemplate->HasTag( "AlienEncounter" ) )
{
bAlienEncounterTag = true;
CASW_Encounter *pEncounter = new CASW_Encounter();
// pick a random spot in this room
Vector vecWorldMins, vecWorldMaxs;
pRoom->GetWorldBounds( &vecWorldMins, &vecWorldMaxs );
Vector vecPos = vecWorldMins + pLayoutSystem->GetRandomFloat( 0, 1 ) * ( vecWorldMaxs - vecWorldMins );
vecPos.z = 0;
pEncounter->SetEncounterPosition( vecPos );
pEncounter->SetEncounterRadius( pLayoutSystem->GetRandomFloat( asw_encounter_radius_min.GetFloat(), asw_encounter_radius_max.GetFloat() ) );
// add spawn defs
// TODO: more spawns in bigger rooms? or rooms with higher weights?
int iSpawnsPerEncounter = pLayoutSystem->GetRandomInt( CurrentSpawnSet()->GetMinSpawnsPerEncounter(), CurrentSpawnSet()->GetMaxSpawnsPerEncounter() );
for ( int i = 0; i < iSpawnsPerEncounter; i++ )
{
CASW_Spawn_Definition* pSpawnDef = CurrentSpawnSet()->GetSpawnDef( ASW_NPC_SPAWN_TYPE_FIXED );
if ( !pSpawnDef )
continue;
pEncounter->AddSpawnDef( pSpawnDef );
}
pLayout->m_Encounters.AddToTail( pEncounter );
}
}
if ( bAlienEncounterTag )
{
pLayout->MarkEncounterRooms();
return;
}
// find area of the mission
int iTotalArea = 0;
for ( int i = 0; i < iRooms; i++ )
{
CRoom *pRoom = pLayout->m_PlacedRooms[i];
iTotalArea += ( pRoom->m_pRoomTemplate->GetTilesX() * ASW_TILE_SIZE ) * ( pRoom->m_pRoomTemplate->GetTilesY() * ASW_TILE_SIZE );
}
// decide how many encounters we want
int iEncounters = pLayoutSystem->GetRandomInt( CurrentSpawnSet()->GetMinEncounters(), CurrentSpawnSet()->GetMaxEncounters() );
// distance between encounters
//float flMinDistance = asw_encounters_distance_min.GetFloat();
// randomly pick rooms for the encounters to be in, using the room weights
CUtlVector<CRoom*> candidates;
float flTotalWeight = 0;
for ( int i = 0; i < iRooms; i++ )
{
CRoom *pRoom = pLayout->m_PlacedRooms[i];
if ( pRoom->GetSpawnWeight() > 0
&& !pRoom->m_pRoomTemplate->IsEscapeRoom()
&& !pRoom->m_pRoomTemplate->IsStartRoom()
&& !pRoom->m_pRoomTemplate->IsBorderRoom() )
{
flTotalWeight += pRoom->GetSpawnWeight();
candidates.AddToTail( pRoom );
}
}
for ( int e = 0; e < iEncounters; e++ )
{
float flChosen = pLayoutSystem->GetRandomFloat( 0, flTotalWeight );
CRoom *pChosenRoom = NULL;
for ( int i = 0; i < candidates.Count(); i++ )
{
CRoom *pRoom = candidates[i];
flChosen -= pRoom->GetSpawnWeight();
if ( flChosen <= 0 )
{
pChosenRoom = pRoom;
break;
}
}
if ( !pChosenRoom )
continue;
CASW_Encounter *pEncounter = new CASW_Encounter();
// pick a random spot in this room
Vector vecWorldMins, vecWorldMaxs;
pChosenRoom->GetWorldBounds( &vecWorldMins, &vecWorldMaxs );
//Vector vecPos = vecWorldMins + pLayoutSystem->GetRandomFloat( 0, 1 ) * ( vecWorldMaxs - vecWorldMins );
Vector vecPos = ( vecWorldMins + vecWorldMaxs ) * 0.5f; // center of the room
vecPos.z = 0;
pEncounter->SetEncounterPosition( vecPos );
pEncounter->SetEncounterRadius( pLayoutSystem->GetRandomFloat( asw_encounter_radius_min.GetFloat(), asw_encounter_radius_max.GetFloat() ) );
// add spawn defs
// TODO: more spawns in bigger rooms? or rooms with higher weights?
int iSpawnsPerEncounter = pLayoutSystem->GetRandomInt( CurrentSpawnSet()->GetMinSpawnsPerEncounter(), CurrentSpawnSet()->GetMaxSpawnsPerEncounter() );
for ( int i = 0; i < iSpawnsPerEncounter; i++ )
{
CASW_Spawn_Definition* pSpawnDef = CurrentSpawnSet()->GetSpawnDef( ASW_NPC_SPAWN_TYPE_FIXED );
if ( !pSpawnDef )
continue;
pEncounter->AddSpawnDef( pSpawnDef );
}
pLayout->m_Encounters.AddToTail( pEncounter );
}
PushEncountersApart( pLayout );
pLayout->MarkEncounterRooms();
}
void CASWMissionChooserNPCs::PushEncountersApart( CMapLayout *pLayout )
{
// get a list of valid rooms
int iRooms = pLayout->m_PlacedRooms.Count();
CUtlVector<CRoom*> candidates;
float flTotalWeight = 0;
for ( int i = 0; i < iRooms; i++ )
{
CRoom *pRoom = pLayout->m_PlacedRooms[i];
if ( pRoom->GetSpawnWeight() > 0
&& !pRoom->m_pRoomTemplate->IsEscapeRoom()
&& !pRoom->m_pRoomTemplate->IsStartRoom()
&& !pRoom->m_pRoomTemplate->IsBorderRoom() )
{
// skip 1x1 rooms
if ( pRoom->m_pRoomTemplate->GetTilesX() * pRoom->m_pRoomTemplate->GetTilesY() <= 1 )
continue;
flTotalWeight += pRoom->GetSpawnWeight();
candidates.AddToTail( pRoom );
}
}
// push the encounters apart
int iSteps = 6;
for ( int i = 0; i < iSteps; i++ )
{
int iEncounters = pLayout->m_Encounters.Count();
for ( int e = 0; e < iEncounters; e++ )
{
Vector vecPush = vec3_origin;
CASW_Encounter *pEncounter = pLayout->m_Encounters[ e ];
// accumulate a force from all other nearby encounters
for ( int k = 0; k < iEncounters; k++ )
{
if ( e == k )
continue;
CASW_Encounter *pOther = pLayout->m_Encounters[ k ];
Vector dir = pEncounter->GetEncounterPosition() - pOther->GetEncounterPosition(); // direction from other to us
float dist = dir.NormalizeInPlace();
float flMinDistance = asw_encounters_distance_min.GetFloat() + pEncounter->GetEncounterRadius() + pOther->GetEncounterRadius();
if ( dist < flMinDistance )
{
float flPush = flMinDistance - dist; // push us exactly out of the min
vecPush += dir * flPush;
}
}
pEncounter->SetEncounterPosition( pEncounter->GetEncounterPosition() + vecPush );
// clamp encounters to room bounds
CRoom *pClosestRoom = NULL;
float flClosestDist = 65535.0f;
bool bWithinARoom = false;
Vector pos = pEncounter->GetEncounterPosition();
for ( int k = 0; k < candidates.Count(); k++ )
{
CRoom *pRoom = candidates[k];
Vector vecMins, vecMaxs;
pRoom->GetWorldBounds( &vecMins, &vecMaxs );
if ( pos.x >= vecMins.x && pos.x <= vecMaxs.x
&& pos.y >= vecMins.y && pos.y <= vecMaxs.y )
{
bWithinARoom = true;
break;
}
Vector vecMiddle = ( vecMins + vecMaxs ) * 0.5f;
float flDist = vecMiddle.DistTo( pos );
if ( flDist < flClosestDist )
{
flClosestDist = flDist;
pClosestRoom = pRoom;
}
}
// if encounter wasn't in any room, then clamp to the closest
if ( !bWithinARoom && pClosestRoom )
{
Vector vecMins, vecMaxs;
pClosestRoom->GetWorldBounds( &vecMins, &vecMaxs );
pos.x = clamp( pos.x, vecMins.x, vecMaxs.x );
pos.y = clamp( pos.y, vecMins.y, vecMaxs.y );
pEncounter->SetEncounterPosition( pos );
}
/*
// put encounter in the center of its room
for ( int k = 0; k < candidates.Count(); k++ )
{
CRoom *pRoom = candidates[k];
Vector vecMins, vecMaxs;
pRoom->GetWorldBounds( &vecMins, &vecMaxs );
if ( pos.x >= vecMins.x && pos.x <= vecMaxs.x
&& pos.y >= vecMins.y && pos.y <= vecMaxs.y )
{
Vector mid = ( vecMins + vecMaxs ) * 0.5f;
mid.z = 0;
pEncounter->SetEncounterPosition ( mid );
break;
}
}
*/
}
}
}