Update contrib.
2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
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".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
30 #include "OstTraceDefinitions.h"
31 #ifdef OST_TRACE_COMPILER_IN_USE
32 #include "TXTINDEXTraces.h"
35 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
36 #include "TXTFMLYR_INTERNAL.H"
37 #include "TXTSTYLE_INTERNAL.H"
40 TGlobalLayerInfoAppend::TGlobalLayerInfoAppend()
41 : iAggParaFormatLayer(NULL),iAggCharFormatLayer(NULL),iComParaFormatLayer(NULL),iComCharFormatLayer(NULL)
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)
51 TTextFragment::TTextFragment():
58 TCurrentIndexRecords::TCurrentIndexRecords()
63 DLLEXPORT_C void CRichTextIndex::__DbgTestInvariant()const
64 // Provides class invariants. Explanations below:
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++)
75 // ASSERT: The basedOn link is valid.
76 CFormatLayer* thisLayer=(*iParaIx)[para].iParaAttribs->iParaFormat;
77 CFormatLayer* base=CONST_CAST(CFormatLayer*,thisLayer->SenseBase());
80 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DBGTESTINVARIANT, "base==NULL" );
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++)
92 const RPhraseAttribsEntry* phrase=&(*iPhraseIx)[currentPhraseElement];
93 CCharFormatLayer* charFormatLayer=phrase->CharFormat();
94 // ASSERT: The basedOn link is valid.
95 if (charFormatLayer->SenseBase()==NULL)
97 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_DBGTESTINVARIANT, "charFormatLayer->SenseBase()==NULL" );
99 __ASSERT_DEBUG(charFormatLayer->SenseBase()!=NULL,User::Invariant());
100 if (TInt(charFormatLayer->SenseBase())<=0x1000)
102 OstTrace0( TRACE_DUMP, DUP2_CRICHTEXTINDEX_DBGTESTINVARIANT, "TInt(charFormatLayer->SenseBase())<=0x1000" );
104 __ASSERT_DEBUG(TInt(charFormatLayer->SenseBase())>0x1000,User::Invariant());
105 sumOfPhraseLengths+=(*iPhraseIx)[currentPhraseElement].Length();
106 if ((*iPhraseIx)[currentPhraseElement].Length()==0)
107 zeroLengthPhraseCount++;
108 currentPhraseElement++;
112 CCharFormatLayer* charFormatLayer=(*iParaIx)[para].iParaAttribs->iCharFormat;
113 // ASSERT: The basedOn link is valid.
114 if (charFormatLayer->SenseBase()==NULL)
116 OstTrace0( TRACE_DUMP, DUP3_CRICHTEXTINDEX_DBGTESTINVARIANT, "charFormatLayer->SenseBase()==NULL" );
118 __ASSERT_DEBUG(charFormatLayer->SenseBase()!=NULL,User::Invariant());
119 sumOfPhraseLengths+=(*iParaIx)[para].iLength;
122 if (sumOfPhraseLengths!=paragraphLength)
124 OstTrace0( TRACE_DUMP, DUP4_CRICHTEXTINDEX_DBGTESTINVARIANT, "sumOfPhraseLengths!=paragraphLength" );
126 __ASSERT_DEBUG(sumOfPhraseLengths==paragraphLength,User::Invariant());
128 // ASSERT: We have no unexpected phrases left over
129 if (currentPhraseElement!=-1 &&
130 currentPhraseElement!=iPhraseIx->Count())
132 OstTrace0( TRACE_DUMP, DUP5_CRICHTEXTINDEX_DBGTESTINVARIANT, "We have no unexpected phrases left over" );
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)))
140 OstTrace0( TRACE_DUMP, DUP6_CRICHTEXTINDEX_DBGTESTINVARIANT, "There is either zero(0) or one(1) zero length phrase in the whole index" );
142 __ASSERT_DEBUG( (zeroLengthPhraseCount==0) ||
143 (zeroLengthPhraseCount==1 && iPendingNewPhrasePos!=EInsertCharFormatReset),
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())
150 CParaAttribs* currentSharedPara;
151 TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
152 while ((currentSharedPara=iterator++)!=NULL)
153 totalReferenceCount+=currentSharedPara->iRefCount;
155 if ((numberOfReferencesToSharedList!=totalReferenceCount) &&
156 (numberOfReferencesToSharedList!=totalReferenceCount-1))
158 OstTrace0( TRACE_DUMP, DUP7_CRICHTEXTINDEX_DBGTESTINVARIANT, "Invariant" );
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.
164 TInt phraseCount=(iPhraseIx) ? iPhraseIx->Count() : 0;
165 for (TInt item=0;item<phraseCount;item++)
166 picCount+=((*iPhraseIx)[item].IsPicturePhrase())
169 if (iPictureCount!=picCount)
171 OstTrace0( TRACE_DUMP, DUP8_CRICHTEXTINDEX_DBGTESTINVARIANT, "Invariant" );
173 __ASSERT_DEBUG(iPictureCount==picCount,User::Invariant());
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.
184 CRichTextIndex* self=new(ELeave) CRichTextIndex(aText);
185 CleanupStack::PushL(self);
186 self->ConstructL(aGlobalParaLayer,aGlobalCharLayer,aParaGran,aPhraseGran);
192 CRichTextIndex::CRichTextIndex(const CRichText& aText):
194 iPendingNewPhrasePos(EInsertCharFormatReset),
195 iSharedParaQueHead(_FOFF(CParaAttribs,link))
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.
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);
220 iPhraseIx=new(ELeave) CArrayFixSeg<RPhraseAttribsEntry>(aPhraseGran);
226 CRichTextIndex::~CRichTextIndex()
227 // Free up all storage allocated in the rich text index.
232 {// Destroy the phrase index and its contents.
235 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_CRICHTEXTINDEX, "EPhraseIxPresentWithNoParaIx" );
237 __ASSERT_ALWAYS(iParaIx,Panic(EPhraseIxPresentWithNoParaIx));
238 count=iPhraseIx->Count();
239 for (TInt offset=0;offset<count;offset++)
240 (*iPhraseIx)[offset].Discard();
244 {// Destroy the para index and its contents.
245 count=iParaIx->Count();
246 for (TInt offset=0;offset<count;offset++)
248 CParaAttribs* pA=(*iParaIx)[offset].iParaAttribs;
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())
264 OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_CRICHTEXTINDEX, "ERichTextIndexIntegrityErr" );
266 __ASSERT_ALWAYS(iSharedParaQueHead.IsEmpty(),Panic(ERichTextIndexIntegrityErr));
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.
275 // If aParaOffset specifies a paragraph that does not exist, EScanEndOfData is returned.
280 if (aParaOffset>=iParaIx->Count())
281 return CPlainText::EScanEndOfData;
283 for (offset=0;offset<aParaOffset;offset++)
284 pos+=(*iParaIx)[offset].iLength;
285 aLength=(*iParaIx)[offset].iLength;
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.
298 ((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionAbsolute);
299 aPos-=iPos.iParaElementOffset;
300 return iPos.iParaElement;
304 void CRichTextIndex::DocumentChanged()const
306 MUTABLE_CAST(TLogicalPosition&,iLastUsed).Clear();
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.
326 if (!((iPendingNewPhrasePos == EInsertCharFormatReset) || (aPos == iPendingNewPhrasePos)))
327 CancelInsertCharFormat();
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.
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.
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);
354 current.iParaAttribs->iPhraseCount++;
356 GetCurrentRecords(current); // Update current records, cos used below.
358 // Extend the lengths in the index.
359 current.iParaEntry->iLength+=aLength; // Increase length of paragraph.
361 current.iPhrase->AdjustLength(aLength); // Increase the length of this phrase.
362 iPos.iPhraseElementOffset+=aLength;
363 iPos.iParaElementOffset+=aLength;
364 iPos.iDocPos+=aLength;
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.
374 TInt bufLen=aBuf.Length();
377 TInt paragraphCount=0;
378 const TText* start=aBuf.Ptr();
379 const TText* end=start+bufLen;
382 if (*start++==CEditableText::EParagraphDelimiter)
385 if (paragraphCount==0)
389 DoSoloInsertL(aPos,bufLen);
390 CancelInsertCharFormat();
395 ScanToPosition(aPos,EScanToPositionMatchLeft);
396 TLogicalPosition pastePos=iPos; // Used to adjust paragraph/phrase lengths later
398 if (paragraphCount>1)
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);
411 theParaAttribs->iRefCount+=paragraphCount-2; // add the extra references
413 TLogicalPosition pos;
415 SplitParagraphAtPastePosL(pastePos,pos,aGlobalParaFormatLayer));
418 RbRemoveInsertedParaAttribsEntries(pastePos.iParaElement,paragraphCount-1);
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;
429 }// Cos weve inserted new paragraphs in front of the governing one.
431 // Sort out the front para
432 TParaAttribsEntry& frontPara=(*iParaIx)[pastePos.iParaElement];
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)
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];
446 phrase->AdjustLength(lengthOfFirstPara); // Adjust the phrase length
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)
454 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_INSERTL, "EInsertEmbeddedParaErr" );
456 __ASSERT_DEBUG(length!=KErrNotFound,Panic(EInsertEmbeddedParaErr));
458 buf.Set(buf.Right(buf.Length()-length));
460 // For final paragrph
461 TInt trailingTextLen=buf.Length();
462 if (trailingTextLen>0)
464 TParaAttribsEntry& backPara=(*iParaIx)[pos.iParaElement];
465 backPara.iLength+=trailingTextLen;
466 if (!backPara.iParaAttribs->IsShared())
468 RPhraseAttribsEntry& phrase=(*iPhraseIx)[pos.iPhraseElement];
469 phrase.AdjustLength(trailingTextLen); // Adjust phrase length
474 if (bufLen>1 && iPendingNewPhrasePos!=EInsertCharFormatReset)
476 iPendingNewPhrasePos=aPos+(bufLen-trailingTextLen);
477 CancelInsertCharFormat();
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.
489 TInt insertPendingPos = iPendingNewPhrasePos;
490 DoSoloInsertL(aPastePos.iDocPos,1);
491 TBool insertCharFormatDeleted = EFalse;
492 if (InsertCharFormatIsActive())
494 insertCharFormatDeleted = DeleteInsertCharFormat();
495 iPendingNewPhrasePos = EInsertCharFormatReset;
498 InsertParagraphL(aPastePos.iDocPos+1,aGlobalParaFormatLayer)); // Split the current para (and maybe phrase index).
501 // locate the character inserted by DoSoloInsertL() above
502 ScanToPosition(aPastePos.iDocPos,EScanToPositionAbsolute);
503 TCurrentIndexRecords current;
504 GetCurrentRecords(current);
505 current.iParaEntry->iLength--;
508 current.iPhrase->AdjustLength(-1); // collapse phrase by right amount
509 if (insertPendingPos!=EInsertCharFormatReset)
511 if (current.iPhrase->Length()!=0)
513 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SPLITPARAGRAPHATPASTEPOSL, "Invariant" );
515 __ASSERT_DEBUG(current.iPhrase->Length()==0,User::Invariant());
516 iPendingNewPhrasePos=insertPendingPos;
518 else if (current.iPhrase->Length()==0)
520 RemoveFromPhraseIx(iPos.iPhraseElement,1);
521 current.iParaAttribs->iPhraseCount--;
522 if (current.iParaAttribs->PhraseCount()<=1)
524 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_SPLITPARAGRAPHATPASTEPOSL, "Invariant" );
526 __ASSERT_DEBUG(current.iParaAttribs->PhraseCount()>1,User::Invariant());
529 OstTrace1( TRACE_DUMP, DUP2_CRICHTEXTINDEX_SPLITPARAGRAPHATPASTEPOSL, "Leave code=%d", ret );
532 if (insertPendingPos != EInsertCharFormatReset)
533 ConsolidateAt(insertPendingPos, insertCharFormatDeleted?
534 EPositionOnly : EFollowingPhrase);
535 ScanToPosition(aPastePos.iDocPos+1,EScanToPositionMatchLeft); // Gives us the next para.
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.
545 {return aBuf.Locate(CEditableText::EParagraphDelimiter);}
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
557 // ASSERT: A valid picture header, referencing a valid picture has been inserted.
558 if (!aPicHdr.iPicture.IsPtr() || aPicHdr.iPicture.AsPtr()==NULL)
560 OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_INSERTL, "EInsertNullPicHdrData" );
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))
566 OstTrace0( TRACE_FATAL, DUP2_CRICHTEXTINDEX_INSERTL, "ESetInsertCharFormatIntegrityErr" );
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);
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;
587 CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimed));
588 current.iParaEntry->iParaAttribs=reclaimed; // Use this reclaimed para attribs (the new CParaAttribs instance)
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)
594 OstTrace0( TRACE_DUMP, DUP3_CRICHTEXTINDEX_INSERTL, "EReclaimShareError" );
596 __ASSERT_DEBUG(current.iPhrase!=NULL,Panic(EReclaimShareError));
598 SplitPhraseL(aPos)); // Phrase may not be split if at boundary.
602 RbInsertPicture(rollbackParaAttribsHandle);//Restore the old CParaAttribs instance
606 TInt offset=(PhraseSplit())?1:0; // Insert position of new phrase relative to current.
607 RPhraseAttribsEntry newPhrase(picture);
609 iPhraseIx->InsertL(iPos.iPhraseElement+offset,newPhrase));
612 RbInsertPicture(rollbackParaAttribsHandle);//Restore the old CParaAttribs instance
618 CleanupStack::Pop();//"Pop" for CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimed));
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
629 rollbackParaAttribsHandle->Release(); // Release hold on original shared paraAttribs.
633 //coverity[memory_leak]
636 void CRichTextIndex::RbInsertPicture(CParaAttribs* aGoodParaAttribs)
637 // Reinstate the original good paraAttribs.
638 // Then rollback the SplitPhrase() call if it succeeded.
641 (*iParaIx)[iPos.iParaElement].iParaAttribs=aGoodParaAttribs;
643 {// Rollback the SplitPhrase()
644 TInt length=(*iPhraseIx)[iPos.iPhraseElement+1].Length();
645 RemoveFromPhraseIx(iPos.iPhraseElement+1);
646 (*iPhraseIx)[iPos.iPhraseElement].AdjustLength(length);
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)
655 void CRichTextIndex::InsertParagraphL(TInt aPos,const CParaFormatLayer& aGlobalParaFormatLayer)
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();
665 charLayer=current.iParaAttribs->iCharFormat;
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();
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.
677 if (currentStyleType==KNormalParagraphStyleUid)
679 else if (currentStyleType==KUserDefinedParagraphStyleUid)
681 else if (iPos.iParaElementOffset<=(current.iParaEntry->iLength-2)) // cos of previous call to DoSoloInsertL()
685 newParaLayer->SetBase((useNormal)
686 ? &aGlobalParaFormatLayer
688 const CCharFormatLayer* newCharBase=(useNormal)
689 ? iText.GlobalCharFormatLayer()
690 : STATIC_CAST(const CParagraphStyle&,currentStyle).CharFormatLayer();
691 CleanupStack::PushL(newParaLayer);
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
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));
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
718 // Insert the new paragraph.
719 newPara.iParaAttribs=newParaAttribs;
721 iParaIx->InsertL(iPos.iParaElement+1,newPara)); // Inserts the new paraAttribsEntry record.
724 RemoveFromPhraseIx(iPos.iPhraseElement+1,1); // inserted by DoSplitPhraseL
725 current.iParaAttribs->iPhraseCount--;
729 iPendingNewPhrasePos=insertPendingPos;
730 CleanupStack::Pop(); // newParaAttribs. All OK now
731 GetCurrentRecords(current); // could be changed by InsertL() above
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;
739 const CArrayFix<RPhraseAttribsEntry>& phraseIx=*iPhraseIx;
740 TInt startPhrase=iPos.iPhraseElement+1;
741 for (TInt ii=startPhrase; ii<startPhrase+newPhraseCount; ii++)
743 RPhraseAttribsEntry phrase=phraseIx[ii];
744 phrase.CharFormat()->SetBase(newCharBase);
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);
752 Share(iParaIx->At(iPos.iParaElement),iPos.iParaBasePhraseElement);
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.
761 void CRichTextIndex::SetForDeleteL(TIndexDeleteInfo& aInfo,TInt aPos,TInt aLength)
764 __TEST_INVARIANT; // Do not need to RebalanceIndex(); part of defined behaviour for delete.
766 aInfo.iDeleteLength=aLength;
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
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;
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;
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;
806 reclaimedStartPara=RequestReclaimShareL(current.iParaAttribs,current.iParaEntry));
809 if (reclaimedEndPara)
811 reclaimedEndPara->Release();
812 RemoveFromPhraseIx(endPosPhrase);
813 origEndParaEntry->iParaAttribs=origParaAttribs;
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.
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;
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.
834 // coverity[memory_leak]
838 TBool CRichTextIndex::DeleteParagraph(TInt aPos,TInt aLength)
839 // Remove aCount entire paragraphs from the text.
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.
846 __TEST_INVARIANT; // Do not need to RebalanceIndex(); part of defined behaviour for delete.
848 CancelInsertCharFormat();
849 ScanToPosition(aPos,EScanToPositionAbsolute,&iLastUsed);
851 if (iPos.iParaElementOffset!=0)
853 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DELETEPARAGRAPH, "EDeleteParagraphInvalidStartValue" );
855 __ASSERT_DEBUG(iPos.iParaElementOffset==0,Panic(EDeleteParagraphInvalidStartValue));
857 TIndexDeleteInfo info;
858 info.iDeleteType=TIndexDeleteInfo::EDeleteParagraph;
859 info.iDeletePos=iPos;
860 info.iStartPara=iPos.iParaElement;
862 TInt documentLength=iText.DocumentLength();
863 TInt pos=(aPos+aLength>documentLength
866 ScanToPosition(pos,EScanToPositionMatchLeft,&iLastUsed); // Forces endPara to be next para when just removing a para delimiter.
868 info.iEndPara=iPos.iParaElement;
869 info.iDeleteLength=aLength;
872 // do not want to call TidyAfterDelete()
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.
886 __TEST_INVARIANT; // Do not need to RebalanceIndex(); part of defined behaviour for delete.
888 ScanToPosition(aPos,EScanToPositionAbsolute);
892 TCurrentIndexRecords current;
893 GetCurrentRecords(current);
894 TInt startParaLength=current.iParaEntry->iLength;
895 TInt lengthRemainingInPara=startParaLength-iPos.iParaElementOffset;
897 if (aLength>=lengthRemainingInPara)
899 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DELETEFROMPARAGRAPH, "EDeleteFromParagraphInvalidRange" );
901 __ASSERT_ALWAYS(aLength<lengthRemainingInPara,Panic(EDeleteFromParagraphInvalidRange));
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;
912 DoDeleteFromParagraph(info);
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.
924 iPos=aInfo.iDeletePos;
925 TInt length=aInfo.iDeleteLength;
926 DeleteParagraphText(length);
927 TidyAfterDelete(aInfo);
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.
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.
946 TBool firstParaRemoved=(FirstPhraseOfParagraph() && iPos.iPhraseElementOffset==0 && aInfo.iDeleteLength>=current.iParaEntry->iLength);
947 // ETrue if the 1st para has been *wholly* deleted.
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++)
954 ScanToPosition(aInfo.iDeletePos.iDocPos,EScanToPositionAbsolute);
955 DeleteParagraphText(leftToDelete);
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);
968 if (aInfo.iDeleteType!=TIndexDeleteInfo::EDeleteParagraph)
969 TidyAfterDelete(aInfo);
976 void CRichTextIndex::TidyAfterDelete(const TIndexDeleteInfo& aInfo)
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;
996 void CRichTextIndex::Normalize(TInt aPos)
999 ScanToPosition(aPos,EScanToPositionAbsolute);
1006 void CRichTextIndex::NormalizeNow(const TLogicalPosition& aNormalizePos)
1009 CParaAttribs* currentParaAttribs=(*iParaIx)[aNormalizePos.iParaElement].iParaAttribs;
1010 if (!currentParaAttribs->IsShared())
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);
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.
1030 CParaAttribs* reservedCell=ReserveCellL();
1031 CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reservedCell));
1032 return reservedCell;
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.
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;
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.
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);
1075 TInt deleteFromPhrase=CurrentPhraseLength()-iPos.iPhraseElementOffset;
1076 TInt maxPhrase=current.iParaAttribs->PhraseCount();
1077 TInt currentPhrase=(iPos.iPhraseElement-iPos.iParaBasePhraseElement);
1078 while (currentPhrase<maxPhrase)
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--;
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;
1096 break; // Nothing left to delete in this paragraph.
1097 // Get the data for the next phrase.
1098 GetCurrentRecords(current);
1099 deleteFromPhrase=CurrentPhraseLength();
1104 TBool CRichTextIndex::InsertCharFormatIsActive()
1106 return iPendingNewPhrasePos != EInsertCharFormatReset;
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.
1114 void CRichTextIndex::SetInsertCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,TInt aPos)
1116 if (InsertCharFormatIsActive() && aPos!=iPendingNewPhrasePos)
1118 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_SETINSERTCHARFORMATL, "ESetInsertCharFormatIntegrityErr" );
1120 __ASSERT_ALWAYS(!InsertCharFormatIsActive() || aPos==iPendingNewPhrasePos,
1121 Panic(ESetInsertCharFormatIntegrityErr));
1122 if (InsertCharFormatIsActive())
1123 UpdateInsertCharFormatL(aFormat, aMask);
1125 NewInsertCharFormatL(aFormat, aMask, aPos);
1128 void CRichTextIndex::NewInsertCharFormatL(const TCharFormatX& aFormat,
1129 const TCharFormatXMask& aMask, TInt aPos)
1131 if (InsertCharFormatIsActive())
1133 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_NEWINSERTCHARFORMATL, "ESetInsertCharFormatIntegrityErr" );
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));
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;
1163 {// Restore the original phrase index.
1167 OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_NEWINSERTCHARFORMATL, "LeaveNoMemory" );
1168 User::LeaveNoMemory();
1170 iPendingNewPhrasePos=aPos;
1173 void CRichTextIndex::UpdateInsertCharFormatL(const TCharFormatX& aFormat,
1174 const TCharFormatXMask& aMask)
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);
1184 CCharFormatLayer* CRichTextIndex::GetCurrentInsertCharFormat()
1186 if (!InsertCharFormatIsActive())
1188 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_GETCURRENTINSERTCHARFORMAT, "ESetInsertCharFormatIntegrityErr" );
1190 __ASSERT_DEBUG(InsertCharFormatIsActive(),
1191 Panic(ESetInsertCharFormatIntegrityErr));
1192 ScanToPosition(iPendingNewPhrasePos,EScanToPositionMatchLeft);
1193 TCurrentIndexRecords current;
1194 GetCurrentRecords(current);
1195 if ((*iPhraseIx)[iPos.iPhraseElement].Length() != 0)
1197 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_GETCURRENTINSERTCHARFORMAT, "ESetInsertCharFormatIntegrityErr" );
1199 __ASSERT_DEBUG((*iPhraseIx)[iPos.iPhraseElement].Length() == 0,
1200 Panic(ESetInsertCharFormatIntegrityErr));
1201 return (*iPhraseIx)[iPos.iPhraseElement].CharFormat();
1204 void CRichTextIndex::DoNewInsertCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,
1205 CCharFormatLayer* aBasedOn,CParaAttribs* aParaAttribs)
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++;
1221 void CRichTextIndex::RebalanceIndex()
1222 // Returns the index to a good state, by releasing the extra share taken on the paraAttribs
1225 if (iRollbackParaAttribsHandle)
1227 // ASSERT: The specified para attribs is indeed in the share list.
1228 if (!iRollbackParaAttribsHandle->IsShared())
1230 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_REBALANCEINDEX, "EParaAttribsNotInSharedList" );
1232 __ASSERT_ALWAYS(iRollbackParaAttribsHandle->IsShared(),Panic(EParaAttribsNotInSharedList));
1233 iRollbackParaAttribsHandle->Release();
1234 iRollbackParaAttribsHandle=NULL;
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.
1245 void CRichTextIndex::CancelInsertCharFormat()
1247 if (InsertCharFormatIsActive())
1249 TBool isDeleted = DeleteInsertCharFormat();
1250 ConsolidateAt(iPendingNewPhrasePos, isDeleted?
1251 EPositionOnly : EFollowingPhrase);
1252 iPendingNewPhrasePos = EInsertCharFormatReset;
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.
1261 TBool CRichTextIndex::DeleteInsertCharFormat()
1263 if (!InsertCharFormatIsActive())
1265 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DELETEINSERTCHARFORMAT, "Invariant" );
1267 __ASSERT_DEBUG(InsertCharFormatIsActive(), User::Invariant());
1268 ScanToPosition(iPendingNewPhrasePos,EScanToPositionMatchLeft);
1269 TCurrentIndexRecords current;
1270 GetCurrentRecords(current);
1271 if (current.iPhrase && current.iPhrase->Length() == 0)
1273 RemoveFromPhraseIx(iPos.iPhraseElement);
1274 current.iParaAttribs->iPhraseCount--; // Para has 1 less phrase in it now.
1280 /** Attempts to merge phrases and share paragraphs.
1282 Phrase boundary here is merged if possible, paragraph here is shared if
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.
1288 void CRichTextIndex::ConsolidateAt(TInt aPosition,
1289 TPositionOrPhrase aPositionOrPhrase)
1291 ScanToPosition(aPosition, EScanToPositionAbsolute);
1292 TCurrentIndexRecords current;
1293 GetCurrentRecords(current);
1294 if (!current.iPhrase)
1297 TInt length = current.iPhrase->Length();
1298 MergePhrases(aPosition);
1299 if (aPositionOrPhrase == EFollowingPhrase)
1301 ScanToPosition(aPosition, EScanToPositionAbsolute);
1302 GetCurrentRecords(current);
1303 if (current.iPhrase)
1304 MergePhrases(aPosition + length);
1306 Normalize(aPosition);
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.
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);
1330 TIndexDeleteInfo deleteInfo;
1331 SetForDeleteL(deleteInfo,aPos,aLength);
1332 TBool parasMerged=DeleteNow(deleteInfo);
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.
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();
1361 currentParaAttribs->iParaFormat->SenseL(pf,applyMask);
1362 currentParaAttribs->iParaFormat->SetL(pf,applyMask);
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);
1376 CParaAttribs* sharedParaAttribs=GetParaAttribsL(newParaLayer,newCharLayer);
1377 CleanupStack::PopAndDestroy(2);
1378 if (sharedParaAttribs)
1379 (*iParaIx)[paraItem].iParaAttribs=sharedParaAttribs;
1380 currentParaAttribs->Release();
1383 CleanupStack::PopAndDestroy(); // pf
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.
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();
1410 currentParaAttribs.iParaFormat->SetBase(&aStyle);
1411 TInt phraseCount=currentParaAttribs.PhraseCount();
1412 for (TInt phraseItem=0;phraseItem<phraseCount;phraseItem++)
1414 CCharFormatLayer& charLayer=*(*iPhraseIx)[paragraphBasePhrase+phraseItem].CharFormat();
1415 if (type==KNormalParagraphStyleUid)
1416 charLayer.SetBase(aCharStyleNormal);
1418 charLayer.SetBase(aStyle.CharFormatLayer());
1419 ModifySpecificFormatting(*currentParaAttribs.iParaFormat,charLayer,aMode);
1421 paragraphBasePhrase+=phraseCount;
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();
1430 newParaLayer=CParaFormatLayer::NewL(currentParaAttribs.iParaFormat);
1431 newParaLayer->SetBase(&aStyle);
1432 CleanupStack::PushL(newParaLayer);
1434 // Make a new char format layer
1435 CCharFormatLayer* newCharLayer=NULL;
1436 if (aMode==CParagraphStyle::ERetainNoSpecificFormats || aMode==CParagraphStyle::ERetainSpecificParaFormat)
1437 newCharLayer=CCharFormatLayer::NewL();
1439 newCharLayer=CCharFormatLayer::NewL(currentParaAttribs.iCharFormat);
1440 if (type==KNormalParagraphStyleUid)
1441 newCharLayer->SetBase(aCharStyleNormal);
1443 newCharLayer->SetBase(aStyle.CharFormatLayer());
1444 CleanupStack::PushL(newCharLayer);
1446 (*iParaIx)[paraItem].iParaAttribs=GetParaAttribsL(newParaLayer,newCharLayer);
1447 CleanupStack::PopAndDestroy(2);
1448 currentParaAttribs.Release();
1454 void CRichTextIndex::ModifySpecificFormatting(CParaFormatLayer& aPl,CCharFormatLayer& aCl,CParagraphStyle::TApplyParaStyleMode aMode)
1460 case(CParagraphStyle::ERetainNoSpecificFormats):
1464 case(CParagraphStyle::ERetainSpecificParaFormat):
1467 case(CParagraphStyle::ERetainSpecificCharFormat):
1470 case(CParagraphStyle::ERetainAllSpecificFormats):
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.
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.
1486 void CRichTextIndex::NotifyStyleChangedL(const CParagraphStyle* aTo,const CParagraphStyle* aFrom,
1487 const CParaFormatLayer& aGlobalParaFormatLayer,
1488 const CCharFormatLayer& aGlobalCharFormatLayer)
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++)
1499 TParaAttribsEntry* currentPara = &(*iParaIx)[paraItem];
1500 CParaAttribs* currentParaAttribs = currentPara->iParaAttribs;
1502 TBool changeParStyleBase = currentParaAttribs->iParaFormat->SenseBase() == aFrom;
1503 if (!currentParaAttribs->IsShared())
1505 if (changeParStyleBase)
1506 currentParaAttribs->iParaFormat->SetBase(newParFormatLayer);
1507 TInt phraseCount = currentParaAttribs->PhraseCount();
1508 for (TInt phraseItem = currentPhrase; phraseItem < (currentPhrase + phraseCount); phraseItem++)
1510 CCharFormatLayer* charFormat = (*iPhraseIx)[phraseItem].CharFormat();
1511 if (charFormat->SenseBase() == oldCharFormatLayer)
1512 charFormat->SetBase(newCharFormatLayer);
1514 currentPhrase += phraseCount;
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;
1537 const CParaFormatLayer* CRichTextIndex::ParagraphStyle(TBool& aStyleChangesOverRange,
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.
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();
1554 for (;para<=endPara;para++)
1556 CParaFormatLayer* nextStyle=(CParaFormatLayer*)(*iParaIx)[para].iParaAttribs->iParaFormat->SenseBase();
1557 if (nextStyle!=style)
1558 aStyleChangesOverRange=ETrue;
1566 void CRichTextIndex::SplitPhraseL(TInt aPhrase,TInt anOffset,RPhraseAttribsEntry& aPhraseAttribs,CParaAttribs& aParaAttribs)
1568 if (anOffset<=0 || anOffset>=aPhraseAttribs.Length())
1570 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SPLITPHRASEL, "Invariant" );
1572 __ASSERT_DEBUG(anOffset>0 && anOffset<aPhraseAttribs.Length(),User::Invariant());
1574 CCharFormatLayer* charLayer=CCharFormatLayer::NewCopyBaseL(aPhraseAttribs.CharFormat());
1575 CleanupStack::PushL(charLayer);
1576 iPhraseIx->InsertL(aPhrase+1,RPhraseAttribsEntry(charLayer,aPhraseAttribs.Length()-anOffset));
1577 CleanupStack::Pop();
1579 // InsertL() has invalidated the phrase index.
1580 iPhraseIx->At(aPhrase).SetLength(anOffset); // Adjust the length of the orginal phrase
1581 aParaAttribs.iPhraseCount++;
1585 TBool CRichTextIndex::MergePhrases(TInt aPhrase,RPhraseAttribsEntry& aPhraseAttribs,CParaAttribs& aParaAttribs)
1587 RPhraseAttribsEntry& prevPhrase=iPhraseIx->At(aPhrase-1);
1588 if (!aPhraseAttribs.IsIdentical(prevPhrase))
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--;
1598 void CRichTextIndex::Share(TParaAttribsEntry& aParaEntry,TInt aPhrase)
1600 // aParaEntry is not shared and can be (phrase count 1), aPhrase is the single phrase element
1603 CParaAttribs* paraAttribs=aParaEntry.iParaAttribs;
1604 if (paraAttribs->iPhraseCount!=1)
1606 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SHARE, "Invariant" );
1608 __ASSERT_DEBUG(paraAttribs->iPhraseCount==1,User::Invariant());
1610 RPhraseAttribsEntry& phraseAttribs=iPhraseIx->At(aPhrase);
1611 if (phraseAttribs.IsPicturePhrase())
1613 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_SHARE, "Invariant" );
1615 __ASSERT_DEBUG(!phraseAttribs.IsPicturePhrase(),User::Invariant());
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;
1624 iPhraseIx->Delete(aPhrase);
1628 void CRichTextIndex::ApplyCharFormatCleanup(TAny* aPtr)
1629 // CLeanup function for ApplyCharFormatL()
1631 {REINTERPRET_CAST(CRichTextIndex*,aPtr)->ApplyCharFormatRollback();}
1634 void CRichTextIndex::ApplyCharFormatRollback()
1635 // Paragraph and phrase we were working on are stored in iPos
1636 // Return them the canonical form
1639 TParaAttribsEntry& paraEntry=iParaIx->At(iPos.iParaElement);
1640 CParaAttribs* paraAttribs=paraEntry.iParaAttribs;
1642 if (paraAttribs->IsShared())
1645 TInt phrase=iPos.iPhraseElement;
1646 TInt base=iPos.iParaBasePhraseElement;
1647 if (phrase<base || phrase>=base+paraAttribs->iPhraseCount)
1649 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPLYCHARFORMATROLLBACK, "Invariant" );
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);
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).
1668 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1670 __ASSERT_DEBUG(aLength>=0,User::Invariant());
1673 ScanToPosition(aPos,EScanToPositionAbsolute);
1674 TInt paraOffset=iPos.iParaElementOffset;
1675 TInt phraseOffset=iPos.iPhraseElementOffset;
1676 TInt phrase=iPos.iPhraseElement;
1678 // prepare for failure
1679 CleanupStack::PushL(TCleanupItem(ApplyCharFormatCleanup,this));
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;
1688 aPos+=charsToFormat;
1691 // STEP 1. Reclaim any shared paragraph into non shared form. Re-use the object if possible
1693 if (paraAttribs->IsShared())
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;
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;
1715 phraseOffset=paraOffset; // we are now in the current position
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
1723 if (phrase>=iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount)
1725 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1727 __ASSERT_DEBUG(phrase<iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount,User::Invariant());
1729 RPhraseAttribsEntry* phraseAttribs=&iPhraseIx->At(phrase);
1730 TInt len=phraseAttribs->Length();
1732 // STEP 2.1 Split the phrase at the beginning of the range?
1735 { // can only happen for the first phrase
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.
1741 // coverity[leave_without_push]
1742 SplitPhraseL(phrase,phraseOffset,*phraseAttribs,*paraAttribs); // inserts new phrase at correct position
1745 iPos.iPhraseElement=++phrase;
1746 phraseAttribs=&iPhraseIx->At(phrase);
1749 // STEP 2.2 Split the phrase at the end of the range?
1751 if (len>charsToFormat)
1752 { // phrase is longer than required format, so split it
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.
1758 // coverity[leave_without_push]
1759 SplitPhraseL(phrase,charsToFormat,*phraseAttribs,*paraAttribs);
1761 phraseAttribs=&iPhraseIx->At(phrase); // SplitPhraseL modifies the index array, we must do this!
1764 if (phraseAttribs->Length()!=len)
1766 OstTrace0( TRACE_DUMP, DUP2_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1768 __ASSERT_DEBUG(phraseAttribs->Length()==len,User::Invariant());
1770 // STEP 2.3 Change the format of the current phrase layer
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);
1779 // STEP 2.4 Check for merging with previous phrase
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;
1786 } while (charsToFormat);
1788 if (phrase!=iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount && aLength!=0)
1790 OstTrace0( TRACE_DUMP, DUP3_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1792 __ASSERT_DEBUG(phrase==iPos.iParaBasePhraseElement+paraAttribs->iPhraseCount || aLength==0,User::Invariant());
1794 // STEP 3 Reduce the paragraph attributes back to canonical form
1796 // STEP 3.1 Check for merging at the end of the changes
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
1801 // STEP 3.2 See if we can re-share the paragraph
1803 if (paraAttribs->iPhraseCount==1)
1804 { // This para has constant character formatting - can be shared.
1805 iPos.iPhraseElement=--phrase;
1806 Share(paraEntry,phrase);
1809 // loop into next paragraph
1812 iPos.iParaElement++;
1814 iPos.iParaBasePhraseElement=phrase;
1816 ScanToPosition(aPos,EScanToPositionAbsolute);
1817 if (iPos.iDocPos!=aPos)
1819 OstTrace0( TRACE_DUMP, DUP4_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1821 __ASSERT_DEBUG(iPos.iDocPos==aPos,User::Invariant());
1822 if (iPos.iPhraseElement!=phrase)
1824 OstTrace0( TRACE_DUMP, DUP5_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1826 __ASSERT_DEBUG(iPos.iPhraseElement==phrase,User::Invariant());
1827 if (iPos.iParaElementOffset!=paraOffset)
1829 OstTrace0( TRACE_DUMP, DUP6_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1831 __ASSERT_DEBUG(iPos.iParaElementOffset==paraOffset,User::Invariant());
1832 if (iPos.iPhraseElementOffset!=phraseOffset)
1834 OstTrace0( TRACE_DUMP, DUP7_CRICHTEXTINDEX_APPLYCHARFORMATL, "Invariant" );
1836 __ASSERT_DEBUG(iPos.iPhraseElementOffset==phraseOffset,User::Invariant());
1841 CleanupStack::Pop(); // rollback item
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.
1858 TInt endPara=OwningParagraph(aPos+aLength);
1859 TInt currentPara=OwningParagraph(aPos);
1860 CParaAttribs* currentParaAttribs=NULL;
1861 while (currentPara<=endPara)
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.
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;
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.
1899 TCharFormatX format; // dummy format
1900 TCharFormatXMask mask;
1902 ApplyCharFormatL(format,mask,aPos,aLength,ETrue);
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.
1915 ScanToPosition(aPos,EScanToPositionAbsolute);
1916 return MergePhrases(iPos);
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.
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;
1944 return phrasesMerged;
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
1953 void CRichTextIndex::RemoveFromPhraseIx(TInt aPhraseIndex,TInt aCount)
1955 // if the phrase being deleted is <= iLastPos
1956 // then iLastPos will become invalid so should be reset
1957 if (aPhraseIndex <= iLastUsed.iPhraseElement )
1960 for (TInt offset=aPhraseIndex;offset<(aPhraseIndex+aCount);offset++)
1962 // discard phrases & book-keep the picture count
1963 RPhraseAttribsEntry& phrase=(*iPhraseIx)[offset];
1965 if (phrase.IsPicturePhrase())
1968 iPhraseIx->Delete(aPhraseIndex,aCount);
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.
1978 TLogicalPosition cachePos(iLastUsed);
1979 TInt para=CONST_CAST(CRichTextIndex*,this)->OwningParagraph(aPos,&cachePos);
1980 (*iParaIx)[para].iParaAttribs->iParaFormat->SenseEffectiveL(aFormat);
1983 void CRichTextIndex::GetSpecificParagraphFormatL(CParaFormat* aFormat,
1984 TParaFormatMask& aMask,
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);
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.
2002 CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
2003 TCurrentIndexRecords current;
2004 GetCurrentRecords(current);
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);
2013 {// Constant character formatting held in the para attribs
2014 charFormatLayer=current.iParaAttribs->iCharFormat;
2015 phraseLength=current.iParaEntry->iLength-iPos.iParaElementOffset;
2017 charFormatLayer->SenseEffective(aFormat);
2018 return phraseLength;
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.
2030 CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
2031 TCurrentIndexRecords current;
2032 GetCurrentRecords(current);
2034 return (current.iPhrase)
2035 ? current.iPhrase->GetPictureSizeInTwips(aSize)
2039 TPictureHeader* CRichTextIndex::PictureHeaderPtr(TInt aPos)
2043 CONST_CAST(CRichTextIndex*,this)->ScanToPosition(aPos,EScanToPositionAbsolute);
2044 TCurrentIndexRecords current;
2045 GetCurrentRecords(current);
2046 return current.iPhrase? current.iPhrase->PictureHeaderPtr() : 0;
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.
2054 const TPictureHeader* p = const_cast<CRichTextIndex*>(this)->PictureHeaderPtr(aPos);
2055 return p? *p : TPictureHeader();
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.
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)
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.
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));
2088 (*iParaIx)[para].iParaAttribs->iParaFormat->SenseEffectiveL(aFormat,aMode); // Sense 1st paras' format.
2090 CParaFormat* format=CParaFormat::NewLC();
2091 for (;para<=endPara;para++)
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)
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));
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);
2143 if (format->TabCount()!=aFormat->TabCount())
2144 aVaries.SetAttrib(EAttTabStop); // TabLists are different.
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);
2158 aVaries.SetAttrib(EAttTabStop);
2162 CleanupStack::PopAndDestroy();
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
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);
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.
2214 if (aLength==0) // Get the format of the character to the left of the cursor.
2215 ((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionMatchLeft);
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);
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);
2235 current.iParaAttribs->iCharFormat->SenseEffective(format);
2236 CheckForUndetermined(format,aFormat,aVaries);
2237 pos+=CurrentPhraseLength();
2242 void CRichTextIndex::GetSpecificCharFormatDirection(TCharFormatX& aFormat,
2243 TCharFormatXMask& aMask,
2245 TBool aGetLeft) const
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);
2257 current.iParaAttribs->iCharFormat->Sense(aFormat,aMask);
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.
2265 GetSpecificCharFormatDirection(aFormat, aMask, aPos, ETrue);
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.
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*).
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;
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
2303 // aCharFormat may be NULL, as may aReservedCell.
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;
2311 charFormat=aCharFormat; // Has a phrase index.
2313 charFormat=aParaAttribs->iCharFormat; // Has constant char formatting.
2314 return GetParaAttribsL(aParaAttribs->iParaFormat,charFormat,aReservedCell);
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.
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);
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.
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.
2344 CParaAttribs* handle=FindSharedParaAttribs(*aParaFormat,*aCharFormat);
2346 return handle; // match found already in the shared list.
2347 // There is no match, so create new sharedPara and add to list.
2349 aReservedCell=CParaAttribs::NewL(aParaFormat,aCharFormat); // Reusing aReservedCell saves an automatic.
2350 iSharedParaQueHead.AddLast(*aReservedCell);
2351 return aReservedCell;
2355 CParaAttribs* CRichTextIndex::GetParaAttribs(CParaAttribs* aParaAttribs,CCharFormatLayer& aCharFormatLayer)
2359 CParaAttribs* handle=FindSharedParaAttribs(*aParaAttribs->iParaFormat,aCharFormatLayer);
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;
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.
2379 CParaAttribs* sourceParaAttribs=(*iParaIx)[aLogicalPosition.iParaElement].iParaAttribs;
2380 RPhraseAttribsEntry* sourcePhrase=&(*iPhraseIx)[aLogicalPosition.iPhraseElement];
2382 CParaAttribs* handle=FindSharedParaAttribs(*sourceParaAttribs->iParaFormat,*sourcePhrase->CharFormat());
2386 {// No match, so piece together new shared paraAttribs and add to shared para list
2387 sourceParaAttribs->iRefCount=1;
2388 if (sourcePhrase->IsPicturePhrase())
2390 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_GETPARAATTRIBS, "EReleasCharFormatLayerOwnershipCalledOnPicturePhrase" );
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;
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.
2409 CParaAttribs* currentSharedPara=NULL;
2410 TBool matched=EFalse;
2411 if (!iSharedParaQueHead.IsEmpty())
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);
2419 matched=aCharFormatLayer.IsIdentical(currentSharedPara->iCharFormat);
2422 // We have a match, so adjust reference count.
2423 currentSharedPara->iRefCount++;
2424 return currentSharedPara;
2427 return currentSharedPara;
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
2439 // The implementation below matches to the right as standard, then checks if a left
2440 // phrase match is available.
2443 if (!aLastUsed || aLastUsed->iDocPos-aLastUsed->iParaElementOffset>aCharPos)
2444 iPos.Clear(); // Reset the internal position record.
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;
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=¶Ix[paraElement];
2461 TInt len=para->iLength;
2462 if (aCharPos>=(startOfPara+len))
2464 iPos.iParaElementOffset=iPos.iPhraseElementOffset=0;
2465 phraseElement=iPos.iParaBasePhraseElement;
2466 const TParaAttribsEntry* end=paraIx.End(paraElement);
2468 {// Find the paragraph...
2470 if (!(para->iParaAttribs->IsShared())) // Adjust position within phrase index
2471 phraseElement+=para->iParaAttribs->iPhraseCount;
2475 para=¶Ix[paraElement];
2476 end=paraIx.End(paraElement);
2480 while (aCharPos>=(startOfPara+len) && (paraElement+1) < paraIx.Count());
2481 iPos.iParaBasePhraseElement=phraseElement;
2482 iPos.iParaElement=paraElement;
2484 TInt startOfPhrase=iPos.iParaElementOffset-iPos.iPhraseElementOffset;
2485 // the offset within the paragraph.
2486 TInt paraElementOffset=aCharPos-startOfPara;
2487 iPos.iParaElementOffset=paraElementOffset;
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;
2496 { // Find the phrase in the paragraph...
2497 len=phrase->Length();
2498 if (paraElementOffset<(startOfPhrase+len))
2501 lastPhraseLength=len;
2503 end=phraseIx.End(phraseElement);
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.
2514 iPos.iPhraseElementOffset=lastPhraseLength;
2517 iPos.iPhraseElementOffset=paraElementOffset-startOfPhrase;
2521 if (iPos.iParaBasePhraseElement!=phraseElement)
2523 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_SCANTOPOSITION, "EDebug" );
2525 __ASSERT_DEBUG(iPos.iParaBasePhraseElement==phraseElement,Panic(EDebug));
2527 iPos.iPhraseElement=phraseElement;
2528 iPos.iDocPos=aCharPos;
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.
2540 {return iPos.iPhraseElement==iPos.iParaBasePhraseElement;}
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.
2548 if ((*iParaIx)[iPos.iParaElement].iParaAttribs->IsShared())
2549 return (*iParaIx)[iPos.iParaElement].iLength;
2551 return (*iPhraseIx)[iPos.iPhraseElement].Length();
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.
2560 if ((*iParaIx)[iPos.iParaElement].iParaAttribs->IsShared())
2561 return iPos.iParaElementOffset; // only 1 phrase in para.
2563 return iPos.iPhraseElementOffset;
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.
2573 aRecord.iParaEntry=&(*iParaIx)[iPos.iParaElement];
2574 aRecord.iParaAttribs=aRecord.iParaEntry->iParaAttribs;
2575 if (aRecord.iParaAttribs->IsShared())
2576 aRecord.iPhrase=NULL;
2578 aRecord.iPhrase=&((*iPhraseIx)[iPos.iPhraseElement]);
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.
2590 CCharFormatLayer* charFormatLayer=NULL;
2591 if (aCurrent.iPhrase)
2592 {// Specific character formatting held by phrase index.
2593 charFormatLayer=aCurrent.iPhrase->CharFormat();
2596 {// Constant character formatting held in the para attribs
2597 charFormatLayer=aCurrent.iParaAttribs->iCharFormat;
2599 charFormatLayer->Sense(aFormat,aMask);
2600 aCharBase=(CCharFormatLayer*)(charFormatLayer->SenseBase());
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.
2609 ((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionMatchLeft,aLastUsed);
2610 return iPos.iParaElement;
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).
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)
2631 OstTrace0( TRACE_FATAL, DUP1_CRICHTEXTINDEX_SPLITPHRASEL, "ESplitPhraseCalledOnSharedPara" );
2633 __ASSERT_ALWAYS(current.iPhrase!=NULL,Panic(ESplitPhraseCalledOnSharedPara));
2634 DoSplitPhraseL(*current.iPhrase,iPos.iPhraseElementOffset,current.iParaAttribs);
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).
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);
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.
2661 // ASSERT: Cannot split a picture phrase.
2662 if (aCurrentPhrase.IsPicturePhrase())
2664 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DOSPLITPHRASEL, "ESplitPhraseCalledOnPicturePhrase" );
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();
2673 // InsertL() has invalidated the current internal position record.
2674 iPhraseIx->At(iPos.iPhraseElement).SetLength(aPhraseOffset);
2675 SetPhraseSplit(ETrue);
2676 aParaAttribs->iPhraseCount++;
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
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)
2692 TInt phraseCount=PhraseCount();
2696 TInt sharedParaCount=SharedParaCount(this);
2697 if (sharedParaCount<1 && (sharedParaCount!=0 || phraseCount<=0))
2699 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_HASMARKUPDATA, "ERichTextIndexIntegrityErr" );
2701 __ASSERT_ALWAYS(sharedParaCount>=1 || (sharedParaCount==0 && phraseCount>0),Panic(ERichTextIndexIntegrityErr));
2702 if (sharedParaCount>1)
2704 const CParaAttribs* paraAttribs=iSharedParaQueHead.First();
2705 if (!paraAttribs->iParaFormat->IsEmpty())
2707 if (!paraAttribs->iCharFormat->IsEmpty())
2709 if (paraAttribs->iParaFormat->SenseBase()!=aGlobalParaFormatLayer)
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.
2720 TInt sharedParaCount = 0;
2721 TDblQueIter<CParaAttribs> iterator( MUTABLE_CAST(TDblQue<CParaAttribs>&, aSource->iSharedParaQueHead) );
2722 while ( iterator++ != NULL )
2724 return sharedParaCount;
2728 void CRichTextIndex::AppendTakingSolePictureOwnershipL(const CRichTextIndex* aSource,const TGlobalLayerInfoAppend& aGlobalLayerInfo)
2729 // No paragraph style information is appended.
2732 CancelInsertCharFormat();
2733 CONST_CAST(CRichTextIndex*,aSource)->CancelInsertCharFormat();
2735 TInt origParaCount=ParagraphCount();
2736 TInt origPhraseCount=iPhraseIx->Count();
2738 AppendParaIndexL(aSource,aGlobalLayerInfo);
2739 AppendPhraseIndexL(aSource,aGlobalLayerInfo);
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
2753 void CRichTextIndex::AppendParaIndexL(const CRichTextIndex* aSource,const TGlobalLayerInfoAppend& aGlobalLayerInfo)
2756 CRichTextStoreMap<CParaAttribs>* map=CRichTextStoreMap<CParaAttribs>::NewLC(SharedParaCount(aSource));
2758 AppendSharedFormatsL(*map,aSource,aGlobalLayerInfo);
2760 // Extend para index by required amount
2761 TInt originalParaCount=iParaIx->Count();
2762 TInt requiredParaCount=aSource->iParaIx->Count();
2763 iParaIx->AppendL(TParaAttribsEntry(),requiredParaCount);
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())
2772 tParaAttribs=map->Item(sParaAttribs);
2773 if (tParaAttribs==NULL)
2775 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPENDPARAINDEXL, "ESharedFormatsMapIntegrityError" );
2777 __ASSERT_DEBUG(tParaAttribs!=NULL,Panic(ESharedFormatsMapIntegrityError));
2778 tParaAttribs->iRefCount++;
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;
2786 TParaAttribsEntry& tParaEntry=(*iParaIx)[originalParaCount+ii];
2787 tParaEntry.iLength=sParaEntry.iLength;
2788 tParaEntry.iParaAttribs=tParaAttribs;
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]
2796 CleanupStack::PopAndDestroy(); // map
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.
2806 TDblQueIter<CParaAttribs> iterator(MUTABLE_CAST(TDblQue<CParaAttribs>&,aSource->iSharedParaQueHead));
2807 CParaAttribs* currentSharedPara;
2808 while ((currentSharedPara=iterator++)!=NULL)
2810 if (!currentSharedPara->IsShared())
2812 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPENDSHAREDFORMATSL, "Invariant" );
2814 __ASSERT_DEBUG(currentSharedPara->IsShared(),User::Invariant());
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)
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);
2831 newParaAttribs->iRefCount--; // we have not yet linked incoming para - taken out no shares just yet
2832 aMap.Bind(currentSharedPara,newParaAttribs);
2836 void CRichTextIndex::AppendPhraseIndexL(const CRichTextIndex* aSource,const TGlobalLayerInfoAppend& aGlobalLayerInfo)
2839 TInt originalPhraseCount=iPhraseIx->Count();
2840 TInt requiredPhraseCount=aSource->iPhraseIx->Count();
2842 // Extend phrase index by required amount
2843 iPhraseIx->AppendL(RPhraseAttribsEntry(),requiredPhraseCount);
2845 CArrayFixFlat<TInt>* pictureMap=new(ELeave) CArrayFixFlat<TInt>(16);
2846 CleanupStack::PushL(pictureMap);
2848 for (TInt jj=0;jj<requiredPhraseCount;jj++)
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());
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);
2869 pictureMap->AppendL(jj);
2873 // transfer pictures now
2874 for (TInt kk=pictureMap->Count();--kk>=0;)
2876 TInt jj=pictureMap->At(kk);
2877 TPictureHeader* sHeader=(*aSource->iPhraseIx)[jj].PictureHeaderPtr();
2880 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_APPENDPHRASEINDEXL, "Invariant" );
2882 __ASSERT_DEBUG(sHeader,User::Invariant());
2883 if (sHeader->iPicture.IsPtr())
2884 { // transfer picture to us
2885 TPictureHeader* tHeader=(*iPhraseIx)[jj+originalPhraseCount].PictureHeaderPtr();
2888 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_APPENDPHRASEINDEXL, "Invariant" );
2890 __ASSERT_DEBUG(tHeader,User::Invariant());
2891 tHeader->iPicture=sHeader->iPicture.AsPtr();
2892 sHeader->iPicture=NULL;
2896 CleanupStack::PopAndDestroy(); // pictureMap
2900 void CRichTextIndex::AppendParagraphL(const CParaFormatLayer* aGlobalParaFormatLayer,
2901 const CCharFormatLayer* aGlobalCharFormatLayer,
2903 // Append aReplicas empty paragraphs, the format of which is based on the
2904 // global format layers.
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.
2918 // add the paragraph records
2919 TParaAttribsEntry paraEntry(1,paUsed);
2920 iParaIx->AppendL(paraEntry,aReplicas);
2921 if (paUsed!=paraAttribs)
2922 CleanupStack::PopAndDestroy();
2924 CleanupStack::Pop();
2926 // set the ref count of paUsed
2927 paUsed->iRefCount+=(aReplicas-1); // compensate for already adding 1 share.
2933 TParaAttribsEntry::TParaAttribsEntry():
2940 TParaAttribsEntry::TParaAttribsEntry(TInt aLength,CParaAttribs* aParaAttribs):
2942 iParaAttribs(aParaAttribs)
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.
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;
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.
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;
2980 CParaAttribs* CParaAttribs::NewL(const CParaAttribs* aParaAttribs)
2982 return NewL(aParaAttribs->iParaFormat,aParaAttribs->iCharFormat);
2986 CParaAttribs::CParaAttribs():
2987 iRefCount(-1) // Ensures Destruct works correctly when called on a semi-initialised object.
2992 void CParaAttribs::Release()
2993 // Release a share on this CParaAttribs.
2994 // If after this, no shares remain, destroy this CParaAttribs.
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.
3014 CParaAttribs::~CParaAttribs()
3015 // Release the memory associated with this object.
3020 {// Constant character formatting - in the shared list.
3022 link.Deque(); // Remove this para attribs from the shared list.
3027 TInt CParaAttribs::PhraseCount()const
3028 // Return a count of the number of phrases in this para attribs.
3030 {return (iRefCount>=1)?1:iPhraseCount;}
3033 DLLEXPORT_C void RPhraseAttribsEntry::__DbgTestInvariant()const
3034 // Class invariants.
3038 // ASSERT: iLength is +ve (applying to character formatting, or is set to indicate a picture phrase.
3039 if (iLength<0 && !IsPicturePhrase())
3041 OstTrace0( TRACE_DUMP, RPHRASEATTRIBSENTRY_DBGTESTINVARIANT, "Invariant" );
3043 __ASSERT_DEBUG(iLength>=0 || IsPicturePhrase(),User::Invariant());
3048 RPhraseAttribsEntry::RPhraseAttribsEntry():
3055 RPhraseAttribsEntry::RPhraseAttribsEntry(CCharFormatLayer* aCharFormat,TInt aLength):
3057 iCharFormat(aCharFormat)
3062 RPhraseAttribsEntry::RPhraseAttribsEntry(CPicturePhrase* aPicturePhrase):
3063 iLength(EPictureIndicator),
3064 iPicturePhrase(aPicturePhrase)
3069 void RPhraseAttribsEntry::AssignAndRelease(const RPhraseAttribsEntry& aPhrase)
3070 // Assign the state of the specified object to this,
3073 iLength=aPhrase.iLength;
3074 iCharFormat=aPhrase.iCharFormat; // both union members share the same address space, so this is fine.
3078 void RPhraseAttribsEntry::Discard()
3084 if (iLength==EPictureIndicator)
3085 delete iPicturePhrase;
3091 void RPhraseAttribsEntry::ExternalizeL(RWriteStream& aStream)const
3092 // Save this phrase into aStream.
3095 TUint8 picIndicator=(TUint8)(IsPicturePhrase()!=EFalse);
3096 aStream.WriteUint8L(picIndicator);
3097 aStream.WriteInt32L(Length());
3098 aStream<< *CharFormat();
3099 if ((TBool)picIndicator)
3100 aStream<< *PictureHeaderPtr();
3104 CCharFormatLayer* RPhraseAttribsEntry::CharFormat()const
3105 // Returns a pointer the CCharFormatLayer of this phrase.
3107 {return (iLength==EPictureIndicator)?iPicturePhrase->iCharFormat:iCharFormat;}
3110 void RPhraseAttribsEntry::SetLength(TInt aLength)
3111 // Sets the phrase length.
3120 void RPhraseAttribsEntry::AdjustLength(TInt aIncrement)
3121 // Adjusts the length of the phrase by the signed value aIncrement.
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) )
3129 OstTrace0( TRACE_DUMP, RPHRASEATTRIBSENTRY_ADJUSTLENGTH, "EModifiedPicturePhraseLength" );
3131 __ASSERT_DEBUG(!IsPicturePhrase() || (IsPicturePhrase() && aIncrement==EPictureIndicator)
3132 || (IsPicturePhrase() && aIncrement==0)
3133 ,Panic(EModifiedPicturePhraseLength));
3135 iLength=(len==EPictureIndicator)
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.
3146 if (!IsPicturePhrase())
3147 return KErrNotFound;
3148 if (iPicturePhrase->iPicHdr.iPicture.IsPtr() && iPicturePhrase->iPicHdr.iPicture.AsPtr())
3149 iPicturePhrase->iPicHdr.iPicture->GetSizeInTwips(aSize);
3151 aSize=iPicturePhrase->iPicHdr.iSize;
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.
3161 return (IsPicturePhrase())
3162 ? iPicturePhrase->iPicHdr
3167 const CPicture* RPhraseAttribsEntry::PictureHandleL(const MPictureFactory* aFactory,
3168 const MRichTextStoreResolver* aResolver,
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.
3178 if (iLength!=EPictureIndicator)
3181 {// Check if the picture is in memory
3182 CPicture* handle=NULL;
3183 if (iPicturePhrase->iPicHdr.iPicture.IsPtr())
3184 handle=iPicturePhrase->iPicHdr.iPicture;
3186 {// Check if the picture should be loaded from persistent storage
3187 if (aForceLoad==MLayDoc::EForceLoadTrue) // picture not in memory, so load it.
3189 if (aResolver==NULL || aFactory==NULL)
3191 const CStreamStore& store=aResolver->StreamStoreL(aPos);
3192 aFactory->NewPictureL(iPicturePhrase->iPicHdr,store);
3193 handle=iPicturePhrase->iPicHdr.iPicture;
3201 TPictureHeader* RPhraseAttribsEntry::PictureHeaderPtr()const
3202 // For a [text] phrase returns NULL.
3203 // For a picture phrase returns a pointer to the picture header.
3205 {return (iLength==EPictureIndicator)? &iPicturePhrase->iPicHdr:NULL;}
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.
3213 CCharFormatLayer*& layer=(iLength==EPictureIndicator)?iPicturePhrase->iCharFormat:iCharFormat;
3214 CCharFormatLayer* charFormatLayer=layer;
3216 return charFormatLayer;
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.
3226 if (iLength<=0) // picture or zero-length
3228 if (aPhrase.iLength<=0) // picture or zero-length
3230 return iCharFormat->IsIdentical(aPhrase.iCharFormat,EFalse); // EFalse=do not compare based-on link
3234 CPicturePhrase* CPicturePhrase::NewL(const TPictureHeader& aPicHdr,TCharFormatX& aFormat,
3235 TCharFormatXMask& aMask,CCharFormatLayer* aCharBase,
3236 TBool& aPictureOwnershipTaken)
3238 CPicturePhrase* self=new(ELeave) CPicturePhrase(aPicHdr,aPictureOwnershipTaken);
3239 CleanupStack::PushL(self);
3240 self->ConstructL(aFormat,aMask,aCharBase);
3241 CleanupStack::Pop();
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)
3252 return new(ELeave) CPicturePhrase(aPicHdr,aCharLayer,aPictureOwnershipTaken);
3256 CPicturePhrase::CPicturePhrase(const TPictureHeader& aPicHdr,TBool& aPictureOwnershipTaken):
3259 aPictureOwnershipTaken=ETrue;
3263 CPicturePhrase::CPicturePhrase(const TPictureHeader& aPicHdr,
3264 CCharFormatLayer* aCharLayer,
3265 TBool& aPictureOwnershipTaken):
3266 iCharFormat(aCharLayer),
3269 aPictureOwnershipTaken=ETrue;
3273 void CPicturePhrase::ConstructL(TCharFormatX& aFormat,TCharFormatXMask& aMask,CCharFormatLayer* aCharBase)
3275 iCharFormat=CCharFormatLayer::NewL(aFormat,aMask);
3276 iCharFormat->SetBase(aCharBase);
3280 CPicturePhrase::~CPicturePhrase()
3282 iPicHdr.DeletePicture();
3286 TLogicalPosition::TLogicalPosition():
3289 iParaElementOffset(0),
3291 iPhraseElementOffset(0)
3296 void TLogicalPosition::Clear()
3297 // Reset this logical position record.
3300 *this=TLogicalPosition();
3301 iParaBasePhraseElement=0;