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.
42 #include "TXTCLIPBOARD.H"
44 #include "OstTraceDefinitions.h"
45 #ifdef OST_TRACE_COMPILER_IN_USE
46 #include "TXTETEXTTraces.h"
49 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
50 #include "TXTETEXT_INTERNAL.H"
53 const TUint KFieldCountLimit = 255;
54 #define UNUSED_VAR(a) a = a
56 // Write some or all of the text in a buffer to a stream, writing the length first if aWriteLength is true.
57 static void ExternalizeTextL(RWriteStream& aStream,const CBufBase& aText,TInt aPos,TInt aLength,TBool aWriteLength)
60 aStream << TCardinality(aLength);
62 // Use the Standard Unicode Compression Scheme.
63 RBufReadStream input_stream(aText,aPos * sizeof(TText));
64 TMemoryStreamUnicodeSource source(input_stream);
66 c.CompressL(aStream,source,KMaxTInt,aLength);
70 // Read text from a stream and write it to a buffer.
71 static void InternalizeTextL(RReadStream& aStream,CBufBase& aText,TInt aLength)
73 // Use the Standard Unicode Compression Scheme.
74 RBufWriteStream output_stream(aText);
75 TMemoryStreamUnicodeSink sink(output_stream);
77 e.ExpandL(sink,aStream,aLength);
78 output_stream.CommitL();
79 output_stream.Close();
82 // Read text from a stream and write it to a buffer; read the length first.
83 static void InternalizeTextL(RReadStream& aStream,CBufBase& aText)
87 InternalizeTextL(aStream,aText,length);
90 Returns the interface corresponding to the
91 specified UID if it exists, or 0 if not. Overridden
92 versions should base call rather than returning 0.
94 @param aInterfaceId The UID indicating the interface to return
95 @param aInterface The interface corresponding to aInterfaceId
96 if it is supported, or 0 if it is not
98 EXPORT_C void CEditableText::ExtendedInterface(TAny*& /*aInterface*/, TUid /*aInterfaceId*/) {}
101 Returns the interface corresponding to the
102 specified UID if it exists, or 0 if not. Overridden
103 versions should base call rather than returning 0.
105 @param aInterfaceId The UID indicating the interface to return
106 @param aInterface The interface corresponding to aInterfaceId
107 if it is supported, or 0 if it is not
109 EXPORT_C void CPlainText::ExtendedInterface(TAny*& /*aInterface*/, TUid /*aInterfaceId*/) {}
115 EXPORT_C void CPlainText::Reserved_2() {}
117 //////////////////////////////////
119 //////////////////////////////////
121 EXPORT_C CEditableText::~CEditableText()
123 delete iOptionalData;
128 EXPORT_C TInt CEditableText::ScanWords(TInt& /*aPos*/,TUint& /*aScanMask*/) const
129 /** Scans the text from a specified document position to a location
130 determined by the flags specified in a bitmask. The function can scan
131 forwards or backwards to the beginning or end of a word.
133 @param aPos A valid document position from which to scan. On return,
134 contains the new document position.
135 @param aScanMask The scan mask to use. See the scanning enumeration defined
137 @return The number of characters skipped to reach the new document position.
138 Notes: If the scan passes the end of text delimiter, on return, aPos is set
139 to EScanEndOfData and the function's return value indicates the
140 number of characters skipped in passing the end of text delimiter. */
146 EXPORT_C TInt CEditableText::ScanParas(TInt& /*aPos*/,TUint& /*aScanMask*/) const
147 /** Scans the text from a specified document position to a location determined
148 by the flags specified in a bitmask. The function can scan forwards or backwards
149 to the beginning or end of a paragraph.
151 @param aPos A valid document position from which to scan. On return, contains
152 the new document position.
153 @param aScanMask The scan mask to use. See the scanning enumeration defined
155 @return The number of characters skipped to reach the new document position.
156 Notes: If the scan passes the end of text delimiter, on return, aPos is set
157 to EScanEndOfData and the function's return value indicates the
158 number of characters skipped in passing the end of text delimiter. */
165 EXPORT_C void CEditableText::SetHasChanged(TBool aHasChanged)
166 /** Sets whether a change has occurred to the editable text object. This is called
167 by functions which change the text object in some way.
169 @param aHasChanged ETrue if a change has occurred to the text object. EFalse
170 if no change has occurred. */
172 iHasChanged = aHasChanged;
176 // Save the editable text type identifier.
177 void CEditableText::ExternalizeL(RWriteStream& aStream)const
179 aStream << KEditableTextUid;
183 void CEditableText::InternalizeL(RReadStream& aStream)
184 // Read from the stream, expecting the editable text type identifier
189 if (uid!=KEditableTextUid)
190 User::Leave(KErrCorrupt);
196 EXPORT_C TStreamId CEditableText::StoreL(CStreamStore& aStore)const
197 /** Stores the text and its components. The components (e.g. fields and pictures)
198 are stored in separate streams within the stream store.
200 @param aStore Stream store to which the text and text components are written.
201 @return The ID of the stream store. */
203 CStoreMap* map=CStoreMap::NewLC(aStore);
204 StoreComponentsL(aStore,*map);
206 RStoreWriteStream stream(*map);
207 TStreamId id=stream.CreateLC(aStore);
212 CleanupStack::PopAndDestroy(2); // map,stream
218 EXPORT_C void CEditableText::RestoreL(const CStreamStore& aStore,TStreamId aStreamId)
219 /** Restores the text and its components from a stream store.
221 @param aStore Stream store containing the text and its components.
222 @param aStreamId The ID of the stream store in which the text was previously
225 // Load text and field components only. (Pictures, if present, are deferred loaded).
226 __ETEXT_WATCH(RESTORE)
228 RStoreReadStream stream;
229 stream.OpenLC(aStore,aStreamId);
232 CleanupStack::PopAndDestroy(); // stream
233 RestoreComponentsL(aStore);
235 __ETEXT_WATCH_END(RESTORE)
238 TBool CEditableText::DeleteWithoutDestroyingFormatL(TInt aPos, TInt aLength)
239 /** Deletes a range of characters. For rich text the format of the deleted character
240 at position aPos is preserved, so that any text subsequently inserted at aPos will have
241 that format applied to it.
242 @param aPos The document position from which to begin deleting including aPos.
243 @param aLength The number of characters to delete. Must be positive or a panic
244 occurs. The sum of aPos and aLength must be less than the document length,
246 @return Indicates whether two paragraphs have been merged together as a result
247 of the delete, indicating that the resulting paragraph must be reformatted.
250 TAny* richTextInterface = NULL;
251 ExtendedInterface(richTextInterface, KUidRichText);
253 if(richTextInterface)
255 return REINTERPRET_CAST(CRichText*, richTextInterface)->DelSetInsertCharFormatL(aPos, aLength);
259 return DeleteL(aPos, aLength);
263 EXPORT_C void CEditableText::StartFepInlineEditL(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument,const TDesC& aInitialInlineText,TInt aPositionOfInlineTextInDocument,TInt aNumberOfCharactersToHide,MFepInlineTextFormatRetriever& aInlineTextFormatRetriever)
266 if (aPositionOfInlineTextInDocument<0 || aNumberOfCharactersToHide<0 || aPositionOfInlineTextInDocument+aNumberOfCharactersToHide>DocumentLength())
268 OstTrace0( TRACE_DUMP, CEDITABLETEXT_STARTFEPINLINEEDITL, "ECharPosBeyondDocument" );
270 __ASSERT_ALWAYS(aPositionOfInlineTextInDocument>=0 && aNumberOfCharactersToHide>=0 && aPositionOfInlineTextInDocument+aNumberOfCharactersToHide<=DocumentLength(),Panic(ECharPosBeyondDocument));
271 if (InlineEditData()!=NULL)
273 OstTrace0( TRACE_DUMP, DUP1_CEDITABLETEXT_STARTFEPINLINEEDITL, "EAlreadyFepInlineEditing" );
275 __ASSERT_ALWAYS(InlineEditData()==NULL,Panic(EAlreadyFepInlineEditing));
276 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
277 aNumberOfCharactersSuccessfullyDeleted=0;
278 aNumberOfCharactersSuccessfullyInserted=0;
279 CInlineEditData* const inlineEditData=new(ELeave) CInlineEditData;
280 CleanupStack::PushL(inlineEditData);
281 HBufC* hiddenText=NULL;
282 CleanupStack::PushL(hiddenText);
283 if (aNumberOfCharactersToHide>0)
285 CleanupStack::Pop(); // hiddenText
286 hiddenText=HBufC::NewLC(aNumberOfCharactersToHide);
287 TPtr hiddenTextAsWritableDescriptor=hiddenText->Des();
288 Extract(hiddenTextAsWritableDescriptor,aPositionOfInlineTextInDocument,aNumberOfCharactersToHide);
289 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=DeleteWithoutDestroyingFormatL(aPositionOfInlineTextInDocument,aNumberOfCharactersToHide);
290 aNumberOfCharactersSuccessfullyDeleted=aNumberOfCharactersToHide;
291 aPositionOfInsertionPointInDocument=aPositionOfInlineTextInDocument;
293 inlineEditData->iPositionOfInlineTextInDocument=aPositionOfInlineTextInDocument;
294 inlineEditData->iLengthOfInlineText=aInitialInlineText.Length();
295 inlineEditData->iInlineText=aInitialInlineText.AllocL();
296 inlineEditData->iHiddenText=hiddenText;
297 CleanupStack::Pop(); // hiddentext now owned by inlineEditData.
298 inlineEditData->iInlineTextFormatRetriever=&aInlineTextFormatRetriever;
299 InsertL(aPositionOfInlineTextInDocument,aInitialInlineText);
300 SetAndTransferOwnershipOfInlineEditDataL(inlineEditData);
301 CleanupStack::Pop(); // inlineEditData
302 aNumberOfCharactersSuccessfullyInserted=inlineEditData->iLengthOfInlineText;
303 aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
307 EXPORT_C void CEditableText::UpdateFepInlineTextL(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument,const TDesC& aNewInlineText)
310 CInlineEditData* const inlineEditData=InlineEditData();
311 if (inlineEditData==NULL)
313 OstTrace0( TRACE_DUMP, CEDITABLETEXT_UPDATEFEPINLINETEXTL, "ENotCurrentlyFepInlineEditing" );
315 __ASSERT_ALWAYS(inlineEditData!=NULL,Panic(ENotCurrentlyFepInlineEditing));
316 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
317 aNumberOfCharactersSuccessfullyDeleted=0;
318 aNumberOfCharactersSuccessfullyInserted=0;
319 HBufC*& inlineText=inlineEditData->iInlineText;
320 HBufC* oldInlineText=inlineText;
321 if (oldInlineText!=NULL && inlineEditData->iLengthOfInlineText!=oldInlineText->Length())
323 OstTrace0( TRACE_DUMP, DUP1_CEDITABLETEXT_UPDATEFEPINLINETEXTL, "EDebug" );
325 __ASSERT_DEBUG(oldInlineText==NULL || inlineEditData->iLengthOfInlineText==oldInlineText->Length(),Panic(EDebug));
326 const TInt lengthOfNewInlineText=aNewInlineText.Length();
327 if (oldInlineText!=NULL && *oldInlineText==aNewInlineText)
329 aNumberOfCharactersSuccessfullyDeleted=lengthOfNewInlineText;
330 aNumberOfCharactersSuccessfullyInserted=lengthOfNewInlineText;
334 if (oldInlineText==NULL)
336 oldInlineText=HBufC::NewL(lengthOfNewInlineText);
338 else if (lengthOfNewInlineText>oldInlineText->Length())
340 oldInlineText=oldInlineText->ReAllocL(lengthOfNewInlineText);
342 CleanupStack::PushL(oldInlineText);
343 inlineText=NULL; // sets inlineEditData->iLengthOfInlineText in case either the delete or the insert leaves
344 const TInt positionOfInlineTextInDocument=inlineEditData->iPositionOfInlineTextInDocument;
345 TInt& lengthOfInlineText=inlineEditData->iLengthOfInlineText;
346 if (lengthOfInlineText>0)
348 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=DeleteWithoutDestroyingFormatL(positionOfInlineTextInDocument,lengthOfInlineText);
349 aNumberOfCharactersSuccessfullyDeleted=lengthOfInlineText;
350 lengthOfInlineText=0; // sets inlineEditData->iLengthOfInlineText in case the insert leaves
351 aPositionOfInsertionPointInDocument=inlineEditData->iPositionOfInlineTextInDocument;
353 InsertL(positionOfInlineTextInDocument,aNewInlineText);
354 lengthOfInlineText=aNewInlineText.Length(); // sets inlineEditData->iLengthOfInlineText
355 aNumberOfCharactersSuccessfullyInserted=lengthOfInlineText;
356 inlineText=oldInlineText;
357 CleanupStack::Pop(); // oldInlineText
358 *oldInlineText=aNewInlineText;
360 aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
364 EXPORT_C void CEditableText::CommitFepInlineEditL(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument)
369 const CInlineEditData* const inlineEditData=InlineEditData();
370 if (inlineEditData==NULL)
372 OstTrace0( TRACE_FATAL, DUP1_CEDITABLETEXT_COMMITFEPINLINEEDITL, "ENotCurrentlyFepInlineEditing" );
374 __ASSERT_ALWAYS(inlineEditData!=NULL,Panic(ENotCurrentlyFepInlineEditing));
375 if (inlineEditData->iInlineText!=NULL && inlineEditData->iLengthOfInlineText!=inlineEditData->iInlineText->Length())
377 OstTrace0( TRACE_DUMP, CEDITABLETEXT_COMMITFEPINLINEEDITL, "EDebug" );
379 __ASSERT_DEBUG(inlineEditData->iInlineText==NULL || inlineEditData->iLengthOfInlineText==inlineEditData->iInlineText->Length(),Panic(EDebug));
380 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
381 const TInt lengthOfInlineText=inlineEditData->iLengthOfInlineText;
382 aNumberOfCharactersSuccessfullyDeleted=lengthOfInlineText;
383 aNumberOfCharactersSuccessfullyInserted=lengthOfInlineText;
384 aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
385 DeleteInlineEditDataAndSetToNull();
389 EXPORT_C void CEditableText::CancelFepInlineEdit(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument)
394 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
395 aNumberOfCharactersSuccessfullyDeleted=0;
396 aNumberOfCharactersSuccessfullyInserted=0;
397 const CInlineEditData* inlineEditData=InlineEditData();
398 if (inlineEditData!=NULL)
400 const TInt positionOfInlineTextInDocument=inlineEditData->iPositionOfInlineTextInDocument;
401 const TInt lengthOfInlineText=inlineEditData->iLengthOfInlineText;
402 if (inlineEditData->iInlineText!=NULL && lengthOfInlineText!=inlineEditData->iInlineText->Length())
404 OstTrace0( TRACE_DUMP, CEDITABLETEXT_CANCELFEPINLINEEDIT, "EDebug" );
406 __ASSERT_DEBUG(inlineEditData->iInlineText==NULL || lengthOfInlineText==inlineEditData->iInlineText->Length(),Panic(EDebug));
408 if (lengthOfInlineText>0)
410 aParagraphContainingStartPositionOfInlineTextHasChangedFormat=DeleteWithoutDestroyingFormatL(positionOfInlineTextInDocument,lengthOfInlineText);
411 aNumberOfCharactersSuccessfullyDeleted=lengthOfInlineText;
412 aPositionOfInsertionPointInDocument=inlineEditData->iPositionOfInlineTextInDocument;
414 const HBufC* const hiddenText=inlineEditData->iHiddenText;
415 if (hiddenText!=NULL)
417 if (hiddenText->Length()<=0)
419 OstTrace0( TRACE_DUMP, DUP1_CEDITABLETEXT_CANCELFEPINLINEEDIT, "EDebug" );
421 __ASSERT_DEBUG(hiddenText->Length()>0, Panic(EDebug));
422 InsertL(positionOfInlineTextInDocument,*hiddenText);
423 aNumberOfCharactersSuccessfullyInserted=hiddenText->Length();
424 aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
428 DeleteInlineEditDataAndSetToNull();
432 EXPORT_C void CEditableText::OverrideFormatOfInlineTextIfApplicable(TPtrC& aView,TCharFormat& aFormat,TInt aStartPos)const
434 const CInlineEditData* inlineEditData=InlineEditData();
435 if (inlineEditData!=NULL)
437 const TInt positionOfInlineTextInDocument=inlineEditData->iPositionOfInlineTextInDocument;
438 const TInt lengthOfInlineText=inlineEditData->iLengthOfInlineText;
439 const TInt originalLengthOfView=aView.Length();
440 TInt maximumLengthOfView=originalLengthOfView;
441 if (aStartPos<positionOfInlineTextInDocument)
443 maximumLengthOfView=positionOfInlineTextInDocument-aStartPos;
445 else if (aStartPos<positionOfInlineTextInDocument+lengthOfInlineText)
447 inlineEditData->iInlineTextFormatRetriever->GetFormatOfFepInlineText(aFormat,maximumLengthOfView,aStartPos-positionOfInlineTextInDocument);
449 if (originalLengthOfView>maximumLengthOfView)
451 aView.Set(aView.Left(maximumLengthOfView));
457 EXPORT_C TInt CEditableText::GetPositionOfInlineTextInDocument() const
459 const CInlineEditData* inlineEditData=InlineEditData();
460 if (inlineEditData==NULL)
462 return inlineEditData->iPositionOfInlineTextInDocument;
466 EXPORT_C TInt CEditableText::GetLengthOfInlineText() const
468 const CInlineEditData* inlineEditData=InlineEditData();
469 if (inlineEditData==NULL)
471 return inlineEditData->iLengthOfInlineText;
475 //////////////////////////////////
476 // TEtextComponentInfo
477 //////////////////////////////////
481 EXPORT_C TEtextComponentInfo::TEtextComponentInfo()
482 : iFieldCount(0),iPictureCount(0),iStyleCount(0)
483 /** C++ constructor overloaded function.
485 The object can be constructed either:by default this initializes the
486 field, picture and style counts to zerowith a field, picture and style
489 @param aFieldCount Specifies the number of fields in the text object.
490 @param aPictureCount Specifies the number of pictures in the text object
492 @param aStyleCount Specifies the number of styles owned or referenced by
493 the text object (rich text only). */
497 EXPORT_C TEtextComponentInfo::TEtextComponentInfo(TInt aFieldCount,TInt aPictureCount,TInt aStyleCount)
498 : iFieldCount(aFieldCount),iPictureCount(aPictureCount),iStyleCount(aStyleCount)
499 /** C++ constructor overloaded function. The object can be constructed either:
501 by default this initializes the field, picture and style counts to zero
503 with a field, picture and style count
505 @param aFieldCount Specifies the number of fields in the text object.
506 @param aPictureCount Specifies the number of pictures in the text object (rich
508 @param aStyleCount Specifies the number of styles owned or referenced by the
509 text object (rich text only). */
513 //////////////////////////////////
515 //////////////////////////////////
518 EXPORT_C void CPlainText::__DbgTestInvariant()const
519 // Provides class invariants. Explanations below:
523 // ASSERT: Storage handle is good.
524 __ASSERT_DEBUG(iByteStore!=NULL,User::Invariant());
525 // ASSERT: The text component must be non-negative in length
526 __ASSERT_DEBUG(DocumentLength()>=0,User::Invariant());
533 EXPORT_C CPlainText* CPlainText::NewL(TDocumentStorage aStorage,TInt aDefaultTextGranularity)
534 /** Allocates and constructs a plain text object overloaded function.
536 The text object's contents may be restored from a stream store. If the
537 text object supports fields, a field factory should be specified.
539 @param aStorage The type of in-memory buffer to use. Defaults to ESegmentedStorage.
540 @param aDefaultTextGranularity Specifies the granularity of the in-memory
541 buffer. Default is EDefaultTextGranularity bytes (=256).
542 @param aStore Stream store from which the object is restored.
543 @param aStreamId ID of the stream store.
544 @param aFieldFactory Pointer to a field factory. A field factory must be
545 provided if the text object supports the addition of fields.
546 @return Pointer to the new plain text object. */
548 CPlainText* self=new(ELeave) CPlainText();
549 CleanupStack::PushL(self);
550 self->ConstructL(aStorage,aDefaultTextGranularity);
556 EXPORT_C CPlainText* CPlainText::NewL(const CStreamStore& aStore,TStreamId aStreamId,MTextFieldFactory* aFieldFactory,TDocumentStorage aStorage)
557 /** Returns a handle to a new instance of this class, restored from the specified
560 CPlainText* self=new(ELeave) CPlainText();
561 CleanupStack::PushL(self);
562 self->ConstructL(aStore,aStreamId,aFieldFactory,aStorage);
568 EXPORT_C CPlainText::CPlainText()
570 SetHasChanged(EFalse);
574 EXPORT_C void CPlainText::ConstructL(TDocumentStorage aStorage,TInt aDefaultTextGranularity)
575 /** Allocates storage of CBufFlat or CBufSeg, according
576 to the parameter aStorage.
577 Creates & initializes the field set.*/
579 DoConstructL(aStorage,aDefaultTextGranularity);
586 EXPORT_C void CPlainText::ConstructL(const CStreamStore& aStore,TStreamId aStreamId,MTextFieldFactory* aFieldFactory,
587 TDocumentStorage aStorage)
588 /** Allocates storage of CBufFlat or CBufSeg, according
589 to the parameter aStorage, restoring contents from the specified read-stream aStream.*/
591 DoConstructL(aStorage,EDefaultTextGranularity,aFieldFactory);
592 RestoreL(aStore,aStreamId);
598 EXPORT_C void CPlainText::DoConstructL(TDocumentStorage aStorage,TInt aDefaultTextGranularity,MTextFieldFactory* aFieldFactory)
599 /** Allocates storage of CBufFlat or CBufSeg, according to the parameter aStorage.
600 Creates & initializes the field set.*/
602 if (iByteStore!=NULL)
604 OstTrace0( TRACE_DUMP, CPLAINTEXT_DOCONSTRUCTL, "EConstructCalledTwice" );
606 __ASSERT_DEBUG(iByteStore==NULL,Panic(EConstructCalledTwice));
608 iByteStore=(aStorage==ESegmentedStorage)
609 ? (CBufBase*)CBufSeg::NewL(aDefaultTextGranularity*sizeof(TText))
610 : (CBufBase*)CBufFlat::NewL(aDefaultTextGranularity*sizeof(TText));
612 SetFieldFactory(aFieldFactory);
617 EXPORT_C CPlainText::~CPlainText()
618 /** The destructor frees the object's text storage and field set, prior to its
626 void CPlainText::KillFieldSet()
627 /** Delete the field set if it is resident in memory.*/
629 if (FieldSetPresent())
630 delete iFieldSet.AsPtr();
636 // CPlainText - Persistence
639 EXPORT_C void CPlainText::StoreComponentsL(CStreamStore& aStore,CStoreMap& aMap)const
640 /** Stores the plain text object's components to the stream store specified.
642 @param aStore Stream store to which the text object's components are written.
643 @param aMap A store map. This binds the address of each component to the stream
644 ID of aStore. This is needed to support the deferred loading of pictures in
647 // Store any field components, then store the field set out-of-line, if present.
648 if (FieldSetPresent())
650 TStreamId id=iFieldSet->StoreL(aStore);
651 aMap.BindL(iFieldSet,id);
657 EXPORT_C void CPlainText::RestoreComponentsL(const CStreamStore& aStore)
658 /** Restores the plain text object's field set from a stream store.
660 @param aStore The stream store from which the field set is restored. */
662 // Load the field set, and load any referenced pictures
663 TStreamId id=iFieldSet.AsId();
664 if (id!=KNullStreamId)
667 iFieldSet->SetFieldFactory(iFieldFactory);
668 iFieldSet->RestoreL(aStore,id);
674 EXPORT_C void CPlainText::StoreFieldComponentsL(CStreamStore& aStore,CStoreMap& aMap)const
675 /** Stores the plain text object's field components to a stream store.
677 @param aStore Stream store to which the fields are written.
678 @param aMap A store map. This binds the address of each text component to the
679 stream ID of aStore. This is needed to support the deferred loading of pictures
682 // 2' StoreComponents()
683 // Only has effect if a field set is present.
684 if (FieldSetPresent())
685 iFieldSet->StoreFieldsL(aStore,aMap);
690 EXPORT_C void CPlainText::RestoreFieldComponentsL(const CStreamStore& aStore)
691 /** Restores the plain text object's field set.
693 @param aStore The stream store from which the fields are restored. */
695 // 2' RestoreComponents()
696 // Only has effect if a field set is present - (has been Internalized())
697 if (FieldSetPresent())
698 iFieldSet->RestoreFieldsL(aStore);
703 EXPORT_C void CPlainText::ExternalizeL(RWriteStream& aStream)const
704 /** Externalises a plain text object to a write stream. The presence of this function
705 means that the standard templated operator<<() (defined in s32strm.h) is available
706 to externalise objects of this class.
708 @param aStream Stream to which the object should be externalised. */
710 // Store this object in the specified write-stream.
711 CEditableText::ExternalizeL(aStream);
712 DoExternalizeFieldDataL(aStream);
713 DoExternalizePlainTextL(aStream);
718 EXPORT_C void CPlainText::InternalizeL(RReadStream& aStream)
719 /** Internalises the text object's text content and field set from a read stream.
720 The presence of this function means that the standard templated operator>>()
721 (defined in s32strm.h) is available to internalise objects of this class.
722 InternalizeL() has construct rather than assignment semantics. You should
723 not use it for fully initialised objects.
725 @param aStream Stream from which the object should be internalised. */
727 // Restores plain text from the specified read-stream.
728 // Internalize has construction semantics, not assignment semantics.
729 CEditableText::InternalizeL(aStream);
730 DoInternalizeFieldDataL(aStream);
731 DoInternalizePlainTextL(aStream);
738 EXPORT_C void CPlainText::ExternalizeFieldDataL(RWriteStream& aStream)const
739 /** Externalises the plain text object's field set.
741 @param aStream The stream to which the field set should be written. */
743 // Save just the field set
746 TUint fieldCount=(TUint)FieldCount();
747 if(fieldCount<KFieldCountLimit)
748 aStream.WriteUint8L(fieldCount);
751 aStream.WriteUint8L(KFieldCountLimit);
752 aStream.WriteUint32L(fieldCount);
755 aStream<< *iFieldSet;
760 EXPORT_C void CPlainText::InternalizeFieldDataL(RReadStream& aStream)
761 /** Internalizes the field set.
763 @param aStream The read stream from which the field set is read. */
766 // Restores field records from the specified read-stream.
767 // Internalize has construction semantics, not assignment semantics.
769 TUint fieldCount=aStream.ReadUint8L();
770 if (fieldCount==KFieldCountLimit)
771 fieldCount=aStream.ReadUint32L();
775 CreateFieldSetL(DocumentLength());
776 aStream>> *iFieldSet;
781 EXPORT_C void CPlainText::DoInternalizeFieldDataL(RReadStream& aStream)
782 /** Read from the stream until the field data is identified, and consume it.*/
784 TUid uid=UidFromStreamL(aStream);
785 while (uid!=KPlainTextFieldDataUid)
787 if (uid==KPlainTextCharacterDataUid)
788 User::Leave(KErrCorrupt); // There is no field Data !!!!!
789 CPlainText::ConsumeAdornmentL(aStream);
790 uid=UidFromStreamL(aStream);
792 if (FieldSetPresent())
793 iFieldFactory=iFieldSet->FieldFactory();
795 aStream>> iFieldSet; // a swizzle
799 EXPORT_C void CPlainText::DoExternalizeFieldDataL(RWriteStream& aStream)const
800 /** Write to the stream, the T.V. representing the field set.*/
802 aStream<< KPlainTextFieldDataUid;
803 if (FieldSetPresent())
806 aStream<< KNullStreamId;
810 EXPORT_C void CPlainText::DoExternalizePlainTextL(RWriteStream& aStream)const
811 /** Write to the stream, the T.V. representing the plain text.*/
813 aStream<< KPlainTextCharacterDataUid;
814 ExternalizePlainTextL(aStream);
819 EXPORT_C void CPlainText::ExternalizePlainTextL(RWriteStream& aStream)const
821 /** Externalises the plain text object's text content (preceded by a length count)
824 @param aStream Stream to which the text content should be externalised. */
826 // Save just the bytestore
828 ::ExternalizeTextL(aStream,*iByteStore,0,iByteStore->Size() / sizeof(TText),TRUE);
831 EXPORT_C void CPlainText::DoInternalizePlainTextL(RReadStream& aStream)
832 /** Read from the stream until the character data is found, and consume it.*/
834 TUid uid=UidFromStreamL(aStream);
835 while (uid!=KPlainTextCharacterDataUid)
837 CPlainText::ConsumeAdornmentL(aStream);
838 uid=UidFromStreamL(aStream);
840 CPlainText::InternalizePlainTextL(aStream);
843 EXPORT_C void CPlainText::InternalizePlainTextL(RReadStream& aStream)
844 /** Internalises an empty text object's text content from a read stream
847 This function has construct rather than assignment semantics. You
848 should not use it for fully initialised objects.NoteThe overload which
849 takes a length argument is not intended for general use and its use is
852 @param aStream Stream from which the object should be internalised.
853 @param aLength Indicates the number of characters which should be
854 read, after expansion from their compressed format. */
856 // Restores plain text content from the specified read-stream.
857 // Internalize has construction semantics, not assignment semantics.
858 ::InternalizeTextL(aStream,*iByteStore);
859 SetHasChanged(EFalse);
865 EXPORT_C void CPlainText::ExternalizePlainTextNoLengthCountL(RWriteStream& aStream)const
866 /** Externalises the plain text object's text content to a write stream.
868 This function differs from ExternalizePlainTextL() in that
869 it does not precede the text content with a length count.
870 This function is not intended for general use and is deprecated.
871 @see void CPlainText::ExternalizePlainTextL(RWriteStream& aStream)const
874 ::ExternalizeTextL(aStream,*iByteStore,0,iByteStore->Size() / sizeof(TText),FALSE);
878 EXPORT_C void CPlainText::InternalizePlainTextL(RReadStream& aStream,TInt aLength)
879 /** Internalises an empty text object's text content from a read stream's
882 This function has construct rather than assignment semantics. You should not
883 use it for fully initialised objects.
887 The overload which takes a length argument is not intended for general use
888 and its use is deprecated.
890 @param aStream Stream from which the object should be internalised.
891 @param aLength Indicates the number of characters which should be read, after
892 expansion from their compressed format.
895 ::InternalizeTextL(aStream,*iByteStore,aLength);
899 // Copy the specified section of plain text to the specified store.
902 EXPORT_C void CPlainText::CopyToStoreL(CStreamStore& aStore,CStreamDictionary& aDictionary,TInt aPos,TInt aLength) const
903 /** Copies plain text including fields, if present, to the clipboard.
905 A panic occurs in any of the following circumstances:
909 aLength is invalid (zero or less)
911 the sum of aPos and aLength is greater than or equal to the number of characters
914 @param aStore Stream store to which the text is written.
915 @param aDictionary The stream dictionary.
916 @param aPos The document position from which to begin copying.
917 @param aLength The number of characters to copy. */
920 DoCopyToStoreL(aStore,aDictionary,aPos,aLength);
924 TStreamId CPlainText::DoCopyToStoreL(CStreamStore& aStore,CStreamDictionary& aDictionary,TInt aPos,TInt aLength) const
927 TInt documentLength = DocumentLength();
928 if (aPos < 0 || aPos > documentLength)
930 OstTrace0( TRACE_FATAL, CPLAINTEXT_DOCOPYTOSTOREL, "ECharPosBeyondDocument" );
932 __ASSERT_ALWAYS(aPos >= 0 && aPos <= documentLength,Panic(ECharPosBeyondDocument));
935 OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_DOCOPYTOSTOREL, "ECopyToStreamNegativeLength" );
937 __ASSERT_ALWAYS(aLength >= 0,Panic(ECopyToStreamNegativeLength));
938 if (aPos + aLength > documentLength)
940 OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_DOCOPYTOSTOREL, "ECharPosBeyondDocument" );
942 __ASSERT_ALWAYS(aPos + aLength <= documentLength,Panic(ECharPosBeyondDocument));
945 return KNullStreamId;
947 CStoreMap* map=CStoreMap::NewLC(aStore);
948 CopyComponentsL(aStore,*map,aPos,aLength);
950 // create custom externalizer over the map
951 TFieldMapExternalizer fMap(*map);
952 RStoreWriteStream stream(fMap);
953 TStreamId id=stream.CreateLC(aStore);
954 CopyToStreamL(stream,aPos,aLength);
957 aDictionary.AssignL(KClipboardUidTypePlainText,id);
959 CleanupStack::PopAndDestroy(2);
966 void CPlainText::CopyComponentsL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength)const
967 // Copy/Paste 2' StoreComponentsL() - only if a field set is present.
970 if (FieldSetPresent())
971 iFieldSet->CopyComponentsL(aStore,aMap,aPos,aLength);
975 // Write the plain text to the stream.
976 void CPlainText::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength)const
979 TInt documentLength = DocumentLength();
980 if (aPos < 0 || aPos > documentLength)
982 OstTrace0( TRACE_FATAL, CPLAINTEXT_COPYTOSTREAML, "ECharPosBeyondDocument" );
984 __ASSERT_ALWAYS(aPos >= 0 && aPos <= documentLength,Panic(ECharPosBeyondDocument));
987 OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_COPYTOSTREAML, "ECopyToStreamNegativeLength" );
989 __ASSERT_ALWAYS(aLength >= 0,Panic(ECopyToStreamNegativeLength));
990 if (aPos + aLength > documentLength)
992 OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_COPYTOSTREAML, "ECharPosBeyondDocument" );
994 __ASSERT_ALWAYS(aPos + aLength <= documentLength,Panic(ECharPosBeyondDocument));
996 aStream.WriteInt32L(aLength);
997 ::ExternalizeTextL(aStream,*iByteStore,aPos,aLength,FALSE);
999 // Write the field set if any.
1000 TBool fieldSetPresent = FieldSetPresent();
1001 aStream.WriteUint8L(fieldSetPresent != EFalse);
1002 if (fieldSetPresent)
1003 iFieldSet->CopyToStreamL(aStream,aPos,aLength);
1009 EXPORT_C TInt CPlainText::PasteFromStoreL(const CStreamStore& aStore,const CStreamDictionary& aDictionary,TInt aPos)
1010 /** Pastes plain text and fields, if present, from the clipboard into the current
1011 text object at the specified document position. The entire contents of the
1014 @param aStore The steam store from which to paste the text.
1015 @param aDictionary The stream dictionary.
1016 @param aPos Document position at which to paste. Must be valid or the function
1018 @return The number of characters pasted. */
1020 // Paste the lesser of aMaxPasteLength and the entire clipboard contents.
1021 // Return the number of characters pasted.
1022 TStreamId id=aDictionary.At(KClipboardUidTypePlainText);
1023 return DoPasteFromStoreL(aStore,id,aPos);
1027 TInt CPlainText::DoPasteFromStoreL(const CStreamStore& aStore,TStreamId aStreamId,TInt aPos)
1029 if (aPos>DocumentLength())
1031 OstTrace0( TRACE_FATAL, CPLAINTEXT_DOPASTEFROMSTOREL, "ECharPosBeyondDocument" );
1033 __ASSERT_ALWAYS(aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1036 if (aStreamId!=KNullStreamId)
1037 {// There is a recognised type in the clipboard.
1038 RStoreReadStream stream;
1039 stream.OpenLC(aStore,aStreamId);
1040 charsPasted=PasteFromStreamL(stream,aPos);
1041 CleanupStack::PopAndDestroy();
1043 PasteComponentsL(aStore,aPos);
1044 SetHasChanged(ETrue);
1052 void CPlainText::PasteComponentsL(const CStreamStore& aStore,TInt aPos)
1053 // Copy/Paste 2' RestoreComponentsL() - only if a field set is present.
1056 if (FieldSetPresent())
1057 iFieldSet->PasteComponentsL(aStore,aPos);
1061 // Paste everything in the stream.
1062 TInt CPlainText::PasteFromStreamL(RReadStream& aStream,TInt aPos)
1064 TInt chars_read = 0;
1065 TInt error = KErrNone;
1067 TRAP(error, chars_read=CPlainText::DoPasteFromStreamL(aStream, aPos));
1069 UpdatePageTable(aPos,chars_read);
1072 If there was an exception delete any inserted text and propagate the exception.
1073 Not deleting the text would cause the size of the text to be inconsistent with the size
1074 implied elsewhere, such as in the formatting information stored in CRichText objects.
1076 if (error != KErrNone)
1078 DoPtDelete(aPos,chars_read);
1079 OstTrace1( TRACE_FATAL, DUP1_CPLAINTEXT_PASTEFROMSTREAML, "Leave code=%d", error );
1088 TInt CPlainText::DoPasteFromStreamL(RReadStream& aStream, TInt aPos)
1090 TInt chars_read = 0;
1092 CBufSeg* buffer = CBufSeg::NewL(512);
1093 CleanupStack::PushL(buffer);
1094 TInt length = aStream.ReadInt32L();
1095 ::InternalizeTextL(aStream,*buffer,length);
1098 Insert the text bit by bit so that memory consumed by the CPlainText object is freed from the buffer;
1099 this is important if pasting huge amounts of text.
1101 while (buffer->Size() > 0)
1103 TPtr8 p8 = buffer->Ptr(0);
1104 TInt bytes = p8.Length();
1105 TInt chars = bytes / sizeof(TText);
1106 TPtrC p((TText*)p8.Ptr(),chars); // platform dependency in the Unicode build; relies on little-endianness
1107 PtInsertL(aPos + chars_read,p);
1108 buffer->Delete(0,bytes);
1109 chars_read += chars;
1112 CleanupStack::PopAndDestroy(); // buffer
1115 // If there's a field set, internalize it.
1116 if (aStream.ReadUint8L() != 0) // next byte is non-zero if there's a field set
1118 if (!FieldSetPresent())
1119 CreateFieldSetL(DocumentLength());
1120 iFieldSet->PasteFromStreamL(aStream,aPos,chars_read);
1127 void CPlainText::InsertEodL()
1128 /** Inserts the end-of-document character upon document construction.*/
1130 // ASSERT: The plain text component is empty.
1131 if (DocumentLength()!=-1)
1133 OstTrace0( TRACE_DUMP, DUP1_CPLAINTEXT_INSERTEODL, "ECorruptTextStore" );
1135 __ASSERT_DEBUG(DocumentLength()==-1,Panic(ECorruptTextStore));
1137 content.Append(EParagraphDelimiter);
1145 EXPORT_C void CPlainText::Reset()
1146 /** Deletes all text content, formatting and fields from the document, leaving
1147 the single paragraph delimiter which terminates the text object. */
1149 // Resets document contents to a single end-of-document character, with no other content.
1150 // (No reset occurs if the component is already in the reset state. Avoids an assertion
1151 // failure in the delete method, where length to delete !> 0).
1154 TInt content=DocumentLength();
1156 DoPtDelete(0,content);
1157 if (FieldSetPresent())
1159 SetHasChanged(ETrue);
1167 EXPORT_C TInt CPlainText::DocumentLength()const
1168 /** Gets the the number of characters in the text object.
1170 Note: the count includes all non-printing characters but excludes the end
1171 of text paragraph delimiter, so that the smallest possible return value is
1174 @return The number of characters in the text object. */
1175 {return ((iByteStore->Size()/sizeof(TText))-1);}
1178 EXPORT_C void CPlainText::InsertL(TInt aInsertPos,const TChar& aChar)
1179 /** Inserts either a single character or a descriptor into the text object
1180 at a specified document position.
1182 Updates the page table.
1184 @param aPos The document position at which to insert the character/descriptor.
1185 Must be valid, or a panic occurs.
1186 @param aChar The character to insert.
1187 @param aBuf The descriptor to insert. */
1190 if (aInsertPos<0 || aInsertPos>DocumentLength())
1192 OstTrace0( TRACE_FATAL, CPLAINTEXT_INSERTL, "ECharPosBeyondDocument" );
1194 __ASSERT_ALWAYS(aInsertPos>=0 && aInsertPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1196 TInt contentLength = 1;
1197 if (aChar < 0x10000)
1200 content.Append(aChar);
1201 DoPtInsertL(aInsertPos,content);
1205 TText16 high = TChar::GetHighSurrogate(aChar);
1206 TText16 low = TChar::GetLowSurrogate(aChar);
1207 RDebug::Print(_L("CPlainText::InsertL(%d), %X expand to %X %X."), aInsertPos, TUint(aChar), high, low);
1211 content.Append(high);
1212 content.Append(low);
1213 DoPtInsertL(aInsertPos,content);
1215 if (FieldSetPresent())
1216 iFieldSet->NotifyInsertion(aInsertPos,contentLength);
1218 SetHasChanged(ETrue);
1224 EXPORT_C void CPlainText::InsertL(TInt aPos,const TDesC& aBuf)
1225 /** Inserts the contents of aBuf into the document at position aPos.*/
1227 PtInsertL(aPos,aBuf);
1228 SetHasChanged(ETrue);
1232 EXPORT_C void CPlainText::PtInsertL(TInt aPos,const TDesC& aBuf)
1233 /** Inserts the contents a aBuf into the document at position aInsertPos.
1234 Maintain field set.*/
1237 if (aPos<0 || aPos>DocumentLength())
1239 OstTrace0( TRACE_FATAL, CPLAINTEXT_PTINSERTL, "ECharPosBeyondDocument" );
1241 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1243 DoPtInsertL(aPos,aBuf);
1244 if (FieldSetPresent())
1245 iFieldSet->NotifyInsertion(aPos,aBuf.Length());
1251 EXPORT_C void CPlainText::DoPtInsertL(TInt aPos,const TDesC& aBuf)
1252 /** Inserts the contents a aBuf into the document at position aInsertPos.
1253 Maintain field set.*/
1255 TPtrC8 buf((TUint8*)aBuf.Ptr(),aBuf.Size());
1256 iByteStore->InsertL(aPos*sizeof(TText),buf);
1257 UpdatePageTable(aPos,aBuf.Length());
1261 void CPlainText::InsertL(TInt aPos,const CPlainText* aText)
1262 /** Insert the specified plain text object at the specified character position.
1263 (Called by CRichText::Insert()*/
1265 TInt lengthRemaining=aText->DocumentLength();
1269 TPtrC view=aText->Read(readPos);
1270 TInt consumed=view.Length();
1271 if (consumed>lengthRemaining)
1272 consumed=lengthRemaining;
1274 lengthRemaining-=consumed;
1275 if (lengthRemaining==0)
1283 EXPORT_C TBool CPlainText::DeleteL(TInt aPos,TInt aLength)
1284 /** Deletes one or more characters beginning at, and including, the character at
1285 a specified document position. Updates the page table. Any fields wholly contained
1286 in the range of characters to delete are removed from the field set.
1288 @param aPos The document position from which to begin deleting. Must be valid
1290 @param aLength The number of characters to delete. Must be positive or a panic
1291 occurs. The sum of aPos and aLength must be less than the document length,
1293 @return Indicates whether two paragraphs have been merged together as a result
1294 of the delete, indicating that the resulting paragraph must be reformatted.
1295 Has no meaning for plain text, so always EFalse. */
1297 return Delete(aPos,aLength);
1301 // A non-virtual non-leaving delete function for use inside ETEXT.
1302 TBool CPlainText::Delete(TInt aPos,TInt aLength)
1306 TBool ret = DoPtDelete(aPos,aLength);
1307 if (FieldSetPresent())
1308 iFieldSet->NotifyDeletion(aPos,aLength);
1309 SetHasChanged(ETrue);
1316 EXPORT_C TBool CPlainText::DoPtDelete(TInt aPos,TInt aLength)
1317 /** Deletes aLength number of characters from the document,
1318 commencing at, and including, position aPos.
1319 The return value indicates if 2 paragraphs have been merged together
1320 as a result of the delete, indicating that the resulting paragraph
1321 must be reformatted.
1322 In global text, this clearly has no reasonable meaning, so always returns
1323 EFalse, so no reformatting occurs.*/
1325 TInt documentLength=DocumentLength()+1;
1326 if (aPos<0 || aPos>documentLength)
1328 OstTrace0( TRACE_FATAL, CPLAINTEXT_DOPTDELETE, "ECharPosBeyondDocument" );
1330 __ASSERT_ALWAYS(aPos>=0 && aPos<=documentLength,Panic(ECharPosBeyondDocument));
1333 OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_DOPTDELETE, "EDeleteNegativeLength" );
1335 __ASSERT_ALWAYS(aLength>=0,Panic(EDeleteNegativeLength));
1336 if (aPos+aLength>documentLength)
1338 OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_DOPTDELETE, "ECharPosBeyondDocument" );
1340 __ASSERT_ALWAYS(aPos+aLength<=documentLength,Panic(ECharPosBeyondDocument));
1342 iByteStore->Delete(aPos*sizeof(TText),aLength*sizeof(TText));
1343 UpdatePageTable(aPos,-aLength);
1350 EXPORT_C TInt CPlainText::ImportTextFileL(TInt aPos,const TDes& aFileName,TTextOrganisation aTextOrganisation)
1351 /** Imports a plain text file into this text object.
1353 Translates non-printing characters in the source text file into Symbian OS
1354 special characters, for instance tabs are converted into
1355 CEditableText::ETabCharacters, and form feeds into CEditableText::EPageBreaks.
1356 Line feeds in the source text file are translated according to the
1357 aTextOrganisation argument.
1359 The function leaves if there is any problem in opening the file.
1361 @param aPos Document position at which to insert the text. Must be a valid
1362 position, or a panic occurs.
1363 @param aFileName The name of the text file to import.
1364 @param aTextOrganisation If EOrganiseByLine, a single line feed or a line feed
1365 and carriage return is converted into a space character. A line feed which
1366 is followed by another line feed is converted into a paragraph delimiter.
1367 If EOrganiseByParagraph, all line feeds are converted into paragraph delimiters.
1369 @return The number of characters imported. */
1371 TInt chars_inserted = 0;
1373 TInt error = file_session.Connect();
1374 if (error == KErrNone)
1377 error = file.Open(file_session,aFileName,EFileStream | EFileRead | EFileShareReadersOnly);
1378 if (error == KErrNone)
1380 RFileReadStream input_stream(file);
1381 TRAP(error,ImportTextL(aPos,input_stream,aTextOrganisation,KMaxTInt,KMaxTInt,&chars_inserted));
1382 input_stream.Close();
1385 file_session.Close();
1389 OstTrace1( TRACE_FATAL, DUP1_CPLAINTEXT_IMPORTTEXTFILEL, "Leave code=%x", error );
1391 User::LeaveIfError(error);
1392 return chars_inserted;
1397 EXPORT_C void CPlainText::ExportAsTextL(const TDes& aFileName,TTextOrganisation aTextOrganisation,TInt aLineWrap)const
1398 /** Writes the contents of the plain text object to a text file.
1400 The filename is given by aFileName. Any existing file with that name is
1401 replaced. A wrap width can be specified. This is only used when exporting
1402 by line (aTextOrganisation is EOrganiseByLine).
1404 The function leaves if there is any problem in creating or replacing the file.
1406 @param aFileName The name of the file to export the text to. If a file with
1407 this name already exists, it is replaced. Otherwise, a new file is created.
1408 @param aTextOrganisation Defines how to translate line delimiters. If
1409 EOrganiseByLine, lines wrap at the wrap width, as specified in aMaxLineLength.
1410 If EOrganiseByParagraph, lines do not wrap and paragraph delimiters are
1411 converted into CR/LF pairs.
1412 @param aMaxLineLength The maximum number of characters in each line, (only
1413 relevant if the text organisation is EOrganiseByLine). */
1415 if (aTextOrganisation != EOrganiseByParagraph && aLineWrap <= 0)
1417 OstTrace0( TRACE_DUMP, CPLAINTEXT_EXPORTASTEXTL, "EExportLineWrapInvalid" );
1419 __ASSERT_DEBUG(aTextOrganisation == EOrganiseByParagraph || aLineWrap > 0,Panic(EExportLineWrapInvalid));
1421 TInt error = file_session.Connect();
1422 if (error == KErrNone)
1425 error = file.Replace(file_session,aFileName,EFileStream | EFileWrite | EFileShareExclusive);
1426 if (error == KErrNone)
1428 RFileWriteStream output_stream(file);
1429 TRAP(error,output_stream.WriteUint16L(EByteOrderMark));
1430 if (error == KErrNone)
1431 TRAP(error,ExportTextL(0,output_stream,aTextOrganisation,KMaxTInt,DocumentLength(),aLineWrap));
1432 output_stream.Close();
1435 file_session.Close();
1439 OstTrace1( TRACE_FATAL, DUP2_CPLAINTEXT_EXPORTASTEXTL, "Leave code=%x", error );
1441 User::LeaveIfError(error);
1446 EXPORT_C void CPlainText::ImportTextL(TInt aPos,RReadStream& aInput,TTextOrganisation aTextOrganisation,
1447 TInt aMaxOutputChars,TInt aMaxInputChars,
1448 TInt* aOutputChars,TInt* aInputChars)
1449 /** Imports plain text from a stream into this text object.
1451 Translates line feeds in the source text according to the
1452 aTextOrganisation argument.
1454 @param aPos Document position at which to insert the text. Must be a valid
1455 position, or a panic occurs.
1456 @param aInput Stream from which to read the text.
1457 @param aTextOrganisation If EOrganiseByLine, a single line feed
1458 or a line feed and carriage return is converted into a space character. A
1459 line feed which is followed by another line feed is converted into a paragraph
1460 delimiter. If EOrganiseByParagraph, all line feeds are converted
1461 into paragraph delimiters.
1462 @param aMaxOutputChars The maximum number of characters to write to the text
1464 @param aMaxInputChars The maximum number of characters to read from the stream.
1465 @param aOutputChars On return, the number of characters written to the text
1467 @param aInputChars On return, the number of characters read from the stream. */
1469 TImportExportParam param;
1470 param.iOrganisation = aTextOrganisation;
1471 param.iMaxOutputChars = aMaxOutputChars;
1472 param.iMaxInputChars = aMaxInputChars;
1473 TImportExportResult result;
1474 ImportTextL(aPos,aInput,param,result);
1476 *aOutputChars = result.iOutputChars;
1478 *aInputChars = result.iInputChars;
1482 EXPORT_C void CPlainText::ExportTextL(TInt aPos,RWriteStream& aOutput,TTextOrganisation aTextOrganisation,
1483 TInt aMaxOutputChars,TInt aMaxInputChars,TInt aMaxLineLength,
1484 TInt* aOutputChars,TInt* aInputChars) const
1485 /** Writes plain text to a stream, optionally converting it from Unicode
1486 into a foreign encoding.
1489 @param aPos A document position in the source plain text object from which
1490 to start reading the text to export.
1491 @param aOutput On return, the write stream to which the text is written.
1492 @param aParam Export parameters, including an optional foreign encoding to
1493 convert the Unicode text into, a file server connection, (this is needed for
1494 the character conversion - if not specified, one will be created), a line
1495 wrap width, and the maximum number of characters to export.
1496 @param aResult On return, contains the number of characters read and written. */
1498 TImportExportParam param;
1499 param.iOrganisation = aTextOrganisation;
1500 param.iMaxOutputChars = aMaxOutputChars;
1501 param.iMaxInputChars = aMaxInputChars;
1502 param.iMaxLineLength = aMaxLineLength;
1503 TImportExportResult result;
1504 ExportTextL(aPos,aOutput,param,result);
1506 *aOutputChars = result.iOutputChars;
1508 *aInputChars = result.iInputChars;
1511 EXPORT_C void CPlainText::ImportTextL(TInt aPos,RReadStream& aInput,
1512 const TImportExportParam& aParam,TImportExportResult& aResult)
1513 /** Imports plain text from a stream into this text object, optionally
1514 converting it from a foreign encoding into Unicode.
1516 @param aPos Document position at which to insert the text. Must be a valid
1517 position, or a panic occurs.
1518 @param aInput Stream from which to read the text.
1519 @param aParam Import parameters, including the foreign encoding to convert
1520 from, whether to guess the foreign encoding and the maximum number of characters
1522 @param aResult On return, contains the number of characters read and written
1523 and the foreign encoding used by the imported text.
1524 @see CPlainText::TImportExportParam
1525 @see CPlainText::TImportExportResult */
1527 CBufSeg* buffer = CBufSeg::NewL(512);
1528 CleanupStack::PushL(buffer);
1529 RBufWriteStream output(*buffer,0);
1530 TImportExportParam param = aParam;
1531 param.iOutputInternal = TRUE; // force output to internal format
1532 TPlainTextReader::TranslateL(param,aResult,output,aInput);
1534 TInt chars_inserted = 0;
1535 while (buffer->Size() > 0)
1537 TPtr8 p8 = buffer->Ptr(0);
1538 TInt bytes = p8.Length();
1539 TInt chars = bytes / sizeof(TText);
1540 TPtrC p((TText*)p8.Ptr(),chars);
1542 Insert text using the virtual function InsertL to allow derived classes
1543 like CRichText to update their attributes.
1545 InsertL(aPos + chars_inserted,p);
1546 buffer->Delete(0,bytes);
1547 chars_inserted += chars;
1550 CleanupStack::PopAndDestroy(); // buffer;
1553 EXPORT_C void CPlainText::ExportTextL(TInt aPos,RWriteStream& aOutput,
1554 const TImportExportParam& aParam,TImportExportResult& aResult) const
1555 /** Writes plain text to a stream, optionally converting it from Unicode into a
1558 @param aPos A document position in the source plain text object from which
1559 to start reading the text to export.
1560 @param aOutput On return, the write stream to which the text is written.
1561 @param aParam Export parameters, including an optional foreign encoding to
1562 convert the Unicode text into, a file server connection, (this is needed for
1563 the character conversion if not specified, one will be created), a line
1564 wrap width, and the maximum number of characters to export.
1565 @param aResult On return, contains the number of characters read and written.
1566 @see CPlainText::TImportExportParam
1567 @see CPlainText::TImportExportResult */
1569 TImportExportParam param = aParam;
1570 param.iInputInternal = TRUE; // force input to internal format
1571 param.iMaxInputChars = Min(param.iMaxInputChars,DocumentLength() - aPos); // ensure final paragraph delimiter is not exported
1572 RBufReadStream input(*iByteStore,aPos);
1573 TPlainTextWriter::TranslateL(param,aResult,aOutput,input);
1577 EXPORT_C TPtrC CPlainText::Read(TInt aStartPos)const
1578 /** Gets a read-only view of a portion of the text object.
1580 The extent of the view is the range of characters starting at aStartPos and ending at whichever of the
1581 following document positions is reached first:
1583 - The end of the document.
1584 - The end of the segment if using segmented storage (the storage type is specified in the NewL().
1586 Therefore, when using a segmented buffer to store the document, the length of the resultant view may be
1587 less than the requested length. In this case multiple calls to Read() may be necessary.
1589 @param aStartPos The document position at which to begin reading. Must be valid or a panic occurs.
1590 @return Constant pointer to a section of the text object.
1594 if (aStartPos<0 || aStartPos>DocumentLength())
1596 OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_READ, "ECharPosBeyondDocument" );
1598 __ASSERT_ALWAYS(aStartPos>=0 && aStartPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1600 TPtr8 buf=iByteStore->Ptr(aStartPos*sizeof(TText));
1601 return TPtrC((TText*)buf.Ptr(),buf.Length()/sizeof(TText));
1605 EXPORT_C TPtrC CPlainText::Read(TInt aStartPos,TInt aLength)const
1606 /** Gets a read-only view of a portion of the text object.
1608 The extent of the view is the range of characters starting at aStartPos and ending at whichever of the
1609 following document positions is reached first:
1611 - The end of the document.
1612 - The end of the segment if using segmented storage (the storage type is specified in the NewL().
1613 - The sum of aStartPos and (aLength-1).
1615 Therefore, when using a segmented buffer to store the document, the length of the resultant view may be
1616 less than the requested length. In this case multiple calls to Read() may be necessary.
1618 @param aStartPos The document position at which to begin reading. Must be valid or a panic occurs.
1619 @param aLength The number of characters to read, inclusive of the character at position aStartPos.
1620 @return Constant pointer to a section of the text object.
1624 if (aStartPos<0 || aStartPos>DocumentLength())
1626 OstTrace0( TRACE_FATAL, CPLAINTEXT_READ, "ECharPosBeyondDocument" );
1628 __ASSERT_ALWAYS(aStartPos>=0 && aStartPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1630 TPtr8 buf=iByteStore->Ptr(aStartPos*sizeof(TText));
1631 TInt length=Min(aLength,((TInt)buf.Length()/sizeof(TText)));
1632 return TPtrC((TText*)buf.Ptr(),length);
1636 EXPORT_C void CPlainText::Extract(TDes& aBuf,TInt aPos)const
1637 /** Copies the contents of the text object into a descriptor overloaded
1638 function. The function copies all characters from and including the document
1639 position specified, to the end of the document or the end of the range
1640 of characters, if specified.
1642 The buffer's maximum length must be greater than or equal to the number
1643 of characters to extract, or a panic occurs.
1645 @param aBuf A buffer; on return contains the extracted text.
1646 @param aPos The document position from which to copy. Must be valid or a
1650 TInt documentLength=DocumentLength();
1651 if (aPos<0 || aPos>documentLength)
1653 OstTrace0( TRACE_FATAL, CPLAINTEXT_EXTRACT, "ECharPosBeyondDocument" );
1655 __ASSERT_ALWAYS(aPos>=0 && aPos<=documentLength,Panic(ECharPosBeyondDocument));
1656 if (aBuf.MaxLength()<documentLength - aPos)
1658 OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_EXTRACT, "EExtractBufferTooSmall" );
1660 __ASSERT_ALWAYS(aBuf.MaxLength()>=documentLength - aPos,Panic(EExtractBufferTooSmall));
1662 DoExtract(aBuf,aPos,documentLength-aPos);
1666 EXPORT_C void CPlainText::Extract(TDes& aBuf,TInt aPos,TInt aLength)const
1667 /** Copies the contents of the text object into a descriptor-overloaded function.
1668 The function copies all characters from and including the document position
1669 specified, to the end of the document or the end of the range of characters,
1672 The buffer's maximum length must be greater than or equal to aLength, or a
1675 @param aBuf A buffer; on return contains the extracted text.
1676 @param aPos The document position from which to copy. Must be valid or a panic
1678 @param aLength The number of characters to copy. */
1681 if (aPos<0 || aPos>DocumentLength())
1683 OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_EXTRACT, "ECharPosBeyondDocument" );
1685 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1686 if (aBuf.MaxLength()<aLength)
1688 OstTrace0( TRACE_FATAL, DUP3_CPLAINTEXT_EXTRACT, "EExtractBufferTooSmall" );
1690 __ASSERT_ALWAYS(aBuf.MaxLength()>=aLength,Panic(EExtractBufferTooSmall));
1692 DoExtract(aBuf,aPos,aLength);
1695 // Extract text, optionally discarding some characters such as control characters and soft hyphens or
1696 // inline text, depending on the flag.
1697 EXPORT_C void CPlainText::ExtractSelectively(TDes& aBuf,TInt aPos,TInt aLength,TUint aFlags)
1699 if (aPos < 0 || aPos > DocumentLength())
1701 OstTrace0( TRACE_FATAL, CPLAINTEXT_EXTRACTSELECTIVELY, "ECharPosBeyondDocument" );
1703 __ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(),Panic(ECharPosBeyondDocument));
1704 DoExtract(aBuf,aPos,aLength,aFlags);
1708 void CPlainText::DoExtract(TDes& aBuf,TInt aPos,TInt aLength,TUint aFlags) const
1709 /** Copies aLength characters, commencing at aPos, into the specified
1710 buffer aBuf. If aLength is greater than the amount of data left,
1711 then all remaining characters are written.*/
1714 TInt remainingLength=Min(aLength,DocumentLength()-aPos);
1718 TPtrC textRead=Read(startPos);
1719 TInt lengthRead=textRead.Length();
1720 if (lengthRead>remainingLength)
1721 lengthRead=remainingLength;
1723 // Remove specific characters
1724 if (aFlags & EExtractVisible)
1726 const TText* p = textRead.Ptr();
1727 const TText* q = p + lengthRead;
1731 if (c == EParagraphDelimiter || c == ELineBreak || c == ETabCharacter)
1733 else if (c != EPotentialHyphen && c.IsPrint())
1738 // Remove inline text from the specified section
1739 else if (aFlags & EExcludeInlineEditedText)
1741 const TInt inlineTextPos = GetPositionOfInlineTextInDocument();
1742 const TText* p = textRead.Ptr();
1744 if (inlineTextPos != KErrNotFound)
1746 for (TInt i=aPos; i<(aPos+lengthRead); i++)
1748 if (!((i >= inlineTextPos) && (i <= (inlineTextPos + GetLengthOfInlineText() - 1))))
1758 aBuf.Append(textRead.Ptr(),lengthRead);
1761 aBuf.Append(textRead.Ptr(),lengthRead);
1762 remainingLength-=lengthRead;
1763 if (remainingLength==0)
1765 startPos+=lengthRead;
1771 EXPORT_C void CPlainText::SetPageTable(TPageTable* aPageTable)
1772 /** Links the text object to a page table. A page table is an array of integers;
1773 each integer represents the number of characters on a page. It is required
1774 for pagination. The page table is updated when changes are made to the document,
1775 e.g. after pasting from the clipboard, inserting, importing or deleting text.
1777 The text object does not take ownership of the page table specified.
1779 @param aPageTable The page table to be referenced by the text object. */
1781 iPageTable=aPageTable;
1782 SetHasChanged(ETrue);
1786 void CPlainText::UpdatePageTable(TInt aPos,TInt aLength)
1787 // Adds aLength number of characters to the page in the page table
1788 // specified by the character position aPos.
1789 // Called from i) Insert ii) Delete iii) Paste from clipboard iv) Text file import.
1793 if (aPos<0 || aPos>DocumentLength()+1)
1795 OstTrace0( TRACE_FATAL, CPLAINTEXT_UPDATEPAGETABLE, "ECharPosBeyondDocument" );
1797 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength()+1,Panic(ECharPosBeyondDocument));
1800 (*iPageTable)[PageContainingPos(aPos)]+=aLength;
1804 EXPORT_C TInt CPlainText::PageContainingPos(TInt aPos)const
1805 /** Gets the number of the page which contains the specified document position.
1806 If no page table has been set up, the function returns a value of zero. Use
1807 SetPageTable() to set up the page table.
1809 @param aPos A document position. Must be valid or a panic occurs.
1810 @return The page number containing document position aPos. */
1813 if (aPos<0 || aPos>DocumentLength())
1815 OstTrace0( TRACE_FATAL, CPLAINTEXT_PAGECONTAININGPOS, "ECharPosBeyondDocument" );
1817 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1819 if (!iPageTable || (iPageTable->Count()<1))
1824 TInt charCount=(*iPageTable)[pageOffset];
1825 while (charCount<=aPos)
1828 charCount+=(*iPageTable)[pageOffset];
1834 EXPORT_C TEtextComponentInfo CPlainText::ComponentInfo()const
1835 /** Gets information about the number of components contained in the text object.
1836 For plain text, only the field count has relevance.
1838 @return Contains the field count. */
1839 {return TEtextComponentInfo(FieldCount(),0,0);}
1843 EXPORT_C TInt CPlainText::FieldCount() const
1844 /** Gets a count of the number of fields in the text object's field set.
1846 @return The number of fields in the field set. */
1850 return (FieldSetPresent())
1851 ? iFieldSet->FieldCount()
1855 EXPORT_C void CPlainText::InsertFieldL(TInt aPos,CTextField* aField,TUid aFieldType)
1856 /** Inserts a field into the text object at a specified document position. creating a zero-length field record.
1857 Maintain the field set.
1859 Note: After insertion, the field should be evaluated in order to make its contents
1860 visible; use UpdateFieldL().
1862 @param aPos The document position at which to insert the field. Must be valid,
1864 @param aField The field to insert, created by NewTextFieldL(). Must not be
1865 NULL, or a panic occurs.
1866 @param aFieldType Identifies the type of field to insert. For the built in
1867 field types, see the UID values defined in flddef.h. */
1870 if (aPos<0 || aPos>DocumentLength())
1872 OstTrace0( TRACE_FATAL, CPLAINTEXT_INSERTFIELDL, "ECharPosBeyondDocument" );
1874 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1877 OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_INSERTFIELDL, "ENoTextField" );
1879 __ASSERT_ALWAYS(aField,Panic(ENoTextField));
1881 if (!FieldSetPresent())
1882 CreateFieldSetL(DocumentLength());
1883 iFieldSet->InsertFieldL(aPos,aField,aFieldType);
1884 SetHasChanged(ETrue);
1889 EXPORT_C void CPlainText::UpdateFieldL(TInt aPos)
1890 /** Re-evaluates the field which covers the document position specified. Re-evaluating
1891 a field means calculating the field's new value, then inserting that value
1892 into the text object, replacing the previous value.
1896 fields have a maximum length of 20 characters.
1898 the first time a field is updated, the position specified should be the position
1899 at which the field was inserted.
1901 @param aPos A document position in the field to be updated. Must be a valid
1902 position, or a panic occurs. */
1905 if (aPos<0 || aPos>DocumentLength())
1907 OstTrace0( TRACE_FATAL, CPLAINTEXT_UPDATEFIELDL, "ECharPosBeyondDocument" );
1909 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1911 if (!FieldSetPresent())
1913 TFindFieldInfo info;
1914 if (iFieldSet->FindFields(info,aPos))
1915 {// a field exists at aPos, so update it.
1916 HBufC* valueBuf=HBufC::NewL(KMaxFieldBufferSize); // will hold the new value,max length 20
1918 Don't put valueBuf on the cleanup stack before calling NewFieldValueL because NewFieldValueL
1919 sometimes changes valueBuf, in which case it itself puts it on the cleanup stack.
1921 iFieldSet->NewFieldValueL(valueBuf,info.iFirstFieldPos); // get the new value
1922 CleanupStack::PushL(valueBuf);
1923 DoPtInsertL(info.iFirstFieldPos,*valueBuf); // insert the new text into the document
1924 DoPtDelete(info.iFirstFieldPos+valueBuf->Length(),info.iFirstFieldLen); // delete the old text of the field
1925 iFieldSet->NotifyFieldUpdate(aPos,valueBuf->Length()); // inform the field set
1926 CleanupStack::PopAndDestroy();
1928 SetHasChanged(ETrue);
1935 EXPORT_C void CPlainText::UpdateAllFieldsL()
1936 /** Re-evaluates all of the fields in the text object. Re-evaluating a field means
1937 calculating the field's new value, then inserting that value into the text
1938 object, replacing the previous value.
1940 Note: Fields have a maximum length of 20 characters. */
1944 TInt numFields=FieldCount();
1946 TFindFieldInfo info;
1947 for (TInt item=0;item<numFields;item++)
1949 // find the next field and go to its beginning
1950 iFieldSet->FindFields(info,pos,DocumentLength()-pos);
1951 pos=info.iFirstFieldPos;
1953 // the field has changed, find out its new length and go to the end of it
1954 iFieldSet->FindFields(info,pos);
1955 pos+=info.iFirstFieldLen;
1963 EXPORT_C TBool CPlainText::RemoveField(TInt aPos)
1964 /** Removes the field covering the document position specified from the field set,
1965 and deletes all text content associated with the field from the text object.
1967 @param aPos A document position within the field to be deleted. Must be a
1968 valid position or a panic occurs.
1969 @return ETrue if a field was located at aPos. EFalse if no field was located
1973 if (aPos<0 || aPos>DocumentLength())
1975 OstTrace0( TRACE_FATAL, CPLAINTEXT_REMOVEFIELD, "ECharPosBeyondDocument" );
1977 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
1979 TBool fieldRemoved=EFalse;
1981 if (FieldSetPresent())
1983 TFindFieldInfo info;
1984 if (iFieldSet->FindFields(info,aPos))
1985 {// remove the field's record & associated text
1986 Delete(info.iFirstFieldPos,info.iFirstFieldLen);
1987 iFieldSet->RemoveField(aPos);
1992 SetHasChanged(ETrue);
1995 return fieldRemoved;
2000 EXPORT_C TBool CPlainText::FindFields(TInt aPos) const
2004 if (FieldSetPresent())
2005 return iFieldSet->FindFields(aPos);
2011 EXPORT_C TBool CPlainText::FindFields(TFindFieldInfo& aInfo,TInt aPos,TInt aRange)const
2012 /** Tests whether a document position is located within a field overloaded
2015 The second overload extracts information about all fields located
2016 within a range of characters.
2018 @param aPos The document position. Must be valid or a panic occurs.
2019 @param aInfo On return, contains the number of fields fully or partially
2020 within the range of characters specified and the start position and length
2021 of the first field found.
2022 @param aRange The number of characters to search, beginning at aPos. Must
2023 not be negative, or a panic occurs. The sum of aPos and aRange must be less
2024 than the document length, or a panic occurs. The default range is zero. In
2025 this case the function just returns information about the single field located
2026 at the position specified.
2027 @return ETrue if aPos is located within a field, or if there were one or more
2028 fields found in the range. EFalse if not. */
2032 if (FieldSetPresent())
2033 return iFieldSet->FindFields(aInfo,aPos,aRange);
2038 EXPORT_C const CTextField* CPlainText::TextField(TInt aPos)const
2039 /** Returns a pointer to the field located at the specified document position.
2041 @param aPos A document position within the field. Must be a valid document
2042 position, or a panic occurs.
2043 @return Pointer to the field which covers position aPos. NULL if there is no
2044 field at the specified position. */
2048 return (FieldSetPresent())
2049 ? iFieldSet->TextField(aPos)
2053 EXPORT_C TBool CPlainText::ConvertFieldToText(TInt aPos)
2054 /** Converts the field containing the specified document position into text, leaving
2055 its current value in the text object. Does not update the field beforehand.
2057 @param aPos A document position within the field to convert. Must be a valid
2058 position or a panic occurs.
2059 @return ETrue if there was a field located at aPos. EFalse if there was no
2060 field located at aPos. */
2063 if (aPos<0 || aPos>DocumentLength())
2065 OstTrace0( TRACE_FATAL, CPLAINTEXT_CONVERTFIELDTOTEXT, "ECharPosBeyondDocument" );
2067 __ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
2069 TBool fieldConverted=EFalse;
2071 if (FieldSetPresent())
2073 TFindFieldInfo info;
2074 iFieldSet->FindFields(info,aPos);
2075 if (iFieldSet->RemoveField(aPos))
2076 fieldConverted=ETrue;
2079 SetHasChanged(ETrue);
2082 return fieldConverted;
2086 EXPORT_C void CPlainText::ConvertAllFieldsToText()
2087 /** Removes all fields from the text object's field set, leaving their current
2088 text value in the text object. Does not update the fields beforehand. */
2092 TInt fieldCount=FieldCount();
2094 for (TInt item=0;item<fieldCount;item++)
2095 {// Convert each field in turn.
2096 TFindFieldInfo info;
2097 iFieldSet->FindFields(info,pos); // find the next field and go to its beginning
2098 pos=info.iFirstFieldPos;
2099 ConvertFieldToText(pos);
2105 EXPORT_C void CPlainText::SetFieldFactory(MTextFieldFactory* aFactory)
2106 /** Sets up a field factory. A field factory is an instance of a class deriving
2107 from MTextFieldFactory. It must implement a NewFieldL() function to create
2108 and return fields of the desired type. The field factory's NewFieldL() function
2109 is called by CPlainText::NewTextFieldL(). A field factory must be set up if
2110 you intend to add any fields to the text object.
2112 @param aFactory The field factory. */
2114 // Set the FieldSet's header factory handle (this points to the built-in factory by default)
2115 //+ This function has no way to report failure, which can happen if CreateFieldSetL leaves. It ought to leave.
2117 // Create field set if necessary & set the factory
2119 if (!FieldSetPresent())
2120 TRAP(error,CreateFieldSetL(DocumentLength()));
2122 iFieldSet->SetFieldFactory(aFactory);
2127 EXPORT_C CTextField* CPlainText::NewTextFieldL(TUid aFieldType)const
2128 /** Creates and returns a new field. Before calling this function, a field
2129 factory should have been set up, either by calling SetFieldFactory(), or by
2130 specifying one in the NewL(). The field factory's NewFieldL() function is
2131 called to create a field of the type specified in the argument. A NULL field
2132 is returned if no field factory has been set up.
2134 @param aFieldType Identifies the field type.
2135 @return Pointer to the new text field. NULL if no field factory has been set
2140 CTextField* field=NULL;
2141 if (FieldSetPresent())
2142 field=iFieldSet->NewFieldL(aFieldType);
2148 void CPlainText::CreateFieldSetL(TInt aDocumentLength)
2150 iFieldSet = CTextFieldSet::NewL(aDocumentLength);
2153 EXPORT_C const MTextFieldFactory* CPlainText::FieldFactory()const
2154 /** Gets a pointer to the field factory used by the text object. The field factory
2155 may be set up using SetFieldFactory(), or may be specified in the NewL().
2157 @return The field factory. NULL if no field factory has been set up. */
2158 {return (FieldSetPresent()) ? iFieldSet->FieldFactory() : NULL;}
2160 CEditableTextOptionalData::CEditableTextOptionalData()
2165 CEditableTextOptionalData::~CEditableTextOptionalData()
2167 delete iInlineEditData;