os/textandloc/textrendering/texthandling/stext/TXTRICH.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description: 
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
sl@0
    18
sl@0
    19
#include <e32std.h>
sl@0
    20
#include <e32base.h>
sl@0
    21
sl@0
    22
#include "TXTRICH.H"
sl@0
    23
#include "TXTINDEX.H"
sl@0
    24
#include "TXTSTD.H"
sl@0
    25
#include "TXTRTPFL.H"
sl@0
    26
#include "ParseLst.h"
sl@0
    27
#include "TXTCLIPBOARD.H"
sl@0
    28
sl@0
    29
#include "OstTraceDefinitions.h"
sl@0
    30
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    31
#include "TXTRICHTraces.h"
sl@0
    32
#endif
sl@0
    33
sl@0
    34
#ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
sl@0
    35
#include "TXTETEXT_INTERNAL.H"
sl@0
    36
#include "TXTRICH_INTERNAL.H"
sl@0
    37
#endif
sl@0
    38
sl@0
    39
EXPORT_C void CRichText::__DbgTestInvariant()const
sl@0
    40
// Provides class invariants.  Explanations below:
sl@0
    41
//
sl@0
    42
	{
sl@0
    43
#ifdef _DEBUG
sl@0
    44
// ASSERT: The global format layers are never null.
sl@0
    45
	if (iGlobalParaFormatLayer == NULL)
sl@0
    46
	    {
sl@0
    47
	    OstTrace0( TRACE_DUMP, CRICHTEXT_DBGTESTINVARIANT, "Invariant" );
sl@0
    48
	    }
sl@0
    49
	__ASSERT_DEBUG(iGlobalParaFormatLayer != NULL, User::Invariant());
sl@0
    50
	if (iGlobalCharFormatLayer == NULL)
sl@0
    51
	    {
sl@0
    52
	    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXT_DBGTESTINVARIANT, "Invariant" );
sl@0
    53
	    }
sl@0
    54
	__ASSERT_DEBUG(iGlobalCharFormatLayer != NULL, User::Invariant());
sl@0
    55
	if (IndexPresent())
sl@0
    56
		{
sl@0
    57
// ASSERT: The sum of para lengths == the length as described by the document storage.
sl@0
    58
		TInt cumulativeParaLength = 0;
sl@0
    59
		TInt maxPara = iIndex->iParaIx->Count();
sl@0
    60
		for (TInt offset = 0; offset < maxPara; offset++)
sl@0
    61
			{
sl@0
    62
			TParaAttribsEntry entry = (*iIndex->iParaIx)[offset];
sl@0
    63
			cumulativeParaLength += entry.iLength;
sl@0
    64
			}
sl@0
    65
		if (cumulativeParaLength != (DocumentLength() + 1))
sl@0
    66
		    {
sl@0
    67
		    OstTrace0( TRACE_DUMP, DUP2_CRICHTEXT_DBGTESTINVARIANT, "Invariant" );
sl@0
    68
		    }
sl@0
    69
		__ASSERT_DEBUG(cumulativeParaLength == (DocumentLength() + 1), User::Invariant());
sl@0
    70
		}
sl@0
    71
		// Change here for defect INC005336.
sl@0
    72
		// This defect is present when the assertion below fails.
sl@0
    73
        if ((iParserData != NULL) &&
sl@0
    74
                (iParserData->HaveRange()) &&
sl@0
    75
                (iParserData->EndParse() > DocumentLength()))
sl@0
    76
            {
sl@0
    77
            OstTrace0( TRACE_DUMP, DUP3_CRICHTEXT_DBGTESTINVARIANT, "Invariant" );
sl@0
    78
            }
sl@0
    79
		__ASSERT_DEBUG( (iParserData == NULL) || \
sl@0
    80
			(!iParserData->HaveRange()) || \
sl@0
    81
			(iParserData->EndParse() <= DocumentLength()), User::Invariant());
sl@0
    82
#endif
sl@0
    83
	}
sl@0
    84
sl@0
    85
EXPORT_C CRichText* CRichText::NewL(const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,
sl@0
    86
									TDocumentStorage aStorage,TInt aDefaultTextGranularity,TParaType aParaType)
sl@0
    87
/** Allocates and constructs an empty rich text object, with a global character
sl@0
    88
and paragraph format layer. A single end-of-document delimiter is inserted.
sl@0
    89
No style list is allocated.
sl@0
    90
sl@0
    91
@param aGlobalParaLayer Pointer to the paragraph format layer referenced by
sl@0
    92
the rich text object. Must not be NULL, or a panic occurs.
sl@0
    93
@param aGlobalCharLayer Pointer to the character format layer referenced by
sl@0
    94
the rich text object. Must not be NULL, or a panic occurs.
sl@0
    95
@param aStorage The type of in-memory buffer to use. Defaults to ESegmentedStorage
sl@0
    96
which should rarely need to be changed.
sl@0
    97
@param aDefaultTextGranularity Specifies the granularity of the in-memory buffer.
sl@0
    98
Default is EDefaultTextGranularity bytes (=256), and this should rarely need
sl@0
    99
to be changed.
sl@0
   100
@param aParaType This argument indicates whether you are using a single paragraph
sl@0
   101
or multiple paragraphs, and thus affects the granularity of aggregate objects
sl@0
   102
used internally for storing paragraph attributes. Default = EMultiPara.
sl@0
   103
@return The rich text object. */
sl@0
   104
	{
sl@0
   105
	// Create new rich text containing just a single end-of-document character.
sl@0
   106
	if (aGlobalParaLayer == NULL)
sl@0
   107
	    {
sl@0
   108
	    OstTrace0( TRACE_FATAL, CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   109
	    }
sl@0
   110
	__ASSERT_ALWAYS(aGlobalParaLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   111
	if (aGlobalCharLayer == NULL)
sl@0
   112
	    {
sl@0
   113
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   114
	    }
sl@0
   115
	__ASSERT_ALWAYS(aGlobalCharLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   116
sl@0
   117
	CRichText* self = new(ELeave) CRichText(aGlobalParaLayer, aGlobalCharLayer);
sl@0
   118
	CleanupStack::PushL(self);
sl@0
   119
	self->ConstructL(aStorage, aDefaultTextGranularity, aParaType);
sl@0
   120
	CleanupStack::Pop();
sl@0
   121
	return self;
sl@0
   122
	}
sl@0
   123
sl@0
   124
sl@0
   125
// Create new rich text that supports Paragraph Styles, containing just a single end-of-document character.
sl@0
   126
EXPORT_C CRichText* CRichText::NewL(const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,
sl@0
   127
									const CStyleList& aStyleList,
sl@0
   128
									TDocumentStorage aStorage,TInt aDefaultTextGranularity,TParaType aParaType)
sl@0
   129
/** Allocates and constructs an empty rich text object which supports styles. It
sl@0
   130
is constructed with a global character and paragraph format layer and a style
sl@0
   131
list. A single end-of-document delimiter is inserted. The rich text object
sl@0
   132
takes ownership of the style list.
sl@0
   133
sl@0
   134
Note:
sl@0
   135
sl@0
   136
A rich text object not constructed with a style list may still use styles,
sl@0
   137
by calling SetStyleListExternallyOwned() at any time after construction. In
sl@0
   138
this case, the rich text object does not own the style list.
sl@0
   139
sl@0
   140
@param aGlobalParaLayer Pointer to the paragraph format layer referenced by
sl@0
   141
the rich text object. Must not be NULL, or a panic occurs.
sl@0
   142
@param aGlobalCharLayer Pointer to the character format layer referenced by
sl@0
   143
the rich text object. Must not be NULL, or a panic occurs.
sl@0
   144
@param aStyleList Style list. Holds the set of paragraph styles which can be
sl@0
   145
used in the rich text object.
sl@0
   146
@param aStorage The type of in-memory buffer to use. Defaults to ESegmentedStorage
sl@0
   147
which should rarely need to be changed.
sl@0
   148
@param aDefaultTextGranularity Specifies the granularity of the in-memory buffer.
sl@0
   149
Default is EDefaultTextGranularity bytes (=256), and this should rarely need
sl@0
   150
to be changed.
sl@0
   151
@param aParaType This argument indicates whether you are using a single paragraph
sl@0
   152
or multiple paragraphs, and thus affects the granularity of aggregate objects
sl@0
   153
used internally for storing paragraph attributes. Default = EMultiPara.
sl@0
   154
@return The new rich text object. */
sl@0
   155
	{
sl@0
   156
	if (aGlobalParaLayer == NULL)
sl@0
   157
	    {
sl@0
   158
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   159
	    }
sl@0
   160
	__ASSERT_ALWAYS(aGlobalParaLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   161
	if (aGlobalCharLayer == NULL)
sl@0
   162
	    {
sl@0
   163
	    OstTrace0( TRACE_FATAL, DUP3_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   164
	    }
sl@0
   165
	__ASSERT_ALWAYS(aGlobalCharLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   166
sl@0
   167
	CRichText* self = new(ELeave) CRichText(aGlobalParaLayer, aGlobalCharLayer, CONST_CAST(CStyleList*, &aStyleList));
sl@0
   168
	CleanupStack::PushL(self);
sl@0
   169
	self->ConstructL(aStorage, aDefaultTextGranularity, aParaType);
sl@0
   170
	CleanupStack::Pop();
sl@0
   171
	return self;
sl@0
   172
	}
sl@0
   173
sl@0
   174
sl@0
   175
// Restore into a new rich text object, using the specified global layers.
sl@0
   176
EXPORT_C CRichText* CRichText::NewL(const CStreamStore& aStore,TStreamId aStreamId,
sl@0
   177
									const CParaFormatLayer* aGlobalParaLayer,const CCharFormatLayer* aGlobalCharLayer,
sl@0
   178
									MTextFieldFactory* aFieldFactory,
sl@0
   179
									TDocumentStorage aStorage)
sl@0
   180
/** Allocates and constructs a rich text object with a field factory. Its text
sl@0
   181
content is internalized from a stream store.
sl@0
   182
sl@0
   183
Note:
sl@0
   184
sl@0
   185
A rich text object not constructed with a field factory may still support
sl@0
   186
the addition of fields, by calling SetFieldFactory(), defined in the base
sl@0
   187
class CPlainText.
sl@0
   188
sl@0
   189
@param aStore Stream store from which the object is restored.
sl@0
   190
@param aStreamId ID of the stream store.
sl@0
   191
@param aGlobalParaLayer Pointer to the paragraph format layer referenced by
sl@0
   192
the rich text object. Must not be NULL, or a panic occurs.
sl@0
   193
@param aGlobalCharLayer Pointer to the character format layer referenced by
sl@0
   194
the rich text object. Must not be NULL, or a panic occurs.
sl@0
   195
@param aFieldFactory Pointer to a field factory. A field factory must be provided
sl@0
   196
if the text object supports fields.
sl@0
   197
@param aStorage The type of in-memory buffer to use. Defaults to ESegmentedStorage
sl@0
   198
which should rarely need to be changed.
sl@0
   199
@return The new rich text object. */
sl@0
   200
	{
sl@0
   201
	if (aGlobalParaLayer == NULL)
sl@0
   202
	    {
sl@0
   203
	    OstTrace0( TRACE_FATAL, DUP4_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   204
	    }
sl@0
   205
	__ASSERT_ALWAYS(aGlobalParaLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   206
	if (aGlobalCharLayer == NULL)
sl@0
   207
	    {
sl@0
   208
	    OstTrace0( TRACE_FATAL, DUP5_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   209
	    }
sl@0
   210
	__ASSERT_ALWAYS(aGlobalCharLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   211
sl@0
   212
	CRichText* self = new(ELeave) CRichText(aGlobalParaLayer, aGlobalCharLayer);
sl@0
   213
	CleanupStack::PushL(self);
sl@0
   214
	self->ConstructL(aStore, aStreamId, NULL, NULL, aFieldFactory, aStorage);
sl@0
   215
	CleanupStack::Pop();
sl@0
   216
	return self;
sl@0
   217
	}
sl@0
   218
sl@0
   219
sl@0
   220
EXPORT_C CRichText* CRichText::NewL(const CStreamStore& aStore, TStreamId aStreamId,
sl@0
   221
									const CParaFormatLayer* aGlobalParaLayer, const CCharFormatLayer* aGlobalCharLayer,
sl@0
   222
									MPictureFactory* aPictureFactory, MRichTextStoreResolver* aStoreResolver,
sl@0
   223
									MTextFieldFactory* aFieldFactory,
sl@0
   224
									TDocumentStorage aStorage)
sl@0
   225
/** Allocates and constructs a rich text object with a field factory and a picture
sl@0
   226
factory. Its text content is internalized from a stream store.
sl@0
   227
sl@0
   228
Note:
sl@0
   229
sl@0
   230
A rich text object not constructed with a field factory may still support
sl@0
   231
the addition of fields, by calling SetFieldFactory(), defined in the base
sl@0
   232
class CPlainText.
sl@0
   233
sl@0
   234
@param aStore Stream store from which the object is restored.
sl@0
   235
@param aStreamId ID of the stream store.
sl@0
   236
@param aGlobalParaLayer Pointer to the paragraph format layer referenced by
sl@0
   237
the rich text object. Must not be NULL, or a panic occurs.
sl@0
   238
@param aGlobalCharLayer Pointer to the character format layer referenced by
sl@0
   239
the rich text object. Must not be NULL, or a panic occurs.
sl@0
   240
@param aPictureFactory The picture factory. This is needed to load pictures
sl@0
   241
into memory, (see PictureHandleL()). If a store resolver is specified (not
sl@0
   242
NULL), then a factory must also be specified, or a panic occurs.
sl@0
   243
@param aStoreResolver A store resolver. This determines which file store the
sl@0
   244
picture is stored in. It is used to get from a reference to an embedded picture
sl@0
   245
within a CRichText object to the actual picture itself. Picture loading is
sl@0
   246
done by the picture factory.
sl@0
   247
@param aFieldFactory Pointer to a field factory. A field factory must be provided
sl@0
   248
if the text object supports fields.
sl@0
   249
@param aStorage The type of in-memory buffer to use. Defaults to ESegmentedStorage
sl@0
   250
which should rarely need to be changed.
sl@0
   251
@return The new rich text object. */
sl@0
   252
	{
sl@0
   253
	// Restore a new rich text from the specified stream, that uses the specified global layers, and the
sl@0
   254
	// specified picture header factory and store, if this rich text supports pictures.
sl@0
   255
	if (!aPictureFactory && aStoreResolver)
sl@0
   256
	    {
sl@0
   257
	    OstTrace0( TRACE_FATAL, DUP6_CRICHTEXT_NEWL, "EInvalidPictureFactorySettings" );
sl@0
   258
	    }
sl@0
   259
	__ASSERT_ALWAYS(!(!aPictureFactory && aStoreResolver), Panic(EInvalidPictureFactorySettings));
sl@0
   260
	if (aGlobalParaLayer == NULL)
sl@0
   261
	    {
sl@0
   262
	    OstTrace0( TRACE_FATAL, DUP7_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   263
	    }
sl@0
   264
	__ASSERT_ALWAYS(aGlobalParaLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   265
	if (aGlobalCharLayer == NULL)
sl@0
   266
	    {
sl@0
   267
	    OstTrace0( TRACE_FATAL, DUP8_CRICHTEXT_NEWL, "ENullFormatLayerHandle" );
sl@0
   268
	    }
sl@0
   269
	__ASSERT_ALWAYS(aGlobalCharLayer != NULL, Panic(ENullFormatLayerHandle));
sl@0
   270
sl@0
   271
	CRichText* self = new(ELeave) CRichText(aGlobalParaLayer, aGlobalCharLayer);
sl@0
   272
	CleanupStack::PushL(self);
sl@0
   273
	self->ConstructL(aStore, aStreamId, aPictureFactory, aStoreResolver, aFieldFactory, aStorage);
sl@0
   274
	CleanupStack::Pop();
sl@0
   275
	return self;
sl@0
   276
	}
sl@0
   277
sl@0
   278
sl@0
   279
EXPORT_C CRichText::CRichText(const CParaFormatLayer* aGlobalParaLayer, const CCharFormatLayer* aGlobalCharLayer,
sl@0
   280
							  CStyleList* aStyleList):
sl@0
   281
	CGlobalText(aGlobalParaLayer, aGlobalCharLayer),
sl@0
   282
	iStyleList(aStyleList)
sl@0
   283
	{
sl@0
   284
	}
sl@0
   285
sl@0
   286
sl@0
   287
EXPORT_C void CRichText::ConstructL(TDocumentStorage aStorage, TInt aDefaultTextGranularity, TParaType aParaType)
sl@0
   288
// Initialises and updates the index following the CPlainText::ConstructL
sl@0
   289
// insertion of the end-of-document character.
sl@0
   290
//
sl@0
   291
	{
sl@0
   292
	CPlainText::ConstructL(aStorage, aDefaultTextGranularity);
sl@0
   293
	SetParaTypeIsSingle(aParaType == ESinglePara);
sl@0
   294
	iParserData = new(ELeave) CParserData(DocumentLength());
sl@0
   295
	TInt a;
sl@0
   296
	TInt b;
sl@0
   297
	ParseText(a, b, ETrue);
sl@0
   298
sl@0
   299
	__TEST_INVARIANT;
sl@0
   300
	}
sl@0
   301
sl@0
   302
sl@0
   303
EXPORT_C void CRichText::ConstructL(const CStreamStore& aStore,TStreamId aStreamId,
sl@0
   304
									MPictureFactory* aPictureFactory, MRichTextStoreResolver* aStoreResolver,
sl@0
   305
									MTextFieldFactory* aFieldFactory,
sl@0
   306
									TDocumentStorage aStorage)
sl@0
   307
// Initialises and updates the index following the CPlainText::ConstructL
sl@0
   308
// insertion of the end-of-document character.
sl@0
   309
// Sets the picture header factory if provided.
sl@0
   310
//
sl@0
   311
	{
sl@0
   312
	CPlainText::ConstructL(aStore, aStreamId, aFieldFactory, aStorage);
sl@0
   313
	SetPictureFactory(aPictureFactory, aStoreResolver);
sl@0
   314
	if (iParserData == NULL)
sl@0
   315
		iParserData = new(ELeave) CParserData(DocumentLength());
sl@0
   316
	TInt a;
sl@0
   317
	TInt b;
sl@0
   318
	ParseText(a, b, ETrue);
sl@0
   319
sl@0
   320
	__TEST_INVARIANT;
sl@0
   321
	}
sl@0
   322
sl@0
   323
sl@0
   324
EXPORT_C CRichText::~CRichText()
sl@0
   325
/** The destructor frees all resources owned by the rich text object, prior to
sl@0
   326
its destruction. */
sl@0
   327
	{
sl@0
   328
	// We cannot call DestroyParserSystem() here because it applies to all instances in the thread
sl@0
   329
	delete iParserData;
sl@0
   330
	KillStyleList();
sl@0
   331
	KillIndex();
sl@0
   332
	}
sl@0
   333
sl@0
   334
sl@0
   335
void CRichText::KillStyleList()
sl@0
   336
// Free up the style table
sl@0
   337
//
sl@0
   338
	{
sl@0
   339
	if (StyleListPresent() && !StyleListExternallyOwned())
sl@0
   340
		{
sl@0
   341
		delete iStyleList.AsPtr();
sl@0
   342
		iStyleList = NULL;
sl@0
   343
		}
sl@0
   344
	}
sl@0
   345
sl@0
   346
sl@0
   347
void CRichText::KillIndex()
sl@0
   348
// Delete the rich text index if it's resident in memory.
sl@0
   349
//
sl@0
   350
	{
sl@0
   351
	if (IndexPresent())
sl@0
   352
		delete iIndex.AsPtr();
sl@0
   353
	iIndex=NULL;
sl@0
   354
	}
sl@0
   355
sl@0
   356
sl@0
   357
TBool CRichText::CreateEmptyMarkupComponentL()
sl@0
   358
// If necessary, creates an empty markup component.
sl@0
   359
// Returns ETrue if the markup component was created as a result of this function,
sl@0
   360
// otherwise returns EFalse.
sl@0
   361
//
sl@0
   362
	{
sl@0
   363
	if (IndexPresent())
sl@0
   364
		return EFalse;
sl@0
   365
	TInt paraGran = (ParaTypeIsSingle()) ? KSingleParaGranularity : KMultiParaGranularity;
sl@0
   366
	TInt phrGran = (ParaTypeIsSingle()) ? KSmallPhraseGranularity : KLargePhraseGranularity;
sl@0
   367
	iIndex = CRichTextIndex::NewL(iGlobalParaFormatLayer, iGlobalCharFormatLayer, *this,paraGran,phrGran);
sl@0
   368
	return ETrue;
sl@0
   369
	}
sl@0
   370
sl@0
   371
sl@0
   372
void CRichText::CreateAndGenerateMarkupComponentL()
sl@0
   373
// Checks if the rich text index needs to be created and does so if necessary.
sl@0
   374
// Called by all public rich text functions that manipulate specific formatting.
sl@0
   375
//
sl@0
   376
	{
sl@0
   377
	if (CreateEmptyMarkupComponentL())
sl@0
   378
		{
sl@0
   379
		TRAPD(ret, GenerateMarkupL());
sl@0
   380
		if (ret != KErrNone)
sl@0
   381
			{// destroy this partially set markup component
sl@0
   382
			delete iIndex.AsPtr();
sl@0
   383
			iIndex = NULL;
sl@0
   384
			User::Leave(ret);
sl@0
   385
			}
sl@0
   386
		}
sl@0
   387
	}
sl@0
   388
sl@0
   389
sl@0
   390
void CRichText::GenerateMarkupL()
sl@0
   391
// Generate markup data corresponding to the current text content.
sl@0
   392
//
sl@0
   393
	{
sl@0
   394
	TInt remainingLength = DocumentLength();
sl@0
   395
	TInt startPos = 0;
sl@0
   396
	while (remainingLength)
sl@0
   397
		{
sl@0
   398
		TPtrC buf = Read(startPos);
sl@0
   399
		TInt consumed = buf.Length();
sl@0
   400
		if (consumed > remainingLength)
sl@0
   401
			{
sl@0
   402
			consumed = remainingLength;
sl@0
   403
			buf.Set(buf.Ptr(), consumed);
sl@0
   404
			}
sl@0
   405
		iIndex->InsertL(startPos, buf, *iGlobalParaFormatLayer);
sl@0
   406
		remainingLength -= consumed;
sl@0
   407
		startPos += consumed;
sl@0
   408
		}
sl@0
   409
	}
sl@0
   410
sl@0
   411
EXPORT_C void CRichText::CopyToStoreL(CStreamStore& aStore,CStreamDictionary& aDictionary,TInt aPos,TInt aLength) const
sl@0
   412
/** Copies a portion of the rich text object, with components to the clipboard.
sl@0
   413
sl@0
   414
A panic occurs in the following circumstances:
sl@0
   415
sl@0
   416
aPos is an invalid document position
sl@0
   417
sl@0
   418
aLength is invalid (zero or less)
sl@0
   419
sl@0
   420
the sum of aPos and aLength is greater than or equal to the number of characters
sl@0
   421
in the document
sl@0
   422
sl@0
   423
@param aStore Stream store to which the rich text is written.
sl@0
   424
@param aDictionary The stream dictionary.
sl@0
   425
@param aPos The document position from which to begin copying.
sl@0
   426
@param aLength The number of characters to copy. */
sl@0
   427
	{
sl@0
   428
	if (aLength > 0)
sl@0
   429
		{
sl@0
   430
		TStreamId plainTextId = CPlainText::DoCopyToStoreL(aStore,aDictionary,aPos,aLength);
sl@0
   431
		TStreamId id = DoCopyToStoreL(aStore,aPos,aLength,plainTextId,FALSE);
sl@0
   432
		aDictionary.AssignL(KClipboardUidTypeRichText,id);
sl@0
   433
		TStreamId idStyles = DoCopyToStoreL(aStore,aPos,aLength,plainTextId,TRUE);
sl@0
   434
		aDictionary.AssignL(KClipboardUidTypeRichTextWithStyles,idStyles);
sl@0
   435
		}
sl@0
   436
	}
sl@0
   437
sl@0
   438
sl@0
   439
// Copy the selected region of rich text, with components, to the specified store.
sl@0
   440
TStreamId CRichText::DoCopyToStoreL(CStreamStore& aStore,TInt aPos,TInt aLength,TStreamId aPlainTextId,TBool aCopyStyles) const
sl@0
   441
	{
sl@0
   442
	__TEST_INVARIANT;
sl@0
   443
sl@0
   444
	TInt documentLength = DocumentLength();
sl@0
   445
	if (aPos < 0 || aPos > documentLength)
sl@0
   446
	    {
sl@0
   447
	    OstTrace0( TRACE_FATAL, CRICHTEXT_DOCOPYTOSTOREL, "ECharPosBeyondDocument" );
sl@0
   448
	    }
sl@0
   449
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= documentLength,Panic(ECharPosBeyondDocument));
sl@0
   450
	if (aLength < 0)
sl@0
   451
	    {
sl@0
   452
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_DOCOPYTOSTOREL, "ECopyToClipboardNegativeLength" );
sl@0
   453
	    }
sl@0
   454
	__ASSERT_ALWAYS(aLength >= 0,Panic(ECopyToClipboardNegativeLength));
sl@0
   455
	if (aPos + aLength > documentLength)
sl@0
   456
	    {
sl@0
   457
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_DOCOPYTOSTOREL, "ECharPosBeyondDocument" );
sl@0
   458
	    }
sl@0
   459
	__ASSERT_ALWAYS(aPos + aLength <= documentLength,Panic(ECharPosBeyondDocument));
sl@0
   460
sl@0
   461
	if (aLength == 0)
sl@0
   462
		return KNullStreamId;
sl@0
   463
sl@0
   464
	CStoreMap* map = CStoreMap::NewLC(aStore);
sl@0
   465
	CopyComponentsL(aStore,*map,aPos,aLength,aPlainTextId);
sl@0
   466
	RStoreWriteStream stream(*map);
sl@0
   467
	TStreamId id = stream.CreateLC(aStore);
sl@0
   468
	CopyToStreamL(stream,aPos,aLength,aPlainTextId,aCopyStyles);
sl@0
   469
	stream.CommitL();
sl@0
   470
	map->Reset();
sl@0
   471
	CleanupStack::PopAndDestroy(2);
sl@0
   472
sl@0
   473
	__TEST_INVARIANT;
sl@0
   474
	return id;
sl@0
   475
	}
sl@0
   476
sl@0
   477
sl@0
   478
EXPORT_C void CRichText::CopyComponentsL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength,TStreamId aPlainTextId) const
sl@0
   479
	{
sl@0
   480
	if (aPlainTextId == KNullStreamId)
sl@0
   481
		CPlainText::CopyComponentsL(aStore,aMap,aPos,aLength);
sl@0
   482
	if (IndexPresent())
sl@0
   483
		iIndex->StorePicturesL(aStore,aMap,aPos,aLength);
sl@0
   484
	}
sl@0
   485
sl@0
   486
sl@0
   487
// Copy the selected region of rich text and components to the specified stream.
sl@0
   488
EXPORT_C void CRichText::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength) const
sl@0
   489
	{
sl@0
   490
	__TEST_INVARIANT;
sl@0
   491
	CopyToStreamL(aStream, aPos, aLength, KNullStreamId);
sl@0
   492
	}
sl@0
   493
sl@0
   494
sl@0
   495
/*
sl@0
   496
Copy the selected region of rich text and components to the specified stream.
sl@0
   497
If aPlainTextId is NULL the plain text component is stored in-line,
sl@0
   498
otherwise the reference to the plain text stream component is stored.
sl@0
   499
*/
sl@0
   500
EXPORT_C void CRichText::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength,TStreamId aPlainTextId) const
sl@0
   501
	{
sl@0
   502
	CopyToStreamL(aStream,aPos,aLength,aPlainTextId,TRUE);
sl@0
   503
	}
sl@0
   504
sl@0
   505
sl@0
   506
void CRichText::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength,TStreamId aPlainTextId,TBool aCopyStyles) const
sl@0
   507
	{
sl@0
   508
	if (aPlainTextId == KNullStreamId)
sl@0
   509
		{
sl@0
   510
		aStream.WriteUint8L(TRUE);
sl@0
   511
		CPlainText::CopyToStreamL(aStream,aPos,aLength);
sl@0
   512
		}
sl@0
   513
	else
sl@0
   514
		{
sl@0
   515
		aStream.WriteUint8L(FALSE);
sl@0
   516
		aStream << aPlainTextId;
sl@0
   517
		}
sl@0
   518
sl@0
   519
	if (!IndexPresent())
sl@0
   520
		CONST_CAST(CRichText*,this)->CreateAndGenerateMarkupComponentL();
sl@0
   521
	aStream.WriteUint8L(TRUE);
sl@0
   522
	iIndex->CopyToStreamL(aStream,aPos,aLength,aCopyStyles);
sl@0
   523
	}
sl@0
   524
sl@0
   525
EXPORT_C TInt CRichText::PasteFromStoreL(const CStreamStore& aStore, const CStreamDictionary& aDictionary, TInt aPos, CParagraphStyle::TStylePasteMode aStylePasteMode)
sl@0
   526
/** Pastes the contents of the clipboard into the rich text object at the
sl@0
   527
specified document position. Returns the number of characters pasted.
sl@0
   528
sl@0
   529
If the text in the clipboard has been formatted using styles, the
sl@0
   530
aStylePasteMode argument indicates whether the styles should be preserved
sl@0
   531
or discarded. If the argument is not specified, the pasted rich text
sl@0
   532
retains all formatting specified in the styles, and any new style
sl@0
   533
definitions are added to the style list of the rich text object into which
sl@0
   534
it is pasted.
sl@0
   535
sl@0
   536
@param aStore The stream store from which to paste the rich text.
sl@0
   537
@param aDictionary The stream dictionary.
sl@0
   538
@param aPos The document position at which to paste the rich text. Must be
sl@0
   539
valid, or a panic occurs.
sl@0
   540
@param aStylePasteMode Indicates whether styles in the pasted text should be
sl@0
   541
preserved or discarded.
sl@0
   542
@return The number of characters pasted. */
sl@0
   543
	{
sl@0
   544
	return DoPasteRtFromStoreL(aStore, aDictionary, aPos, aStylePasteMode);
sl@0
   545
	}
sl@0
   546
sl@0
   547
sl@0
   548
EXPORT_C TInt CRichText::PasteFromStoreL(const CStreamStore& aStore, const CStreamDictionary& aDictionary, TInt aPos)
sl@0
   549
// Paste the lesser of i) aMaxPasteLength ii) the entire clipboard contents.
sl@0
   550
// Return a count of the number of characters pasted.
sl@0
   551
//
sl@0
   552
	{
sl@0
   553
	return DoPasteRtFromStoreL(aStore, aDictionary, aPos, CParagraphStyle::EAddNewStyles);
sl@0
   554
	}
sl@0
   555
sl@0
   556
sl@0
   557
TInt CRichText::DoPasteRtFromStoreL(const CStreamStore& aStore, const CStreamDictionary& aDictionary, TInt aPos, CParagraphStyle::TStylePasteMode aStylePasteMode)
sl@0
   558
// Paste the rich text data from the specified clipboard, inserting it into this
sl@0
   559
// instance at character position aPos.  Returns the number of characters pasted.
sl@0
   560
// May be 0.
sl@0
   561
//
sl@0
   562
	{
sl@0
   563
	__TEST_INVARIANT;
sl@0
   564
	if (aPos > DocumentLength())
sl@0
   565
	    {
sl@0
   566
	    OstTrace0( TRACE_FATAL, CRICHTEXT_DOPASTERTFROMSTOREL, "ECharPosBeyondDocument" );
sl@0
   567
	    }
sl@0
   568
	__ASSERT_ALWAYS(aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   569
sl@0
   570
	TUid type = KClipboardUidTypeRichTextWithStyles;
sl@0
   571
	TStreamId id = aDictionary.At(type);
sl@0
   572
	if (id == KNullStreamId)
sl@0
   573
		{
sl@0
   574
		type = KClipboardUidTypeRichText;
sl@0
   575
		id = aDictionary.At(type);
sl@0
   576
		}
sl@0
   577
	if (id == KNullStreamId)
sl@0
   578
		{
sl@0
   579
		type = KClipboardUidTypePlainText;
sl@0
   580
		id = aDictionary.At(type);
sl@0
   581
		}
sl@0
   582
	TInt consumed = 0;
sl@0
   583
	if (id == KNullStreamId)
sl@0
   584
		return consumed;
sl@0
   585
	if (type == KClipboardUidTypeRichText || type == KClipboardUidTypeRichTextWithStyles)
sl@0
   586
		consumed = PasteRichTextFromStoreL(aStore, aDictionary, id, aPos, aStylePasteMode);
sl@0
   587
	else if (type == KClipboardUidTypePlainText)
sl@0
   588
		consumed = PastePlainTextFromStoreL(aStore, id, aPos);
sl@0
   589
sl@0
   590
	if (consumed)
sl@0
   591
		{
sl@0
   592
		iParserData->MergeRange(aPos,0,consumed);
sl@0
   593
		CallEditObserver(aPos, consumed);
sl@0
   594
		}
sl@0
   595
sl@0
   596
	SetHasChanged(ETrue);
sl@0
   597
	__TEST_INVARIANT;
sl@0
   598
	return consumed;
sl@0
   599
	}
sl@0
   600
sl@0
   601
sl@0
   602
TInt CRichText::PasteRichTextFromStoreL(const CStreamStore& aStore, const CStreamDictionary& aDictionary, TStreamId& aRichTextStreamId, TInt aPos,CParagraphStyle::TStylePasteMode aStylePasteMode)
sl@0
   603
// Paste the plain text stream, then open and paste the rich text markup from the specified stream.
sl@0
   604
//
sl@0
   605
	{
sl@0
   606
	if (!IndexPresent())
sl@0
   607
		CreateAndGenerateMarkupComponentL();  // create the index if it does not already exist
sl@0
   608
	TStreamId id = aDictionary.At(KClipboardUidTypePlainText);
sl@0
   609
// ASSERT: We have rich text, so the plain text stream must exist.
sl@0
   610
	if (id == KNullStreamId)
sl@0
   611
	    {
sl@0
   612
	    OstTrace0( TRACE_FATAL, CRICHTEXT_PASTERICHTEXTFROMSTOREL, "EClipboardIntegrity" );
sl@0
   613
	    }
sl@0
   614
	__ASSERT_ALWAYS(id != KNullStreamId,Panic(EClipboardIntegrity));
sl@0
   615
	TInt consumed = CPlainText::DoPasteFromStoreL(aStore, id, aPos);
sl@0
   616
	TRAPD(ret, CompletePasteRichTextFromStoreL(aStore, aRichTextStreamId, aPos, aStylePasteMode));
sl@0
   617
	if (ret != KErrNone)
sl@0
   618
		{
sl@0
   619
		CPlainText::Delete(aPos,consumed);
sl@0
   620
		OstTrace1( TRACE_FATAL, DUP1_CRICHTEXT_PASTERICHTEXTFROMSTOREL, "Leave code=%d", ret );
sl@0
   621
		User::Leave(ret);
sl@0
   622
		}
sl@0
   623
	return consumed;
sl@0
   624
	}
sl@0
   625
sl@0
   626
sl@0
   627
void CRichText::CompletePasteRichTextFromStoreL(const CStreamStore& aStore, TStreamId& aRichTextStreamId, TInt aPos, CParagraphStyle::TStylePasteMode aStylePasteMode)
sl@0
   628
//
sl@0
   629
	{
sl@0
   630
	RStoreReadStream stream;
sl@0
   631
	stream.OpenLC(aStore, aRichTextStreamId);
sl@0
   632
	TBool plainTextInline = (TBool)stream.ReadUint8L();
sl@0
   633
	if (!plainTextInline)
sl@0
   634
		{
sl@0
   635
		TStreamId dummy;
sl@0
   636
		stream >> dummy;
sl@0
   637
		}
sl@0
   638
	TBool markupPresentInClipboard = (TBool)stream.ReadUint8L();
sl@0
   639
	if (markupPresentInClipboard)
sl@0
   640
		iIndex->PasteFromStreamL(aStore, stream, aPos,aStylePasteMode, iGlobalParaFormatLayer, iGlobalCharFormatLayer);
sl@0
   641
	CleanupStack::PopAndDestroy();  // stream
sl@0
   642
	}
sl@0
   643
sl@0
   644
sl@0
   645
TInt CRichText::PastePlainTextFromStoreL(const CStreamStore& aStore, TStreamId& anId, TInt aPos)
sl@0
   646
// Paste the plain text stream, then update the rich text index to show any paragraph delimiters pasted.
sl@0
   647
//
sl@0
   648
	{
sl@0
   649
	TInt consumed = CPlainText::DoPasteFromStoreL(aStore, anId, aPos);
sl@0
   650
	//
sl@0
   651
	if (IndexPresent())
sl@0
   652
		{
sl@0
   653
		TRAPD(ret, CompletePastePlainTextL(aPos, consumed));
sl@0
   654
		if (ret != KErrNone)
sl@0
   655
			{
sl@0
   656
			CPlainText::Delete(aPos,consumed);
sl@0
   657
			User::Leave(ret);
sl@0
   658
			}
sl@0
   659
		}
sl@0
   660
	return consumed;
sl@0
   661
	}
sl@0
   662
sl@0
   663
void CRichText::CompletePastePlainTextL(TInt aPos, TInt aCharacterCount)
sl@0
   664
// Updates the rich text index following the pasting of global text.
sl@0
   665
//
sl@0
   666
	{
sl@0
   667
	HBufC* buf = HBufC::NewLC(aCharacterCount);
sl@0
   668
	TPtr bf = buf->Des();
sl@0
   669
	Extract(bf, aPos, aCharacterCount);
sl@0
   670
	iIndex->InsertL(aPos, bf, *iGlobalParaFormatLayer);
sl@0
   671
	CleanupStack::PopAndDestroy();  // buf
sl@0
   672
	}
sl@0
   673
sl@0
   674
EXPORT_C void CRichText::Reset()
sl@0
   675
/** Resets the document's contents to a single end-of-document delimiter. Also
sl@0
   676
deletes the style list if owned by the object. */
sl@0
   677
	{
sl@0
   678
	// Resets document contents to single end-of-document charcter,
sl@0
   679
	// and resets the index to match document content.
sl@0
   680
	TInt length = DocumentLength();
sl@0
   681
	CPlainText::Reset();  // remove all content bar eod character
sl@0
   682
	KillStyleList();  // does not get destroyed if externally owned
sl@0
   683
	KillIndex();
sl@0
   684
	SetHasChanged(ETrue);
sl@0
   685
	iParserData->KillRange();
sl@0
   686
	iParserData->iLastKnownCursor = -1;
sl@0
   687
	CallEditObserver(0, -length);
sl@0
   688
sl@0
   689
	__TEST_INVARIANT;
sl@0
   690
	}
sl@0
   691
sl@0
   692
EXPORT_C void CRichText::InsertL(TInt aPos, const TChar& aChar)
sl@0
   693
/** Inserts either a single character or a descriptor into the text object
sl@0
   694
at a specified document position.
sl@0
   695
sl@0
   696
The insertion position must be valid or a panic occurs.Note:A panic
sl@0
   697
occurs if the text object is in an "insert pending" state (i.e.
sl@0
   698
SetInsertCharFormatL() has been called and has not been
sl@0
   699
cancelled using CancelInsertCharFormat()) and aPos is not
sl@0
   700
the same as the insertion point.
sl@0
   701
sl@0
   702
@param aPos The document position at which to insert the character/descriptor.
sl@0
   703
@param aChar The character to insert. Must not be a picture character or a
sl@0
   704
panic occurs.
sl@0
   705
@param aBuf The descriptor to insert. */
sl@0
   706
	{
sl@0
   707
	__TEST_INVARIANT;
sl@0
   708
	if (aPos < 0 || aPos > DocumentLength())
sl@0
   709
	    {
sl@0
   710
	    OstTrace0( TRACE_FATAL, CRICHTEXT_INSERTL, "ECharPosBeyondDocument" );
sl@0
   711
	    }
sl@0
   712
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   713
	if (aChar == EPictureCharacter)
sl@0
   714
	    {
sl@0
   715
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_INSERTL, "ENonOverloadedInsertCalledWithPictureCharacter" );
sl@0
   716
	    }
sl@0
   717
	__ASSERT_ALWAYS(aChar!= EPictureCharacter, Panic(ENonOverloadedInsertCalledWithPictureCharacter));
sl@0
   718
sl@0
   719
	if (aChar < 0x10000)
sl@0
   720
		{
sl@0
   721
		TBuf<1> content;
sl@0
   722
		content.Append(aChar);
sl@0
   723
		RtInsertL(aPos, content);
sl@0
   724
		}
sl@0
   725
	else
sl@0
   726
		{
sl@0
   727
		TText16 high = TChar::GetHighSurrogate(aChar);
sl@0
   728
		TText16 low = TChar::GetLowSurrogate(aChar);
sl@0
   729
		RDebug::Print(_L("CRichText::InsertL(), %X expand to %X %X."), TUint(aChar), high, low);
sl@0
   730
		
sl@0
   731
		TBuf<2> content;
sl@0
   732
		content.Append(high);
sl@0
   733
		content.Append(low);
sl@0
   734
		RtInsertL(aPos, content);
sl@0
   735
		}
sl@0
   736
sl@0
   737
	__TEST_INVARIANT;
sl@0
   738
	}
sl@0
   739
sl@0
   740
sl@0
   741
EXPORT_C void CRichText::InsertL(TInt aPos, const TDesC& aBuf)
sl@0
   742
// Inserts the contents of aBuf into the text component at character position aPos.
sl@0
   743
//
sl@0
   744
	{
sl@0
   745
	__ETEXT_WATCH(INSERT_DESC);
sl@0
   746
sl@0
   747
	RtInsertL(aPos, aBuf);
sl@0
   748
sl@0
   749
	__ETEXT_WATCH_END(INSERT_DESC);
sl@0
   750
	}
sl@0
   751
sl@0
   752
sl@0
   753
EXPORT_C void CRichText::RtInsertL(TInt aPos, const TDesC& aBuf)
sl@0
   754
// Inserts the contents a aBuf into the text component at position aPos.
sl@0
   755
// Updates the index accordingly, and accounts for all embedded paragraph delimiters
sl@0
   756
// in the passed buffer aBuf.
sl@0
   757
//
sl@0
   758
	{
sl@0
   759
	__TEST_INVARIANT;
sl@0
   760
	if (aPos < 0 || aPos > DocumentLength())
sl@0
   761
	    {
sl@0
   762
	    OstTrace0( TRACE_FATAL, CRICHTEXT_RTINSERTL, "ECharPosBeyondDocument" );
sl@0
   763
	    }
sl@0
   764
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   765
sl@0
   766
	int length = aBuf.Length();
sl@0
   767
	CPlainText::InsertL(aPos, aBuf);
sl@0
   768
	if (IndexPresent())
sl@0
   769
		{
sl@0
   770
		TRAPD(ret, iIndex->InsertL(aPos, aBuf, *iGlobalParaFormatLayer));
sl@0
   771
		if (ret != KErrNone)
sl@0
   772
			{
sl@0
   773
			CPlainText::Delete(aPos,length);
sl@0
   774
			User::Leave(ret);
sl@0
   775
			}
sl@0
   776
		SetHasChanged(ETrue);
sl@0
   777
		}
sl@0
   778
sl@0
   779
	iParserData->MergeRange(aPos,0,length);
sl@0
   780
	CallEditObserver(aPos,length);
sl@0
   781
sl@0
   782
	__TEST_INVARIANT;
sl@0
   783
	}
sl@0
   784
sl@0
   785
sl@0
   786
EXPORT_C void CRichText::InsertL(TInt aPos,const TPictureHeader& aHeader)
sl@0
   787
/**Inserts a picture header into the text object at a specified document
sl@0
   788
position. The picture header specified must reference a valid picture, or
sl@0
   789
a panic occurs.
sl@0
   790
sl@0
   791
A panic also occurs if the text object is in an "insert pending" state
sl@0
   792
(SetInsertCharFormatL() has been called and has not been cancelled using
sl@0
   793
CancelInsertCharFormat()) and aPos is not the same as the insertion point.
sl@0
   794
sl@0
   795
This method takes ownership of a picture referenced in aHeader.
sl@0
   796
sl@0
   797
@param aPos The document position at which to insert the picture header. Must
sl@0
   798
be valid, or a panic occurs.
sl@0
   799
@param aHeader A picture header. This holds a pointer to the picture to insert,
sl@0
   800
and information about the picture. */
sl@0
   801
	{
sl@0
   802
	__TEST_INVARIANT;
sl@0
   803
	if (aPos < 0 || aPos > DocumentLength())
sl@0
   804
	    {
sl@0
   805
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_INSERTL, "ECharPosBeyondDocument" );
sl@0
   806
	    }
sl@0
   807
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   808
sl@0
   809
	CleanupStack::PushL(aHeader.iPicture);
sl@0
   810
sl@0
   811
	if (!IndexPresent())
sl@0
   812
		CreateAndGenerateMarkupComponentL();
sl@0
   813
	TBuf<1> picture;
sl@0
   814
	picture.Append(EPictureCharacter);
sl@0
   815
	CPlainText::InsertL(aPos, picture);
sl@0
   816
sl@0
   817
    CleanupStack::Pop(aHeader.iPicture);
sl@0
   818
sl@0
   819
	TBool pictureOwnershipTaken(EFalse);
sl@0
   820
	TRAPD(ret, iIndex->InsertL(aPos, aHeader,pictureOwnershipTaken));
sl@0
   821
	if (ret != KErrNone)
sl@0
   822
		{
sl@0
   823
		if(!pictureOwnershipTaken)
sl@0
   824
		    {
sl@0
   825
		    const_cast <TPictureHeader&> (aHeader).DeletePicture();
sl@0
   826
		    }
sl@0
   827
		CPlainText::Delete(aPos,picture.Length());  // remove the picture place-holder
sl@0
   828
    	User::Leave(ret);
sl@0
   829
		}
sl@0
   830
sl@0
   831
	SetHasChanged(ETrue);
sl@0
   832
	iParserData->MergeRange(aPos,0,1);
sl@0
   833
	CallEditObserver(aPos,1);
sl@0
   834
	__TEST_INVARIANT;
sl@0
   835
	}
sl@0
   836
sl@0
   837
sl@0
   838
EXPORT_C TBool CRichText::DeleteL(TInt aPos, TInt aLength)
sl@0
   839
/** Deletes one or more characters beginning at, and including, the character at
sl@0
   840
a specified document position. Can leave because paragraphs may be merged
sl@0
   841
and reformatted by the function.
sl@0
   842
sl@0
   843
@param aPos The start of the range of characters to delete. Must be valid
sl@0
   844
or a panic occurs.
sl@0
   845
@param aLength The number of characters to delete. Must be positive or a panic
sl@0
   846
occurs. The sum of aPos and aLength must be less than the document length,
sl@0
   847
or a panic occurs.
sl@0
   848
@return Indicates whether two paragraphs have been merged as a result of the
sl@0
   849
delete, indicating that the formatting of part of the resulting paragraph
sl@0
   850
may have changed. */
sl@0
   851
	{
sl@0
   852
	// Deletes aRange number of characters from the text component
sl@0
   853
	// and the corresponding index data.
sl@0
   854
	// Delete commences at, and includes, character position aPos.
sl@0
   855
	__TEST_INVARIANT;
sl@0
   856
sl@0
   857
	if (aPos < 0 || aPos > DocumentLength())
sl@0
   858
	    {
sl@0
   859
	    OstTrace0( TRACE_FATAL, CRICHTEXT_DELETEL, "ECharPosBeyondDocument" );
sl@0
   860
	    }
sl@0
   861
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   862
	if (aLength < 0)
sl@0
   863
	    {
sl@0
   864
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_DELETEL, "EDebugDeleteZeroLength" );
sl@0
   865
	    }
sl@0
   866
	__ASSERT_ALWAYS(aLength >= 0, Panic(EDebugDeleteZeroLength));
sl@0
   867
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
   868
	    {
sl@0
   869
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_DELETEL, "ECharPosBeyondDocument" );
sl@0
   870
	    }
sl@0
   871
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   872
sl@0
   873
	TBool requireMerge = EFalse;
sl@0
   874
	if (!IndexPresent())
sl@0
   875
		CPlainText::Delete(aPos,aLength);
sl@0
   876
	else
sl@0
   877
		{
sl@0
   878
		iIndex->CancelInsertCharFormat();
sl@0
   879
		TIndexDeleteInfo info;
sl@0
   880
		iIndex->SetForDeleteL(info, aPos, aLength);
sl@0
   881
		CPlainText::Delete(aPos,aLength);
sl@0
   882
		requireMerge = iIndex->DeleteNow(info);
sl@0
   883
		}
sl@0
   884
	iParserData->MergeRange(aPos,aLength,0);
sl@0
   885
	CallEditObserver(aPos,-aLength);
sl@0
   886
sl@0
   887
	__TEST_INVARIANT;
sl@0
   888
	return requireMerge;
sl@0
   889
	}
sl@0
   890
sl@0
   891
EXPORT_C void CRichText::DeleteParagraph(TInt aPos, TInt aLength)
sl@0
   892
/** Deletes one or more whole paragraphs of text. No paragraphs can be merged together
sl@0
   893
by this function, so it cannot leave it must only be used to delete entire
sl@0
   894
paragraphs.
sl@0
   895
sl@0
   896
@param aPos The document position of the start of the first paragraph to delete.
sl@0
   897
Must be a valid position or a panic occurs.
sl@0
   898
@param aLength The number of characters to delete. */
sl@0
   899
	{
sl@0
   900
	// Use to remove entire paragraphs of text.
sl@0
   901
	// Leave-safe if called in this context.
sl@0
   902
	// MUST NOT CALL if not in this context.
sl@0
   903
	__TEST_INVARIANT;
sl@0
   904
sl@0
   905
	// Store the length of the text before we commence with deletions.
sl@0
   906
	TInt initialDocLen=DocumentLength();
sl@0
   907
sl@0
   908
	if (aPos < 0 || aPos > initialDocLen)
sl@0
   909
	    {
sl@0
   910
	    OstTrace0( TRACE_FATAL, CRICHTEXT_DELETEPARAGRAPH, "ECharPosBeyondDocument" );
sl@0
   911
	    }
sl@0
   912
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= initialDocLen, Panic(ECharPosBeyondDocument));
sl@0
   913
	if (aPos + aLength > initialDocLen + 1)
sl@0
   914
	    {
sl@0
   915
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_DELETEPARAGRAPH, "ECharPosBeyondDocument" );
sl@0
   916
	    }
sl@0
   917
	__ASSERT_ALWAYS(aPos + aLength <= initialDocLen + 1, Panic(ECharPosBeyondDocument));
sl@0
   918
sl@0
   919
	if (aLength <= 0)
sl@0
   920
		return;
sl@0
   921
sl@0
   922
	if (IndexPresent())
sl@0
   923
		iIndex->DeleteParagraph(aPos, aLength);
sl@0
   924
	CPlainText::Delete(aPos,aLength);
sl@0
   925
sl@0
   926
	// Change here for defect INC005336.
sl@0
   927
	// Previously the MergeRange call (see else line) was incorrectly handling
sl@0
   928
	// the scenario where all but the first paragraph was deleted e.g. pasting
sl@0
   929
	// multi line buffer into a single line text control in a new email on Techview
sl@0
   930
	// emulator.
sl@0
   931
	// This was because the deletion in the two lines above would delete the
sl@0
   932
	// 'end-of-text' maker causing the iEndParse member of CParserData to
sl@0
   933
	// index the char right after the end of test marker. This would panic
sl@0
   934
	// plugin parsers thatsubsquently executed.
sl@0
   935
sl@0
   936
	if (aPos+aLength > initialDocLen)
sl@0
   937
		// When deletion includes the end-of-text marker, adjust start
sl@0
   938
		// supplied so that it appears to MergeRange that we are
sl@0
   939
		// deleting the paragraph and CR just before it which belongs
sl@0
   940
		// to the previous paragraph and not the end-of-text marker.
sl@0
   941
		// This is actually the end result of the deletion anyway!
sl@0
   942
		iParserData->MergeRange(initialDocLen-aLength,aLength,0);
sl@0
   943
	else
sl@0
   944
		// All other deletions which don't.
sl@0
   945
		iParserData->MergeRange(aPos,aLength,0);
sl@0
   946
sl@0
   947
	CallEditObserver(aPos,-aLength);
sl@0
   948
sl@0
   949
	__TEST_INVARIANT;
sl@0
   950
	}
sl@0
   951
sl@0
   952
EXPORT_C void CRichText::DeleteFromParagraph(TInt aPos, TInt aLength)
sl@0
   953
/** Removes a range of characters from within a single paragraph only. Should not
sl@0
   954
be used for deleting an entire paragraph or paragraphs because this may cause
sl@0
   955
it to leave. Otherwise, it is guaranteed not to leave.
sl@0
   956
sl@0
   957
@param aPos The document position of the start of the range of characters
sl@0
   958
to delete. Must be a valid document position, or a panic occurs.
sl@0
   959
@param aLength The number of characters to delete. The sum of aPos and aLength
sl@0
   960
must be less than the document length, or a panic occurs. */
sl@0
   961
	{
sl@0
   962
	// Removes aLength characters from *within* a single paragraph only.
sl@0
   963
	// Guaranteed not to leave if this pre-condition holds true.
sl@0
   964
	__TEST_INVARIANT;
sl@0
   965
sl@0
   966
	if (aPos < 0 || aPos > DocumentLength())
sl@0
   967
	    {
sl@0
   968
	    OstTrace0( TRACE_FATAL, CRICHTEXT_DELETEFROMPARAGRAPH, "ECharPosBeyondDocument" );
sl@0
   969
	    }
sl@0
   970
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   971
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
   972
	    {
sl@0
   973
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_DELETEFROMPARAGRAPH, "ECharPosBeyondDocument" );
sl@0
   974
	    }
sl@0
   975
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
   976
sl@0
   977
	if (aLength <= 0)
sl@0
   978
		return;
sl@0
   979
sl@0
   980
	if (IndexPresent())
sl@0
   981
		iIndex->DeleteFromParagraph(aPos, aLength);
sl@0
   982
	CPlainText::Delete(aPos,aLength);
sl@0
   983
	iParserData->MergeRange(aPos,aLength,0);
sl@0
   984
	CallEditObserver(aPos,-aLength);
sl@0
   985
sl@0
   986
	__TEST_INVARIANT;
sl@0
   987
	}
sl@0
   988
sl@0
   989
EXPORT_C TInt CRichText::ParagraphCount()const
sl@0
   990
/** Gets a count of the number of paragraphs in the text object.
sl@0
   991
sl@0
   992
Note:
sl@0
   993
sl@0
   994
The paragraph delimiter which terminates every text object means this function
sl@0
   995
always returns a count of at least one.
sl@0
   996
sl@0
   997
@return The number of paragraphs in the document. */
sl@0
   998
	{
sl@0
   999
	// Returns a count of the number of paragraphs
sl@0
  1000
	// in the document.
sl@0
  1001
	//
sl@0
  1002
	__TEST_INVARIANT;
sl@0
  1003
sl@0
  1004
	if (IndexPresent())
sl@0
  1005
		return iIndex->ParagraphCount();
sl@0
  1006
	else
sl@0
  1007
		return CPlainText::ParagraphCount();
sl@0
  1008
	}
sl@0
  1009
sl@0
  1010
sl@0
  1011
EXPORT_C TEtextComponentInfo CRichText::ComponentInfo() const
sl@0
  1012
/** Gets information about the number of components contained in the text object
sl@0
  1013
(the field count, the picture count and the style count).
sl@0
  1014
sl@0
  1015
@return Contains the component information. */
sl@0
  1016
	{
sl@0
  1017
	return TEtextComponentInfo(FieldCount(), PictureCount(), StyleCount());
sl@0
  1018
	}
sl@0
  1019
sl@0
  1020
EXPORT_C TInt CRichText::CharPosOfParagraph(TInt& aLength, TInt aParaOffset) const
sl@0
  1021
/** Finds the length and the start position of a paragraph identified by its
sl@0
  1022
paragraph number. The first paragraph is numbered zero.
sl@0
  1023
sl@0
  1024
Notes:
sl@0
  1025
sl@0
  1026
if aParaOffset is invalid, (equal to or greater than the total number of
sl@0
  1027
paragraphs), the function's return value is EScanEndOfData (= -1)
sl@0
  1028
sl@0
  1029
@param aLength On return contains the length of the specified paragraph.
sl@0
  1030
@param aParaOffset The paragraph number. The first paragraph is numbered zero.
sl@0
  1031
@return The document position of the first character in the paragraph. */
sl@0
  1032
	{
sl@0
  1033
	// Returns the character position of the first character of paragraph aParaOffset,
sl@0
  1034
	// where aParaOffset specifies the nth paragraph.
sl@0
  1035
	// The length of this nth paragraph is written to aLength.
sl@0
  1036
	//
sl@0
  1037
	// If aParaOffset specifies a paragraph that does not exist, EScanEndOfData is returned.
sl@0
  1038
	//
sl@0
  1039
	__TEST_INVARIANT;
sl@0
  1040
sl@0
  1041
	if (IndexPresent())
sl@0
  1042
		return iIndex->CharPosOfParagraph(aLength, aParaOffset);
sl@0
  1043
	else
sl@0
  1044
		return CPlainText::CharPosOfParagraph(aLength, aParaOffset);
sl@0
  1045
	}
sl@0
  1046
sl@0
  1047
EXPORT_C TInt CRichText::ParagraphNumberForPos(TInt& aPos) const
sl@0
  1048
/** Gets the number of the paragraph which contains a document position.
sl@0
  1049
Paragraph numbering begins at zero.
sl@0
  1050
sl@0
  1051
@param aPos A document position. Must be valid or a panic occurs. On return,
sl@0
  1052
contains the document position of the first character in the paragraph in
sl@0
  1053
which it is located.
sl@0
  1054
@return The number of the paragraph containing the specified document position. */
sl@0
  1055
	{
sl@0
  1056
	// Returns the paragraph offset for the specified character position aPos.
sl@0
  1057
	// aPos is in turn modified to hold the character position of the first character
sl@0
  1058
	// of this paragraph.  If aPos is already on a paragraph boundary then do nothing.
sl@0
  1059
	//
sl@0
  1060
	__TEST_INVARIANT;
sl@0
  1061
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1062
	    {
sl@0
  1063
	    OstTrace0( TRACE_FATAL, CRICHTEXT_PARAGRAPHNUMBERFORPOS, "ECharPosBeyondDocument" );
sl@0
  1064
	    }
sl@0
  1065
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1066
sl@0
  1067
	if (IndexPresent())
sl@0
  1068
		return iIndex->ParagraphNumberForPos(aPos);
sl@0
  1069
	else
sl@0
  1070
		return CPlainText::ParagraphNumberForPos(aPos);
sl@0
  1071
	}
sl@0
  1072
sl@0
  1073
/** Applies character formatting to a zero length selection, for example
sl@0
  1074
turning bold on. This has the effect that the formatting will be applied to
sl@0
  1075
text subsequently inserted at the position. This "insert pending" state is
sl@0
  1076
cancelled by calling CancelInsertCharFormat().
sl@0
  1077
sl@0
  1078
Note 1: After calling this function, if text is inserted at a different
sl@0
  1079
position to aPos, a panic will occur, unless CancelInsertCharFormat() has been
sl@0
  1080
called before the insertion to cancel the "insert pending" state.
sl@0
  1081
sl@0
  1082
Note 2: If the insert character format is being set for the end of the
sl@0
  1083
paragraph, the paragraph delimiter is set to that format as well. This helps
sl@0
  1084
end-of-paragraph behaviour be more similar to other places.
sl@0
  1085
sl@0
  1086
@param aFormat The character format values to apply.
sl@0
  1087
@param aMask Character format mask specifying the attributes affected.
sl@0
  1088
@param aPos The document position at which to insert the character format.
sl@0
  1089
@pre aPos must be a valid position, or a panic will occur. */
sl@0
  1090
EXPORT_C void CRichText::SetInsertCharFormatL(const TCharFormat& aFormat,
sl@0
  1091
	const TCharFormatMask& aMask, TInt aPos)
sl@0
  1092
	{
sl@0
  1093
	SetExtendedInsertCharFormatL(aFormat, aMask,aPos);
sl@0
  1094
	}
sl@0
  1095
sl@0
  1096
void CancelInsertCharFormat(TAny* aCRichTextIndex)
sl@0
  1097
	{
sl@0
  1098
	reinterpret_cast<CRichTextIndex*>(aCRichTextIndex)->CancelInsertCharFormat();
sl@0
  1099
	}
sl@0
  1100
sl@0
  1101
void CRichText::SetExtendedInsertCharFormatL(const TCharFormatX& aFormat, const TCharFormatXMask& aMask, TInt aPos)
sl@0
  1102
	{
sl@0
  1103
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1104
	    {
sl@0
  1105
	    OstTrace0( TRACE_FATAL, CRICHTEXT_SETEXTENDEDINSERTCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1106
	    }
sl@0
  1107
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1108
	CreateAndGenerateMarkupComponentL();
sl@0
  1109
	CRichTextIndex* index = iIndex.AsPtr();
sl@0
  1110
	if (index->InsertCharFormatIsActive())
sl@0
  1111
		{
sl@0
  1112
		TCharFormatX format = aFormat;
sl@0
  1113
        TCharFormatXMask mask = aMask;
sl@0
  1114
        CCharFormatLayer* currentLayer = index->GetCurrentInsertCharFormat();
sl@0
  1115
        currentLayer->Sense(format,mask);
sl@0
  1116
		CCharFormatLayer* newLayer = CCharFormatLayer::NewCopyBaseL(currentLayer);
sl@0
  1117
		CleanupStack::PushL(newLayer);
sl@0
  1118
        newLayer->SetL(format,mask);
sl@0
  1119
		if (Read(aPos, 1)[0] == EParagraphDelimiter)
sl@0
  1120
			index->ApplyCharFormatL(aFormat, aMask, aPos, 1, EFalse);
sl@0
  1121
		if (index->InsertCharFormatIsActive())
sl@0
  1122
			{
sl@0
  1123
			currentLayer = index->GetCurrentInsertCharFormat();
sl@0
  1124
			currentLayer->Swap(*newLayer);
sl@0
  1125
			}
sl@0
  1126
		CleanupStack::PopAndDestroy(newLayer);
sl@0
  1127
		}
sl@0
  1128
	else
sl@0
  1129
		{
sl@0
  1130
		TCleanupItem cleanup(::CancelInsertCharFormat, index);
sl@0
  1131
		CleanupStack::PushL(cleanup);
sl@0
  1132
		index->NewInsertCharFormatL(aFormat, aMask, aPos);
sl@0
  1133
		if (Read(aPos, 1)[0] == EParagraphDelimiter)
sl@0
  1134
			index->ApplyCharFormatL(aFormat, aMask, aPos, 1, EFalse);
sl@0
  1135
		CleanupStack::Pop();
sl@0
  1136
		}
sl@0
  1137
	SetHasChanged(TRUE);
sl@0
  1138
	}
sl@0
  1139
	
sl@0
  1140
EXPORT_C void CRichText::ExtendedInterface(TAny*& aInterface, TUid aInterfaceId)
sl@0
  1141
/**
sl@0
  1142
Returns the interface corresponding to the specified UID if it exists, or 0 if not. 
sl@0
  1143
Overridden versions should base call rather than returning 0.
sl@0
  1144
For KUidRichText, CRichText will be returned if rich text is supported.
sl@0
  1145
sl@0
  1146
@param aInterfaceId The UID indicating the interface to return
sl@0
  1147
@param aInterface The interface corresponding to aInterfaceId
sl@0
  1148
if it is supported, or 0 if it is not
sl@0
  1149
*/
sl@0
  1150
	{ 
sl@0
  1151
	if(KUidRichText == aInterfaceId) 
sl@0
  1152
		{
sl@0
  1153
		aInterface = REINTERPRET_CAST(TAny*, this);
sl@0
  1154
		}
sl@0
  1155
	else
sl@0
  1156
		{
sl@0
  1157
	    CGlobalText::ExtendedInterface(aInterface, aInterfaceId);
sl@0
  1158
		}
sl@0
  1159
	}
sl@0
  1160
sl@0
  1161
EXPORT_C TBool CRichText::DelSetInsertCharFormatL(TInt aPos, TInt aLength)
sl@0
  1162
/** Deletes a range of characters. The range affected is from aPos to
sl@0
  1163
aPos+(aLength-1) inclusive. It differs from DeleteL() in that this function
sl@0
  1164
preserves the formatting of the deleted character at position aPos, so that
sl@0
  1165
any text subsequently inserted at aPos will have that formatting applied to it.
sl@0
  1166
sl@0
  1167
A panic occurs if:
sl@0
  1168
sl@0
  1169
after calling this function, text is inserted at a different position to aPos,
sl@0
  1170
without calling CancelInsertCharFormat() before the insertion
sl@0
  1171
sl@0
  1172
aPos is invalid
sl@0
  1173
sl@0
  1174
aLength is negative
sl@0
  1175
sl@0
  1176
the range goes beyond the end of the document
sl@0
  1177
sl@0
  1178
@param aPos The document position of the first character to delete.
sl@0
  1179
@param aLength The number of characters to delete.
sl@0
  1180
@return ETrue if two paragraphs have been merged as a result of the deletion;
sl@0
  1181
EFalse if there has been no paragraph merge. */
sl@0
  1182
	{
sl@0
  1183
	// Delete aLength characters, commencing at, and including, aPos.
sl@0
  1184
	// If aPos is on a phrase boundary, and the whole phrase or more is deleted then
sl@0
  1185
	// remember temporarily the phrase format.  This is applied to any content that is
sl@0
  1186
	// immediately inserted.
sl@0
  1187
	//
sl@0
  1188
	__TEST_INVARIANT;
sl@0
  1189
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1190
	    {
sl@0
  1191
	    OstTrace0( TRACE_FATAL, CRICHTEXT_DELSETINSERTCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1192
	    }
sl@0
  1193
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1194
	if (aLength < 0)
sl@0
  1195
	    {
sl@0
  1196
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_DELSETINSERTCHARFORMATL, "EDebugDeleteZeroLength" );
sl@0
  1197
	    }
sl@0
  1198
	__ASSERT_ALWAYS(aLength >= 0, Panic(EDebugDeleteZeroLength));
sl@0
  1199
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
  1200
	    {
sl@0
  1201
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_DELSETINSERTCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1202
	    }
sl@0
  1203
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1204
sl@0
  1205
	TBool parasMerged = EFalse;
sl@0
  1206
	if (!IndexPresent())
sl@0
  1207
		CPlainText::DeleteL(aPos, aLength);
sl@0
  1208
	else
sl@0
  1209
		{
sl@0
  1210
		parasMerged = iIndex->DelSetInsertCharFormatL(aPos, aLength);
sl@0
  1211
		CPlainText::Delete(aPos,aLength);
sl@0
  1212
		SetHasChanged(ETrue);
sl@0
  1213
		}
sl@0
  1214
	iParserData->MergeRange(aPos,aLength,0);
sl@0
  1215
	CallEditObserver(aPos,-aLength);
sl@0
  1216
sl@0
  1217
	__TEST_INVARIANT;
sl@0
  1218
	return parasMerged;
sl@0
  1219
	}
sl@0
  1220
sl@0
  1221
EXPORT_C void CRichText::CancelInsertCharFormat()
sl@0
  1222
/** Cancels the "insert pending" state set by a call to SetInsertCharFormatL()
sl@0
  1223
or DelSetInsertCharFormatL().
sl@0
  1224
sl@0
  1225
This function removes the restriction on the text insertion position imposed
sl@0
  1226
by these two functions. It is recommended that it is called before every cursor
sl@0
  1227
movement, scroll, paste, etc. This call is a small overhead, and has no effect
sl@0
  1228
if not applicable. */
sl@0
  1229
	{
sl@0
  1230
	// Cancels the transitory state where a specified character format
sl@0
  1231
	// is applied *on top of* any inherited formatting.  eg, when bold is on.
sl@0
  1232
	// Cancel when: (1) the text position is altered.  (2) the first character
sl@0
  1233
	// has been inserted following the setting of this state.
sl@0
  1234
	//
sl@0
  1235
	if (IndexPresent() && iIndex->InsertCharFormatIsActive())
sl@0
  1236
		{
sl@0
  1237
		iIndex->CancelInsertCharFormat();
sl@0
  1238
		SetHasChanged(ETrue);
sl@0
  1239
		}
sl@0
  1240
	}
sl@0
  1241
sl@0
  1242
EXPORT_C void CRichText::ApplyParaFormatL(const CParaFormat* aFormat, const TParaFormatMask& aMask, TInt aPos, TInt aLength)
sl@0
  1243
/** Applies paragraph formatting to a range of paragraphs. The attributes which
sl@0
  1244
are set in the mask are taken from aFormat and applied. The attributes which
sl@0
  1245
are not set in the mask are not changed.
sl@0
  1246
sl@0
  1247
The region affected consists of every paragraph containing one or more
sl@0
  1248
characters in the range aPos to aPos+(aLength-1).
sl@0
  1249
sl@0
  1250
@param aFormat Contains the paragraph format attribute values to apply.
sl@0
  1251
@param aMask Specifies which paragraph format attributes should be affected.
sl@0
  1252
@param aPos The document position of the start of the range.
sl@0
  1253
@param aLength The number of characters in the range. */
sl@0
  1254
	{
sl@0
  1255
	// Applies the specified format attributes to the paragraphs covering
sl@0
  1256
	// character position aPos to aPos+aLength-1.
sl@0
  1257
	//
sl@0
  1258
	__TEST_INVARIANT;
sl@0
  1259
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1260
	    {
sl@0
  1261
	    OstTrace0( TRACE_FATAL, CRICHTEXT_APPLYPARAFORMATL, "ECharPosBeyondDocument" );
sl@0
  1262
	    }
sl@0
  1263
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1264
	if (aLength < 0)
sl@0
  1265
	    {
sl@0
  1266
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_APPLYPARAFORMATL, "EApplyParaFormatNegativeLength" );
sl@0
  1267
	    }
sl@0
  1268
	__ASSERT_ALWAYS(aLength >= 0,Panic(EApplyParaFormatNegativeLength));
sl@0
  1269
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
  1270
	    {
sl@0
  1271
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_APPLYPARAFORMATL, "ECharPosBeyondDocument" );
sl@0
  1272
	    }
sl@0
  1273
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1274
sl@0
  1275
	CreateAndGenerateMarkupComponentL();
sl@0
  1276
	iIndex->ApplyParaFormatL(aFormat, aMask, aPos, aLength);
sl@0
  1277
	SetHasChanged(ETrue);
sl@0
  1278
sl@0
  1279
	__TEST_INVARIANT;
sl@0
  1280
	}
sl@0
  1281
sl@0
  1282
EXPORT_C void CRichText::ApplyCharFormatL(const TCharFormat& aFormat, const TCharFormatMask& aMask, TInt aPos, TInt aLength)
sl@0
  1283
/** Applies character formatting to a range of characters. The attributes which
sl@0
  1284
are set in the mask are read from aFormat and applied. The attributes which
sl@0
  1285
are not set in the mask are not changed. The range of characters affected
sl@0
  1286
is from aPos to aPos+(aLength-1) inclusive. The sum of aPos and aLength
sl@0
  1287
must be less than or equal to the document length, or a panic occurs.
sl@0
  1288
sl@0
  1289
@param aFormat Contains the character format attribute values to apply.
sl@0
  1290
@param aMask Bitmask specifying which character format attributes should be
sl@0
  1291
applied.
sl@0
  1292
@param aPos Document position from which to apply the new character formatting.
sl@0
  1293
Must be greater than or equal to zero, or a panic occurs.
sl@0
  1294
@param aLength The number of characters to which the new formatting should
sl@0
  1295
be applied. Must be greater than or equal to zero, or a panic occurs. If the
sl@0
  1296
length is zero, the function has the same effect as SetInsertCharFormatL(). */
sl@0
  1297
	{
sl@0
  1298
	// Applies the specified character formatting to the characters conatined within the range
sl@0
  1299
	// aPos to aPos+(aLength-1).
sl@0
  1300
	// If aLength is zero, the SetInsertCharFormat state is called.
sl@0
  1301
	//
sl@0
  1302
	__ETEXT_WATCH(APPLY_CHAR_FORMAT);
sl@0
  1303
	__TEST_INVARIANT;
sl@0
  1304
	
sl@0
  1305
	TInt document_length = DocumentLength();
sl@0
  1306
	if (aPos < 0)
sl@0
  1307
	    {
sl@0
  1308
	    OstTrace0( TRACE_FATAL, CRICHTEXT_APPLYCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1309
	    }
sl@0
  1310
	__ASSERT_ALWAYS(aPos >= 0,Panic(ECharPosBeyondDocument));
sl@0
  1311
	if (aLength < 0)
sl@0
  1312
	    {
sl@0
  1313
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_APPLYCHARFORMATL, "EApplyCharFormatNegativeLength" );
sl@0
  1314
	    }
sl@0
  1315
	__ASSERT_ALWAYS(aLength >= 0,Panic(EApplyCharFormatNegativeLength));
sl@0
  1316
	if (aPos + aLength - 1 > document_length)
sl@0
  1317
	    {
sl@0
  1318
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_APPLYCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1319
	    }
sl@0
  1320
	__ASSERT_ALWAYS(aPos + aLength - 1 <= document_length,Panic(ECharPosBeyondDocument));
sl@0
  1321
sl@0
  1322
	//If some characters are highlighted AND current position + highlighted txt = document length	
sl@0
  1323
	// Fix for INC097216. Compensate for the changes introduced to Form in defect fix INC087637,
sl@0
  1324
	// which now considers the height of the EOD character; meaning that this character now 
sl@0
  1325
	// needs to be formatted along with rest of text.
sl@0
  1326
	if ((aLength > 0) && (aPos + aLength == document_length))
sl@0
  1327
		{
sl@0
  1328
		aLength++;
sl@0
  1329
		}
sl@0
  1330
	
sl@0
  1331
	DoApplyExtendedCharFormatL(aFormat, aMask, aPos, aLength);
sl@0
  1332
	
sl@0
  1333
	__TEST_INVARIANT;
sl@0
  1334
	__ETEXT_WATCH_END(APPLY_CHAR_FORMAT);
sl@0
  1335
	}
sl@0
  1336
	
sl@0
  1337
// This method is used internally only. It does not format the EOD character.
sl@0
  1338
void CRichText::ApplyExtendedCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,TInt aPos,TInt aLength)
sl@0
  1339
	{
sl@0
  1340
	TInt document_length = DocumentLength();
sl@0
  1341
	if (aPos < 0)
sl@0
  1342
	    {
sl@0
  1343
	    OstTrace0( TRACE_FATAL, CRICHTEXT_APPLYEXTENDEDCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1344
	    }
sl@0
  1345
	__ASSERT_ALWAYS(aPos >= 0,Panic(ECharPosBeyondDocument));
sl@0
  1346
	if (aLength < 0)
sl@0
  1347
	    {
sl@0
  1348
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_APPLYEXTENDEDCHARFORMATL, "EApplyCharFormatNegativeLength" );
sl@0
  1349
	    }
sl@0
  1350
	__ASSERT_ALWAYS(aLength >= 0,Panic(EApplyCharFormatNegativeLength));
sl@0
  1351
	if (aPos + aLength - 1 > document_length)
sl@0
  1352
	    {
sl@0
  1353
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_APPLYEXTENDEDCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1354
	    }
sl@0
  1355
	__ASSERT_ALWAYS(aPos + aLength - 1 <= document_length,Panic(ECharPosBeyondDocument));
sl@0
  1356
	
sl@0
  1357
	DoApplyExtendedCharFormatL(aFormat, aMask, aPos, aLength);
sl@0
  1358
	}	
sl@0
  1359
	
sl@0
  1360
// Apply the extended character formatting.	
sl@0
  1361
void CRichText::DoApplyExtendedCharFormatL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask,TInt aPos,TInt aLength)
sl@0
  1362
	{
sl@0
  1363
	if (aLength > 0)
sl@0
  1364
		{
sl@0
  1365
		CreateAndGenerateMarkupComponentL();
sl@0
  1366
		iIndex->ApplyCharFormatL(aFormat,aMask,aPos,aLength);
sl@0
  1367
		SetHasChanged(ETrue);
sl@0
  1368
		}
sl@0
  1369
	else
sl@0
  1370
		SetExtendedInsertCharFormatL(aFormat,aMask,aPos);
sl@0
  1371
	}		
sl@0
  1372
sl@0
  1373
EXPORT_C void CRichText::SetStyleListExternallyOwned(const CStyleList& aStyleList)
sl@0
  1374
/** Assigns an externally owned style list to the rich text object.
sl@0
  1375
Replaces any previous style list used by the object. Calls
sl@0
  1376
SetStyleListExternallyOwned(ETrue).
sl@0
  1377
sl@0
  1378
@param aExternallyOwned The style list to assign to this rich text object.
sl@0
  1379
Not owned by the rich text object. */
sl@0
  1380
	{
sl@0
  1381
	CStyleList* styleList = CONST_CAST(CStyleList*, &aStyleList);
sl@0
  1382
	iStyleList = styleList;
sl@0
  1383
	SetStyleListExternallyOwned(ETrue);
sl@0
  1384
	}
sl@0
  1385
sl@0
  1386
EXPORT_C void CRichText::ApplyParagraphStyleL(const CParagraphStyle& aStyle, TInt aPos, TInt aLength, CParagraphStyle::TApplyParaStyleMode aMode)
sl@0
  1387
/** Applies a specified paragraph style to a range of paragraphs. The region
sl@0
  1388
affected consists of every paragraph containing one or more characters in the
sl@0
  1389
range aPos to aPos+(aLength-1).
sl@0
  1390
sl@0
  1391
A panic occurs if:
sl@0
  1392
sl@0
  1393
aPos is invalid, or
sl@0
  1394
sl@0
  1395
aLength is negative, or
sl@0
  1396
sl@0
  1397
the range goes beyond the end of the document, or
sl@0
  1398
sl@0
  1399
the rich text object has no style list
sl@0
  1400
sl@0
  1401
@param aStyle The style to apply.
sl@0
  1402
@param aPos The document position of the start of the range.
sl@0
  1403
@param aLength The number of characters in the range.
sl@0
  1404
@param aMode Controls what specific formatting, if any, should be preserved
sl@0
  1405
when the style is applied. */
sl@0
  1406
	{
sl@0
  1407
	// Applies the specified paragraph style to the paragraphs covering
sl@0
  1408
	// character positions aPos to aPos+aLength-1.
sl@0
  1409
	//
sl@0
  1410
	__TEST_INVARIANT;
sl@0
  1411
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1412
	    {
sl@0
  1413
	    OstTrace0( TRACE_FATAL, CRICHTEXT_APPLYPARAGRAPHSTYLEL, "ECharPosBeyondDocument" );
sl@0
  1414
	    }
sl@0
  1415
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1416
	if (aLength < 0)
sl@0
  1417
	    {
sl@0
  1418
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_APPLYPARAGRAPHSTYLEL, "EApplyParaStyleNegativeLength" );
sl@0
  1419
	    }
sl@0
  1420
	__ASSERT_ALWAYS(aLength >= 0, Panic(EApplyParaStyleNegativeLength));
sl@0
  1421
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
  1422
	    {
sl@0
  1423
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_APPLYPARAGRAPHSTYLEL, "ECharPosBeyondDocument" );
sl@0
  1424
	    }
sl@0
  1425
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1426
	if (!StyleListPresent())
sl@0
  1427
	    {
sl@0
  1428
	    OstTrace0( TRACE_FATAL, DUP3_CRICHTEXT_APPLYPARAGRAPHSTYLEL, "ERichTextNotSetForUsingStyles" );
sl@0
  1429
	    }
sl@0
  1430
	__ASSERT_ALWAYS(StyleListPresent(), Panic(ERichTextNotSetForUsingStyles));
sl@0
  1431
sl@0
  1432
	CreateAndGenerateMarkupComponentL();
sl@0
  1433
	iIndex->ApplyParagraphStyleL(aStyle, aPos, aLength, iGlobalCharFormatLayer, aMode);
sl@0
  1434
	SetHasChanged(ETrue);
sl@0
  1435
sl@0
  1436
	__TEST_INVARIANT;
sl@0
  1437
	}
sl@0
  1438
sl@0
  1439
EXPORT_C void CRichText::NotifyStyleChangedL(const CParagraphStyle* aTo, const CParagraphStyle* aFrom)
sl@0
  1440
/** Removes a style from every paragraph in the document to which it applies,
sl@0
  1441
and replaces it with another.
sl@0
  1442
sl@0
  1443
If style aTo is NULL, aFrom is replaced by the global character and paragraph
sl@0
  1444
format layers, so that in effect, style aFrom is removed. Any specific
sl@0
  1445
formatting which has been applied to the paragraphs is retained.
sl@0
  1446
sl@0
  1447
Notes:
sl@0
  1448
sl@0
  1449
This function should be called on the text content object after changing a
sl@0
  1450
style in the style list.
sl@0
  1451
sl@0
  1452
A panic occurs if the rich text object does not use a style list (this can
sl@0
  1453
be tested for using StyleListPresent()).
sl@0
  1454
sl@0
  1455
@param aTo The new paragraph style to apply.
sl@0
  1456
@param aFrom The paragraph style to remove. */
sl@0
  1457
	{
sl@0
  1458
	// Update the rich text index following the change of an applied paragraph style.
sl@0
  1459
	//
sl@0
  1460
	__TEST_INVARIANT;
sl@0
  1461
	if (!StyleListPresent())
sl@0
  1462
	    {
sl@0
  1463
	    OstTrace0( TRACE_FATAL, CRICHTEXT_NOTIFYSTYLECHANGEDL, "ERichTextNotSetForUsingStyles" );
sl@0
  1464
	    }
sl@0
  1465
	__ASSERT_ALWAYS(StyleListPresent(), Panic(ERichTextNotSetForUsingStyles));
sl@0
  1466
sl@0
  1467
	CreateAndGenerateMarkupComponentL();
sl@0
  1468
	iIndex->NotifyStyleChangedL(aTo, aFrom, *iGlobalParaFormatLayer, *iGlobalCharFormatLayer);
sl@0
  1469
	SetHasChanged(ETrue);
sl@0
  1470
sl@0
  1471
	__TEST_INVARIANT;
sl@0
  1472
	}
sl@0
  1473
sl@0
  1474
EXPORT_C const CParaFormatLayer* CRichText::ParagraphStyle(TBool& aStyleChangesOverRange, TInt aPos, TInt aLength) const
sl@0
  1475
/** Gets a pointer to the first paragraph style encountered in the specified
sl@0
  1476
range.
sl@0
  1477
sl@0
  1478
@param aStyleChangesOverRange On return, set to ETrue if more than one paragraph
sl@0
  1479
style is used over the specified range of characters. Otherwise EFalse
sl@0
  1480
@param aPos The document position of the start of the range. Must be valid.
sl@0
  1481
@param aLength The number of characters in the range. Must be greater than
sl@0
  1482
or equal to zero.
sl@0
  1483
@return Pointer to the paragraph style which applies to the paragraph containing
sl@0
  1484
document position aPos. Its type (returned by CParaFormatLayer::Type())
sl@0
  1485
indicates whether this object is a style, or just a paragraph format layer. */
sl@0
  1486
	{
sl@0
  1487
	__TEST_INVARIANT;
sl@0
  1488
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1489
	    {
sl@0
  1490
	    OstTrace0( TRACE_FATAL, CRICHTEXT_PARAGRAPHSTYLE, "ECharPosBeyondDocument" );
sl@0
  1491
	    }
sl@0
  1492
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1493
	if (aLength < 0)
sl@0
  1494
	    {
sl@0
  1495
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_PARAGRAPHSTYLE, "EParagraphStyleNegativeLength" );
sl@0
  1496
	    }
sl@0
  1497
	__ASSERT_ALWAYS(aLength >= 0,Panic(EParagraphStyleNegativeLength));
sl@0
  1498
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
  1499
	    {
sl@0
  1500
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_PARAGRAPHSTYLE, "ECharPosBeyondDocument" );
sl@0
  1501
	    }
sl@0
  1502
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1503
sl@0
  1504
	if (IndexPresent())
sl@0
  1505
		return iIndex->ParagraphStyle(aStyleChangesOverRange, aPos, aLength);
sl@0
  1506
	else
sl@0
  1507
		{
sl@0
  1508
		aStyleChangesOverRange = EFalse;
sl@0
  1509
		return iGlobalParaFormatLayer;
sl@0
  1510
		}
sl@0
  1511
	}
sl@0
  1512
sl@0
  1513
EXPORT_C void CRichText::SetHasChanged(TBool aHasChanged)
sl@0
  1514
/** Sets whether the document's content or formatting has changed. This function
sl@0
  1515
is called with an value of ETrue by all functions which modify the text content
sl@0
  1516
or formatting. Use CEditableText::HasChanged() to test whether the document
sl@0
  1517
has changed.
sl@0
  1518
sl@0
  1519
@param aHasChanged ETrue if the text object has been changed, EFalse if not. */
sl@0
  1520
	{
sl@0
  1521
	// Replaces the base class method of the same name.
sl@0
  1522
	//
sl@0
  1523
	if (aHasChanged && IndexPresent())
sl@0
  1524
		iIndex->DocumentChanged();
sl@0
  1525
	CGlobalText::SetHasChanged(aHasChanged);
sl@0
  1526
	}
sl@0
  1527
sl@0
  1528
EXPORT_C void CRichText::RemoveSpecificParaFormatL(TInt aPos, TInt aLength)
sl@0
  1529
/** Removes all specific paragraph formatting from a range of paragraphs. This
sl@0
  1530
does not remove formatting from the object's global paragraph format layer.
sl@0
  1531
The region affected consists of every paragraph containing one or more
sl@0
  1532
characters in the range covered by document position aPos to aPos+(aLength-1)
sl@0
  1533
inclusive.
sl@0
  1534
sl@0
  1535
A panic occurs in the following situations:
sl@0
  1536
sl@0
  1537
the position is negative,
sl@0
  1538
sl@0
  1539
the length is negative,
sl@0
  1540
sl@0
  1541
the range goes beyond the end of the document
sl@0
  1542
sl@0
  1543
@param aPos The document position of the start of the range.
sl@0
  1544
@param aLength The number of characters in the range. */
sl@0
  1545
	{
sl@0
  1546
	// Removes all specific paragraph formatting from the selected region.
sl@0
  1547
	//
sl@0
  1548
	__ETEXT_WATCH(REMOVE_PARA_FORMAT);
sl@0
  1549
sl@0
  1550
	__TEST_INVARIANT;
sl@0
  1551
sl@0
  1552
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1553
	    {
sl@0
  1554
	    OstTrace0( TRACE_FATAL, CRICHTEXT_REMOVESPECIFICPARAFORMATL, "ECharPosBeyondDocument" );
sl@0
  1555
	    }
sl@0
  1556
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1557
	if (aLength < 0)
sl@0
  1558
	    {
sl@0
  1559
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_REMOVESPECIFICPARAFORMATL, "ERemoveSpecificParaFormatNegativeLength" );
sl@0
  1560
	    }
sl@0
  1561
	__ASSERT_ALWAYS(aLength >= 0, Panic(ERemoveSpecificParaFormatNegativeLength));
sl@0
  1562
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
  1563
	    {
sl@0
  1564
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_REMOVESPECIFICPARAFORMATL, "ECharPosBeyondDocument" );
sl@0
  1565
	    }
sl@0
  1566
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1567
sl@0
  1568
	if (IndexPresent())
sl@0
  1569
		{
sl@0
  1570
		iIndex->RemoveSpecificParaFormatL(aPos, aLength);
sl@0
  1571
		SetHasChanged(ETrue);
sl@0
  1572
		}
sl@0
  1573
sl@0
  1574
	__TEST_INVARIANT;
sl@0
  1575
sl@0
  1576
	__ETEXT_WATCH_END(REMOVE_PARA_FORMAT);
sl@0
  1577
	}
sl@0
  1578
sl@0
  1579
EXPORT_C void CRichText::RemoveSpecificCharFormatL(TInt aPos, TInt aLength)
sl@0
  1580
/** Removes all specific character formatting from a range of characters (does
sl@0
  1581
not remove the formatting which has been taken from the object's global character
sl@0
  1582
format layer). A panic occurs in the following situations:
sl@0
  1583
sl@0
  1584
the position is negative,
sl@0
  1585
sl@0
  1586
the length is negative,
sl@0
  1587
sl@0
  1588
the range goes beyond the end of the document
sl@0
  1589
sl@0
  1590
@param aPos The document position of the start of the region affected.
sl@0
  1591
@param aLength The number of characters in the region affected. If zero, the
sl@0
  1592
function has no effect. */
sl@0
  1593
	{
sl@0
  1594
	__TEST_INVARIANT;
sl@0
  1595
	
sl@0
  1596
	TInt document_length = DocumentLength();
sl@0
  1597
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1598
	    {
sl@0
  1599
	    OstTrace0( TRACE_FATAL, CRICHTEXT_REMOVESPECIFICCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1600
	    }
sl@0
  1601
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1602
	if (aLength < 0)
sl@0
  1603
	    {
sl@0
  1604
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_REMOVESPECIFICCHARFORMATL, "ERemoveSpecificParaFormatNegativeLength" );
sl@0
  1605
	    }
sl@0
  1606
	__ASSERT_ALWAYS(aLength >= 0, Panic(ERemoveSpecificParaFormatNegativeLength));
sl@0
  1607
	if (aPos + (aLength - 1) > DocumentLength())
sl@0
  1608
	    {
sl@0
  1609
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_REMOVESPECIFICCHARFORMATL, "ECharPosBeyondDocument" );
sl@0
  1610
	    }
sl@0
  1611
	__ASSERT_ALWAYS(aPos + (aLength - 1) <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1612
	
sl@0
  1613
	// in correspondance to INC097216, character format removing considers the height of end of document
sl@0
  1614
    // character
sl@0
  1615
    if (aPos + aLength == document_length)
sl@0
  1616
   		{
sl@0
  1617
        aLength++;
sl@0
  1618
        }
sl@0
  1619
sl@0
  1620
	if (aLength > 0 && IndexPresent())
sl@0
  1621
		{
sl@0
  1622
		iIndex->RemoveSpecificCharFormatL(aPos, aLength);
sl@0
  1623
		SetHasChanged(ETrue);
sl@0
  1624
		}
sl@0
  1625
sl@0
  1626
	__TEST_INVARIANT;
sl@0
  1627
	}
sl@0
  1628
sl@0
  1629
EXPORT_C void CRichText::GetChars(TPtrC& aText, TCharFormat& aFormat, TInt aPos) const
sl@0
  1630
/** Gets a constant pointer descriptor to a portion of the text object
sl@0
  1631
with constant character formatting.
sl@0
  1632
sl@0
  1633
The view starts at the document position specified, and ends at: the
sl@0
  1634
last character which shares the same character formatting, orthe end
sl@0
  1635
of the document, orthe end of the segment, if segmented storage is
sl@0
  1636
being usedwhichever occurs first. Also fills a character format object
sl@0
  1637
with the character formatting of the range of characters.
sl@0
  1638
sl@0
  1639
@param aView On return, a constant pointer to a portion of the text.
sl@0
  1640
@param aFormat On return, contains the character formatting of the text.
sl@0
  1641
@param aPos The start position for the view. Must be a valid document
sl@0
  1642
position, or a panic occurs. */
sl@0
  1643
	{
sl@0
  1644
	// Get a run of text and its format, starting at aPos.
sl@0
  1645
	__ETEXT_WATCH(GET_CHARS)
sl@0
  1646
	TCharFormatX format;
sl@0
  1647
	GetTextAndExtendedFormat(aText, format, aPos);
sl@0
  1648
	OverrideFormatForParsersIfApplicable(aText, format, aPos);
sl@0
  1649
	aFormat = format.iCharFormat;
sl@0
  1650
	OverrideFormatOfInlineTextIfApplicable(aText, aFormat, aPos);
sl@0
  1651
	__ETEXT_WATCH_END(GET_CHARS)
sl@0
  1652
	}
sl@0
  1653
sl@0
  1654
sl@0
  1655
void CRichText::GetTextAndExtendedFormat(TPtrC& aText,TCharFormatX& aFormat,TInt aPos) const
sl@0
  1656
	{
sl@0
  1657
	__TEST_INVARIANT;
sl@0
  1658
	TInt documentLength = DocumentLength();
sl@0
  1659
	if (aPos < 0 || aPos > documentLength)
sl@0
  1660
	    {
sl@0
  1661
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETTEXTANDEXTENDEDFORMAT, "ECharPosBeyondDocument" );
sl@0
  1662
	    }
sl@0
  1663
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= documentLength, Panic(ECharPosBeyondDocument));
sl@0
  1664
	if (!IndexPresent())
sl@0
  1665
		{
sl@0
  1666
		aText.Set(Read(aPos));
sl@0
  1667
		iGlobalCharFormatLayer->SenseEffective(aFormat);
sl@0
  1668
		}
sl@0
  1669
	else
sl@0
  1670
		{
sl@0
  1671
		int phrase_length = iIndex->GetChars(aFormat,aPos);
sl@0
  1672
		aText.Set(Read(aPos,phrase_length));
sl@0
  1673
		}
sl@0
  1674
	__TEST_INVARIANT;
sl@0
  1675
	}
sl@0
  1676
sl@0
  1677
EXPORT_C TInt CRichText::GetPictureSizeInTwips(TSize& aSize, TInt aPos) const
sl@0
  1678
/** Gets the size of a picture located at a specified document position.
sl@0
  1679
sl@0
  1680
@param aSize On return, contains the size of the picture located at aPos.
sl@0
  1681
@param aPos Document position of the picture. Must be a valid position.
sl@0
  1682
@return KErrNotFound if there is no picture at the specified document position,
sl@0
  1683
KErrNone if there is. */
sl@0
  1684
	{
sl@0
  1685
	__TEST_INVARIANT;
sl@0
  1686
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1687
	    {
sl@0
  1688
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETPICTURESIZEINTWIPS, "ECharPosBeyondDocument" );
sl@0
  1689
	    }
sl@0
  1690
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1691
sl@0
  1692
	if (IndexPresent())
sl@0
  1693
		return iIndex->GetPictureSizeInTwips(aSize, aPos);
sl@0
  1694
	else
sl@0
  1695
		return KErrNotFound;
sl@0
  1696
	}
sl@0
  1697
sl@0
  1698
EXPORT_C CPicture* CRichText::PictureHandleL(TInt aPos, MLayDoc::TForcePictureLoad aForceLoad) const
sl@0
  1699
/** Gets a pointer to the picture located at a specified document position, if
sl@0
  1700
one exists. If the picture is not in memory, the function loads it (if the
sl@0
  1701
second argument has a value of MLayDoc::EForceLoadTrue).
sl@0
  1702
sl@0
  1703
Note:
sl@0
  1704
sl@0
  1705
In order to load the picture, a picture factory and a store resolver must
sl@0
  1706
have been set.
sl@0
  1707
sl@0
  1708
@param aPos Document position of the picture character. Must be a valid position.
sl@0
  1709
@param aForceLoad If the picture is not loaded into memory,
sl@0
  1710
MLayDoc::EForceLoadTrue loads it using the picture factory;
sl@0
  1711
MLayDoc::EForceLoadFalse does not, and in this case, the function returns NULL.
sl@0
  1712
@return A pointer to the picture located at aPos. NULL if aPos does not specify
sl@0
  1713
a picture character, or if there is a picture at aPos which is not in memory,
sl@0
  1714
and the second argument is MLayDoc::EForceLoadFalse. */
sl@0
  1715
	{
sl@0
  1716
	__ETEXT_WATCH(PICTURE_HANDLE);
sl@0
  1717
sl@0
  1718
	__TEST_INVARIANT;
sl@0
  1719
	if (aPos<0 || aPos>DocumentLength())
sl@0
  1720
	    {
sl@0
  1721
	    OstTrace0( TRACE_FATAL, CRICHTEXT_PICTUREHANDLEL, "ECharPosBeyondDocument" );
sl@0
  1722
	    }
sl@0
  1723
	__ASSERT_ALWAYS(aPos>=0 && aPos<=DocumentLength(),Panic(ECharPosBeyondDocument));
sl@0
  1724
sl@0
  1725
	if (IndexPresent())
sl@0
  1726
		return iIndex->PictureHandleL(aPos, aForceLoad);
sl@0
  1727
	else
sl@0
  1728
		return NULL;
sl@0
  1729
sl@0
  1730
	__ETEXT_WATCH_END(PICTURE_HANDLE);
sl@0
  1731
	}
sl@0
  1732
sl@0
  1733
EXPORT_C void CRichText::GetParagraphFormatL(CParaFormat* aFormat, TInt aPos) const
sl@0
  1734
/** Gets the effective paragraph formatting which applies to the paragraph which
sl@0
  1735
contains a specified document position. On return, aFormat is filled with
sl@0
  1736
values for all paragraph format attributes.
sl@0
  1737
sl@0
  1738
@param aFormat On return, filled with the paragraph's effective paragraph
sl@0
  1739
formatting.
sl@0
  1740
@param aPos Any document position within the paragraph of interest. */
sl@0
  1741
	{
sl@0
  1742
	__ETEXT_WATCH(GET_PARAGRAPH_FORMAT)
sl@0
  1743
sl@0
  1744
	__TEST_INVARIANT;
sl@0
  1745
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1746
	    {
sl@0
  1747
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETPARAGRAPHFORMATL, "ECharPosBeyondDocument" );
sl@0
  1748
	    }
sl@0
  1749
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1750
sl@0
  1751
	if (!IndexPresent())
sl@0
  1752
		CGlobalText::GetParagraphFormatL(aFormat, aPos);
sl@0
  1753
	else
sl@0
  1754
		{
sl@0
  1755
		aFormat->Reset();
sl@0
  1756
		iIndex->GetParagraphFormatL(aFormat, aPos);
sl@0
  1757
		}
sl@0
  1758
sl@0
  1759
	__ETEXT_WATCH_END(GET_PARAGRAPH_FORMAT)
sl@0
  1760
	}
sl@0
  1761
sl@0
  1762
EXPORT_C void CRichText::GetSpecificParagraphFormatL(CParaFormat* aFormat,
sl@0
  1763
													 TParaFormatMask& aMask,
sl@0
  1764
													 TInt aPos) const
sl@0
  1765
// Fills aFormat with the effective Paragraph format attributes for the paragraph
sl@0
  1766
// in which character position aPos is contained.
sl@0
  1767
//
sl@0
  1768
	{
sl@0
  1769
	__ETEXT_WATCH(GET_PARAGRAPH_FORMAT)
sl@0
  1770
sl@0
  1771
	__TEST_INVARIANT;
sl@0
  1772
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1773
	    {
sl@0
  1774
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETSPECIFICPARAGRAPHFORMATL, "ECharPosBeyondDocument" );
sl@0
  1775
	    }
sl@0
  1776
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1777
sl@0
  1778
	aFormat->Reset();
sl@0
  1779
	aMask.ClearAll();
sl@0
  1780
	if (IndexPresent())
sl@0
  1781
		iIndex->GetSpecificParagraphFormatL(aFormat, aMask, aPos);
sl@0
  1782
sl@0
  1783
	__ETEXT_WATCH_END(GET_PARAGRAPH_FORMAT)
sl@0
  1784
	}
sl@0
  1785
sl@0
  1786
EXPORT_C void CRichText::GetParaFormatL(CParaFormat* aFormat, TParaFormatMask& aVaries, TInt aPos, TInt aLength,
sl@0
  1787
										CParaFormat::TParaFormatGetMode aMode) const
sl@0
  1788
/** Gets the effective paragraph formatting which applies to a range of paragraphs.
sl@0
  1789
The region involved is every paragraph containing one or more characters in
sl@0
  1790
the range aPos to aPos+(aLength-1) inclusive. On return, aFormat is filled
sl@0
  1791
with values for all paragraph format attributes and the mask indicates the
sl@0
  1792
values that change over the region, and whose value is therefore indeterminate.
sl@0
  1793
sl@0
  1794
Note:
sl@0
  1795
sl@0
  1796
If aMode has a value of EFixedAttributes, the function cannot leave.
sl@0
  1797
sl@0
  1798
@param aFormat Must not be NULL or a panic occurs. On return, contains the
sl@0
  1799
effective paragraph formatting for the range of paragraphs.
sl@0
  1800
@param aVaries On return, a bitmask indicating which paragraph format attributes
sl@0
  1801
vary over the range of characters selected.
sl@0
  1802
@param aPos The document position of the start of the range.
sl@0
  1803
@param aLength The number of characters in the range.
sl@0
  1804
@param aMode The default, EAllAttributes means that values for all paragraph
sl@0
  1805
format attributes are written to aFormat. EFixedAttributes means that tabs,
sl@0
  1806
bullets and borders are not written to aFormat. */
sl@0
  1807
	{
sl@0
  1808
	// Senses the paragraph format of para(s) covered by the region aPos to aPos+aLength-1.
sl@0
  1809
	// aFormat takes the values of all attributes, and the mask aMask indicates those values that change
sl@0
  1810
	// over the selected region, and are therefore *indeterminate*.
sl@0
  1811
	// Application: seeding paragraph formatting dialogs.
sl@0
  1812
	//
sl@0
  1813
	__TEST_INVARIANT;
sl@0
  1814
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1815
	    {
sl@0
  1816
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETPARAFORMATL, "ECharPosBeyondDocument" );
sl@0
  1817
	    }
sl@0
  1818
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1819
	if (aLength < 0)
sl@0
  1820
	    {
sl@0
  1821
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_GETPARAFORMATL, "EGetParaFormatNegativeLength" );
sl@0
  1822
	    }
sl@0
  1823
	__ASSERT_ALWAYS(aLength >= 0, Panic(EGetParaFormatNegativeLength));
sl@0
  1824
	if (aPos + aLength > DocumentLength())
sl@0
  1825
	    {
sl@0
  1826
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_GETPARAFORMATL, "ECharPosBeyondDocument" );
sl@0
  1827
	    }
sl@0
  1828
	__ASSERT_ALWAYS(aPos + aLength <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1829
sl@0
  1830
	if (IndexPresent())
sl@0
  1831
		iIndex->GetParaFormatL(aFormat, aVaries, aPos, aLength, aMode);
sl@0
  1832
	else
sl@0
  1833
		CGlobalText::GetParaFormatL(aFormat, aVaries, aPos, aLength, aMode);
sl@0
  1834
	}
sl@0
  1835
sl@0
  1836
EXPORT_C void CRichText::GetCharFormat(TCharFormat& aFormat, TCharFormatMask& aVaries, TInt aPos, TInt aLength) const
sl@0
  1837
/** Gets the effective character formatting which applies to a range of
sl@0
  1838
characters. The range of characters involved is from aPos to aPos+(aLength-1)
sl@0
  1839
inclusive. On return, aFormat is filled with values for all character format
sl@0
  1840
attributes, and on return, the mask indicates the values that change over the
sl@0
  1841
region, and whose value is therefore indeterminate.
sl@0
  1842
sl@0
  1843
The length value can be zero. In this case, the character formatting sensed
sl@0
  1844
is that of the character immediately to the left of the position specified.
sl@0
  1845
sl@0
  1846
@param aFormat On return, contains the character format values for the range
sl@0
  1847
of characters.
sl@0
  1848
@param aVaries On return, indicates which character format attributes vary
sl@0
  1849
over the range.
sl@0
  1850
@param aPos Document position of the start of the range. Must be valid or a
sl@0
  1851
panic occurs.
sl@0
  1852
@param aLength Number of characters in the range. Must be greater than or equal
sl@0
  1853
to zero, or a panic occurs. */
sl@0
  1854
	{
sl@0
  1855
	// Senses the character formatting of the phrase(s) covered by the region aPos to aPos+aLength-1.
sl@0
  1856
	// May be called with zero length.
sl@0
  1857
	// aFormat takes the values of all character format attributes, and the mask aMask indicates those
sl@0
  1858
	// values that change over the selected region, and are therefore *indeterminate*.
sl@0
  1859
	// Application: seeding character formatting dialogs.
sl@0
  1860
	//
sl@0
  1861
	TCharFormatX format;
sl@0
  1862
	TCharFormatXMask varies;
sl@0
  1863
	GetExtendedCharFormat(format, varies, aPos, aLength);
sl@0
  1864
	aFormat = format.iCharFormat;
sl@0
  1865
	varies.ClearExtendedAttribs();
sl@0
  1866
	aVaries = varies;
sl@0
  1867
	}
sl@0
  1868
sl@0
  1869
sl@0
  1870
void CRichText::GetExtendedCharFormat(TCharFormatX& aFormat, TCharFormatXMask& aVaries, TInt aPos, TInt aLength) const
sl@0
  1871
	{
sl@0
  1872
	__TEST_INVARIANT;
sl@0
  1873
	int document_length = DocumentLength();
sl@0
  1874
	if (aPos < 0 || aPos > document_length)
sl@0
  1875
	    {
sl@0
  1876
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETEXTENDEDCHARFORMAT, "ECharPosBeyondDocument" );
sl@0
  1877
	    }
sl@0
  1878
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= document_length, Panic(ECharPosBeyondDocument));
sl@0
  1879
	if (aLength < 0)
sl@0
  1880
	    {
sl@0
  1881
	    OstTrace0( TRACE_FATAL, DUP1_CRICHTEXT_GETEXTENDEDCHARFORMAT, "EGetCharFormatNegativeLength" );
sl@0
  1882
	    }
sl@0
  1883
	__ASSERT_ALWAYS(aLength >= 0, Panic(EGetCharFormatNegativeLength));
sl@0
  1884
	if (aPos + aLength - 1 > document_length)
sl@0
  1885
	    {
sl@0
  1886
	    OstTrace0( TRACE_FATAL, DUP2_CRICHTEXT_GETEXTENDEDCHARFORMAT, "ECharPosBeyondDocument" );
sl@0
  1887
	    }
sl@0
  1888
	__ASSERT_ALWAYS(aPos + aLength - 1 <= document_length, Panic(ECharPosBeyondDocument));
sl@0
  1889
sl@0
  1890
	if (IndexPresent())
sl@0
  1891
		iIndex->GetCharFormat(aFormat, aVaries, aPos, aLength);
sl@0
  1892
	else
sl@0
  1893
		{
sl@0
  1894
		iGlobalCharFormatLayer->SenseEffective(aFormat);
sl@0
  1895
		aVaries.ClearAll();
sl@0
  1896
		}
sl@0
  1897
	}
sl@0
  1898
sl@0
  1899
sl@0
  1900
void CRichText::GetSpecificCharFormatLeftRight(TCharFormat& aFormat,
sl@0
  1901
									TCharFormatMask& aMask,
sl@0
  1902
									TInt aPos,
sl@0
  1903
									TBool aLeft) const
sl@0
  1904
	{
sl@0
  1905
	__ETEXT_WATCH(GET_SPECIFIC_CHARS);
sl@0
  1906
sl@0
  1907
	__TEST_INVARIANT;
sl@0
  1908
sl@0
  1909
	if (aPos < 0 || aPos > DocumentLength())
sl@0
  1910
	    {
sl@0
  1911
	    OstTrace0( TRACE_FATAL, CRICHTEXT_GETSPECIFICCHARFORMATLEFTRIGHT, "ECharPosBeyondDocument" );
sl@0
  1912
	    }
sl@0
  1913
	__ASSERT_ALWAYS(aPos >= 0 && aPos <= DocumentLength(), Panic(ECharPosBeyondDocument));
sl@0
  1914
sl@0
  1915
	if (IndexPresent())
sl@0
  1916
		{
sl@0
  1917
		TCharFormatX format;
sl@0
  1918
		TCharFormatXMask mask;
sl@0
  1919
		iIndex->GetSpecificCharFormatDirection(format, mask, aPos, aLeft);
sl@0
  1920
		aFormat = format.iCharFormat;
sl@0
  1921
		mask.ClearExtendedAttribs();
sl@0
  1922
		aMask = mask;
sl@0
  1923
		}
sl@0
  1924
sl@0
  1925
	__ETEXT_WATCH_END(GET_SPECIFIC_CHARS);
sl@0
  1926
	}
sl@0
  1927
sl@0
  1928
EXPORT_C void CRichText::GetSpecificCharFormat(TCharFormat& aFormat, TCharFormatMask& aMask, TInt aPos) const
sl@0
  1929
/** Gets the specific character formatting which applies to the character to the
sl@0
  1930
left of a document position. Specific formatting is just the formatting which
sl@0
  1931
has been applied over the object's global format layers - it is not the
sl@0
  1932
same as the effective formatting.
sl@0
  1933
sl@0
  1934
@param aFormat On return contains the specific character formatting which
sl@0
  1935
applies to the character to the left of the document position.
sl@0
  1936
@param aMask On return, indicates which character format attributes have been
sl@0
  1937
applied specifically, (not taken from the global layers).
sl@0
  1938
@param aPos The document position. Must be valid or a panic occurs. */
sl@0
  1939
	{
sl@0
  1940
	// Return the format attributes store in the specific layer only, for the specified document position.
sl@0
  1941
	// THIS IS NOT THE EFFECTIVE FORMAT, BUT THE SPECIFIC FORMATTING ONLY.
sl@0
  1942
	//
sl@0
  1943
	GetSpecificCharFormatLeftRight(aFormat, aMask, aPos, ETrue);
sl@0
  1944
	}
sl@0
  1945
sl@0
  1946
EXPORT_C void CRichText::GetSpecificCharFormatRight(TCharFormat& aFormat, TCharFormatMask& aMask, TInt aPos) const
sl@0
  1947
/** Gets the specific character formatting which applies to the character logically
sl@0
  1948
after a document position. Note that this is not necessarily the character
sl@0
  1949
to the right of the document position, because right to left text is supported.
sl@0
  1950
sl@0
  1951
Specific formatting is just the formatting which has been applied over the
sl@0
  1952
object's global format layers it is not the same as the effective formatting.
sl@0
  1953
sl@0
  1954
@param aFormat On return, contains the specific character formatting which
sl@0
  1955
applies to the character logically after the document position.
sl@0
  1956
@param aMask On return, indicates which character format attributes have been
sl@0
  1957
applied specifically, (not taken from the global layers).
sl@0
  1958
@param aPos The document position. Must be valid or a panic occurs. */
sl@0
  1959
	{
sl@0
  1960
	GetSpecificCharFormatLeftRight(aFormat, aMask, aPos, EFalse);
sl@0
  1961
	}
sl@0
  1962
sl@0
  1963
TBool CRichText::IndexPresent()const
sl@0
  1964
// Returns ETrue if the rich text index is present, otherwise
sl@0
  1965
// returns EFalse.
sl@0
  1966
//
sl@0
  1967
	{
sl@0
  1968
	return iIndex.IsPtr() && iIndex.AsPtr();
sl@0
  1969
	}
sl@0
  1970
sl@0
  1971
sl@0
  1972
EXPORT_C TInt CRichText::PictureCount() const
sl@0
  1973
/** Gets a count of the number of pictures contained in the rich text object.
sl@0
  1974
sl@0
  1975
@return The picture count. */
sl@0
  1976
	{return (IndexPresent()) ? iIndex->iPictureCount : 0;}
sl@0
  1977
sl@0
  1978
sl@0
  1979
void CRichText::SetParaTypeIsSingle(TBool aBool)
sl@0
  1980
	{
sl@0
  1981
	if (aBool)
sl@0
  1982
		iFlags |= KParaTypeIsSingle;
sl@0
  1983
	else
sl@0
  1984
		iFlags &= ~KParaTypeIsSingle;
sl@0
  1985
	}
sl@0
  1986
sl@0
  1987
sl@0
  1988
TBool CRichText::ParaTypeIsSingle() const
sl@0
  1989
	{return iFlags & KParaTypeIsSingle;}
sl@0
  1990
sl@0
  1991
sl@0
  1992
EXPORT_C void CRichText::AppendTakingSolePictureOwnershipL(const CRichText& aSource)
sl@0
  1993
/** Appends a rich text object to this one. The text is appended immediately after
sl@0
  1994
the end-of-text paragraph delimiter. The incoming text's formatting is set
sl@0
  1995
to be based on the global formatting of this rich text object.
sl@0
  1996
sl@0
  1997
Notes:
sl@0
  1998
sl@0
  1999
If this rich text object is empty (e.g. because it is newly initialised, or
sl@0
  2000
has been reset), then the end-of-text delimiter of the incoming rich text
sl@0
  2001
is not appended. This avoids the possibility of having a trailing paragraph
sl@0
  2002
delimiter, giving one more empty line than would typically be desired.
sl@0
  2003
sl@0
  2004
If the incoming rich text contains pictures which have been loaded into memory,
sl@0
  2005
their sole ownership is transferred to the current rich text object. In aSource,
sl@0
  2006
these picture handles are set to NULL.
sl@0
  2007
sl@0
  2008
@param aSource The rich text object to append. */
sl@0
  2009
	{
sl@0
  2010
	// Appends the specified rich text object to this one.
sl@0
  2011
	// If this rich text is empty or has been newly initialised or reset, then the final paragraph delimiter
sl@0
  2012
	// of the incoming rich text is NOT appended, thus avoiding the prospect of having a trailing, empty
sl@0
  2013
	// paragraph delimiter, giving one more empty line than would typically be desired.
sl@0
  2014
	//
sl@0
  2015
	__ETEXT_WATCH(APPEND_RICH_TEXT);
sl@0
  2016
sl@0
  2017
	// Append the text
sl@0
  2018
	const TInt thisDocumentLength = DocumentLength();
sl@0
  2019
	MPictureFactory* factory = NULL;
sl@0
  2020
	MRichTextStoreResolver* resolver = NULL;
sl@0
  2021
	if (thisDocumentLength == 0)  // this is an empty document
sl@0
  2022
		{
sl@0
  2023
		factory = iPictureFactory;
sl@0
  2024
		resolver = iStoreResolver;
sl@0
  2025
		Reset();  // destroy the markup component if it exists.  Makes the job easier
sl@0
  2026
		}
sl@0
  2027
	//
sl@0
  2028
	PrepareAppendMarkupL(aSource);  // requires no rollback - objects in a (bigger) stable state
sl@0
  2029
	//
sl@0
  2030
	TRAPD(ret, DoAppendTakingSolePictureOwnershipL(aSource));
sl@0
  2031
	if (ret != KErrNone)
sl@0
  2032
		{
sl@0
  2033
		CPlainText::Delete(thisDocumentLength,DocumentLength() - thisDocumentLength);
sl@0
  2034
		User::Leave(ret);
sl@0
  2035
		}
sl@0
  2036
	if (thisDocumentLength == 0)
sl@0
  2037
		{
sl@0
  2038
		DeleteParagraph(0, 1);   // remove excess first paragraph from empty documents
sl@0
  2039
		SetPictureFactory(factory, resolver);
sl@0
  2040
		}
sl@0
  2041
	int new_length = DocumentLength() - thisDocumentLength;
sl@0
  2042
	iParserData->MergeRange(thisDocumentLength,0,new_length);
sl@0
  2043
	CallEditObserver(thisDocumentLength,new_length);
sl@0
  2044
sl@0
  2045
	__TEST_INVARIANT;
sl@0
  2046
sl@0
  2047
	__ETEXT_WATCH_END(APPEND_RICH_TEXT);
sl@0
  2048
	}
sl@0
  2049
sl@0
  2050
sl@0
  2051
void CRichText::DoAppendTakingSolePictureOwnershipL(const CRichText& aSource)
sl@0
  2052
	{
sl@0
  2053
	TInt lengthRemaining = aSource.DocumentLength() + 1;  // we want to append the para delimiters also!
sl@0
  2054
	TInt consumed = 0;
sl@0
  2055
	FOREVER
sl@0
  2056
		{
sl@0
  2057
		TPtrC view = aSource.Read(consumed);
sl@0
  2058
		if (view.Length() > lengthRemaining)
sl@0
  2059
			view.Set(view.Ptr(), lengthRemaining);
sl@0
  2060
		CPlainText::DoPtInsertL(DocumentLength() + 1, view);  // insert AFTER the final paragraph delimiter
sl@0
  2061
		TInt viewLength = view.Length();
sl@0
  2062
		lengthRemaining -= viewLength;
sl@0
  2063
		if (lengthRemaining == 0)
sl@0
  2064
			break;
sl@0
  2065
		consumed += viewLength;
sl@0
  2066
		}
sl@0
  2067
sl@0
  2068
	if (IndexPresent())
sl@0
  2069
		{
sl@0
  2070
		if (!aSource.IndexPresent())
sl@0
  2071
		    {
sl@0
  2072
		    OstTrace0( TRACE_DUMP, DUP1_CRICHTEXT_DOAPPENDTAKINGSOLEPICTUREOWNERSHIPL, "Invariant" );
sl@0
  2073
		    }
sl@0
  2074
		__ASSERT_DEBUG(aSource.IndexPresent(), User::Invariant());       // PrepareAppend should have sorted this
sl@0
  2075
sl@0
  2076
		TGlobalLayerInfoAppend info(GlobalParaFormatLayer(), GlobalCharFormatLayer(), aSource.GlobalParaFormatLayer(), aSource.GlobalCharFormatLayer());
sl@0
  2077
		iIndex->AppendTakingSolePictureOwnershipL(aSource.iIndex, info);
sl@0
  2078
		}
sl@0
  2079
sl@0
  2080
	__TEST_INVARIANT;
sl@0
  2081
	}
sl@0
  2082
sl@0
  2083
sl@0
  2084
void CRichText::PrepareAppendMarkupL(const CRichText& aSource)
sl@0
  2085
// Guarantees that both the component and aggregate objects have a valid markup component.
sl@0
  2086
//
sl@0
  2087
	{
sl@0
  2088
	if (aSource.HasMarkupData() && !HasMarkupData())
sl@0
  2089
		CreateAndGenerateMarkupComponentL();
sl@0
  2090
	else if (IndexPresent())
sl@0
  2091
		CONST_CAST(CRichText&, aSource).CreateAndGenerateMarkupComponentL();
sl@0
  2092
	}
sl@0
  2093
sl@0
  2094
sl@0
  2095
EXPORT_C void CRichText::AppendParagraphL(TInt aReplicas)
sl@0
  2096
/** Appends one or more empty paragraphs to the document. The new paragraphs take
sl@0
  2097
on the formatting specified in the global format layers.
sl@0
  2098
sl@0
  2099
@param aReplicas The number of empty paragraphs to append to the document.
sl@0
  2100
By default, a single paragraph. */
sl@0
  2101
	{
sl@0
  2102
	// Inserts an empty paragraph at the end of the document.
sl@0
  2103
	// The new paragraph takes on the format as described
sl@0
  2104
	// by the global format layers.
sl@0
  2105
	//
sl@0
  2106
	__ETEXT_WATCH(APPEND_PARAGRAPH);
sl@0
  2107
sl@0
  2108
	__TEST_INVARIANT;
sl@0
  2109
sl@0
  2110
	TInt documentLength = DocumentLength();
sl@0
  2111
	if (aReplicas == 1)
sl@0
  2112
		CPlainText::InsertL(documentLength, CEditableText::EParagraphDelimiter);
sl@0
  2113
	else
sl@0
  2114
		{
sl@0
  2115
		HBufC* bb = HBufC::NewLC(aReplicas);
sl@0
  2116
		TPtr buf = bb->Des();
sl@0
  2117
		buf.Fill(CEditableText::EParagraphDelimiter, aReplicas);
sl@0
  2118
		CPlainText::InsertL(documentLength, buf);
sl@0
  2119
		CleanupStack::PopAndDestroy();  // bb
sl@0
  2120
		}
sl@0
  2121
	//
sl@0
  2122
	if (IndexPresent())
sl@0
  2123
		{
sl@0
  2124
		TRAPD(ret,
sl@0
  2125
		iIndex->AppendParagraphL(iGlobalParaFormatLayer, iGlobalCharFormatLayer, aReplicas));
sl@0
  2126
		if (ret != KErrNone)
sl@0
  2127
			{
sl@0
  2128
			CPlainText::Delete(DocumentLength() - aReplicas, aReplicas);
sl@0
  2129
			User::Leave(ret);
sl@0
  2130
			}
sl@0
  2131
		SetHasChanged(ETrue);
sl@0
  2132
		}
sl@0
  2133
sl@0
  2134
	int new_length = DocumentLength() - documentLength;
sl@0
  2135
	iParserData->MergeRange(documentLength,0,new_length);
sl@0
  2136
	CallEditObserver(documentLength,new_length);
sl@0
  2137
	__TEST_INVARIANT;
sl@0
  2138
sl@0
  2139
	__ETEXT_WATCH_END(APPEND_PARAGRAPH);
sl@0
  2140
	}