Update contrib.
2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
33 EXPORT_C TFieldMapExternalizer::TFieldMapExternalizer(const CStoreMap& aMap)
37 EXPORT_C void TFieldMapExternalizer::ExternalizeL(const TStreamRef& aRef,RWriteStream& aStream) const
38 // Write the stream id bound to aRef to aStream. If not bound, write KNullStreamId
41 TSwizzleC<TAny> swizzle=aRef;
42 aStream<<iMap->At(swizzle);
46 EXPORT_C TStreamId CTextFieldSet::StoreL(CStreamStore& aStore)const
47 // Save the fields and the fieldSet in their own streams
48 // Encapsulates the storing of its components.
51 CStoreMap* map=CStoreMap::NewLC(aStore);
52 StoreFieldsL(aStore,*map); // binds id's to swizzles
54 // create custom externalizer over the map
55 TFieldMapExternalizer fMap(*map);
56 RStoreWriteStream stream(fMap);
57 TStreamId id=stream.CreateLC(aStore);
60 CleanupStack::PopAndDestroy(); // stream
63 CleanupStack::PopAndDestroy(); // map
68 EXPORT_C void CTextFieldSet::StoreFieldsL(CStreamStore& aStore,CStoreMap& aMap)const
69 // Stores all fields in the set
71 {StoreFieldsL(aStore,aMap,iFieldArray);}
74 void CTextFieldSet::StoreFieldsL(CStreamStore& aStore,CStoreMap& aMap,CArrayFixSeg<TTextFieldEntry>* aArray)const
75 // Stores all fields contained in the set provided
80 for (TInt i=0 ; i<(aArray->Count()-1) ; i++)
82 TStreamId id=(*aArray)[i].iFieldHeader.iField->StoreL(aStore);
83 if (id!=KNullStreamId)
84 aMap.BindL((*aArray)[i].iFieldHeader.iField,id);
89 EXPORT_C void CTextFieldSet::ExternalizeL(RWriteStream& aStream)const
93 ExternalizeL(aStream,iFieldArray);
97 void CTextFieldSet::ExternalizeL(RWriteStream& aStream,CArrayFixSeg<TTextFieldEntry>* aArray)const
99 TInt numFieldEntries = aArray->Count();
100 aStream.WriteInt32L(numFieldEntries);
102 for (TInt i=0 ; i<numFieldEntries-1 ; i++)
103 aStream<< (*aArray)[i];
104 // write out last entry in array: the bit after the last field
105 aStream.WriteInt32L((*aArray)[numFieldEntries-1].iPreFieldLen);
109 EXPORT_C void CTextFieldSet::RestoreL(const CStreamStore& aFieldStore,TStreamId aStreamId)
111 // reset the array and stream into it
113 DoRestoreL(aFieldStore,aStreamId);
117 EXPORT_C void CTextFieldSet::RestoreFieldsL(const CStreamStore& aFieldStore)
119 DoRestoreFieldsL(iFieldArray,aFieldStore); // restore the fields individually from their own streams
123 void CTextFieldSet::DoRestoreL(const CStreamStore& aFieldStore,TStreamId aStreamId)
124 // Restores a field set and its associated fields from the store provded.
127 __ASSERT_ALWAYS(iFieldArray->Count()==1,Panic(EArrayNotEmptyOnRestore)); // array must be empty
128 __ASSERT_ALWAYS((*iFieldArray)[0].iPreFieldLen==0,Panic(EArrayNotEmptyOnRestore));
130 // retrieve the headstream from the store
131 RStoreReadStream stream;
132 stream.OpenLC(aFieldStore,aStreamId);
133 // restore the set, then the individual fields
134 stream>> *this; // internalize the field set (the headers)
135 CleanupStack::PopAndDestroy(); // stream
136 DoRestoreFieldsL(iFieldArray,aFieldStore); // restore the fields individually from their own streams
140 EXPORT_C void CTextFieldSet::InternalizeL(RReadStream& aStream)
142 InternalizeL(iFieldArray,aStream);
148 void CTextFieldSet::InternalizeL(CArrayFixSeg<TTextFieldEntry>* aArray,RReadStream& aStream)
149 {// assume the array is empty
150 TInt numFieldEntries = aStream.ReadInt32L();
151 // read in the fields
152 TTextFieldEntry entry;
153 for (TInt i=0 ; i<numFieldEntries-1 ; i++)
156 InsertEntryL(i,entry,aArray); // insert new entry
158 // read in the last entry: the bit after the last field. This will not contain a field
159 (*aArray)[numFieldEntries-1].iPreFieldLen = aStream.ReadInt32L();
163 void CTextFieldSet::DoRestoreFieldsL(CArrayFixSeg<TTextFieldEntry>* aArray,const CStreamStore& aFieldStore,TInt aStartIndex)
164 // This fn is called after all FieldHeaders have been internalized - the Swizzles hold the stream id's.
165 // One by one the fields are created (with the factory) and then have their settings streamed in from the store.
166 // If no factory exists all fields are converted to text.
169 TInt ii=aArray->Count()-2; // -2 because we skip the last (empty) entry
170 while (ii>=aStartIndex)
172 if ((*aArray)[ii].iFieldHeader.iField.IsId())
173 {// restore the field only if it isn't the very last (dummy) entry
174 if (iFieldFactory==NULL)
175 // no factory - convert the field to text
176 DeleteFieldEntry(aArray,ii);
179 TStreamId id = (*aArray)[ii].iFieldHeader.iField.AsId();
180 (*aArray)[ii].iFieldHeader.iField = iFieldFactory->NewFieldL((*aArray)[ii].iFieldHeader.iFieldType);
181 if ((*aArray)[ii].iFieldHeader.iField!=NULL)
182 (*aArray)[ii].iFieldHeader.iField->RestoreL(aFieldStore,id);
184 DeleteFieldEntry(aArray,ii); // handle "field type not recognised" by converting field to text
192 /***************************************** cut & paste *******************************************/
195 EXPORT_C TStreamId CTextFieldSet::CopyToStoreL(CStreamStore& aStore,TInt aPos,TInt aLength)const
196 // Copy any fields in the selected region to the specified store, returning the id of the head-stream.
200 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
201 __ASSERT_ALWAYS(aLength>=0,Panic(ENegativeRange));
202 __ASSERT_DEBUG((aPos+aLength)<=CharCount(),Panic(EPosOutOfRange));
204 // Create a store map and store the fields themselves
205 CStoreMap* map=CStoreMap::NewLC(aStore);
206 CopyComponentsL(aStore,*map,aPos,aLength);
208 // Create a head-stream in which to store the field entries and do so
209 RStoreWriteStream stream(*map);
210 TStreamId id=stream.CreateLC(aStore);
211 CopyToStreamL(stream,aPos,aLength);
216 CleanupStack::PopAndDestroy(2); // map, stream
221 EXPORT_C void CTextFieldSet::CopyComponentsL(CStreamStore& aStore,CStoreMap& aMap,TInt aPos,TInt aLength)const
222 // Stores all fields in the set
226 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
227 __ASSERT_ALWAYS(aLength>=0,Panic(ENegativeRange));
228 __ASSERT_DEBUG((aPos+aLength)<=CharCount(),Panic(EPosOutOfRange));
230 // Create an array of the fields to be cut/copied
231 CArrayFixSeg<TTextFieldEntry>* tempArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
232 CleanupStack::PushL(tempArray);
233 CopyToArrayL(tempArray,aPos,aLength);
235 // stream the required fields in their own streams
236 StoreFieldsL(aStore,aMap,tempArray);
237 CleanupStack::PopAndDestroy(); // tempArray
241 EXPORT_C void CTextFieldSet::CopyToStreamL(RWriteStream& aStream,TInt aPos,TInt aLength)const
242 // Stores all fields in the set
246 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
247 __ASSERT_ALWAYS(aLength>=0,Panic(ENegativeRange));
248 __ASSERT_DEBUG((aPos+aLength)<=CharCount(),Panic(EPosOutOfRange));
250 // Create an array of the fields to be cut/copied
251 CArrayFixSeg<TTextFieldEntry>* tempArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
252 CleanupStack::PushL(tempArray);
253 CopyToArrayL(tempArray,aPos,aLength);
255 // stream the field entries in the temp array
256 ExternalizeL(aStream,tempArray);
257 CleanupStack::PopAndDestroy(); // tempArray
261 void CTextFieldSet::CopyToArrayL(CArrayFixSeg<TTextFieldEntry>* aArray,TInt aPos,TInt aLength)const
263 TInt index; TInt offset;
264 if (InField(aPos,index,offset))
265 offset += (*iFieldArray)[index].iPreFieldLen; // make offset relative to start of entry
266 // split first entry in range
267 TTextFieldEntry entry = SplitEntry(index,offset,aLength);
269 TInt charsCopied=EntryLen(entry);
270 // split second if neccessary
271 if ((!entry.iFieldHeader.iField)&&(charsCopied<aLength))
273 TInt preFieldLen = entry.iPreFieldLen;
274 entry = SplitEntry(index,0,aLength-preFieldLen);
275 entry.iPreFieldLen += preFieldLen;
276 charsCopied = EntryLen(entry);
279 ((CTextFieldSet*)this)->AppendEntryL(entry,aArray); // append the first entry to the storage array
280 // write out all whole entries
281 while (charsCopied<aLength)
283 if ((charsCopied+EntryLen(index))<=aLength)
284 ((CTextFieldSet*)this)->AppendEntryL((*iFieldArray)[index],aArray);
285 charsCopied += EntryLen(index);
288 // split last entry if neccessary
289 if (charsCopied>aLength)
290 {// The last entry needs to be split
291 // first get back to the beginning of the entry
293 charsCopied -= EntryLen(index);
294 entry = SplitEntry(index,0,aLength-charsCopied); // split up the last entry as required
295 ((CTextFieldSet*)this)->AppendEntryL(entry,aArray); // append the last entry to the storage array
297 // add an empty last entry if neccessary
298 TInt numFieldEntries = aArray->Count();
299 if (((*aArray)[numFieldEntries-1].iFieldHeader.iField) || ((*aArray)[numFieldEntries-1].iFieldValueLen!=0))
301 entry.iPreFieldLen = 0;
302 entry.iFieldValueLen = 0;
303 entry.iFieldHeader.iFieldType = KNullUid;
304 entry.iFieldHeader.iField = NULL;
305 ((CTextFieldSet*)this)->AppendEntryL(entry,aArray);
311 EXPORT_C void CTextFieldSet::PasteFromStoreL(const CStreamStore& aFieldStore,TStreamId aStreamId,TInt aPos,TInt aMaxLen)
312 // Paste from aStore into the document at insert position aPos.
313 // Optionally the pasted text can be clipped to a maximum length aMaxLen.
316 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutOfRange));
317 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutOfRange));
319 // retrieve the headstream from the store
320 RStoreReadStream stream;
321 stream.OpenLC(aFieldStore,aStreamId);
323 // restore the set...
324 PasteFromStreamL(stream,aPos,aMaxLen); // internalize the field set (the headers)
325 CleanupStack::PopAndDestroy(); // stream
327 // ...then the individual fields
328 PasteComponentsL(aFieldStore,aPos); // restore the fields individually from their own streams
332 EXPORT_C void CTextFieldSet::PasteFromStreamL(RReadStream& aStream,TInt aPos,TInt aMaxLen)
333 // streams the field entries into a temporary array, which is returned.
334 // PasteComponents() must be called after this to actually carry out the paste...
336 // create a temporary array to stream in to, inserting the first entry
337 CArrayFixSeg<TTextFieldEntry>* tempFieldArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
338 CleanupStack::PushL(tempFieldArray);
339 AddInitialFieldEntryL(tempFieldArray,0);
341 // internalize the field entries
342 InternalizeL(tempFieldArray,aStream);
344 // trim off any entries that lie beyond aMaxLength
345 if (aMaxLen!=ENoPasteLimit)
346 {// if aMaxLen is not ENoPasteLimit discard the excess fields
347 __ASSERT_ALWAYS(aMaxLen>=0,Panic(ELengthOutOfRange));
351 for (i=0 ; (length<aMaxLen)&&(i<tempFieldArray->Count()) ; i++)
352 length += EntryLen((*tempFieldArray)[i]);
354 {// make first entry zero len, delete all others
356 (*tempFieldArray)[i-1].iPreFieldLen = 0;
358 else if (length>aMaxLen)
359 // truncate the last field in range
360 (*tempFieldArray)[i-1].iPreFieldLen += (*tempFieldArray)[i-1].iFieldValueLen-(length-aMaxLen);
361 else if ((length==aMaxLen) && ((*tempFieldArray)[i-1].iFieldHeader.iField!=NULL))
362 {// if the terminating entry has a field add a zero length entry, the mandatory last entry
364 (*tempFieldArray)[i-1].iPreFieldLen = 0;
366 // ensure the last entry is of the correct format
367 (*tempFieldArray)[i-1].iFieldValueLen = 0;
368 (*tempFieldArray)[i-1].iFieldHeader.iFieldType = KNullUid;
369 (*tempFieldArray)[i-1].iFieldHeader.iField = NULL;
370 // delete all the fields wholely out of range
371 for (TInt index=i ; index<tempFieldArray->Count() ; index++)
372 (*tempFieldArray)[index].iFieldHeader.iField = NULL;
373 tempFieldArray->Delete(i,tempFieldArray->Count()-i); // pos,count
376 DoPasteL(tempFieldArray,aPos);
377 CleanupStack::PopAndDestroy(); // tempFieldArray
381 EXPORT_C void CTextFieldSet::PasteComponentsL(const CStreamStore& aFieldStore,TInt aPos)
383 // Restore the fields individually from their own streams
384 TInt index; TInt offset;
385 // We don't need to make any difference between in and not in field situation here
386 // all we need is the index
387 TBool isInField = InField(aPos,index,offset);
388 DoRestoreFieldsL(iFieldArray,aFieldStore,index);
392 void CTextFieldSet::DoPasteL(CArrayFixSeg<TTextFieldEntry>* aSourceArray,TInt aPos)
393 // Insert into this instance, at character position aPos, the entire (field entry) contents of the field array aSourceArray.
394 // All iField objects in aSourceArray are ID's at this time.
397 // are we inserting into a field?
398 TInt numFieldEntries = aSourceArray->Count();
399 TInt index; TInt offset;
401 TBool inField = InField(aPos,index,offset);
402 // record the rollback info
403 RecordRollbackInfoL(index);
404 if ((inField)&&(offset!=0))
405 {// everything we insert will become text - no chance of leaving
406 // insert all but last entry
408 for (; i<numFieldEntries-1 ; i++)
409 {// copy text (no need to delete field)
410 (*iFieldArray)[index].iFieldValueLen += EntryLen((*aSourceArray)[i]);
412 // read in the last entry (has no field attached)
413 (*iFieldArray)[index].iFieldValueLen += (*aSourceArray)[i].iPreFieldLen;
416 {// else split the entry we are going to bisect - this may leave
418 offset = (*iFieldArray)[index].iPreFieldLen; // must be at start of field
419 if (numFieldEntries>1)
420 {// read 1st field & carry out split.
421 InsertEntryL(index,(*aSourceArray)[0]); // if this leaves the model will be intact
422 (*iFieldArray)[index].iPreFieldLen += offset;
423 (*iFieldArray)[index+1].iPreFieldLen -= offset;
426 // insert all other fields except last.
427 for (TInt i=1 ; i<numFieldEntries-1 ; i++)
430 InsertEntryL(index,(*aSourceArray)[i]));
432 {// do rollback, then propagate leave
438 // join last field up to successor
439 (*iFieldArray)[index].iPreFieldLen += (*aSourceArray)[numFieldEntries-1].iPreFieldLen;
446 void CTextFieldSet::RecordRollbackInfoL(TInt aIndex)
448 delete iRollbackInfo;
449 iRollbackInfo = new(ELeave) TRollbackInfo();
450 iRollbackInfo->iEntryNum = aIndex;
451 iRollbackInfo->iPreFieldLen = (*iFieldArray)[aIndex].iPreFieldLen;
452 iRollbackInfo->iFieldValueLen = (*iFieldArray)[aIndex].iFieldValueLen;
453 iRollbackInfo->iTotalEntries = iFieldArray->Count();
457 EXPORT_C void CTextFieldSet::RollbackPaste()
458 // Carries out rollback from a paste function
459 // This will only have an effect after a PasteFromStream() has been called
460 // nb it would be distasterous if this were called at random some time after a paste!
464 return; // nothing to do
465 // remove added entries from array
466 TInt entriesToRemove=iFieldArray->Count()-iRollbackInfo->iTotalEntries;
468 for (i=iRollbackInfo->iEntryNum ; i<iRollbackInfo->iEntryNum+entriesToRemove ; i++)
470 if ((*iFieldArray)[i].iFieldHeader.iField.IsPtr())
471 delete (*iFieldArray)[i].iFieldHeader.iField.AsPtr(); // Delete the textField object
472 iFieldArray->Delete(i);
474 // now right num entries, but wrong length - use backup info to correct length
475 (*iFieldArray)[i].iPreFieldLen = iRollbackInfo->iPreFieldLen;
476 (*iFieldArray)[i].iFieldValueLen = iRollbackInfo->iFieldValueLen;
478 __ASSERT_DEBUG(iFieldArray->Count()==iRollbackInfo->iTotalEntries,Panic(EDebug));
479 delete iRollbackInfo;