You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
977 lines
28 KiB
977 lines
28 KiB
//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
//=============================================================================//
|
|
|
|
#include <windows.h>
|
|
#include "interface.h"
|
|
#include "tier0/icommandline.h"
|
|
#include "filesystem_tools.h"
|
|
#include "KeyValues.h"
|
|
#include "UtlBuffer.h"
|
|
#include <io.h>
|
|
#include <fcntl.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <stdio.h>
|
|
#include "ConfigManager.h"
|
|
#include "SourceAppInfo.h"
|
|
#include "steam/steam_api.h"
|
|
|
|
extern CSteamAPIContext *steamapicontext;
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include <tier0/memdbgon.h>
|
|
|
|
#define GAME_CONFIG_FILENAME "GameConfig.txt"
|
|
#define TOKEN_SDK_VERSION "SDKVersion"
|
|
|
|
// Version history:
|
|
// 0 - Initial release
|
|
// 1 - Versioning added, DoD configuration added
|
|
// 2 - Ep1 added
|
|
// 3 - Ep2, TF2, and Portal added
|
|
// 4 - Portal 2 added
|
|
// 5 - CSS 1.5
|
|
|
|
#define SDK_LAUNCHER_VERSION 5
|
|
|
|
// Half-Life 2
|
|
defaultConfigInfo_t HL2Info =
|
|
{
|
|
"Half-Life 2",
|
|
"hl2",
|
|
"halflife2.fgd",
|
|
"half-life 2",
|
|
"info_player_start",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_HL2 )
|
|
};
|
|
|
|
// Counter-Strike: Source
|
|
defaultConfigInfo_t CStrikeInfo =
|
|
{
|
|
"Counter-Strike: Source",
|
|
"cstrike",
|
|
"cstrike.fgd",
|
|
"counter-strike source",
|
|
"info_player_terrorist",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_CSS )
|
|
};
|
|
|
|
// Counter-Strike: Source
|
|
defaultConfigInfo_t CStrike15Info =
|
|
{
|
|
"Counter-Strike: Global Offensive",
|
|
"csgo",
|
|
"csgo.fgd",
|
|
"counter-strike global offensive",
|
|
"info_player_terrorist",
|
|
"csgo.exe",
|
|
GetAppSteamAppId( k_App_CSS15 )
|
|
};
|
|
|
|
//Half-Life 2: Deathmatch
|
|
defaultConfigInfo_t HL2DMInfo =
|
|
{
|
|
"Half-Life 2: Deathmatch",
|
|
"hl2mp",
|
|
"hl2mp.fgd",
|
|
"half-life 2 deathmatch",
|
|
"info_player_deathmatch",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_HL2MP )
|
|
};
|
|
|
|
// Day of Defeat: Source
|
|
defaultConfigInfo_t DODInfo =
|
|
{
|
|
"Day of Defeat: Source",
|
|
"dod",
|
|
"dod.fgd",
|
|
"day of defeat source",
|
|
"info_player_allies",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_DODS )
|
|
};
|
|
|
|
// Half-Life 2 Episode 1
|
|
defaultConfigInfo_t Episode1Info =
|
|
{
|
|
"Half-Life 2: Episode One",
|
|
"episodic",
|
|
"halflife2.fgd",
|
|
"half-life 2 episode one",
|
|
"info_player_start",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_HL2_EP1 )
|
|
};
|
|
|
|
// Half-Life 2 Episode 2
|
|
defaultConfigInfo_t Episode2Info =
|
|
{
|
|
"Half-Life 2: Episode Two",
|
|
"ep2",
|
|
"halflife2.fgd",
|
|
"half-life 2 episode two",
|
|
"info_player_start",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_HL2_EP2 )
|
|
};
|
|
|
|
// Team Fortress 2
|
|
defaultConfigInfo_t TF2Info =
|
|
{
|
|
"Team Fortress 2",
|
|
"tf",
|
|
"tf.fgd",
|
|
"team fortress 2",
|
|
"info_player_teamspawn",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_TF2 )
|
|
};
|
|
|
|
// Portal
|
|
defaultConfigInfo_t PortalInfo =
|
|
{
|
|
"Portal",
|
|
"portal",
|
|
"portal.fgd",
|
|
"portal",
|
|
"info_player_start",
|
|
"hl2.exe",
|
|
GetAppSteamAppId( k_App_PORTAL )
|
|
};
|
|
|
|
// Portal 2
|
|
defaultConfigInfo_t Portal2Info =
|
|
{
|
|
"Portal 2",
|
|
"portal2",
|
|
"portal2.fgd",
|
|
"portal 2",
|
|
"info_player_start",
|
|
"portal2.exe",
|
|
GetAppSteamAppId( k_App_PORTAL2 )
|
|
};
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Constructor
|
|
//-----------------------------------------------------------------------------
|
|
CGameConfigManager::CGameConfigManager( void ) : m_pData( NULL ), m_LoadStatus( LOADSTATUS_NONE )
|
|
{
|
|
// Start with default directory
|
|
GetModuleFileName( ( HINSTANCE )GetModuleHandle( NULL ), m_szBaseDirectory, sizeof( m_szBaseDirectory ) );
|
|
Q_StripLastDir( m_szBaseDirectory, sizeof( m_szBaseDirectory ) ); // Get rid of the filename.
|
|
Q_StripTrailingSlash( m_szBaseDirectory );
|
|
m_eSDKEpoch = (eSDKEpochs) SDK_LAUNCHER_VERSION;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Destructor
|
|
//-----------------------------------------------------------------------------
|
|
CGameConfigManager::~CGameConfigManager( void )
|
|
{
|
|
// Release the keyvalues
|
|
if ( m_pData != NULL )
|
|
{
|
|
m_pData->deleteThis();
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Config loading interface
|
|
// Input : *baseDir - base directory for our file
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::LoadConfigs( const char *baseDir )
|
|
{
|
|
return LoadConfigsInternal( baseDir, false );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Loads a file into the given utlbuffer.
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool ReadUtlBufferFromFile( CUtlBuffer &buffer, const char *szPath )
|
|
{
|
|
struct _stat fileInfo;
|
|
if ( _stat( szPath, &fileInfo ) == -1 )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
buffer.EnsureCapacity( fileInfo.st_size );
|
|
|
|
int nFile = _open( szPath, _O_BINARY | _O_RDONLY );
|
|
if ( nFile == -1 )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if ( _read( nFile, buffer.Base(), fileInfo.st_size ) != fileInfo.st_size )
|
|
{
|
|
_close( nFile );
|
|
return false;
|
|
}
|
|
|
|
_close( nFile );
|
|
buffer.SeekPut( CUtlBuffer::SEEK_HEAD, fileInfo.st_size );
|
|
return true;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Loads a file into the given utlbuffer.
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool SaveUtlBufferToFile( CUtlBuffer &buffer, const char *szPath )
|
|
{
|
|
int nFile = _open( szPath, _O_TEXT | _O_CREAT | _O_TRUNC | _O_RDWR, _S_IWRITE );
|
|
if ( nFile == -1 )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
int nSize = buffer.TellMaxPut();
|
|
|
|
if ( _write( nFile, buffer.Base(), nSize ) < nSize )
|
|
{
|
|
_close( nFile );
|
|
return false;
|
|
}
|
|
|
|
_close( nFile );
|
|
return true;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Load a game configuration file (with fail-safes)
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::LoadConfigsInternal( const char *baseDir, bool bRecursiveCall )
|
|
{
|
|
// Init the config if it doesn't exist
|
|
if ( !IsLoaded() )
|
|
{
|
|
m_pData = new KeyValues( GAME_CONFIG_FILENAME );
|
|
|
|
if ( !IsLoaded() )
|
|
{
|
|
m_LoadStatus = LOADSTATUS_ERROR;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Clear it out
|
|
m_pData->Clear();
|
|
|
|
// Build our default directory
|
|
if ( baseDir != NULL && baseDir[0] != NULL )
|
|
{
|
|
SetBaseDirectory( baseDir );
|
|
}
|
|
|
|
// Make a full path name
|
|
char szPath[MAX_PATH];
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", GetBaseDirectory(), GAME_CONFIG_FILENAME );
|
|
|
|
bool bLoaded = false;
|
|
|
|
CUtlBuffer buffer( 0, 0, CUtlBuffer::TEXT_BUFFER );
|
|
if ( ReadUtlBufferFromFile( buffer, szPath ) )
|
|
{
|
|
bLoaded = m_pData->LoadFromBuffer( szPath, buffer, NULL, NULL );
|
|
}
|
|
|
|
if ( !bLoaded )
|
|
{
|
|
// Attempt to re-create the configs
|
|
if ( CreateAllDefaultConfigs() )
|
|
{
|
|
// Only allow this once
|
|
if ( !bRecursiveCall )
|
|
return LoadConfigsInternal( baseDir, true );
|
|
|
|
// Version the config.
|
|
VersionConfig();
|
|
}
|
|
|
|
m_LoadStatus = LOADSTATUS_ERROR;
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
// Check to see if the gameconfig.txt is up to date.
|
|
UpdateConfigsInternal();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Add to the current config.
|
|
//-----------------------------------------------------------------------------
|
|
void CGameConfigManager::UpdateConfigsInternal( void )
|
|
{
|
|
// Check to a valid gameconfig.txt file buffer.
|
|
if ( !IsLoaded() )
|
|
return;
|
|
|
|
// Check for version first. If the version is up to date, it is assumed to be accurate
|
|
if ( IsConfigCurrent() )
|
|
return;
|
|
|
|
KeyValues *pGameBlock = GetGameBlock();
|
|
if ( !pGameBlock )
|
|
{
|
|
// If we don't have a game block, reset the config file.
|
|
ResetConfigs();
|
|
return;
|
|
}
|
|
|
|
KeyValues *pDefaultBlock = new KeyValues( "DefaultConfigs" );
|
|
if ( pDefaultBlock != NULL )
|
|
{
|
|
// Compile our default configurations
|
|
GetDefaultGameBlock( pDefaultBlock );
|
|
|
|
// Compare our default block to our current configs
|
|
KeyValues *pNextSubKey = pDefaultBlock->GetFirstTrueSubKey();
|
|
while ( pNextSubKey != NULL )
|
|
{
|
|
// If we already have the name, we don't care about it
|
|
if ( pGameBlock->FindKey( pNextSubKey->GetName() ) )
|
|
{
|
|
// Advance by one key
|
|
pNextSubKey = pNextSubKey->GetNextTrueSubKey();
|
|
continue;
|
|
}
|
|
|
|
// Copy the data through to our game block
|
|
KeyValues *pKeyCopy = pNextSubKey->MakeCopy();
|
|
pGameBlock->AddSubKey( pKeyCopy );
|
|
|
|
// Advance by one key
|
|
pNextSubKey = pNextSubKey->GetNextTrueSubKey();
|
|
}
|
|
|
|
// All done
|
|
pDefaultBlock->deleteThis();
|
|
}
|
|
|
|
// Save the new config.
|
|
SaveConfigs();
|
|
|
|
// Add the new version as we have been updated.
|
|
VersionConfig();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Update the gameconfig.txt version number.
|
|
//-----------------------------------------------------------------------------
|
|
void CGameConfigManager::VersionConfig( void )
|
|
{
|
|
// Check to a valid gameconfig.txt file buffer.
|
|
if ( !IsLoaded() )
|
|
return;
|
|
|
|
// Look for the a version key value pair and update it.
|
|
KeyValues *pKeyVersion = m_pData->FindKey( TOKEN_SDK_VERSION );
|
|
|
|
// Update the already existing version key value pair.
|
|
if ( pKeyVersion )
|
|
{
|
|
if ( pKeyVersion->GetInt() == m_eSDKEpoch )
|
|
return;
|
|
|
|
m_pData->SetInt( TOKEN_SDK_VERSION, m_eSDKEpoch );
|
|
}
|
|
// Create a new version key value pair.
|
|
else
|
|
{
|
|
m_pData->SetInt( TOKEN_SDK_VERSION, m_eSDKEpoch );
|
|
}
|
|
|
|
// Save the configuration.
|
|
SaveConfigs();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Check to see if the version of the gameconfig.txt is up to date.
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::IsConfigCurrent( void )
|
|
{
|
|
// Check to a valid gameconfig.txt file buffer.
|
|
if ( !IsLoaded() )
|
|
return false;
|
|
|
|
KeyValues *pKeyValue = m_pData->FindKey( TOKEN_SDK_VERSION );
|
|
if ( !pKeyValue )
|
|
return false;
|
|
|
|
int nVersion = pKeyValue->GetInt();
|
|
if ( nVersion == m_eSDKEpoch )
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Get the base path for a default config's install (handling steam's paths)
|
|
//-----------------------------------------------------------------------------
|
|
void CGameConfigManager::GetRootGameDirectory( char *out, size_t outLen, const char *rootDir, const char *steamDir )
|
|
{
|
|
// NOTE: This has since been depricated due to Steam filesystem changes -- jdw
|
|
Q_strncpy( out, rootDir, outLen );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Get the base path for a default config's content sources (handling steam's paths)
|
|
//-----------------------------------------------------------------------------
|
|
void CGameConfigManager::GetRootContentDirectory( char *out, size_t outLen, const char *rootDir )
|
|
{
|
|
// Steam install is different
|
|
if ( IsSDKDeployment() )
|
|
{
|
|
Q_snprintf( out, outLen, "%s\\sdk_content", rootDir );
|
|
}
|
|
else
|
|
{
|
|
Q_snprintf( out, outLen, "%s\\content", rootDir );
|
|
}
|
|
}
|
|
|
|
// Default game configuration template
|
|
const char szDefaultConfigText[] =
|
|
"\"%gamename%\"\
|
|
{\
|
|
\"GameDir\" \"%gamedir%\"\
|
|
\"Hammer\"\
|
|
{\
|
|
\"TextureFormat\" \"5\"\
|
|
\"MapFormat\" \"4\"\
|
|
\"DefaultTextureScale\" \"0.250000\"\
|
|
\"DefaultLightmapScale\" \"16\"\
|
|
\"DefaultSolidEntity\" \"func_detail\"\
|
|
\"DefaultPointEntity\" \"%defaultpointentity%\"\
|
|
\"GameExeDir\" \"%gameexe%\"\
|
|
\"MapDir\" \"%gamemaps%\"\
|
|
\"CordonTexture\" \"tools\\toolsskybox\"\
|
|
\"MaterialExcludeCount\" \"0\"\
|
|
\"GameExe\" \"%gameEXE%\"\
|
|
\"BSP\" \"%bspdir%\"\
|
|
\"Vis\" \"%visdir%\"\
|
|
\"Light\" \"%lightdir%\"\
|
|
}}";
|
|
|
|
// NOTE: This function could use some re-write, it can't handle non-retail paths well
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Add a templated default configuration with proper paths
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::AddDefaultConfig( const defaultConfigInfo_t &info, KeyValues *out, const char *rootDirectory, const char *gameExeDir )
|
|
{
|
|
// NOTE: Freed by head keyvalue
|
|
KeyValues *newConfig = new KeyValues( info.gameName );
|
|
|
|
// Set this up to autodelete until we know we're actually going to use it
|
|
KeyValues::AutoDelete autodelete_key( newConfig );
|
|
|
|
if ( newConfig->LoadFromBuffer( "defaultcfg.txt", szDefaultConfigText ) == false )
|
|
return false;
|
|
|
|
newConfig->SetName( info.gameName );
|
|
|
|
// Game's root directory (with special steam name handling)
|
|
char rootGameDir[MAX_PATH];
|
|
GetRootGameDirectory( rootGameDir, sizeof( rootGameDir ), rootDirectory, info.steamPath );
|
|
|
|
// Game's content directory
|
|
char contentRootDir[MAX_PATH];
|
|
GetRootContentDirectory( contentRootDir, sizeof( contentRootDir ), rootDirectory );
|
|
|
|
char szPath[MAX_PATH];
|
|
|
|
// Game directory
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", rootGameDir, info.gameDir );
|
|
newConfig->SetString( "GameDir", szPath );
|
|
|
|
// Create the Hammer portion of this block
|
|
KeyValues *hammerBlock = newConfig->FindKey( "Hammer" );
|
|
|
|
if ( hammerBlock == NULL )
|
|
return false;
|
|
|
|
hammerBlock->SetString( "GameExeDir", gameExeDir );
|
|
|
|
// Fill in the proper default point entity
|
|
hammerBlock->SetString( "DefaultPointEntity", info.defaultPointEntity );
|
|
|
|
// Fill in the default VMF directory
|
|
char contentMapDir[MAX_PATH];
|
|
Q_snprintf( contentMapDir, sizeof( contentMapDir ), "%s\\maps", contentRootDir );
|
|
hammerBlock->SetString( "MapDir", contentMapDir );
|
|
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s\\maps", rootGameDir, info.gameDir );
|
|
hammerBlock->SetString( "BSPDir", szPath );
|
|
|
|
// Fill in the game executable
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", gameExeDir, info.exeName );
|
|
hammerBlock->SetString( "GameEXE", szPath );
|
|
|
|
//Fill in game FGDs
|
|
if ( info.FGD[0] != '\0' )
|
|
{
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", GetBaseDirectory(), info.FGD );
|
|
hammerBlock->SetString( "GameData0", szPath );
|
|
}
|
|
|
|
// Fill in the tools path
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\vbsp.exe", GetBaseDirectory() );
|
|
hammerBlock->SetString( "BSP", szPath );
|
|
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\vvis.exe", GetBaseDirectory() );
|
|
hammerBlock->SetString( "Vis", szPath );
|
|
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\vrad.exe", GetBaseDirectory() );
|
|
hammerBlock->SetString( "Light", szPath );
|
|
|
|
// Get our insertion point
|
|
KeyValues *insertSpot = out->GetFirstTrueSubKey();
|
|
|
|
// detach the autodelete pointer
|
|
autodelete_key.Assign(NULL);
|
|
// Set this as the sub key if there's nothing already there
|
|
if ( insertSpot == NULL )
|
|
{
|
|
out->AddSubKey( newConfig );
|
|
}
|
|
else
|
|
{
|
|
// Find the last subkey
|
|
while ( insertSpot->GetNextTrueSubKey() )
|
|
{
|
|
insertSpot = insertSpot->GetNextTrueSubKey();
|
|
}
|
|
|
|
// Become a peer to it
|
|
insertSpot->SetNextKey( newConfig );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Determines whether the requested appID is installed on this computer
|
|
// Input : nAppID - ID to verify
|
|
// Output : Returns true if installed, false if not.
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::IsAppSubscribed( int nAppID )
|
|
{
|
|
bool bIsSubscribed = false;
|
|
|
|
if ( g_pFullFileSystem != NULL && g_pFullFileSystem->IsSteam() )
|
|
{
|
|
if ( steamapicontext->SteamApps() )
|
|
{
|
|
// See if specified app is installed
|
|
bIsSubscribed = steamapicontext->SteamApps()->BIsSubscribedApp( nAppID );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If we aren't running FileSystem Steam then we must be doing internal development. Give everything.
|
|
bIsSubscribed = true;
|
|
}
|
|
|
|
return bIsSubscribed;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Create default configurations for all Valve retail applications
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::CreateAllDefaultConfigs( void )
|
|
{
|
|
bool bRetVal = true;
|
|
|
|
// Start our new block
|
|
KeyValues *configBlock = new KeyValues( "Configs" );
|
|
KeyValues *gameBlock = configBlock->CreateNewKey();
|
|
gameBlock->SetName( "Games" );
|
|
|
|
GetDefaultGameBlock( gameBlock );
|
|
|
|
bRetVal = !gameBlock->IsEmpty();
|
|
|
|
// Make a full path name
|
|
char szPath[MAX_PATH];
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", GetBaseDirectory(), GAME_CONFIG_FILENAME );
|
|
|
|
CUtlBuffer buffer;
|
|
configBlock->RecursiveSaveToFile( buffer, 0 );
|
|
SaveUtlBufferToFile( buffer, szPath );
|
|
|
|
configBlock->deleteThis();
|
|
|
|
m_LoadStatus = LOADSTATUS_CREATED;
|
|
|
|
return bRetVal;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Load game information from an INI file
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::ConvertGameConfigsINI( void )
|
|
{
|
|
const char *iniFilePath = GetIniFilePath();
|
|
|
|
// Load our INI file
|
|
int nNumConfigs = GetPrivateProfileInt( "Configs", "NumConfigs", 0, iniFilePath );
|
|
if ( nNumConfigs <= 0 )
|
|
return false;
|
|
|
|
// Build a new keyvalue file
|
|
KeyValues *headBlock = new KeyValues( "Configs" );
|
|
|
|
// Create the block for games
|
|
KeyValues *gamesBlock = headBlock->CreateNewKey( );
|
|
gamesBlock->SetName( "Games" );
|
|
|
|
int i;
|
|
int nStrlen;
|
|
char szSectionName[MAX_PATH];
|
|
char textBuffer[MAX_PATH];
|
|
|
|
// Parse all the configs
|
|
for ( int nConfig = 0; nConfig < nNumConfigs; nConfig++ )
|
|
{
|
|
// Each came configuration is stored in a different section, named "GameConfig0..GameConfigN".
|
|
// If the "Name" key exists in this section, try to load the configuration from this section.
|
|
sprintf(szSectionName, "GameConfig%d", nConfig);
|
|
|
|
int nCount = GetPrivateProfileString(szSectionName, "Name", "", textBuffer, sizeof(textBuffer), iniFilePath);
|
|
if (nCount > 0)
|
|
{
|
|
// Make a new section
|
|
KeyValues *subGame = gamesBlock->CreateNewKey();
|
|
subGame->SetName( textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "ModDir", "", textBuffer, sizeof(textBuffer), iniFilePath);
|
|
|
|
// Add the mod dir
|
|
subGame->SetString( "GameDir", textBuffer );
|
|
|
|
// Start a block for Hammer settings
|
|
KeyValues *hammerBlock = subGame->CreateNewKey();
|
|
hammerBlock->SetName( "Hammer" );
|
|
|
|
i = 0;
|
|
|
|
// Get all FGDs
|
|
do
|
|
{
|
|
char szGameData[MAX_PATH];
|
|
|
|
sprintf( szGameData, "GameData%d", i );
|
|
nStrlen = GetPrivateProfileString( szSectionName, szGameData, "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
|
|
if ( nStrlen > 0 )
|
|
{
|
|
hammerBlock->SetString( szGameData, textBuffer );
|
|
i++;
|
|
}
|
|
} while ( nStrlen > 0 );
|
|
|
|
hammerBlock->SetInt( "TextureFormat", GetPrivateProfileInt( szSectionName, "TextureFormat", 5 /*FIXME: tfVMT*/, iniFilePath ) );
|
|
hammerBlock->SetInt( "MapFormat", GetPrivateProfileInt( szSectionName, "MapFormat", 4 /*FIXME: mfHalfLife2*/, iniFilePath ) );
|
|
|
|
// Default texture scale
|
|
GetPrivateProfileString( szSectionName, "DefaultTextureScale", "1", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
float defaultTextureScale = (float) atof( textBuffer );
|
|
if ( defaultTextureScale == 0 )
|
|
{
|
|
defaultTextureScale = 1.0f;
|
|
}
|
|
|
|
hammerBlock->SetFloat( "DefaultTextureScale", defaultTextureScale );
|
|
|
|
hammerBlock->SetInt( "DefaultLightmapScale", GetPrivateProfileInt( szSectionName, "DefaultLightmapScale", 16 /*FIXME: DEFAULT_LIGHTMAP_SCALE*/, iniFilePath ) );
|
|
|
|
GetPrivateProfileString( szSectionName, "GameExe", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "GameExe", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "DefaultSolidEntity", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "DefaultSolidEntity", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "DefaultPointEntity", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "DefaultPointEntity", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "BSP", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "BSP", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "Vis", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "Vis", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "Light", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "Light", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "GameExeDir", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "GameExeDir", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "MapDir", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "MapDir", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "BSPDir", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "BSPDir", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "CordonTexture", "", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
hammerBlock->SetString( "CordonTexture", textBuffer );
|
|
|
|
GetPrivateProfileString( szSectionName, "MaterialExcludeCount", "0", textBuffer, sizeof(textBuffer), iniFilePath );
|
|
int materialExcludeCount = atoi( textBuffer );
|
|
hammerBlock->SetInt( "MaterialExcludeCount", materialExcludeCount );
|
|
|
|
char excludeDir[MAX_PATH];
|
|
|
|
// Write out all excluded directories
|
|
for( i = 0; i < materialExcludeCount; i++ )
|
|
{
|
|
sprintf( &excludeDir[0], "-MaterialExcludeDir%d", i );
|
|
GetPrivateProfileString( szSectionName, excludeDir, "", textBuffer, sizeof( textBuffer ), iniFilePath );
|
|
hammerBlock->SetString( excludeDir, textBuffer );
|
|
}
|
|
}
|
|
}
|
|
// Make a full path name
|
|
char szPath[MAX_PATH];
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", GetBaseDirectory(), GAME_CONFIG_FILENAME );
|
|
|
|
CUtlBuffer buffer;
|
|
headBlock->RecursiveSaveToFile( buffer, 0 );
|
|
SaveUtlBufferToFile( buffer, szPath );
|
|
|
|
// Rename the old INI file
|
|
char newFilePath[MAX_PATH];
|
|
Q_snprintf( newFilePath, sizeof( newFilePath ), "%s.OLD", iniFilePath );
|
|
|
|
rename( iniFilePath, newFilePath );
|
|
|
|
// Notify that we were converted
|
|
m_LoadStatus = LOADSTATUS_CONVERTED;
|
|
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Write out a game configuration file
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::SaveConfigs( const char *baseDir )
|
|
{
|
|
if ( !IsLoaded() )
|
|
return false;
|
|
|
|
// Build our default directory
|
|
if ( baseDir != NULL && baseDir[0] != NULL )
|
|
{
|
|
SetBaseDirectory( baseDir );
|
|
}
|
|
|
|
// Make a full path name
|
|
char szPath[MAX_PATH];
|
|
Q_strncpy( szPath, GetBaseDirectory(), sizeof(szPath) );
|
|
Q_AppendSlash( szPath, sizeof(szPath) );
|
|
Q_strncat( szPath, GAME_CONFIG_FILENAME, sizeof( szPath ), COPY_ALL_CHARACTERS );
|
|
|
|
CUtlBuffer buffer;
|
|
m_pData->RecursiveSaveToFile( buffer, 0 );
|
|
|
|
return SaveUtlBufferToFile( buffer, szPath );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Find the directory our .exe is based out of
|
|
//-----------------------------------------------------------------------------
|
|
const char *CGameConfigManager::GetBaseDirectory( void )
|
|
{
|
|
return m_szBaseDirectory;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Find the root directory
|
|
//-----------------------------------------------------------------------------
|
|
const char *CGameConfigManager::GetRootDirectory( void )
|
|
{
|
|
static char path[MAX_PATH] = {0};
|
|
if ( path[0] == 0 )
|
|
{
|
|
Q_strncpy( path, GetBaseDirectory(), sizeof( path ) );
|
|
Q_StripLastDir( path, sizeof( path ) ); // Get rid of the 'bin' directory
|
|
Q_StripTrailingSlash( path );
|
|
|
|
if ( g_pFullFileSystem && g_pFullFileSystem->IsSteam() )
|
|
{
|
|
Q_StripLastDir( path, sizeof( path ) ); // // Get rid of the 'orangebox' directory
|
|
Q_StripTrailingSlash( path );
|
|
Q_StripLastDir( path, sizeof( path ) ); // Get rid of the 'bin' directory
|
|
Q_StripTrailingSlash( path );
|
|
Q_StripLastDir( path, sizeof( path ) ); // Get rid of the 'sourcesdk' directory
|
|
Q_StripTrailingSlash( path );
|
|
}
|
|
}
|
|
return path;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Returns the game configuation block
|
|
//-----------------------------------------------------------------------------
|
|
KeyValues *CGameConfigManager::GetGameBlock( void )
|
|
{
|
|
if ( !IsLoaded() )
|
|
return NULL;
|
|
|
|
return ( m_pData->FindKey( TOKEN_GAMES ) );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Returns a piece of the game configuation block of the given name
|
|
// Input : *keyName - name of the block to return
|
|
//-----------------------------------------------------------------------------
|
|
KeyValues *CGameConfigManager::GetGameSubBlock( const char *keyName )
|
|
{
|
|
if ( !IsLoaded() )
|
|
return NULL;
|
|
|
|
KeyValues *pGameBlock = GetGameBlock();
|
|
if ( pGameBlock == NULL )
|
|
return NULL;
|
|
|
|
// Return the data
|
|
KeyValues *pSubBlock = pGameBlock->FindKey( keyName );
|
|
|
|
return pSubBlock;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Get the gamecfg.ini file for conversion
|
|
//-----------------------------------------------------------------------------
|
|
const char *CGameConfigManager::GetIniFilePath( void )
|
|
{
|
|
static char iniFilePath[MAX_PATH] = {0};
|
|
if ( iniFilePath[0] == 0 )
|
|
{
|
|
Q_strncpy( iniFilePath, GetBaseDirectory(), sizeof( iniFilePath ) );
|
|
Q_strncat( iniFilePath, "\\gamecfg.ini", sizeof( iniFilePath ), COPY_ALL_CHARACTERS );
|
|
}
|
|
|
|
return iniFilePath;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Deletes the current config and recreates it with default values
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::ResetConfigs( const char *baseDir /*= NULL*/ )
|
|
{
|
|
// Build our default directory
|
|
if ( baseDir != NULL && baseDir[0] != NULL )
|
|
{
|
|
SetBaseDirectory( baseDir );
|
|
}
|
|
|
|
// Make a full path name
|
|
char szPath[MAX_PATH];
|
|
Q_snprintf( szPath, sizeof( szPath ), "%s\\%s", GetBaseDirectory(), GAME_CONFIG_FILENAME );
|
|
|
|
// Delete the file
|
|
if ( unlink( szPath ) )
|
|
return false;
|
|
|
|
// Load the file again (causes defaults to be created)
|
|
if ( LoadConfigsInternal( baseDir, false ) == false )
|
|
return false;
|
|
|
|
// Save it out
|
|
return SaveConfigs( baseDir );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CGameConfigManager::SetBaseDirectory( const char *pDirectory )
|
|
{
|
|
// Clear it
|
|
if ( pDirectory == NULL || pDirectory[0] == '\0' )
|
|
{
|
|
m_szBaseDirectory[0] = '\0';
|
|
return;
|
|
}
|
|
|
|
// Copy it
|
|
Q_strncpy( m_szBaseDirectory, pDirectory, sizeof( m_szBaseDirectory ) );
|
|
Q_StripTrailingSlash( m_szBaseDirectory );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Create a block of keyvalues containing our default configurations
|
|
// Output : A block of keyvalues
|
|
//-----------------------------------------------------------------------------
|
|
bool CGameConfigManager::GetDefaultGameBlock( KeyValues *pIn )
|
|
{
|
|
CUtlVector<defaultConfigInfo_t> defaultConfigs;
|
|
|
|
// Add HL2 games to list
|
|
if ( m_eSDKEpoch == SDK_EPOCH_HL2 || m_eSDKEpoch == SDK_EPOCH_EP1 )
|
|
{
|
|
defaultConfigs.AddToTail( HL2Info );
|
|
defaultConfigs.AddToTail( CStrikeInfo );
|
|
defaultConfigs.AddToTail( HL2DMInfo );
|
|
}
|
|
// Add EP1 game to list
|
|
if ( m_eSDKEpoch == SDK_EPOCH_EP1 )
|
|
{
|
|
defaultConfigs.AddToTail( Episode1Info );
|
|
}
|
|
|
|
// Add EP2 games to list
|
|
if ( m_eSDKEpoch == SDK_EPOCH_EP2 )
|
|
{
|
|
defaultConfigs.AddToTail( Episode2Info );
|
|
}
|
|
|
|
if ( m_eSDKEpoch == SDK_EPOCH_PORTAL2 )
|
|
{
|
|
defaultConfigs.AddToTail( Portal2Info );
|
|
}
|
|
|
|
// CSS 1.5
|
|
if ( m_eSDKEpoch == SDK_EPOCH_CSS15 )
|
|
{
|
|
defaultConfigs.AddToTail( CStrike15Info );
|
|
}
|
|
|
|
if ( pIn == NULL )
|
|
return false;
|
|
|
|
char szPath[MAX_PATH];
|
|
|
|
// Add all default configs
|
|
int nNumConfigs = defaultConfigs.Count();
|
|
for ( int i = 0; i < nNumConfigs; i++ )
|
|
{
|
|
// If it's installed, add it
|
|
if ( IsAppSubscribed( defaultConfigs[i].steamAppID ) )
|
|
{
|
|
GetRootGameDirectory( szPath, sizeof( szPath ), GetRootDirectory(), defaultConfigs[i].steamPath );
|
|
AddDefaultConfig( defaultConfigs[i], pIn, GetRootDirectory(), szPath );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|