]>
git.saurik.com Git - wxWidgets.git/blob - src/common/config.cpp
   1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/config.cpp 
   3 // Purpose:     implementation of wxConfigBase class 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 1997 Karsten Ballüder   Ballueder@usa.net 
   9 //                       Vadim Zeitlin      <zeitlin@dptmaths.ens-cachan.fr> 
  10 // Licence:     wxWindows licence 
  11 /////////////////////////////////////////////////////////////////////////////// 
  13 // ---------------------------------------------------------------------------- 
  15 // ---------------------------------------------------------------------------- 
  17 #include "wx/wxprec.h" 
  23 #ifndef wxUSE_CONFIG_NATIVE 
  24     #define wxUSE_CONFIG_NATIVE 1 
  27 #include "wx/config.h" 
  34     #include "wx/arrstr.h" 
  38 #if wxUSE_CONFIG && ((wxUSE_FILE && wxUSE_TEXTFILE) || wxUSE_CONFIG_NATIVE) 
  44 #include <limits.h>     // for INT_MAX 
  46 // ---------------------------------------------------------------------------- 
  47 // global and class static variables 
  48 // ---------------------------------------------------------------------------- 
  50 wxConfigBase 
*wxConfigBase::ms_pConfig     
= NULL
; 
  51 bool          wxConfigBase::ms_bAutoCreate 
= true; 
  53 // ============================================================================ 
  55 // ============================================================================ 
  57 // ---------------------------------------------------------------------------- 
  59 // ---------------------------------------------------------------------------- 
  61 // Not all args will always be used by derived classes, but including them all 
  62 // in each class ensures compatibility. 
  63 wxConfigBase::wxConfigBase(const wxString
& appName
, 
  64                            const wxString
& vendorName
, 
  65                            const wxString
& WXUNUSED(localFilename
), 
  66                            const wxString
& WXUNUSED(globalFilename
), 
  68             : m_appName(appName
), m_vendorName(vendorName
), m_style(style
) 
  70     m_bExpandEnvVars 
= true; 
  71     m_bRecordDefaults 
= false; 
  74 wxConfigBase::~wxConfigBase() 
  76     // required here for Darwin 
  79 wxConfigBase 
*wxConfigBase::Set(wxConfigBase 
*pConfig
) 
  81   wxConfigBase 
*pOld 
= ms_pConfig
; 
  86 wxConfigBase 
*wxConfigBase::Create() 
  88   if ( ms_bAutoCreate 
&& ms_pConfig 
== NULL 
) { 
  90     #if defined(__WXMSW__) && wxUSE_CONFIG_NATIVE 
  91         new wxRegConfig(wxTheApp
->GetAppName(), wxTheApp
->GetVendorName()); 
  92     #elif defined(__WXPALMOS__) && wxUSE_CONFIG_NATIVE 
  93         new wxPrefConfig(wxTheApp
->GetAppName()); 
  94     #else // either we're under Unix or wish to use files even under Windows 
  95       new wxFileConfig(wxTheApp
->GetAppName()); 
 102 // ---------------------------------------------------------------------------- 
 103 // wxConfigBase reading entries 
 104 // ---------------------------------------------------------------------------- 
 106 // implement both Read() overloads for the given type in terms of DoRead() 
 107 #define IMPLEMENT_READ_FOR_TYPE(name, type, deftype, extra)                 \ 
 108     bool wxConfigBase::Read(const wxString& key, type *val) const           \ 
 110         wxCHECK_MSG( val, false, _T("wxConfig::Read(): NULL parameter") );  \ 
 112         if ( !DoRead##name(key, val) )                                      \ 
 115         *val = extra(*val);                                                 \ 
 120     bool wxConfigBase::Read(const wxString& key,                            \ 
 122                             deftype defVal) const                           \ 
 124         wxCHECK_MSG( val, false, _T("wxConfig::Read(): NULL parameter") );  \ 
 126         bool read = DoRead##name(key, val);                                 \ 
 129             if ( IsRecordingDefaults() )                                    \ 
 131                 ((wxConfigBase *)this)->DoWrite##name(key, defVal);         \ 
 137         *val = extra(*val);                                                 \ 
 143 IMPLEMENT_READ_FOR_TYPE(String
, wxString
, const wxString
&, ExpandEnvVars
) 
 144 IMPLEMENT_READ_FOR_TYPE(Long
, long, long, long) 
 145 IMPLEMENT_READ_FOR_TYPE(Int
, int, int, int) 
 146 IMPLEMENT_READ_FOR_TYPE(Double
, double, double, double) 
 147 IMPLEMENT_READ_FOR_TYPE(Bool
, bool, bool, bool) 
 149 #undef IMPLEMENT_READ_FOR_TYPE 
 151 // the DoReadXXX() for the other types have implementation in the base class 
 152 // but can be overridden in the derived ones 
 153 bool wxConfigBase::DoReadInt(const wxString
& key
, int *pi
) const 
 155     wxCHECK_MSG( pi
, false, _T("wxConfig::Read(): NULL parameter") ); 
 158     if ( !DoReadLong(key
, &l
) ) 
 161     wxASSERT_MSG( l 
< INT_MAX
, _T("overflow in wxConfig::DoReadInt") ); 
 168 bool wxConfigBase::DoReadBool(const wxString
& key
, bool* val
) const 
 170     wxCHECK_MSG( val
, false, _T("wxConfig::Read(): NULL parameter") ); 
 173     if ( !DoReadLong(key
, &l
) ) 
 176     wxASSERT_MSG( l 
== 0 || l 
== 1, _T("bad bool value in wxConfig::DoReadInt") ); 
 183 bool wxConfigBase::DoReadDouble(const wxString
& key
, double* val
) const 
 186     if ( Read(key
, &str
) ) 
 188         return str
.ToDouble(val
); 
 194 // string reading helper 
 195 wxString 
wxConfigBase::ExpandEnvVars(const wxString
& str
) const 
 197     wxString tmp
; // Required for BC++ 
 198     if (IsExpandingEnvVars()) 
 199         tmp 
= wxExpandEnvVars(str
); 
 205 // ---------------------------------------------------------------------------- 
 206 // wxConfigBase writing 
 207 // ---------------------------------------------------------------------------- 
 209 bool wxConfigBase::DoWriteDouble(const wxString
& key
, double val
) 
 211     return DoWriteString(key
, wxString::Format(_T("%g"), val
)); 
 214 bool wxConfigBase::DoWriteInt(const wxString
& key
, int value
) 
 216     return DoWriteLong(key
, (long)value
); 
 219 bool wxConfigBase::DoWriteBool(const wxString
& key
, bool value
) 
 221     return DoWriteLong(key
, value 
? 1l : 0l); 
 224 // ---------------------------------------------------------------------------- 
 225 // wxConfigPathChanger 
 226 // ---------------------------------------------------------------------------- 
 228 wxConfigPathChanger::wxConfigPathChanger(const wxConfigBase 
*pContainer
, 
 229                                          const wxString
& strEntry
) 
 232   m_pContainer 
= (wxConfigBase 
*)pContainer
; 
 234   // the path is everything which precedes the last slash 
 235   wxString strPath 
= strEntry
.BeforeLast(wxCONFIG_PATH_SEPARATOR
); 
 237   // except in the special case of "/keyname" when there is nothing before "/" 
 238   if ( strPath
.empty() && 
 239        ((!strEntry
.empty()) && strEntry
[0] == wxCONFIG_PATH_SEPARATOR
) ) 
 241     strPath 
= wxCONFIG_PATH_SEPARATOR
; 
 244   if ( !strPath
.empty() ) 
 246     if ( m_pContainer
->GetPath() != strPath 
) 
 248         // we do change the path so restore it later 
 251         /* JACS: work around a memory bug that causes an assert 
 252            when using wxRegConfig, related to reference-counting. 
 253            Can be reproduced by removing (const wxChar*) below and 
 254            adding the following code to the config sample OnInit under 
 257            pConfig->SetPath(wxT("MySettings")); 
 258            pConfig->SetPath(wxT("..")); 
 260            pConfig->Read(_T("MainWindowX"), & value); 
 262         m_strOldPath 
= (const wxChar
*) m_pContainer
->GetPath(); 
 263         if ( *m_strOldPath
.c_str() != wxCONFIG_PATH_SEPARATOR 
) 
 264           m_strOldPath 
+= wxCONFIG_PATH_SEPARATOR
; 
 265         m_pContainer
->SetPath(strPath
); 
 268     // in any case, use the just the name, not full path 
 269     m_strName 
= strEntry
.AfterLast(wxCONFIG_PATH_SEPARATOR
); 
 272     // it's a name only, without path - nothing to do 
 273     m_strName 
= strEntry
; 
 277 void wxConfigPathChanger::UpdateIfDeleted() 
 279     // we don't have to do anything at all if we didn't change the path 
 283     // find the deepest still existing parent path of the original path 
 284     while ( !m_pContainer
->HasGroup(m_strOldPath
) ) 
 286         m_strOldPath 
= m_strOldPath
.BeforeLast(wxCONFIG_PATH_SEPARATOR
); 
 287         if ( m_strOldPath
.empty() ) 
 288             m_strOldPath 
= wxCONFIG_PATH_SEPARATOR
; 
 292 wxConfigPathChanger::~wxConfigPathChanger() 
 294   // only restore path if it was changed 
 296     m_pContainer
->SetPath(m_strOldPath
); 
 300 #endif // wxUSE_CONFIG 
 302 // ---------------------------------------------------------------------------- 
 303 // static & global functions 
 304 // ---------------------------------------------------------------------------- 
 306 // understands both Unix and Windows (but only under Windows) environment 
 307 // variables expansion: i.e. $var, $(var) and ${var} are always understood 
 308 // and in addition under Windows %var% is also. 
 310 // don't change the values the enum elements: they must be equal 
 311 // to the matching [closing] delimiter. 
 315   Bracket_Normal  
= ')', 
 318   Bracket_Windows 
= '%',    // yeah, Windows people are a bit strange ;-) 
 323 wxString 
wxExpandEnvVars(const wxString
& str
) 
 326   strResult
.Alloc(str
.length()); 
 329   for ( size_t n 
= 0; n 
< str
.length(); n
++ ) { 
 338             if ( str
[n
] == wxT('%') ) 
 339               bracket 
= Bracket_Windows
; 
 342           if ( n 
== str
.length() - 1 ) { 
 343             bracket 
= Bracket_None
; 
 346             switch ( str
[n 
+ 1] ) { 
 348                 bracket 
= Bracket_Normal
; 
 349                 n
++;                   // skip the bracket 
 353                 bracket 
= Bracket_Curly
; 
 354                 n
++;                   // skip the bracket 
 358                 bracket 
= Bracket_None
; 
 364           while ( m 
< str
.length() && (wxIsalnum(str
[m
]) || str
[m
] == wxT('_')) ) 
 367           wxString 
strVarName(str
.c_str() + n 
+ 1, m 
- n 
- 1); 
 370           const wxChar 
*pszValue 
= NULL
; 
 372           // NB: use wxGetEnv instead of wxGetenv as otherwise variables 
 373           //     set through wxSetEnv may not be read correctly! 
 374           const wxChar 
*pszValue 
= NULL
; 
 376           if (wxGetEnv(strVarName
, &tmp
)) 
 379           if ( pszValue 
!= NULL 
) { 
 380             strResult 
+= pszValue
; 
 383             // variable doesn't exist => don't change anything 
 385               if ( bracket 
!= Bracket_Windows 
) 
 387                 if ( bracket 
!= Bracket_None 
) 
 388                   strResult 
<< str
[n 
- 1]; 
 389             strResult 
<< str
[n
] << strVarName
; 
 392           // check the closing bracket 
 393           if ( bracket 
!= Bracket_None 
) { 
 394             if ( m 
== str
.length() || str
[m
] != (wxChar
)bracket 
) { 
 395               // under MSW it's common to have '%' characters in the registry 
 396               // and it's annoying to have warnings about them each time, so 
 397               // ignroe them silently if they are not used for env vars 
 399               // under Unix, OTOH, this warning could be useful for the user to 
 400               // understand why isn't the variable expanded as intended 
 402                 wxLogWarning(_("Environment variables expansion failed: missing '%c' at position %u in '%s'."), 
 403                              (char)bracket
, (unsigned int) (m 
+ 1), str
.c_str()); 
 407               // skip closing bracket unless the variables wasn't expanded 
 408               if ( pszValue 
== NULL 
) 
 409                 strResult 
<< (char)bracket
; 
 414           n 
= m 
- 1;  // skip variable name 
 419         // backslash can be used to suppress special meaning of % and $ 
 420         if ( n 
!= str
.length() - 1 && 
 421                 (str
[n 
+ 1] == wxT('%') || str
[n 
+ 1] == wxT('$')) ) { 
 422           strResult 
+= str
[++n
]; 
 436 // this function is used to properly interpret '..' in path 
 437 void wxSplitPath(wxArrayString
& aParts
, const wxChar 
*sz
) 
 442   const wxChar 
*pc 
= sz
; 
 444     if ( *pc 
== wxT('\0') || *pc 
== wxCONFIG_PATH_SEPARATOR 
) { 
 445       if ( strCurrent 
== wxT(".") ) { 
 448       else if ( strCurrent 
== wxT("..") ) { 
 450         if ( aParts
.size() == 0 ) 
 451           wxLogWarning(_("'%s' has extra '..', ignored."), sz
); 
 453           aParts
.erase(aParts
.end() - 1); 
 457       else if ( !strCurrent
.empty() ) { 
 458         aParts
.push_back(strCurrent
); 
 462         // could log an error here, but we prefer to ignore extra '/' 
 464       if ( *pc 
== wxT('\0') )