]>
Commit | Line | Data |
---|---|---|
1 | /////////////////////////////////////////////////////////////////////////////// | |
2 | // Name: confbase.h | |
3 | // Purpose: declaration of the base class of all config implementations | |
4 | // (see also: fileconf.h and msw/regconf.h and iniconf.h) | |
5 | // Author: Karsten Ballueder & Vadim Zeitlin | |
6 | // Modified by: | |
7 | // Created: 07.04.98 (adapted from appconf.h) | |
8 | // RCS-ID: $Id$ | |
9 | // Copyright: (c) 1997 Karsten Ballueder Ballueder@usa.net | |
10 | // Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> | |
11 | // Licence: wxWindows licence | |
12 | /////////////////////////////////////////////////////////////////////////////// | |
13 | ||
14 | #ifndef _WX_CONFBASE_H_ | |
15 | #define _WX_CONFBASE_H_ | |
16 | ||
17 | #include "wx/defs.h" | |
18 | #include "wx/string.h" | |
19 | #include "wx/object.h" | |
20 | #include "wx/base64.h" | |
21 | ||
22 | class WXDLLIMPEXP_FWD_BASE wxArrayString; | |
23 | ||
24 | // ---------------------------------------------------------------------------- | |
25 | // constants | |
26 | // ---------------------------------------------------------------------------- | |
27 | ||
28 | /// shall we be case sensitive in parsing variable names? | |
29 | #ifndef wxCONFIG_CASE_SENSITIVE | |
30 | #define wxCONFIG_CASE_SENSITIVE 0 | |
31 | #endif | |
32 | ||
33 | /// separates group and entry names (probably shouldn't be changed) | |
34 | #ifndef wxCONFIG_PATH_SEPARATOR | |
35 | #define wxCONFIG_PATH_SEPARATOR _T('/') | |
36 | #endif | |
37 | ||
38 | /// introduces immutable entries | |
39 | // (i.e. the ones which can't be changed from the local config file) | |
40 | #ifndef wxCONFIG_IMMUTABLE_PREFIX | |
41 | #define wxCONFIG_IMMUTABLE_PREFIX _T('!') | |
42 | #endif | |
43 | ||
44 | #if wxUSE_CONFIG | |
45 | ||
46 | #include "wx/string.h" | |
47 | ||
48 | /// should we use registry instead of configuration files under Windows? | |
49 | // (i.e. whether wxConfigBase::Create() will create a wxFileConfig (if it's | |
50 | // false) or wxRegConfig (if it's true and we're under Win32)) | |
51 | #ifndef wxUSE_CONFIG_NATIVE | |
52 | #define wxUSE_CONFIG_NATIVE 1 | |
53 | #endif | |
54 | ||
55 | // Style flags for constructor style parameter | |
56 | enum | |
57 | { | |
58 | wxCONFIG_USE_LOCAL_FILE = 1, | |
59 | wxCONFIG_USE_GLOBAL_FILE = 2, | |
60 | wxCONFIG_USE_RELATIVE_PATH = 4, | |
61 | wxCONFIG_USE_NO_ESCAPE_CHARACTERS = 8, | |
62 | wxCONFIG_USE_SUBDIR = 16 | |
63 | }; | |
64 | ||
65 | // ---------------------------------------------------------------------------- | |
66 | // abstract base class wxConfigBase which defines the interface for derived | |
67 | // classes | |
68 | // | |
69 | // wxConfig organizes the items in a tree-like structure (modeled after the | |
70 | // Unix/Dos filesystem). There are groups (directories) and keys (files). | |
71 | // There is always one current group given by the current path. | |
72 | // | |
73 | // Keys are pairs "key_name = value" where value may be of string or integer | |
74 | // (long) type (TODO doubles and other types such as wxDate coming soon). | |
75 | // ---------------------------------------------------------------------------- | |
76 | ||
77 | class WXDLLIMPEXP_BASE wxConfigBase : public wxObject | |
78 | { | |
79 | public: | |
80 | // constants | |
81 | // the type of an entry | |
82 | enum EntryType | |
83 | { | |
84 | Type_Unknown, | |
85 | Type_String, | |
86 | Type_Boolean, | |
87 | Type_Integer, // use Read(long *) | |
88 | Type_Float // use Read(double *) | |
89 | }; | |
90 | ||
91 | // static functions | |
92 | // sets the config object, returns the previous pointer | |
93 | static wxConfigBase *Set(wxConfigBase *pConfig); | |
94 | // get the config object, creates it on demand unless DontCreateOnDemand | |
95 | // was called | |
96 | static wxConfigBase *Get(bool createOnDemand = true) | |
97 | { if ( createOnDemand && (!ms_pConfig) ) Create(); return ms_pConfig; } | |
98 | // create a new config object: this function will create the "best" | |
99 | // implementation of wxConfig available for the current platform, see | |
100 | // comments near definition wxUSE_CONFIG_NATIVE for details. It returns | |
101 | // the created object and also sets it as ms_pConfig. | |
102 | static wxConfigBase *Create(); | |
103 | // should Get() try to create a new log object if the current one is NULL? | |
104 | static void DontCreateOnDemand() { ms_bAutoCreate = false; } | |
105 | ||
106 | // ctor & virtual dtor | |
107 | // ctor (can be used as default ctor too) | |
108 | // | |
109 | // Not all args will always be used by derived classes, but including | |
110 | // them all in each class ensures compatibility. If appName is empty, | |
111 | // uses wxApp name | |
112 | wxConfigBase(const wxString& appName = wxEmptyString, | |
113 | const wxString& vendorName = wxEmptyString, | |
114 | const wxString& localFilename = wxEmptyString, | |
115 | const wxString& globalFilename = wxEmptyString, | |
116 | long style = 0); | |
117 | ||
118 | // empty but ensures that dtor of all derived classes is virtual | |
119 | virtual ~wxConfigBase(); | |
120 | ||
121 | // path management | |
122 | // set current path: if the first character is '/', it's the absolute path, | |
123 | // otherwise it's a relative path. '..' is supported. If the strPath | |
124 | // doesn't exist it is created. | |
125 | virtual void SetPath(const wxString& strPath) = 0; | |
126 | // retrieve the current path (always as absolute path) | |
127 | virtual const wxString& GetPath() const = 0; | |
128 | ||
129 | // enumeration: all functions here return false when there are no more items. | |
130 | // you must pass the same lIndex to GetNext and GetFirst (don't modify it) | |
131 | // enumerate subgroups | |
132 | virtual bool GetFirstGroup(wxString& str, long& lIndex) const = 0; | |
133 | virtual bool GetNextGroup (wxString& str, long& lIndex) const = 0; | |
134 | // enumerate entries | |
135 | virtual bool GetFirstEntry(wxString& str, long& lIndex) const = 0; | |
136 | virtual bool GetNextEntry (wxString& str, long& lIndex) const = 0; | |
137 | // get number of entries/subgroups in the current group, with or without | |
138 | // it's subgroups | |
139 | virtual size_t GetNumberOfEntries(bool bRecursive = false) const = 0; | |
140 | virtual size_t GetNumberOfGroups(bool bRecursive = false) const = 0; | |
141 | ||
142 | // tests of existence | |
143 | // returns true if the group by this name exists | |
144 | virtual bool HasGroup(const wxString& strName) const = 0; | |
145 | // same as above, but for an entry | |
146 | virtual bool HasEntry(const wxString& strName) const = 0; | |
147 | // returns true if either a group or an entry with a given name exist | |
148 | bool Exists(const wxString& strName) const | |
149 | { return HasGroup(strName) || HasEntry(strName); } | |
150 | ||
151 | // get the entry type | |
152 | virtual EntryType GetEntryType(const wxString& name) const | |
153 | { | |
154 | // by default all entries are strings | |
155 | return HasEntry(name) ? Type_String : Type_Unknown; | |
156 | } | |
157 | ||
158 | // key access: returns true if value was really read, false if default used | |
159 | // (and if the key is not found the default value is returned.) | |
160 | ||
161 | // read a string from the key | |
162 | bool Read(const wxString& key, wxString *pStr) const; | |
163 | bool Read(const wxString& key, wxString *pStr, const wxString& defVal) const; | |
164 | ||
165 | // read a number (long) | |
166 | bool Read(const wxString& key, long *pl) const; | |
167 | bool Read(const wxString& key, long *pl, long defVal) const; | |
168 | ||
169 | // read an int (wrapper around `long' version) | |
170 | bool Read(const wxString& key, int *pi) const; | |
171 | bool Read(const wxString& key, int *pi, int defVal) const; | |
172 | ||
173 | // read a double | |
174 | bool Read(const wxString& key, double* val) const; | |
175 | bool Read(const wxString& key, double* val, double defVal) const; | |
176 | ||
177 | // read a bool | |
178 | bool Read(const wxString& key, bool* val) const; | |
179 | bool Read(const wxString& key, bool* val, bool defVal) const; | |
180 | ||
181 | #if wxUSE_BASE64 | |
182 | // read a binary data block | |
183 | bool Read(const wxString& key, wxMemoryBuffer* data) const | |
184 | { return DoReadBinary(key, data); } | |
185 | // no default version since it does not make sense for binary data | |
186 | #endif // wxUSE_BASE64 | |
187 | ||
188 | // Causes ambiguities in VC++ 6 and OpenVMS (at least) | |
189 | #if ( (!defined(__VISUALC__) || __VISUALC__ > 1200) && !defined( __VMS ) && !defined (__DMC__)) | |
190 | // read other types, for which wxFromString is defined | |
191 | template <typename T> | |
192 | bool Read(const wxString& key, T* value) const | |
193 | { | |
194 | wxString s; | |
195 | if ( !Read(key, &s) ) | |
196 | return false; | |
197 | return wxFromString(s, value); | |
198 | } | |
199 | ||
200 | template <typename T> | |
201 | bool Read(const wxString& key, T* value, const T& defVal) const | |
202 | { | |
203 | const bool found = Read(key, value); | |
204 | if ( !found ) | |
205 | { | |
206 | if (IsRecordingDefaults()) | |
207 | ((wxConfigBase *)this)->Write(key, defVal); | |
208 | *value = defVal; | |
209 | } | |
210 | return found; | |
211 | } | |
212 | #endif | |
213 | ||
214 | // convenience functions returning directly the value | |
215 | wxString Read(const wxString& key, | |
216 | const wxString& defVal = wxEmptyString) const | |
217 | { wxString s; (void)Read(key, &s, defVal); return s; } | |
218 | ||
219 | // we have to provide a separate version for C strings as otherwise the | |
220 | // template Read() would be used | |
221 | wxString Read(const wxString& key, const char* defVal) const | |
222 | { return Read(key, wxString(defVal)); } | |
223 | #if wxUSE_WCHAR_T | |
224 | wxString Read(const wxString& key, const wchar_t* defVal) const | |
225 | { return Read(key, wxString(defVal)); } | |
226 | #endif | |
227 | ||
228 | long ReadLong(const wxString& key, long defVal) const | |
229 | { long l; (void)Read(key, &l, defVal); return l; } | |
230 | ||
231 | double ReadDouble(const wxString& key, double defVal) const | |
232 | { double d; (void)Read(key, &d, defVal); return d; } | |
233 | ||
234 | bool ReadBool(const wxString& key, bool defVal) const | |
235 | { bool b; (void)Read(key, &b, defVal); return b; } | |
236 | ||
237 | template <typename T> | |
238 | T ReadObject(const wxString& key, T const& defVal) const | |
239 | { T t; (void)Read(key, &t, defVal); return t; } | |
240 | ||
241 | // for compatibility with wx 2.8 | |
242 | long Read(const wxString& key, long defVal) const | |
243 | { return ReadLong(key, defVal); } | |
244 | ||
245 | ||
246 | // write the value (return true on success) | |
247 | bool Write(const wxString& key, const wxString& value) | |
248 | { return DoWriteString(key, value); } | |
249 | ||
250 | bool Write(const wxString& key, long value) | |
251 | { return DoWriteLong(key, value); } | |
252 | ||
253 | bool Write(const wxString& key, double value) | |
254 | { return DoWriteDouble(key, value); } | |
255 | ||
256 | bool Write(const wxString& key, bool value) | |
257 | { return DoWriteBool(key, value); } | |
258 | ||
259 | #if wxUSE_BASE64 | |
260 | bool Write(const wxString& key, const wxMemoryBuffer& buf) | |
261 | { return DoWriteBinary(key, buf); } | |
262 | #endif // wxUSE_BASE64 | |
263 | ||
264 | // we have to provide a separate version for C strings as otherwise they | |
265 | // would be converted to bool and not to wxString as expected! | |
266 | bool Write(const wxString& key, const char *value) | |
267 | { return Write(key, wxString(value)); } | |
268 | bool Write(const wxString& key, const unsigned char *value) | |
269 | { return Write(key, wxString(value)); } | |
270 | #if wxUSE_WCHAR_T | |
271 | bool Write(const wxString& key, const wchar_t *value) | |
272 | { return Write(key, wxString(value)); } | |
273 | #endif | |
274 | ||
275 | ||
276 | // we also have to provide specializations for other types which we want to | |
277 | // handle using the specialized DoWriteXXX() instead of the generic template | |
278 | // version below | |
279 | bool Write(const wxString& key, short value) | |
280 | { return DoWriteLong(key, value); } | |
281 | ||
282 | bool Write(const wxString& key, unsigned short value) | |
283 | { return DoWriteLong(key, value); } | |
284 | ||
285 | bool Write(const wxString& key, unsigned int value) | |
286 | { return DoWriteLong(key, value); } | |
287 | ||
288 | bool Write(const wxString& key, int value) | |
289 | { return DoWriteLong(key, value); } | |
290 | ||
291 | bool Write(const wxString& key, unsigned long value) | |
292 | { return DoWriteLong(key, value); } | |
293 | ||
294 | bool Write(const wxString& key, float value) | |
295 | { return DoWriteDouble(key, value); } | |
296 | ||
297 | // Causes ambiguities in VC++ 6 and OpenVMS (at least) | |
298 | #if ( (!defined(__VISUALC__) || __VISUALC__ > 1200) && !defined( __VMS ) && !defined (__DMC__)) | |
299 | // for other types, use wxToString() | |
300 | template <typename T> | |
301 | bool Write(const wxString& key, T const& value) | |
302 | { return Write(key, wxToString(value)); } | |
303 | #endif | |
304 | ||
305 | // permanently writes all changes | |
306 | virtual bool Flush(bool bCurrentOnly = false) = 0; | |
307 | ||
308 | // renaming, all functions return false on failure (probably because the new | |
309 | // name is already taken by an existing entry) | |
310 | // rename an entry | |
311 | virtual bool RenameEntry(const wxString& oldName, | |
312 | const wxString& newName) = 0; | |
313 | // rename a group | |
314 | virtual bool RenameGroup(const wxString& oldName, | |
315 | const wxString& newName) = 0; | |
316 | ||
317 | // delete entries/groups | |
318 | // deletes the specified entry and the group it belongs to if | |
319 | // it was the last key in it and the second parameter is true | |
320 | virtual bool DeleteEntry(const wxString& key, | |
321 | bool bDeleteGroupIfEmpty = true) = 0; | |
322 | // delete the group (with all subgroups) | |
323 | virtual bool DeleteGroup(const wxString& key) = 0; | |
324 | // delete the whole underlying object (disk file, registry key, ...) | |
325 | // primarily for use by uninstallation routine. | |
326 | virtual bool DeleteAll() = 0; | |
327 | ||
328 | // options | |
329 | // we can automatically expand environment variables in the config entries | |
330 | // (this option is on by default, you can turn it on/off at any time) | |
331 | bool IsExpandingEnvVars() const { return m_bExpandEnvVars; } | |
332 | void SetExpandEnvVars(bool bDoIt = true) { m_bExpandEnvVars = bDoIt; } | |
333 | // recording of default values | |
334 | void SetRecordDefaults(bool bDoIt = true) { m_bRecordDefaults = bDoIt; } | |
335 | bool IsRecordingDefaults() const { return m_bRecordDefaults; } | |
336 | // does expansion only if needed | |
337 | wxString ExpandEnvVars(const wxString& str) const; | |
338 | ||
339 | // misc accessors | |
340 | wxString GetAppName() const { return m_appName; } | |
341 | wxString GetVendorName() const { return m_vendorName; } | |
342 | ||
343 | // Used wxIniConfig to set members in constructor | |
344 | void SetAppName(const wxString& appName) { m_appName = appName; } | |
345 | void SetVendorName(const wxString& vendorName) { m_vendorName = vendorName; } | |
346 | ||
347 | void SetStyle(long style) { m_style = style; } | |
348 | long GetStyle() const { return m_style; } | |
349 | ||
350 | protected: | |
351 | static bool IsImmutable(const wxString& key) | |
352 | { return !key.IsEmpty() && key[0] == wxCONFIG_IMMUTABLE_PREFIX; } | |
353 | ||
354 | // return the path without trailing separator, if any: this should be called | |
355 | // to sanitize paths referring to the group names before passing them to | |
356 | // wxConfigPathChanger as "/foo/bar/" should be the same as "/foo/bar" and it | |
357 | // isn't interpreted in the same way by it (and this can't be changed there | |
358 | // as it's not the same for the entries names) | |
359 | static wxString RemoveTrailingSeparator(const wxString& key); | |
360 | ||
361 | // do read/write the values of different types | |
362 | virtual bool DoReadString(const wxString& key, wxString *pStr) const = 0; | |
363 | virtual bool DoReadLong(const wxString& key, long *pl) const = 0; | |
364 | virtual bool DoReadDouble(const wxString& key, double* val) const; | |
365 | virtual bool DoReadBool(const wxString& key, bool* val) const; | |
366 | #if wxUSE_BASE64 | |
367 | virtual bool DoReadBinary(const wxString& key, wxMemoryBuffer* buf) const = 0; | |
368 | #endif // wxUSE_BASE64 | |
369 | ||
370 | virtual bool DoWriteString(const wxString& key, const wxString& value) = 0; | |
371 | virtual bool DoWriteLong(const wxString& key, long value) = 0; | |
372 | virtual bool DoWriteDouble(const wxString& key, double value); | |
373 | virtual bool DoWriteBool(const wxString& key, bool value); | |
374 | #if wxUSE_BASE64 | |
375 | virtual bool DoWriteBinary(const wxString& key, const wxMemoryBuffer& buf) = 0; | |
376 | #endif // wxUSE_BASE64 | |
377 | ||
378 | private: | |
379 | // are we doing automatic environment variable expansion? | |
380 | bool m_bExpandEnvVars; | |
381 | // do we record default values? | |
382 | bool m_bRecordDefaults; | |
383 | ||
384 | // static variables | |
385 | static wxConfigBase *ms_pConfig; | |
386 | static bool ms_bAutoCreate; | |
387 | ||
388 | // Application name and organisation name | |
389 | wxString m_appName; | |
390 | wxString m_vendorName; | |
391 | ||
392 | // Style flag | |
393 | long m_style; | |
394 | ||
395 | DECLARE_ABSTRACT_CLASS(wxConfigBase) | |
396 | }; | |
397 | ||
398 | // a handy little class which changes current path to the path of given entry | |
399 | // and restores it in dtor: so if you declare a local variable of this type, | |
400 | // you work in the entry directory and the path is automatically restored | |
401 | // when the function returns | |
402 | // Taken out of wxConfig since not all compilers can cope with nested classes. | |
403 | class WXDLLIMPEXP_BASE wxConfigPathChanger | |
404 | { | |
405 | public: | |
406 | // ctor/dtor do path changing/restoring of the path | |
407 | wxConfigPathChanger(const wxConfigBase *pContainer, const wxString& strEntry); | |
408 | ~wxConfigPathChanger(); | |
409 | ||
410 | // get the key name | |
411 | const wxString& Name() const { return m_strName; } | |
412 | ||
413 | // this method must be called if the original path (i.e. the current path at | |
414 | // the moment of creation of this object) could have been deleted to prevent | |
415 | // us from restoring the not existing (any more) path | |
416 | // | |
417 | // if the original path doesn't exist any more, the path will be restored to | |
418 | // the deepest still existing component of the old path | |
419 | void UpdateIfDeleted(); | |
420 | ||
421 | private: | |
422 | wxConfigBase *m_pContainer; // object we live in | |
423 | wxString m_strName, // name of entry (i.e. name only) | |
424 | m_strOldPath; // saved path | |
425 | bool m_bChanged; // was the path changed? | |
426 | ||
427 | DECLARE_NO_COPY_CLASS(wxConfigPathChanger) | |
428 | }; | |
429 | ||
430 | ||
431 | #endif // wxUSE_CONFIG | |
432 | ||
433 | /* | |
434 | Replace environment variables ($SOMETHING) with their values. The format is | |
435 | $VARNAME or ${VARNAME} where VARNAME contains alphanumeric characters and | |
436 | '_' only. '$' must be escaped ('\$') in order to be taken literally. | |
437 | */ | |
438 | ||
439 | WXDLLIMPEXP_BASE wxString wxExpandEnvVars(const wxString &sz); | |
440 | ||
441 | /* | |
442 | Split path into parts removing '..' in progress | |
443 | */ | |
444 | WXDLLIMPEXP_BASE void wxSplitPath(wxArrayString& aParts, const wxString& path); | |
445 | ||
446 | #endif // _WX_CONFBASE_H_ | |
447 |