phonebookui/Phonebook2/xSPExtensionManager/src/CxSPContactManager.cpp
changeset 0 e686773b3f54
child 11 2828b4d142c0
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/phonebookui/Phonebook2/xSPExtensionManager/src/CxSPContactManager.cpp	Tue Feb 02 10:12:17 2010 +0200
@@ -0,0 +1,354 @@
+/*
+* Copyright (c) 2006 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: 
+*       Contact manager.
+*
+*/
+
+
+// INCLUDE FILES
+#include <s32file.h>
+#include <s32mem.h>
+#include <CPbk2ViewState.h>
+#include <Pbk2ViewId.hrh>
+#include "CxSPContactManager.h"
+#include "MxSPFactory.h"
+#include <MVPbkContactLink.h>
+#include <MVPbkStreamable.h>
+#include <CVPbkContactManager.h>
+#include <CVPbkContactLinkArray.h>
+
+// CONSTANTS
+namespace
+	{
+	_LIT( KExtContactMapFile, "xSPcontactMap.dat" );
+	const TInt KExpandSize = 4; // needed data buffer size
+	}
+
+// ==================== MEMBER FUNCTIONS ====================
+
+CxSPPbkContactMap::CxSPPbkContactMap()
+	{	
+	}
+        
+CxSPPbkContactMap::~CxSPPbkContactMap()
+	{
+	delete iPbkContactLink;
+	}
+
+void CxSPPbkContactMap::ExternalizeL( RWriteStream& aStream ) const
+	{
+	aStream.WriteUint32L( iId );
+	aStream.WriteInt32L( ixSPContactId );
+	HBufC8* buffer = CxSPContactManager::LinkToBufferL( iPbkContactLink );
+	CleanupStack::PushL( buffer );
+	aStream.WriteInt32L( buffer->Length() );
+	aStream.WriteL( *buffer, buffer->Length() );
+	CleanupStack::PopAndDestroy(); // buffer		
+	}
+
+void CxSPPbkContactMap::InternalizeL( RReadStream& aStream,
+										const MVPbkContactStoreList& aContactStores )
+	{
+	iId = aStream.ReadUint32L();
+	ixSPContactId = aStream.ReadInt32L();
+	TInt length = aStream.ReadInt32L();
+	HBufC8* bufLink = HBufC8::NewLC( length );
+	TPtr8 ptr = bufLink->Des();
+	aStream.ReadL( ptr, length );		
+	iPbkContactLink = CxSPContactManager::BufferToLinkL( *bufLink, aContactStores );	
+	CleanupStack::PopAndDestroy(); // bufLink	
+	}	
+
+CxSPContactManager::CxSPContactManager()
+    {
+    }
+    
+void CxSPContactManager::ConstructL( CArrayPtrFlat<MxSPFactory>& aFactories,
+										CVPbkContactManager& aVPbkContactManager )
+    {
+    MVPbkContactStoreList& contactStoreList = aVPbkContactManager.ContactStoresL();    
+    RestoreArrayL( KExtContactMapFile, aFactories, contactStoreList );	
+	// Register this instance as the contact manager of all xSP factories
+	for( TInt i = 0; i < aFactories.Count(); i++ )
+		{
+		aFactories[i]->RegisterContactManager( *this );
+		}
+    } 
+         
+CxSPContactManager::~CxSPContactManager()
+    {
+    iContactMap.ResetAndDestroy(); 
+    iContactMap.Close();
+    delete iState;
+    }
+
+CxSPContactManager* CxSPContactManager::NewL( CArrayPtrFlat<MxSPFactory>& aFactories,
+												CVPbkContactManager& aVPbkContactManager )
+    {
+    CxSPContactManager* self = new (ELeave) CxSPContactManager;
+    CleanupStack::PushL(self);
+    self->ConstructL( aFactories, aVPbkContactManager );
+    CleanupStack::Pop(self);
+    return self;
+    }
+    
+void CxSPContactManager::MapContactL( TUint32 aId, const MVPbkContactLink& aPbkContactLink,
+        								TInt32 axSPContactId )
+	{
+	TInt index = FindMap( aId, aPbkContactLink, axSPContactId );
+	if( index == KErrNotFound )
+		{
+		CxSPPbkContactMap* map = new (ELeave) CxSPPbkContactMap;
+		CleanupStack::PushL( map );		
+		map->iId = aId;
+		map->ixSPContactId = axSPContactId;
+		map->iPbkContactLink = aPbkContactLink.CloneLC();
+		CleanupStack::Pop(); // map->iPbkContactLink
+		User::LeaveIfError( iContactMap.Append( map ) );		
+		CleanupStack::Pop(); // map		
+		SaveArrayL( KExtContactMapFile );
+		}
+	}
+	
+void CxSPContactManager::UnmapContactL( TUint32 aId, const MVPbkContactLink& aPbkContactLink,
+        								TInt32 axSPContactId )
+	{
+	TInt index = FindMap( aId, aPbkContactLink, axSPContactId );
+	if( index != KErrNotFound )
+		{
+		CxSPPbkContactMap* map = iContactMap[index];
+		delete map;
+		iContactMap.Remove( index );		
+		SaveArrayL( KExtContactMapFile );
+		}
+	}
+		
+void CxSPContactManager::GetMappedPbkContactsL( TUint32 aId, TInt32 axSPContactId,
+        							RPointerArray<MVPbkContactLink>& aPbkContactLinks ) const
+	{
+	TInt count = iContactMap.Count();
+	for( TInt i = 0; i < count; i++ )
+		{
+		const CxSPPbkContactMap* map = iContactMap[i];
+		if( map->iId == aId && map->ixSPContactId == axSPContactId && map->iPbkContactLink )
+			{
+			User::LeaveIfError( aPbkContactLinks.Append( map->iPbkContactLink ) );
+			}
+		}
+	}		
+
+void CxSPContactManager::GetMappedxSPContactsL( TUint32 aId, 
+												const MVPbkContactLink& aPbkContactLink,
+        										RArray<TInt32>& axSPContactIds ) const
+	{
+	const TInt count = iContactMap.Count();
+	for( TInt i = 0; i < count; i++ )
+		{
+		const CxSPPbkContactMap* map = iContactMap[i];
+		if( map->iId == aId && map->iPbkContactLink )
+			{
+			if( map->iPbkContactLink->IsSame( aPbkContactLink ) )
+				{
+				User::LeaveIfError( axSPContactIds.Append( map->ixSPContactId ) );
+				}
+			}			
+		}
+	}        										
+	
+void CxSPContactManager::ViewActivatedL( const TVwsViewId& aPrevViewId,
+                         				TUid aCustomMessageId,
+                         				const TDesC8& aCustomMessage )
+	{
+	if( aCustomMessageId == CPbk2ViewState::Uid() && 
+    						aPrevViewId.iViewUid.iUid == EPbk2NamesListViewId )
+		{
+		delete iState;
+		iState = NULL;
+		iState = CPbk2ViewState::NewL( aCustomMessage );
+		}								
+	}
+			
+const CPbk2ViewState* CxSPContactManager::NamesListState() const
+	{
+	return iState;
+	}		  	   
+ 
+TInt CxSPContactManager::CreatePrivateFolder( RFs& aSession, 
+											  TDes& aPath, 
+											  TDriveNumber aDrive ) const
+	{
+	_LIT( KColon, ":" );
+	TChar drive;
+	TInt err = aSession.DriveToChar( aDrive, drive );							
+	if( !err )
+		{
+		err = aSession.CreatePrivatePath( aDrive );
+		if( !err )
+			{
+			TFileName path;			
+			aSession.PrivatePath( path );
+			aPath.Zero();
+			aPath.Append( drive );
+			aPath.Append( KColon );
+			aPath.Append( path );														
+			}		
+		}	
+	return err;
+	} 
+ 
+TInt CxSPContactManager::FindMap( TUint32 aId, 
+									const MVPbkContactLink& aPbkContactLink, 
+									TInt32 axSPContactId ) const
+	{
+	TInt index( KErrNotFound );
+	TInt count = iContactMap.Count();
+	for( TInt i = 0; i < count && index == KErrNotFound; i++ )
+		{
+		const CxSPPbkContactMap* map = iContactMap[i];
+		if( map->iId == aId && map->ixSPContactId == axSPContactId && map->iPbkContactLink )
+			{
+			if( map->iPbkContactLink->IsSame( aPbkContactLink ) )
+				{
+				index = i;
+				}
+			}			
+		}
+	return index;
+	}
+	
+void CxSPContactManager::SaveArrayL( const TDesC& aFileName )
+	{
+	RFs fs; // CSI: 76 #
+	User::LeaveIfError( fs.Connect() );
+	CleanupClosePushL( fs );
+	TFileName orderFile;
+	User::LeaveIfError( CreatePrivateFolder( fs, orderFile, EDriveC ) );
+	orderFile.Append( aFileName );
+	RFileWriteStream writeStream;
+	TInt err = writeStream.Replace( fs, orderFile, EFileWrite );
+	if( !err )
+		{
+		CleanupClosePushL( writeStream );		
+		TUint32 count = iContactMap.Count();
+		writeStream.WriteUint32L( count );
+		for( TUint32 i = 0; i < count; i++ )
+			{
+			writeStream << *iContactMap[i];
+			}			
+		CleanupStack::PopAndDestroy(); // writeStream
+		}
+	CleanupStack::PopAndDestroy(); // fs
+	}
+	
+void CxSPContactManager::RestoreArrayL( const TDesC& aFileName,
+										CArrayPtrFlat<MxSPFactory>& aFactories,
+										MVPbkContactStoreList& aContactStoreList )
+	{
+	// read the array from file
+    RFs fs; // CSI: 76 #
+	User::LeaveIfError( fs.Connect() );
+	CleanupClosePushL( fs );	
+	TFileName orderFile;
+	User::LeaveIfError( CreatePrivateFolder( fs, orderFile, EDriveC ) );
+	orderFile.Append( aFileName );
+	RFileReadStream readStream;
+	TInt err = readStream.Open( fs, orderFile, EFileRead );
+	if( !err )
+		{
+		CleanupClosePushL( readStream );
+		TUint32 count = readStream.ReadUint32L();
+		for( TUint32 i = 0; i < count; i++ )
+			{
+			CxSPPbkContactMap* map = new (ELeave) CxSPPbkContactMap;
+			CleanupStack::PushL( map );
+			map->InternalizeL( readStream, aContactStoreList );
+			
+			// only use mappings from the present xSP extensions
+			// removed xSP extension mappings are not used
+			TBool found( EFalse );
+			for( TInt j = 0; j < aFactories.Count() && !found; j++ )
+				{
+				if( map->iId == aFactories[j]->Id() )
+					{
+					found = ETrue;
+					}
+				}			
+			if( found )
+				{
+				User::LeaveIfError( iContactMap.Append( map ) );
+				CleanupStack::Pop(); // map	
+				}
+			else
+				{
+				CleanupStack::PopAndDestroy(); // map
+				}			
+			}
+		CleanupStack::PopAndDestroy(); // readStream
+		}		
+	CleanupStack::PopAndDestroy(); // fs
+	}	
+	
+HBufC8* CxSPContactManager::LinkToBufferL( const MVPbkContactLink* aLink )
+	{
+	HBufC8* buffer = NULL;	
+	if( aLink )
+		{
+		const MVPbkStreamable* streamable = aLink->Streamable();
+		if( streamable )
+			{
+			CBufFlat* buf = CBufFlat::NewL( KExpandSize );
+			CleanupStack::PushL( buf );
+			RBufWriteStream writeStream( *buf );
+			CleanupClosePushL( writeStream );
+			streamable->ExternalizeL( writeStream );
+			writeStream.CommitL();
+			CleanupStack::PopAndDestroy(); // writeStream				
+			TPtr8 ptr = buf->Ptr( 0 );
+			buffer = ptr.AllocL();	// CSI: 35 #
+			CleanupStack::PopAndDestroy(); // buf			
+			}
+		}
+	if( !buffer )
+		{
+		buffer = KNullDesC8().AllocL();
+		}
+	return buffer;
+	}
+	
+MVPbkContactLink* CxSPContactManager::BufferToLinkL( const TDesC8& aBuffer,
+									const MVPbkContactStoreList& aContactStores )
+	{
+	MVPbkContactLink* contactLink = NULL;
+	if( aBuffer.Length() )
+		{
+		CBufFlat* buf = CBufFlat::NewL( KExpandSize );
+		CleanupStack::PushL( buf );
+		buf->InsertL( 0, aBuffer );
+		RBufReadStream readStream( *buf );
+		CleanupClosePushL( readStream );	
+		CVPbkContactLinkArray* array = CVPbkContactLinkArray::NewLC( readStream, aContactStores );
+		TInt count = array->Count();												
+		if( count == 1 ) // should contain only single contact link
+			{
+			const MVPbkContactLink& link = array->At( 0 );
+			contactLink = link.CloneLC();
+			CleanupStack::Pop(); // contactLink
+			}						
+		CleanupStack::PopAndDestroy( 3 ); // array, readStream, buf	
+		}
+	return contactLink;
+	}			
+        								
+// End of File