| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393 | //-----------------------------------------------------------------------------// Copyright (c) 2012 GarageGames, LLC//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions://// The above copyright notice and this permission notice shall be included in// all copies or substantial portions of the Software.//// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS// IN THE SOFTWARE.//-----------------------------------------------------------------------------#include "persistenceManager.h"#include "console/simSet.h"#include "console/consoleTypes.h"#include "core/stream/fileStream.h"#include "gui/core/guiTypes.h"#include "materials/customMaterialDefinition.h"#include "ts/tsShapeConstruct.h"#include "sim/netStringTable.h"IMPLEMENT_CONOBJECT(PersistenceManager);ConsoleDocClass( PersistenceManager,				"@brief this class manages updating SimObjects in the file they were "				"created in non-destructively (mostly aimed at datablocks and materials).\n\n"				"Basic scripting interface:\n\n"				"	- Creation: new PersistenceManager(FooManager);\n"				"	- Flag objects as dirty: FooManager.setDirty(<object name or id>);\n"				"	- Remove objects from dirty list: FooManager.removeDirty(<object name or id>);\n"				"	- List all currently dirty objects: FooManager.listDirty();\n"				"	- Check to see if an object is dirty: FooManager.isDirty(<object name or id>);\n"				"	- Save dirty objects to their files: FooManager.saveDirty();\n\n"				"@note Dirty objects don't update their files until saveDirty() is "				"called so you can change their properties after you flag them as dirty\n\n"				"@note Currently only used by editors, not intended for actual game development\n\n"				"@ingroup Console\n"				"@ingroup Editors\n"				"@internal");PersistenceManager::PersistenceManager(){   mCurrentObject = NULL;   mCurrentFile = NULL;   VECTOR_SET_ASSOCIATION(mLineBuffer);   mLineBuffer.reserve(2048);}PersistenceManager::~PersistenceManager(){   mDirtyObjects.clear();}bool PersistenceManager::onAdd(){   if (!Parent::onAdd())      return false;   return true;}void PersistenceManager::onRemove(){   Parent::onRemove();}void PersistenceManager::clearLineBuffer(){   for (U32 i = 0; i < mLineBuffer.size(); i++)   {      dFree( mLineBuffer[ i ] );      mLineBuffer[ i ] = NULL;   }   mLineBuffer.clear();}void PersistenceManager::deleteObject(ParsedObject* object){   if (object)   {      // Clear up used property memory      for (U32 j = 0; j < object->properties.size(); j++)      {         ParsedProperty& prop = object->properties[j];         if (prop.value)         {            dFree( prop.value );            prop.value = NULL;         }      }      object->properties.clear();      // Delete the parsed object      SAFE_DELETE(object);   }}void PersistenceManager::clearObjects(){   // Clean up the object buffer   for (U32 i = 0; i < mObjectBuffer.size(); i++)      deleteObject(mObjectBuffer[i]);   mObjectBuffer.clear();   // We shouldn't have anything in the object stack   // but let's clean it up just in case   // Clean up the object buffer   for (U32 i = 0; i < mObjectStack.size(); i++)      deleteObject(mObjectStack[i]);   mObjectStack.clear();   // Finally make sure there isn't a current object   deleteObject(mCurrentObject);}void PersistenceManager::clearFileData(){   // Clear the active file name   if (mCurrentFile)   {      dFree( mCurrentFile );      mCurrentFile = NULL;   }   // Clear the file objects   clearObjects();   // Clear the line buffer   clearLineBuffer();   // Clear the tokenizer data   mParser.clear();}void PersistenceManager::clearAll(){   // Clear the file data in case it hasn't cleared yet   clearFileData();   // Clear the dirty object list   mDirtyObjects.clear();   // Clear the remove field list   mRemoveFields.clear();}bool PersistenceManager::readFile(const char* fileName){   // Clear our previous file buffers just in   // case saveDirtyFile() didn't catch it   clearFileData();   // Handle an object writing out to a new file   if ( !Torque::FS::IsFile( fileName ) )   {      // Set our current file      mCurrentFile = dStrdup(fileName);      return true;   }   // Try to open the file   FileStream  stream;   stream.open( fileName, Torque::FS::File::Read );   if ( stream.getStatus() != Stream::Ok )   {      Con::errorf("PersistenceManager::readFile() - Failed to open %s", fileName);      return false;   }   // The file is good so read it in   mCurrentFile = dStrdup(fileName);   while(stream.getStatus() != Stream::EOS)   {      U8* buffer = ( U8* ) dMalloc( 2048 );      dMemset(buffer, 0, 2048);      stream.readLine(buffer, 2048);      mLineBuffer.push_back((const char*)buffer);   }   // Because of the way that writeLine() works we need to   // make sure we don't have an empty last line or else   // we will get an extra line break   if (mLineBuffer.size() > 0)   {      if (mLineBuffer.last() && mLineBuffer.last()[0] == 0)      {         dFree(mLineBuffer.last());         mLineBuffer.pop_back();      }   }   stream.close();   //Con::printf("Successfully opened and read %s", mCurrentFile);   return true;}void PersistenceManager::killObject(){   // Don't save this object   SAFE_DELETE(mCurrentObject);   // If there is an object in the stack restore it   if (mObjectStack.size() > 0)   {      mCurrentObject = mObjectStack.last();      mObjectStack.pop_back();   }}void PersistenceManager::saveObject(){   // Now that we have all of the data attempt to   // find the corresponding SimObject   mCurrentObject->simObject = Sim::findObject(mCurrentFile, mCurrentObject->endLine + 1);   // Save this object   mObjectBuffer.push_back(mCurrentObject);   mCurrentObject = NULL;   // If there is an object in the stack restore it   if (mObjectStack.size() > 0)   {      mCurrentObject = mObjectStack.last();      mObjectStack.pop_back();   }}void PersistenceManager::parseObject(){   // We *should* already be in position but just in case...   if (!mParser.tokenICmp("new") &&       !mParser.tokenICmp("singleton") &&       !mParser.tokenICmp("datablock"))   {      Con::errorf("PersistenceManager::parseObject() - handed a position that doesn't point to an object \         creation keyword (new, singleton, datablock)");      return;   }   // If there is an object already being parsed then   // push it into the stack to finish later   if (mCurrentObject)      mObjectStack.push_back(mCurrentObject);   mCurrentObject = new ParsedObject;   //// If this object declaration is being assigned to a variable then   //// consider that the "start" of the declaration (otherwise we could   //// get a script compile error if we delete the object declaration)   mParser.regressToken(true);   if (mParser.tokenICmp("="))   {      // Ok, we are at an '='...back up to the beginning of that variable      mParser.regressToken(true);      // Get the startLine and startPosition      mCurrentObject->startLine = mParser.getCurrentLine();      mCurrentObject->startPosition = mParser.getTokenLineOffset();      // Advance back to the object declaration      mParser.advanceToken(true);      mParser.advanceToken(true);   }   else   {      // Advance back to the object declaration      mParser.advanceToken(true);      // Get the startLine and startPosition      mCurrentObject->startLine = mParser.getCurrentLine();      mCurrentObject->startPosition = mParser.getTokenLineOffset();   }   if (mObjectStack.size() > 0)      mCurrentObject->parentObject = mObjectStack.last();   // The next token should be the className   mCurrentObject->className = StringTable->insert(mParser.getNextToken());   // Advance to '('   mParser.advanceToken(true);   if (!mParser.tokenICmp("("))   {      Con::errorf("PersistenceManager::parseObject() - badly formed object \         declaration on line %d - was expecting a '(' character", mParser.getCurrentLine()+1);      // Remove this object without saving it      killObject();      return;   }   // The next token should either be the object name or ')'   mParser.advanceToken(true);   if (mParser.tokenICmp(")"))   {      mCurrentObject->name = StringTable->insert("");      mCurrentObject->nameLine = mParser.getCurrentLine();      mCurrentObject->namePosition = mParser.getTokenLineOffset();   }   else   {      mCurrentObject->name = StringTable->insert(mParser.getToken());      mCurrentObject->nameLine = mParser.getCurrentLine();      mCurrentObject->namePosition = mParser.getTokenLineOffset();      // Advance to either ')' or ':'      mParser.advanceToken(true);      if (mParser.tokenICmp(":"))      {         // Advance past the object we are copying from         mParser.advanceToken(true);         // Advance to ')'         mParser.advanceToken(true);      }      if (!mParser.tokenICmp(")"))      {         Con::errorf("PersistenceManager::parseObject() - badly formed object \            declaration on line %d - was expecting a ')' character", mParser.getCurrentLine()+1);         // Remove this object without saving it         killObject();         return;      }   }   // The next token should either be a ';' or a '{'   mParser.advanceToken(true);   if (mParser.tokenICmp(";"))   {      // Save the end line number      mCurrentObject->endLine = mParser.getCurrentLine();      // Save the end position      mCurrentObject->endPosition = mParser.getTokenLineOffset();      // Flag this object as not having braces      mCurrentObject->hasBraces = false;      saveObject(); // Object has no fields      return;   }   else if (!mParser.tokenICmp("{"))   {      Con::errorf("PersistenceManager::parseObject() - badly formed object \         declaration on line %d - was expecting a '{' character", mParser.getCurrentLine()+1);      // Remove this object without saving it      killObject();      return;   }   while (mParser.advanceToken(true))   {      // Check for a subobject      if (mParser.tokenICmp("new") ||          mParser.tokenICmp("singleton") ||          mParser.tokenICmp("datablock"))      {         parseObject();      }      // Check to see if we have a property      if (mParser.tokenICmp("="))      {         // Ok, we are at an '='...back up to find out         // what variable is getting assigned         mParser.regressToken(true);         const char* variable = mParser.getToken();         if (variable && dStrlen(variable) > 0)         {            // See if it is a global or a local variable            if (variable[0] == '%' || variable[0] == '$')            {               // We ignore this variable and go               // back to our previous place               mParser.advanceToken(true);            }            // Could also potentially be a <object>.<variable>            // assignment which we don't care about either            else if (dStrchr(variable, '.'))            {               // We ignore this variable and go               // back to our previous place               mParser.advanceToken(true);            }            // If we made it to here assume it is a variable            // for the current object            else            {               // Create our new property               mCurrentObject->properties.increment();               ParsedProperty& prop = mCurrentObject->properties.last();               // Check to see if this is an array variable               if (dStrlen(variable) > 3 && variable[dStrlen(variable) - 1] == ']')               {                  // The last character is a ']' which *should* mean                  // there is also a corresponding '['                  const char* arrayPosStart = dStrrchr(variable, '[');                  if (!arrayPosStart)                  {                     Con::errorf("PersistenceManager::parseObject() - error parsing array position - \                        was expecting a '[' character");                  }                  else                  {                     // Parse the array position for the variable name                     S32 arrayPos = -1;                     dSscanf(arrayPosStart, "[%d]", &arrayPos);                     // If we got a valid array position then set it                     if (arrayPos > -1)                        prop.arrayPos = arrayPos;                     // Trim off the [<pos>] from the variable name                     char* variableName = dStrdup(variable);                     variableName[arrayPosStart - variable] = 0;                     // Set the variable name to our new shortened name                     variable = StringTable->insert(variableName, true);                     // Cleanup our variableName buffer                     dFree( variableName );                  }               }               // Set back the variable name               prop.name = StringTable->insert(variable, true);               // Store the start position for this variable               prop.startLine     = mParser.getCurrentLine();               prop.startPosition = mParser.getTokenLineOffset();               // Advance back to the '='               mParser.advanceToken(true);               // Sanity check               if (!mParser.tokenICmp("="))                  Con::errorf("PersistenceManager::parseObject() - somehow we aren't \                               pointing at the expected '=' character");               else               {                  // The next token should be the value                  // being assigned to the variable                  mParser.advanceToken(true);                  // Store the line number for this value                  prop.valueLine = mParser.getCurrentLine();                  // Store the values beginning position                  prop.valuePosition = mParser.getTokenLineOffset();                  // Read tokens up to the semicolon.                  // Quoted tokens skip the leading and trailing quote characters. eg.                  // "this" becomes: this                  // "this" TAB "that" becomes: this" TAB "that                  // "this" TAB "that" TAB "other" becomes: this" TAB "that" TAB "other                  String value;                  bool wasQuoted = false;                  while (!mParser.endOfFile() && !mParser.tokenICmp(";"))                  {                     // Join tokens together (skipped first time through when string is empty)                     if (value.length() > 0)                     {                        if (wasQuoted)                           value += "\" ";                  // quoted followed by non-quoted                        else if (mParser.tokenIsQuoted())                           value += " \"";                  // non-quoted followed by quoted                        else                           value += " ";                    // non-quoted followed by non-quoted                     }                     value += mParser.getToken();                     wasQuoted = mParser.tokenIsQuoted();                     mParser.advanceToken(true);                  }                  // TODO: make sure this doesn't leak                  prop.value = dStrdup(value.c_str());                  if (!mParser.tokenICmp(";"))                     Con::errorf("PersistenceManager::parseObject() - badly formed variable "                                  "assignment on line %d - was expecting a ';' character", mParser.getCurrentLine()+1);                  // Store the end position for this variable                  prop.endLine     = mParser.getCurrentLine();                  prop.endPosition = mParser.getTokenLineOffset();                  if (wasQuoted)                     prop.endPosition -= 1;               }            }         }      }      // Check for the end of the object declaration      if (mParser.tokenICmp("}"))      {         // See if the next token is a ';'         mParser.advanceToken(true);         if (mParser.tokenICmp(";"))         {            // Save the end line number            mCurrentObject->endLine = mParser.getCurrentLine();            // Save the end position            mCurrentObject->endPosition = mParser.getTokenLineOffset();            saveObject();            break;         }      }   }}bool PersistenceManager::parseFile(const char* fileName){   // Read the file into the line buffer   if (!readFile(fileName))      return false;   // Load it into our Tokenizer parser   if (!mParser.openFile(fileName))   {      // Handle an object writing out to a new file      if ( !Torque::FS::IsFile( fileName ) )         return true;      return false;   }   // Set our reserved "single" tokens   mParser.setSingleTokens("(){};=:");   // Search object declarations   while (mParser.advanceToken(true))   {      if (mParser.tokenICmp("new") ||          mParser.tokenICmp("singleton") ||          mParser.tokenICmp("datablock"))      {         parseObject();      }   }   // If we had an object that didn't end properly   // then we could have objects on the stack   while (mCurrentObject)      saveObject();   //Con::errorf("Parsed Results:");   //for (U32 i = 0; i < mObjectBuffer.size(); i++)   //{   //   ParsedObject* parsedObject = mObjectBuffer[i];   //   Con::warnf("   mObjectBuffer[%d]:", i);   //   Con::warnf("      name = %s",      parsedObject->name);   //   Con::warnf("      className = %s", parsedObject->className);   //   Con::warnf("      startLine = %d", parsedObject->startLine + 1);   //   Con::warnf("      endLine   = %d", parsedObject->endLine + 1);   //   //if (mObjectBuffer[i]->properties.size() > 0)   //   //{   //   //   Con::warnf("   properties:");   //   //   for (U32 j = 0; j < mObjectBuffer[i]->properties.size(); j++)   //   //      Con::warnf("      %s = %s;", mObjectBuffer[i]->properties[j].name,   //   //                                   mObjectBuffer[i]->properties[j].value);   //   //}   //   if (!parsedObject->simObject.isNull())   //   {   //      SimObject* simObject = parsedObject->simObject;   //      Con::warnf("      SimObject(%s) %d:", simObject->getName(), simObject->getId());   //      Con::warnf("         declaration line = %d", simObject->getDeclarationLine());   //   }   //}   return true;}S32 PersistenceManager::getPropertyIndex(ParsedObject* parsedObject, const char* fieldName, U32 arrayPos){   S32 propertyIndex = -1;   if (!parsedObject)      return propertyIndex;   for (U32 i = 0; i < parsedObject->properties.size(); i++)   {      if (dStricmp(fieldName, parsedObject->properties[i].name) == 0 &&          parsedObject->properties[i].arrayPos == arrayPos)      {         propertyIndex = i;         break;      }   }   return propertyIndex;}char* PersistenceManager::getObjectIndent(ParsedObject* object){   char* indent = Con::getReturnBuffer(2048);   indent[0] = 0;   if (!object)      return indent;   if (object->startLine < 0 || object->startLine >= mLineBuffer.size())      return indent;   const char* line = mLineBuffer[object->startLine];   if (line)   {      const char* nonSpace = line;      U32 strLen = dStrlen(line);      for (U32 i = 0; i < strLen; i++)      {         if (*nonSpace != ' ')            break;         nonSpace++;      }      dStrncpy(indent, line, nonSpace - line);      indent[nonSpace - line] = 0;   }   return indent;}void PersistenceManager::updatePositions(U32 lineNumber, U32 startPos, S32 diff){   if (diff == 0)      return;   for (U32 i = 0; i < mObjectBuffer.size(); i++)   {      ParsedObject* object = mObjectBuffer[i];      if (object->nameLine == lineNumber && object->namePosition > startPos)         object->namePosition += diff;      if (object->endLine == lineNumber && object->endPosition > startPos)         object->endPosition += diff;      if (lineNumber >= object->startLine && lineNumber <= object->endLine)      {         for (U32 j = 0; j < object->properties.size(); j++)         {            ParsedProperty& prop = object->properties[j];            S32 propStartPos = prop.startPosition;            S32 endPos       = prop.endPosition;            S32 valuePos     = prop.valuePosition;            if (lineNumber == prop.startLine && propStartPos > startPos)            {               propStartPos += diff;               if (propStartPos < 0)                  propStartPos = 0;               prop.startPosition = valuePos;            }            if (lineNumber == prop.endLine && endPos > startPos)            {               endPos += diff;               if (endPos < 0)                  endPos = 0;               prop.endPosition = endPos;            }            if (lineNumber == prop.valueLine && valuePos > startPos)            {               valuePos += diff;               if (valuePos < 0)                  valuePos = 0;               prop.valuePosition = valuePos;            }         }      }   }}void PersistenceManager::updateLineOffsets(U32 startLine, S32 diff, ParsedObject* skipObject){   if (diff == 0)      return;   if (startLine >= mLineBuffer.size())      return;   if (startLine + diff >= mLineBuffer.size())      return;   // Make sure we don't double offset a SimObject's   // declaration line   SimObjectList updated;   if (skipObject && !skipObject->simObject.isNull())      updated.push_back_unique(skipObject->simObject);   for (U32 i = 0; i < mObjectBuffer.size(); i++)   {      ParsedObject* object = mObjectBuffer[i];      // See if this is the skipObject      if (skipObject && skipObject == object)         continue;      // We can safely ignore objects that      // came earlier in the file      if (object->endLine < startLine)         continue;      if (object->startLine >= startLine)         object->startLine += diff;      if (object->nameLine >= startLine)         object->nameLine += diff;      for (U32 j = 0; j < object->properties.size(); j++)      {         if (object->properties[j].startLine >= startLine)            object->properties[j].startLine += diff;         if (object->properties[j].endLine >= startLine)            object->properties[j].endLine += diff;         if (object->properties[j].valueLine >= startLine)            object->properties[j].valueLine += diff;      }      if (object->endLine >= startLine)         object->endLine += diff;      if (!object->simObject.isNull() &&          object->simObject->getDeclarationLine() > startLine)      {         // Check for already updated SimObject's         U32 currSize = updated.size();         updated.push_back_unique(object->simObject);         if (updated.size() == currSize)            continue;         S32 newDeclLine = object->simObject->getDeclarationLine() + diff;         if (newDeclLine < 0)            newDeclLine = 0;         object->simObject->setDeclarationLine(newDeclLine);      }   }}PersistenceManager::ParsedObject* PersistenceManager::findParentObject(SimObject* object, ParsedObject* parentObject){   ParsedObject* ret = NULL;   if (!object)      return ret;   // First test for the SimGroup it belongs to   ret = findParsedObject(object->getGroup(), parentObject);   if (ret)      return ret;   // TODO: Test all of the SimSet's that this object belongs to   return ret;}PersistenceManager::ParsedObject* PersistenceManager::findParsedObject(SimObject* object, ParsedObject* parentObject){   if (!object)      return NULL;   // See if our object belongs to a parent   if (!parentObject)      parentObject = findParentObject(object, parentObject);   // First let's compare the object to the SimObject's that   // we matched to our ParsedObject's when we loaded them   for (U32 i = 0; i < mObjectBuffer.size(); i++)   {      ParsedObject* testObj = mObjectBuffer[i];      if (testObj->simObject == object)      {         // Deal with children objects         if (testObj->parentObject != parentObject)            continue;         return testObj;      }   }   // Didn't find it in our ParsedObject's SimObject's   // so see if we can find one that corresponds to the   // same name and className   const char *originalName = object->getOriginalName();   // Find the corresponding ParsedObject   if (originalName && originalName[0])   {      for (U32 i = 0; i < mObjectBuffer.size(); i++)      {         ParsedObject* testObj = mObjectBuffer[i];         if (testObj->name == originalName)         {            // Deal with children objects            if (testObj->parentObject != parentObject)               continue;            return testObj;         }      }   }   //Check internal names   if (object->getInternalName())   {      for (U32 i = 0; i < mObjectBuffer.size(); i++)      {         ParsedObject* testObj = mObjectBuffer[i];         for (U32 j = 0; j < testObj->properties.size(); j++)         {            const ParsedProperty &prop = testObj->properties[j];            if (	dStrcmp( prop.name, "internalName" ) == 0 &&                dStrcmp( prop.value, object->getInternalName() ) == 0 )               return testObj;            else if ( dStrcmp(prop.name, "internalName") == 0)               break;         }      }   }   return NULL;}void PersistenceManager::updateToken( const U32 lineNumber, const U32 linePosition, const U32 oldValueLen, const char* newValue, bool addQuotes ){   // Make sure we have a valid lineNumber   if (lineNumber < 0 || linePosition < 0 ||       lineNumber >= mLineBuffer.size())      return;   // Grab the line that the value is on   const char* line = mLineBuffer[lineNumber];   U32 newValueLen = ( newValue ) ? dStrlen(newValue) : 0;   // Make sure we have a valid linePosition   if (linePosition >= dStrlen(line) ||       linePosition + oldValueLen > dStrlen(line))      return;   // Get all of the characters up to the value position   U32 preStringLen = linePosition;   bool needQuotes = false;   if( addQuotes && line[ linePosition - 1 ] != '"' )   {      preStringLen ++;      needQuotes = true;   }   char* preString = ( char* ) dMalloc( preStringLen + 1 );   dMemcpy( preString, line, linePosition );   if( needQuotes )   {      preString[ linePosition ] = '"';      preString[ linePosition + 1 ] = 0;   }   else      preString[ linePosition ] = 0;   // Get all of the characters that occur after the value   const char* postStringSrc = line + linePosition + oldValueLen;   U32 postStringLen = dStrlen( postStringSrc );   if( needQuotes )      postStringLen ++;   char* postString = ( char* ) dMalloc( postStringLen + 1 );   if( needQuotes )      postString[ 0 ] = '"';   dStrcpy( &postString[ needQuotes ? 1 : 0 ], postStringSrc );   postString[ postStringLen ] = 0;   // Calculate the length of our new line   U32 newLineLen = 0;   newLineLen += preStringLen;   newLineLen += newValueLen;   newLineLen += postStringLen;   // Create a buffer for our new line and   // null terminate it   char* newLine = ( char* ) dMalloc( newLineLen + 1 );   newLine[0] = 0;   // Build the new line with the   // preString + newValue + postString   dStrcat(newLine, preString);   if ( newValue )      dStrcat(newLine, newValue);   dStrcat(newLine, postString);   // Clear our existing line   if (mLineBuffer[lineNumber])   {      dFree( mLineBuffer[ lineNumber ] );      mLineBuffer[ lineNumber ] = NULL;   }   // Set the new line   mLineBuffer[lineNumber] = newLine;   // Figure out the size difference of the old value   // and new value in case we need to update any else   // on the line after it   S32 diff = newValueLen - oldValueLen;   // Update anything that is on the line after this that needs   // to change its offsets to reflect the new line   updatePositions(lineNumber, linePosition, diff);   // Clean up our buffers   dFree( preString );   dFree( postString );}const char* PersistenceManager::getFieldValue(SimObject* object, const char* fieldName, U32 arrayPos){   // Our return string   char* ret = NULL;   // Buffer to hold the string equivalent of the arrayPos   char arrayPosStr[8];   dSprintf(arrayPosStr, 8, "%d", arrayPos);   // Get the object's value   const char *value = object->getDataField(fieldName, arrayPosStr );   if (value)      ret = dStrdup(value);   return ret;}const char* PersistenceManager::createNewProperty(const char* name, const char* value, bool isArray, U32 arrayPos){   if (!name || !value)      return NULL;   AssertFatal( value[0] != StringTagPrefixByte, "Got tagged string!" );   char* newProp = ( char* ) dMalloc( 2048 );   dMemset(newProp, 0, 2048);   if (value)   {      if (isArray)         dSprintf(newProp, 2048, "%s[%d] = \"%s\";", name, arrayPos, value);      else         dSprintf(newProp, 2048, "%s = \"%s\";", name, value);   }   else   {      if (isArray)         dSprintf(newProp, 2048, "%s[%d] = \"\";", name, arrayPos);      else         dSprintf(newProp, 2048, "%s = \"\";", name);   }   return newProp;}bool PersistenceManager::isEmptyLine(const char* line){   // Simple test first   if (!line || dStrlen(line) == 0)      return true;   U32 len = dStrlen(line);   for (U32 i = 0; i < len; i++)   {      const char& c = line[i];      // Skip "empty" characters      if (c == ' '  ||          c == '\t' ||          c == '\r' ||          c == '\n')      {         continue;      }      // If we have made it to the an end of the line      // comment then consider this an empty line      if (c == '/')      {         if (i < len - 1)         {            if (line[i + 1] == '/')               return true;         }      }      // If it isn't an "empty" character or a comment then      // we have a valid character on the line and it isn't empty      return false;   }   return true;}void PersistenceManager::removeLine(U32 lineNumber){   if (lineNumber >= mLineBuffer.size())      return;   if (mLineBuffer[lineNumber])   {      dFree( mLineBuffer[ lineNumber ] );      mLineBuffer[ lineNumber ] = NULL;   }   mLineBuffer.erase(lineNumber);   updateLineOffsets(lineNumber, -1);}void PersistenceManager::removeTextBlock(U32 startLine, U32 endLine, U32 startPos, U32 endPos, bool removeEmptyLines){   // Make sure we have valid lines   if (startLine >= mLineBuffer.size() || endLine >= mLineBuffer.size())      return;   // We assume that the startLine is before the endLine   if (startLine > endLine)      return;   // Grab the lines (they may be the same)   const char* startLineText = mLineBuffer[startLine];   const char* endLineText   = mLineBuffer[endLine];   // Make sure we have a valid startPos   if (startPos >= dStrlen(startLineText))      return;   // Make sure we have a valid endPos   if (endPos > dStrlen(endLineText))      return;   if (startLine == endLine)   {      // Now let updateToken do the heavy lifting on removing it      updateToken(startLine, startPos, endPos - startPos, "");      // Handle removing an empty lines if desired      if (removeEmptyLines)      {         const char* line = mLineBuffer[startLine];         if (isEmptyLine(line))            removeLine(startLine);      }   }   else   {      // Start with clearing the startLine from startPos to the end      updateToken(startLine, startPos, dStrlen(startLineText + startPos), "");      // Then clear the endLine from beginning to endPos      updateToken(endLine, 0, endPos, "");      // Handle removing an empty endLine if desired      if (removeEmptyLines)      {         const char* line = mLineBuffer[endLine];         if (isEmptyLine(line))            removeLine(endLine);      }      // Handle removing any lines between the startLine and endLine      for (U32 i = startLine + 1; i < endLine; i++)         removeLine(startLine + 1);      // Handle removing an empty startLine if desired      if (removeEmptyLines)      {         const char* line = mLineBuffer[startLine];         if (isEmptyLine(line))            removeLine(startLine);      }   }}void PersistenceManager::removeParsedObject(ParsedObject* parsedObject){   if (!parsedObject)      return;   if (parsedObject->startLine < 0 || parsedObject->startLine >= mLineBuffer.size())      return;   if (parsedObject->endLine < 0 || parsedObject->startLine >= mLineBuffer.size())      return;   removeTextBlock(parsedObject->startLine,     parsedObject->endLine,                   parsedObject->startPosition, parsedObject->endPosition+1, true); // +1 to remove trailing semicolon as well   parsedObject->parentObject = NULL;   parsedObject->simObject    = NULL;}void PersistenceManager::removeField(const ParsedProperty& prop){   if (prop.startLine < 0 || prop.startLine >= mLineBuffer.size())      return;   if (prop.endLine < 0 || prop.endLine >= mLineBuffer.size())      return;   S32 endPosition = prop.endPosition+1;                    // +1 to remove trailing semicolon as well   if ((endPosition < dStrlen(mLineBuffer[prop.endLine])) &&       (mLineBuffer[prop.endLine][endPosition] == ';'))     // adjust end position for quoted values (otherwise a trailing semicolon will remain)      endPosition++;  removeTextBlock(prop.startLine, prop.endLine, prop.startPosition, endPosition, true);}U32 PersistenceManager::writeProperties(const Vector<const char*>& properties, const U32 insertLine, const char* objectIndent){   U32 currInsertLine = insertLine;   for (U32 i = 0; i < properties.size(); i++)   {      const char* prop = properties[i];      if (!prop || dStrlen(prop) == 0)         continue;      U32 len = dStrlen(objectIndent) + dStrlen(prop) + 4;      char* newLine = ( char* ) dMalloc( len );      dSprintf(newLine, len, "%s   %s", objectIndent, prop);      mLineBuffer.insert(currInsertLine, newLine);      currInsertLine++;   }   return currInsertLine - insertLine;}PersistenceManager::ParsedObject* PersistenceManager::writeNewObject(SimObject* object, const Vector<const char*>& properties, const U32 insertLine, ParsedObject* parentObject){   ParsedObject* parsedObject = new ParsedObject;   parsedObject->name      = object->getName();   parsedObject->className = object->getClassName();   parsedObject->simObject = object;   U32 currInsertLine = insertLine;   // If the parentObject isn't set see if   // we can find it in the file   if (!parentObject)      parentObject = findParentObject(object);   parsedObject->parentObject = parentObject;   char* indent = getObjectIndent(parentObject);   if (parentObject)      dStrcat(indent, "   \0");   // Write out the beginning of the object declaration   const char* dclToken = "new";   if (dynamic_cast<Material*>(object) ||       dynamic_cast<CustomMaterial*>(object) ||       dynamic_cast<GuiControlProfile*>(object) ||       dynamic_cast<TSShapeConstructor*>(object))      dclToken = "singleton";   else if( dynamic_cast< SimDataBlock* >( object ) )   {      SimDataBlock* db = static_cast<SimDataBlock*>(object);      if( db->isClientOnly() )      {         if( db->getName() && db->getName()[ 0 ] )            dclToken = "singleton";      }      else         dclToken = "datablock";   }   char newLine[ 4096 ];   dMemset(newLine, 0, sizeof( newLine));   // New line before an object declaration   dSprintf(newLine, sizeof( newLine ), "");   mLineBuffer.insert(currInsertLine, dStrdup(newLine));   currInsertLine++;   dMemset(newLine, 0, sizeof( newLine ));   parsedObject->startLine    = currInsertLine;   parsedObject->nameLine     = currInsertLine;   parsedObject->namePosition = dStrlen(indent) + dStrlen(dclToken) + dStrlen(object->getClassName()) + 2;   // Objects that had no name were getting saved out as: Object((null))   if ( object->getName() != NULL )   {      if( object->getCopySource() )         dSprintf(newLine, sizeof( newLine ), "%s%s %s(%s : %s)", indent, dclToken, object->getClassName(), object->getName(),            object->getCopySource() ? object->getCopySource()->getName() : "" );      else         dSprintf(newLine, sizeof( newLine ), "%s%s %s(%s)", indent, dclToken, object->getClassName(), object->getName());   }   else      dSprintf(newLine, sizeof( newLine ), "%s%s %s()", indent, dclToken, object->getClassName() );   mLineBuffer.insert(currInsertLine, dStrdup(newLine));   currInsertLine++;   dMemset(newLine, 0, sizeof( newLine ));   dSprintf(newLine, sizeof( newLine ), "%s{", indent);   mLineBuffer.insert(currInsertLine, dStrdup(newLine));   currInsertLine++;   dMemset(newLine, 0, sizeof( newLine ));   currInsertLine += writeProperties(properties, currInsertLine, indent);   parsedObject->endLine = currInsertLine;   parsedObject->updated = true;   dSprintf(newLine, sizeof( newLine ), "%s};", indent);   mLineBuffer.insert(currInsertLine, dStrdup(newLine));   currInsertLine++;   updateLineOffsets(insertLine, currInsertLine - insertLine, parsedObject);   mObjectBuffer.push_back(parsedObject);   // Update the SimObject to reflect its saved name and declaration line.      // These values should always reflect what is in the file, even if the object   // has not actually been re-created from an execution of that file yet.   object->setOriginalName( object->getName() );   object->setDeclarationLine( currInsertLine );      if (mCurrentFile)      object->setFilename(mCurrentFile);   return parsedObject;}void PersistenceManager::updateObject(SimObject* object, ParsedObject* parentObject){   // Create a default object of the same type   ConsoleObject *defaultConObject = ConsoleObject::create(object->getClassName());   SimObject* defaultObject = dynamic_cast<SimObject*>(defaultConObject);      // ***Really*** shouldn't happen   if (!defaultObject)      return;   Vector<const char*> newLines;   ParsedObject* parsedObject = findParsedObject(object, parentObject);   // If we don't already have an association between the ParsedObject   // and the SimObject then go ahead and create it   if (parsedObject && parsedObject->simObject.isNull())      parsedObject->simObject = object;         // Kill all fields on the remove list.      for( U32 i = 0; i < mRemoveFields.size(); ++ i )   {      RemoveField& field = mRemoveFields[ i ];      if( field.object != object )         continue;               S32 propertyIndex = getPropertyIndex( parsedObject, field.fieldName, field.arrayPos );      if( propertyIndex != -1 )         removeField( parsedObject->properties[ propertyIndex ] );   }   // Get our field list   const AbstractClassRep::FieldList &list = object->getFieldList();   for(U32 i = 0; i < list.size(); i++)   {      const AbstractClassRep::Field* f = &list[i];      // Skip the special field types.      if ( f->type >= AbstractClassRep::ARCFirstCustomField )         continue;      for(U32 j = 0; S32(j) < f->elementCount; j++)      {         const char* value = getFieldValue(object, f->pFieldname, j);         // Make sure we got a value         if (!value)            continue;         // Let's see if this field is already in the file         S32 propertyIndex = getPropertyIndex(parsedObject, f->pFieldname, j);         if (propertyIndex > -1)         {            ParsedProperty& prop = parsedObject->properties[propertyIndex];            // If this field is on the remove list then remove it and continue            if (findRemoveField(object, f->pFieldname, j) || !object->writeField(f->pFieldname, value))            {               removeField( parsedObject->properties[ propertyIndex ] );               dFree( value );               continue;            }            // Run the parsed value through the console system conditioners so            // that it will better match the data we got back from the object.            const char* evalue = Con::getFormattedData(f->type, prop.value, f->table, f->flag);            // If our data doesn't match then we get to update it.            //            // As for copy-sources, we just assume here that if a property setting            // is there in the file, the user does not want it inherited from the copy-source            // even in the case the actual values are identical.                        if( dStricmp(value, evalue) != 0 )            {               if( value[ 0 ] == '\0' &&                   dStricmp( getFieldValue( defaultObject, f->pFieldname, j ), value ) == 0 &&                   ( !object->getCopySource() || dStricmp( getFieldValue( object->getCopySource(), f->pFieldname, j ), value ) == 0 ) )               {                  removeField( prop );               }               else               {                  // TODO: This should be wrapped in a helper method... probably.                  // Detect and collapse relative path information                  if (f->type == TypeFilename ||                     f->type == TypeStringFilename ||                     f->type == TypeImageFilename ||                     f->type == TypePrefabFilename ||                     f->type == TypeShapeFilename)                  {                     char fnBuf[1024];                     Con::collapseScriptFilename(fnBuf, 1024, value);                     updateToken(prop.valueLine, prop.valuePosition, prop.endPosition - prop.valuePosition, fnBuf, true);                  }                  else                     updateToken(prop.valueLine, prop.valuePosition, prop.endPosition - prop.valuePosition, value, true);               }            }         }         else         {            // No need to process a removed field that doesn't exist in the file            if (findRemoveField(object, f->pFieldname, j) || !object->writeField(f->pFieldname, value))            {               dFree( value );               continue;            }                        bool mustUpdate = false;            // If we didn't find the property in the ParsedObject            // then we need to compare against the default value            // for this property and save it out if it is different            const char* defaultValue = getFieldValue(defaultObject, f->pFieldname, j);            if( !defaultValue || dStricmp( value, defaultValue ) != 0 )            {               // Value differs.  Check whether it also differs from the               // value in the copy source if there is one.                              if( object->getCopySource() )               {                  const char* copySourceValue = getFieldValue( object->getCopySource(), f->pFieldname, j );                  if( !copySourceValue || dStricmp( copySourceValue, value ) != 0 )                     mustUpdate = true;                                       if( copySourceValue )                     dFree( copySourceValue );               }               else                  mustUpdate = true;            }            else            {               // Value does not differ.  If it differs from the copy source's value,               // though, we still want to write it out as otherwise we'll see the               // copy source's value override us.                              if( object->getCopySource() )               {                  const char* copySourceValue = getFieldValue( object->getCopySource(), f->pFieldname, j );                  if( copySourceValue && dStricmp( copySourceValue, value ) != 0 )                     mustUpdate = true;                                       if( copySourceValue )                     dFree( copySourceValue );               }            }            // The default value for most string type fields is            // NULL so we can't just continue here or we'd never ever            // write them out...            //            //if (!defaultValue)            //   continue;            // If the object's value is different from the default            // value then add it to the ParsedObject's newLines                                    if ( mustUpdate )            {               // TODO: This should be wrapped in a helper method... probably.               // Detect and collapse relative path information               if (f->type == TypeFilename ||                   f->type == TypeStringFilename ||                   f->type == TypeImageFilename ||                   f->type == TypePrefabFilename ||                   f->type == TypeShapeFilename)               {                  char fnBuf[1024];                  Con::collapseScriptFilename(fnBuf, 1024, value);                  newLines.push_back(createNewProperty(f->pFieldname, fnBuf, f->elementCount > 1, j));               }               else                  newLines.push_back(createNewProperty(f->pFieldname, value, f->elementCount > 1, j));                          }            if (defaultValue)               dFree( defaultValue );         }         dFree( value );      }   }   // Handle dynamic fields   SimFieldDictionary* fieldDict = object->getFieldDictionary();   for(SimFieldDictionaryIterator itr(fieldDict); *itr; ++itr)   {      SimFieldDictionary::Entry * entry = (*itr);      if( !entry->value )         continue;      // Let's see if this field is already in the file      S32 propertyIndex = getPropertyIndex(parsedObject, entry->slotName);      if (propertyIndex > -1)      {         ParsedProperty& prop = parsedObject->properties[propertyIndex];         // If this field is on the remove list then remove it and continue         if (findRemoveField(object, entry->slotName) || !object->writeField(entry->slotName, entry->value))         {            removeField( parsedObject->properties[ propertyIndex ] );            continue;         }         if( object->getCopySource() )         {            const char* copySourceFieldValue = object->getCopySource()->getDataField( entry->slotName, NULL );            if( dStrcmp( copySourceFieldValue, entry->value ) == 0 )            {               removeField( prop );               continue;            }         }         const char* evalue = prop.value;         const char *entryVal = entry->value;         if ( entryVal[0] == StringTagPrefixByte )                       entryVal = gNetStringTable->lookupString( dAtoi( entryVal+1 ) );         else         {            // Run the parsed value through the console system conditioners so            // that it will better match the data we got back from the object.            evalue = Con::getFormattedData(TypeString, evalue);         }         // If our data doesn't match then we get to update it         if (dStricmp(entryVal, evalue) != 0)            updateToken(prop.valueLine, prop.valuePosition, prop.endPosition - prop.valuePosition, entryVal);      }      else      {         // No need to process a removed field that doesn't exist in the file         if (findRemoveField(object, entry->slotName) || !object->writeField(entry->slotName, entry->value))            continue;         if( object->getCopySource() )         {            const char* copySourceFieldValue = object->getCopySource()->getDataField( entry->slotName, NULL );            if( dStrcmp( copySourceFieldValue, entry->value ) == 0 )               continue;         }         newLines.push_back(createNewProperty(entry->slotName, entry->value));      }   }      // If we have a parsedObject and the name changed   // then update the parsedObject to the new name.      // NOTE: an object 'can' have a NULL name which crashes in dStricmp.   if (parsedObject && parsedObject->name != StringTable->insert(object->getName(), true) )   {      StringTableEntry objectName = StringTable->insert(object->getName(), true);      if (parsedObject->name != objectName)      {         // Update the name in the file         updateToken(parsedObject->nameLine, parsedObject->namePosition, dStrlen(parsedObject->name), object->getName());         // Updated the parsedObject's name         parsedObject->name = objectName;         // Updated the object's "original" name to the one that is now in the file         object->setOriginalName(objectName);      }   }   if (parsedObject && newLines.size() > 0)   {      U32 lastPropLine = parsedObject->endLine;      if (parsedObject->properties.size() > 0)         lastPropLine = parsedObject->properties.last().valueLine + 1;      U32 currInsertLine = lastPropLine;      const char* indent = getObjectIndent(parsedObject);      // This should handle adding the opening { to an object      // that formerly did not have {};      if (!parsedObject->hasBraces)      {         updateToken(parsedObject->endLine, parsedObject->endPosition, 1, "\r\n{");         currInsertLine++;      }      currInsertLine += writeProperties(newLines, currInsertLine, indent);      // This should handle adding the opening } to an object      // that formerly did not have {};      if (!parsedObject->hasBraces)      {         U32 len = dStrlen(indent) + 3;         char* newLine = ( char* ) dMalloc( len );         dSprintf(newLine, len, "%s};", indent);         mLineBuffer.insert(currInsertLine, newLine);         currInsertLine++;      }      // Update the line offsets to account for the new lines      updateLineOffsets(lastPropLine, currInsertLine - lastPropLine);   }   else if (!parsedObject)   {      U32 insertLine = mLineBuffer.size();      if (!parentObject)         parentObject = findParentObject(object, parentObject);      if (parentObject && parentObject->endLine > -1)         insertLine = parentObject->endLine;      parsedObject = writeNewObject(object, newLines, insertLine, parentObject);   }   // Clean up the newLines memory   for (U32 i = 0; i < newLines.size(); i++)   {      if (newLines[i])      {         dFree(newLines[i]);         newLines[ i ] = NULL;      }   }   newLines.clear();   SimSet* set = dynamic_cast<SimSet*>(object);   if (set)   {      for(SimSet::iterator i = set->begin(); i != set->end(); i++)      {         SimObject* subObject = (SimObject*)(*i);         updateObject(subObject, parsedObject);      }   }   // Loop through the children of this parsedObject   // If they haven't been updated then assume that they   // don't exist in the file anymore   if (parsedObject)   {      for (S32 i = 0; i < mObjectBuffer.size(); i++)      {         ParsedObject* removeObj = mObjectBuffer[i];         if (removeObj->parentObject == parsedObject && !removeObj->updated)         {            removeParsedObject(removeObj);            mObjectBuffer.erase(i);            i--;            deleteObject(removeObj);         }      }   }   // Flag this as an updated object   if (parsedObject)      parsedObject->updated = true;      // Cleanup our created default object   delete defaultConObject;}bool PersistenceManager::saveDirtyFile(){   FileStream  stream;   stream.open( mCurrentFile, Torque::FS::File::Write );   if ( stream.getStatus() != Stream::Ok )   {      clearFileData();      return false;   }   for (U32 i = 0; i < mLineBuffer.size(); i++)      stream.writeLine((const U8*)mLineBuffer[i]);   stream.close();   //Con::printf("Successfully opened and wrote %s", mCurrentFile);   //Con::errorf("Updated Results:");   //for (U32 i = 0; i < mObjectBuffer.size(); i++)   //{   //   ParsedObject* parsedObject = mObjectBuffer[i];   //   Con::warnf("   mObjectBuffer[%d]:", i);   //   Con::warnf("      name = %s",      parsedObject->name);   //   Con::warnf("      className = %s", parsedObject->className);   //   Con::warnf("      startLine = %d", parsedObject->startLine + 1);   //   Con::warnf("      endLine   = %d", parsedObject->endLine + 1);   //   //if (mObjectBuffer[i]->properties.size() > 0)   //   //{   //   //   Con::warnf("   properties:");   //   //   for (U32 j = 0; j < mObjectBuffer[i]->properties.size(); j++)   //   //      Con::warnf("      %s = %s;", mObjectBuffer[i]->properties[j].name,   //   //                                   mObjectBuffer[i]->properties[j].value);   //   //}   //   if (!parsedObject->simObject.isNull())   //   {   //      SimObject* simObject = parsedObject->simObject;   //      Con::warnf("      SimObject(%s) %d:", simObject->getName(), simObject->getId());   //      Con::warnf("         declaration line = %d", simObject->getDeclarationLine());   //   }   //}   // Clear our file data   clearFileData();   return true;}S32 QSORT_CALLBACK PersistenceManager::compareFiles(const void* a,const void* b){   DirtyObject* objectA = (DirtyObject*)(a);   DirtyObject* objectB = (DirtyObject*)(b);   if (objectA->isNull())      return -1;   else if (objectB->isNull())      return 1;   if (objectA->fileName == objectB->fileName)      return objectA->getObject()->getDeclarationLine() - objectB->getObject()->getDeclarationLine();   return dStricmp(objectA->fileName, objectB->fileName);}bool PersistenceManager::setDirty(SimObject* inObject, const char* inFileName){   // Check if the object is already in the dirty list...   DirtyObject *pDirty = findDirtyObject( inObject );      // The filename we will save this object to (later)..   String saveFile;   // Expand the script filename if we were passed one.   if ( inFileName )   {      char buffer[4096];      Con::expandScriptFilename( buffer, 4096, inFileName );      saveFile = buffer;   }   // If no filename was passed in, and the object was already dirty,   // we have nothing to do.      if ( saveFile.isEmpty() && pDirty )      return true;   // Otherwise default to the simObject's filename.   if ( saveFile.isEmpty() )      saveFile = inObject->getFilename();      // Error if still no filename.   if ( saveFile.isEmpty() )   {      if (inObject->getName())         Con::errorf("PersistenceManager::setDirty() - SimObject %s has no file name associated with it - can not save", inObject->getName());      else         Con::errorf("PersistenceManager::setDirty() - SimObject %d has no file name associated with it - can not save", inObject->getId());      return false;   }   // Update the DirtyObject's fileName if we have it   // else create a new one.   if ( pDirty )      pDirty->fileName = StringTable->insert( saveFile );   else   {          // Add the newly dirty object.      mDirtyObjects.increment();      mDirtyObjects.last().setObject( inObject );      mDirtyObjects.last().fileName = StringTable->insert( saveFile );   }   return true;}void PersistenceManager::removeDirty(SimObject* object){   for (U32 i = 0; i < mDirtyObjects.size(); i++)   {      const DirtyObject& dirtyObject = mDirtyObjects[i];      if (dirtyObject.isNull())         continue;      if (dirtyObject.getObject() == object)      {         mDirtyObjects.erase(i);         break;      }   }   for (U32 i = 0; i < mRemoveFields.size(); i++)   {      const RemoveField& field = mRemoveFields[i];      if (field.object != object)         continue;      mRemoveFields.erase(i);      if (i > 0)         i--;   }}void PersistenceManager::addRemoveField(SimObject* object, const char* fieldName){   // Check to see if this is an array variable   U32 arrayPos = 0;   const char* name = fieldName;   if (dStrlen(fieldName) > 3 && fieldName[dStrlen(fieldName) - 1] == ']')   {      // The last character is a ']' which *should* mean      // there is also a corresponding '['      const char* arrayPosStart = dStrrchr(fieldName, '[');      if (!arrayPosStart)      {         Con::errorf("PersistenceManager::addRemoveField() - error parsing array position - \                      was expecting a '[' character");      }      else      {         // Parse the array position for the variable name         dSscanf(arrayPosStart, "[%d]", &arrayPos);         // Trim off the [<pos>] from the variable name         char* variableName = dStrdup(fieldName);         variableName[arrayPosStart - fieldName] = 0;         // Set the variable name to our new shortened name         name = StringTable->insert(variableName, true);         // Cleanup our variableName buffer         dFree( variableName );      }   }   // Make sure this field isn't already on the list   if (!findRemoveField(object, name, arrayPos))   {      mRemoveFields.increment();      RemoveField& field = mRemoveFields.last();      field.object = object;      field.fieldName = StringTable->insert(name);      field.arrayPos = arrayPos;   }}bool PersistenceManager::isDirty(SimObject* object){   return ( findDirtyObject( object ) != NULL );}PersistenceManager::DirtyObject* PersistenceManager::findDirtyObject(SimObject* object){   for (U32 i = 0; i < mDirtyObjects.size(); i++)   {      const DirtyObject& dirtyObject = mDirtyObjects[i];      if (dirtyObject.isNull())         continue;      if (dirtyObject.getObject() == object)         return &mDirtyObjects[i];   }   return NULL;}bool PersistenceManager::findRemoveField(SimObject* object, const char* fieldName, U32 arrayPos){   for (U32 i = 0; i < mRemoveFields.size(); i++)   {      if (mRemoveFields[i].object == object &&          mRemoveFields[i].arrayPos == arrayPos &&          dStricmp(mRemoveFields[i].fieldName, fieldName) == 0)      {         return true;      }   }   return false;}bool PersistenceManager::saveDirty(){   // Remove any null SimObject's first   for (S32 i = 0; i < mDirtyObjects.size(); i++)   {      const DirtyObject& dirtyObject = mDirtyObjects[i];      if (dirtyObject.isNull())      {         mDirtyObjects.erase(i);         i--;      }   }   // Sort by filename and declaration lines   dQsort(mDirtyObjects.address(), mDirtyObjects.size(), sizeof(DirtyList::value_type), compareFiles);   for (U32 i = 0; i < mDirtyObjects.size(); i++)   {      const DirtyObject& dirtyObject = mDirtyObjects[i];      if (dirtyObject.isNull())         continue;      SimObject* object = dirtyObject.getObject();      if (!mCurrentFile || dStricmp(mCurrentFile, dirtyObject.fileName) != 0)      {         // If mCurrentFile is set then that means we         // changed file names so save our previous one         if (mCurrentFile)            saveDirtyFile();         // Open our new file and parse it         bool success = parseFile(dirtyObject.fileName);         if (!success)         {            const char *name = object->getName();            if (name)            {               Con::errorf("PersistenceManager::saveDirty(): Unable to open %s to save %s %s (%d)",                  dirtyObject.fileName, object->getClassName(), name, object->getId());            }            else            {               Con::errorf("PersistenceManager::saveDirty(): Unable to open %s to save %s (%d)",                  dirtyObject.fileName, object->getClassName(), object->getId());            }            continue;         }      }      // Update this object's properties      //      // An empty script file (with 1 line) gets here with zero      // elements in the linebuffer, so this would prevent us from      // ever writing to it... Or is this test preventing bad things from      // happening if the file didn't exist at all?      //      if (mCurrentFile /*&& mLineBuffer.size() > 0*/)         updateObject(object);   }   // Save out our last file   if (mCurrentFile)      saveDirtyFile();   // Done writing out our dirty objects so reset everything   clearAll();   return true;}bool PersistenceManager::saveDirtyObject(SimObject* object){   // find our object passed in   for (U32 i = 0; i < mDirtyObjects.size(); i++)   {      const DirtyObject& dirtyObject = mDirtyObjects[i];      if (dirtyObject.isNull())         continue;      if (dirtyObject.getObject() == object)      {         // Open our new file and parse it         bool success = parseFile(dirtyObject.fileName);         if (!success)         {            const char *name = object->getName();            if (name)            {			      Con::errorf("PersistenceManager::saveDirtyObject(): Unable to open %s to save %s %s (%d)",				      dirtyObject.fileName, object->getClassName(), name, object->getId());		      }		      else		      {			      Con::errorf("PersistenceManager::saveDirtyObject(): Unable to open %s to save %s (%d)",				      dirtyObject.fileName, object->getClassName(), object->getId());		      }		      return false;		   }         // if the file exists then lets update and save		   if(mCurrentFile)		   {		      updateObject(object);            saveDirtyFile();		   }                  break;      }   }   // remove this object from the dirty list   removeDirty(object);   return true;}void PersistenceManager::removeObjectFromFile(SimObject* object, const char* fileName){   if (mCurrentFile)   {      Con::errorf("PersistenceManager::removeObjectFromFile(): Can't remove an object from a \                  file while another is currently opened");      return;   }      const char* file = object->getFilename();   if (fileName)   {      char buffer[1024];      Con::expandScriptFilename( buffer, 1024, fileName );      file = StringTable->insert(buffer);   }   bool success = false;      if ( file && file[ 0 ] )      success = parseFile(file);   if (!success)   {      const char *name = object->getName();      String errorNameStr;      if ( name )         errorNameStr = String::ToString( "%s %s (%d)", object->getClassName(), name, object->getId() );      else         errorNameStr = String::ToString( "%s (%d)", object->getClassName(), object->getId() );      if ( !file )         Con::errorf("PersistenceManager::removeObjectFromFile(): File was null trying to save %s", errorNameStr.c_str() );      else                     Con::errorf("PersistenceManager::removeObjectFromFile(): Unable to open %s to save %s", file, errorNameStr.c_str() );                  // Reset everything      clearAll();      return;   }   ParsedObject* parsedObject = findParsedObject(object);   if (!parsedObject)   {      const char *name = object->getName();      if (name)      {         Con::errorf("PersistenceManager::removeObjectFromFile(): Unable to find %s %s (%d) in %s",            object->getClassName(), name, object->getId(), file);      }      else      {         Con::errorf("PersistenceManager::removeObjectFromFile(): Unable to find %s (%d) in %s",            object->getClassName(), object->getId(), file);      }      // Reset everything      clearAll();      return;   }   removeParsedObject(parsedObject);   for (U32 i = 0; i < mObjectBuffer.size(); i++)   {      ParsedObject* removeObj = mObjectBuffer[i];      if (removeObj == parsedObject)      {         mObjectBuffer.erase(i);         break;      }   }   deleteObject(parsedObject);   // Save out the file   if (mCurrentFile)      saveDirtyFile();   // Reset everything   clearAll();}void PersistenceManager::deleteObjectsFromFile(const char* fileName){   if ( mCurrentFile )   {      Con::errorf( "PersistenceManager::deleteObjectsFromFile(): Cannot process while file while another is currently open." );      return;   }   // Expand Script File.   char buffer[1024];   Con::expandScriptFilename( buffer, 1024, fileName );   // Parse File.   if ( !parseFile( StringTable->insert(buffer) ) )   {      // Invalid.      return;   }   // Iterate over the objects.   for ( Vector<ParsedObject*>::iterator itr = mObjectBuffer.begin(); itr != mObjectBuffer.end(); itr++ )   {      SimObject *object;      if ( Sim::findObject( ( *itr )->name, object ) )      {         // Delete the Object.         object->deleteObject();      }   }   // Clear.   clearAll();}ConsoleMethod( PersistenceManager, deleteObjectsFromFile, void, 3, 3, "( fileName )"              "Delete all of the objects that are created from the given file." ){   // Delete Objects.   object->deleteObjectsFromFile( argv[2] );}ConsoleMethod( PersistenceManager, setDirty, void, 3, 4, "(SimObject object, [filename])"              "Mark an existing SimObject as dirty (will be written out when saveDirty() is called)."){   SimObject *dirtyObject = NULL;   if (argv[2][0])   {      if (!Sim::findObject(argv[2], dirtyObject))      {         Con::printf("%s(): Invalid SimObject: %s", argv[0], argv[2]);         return;      }   }      // Prevent ourselves from shooting us in the foot.      if( dirtyObject == Sim::getRootGroup() )   {      Con::errorf( "%s(): Cannot save RootGroup", argv[ 0 ] );      return;   }   if (dirtyObject)   {      if (argc == 4 && argv[3][0])         object->setDirty(dirtyObject, argv[3]);      else         object->setDirty(dirtyObject);   }}ConsoleMethod( PersistenceManager, removeDirty, void, 3, 3, "(SimObject object)"              "Remove a SimObject from the dirty list."){   SimObject *dirtyObject = NULL;   if (argv[2][0])   {      if (!Sim::findObject(argv[2], dirtyObject))      {         Con::printf("%s(): Invalid SimObject: %s", argv[0], argv[2]);         return;      }   }   if (dirtyObject)      object->removeDirty(dirtyObject);}ConsoleMethod( PersistenceManager, isDirty, bool, 3, 3, "(SimObject object)"              "Returns true if the SimObject is on the dirty list."){   SimObject *dirtyObject = NULL;   if (argv[2][0])   {      if (!Sim::findObject(argv[2], dirtyObject))      {         Con::printf("%s(): Invalid SimObject: %s", argv[0], argv[2]);         return false;      }   }   if (dirtyObject)      return object->isDirty(dirtyObject);   return false;}ConsoleMethod( PersistenceManager, hasDirty, bool, 2, 2, "()"              "Returns true if the manager has dirty objects to save." ){   return object->hasDirty();}ConsoleMethod( PersistenceManager, getDirtyObjectCount, S32, 2, 2, "()"              "Returns the number of dirty objects." ){   return object->getDirtyList().size();}ConsoleMethod( PersistenceManager, getDirtyObject, S32, 3, 3, "( index )"              "Returns the ith dirty object." ){   const S32 index = dAtoi( argv[2] );   if ( index < 0 || index >= object->getDirtyList().size() )   {      Con::warnf( "PersistenceManager::getDirtyObject() - Index (%s) out of range.", argv[2] );      return 0;   }   // Fetch Object.   const PersistenceManager::DirtyObject& dirtyObject = object->getDirtyList()[index];   // Return Id.   return ( dirtyObject.getObject() ) ? dirtyObject.getObject()->getId() : 0;}ConsoleMethod( PersistenceManager, listDirty, void, 2, 2, "()"              "Prints the dirty list to the console."){   const PersistenceManager::DirtyList dirtyList = object->getDirtyList();   for(U32 i = 0; i < dirtyList.size(); i++)   {      const PersistenceManager::DirtyObject& dirtyObject = dirtyList[i];      if (dirtyObject.isNull())         continue;      SimObject *obj = dirtyObject.getObject();      bool isSet = dynamic_cast<SimSet *>(obj) != 0;      const char *name = obj->getName();      if (name)      {         Con::printf("   %d,\"%s\": %s %s %s", obj->getId(), name,         obj->getClassName(), dirtyObject.fileName, isSet ? "(g)":"");      }      else      {         Con::printf("   %d: %s %s, %s", obj->getId(), obj->getClassName(),         dirtyObject.fileName, isSet ? "(g)" : "");      }   }}ConsoleMethod( PersistenceManager, saveDirty, bool, 2, 2, "()"              "Saves all of the SimObject's on the dirty list to their respective files."){   return object->saveDirty();}ConsoleMethod( PersistenceManager, saveDirtyObject, bool, 3, 3, "(SimObject object)"              "Save a dirty SimObject to it's file."){   SimObject *dirtyObject = NULL;   if (argv[2][0])   {      if (!Sim::findObject(argv[2], dirtyObject))      {         Con::printf("%s(): Invalid SimObject: %s", argv[0], argv[2]);         return false;      }   }   if (dirtyObject)      return object->saveDirtyObject(dirtyObject);   return false;}ConsoleMethod( PersistenceManager, clearAll, void, 2, 2, "()"              "Clears all the tracked objects without saving them." ){   object->clearAll();}ConsoleMethod( PersistenceManager, removeObjectFromFile, void, 3, 4, "(SimObject object, [filename])"              "Remove an existing SimObject from a file (can optionally specify a different file than \               the one it was created in."){   SimObject *dirtyObject = NULL;   if (argv[2][0])   {      if (!Sim::findObject(argv[2], dirtyObject))      {         Con::printf("%s(): Invalid SimObject: %s", argv[0], argv[2]);         return;      }   }   if (dirtyObject)   {      if (argc == 4 && argv[3][0])         object->removeObjectFromFile(dirtyObject, argv[3]);      else         object->removeObjectFromFile(dirtyObject);   }}ConsoleMethod( PersistenceManager, removeField, void, 4, 4, "(SimObject object, string fieldName)"              "Remove a specific field from an object declaration."){   SimObject *dirtyObject = NULL;   if (argv[2][0])   {      if (!Sim::findObject(argv[2], dirtyObject))      {         Con::printf("%s(): Invalid SimObject: %s", argv[0], argv[2]);         return;      }   }   if (dirtyObject)   {      if (argv[3][0])         object->addRemoveField(dirtyObject, argv[3]);   }}
 |