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 "TXTIXSTRTraces.h"
35 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
36 #include "TXTFMLYR_INTERNAL.H"
39 const TUint8 KRegisterItemPresent=1;
42 GLDEF_C void DiscardPhraseOnCleanup(TAny* aPhrase)
43 // Allows RPhraseAttribsEntry on the CleanupStack to be destroyed correctly,
44 // when included as a PushL(TCleanupItem).
47 ((RPhraseAttribsEntry*)aPhrase)->Discard();
52 TRtPasteContext::TRtPasteContext(const CStreamStore* aStore,const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,const CStyleList* aStyleList)
55 iGlobalParaLayer(aGlobalParaLayer),iGlobalCharLayer(aGlobalCharLayer),iStyleList(aStyleList),
56 iParagraphStylesFlag(EFalse),iIncompleteParaFlag(ETrue),iPastePos(),iParasPasted(0)
59 //////////////////////////////////////////////////////////////////////////////////////
62 CRichTextIndex* CRichTextIndex::NewL(const CStreamStore& aStore,TStreamId aId,
63 // MPictureFactory* aPictureFactory,MRichTextStoreResolver* aStoreResolver,
64 const CParaFormatLayer* aGlobalParaLayer,
65 const CCharFormatLayer* aGlobalCharLayer,
66 const CRichText& aText)
67 // const CStyleList* aStyleList)
70 CRichTextIndex* self=new(ELeave) CRichTextIndex(aText);
71 CleanupStack::PushL(self);
72 self->ConstructL(aGlobalParaLayer,aGlobalCharLayer,KMultiParaGranularity,KLargePhraseGranularity);
73 self->RestoreL(aStore,aId,aGlobalParaLayer,aGlobalCharLayer,aText.StyleList());
74 // self->SetPictureFactory(aPictureFactory,aStoreResolver);
80 void CRichTextIndex::RestoreL(const CStreamStore& aStore,TStreamId aId,
81 const CParaFormatLayer* aGlobalParaLayer,
82 const CCharFormatLayer* aGlobalCharLayer,
83 const CStyleList* aStyleList)
86 RStoreReadStream stream;
87 stream.OpenLC(aStore,aId);
89 InternalizeL(stream,aGlobalParaLayer,aGlobalCharLayer,aStyleList);
90 CleanupStack::PopAndDestroy(); // stream
94 TStreamId CRichTextIndex::StoreMarkupL(CStreamStore& aStore,const CStyleList* /*aStyleList*/) const
95 // Store this rich text markup out-of-line along with its components.
98 CStoreMap* map=CStoreMap::NewLC(aStore);
99 StorePicturesL(aStore,*map);
101 RStoreWriteStream stream(*map);
102 TStreamId id=stream.CreateLC(aStore);
103 // CONST_CAST(CRichTextIndex*,this)->iStyleList=CONST_CAST(CStyleList*,aStyleList);
104 ExternalizeL(stream);
108 CleanupStack::PopAndDestroy(2); // map,stream
113 void CRichTextIndex::StorePicturesL(CStreamStore& aStore,CStoreMap& aMap) const
114 // Store any picture components in the specified store.
115 // Due to deferred loading, any pictures that are not in memory at this point are loaded,
121 StorePicturesL(aStore,aMap,0,iText.DocumentLength()+1);
125 void CRichTextIndex::StorePicturesL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength) const
126 // Store all picture components that are contained within the specified range (inclusive), in the store provided.
127 // Prior to storing the picture, its size is written to its picture header which is stored in-line in
129 // Any pictures that are not in memory are now deleted, as a previous call to LoadAllPicturesNowL()
130 // should have been made.
131 // We are guaranteed that all pictures that are in memory are fully restored & detached from their store
136 if (iPictureCount==0)
138 CONST_CAST(CRichTextIndex*,this)->DetachFromStoreL(CPicture::EDetachFull,aPos,aLength); // ensure all pictures have been loaded
139 TInt currentPos=aPos;
140 DocumentChanged(); // reset the cache logical position
141 while (currentPos<aPos+aLength)
142 {// Store next picture & increment pos.
143 CONST_CAST(CRichTextIndex*,this)->ScanToPosition(currentPos,EScanToPositionAbsolute,&MUTABLE_CAST(TLogicalPosition&,iLastUsed));
144 TCurrentIndexRecords current;
145 GetCurrentRecords(current);
146 if (current.iPhrase && current.iPhrase->IsPicturePhrase())
148 RPhraseAttribsEntry& phrase=*current.iPhrase;
149 TPictureHeader* hdr=phrase.PictureHeaderPtr();
152 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_STOREPICTURESL, "ERichTextStorePictureIntegrityError" );
154 __ASSERT_DEBUG(hdr,Panic(ERichTextStorePictureIntegrityError));
155 TBool pictureInMemory=(hdr->iPicture.IsPtr() && hdr->iPicture.AsPtr());
157 {// Store the picture
158 CPicture& picture=*hdr->iPicture;
159 picture.GetSizeInTwips(hdr->iSize); // write picture size to picture header.
160 TStreamId id=picture.StoreL(aStore);
161 aMap.BindL(hdr->iPicture,id);
164 {// Replace the picture phrase with a simple text equivalent.
165 CCharFormatLayer* charLayer=phrase.ReleaseCharFormatLayerOwnership();
166 phrase.Discard(); // destroy the redundant picture phrase
167 RPhraseAttribsEntry newTextPhrase(charLayer,1);
168 (*iPhraseIx)[iPos.iPhraseElement]=newTextPhrase;
169 CONST_CAST(CRichTextIndex*,this)->iPictureCount--; // the picture has now been removed
170 // I do not bother tidying up the phrase index here,
171 // since the benefit does not currently match the cost involved.
176 // increment the current position to the *start* of the next phrase/paragraph
177 TInt offsetIntoUnit=(current.iPhrase)?iPos.iPhraseElementOffset:iPos.iParaElementOffset;
178 currentPos+=CurrentPhraseLength()-offsetIntoUnit;
183 void CRichTextIndex::DetachFromStoreL(CPicture::TDetach aDegree,TInt aPos,TInt aLength)
184 // Attempts to restore all pictures not already present in memory.
185 // A deep (door+document) or shallow (door only) restoration may occur, depending
186 // on the value of detach.
191 if (iPictureCount==0)
193 DocumentChanged(); // reset the cache logical position
194 TInt currentPos=aPos;
195 while (currentPos<aPos+aLength)
197 CONST_CAST(CRichTextIndex*,this)->ScanToPosition(currentPos,EScanToPositionAbsolute,&iLastUsed);
198 TCurrentIndexRecords current;
199 GetCurrentRecords(current);
202 RPhraseAttribsEntry& phrase=*current.iPhrase;
203 if (phrase.IsPicturePhrase())
205 TPictureHeader* hdr=phrase.PictureHeaderPtr();
208 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_DETACHFROMSTOREL, "ERichTextStorePictureIntegrityError" );
210 __ASSERT_DEBUG(hdr,Panic(ERichTextStorePictureIntegrityError));
212 if (hdr->iPicture.IsId())
215 phrase.PictureHandleL(iText.PictureFactory(),iText.StoreResolver(),iPos.iDocPos,MLayDoc::EForceLoadTrue)); // swizzles
216 if (r!=KErrNone && !hdr->iPicture.IsId())
218 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_DETACHFROMSTOREL, "ERichTextStorePictureIntegrityError" );
220 __ASSERT_DEBUG(r==KErrNone || hdr->iPicture.IsId(),Panic(ERichTextStorePictureIntegrityError));
221 if (r==KErrNotSupported) // we don't recognise the picture type
223 TInt offsetIntoUnit=(current.iPhrase)?iPos.iPhraseElementOffset:iPos.iParaElementOffset;
224 currentPos+=CurrentPhraseLength()-offsetIntoUnit;
228 OstTrace1( TRACE_FATAL, DUP3_CRICHTEXTINDEX_DETACHFROMSTOREL, "Leave code=%d", r );
229 User::LeaveIfError(r);
232 // recurse the call to detach the picture from the store
233 if (!hdr->iPicture.IsPtr())
235 OstTrace0( TRACE_DUMP, DUP2_CRICHTEXTINDEX_DETACHFROMSTOREL, "ERichTextStorePictureIntegrityError" );
237 __ASSERT_DEBUG(hdr->iPicture.IsPtr(),Panic(ERichTextStorePictureIntegrityError));
238 hdr->iPicture->DetachFromStoreL(aDegree);
242 // increment the current position to the *start* of the next phrase/paragraph
243 TInt offsetIntoUnit=(current.iPhrase)?iPos.iPhraseElementOffset:iPos.iParaElementOffset;
244 currentPos+=CurrentPhraseLength()-offsetIntoUnit;
249 //////////////////////////////////////////////////////////////////////////////////////
251 void CRichTextIndex::ExternalizeL(RWriteStream& aStream)const
252 // Persist the current state of this object in the specified stream.
253 // Saved records include the terminating paragraph delimiter.
259 CONST_CAST(CRichTextIndex*,this)->CancelInsertCharFormat(); // cancel any insert-pending state.
260 TInt charCount=iText.DocumentLength()+1; // includes the end of document character
261 ((CRichTextIndex*)this)->ScanToPosition(0,EScanToPositionAbsolute);
262 TLogicalPosition startPos=iPos;
263 ((CRichTextIndex*)this)->ScanToPosition(charCount-1,EScanToPositionMatchLeft);
264 TLogicalPosition endPos=iPos;
265 CStyleList* styleList=iText.StyleList();
266 ExternalizeRtiHeaderL(aStream,endPos,styleList);
267 ExternalizeSharedFormatsL(aStream,startPos,endPos,styleList);
268 // DEF126934 begin: memory leak
269 RPhraseAttribsEntry *pForRelease = ExternalizeParaIxL(aStream,startPos,endPos,styleList);
273 ExternalizePhraseIxL(aStream);
277 void CRichTextIndex::InternalizeL(RReadStream& aStream,const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,const CStyleList* aStyleList)
278 // Load a rich text index. Has construct semantics only!
279 // Restores this object from the specified stream. As specific format layers are restored, they are based
280 // on the specified global format layers.
283 if (iParaIx->Count()!=1 || (*iParaIx)[0].iLength!=1)
285 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_INTERNALIZEL, "ERtIndexInternalizeCalledOnNonEmptySource" );
287 __ASSERT_ALWAYS(iParaIx->Count()==1 && (*iParaIx)[0].iLength==1,Panic(ERtIndexInternalizeCalledOnNonEmptySource));
288 TRtPasteContext context(NULL,aGlobalParaLayer,aGlobalCharLayer,aStyleList);
289 InternalizeRtiHeaderL(aStream,context);
290 InternalizeSharedFormatsL(aStream,context);
291 InternalizeParaIxL(aStream,context);
292 InternalizePhraseIxL(aStream,aGlobalCharLayer);
293 if (context.iParagraphStylesFlag)
294 GenerateAllPhraseLinksL();
295 SetSpecificMarkupInternalized(ETrue);
300 //////////////////////////////////////////////////////////////////////////////////////
303 void CRichTextIndex::GenerateAllPhraseLinksL()
304 // Go through the index, and for each specific character format layer (shared or otherwise), regenerate
305 // the appropriate based-on link, (normal or style).
308 TInt count=ParagraphCount();
309 TInt phraseElement=0;
310 for (TInt ii=0; ii<count; ii++)
312 const CParaAttribs& paraAttribs=*(*iParaIx)[ii].iParaAttribs;
313 const CParaFormatLayer* base=STATIC_CAST(const CParaFormatLayer*,paraAttribs.iParaFormat->SenseBase());
314 TInt phraseCount=paraAttribs.PhraseCount();
317 // If the reference count is zero, iCharFormat cannot be valid, so the CParaAttribs object is corrupt.
318 if (paraAttribs.iRefCount == 0)
319 User::Leave(KErrCorrupt);
321 GeneratePhraseLink(paraAttribs.iCharFormat,base); // constant character formatting
327 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_GENERATEALLPHRASELINKSL, "EDebug" );
329 __ASSERT_DEBUG(phraseCount>1,Panic(EDebug));
331 // If the phrase count is too great the CParaAttribs object is corrupt.
332 if (phraseElement + phraseCount > iPhraseIx->Count())
333 User::Leave(KErrCorrupt);
335 for (TInt nn=0; nn<phraseCount; nn++)
336 GeneratePhraseLink((*iPhraseIx)[phraseElement+nn].CharFormat(),base);
337 phraseElement+=phraseCount;
343 void CRichTextIndex::GeneratePhraseLink(CCharFormatLayer* aPhraseCharFormatLayer,const CParaFormatLayer* aBase)
344 // Set the based on link for this character format layer appropriately.
345 // By default, all specific character format layers are based on the global default.
348 TUid type=aBase->Type();
349 if (type!=KNormalParagraphStyleUid)
350 aPhraseCharFormatLayer->SetBase(((CParagraphStyle*)aBase)->CharFormatLayer());
354 //////////////////////////////////////////////////////////////////////////////////////
356 void CRichTextIndex::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength, TBool aCopyStyles)const
357 // Copies the markup corresponding to the specified rich text region to the specified write stream.
358 // aPos/aLength has already been validated in RichText
359 // Saved records can NOT include the terminating paragraph delimiter.
364 ((CRichTextIndex*)this)->ScanToPosition(aPos,EScanToPositionAbsolute);
365 TLogicalPosition startPos=iPos;
366 ((CRichTextIndex*)this)->ScanToPosition((aPos+aLength)-1,EScanToPositionAbsolute);
367 TLogicalPosition endPos=iPos;
369 CStyleList* docStyleList = NULL;
371 docStyleList = (CStyleList*)iText.StyleList();
373 ExternalizeRtiHeaderL(aStream,endPos,docStyleList);
376 ExternalizeReferencedStylesL(aStream,startPos,endPos);
378 ExternalizeSharedFormatsL(aStream,startPos,endPos,docStyleList);
379 RPhraseAttribsEntry* virtualTrailingText=ExternalizeParaIxL(aStream,startPos,endPos,docStyleList);
380 if (virtualTrailingText)
381 CleanupStack::PushL(virtualTrailingText);
382 ExternalizePhraseIxL(aStream,startPos,endPos,virtualTrailingText);
383 if (virtualTrailingText)
384 CleanupStack::PopAndDestroy(virtualTrailingText);
387 //////////////////////////////////////////////////////////////////////////////////////
390 void CRichTextIndex::ExternalizeRtiHeaderL(RWriteStream& aStream,const TLogicalPosition& aEnd,const CStyleList* aStyleList)const
391 // Stores index specific information.
394 TBool incompleteParaFlag=((*iParaIx)[aEnd.iParaElement].iLength>(aEnd.iParaElementOffset+1));
395 aStream.WriteUint8L((TUint8)(aStyleList!=NULL)); // flags the presence of paragraph styles.
396 aStream.WriteUint8L((TUint8)(incompleteParaFlag!=EFalse)); // the last para has no paragraph delimiter
397 if(incompleteParaFlag)
398 {//apply paragraph format to final text fragment if logical end is at document end
399 aStream.WriteUint8L((TUint8)(aEnd.iDocPos+1 == iText.DocumentLength()));
404 void CRichTextIndex::InternalizeRtiHeaderL(RReadStream& aStream,TRtPasteContext& aContext)
405 // Load index specific information.
410 aContext.iParagraphStylesFlag=(TBool)flag;
412 aContext.iIncompleteParaFlag=(TBool)flag;
413 aContext.iApplyFormatToLastFlag=EFalse;
414 if(aContext.iIncompleteParaFlag)
417 aContext.iApplyFormatToLastFlag=(TBool)flag;
421 //////////////////////////////////////////////////////////////////////////////////////
423 void CRichTextIndex::ExternalizeReferencedStylesL(RWriteStream& aStream,const TLogicalPosition& aStart,
424 const TLogicalPosition& aEnd) const
425 // Write those styles that are referenced by the paragraphs in the range aStart to aEnd.
428 CStyleList* list = iText.StyleList();
431 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEREFERENCEDSTYLESL, "EStyleClipboardIntegrityError" );
433 __ASSERT_DEBUG(list,Panic(EStyleClipboardIntegrityError));
434 if (aStart.iParaElement > aEnd.iParaElement)
436 OstTrace0( TRACE_DUMP, DUP1_CRICHTEXTINDEX_EXTERNALIZEREFERENCEDSTYLESL, "EStyleClipboardIntegrityError" );
438 __ASSERT_DEBUG(aStart.iParaElement <= aEnd.iParaElement,Panic(EStyleClipboardIntegrityError));
440 TUint8 numStyles = 0;
442 numStyles = (TUint8)list->Count();
444 HBufC8* buf=HBufC8::NewL(numStyles);
445 TPtr8 paraRegister=buf->Des();
446 paraRegister.FillZ(numStyles);
447 TInt refCount=MarkStyleRegister(paraRegister,aStart.iParaElement,aEnd.iParaElement);
448 CleanupStack::PushL(buf);
449 ExternalizeItemsPresentInStyleRegisterL(aStream,refCount,paraRegister);
450 CleanupStack::PopAndDestroy();
454 void CRichTextIndex::InternalizeSharedFormatsL(RReadStream& aStream, const TRtPasteContext& aContext)
455 // Restores the list of shared para attribs.
456 // All existing CParaAttribs on the shared list are removed.
457 // The reference counts are not restored, but modified dynamically as the para index is loaded.
460 iParaIx->Reset(); // stops paraIx referencing shared paraAttribs that are about to be released.
461 CParaAttribs* currentSharedPara;
462 TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
463 while ( (currentSharedPara = iterator++) != NULL )
464 currentSharedPara->Release(currentSharedPara->iRefCount);
465 TUint8 sharedParaCount = ReadTUint8CountL(aStream);
466 for (TUint8 paraItem = 0; paraItem < sharedParaCount; paraItem++)
468 (void)aStream.ReadInt32L(); // consume the data.
469 CParaFormatLayer* paraLayer = InternalizeParagraphFormatL(aStream, aContext);
470 CleanupStack::PushL(paraLayer);
472 CCharFormatLayer* charLayer = CCharFormatLayer::NewL(aStream); // Does not restore based on link.
473 charLayer->SetBase(aContext.iGlobalCharLayer); // base on 'Normal' global char format
474 CleanupStack::PushL(charLayer);
475 CParaAttribs* paraAttribs = CParaAttribs::NewL(paraLayer, charLayer);
476 CleanupStack::PopAndDestroy(2);
477 paraAttribs->iRefCount = 0; // Otherwise recalculating reference counts on internalization of paraIx will result in 1 too many.
478 iSharedParaQueHead.AddLast(*paraAttribs);
483 void CRichTextIndex::ExternalizeSharedFormatsL(RWriteStream& aStream,const TLogicalPosition& aStart,const TLogicalPosition& aEnd,const CStyleList* aStyleList)const
484 // Persist those items in the shared list that are referenced by the paragraphs in the
485 // range aStart para to aEnd para.
488 TUint8 sharedParaCount=SharedParaAttribsEntryCountL();
489 // Some paras coverd, so take register
490 HBufC8* buf=HBufC8::NewL(sharedParaCount);
491 TPtr8 sharedAttribsRegister=buf->Des();
492 sharedAttribsRegister.FillZ(sharedParaCount);
493 TInt sharedCount=MarkRegister(sharedAttribsRegister,aStart.iParaElement,aEnd.iParaElement);
494 CleanupStack::PushL(buf);
495 ExternalizeItemsPresentInRegisterL(aStream,sharedCount,sharedAttribsRegister,aStyleList);
496 CleanupStack::PopAndDestroy();
500 TUint8 CRichTextIndex::SharedParaAttribsEntryCountL()const
501 // Return a count of the number of items currently
502 // in the sharedParaFormats list.
504 // This will leave if more than 255 found, as this is not supported
506 TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
507 TUint16 sharedParaCount = 0;
508 while (iterator++ != NULL)
511 // Currently more than 255 paragraph formats are not supported
512 // If the current document has more, leave with KErrNotSupported
513 if (sharedParaCount > 255)
514 User::Leave(KErrNotSupported);
516 return (TUint8)sharedParaCount;
520 TInt CRichTextIndex::MarkRegister(TDes8& aBuf,TInt aStartPara,TInt aEndPara)const
521 // Indicate in the register, aBuf, once and once only, each paraAttribs that is
522 // referenced by each of the paragraphs in the specified range.
523 // Return the number of shared para formats referenced.
526 TInt parasReferenced=0;
527 for (TInt item=aStartPara;item<=aEndPara;item++)
529 CParaAttribs* paraAttribs=(*iParaIx)[item].iParaAttribs;
530 TInt offset=RefNum(paraAttribs); // 0 return shows para attribs not in shared list
532 {// para entry references a shared para attribs
533 if (aBuf[offset-1]==0)
535 aBuf[offset-1]=KRegisterItemPresent; // mark item as needing to be stored
540 return parasReferenced;
544 TInt CRichTextIndex::MarkStyleRegister(TDes8& aBuf,TInt aStartPara,TInt aEndPara)const
545 // Indicate in the register, aBuf, once and once only, each paragraph style thar is
546 // referenced by each of the paragraphs in the speciifed range.
547 // Return the umber of paragraph styles referenced.
550 TInt stylesReferenced=0;
551 const CParaAttribs* para=NULL;
552 for (TInt item=aStartPara;item<=aEndPara;item++)
554 para=(*iParaIx)[item].iParaAttribs;
555 TInt index=iText.StyleList()->IndexByPtr(STATIC_CAST(const CParaFormatLayer*,para->iParaFormat->SenseBase()));
556 if (index!=KErrNotFound)
558 if (index>=aBuf.Length())
560 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_MARKSTYLEREGISTER, "EStyleClipboardIntegrityError" );
562 __ASSERT_DEBUG(index<aBuf.Length(),Panic(EStyleClipboardIntegrityError));
563 if (aBuf[index]!=KRegisterItemPresent)
565 aBuf[index]=KRegisterItemPresent; // mark item as needing to be stored
571 return stylesReferenced;
575 void CRichTextIndex::ExternalizeItemsPresentInRegisterL(RWriteStream& aStream,TInt aSharedCount,const TDes8& aBuf,const CStyleList* aStyleList)const
576 // Externalize each object from the shared list that has a corresponding mark in the
580 if (aSharedCount>(TInt)KMaxTUint8)
582 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEITEMSPRESENTINREGISTERL, "ESharedParaCountStreamOverflow" );
584 __ASSERT_DEBUG(aSharedCount<=(TInt)KMaxTUint8,Panic(ESharedParaCountStreamOverflow));
585 aStream.WriteUint8L(aSharedCount);
586 CParaAttribs* currentSharedPara=NULL;
587 TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
589 while ((currentSharedPara=iterator++)!=NULL)
591 if (aBuf[offset]==KRegisterItemPresent)
593 aStream.WriteInt32L(offset+1); // the ref-no of the shared para - so it can be spotted in the paste.
594 ExternalizeParagraphFormatL(aStream,*currentSharedPara->iParaFormat,aStyleList);
595 aStream<< *currentSharedPara->iCharFormat;
602 void CRichTextIndex::ExternalizeItemsPresentInStyleRegisterL(RWriteStream& aStream,TInt aRefStyleCount,
603 const TDes8& aBuf) const
604 // Externalize each object from the paragraph style list that has a corresponding mark in the register aBuf.
607 if (aRefStyleCount > (TInt)KMaxTUint8)
609 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEITEMSPRESENTINSTYLEREGISTERL, "EStyleClipboardIntegrityError" );
611 __ASSERT_DEBUG(aRefStyleCount <= (TInt)KMaxTUint8,Panic(EStyleClipboardIntegrityError));
612 aStream.WriteUint8L(aRefStyleCount);
613 TInt count=aBuf.Length();
615 for (TInt ii=0;ii<count;ii++)
617 if (aBuf[ii]==KRegisterItemPresent)
619 aStream.WriteInt8L(ii+1); // cannot use 0 index as this represents an absence.
620 iText.StyleList()->At(ii).iStyle->ExternalizeL(aStream);
626 RPhraseAttribsEntry* CRichTextIndex::ExternalizeParaIxL(RWriteStream& aStream,
627 const TLogicalPosition& aStart,
628 const TLogicalPosition& aEnd,
629 const CStyleList* aStyleList)const
630 // Externalize each para entry from the para index that falls between
631 // aStart paragraph and aEnd paragraph inclusive.
632 // Any specific paraAttribs (ie not on the shared list) are also externalized.
634 // * paragraph length
635 // * ref no. (ref no. 0 means specific/non-shared one)
636 // * [specific paragraph format layer - including based-on link]
637 // * [specific phrase count]
640 TParaAttribsEntry lastParaEntry=(*iParaIx)[aEnd.iParaElement];
641 TBool incompleteParaFlag=(lastParaEntry.iLength>(aEnd.iParaElementOffset+1));
642 aStream.WriteInt32L(1+aEnd.iParaElement-aStart.iParaElement); // paragraph count
643 for (TInt paraItem=aStart.iParaElement;paraItem<aEnd.iParaElement;paraItem++)
644 {// Externalize all but last TParaAttribsEntrys from the para index
645 TParaAttribsEntry paraEntry=(*iParaIx)[paraItem];
646 TInt phraseCount=(paraEntry.iParaAttribs->IsShared())
648 : paraEntry.iParaAttribs->iPhraseCount;
649 if (paraItem==aStart.iParaElement)
650 {// Fix & write length & phrase count of the first paragraph.
651 paraEntry.iLength=(paraEntry.iLength-aStart.iParaElementOffset); // fix 1st para length
653 phraseCount=phraseCount-(aStart.iPhraseElement-aStart.iParaBasePhraseElement);
655 aStream.WriteInt32L(paraEntry.iLength);
656 CParaAttribs* paraAttribs=paraEntry.iParaAttribs;
657 TUint8 refNo=RefNum(paraAttribs);
658 aStream.WriteUint8L(refNo);
660 {// Write specific para layer & phrase count
661 ExternalizeParagraphFormatL(aStream,*paraAttribs->iParaFormat,aStyleList);
662 aStream.WriteInt32L(phraseCount);
665 // Fix & write the length of the last paragraph.
666 if (aStart.iParaElement==aEnd.iParaElement)
667 lastParaEntry.iLength=(aEnd.iParaElementOffset-aStart.iParaElementOffset)+1; // copied text contained within 1 para
669 lastParaEntry.iLength-=(lastParaEntry.iLength-aEnd.iParaElementOffset)-1;
670 aStream.WriteInt32L(lastParaEntry.iLength);
671 // Fix & write the phrase count of the last paragraph if it has specific char format
672 CParaAttribs* lastParaAttribs=lastParaEntry.iParaAttribs;
673 TUint8 refNo=RefNum(lastParaAttribs);
674 RPhraseAttribsEntry* virtualTextPhrase=NULL;
675 if (incompleteParaFlag)
676 {// Adjust the phrase count for this paragraph that is not complete
678 if (aStart.iParaElement==aEnd.iParaElement)
679 phraseCount=(aEnd.iPhraseElement-aStart.iPhraseElement)+1; // copied text contained within 1 para
681 phraseCount=(aEnd.iPhraseElement-aEnd.iParaBasePhraseElement)+1;
682 if (phraseCount>lastParaAttribs->iPhraseCount)
684 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_EXTERNALIZEPARAIXL, "ERtExternalizeParaIx" );
686 __ASSERT_DEBUG(phraseCount<=lastParaAttribs->iPhraseCount,Panic(ERtExternalizeParaIx));
687 aStream.WriteInt32L(phraseCount);
689 {// Set the virtual phrase representing the trailing text from the shared paragraph.
690 virtualTextPhrase=new(ELeave) RPhraseAttribsEntry(lastParaAttribs->iCharFormat,lastParaEntry.iLength);
691 CleanupStack::PushL(virtualTextPhrase);
694 // Write out the paragraph format for the paragraph that is not complete
695 // - necessary since we need to patch up the style character formatting
696 if (aStyleList || aEnd.iDocPos+1 == iText.DocumentLength())
697 ExternalizeParagraphFormatL(aStream, *lastParaAttribs->iParaFormat, aStyleList);
700 {// This is a complete para - copy region ends on a paragraph delimiter.
701 aStream.WriteUint8L(refNo);
703 {// Write specific para layer & phrase count
704 ExternalizeParagraphFormatL(aStream,*lastParaAttribs->iParaFormat,aStyleList);
705 TInt phraseCount=lastParaAttribs->iPhraseCount;
706 if (aStart.iParaElement==aEnd.iParaElement && aStart.iParaElementOffset && aStart.iParaElementOffset>0)
707 phraseCount=(aEnd.iPhraseElement-aStart.iPhraseElement)+1; // copied text contained within 1 para
708 aStream.WriteInt32L(phraseCount);
711 if (virtualTextPhrase)
712 CleanupStack::Pop(virtualTextPhrase);
713 return virtualTextPhrase;
717 void CRichTextIndex::ExternalizeParagraphFormatL(RWriteStream& aStream,const CParaFormatLayer& aLayer,const CStyleList* aStyleList)const
718 // If a style list is present, write out the based-on links for every paragraph format layer,
719 // in order to reconnect each paragraph to the right style on loading.
726 refNo=aStyleList->IndexByPtr(((CParaFormatLayer*)aLayer.SenseBase()));
727 // returns the offset in the style list, or a negative error.
729 refNo=(refNo!=KErrNotFound)
730 ? refNo+1 // the nth style where 1st item is offset 1.
731 : 0; // An error value was returned so this thing has no paragraph style as its base
732 aStream.WriteInt8L((TInt8)-refNo);
737 CParaFormatLayer* CRichTextIndex::InternalizeParagraphFormatL(RReadStream& aStream,const TRtPasteContext& aContext)
740 CParaFormatLayer* paraLayer=CParaFormatLayer::NewL(aStream); // specific paragraph format layer
742 if (aContext.iParagraphStylesFlag)
744 CleanupStack::PushL(paraLayer);
745 TInt refNo=aStream.ReadInt8L(); // Read in the based on link to fix up to the style table
749 if (index>0 && aContext.iStyleList)
750 paraLayer->SetBase(aContext.iStyleList->At(index-1).iStyle);
752 paraLayer->SetBase(aContext.iGlobalParaLayer);
757 CParaFormatLayer* CRichTextIndex::PasteParagraphFormatL(RReadStream& aStream,const TRtPasteContext& aContext,CStyleMap* aStyleMap)
760 CParaFormatLayer* paraLayer=CParaFormatLayer::NewL(aStream); // specific paragraph format layer
763 if (aContext.iParagraphStylesFlag)
765 CleanupStack::PushL(paraLayer);
766 TInt refNo=aStream.ReadInt8L(); // Read in the based on link to fix up to the style table
767 CleanupStack::Pop(paraLayer); // paraLayer
771 // If a style exists for this paragraph, set it as the base.
772 // If a style list doesn't exist, the base paragraph style will be set instead.
774 if (index>0 && aContext.iStylePasteMode != CParagraphStyle::EIgnoreNewStyles)
776 if (!aContext.iStyleList || (aContext.iStyleList && aContext.iStylePasteMode == CParagraphStyle::EConvertNewStyles))
778 CleanupStack::PushL(paraLayer);
779 iLastCharacterStyle = (aStyleMap->Item(index))->iCharFormatLayer;
781 // If the application doesn't have a style list,
782 // the new paragraph style has to be enforced by merging the old
783 // paralayer with the style paralayer
784 CParaFormat* MergedParaFormat = CParaFormat::NewLC();
785 TParaFormatMask MergedParaFormatMask;
787 CParaFormatLayer* newParaLayer;
789 // Extract the masks from both layers
790 paraLayer->SenseL(MergedParaFormat, MergedParaFormatMask);
791 (aStyleMap->Item(index))->SenseL(MergedParaFormat, MergedParaFormatMask);
793 newParaLayer = CParaFormatLayer::NewL(MergedParaFormat, MergedParaFormatMask);
795 // Now on stack: MergedParaFormat, paraLayer, ...
796 CleanupStack::PopAndDestroy(); // delete MergedParaFormat
798 // Now on stack: paraLayer, ...
799 CleanupStack::PopAndDestroy(); // paraLayer
800 paraLayer = newParaLayer;
802 paraLayer->SetBase(aContext.iGlobalParaLayer);
807 paraLayer->SetBase(aStyleMap->Item(index));
808 iLastCharacterStyle = NULL;
812 // Otherwise set the character style to NULL, which will set the global
813 // character style (see line 983 approx)
815 iLastCharacterStyle = NULL;
816 paraLayer->SetBase(aContext.iGlobalParaLayer);
823 void CRichTextIndex::InternalizeParaIxL(RReadStream& aStream,const TRtPasteContext& aContext)
824 // Restore the paragraph index & all associated specific para formats,
825 // All existing index data is lost.
828 // The para index has been reset at the start of internalize shared formats.
829 TInt paraIxCount=aStream.ReadInt32L(); // paragraph count
830 for (TInt currentPara=0;currentPara<paraIxCount;currentPara++)
831 {// Restore each paragraph
832 TParaAttribsEntry para;
833 para.iLength=aStream.ReadInt32L(); // paragraph length
834 TUint8 refNo=aStream.ReadUint8L(); // ref no
836 {// Link to para attribs in shared list & up its reference count
837 para.iParaAttribs=SharedParaAttribs(refNo);
838 para.iParaAttribs->iRefCount++;
839 iParaIx->AppendL(para);
842 {// Have to build up the specific para attribs
843 CParaFormatLayer* paraLayer=InternalizeParagraphFormatL(aStream,aContext);
844 CleanupStack::PushL(paraLayer);
845 CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraLayer);
846 CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,specificParaAttribs));
847 specificParaAttribs->iPhraseCount=aStream.ReadInt32L(); // specific phrase count
848 CleanupStack::Pop(); // specificParaAttribs
849 CleanupStack::PopAndDestroy(); // paraLayer
850 para.iParaAttribs=specificParaAttribs;
851 CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,specificParaAttribs));
852 iParaIx->AppendL(para);
853 CleanupStack::Pop(); // specificParaAttribs
858 void CRichTextIndex::ExternalizePhraseIxL(RWriteStream& aStream)const
859 // Called from an Externalize.
860 // All phrases must be saved.
862 {ExternalizePhrasesL(aStream,0,PhraseCount());}
865 void CRichTextIndex::ExternalizePhraseIxL(RWriteStream& aStream,
866 const TLogicalPosition& aStart,
867 TLogicalPosition& aEnd,
868 const RPhraseAttribsEntry* aVirtualTrailingText)const
869 // Called from CopyToStream.
870 // Here, a text fragment only, is to be saved.
871 // The last argument may be NULL. If it is other than NULL, it represents trailing text
872 // that has no paragraph delimiter, which was part of a shared paragraph. This must be presented as
873 // a distinct phrase, since this is the context in which it is pasted back into a document.
876 TInt phraseCount=PhraseCount();
877 phraseCount+=(aVirtualTrailingText) ? 1 : 0;
878 TBool noRealPhrasesInRegion=ETrue;
879 TInt nn=aStart.iParaElement;
880 while (nn<=aEnd.iParaElement && noRealPhrasesInRegion)
882 if (!(*iParaIx)[nn].iParaAttribs->IsShared())
883 noRealPhrasesInRegion=EFalse;
887 if (phraseCount==0 || (phraseCount==1 && aVirtualTrailingText || noRealPhrasesInRegion)) // save this phrase count
888 ExternalizePhrasesL(aStream,aStart.iPhraseElement,0,aVirtualTrailingText);
891 TBool endParaShared=(*iParaIx)[aEnd.iParaElement].iParaAttribs->IsShared();
892 if (endParaShared && aEnd.iPhraseElement>0)
893 aEnd.iPhraseElement--; // Due to action of ScanToPosition
895 RPhraseAttribsEntry* backPhrase=&(*iPhraseIx)[aEnd.iPhraseElement];
896 TInt backLength=backPhrase->Length();
897 if (!(endParaShared || backPhrase->IsPicturePhrase()))
898 backPhrase->SetLength(aEnd.iPhraseElementOffset+1);
900 RPhraseAttribsEntry* frontPhrase=&(*iPhraseIx)[aStart.iPhraseElement];
901 TInt frontLength=frontPhrase->Length();
902 if (!frontPhrase->IsPicturePhrase()) // Fix length of first phrase.
903 frontPhrase->SetLength(frontLength-aStart.iPhraseElementOffset);
905 ExternalizePhrasesL(aStream,aStart.iPhraseElement,aEnd.iPhraseElement-aStart.iPhraseElement+1,aVirtualTrailingText));
907 // Now fix the altered phrase lengths.
908 if (!frontPhrase->IsPicturePhrase())
909 frontPhrase->SetLength(frontLength);
910 if (!(endParaShared || backPhrase->IsPicturePhrase()))
911 backPhrase->SetLength(backLength);
913 __TEST_INVARIANT; // we lied about being const
915 User::LeaveIfError(ret);
920 void CRichTextIndex::ExternalizePhrasesL(RWriteStream& aStream,TInt aStart,TInt aPhraseCount,
921 const RPhraseAttribsEntry* aVirtualPhrase)const
922 // Save the specified number of phrases present in the phrase index,
923 // starting from phrase offset aStart.
926 ExternalizePhraseCountL(aStream,aPhraseCount+ ((aVirtualPhrase)?1:0) );
927 for (TInt phraseItem=aStart;phraseItem<aStart+aPhraseCount;phraseItem++)
929 RPhraseAttribsEntry& phraseEntry=(*iPhraseIx)[phraseItem];
930 aStream<< phraseEntry;
933 aStream<< *aVirtualPhrase;
937 void CRichTextIndex::InternalizePhraseIxL(RReadStream& aStream,const CCharFormatLayer* aGlobalCharFormat)
938 // Load all the phrases from the stream, and insert them into the phrase index
942 TInt phraseCount=aStream.ReadInt32L();
944 // Extend phrase index by required amount
945 iPhraseIx->AppendL(RPhraseAttribsEntry(),phraseCount);
947 for (TInt phraseItem=0;phraseItem<phraseCount;phraseItem++)
948 {// Restore each phrase & insert into the phrase index.
949 TBool isPicture=(TBool)aStream.ReadUint8L();
950 TInt phraseLength=aStream.ReadInt32L();
951 CCharFormatLayer* charLayer=CCharFormatLayer::NewL(aStream);
952 charLayer->SetBase(aGlobalCharFormat);
953 RPhraseAttribsEntry& tPhrase=iPhraseIx->At(phraseItem);
955 new(&tPhrase) RPhraseAttribsEntry(charLayer,phraseLength);
957 {// Manufacture new picture header & set its picture store
958 CleanupStack::PushL(charLayer);
961 TBool ownershipTaken(EFalse);
962 CPicturePhrase* picPhrase=CPicturePhrase::NewL(hdr,charLayer,ownershipTaken);
963 CleanupStack::Pop(); // char layer
964 new(&tPhrase) RPhraseAttribsEntry(picPhrase);
971 CParaAttribs* CRichTextIndex::SharedParaAttribs(TUint8 aOrdinal)
972 // Return the handle of the para attribs in the shared para attribs list,
973 // whose position in the list is specified by the argument aOrdinal.
976 CParaAttribs* currentSharedPara;
977 TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
979 while ((currentSharedPara=iterator++)!=NULL && aOrdinal!=match)
981 if (currentSharedPara==NULL)
983 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_SHAREDPARAATTRIBS, "EEndOfSharedParaListEncountered" );
985 __ASSERT_ALWAYS(currentSharedPara!=NULL,Panic(EEndOfSharedParaListEncountered));
986 return currentSharedPara;
990 TUint8 CRichTextIndex::RefNum(const CParaAttribs* aParaAttribs)const
991 // If the para attribs argument is present in the shared list, return a
992 // reference to it; otherwise return zero as the reference.
995 CParaAttribs* currentSharedPara;
996 TDblQueIter<CParaAttribs> iterator(((CRichTextIndex*)this)->iSharedParaQueHead);
998 while ((currentSharedPara=iterator++)!=NULL)
1000 if (currentSharedPara==aParaAttribs)
1007 //////////////////////////////////////////////////
1008 //////////////////////////////////////////////////
1009 //////////////////////////////////////////////////
1011 void CRichTextIndex::PasteFromStreamL(const CStreamStore& aStore,RReadStream& aStream,TInt aPos,CParagraphStyle::TStylePasteMode aStylePasteMode,const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer)
1016 CancelInsertCharFormat();
1017 ScanToPosition(aPos,EScanToPositionAbsolute);
1021 TRtPasteContext context(&aStore,aGlobalParaLayer,aGlobalCharLayer,iText.StyleList());
1022 context.iPastePos=iPos;
1023 context.iStylePasteMode = aStylePasteMode;
1025 InternalizeRtiHeaderL(aStream,context); // no rollback required on the header. Does not alter iPos.
1027 // Get the pargraph styles
1029 CStyleMap* styleMap = NULL;
1030 if (context.iParagraphStylesFlag)
1032 TUint8 styleCount=ReadTUint8CountL(aStream);
1033 styleMap=CStyleMap::NewLC(styleCount);
1035 PasteStylesL(aStream,*styleMap,context)); // Does not alter iPos.
1036 if (ret!=KErrNone) {RbPasteSharedFormatsL(ret);}
1039 // Get the shared formats
1040 TUint8 sharedParaCount=ReadTUint8CountL(aStream);
1041 CParaAttribsMap* paraMap=CParaAttribsMap::NewLC(sharedParaCount);
1043 PasteSharedFormatsL(aStream,*paraMap,context,styleMap)); // Does not alter iPos.
1046 // beginning of fixing DEF 126651 (1/2)
1047 // free orphan object before leave.
1048 if (context.iParagraphStylesFlag)
1050 if (!context.iStyleList || context.iStylePasteMode == CParagraphStyle::EConvertNewStyles)
1052 TInt maxSize = styleMap->Count();
1053 for (TInt count=0; count<maxSize; count++)
1055 delete (CParagraphStyle*)styleMap->At(count).iT;
1058 CleanupStack::PopAndDestroy(); // cleanup styleMap
1060 CleanupStack::PopAndDestroy(); // cleanup paraMap
1061 // end of fixing DEF 126651 (1/2)
1062 RbPasteSharedFormatsL(ret);
1067 PasteIxL(aStream,context,*paraMap, styleMap)); // context now has both global layers & pastePos
1070 // beginning of fixing DEF 126651 (2/2)
1071 // free orphan object before leave.
1072 if (context.iParagraphStylesFlag)
1074 if (!context.iStyleList || context.iStylePasteMode == CParagraphStyle::EConvertNewStyles)
1076 TInt maxSize = styleMap->Count();
1077 for (TInt count=0; count<maxSize; count++)
1079 delete (CParagraphStyle*)styleMap->At(count).iT;
1082 CleanupStack::PopAndDestroy(); // cleanup styleMap
1084 CleanupStack::PopAndDestroy(); // cleanup paraMap
1085 // end of fixing DEF 126651 (2/2)
1086 RbPasteSharedFormatsL(ret);
1090 if (context.iParagraphStylesFlag)
1092 TRAP(ret,GenerateAllPhraseLinksL());
1093 if (ret != KErrNone)
1094 RbPasteSharedFormatsL(ret);
1096 if (!context.iStyleList || context.iStylePasteMode == CParagraphStyle::EConvertNewStyles)
1098 TInt maxSize = styleMap->Count();
1099 for (TInt count=0; count<maxSize; count++)
1101 delete (CParagraphStyle*)styleMap->At(count).iT;
1104 CleanupStack::PopAndDestroy(); // cleanup styleMap
1107 CleanupStack::PopAndDestroy(); // cleanup paraMap
1113 void CRichTextIndex::PasteStylesL(RReadStream& aStream,CStyleMap& aMap,const TRtPasteContext& aContext)
1114 // Restore the paragraph styles from the specified stream.
1115 // Restoration is controlled by the flag TStylePasteMode.
1119 TInt styleCount=aMap.iCapacity;
1120 // TBool docSupportsStyles = iText.StyleListPresent(); // detect if document has style list
1121 CParagraphStyle* style=NULL;
1122 for(TUint8 ii=0; ii<styleCount; ii++)
1124 TInt refNo=aStream.ReadInt8L();
1125 style=CParagraphStyle::NewL(aStream,*aContext.iGlobalParaLayer,*aContext.iGlobalCharLayer);
1127 // Now attempt to map this style to one in the document we are adding it to
1128 // refNo contains the style number when the selection was cut, which
1129 // is needed to map to the reference no's coming in with the paragraph
1132 if (aContext.iStylePasteMode == CParagraphStyle::EIgnoreNewStyles)
1133 // We're ignoring all style info, so delete it now
1138 if (aContext.iStyleList && aContext.iStylePasteMode == CParagraphStyle::EAddNewStyles)
1140 TInt docStyle = iText.StyleList()->IndexByName(style->iName);
1141 if (docStyle!=KErrNotFound)
1143 /* Add the existing style into the stylemap */
1144 aMap.Bind(refNo,iText.StyleList()->PtrByName(style->iName)->iStyle);
1146 /* Since this style exists, we can safely delete this copy */
1152 // Ok, the document doesn't have this style in it.
1153 // So let's add it in, and put a reference in the stylemap
1155 RParagraphStyleInfo newStyle(style);
1157 iText.StyleList()->AppendL(&newStyle);
1159 aMap.Bind(refNo,style);
1161 // the StyeList takes care of style destruction, so no deletion
1162 // is necessary here
1167 // Document doesn't support styles, so save them so they can
1168 // be converted to character specific formatting later on
1169 aMap.Bind(refNo, style);
1177 void CRichTextIndex::ImposeCharacterStyleL(CCharFormatLayer** aCharLayer)
1179 // This function is used to impose the current character style onto
1180 // the character layer for this paragraph. It's used when translating
1181 // style information into specific formatting when pasting text containing
1182 // styles into text without styles.
1184 TCharFormatX MergedCharFormat;
1185 TCharFormatXMask MergedCharFormatMask;
1187 CCharFormatLayer* newCharLayer;
1189 // Extract the masks from both layers.
1190 // use MergedCharFormat to hold the char info which we don't want yet
1192 (*aCharLayer)->Sense(MergedCharFormat, MergedCharFormatMask);
1193 iLastCharacterStyle->Sense(MergedCharFormat, MergedCharFormatMask);
1195 // Re-build charLayer;
1196 CleanupStack::PushL(*aCharLayer);
1197 newCharLayer = CCharFormatLayer::NewL(MergedCharFormat,MergedCharFormatMask);
1198 CleanupStack::Pop(*aCharLayer);
1200 delete (*aCharLayer);
1201 (*aCharLayer) = newCharLayer;
1205 void CRichTextIndex::PasteSharedFormatsL(RReadStream& aStream,CParaAttribsMap& aMap,const TRtPasteContext& aContext,CStyleMap* aStyleMap)
1206 // Load shared formats from the specified stream.
1207 // Only adds the loaded shared format to the list of shared formats
1208 // if it does not already exist in that list.
1209 // A map is kept, that for each loaded format specifies its offset within
1214 TInt mapCount=aMap.iCapacity;
1215 for (TUint8 paraItem=0;paraItem<mapCount;paraItem++)
1217 TInt refNo=aStream.ReadInt32L();
1218 CParaFormatLayer* paraLayer=PasteParagraphFormatL(aStream,aContext,aStyleMap);
1219 CleanupStack::PushL(paraLayer);
1221 CCharFormatLayer* charLayer=CCharFormatLayer::NewL(aStream); // Does not restore based on link.
1222 //// Change specific formatting for this paragraph
1224 if (iLastCharacterStyle != NULL)
1225 ImposeCharacterStyleL(&charLayer);
1227 charLayer->SetBase(aContext.iGlobalCharLayer);
1229 CleanupStack::PushL(charLayer);
1230 CParaAttribs* paraAttribs=GetParaAttribsL(paraLayer,charLayer);
1231 CleanupStack::PopAndDestroy(2); // charLayer & paraLayer
1232 aMap.Bind(refNo,paraAttribs);
1233 paraAttribs->iRefCount--;
1238 void CRichTextIndex::PasteIxL(RReadStream& aStream,TRtPasteContext& aContext,const CParaAttribsMap& aMap,CStyleMap* aStyleMap/*,CParaAttribs* aSecondReserved*/)
1241 TInt completeParaCount=aStream.ReadInt32L();
1242 completeParaCount-=(aContext.iIncompleteParaFlag)? 1 : 0;
1243 // Create rollback states
1244 TParaAttribsEntry para=(*iParaIx)[aContext.iPastePos.iParaElement];
1245 CParaAttribs* reclaimed=RequestReclaimShareL(para.iParaAttribs,¶); // does not release share
1246 iRollbackParaAttribsHandle=NULL;
1249 iRollbackParaAttribsHandle=para.iParaAttribs;
1250 CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,reclaimed));
1251 (*iParaIx)[aContext.iPastePos.iParaElement].iParaAttribs=reclaimed; // Use this reclaimed para attribs
1253 // Split the phrase at the paste position, ready to insert clipboard content
1255 SplitPhraseL(aContext.iPastePos.iDocPos));
1256 if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
1257 ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute); // pick up the changes
1258 aContext.iPastePos=iPos;
1259 // Paste all complete paragraphs from the clipboard
1260 TInt characterCount=0;
1261 RPhraseAttribsEntry* firstParaVirtualPhrase=NULL;
1263 characterCount=PasteParaIxL(aStream,aContext,completeParaCount,aMap,firstParaVirtualPhrase,aStyleMap));
1264 if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
1265 CleanupStack::PushL(firstParaVirtualPhrase);
1266 // Paste any remaining text fragment.
1267 TTextFragment textFragment;
1268 if (aContext.iIncompleteParaFlag)
1271 textFragment=GetTextFragmentL(aStream));
1272 if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
1273 characterCount+=textFragment.iLength;
1275 // Restore the character stle info for the final text fragment
1276 if (aContext.iParagraphStylesFlag || aContext.iApplyFormatToLastFlag)
1278 CParaFormatLayer* paraLayer = PasteParagraphFormatL(aStream, aContext, aStyleMap);
1279 CleanupStack::PushL(paraLayer);
1280 //Restore the paragraph formatting for the final text fragment
1281 if (aContext.iApplyFormatToLastFlag)
1283 CParaFormat* paraFormat = CParaFormat::NewLC();
1284 TParaFormatMask mask;
1285 paraLayer->SenseL(paraFormat,mask);
1286 (*iParaIx)[completeParaCount].iParaAttribs->iParaFormat->SetL(paraFormat,mask);
1287 CleanupStack::PopAndDestroy(paraFormat);
1289 // We can now discard the format layer, but the character format has been safely
1290 // patched to the style character format
1291 CleanupStack::PopAndDestroy(paraLayer);
1296 PastePhraseIxL(aStream,aContext,firstParaVirtualPhrase));
1297 if (ret!=KErrNone) {RbPasteParaIxL(aContext.iPastePos,aContext.iParasPasted,ret);}
1298 CleanupStack::PopAndDestroy(); // firstParaVirtualPhrase / shallow destroy only - deep copy in phraseIx
1300 // ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute); // phraseIx not pasted at this point
1301 // TLogicalPosition headParaNormalizePos=iPos;
1304 if (completeParaCount==0)
1305 {// Adjust paragraph length & phrase count
1306 TParaAttribsEntry* para=&(*iParaIx)[aContext.iPastePos.iParaElement];
1307 para->iLength+=textFragment.iLength;
1308 para->iParaAttribs->iPhraseCount+=textFragment.iPhraseCount;
1309 ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute);
1310 TLogicalPosition headParaNormalizePos=iPos;
1311 MergePhrases(headParaNormalizePos);
1312 MergePhrases(aContext.iPastePos.iDocPos+characterCount);
1313 DoPasteCleanup(headParaNormalizePos,reclaimed);
1316 {// Adjust paragraph lengths & phrase counts
1317 TParaAttribsEntry* firstPara=&(*iParaIx)[aContext.iPastePos.iParaElement];
1318 firstPara->iLength+=aContext.iPastePos.iParaElementOffset; // Update length of the first para
1319 firstPara->iParaAttribs->iPhraseCount+=(aContext.iPastePos.iPhraseElement-aContext.iPastePos.iParaBasePhraseElement); // Update the phrase count of the first para attribs
1320 ScanToPosition(aContext.iPastePos.iDocPos,EScanToPositionAbsolute);
1321 TLogicalPosition headParaNormalizePos=iPos;
1322 MergePhrases(headParaNormalizePos);
1323 DoPasteCleanup(headParaNormalizePos,reclaimed);
1325 // Adjust the length & phrase count of the new final paragraph
1326 TParaAttribsEntry* lastPara=&(*iParaIx)[aContext.iPastePos.iParaElement+completeParaCount];
1327 lastPara->iLength=(lastPara->iLength-aContext.iPastePos.iParaElementOffset)+textFragment.iLength;
1328 lastPara->iParaAttribs->iPhraseCount-=(aContext.iPastePos.iPhraseElement-aContext.iPastePos.iParaBasePhraseElement);
1329 // phrase count may be wrong - since the added phrase may have been merged into the current one.
1330 // Cant just add these phrases.
1331 lastPara->iParaAttribs->iPhraseCount+=textFragment.iPhraseCount;
1332 ScanToPosition(aContext.iPastePos.iDocPos+characterCount,EScanToPositionAbsolute);
1334 DoPasteCleanup(iPos,(CParaAttribs*)NULL);
1337 NormalizeSharedList();
1341 void CRichTextIndex::DoPasteCleanup(TLogicalPosition& aNormalizePos,CParaAttribs* aReclaimed)
1344 NormalizeNow(aNormalizePos);
1346 CleanupStack::Pop();
1350 TInt CRichTextIndex::PasteParaIxL(RReadStream& aStream,TRtPasteContext& aContext,TInt aCompleteParaCount,const CParaAttribsMap& aMap,RPhraseAttribsEntry*& aFirstParaVirtualPhrase, CStyleMap* aStyleMap)
1353 TParaAttribsEntry para;
1354 iParaIx->InsertL(aContext.iPastePos.iParaElement,para,aCompleteParaCount);
1355 aContext.iParasPasted=aCompleteParaCount;
1356 TInt characterCount=0;
1357 if (aCompleteParaCount>0)
1359 para=DoPasteFirstIntoParaL(aStream,aMap,aContext,aFirstParaVirtualPhrase, aStyleMap);
1361 (*iParaIx)[aContext.iPastePos.iParaElement+paraItem]=para;
1362 characterCount+=para.iLength;
1364 CleanupStack::PushL(TCleanupItem(DiscardPhraseOnCleanup,aFirstParaVirtualPhrase));
1365 for (TInt paraItem=1;paraItem<aCompleteParaCount;paraItem++)
1366 {// Paste all the paras that have paragraph delimiters.
1367 para=DoPasteIntoParaL(aStream,aMap,aContext,aStyleMap);
1368 (*iParaIx)[aContext.iPastePos.iParaElement+paraItem]=para;
1369 characterCount+=para.iLength;
1371 CleanupStack::Pop(); // firstParaVirtualPhrase
1372 // ASSERT: At this point we have pasted all paras that were in the stream.
1373 if (aContext.iParasPasted!=aCompleteParaCount)
1375 OstTrace0( TRACE_DUMP, CRICHTEXTINDEX_PASTEPARAIXL, "EPasteParaIxError" );
1377 __ASSERT_DEBUG(aContext.iParasPasted==aCompleteParaCount,Panic(EPasteParaIxError));
1378 return characterCount;
1382 TParaAttribsEntry CRichTextIndex::DoPasteFirstIntoParaL(RReadStream& aStream,const CParaAttribsMap& aMap,const TRtPasteContext& aContext,RPhraseAttribsEntry*& aFirstParaVirtualPhrase,CStyleMap* aStyleMap)
1385 TParaAttribsEntry para;
1386 para.iLength=aStream.ReadInt32L();
1387 TUint8 refNo=aStream.ReadUint8L();
1389 {// This is the first pasted para, so if shared, must convert to phrase on the fly.
1390 CParaAttribs* paraAttribs=aMap.Item(refNo);
1391 CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraAttribs->iParaFormat);
1392 CleanupStack::PushL(TCleanupItem(ReleaseOnCleanup,specificParaAttribs));
1394 CCharFormatLayer* layer=CCharFormatLayer::NewCopyBaseL(paraAttribs->iCharFormat);
1395 CleanupStack::PushL(layer);
1396 aFirstParaVirtualPhrase=new(ELeave) RPhraseAttribsEntry(layer,para.iLength);
1397 CleanupStack::Pop(2); // layer & specificParaAttribs
1399 para.iParaAttribs=specificParaAttribs;
1400 if (para.iParaAttribs==NULL)
1402 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DOPASTEFIRSTINTOPARAL, "ESharedFormatsMapIntegrityError" );
1404 __ASSERT_ALWAYS(para.iParaAttribs!=NULL,Panic(ESharedFormatsMapIntegrityError));
1407 {// Have to build up the specific para attribs
1408 CParaFormatLayer* paraLayer=PasteParagraphFormatL(aStream,aContext,aStyleMap);
1409 CleanupStack::PushL(paraLayer);
1410 TInt phraseCount=aStream.ReadInt32L(); // specific phrase count
1411 CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraLayer);
1412 specificParaAttribs->iPhraseCount=phraseCount;
1413 para.iParaAttribs=specificParaAttribs;
1414 CleanupStack::PopAndDestroy(); // a copy of paraLayer is taken!
1420 TParaAttribsEntry CRichTextIndex::DoPasteIntoParaL(RReadStream& aStream,const CParaAttribsMap& aMap,const TRtPasteContext& aContext,CStyleMap* aStyleMap)
1423 TParaAttribsEntry para;
1424 para.iLength=aStream.ReadInt32L();
1425 TUint8 refNo=aStream.ReadUint8L();
1427 {// Link to para attribs in shared list & up its reference count
1428 para.iParaAttribs=aMap.Item(refNo);
1429 if (para.iParaAttribs==NULL)
1431 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DOPASTEINTOPARAL, "ESharedFormatsMapIntegrityError" );
1433 __ASSERT_ALWAYS(para.iParaAttribs!=NULL,Panic(ESharedFormatsMapIntegrityError));
1434 para.iParaAttribs->iRefCount++;
1437 {// Have to build up the specific para attribs
1438 CParaFormatLayer* paraLayer=PasteParagraphFormatL(aStream,aContext,aStyleMap);
1439 CleanupStack::PushL(paraLayer);
1440 TInt phraseCount=aStream.ReadInt32L(); // specific phrase count
1441 CParaAttribs* specificParaAttribs=CParaAttribs::NewL(paraLayer);
1442 specificParaAttribs->iPhraseCount=phraseCount;
1443 para.iParaAttribs=specificParaAttribs;
1444 CleanupStack::PopAndDestroy(); // a copy of paraLayer is taken!
1450 TTextFragment CRichTextIndex::GetTextFragmentL(RReadStream& aStream)
1453 TTextFragment textFragment;
1454 textFragment.iLength=aStream.ReadInt32L();
1455 textFragment.iPhraseCount=aStream.ReadInt32L();
1456 return textFragment;
1460 void CRichTextIndex::PastePhraseIxL(RReadStream& aStream,TRtPasteContext& aContext,const RPhraseAttribsEntry* aFirstParaVirtualPhrase)
1461 // The character position to paste at should now fall on a phrase boundary.
1464 // ASSERT: Having pasted the paraIx, the para containig pastePos has had the containing phrase split at that point.
1465 if (aContext.iPastePos.iPhraseElementOffset!=0)
1467 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_PASTEPHRASEIXL, "EPastePhraseIxErr" );
1469 __ASSERT_ALWAYS(aContext.iPastePos.iPhraseElementOffset==0,Panic(EPastePhraseIxErr));
1472 TInt phraseCount=aStream.ReadInt32L(); // leave caught by calling function. No state change yet.
1473 if (aFirstParaVirtualPhrase)
1475 iPhraseIx->InsertL(aContext.iPastePos.iPhraseElement,*aFirstParaVirtualPhrase);
1478 RPhraseAttribsEntry holdingPhrase;
1479 iPhraseIx->InsertL(aContext.iPastePos.iPhraseElement+offset,holdingPhrase,phraseCount);
1480 for (TInt phraseItem=0;phraseItem<phraseCount;phraseItem++)
1481 {// Restore each phrase & insert into the phrase index.
1482 RPhraseAttribsEntry phrase;
1484 // phrase=DoPastePhraseL(aStream,aContext)); // !! delete this if the code works
1485 DoPastePhraseL(aStream,aContext,phrase));
1486 if (ret!=KErrNone) {RbPastePhraseIxL(aContext.iPastePos,phraseCount+offset,ret);}
1487 (*iPhraseIx)[aContext.iPastePos.iPhraseElement+phraseItem+offset]=phrase;
1488 if (phrase.IsPicturePhrase() && iText.PictureFactory())
1494 void CRichTextIndex::DoPastePhraseL(RReadStream& aStream,const TRtPasteContext& aContext,RPhraseAttribsEntry& aPhrase)
1497 TBool isPicture=(TBool)aStream.ReadUint8L();
1498 TInt phraseLength=aStream.ReadInt32L();
1500 if (!(isPicture && phraseLength==1 || !isPicture))
1502 OstTrace0( TRACE_FATAL, CRICHTEXTINDEX_DOPASTEPHRASEL, "KErrCorrupt" );
1504 __ASSERT_ALWAYS(isPicture && phraseLength==1 || !isPicture,User::Leave(KErrCorrupt));
1506 CCharFormatLayer* charLayer=CCharFormatLayer::NewL(aStream);
1507 /// Change this character format if style formatting is to be applied
1509 // Paste style for the trailing text
1510 if (iLastCharacterStyle != NULL)
1511 ImposeCharacterStyleL(&charLayer);
1513 charLayer->SetBase(aContext.iGlobalCharLayer);
1515 CPicturePhrase* picPhrase=NULL;
1516 const MPictureFactory* factory=iText.PictureFactory();
1517 TBool pictureLoadError=EFalse;
1521 aStream>> hdr; // hdr.iPicture always references a picture in the deferred store.
1522 if (isPicture && factory)
1523 {// Manufacture new picture phrase & set its picture store
1524 CleanupStack::PushL(charLayer);
1525 TBool ownershipTaken(EFalse);
1526 picPhrase=CPicturePhrase::NewL(hdr,charLayer,ownershipTaken);
1527 CleanupStack::Pop(); // charLayer - picPhrase takes ownership
1528 CleanupStack::PushL(picPhrase);
1529 TRAPD(r,factory->NewPictureL(hdr,*aContext.iStore)); // load picture since clipboard store is transient.
1530 // r=KErrNotSupported // we don't recognise the picture type
1533 picPhrase->iPicHdr.iPicture=hdr.iPicture; // make picPhrase point at the restored picture object.
1535 hdr.iPicture->DetachFromStoreL(CPicture::EDetachFull)); // recurse the call to detach the picture from the store
1542 pictureLoadError=ETrue;
1543 picPhrase->iCharFormat=NULL;
1544 CleanupStack::PopAndDestroy(picPhrase);
1545 if (r!=KErrNotSupported)
1547 OstTrace1( TRACE_FATAL, DUP2_CRICHTEXTINDEX_DOPASTEPHRASEL, "Leave code=%d", r );
1553 if (isPicture && factory && !pictureLoadError)
1554 new(&aPhrase) RPhraseAttribsEntry(picPhrase);
1556 new(&aPhrase) RPhraseAttribsEntry(charLayer,phraseLength);
1558 // The ownership has been transfered to RPhraseAttribsEntry
1560 CleanupStack::Pop(picPhrase);
1564 TUint8 CRichTextIndex::ReadTUint8CountL(RReadStream& aStream)const
1565 {return aStream.ReadUint8L();}
1568 void CRichTextIndex::RbPasteSharedFormatsL(TInt aRet)
1569 // For each para attribs that has been read in, release all shares on it, if it
1570 // does not already exist in shared list.
1573 NormalizeSharedList();
1574 OstTrace1( TRACE_FATAL, CRICHTEXTINDEX_RBPASTESHAREDFORMATSL, "Leave code=%d", aRet );
1579 void CRichTextIndex::NormalizeSharedList()
1580 // Removes unreferenced (non-shared) shared paras from the shared list.
1581 // Called in rollback situations.
1582 // Also called naturally following a paste, since an incoming shared para
1583 // may be pasted into an existing para, making it non-shared.
1586 CParaAttribs* currentSharedPara=NULL;
1587 TDblQueIter<CParaAttribs> iterator(iSharedParaQueHead);
1588 while ((currentSharedPara=iterator++)!=NULL)
1589 {// Rollback the shared list.
1590 if (currentSharedPara->iRefCount==0)
1591 {// Remove this unreferenced item from the shared list.
1592 currentSharedPara->iRefCount=1;
1593 currentSharedPara->Release();
1599 void CRichTextIndex::RbRemoveInsertedParaAttribsEntries(TInt aFirstParaInsertPos,TInt aInsertedParaCount)
1600 // Rollback on leaving part way through inserting paragraph records into the para index.
1601 // For all pasted paragraph records, release their share on their paraAttribs if present.
1604 for (TInt ii=0;ii<aInsertedParaCount;ii++)
1606 CParaAttribs* paraAttribs=(*iParaIx)[aFirstParaInsertPos+ii].iParaAttribs;
1608 paraAttribs->Release();
1610 iParaIx->Delete(aFirstParaInsertPos,aInsertedParaCount);
1614 void CRichTextIndex::RbPasteParaIxL(const TLogicalPosition& aPos,TInt aParasPasted,TInt aRet)
1615 // Rollback on leaving part way through the pasting of the para index.
1616 // For all pasted paras, release their share on their paraAttribs.
1619 RbRemoveInsertedParaAttribsEntries(aPos.iParaElement,aParasPasted);
1620 MergePhrases(aPos.iDocPos); // updates iPos to paste pos.
1623 if (iRollbackParaAttribsHandle)
1625 RemoveFromPhraseIx(iPos.iPhraseElement,1); // Removes the phrase created from ResquestReclaim on para
1626 (*iParaIx)[aPos.iParaElement].iParaAttribs=iRollbackParaAttribsHandle;
1627 iRollbackParaAttribsHandle=NULL;
1629 OstTrace1( TRACE_FATAL, CRICHTEXTINDEX_RBPASTEPARAIXL, "Leave code=%d", aRet );
1634 void CRichTextIndex::RbRemoveInsertedPhraseAttribsEntries(TInt aFirstPhraseInsertPos,TInt aInsertedPhraseCount)
1635 // Rollback on leaving part way through the pasting of the phrase index.
1636 // For all pasted phrases, discard their components.
1639 for (TInt ii=0;ii<aInsertedPhraseCount;ii++)
1641 RPhraseAttribsEntry& phrase=(*iPhraseIx)[aFirstPhraseInsertPos+ii];
1642 if (phrase.IsPicturePhrase())
1644 if (phrase.CharFormat())
1647 iPhraseIx->Delete(aFirstPhraseInsertPos,aInsertedPhraseCount);
1651 void CRichTextIndex::RbPastePhraseIxL(const TLogicalPosition& aPos,TInt aPhraseCount,TInt aRet)
1652 // Rollback on leaving part way through the pasting of the phrase index.
1653 // For all pasted phrases, discard their components.
1656 RbRemoveInsertedPhraseAttribsEntries(aPos.iPhraseElement,aPhraseCount);
1657 OstTrace1( TRACE_FATAL, CRICHTEXTINDEX_RBPASTEPHRASEIXL, "Leave code=%d", aRet );