mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-12 03:32:11 +08:00
1028 lines
30 KiB
C++
1028 lines
30 KiB
C++
#include "VMFExporter.h"
|
|
#include "KeyValues.h"
|
|
|
|
#include "TileSource/RoomTemplate.h"
|
|
#include "TileSource/Room.h"
|
|
#include "TileSource/LevelTheme.h"
|
|
#include "TileSource/MapLayout.h"
|
|
#include "TileGenDialog.h"
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include <tier0/memdbgon.h>
|
|
|
|
ConVar tilegen_use_instancing( "tilegen_use_instancing", "0", FCVAR_REPLICATED );
|
|
|
|
// TODO: Read room templates into keyvalues and output them after the whole room template has been loaded
|
|
// This way we can modify state based on the complete picture of that room, rather than just making changes on a key by key basis
|
|
|
|
VMFExporter::VMFExporter()
|
|
{
|
|
m_pMapLayout = NULL;
|
|
|
|
Init();
|
|
}
|
|
|
|
VMFExporter::~VMFExporter()
|
|
{
|
|
ClearExportErrors();
|
|
}
|
|
|
|
void VMFExporter::Init()
|
|
{
|
|
m_szLastExporterError[0] = '\0';
|
|
m_SideTranslations.Purge();
|
|
m_NodeTranslations.Purge();
|
|
m_vecLastPlaneOffset = vec3_origin;
|
|
m_bWritingLevelContainer = false;
|
|
m_iEntityCount = 1; // 1 already, from the worldspawn
|
|
m_iSideCount = 0;
|
|
m_pTemplateKeys = NULL;
|
|
|
|
m_iNextNodeID = 0;
|
|
m_pRoom = NULL;
|
|
m_iCurrentRoom = 0;
|
|
|
|
m_iMapExtents_XMin = 0;
|
|
m_iMapExtents_YMin = 0;
|
|
m_iMapExtents_XMax = 0;
|
|
m_iMapExtents_YMax = 0;
|
|
m_pExportKeys = NULL;
|
|
m_vecStartRoomOrigin = vec3_origin;
|
|
ClearExportErrors();
|
|
}
|
|
|
|
void VMFExporter::ExportError( const char *pMsg, ... )
|
|
{
|
|
char msg[4096];
|
|
va_list marker;
|
|
va_start( marker, pMsg );
|
|
Q_vsnprintf( msg, sizeof( msg ), pMsg, marker );
|
|
va_end( marker );
|
|
|
|
m_ExportErrors.AddToTail( TileGenCopyString( msg ) );
|
|
}
|
|
|
|
void VMFExporter::ClearExportErrors()
|
|
{
|
|
m_ExportErrors.PurgeAndDeleteElements();
|
|
}
|
|
|
|
bool VMFExporter::ShowExportErrors()
|
|
{
|
|
if ( m_ExportErrors.Count() <= 0 )
|
|
return false;
|
|
char msg[4096];
|
|
msg[0] = 0;
|
|
for ( int i=0; i<m_ExportErrors.Count(); i++ )
|
|
{
|
|
Q_snprintf( msg, sizeof( msg ), "%s\n%s", msg, m_ExportErrors[i] );
|
|
}
|
|
VGUIMessageBox( g_pTileGenDialog, "Export problems:", msg );
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ExportVMF( CMapLayout* pLayout, const char *mapname, bool bPopupWarnings )
|
|
{
|
|
m_bPopupWarnings = bPopupWarnings;
|
|
|
|
Init();
|
|
|
|
m_pMapLayout = pLayout;
|
|
|
|
if ( pLayout->m_PlacedRooms.Count() <= 0 )
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to export: No rooms placed in the map layout!\n" );
|
|
return false;
|
|
}
|
|
|
|
// see if we have a start room
|
|
bool bHasStartRoom = false;
|
|
for ( int i = 0 ; i < pLayout->m_PlacedRooms.Count() ; i++ )
|
|
{
|
|
if ( pLayout->m_PlacedRooms[i]->m_pRoomTemplate->IsStartRoom() )
|
|
{
|
|
int half_map_size = MAP_LAYOUT_TILES_WIDE * 0.5f; // shift back so the middle of our grid is the origin
|
|
m_vecStartRoomOrigin.x = ( pLayout->m_PlacedRooms[i]->m_iPosX - half_map_size ) * ASW_TILE_SIZE;
|
|
m_vecStartRoomOrigin.y = ( pLayout->m_PlacedRooms[i]->m_iPosY - half_map_size ) * ASW_TILE_SIZE;
|
|
bHasStartRoom = true;
|
|
break;
|
|
}
|
|
}
|
|
LoadUniqueKeyList();
|
|
|
|
m_iNextNodeID = 0;
|
|
|
|
m_pExportKeys = new KeyValues( "ExportKeys" );
|
|
|
|
m_pExportKeys->AddSubKey( GetVersionInfo() );
|
|
m_pExportKeys->AddSubKey( GetDefaultVisGroups() );
|
|
m_pExportKeys->AddSubKey( GetViewSettings() );
|
|
m_pExportWorldKeys = GetDefaultWorldChunk();
|
|
if ( !m_pExportWorldKeys )
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to save world chunk start\n");
|
|
return false;
|
|
}
|
|
m_pExportKeys->AddSubKey( m_pExportWorldKeys );
|
|
|
|
|
|
// save out the big cube the whole level sits in
|
|
if ( !AddLevelContainer() )
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to save level container\n");
|
|
return false;
|
|
}
|
|
|
|
if ( tilegen_use_instancing.GetBool() )
|
|
{
|
|
int nLogicalRooms = m_pMapLayout->m_LogicalRooms.Count();
|
|
int nPlacedRooms = m_pMapLayout->m_PlacedRooms.Count();
|
|
|
|
m_pRoom = NULL;
|
|
for ( int i = 0; i < nLogicalRooms; ++ i )
|
|
{
|
|
AddRoomInstance( m_pMapLayout->m_LogicalRooms[i] );
|
|
}
|
|
|
|
for ( int i = 0; i < nPlacedRooms; ++ i )
|
|
{
|
|
m_pRoom = m_pMapLayout->m_PlacedRooms[i];
|
|
AddRoomInstance( m_pRoom->m_pRoomTemplate, i );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// write out logical room solids
|
|
int iLogicalRooms = m_pMapLayout->m_LogicalRooms.Count();
|
|
m_pRoom = NULL;
|
|
for ( int i = 0 ; i < iLogicalRooms ; i++ )
|
|
{
|
|
// start logical room IDs at 5000 (assumes we'll never place 5000 real rooms)
|
|
m_iCurrentRoom = 5000 + i;
|
|
CRoomTemplate *pRoomTemplate = m_pMapLayout->m_LogicalRooms[i];
|
|
if ( !pRoomTemplate )
|
|
continue;
|
|
|
|
if ( !AddRoomTemplateSolids( pRoomTemplate ) )
|
|
return false;
|
|
}
|
|
|
|
// go through each CRoom and write out its world solids
|
|
int iRooms = m_pMapLayout->m_PlacedRooms.Count();
|
|
for ( m_iCurrentRoom = 0 ; m_iCurrentRoom<iRooms ; m_iCurrentRoom++)
|
|
{
|
|
m_pRoom = m_pMapLayout->m_PlacedRooms[m_iCurrentRoom];
|
|
if (!m_pRoom)
|
|
continue;
|
|
const CRoomTemplate *pRoomTemplate = m_pRoom->m_pRoomTemplate;
|
|
if (!pRoomTemplate)
|
|
continue;
|
|
|
|
if ( !AddRoomTemplateSolids( pRoomTemplate ) )
|
|
return false;
|
|
}
|
|
|
|
// write out logical room entities
|
|
m_pRoom = NULL;
|
|
for ( int i = 0 ; i < iLogicalRooms ; i++ )
|
|
{
|
|
// start logical room IDs at 5000 (assumes we'll never place 5000 real rooms)
|
|
m_iCurrentRoom = 5000 + i;
|
|
CRoomTemplate *pRoomTemplate = m_pMapLayout->m_LogicalRooms[i];
|
|
if ( !pRoomTemplate )
|
|
continue;
|
|
|
|
if ( !AddRoomTemplateEntities( pRoomTemplate ) )
|
|
return false;
|
|
}
|
|
|
|
// go through each CRoom and add its entities
|
|
for ( m_iCurrentRoom = 0 ; m_iCurrentRoom<iRooms ; m_iCurrentRoom++)
|
|
{
|
|
m_pRoom = m_pMapLayout->m_PlacedRooms[m_iCurrentRoom];
|
|
if (!m_pRoom)
|
|
continue;
|
|
const CRoomTemplate *pRoomTemplate = m_pRoom->m_pRoomTemplate;
|
|
if (!pRoomTemplate)
|
|
continue;
|
|
|
|
if ( !AddRoomTemplateEntities( pRoomTemplate ) )
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// add some player starts to the map in the tile the user selected
|
|
if ( !bHasStartRoom )
|
|
{
|
|
m_pExportKeys->AddSubKey( GetPlayerStarts() );
|
|
}
|
|
|
|
m_pExportKeys->AddSubKey( GetGameRulesProxy() );
|
|
m_pExportKeys->AddSubKey( GetDefaultCamera() );
|
|
|
|
// save out the export keys
|
|
char filename[512];
|
|
Q_snprintf( filename, sizeof(filename), "maps\\%s", mapname );
|
|
Q_SetExtension( filename, "vmf", sizeof( filename ) );
|
|
CUtlBuffer buf( 0, 0, CUtlBuffer::TEXT_BUFFER );
|
|
for ( KeyValues *pKey = m_pExportKeys->GetFirstSubKey(); pKey; pKey = pKey->GetNextKey() )
|
|
{
|
|
pKey->RecursiveSaveToFile( buf, 0 );
|
|
}
|
|
if ( !g_pFullFileSystem->WriteFile( filename, "GAME", buf ) )
|
|
{
|
|
Msg( "Failed to SaveToFile %s\n", filename );
|
|
return false;
|
|
}
|
|
|
|
// save the map layout there (so the game can get information about rooms during play)
|
|
Q_snprintf( filename, sizeof( filename ), "maps\\%s", mapname );
|
|
Q_SetExtension( filename, "layout", sizeof( filename ) );
|
|
if ( !m_pMapLayout->SaveMapLayout( filename ) )
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to save .layout file\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::AddRoomTemplateSolids( const CRoomTemplate *pRoomTemplate )
|
|
{
|
|
// open its vmf file
|
|
char roomvmfname[MAX_PATH];
|
|
Q_snprintf(roomvmfname, sizeof(roomvmfname), "tilegen/roomtemplates/%s/%s.vmf",
|
|
pRoomTemplate->m_pLevelTheme->m_szName,
|
|
pRoomTemplate->GetFullName() );
|
|
m_pTemplateKeys = new KeyValues( "RoomTemplateVMF" );
|
|
m_pTemplateKeys->LoadFromFile( g_pFullFileSystem, roomvmfname, "GAME" );
|
|
|
|
// look for world key
|
|
for ( KeyValues *pKeys = m_pTemplateKeys; pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "world" ) ) // find the world key in our room template
|
|
{
|
|
if ( !ProcessWorld( pKeys ) ) // fix up solid positions
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to copy world from room %s\n", pRoomTemplate->GetFullName() );
|
|
return false;
|
|
}
|
|
for ( KeyValues *pSubKey = pKeys->GetFirstSubKey(); pSubKey; pSubKey = pSubKey->GetNextKey() ) // convert each solid to a func_detail entity
|
|
{
|
|
if ( !Q_stricmp( pSubKey->GetName(), "solid" ) )
|
|
{
|
|
if ( IsDisplacementBrush( pSubKey ) )
|
|
{
|
|
// add to world section
|
|
m_pExportWorldKeys->AddSubKey( pSubKey->MakeCopy() );
|
|
}
|
|
else
|
|
{
|
|
// put into entity section as a func_detail
|
|
KeyValues *pFuncDetail = new KeyValues( "entity" );
|
|
pFuncDetail->SetInt( "id", ++m_iEntityCount );
|
|
pFuncDetail->SetString( "classname", "func_detail" );
|
|
pFuncDetail->AddSubKey( pSubKey->MakeCopy() );
|
|
m_pExportKeys->AddSubKey( pFuncDetail );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
m_pTemplateKeys->deleteThis();
|
|
m_pTemplateKeys = NULL;
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::AddRoomTemplateEntities( const CRoomTemplate *pRoomTemplate )
|
|
{
|
|
m_SideTranslations.PurgeAndDeleteElements();
|
|
m_NodeTranslations.PurgeAndDeleteElements();
|
|
|
|
// reopen the source vmf
|
|
char roomvmfname[MAX_PATH];
|
|
Q_snprintf( roomvmfname, sizeof(roomvmfname), "tilegen/roomtemplates/%s/%s.vmf",
|
|
pRoomTemplate->m_pLevelTheme->m_szName,
|
|
pRoomTemplate->GetFullName() );
|
|
m_pTemplateKeys = new KeyValues( "RoomTemplateVMF" );
|
|
m_pTemplateKeys->LoadFromFile( g_pFullFileSystem, roomvmfname, "GAME" );
|
|
|
|
// make all node IDs unique
|
|
MakeNodeIDsUnique();
|
|
|
|
// sets priority of objective entities based on the generation options
|
|
ReorderObjectives( pRoomTemplate, m_pTemplateKeys );
|
|
|
|
// look for entity keys
|
|
for ( KeyValues *pKeys = m_pTemplateKeys; pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "entity" ) )
|
|
{
|
|
if ( !ProcessEntity( pKeys ) )
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to copy entity from room %s\n", pRoomTemplate->GetFullName());
|
|
return false;
|
|
}
|
|
m_pExportKeys->AddSubKey( pKeys->MakeCopy() );
|
|
}
|
|
}
|
|
|
|
m_pTemplateKeys->deleteThis();
|
|
m_pTemplateKeys = NULL;
|
|
return true;
|
|
};
|
|
|
|
bool VMFExporter::IsDisplacementBrush( KeyValues *pSolidKeys )
|
|
{
|
|
// go through each side
|
|
for ( KeyValues *pKeys = pSolidKeys->GetFirstSubKey(); pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "side" ) && pKeys->FindKey( "dispinfo" ) )
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool VMFExporter::ProcessWorld( KeyValues *pWorldKeys )
|
|
{
|
|
// look for solid keys
|
|
KeyValues *pSolidKeys = pWorldKeys->GetFirstSubKey();
|
|
while ( pSolidKeys )
|
|
{
|
|
if ( !Q_stricmp( pSolidKeys->GetName(), "solid" ) )
|
|
{
|
|
if ( !ProcessSolid( pSolidKeys ) )
|
|
return false;
|
|
}
|
|
pSolidKeys = pSolidKeys->GetNextKey();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessSolid( KeyValues *pSolidKey )
|
|
{
|
|
pSolidKey->SetInt( "id", ++m_iEntityCount );
|
|
|
|
// now process each side
|
|
KeyValues *pKeys = pSolidKey->GetFirstSubKey();
|
|
while ( pKeys )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "side" ) )
|
|
{
|
|
if ( !ProcessSide( pKeys ) )
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
if ( !ProcessGenericRecursive( pKeys ) )
|
|
return false;
|
|
}
|
|
pKeys = pKeys->GetNextKey();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessSide( KeyValues *pSideKey )
|
|
{
|
|
for ( KeyValues *pKeys = pSideKey->GetFirstSubKey(); pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "dispinfo" ) )
|
|
{
|
|
if ( !ProcessGenericRecursive( pKeys ) )
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
if ( !ProcessSideKey( pKeys ) )
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessGenericRecursive( KeyValues *pKey )
|
|
{
|
|
const char *szKey = pKey->GetName();
|
|
const char *szValue = pKey->GetString();
|
|
|
|
if ( !Q_stricmp( szKey, "startposition" ) )
|
|
{
|
|
Vector Origin;
|
|
int nRead = sscanf(szValue, "[%f %f %f]", &Origin[0], &Origin[1], &Origin[2]);
|
|
|
|
if (nRead != 3)
|
|
return false;
|
|
|
|
// move the points to where our room is
|
|
Origin += GetCurrentRoomOffset();
|
|
|
|
char buffer[256];
|
|
Q_snprintf(buffer, sizeof(buffer), "[%f %f %f]", Origin[0], Origin[1], Origin[2]);
|
|
|
|
pKey->SetStringValue( buffer );
|
|
}
|
|
|
|
if ( pKey->GetFirstSubKey() ) // if this keyvalues entry has subkeys, then process them
|
|
{
|
|
for ( KeyValues *pKeys = pKey->GetFirstSubKey(); pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
if ( !ProcessGenericRecursive( pKeys ) )
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessSideKey( KeyValues *pKey )
|
|
{
|
|
const char *szKey = pKey->GetName();
|
|
const char *szValue = pKey->GetString();
|
|
|
|
if (!stricmp(szKey, "id"))
|
|
{
|
|
// store the side
|
|
SideTranslation_t *pSideTranslation = new SideTranslation_t;
|
|
pSideTranslation->m_iOriginalSide = atoi( szValue );
|
|
|
|
// give this side a unique ID
|
|
char buffer[32];
|
|
Q_snprintf(buffer, sizeof(buffer), "%d", ++m_iSideCount );
|
|
pKey->SetStringValue( buffer );
|
|
|
|
pSideTranslation->m_iNewSide = m_iSideCount;
|
|
m_SideTranslations.AddToTail( pSideTranslation );
|
|
|
|
return true;
|
|
}
|
|
else if (!stricmp(szKey, "plane"))
|
|
{
|
|
Vector planepts[3];
|
|
int nRead = sscanf(szValue, "(%f %f %f) (%f %f %f) (%f %f %f)",
|
|
&planepts[0][0], &planepts[0][1], &planepts[0][2],
|
|
&planepts[1][0], &planepts[1][1], &planepts[1][2],
|
|
&planepts[2][0], &planepts[2][1], &planepts[2][2]);
|
|
|
|
if (nRead != 9)
|
|
return false;
|
|
|
|
if (m_bWritingLevelContainer)
|
|
{
|
|
// we're currently writing out the big cube that the level sits in
|
|
// need to adjust the x and y coordinates of this box to encase the whole map layout
|
|
|
|
for (int p=0;p<3;p++)
|
|
{
|
|
if (planepts[p][0] < 0)
|
|
planepts[p][0] += m_iMapExtents_XMin * ASW_TILE_SIZE;
|
|
else
|
|
planepts[p][0] += m_iMapExtents_XMax * ASW_TILE_SIZE;
|
|
if (planepts[p][1] < 0)
|
|
planepts[p][1] += m_iMapExtents_YMin * ASW_TILE_SIZE;
|
|
else
|
|
planepts[p][1] += m_iMapExtents_YMax * ASW_TILE_SIZE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
|
|
// move the points to where our room is
|
|
for (int p=0;p<3;p++)
|
|
{
|
|
planepts[p] += GetCurrentRoomOffset();
|
|
}
|
|
|
|
// store these so we know how much to shift texture alignment
|
|
m_vecLastPlaneOffset = GetCurrentRoomOffset();
|
|
}
|
|
|
|
char buffer[256];
|
|
Q_snprintf(buffer, sizeof(buffer), "(%f %f %f) (%f %f %f) (%f %f %f)",
|
|
planepts[0][0], planepts[0][1], planepts[0][2],
|
|
planepts[1][0], planepts[1][1], planepts[1][2],
|
|
planepts[2][0], planepts[2][1], planepts[2][2]);
|
|
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
else if ( !stricmp(szKey, "uaxis") || !stricmp(szKey, "vaxis") ) // fix texture alignment (similar to moving a brush with texture lock turned on in hammer)
|
|
{
|
|
// NOTE: This assumes the plane was read in previously and set m_vecLastPlaneOffset
|
|
Vector axis;
|
|
float offset, scale;
|
|
offset = 0;
|
|
scale = 1.0f;
|
|
if ( sscanf( szValue, "[%f %f %f %f] %f", &axis.x, &axis.y, &axis.z, &offset, &scale ) == 5 )
|
|
{
|
|
offset -= m_vecLastPlaneOffset.Dot( axis ) / scale;
|
|
|
|
char buffer[256];
|
|
Q_snprintf( buffer, sizeof(buffer), "[%f %f %f %f] %f",
|
|
axis.x, axis.y, axis.z, offset, scale );
|
|
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
Msg( "Error loading in uvaxis\n" );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessEntity( KeyValues *pEntityKeys )
|
|
{
|
|
for ( KeyValues *pKeys = pEntityKeys->GetFirstSubKey(); pKeys; )
|
|
{
|
|
if ( pKeys->GetFirstSubKey() )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "solid" ) )
|
|
{
|
|
if ( !ProcessSolid( pKeys ) )
|
|
return false;
|
|
}
|
|
else if ( !Q_stricmp( pKeys->GetName(), "connections" ) )
|
|
{
|
|
if ( !ProcessConnections( pKeys ) )
|
|
return false;
|
|
}
|
|
else if ( !Q_stricmp( pKeys->GetName(), "editor" ) ) // remove editor keys
|
|
{
|
|
KeyValues *pRemoveKey = pKeys;
|
|
pKeys = pKeys->GetNextKey();
|
|
pEntityKeys->RemoveSubKey( pRemoveKey );
|
|
pRemoveKey->deleteThis();
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( !ProcessEntityKey( pKeys ) )
|
|
return false;
|
|
}
|
|
pKeys = pKeys->GetNextKey();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessEntityKey( KeyValues *pKey )
|
|
{
|
|
const char *szKey = pKey->GetName();
|
|
const char *szValue = pKey->GetString();
|
|
|
|
if (!stricmp(szKey, "id"))
|
|
{
|
|
// give this entity a unique ID
|
|
char buffer[32];
|
|
Q_snprintf(buffer, sizeof(buffer), "%d", ++m_iEntityCount );
|
|
pKey->SetStringValue( buffer );
|
|
|
|
return true;
|
|
}
|
|
else if (!stricmp(szKey, "origin"))
|
|
{
|
|
Vector Origin;
|
|
int nRead = sscanf(szValue, "%f %f %f", &Origin[0], &Origin[1], &Origin[2]);
|
|
|
|
if (nRead != 3)
|
|
return false;
|
|
|
|
// move the points to where our room is
|
|
Origin += GetCurrentRoomOffset();
|
|
|
|
char buffer[256];
|
|
Q_snprintf(buffer, sizeof(buffer), "%f %f %f", Origin[0], Origin[1], Origin[2]);
|
|
pKey->SetStringValue( buffer );
|
|
|
|
return true;
|
|
}
|
|
// check for overlay values
|
|
if (!stricmp(szKey, "sides"))
|
|
{
|
|
// just deal with one side for now
|
|
// TODO: handle multiple sides in the value and handle sides in the world block
|
|
int iSide = atoi(szValue);
|
|
for (int i=0;i<m_SideTranslations.Count();i++)
|
|
{
|
|
SideTranslation_t *pSideTranslation = m_SideTranslations[i];
|
|
if ( pSideTranslation->m_iOriginalSide == iSide )
|
|
{
|
|
char buffer[32];
|
|
Q_snprintf(buffer, sizeof(buffer), "%d", pSideTranslation->m_iNewSide );
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
else if (!stricmp(szKey, "BasisOrigin"))
|
|
{
|
|
Vector Origin;
|
|
int nRead = sscanf(szValue, "%f %f %f", &Origin[0], &Origin[1], &Origin[2]);
|
|
|
|
if (nRead != 3)
|
|
return false;
|
|
|
|
// move the points to where our room is
|
|
Origin += GetCurrentRoomOffset();
|
|
|
|
char buffer[256];
|
|
Q_snprintf(buffer, sizeof(buffer), "%f %f %f", Origin[0], Origin[1], Origin[2]);
|
|
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
|
|
// check for unique string keys (targetname, etc.)
|
|
int nCount = m_UniqueKeys.Count();
|
|
for ( int i = 0; i < nCount; ++i )
|
|
{
|
|
const char *pUnique = m_UniqueKeys[i];
|
|
if ( !stricmp(szKey, pUnique) && szValue && szValue[0] != '@' ) // don't make names unique if they start with special character
|
|
{
|
|
// prepend room id to make this unique
|
|
char buffer[256];
|
|
Q_snprintf( buffer, sizeof(buffer), "Room%d_%s", m_iCurrentRoom, szValue );
|
|
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// remap node IDs
|
|
nCount = m_NodeIDKeys.Count();
|
|
for ( int i = 0; i < nCount; ++i )
|
|
{
|
|
const char *pNodeKey = m_NodeIDKeys[i];
|
|
if (!stricmp(szKey, pNodeKey))
|
|
{
|
|
int iNodeID = atoi(szValue);
|
|
for (int i=0;i<m_NodeTranslations.Count();i++)
|
|
{
|
|
NodeTranslation_t *pTranslation = m_NodeTranslations[i];
|
|
if ( pTranslation->m_iOriginalNodeID == iNodeID )
|
|
{
|
|
|
|
char buffer[32];
|
|
Q_snprintf(buffer, sizeof(buffer), "%d", pTranslation->m_iNewNodeID );
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessConnections( KeyValues *pConnections )
|
|
{
|
|
for ( KeyValues *pKeys = pConnections->GetFirstSubKey(); pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
if ( !ProcessConnectionsKey( pKeys ) )
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool VMFExporter::ProcessConnectionsKey( KeyValues *pKey )
|
|
{
|
|
const char *szValue = pKey->GetString();
|
|
|
|
char buffer[256];
|
|
// prepend room id to targetname
|
|
if ( szValue && szValue[0] != '@' ) // don't make names unique if they start with special character
|
|
{
|
|
Q_snprintf( buffer, sizeof( buffer ), "Room%d_%s", m_iCurrentRoom, szValue );
|
|
}
|
|
else
|
|
{
|
|
Q_snprintf( buffer, sizeof( buffer ), "%s", szValue );
|
|
}
|
|
|
|
pKey->SetStringValue( buffer );
|
|
return true;
|
|
}
|
|
|
|
// sets priority of objective entities based on the order the rooms were listed in the mission/objective txt
|
|
void VMFExporter::ReorderObjectives( const CRoomTemplate *pTemplate, KeyValues *pTemplateKeys )
|
|
{
|
|
KeyValues *pKeys = m_pTemplateKeys;
|
|
while ( pKeys )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "entity" ) && !Q_strnicmp( pKeys->GetString( "classname" ), "asw_objective", 13 ) )
|
|
{
|
|
if ( pKeys->GetFloat( "Priority" ) != 0 ) // if level designer has already set priority, then don't override it
|
|
{
|
|
pKeys = pKeys->GetNextKey();
|
|
continue;
|
|
}
|
|
|
|
int iPriority = 100;
|
|
// We no longer have a requested rooms array, so this code is not valid. Need to replace it with something else to ensure priorities are set correctly.
|
|
// for ( int i = 0; i < m_pMapLayout->m_pRequestedRooms.Count(); i++ )
|
|
// {
|
|
// if ( m_pMapLayout->m_pRequestedRooms[i]->m_pRoomTemplate == pTemplate )
|
|
// {
|
|
// iPriority = m_pMapLayout->m_pRequestedRooms[i]->m_iMissionTextOrder;
|
|
// }
|
|
// }
|
|
pKeys->SetFloat( "Priority", iPriority );
|
|
}
|
|
pKeys = pKeys->GetNextKey();
|
|
}
|
|
}
|
|
|
|
int VMFExporter::MakeNodeIDsUnique()
|
|
{
|
|
int iNodes = 0;
|
|
KeyValues *pKeys = m_pTemplateKeys;
|
|
while ( pKeys )
|
|
{
|
|
if ( !Q_stricmp( pKeys->GetName(), "entity" ) ) // go through all entities in this room
|
|
{
|
|
KeyValues *pFieldKey = pKeys->GetFirstSubKey();
|
|
while ( pFieldKey ) // go through all properties of this entity
|
|
{
|
|
if ( !pFieldKey->GetFirstSubKey() ) // leaf
|
|
{
|
|
if ( !stricmp(pFieldKey->GetName(), "nodeid" ) )
|
|
{
|
|
NodeTranslation_t *pNodeTranslation = new NodeTranslation_t; // store a translation so we can fix up info links
|
|
pNodeTranslation->m_iOriginalNodeID = atoi( pFieldKey->GetString() );
|
|
pNodeTranslation->m_iNewNodeID = m_iNextNodeID;
|
|
m_NodeTranslations.AddToTail( pNodeTranslation );
|
|
char buffer[16];
|
|
Q_snprintf( buffer, sizeof( buffer ), "%d", m_iNextNodeID);
|
|
pFieldKey->SetStringValue( buffer );
|
|
m_iNextNodeID++;
|
|
iNodes++;
|
|
}
|
|
}
|
|
pFieldKey = pFieldKey->GetNextKey();
|
|
}
|
|
}
|
|
pKeys = pKeys->GetNextKey();
|
|
}
|
|
return iNodes;
|
|
}
|
|
|
|
const Vector& VMFExporter::GetCurrentRoomOffset()
|
|
{
|
|
static Vector s_vecCurrentRoomOffset = vec3_origin;
|
|
|
|
if ( m_pRoom )
|
|
{
|
|
int half_map_size = MAP_LAYOUT_TILES_WIDE * 0.5f;
|
|
s_vecCurrentRoomOffset.x = ( m_pRoom->m_iPosX - half_map_size ) * ASW_TILE_SIZE;
|
|
s_vecCurrentRoomOffset.y = ( m_pRoom->m_iPosY - half_map_size ) * ASW_TILE_SIZE;
|
|
s_vecCurrentRoomOffset.z = 0;
|
|
}
|
|
else
|
|
{
|
|
// place logical rooms at the origin and up above the camera
|
|
s_vecCurrentRoomOffset.x = 0;
|
|
s_vecCurrentRoomOffset.y = 0;
|
|
s_vecCurrentRoomOffset.z = 1024;
|
|
}
|
|
return s_vecCurrentRoomOffset;
|
|
}
|
|
|
|
#define UNIQUE_KEYS_FILE "tilegen/unique_keys.txt"
|
|
void VMFExporter::LoadUniqueKeyList()
|
|
{
|
|
m_UniqueKeys.RemoveAll();
|
|
m_NodeIDKeys.RemoveAll();
|
|
|
|
// Open the manifest file, and read the particles specified inside it
|
|
KeyValues *manifest = new KeyValues( UNIQUE_KEYS_FILE );
|
|
if ( manifest->LoadFromFile( g_pFullFileSystem, UNIQUE_KEYS_FILE, "GAME" ) )
|
|
{
|
|
for ( KeyValues *sub = manifest->GetFirstSubKey(); sub != NULL; sub = sub->GetNextKey() )
|
|
{
|
|
if ( !Q_stricmp( sub->GetName(), "key" ) )
|
|
{
|
|
m_UniqueKeys.AddToTail( sub->GetString() );
|
|
continue;
|
|
}
|
|
if ( !Q_stricmp( sub->GetName(), "NodeID" ) )
|
|
{
|
|
m_NodeIDKeys.AddToTail( sub->GetString() );
|
|
continue;
|
|
}
|
|
|
|
Warning( "VMFExporter::LoadUniqueKeyList: Manifest '%s' with bogus file type '%s', expecting 'key' or 'NodeID'\n", UNIQUE_KEYS_FILE, sub->GetName() );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Warning( "VMFExporter: Unable to load manifest file '%s'\n", UNIQUE_KEYS_FILE );
|
|
}
|
|
|
|
manifest->deleteThis();
|
|
}
|
|
|
|
void VMFExporter::AddRoomInstance( const CRoomTemplate *pRoomTemplate, int nPlacedRoomIndex )
|
|
{
|
|
KeyValues *pFuncInstance = new KeyValues( "entity" );
|
|
pFuncInstance->SetInt( "id", ++ m_iEntityCount );
|
|
pFuncInstance->SetString( "classname", "func_instance" );
|
|
pFuncInstance->SetString( "angles", "0 0 0" );
|
|
|
|
// Used to identify rooms for later fixup (e.g. adding/swapping instances and models)
|
|
if ( nPlacedRoomIndex != -1 )
|
|
{
|
|
pFuncInstance->SetInt( "PlacedRoomIndex", nPlacedRoomIndex );
|
|
}
|
|
|
|
char vmfName[MAX_PATH];
|
|
Q_snprintf( vmfName, sizeof( vmfName ), "tilegen/roomtemplates/%s/%s.vmf", pRoomTemplate->m_pLevelTheme->m_szName, pRoomTemplate->GetFullName() );
|
|
// Convert backslashes to forward slashes to please the VMF parser
|
|
int nStrLen = Q_strlen( vmfName );
|
|
for ( int i = 0; i < nStrLen; ++ i )
|
|
{
|
|
if ( vmfName[i] == '\\' ) vmfName[i] = '/';
|
|
}
|
|
|
|
pFuncInstance->SetString( "file", vmfName );
|
|
|
|
Vector vOrigin = GetCurrentRoomOffset();
|
|
char buf[128];
|
|
Q_snprintf( buf, 128, "%f %f %f", vOrigin.x, vOrigin.y, vOrigin.z );
|
|
pFuncInstance->SetString( "origin", buf );
|
|
m_pExportKeys->AddSubKey( pFuncInstance );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Saves the version information chunk.
|
|
// Input : *pFile -
|
|
// Output : Returns ChunkFile_Ok on success, an error code on failure.
|
|
//-----------------------------------------------------------------------------
|
|
KeyValues* VMFExporter::GetVersionInfo()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "versioninfo" );
|
|
pKeys->SetInt( "editorversion", 400 );
|
|
pKeys->SetInt( "editorbuild", 3900 );
|
|
pKeys->SetInt( "mapversion", 1 );
|
|
pKeys->SetInt( "formatversion", 100 );
|
|
pKeys->SetBool( "prefab", false );
|
|
return pKeys;
|
|
}
|
|
|
|
// just save out an empty visgroups section
|
|
KeyValues* VMFExporter::GetDefaultVisGroups()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "visgroups" );
|
|
return pKeys;
|
|
}
|
|
|
|
KeyValues* VMFExporter::GetViewSettings()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "viewsettings" );
|
|
pKeys->SetBool( "bSnapToGrid", true );
|
|
pKeys->SetBool( "bShowGrid", true );
|
|
pKeys->SetBool( "bShowLogicalGrid", false );
|
|
pKeys->SetInt( "nGridSpacing", ASW_TILE_SIZE );
|
|
pKeys->SetBool( "bShow3DGrid", false );
|
|
return pKeys;
|
|
}
|
|
|
|
KeyValues* VMFExporter::GetDefaultCamera()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "cameras" );
|
|
pKeys->SetInt( "activecamera", 0 );
|
|
|
|
KeyValues *pSubKeys = new KeyValues( "camera" );
|
|
pSubKeys->SetString( "position", "[135.491 -60.314 364.02]" );
|
|
pSubKeys->SetString( "look", "[141.195 98.7571 151.25]" );
|
|
pKeys->AddSubKey( pSubKeys );
|
|
|
|
return pKeys;
|
|
}
|
|
|
|
KeyValues* VMFExporter::GetPlayerStarts()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "entity" );
|
|
pKeys->SetInt( "id", m_iEntityCount++ );
|
|
pKeys->SetString( "classname", "info_player_start" );
|
|
pKeys->SetString( "angles", "0 0 0" );
|
|
|
|
// put a single player start in the centre of the player start tile selected
|
|
float player_start_x = 128.0f;
|
|
float player_start_y = 128.0f;
|
|
int half_map_size = MAP_LAYOUT_TILES_WIDE * 0.5f;
|
|
player_start_x += (m_pMapLayout->m_iPlayerStartTileX - half_map_size) * ASW_TILE_SIZE;
|
|
player_start_y += (m_pMapLayout->m_iPlayerStartTileY - half_map_size) * ASW_TILE_SIZE;
|
|
|
|
char buffer[128];
|
|
Q_snprintf(buffer, sizeof(buffer), "%f %f 1.0", player_start_x, player_start_y);
|
|
pKeys->SetString( "origin", buffer );
|
|
|
|
return pKeys;
|
|
}
|
|
|
|
KeyValues* VMFExporter::GetGameRulesProxy()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "entity" );
|
|
pKeys->SetInt( "id", m_iEntityCount++ );
|
|
pKeys->SetString( "classname", "asw_gamerules" );
|
|
pKeys->SetString( "targetname", "@asw_gamerules" );
|
|
|
|
char buffer[128];
|
|
Q_snprintf(buffer, sizeof(buffer), "%f %f %f", m_vecStartRoomOrigin.x, m_vecStartRoomOrigin.y, m_vecStartRoomOrigin.z);
|
|
pKeys->SetString( "origin", buffer );
|
|
|
|
KeyValues *pGenerationOptions = m_pMapLayout->GetGenerationOptions();
|
|
int nDifficultyModifier = 0;
|
|
if ( pGenerationOptions != NULL )
|
|
{
|
|
nDifficultyModifier = pGenerationOptions->GetInt( "Difficulty", 5 ) - 5;
|
|
}
|
|
|
|
pKeys->SetInt( "difficultymodifier", nDifficultyModifier );
|
|
|
|
return pKeys;
|
|
}
|
|
|
|
KeyValues* VMFExporter::GetDefaultWorldChunk()
|
|
{
|
|
KeyValues *pKeys = new KeyValues( "world" );
|
|
pKeys->SetInt( "id", 1 );
|
|
pKeys->SetInt( "mapversion", 1 );
|
|
pKeys->SetString( "classname", "worldspawn" );
|
|
pKeys->SetString( "skyname", "blacksky" );
|
|
pKeys->SetInt( "maxpropscreenwidth", -1 );
|
|
pKeys->SetString( "detailvbsp", "detail.vbsp" );
|
|
pKeys->SetString( "detailmaterial", "detail/detailsprites" );
|
|
pKeys->SetInt( "speedruntime", 180 );
|
|
return pKeys;
|
|
}
|
|
|
|
bool VMFExporter::AddLevelContainer()
|
|
{
|
|
KeyValues *pLevelContainerKeys = new KeyValues( "LevelContainer" );
|
|
if ( !pLevelContainerKeys->LoadFromFile( g_pFullFileSystem, "tilegen/roomtemplates/levelcontainer.vmf.no_func_detail", "GAME" ) )
|
|
return false;
|
|
|
|
m_bWritingLevelContainer = true;
|
|
|
|
// set the extents of the map
|
|
m_pMapLayout->GetExtents(m_iMapExtents_XMin, m_iMapExtents_XMax, m_iMapExtents_YMin, m_iMapExtents_YMax);
|
|
Msg( "Layout extents: Topleft: %f %f - Lower right: %f %f\n", m_iMapExtents_XMin, m_iMapExtents_YMin, m_iMapExtents_XMax, m_iMapExtents_YMax );
|
|
// adjust to be relative to the centre of the map
|
|
int half_map_size = MAP_LAYOUT_TILES_WIDE * 0.5f;
|
|
m_iMapExtents_XMin -= half_map_size;
|
|
m_iMapExtents_XMax -= half_map_size;
|
|
m_iMapExtents_YMin -= half_map_size;
|
|
m_iMapExtents_YMax -= half_map_size;
|
|
// pad them by 2 blocks, so the player doesn't move his camera into the wall when at an edge block
|
|
m_iMapExtents_XMin -= 2;
|
|
m_iMapExtents_XMax += 2;
|
|
m_iMapExtents_YMin -= 2;
|
|
m_iMapExtents_YMax += 2;
|
|
|
|
Msg( " Adjusted to: Topleft: %d %d - Lower right: %d %d\n", m_iMapExtents_XMin, m_iMapExtents_YMin, m_iMapExtents_XMax, m_iMapExtents_YMax );
|
|
|
|
// find world keys
|
|
KeyValues *pWorldKeys = NULL;
|
|
for ( KeyValues *pKeys = pLevelContainerKeys; pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
const char *szName = pKeys->GetName();
|
|
if ( !Q_stricmp( szName, "world" ) )
|
|
{
|
|
pWorldKeys = pKeys;
|
|
break;
|
|
}
|
|
}
|
|
if ( !pWorldKeys || !ProcessWorld( pWorldKeys ) )
|
|
{
|
|
Q_snprintf( m_szLastExporterError, sizeof(m_szLastExporterError), "Failed to copy level container\n" );
|
|
return false;
|
|
}
|
|
|
|
m_bWritingLevelContainer = false;
|
|
|
|
for ( KeyValues *pKeys = pWorldKeys->GetFirstSubKey(); pKeys; pKeys = pKeys->GetNextKey() )
|
|
{
|
|
const char *szName = pKeys->GetName();
|
|
if ( !Q_stricmp( szName, "solid" ) )
|
|
{
|
|
m_pExportWorldKeys->AddSubKey( pKeys->MakeCopy() );
|
|
}
|
|
}
|
|
pLevelContainerKeys->deleteThis();
|
|
|
|
m_pTemplateKeys->deleteThis();
|
|
m_pTemplateKeys = NULL;
|
|
|
|
return true;
|
|
}
|