/*
===========================================================================
Doom 3 BFG Edition GPL Source Code
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Doom 3 BFG Edition Source Code. If not, see .
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
===========================================================================
*/
#pragma hdrstop
#include "../../idlib/precompiled.h"
#include "../Game_local.h"
/*
Save game related helper classes.
Save games are implemented in two classes, idSaveGame and idRestoreGame, that implement write/read functions for
common types. They're passed in to each entity and object for them to archive themselves. Each class
implements save/restore functions for it's own data. When restoring, all the objects are instantiated,
then the restore function is called on each, superclass first, then subclasses.
Pointers are restored by saving out an object index for each unique object pointer and adding them to a list of
objects that are to be saved. Restore instantiates all the objects in the list before calling the Restore function
on each object so that the pointers returned are valid. No object's restore function should rely on any other objects
being fully instantiated until after the restore process is complete. Post restore fixup should be done by posting
events with 0 delay.
The savegame header will have the Game Name, Version, Map Name, and Player Persistent Info.
Changes in version make savegames incompatible, and the game will start from the beginning of the level with
the player's persistent info.
Changes to classes that don't need to break compatibilty can use the build number as the savegame version.
Later versions are responsible for restoring from previous versions by ignoring any unused data and initializing
variables that weren't in previous versions with safe information.
At the head of the save game is enough information to restore the player to the beginning of the level should the
file be unloadable in some way (for example, due to script changes).
*/
/*
================
idSaveGame::idSaveGame()
================
*/
idSaveGame::idSaveGame( idFile* savefile, idFile* stringTableFile, int saveVersion )
{
//compressor = idCompressor::AllocLZW();
//compressor->Init( savefile, true, 8 );
//file = compressor;
file = savefile;
stringFile = stringTableFile;
version = saveVersion;
// Put NULL at the start of the list so we can skip over it.
objects.Clear();
objects.Append( NULL );
curStringTableOffset = 0;
}
/*
================
idSaveGame::~idSaveGame()
================
*/
idSaveGame::~idSaveGame()
{
//compressor->FinishCompress();
//delete compressor;
if( objects.Num() )
{
Close();
}
}
/*
================
idSaveGame::Close
================
*/
void idSaveGame::Close()
{
WriteSoundCommands();
// read trace models
idClipModel::SaveTraceModels( this );
for( int i = 1; i < objects.Num(); i++ )
{
CallSave_r( objects[ i ]->GetType(), objects[ i ] );
}
objects.Clear();
// Save out the string table at the end of the file
for( int i = 0; i < stringTable.Num(); ++i )
{
stringFile->WriteString( stringTable[i].string );
}
stringHash.Free();
stringTable.Clear();
if( file->Length() > MIN_SAVEGAME_SIZE_BYTES || stringFile->Length() > MAX_SAVEGAME_STRING_TABLE_SIZE )
{
idLib::FatalError( "OVERFLOWED SAVE GAME FILE BUFFER" );
}
#ifdef ID_DEBUG_MEMORY
idStr gameState = file->GetName();
gameState.StripFileExtension();
WriteGameState_f( idCmdArgs( va( "test %s_save", gameState.c_str() ), false ) );
#endif
}
/*
================
idSaveGame::WriteDecls
================
*/
void idSaveGame::WriteDecls()
{
// Write out all loaded decls
for( int t = 0; t < declManager->GetNumDeclTypes(); t++ )
{
for( int d = 0; d < declManager->GetNumDecls( ( declType_t )t ); d++ )
{
const idDecl* decl = declManager->DeclByIndex( ( declType_t )t, d, false );
if( decl == NULL || decl->GetState() == DS_UNPARSED )
{
continue;
}
const char* declName = decl->GetName();
if( declName[0] == 0 )
{
continue;
}
WriteString( declName );
}
WriteString( 0 );
}
}
/*
================
idSaveGame::WriteObjectList
================
*/
void idSaveGame::WriteObjectList()
{
WriteInt( objects.Num() - 1 );
for( int i = 1; i < objects.Num(); i++ )
{
WriteString( objects[ i ]->GetClassname() );
}
}
/*
================
idSaveGame::CallSave_r
================
*/
void idSaveGame::CallSave_r( const idTypeInfo* cls, const idClass* obj )
{
if( cls->super )
{
CallSave_r( cls->super, obj );
if( cls->super->Save == cls->Save )
{
// don't call save on this inheritance level since the function was called in the super class
return;
}
}
( obj->*cls->Save )( this );
}
/*
================
idSaveGame::AddObject
================
*/
void idSaveGame::AddObject( const idClass* obj )
{
objects.AddUnique( obj );
}
/*
================
idSaveGame::Write
================
*/
void idSaveGame::Write( const void* buffer, int len )
{
file->Write( buffer, len );
}
/*
================
idSaveGame::WriteInt
================
*/
void idSaveGame::WriteInt( const int value )
{
file->WriteBig( value );
}
/*
================
idSaveGame::WriteJoint
================
*/
void idSaveGame::WriteJoint( const jointHandle_t value )
{
file->WriteBig( ( int& )value );
}
/*
================
idSaveGame::WriteShort
================
*/
void idSaveGame::WriteShort( const short value )
{
file->WriteBig( value );
}
/*
================
idSaveGame::WriteByte
================
*/
void idSaveGame::WriteByte( const byte value )
{
file->Write( &value, sizeof( value ) );
}
/*
================
idSaveGame::WriteSignedChar
================
*/
void idSaveGame::WriteSignedChar( const signed char value )
{
file->Write( &value, sizeof( value ) );
}
/*
================
idSaveGame::WriteFloat
================
*/
void idSaveGame::WriteFloat( const float value )
{
file->WriteBig( value );
}
/*
================
idSaveGame::WriteBool
================
*/
void idSaveGame::WriteBool( const bool value )
{
file->WriteBool( value );
}
/*
================
idSaveGame::WriteString
================
*/
void idSaveGame::WriteString( const char* string )
{
if( string == NULL || *string == 0 )
{
WriteInt( -1 );
return;
}
// If we already have this string in our hash, write out of the offset in the table and return
int hash = stringHash.GenerateKey( string );
for( int i = stringHash.First( hash ); i != -1; i = stringHash.Next( i ) )
{
if( stringTable[i].string.Cmp( string ) == 0 )
{
WriteInt( stringTable[i].offset );
return;
}
}
// Add the string to our hash, generate the index, and update our current table offset
stringTableIndex_s& tableIndex = stringTable.Alloc();
tableIndex.offset = curStringTableOffset;
tableIndex.string = string;
stringHash.Add( hash, stringTable.Num() - 1 );
WriteInt( curStringTableOffset );
curStringTableOffset += ( strlen( string ) + 4 );
}
/*
================
idSaveGame::WriteVec2
================
*/
void idSaveGame::WriteVec2( const idVec2& vec )
{
file->WriteBig( vec );
}
/*
================
idSaveGame::WriteVec3
================
*/
void idSaveGame::WriteVec3( const idVec3& vec )
{
file->WriteBig( vec );
}
/*
================
idSaveGame::WriteVec4
================
*/
void idSaveGame::WriteVec4( const idVec4& vec )
{
file->WriteBig( vec );
}
/*
================
idSaveGame::WriteVec6
================
*/
void idSaveGame::WriteVec6( const idVec6& vec )
{
file->WriteBig( vec );
}
/*
================
idSaveGame::WriteBounds
================
*/
void idSaveGame::WriteBounds( const idBounds& bounds )
{
file->WriteBig( bounds );
}
/*
================
idSaveGame::WriteBounds
================
*/
void idSaveGame::WriteWinding( const idWinding& w )
{
int i, num;
num = w.GetNumPoints();
file->WriteBig( num );
for( i = 0; i < num; i++ )
{
idVec5 v = w[i];
file->WriteBig( v );
}
}
/*
================
idSaveGame::WriteMat3
================
*/
void idSaveGame::WriteMat3( const idMat3& mat )
{
file->WriteBig( mat );
}
/*
================
idSaveGame::WriteAngles
================
*/
void idSaveGame::WriteAngles( const idAngles& angles )
{
file->WriteBig( angles );
}
/*
================
idSaveGame::WriteObject
================
*/
void idSaveGame::WriteObject( const idClass* obj )
{
int index;
index = objects.FindIndex( obj );
if( index < 0 )
{
gameLocal.DPrintf( "idSaveGame::WriteObject - WriteObject FindIndex failed\n" );
// Use the NULL index
index = 0;
}
WriteInt( index );
}
/*
================
idSaveGame::WriteStaticObject
================
*/
void idSaveGame::WriteStaticObject( const idClass& obj )
{
CallSave_r( obj.GetType(), &obj );
}
/*
================
idSaveGame::WriteDict
================
*/
void idSaveGame::WriteDict( const idDict* dict )
{
int num;
int i;
const idKeyValue* kv;
if( !dict )
{
WriteInt( -1 );
}
else
{
num = dict->GetNumKeyVals();
WriteInt( num );
for( i = 0; i < num; i++ )
{
kv = dict->GetKeyVal( i );
WriteString( kv->GetKey() );
WriteString( kv->GetValue() );
}
}
}
/*
================
idSaveGame::WriteMaterial
================
*/
void idSaveGame::WriteMaterial( const idMaterial* material )
{
if( !material )
{
WriteString( "" );
}
else
{
WriteString( material->GetName() );
}
}
/*
================
idSaveGame::WriteSkin
================
*/
void idSaveGame::WriteSkin( const idDeclSkin* skin )
{
if( !skin )
{
WriteString( "" );
}
else
{
WriteString( skin->GetName() );
}
}
/*
================
idSaveGame::WriteParticle
================
*/
void idSaveGame::WriteParticle( const idDeclParticle* particle )
{
if( !particle )
{
WriteString( "" );
}
else
{
WriteString( particle->GetName() );
}
}
/*
================
idSaveGame::WriteFX
================
*/
void idSaveGame::WriteFX( const idDeclFX* fx )
{
if( !fx )
{
WriteString( "" );
}
else
{
WriteString( fx->GetName() );
}
}
/*
================
idSaveGame::WriteModelDef
================
*/
void idSaveGame::WriteModelDef( const idDeclModelDef* modelDef )
{
if( !modelDef )
{
WriteString( "" );
}
else
{
WriteString( modelDef->GetName() );
}
}
/*
================
idSaveGame::WriteSoundShader
================
*/
void idSaveGame::WriteSoundShader( const idSoundShader* shader )
{
const char* name;
if( !shader )
{
WriteString( "" );
}
else
{
name = shader->GetName();
WriteString( name );
}
}
/*
================
idSaveGame::WriteModel
================
*/
void idSaveGame::WriteModel( const idRenderModel* model )
{
const char* name;
if( !model )
{
WriteString( "" );
}
else
{
name = model->Name();
WriteString( name );
}
}
/*
================
idSaveGame::WriteUserInterface
================
*/
void idSaveGame::WriteUserInterface( const idUserInterface* ui, bool unique )
{
const char* name;
if( !ui )
{
WriteString( "" );
}
else
{
name = ui->Name();
WriteString( name );
WriteBool( unique );
if( ui->WriteToSaveGame( file ) == false )
{
gameLocal.Error( "idSaveGame::WriteUserInterface: ui failed to write properly\n" );
}
}
}
/*
================
idSaveGame::WriteRenderEntity
================
*/
void idSaveGame::WriteRenderEntity( const renderEntity_t& renderEntity )
{
int i;
WriteModel( renderEntity.hModel );
WriteInt( renderEntity.entityNum );
WriteInt( renderEntity.bodyId );
WriteBounds( renderEntity.bounds );
// callback is set by class's Restore function
WriteInt( renderEntity.suppressSurfaceInViewID );
WriteInt( renderEntity.suppressShadowInViewID );
WriteInt( renderEntity.suppressShadowInLightID );
WriteInt( renderEntity.allowSurfaceInViewID );
WriteVec3( renderEntity.origin );
WriteMat3( renderEntity.axis );
WriteMaterial( renderEntity.customShader );
WriteMaterial( renderEntity.referenceShader );
WriteSkin( renderEntity.customSkin );
if( renderEntity.referenceSound != NULL )
{
WriteInt( renderEntity.referenceSound->Index() );
}
else
{
WriteInt( 0 );
}
for( i = 0; i < MAX_ENTITY_SHADER_PARMS; i++ )
{
WriteFloat( renderEntity.shaderParms[ i ] );
}
for( i = 0; i < MAX_RENDERENTITY_GUI; i++ )
{
WriteUserInterface( renderEntity.gui[ i ], renderEntity.gui[ i ] ? renderEntity.gui[ i ]->IsUniqued() : false );
}
WriteFloat( renderEntity.modelDepthHack );
WriteBool( renderEntity.noSelfShadow );
WriteBool( renderEntity.noShadow );
WriteBool( renderEntity.noDynamicInteractions );
WriteBool( renderEntity.weaponDepthHack );
WriteInt( renderEntity.forceUpdate );
WriteInt( renderEntity.timeGroup );
WriteInt( renderEntity.xrayIndex );
}
/*
================
idSaveGame::WriteRenderLight
================
*/
void idSaveGame::WriteRenderLight( const renderLight_t& renderLight )
{
int i;
WriteMat3( renderLight.axis );
WriteVec3( renderLight.origin );
WriteInt( renderLight.suppressLightInViewID );
WriteInt( renderLight.allowLightInViewID );
WriteBool( renderLight.noShadows );
WriteBool( renderLight.noSpecular );
WriteBool( renderLight.pointLight );
WriteBool( renderLight.parallel );
WriteVec3( renderLight.lightRadius );
WriteVec3( renderLight.lightCenter );
WriteVec3( renderLight.target );
WriteVec3( renderLight.right );
WriteVec3( renderLight.up );
WriteVec3( renderLight.start );
WriteVec3( renderLight.end );
// only idLight has a prelightModel and it's always based on the entityname, so we'll restore it there
// WriteModel( renderLight.prelightModel );
WriteInt( renderLight.lightId );
WriteMaterial( renderLight.shader );
for( i = 0; i < MAX_ENTITY_SHADER_PARMS; i++ )
{
WriteFloat( renderLight.shaderParms[ i ] );
}
if( renderLight.referenceSound != NULL )
{
WriteInt( renderLight.referenceSound->Index() );
}
else
{
WriteInt( 0 );
}
}
/*
================
idSaveGame::WriteRefSound
================
*/
void idSaveGame::WriteRefSound( const refSound_t& refSound )
{
if( refSound.referenceSound )
{
WriteInt( refSound.referenceSound->Index() );
}
else
{
WriteInt( 0 );
}
WriteVec3( refSound.origin );
WriteInt( refSound.listenerId );
WriteSoundShader( refSound.shader );
WriteFloat( refSound.diversity );
WriteBool( refSound.waitfortrigger );
WriteFloat( refSound.parms.minDistance );
WriteFloat( refSound.parms.maxDistance );
WriteFloat( refSound.parms.volume );
WriteFloat( refSound.parms.shakes );
WriteInt( refSound.parms.soundShaderFlags );
WriteInt( refSound.parms.soundClass );
}
/*
================
idSaveGame::WriteRenderView
================
*/
void idSaveGame::WriteRenderView( const renderView_t& view )
{
int i;
WriteInt( view.viewID );
WriteInt( 0 /* view.x */ );
WriteInt( 0 /* view.y */ );
WriteInt( 0 /* view.width */ );
WriteInt( 0 /* view.height */ );
WriteFloat( view.fov_x );
WriteFloat( view.fov_y );
WriteVec3( view.vieworg );
WriteMat3( view.viewaxis );
WriteBool( view.cramZNear );
WriteInt( view.time[0] );
for( i = 0; i < MAX_GLOBAL_SHADER_PARMS; i++ )
{
WriteFloat( view.shaderParms[ i ] );
}
}
/*
===================
idSaveGame::WriteUsercmd
===================
*/
void idSaveGame::WriteUsercmd( const usercmd_t& usercmd )
{
WriteByte( usercmd.buttons );
WriteSignedChar( usercmd.forwardmove );
WriteSignedChar( usercmd.rightmove );
WriteShort( usercmd.angles[0] );
WriteShort( usercmd.angles[1] );
WriteShort( usercmd.angles[2] );
WriteShort( usercmd.mx );
WriteShort( usercmd.my );
WriteByte( usercmd.impulse );
WriteByte( usercmd.impulseSequence );
}
/*
===================
idSaveGame::WriteContactInfo
===================
*/
void idSaveGame::WriteContactInfo( const contactInfo_t& contactInfo )
{
WriteInt( ( int )contactInfo.type );
WriteVec3( contactInfo.point );
WriteVec3( contactInfo.normal );
WriteFloat( contactInfo.dist );
WriteInt( contactInfo.contents );
WriteMaterial( contactInfo.material );
WriteInt( contactInfo.modelFeature );
WriteInt( contactInfo.trmFeature );
WriteInt( contactInfo.entityNum );
WriteInt( contactInfo.id );
}
/*
===================
idSaveGame::WriteTrace
===================
*/
void idSaveGame::WriteTrace( const trace_t& trace )
{
WriteFloat( trace.fraction );
WriteVec3( trace.endpos );
WriteMat3( trace.endAxis );
WriteContactInfo( trace.c );
}
/*
===================
idRestoreGame::WriteTraceModel
===================
*/
void idSaveGame::WriteTraceModel( const idTraceModel& trace )
{
int j, k;
WriteInt( ( int& )trace.type );
WriteInt( trace.numVerts );
for( j = 0; j < MAX_TRACEMODEL_VERTS; j++ )
{
WriteVec3( trace.verts[j] );
}
WriteInt( trace.numEdges );
for( j = 0; j < ( MAX_TRACEMODEL_EDGES + 1 ); j++ )
{
WriteInt( trace.edges[j].v[0] );
WriteInt( trace.edges[j].v[1] );
WriteVec3( trace.edges[j].normal );
}
WriteInt( trace.numPolys );
for( j = 0; j < MAX_TRACEMODEL_POLYS; j++ )
{
WriteVec3( trace.polys[j].normal );
WriteFloat( trace.polys[j].dist );
WriteBounds( trace.polys[j].bounds );
WriteInt( trace.polys[j].numEdges );
for( k = 0; k < MAX_TRACEMODEL_POLYEDGES; k++ )
{
WriteInt( trace.polys[j].edges[k] );
}
}
WriteVec3( trace.offset );
WriteBounds( trace.bounds );
WriteBool( trace.isConvex );
// padding win32 native structs
char tmp[3];
memset( tmp, 0, sizeof( tmp ) );
file->Write( tmp, 3 );
}
/*
===================
idSaveGame::WriteClipModel
===================
*/
void idSaveGame::WriteClipModel( const idClipModel* clipModel )
{
if( clipModel != NULL )
{
WriteBool( true );
clipModel->Save( this );
}
else
{
WriteBool( false );
}
}
/*
===================
idSaveGame::WriteSoundCommands
===================
*/
void idSaveGame::WriteSoundCommands()
{
gameSoundWorld->WriteToSaveGame( file );
}
/*
======================
idSaveGame::WriteBuildNumber
======================
*/
void idSaveGame::WriteBuildNumber( const int value )
{
WriteInt( BUILD_NUMBER );
}
/***********************************************************************
idRestoreGame
***********************************************************************/
/*
================
idRestoreGame::RestoreGame
================
*/
idRestoreGame::idRestoreGame( idFile* savefile, idFile* stringTableFile, int saveVersion )
{
file = savefile;
stringFile = stringTableFile;
version = saveVersion;
}
/*
================
idRestoreGame::~idRestoreGame()
================
*/
idRestoreGame::~idRestoreGame()
{
}
/*
================
idRestoreGame::ReadDecls
================
*/
void idRestoreGame::ReadDecls()
{
idStr declName;
for( int t = 0; t < declManager->GetNumDeclTypes(); t++ )
{
while( true )
{
ReadString( declName );
if( declName.IsEmpty() )
{
break;
}
declManager->FindType( ( declType_t )t, declName );
}
}
}
/*
================
void idRestoreGame::CreateObjects
================
*/
void idRestoreGame::CreateObjects()
{
int i, num;
idStr classname;
idTypeInfo* type;
ReadInt( num );
// create all the objects
objects.SetNum( num + 1 );
memset( objects.Ptr(), 0, sizeof( objects[ 0 ] ) * objects.Num() );
for( i = 1; i < objects.Num(); i++ )
{
ReadString( classname );
type = idClass::GetClass( classname );
if( type == NULL )
{
Error( "idRestoreGame::CreateObjects: Unknown class '%s'", classname.c_str() );
return;
}
objects[ i ] = type->CreateInstance();
#ifdef ID_DEBUG_MEMORY
InitTypeVariables( objects[i], type->classname, 0xce );
#endif
}
}
/*
================
void idRestoreGame::RestoreObjects
================
*/
void idRestoreGame::RestoreObjects()
{
int i;
ReadSoundCommands();
// read trace models
idClipModel::RestoreTraceModels( this );
// restore all the objects
for( i = 1; i < objects.Num(); i++ )
{
CallRestore_r( objects[ i ]->GetType(), objects[ i ] );
}
// regenerate render entities and render lights because are not saved
for( i = 1; i < objects.Num(); i++ )
{
if( objects[ i ]->IsType( idEntity::Type ) )
{
idEntity* ent = static_cast( objects[ i ] );
ent->UpdateVisuals();
ent->Present();
}
}
#ifdef ID_DEBUG_MEMORY
idStr gameState = file->GetName();
gameState.StripFileExtension();
WriteGameState_f( idCmdArgs( va( "test %s_restore", gameState.c_str() ), false ) );
//CompareGameState_f( idCmdArgs( va( "test %s_save", gameState.c_str() ) ) );
gameLocal.Error( "dumped game states" );
#endif
}
/*
====================
void idRestoreGame::DeleteObjects
====================
*/
void idRestoreGame::DeleteObjects()
{
// Remove the NULL object before deleting
objects.RemoveIndex( 0 );
objects.DeleteContents( true );
}
/*
================
idRestoreGame::Error
================
*/
void idRestoreGame::Error( const char* fmt, ... )
{
va_list argptr;
char text[ 1024 ];
va_start( argptr, fmt );
vsprintf( text, fmt, argptr );
va_end( argptr );
objects.DeleteContents( true );
gameLocal.Error( "%s", text );
}
/*
================
idRestoreGame::CallRestore_r
================
*/
void idRestoreGame::CallRestore_r( const idTypeInfo* cls, idClass* obj )
{
if( cls->super )
{
CallRestore_r( cls->super, obj );
if( cls->super->Restore == cls->Restore )
{
// don't call save on this inheritance level since the function was called in the super class
return;
}
}
( obj->*cls->Restore )( this );
}
/*
================
idRestoreGame::Read
================
*/
void idRestoreGame::Read( void* buffer, int len )
{
file->Read( buffer, len );
}
/*
================
idRestoreGame::ReadInt
================
*/
void idRestoreGame::ReadInt( int& value )
{
file->ReadBig( value );
}
/*
================
idRestoreGame::ReadJoint
================
*/
void idRestoreGame::ReadJoint( jointHandle_t& value )
{
file->ReadBig( ( int& )value );
}
/*
================
idRestoreGame::ReadShort
================
*/
void idRestoreGame::ReadShort( short& value )
{
file->ReadBig( value );
}
/*
================
idRestoreGame::ReadByte
================
*/
void idRestoreGame::ReadByte( byte& value )
{
file->Read( &value, sizeof( value ) );
}
/*
================
idRestoreGame::ReadSignedChar
================
*/
void idRestoreGame::ReadSignedChar( signed char& value )
{
file->Read( &value, sizeof( value ) );
}
/*
================
idRestoreGame::ReadFloat
================
*/
void idRestoreGame::ReadFloat( float& value )
{
file->ReadBig( value );
}
/*
================
idRestoreGame::ReadBool
================
*/
void idRestoreGame::ReadBool( bool& value )
{
file->ReadBig( value );
}
/*
================
idRestoreGame::ReadString
================
*/
void idRestoreGame::ReadString( idStr& string )
{
string.Empty();
int offset = -1;
ReadInt( offset );
if( offset < 0 )
{
return;
}
stringFile->Seek( offset, FS_SEEK_SET );
stringFile->ReadString( string );
return;
}
/*
================
idRestoreGame::ReadVec2
================
*/
void idRestoreGame::ReadVec2( idVec2& vec )
{
file->ReadBig( vec );
}
/*
================
idRestoreGame::ReadVec3
================
*/
void idRestoreGame::ReadVec3( idVec3& vec )
{
file->ReadBig( vec );
}
/*
================
idRestoreGame::ReadVec4
================
*/
void idRestoreGame::ReadVec4( idVec4& vec )
{
file->ReadBig( vec );
}
/*
================
idRestoreGame::ReadVec6
================
*/
void idRestoreGame::ReadVec6( idVec6& vec )
{
file->ReadBig( vec );
}
/*
================
idRestoreGame::ReadBounds
================
*/
void idRestoreGame::ReadBounds( idBounds& bounds )
{
file->ReadBig( bounds );
}
/*
================
idRestoreGame::ReadWinding
================
*/
void idRestoreGame::ReadWinding( idWinding& w )
{
int i, num;
ReadInt( num );
w.SetNumPoints( num );
for( i = 0; i < num; i++ )
{
idVec5& v = w[i];
file->ReadBig( v );
}
}
/*
================
idRestoreGame::ReadMat3
================
*/
void idRestoreGame::ReadMat3( idMat3& mat )
{
file->ReadBig( mat );
}
/*
================
idRestoreGame::ReadAngles
================
*/
void idRestoreGame::ReadAngles( idAngles& angles )
{
file->ReadBig( angles );
}
/*
================
idRestoreGame::ReadObject
================
*/
void idRestoreGame::ReadObject( idClass*& obj )
{
int index;
ReadInt( index );
if( ( index < 0 ) || ( index >= objects.Num() ) )
{
Error( "idRestoreGame::ReadObject: invalid object index" );
}
obj = objects[ index ];
}
/*
================
idRestoreGame::ReadStaticObject
================
*/
void idRestoreGame::ReadStaticObject( idClass& obj )
{
CallRestore_r( obj.GetType(), &obj );
}
/*
================
idRestoreGame::ReadDict
================
*/
void idRestoreGame::ReadDict( idDict* dict )
{
int num;
int i;
idStr key;
idStr value;
ReadInt( num );
if( num < 0 )
{
dict = NULL;
}
else
{
dict->Clear();
for( i = 0; i < num; i++ )
{
ReadString( key );
ReadString( value );
dict->Set( key, value );
}
}
}
/*
================
idRestoreGame::ReadMaterial
================
*/
void idRestoreGame::ReadMaterial( const idMaterial*& material )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
material = NULL;
}
else
{
material = declManager->FindMaterial( name );
}
}
/*
================
idRestoreGame::ReadSkin
================
*/
void idRestoreGame::ReadSkin( const idDeclSkin*& skin )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
skin = NULL;
}
else
{
skin = declManager->FindSkin( name );
}
}
/*
================
idRestoreGame::ReadParticle
================
*/
void idRestoreGame::ReadParticle( const idDeclParticle*& particle )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
particle = NULL;
}
else
{
particle = static_cast( declManager->FindType( DECL_PARTICLE, name ) );
}
}
/*
================
idRestoreGame::ReadFX
================
*/
void idRestoreGame::ReadFX( const idDeclFX*& fx )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
fx = NULL;
}
else
{
fx = static_cast( declManager->FindType( DECL_FX, name ) );
}
}
/*
================
idRestoreGame::ReadSoundShader
================
*/
void idRestoreGame::ReadSoundShader( const idSoundShader*& shader )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
shader = NULL;
}
else
{
shader = declManager->FindSound( name );
}
}
/*
================
idRestoreGame::ReadModelDef
================
*/
void idRestoreGame::ReadModelDef( const idDeclModelDef*& modelDef )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
modelDef = NULL;
}
else
{
modelDef = static_cast( declManager->FindType( DECL_MODELDEF, name, false ) );
}
}
/*
================
idRestoreGame::ReadModel
================
*/
void idRestoreGame::ReadModel( idRenderModel*& model )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
model = NULL;
}
else
{
model = renderModelManager->FindModel( name );
}
}
/*
================
idRestoreGame::ReadUserInterface
================
*/
void idRestoreGame::ReadUserInterface( idUserInterface*& ui )
{
idStr name;
ReadString( name );
if( !name.Length() )
{
ui = NULL;
}
else
{
bool unique;
ReadBool( unique );
ui = uiManager->FindGui( name, true, unique );
if( ui )
{
if( ui->ReadFromSaveGame( file ) == false )
{
Error( "idSaveGame::ReadUserInterface: ui failed to read properly\n" );
}
else
{
ui->StateChanged( gameLocal.time );
}
}
}
}
/*
================
idRestoreGame::ReadRenderEntity
================
*/
void idRestoreGame::ReadRenderEntity( renderEntity_t& renderEntity )
{
int i;
int index;
ReadModel( renderEntity.hModel );
ReadInt( renderEntity.entityNum );
ReadInt( renderEntity.bodyId );
ReadBounds( renderEntity.bounds );
// callback is set by class's Restore function
renderEntity.callback = NULL;
renderEntity.callbackData = NULL;
ReadInt( renderEntity.suppressSurfaceInViewID );
ReadInt( renderEntity.suppressShadowInViewID );
ReadInt( renderEntity.suppressShadowInLightID );
ReadInt( renderEntity.allowSurfaceInViewID );
ReadVec3( renderEntity.origin );
ReadMat3( renderEntity.axis );
ReadMaterial( renderEntity.customShader );
ReadMaterial( renderEntity.referenceShader );
ReadSkin( renderEntity.customSkin );
ReadInt( index );
renderEntity.referenceSound = gameSoundWorld->EmitterForIndex( index );
for( i = 0; i < MAX_ENTITY_SHADER_PARMS; i++ )
{
ReadFloat( renderEntity.shaderParms[ i ] );
}
for( i = 0; i < MAX_RENDERENTITY_GUI; i++ )
{
ReadUserInterface( renderEntity.gui[ i ] );
}
// idEntity will restore "cameraTarget", which will be used in idEntity::Present to restore the remoteRenderView
renderEntity.remoteRenderView = NULL;
renderEntity.joints = NULL;
renderEntity.numJoints = 0;
ReadFloat( renderEntity.modelDepthHack );
ReadBool( renderEntity.noSelfShadow );
ReadBool( renderEntity.noShadow );
ReadBool( renderEntity.noDynamicInteractions );
ReadBool( renderEntity.weaponDepthHack );
ReadInt( renderEntity.forceUpdate );
ReadInt( renderEntity.timeGroup );
ReadInt( renderEntity.xrayIndex );
}
/*
================
idRestoreGame::ReadRenderLight
================
*/
void idRestoreGame::ReadRenderLight( renderLight_t& renderLight )
{
int index;
int i;
ReadMat3( renderLight.axis );
ReadVec3( renderLight.origin );
ReadInt( renderLight.suppressLightInViewID );
ReadInt( renderLight.allowLightInViewID );
ReadBool( renderLight.noShadows );
ReadBool( renderLight.noSpecular );
ReadBool( renderLight.pointLight );
ReadBool( renderLight.parallel );
ReadVec3( renderLight.lightRadius );
ReadVec3( renderLight.lightCenter );
ReadVec3( renderLight.target );
ReadVec3( renderLight.right );
ReadVec3( renderLight.up );
ReadVec3( renderLight.start );
ReadVec3( renderLight.end );
// only idLight has a prelightModel and it's always based on the entityname, so we'll restore it there
// ReadModel( renderLight.prelightModel );
renderLight.prelightModel = NULL;
ReadInt( renderLight.lightId );
ReadMaterial( renderLight.shader );
for( i = 0; i < MAX_ENTITY_SHADER_PARMS; i++ )
{
ReadFloat( renderLight.shaderParms[ i ] );
}
ReadInt( index );
renderLight.referenceSound = gameSoundWorld->EmitterForIndex( index );
}
/*
================
idRestoreGame::ReadRefSound
================
*/
void idRestoreGame::ReadRefSound( refSound_t& refSound )
{
int index;
ReadInt( index );
refSound.referenceSound = gameSoundWorld->EmitterForIndex( index );
ReadVec3( refSound.origin );
ReadInt( refSound.listenerId );
ReadSoundShader( refSound.shader );
ReadFloat( refSound.diversity );
ReadBool( refSound.waitfortrigger );
ReadFloat( refSound.parms.minDistance );
ReadFloat( refSound.parms.maxDistance );
ReadFloat( refSound.parms.volume );
ReadFloat( refSound.parms.shakes );
ReadInt( refSound.parms.soundShaderFlags );
ReadInt( refSound.parms.soundClass );
}
/*
================
idRestoreGame::ReadRenderView
================
*/
void idRestoreGame::ReadRenderView( renderView_t& view )
{
int i;
ReadInt( view.viewID );
ReadInt( i /* view.x */ );
ReadInt( i /* view.y */ );
ReadInt( i /* view.width */ );
ReadInt( i /* view.height */ );
ReadFloat( view.fov_x );
ReadFloat( view.fov_y );
ReadVec3( view.vieworg );
ReadMat3( view.viewaxis );
ReadBool( view.cramZNear );
ReadInt( view.time[0] );
for( i = 0; i < MAX_GLOBAL_SHADER_PARMS; i++ )
{
ReadFloat( view.shaderParms[ i ] );
}
}
/*
=================
idRestoreGame::ReadUsercmd
=================
*/
void idRestoreGame::ReadUsercmd( usercmd_t& usercmd )
{
ReadByte( usercmd.buttons );
ReadSignedChar( usercmd.forwardmove );
ReadSignedChar( usercmd.rightmove );
ReadShort( usercmd.angles[0] );
ReadShort( usercmd.angles[1] );
ReadShort( usercmd.angles[2] );
ReadShort( usercmd.mx );
ReadShort( usercmd.my );
ReadByte( usercmd.impulse );
ReadByte( usercmd.impulseSequence );
}
/*
===================
idRestoreGame::ReadContactInfo
===================
*/
void idRestoreGame::ReadContactInfo( contactInfo_t& contactInfo )
{
ReadInt( ( int& )contactInfo.type );
ReadVec3( contactInfo.point );
ReadVec3( contactInfo.normal );
ReadFloat( contactInfo.dist );
ReadInt( contactInfo.contents );
ReadMaterial( contactInfo.material );
ReadInt( contactInfo.modelFeature );
ReadInt( contactInfo.trmFeature );
ReadInt( contactInfo.entityNum );
ReadInt( contactInfo.id );
}
/*
===================
idRestoreGame::ReadTrace
===================
*/
void idRestoreGame::ReadTrace( trace_t& trace )
{
ReadFloat( trace.fraction );
ReadVec3( trace.endpos );
ReadMat3( trace.endAxis );
ReadContactInfo( trace.c );
}
/*
===================
idRestoreGame::ReadTraceModel
===================
*/
void idRestoreGame::ReadTraceModel( idTraceModel& trace )
{
int j, k;
ReadInt( ( int& )trace.type );
ReadInt( trace.numVerts );
for( j = 0; j < MAX_TRACEMODEL_VERTS; j++ )
{
ReadVec3( trace.verts[j] );
}
ReadInt( trace.numEdges );
for( j = 0; j < ( MAX_TRACEMODEL_EDGES + 1 ); j++ )
{
ReadInt( trace.edges[j].v[0] );
ReadInt( trace.edges[j].v[1] );
ReadVec3( trace.edges[j].normal );
}
ReadInt( trace.numPolys );
for( j = 0; j < MAX_TRACEMODEL_POLYS; j++ )
{
ReadVec3( trace.polys[j].normal );
ReadFloat( trace.polys[j].dist );
ReadBounds( trace.polys[j].bounds );
ReadInt( trace.polys[j].numEdges );
for( k = 0; k < MAX_TRACEMODEL_POLYEDGES; k++ )
{
ReadInt( trace.polys[j].edges[k] );
}
}
ReadVec3( trace.offset );
ReadBounds( trace.bounds );
ReadBool( trace.isConvex );
// padding win32 native structs
char tmp[3];
file->Read( tmp, 3 );
}
/*
=====================
idRestoreGame::ReadClipModel
=====================
*/
void idRestoreGame::ReadClipModel( idClipModel*& clipModel )
{
bool restoreClipModel;
ReadBool( restoreClipModel );
if( restoreClipModel )
{
clipModel = new( TAG_SAVEGAMES ) idClipModel();
clipModel->Restore( this );
}
else
{
clipModel = NULL;
}
}
/*
=====================
idRestoreGame::ReadSoundCommands
=====================
*/
void idRestoreGame::ReadSoundCommands()
{
gameSoundWorld->StopAllSounds();
gameSoundWorld->ReadFromSaveGame( file );
}