]> git.saurik.com Git - wxWidgets.git/blob - src/common/config.cpp
TWIN32 compatibility added; wxMotif uses wxGTK's wxPostScriptDC;
[wxWidgets.git] / src / common / config.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: config.cpp
3 // Purpose: implementation of wxConfigBase class
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 07.04.98
7 // RCS-ID: $Id$
8 // Copyright: (c) 1997 Karsten Ballüder Ballueder@usa.net
9 // Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
10 // Licence: wxWindows license
11 ///////////////////////////////////////////////////////////////////////////////
12
13 // ----------------------------------------------------------------------------
14 // headers
15 // ----------------------------------------------------------------------------
16 #ifdef __GNUG__
17 #pragma implementation "confbase.h"
18 #endif
19
20 #include "wx/wxprec.h"
21
22 #ifndef WX_PRECOMP
23 #include "wx/wx.h"
24 #endif
25
26 #include <wx/confbase.h>
27
28 #ifdef wxUSE_CONFIG
29
30 #ifdef __BORLANDC__
31 #pragma hdrstop
32 #endif //__BORLANDC__
33
34 #include <wx/app.h>
35 #include <wx/file.h>
36 #include <wx/log.h>
37 #include <wx/textfile.h>
38 #include <wx/utils.h>
39
40 // we must include (one of) these files for wxConfigBase::Create
41 #if defined(__WXMSW__) && defined(wxCONFIG_WIN32_NATIVE)
42 #ifdef __WIN32__
43 #include <wx/msw/regconf.h>
44 #else //WIN16
45 #include <wx/msw/iniconf.h>
46 #endif
47 #else // either we're under Unix or wish to use files even under Windows
48 #include <wx/fileconf.h>
49 #endif
50
51 #include <stdlib.h>
52 #include <math.h>
53 #include <ctype.h> // for isalnum()
54
55 // ----------------------------------------------------------------------------
56 // global and class static variables
57 // ----------------------------------------------------------------------------
58
59 wxConfigBase *wxConfigBase::ms_pConfig = NULL;
60 bool wxConfigBase::ms_bAutoCreate = TRUE;
61
62 // ============================================================================
63 // implementation
64 // ============================================================================
65
66 // ----------------------------------------------------------------------------
67 // wxConfigBase
68 // ----------------------------------------------------------------------------
69
70 // Not all args will always be used by derived classes, but
71 // including them all in each class ensures compatibility.
72 wxConfigBase::wxConfigBase(const wxString& appName, const wxString& vendorName,
73 const wxString& WXUNUSED(localFilename), const wxString& WXUNUSED(globalFilename), long style):
74 m_appName(appName), m_vendorName(vendorName), m_style(style)
75 {
76 m_bExpandEnvVars = TRUE; m_bRecordDefaults = FALSE;
77 }
78
79 wxConfigBase *wxConfigBase::Set(wxConfigBase *pConfig)
80 {
81 wxConfigBase *pOld = ms_pConfig;
82 ms_pConfig = pConfig;
83 return pOld;
84 }
85
86 wxConfigBase *wxConfigBase::Create()
87 {
88 if ( ms_bAutoCreate && ms_pConfig == NULL ) {
89 ms_pConfig =
90 #if defined(__WXMSW__) && defined(wxCONFIG_WIN32_NATIVE)
91 #ifdef __WIN32__
92 new wxRegConfig(wxTheApp->GetAppName(), wxTheApp->GetVendorName());
93 #else //WIN16
94 new wxIniConfig(wxTheApp->GetAppName(), wxTheApp->GetVendorName());
95 #endif
96 #else // either we're under Unix or wish to use files even under Windows
97 new wxFileConfig(wxTheApp->GetAppName());
98 #endif
99 }
100
101 return ms_pConfig;
102 }
103
104 wxString wxConfigBase::Read(const wxString& key, const wxString& defVal) const
105 {
106 wxString s;
107 Read(key, &s, defVal);
108 return s;
109 }
110
111 bool wxConfigBase::Read(const wxString& key, wxString *str, const wxString& defVal) const
112 {
113 if (!Read(key, str))
114 {
115 *str = ExpandEnvVars(defVal);
116 return FALSE;
117 }
118 else
119 return TRUE;
120 }
121
122 bool wxConfigBase::Read(const wxString& key, long *pl, long defVal) const
123 {
124 if (!Read(key, pl))
125 {
126 *pl = defVal;
127 return FALSE;
128 }
129 else
130 return TRUE;
131 }
132
133 bool wxConfigBase::Read(const wxString& key, double* val) const
134 {
135 wxString str;
136 if (Read(key, & str))
137 {
138 *val = atof(str);
139 return TRUE;
140 }
141 else
142 return FALSE;
143 }
144
145 bool wxConfigBase::Read(const wxString& key, double* val, double defVal) const
146 {
147 if (!Read(key, val))
148 {
149 *val = defVal;
150 return FALSE;
151 }
152 else
153 return TRUE;
154 }
155
156 bool wxConfigBase::Read(const wxString& key, bool* val) const
157 {
158 long l;
159 if (Read(key, & l))
160 {
161 *val = (l != 0);
162 return TRUE;
163 }
164 else
165 return FALSE;
166 }
167
168 bool wxConfigBase::Read(const wxString& key, bool* val, bool defVal) const
169 {
170 if (!Read(key, val))
171 {
172 *val = defVal;
173 return FALSE;
174 }
175 else
176 return TRUE;
177 }
178
179 // Convenience functions
180
181 bool wxConfigBase::Read(const wxString& key, int *pi) const
182 {
183 long l;
184 bool ret = Read(key, &l);
185 if (ret)
186 *pi = (int) l;
187 return ret;
188 }
189
190 bool wxConfigBase::Read(const wxString& key, int *pi, int defVal) const
191 {
192 long l;
193 bool ret = Read(key, &l, (long) defVal);
194 *pi = (int) l;
195 return ret;
196 }
197
198 bool wxConfigBase::Write(const wxString& key, double val)
199 {
200 wxString str;
201 str.Printf("%f", val);
202 return Write(key, str);
203 }
204
205 bool wxConfigBase::Write(const wxString& key, bool value)
206 {
207 long l = (value ? 1 : 0);
208 return Write(key, l);
209 }
210
211 wxString wxConfigBase::ExpandEnvVars(const wxString& str) const
212 {
213 wxString tmp; // Required for BC++
214 if (IsExpandingEnvVars())
215 tmp = wxExpandEnvVars(str);
216 else
217 tmp = str;
218 return tmp;
219 }
220
221 // ----------------------------------------------------------------------------
222 // wxConfigPathChanger
223 // ----------------------------------------------------------------------------
224
225 wxConfigPathChanger::wxConfigPathChanger(const wxConfigBase *pContainer,
226 const wxString& strEntry)
227 {
228 m_pContainer = (wxConfigBase *)pContainer;
229 wxString strPath = strEntry.Before(wxCONFIG_PATH_SEPARATOR);
230
231 // special case of "/keyname" when there is nothing before "/"
232 if ( strPath.IsEmpty() && ((!strEntry.IsEmpty()) && strEntry[0] == wxCONFIG_PATH_SEPARATOR ))
233 strPath = wxCONFIG_PATH_SEPARATOR;
234
235 if ( !strPath.IsEmpty() ) {
236 // do change the path
237 m_bChanged = TRUE;
238 m_strName = strEntry.Right(wxCONFIG_PATH_SEPARATOR);
239 m_strOldPath = m_pContainer->GetPath();
240 m_strOldPath += wxCONFIG_PATH_SEPARATOR;
241 m_pContainer->SetPath(strPath);
242 }
243 else {
244 // it's a name only, without path - nothing to do
245 m_bChanged = FALSE;
246 m_strName = strEntry;
247 }
248 }
249
250 wxConfigPathChanger::~wxConfigPathChanger()
251 {
252 // only restore path if it was changed
253 if ( m_bChanged ) {
254 m_pContainer->SetPath(m_strOldPath);
255 }
256 }
257
258 // ----------------------------------------------------------------------------
259 // static & global functions
260 // ----------------------------------------------------------------------------
261
262 // understands both Unix and Windows (but only under Windows) environment
263 // variables expansion: i.e. $var, $(var) and ${var} are always understood
264 // and in addition under Windows %var% is also.
265 wxString wxExpandEnvVars(const wxString& str)
266 {
267 wxString strResult;
268 strResult.Alloc(str.Len());
269
270 // don't change the values the enum elements: they must be equal
271 // to the matching [closing] delimiter.
272 enum Bracket
273 {
274 Bracket_None,
275 Bracket_Normal = ')',
276 Bracket_Curly = '}'
277 #ifdef __WXMSW__
278 ,Bracket_Windows = '%' // yeah, Windows people are a bit strange ;-)
279 #endif
280 };
281
282 size_t m;
283 for ( size_t n = 0; n < str.Len(); n++ ) {
284 switch ( str[n] ) {
285 #ifdef __WXMSW__
286 case '%':
287 #endif //WINDOWS
288 case '$':
289 {
290 Bracket bracket;
291 #ifdef __WXMSW__
292 if ( str[n] == '%' )
293 bracket = Bracket_Windows;
294 else
295 #endif //WINDOWS
296 if ( n == str.Len() - 1 ) {
297 bracket = Bracket_None;
298 }
299 else {
300 switch ( str[n + 1] ) {
301 case '(':
302 bracket = Bracket_Normal;
303 n++; // skip the bracket
304 break;
305
306 case '{':
307 bracket = Bracket_Curly;
308 n++; // skip the bracket
309 break;
310
311 default:
312 bracket = Bracket_None;
313 }
314 }
315
316 m = n + 1;
317
318 while ( m < str.Len() && (isalnum(str[m]) || str[m] == '_') )
319 m++;
320
321 wxString strVarName(str.c_str() + n + 1, m - n - 1);
322
323 const char *pszValue = getenv(strVarName);
324 if ( pszValue != NULL ) {
325 strResult += pszValue;
326 }
327 else {
328 // variable doesn't exist => don't change anything
329 #ifdef __WXMSW__
330 if ( bracket != Bracket_Windows )
331 #endif
332 if ( bracket != Bracket_None )
333 strResult << str[n - 1];
334 strResult << str[n] << strVarName;
335 }
336
337 // check the closing bracket
338 if ( bracket != Bracket_None ) {
339 if ( m == str.Len() || str[m] != (char)bracket ) {
340 wxLogWarning(_("Environment variables expansion failed: "
341 "missing '%c' at position %d in '%s'."),
342 (char)bracket, m + 1, str.c_str());
343 }
344 else {
345 // skip closing bracket unless the variables wasn't expanded
346 if ( pszValue == NULL )
347 strResult << (char)bracket;
348 m++;
349 }
350 }
351
352 n = m - 1; // skip variable name
353 }
354 break;
355
356 case '\\':
357 // backslash can be used to suppress special meaning of % and $
358 if ( n != str.Len() && (str[n + 1] == '%' || str[n + 1] == '$') ) {
359 strResult += str[++n];
360
361 break;
362 }
363 //else: fall through
364
365 default:
366 strResult += str[n];
367 }
368 }
369
370 return strResult;
371 }
372
373 // this function is used to properly interpret '..' in path
374 void wxSplitPath(wxArrayString& aParts, const char *sz)
375 {
376 aParts.Empty();
377
378 wxString strCurrent;
379 const char *pc = sz;
380 for ( ;; ) {
381 if ( *pc == '\0' || *pc == wxCONFIG_PATH_SEPARATOR ) {
382 if ( strCurrent == "." ) {
383 // ignore
384 }
385 else if ( strCurrent == ".." ) {
386 // go up one level
387 if ( aParts.IsEmpty() )
388 wxLogWarning(_("'%s' has extra '..', ignored."), sz);
389 else
390 aParts.Remove(aParts.Count() - 1);
391
392 strCurrent.Empty();
393 }
394 else if ( !strCurrent.IsEmpty() ) {
395 aParts.Add(strCurrent);
396 strCurrent.Empty();
397 }
398 //else:
399 // could log an error here, but we prefer to ignore extra '/'
400
401 if ( *pc == '\0' )
402 return;
403 }
404 else
405 strCurrent += *pc;
406
407 pc++;
408 }
409 }
410
411 #endif
412
413 // wxUSE_CONFIG
414