contentmgmt/referencedrmagent/contentiterator/embeddedcontentiteratorbase.cpp
author Santosh V Patil <santosh.v.patil@nokia.com>
Tue, 08 Jun 2010 10:31:52 +0530
branchRCL_3
changeset 68 564220cc963b
parent 15 da2ae96f639b
child 102 deec7e509f66
permissions -rw-r--r--
Transplanting changeset f3b0b5725c58 (Fix for bug 1301)

/*
* Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of the License "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 <f32file.h>
#include <apmstd.h>

#include "contentIterator.h"
#include "virtualpath.h"
#include "content.h"
#include "EmbeddedcontentIteratorBase.h"
#include "embeddedobject.h"

using namespace ContentAccess;


CEmbeddedContentIteratorBase* CEmbeddedContentIteratorBase ::NewL(CContent& aContent, TBool aRecursive, const TDesC8& aMimeType)
	{
	CEmbeddedContentIteratorBase* self = new (ELeave) CEmbeddedContentIteratorBase(aContent, aRecursive, aMimeType);
	CleanupStack::PushL(self);
	self->ConstructL();
	CleanupStack::Pop(self);
	return self;
	}

void CEmbeddedContentIteratorBase::ConstructL()
	{
	// populate the array with the items at this level of the content file
	iContent.GetEmbeddedObjectsL(iEmbeddedContentObjects, EContentObject);
	iContent.GetEmbeddedObjectsL(iEmbeddedContainerObjects, EContainerObject);

	// Find the Next,  (ie. first content object)
	// This will leave if there is no object
	User::LeaveIfError(Next());
	}

CEmbeddedContentIteratorBase::CEmbeddedContentIteratorBase(CContent& aContent, TBool aRecursive, const TDesC8& aMimeType) 
	: iContent(aContent), iRecursive(aRecursive), iMimeType(aMimeType)
	{
	iContentIndex = 0;
	iContainerIndex = 0;
	}

CEmbeddedContentIteratorBase::~CEmbeddedContentIteratorBase()
	{
	delete iSubIterator;

	// Clear array 
	iEmbeddedContainerObjects.Close();
	iEmbeddedContentObjects.Close();
	}	

CEmbeddedObject& CEmbeddedContentIteratorBase::EmbeddedObject()
	{
	// The next pointer is currently on an object inside the subiterator
	if(iSubIterator)
		{
		return iSubIterator->EmbeddedObject();
		}
	else
		{	
		return *iEmbeddedContentObjects[iContentIndex - 1];
		}
	}
		
TInt CEmbeddedContentIteratorBase::Next()
	{
	// Must figure out if there's a constant for this
	TBuf8 <KMaxDataTypeLength> mimeType;
	TInt ret = KErrNotFound;

	// If we are already looking into a container, try finding the next in there
	if(iSubIterator)
		{
		ret = iSubIterator->Next();
		if(ret == KErrNone)
			{
			return ret;
			}
		else 
			{
			// come back to our level
			iContent.CloseContainer();

			// Delete sub-iterator
			delete iSubIterator;
			iSubIterator = NULL;
			}
		}
	

	// Find content objects inside this container
	for( ; iContentIndex < iEmbeddedContentObjects.Count(); iContentIndex++)
		{
		// if we don't care about mime type, this content object will do
		if(iMimeType.Length() == 0)
			{
			// make sure next time we look at the next item in our array
			iContentIndex++;
			return KErrNone;
			}

		// See if the content object has the right mime type that we are looking for
		mimeType.Copy(iEmbeddedContentObjects[iContentIndex]->MimeType());
		if(iMimeType == mimeType)
			{
			// make sure next time we look at the next item in our array
			iContentIndex++;
			return KErrNone;
			}
		// otherwise continue to next iteration, for loop incrementes iContentIndex		
		}		
	
	// Free memory allocated for content objects
	if(iContentIndex)
		{
		iEmbeddedContentObjects.ResetAndDestroy();
		iContentIndex = 0;
		}

	// Find content objects within nested containers
	for( ; iContainerIndex < iEmbeddedContainerObjects.Count(); iContainerIndex++)
		{
		// If it's a container look inside 
		iContent.OpenContainer(iEmbeddedContainerObjects[iContainerIndex]->UniqueId());
		TRAPD(err, iSubIterator = CEmbeddedContentIteratorBase::NewL(iContent, iRecursive, iMimeType));
		if(err == KErrNone)
			{
			// must have found something inside
			// make sure next time we search at the next item in our array
			iContainerIndex++;
			return KErrNone;
			}
		// otherwise continue to next iteration
		}

	// must be at the end of the array, ie. can't find any more content objects
	return KErrNotFound;
	}