os/textandloc/textrendering/texthandling/stext/TXTETEXT.CPP
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/textandloc/textrendering/texthandling/stext/TXTETEXT.CPP	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,2168 @@
     1.4 +/*
     1.5 +* Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
     1.6 +* All rights reserved.
     1.7 +* This component and the accompanying materials are made available
     1.8 +* under the terms of "Eclipse Public License v1.0"
     1.9 +* which accompanies this distribution, and is available
    1.10 +* at the URL "http://www.eclipse.org/legal/epl-v10.html".
    1.11 +*
    1.12 +* Initial Contributors:
    1.13 +* Nokia Corporation - initial contribution.
    1.14 +*
    1.15 +* Contributors:
    1.16 +*
    1.17 +* Description: 
    1.18 +*
    1.19 +*/
    1.20 +
    1.21 +
    1.22 +#include <e32std.h>
    1.23 +#include <e32base.h>
    1.24 +
    1.25 +#include <s32std.h>
    1.26 +#include <s32strm.h>
    1.27 +#include <s32stor.h>
    1.28 +#include <s32mem.h>
    1.29 +#include <s32file.h>
    1.30 +#include <s32ucmp.h>
    1.31 +
    1.32 +#include <fepitfr.h>
    1.33 +
    1.34 +#include "FLDDEF.H"
    1.35 +#include "FLDINFO.H"
    1.36 +#include "FLDSET.H"
    1.37 +
    1.38 +#include "TXTETEXT.H"
    1.39 +#include "TXTRICH.H"
    1.40 +#include "TXTOPT.H"
    1.41 +#include "TXTFEP.H"
    1.42 +#include "TXTPLAIN.H"
    1.43 +#include "TXTSTD.H"
    1.44 +#include "TXTRTPFL.H"
    1.45 +#include "TXTCLIPBOARD.H"
    1.46 +
    1.47 +#include "OstTraceDefinitions.h"
    1.48 +#ifdef OST_TRACE_COMPILER_IN_USE
    1.49 +#include "TXTETEXTTraces.h"
    1.50 +#endif
    1.51 +
    1.52 +#ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
    1.53 +#include "TXTETEXT_INTERNAL.H"
    1.54 +#endif
    1.55 +
    1.56 +const TUint KFieldCountLimit = 255;
    1.57 +#define UNUSED_VAR(a) a = a
    1.58 +
    1.59 +// Write some or all of the text in a buffer to a stream, writing the length first if aWriteLength is true.
    1.60 +static void ExternalizeTextL(RWriteStream& aStream,const CBufBase& aText,TInt aPos,TInt aLength,TBool aWriteLength)
    1.61 +	{
    1.62 +	if (aWriteLength)
    1.63 +		aStream << TCardinality(aLength);
    1.64 +
    1.65 +	// Use the Standard Unicode Compression Scheme.
    1.66 +	RBufReadStream input_stream(aText,aPos * sizeof(TText));
    1.67 +	TMemoryStreamUnicodeSource source(input_stream);
    1.68 +	TUnicodeCompressor c;
    1.69 +	c.CompressL(aStream,source,KMaxTInt,aLength);
    1.70 +	input_stream.Close();
    1.71 +	}
    1.72 +
    1.73 +// Read text from a stream and write it to a buffer.
    1.74 +static void InternalizeTextL(RReadStream& aStream,CBufBase& aText,TInt aLength)
    1.75 +	{
    1.76 +	// Use the Standard Unicode Compression Scheme.
    1.77 +	RBufWriteStream output_stream(aText);
    1.78 +	TMemoryStreamUnicodeSink sink(output_stream);
    1.79 +	TUnicodeExpander e;
    1.80 +	e.ExpandL(sink,aStream,aLength);
    1.81 +	output_stream.CommitL();
    1.82 +	output_stream.Close();
    1.83 +	}
    1.84 +
    1.85 +// Read text from a stream and write it to a buffer; read the length first.
    1.86 +static void InternalizeTextL(RReadStream& aStream,CBufBase& aText)
    1.87 +	{
    1.88 +	TCardinality length;
    1.89 +	aStream >> length;
    1.90 +	InternalizeTextL(aStream,aText,length);
    1.91 +	}
    1.92 +/**
    1.93 +Returns the interface corresponding to the
    1.94 +specified UID if it exists, or 0 if not. Overridden
    1.95 +versions should base call rather than returning 0.
    1.96 +
    1.97 +@param aInterfaceId The UID indicating the interface to return
    1.98 +@param aInterface The interface corresponding to aInterfaceId
    1.99 +if it is supported, or 0 if it is not
   1.100 +*/
   1.101 +EXPORT_C void CEditableText::ExtendedInterface(TAny*& /*aInterface*/, TUid /*aInterfaceId*/) {}
   1.102 +
   1.103 +/**
   1.104 +Returns the interface corresponding to the
   1.105 +specified UID if it exists, or 0 if not. Overridden
   1.106 +versions should base call rather than returning 0.
   1.107 +
   1.108 +@param aInterfaceId The UID indicating the interface to return
   1.109 +@param aInterface The interface corresponding to aInterfaceId
   1.110 +if it is supported, or 0 if it is not
   1.111 +*/
   1.112 +EXPORT_C void CPlainText::ExtendedInterface(TAny*& /*aInterface*/, TUid /*aInterfaceId*/) {}
   1.113 +
   1.114 +/**
   1.115 + @internalAll
   1.116 + @released
   1.117 + */
   1.118 +EXPORT_C void CPlainText::Reserved_2() {}
   1.119 +
   1.120 +//////////////////////////////////
   1.121 +// CEditableText
   1.122 +//////////////////////////////////
   1.123 +
   1.124 +EXPORT_C CEditableText::~CEditableText()
   1.125 +	{
   1.126 +	delete iOptionalData;
   1.127 +	}
   1.128 +
   1.129 +
   1.130 +
   1.131 +EXPORT_C TInt CEditableText::ScanWords(TInt& /*aPos*/,TUint& /*aScanMask*/) const
   1.132 +/** Scans the text from a specified document position to a location 
   1.133 +determined by the flags specified in a bitmask. The function can scan 
   1.134 +forwards or backwards to the beginning or end of a word.
   1.135 +
   1.136 +@param aPos A valid document position from which to scan. On return, 
   1.137 +contains the new document position. 
   1.138 +@param aScanMask The scan mask to use. See the scanning enumeration defined 
   1.139 +in class CPlainText. 
   1.140 +@return The number of characters skipped to reach the new document position. 
   1.141 +Notes: If the scan passes the end of text delimiter, on return, aPos is set 
   1.142 +to EScanEndOfData  and the function's return value indicates the 
   1.143 +number of characters skipped in passing the end of text delimiter. */
   1.144 +	{
   1.145 +	return 0;
   1.146 +	}
   1.147 +
   1.148 +
   1.149 +EXPORT_C TInt CEditableText::ScanParas(TInt& /*aPos*/,TUint& /*aScanMask*/) const
   1.150 +/** Scans the text from a specified document position to a location determined 
   1.151 +by the flags specified in a bitmask. The function can scan forwards or backwards 
   1.152 +to the beginning or end of a paragraph.
   1.153 +
   1.154 +@param aPos A valid document position from which to scan. On return, contains 
   1.155 +the new document position. 
   1.156 +@param aScanMask The scan mask to use. See the scanning enumeration defined 
   1.157 +in class CPlainText. 
   1.158 +@return The number of characters skipped to reach the new document position. 
   1.159 +Notes: If the scan passes the end of text delimiter, on return, aPos is set 
   1.160 +to EScanEndOfData  and the function's return value indicates the 
   1.161 +number of characters skipped in passing the end of text delimiter. */
   1.162 +	{
   1.163 +	return 0;
   1.164 +	}
   1.165 +
   1.166 +
   1.167 + 
   1.168 +EXPORT_C void CEditableText::SetHasChanged(TBool aHasChanged)
   1.169 +/** Sets whether a change has occurred to the editable text object. This is called 
   1.170 +by functions which change the text object in some way.
   1.171 +
   1.172 +@param aHasChanged ETrue if a change has occurred to the text object. EFalse 
   1.173 +if no change has occurred. */
   1.174 +	{
   1.175 +	iHasChanged = aHasChanged;
   1.176 +	}
   1.177 +
   1.178 +
   1.179 +// Save the editable text type identifier.
   1.180 +void CEditableText::ExternalizeL(RWriteStream& aStream)const
   1.181 +	{
   1.182 +	aStream << KEditableTextUid;
   1.183 +	}
   1.184 +
   1.185 +
   1.186 +void CEditableText::InternalizeL(RReadStream& aStream)
   1.187 +// Read from the stream, expecting the editable text type identifier
   1.188 +//
   1.189 +	{
   1.190 +	TUid uid;
   1.191 +	aStream>> uid;
   1.192 +	if (uid!=KEditableTextUid)
   1.193 +		User::Leave(KErrCorrupt);
   1.194 +	}
   1.195 +
   1.196 +
   1.197 + 
   1.198 +
   1.199 +EXPORT_C TStreamId CEditableText::StoreL(CStreamStore& aStore)const
   1.200 +/** Stores the text and its components. The components (e.g. fields and pictures) 
   1.201 +are stored in separate streams within the stream store.
   1.202 +
   1.203 +@param aStore Stream store to which the text and text components are written. 
   1.204 +@return The ID of the stream store. */
   1.205 +	{
   1.206 +	CStoreMap* map=CStoreMap::NewLC(aStore);
   1.207 +	StoreComponentsL(aStore,*map);
   1.208 +//
   1.209 +	RStoreWriteStream stream(*map);
   1.210 +	TStreamId id=stream.CreateLC(aStore);
   1.211 +	stream<< *this;
   1.212 +	stream.CommitL();
   1.213 +//
   1.214 +	map->Reset();
   1.215 +	CleanupStack::PopAndDestroy(2);  // map,stream
   1.216 +	return id;
   1.217 +	}
   1.218 +
   1.219 +
   1.220 + 
   1.221 +EXPORT_C void CEditableText::RestoreL(const CStreamStore& aStore,TStreamId aStreamId)
   1.222 +/** Restores the text and its components from a stream store.
   1.223 +
   1.224 +@param aStore Stream store containing the text and its components. 
   1.225 +@param aStreamId The ID of the stream store in which the text was previously 
   1.226 +stored. */
   1.227 +	{
   1.228 +	// Load text and field components only.  (Pictures, if present, are deferred loaded).
   1.229 +	__ETEXT_WATCH(RESTORE)
   1.230 +
   1.231 +	RStoreReadStream stream;
   1.232 +	stream.OpenLC(aStore,aStreamId);
   1.233 +	//
   1.234 +	stream>> *this;
   1.235 +	CleanupStack::PopAndDestroy();  // stream
   1.236 +	RestoreComponentsL(aStore);
   1.237 +
   1.238 +	__ETEXT_WATCH_END(RESTORE)
   1.239 +	}
   1.240 +
   1.241 +TBool CEditableText::DeleteWithoutDestroyingFormatL(TInt aPos, TInt aLength)
   1.242 +/** Deletes a range of characters. For rich text the format of the deleted character 
   1.243 +at position aPos is preserved, so that any text subsequently inserted at aPos will have 
   1.244 +that format applied to it.
   1.245 +@param aPos The document position from which to begin deleting including aPos.
   1.246 +@param aLength The number of characters to delete. Must be positive or a panic 
   1.247 +occurs. The sum of aPos and aLength must be less than the document length, 
   1.248 +or a panic occurs.  
   1.249 +@return Indicates whether two paragraphs have been merged together as a result 
   1.250 +of the delete, indicating that the resulting paragraph must be reformatted. 
   1.251 +*/	
   1.252 +	{
   1.253 +	TAny* richTextInterface = NULL;	
   1.254 +	ExtendedInterface(richTextInterface, KUidRichText);
   1.255 +
   1.256 +	if(richTextInterface)
   1.257 +		{
   1.258 +		return REINTERPRET_CAST(CRichText*, richTextInterface)->DelSetInsertCharFormatL(aPos, aLength);
   1.259 +		}
   1.260 +	else
   1.261 +		{
   1.262 +		return DeleteL(aPos, aLength);
   1.263 +		}
   1.264 +	}
   1.265 +
   1.266 +EXPORT_C void CEditableText::StartFepInlineEditL(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument,const TDesC& aInitialInlineText,TInt aPositionOfInlineTextInDocument,TInt aNumberOfCharactersToHide,MFepInlineTextFormatRetriever& aInlineTextFormatRetriever)
   1.267 +/** @internalAll */	
   1.268 +	{
   1.269 +	if (aPositionOfInlineTextInDocument<0 || aNumberOfCharactersToHide<0 || aPositionOfInlineTextInDocument+aNumberOfCharactersToHide>DocumentLength())
   1.270 +	    {
   1.271 +	    OstTrace0( TRACE_DUMP, CEDITABLETEXT_STARTFEPINLINEEDITL, "ECharPosBeyondDocument" );
   1.272 +	    }
   1.273 +	__ASSERT_ALWAYS(aPositionOfInlineTextInDocument>=0 && aNumberOfCharactersToHide>=0 && aPositionOfInlineTextInDocument+aNumberOfCharactersToHide<=DocumentLength(),Panic(ECharPosBeyondDocument));
   1.274 +	if (InlineEditData()!=NULL)
   1.275 +	    {
   1.276 +	    OstTrace0( TRACE_DUMP, DUP1_CEDITABLETEXT_STARTFEPINLINEEDITL, "EAlreadyFepInlineEditing" );
   1.277 +	    }
   1.278 +	__ASSERT_ALWAYS(InlineEditData()==NULL,Panic(EAlreadyFepInlineEditing));
   1.279 +	aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
   1.280 +	aNumberOfCharactersSuccessfullyDeleted=0;
   1.281 +	aNumberOfCharactersSuccessfullyInserted=0;
   1.282 +	CInlineEditData* const inlineEditData=new(ELeave) CInlineEditData;
   1.283 +	CleanupStack::PushL(inlineEditData);
   1.284 +	HBufC* hiddenText=NULL;
   1.285 +	CleanupStack::PushL(hiddenText);
   1.286 +	if (aNumberOfCharactersToHide>0)
   1.287 +		{
   1.288 +		CleanupStack::Pop(); // hiddenText
   1.289 +		hiddenText=HBufC::NewLC(aNumberOfCharactersToHide);
   1.290 +		TPtr hiddenTextAsWritableDescriptor=hiddenText->Des();
   1.291 +		Extract(hiddenTextAsWritableDescriptor,aPositionOfInlineTextInDocument,aNumberOfCharactersToHide);
   1.292 +		aParagraphContainingStartPositionOfInlineTextHasChangedFormat=DeleteWithoutDestroyingFormatL(aPositionOfInlineTextInDocument,aNumberOfCharactersToHide);
   1.293 +		aNumberOfCharactersSuccessfullyDeleted=aNumberOfCharactersToHide;
   1.294 +		aPositionOfInsertionPointInDocument=aPositionOfInlineTextInDocument;
   1.295 +		}
   1.296 +	inlineEditData->iPositionOfInlineTextInDocument=aPositionOfInlineTextInDocument;
   1.297 +	inlineEditData->iLengthOfInlineText=aInitialInlineText.Length();
   1.298 +	inlineEditData->iInlineText=aInitialInlineText.AllocL();
   1.299 +	inlineEditData->iHiddenText=hiddenText;
   1.300 +	CleanupStack::Pop(); // hiddentext now owned by inlineEditData. 
   1.301 +	inlineEditData->iInlineTextFormatRetriever=&aInlineTextFormatRetriever;
   1.302 +	InsertL(aPositionOfInlineTextInDocument,aInitialInlineText);
   1.303 +	SetAndTransferOwnershipOfInlineEditDataL(inlineEditData);
   1.304 +	CleanupStack::Pop(); // inlineEditData
   1.305 +	aNumberOfCharactersSuccessfullyInserted=inlineEditData->iLengthOfInlineText;
   1.306 +	aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
   1.307 +	}
   1.308 +
   1.309 +
   1.310 +EXPORT_C void CEditableText::UpdateFepInlineTextL(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument,const TDesC& aNewInlineText)
   1.311 +/** @internalAll */	
   1.312 +    {
   1.313 +	CInlineEditData* const inlineEditData=InlineEditData();
   1.314 +	if (inlineEditData==NULL)
   1.315 +	    {
   1.316 +	    OstTrace0( TRACE_DUMP, CEDITABLETEXT_UPDATEFEPINLINETEXTL, "ENotCurrentlyFepInlineEditing" );
   1.317 +	    }
   1.318 +	__ASSERT_ALWAYS(inlineEditData!=NULL,Panic(ENotCurrentlyFepInlineEditing));
   1.319 +	aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
   1.320 +	aNumberOfCharactersSuccessfullyDeleted=0;
   1.321 +	aNumberOfCharactersSuccessfullyInserted=0;
   1.322 +	HBufC*& inlineText=inlineEditData->iInlineText;
   1.323 +	HBufC* oldInlineText=inlineText;
   1.324 +	if (oldInlineText!=NULL && inlineEditData->iLengthOfInlineText!=oldInlineText->Length())
   1.325 +	    {
   1.326 +	    OstTrace0( TRACE_DUMP, DUP1_CEDITABLETEXT_UPDATEFEPINLINETEXTL, "EDebug" );
   1.327 +	    }
   1.328 +	__ASSERT_DEBUG(oldInlineText==NULL || inlineEditData->iLengthOfInlineText==oldInlineText->Length(),Panic(EDebug));
   1.329 +	const TInt lengthOfNewInlineText=aNewInlineText.Length();
   1.330 +	if (oldInlineText!=NULL && *oldInlineText==aNewInlineText)
   1.331 +		{
   1.332 +		aNumberOfCharactersSuccessfullyDeleted=lengthOfNewInlineText;
   1.333 +		aNumberOfCharactersSuccessfullyInserted=lengthOfNewInlineText;
   1.334 +		}
   1.335 +	else
   1.336 +		{
   1.337 +		if (oldInlineText==NULL)
   1.338 +			{
   1.339 +			oldInlineText=HBufC::NewL(lengthOfNewInlineText);
   1.340 +			}
   1.341 +		else if (lengthOfNewInlineText>oldInlineText->Length())
   1.342 +			{
   1.343 +			oldInlineText=oldInlineText->ReAllocL(lengthOfNewInlineText);
   1.344 +			}
   1.345 +		CleanupStack::PushL(oldInlineText);
   1.346 +		inlineText=NULL; // sets inlineEditData->iLengthOfInlineText in case either the delete or the insert leaves
   1.347 +		const TInt positionOfInlineTextInDocument=inlineEditData->iPositionOfInlineTextInDocument;
   1.348 +		TInt& lengthOfInlineText=inlineEditData->iLengthOfInlineText;
   1.349 +		if (lengthOfInlineText>0)
   1.350 +			{
   1.351 +			aParagraphContainingStartPositionOfInlineTextHasChangedFormat=DeleteWithoutDestroyingFormatL(positionOfInlineTextInDocument,lengthOfInlineText);
   1.352 +			aNumberOfCharactersSuccessfullyDeleted=lengthOfInlineText;
   1.353 +			lengthOfInlineText=0; // sets inlineEditData->iLengthOfInlineText in case the insert leaves
   1.354 +			aPositionOfInsertionPointInDocument=inlineEditData->iPositionOfInlineTextInDocument;
   1.355 +			}
   1.356 +		InsertL(positionOfInlineTextInDocument,aNewInlineText);
   1.357 +		lengthOfInlineText=aNewInlineText.Length(); // sets inlineEditData->iLengthOfInlineText
   1.358 +		aNumberOfCharactersSuccessfullyInserted=lengthOfInlineText;
   1.359 +		inlineText=oldInlineText;
   1.360 +		CleanupStack::Pop(); // oldInlineText
   1.361 +		*oldInlineText=aNewInlineText;
   1.362 +		}
   1.363 +	aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
   1.364 +	}
   1.365 +
   1.366 +
   1.367 +EXPORT_C void CEditableText::CommitFepInlineEditL(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument)
   1.368 +/**
   1.369 + * @internalAll
   1.370 + */	
   1.371 +    {
   1.372 +	const CInlineEditData* const inlineEditData=InlineEditData();
   1.373 +	if (inlineEditData==NULL)
   1.374 +	    {
   1.375 +	    OstTrace0( TRACE_FATAL, DUP1_CEDITABLETEXT_COMMITFEPINLINEEDITL, "ENotCurrentlyFepInlineEditing" );
   1.376 +	    }
   1.377 +	__ASSERT_ALWAYS(inlineEditData!=NULL,Panic(ENotCurrentlyFepInlineEditing));
   1.378 +	if (inlineEditData->iInlineText!=NULL && inlineEditData->iLengthOfInlineText!=inlineEditData->iInlineText->Length())
   1.379 +	    {
   1.380 +	    OstTrace0( TRACE_DUMP, CEDITABLETEXT_COMMITFEPINLINEEDITL, "EDebug" );
   1.381 +	    }
   1.382 +	__ASSERT_DEBUG(inlineEditData->iInlineText==NULL || inlineEditData->iLengthOfInlineText==inlineEditData->iInlineText->Length(),Panic(EDebug));
   1.383 +	aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
   1.384 +	const TInt lengthOfInlineText=inlineEditData->iLengthOfInlineText;
   1.385 +	aNumberOfCharactersSuccessfullyDeleted=lengthOfInlineText;
   1.386 +	aNumberOfCharactersSuccessfullyInserted=lengthOfInlineText;
   1.387 +	aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
   1.388 +	DeleteInlineEditDataAndSetToNull();
   1.389 +	}
   1.390 +
   1.391 +
   1.392 +EXPORT_C void CEditableText::CancelFepInlineEdit(TBool& aParagraphContainingStartPositionOfInlineTextHasChangedFormat,TInt& aNumberOfCharactersSuccessfullyDeleted,TInt& aNumberOfCharactersSuccessfullyInserted,TInt& aPositionOfInsertionPointInDocument,TInt aNewPositionOfInsertionPointInDocument)
   1.393 +/**
   1.394 + * @internalAll
   1.395 + */	
   1.396 +    {
   1.397 +	aParagraphContainingStartPositionOfInlineTextHasChangedFormat=EFalse;
   1.398 +	aNumberOfCharactersSuccessfullyDeleted=0;
   1.399 +	aNumberOfCharactersSuccessfullyInserted=0;
   1.400 +	const CInlineEditData* inlineEditData=InlineEditData();
   1.401 +	if (inlineEditData!=NULL)
   1.402 +		{
   1.403 +		const TInt positionOfInlineTextInDocument=inlineEditData->iPositionOfInlineTextInDocument;
   1.404 +		const TInt lengthOfInlineText=inlineEditData->iLengthOfInlineText;
   1.405 +		if (inlineEditData->iInlineText!=NULL && lengthOfInlineText!=inlineEditData->iInlineText->Length())
   1.406 +		    {
   1.407 +		    OstTrace0( TRACE_DUMP, CEDITABLETEXT_CANCELFEPINLINEEDIT, "EDebug" );
   1.408 +		    }
   1.409 +		__ASSERT_DEBUG(inlineEditData->iInlineText==NULL || lengthOfInlineText==inlineEditData->iInlineText->Length(),Panic(EDebug));
   1.410 +		TRAPD(notUsed,
   1.411 +						if (lengthOfInlineText>0)
   1.412 +							{
   1.413 +							aParagraphContainingStartPositionOfInlineTextHasChangedFormat=DeleteWithoutDestroyingFormatL(positionOfInlineTextInDocument,lengthOfInlineText);
   1.414 +							aNumberOfCharactersSuccessfullyDeleted=lengthOfInlineText;
   1.415 +							aPositionOfInsertionPointInDocument=inlineEditData->iPositionOfInlineTextInDocument;
   1.416 +							}
   1.417 +						const HBufC* const hiddenText=inlineEditData->iHiddenText;
   1.418 +						if (hiddenText!=NULL)
   1.419 +							{
   1.420 +							if (hiddenText->Length()<=0)
   1.421 +							    {
   1.422 +							    OstTrace0( TRACE_DUMP, DUP1_CEDITABLETEXT_CANCELFEPINLINEEDIT, "EDebug" );
   1.423 +							    }
   1.424 +							__ASSERT_DEBUG(hiddenText->Length()>0, Panic(EDebug));
   1.425 +							InsertL(positionOfInlineTextInDocument,*hiddenText);
   1.426 +							aNumberOfCharactersSuccessfullyInserted=hiddenText->Length();
   1.427 +							aPositionOfInsertionPointInDocument=aNewPositionOfInsertionPointInDocument;
   1.428 +							}
   1.429 +			);
   1.430 +        UNUSED_VAR(notUsed);
   1.431 +		DeleteInlineEditDataAndSetToNull();
   1.432 +		}
   1.433 +	}
   1.434 +
   1.435 +EXPORT_C void CEditableText::OverrideFormatOfInlineTextIfApplicable(TPtrC& aView,TCharFormat& aFormat,TInt aStartPos)const
   1.436 +	{
   1.437 +	const CInlineEditData* inlineEditData=InlineEditData();
   1.438 +	if (inlineEditData!=NULL)
   1.439 +		{
   1.440 +		const TInt positionOfInlineTextInDocument=inlineEditData->iPositionOfInlineTextInDocument;
   1.441 +		const TInt lengthOfInlineText=inlineEditData->iLengthOfInlineText;
   1.442 +		const TInt originalLengthOfView=aView.Length();
   1.443 +		TInt maximumLengthOfView=originalLengthOfView;
   1.444 +		if (aStartPos<positionOfInlineTextInDocument)
   1.445 +			{
   1.446 +			maximumLengthOfView=positionOfInlineTextInDocument-aStartPos;
   1.447 +			}
   1.448 +		else if (aStartPos<positionOfInlineTextInDocument+lengthOfInlineText)
   1.449 +			{
   1.450 +			inlineEditData->iInlineTextFormatRetriever->GetFormatOfFepInlineText(aFormat,maximumLengthOfView,aStartPos-positionOfInlineTextInDocument);
   1.451 +			}
   1.452 +		if (originalLengthOfView>maximumLengthOfView)
   1.453 +			{
   1.454 +			aView.Set(aView.Left(maximumLengthOfView));
   1.455 +			}
   1.456 +		}
   1.457 +	}
   1.458 +
   1.459 +
   1.460 +EXPORT_C TInt CEditableText::GetPositionOfInlineTextInDocument() const
   1.461 +	{
   1.462 +	const CInlineEditData* inlineEditData=InlineEditData();
   1.463 +	if (inlineEditData==NULL)
   1.464 +		return KErrNotFound;
   1.465 +	return inlineEditData->iPositionOfInlineTextInDocument;	
   1.466 +	}
   1.467 +
   1.468 +
   1.469 +EXPORT_C TInt CEditableText::GetLengthOfInlineText() const
   1.470 +	{
   1.471 +	const CInlineEditData* inlineEditData=InlineEditData();
   1.472 +	if (inlineEditData==NULL)
   1.473 +		return KErrNotFound;
   1.474 +	return inlineEditData->iLengthOfInlineText;		
   1.475 +	}
   1.476 +
   1.477 +
   1.478 +//////////////////////////////////
   1.479 +// TEtextComponentInfo
   1.480 +//////////////////////////////////
   1.481 +
   1.482 +
   1.483 +
   1.484 +EXPORT_C TEtextComponentInfo::TEtextComponentInfo()
   1.485 +	: iFieldCount(0),iPictureCount(0),iStyleCount(0)
   1.486 +/** C++ constructor overloaded function.
   1.487 +
   1.488 +The object can be constructed either:by default this initializes the
   1.489 +field, picture and style counts to zerowith a field, picture and style
   1.490 +count
   1.491 +
   1.492 +@param aFieldCount Specifies the number of fields in the text object.
   1.493 +@param aPictureCount Specifies the number of pictures in the text object
   1.494 +(rich text only).
   1.495 +@param aStyleCount  Specifies the number of styles owned or referenced by
   1.496 +the text object (rich text only). */
   1.497 +    {}
   1.498 +
   1.499 +
   1.500 +EXPORT_C TEtextComponentInfo::TEtextComponentInfo(TInt aFieldCount,TInt aPictureCount,TInt aStyleCount)
   1.501 +	: iFieldCount(aFieldCount),iPictureCount(aPictureCount),iStyleCount(aStyleCount)
   1.502 +/** C++ constructor overloaded function. The object can be constructed either:
   1.503 +
   1.504 +by default  this initializes the field, picture and style counts to zero
   1.505 +
   1.506 +with a field, picture and style count
   1.507 +
   1.508 +@param aFieldCount Specifies the number of fields in the text object. 
   1.509 +@param aPictureCount Specifies the number of pictures in the text object (rich 
   1.510 +text only). 
   1.511 +@param aStyleCount Specifies the number of styles owned or referenced by the 
   1.512 +text object (rich text only). */
   1.513 +	{}
   1.514 +
   1.515 +
   1.516 +//////////////////////////////////
   1.517 +// CPlainText
   1.518 +//////////////////////////////////
   1.519 +
   1.520 +
   1.521 +EXPORT_C void CPlainText::__DbgTestInvariant()const
   1.522 +// Provides class invariants.  Explanations below:
   1.523 +//
   1.524 +	{
   1.525 +#ifdef _DEBUG
   1.526 +// ASSERT: Storage handle is good.
   1.527 +	__ASSERT_DEBUG(iByteStore!=NULL,User::Invariant());
   1.528 +// ASSERT: The text component must be non-negative in length
   1.529 +	__ASSERT_DEBUG(DocumentLength()>=0,User::Invariant());
   1.530 +#endif
   1.531 +	}
   1.532 +
   1.533 +
   1.534 +
   1.535 +
   1.536 +EXPORT_C CPlainText* CPlainText::NewL(TDocumentStorage aStorage,TInt aDefaultTextGranularity)
   1.537 +/** Allocates and constructs a plain text object overloaded function.
   1.538 +
   1.539 +The text object's contents may be restored from a stream store. If the
   1.540 +text object supports fields, a field factory should be specified.
   1.541 +
   1.542 +@param aStorage  The type of in-memory buffer to use. Defaults to ESegmentedStorage.
   1.543 +@param aDefaultTextGranularity  Specifies the granularity of the in-memory 
   1.544 +buffer. Default is EDefaultTextGranularity bytes (=256).
   1.545 +@param aStore   Stream store from which the object is restored.
   1.546 +@param aStreamId  ID of the stream store.
   1.547 +@param aFieldFactory Pointer to a field factory. A field factory must be
   1.548 +provided if the text object supports the addition of fields.
   1.549 +@return Pointer to the new plain text object. */	
   1.550 +	{
   1.551 +	CPlainText* self=new(ELeave) CPlainText();
   1.552 +	CleanupStack::PushL(self);
   1.553 +	self->ConstructL(aStorage,aDefaultTextGranularity);
   1.554 +	CleanupStack::Pop();
   1.555 +	return self;
   1.556 +	}
   1.557 +
   1.558 +
   1.559 +EXPORT_C CPlainText* CPlainText::NewL(const CStreamStore& aStore,TStreamId aStreamId,MTextFieldFactory* aFieldFactory,TDocumentStorage aStorage)
   1.560 +/** Returns a handle to a new instance of this class, restored from the specified
   1.561 +read stream.*/
   1.562 +	{
   1.563 +	CPlainText* self=new(ELeave) CPlainText();
   1.564 +	CleanupStack::PushL(self);
   1.565 +	self->ConstructL(aStore,aStreamId,aFieldFactory,aStorage);
   1.566 +	CleanupStack::Pop();
   1.567 +	return self;
   1.568 +	}
   1.569 +
   1.570 +
   1.571 +EXPORT_C CPlainText::CPlainText()
   1.572 +	{
   1.573 +	SetHasChanged(EFalse);
   1.574 +	}
   1.575 +
   1.576 +
   1.577 +EXPORT_C void CPlainText::ConstructL(TDocumentStorage aStorage,TInt aDefaultTextGranularity)
   1.578 +/** Allocates storage of CBufFlat or CBufSeg, according
   1.579 +to the parameter aStorage.
   1.580 +Creates & initializes the field set.*/
   1.581 +	{
   1.582 +	DoConstructL(aStorage,aDefaultTextGranularity);
   1.583 +	InsertEodL();
   1.584 +
   1.585 +	__TEST_INVARIANT;
   1.586 +	}
   1.587 +
   1.588 +
   1.589 +EXPORT_C void CPlainText::ConstructL(const CStreamStore& aStore,TStreamId aStreamId,MTextFieldFactory* aFieldFactory,
   1.590 +									 TDocumentStorage aStorage)
   1.591 +/** Allocates storage of CBufFlat or CBufSeg, according
   1.592 + to the parameter aStorage, restoring contents from the specified read-stream aStream.*/
   1.593 +	{
   1.594 +	DoConstructL(aStorage,EDefaultTextGranularity,aFieldFactory);	
   1.595 +	RestoreL(aStore,aStreamId);
   1.596 +
   1.597 +	__TEST_INVARIANT;
   1.598 +	}
   1.599 +
   1.600 +
   1.601 +EXPORT_C void CPlainText::DoConstructL(TDocumentStorage aStorage,TInt aDefaultTextGranularity,MTextFieldFactory* aFieldFactory)
   1.602 +/** Allocates storage of CBufFlat or CBufSeg, according to the parameter aStorage.
   1.603 +Creates & initializes the field set.*/
   1.604 +	{
   1.605 +	if (iByteStore!=NULL)
   1.606 +	    {
   1.607 +	    OstTrace0( TRACE_DUMP, CPLAINTEXT_DOCONSTRUCTL, "EConstructCalledTwice" );
   1.608 +	    }
   1.609 +	__ASSERT_DEBUG(iByteStore==NULL,Panic(EConstructCalledTwice));
   1.610 +	
   1.611 +	iByteStore=(aStorage==ESegmentedStorage)
   1.612 +		? (CBufBase*)CBufSeg::NewL(aDefaultTextGranularity*sizeof(TText))
   1.613 +		: (CBufBase*)CBufFlat::NewL(aDefaultTextGranularity*sizeof(TText));
   1.614 +	if (aFieldFactory)
   1.615 +		SetFieldFactory(aFieldFactory);
   1.616 +	}
   1.617 +
   1.618 +
   1.619 +
   1.620 +EXPORT_C CPlainText::~CPlainText()
   1.621 +/** The destructor frees the object's text storage and field set, prior to its 
   1.622 +destruction. */
   1.623 +	{
   1.624 +	KillFieldSet();
   1.625 +	delete iByteStore;
   1.626 +	}
   1.627 +
   1.628 +
   1.629 +void CPlainText::KillFieldSet()
   1.630 +/** Delete the field set if it is resident in memory.*/
   1.631 +	{
   1.632 +	if (FieldSetPresent())
   1.633 +		delete iFieldSet.AsPtr();
   1.634 +	iFieldSet=NULL;
   1.635 +	}
   1.636 +
   1.637 +
   1.638 +//
   1.639 +// CPlainText - Persistence
   1.640 +
   1.641 +
   1.642 +EXPORT_C void CPlainText::StoreComponentsL(CStreamStore& aStore,CStoreMap& aMap)const
   1.643 +/** Stores the plain text object's components to the stream store specified.
   1.644 +
   1.645 +@param aStore Stream store to which the text object's components are written. 
   1.646 +@param aMap A store map. This binds the address of each component to the stream 
   1.647 +ID of aStore. This is needed to support the deferred loading of pictures in 
   1.648 +rich text. */
   1.649 +	{
   1.650 +	// Store any field components, then store the field set out-of-line, if present.
   1.651 +	if (FieldSetPresent())
   1.652 +		{
   1.653 +		TStreamId id=iFieldSet->StoreL(aStore);
   1.654 +		aMap.BindL(iFieldSet,id);
   1.655 +		}
   1.656 +	}
   1.657 +
   1.658 +
   1.659 +
   1.660 +EXPORT_C void CPlainText::RestoreComponentsL(const CStreamStore& aStore)
   1.661 +/** Restores the plain text object's field set from a stream store.
   1.662 +
   1.663 +@param aStore The stream store from which the field set is restored. */
   1.664 +	{
   1.665 +	// Load the field set, and load any referenced pictures
   1.666 +	TStreamId id=iFieldSet.AsId();
   1.667 +	if (id!=KNullStreamId)
   1.668 +		{
   1.669 +		CreateFieldSetL(0);
   1.670 +		iFieldSet->SetFieldFactory(iFieldFactory);
   1.671 +		iFieldSet->RestoreL(aStore,id);
   1.672 +		}
   1.673 +	}
   1.674 +
   1.675 +
   1.676 +
   1.677 +EXPORT_C void CPlainText::StoreFieldComponentsL(CStreamStore& aStore,CStoreMap& aMap)const
   1.678 +/** Stores the plain text object's field components to a stream store.
   1.679 +
   1.680 +@param aStore Stream store to which the fields are written. 
   1.681 +@param aMap A store map. This binds the address of each text component to the 
   1.682 +stream ID of aStore. This is needed to support the deferred loading of pictures 
   1.683 +in rich text. */
   1.684 +	{
   1.685 +	// 2' StoreComponents() 
   1.686 +	// Only has effect if a field set is present.
   1.687 +	if (FieldSetPresent())
   1.688 +		iFieldSet->StoreFieldsL(aStore,aMap);
   1.689 +	}
   1.690 +
   1.691 +
   1.692 +
   1.693 +EXPORT_C void CPlainText::RestoreFieldComponentsL(const CStreamStore& aStore)
   1.694 +/** Restores the plain text object's field set.
   1.695 +
   1.696 +@param aStore The stream store from which the fields are restored. */
   1.697 +	{
   1.698 +	// 2' RestoreComponents()
   1.699 +	// Only has effect if a field set is present - (has been Internalized())
   1.700 +	if (FieldSetPresent())
   1.701 +		iFieldSet->RestoreFieldsL(aStore);
   1.702 +	}
   1.703 +
   1.704 +
   1.705 +	
   1.706 +EXPORT_C void CPlainText::ExternalizeL(RWriteStream& aStream)const
   1.707 +/** Externalises a plain text object to a write stream. The presence of this function 
   1.708 +means that the standard templated operator<<() (defined in s32strm.h) is available 
   1.709 +to externalise objects of this class.
   1.710 +
   1.711 +@param aStream Stream to which the object should be externalised. */
   1.712 +	{
   1.713 +	// Store this object in the specified write-stream.
   1.714 +	CEditableText::ExternalizeL(aStream);
   1.715 +	DoExternalizeFieldDataL(aStream);
   1.716 +	DoExternalizePlainTextL(aStream);
   1.717 +	}
   1.718 +
   1.719 +
   1.720 +
   1.721 +EXPORT_C void CPlainText::InternalizeL(RReadStream& aStream)
   1.722 +/** Internalises the text object's text content and field set from a read stream. 
   1.723 +The presence of this function means that the standard templated operator>>() 
   1.724 +(defined in s32strm.h) is available to internalise objects of this class. 
   1.725 +InternalizeL() has construct rather than assignment semantics. You should 
   1.726 +not use it for fully initialised objects.
   1.727 +
   1.728 +@param aStream Stream from which the object should be internalised. */
   1.729 +	{
   1.730 +	// Restores plain text from the specified read-stream.
   1.731 +	// Internalize has construction semantics, not assignment semantics.
   1.732 +	CEditableText::InternalizeL(aStream);
   1.733 +	DoInternalizeFieldDataL(aStream);
   1.734 +	DoInternalizePlainTextL(aStream);
   1.735 +	
   1.736 +	__TEST_INVARIANT;
   1.737 +	}
   1.738 +
   1.739 +
   1.740 +
   1.741 +EXPORT_C void CPlainText::ExternalizeFieldDataL(RWriteStream& aStream)const
   1.742 +/** Externalises the plain text object's field set.
   1.743 +
   1.744 +@param aStream The stream to which the field set should be written. */
   1.745 +	{
   1.746 +	// Save just the field set
   1.747 +	__TEST_INVARIANT;
   1.748 +
   1.749 +	TUint fieldCount=(TUint)FieldCount();
   1.750 +	if(fieldCount<KFieldCountLimit)
   1.751 +		aStream.WriteUint8L(fieldCount);
   1.752 +	else
   1.753 +		{
   1.754 +		aStream.WriteUint8L(KFieldCountLimit);
   1.755 +		aStream.WriteUint32L(fieldCount);
   1.756 +		}
   1.757 +	if (fieldCount>0)
   1.758 +		aStream<< *iFieldSet;
   1.759 +	}
   1.760 +
   1.761 +
   1.762 +
   1.763 +EXPORT_C void CPlainText::InternalizeFieldDataL(RReadStream& aStream)
   1.764 +/** Internalizes the field set.
   1.765 +
   1.766 +@param aStream The read stream from which the field set is read. */
   1.767 +	{
   1.768 +	// 2' InternalizeL()
   1.769 +	// Restores field records from the specified read-stream.
   1.770 +	// Internalize has construction semantics, not assignment semantics.
   1.771 +
   1.772 +	TUint fieldCount=aStream.ReadUint8L();
   1.773 +	if (fieldCount==KFieldCountLimit)
   1.774 +		fieldCount=aStream.ReadUint32L();
   1.775 +	if (fieldCount>0)
   1.776 +		{
   1.777 +		if (!iFieldSet)
   1.778 +			CreateFieldSetL(DocumentLength());
   1.779 +		aStream>> *iFieldSet;
   1.780 +		}
   1.781 +	}
   1.782 +
   1.783 +
   1.784 +EXPORT_C void CPlainText::DoInternalizeFieldDataL(RReadStream& aStream)
   1.785 +/** Read from the stream until the field data is identified, and consume it.*/
   1.786 +	{
   1.787 +	TUid uid=UidFromStreamL(aStream);
   1.788 +	while (uid!=KPlainTextFieldDataUid)
   1.789 + 		{
   1.790 +		if (uid==KPlainTextCharacterDataUid)
   1.791 +			User::Leave(KErrCorrupt);  // There is no field Data !!!!!
   1.792 +		CPlainText::ConsumeAdornmentL(aStream);
   1.793 +		uid=UidFromStreamL(aStream);
   1.794 +		}
   1.795 +	if (FieldSetPresent())
   1.796 +		iFieldFactory=iFieldSet->FieldFactory();
   1.797 +	KillFieldSet();
   1.798 +	aStream>> iFieldSet;  // a swizzle
   1.799 +	}
   1.800 +
   1.801 +
   1.802 +EXPORT_C void CPlainText::DoExternalizeFieldDataL(RWriteStream& aStream)const
   1.803 +/** Write to the stream, the T.V. representing the field set.*/
   1.804 +	{
   1.805 +	aStream<< KPlainTextFieldDataUid;
   1.806 +	if (FieldSetPresent())
   1.807 +		aStream<< iFieldSet;
   1.808 +	else
   1.809 +		aStream<< KNullStreamId;
   1.810 +	}
   1.811 +
   1.812 +
   1.813 +EXPORT_C void CPlainText::DoExternalizePlainTextL(RWriteStream& aStream)const
   1.814 +/** Write to the stream, the T.V. representing the plain text.*/
   1.815 +	{
   1.816 +	aStream<< KPlainTextCharacterDataUid;
   1.817 +	ExternalizePlainTextL(aStream);
   1.818 +	}
   1.819 +
   1.820 +
   1.821 +
   1.822 +EXPORT_C void CPlainText::ExternalizePlainTextL(RWriteStream& aStream)const
   1.823 +	
   1.824 +/** Externalises the plain text object's text content (preceded by a length count) 
   1.825 +to a write stream.
   1.826 +
   1.827 +@param aStream Stream to which the text content should be externalised. */
   1.828 +	{
   1.829 +	// Save just the bytestore
   1.830 +	__TEST_INVARIANT;
   1.831 +	::ExternalizeTextL(aStream,*iByteStore,0,iByteStore->Size() / sizeof(TText),TRUE);
   1.832 +	}
   1.833 +
   1.834 +EXPORT_C void CPlainText::DoInternalizePlainTextL(RReadStream& aStream)
   1.835 +/** Read from the stream until the character data is found, and consume it.*/
   1.836 +	{
   1.837 +	TUid uid=UidFromStreamL(aStream);
   1.838 +	while (uid!=KPlainTextCharacterDataUid)
   1.839 +		{
   1.840 +		CPlainText::ConsumeAdornmentL(aStream);
   1.841 +		uid=UidFromStreamL(aStream);
   1.842 +		}
   1.843 +	CPlainText::InternalizePlainTextL(aStream);
   1.844 +	}
   1.845 +
   1.846 +EXPORT_C void CPlainText::InternalizePlainTextL(RReadStream& aStream)
   1.847 +/** Internalises an empty text object's text content from a read stream
   1.848 +overloaded function.
   1.849 +
   1.850 +This function has construct rather than assignment semantics. You
   1.851 +should not use it for fully initialised objects.NoteThe overload which
   1.852 +takes a length argument is not intended for general use and its use is
   1.853 +deprecated.
   1.854 +
   1.855 +@param aStream Stream from which the object should be internalised.
   1.856 +@param  aLength Indicates the number of characters which should be
   1.857 +read, after expansion from their compressed format. */	
   1.858 +    {
   1.859 +	// Restores plain text content from the specified read-stream.
   1.860 +	// Internalize has construction semantics, not assignment semantics.
   1.861 +	::InternalizeTextL(aStream,*iByteStore);
   1.862 +	SetHasChanged(EFalse);
   1.863 +
   1.864 +	__TEST_INVARIANT;
   1.865 +	}
   1.866 +
   1.867 +
   1.868 +EXPORT_C void CPlainText::ExternalizePlainTextNoLengthCountL(RWriteStream& aStream)const
   1.869 +/** Externalises the plain text object's text content to a write stream.
   1.870 +
   1.871 +This function differs from ExternalizePlainTextL() in that 
   1.872 +it does not precede the text content with a length count.
   1.873 +This function is not intended for general use and is deprecated.
   1.874 +@see void CPlainText::ExternalizePlainTextL(RWriteStream& aStream)const
   1.875 +@deprecated */
   1.876 +	{
   1.877 +	::ExternalizeTextL(aStream,*iByteStore,0,iByteStore->Size() / sizeof(TText),FALSE);
   1.878 +	}
   1.879 +
   1.880 +
   1.881 +EXPORT_C void CPlainText::InternalizePlainTextL(RReadStream& aStream,TInt aLength)
   1.882 +/** Internalises an empty text object's text content from a read stream's 
   1.883 +overloaded function.
   1.884 +
   1.885 +This function has construct rather than assignment semantics. You should not 
   1.886 +use it for fully initialised objects.
   1.887 +
   1.888 +Note
   1.889 +
   1.890 +The overload which takes a length argument is not intended for general use 
   1.891 +and its use is deprecated. 
   1.892 +
   1.893 +@param aStream Stream from which the object should be internalised. 
   1.894 +@param aLength Indicates the number of characters which should be read, after 
   1.895 +expansion from their compressed format. 
   1.896 +@deprecated */
   1.897 +	{
   1.898 +	::InternalizeTextL(aStream,*iByteStore,aLength);
   1.899 +	}
   1.900 +
   1.901 +
   1.902 +// Copy the specified section of plain text to the specified store.
   1.903 +
   1.904 +
   1.905 +EXPORT_C void CPlainText::CopyToStoreL(CStreamStore& aStore,CStreamDictionary& aDictionary,TInt aPos,TInt aLength) const
   1.906 +/** Copies plain text including fields, if present, to the clipboard.
   1.907 +
   1.908 +A panic occurs in any of the following circumstances:
   1.909 +
   1.910 +aPos is invalid
   1.911 +
   1.912 +aLength is invalid (zero or less)
   1.913 +
   1.914 +the sum of aPos and aLength is greater than or equal to the number of characters 
   1.915 +in the document
   1.916 +
   1.917 +@param aStore Stream store to which the text is written. 
   1.918 +@param aDictionary The stream dictionary. 
   1.919 +@param aPos The document position from which to begin copying. 
   1.920 +@param aLength The number of characters to copy. */
   1.921 +	{
   1.922 +	if (aLength > 0)
   1.923 +		DoCopyToStoreL(aStore,aDictionary,aPos,aLength);
   1.924 +	}
   1.925 +
   1.926 +
   1.927 +TStreamId CPlainText::DoCopyToStoreL(CStreamStore& aStore,CStreamDictionary& aDictionary,TInt aPos,TInt aLength) const
   1.928 +	{
   1.929 +	__TEST_INVARIANT;
   1.930 +	TInt documentLength = DocumentLength();
   1.931 +	if (aPos < 0 || aPos > documentLength)
   1.932 +	    {
   1.933 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_DOCOPYTOSTOREL, "ECharPosBeyondDocument" );
   1.934 +	    }
   1.935 +	__ASSERT_ALWAYS(aPos >= 0 && aPos <= documentLength,Panic(ECharPosBeyondDocument));
   1.936 +	if (aLength < 0)
   1.937 +	    {
   1.938 +	    OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_DOCOPYTOSTOREL, "ECopyToStreamNegativeLength" );
   1.939 +	    }
   1.940 +	__ASSERT_ALWAYS(aLength >= 0,Panic(ECopyToStreamNegativeLength));
   1.941 +	if (aPos + aLength > documentLength)
   1.942 +	    {
   1.943 +	    OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_DOCOPYTOSTOREL, "ECharPosBeyondDocument" );
   1.944 +	    }
   1.945 +	__ASSERT_ALWAYS(aPos + aLength <= documentLength,Panic(ECharPosBeyondDocument));
   1.946 +
   1.947 +	if (aLength == 0)
   1.948 +		return KNullStreamId;
   1.949 +
   1.950 +	CStoreMap* map=CStoreMap::NewLC(aStore);
   1.951 +	CopyComponentsL(aStore,*map,aPos,aLength);
   1.952 +
   1.953 +	// create custom externalizer over the map
   1.954 +	TFieldMapExternalizer fMap(*map);
   1.955 +	RStoreWriteStream stream(fMap);
   1.956 +	TStreamId id=stream.CreateLC(aStore);
   1.957 +	CopyToStreamL(stream,aPos,aLength);
   1.958 +	stream.CommitL();
   1.959 +
   1.960 +	aDictionary.AssignL(KClipboardUidTypePlainText,id);
   1.961 +	map->Reset();
   1.962 +	CleanupStack::PopAndDestroy(2);
   1.963 +
   1.964 +	__TEST_INVARIANT;
   1.965 +	return id;
   1.966 +	}
   1.967 +
   1.968 +
   1.969 +void CPlainText::CopyComponentsL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength)const
   1.970 +// Copy/Paste 2' StoreComponentsL() - only if a field set is present.
   1.971 +// 
   1.972 +	{
   1.973 +	if (FieldSetPresent())
   1.974 +		iFieldSet->CopyComponentsL(aStore,aMap,aPos,aLength);
   1.975 +	}
   1.976 +
   1.977 +
   1.978 +// Write the plain text to the stream.
   1.979 +void CPlainText::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength)const
   1.980 +	{
   1.981 +	__TEST_INVARIANT;
   1.982 +	TInt documentLength = DocumentLength();
   1.983 +	if (aPos < 0 || aPos > documentLength)
   1.984 +	    {
   1.985 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_COPYTOSTREAML, "ECharPosBeyondDocument" );
   1.986 +	    }
   1.987 +	__ASSERT_ALWAYS(aPos >= 0 && aPos <= documentLength,Panic(ECharPosBeyondDocument));
   1.988 +	if (aLength < 0)
   1.989 +	    {
   1.990 +	    OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_COPYTOSTREAML, "ECopyToStreamNegativeLength" );
   1.991 +	    }
   1.992 +	__ASSERT_ALWAYS(aLength >= 0,Panic(ECopyToStreamNegativeLength));
   1.993 +	if (aPos + aLength > documentLength)
   1.994 +	    {
   1.995 +	    OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_COPYTOSTREAML, "ECharPosBeyondDocument" );
   1.996 +	    }
   1.997 +	__ASSERT_ALWAYS(aPos + aLength <= documentLength,Panic(ECharPosBeyondDocument));
   1.998 +
   1.999 +	aStream.WriteInt32L(aLength);
  1.1000 +	::ExternalizeTextL(aStream,*iByteStore,aPos,aLength,FALSE);
  1.1001 +
  1.1002 +	// Write the field set if any.
  1.1003 +	TBool fieldSetPresent = FieldSetPresent();
  1.1004 +	aStream.WriteUint8L(fieldSetPresent != EFalse);
  1.1005 +	if (fieldSetPresent)
  1.1006 +		iFieldSet->CopyToStreamL(aStream,aPos,aLength);
  1.1007 +	}
  1.1008 +
  1.1009 +
  1.1010 + 
  1.1011 +
  1.1012 +EXPORT_C TInt CPlainText::PasteFromStoreL(const CStreamStore& aStore,const CStreamDictionary& aDictionary,TInt aPos)
  1.1013 +/** Pastes plain text and fields, if present, from the clipboard into the current 
  1.1014 +text object at the specified document position. The entire contents of the 
  1.1015 +store are pasted.
  1.1016 +
  1.1017 +@param aStore The steam store from which to paste the text. 
  1.1018 +@param aDictionary The stream dictionary. 
  1.1019 +@param aPos Document position at which to paste. Must be valid or the function 
  1.1020 +raises a panic. 
  1.1021 +@return The number of characters pasted. */
  1.1022 +	{
  1.1023 +	// Paste the lesser of aMaxPasteLength and the entire clipboard contents.
  1.1024 +	// Return the number of characters pasted.
  1.1025 +	TStreamId id=aDictionary.At(KClipboardUidTypePlainText);
  1.1026 +	return DoPasteFromStoreL(aStore,id,aPos);
  1.1027 +	}
  1.1028 +
  1.1029 +
  1.1030 +TInt CPlainText::DoPasteFromStoreL(const CStreamStore& aStore,TStreamId aStreamId,TInt aPos)
  1.1031 +	{
  1.1032 +	if (aPos>DocumentLength())
  1.1033 +	    {
  1.1034 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_DOPASTEFROMSTOREL, "ECharPosBeyondDocument" );
  1.1035 +	    }
  1.1036 +	__ASSERT_ALWAYS(aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1037 +
  1.1038 +	TInt charsPasted=0;
  1.1039 +	if (aStreamId!=KNullStreamId)
  1.1040 +		{// There is a recognised type in the clipboard.
  1.1041 +		RStoreReadStream stream;
  1.1042 +		stream.OpenLC(aStore,aStreamId);
  1.1043 +		charsPasted=PasteFromStreamL(stream,aPos);
  1.1044 +		CleanupStack::PopAndDestroy();
  1.1045 +		//
  1.1046 +		PasteComponentsL(aStore,aPos);
  1.1047 +		SetHasChanged(ETrue);
  1.1048 +		}
  1.1049 +
  1.1050 +	__TEST_INVARIANT;
  1.1051 +	return charsPasted;
  1.1052 +	}
  1.1053 +
  1.1054 +
  1.1055 +void CPlainText::PasteComponentsL(const CStreamStore& aStore,TInt aPos)
  1.1056 +// Copy/Paste 2' RestoreComponentsL() - only if a field set is present.
  1.1057 +//
  1.1058 +	{
  1.1059 +	if (FieldSetPresent())
  1.1060 +		iFieldSet->PasteComponentsL(aStore,aPos);
  1.1061 +	}
  1.1062 +
  1.1063 +
  1.1064 +// Paste everything in the stream.
  1.1065 +TInt CPlainText::PasteFromStreamL(RReadStream& aStream,TInt aPos)
  1.1066 +	{
  1.1067 +	TInt chars_read = 0;
  1.1068 +	TInt error = KErrNone;
  1.1069 +
  1.1070 +	TRAP(error, chars_read=CPlainText::DoPasteFromStreamL(aStream, aPos));
  1.1071 +
  1.1072 +	UpdatePageTable(aPos,chars_read);
  1.1073 +
  1.1074 +	/*
  1.1075 +	If there was an exception delete any inserted text and propagate the exception.
  1.1076 +	Not deleting the text would cause the size of the text to be inconsistent with the size
  1.1077 +	implied elsewhere, such as in the formatting information stored in CRichText objects.
  1.1078 +	*/
  1.1079 +	if (error != KErrNone)
  1.1080 +		{
  1.1081 +		DoPtDelete(aPos,chars_read);
  1.1082 +		OstTrace1( TRACE_FATAL, DUP1_CPLAINTEXT_PASTEFROMSTREAML, "Leave code=%d", error );
  1.1083 +		User::Leave(error);
  1.1084 +		}
  1.1085 +
  1.1086 +	__TEST_INVARIANT;
  1.1087 +	return chars_read;
  1.1088 +	}
  1.1089 +
  1.1090 +
  1.1091 +TInt CPlainText::DoPasteFromStreamL(RReadStream& aStream, TInt aPos)
  1.1092 +	{
  1.1093 +	TInt chars_read = 0;
  1.1094 +
  1.1095 +	CBufSeg* buffer = CBufSeg::NewL(512);
  1.1096 +	CleanupStack::PushL(buffer);
  1.1097 +	TInt length = aStream.ReadInt32L();
  1.1098 +	::InternalizeTextL(aStream,*buffer,length);
  1.1099 +
  1.1100 +	/*
  1.1101 +	Insert the text bit by bit so that memory consumed by the CPlainText object is freed from the buffer;
  1.1102 +	this is important if pasting huge amounts of text.
  1.1103 +	*/
  1.1104 +	while (buffer->Size() > 0)
  1.1105 +		{
  1.1106 +		TPtr8 p8 = buffer->Ptr(0);
  1.1107 +		TInt bytes = p8.Length();
  1.1108 +		TInt chars = bytes / sizeof(TText);
  1.1109 +		TPtrC p((TText*)p8.Ptr(),chars);	// platform dependency in the Unicode build; relies on little-endianness
  1.1110 +		PtInsertL(aPos + chars_read,p);
  1.1111 +		buffer->Delete(0,bytes);
  1.1112 +		chars_read += chars;
  1.1113 +		}
  1.1114 +
  1.1115 +	CleanupStack::PopAndDestroy();	// buffer
  1.1116 +	buffer = NULL;
  1.1117 +
  1.1118 +	// If there's a field set, internalize it.
  1.1119 +	if (aStream.ReadUint8L() != 0)	// next byte is non-zero if there's a field set
  1.1120 +		{
  1.1121 +		if (!FieldSetPresent())
  1.1122 +			CreateFieldSetL(DocumentLength());
  1.1123 +		iFieldSet->PasteFromStreamL(aStream,aPos,chars_read);
  1.1124 +		}
  1.1125 +	
  1.1126 +	return chars_read;
  1.1127 +	}
  1.1128 +
  1.1129 +
  1.1130 +void CPlainText::InsertEodL()
  1.1131 +/** Inserts the end-of-document character upon document construction.*/
  1.1132 +	{
  1.1133 +// ASSERT: The plain text component is empty.
  1.1134 +	if (DocumentLength()!=-1)
  1.1135 +	    {
  1.1136 +	    OstTrace0( TRACE_DUMP, DUP1_CPLAINTEXT_INSERTEODL, "ECorruptTextStore" );
  1.1137 +	    }
  1.1138 +	__ASSERT_DEBUG(DocumentLength()==-1,Panic(ECorruptTextStore));
  1.1139 +	TBuf<1> content;
  1.1140 +	content.Append(EParagraphDelimiter);
  1.1141 +	TPtrC eod(content);
  1.1142 +	DoPtInsertL(0,eod);
  1.1143 +
  1.1144 +	__TEST_INVARIANT;
  1.1145 +	}
  1.1146 +
  1.1147 + 
  1.1148 +EXPORT_C void CPlainText::Reset()
  1.1149 +/** Deletes all text content, formatting and fields from the document, leaving 
  1.1150 +the single paragraph delimiter which terminates the text object. */
  1.1151 +	{	
  1.1152 +	// Resets document contents to a single end-of-document character, with no other content.
  1.1153 +	// (No reset occurs if the component is already in the reset state.  Avoids an assertion
  1.1154 +	// failure in the delete method, where length to delete !> 0).
  1.1155 +	__TEST_INVARIANT;
  1.1156 +		
  1.1157 +	TInt content=DocumentLength();
  1.1158 +	if (content>0)
  1.1159 +		DoPtDelete(0,content);
  1.1160 +	if (FieldSetPresent())
  1.1161 +		KillFieldSet();
  1.1162 +	SetHasChanged(ETrue);
  1.1163 +	
  1.1164 +	__TEST_INVARIANT;
  1.1165 +	}
  1.1166 +
  1.1167 +
  1.1168 + 
  1.1169 +
  1.1170 +EXPORT_C TInt CPlainText::DocumentLength()const
  1.1171 +/** Gets the the number of characters in the text object.
  1.1172 +
  1.1173 +Note: the count includes all non-printing characters but excludes the end 
  1.1174 +of text paragraph delimiter, so that the smallest possible return value is 
  1.1175 +zero.
  1.1176 +
  1.1177 +@return The number of characters in the text object. */
  1.1178 +	{return ((iByteStore->Size()/sizeof(TText))-1);}
  1.1179 +
  1.1180 +
  1.1181 +EXPORT_C void CPlainText::InsertL(TInt aInsertPos,const TChar& aChar)
  1.1182 +/** Inserts either a single character or a descriptor into the text object
  1.1183 +at a specified document position.
  1.1184 +
  1.1185 +Updates the page table.
  1.1186 +
  1.1187 +@param aPos The document position at which to insert the character/descriptor. 
  1.1188 +Must be valid, or a panic occurs.
  1.1189 +@param aChar The character to insert.
  1.1190 +@param aBuf The descriptor to insert. */
  1.1191 +	{
  1.1192 +	__TEST_INVARIANT;
  1.1193 +	if (aInsertPos<0 || aInsertPos>DocumentLength())
  1.1194 +	    {
  1.1195 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_INSERTL, "ECharPosBeyondDocument" );
  1.1196 +	    }
  1.1197 +	__ASSERT_ALWAYS(aInsertPos>=0 && aInsertPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1198 +
  1.1199 +	TInt contentLength = 1;
  1.1200 +	if (aChar < 0x10000)
  1.1201 +		{
  1.1202 +		TBuf<1> content;
  1.1203 +		content.Append(aChar);
  1.1204 +		DoPtInsertL(aInsertPos,content);
  1.1205 +		}
  1.1206 +	else
  1.1207 +		{
  1.1208 +		TText16 high = TChar::GetHighSurrogate(aChar);
  1.1209 +		TText16 low = TChar::GetLowSurrogate(aChar);
  1.1210 +		RDebug::Print(_L("CPlainText::InsertL(%d), %X expand to %X %X."), aInsertPos, TUint(aChar), high, low);
  1.1211 +	
  1.1212 +		TBuf<2> content;
  1.1213 +		contentLength = 2;
  1.1214 +		content.Append(high);
  1.1215 +		content.Append(low);
  1.1216 +		DoPtInsertL(aInsertPos,content);
  1.1217 +		}
  1.1218 +	if (FieldSetPresent())
  1.1219 +		iFieldSet->NotifyInsertion(aInsertPos,contentLength);
  1.1220 +
  1.1221 +	SetHasChanged(ETrue);
  1.1222 +
  1.1223 +	__TEST_INVARIANT;
  1.1224 +	}
  1.1225 +
  1.1226 +
  1.1227 +EXPORT_C void CPlainText::InsertL(TInt aPos,const TDesC& aBuf)
  1.1228 +/** Inserts the contents of aBuf into the document at position aPos.*/	
  1.1229 +	{
  1.1230 +	PtInsertL(aPos,aBuf);
  1.1231 +	SetHasChanged(ETrue);
  1.1232 +	}
  1.1233 +
  1.1234 +
  1.1235 +EXPORT_C void CPlainText::PtInsertL(TInt aPos,const TDesC& aBuf)
  1.1236 +/** Inserts the contents a aBuf into the document at position aInsertPos.
  1.1237 +Maintain field set.*/
  1.1238 +	{
  1.1239 +	__TEST_INVARIANT;
  1.1240 +	if (aPos<0 || aPos>DocumentLength())
  1.1241 +	    {
  1.1242 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_PTINSERTL, "ECharPosBeyondDocument" );
  1.1243 +	    }
  1.1244 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1245 +
  1.1246 +	DoPtInsertL(aPos,aBuf);
  1.1247 +	if (FieldSetPresent())
  1.1248 +		iFieldSet->NotifyInsertion(aPos,aBuf.Length());
  1.1249 +
  1.1250 +	__TEST_INVARIANT;
  1.1251 +	}
  1.1252 +
  1.1253 +
  1.1254 +EXPORT_C void CPlainText::DoPtInsertL(TInt aPos,const TDesC& aBuf)
  1.1255 +/** Inserts the contents a aBuf into the document at position aInsertPos.
  1.1256 +Maintain field set.*/
  1.1257 +	{
  1.1258 +	TPtrC8 buf((TUint8*)aBuf.Ptr(),aBuf.Size());
  1.1259 +	iByteStore->InsertL(aPos*sizeof(TText),buf);
  1.1260 +	UpdatePageTable(aPos,aBuf.Length());
  1.1261 +	}
  1.1262 +
  1.1263 +
  1.1264 +void CPlainText::InsertL(TInt aPos,const CPlainText* aText)
  1.1265 +/** Insert the specified plain text object at the specified character position.
  1.1266 +(Called by CRichText::Insert()*/
  1.1267 +	{
  1.1268 +	TInt lengthRemaining=aText->DocumentLength();
  1.1269 +	TInt readPos=0;
  1.1270 +	FOREVER
  1.1271 +		{
  1.1272 +		TPtrC view=aText->Read(readPos);
  1.1273 +		TInt consumed=view.Length();
  1.1274 +		if (consumed>lengthRemaining)
  1.1275 +			consumed=lengthRemaining;
  1.1276 +		InsertL(aPos,view);
  1.1277 +		lengthRemaining-=consumed;
  1.1278 +		if (lengthRemaining==0)
  1.1279 +			return;
  1.1280 +		aPos+=consumed;
  1.1281 +		readPos+=consumed;
  1.1282 +		}
  1.1283 +	}
  1.1284 +
  1.1285 +
  1.1286 +EXPORT_C TBool CPlainText::DeleteL(TInt aPos,TInt aLength)
  1.1287 +/** Deletes one or more characters beginning at, and including, the character at 
  1.1288 +a specified document position. Updates the page table. Any fields wholly contained 
  1.1289 +in the range of characters to delete are removed from the field set.
  1.1290 +
  1.1291 +@param aPos The document position from which to begin deleting. Must be valid 
  1.1292 +or a panic occurs. 
  1.1293 +@param aLength The number of characters to delete. Must be positive or a panic 
  1.1294 +occurs. The sum of aPos and aLength must be less than the document length, 
  1.1295 +or a panic occurs. 
  1.1296 +@return Indicates whether two paragraphs have been merged together as a result 
  1.1297 +of the delete, indicating that the resulting paragraph must be reformatted. 
  1.1298 +Has no meaning for plain text, so always EFalse. */
  1.1299 +	{
  1.1300 +	return Delete(aPos,aLength);
  1.1301 +	}
  1.1302 +
  1.1303 +
  1.1304 +// A non-virtual non-leaving delete function for use inside ETEXT.
  1.1305 +TBool CPlainText::Delete(TInt aPos,TInt aLength)
  1.1306 +	{
  1.1307 +	__TEST_INVARIANT;
  1.1308 +
  1.1309 +	TBool ret = DoPtDelete(aPos,aLength);
  1.1310 +	if (FieldSetPresent())
  1.1311 +		iFieldSet->NotifyDeletion(aPos,aLength);
  1.1312 +	SetHasChanged(ETrue);
  1.1313 +
  1.1314 +	__TEST_INVARIANT;
  1.1315 +	return ret;
  1.1316 +	}
  1.1317 +
  1.1318 +
  1.1319 +EXPORT_C TBool CPlainText::DoPtDelete(TInt aPos,TInt aLength)
  1.1320 +/** Deletes aLength number of characters from the document,
  1.1321 + commencing at, and including, position aPos.
  1.1322 + The return value indicates if 2 paragraphs have been merged together
  1.1323 + as a result of the delete, indicating that the resulting paragraph
  1.1324 + must be reformatted.
  1.1325 + In global text, this clearly has no reasonable meaning, so always returns
  1.1326 + EFalse, so no reformatting occurs.*/
  1.1327 +	{
  1.1328 +	TInt documentLength=DocumentLength()+1;
  1.1329 +	if (aPos<0 || aPos>documentLength)
  1.1330 +	    {
  1.1331 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_DOPTDELETE, "ECharPosBeyondDocument" );
  1.1332 +	    }
  1.1333 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=documentLength,Panic(ECharPosBeyondDocument));
  1.1334 +	if (aLength<0)
  1.1335 +	    {
  1.1336 +	    OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_DOPTDELETE, "EDeleteNegativeLength" );
  1.1337 +	    }
  1.1338 +	__ASSERT_ALWAYS(aLength>=0,Panic(EDeleteNegativeLength));
  1.1339 +	if (aPos+aLength>documentLength)
  1.1340 +	    {
  1.1341 +	    OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_DOPTDELETE, "ECharPosBeyondDocument" );
  1.1342 +	    }
  1.1343 +	__ASSERT_ALWAYS(aPos+aLength<=documentLength,Panic(ECharPosBeyondDocument));
  1.1344 +
  1.1345 +	iByteStore->Delete(aPos*sizeof(TText),aLength*sizeof(TText));
  1.1346 +	UpdatePageTable(aPos,-aLength);
  1.1347 +
  1.1348 +	return EFalse;
  1.1349 +	}
  1.1350 +
  1.1351 +
  1.1352 +
  1.1353 +EXPORT_C TInt CPlainText::ImportTextFileL(TInt aPos,const TDes& aFileName,TTextOrganisation aTextOrganisation)
  1.1354 +/** Imports a plain text file into this text object.
  1.1355 +
  1.1356 +Translates non-printing characters in the source text file into Symbian OS 
  1.1357 +special characters, for instance tabs are converted into 
  1.1358 +CEditableText::ETabCharacters, and form feeds into CEditableText::EPageBreaks. 
  1.1359 +Line feeds in the source text file are translated according to the 
  1.1360 +aTextOrganisation argument.
  1.1361 +
  1.1362 +The function leaves if there is any problem in opening the file.
  1.1363 +
  1.1364 +@param aPos Document position at which to insert the text. Must be a valid 
  1.1365 +position, or a panic occurs.
  1.1366 +@param aFileName The name of the text file to import.
  1.1367 +@param aTextOrganisation If EOrganiseByLine, a single line feed or a line feed 
  1.1368 +and carriage return is converted into a space character. A line feed which 
  1.1369 +is followed by another line feed is converted into a paragraph delimiter. 
  1.1370 +If EOrganiseByParagraph, all line feeds are converted into paragraph delimiters. 
  1.1371 +
  1.1372 +@return The number of characters imported. */
  1.1373 +	{
  1.1374 +	TInt chars_inserted = 0;
  1.1375 +	RFs file_session;
  1.1376 +	TInt error = file_session.Connect();
  1.1377 +	if (error == KErrNone)
  1.1378 +		{
  1.1379 +		RFile file;
  1.1380 +		error = file.Open(file_session,aFileName,EFileStream | EFileRead | EFileShareReadersOnly);
  1.1381 +		if (error == KErrNone)
  1.1382 +			{
  1.1383 +			RFileReadStream input_stream(file);
  1.1384 +			TRAP(error,ImportTextL(aPos,input_stream,aTextOrganisation,KMaxTInt,KMaxTInt,&chars_inserted));
  1.1385 +			input_stream.Close();
  1.1386 +			}
  1.1387 +		file.Close();
  1.1388 +		file_session.Close();
  1.1389 +		}
  1.1390 +	if (error < 0)
  1.1391 +	    {
  1.1392 +	    OstTrace1( TRACE_FATAL, DUP1_CPLAINTEXT_IMPORTTEXTFILEL, "Leave code=%x", error );
  1.1393 +	    }
  1.1394 +	User::LeaveIfError(error);
  1.1395 +	return chars_inserted;
  1.1396 +	}
  1.1397 +
  1.1398 +
  1.1399 +
  1.1400 +EXPORT_C void CPlainText::ExportAsTextL(const TDes& aFileName,TTextOrganisation aTextOrganisation,TInt aLineWrap)const
  1.1401 +/** Writes the contents of the plain text object to a text file.
  1.1402 +	
  1.1403 +The filename is given by aFileName. Any existing file with that name is 
  1.1404 +replaced. A wrap width can be specified. This is only used when exporting 
  1.1405 +by line (aTextOrganisation is EOrganiseByLine).
  1.1406 +
  1.1407 +The function leaves if there is any problem in creating or replacing the file.
  1.1408 +
  1.1409 +@param aFileName The name of the file to export the text to. If a file with 
  1.1410 +this name already exists, it is replaced. Otherwise, a new file is created.
  1.1411 +@param aTextOrganisation Defines how to translate line delimiters. If 
  1.1412 +EOrganiseByLine, lines wrap at the wrap width, as specified in aMaxLineLength. 
  1.1413 +If EOrganiseByParagraph, lines do not wrap and paragraph delimiters are 
  1.1414 +converted into CR/LF pairs.
  1.1415 +@param aMaxLineLength The maximum number of characters in each line, (only 
  1.1416 +relevant if the text organisation is EOrganiseByLine). */
  1.1417 +	{
  1.1418 +	if (aTextOrganisation != EOrganiseByParagraph && aLineWrap <= 0)
  1.1419 +	    {
  1.1420 +	    OstTrace0( TRACE_DUMP, CPLAINTEXT_EXPORTASTEXTL, "EExportLineWrapInvalid" );
  1.1421 +	    }
  1.1422 +	__ASSERT_DEBUG(aTextOrganisation == EOrganiseByParagraph || aLineWrap > 0,Panic(EExportLineWrapInvalid));
  1.1423 +	RFs file_session;
  1.1424 +	TInt error = file_session.Connect();
  1.1425 +	if (error == KErrNone)
  1.1426 +		{
  1.1427 +		RFile file;
  1.1428 +		error = file.Replace(file_session,aFileName,EFileStream | EFileWrite | EFileShareExclusive);
  1.1429 +		if (error == KErrNone)
  1.1430 +			{
  1.1431 +			RFileWriteStream output_stream(file);
  1.1432 +			TRAP(error,output_stream.WriteUint16L(EByteOrderMark));
  1.1433 +			if (error == KErrNone)
  1.1434 +				TRAP(error,ExportTextL(0,output_stream,aTextOrganisation,KMaxTInt,DocumentLength(),aLineWrap));
  1.1435 +			output_stream.Close();
  1.1436 +			}
  1.1437 +		file.Close();
  1.1438 +		file_session.Close();
  1.1439 +		}
  1.1440 +	if (error < 0)
  1.1441 +	    {
  1.1442 +	    OstTrace1( TRACE_FATAL, DUP2_CPLAINTEXT_EXPORTASTEXTL, "Leave code=%x", error );
  1.1443 +	    }
  1.1444 +	User::LeaveIfError(error);
  1.1445 +	}
  1.1446 +
  1.1447 +
  1.1448 +
  1.1449 +EXPORT_C void CPlainText::ImportTextL(TInt aPos,RReadStream& aInput,TTextOrganisation aTextOrganisation,
  1.1450 +									  TInt aMaxOutputChars,TInt aMaxInputChars,
  1.1451 +									  TInt* aOutputChars,TInt* aInputChars)
  1.1452 +/** Imports plain text from a stream into this text object.
  1.1453 +
  1.1454 +Translates line feeds in the source text according to the
  1.1455 +aTextOrganisation argument.
  1.1456 +
  1.1457 +@param aPos Document position at which to insert the text. Must be a valid 
  1.1458 +position, or a panic occurs.
  1.1459 +@param aInput Stream from which to read the text.
  1.1460 +@param aTextOrganisation If EOrganiseByLine, a single line feed 
  1.1461 +or a line feed and carriage return is converted into a space character. A 
  1.1462 +line feed which is followed by another line feed is converted into a paragraph
  1.1463 +delimiter. If EOrganiseByParagraph, all line feeds are converted 
  1.1464 +into paragraph delimiters.
  1.1465 +@param aMaxOutputChars The maximum number of characters to write to the text
  1.1466 +object.
  1.1467 +@param aMaxInputChars The maximum number of characters to read from the stream.
  1.1468 +@param aOutputChars  On return, the number of characters written to the text 
  1.1469 +object.
  1.1470 +@param aInputChars  On return, the number of characters read from the stream. */
  1.1471 +    {
  1.1472 +	TImportExportParam param;
  1.1473 +	param.iOrganisation = aTextOrganisation;
  1.1474 +	param.iMaxOutputChars = aMaxOutputChars;
  1.1475 +	param.iMaxInputChars = aMaxInputChars;
  1.1476 +	TImportExportResult result;
  1.1477 +	ImportTextL(aPos,aInput,param,result);
  1.1478 +	if (aOutputChars)
  1.1479 +		*aOutputChars = result.iOutputChars;
  1.1480 +	if (aInputChars)
  1.1481 +		*aInputChars = result.iInputChars;
  1.1482 +	}
  1.1483 +
  1.1484 +
  1.1485 +EXPORT_C void CPlainText::ExportTextL(TInt aPos,RWriteStream& aOutput,TTextOrganisation aTextOrganisation,
  1.1486 +									  TInt aMaxOutputChars,TInt aMaxInputChars,TInt aMaxLineLength,
  1.1487 +									  TInt* aOutputChars,TInt* aInputChars) const
  1.1488 +/**  Writes plain text to a stream, optionally converting it from Unicode
  1.1489 +into a foreign encoding.
  1.1490 +
  1.1491 +@since 6.1
  1.1492 +@param aPos A document position in the source plain text object from which 
  1.1493 +to start reading the text to export.
  1.1494 +@param aOutput  On return, the write stream to which the text is  written.
  1.1495 +@param aParam  Export parameters, including an optional foreign encoding to 
  1.1496 +convert the Unicode text into, a file  server connection, (this is needed for 
  1.1497 +the character conversion - if not specified, one will be created), a line 
  1.1498 +wrap width, and the maximum number of characters to export.
  1.1499 +@param aResult  On return, contains the number of characters read and written. */
  1.1500 +    {
  1.1501 +	TImportExportParam param;
  1.1502 +	param.iOrganisation = aTextOrganisation;
  1.1503 +	param.iMaxOutputChars = aMaxOutputChars;
  1.1504 +	param.iMaxInputChars = aMaxInputChars;
  1.1505 +	param.iMaxLineLength = aMaxLineLength;
  1.1506 +	TImportExportResult result;
  1.1507 +	ExportTextL(aPos,aOutput,param,result);
  1.1508 +	if (aOutputChars)
  1.1509 +		*aOutputChars = result.iOutputChars;
  1.1510 +	if (aInputChars)
  1.1511 +		*aInputChars = result.iInputChars;
  1.1512 +	}
  1.1513 +
  1.1514 +EXPORT_C void CPlainText::ImportTextL(TInt aPos,RReadStream& aInput,
  1.1515 +									  const TImportExportParam& aParam,TImportExportResult& aResult)
  1.1516 +/** Imports plain text from a stream into this text object, optionally 
  1.1517 +converting it from a foreign encoding into Unicode.
  1.1518 +
  1.1519 +@param aPos Document position at which to insert the text. Must be a valid 
  1.1520 +position, or a panic occurs.
  1.1521 +@param aInput Stream from which to read the text.
  1.1522 +@param aParam Import parameters, including the foreign encoding to convert 
  1.1523 +from, whether to guess the foreign encoding and the maximum number of characters 
  1.1524 +to import.
  1.1525 +@param aResult On return, contains the number of characters read and written 
  1.1526 +and the foreign encoding used by the imported text.
  1.1527 +@see CPlainText::TImportExportParam
  1.1528 +@see CPlainText::TImportExportResult */
  1.1529 +	{
  1.1530 +	CBufSeg* buffer = CBufSeg::NewL(512);
  1.1531 +	CleanupStack::PushL(buffer);
  1.1532 +	RBufWriteStream output(*buffer,0);
  1.1533 +	TImportExportParam param = aParam;
  1.1534 +	param.iOutputInternal = TRUE; // force output to internal format
  1.1535 +	TPlainTextReader::TranslateL(param,aResult,output,aInput);
  1.1536 +
  1.1537 +	TInt chars_inserted = 0;
  1.1538 +	while (buffer->Size() > 0)
  1.1539 +		{
  1.1540 +		TPtr8 p8 = buffer->Ptr(0);
  1.1541 +		TInt bytes = p8.Length();
  1.1542 +		TInt chars = bytes / sizeof(TText);
  1.1543 +		TPtrC p((TText*)p8.Ptr(),chars);
  1.1544 +		/*
  1.1545 +		Insert text using the virtual function InsertL to allow derived classes
  1.1546 +		like CRichText to update their attributes.
  1.1547 +		*/
  1.1548 +		InsertL(aPos + chars_inserted,p);
  1.1549 +		buffer->Delete(0,bytes);
  1.1550 +		chars_inserted += chars;
  1.1551 +		}
  1.1552 +
  1.1553 +	CleanupStack::PopAndDestroy();	// buffer;
  1.1554 +	}
  1.1555 +
  1.1556 +EXPORT_C void CPlainText::ExportTextL(TInt aPos,RWriteStream& aOutput,
  1.1557 +									  const TImportExportParam& aParam,TImportExportResult& aResult) const
  1.1558 +/** Writes plain text to a stream, optionally converting it from Unicode into a 
  1.1559 +foreign encoding.
  1.1560 +
  1.1561 +@param aPos A document position in the source plain text object from which 
  1.1562 +to start reading the text to export.
  1.1563 +@param aOutput On return, the write stream to which the text is written.
  1.1564 +@param aParam Export parameters, including an optional foreign encoding to 
  1.1565 +convert the Unicode text into, a file server connection, (this is needed for 
  1.1566 +the character conversion if not specified, one will be created), a line 
  1.1567 +wrap width, and the maximum number of characters to export.
  1.1568 +@param aResult On return, contains the number of characters read and written.
  1.1569 +@see CPlainText::TImportExportParam
  1.1570 +@see CPlainText::TImportExportResult */
  1.1571 +	{
  1.1572 +	TImportExportParam param = aParam;
  1.1573 +	param.iInputInternal = TRUE; // force input to internal format
  1.1574 +	param.iMaxInputChars = Min(param.iMaxInputChars,DocumentLength() - aPos); // ensure final paragraph delimiter is not exported
  1.1575 +	RBufReadStream input(*iByteStore,aPos);
  1.1576 +	TPlainTextWriter::TranslateL(param,aResult,aOutput,input);
  1.1577 +	}
  1.1578 +
  1.1579 +
  1.1580 +EXPORT_C TPtrC CPlainText::Read(TInt aStartPos)const
  1.1581 +/** Gets a read-only view of a portion of the text object.
  1.1582 +
  1.1583 +The extent of the view is the range of characters starting at aStartPos and ending at whichever of the
  1.1584 +following document positions is reached first:
  1.1585 +
  1.1586 +- The end of the document. 
  1.1587 +- The end of the segment if using segmented storage (the storage type is specified in the NewL(). 
  1.1588 +
  1.1589 +Therefore, when using a segmented buffer to store the document, the length of the resultant view may be
  1.1590 +less than the requested length. In this case multiple calls to Read() may be necessary.
  1.1591 +
  1.1592 +@param aStartPos The document position at which to begin reading. Must be valid or a panic occurs.
  1.1593 +@return Constant pointer to a section of the text object. 
  1.1594 +*/	
  1.1595 +    {
  1.1596 +	__TEST_INVARIANT;
  1.1597 +	if (aStartPos<0 || aStartPos>DocumentLength())
  1.1598 +	    {
  1.1599 +	    OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_READ, "ECharPosBeyondDocument" );
  1.1600 +	    }
  1.1601 +	__ASSERT_ALWAYS(aStartPos>=0 && aStartPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1602 +	
  1.1603 +	TPtr8 buf=iByteStore->Ptr(aStartPos*sizeof(TText));
  1.1604 +	return TPtrC((TText*)buf.Ptr(),buf.Length()/sizeof(TText));
  1.1605 +	}
  1.1606 +
  1.1607 +
  1.1608 +EXPORT_C TPtrC CPlainText::Read(TInt aStartPos,TInt aLength)const
  1.1609 + /** Gets a read-only view of a portion of the text object.
  1.1610 +
  1.1611 +The extent of the view is the range of characters starting at aStartPos and ending at whichever of the
  1.1612 +following document positions is reached first:
  1.1613 +
  1.1614 +- The end of the document.
  1.1615 +- The end of the segment if using segmented storage (the storage type is specified in the NewL().
  1.1616 +- The sum of aStartPos and (aLength-1).
  1.1617 +
  1.1618 +Therefore, when using a segmented buffer to store the document, the length of the resultant view may be
  1.1619 +less than the requested length. In this case multiple calls to Read() may be necessary.
  1.1620 +
  1.1621 +@param aStartPos The document position at which to begin reading. Must be valid or a panic occurs. 
  1.1622 +@param aLength The number of characters to read, inclusive of the character at position aStartPos. 
  1.1623 +@return Constant pointer to a section of the text object.
  1.1624 +*/
  1.1625 +	{
  1.1626 +	__TEST_INVARIANT;
  1.1627 +	if (aStartPos<0 || aStartPos>DocumentLength())
  1.1628 +	    {
  1.1629 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_READ, "ECharPosBeyondDocument" );
  1.1630 +	    }
  1.1631 +	__ASSERT_ALWAYS(aStartPos>=0 && aStartPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1632 +	
  1.1633 +	TPtr8 buf=iByteStore->Ptr(aStartPos*sizeof(TText));
  1.1634 +	TInt length=Min(aLength,((TInt)buf.Length()/sizeof(TText)));
  1.1635 +	return TPtrC((TText*)buf.Ptr(),length);
  1.1636 +	}
  1.1637 +
  1.1638 +
  1.1639 +EXPORT_C void CPlainText::Extract(TDes& aBuf,TInt aPos)const
  1.1640 +/**  Copies the contents of the text object into a descriptor overloaded
  1.1641 +function. The function copies all characters from and including the document
  1.1642 +position specified, to the end of the document or the end of the range
  1.1643 +of characters, if specified.
  1.1644 +
  1.1645 +The buffer's maximum length must be greater than or equal to the number 
  1.1646 +of characters to extract, or a panic occurs.
  1.1647 +
  1.1648 +@param aBuf A buffer; on return contains the extracted text.
  1.1649 +@param aPos The document position from which to copy. Must be valid or a 
  1.1650 +panic occurs.*/
  1.1651 +    {
  1.1652 +	__TEST_INVARIANT;
  1.1653 +	TInt documentLength=DocumentLength();
  1.1654 +	if (aPos<0 || aPos>documentLength)
  1.1655 +	    {
  1.1656 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_EXTRACT, "ECharPosBeyondDocument" );
  1.1657 +	    }
  1.1658 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=documentLength,Panic(ECharPosBeyondDocument));
  1.1659 +	if (aBuf.MaxLength()<documentLength - aPos)
  1.1660 +	    {
  1.1661 +	    OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_EXTRACT, "EExtractBufferTooSmall" );
  1.1662 +	    }
  1.1663 +	__ASSERT_ALWAYS(aBuf.MaxLength()>=documentLength - aPos,Panic(EExtractBufferTooSmall));
  1.1664 +
  1.1665 +	DoExtract(aBuf,aPos,documentLength-aPos);
  1.1666 +	}
  1.1667 +
  1.1668 +
  1.1669 +EXPORT_C void CPlainText::Extract(TDes& aBuf,TInt aPos,TInt aLength)const
  1.1670 +/** Copies the contents of the text object into a descriptor-overloaded function. 
  1.1671 +The function copies all characters from and including the document position 
  1.1672 +specified, to the end of the document or the end of the range of characters, 
  1.1673 +if specified.
  1.1674 +
  1.1675 +The buffer's maximum length must be greater than or equal to aLength, or a 
  1.1676 +panic ocurs.
  1.1677 +
  1.1678 +@param aBuf A buffer; on return contains the extracted text. 
  1.1679 +@param aPos The document position from which to copy. Must be valid or a panic 
  1.1680 +occurs. 
  1.1681 +@param aLength The number of characters to copy. */
  1.1682 +	{
  1.1683 +	__TEST_INVARIANT;
  1.1684 +	if (aPos<0 || aPos>DocumentLength())
  1.1685 +	    {
  1.1686 +	    OstTrace0( TRACE_FATAL, DUP2_CPLAINTEXT_EXTRACT, "ECharPosBeyondDocument" );
  1.1687 +	    }
  1.1688 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1689 +	if (aBuf.MaxLength()<aLength)
  1.1690 +	    {
  1.1691 +	    OstTrace0( TRACE_FATAL, DUP3_CPLAINTEXT_EXTRACT, "EExtractBufferTooSmall" );
  1.1692 +	    }
  1.1693 +	__ASSERT_ALWAYS(aBuf.MaxLength()>=aLength,Panic(EExtractBufferTooSmall));
  1.1694 +
  1.1695 +	DoExtract(aBuf,aPos,aLength);
  1.1696 +	}
  1.1697 +
  1.1698 +// Extract text, optionally discarding some characters such as control characters and soft hyphens or
  1.1699 +// inline text, depending on the flag.
  1.1700 +EXPORT_C void CPlainText::ExtractSelectively(TDes& aBuf,TInt aPos,TInt aLength,TUint aFlags)
  1.1701 +	{
  1.1702 +	if (aPos < 0 || aPos > DocumentLength())
  1.1703 +	    {
  1.1704 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_EXTRACTSELECTIVELY, "ECharPosBeyondDocument" );
  1.1705 +	    }
  1.1706 +	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1707 +	DoExtract(aBuf,aPos,aLength,aFlags);
  1.1708 +	}
  1.1709 +
  1.1710 +
  1.1711 +void CPlainText::DoExtract(TDes& aBuf,TInt aPos,TInt aLength,TUint aFlags) const
  1.1712 +/** Copies aLength characters, commencing at aPos, into the specified
  1.1713 +buffer aBuf.  If aLength is greater than the amount of data left,
  1.1714 +then all remaining characters are written.*/
  1.1715 +	{
  1.1716 +	aBuf.SetLength(0);
  1.1717 +	TInt remainingLength=Min(aLength,DocumentLength()-aPos);
  1.1718 +	TInt startPos=aPos;
  1.1719 +	FOREVER
  1.1720 +		{
  1.1721 +		TPtrC textRead=Read(startPos);
  1.1722 +		TInt lengthRead=textRead.Length();
  1.1723 +		if (lengthRead>remainingLength)
  1.1724 +			lengthRead=remainingLength;
  1.1725 +		
  1.1726 +		// Remove specific characters
  1.1727 +		if (aFlags & EExtractVisible)
  1.1728 +			{
  1.1729 +			const TText* p = textRead.Ptr();
  1.1730 +			const TText* q = p + lengthRead;
  1.1731 +			while (p < q)
  1.1732 +				{
  1.1733 +				TChar c(*p++);
  1.1734 +				if (c == EParagraphDelimiter || c == ELineBreak || c == ETabCharacter)
  1.1735 +					aBuf.Append(' ');
  1.1736 +				else if (c != EPotentialHyphen && c.IsPrint())
  1.1737 +					aBuf.Append(c);
  1.1738 +				}
  1.1739 +			}
  1.1740 +			
  1.1741 +		// Remove inline text from the specified section
  1.1742 +		else if (aFlags & EExcludeInlineEditedText)
  1.1743 +			{
  1.1744 +			const TInt inlineTextPos = GetPositionOfInlineTextInDocument();
  1.1745 +			const TText* p = textRead.Ptr();
  1.1746 +			
  1.1747 +			if (inlineTextPos != KErrNotFound)
  1.1748 +				{
  1.1749 +				for (TInt i=aPos; i<(aPos+lengthRead); i++)
  1.1750 +					{
  1.1751 +		    		if (!((i >= inlineTextPos) && (i <= (inlineTextPos + GetLengthOfInlineText() - 1))))
  1.1752 +		    			{
  1.1753 +						TChar c(*p++);
  1.1754 +						aBuf.Append(c);
  1.1755 +						continue;
  1.1756 +		    			}
  1.1757 +		    		++p;
  1.1758 +					}
  1.1759 +				}
  1.1760 +			else
  1.1761 +				aBuf.Append(textRead.Ptr(),lengthRead);
  1.1762 +			}
  1.1763 +		else
  1.1764 +			aBuf.Append(textRead.Ptr(),lengthRead);
  1.1765 +		remainingLength-=lengthRead;
  1.1766 +		if (remainingLength==0)
  1.1767 +			return;
  1.1768 +		startPos+=lengthRead;
  1.1769 +		}
  1.1770 +	}
  1.1771 +
  1.1772 +
  1.1773 +
  1.1774 +EXPORT_C void CPlainText::SetPageTable(TPageTable* aPageTable)
  1.1775 +/** Links the text object to a page table. A page table is an array of integers; 
  1.1776 +each integer represents the number of characters on a page. It is required 
  1.1777 +for pagination. The page table is updated when changes are made to the document, 
  1.1778 +e.g. after pasting from the clipboard, inserting, importing or deleting text.
  1.1779 +
  1.1780 +The text object does not take ownership of the page table specified.
  1.1781 +
  1.1782 +@param aPageTable The page table to be referenced by the text object. */
  1.1783 +	{
  1.1784 +	iPageTable=aPageTable;
  1.1785 +	SetHasChanged(ETrue);
  1.1786 +	}
  1.1787 +
  1.1788 +
  1.1789 +void CPlainText::UpdatePageTable(TInt aPos,TInt aLength)
  1.1790 +// Adds aLength number of characters to the page in the page table
  1.1791 +// specified by the character position aPos.
  1.1792 +// Called from i) Insert  ii) Delete  iii) Paste from clipboard  iv) Text file import.
  1.1793 +//
  1.1794 +	{
  1.1795 +	__TEST_INVARIANT;
  1.1796 +	if (aPos<0 || aPos>DocumentLength()+1)
  1.1797 +	    {
  1.1798 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_UPDATEPAGETABLE, "ECharPosBeyondDocument" );
  1.1799 +	    }
  1.1800 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength()+1,Panic(ECharPosBeyondDocument));
  1.1801 +	
  1.1802 +	if (iPageTable)
  1.1803 +		(*iPageTable)[PageContainingPos(aPos)]+=aLength;
  1.1804 +	}
  1.1805 +
  1.1806 +
  1.1807 +EXPORT_C TInt CPlainText::PageContainingPos(TInt aPos)const
  1.1808 +/** Gets the number of the page which contains the specified document position. 
  1.1809 +If no page table has been set up, the function returns a value of zero. Use 
  1.1810 +SetPageTable() to set up the page table.
  1.1811 +
  1.1812 +@param aPos A document position. Must be valid or a panic occurs. 
  1.1813 +@return The page number containing document position aPos. */
  1.1814 +    {
  1.1815 +	__TEST_INVARIANT;
  1.1816 +	if (aPos<0 || aPos>DocumentLength())
  1.1817 +	    {
  1.1818 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_PAGECONTAININGPOS, "ECharPosBeyondDocument" );
  1.1819 +	    }
  1.1820 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1821 +
  1.1822 +    if (!iPageTable || (iPageTable->Count()<1))
  1.1823 +        return 0;
  1.1824 +    else
  1.1825 +        {
  1.1826 +        TInt pageOffset=0;
  1.1827 +        TInt charCount=(*iPageTable)[pageOffset];
  1.1828 +        while (charCount<=aPos)
  1.1829 +            {
  1.1830 +            pageOffset++;
  1.1831 +            charCount+=(*iPageTable)[pageOffset];
  1.1832 +            }
  1.1833 +        return pageOffset;
  1.1834 +        }
  1.1835 +    }
  1.1836 +
  1.1837 +EXPORT_C TEtextComponentInfo CPlainText::ComponentInfo()const
  1.1838 +/** Gets information about the number of components contained in the text object. 
  1.1839 +For plain text, only the field count has relevance.
  1.1840 +
  1.1841 +@return Contains the field count. */
  1.1842 +	{return TEtextComponentInfo(FieldCount(),0,0);}
  1.1843 +
  1.1844 +
  1.1845 +
  1.1846 +EXPORT_C TInt CPlainText::FieldCount() const
  1.1847 +/** Gets a count of the number of fields in the text object's field set.
  1.1848 +
  1.1849 +@return The number of fields in the field set. */
  1.1850 +	{
  1.1851 +	__TEST_INVARIANT;
  1.1852 +
  1.1853 +	return (FieldSetPresent())
  1.1854 +		? iFieldSet->FieldCount()
  1.1855 +		: 0;
  1.1856 +	}
  1.1857 +
  1.1858 +EXPORT_C void CPlainText::InsertFieldL(TInt aPos,CTextField* aField,TUid aFieldType)
  1.1859 +/** Inserts a field into the text object at a specified document position. creating a zero-length field record.
  1.1860 + Maintain the field set.
  1.1861 +
  1.1862 +Note: After insertion, the field should be evaluated in order to make its contents 
  1.1863 +visible; use UpdateFieldL().
  1.1864 +
  1.1865 +@param aPos The document position at which to insert the field. Must be valid, 
  1.1866 +or a panic occurs. 
  1.1867 +@param aField The field to insert, created by NewTextFieldL(). Must not be 
  1.1868 +NULL, or a panic occurs. 
  1.1869 +@param aFieldType Identifies the type of field to insert. For the built in 
  1.1870 +field types, see the UID values defined in flddef.h. */
  1.1871 +	{
  1.1872 +	__TEST_INVARIANT;
  1.1873 +	if (aPos<0 || aPos>DocumentLength())
  1.1874 +	    {
  1.1875 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_INSERTFIELDL, "ECharPosBeyondDocument" );
  1.1876 +	    }
  1.1877 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1878 +	if (!aField)
  1.1879 +	    {
  1.1880 +	    OstTrace0( TRACE_FATAL, DUP1_CPLAINTEXT_INSERTFIELDL, "ENoTextField" );
  1.1881 +	    }
  1.1882 +	__ASSERT_ALWAYS(aField,Panic(ENoTextField));
  1.1883 +
  1.1884 +	if (!FieldSetPresent())
  1.1885 +		CreateFieldSetL(DocumentLength());
  1.1886 +	iFieldSet->InsertFieldL(aPos,aField,aFieldType);
  1.1887 +	SetHasChanged(ETrue);
  1.1888 +
  1.1889 +	__TEST_INVARIANT;
  1.1890 +	}
  1.1891 +
  1.1892 +EXPORT_C void CPlainText::UpdateFieldL(TInt aPos)
  1.1893 +/** Re-evaluates the field which covers the document position specified. Re-evaluating 
  1.1894 +a field means calculating the field's new value, then inserting that value 
  1.1895 +into the text object, replacing the previous value.
  1.1896 +
  1.1897 +Notes:
  1.1898 +
  1.1899 +fields have a maximum length of 20 characters.
  1.1900 +
  1.1901 +the first time a field is updated, the position specified should be the position 
  1.1902 +at which the field was inserted.
  1.1903 +
  1.1904 +@param aPos A document position in the field to be updated. Must be a valid 
  1.1905 +position, or a panic occurs. */
  1.1906 +	{
  1.1907 +	__TEST_INVARIANT;
  1.1908 +	if (aPos<0 || aPos>DocumentLength())
  1.1909 +	    {
  1.1910 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_UPDATEFIELDL, "ECharPosBeyondDocument" );
  1.1911 +	    }
  1.1912 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1913 +
  1.1914 +	if (!FieldSetPresent())
  1.1915 +		return;
  1.1916 +	TFindFieldInfo info;
  1.1917 +	if (iFieldSet->FindFields(info,aPos))
  1.1918 +		{// a field exists at aPos, so update it.
  1.1919 +		HBufC* valueBuf=HBufC::NewL(KMaxFieldBufferSize); // will hold the new value,max length 20
  1.1920 +		/*
  1.1921 +		Don't put valueBuf on the cleanup stack before calling NewFieldValueL because NewFieldValueL
  1.1922 +		sometimes changes valueBuf, in which case it itself puts it on the cleanup stack.
  1.1923 +		*/
  1.1924 +		iFieldSet->NewFieldValueL(valueBuf,info.iFirstFieldPos);  // get the new value
  1.1925 +		CleanupStack::PushL(valueBuf);
  1.1926 +		DoPtInsertL(info.iFirstFieldPos,*valueBuf);  // insert the new text into the document
  1.1927 +		DoPtDelete(info.iFirstFieldPos+valueBuf->Length(),info.iFirstFieldLen);  // delete the old text of the field
  1.1928 +		iFieldSet->NotifyFieldUpdate(aPos,valueBuf->Length());  // inform the field set
  1.1929 +		CleanupStack::PopAndDestroy();
  1.1930 +		}
  1.1931 +	SetHasChanged(ETrue);
  1.1932 +
  1.1933 +	__TEST_INVARIANT;
  1.1934 +	}
  1.1935 +
  1.1936 +
  1.1937 +
  1.1938 +EXPORT_C void CPlainText::UpdateAllFieldsL()
  1.1939 +/** Re-evaluates all of the fields in the text object. Re-evaluating a field means 
  1.1940 +calculating the field's new value, then inserting that value into the text 
  1.1941 +object, replacing the previous value.
  1.1942 +
  1.1943 +Note: Fields have a maximum length of 20 characters. */
  1.1944 +	{
  1.1945 +	__TEST_INVARIANT;
  1.1946 +
  1.1947 +	TInt numFields=FieldCount();
  1.1948 +	TInt pos=0;
  1.1949 +	TFindFieldInfo info;
  1.1950 +	for (TInt item=0;item<numFields;item++)
  1.1951 +		{
  1.1952 +		// find the next field and go to its beginning
  1.1953 +		iFieldSet->FindFields(info,pos,DocumentLength()-pos);
  1.1954 +		pos=info.iFirstFieldPos;
  1.1955 +		UpdateFieldL(pos);
  1.1956 +		// the field has changed, find out its new length and go to the end of it
  1.1957 +		iFieldSet->FindFields(info,pos);
  1.1958 +		pos+=info.iFirstFieldLen;
  1.1959 +		}
  1.1960 +
  1.1961 +	__TEST_INVARIANT;
  1.1962 +	}
  1.1963 +
  1.1964 +
  1.1965 +
  1.1966 +EXPORT_C TBool CPlainText::RemoveField(TInt aPos)
  1.1967 +/** Removes the field covering the document position specified from the field set, 
  1.1968 +and deletes all text content associated with the field from the text object.
  1.1969 +
  1.1970 +@param aPos A document position within the field to be deleted. Must be a 
  1.1971 +valid position or a panic occurs. 
  1.1972 +@return ETrue if a field was located at aPos. EFalse if no field was located 
  1.1973 +at aPos. */
  1.1974 +	{
  1.1975 +	__TEST_INVARIANT;
  1.1976 +	if (aPos<0 || aPos>DocumentLength())
  1.1977 +	    {
  1.1978 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_REMOVEFIELD, "ECharPosBeyondDocument" );
  1.1979 +	    }
  1.1980 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.1981 +
  1.1982 +	TBool fieldRemoved=EFalse;
  1.1983 +	
  1.1984 +	if (FieldSetPresent())
  1.1985 +		{
  1.1986 +		TFindFieldInfo info;
  1.1987 +		if (iFieldSet->FindFields(info,aPos))
  1.1988 +			{// remove the field's record & associated text
  1.1989 +			Delete(info.iFirstFieldPos,info.iFirstFieldLen);
  1.1990 +			iFieldSet->RemoveField(aPos);
  1.1991 +			fieldRemoved=ETrue;
  1.1992 +			}
  1.1993 +		}
  1.1994 +	if (fieldRemoved)
  1.1995 +		SetHasChanged(ETrue);
  1.1996 +	
  1.1997 +	__TEST_INVARIANT;
  1.1998 +	return fieldRemoved;
  1.1999 +	}
  1.2000 +
  1.2001 +
  1.2002 +
  1.2003 +EXPORT_C TBool CPlainText::FindFields(TInt aPos) const
  1.2004 +	{
  1.2005 +	__TEST_INVARIANT;
  1.2006 +
  1.2007 +	if (FieldSetPresent())
  1.2008 +		return iFieldSet->FindFields(aPos);
  1.2009 +	else
  1.2010 +		return EFalse;
  1.2011 +	}
  1.2012 +
  1.2013 +
  1.2014 +EXPORT_C TBool CPlainText::FindFields(TFindFieldInfo& aInfo,TInt aPos,TInt aRange)const
  1.2015 +/** Tests whether a document position is located within a field overloaded
  1.2016 +function.
  1.2017 +
  1.2018 +The second overload extracts information about all fields located
  1.2019 +within a range of characters.
  1.2020 +
  1.2021 +@param aPos The document position. Must be valid or a panic occurs.
  1.2022 +@param aInfo  On return, contains the number of fields fully or partially 
  1.2023 +within the range of characters specified and the start position and length 
  1.2024 +of the first field found.
  1.2025 +@param   aRange The number of characters to search, beginning at aPos. Must 
  1.2026 +not be negative, or a panic occurs. The sum of aPos and aRange must be less 
  1.2027 +than the document length, or a panic occurs. The default range is zero. In 
  1.2028 +this case the function just returns information about the single field located 
  1.2029 +at the position specified.
  1.2030 +@return ETrue if aPos is located within a field, or if there were one or more 
  1.2031 +fields found in the range. EFalse if not. */
  1.2032 +	{
  1.2033 +	__TEST_INVARIANT;
  1.2034 +
  1.2035 +	if (FieldSetPresent())
  1.2036 +		return iFieldSet->FindFields(aInfo,aPos,aRange);
  1.2037 +	else
  1.2038 +		return EFalse;
  1.2039 +	}
  1.2040 +
  1.2041 +EXPORT_C const CTextField* CPlainText::TextField(TInt aPos)const
  1.2042 +/** Returns a pointer to the field located at the specified document position.
  1.2043 +
  1.2044 +@param aPos A document position within the field. Must be a valid document 
  1.2045 +position, or a panic occurs. 
  1.2046 +@return Pointer to the field which covers position aPos. NULL if there is no 
  1.2047 +field at the specified position. */
  1.2048 +	{
  1.2049 +	__TEST_INVARIANT;
  1.2050 +
  1.2051 +	return (FieldSetPresent())
  1.2052 +		? iFieldSet->TextField(aPos)
  1.2053 +		: NULL;
  1.2054 +	}
  1.2055 +
  1.2056 +EXPORT_C TBool CPlainText::ConvertFieldToText(TInt aPos)
  1.2057 +/** Converts the field containing the specified document position into text, leaving 
  1.2058 +its current value in the text object. Does not update the field beforehand.
  1.2059 +
  1.2060 +@param aPos A document position within the field to convert. Must be a valid 
  1.2061 +position or a panic occurs. 
  1.2062 +@return ETrue if there was a field located at aPos. EFalse if there was no 
  1.2063 +field located at aPos. */
  1.2064 +	{
  1.2065 +	__TEST_INVARIANT;
  1.2066 +	if (aPos<0 || aPos>DocumentLength())
  1.2067 +	    {
  1.2068 +	    OstTrace0( TRACE_FATAL, CPLAINTEXT_CONVERTFIELDTOTEXT, "ECharPosBeyondDocument" );
  1.2069 +	    }
  1.2070 +	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
  1.2071 +
  1.2072 +	TBool fieldConverted=EFalse;
  1.2073 +	
  1.2074 +	if (FieldSetPresent())
  1.2075 +		{
  1.2076 +		TFindFieldInfo info;
  1.2077 +		iFieldSet->FindFields(info,aPos);
  1.2078 +		if (iFieldSet->RemoveField(aPos))
  1.2079 +			fieldConverted=ETrue;
  1.2080 +		}
  1.2081 +	if (fieldConverted)
  1.2082 +		SetHasChanged(ETrue);
  1.2083 +	
  1.2084 +	__TEST_INVARIANT;
  1.2085 +	return fieldConverted;
  1.2086 +	}
  1.2087 +
  1.2088 +
  1.2089 +EXPORT_C void CPlainText::ConvertAllFieldsToText()
  1.2090 +/** Removes all fields from the text object's field set, leaving their current 
  1.2091 +text value in the text object. Does not update the fields beforehand. */
  1.2092 +	{
  1.2093 +	__TEST_INVARIANT;
  1.2094 +
  1.2095 +	TInt fieldCount=FieldCount();
  1.2096 +	TInt pos=0;
  1.2097 +	for (TInt item=0;item<fieldCount;item++)
  1.2098 +		{// Convert each field in turn.
  1.2099 +		TFindFieldInfo info;
  1.2100 +		iFieldSet->FindFields(info,pos);  // find the next field and go to its beginning
  1.2101 +		pos=info.iFirstFieldPos;
  1.2102 +		ConvertFieldToText(pos);
  1.2103 +		}
  1.2104 +
  1.2105 +	__TEST_INVARIANT;
  1.2106 +	}
  1.2107 +
  1.2108 +EXPORT_C void CPlainText::SetFieldFactory(MTextFieldFactory* aFactory)
  1.2109 +/** Sets up a field factory. A field factory is an instance of a class deriving 
  1.2110 +from MTextFieldFactory. It must implement a NewFieldL() function to create 
  1.2111 +and return fields of the desired type. The field factory's NewFieldL() function 
  1.2112 +is called by CPlainText::NewTextFieldL(). A field factory must be set up if 
  1.2113 +you intend to add any fields to the text object.
  1.2114 +
  1.2115 +@param aFactory The field factory. */
  1.2116 +	{
  1.2117 +	// Set the FieldSet's header factory handle (this points to the built-in factory by default)
  1.2118 +	//+ This function has no way to report failure, which can happen if CreateFieldSetL leaves. It ought to leave.
  1.2119 +	
  1.2120 +	// Create field set if necessary & set the factory
  1.2121 +	int error = 0;
  1.2122 +	if (!FieldSetPresent())
  1.2123 +		TRAP(error,CreateFieldSetL(DocumentLength()));
  1.2124 +	if (!error)
  1.2125 +		iFieldSet->SetFieldFactory(aFactory);
  1.2126 +	}
  1.2127 +
  1.2128 +
  1.2129 +
  1.2130 +EXPORT_C CTextField* CPlainText::NewTextFieldL(TUid aFieldType)const
  1.2131 +/** Creates and returns a new field. Before calling this function, a field 
  1.2132 +factory should have been set up, either by calling SetFieldFactory(), or by 
  1.2133 +specifying one in the NewL(). The field factory's NewFieldL() function is 
  1.2134 +called to create a field of the type specified in the argument. A NULL field 
  1.2135 +is returned if no field factory has been set up.
  1.2136 +
  1.2137 +@param aFieldType Identifies the field type. 
  1.2138 +@return Pointer to the new text field. NULL if no field factory has been set 
  1.2139 +up. */
  1.2140 +	{
  1.2141 +	__TEST_INVARIANT;
  1.2142 +
  1.2143 +	CTextField* field=NULL;
  1.2144 +	if (FieldSetPresent())
  1.2145 +		field=iFieldSet->NewFieldL(aFieldType);
  1.2146 +
  1.2147 +	__TEST_INVARIANT;
  1.2148 +	return field;
  1.2149 +	}
  1.2150 +
  1.2151 +void CPlainText::CreateFieldSetL(TInt aDocumentLength)
  1.2152 +	{
  1.2153 +	iFieldSet = CTextFieldSet::NewL(aDocumentLength);
  1.2154 +	}
  1.2155 +
  1.2156 +EXPORT_C const MTextFieldFactory* CPlainText::FieldFactory()const
  1.2157 +/** Gets a pointer to the field factory used by the text object. The field factory 
  1.2158 +may be set up using SetFieldFactory(), or may be specified in the NewL().
  1.2159 +
  1.2160 +@return The field factory. NULL if no field factory has been set up. */
  1.2161 +	{return (FieldSetPresent()) ? iFieldSet->FieldFactory() : NULL;}
  1.2162 +
  1.2163 +CEditableTextOptionalData::CEditableTextOptionalData()
  1.2164 +	{
  1.2165 +	// do nothing
  1.2166 +	}
  1.2167 +
  1.2168 +CEditableTextOptionalData::~CEditableTextOptionalData()
  1.2169 +	{
  1.2170 +	delete iInlineEditData;
  1.2171 +	}