]>
git.saurik.com Git - wxWidgets.git/blob - src/common/config.cpp
   1 /////////////////////////////////////////////////////////////////////////////// 
   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" 
  30 #include "wx/arrstr.h" 
  32 #if wxUSE_CONFIG && ((wxUSE_FILE && wxUSE_TEXTFILE) || wxUSE_CONFIG_NATIVE) 
  36 #include "wx/textfile.h" 
  43 #include <limits.h>     // for INT_MAX 
  45 // ---------------------------------------------------------------------------- 
  46 // global and class static variables 
  47 // ---------------------------------------------------------------------------- 
  49 wxConfigBase 
*wxConfigBase::ms_pConfig     
= NULL
; 
  50 bool          wxConfigBase::ms_bAutoCreate 
= true; 
  52 // ============================================================================ 
  54 // ============================================================================ 
  56 // ---------------------------------------------------------------------------- 
  58 // ---------------------------------------------------------------------------- 
  60 // Not all args will always be used by derived classes, but including them all 
  61 // in each class ensures compatibility. 
  62 wxConfigBase::wxConfigBase(const wxString
& appName
, 
  63                            const wxString
& vendorName
, 
  64                            const wxString
& WXUNUSED(localFilename
), 
  65                            const wxString
& WXUNUSED(globalFilename
), 
  67             : m_appName(appName
), m_vendorName(vendorName
), m_style(style
) 
  69     m_bExpandEnvVars 
= true; 
  70     m_bRecordDefaults 
= false; 
  73 wxConfigBase::~wxConfigBase() 
  75     // required here for Darwin 
  78 wxConfigBase 
*wxConfigBase::Set(wxConfigBase 
*pConfig
) 
  80   wxConfigBase 
*pOld 
= ms_pConfig
; 
  85 wxConfigBase 
*wxConfigBase::Create() 
  87   if ( ms_bAutoCreate 
&& ms_pConfig 
== NULL 
) { 
  89     #if defined(__WXMSW__) && wxUSE_CONFIG_NATIVE 
  90         new wxRegConfig(wxTheApp
->GetAppName(), wxTheApp
->GetVendorName()); 
  91     #elif defined(__WXPALMOS__) && wxUSE_CONFIG_NATIVE 
  92         new wxPrefConfig(wxTheApp
->GetAppName()); 
  93     #else // either we're under Unix or wish to use files even under Windows 
  94       new wxFileConfig(wxTheApp
->GetAppName()); 
 101 // ---------------------------------------------------------------------------- 
 102 // wxConfigBase reading entries 
 103 // ---------------------------------------------------------------------------- 
 105 // implement both Read() overloads for the given type in terms of DoRead() 
 106 #define IMPLEMENT_READ_FOR_TYPE(name, type, deftype, extra)                 \ 
 107     bool wxConfigBase::Read(const wxString& key, type *val) const           \ 
 109         wxCHECK_MSG( val, false, _T("wxConfig::Read(): NULL parameter") );  \ 
 111         if ( !DoRead##name(key, val) )                                      \ 
 114         *val = extra(*val);                                                 \ 
 119     bool wxConfigBase::Read(const wxString& key,                            \ 
 121                             deftype defVal) const                           \ 
 123         wxCHECK_MSG( val, false, _T("wxConfig::Read(): NULL parameter") );  \ 
 125         bool read = DoRead##name(key, val);                                 \ 
 128             if ( IsRecordingDefaults() )                                    \ 
 130                 ((wxConfigBase *)this)->DoWrite##name(key, defVal);         \ 
 136         *val = extra(*val);                                                 \ 
 142 IMPLEMENT_READ_FOR_TYPE(String
, wxString
, const wxString
&, ExpandEnvVars
) 
 143 IMPLEMENT_READ_FOR_TYPE(Long
, long, long, long) 
 144 IMPLEMENT_READ_FOR_TYPE(Int
, int, int, int) 
 145 IMPLEMENT_READ_FOR_TYPE(Double
, double, double, double) 
 146 IMPLEMENT_READ_FOR_TYPE(Bool
, bool, bool, bool) 
 148 #undef IMPLEMENT_READ_FOR_TYPE 
 150 // the DoReadXXX() for the other types have implementation in the base class 
 151 // but can be overridden in the derived ones 
 152 bool wxConfigBase::DoReadInt(const wxString
& key
, int *pi
) const 
 154     wxCHECK_MSG( pi
, false, _T("wxConfig::Read(): NULL parameter") ); 
 157     if ( !DoReadLong(key
, &l
) ) 
 160     wxASSERT_MSG( l 
< INT_MAX
, _T("overflow in wxConfig::DoReadInt") ); 
 167 bool wxConfigBase::DoReadBool(const wxString
& key
, bool* val
) const 
 169     wxCHECK_MSG( val
, false, _T("wxConfig::Read(): NULL parameter") ); 
 172     if ( !DoReadLong(key
, &l
) ) 
 175     wxASSERT_MSG( l 
== 0 || l 
== 1, _T("bad bool value in wxConfig::DoReadInt") ); 
 182 bool wxConfigBase::DoReadDouble(const wxString
& key
, double* val
) const 
 185     if ( Read(key
, &str
) ) 
 187         return str
.ToDouble(val
); 
 193 // string reading helper 
 194 wxString 
wxConfigBase::ExpandEnvVars(const wxString
& str
) const 
 196     wxString tmp
; // Required for BC++ 
 197     if (IsExpandingEnvVars()) 
 198         tmp 
= wxExpandEnvVars(str
); 
 204 // ---------------------------------------------------------------------------- 
 205 // wxConfigBase writing 
 206 // ---------------------------------------------------------------------------- 
 208 bool wxConfigBase::DoWriteDouble(const wxString
& key
, double val
) 
 210     return DoWriteString(key
, wxString::Format(_T("%g"), val
)); 
 213 bool wxConfigBase::DoWriteInt(const wxString
& key
, int value
) 
 215     return DoWriteLong(key
, (long)value
); 
 218 bool wxConfigBase::DoWriteBool(const wxString
& key
, bool value
) 
 220     return DoWriteLong(key
, value 
? 1l : 0l); 
 223 // ---------------------------------------------------------------------------- 
 224 // wxConfigPathChanger 
 225 // ---------------------------------------------------------------------------- 
 227 wxConfigPathChanger::wxConfigPathChanger(const wxConfigBase 
*pContainer
, 
 228                                          const wxString
& strEntry
) 
 231   m_pContainer 
= (wxConfigBase 
*)pContainer
; 
 233   // the path is everything which precedes the last slash 
 234   wxString strPath 
= strEntry
.BeforeLast(wxCONFIG_PATH_SEPARATOR
); 
 236   // except in the special case of "/keyname" when there is nothing before "/" 
 237   if ( strPath
.empty() && 
 238        ((!strEntry
.empty()) && strEntry
[0] == wxCONFIG_PATH_SEPARATOR
) ) 
 240     strPath 
= wxCONFIG_PATH_SEPARATOR
; 
 243   if ( !strPath
.empty() ) 
 245     if ( m_pContainer
->GetPath() != strPath 
) 
 247         // we do change the path so restore it later 
 250         /* JACS: work around a memory bug that causes an assert 
 251            when using wxRegConfig, related to reference-counting. 
 252            Can be reproduced by removing (const wxChar*) below and 
 253            adding the following code to the config sample OnInit under 
 256            pConfig->SetPath(wxT("MySettings")); 
 257            pConfig->SetPath(wxT("..")); 
 259            pConfig->Read(_T("MainWindowX"), & value); 
 261         m_strOldPath 
= (const wxChar
*) m_pContainer
->GetPath(); 
 262         if ( *m_strOldPath
.c_str() != wxCONFIG_PATH_SEPARATOR 
) 
 263           m_strOldPath 
+= wxCONFIG_PATH_SEPARATOR
; 
 264         m_pContainer
->SetPath(strPath
); 
 267     // in any case, use the just the name, not full path 
 268     m_strName 
= strEntry
.AfterLast(wxCONFIG_PATH_SEPARATOR
); 
 271     // it's a name only, without path - nothing to do 
 272     m_strName 
= strEntry
; 
 276 wxConfigPathChanger::~wxConfigPathChanger() 
 278   // only restore path if it was changed 
 280     m_pContainer
->SetPath(m_strOldPath
); 
 284 #endif // wxUSE_CONFIG 
 286 // ---------------------------------------------------------------------------- 
 287 // static & global functions 
 288 // ---------------------------------------------------------------------------- 
 290 // understands both Unix and Windows (but only under Windows) environment 
 291 // variables expansion: i.e. $var, $(var) and ${var} are always understood 
 292 // and in addition under Windows %var% is also. 
 294 // don't change the values the enum elements: they must be equal 
 295 // to the matching [closing] delimiter. 
 299   Bracket_Normal  
= ')', 
 302   Bracket_Windows 
= '%',    // yeah, Windows people are a bit strange ;-) 
 307 wxString 
wxExpandEnvVars(const wxString
& str
) 
 310   strResult
.Alloc(str
.Len()); 
 313   for ( size_t n 
= 0; n 
< str
.Len(); n
++ ) { 
 322             if ( str
[n
] == wxT('%') ) 
 323               bracket 
= Bracket_Windows
; 
 326           if ( n 
== str
.Len() - 1 ) { 
 327             bracket 
= Bracket_None
; 
 330             switch ( str
[n 
+ 1] ) { 
 332                 bracket 
= Bracket_Normal
; 
 333                 n
++;                   // skip the bracket 
 337                 bracket 
= Bracket_Curly
; 
 338                 n
++;                   // skip the bracket 
 342                 bracket 
= Bracket_None
; 
 348           while ( m 
< str
.Len() && (wxIsalnum(str
[m
]) || str
[m
] == wxT('_')) ) 
 351           wxString 
strVarName(str
.c_str() + n 
+ 1, m 
- n 
- 1); 
 354           const wxChar 
*pszValue 
= NULL
; 
 356           const wxChar 
*pszValue 
= wxGetenv(strVarName
); 
 358           if ( pszValue 
!= NULL 
) { 
 359             strResult 
+= pszValue
; 
 362             // variable doesn't exist => don't change anything 
 364               if ( bracket 
!= Bracket_Windows 
) 
 366                 if ( bracket 
!= Bracket_None 
) 
 367                   strResult 
<< str
[n 
- 1]; 
 368             strResult 
<< str
[n
] << strVarName
; 
 371           // check the closing bracket 
 372           if ( bracket 
!= Bracket_None 
) { 
 373             if ( m 
== str
.Len() || str
[m
] != (wxChar
)bracket 
) { 
 374               // under MSW it's common to have '%' characters in the registry 
 375               // and it's annoying to have warnings about them each time, so 
 376               // ignroe them silently if they are not used for env vars 
 378               // under Unix, OTOH, this warning could be useful for the user to 
 379               // understand why isn't the variable expanded as intended 
 381                 wxLogWarning(_("Environment variables expansion failed: missing '%c' at position %u in '%s'."), 
 382                              (char)bracket
, (unsigned int) (m 
+ 1), str
.c_str()); 
 386               // skip closing bracket unless the variables wasn't expanded 
 387               if ( pszValue 
== NULL 
) 
 388                 strResult 
<< (char)bracket
; 
 393           n 
= m 
- 1;  // skip variable name 
 398         // backslash can be used to suppress special meaning of % and $ 
 399         if ( n 
!= str
.Len() - 1 && 
 400                 (str
[n 
+ 1] == wxT('%') || str
[n 
+ 1] == wxT('$')) ) { 
 401           strResult 
+= str
[++n
]; 
 415 // this function is used to properly interpret '..' in path 
 416 void wxSplitPath(wxArrayString
& aParts
, const wxChar 
*sz
) 
 421   const wxChar 
*pc 
= sz
; 
 423     if ( *pc 
== wxT('\0') || *pc 
== wxCONFIG_PATH_SEPARATOR 
) { 
 424       if ( strCurrent 
== wxT(".") ) { 
 427       else if ( strCurrent 
== wxT("..") ) { 
 429         if ( aParts
.size() == 0 ) 
 430           wxLogWarning(_("'%s' has extra '..', ignored."), sz
); 
 432           aParts
.erase(aParts
.end() - 1); 
 436       else if ( !strCurrent
.empty() ) { 
 437         aParts
.push_back(strCurrent
); 
 441         // could log an error here, but we prefer to ignore extra '/' 
 443       if ( *pc 
== wxT('\0') )