os/textandloc/textrendering/texthandling/stext/TXTINDEX.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 #include <s32stor.h>
    24 #include "TXTFMLYR.H"
    25 #include "TXTETEXT.H"
    26 #include "TXTLAYDC.H"
    27 #include "TXTSTYLE.H"
    28 #include "TXTINDEX.H"
    29 
    30 #include "OstTraceDefinitions.h"
    31 #ifdef OST_TRACE_COMPILER_IN_USE
    32 #include "TXTINDEXTraces.h"
    33 #endif
    34 
    35 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
    36 #include "TXTFMLYR_INTERNAL.H"
    37 #include "TXTSTYLE_INTERNAL.H"
    38 #endif
    39 
    40 TGlobalLayerInfoAppend::TGlobalLayerInfoAppend()
    41 	: iAggParaFormatLayer(NULL),iAggCharFormatLayer(NULL),iComParaFormatLayer(NULL),iComCharFormatLayer(NULL)
    42 	{}
    43 
    44 TGlobalLayerInfoAppend::TGlobalLayerInfoAppend(const CParaFormatLayer* aAggParaFormatLayer,const CCharFormatLayer* aAggCharFormatLayer,
    45 							const CParaFormatLayer* aComParaFormatLayer,const CCharFormatLayer* aComCharFormatLayer)
    46 	: iAggParaFormatLayer(aAggParaFormatLayer),iAggCharFormatLayer(aAggCharFormatLayer),
    47 	  iComParaFormatLayer(aComParaFormatLayer),iComCharFormatLayer(aComCharFormatLayer)
    48 	{}
    49 
    50 
    51 TTextFragment::TTextFragment():
    52 	iLength(0),
    53 	iPhraseCount(0)
    54 	{
    55 	}
    56 
    57 
    58 TCurrentIndexRecords::TCurrentIndexRecords()
    59 	{
    60 	}
    61 
    62 
    63 DLLEXPORT_C void CRichTextIndex::__DbgTestInvariant()const
    64 // Provides class invariants.  Explanations below:
    65 //
    66 	{
    67 #ifdef _DEBUG
    68 // ASSERT: Every phrase index is consistent with its corresponding paragraph.
    69 	TInt zeroLengthPhraseCount=0;
    70 	TInt maxPara=iParaIx->Count();
    71 	TInt numberOfReferencesToSharedList=0;
    72 	TInt currentPhraseElement=0;
    73 	for (TInt para=0;para<maxPara;para++)
    74 		{
    75 		// ASSERT: The basedOn link is valid.
    76 		CFormatLayer* thisLayer=(*iParaIx)[para].iParaAttribs->iParaFormat;
    77 		CFormatLayer* base=CONST_CAST(CFormatLayer*,thisLayer->SenseBase());
    78 		if (base==NULL)
    79 		    {
    80 		    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DBGTESTINVARIANT, "base==NULL" );
    81 		    }
    82 		__ASSERT_DEBUG(base!=NULL,User::Invariant());
    83 		if ((*iParaIx)[para].iParaAttribs->iRefCount>0)
    84 			numberOfReferencesToSharedList++;
    85 		TInt paragraphLength=(*iParaIx)[para].iLength;
    86 		TInt sumOfPhraseLengths=0;
    87 		TInt maxPhrase=(*iParaIx)[para].iParaAttribs->PhraseCount();
    88 		for (TInt phrase=0;phrase<maxPhrase;phrase++)
    89 			{
    90 			if (maxPhrase>1)
    91 				{
    92 				const RPhraseAttribsEntry* phrase=&(*iPhraseIx)[currentPhraseElement];
    93 				CCharFormatLayer* charFormatLayer=phrase->CharFormat();
    94 				// ASSERT: The basedOn link is valid.
    95 				if (charFormatLayer->SenseBase()==NULL)
    96 				    {
    97 				    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_DBGTESTINVARIANT, "charFormatLayer->SenseBase()==NULL" );
    98 				    }
    99 				__ASSERT_DEBUG(charFormatLayer->SenseBase()!=NULL,User::Invariant());
   100 				if (TInt(charFormatLayer->SenseBase())<=0x1000)
   101 				    {
   102 				    OstTrace0( TRACE_DUMP, DUP2_CRICHTEXTINDEX_DBGTESTINVARIANT, "TInt(charFormatLayer->SenseBase())<=0x1000" );
   103 				    }
   104 				__ASSERT_DEBUG(TInt(charFormatLayer->SenseBase())>0x1000,User::Invariant());
   105 				sumOfPhraseLengths+=(*iPhraseIx)[currentPhraseElement].Length();
   106 				if ((*iPhraseIx)[currentPhraseElement].Length()==0)
   107 					zeroLengthPhraseCount++;
   108 				currentPhraseElement++;
   109 				}
   110 			else
   111 				{
   112 				CCharFormatLayer* charFormatLayer=(*iParaIx)[para].iParaAttribs->iCharFormat;
   113 				// ASSERT: The basedOn link is valid.
   114 				if (charFormatLayer->SenseBase()==NULL)
   115 				    {
   116 				    OstTrace0( TRACE_DUMP, DUP3_CRICHTEXTINDEX_DBGTESTINVARIANT, "charFormatLayer->SenseBase()==NULL" );
   117 				    }
   118 				__ASSERT_DEBUG(charFormatLayer->SenseBase()!=NULL,User::Invariant());
   119 				sumOfPhraseLengths+=(*iParaIx)[para].iLength;
   120 				}
   121 			}
   122 		if (sumOfPhraseLengths!=paragraphLength)
   123 		    {
   124 		    OstTrace0( TRACE_DUMP, DUP4_CRICHTEXTINDEX_DBGTESTINVARIANT, "sumOfPhraseLengths!=paragraphLength" );
   125 		    }
   126 		__ASSERT_DEBUG(sumOfPhraseLengths==paragraphLength,User::Invariant());
   127 		}
   128 // ASSERT: We have no unexpected phrases left over
   129 	if (currentPhraseElement!=-1 &&
   130             currentPhraseElement!=iPhraseIx->Count())
   131 	    {
   132 	    OstTrace0( TRACE_DUMP, DUP5_CRICHTEXTINDEX_DBGTESTINVARIANT, "We have no unexpected phrases left over" );
   133 	    }
   134 	__ASSERT_DEBUG(currentPhraseElement==-1 ||
   135 					currentPhraseElement==iPhraseIx->Count(),User::Invariant());
   136 // ASSERT: There is either zero(0) or one(1) zero length phrase in the whole index
   137 	if (!((zeroLengthPhraseCount==0) ||
   138             (zeroLengthPhraseCount==1 && iPendingNewPhrasePos!=EInsertCharFormatReset)))
   139 	    {
   140 	    OstTrace0( TRACE_DUMP, DUP6_CRICHTEXTINDEX_DBGTESTINVARIANT, "There is either zero(0) or one(1) zero length phrase in the whole index" );
   141 	    }
   142 	__ASSERT_DEBUG( (zeroLengthPhraseCount==0) ||
   143 					(zeroLengthPhraseCount==1 && iPendingNewPhrasePos!=EInsertCharFormatReset),
   144 					User::Invariant());
   145 // ASSERT: the number of paraEntries with paraAttribs of refCount>0 == the sum of refCounts in the shared list.
   146 //			or is only one less - as is set when SetInsertCharFormat is called on a shared paraAttribs.
   147 	TInt totalReferenceCount=0;
   148 	if (!iSharedParaQueHead.IsEmpty())
   149 		{
   150 		CParaAttribs* currentSharedPara;
   151 		TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
   152 		while ((currentSharedPara=iterator++)!=NULL)
   153 			totalReferenceCount+=currentSharedPara->iRefCount;
   154 		}
   155 	if ((numberOfReferencesToSharedList!=totalReferenceCount) &&
   156             (numberOfReferencesToSharedList!=totalReferenceCount-1))
   157 	    {
   158 	    OstTrace0( TRACE_DUMP, DUP7_CRICHTEXTINDEX_DBGTESTINVARIANT, "Invariant" );
   159 	    }
   160 	__ASSERT_DEBUG((numberOfReferencesToSharedList==totalReferenceCount) ||
   161 				   (numberOfReferencesToSharedList==totalReferenceCount-1),User::Invariant());
   162 // ASSERT: iPictureCount corresponds to the number of pictures in the stored in the index.
   163 	TInt picCount=0;
   164 	TInt phraseCount=(iPhraseIx) ? iPhraseIx->Count() : 0;
   165 	for (TInt item=0;item<phraseCount;item++)
   166 		picCount+=((*iPhraseIx)[item].IsPicturePhrase())
   167 					? 1
   168 					: 0;
   169 	if (iPictureCount!=picCount)
   170 	    {
   171 	    OstTrace0( TRACE_DUMP, DUP8_CRICHTEXTINDEX_DBGTESTINVARIANT, "Invariant" );
   172 	    }
   173 	__ASSERT_DEBUG(iPictureCount==picCount,User::Invariant());
   174 #endif
   175 	}
   176 
   177 
   178 CRichTextIndex* CRichTextIndex::NewL(const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,
   179 									 const CRichText& aText,TInt aParaGran,TInt aPhraseGran)
   180 // Return a handle to a new instance of this class.
   181 // Requires the global format layer handles on which to base the first content.
   182 //
   183 	{
   184 	CRichTextIndex* self=new(ELeave) CRichTextIndex(aText);
   185 	CleanupStack::PushL(self);
   186 	self->ConstructL(aGlobalParaLayer,aGlobalCharLayer,aParaGran,aPhraseGran);
   187 	CleanupStack::Pop();
   188 	return self;
   189 	}
   190 
   191 
   192 CRichTextIndex::CRichTextIndex(const CRichText& aText):
   193 	iText(aText),
   194 	iPendingNewPhrasePos(EInsertCharFormatReset),
   195 	iSharedParaQueHead(_FOFF(CParaAttribs,link))
   196 	{
   197 	}
   198 
   199 
   200 void CRichTextIndex::ConstructL(const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,TInt aParaGran,TInt aPhraseGran)
   201 // Provides a fully initialised rich text index.
   202 // Upon construction, the index contains a single paragraph of length 1 (para.terminator).
   203 // This first paragraph initially has constant character formatting, ie the shared para list
   204 // must have one item in it.
   205 //
   206 	{
   207 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL();  // Creates empty layer.
   208 	paraLayer->SetBase(aGlobalParaLayer);  // Sets basedOn to global default.
   209 	CleanupStack::PushL(paraLayer);
   210 	CCharFormatLayer* charLayer=CCharFormatLayer::NewL();  // Creates empty layer.
   211 	charLayer->SetBase(aGlobalCharLayer);  // Sets basedOn to global default.
   212 	CleanupStack::PushL(charLayer);
   213 	CParaAttribs* paraAttribs=GetParaAttribsL(paraLayer,charLayer);
   214 	CleanupStack::PopAndDestroy(2);
   215 	CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,paraAttribs));
   216 	iParaIx=new(ELeave) CArrayFixSeg<TParaAttribsEntry>(aParaGran);
   217 	TParaAttribsEntry para(1,paraAttribs);
   218 	iParaIx->AppendL(para);
   219 	CleanupStack::Pop();
   220 	iPhraseIx=new(ELeave) CArrayFixSeg<RPhraseAttribsEntry>(aPhraseGran);
   221 
   222 	__TEST_INVARIANT;
   223 	}
   224 
   225 
   226 CRichTextIndex::~CRichTextIndex()
   227 // Free up all storage allocated in the rich text index.
   228 //
   229 	{
   230 	TInt count;
   231 	if (iPhraseIx)
   232 		{// Destroy the phrase index and its contents.
   233 		if (!iParaIx)
   234 		    {
   235 		    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_CRICHTEXTINDEX, "EPhraseIxPresentWithNoParaIx" );
   236 		    }
   237 		__ASSERT_ALWAYS(iParaIx,Panic(EPhraseIxPresentWithNoParaIx));
   238 		count=iPhraseIx->Count();
   239 		for (TInt offset=0;offset<count;offset++)
   240 			(*iPhraseIx)[offset].Discard();
   241 		}
   242 	delete iPhraseIx;
   243 	if (iParaIx)
   244 		{// Destroy the para index and its contents.
   245 		count=iParaIx->Count();
   246 		for (TInt offset=0;offset<count;offset++)
   247 			{
   248 			CParaAttribs* pA=(*iParaIx)[offset].iParaAttribs;
   249 			if (!pA->IsShared())
   250 				pA->Release();
   251 			}
   252 		}
   253 	delete iParaIx;
   254 	RebalanceIndex();
   255 	//
   256 	// Clear the shared list.  (Will usually be empty by now
   257 	// unless internalize failed after getting shared list & before getting all para data.
   258 	CParaAttribs* currentSharedPara;
   259 	TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
   260 	while ((currentSharedPara=iterator++)!=NULL)
   261 		currentSharedPara->Release(currentSharedPara->iRefCount);
   262 	if (!iSharedParaQueHead.IsEmpty())
   263 	    {
   264 	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_CRICHTEXTINDEX, "ERichTextIndexIntegrityErr" );
   265 	    }
   266 	__ASSERT_ALWAYS(iSharedParaQueHead.IsEmpty(),Panic(ERichTextIndexIntegrityErr));
   267 	}
   268 
   269 
   270 TInt CRichTextIndex::CharPosOfParagraph(TInt& aLength,TInt aParaOffset)const
   271 // Returns the character position of the first character of paragraph aParaOffset,
   272 // where aParaOffset specifies the nth paragraph.
   273 // The length of this nth paragraph is written to aLength.
   274 //
   275 // If aParaOffset specifies a paragraph that does not exist, EScanEndOfData is returned.
   276 //
   277 	{
   278 	__TEST_INVARIANT;
   279 
   280 	if (aParaOffset>=iParaIx->Count())
   281 		return CPlainText::EScanEndOfData;
   282 	TInt pos=0,offset=0;
   283 	for (offset=0;offset<aParaOffset;offset++)
   284 		pos+=(*iParaIx)[offset].iLength;
   285 	aLength=(*iParaIx)[offset].iLength;
   286 	return pos;
   287 	}
   288 
   289 
   290 TInt CRichTextIndex::ParagraphNumberForPos(TInt& aPos)const
   291 // Returns the paragraph offset for the specified character position aPos.
   292 // aPos is in turn modified to hold the character position of the first character
   293 // of this paragraph.  If aPos is already on a paragraph boundary then do nothing.
   294 //
   295 	{
   296 	__TEST_INVARIANT;
   297 
   298 	((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionAbsolute);
   299 	aPos-=iPos.iParaElementOffset;
   300 	return iPos.iParaElement;
   301 	}
   302 
   303 
   304 void CRichTextIndex::DocumentChanged()const
   305 	{
   306  	MUTABLE_CAST(TLogicalPosition&,iLastUsed).Clear();
   307 	}
   308 
   309 
   310 void CRichTextIndex::DoSoloInsertL(TInt aPos,TInt aLength)
   311 // Updates the index following the insertion of content into a single phrase.
   312 // First find the phrase record the governs the insert pos.
   313 // (1) If the current phrase is a text phrase, then simply extend the length of the phrase record.
   314 // (2) If the current phrase is a picture phrase then this cannot be extended.
   315 // There is 1 pathological case here:
   316 // (i) The [current] picture phrase is the 1st phrase in the paragraph, and the insert pos
   317 // is before this phrase. (paragraph insert pos == 0).
   318 // In this case we must insert a new text phrase *before* the picture phrase.
   319 // (ii) In normal circumstances the inserted text is after the picture phrase.  So, if the
   320 // following phrase is of the same character format as the picture then we can re-use this phrase.
   321 // If it is not of the same character format (or is also a picture phrase),
   322 // then we must insert a new text phrase of the correct format immediately following the current picture phrase.
   323 //
   324 	{
   325 	RebalanceIndex();
   326 	if (!((iPendingNewPhrasePos == EInsertCharFormatReset) || (aPos == iPendingNewPhrasePos)))
   327 		CancelInsertCharFormat();
   328 
   329 	ScanToPosition(aPos,EScanToPositionMatchLeft);
   330 	TCurrentIndexRecords current; GetCurrentRecords(current);
   331 	if (current.iPhrase && current.iPhrase->IsPicturePhrase())
   332 		{// Paragraph has specific char format, and current phrase is picture phrase.
   333 		TInt newPhraseRequired=EFalse;
   334 		CCharFormatLayer* charLayer=current.iPhrase->CharFormat();
   335 		if (FirstPhraseOfParagraph() && iPos.iPhraseElementOffset==0)
   336 			newPhraseRequired=ETrue;  // Text is inserted at the start of the para behind the picture.
   337 		else
   338 			{// Check for re-use of the next text phrase.
   339 			iPos.iPhraseElement++;
   340 			iPos.iPhraseElementOffset=0;
   341 			const RPhraseAttribsEntry& nextPhrase=(*iPhraseIx)[iPos.iPhraseElement];
   342 			if (nextPhrase.IsPicturePhrase() || !nextPhrase.CharFormat()->IsIdentical(charLayer,EFalse))
   343 				{// Need a new phrase if the formats don't match OR the next phrase is a picture phrase.
   344 				newPhraseRequired=ETrue;  // Need to create a new phrase to take this insert.
   345 				}
   346 			}
   347 		if (newPhraseRequired)
   348 			{// Insert new phrase & record this fact
   349 			CCharFormatLayer* charFormat=CCharFormatLayer::NewCopyBaseL(charLayer);
   350 			RPhraseAttribsEntry newPhrase(charFormat);
   351 			CleanupStack::PushL(charFormat);
   352 			iPhraseIx->InsertL(iPos.iPhraseElement,newPhrase);
   353 			CleanupStack::Pop();
   354 			current.iParaAttribs->iPhraseCount++;
   355 			}
   356 		GetCurrentRecords(current);  // Update current records, cos used below.
   357 		}
   358 	// Extend the lengths in the index.
   359 	current.iParaEntry->iLength+=aLength;  // Increase length of paragraph.
   360 	if (current.iPhrase)
   361 		current.iPhrase->AdjustLength(aLength);  // Increase the length of this phrase.
   362 	iPos.iPhraseElementOffset+=aLength;
   363 	iPos.iParaElementOffset+=aLength;
   364 	iPos.iDocPos+=aLength;
   365 	}
   366 
   367 
   368 void CRichTextIndex::InsertL(TInt aPos,const TDesC& aBuf,const CParaFormatLayer& aGlobalParaFormatLayer)
   369 // Updates the index following the insertion of a descriptor of text into the document.
   370 // Correctly handles embedded paragraph delimiters.
   371 //
   372 	{
   373 	__TEST_INVARIANT;
   374 	TInt bufLen=aBuf.Length();
   375 
   376 	RebalanceIndex();
   377 	TInt paragraphCount=0;
   378 	const TText* start=aBuf.Ptr();
   379 	const TText* end=start+bufLen;
   380 	while (start<end)
   381 		{
   382 		if (*start++==CEditableText::EParagraphDelimiter)
   383 			paragraphCount++;
   384 		}
   385 	if (paragraphCount==0)
   386 		{
   387 		if (bufLen > 0)
   388 			{
   389 			DoSoloInsertL(aPos,bufLen);
   390 			CancelInsertCharFormat();
   391 			}
   392 		__TEST_INVARIANT;
   393 		return;
   394 		}
   395 	ScanToPosition(aPos,EScanToPositionMatchLeft);
   396 	TLogicalPosition pastePos=iPos;  // Used to adjust paragraph/phrase lengths later
   397 
   398 	if (paragraphCount>1)
   399 		{
   400 		TCurrentIndexRecords current;
   401 		GetCurrentRecords(current);
   402 		const CParaAttribs& paraAttribs=*current.iParaAttribs;
   403 		CCharFormatLayer* charLayer=(paraAttribs.IsShared())
   404 								? paraAttribs.iCharFormat
   405 								: (*iPhraseIx)[iPos.iPhraseElement].CharFormat();
   406 		CParaFormatLayer* paraLayer=paraAttribs.iParaFormat;
   407 		CParaAttribs* theParaAttribs=GetParaAttribsL(paraLayer,charLayer);
   408 		CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,theParaAttribs));
   409 		iParaIx->InsertL(pastePos.iParaElement,TParaAttribsEntry(0,theParaAttribs),paragraphCount-1);
   410 		CleanupStack::Pop();
   411 		theParaAttribs->iRefCount+=paragraphCount-2;	// add the extra references
   412 		}
   413 	TLogicalPosition pos;
   414 	TRAPD(ret,
   415 	SplitParagraphAtPastePosL(pastePos,pos,aGlobalParaFormatLayer));
   416 		if (ret!=KErrNone)
   417 			{
   418 			RbRemoveInsertedParaAttribsEntries(pastePos.iParaElement,paragraphCount-1);
   419 			User::Leave(ret);
   420 			}
   421 
   422 	if (paragraphCount>1)
   423 		{	// swap the entries for the split and first inserted paragraph
   424 		TParaAttribsEntry& paraEntry=(*iParaIx)[pastePos.iParaElement+paragraphCount-1];
   425 		TParaAttribsEntry& insertEntry=(*iParaIx)[pastePos.iParaElement];
   426 		TParaAttribsEntry temp=paraEntry;
   427 		paraEntry=insertEntry;
   428 		insertEntry=temp;
   429 		}// Cos weve inserted new paragraphs in front of the governing one.
   430 
   431 	// Sort out the front para
   432 	TParaAttribsEntry& frontPara=(*iParaIx)[pastePos.iParaElement];
   433 	TPtrC buf(aBuf);
   434 	TInt lengthOfFirstPara=ParaLengthFromBuffer(buf);
   435 	frontPara.iLength+=lengthOfFirstPara;  // Adjust the para length
   436 	buf.Set(aBuf.Right(aBuf.Length()-lengthOfFirstPara-1));
   437 	if (!frontPara.iParaAttribs->IsShared() && lengthOfFirstPara != 0)
   438 		{
   439 		RPhraseAttribsEntry* phrase=&(*iPhraseIx)[pastePos.iPhraseElement];
   440 		if (phrase->IsPicturePhrase())
   441 			{	// move insertion past any picture phrase
   442 			pastePos.iPhraseElement++;
   443 			pastePos.iPhraseElementOffset=0;
   444 			phrase=&(*iPhraseIx)[pastePos.iPhraseElement];
   445 			}
   446 		phrase->AdjustLength(lengthOfFirstPara);  // Adjust the phrase length
   447 		}
   448 	for (TInt paraItem=1;paraItem<paragraphCount;paraItem++)
   449 		{// For each para inserted between the fist and last
   450 		TParaAttribsEntry& para=(*iParaIx)[pastePos.iParaElement+paraItem];
   451 		TInt length=ParaLengthFromBuffer(buf)+1;
   452 		if (length==KErrNotFound)
   453 		    {
   454 		    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_INSERTL, "EInsertEmbeddedParaErr" );
   455 		    }
   456 		__ASSERT_DEBUG(length!=KErrNotFound,Panic(EInsertEmbeddedParaErr));
   457 		para.iLength=length;
   458 		buf.Set(buf.Right(buf.Length()-length));
   459 		}
   460 	// For final paragrph
   461 	TInt trailingTextLen=buf.Length();
   462 	if (trailingTextLen>0)
   463 		{
   464 		TParaAttribsEntry& backPara=(*iParaIx)[pos.iParaElement];
   465 		backPara.iLength+=trailingTextLen;
   466 		if (!backPara.iParaAttribs->IsShared())
   467 			{
   468 			RPhraseAttribsEntry& phrase=(*iPhraseIx)[pos.iPhraseElement];
   469 			phrase.AdjustLength(trailingTextLen);  // Adjust phrase length
   470 			}
   471 		}
   472 	//
   473 	// Now tidy up
   474 	if (bufLen>1 && iPendingNewPhrasePos!=EInsertCharFormatReset)
   475 		{
   476 		iPendingNewPhrasePos=aPos+(bufLen-trailingTextLen);
   477 		CancelInsertCharFormat();
   478 		}
   479 
   480 	__TEST_INVARIANT;
   481 	}
   482 
   483 
   484 void CRichTextIndex::SplitParagraphAtPastePosL(TLogicalPosition& aPastePos,TLogicalPosition& aNewPos,
   485 												const CParaFormatLayer& aGlobalParaFormatLayer)
   486 // Breaks the paragraph specified by the logical position aPastePos, inserting a paragraph delimiter.
   487 //
   488 	{
   489 	TInt insertPendingPos = iPendingNewPhrasePos;
   490 	DoSoloInsertL(aPastePos.iDocPos,1);
   491 	TBool insertCharFormatDeleted = EFalse;
   492 	if (InsertCharFormatIsActive())
   493 		{
   494 		insertCharFormatDeleted = DeleteInsertCharFormat();
   495 		iPendingNewPhrasePos = EInsertCharFormatReset;
   496 		}
   497 	TRAPD(ret,
   498 	InsertParagraphL(aPastePos.iDocPos+1,aGlobalParaFormatLayer));  // Split the current para (and maybe phrase index).
   499 	if (ret!=KErrNone)
   500 		{
   501 		// locate the character inserted by DoSoloInsertL() above
   502 		ScanToPosition(aPastePos.iDocPos,EScanToPositionAbsolute);
   503 		TCurrentIndexRecords current;
   504 		GetCurrentRecords(current);
   505 		current.iParaEntry->iLength--;
   506 		if (current.iPhrase)
   507 			{
   508 			current.iPhrase->AdjustLength(-1);  // collapse phrase by right amount
   509 			if (insertPendingPos!=EInsertCharFormatReset)
   510 				{
   511 				if (current.iPhrase->Length()!=0)
   512 				    {
   513 				    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SPLITPARAGRAPHATPASTEPOSL, "Invariant" );
   514 				    }
   515 				__ASSERT_DEBUG(current.iPhrase->Length()==0,User::Invariant());
   516 				iPendingNewPhrasePos=insertPendingPos;
   517 				}
   518 			else if (current.iPhrase->Length()==0)
   519 				{
   520 				RemoveFromPhraseIx(iPos.iPhraseElement,1);
   521 				current.iParaAttribs->iPhraseCount--;
   522 				if (current.iParaAttribs->PhraseCount()<=1)
   523 				    {
   524 				    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_SPLITPARAGRAPHATPASTEPOSL, "Invariant" );
   525 				    }
   526 				__ASSERT_DEBUG(current.iParaAttribs->PhraseCount()>1,User::Invariant());
   527 				}
   528 			}
   529 		OstTrace1( TRACE_DUMP, DUP2_CRICHTEXTINDEX_SPLITPARAGRAPHATPASTEPOSL, "Leave code=%d", ret );
   530 		User::Leave(ret);
   531 		}
   532 	if (insertPendingPos != EInsertCharFormatReset)
   533 		ConsolidateAt(insertPendingPos, insertCharFormatDeleted?
   534 			EPositionOnly : EFollowingPhrase);
   535 	ScanToPosition(aPastePos.iDocPos+1,EScanToPositionMatchLeft);  // Gives us the next para.
   536 	aNewPos=iPos;
   537 	}
   538 
   539 
   540 TInt CRichTextIndex::ParaLengthFromBuffer(TDesC& aBuf)const
   541 // Returns the length of the first para found in the buffer.
   542 // The returned length excludes the paragraph delimiter character.
   543 // Returns KNotFound if there is no paragraph delimiter.
   544 //
   545 	{return aBuf.Locate(CEditableText::EParagraphDelimiter);}
   546 
   547 
   548 void CRichTextIndex::InsertL(TInt aPos,const TPictureHeader& aPicHdr, TBool& aPictureOwnershipTaken)
   549 // Updates the index following the insertion of a picture header object into the text
   550 // component.  This is accomplished by creating & inserting a picture phrase at the
   551 // relevant place.
   552 //
   553 	{
   554 	__TEST_INVARIANT;
   555 
   556 	RebalanceIndex();
   557 // ASSERT: A valid picture header, referencing a valid picture has been inserted.
   558 	if (!aPicHdr.iPicture.IsPtr() || aPicHdr.iPicture.AsPtr()==NULL)
   559 	    {
   560 	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_INSERTL, "EInsertNullPicHdrData" );
   561 	    }
   562 	__ASSERT_ALWAYS(aPicHdr.iPicture.IsPtr() && aPicHdr.iPicture.AsPtr()!=NULL,Panic(EInsertNullPicHdrData));
   563 // ASSERT: The current insert pos hasn't been changed without cancelling SetInsertCharFormat.
   564 	if ((iPendingNewPhrasePos!=EInsertCharFormatReset) && (aPos!=iPendingNewPhrasePos))
   565 	    {
   566 	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXTINDEX_INSERTL, "ESetInsertCharFormatIntegrityErr" );
   567 	    }
   568 	__ASSERT_ALWAYS((iPendingNewPhrasePos==EInsertCharFormatReset) || (aPos==iPendingNewPhrasePos),
   569 					Panic(ESetInsertCharFormatIntegrityErr));
   570 	if (iPendingNewPhrasePos!=EInsertCharFormatReset)
   571 		CancelInsertCharFormat();  // Cancel this state before inserting picture. Rebalances the index.
   572 	ScanToPosition(aPos,EScanToPositionMatchLeft);
   573 	TCurrentIndexRecords current; GetCurrentRecords(current);
   574 	TCharFormatX format;
   575 	TCharFormatXMask mask;  //...and build up its format.
   576 	CCharFormatLayer* baseChar;
   577 	GetPhraseFormat(current,format,mask,baseChar);  //...inherit format from prev. phrase.
   578 	// Create the picture phrase. Takes ownership of the aPicHdr.iPicture
   579 	CPicturePhrase* picture=CPicturePhrase::NewL(aPicHdr,format,mask,baseChar,aPictureOwnershipTaken);
   580 	CleanupStack::PushL(picture);
   581 	//New reclaimed CParaAttribs instance
   582 	CParaAttribs* reclaimed=RequestReclaimShareL(current.iParaAttribs,current.iParaEntry);
   583     //Store the old CParaAttribs instance in rollbackParaAttribsHandle
   584 	CParaAttribs* rollbackParaAttribsHandle=current.iParaAttribs;
   585 	if (reclaimed)
   586 		{
   587 		CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimed));
   588 		current.iParaEntry->iParaAttribs=reclaimed;  // Use this reclaimed para attribs (the new CParaAttribs instance)
   589 		}
   590 	GetCurrentRecords(current);
   591 //	ASSERT: The reclaim succeeded.  We must always end up with a PhraseIx-not constant char format.
   592 	if (current.iPhrase==NULL)
   593 	    {
   594 	    OstTrace0( TRACE_DUMP, DUP3_CRICHTEXTINDEX_INSERTL, "EReclaimShareError" );
   595 	    }
   596 	__ASSERT_DEBUG(current.iPhrase!=NULL,Panic(EReclaimShareError));
   597 	TRAPD(ret1,
   598 	SplitPhraseL(aPos));  // Phrase may not be split if at boundary.
   599 
   600     if (ret1!=KErrNone)
   601         {
   602         RbInsertPicture(rollbackParaAttribsHandle);//Restore the old CParaAttribs instance
   603         User::Leave(ret1);
   604         }
   605 
   606 	TInt offset=(PhraseSplit())?1:0;  // Insert position of new phrase relative to current.
   607 	RPhraseAttribsEntry newPhrase(picture);
   608 	TRAPD(ret2,
   609 	iPhraseIx->InsertL(iPos.iPhraseElement+offset,newPhrase));
   610     if (ret2!=KErrNone)
   611         {
   612         RbInsertPicture(rollbackParaAttribsHandle);//Restore the old CParaAttribs instance
   613         User::Leave(ret2);
   614         }
   615 
   616 	if(reclaimed)
   617 	    {
   618         CleanupStack::Pop();//"Pop" for CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimed));
   619 	    }
   620 	CleanupStack::Pop(picture);
   621 	// Update counts etc. - cannot leave now.
   622 	current.iParaEntry->iLength+=RPhraseAttribsEntry::EPicturePhraseLength;
   623 	current.iParaAttribs->iPhraseCount++;  // for the picture phrase
   624 	iPictureCount++;
   625 
   626 	// Commit
   627 	if(reclaimed)
   628 	    {
   629         rollbackParaAttribsHandle->Release();  // Release hold on original shared paraAttribs.
   630 	    }
   631 
   632 	__TEST_INVARIANT;
   633 	//coverity[memory_leak]
   634 	}
   635 
   636 void CRichTextIndex::RbInsertPicture(CParaAttribs* aGoodParaAttribs)
   637 // Reinstate the original good paraAttribs.
   638 // Then rollback the SplitPhrase() call if it succeeded.
   639 //
   640 	{
   641 	(*iParaIx)[iPos.iParaElement].iParaAttribs=aGoodParaAttribs;
   642 	if (PhraseSplit())
   643 		{// Rollback the SplitPhrase()
   644 		TInt length=(*iPhraseIx)[iPos.iPhraseElement+1].Length();
   645 		RemoveFromPhraseIx(iPos.iPhraseElement+1);
   646 		(*iPhraseIx)[iPos.iPhraseElement].AdjustLength(length);
   647 		}
   648 	}
   649 
   650 
   651 // Insert a new paragraph immediately following character position aPos, fixing the length of the preceeding
   652 // paragraph.  The new paragraph preserves any explicit paragraph/character formatting, and is based on the
   653 // global layers.  (Do not need to rebalance the index here; a previous call to DoSoloInsertL accomplishes this)
   654 
   655 void CRichTextIndex::InsertParagraphL(TInt aPos,const CParaFormatLayer& aGlobalParaFormatLayer)
   656 	{
   657 	ScanToPosition(aPos,EScanToPositionMatchLeft);
   658 	TCurrentIndexRecords current;
   659 	GetCurrentRecords(current);
   660 	TParaAttribsEntry newPara;
   661 	CCharFormatLayer* charLayer;
   662   	if (current.iPhrase)  // entry in phrase index
   663 		charLayer=current.iPhrase->CharFormat();
   664 	else
   665 		charLayer=current.iParaAttribs->iCharFormat;
   666 	//
   667 	// New para format layer, based on normal, inheriting specific format
   668 	CParaFormatLayer* currentParaFormat=current.iParaAttribs->iParaFormat;
   669 	CParaFormatLayer* newParaLayer=CParaFormatLayer::NewL(currentParaFormat);
   670 	const CParaFormatLayer& currentStyle=STATIC_CAST(const CParaFormatLayer&,*currentParaFormat->SenseBase());
   671 	const TUid currentStyleType=currentStyle.Type();
   672 
   673 	// !!
   674 	// Only change to Normal if current style is a built-in one
   675 	// or we are not at the end of a heading style.
   676 	TBool useNormal;
   677 	if (currentStyleType==KNormalParagraphStyleUid)
   678 		useNormal=ETrue;
   679 	else if (currentStyleType==KUserDefinedParagraphStyleUid)
   680 		useNormal=EFalse;
   681 	else if (iPos.iParaElementOffset<=(current.iParaEntry->iLength-2))  // cos of previous call to DoSoloInsertL()
   682 		useNormal=EFalse;
   683 	else
   684 		useNormal=ETrue;
   685 	newParaLayer->SetBase((useNormal)
   686 		? &aGlobalParaFormatLayer
   687 		: &currentStyle);
   688 	const CCharFormatLayer* newCharBase=(useNormal)
   689 		? iText.GlobalCharFormatLayer()
   690 		: STATIC_CAST(const CParagraphStyle&,currentStyle).CharFormatLayer();
   691 	CleanupStack::PushL(newParaLayer);
   692 	//
   693 	if (current.iParaAttribs->IsShared())
   694 		{// Current para has constant char format - so the new one also has constant char format
   695 		// New char format layer, based on normal, inheriting specific format
   696 		CCharFormatLayer* newCharLayer=CCharFormatLayer::NewL(charLayer);
   697 		newCharLayer->SetBase(newCharBase);
   698 		CleanupStack::PushL(newCharLayer);
   699 		newPara.iParaAttribs=GetParaAttribsL(newParaLayer,newCharLayer);
   700 		CleanupStack::PopAndDestroy(2);  // newCharLayer/newParaLayer
   701 		CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,newPara.iParaAttribs));
   702 		iParaIx->InsertL(iPos.iParaElement+1,newPara);
   703 		CleanupStack::Pop();  // paraAttribs cleanup item
   704 		GetCurrentRecords(current);		// could be changed by InsertL() above
   705 		}
   706 	else  // Do the split myself since this para has specific character formatting.
   707 		{// Make the new CParaAttribs
   708 		CParaAttribs* newParaAttribs=CParaAttribs::NewL(newParaLayer);
   709 		CleanupStack::PopAndDestroy();  // newParaLayer - copy owned by newParaAttribs
   710 		CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,newParaAttribs));
   711 		//
   712 		// Split current phrase & insert if necessary.
   713 		// Split even when we are at a phrase boundary -> this introduces an z.l.p. for the insertion point
   714 		RPhraseAttribsEntry& insertPhrase=iPhraseIx->At(iPos.iPhraseElement);
   715 		TInt insertPendingPos=(iPos.iPhraseElementOffset==insertPhrase.Length()) ? aPos : EInsertCharFormatReset;
   716 		DoSplitPhraseL(insertPhrase,iPos.iPhraseElementOffset,current.iParaAttribs);  // Ups iPhraseCount
   717 		//
   718 		// Insert the new paragraph.
   719 		newPara.iParaAttribs=newParaAttribs;
   720 		TRAPD(ret,
   721 		iParaIx->InsertL(iPos.iParaElement+1,newPara));  // Inserts the new paraAttribsEntry record.
   722 			if (ret!=KErrNone)
   723 				{
   724 				RemoveFromPhraseIx(iPos.iPhraseElement+1,1);	// inserted by DoSplitPhraseL
   725 				current.iParaAttribs->iPhraseCount--;
   726 				User::Leave(ret);
   727 				}
   728 
   729 		iPendingNewPhrasePos=insertPendingPos;
   730 		CleanupStack::Pop();			// newParaAttribs. All OK now
   731 		GetCurrentRecords(current);		// could be changed by InsertL() above
   732 		//
   733 		// Calculate new paraAttribs phrase counts.
   734 		TInt remainder=(iPos.iPhraseElement+1)-iPos.iParaBasePhraseElement;
   735 		TInt newPhraseCount=current.iParaAttribs->iPhraseCount-remainder;
   736 		newParaAttribs->iPhraseCount=newPhraseCount;
   737 		current.iParaAttribs->iPhraseCount=remainder;
   738 
   739 		const CArrayFix<RPhraseAttribsEntry>& phraseIx=*iPhraseIx;
   740 		TInt startPhrase=iPos.iPhraseElement+1;
   741 		for (TInt ii=startPhrase; ii<startPhrase+newPhraseCount; ii++)
   742 			{
   743 			RPhraseAttribsEntry phrase=phraseIx[ii];
   744 			phrase.CharFormat()->SetBase(newCharBase);
   745 			}
   746 		//
   747 		// The index now reflects the correct state.
   748 		// Next, the efficiency thing - see if the new paras can share existing ones.
   749 		if (newPhraseCount==1)
   750 			Share(iParaIx->At(iPos.iParaElement+1),iPos.iParaBasePhraseElement+remainder);
   751 		if (remainder==1)
   752 			Share(iParaIx->At(iPos.iParaElement),iPos.iParaBasePhraseElement);
   753 		}
   754 	// Alter the length of the original paragraph and the new paragraph.
   755 	TInt currentLength=current.iParaEntry->iLength;
   756 	current.iParaEntry->iLength=iPos.iParaElementOffset;
   757 	((*iParaIx)[iPos.iParaElement+1]).iLength+=currentLength-current.iParaEntry->iLength;  // Alters the length of the copy of aNewPara.
   758 	}
   759 
   760 
   761 void CRichTextIndex::SetForDeleteL(TIndexDeleteInfo& aInfo,TInt aPos,TInt aLength)
   762 //
   763 	{
   764 	__TEST_INVARIANT;  // Do not need to RebalanceIndex(); part of defined behaviour for delete.
   765 
   766 	aInfo.iDeleteLength=aLength;
   767 	//
   768 	// Check for simple cases first
   769 	DocumentChanged();  // clears internal position record.
   770 	ScanToPosition(aPos,EScanToPositionAbsolute,&iLastUsed);
   771 	TCurrentIndexRecords current;
   772 	GetCurrentRecords(current);
   773 	aInfo.iStartPara=iPos.iParaElement;
   774 	aInfo.iEndPara=iPos.iParaElement;  // default
   775 	aInfo.iDeletePos=iPos;  // default
   776 	//
   777 	TInt startParaLength=current.iParaEntry->iLength;
   778 	TInt lengthRemainingInPara=startParaLength-iPos.iParaElementOffset;
   779 	if (aLength<lengthRemainingInPara)
   780 		{// Case is delete-from-paragraph
   781 		aInfo.iDeleteType=TIndexDeleteInfo::EDeleteFromParagraph;
   782 		return;
   783 		}
   784 	//
   785 	ScanToPosition(aPos+aLength,EScanToPositionMatchLeft,&iLastUsed);  // Forces endPara to be next para when just removing a para delimiter.
   786 	aInfo.iEndPara=iPos.iParaElement;
   787 //	if (iPos.iParaElementOffset==0)
   788 //		{// Can use delete-paragraph
   789 //		aInfo.iDeleteType=TIndexDeleteInfo::EDeleteParagraph;
   790 //		return;
   791 //		}
   792 	//
   793 	// Set for the general (leaving) delete.
   794 	GetCurrentRecords(current);
   795 	CParaAttribs* reclaimedEndPara=RequestReclaimShareL(current.iParaAttribs,current.iParaEntry); // does not release share.
   796 	TParaAttribsEntry* origEndParaEntry=current.iParaEntry;
   797 	CParaAttribs* origParaAttribs=current.iParaAttribs;
   798 	TInt endPosPhrase=iPos.iPhraseElement;
   799 	if (reclaimedEndPara)
   800 		origEndParaEntry->iParaAttribs=reclaimedEndPara;
   801 	// Get start para info.
   802 	ScanToPosition(aPos,EScanToPositionAbsolute);
   803 	GetCurrentRecords(current);
   804 	CParaAttribs* reclaimedStartPara=NULL;
   805 	TRAPD(ret,
   806 	reclaimedStartPara=RequestReclaimShareL(current.iParaAttribs,current.iParaEntry));
   807 		if (ret!=KErrNone)
   808 			{
   809 			if (reclaimedEndPara)
   810 				{
   811 				reclaimedEndPara->Release();
   812 				RemoveFromPhraseIx(endPosPhrase);
   813 				origEndParaEntry->iParaAttribs=origParaAttribs;
   814 				}
   815 			User::Leave(ret);
   816 			}
   817 	if (reclaimedEndPara)
   818 		origParaAttribs->Release();  // Release share on the original end para attribs
   819 	if (reclaimedStartPara)
   820 		{// Use the specific start  para
   821 		current.iParaAttribs->Release();
   822 		current.iParaEntry->iParaAttribs=reclaimedStartPara;
   823 		ScanToPosition(aPos,EScanToPositionAbsolute);  // Pick up reclaimed phrase.
   824 		}
   825 	aInfo.iDeletePos=iPos;  // internal position of aPos after any reclaim
   826 	// Note: iDeleteType can surely be made obsolete now? TPB 7/11/2000
   827 	aInfo.iDeleteType=TIndexDeleteInfo::EDeleteFromParagraph;
   828 	
   829 	/*
   830 	 * Pointer to memory allocated to 'reclaimedEndPara' is assigned to 
   831 	 * 'origEndParaEntry->iParaAttribs' on line 706. The memory will be 
   832 	 * released in CRichTextIndex's destructor.
   833 	 */ 
   834 	// coverity[memory_leak]
   835 	}
   836 
   837 
   838 TBool CRichTextIndex::DeleteParagraph(TInt aPos,TInt aLength)
   839 // Remove aCount entire paragraphs from the text.
   840 // Leave-safe
   841 // Returns EFalse indicating that no paragraphs were merged together,
   842 // as a result of the delete action.
   843 // Does NOT preserve any zero-length/insert pending state.
   844 //
   845 	{
   846 	__TEST_INVARIANT;  // Do not need to RebalanceIndex(); part of defined behaviour for delete.
   847 
   848 	CancelInsertCharFormat();
   849 	ScanToPosition(aPos,EScanToPositionAbsolute,&iLastUsed);
   850 
   851 	if (iPos.iParaElementOffset!=0)
   852 	    {
   853 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DELETEPARAGRAPH, "EDeleteParagraphInvalidStartValue" );
   854 	    }
   855 	__ASSERT_DEBUG(iPos.iParaElementOffset==0,Panic(EDeleteParagraphInvalidStartValue));
   856 
   857 	TIndexDeleteInfo info;
   858 	info.iDeleteType=TIndexDeleteInfo::EDeleteParagraph;
   859 	info.iDeletePos=iPos;
   860 	info.iStartPara=iPos.iParaElement;
   861 	//
   862 	TInt documentLength=iText.DocumentLength();
   863 	TInt pos=(aPos+aLength>documentLength
   864 		? documentLength
   865 		: aPos+aLength);
   866 	ScanToPosition(pos,EScanToPositionMatchLeft,&iLastUsed);  // Forces endPara to be next para when just removing a para delimiter.
   867 
   868 	info.iEndPara=iPos.iParaElement;
   869 	info.iDeleteLength=aLength;
   870 
   871 	DeleteNow(info);
   872 	// do not want to call TidyAfterDelete()
   873 
   874 	return EFalse;
   875 	}
   876 
   877 
   878 
   879 void CRichTextIndex::DeleteFromParagraph(TInt aPos,TInt aLength)
   880 // Special case delete for removing content from within a single paragraph only.
   881 // Not to be used for deleting an entire paragraph or paragraphs.
   882 // Returns EFalse indicating that no paragraphs were merged together,
   883 // as a result of the delete action.
   884 //
   885 	{
   886 	__TEST_INVARIANT;  // Do not need to RebalanceIndex(); part of defined behaviour for delete.
   887 
   888 	ScanToPosition(aPos,EScanToPositionAbsolute);
   889 
   890 #ifdef _DEBUG
   891 	{
   892 	TCurrentIndexRecords current;
   893 	GetCurrentRecords(current);
   894 	TInt startParaLength=current.iParaEntry->iLength;
   895 	TInt lengthRemainingInPara=startParaLength-iPos.iParaElementOffset;
   896 
   897 	if (aLength>=lengthRemainingInPara)
   898 	    {
   899 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DELETEFROMPARAGRAPH, "EDeleteFromParagraphInvalidRange" );
   900 	    }
   901 	__ASSERT_ALWAYS(aLength<lengthRemainingInPara,Panic(EDeleteFromParagraphInvalidRange));
   902 	}
   903 #endif
   904 
   905 	TIndexDeleteInfo info;
   906 	info.iDeleteLength=aLength;
   907 	info.iStartPara=iPos.iParaElement;
   908 	info.iEndPara=iPos.iParaElement;
   909 	info.iDeletePos=iPos;
   910 	info.iDeleteType=TIndexDeleteInfo::EDeleteFromParagraph;
   911 
   912 	DoDeleteFromParagraph(info);
   913 
   914 	__TEST_INVARIANT;
   915 	}
   916 
   917 
   918 TBool CRichTextIndex::DoDeleteFromParagraph(const TIndexDeleteInfo& aInfo)
   919 // Delete content from *within* the boundary of a single paragraph only.0
   920 // Returns EFalse indicating that no paragraphs were merged together,
   921 // as a result of the delete action.
   922 //
   923 	{
   924 	iPos=aInfo.iDeletePos;
   925 	TInt length=aInfo.iDeleteLength;
   926 	DeleteParagraphText(length);
   927 	TidyAfterDelete(aInfo);
   928 
   929 	return EFalse;
   930 	}
   931 
   932 
   933 TBool CRichTextIndex::DeleteNow(TIndexDeleteInfo& aInfo)
   934 // Deletes index data corresponding the info argument.
   935 // Returns ETrue is 2 paragraphs are merged as a result of the delete, otherwise returns false.
   936 //
   937 	{
   938 	iPos=aInfo.iDeletePos;
   939 	TCurrentIndexRecords current;
   940 	GetCurrentRecords(current);
   941 	TInt leftToDelete=aInfo.iDeleteLength;
   942 	TInt charsLeftInPara=(current.iParaEntry->iLength)-(iPos.iParaElementOffset);
   943 	TBool doParaMerge=((iPos.iPhraseElement>0 || iPos.iPhraseElementOffset>0) && leftToDelete>=charsLeftInPara);
   944 	// ETrue if the 1st para has content remaining but no paragraph delimiter.
   945 	//
   946 	TBool firstParaRemoved=(FirstPhraseOfParagraph() && iPos.iPhraseElementOffset==0 && aInfo.iDeleteLength>=current.iParaEntry->iLength);
   947 	// ETrue if the 1st para has been *wholly* deleted.
   948 	//
   949 	DeleteParagraphText(leftToDelete);  // Delete range will be in a minimum of 1 paragraph.
   950 	if (aInfo.iStartPara<aInfo.iEndPara)
   951 		{// The delete range crosses paragraph boundaries.
   952 		for (TInt currentPara=aInfo.iStartPara+1;currentPara<=aInfo.iEndPara;currentPara++)
   953 			{
   954 			ScanToPosition(aInfo.iDeletePos.iDocPos,EScanToPositionAbsolute);
   955 			DeleteParagraphText(leftToDelete);
   956 			}
   957 		}
   958 	// Now tidy up
   959 	if (doParaMerge && !firstParaRemoved)
   960 		{// Merge the 2 paras together.
   961 		TParaAttribsEntry* paraEntry=&(*iParaIx)[aInfo.iStartPara];
   962 		TParaAttribsEntry* paraEntryFollowing=&(*iParaIx)[aInfo.iStartPara+1];
   963 		paraEntryFollowing->iLength+=paraEntry->iLength;  // Extend length of remaining para.
   964 		paraEntryFollowing->iParaAttribs->iPhraseCount+=paraEntry->iParaAttribs->iPhraseCount;  // Extend phrase count
   965 		paraEntry->iParaAttribs->Release();
   966 		iParaIx->Delete(aInfo.iStartPara);
   967 		}
   968 	if (aInfo.iDeleteType!=TIndexDeleteInfo::EDeleteParagraph)
   969 		TidyAfterDelete(aInfo);
   970 
   971 	__TEST_INVARIANT;
   972 	return doParaMerge;
   973 	}
   974 
   975 
   976 void CRichTextIndex::TidyAfterDelete(const TIndexDeleteInfo& aInfo)
   977 //
   978 //
   979 	{
   980 	MergePhrases(aInfo.iDeletePos.iDocPos);  // Alters internal position record.
   981 	TCurrentIndexRecords current;
   982 	GetCurrentRecords(current);  // So must get records again.
   983 	if (!current.iParaAttribs->IsShared())
   984 		{// May be able to reclaim a share from this *specific* record
   985 		CParaAttribs* sharedParaAttribs=RequestShare(iPos);
   986 		if (sharedParaAttribs!=NULL && current.iParaAttribs!=sharedParaAttribs)
   987 			{// Use this shared record
   988 			current.iParaAttribs->Release();
   989 			RemoveFromPhraseIx(iPos.iPhraseElement);
   990 			current.iParaEntry->iParaAttribs=sharedParaAttribs;
   991 			}
   992 		}
   993 	}
   994 
   995 
   996 void CRichTextIndex::Normalize(TInt aPos)
   997 //
   998 	{
   999 	ScanToPosition(aPos,EScanToPositionAbsolute);
  1000 	NormalizeNow(iPos);
  1001 
  1002 	__TEST_INVARIANT;
  1003 	}
  1004 
  1005 
  1006 void CRichTextIndex::NormalizeNow(const TLogicalPosition& aNormalizePos)
  1007 //
  1008 	{
  1009 	CParaAttribs* currentParaAttribs=(*iParaIx)[aNormalizePos.iParaElement].iParaAttribs;
  1010 	if (!currentParaAttribs->IsShared())
  1011 		{
  1012 		CParaAttribs* sharedParaAttribs=RequestShare(iPos);
  1013 		if (sharedParaAttribs!=NULL && currentParaAttribs!=sharedParaAttribs)
  1014 			{// We must have been given a share on something already in the shared list.  Dump current stuff.
  1015 			currentParaAttribs->Release();
  1016 			(*iParaIx)[aNormalizePos.iParaElement].iParaAttribs=sharedParaAttribs;
  1017 			RemoveFromPhraseIx(aNormalizePos.iParaBasePhraseElement);
  1018 			}
  1019 		}
  1020 	}
  1021 
  1022 
  1023 CParaAttribs* CRichTextIndex::ReserveCellLC()
  1024 // Returns a handle to a newly created CParaAttribs object.  This may be used
  1025 // during a call to GetParaAttribsL() as a pre-allocated cell, thus ensuring
  1026 // that the call cannot possibly leave.
  1027 // ASSUMES: that the internal position record has been set correctly.
  1028 //
  1029 	{
  1030 	CParaAttribs* reservedCell=ReserveCellL();
  1031 	CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reservedCell));
  1032 	return reservedCell;
  1033 	}
  1034 
  1035 
  1036 CParaAttribs* CRichTextIndex::ReserveCellL()
  1037 // Returns a handle to a newly created CParaAttribs object.  This may be used
  1038 // during a call to GetParaAttribsL() as a pre-allocated cell, thus ensuring
  1039 // that the call cannot possibly leave.
  1040 // ASSUMES: that the internal position record has been set correctly.
  1041 //
  1042 	{
  1043 	TCurrentIndexRecords current;
  1044 	GetCurrentRecords(current);
  1045 	const CParaAttribs& paraAttribs=*current.iParaAttribs;
  1046 	CParaFormatLayer* paraLayer=paraAttribs.iParaFormat;
  1047 	CCharFormatLayer* charLayer=(paraAttribs.IsShared())
  1048 								? paraAttribs.iCharFormat
  1049 								: (*iPhraseIx)[iPos.iPhraseElement].CharFormat();
  1050 	CParaAttribs* reservedCell=CParaAttribs::NewL(paraLayer,charLayer);
  1051 	return reservedCell;
  1052 	}
  1053 
  1054 
  1055 TBool CRichTextIndex::DeleteParagraphText(TInt& aLength)
  1056 // Called once for each paragraph that's included in the delete range.
  1057 // Assumes the internal position record has already been set correctly.
  1058 // The delete range may cover: (1) The entire paragraph - so just destroy the paragraph,
  1059 // (2i) At least a portion of a single phrase, and possibly
  1060 // (2ii) 0..n whole contiguous phrases, and possibly
  1061 // (2iii) a trailing partial phrase.
  1062 // Returns ETrue if the full paragraph is deleted, otherwise returns EFalase.
  1063 //
  1064 	{
  1065 	TCurrentIndexRecords current; GetCurrentRecords(current);
  1066 	if (FirstPhraseOfParagraph() && iPos.iPhraseElementOffset==0 && aLength>=current.iParaEntry->iLength)
  1067 		{// The entire paragraph needs to be deleted.
  1068 		aLength-=current.iParaEntry->iLength;
  1069 		if (!current.iParaAttribs->IsShared())
  1070 			RemoveFromPhraseIx(iPos.iParaBasePhraseElement,current.iParaAttribs->iPhraseCount);
  1071 		current.iParaAttribs->Release();
  1072 		iParaIx->Delete(iPos.iParaElement);
  1073 		return ETrue;
  1074 		}
  1075 	TInt deleteFromPhrase=CurrentPhraseLength()-iPos.iPhraseElementOffset;
  1076 	TInt maxPhrase=current.iParaAttribs->PhraseCount();
  1077 	TInt currentPhrase=(iPos.iPhraseElement-iPos.iParaBasePhraseElement);
  1078 	while (currentPhrase<maxPhrase)
  1079 		{
  1080 		TInt deletable=Min(deleteFromPhrase,aLength);
  1081 		current.iParaEntry->iLength-=deletable;  // Adjust the paragraph length.
  1082 		if (current.iPhrase && deletable>=CurrentPhraseLength())
  1083 			{// Remove the now empty phrase from the phrase index.
  1084 			RemoveFromPhraseIx(iPos.iPhraseElement);
  1085 			current.iParaAttribs->iPhraseCount--;
  1086 			}
  1087 		else if (current.iPhrase)
  1088 			{// Adjust phrase length and move onto the next phrase.
  1089 			current.iPhrase->AdjustLength(-deletable);
  1090 			iPos.iPhraseElement++;
  1091 			iPos.iPhraseElementOffset=0;
  1092 			}
  1093 		currentPhrase++;
  1094 		aLength-=deletable;
  1095 		if(aLength==0)
  1096 			break;  // Nothing left to delete in this paragraph.
  1097 		// Get the data for the next phrase.
  1098 		GetCurrentRecords(current);
  1099 		deleteFromPhrase=CurrentPhraseLength();
  1100 		}
  1101 	return EFalse;
  1102 	}
  1103 
  1104 TBool CRichTextIndex::InsertCharFormatIsActive()
  1105 	{
  1106 	return iPendingNewPhrasePos != EInsertCharFormatReset;
  1107 	}
  1108 
  1109 /** Sets an *InsertPending* state, where format has been inserted into the
  1110 text, but no content has yet been inserted. This *state* is cancelled by cursor
  1111 movement etc. Split the current phrase at aPos (if necessary) and insert a zero
  1112 length phrase, ready to accept the pending content of the specified format.
  1113 */
  1114 void CRichTextIndex::SetInsertCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,TInt aPos)
  1115 	{
  1116 	if (InsertCharFormatIsActive() && aPos!=iPendingNewPhrasePos)
  1117 	    {
  1118 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_SETINSERTCHARFORMATL, "ESetInsertCharFormatIntegrityErr" );
  1119 	    }
  1120 	__ASSERT_ALWAYS(!InsertCharFormatIsActive() || aPos==iPendingNewPhrasePos,
  1121 					Panic(ESetInsertCharFormatIntegrityErr));
  1122 	if (InsertCharFormatIsActive())
  1123 		UpdateInsertCharFormatL(aFormat, aMask);
  1124 	else
  1125 		NewInsertCharFormatL(aFormat, aMask, aPos);
  1126 	}
  1127 
  1128 void CRichTextIndex::NewInsertCharFormatL(const TCharFormatX& aFormat,
  1129 	const TCharFormatXMask& aMask, TInt aPos)
  1130 	{
  1131 	if (InsertCharFormatIsActive())
  1132 	    {
  1133 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_NEWINSERTCHARFORMATL, "ESetInsertCharFormatIntegrityErr" );
  1134 	    }
  1135 	__ASSERT_ALWAYS(!InsertCharFormatIsActive(),
  1136 		Panic(ESetInsertCharFormatIntegrityErr));
  1137 	ScanToPosition(aPos,EScanToPositionMatchLeft);
  1138 	TCurrentIndexRecords current;
  1139 	GetCurrentRecords(current);
  1140 	CCharFormatLayer* basedOn;
  1141 	TCharFormatX applyFormat=aFormat;
  1142 	TCharFormatXMask applyMask=aMask;
  1143 	GetPhraseFormat(current,applyFormat,applyMask,basedOn);  // Inherit phrase attributes to the left, over what is present.
  1144 	TBool origParaAttribsShared=current.iParaAttribs->IsShared();
  1145 	if (origParaAttribsShared)
  1146 		{// Current paraAttribs is shared.
  1147 		iRollbackParaAttribsHandle=current.iParaAttribs;
  1148 		current.iParaEntry->iParaAttribs=RequestReclaimShareL(current.iParaAttribs,current.iParaEntry);  // Does not release share.
  1149 		ScanToPosition(aPos,EScanToPositionMatchLeft);  // Pick up reclaimed phrase.
  1150 		}  // Now current.iParaAttribs has specific character formatting - guaranteed.
  1151 	GetCurrentRecords(current);
  1152 	TRAPD(ret, DoNewInsertCharFormatL(applyFormat, applyMask,
  1153 		basedOn, current.iParaAttribs));
  1154 	if (ret!=KErrNone)
  1155 		{// Rollback as if this function call never happened.
  1156 		if (origParaAttribsShared)
  1157 			{// Revert back to sharing the original.
  1158 			current.iParaAttribs->Release();
  1159 			RemoveFromPhraseIx(iPos.iPhraseElement);
  1160 			current.iParaEntry->iParaAttribs=iRollbackParaAttribsHandle;
  1161 			}
  1162 		else
  1163 			{// Restore the original phrase index.
  1164 			if (PhraseSplit())
  1165 				MergePhrases(aPos);
  1166 			}
  1167 		OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_NEWINSERTCHARFORMATL, "LeaveNoMemory" );
  1168 		User::LeaveNoMemory();
  1169 		}
  1170 	iPendingNewPhrasePos=aPos;
  1171 	}
  1172 
  1173 void CRichTextIndex::UpdateInsertCharFormatL(const TCharFormatX& aFormat,
  1174 	const TCharFormatXMask& aMask)
  1175 	{
  1176 	CCharFormatLayer* currentLayer = GetCurrentInsertCharFormat();
  1177 	CCharFormatLayer* newLayer = CCharFormatLayer::NewCopyBaseL(currentLayer);
  1178 	CleanupStack::PushL(newLayer);
  1179 	newLayer->SetL(aFormat, aMask);
  1180 	currentLayer->Swap(*newLayer);
  1181 	CleanupStack::PopAndDestroy(newLayer);
  1182 	}
  1183 
  1184 CCharFormatLayer* CRichTextIndex::GetCurrentInsertCharFormat()
  1185 	{
  1186 	if (!InsertCharFormatIsActive())
  1187 	    {
  1188 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_GETCURRENTINSERTCHARFORMAT, "ESetInsertCharFormatIntegrityErr" );
  1189 	    }
  1190 	__ASSERT_DEBUG(InsertCharFormatIsActive(),
  1191 		Panic(ESetInsertCharFormatIntegrityErr));
  1192 	ScanToPosition(iPendingNewPhrasePos,EScanToPositionMatchLeft);
  1193 	TCurrentIndexRecords current;
  1194 	GetCurrentRecords(current);
  1195 	if ((*iPhraseIx)[iPos.iPhraseElement].Length() != 0)
  1196 	    {
  1197 	    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_GETCURRENTINSERTCHARFORMAT, "ESetInsertCharFormatIntegrityErr" );
  1198 	    }
  1199 	__ASSERT_DEBUG((*iPhraseIx)[iPos.iPhraseElement].Length() == 0,
  1200 		Panic(ESetInsertCharFormatIntegrityErr));
  1201 	return (*iPhraseIx)[iPos.iPhraseElement].CharFormat();
  1202 	}
  1203 
  1204 void CRichTextIndex::DoNewInsertCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,
  1205 											CCharFormatLayer* aBasedOn,CParaAttribs* aParaAttribs)
  1206 	{
  1207 	SplitPhraseL(iPos.iPhraseElement,iPos.iPhraseElementOffset,aParaAttribs);
  1208 	CCharFormatLayer* layer=CCharFormatLayer::NewL();
  1209 	layer->SetBase(aBasedOn);  // must be done before the SetL().
  1210 	CleanupStack::PushL(layer);
  1211 	layer->SetL(aFormat,aMask);
  1212 	RPhraseAttribsEntry pendingNewPhrase(layer);
  1213 	TInt pendingNewPhraseElement=(FirstPhraseOfParagraph() && iPos.iPhraseElementOffset==0)
  1214 		?iPos.iParaBasePhraseElement:iPos.iPhraseElement+1;
  1215 	iPhraseIx->InsertL(pendingNewPhraseElement,pendingNewPhrase);
  1216 	CleanupStack::Pop();
  1217 	aParaAttribs->iPhraseCount++;
  1218 	}
  1219 
  1220 
  1221 void CRichTextIndex::RebalanceIndex()
  1222 // Returns the index to a good state, by releasing the extra share taken on the paraAttribs
  1223 //
  1224 	{
  1225 	if (iRollbackParaAttribsHandle)
  1226 		{
  1227 		// ASSERT: The specified para attribs is indeed in the share list.
  1228 		if (!iRollbackParaAttribsHandle->IsShared())
  1229 		    {
  1230 		    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_REBALANCEINDEX, "EParaAttribsNotInSharedList" );
  1231 		    }
  1232 		__ASSERT_ALWAYS(iRollbackParaAttribsHandle->IsShared(),Panic(EParaAttribsNotInSharedList));
  1233 		iRollbackParaAttribsHandle->Release();
  1234 		iRollbackParaAttribsHandle=NULL;
  1235 		}
  1236 	}
  1237 
  1238 /** Cancels the transitory state where a specified character format is applied
  1239 on top of any inherited formatting. eg, when bold is on. Cancel when: (1) the
  1240 text position is altered. (2) the first character (or picture) has been
  1241 inserted following the setting of this state. If a zero length phrase is
  1242 removed OR has content entered into it, the newly abutting phrases are checked
  1243 to see if they can be merged. Then a request share of this para is issued.
  1244 */
  1245 void CRichTextIndex::CancelInsertCharFormat()
  1246 	{
  1247 	if (InsertCharFormatIsActive())
  1248 		{
  1249 		TBool isDeleted = DeleteInsertCharFormat();
  1250 		ConsolidateAt(iPendingNewPhrasePos, isDeleted?
  1251 			EPositionOnly : EFollowingPhrase);
  1252 		iPendingNewPhrasePos = EInsertCharFormatReset;
  1253 		}
  1254 	}
  1255 
  1256 /** Attempts to delete a zero-length phrase at the insert character format
  1257 position. Does not delete any phrase of non-zero length.
  1258 @pre The insert character format must be active
  1259 @return ETrue if a zero-length phrase was deleted.
  1260 */
  1261 TBool CRichTextIndex::DeleteInsertCharFormat()
  1262 	{
  1263 	if (!InsertCharFormatIsActive())
  1264 	    {
  1265 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DELETEINSERTCHARFORMAT, "Invariant" );
  1266 	    }
  1267 	__ASSERT_DEBUG(InsertCharFormatIsActive(), User::Invariant());
  1268 	ScanToPosition(iPendingNewPhrasePos,EScanToPositionMatchLeft);
  1269 	TCurrentIndexRecords current;
  1270 	GetCurrentRecords(current);
  1271 	if (current.iPhrase && current.iPhrase->Length() == 0)
  1272 		{
  1273 		RemoveFromPhraseIx(iPos.iPhraseElement);
  1274 		current.iParaAttribs->iPhraseCount--;  // Para has 1 less phrase in it now.
  1275 		return ETrue;
  1276 		}
  1277 	return EFalse;
  1278 	}
  1279 
  1280 /** Attempts to merge phrases and share paragraphs.
  1281 @param aPosition
  1282 	Phrase boundary here is merged if possible, paragraph here is shared if
  1283 	possible.
  1284 @param aPositionOrPhrase
  1285 	If EPositionOnly the phrases either side of aPosition are considered for merging. If EFollowingPhrase,
  1286 	the end of the phrase following aPosition is also considered.
  1287 */
  1288 void CRichTextIndex::ConsolidateAt(TInt aPosition,
  1289 	TPositionOrPhrase aPositionOrPhrase)
  1290 	{
  1291 	ScanToPosition(aPosition, EScanToPositionAbsolute);
  1292 	TCurrentIndexRecords current;
  1293 	GetCurrentRecords(current);
  1294 	if (!current.iPhrase)
  1295 		return;
  1296 
  1297 	TInt length = current.iPhrase->Length();
  1298 	MergePhrases(aPosition);
  1299 	if (aPositionOrPhrase == EFollowingPhrase)
  1300 		{
  1301 		ScanToPosition(aPosition, EScanToPositionAbsolute);
  1302 		GetCurrentRecords(current);
  1303 		if (current.iPhrase)
  1304 			MergePhrases(aPosition + length);
  1305 		}
  1306 	Normalize(aPosition);
  1307 	RebalanceIndex();
  1308 	}
  1309 
  1310 TBool CRichTextIndex::DelSetInsertCharFormatL(TInt aPos,TInt aLength)
  1311 // Delete aLength characters, commencing at, and including, aPos.
  1312 // Adds value by the following behaviour:
  1313 // If aPos is on a phrase boundary, then remember temporarily the phrase format.
  1314 // This is applied to any content that is immediately inserted.
  1315 //
  1316 	{
  1317 	__TEST_INVARIANT;
  1318 
  1319 	CancelInsertCharFormat();
  1320 	ScanToPosition(aPos,EScanToPositionAbsolute);
  1321 	TCurrentIndexRecords current; GetCurrentRecords(current);
  1322 	if ((!current.iParaAttribs->IsShared()) && iPos.iPhraseElementOffset==0)
  1323 		{// aPos is on phrase boundary so SetState.
  1324 		TCharFormatX format;
  1325 		TCharFormatXMask mask;
  1326 		CCharFormatLayer* charBase;
  1327 		GetPhraseFormat(current,format,mask,charBase);
  1328 		SetInsertCharFormatL(format,mask,aPos);
  1329 		}
  1330 	TIndexDeleteInfo deleteInfo;
  1331 	SetForDeleteL(deleteInfo,aPos,aLength);
  1332 	TBool parasMerged=DeleteNow(deleteInfo);
  1333 
  1334 	__TEST_INVARIANT;
  1335 	return parasMerged;
  1336 	}
  1337 
  1338 
  1339 void CRichTextIndex::ApplyParaFormatL(const CParaFormat* aFormat,const TParaFormatMask& aMask,TInt aPos,TInt aLength)
  1340 // Applies the specified format attributes to the paragraphs covering character position aPos to aPos+aLength-1.
  1341 // Preserves any attributes that are currently stored in this layer.
  1342 // If the specified para(s) is in the shared list, a new shared para of the desired format must be created,
  1343 // and the usage count of the original decremented.
  1344 //
  1345 	{
  1346 	__TEST_INVARIANT;
  1347 
  1348 	TInt offset=(aLength==0)?0 :-1;
  1349 	TInt endPara=OwningParagraph(aPos+(aLength+offset));
  1350 	TInt paraItem=OwningParagraph(aPos);
  1351 	TCurrentIndexRecords current; GetCurrentRecords(current);
  1352 	CParaFormat* pf=CParaFormat::NewL(*aFormat);  // preserve the desired tablist.
  1353 	CleanupStack::PushL(pf);
  1354 	for (;paraItem<=endPara;paraItem++)
  1355 		{// For each paragraph, apply the specified format.
  1356 		TParaFormatMask applyMask=aMask;
  1357 		CParaAttribs* currentParaAttribs=(*iParaIx)[paraItem].iParaAttribs;
  1358 		TBool shared=currentParaAttribs->IsShared();
  1359 		if (!shared)
  1360 			{
  1361 			currentParaAttribs->iParaFormat->SenseL(pf,applyMask);
  1362 			currentParaAttribs->iParaFormat->SetL(pf,applyMask);
  1363 			}
  1364 		else
  1365 			{// Must create a new shared para attribs of the specified format
  1366 			// Make a new para format layer
  1367 			currentParaAttribs->iParaFormat->SenseL(pf,applyMask);
  1368 			CParaFormatLayer* newParaLayer=CParaFormatLayer::NewL(pf,applyMask);
  1369 			newParaLayer->SetBase(currentParaAttribs->iParaFormat->SenseBase());
  1370 			CleanupStack::PushL(newParaLayer);
  1371 			// Make a new char format layer
  1372 			CCharFormatLayer* newCharLayer=CCharFormatLayer::NewL(currentParaAttribs->iCharFormat);
  1373 			newCharLayer->SetBase(currentParaAttribs->iCharFormat->SenseBase());
  1374 			CleanupStack::PushL(newCharLayer);
  1375 			//
  1376 			CParaAttribs* sharedParaAttribs=GetParaAttribsL(newParaLayer,newCharLayer);
  1377 			CleanupStack::PopAndDestroy(2);
  1378 			if (sharedParaAttribs)
  1379 				(*iParaIx)[paraItem].iParaAttribs=sharedParaAttribs;
  1380 			currentParaAttribs->Release();
  1381 			}
  1382 		}
  1383 	CleanupStack::PopAndDestroy();  // pf
  1384 	__TEST_INVARIANT;
  1385 	}
  1386 
  1387 
  1388 void CRichTextIndex::ApplyParagraphStyleL(const CParagraphStyle& aStyle,TInt aPos,TInt aLength,
  1389 										  const CCharFormatLayer* aCharStyleNormal,CParagraphStyle::TApplyParaStyleMode aMode)
  1390 // Applies the specified paragraph style to the paragraphs covering
  1391 // character positions aPos to aPos+aLength-1.
  1392 // Alters the specific formatting of the covered paragraphs as specified by aMode.
  1393 //
  1394 	{
  1395 	__TEST_INVARIANT;
  1396 
  1397 	TInt offset=(aLength==0)?0 :-1;
  1398 	TInt endPara=OwningParagraph(aPos+(aLength+offset));
  1399 	TInt paraItem=OwningParagraph(aPos);
  1400 	TInt paragraphBasePhrase=iPos.iParaBasePhraseElement;
  1401 	TCurrentIndexRecords current;
  1402 	GetCurrentRecords(current);
  1403 	for (;paraItem<=endPara;paraItem++)
  1404 		{// For each paragraph, apply the specified style
  1405 		CParaAttribs& currentParaAttribs=*(*iParaIx)[paraItem].iParaAttribs;
  1406 		TBool shared=currentParaAttribs.IsShared();
  1407 		TUid type=aStyle.Type();
  1408 		if (!shared)
  1409 			{
  1410 			currentParaAttribs.iParaFormat->SetBase(&aStyle);
  1411 			TInt phraseCount=currentParaAttribs.PhraseCount();
  1412 			for (TInt phraseItem=0;phraseItem<phraseCount;phraseItem++)
  1413 				{
  1414 				CCharFormatLayer& charLayer=*(*iPhraseIx)[paragraphBasePhrase+phraseItem].CharFormat();
  1415 				if (type==KNormalParagraphStyleUid)
  1416 					charLayer.SetBase(aCharStyleNormal);
  1417 				else
  1418 					charLayer.SetBase(aStyle.CharFormatLayer());
  1419 				ModifySpecificFormatting(*currentParaAttribs.iParaFormat,charLayer,aMode);
  1420 				}
  1421 			paragraphBasePhrase+=phraseCount;
  1422 			}
  1423 		else
  1424 			{// Must create a new shared para attribs of the same format, but a different based on link
  1425 			// Make a new para format layer
  1426 			CParaFormatLayer* newParaLayer=NULL;
  1427 			if (aMode==CParagraphStyle::ERetainNoSpecificFormats || aMode==CParagraphStyle::ERetainSpecificCharFormat)
  1428 				newParaLayer=CParaFormatLayer::NewL();
  1429 			else
  1430 				newParaLayer=CParaFormatLayer::NewL(currentParaAttribs.iParaFormat);
  1431 			newParaLayer->SetBase(&aStyle);
  1432 			CleanupStack::PushL(newParaLayer);
  1433 			//
  1434 			// Make a new char format layer
  1435 			CCharFormatLayer* newCharLayer=NULL;
  1436 			if (aMode==CParagraphStyle::ERetainNoSpecificFormats || aMode==CParagraphStyle::ERetainSpecificParaFormat)
  1437 				newCharLayer=CCharFormatLayer::NewL();
  1438 			else
  1439 				newCharLayer=CCharFormatLayer::NewL(currentParaAttribs.iCharFormat);
  1440 			if (type==KNormalParagraphStyleUid)
  1441 				newCharLayer->SetBase(aCharStyleNormal);
  1442 			else
  1443 				newCharLayer->SetBase(aStyle.CharFormatLayer());
  1444 			CleanupStack::PushL(newCharLayer);
  1445 			//
  1446 			(*iParaIx)[paraItem].iParaAttribs=GetParaAttribsL(newParaLayer,newCharLayer);
  1447 			CleanupStack::PopAndDestroy(2);
  1448 			currentParaAttribs.Release();
  1449 			}
  1450 		}
  1451 	}
  1452 
  1453 
  1454 void CRichTextIndex::ModifySpecificFormatting(CParaFormatLayer& aPl,CCharFormatLayer& aCl,CParagraphStyle::TApplyParaStyleMode aMode)
  1455 //
  1456 //
  1457 	{
  1458 	switch(aMode)
  1459 		{
  1460 		case(CParagraphStyle::ERetainNoSpecificFormats):
  1461 			aPl.Reset();
  1462 			aCl.Reset();
  1463 			break;
  1464 		case(CParagraphStyle::ERetainSpecificParaFormat):
  1465 			aCl.Reset();
  1466 			break;
  1467 		case(CParagraphStyle::ERetainSpecificCharFormat):
  1468 			aPl.Reset();
  1469 			break;
  1470 		case(CParagraphStyle::ERetainAllSpecificFormats):
  1471 		default:
  1472 			break;
  1473 		}
  1474 	}
  1475 
  1476 
  1477 /*
  1478 For every paragraph in the document: (i) if it uses the style aFrom, make it use the style aTo,
  1479 or the global style if aTo is null; (ii) if any phrase in the paragraph has a character format
  1480 based on the character format owned by aFrom, change it so that it is based on the character
  1481 format owned by aTo, or the global character format if aTo is null.
  1482 
  1483 The action described in (ii) should only occur for an 'orphaned' character insertion format; that is
  1484 an insertion format left after deletion of a block in a certain style that is itself then deleted.
  1485 */
  1486 void CRichTextIndex::NotifyStyleChangedL(const CParagraphStyle* aTo,const CParagraphStyle* aFrom,
  1487 										 const CParaFormatLayer& aGlobalParaFormatLayer,
  1488 										 const CCharFormatLayer& aGlobalCharFormatLayer)
  1489 	{
  1490 	__TEST_INVARIANT;
  1491 
  1492 	TInt paraCount=ParagraphCount();
  1493 	TInt currentPhrase = 0;
  1494 	const CCharFormatLayer* oldCharFormatLayer = aFrom->CharFormatLayer();
  1495 	const CParaFormatLayer* newParFormatLayer = aTo ? aTo : &aGlobalParaFormatLayer;
  1496 	const CCharFormatLayer* newCharFormatLayer = aTo ? aTo->CharFormatLayer() : &aGlobalCharFormatLayer;
  1497 	for (TInt paraItem = 0;paraItem < paraCount; paraItem++)
  1498 		{
  1499 		TParaAttribsEntry* currentPara = &(*iParaIx)[paraItem];
  1500 		CParaAttribs* currentParaAttribs = currentPara->iParaAttribs;
  1501 
  1502 		TBool changeParStyleBase = currentParaAttribs->iParaFormat->SenseBase() == aFrom;
  1503 		if (!currentParaAttribs->IsShared())
  1504 			{
  1505 			if (changeParStyleBase)
  1506 				currentParaAttribs->iParaFormat->SetBase(newParFormatLayer);
  1507 			TInt phraseCount = currentParaAttribs->PhraseCount();
  1508 			for (TInt phraseItem = currentPhrase; phraseItem < (currentPhrase + phraseCount); phraseItem++)
  1509 				{
  1510 				CCharFormatLayer* charFormat = (*iPhraseIx)[phraseItem].CharFormat();
  1511 				if (charFormat->SenseBase() == oldCharFormatLayer)
  1512 					charFormat->SetBase(newCharFormatLayer);
  1513 				}
  1514 			currentPhrase += phraseCount;
  1515 			}
  1516 		else
  1517 			{	// Maintain the shared list reference
  1518 			CParaAttribs* resultantParaAttribs = CParaAttribs::NewL(currentParaAttribs);
  1519 			CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,resultantParaAttribs));
  1520 			if (changeParStyleBase)
  1521 				resultantParaAttribs->iParaFormat->SetBase(newParFormatLayer);
  1522 			if (resultantParaAttribs->iCharFormat->SenseBase() == oldCharFormatLayer)
  1523 				resultantParaAttribs->iCharFormat->SetBase(newCharFormatLayer);
  1524 			CParaAttribs* shared = RequestShareL(resultantParaAttribs); // will return a non-NULL handle
  1525 			__ASSERT_DEBUG(shared,Panic(EDebug));
  1526 			currentParaAttribs->Release();
  1527 			iSharedParaQueHead.AddLast(*resultantParaAttribs); // allows correct release of cell.
  1528 			CleanupStack::PopAndDestroy();
  1529 			currentPara->iParaAttribs = shared;
  1530 			}
  1531 		}
  1532 
  1533 	__TEST_INVARIANT;
  1534 	}
  1535 
  1536 
  1537 const CParaFormatLayer* CRichTextIndex::ParagraphStyle(TBool& aStyleChangesOverRange,
  1538 															   TInt aPos,
  1539 															   TInt aLength)const
  1540 // Return the handle of the first paragraph style encountered in the specified range.
  1541 // Set aStyleChangesOverRange to ETrue, if different paragraph styles are encountered
  1542 // across the specified range, otherwise set it to EFalse.
  1543 //
  1544 	{
  1545 	__TEST_INVARIANT;
  1546 
  1547 	aStyleChangesOverRange=EFalse;
  1548 	CParaFormatLayer* style=NULL;
  1549 	TInt para=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos);
  1550 	TInt offset=(aLength==0)?0 :-1;
  1551 	TInt endPara=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos+(aLength+offset));
  1552 	style=(CParaFormatLayer*)(*iParaIx)[para].iParaAttribs->iParaFormat->SenseBase();
  1553 	++para;
  1554 	for (;para<=endPara;para++)
  1555 		{
  1556 		CParaFormatLayer* nextStyle=(CParaFormatLayer*)(*iParaIx)[para].iParaAttribs->iParaFormat->SenseBase();
  1557 		if (nextStyle!=style)
  1558 			aStyleChangesOverRange=ETrue;
  1559 		}
  1560 
  1561 	__TEST_INVARIANT;
  1562 	return style;
  1563 	}
  1564 
  1565 
  1566 void CRichTextIndex::SplitPhraseL(TInt aPhrase,TInt anOffset,RPhraseAttribsEntry& aPhraseAttribs,CParaAttribs& aParaAttribs)
  1567 	{
  1568 	if (anOffset<=0 || anOffset>=aPhraseAttribs.Length())
  1569 	    {
  1570 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SPLITPHRASEL, "Invariant" );
  1571 	    }
  1572 	__ASSERT_DEBUG(anOffset>0 && anOffset<aPhraseAttribs.Length(),User::Invariant());
  1573 //
  1574 	CCharFormatLayer* charLayer=CCharFormatLayer::NewCopyBaseL(aPhraseAttribs.CharFormat());
  1575 	CleanupStack::PushL(charLayer);
  1576 	iPhraseIx->InsertL(aPhrase+1,RPhraseAttribsEntry(charLayer,aPhraseAttribs.Length()-anOffset));
  1577 	CleanupStack::Pop();
  1578 	//
  1579 	// InsertL() has invalidated the phrase index.
  1580 	iPhraseIx->At(aPhrase).SetLength(anOffset);  // Adjust the length of the orginal phrase
  1581 	aParaAttribs.iPhraseCount++;
  1582 	}
  1583 
  1584 
  1585 TBool CRichTextIndex::MergePhrases(TInt aPhrase,RPhraseAttribsEntry& aPhraseAttribs,CParaAttribs& aParaAttribs)
  1586 	{
  1587 	RPhraseAttribsEntry& prevPhrase=iPhraseIx->At(aPhrase-1);
  1588 	if (!aPhraseAttribs.IsIdentical(prevPhrase))
  1589 		return EFalse;
  1590 	// Merge the abutting phrases together.
  1591 	prevPhrase.AdjustLength(aPhraseAttribs.Length());  //  Extend the remaining phrase
  1592 	RemoveFromPhraseIx(aPhrase);		// Free the resources taken by the redundant phrase
  1593 	aParaAttribs.iPhraseCount--;
  1594 	return ETrue;
  1595 	}
  1596 
  1597 
  1598 void CRichTextIndex::Share(TParaAttribsEntry& aParaEntry,TInt aPhrase)
  1599 //
  1600 // aParaEntry is not shared and can be (phrase count 1), aPhrase is the single phrase element
  1601 //
  1602 	{
  1603 	CParaAttribs* paraAttribs=aParaEntry.iParaAttribs;
  1604 	if (paraAttribs->iPhraseCount!=1)
  1605 	    {
  1606 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SHARE, "Invariant" );
  1607 	    }
  1608 	__ASSERT_DEBUG(paraAttribs->iPhraseCount==1,User::Invariant());
  1609 
  1610 	RPhraseAttribsEntry& phraseAttribs=iPhraseIx->At(aPhrase);
  1611 	if (phraseAttribs.IsPicturePhrase())
  1612 	    {
  1613 	    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_SHARE, "Invariant" );
  1614 	    }
  1615 	__ASSERT_DEBUG(!phraseAttribs.IsPicturePhrase(),User::Invariant());
  1616 
  1617 	CParaAttribs* share=GetParaAttribs(paraAttribs,*phraseAttribs.CharFormat());
  1618 	if (share!=paraAttribs)
  1619 		{	// re-use an existing share, so release the current attribs
  1620 		paraAttribs->Release();
  1621 		phraseAttribs.Discard();
  1622 		aParaEntry.iParaAttribs=share;
  1623 		}
  1624 	iPhraseIx->Delete(aPhrase);
  1625 	}
  1626 
  1627 
  1628 void CRichTextIndex::ApplyCharFormatCleanup(TAny* aPtr)
  1629 // CLeanup function for ApplyCharFormatL()
  1630 //
  1631 	{REINTERPRET_CAST(CRichTextIndex*,aPtr)->ApplyCharFormatRollback();}
  1632 
  1633 
  1634 void CRichTextIndex::ApplyCharFormatRollback()
  1635 // Paragraph and phrase we were working on are stored in iPos
  1636 // Return them the canonical form
  1637 //
  1638 	{
  1639 	TParaAttribsEntry& paraEntry=iParaIx->At(iPos.iParaElement);
  1640 	CParaAttribs* paraAttribs=paraEntry.iParaAttribs;
  1641 
  1642 	if (paraAttribs->IsShared())
  1643 		return;
  1644 
  1645 	TInt phrase=iPos.iPhraseElement;
  1646 	TInt base=iPos.iParaBasePhraseElement;
  1647 	if (phrase<base || phrase>=base+paraAttribs->iPhraseCount)
  1648 	    {
  1649 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPLYCHARFORMATROLLBACK, "Invariant" );
  1650 	    }
  1651 	__ASSERT_DEBUG(phrase>=base && phrase<base+paraAttribs->iPhraseCount,User::Invariant());
  1652 	if (phrase<base+paraAttribs->iPhraseCount-1)	// merge to the right
  1653 		MergePhrases(phrase+1,iPhraseIx->At(phrase+1),*paraAttribs);
  1654 	if (phrase>base)								// merge to the left
  1655 		MergePhrases(phrase,iPhraseIx->At(phrase),*paraAttribs);
  1656 	if (paraAttribs->iPhraseCount==1)				// Share the paragraph
  1657 		Share(paraEntry,base);
  1658 	}
  1659 
  1660 
  1661 void CRichTextIndex::ApplyCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,TInt aPos,TInt aLength,TBool aRemoveSpecific)
  1662 // Applies the specified character formatting to the characters contained within the range
  1663 // aPos to aPos+(aLength-1).
  1664 //
  1665 	{
  1666 	if (aLength<0)
  1667 	    {
  1668 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1669 	    }
  1670 	__ASSERT_DEBUG(aLength>=0,User::Invariant());
  1671 	__TEST_INVARIANT;
  1672 
  1673 	ScanToPosition(aPos,EScanToPositionAbsolute);
  1674 	TInt paraOffset=iPos.iParaElementOffset;
  1675 	TInt phraseOffset=iPos.iPhraseElementOffset;
  1676 	TInt phrase=iPos.iPhraseElement;
  1677 
  1678 // prepare for failure
  1679 	CleanupStack::PushL(TCleanupItem(ApplyCharFormatCleanup,this));
  1680 
  1681 	for (;;)
  1682 		{	// a paragraph at a time
  1683 		TParaAttribsEntry& paraEntry=iParaIx->At(iPos.iParaElement);
  1684 		CParaAttribs* paraAttribs=paraEntry.iParaAttribs;
  1685 		TInt charsToFormat=Min(aLength,paraEntry.iLength-paraOffset);
  1686 		aLength-=charsToFormat;
  1687 #ifdef _DEBUG
  1688 		aPos+=charsToFormat;
  1689 #endif
  1690 
  1691 // STEP 1. Reclaim any shared paragraph into non shared form. Re-use the object if possible
  1692 
  1693 		if (paraAttribs->IsShared())
  1694 			{
  1695 			CCharFormatLayer* charLayer=paraAttribs->iCharFormat;
  1696 			if (paraAttribs->iRefCount==CParaAttribs::EPrimeSharedCount)
  1697 				{	// we are the sole user of this attribute
  1698 				iPhraseIx->InsertL(phrase,RPhraseAttribsEntry(charLayer,paraEntry.iLength));
  1699 				// adjust attribute to be non-shared
  1700 				paraAttribs->link.Deque();
  1701 				paraAttribs->iRefCount=CParaAttribs::EPrimeNonSharedCount;
  1702 				paraAttribs->iPhraseCount=1;
  1703 				}
  1704 			else
  1705 				{	// create a new para attribs object
  1706 				CParaAttribs* newAttribs=CParaAttribs::NewL(paraAttribs->iParaFormat);
  1707 				CleanupReleasePushL(*newAttribs);
  1708 				charLayer=CCharFormatLayer::NewCopyBaseL(charLayer);
  1709 				CleanupStack::PushL(charLayer);
  1710 				iPhraseIx->InsertL(phrase,RPhraseAttribsEntry(charLayer,paraEntry.iLength));
  1711 				CleanupStack::Pop(2);		// charlayer, newAttribs
  1712 				paraAttribs->Release();		// lose a share on the old attribs
  1713 				paraEntry.iParaAttribs=paraAttribs=newAttribs;
  1714 				}
  1715 			phraseOffset=paraOffset; // we are now in the current position
  1716 			}
  1717 
  1718 // STEP 2.	Walk through all affected phrases in this paragraph
  1719 //			For each one, we may need to split it, and then apply the new format
  1720 
  1721 		do
  1722 			{
  1723 			if (phrase>=iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount)
  1724 			    {
  1725 			    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1726 			    }
  1727 			__ASSERT_DEBUG(phrase<iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount,User::Invariant());
  1728 //
  1729 			RPhraseAttribsEntry* phraseAttribs=&iPhraseIx->At(phrase);
  1730 			TInt len=phraseAttribs->Length();
  1731 
  1732 // STEP 2.1	Split the phrase at the beginning of the range?
  1733 
  1734 			if (phraseOffset>0)
  1735 				{		// can only happen for the first phrase
  1736 				/*
  1737 				 * The pointer paraAttribs is also stored in
  1738 				 * 'paraEntry.iParaAttribs'. The memory pointed to by this
  1739 				 * pointer will be released in CRichTextIndex's destructor.
  1740 				 */
  1741 				// coverity[leave_without_push]
  1742 				SplitPhraseL(phrase,phraseOffset,*phraseAttribs,*paraAttribs);	// inserts new phrase at correct position
  1743 				len-=phraseOffset;
  1744 				phraseOffset=0;
  1745 				iPos.iPhraseElement=++phrase;
  1746 				phraseAttribs=&iPhraseIx->At(phrase);
  1747 				}
  1748 
  1749 // STEP 2.2	Split the phrase at the end of the range?
  1750 
  1751 			if (len>charsToFormat)
  1752 				{	// phrase is longer than required format, so split it
  1753 				/*
  1754 				 * The pointer paraAttribs is also stored in
  1755 				 * 'paraEntry.iParaAttribs'. The memory pointed to by this
  1756 				 * pointer will be released in CRichTextIndex's destructor.
  1757 				 */
  1758 				// coverity[leave_without_push]
  1759 				SplitPhraseL(phrase,charsToFormat,*phraseAttribs,*paraAttribs);
  1760 				len=charsToFormat;
  1761 				phraseAttribs=&iPhraseIx->At(phrase);		// SplitPhraseL modifies the index array, we must do this!
  1762 				}
  1763 
  1764 			if (phraseAttribs->Length()!=len)
  1765 			    {
  1766 			    OstTrace0( TRACE_DUMP, DUP2_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1767 			    }
  1768 			__ASSERT_DEBUG(phraseAttribs->Length()==len,User::Invariant());
  1769 
  1770 // STEP 2.3	Change the format of the current phrase layer
  1771 
  1772 			TCharFormatX format=aFormat;
  1773 			TCharFormatXMask mask=aMask;
  1774 			CCharFormatLayer* charLayer=phraseAttribs->CharFormat();
  1775 			if (!aRemoveSpecific)
  1776 				charLayer->Sense(format,mask);  // preserve current specific character formatting
  1777 			charLayer->SetL(format,mask);
  1778 
  1779 // STEP 2.4	Check for merging with previous phrase
  1780 
  1781 			if (phrase==iPos.iParaBasePhraseElement || !MergePhrases(phrase,*phraseAttribs,*paraAttribs))
  1782 				// if we don't merge this phrase, move on to the next one
  1783 				iPos.iPhraseElement=++phrase;
  1784 
  1785 			charsToFormat-=len;
  1786 			} while (charsToFormat);
  1787 
  1788 		if (phrase!=iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount && aLength!=0)
  1789 		    {
  1790 		    OstTrace0( TRACE_DUMP, DUP3_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1791 		    }
  1792 		__ASSERT_DEBUG(phrase==iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount || aLength==0,User::Invariant());
  1793 
  1794 // STEP 3	Reduce the paragraph attributes back to canonical form
  1795 
  1796 // STEP 3.1	Check for merging at the end of the changes
  1797 
  1798 		if (phrase>iPos.iParaBasePhraseElement && phrase<iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount)
  1799 			MergePhrases(phrase,iPhraseIx->At(phrase),*paraAttribs);	// mustn't adjust phrase index to follow merge
  1800 
  1801 // STEP 3.2	See if we can re-share the paragraph
  1802 
  1803 		if (paraAttribs->iPhraseCount==1)
  1804 			{	// This para has constant character formatting - can be shared.
  1805 			iPos.iPhraseElement=--phrase;
  1806 			Share(paraEntry,phrase);
  1807 			}
  1808 
  1809 // loop into next paragraph
  1810 		if (aLength==0)
  1811 			break;
  1812 		iPos.iParaElement++;
  1813 		paraOffset=0;
  1814 		iPos.iParaBasePhraseElement=phrase;
  1815 #ifdef _DEBUG
  1816 		ScanToPosition(aPos,EScanToPositionAbsolute);
  1817 		if (iPos.iDocPos!=aPos)
  1818 		    {
  1819 		    OstTrace0( TRACE_DUMP, DUP4_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1820 		    }
  1821 		__ASSERT_DEBUG(iPos.iDocPos==aPos,User::Invariant());
  1822 		if (iPos.iPhraseElement!=phrase)
  1823 		    {
  1824 		    OstTrace0( TRACE_DUMP, DUP5_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1825 		    }
  1826 		__ASSERT_DEBUG(iPos.iPhraseElement==phrase,User::Invariant());
  1827 		if (iPos.iParaElementOffset!=paraOffset)
  1828 		    {
  1829 		    OstTrace0( TRACE_DUMP, DUP6_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1830 		    }
  1831 		__ASSERT_DEBUG(iPos.iParaElementOffset==paraOffset,User::Invariant());
  1832 		if (iPos.iPhraseElementOffset!=phraseOffset)
  1833 		    {
  1834 		    OstTrace0( TRACE_DUMP, DUP7_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
  1835 		    }
  1836 		__ASSERT_DEBUG(iPos.iPhraseElementOffset==phraseOffset,User::Invariant());
  1837 		__TEST_INVARIANT;
  1838 #endif
  1839 		}
  1840 
  1841 	CleanupStack::Pop();	// rollback item
  1842 
  1843 	__TEST_INVARIANT;
  1844 	}
  1845 
  1846 
  1847 void CRichTextIndex::RemoveSpecificParaFormatL(TInt aPos,TInt aLength)
  1848 // Removes all specific paragraph formatting from the specified region.
  1849 // For each paragraph covered by the range, check if its para attribs is in the
  1850 // shared list, or not.
  1851 // If its not shared, then simply reset the para format layer.
  1852 // If it is in the shared list, then a new shared para attribs must be created,
  1853 // and the reference count of the original decremented.
  1854 //
  1855 	{
  1856 	__TEST_INVARIANT;
  1857 
  1858 	TInt endPara=OwningParagraph(aPos+aLength);
  1859 	TInt currentPara=OwningParagraph(aPos);
  1860 	CParaAttribs* currentParaAttribs=NULL;
  1861 	while (currentPara<=endPara)
  1862 		{
  1863 		currentParaAttribs=(*iParaIx)[currentPara].iParaAttribs;
  1864 		if (!currentParaAttribs->IsShared())
  1865 			{// Reset specific paragraph format layer to be empty.
  1866 			currentParaAttribs->iParaFormat->Reset();  // remove specific formatting.
  1867 			}
  1868 		else
  1869 			{// Maintain the shared list reference
  1870 			CParaAttribs* resultantParaAttribs=CParaAttribs::NewL(currentParaAttribs);
  1871 			CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,resultantParaAttribs));
  1872 			resultantParaAttribs->iParaFormat->Reset();  // remove specific formatting
  1873 			CParaAttribs* shared=RequestShareL(resultantParaAttribs);  // Will return a non-NULL handle.
  1874 			__ASSERT_DEBUG(shared,Panic(EDebug));
  1875 			currentParaAttribs->Release();
  1876 			iSharedParaQueHead.AddLast(*resultantParaAttribs);  // Allows correct release of cell.
  1877 			CleanupStack::PopAndDestroy();
  1878 			(*iParaIx)[currentPara].iParaAttribs=shared;
  1879 			}
  1880 		currentPara++;
  1881 		}
  1882 
  1883 
  1884 	__TEST_INVARIANT;
  1885 	}
  1886 
  1887 
  1888 void CRichTextIndex::RemoveSpecificCharFormatL(TInt aPos,TInt aLength)
  1889 // Removes all specific character formatting from the specified region.
  1890 // For each paragraph covered by the range, check if its para attribs is in the
  1891 // shared list, or not.
  1892 // If its not shared, then simply reset the para format layer.
  1893 // If it is in the shared list, then a new shared para attribs must be created,
  1894 // and the reference count of the original decremented.
  1895 //
  1896 	{
  1897 	__TEST_INVARIANT;
  1898 
  1899 	TCharFormatX format;  // dummy format
  1900 	TCharFormatXMask mask;
  1901 	mask.ClearAll();
  1902 	ApplyCharFormatL(format,mask,aPos,aLength,ETrue);
  1903 
  1904 	__TEST_INVARIANT;
  1905 	}
  1906 
  1907 
  1908 TBool CRichTextIndex::MergePhrases(TInt aPos)
  1909 // Checks if the specified character position aPos is a phrase boundary.
  1910 // If so, then examines the two abutting phrases at aPos to see if they
  1911 // are identical;  in which case they are merged into one phrase that covers
  1912 // the sum of their lengths.
  1913 //
  1914 	{
  1915 	ScanToPosition(aPos,EScanToPositionAbsolute);
  1916 	return MergePhrases(iPos);
  1917 	}
  1918 
  1919 
  1920 TBool CRichTextIndex::MergePhrases(const TLogicalPosition& aPos)
  1921 // Checks if the specified character position record is a phrase boundary.
  1922 // If so, then examines the two abutting phrases at aPos to see if they
  1923 // are identical;  in which case they are merged into one phrase that covers
  1924 // the sum of their lengths.
  1925 //
  1926 	{
  1927 	TCurrentIndexRecords current;
  1928 	GetCurrentRecords(current);
  1929 	TBool phrasesMerged=EFalse;
  1930 	if (!FirstPhraseOfParagraph() && iPos.iPhraseElementOffset==0)
  1931 		{// Check if the 2 abutting phrases can be merged together.
  1932 		TInt rightPhraseElement=iPos.iPhraseElement;
  1933 		RPhraseAttribsEntry* rightPhrase=&(*iPhraseIx)[rightPhraseElement];
  1934 		RPhraseAttribsEntry* leftPhrase=&(*iPhraseIx)[rightPhraseElement-1];
  1935 		if (rightPhrase->IsIdentical(*leftPhrase))
  1936 			{// Merge the abutting phrases together.  Cannot merge picture/non-picture/z.l.p. phrase combinations.
  1937 			rightPhrase->AdjustLength(leftPhrase->Length());  //  Extend the right phrase length
  1938 			RemoveFromPhraseIx(rightPhraseElement-1);  // Free the resources taken by the left phrase - redundant
  1939 			(*iParaIx)[iPos.iParaElement].iParaAttribs->iPhraseCount--; // Update phrase count of owning CParaAttribs
  1940 			ScanToPosition(aPos.iDocPos,EScanToPositionAbsolute);  // Pick up new phrase index.
  1941 			phrasesMerged=ETrue;
  1942 			}
  1943 		}
  1944 	return phrasesMerged;
  1945 	}
  1946 
  1947 
  1948 /** Remove phrases from the containing object.  This includes
  1949 freeing referenced resources. (pictures etc.)
  1950 @param aPhraseIndex The first phrase to be deleted
  1951 @param aCount The number of phrases to be deleted
  1952 */
  1953 void CRichTextIndex::RemoveFromPhraseIx(TInt aPhraseIndex,TInt aCount)
  1954 	{
  1955  	// if the phrase being deleted is <= iLastPos  
  1956  	// then iLastPos will become invalid so should be reset
  1957  	if (aPhraseIndex <= iLastUsed.iPhraseElement )
  1958  		iLastUsed.Clear();
  1959  		
  1960 	for (TInt offset=aPhraseIndex;offset<(aPhraseIndex+aCount);offset++)
  1961 		{
  1962 		// discard phrases & book-keep the picture count
  1963 		RPhraseAttribsEntry& phrase=(*iPhraseIx)[offset];
  1964 		phrase.Discard();
  1965 		if (phrase.IsPicturePhrase())
  1966 			iPictureCount--;
  1967 		}
  1968 	iPhraseIx->Delete(aPhraseIndex,aCount);
  1969 	}
  1970 
  1971 void CRichTextIndex::GetParagraphFormatL(CParaFormat* aFormat,TInt aPos)const
  1972 // Fills aFormat with the effective Paragraph format attributes for the paragraph
  1973 // in which character position aPos is contained.
  1974 //
  1975 	{
  1976 	__TEST_INVARIANT;
  1977 
  1978 	TLogicalPosition cachePos(iLastUsed);
  1979 	TInt para=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos,&cachePos);
  1980 	(*iParaIx)[para].iParaAttribs->iParaFormat->SenseEffectiveL(aFormat);
  1981 	}
  1982 
  1983 void CRichTextIndex::GetSpecificParagraphFormatL(CParaFormat* aFormat,
  1984 												 TParaFormatMask& aMask,
  1985 												 TInt aPos)const
  1986 	{
  1987 	__TEST_INVARIANT;
  1988 
  1989 	TLogicalPosition cachePos(iLastUsed);
  1990 	TInt para=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos,&cachePos);
  1991 	CParaFormatLayer* pLayer = (*iParaIx)[para].iParaAttribs->iParaFormat;
  1992 	pLayer->SenseL(aFormat, aMask);
  1993 	}
  1994 
  1995 TInt CRichTextIndex::GetChars(TCharFormatX& aFormat,TInt aPos) const
  1996 // Returns the number of characters, commencing at aStartPos, that occupy the same phrase, and
  1997 // modifies aFormat, to hold the effective format of that phrase.
  1998 //
  1999 	{
  2000 	__TEST_INVARIANT;
  2001 
  2002 	CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
  2003 	TCurrentIndexRecords current;
  2004 	GetCurrentRecords(current);
  2005 	TInt phraseLength;
  2006 	CCharFormatLayer* charFormatLayer;
  2007 	if (current.iPhrase)
  2008 		{// Specific character formatting held in phrase index.
  2009 		charFormatLayer=current.iPhrase->CharFormat();
  2010 		phraseLength=(current.iPhrase->Length())-(iPos.iPhraseElementOffset);
  2011 		}
  2012 	else
  2013 		{// Constant character formatting held in the para attribs
  2014 		charFormatLayer=current.iParaAttribs->iCharFormat;
  2015 		phraseLength=current.iParaEntry->iLength-iPos.iParaElementOffset;
  2016 		}
  2017 	charFormatLayer->SenseEffective(aFormat);
  2018 	return phraseLength;
  2019 	}
  2020 
  2021 
  2022 TInt CRichTextIndex::GetPictureSizeInTwips(TSize& aSize,TInt aPos)const
  2023 // Get the size of the specified picture into aSize.  The picture is specified by its
  2024 // character position.  Return KErrNotFound if there is no picture at the specified
  2025 // document position.
  2026 //
  2027 	{
  2028 	__TEST_INVARIANT;
  2029 
  2030 	CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
  2031 	TCurrentIndexRecords current;
  2032 	GetCurrentRecords(current);
  2033 
  2034 	return (current.iPhrase)
  2035 		? current.iPhrase->GetPictureSizeInTwips(aSize)
  2036 		: KErrNotFound;
  2037 	}
  2038 
  2039 TPictureHeader* CRichTextIndex::PictureHeaderPtr(TInt aPos)
  2040 	{
  2041 	__TEST_INVARIANT;
  2042 
  2043 	CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute);
  2044 	TCurrentIndexRecords current;
  2045 	GetCurrentRecords(current);
  2046 	return current.iPhrase? current.iPhrase->PictureHeaderPtr() : 0;
  2047 	}
  2048 
  2049 TPictureHeader CRichTextIndex::PictureHeader(TInt aPos) const
  2050 // Return the picture header describing the picture at character position aPos.
  2051 // If there is no picture at character position aPos, a default picture header is returned.
  2052 //
  2053 	{
  2054 	const TPictureHeader* p = const_cast<CRichTextIndex*>(this)->PictureHeaderPtr(aPos);
  2055 	return p? *p : TPictureHeader();
  2056 	}
  2057 
  2058 
  2059 CPicture* CRichTextIndex::PictureHandleL(TInt aPos,MLayDoc::TForcePictureLoad aForceLoad)const
  2060 // Returns the handle of the concrete picture at character position aPos, if one exists;
  2061 // otherwise returns NULL.
  2062 //
  2063 	{
  2064 	__TEST_INVARIANT;
  2065 
  2066 	CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
  2067 	TCurrentIndexRecords current;
  2068 	GetCurrentRecords(current);
  2069 	return (current.iPhrase)
  2070 		? (CPicture*)current.iPhrase->PictureHandleL(iText.PictureFactory(),iText.StoreResolver(),aPos,aForceLoad)
  2071 		: NULL;
  2072 	}
  2073 
  2074 
  2075 void CRichTextIndex::GetParaFormatL(CParaFormat* aFormat,TParaFormatMask& aVaries,TInt aPos,TInt aLength,CParaFormat::TParaFormatGetMode aMode)const
  2076 // Senses the paragraph format of para(s) covered by the region aPos to aPos+aLength-1.
  2077 // aFormat takes the values of all attributes, and the mask aMask indicates those values that change
  2078 // over the selected region, and are therefore *indeterminate*.
  2079 // Application: seeding paragraph formatting dialogs.
  2080 //
  2081 	{
  2082 	__TEST_INVARIANT;
  2083 
  2084 	TInt para=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos);
  2085 	TInt offset=(aLength==0)?0 :-1;
  2086 	TInt endPara=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos+(aLength+offset));
  2087 	aVaries.ClearAll();
  2088 	(*iParaIx)[para].iParaAttribs->iParaFormat->SenseEffectiveL(aFormat,aMode);  // Sense 1st paras' format.
  2089 	++para;
  2090 	CParaFormat* format=CParaFormat::NewLC();
  2091 	for (;para<=endPara;para++)
  2092 		{
  2093 		(*iParaIx)[para].iParaAttribs->iParaFormat->SenseEffectiveL(format,aMode);
  2094 		if (format->iLanguage!=aFormat->iLanguage)
  2095 			aVaries.SetAttrib(EAttParaLanguage);
  2096 		if (format->iFillColor!=aFormat->iFillColor)
  2097 			aVaries.SetAttrib(EAttFillColor);
  2098 		if (format->iLeftMarginInTwips!=aFormat->iLeftMarginInTwips)
  2099 			aVaries.SetAttrib(EAttLeftMargin);
  2100 		if (format->iRightMarginInTwips!=aFormat->iRightMarginInTwips)
  2101 			aVaries.SetAttrib(EAttRightMargin);
  2102 		if (format->iIndentInTwips!=aFormat->iIndentInTwips)
  2103 			aVaries.SetAttrib(EAttIndent);
  2104 		if (format->iHorizontalAlignment!=aFormat->iHorizontalAlignment)
  2105 			aVaries.SetAttrib(EAttAlignment);
  2106 		if (format->iVerticalAlignment!=aFormat->iVerticalAlignment)
  2107 			aVaries.SetAttrib(EAttVerticalAlignment);
  2108 		if (format->iLineSpacingInTwips!=aFormat->iLineSpacingInTwips)
  2109 			aVaries.SetAttrib(EAttLineSpacing);
  2110 		if (format->iLineSpacingControl!=aFormat->iLineSpacingControl)
  2111 			aVaries.SetAttrib(EAttLineSpacingControl);
  2112 		if (format->iSpaceBeforeInTwips!=aFormat->iSpaceBeforeInTwips)
  2113 			aVaries.SetAttrib(EAttSpaceBefore);
  2114 		if (format->iSpaceAfterInTwips!=aFormat->iSpaceAfterInTwips)
  2115 			aVaries.SetAttrib(EAttSpaceAfter);
  2116 		if (format->iKeepTogether!=aFormat->iKeepTogether)
  2117 			aVaries.SetAttrib(EAttKeepTogether);
  2118 		if (format->iKeepWithNext!=aFormat->iKeepWithNext)
  2119 			aVaries.SetAttrib(EAttKeepWithNext);
  2120 		if (format->iWidowOrphan!=aFormat->iWidowOrphan)
  2121 			aVaries.SetAttrib(EAttWidowOrphan);
  2122 		if (format->iWrap!=aFormat->iWrap)
  2123 			aVaries.SetAttrib(EAttWrap);
  2124 		if (format->iBorderMarginInTwips!=aFormat->iBorderMarginInTwips)
  2125 			aVaries.SetAttrib(EAttBorderMargin);
  2126 		if (format->iDefaultTabWidthInTwips!=aFormat->iDefaultTabWidthInTwips)
  2127 			aVaries.SetAttrib(EAttDefaultTabWidth);
  2128 		if (aMode==CParaFormat::EAllAttributes)
  2129 			{
  2130 			// Borders
  2131 			for (TInt border=0;border<CParaFormat::EMaxParaBorder;border++)
  2132 				{// Check each para border individually.  Assumes border format attributes run consecutively.
  2133 				if (!format->IsBorderEqual((CParaFormat::TParaBorderSide)border,*aFormat))
  2134 					aVaries.SetAttrib((TTextFormatAttribute)(EAttTopBorder+border));
  2135 				}
  2136 			// Bullet
  2137 			if (!format->iBullet && !aFormat->iBullet)
  2138 				{ /* neither para has bullet, so no variation */ }
  2139 			else if (!format->iBullet || !aFormat->iBullet
  2140 				|| *format->iBullet!=*aFormat->iBullet)
  2141 				aVaries.SetAttrib(EAttBullet);
  2142 			// The Tab-List
  2143 			if (format->TabCount()!=aFormat->TabCount())
  2144 				aVaries.SetAttrib(EAttTabStop);  // TabLists are different.
  2145 			else
  2146 				{// The 2 tablists have the same number of tab stops - but are not necessarily the same.
  2147 				TBool matched=ETrue;
  2148 				TInt tabCount=format->TabCount();
  2149 				for (TInt tabItem=0;tabItem<tabCount;tabItem++)
  2150 					{// Compare the 2 tabs.
  2151 					TTabStop comp1,comp2;
  2152 					comp1=format->TabStop(tabItem);
  2153 					comp2=aFormat->TabStop(tabItem);
  2154 					if (comp1!=comp2)
  2155 						matched=EFalse;
  2156 					}
  2157 				if (!matched)
  2158 					aVaries.SetAttrib(EAttTabStop);
  2159 				}
  2160 			}
  2161 		}
  2162 	CleanupStack::PopAndDestroy();
  2163 	}
  2164 
  2165 
  2166 // Compare all attributes in two formats and where they differ set the appropriate flag in the aVaries mask.
  2167 void CRichTextIndex::CheckForUndetermined(const TCharFormatX& aFormatA,const TCharFormatX& aFormatB,
  2168 										  TCharFormatXMask& aVaries) const
  2169 	{
  2170 	const TCharFormat& a = aFormatA.iCharFormat;
  2171 	const TCharFormat& b = aFormatB.iCharFormat;
  2172 	if (a.iLanguage!=b.iLanguage)
  2173 		aVaries.SetAttrib(EAttCharLanguage);
  2174 	if (a.iFontPresentation.iTextColor!=b.iFontPresentation.iTextColor)
  2175 		aVaries.SetAttrib(EAttColor);
  2176 	if (a.iFontPresentation.iHighlightColor!=b.iFontPresentation.iHighlightColor)
  2177 		aVaries.SetAttrib(EAttFontHighlightColor);
  2178 	if (a.iFontPresentation.iHighlightStyle!=b.iFontPresentation.iHighlightStyle)
  2179 		aVaries.SetAttrib(EAttFontHighlightStyle);
  2180 	if (a.iFontPresentation.iStrikethrough!=b.iFontPresentation.iStrikethrough)
  2181 		aVaries.SetAttrib(EAttFontStrikethrough);
  2182 	if (a.iFontPresentation.iUnderline!=b.iFontPresentation.iUnderline)
  2183 		aVaries.SetAttrib(EAttFontUnderline);
  2184 	if (a.iFontPresentation.iHiddenText!=b.iFontPresentation.iHiddenText)
  2185 		aVaries.SetAttrib(EAttFontHiddenText);
  2186 	if (a.iFontPresentation.iPictureAlignment!=b.iFontPresentation.iPictureAlignment)
  2187 		aVaries.SetAttrib(EAttFontPictureAlignment);
  2188 	if (a.iFontSpec.iHeight!=b.iFontSpec.iHeight)
  2189 		aVaries.SetAttrib(EAttFontHeight);
  2190 	if (!(a.iFontSpec.iTypeface==b.iFontSpec.iTypeface))
  2191 		aVaries.SetAttrib(EAttFontTypeface);
  2192 	if (a.iFontSpec.iFontStyle.Posture()!=b.iFontSpec.iFontStyle.Posture())
  2193 		aVaries.SetAttrib(EAttFontPosture);
  2194 	if (a.iFontSpec.iFontStyle.StrokeWeight()!=b.iFontSpec.iFontStyle.StrokeWeight())
  2195 		aVaries.SetAttrib(EAttFontStrokeWeight);
  2196 	if (a.iFontSpec.iFontStyle.PrintPosition()!=b.iFontSpec.iFontStyle.PrintPosition())
  2197 		aVaries.SetAttrib(EAttFontPrintPos);
  2198 	if (aFormatA.iParserTag != aFormatB.iParserTag)
  2199 		aVaries.SetAttrib(EAttParserTag);
  2200 	}
  2201 
  2202 
  2203 void CRichTextIndex::GetCharFormat(TCharFormatX& aFormat,TCharFormatXMask& aVaries,TInt aPos,TInt aLength)const
  2204 // Senses the character formatting of the phrase(s) covered by the region aPos to aPos+aLength-1.
  2205 // aFormat takes the values of all character format attributes, and the mask aMask indicates those
  2206 // values that change over the selected region, and are therefore *indeterminate*.
  2207 // Application: seeding character formatting dialogs.
  2208 // If aLength is zero, the character format sensed is that of the charcter immediatley to the left (behind) the cursor.
  2209 //
  2210 	{
  2211 	__TEST_INVARIANT;
  2212 
  2213 	aVaries.ClearAll();
  2214 	if (aLength==0)  // Get the format of the character to the left of the cursor.
  2215 		((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionMatchLeft);
  2216 	else
  2217 		((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionAbsolute);
  2218 	// Get char format of first phrase
  2219 	TCurrentIndexRecords current;
  2220 	GetCurrentRecords(current);
  2221 	if (current.iPhrase)
  2222 		current.iPhrase->CharFormat()->SenseEffective(aFormat);
  2223 	else
  2224 		current.iParaAttribs->iCharFormat->SenseEffective(aFormat);
  2225 	// Place pos at start of next phrase
  2226 	TInt pos=aPos+(CurrentPhraseLength()-CurrentPhraseOffset());
  2227 	while (pos<=aPos+aLength-1)
  2228 		{// Get the format of the next phrase and check if attributes change value
  2229 		((CRichTextIndex*)this)->ScanToPosition(pos,EScanToPositionAbsolute);
  2230 		GetCurrentRecords(current);
  2231 		TCharFormatX format;
  2232 		if (current.iPhrase)
  2233 			current.iPhrase->CharFormat()->SenseEffective(format);
  2234 		else
  2235 			current.iParaAttribs->iCharFormat->SenseEffective(format);
  2236 		CheckForUndetermined(format,aFormat,aVaries);
  2237 		pos+=CurrentPhraseLength();
  2238 		}
  2239 	}
  2240 
  2241 
  2242 void CRichTextIndex::GetSpecificCharFormatDirection(TCharFormatX& aFormat,
  2243 												   TCharFormatXMask& aMask,
  2244 												   TInt aPos,
  2245 												   TBool aGetLeft) const
  2246 	{
  2247 	__TEST_INVARIANT;
  2248 
  2249 	aMask.ClearAll();
  2250 	((CRichTextIndex*)this)->ScanToPosition(aPos,
  2251 		aGetLeft? EScanToPositionMatchLeft : EScanToPositionAbsolute);
  2252 	TCurrentIndexRecords current;
  2253 	GetCurrentRecords(current);
  2254 	if (current.iPhrase)
  2255 		current.iPhrase->CharFormat()->Sense(aFormat,aMask);
  2256 	else
  2257 		current.iParaAttribs->iCharFormat->Sense(aFormat,aMask);
  2258 	}
  2259 
  2260 void CRichTextIndex::GetSpecificCharFormat(TCharFormatX& aFormat,TCharFormatXMask& aMask,TInt aPos)const
  2261 // Return the format attributes store in the specific layer only, for the specified document position.
  2262 // THIS IS NOT THE EFFECTIVE FORMAT, BUT THE SPECIFIC FORMATTING ONLY.
  2263 //
  2264 	{
  2265 	GetSpecificCharFormatDirection(aFormat, aMask, aPos, ETrue);
  2266 	}
  2267 
  2268 
  2269 CParaAttribs* CRichTextIndex::RequestReclaimShareL(CParaAttribs* aParaAttribs,TParaAttribsEntry* aParaEntry)
  2270 // If the specified para attribs is currently on the shared list, then
  2271 // a specific para attribs is *reclaimed* and returned.
  2272 // The current share on the CParaAttribs is not *Released*.
  2273 // A new CParaAttribs is created of the same paragraph format as the shared one,
  2274 // but with a reference count of zero, (as this now has specific character formatting),
  2275 // and 1 phrase that is of the same character format as the shared one.
  2276 // The reclaimed specific para attribs is attactched to the specified para entry.
  2277 // NOTE:
  2278 // If the specified CParaAttribs is not currently on the shared list, NULL
  2279 // is returned, and aParaAttribs left unchanged..
  2280 // Assumes a previous call to ScanToPosition has correctly set the internal position record.
  2281 // This function can be called safely in any situation. (Except that it may *LEAVE*).
  2282 //
  2283 	{
  2284 	if (!(aParaAttribs->IsShared()))
  2285 		return NULL;  // This para attribs is not currently shared.
  2286 	// We are dealing with a shared paraAttribs from now on.
  2287 	CParaAttribs* reclaimedPara=CParaAttribs::NewL(aParaAttribs->iParaFormat);  // Create the re-claimed paraAttribs.
  2288 	CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimedPara));
  2289 	CCharFormatLayer* newFormat=CCharFormatLayer::NewCopyBaseL(aParaAttribs->iCharFormat);
  2290 	CleanupStack::PushL(newFormat);
  2291 	RPhraseAttribsEntry phrase(newFormat,aParaEntry->iLength);
  2292 	// Now insert this phrase into the index.
  2293 	iPhraseIx->InsertL(iPos.iPhraseElement,phrase);
  2294 	CleanupStack::Pop(2);  // newFormat, reclaimedPara
  2295 	return reclaimedPara;
  2296 	}
  2297 
  2298 
  2299 CParaAttribs* CRichTextIndex::RequestShareL(CParaAttribs* aParaAttribs,CCharFormatLayer* aCharFormat,CParaAttribs* aReservedCell)
  2300 // Attempts to re-use an existing paraAttribs that matches the one specified.
  2301 // Returns the handle of a CParaAttribs, or NULL if the specified argument cannot
  2302 // be shared.
  2303 // aCharFormat may be NULL, as may aReservedCell.
  2304 //
  2305 	{
  2306 	if (aParaAttribs->iRefCount<=0 && aParaAttribs->iPhraseCount>1)
  2307 		return NULL;  // This para has specific character formatting & multiple phrases and so cannot be shared.
  2308 	// This para has constant character formatting - can be shared.
  2309 	CCharFormatLayer* charFormat;
  2310 	if (aCharFormat)
  2311 		charFormat=aCharFormat;  // Has a phrase index.
  2312 	else
  2313 		charFormat=aParaAttribs->iCharFormat;  // Has constant char formatting.
  2314 	return GetParaAttribsL(aParaAttribs->iParaFormat,charFormat,aReservedCell);
  2315 	}
  2316 
  2317 
  2318 CParaAttribs* CRichTextIndex::RequestShare(const TLogicalPosition& aLogicalPosition)
  2319 // Returns a handle to a paraAttribs on the shared list that matches the paragraph at the
  2320 // specified position.  Returns NULL if the specified paragraph does not have constant
  2321 // character formatting.
  2322 //
  2323 	{
  2324 	CParaAttribs* paraAttribs=(*iParaIx)[aLogicalPosition.iParaElement].iParaAttribs;
  2325 	if (paraAttribs->iRefCount<=0 && paraAttribs->iPhraseCount>1)
  2326 		return NULL;  // This para has specific character formatting and so cannot be shared.
  2327 	// This para has constant character formatting - can be shared.
  2328 	return GetParaAttribs(aLogicalPosition);
  2329 	}
  2330 
  2331 
  2332 CParaAttribs* CRichTextIndex::GetParaAttribsL(const CParaFormatLayer* aParaFormat,const CCharFormatLayer* aCharFormat,
  2333 											  CParaAttribs* aReservedCell)
  2334 // Attempts to match the specified arguments to a CParaAttribs in the shared list.
  2335 // If matched, the handle of the matched para attribs is returned, and the reference count
  2336 // of that item is incremented.
  2337 //
  2338 // aReservedCell is a pre-allocated CParaAttribs that has been correctly setup.  If no match
  2339 // is found, and the reserved cell is specified, this is used in preference to creating a new one.
  2340 // If the reserved cell is not specified and there is no match, a new CParaAttribs of the correct
  2341 // specification is created and added to the shared list.
  2342 //
  2343 	{
  2344 	CParaAttribs* handle=FindSharedParaAttribs(*aParaFormat,*aCharFormat);
  2345 	if (handle)
  2346 		return handle;  // match found already in the shared list.
  2347 // There is no match, so create new sharedPara and add to list.
  2348 	if (!aReservedCell)
  2349 		aReservedCell=CParaAttribs::NewL(aParaFormat,aCharFormat);  // Reusing aReservedCell saves an automatic.
  2350 	iSharedParaQueHead.AddLast(*aReservedCell);
  2351 	return aReservedCell;
  2352 	}
  2353 
  2354 
  2355 CParaAttribs* CRichTextIndex::GetParaAttribs(CParaAttribs* aParaAttribs,CCharFormatLayer& aCharFormatLayer)
  2356 // Called by Reset.
  2357 //
  2358 	{
  2359 	CParaAttribs* handle=FindSharedParaAttribs(*aParaAttribs->iParaFormat,aCharFormatLayer);
  2360 	if (handle)
  2361 		return handle;
  2362 	else
  2363 		{// No match, so piece together new shared paraAttribs and add to shared para list.
  2364 		aParaAttribs->iRefCount=1;
  2365 		aParaAttribs->iCharFormat=&aCharFormatLayer;
  2366 		iSharedParaQueHead.AddLast(*aParaAttribs);
  2367 		return aParaAttribs;
  2368 		}
  2369 	}
  2370 
  2371 
  2372 CParaAttribs* CRichTextIndex::GetParaAttribs(const TLogicalPosition& aLogicalPosition)
  2373 // Attempts to match the specified arguments to a CParaAttribs in the shared list.
  2374 // If matched, the handle of the matched para attribs is returned, and the reference count
  2375 // of that item is incremented. If no match is found, the current para attribs is
  2376 // transformed into one that is placed in the shared list.
  2377 //
  2378 	{
  2379 	CParaAttribs* sourceParaAttribs=(*iParaIx)[aLogicalPosition.iParaElement].iParaAttribs;
  2380 	RPhraseAttribsEntry* sourcePhrase=&(*iPhraseIx)[aLogicalPosition.iPhraseElement];
  2381 	//
  2382 	CParaAttribs* handle=FindSharedParaAttribs(*sourceParaAttribs->iParaFormat,*sourcePhrase->CharFormat());
  2383 	if (handle)
  2384 		return handle;
  2385 	else
  2386 		{// No match, so piece together new shared paraAttribs and add to shared para list
  2387 		sourceParaAttribs->iRefCount=1;
  2388 		if (sourcePhrase->IsPicturePhrase())
  2389 		    {
  2390 		    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_GETPARAATTRIBS, "EReleasCharFormatLayerOwnershipCalledOnPicturePhrase" );
  2391 		    }
  2392 		__ASSERT_ALWAYS(!sourcePhrase->IsPicturePhrase(),Panic(EReleasCharFormatLayerOwnershipCalledOnPicturePhrase));
  2393 		sourceParaAttribs->iCharFormat=sourcePhrase->ReleaseCharFormatLayerOwnership();
  2394 		sourcePhrase->Discard();
  2395 		iPhraseIx->Delete(aLogicalPosition.iPhraseElement);  // remove the deleted phrase from the phrase index.
  2396 		iSharedParaQueHead.AddLast(*sourceParaAttribs);
  2397 		return sourceParaAttribs;
  2398 		}
  2399 	}
  2400 
  2401 
  2402 CParaAttribs* CRichTextIndex::FindSharedParaAttribs(const CParaFormatLayer& aParaFormatLayer,const CCharFormatLayer& aCharFormatLayer)
  2403 // Attempts to match the specified arguments to an item in the shared para list.
  2404 // If found, the handle of the matched para attribs is returned, and the reference count
  2405 // of that item is incremented.
  2406 // If no match is made NULL is returned.
  2407 //
  2408 	{
  2409 	CParaAttribs* currentSharedPara=NULL;
  2410 	TBool matched=EFalse;
  2411 	if (!iSharedParaQueHead.IsEmpty())
  2412 		{
  2413 		TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
  2414 		while ((currentSharedPara=iterator++)!=NULL)
  2415 			{// Try and match each item in the shared para list.
  2416 			matched=aParaFormatLayer.IsIdentical(currentSharedPara->iParaFormat);
  2417 			if (!matched)
  2418 				continue;
  2419 			matched=aCharFormatLayer.IsIdentical(currentSharedPara->iCharFormat);
  2420 			if (!matched)
  2421 				continue;
  2422 			// We have a match, so adjust reference count.
  2423 			currentSharedPara->iRefCount++;
  2424 			return currentSharedPara;
  2425 			}
  2426 		}
  2427 	return currentSharedPara;
  2428 	}
  2429 
  2430 
  2431 void CRichTextIndex::ScanToPosition(TInt aCharPos,TScanToPositionMode aMode,TLogicalPosition* aLastUsed/*=NULL*/)
  2432 // Move the internal position record to that indicated by the specified character position, aCharPos.
  2433 // Behaviour follows:
  2434 // aCharPos is considered to be goverened by the phrase covering aCharPos-1,
  2435 // except when aCharPos is the fist character of a paragraph, when
  2436 // aCharPos is goverened by the phrase coverng aCharPos.
  2437 // (If nothing else, aCharPos will be a paragraph delimiter or the end-of-document
  2438 // character.
  2439 // The implementation below matches to the right as standard, then checks if a left
  2440 // phrase match is available.
  2441 //
  2442 	{
  2443 	if (!aLastUsed || aLastUsed->iDocPos-aLastUsed->iParaElementOffset>aCharPos)
  2444 		iPos.Clear();  // Reset the internal position record.
  2445 	else
  2446 		{
  2447 		iPos=*aLastUsed;
  2448 		if (iPos.iDocPos>aCharPos || (aMode==EScanToPositionMatchLeft && iPos.iDocPos==aCharPos))
  2449 			{// reset to the start of paragraph if aPos < cache pos or aPos==chache Pos whilst matching left
  2450 			iPos.iDocPos-=iPos.iParaElementOffset;
  2451 			iPos.iParaElementOffset=iPos.iPhraseElementOffset=0;
  2452 			iPos.iPhraseElement=iPos.iParaBasePhraseElement;
  2453 			}
  2454 		}
  2455 
  2456 	TInt phraseElement=iPos.iPhraseElement;
  2457 	TInt startOfPara=iPos.iDocPos-iPos.iParaElementOffset;
  2458 	TInt paraElement=iPos.iParaElement;
  2459 	const CArrayFix<TParaAttribsEntry>& paraIx=*iParaIx;
  2460 	const TParaAttribsEntry* para=&paraIx[paraElement];
  2461 	TInt len=para->iLength;
  2462 	if (aCharPos>=(startOfPara+len))
  2463 		{
  2464 		iPos.iParaElementOffset=iPos.iPhraseElementOffset=0;
  2465 		phraseElement=iPos.iParaBasePhraseElement;
  2466 		const TParaAttribsEntry* end=paraIx.End(paraElement);
  2467 		do
  2468 			{// Find the paragraph...
  2469 			startOfPara+=len;
  2470 			if (!(para->iParaAttribs->IsShared()))  // Adjust position within phrase index
  2471 				phraseElement+=para->iParaAttribs->iPhraseCount;
  2472 			++paraElement;
  2473 			if (++para==end)
  2474 				{
  2475 				para=&paraIx[paraElement];
  2476 				end=paraIx.End(paraElement);
  2477 				}
  2478 			len=para->iLength;
  2479 			}
  2480 				while (aCharPos>=(startOfPara+len) && (paraElement+1) < paraIx.Count());
  2481 		iPos.iParaBasePhraseElement=phraseElement;
  2482 		iPos.iParaElement=paraElement;
  2483 		}
  2484 	TInt startOfPhrase=iPos.iParaElementOffset-iPos.iPhraseElementOffset;
  2485 // the offset within the paragraph.
  2486 	TInt paraElementOffset=aCharPos-startOfPara;
  2487 	iPos.iParaElementOffset=paraElementOffset;
  2488 
  2489 	if (!(para->iParaAttribs->IsShared()))
  2490 		{// Find phrase & offset within it.
  2491 		TInt lastPhraseLength=-1;  // Record phrase length in case left match required.
  2492 		const CArrayFix<RPhraseAttribsEntry>& phraseIx=*iPhraseIx;
  2493 		const RPhraseAttribsEntry* phrase=&phraseIx[phraseElement];
  2494 		const RPhraseAttribsEntry* end=NULL;
  2495 		for (;;)
  2496 			{	// Find the phrase in the paragraph...
  2497 			len=phrase->Length();
  2498 			if (paraElementOffset<(startOfPhrase+len))
  2499 				break;
  2500 			startOfPhrase+=len;
  2501 			lastPhraseLength=len;
  2502 			if (end==NULL)
  2503 				end=phraseIx.End(phraseElement);
  2504 			phraseElement++;
  2505 			if (++phrase<end)
  2506 				continue;
  2507 			phrase=&phraseIx[phraseElement];
  2508 			end=phraseIx.End(phraseElement);
  2509 			}//...and the offset within this.
  2510 		// Check now for match left.
  2511 		if ((aMode==EScanToPositionMatchLeft) && lastPhraseLength>=0 && paraElementOffset==startOfPhrase)
  2512 			{// Match to the left most phrase if at the start of a phrase.
  2513 			phraseElement--;
  2514 			iPos.iPhraseElementOffset=lastPhraseLength;
  2515 			}
  2516 		else
  2517 			iPos.iPhraseElementOffset=paraElementOffset-startOfPhrase;
  2518 		}
  2519 	else
  2520 		{
  2521 		if (iPos.iParaBasePhraseElement!=phraseElement)
  2522 		    {
  2523 		    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SCANTOPOSITION, "EDebug" );
  2524 		    }
  2525 		__ASSERT_DEBUG(iPos.iParaBasePhraseElement==phraseElement,Panic(EDebug));
  2526 		}
  2527 	iPos.iPhraseElement=phraseElement;
  2528 	iPos.iDocPos=aCharPos;
  2529 	if (aLastUsed)
  2530 		*aLastUsed=iPos;
  2531 	}
  2532 
  2533 
  2534 TBool CRichTextIndex::FirstPhraseOfParagraph()const
  2535 // Interogates the current internal position record.
  2536 // Return ETrue if the current phrase element is the first phrase
  2537 // of specific character format in the current paragraph;
  2538 // Otherwise return EFalse.
  2539 //
  2540 	{return iPos.iPhraseElement==iPos.iParaBasePhraseElement;}
  2541 
  2542 
  2543 TInt CRichTextIndex::CurrentPhraseLength()const
  2544 // Return the length of the current phrase, where the current
  2545 // phrase is specified by the state of the internal position record.
  2546 //
  2547 	{
  2548 	if ((*iParaIx)[iPos.iParaElement].iParaAttribs->IsShared())
  2549 		return (*iParaIx)[iPos.iParaElement].iLength;
  2550 	else
  2551 		return (*iPhraseIx)[iPos.iPhraseElement].Length();
  2552 	}
  2553 
  2554 
  2555 TInt CRichTextIndex::CurrentPhraseOffset()const
  2556 // Returns the offset within the current phrase, where the current
  2557 // phrase is specified by the state of the internal position record.
  2558 //
  2559 	{
  2560 	if ((*iParaIx)[iPos.iParaElement].iParaAttribs->IsShared())
  2561 		return iPos.iParaElementOffset;  // only 1 phrase in para.
  2562 	else
  2563 		return iPos.iPhraseElementOffset;
  2564 	}
  2565 
  2566 
  2567 void CRichTextIndex::GetCurrentRecords(TCurrentIndexRecords& aRecord)const
  2568 // Package the phrase and paragraph index records that apply to the
  2569 // current paragraph and return this package.  It is assumed that
  2570 // the caller has already set the internal position record to a valid state.
  2571 //
  2572 	{
  2573 	aRecord.iParaEntry=&(*iParaIx)[iPos.iParaElement];
  2574 	aRecord.iParaAttribs=aRecord.iParaEntry->iParaAttribs;
  2575 	if (aRecord.iParaAttribs->IsShared())
  2576 		aRecord.iPhrase=NULL;
  2577 	else
  2578 		aRecord.iPhrase=&((*iPhraseIx)[iPos.iPhraseElement]);
  2579 	}
  2580 
  2581 
  2582 void CRichTextIndex::GetPhraseFormat(TCurrentIndexRecords& aCurrent,TCharFormatX& aFormat,TCharFormatXMask& aMask,
  2583 									 CCharFormatLayer*& aCharBase)const
  2584 // Fills aFormat and aMask with the character formatting information of the current record.
  2585 // aCharBase is set to the basedOn link if present.
  2586 // Encapsulates the concepts of the specific phrase index, and the constant character format.
  2587 // Only senses the format in the layer, does *NOT* perform a SenseEffective.
  2588 //
  2589 	{
  2590 	CCharFormatLayer* charFormatLayer=NULL;
  2591 	if (aCurrent.iPhrase)
  2592 		{// Specific character formatting held by phrase index.
  2593 		charFormatLayer=aCurrent.iPhrase->CharFormat();
  2594 		}
  2595 	else
  2596 		{// Constant character formatting held in the para attribs
  2597 		charFormatLayer=aCurrent.iParaAttribs->iCharFormat;
  2598 		}
  2599 	charFormatLayer->Sense(aFormat,aMask);
  2600 	aCharBase=(CCharFormatLayer*)(charFormatLayer->SenseBase());
  2601 	}
  2602 
  2603 
  2604 TInt CRichTextIndex::OwningParagraph(TInt aPos,TLogicalPosition* aLastUsed/*=NULL*/)const
  2605 // Return the paragraph element number that contains character position aPos.
  2606 // Assumes the caller has validated aPos.  Alters the internal record position.
  2607 //
  2608 	{
  2609 	((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionMatchLeft,aLastUsed);
  2610 	return iPos.iParaElement;
  2611 	}
  2612 
  2613 
  2614 void CRichTextIndex::SplitPhraseL(TInt aSplitPos)
  2615 // Splits the phrase at the offset aSplitPos, creating a new phrase
  2616 // which is filled with the split part of the current phrase, includig aSplitPos.
  2617 // The character format applied to the new phrase is the format of the phrase from which it has been split.
  2618 // The resulting new phrase is inserted into the phrase index immediately following the
  2619 // current element.  If aSplitPos is already at a phrase boundary, then no split is performed.
  2620 // (This means that a picture phrase in effect can never be split).
  2621 //
  2622 	{
  2623 	SetPhraseSplit(EFalse);
  2624 	ScanToPosition(aSplitPos,EScanToPositionAbsolute);
  2625 	if (iPos.iPhraseElementOffset==0)
  2626 		return;  // aSplitPos on a phrase boundary; urgo no split.
  2627 	TCurrentIndexRecords current; GetCurrentRecords(current);
  2628 // ASSERT: This function set can only be called on CParaAttribs that specific char format.
  2629 	if (current.iPhrase==NULL)
  2630 	    {
  2631 	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_SPLITPHRASEL, "ESplitPhraseCalledOnSharedPara" );
  2632 	    }
  2633 	__ASSERT_ALWAYS(current.iPhrase!=NULL,Panic(ESplitPhraseCalledOnSharedPara));
  2634 	DoSplitPhraseL(*current.iPhrase,iPos.iPhraseElementOffset,current.iParaAttribs);
  2635 	}
  2636 
  2637 
  2638 void CRichTextIndex::SplitPhraseL(TInt aPhraseElement,TInt aPhraseOffset,CParaAttribs* aParaAttribs)
  2639 // Splits the specified phrase at the offset aOffsetInPhrase, creating a new phrase
  2640 // which is filled with the split part of the current phrase, includig the split pos.
  2641 // The character format applied to the new phrase is the format of the phrase from which it has been split.
  2642 // The resulting new phrase is inserted into the phrase index immediately following the
  2643 // current element.  If the split pos is already at a phrase boundary, then no split is performed.
  2644 // (This means that a picture phrase in effect can never be split).
  2645 //
  2646 	{
  2647 	SetPhraseSplit(EFalse);
  2648 	RPhraseAttribsEntry& phrase=(*iPhraseIx)[aPhraseElement];
  2649 	if ((aPhraseOffset>0) && (aPhraseOffset<phrase.Length()))
  2650 		{// Not at a phrase boundary so split the current phrase.
  2651 		DoSplitPhraseL(phrase,aPhraseOffset,aParaAttribs);
  2652 		}
  2653 	}
  2654 
  2655 
  2656 void CRichTextIndex::DoSplitPhraseL(RPhraseAttribsEntry& aCurrentPhrase,TInt aPhraseOffset,CParaAttribs* aParaAttribs)
  2657 // Splits the specified phrase, creating a new phrase of the same character format.
  2658 // This new phrase is inserted into the phrase index immediately following the current one.
  2659 //
  2660 	{
  2661 // ASSERT: Cannot split a picture phrase.
  2662 	if (aCurrentPhrase.IsPicturePhrase())
  2663 	    {
  2664 	    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DOSPLITPHRASEL, "ESplitPhraseCalledOnPicturePhrase" );
  2665 	    }
  2666 	__ASSERT_DEBUG(!aCurrentPhrase.IsPicturePhrase(),Panic(ESplitPhraseCalledOnPicturePhrase));
  2667 	CCharFormatLayer* layer=CCharFormatLayer::NewCopyBaseL(aCurrentPhrase.CharFormat());
  2668 	CleanupStack::PushL(layer);
  2669 	RPhraseAttribsEntry newPhrase(layer,aCurrentPhrase.Length()-aPhraseOffset);
  2670 	iPhraseIx->InsertL(iPos.iPhraseElement+1,newPhrase);
  2671 	CleanupStack::Pop();
  2672 	//
  2673 	// InsertL() has invalidated the current internal position record.
  2674 	iPhraseIx->At(iPos.iPhraseElement).SetLength(aPhraseOffset);
  2675 	SetPhraseSplit(ETrue);
  2676 	aParaAttribs->iPhraseCount++;
  2677 	}
  2678 
  2679 
  2680 TBool CRichTextIndex::HasMarkupData(const CFormatLayer* aGlobalParaFormatLayer)const
  2681 // Returns ETure if this rich text instance has any specific markup,
  2682 // otherwise returns EFalse.
  2683 // The presence of specific markup is indicated by the following...
  2684 // 1) Style list is present (if style table is owned by the rich text)
  2685 // 2) Any phrase index content
  2686 // 3) >1 shared para attribs
  2687 //   or
  2688 // 3) 1 shared para attribs that has specific markup
  2689 // 4) any paragraph is based on a style other than normal (the global paraformatlayer)
  2690 //
  2691 	{
  2692 	TInt phraseCount=PhraseCount();
  2693 	if (phraseCount>0)
  2694 		return ETrue;
  2695 	//
  2696 	TInt sharedParaCount=SharedParaCount(this);
  2697 	if (sharedParaCount<1 && (sharedParaCount!=0 || phraseCount<=0))
  2698 	    {
  2699 	    OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_HASMARKUPDATA, "ERichTextIndexIntegrityErr" );
  2700 	    }
  2701 	__ASSERT_ALWAYS(sharedParaCount>=1 || (sharedParaCount==0 && phraseCount>0),Panic(ERichTextIndexIntegrityErr));
  2702 	if (sharedParaCount>1)
  2703 		return ETrue;
  2704 	const CParaAttribs* paraAttribs=iSharedParaQueHead.First();
  2705 	if (!paraAttribs->iParaFormat->IsEmpty())
  2706 		return ETrue;
  2707 	if (!paraAttribs->iCharFormat->IsEmpty())
  2708 		return ETrue;
  2709 	if (paraAttribs->iParaFormat->SenseBase()!=aGlobalParaFormatLayer)
  2710 		return ETrue;
  2711 	return EFalse;
  2712 	}
  2713 
  2714 
  2715 TInt CRichTextIndex::SharedParaCount(const CRichTextIndex* aSource)const
  2716 // Return a count of the number of shared paragraph formats present
  2717 // in the specified object.
  2718 //
  2719 	{
  2720 	TInt sharedParaCount = 0;
  2721 	TDblQueIter<CParaAttribs> iterator( MUTABLE_CAST(TDblQue<CParaAttribs>&, aSource->iSharedParaQueHead) );
  2722 	while ( iterator++ != NULL )
  2723 		sharedParaCount++;
  2724 	return sharedParaCount;
  2725 	}
  2726 
  2727 
  2728 void CRichTextIndex::AppendTakingSolePictureOwnershipL(const CRichTextIndex* aSource,const TGlobalLayerInfoAppend& aGlobalLayerInfo)
  2729 // No paragraph style information is appended.
  2730 //
  2731 	{
  2732 	CancelInsertCharFormat();
  2733 	CONST_CAST(CRichTextIndex*,aSource)->CancelInsertCharFormat();
  2734 
  2735 	TInt origParaCount=ParagraphCount();
  2736 	TInt origPhraseCount=iPhraseIx->Count();
  2737 	TRAPD(ret,
  2738 		AppendParaIndexL(aSource,aGlobalLayerInfo);
  2739 		AppendPhraseIndexL(aSource,aGlobalLayerInfo);
  2740 		);
  2741 	if (ret!=KErrNone)
  2742 		{
  2743 		RemoveFromPhraseIx(origPhraseCount,iPhraseIx->Count()-origPhraseCount);	// remove any added phrases etc.
  2744 		RbRemoveInsertedParaAttribsEntries(origParaCount,ParagraphCount()-origParaCount);	// remove any added paragraphs etc.
  2745 		NormalizeSharedList();		// remove any added shared paragraph attributes
  2746 		User::Leave(ret);
  2747 		}
  2748 
  2749 	__TEST_INVARIANT;
  2750 	}
  2751 
  2752 
  2753 void CRichTextIndex::AppendParaIndexL(const CRichTextIndex* aSource,const TGlobalLayerInfoAppend& aGlobalLayerInfo)
  2754 //
  2755 	{
  2756 	CRichTextStoreMap<CParaAttribs>* map=CRichTextStoreMap<CParaAttribs>::NewLC(SharedParaCount(aSource));
  2757 
  2758 	AppendSharedFormatsL(*map,aSource,aGlobalLayerInfo);
  2759 
  2760 	// Extend para index by required amount
  2761 	TInt originalParaCount=iParaIx->Count();
  2762 	TInt requiredParaCount=aSource->iParaIx->Count();
  2763 	iParaIx->AppendL(TParaAttribsEntry(),requiredParaCount);
  2764 
  2765 	for (TInt ii=0;ii<requiredParaCount;ii++)
  2766 		{// Copy the paragraph data for each of the appended paragraphs.
  2767 		const TParaAttribsEntry& sParaEntry=(*aSource->iParaIx)[ii];
  2768 		const CParaAttribs* sParaAttribs=sParaEntry.iParaAttribs;
  2769 		CParaAttribs* tParaAttribs;
  2770 		if (sParaAttribs->IsShared())
  2771 			{
  2772 			tParaAttribs=map->Item(sParaAttribs);
  2773 			if (tParaAttribs==NULL)
  2774 			    {
  2775 			    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPENDPARAINDEXL, "ESharedFormatsMapIntegrityError" );
  2776 			    }
  2777 			__ASSERT_DEBUG(tParaAttribs!=NULL,Panic(ESharedFormatsMapIntegrityError));
  2778 			tParaAttribs->iRefCount++;
  2779 			}
  2780 		else
  2781 			{// Have to build up the specific para attribs
  2782 			tParaAttribs=CParaAttribs::NewL(sParaAttribs->iParaFormat);  // sets iRefCount=0, copies the format layer
  2783 			tParaAttribs->iParaFormat->SetBase(aGlobalLayerInfo.iAggParaFormatLayer);
  2784 			tParaAttribs->iPhraseCount=sParaAttribs->iPhraseCount;
  2785 			}
  2786 		TParaAttribsEntry& tParaEntry=(*iParaIx)[originalParaCount+ii];
  2787 		tParaEntry.iLength=sParaEntry.iLength;
  2788 		tParaEntry.iParaAttribs=tParaAttribs;
  2789 		
  2790 		// tParaAttribs is attached to CRichTextIndex::iParaIx, and will be
  2791 		// released in destructor CRichTextIndex::~CRichTextIndex().
  2792 		// To prevent Coverity from reporting defect, add a comment:
  2793 		// coverity[memory_leak]
  2794 		}
  2795 
  2796 	CleanupStack::PopAndDestroy();  // map
  2797 	}
  2798 
  2799 
  2800 void CRichTextIndex::AppendSharedFormatsL(CParaAttribsMap& aMap,const CRichTextIndex* aSource,
  2801 											const TGlobalLayerInfoAppend& aGlobalLayerInfo)
  2802 // A map is kept, that for each original format specifies the corresponding new one that appended
  2803 // paragraphs should use.
  2804 //
  2805 	{
  2806 	TDblQueIter<CParaAttribs> iterator(MUTABLE_CAST(TDblQue<CParaAttribs>&,aSource->iSharedParaQueHead));
  2807 	CParaAttribs* currentSharedPara;
  2808 	while ((currentSharedPara=iterator++)!=NULL)
  2809 		{
  2810 		if (!currentSharedPara->IsShared())
  2811 		    {
  2812 		    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPENDSHAREDFORMATSL, "Invariant" );
  2813 		    }
  2814 		__ASSERT_DEBUG(currentSharedPara->IsShared(),User::Invariant());
  2815 
  2816 		CParaFormatLayer* sPl=currentSharedPara->iParaFormat;
  2817 		CCharFormatLayer* sCl=currentSharedPara->iCharFormat;
  2818 		sPl->SetBase(aGlobalLayerInfo.iAggParaFormatLayer);  // alter the original so that the following GetParaAttribsL() call will
  2819 		sCl->SetBase(aGlobalLayerInfo.iAggCharFormatLayer);  // match based on our global format layers, not those of aSource's.
  2820 		CParaAttribs* newParaAttribs=FindSharedParaAttribs(*sPl,*sCl);
  2821 		sPl->SetBase(aGlobalLayerInfo.iComParaFormatLayer);  // set the global format layers back again, cos we don't want to
  2822 		sCl->SetBase(aGlobalLayerInfo.iComCharFormatLayer);  // corrupt aSource.
  2823 		if (newParaAttribs==NULL)
  2824 			{
  2825 			newParaAttribs=CParaAttribs::NewL(currentSharedPara);
  2826 			// change the based-on links (they are copied in the construction)
  2827 			newParaAttribs->iParaFormat->SetBase(aGlobalLayerInfo.iAggParaFormatLayer);
  2828 			newParaAttribs->iCharFormat->SetBase(aGlobalLayerInfo.iAggCharFormatLayer);
  2829 			iSharedParaQueHead.AddLast(*newParaAttribs);
  2830 			}
  2831 		newParaAttribs->iRefCount--;  // we have not yet linked incoming para - taken out no shares just yet
  2832 		aMap.Bind(currentSharedPara,newParaAttribs);
  2833 		}
  2834 	}
  2835 
  2836 void CRichTextIndex::AppendPhraseIndexL(const CRichTextIndex* aSource,const TGlobalLayerInfoAppend& aGlobalLayerInfo)
  2837 //
  2838 	{
  2839 	TInt originalPhraseCount=iPhraseIx->Count();
  2840 	TInt requiredPhraseCount=aSource->iPhraseIx->Count();
  2841 
  2842 	// Extend phrase index by required amount
  2843 	iPhraseIx->AppendL(RPhraseAttribsEntry(),requiredPhraseCount);
  2844 
  2845 	CArrayFixFlat<TInt>* pictureMap=new(ELeave) CArrayFixFlat<TInt>(16);
  2846 	CleanupStack::PushL(pictureMap);
  2847 
  2848 	for (TInt jj=0;jj<requiredPhraseCount;jj++)
  2849 		{
  2850 		RPhraseAttribsEntry& sPhrase=(*aSource->iPhraseIx)[jj];
  2851 		CCharFormatLayer* sCharFormatLayer=sPhrase.CharFormat();
  2852 		CCharFormatLayer* charLayer=CCharFormatLayer::NewL(sCharFormatLayer);
  2853 		charLayer->SetBase(aGlobalLayerInfo.iAggCharFormatLayer);
  2854 		RPhraseAttribsEntry& tPhrase=(*iPhraseIx)[jj+originalPhraseCount];
  2855 		if (!sPhrase.IsPicturePhrase())
  2856 			tPhrase=RPhraseAttribsEntry(charLayer,sPhrase.Length());
  2857 		else
  2858 			{
  2859 			TPictureHeader hdr;
  2860 			hdr=sPhrase.PictureHeader();  // copy the header from the source
  2861 			if (hdr.iPicture.IsPtr())
  2862 				hdr.iPicture=NULL;		  // pic.ownership transferred later.
  2863 			CleanupStack::PushL(charLayer);
  2864 			TBool ownershipTaken(EFalse);
  2865 			CPicturePhrase* picPhrase=CPicturePhrase::NewL(hdr,charLayer,ownershipTaken);
  2866 			CleanupStack::Pop();  // charLayer
  2867 			tPhrase=RPhraseAttribsEntry(picPhrase);
  2868 			iPictureCount++;
  2869 			pictureMap->AppendL(jj);
  2870 			}
  2871 		}
  2872 
  2873 	// transfer pictures now
  2874 	for (TInt kk=pictureMap->Count();--kk>=0;)
  2875 		{
  2876 		TInt jj=pictureMap->At(kk);
  2877 		TPictureHeader* sHeader=(*aSource->iPhraseIx)[jj].PictureHeaderPtr();
  2878 		if (!sHeader)
  2879 		    {
  2880 		    OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPENDPHRASEINDEXL, "Invariant" );
  2881 		    }
  2882 		__ASSERT_DEBUG(sHeader,User::Invariant());
  2883 		if (sHeader->iPicture.IsPtr())
  2884 			{	// transfer picture to us
  2885 			TPictureHeader* tHeader=(*iPhraseIx)[jj+originalPhraseCount].PictureHeaderPtr();
  2886 			if (!tHeader)
  2887 			    {
  2888 			    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_APPENDPHRASEINDEXL, "Invariant" );
  2889 			    }
  2890 			__ASSERT_DEBUG(tHeader,User::Invariant());
  2891 			tHeader->iPicture=sHeader->iPicture.AsPtr();
  2892 			sHeader->iPicture=NULL;
  2893 			}
  2894 		}
  2895 
  2896 	CleanupStack::PopAndDestroy();	// pictureMap
  2897 	}
  2898 
  2899 
  2900 void CRichTextIndex::AppendParagraphL(const CParaFormatLayer* aGlobalParaFormatLayer,
  2901 									  const CCharFormatLayer* aGlobalCharFormatLayer,
  2902 									  TInt aReplicas)
  2903 // Append aReplicas empty paragraphs, the format of which is based on the
  2904 // global format layers.
  2905 //
  2906 	{
  2907 	__TEST_INVARIANT;
  2908 
  2909 	// add the shared para format record
  2910 	CParaAttribs* paraAttribs=CParaAttribs::NewL(aGlobalParaFormatLayer,aGlobalCharFormatLayer);
  2911 	paraAttribs->iParaFormat->SetBase(aGlobalParaFormatLayer);  // reset the base properly
  2912 	paraAttribs->iCharFormat->SetBase(aGlobalCharFormatLayer);  // reset the base properly.
  2913 	CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,paraAttribs));
  2914 	CParaAttribs* paUsed=GetParaAttribsL(aGlobalParaFormatLayer,aGlobalCharFormatLayer,paraAttribs);
  2915 	// guaranteed not to leave as 3rd argument specified.
  2916 	// The refCount of paUsed is incremented here by one share.
  2917 	//
  2918 	// add the paragraph records
  2919 	TParaAttribsEntry paraEntry(1,paUsed);
  2920 	iParaIx->AppendL(paraEntry,aReplicas);
  2921 	if (paUsed!=paraAttribs)
  2922 		CleanupStack::PopAndDestroy();
  2923 	else
  2924 		CleanupStack::Pop();
  2925 	//
  2926 	// set the ref count of paUsed
  2927 	paUsed->iRefCount+=(aReplicas-1);  // compensate for already adding 1 share.
  2928 
  2929 	__TEST_INVARIANT;
  2930 	}
  2931 
  2932 
  2933 TParaAttribsEntry::TParaAttribsEntry():
  2934 	iLength(0),
  2935 	iParaAttribs(NULL)
  2936 	{
  2937 	}
  2938 
  2939 
  2940 TParaAttribsEntry::TParaAttribsEntry(TInt aLength,CParaAttribs* aParaAttribs):
  2941 	iLength(aLength),
  2942 	iParaAttribs(aParaAttribs)
  2943 	{
  2944 	}
  2945 
  2946 
  2947 CParaAttribs* CParaAttribs::NewL(const CParaFormatLayer* aParaLayer,const CCharFormatLayer* aCharLayer)
  2948 // Returns a handle to a new instance of this class.
  2949 // Creates a CParaAttribs with constant character formatting.
  2950 //
  2951 	{
  2952 	CParaAttribs* self=new(ELeave) CParaAttribs();
  2953 	CleanupStack::PushL(self);
  2954 	self->iParaFormat=CParaFormatLayer::NewCopyBaseL(aParaLayer);
  2955 	// iParaFormat will be released in destructor.
  2956 	// To prevent Coverity from reporting defect, add a comment:
  2957 	// coverity[leave_without_push]
  2958 	self->iCharFormat=CCharFormatLayer::NewCopyBaseL(aCharLayer);
  2959 	CleanupStack::Pop();
  2960 	self->iRefCount=EPrimeSharedCount;
  2961 	return self;
  2962 	}
  2963 
  2964 
  2965 CParaAttribs* CParaAttribs::NewL(const CParaFormatLayer* aParaLayer)
  2966 // Returns a handle to a new instance of this class.
  2967 // Creates a CParaAttribs for specific character formatting.
  2968 //
  2969 	{
  2970 	CParaAttribs* self=new(ELeave) CParaAttribs();
  2971 	CleanupStack::PushL(self);
  2972 	self->iParaFormat=CParaFormatLayer::NewCopyBaseL(aParaLayer);
  2973 	CleanupStack::Pop();
  2974 	self->iRefCount=EPrimeNonSharedCount;
  2975 	self->iPhraseCount=1;
  2976 	return self;
  2977 	}
  2978 
  2979 
  2980 CParaAttribs* CParaAttribs::NewL(const CParaAttribs* aParaAttribs)
  2981 	{
  2982 	return NewL(aParaAttribs->iParaFormat,aParaAttribs->iCharFormat);
  2983 	}
  2984 
  2985 
  2986 CParaAttribs::CParaAttribs():
  2987 	iRefCount(-1)  // Ensures Destruct works correctly when called on a semi-initialised object.
  2988 	{
  2989 	}
  2990 
  2991 
  2992 void CParaAttribs::Release()
  2993 // Release a share on this CParaAttribs.
  2994 // If after this, no shares remain, destroy this CParaAttribs.
  2995 //
  2996 	{
  2997 	iRefCount--;
  2998 	if (iRefCount<=0)
  2999 		delete this;
  3000 	}
  3001 
  3002 
  3003 void CParaAttribs::Release(TInt aCount)
  3004 // Release aCount number of shares of this CParaAttribs.
  3005 // If after this, no shares remain, destroy this CParaAttribs.
  3006 //
  3007 	{
  3008 	iRefCount-=aCount;
  3009 	if (iRefCount<=0)
  3010 		delete this;
  3011 	}
  3012 
  3013 
  3014 CParaAttribs::~CParaAttribs()
  3015 // Release the memory associated with this object.
  3016 //
  3017 	{
  3018 	delete iParaFormat;
  3019 	if (iRefCount==0)
  3020 		{// Constant character formatting - in the shared list.
  3021 		delete iCharFormat;
  3022 		link.Deque();  // Remove this para attribs from the shared list.
  3023 		}
  3024 	}
  3025 
  3026 
  3027 TInt CParaAttribs::PhraseCount()const
  3028 // Return a count of the number of phrases in this para attribs.
  3029 //
  3030 	{return (iRefCount>=1)?1:iPhraseCount;}
  3031 
  3032 
  3033 DLLEXPORT_C void RPhraseAttribsEntry::__DbgTestInvariant()const
  3034 // Class invariants.
  3035 //
  3036 	{
  3037 #ifdef _DEBUG
  3038 // ASSERT: iLength is +ve (applying to character formatting, or	is set to indicate a picture phrase.
  3039 	if (iLength<0 && !IsPicturePhrase())
  3040 	    {
  3041 	    OstTrace0( TRACE_DUMP, RPHRASEATTRIBSENTRY_DBGTESTINVARIANT, "Invariant" );
  3042 	    }
  3043 	__ASSERT_DEBUG(iLength>=0 || IsPicturePhrase(),User::Invariant());
  3044 #endif
  3045 	}
  3046 
  3047 
  3048 RPhraseAttribsEntry::RPhraseAttribsEntry():
  3049 	iLength(0),
  3050 	iCharFormat(NULL)
  3051 	{
  3052 	}
  3053 
  3054 
  3055 RPhraseAttribsEntry::RPhraseAttribsEntry(CCharFormatLayer* aCharFormat,TInt aLength):
  3056 	iLength(aLength),
  3057 	iCharFormat(aCharFormat)
  3058 	{
  3059 	}
  3060 
  3061 
  3062 RPhraseAttribsEntry::RPhraseAttribsEntry(CPicturePhrase* aPicturePhrase):
  3063 	iLength(EPictureIndicator),
  3064 	iPicturePhrase(aPicturePhrase)
  3065 	{
  3066 	}
  3067 
  3068 
  3069 void RPhraseAttribsEntry::AssignAndRelease(const RPhraseAttribsEntry& aPhrase)
  3070 // Assign the state of the specified object to this,
  3071 //
  3072 	{
  3073 	iLength=aPhrase.iLength;
  3074 	iCharFormat=aPhrase.iCharFormat;  // both union members share the same address space, so this is fine.
  3075 	}
  3076 
  3077 
  3078 void RPhraseAttribsEntry::Discard()
  3079 // Free storage.
  3080 //
  3081 	{
  3082 	__TEST_INVARIANT;
  3083 
  3084 	if (iLength==EPictureIndicator)
  3085 		delete iPicturePhrase;
  3086 	else
  3087 		delete iCharFormat;
  3088 	}
  3089 
  3090 
  3091 void RPhraseAttribsEntry::ExternalizeL(RWriteStream& aStream)const
  3092 // Save this phrase into aStream.
  3093 //
  3094 	{
  3095 	TUint8 picIndicator=(TUint8)(IsPicturePhrase()!=EFalse);
  3096 	aStream.WriteUint8L(picIndicator);
  3097 	aStream.WriteInt32L(Length());
  3098 	aStream<< *CharFormat();
  3099 	if ((TBool)picIndicator)
  3100 		aStream<< *PictureHeaderPtr();
  3101 	}
  3102 
  3103 
  3104 CCharFormatLayer* RPhraseAttribsEntry::CharFormat()const
  3105  // Returns a pointer the CCharFormatLayer of this phrase.
  3106  //
  3107  	{return (iLength==EPictureIndicator)?iPicturePhrase->iCharFormat:iCharFormat;}
  3108 
  3109 
  3110 void RPhraseAttribsEntry::SetLength(TInt aLength)
  3111 // Sets the phrase length.
  3112 //
  3113 	{
  3114 	__TEST_INVARIANT;
  3115 
  3116 	iLength=aLength;
  3117 	}
  3118 
  3119 
  3120 void RPhraseAttribsEntry::AdjustLength(TInt aIncrement)
  3121 // Adjusts the length of the phrase by the signed value aIncrement.
  3122 //
  3123 	{
  3124 // ASSERT: The length of a picture phrase may only be altered by deleting it, in which case
  3125 //			the only adjustment made will be an increment of -1 (EPictureIndicator).
  3126 	if (IsPicturePhrase() && (!IsPicturePhrase() || aIncrement!=EPictureIndicator)
  3127                                       && (!IsPicturePhrase() || aIncrement!=0) )
  3128 	    {
  3129 	    OstTrace0( TRACE_DUMP, RPHRASEATTRIBSENTRY_ADJUSTLENGTH, "EModifiedPicturePhraseLength" );
  3130 	    }
  3131 	__ASSERT_DEBUG(!IsPicturePhrase() || (IsPicturePhrase() && aIncrement==EPictureIndicator)
  3132 									  || (IsPicturePhrase() && aIncrement==0)
  3133 									  ,Panic(EModifiedPicturePhraseLength));
  3134 	TInt len=iLength;
  3135 	iLength=(len==EPictureIndicator)
  3136 		? len-aIncrement
  3137 		: len+aIncrement;
  3138 	}
  3139 
  3140 
  3141 TInt RPhraseAttribsEntry::GetPictureSizeInTwips(TSize& aSize)const
  3142 // If this is a picture phrase, write the size of the picture to aSize,
  3143 // otherwise return KErrNotFound.
  3144 //
  3145 	{
  3146 	if (!IsPicturePhrase())
  3147 		return KErrNotFound;
  3148 	if (iPicturePhrase->iPicHdr.iPicture.IsPtr() && iPicturePhrase->iPicHdr.iPicture.AsPtr())
  3149 		iPicturePhrase->iPicHdr.iPicture->GetSizeInTwips(aSize);
  3150 	else
  3151 		aSize=iPicturePhrase->iPicHdr.iSize;
  3152 	return KErrNone;
  3153 	}
  3154 
  3155 
  3156 TPictureHeader RPhraseAttribsEntry::PictureHeader()const
  3157 // Return the picture header describing the picture at character position aPos.
  3158 // If there is no picture at character position aPos, a default picture header is returned.
  3159 //
  3160 	{
  3161 	return (IsPicturePhrase())
  3162 		? iPicturePhrase->iPicHdr
  3163 		: TPictureHeader();
  3164 	}
  3165 
  3166 
  3167 const CPicture* RPhraseAttribsEntry::PictureHandleL(const MPictureFactory* aFactory,
  3168 													const MRichTextStoreResolver* aResolver,
  3169 													TInt aPos,
  3170 													MLayDoc::TForcePictureLoad aForceLoad)const
  3171 // For a text phrase returns NULL.
  3172 // For a picture phrase returns a pointer to the picture object itself.
  3173 // May leave since loading of pictures is deferred until this point,
  3174 // ie, when the picture is required.
  3175 // Also returns NULL if the picture is not in memory *and* TForcePictureLoad is false.
  3176 //
  3177 	{
  3178 	if (iLength!=EPictureIndicator)
  3179 		return NULL;
  3180 	else
  3181 		{// Check if the picture is in memory
  3182 		CPicture* handle=NULL;
  3183 		if (iPicturePhrase->iPicHdr.iPicture.IsPtr())
  3184 			handle=iPicturePhrase->iPicHdr.iPicture;
  3185 		else
  3186 			{// Check if the picture should be loaded from persistent storage
  3187 			if (aForceLoad==MLayDoc::EForceLoadTrue)  // picture not in memory, so load it.
  3188 				{
  3189 				if (aResolver==NULL || aFactory==NULL)
  3190 					return NULL;
  3191 				const CStreamStore& store=aResolver->StreamStoreL(aPos);
  3192 				aFactory->NewPictureL(iPicturePhrase->iPicHdr,store);
  3193 				handle=iPicturePhrase->iPicHdr.iPicture;
  3194 				}
  3195 			}
  3196 		return handle;
  3197 		}
  3198 	}
  3199 
  3200 
  3201 TPictureHeader* RPhraseAttribsEntry::PictureHeaderPtr()const
  3202 // For a [text] phrase returns NULL.
  3203 // For a picture phrase returns a pointer to the picture header.
  3204 //
  3205 	{return (iLength==EPictureIndicator)? &iPicturePhrase->iPicHdr:NULL;}
  3206 
  3207 
  3208 CCharFormatLayer* RPhraseAttribsEntry::ReleaseCharFormatLayerOwnership()
  3209 // Return a handle to the character format layer, then set this handle to it to NULL.
  3210 // thus giving up ownership of the object.
  3211 //
  3212 	{
  3213 	CCharFormatLayer*& layer=(iLength==EPictureIndicator)?iPicturePhrase->iCharFormat:iCharFormat;
  3214 	CCharFormatLayer* charFormatLayer=layer;
  3215 	layer=NULL;
  3216 	return charFormatLayer;
  3217 	}
  3218 
  3219 
  3220 TBool RPhraseAttribsEntry::IsIdentical(const RPhraseAttribsEntry& aPhrase)const
  3221 // Returns ETrue if this phrase is identical to the specified phrase.
  3222 // otherwise returns EFalse.
  3223 // The 2 are equal only if they are both non-zero length text phrases, and of identical character format.
  3224 //
  3225 	{
  3226 	if (iLength<=0)				// picture or zero-length
  3227 		return EFalse;
  3228 	if (aPhrase.iLength<=0)		// picture or zero-length
  3229 		return EFalse;
  3230 	return iCharFormat->IsIdentical(aPhrase.iCharFormat,EFalse); // EFalse=do not compare based-on link
  3231 	}
  3232 
  3233 
  3234 CPicturePhrase* CPicturePhrase::NewL(const TPictureHeader& aPicHdr,TCharFormatX& aFormat,
  3235 									 TCharFormatXMask& aMask,CCharFormatLayer* aCharBase,
  3236 									 TBool& aPictureOwnershipTaken)
  3237 	{
  3238 	CPicturePhrase* self=new(ELeave) CPicturePhrase(aPicHdr,aPictureOwnershipTaken);
  3239 	CleanupStack::PushL(self);
  3240 	self->ConstructL(aFormat,aMask,aCharBase);
  3241 	CleanupStack::Pop();
  3242 	return self;
  3243 	}
  3244 
  3245 
  3246 // The charLayer is assumed to have a valid based-on link already set.
  3247 // Called as part of the rich text index Internalize only.
  3248 CPicturePhrase* CPicturePhrase::NewL(const TPictureHeader& aPicHdr,
  3249                                      CCharFormatLayer* aCharLayer,
  3250                                      TBool& aPictureOwnershipTaken)
  3251 	{
  3252 	return new(ELeave) CPicturePhrase(aPicHdr,aCharLayer,aPictureOwnershipTaken);
  3253 	}
  3254 
  3255 
  3256 CPicturePhrase::CPicturePhrase(const TPictureHeader& aPicHdr,TBool& aPictureOwnershipTaken):
  3257 	iPicHdr(aPicHdr)
  3258 	{
  3259    	aPictureOwnershipTaken=ETrue;
  3260 	}
  3261 
  3262 
  3263 CPicturePhrase::CPicturePhrase(const TPictureHeader& aPicHdr,
  3264                                CCharFormatLayer* aCharLayer,
  3265                                TBool& aPictureOwnershipTaken):
  3266 	iCharFormat(aCharLayer),
  3267 	iPicHdr(aPicHdr)
  3268 	{
  3269 	aPictureOwnershipTaken=ETrue;
  3270 	}
  3271 
  3272 
  3273 void CPicturePhrase::ConstructL(TCharFormatX& aFormat,TCharFormatXMask& aMask,CCharFormatLayer* aCharBase)
  3274 	{
  3275 	iCharFormat=CCharFormatLayer::NewL(aFormat,aMask);
  3276 	iCharFormat->SetBase(aCharBase);
  3277 	}
  3278 
  3279 
  3280 CPicturePhrase::~CPicturePhrase()
  3281 	{
  3282 	iPicHdr.DeletePicture();
  3283 	delete iCharFormat;
  3284 	}
  3285 
  3286 TLogicalPosition::TLogicalPosition():
  3287 	iDocPos(0),
  3288 	iParaElement(0),
  3289 	iParaElementOffset(0),
  3290 	iPhraseElement(0),
  3291 	iPhraseElementOffset(0)
  3292 	{
  3293 	}
  3294 
  3295 
  3296 void TLogicalPosition::Clear()
  3297 // Reset this logical position record.
  3298 //
  3299 	{
  3300 	*this=TLogicalPosition();
  3301 	iParaBasePhraseElement=0;
  3302 	}