os/textandloc/textrendering/texthandling/stext/TXTIXSTR.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
     3 * All rights reserved.
     4 * This component and the accompanying materials are made available
     5 * under the terms of "Eclipse Public License v1.0"
     6 * which accompanies this distribution, and is available
     7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
     8 *
     9 * Initial Contributors:
    10 * Nokia Corporation - initial contribution.
    11 *
    12 * Contributors:
    13 *
    14 * Description: 
    15 *
    16 */
    17 
    18 
    19 #include <e32std.h>
    20 #include <e32base.h>
    21 
    22 #include <gdi.h>
    23 
    24 #include "TXTFMLYR.H"
    25 #include "TXTMRTSR.H"
    26 #include "TXTRICH.H"
    27 #include "TXTINDEX.H"
    28 #include "TXTSTD.H"
    29 
    30 #include "OstTraceDefinitions.h"
    31 #ifdef OST_TRACE_COMPILER_IN_USE
    32 #include "TXTIXSTRTraces.h"
    33 #endif
    34 
    35 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
    36 #include "TXTFMLYR_INTERNAL.H"
    37 #endif
    38 
    39 const TUint8 KRegisterItemPresent=1;
    40 
    41 
    42 GLDEF_C void DiscardPhraseOnCleanup(TAny* aPhrase)
    43 // Allows RPhraseAttribsEntry on the CleanupStack to be destroyed correctly,
    44 // when included as a PushL(TCleanupItem).
    45 //
    46 	{
    47 	((RPhraseAttribsEntry*)aPhrase)->Discard();
    48 	User::Free(aPhrase);
    49 	}
    50 
    51 
    52 TRtPasteContext::TRtPasteContext(const CStreamStore* aStore,const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,const CStyleList* aStyleList)
    53 //
    54 	:iStore(aStore),
    55 	iGlobalParaLayer(aGlobalParaLayer),iGlobalCharLayer(aGlobalCharLayer),iStyleList(aStyleList),
    56 	iParagraphStylesFlag(EFalse),iIncompleteParaFlag(ETrue),iPastePos(),iParasPasted(0)
    57 	{}
    58 
    59 //////////////////////////////////////////////////////////////////////////////////////
    60 
    61 
    62 CRichTextIndex* CRichTextIndex::NewL(const CStreamStore& aStore,TStreamId aId,
    63 //										MPictureFactory* aPictureFactory,MRichTextStoreResolver* aStoreResolver,
    64 										const CParaFormatLayer* aGlobalParaLayer,
    65 										const CCharFormatLayer* aGlobalCharLayer,
    66 										const CRichText& aText)
    67 //										const CStyleList* aStyleList)
    68 //
    69 	{
    70 	CRichTextIndex* self=new(ELeave) CRichTextIndex(aText);
    71 	CleanupStack::PushL(self);
    72 	self->ConstructL(aGlobalParaLayer,aGlobalCharLayer,KMultiParaGranularity,KLargePhraseGranularity);
    73 	self->RestoreL(aStore,aId,aGlobalParaLayer,aGlobalCharLayer,aText.StyleList());
    74 //	self->SetPictureFactory(aPictureFactory,aStoreResolver);
    75 	CleanupStack::Pop();
    76 	return self;
    77 	}
    78 
    79 
    80 void CRichTextIndex::RestoreL(const CStreamStore& aStore,TStreamId aId,
    81 									   const CParaFormatLayer* aGlobalParaLayer,
    82 									   const CCharFormatLayer* aGlobalCharLayer,
    83 									   const CStyleList* aStyleList)
    84 //
    85 	{
    86 	RStoreReadStream stream;
    87 	stream.OpenLC(aStore,aId);
    88 	//
    89 	InternalizeL(stream,aGlobalParaLayer,aGlobalCharLayer,aStyleList);
    90 	CleanupStack::PopAndDestroy();  // stream
    91 	}
    92 
    93 
    94 TStreamId CRichTextIndex::StoreMarkupL(CStreamStore& aStore,const CStyleList* /*aStyleList*/) const
    95 // Store this rich text markup out-of-line along with its components.
    96 //
    97 	{
    98 	CStoreMap* map=CStoreMap::NewLC(aStore);
    99 	StorePicturesL(aStore,*map);
   100 //
   101 	RStoreWriteStream stream(*map);
   102 	TStreamId id=stream.CreateLC(aStore);
   103 //	CONST_CAST(CRichTextIndex*,this)->iStyleList=CONST_CAST(CStyleList*,aStyleList);
   104 	ExternalizeL(stream);
   105 	stream.CommitL();
   106 //
   107 	map->Reset();
   108 	CleanupStack::PopAndDestroy(2);  // map,stream
   109 	return id;
   110 	}
   111 
   112 
   113 void CRichTextIndex::StorePicturesL(CStreamStore& aStore,CStoreMap& aMap) const
   114 // Store any picture components in the specified store.
   115 // Due to deferred loading, any pictures that are not in memory at this point are loaded,
   116 // and then saved.
   117 //
   118 	{
   119 	__TEST_INVARIANT;
   120 
   121 	StorePicturesL(aStore,aMap,0,iText.DocumentLength()+1);
   122 	}
   123 
   124 
   125 void CRichTextIndex::StorePicturesL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength) const
   126 // Store all picture components that are contained within the specified range (inclusive), in the store provided.
   127 // Prior to storing the picture, its size is written to its picture header which is stored in-line in 
   128 // the markup.
   129 // Any pictures that are not in memory are now deleted, as a previous call to LoadAllPicturesNowL()
   130 // should have been made.
   131 // We are guaranteed that all pictures that are in memory are fully restored & detached from their store
   132 //
   133 	{
   134 	__TEST_INVARIANT;
   135 
   136 	if (iPictureCount==0)
   137 		return;
   138 	CONST_CAST(CRichTextIndex*,this)->DetachFromStoreL(CPicture::EDetachFull,aPos,aLength);	// ensure all pictures have been loaded
   139 	TInt currentPos=aPos;
   140 	DocumentChanged();  // reset the cache logical position
   141 	while (currentPos<aPos+aLength)
   142 		{// Store next picture & increment pos.
   143 		CONST_CAST(CRichTextIndex*,this)->ScanToPosition(currentPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
   144 		TCurrentIndexRecords current;
   145 		GetCurrentRecords(current);
   146 		if (current.iPhrase && current.iPhrase->IsPicturePhrase())
   147 			{
   148 			RPhraseAttribsEntry& phrase=*current.iPhrase;
   149 			TPictureHeader* hdr=phrase.PictureHeaderPtr();
   150 			if (!hdr)
   151 			    {
   152 			    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_STOREPICTURESL, "ERichTextStorePictureIntegrityError" );
   153 			    }
   154 			__ASSERT_DEBUG(hdr,Panic(ERichTextStorePictureIntegrityError));
   155 			TBool pictureInMemory=(hdr->iPicture.IsPtr() && hdr->iPicture.AsPtr());
   156 			if (pictureInMemory)
   157 				{// Store the picture
   158 				CPicture& picture=*hdr->iPicture;
   159 				picture.GetSizeInTwips(hdr->iSize);  //  write picture size to picture header.
   160 				TStreamId id=picture.StoreL(aStore);
   161 				aMap.BindL(hdr->iPicture,id);
   162 				}
   163 			else
   164 				{// Replace the picture phrase with a simple text equivalent.
   165 				CCharFormatLayer* charLayer=phrase.ReleaseCharFormatLayerOwnership();
   166 				phrase.Discard();  // destroy the redundant picture phrase
   167 				RPhraseAttribsEntry newTextPhrase(charLayer,1);
   168 				(*iPhraseIx)[iPos.iPhraseElement]=newTextPhrase;
   169 				CONST_CAST(CRichTextIndex*,this)->iPictureCount--;  // the picture has now been removed
   170 				// I do not bother tidying up the phrase index here,
   171 				// since the benefit does not currently match the cost involved.
   172 				//
   173 				}
   174 			}
   175 		//
   176 		// increment the current position to the *start* of the next phrase/paragraph
   177 		TInt offsetIntoUnit=(current.iPhrase)?iPos.iPhraseElementOffset:iPos.iParaElementOffset;
   178 		currentPos+=CurrentPhraseLength()-offsetIntoUnit;
   179 		}
   180 	}
   181 
   182 
   183 void CRichTextIndex::DetachFromStoreL(CPicture::TDetach aDegree,TInt aPos,TInt aLength)
   184 // Attempts to restore all pictures not already present in memory.
   185 // A deep (door+document) or shallow (door only) restoration may occur, depending
   186 // on the value of detach.
   187 //
   188 	{
   189 	__TEST_INVARIANT;
   190 
   191 	if (iPictureCount==0)
   192 		return;
   193 	DocumentChanged();  // reset the cache logical position
   194 	TInt currentPos=aPos;
   195 	while (currentPos<aPos+aLength)
   196 		{
   197 		CONST_CAST(CRichTextIndex*,this)->ScanToPosition(currentPos,EScanToPositionAbsolute,&iLastUsed);
   198 		TCurrentIndexRecords current;
   199 		GetCurrentRecords(current);
   200 		if (current.iPhrase)
   201 			{
   202 			RPhraseAttribsEntry& phrase=*current.iPhrase;
   203 			if (phrase.IsPicturePhrase())
   204 				{
   205 				TPictureHeader* hdr=phrase.PictureHeaderPtr();
   206 				if (!hdr)
   207 				    {
   208 				    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DETACHFROMSTOREL, "ERichTextStorePictureIntegrityError" );
   209 				    }
   210 				__ASSERT_DEBUG(hdr,Panic(ERichTextStorePictureIntegrityError));
   211 				//
   212 				if (hdr->iPicture.IsId())
   213 					{
   214 					TRAPD(r,
   215 					phrase.PictureHandleL(iText.PictureFactory(),iText.StoreResolver(),iPos.iDocPos,MLayDoc::EForceLoadTrue));  // swizzles
   216 					if (r!=KErrNone && !hdr->iPicture.IsId())
   217 					    {
   218 					    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_DETACHFROMSTOREL, "ERichTextStorePictureIntegrityError" );    
   219 					    }
   220 					__ASSERT_DEBUG(r==KErrNone || hdr->iPicture.IsId(),Panic(ERichTextStorePictureIntegrityError));
   221 					if (r==KErrNotSupported)  // we don't recognise the picture type
   222 						{
   223 						TInt offsetIntoUnit=(current.iPhrase)?iPos.iPhraseElementOffset:iPos.iParaElementOffset;
   224 						currentPos+=CurrentPhraseLength()-offsetIntoUnit;
   225 						continue;
   226 						}
   227 					
   228 					OstTrace1( TRACE_FATAL, DUP3_CRICHTEXTINDEX_DETACHFROMSTOREL, "Leave code=%d", r );
   229 					User::LeaveIfError(r);
   230 					}
   231 				//
   232 				// recurse the call to detach the picture from the store
   233 				if (!hdr->iPicture.IsPtr())
   234 				    {
   235 				    OstTrace0( TRACE_DUMP, DUP2_CRICHTEXTINDEX_DETACHFROMSTOREL, "ERichTextStorePictureIntegrityError" );
   236 				    }
   237 				__ASSERT_DEBUG(hdr->iPicture.IsPtr(),Panic(ERichTextStorePictureIntegrityError));
   238 				hdr->iPicture->DetachFromStoreL(aDegree);
   239 				}
   240 			}
   241 		//
   242 		// increment the current position to the *start* of the next phrase/paragraph
   243 		TInt offsetIntoUnit=(current.iPhrase)?iPos.iPhraseElementOffset:iPos.iParaElementOffset;
   244 		currentPos+=CurrentPhraseLength()-offsetIntoUnit;
   245 		}
   246 	__TEST_INVARIANT;
   247 	}
   248 
   249 //////////////////////////////////////////////////////////////////////////////////////
   250 
   251 void CRichTextIndex::ExternalizeL(RWriteStream& aStream)const
   252 // Persist the current state of this object in the specified stream.
   253 // Saved records include the terminating paragraph delimiter. 
   254 //
   255 //
   256 	{
   257 	__TEST_INVARIANT;
   258 
   259 	CONST_CAST(CRichTextIndex*,this)->CancelInsertCharFormat();  // cancel any insert-pending state.
   260 	TInt charCount=iText.DocumentLength()+1;  // includes the end of document character
   261 	((CRichTextIndex*)this)->ScanToPosition(0,EScanToPositionAbsolute);
   262 	TLogicalPosition startPos=iPos;
   263 	((CRichTextIndex*)this)->ScanToPosition(charCount-1,EScanToPositionMatchLeft);
   264 	TLogicalPosition endPos=iPos;
   265 	CStyleList* styleList=iText.StyleList();
   266 	ExternalizeRtiHeaderL(aStream,endPos,styleList);
   267 	ExternalizeSharedFormatsL(aStream,startPos,endPos,styleList);
   268 	// DEF126934 begin: memory leak
   269 	RPhraseAttribsEntry *pForRelease = ExternalizeParaIxL(aStream,startPos,endPos,styleList);
   270 	if (pForRelease)
   271 		delete pForRelease;
   272 	// DEF126934 end
   273 	ExternalizePhraseIxL(aStream);
   274 	}
   275 
   276 
   277 void CRichTextIndex::InternalizeL(RReadStream& aStream,const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,const CStyleList* aStyleList)
   278 // Load a rich text index.  Has construct semantics only!
   279 // Restores this object from the specified stream.  As specific format layers are restored, they are based
   280 // on the specified global format layers. 
   281 //
   282 	{
   283 	if (iParaIx->Count()!=1 || (*iParaIx)[0].iLength!=1)
   284 	    {
   285 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_INTERNALIZEL, "ERtIndexInternalizeCalledOnNonEmptySource" );
   286 	    }
   287 	__ASSERT_ALWAYS(iParaIx->Count()==1 && (*iParaIx)[0].iLength==1,Panic(ERtIndexInternalizeCalledOnNonEmptySource));
   288 	TRtPasteContext context(NULL,aGlobalParaLayer,aGlobalCharLayer,aStyleList);
   289 	InternalizeRtiHeaderL(aStream,context);
   290 	InternalizeSharedFormatsL(aStream,context);
   291 	InternalizeParaIxL(aStream,context);
   292 	InternalizePhraseIxL(aStream,aGlobalCharLayer);
   293 	if (context.iParagraphStylesFlag)
   294 		GenerateAllPhraseLinksL();
   295 	SetSpecificMarkupInternalized(ETrue);
   296 
   297 	__TEST_INVARIANT;
   298 	}
   299 
   300 //////////////////////////////////////////////////////////////////////////////////////
   301 
   302 
   303 void CRichTextIndex::GenerateAllPhraseLinksL()
   304 // Go through the index, and for each specific character format layer (shared or otherwise), regenerate
   305 // the appropriate based-on link, (normal or style).
   306 // 
   307 	{
   308 	TInt count=ParagraphCount();
   309 	TInt phraseElement=0;
   310 	for (TInt ii=0; ii<count; ii++)
   311 		{
   312 		const CParaAttribs& paraAttribs=*(*iParaIx)[ii].iParaAttribs;
   313 		const CParaFormatLayer* base=STATIC_CAST(const CParaFormatLayer*,paraAttribs.iParaFormat->SenseBase());
   314 		TInt phraseCount=paraAttribs.PhraseCount();
   315 		if (phraseCount==1)
   316 			{
   317 			// If the reference count is zero, iCharFormat cannot be valid, so the CParaAttribs object is corrupt.
   318 			if (paraAttribs.iRefCount == 0)
   319 				User::Leave(KErrCorrupt);
   320 
   321 			GeneratePhraseLink(paraAttribs.iCharFormat,base);  // constant character formatting
   322 			}
   323 		else
   324 			{
   325 			if (phraseCount<=1)
   326 			    {
   327 			    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_GENERATEALLPHRASELINKSL, "EDebug" );
   328 			    }
   329 			 __ASSERT_DEBUG(phraseCount>1,Panic(EDebug));
   330 
   331 			// If the phrase count is too great the CParaAttribs object is corrupt.
   332 			if (phraseElement + phraseCount > iPhraseIx->Count())
   333 				User::Leave(KErrCorrupt);
   334 
   335 			for (TInt nn=0; nn<phraseCount; nn++)
   336 				GeneratePhraseLink((*iPhraseIx)[phraseElement+nn].CharFormat(),base);
   337 			phraseElement+=phraseCount;
   338 			}
   339 		}
   340 	}
   341 
   342 
   343 void CRichTextIndex::GeneratePhraseLink(CCharFormatLayer* aPhraseCharFormatLayer,const CParaFormatLayer* aBase)
   344 // Set the based on link for this character format layer appropriately.
   345 // By default, all specific character format layers are based on the global default.
   346 //
   347 	{
   348 	TUid type=aBase->Type();
   349 	if (type!=KNormalParagraphStyleUid)
   350 		aPhraseCharFormatLayer->SetBase(((CParagraphStyle*)aBase)->CharFormatLayer());
   351 	}
   352 
   353 
   354 //////////////////////////////////////////////////////////////////////////////////////
   355 
   356 void CRichTextIndex::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength, TBool aCopyStyles)const
   357 // Copies the markup corresponding to the specified rich text region to the specified write stream.
   358 // aPos/aLength has already been validated in RichText
   359 // Saved records can NOT include the terminating paragraph delimiter.
   360 //
   361 	{
   362 	__TEST_INVARIANT;
   363 
   364 	((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionAbsolute);
   365 	TLogicalPosition startPos=iPos;
   366 	((CRichTextIndex*)this)->ScanToPosition((aPos+aLength)-1,EScanToPositionAbsolute);
   367 	TLogicalPosition endPos=iPos;
   368 
   369 	CStyleList* docStyleList = NULL;
   370 	if (aCopyStyles)
   371 		docStyleList = (CStyleList*)iText.StyleList();
   372 
   373 	ExternalizeRtiHeaderL(aStream,endPos,docStyleList);
   374 	
   375 	if (docStyleList)
   376 		ExternalizeReferencedStylesL(aStream,startPos,endPos);
   377 	
   378 	ExternalizeSharedFormatsL(aStream,startPos,endPos,docStyleList); 
   379 	RPhraseAttribsEntry* virtualTrailingText=ExternalizeParaIxL(aStream,startPos,endPos,docStyleList);
   380 	if (virtualTrailingText)
   381 	    CleanupStack::PushL(virtualTrailingText);
   382 	ExternalizePhraseIxL(aStream,startPos,endPos,virtualTrailingText);
   383 	if (virtualTrailingText)
   384 	    CleanupStack::PopAndDestroy(virtualTrailingText);
   385 	}
   386 
   387 //////////////////////////////////////////////////////////////////////////////////////
   388 
   389 
   390 void CRichTextIndex::ExternalizeRtiHeaderL(RWriteStream& aStream,const TLogicalPosition& aEnd,const CStyleList* aStyleList)const
   391 // Stores index specific information.
   392 //
   393 	{
   394 	TBool incompleteParaFlag=((*iParaIx)[aEnd.iParaElement].iLength>(aEnd.iParaElementOffset+1));
   395 	aStream.WriteUint8L((TUint8)(aStyleList!=NULL));  // flags the presence of paragraph styles.
   396 	aStream.WriteUint8L((TUint8)(incompleteParaFlag!=EFalse)); // the last para has no paragraph delimiter
   397 	if(incompleteParaFlag)
   398 		{//apply paragraph format to final text fragment if logical end is at document end
   399 		aStream.WriteUint8L((TUint8)(aEnd.iDocPos+1 == iText.DocumentLength()));
   400 		}
   401 	}
   402 
   403 
   404 void CRichTextIndex::InternalizeRtiHeaderL(RReadStream& aStream,TRtPasteContext& aContext)
   405 // Load index specific information.
   406 //
   407 	{
   408 	TUint8 flag=0;
   409 	aStream>> flag;
   410 	aContext.iParagraphStylesFlag=(TBool)flag;
   411 	aStream>> flag;
   412 	aContext.iIncompleteParaFlag=(TBool)flag;
   413 	aContext.iApplyFormatToLastFlag=EFalse;
   414 	if(aContext.iIncompleteParaFlag)
   415 		{
   416 		aStream>> flag;
   417 		aContext.iApplyFormatToLastFlag=(TBool)flag;
   418 		}
   419 	}
   420 
   421 //////////////////////////////////////////////////////////////////////////////////////
   422                                                                                                                      
   423 void CRichTextIndex::ExternalizeReferencedStylesL(RWriteStream& aStream,const TLogicalPosition& aStart,
   424 												  const TLogicalPosition& aEnd) const
   425 // Write those styles that are referenced by the paragraphs in the range aStart to aEnd.
   426 //
   427 	{
   428 	CStyleList* list = iText.StyleList();
   429 	if (!list)
   430 	    {
   431 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEREFERENCEDSTYLESL, "EStyleClipboardIntegrityError" );
   432 	    }
   433 	__ASSERT_DEBUG(list,Panic(EStyleClipboardIntegrityError));
   434 	if (aStart.iParaElement > aEnd.iParaElement)
   435 	    {
   436 	    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_EXTERNALIZEREFERENCEDSTYLESL, "EStyleClipboardIntegrityError" );
   437 	    }
   438 	__ASSERT_DEBUG(aStart.iParaElement <= aEnd.iParaElement,Panic(EStyleClipboardIntegrityError));
   439 
   440 	TUint8 numStyles = 0;
   441 	if (list) 
   442 		numStyles = (TUint8)list->Count();
   443 
   444 	HBufC8* buf=HBufC8::NewL(numStyles);
   445 	TPtr8 paraRegister=buf->Des();
   446 	paraRegister.FillZ(numStyles);
   447 	TInt refCount=MarkStyleRegister(paraRegister,aStart.iParaElement,aEnd.iParaElement);
   448 	CleanupStack::PushL(buf);
   449 	ExternalizeItemsPresentInStyleRegisterL(aStream,refCount,paraRegister);
   450 	CleanupStack::PopAndDestroy();
   451 	}
   452 
   453 
   454 void CRichTextIndex::InternalizeSharedFormatsL(RReadStream& aStream, const TRtPasteContext& aContext)
   455 // Restores the list of shared para attribs.
   456 // All existing CParaAttribs on the shared list are removed.
   457 // The reference counts are not restored, but modified dynamically as the para index is loaded.
   458 //
   459 	{
   460 	iParaIx->Reset();  // stops paraIx referencing shared paraAttribs that are about to be released.
   461 	CParaAttribs* currentSharedPara;
   462 	TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
   463 	while ( (currentSharedPara = iterator++) != NULL )
   464 		currentSharedPara->Release(currentSharedPara->iRefCount);
   465 	TUint8 sharedParaCount = ReadTUint8CountL(aStream);
   466 	for (TUint8 paraItem = 0; paraItem < sharedParaCount; paraItem++)
   467 		{
   468 		(void)aStream.ReadInt32L();  // consume the data.
   469 		CParaFormatLayer* paraLayer = InternalizeParagraphFormatL(aStream, aContext);
   470 		CleanupStack::PushL(paraLayer);
   471 
   472 		CCharFormatLayer* charLayer = CCharFormatLayer::NewL(aStream);  // Does not restore based on link.
   473 		charLayer->SetBase(aContext.iGlobalCharLayer);  // base on 'Normal' global char format
   474 		CleanupStack::PushL(charLayer);
   475 		CParaAttribs* paraAttribs = CParaAttribs::NewL(paraLayer, charLayer);
   476 		CleanupStack::PopAndDestroy(2);
   477 		paraAttribs->iRefCount = 0;  // Otherwise recalculating reference counts on internalization of paraIx will result in 1 too many.
   478 		iSharedParaQueHead.AddLast(*paraAttribs);
   479 		}
   480 	}
   481 
   482 
   483 void CRichTextIndex::ExternalizeSharedFormatsL(RWriteStream& aStream,const TLogicalPosition& aStart,const TLogicalPosition& aEnd,const CStyleList* aStyleList)const
   484 // Persist those items in the shared list that are referenced by the paragraphs in the 
   485 // range aStart para to aEnd para.
   486 //
   487 	{
   488 	TUint8 sharedParaCount=SharedParaAttribsEntryCountL();
   489 	// Some paras coverd, so take register
   490 	HBufC8* buf=HBufC8::NewL(sharedParaCount);
   491 	TPtr8 sharedAttribsRegister=buf->Des();
   492 	sharedAttribsRegister.FillZ(sharedParaCount);
   493 	TInt sharedCount=MarkRegister(sharedAttribsRegister,aStart.iParaElement,aEnd.iParaElement);
   494 	CleanupStack::PushL(buf);
   495 	ExternalizeItemsPresentInRegisterL(aStream,sharedCount,sharedAttribsRegister,aStyleList);
   496 	CleanupStack::PopAndDestroy();
   497 	}
   498 
   499 
   500 TUint8 CRichTextIndex::SharedParaAttribsEntryCountL()const
   501 // Return a count of the number of items currently
   502 // in the sharedParaFormats list.
   503 //
   504 // This will leave if more than 255 found, as this is not supported
   505 	{
   506 	TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
   507 	TUint16 sharedParaCount = 0;
   508 	while (iterator++ != NULL)
   509 		sharedParaCount++;
   510 		
   511 	// Currently more than 255 paragraph formats are not supported 
   512 	// If the current document has more, leave with KErrNotSupported
   513 	if (sharedParaCount > 255)
   514 		User::Leave(KErrNotSupported);
   515 	
   516 	return (TUint8)sharedParaCount;
   517 	}
   518 
   519 
   520 TInt CRichTextIndex::MarkRegister(TDes8& aBuf,TInt aStartPara,TInt aEndPara)const
   521 // Indicate in the register, aBuf, once and once only, each paraAttribs that is
   522 // referenced by each of the paragraphs in the specified range.
   523 // Return the number of shared para formats referenced.
   524 //
   525 	{
   526 	TInt parasReferenced=0;
   527 	for (TInt item=aStartPara;item<=aEndPara;item++)
   528 		{
   529 		CParaAttribs* paraAttribs=(*iParaIx)[item].iParaAttribs;
   530 		TInt offset=RefNum(paraAttribs);  // 0 return shows para attribs not in shared list
   531 		if (offset>0)
   532 			{// para entry references a shared para attribs
   533 			if (aBuf[offset-1]==0)
   534 				{
   535 				aBuf[offset-1]=KRegisterItemPresent;  // mark item as needing to be stored
   536 				parasReferenced++;
   537 				}
   538 			}
   539 		}
   540 	return parasReferenced;
   541 	}
   542 
   543 
   544 TInt CRichTextIndex::MarkStyleRegister(TDes8& aBuf,TInt aStartPara,TInt aEndPara)const
   545 // Indicate in the register, aBuf, once and once only, each paragraph style thar is
   546 // referenced by each of the paragraphs in the speciifed range.
   547 // Return the umber of paragraph styles referenced.
   548 //
   549 	{
   550 	TInt stylesReferenced=0;
   551 	const CParaAttribs* para=NULL;
   552 	for (TInt item=aStartPara;item<=aEndPara;item++)
   553 		{
   554 		para=(*iParaIx)[item].iParaAttribs;
   555 		TInt index=iText.StyleList()->IndexByPtr(STATIC_CAST(const CParaFormatLayer*,para->iParaFormat->SenseBase()));
   556 		if (index!=KErrNotFound)
   557 			{
   558 			if (index>=aBuf.Length())
   559 			    {
   560 			    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_MARKSTYLEREGISTER, "EStyleClipboardIntegrityError" );
   561 			    }
   562 			__ASSERT_DEBUG(index<aBuf.Length(),Panic(EStyleClipboardIntegrityError));
   563 			if (aBuf[index]!=KRegisterItemPresent)
   564 				{
   565 				aBuf[index]=KRegisterItemPresent;  // mark item as needing to be stored
   566 		
   567 				stylesReferenced++;
   568 				}
   569 			}
   570 		}
   571 	return stylesReferenced;
   572 	}
   573 
   574 
   575 void CRichTextIndex::ExternalizeItemsPresentInRegisterL(RWriteStream& aStream,TInt aSharedCount,const TDes8& aBuf,const CStyleList* aStyleList)const
   576 // Externalize each object from the shared list that has a corresponding mark in the
   577 // register, aBuf.
   578 //
   579 	{
   580 	if (aSharedCount>(TInt)KMaxTUint8)
   581 	    {
   582 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEITEMSPRESENTINREGISTERL, "ESharedParaCountStreamOverflow" );
   583 	    }
   584 	__ASSERT_DEBUG(aSharedCount<=(TInt)KMaxTUint8,Panic(ESharedParaCountStreamOverflow));
   585 	aStream.WriteUint8L(aSharedCount);
   586 	CParaAttribs* currentSharedPara=NULL;
   587 	TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
   588 	TInt offset=0;
   589 	while ((currentSharedPara=iterator++)!=NULL)
   590 		{
   591 		if (aBuf[offset]==KRegisterItemPresent)
   592 			{
   593 			aStream.WriteInt32L(offset+1);  // the ref-no of the shared para - so it can be spotted in the paste.
   594 			ExternalizeParagraphFormatL(aStream,*currentSharedPara->iParaFormat,aStyleList);
   595 			aStream<< *currentSharedPara->iCharFormat;
   596 			}
   597 		offset++;
   598 		}
   599 	}
   600 
   601 
   602 void CRichTextIndex::ExternalizeItemsPresentInStyleRegisterL(RWriteStream& aStream,TInt aRefStyleCount,
   603 															 const TDes8& aBuf) const
   604 // Externalize each object from the paragraph style list that has a corresponding mark in the register aBuf.
   605 //
   606 	{
   607 	if (aRefStyleCount > (TInt)KMaxTUint8)
   608 	    {
   609 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEITEMSPRESENTINSTYLEREGISTERL, "EStyleClipboardIntegrityError" );
   610 	    }
   611 	__ASSERT_DEBUG(aRefStyleCount <= (TInt)KMaxTUint8,Panic(EStyleClipboardIntegrityError));
   612 	aStream.WriteUint8L(aRefStyleCount);
   613 	TInt count=aBuf.Length();
   614 
   615 	for (TInt ii=0;ii<count;ii++)
   616 		{
   617 		if (aBuf[ii]==KRegisterItemPresent)
   618 			{
   619 			aStream.WriteInt8L(ii+1);  // cannot use 0 index as this represents an absence.
   620 			iText.StyleList()->At(ii).iStyle->ExternalizeL(aStream);
   621 			}
   622 		}
   623   }
   624 
   625 
   626 RPhraseAttribsEntry* CRichTextIndex::ExternalizeParaIxL(RWriteStream& aStream,
   627 										const TLogicalPosition& aStart,
   628 										const TLogicalPosition& aEnd,
   629 										const CStyleList* aStyleList)const
   630 // Externalize each para entry from the para index that falls between 
   631 // aStart paragraph and aEnd paragraph inclusive.
   632 // Any specific paraAttribs (ie not on the shared list) are also externalized.
   633 // * paragraph count
   634 // * 	paragraph length
   635 // *	ref no.  (ref no. 0 means specific/non-shared one)
   636 // *	[specific paragraph format layer - including based-on link]
   637 // *	[specific phrase count]
   638 //
   639 	{
   640 	TParaAttribsEntry lastParaEntry=(*iParaIx)[aEnd.iParaElement];
   641 	TBool incompleteParaFlag=(lastParaEntry.iLength>(aEnd.iParaElementOffset+1));
   642 	aStream.WriteInt32L(1+aEnd.iParaElement-aStart.iParaElement);  // paragraph count
   643 	for (TInt paraItem=aStart.iParaElement;paraItem<aEnd.iParaElement;paraItem++)
   644 		{// Externalize all but last TParaAttribsEntrys from the para index
   645 		TParaAttribsEntry paraEntry=(*iParaIx)[paraItem];
   646 		TInt phraseCount=(paraEntry.iParaAttribs->IsShared()) 
   647 			? 0 
   648 			: paraEntry.iParaAttribs->iPhraseCount;
   649 		if (paraItem==aStart.iParaElement)
   650 			{// Fix & write length & phrase count of the first paragraph.
   651 			paraEntry.iLength=(paraEntry.iLength-aStart.iParaElementOffset);  // fix 1st para length
   652 			if (phraseCount>0)
   653 				phraseCount=phraseCount-(aStart.iPhraseElement-aStart.iParaBasePhraseElement);
   654 			}
   655 		aStream.WriteInt32L(paraEntry.iLength);
   656 		CParaAttribs* paraAttribs=paraEntry.iParaAttribs;
   657 		TUint8 refNo=RefNum(paraAttribs);
   658 		aStream.WriteUint8L(refNo);
   659 		if (refNo==0)
   660 			{// Write specific para layer & phrase count
   661 			ExternalizeParagraphFormatL(aStream,*paraAttribs->iParaFormat,aStyleList);
   662 			aStream.WriteInt32L(phraseCount);
   663 			}
   664 		}
   665 	// Fix & write the length of the last paragraph.
   666 	if (aStart.iParaElement==aEnd.iParaElement)
   667 		lastParaEntry.iLength=(aEnd.iParaElementOffset-aStart.iParaElementOffset)+1;  // copied text contained within 1 para
   668 	else
   669 		lastParaEntry.iLength-=(lastParaEntry.iLength-aEnd.iParaElementOffset)-1;
   670 	aStream.WriteInt32L(lastParaEntry.iLength);
   671 	// Fix & write the phrase count of the last paragraph if it has specific char format
   672 	CParaAttribs* lastParaAttribs=lastParaEntry.iParaAttribs;
   673 	TUint8 refNo=RefNum(lastParaAttribs);
   674 	RPhraseAttribsEntry* virtualTextPhrase=NULL;
   675 	if (incompleteParaFlag)
   676 		{// Adjust the phrase count for this paragraph that is not complete
   677 		TInt phraseCount=0;
   678 		if (aStart.iParaElement==aEnd.iParaElement)
   679 			phraseCount=(aEnd.iPhraseElement-aStart.iPhraseElement)+1;  // copied text contained within 1 para
   680 		else
   681 			phraseCount=(aEnd.iPhraseElement-aEnd.iParaBasePhraseElement)+1;
   682 		if (phraseCount>lastParaAttribs->iPhraseCount)
   683 		    {
   684 		    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEPARAIXL, "ERtExternalizeParaIx" );
   685 		    }
   686 		__ASSERT_DEBUG(phraseCount<=lastParaAttribs->iPhraseCount,Panic(ERtExternalizeParaIx));
   687 		aStream.WriteInt32L(phraseCount);
   688 		if (refNo>0)
   689 			{// Set the virtual phrase representing the trailing text from the shared paragraph.
   690 			virtualTextPhrase=new(ELeave) RPhraseAttribsEntry(lastParaAttribs->iCharFormat,lastParaEntry.iLength);
   691 			CleanupStack::PushL(virtualTextPhrase);
   692 			}
   693 
   694 		// Write out the paragraph format for the paragraph that is not complete
   695 		// - necessary since we need to patch up the style character formatting
   696 		if (aStyleList || aEnd.iDocPos+1 == iText.DocumentLength())
   697 			ExternalizeParagraphFormatL(aStream, *lastParaAttribs->iParaFormat, aStyleList);
   698 		}
   699 	else
   700 		{// This is a complete para - copy region ends on a paragraph delimiter.
   701 		aStream.WriteUint8L(refNo);
   702 		if (refNo==0)
   703 			{// Write specific para layer & phrase count
   704 			ExternalizeParagraphFormatL(aStream,*lastParaAttribs->iParaFormat,aStyleList);
   705 			TInt phraseCount=lastParaAttribs->iPhraseCount;
   706 			if (aStart.iParaElement==aEnd.iParaElement && aStart.iParaElementOffset && aStart.iParaElementOffset>0)
   707 				phraseCount=(aEnd.iPhraseElement-aStart.iPhraseElement)+1;  // copied text contained within 1 para
   708 			aStream.WriteInt32L(phraseCount);
   709 			}
   710 		}
   711 	if (virtualTextPhrase)
   712 	    CleanupStack::Pop(virtualTextPhrase);
   713 	return virtualTextPhrase;
   714 	}
   715 
   716 
   717 void CRichTextIndex::ExternalizeParagraphFormatL(RWriteStream& aStream,const CParaFormatLayer& aLayer,const CStyleList* aStyleList)const
   718 // If a style list is present, write out the based-on links for every paragraph format layer,
   719 // in order to reconnect each paragraph to the right style on loading.
   720 //
   721 	{
   722 	aStream<< aLayer;
   723 	if (aStyleList)
   724 		{
   725 		TInt refNo=0;
   726 		refNo=aStyleList->IndexByPtr(((CParaFormatLayer*)aLayer.SenseBase()));
   727 		// returns the offset in the style list, or a negative error.
   728 		//
   729 		refNo=(refNo!=KErrNotFound)
   730 			? refNo+1  // the nth style where 1st item is offset 1.
   731 			: 0;  // An error value was returned so this thing has no paragraph style as its base
   732 		aStream.WriteInt8L((TInt8)-refNo);
   733 		}
   734  	}
   735 
   736 
   737 CParaFormatLayer* CRichTextIndex::InternalizeParagraphFormatL(RReadStream& aStream,const TRtPasteContext& aContext)
   738 // 
   739 	{
   740 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL(aStream);  // specific paragraph format layer
   741 	TUint index=0;
   742 	if (aContext.iParagraphStylesFlag)
   743 		{
   744 		CleanupStack::PushL(paraLayer);
   745 		TInt refNo=aStream.ReadInt8L();  // Read in the based on link to fix up to the style table
   746 		CleanupStack::Pop();
   747 		index=Abs(refNo);
   748 		}
   749 	if (index>0 && aContext.iStyleList)
   750 		paraLayer->SetBase(aContext.iStyleList->At(index-1).iStyle);
   751 	else
   752 		paraLayer->SetBase(aContext.iGlobalParaLayer);
   753 	return paraLayer;
   754 	}
   755 
   756 
   757 CParaFormatLayer* CRichTextIndex::PasteParagraphFormatL(RReadStream& aStream,const TRtPasteContext& aContext,CStyleMap* aStyleMap)
   758 // 
   759 	{
   760 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL(aStream);  // specific paragraph format layer
   761  
   762 	TUint index=0;
   763 	if (aContext.iParagraphStylesFlag)
   764 		{
   765 		CleanupStack::PushL(paraLayer);
   766 		TInt refNo=aStream.ReadInt8L();  // Read in the based on link to fix up to the style table
   767 		CleanupStack::Pop(paraLayer);	// paraLayer
   768 		index=Abs(refNo);
   769 		}
   770 
   771 	// If a style exists for this paragraph, set it as the base.
   772 	// If a style list doesn't exist, the base paragraph style will be set instead.
   773 	
   774 	if (index>0 && aContext.iStylePasteMode != CParagraphStyle::EIgnoreNewStyles)
   775 	{
   776 		if (!aContext.iStyleList || (aContext.iStyleList && aContext.iStylePasteMode == CParagraphStyle::EConvertNewStyles))
   777 		{
   778 			CleanupStack::PushL(paraLayer);
   779 			iLastCharacterStyle = (aStyleMap->Item(index))->iCharFormatLayer;
   780 
   781 			// If the application doesn't have a style list, 
   782 			// the new paragraph style has to be enforced by merging the old
   783 			// paralayer with the style paralayer
   784 			CParaFormat* MergedParaFormat = CParaFormat::NewLC();
   785 			TParaFormatMask MergedParaFormatMask;
   786 
   787 			CParaFormatLayer* newParaLayer;
   788 
   789 			// Extract the masks from both layers
   790 			paraLayer->SenseL(MergedParaFormat, MergedParaFormatMask);
   791 			(aStyleMap->Item(index))->SenseL(MergedParaFormat, MergedParaFormatMask);
   792 
   793 			newParaLayer = CParaFormatLayer::NewL(MergedParaFormat, MergedParaFormatMask);
   794 
   795 			// Now on stack: MergedParaFormat, paraLayer, ...
   796 			CleanupStack::PopAndDestroy();	// delete MergedParaFormat
   797 
   798 			// Now on stack: paraLayer, ...
   799 			CleanupStack::PopAndDestroy();	// paraLayer
   800 			paraLayer = newParaLayer;
   801 
   802 			paraLayer->SetBase(aContext.iGlobalParaLayer);
   803 
   804 		}
   805 		else 
   806 		{
   807 			paraLayer->SetBase(aStyleMap->Item(index));		
   808 			iLastCharacterStyle = NULL;
   809 		}
   810 	}
   811 	else 
   812 	// Otherwise set the character style to NULL, which will set the global
   813 	// character style (see line 983 approx)
   814 	{
   815 		iLastCharacterStyle = NULL;
   816 		paraLayer->SetBase(aContext.iGlobalParaLayer);
   817 	}
   818 		
   819 	return paraLayer;
   820 	}
   821 
   822 
   823 void CRichTextIndex::InternalizeParaIxL(RReadStream& aStream,const TRtPasteContext& aContext)
   824 // Restore the paragraph index & all associated specific para formats,
   825 // All existing index data is lost.
   826 //
   827 	{
   828 	// The para index has been reset at the start of internalize shared formats.
   829 	TInt paraIxCount=aStream.ReadInt32L();  // paragraph count
   830 	for (TInt currentPara=0;currentPara<paraIxCount;currentPara++)
   831 		{// Restore each paragraph
   832 		TParaAttribsEntry para;
   833 		para.iLength=aStream.ReadInt32L();  // paragraph length
   834 		TUint8 refNo=aStream.ReadUint8L();  // ref no
   835 		if (refNo>0)
   836 			{// Link to para attribs in shared list & up its reference count
   837 			para.iParaAttribs=SharedParaAttribs(refNo);
   838 			para.iParaAttribs->iRefCount++;
   839 			iParaIx->AppendL(para);
   840 			}
   841 		else
   842 			{// Have to build up the specific para attribs
   843 			CParaFormatLayer* paraLayer=InternalizeParagraphFormatL(aStream,aContext);
   844 			CleanupStack::PushL(paraLayer);
   845 			CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraLayer);
   846 				CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,specificParaAttribs));
   847 			specificParaAttribs->iPhraseCount=aStream.ReadInt32L();  // specific phrase count
   848 				CleanupStack::Pop();	// specificParaAttribs
   849 			CleanupStack::PopAndDestroy(); // paraLayer
   850 			para.iParaAttribs=specificParaAttribs;
   851 			CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,specificParaAttribs));
   852 			iParaIx->AppendL(para);
   853 			CleanupStack::Pop();  // specificParaAttribs
   854 			}
   855 		}
   856 	}
   857 
   858 void CRichTextIndex::ExternalizePhraseIxL(RWriteStream& aStream)const
   859 // Called from an Externalize.
   860 // All phrases must be saved.
   861 //
   862 	{ExternalizePhrasesL(aStream,0,PhraseCount());}
   863 
   864 
   865 void CRichTextIndex::ExternalizePhraseIxL(RWriteStream& aStream,
   866 										  const TLogicalPosition& aStart,
   867 										  TLogicalPosition& aEnd,
   868 										  const RPhraseAttribsEntry* aVirtualTrailingText)const
   869 // Called from CopyToStream.
   870 // Here, a text fragment only, is to be saved.
   871 // The last argument may be NULL.  If it is other than NULL, it represents trailing text
   872 // that has no paragraph delimiter, which was part of a shared paragraph.  This must be presented as
   873 // a distinct phrase, since this is the context in which it is pasted back into a document.
   874 //
   875 	{
   876 	TInt phraseCount=PhraseCount();
   877 	phraseCount+=(aVirtualTrailingText) ? 1 : 0;
   878 	TBool noRealPhrasesInRegion=ETrue;
   879 	TInt nn=aStart.iParaElement;
   880 	while (nn<=aEnd.iParaElement && noRealPhrasesInRegion)
   881 		{
   882 		if (!(*iParaIx)[nn].iParaAttribs->IsShared())
   883 			noRealPhrasesInRegion=EFalse;
   884 		nn++;
   885 		}
   886 	//
   887 	if (phraseCount==0 || (phraseCount==1 && aVirtualTrailingText || noRealPhrasesInRegion))  // save this phrase count
   888 		ExternalizePhrasesL(aStream,aStart.iPhraseElement,0,aVirtualTrailingText);
   889 	else
   890 		{
   891 		TBool endParaShared=(*iParaIx)[aEnd.iParaElement].iParaAttribs->IsShared();
   892 		if (endParaShared && aEnd.iPhraseElement>0)
   893 			aEnd.iPhraseElement--;  // Due to action of ScanToPosition
   894 		//
   895 		RPhraseAttribsEntry* backPhrase=&(*iPhraseIx)[aEnd.iPhraseElement];
   896 		TInt backLength=backPhrase->Length();
   897 		if (!(endParaShared || backPhrase->IsPicturePhrase()))
   898 			backPhrase->SetLength(aEnd.iPhraseElementOffset+1);
   899 		//
   900 		RPhraseAttribsEntry* frontPhrase=&(*iPhraseIx)[aStart.iPhraseElement];
   901 		TInt frontLength=frontPhrase->Length();
   902 		if (!frontPhrase->IsPicturePhrase())  // Fix length of first phrase.
   903 			frontPhrase->SetLength(frontLength-aStart.iPhraseElementOffset);
   904 		TRAPD(ret,
   905 			ExternalizePhrasesL(aStream,aStart.iPhraseElement,aEnd.iPhraseElement-aStart.iPhraseElement+1,aVirtualTrailingText));
   906 		//
   907 		// Now fix the altered phrase lengths.
   908 		if (!frontPhrase->IsPicturePhrase())
   909 			frontPhrase->SetLength(frontLength);
   910 		if (!(endParaShared || backPhrase->IsPicturePhrase()))
   911 			backPhrase->SetLength(backLength);
   912 
   913 		__TEST_INVARIANT;	// we lied about being const
   914 
   915 		User::LeaveIfError(ret);
   916 		}
   917 	}
   918 
   919 
   920 void CRichTextIndex::ExternalizePhrasesL(RWriteStream& aStream,TInt aStart,TInt aPhraseCount,
   921 										 const RPhraseAttribsEntry* aVirtualPhrase)const
   922 // Save the specified number of phrases present in the phrase index,
   923 // starting from phrase offset aStart.
   924 //
   925 	{
   926 	ExternalizePhraseCountL(aStream,aPhraseCount+ ((aVirtualPhrase)?1:0) );
   927 	for (TInt phraseItem=aStart;phraseItem<aStart+aPhraseCount;phraseItem++)
   928 		{
   929 		RPhraseAttribsEntry& phraseEntry=(*iPhraseIx)[phraseItem];
   930 		aStream<< phraseEntry;
   931 		}
   932 	if (aVirtualPhrase)
   933 		aStream<< *aVirtualPhrase;
   934 	}
   935 
   936 
   937 void CRichTextIndex::InternalizePhraseIxL(RReadStream& aStream,const CCharFormatLayer* aGlobalCharFormat)
   938 // Load all the phrases from the stream, and insert them into the phrase index
   939 //
   940 	{
   941 	iPhraseIx->Reset();
   942 	TInt phraseCount=aStream.ReadInt32L();
   943 
   944 	// Extend phrase index by required amount
   945 	iPhraseIx->AppendL(RPhraseAttribsEntry(),phraseCount);
   946 	
   947 	for (TInt phraseItem=0;phraseItem<phraseCount;phraseItem++)
   948 		{// Restore each phrase & insert into the phrase index.
   949 		TBool isPicture=(TBool)aStream.ReadUint8L();
   950 		TInt phraseLength=aStream.ReadInt32L();
   951 		CCharFormatLayer* charLayer=CCharFormatLayer::NewL(aStream);
   952 		charLayer->SetBase(aGlobalCharFormat);
   953 		RPhraseAttribsEntry& tPhrase=iPhraseIx->At(phraseItem);
   954 		if (!isPicture)
   955 			new(&tPhrase) RPhraseAttribsEntry(charLayer,phraseLength);
   956 		else
   957 			{// Manufacture new picture header & set its picture store
   958 			CleanupStack::PushL(charLayer);
   959 			TPictureHeader hdr;
   960 			aStream>> hdr;
   961 			TBool ownershipTaken(EFalse);
   962 			CPicturePhrase* picPhrase=CPicturePhrase::NewL(hdr,charLayer,ownershipTaken);
   963 			CleanupStack::Pop();  // char layer
   964 			new(&tPhrase) RPhraseAttribsEntry(picPhrase);
   965 			iPictureCount++;
   966 			}
   967 		}
   968 	}
   969 
   970 
   971 CParaAttribs* CRichTextIndex::SharedParaAttribs(TUint8 aOrdinal)
   972 // Return the handle of the para attribs in the shared para attribs list,
   973 // whose position in the list is specified by the argument aOrdinal.
   974 //
   975 	{
   976 	CParaAttribs* currentSharedPara;
   977 	TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
   978 	TInt match=1;
   979 	while ((currentSharedPara=iterator++)!=NULL && aOrdinal!=match)
   980 		match++;
   981 	if (currentSharedPara==NULL)
   982 	    {
   983 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_SHAREDPARAATTRIBS, "EEndOfSharedParaListEncountered" );
   984 	    }
   985 	__ASSERT_ALWAYS(currentSharedPara!=NULL,Panic(EEndOfSharedParaListEncountered));
   986 	return currentSharedPara;
   987 	}
   988 
   989 
   990 TUint8 CRichTextIndex::RefNum(const CParaAttribs* aParaAttribs)const
   991 // If the para attribs argument is present in the shared list, return a 
   992 // reference to it; otherwise return zero as the reference.
   993 //
   994 	{
   995 	CParaAttribs* currentSharedPara;
   996 	TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
   997 	TUint8	 refNo=1;
   998 	while ((currentSharedPara=iterator++)!=NULL)
   999 		{
  1000 		if (currentSharedPara==aParaAttribs)
  1001 			return refNo;
  1002 		refNo++;
  1003 		}
  1004 	return 0;
  1005 	}
  1006 
  1007 //////////////////////////////////////////////////
  1008 //////////////////////////////////////////////////
  1009 //////////////////////////////////////////////////
  1010 
  1011 void CRichTextIndex::PasteFromStreamL(const CStreamStore& aStore,RReadStream& aStream,TInt aPos,CParagraphStyle::TStylePasteMode aStylePasteMode,const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer)
  1012 //
  1013 	{
  1014 	__TEST_INVARIANT;
  1015 	
  1016 	CancelInsertCharFormat();
  1017 	ScanToPosition(aPos,EScanToPositionAbsolute);
  1018 
  1019 	//
  1020 	//  Get the header
  1021 	TRtPasteContext context(&aStore,aGlobalParaLayer,aGlobalCharLayer,iText.StyleList());
  1022 	context.iPastePos=iPos;
  1023 	context.iStylePasteMode = aStylePasteMode;
  1024 
  1025 	InternalizeRtiHeaderL(aStream,context);  // no rollback required on the header.  Does not alter iPos.
  1026 	//
  1027 	// Get the pargraph styles
  1028 
  1029 	CStyleMap* styleMap = NULL;
  1030 	if (context.iParagraphStylesFlag)
  1031 	{
  1032 		TUint8 styleCount=ReadTUint8CountL(aStream);
  1033 		styleMap=CStyleMap::NewLC(styleCount);
  1034 		TRAPD(ret,
  1035 		PasteStylesL(aStream,*styleMap,context));  // Does not alter iPos.
  1036 		if (ret!=KErrNone) {RbPasteSharedFormatsL(ret);}
  1037 	}
  1038 	//
  1039 	// Get the shared formats
  1040 	TUint8 sharedParaCount=ReadTUint8CountL(aStream);
  1041 	CParaAttribsMap* paraMap=CParaAttribsMap::NewLC(sharedParaCount);
  1042 	TRAPD(ret,
  1043 	PasteSharedFormatsL(aStream,*paraMap,context,styleMap));  // Does not alter iPos.
  1044 	if (ret!=KErrNone)
  1045 		{
  1046 		// beginning of fixing DEF 126651 (1/2)
  1047 		// free orphan object before leave.
  1048 		if (context.iParagraphStylesFlag)
  1049 		{
  1050 			if (!context.iStyleList || context.iStylePasteMode == CParagraphStyle::EConvertNewStyles)
  1051 			{
  1052 			TInt maxSize = styleMap->Count();
  1053 			for (TInt count=0; count<maxSize; count++)
  1054 				{
  1055 				delete (CParagraphStyle*)styleMap->At(count).iT;	
  1056 				}
  1057 			}
  1058 			CleanupStack::PopAndDestroy();	// cleanup styleMap
  1059 		}
  1060 		CleanupStack::PopAndDestroy();  // cleanup paraMap
  1061 		// end of fixing DEF 126651 (1/2)
  1062 		RbPasteSharedFormatsL(ret);
  1063 		}
  1064 	//
  1065 	// Get the markup
  1066 	TRAP(ret,
  1067 	PasteIxL(aStream,context,*paraMap, styleMap));  // context now has both global layers & pastePos
  1068 	if (ret!=KErrNone)
  1069 		{
  1070 		// beginning of fixing DEF 126651 (2/2)
  1071 		// free orphan object before leave.
  1072 		if (context.iParagraphStylesFlag)
  1073 		{
  1074 			if (!context.iStyleList || context.iStylePasteMode == CParagraphStyle::EConvertNewStyles)
  1075 			{
  1076 			TInt maxSize = styleMap->Count();
  1077 			for (TInt count=0; count<maxSize; count++)
  1078 				{
  1079 				delete (CParagraphStyle*)styleMap->At(count).iT;	
  1080 				}
  1081 			}
  1082 			CleanupStack::PopAndDestroy();	// cleanup styleMap
  1083 		}
  1084 		CleanupStack::PopAndDestroy();  // cleanup paraMap
  1085 		// end of fixing DEF 126651 (2/2)
  1086 		RbPasteSharedFormatsL(ret);
  1087 		}
  1088 
  1089 
  1090 	if (context.iParagraphStylesFlag)
  1091 	{
  1092 		TRAP(ret,GenerateAllPhraseLinksL());
  1093 		if (ret != KErrNone)
  1094 			RbPasteSharedFormatsL(ret);
  1095 
  1096 		if (!context.iStyleList || context.iStylePasteMode == CParagraphStyle::EConvertNewStyles)
  1097 		{
  1098 		TInt maxSize = styleMap->Count();
  1099 		for (TInt count=0; count<maxSize; count++)
  1100 			{
  1101 			delete (CParagraphStyle*)styleMap->At(count).iT;	
  1102 			}
  1103 		}
  1104 		CleanupStack::PopAndDestroy();	// cleanup styleMap
  1105 	}
  1106 
  1107 	CleanupStack::PopAndDestroy();  // cleanup paraMap
  1108 
  1109 	__TEST_INVARIANT;
  1110 	}
  1111 
  1112 
  1113 void CRichTextIndex::PasteStylesL(RReadStream& aStream,CStyleMap& aMap,const TRtPasteContext& aContext)
  1114 // Restore the paragraph styles from the specified stream.
  1115 // Restoration is controlled by the flag TStylePasteMode.
  1116 //
  1117 	{
  1118 	
  1119 	TInt styleCount=aMap.iCapacity;
  1120 //	TBool docSupportsStyles = iText.StyleListPresent();	// detect if document has style list
  1121 	CParagraphStyle* style=NULL;
  1122 	for(TUint8 ii=0; ii<styleCount; ii++)
  1123 	{
  1124 		TInt refNo=aStream.ReadInt8L();
  1125 		style=CParagraphStyle::NewL(aStream,*aContext.iGlobalParaLayer,*aContext.iGlobalCharLayer);
  1126 
  1127 		// Now attempt to map this style to one in the document we are adding it to
  1128 		// refNo contains the style number when the selection was cut, which
  1129 		//   is needed to map to the reference no's coming in with the paragraph
  1130 		//   format info
  1131 
  1132 		if (aContext.iStylePasteMode == CParagraphStyle::EIgnoreNewStyles)
  1133 			// We're ignoring all style info, so delete it now
  1134 			delete style;
  1135 		else
  1136 		{
  1137 
  1138 		if (aContext.iStyleList && aContext.iStylePasteMode == CParagraphStyle::EAddNewStyles)
  1139 		{
  1140 			TInt docStyle = iText.StyleList()->IndexByName(style->iName);
  1141 			if (docStyle!=KErrNotFound)
  1142 			{
  1143 				/* Add the existing style into the stylemap */
  1144 				aMap.Bind(refNo,iText.StyleList()->PtrByName(style->iName)->iStyle);
  1145 
  1146 				/* Since this style exists, we can safely delete this copy */
  1147 				delete style;
  1148 
  1149 			}
  1150 			else
  1151 			{
  1152 				// Ok, the document doesn't have this style in it.
  1153 				// So let's add it in, and put a reference in the stylemap
  1154 
  1155 				RParagraphStyleInfo newStyle(style);
  1156 
  1157 				iText.StyleList()->AppendL(&newStyle);
  1158 
  1159 				aMap.Bind(refNo,style);
  1160 
  1161 				// the StyeList takes care of style destruction, so no deletion
  1162 				// is necessary here
  1163 			}
  1164 		}
  1165 		else
  1166 		{
  1167 			// Document doesn't support styles, so save them so they can
  1168 			// be converted to character specific formatting later on
  1169 			aMap.Bind(refNo, style);
  1170 		}
  1171 		}
  1172 
  1173 	}
  1174 }
  1175 
  1176 
  1177 void CRichTextIndex::ImposeCharacterStyleL(CCharFormatLayer** aCharLayer)
  1178 {
  1179 	// This function is used to impose the current character style onto
  1180 	// the character layer for this paragraph. It's used when translating
  1181 	// style information into specific formatting when pasting text containing
  1182 	// styles into text without styles.
  1183 
  1184 	TCharFormatX MergedCharFormat;
  1185 	TCharFormatXMask MergedCharFormatMask;
  1186 			
  1187 	CCharFormatLayer* newCharLayer;
  1188 
  1189 	// Extract the masks from both layers.
  1190 	// use MergedCharFormat to hold the char info which we don't want yet
  1191 
  1192 	(*aCharLayer)->Sense(MergedCharFormat, MergedCharFormatMask);
  1193 	iLastCharacterStyle->Sense(MergedCharFormat, MergedCharFormatMask);
  1194 			
  1195 	// Re-build charLayer;
  1196 	CleanupStack::PushL(*aCharLayer);
  1197 	newCharLayer = CCharFormatLayer::NewL(MergedCharFormat,MergedCharFormatMask);
  1198 	CleanupStack::Pop(*aCharLayer);
  1199 
  1200 	delete (*aCharLayer);
  1201 	(*aCharLayer) = newCharLayer;
  1202 }
  1203 
  1204 
  1205 void CRichTextIndex::PasteSharedFormatsL(RReadStream& aStream,CParaAttribsMap& aMap,const TRtPasteContext& aContext,CStyleMap* aStyleMap)
  1206 // Load shared formats from the specified stream.
  1207 // Only adds the loaded shared format to the list of shared formats
  1208 // if it does not already exist in that list.
  1209 // A map is kept, that for each loaded format specifies its offset within
  1210 // the shared list.
  1211 //
  1212 
  1213 	{
  1214 	TInt mapCount=aMap.iCapacity;
  1215 	for (TUint8 paraItem=0;paraItem<mapCount;paraItem++)
  1216 		{
  1217 		TInt refNo=aStream.ReadInt32L();
  1218 		CParaFormatLayer* paraLayer=PasteParagraphFormatL(aStream,aContext,aStyleMap);
  1219 		CleanupStack::PushL(paraLayer);
  1220 
  1221 		CCharFormatLayer* charLayer=CCharFormatLayer::NewL(aStream);  // Does not restore based on link.
  1222 //// Change specific formatting for this paragraph
  1223 
  1224 		if (iLastCharacterStyle != NULL)
  1225 			ImposeCharacterStyleL(&charLayer);
  1226 		
  1227 		charLayer->SetBase(aContext.iGlobalCharLayer);
  1228 
  1229 		CleanupStack::PushL(charLayer);
  1230 		CParaAttribs* paraAttribs=GetParaAttribsL(paraLayer,charLayer);
  1231 		CleanupStack::PopAndDestroy(2);  // charLayer & paraLayer
  1232 		aMap.Bind(refNo,paraAttribs);
  1233 		paraAttribs->iRefCount--;
  1234 		}
  1235 	}
  1236 
  1237 
  1238 void CRichTextIndex::PasteIxL(RReadStream& aStream,TRtPasteContext& aContext,const CParaAttribsMap& aMap,CStyleMap* aStyleMap/*,CParaAttribs* aSecondReserved*/)
  1239 //
  1240 	{
  1241 	TInt completeParaCount=aStream.ReadInt32L();
  1242 	completeParaCount-=(aContext.iIncompleteParaFlag)? 1 : 0;
  1243 //	Create rollback states
  1244 	TParaAttribsEntry para=(*iParaIx)[aContext.iPastePos.iParaElement];
  1245 	CParaAttribs* reclaimed=RequestReclaimShareL(para.iParaAttribs,&para);  // does not release share
  1246 	iRollbackParaAttribsHandle=NULL;
  1247 	if (reclaimed)
  1248 		{
  1249 		iRollbackParaAttribsHandle=para.iParaAttribs;
  1250 		CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimed));
  1251 		(*iParaIx)[aContext.iPastePos.iParaElement].iParaAttribs=reclaimed;  // Use this reclaimed para attribs
  1252 		}
  1253 //	Split the phrase at the paste position, ready to insert clipboard content
  1254 	TRAPD(ret,
  1255 	SplitPhraseL(aContext.iPastePos.iDocPos));
  1256 		if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
  1257 	ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute);  // pick up the changes
  1258 	aContext.iPastePos=iPos;
  1259 //	Paste all complete paragraphs from the clipboard
  1260 	TInt characterCount=0;
  1261 	RPhraseAttribsEntry* firstParaVirtualPhrase=NULL;
  1262 	TRAP(ret,
  1263 	characterCount=PasteParaIxL(aStream,aContext,completeParaCount,aMap,firstParaVirtualPhrase,aStyleMap));
  1264 		if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
  1265 	CleanupStack::PushL(firstParaVirtualPhrase);
  1266 //	Paste any remaining text fragment.
  1267 	TTextFragment textFragment;
  1268 	if (aContext.iIncompleteParaFlag)
  1269 		{
  1270 		TRAPD(ret,
  1271 		textFragment=GetTextFragmentL(aStream));
  1272 			if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}			
  1273 		characterCount+=textFragment.iLength;
  1274 
  1275 		// Restore the character stle info for the final text fragment
  1276 		if (aContext.iParagraphStylesFlag || aContext.iApplyFormatToLastFlag)
  1277 			{
  1278 			CParaFormatLayer* paraLayer = PasteParagraphFormatL(aStream, aContext, aStyleMap);
  1279 			CleanupStack::PushL(paraLayer);
  1280 			//Restore the paragraph formatting for the final text fragment
  1281 			if (aContext.iApplyFormatToLastFlag)
  1282 				{
  1283 				CParaFormat* paraFormat = CParaFormat::NewLC();
  1284 				TParaFormatMask mask;
  1285 				paraLayer->SenseL(paraFormat,mask);
  1286 				(*iParaIx)[completeParaCount].iParaAttribs->iParaFormat->SetL(paraFormat,mask);
  1287 				CleanupStack::PopAndDestroy(paraFormat);
  1288 				}
  1289 			// We can now discard the format layer, but the character format has been safely
  1290 			// patched to the style character format
  1291 			CleanupStack::PopAndDestroy(paraLayer);
  1292 			}
  1293 		}
  1294 //
  1295 	TRAP(ret,
  1296 	PastePhraseIxL(aStream,aContext,firstParaVirtualPhrase));
  1297 		if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
  1298 	CleanupStack::PopAndDestroy();  // firstParaVirtualPhrase / shallow destroy only - deep copy in phraseIx
  1299 
  1300 //	ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute);  // phraseIx not pasted at this point
  1301 //	TLogicalPosition headParaNormalizePos=iPos;
  1302 	
  1303 	
  1304 	if (completeParaCount==0)
  1305 		{// Adjust paragraph length & phrase count
  1306 		TParaAttribsEntry* para=&(*iParaIx)[aContext.iPastePos.iParaElement];
  1307 		para->iLength+=textFragment.iLength;
  1308 		para->iParaAttribs->iPhraseCount+=textFragment.iPhraseCount;
  1309 		ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute);
  1310 		TLogicalPosition headParaNormalizePos=iPos;
  1311 		MergePhrases(headParaNormalizePos);
  1312 		MergePhrases(aContext.iPastePos.iDocPos+characterCount);
  1313 		DoPasteCleanup(headParaNormalizePos,reclaimed);
  1314 		}
  1315 	else
  1316 		{// Adjust paragraph lengths & phrase counts	
  1317 		TParaAttribsEntry* firstPara=&(*iParaIx)[aContext.iPastePos.iParaElement];
  1318 		firstPara->iLength+=aContext.iPastePos.iParaElementOffset;  // Update length of the first para
  1319 		firstPara->iParaAttribs->iPhraseCount+=(aContext.iPastePos.iPhraseElement-aContext.iPastePos.iParaBasePhraseElement);  // Update the phrase count of the first para attribs
  1320 		ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute);
  1321 		TLogicalPosition headParaNormalizePos=iPos;
  1322 		MergePhrases(headParaNormalizePos);
  1323 		DoPasteCleanup(headParaNormalizePos,reclaimed);
  1324 		//
  1325 		// Adjust the length & phrase count of the new final paragraph
  1326 		TParaAttribsEntry* lastPara=&(*iParaIx)[aContext.iPastePos.iParaElement+completeParaCount];
  1327 		lastPara->iLength=(lastPara->iLength-aContext.iPastePos.iParaElementOffset)+textFragment.iLength;
  1328 		lastPara->iParaAttribs->iPhraseCount-=(aContext.iPastePos.iPhraseElement-aContext.iPastePos.iParaBasePhraseElement);
  1329 		// phrase count may be wrong - since the added phrase may have been merged into the current one.
  1330 		// Cant just add these phrases.
  1331 		lastPara->iParaAttribs->iPhraseCount+=textFragment.iPhraseCount;
  1332 		ScanToPosition(aContext.iPastePos.iDocPos+characterCount,EScanToPositionAbsolute);
  1333 		MergePhrases(iPos);
  1334 		DoPasteCleanup(iPos,(CParaAttribs*)NULL);
  1335 		}
  1336 	RebalanceIndex();
  1337 	NormalizeSharedList();
  1338 	}
  1339 
  1340 
  1341 void CRichTextIndex::DoPasteCleanup(TLogicalPosition& aNormalizePos,CParaAttribs* aReclaimed)
  1342 //
  1343 	{
  1344 	NormalizeNow(aNormalizePos);
  1345 	if (aReclaimed)
  1346 		CleanupStack::Pop();
  1347 	}
  1348 
  1349 
  1350 TInt CRichTextIndex::PasteParaIxL(RReadStream& aStream,TRtPasteContext& aContext,TInt aCompleteParaCount,const CParaAttribsMap& aMap,RPhraseAttribsEntry*& aFirstParaVirtualPhrase, CStyleMap* aStyleMap)
  1351 //
  1352 	{
  1353 	TParaAttribsEntry para;
  1354 	iParaIx->InsertL(aContext.iPastePos.iParaElement,para,aCompleteParaCount);	
  1355 	aContext.iParasPasted=aCompleteParaCount;
  1356 	TInt characterCount=0;
  1357 	if (aCompleteParaCount>0)
  1358 		{
  1359 		para=DoPasteFirstIntoParaL(aStream,aMap,aContext,aFirstParaVirtualPhrase, aStyleMap);
  1360 		TInt paraItem=0;
  1361 		(*iParaIx)[aContext.iPastePos.iParaElement+paraItem]=para;
  1362 		characterCount+=para.iLength;
  1363 		}
  1364 	CleanupStack::PushL(TCleanupItem(DiscardPhraseOnCleanup,aFirstParaVirtualPhrase));
  1365 	for (TInt paraItem=1;paraItem<aCompleteParaCount;paraItem++)
  1366 		{// Paste all the paras that have paragraph delimiters.
  1367 		para=DoPasteIntoParaL(aStream,aMap,aContext,aStyleMap);
  1368 		(*iParaIx)[aContext.iPastePos.iParaElement+paraItem]=para;
  1369 		characterCount+=para.iLength;
  1370 		}
  1371 	CleanupStack::Pop();  // firstParaVirtualPhrase
  1372 // ASSERT: At this point we have pasted all paras that were in the stream.
  1373 	if (aContext.iParasPasted!=aCompleteParaCount)
  1374 	    {
  1375 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_PASTEPARAIXL, "EPasteParaIxError" );
  1376 	    }
  1377 	__ASSERT_DEBUG(aContext.iParasPasted==aCompleteParaCount,Panic(EPasteParaIxError));
  1378 	return characterCount;
  1379 	}
  1380 			
  1381 			
  1382 TParaAttribsEntry CRichTextIndex::DoPasteFirstIntoParaL(RReadStream& aStream,const CParaAttribsMap& aMap,const TRtPasteContext& aContext,RPhraseAttribsEntry*& aFirstParaVirtualPhrase,CStyleMap* aStyleMap)
  1383 //
  1384 	{
  1385 	TParaAttribsEntry para;
  1386 	para.iLength=aStream.ReadInt32L();
  1387 	TUint8 refNo=aStream.ReadUint8L();
  1388 	if (refNo>0)
  1389 		{// This is the first pasted para, so if shared, must convert to phrase on the fly.
  1390 		CParaAttribs* paraAttribs=aMap.Item(refNo);
  1391 		CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraAttribs->iParaFormat);
  1392 		CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,specificParaAttribs));
  1393 		//
  1394 		CCharFormatLayer* layer=CCharFormatLayer::NewCopyBaseL(paraAttribs->iCharFormat);
  1395 		CleanupStack::PushL(layer);
  1396 		aFirstParaVirtualPhrase=new(ELeave) RPhraseAttribsEntry(layer,para.iLength);
  1397 		CleanupStack::Pop(2);  // layer & specificParaAttribs
  1398 		//
  1399 		para.iParaAttribs=specificParaAttribs;
  1400 		if (para.iParaAttribs==NULL)
  1401 		    {
  1402 		    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DOPASTEFIRSTINTOPARAL, "ESharedFormatsMapIntegrityError" );
  1403 		    }
  1404 		__ASSERT_ALWAYS(para.iParaAttribs!=NULL,Panic(ESharedFormatsMapIntegrityError));
  1405 		}
  1406 	else
  1407 		{// Have to build up the specific para attribs
  1408 		CParaFormatLayer* paraLayer=PasteParagraphFormatL(aStream,aContext,aStyleMap);
  1409 		CleanupStack::PushL(paraLayer);
  1410 		TInt phraseCount=aStream.ReadInt32L();  // specific phrase count
  1411 		CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraLayer);
  1412 		specificParaAttribs->iPhraseCount=phraseCount;
  1413 		para.iParaAttribs=specificParaAttribs;
  1414 		CleanupStack::PopAndDestroy();  // a copy of paraLayer is taken!
  1415 		}
  1416 	return para;
  1417 	}
  1418 
  1419 
  1420 TParaAttribsEntry CRichTextIndex::DoPasteIntoParaL(RReadStream& aStream,const CParaAttribsMap& aMap,const TRtPasteContext& aContext,CStyleMap* aStyleMap)
  1421 //
  1422 	{
  1423 	TParaAttribsEntry para;
  1424 	para.iLength=aStream.ReadInt32L();
  1425 	TUint8 refNo=aStream.ReadUint8L();
  1426 	if (refNo>0)
  1427 		{// Link to para attribs in shared list & up its reference count
  1428 		para.iParaAttribs=aMap.Item(refNo);
  1429 		if (para.iParaAttribs==NULL)
  1430 		    {
  1431 		    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DOPASTEINTOPARAL, "ESharedFormatsMapIntegrityError" );
  1432 		    }
  1433 		__ASSERT_ALWAYS(para.iParaAttribs!=NULL,Panic(ESharedFormatsMapIntegrityError));
  1434 		para.iParaAttribs->iRefCount++;
  1435 		}
  1436 	else
  1437 		{// Have to build up the specific para attribs
  1438 		CParaFormatLayer* paraLayer=PasteParagraphFormatL(aStream,aContext,aStyleMap);
  1439 		CleanupStack::PushL(paraLayer);
  1440 		TInt phraseCount=aStream.ReadInt32L();  // specific phrase count
  1441 		CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraLayer);
  1442 		specificParaAttribs->iPhraseCount=phraseCount;
  1443 		para.iParaAttribs=specificParaAttribs;
  1444 		CleanupStack::PopAndDestroy();  // a copy of paraLayer is taken!
  1445 		}
  1446 	return para;
  1447 	}
  1448 
  1449 
  1450 TTextFragment CRichTextIndex::GetTextFragmentL(RReadStream& aStream)
  1451 //
  1452 	{
  1453 	TTextFragment textFragment;
  1454 	textFragment.iLength=aStream.ReadInt32L();
  1455 	textFragment.iPhraseCount=aStream.ReadInt32L();
  1456 	return textFragment;
  1457 	}
  1458 
  1459 
  1460 void CRichTextIndex::PastePhraseIxL(RReadStream& aStream,TRtPasteContext& aContext,const RPhraseAttribsEntry* aFirstParaVirtualPhrase)
  1461 // The character position to paste at should now fall on a phrase boundary.
  1462 //
  1463 	{
  1464 // ASSERT: Having pasted the paraIx, the para containig pastePos has had the containing phrase split at that point.
  1465 	if (aContext.iPastePos.iPhraseElementOffset!=0)
  1466 	    {
  1467 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_PASTEPHRASEIXL, "EPastePhraseIxErr" );
  1468 	    }
  1469 	__ASSERT_ALWAYS(aContext.iPastePos.iPhraseElementOffset==0,Panic(EPastePhraseIxErr));
  1470 
  1471 	TInt offset=0;
  1472 	TInt phraseCount=aStream.ReadInt32L();  // leave caught by calling function.  No state change yet.
  1473 	if (aFirstParaVirtualPhrase)
  1474 		{
  1475 		iPhraseIx->InsertL(aContext.iPastePos.iPhraseElement,*aFirstParaVirtualPhrase);
  1476 		offset++;
  1477 		}
  1478 	RPhraseAttribsEntry holdingPhrase;
  1479 	iPhraseIx->InsertL(aContext.iPastePos.iPhraseElement+offset,holdingPhrase,phraseCount);
  1480 	for (TInt phraseItem=0;phraseItem<phraseCount;phraseItem++)
  1481 		{// Restore each phrase & insert into the phrase index.
  1482 		RPhraseAttribsEntry phrase;
  1483 		TRAPD(ret,
  1484 //		phrase=DoPastePhraseL(aStream,aContext));  // !! delete this if the code works
  1485 		DoPastePhraseL(aStream,aContext,phrase));
  1486 			if (ret!=KErrNone) {RbPastePhraseIxL(aContext.iPastePos,phraseCount+offset,ret);}
  1487 		(*iPhraseIx)[aContext.iPastePos.iPhraseElement+phraseItem+offset]=phrase;
  1488 		if (phrase.IsPicturePhrase() && iText.PictureFactory())
  1489 			iPictureCount++;
  1490 		}
  1491 	}
  1492 
  1493 
  1494 void CRichTextIndex::DoPastePhraseL(RReadStream& aStream,const TRtPasteContext& aContext,RPhraseAttribsEntry& aPhrase)
  1495 //
  1496 	{
  1497 	TBool isPicture=(TBool)aStream.ReadUint8L();
  1498 	TInt phraseLength=aStream.ReadInt32L();
  1499 
  1500 	if (!(isPicture && phraseLength==1 || !isPicture))
  1501 	    {
  1502 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DOPASTEPHRASEL, "KErrCorrupt" );
  1503 	    }
  1504 	__ASSERT_ALWAYS(isPicture && phraseLength==1 || !isPicture,User::Leave(KErrCorrupt));
  1505 	
  1506 	CCharFormatLayer* charLayer=CCharFormatLayer::NewL(aStream);
  1507 /// Change this character format if style formatting is to be applied
  1508 
  1509 // Paste style for the trailing text
  1510 	if (iLastCharacterStyle != NULL)
  1511 			ImposeCharacterStyleL(&charLayer);
  1512 	
  1513 		charLayer->SetBase(aContext.iGlobalCharLayer);
  1514 
  1515 	CPicturePhrase* picPhrase=NULL;
  1516 	const MPictureFactory* factory=iText.PictureFactory();
  1517 	TBool pictureLoadError=EFalse;
  1518 	if (isPicture)
  1519 		{
  1520 		TPictureHeader hdr;
  1521 		aStream>> hdr;  // hdr.iPicture always references a picture in the deferred store.
  1522 		if (isPicture && factory)
  1523 			{// Manufacture new picture phrase & set its picture store
  1524 			CleanupStack::PushL(charLayer);
  1525 			TBool ownershipTaken(EFalse);
  1526 			picPhrase=CPicturePhrase::NewL(hdr,charLayer,ownershipTaken);
  1527 			CleanupStack::Pop();  // charLayer - picPhrase takes ownership
  1528 			CleanupStack::PushL(picPhrase);
  1529 			TRAPD(r,factory->NewPictureL(hdr,*aContext.iStore));  // load picture since clipboard store is transient.
  1530 			// r=KErrNotSupported  // we don't recognise the picture type
  1531 			if (r==KErrNone)
  1532 				{
  1533 				picPhrase->iPicHdr.iPicture=hdr.iPicture;  // make picPhrase point at the restored picture object.
  1534 				TRAP(r,
  1535 				hdr.iPicture->DetachFromStoreL(CPicture::EDetachFull));  // recurse the call to detach the picture from the store
  1536 				}
  1537 			switch (r)
  1538 				{
  1539 				case(KErrNone):
  1540 					break;
  1541 				default:
  1542 					pictureLoadError=ETrue;
  1543 					picPhrase->iCharFormat=NULL;
  1544 					CleanupStack::PopAndDestroy(picPhrase);
  1545 					if (r!=KErrNotSupported)
  1546 					    {
  1547 						OstTrace1( TRACE_FATAL, DUP2_CRICHTEXTINDEX_DOPASTEPHRASEL, "Leave code=%d", r );
  1548 					    User::Leave(r);
  1549 					    }
  1550 				}
  1551 			}
  1552 		}
  1553 	if (isPicture && factory && !pictureLoadError)
  1554 		new(&aPhrase) RPhraseAttribsEntry(picPhrase);
  1555 	else
  1556 		new(&aPhrase) RPhraseAttribsEntry(charLayer,phraseLength);
  1557 	
  1558 	// The ownership has been transfered to RPhraseAttribsEntry
  1559 	if(picPhrase)
  1560 	    CleanupStack::Pop(picPhrase);
  1561 	}
  1562 
  1563 
  1564 TUint8 CRichTextIndex::ReadTUint8CountL(RReadStream& aStream)const
  1565 	{return aStream.ReadUint8L();}
  1566 
  1567 
  1568 void CRichTextIndex::RbPasteSharedFormatsL(TInt aRet)
  1569 // For each para attribs that has been read in, release all shares on it, if it
  1570 // does not already exist in shared list.
  1571 //
  1572 	{
  1573 	NormalizeSharedList();
  1574 	OstTrace1( TRACE_FATAL, CRICHTEXTINDEX_RBPASTESHAREDFORMATSL, "Leave code=%d", aRet );
  1575 	User::Leave(aRet);
  1576 	}
  1577 		
  1578 	
  1579 void CRichTextIndex::NormalizeSharedList()
  1580 // Removes unreferenced (non-shared) shared paras from the shared list.
  1581 // Called in rollback situations.
  1582 // Also called naturally following a paste, since an incoming shared para
  1583 // may be pasted into an existing para, making it non-shared.
  1584 //
  1585 	{
  1586 	CParaAttribs* currentSharedPara=NULL;
  1587 	TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
  1588 	while ((currentSharedPara=iterator++)!=NULL)
  1589 		{// Rollback the shared list.
  1590 		if (currentSharedPara->iRefCount==0)
  1591 			{// Remove this unreferenced item from the shared list.
  1592 			currentSharedPara->iRefCount=1;
  1593 			currentSharedPara->Release();
  1594 			}
  1595 		}
  1596 	}
  1597 
  1598 
  1599 void CRichTextIndex::RbRemoveInsertedParaAttribsEntries(TInt aFirstParaInsertPos,TInt aInsertedParaCount)
  1600 // Rollback on leaving part way through inserting paragraph records into the para index.
  1601 // For all pasted paragraph records, release their share on their paraAttribs if present.
  1602 //
  1603 	{
  1604 	for (TInt ii=0;ii<aInsertedParaCount;ii++)
  1605 		{
  1606 		CParaAttribs* paraAttribs=(*iParaIx)[aFirstParaInsertPos+ii].iParaAttribs;
  1607 		if (paraAttribs)
  1608 			paraAttribs->Release();
  1609 		}
  1610 	iParaIx->Delete(aFirstParaInsertPos,aInsertedParaCount);
  1611 	}
  1612 
  1613 
  1614 void CRichTextIndex::RbPasteParaIxL(const TLogicalPosition& aPos,TInt aParasPasted,TInt aRet)
  1615 // Rollback on leaving part way through the pasting of the para index.
  1616 // For all pasted paras, release their share on their paraAttribs.
  1617 //
  1618 	{
  1619 	RbRemoveInsertedParaAttribsEntries(aPos.iParaElement,aParasPasted);
  1620 	MergePhrases(aPos.iDocPos);  // updates iPos to paste pos.
  1621 	// 
  1622 	//
  1623 	if (iRollbackParaAttribsHandle)
  1624 		{
  1625 		RemoveFromPhraseIx(iPos.iPhraseElement,1);  // Removes the phrase created from ResquestReclaim on para
  1626 		(*iParaIx)[aPos.iParaElement].iParaAttribs=iRollbackParaAttribsHandle;
  1627 		iRollbackParaAttribsHandle=NULL;
  1628 		}
  1629 	OstTrace1( TRACE_FATAL, CRICHTEXTINDEX_RBPASTEPARAIXL, "Leave code=%d", aRet );
  1630 	User::Leave(aRet);
  1631 	}
  1632 
  1633 
  1634 void CRichTextIndex::RbRemoveInsertedPhraseAttribsEntries(TInt aFirstPhraseInsertPos,TInt aInsertedPhraseCount)
  1635 // Rollback on leaving part way through the pasting of the phrase index.
  1636 // For all pasted phrases, discard their components.
  1637 //
  1638 	{
  1639 	for (TInt ii=0;ii<aInsertedPhraseCount;ii++)
  1640 		{
  1641 		RPhraseAttribsEntry& phrase=(*iPhraseIx)[aFirstPhraseInsertPos+ii];
  1642 		if (phrase.IsPicturePhrase())
  1643 			iPictureCount--;
  1644 		if (phrase.CharFormat())
  1645 			phrase.Discard();
  1646 		}
  1647 	iPhraseIx->Delete(aFirstPhraseInsertPos,aInsertedPhraseCount);
  1648 	}
  1649 
  1650 
  1651 void CRichTextIndex::RbPastePhraseIxL(const TLogicalPosition& aPos,TInt aPhraseCount,TInt aRet)
  1652 // Rollback on leaving part way through the pasting of the phrase index.
  1653 // For all pasted phrases, discard their components.
  1654 //
  1655 	{
  1656 	RbRemoveInsertedPhraseAttribsEntries(aPos.iPhraseElement,aPhraseCount);
  1657 	OstTrace1( TRACE_FATAL, CRICHTEXTINDEX_RBPASTEPHRASEIXL, "Leave code=%d", aRet );
  1658 	User::Leave(aRet);
  1659 	}