Update contrib.
1 // Copyright (c) 2005-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 #ifndef __INITEMPLATE_H__
19 #define __INITEMPLATE_H__
29 //Line Type enumeration //
37 //utils template class//
38 static void GetBufferL(RFs& aFs,const TDesC& aFileName,HBufC8*& aBufferPtr)
40 //open the file for reading
42 User::LeaveIfError(file.Open(aFs,aFileName,EFileShareReadersOnly));
44 //only process the file if it exists
46 CleanupClosePushL(file);
47 User::LeaveIfError(file.Size(filesize));
48 aBufferPtr=HBufC8::NewLC(filesize + 2); //In case '\r\n' needs to be appended
49 TPtr8 asWritableBuffer(aBufferPtr->Des());
50 User::LeaveIfError(file.Read(0,asWritableBuffer,filesize));
52 //check if '\n' is present
53 if( filesize > 0 && asWritableBuffer[filesize - 1] != '\n' )
55 asWritableBuffer.Append(_L8("\r\n"));
60 CleanupStack::PopAndDestroy();
63 template <class TLexX>
64 TBool IsEosAfterSpace(TLexX& aLex)
70 template<class TPtrCX>
71 void RemoveComment(TPtrCX& aLine)
73 //Trim anything on the rhs of any inline comment ';' or '#'
74 //in this case we need to locate which is the first occuring from left
75 TInt semiColon=aLine.Locate(';');
76 TInt hash=aLine.Locate('#');
77 if (semiColon!=KErrNotFound || hash!=KErrNotFound)
79 if (semiColon!=KErrNotFound && hash!=KErrNotFound)
81 aLine.Set(aLine.Left(semiColon<hash?semiColon:hash));
85 aLine.Set(aLine.Left(semiColon!=KErrNotFound?semiColon:hash));
90 template <class TPtrCX,class TLexX>
91 void RemoveWSBeforeAndAfter(const TPtrCX& aString,TPtrCX& aOutput)
93 TLexX parser(aString);
94 parser.SkipSpaceAndMark();
95 TInt offset=parser.Offset();
98 //now skip for whitespace and check end of string
99 //if this is eos we are done unget to mark and break
100 parser.SkipCharacters();
102 if (IsEosAfterSpace(parser))
104 parser.UnGetToMark();
108 TInt endOffset=parser.MarkedOffset();
109 aOutput.Set(aString.Mid(offset,endOffset-offset));
112 template<class TPtrCX,class TLexX>
113 TInt CountTokens(TPtrCX aLine)
115 TLexX myLexer(aLine);
117 while (!myLexer.Eos())
119 TPtrCX next = myLexer.NextToken();
128 static void CorruptIfL(TBool aIsCorrupt)
132 User::Leave(KErrCorrupt);
136 template<class HBufCX, class TPtrCX>
137 TPtrCX ExtractLineFromBuffer(const TPtrCX& aFileBuffer, TInt aStartOffset)
140 TPtrCX partialLine = aFileBuffer.Mid(aStartOffset, aFileBuffer.Length()- aStartOffset);
141 TInt delimOffset= partialLine.Locate(delim);
142 if (delimOffset==KErrNotFound)
146 return partialLine.Left(delimOffset+1);
149 //ini line template class //
150 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
151 NONSHARABLE_CLASS(CIniLine) : public CBase
154 static CIniLine* NewL(const TDesCX& aLine);
155 static CIniLine* NewLC(const TDesCX& aLine);
156 static CIniLine* NewLC(HBufCX* aLine, TLineType aTypeOfLine);
157 static CIniLine* NewFromSectionNameLC(const TDesCX& aSectionName);
158 static CIniLine* NewFromKeyValueLineLC(const TDesCX& aKeyName, const TDesCX& aKeyValue);
161 TLineType LineType() const;
162 void InsertBefore(CIniLine* aLine);
163 const HBufCX& LineBuffer() const;
164 TBool operator==(const CIniLine&) const;
168 TLineType DetermineTypeOfLine(const TDesCX& aLine);
169 void ConstructL(const TDesCX& aLine);
170 CIniLine(HBufCX* aLine, TLineType aLineType);
177 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
178 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>* CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::NewL(const TDesCX& aLine)
180 CIniLine* self=CIniLine::NewLC(aLine);
181 CleanupStack::Pop(self);
185 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
186 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>* CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::NewLC(const TDesCX& aLine)
188 CIniLine* self=new (ELeave) CIniLine();
189 CleanupStack::PushL(self);
190 self->ConstructL(aLine);
194 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
195 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>* CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::NewLC(HBufCX* aLine, TLineType aTypeOfLine)
197 CIniLine* self=new (ELeave) CIniLine(aLine, aTypeOfLine);
198 CleanupStack::PushL(self);
202 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
203 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::CIniLine(HBufCX* aLine, TLineType aTypeOfLine):iLine(aLine), iLineType(aTypeOfLine)
206 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
207 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::CIniLine()
210 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
211 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::~CIniLine()
217 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
218 void CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::ConstructL(const TDesCX& aLine)
220 HBufCX* myHBuf = aLine.AllocL();
221 this->iLine = myHBuf;
222 this->iLineType = this->DetermineTypeOfLine(aLine);
225 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
226 TLineType CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::DetermineTypeOfLine(const TDesCX& aLine)
228 // First recognized character wins. Otherwise return a comment.
229 for (TInt i=0; i< aLine.Length();i++)
231 TText myChar = aLine[i];
251 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
252 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>* CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::NewFromSectionNameLC(const TDesCX& aSectionName)
254 //Fabricate a section name.
255 HBufCX* myBuffer=HBufCX::NewLC(aSectionName.Size()+ 6); //6= "2 cr's and \r" + "[]".
256 TPtrX myBuf (myBuffer->Des());
257 myBuf.Append(_L("\r\n["));
258 myBuf.Append(aSectionName);
259 myBuf.Append(_L("]\r\n"));
260 CIniLine* myLine = CIniLine::NewLC(myBuffer, ESection);
261 CleanupStack::Pop(2);
262 CleanupStack::PushL(myLine);
266 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
267 CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>* CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::NewFromKeyValueLineLC(const TDesCX& aKeyName, const TDesCX& aKeyValue)
269 //Fabricate a key = value line.
270 HBufCX* myBuffer=HBufCX::NewLC(aKeyName.Size()+ aKeyValue.Size() + 3); //3= "1 cr + \r" + "=".
271 TPtrX myBuf (myBuffer->Des());
272 myBuf.Set(myBuffer->Des());
273 myBuf.Append(aKeyName);
275 myBuf.Append(aKeyValue);
276 myBuf.Append(_L("\r\n"));
277 CIniLine* myLine = CIniLine::NewLC(myBuffer, EKeyValue);
278 CleanupStack::Pop(2);
279 CleanupStack::PushL(myLine);
283 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
284 TLineType CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::LineType() const
289 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
290 void CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::InsertBefore(CIniLine* aLine)
292 aLine->iLink.AddBefore(&(this->iLink));
295 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
296 const HBufCX& CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::LineBuffer() const
301 template<class TDesCX,class HBufCX,class TPtrCX, class TPtrX>
302 TBool CIniLine<TDesCX,HBufCX,TPtrCX,TPtrX>::operator==(const CIniLine& aLine)const
304 if (LineType() != aLine.LineType())
307 if ((LineBuffer()).Compare(aLine.LineBuffer()) != 0)
312 //ini key template class//
313 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX, class TPtrX,class TLexX>
314 NONSHARABLE_CLASS(CIniKey): public CBase
317 static CIniKey* NewL(const TDesCX& aKeyName,const TDesCX& aKeyValue);
318 static CIniKey* NewLC(const TDesCX& aKeyName,const TDesCX& aKeyValue);
319 static CIniKey* NewLC(CIniLineX* aLine);
320 static TInt CompareKey(const CIniKey& aFirstKey,const CIniKey& aSecondKey);
322 void SetKeyValue(const TDesCX& aKeyValue);
323 TPtrCX KeyName() const;
324 TPtrCX KeyValue() const;
325 CIniLineX* LineSrc() const;
328 CIniKey(const TDesCX& aKeyName, const TDesCX& aKeyValue, CIniLineX* aLine);
329 CIniKey(const TDesCX& aKeyName, const TDesCX& aKeyValue);
330 static CIniKey* ParseKeyLineL(CIniLineX* aLine);
337 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
338 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>* CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::NewL(const TDesCX& aKeyName,const TDesCX& aKeyValue)
340 CIniKey* self=new (ELeave) CIniKey(aKeyName, aKeyValue);
344 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
345 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>* CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::NewLC(const TDesCX& aKeyName,const TDesCX& aKeyValue)
347 CIniKey* self=CIniKey::NewL(aKeyName,aKeyValue);
348 CleanupStack::PushL(self);
352 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
353 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>* CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::NewLC(CIniLineX* aLine)
355 CIniKey* self=ParseKeyLineL(aLine);
356 CleanupStack::PushL(self);
357 self->iLineSrc=aLine;
361 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
362 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::CIniKey(const TDesCX& aKeyName, const TDesCX& aKeyValue, CIniLineX* aLine):iKeyName(aKeyName),iKeyValue(aKeyValue),iLineSrc(aLine)
365 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
366 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::CIniKey(const TDesCX& aKeyName, const TDesCX& aKeyValue):iKeyName(aKeyName),iKeyValue(aKeyValue),iLineSrc(NULL)
369 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
370 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::~CIniKey()
373 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
374 TPtrCX CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::KeyName() const
379 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
380 TPtrCX CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::KeyValue() const
385 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
386 CIniLineX* CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::LineSrc() const
391 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
392 void CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::SetKeyValue(const TDesCX& aKeyValue)
394 iKeyValue.Set(aKeyValue);
397 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
398 TInt CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::CompareKey(const CIniKey& aFirstKey,const CIniKey& aSecondKey)
400 return (aFirstKey.KeyName()).Compare(aSecondKey.KeyName());
403 template<class TDesCX,class HBufCX,class TPtrCX, class CIniLineX,class TPtrX,class TLexX>
404 CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>* CIniKey<TDesCX,HBufCX,TPtrCX,CIniLineX,TPtrX,TLexX>::ParseKeyLineL(CIniLineX* aIniLine)
406 const HBufCX& myLine = aIniLine->LineBuffer();
407 //Get the first occurence of '=' sign
408 TInt equalOffset=myLine.Locate('=');
409 CorruptIfL(equalOffset==KErrNotFound);
411 TInt tokenCount = CountTokens<TPtrCX, TLexX>(myLine.Left(equalOffset));
412 CorruptIfL(tokenCount != 1);
414 //construct a TLex for checking the lhs of equal sign
416 parser.Assign(myLine.Left(equalOffset));
417 parser.SkipSpaceAndMark();
418 parser.SkipCharacters();
419 TPtrCX keyname=parser.MarkedToken();
420 TBool check = (keyname.Length()==0);
423 //strip off any white space before and after the value
424 TInt edgeOfValue=equalOffset;
425 while(++edgeOfValue < myLine.Length())
427 TChar t = (myLine)[edgeOfValue];
434 if (edgeOfValue < myLine.Length())
436 keyvalue.Set(myLine.Mid(edgeOfValue, (myLine.Length()-edgeOfValue-1)));
439 TInt hashOffset=keyvalue.Locate('#');
442 keyvalue.Set(keyvalue.Left(hashOffset));
444 TInt commentOffset=keyvalue.Locate(';');
445 if (commentOffset > 0)
447 keyvalue.Set(keyvalue.Left(commentOffset));
450 TInt index = keyvalue.Length()-1;
455 TChar t = keyvalue[index];
465 if (index < (keyvalue.Length()-1))
467 keyvalue.Set(keyvalue.Left(index+1));
471 CIniKey* myKey = new (ELeave) CIniKey(keyname,keyvalue, aIniLine);
475 //section template class//
476 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX,class CIniLineX,class TPtrX, class TLexX>
477 NONSHARABLE_CLASS(CIniSection): public CBase
480 static CIniSection* NewL(const TDesCX& aSectionName);
481 static CIniSection* NewLC(const TDesCX& aSectionName);
482 static CIniSection* NewL(const TDesCX& aSectionName, CIniLineX* aLineSrc);
483 static CIniSection* NewLC(CIniLineX* aLineSrc);
484 static TInt CompareSection(const CIniSection& aFirstSection,const CIniSection& aSecondSection);
487 void InsertKeyL(const TDesCX& aKeyName,const TDesCX& aKeyValue);
488 void InsertKeyL(const CIniKeyX* aKey);
489 void RemoveKeyL(const TDesCX& aKeyName);
491 TPtrCX KeyValueL(const TDesCX& aKeyName) const;
492 TInt KeyCount() const;
493 CIniKeyX* FindKeyL(const TDesCX& aKeyName) const;
494 TPtrCX SectionName() const;
496 CIniLineX* SrcLine();
497 void SetSrcLine(CIniLineX* aLineSrc);
499 //to be used to access individual item
500 const CIniKeyX* Key(TInt aIndex) const {return iKeyArray[aIndex];}
501 void ReserveSpaceInKeyArrayL(){iKeyArray.ReserveL(1);}
503 CIniSection(TPtrCX aSectionName);
504 CIniSection(TPtrCX aSectionName, CIniLineX* aLine);
505 static TPtrCX ParseSectionLineL(const HBufCX& aLine);
509 RPointerArray<CIniKeyX> iKeyArray;
510 CIniLineX* iLineSrc; //If no document object exists, then this object is not used.
513 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
514 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>* CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::NewL(const TDesCX& aSectionName)
516 CIniSection* self=new (ELeave) CIniSection(aSectionName);
520 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
521 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>* CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::NewL(const TDesCX& aSectionName,CIniLineX* aLineSrc)
523 CIniSection* self=new (ELeave) CIniSection(aSectionName, aLineSrc);
527 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
528 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>* CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::NewLC(const TDesCX& aSectionName)
530 CIniSection* self=new (ELeave) CIniSection(aSectionName);
531 CleanupStack::PushL(self);
535 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
536 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>* CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::NewLC(CIniLineX* aLineSrc)
538 const HBufCX& buffer = aLineSrc->LineBuffer();
539 TPtrCX sectionName = ParseSectionLineL(buffer);
540 CIniSection* self=new (ELeave) CIniSection(sectionName, aLineSrc);
541 CleanupStack::PushL(self);
545 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
546 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::CIniSection(TPtrCX aSectionName, CIniLineX* aLine):iSectionName(aSectionName), iLineSrc(aLine)
551 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
552 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::CIniSection(TPtrCX aSectionName):iSectionName(aSectionName)
557 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
558 CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::~CIniSection()
560 iKeyArray.ResetAndDestroy();
563 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX, class TLexX>
564 void CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::SetSrcLine(CIniLineX* aLineSrc)
573 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
574 TInt CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::KeyCount() const
576 return iKeyArray.Count();
579 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
580 TPtrCX CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::SectionName() const
585 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
586 CIniLineX* CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::SrcLine()
591 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
592 TInt CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::CompareSection(const CIniSection& aFirstSection,const CIniSection& aSecondSection)
594 return aFirstSection.SectionName().Compare(aSecondSection.SectionName());
597 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX, class TLexX>
598 void CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::InsertKeyL(const TDesCX& aKeyName,const TDesCX& aKeyValue)
600 CIniKeyX* newKey=CIniKeyX::NewLC(aKeyName,aKeyValue);
602 CleanupStack::Pop(newKey);
605 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX, class TLexX>
606 void CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::InsertKeyL(const CIniKeyX* aKey)
608 iKeyArray.InsertInOrderL(aKey,TLinearOrder<CIniKeyX>(CIniKeyX::CompareKey));
611 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
612 void CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::RemoveKeyL(const TDesCX& aKeyName)
614 CIniKeyX* key=FindKeyL(aKeyName);
615 CIniLineX* myLine = key->LineSrc();
616 TInt index=iKeyArray.FindInOrderL(key,TLinearOrder<CIniKeyX>(CIniKeyX::CompareKey));
617 iKeyArray.Remove(index);
622 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX, class TLexX>
623 CIniKeyX* CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::FindKeyL(const TDesCX& aKeyName) const
625 CIniKeyX* key=CIniKeyX::NewLC(aKeyName, aKeyName);
626 TInt index=iKeyArray.FindInOrderL(key,TLinearOrder<CIniKeyX>(CIniKeyX::CompareKey));
627 CleanupStack::PopAndDestroy(key);
628 return iKeyArray[index];
631 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
632 TPtrCX CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::KeyValueL(const TDesCX& aKeyName) const
634 CIniKeyX* key=FindKeyL(aKeyName);
635 return key->KeyValue();
638 template<class TDesCX,class HBufCX,class TPtrCX,class CIniKeyX, class CIniLineX, class TPtrX,class TLexX>
639 TPtrCX CIniSection<TDesCX,HBufCX,TPtrCX,CIniKeyX,CIniLineX,TPtrX,TLexX>::ParseSectionLineL(const HBufCX& aLine)
641 //find the last terminating bracket ']'
642 //anything in between is considered the section name
643 TInt endBracket=aLine.Locate(']');
644 //Check if the terminating bracket exits. Leave with an error code if not found.
645 CorruptIfL(endBracket == KErrNotFound);
647 TInt startBracket=aLine.Locate('[');
648 CorruptIfL(startBracket == KErrNotFound);
650 TPtrCX sectionName=aLine.Mid(startBracket+1, (endBracket-startBracket)-1);
652 //corrupt if empty section e.g []
653 TBool check = (sectionName.Length()==0);
656 TLexX lex(sectionName);
658 //Check for any white space within section name
659 //check any white spaces directly after the '['
661 check = (lex.Offset()>lex.MarkedOffset());
663 lex.SkipCharacters();
665 //At this stage we can extract the section name
666 sectionName.Set(lex.MarkedToken());
667 return (sectionName);
670 //iterator template class//
671 template<class TDesCX,class TPtrCX,class CIniSectionX,class CIniDocumentX,class CIniKeyX>
672 NONSHARABLE_CLASS(CIniSecIterImplX): public CBase
675 CIniSecIterImplX():iCurrentIndex(0),iSection(NULL){}
676 TBool Next(TPtrCX& aKeyName,TPtrCX& aKeyValue);
677 void Reset(){iCurrentIndex=0;}
678 TBool End(){return iCurrentIndex>=iSection->KeyCount();}
681 CIniSectionX* iSection;
684 template<class TDesCX,class TPtrCX,class CIniSectionX,class CIniDocumentX,class CIniKeyX>
685 TBool CIniSecIterImplX<TDesCX,TPtrCX,CIniSectionX,CIniDocumentX,CIniKeyX>::Next(TPtrCX& aKeyName,TPtrCX& aKeyValue)
689 if (iCurrentIndex<iSection->KeyCount())
691 const CIniKeyX* key=iSection->Key(iCurrentIndex);
692 aKeyName.Set(key->KeyName());
693 aKeyValue.Set(key->KeyValue());
702 //inidocument template class//
703 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
704 NONSHARABLE_CLASS(CIniDocumentTmplX) : public CBase
707 CIniDocumentTmplX(RFs& aFs,TBool aNarrow);
708 ~CIniDocumentTmplX();
709 void FlushL(const TDesC& aFileName);
711 TInt GetSectionList(RArray<TPtrCX>& aSectionList) const;
712 void GetKeyValueL(const TDesCX& aSectionName,const TDesCX& aKeyName,TPtrCX& aKeyValue) const;
713 CIniSectionX* SectionL(const TDesCX& aSectionName) const;
715 CIniSectionX* AddSectionL(const TDesCX& aSectionName, CIniLineX* aLine);
716 CIniSectionX* AddSectionL(const TDesCX& aSectionName);
717 void AddSectionL(const CIniSectionX* aSection);
719 void RemoveSectionL(const TDesCX& aSectionName);
720 void SetKeyL(const TDesCX& aSectionName,const TDesCX& aKeyName,const TDesCX& aKeyValue);
721 void RemoveKeyL(const TDesCX& aSectionName,const TDesCX& aKeyName);
722 void RemoveLineL(CIniLineX* aLine);
723 void InsertKeyValueIntoQueueL(CIniLineX* aKeyValue, CIniSectionX& aSection);
724 void AppendIntoQueue(CIniLineX* aSection);
725 TBool CompareDocs(CIniDocumentTmplX& aDoc);
728 CIniSectionX* FindSectionL(const TDesCX& aSectionName) const;
729 void ConstructL(const TDesC& aFileName);
734 TDblQue<CIniLineX> iDocument;
735 RPointerArray<CIniSectionX> iSectionArray;
738 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX, class CIniLineX,class CIniKeyX>
739 CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::CIniDocumentTmplX(RFs& aFs,TBool aNarrow):iNarrow(aNarrow),iFs(aFs)
741 iSectionArray.Reset();
742 iDocument.SetOffset(_FOFF(CIniLineX, iLink));
746 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
747 TBool CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::CompareDocs(CIniDocumentTmplX& aDoc)
749 CIniLineX* lineFromA; // This object is a. The passed in one is b.
750 CIniLineX* lineFromB;
751 TDblQueIter<CIniLineX> itera(this->iDocument);
752 TDblQueIter<CIniLineX> iterb(aDoc.iDocument);
754 TBool NotDoneYet= ETrue;
759 if ((lineFromA == NULL) || (lineFromB == NULL))
762 if (!(*lineFromA == *lineFromB))
768 if ( (lineFromA) || (lineFromB))
771 if ((this->iSectionArray.Count()) != (aDoc.iSectionArray.Count()))
776 for (TInt i=0; i<this->iSectionArray.Count(); i++)
778 TPtrCX secNameA = this->iSectionArray[i]->SectionName();
779 TPtrCX secNameB = aDoc.iSectionArray[i]->SectionName();
780 if ((secNameA.Compare(secNameB))!= 0)
783 if ((this->iSectionArray[i]->KeyCount()) != (aDoc.iSectionArray[i]->KeyCount()))
786 lineFromA = this->iSectionArray[i]->SrcLine();
787 lineFromB = aDoc.iSectionArray[i]->SrcLine();
789 if (!(*lineFromA == *lineFromB))
794 for (TInt k=0; k<this->iSectionArray[i]->KeyCount(); k++)
796 const CIniKeyX* keyA = this->iSectionArray[i]->Key(k);
797 const CIniKeyX* keyB = aDoc.iSectionArray[i]->Key(k);
798 if (keyA->KeyName().Compare(keyB->KeyName()) != 0)
800 if (keyA->KeyValue().Compare(keyB->KeyValue()) != 0)
803 lineFromA = keyA->LineSrc();
804 lineFromB = keyB->LineSrc();
806 if (!(*lineFromA == *lineFromB))
816 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX, class CIniLineX,class CIniKeyX>
817 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::AppendIntoQueue(CIniLineX* aLine)
819 if (iDocument.IsEmpty())
821 iDocument.AddFirst(*aLine);
825 iDocument.AddLast(*aLine); //always at the end of the list.
829 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX, class CIniLineX,class CIniKeyX>
830 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::InsertKeyValueIntoQueueL(CIniLineX* aKeyValue,CIniSectionX& aSection)
832 // Find the end of the section (if exists).
833 CIniLineX* lastEntry;
835 TDblQueIter<CIniLineX> iter(iDocument);
837 lastEntry = aSection.SrcLine(); //by default insert into queue after this line entry.
838 iter.Set(*lastEntry); // search after this point for a keyvalue line.
839 iter++; //point to next line
840 while ((line = iter++) != NULL)
843 TLineType typeOfLine = line->LineType();
844 if (typeOfLine == ESection)
852 AppendIntoQueue(aKeyValue);
856 if (iDocument.IsLast(lastEntry))
858 AppendIntoQueue(aKeyValue);
862 lastEntry->InsertBefore(aKeyValue);
867 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX, class CIniLineX,class CIniKeyX>
868 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::FlushL(const TDesC& aFileName)
870 //we need to get the path for the target file
872 User::LeaveIfError(originFile.Set(aFileName,NULL,NULL));
874 //Need to ensure path exists
875 TInt error=iFs.MkDirAll(originFile.DriveAndPath());
876 if (error!=KErrAlreadyExists)
878 User::LeaveIfError(error);
881 //Constructing the temp file name with the same directory and path
884 User::LeaveIfError(file.Temp(iFs,originFile.DriveAndPath(),tempFile,EFileWrite));
885 CleanupClosePushL(file);
890 TChar myChar(0xFEFF);
891 const unsigned char* myChar8=reinterpret_cast<const unsigned char*>(&myChar);
892 bufPtr.Set(myChar8, 2);
893 User::LeaveIfError(file.Write(bufPtr));
896 TDblQueIter<CIniLineX> iter(iDocument);
898 while ((line = iter++) != NULL)
900 const HBufCX& tempBuffer=line->LineBuffer();
901 const TUint8* rawptr8=reinterpret_cast<const TUint8*>(tempBuffer.Ptr());
904 bufPtr.Set(rawptr8,tempBuffer.Length());
908 bufPtr.Set(rawptr8,tempBuffer.Length()*2);
910 User::LeaveIfError(file.Write(bufPtr));
912 User::LeaveIfError(file.Flush());
913 CleanupStack::PopAndDestroy();
915 //Finally try replacing or creating a new file
916 //depending on whether file exist or not
918 if (iFs.Entry(aFileName,entry)==KErrNone)
920 User::LeaveIfError(iFs.Replace(tempFile,aFileName));
924 User::LeaveIfError(iFs.Rename(tempFile,aFileName));
928 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX, class CIniLineX,class CIniKeyX>
929 TInt CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::GetSectionList(RArray<TPtrCX>& aSectionList) const
931 //Reset the list first
932 aSectionList.Reset();
933 for (TInt i=0;i<iSectionArray.Count();i++)
935 TPtrCX nameptr=iSectionArray[i]->SectionName();
936 TInt ret=aSectionList.Append(nameptr);
937 //If error half way fo the copying, reset the list first
940 aSectionList.Reset();
947 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX, class CIniLineX,class CIniKeyX>
948 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::GetKeyValueL(const TDesCX& aSectionName,const TDesCX& aKeyName,TPtrCX& aKeyValue) const
950 //Find if section exists
951 CIniSectionX* section = SectionL(aSectionName);
952 aKeyValue.Set(section->KeyValueL(aKeyName));
955 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
956 CIniSectionX* CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::SectionL(const TDesCX& aSectionName) const
958 CIniSectionX* section = FindSectionL(aSectionName);
962 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
963 CIniSectionX* CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::AddSectionL(const TDesCX& aSectionName)
965 //Find if section exists
966 CIniSectionX* section = NULL;
967 TRAPD(ret, section = FindSectionL(aSectionName));
968 if ((ret == KErrNone) && (section != NULL))
969 User::Leave(KErrAlreadyExists);
971 //Fabricate objects upfront and on the cleanup stack to ensure their deletion if OOM conditions occur.
972 CIniLineX* myLine = CIniLineX::NewFromSectionNameLC(aSectionName);
973 AppendIntoQueue(myLine);
975 // reserve 1 slot for the new section name to provoke the out of memory condition
976 // before actually inserting the object to avoid having to go back and remove it if it fails.
977 this->iSectionArray.ReserveL(1);
979 // If we have not left, we are sure adding a section should not leave due to OOM.
980 CIniSectionX* newSection=AddSectionL(aSectionName, myLine);
981 CleanupStack::Pop(myLine);
985 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
986 CIniSectionX* CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::AddSectionL(const TDesCX& aSectionName, CIniLineX* aLine)
988 CIniSectionX* newSection=CIniSectionX::NewLC(aSectionName);
989 AddSectionL(newSection);
990 newSection->SetSrcLine(aLine);
991 CleanupStack::Pop(newSection);
995 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
996 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::AddSectionL(const CIniSectionX* aSection)
998 iSectionArray.InsertInOrderL(aSection,TLinearOrder<CIniSectionX>(CIniSectionX::CompareSection));
1001 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
1002 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::RemoveSectionL(const TDesCX& aSectionName)
1004 CIniSectionX* section = FindSectionL(aSectionName);
1005 CIniLineX* line = section->SrcLine();
1006 ASSERT(line); //This should never happen ie. the Key object must have a line associated with it.
1008 // remove the lines after to the start of the next section
1009 TDblQueIter<CIniLineX> iter(iDocument);
1010 iter.Set(*line); // search after this point for a keyvalue line.
1011 TBool atStartofSection = ETrue;
1012 while ((line = iter++) != NULL)
1014 TLineType typeOfLine = line->LineType();
1015 if ((typeOfLine == ESection))
1017 if (!atStartofSection)
1024 atStartofSection = EFalse;
1029 delete line; //includes comment lines which are only in the document object. Otherwise you could just ~section.
1033 TInt index=this->iSectionArray.FindInOrderL(section,TLinearOrder<CIniSectionX>(CIniSectionX::CompareSection));
1034 iSectionArray.Remove(index);
1038 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
1039 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::SetKeyL(const TDesCX& aSectionName,const TDesCX& aKeyName,const TDesCX& aKeyValue)
1041 //Find if section exists
1042 CIniSectionX* section = NULL;
1043 TRAPD(ret, section = FindSectionL(aSectionName));
1044 if (ret == KErrNone)
1046 TRAPD(ret2, RemoveKeyL(aSectionName, aKeyName));
1047 if ((ret2 != KErrNone) && (ret2 != KErrNotFound))
1052 CIniLineX* myLine2 = CIniLineX::NewFromKeyValueLineLC(aKeyName, aKeyValue);
1053 CIniKeyX* myKey = CIniKeyX::NewLC(myLine2);
1054 InsertKeyValueIntoQueueL(myLine2, *section);
1056 // reserve 1 slot in the key array to provoke the out of memory condition
1057 // before inserting the key to avoid having to go back and remove the key if OOM occurs.
1058 section->ReserveSpaceInKeyArrayL();
1059 section->InsertKeyL(myKey);
1061 CleanupStack::Pop(2);
1064 if (ret!=KErrNotFound)
1069 //Fabricate objects upfront and on the cleanup stack to ensure their deletion if OOM conditions occur.
1070 CIniLineX* myLine = CIniLineX::NewFromSectionNameLC(aSectionName);
1071 AppendIntoQueue(myLine);
1072 CIniLineX* myLine3 = CIniLineX::NewFromKeyValueLineLC(aKeyName, aKeyValue);
1073 AppendIntoQueue(myLine3);
1074 CIniKeyX* myKey = CIniKeyX::NewLC(myLine3);
1076 // reserve 1 slot for the new section name to provoke the out of memory condition
1077 // before actually inserting the object to avoid having to go back and remove it if it fails.
1078 this->iSectionArray.ReserveL(1);
1080 // If we have not left, we are sure adding a section should not leave due to OOM.
1081 CIniSectionX* newSection=AddSectionL(aSectionName, myLine);
1083 // reserve 1 slot in the key array of the newly created empty section to provoke the out of memory condition
1084 // before inserting the key to avoid having to go back and remove the key if OOM occurs.
1085 TRAPD(err, newSection->ReserveSpaceInKeyArrayL());
1086 if (err == KErrNoMemory)
1088 // In this case, we have an empty newSection object to be removed.
1089 TInt index=this->iSectionArray.FindInOrder(newSection,TLinearOrder<CIniSectionX>(CIniSectionX::CompareSection));
1090 iSectionArray.Remove(index);
1092 CleanupStack::PopAndDestroy(3);
1093 User::Leave(KErrNoMemory);
1101 // we are now sure the next inserts will not fail due to OOM.
1102 newSection->InsertKeyL(myKey);
1103 CleanupStack::Pop(3);
1106 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
1107 void CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::RemoveKeyL(const TDesCX& aSectionName,const TDesCX& aKeyName)
1109 //Find if section exists
1110 CIniSectionX* section = FindSectionL(aSectionName);
1111 section->RemoveKeyL(aKeyName);
1114 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
1115 CIniSectionX* CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::FindSectionL(const TDesCX& aSectionName) const
1117 CIniSectionX* newSection=CIniSectionX::NewLC(aSectionName);
1118 TInt index=iSectionArray.FindInOrderL(newSection,TLinearOrder<CIniSectionX>(CIniSectionX::CompareSection));
1119 CleanupStack::PopAndDestroy(newSection);
1120 return iSectionArray[index];
1123 template<class TDesCX,class TPtrCX,class CIniSectionX,class TLexX,class TPtrX,class HBufCX,class CIniLineX,class CIniKeyX>
1124 CIniDocumentTmplX<TDesCX,TPtrCX,CIniSectionX,TLexX,TPtrX,HBufCX,CIniLineX,CIniKeyX>::~CIniDocumentTmplX()
1127 TDblQueIter<CIniLineX> iter(iDocument);
1129 while ((line = iter++) != NULL)
1133 iSectionArray.ResetAndDestroy();
1136 //light weight template class//
1137 template<class TDesCX,class TPtrCX,class HBufCX,class TLexX,class TPtrX,class TBufX,class CIniLineX>
1138 NONSHARABLE_CLASS(CIniFileImplX): public CBase
1141 CIniFileImplX():iBuffer(NULL) {}
1142 TInt FindVar(const TDesCX& aSection,const TDesCX& aKey,TPtrCX& aValue);
1143 void ProcessBufferL(const TDesCX& aPtr);
1152 template<class TDesCX,class TPtrCX,class HBufCX,class TLexX,class TPtrX,class TBufX,class CIniLineX>
1153 void CIniFileImplX<TDesCX,TPtrCX,HBufCX,TLexX,TPtrX,TBufX,CIniLineX>::ProcessBufferL(const TDesCX& aPtr)
1155 iBuffer = HBufCX::NewL(aPtr.Length() + sizeof('\n')); //1 for cr.
1156 TPtrX bufferPtr(iBuffer->Des());
1157 TLexX lineParser(aPtr);
1158 while (!lineParser.Eos())
1160 //Get line by line which is terminated by \n
1161 lineParser.SkipSpaceAndMark();
1162 while (!lineParser.Eos() && lineParser.Peek()!='\n')
1166 TPtrCX line=lineParser.MarkedToken();
1167 //Now append the LHS of the comment to the buffer
1168 RemoveComment(line);
1170 //trim any white space before section,key- value and comments
1171 TLexX removeWs(line);
1172 removeWs.SkipSpace();
1173 if (!removeWs.Eos())
1175 bufferPtr.Append(line);
1176 //this added own delimiter is useful for syntax checking
1177 bufferPtr.Append('\n');
1182 template<class TDesCX,class TPtrCX,class HBufCX,class TLexX,class TPtrX,class TBufX,class CIniLineX>
1183 TInt CIniFileImplX<TDesCX,TPtrCX,HBufCX,TLexX,TPtrX,TBufX,CIniLineX>::FindVar(const TDesCX& aSection,const TDesCX& aKey,TPtrCX& aValue)
1185 //first find the start of the section [aSectionName]
1186 //5 here is []\n and the wildcard *\n[sectionname]*
1187 HBufCX* section=HBufCX::New(aSection.Length()+5);
1189 return KErrNoMemory;
1190 TPtrX sectionPtr(section->Des());
1191 sectionPtr.Append(_L("*\n["));
1192 sectionPtr.Append(aSection);
1193 sectionPtr.Append(_L("]*"));
1195 // Find a match purely on the section and not on the value, i.e. checking for a match on
1196 // [SECTION] does not find a match on key=[SECTION].
1198 TBool firstSection=ETrue;
1199 // Matching on first section is different as it should be the first
1200 // entry, i.e. no *\n before [aSection]* .
1201 TInt sectionStart=iBuffer->Match(sectionPtr.Mid(2));
1202 if (sectionStart==KErrNotFound)
1204 firstSection=EFalse;
1205 // Then try to match on any section after the first, i.e. *\n[aSection]* .
1206 sectionStart=iBuffer->Match(sectionPtr);
1209 if (sectionStart!=KErrNotFound)
1211 // There will be always \n added after section, key or value.
1212 TPtrCX searchBoundary(iBuffer->Mid(sectionStart+aSection.Length()+(firstSection?2:3)));
1213 searchBoundary.Set(searchBoundary.Mid(searchBoundary.Locate('\n')));
1214 //now need to locate the start of next section or end of file
1216 matchPattern.Append(_L("*\n[*]*"));
1217 TInt endSearchPos=searchBoundary.Match(matchPattern);
1218 if (endSearchPos!=KErrNotFound)
1220 //if can find next section block reduce the search boundary
1221 searchBoundary.Set(searchBoundary.Left(++endSearchPos));
1223 //now time to find the first exact match key i.e "key" does not match "subkey"
1225 //search through the boundary for matching key
1226 while ((equalOffset=searchBoundary.Locate('='))!=KErrNotFound)
1228 //skip over the \n char
1229 searchBoundary.Set(searchBoundary.Mid(1));
1230 TLexX keyCheck(searchBoundary.Left(--equalOffset));
1231 keyCheck.SkipSpaceAndMark();
1232 keyCheck.SkipCharacters();
1233 TPtrCX key(keyCheck.MarkedToken());
1234 keyCheck.SkipSpace();
1235 if (keyCheck.Eos() && key.Compare(aKey)==0)
1237 TInt location = searchBoundary.Locate('\n');
1238 __ASSERT_DEBUG(location >= 0, User::Invariant());
1239 //trim any white space before and after the value
1240 TPtrCX rhs(searchBoundary.Mid(equalOffset+1,location-equalOffset-1));
1241 RemoveWSBeforeAndAfter<TPtrCX,TLexX>(rhs,aValue);
1244 //update the next search boundary
1245 searchBoundary.Set(searchBoundary.Mid(searchBoundary.Locate('\n')));
1249 return KErrNotFound;
1253 #endif//__BAINIPARSERIMPL_H__