]>
git.saurik.com Git - wxWidgets.git/blob - src/msw/regconf.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: msw/regconf.cpp
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "regconf.h"
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
24 #include "wx/string.h"
34 #include "wx/config.h"
38 #include "wx/msw/registry.h"
39 #include "wx/msw/regconf.h"
41 // ----------------------------------------------------------------------------
43 // ----------------------------------------------------------------------------
45 // we put our data in HKLM\SOFTWARE_KEY\appname
46 #define SOFTWARE_KEY wxString("Software\\")
48 // ----------------------------------------------------------------------------
50 // ----------------------------------------------------------------------------
52 // get the value if the key is opened and it exists
53 bool TryGetValue(const wxRegKey
& key
, const wxString
& str
, wxString
& strVal
)
55 return key
.IsOpened() && key
.HasValue(str
) && key
.QueryValue(str
, strVal
);
58 bool TryGetValue(const wxRegKey
& key
, const wxString
& str
, long *plVal
)
60 return key
.IsOpened() && key
.HasValue(str
) && key
.QueryValue(str
, plVal
);
63 // ============================================================================
65 // ============================================================================
67 // ----------------------------------------------------------------------------
69 // ----------------------------------------------------------------------------
71 // create the config object which stores its data under HKCU\vendor\app and, if
72 // style & wxCONFIG_USE_GLOBAL_FILE, under HKLM\vendor\app
73 wxRegConfig::wxRegConfig(const wxString
& appName
, const wxString
& vendorName
,
74 const wxString
& strLocal
, const wxString
& strGlobal
,
76 : wxConfigBase(appName
, vendorName
, strLocal
, strGlobal
, style
)
80 bool bDoUseGlobal
= (style
& wxCONFIG_USE_GLOBAL_FILE
) != 0;
82 // the convention is to put the programs keys under <vendor>\<appname>
83 // (but it can be overriden by specifying the pathes explicitly in strLocal
85 if ( strLocal
.IsEmpty() || (strGlobal
.IsEmpty() && bDoUseGlobal
) )
87 if ( vendorName
.IsEmpty() )
90 strRoot
= wxTheApp
->GetVendorName();
97 // no '\\' needed if no vendor name
98 if ( !strRoot
.IsEmpty() )
103 if ( appName
.IsEmpty() )
105 wxCHECK_RET( wxTheApp
, wxT("No application name in wxRegConfig ctor!") );
106 strRoot
<< wxTheApp
->GetAppName();
113 //else: we don't need to do all the complicated stuff above
115 wxString str
= strLocal
.IsEmpty() ? strRoot
: strLocal
;
117 // as we're going to change the name of these keys fairly often and as
118 // there are only few of wxRegConfig objects (usually 1), we can allow
119 // ourselves to be generous and spend some memory to significantly improve
120 // performance of SetPath()
121 static const size_t MEMORY_PREALLOC
= 512;
123 m_keyLocalRoot
.ReserveMemoryForName(MEMORY_PREALLOC
);
124 m_keyLocal
.ReserveMemoryForName(MEMORY_PREALLOC
);
126 m_keyLocalRoot
.SetName(wxRegKey::HKCU
, SOFTWARE_KEY
+ str
);
127 m_keyLocal
.SetName(m_keyLocalRoot
, _T(""));
131 str
= strGlobal
.IsEmpty() ? strRoot
: strGlobal
;
133 m_keyGlobalRoot
.ReserveMemoryForName(MEMORY_PREALLOC
);
134 m_keyGlobal
.ReserveMemoryForName(MEMORY_PREALLOC
);
136 m_keyGlobalRoot
.SetName(wxRegKey::HKLM
, SOFTWARE_KEY
+ str
);
137 m_keyGlobal
.SetName(m_keyGlobalRoot
, _T(""));
140 // Create() will Open() if key already exists
141 m_keyLocalRoot
.Create();
143 // as it's the same key, Open() shouldn't fail (i.e. no need for Create())
146 // OTOH, this key may perfectly not exist, so suppress error messages the call
147 // to Open() might generate
151 m_keyGlobalRoot
.Open();
156 wxRegConfig::~wxRegConfig()
158 // nothing to do - key will be closed in their dtors
161 // ----------------------------------------------------------------------------
163 // ----------------------------------------------------------------------------
165 // this function is called a *lot* of times (as I learned after seeing from
166 // profiler output that it is called ~12000 times from Mahogany start up code!)
167 // so it is important to optimize it - in particular, avoid using generic
168 // string functions here and do everything manually because it is faster
170 // I still kept the old version to be able to check that the optimized code has
171 // the same output as the non optimized version.
172 void wxRegConfig::SetPath(const wxString
& strPath
)
174 // remember the old path
175 wxString strOldPath
= m_strPath
;
177 #ifdef WX_DEBUG_SET_PATH // non optimized version kept here for testing
178 wxString m_strPathAlt
;
181 wxArrayString aParts
;
183 // because GetPath() returns "" when we're at root, we must understand
184 // empty string as "/"
185 if ( strPath
.IsEmpty() || (strPath
[0] == wxCONFIG_PATH_SEPARATOR
) ) {
187 wxSplitPath(aParts
, strPath
);
190 // relative path, combine with current one
191 wxString strFullPath
= GetPath();
192 strFullPath
<< wxCONFIG_PATH_SEPARATOR
<< strPath
;
193 wxSplitPath(aParts
, strFullPath
);
196 // recombine path parts in one variable
198 m_strPathAlt
.Empty();
199 for ( size_t n
= 0; n
< aParts
.Count(); n
++ ) {
200 strRegPath
<< '\\' << aParts
[n
];
201 m_strPathAlt
<< wxCONFIG_PATH_SEPARATOR
<< aParts
[n
];
206 // check for the most common case first
207 if ( strPath
.empty() )
209 m_strPath
= wxCONFIG_PATH_SEPARATOR
;
213 // construct the full path
214 wxString strFullPath
;
215 if ( strPath
[0u] == wxCONFIG_PATH_SEPARATOR
)
218 strFullPath
= strPath
;
220 else // relative path
222 strFullPath
.reserve(2*m_strPath
.length());
224 strFullPath
<< m_strPath
;
225 if ( strFullPath
.Len() == 0 ||
226 strFullPath
.Last() != wxCONFIG_PATH_SEPARATOR
)
227 strFullPath
<< wxCONFIG_PATH_SEPARATOR
;
228 strFullPath
<< strPath
;
231 // simplify it: we need to handle ".." here
233 // count the total number of slashes we have to know if we can go upper
234 size_t totalSlashes
= 0;
236 // position of the last slash to be able to backtrack to it quickly if
237 // needed, but we set it to -1 if we don't have a valid position
239 // we only remember the last position which means that we handle ".."
240 // quite efficiently but not "../.." - however the latter should be
241 // much more rare, so it is probably ok
242 int posLastSlash
= -1;
244 const wxChar
*src
= strFullPath
.c_str();
245 size_t len
= strFullPath
.length();
246 const wxChar
*end
= src
+ len
;
248 wxChar
*dst
= m_strPath
.GetWriteBuf(len
);
251 for ( ; src
< end
; src
++, dst
++ )
253 if ( *src
== wxCONFIG_PATH_SEPARATOR
)
257 // note that we don't have to check for src < end here as
258 // *end == 0 so can't be '.'
259 if ( src
[1] == _T('.') && src
[2] == _T('.') &&
260 (src
+ 3 == end
|| src
[3] == wxCONFIG_PATH_SEPARATOR
) )
264 wxLogWarning(_("'%s' has extra '..', ignored."),
265 strFullPath
.c_str());
267 else // return to the previous path component
269 // do we already have its position?
270 if ( posLastSlash
== -1 )
272 // no, find it: note that we are sure to have one
273 // because totalSlashes > 0 so we don't have to
274 // check the boundary condition below
276 // this is more efficient than strrchr()
277 while ( *dst
!= wxCONFIG_PATH_SEPARATOR
)
282 else // the position of last slash was stored
285 dst
= start
+ posLastSlash
;
287 // invalidate posLastSlash
291 // this shouldn't happen
292 wxASSERT_MSG( *dst
== wxCONFIG_PATH_SEPARATOR
,
293 _T("error in wxRegConfig::SetPath") );
304 if ( (dst
== start
) || (dst
[-1] != wxCONFIG_PATH_SEPARATOR
) )
306 *dst
= wxCONFIG_PATH_SEPARATOR
;
308 posLastSlash
= dst
- start
;
312 else // previous char was a slash too
314 // squeeze several subsequent slashes into one: i.e.
315 // just ignore this one
320 else // normal character
327 // NUL terminate the string
328 if ( dst
[-1] == wxCONFIG_PATH_SEPARATOR
&& (dst
!= start
+ 1) )
330 // if it has a trailing slash we remove it unless it is the only
337 m_strPath
.UngetWriteBuf(dst
- start
);
340 #ifdef WX_DEBUG_SET_PATH
341 wxASSERT( m_strPath
== m_strPathAlt
);
344 if ( m_strPath
== strOldPath
)
347 // registry APIs want backslashes instead of slashes
349 size_t len
= m_strPath
.length();
351 const wxChar
*src
= m_strPath
.c_str();
352 wxChar
*dst
= strRegPath
.GetWriteBuf(len
);
354 const wxChar
*end
= src
+ len
;
355 for ( ; src
< end
; src
++, dst
++ )
357 if ( *src
== wxCONFIG_PATH_SEPARATOR
)
363 strRegPath
.UngetWriteBuf(len
);
365 // this is not needed any longer as we don't create keys unnecessarily any
366 // more (now it is done on demand, i.e. only when they're going to contain
369 // as we create the registry key when SetPath(key) is done, we can be left
370 // with plenty of empty keys if this was only done to try to read some
371 // value which, in fact, doesn't exist - to prevent this from happening we
372 // automatically delete the old key if it was empty
373 if ( m_keyLocal
.Exists() && LocalKey().IsEmpty() )
375 m_keyLocal
.DeleteSelf();
379 // change current key(s)
380 m_keyLocal
.SetName(m_keyLocalRoot
, strRegPath
);
381 m_keyGlobal
.SetName(m_keyGlobalRoot
, strRegPath
);
383 // don't create it right now, wait until it is accessed
384 //m_keyLocal.Create();
390 // ----------------------------------------------------------------------------
391 // enumeration (works only with current group)
392 // ----------------------------------------------------------------------------
395 We want to enumerate all local keys/values after the global ones, but, of
396 course, we don't want to repeat a key which appears locally as well as
399 We use the 15th bit of lIndex for distinction between global and local.
402 #define LOCAL_MASK 0x8000
403 #define IS_LOCAL_INDEX(l) (((l) & LOCAL_MASK) != 0)
405 bool wxRegConfig::GetFirstGroup(wxString
& str
, long& lIndex
) const
408 return GetNextGroup(str
, lIndex
);
411 bool wxRegConfig::GetNextGroup(wxString
& str
, long& lIndex
) const
413 // are we already enumerating local entries?
414 if ( m_keyGlobal
.IsOpened() && !IS_LOCAL_INDEX(lIndex
) ) {
415 // try to find a global entry which doesn't appear locally
416 while ( m_keyGlobal
.GetNextKey(str
, lIndex
) ) {
417 if ( !m_keyLocal
.Exists() || !LocalKey().HasSubKey(str
) ) {
418 // ok, found one - return it
423 // no more global entries
424 lIndex
|= LOCAL_MASK
;
427 // if we don't have the key at all, don't try to enumerate anything under it
428 if ( !m_keyLocal
.Exists() )
431 // much easier with local entries: get the next one we find
432 // (don't forget to clear our flag bit and set it again later)
433 lIndex
&= ~LOCAL_MASK
;
434 bool bOk
= LocalKey().GetNextKey(str
, lIndex
);
435 lIndex
|= LOCAL_MASK
;
440 bool wxRegConfig::GetFirstEntry(wxString
& str
, long& lIndex
) const
443 return GetNextEntry(str
, lIndex
);
446 bool wxRegConfig::GetNextEntry(wxString
& str
, long& lIndex
) const
448 // are we already enumerating local entries?
449 if ( m_keyGlobal
.IsOpened() && !IS_LOCAL_INDEX(lIndex
) ) {
450 // try to find a global entry which doesn't appear locally
451 while ( m_keyGlobal
.GetNextValue(str
, lIndex
) ) {
452 if ( !m_keyLocal
.Exists() || !LocalKey().HasValue(str
) ) {
453 // ok, found one - return it
458 // no more global entries
459 lIndex
|= LOCAL_MASK
;
462 // if we don't have the key at all, don't try to enumerate anything under it
463 if ( !m_keyLocal
.Exists() )
466 // much easier with local entries: get the next one we find
467 // (don't forget to clear our flag bit and set it again later)
468 lIndex
&= ~LOCAL_MASK
;
469 bool bOk
= LocalKey().GetNextValue(str
, lIndex
);
470 lIndex
|= LOCAL_MASK
;
475 size_t wxRegConfig::GetNumberOfEntries(bool WXUNUSED(bRecursive
)) const
482 bool bCont
= ((wxRegConfig
*)this)->GetFirstEntry(str
, l
);
486 bCont
= ((wxRegConfig
*)this)->GetNextEntry(str
, l
);
492 size_t wxRegConfig::GetNumberOfGroups(bool WXUNUSED(bRecursive
)) const
499 bool bCont
= ((wxRegConfig
*)this)->GetFirstGroup(str
, l
);
503 bCont
= ((wxRegConfig
*)this)->GetNextGroup(str
, l
);
509 // ----------------------------------------------------------------------------
510 // tests for existence
511 // ----------------------------------------------------------------------------
513 bool wxRegConfig::HasGroup(const wxString
& key
) const
515 wxConfigPathChanger
path(this, key
);
517 wxString
strName(path
.Name());
519 return (m_keyLocal
.Exists() && LocalKey().HasSubKey(strName
)) ||
520 m_keyGlobal
.HasSubKey(strName
);
523 bool wxRegConfig::HasEntry(const wxString
& key
) const
525 wxConfigPathChanger
path(this, key
);
527 wxString
strName(path
.Name());
529 return (m_keyLocal
.Exists() && LocalKey().HasValue(strName
)) ||
530 m_keyGlobal
.HasValue(strName
);
533 wxConfigBase::EntryType
wxRegConfig::GetEntryType(const wxString
& key
) const
535 wxConfigPathChanger
path(this, key
);
537 wxString
strName(path
.Name());
540 if ( m_keyLocal
.Exists() && LocalKey().HasValue(strName
) )
541 isNumeric
= m_keyLocal
.IsNumericValue(strName
);
542 else if ( m_keyGlobal
.HasValue(strName
) )
543 isNumeric
= m_keyGlobal
.IsNumericValue(strName
);
545 return wxConfigBase::Type_Unknown
;
547 return isNumeric
? wxConfigBase::Type_Integer
: wxConfigBase::Type_String
;
550 // ----------------------------------------------------------------------------
552 // ----------------------------------------------------------------------------
554 bool wxRegConfig::Read(const wxString
& key
, wxString
*pStr
) const
556 wxConfigPathChanger
path(this, key
);
558 bool bQueryGlobal
= TRUE
;
560 // if immutable key exists in global key we must check that it's not
561 // overriden by the local key with the same name
562 if ( IsImmutable(path
.Name()) ) {
563 if ( TryGetValue(m_keyGlobal
, path
.Name(), *pStr
) ) {
564 if ( m_keyLocal
.Exists() && LocalKey().HasValue(path
.Name()) ) {
565 wxLogWarning(wxT("User value for immutable key '%s' ignored."),
566 path
.Name().c_str());
568 *pStr
= wxConfigBase::ExpandEnvVars(*pStr
);
572 // don't waste time - it's not there anyhow
573 bQueryGlobal
= FALSE
;
577 // first try local key
578 if ( (m_keyLocal
.Exists() && TryGetValue(LocalKey(), path
.Name(), *pStr
)) ||
579 (bQueryGlobal
&& TryGetValue(m_keyGlobal
, path
.Name(), *pStr
)) ) {
582 *pStr
= wxConfigBase::ExpandEnvVars(*pStr
);
589 bool wxRegConfig::Read(const wxString
& key
, wxString
*pStr
,
590 const wxString
& szDefault
) const
592 wxConfigPathChanger
path(this, key
);
594 bool bQueryGlobal
= TRUE
;
596 // if immutable key exists in global key we must check that it's not
597 // overriden by the local key with the same name
598 if ( IsImmutable(path
.Name()) ) {
599 if ( TryGetValue(m_keyGlobal
, path
.Name(), *pStr
) ) {
600 if ( m_keyLocal
.Exists() && LocalKey().HasValue(path
.Name()) ) {
601 wxLogWarning(wxT("User value for immutable key '%s' ignored."),
602 path
.Name().c_str());
608 // don't waste time - it's not there anyhow
609 bQueryGlobal
= FALSE
;
613 // first try local key
614 if ( (m_keyLocal
.Exists() && TryGetValue(LocalKey(), path
.Name(), *pStr
)) ||
615 (bQueryGlobal
&& TryGetValue(m_keyGlobal
, path
.Name(), *pStr
)) ) {
616 *pStr
= wxConfigBase::ExpandEnvVars(*pStr
);
620 if ( IsRecordingDefaults() ) {
621 ((wxRegConfig
*)this)->Write(key
, szDefault
);
628 *pStr
= wxConfigBase::ExpandEnvVars(*pStr
);
633 bool wxRegConfig::Read(const wxString
& key
, long *plResult
) const
635 wxConfigPathChanger
path(this, key
);
637 bool bQueryGlobal
= TRUE
;
639 // if immutable key exists in global key we must check that it's not
640 // overriden by the local key with the same name
641 if ( IsImmutable(path
.Name()) ) {
642 if ( TryGetValue(m_keyGlobal
, path
.Name(), plResult
) ) {
643 if ( m_keyLocal
.Exists() && LocalKey().HasValue(path
.Name()) ) {
644 wxLogWarning(wxT("User value for immutable key '%s' ignored."),
645 path
.Name().c_str());
651 // don't waste time - it's not there anyhow
652 bQueryGlobal
= FALSE
;
656 // first try local key
657 if ( (m_keyLocal
.Exists() && TryGetValue(LocalKey(), path
.Name(), plResult
)) ||
658 (bQueryGlobal
&& TryGetValue(m_keyGlobal
, path
.Name(), plResult
)) ) {
664 bool wxRegConfig::Write(const wxString
& key
, const wxString
& szValue
)
666 wxConfigPathChanger
path(this, key
);
668 if ( IsImmutable(path
.Name()) ) {
669 wxLogError(wxT("Can't change immutable entry '%s'."), path
.Name().c_str());
673 return LocalKey().SetValue(path
.Name(), szValue
);
676 bool wxRegConfig::Write(const wxString
& key
, long lValue
)
678 wxConfigPathChanger
path(this, key
);
680 if ( IsImmutable(path
.Name()) ) {
681 wxLogError(wxT("Can't change immutable entry '%s'."), path
.Name().c_str());
685 return LocalKey().SetValue(path
.Name(), lValue
);
688 // ----------------------------------------------------------------------------
690 // ----------------------------------------------------------------------------
692 bool wxRegConfig::RenameEntry(const wxString
& oldName
, const wxString
& newName
)
694 // check that the old entry exists...
695 if ( !HasEntry(oldName
) )
698 // and that the new one doesn't
699 if ( HasEntry(newName
) )
702 return m_keyLocal
.RenameValue(oldName
, newName
);
705 bool wxRegConfig::RenameGroup(const wxString
& oldName
, const wxString
& newName
)
707 // check that the old group exists...
708 if ( !HasGroup(oldName
) )
711 // and that the new one doesn't
712 if ( HasGroup(newName
) )
715 return wxRegKey(m_keyLocal
, oldName
).Rename(newName
);
718 // ----------------------------------------------------------------------------
720 // ----------------------------------------------------------------------------
721 bool wxRegConfig::DeleteEntry(const wxString
& value
, bool WXUNUSED(bGroupIfEmptyAlso
))
723 wxConfigPathChanger
path(this, value
);
725 if ( m_keyLocal
.Exists() ) {
726 if ( !m_keyLocal
.DeleteValue(path
.Name()) )
729 if ( m_keyLocal
.IsEmpty() ) {
730 wxString strKey
= GetPath().AfterLast(wxCONFIG_PATH_SEPARATOR
);
731 SetPath(".."); // changes m_keyLocal
732 return LocalKey().DeleteKey(strKey
);
739 bool wxRegConfig::DeleteGroup(const wxString
& key
)
741 wxConfigPathChanger
path(this, key
);
743 return m_keyLocal
.Exists() ? LocalKey().DeleteKey(path
.Name()) : TRUE
;
746 bool wxRegConfig::DeleteAll()
751 bool bOk
= m_keyLocalRoot
.DeleteSelf();
753 // make sure that we opened m_keyGlobalRoot and so it has a reasonable name:
754 // otherwise we will delete HKEY_CLASSES_ROOT recursively
755 if ( bOk
&& m_keyGlobalRoot
.IsOpened() )
756 bOk
= m_keyGlobalRoot
.DeleteSelf();