--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/persistentstorage/dbms/pcdbms/utable/UT_TBSRC.CPP Fri Jan 22 11:06:30 2010 +0200
@@ -0,0 +1,309 @@
+// Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
+// All rights reserved.
+// This component and the accompanying materials are made available
+// under the terms of "Eclipse Public License v1.0"
+// which accompanies this distribution, and is available
+// at the URL "http://www.eclipse.org/legal/epl-v10.html".
+//
+// Initial Contributors:
+// Nokia Corporation - initial contribution.
+//
+// Contributors:
+//
+// Description:
+//
+
+#include "UT_STD.H"
+
+// Class CDbReverseIterator
+
+NONSHARABLE_CLASS(CDbReverseIter) : public CDbRecordIter
+ {
+public:
+ CDbReverseIter(CDbRecordIter* anIterator);
+ ~CDbReverseIter();
+private:
+ TInt Count() const;
+ TDbRecordId CurrentL();
+ TBool GotoL(TDbPosition aPosition);
+ TBool GotoL(TDbRecordId aRecordId,RDbTableRow& aBuffer);
+ TBool SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison);
+ TDeleted DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow);
+private:
+ CDbRecordIter* iIter;
+ };
+
+CDbReverseIter::CDbReverseIter(CDbRecordIter* aIterator)
+ : CDbRecordIter(aIterator->Host()),iIter(aIterator)
+ {}
+
+CDbReverseIter::~CDbReverseIter()
+ {
+ delete iIter;
+ }
+
+TInt CDbReverseIter::Count() const
+ {
+ return iIter->Count();
+ }
+
+TDbRecordId CDbReverseIter::CurrentL()
+ {
+ return iIter->CurrentL();
+ }
+
+TBool CDbReverseIter::GotoL(TDbPosition aPosition)
+ {
+ return iIter->GotoL(TDbPosition(EDbLast-aPosition));
+ }
+
+TBool CDbReverseIter::GotoL(TDbRecordId aRecord,RDbTableRow& aRow)
+ {
+ return iIter->GotoL(aRecord,aRow);
+ }
+
+TBool CDbReverseIter::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison)
+//
+// Reverse the comparison
+//
+ {
+ return iIter->SeekL(aKey,RDbTable::TComparison(RDbTable::EGreaterThan-aComparison));
+ }
+
+CDbReverseIter::TDeleted CDbReverseIter::DoDeletedL(TDbPosition aPosition,TDbRecordId aRecordId,const RDbTableRow* aRow)
+//
+// Need to invert the position.
+//
+ {
+ return iIter->DoDeletedL(TDbPosition(EDbLast-aPosition),aRecordId,aRow);
+ }
+
+// Class CDbTableSource
+
+CDbTableSource::CDbTableSource()
+ {}
+
+CDbTableSource::~CDbTableSource()
+ {
+ delete iIter;
+ iRow.Close();
+ }
+
+void CDbTableSource::Construct(CDbTable* aTable)
+//
+// We own the table pass in
+//
+ {
+ iIterMark.Construct(aTable->Generation());
+ iRow.Open(aTable);
+ }
+
+void CDbTableSource::ReverseIteratorL()
+//
+// Plug in a reverse iterator adaptor for the iterator
+//
+ {
+ iIter=new(ELeave) CDbReverseIter(iIter);
+ }
+
+TInt CDbTableSource::ColumnCount() const
+ {
+ return iRow.Table().Def().Columns().Count();
+ }
+
+const TDbColumnDef& CDbTableSource::ColumnDef(TDbColNo aCol) const
+ {
+ return iRow.Table().Def().Columns()[aCol];
+ }
+
+RDbRow* CDbTableSource::RowBuffer()
+ {
+ return &iRow;
+ }
+
+TDbColumn CDbTableSource::Column(TDbColNo aColNo)
+ {
+ return TDbColumn(iRow,aColNo);
+ }
+
+void CDbTableSource::Reset()
+ {
+ iReset=ETrue;
+ }
+
+TBool CDbTableSource::EvaluateL(TInt&,TDbRecordId&,TBool&)
+ {
+ return EFalse;
+ }
+
+TBool CDbTableSource::Unevaluated()
+ {
+ return EFalse;
+ }
+
+TInt CDbTableSource::CountL()
+//
+// Count the records
+//
+ {
+ if (iReset)
+ OpenIteratorL();
+ return iIter->Count();
+ }
+
+void CDbTableSource::OpenIteratorL()
+//
+// Re-open the iterator after a reset
+//
+ {
+ __ASSERT(iIter);
+ if (iIter->Host().OpenL())
+ __LEAVE(KErrCorrupt);
+ iReset=EFalse;
+ }
+
+CDbTableSource::TGoto CDbTableSource::GotoL(TInt& aWork,TDbPosition aPosition,TDbRecordId& aRecordId)
+//
+// Move to the position specified. If out of synch, try to resynch the iterator
+// using aRecordId
+//
+ {
+ __ASSERT(iIter);
+ TInt work=aWork;
+ if (work<0)
+ return EExhausted;
+ aWork=work-EWorkToIterate;
+ CDbRecordIter& iter=*iIter;
+ if (iIterMark.Changed())
+ { // two possible resaons for this
+ if (iReset)
+ OpenIteratorL();
+ if (aPosition==EDbNext || aPosition==EDbPrevious)
+ { // table modified
+ if (aRecordId==KDbNullRecordId)
+ return ESynchFailure;
+ if (!iter.GotoL(aRecordId,iRow))
+ { // record has been deleted
+ switch(iter.DeletedL(aPosition,aRecordId))
+ {
+ case CDbRecordIter::EAtRow: // found it
+ aRecordId=iter.CurrentL();
+ return ESuccess;
+ case CDbRecordIter::ENoRow: // no more rows
+ aRecordId=KDbNullRecordId;
+ return ENoRow;
+ case CDbRecordIter::ENotSupported: // couldn't re-snych
+ return ESynchFailure;
+ }
+ }
+ }
+ iIterMark.Mark();
+ }
+ if (iter.GotoL(aPosition))
+ {
+ aRecordId=iter.CurrentL();
+ return ESuccess;
+ }
+ else
+ {
+ aRecordId=KDbNullRecordId;
+ return ENoRow;
+ }
+ }
+
+TBool CDbTableSource::SynchL(TDbRecordId aRecordId)
+ {
+ if (iReset)
+ OpenIteratorL();
+ TBool found=iIter->GotoL(aRecordId,iRow);
+ iIterMark.Mark();
+ return found;
+ }
+
+TBool CDbTableSource::GotoL(TDbRecordId aRecordId)
+ {
+ return iRow.Table().ExistsL(aRecordId) ? SynchL(aRecordId) : EFalse;
+ }
+
+void CDbTableSource::ReadRowL(TDbRecordId aRecordId)
+ {
+ iRow.ReadL(aRecordId);
+ }
+
+void CDbTableSource::NewRowL(TDbRecordId aCopyRecord)
+ {
+ if (iIter!=NULL&&iReset)
+ OpenIteratorL();
+ iRow.NewL(aCopyRecord);
+ }
+
+void CDbTableSource::PrepareToWriteRowL(TWrite aWrite)
+ {
+ __ASSERT(aWrite==EReplace||aWrite==EAppend);
+ if (aWrite==EReplace)
+ iRow.PrepareReplaceL();
+ else
+ iRow.PrepareAppendL();
+ }
+
+TDbRecordId CDbTableSource::WriteRowL(TWrite aWrite,TSynch aSynch)
+ {
+ __ASSERT(aWrite==EReplace||aWrite==EAppend);
+ TDbRecordId id=aWrite==EReplace ? iRow.ReplaceL() : iRow.AppendL();
+ if (aSynch==ESynch)
+ {
+ __DEBUG(TBool dbgchk=) SynchL(id);
+ __ASSERT(dbgchk);
+ }
+ return id;
+ }
+
+CDbTableSource::TDelete CDbTableSource::DeleteRowL(TDbRecordId& aRecordId,TSynch aSynch)
+//
+// Remove the row from the table. Synchronise the iterator if required
+// When not synchronising, the return value is not significant
+//
+ {
+ // if the table has changed, we may need to re-synch the iterator before deleting
+ // this is important if neighbouring rows have been deleted by another view
+ if (aSynch==ESynch && iIterMark.Changed() && !SynchL(aRecordId))
+ __LEAVE(KErrNotFound); // it has been deleted already!
+ iRow.DeleteL(aRecordId);
+ if (aSynch==ESynch)
+ {
+ CDbRecordIter::TDeleted del=iIter->DeletedL(EDbNext,aRecordId,iRow);
+ __ASSERT(del!=CDbRecordIter::ENotSupported); // must always be able to synch with row data
+ iIterMark.Mark();
+ if (del==CDbRecordIter::EAtRow)
+ { // pass back the next record id
+ aRecordId=iIter->CurrentL();
+ return EDeletedAtNext;
+ }
+ }
+ return EDeletedAtEnd;
+ }
+
+void CDbTableSource::SetIndexL(const TDesC* anIndex)
+ {
+ CDbRecordIter* iter=anIndex ? iRow.Table().IteratorL(*anIndex) : iRow.Table().IteratorL();
+ delete iIter;
+ iIter=iter;
+ }
+
+TBool CDbTableSource::SeekL(const TDbLookupKey& aKey,RDbTable::TComparison aComparison,TDbRecordId& aRecordId)
+ {
+ if (iReset)
+ OpenIteratorL();
+ CDbRecordIter& iter=*iIter;
+ TBool atrow=iter.SeekL(aKey,aComparison);
+ aRecordId=atrow ? iter.CurrentL() : KDbNullRecordId;
+ iIterMark.Mark();
+ return atrow;
+ }
+
+CSqlSearchCondition* CDbTableSource::ParseConstraintLC(const TDesC& aCondition)
+ {
+ CSqlSearchCondition* sc=Sql::ParseSearchConditionLC(aCondition);
+ sc->BindL(iRow);
+ return sc;
+ }