123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944 |
- /*
- ===========================================================================
- 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 <http://www.gnu.org/licenses/>.
- 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.
- ===========================================================================
- */
- #include "precompiled.h"
- #pragma hdrstop
- idStrPool idDict::globalKeys;
- idStrPool idDict::globalValues;
- /*
- ================
- idDict::operator=
- clear existing key/value pairs and copy all key/value pairs from other
- ================
- */
- idDict &idDict::operator=( const idDict &other ) {
- int i;
- // check for assignment to self
- if ( this == &other ) {
- return *this;
- }
- Clear();
- args = other.args;
- argHash = other.argHash;
- for ( i = 0; i < args.Num(); i++ ) {
- args[i].key = globalKeys.CopyString( args[i].key );
- args[i].value = globalValues.CopyString( args[i].value );
- }
- return *this;
- }
- /*
- ================
- idDict::Copy
- copy all key value pairs without removing existing key/value pairs not present in the other dict
- ================
- */
- void idDict::Copy( const idDict &other ) {
- int i, n, *found;
- idKeyValue kv;
- // check for assignment to self
- if ( this == &other ) {
- return;
- }
- n = other.args.Num();
- if ( args.Num() ) {
- found = (int *) _alloca16( other.args.Num() * sizeof( int ) );
- for ( i = 0; i < n; i++ ) {
- found[i] = FindKeyIndex( other.args[i].GetKey() );
- }
- } else {
- found = NULL;
- }
- for ( i = 0; i < n; i++ ) {
- if ( found && found[i] != -1 ) {
- // first set the new value and then free the old value to allow proper self copying
- const idPoolStr *oldValue = args[found[i]].value;
- args[found[i]].value = globalValues.CopyString( other.args[i].value );
- globalValues.FreeString( oldValue );
- } else {
- kv.key = globalKeys.CopyString( other.args[i].key );
- kv.value = globalValues.CopyString( other.args[i].value );
- argHash.Add( argHash.GenerateKey( kv.GetKey(), false ), args.Append( kv ) );
- }
- }
- }
- /*
- ================
- idDict::TransferKeyValues
- clear existing key/value pairs and transfer key/value pairs from other
- ================
- */
- void idDict::TransferKeyValues( idDict &other ) {
- int i, n;
- if ( this == &other ) {
- return;
- }
- if ( other.args.Num() && other.args[0].key->GetPool() != &globalKeys ) {
- common->FatalError( "idDict::TransferKeyValues: can't transfer values across a DLL boundary" );
- return;
- }
- Clear();
- n = other.args.Num();
- args.SetNum( n );
- for ( i = 0; i < n; i++ ) {
- args[i].key = other.args[i].key;
- args[i].value = other.args[i].value;
- }
- argHash = other.argHash;
- other.args.Clear();
- other.argHash.Free();
- }
- /*
- ================
- idDict::Parse
- ================
- */
- bool idDict::Parse( idParser &parser ) {
- idToken token;
- idToken token2;
- bool errors;
- errors = false;
- parser.ExpectTokenString( "{" );
- parser.ReadToken( &token );
- while( ( token.type != TT_PUNCTUATION ) || ( token != "}" ) ) {
- if ( token.type != TT_STRING ) {
- parser.Error( "Expected quoted string, but found '%s'", token.c_str() );
- }
- if ( !parser.ReadToken( &token2 ) ) {
- parser.Error( "Unexpected end of file" );
- }
- if ( FindKey( token ) ) {
- parser.Warning( "'%s' already defined", token.c_str() );
- errors = true;
- }
- Set( token, token2 );
- if ( !parser.ReadToken( &token ) ) {
- parser.Error( "Unexpected end of file" );
- }
- }
- return !errors;
- }
- /*
- ================
- idDict::SetDefaults
- ================
- */
- void idDict::SetDefaults( const idDict *dict ) {
- int i, n;
- const idKeyValue *kv, *def;
- idKeyValue newkv;
- n = dict->args.Num();
- for( i = 0; i < n; i++ ) {
- def = &dict->args[i];
- kv = FindKey( def->GetKey() );
- if ( !kv ) {
- newkv.key = globalKeys.CopyString( def->key );
- newkv.value = globalValues.CopyString( def->value );
- argHash.Add( argHash.GenerateKey( newkv.GetKey(), false ), args.Append( newkv ) );
- }
- }
- }
- /*
- ================
- idDict::Clear
- ================
- */
- void idDict::Clear() {
- int i;
- for( i = 0; i < args.Num(); i++ ) {
- globalKeys.FreeString( args[i].key );
- globalValues.FreeString( args[i].value );
- }
- args.Clear();
- argHash.Free();
- }
- /*
- ================
- idDict::Print
- ================
- */
- void idDict::Print() const {
- int i;
- int n;
- n = args.Num();
- for( i = 0; i < n; i++ ) {
- idLib::common->Printf( "%s = %s\n", args[i].GetKey().c_str(), args[i].GetValue().c_str() );
- }
- }
- int KeyCompare( const idKeyValue *a, const idKeyValue *b ) {
- return idStr::Cmp( a->GetKey(), b->GetKey() );
- }
- /*
- ================
- idDict::Checksum
- ================
- */
- int idDict::Checksum() const {
- unsigned long ret;
- int i, n;
- idList<idKeyValue> sorted = args;
- sorted.SortWithTemplate( idSort_KeyValue() );
- n = sorted.Num();
- CRC32_InitChecksum( ret );
- for( i = 0; i < n; i++ ) {
- CRC32_UpdateChecksum( ret, sorted[i].GetKey().c_str(), sorted[i].GetKey().Length() );
- CRC32_UpdateChecksum( ret, sorted[i].GetValue().c_str(), sorted[i].GetValue().Length() );
- }
- CRC32_FinishChecksum( ret );
- return ret;
- }
- /*
- ================
- idDict::Allocated
- ================
- */
- size_t idDict::Allocated() const {
- int i;
- size_t size;
- size = args.Allocated() + argHash.Allocated();
- for( i = 0; i < args.Num(); i++ ) {
- size += args[i].Size();
- }
- return size;
- }
- /*
- ================
- idDict::Set
- ================
- */
- void idDict::Set( const char *key, const char *value ) {
- int i;
- idKeyValue kv;
- if ( key == NULL || key[0] == '\0' ) {
- return;
- }
- i = FindKeyIndex( key );
- if ( i != -1 ) {
- // first set the new value and then free the old value to allow proper self copying
- const idPoolStr *oldValue = args[i].value;
- args[i].value = globalValues.AllocString( value );
- globalValues.FreeString( oldValue );
- } else {
- kv.key = globalKeys.AllocString( key );
- kv.value = globalValues.AllocString( value );
- argHash.Add( argHash.GenerateKey( kv.GetKey(), false ), args.Append( kv ) );
- }
- }
- /*
- ================
- idDict::GetFloat
- ================
- */
- bool idDict::GetFloat( const char *key, const char *defaultString, float &out ) const {
- const char *s;
- bool found;
- found = GetString( key, defaultString, &s );
- out = atof( s );
- return found;
- }
- /*
- ================
- idDict::GetInt
- ================
- */
- bool idDict::GetInt( const char *key, const char *defaultString, int &out ) const {
- const char *s;
- bool found;
- found = GetString( key, defaultString, &s );
- out = atoi( s );
- return found;
- }
- /*
- ================
- idDict::GetBool
- ================
- */
- bool idDict::GetBool( const char *key, const char *defaultString, bool &out ) const {
- const char *s;
- bool found;
- found = GetString( key, defaultString, &s );
- out = ( atoi( s ) != 0 );
- return found;
- }
- /*
- ================
- idDict::GetFloat
- ================
- */
- bool idDict::GetFloat( const char *key, const float defaultFloat, float &out ) const {
- const idKeyValue *kv = FindKey( key );
- if ( kv ) {
- out = atof( kv->GetValue() );
- return true;
- } else {
- out = defaultFloat;
- return false;
- }
- }
- /*
- ================
- idDict::GetInt
- ================
- */
- bool idDict::GetInt( const char *key, const int defaultInt, int &out ) const {
- const idKeyValue *kv = FindKey( key );
- if ( kv ) {
- out = atoi( kv->GetValue() );
- return true;
- } else {
- out = defaultInt;
- return false;
- }
- }
- /*
- ================
- idDict::GetBool
- ================
- */
- bool idDict::GetBool( const char *key, const bool defaultBool, bool &out ) const {
- const idKeyValue *kv = FindKey( key );
- if ( kv ) {
- out = ( atoi( kv->GetValue() ) != 0 );
- return true;
- } else {
- out = defaultBool;
- return false;
- }
- }
- /*
- ================
- idDict::GetAngles
- ================
- */
- bool idDict::GetAngles( const char *key, const char *defaultString, idAngles &out ) const {
- bool found;
- const char *s;
-
- if ( !defaultString ) {
- defaultString = "0 0 0";
- }
- found = GetString( key, defaultString, &s );
- out.Zero();
- sscanf( s, "%f %f %f", &out.pitch, &out.yaw, &out.roll );
- return found;
- }
- /*
- ================
- idDict::GetVector
- ================
- */
- bool idDict::GetVector( const char *key, const char *defaultString, idVec3 &out ) const {
- bool found;
- const char *s;
-
- if ( !defaultString ) {
- defaultString = "0 0 0";
- }
- found = GetString( key, defaultString, &s );
- out.Zero();
- sscanf( s, "%f %f %f", &out.x, &out.y, &out.z );
- return found;
- }
- /*
- ================
- idDict::GetVec2
- ================
- */
- bool idDict::GetVec2( const char *key, const char *defaultString, idVec2 &out ) const {
- bool found;
- const char *s;
-
- if ( !defaultString ) {
- defaultString = "0 0";
- }
- found = GetString( key, defaultString, &s );
- out.Zero();
- sscanf( s, "%f %f", &out.x, &out.y );
- return found;
- }
- /*
- ================
- idDict::GetVec4
- ================
- */
- bool idDict::GetVec4( const char *key, const char *defaultString, idVec4 &out ) const {
- bool found;
- const char *s;
-
- if ( !defaultString ) {
- defaultString = "0 0 0 0";
- }
- found = GetString( key, defaultString, &s );
- out.Zero();
- sscanf( s, "%f %f %f %f", &out.x, &out.y, &out.z, &out.w );
- return found;
- }
- /*
- ================
- idDict::GetMatrix
- ================
- */
- bool idDict::GetMatrix( const char *key, const char *defaultString, idMat3 &out ) const {
- const char *s;
- bool found;
-
- if ( !defaultString ) {
- defaultString = "1 0 0 0 1 0 0 0 1";
- }
- found = GetString( key, defaultString, &s );
- out.Identity(); // sccanf has a bug in it on Mac OS 9. Sigh.
- sscanf( s, "%f %f %f %f %f %f %f %f %f", &out[0].x, &out[0].y, &out[0].z, &out[1].x, &out[1].y, &out[1].z, &out[2].x, &out[2].y, &out[2].z );
- return found;
- }
- /*
- ================
- WriteString
- ================
- */
- static void WriteString( const char *s, idFile *f ) {
- int len = strlen( s );
- if ( len >= MAX_STRING_CHARS-1 ) {
- idLib::common->Error( "idDict::WriteToFileHandle: bad string" );
- }
- f->Write( s, strlen(s) + 1 );
- }
- /*
- ================
- idDict::FindKey
- ================
- */
- const idKeyValue *idDict::FindKey( const char *key ) const {
- int i, hash;
- if ( key == NULL || key[0] == '\0' ) {
- idLib::common->DWarning( "idDict::FindKey: empty key" );
- return NULL;
- }
- hash = argHash.GenerateKey( key, false );
- for ( i = argHash.First( hash ); i != -1; i = argHash.Next( i ) ) {
- if ( args[i].GetKey().Icmp( key ) == 0 ) {
- return &args[i];
- }
- }
- return NULL;
- }
- /*
- ================
- idDict::FindKeyIndex
- ================
- */
- int idDict::FindKeyIndex( const char *key ) const {
- if ( key == NULL || key[0] == '\0' ) {
- idLib::common->DWarning( "idDict::FindKeyIndex: empty key" );
- return 0;
- }
- int hash = argHash.GenerateKey( key, false );
- for ( int i = argHash.First( hash ); i != -1; i = argHash.Next( i ) ) {
- if ( args[i].GetKey().Icmp( key ) == 0 ) {
- return i;
- }
- }
- return -1;
- }
- /*
- ================
- idDict::Delete
- ================
- */
- void idDict::Delete( const char *key ) {
- int hash, i;
- hash = argHash.GenerateKey( key, false );
- for ( i = argHash.First( hash ); i != -1; i = argHash.Next( i ) ) {
- if ( args[i].GetKey().Icmp( key ) == 0 ) {
- globalKeys.FreeString( args[i].key );
- globalValues.FreeString( args[i].value );
- args.RemoveIndex( i );
- argHash.RemoveIndex( hash, i );
- break;
- }
- }
- #if 0
- // make sure all keys can still be found in the hash index
- for ( i = 0; i < args.Num(); i++ ) {
- assert( FindKey( args[i].GetKey() ) != NULL );
- }
- #endif
- }
- /*
- ================
- idDict::MatchPrefix
- ================
- */
- const idKeyValue *idDict::MatchPrefix( const char *prefix, const idKeyValue *lastMatch ) const {
- int i;
- int len;
- int start;
- assert( prefix );
- len = strlen( prefix );
- start = -1;
- if ( lastMatch ) {
- start = args.FindIndex( *lastMatch );
- assert( start >= 0 );
- if ( start < 1 ) {
- start = 0;
- }
- }
- for( i = start + 1; i < args.Num(); i++ ) {
- if ( !args[i].GetKey().Icmpn( prefix, len ) ) {
- return &args[i];
- }
- }
- return NULL;
- }
- /*
- ================
- idDict::RandomPrefix
- ================
- */
- const char *idDict::RandomPrefix( const char *prefix, idRandom &random ) const {
- int count;
- const int MAX_RANDOM_KEYS = 2048;
- const char *list[MAX_RANDOM_KEYS];
- const idKeyValue *kv;
- list[0] = "";
- for ( count = 0, kv = MatchPrefix( prefix ); kv != NULL && count < MAX_RANDOM_KEYS; kv = MatchPrefix( prefix, kv ) ) {
- list[count++] = kv->GetValue().c_str();
- }
- return list[random.RandomInt( count )];
- }
- /*
- ================
- idDict::WriteToFileHandle
- ================
- */
- void idDict::WriteToFileHandle( idFile *f ) const {
- int c = LittleLong( args.Num() );
- f->Write( &c, sizeof( c ) );
- for ( int i = 0; i < args.Num(); i++ ) { // don't loop on the swapped count use the original
- WriteString( args[i].GetKey().c_str(), f );
- WriteString( args[i].GetValue().c_str(), f );
- }
- }
- /*
- ================
- ReadString
- ================
- */
- static idStr ReadString( idFile *f ) {
- char str[MAX_STRING_CHARS];
- int len;
- for ( len = 0; len < MAX_STRING_CHARS; len++ ) {
- f->Read( (void *)&str[len], 1 );
- if ( str[len] == 0 ) {
- break;
- }
- }
- if ( len == MAX_STRING_CHARS ) {
- idLib::common->Error( "idDict::ReadFromFileHandle: bad string" );
- }
- return idStr( str );
- }
- /*
- ================
- idDict::ReadFromFileHandle
- ================
- */
- void idDict::ReadFromFileHandle( idFile *f ) {
- int c;
- idStr key, val;
- Clear();
- f->Read( &c, sizeof( c ) );
- c = LittleLong( c );
- for ( int i = 0; i < c; i++ ) {
- key = ReadString( f );
- val = ReadString( f );
- Set( key, val );
- }
- }
- /*
- ========================
- idDict::Serialize
- ========================
- */
- void idDict::Serialize( idSerializer & ser ) {
- if ( ser.IsReading() ) {
- Clear();
- }
- int num = args.Num();
- ser.SerializePacked( num );
- for ( int i = 0; i < num; i++ ) {
- idStr key;
- idStr val;
- if ( ser.IsWriting() ) {
- key = args[i].GetKey();
- val = args[i].GetValue();
- }
- ser.SerializeString( key );
- ser.SerializeString( val );
- if ( ser.IsReading() ) {
- Set( key.c_str(), val.c_str() );
- }
- }
- }
- /*
- ================
- idDict::WriteToIniFile
- ================
- */
- void idDict::WriteToIniFile( idFile * f ) const {
- // make a copy so we don't affect the checksum of the original dict
- idList< idKeyValue > sortedArgs( args );
- sortedArgs.SortWithTemplate( idSort_KeyValue() );
- idList< idStr > prefixList;
- idTempArray< int > prefixIndex( sortedArgs.Num() ); // for each keyValue in the args, this is an index into which prefix it uses.
- // 0 means no prefix, otherwise, it's an index + (-1) into prefixList
- // we do this so we can print all the non-prefix based pairs first
- idStr prevPrefix = "";
- idStr skipFirstLine = "";
- // Scan for all the prefixes
- for ( int i = 0; i < sortedArgs.Num(); i++ ) {
- const idKeyValue * kv = &sortedArgs[i];
- int slashPosition = kv->GetKey().Last( '/' );
- if ( slashPosition != idStr::INVALID_POSITION ) {
- idStr prefix = kv->GetKey().Mid( 0, slashPosition );
- if ( prefix != prevPrefix ) {
- prevPrefix = prefix;
- prefixList.Append( prefix );
- }
- prefixIndex[i] = prefixList.Num();
- } else {
- prefixIndex[i] = 0;
- // output all the prefix-less first
- idStr str = va( "%s=%s\n", kv->GetKey().c_str(), idStr::CStyleQuote( kv->GetValue() ) );
- f->Write( (void *)str.c_str(), str.Length() );
- skipFirstLine = "\n";
- }
- }
- int prevPrefixIndex = 0;
- int prefixLength = 0;
- // output all the rest without their prefix
- for ( int i = 0; i < sortedArgs.Num(); i++ ) {
- if ( prefixIndex[i] == 0 ) {
- continue;
- }
- if ( prefixIndex[i] != prevPrefixIndex ) {
- prevPrefixIndex = prefixIndex[i];
- prefixLength = prefixList[prevPrefixIndex - 1].Length() + 1; // to skip past the '/' too
- // output prefix
- idStr str = va( "%s[%s]\n", skipFirstLine.c_str(), prefixList[prevPrefixIndex - 1].c_str() );
- f->Write( (void *)str.c_str(), str.Length() );
- }
- const idKeyValue * kv = &sortedArgs[i];
- idStr str = va( "%s=%s\n", kv->GetKey().c_str() + prefixLength, idStr::CStyleQuote( kv->GetValue() ) );
- f->Write( (void *)str.c_str(), str.Length() );
- }
- }
- /*
- ================
- idDict::ReadFromIniFile
- ================
- */
- bool idDict::ReadFromIniFile( idFile * f ) {
- int length = f->Length();
- idTempArray< char > buffer( length );
- if ( (int)f->Read( buffer.Ptr(), length ) != length ) {
- return false;
- }
- buffer[length-1] = NULL; // Since the .ini files are not null terminated, make sure we mark where the end of the .ini file is in our read buffer
- idLexer parser( LEXFL_NOFATALERRORS | LEXFL_ALLOWPATHNAMES /*| LEXFL_ONLYSTRINGS */);
- idStr name = f->GetName();
- name.Append( " dictionary INI reader" );
- if ( !parser.LoadMemory( ( const char* )buffer.Ptr(), length, name.c_str() ) ) {
- return false;
- }
- idToken token;
- idToken token2;
- idStr prefix = "";
- idStr valueStr;
- bool success = true;
- Clear();
- const punctuation_t ini_punctuations[] = {
- { "[", P_SQBRACKETOPEN },
- { "]", P_SQBRACKETCLOSE },
- { "=", P_ASSIGN },
- { NULL, 0 }
- };
- parser.SetPunctuations( ini_punctuations );
- while ( success && !parser.EndOfFile() ) {
- if ( parser.PeekTokenType( TT_PUNCTUATION, P_SQBRACKETOPEN, &token ) ) {
- success = success && parser.ExpectTokenType( TT_PUNCTUATION, P_SQBRACKETOPEN, &token );
- success = success && parser.ReadToken( &token );
- prefix = token.c_str();
- prefix.Append( '/' );
- success = success && parser.ExpectTokenType( TT_PUNCTUATION, P_SQBRACKETCLOSE, &token );
- }
-
- if ( !parser.PeekTokenType( TT_NAME, 0, &token ) ) {
- // end of file most likely
- break;
- }
- success = success && parser.ExpectTokenType( TT_NAME, 0, &token );
- success = success && parser.ExpectTokenType( TT_PUNCTUATION, P_ASSIGN, &token2 );
- success = success && ( parser.ParseRestOfLine( valueStr ) != NULL );
- valueStr = idStr::CStyleUnQuote( valueStr );
- idStr key = va( "%s%s", prefix.c_str(), token.c_str() );
- if ( FindKey( key.c_str() ) ) {
- parser.Warning( "'%s' already defined", key.c_str() );
- }
- Set( key.c_str(), valueStr.c_str() );
- }
- return success;
- }
- CONSOLE_COMMAND( TestDictIniFile, "Tests the writing/reading of various items in a dict to/from an ini file", 0 ) {
- // Write to the file
- idFile * file = fileSystem->OpenFileWrite( "idDict_ini_test.ini" );
- if ( file == NULL ) {
- idLib::Printf( "[^1FAILED^0] Couldn't open file for writing.\n" );
- return;
- }
- idDict vars;
- vars.SetInt( "section1/section3/a", -1 );
- vars.SetInt( "section1/section3/b", 0 );
- vars.SetInt( "section1/section3/c", 3 );
- vars.SetFloat( "section2/d", 4.0f );
- vars.SetFloat( "section2/e", -5.0f );
- vars.SetBool( "section2/f", true );
- vars.SetBool( "section1/g", false );
- vars.Set( "section1/h", "test1" );
- vars.Set( "i", "1234" );
- vars.SetInt( "j", 9 );
- vars.WriteToIniFile( file );
- delete file;
- // Read from the file
- file = fileSystem->OpenFileRead( "idDict_ini_test.ini" );
- if ( file == NULL ) {
- idLib::Printf( "[^1FAILED^0] Couldn't open file for reading.\n" );
- }
- idDict readVars;
- readVars.ReadFromIniFile( file );
- delete file;
- if ( vars.Checksum() != readVars.Checksum() ) {
- idLib::Printf( "[^1FAILED^0] Dictionaries do not match.\n" );
- } else {
- idLib::Printf( "[^2PASSED^0] Dictionaries match.\n" );
- }
- // Output results
- for ( int i = 0; i < readVars.GetNumKeyVals(); i++ ) {
- const idKeyValue * kv = readVars.GetKeyVal( i );
- idLib::Printf( "%s=%s\n", kv->GetKey().c_str(), kv->GetValue().c_str() );
- }
- }
- /*
- ================
- idDict::Init
- ================
- */
- void idDict::Init() {
- globalKeys.SetCaseSensitive( false );
- globalValues.SetCaseSensitive( true );
- }
- /*
- ================
- idDict::Shutdown
- ================
- */
- void idDict::Shutdown() {
- globalKeys.Clear();
- globalValues.Clear();
- }
- /*
- ================
- idDict::ShowMemoryUsage_f
- ================
- */
- void idDict::ShowMemoryUsage_f( const idCmdArgs &args ) {
- idLib::common->Printf( "%5d KB in %d keys\n", globalKeys.Size() >> 10, globalKeys.Num() );
- idLib::common->Printf( "%5d KB in %d values\n", globalValues.Size() >> 10, globalValues.Num() );
- }
- /*
- ================
- idDict::ListKeys_f
- ================
- */
- void idDict::ListKeys_f( const idCmdArgs &args ) {
- idLib::Printf( "Not implemented due to sort impl issues.\n" );
- //int i;
- //idList<const idPoolStr *> keyStrings;
- //for ( i = 0; i < globalKeys.Num(); i++ ) {
- // keyStrings.Append( globalKeys[i] );
- //}
- //keyStrings.SortWithTemplate( idSort_PoolStrPtr() );
- //for ( i = 0; i < keyStrings.Num(); i++ ) {
- // idLib::common->Printf( "%s\n", keyStrings[i]->c_str() );
- //}
- //idLib::common->Printf( "%5d keys\n", keyStrings.Num() );
- }
- /*
- ================
- idDict::ListValues_f
- ================
- */
- void idDict::ListValues_f( const idCmdArgs &args ) {
- idLib::Printf( "Not implemented due to sort impl issues.\n" );
- //int i;
- //idList<const idPoolStr *> valueStrings;
- //for ( i = 0; i < globalValues.Num(); i++ ) {
- // valueStrings.Append( globalValues[i] );
- //}
- //valueStrings.SortWithTemplate( idSort_PoolStrPtr() );
- //for ( i = 0; i < valueStrings.Num(); i++ ) {
- // idLib::common->Printf( "%s\n", valueStrings[i]->c_str() );
- //}
- //idLib::common->Printf( "%5d values\n", valueStrings.Num() );
- }
|