os/textandloc/textrendering/texthandling/sfields/FLDSTRM.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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 <s32strm.h>
sl@0
    23
#include <s32stor.h>
sl@0
    24
sl@0
    25
#include "FLDDEF.H"
sl@0
    26
#include "FLDSET.H"
sl@0
    27
#include "FLDARRAY.H"
sl@0
    28
sl@0
    29
#include "FLDSTD.H"
sl@0
    30
sl@0
    31
sl@0
    32
sl@0
    33
EXPORT_C TFieldMapExternalizer::TFieldMapExternalizer(const CStoreMap& aMap)
sl@0
    34
	: iMap(&aMap)
sl@0
    35
	{}
sl@0
    36
sl@0
    37
EXPORT_C void TFieldMapExternalizer::ExternalizeL(const TStreamRef& aRef,RWriteStream& aStream) const
sl@0
    38
// Write the stream id bound to aRef to aStream. If not bound, write KNullStreamId
sl@0
    39
//
sl@0
    40
	{
sl@0
    41
	TSwizzleC<TAny> swizzle=aRef;
sl@0
    42
	aStream<<iMap->At(swizzle);
sl@0
    43
	}
sl@0
    44
sl@0
    45
sl@0
    46
EXPORT_C TStreamId CTextFieldSet::StoreL(CStreamStore& aStore)const
sl@0
    47
// Save the fields and the fieldSet in their own streams
sl@0
    48
// Encapsulates the storing of its components.
sl@0
    49
//
sl@0
    50
	{
sl@0
    51
	CStoreMap* map=CStoreMap::NewLC(aStore);
sl@0
    52
	StoreFieldsL(aStore,*map);  // binds id's to swizzles
sl@0
    53
//
sl@0
    54
	// create custom externalizer over the map
sl@0
    55
	TFieldMapExternalizer fMap(*map);
sl@0
    56
	RStoreWriteStream stream(fMap);
sl@0
    57
	TStreamId id=stream.CreateLC(aStore);
sl@0
    58
	stream<< *this;
sl@0
    59
	stream.CommitL();
sl@0
    60
	CleanupStack::PopAndDestroy();  // stream
sl@0
    61
//
sl@0
    62
	map->Reset();
sl@0
    63
	CleanupStack::PopAndDestroy();  // map
sl@0
    64
	return id;
sl@0
    65
	}
sl@0
    66
sl@0
    67
sl@0
    68
EXPORT_C void CTextFieldSet::StoreFieldsL(CStreamStore& aStore,CStoreMap& aMap)const
sl@0
    69
// Stores all fields in the set
sl@0
    70
//
sl@0
    71
	{StoreFieldsL(aStore,aMap,iFieldArray);}
sl@0
    72
sl@0
    73
sl@0
    74
void CTextFieldSet::StoreFieldsL(CStreamStore& aStore,CStoreMap& aMap,CArrayFixSeg<TTextFieldEntry>* aArray)const
sl@0
    75
// Stores all fields contained in the set provided
sl@0
    76
//
sl@0
    77
	{
sl@0
    78
	__TEST_INVARIANT;
sl@0
    79
sl@0
    80
	for (TInt i=0 ; i<(aArray->Count()-1) ; i++)
sl@0
    81
		{
sl@0
    82
		TStreamId id=(*aArray)[i].iFieldHeader.iField->StoreL(aStore);
sl@0
    83
		if (id!=KNullStreamId)
sl@0
    84
			aMap.BindL((*aArray)[i].iFieldHeader.iField,id);
sl@0
    85
		}
sl@0
    86
	}
sl@0
    87
sl@0
    88
sl@0
    89
EXPORT_C void CTextFieldSet::ExternalizeL(RWriteStream& aStream)const
sl@0
    90
	{
sl@0
    91
	__TEST_INVARIANT;
sl@0
    92
sl@0
    93
	ExternalizeL(aStream,iFieldArray);
sl@0
    94
	}
sl@0
    95
sl@0
    96
sl@0
    97
void CTextFieldSet::ExternalizeL(RWriteStream& aStream,CArrayFixSeg<TTextFieldEntry>* aArray)const
sl@0
    98
	{
sl@0
    99
	TInt numFieldEntries = aArray->Count();
sl@0
   100
	aStream.WriteInt32L(numFieldEntries);
sl@0
   101
	// write out fields
sl@0
   102
	for (TInt i=0 ; i<numFieldEntries-1 ; i++)
sl@0
   103
		aStream<< (*aArray)[i];
sl@0
   104
	// write out last entry in array: the bit after the last field
sl@0
   105
	aStream.WriteInt32L((*aArray)[numFieldEntries-1].iPreFieldLen);
sl@0
   106
	}
sl@0
   107
sl@0
   108
sl@0
   109
EXPORT_C void CTextFieldSet::RestoreL(const CStreamStore& aFieldStore,TStreamId aStreamId)
sl@0
   110
	{
sl@0
   111
	// reset the array and stream into it
sl@0
   112
	Reset();
sl@0
   113
	DoRestoreL(aFieldStore,aStreamId);
sl@0
   114
	}
sl@0
   115
sl@0
   116
sl@0
   117
EXPORT_C void CTextFieldSet::RestoreFieldsL(const CStreamStore& aFieldStore)
sl@0
   118
	{
sl@0
   119
	DoRestoreFieldsL(iFieldArray,aFieldStore); // restore the fields individually from their own streams
sl@0
   120
	}
sl@0
   121
sl@0
   122
sl@0
   123
void CTextFieldSet::DoRestoreL(const CStreamStore& aFieldStore,TStreamId aStreamId)
sl@0
   124
// Restores a field set and its associated fields from the store provded.
sl@0
   125
// 
sl@0
   126
	{
sl@0
   127
	__ASSERT_ALWAYS(iFieldArray->Count()==1,Panic(EArrayNotEmptyOnRestore)); // array must be empty
sl@0
   128
	__ASSERT_ALWAYS((*iFieldArray)[0].iPreFieldLen==0,Panic(EArrayNotEmptyOnRestore));
sl@0
   129
sl@0
   130
	// retrieve the headstream from the store
sl@0
   131
	RStoreReadStream stream;
sl@0
   132
	stream.OpenLC(aFieldStore,aStreamId);
sl@0
   133
	// restore the set, then the individual fields
sl@0
   134
	stream>> *this;  // internalize the field set (the headers)
sl@0
   135
	CleanupStack::PopAndDestroy(); // stream
sl@0
   136
	DoRestoreFieldsL(iFieldArray,aFieldStore); // restore the fields individually from their own streams
sl@0
   137
	}
sl@0
   138
sl@0
   139
sl@0
   140
EXPORT_C void CTextFieldSet::InternalizeL(RReadStream& aStream)
sl@0
   141
	{
sl@0
   142
	InternalizeL(iFieldArray,aStream);
sl@0
   143
sl@0
   144
	__TEST_INVARIANT;
sl@0
   145
	}
sl@0
   146
sl@0
   147
sl@0
   148
void CTextFieldSet::InternalizeL(CArrayFixSeg<TTextFieldEntry>* aArray,RReadStream& aStream)
sl@0
   149
	{// assume the array is empty
sl@0
   150
	TInt numFieldEntries = aStream.ReadInt32L();
sl@0
   151
	// read in the fields
sl@0
   152
	TTextFieldEntry entry;
sl@0
   153
	for (TInt i=0 ; i<numFieldEntries-1 ; i++)
sl@0
   154
		{
sl@0
   155
		aStream>> entry;
sl@0
   156
		InsertEntryL(i,entry,aArray); // insert new entry
sl@0
   157
		}
sl@0
   158
	// read in the last entry: the bit after the last field. This will not contain a field
sl@0
   159
	(*aArray)[numFieldEntries-1].iPreFieldLen = aStream.ReadInt32L();
sl@0
   160
	}
sl@0
   161
sl@0
   162
sl@0
   163
void CTextFieldSet::DoRestoreFieldsL(CArrayFixSeg<TTextFieldEntry>* aArray,const CStreamStore& aFieldStore,TInt aStartIndex)
sl@0
   164
// This fn is called after all FieldHeaders have been internalized - the Swizzles hold the stream id's.
sl@0
   165
// One by one the fields are created (with the factory) and then have their settings streamed in from the store.
sl@0
   166
// If no factory exists all fields are converted to text.
sl@0
   167
//
sl@0
   168
	{
sl@0
   169
	TInt ii=aArray->Count()-2;  // -2 because we skip the last (empty) entry
sl@0
   170
	while (ii>=aStartIndex)
sl@0
   171
		{
sl@0
   172
		if ((*aArray)[ii].iFieldHeader.iField.IsId()) 
sl@0
   173
			{// restore the field only if it isn't the very last (dummy) entry
sl@0
   174
			if (iFieldFactory==NULL)
sl@0
   175
				// no factory - convert the field to text
sl@0
   176
				DeleteFieldEntry(aArray,ii);
sl@0
   177
			else
sl@0
   178
				{
sl@0
   179
				TStreamId id = (*aArray)[ii].iFieldHeader.iField.AsId();
sl@0
   180
				(*aArray)[ii].iFieldHeader.iField = iFieldFactory->NewFieldL((*aArray)[ii].iFieldHeader.iFieldType);
sl@0
   181
				if ((*aArray)[ii].iFieldHeader.iField!=NULL)
sl@0
   182
					(*aArray)[ii].iFieldHeader.iField->RestoreL(aFieldStore,id);
sl@0
   183
				else
sl@0
   184
					DeleteFieldEntry(aArray,ii); // handle "field type not recognised" by converting field to text
sl@0
   185
				}
sl@0
   186
			}
sl@0
   187
		ii--;
sl@0
   188
		}
sl@0
   189
	}
sl@0
   190
sl@0
   191
sl@0
   192
/***************************************** cut & paste *******************************************/
sl@0
   193
sl@0
   194
sl@0
   195
EXPORT_C TStreamId CTextFieldSet::CopyToStoreL(CStreamStore& aStore,TInt aPos,TInt aLength)const
sl@0
   196
// Copy any fields in the selected region to the specified store, returning the id of the head-stream.
sl@0
   197
//
sl@0
   198
	{
sl@0
   199
	__TEST_INVARIANT;
sl@0
   200
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
sl@0
   201
	__ASSERT_ALWAYS(aLength>=0,Panic(ENegativeRange));
sl@0
   202
	__ASSERT_DEBUG((aPos+aLength)<=CharCount(),Panic(EPosOutOfRange));
sl@0
   203
sl@0
   204
	// Create a store map and store the fields themselves
sl@0
   205
	CStoreMap* map=CStoreMap::NewLC(aStore);
sl@0
   206
	CopyComponentsL(aStore,*map,aPos,aLength);
sl@0
   207
sl@0
   208
	// Create a head-stream in which to store the field entries and do so
sl@0
   209
	RStoreWriteStream stream(*map);
sl@0
   210
	TStreamId id=stream.CreateLC(aStore);
sl@0
   211
	CopyToStreamL(stream,aPos,aLength);
sl@0
   212
sl@0
   213
	// tidy up
sl@0
   214
	stream.CommitL();
sl@0
   215
	map->Reset();
sl@0
   216
	CleanupStack::PopAndDestroy(2); // map, stream
sl@0
   217
	return id;
sl@0
   218
	}
sl@0
   219
sl@0
   220
sl@0
   221
EXPORT_C void CTextFieldSet::CopyComponentsL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength)const
sl@0
   222
// Stores all fields in the set
sl@0
   223
//
sl@0
   224
	{
sl@0
   225
	__TEST_INVARIANT;
sl@0
   226
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
sl@0
   227
	__ASSERT_ALWAYS(aLength>=0,Panic(ENegativeRange));
sl@0
   228
	__ASSERT_DEBUG((aPos+aLength)<=CharCount(),Panic(EPosOutOfRange));
sl@0
   229
sl@0
   230
	// Create an array of the fields to be cut/copied
sl@0
   231
	CArrayFixSeg<TTextFieldEntry>* tempArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
sl@0
   232
	CleanupStack::PushL(tempArray);
sl@0
   233
	CopyToArrayL(tempArray,aPos,aLength);
sl@0
   234
sl@0
   235
	// stream the required fields in their own streams
sl@0
   236
	StoreFieldsL(aStore,aMap,tempArray);
sl@0
   237
	CleanupStack::PopAndDestroy(); // tempArray
sl@0
   238
	}
sl@0
   239
sl@0
   240
sl@0
   241
EXPORT_C void CTextFieldSet::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength)const
sl@0
   242
// Stores all fields in the set
sl@0
   243
//
sl@0
   244
	{
sl@0
   245
	__TEST_INVARIANT;
sl@0
   246
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
sl@0
   247
	__ASSERT_ALWAYS(aLength>=0,Panic(ENegativeRange));
sl@0
   248
	__ASSERT_DEBUG((aPos+aLength)<=CharCount(),Panic(EPosOutOfRange));
sl@0
   249
sl@0
   250
	// Create an array of the fields to be cut/copied
sl@0
   251
	CArrayFixSeg<TTextFieldEntry>* tempArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
sl@0
   252
	CleanupStack::PushL(tempArray);
sl@0
   253
	CopyToArrayL(tempArray,aPos,aLength);
sl@0
   254
sl@0
   255
	// stream the field entries in the temp array
sl@0
   256
	ExternalizeL(aStream,tempArray);
sl@0
   257
	CleanupStack::PopAndDestroy(); // tempArray
sl@0
   258
	}
sl@0
   259
sl@0
   260
sl@0
   261
void CTextFieldSet::CopyToArrayL(CArrayFixSeg<TTextFieldEntry>* aArray,TInt aPos,TInt aLength)const
sl@0
   262
	{
sl@0
   263
	TInt index; TInt offset;
sl@0
   264
	if (InField(aPos,index,offset))
sl@0
   265
		offset += (*iFieldArray)[index].iPreFieldLen; // make offset relative to start of entry
sl@0
   266
	// split first entry in range
sl@0
   267
	TTextFieldEntry entry = SplitEntry(index,offset,aLength);
sl@0
   268
	index++;
sl@0
   269
	TInt charsCopied=EntryLen(entry);
sl@0
   270
	// split second if neccessary
sl@0
   271
	if ((!entry.iFieldHeader.iField)&&(charsCopied<aLength))
sl@0
   272
		{
sl@0
   273
		TInt preFieldLen = entry.iPreFieldLen;
sl@0
   274
		entry = SplitEntry(index,0,aLength-preFieldLen);
sl@0
   275
		entry.iPreFieldLen += preFieldLen;
sl@0
   276
		charsCopied = EntryLen(entry);
sl@0
   277
		index++;
sl@0
   278
		}
sl@0
   279
	((CTextFieldSet*)this)->AppendEntryL(entry,aArray); // append the first entry to the storage array
sl@0
   280
	// write out all whole entries
sl@0
   281
	while (charsCopied<aLength)
sl@0
   282
		{
sl@0
   283
		if ((charsCopied+EntryLen(index))<=aLength)
sl@0
   284
			((CTextFieldSet*)this)->AppendEntryL((*iFieldArray)[index],aArray); 
sl@0
   285
		charsCopied += EntryLen(index);
sl@0
   286
		index++;
sl@0
   287
		}
sl@0
   288
	// split last entry if neccessary
sl@0
   289
	if (charsCopied>aLength)
sl@0
   290
		{// The last entry needs to be split
sl@0
   291
		// first get back to the beginning of the entry
sl@0
   292
		index--;
sl@0
   293
		charsCopied -= EntryLen(index);
sl@0
   294
		entry = SplitEntry(index,0,aLength-charsCopied); // split up the last entry as required
sl@0
   295
		((CTextFieldSet*)this)->AppendEntryL(entry,aArray); // append the last entry to the storage array
sl@0
   296
		}
sl@0
   297
	// add an empty last entry if neccessary
sl@0
   298
	TInt numFieldEntries = aArray->Count();
sl@0
   299
	if (((*aArray)[numFieldEntries-1].iFieldHeader.iField) || ((*aArray)[numFieldEntries-1].iFieldValueLen!=0))
sl@0
   300
		{
sl@0
   301
		entry.iPreFieldLen = 0;
sl@0
   302
		entry.iFieldValueLen = 0;
sl@0
   303
		entry.iFieldHeader.iFieldType = KNullUid;
sl@0
   304
		entry.iFieldHeader.iField = NULL;
sl@0
   305
		((CTextFieldSet*)this)->AppendEntryL(entry,aArray);
sl@0
   306
		numFieldEntries++;
sl@0
   307
		}
sl@0
   308
	}
sl@0
   309
sl@0
   310
sl@0
   311
EXPORT_C void CTextFieldSet::PasteFromStoreL(const CStreamStore& aFieldStore,TStreamId aStreamId,TInt aPos,TInt aMaxLen)
sl@0
   312
// Paste from aStore into the document at insert position aPos.
sl@0
   313
// Optionally the pasted text can be clipped to a maximum length aMaxLen.
sl@0
   314
//
sl@0
   315
	{
sl@0
   316
	__ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
sl@0
   317
	__ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutOfRange));
sl@0
   318
sl@0
   319
	// retrieve the headstream from the store
sl@0
   320
	RStoreReadStream stream;
sl@0
   321
	stream.OpenLC(aFieldStore,aStreamId);
sl@0
   322
sl@0
   323
	// restore the set...
sl@0
   324
	PasteFromStreamL(stream,aPos,aMaxLen); // internalize the field set (the headers)
sl@0
   325
	CleanupStack::PopAndDestroy(); // stream
sl@0
   326
sl@0
   327
	// ...then the individual fields
sl@0
   328
	PasteComponentsL(aFieldStore,aPos); // restore the fields individually from their own streams
sl@0
   329
	}
sl@0
   330
sl@0
   331
sl@0
   332
EXPORT_C void CTextFieldSet::PasteFromStreamL(RReadStream& aStream,TInt aPos,TInt aMaxLen)
sl@0
   333
// streams the field entries into a temporary array, which is returned. 
sl@0
   334
// PasteComponents() must be called after this to actually carry out the paste...
sl@0
   335
	{
sl@0
   336
	// create a temporary array to stream in to, inserting the first entry
sl@0
   337
	CArrayFixSeg<TTextFieldEntry>* tempFieldArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
sl@0
   338
	CleanupStack::PushL(tempFieldArray);
sl@0
   339
	AddInitialFieldEntryL(tempFieldArray,0);
sl@0
   340
sl@0
   341
	// internalize the field entries
sl@0
   342
	InternalizeL(tempFieldArray,aStream);
sl@0
   343
sl@0
   344
	// trim off any entries that lie beyond aMaxLength
sl@0
   345
	if (aMaxLen!=ENoPasteLimit)
sl@0
   346
		{// if aMaxLen is not ENoPasteLimit discard the excess fields
sl@0
   347
		__ASSERT_ALWAYS(aMaxLen>=0,Panic(ELengthOutOfRange));
sl@0
   348
		//
sl@0
   349
		TInt length=0;
sl@0
   350
		TInt i=0;
sl@0
   351
		for (i=0 ; (length<aMaxLen)&&(i<tempFieldArray->Count()) ; i++)
sl@0
   352
			length += EntryLen((*tempFieldArray)[i]);
sl@0
   353
		if (aMaxLen==0)
sl@0
   354
			{// make first entry zero len, delete all others
sl@0
   355
			i++;
sl@0
   356
			(*tempFieldArray)[i-1].iPreFieldLen = 0;
sl@0
   357
			}
sl@0
   358
		else if (length>aMaxLen)
sl@0
   359
			// truncate the last field in range
sl@0
   360
			(*tempFieldArray)[i-1].iPreFieldLen += (*tempFieldArray)[i-1].iFieldValueLen-(length-aMaxLen);
sl@0
   361
		else if ((length==aMaxLen) && ((*tempFieldArray)[i-1].iFieldHeader.iField!=NULL))
sl@0
   362
			{// if the terminating entry has a field add a zero length entry, the mandatory last entry
sl@0
   363
			i++;
sl@0
   364
			(*tempFieldArray)[i-1].iPreFieldLen = 0;
sl@0
   365
			}
sl@0
   366
		// ensure the last entry is of the correct format
sl@0
   367
		(*tempFieldArray)[i-1].iFieldValueLen = 0;
sl@0
   368
		(*tempFieldArray)[i-1].iFieldHeader.iFieldType = KNullUid;
sl@0
   369
		(*tempFieldArray)[i-1].iFieldHeader.iField = NULL;
sl@0
   370
		// delete all the fields wholely out of range
sl@0
   371
		for (TInt index=i ; index<tempFieldArray->Count() ; index++)
sl@0
   372
			(*tempFieldArray)[index].iFieldHeader.iField = NULL;
sl@0
   373
		tempFieldArray->Delete(i,tempFieldArray->Count()-i); // pos,count
sl@0
   374
		}
sl@0
   375
sl@0
   376
	DoPasteL(tempFieldArray,aPos);
sl@0
   377
	CleanupStack::PopAndDestroy(); // tempFieldArray
sl@0
   378
	}
sl@0
   379
sl@0
   380
sl@0
   381
EXPORT_C void CTextFieldSet::PasteComponentsL(const CStreamStore& aFieldStore,TInt aPos)
sl@0
   382
	{
sl@0
   383
	// Restore the fields individually from their own streams
sl@0
   384
	TInt index; TInt offset;
sl@0
   385
	// We don't need to make any difference between in and not in field situation here
sl@0
   386
	// all we need is the index
sl@0
   387
	TBool isInField = InField(aPos,index,offset);
sl@0
   388
	DoRestoreFieldsL(iFieldArray,aFieldStore,index); 
sl@0
   389
	}
sl@0
   390
sl@0
   391
sl@0
   392
void CTextFieldSet::DoPasteL(CArrayFixSeg<TTextFieldEntry>* aSourceArray,TInt aPos)
sl@0
   393
// Insert into this instance, at character position aPos, the entire (field entry) contents of the field array aSourceArray.
sl@0
   394
// All iField objects in aSourceArray are ID's at this time.
sl@0
   395
//
sl@0
   396
	{
sl@0
   397
	// are we inserting into a field?
sl@0
   398
	TInt numFieldEntries = aSourceArray->Count();
sl@0
   399
	TInt index; TInt offset;
sl@0
   400
	
sl@0
   401
	TBool inField = InField(aPos,index,offset);
sl@0
   402
	// record the rollback info
sl@0
   403
	RecordRollbackInfoL(index);
sl@0
   404
	if ((inField)&&(offset!=0))
sl@0
   405
		{// everything we insert will become text - no chance of leaving
sl@0
   406
		// insert all but last entry
sl@0
   407
		TInt i=0;
sl@0
   408
		for (; i<numFieldEntries-1 ; i++)
sl@0
   409
			{// copy text (no need to delete field)
sl@0
   410
			(*iFieldArray)[index].iFieldValueLen += EntryLen((*aSourceArray)[i]);
sl@0
   411
			}
sl@0
   412
		// read in the last entry (has no field attached)
sl@0
   413
		(*iFieldArray)[index].iFieldValueLen += (*aSourceArray)[i].iPreFieldLen;
sl@0
   414
		}
sl@0
   415
	else
sl@0
   416
		{// else split the entry we are going to bisect - this may leave
sl@0
   417
		if (inField)
sl@0
   418
			offset = (*iFieldArray)[index].iPreFieldLen; // must be at start of field
sl@0
   419
		if (numFieldEntries>1)
sl@0
   420
			{// read 1st field & carry out split.
sl@0
   421
			InsertEntryL(index,(*aSourceArray)[0]); // if this leaves the model will be intact
sl@0
   422
			(*iFieldArray)[index].iPreFieldLen += offset;
sl@0
   423
			(*iFieldArray)[index+1].iPreFieldLen -= offset;
sl@0
   424
			index++;
sl@0
   425
			}
sl@0
   426
		// insert all other fields except last.
sl@0
   427
		for (TInt i=1 ; i<numFieldEntries-1 ; i++)
sl@0
   428
			{
sl@0
   429
			TRAPD(ret,\
sl@0
   430
				InsertEntryL(index,(*aSourceArray)[i]));
sl@0
   431
			if (ret!=KErrNone)
sl@0
   432
				{// do rollback, then propagate leave
sl@0
   433
				RollbackPaste();
sl@0
   434
				User::Leave(ret);
sl@0
   435
				}
sl@0
   436
			index++;
sl@0
   437
			}
sl@0
   438
		// join last field up to successor
sl@0
   439
		(*iFieldArray)[index].iPreFieldLen += (*aSourceArray)[numFieldEntries-1].iPreFieldLen;
sl@0
   440
		}
sl@0
   441
sl@0
   442
	__TEST_INVARIANT;
sl@0
   443
	}
sl@0
   444
sl@0
   445
sl@0
   446
void CTextFieldSet::RecordRollbackInfoL(TInt aIndex)
sl@0
   447
	{
sl@0
   448
	delete iRollbackInfo;
sl@0
   449
	iRollbackInfo = new(ELeave) TRollbackInfo();
sl@0
   450
	iRollbackInfo->iEntryNum = aIndex;
sl@0
   451
	iRollbackInfo->iPreFieldLen = (*iFieldArray)[aIndex].iPreFieldLen;
sl@0
   452
	iRollbackInfo->iFieldValueLen = (*iFieldArray)[aIndex].iFieldValueLen;
sl@0
   453
	iRollbackInfo->iTotalEntries = iFieldArray->Count();
sl@0
   454
	}
sl@0
   455
sl@0
   456
sl@0
   457
EXPORT_C void CTextFieldSet::RollbackPaste()
sl@0
   458
// Carries out rollback from a paste function
sl@0
   459
// This will only have an effect after a PasteFromStream() has been called
sl@0
   460
// nb it would be distasterous if this were called at random some time after a paste!
sl@0
   461
//
sl@0
   462
	{
sl@0
   463
	if (!iRollbackInfo)
sl@0
   464
		return; // nothing to do
sl@0
   465
	// remove added entries from array
sl@0
   466
	TInt entriesToRemove=iFieldArray->Count()-iRollbackInfo->iTotalEntries;
sl@0
   467
	TInt i=0;
sl@0
   468
	for (i=iRollbackInfo->iEntryNum ; i<iRollbackInfo->iEntryNum+entriesToRemove ; i++)
sl@0
   469
		{
sl@0
   470
		if ((*iFieldArray)[i].iFieldHeader.iField.IsPtr())
sl@0
   471
			delete (*iFieldArray)[i].iFieldHeader.iField.AsPtr(); // Delete the textField object
sl@0
   472
		iFieldArray->Delete(i);
sl@0
   473
		}
sl@0
   474
	// now right num entries, but wrong length - use backup info to correct length
sl@0
   475
	(*iFieldArray)[i].iPreFieldLen = iRollbackInfo->iPreFieldLen;
sl@0
   476
	(*iFieldArray)[i].iFieldValueLen = iRollbackInfo->iFieldValueLen;
sl@0
   477
sl@0
   478
	__ASSERT_DEBUG(iFieldArray->Count()==iRollbackInfo->iTotalEntries,Panic(EDebug));
sl@0
   479
	delete iRollbackInfo;
sl@0
   480
	}
sl@0
   481