xref: /trunk/main/tools/source/generic/config.cxx (revision cdf0e10c4e3984b49a9502b011690b615761d4a3)
1*cdf0e10cSrcweir /*************************************************************************
2*cdf0e10cSrcweir  *
3*cdf0e10cSrcweir  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4*cdf0e10cSrcweir  *
5*cdf0e10cSrcweir  * Copyright 2000, 2010 Oracle and/or its affiliates.
6*cdf0e10cSrcweir  *
7*cdf0e10cSrcweir  * OpenOffice.org - a multi-platform office productivity suite
8*cdf0e10cSrcweir  *
9*cdf0e10cSrcweir  * This file is part of OpenOffice.org.
10*cdf0e10cSrcweir  *
11*cdf0e10cSrcweir  * OpenOffice.org is free software: you can redistribute it and/or modify
12*cdf0e10cSrcweir  * it under the terms of the GNU Lesser General Public License version 3
13*cdf0e10cSrcweir  * only, as published by the Free Software Foundation.
14*cdf0e10cSrcweir  *
15*cdf0e10cSrcweir  * OpenOffice.org is distributed in the hope that it will be useful,
16*cdf0e10cSrcweir  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17*cdf0e10cSrcweir  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18*cdf0e10cSrcweir  * GNU Lesser General Public License version 3 for more details
19*cdf0e10cSrcweir  * (a copy is included in the LICENSE file that accompanied this code).
20*cdf0e10cSrcweir  *
21*cdf0e10cSrcweir  * You should have received a copy of the GNU Lesser General Public License
22*cdf0e10cSrcweir  * version 3 along with OpenOffice.org.  If not, see
23*cdf0e10cSrcweir  * <http://www.openoffice.org/license.html>
24*cdf0e10cSrcweir  * for a copy of the LGPLv3 License.
25*cdf0e10cSrcweir  *
26*cdf0e10cSrcweir  ************************************************************************/
27*cdf0e10cSrcweir 
28*cdf0e10cSrcweir // MARKER(update_precomp.py): autogen include statement, do not remove
29*cdf0e10cSrcweir #include "precompiled_tools.hxx"
30*cdf0e10cSrcweir 
31*cdf0e10cSrcweir #define _CONFIG_CXX
32*cdf0e10cSrcweir 
33*cdf0e10cSrcweir #include <cstddef>
34*cdf0e10cSrcweir #include <cstdlib>
35*cdf0e10cSrcweir #include <limits>
36*cdf0e10cSrcweir #include <new>
37*cdf0e10cSrcweir #include <string.h>
38*cdf0e10cSrcweir 
39*cdf0e10cSrcweir #ifdef WNT
40*cdf0e10cSrcweir #include "stdlib.h"
41*cdf0e10cSrcweir #endif
42*cdf0e10cSrcweir #include <osl/file.hxx>
43*cdf0e10cSrcweir #include <tools/stream.hxx>
44*cdf0e10cSrcweir #include <tools/debug.hxx>
45*cdf0e10cSrcweir #include <tools/config.hxx>
46*cdf0e10cSrcweir #include <osl/security.h>
47*cdf0e10cSrcweir 
48*cdf0e10cSrcweir #define MAXBUFLEN   1024        // Fuer Buffer bei VOS-Funktionen
49*cdf0e10cSrcweir 
50*cdf0e10cSrcweir // -----------------
51*cdf0e10cSrcweir // - ImplConfigData -
52*cdf0e10cSrcweir // -----------------
53*cdf0e10cSrcweir 
54*cdf0e10cSrcweir struct ImplKeyData
55*cdf0e10cSrcweir {
56*cdf0e10cSrcweir     ImplKeyData*    mpNext;
57*cdf0e10cSrcweir     ByteString      maKey;
58*cdf0e10cSrcweir     ByteString      maValue;
59*cdf0e10cSrcweir     sal_Bool            mbIsComment;
60*cdf0e10cSrcweir };
61*cdf0e10cSrcweir 
62*cdf0e10cSrcweir struct ImplGroupData
63*cdf0e10cSrcweir {
64*cdf0e10cSrcweir     ImplGroupData*  mpNext;
65*cdf0e10cSrcweir     ImplKeyData*    mpFirstKey;
66*cdf0e10cSrcweir     ByteString      maGroupName;
67*cdf0e10cSrcweir     sal_uInt16          mnEmptyLines;
68*cdf0e10cSrcweir };
69*cdf0e10cSrcweir 
70*cdf0e10cSrcweir struct ImplConfigData
71*cdf0e10cSrcweir {
72*cdf0e10cSrcweir     ImplGroupData*  mpFirstGroup;
73*cdf0e10cSrcweir     XubString       maFileName;
74*cdf0e10cSrcweir     sal_uIntPtr         mnDataUpdateId;
75*cdf0e10cSrcweir     sal_uIntPtr         mnTimeStamp;
76*cdf0e10cSrcweir     LineEnd         meLineEnd;
77*cdf0e10cSrcweir     sal_uInt16          mnRefCount;
78*cdf0e10cSrcweir     sal_Bool            mbModified;
79*cdf0e10cSrcweir     sal_Bool            mbRead;
80*cdf0e10cSrcweir     sal_Bool            mbIsUTF8BOM;
81*cdf0e10cSrcweir };
82*cdf0e10cSrcweir 
83*cdf0e10cSrcweir // =======================================================================
84*cdf0e10cSrcweir 
85*cdf0e10cSrcweir static ByteString& getEmptyByteString()
86*cdf0e10cSrcweir {
87*cdf0e10cSrcweir     static ByteString aEmpty;
88*cdf0e10cSrcweir     return aEmpty;
89*cdf0e10cSrcweir }
90*cdf0e10cSrcweir 
91*cdf0e10cSrcweir // =======================================================================
92*cdf0e10cSrcweir 
93*cdf0e10cSrcweir static String toUncPath( const String& rPath )
94*cdf0e10cSrcweir {
95*cdf0e10cSrcweir     ::rtl::OUString aFileURL;
96*cdf0e10cSrcweir 
97*cdf0e10cSrcweir     // check if rFileName is already a URL; if not make it so
98*cdf0e10cSrcweir     if( rPath.CompareToAscii( "file://", 7 ) == COMPARE_EQUAL )
99*cdf0e10cSrcweir         aFileURL = rPath;
100*cdf0e10cSrcweir     else if( ::osl::FileBase::getFileURLFromSystemPath( rPath, aFileURL ) != ::osl::FileBase::E_None )
101*cdf0e10cSrcweir         aFileURL = rPath;
102*cdf0e10cSrcweir 
103*cdf0e10cSrcweir     return aFileURL;
104*cdf0e10cSrcweir }
105*cdf0e10cSrcweir 
106*cdf0e10cSrcweir static sal_uIntPtr ImplSysGetConfigTimeStamp( const XubString& rFileName )
107*cdf0e10cSrcweir {
108*cdf0e10cSrcweir     sal_uIntPtr nTimeStamp = 0;
109*cdf0e10cSrcweir     ::osl::DirectoryItem aItem;
110*cdf0e10cSrcweir     ::osl::FileStatus aStatus( osl_FileStatus_Mask_ModifyTime );
111*cdf0e10cSrcweir 
112*cdf0e10cSrcweir     int nError = 0;
113*cdf0e10cSrcweir     if( ( nError = ::osl::DirectoryItem::get( rFileName, aItem ) ) == ::osl::FileBase::E_None &&
114*cdf0e10cSrcweir         aItem.getFileStatus( aStatus ) == ::osl::FileBase::E_None )
115*cdf0e10cSrcweir     {
116*cdf0e10cSrcweir         nTimeStamp = aStatus.getModifyTime().Seconds;
117*cdf0e10cSrcweir     }
118*cdf0e10cSrcweir 
119*cdf0e10cSrcweir     return nTimeStamp;
120*cdf0e10cSrcweir }
121*cdf0e10cSrcweir 
122*cdf0e10cSrcweir // -----------------------------------------------------------------------
123*cdf0e10cSrcweir 
124*cdf0e10cSrcweir static sal_uInt8* ImplSysReadConfig( const XubString& rFileName,
125*cdf0e10cSrcweir                                 sal_uInt64& rRead, sal_Bool& rbRead, sal_Bool& rbIsUTF8BOM, sal_uIntPtr& rTimeStamp )
126*cdf0e10cSrcweir {
127*cdf0e10cSrcweir     sal_uInt8*          pBuf = NULL;
128*cdf0e10cSrcweir     ::osl::File aFile( rFileName );
129*cdf0e10cSrcweir 
130*cdf0e10cSrcweir     if( aFile.open( osl_File_OpenFlag_Read ) == ::osl::FileBase::E_None )
131*cdf0e10cSrcweir     {
132*cdf0e10cSrcweir         sal_uInt64 nPos = 0, nRead = 0;
133*cdf0e10cSrcweir         if( aFile.getSize( nPos ) == ::osl::FileBase::E_None )
134*cdf0e10cSrcweir         {
135*cdf0e10cSrcweir             if (nPos > std::numeric_limits< std::size_t >::max()) {
136*cdf0e10cSrcweir                 aFile.close();
137*cdf0e10cSrcweir                 return 0;
138*cdf0e10cSrcweir             }
139*cdf0e10cSrcweir             pBuf = new sal_uInt8[static_cast< std::size_t >(nPos)];
140*cdf0e10cSrcweir             if( aFile.read( pBuf, nPos, nRead ) == ::osl::FileBase::E_None && nRead == nPos )
141*cdf0e10cSrcweir             {
142*cdf0e10cSrcweir                 //skip the byte-order-mark 0xEF 0xBB 0xBF, if it was UTF8 files
143*cdf0e10cSrcweir                 unsigned char BOM[3] = {0xEF, 0xBB, 0xBF};
144*cdf0e10cSrcweir                 if (nRead > 2 && memcmp(pBuf, BOM, 3) == 0)
145*cdf0e10cSrcweir                 {
146*cdf0e10cSrcweir                     nRead -= 3;
147*cdf0e10cSrcweir                     rtl_moveMemory(pBuf, pBuf + 3, sal::static_int_cast<sal_Size>(nRead * sizeof(sal_uInt8)) );
148*cdf0e10cSrcweir                     rbIsUTF8BOM = sal_True;
149*cdf0e10cSrcweir                 }
150*cdf0e10cSrcweir 
151*cdf0e10cSrcweir                 rTimeStamp = ImplSysGetConfigTimeStamp( rFileName );
152*cdf0e10cSrcweir                 rbRead = sal_True;
153*cdf0e10cSrcweir                 rRead = nRead;
154*cdf0e10cSrcweir             }
155*cdf0e10cSrcweir             else
156*cdf0e10cSrcweir             {
157*cdf0e10cSrcweir                 delete[] pBuf;
158*cdf0e10cSrcweir                 pBuf = NULL;
159*cdf0e10cSrcweir             }
160*cdf0e10cSrcweir         }
161*cdf0e10cSrcweir         aFile.close();
162*cdf0e10cSrcweir     }
163*cdf0e10cSrcweir 
164*cdf0e10cSrcweir     return pBuf;
165*cdf0e10cSrcweir }
166*cdf0e10cSrcweir 
167*cdf0e10cSrcweir // -----------------------------------------------------------------------
168*cdf0e10cSrcweir 
169*cdf0e10cSrcweir static sal_Bool ImplSysWriteConfig( const XubString& rFileName,
170*cdf0e10cSrcweir                                 const sal_uInt8* pBuf, sal_uIntPtr nBufLen, sal_Bool rbIsUTF8BOM, sal_uIntPtr& rTimeStamp )
171*cdf0e10cSrcweir {
172*cdf0e10cSrcweir     sal_Bool bSuccess = sal_False;
173*cdf0e10cSrcweir     sal_Bool bUTF8BOMSuccess = sal_False;
174*cdf0e10cSrcweir 
175*cdf0e10cSrcweir     ::osl::File aFile( rFileName );
176*cdf0e10cSrcweir     ::osl::FileBase::RC eError = aFile.open( osl_File_OpenFlag_Write | osl_File_OpenFlag_Create );
177*cdf0e10cSrcweir     if( eError != ::osl::FileBase::E_None )
178*cdf0e10cSrcweir         eError = aFile.open( osl_File_OpenFlag_Write );
179*cdf0e10cSrcweir     if( eError == ::osl::FileBase::E_None )
180*cdf0e10cSrcweir     {
181*cdf0e10cSrcweir         // truncate
182*cdf0e10cSrcweir         aFile.setSize( 0 );
183*cdf0e10cSrcweir         sal_uInt64 nWritten;
184*cdf0e10cSrcweir 
185*cdf0e10cSrcweir         //write the the byte-order-mark 0xEF 0xBB 0xBF first , if it was UTF8 files
186*cdf0e10cSrcweir         if ( rbIsUTF8BOM )
187*cdf0e10cSrcweir         {
188*cdf0e10cSrcweir             unsigned char BOM[3] = {0xEF, 0xBB, 0xBF};
189*cdf0e10cSrcweir             sal_uInt64 nUTF8BOMWritten;
190*cdf0e10cSrcweir             if( aFile.write( BOM, 3, nUTF8BOMWritten ) == ::osl::FileBase::E_None && 3 == nUTF8BOMWritten )
191*cdf0e10cSrcweir             {
192*cdf0e10cSrcweir                 bUTF8BOMSuccess = sal_True;
193*cdf0e10cSrcweir             }
194*cdf0e10cSrcweir         }
195*cdf0e10cSrcweir 
196*cdf0e10cSrcweir         if( aFile.write( pBuf, nBufLen, nWritten ) == ::osl::FileBase::E_None && nWritten == nBufLen )
197*cdf0e10cSrcweir         {
198*cdf0e10cSrcweir             bSuccess = sal_True;
199*cdf0e10cSrcweir         }
200*cdf0e10cSrcweir         if ( rbIsUTF8BOM ? bSuccess && bUTF8BOMSuccess : bSuccess )
201*cdf0e10cSrcweir         {
202*cdf0e10cSrcweir             rTimeStamp = ImplSysGetConfigTimeStamp( rFileName );
203*cdf0e10cSrcweir         }
204*cdf0e10cSrcweir     }
205*cdf0e10cSrcweir 
206*cdf0e10cSrcweir     return rbIsUTF8BOM ? bSuccess && bUTF8BOMSuccess : bSuccess;
207*cdf0e10cSrcweir }
208*cdf0e10cSrcweir 
209*cdf0e10cSrcweir // -----------------------------------------------------------------------
210*cdf0e10cSrcweir 
211*cdf0e10cSrcweir static String ImplMakeConfigName( const XubString* pFileName,
212*cdf0e10cSrcweir                                   const XubString* pPathName )
213*cdf0e10cSrcweir {
214*cdf0e10cSrcweir     ::rtl::OUString aFileName;
215*cdf0e10cSrcweir     ::rtl::OUString aPathName;
216*cdf0e10cSrcweir     if ( pFileName )
217*cdf0e10cSrcweir     {
218*cdf0e10cSrcweir #ifdef UNX
219*cdf0e10cSrcweir         aFileName = ::rtl::OUString::createFromAscii( "." );
220*cdf0e10cSrcweir         aFileName += *pFileName;
221*cdf0e10cSrcweir         aFileName += ::rtl::OUString::createFromAscii( "rc" );
222*cdf0e10cSrcweir #else
223*cdf0e10cSrcweir         aFileName = *pFileName;
224*cdf0e10cSrcweir         aFileName += ::rtl::OUString::createFromAscii( ".ini" );
225*cdf0e10cSrcweir #endif
226*cdf0e10cSrcweir     }
227*cdf0e10cSrcweir     else
228*cdf0e10cSrcweir     {
229*cdf0e10cSrcweir #ifdef UNX
230*cdf0e10cSrcweir         aFileName = ::rtl::OUString::createFromAscii( ".sversionrc" );
231*cdf0e10cSrcweir #else
232*cdf0e10cSrcweir         aFileName = ::rtl::OUString::createFromAscii( "sversion.ini" );
233*cdf0e10cSrcweir #endif
234*cdf0e10cSrcweir     }
235*cdf0e10cSrcweir 
236*cdf0e10cSrcweir     // #88208# in case pPathName is set but empty and pFileName is set
237*cdf0e10cSrcweir     // and not empty just return the filename; on the default case
238*cdf0e10cSrcweir     // prepend default path as usual
239*cdf0e10cSrcweir     if ( pPathName && pPathName->Len() )
240*cdf0e10cSrcweir         aPathName = toUncPath( *pPathName );
241*cdf0e10cSrcweir     else if( pPathName && pFileName && pFileName->Len() )
242*cdf0e10cSrcweir         return aFileName;
243*cdf0e10cSrcweir     else
244*cdf0e10cSrcweir     {
245*cdf0e10cSrcweir         oslSecurity aSec = osl_getCurrentSecurity();
246*cdf0e10cSrcweir         osl_getConfigDir( aSec, &aPathName.pData );
247*cdf0e10cSrcweir         osl_freeSecurityHandle( aSec );
248*cdf0e10cSrcweir     }
249*cdf0e10cSrcweir 
250*cdf0e10cSrcweir     ::rtl::OUString aName( aPathName );
251*cdf0e10cSrcweir     aName += ::rtl::OUString::createFromAscii( "/" );
252*cdf0e10cSrcweir     aName += aFileName;
253*cdf0e10cSrcweir 
254*cdf0e10cSrcweir     return aName;
255*cdf0e10cSrcweir }
256*cdf0e10cSrcweir 
257*cdf0e10cSrcweir // -----------------------------------------------------------------------
258*cdf0e10cSrcweir 
259*cdf0e10cSrcweir namespace {
260*cdf0e10cSrcweir 
261*cdf0e10cSrcweir ByteString makeByteString(sal_uInt8 const * p, sal_uInt64 n) {
262*cdf0e10cSrcweir     if (n > STRING_MAXLEN) {
263*cdf0e10cSrcweir         #ifdef WNT
264*cdf0e10cSrcweir         abort();
265*cdf0e10cSrcweir         #else
266*cdf0e10cSrcweir         ::std::abort(); //TODO: handle this gracefully
267*cdf0e10cSrcweir         #endif
268*cdf0e10cSrcweir     }
269*cdf0e10cSrcweir     return ByteString(
270*cdf0e10cSrcweir         reinterpret_cast< char const * >(p),
271*cdf0e10cSrcweir         sal::static_int_cast< xub_StrLen >(n));
272*cdf0e10cSrcweir }
273*cdf0e10cSrcweir 
274*cdf0e10cSrcweir }
275*cdf0e10cSrcweir 
276*cdf0e10cSrcweir static void ImplMakeConfigList( ImplConfigData* pData,
277*cdf0e10cSrcweir                                 const sal_uInt8* pBuf, sal_uInt64 nLen )
278*cdf0e10cSrcweir {
279*cdf0e10cSrcweir     // kein Buffer, keine Daten
280*cdf0e10cSrcweir     if ( !nLen )
281*cdf0e10cSrcweir         return;
282*cdf0e10cSrcweir 
283*cdf0e10cSrcweir     // Buffer parsen und Liste zusammenbauen
284*cdf0e10cSrcweir     sal_uInt64 nStart;
285*cdf0e10cSrcweir     sal_uInt64 nLineLen;
286*cdf0e10cSrcweir     xub_StrLen      nNameLen;
287*cdf0e10cSrcweir     xub_StrLen      nKeyLen;
288*cdf0e10cSrcweir     sal_uInt64 i;
289*cdf0e10cSrcweir     const sal_uInt8*    pLine;
290*cdf0e10cSrcweir     ImplKeyData*    pPrevKey = NULL;
291*cdf0e10cSrcweir     ImplKeyData*    pKey;
292*cdf0e10cSrcweir     ImplGroupData*  pPrevGroup = NULL;
293*cdf0e10cSrcweir     ImplGroupData*  pGroup = NULL;
294*cdf0e10cSrcweir     i = 0;
295*cdf0e10cSrcweir     while ( i < nLen )
296*cdf0e10cSrcweir     {
297*cdf0e10cSrcweir         // Ctrl+Z
298*cdf0e10cSrcweir         if ( pBuf[i] == 0x1A )
299*cdf0e10cSrcweir             break;
300*cdf0e10cSrcweir 
301*cdf0e10cSrcweir         // Spaces und Tabs entfernen
302*cdf0e10cSrcweir         while ( (pBuf[i] == ' ') || (pBuf[i] == '\t') )
303*cdf0e10cSrcweir             i++;
304*cdf0e10cSrcweir 
305*cdf0e10cSrcweir         // Zeilenanfang merken
306*cdf0e10cSrcweir         nStart = i;
307*cdf0e10cSrcweir         pLine = pBuf+i;
308*cdf0e10cSrcweir 
309*cdf0e10cSrcweir         // Zeilenende suchen
310*cdf0e10cSrcweir         while (  (i < nLen) && pBuf[i] && (pBuf[i] != '\r') && (pBuf[i] != '\n') &&
311*cdf0e10cSrcweir                 (pBuf[i] != 0x1A) )
312*cdf0e10cSrcweir             i++;
313*cdf0e10cSrcweir 
314*cdf0e10cSrcweir         nLineLen = i-nStart;
315*cdf0e10cSrcweir 
316*cdf0e10cSrcweir         // Wenn Zeilenende (CR/LF), dann noch einen weiterschalten
317*cdf0e10cSrcweir         if ( (i+1 < nLen) &&
318*cdf0e10cSrcweir              (pBuf[i] != pBuf[i+1]) &&
319*cdf0e10cSrcweir              ((pBuf[i+1] == '\r') || (pBuf[i+1] == '\n')) )
320*cdf0e10cSrcweir             i++;
321*cdf0e10cSrcweir         i++;
322*cdf0e10cSrcweir 
323*cdf0e10cSrcweir         // Zeile auswerten
324*cdf0e10cSrcweir         if ( *pLine == '[' )
325*cdf0e10cSrcweir         {
326*cdf0e10cSrcweir             pGroup               = new ImplGroupData;
327*cdf0e10cSrcweir             pGroup->mpNext       = NULL;
328*cdf0e10cSrcweir             pGroup->mpFirstKey   = NULL;
329*cdf0e10cSrcweir             pGroup->mnEmptyLines = 0;
330*cdf0e10cSrcweir             if ( pPrevGroup )
331*cdf0e10cSrcweir                 pPrevGroup->mpNext = pGroup;
332*cdf0e10cSrcweir             else
333*cdf0e10cSrcweir                 pData->mpFirstGroup = pGroup;
334*cdf0e10cSrcweir             pPrevGroup  = pGroup;
335*cdf0e10cSrcweir             pPrevKey    = NULL;
336*cdf0e10cSrcweir             pKey        = NULL;
337*cdf0e10cSrcweir 
338*cdf0e10cSrcweir             // Gruppennamen rausfiltern
339*cdf0e10cSrcweir             pLine++;
340*cdf0e10cSrcweir             nLineLen--;
341*cdf0e10cSrcweir             // Spaces und Tabs entfernen
342*cdf0e10cSrcweir             while ( (*pLine == ' ') || (*pLine == '\t') )
343*cdf0e10cSrcweir             {
344*cdf0e10cSrcweir                 nLineLen--;
345*cdf0e10cSrcweir                 pLine++;
346*cdf0e10cSrcweir             }
347*cdf0e10cSrcweir             nNameLen = 0;
348*cdf0e10cSrcweir             while ( (nNameLen < nLineLen) && (pLine[nNameLen] != ']') )
349*cdf0e10cSrcweir                 nNameLen++;
350*cdf0e10cSrcweir             if ( nNameLen )
351*cdf0e10cSrcweir             {
352*cdf0e10cSrcweir                 while ( (pLine[nNameLen-1] == ' ') || (pLine[nNameLen-1] == '\t') )
353*cdf0e10cSrcweir                     nNameLen--;
354*cdf0e10cSrcweir             }
355*cdf0e10cSrcweir             pGroup->maGroupName = ByteString( (const sal_Char*)pLine, nNameLen );
356*cdf0e10cSrcweir         }
357*cdf0e10cSrcweir         else
358*cdf0e10cSrcweir         {
359*cdf0e10cSrcweir             if ( nLineLen )
360*cdf0e10cSrcweir             {
361*cdf0e10cSrcweir                 // Wenn noch keine Gruppe existiert, dann alle Keys in die
362*cdf0e10cSrcweir                 // Default-Gruppe
363*cdf0e10cSrcweir                 if ( !pGroup )
364*cdf0e10cSrcweir                 {
365*cdf0e10cSrcweir                     pGroup              = new ImplGroupData;
366*cdf0e10cSrcweir                     pGroup->mpNext      = NULL;
367*cdf0e10cSrcweir                     pGroup->mpFirstKey  = NULL;
368*cdf0e10cSrcweir                     pGroup->mnEmptyLines = 0;
369*cdf0e10cSrcweir                     if ( pPrevGroup )
370*cdf0e10cSrcweir                         pPrevGroup->mpNext = pGroup;
371*cdf0e10cSrcweir                     else
372*cdf0e10cSrcweir                         pData->mpFirstGroup = pGroup;
373*cdf0e10cSrcweir                     pPrevGroup  = pGroup;
374*cdf0e10cSrcweir                     pPrevKey    = NULL;
375*cdf0e10cSrcweir                 }
376*cdf0e10cSrcweir 
377*cdf0e10cSrcweir                 // Falls Leerzeile vorhanden, dann anhaengen
378*cdf0e10cSrcweir                 if ( pPrevKey )
379*cdf0e10cSrcweir                 {
380*cdf0e10cSrcweir                     while ( pGroup->mnEmptyLines )
381*cdf0e10cSrcweir                     {
382*cdf0e10cSrcweir                         pKey                = new ImplKeyData;
383*cdf0e10cSrcweir                         pKey->mbIsComment   = sal_True;
384*cdf0e10cSrcweir                         pPrevKey->mpNext    = pKey;
385*cdf0e10cSrcweir                         pPrevKey            = pKey;
386*cdf0e10cSrcweir                         pGroup->mnEmptyLines--;
387*cdf0e10cSrcweir                     }
388*cdf0e10cSrcweir                 }
389*cdf0e10cSrcweir 
390*cdf0e10cSrcweir                 // Neuen Key erzeugen
391*cdf0e10cSrcweir                 pKey        = new ImplKeyData;
392*cdf0e10cSrcweir                 pKey->mpNext = NULL;
393*cdf0e10cSrcweir                 if ( pPrevKey )
394*cdf0e10cSrcweir                     pPrevKey->mpNext = pKey;
395*cdf0e10cSrcweir                 else
396*cdf0e10cSrcweir                     pGroup->mpFirstKey = pKey;
397*cdf0e10cSrcweir                 pPrevKey = pKey;
398*cdf0e10cSrcweir                 if ( pLine[0] == ';' )
399*cdf0e10cSrcweir                 {
400*cdf0e10cSrcweir                     pKey->maValue = makeByteString(pLine, nLineLen);
401*cdf0e10cSrcweir                     pKey->mbIsComment = sal_True;
402*cdf0e10cSrcweir                 }
403*cdf0e10cSrcweir                 else
404*cdf0e10cSrcweir                 {
405*cdf0e10cSrcweir                     pKey->mbIsComment = sal_False;
406*cdf0e10cSrcweir                     nNameLen = 0;
407*cdf0e10cSrcweir                     while ( (nNameLen < nLineLen) && (pLine[nNameLen] != '=') )
408*cdf0e10cSrcweir                         nNameLen++;
409*cdf0e10cSrcweir                     nKeyLen = nNameLen;
410*cdf0e10cSrcweir                     // Spaces und Tabs entfernen
411*cdf0e10cSrcweir                     if ( nNameLen )
412*cdf0e10cSrcweir                     {
413*cdf0e10cSrcweir                         while ( (pLine[nNameLen-1] == ' ') || (pLine[nNameLen-1] == '\t') )
414*cdf0e10cSrcweir                             nNameLen--;
415*cdf0e10cSrcweir                     }
416*cdf0e10cSrcweir                     pKey->maKey = ByteString( (const sal_Char*)pLine, nNameLen );
417*cdf0e10cSrcweir                     nKeyLen++;
418*cdf0e10cSrcweir                     if ( nKeyLen < nLineLen )
419*cdf0e10cSrcweir                     {
420*cdf0e10cSrcweir                         pLine += nKeyLen;
421*cdf0e10cSrcweir                         nLineLen -= nKeyLen;
422*cdf0e10cSrcweir                         // Spaces und Tabs entfernen
423*cdf0e10cSrcweir                         while ( (*pLine == ' ') || (*pLine == '\t') )
424*cdf0e10cSrcweir                         {
425*cdf0e10cSrcweir                             nLineLen--;
426*cdf0e10cSrcweir                             pLine++;
427*cdf0e10cSrcweir                         }
428*cdf0e10cSrcweir                         if ( nLineLen )
429*cdf0e10cSrcweir                         {
430*cdf0e10cSrcweir                             while ( (pLine[nLineLen-1] == ' ') || (pLine[nLineLen-1] == '\t') )
431*cdf0e10cSrcweir                                 nLineLen--;
432*cdf0e10cSrcweir                             pKey->maValue = makeByteString(pLine, nLineLen);
433*cdf0e10cSrcweir                         }
434*cdf0e10cSrcweir                     }
435*cdf0e10cSrcweir                 }
436*cdf0e10cSrcweir             }
437*cdf0e10cSrcweir             else
438*cdf0e10cSrcweir             {
439*cdf0e10cSrcweir                 // Leerzeilen werden nur gezaehlt und beim Erzeugen des
440*cdf0e10cSrcweir                 // naechsten Keys angehaengt, da wir Leerzeilen am Ende
441*cdf0e10cSrcweir                 // einer Gruppe auch nach hinzufuegen von neuen Keys nur
442*cdf0e10cSrcweir                 // am Ende der Gruppe wieder speichern wollen
443*cdf0e10cSrcweir                 if ( pGroup )
444*cdf0e10cSrcweir                     pGroup->mnEmptyLines++;
445*cdf0e10cSrcweir             }
446*cdf0e10cSrcweir         }
447*cdf0e10cSrcweir     }
448*cdf0e10cSrcweir }
449*cdf0e10cSrcweir 
450*cdf0e10cSrcweir // -----------------------------------------------------------------------
451*cdf0e10cSrcweir 
452*cdf0e10cSrcweir static sal_uInt8* ImplGetConfigBuffer( const ImplConfigData* pData, sal_uIntPtr& rLen )
453*cdf0e10cSrcweir {
454*cdf0e10cSrcweir     sal_uInt8*          pWriteBuf;
455*cdf0e10cSrcweir     sal_uInt8*          pBuf;
456*cdf0e10cSrcweir     sal_uInt8           aLineEndBuf[2] = {0, 0};
457*cdf0e10cSrcweir     ImplKeyData*    pKey;
458*cdf0e10cSrcweir     ImplGroupData*  pGroup;
459*cdf0e10cSrcweir     unsigned int    nBufLen;
460*cdf0e10cSrcweir     sal_uInt16          nValueLen;
461*cdf0e10cSrcweir     sal_uInt16          nKeyLen;
462*cdf0e10cSrcweir     sal_uInt16          nLineEndLen;
463*cdf0e10cSrcweir 
464*cdf0e10cSrcweir     if ( pData->meLineEnd == LINEEND_CR )
465*cdf0e10cSrcweir     {
466*cdf0e10cSrcweir         aLineEndBuf[0] = _CR;
467*cdf0e10cSrcweir         nLineEndLen = 1;
468*cdf0e10cSrcweir     }
469*cdf0e10cSrcweir     else if ( pData->meLineEnd == LINEEND_LF )
470*cdf0e10cSrcweir     {
471*cdf0e10cSrcweir         aLineEndBuf[0] = _LF;
472*cdf0e10cSrcweir         nLineEndLen = 1;
473*cdf0e10cSrcweir     }
474*cdf0e10cSrcweir     else
475*cdf0e10cSrcweir     {
476*cdf0e10cSrcweir         aLineEndBuf[0] = _CR;
477*cdf0e10cSrcweir         aLineEndBuf[1] = _LF;
478*cdf0e10cSrcweir         nLineEndLen = 2;
479*cdf0e10cSrcweir     }
480*cdf0e10cSrcweir 
481*cdf0e10cSrcweir     // Buffergroesse ermitteln
482*cdf0e10cSrcweir     nBufLen = 0;
483*cdf0e10cSrcweir     pGroup = pData->mpFirstGroup;
484*cdf0e10cSrcweir     while ( pGroup )
485*cdf0e10cSrcweir     {
486*cdf0e10cSrcweir         // Leere Gruppen werden nicht geschrieben
487*cdf0e10cSrcweir         if ( pGroup->mpFirstKey )
488*cdf0e10cSrcweir         {
489*cdf0e10cSrcweir             nBufLen += pGroup->maGroupName.Len() + nLineEndLen + 2;
490*cdf0e10cSrcweir             pKey = pGroup->mpFirstKey;
491*cdf0e10cSrcweir             while ( pKey )
492*cdf0e10cSrcweir             {
493*cdf0e10cSrcweir                 nValueLen = pKey->maValue.Len();
494*cdf0e10cSrcweir                 if ( pKey->mbIsComment )
495*cdf0e10cSrcweir                     nBufLen += nValueLen + nLineEndLen;
496*cdf0e10cSrcweir                 else
497*cdf0e10cSrcweir                     nBufLen += pKey->maKey.Len() + nValueLen + nLineEndLen + 1;
498*cdf0e10cSrcweir 
499*cdf0e10cSrcweir                 pKey = pKey->mpNext;
500*cdf0e10cSrcweir             }
501*cdf0e10cSrcweir 
502*cdf0e10cSrcweir             // Leerzeile nach jeder Gruppe auch wieder speichern
503*cdf0e10cSrcweir             if ( !pGroup->mnEmptyLines )
504*cdf0e10cSrcweir                 pGroup->mnEmptyLines = 1;
505*cdf0e10cSrcweir             nBufLen += nLineEndLen * pGroup->mnEmptyLines;
506*cdf0e10cSrcweir         }
507*cdf0e10cSrcweir 
508*cdf0e10cSrcweir         pGroup = pGroup->mpNext;
509*cdf0e10cSrcweir     }
510*cdf0e10cSrcweir 
511*cdf0e10cSrcweir     // Laenge dem Aufrufer mitteilen
512*cdf0e10cSrcweir     rLen = nBufLen;
513*cdf0e10cSrcweir     if ( !nBufLen )
514*cdf0e10cSrcweir     {
515*cdf0e10cSrcweir         pWriteBuf = new sal_uInt8[nLineEndLen];
516*cdf0e10cSrcweir         if ( pWriteBuf )
517*cdf0e10cSrcweir         {
518*cdf0e10cSrcweir             pWriteBuf[0] = aLineEndBuf[0];
519*cdf0e10cSrcweir             if ( nLineEndLen == 2 )
520*cdf0e10cSrcweir                 pWriteBuf[1] = aLineEndBuf[1];
521*cdf0e10cSrcweir             return pWriteBuf;
522*cdf0e10cSrcweir         }
523*cdf0e10cSrcweir         else
524*cdf0e10cSrcweir             return 0;
525*cdf0e10cSrcweir     }
526*cdf0e10cSrcweir 
527*cdf0e10cSrcweir     // Schreibbuffer anlegen (wird vom Aufrufer zerstoert)
528*cdf0e10cSrcweir     pWriteBuf = new sal_uInt8[nBufLen];
529*cdf0e10cSrcweir     if ( !pWriteBuf )
530*cdf0e10cSrcweir         return 0;
531*cdf0e10cSrcweir 
532*cdf0e10cSrcweir     // Buffer fuellen
533*cdf0e10cSrcweir     pBuf = pWriteBuf;
534*cdf0e10cSrcweir     pGroup = pData->mpFirstGroup;
535*cdf0e10cSrcweir     while ( pGroup )
536*cdf0e10cSrcweir     {
537*cdf0e10cSrcweir         // Leere Gruppen werden nicht geschrieben
538*cdf0e10cSrcweir         if ( pGroup->mpFirstKey )
539*cdf0e10cSrcweir         {
540*cdf0e10cSrcweir             *pBuf = '[';    pBuf++;
541*cdf0e10cSrcweir             memcpy( pBuf, pGroup->maGroupName.GetBuffer(), pGroup->maGroupName.Len() );
542*cdf0e10cSrcweir             pBuf += pGroup->maGroupName.Len();
543*cdf0e10cSrcweir             *pBuf = ']';    pBuf++;
544*cdf0e10cSrcweir             *pBuf = aLineEndBuf[0]; pBuf++;
545*cdf0e10cSrcweir             if ( nLineEndLen == 2 )
546*cdf0e10cSrcweir             {
547*cdf0e10cSrcweir                 *pBuf = aLineEndBuf[1]; pBuf++;
548*cdf0e10cSrcweir             }
549*cdf0e10cSrcweir             pKey = pGroup->mpFirstKey;
550*cdf0e10cSrcweir             while ( pKey )
551*cdf0e10cSrcweir             {
552*cdf0e10cSrcweir                 nValueLen = pKey->maValue.Len();
553*cdf0e10cSrcweir                 if ( pKey->mbIsComment )
554*cdf0e10cSrcweir                 {
555*cdf0e10cSrcweir                     if ( nValueLen )
556*cdf0e10cSrcweir                     {
557*cdf0e10cSrcweir                         memcpy( pBuf, pKey->maValue.GetBuffer(), nValueLen );
558*cdf0e10cSrcweir                         pBuf += nValueLen;
559*cdf0e10cSrcweir                     }
560*cdf0e10cSrcweir                     *pBuf = aLineEndBuf[0]; pBuf++;
561*cdf0e10cSrcweir                     if ( nLineEndLen == 2 )
562*cdf0e10cSrcweir                     {
563*cdf0e10cSrcweir                         *pBuf = aLineEndBuf[1]; pBuf++;
564*cdf0e10cSrcweir                     }
565*cdf0e10cSrcweir                 }
566*cdf0e10cSrcweir                 else
567*cdf0e10cSrcweir                 {
568*cdf0e10cSrcweir                     nKeyLen = pKey->maKey.Len();
569*cdf0e10cSrcweir                     memcpy( pBuf, pKey->maKey.GetBuffer(), nKeyLen );
570*cdf0e10cSrcweir                     pBuf += nKeyLen;
571*cdf0e10cSrcweir                     *pBuf = '=';    pBuf++;
572*cdf0e10cSrcweir                     memcpy( pBuf, pKey->maValue.GetBuffer(), nValueLen );
573*cdf0e10cSrcweir                     pBuf += nValueLen;
574*cdf0e10cSrcweir                     *pBuf = aLineEndBuf[0]; pBuf++;
575*cdf0e10cSrcweir                     if ( nLineEndLen == 2 )
576*cdf0e10cSrcweir                     {
577*cdf0e10cSrcweir                         *pBuf = aLineEndBuf[1]; pBuf++;
578*cdf0e10cSrcweir                     }
579*cdf0e10cSrcweir                 }
580*cdf0e10cSrcweir 
581*cdf0e10cSrcweir                 pKey = pKey->mpNext;
582*cdf0e10cSrcweir             }
583*cdf0e10cSrcweir 
584*cdf0e10cSrcweir             // Leerzeile nach jeder Gruppe auch wieder speichern
585*cdf0e10cSrcweir             sal_uInt16 nEmptyLines = pGroup->mnEmptyLines;
586*cdf0e10cSrcweir             while ( nEmptyLines )
587*cdf0e10cSrcweir             {
588*cdf0e10cSrcweir                 *pBuf = aLineEndBuf[0]; pBuf++;
589*cdf0e10cSrcweir                 if ( nLineEndLen == 2 )
590*cdf0e10cSrcweir                 {
591*cdf0e10cSrcweir                     *pBuf = aLineEndBuf[1]; pBuf++;
592*cdf0e10cSrcweir                 }
593*cdf0e10cSrcweir                 nEmptyLines--;
594*cdf0e10cSrcweir             }
595*cdf0e10cSrcweir         }
596*cdf0e10cSrcweir 
597*cdf0e10cSrcweir         pGroup = pGroup->mpNext;
598*cdf0e10cSrcweir     }
599*cdf0e10cSrcweir 
600*cdf0e10cSrcweir     return pWriteBuf;
601*cdf0e10cSrcweir }
602*cdf0e10cSrcweir 
603*cdf0e10cSrcweir // -----------------------------------------------------------------------
604*cdf0e10cSrcweir 
605*cdf0e10cSrcweir static void ImplReadConfig( ImplConfigData* pData )
606*cdf0e10cSrcweir {
607*cdf0e10cSrcweir     sal_uIntPtr         nTimeStamp = 0;
608*cdf0e10cSrcweir     sal_uInt64 nRead = 0;
609*cdf0e10cSrcweir     sal_Bool            bRead = sal_False;
610*cdf0e10cSrcweir     sal_Bool                bIsUTF8BOM =sal_False;
611*cdf0e10cSrcweir     sal_uInt8*          pBuf = ImplSysReadConfig( pData->maFileName, nRead, bRead, bIsUTF8BOM, nTimeStamp );
612*cdf0e10cSrcweir 
613*cdf0e10cSrcweir     // Aus dem Buffer die Config-Verwaltungsliste aufbauen
614*cdf0e10cSrcweir     if ( pBuf )
615*cdf0e10cSrcweir     {
616*cdf0e10cSrcweir         ImplMakeConfigList( pData, pBuf, nRead );
617*cdf0e10cSrcweir         delete[] pBuf;
618*cdf0e10cSrcweir     }
619*cdf0e10cSrcweir     pData->mnTimeStamp = nTimeStamp;
620*cdf0e10cSrcweir     pData->mbModified  = sal_False;
621*cdf0e10cSrcweir     if ( bRead )
622*cdf0e10cSrcweir         pData->mbRead = sal_True;
623*cdf0e10cSrcweir     if ( bIsUTF8BOM )
624*cdf0e10cSrcweir         pData->mbIsUTF8BOM = sal_True;
625*cdf0e10cSrcweir }
626*cdf0e10cSrcweir 
627*cdf0e10cSrcweir // -----------------------------------------------------------------------
628*cdf0e10cSrcweir 
629*cdf0e10cSrcweir static void ImplWriteConfig( ImplConfigData* pData )
630*cdf0e10cSrcweir {
631*cdf0e10cSrcweir #ifdef DBG_UTIL
632*cdf0e10cSrcweir     if ( DbgIsAssert() )
633*cdf0e10cSrcweir     {
634*cdf0e10cSrcweir         if ( pData->mnTimeStamp != ImplSysGetConfigTimeStamp( pData->maFileName ) )
635*cdf0e10cSrcweir         {
636*cdf0e10cSrcweir             DBG_ERROR1( "Config overwrites modified configfile:\n %s", ByteString( pData->maFileName, RTL_TEXTENCODING_UTF8 ).GetBuffer() );
637*cdf0e10cSrcweir         }
638*cdf0e10cSrcweir     }
639*cdf0e10cSrcweir #endif
640*cdf0e10cSrcweir 
641*cdf0e10cSrcweir     // Aus der Config-Liste einen Buffer zusammenbauen
642*cdf0e10cSrcweir     sal_uIntPtr nBufLen;
643*cdf0e10cSrcweir     sal_uInt8*  pBuf = ImplGetConfigBuffer( pData, nBufLen );
644*cdf0e10cSrcweir     if ( pBuf )
645*cdf0e10cSrcweir     {
646*cdf0e10cSrcweir         if ( ImplSysWriteConfig( pData->maFileName, pBuf, nBufLen, pData->mbIsUTF8BOM, pData->mnTimeStamp ) )
647*cdf0e10cSrcweir             pData->mbModified = sal_False;
648*cdf0e10cSrcweir         delete[] pBuf;
649*cdf0e10cSrcweir     }
650*cdf0e10cSrcweir     else
651*cdf0e10cSrcweir         pData->mbModified = sal_False;
652*cdf0e10cSrcweir }
653*cdf0e10cSrcweir 
654*cdf0e10cSrcweir // -----------------------------------------------------------------------
655*cdf0e10cSrcweir 
656*cdf0e10cSrcweir static void ImplDeleteConfigData( ImplConfigData* pData )
657*cdf0e10cSrcweir {
658*cdf0e10cSrcweir     ImplKeyData*    pTempKey;
659*cdf0e10cSrcweir     ImplKeyData*    pKey;
660*cdf0e10cSrcweir     ImplGroupData*  pTempGroup;
661*cdf0e10cSrcweir     ImplGroupData*  pGroup = pData->mpFirstGroup;
662*cdf0e10cSrcweir     while ( pGroup )
663*cdf0e10cSrcweir     {
664*cdf0e10cSrcweir         pTempGroup = pGroup->mpNext;
665*cdf0e10cSrcweir 
666*cdf0e10cSrcweir         // Alle Keys loeschen
667*cdf0e10cSrcweir         pKey = pGroup->mpFirstKey;
668*cdf0e10cSrcweir         while ( pKey )
669*cdf0e10cSrcweir         {
670*cdf0e10cSrcweir             pTempKey = pKey->mpNext;
671*cdf0e10cSrcweir             delete pKey;
672*cdf0e10cSrcweir             pKey = pTempKey;
673*cdf0e10cSrcweir         }
674*cdf0e10cSrcweir 
675*cdf0e10cSrcweir         // Gruppe loeschen und weiterschalten
676*cdf0e10cSrcweir         delete pGroup;
677*cdf0e10cSrcweir         pGroup = pTempGroup;
678*cdf0e10cSrcweir     }
679*cdf0e10cSrcweir 
680*cdf0e10cSrcweir     pData->mpFirstGroup = NULL;
681*cdf0e10cSrcweir }
682*cdf0e10cSrcweir 
683*cdf0e10cSrcweir // =======================================================================
684*cdf0e10cSrcweir 
685*cdf0e10cSrcweir static ImplConfigData* ImplGetConfigData( const XubString& rFileName )
686*cdf0e10cSrcweir {
687*cdf0e10cSrcweir     ImplConfigData* pData;
688*cdf0e10cSrcweir 
689*cdf0e10cSrcweir     pData                   = new ImplConfigData;
690*cdf0e10cSrcweir     pData->maFileName       = rFileName;
691*cdf0e10cSrcweir     pData->mpFirstGroup     = NULL;
692*cdf0e10cSrcweir     pData->mnDataUpdateId   = 0;
693*cdf0e10cSrcweir     pData->meLineEnd        = LINEEND_CRLF;
694*cdf0e10cSrcweir     pData->mnRefCount       = 0;
695*cdf0e10cSrcweir     pData->mbRead           = sal_False;
696*cdf0e10cSrcweir     pData->mbIsUTF8BOM      = sal_False;
697*cdf0e10cSrcweir     ImplReadConfig( pData );
698*cdf0e10cSrcweir 
699*cdf0e10cSrcweir     return pData;
700*cdf0e10cSrcweir }
701*cdf0e10cSrcweir 
702*cdf0e10cSrcweir // -----------------------------------------------------------------------
703*cdf0e10cSrcweir 
704*cdf0e10cSrcweir static void ImplFreeConfigData( ImplConfigData* pDelData )
705*cdf0e10cSrcweir {
706*cdf0e10cSrcweir     ImplDeleteConfigData( pDelData );
707*cdf0e10cSrcweir     delete pDelData;
708*cdf0e10cSrcweir }
709*cdf0e10cSrcweir 
710*cdf0e10cSrcweir // =======================================================================
711*cdf0e10cSrcweir 
712*cdf0e10cSrcweir sal_Bool Config::ImplUpdateConfig() const
713*cdf0e10cSrcweir {
714*cdf0e10cSrcweir     // Wenn sich TimeStamp unterscheidet, dann Datei neu einlesen
715*cdf0e10cSrcweir     if ( mpData->mnTimeStamp != ImplSysGetConfigTimeStamp( maFileName ) )
716*cdf0e10cSrcweir     {
717*cdf0e10cSrcweir         ImplDeleteConfigData( mpData );
718*cdf0e10cSrcweir         ImplReadConfig( mpData );
719*cdf0e10cSrcweir         mpData->mnDataUpdateId++;
720*cdf0e10cSrcweir         return sal_True;
721*cdf0e10cSrcweir     }
722*cdf0e10cSrcweir     else
723*cdf0e10cSrcweir         return sal_False;
724*cdf0e10cSrcweir }
725*cdf0e10cSrcweir 
726*cdf0e10cSrcweir // -----------------------------------------------------------------------
727*cdf0e10cSrcweir 
728*cdf0e10cSrcweir ImplGroupData* Config::ImplGetGroup() const
729*cdf0e10cSrcweir {
730*cdf0e10cSrcweir     if ( !mpActGroup || (mnDataUpdateId != mpData->mnDataUpdateId) )
731*cdf0e10cSrcweir     {
732*cdf0e10cSrcweir         ImplGroupData* pPrevGroup = NULL;
733*cdf0e10cSrcweir         ImplGroupData* pGroup = mpData->mpFirstGroup;
734*cdf0e10cSrcweir         while ( pGroup )
735*cdf0e10cSrcweir         {
736*cdf0e10cSrcweir             if ( pGroup->maGroupName.EqualsIgnoreCaseAscii( maGroupName ) )
737*cdf0e10cSrcweir                 break;
738*cdf0e10cSrcweir 
739*cdf0e10cSrcweir             pPrevGroup = pGroup;
740*cdf0e10cSrcweir             pGroup = pGroup->mpNext;
741*cdf0e10cSrcweir         }
742*cdf0e10cSrcweir 
743*cdf0e10cSrcweir         // Falls Gruppe noch nicht existiert, dann dazufuegen
744*cdf0e10cSrcweir         if ( !pGroup )
745*cdf0e10cSrcweir         {
746*cdf0e10cSrcweir             pGroup               = new ImplGroupData;
747*cdf0e10cSrcweir             pGroup->mpNext       = NULL;
748*cdf0e10cSrcweir             pGroup->mpFirstKey   = NULL;
749*cdf0e10cSrcweir             pGroup->mnEmptyLines = 1;
750*cdf0e10cSrcweir             if ( pPrevGroup )
751*cdf0e10cSrcweir                 pPrevGroup->mpNext = pGroup;
752*cdf0e10cSrcweir             else
753*cdf0e10cSrcweir                 mpData->mpFirstGroup = pGroup;
754*cdf0e10cSrcweir         }
755*cdf0e10cSrcweir 
756*cdf0e10cSrcweir         // Gruppenname immer uebernehmen, da er auch in dieser Form
757*cdf0e10cSrcweir         // geschrieben werden soll. Ausserdem die Cache-Members der
758*cdf0e10cSrcweir         // Config-Klasse updaten
759*cdf0e10cSrcweir         pGroup->maGroupName             = maGroupName;
760*cdf0e10cSrcweir         ((Config*)this)->mnDataUpdateId = mpData->mnDataUpdateId;
761*cdf0e10cSrcweir         ((Config*)this)->mpActGroup     = pGroup;
762*cdf0e10cSrcweir     }
763*cdf0e10cSrcweir 
764*cdf0e10cSrcweir     return mpActGroup;
765*cdf0e10cSrcweir }
766*cdf0e10cSrcweir 
767*cdf0e10cSrcweir // =======================================================================
768*cdf0e10cSrcweir 
769*cdf0e10cSrcweir Config::Config()
770*cdf0e10cSrcweir {
771*cdf0e10cSrcweir     // Daten initialisieren und einlesen
772*cdf0e10cSrcweir     maFileName      = ImplMakeConfigName( NULL, NULL );
773*cdf0e10cSrcweir     mpData          = ImplGetConfigData( maFileName );
774*cdf0e10cSrcweir     mpActGroup      = NULL;
775*cdf0e10cSrcweir     mnDataUpdateId  = 0;
776*cdf0e10cSrcweir     mnLockCount     = 1;
777*cdf0e10cSrcweir     mbPersistence   = sal_True;
778*cdf0e10cSrcweir 
779*cdf0e10cSrcweir #ifdef DBG_UTIL
780*cdf0e10cSrcweir     DBG_TRACE( "Config::Config()" );
781*cdf0e10cSrcweir #endif
782*cdf0e10cSrcweir }
783*cdf0e10cSrcweir 
784*cdf0e10cSrcweir // -----------------------------------------------------------------------
785*cdf0e10cSrcweir 
786*cdf0e10cSrcweir Config::Config( const XubString& rFileName )
787*cdf0e10cSrcweir {
788*cdf0e10cSrcweir     // Daten initialisieren und einlesen
789*cdf0e10cSrcweir     maFileName      = toUncPath( rFileName );
790*cdf0e10cSrcweir     mpData          = ImplGetConfigData( maFileName );
791*cdf0e10cSrcweir     mpActGroup      = NULL;
792*cdf0e10cSrcweir     mnDataUpdateId  = 0;
793*cdf0e10cSrcweir     mnLockCount     = 1;
794*cdf0e10cSrcweir     mbPersistence   = sal_True;
795*cdf0e10cSrcweir 
796*cdf0e10cSrcweir #ifdef DBG_UTIL
797*cdf0e10cSrcweir     ByteString aTraceStr( "Config::Config( " );
798*cdf0e10cSrcweir     aTraceStr += ByteString( maFileName, RTL_TEXTENCODING_UTF8 );
799*cdf0e10cSrcweir     aTraceStr += " )";
800*cdf0e10cSrcweir     DBG_TRACE( aTraceStr.GetBuffer() );
801*cdf0e10cSrcweir #endif
802*cdf0e10cSrcweir }
803*cdf0e10cSrcweir 
804*cdf0e10cSrcweir // -----------------------------------------------------------------------
805*cdf0e10cSrcweir 
806*cdf0e10cSrcweir Config::~Config()
807*cdf0e10cSrcweir {
808*cdf0e10cSrcweir #ifdef DBG_UTIL
809*cdf0e10cSrcweir     DBG_TRACE( "Config::~Config()" );
810*cdf0e10cSrcweir #endif
811*cdf0e10cSrcweir 
812*cdf0e10cSrcweir     Flush();
813*cdf0e10cSrcweir     ImplFreeConfigData( mpData );
814*cdf0e10cSrcweir }
815*cdf0e10cSrcweir 
816*cdf0e10cSrcweir // -----------------------------------------------------------------------
817*cdf0e10cSrcweir 
818*cdf0e10cSrcweir String Config::GetDefDirectory()
819*cdf0e10cSrcweir {
820*cdf0e10cSrcweir     ::rtl::OUString aDefConfig;
821*cdf0e10cSrcweir     oslSecurity aSec = osl_getCurrentSecurity();
822*cdf0e10cSrcweir     osl_getConfigDir( aSec, &aDefConfig.pData );
823*cdf0e10cSrcweir     osl_freeSecurityHandle( aSec );
824*cdf0e10cSrcweir 
825*cdf0e10cSrcweir     return aDefConfig;
826*cdf0e10cSrcweir }
827*cdf0e10cSrcweir 
828*cdf0e10cSrcweir // -----------------------------------------------------------------------
829*cdf0e10cSrcweir 
830*cdf0e10cSrcweir XubString Config::GetConfigName( const XubString& rPath,
831*cdf0e10cSrcweir                                  const XubString& rBaseName )
832*cdf0e10cSrcweir {
833*cdf0e10cSrcweir     return ImplMakeConfigName( &rBaseName, &rPath );
834*cdf0e10cSrcweir }
835*cdf0e10cSrcweir 
836*cdf0e10cSrcweir // -----------------------------------------------------------------------
837*cdf0e10cSrcweir 
838*cdf0e10cSrcweir void Config::SetGroup( const ByteString& rGroup )
839*cdf0e10cSrcweir {
840*cdf0e10cSrcweir     // Wenn neue Gruppe gesetzt wird, muss beim naechsten mal die
841*cdf0e10cSrcweir     // Gruppe neu ermittelt werden
842*cdf0e10cSrcweir     if ( maGroupName != rGroup )
843*cdf0e10cSrcweir     {
844*cdf0e10cSrcweir         maGroupName     = rGroup;
845*cdf0e10cSrcweir         mnDataUpdateId  = mpData->mnDataUpdateId-1;
846*cdf0e10cSrcweir     }
847*cdf0e10cSrcweir }
848*cdf0e10cSrcweir 
849*cdf0e10cSrcweir // -----------------------------------------------------------------------
850*cdf0e10cSrcweir 
851*cdf0e10cSrcweir void Config::DeleteGroup( const ByteString& rGroup )
852*cdf0e10cSrcweir {
853*cdf0e10cSrcweir     // Config-Daten evt. updaten
854*cdf0e10cSrcweir     if ( !mnLockCount || !mpData->mbRead )
855*cdf0e10cSrcweir     {
856*cdf0e10cSrcweir         ImplUpdateConfig();
857*cdf0e10cSrcweir         mpData->mbRead = sal_True;
858*cdf0e10cSrcweir     }
859*cdf0e10cSrcweir 
860*cdf0e10cSrcweir     ImplGroupData* pPrevGroup = NULL;
861*cdf0e10cSrcweir     ImplGroupData* pGroup = mpData->mpFirstGroup;
862*cdf0e10cSrcweir     while ( pGroup )
863*cdf0e10cSrcweir     {
864*cdf0e10cSrcweir         if ( pGroup->maGroupName.EqualsIgnoreCaseAscii( rGroup ) )
865*cdf0e10cSrcweir             break;
866*cdf0e10cSrcweir 
867*cdf0e10cSrcweir         pPrevGroup = pGroup;
868*cdf0e10cSrcweir         pGroup = pGroup->mpNext;
869*cdf0e10cSrcweir     }
870*cdf0e10cSrcweir 
871*cdf0e10cSrcweir     if ( pGroup )
872*cdf0e10cSrcweir     {
873*cdf0e10cSrcweir         // Alle Keys loeschen
874*cdf0e10cSrcweir         ImplKeyData* pTempKey;
875*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
876*cdf0e10cSrcweir         while ( pKey )
877*cdf0e10cSrcweir         {
878*cdf0e10cSrcweir             pTempKey = pKey->mpNext;
879*cdf0e10cSrcweir             delete pKey;
880*cdf0e10cSrcweir             pKey = pTempKey;
881*cdf0e10cSrcweir         }
882*cdf0e10cSrcweir 
883*cdf0e10cSrcweir         // Gruppe weiterschalten und loeschen
884*cdf0e10cSrcweir         if ( pPrevGroup )
885*cdf0e10cSrcweir             pPrevGroup->mpNext = pGroup->mpNext;
886*cdf0e10cSrcweir         else
887*cdf0e10cSrcweir             mpData->mpFirstGroup = pGroup->mpNext;
888*cdf0e10cSrcweir         delete pGroup;
889*cdf0e10cSrcweir 
890*cdf0e10cSrcweir         // Config-Datei neu schreiben
891*cdf0e10cSrcweir         if ( !mnLockCount && mbPersistence )
892*cdf0e10cSrcweir             ImplWriteConfig( mpData );
893*cdf0e10cSrcweir         else
894*cdf0e10cSrcweir         {
895*cdf0e10cSrcweir             mpData->mbModified = sal_True;
896*cdf0e10cSrcweir         }
897*cdf0e10cSrcweir 
898*cdf0e10cSrcweir         // Gruppen auf ungluetig setzen
899*cdf0e10cSrcweir         mnDataUpdateId = mpData->mnDataUpdateId;
900*cdf0e10cSrcweir         mpData->mnDataUpdateId++;
901*cdf0e10cSrcweir     }
902*cdf0e10cSrcweir }
903*cdf0e10cSrcweir 
904*cdf0e10cSrcweir // -----------------------------------------------------------------------
905*cdf0e10cSrcweir 
906*cdf0e10cSrcweir ByteString Config::GetGroupName( sal_uInt16 nGroup ) const
907*cdf0e10cSrcweir {
908*cdf0e10cSrcweir     // Config-Daten evt. updaten
909*cdf0e10cSrcweir     if ( !mnLockCount )
910*cdf0e10cSrcweir         ImplUpdateConfig();
911*cdf0e10cSrcweir 
912*cdf0e10cSrcweir     ImplGroupData*  pGroup = mpData->mpFirstGroup;
913*cdf0e10cSrcweir     sal_uInt16          nGroupCount = 0;
914*cdf0e10cSrcweir     ByteString      aGroupName;
915*cdf0e10cSrcweir     while ( pGroup )
916*cdf0e10cSrcweir     {
917*cdf0e10cSrcweir         if ( nGroup == nGroupCount )
918*cdf0e10cSrcweir         {
919*cdf0e10cSrcweir             aGroupName = pGroup->maGroupName;
920*cdf0e10cSrcweir             break;
921*cdf0e10cSrcweir         }
922*cdf0e10cSrcweir 
923*cdf0e10cSrcweir         nGroupCount++;
924*cdf0e10cSrcweir         pGroup = pGroup->mpNext;
925*cdf0e10cSrcweir     }
926*cdf0e10cSrcweir 
927*cdf0e10cSrcweir     return aGroupName;
928*cdf0e10cSrcweir }
929*cdf0e10cSrcweir 
930*cdf0e10cSrcweir // -----------------------------------------------------------------------
931*cdf0e10cSrcweir 
932*cdf0e10cSrcweir sal_uInt16 Config::GetGroupCount() const
933*cdf0e10cSrcweir {
934*cdf0e10cSrcweir     // Config-Daten evt. updaten
935*cdf0e10cSrcweir     if ( !mnLockCount )
936*cdf0e10cSrcweir         ImplUpdateConfig();
937*cdf0e10cSrcweir 
938*cdf0e10cSrcweir     ImplGroupData*  pGroup = mpData->mpFirstGroup;
939*cdf0e10cSrcweir     sal_uInt16          nGroupCount = 0;
940*cdf0e10cSrcweir     while ( pGroup )
941*cdf0e10cSrcweir     {
942*cdf0e10cSrcweir         nGroupCount++;
943*cdf0e10cSrcweir         pGroup = pGroup->mpNext;
944*cdf0e10cSrcweir     }
945*cdf0e10cSrcweir 
946*cdf0e10cSrcweir     return nGroupCount;
947*cdf0e10cSrcweir }
948*cdf0e10cSrcweir 
949*cdf0e10cSrcweir // -----------------------------------------------------------------------
950*cdf0e10cSrcweir 
951*cdf0e10cSrcweir sal_Bool Config::HasGroup( const ByteString& rGroup ) const
952*cdf0e10cSrcweir {
953*cdf0e10cSrcweir     // Config-Daten evt. updaten
954*cdf0e10cSrcweir     if ( !mnLockCount )
955*cdf0e10cSrcweir         ImplUpdateConfig();
956*cdf0e10cSrcweir 
957*cdf0e10cSrcweir     ImplGroupData*  pGroup = mpData->mpFirstGroup;
958*cdf0e10cSrcweir     sal_Bool            bRet = sal_False;
959*cdf0e10cSrcweir 
960*cdf0e10cSrcweir     while( pGroup )
961*cdf0e10cSrcweir     {
962*cdf0e10cSrcweir         if( pGroup->maGroupName.EqualsIgnoreCaseAscii( rGroup ) )
963*cdf0e10cSrcweir         {
964*cdf0e10cSrcweir             bRet = sal_True;
965*cdf0e10cSrcweir             break;
966*cdf0e10cSrcweir         }
967*cdf0e10cSrcweir 
968*cdf0e10cSrcweir         pGroup = pGroup->mpNext;
969*cdf0e10cSrcweir     }
970*cdf0e10cSrcweir 
971*cdf0e10cSrcweir     return bRet;
972*cdf0e10cSrcweir }
973*cdf0e10cSrcweir 
974*cdf0e10cSrcweir // -----------------------------------------------------------------------
975*cdf0e10cSrcweir 
976*cdf0e10cSrcweir ByteString Config::ReadKey( const ByteString& rKey ) const
977*cdf0e10cSrcweir {
978*cdf0e10cSrcweir     return ReadKey( rKey, getEmptyByteString() );
979*cdf0e10cSrcweir }
980*cdf0e10cSrcweir 
981*cdf0e10cSrcweir // -----------------------------------------------------------------------
982*cdf0e10cSrcweir 
983*cdf0e10cSrcweir UniString Config::ReadKey( const ByteString& rKey, rtl_TextEncoding eEncoding ) const
984*cdf0e10cSrcweir {
985*cdf0e10cSrcweir     if ( mpData->mbIsUTF8BOM )
986*cdf0e10cSrcweir         eEncoding = RTL_TEXTENCODING_UTF8;
987*cdf0e10cSrcweir     return UniString( ReadKey( rKey ), eEncoding );
988*cdf0e10cSrcweir }
989*cdf0e10cSrcweir 
990*cdf0e10cSrcweir // -----------------------------------------------------------------------
991*cdf0e10cSrcweir 
992*cdf0e10cSrcweir ByteString Config::ReadKey( const ByteString& rKey, const ByteString& rDefault ) const
993*cdf0e10cSrcweir {
994*cdf0e10cSrcweir #ifdef DBG_UTIL
995*cdf0e10cSrcweir     ByteString aTraceStr( "Config::ReadKey( " );
996*cdf0e10cSrcweir     aTraceStr += rKey;
997*cdf0e10cSrcweir     aTraceStr += " ) from ";
998*cdf0e10cSrcweir     aTraceStr += GetGroup();
999*cdf0e10cSrcweir     aTraceStr += " in ";
1000*cdf0e10cSrcweir     aTraceStr += ByteString( maFileName, RTL_TEXTENCODING_UTF8 );
1001*cdf0e10cSrcweir     DBG_TRACE( aTraceStr.GetBuffer() );
1002*cdf0e10cSrcweir #endif
1003*cdf0e10cSrcweir 
1004*cdf0e10cSrcweir     // Config-Daten evt. updaten
1005*cdf0e10cSrcweir     if ( !mnLockCount )
1006*cdf0e10cSrcweir         ImplUpdateConfig();
1007*cdf0e10cSrcweir 
1008*cdf0e10cSrcweir     // Key suchen und Value zurueckgeben
1009*cdf0e10cSrcweir     ImplGroupData* pGroup = ImplGetGroup();
1010*cdf0e10cSrcweir     if ( pGroup )
1011*cdf0e10cSrcweir     {
1012*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
1013*cdf0e10cSrcweir         while ( pKey )
1014*cdf0e10cSrcweir         {
1015*cdf0e10cSrcweir             if ( !pKey->mbIsComment && pKey->maKey.EqualsIgnoreCaseAscii( rKey ) )
1016*cdf0e10cSrcweir                 return pKey->maValue;
1017*cdf0e10cSrcweir 
1018*cdf0e10cSrcweir             pKey = pKey->mpNext;
1019*cdf0e10cSrcweir         }
1020*cdf0e10cSrcweir     }
1021*cdf0e10cSrcweir 
1022*cdf0e10cSrcweir     return rDefault;
1023*cdf0e10cSrcweir }
1024*cdf0e10cSrcweir 
1025*cdf0e10cSrcweir // -----------------------------------------------------------------------
1026*cdf0e10cSrcweir 
1027*cdf0e10cSrcweir void Config::WriteKey( const ByteString& rKey, const ByteString& rStr )
1028*cdf0e10cSrcweir {
1029*cdf0e10cSrcweir #ifdef DBG_UTIL
1030*cdf0e10cSrcweir     ByteString aTraceStr( "Config::WriteKey( " );
1031*cdf0e10cSrcweir     aTraceStr += rKey;
1032*cdf0e10cSrcweir     aTraceStr += ", ";
1033*cdf0e10cSrcweir     aTraceStr += rStr;
1034*cdf0e10cSrcweir     aTraceStr += " ) to ";
1035*cdf0e10cSrcweir     aTraceStr += GetGroup();
1036*cdf0e10cSrcweir     aTraceStr += " in ";
1037*cdf0e10cSrcweir     aTraceStr += ByteString( maFileName, RTL_TEXTENCODING_UTF8 );
1038*cdf0e10cSrcweir     DBG_TRACE( aTraceStr.GetBuffer() );
1039*cdf0e10cSrcweir     DBG_ASSERTWARNING( rStr != ReadKey( rKey ), "Config::WriteKey() with the same Value" );
1040*cdf0e10cSrcweir #endif
1041*cdf0e10cSrcweir 
1042*cdf0e10cSrcweir     // Config-Daten evt. updaten
1043*cdf0e10cSrcweir     if ( !mnLockCount || !mpData->mbRead )
1044*cdf0e10cSrcweir     {
1045*cdf0e10cSrcweir         ImplUpdateConfig();
1046*cdf0e10cSrcweir         mpData->mbRead = sal_True;
1047*cdf0e10cSrcweir     }
1048*cdf0e10cSrcweir 
1049*cdf0e10cSrcweir     // Key suchen und Value setzen
1050*cdf0e10cSrcweir     ImplGroupData* pGroup = ImplGetGroup();
1051*cdf0e10cSrcweir     if ( pGroup )
1052*cdf0e10cSrcweir     {
1053*cdf0e10cSrcweir         ImplKeyData* pPrevKey = NULL;
1054*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
1055*cdf0e10cSrcweir         while ( pKey )
1056*cdf0e10cSrcweir         {
1057*cdf0e10cSrcweir             if ( !pKey->mbIsComment && pKey->maKey.EqualsIgnoreCaseAscii( rKey ) )
1058*cdf0e10cSrcweir                 break;
1059*cdf0e10cSrcweir 
1060*cdf0e10cSrcweir             pPrevKey = pKey;
1061*cdf0e10cSrcweir             pKey = pKey->mpNext;
1062*cdf0e10cSrcweir         }
1063*cdf0e10cSrcweir 
1064*cdf0e10cSrcweir         sal_Bool bNewValue;
1065*cdf0e10cSrcweir         if ( !pKey )
1066*cdf0e10cSrcweir         {
1067*cdf0e10cSrcweir             pKey              = new ImplKeyData;
1068*cdf0e10cSrcweir             pKey->mpNext      = NULL;
1069*cdf0e10cSrcweir             pKey->maKey       = rKey;
1070*cdf0e10cSrcweir             pKey->mbIsComment = sal_False;
1071*cdf0e10cSrcweir             if ( pPrevKey )
1072*cdf0e10cSrcweir                 pPrevKey->mpNext = pKey;
1073*cdf0e10cSrcweir             else
1074*cdf0e10cSrcweir                 pGroup->mpFirstKey = pKey;
1075*cdf0e10cSrcweir             bNewValue = sal_True;
1076*cdf0e10cSrcweir         }
1077*cdf0e10cSrcweir         else
1078*cdf0e10cSrcweir             bNewValue = pKey->maValue != rStr;
1079*cdf0e10cSrcweir 
1080*cdf0e10cSrcweir         if ( bNewValue )
1081*cdf0e10cSrcweir         {
1082*cdf0e10cSrcweir             pKey->maValue = rStr;
1083*cdf0e10cSrcweir 
1084*cdf0e10cSrcweir             if ( !mnLockCount && mbPersistence )
1085*cdf0e10cSrcweir                 ImplWriteConfig( mpData );
1086*cdf0e10cSrcweir             else
1087*cdf0e10cSrcweir             {
1088*cdf0e10cSrcweir                 mpData->mbModified = sal_True;
1089*cdf0e10cSrcweir             }
1090*cdf0e10cSrcweir         }
1091*cdf0e10cSrcweir     }
1092*cdf0e10cSrcweir }
1093*cdf0e10cSrcweir 
1094*cdf0e10cSrcweir // -----------------------------------------------------------------------
1095*cdf0e10cSrcweir 
1096*cdf0e10cSrcweir void Config::WriteKey( const ByteString& rKey, const UniString& rValue, rtl_TextEncoding eEncoding )
1097*cdf0e10cSrcweir {
1098*cdf0e10cSrcweir     if ( mpData->mbIsUTF8BOM  )
1099*cdf0e10cSrcweir         eEncoding = RTL_TEXTENCODING_UTF8;
1100*cdf0e10cSrcweir     WriteKey( rKey, ByteString( rValue, eEncoding ) );
1101*cdf0e10cSrcweir }
1102*cdf0e10cSrcweir 
1103*cdf0e10cSrcweir // -----------------------------------------------------------------------
1104*cdf0e10cSrcweir 
1105*cdf0e10cSrcweir void Config::DeleteKey( const ByteString& rKey )
1106*cdf0e10cSrcweir {
1107*cdf0e10cSrcweir     // Config-Daten evt. updaten
1108*cdf0e10cSrcweir     if ( !mnLockCount || !mpData->mbRead )
1109*cdf0e10cSrcweir     {
1110*cdf0e10cSrcweir         ImplUpdateConfig();
1111*cdf0e10cSrcweir         mpData->mbRead = sal_True;
1112*cdf0e10cSrcweir     }
1113*cdf0e10cSrcweir 
1114*cdf0e10cSrcweir     // Key suchen und Value setzen
1115*cdf0e10cSrcweir     ImplGroupData* pGroup = ImplGetGroup();
1116*cdf0e10cSrcweir     if ( pGroup )
1117*cdf0e10cSrcweir     {
1118*cdf0e10cSrcweir         ImplKeyData* pPrevKey = NULL;
1119*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
1120*cdf0e10cSrcweir         while ( pKey )
1121*cdf0e10cSrcweir         {
1122*cdf0e10cSrcweir             if ( !pKey->mbIsComment && pKey->maKey.EqualsIgnoreCaseAscii( rKey ) )
1123*cdf0e10cSrcweir                 break;
1124*cdf0e10cSrcweir 
1125*cdf0e10cSrcweir             pPrevKey = pKey;
1126*cdf0e10cSrcweir             pKey = pKey->mpNext;
1127*cdf0e10cSrcweir         }
1128*cdf0e10cSrcweir 
1129*cdf0e10cSrcweir         if ( pKey )
1130*cdf0e10cSrcweir         {
1131*cdf0e10cSrcweir             // Gruppe weiterschalten und loeschen
1132*cdf0e10cSrcweir             if ( pPrevKey )
1133*cdf0e10cSrcweir                 pPrevKey->mpNext = pKey->mpNext;
1134*cdf0e10cSrcweir             else
1135*cdf0e10cSrcweir                 pGroup->mpFirstKey = pKey->mpNext;
1136*cdf0e10cSrcweir             delete pKey;
1137*cdf0e10cSrcweir 
1138*cdf0e10cSrcweir             // Config-Datei neu schreiben
1139*cdf0e10cSrcweir             if ( !mnLockCount && mbPersistence )
1140*cdf0e10cSrcweir                 ImplWriteConfig( mpData );
1141*cdf0e10cSrcweir             else
1142*cdf0e10cSrcweir             {
1143*cdf0e10cSrcweir                 mpData->mbModified = sal_True;
1144*cdf0e10cSrcweir             }
1145*cdf0e10cSrcweir         }
1146*cdf0e10cSrcweir     }
1147*cdf0e10cSrcweir }
1148*cdf0e10cSrcweir 
1149*cdf0e10cSrcweir // -----------------------------------------------------------------------
1150*cdf0e10cSrcweir 
1151*cdf0e10cSrcweir sal_uInt16 Config::GetKeyCount() const
1152*cdf0e10cSrcweir {
1153*cdf0e10cSrcweir #ifdef DBG_UTIL
1154*cdf0e10cSrcweir     ByteString aTraceStr( "Config::GetKeyCount()" );
1155*cdf0e10cSrcweir     aTraceStr += " from ";
1156*cdf0e10cSrcweir     aTraceStr += GetGroup();
1157*cdf0e10cSrcweir     aTraceStr += " in ";
1158*cdf0e10cSrcweir     aTraceStr += ByteString( maFileName, RTL_TEXTENCODING_UTF8 );
1159*cdf0e10cSrcweir     DBG_TRACE( aTraceStr.GetBuffer() );
1160*cdf0e10cSrcweir #endif
1161*cdf0e10cSrcweir 
1162*cdf0e10cSrcweir     // Config-Daten evt. updaten
1163*cdf0e10cSrcweir     if ( !mnLockCount )
1164*cdf0e10cSrcweir         ImplUpdateConfig();
1165*cdf0e10cSrcweir 
1166*cdf0e10cSrcweir     // Key suchen und Value zurueckgeben
1167*cdf0e10cSrcweir     sal_uInt16 nCount = 0;
1168*cdf0e10cSrcweir     ImplGroupData* pGroup = ImplGetGroup();
1169*cdf0e10cSrcweir     if ( pGroup )
1170*cdf0e10cSrcweir     {
1171*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
1172*cdf0e10cSrcweir         while ( pKey )
1173*cdf0e10cSrcweir         {
1174*cdf0e10cSrcweir             if ( !pKey->mbIsComment )
1175*cdf0e10cSrcweir                 nCount++;
1176*cdf0e10cSrcweir 
1177*cdf0e10cSrcweir             pKey = pKey->mpNext;
1178*cdf0e10cSrcweir         }
1179*cdf0e10cSrcweir     }
1180*cdf0e10cSrcweir 
1181*cdf0e10cSrcweir     return nCount;
1182*cdf0e10cSrcweir }
1183*cdf0e10cSrcweir 
1184*cdf0e10cSrcweir // -----------------------------------------------------------------------
1185*cdf0e10cSrcweir 
1186*cdf0e10cSrcweir ByteString Config::GetKeyName( sal_uInt16 nKey ) const
1187*cdf0e10cSrcweir {
1188*cdf0e10cSrcweir #ifdef DBG_UTIL
1189*cdf0e10cSrcweir     ByteString aTraceStr( "Config::GetKeyName( " );
1190*cdf0e10cSrcweir     aTraceStr += ByteString::CreateFromInt32(nKey);
1191*cdf0e10cSrcweir     aTraceStr += " ) from ";
1192*cdf0e10cSrcweir     aTraceStr += GetGroup();
1193*cdf0e10cSrcweir     aTraceStr += " in ";
1194*cdf0e10cSrcweir     aTraceStr += ByteString( maFileName, RTL_TEXTENCODING_UTF8 );
1195*cdf0e10cSrcweir     DBG_TRACE( aTraceStr.GetBuffer() );
1196*cdf0e10cSrcweir #endif
1197*cdf0e10cSrcweir 
1198*cdf0e10cSrcweir     // Key suchen und Name zurueckgeben
1199*cdf0e10cSrcweir     ImplGroupData* pGroup = ImplGetGroup();
1200*cdf0e10cSrcweir     if ( pGroup )
1201*cdf0e10cSrcweir     {
1202*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
1203*cdf0e10cSrcweir         while ( pKey )
1204*cdf0e10cSrcweir         {
1205*cdf0e10cSrcweir             if ( !pKey->mbIsComment )
1206*cdf0e10cSrcweir             {
1207*cdf0e10cSrcweir                 if ( !nKey )
1208*cdf0e10cSrcweir                     return pKey->maKey;
1209*cdf0e10cSrcweir                 nKey--;
1210*cdf0e10cSrcweir             }
1211*cdf0e10cSrcweir 
1212*cdf0e10cSrcweir             pKey = pKey->mpNext;
1213*cdf0e10cSrcweir         }
1214*cdf0e10cSrcweir     }
1215*cdf0e10cSrcweir 
1216*cdf0e10cSrcweir     return getEmptyByteString();
1217*cdf0e10cSrcweir }
1218*cdf0e10cSrcweir 
1219*cdf0e10cSrcweir // -----------------------------------------------------------------------
1220*cdf0e10cSrcweir 
1221*cdf0e10cSrcweir ByteString Config::ReadKey( sal_uInt16 nKey ) const
1222*cdf0e10cSrcweir {
1223*cdf0e10cSrcweir #ifdef DBG_UTIL
1224*cdf0e10cSrcweir     ByteString aTraceStr( "Config::ReadKey( " );
1225*cdf0e10cSrcweir     aTraceStr += ByteString::CreateFromInt32( nKey );
1226*cdf0e10cSrcweir     aTraceStr += " ) from ";
1227*cdf0e10cSrcweir     aTraceStr += GetGroup();
1228*cdf0e10cSrcweir     aTraceStr += " in ";
1229*cdf0e10cSrcweir     aTraceStr += ByteString( maFileName, RTL_TEXTENCODING_UTF8 );
1230*cdf0e10cSrcweir     DBG_TRACE( aTraceStr.GetBuffer() );
1231*cdf0e10cSrcweir #endif
1232*cdf0e10cSrcweir 
1233*cdf0e10cSrcweir     // Key suchen und Value zurueckgeben
1234*cdf0e10cSrcweir     ImplGroupData* pGroup = ImplGetGroup();
1235*cdf0e10cSrcweir     if ( pGroup )
1236*cdf0e10cSrcweir     {
1237*cdf0e10cSrcweir         ImplKeyData* pKey = pGroup->mpFirstKey;
1238*cdf0e10cSrcweir         while ( pKey )
1239*cdf0e10cSrcweir         {
1240*cdf0e10cSrcweir             if ( !pKey->mbIsComment )
1241*cdf0e10cSrcweir             {
1242*cdf0e10cSrcweir                 if ( !nKey )
1243*cdf0e10cSrcweir                     return pKey->maValue;
1244*cdf0e10cSrcweir                 nKey--;
1245*cdf0e10cSrcweir             }
1246*cdf0e10cSrcweir 
1247*cdf0e10cSrcweir             pKey = pKey->mpNext;
1248*cdf0e10cSrcweir         }
1249*cdf0e10cSrcweir     }
1250*cdf0e10cSrcweir 
1251*cdf0e10cSrcweir     return getEmptyByteString();
1252*cdf0e10cSrcweir }
1253*cdf0e10cSrcweir 
1254*cdf0e10cSrcweir // -----------------------------------------------------------------------
1255*cdf0e10cSrcweir 
1256*cdf0e10cSrcweir void Config::EnterLock()
1257*cdf0e10cSrcweir {
1258*cdf0e10cSrcweir     // Config-Daten evt. updaten
1259*cdf0e10cSrcweir     if ( !mnLockCount )
1260*cdf0e10cSrcweir         ImplUpdateConfig();
1261*cdf0e10cSrcweir 
1262*cdf0e10cSrcweir     mnLockCount++;
1263*cdf0e10cSrcweir }
1264*cdf0e10cSrcweir 
1265*cdf0e10cSrcweir // -----------------------------------------------------------------------
1266*cdf0e10cSrcweir 
1267*cdf0e10cSrcweir void Config::LeaveLock()
1268*cdf0e10cSrcweir {
1269*cdf0e10cSrcweir     DBG_ASSERT( mnLockCount, "Config::LeaveLook() without Config::EnterLook()" );
1270*cdf0e10cSrcweir     mnLockCount--;
1271*cdf0e10cSrcweir 
1272*cdf0e10cSrcweir     if ( (mnLockCount == 0) && mpData->mbModified && mbPersistence )
1273*cdf0e10cSrcweir         ImplWriteConfig( mpData );
1274*cdf0e10cSrcweir }
1275*cdf0e10cSrcweir 
1276*cdf0e10cSrcweir // -----------------------------------------------------------------------
1277*cdf0e10cSrcweir 
1278*cdf0e10cSrcweir sal_Bool Config::Update()
1279*cdf0e10cSrcweir {
1280*cdf0e10cSrcweir     return ImplUpdateConfig();
1281*cdf0e10cSrcweir }
1282*cdf0e10cSrcweir 
1283*cdf0e10cSrcweir // -----------------------------------------------------------------------
1284*cdf0e10cSrcweir 
1285*cdf0e10cSrcweir void Config::Flush()
1286*cdf0e10cSrcweir {
1287*cdf0e10cSrcweir     if ( mpData->mbModified && mbPersistence )
1288*cdf0e10cSrcweir         ImplWriteConfig( mpData );
1289*cdf0e10cSrcweir }
1290*cdf0e10cSrcweir 
1291*cdf0e10cSrcweir // -----------------------------------------------------------------------
1292*cdf0e10cSrcweir 
1293*cdf0e10cSrcweir void Config::SetLineEnd( LineEnd eLineEnd )
1294*cdf0e10cSrcweir {
1295*cdf0e10cSrcweir     mpData->meLineEnd = eLineEnd;
1296*cdf0e10cSrcweir }
1297*cdf0e10cSrcweir 
1298*cdf0e10cSrcweir // -----------------------------------------------------------------------
1299*cdf0e10cSrcweir 
1300*cdf0e10cSrcweir LineEnd Config::GetLineEnd() const
1301*cdf0e10cSrcweir {
1302*cdf0e10cSrcweir     return mpData->meLineEnd;
1303*cdf0e10cSrcweir }
1304*cdf0e10cSrcweir 
1305