Update contrib.
1 // Copyright (c) 1998-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 // Class CDbReverseIterator
20 NONSHARABLE_CLASS(CDbReverseIter) : public CDbRecordIter
23 CDbReverseIter(CDbRecordIter* anIterator);
27 TDbRecordId CurrentL();
28 TBool GotoL(TDbPosition aPosition);
29 TBool GotoL(TDbRecordId aRecordId,RDbTableRow& aBuffer);
30 TBool SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison);
31 TDeleted DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow);
36 CDbReverseIter::CDbReverseIter(CDbRecordIter* aIterator)
37 : CDbRecordIter(aIterator->Host()),iIter(aIterator)
40 CDbReverseIter::~CDbReverseIter()
45 TInt CDbReverseIter::Count() const
47 return iIter->Count();
50 TDbRecordId CDbReverseIter::CurrentL()
52 return iIter->CurrentL();
55 TBool CDbReverseIter::GotoL(TDbPosition aPosition)
57 return iIter->GotoL(TDbPosition(EDbLast-aPosition));
60 TBool CDbReverseIter::GotoL(TDbRecordId aRecord,RDbTableRow& aRow)
62 return iIter->GotoL(aRecord,aRow);
65 TBool CDbReverseIter::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison)
67 // Reverse the comparison
70 return iIter->SeekL(aKey,RDbTable::TComparison(RDbTable::EGreaterThan-aComparison));
73 CDbReverseIter::TDeleted CDbReverseIter::DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow)
75 // Need to invert the position.
78 return iIter->DoDeletedL(TDbPosition(EDbLast-aPosition),aRecordId,aRow);
81 // Class CDbTableSource
83 CDbTableSource::CDbTableSource()
86 CDbTableSource::~CDbTableSource()
92 void CDbTableSource::Construct(CDbTable* aTable)
94 // We own the table pass in
97 iIterMark.Construct(aTable->Generation());
101 void CDbTableSource::ReverseIteratorL()
103 // Plug in a reverse iterator adaptor for the iterator
106 iIter=new(ELeave) CDbReverseIter(iIter);
109 TInt CDbTableSource::ColumnCount() const
111 return iRow.Table().Def().Columns().Count();
114 const TDbColumnDef& CDbTableSource::ColumnDef(TDbColNo aCol) const
116 return iRow.Table().Def().Columns()[aCol];
119 RDbRow* CDbTableSource::RowBuffer()
124 TDbColumn CDbTableSource::Column(TDbColNo aColNo)
126 return TDbColumn(iRow,aColNo);
129 void CDbTableSource::Reset()
134 TBool CDbTableSource::EvaluateL(TInt&,TDbRecordId&,TBool&)
139 TBool CDbTableSource::Unevaluated()
144 TInt CDbTableSource::CountL()
151 return iIter->Count();
154 void CDbTableSource::OpenIteratorL()
156 // Re-open the iterator after a reset
160 if (iIter->Host().OpenL())
161 __LEAVE(KErrCorrupt);
165 CDbTableSource::TGoto CDbTableSource::GotoL(TInt& aWork,TDbPosition aPosition,TDbRecordId& aRecordId)
167 // Move to the position specified. If out of synch, try to resynch the iterator
175 aWork=work-EWorkToIterate;
176 CDbRecordIter& iter=*iIter;
177 if (iIterMark.Changed())
178 { // two possible resaons for this
181 if (aPosition==EDbNext || aPosition==EDbPrevious)
183 if (aRecordId==KDbNullRecordId)
184 return ESynchFailure;
185 if (!iter.GotoL(aRecordId,iRow))
186 { // record has been deleted
187 switch(iter.DeletedL(aPosition,aRecordId))
189 case CDbRecordIter::EAtRow: // found it
190 aRecordId=iter.CurrentL();
192 case CDbRecordIter::ENoRow: // no more rows
193 aRecordId=KDbNullRecordId;
195 case CDbRecordIter::ENotSupported: // couldn't re-snych
196 return ESynchFailure;
202 if (iter.GotoL(aPosition))
204 aRecordId=iter.CurrentL();
209 aRecordId=KDbNullRecordId;
214 TBool CDbTableSource::SynchL(TDbRecordId aRecordId)
218 TBool found=iIter->GotoL(aRecordId,iRow);
223 TBool CDbTableSource::GotoL(TDbRecordId aRecordId)
225 return iRow.Table().ExistsL(aRecordId) ? SynchL(aRecordId) : EFalse;
228 void CDbTableSource::ReadRowL(TDbRecordId aRecordId)
230 iRow.ReadL(aRecordId);
233 void CDbTableSource::NewRowL(TDbRecordId aCopyRecord)
235 if (iIter!=NULL&&iReset)
237 iRow.NewL(aCopyRecord);
240 void CDbTableSource::PrepareToWriteRowL(TWrite aWrite)
242 __ASSERT(aWrite==EReplace||aWrite==EAppend);
243 if (aWrite==EReplace)
244 iRow.PrepareReplaceL();
246 iRow.PrepareAppendL();
249 TDbRecordId CDbTableSource::WriteRowL(TWrite aWrite,TSynch aSynch)
251 __ASSERT(aWrite==EReplace||aWrite==EAppend);
252 TDbRecordId id=aWrite==EReplace ? iRow.ReplaceL() : iRow.AppendL();
255 __DEBUG(TBool dbgchk=) SynchL(id);
261 CDbTableSource::TDelete CDbTableSource::DeleteRowL(TDbRecordId& aRecordId,TSynch aSynch)
263 // Remove the row from the table. Synchronise the iterator if required
264 // When not synchronising, the return value is not significant
267 // if the table has changed, we may need to re-synch the iterator before deleting
268 // this is important if neighbouring rows have been deleted by another view
269 if (aSynch==ESynch && iIterMark.Changed() && !SynchL(aRecordId))
270 __LEAVE(KErrNotFound); // it has been deleted already!
271 iRow.DeleteL(aRecordId);
274 CDbRecordIter::TDeleted del=iIter->DeletedL(EDbNext,aRecordId,iRow);
275 __ASSERT(del!=CDbRecordIter::ENotSupported); // must always be able to synch with row data
277 if (del==CDbRecordIter::EAtRow)
278 { // pass back the next record id
279 aRecordId=iIter->CurrentL();
280 return EDeletedAtNext;
283 return EDeletedAtEnd;
286 void CDbTableSource::SetIndexL(const TDesC* anIndex)
288 CDbRecordIter* iter=anIndex ? iRow.Table().IteratorL(*anIndex) : iRow.Table().IteratorL();
293 TBool CDbTableSource::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison,TDbRecordId& aRecordId)
297 CDbRecordIter& iter=*iIter;
298 TBool atrow=iter.SeekL(aKey,aComparison);
299 aRecordId=atrow ? iter.CurrentL() : KDbNullRecordId;
304 CSqlSearchCondition* CDbTableSource::ParseConstraintLC(const TDesC& aCondition)
306 CSqlSearchCondition* sc=Sql::ParseSearchConditionLC(aCondition);