| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948 | //-----------------------------------------------------------------------------// 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 "console/simBase.h"#include "platform/nativeDialogs/fileDialog.h"#include "platform/threads/mutex.h"#include "platformWin32/platformWin32.h"#include "core/util/safeDelete.h"#include "math/mMath.h"#include "core/strings/unicode.h"#include "console/consoleTypes.h"#include "platform/profiler.h"#include <ShlObj.h>#include <WindowsX.h>#include "console/engineAPI.h"#ifdef TORQUE_TOOLS//-----------------------------------------------------------------------------// PlatformFileDlgData Implementation//-----------------------------------------------------------------------------FileDialogData::FileDialogData(){   // Default Path   //   //  Try to provide consistent experience by recalling the last file path   // - else   //  Default to Working Directory if last path is not set or is invalid   mDefaultPath = StringTable->insert( Con::getVariable("Tools::FileDialogs::LastFilePath") );   if( mDefaultPath == StringTable->lookup("") || !Platform::isDirectory( mDefaultPath ) )      mDefaultPath = Platform::getCurrentDirectory();   mDefaultFile = StringTable->insert("");   mFilters = StringTable->insert("");   mFile = StringTable->insert("");   mTitle = StringTable->insert("");   mStyle = 0;}FileDialogData::~FileDialogData(){}static LRESULT PASCAL OKBtnFolderHackProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam){   WNDPROC oldProc = (WNDPROC)GetProp(hWnd, dT("OldWndProc"));   switch(uMsg)   {      case WM_COMMAND:         if(LOWORD(wParam) == IDOK)         {            LPOPENFILENAME ofn = (LPOPENFILENAME)GetProp(hWnd, dT("OFN"));            if(ofn == NULL)               break;            SendMessage(hWnd, CDM_GETFILEPATH, ofn->nMaxFile, (LPARAM)ofn->lpstrFile);            char *filePath;#ifdef UNICODE            char fileBuf[MAX_PATH];            convertUTF16toUTF8(ofn->lpstrFile, fileBuf);            filePath = fileBuf;#else            filePath = ofn->lpstrFile;#endif            if(Platform::isDirectory(filePath))            {               // Got a directory               EndDialog(hWnd, IDOK);            }         }         break;   }   if(oldProc)      return CallWindowProc(oldProc, hWnd, uMsg, wParam, lParam);   else      return DefWindowProc(hWnd, uMsg, wParam, lParam);}static UINT_PTR CALLBACK FolderHookProc(HWND hdlg, UINT uMsg, WPARAM wParam, LPARAM lParam){   HWND hParent = GetParent(hdlg);      switch(uMsg)   {      case WM_INITDIALOG:         {            LPOPENFILENAME lpofn = (LPOPENFILENAME)lParam;            SendMessage(hParent, CDM_SETCONTROLTEXT, stc3, (LPARAM)dT("Folder name:"));            SendMessage(hParent, CDM_HIDECONTROL, cmb1, 0);            SendMessage(hParent, CDM_HIDECONTROL, stc2, 0);            LONG oldProc = SetWindowLong(hParent, GWLP_WNDPROC, (LONG)OKBtnFolderHackProc);            SetProp(hParent, dT("OldWndProc"), (HANDLE)oldProc);            SetProp(hParent, dT("OFN"), (HANDLE)lpofn);         }         break;      case WM_NOTIFY:         {            LPNMHDR nmhdr = (LPNMHDR)lParam;            switch(nmhdr->code)            {               case CDN_FOLDERCHANGE:                  {                     LPOFNOTIFY lpofn = (LPOFNOTIFY)lParam;                     OpenFolderDialog *ofd = (OpenFolderDialog *)lpofn->lpOFN->lCustData;                     #ifdef UNICODE                     UTF16 buf[MAX_PATH];#else                     char buf[MAX_PATH];#endif                     SendMessage(hParent, CDM_GETFOLDERPATH, sizeof(buf), (LPARAM)buf);                     char filePath[MAX_PATH];#ifdef UNICODE                     convertUTF16toUTF8(buf, filePath);#else                     dStrcpy( filePath, buf );#endif                     // [tom, 12/8/2006] Hack to remove files from the list because                     // CDN_INCLUDEITEM doesn't work for regular files and folders.                     HWND shellView = GetDlgItem(hParent, lst2);                     HWND listView = FindWindowEx(shellView, 0, WC_LISTVIEW, NULL);                     if(listView)                     {                        S32 count = ListView_GetItemCount(listView);                        for(S32 i = count - 1;i >= 0;--i)                        {                           ListView_GetItemText(listView, i, 0, buf, sizeof(buf));                           #ifdef UNICODE                           char buf2[MAX_PATH];                           convertUTF16toUTF8(buf, buf2);#else                           char *buf2 = buf;#endif                           char full[MAX_PATH];                           dSprintf(full, sizeof(full), "%s\\%s", filePath, buf2);                           if(!Platform::isDirectory(full))                           {                              ListView_DeleteItem(listView, i);                           }                        }                     }                     if(ofd->mMustExistInDir == NULL || *ofd->mMustExistInDir == 0)                        break;                     HWND hOK = GetDlgItem(hParent, IDOK);                     if(hOK == NULL)                        break;                     char checkPath[MAX_PATH];                     dSprintf(checkPath, sizeof(checkPath), "%s\\%s", filePath, ofd->mMustExistInDir);                     EnableWindow(hOK, Platform::isFile(checkPath));                  }                  break;            }         }         break;   }   return 0;}//-----------------------------------------------------------------------------// FileDialog Implementation//-----------------------------------------------------------------------------IMPLEMENT_CONOBJECT(FileDialog);ConsoleDocClass( FileDialog,   "@brief Base class responsible for displaying an OS file browser.\n\n"   "FileDialog is a platform agnostic dialog interface for querying the user for "   "file locations. It is designed to be used through the exposed scripting interface.\n\n"      "FileDialog is the base class for Native File Dialog controls in Torque. It provides these basic areas of functionality:\n\n"   "   - Inherits from SimObject and is exposed to the scripting interface\n"   "   - Provides blocking interface to allow instant return to script execution\n"   "   - Simple object configuration makes practical use easy and effective\n\n"      "FileDialog is *NOT* intended to be used directly in script and is only exposed to script to expose generic file dialog attributes.\n\n"   "This base class is usable in TorqueScript, but is does not specify what functionality is intended (open or save?). "   "Its children, OpenFileDialog and SaveFileDialog, do make use of DialogStyle flags and do make use of specific funcationality. "   "These are the preferred classes to use\n\n"   "However, the FileDialog base class does contain the key properties and important method for file browing. The most "   "important function is Execute(). This is used by both SaveFileDialog and OpenFileDialog to initiate the browser.\n\n"   "@tsexample\n"   "// NOTE: This is not he preferred class to use, but this still works\n\n"   "// Create the file dialog\n"   "%baseFileDialog = new FileDialog()\n"   "{\n"   "   // Allow browsing of all file types\n"   "   filters = \"*.*\";\n\n"   "   // No default file\n"   "   defaultFile = "";\n\n"   "   // Set default path relative to project\n"   "   defaultPath = \"./\";\n\n"   "   // Set the title\n"   "   title = \"Durpa\";\n\n"   "   // Allow changing of path you are browsing\n"   "   changePath = true;\n"   "};\n\n"   " // Launch the file dialog\n"   " %baseFileDialog.Execute();\n"   " \n"   " // Don't forget to cleanup\n"   " %baseFileDialog.delete();\n\n\n"   "@endtsexample\n\n"   "@note FileDialog and its related classes are only availble in a Tools build of Torque.\n\n"   "@see OpenFileDialog for a practical example on opening a file\n"   "@see SaveFileDialog for a practical example of saving a file\n"   "@ingroup FileSystem\n");FileDialog::FileDialog() : mData(){   // Default to File Must Exist Open Dialog style   mData.mStyle = FileDialogData::FDS_OPEN | FileDialogData::FDS_MUSTEXIST;   mChangePath = false;}FileDialog::~FileDialog(){}void FileDialog::initPersistFields(){   addProtectedField( "defaultPath", TypeString, Offset(mData.mDefaultPath, FileDialog), &setDefaultPath, &defaultProtectedGetFn,       "The default directory path when the dialog is shown." );         addProtectedField( "defaultFile", TypeString, Offset(mData.mDefaultFile, FileDialog), &setDefaultFile, &defaultProtectedGetFn,       "The default file path when the dialog is shown." );               addProtectedField( "fileName", TypeString, Offset(mData.mFile, FileDialog), &setFile, &defaultProtectedGetFn,       "The default file name when the dialog is shown." );         addProtectedField( "filters", TypeString, Offset(mData.mFilters, FileDialog), &setFilters, &defaultProtectedGetFn,       "The filter string for limiting the types of files visible in the dialog.  It makes use of the pipe symbol '|' "      "as a delimiter.  For example:\n\n"      "'All Files|*.*'\n\n"      "'Image Files|*.png;*.jpg|Png Files|*.png|Jepg Files|*.jpg'" );         addField( "title", TypeString, Offset(mData.mTitle, FileDialog),       "The title for the dialog." );      addProtectedField( "changePath", TypeBool, Offset(mChangePath, FileDialog), &setChangePath, &getChangePath,      "True/False whether to set the working directory to the directory returned by the dialog." );      Parent::initPersistFields();}static const U32 convertUTF16toUTF8DoubleNULL( const UTF16 *unistring, UTF8  *outbuffer, U32 len){   AssertFatal(len >= 1, "Buffer for unicode conversion must be large enough to hold at least the null terminator.");   PROFILE_START(convertUTF16toUTF8DoubleNULL);   U32 walked, nCodeunits, codeunitLen;   UTF32 middleman;   nCodeunits=0;   while( ! (*unistring == '\0' && *(unistring + 1) == '\0') && nCodeunits + 3 < len )   {      walked = 1;      middleman  = oneUTF16toUTF32(unistring,&walked);      codeunitLen = oneUTF32toUTF8(middleman, &outbuffer[nCodeunits]);      unistring += walked;      nCodeunits += codeunitLen;   }   nCodeunits = getMin(nCodeunits,len - 1);   outbuffer[nCodeunits] = '\0';   outbuffer[nCodeunits+1] = '\0';   PROFILE_END();   return nCodeunits;}//// Execute Method//bool FileDialog::Execute(){   static char pszResult[MAX_PATH];#ifdef UNICODE   UTF16 pszFile[MAX_PATH];   UTF16 pszInitialDir[MAX_PATH];   UTF16 pszTitle[MAX_PATH];   UTF16 pszFilter[1024];   UTF16 pszFileTitle[MAX_PATH];   UTF16 pszDefaultExtension[MAX_PATH];   // Convert parameters to UTF16*'s   convertUTF8toUTF16((UTF8 *)mData.mDefaultFile, pszFile);   convertUTF8toUTF16((UTF8 *)mData.mDefaultPath, pszInitialDir);   convertUTF8toUTF16((UTF8 *)mData.mTitle, pszTitle);   convertUTF8toUTF16((UTF8 *)mData.mFilters, pszFilter);#else   // Not Unicode, All char*'s!   char pszFile[MAX_PATH];   char pszFilter[1024];   char pszFileTitle[MAX_PATH];   dStrcpy( pszFile, mData.mDefaultFile );   dStrcpy( pszFilter, mData.mFilters );   const char* pszInitialDir = mData.mDefaultPath;   const char* pszTitle = mData.mTitle;   #endif   pszFileTitle[0] = 0;   // Convert Filters   U32 filterLen = dStrlen( pszFilter );   S32 dotIndex = -1;   for( U32 i = 0; i < filterLen; i++ )   {      if( pszFilter[i] == '|' )         pszFilter[i] = '\0';      if( pszFilter[ i ] == '.' && dotIndex == -1 )         dotIndex = i;   }   // Add second NULL terminator at the end   pszFilter[ filterLen + 1 ] = '\0';   // Get default extension.   dMemset( pszDefaultExtension, 0, sizeof( pszDefaultExtension ) );   if( dotIndex != -1 )   {      for( U32 i = 0; i < MAX_PATH; ++ i )      {         UTF16 ch = pszFilter[ dotIndex + 1 + i ];         if( !ch || ch == ';' || ch == '|' || dIsspace( ch ) )            break;         pszDefaultExtension[ i ] = ch;      }   }   OPENFILENAME ofn;   dMemset(&ofn, 0, sizeof(ofn));   ofn.lStructSize      = sizeof(ofn);   ofn.hwndOwner        = getWin32WindowHandle();   ofn.lpstrFile        = pszFile;   if( !dStrncmp( mData.mDefaultFile, "", 1 ) )      ofn.lpstrFile[0]     = '\0';   ofn.nMaxFile         = sizeof(pszFile);   ofn.lpstrFilter      = pszFilter;   ofn.nFilterIndex     = 1;   ofn.lpstrInitialDir  = pszInitialDir;   ofn.lCustData        = (LPARAM)this;   ofn.lpstrFileTitle   = pszFileTitle;   ofn.nMaxFileTitle    = sizeof(pszFileTitle);   ofn.lpstrDefExt      = pszDefaultExtension[ 0 ] ? pszDefaultExtension : NULL;   if( mData.mTitle != StringTable->lookup("") )      ofn.lpstrTitle = pszTitle;   // Build Proper Flags.   ofn.Flags = OFN_EXPLORER | OFN_ENABLESIZING | OFN_HIDEREADONLY;   if(mData.mStyle & FileDialogData::FDS_BROWSEFOLDER)   {      ofn.lpfnHook = FolderHookProc;      ofn.Flags |= OFN_ENABLEHOOK;   }      if( !(mData.mStyle & FileDialogData::FDS_CHANGEPATH) )      ofn.Flags |= OFN_NOCHANGEDIR;   if( mData.mStyle & FileDialogData::FDS_MUSTEXIST )      ofn.Flags |= OFN_FILEMUSTEXIST;   if( mData.mStyle & FileDialogData::FDS_OPEN && mData.mStyle & FileDialogData::FDS_MULTIPLEFILES )      ofn.Flags |= OFN_ALLOWMULTISELECT;   if( mData.mStyle & FileDialogData::FDS_OVERWRITEPROMPT )      ofn.Flags |= OFN_OVERWRITEPROMPT;   // Flag we're showing file browser so we can do some render hacking   winState.renderThreadBlocked = true;   // Get the current working directory, so we can back up to it once Windows has   // done its craziness and messed with it.   StringTableEntry cwd = Platform::getCurrentDirectory();   // Execute Dialog (Blocking Call)   bool dialogSuccess = false;   if( mData.mStyle & FileDialogData::FDS_OPEN )      dialogSuccess = GetOpenFileName(&ofn);   else if( mData.mStyle & FileDialogData::FDS_SAVE )      dialogSuccess = GetSaveFileName(&ofn);   // Dialog is gone.   winState.renderThreadBlocked = false;   // Restore the working directory.   Platform::setCurrentDirectory( cwd );   // Did we select a file?   if( !dialogSuccess )      return false;   // Handle Result Properly for Unicode as well as ANSI#ifdef UNICODE   if(pszFileTitle[0] || ! ( mData.mStyle & FileDialogData::FDS_OPEN && mData.mStyle & FileDialogData::FDS_MULTIPLEFILES ))      convertUTF16toUTF8( (UTF16*)pszFile, pszResult);   else      convertUTF16toUTF8DoubleNULL( (UTF16*)pszFile, (UTF8*)pszResult, sizeof(pszResult));#else   if(pszFileTitle[0] || ! ( mData.mStyle & FileDialogData::FDS_OPEN && mData.mStyle & FileDialogData::FDS_MULTIPLEFILES ))      dStrcpy(pszResult,pszFile);   else   {      // [tom, 1/4/2007] pszResult is a double-NULL terminated, NULL separated list in this case so we can't just dSstrcpy()      char *sptr = pszFile, *dptr = pszResult;      while(! (*sptr == 0 && *(sptr+1) == 0))         *dptr++ = *sptr++;      *dptr++ = 0;   }#endif   forwardslash(pszResult);   // [tom, 1/5/2007] Windows is ridiculously dumb. If you select a single file in a multiple   // select file dialog then it will return the file the same way as it would in a single   // select dialog. The only difference is pszFileTitle is empty if multiple files   // are selected.   // Store the result on our object   if( mData.mStyle & FileDialogData::FDS_BROWSEFOLDER || ( pszFileTitle[0] && ! ( mData.mStyle & FileDialogData::FDS_OPEN && mData.mStyle & FileDialogData::FDS_MULTIPLEFILES )))   {      // Single file selection, do it the easy way      mData.mFile = StringTable->insert( pszResult );   }   else if(pszFileTitle[0] && ( mData.mStyle & FileDialogData::FDS_OPEN && mData.mStyle & FileDialogData::FDS_MULTIPLEFILES ))   {      // Single file selection in a multiple file selection dialog      setDataField(StringTable->insert("files"), "0", pszResult);      setDataField(StringTable->insert("fileCount"), NULL, "1");   }   else   {      // Multiple file selection, break out into an array      S32 numFiles = 0;      const char *dir = pszResult;      const char *file = dir + dStrlen(dir) + 1;      char buffer[1024];      while(*file)      {         Platform::makeFullPathName(file, buffer, sizeof(buffer), dir);         setDataField(StringTable->insert("files"), Con::getIntArg(numFiles++), buffer);         file = file + dStrlen(file) + 1;      }      setDataField(StringTable->insert("fileCount"), NULL, Con::getIntArg(numFiles));   }   // Return success.   return true;}DefineEngineMethod( FileDialog, Execute, bool, (),,   "@brief Launches the OS file browser\n\n"   "After an Execute() call, the chosen file name and path is available in one of two areas.  "   "If only a single file selection is permitted, the results will be stored in the @a fileName "   "attribute.\n\n"   "If multiple file selection is permitted, the results will be stored in the "   "@a files array.  The total number of files in the array will be stored in the "   "@a fileCount attribute.\n\n"   "@tsexample\n"   "// NOTE: This is not he preferred class to use, but this still works\n\n"   "// Create the file dialog\n"   "%baseFileDialog = new FileDialog()\n"   "{\n"   "   // Allow browsing of all file types\n"   "   filters = \"*.*\";\n\n"   "   // No default file\n"   "   defaultFile = "";\n\n"   "   // Set default path relative to project\n"   "   defaultPath = \"./\";\n\n"   "   // Set the title\n"   "   title = \"Durpa\";\n\n"   "   // Allow changing of path you are browsing\n"   "   changePath = true;\n"   "};\n\n"   " // Launch the file dialog\n"   " %baseFileDialog.Execute();\n"   " \n"   " // Don't forget to cleanup\n"   " %baseFileDialog.delete();\n\n\n"   " // A better alternative is to use the \n"   " // derived classes which are specific to file open and save\n\n"   " // Create a dialog dedicated to opening files\n"   " %openFileDlg = new OpenFileDialog()\n"   " {\n"   "    // Look for jpg image files\n"   "    // First part is the descriptor|second part is the extension\n"   "    Filters = \"Jepg Files|*.jpg\";\n"   "    // Allow browsing through other folders\n"   "    ChangePath = true;\n\n"   "    // Only allow opening of one file at a time\n"   "    MultipleFiles = false;\n"   " };\n\n"   " // Launch the open file dialog\n"   " %result = %openFileDlg.Execute();\n\n"   " // Obtain the chosen file name and path\n"   " if ( %result )\n"   " {\n"   "    %seletedFile = %openFileDlg.file;\n"   " }\n"   " else\n"   " {\n"   "    %selectedFile = \"\";\n"   " }\n"   " // Cleanup\n"   " %openFileDlg.delete();\n\n\n"   " // Create a dialog dedicated to saving a file\n"   " %saveFileDlg = new SaveFileDialog()\n"   " {\n"   "    // Only allow for saving of COLLADA files\n"   "    Filters = \"COLLADA Files (*.dae)|*.dae|\";\n\n"   "    // Default save path to where the WorldEditor last saved\n"   "    DefaultPath = $pref::WorldEditor::LastPath;\n\n"   "    // No default file specified\n"   "    DefaultFile = \"\";\n\n"   "    // Do not allow the user to change to a new directory\n"   "    ChangePath = false;\n\n"   "    // Prompt the user if they are going to overwrite an existing file\n"   "    OverwritePrompt = true;\n"   " };\n\n"   " // Launch the save file dialog\n"   " %result = %saveFileDlg.Execute();\n\n"   " // Obtain the file name\n"   " %selectedFile = \"\";\n"   " if ( %result )\n"   "    %selectedFile = %saveFileDlg.file;\n\n"   " // Cleanup\n"   " %saveFileDlg.delete();\n"   "@endtsexample\n\n"   "@return True if the file was selected was successfully found (opened) or declared (saved)."){   return object->Execute();}//-----------------------------------------------------------------------------// Dialog Filters//-----------------------------------------------------------------------------bool FileDialog::setFilters( void *object, const char *index, const char *data ){   // Will do validate on write at some point.   if( !data )      return true;   return true;};//-----------------------------------------------------------------------------// Default Path Property - String Validated on Write//-----------------------------------------------------------------------------bool FileDialog::setDefaultPath( void *object, const char *index, const char *data ){   if( !data || !dStrncmp( data, "", 1 ) )      return true;   // Copy and Backslash the path (Windows dialogs are VERY picky about this format)   static char szPathValidate[512];   dStrcpy( szPathValidate, data );   Platform::makeFullPathName( data,szPathValidate, sizeof(szPathValidate));   backslash( szPathValidate );   // Remove any trailing \'s   S8 validateLen = dStrlen( szPathValidate );   if( szPathValidate[ validateLen - 1 ] == '\\' )      szPathValidate[ validateLen - 1 ] = '\0';   // Now check    if( Platform::isDirectory( szPathValidate ) )   {      // Finally, assign in proper format.      FileDialog *pDlg = static_cast<FileDialog*>( object );      pDlg->mData.mDefaultPath = StringTable->insert( szPathValidate );   }#ifdef TORQUE_DEBUG   else      Con::errorf(ConsoleLogEntry::GUI, "FileDialog - Invalid Default Path Specified!");#endif   return false;};//-----------------------------------------------------------------------------// Default File Property - String Validated on Write//-----------------------------------------------------------------------------bool FileDialog::setDefaultFile( void *object, const char *index, const char *data ){   if( !data || !dStrncmp( data, "", 1 ) )      return true;   // Copy and Backslash the path (Windows dialogs are VERY picky about this format)   static char szPathValidate[512];   Platform::makeFullPathName( data,szPathValidate, sizeof(szPathValidate) );   backslash( szPathValidate );   // Remove any trailing \'s   S8 validateLen = dStrlen( szPathValidate );   if( szPathValidate[ validateLen - 1 ] == '\\' )      szPathValidate[ validateLen - 1 ] = '\0';   // Finally, assign in proper format.   FileDialog *pDlg = static_cast<FileDialog*>( object );   pDlg->mData.mDefaultFile = StringTable->insert( szPathValidate );   return false;};//-----------------------------------------------------------------------------// ChangePath Property - Change working path on successful file selection//-----------------------------------------------------------------------------bool FileDialog::setChangePath( void *object, const char *index, const char *data ){   bool bMustExist = dAtob( data );   FileDialog *pDlg = static_cast<FileDialog*>( object );   if( bMustExist )      pDlg->mData.mStyle |= FileDialogData::FDS_CHANGEPATH;   else      pDlg->mData.mStyle &= ~FileDialogData::FDS_CHANGEPATH;   return true;};const char* FileDialog::getChangePath(void* obj, const char* data){   FileDialog *pDlg = static_cast<FileDialog*>( obj );   if( pDlg->mData.mStyle & FileDialogData::FDS_CHANGEPATH )      return StringTable->insert("true");   else      return StringTable->insert("false");}bool FileDialog::setFile( void *object, const char *index, const char *data ){   return false;};//-----------------------------------------------------------------------------// OpenFileDialog Implementation//-----------------------------------------------------------------------------ConsoleDocClass( OpenFileDialog,   "@brief Derived from FileDialog, this class is responsible for opening a file browser with the intention of opening a file.\n\n"   "The core usage of this dialog is to locate a file in the OS and return the path and name. This does not handle "   "the actual file parsing or data manipulation. That functionality is left up to the FileObject class.\n\n"      "@tsexample\n"   " // Create a dialog dedicated to opening files\n"   " %openFileDlg = new OpenFileDialog()\n"   " {\n"   "    // Look for jpg image files\n"   "    // First part is the descriptor|second part is the extension\n"   "    Filters = \"Jepg Files|*.jpg\";\n"   "    // Allow browsing through other folders\n"   "    ChangePath = true;\n\n"   "    // Only allow opening of one file at a time\n"   "    MultipleFiles = false;\n"   " };\n\n"   " // Launch the open file dialog\n"   " %result = %openFileDlg.Execute();\n\n"   " // Obtain the chosen file name and path\n"   " if ( %result )\n"   " {\n"   "    %seletedFile = %openFileDlg.file;\n"   " }\n"   " else\n"   " {\n"   "    %selectedFile = \"\";\n"   " }\n\n"   " // Cleanup\n"   " %openFileDlg.delete();\n\n\n"   "@endtsexample\n\n"   "@note FileDialog and its related classes are only availble in a Tools build of Torque.\n\n"   "@see FileDialog\n"   "@see SaveFileDialog\n"   "@see FileObject\n"   "@ingroup FileSystem\n");OpenFileDialog::OpenFileDialog(){   // Default File Must Exist   mData.mStyle = FileDialogData::FDS_OPEN | FileDialogData::FDS_MUSTEXIST;}OpenFileDialog::~OpenFileDialog(){   mMustExist = true;   mMultipleFiles = false;}IMPLEMENT_CONOBJECT(OpenFileDialog);//-----------------------------------------------------------------------------// Console Properties//-----------------------------------------------------------------------------void OpenFileDialog::initPersistFields(){   addProtectedField("MustExist", TypeBool, Offset(mMustExist, OpenFileDialog), &setMustExist, &getMustExist, "True/False whether the file returned must exist or not" );   addProtectedField("MultipleFiles", TypeBool, Offset(mMultipleFiles, OpenFileDialog), &setMultipleFiles, &getMultipleFiles, "True/False whether multiple files may be selected and returned or not" );      Parent::initPersistFields();}//-----------------------------------------------------------------------------// File Must Exist - Boolean//-----------------------------------------------------------------------------bool OpenFileDialog::setMustExist( void *object, const char *index, const char *data ){   bool bMustExist = dAtob( data );   OpenFileDialog *pDlg = static_cast<OpenFileDialog*>( object );      if( bMustExist )      pDlg->mData.mStyle |= FileDialogData::FDS_MUSTEXIST;   else      pDlg->mData.mStyle &= ~FileDialogData::FDS_MUSTEXIST;   return true;};const char* OpenFileDialog::getMustExist(void* obj, const char* data){   OpenFileDialog *pDlg = static_cast<OpenFileDialog*>( obj );   if( pDlg->mData.mStyle & FileDialogData::FDS_MUSTEXIST )      return StringTable->insert("true");   else      return StringTable->insert("false");}//-----------------------------------------------------------------------------// Can Select Multiple Files - Boolean//-----------------------------------------------------------------------------bool OpenFileDialog::setMultipleFiles( void *object, const char *index, const char *data ){   bool bMustExist = dAtob( data );   OpenFileDialog *pDlg = static_cast<OpenFileDialog*>( object );   if( bMustExist )      pDlg->mData.mStyle |= FileDialogData::FDS_MULTIPLEFILES;   else      pDlg->mData.mStyle &= ~FileDialogData::FDS_MULTIPLEFILES;   return true;};const char* OpenFileDialog::getMultipleFiles(void* obj, const char* data){   OpenFileDialog *pDlg = static_cast<OpenFileDialog*>( obj );   if( pDlg->mData.mStyle & FileDialogData::FDS_MULTIPLEFILES )      return StringTable->insert("true");   else      return StringTable->insert("false");}//-----------------------------------------------------------------------------// SaveFileDialog Implementation//-----------------------------------------------------------------------------ConsoleDocClass( SaveFileDialog,   "@brief Derived from FileDialog, this class is responsible for opening a file browser with the intention of saving a file.\n\n"   "The core usage of this dialog is to locate a file in the OS and return the path and name. This does not handle "   "the actual file writing or data manipulation. That functionality is left up to the FileObject class.\n\n"      "@tsexample\n"   " // Create a dialog dedicated to opening file\n"   " %saveFileDlg = new SaveFileDialog()\n"   " {\n"   "    // Only allow for saving of COLLADA files\n"   "    Filters        = \"COLLADA Files (*.dae)|*.dae|\";\n\n"   "    // Default save path to where the WorldEditor last saved\n"   "    DefaultPath    = $pref::WorldEditor::LastPath;\n\n"   "    // No default file specified\n"   "    DefaultFile    = \"\";\n\n"   "    // Do not allow the user to change to a new directory\n"   "    ChangePath     = false;\n\n"   "    // Prompt the user if they are going to overwrite an existing file\n"   "    OverwritePrompt   = true;\n"   " };\n\n"   " // Launch the save file dialog\n"   " %saveFileDlg.Execute();\n\n"   " if ( %result )\n"   " {\n"   "    %seletedFile = %openFileDlg.file;\n"   " }\n"   " else\n"   " {\n"   "    %selectedFile = \"\";\n"   " }\n\n"   " // Cleanup\n"   " %saveFileDlg.delete();\n"   "@endtsexample\n\n"   "@note FileDialog and its related classes are only availble in a Tools build of Torque.\n\n"   "@see FileDialog\n"   "@see OpenFileDialog\n"   "@see FileObject\n"   "@ingroup FileSystem\n");SaveFileDialog::SaveFileDialog(){   // Default File Must Exist   mData.mStyle = FileDialogData::FDS_SAVE | FileDialogData::FDS_OVERWRITEPROMPT;   mOverwritePrompt = true;}SaveFileDialog::~SaveFileDialog(){}IMPLEMENT_CONOBJECT(SaveFileDialog);//-----------------------------------------------------------------------------// Console Properties//-----------------------------------------------------------------------------void SaveFileDialog::initPersistFields(){   addProtectedField("OverwritePrompt", TypeBool, Offset(mOverwritePrompt, SaveFileDialog), &setOverwritePrompt, &getOverwritePrompt, "True/False whether the dialog should prompt before accepting an existing file name" );      Parent::initPersistFields();}//-----------------------------------------------------------------------------// Prompt on Overwrite - Boolean//-----------------------------------------------------------------------------bool SaveFileDialog::setOverwritePrompt( void *object, const char *index, const char *data ){   bool bMustExist = dAtob( data );   SaveFileDialog *pDlg = static_cast<SaveFileDialog*>( object );   if( bMustExist )      pDlg->mData.mStyle |= FileDialogData::FDS_OVERWRITEPROMPT;   else      pDlg->mData.mStyle &= ~FileDialogData::FDS_OVERWRITEPROMPT;   return true;};const char* SaveFileDialog::getOverwritePrompt(void* obj, const char* data){   SaveFileDialog *pDlg = static_cast<SaveFileDialog*>( obj );   if( pDlg->mData.mStyle & FileDialogData::FDS_OVERWRITEPROMPT )      return StringTable->insert("true");   else      return StringTable->insert("false");}//-----------------------------------------------------------------------------// OpenFolderDialog Implementation//-----------------------------------------------------------------------------OpenFolderDialog::OpenFolderDialog(){   mData.mStyle = FileDialogData::FDS_OPEN | FileDialogData::FDS_OVERWRITEPROMPT | FileDialogData::FDS_BROWSEFOLDER;   mMustExistInDir = "";}IMPLEMENT_CONOBJECT(OpenFolderDialog);ConsoleDocClass( OpenFolderDialog,   "@brief OS level dialog used for browsing folder structures.\n\n"   "This is essentially an OpenFileDialog, but only used for returning directory paths, not files.\n\n"   "@note FileDialog and its related classes are only availble in a Tools build of Torque.\n\n"   "@see OpenFileDialog for more details on functionality.\n\n"   "@ingroup FileSystem\n");void OpenFolderDialog::initPersistFields(){   addField("fileMustExist", TypeFilename, Offset(mMustExistInDir, OpenFolderDialog), "File that must be in selected folder for it to be valid");   Parent::initPersistFields();}#endif
 |