1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/fileconf.cpp 
   3 // Purpose:     implementation of wxFileConfig derivation of wxConfig 
   4 // Author:      Vadim Zeitlin 
   6 // Created:     07.04.98 (adapted from appconf.cpp) 
   8 // Copyright:   (c) 1997 Karsten Ballüder   &  Vadim Zeitlin 
   9 //                       Ballueder@usa.net     <zeitlin@dptmaths.ens-cachan.fr> 
  10 // Licence:     wxWindows licence 
  11 /////////////////////////////////////////////////////////////////////////////// 
  13 // ---------------------------------------------------------------------------- 
  15 // ---------------------------------------------------------------------------- 
  17 // For compilers that support precompilation, includes "wx.h". 
  18 #include  "wx/wxprec.h" 
  24 #if wxUSE_CONFIG && wxUSE_FILECONFIG 
  27     #include  "wx/dynarray.h" 
  28     #include  "wx/string.h" 
  32     #include  "wx/utils.h"    // for wxGetHomeDir 
  34         #include  "wx/stream.h" 
  35     #endif // wxUSE_STREAMS 
  39 #include  "wx/textfile.h" 
  40 #include  "wx/memtext.h" 
  41 #include  "wx/config.h" 
  42 #include  "wx/fileconf.h" 
  43 #include  "wx/filefn.h" 
  45 #if defined(__WXMAC__) 
  46     #include  "wx/mac/private.h"  // includes mac headers 
  47     #include  "wx/filename.h"     // for MacSetTypeAndCreator 
  50 #if defined(__WXMSW__) 
  51     #include "wx/msw/private.h" 
  61 // ---------------------------------------------------------------------------- 
  63 // ---------------------------------------------------------------------------- 
  64 #define CONST_CAST ((wxFileConfig *)this)-> 
  66 // ---------------------------------------------------------------------------- 
  68 // ---------------------------------------------------------------------------- 
  74 #define FILECONF_TRACE_MASK _T("fileconf") 
  76 // ---------------------------------------------------------------------------- 
  77 // global functions declarations 
  78 // ---------------------------------------------------------------------------- 
  80 // compare functions for sorting the arrays 
  81 static int LINKAGEMODE 
CompareEntries(wxFileConfigEntry 
*p1
, wxFileConfigEntry 
*p2
); 
  82 static int LINKAGEMODE 
CompareGroups(wxFileConfigGroup 
*p1
, wxFileConfigGroup 
*p2
); 
  85 static wxString 
FilterInValue(const wxString
& str
); 
  86 static wxString 
FilterOutValue(const wxString
& str
); 
  88 static wxString 
FilterInEntryName(const wxString
& str
); 
  89 static wxString 
FilterOutEntryName(const wxString
& str
); 
  91 // get the name to use in wxFileConfig ctor 
  92 static wxString 
GetAppName(const wxString
& appname
); 
  94 // ============================================================================ 
  96 // ============================================================================ 
  98 // ---------------------------------------------------------------------------- 
  99 // "template" array types 
 100 // ---------------------------------------------------------------------------- 
 102 #ifdef WXMAKINGDLL_BASE 
 103     WX_DEFINE_SORTED_USER_EXPORTED_ARRAY(wxFileConfigEntry 
*, ArrayEntries
, 
 105     WX_DEFINE_SORTED_USER_EXPORTED_ARRAY(wxFileConfigGroup 
*, ArrayGroups
, 
 108     WX_DEFINE_SORTED_ARRAY(wxFileConfigEntry 
*, ArrayEntries
); 
 109     WX_DEFINE_SORTED_ARRAY(wxFileConfigGroup 
*, ArrayGroups
); 
 112 // ---------------------------------------------------------------------------- 
 113 // wxFileConfigLineList 
 114 // ---------------------------------------------------------------------------- 
 116 // we store all lines of the local config file as a linked list in memory 
 117 class wxFileConfigLineList
 
 120   void SetNext(wxFileConfigLineList 
*pNext
)  { m_pNext 
= pNext
; } 
 121   void SetPrev(wxFileConfigLineList 
*pPrev
)  { m_pPrev 
= pPrev
; } 
 124   wxFileConfigLineList(const wxString
& str
, 
 125                        wxFileConfigLineList 
*pNext 
= NULL
) : m_strLine(str
) 
 126     { SetNext(pNext
); SetPrev(NULL
); } 
 128   // next/prev nodes in the linked list 
 129   wxFileConfigLineList 
*Next() const { return m_pNext
;  } 
 130   wxFileConfigLineList 
*Prev() const { return m_pPrev
;  } 
 132   // get/change lines text 
 133   void SetText(const wxString
& str
) { m_strLine 
= str
;  } 
 134   const wxString
& Text() const { return m_strLine
; } 
 137   wxString  m_strLine
;                  // line contents 
 138   wxFileConfigLineList 
*m_pNext
,        // next node 
 139                        *m_pPrev
;        // previous one 
 141     DECLARE_NO_COPY_CLASS(wxFileConfigLineList
) 
 144 // ---------------------------------------------------------------------------- 
 145 // wxFileConfigEntry: a name/value pair 
 146 // ---------------------------------------------------------------------------- 
 148 class wxFileConfigEntry
 
 151   wxFileConfigGroup 
*m_pParent
; // group that contains us 
 153   wxString      m_strName
,      // entry name 
 155   bool          m_bImmutable
:1, // can be overriden locally? 
 156                 m_bHasValue
:1;  // set after first call to SetValue() 
 158   int           m_nLine
;        // used if m_pLine == NULL only 
 160   // pointer to our line in the linked list or NULL if it was found in global 
 161   // file (which we don't modify) 
 162   wxFileConfigLineList 
*m_pLine
; 
 165   wxFileConfigEntry(wxFileConfigGroup 
*pParent
, 
 166                     const wxString
& strName
, int nLine
); 
 169   const wxString
& Name()        const { return m_strName
;    } 
 170   const wxString
& Value()       const { return m_strValue
;   } 
 171   wxFileConfigGroup 
*Group()    const { return m_pParent
;    } 
 172   bool            IsImmutable() const { return m_bImmutable
; } 
 173   bool            IsLocal()     const { return m_pLine 
!= 0; } 
 174   int             Line()        const { return m_nLine
;      } 
 175   wxFileConfigLineList 
* 
 176                   GetLine()     const { return m_pLine
;      } 
 178   // modify entry attributes 
 179   void SetValue(const wxString
& strValue
, bool bUser 
= true); 
 180   void SetLine(wxFileConfigLineList 
*pLine
); 
 182     DECLARE_NO_COPY_CLASS(wxFileConfigEntry
) 
 185 // ---------------------------------------------------------------------------- 
 186 // wxFileConfigGroup: container of entries and other groups 
 187 // ---------------------------------------------------------------------------- 
 189 class wxFileConfigGroup
 
 192   wxFileConfig 
*m_pConfig
;          // config object we belong to 
 193   wxFileConfigGroup  
*m_pParent
;    // parent group (NULL for root group) 
 194   ArrayEntries  m_aEntries
;         // entries in this group 
 195   ArrayGroups   m_aSubgroups
;       // subgroups 
 196   wxString      m_strName
;          // group's name 
 197   wxFileConfigLineList 
*m_pLine
;    // pointer to our line in the linked list 
 198   wxFileConfigEntry 
*m_pLastEntry
;  // last entry/subgroup of this group in the 
 199   wxFileConfigGroup 
*m_pLastGroup
;  // local file (we insert new ones after it) 
 201   // DeleteSubgroupByName helper 
 202   bool DeleteSubgroup(wxFileConfigGroup 
*pGroup
); 
 205   void UpdateGroupAndSubgroupsLines(); 
 209   wxFileConfigGroup(wxFileConfigGroup 
*pParent
, const wxString
& strName
, wxFileConfig 
*); 
 211   // dtor deletes all entries and subgroups also 
 212   ~wxFileConfigGroup(); 
 215   const wxString
& Name()    const { return m_strName
; } 
 216   wxFileConfigGroup    
*Parent()  const { return m_pParent
; } 
 217   wxFileConfig   
*Config()  const { return m_pConfig
; } 
 219   const ArrayEntries
& Entries() const { return m_aEntries
;   } 
 220   const ArrayGroups
&  Groups()  const { return m_aSubgroups
; } 
 221   bool  IsEmpty() const { return Entries().IsEmpty() && Groups().IsEmpty(); } 
 223   // find entry/subgroup (NULL if not found) 
 224   wxFileConfigGroup 
*FindSubgroup(const wxChar 
*szName
) const; 
 225   wxFileConfigEntry 
*FindEntry   (const wxChar 
*szName
) const; 
 227   // delete entry/subgroup, return false if doesn't exist 
 228   bool DeleteSubgroupByName(const wxChar 
*szName
); 
 229   bool DeleteEntry(const wxChar 
*szName
); 
 231   // create new entry/subgroup returning pointer to newly created element 
 232   wxFileConfigGroup 
*AddSubgroup(const wxString
& strName
); 
 233   wxFileConfigEntry 
*AddEntry   (const wxString
& strName
, int nLine 
= wxNOT_FOUND
); 
 235   void SetLine(wxFileConfigLineList 
*pLine
); 
 237   // rename: no checks are done to ensure that the name is unique! 
 238   void Rename(const wxString
& newName
); 
 241   wxString 
GetFullName() const; 
 243   // get the last line belonging to an entry/subgroup of this group 
 244   wxFileConfigLineList 
*GetGroupLine();     // line which contains [group] 
 245   wxFileConfigLineList 
*GetLastEntryLine(); // after which our subgroups start 
 246   wxFileConfigLineList 
*GetLastGroupLine(); // after which the next group starts 
 248   // called by entries/subgroups when they're created/deleted 
 249   void SetLastEntry(wxFileConfigEntry 
*pEntry
); 
 250   void SetLastGroup(wxFileConfigGroup 
*pGroup
) 
 251     { m_pLastGroup 
= pGroup
; } 
 253   DECLARE_NO_COPY_CLASS(wxFileConfigGroup
) 
 256 // ============================================================================ 
 258 // ============================================================================ 
 260 // ---------------------------------------------------------------------------- 
 262 // ---------------------------------------------------------------------------- 
 263 wxString 
wxFileConfig::GetGlobalDir() 
 267 #ifdef __VMS__ // Note if __VMS is defined __UNIX is also defined 
 268     strDir 
= wxT("sys$manager:"); 
 269 #elif defined(__WXMAC__) 
 270     strDir 
= wxMacFindFolder(  (short) kOnSystemDisk
, kPreferencesFolderType
, kDontCreateFolder 
) ; 
 271 #elif defined( __UNIX__ ) 
 272     strDir 
= wxT("/etc/"); 
 273 #elif defined(__OS2__) 
 274     ULONG aulSysInfo
[QSV_MAX
] = {0}; 
 278     rc 
= DosQuerySysInfo( 1L, QSV_MAX
, (PVOID
)aulSysInfo
, sizeof(ULONG
)*QSV_MAX
); 
 281         drive 
= aulSysInfo
[QSV_BOOT_DRIVE 
- 1]; 
 282         strDir
.Printf(wxT("%c:\\OS2\\"), 'A'+drive
-1); 
 284 #elif defined(__WXSTUBS__) 
 285     wxFAIL_MSG( wxT("TODO") ); 
 286 #elif defined(__DOS__) 
 287     // There's no such thing as global cfg dir in MS-DOS, let's return 
 288     // current directory (FIXME_MGL?) 
 290 #elif defined(__WXWINCE__) 
 291     strDir 
= wxT("\\Windows\\"); 
 294     wxChar szWinDir
[MAX_PATH
]; 
 295     ::GetWindowsDirectory(szWinDir
, MAX_PATH
); 
 299 #endif // Unix/Windows 
 304 wxString 
wxFileConfig::GetLocalDir() 
 308 #if defined(__WXMAC__) || defined(__DOS__) 
 309     // no local dir concept on Mac OS 9 or MS-DOS 
 310     strDir 
<< GetGlobalDir() ; 
 312     wxGetHomeDir(&strDir
); 
 316             (strDir
.Last() != wxT('/')) 
 318             && (strDir
.Last() != wxT(']')) 
 323         if (strDir
.Last() != wxT('\\')) 
 331 wxString 
wxFileConfig::GetGlobalFileName(const wxChar 
*szFile
) 
 333     wxString str 
= GetGlobalDir(); 
 336     if ( wxStrchr(szFile
, wxT('.')) == NULL 
) 
 337 #if defined( __WXMAC__ ) 
 338         str 
<< wxT(" Preferences") ; 
 339 #elif defined( __UNIX__ ) 
 348 wxString 
wxFileConfig::GetLocalFileName(const wxChar 
*szFile
) 
 351     // On VMS I saw the problem that the home directory was appended 
 352     // twice for the configuration file. Does that also happen for 
 354     wxString str 
= wxT( '.' ); 
 356     wxString str 
= GetLocalDir(); 
 359 #if defined( __UNIX__ ) && !defined( __VMS ) && !defined( __WXMAC__ ) 
 365 #if defined(__WINDOWS__) || defined(__DOS__) 
 366     if ( wxStrchr(szFile
, wxT('.')) == NULL 
) 
 371     str 
<< wxT(" Preferences") ; 
 377 // ---------------------------------------------------------------------------- 
 379 // ---------------------------------------------------------------------------- 
 381 void wxFileConfig::Init() 
 384     m_pRootGroup    
= new wxFileConfigGroup(NULL
, wxEmptyString
, this); 
 389     // It's not an error if (one of the) file(s) doesn't exist. 
 391     // parse the global file 
 392     if ( !m_strGlobalFile
.empty() && wxFile::Exists(m_strGlobalFile
) ) 
 394         wxTextFile 
fileGlobal(m_strGlobalFile
); 
 396         if ( fileGlobal
.Open(*m_conv
/*ignored in ANSI build*/) ) 
 398             Parse(fileGlobal
, false /* global */); 
 403             wxLogWarning(_("can't open global configuration file '%s'."), m_strGlobalFile
.c_str()); 
 407     // parse the local file 
 408     if ( !m_strLocalFile
.empty() && wxFile::Exists(m_strLocalFile
) ) 
 410         wxTextFile 
fileLocal(m_strLocalFile
); 
 411         if ( fileLocal
.Open(*m_conv
/*ignored in ANSI build*/) ) 
 413             Parse(fileLocal
, true /* local */); 
 418             wxLogWarning(_("can't open user configuration file '%s'."),  m_strLocalFile
.c_str() ); 
 425 // constructor supports creation of wxFileConfig objects of any type 
 426 wxFileConfig::wxFileConfig(const wxString
& appName
, const wxString
& vendorName
, 
 427                            const wxString
& strLocal
, const wxString
& strGlobal
, 
 429                            const wxMBConv
& conv
) 
 430             : wxConfigBase(::GetAppName(appName
), vendorName
, 
 433               m_strLocalFile(strLocal
), m_strGlobalFile(strGlobal
), 
 436     // Make up names for files if empty 
 437     if ( m_strLocalFile
.empty() && (style 
& wxCONFIG_USE_LOCAL_FILE
) ) 
 438         m_strLocalFile 
= GetLocalFileName(GetAppName()); 
 440     if ( m_strGlobalFile
.empty() && (style 
& wxCONFIG_USE_GLOBAL_FILE
) ) 
 441         m_strGlobalFile 
= GetGlobalFileName(GetAppName()); 
 443     // Check if styles are not supplied, but filenames are, in which case 
 444     // add the correct styles. 
 445     if ( !m_strLocalFile
.empty() ) 
 446         SetStyle(GetStyle() | wxCONFIG_USE_LOCAL_FILE
); 
 448     if ( !m_strGlobalFile
.empty() ) 
 449         SetStyle(GetStyle() | wxCONFIG_USE_GLOBAL_FILE
); 
 451     // if the path is not absolute, prepend the standard directory to it 
 452     // UNLESS wxCONFIG_USE_RELATIVE_PATH style is set 
 453     if ( !(style 
& wxCONFIG_USE_RELATIVE_PATH
) ) 
 455         if ( !m_strLocalFile
.empty() && !wxIsAbsolutePath(m_strLocalFile
) ) 
 457             const wxString strLocalOrig 
= m_strLocalFile
; 
 458             m_strLocalFile 
= GetLocalDir(); 
 459             m_strLocalFile 
<< strLocalOrig
; 
 462         if ( !m_strGlobalFile
.empty() && !wxIsAbsolutePath(m_strGlobalFile
) ) 
 464             const wxString strGlobalOrig 
= m_strGlobalFile
; 
 465             m_strGlobalFile 
= GetGlobalDir(); 
 466             m_strGlobalFile 
<< strGlobalOrig
; 
 477 wxFileConfig::wxFileConfig(wxInputStream 
&inStream
, const wxMBConv
& conv
) 
 478             : m_conv(conv
.Clone()) 
 480     // always local_file when this constructor is called (?) 
 481     SetStyle(GetStyle() | wxCONFIG_USE_LOCAL_FILE
); 
 484     m_pRootGroup    
= new wxFileConfigGroup(NULL
, wxEmptyString
, this); 
 489     // translate everything to the current (platform-dependent) line 
 490     // termination character 
 498             inStream
.Read(buf
, WXSIZEOF(buf
)-1);  // leave room for the NULL 
 500             const wxStreamError err 
= inStream
.GetLastError(); 
 502             if ( err 
!= wxSTREAM_NO_ERROR 
&& err 
!= wxSTREAM_EOF 
) 
 504                 wxLogError(_("Error reading config options.")); 
 508             // FIXME: this is broken because if we have part of multibyte 
 509             //        character in the buffer (and another part hasn't been 
 510             //        read yet) we're going to lose data because of conversion 
 512             buf
[inStream
.LastRead()] = '\0'; 
 513             strTmp 
+= conv
.cMB2WX(buf
); 
 515         while ( !inStream
.Eof() ); 
 517         strTrans 
= wxTextBuffer::Translate(strTmp
); 
 520     wxMemoryText memText
; 
 522     // Now we can add the text to the memory text. To do this we extract line 
 523     // by line from the translated string, until we've reached the end. 
 525     // VZ: all this is horribly inefficient, we should do the translation on 
 526     //     the fly in one pass saving both memory and time (TODO) 
 528     const wxChar 
*pEOL 
= wxTextBuffer::GetEOL(wxTextBuffer::typeDefault
); 
 529     const size_t EOLLen 
= wxStrlen(pEOL
); 
 531     int posLineStart 
= strTrans
.Find(pEOL
); 
 532     while ( posLineStart 
!= -1 ) 
 534         wxString 
line(strTrans
.Left(posLineStart
)); 
 536         memText
.AddLine(line
); 
 538         strTrans 
= strTrans
.Mid(posLineStart 
+ EOLLen
); 
 540         posLineStart 
= strTrans
.Find(pEOL
); 
 543     // also add whatever we have left in the translated string. 
 544     if ( !strTrans
.empty() ) 
 545         memText
.AddLine(strTrans
); 
 547     // Finally we can parse it all. 
 548     Parse(memText
, true /* local */); 
 554 #endif // wxUSE_STREAMS 
 556 void wxFileConfig::CleanUp() 
 560     wxFileConfigLineList 
*pCur 
= m_linesHead
; 
 561     while ( pCur 
!= NULL 
) { 
 562         wxFileConfigLineList 
*pNext 
= pCur
->Next(); 
 568 wxFileConfig::~wxFileConfig() 
 577 // ---------------------------------------------------------------------------- 
 578 // parse a config file 
 579 // ---------------------------------------------------------------------------- 
 581 void wxFileConfig::Parse(const wxTextBuffer
& buffer
, bool bLocal
) 
 583   const wxChar 
*pStart
; 
 587   size_t nLineCount 
= buffer
.GetLineCount(); 
 589   for ( size_t n 
= 0; n 
< nLineCount
; n
++ ) 
 593     // add the line to linked list 
 596       LineListAppend(strLine
); 
 598       // let the root group have its start line as well 
 601         m_pCurrentGroup
->SetLine(m_linesTail
); 
 606     // skip leading spaces 
 607     for ( pStart 
= strLine
; wxIsspace(*pStart
); pStart
++ ) 
 610     // skip blank/comment lines 
 611     if ( *pStart 
== wxT('\0')|| *pStart 
== wxT(';') || *pStart 
== wxT('#') ) 
 614     if ( *pStart 
== wxT('[') ) {          // a new group 
 617       while ( *++pEnd 
!= wxT(']') ) { 
 618         if ( *pEnd 
== wxT('\\') ) { 
 619             // the next char is escaped, so skip it even if it is ']' 
 623         if ( *pEnd 
== wxT('\n') || *pEnd 
== wxT('\0') ) { 
 624             // we reached the end of line, break out of the loop 
 629       if ( *pEnd 
!= wxT(']') ) { 
 630         wxLogError(_("file '%s': unexpected character %c at line %d."), 
 631                    buffer
.GetName(), *pEnd
, n 
+ 1); 
 632         continue; // skip this line 
 635       // group name here is always considered as abs path 
 638       strGroup 
<< wxCONFIG_PATH_SEPARATOR
 
 639                << FilterInEntryName(wxString(pStart
, pEnd 
- pStart
)); 
 641       // will create it if doesn't yet exist 
 646         if ( m_pCurrentGroup
->Parent() ) 
 647           m_pCurrentGroup
->Parent()->SetLastGroup(m_pCurrentGroup
); 
 648         m_pCurrentGroup
->SetLine(m_linesTail
); 
 651       // check that there is nothing except comments left on this line 
 653       while ( *++pEnd 
!= wxT('\0') && bCont 
) { 
 662             // ignore whitespace ('\n' impossible here) 
 666             wxLogWarning(_("file '%s', line %d: '%s' ignored after group header."), 
 667                          buffer
.GetName(), n 
+ 1, pEnd
); 
 674       while ( *pEnd 
&& *pEnd 
!= wxT('=') /* && !wxIsspace(*pEnd)*/ ) { 
 675         if ( *pEnd 
== wxT('\\') ) { 
 676           // next character may be space or not - still take it because it's 
 677           // quoted (unless there is nothing) 
 680             // the error message will be given below anyhow 
 688       wxString 
strKey(FilterInEntryName(wxString(pStart
, pEnd
).Trim())); 
 691       while ( wxIsspace(*pEnd
) ) 
 694       if ( *pEnd
++ != wxT('=') ) { 
 695         wxLogError(_("file '%s', line %d: '=' expected."), 
 696                    buffer
.GetName(), n 
+ 1); 
 699         wxFileConfigEntry 
*pEntry 
= m_pCurrentGroup
->FindEntry(strKey
); 
 701         if ( pEntry 
== NULL 
) { 
 703           pEntry 
= m_pCurrentGroup
->AddEntry(strKey
, n
); 
 706           if ( bLocal 
&& pEntry
->IsImmutable() ) { 
 707             // immutable keys can't be changed by user 
 708             wxLogWarning(_("file '%s', line %d: value for immutable key '%s' ignored."), 
 709                          buffer
.GetName(), n 
+ 1, strKey
.c_str()); 
 712           // the condition below catches the cases (a) and (b) but not (c): 
 713           //  (a) global key found second time in global file 
 714           //  (b) key found second (or more) time in local file 
 715           //  (c) key from global file now found in local one 
 716           // which is exactly what we want. 
 717           else if ( !bLocal 
|| pEntry
->IsLocal() ) { 
 718             wxLogWarning(_("file '%s', line %d: key '%s' was first found at line %d."), 
 719                          buffer
.GetName(), n 
+ 1, strKey
.c_str(), pEntry
->Line()); 
 725           pEntry
->SetLine(m_linesTail
); 
 728         while ( wxIsspace(*pEnd
) ) 
 731         wxString value 
= pEnd
; 
 732         if ( !(GetStyle() & wxCONFIG_USE_NO_ESCAPE_CHARACTERS
) ) 
 733             value 
= FilterInValue(value
); 
 735         pEntry
->SetValue(value
, false); 
 741 // ---------------------------------------------------------------------------- 
 743 // ---------------------------------------------------------------------------- 
 745 void wxFileConfig::SetRootPath() 
 748     m_pCurrentGroup 
= m_pRootGroup
; 
 752 wxFileConfig::DoSetPath(const wxString
& strPath
, bool createMissingComponents
) 
 754     wxArrayString aParts
; 
 756     if ( strPath
.empty() ) { 
 761     if ( strPath
[0] == wxCONFIG_PATH_SEPARATOR 
) { 
 763         wxSplitPath(aParts
, strPath
); 
 766         // relative path, combine with current one 
 767         wxString strFullPath 
= m_strPath
; 
 768         strFullPath 
<< wxCONFIG_PATH_SEPARATOR 
<< strPath
; 
 769         wxSplitPath(aParts
, strFullPath
); 
 772     // change current group 
 774     m_pCurrentGroup 
= m_pRootGroup
; 
 775     for ( n 
= 0; n 
< aParts
.Count(); n
++ ) { 
 776         wxFileConfigGroup 
*pNextGroup 
= m_pCurrentGroup
->FindSubgroup(aParts
[n
]); 
 777         if ( pNextGroup 
== NULL 
) 
 779             if ( !createMissingComponents 
) 
 782             pNextGroup 
= m_pCurrentGroup
->AddSubgroup(aParts
[n
]); 
 785         m_pCurrentGroup 
= pNextGroup
; 
 788     // recombine path parts in one variable 
 790     for ( n 
= 0; n 
< aParts
.Count(); n
++ ) { 
 791         m_strPath 
<< wxCONFIG_PATH_SEPARATOR 
<< aParts
[n
]; 
 797 void wxFileConfig::SetPath(const wxString
& strPath
) 
 799     DoSetPath(strPath
, true /* create missing path components */); 
 802 // ---------------------------------------------------------------------------- 
 804 // ---------------------------------------------------------------------------- 
 806 bool wxFileConfig::GetFirstGroup(wxString
& str
, long& lIndex
) const 
 809     return GetNextGroup(str
, lIndex
); 
 812 bool wxFileConfig::GetNextGroup (wxString
& str
, long& lIndex
) const 
 814     if ( size_t(lIndex
) < m_pCurrentGroup
->Groups().Count() ) { 
 815         str 
= m_pCurrentGroup
->Groups()[(size_t)lIndex
++]->Name(); 
 822 bool wxFileConfig::GetFirstEntry(wxString
& str
, long& lIndex
) const 
 825     return GetNextEntry(str
, lIndex
); 
 828 bool wxFileConfig::GetNextEntry (wxString
& str
, long& lIndex
) const 
 830     if ( size_t(lIndex
) < m_pCurrentGroup
->Entries().Count() ) { 
 831         str 
= m_pCurrentGroup
->Entries()[(size_t)lIndex
++]->Name(); 
 838 size_t wxFileConfig::GetNumberOfEntries(bool bRecursive
) const 
 840     size_t n 
= m_pCurrentGroup
->Entries().Count(); 
 842         wxFileConfigGroup 
*pOldCurrentGroup 
= m_pCurrentGroup
; 
 843         size_t nSubgroups 
= m_pCurrentGroup
->Groups().Count(); 
 844         for ( size_t nGroup 
= 0; nGroup 
< nSubgroups
; nGroup
++ ) { 
 845             CONST_CAST m_pCurrentGroup 
= m_pCurrentGroup
->Groups()[nGroup
]; 
 846             n 
+= GetNumberOfEntries(true); 
 847             CONST_CAST m_pCurrentGroup 
= pOldCurrentGroup
; 
 854 size_t wxFileConfig::GetNumberOfGroups(bool bRecursive
) const 
 856     size_t n 
= m_pCurrentGroup
->Groups().Count(); 
 858         wxFileConfigGroup 
*pOldCurrentGroup 
= m_pCurrentGroup
; 
 859         size_t nSubgroups 
= m_pCurrentGroup
->Groups().Count(); 
 860         for ( size_t nGroup 
= 0; nGroup 
< nSubgroups
; nGroup
++ ) { 
 861             CONST_CAST m_pCurrentGroup 
= m_pCurrentGroup
->Groups()[nGroup
]; 
 862             n 
+= GetNumberOfGroups(true); 
 863             CONST_CAST m_pCurrentGroup 
= pOldCurrentGroup
; 
 870 // ---------------------------------------------------------------------------- 
 871 // tests for existence 
 872 // ---------------------------------------------------------------------------- 
 874 bool wxFileConfig::HasGroup(const wxString
& strName
) const 
 876     // special case: DoSetPath("") does work as it's equivalent to DoSetPath("/") 
 877     // but there is no group with empty name so treat this separately 
 878     if ( strName
.empty() ) 
 881     const wxString pathOld 
= GetPath(); 
 883     wxFileConfig 
*self 
= wx_const_cast(wxFileConfig 
*, this); 
 885         rc 
= self
->DoSetPath(strName
, false /* don't create missing components */); 
 887     self
->SetPath(pathOld
); 
 892 bool wxFileConfig::HasEntry(const wxString
& strName
) const 
 894     wxConfigPathChanger 
path(this, strName
); 
 896     wxFileConfigEntry 
*pEntry 
= m_pCurrentGroup
->FindEntry(path
.Name()); 
 897     return pEntry 
!= NULL
; 
 900 // ---------------------------------------------------------------------------- 
 902 // ---------------------------------------------------------------------------- 
 904 bool wxFileConfig::DoReadString(const wxString
& key
, wxString
* pStr
) const 
 906     wxConfigPathChanger 
path(this, key
); 
 908     wxFileConfigEntry 
*pEntry 
= m_pCurrentGroup
->FindEntry(path
.Name()); 
 909     if (pEntry 
== NULL
) { 
 913     *pStr 
= pEntry
->Value(); 
 918 bool wxFileConfig::DoReadLong(const wxString
& key
, long *pl
) const 
 921     if ( !Read(key
, &str
) ) 
 924     // extra spaces shouldn't prevent us from reading numeric values 
 927     return str
.ToLong(pl
); 
 930 bool wxFileConfig::DoWriteString(const wxString
& key
, const wxString
& szValue
) 
 932     wxConfigPathChanger     
path(this, key
); 
 933     wxString                strName 
= path
.Name(); 
 935     wxLogTrace( FILECONF_TRACE_MASK
, 
 936                 _T("  Writing String '%s' = '%s' to Group '%s'"), 
 941     if ( strName
.empty() ) 
 943             // setting the value of a group is an error 
 945         wxASSERT_MSG( szValue
.empty(), wxT("can't set value of a group!") ); 
 947             // ... except if it's empty in which case it's a way to force it's creation 
 949         wxLogTrace( FILECONF_TRACE_MASK
, 
 950                     _T("  Creating group %s"), 
 951                     m_pCurrentGroup
->Name().c_str() ); 
 955             // this will add a line for this group if it didn't have it before 
 957         (void)m_pCurrentGroup
->GetGroupLine(); 
 961         // writing an entry check that the name is reasonable 
 962         if ( strName
[0u] == wxCONFIG_IMMUTABLE_PREFIX 
) 
 964             wxLogError( _("Config entry name cannot start with '%c'."), 
 965                         wxCONFIG_IMMUTABLE_PREFIX
); 
 969         wxFileConfigEntry   
*pEntry 
= m_pCurrentGroup
->FindEntry(strName
); 
 973             wxLogTrace( FILECONF_TRACE_MASK
, 
 974                         _T("  Adding Entry %s"), 
 976             pEntry 
= m_pCurrentGroup
->AddEntry(strName
); 
 979         wxLogTrace( FILECONF_TRACE_MASK
, 
 980                     _T("  Setting value %s"), 
 982         pEntry
->SetValue(szValue
); 
 990 bool wxFileConfig::DoWriteLong(const wxString
& key
, long lValue
) 
 992   return Write(key
, wxString::Format(_T("%ld"), lValue
)); 
 995 bool wxFileConfig::Flush(bool /* bCurrentOnly */) 
 997   if ( !IsDirty() || !m_strLocalFile 
) 
1000   // set the umask if needed 
1001   wxCHANGE_UMASK(m_umask
); 
1003   wxTempFile 
file(m_strLocalFile
); 
1005   if ( !file
.IsOpened() ) 
1007     wxLogError(_("can't open user configuration file.")); 
1011   // write all strings to file 
1012   for ( wxFileConfigLineList 
*p 
= m_linesHead
; p 
!= NULL
; p 
= p
->Next() ) 
1014     wxString line 
= p
->Text(); 
1015     line 
+= wxTextFile::GetEOL(); 
1016     if ( !file
.Write(line
, *m_conv
) ) 
1018       wxLogError(_("can't write user configuration file.")); 
1023   if ( !file
.Commit() ) 
1025       wxLogError(_("Failed to update user configuration file.")); 
1032 #if defined(__WXMAC__) 
1033   wxFileName(m_strLocalFile
).MacSetTypeAndCreator('TEXT', 'ttxt'); 
1041 bool wxFileConfig::Save(wxOutputStream
& os
, const wxMBConv
& conv
) 
1043     // save unconditionally, even if not dirty 
1044     for ( wxFileConfigLineList 
*p 
= m_linesHead
; p 
!= NULL
; p 
= p
->Next() ) 
1046         wxString line 
= p
->Text(); 
1047         line 
+= wxTextFile::GetEOL(); 
1049         wxCharBuffer 
buf(line
.mb_str(conv
)); 
1050         if ( !os
.Write(buf
, strlen(buf
)) ) 
1052             wxLogError(_("Error saving user configuration data.")); 
1063 #endif // wxUSE_STREAMS 
1065 // ---------------------------------------------------------------------------- 
1066 // renaming groups/entries 
1067 // ---------------------------------------------------------------------------- 
1069 bool wxFileConfig::RenameEntry(const wxString
& oldName
, 
1070                                const wxString
& newName
) 
1072     wxASSERT_MSG( !wxStrchr(oldName
, wxCONFIG_PATH_SEPARATOR
), 
1073                    _T("RenameEntry(): paths are not supported") ); 
1075     // check that the entry exists 
1076     wxFileConfigEntry 
*oldEntry 
= m_pCurrentGroup
->FindEntry(oldName
); 
1080     // check that the new entry doesn't already exist 
1081     if ( m_pCurrentGroup
->FindEntry(newName
) ) 
1084     // delete the old entry, create the new one 
1085     wxString value 
= oldEntry
->Value(); 
1086     if ( !m_pCurrentGroup
->DeleteEntry(oldName
) ) 
1091     wxFileConfigEntry 
*newEntry 
= m_pCurrentGroup
->AddEntry(newName
); 
1092     newEntry
->SetValue(value
); 
1097 bool wxFileConfig::RenameGroup(const wxString
& oldName
, 
1098                                const wxString
& newName
) 
1100     // check that the group exists 
1101     wxFileConfigGroup 
*group 
= m_pCurrentGroup
->FindSubgroup(oldName
); 
1105     // check that the new group doesn't already exist 
1106     if ( m_pCurrentGroup
->FindSubgroup(newName
) ) 
1109     group
->Rename(newName
); 
1116 // ---------------------------------------------------------------------------- 
1117 // delete groups/entries 
1118 // ---------------------------------------------------------------------------- 
1120 bool wxFileConfig::DeleteEntry(const wxString
& key
, bool bGroupIfEmptyAlso
) 
1122   wxConfigPathChanger 
path(this, key
); 
1124   if ( !m_pCurrentGroup
->DeleteEntry(path
.Name()) ) 
1129   if ( bGroupIfEmptyAlso 
&& m_pCurrentGroup
->IsEmpty() ) { 
1130     if ( m_pCurrentGroup 
!= m_pRootGroup 
) { 
1131       wxFileConfigGroup 
*pGroup 
= m_pCurrentGroup
; 
1132       SetPath(wxT(".."));  // changes m_pCurrentGroup! 
1133       m_pCurrentGroup
->DeleteSubgroupByName(pGroup
->Name()); 
1135     //else: never delete the root group 
1141 bool wxFileConfig::DeleteGroup(const wxString
& key
) 
1143   wxConfigPathChanger 
path(this, key
); 
1145   if ( !m_pCurrentGroup
->DeleteSubgroupByName(path
.Name()) ) 
1153 bool wxFileConfig::DeleteAll() 
1157   if ( !m_strLocalFile
.empty() ) 
1159       if ( wxFile::Exists(m_strLocalFile
) && wxRemove(m_strLocalFile
) == -1 ) 
1161           wxLogSysError(_("can't delete user configuration file '%s'"), 
1162                         m_strLocalFile
.c_str()); 
1172 // ---------------------------------------------------------------------------- 
1173 // linked list functions 
1174 // ---------------------------------------------------------------------------- 
1176     // append a new line to the end of the list 
1178 wxFileConfigLineList 
*wxFileConfig::LineListAppend(const wxString
& str
) 
1180     wxLogTrace( FILECONF_TRACE_MASK
, 
1181                 _T("    ** Adding Line '%s'"), 
1183     wxLogTrace( FILECONF_TRACE_MASK
, 
1185                 ((m_linesHead
) ? m_linesHead
->Text().c_str() : wxEmptyString
) ); 
1186     wxLogTrace( FILECONF_TRACE_MASK
, 
1188                 ((m_linesTail
) ? m_linesTail
->Text().c_str() : wxEmptyString
) ); 
1190     wxFileConfigLineList 
*pLine 
= new wxFileConfigLineList(str
); 
1192     if ( m_linesTail 
== NULL 
) 
1195         m_linesHead 
= pLine
; 
1200         m_linesTail
->SetNext(pLine
); 
1201         pLine
->SetPrev(m_linesTail
); 
1204     m_linesTail 
= pLine
; 
1206     wxLogTrace( FILECONF_TRACE_MASK
, 
1208                 ((m_linesHead
) ? m_linesHead
->Text().c_str() : wxEmptyString
) ); 
1209     wxLogTrace( FILECONF_TRACE_MASK
, 
1211                 ((m_linesTail
) ? m_linesTail
->Text().c_str() : wxEmptyString
) ); 
1216 // insert a new line after the given one or in the very beginning if !pLine 
1217 wxFileConfigLineList 
*wxFileConfig::LineListInsert(const wxString
& str
, 
1218                                                    wxFileConfigLineList 
*pLine
) 
1220     wxLogTrace( FILECONF_TRACE_MASK
, 
1221                 _T("    ** Inserting Line '%s' after '%s'"), 
1223                 ((pLine
) ? pLine
->Text().c_str() : wxEmptyString
) ); 
1224     wxLogTrace( FILECONF_TRACE_MASK
, 
1226                 ((m_linesHead
) ? m_linesHead
->Text().c_str() : wxEmptyString
) ); 
1227     wxLogTrace( FILECONF_TRACE_MASK
, 
1229                 ((m_linesTail
) ? m_linesTail
->Text().c_str() : wxEmptyString
) ); 
1231     if ( pLine 
== m_linesTail 
) 
1232         return LineListAppend(str
); 
1234     wxFileConfigLineList 
*pNewLine 
= new wxFileConfigLineList(str
); 
1235     if ( pLine 
== NULL 
) 
1237         // prepend to the list 
1238         pNewLine
->SetNext(m_linesHead
); 
1239         m_linesHead
->SetPrev(pNewLine
); 
1240         m_linesHead 
= pNewLine
; 
1244         // insert before pLine 
1245         wxFileConfigLineList 
*pNext 
= pLine
->Next(); 
1246         pNewLine
->SetNext(pNext
); 
1247         pNewLine
->SetPrev(pLine
); 
1248         pNext
->SetPrev(pNewLine
); 
1249         pLine
->SetNext(pNewLine
); 
1252     wxLogTrace( FILECONF_TRACE_MASK
, 
1254                 ((m_linesHead
) ? m_linesHead
->Text().c_str() : wxEmptyString
) ); 
1255     wxLogTrace( FILECONF_TRACE_MASK
, 
1257                 ((m_linesTail
) ? m_linesTail
->Text().c_str() : wxEmptyString
) ); 
1262 void wxFileConfig::LineListRemove(wxFileConfigLineList 
*pLine
) 
1264     wxLogTrace( FILECONF_TRACE_MASK
, 
1265                 _T("    ** Removing Line '%s'"), 
1266                 pLine
->Text().c_str() ); 
1267     wxLogTrace( FILECONF_TRACE_MASK
, 
1269                 ((m_linesHead
) ? m_linesHead
->Text().c_str() : wxEmptyString
) ); 
1270     wxLogTrace( FILECONF_TRACE_MASK
, 
1272                 ((m_linesTail
) ? m_linesTail
->Text().c_str() : wxEmptyString
) ); 
1274     wxFileConfigLineList    
*pPrev 
= pLine
->Prev(), 
1275                             *pNext 
= pLine
->Next(); 
1279     if ( pPrev 
== NULL 
) 
1280         m_linesHead 
= pNext
; 
1282         pPrev
->SetNext(pNext
); 
1286     if ( pNext 
== NULL 
) 
1287         m_linesTail 
= pPrev
; 
1289         pNext
->SetPrev(pPrev
); 
1291     if ( m_pRootGroup
->GetGroupLine() == pLine 
) 
1292         m_pRootGroup
->SetLine(m_linesHead
); 
1294     wxLogTrace( FILECONF_TRACE_MASK
, 
1296                 ((m_linesHead
) ? m_linesHead
->Text().c_str() : wxEmptyString
) ); 
1297     wxLogTrace( FILECONF_TRACE_MASK
, 
1299                 ((m_linesTail
) ? m_linesTail
->Text().c_str() : wxEmptyString
) ); 
1304 bool wxFileConfig::LineListIsEmpty() 
1306     return m_linesHead 
== NULL
; 
1309 // ============================================================================ 
1310 // wxFileConfig::wxFileConfigGroup 
1311 // ============================================================================ 
1313 // ---------------------------------------------------------------------------- 
1315 // ---------------------------------------------------------------------------- 
1318 wxFileConfigGroup::wxFileConfigGroup(wxFileConfigGroup 
*pParent
, 
1319                                        const wxString
& strName
, 
1320                                        wxFileConfig 
*pConfig
) 
1321                          : m_aEntries(CompareEntries
), 
1322                            m_aSubgroups(CompareGroups
), 
1325   m_pConfig 
= pConfig
; 
1326   m_pParent 
= pParent
; 
1329   m_pLastEntry 
= NULL
; 
1330   m_pLastGroup 
= NULL
; 
1333 // dtor deletes all children 
1334 wxFileConfigGroup::~wxFileConfigGroup() 
1337   size_t n
, nCount 
= m_aEntries
.Count(); 
1338   for ( n 
= 0; n 
< nCount
; n
++ ) 
1339     delete m_aEntries
[n
]; 
1342   nCount 
= m_aSubgroups
.Count(); 
1343   for ( n 
= 0; n 
< nCount
; n
++ ) 
1344     delete m_aSubgroups
[n
]; 
1347 // ---------------------------------------------------------------------------- 
1349 // ---------------------------------------------------------------------------- 
1351 void wxFileConfigGroup::SetLine(wxFileConfigLineList 
*pLine
) 
1353     // for a normal (i.e. not root) group this method shouldn't be called twice 
1354     // unless we are resetting the line 
1355     wxASSERT_MSG( !m_pParent 
|| !m_pLine 
|| !pLine
, 
1356                    _T("changing line for a non-root group?") ); 
1362   This is a bit complicated, so let me explain it in details. All lines that 
1363   were read from the local file (the only one we will ever modify) are stored 
1364   in a (doubly) linked list. Our problem is to know at which position in this 
1365   list should we insert the new entries/subgroups. To solve it we keep three 
1366   variables for each group: m_pLine, m_pLastEntry and m_pLastGroup. 
1368   m_pLine points to the line containing "[group_name]" 
1369   m_pLastEntry points to the last entry of this group in the local file. 
1370   m_pLastGroup                   subgroup 
1372   Initially, they're NULL all three. When the group (an entry/subgroup) is read 
1373   from the local file, the corresponding variable is set. However, if the group 
1374   was read from the global file and then modified or created by the application 
1375   these variables are still NULL and we need to create the corresponding lines. 
1376   See the following functions (and comments preceding them) for the details of 
1379   Also, when our last entry/group are deleted we need to find the new last 
1380   element - the code in DeleteEntry/Subgroup does this by backtracking the list 
1381   of lines until it either founds an entry/subgroup (and this is the new last 
1382   element) or the m_pLine of the group, in which case there are no more entries 
1383   (or subgroups) left and m_pLast<element> becomes NULL. 
1385   NB: This last problem could be avoided for entries if we added new entries 
1386       immediately after m_pLine, but in this case the entries would appear 
1387       backwards in the config file (OTOH, it's not that important) and as we 
1388       would still need to do it for the subgroups the code wouldn't have been 
1389       significantly less complicated. 
1392 // Return the line which contains "[our name]". If we're still not in the list, 
1393 // add our line to it immediately after the last line of our parent group if we 
1394 // have it or in the very beginning if we're the root group. 
1395 wxFileConfigLineList 
*wxFileConfigGroup::GetGroupLine() 
1397     wxLogTrace( FILECONF_TRACE_MASK
, 
1398                 _T("  GetGroupLine() for Group '%s'"), 
1403         wxLogTrace( FILECONF_TRACE_MASK
, 
1404                     _T("    Getting Line item pointer") ); 
1406         wxFileConfigGroup   
*pParent 
= Parent(); 
1408         // this group wasn't present in local config file, add it now 
1411             wxLogTrace( FILECONF_TRACE_MASK
, 
1412                         _T("    checking parent '%s'"), 
1413                         pParent
->Name().c_str() ); 
1415             wxString    strFullName
; 
1417             // add 1 to the name because we don't want to start with '/' 
1418             strFullName 
<< wxT("[") 
1419                         << FilterOutEntryName(GetFullName().c_str() + 1) 
1421             m_pLine 
= m_pConfig
->LineListInsert(strFullName
, 
1422                                                 pParent
->GetLastGroupLine()); 
1423             pParent
->SetLastGroup(this);  // we're surely after all the others 
1425         //else: this is the root group and so we return NULL because we don't 
1426         //      have any group line 
1432 // Return the last line belonging to the subgroups of this group (after which 
1433 // we can add a new subgroup), if we don't have any subgroups or entries our 
1434 // last line is the group line (m_pLine) itself. 
1435 wxFileConfigLineList 
*wxFileConfigGroup::GetLastGroupLine() 
1437     // if we have any subgroups, our last line is the last line of the last 
1441         wxFileConfigLineList 
*pLine 
= m_pLastGroup
->GetLastGroupLine(); 
1443         wxASSERT_MSG( pLine
, _T("last group must have !NULL associated line") ); 
1448     // no subgroups, so the last line is the line of thelast entry (if any) 
1449     return GetLastEntryLine(); 
1452 // return the last line belonging to the entries of this group (after which 
1453 // we can add a new entry), if we don't have any entries we will add the new 
1454 // one immediately after the group line itself. 
1455 wxFileConfigLineList 
*wxFileConfigGroup::GetLastEntryLine() 
1457     wxLogTrace( FILECONF_TRACE_MASK
, 
1458                 _T("  GetLastEntryLine() for Group '%s'"), 
1463         wxFileConfigLineList    
*pLine 
= m_pLastEntry
->GetLine(); 
1465         wxASSERT_MSG( pLine
, _T("last entry must have !NULL associated line") ); 
1470     // no entries: insert after the group header, if any 
1471     return GetGroupLine(); 
1474 void wxFileConfigGroup::SetLastEntry(wxFileConfigEntry 
*pEntry
) 
1476     m_pLastEntry 
= pEntry
; 
1480         // the only situation in which a group without its own line can have 
1481         // an entry is when the first entry is added to the initially empty 
1482         // root pseudo-group 
1483         wxASSERT_MSG( !m_pParent
, _T("unexpected for non root group") ); 
1485         // let the group know that it does have a line in the file now 
1486         m_pLine 
= pEntry
->GetLine(); 
1490 // ---------------------------------------------------------------------------- 
1492 // ---------------------------------------------------------------------------- 
1494 void wxFileConfigGroup::UpdateGroupAndSubgroupsLines() 
1496     // update the line of this group 
1497     wxFileConfigLineList 
*line 
= GetGroupLine(); 
1498     wxCHECK_RET( line
, _T("a non root group must have a corresponding line!") ); 
1500     // +1: skip the leading '/' 
1501     line
->SetText(wxString::Format(_T("[%s]"), GetFullName().c_str() + 1)); 
1504     // also update all subgroups as they have this groups name in their lines 
1505     const size_t nCount 
= m_aSubgroups
.Count(); 
1506     for ( size_t n 
= 0; n 
< nCount
; n
++ ) 
1508         m_aSubgroups
[n
]->UpdateGroupAndSubgroupsLines(); 
1512 void wxFileConfigGroup::Rename(const wxString
& newName
) 
1514     wxCHECK_RET( m_pParent
, _T("the root group can't be renamed") ); 
1516     m_strName 
= newName
; 
1518     // update the group lines recursively 
1519     UpdateGroupAndSubgroupsLines(); 
1522 wxString 
wxFileConfigGroup::GetFullName() const 
1526         fullname 
= Parent()->GetFullName() + wxCONFIG_PATH_SEPARATOR 
+ Name(); 
1531 // ---------------------------------------------------------------------------- 
1533 // ---------------------------------------------------------------------------- 
1535 // use binary search because the array is sorted 
1537 wxFileConfigGroup::FindEntry(const wxChar 
*szName
) const 
1541        hi 
= m_aEntries
.Count(); 
1543   wxFileConfigEntry 
*pEntry
; 
1547     pEntry 
= m_aEntries
[i
]; 
1549     #if wxCONFIG_CASE_SENSITIVE 
1550       res 
= wxStrcmp(pEntry
->Name(), szName
); 
1552       res 
= wxStricmp(pEntry
->Name(), szName
); 
1567 wxFileConfigGroup::FindSubgroup(const wxChar 
*szName
) const 
1571        hi 
= m_aSubgroups
.Count(); 
1573   wxFileConfigGroup 
*pGroup
; 
1577     pGroup 
= m_aSubgroups
[i
]; 
1579     #if wxCONFIG_CASE_SENSITIVE 
1580       res 
= wxStrcmp(pGroup
->Name(), szName
); 
1582       res 
= wxStricmp(pGroup
->Name(), szName
); 
1596 // ---------------------------------------------------------------------------- 
1597 // create a new item 
1598 // ---------------------------------------------------------------------------- 
1600 // create a new entry and add it to the current group 
1601 wxFileConfigEntry 
*wxFileConfigGroup::AddEntry(const wxString
& strName
, int nLine
) 
1603     wxASSERT( FindEntry(strName
) == 0 ); 
1605     wxFileConfigEntry   
*pEntry 
= new wxFileConfigEntry(this, strName
, nLine
); 
1607     m_aEntries
.Add(pEntry
); 
1611 // create a new group and add it to the current group 
1612 wxFileConfigGroup 
*wxFileConfigGroup::AddSubgroup(const wxString
& strName
) 
1614     wxASSERT( FindSubgroup(strName
) == 0 ); 
1616     wxFileConfigGroup   
*pGroup 
= new wxFileConfigGroup(this, strName
, m_pConfig
); 
1618     m_aSubgroups
.Add(pGroup
); 
1622 // ---------------------------------------------------------------------------- 
1624 // ---------------------------------------------------------------------------- 
1627   The delete operations are _very_ slow if we delete the last item of this 
1628   group (see comments before GetXXXLineXXX functions for more details), 
1629   so it's much better to start with the first entry/group if we want to 
1630   delete several of them. 
1633 bool wxFileConfigGroup::DeleteSubgroupByName(const wxChar 
*szName
) 
1635     wxFileConfigGroup 
* const pGroup 
= FindSubgroup(szName
); 
1637     return pGroup 
? DeleteSubgroup(pGroup
) : false; 
1640 // Delete the subgroup and remove all references to it from 
1641 // other data structures. 
1642 bool wxFileConfigGroup::DeleteSubgroup(wxFileConfigGroup 
*pGroup
) 
1644     wxCHECK_MSG( pGroup
, false, _T("deleting non existing group?") ); 
1646     wxLogTrace( FILECONF_TRACE_MASK
, 
1647                 _T("Deleting group '%s' from '%s'"), 
1648                 pGroup
->Name().c_str(), 
1651     wxLogTrace( FILECONF_TRACE_MASK
, 
1652                 _T("  (m_pLine) = prev: %p, this %p, next %p"), 
1653                 ((m_pLine
) ? m_pLine
->Prev() : 0), 
1655                 ((m_pLine
) ? m_pLine
->Next() : 0) ); 
1656     wxLogTrace( FILECONF_TRACE_MASK
, 
1658                 ((m_pLine
) ? m_pLine
->Text().c_str() : wxEmptyString
) ); 
1660     // delete all entries... 
1661     size_t nCount 
= pGroup
->m_aEntries
.Count(); 
1663     wxLogTrace(FILECONF_TRACE_MASK
, 
1664                _T("Removing %lu entries"), (unsigned long)nCount 
); 
1666     for ( size_t nEntry 
= 0; nEntry 
< nCount
; nEntry
++ ) 
1668         wxFileConfigLineList 
*pLine 
= pGroup
->m_aEntries
[nEntry
]->GetLine(); 
1672             wxLogTrace( FILECONF_TRACE_MASK
, 
1674                         pLine
->Text().c_str() ); 
1675             m_pConfig
->LineListRemove(pLine
); 
1679     // ...and subgroups of this subgroup 
1680     nCount 
= pGroup
->m_aSubgroups
.Count(); 
1682     wxLogTrace( FILECONF_TRACE_MASK
, 
1683                 _T("Removing %lu subgroups"), (unsigned long)nCount 
); 
1685     for ( size_t nGroup 
= 0; nGroup 
< nCount
; nGroup
++ ) 
1687         pGroup
->DeleteSubgroup(pGroup
->m_aSubgroups
[0]); 
1690     // and then finally the group itself 
1691     wxFileConfigLineList 
*pLine 
= pGroup
->m_pLine
; 
1694         wxLogTrace( FILECONF_TRACE_MASK
, 
1695                     _T("  Removing line for group '%s' : '%s'"), 
1696                     pGroup
->Name().c_str(), 
1697                     pLine
->Text().c_str() ); 
1698         wxLogTrace( FILECONF_TRACE_MASK
, 
1699                     _T("  Removing from group '%s' : '%s'"), 
1701                     ((m_pLine
) ? m_pLine
->Text().c_str() : wxEmptyString
) ); 
1703         // notice that we may do this test inside the previous "if" 
1704         // because the last entry's line is surely !NULL 
1705         if ( pGroup 
== m_pLastGroup 
) 
1707             wxLogTrace( FILECONF_TRACE_MASK
, 
1708                         _T("  Removing last group") ); 
1710             // our last entry is being deleted, so find the last one which 
1711             // stays by going back until we find a subgroup or reach the 
1713             const size_t nSubgroups 
= m_aSubgroups
.Count(); 
1715             m_pLastGroup 
= NULL
; 
1716             for ( wxFileConfigLineList 
*pl 
= pLine
->Prev(); 
1717                   pl 
&& pl 
!= m_pLine 
&& !m_pLastGroup
; 
1720                 // does this line belong to our subgroup? 
1721                 for ( size_t n 
= 0; n 
< nSubgroups
; n
++ ) 
1723                     // do _not_ call GetGroupLine! we don't want to add it to 
1724                     // the local file if it's not already there 
1725                     if ( m_aSubgroups
[n
]->m_pLine 
== pl 
) 
1727                         m_pLastGroup 
= m_aSubgroups
[n
]; 
1734         m_pConfig
->LineListRemove(pLine
); 
1738         wxLogTrace( FILECONF_TRACE_MASK
, 
1739                     _T("  No line entry for Group '%s'?"), 
1740                     pGroup
->Name().c_str() ); 
1743     m_aSubgroups
.Remove(pGroup
); 
1749 bool wxFileConfigGroup::DeleteEntry(const wxChar 
*szName
) 
1751   wxFileConfigEntry 
*pEntry 
= FindEntry(szName
); 
1754       // entry doesn't exist, nothing to do 
1758   wxFileConfigLineList 
*pLine 
= pEntry
->GetLine(); 
1759   if ( pLine 
!= NULL 
) { 
1760     // notice that we may do this test inside the previous "if" because the 
1761     // last entry's line is surely !NULL 
1762     if ( pEntry 
== m_pLastEntry 
) { 
1763       // our last entry is being deleted - find the last one which stays 
1764       wxASSERT( m_pLine 
!= NULL 
);  // if we have an entry with !NULL pLine... 
1766       // go back until we find another entry or reach the group's line 
1767       wxFileConfigEntry 
*pNewLast 
= NULL
; 
1768       size_t n
, nEntries 
= m_aEntries
.Count(); 
1769       wxFileConfigLineList 
*pl
; 
1770       for ( pl 
= pLine
->Prev(); pl 
!= m_pLine
; pl 
= pl
->Prev() ) { 
1771         // is it our subgroup? 
1772         for ( n 
= 0; (pNewLast 
== NULL
) && (n 
< nEntries
); n
++ ) { 
1773           if ( m_aEntries
[n
]->GetLine() == m_pLine 
) 
1774             pNewLast 
= m_aEntries
[n
]; 
1777         if ( pNewLast 
!= NULL 
) // found? 
1781       if ( pl 
== m_pLine 
) { 
1782         wxASSERT( !pNewLast 
);  // how comes it has the same line as we? 
1784         // we've reached the group line without finding any subgroups 
1785         m_pLastEntry 
= NULL
; 
1788         m_pLastEntry 
= pNewLast
; 
1791     m_pConfig
->LineListRemove(pLine
); 
1794   m_aEntries
.Remove(pEntry
); 
1800 // ============================================================================ 
1801 // wxFileConfig::wxFileConfigEntry 
1802 // ============================================================================ 
1804 // ---------------------------------------------------------------------------- 
1806 // ---------------------------------------------------------------------------- 
1807 wxFileConfigEntry::wxFileConfigEntry(wxFileConfigGroup 
*pParent
, 
1808                                        const wxString
& strName
, 
1810                          : m_strName(strName
) 
1812   wxASSERT( !strName
.empty() ); 
1814   m_pParent 
= pParent
; 
1818   m_bHasValue 
= false; 
1820   m_bImmutable 
= strName
[0] == wxCONFIG_IMMUTABLE_PREFIX
; 
1822     m_strName
.erase(0, 1);  // remove first character 
1825 // ---------------------------------------------------------------------------- 
1827 // ---------------------------------------------------------------------------- 
1829 void wxFileConfigEntry::SetLine(wxFileConfigLineList 
*pLine
) 
1831   if ( m_pLine 
!= NULL 
) { 
1832     wxLogWarning(_("entry '%s' appears more than once in group '%s'"), 
1833                  Name().c_str(), m_pParent
->GetFullName().c_str()); 
1837   Group()->SetLastEntry(this); 
1840 // second parameter is false if we read the value from file and prevents the 
1841 // entry from being marked as 'dirty' 
1842 void wxFileConfigEntry::SetValue(const wxString
& strValue
, bool bUser
) 
1844     if ( bUser 
&& IsImmutable() ) 
1846         wxLogWarning( _("attempt to change immutable key '%s' ignored."), 
1851     // do nothing if it's the same value: but don't test for it if m_bHasValue 
1852     // hadn't been set yet or we'd never write empty values to the file 
1853     if ( m_bHasValue 
&& strValue 
== m_strValue 
) 
1857     m_strValue 
= strValue
; 
1861         wxString strValFiltered
; 
1863         if ( Group()->Config()->GetStyle() & wxCONFIG_USE_NO_ESCAPE_CHARACTERS 
) 
1865             strValFiltered 
= strValue
; 
1868             strValFiltered 
= FilterOutValue(strValue
); 
1872         strLine 
<< FilterOutEntryName(m_strName
) << wxT('=') << strValFiltered
; 
1876             // entry was read from the local config file, just modify the line 
1877             m_pLine
->SetText(strLine
); 
1879         else // this entry didn't exist in the local file 
1881             // add a new line to the file: note that line returned by 
1882             // GetLastEntryLine() may be NULL if we're in the root group and it 
1883             // doesn't have any entries yet, but this is ok as passing NULL 
1884             // line to LineListInsert() means to prepend new line to the list 
1885             wxFileConfigLineList 
*line 
= Group()->GetLastEntryLine(); 
1886             m_pLine 
= Group()->Config()->LineListInsert(strLine
, line
); 
1888             Group()->SetLastEntry(this); 
1893 // ============================================================================ 
1895 // ============================================================================ 
1897 // ---------------------------------------------------------------------------- 
1898 // compare functions for array sorting 
1899 // ---------------------------------------------------------------------------- 
1901 int CompareEntries(wxFileConfigEntry 
*p1
, wxFileConfigEntry 
*p2
) 
1903 #if wxCONFIG_CASE_SENSITIVE 
1904     return wxStrcmp(p1
->Name(), p2
->Name()); 
1906     return wxStricmp(p1
->Name(), p2
->Name()); 
1910 int CompareGroups(wxFileConfigGroup 
*p1
, wxFileConfigGroup 
*p2
) 
1912 #if wxCONFIG_CASE_SENSITIVE 
1913     return wxStrcmp(p1
->Name(), p2
->Name()); 
1915     return wxStricmp(p1
->Name(), p2
->Name()); 
1919 // ---------------------------------------------------------------------------- 
1921 // ---------------------------------------------------------------------------- 
1923 // undo FilterOutValue 
1924 static wxString 
FilterInValue(const wxString
& str
) 
1927   strResult
.Alloc(str
.Len()); 
1929   bool bQuoted 
= !str
.empty() && str
[0] == '"'; 
1931   for ( size_t n 
= bQuoted 
? 1 : 0; n 
< str
.Len(); n
++ ) { 
1932     if ( str
[n
] == wxT('\\') ) { 
1933       switch ( str
[++n
] ) { 
1935           strResult 
+= wxT('\n'); 
1939           strResult 
+= wxT('\r'); 
1943           strResult 
+= wxT('\t'); 
1947           strResult 
+= wxT('\\'); 
1951           strResult 
+= wxT('"'); 
1956       if ( str
[n
] != wxT('"') || !bQuoted 
) 
1957         strResult 
+= str
[n
]; 
1958       else if ( n 
!= str
.Len() - 1 ) { 
1959         wxLogWarning(_("unexpected \" at position %d in '%s'."), 
1962       //else: it's the last quote of a quoted string, ok 
1969 // quote the string before writing it to file 
1970 static wxString 
FilterOutValue(const wxString
& str
) 
1976   strResult
.Alloc(str
.Len()); 
1978   // quoting is necessary to preserve spaces in the beginning of the string 
1979   bool bQuote 
= wxIsspace(str
[0]) || str
[0] == wxT('"'); 
1982     strResult 
+= wxT('"'); 
1985   for ( size_t n 
= 0; n 
< str
.Len(); n
++ ) { 
2008         //else: fall through 
2011         strResult 
+= str
[n
]; 
2012         continue;   // nothing special to do 
2015     // we get here only for special characters 
2016     strResult 
<< wxT('\\') << c
; 
2020     strResult 
+= wxT('"'); 
2025 // undo FilterOutEntryName 
2026 static wxString 
FilterInEntryName(const wxString
& str
) 
2029   strResult
.Alloc(str
.Len()); 
2031   for ( const wxChar 
*pc 
= str
.c_str(); *pc 
!= '\0'; pc
++ ) { 
2032     if ( *pc 
== wxT('\\') ) { 
2033       // we need to test it here or we'd skip past the NUL in the loop line 
2034       if ( *++pc 
== _T('\0') ) 
2044 // sanitize entry or group name: insert '\\' before any special characters 
2045 static wxString 
FilterOutEntryName(const wxString
& str
) 
2048   strResult
.Alloc(str
.Len()); 
2050   for ( const wxChar 
*pc 
= str
.c_str(); *pc 
!= wxT('\0'); pc
++ ) { 
2051     const wxChar c 
= *pc
; 
2053     // we explicitly allow some of "safe" chars and 8bit ASCII characters 
2054     // which will probably never have special meaning and with which we can't 
2055     // use isalnum() anyhow (in ASCII built, in Unicode it's just fine) 
2057     // NB: note that wxCONFIG_IMMUTABLE_PREFIX and wxCONFIG_PATH_SEPARATOR 
2058     //     should *not* be quoted 
2061             ((unsigned char)c 
< 127) && 
2063          !wxIsalnum(c
) && !wxStrchr(wxT("@_/-!.*%"), c
) ) 
2065       strResult 
+= wxT('\\'); 
2074 // we can't put ?: in the ctor initializer list because it confuses some 
2075 // broken compilers (Borland C++) 
2076 static wxString 
GetAppName(const wxString
& appName
) 
2078     if ( !appName 
&& wxTheApp 
) 
2079         return wxTheApp
->GetAppName(); 
2084 #endif // wxUSE_CONFIG