]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/iniconf.cpp
Halfway reasonable implementation of wxFont for wxCocoa.
[wxWidgets.git] / src / msw / iniconf.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/iniconf.cpp
3// Purpose: implementation of wxIniConfig class
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 27.07.98
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows licence
10///////////////////////////////////////////////////////////////////////////////
11
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifdef __BORLANDC__
16 #pragma hdrstop
17#endif
18
19#if wxUSE_INICONF
20
21#ifndef WX_PRECOMP
22 #include "wx/msw/wrapwin.h"
23 #include "wx/dynarray.h"
24 #include "wx/string.h"
25 #include "wx/intl.h"
26 #include "wx/event.h"
27 #include "wx/app.h"
28 #include "wx/utils.h"
29 #include "wx/log.h"
30#endif //WX_PRECOMP
31
32#include "wx/config.h"
33#include "wx/file.h"
34
35#include "wx/msw/iniconf.h"
36
37// ----------------------------------------------------------------------------
38// constants
39// ----------------------------------------------------------------------------
40
41// we replace all path separators with this character
42#define PATH_SEP_REPLACE '_'
43
44// ============================================================================
45// implementation
46// ============================================================================
47
48// ----------------------------------------------------------------------------
49// ctor & dtor
50// ----------------------------------------------------------------------------
51IMPLEMENT_ABSTRACT_CLASS(wxIniConfig, wxConfigBase)
52
53wxIniConfig::wxIniConfig(const wxString& strAppName,
54 const wxString& strVendor,
55 const wxString& localFilename,
56 const wxString& globalFilename,
57 long style)
58 : wxConfigBase(strAppName, strVendor, localFilename, globalFilename, style)
59
60#if 0 // This is too complex for some compilers, e.g. BC++ 5.01
61 : wxConfigBase((strAppName.empty() && wxTheApp) ? wxTheApp->GetAppName()
62 : strAppName,
63 strVendor.empty() ? (wxTheApp ? wxTheApp->GetVendorName()
64 : strAppName)
65 : strVendor,
66 localFilename, globalFilename, style)
67#endif
68{
69 if (strAppName.empty() && wxTheApp)
70 SetAppName(wxTheApp->GetAppName());
71 if (strVendor.empty() && wxTheApp)
72 SetVendorName(wxTheApp->GetVendorName());
73
74 m_strLocalFilename = localFilename;
75 if (m_strLocalFilename.empty())
76 {
77 m_strLocalFilename = GetAppName() + wxT(".ini");
78 }
79
80 // append the extension if none given and it's not an absolute file name
81 // (otherwise we assume that they know what they're doing)
82 if ( !wxIsPathSeparator(m_strLocalFilename[0u]) &&
83 m_strLocalFilename.Find(wxT('.')) == wxNOT_FOUND )
84 {
85 m_strLocalFilename << wxT(".ini");
86 }
87
88 // set root path
89 SetPath(wxEmptyString);
90}
91
92wxIniConfig::~wxIniConfig()
93{
94}
95
96// ----------------------------------------------------------------------------
97// path management
98// ----------------------------------------------------------------------------
99
100void wxIniConfig::SetPath(const wxString& strPath)
101{
102 wxArrayString aParts;
103
104 if ( strPath.empty() ) {
105 // nothing
106 }
107 else if ( strPath[0u] == wxCONFIG_PATH_SEPARATOR ) {
108 // absolute path
109 wxSplitPath(aParts, strPath);
110 }
111 else {
112 // relative path, combine with current one
113 wxString strFullPath = GetPath();
114 strFullPath << wxCONFIG_PATH_SEPARATOR << strPath;
115 wxSplitPath(aParts, strFullPath);
116 }
117
118 size_t nPartsCount = aParts.Count();
119 m_strPath.Empty();
120 if ( nPartsCount == 0 ) {
121 // go to the root
122 m_strGroup = PATH_SEP_REPLACE;
123 }
124 else {
125 // translate
126 m_strGroup = aParts[0u];
127 for ( size_t nPart = 1; nPart < nPartsCount; nPart++ ) {
128 if ( nPart > 1 )
129 m_strPath << PATH_SEP_REPLACE;
130 m_strPath << aParts[nPart];
131 }
132 }
133
134 // other functions assume that all this is true, i.e. there are no trailing
135 // underscores at the end except if the group is the root one
136 wxASSERT( (m_strPath.empty() || m_strPath.Last() != PATH_SEP_REPLACE) &&
137 (m_strGroup == wxString(PATH_SEP_REPLACE) ||
138 m_strGroup.Last() != PATH_SEP_REPLACE) );
139}
140
141const wxString& wxIniConfig::GetPath() const
142{
143 static wxString s_str;
144
145 // always return abs path
146 s_str = wxCONFIG_PATH_SEPARATOR;
147
148 if ( m_strGroup == wxString(PATH_SEP_REPLACE) ) {
149 // we're at the root level, nothing to do
150 }
151 else {
152 s_str << m_strGroup;
153 if ( !m_strPath.empty() )
154 s_str << wxCONFIG_PATH_SEPARATOR;
155 for ( const wxStringCharType *p = m_strPath.wx_str(); *p != '\0'; p++ ) {
156 s_str << (*p == PATH_SEP_REPLACE ? wxCONFIG_PATH_SEPARATOR : *p);
157 }
158 }
159
160 return s_str;
161}
162
163wxString wxIniConfig::GetPrivateKeyName(const wxString& szKey) const
164{
165 wxString strKey;
166
167 if ( !m_strPath.empty() )
168 strKey << m_strPath << PATH_SEP_REPLACE;
169
170 strKey << szKey;
171
172 return strKey;
173}
174
175wxString wxIniConfig::GetKeyName(const wxString& szKey) const
176{
177 wxString strKey;
178
179 if ( m_strGroup != wxString(PATH_SEP_REPLACE) )
180 strKey << m_strGroup << PATH_SEP_REPLACE;
181 if ( !m_strPath.empty() )
182 strKey << m_strPath << PATH_SEP_REPLACE;
183
184 strKey << szKey;
185
186 return strKey;
187}
188
189// ----------------------------------------------------------------------------
190// enumeration
191// ----------------------------------------------------------------------------
192
193// not implemented
194bool wxIniConfig::GetFirstGroup(wxString& WXUNUSED(str), long& WXUNUSED(lIndex)) const
195{
196 wxFAIL_MSG("not implemented");
197
198 return false;
199}
200
201bool wxIniConfig::GetNextGroup (wxString& WXUNUSED(str), long& WXUNUSED(lIndex)) const
202{
203 wxFAIL_MSG("not implemented");
204
205 return false;
206}
207
208bool wxIniConfig::GetFirstEntry(wxString& WXUNUSED(str), long& WXUNUSED(lIndex)) const
209{
210 wxFAIL_MSG("not implemented");
211
212 return false;
213}
214
215bool wxIniConfig::GetNextEntry (wxString& WXUNUSED(str), long& WXUNUSED(lIndex)) const
216{
217 wxFAIL_MSG("not implemented");
218
219 return false;
220}
221
222// ----------------------------------------------------------------------------
223// misc info
224// ----------------------------------------------------------------------------
225
226// not implemented
227size_t wxIniConfig::GetNumberOfEntries(bool WXUNUSED(bRecursive)) const
228{
229 wxFAIL_MSG("not implemented");
230
231 return (size_t)-1;
232}
233
234size_t wxIniConfig::GetNumberOfGroups(bool WXUNUSED(bRecursive)) const
235{
236 wxFAIL_MSG("not implemented");
237
238 return (size_t)-1;
239}
240
241bool wxIniConfig::HasGroup(const wxString& WXUNUSED(strName)) const
242{
243 wxFAIL_MSG("not implemented");
244
245 return false;
246}
247
248bool wxIniConfig::HasEntry(const wxString& WXUNUSED(strName)) const
249{
250 wxFAIL_MSG("not implemented");
251
252 return false;
253}
254
255// is current group empty?
256bool wxIniConfig::IsEmpty() const
257{
258 wxChar szBuf[1024];
259
260 GetPrivateProfileString(m_strGroup.wx_str(), NULL, _T(""),
261 szBuf, WXSIZEOF(szBuf),
262 m_strLocalFilename.wx_str());
263 if ( !wxIsEmpty(szBuf) )
264 return false;
265
266 GetProfileString(m_strGroup.wx_str(), NULL, _T(""), szBuf, WXSIZEOF(szBuf));
267 if ( !wxIsEmpty(szBuf) )
268 return false;
269
270 return true;
271}
272
273// ----------------------------------------------------------------------------
274// read/write
275// ----------------------------------------------------------------------------
276
277bool wxIniConfig::DoReadString(const wxString& szKey, wxString *pstr) const
278{
279 wxConfigPathChanger path(this, szKey);
280 wxString strKey = GetPrivateKeyName(path.Name());
281
282 wxChar szBuf[1024]; // FIXME: should dynamically allocate memory...
283
284 // first look in the private INI file
285
286 // NB: the lpDefault param to GetPrivateProfileString can't be NULL
287 GetPrivateProfileString(m_strGroup.wx_str(), strKey.wx_str(), _T(""),
288 szBuf, WXSIZEOF(szBuf),
289 m_strLocalFilename.wx_str());
290 if ( wxIsEmpty(szBuf) ) {
291 // now look in win.ini
292 wxString strKey = GetKeyName(path.Name());
293 GetProfileString(m_strGroup.wx_str(), strKey.wx_str(),
294 _T(""), szBuf, WXSIZEOF(szBuf));
295 }
296
297 if ( wxIsEmpty(szBuf) )
298 return false;
299
300 *pstr = szBuf;
301 return true;
302}
303
304bool wxIniConfig::DoReadLong(const wxString& szKey, long *pl) const
305{
306 wxConfigPathChanger path(this, szKey);
307 wxString strKey = GetPrivateKeyName(path.Name());
308
309 // hack: we have no mean to know if it really found the default value or
310 // didn't find anything, so we call it twice
311
312 static const int nMagic = 17; // 17 is some "rare" number
313 static const int nMagic2 = 28; // arbitrary number != nMagic
314 long lVal = GetPrivateProfileInt(m_strGroup.wx_str(), strKey.wx_str(),
315 nMagic, m_strLocalFilename.wx_str());
316 if ( lVal != nMagic ) {
317 // the value was read from the file
318 *pl = lVal;
319 return true;
320 }
321
322 // is it really nMagic?
323 lVal = GetPrivateProfileInt(m_strGroup.wx_str(), strKey.wx_str(),
324 nMagic2, m_strLocalFilename.wx_str());
325 if ( lVal != nMagic2 ) {
326 // the nMagic it returned was indeed read from the file
327 *pl = lVal;
328 return true;
329 }
330
331 // CS : I have no idea why they should look up in win.ini
332 // and if at all they have to do the same procedure using the two magic numbers
333 // otherwise it always returns true, even if the key was not there at all
334#if 0
335 // no, it was just returning the default value, so now look in win.ini
336 *pl = GetProfileInt(GetVendorName(), GetKeyName(szKey), *pl);
337
338 return true;
339#endif
340 return false ;
341}
342
343bool wxIniConfig::DoWriteString(const wxString& szKey, const wxString& szValue)
344{
345 wxConfigPathChanger path(this, szKey);
346 wxString strKey = GetPrivateKeyName(path.Name());
347
348 bool bOk = WritePrivateProfileString(m_strGroup.wx_str(), strKey.wx_str(),
349 szValue.wx_str(),
350 m_strLocalFilename.wx_str()) != 0;
351
352 if ( !bOk )
353 wxLogLastError(wxT("WritePrivateProfileString"));
354
355 return bOk;
356}
357
358bool wxIniConfig::DoWriteLong(const wxString& szKey, long lValue)
359{
360 return Write(szKey, wxString::Format(_T("%ld"), lValue));
361}
362
363bool wxIniConfig::DoReadBinary(const wxString& WXUNUSED(key),
364 wxMemoryBuffer * WXUNUSED(buf)) const
365{
366 wxFAIL_MSG("not implemented");
367
368 return false;
369}
370
371bool wxIniConfig::DoWriteBinary(const wxString& WXUNUSED(key),
372 const wxMemoryBuffer& WXUNUSED(buf))
373{
374 wxFAIL_MSG("not implemented");
375
376 return false;
377}
378
379bool wxIniConfig::Flush(bool /* bCurrentOnly */)
380{
381 // this is just the way it works
382 return WritePrivateProfileString(NULL, NULL, NULL,
383 m_strLocalFilename.wx_str()) != 0;
384}
385
386// ----------------------------------------------------------------------------
387// delete
388// ----------------------------------------------------------------------------
389
390bool wxIniConfig::DeleteEntry(const wxString& szKey, bool bGroupIfEmptyAlso)
391{
392 // passing NULL as value to WritePrivateProfileString deletes the key
393 wxConfigPathChanger path(this, szKey);
394 wxString strKey = GetPrivateKeyName(path.Name());
395
396 if (WritePrivateProfileString(m_strGroup.wx_str(), strKey.wx_str(),
397 NULL, m_strLocalFilename.wx_str()) == 0)
398 return false;
399
400 if ( !bGroupIfEmptyAlso || !IsEmpty() )
401 return true;
402
403 // delete the current group too
404 bool bOk = WritePrivateProfileString(m_strGroup.wx_str(), NULL,
405 NULL, m_strLocalFilename.wx_str()) != 0;
406
407 if ( !bOk )
408 wxLogLastError(wxT("WritePrivateProfileString"));
409
410 return bOk;
411}
412
413bool wxIniConfig::DeleteGroup(const wxString& szKey)
414{
415 wxConfigPathChanger path(this, szKey);
416
417 // passing NULL as section name to WritePrivateProfileString deletes the
418 // whole section according to the docs
419 bool bOk = WritePrivateProfileString(path.Name().wx_str(), NULL,
420 NULL, m_strLocalFilename.wx_str()) != 0;
421
422 if ( !bOk )
423 wxLogLastError(wxT("WritePrivateProfileString"));
424
425 return bOk;
426}
427
428#ifndef MAX_PATH
429#define MAX_PATH 256
430#endif
431
432bool wxIniConfig::DeleteAll()
433{
434 // first delete our group in win.ini
435 WriteProfileString(GetVendorName().wx_str(), NULL, NULL);
436
437 // then delete our own ini file
438 wxChar szBuf[MAX_PATH];
439 size_t nRc = GetWindowsDirectory(szBuf, WXSIZEOF(szBuf));
440 if ( nRc == 0 )
441 {
442 wxLogLastError(wxT("GetWindowsDirectory"));
443 }
444 else if ( nRc > WXSIZEOF(szBuf) )
445 {
446 wxFAIL_MSG(wxT("buffer is too small for Windows directory."));
447 }
448
449 wxString strFile = szBuf;
450 strFile << '\\' << m_strLocalFilename;
451
452 if ( wxFile::Exists(strFile) && !wxRemoveFile(strFile) ) {
453 wxLogSysError(_("Can't delete the INI file '%s'"), strFile.c_str());
454 return false;
455 }
456
457 return true;
458}
459
460bool wxIniConfig::RenameEntry(const wxString& WXUNUSED(oldName),
461 const wxString& WXUNUSED(newName))
462{
463 // Not implemented
464 return false;
465}
466
467bool wxIniConfig::RenameGroup(const wxString& WXUNUSED(oldName),
468 const wxString& WXUNUSED(newName))
469{
470 // Not implemented
471 return false;
472}
473
474#endif // wxUSE_INICONF