os/textandloc/textrendering/texthandling/sfields/FLDSET.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 "FLDSET.H"
sl@0
    23
#include "FLDDEF.H"
sl@0
    24
#include "FLDARRAY.H"
sl@0
    25
#include "FLDSTD.H"
sl@0
    26
sl@0
    27
sl@0
    28
EXPORT_C void CTextFieldSet::__DbgTestInvariant()const
sl@0
    29
// Provides class invariants.  Explanations below:
sl@0
    30
//
sl@0
    31
	{
sl@0
    32
#ifdef _DEBUG
sl@0
    33
	if (!iFieldArray)
sl@0
    34
		return;
sl@0
    35
// check that every entry in the array has length >=0 and that all but the last have valid header handles
sl@0
    36
	TBool valid = ETrue;
sl@0
    37
	TInt index = iFieldArray->Count()-1;
sl@0
    38
	if (index<0)
sl@0
    39
		valid = EFalse;
sl@0
    40
	else 
sl@0
    41
		{// check last entry: should have null fieldHeader, zero field length and non-negative PreFieldLen
sl@0
    42
		if ( (*iFieldArray)[index].iFieldHeader.iField
sl@0
    43
		||(*iFieldArray)[index].iFieldValueLen!=0
sl@0
    44
		||(*iFieldArray)[index].iPreFieldLen<0 )
sl@0
    45
			{
sl@0
    46
			valid = EFalse;
sl@0
    47
			}
sl@0
    48
		index--;
sl@0
    49
		// check the other entries: should have non-null fieldHeader and non-negative field length and PreFieldLen
sl@0
    50
		for (; (index>=0)&&(valid) ; index--)
sl@0
    51
			{
sl@0
    52
			if ((*iFieldArray)[index].iFieldValueLen<0
sl@0
    53
				||(*iFieldArray)[index].iPreFieldLen<0 )
sl@0
    54
					{
sl@0
    55
					valid = EFalse;
sl@0
    56
					}
sl@0
    57
			}
sl@0
    58
		}
sl@0
    59
	__ASSERT_ALWAYS(valid,User::Invariant());
sl@0
    60
#endif
sl@0
    61
	}
sl@0
    62
sl@0
    63
sl@0
    64
EXPORT_C CTextFieldSet* CTextFieldSet::NewL(TInt aDocumentLength)
sl@0
    65
	{
sl@0
    66
	CTextFieldSet* self=new(ELeave) CTextFieldSet();
sl@0
    67
	CleanupStack::PushL(self);
sl@0
    68
	self->ConstructL(aDocumentLength);
sl@0
    69
	CleanupStack::Pop();
sl@0
    70
	return self;
sl@0
    71
	}
sl@0
    72
sl@0
    73
sl@0
    74
EXPORT_C CTextFieldSet* CTextFieldSet::NewL(const MTextFieldFactory* aFactory,const CStreamStore& aFieldStore,TStreamId aStreamId)
sl@0
    75
	{
sl@0
    76
	CTextFieldSet* self=new(ELeave) CTextFieldSet();
sl@0
    77
	CleanupStack::PushL(self);
sl@0
    78
	self->SetFieldFactory(CONST_CAST(MTextFieldFactory*,aFactory));
sl@0
    79
	self->ConstructL(0);
sl@0
    80
	self->DoRestoreL(aFieldStore,aStreamId);
sl@0
    81
	CleanupStack::Pop();
sl@0
    82
	return self;
sl@0
    83
	}
sl@0
    84
sl@0
    85
sl@0
    86
CTextFieldSet::CTextFieldSet()
sl@0
    87
	{}
sl@0
    88
sl@0
    89
sl@0
    90
void CTextFieldSet::ConstructL(TInt aDocumentLength)
sl@0
    91
// Creates an array in which to store all fields
sl@0
    92
// Inserts an initial entry into the array to cover any text that lies after the last field
sl@0
    93
//
sl@0
    94
	{
sl@0
    95
	iFieldArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
sl@0
    96
	AddInitialFieldEntryL(iFieldArray,aDocumentLength);
sl@0
    97
sl@0
    98
	__TEST_INVARIANT;
sl@0
    99
	}
sl@0
   100
sl@0
   101
sl@0
   102
EXPORT_C CTextFieldSet::~CTextFieldSet()
sl@0
   103
	{
sl@0
   104
	delete iRollbackInfo;
sl@0
   105
	if (iFieldArray)
sl@0
   106
		{
sl@0
   107
		TInt fieldCount=iFieldArray->Count();
sl@0
   108
		for (TInt index=fieldCount-1;index>=0;index--)
sl@0
   109
			delete (*iFieldArray)[index].iFieldHeader.iField.AsPtr(); // Delete the textField objects
sl@0
   110
		delete iFieldArray;
sl@0
   111
		}
sl@0
   112
	}
sl@0
   113
sl@0
   114
sl@0
   115
EXPORT_C void CTextFieldSet::SetFieldFactory(MTextFieldFactory* aFactory)
sl@0
   116
	{
sl@0
   117
	iFieldFactory = aFactory;
sl@0
   118
	}
sl@0
   119
sl@0
   120
sl@0
   121
EXPORT_C MTextFieldFactory* CTextFieldSet::FieldFactory()const
sl@0
   122
	{
sl@0
   123
	return iFieldFactory;
sl@0
   124
	}
sl@0
   125
sl@0
   126
sl@0
   127
void CTextFieldSet::InsertEntryL(TInt aIndex,TTextFieldEntry& aEntry)
sl@0
   128
	{
sl@0
   129
	InsertEntryL(aIndex,aEntry,iFieldArray);
sl@0
   130
	}
sl@0
   131
sl@0
   132
sl@0
   133
void CTextFieldSet::InsertEntryL(TInt aIndex,TTextFieldEntry& aEntry,CArrayFixSeg<TTextFieldEntry>* aArray)
sl@0
   134
// if this function leaves it will be as if it had never been called...
sl@0
   135
//
sl@0
   136
	{
sl@0
   137
	if (aEntry.iFieldHeader.iField.IsPtr())
sl@0
   138
		CleanupStack::PushL(aEntry.iFieldHeader.iField.AsPtr());
sl@0
   139
	aArray->InsertL(aIndex,aEntry); // insert new entry
sl@0
   140
	if (aEntry.iFieldHeader.iField.IsPtr())
sl@0
   141
		CleanupStack::Pop();
sl@0
   142
	}
sl@0
   143
sl@0
   144
sl@0
   145
void CTextFieldSet::AppendEntryL(TTextFieldEntry& aEntry)
sl@0
   146
	{
sl@0
   147
	AppendEntryL(aEntry,iFieldArray);
sl@0
   148
	}
sl@0
   149
sl@0
   150
sl@0
   151
void CTextFieldSet::AppendEntryL(TTextFieldEntry& aEntry,CArrayFixSeg<TTextFieldEntry>* aArray)
sl@0
   152
	{
sl@0
   153
	if (aEntry.iFieldHeader.iField.IsPtr())
sl@0
   154
		CleanupStack::PushL(aEntry.iFieldHeader.iField.AsPtr());
sl@0
   155
	aArray->AppendL(aEntry); // insert new entry
sl@0
   156
	if (aEntry.iFieldHeader.iField.IsPtr())
sl@0
   157
		CleanupStack::Pop();
sl@0
   158
	}
sl@0
   159
sl@0
   160
sl@0
   161
TInt CTextFieldSet::EntryLen(TInt aIndex)const
sl@0
   162
	{
sl@0
   163
	return EntryLen((*iFieldArray)[aIndex]);
sl@0
   164
	}
sl@0
   165
sl@0
   166
sl@0
   167
TInt CTextFieldSet::EntryLen(const TTextFieldEntry& aEntry)const
sl@0
   168
	{
sl@0
   169
	return aEntry.iPreFieldLen+aEntry.iFieldValueLen;
sl@0
   170
	}
sl@0
   171
sl@0
   172
sl@0
   173
TTextFieldEntry CTextFieldSet::SplitEntry(TInt aIndex,TInt aOffset, TInt aRange)const
sl@0
   174
// Splits the entry aIndex, returning the part demarked by the offset (from the start of the entry) and the range
sl@0
   175
//
sl@0
   176
	{
sl@0
   177
	__TEST_INVARIANT;
sl@0
   178
sl@0
   179
	__ASSERT_DEBUG((aIndex>=0)&&(aIndex<iFieldArray->Count()),Panic(EIndexOutOfRange));
sl@0
   180
	TInt entryLength = EntryLen(aIndex);
sl@0
   181
	__ASSERT_DEBUG((aOffset>=0)&&(aOffset<=entryLength),Panic(EPosOutOfRange));
sl@0
   182
	__ASSERT_DEBUG((aRange>=0),Panic(ENegativeRange));
sl@0
   183
sl@0
   184
	if ((aOffset+aRange)>entryLength)
sl@0
   185
		aRange = entryLength-aOffset; // scale range down to entry size if neccessary
sl@0
   186
sl@0
   187
	if ((aOffset==0)&&(aRange==entryLength))
sl@0
   188
		return (*iFieldArray)[aIndex]; //entry does not need to be split
sl@0
   189
sl@0
   190
	TInt charsCopied=0;
sl@0
   191
	TTextFieldEntry entry;
sl@0
   192
	entry.iPreFieldLen = 0;
sl@0
   193
	entry.iFieldValueLen = 0;
sl@0
   194
	entry.iFieldHeader.iField = NULL;
sl@0
   195
	if (aOffset<(*iFieldArray)[aIndex].iPreFieldLen)
sl@0
   196
		{// At least some of the pre needs to be copied
sl@0
   197
		entry.iPreFieldLen = (*iFieldArray)[aIndex].iPreFieldLen-aOffset;
sl@0
   198
		if ((entry.iPreFieldLen)>aRange)
sl@0
   199
			entry.iPreFieldLen = aRange;
sl@0
   200
		charsCopied = entry.iPreFieldLen;
sl@0
   201
		}
sl@0
   202
sl@0
   203
	if (charsCopied<aRange)
sl@0
   204
		{// more to do, so at least some of the field needs to be copied
sl@0
   205
		if ((aOffset+aRange)==entryLength)
sl@0
   206
			{// whole field in range so copy it
sl@0
   207
			entry.iFieldValueLen = (*iFieldArray)[aIndex].iFieldValueLen;
sl@0
   208
			entry.iFieldHeader = (*iFieldArray)[aIndex].iFieldHeader;
sl@0
   209
			}
sl@0
   210
		else
sl@0
   211
			// only part of field in range so convert it to text
sl@0
   212
			entry.iPreFieldLen += aRange-charsCopied;
sl@0
   213
		}
sl@0
   214
	return entry;
sl@0
   215
	}
sl@0
   216
sl@0
   217
sl@0
   218
void CTextFieldSet::AddInitialFieldEntryL(CArrayFixSeg<TTextFieldEntry>* aArray,TInt aDocumentLength)
sl@0
   219
// Add initial entry
sl@0
   220
	{
sl@0
   221
	TTextFieldEntry initialEntry;
sl@0
   222
	initialEntry.iPreFieldLen = aDocumentLength;
sl@0
   223
	initialEntry.iFieldValueLen = 0;
sl@0
   224
	initialEntry.iFieldHeader.iFieldType = KNullUid;
sl@0
   225
	initialEntry.iFieldHeader.iField = NULL;
sl@0
   226
	aArray->AppendL(initialEntry);
sl@0
   227
	}
sl@0
   228
sl@0
   229
sl@0
   230
EXPORT_C void CTextFieldSet::Reset()
sl@0
   231
// deletes all fields (but not corresponding text) and reinitialises the field array
sl@0
   232
//
sl@0
   233
	{
sl@0
   234
	__TEST_INVARIANT;
sl@0
   235
sl@0
   236
	for (TInt index=FieldCount()-1 ; index>=0 ; index--)
sl@0
   237
		delete (*iFieldArray)[index].iFieldHeader.iField.AsPtr(); // Delete the textField objects
sl@0
   238
	iFieldArray->Reset();
sl@0
   239
	AddInitialFieldEntryL(iFieldArray,0); // cannot leave in this context
sl@0
   240
	iFieldArray->Compress(); // compress array
sl@0
   241
sl@0
   242
	__TEST_INVARIANT;
sl@0
   243
	}
sl@0
   244
sl@0
   245
sl@0
   246
EXPORT_C CTextField* CTextFieldSet::NewFieldL(TUid aFieldType)
sl@0
   247
	{
sl@0
   248
	if (iFieldFactory)
sl@0
   249
		return iFieldFactory->NewFieldL(aFieldType);
sl@0
   250
	else
sl@0
   251
		return NULL;
sl@0
   252
	}
sl@0
   253
sl@0
   254
sl@0
   255
EXPORT_C TInt CTextFieldSet::InsertFieldL(TInt aPos,CTextField* aField,TUid aFieldType)
sl@0
   256
// Inserts a field header aField at aPos (aField should be declared on the heap)
sl@0
   257
// The field initially has zero length: Update must be called afterward
sl@0
   258
//
sl@0
   259
	{
sl@0
   260
	__TEST_INVARIANT;
sl@0
   261
	__ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
sl@0
   262
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   263
	__ASSERT_ALWAYS(aField,Panic(ENoTextField));
sl@0
   264
sl@0
   265
	TInt errLevel=KErrAlreadyExists;
sl@0
   266
	TTextFieldEntry entry;
sl@0
   267
	entry.iFieldHeader.iField = aField;
sl@0
   268
	entry.iFieldHeader.iFieldType = aFieldType;
sl@0
   269
	entry.iFieldValueLen = 0;
sl@0
   270
	TInt index; TInt offset;
sl@0
   271
	TBool inField = InField(aPos,index,offset);
sl@0
   272
	if (!inField)
sl@0
   273
		{// not inserting into a field
sl@0
   274
		entry.iPreFieldLen = offset;
sl@0
   275
		iFieldArray->InsertL(index,entry); // insert new entry
sl@0
   276
		(*iFieldArray)[index+1].iPreFieldLen -= offset; // update old entry
sl@0
   277
		errLevel = KErrNone;
sl@0
   278
		}
sl@0
   279
	else if (offset==0)
sl@0
   280
		{// at start of field
sl@0
   281
		entry.iPreFieldLen = (*iFieldArray)[index].iPreFieldLen;
sl@0
   282
		iFieldArray->InsertL(index,entry); // insert new entry
sl@0
   283
		(*iFieldArray)[index+1].iPreFieldLen = 0; // update old entry
sl@0
   284
		errLevel = KErrNone;
sl@0
   285
		}
sl@0
   286
sl@0
   287
	__TEST_INVARIANT;
sl@0
   288
	return errLevel;
sl@0
   289
	}
sl@0
   290
sl@0
   291
sl@0
   292
EXPORT_C const CTextField* CTextFieldSet::TextField(TInt aPos)const
sl@0
   293
// Return a handle to the concrete field at document position aPos.
sl@0
   294
// Returns NULL if there is no field at position aPos.
sl@0
   295
//
sl@0
   296
	{
sl@0
   297
	__TEST_INVARIANT;
sl@0
   298
sl@0
   299
	TInt index=-1;
sl@0
   300
	TInt offset=0;
sl@0
   301
	TBool inField=InField(aPos,index,offset);
sl@0
   302
	if (!inField)
sl@0
   303
		return NULL;
sl@0
   304
	TSwizzle<CTextField> field=(*iFieldArray)[index].iFieldHeader.iField;
sl@0
   305
	__ASSERT_DEBUG(field.IsPtr(),User::Invariant());
sl@0
   306
	return field.AsPtr();
sl@0
   307
	}
sl@0
   308
sl@0
   309
sl@0
   310
EXPORT_C TInt CTextFieldSet::RemoveField(TInt aPos)
sl@0
   311
// Removes the field from the array, adding it's content to the "before" of the next field.
sl@0
   312
// After this function is called the text the field contained should be deleted. If this does
sl@0
   313
//   not happen this function acts as a "ConvertFieldToText()"
sl@0
   314
//
sl@0
   315
	{
sl@0
   316
	__TEST_INVARIANT;
sl@0
   317
	__ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
sl@0
   318
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   319
sl@0
   320
	TInt errLevel=KErrNone;
sl@0
   321
	TInt index; TInt offset;
sl@0
   322
	if (!InField(aPos,index,offset))
sl@0
   323
		errLevel = KErrNotFound;
sl@0
   324
	else
sl@0
   325
		// delete the field entry tidily
sl@0
   326
		DeleteFieldEntry(index);
sl@0
   327
sl@0
   328
	__TEST_INVARIANT;
sl@0
   329
	return errLevel;
sl@0
   330
	}
sl@0
   331
sl@0
   332
sl@0
   333
EXPORT_C TInt CTextFieldSet::FieldCount() const
sl@0
   334
	{
sl@0
   335
	__TEST_INVARIANT;
sl@0
   336
sl@0
   337
	// return count-1 because there is always an entry in the array for the text at the
sl@0
   338
	//  end of the document, after the last field (maybe just the end of doc char)
sl@0
   339
	return (iFieldArray->Count()-1);
sl@0
   340
	}
sl@0
   341
sl@0
   342
sl@0
   343
EXPORT_C TInt CTextFieldSet::CharCount() const
sl@0
   344
// returns the number of characters in the document according to the field array
sl@0
   345
	{
sl@0
   346
	__TEST_INVARIANT;
sl@0
   347
sl@0
   348
	TInt charCount = 0;
sl@0
   349
	for (TInt index=FieldCount() ; index>=0 ; index--)
sl@0
   350
		charCount += (*iFieldArray)[index].iPreFieldLen+(*iFieldArray)[index].iFieldValueLen;
sl@0
   351
	return charCount;
sl@0
   352
	}
sl@0
   353
sl@0
   354
sl@0
   355
EXPORT_C TBool CTextFieldSet::FindFields(TInt aPos) const
sl@0
   356
// Return ETrue if aPos is in a field
sl@0
   357
//
sl@0
   358
	{
sl@0
   359
	TFindFieldInfo dummy;
sl@0
   360
	return (FindFields(dummy,aPos,0));
sl@0
   361
	}
sl@0
   362
sl@0
   363
sl@0
   364
EXPORT_C TBool CTextFieldSet::FindFields(TFindFieldInfo& aInfo,TInt aPos, TInt aRange) const
sl@0
   365
// Check whether aPos is in a field, then check whether any fields start in aRange
sl@0
   366
//
sl@0
   367
	{
sl@0
   368
	__TEST_INVARIANT;
sl@0
   369
	__ASSERT_ALWAYS(aRange>=0,Panic(ENegativeRange));
sl@0
   370
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutsideDoc));
sl@0
   371
	__ASSERT_ALWAYS(aPos+aRange<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   372
	
sl@0
   373
	aInfo.iFieldCountInRange = 0;
sl@0
   374
	aInfo.iFirstFieldLen = 0;
sl@0
   375
	aInfo.iFirstFieldPos = 0;
sl@0
   376
	TInt pos=aPos; // position in doc
sl@0
   377
	// are we in a field to begin with?
sl@0
   378
	TInt index; TInt offset;
sl@0
   379
	if (InField(aPos,index,offset)) 
sl@0
   380
		{
sl@0
   381
		aInfo.iFieldCountInRange++;
sl@0
   382
		aInfo.iFirstFieldLen = (*iFieldArray)[index].iFieldValueLen;
sl@0
   383
		aInfo.iFirstFieldPos = aPos-offset;
sl@0
   384
		pos += (*iFieldArray)[index].iFieldValueLen-offset+(*iFieldArray)[index+1].iPreFieldLen;
sl@0
   385
		index++;
sl@0
   386
		}
sl@0
   387
	else
sl@0
   388
		pos += (*iFieldArray)[index].iPreFieldLen-offset;
sl@0
   389
	// step through array until aRange
sl@0
   390
	while (pos<aPos+aRange)
sl@0
   391
		// When entering this loop we're always at the beginning of a field value
sl@0
   392
		// set aFieldEntry to first field found (if fieldcount==0)
sl@0
   393
		// for each start of field encountered, fieldCount++
sl@0
   394
		{
sl@0
   395
		if (aInfo.iFieldCountInRange==0)
sl@0
   396
			{
sl@0
   397
			aInfo.iFirstFieldLen = (*iFieldArray)[index].iFieldValueLen;
sl@0
   398
			aInfo.iFirstFieldPos = pos;
sl@0
   399
			}
sl@0
   400
		aInfo.iFieldCountInRange++;
sl@0
   401
		pos += (*iFieldArray)[index].iFieldValueLen+(*iFieldArray)[index+1].iPreFieldLen;
sl@0
   402
		index++;
sl@0
   403
		}
sl@0
   404
	// (if sensing right) check that we haven't ended adjacent to one or more zero-length fields 
sl@0
   405
	if ( (aRange==0) && (pos==aPos) && (index<(iFieldArray->Count()-1)) && ((*iFieldArray)[index].iFieldValueLen==0) )
sl@0
   406
		{
sl@0
   407
		aInfo.iFieldCountInRange++;
sl@0
   408
		index++;
sl@0
   409
		while ( ((*iFieldArray)[index].iPreFieldLen==0) && ((*iFieldArray)[index].iFieldValueLen==0) && (index<(iFieldArray->Count()-1)) )
sl@0
   410
			{
sl@0
   411
			aInfo.iFieldCountInRange++;
sl@0
   412
			index++;
sl@0
   413
			}
sl@0
   414
		}
sl@0
   415
sl@0
   416
	__ASSERT_DEBUG(aInfo.iFieldCountInRange<=FieldCount(),Panic(EDebug));
sl@0
   417
	return aInfo.iFieldCountInRange;
sl@0
   418
	}
sl@0
   419
sl@0
   420
sl@0
   421
EXPORT_C TInt CTextFieldSet::NewFieldValueL(HBufC*& aBuf, TInt aPos)
sl@0
   422
// Returns the new value of the field at aPos (if applicable).
sl@0
   423
// The method might reallocate aBuf parameter, so don't forget to:
sl@0
   424
// 1) Push aBuf parameter in the CleanupStack before the call. The call
sl@0
   425
//      may fail, then aBuf gets lost.
sl@0
   426
// 2) Pop aBuf parameter from the CleanupStack after the call - aBuf may get
sl@0
   427
//      reallocated, so the pushed pointer becomes invalid.
sl@0
   428
// 3) Push aBuf in the CleanupStack again.
sl@0
   429
	{
sl@0
   430
	__TEST_INVARIANT;
sl@0
   431
	__ASSERT_ALWAYS(aBuf,Panic(ENoBuffer));
sl@0
   432
	__ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
sl@0
   433
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   434
	
sl@0
   435
	TInt errLevel = KErrNotFound;
sl@0
   436
	TInt index; TInt offset;
sl@0
   437
	if (InField(aPos,index,offset))
sl@0
   438
		{// There's a field at aPos
sl@0
   439
		TPtr bufPtr = aBuf->Des();
sl@0
   440
		TInt reqLen = (*iFieldArray)[index].iFieldHeader.iField->Value(bufPtr);
sl@0
   441
		while (reqLen>0)
sl@0
   442
			{// If more storage is required for the value then reallocate the buffer
sl@0
   443
			aBuf = aBuf->ReAllocL(reqLen); // for unicode compatability, rounds up
sl@0
   444
			TPtr pointer = aBuf->Des();
sl@0
   445
			reqLen = (*iFieldArray)[index].iFieldHeader.iField->Value(pointer);
sl@0
   446
			}
sl@0
   447
		// dont set new field length - this will be done in a subsequent NotifyInsert()
sl@0
   448
		errLevel = KErrNone;
sl@0
   449
		}
sl@0
   450
sl@0
   451
	__TEST_INVARIANT;
sl@0
   452
	return errLevel;
sl@0
   453
	}
sl@0
   454
sl@0
   455
sl@0
   456
EXPORT_C void CTextFieldSet::NotifyInsertion(TInt aPos, TInt aNumberAdded)
sl@0
   457
// Informs the array that aNumberAdded characters have been inserted in the document
sl@0
   458
//
sl@0
   459
	{
sl@0
   460
	__TEST_INVARIANT;
sl@0
   461
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutsideDoc));
sl@0
   462
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   463
	__ASSERT_ALWAYS(aNumberAdded>=0,Panic(EIllegalNegativeValue));
sl@0
   464
sl@0
   465
	TInt index; TInt offset;
sl@0
   466
	// are the extra characters in a field (matching away from fields)?
sl@0
   467
	if (!InField(aPos,index,offset))
sl@0
   468
		(*iFieldArray)[index].iPreFieldLen += aNumberAdded;
sl@0
   469
	else
sl@0
   470
		if (offset>0)
sl@0
   471
			(*iFieldArray)[index].iFieldValueLen += aNumberAdded;
sl@0
   472
		else
sl@0
   473
			(*iFieldArray)[index].iPreFieldLen += aNumberAdded;
sl@0
   474
sl@0
   475
	__TEST_INVARIANT;
sl@0
   476
	}                             
sl@0
   477
sl@0
   478
sl@0
   479
EXPORT_C void CTextFieldSet::NotifyFieldUpdate(TInt aPos, TInt aNewFieldValueLength)
sl@0
   480
	{
sl@0
   481
	__TEST_INVARIANT;
sl@0
   482
	__ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
sl@0
   483
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   484
	__ASSERT_ALWAYS(aNewFieldValueLength>=0,Panic(EIllegalNegativeValue));
sl@0
   485
sl@0
   486
	// Is the insert pos in a field? If so which?
sl@0
   487
	TInt index; TInt offset;
sl@0
   488
	__ASSERT_ALWAYS(InField(aPos,index,offset),Panic(EPosNotInField));
sl@0
   489
	// Update the length of the relevant field
sl@0
   490
	(*iFieldArray)[index].iFieldValueLen = aNewFieldValueLength;
sl@0
   491
	
sl@0
   492
	__TEST_INVARIANT;
sl@0
   493
	}
sl@0
   494
sl@0
   495
sl@0
   496
EXPORT_C void CTextFieldSet::NotifyDeletion(TInt aPos,TInt aTotalRemoved)
sl@0
   497
// Informs the array that aTotalRemoved characters have been removed from the document
sl@0
   498
// Any fields wholely contained will be removed, those partially intersecting will just be shortened
sl@0
   499
//
sl@0
   500
	{
sl@0
   501
	__TEST_INVARIANT;
sl@0
   502
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutsideDoc));
sl@0
   503
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   504
	__ASSERT_ALWAYS(aTotalRemoved>=0,Panic(EIllegalNegativeValue));
sl@0
   505
	
sl@0
   506
	TInt charCount = CharCount();
sl@0
   507
	
sl@0
   508
	//There is a possibility that there exists hidden end-of-document character in rich text objects.
sl@0
   509
	//This is checked by the if statement & accordingly aTotalRemoved is decremented.
sl@0
   510
	//i.e. aPos + aTotalRemoved could be greater by 1 than the CharCount(),  
sl@0
   511
	//this is because the aTotalRemoved might also include 1 hidden character. - DEF095911
sl@0
   512
	if(aPos+aTotalRemoved > charCount)
sl@0
   513
		aTotalRemoved--;
sl@0
   514
sl@0
   515
	__ASSERT_DEBUG(aPos+aTotalRemoved<=charCount, Panic(EPosOutsideDoc));
sl@0
   516
sl@0
   517
	int index = 0;
sl@0
   518
	int offset = 0;
sl@0
   519
	int cur_pos = aPos;
sl@0
   520
	int end_pos = aPos + aTotalRemoved;
sl@0
   521
	TBool in_field = InField(cur_pos,index,offset);
sl@0
   522
	TTextFieldEntry* field = NULL;
sl@0
   523
	int field_start = 0;
sl@0
   524
	int field_end = 0;
sl@0
   525
	if (index >= 0 && index < iFieldArray->Count())
sl@0
   526
		{
sl@0
   527
		field = &(*iFieldArray)[index];
sl@0
   528
		field_start = cur_pos - offset;
sl@0
   529
		if (!in_field)
sl@0
   530
			field_start += field->iPreFieldLen;
sl@0
   531
		field_end = field_start + field->iFieldValueLen;
sl@0
   532
		}
sl@0
   533
sl@0
   534
	while (field)
sl@0
   535
		{
sl@0
   536
		// Reduce the size of the gap before the field if any.
sl@0
   537
		int gap = Min(end_pos,field_start) - cur_pos;
sl@0
   538
		if (gap > 0)
sl@0
   539
			{
sl@0
   540
			cur_pos += gap;
sl@0
   541
			field->iPreFieldLen -= gap;
sl@0
   542
			}
sl@0
   543
		if (cur_pos >= end_pos)
sl@0
   544
			break;
sl@0
   545
sl@0
   546
		// Reduce the field length.
sl@0
   547
		int remove_start = cur_pos;
sl@0
   548
		int remove_end = Min(field_end,end_pos);
sl@0
   549
		cur_pos = field_end;
sl@0
   550
		field->iFieldValueLen -= (remove_end - remove_start);
sl@0
   551
sl@0
   552
		// Delete the field if it is now of zero length.
sl@0
   553
		int added_to_next = 0;
sl@0
   554
		if (field->iFieldValueLen == 0)
sl@0
   555
			{
sl@0
   556
			added_to_next = field->iPreFieldLen;
sl@0
   557
			DeleteFieldEntry(index);
sl@0
   558
			}
sl@0
   559
		else
sl@0
   560
			index++;
sl@0
   561
sl@0
   562
		// Move to the next field.
sl@0
   563
		if (index < iFieldArray->Count())
sl@0
   564
			{
sl@0
   565
			field = &(*iFieldArray)[index];
sl@0
   566
			field_start = cur_pos + field->iPreFieldLen - added_to_next;
sl@0
   567
			field_end = field_start + field->iFieldValueLen;
sl@0
   568
			}
sl@0
   569
		else
sl@0
   570
			field = NULL;
sl@0
   571
		}
sl@0
   572
sl@0
   573
	__TEST_INVARIANT;
sl@0
   574
	}
sl@0
   575
sl@0
   576
sl@0
   577
TBool CTextFieldSet::InField(const TInt aPos, TInt& anIndex, TInt& anOffset) const
sl@0
   578
// works out whether or not aPos is in a field (matching right),
sl@0
   579
//  sets anIndex to the index number of the field entry,
sl@0
   580
//  and sets anOffset to the distance aPos is into the field or its preceeding gap
sl@0
   581
//
sl@0
   582
	{
sl@0
   583
	__ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
sl@0
   584
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
sl@0
   585
sl@0
   586
	TBool inField;
sl@0
   587
	TInt currentPos = 0;
sl@0
   588
	anIndex = 0;
sl@0
   589
	TInt lengthOfDoc = CharCount();
sl@0
   590
	TInt lastFieldNum = iFieldArray->Count()-1;
sl@0
   591
	// Find the index of the entry containing aPos
sl@0
   592
	if (aPos == lengthOfDoc)
sl@0
   593
		{
sl@0
   594
		anIndex = lastFieldNum;
sl@0
   595
		currentPos = aPos;
sl@0
   596
		}
sl@0
   597
	else 
sl@0
   598
		{
sl@0
   599
		anIndex = -1;
sl@0
   600
		while (aPos >= currentPos)
sl@0
   601
			{
sl@0
   602
			anIndex++;
sl@0
   603
			currentPos += (*iFieldArray)[anIndex].iPreFieldLen+(*iFieldArray)[anIndex].iFieldValueLen;
sl@0
   604
			}
sl@0
   605
		}
sl@0
   606
	// Check that we have not skipped over any zero-length fields
sl@0
   607
	if ( ((anIndex-1)>=0) && ((*iFieldArray)[anIndex-1].iFieldValueLen==0) )
sl@0
   608
		{
sl@0
   609
		TInt temp = currentPos - ((*iFieldArray)[anIndex].iPreFieldLen+(*iFieldArray)[anIndex].iFieldValueLen);
sl@0
   610
		if (temp==aPos)
sl@0
   611
			{// aPos is on a field boundary
sl@0
   612
			currentPos = temp;
sl@0
   613
			anIndex--;
sl@0
   614
			while ( ((anIndex-1)>=0)
sl@0
   615
				&&((*iFieldArray)[anIndex].iPreFieldLen==0)
sl@0
   616
				&&((*iFieldArray)[anIndex-1].iFieldValueLen==0) )
sl@0
   617
				{
sl@0
   618
				anIndex--;
sl@0
   619
				}
sl@0
   620
			}
sl@0
   621
		}
sl@0
   622
	// Move to the start of the field (end of prefield) in entry [anIndex]
sl@0
   623
	currentPos -= (*iFieldArray)[anIndex].iFieldValueLen;
sl@0
   624
	// Determine whether or not aPos is in the field of entry[anIndex]
sl@0
   625
	if (anIndex == lastFieldNum)
sl@0
   626
		inField = EFalse;
sl@0
   627
	else if (aPos >= currentPos)
sl@0
   628
			inField = ETrue;
sl@0
   629
		else
sl@0
   630
			inField = EFalse;
sl@0
   631
	// Calculate the offset
sl@0
   632
	if (inField)
sl@0
   633
		anOffset = aPos-currentPos;
sl@0
   634
	else
sl@0
   635
		anOffset = aPos+(*iFieldArray)[anIndex].iPreFieldLen-currentPos;
sl@0
   636
	return inField;
sl@0
   637
	}
sl@0
   638
sl@0
   639
sl@0
   640
void CTextFieldSet::DeleteFieldEntry(TInt anIndex)
sl@0
   641
	{
sl@0
   642
	DeleteFieldEntry(iFieldArray,anIndex);
sl@0
   643
	}
sl@0
   644
sl@0
   645
sl@0
   646
void CTextFieldSet::DeleteFieldEntry(CArrayFixSeg<TTextFieldEntry>* aArray,TInt anIndex)
sl@0
   647
// remove the entry anIndex from the array but don't delete the text from the doc.
sl@0
   648
//
sl@0
   649
	{
sl@0
   650
	__ASSERT_ALWAYS(anIndex<(aArray->Count()-1),Panic(EIndexOutOfRange));
sl@0
   651
sl@0
   652
	// add the entry's "before" to the "before" of the next entry.
sl@0
   653
	(*aArray)[anIndex+1].iPreFieldLen += (*aArray)[anIndex].iPreFieldLen;
sl@0
   654
	// add the field's length to the "before" of the next entry.	
sl@0
   655
	(*aArray)[anIndex+1].iPreFieldLen += (*aArray)[anIndex].iFieldValueLen;
sl@0
   656
	if ((*aArray)[anIndex].iFieldHeader.iField.IsPtr())
sl@0
   657
		delete (*aArray)[anIndex].iFieldHeader.iField.AsPtr(); // delete the field
sl@0
   658
	aArray->Delete(anIndex); // remove the entry from the array
sl@0
   659
	}
sl@0
   660
sl@0
   661
sl@0
   662
///////////////////////////////////////////////
sl@0
   663
// TTextFieldEntry
sl@0
   664
///////////////////////////////////////////////
sl@0
   665
sl@0
   666
sl@0
   667
EXPORT_C void TTextFieldEntry::InternalizeL(RReadStream& aStream)
sl@0
   668
// entry must have a header (ie cant be last entry in the array)
sl@0
   669
	{
sl@0
   670
	iPreFieldLen = aStream.ReadInt32L();
sl@0
   671
	iFieldValueLen = aStream.ReadInt32L();
sl@0
   672
	aStream>> iFieldHeader;
sl@0
   673
	}
sl@0
   674
sl@0
   675
sl@0
   676
EXPORT_C void TTextFieldEntry::ExternalizeL(RWriteStream& aStream)const
sl@0
   677
// entry must have a header (ie cant be last entry in the array)
sl@0
   678
	{
sl@0
   679
	aStream.WriteInt32L(iPreFieldLen);
sl@0
   680
	aStream.WriteInt32L(iFieldValueLen);
sl@0
   681
	aStream<< iFieldHeader;
sl@0
   682
	}
sl@0
   683
sl@0
   684