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.
28 EXPORT_C void CTextFieldSet::__DbgTestInvariant()const
29 // Provides class invariants. Explanations below:
35 // check that every entry in the array has length >=0 and that all but the last have valid header handles
37 TInt index = iFieldArray->Count()-1;
41 {// check last entry: should have null fieldHeader, zero field length and non-negative PreFieldLen
42 if ( (*iFieldArray)[index].iFieldHeader.iField
43 ||(*iFieldArray)[index].iFieldValueLen!=0
44 ||(*iFieldArray)[index].iPreFieldLen<0 )
49 // check the other entries: should have non-null fieldHeader and non-negative field length and PreFieldLen
50 for (; (index>=0)&&(valid) ; index--)
52 if ((*iFieldArray)[index].iFieldValueLen<0
53 ||(*iFieldArray)[index].iPreFieldLen<0 )
59 __ASSERT_ALWAYS(valid,User::Invariant());
64 EXPORT_C CTextFieldSet* CTextFieldSet::NewL(TInt aDocumentLength)
66 CTextFieldSet* self=new(ELeave) CTextFieldSet();
67 CleanupStack::PushL(self);
68 self->ConstructL(aDocumentLength);
74 EXPORT_C CTextFieldSet* CTextFieldSet::NewL(const MTextFieldFactory* aFactory,const CStreamStore& aFieldStore,TStreamId aStreamId)
76 CTextFieldSet* self=new(ELeave) CTextFieldSet();
77 CleanupStack::PushL(self);
78 self->SetFieldFactory(CONST_CAST(MTextFieldFactory*,aFactory));
80 self->DoRestoreL(aFieldStore,aStreamId);
86 CTextFieldSet::CTextFieldSet()
90 void CTextFieldSet::ConstructL(TInt aDocumentLength)
91 // Creates an array in which to store all fields
92 // Inserts an initial entry into the array to cover any text that lies after the last field
95 iFieldArray = new(ELeave) CArrayFixSeg<TTextFieldEntry>(KFieldArrayGranularity);
96 AddInitialFieldEntryL(iFieldArray,aDocumentLength);
102 EXPORT_C CTextFieldSet::~CTextFieldSet()
104 delete iRollbackInfo;
107 TInt fieldCount=iFieldArray->Count();
108 for (TInt index=fieldCount-1;index>=0;index--)
109 delete (*iFieldArray)[index].iFieldHeader.iField.AsPtr(); // Delete the textField objects
115 EXPORT_C void CTextFieldSet::SetFieldFactory(MTextFieldFactory* aFactory)
117 iFieldFactory = aFactory;
121 EXPORT_C MTextFieldFactory* CTextFieldSet::FieldFactory()const
123 return iFieldFactory;
127 void CTextFieldSet::InsertEntryL(TInt aIndex,TTextFieldEntry& aEntry)
129 InsertEntryL(aIndex,aEntry,iFieldArray);
133 void CTextFieldSet::InsertEntryL(TInt aIndex,TTextFieldEntry& aEntry,CArrayFixSeg<TTextFieldEntry>* aArray)
134 // if this function leaves it will be as if it had never been called...
137 if (aEntry.iFieldHeader.iField.IsPtr())
138 CleanupStack::PushL(aEntry.iFieldHeader.iField.AsPtr());
139 aArray->InsertL(aIndex,aEntry); // insert new entry
140 if (aEntry.iFieldHeader.iField.IsPtr())
145 void CTextFieldSet::AppendEntryL(TTextFieldEntry& aEntry)
147 AppendEntryL(aEntry,iFieldArray);
151 void CTextFieldSet::AppendEntryL(TTextFieldEntry& aEntry,CArrayFixSeg<TTextFieldEntry>* aArray)
153 if (aEntry.iFieldHeader.iField.IsPtr())
154 CleanupStack::PushL(aEntry.iFieldHeader.iField.AsPtr());
155 aArray->AppendL(aEntry); // insert new entry
156 if (aEntry.iFieldHeader.iField.IsPtr())
161 TInt CTextFieldSet::EntryLen(TInt aIndex)const
163 return EntryLen((*iFieldArray)[aIndex]);
167 TInt CTextFieldSet::EntryLen(const TTextFieldEntry& aEntry)const
169 return aEntry.iPreFieldLen+aEntry.iFieldValueLen;
173 TTextFieldEntry CTextFieldSet::SplitEntry(TInt aIndex,TInt aOffset, TInt aRange)const
174 // Splits the entry aIndex, returning the part demarked by the offset (from the start of the entry) and the range
179 __ASSERT_DEBUG((aIndex>=0)&&(aIndex<iFieldArray->Count()),Panic(EIndexOutOfRange));
180 TInt entryLength = EntryLen(aIndex);
181 __ASSERT_DEBUG((aOffset>=0)&&(aOffset<=entryLength),Panic(EPosOutOfRange));
182 __ASSERT_DEBUG((aRange>=0),Panic(ENegativeRange));
184 if ((aOffset+aRange)>entryLength)
185 aRange = entryLength-aOffset; // scale range down to entry size if neccessary
187 if ((aOffset==0)&&(aRange==entryLength))
188 return (*iFieldArray)[aIndex]; //entry does not need to be split
191 TTextFieldEntry entry;
192 entry.iPreFieldLen = 0;
193 entry.iFieldValueLen = 0;
194 entry.iFieldHeader.iField = NULL;
195 if (aOffset<(*iFieldArray)[aIndex].iPreFieldLen)
196 {// At least some of the pre needs to be copied
197 entry.iPreFieldLen = (*iFieldArray)[aIndex].iPreFieldLen-aOffset;
198 if ((entry.iPreFieldLen)>aRange)
199 entry.iPreFieldLen = aRange;
200 charsCopied = entry.iPreFieldLen;
203 if (charsCopied<aRange)
204 {// more to do, so at least some of the field needs to be copied
205 if ((aOffset+aRange)==entryLength)
206 {// whole field in range so copy it
207 entry.iFieldValueLen = (*iFieldArray)[aIndex].iFieldValueLen;
208 entry.iFieldHeader = (*iFieldArray)[aIndex].iFieldHeader;
211 // only part of field in range so convert it to text
212 entry.iPreFieldLen += aRange-charsCopied;
218 void CTextFieldSet::AddInitialFieldEntryL(CArrayFixSeg<TTextFieldEntry>* aArray,TInt aDocumentLength)
221 TTextFieldEntry initialEntry;
222 initialEntry.iPreFieldLen = aDocumentLength;
223 initialEntry.iFieldValueLen = 0;
224 initialEntry.iFieldHeader.iFieldType = KNullUid;
225 initialEntry.iFieldHeader.iField = NULL;
226 aArray->AppendL(initialEntry);
230 EXPORT_C void CTextFieldSet::Reset()
231 // deletes all fields (but not corresponding text) and reinitialises the field array
236 for (TInt index=FieldCount()-1 ; index>=0 ; index--)
237 delete (*iFieldArray)[index].iFieldHeader.iField.AsPtr(); // Delete the textField objects
238 iFieldArray->Reset();
239 AddInitialFieldEntryL(iFieldArray,0); // cannot leave in this context
240 iFieldArray->Compress(); // compress array
246 EXPORT_C CTextField* CTextFieldSet::NewFieldL(TUid aFieldType)
249 return iFieldFactory->NewFieldL(aFieldType);
255 EXPORT_C TInt CTextFieldSet::InsertFieldL(TInt aPos,CTextField* aField,TUid aFieldType)
256 // Inserts a field header aField at aPos (aField should be declared on the heap)
257 // The field initially has zero length: Update must be called afterward
261 __ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
262 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
263 __ASSERT_ALWAYS(aField,Panic(ENoTextField));
265 TInt errLevel=KErrAlreadyExists;
266 TTextFieldEntry entry;
267 entry.iFieldHeader.iField = aField;
268 entry.iFieldHeader.iFieldType = aFieldType;
269 entry.iFieldValueLen = 0;
270 TInt index; TInt offset;
271 TBool inField = InField(aPos,index,offset);
273 {// not inserting into a field
274 entry.iPreFieldLen = offset;
275 iFieldArray->InsertL(index,entry); // insert new entry
276 (*iFieldArray)[index+1].iPreFieldLen -= offset; // update old entry
280 {// at start of field
281 entry.iPreFieldLen = (*iFieldArray)[index].iPreFieldLen;
282 iFieldArray->InsertL(index,entry); // insert new entry
283 (*iFieldArray)[index+1].iPreFieldLen = 0; // update old entry
292 EXPORT_C const CTextField* CTextFieldSet::TextField(TInt aPos)const
293 // Return a handle to the concrete field at document position aPos.
294 // Returns NULL if there is no field at position aPos.
301 TBool inField=InField(aPos,index,offset);
304 TSwizzle<CTextField> field=(*iFieldArray)[index].iFieldHeader.iField;
305 __ASSERT_DEBUG(field.IsPtr(),User::Invariant());
306 return field.AsPtr();
310 EXPORT_C TInt CTextFieldSet::RemoveField(TInt aPos)
311 // Removes the field from the array, adding it's content to the "before" of the next field.
312 // After this function is called the text the field contained should be deleted. If this does
313 // not happen this function acts as a "ConvertFieldToText()"
317 __ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
318 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
320 TInt errLevel=KErrNone;
321 TInt index; TInt offset;
322 if (!InField(aPos,index,offset))
323 errLevel = KErrNotFound;
325 // delete the field entry tidily
326 DeleteFieldEntry(index);
333 EXPORT_C TInt CTextFieldSet::FieldCount() const
337 // return count-1 because there is always an entry in the array for the text at the
338 // end of the document, after the last field (maybe just the end of doc char)
339 return (iFieldArray->Count()-1);
343 EXPORT_C TInt CTextFieldSet::CharCount() const
344 // returns the number of characters in the document according to the field array
349 for (TInt index=FieldCount() ; index>=0 ; index--)
350 charCount += (*iFieldArray)[index].iPreFieldLen+(*iFieldArray)[index].iFieldValueLen;
355 EXPORT_C TBool CTextFieldSet::FindFields(TInt aPos) const
356 // Return ETrue if aPos is in a field
359 TFindFieldInfo dummy;
360 return (FindFields(dummy,aPos,0));
364 EXPORT_C TBool CTextFieldSet::FindFields(TFindFieldInfo& aInfo,TInt aPos, TInt aRange) const
365 // Check whether aPos is in a field, then check whether any fields start in aRange
369 __ASSERT_ALWAYS(aRange>=0,Panic(ENegativeRange));
370 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutsideDoc));
371 __ASSERT_ALWAYS(aPos+aRange<=CharCount(),Panic(EPosOutsideDoc));
373 aInfo.iFieldCountInRange = 0;
374 aInfo.iFirstFieldLen = 0;
375 aInfo.iFirstFieldPos = 0;
376 TInt pos=aPos; // position in doc
377 // are we in a field to begin with?
378 TInt index; TInt offset;
379 if (InField(aPos,index,offset))
381 aInfo.iFieldCountInRange++;
382 aInfo.iFirstFieldLen = (*iFieldArray)[index].iFieldValueLen;
383 aInfo.iFirstFieldPos = aPos-offset;
384 pos += (*iFieldArray)[index].iFieldValueLen-offset+(*iFieldArray)[index+1].iPreFieldLen;
388 pos += (*iFieldArray)[index].iPreFieldLen-offset;
389 // step through array until aRange
390 while (pos<aPos+aRange)
391 // When entering this loop we're always at the beginning of a field value
392 // set aFieldEntry to first field found (if fieldcount==0)
393 // for each start of field encountered, fieldCount++
395 if (aInfo.iFieldCountInRange==0)
397 aInfo.iFirstFieldLen = (*iFieldArray)[index].iFieldValueLen;
398 aInfo.iFirstFieldPos = pos;
400 aInfo.iFieldCountInRange++;
401 pos += (*iFieldArray)[index].iFieldValueLen+(*iFieldArray)[index+1].iPreFieldLen;
404 // (if sensing right) check that we haven't ended adjacent to one or more zero-length fields
405 if ( (aRange==0) && (pos==aPos) && (index<(iFieldArray->Count()-1)) && ((*iFieldArray)[index].iFieldValueLen==0) )
407 aInfo.iFieldCountInRange++;
409 while ( ((*iFieldArray)[index].iPreFieldLen==0) && ((*iFieldArray)[index].iFieldValueLen==0) && (index<(iFieldArray->Count()-1)) )
411 aInfo.iFieldCountInRange++;
416 __ASSERT_DEBUG(aInfo.iFieldCountInRange<=FieldCount(),Panic(EDebug));
417 return aInfo.iFieldCountInRange;
421 EXPORT_C TInt CTextFieldSet::NewFieldValueL(HBufC*& aBuf, TInt aPos)
422 // Returns the new value of the field at aPos (if applicable).
423 // The method might reallocate aBuf parameter, so don't forget to:
424 // 1) Push aBuf parameter in the CleanupStack before the call. The call
425 // may fail, then aBuf gets lost.
426 // 2) Pop aBuf parameter from the CleanupStack after the call - aBuf may get
427 // reallocated, so the pushed pointer becomes invalid.
428 // 3) Push aBuf in the CleanupStack again.
431 __ASSERT_ALWAYS(aBuf,Panic(ENoBuffer));
432 __ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
433 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
435 TInt errLevel = KErrNotFound;
436 TInt index; TInt offset;
437 if (InField(aPos,index,offset))
438 {// There's a field at aPos
439 TPtr bufPtr = aBuf->Des();
440 TInt reqLen = (*iFieldArray)[index].iFieldHeader.iField->Value(bufPtr);
442 {// If more storage is required for the value then reallocate the buffer
443 aBuf = aBuf->ReAllocL(reqLen); // for unicode compatability, rounds up
444 TPtr pointer = aBuf->Des();
445 reqLen = (*iFieldArray)[index].iFieldHeader.iField->Value(pointer);
447 // dont set new field length - this will be done in a subsequent NotifyInsert()
456 EXPORT_C void CTextFieldSet::NotifyInsertion(TInt aPos, TInt aNumberAdded)
457 // Informs the array that aNumberAdded characters have been inserted in the document
461 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutsideDoc));
462 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
463 __ASSERT_ALWAYS(aNumberAdded>=0,Panic(EIllegalNegativeValue));
465 TInt index; TInt offset;
466 // are the extra characters in a field (matching away from fields)?
467 if (!InField(aPos,index,offset))
468 (*iFieldArray)[index].iPreFieldLen += aNumberAdded;
471 (*iFieldArray)[index].iFieldValueLen += aNumberAdded;
473 (*iFieldArray)[index].iPreFieldLen += aNumberAdded;
479 EXPORT_C void CTextFieldSet::NotifyFieldUpdate(TInt aPos, TInt aNewFieldValueLength)
482 __ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
483 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
484 __ASSERT_ALWAYS(aNewFieldValueLength>=0,Panic(EIllegalNegativeValue));
486 // Is the insert pos in a field? If so which?
487 TInt index; TInt offset;
488 __ASSERT_ALWAYS(InField(aPos,index,offset),Panic(EPosNotInField));
489 // Update the length of the relevant field
490 (*iFieldArray)[index].iFieldValueLen = aNewFieldValueLength;
496 EXPORT_C void CTextFieldSet::NotifyDeletion(TInt aPos,TInt aTotalRemoved)
497 // Informs the array that aTotalRemoved characters have been removed from the document
498 // Any fields wholely contained will be removed, those partially intersecting will just be shortened
502 __ASSERT_ALWAYS(aPos>=0,Panic(EPosOutsideDoc));
503 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
504 __ASSERT_ALWAYS(aTotalRemoved>=0,Panic(EIllegalNegativeValue));
506 TInt charCount = CharCount();
508 //There is a possibility that there exists hidden end-of-document character in rich text objects.
509 //This is checked by the if statement & accordingly aTotalRemoved is decremented.
510 //i.e. aPos + aTotalRemoved could be greater by 1 than the CharCount(),
511 //this is because the aTotalRemoved might also include 1 hidden character. - DEF095911
512 if(aPos+aTotalRemoved > charCount)
515 __ASSERT_DEBUG(aPos+aTotalRemoved<=charCount, Panic(EPosOutsideDoc));
520 int end_pos = aPos + aTotalRemoved;
521 TBool in_field = InField(cur_pos,index,offset);
522 TTextFieldEntry* field = NULL;
525 if (index >= 0 && index < iFieldArray->Count())
527 field = &(*iFieldArray)[index];
528 field_start = cur_pos - offset;
530 field_start += field->iPreFieldLen;
531 field_end = field_start + field->iFieldValueLen;
536 // Reduce the size of the gap before the field if any.
537 int gap = Min(end_pos,field_start) - cur_pos;
541 field->iPreFieldLen -= gap;
543 if (cur_pos >= end_pos)
546 // Reduce the field length.
547 int remove_start = cur_pos;
548 int remove_end = Min(field_end,end_pos);
550 field->iFieldValueLen -= (remove_end - remove_start);
552 // Delete the field if it is now of zero length.
553 int added_to_next = 0;
554 if (field->iFieldValueLen == 0)
556 added_to_next = field->iPreFieldLen;
557 DeleteFieldEntry(index);
562 // Move to the next field.
563 if (index < iFieldArray->Count())
565 field = &(*iFieldArray)[index];
566 field_start = cur_pos + field->iPreFieldLen - added_to_next;
567 field_end = field_start + field->iFieldValueLen;
577 TBool CTextFieldSet::InField(const TInt aPos, TInt& anIndex, TInt& anOffset) const
578 // works out whether or not aPos is in a field (matching right),
579 // sets anIndex to the index number of the field entry,
580 // and sets anOffset to the distance aPos is into the field or its preceeding gap
583 __ASSERT_ALWAYS((aPos>=0),Panic(EPosOutsideDoc));
584 __ASSERT_DEBUG(aPos<=CharCount(),Panic(EPosOutsideDoc));
589 TInt lengthOfDoc = CharCount();
590 TInt lastFieldNum = iFieldArray->Count()-1;
591 // Find the index of the entry containing aPos
592 if (aPos == lengthOfDoc)
594 anIndex = lastFieldNum;
600 while (aPos >= currentPos)
603 currentPos += (*iFieldArray)[anIndex].iPreFieldLen+(*iFieldArray)[anIndex].iFieldValueLen;
606 // Check that we have not skipped over any zero-length fields
607 if ( ((anIndex-1)>=0) && ((*iFieldArray)[anIndex-1].iFieldValueLen==0) )
609 TInt temp = currentPos - ((*iFieldArray)[anIndex].iPreFieldLen+(*iFieldArray)[anIndex].iFieldValueLen);
611 {// aPos is on a field boundary
614 while ( ((anIndex-1)>=0)
615 &&((*iFieldArray)[anIndex].iPreFieldLen==0)
616 &&((*iFieldArray)[anIndex-1].iFieldValueLen==0) )
622 // Move to the start of the field (end of prefield) in entry [anIndex]
623 currentPos -= (*iFieldArray)[anIndex].iFieldValueLen;
624 // Determine whether or not aPos is in the field of entry[anIndex]
625 if (anIndex == lastFieldNum)
627 else if (aPos >= currentPos)
631 // Calculate the offset
633 anOffset = aPos-currentPos;
635 anOffset = aPos+(*iFieldArray)[anIndex].iPreFieldLen-currentPos;
640 void CTextFieldSet::DeleteFieldEntry(TInt anIndex)
642 DeleteFieldEntry(iFieldArray,anIndex);
646 void CTextFieldSet::DeleteFieldEntry(CArrayFixSeg<TTextFieldEntry>* aArray,TInt anIndex)
647 // remove the entry anIndex from the array but don't delete the text from the doc.
650 __ASSERT_ALWAYS(anIndex<(aArray->Count()-1),Panic(EIndexOutOfRange));
652 // add the entry's "before" to the "before" of the next entry.
653 (*aArray)[anIndex+1].iPreFieldLen += (*aArray)[anIndex].iPreFieldLen;
654 // add the field's length to the "before" of the next entry.
655 (*aArray)[anIndex+1].iPreFieldLen += (*aArray)[anIndex].iFieldValueLen;
656 if ((*aArray)[anIndex].iFieldHeader.iField.IsPtr())
657 delete (*aArray)[anIndex].iFieldHeader.iField.AsPtr(); // delete the field
658 aArray->Delete(anIndex); // remove the entry from the array
662 ///////////////////////////////////////////////
664 ///////////////////////////////////////////////
667 EXPORT_C void TTextFieldEntry::InternalizeL(RReadStream& aStream)
668 // entry must have a header (ie cant be last entry in the array)
670 iPreFieldLen = aStream.ReadInt32L();
671 iFieldValueLen = aStream.ReadInt32L();
672 aStream>> iFieldHeader;
676 EXPORT_C void TTextFieldEntry::ExternalizeL(RWriteStream& aStream)const
677 // entry must have a header (ie cant be last entry in the array)
679 aStream.WriteInt32L(iPreFieldLen);
680 aStream.WriteInt32L(iFieldValueLen);
681 aStream<< iFieldHeader;