os/textandloc/textrendering/texthandling/stext/TXTFMSTM.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
     3 * All rights reserved.
     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".
     8 *
     9 * Initial Contributors:
    10 * Nokia Corporation - initial contribution.
    11 *
    12 * Contributors:
    13 *
    14 * Description: 
    15 *
    16 */
    17 
    18 
    19 #include <e32std.h>
    20 
    21 #include <s32strm.h>
    22 #include <s32mem.h>
    23 #include "TXTFRMAT.H"
    24 #include <txtfmstm.h>
    25 
    26 #include "TXTSTD.H"
    27 #include "OstTraceDefinitions.h"
    28 #ifdef OST_TRACE_COMPILER_IN_USE
    29 #include "TXTFMSTMTraces.h"
    30 #endif
    31 
    32 
    33 const TInt KMaxFormatStreamLength=0x400;  // 1024 bytes
    34 
    35 // Standard attributes
    36 const TUint KFontProportional=0x01;
    37 const TUint KFontSerif=0x02;
    38 const TUint KFontSymbol=0x04;
    39 
    40 const TUint KTypefaceFlags=sizeof(TUint8);
    41 const TInt KTypeSize=sizeof(TUint8);
    42 const TInt KColor=sizeof(TUint8)+sizeof(TUint8)+sizeof(TUint8);
    43 const TInt KFontHeight=sizeof(TInt32);
    44 const TInt KParaBorderThickness=sizeof(TInt32);
    45 const TInt KTabPosition=sizeof(TUint32);
    46 const TInt KTabType=sizeof(TUint8);
    47 
    48 // Paragraph format attributes
    49 const TInt KVariableLengthAttribute=0;
    50 const TInt KParaLanguage=sizeof(TUint8);
    51 const TInt KParaFillColor=KColor;
    52 const TInt KParaLeftMargin=sizeof(TInt32);
    53 const TInt KParaRightMargin=sizeof(TInt32);
    54 const TInt KParaIndent=sizeof(TInt32);
    55 const TInt KParaAlignment=sizeof(TUint8);
    56 const TInt KParaVerticalAlignment=sizeof(TUint8);
    57 const TInt KParaLineSpacing=sizeof(TInt32);
    58 const TInt KParaLineSpacingControl=sizeof(TUint8);
    59 const TInt KParaSpaceBefore=sizeof(TInt32);
    60 const TInt KParaSpaceAfter=sizeof(TInt32);
    61 const TInt KParaKeepTogether=sizeof(TUint8);
    62 const TInt KParaKeepWithNext=sizeof(TUint8);
    63 const TInt KParaStartNewPage=sizeof(TUint8);
    64 const TInt KParaWidowOrphan=sizeof(TUint8);
    65 const TInt KParaWrap=sizeof(TUint8);
    66 const TInt KParaBorderMargin=sizeof(TInt32);
    67 const TInt KParaTopBorder=sizeof(TUint8)+KParaBorderThickness+KColor+sizeof(TUint8);//linestyle/thickness/color/autocolor
    68 const TInt KParaBottomBorder=KParaTopBorder;
    69 const TInt KParaLeftBorder=KParaTopBorder;
    70 const TInt KParaRightBorder=KParaTopBorder;
    71 const TInt KParaBullet=KVariableLengthAttribute;
    72 const TInt KParaDefaultTabWidth=sizeof(TUint32);
    73 const TInt KParaTabStop=KTabPosition+KTabType;
    74 const TInt KParaFillSystemColor=sizeof(TUint8);
    75 const TInt KParaBulletSystemColor=sizeof(TUint8);
    76 const TInt KParaTopBorderSystemColor=sizeof(TUint8);
    77 const TInt KParaBottomBorderSystemColor=sizeof(TUint8);
    78 const TInt KParaLeftBorderSystemColor=sizeof(TUint8);
    79 const TInt KParaRightBorderSystemColor=sizeof(TUint8);
    80 const TInt KParaLanguageX=sizeof(TInt32);
    81 const TInt KParaBulletX=sizeof(TInt32) + 2;
    82 const TInt KBitmapType=sizeof(TUint8);
    83 
    84 // Character format attributes
    85 const TInt KCharLanguage=sizeof(TUint8);
    86 const TInt KCharColor=KColor;
    87 const TInt KCharHighlightColor=KColor;
    88 const TInt KCharHighlightStyle=sizeof(TUint8);
    89 const TInt KCharStrikethrough=sizeof(TUint8);
    90 const TInt KCharUnderline=sizeof(TUint8);
    91 const TInt KCharStrokeWeight=sizeof(TUint8);
    92 const TInt KCharPosture=sizeof(TUint8);
    93 const TInt KCharPrintPos=sizeof(TUint8);
    94 const TInt KCharFontHeight=KFontHeight;
    95 const TInt KCharTypeface=KVariableLengthAttribute;
    96 const TInt KCharHiddenText=sizeof(TUint8);
    97 const TInt KCharPictureAlignment=sizeof(TUint8);
    98 const TInt KCharTextSystemColor=sizeof(TUint8);
    99 const TInt KCharFontHighlightSystemColor=sizeof(TUint8);
   100 const TInt KCharLanguageX=sizeof(TInt32);
   101 const TInt KCharParserTag=sizeof(TUint32);
   102 
   103 /*
   104 Lookup table indexed by TTextFormatAttribute enumerated constants.
   105 Specifies the length in bytes of the format attributes.
   106 */
   107 static const TInt8 TheAttributeLength[EAttributeCount] =
   108 	{
   109 	// Paragraph attribute value lengths.	
   110 	KParaLanguage,
   111 	KParaFillColor,
   112 	KParaLeftMargin,
   113 	KParaRightMargin,
   114 	KParaIndent,
   115 	KParaAlignment,
   116 	KParaVerticalAlignment,
   117 	KParaLineSpacing,
   118 	KParaLineSpacingControl,
   119 	KParaSpaceBefore,
   120 	KParaSpaceAfter,
   121 	KParaKeepTogether,
   122 	KParaKeepWithNext,
   123 	KParaStartNewPage,
   124 	KParaWidowOrphan,
   125 	KParaWrap,
   126 	KParaBorderMargin,
   127 	KParaTopBorder,
   128 	KParaBottomBorder,
   129 	KParaLeftBorder,
   130 	KParaRightBorder,
   131 	KParaBullet,
   132 	KParaDefaultTabWidth,
   133 	KParaTabStop,
   134 
   135 	// Character attribute value lengths.
   136 	KCharLanguage,
   137 	KCharColor,
   138 	KCharHighlightColor,
   139 	KCharHighlightStyle,
   140 	KCharFontHeight,
   141 	KCharStrikethrough,
   142 	KCharUnderline,
   143 	KCharStrokeWeight,
   144 	KCharPosture,
   145 	KCharPrintPos,
   146 	KCharTypeface,
   147 	KCharHiddenText,
   148 	KCharPictureAlignment,
   149 
   150 	// Lengths of extended attributes.
   151 	KParaFillSystemColor,
   152 	KParaBulletSystemColor,
   153 	KParaTopBorderSystemColor,
   154 	KParaBottomBorderSystemColor,
   155 	KParaLeftBorderSystemColor,
   156 	KParaRightBorderSystemColor,
   157 	KCharTextSystemColor,
   158 	KCharFontHighlightSystemColor,
   159 	KParaLanguageX,
   160 	KCharLanguageX,
   161 	KParaBulletX,
   162     KBitmapType,
   163 
   164 	// Lengths of internal attributes
   165 	KCharParserTag
   166 	};
   167 
   168 
   169 DLLEXPORT_C void RFormatStream::__DbgTestInvariant()const
   170 // Provides class invariants.
   171 // This class invariant checks the integrity of a completed format stream.
   172 // As such, this invariant is only called by those methods that act upon a completed
   173 // format stream; that is, not the set methods, since the format stream will not be complete until
   174 // this call has completed.
   175 //
   176 	{
   177 #ifdef _DEBUG
   178 		__ASSERT_DEBUG(DoInvariantCheck(),User::Invariant());
   179 #endif
   180 	}
   181 
   182 void RFormatStream::TestInvariantL()const
   183 // Provides class invariants.
   184 // This class invariant checks the integrity of a completed format stream.
   185 // As such, this invariant is only called by those methods that act upon a completed
   186 // format stream; that is, not the set methods, since the format stream will not be complete until
   187 // this call has completed.
   188 //
   189 	{
   190 	if (!DoInvariantCheck())
   191 		User::Leave(KErrCorrupt);
   192 	}
   193 
   194 TBool RFormatStream::DoInvariantCheck() const
   195 // Provides class invariants.
   196 // This class invariant checks the integrity of a completed format stream.
   197 // As such, this invariant is only called by those methods that act upon a completed
   198 // format stream; that is, not the set methods, since the format stream will not be complete until
   199 // this call has completed.
   200 //
   201 	{
   202 	if (!iBase)
   203 		return ETrue;
   204 	if (iEnd>iBase)
   205 		{// Assert: stream is self consistent
   206 		// Walks through the stream (aBuffer), checking that it is in a consistent state.
   207 		// (1) All entries in the buffer conform to a TYPE-VALUE structure, checking that
   208 		// types do not occur in contiguous bytes.  ie there is a value.
   209 		// (2) Checks that all encountered types are valid types.
   210 		// (3) Checks that the buffer has not changed size as a result of this check
   211 		// (4) Checks that all attributes in the stream occur only once.  Done by taking a register
   212 		// as each attribute is read from the stream.  (The exception to this rule are Tab identifiers
   213 		// which, if present, will typically occur several times in the stream).
   214 		//
   215 		enum {ENotPresent,EPresent};
   216 		TInt8 attributeRegister[EAttributeCount];
   217 		for (TInt offset=0;offset<EAttributeCount;offset++)
   218 			{// Mark attributeRegister as all absent
   219 			attributeRegister[offset]=(TInt8)ENotPresent;  // clear all items in the register
   220 			}
   221 		TUint8* tempStreamLoc=iBase;
   222 		TUint8* streamLoc=iBase;
   223 		TUint8* endOfStreamLoc=iEnd;
   224 		TUint8 data=0;
   225 		while (streamLoc<endOfStreamLoc)
   226 			{
   227 			data=*streamLoc;
   228 			streamLoc++;
   229 // Assert: data read is a valid format attribute type.
   230 			if (!(data<((TUint8)EAttributeCount))) return EFalse;
   231 			if (data!=EAttTabStop)
   232 				{// Tab Identifiers can occur multiple times in a stream!!
   233 // Assert: attribute type does not already exist in attributeRegister.
   234 				if (!(attributeRegister[data]!=(TInt8)EPresent)) return EFalse;
   235 				}
   236 			attributeRegister[data]=(TInt8)EPresent;
   237 			tempStreamLoc=streamLoc+Length(streamLoc,(TTextFormatAttribute)data);
   238 			if (!(tempStreamLoc>=streamLoc)) return EFalse;
   239 			streamLoc=tempStreamLoc;
   240 			}
   241 // Assert: everything is still the same size.
   242 		if (!(streamLoc==endOfStreamLoc)) return EFalse;
   243 		}
   244 	return ETrue;
   245 	}
   246 
   247 static inline TUint32 Read32(const TUint8* aPtr)
   248 	{
   249 	TUint32 val = aPtr[0];
   250 	val |= aPtr[1] << 8;
   251 	val |= aPtr[2] << 16;
   252 	val |= aPtr[3] << 24;
   253 	return val;
   254 	}
   255 
   256 
   257 static inline void Write32(TUint8*& aPtr,TUint32 aVal)
   258 	{
   259 	*aPtr++ = TUint8(aVal);
   260 	*aPtr++ = TUint8(aVal >> 8);
   261 	*aPtr++ = TUint8(aVal >> 16);
   262 	*aPtr++ = TUint8(aVal >> 24);
   263 	}
   264 
   265 
   266 RFormatStream::RFormatStream():
   267 	iBase(NULL),
   268 	iEnd(NULL)
   269 	{
   270 	}
   271 
   272 		
   273 // Allocate the buffer
   274 void RFormatStream::AllocL(TInt aSize)
   275 	{
   276     TUint8* pT = reinterpret_cast<TUint8*>(User::ReAllocL(iBase, aSize));
   277     
   278     iBase=pT;
   279     iEnd=pT+aSize;
   280 	}
   281 
   282 
   283 void RFormatStream::Reset()
   284 // Free all storage
   285 //
   286 	{
   287 	if (iBase)
   288 		{
   289 		User::Free(iBase);
   290 		iBase = iEnd = NULL;
   291 		}
   292 	}
   293 
   294 
   295 void RFormatStream::CopyL(const RFormatStream& aSource)
   296 	{
   297 	TInt size = aSource.iEnd - aSource.iBase;
   298 	if (size < 0)
   299 	    {
   300 	    OstTrace0( TRACE_DUMP, RFORMATSTREAM_COPYL, "EDebug" );
   301 	    }
   302 	__ASSERT_DEBUG(size >= 0,Panic(EDebug));
   303 	if (size == 0)
   304 		Reset();
   305 	else
   306 		{
   307 		AllocL(size);
   308 		Mem::Copy(iBase,aSource.iBase,size);
   309 		}
   310 	}
   311 
   312 
   313 /*
   314 Write the bytecode to a stream. Don't write internal attributes; these are not part of the stored format,
   315 but are used for transitory marking of text by URL parsers, etc.
   316 */
   317 void RFormatStream::ExternalizeL(RWriteStream& aStream) const
   318 	{
   319 	__TEST_INVARIANT;
   320 	
   321 	const TUint8* base=iBase;
   322 	if (base)
   323 		{
   324 		const TUint8* end = base;
   325 		while (end < iEnd && *end < EExternalizedAttributeCount)
   326 			{
   327 			int bytes = TheAttributeLength[*end++];
   328 			if (bytes == 0)
   329 				bytes = *end++;
   330 			end += bytes;
   331 			}
   332 		int length = end - base;
   333 		aStream.WriteInt32L(length);
   334 		aStream.WriteL(base,length);
   335 		}
   336 	else
   337 		aStream.WriteInt32L(0);
   338 	}
   339 
   340 
   341 void RFormatStream::InternalizeL(RReadStream& aStream)
   342 // Load the buffer from the specified stream
   343 //
   344 	{
   345 	TestInvariantL();
   346 
   347 	TInt length=aStream.ReadInt32L();
   348 	//
   349 	if (length<0 || length>KMaxFormatStreamLength)
   350 		User::Leave(KErrCorrupt);
   351 	//
   352 	if (length==0)
   353 		Reset();
   354 	else
   355 		{
   356 		AllocL(length);
   357 		aStream.ReadL(iBase,length);
   358 		}
   359 
   360 	TestInvariantL();
   361 	}
   362 
   363 
   364 // Return a pointer to the stored bytecode and put its length in bytes in aLength.
   365 const TUint8* RFormatStream::Ptr(TInt& aLength) const
   366 	{
   367 	__TEST_INVARIANT;
   368 	aLength=iEnd-iBase;
   369 	if (!((iBase==NULL && aLength==0) || (iBase!=NULL && aLength>0)))
   370 	    {
   371 	    OstTrace0( TRACE_DUMP, RFORMATSTREAM_PTR, "ECorruptFormatLayer" );
   372 	    }
   373 	__ASSERT_DEBUG((iBase==NULL && aLength==0) || (iBase!=NULL && aLength>0),Panic(ECorruptFormatLayer));
   374 	return iBase;
   375 	}
   376 
   377 
   378 // Save the attributes of aCharFormat specified by the corresponding mask aMask.
   379 void RFormatStream::SetCharFormatL(const TCharFormatX& aCharFormatX,const TCharFormatXMask& aMask)
   380 	{
   381 	TInt size=DoCalcSizeCharFormat(aCharFormatX,aMask);
   382 	if (size==0)
   383 		Reset();
   384 	else
   385 		{
   386 		AllocL(size);  // delete the current contents, after allocing a temporary
   387 		DoStoreCharFormat(aCharFormatX,aMask);
   388 		}
   389 
   390 	__TEST_INVARIANT;
   391 	}
   392 
   393 
   394 void RFormatStream::SetParaFormatL(const CParaFormat& aDesiredFormat,const TParaFormatMask& aDesiredMask,
   395 											const CParaFormat& aCurrentFormat)
   396 // Sets the format layer with the specified paragraph format attributes.
   397 // If a leave occurs at any stage, then revert back to original state, and
   398 // propagate the leave.
   399 //
   400 	{
   401 	TInt size=DoCalcSizeParaFormat(aDesiredFormat,aDesiredMask,aCurrentFormat);
   402 	if (size==0)
   403 		Reset();
   404 	else
   405 		{
   406 		AllocL(size);  // delete the current contents, after allocing a temporary
   407 		DoSetParaFormat(aDesiredFormat,aDesiredMask,aCurrentFormat);
   408 		}
   409 
   410 	__TEST_INVARIANT;
   411 	}
   412 	
   413 // Write an auxiliary attribute for a system colour byte. Update aPtr.
   414 static void WriteSystemColor(TUint8*& aPtr,TTextFormatNonMaskableAttribute aAttrib,const TLogicalRgb& aColor)
   415 	{
   416 	TUint8 index = (TUint8)aColor.SystemColorIndex();
   417 	if (index)
   418 		{
   419 		*aPtr++ = TUint8(aAttrib);
   420 		*aPtr++ = index;
   421 		}
   422 	}
   423 
   424 
   425 // Read a system colour byte into a logical colour. Don't change aPtr.
   426 static void ReadSystemColor(const TUint8* aPtr,TLogicalRgb& aColor)
   427 	{
   428 	TUint index = *aPtr;
   429 	aColor.SetSystemColorIndex(index); 
   430 	}
   431 
   432 
   433 // Write paragraph attributes specified by aDesiredMask from aDesiredFormat to the stream.
   434 void RFormatStream::DoSetParaFormat(const CParaFormat& aDesiredFormat,
   435 									TParaFormatMask aMask,
   436 									const CParaFormat& aCurrentFormat)
   437 	{
   438 	TUint8* ptr=iBase;
   439 	if (aMask.AttribIsSet(EAttFillColor))
   440 		{
   441 		*ptr++=TUint8(EAttFillColor);
   442 		ptr=Store(ptr,aDesiredFormat.iFillColor);
   443 		}
   444 	if (aMask.AttribIsSet(EAttLeftMargin))
   445 		{
   446 		*ptr++=TUint8(EAttLeftMargin);
   447 		Write32(ptr,aDesiredFormat.iLeftMarginInTwips);
   448 		}
   449 	if (aMask.AttribIsSet(EAttRightMargin))
   450 		{
   451 		*ptr++=TUint8(EAttRightMargin);
   452 		Write32(ptr,aDesiredFormat.iRightMarginInTwips);
   453 		}
   454 	if (aMask.AttribIsSet(EAttIndent))
   455 		{
   456 		*ptr++=TUint8(EAttIndent);
   457 		Write32(ptr,aDesiredFormat.iIndentInTwips);
   458 		}
   459 	if (aMask.AttribIsSet(EAttAlignment))
   460 		{
   461 		*ptr++=TUint8(EAttAlignment);
   462 		*ptr++=TUint8(aDesiredFormat.iHorizontalAlignment);
   463 		}
   464 	if (aMask.AttribIsSet(EAttVerticalAlignment))
   465 		{
   466 		*ptr++=TUint8(EAttVerticalAlignment);
   467 		*ptr++=TUint8(aDesiredFormat.iVerticalAlignment);
   468 		}
   469 	if (aMask.AttribIsSet(EAttLineSpacing))
   470 		{
   471 		*ptr++=TUint8(EAttLineSpacing);
   472 		Write32(ptr,aDesiredFormat.iLineSpacingInTwips);
   473 		}
   474 	if (aMask.AttribIsSet(EAttLineSpacingControl))
   475 		{
   476 		*ptr++=TUint8(EAttLineSpacingControl);
   477 		*ptr++=TUint8(aDesiredFormat.iLineSpacingControl);
   478 		}
   479 	if (aMask.AttribIsSet(EAttSpaceBefore))
   480 		{
   481 		*ptr++=TUint8(EAttSpaceBefore);
   482 		Write32(ptr,aDesiredFormat.iSpaceBeforeInTwips);
   483 		}
   484 	if (aMask.AttribIsSet(EAttSpaceAfter))
   485 		{
   486 		*ptr++=TUint8(EAttSpaceAfter);
   487 		Write32(ptr,aDesiredFormat.iSpaceAfterInTwips);
   488 		}
   489 	if (aMask.AttribIsSet(EAttKeepTogether))
   490 		{
   491 		*ptr++=TUint8(EAttKeepTogether);
   492 		*ptr++=TUint8(aDesiredFormat.iKeepTogether!=EFalse);
   493 		}
   494 	if (aMask.AttribIsSet(EAttKeepWithNext))
   495 		{
   496 		*ptr++=TUint8(EAttKeepWithNext);
   497 		*ptr++=TUint8(aDesiredFormat.iKeepWithNext!=EFalse);
   498 		}
   499 	if (aMask.AttribIsSet(EAttStartNewPage))
   500 		{
   501 		*ptr++=TUint8(EAttStartNewPage);
   502 		*ptr++=TUint8(aDesiredFormat.iStartNewPage!=EFalse);
   503 		}
   504 	if (aMask.AttribIsSet(EAttWidowOrphan))
   505 		{
   506 		*ptr++=TUint8(EAttWidowOrphan);
   507 		*ptr++=TUint8(aDesiredFormat.iWidowOrphan!=EFalse);
   508 		}
   509 	if (aMask.AttribIsSet(EAttWrap))
   510 		{
   511 		*ptr++=TUint8(EAttWrap);
   512 		*ptr++=TUint8(aDesiredFormat.iWrap!=EFalse);
   513 		}
   514 	if (aMask.AttribIsSet(EAttBorderMargin))
   515 		{
   516 		*ptr++=TUint8(EAttBorderMargin);
   517 		Write32(ptr,aDesiredFormat.iBorderMarginInTwips);
   518 		}
   519 	if ( aDesiredFormat.BordersPresent() || aCurrentFormat.BordersPresent() )
   520 		{
   521 		if (aMask.AttribIsSet(EAttTopBorder))
   522 			ptr=StoreBorder(ptr,EAttTopBorder,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderTop));
   523 		if (aMask.AttribIsSet(EAttBottomBorder))
   524 			ptr=StoreBorder(ptr,EAttBottomBorder,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderBottom));
   525 		if (aMask.AttribIsSet(EAttLeftBorder))
   526 			ptr=StoreBorder(ptr,EAttLeftBorder,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderLeft));
   527 		if (aMask.AttribIsSet(EAttRightBorder))
   528 			ptr=StoreBorder(ptr,EAttRightBorder,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderRight));
   529 		}
   530 	if (aMask.AttribIsSet(EAttDefaultTabWidth))
   531 		{
   532 		*ptr++=TUint8(EAttDefaultTabWidth);
   533 		Write32(ptr,aDesiredFormat.iDefaultTabWidthInTwips);
   534 		}
   535 	if (aMask.AttribIsSet(EAttTabStop))
   536 		{
   537 		TUint8* tptr=ptr;	// prevent stacking of ptr;
   538 		StoreTabs(tptr,aDesiredFormat,aCurrentFormat,ETrue);
   539 		ptr=tptr;
   540 		}
   541 	if (aMask.AttribIsSet(EAttBullet))
   542 		{
   543 		if (aDesiredFormat.iBullet || aCurrentFormat.iBullet)
   544 			{
   545 			if (aDesiredFormat.iBullet)
   546 				ptr = StoreBullet(ptr,*aDesiredFormat.iBullet);
   547 			else if (aCurrentFormat.iBullet)
   548 				ptr = StoreBullet(ptr,TBullet());
   549 			}
   550 		}
   551 	if (!(aDesiredFormat.iLanguage & ~0xFF) && aMask.AttribIsSet(EAttParaLanguage))
   552 		{
   553 		*ptr++=TUint8(EAttParaLanguage);
   554 		*ptr++=TUint8(aDesiredFormat.iLanguage);
   555 		}
   556 
   557 	/*
   558 	Write the auxiliary attributes for system colours, language codes greater than 255, etc.
   559 	These must go at the end of the stream because they will not be recognised by earlier versions
   560 	of ETEXT and will prevent any further attributes from being read.
   561 	*/
   562 	if (aMask.AttribIsSet(EAttFillColor))
   563 		WriteSystemColor(ptr,EAttFillSystemColor,aDesiredFormat.iFillColor);
   564 	if (aMask.AttribIsSet(EAttBullet))
   565 		{
   566 		if (aDesiredFormat.iBullet)
   567 			WriteSystemColor(ptr,EAttBulletSystemColor,aDesiredFormat.iBullet->iColor);
   568 		else if (aCurrentFormat.iBullet)
   569 			WriteSystemColor(ptr,EAttBulletSystemColor,TBullet().iColor);
   570 		}
   571 	if (aDesiredFormat.BordersPresent())
   572 		{
   573 		if (aMask.AttribIsSet(EAttTopBorder))
   574 			WriteSystemColor(ptr,EAttTopBorderSystemColor,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderTop).iColor);
   575 		if (aMask.AttribIsSet(EAttBottomBorder))
   576 			WriteSystemColor(ptr,EAttBottomBorderSystemColor,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderBottom).iColor);
   577 		if (aMask.AttribIsSet(EAttLeftBorder))
   578 			WriteSystemColor(ptr,EAttLeftBorderSystemColor,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderLeft).iColor);
   579 		if (aMask.AttribIsSet(EAttRightBorder))
   580 			WriteSystemColor(ptr,EAttRightBorderSystemColor,aDesiredFormat.ParaBorder(CParaFormat::EParaBorderRight).iColor);
   581 		}
   582 	if ((aDesiredFormat.iLanguage & ~0xFF) && aMask.AttribIsSet(EAttParaLanguage))
   583 		{
   584 		*ptr++ = TUint8(EAttParaLanguageX);
   585 		Write32(ptr,aDesiredFormat.iLanguage);
   586 		}
   587 	if (aMask.AttribIsSet(EAttBullet))
   588 		{
   589 		if (aDesiredFormat.iBullet || aCurrentFormat.iBullet)
   590 			{
   591 			TBullet bullet;
   592 			const TBullet* b = aDesiredFormat.iBullet;
   593 			if (!b)
   594 				b = &bullet;
   595 			*ptr++ = TUint8(EAttBulletX);
   596 			*ptr++ = TUint8(b->iStyle);
   597 			Write32(ptr,b->iStartNumber);
   598 			*ptr++ = TUint8(b->iAlignment);
   599 			}
   600 		}
   601 	}
   602 
   603 
   604 TInt RFormatStream::DoCalcSizeParaFormat(const CParaFormat& aDesiredFormat,TParaFormatMask aMask,
   605 										 const CParaFormat& aCurrentFormat)
   606 // determine the amount of memory required to store the
   607 // specified attriubutes from the specified paragraph format.
   608 //
   609 	{
   610 	TInt size=0;
   611 	if (aMask.AttribIsSet(EAttFillColor))
   612 		{
   613 		size+=(KTypeSize+KParaFillColor);
   614 		if (aDesiredFormat.iFillColor.SystemColorIndex())
   615 			size += KTypeSize + KParaFillSystemColor;
   616 		}
   617 	if (aMask.AttribIsSet(EAttLeftMargin))
   618 		size+=(KTypeSize+KParaLeftMargin);
   619 	if (aMask.AttribIsSet(EAttRightMargin))
   620 		size+=(KTypeSize+KParaRightMargin);
   621 	if (aMask.AttribIsSet(EAttIndent))
   622 		size+=(KTypeSize+KParaIndent);
   623 	if (aMask.AttribIsSet(EAttAlignment))
   624 		size+=(KTypeSize+KParaAlignment);
   625 	if (aMask.AttribIsSet(EAttVerticalAlignment))
   626 		size+=(KTypeSize+KParaVerticalAlignment);
   627 	if (aMask.AttribIsSet(EAttLineSpacing))
   628 		size+=(KTypeSize+KParaLineSpacing);
   629 	if (aMask.AttribIsSet(EAttLineSpacingControl))
   630 		size+=(KTypeSize+KParaLineSpacingControl);
   631 	if (aMask.AttribIsSet(EAttSpaceBefore))
   632 		size+=(KTypeSize+KParaSpaceBefore);
   633 	if (aMask.AttribIsSet(EAttSpaceAfter))
   634 		size+=(KTypeSize+KParaSpaceAfter);
   635 	if (aMask.AttribIsSet(EAttKeepTogether))
   636 		size+=(KTypeSize+KParaKeepTogether);
   637 	if (aMask.AttribIsSet(EAttKeepWithNext))
   638 		size+=(KTypeSize+KParaKeepWithNext);
   639 	if (aMask.AttribIsSet(EAttStartNewPage))
   640 		size+=(KTypeSize+KParaStartNewPage);
   641 	if (aMask.AttribIsSet(EAttWidowOrphan))
   642 		size+=(KTypeSize+KParaWidowOrphan);
   643 	if (aMask.AttribIsSet(EAttWrap))
   644 		size+=(KTypeSize+KParaWrap);
   645 	if (aMask.AttribIsSet(EAttBorderMargin))
   646 		size+=(KTypeSize+KParaBorderMargin);
   647 	if ( aDesiredFormat.BordersPresent() || aCurrentFormat.BordersPresent() )
   648 		{
   649 		if (aMask.AttribIsSet(EAttTopBorder))
   650 			{
   651 			size+=(KTypeSize+
   652 			sizeof(TUint8)+  // line style
   653 			sizeof(TInt32)+  // line thickness
   654 			KColor+
   655 			sizeof(TUint8));  // auto color flag
   656 			if (aDesiredFormat.ParaBorder(CParaFormat::EParaBorderTop).iColor.SystemColorIndex())
   657 				size += KTypeSize + KParaTopBorderSystemColor;
   658 			}
   659 		if (aMask.AttribIsSet(EAttBottomBorder))
   660 			{
   661 			size+=(KTypeSize+
   662 			sizeof(TUint8)+  // line style
   663 			sizeof(TInt32)+  // line thickness
   664 			KColor+
   665 			sizeof(TUint8));  // auto color flag
   666 			if (aDesiredFormat.ParaBorder(CParaFormat::EParaBorderBottom).iColor.SystemColorIndex())
   667 				size += KTypeSize + KParaBottomBorderSystemColor;
   668 			}
   669 		if (aMask.AttribIsSet(EAttLeftBorder))
   670 			{
   671 			size+=(KTypeSize+
   672 			sizeof(TUint8)+  // line style
   673 			sizeof(TInt32)+  // line thickness
   674 			KColor+
   675 			sizeof(TUint8));  // auto color flag
   676 			if (aDesiredFormat.ParaBorder(CParaFormat::EParaBorderLeft).iColor.SystemColorIndex())
   677 				size += KTypeSize + KParaLeftBorderSystemColor;
   678 			}
   679 		if (aMask.AttribIsSet(EAttRightBorder))
   680 			{
   681 			size+=(KTypeSize+
   682 			sizeof(TUint8)+  // line style
   683 			sizeof(TInt32)+  // line thickness
   684 			KColor+
   685 			sizeof(TUint8));  // auto color flag
   686 			if (aDesiredFormat.ParaBorder(CParaFormat::EParaBorderRight).iColor.SystemColorIndex())
   687 				size += KTypeSize + KParaBulletSystemColor;
   688 			}
   689 		}
   690 	if (aMask.AttribIsSet(EAttDefaultTabWidth))
   691 		size+=(KTypeSize+KParaDefaultTabWidth);
   692 	if (aMask.AttribIsSet(EAttTabStop))
   693 		{
   694 		TUint8* ptr=NULL;
   695 		size+=StoreTabs(ptr,aDesiredFormat,aCurrentFormat,EFalse);
   696 		}
   697 	if (aMask.AttribIsSet(EAttBullet))
   698 		{
   699 		if (aDesiredFormat.iBullet || aCurrentFormat.iBullet)
   700 			{
   701 			size += KTypeSize +
   702 					sizeof(TUint8) +		// length of following data
   703 					sizeof(TText) +			// iCharacterCode
   704 					KFontHeight +			// iHeightInTwips
   705 					sizeof(TUint8) +		// iHanging indent
   706 					KCharColor +			// iColor
   707 					sizeof(TUint8) +		// typeface name size
   708 					KTypefaceFlags;			// typeface flags
   709 
   710 			if (aDesiredFormat.iBullet)
   711 				size += aDesiredFormat.iBullet->iTypeface.iName.Size();  // font name
   712 
   713 			if ((aDesiredFormat.iBullet && aDesiredFormat.iBullet->iColor.SystemColorIndex()) ||
   714 				(aCurrentFormat.iBullet))
   715 				size += KTypeSize + KParaBulletSystemColor;
   716 
   717 			size += KTypeSize + KParaBulletX;
   718 			}
   719 		}
   720 	if (aMask.AttribIsSet(EAttParaLanguage))
   721 		{
   722 		if (aDesiredFormat.iLanguage & ~0xFF)
   723 			size += KTypeSize + KParaLanguageX;
   724 		else
   725 			size += KTypeSize + KParaLanguage;
   726 		}
   727 	return size;
   728 	}
   729 
   730 
   731 // Store the specified values in this (allocated) format stream.
   732 void RFormatStream::DoStoreCharFormat(const TCharFormatX& aCharFormat,TCharFormatXMask aMask)
   733 	{
   734 	TUint8* ptr = iBase;
   735 	const TCharFormat format = aCharFormat.iCharFormat;
   736 
   737 	if (aMask.AttribIsSet(EAttColor))
   738 		{
   739 		*ptr++=TUint8(EAttColor);
   740 		ptr=Store(ptr,format.iFontPresentation.iTextColor);
   741 		}
   742 	if (aMask.AttribIsSet(EAttFontHighlightColor))
   743 		{
   744 		*ptr++=TUint8(EAttFontHighlightColor);
   745 		ptr=Store(ptr,format.iFontPresentation.iHighlightColor);
   746 		}
   747 	if (aMask.AttribIsSet(EAttFontHighlightStyle))
   748 		{
   749 		*ptr++=TUint8(EAttFontHighlightStyle);
   750 		*ptr++=(TUint8)format.iFontPresentation.iHighlightStyle;
   751 		}
   752 	if (aMask.AttribIsSet(EAttFontStrikethrough))
   753 		{
   754 		*ptr++=TUint8(EAttFontStrikethrough);
   755 		*ptr++=(TUint8)format.iFontPresentation.iStrikethrough;
   756 		}
   757 	if (aMask.AttribIsSet(EAttFontUnderline))
   758 		{
   759 		*ptr++=TUint8(EAttFontUnderline);
   760 		*ptr++=(TUint8)format.iFontPresentation.iUnderline;
   761 		}
   762 	if (aMask.AttribIsSet(EAttFontHeight))
   763 		{
   764 		*ptr++=TUint8(EAttFontHeight);
   765 		Write32(ptr,format.iFontSpec.iHeight);
   766 		}
   767 	if (aMask.AttribIsSet(EAttFontPosture))
   768 		{
   769 		*ptr++=TUint8(EAttFontPosture);
   770 		*ptr++=TUint8(format.iFontSpec.iFontStyle.Posture());
   771 		}
   772 	if (aMask.AttribIsSet(EAttFontStrokeWeight))
   773 		{
   774 		*ptr++=TUint8(EAttFontStrokeWeight);
   775 		*ptr++=TUint8(format.iFontSpec.iFontStyle.StrokeWeight());
   776 		}
   777 	if (aMask.AttribIsSet(EAttFontPrintPos))
   778 		{
   779 		*ptr++=TUint8(EAttFontPrintPos);
   780 		*ptr++=TUint8(format.iFontSpec.iFontStyle.PrintPosition());
   781 		}
   782 	if (aMask.AttribIsSet(EAttFontTypeface))
   783 		{
   784 		*ptr++=TUint8(EAttFontTypeface);
   785 		ptr=Store(ptr,format.iFontSpec.iTypeface);
   786 		*ptr++=TUint8(EAttBitmapType);
   787 		*ptr++=TUint8(format.iFontSpec.iFontStyle.BitmapType());
   788 		}
   789 	if (!(format.iLanguage & ~0xFF) && aMask.AttribIsSet(EAttCharLanguage))
   790 		{
   791 		*ptr++=TUint8(EAttCharLanguage);
   792 		*ptr++=TUint8(format.iLanguage);
   793 		}
   794 	if (aMask.AttribIsSet(EAttFontHiddenText))
   795 		{
   796 		*ptr++=TUint8(EAttFontHiddenText);
   797 		*ptr++=TUint8(format.iFontPresentation.iHiddenText!=EFalse);
   798 		}
   799 	if (aMask.AttribIsSet(EAttFontPictureAlignment))
   800 		{
   801 		*ptr++=TUint8(EAttFontPictureAlignment);
   802 		*ptr++=TUint8(format.iFontPresentation.iPictureAlignment);
   803 		}
   804 
   805 	// Auxiliary attributes for system colours, etc.
   806 	if (aMask.AttribIsSet(EAttColor))
   807 		WriteSystemColor(ptr,EAttTextSystemColor,format.iFontPresentation.iTextColor);
   808 	if (aMask.AttribIsSet(EAttFontHighlightColor))
   809 		WriteSystemColor(ptr,EAttFontHighlightSystemColor,format.iFontPresentation.iHighlightColor);
   810 	if ((format.iLanguage & ~0xFF) && aMask.AttribIsSet(EAttCharLanguage))
   811 		{
   812 		*ptr++ = TUint8(EAttCharLanguageX);
   813 		Write32(ptr,format.iLanguage);
   814 		}
   815 
   816 	// Internal character attributes:
   817 	if (aMask.AttribIsSet(EAttParserTag))
   818 		{
   819 		*ptr++ = TUint8(EAttParserTag);
   820 		Write32(ptr,aCharFormat.iParserTag);
   821 		}
   822 	}
   823 
   824 
   825 TInt RFormatStream::DoCalcSizeCharFormat(const TCharFormatX& aCharFormatX,const TCharFormatXMask& aMask)
   826 	{
   827 	TInt size = 0;
   828 	const TCharFormat format = aCharFormatX.iCharFormat;
   829 
   830 	if (aMask.AttribIsSet(EAttColor))
   831 		{
   832 		size+=(KTypeSize+KCharColor);
   833 		if (format.iFontPresentation.iTextColor.SystemColorIndex())
   834 			size += KTypeSize + KCharTextSystemColor;
   835 		}
   836 	if (aMask.AttribIsSet(EAttFontHighlightColor))
   837 		{
   838 		size+=(KTypeSize+KCharHighlightColor);
   839 		if (format.iFontPresentation.iHighlightColor.SystemColorIndex())
   840 			size += KTypeSize + KCharFontHighlightSystemColor;
   841 		}
   842 	if (aMask.AttribIsSet(EAttFontHighlightStyle))
   843 		size+=(KTypeSize+KCharHighlightStyle);
   844 	if (aMask.AttribIsSet(EAttFontStrikethrough))
   845 		size+=(KTypeSize+KCharStrikethrough);
   846 	if (aMask.AttribIsSet(EAttFontUnderline))
   847 		size+=(KTypeSize+KCharUnderline);
   848 	if (aMask.AttribIsSet(EAttFontHeight))
   849 		size+=(KTypeSize+KCharFontHeight);
   850 	if (aMask.AttribIsSet(EAttFontPosture))
   851 		size+=(KTypeSize+KCharPosture);
   852 	if (aMask.AttribIsSet(EAttFontStrokeWeight))
   853 		size+=(KTypeSize+KCharStrokeWeight);
   854 	if (aMask.AttribIsSet(EAttFontPrintPos))
   855 		size+=(KTypeSize+KCharPrintPos);
   856 	if (aMask.AttribIsSet(EAttFontTypeface))
   857 		{
   858 		TUint8 length=(TUint8)format.iFontSpec.iTypeface.iName.Size();
   859 		size+=(KTypeSize+sizeof(TUint8)+length+sizeof(TUint8));  // size in bytes
   860 		size+=(KTypeSize+KBitmapType);
   861 		}
   862 	if (aMask.AttribIsSet(EAttCharLanguage))
   863 		{
   864 		if (format.iLanguage & ~0xFF)
   865 			size += KTypeSize + KCharLanguageX;
   866 		else
   867 			size += KTypeSize + KCharLanguage;
   868 		}
   869 	if (aMask.AttribIsSet(EAttFontHiddenText))
   870 		size+=(KTypeSize+KCharHiddenText);
   871 	if (aMask.AttribIsSet(EAttFontPictureAlignment))
   872 		size+=(KTypeSize+KCharPictureAlignment);
   873 
   874 	// Internal character attributes
   875 	if (aMask.AttribIsSet(EAttParserTag))
   876 		size += KTypeSize + KCharParserTag;
   877 	
   878 	return size;
   879 	}
   880 
   881 void RFormatStream::RemoveRedundantCharFormat(TCharFormatMask& aMask,
   882 											  const TCharFormatX& aFormat,const TCharFormatX& aEffectiveFormat)
   883 	{
   884 	TCharFormatXMask mask = aMask;
   885 	const TCharFormat format = aFormat.iCharFormat;
   886 	const TCharFormat effective_format = aEffectiveFormat.iCharFormat;
   887 
   888 	if (mask.AttribIsSet(EAttColor))
   889 		{
   890 		if (format.iFontPresentation.iTextColor==effective_format.iFontPresentation.iTextColor)
   891 			mask.ClearAttrib(EAttColor);
   892 		}
   893 	if (mask.AttribIsSet(EAttFontHighlightColor))
   894 		{
   895 		if (format.iFontPresentation.iHighlightColor==effective_format.iFontPresentation.iHighlightColor)
   896 			mask.ClearAttrib(EAttFontHighlightColor);
   897 		}
   898 	if (mask.AttribIsSet(EAttFontHighlightStyle))
   899 		{
   900 		if (format.iFontPresentation.iHighlightStyle==effective_format.iFontPresentation.iHighlightStyle)
   901 			mask.ClearAttrib(EAttFontHighlightStyle);
   902 		}
   903 	if (mask.AttribIsSet(EAttFontStrikethrough))
   904 		{
   905 		if (format.iFontPresentation.iStrikethrough==effective_format.iFontPresentation.iStrikethrough)
   906 			mask.ClearAttrib(EAttFontStrikethrough);
   907 		}
   908 	if (mask.AttribIsSet(EAttFontUnderline))
   909 		{
   910 		if (format.iFontPresentation.iUnderline==effective_format.iFontPresentation.iUnderline)
   911 			mask.ClearAttrib(EAttFontUnderline);
   912 		}
   913 	if (mask.AttribIsSet(EAttFontHeight))
   914 		{
   915 		if (format.iFontSpec.iHeight==effective_format.iFontSpec.iHeight)
   916 			mask.ClearAttrib(EAttFontHeight);
   917 		}
   918 	if (mask.AttribIsSet(EAttFontPosture))
   919 		{
   920 		if (format.iFontSpec.iFontStyle.Posture()==effective_format.iFontSpec.iFontStyle.Posture())
   921 			mask.ClearAttrib(EAttFontPosture);
   922 		}
   923 	if (mask.AttribIsSet(EAttFontStrokeWeight))
   924 		{
   925 		if (format.iFontSpec.iFontStyle.StrokeWeight()==effective_format.iFontSpec.iFontStyle.StrokeWeight())
   926 			mask.ClearAttrib(EAttFontStrokeWeight);
   927 		}
   928 	if (mask.AttribIsSet(EAttFontPrintPos))
   929 		{
   930 		if (format.iFontSpec.iFontStyle.PrintPosition()==effective_format.iFontSpec.iFontStyle.PrintPosition())
   931 			mask.ClearAttrib(EAttFontPrintPos);
   932 		}
   933 	if (mask.AttribIsSet(EAttFontTypeface))
   934 		{
   935 		if (format.iFontSpec.iTypeface==effective_format.iFontSpec.iTypeface)
   936 			mask.ClearAttrib(EAttFontTypeface);
   937 		}
   938 	if (mask.AttribIsSet(EAttCharLanguage))
   939 		{
   940 		if (format.iLanguage==effective_format.iLanguage)
   941 			mask.ClearAttrib(EAttCharLanguage);
   942 		}
   943 	if (mask.AttribIsSet(EAttFontHiddenText))
   944 		{
   945 		if (format.iFontPresentation.iHiddenText==effective_format.iFontPresentation.iHiddenText)
   946 			mask.ClearAttrib(EAttFontHiddenText);
   947 		}
   948 	if (mask.AttribIsSet(EAttFontPictureAlignment))
   949 		{
   950 		if (format.iFontPresentation.iPictureAlignment==effective_format.iFontPresentation.iPictureAlignment)
   951 			mask.ClearAttrib(EAttFontPictureAlignment);
   952 		}
   953 	if (mask.AttribIsSet(EAttParserTag))
   954 		{
   955 		if (aFormat.iParserTag == aEffectiveFormat.iParserTag)
   956 			mask.ClearAttrib(EAttParserTag);
   957 		}
   958 	aMask = mask;
   959 	}
   960 
   961 
   962 void RFormatStream::SenseParaFormatL(CParaFormat& aParaFormat,TParaFormatMask& aMask,CParaFormat::TParaFormatGetMode aMode)const
   963 // Dumps contents of the stream into aParaFormat.  The attributes written to aParaFormat
   964 // are determined by the mode, aMode.
   965 // If the mode is EAllAttributes, then all attributes are written,
   966 // if the mode is EFixedAttributes, then only the fixed atttibutes are written,
   967 // ie, not tabs,borders,bullet (which are variable).
   968 // For each format attribute found in the stream:
   969 // write it into a ParaFormat if the corresponding flag in aMask is not set,
   970 // and set this flag showing the attribute has been written into aParaFormat.
   971 //
   972 // If aMode is EAllAttributes this function may leave through out-of-memory allocating paragraph borders,
   973 // bullets or tabs.
   974 // If aMode is EFixedAttributes, this function is guaranteed not to leave.
   975 //
   976 	{
   977 	__TEST_INVARIANT;
   978 
   979 	TUint8* ptr=iBase;
   980 	if (ptr==NULL)
   981 		return;
   982 	TParaFormatMask old_mask = aMask;
   983 	TParaFormatMask new_mask = aMask;
   984 	TUint8* end=iEnd;
   985 	TParaBorder* b = NULL;
   986 
   987 	while (ptr < end)
   988 		{
   989 		TUint8 type = *ptr;
   990 		ptr += KTypeSize;
   991 		switch (type)
   992 			{
   993 			case EAttFillColor:
   994 				if (!old_mask.AttribIsSet(EAttFillColor))
   995 					{
   996 					new_mask.SetAttrib(EAttFillColor);
   997 					ptr=ReadValue(ptr,aParaFormat.iFillColor);
   998 					}
   999 				else
  1000 					ptr+=KParaFillColor;
  1001 				break;
  1002 			case EAttLeftMargin:
  1003 				if (!old_mask.AttribIsSet(EAttLeftMargin))
  1004 					{
  1005 					new_mask.SetAttrib(EAttLeftMargin);
  1006 					aParaFormat.iLeftMarginInTwips = Read32(ptr);
  1007 					}
  1008 				ptr+=KParaLeftMargin;
  1009 				break;
  1010 			case EAttRightMargin:
  1011 				if (!old_mask.AttribIsSet(EAttRightMargin))
  1012 					{
  1013 					new_mask.SetAttrib(EAttRightMargin);
  1014 					aParaFormat.iRightMarginInTwips = Read32(ptr);
  1015 					}
  1016 				ptr+=KParaRightMargin;
  1017 				break;
  1018 			case EAttIndent:
  1019 				if (!old_mask.AttribIsSet(EAttIndent))
  1020 					{
  1021 					new_mask.SetAttrib(EAttIndent);
  1022 					aParaFormat.iIndentInTwips = Read32(ptr);
  1023 					}
  1024 				ptr+=KParaIndent;
  1025 				break;
  1026 			case EAttAlignment:
  1027 				if (!old_mask.AttribIsSet(EAttAlignment))
  1028 					{
  1029 					new_mask.SetAttrib(EAttAlignment);
  1030 					aParaFormat.iHorizontalAlignment=CParaFormat::TAlignment(*ptr);
  1031 					}
  1032 				ptr+=KParaAlignment;
  1033 				break;
  1034 			case EAttVerticalAlignment:
  1035 				if (!old_mask.AttribIsSet(EAttVerticalAlignment))
  1036 					{
  1037 					new_mask.SetAttrib(EAttVerticalAlignment);
  1038 					aParaFormat.iVerticalAlignment=CParaFormat::TAlignment(*ptr);
  1039 					}
  1040 				ptr+=KParaVerticalAlignment;
  1041 				break;
  1042 			case EAttLineSpacing:
  1043 				if (!old_mask.AttribIsSet(EAttLineSpacing))
  1044 					{
  1045 					new_mask.SetAttrib(EAttLineSpacing);
  1046 					aParaFormat.iLineSpacingInTwips = Read32(ptr);
  1047 					}
  1048 				ptr+=KParaLineSpacing;
  1049 				break;
  1050 			case EAttLineSpacingControl:
  1051 				if (!old_mask.AttribIsSet(EAttLineSpacingControl))
  1052 					{
  1053 					new_mask.SetAttrib(EAttLineSpacingControl);
  1054 					aParaFormat.iLineSpacingControl=(CParaFormat::TLineSpacingControl)*ptr;
  1055 					}
  1056 				ptr+=KParaLineSpacingControl;
  1057 				break;
  1058 			case EAttSpaceBefore:
  1059 				if (!old_mask.AttribIsSet(EAttSpaceBefore))
  1060 					{
  1061 					new_mask.SetAttrib(EAttSpaceBefore);
  1062 					aParaFormat.iSpaceBeforeInTwips = Read32(ptr);
  1063 					}
  1064 				ptr+=KParaSpaceBefore;
  1065 				break;
  1066 			case EAttSpaceAfter:
  1067 				if (!old_mask.AttribIsSet(EAttSpaceAfter))
  1068 					{
  1069 					new_mask.SetAttrib(EAttSpaceAfter);
  1070 					aParaFormat.iSpaceAfterInTwips = Read32(ptr);
  1071 					}
  1072 				ptr+=KParaSpaceAfter;
  1073 				break;
  1074 			case EAttKeepTogether:
  1075 				if (!old_mask.AttribIsSet(EAttKeepTogether))
  1076 					{
  1077 					new_mask.SetAttrib(EAttKeepTogether);
  1078 					aParaFormat.iKeepTogether=TBool(*ptr);
  1079 					}
  1080 				ptr+=KParaKeepTogether;
  1081 				break;
  1082 			case EAttKeepWithNext:
  1083 				if (!old_mask.AttribIsSet(EAttKeepWithNext))
  1084 					{
  1085 					new_mask.SetAttrib(EAttKeepWithNext);
  1086 					aParaFormat.iKeepWithNext=(TBool)*ptr;
  1087 					}
  1088 				ptr+=KParaKeepWithNext;
  1089 				break;
  1090 			case EAttStartNewPage:
  1091 				if (!old_mask.AttribIsSet(EAttStartNewPage))
  1092 					{
  1093 					new_mask.SetAttrib(EAttStartNewPage);
  1094 					aParaFormat.iStartNewPage=TBool(*ptr);
  1095 					}
  1096 				ptr+=KParaStartNewPage;
  1097 				break;
  1098 			case EAttWidowOrphan:
  1099 				if (!old_mask.AttribIsSet(EAttWidowOrphan))
  1100 					{
  1101 					new_mask.SetAttrib(EAttWidowOrphan);
  1102 					aParaFormat.iWidowOrphan=TBool(*ptr);
  1103 					}
  1104 				ptr+=KParaWidowOrphan;
  1105 				break;
  1106 			case EAttWrap:
  1107 				if (!old_mask.AttribIsSet(EAttWrap))
  1108 					{
  1109 					new_mask.SetAttrib(EAttWrap);
  1110 					aParaFormat.iWrap=TBool(*ptr);
  1111 					}
  1112 				ptr+=KParaWrap;
  1113 				break;
  1114 			case EAttBorderMargin:
  1115 				if (!old_mask.AttribIsSet(EAttBorderMargin))
  1116 					{
  1117 					new_mask.SetAttrib(EAttBorderMargin);
  1118 					aParaFormat.iBorderMarginInTwips = Read32(ptr);
  1119 					}
  1120 				ptr+=KParaBorderMargin;
  1121 				break;
  1122 			case EAttTopBorder:
  1123 				if (aMode==CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttTopBorder) )
  1124 					{
  1125 					ptr+=KParaTopBorder;
  1126 					}
  1127 				else
  1128 					{
  1129 					TParaBorder border;
  1130 					ptr=ReadValue(ptr,border);
  1131 					aParaFormat.SetParaBorderL(CParaFormat::EParaBorderTop,border);
  1132 					new_mask.SetAttrib(EAttTopBorder);
  1133 					}
  1134 				break;
  1135 			case EAttBottomBorder:
  1136 				if (aMode==CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttBottomBorder))
  1137 					{
  1138 					ptr+=KParaBottomBorder;
  1139 					}
  1140 				else
  1141 					{
  1142 					TParaBorder border;
  1143 					ptr=ReadValue(ptr,border);
  1144 					aParaFormat.SetParaBorderL(CParaFormat::EParaBorderBottom,border);
  1145 					new_mask.SetAttrib(EAttBottomBorder);
  1146 					}
  1147 				break;
  1148 			case EAttLeftBorder:
  1149 				if (aMode==CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttLeftBorder))
  1150 					{
  1151 					ptr+=KParaLeftBorder;
  1152 					}
  1153 				else
  1154 					{
  1155 					TParaBorder border;
  1156 					ptr=ReadValue(ptr,border);
  1157 					aParaFormat.SetParaBorderL(CParaFormat::EParaBorderLeft,border);
  1158 					new_mask.SetAttrib(EAttLeftBorder);
  1159 					}
  1160 				break;
  1161 			case EAttRightBorder:
  1162 				if (aMode==CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttRightBorder))
  1163 					{
  1164 					ptr+=KParaRightBorder;
  1165 					}
  1166 				else
  1167 					{
  1168 					TParaBorder border;
  1169 					ptr=ReadValue(ptr,border);
  1170 					aParaFormat.SetParaBorderL(CParaFormat::EParaBorderRight,border);
  1171 					new_mask.SetAttrib(EAttRightBorder);
  1172 					}
  1173 				break;
  1174 			case EAttDefaultTabWidth:
  1175 				if (!old_mask.AttribIsSet(EAttDefaultTabWidth))
  1176 					{
  1177 					new_mask.SetAttrib(EAttDefaultTabWidth);
  1178 					aParaFormat.iDefaultTabWidthInTwips = Read32(ptr);
  1179 					}
  1180 				ptr+=KParaDefaultTabWidth;
  1181 				break;
  1182 			case EAttTabStop:
  1183 				{
  1184 				if (aMode==CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttTabStop))
  1185 					ptr += KParaTabStop;
  1186 				else
  1187 					{
  1188 					ptr=ReadTabL(ptr,aParaFormat);
  1189 					new_mask.SetAttrib(EAttTabStop);
  1190 					}
  1191 				break;
  1192 				}
  1193 			case EAttBullet:
  1194 				{
  1195 				if (aMode==CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttBullet))
  1196 					{
  1197 					ptr+=*ptr+1;  // variable length attribute
  1198 					}
  1199 				else
  1200 					{
  1201 					TBullet* bullet=aParaFormat.iBullet;
  1202 					if (!bullet)
  1203 						aParaFormat.iBullet=bullet=new(ELeave) TBullet;
  1204 					ptr=ReadValue(ptr,*bullet);
  1205 					new_mask.SetAttrib(EAttBullet);
  1206 					//coverity[memory_leak]
  1207 					}
  1208 				break;
  1209 				}
  1210 			case EAttParaLanguage:
  1211 				if (!old_mask.AttribIsSet(EAttParaLanguage))
  1212 					{
  1213 					new_mask.SetAttrib(EAttParaLanguage);
  1214 					aParaFormat.iLanguage=TInt32(*ptr);
  1215 					}
  1216 				ptr+=KParaLanguage;
  1217 				break;
  1218 
  1219 			// Auxiliary attributes for system colours, etc.
  1220 			case EAttFillSystemColor:
  1221 				if (!old_mask.AttribIsSet(EAttFillColor))
  1222 					ReadSystemColor(ptr,aParaFormat.iFillColor);
  1223 				ptr++;
  1224 				break;
  1225 			case EAttBulletSystemColor:
  1226 				if (aMode != CParaFormat::EFixedAttributes && !old_mask.AttribIsSet(EAttBullet) &&
  1227 					aParaFormat.iBullet)
  1228 					ReadSystemColor(ptr,aParaFormat.iBullet->iColor);
  1229 				ptr++;					
  1230 				break;
  1231 			case EAttTopBorderSystemColor:
  1232 				if (aMode != CParaFormat::EFixedAttributes && !old_mask.AttribIsSet(EAttTopBorder))
  1233 					{
  1234 					b = aParaFormat.ParaBorderPtr(CParaFormat::EParaBorderTop);
  1235 					if (b)
  1236 						ReadSystemColor(ptr,b->iColor);
  1237 					}
  1238 				ptr++;
  1239 				break;
  1240 			case EAttBottomBorderSystemColor:
  1241 				if (aMode != CParaFormat::EFixedAttributes && !old_mask.AttribIsSet(EAttBottomBorder))
  1242 					{
  1243 					b = aParaFormat.ParaBorderPtr(CParaFormat::EParaBorderBottom);
  1244 					if (b)
  1245 						ReadSystemColor(ptr,b->iColor);
  1246 					}
  1247 				ptr++;
  1248 				break;
  1249 			case EAttLeftBorderSystemColor:
  1250 				if (aMode != CParaFormat::EFixedAttributes && !old_mask.AttribIsSet(EAttLeftBorder))
  1251 					{
  1252 					b = aParaFormat.ParaBorderPtr(CParaFormat::EParaBorderLeft);
  1253 					if (b)
  1254 						ReadSystemColor(ptr,b->iColor);
  1255 					}
  1256 				ptr++;
  1257 				break;
  1258 			case EAttRightBorderSystemColor:
  1259 				if (aMode != CParaFormat::EFixedAttributes && !old_mask.AttribIsSet(EAttRightBorder))
  1260 					{
  1261 					b = aParaFormat.ParaBorderPtr(CParaFormat::EParaBorderRight);
  1262 					if (b)
  1263 						ReadSystemColor(ptr,b->iColor);
  1264 					}
  1265 				ptr++;
  1266 				break;
  1267 			case EAttParaLanguageX:
  1268 				if (!old_mask.AttribIsSet(EAttParaLanguage))
  1269 					{
  1270 					new_mask.SetAttrib(EAttParaLanguage);
  1271 					aParaFormat.iLanguage = Read32(ptr);
  1272 					}
  1273 				ptr += KParaLanguageX;
  1274 				break;
  1275 			case EAttBulletX:
  1276 				if (aMode == CParaFormat::EFixedAttributes || old_mask.AttribIsSet(EAttBullet) ||
  1277 					!aParaFormat.iBullet)
  1278 					ptr += KParaBulletX;
  1279 				else
  1280 					{
  1281 					aParaFormat.iBullet->iStyle = (TBullet::TStyle)(*ptr++);
  1282 					aParaFormat.iBullet->iStartNumber = Read32(ptr);
  1283 					ptr += sizeof(TInt32);
  1284 					aParaFormat.iBullet->iAlignment = (TBullet::TAlignment)(*ptr++);
  1285 					}
  1286 				break;
  1287 
  1288 			default:  // have maybe read an attribute defined by a later version; stop here
  1289 				aMask = new_mask;
  1290 				return;
  1291 			}
  1292 		}
  1293 	aMask = new_mask;
  1294 	}
  1295 
  1296 /*
  1297 Extract format attributes selected by cleared bits in aMask.
  1298 Put them into aCharFormatX if the corresponding bit in aMask is NOT set, and set the bit.
  1299 */
  1300 void RFormatStream::SenseCharFormat(TCharFormatX& aCharFormat,TCharFormatXMask& aMask) const
  1301 	{
  1302 	__TEST_INVARIANT;
  1303 
  1304 	TUint8* ptr = iBase;
  1305 	if (ptr == NULL)
  1306 		return;
  1307 	TCharFormatXMask old_mask = aMask;
  1308 	TCharFormatXMask new_mask = aMask;
  1309 	TUint8* end = iEnd;
  1310 	TCharFormat& format = aCharFormat.iCharFormat;
  1311 
  1312 	while (ptr<end)
  1313 		{
  1314 		TUint8 type = *ptr;
  1315 		ptr+=KTypeSize;
  1316 		switch (type)
  1317 			{
  1318 			case EAttColor:
  1319 				if (!old_mask.AttribIsSet(EAttColor))
  1320 					{
  1321 					new_mask.SetAttrib(EAttColor);
  1322 					ptr=ReadValue(ptr,format.iFontPresentation.iTextColor);
  1323 					}
  1324 				else
  1325 					ptr+=KCharColor;
  1326 				break;
  1327 			case EAttFontHighlightColor:
  1328 				if (!old_mask.AttribIsSet(EAttFontHighlightColor))
  1329 					{
  1330 					new_mask.SetAttrib(EAttFontHighlightColor);
  1331 					ptr=ReadValue(ptr,format.iFontPresentation.iHighlightColor);
  1332 					}
  1333 				else
  1334 					ptr+=KCharHighlightColor;
  1335 				break;
  1336 			case EAttFontHighlightStyle:
  1337 				if (!old_mask.AttribIsSet(EAttFontHighlightStyle))
  1338 					{
  1339 					new_mask.SetAttrib(EAttFontHighlightStyle);
  1340 					format.iFontPresentation.iHighlightStyle=(TFontPresentation::TFontHighlightStyle)*ptr;
  1341 					}
  1342 				ptr+=KCharHighlightStyle;
  1343 				break;
  1344 			case EAttFontStrikethrough:
  1345 				if (!old_mask.AttribIsSet(EAttFontStrikethrough))
  1346 					{
  1347 					new_mask.SetAttrib(EAttFontStrikethrough);
  1348 					format.iFontPresentation.iStrikethrough=(TFontStrikethrough)*ptr;
  1349 					}
  1350 				ptr+=KCharStrikethrough;
  1351 				break;
  1352 			case EAttFontUnderline:
  1353 				if (!old_mask.AttribIsSet(EAttFontUnderline))
  1354 					{
  1355 					new_mask.SetAttrib(EAttFontUnderline);
  1356 					format.iFontPresentation.iUnderline=(TFontUnderline)*ptr;
  1357 					}
  1358 				ptr+=KCharUnderline;
  1359 				break;
  1360 			case EAttFontHeight:
  1361 				if (!old_mask.AttribIsSet(EAttFontHeight))
  1362 					{
  1363 					new_mask.SetAttrib(EAttFontHeight);
  1364 					format.iFontSpec.iHeight = Read32(ptr);
  1365 					}
  1366 				ptr+=KCharFontHeight;
  1367 				break;
  1368 			case EAttFontPosture:
  1369 				if (!old_mask.AttribIsSet(EAttFontPosture))
  1370 					{
  1371 					new_mask.SetAttrib(EAttFontPosture);
  1372 					format.iFontSpec.iFontStyle.SetPosture((TFontPosture)*ptr);
  1373 					}
  1374 				ptr+=KCharPosture;
  1375 				break;
  1376 			case EAttFontStrokeWeight:
  1377 				if (!old_mask.AttribIsSet(EAttFontStrokeWeight))
  1378 					{
  1379 					new_mask.SetAttrib(EAttFontStrokeWeight);
  1380 					format.iFontSpec.iFontStyle.SetStrokeWeight((TFontStrokeWeight)*ptr);
  1381 					}
  1382 				ptr+=KCharStrokeWeight;
  1383 				break;
  1384 			case EAttFontPrintPos:
  1385 				if (!old_mask.AttribIsSet(EAttFontPrintPos))
  1386 					{
  1387 					new_mask.SetAttrib(EAttFontPrintPos);
  1388 					format.iFontSpec.iFontStyle.SetPrintPosition((TFontPrintPosition)*ptr);
  1389 					}
  1390 				ptr+=KCharPrintPos;
  1391 				break;
  1392 			case EAttFontTypeface:
  1393 				if (!old_mask.AttribIsSet(EAttFontTypeface))
  1394 					{
  1395 					new_mask.SetAttrib(EAttFontTypeface);
  1396 					ptr=ReadValue(ptr,format.iFontSpec.iTypeface);  // updates ptr
  1397 					}
  1398 				else
  1399 					ptr+=*ptr+1;	// variable length attribute
  1400 				break;
  1401 			case EAttBitmapType:
  1402 				if (!old_mask.AttribIsSet(EAttFontTypeface))
  1403 					{
  1404 					format.iFontSpec.iFontStyle.SetBitmapType(static_cast <TGlyphBitmapType> (*ptr));
  1405 					}
  1406 				ptr+=KBitmapType;
  1407 				break;
  1408 			case EAttCharLanguage:
  1409 				if (!old_mask.AttribIsSet(EAttCharLanguage))
  1410 					{
  1411 					new_mask.SetAttrib(EAttCharLanguage);
  1412 					format.iLanguage=*ptr;
  1413 					}
  1414 				ptr+=KCharLanguage;
  1415 				break;
  1416 			case EAttFontHiddenText:
  1417 				if (!old_mask.AttribIsSet(EAttFontHiddenText))
  1418 					{
  1419 					new_mask.SetAttrib(EAttFontHiddenText);
  1420 					format.iFontPresentation.iHiddenText=(TBool)*ptr;
  1421 					}
  1422 				ptr+=KCharHiddenText;
  1423 				break;
  1424 			case EAttFontPictureAlignment:
  1425 				if (!old_mask.AttribIsSet(EAttFontPictureAlignment))
  1426 					{
  1427 					new_mask.SetAttrib(EAttFontPictureAlignment);
  1428 					format.iFontPresentation.iPictureAlignment=(TFontPresentation::TAlignment)*ptr;
  1429 					}
  1430 				ptr+=KCharPictureAlignment;
  1431 				break;
  1432 
  1433 			// Auxiliary attributes for system colours, etc.
  1434 			case EAttTextSystemColor:
  1435 				if (!old_mask.AttribIsSet(EAttColor))
  1436 					ReadSystemColor(ptr,format.iFontPresentation.iTextColor);
  1437 				ptr++;
  1438 				break;
  1439 
  1440 			case EAttFontHighlightSystemColor:
  1441 				if (!old_mask.AttribIsSet(EAttFontHighlightColor))
  1442 					ReadSystemColor(ptr,format.iFontPresentation.iHighlightColor);
  1443 				ptr++;
  1444 				break;
  1445 
  1446 			case EAttCharLanguageX:
  1447 				if (!old_mask.AttribIsSet(EAttCharLanguage))
  1448 					{
  1449 					new_mask.SetAttrib(EAttCharLanguage);
  1450 					format.iLanguage = Read32(ptr);
  1451 					}
  1452 				ptr += KCharLanguageX;
  1453 				break;
  1454 
  1455 			// Internal character attributes:
  1456 			case EAttParserTag:
  1457 				if (!old_mask.AttribIsSet(EAttParserTag))
  1458 					{
  1459 					new_mask.SetAttrib(EAttParserTag);
  1460 					aCharFormat.iParserTag = Read32(ptr);
  1461 					}
  1462 				ptr += KCharParserTag;
  1463 				break;
  1464 
  1465 			default:	// have maybe read an attribute defined by a later version; stop here
  1466 				aMask = new_mask;
  1467 				return;
  1468 			}
  1469 		}
  1470 	aMask = new_mask;
  1471 	}
  1472 
  1473 /** Swaps the contents of this stream with aStream.
  1474 @param aStream Object to swap contents with. */
  1475 void RFormatStream::Swap(RFormatStream& aStream)
  1476 	{
  1477 	TUint8* t = iBase;
  1478 	iBase = aStream.iBase;
  1479 	aStream.iBase = t;
  1480 	t = iEnd;
  1481 	iEnd = aStream.iEnd;
  1482 	aStream.iEnd = t;
  1483 	}
  1484 
  1485 TInt RFormatStream::StoreTabs(TUint8*& aPtr,const CParaFormat& aDesiredFormat,const CParaFormat& aCurrentFormat,TBool aStoreData)
  1486 // Merges the tabs from the desired paraFormat with those from the current effective paraFormat, resolving differences
  1487 // as described below, and store the resultant tablist in this layer.
  1488 // If aStoreData is false, then return the number of bytes necessary to store all required tabs
  1489 //
  1490 	{
  1491 	TInt requiredTabCount=0;
  1492 	TInt desiredTabCount=aDesiredFormat.TabCount();
  1493 	TInt currentTabCount=aCurrentFormat.TabCount();
  1494 	if (desiredTabCount==0 && currentTabCount==0)
  1495 		{// Nothing needs to be stored.
  1496 		return requiredTabCount;
  1497 		}
  1498 	if (desiredTabCount>0 && currentTabCount>0)
  1499 		{// The 2 tab lists need merging.
  1500 		requiredTabCount=MergeTabLists(aPtr,aDesiredFormat,desiredTabCount,aCurrentFormat,currentTabCount,aStoreData);
  1501 		}
  1502 	else if (desiredTabCount>0 && currentTabCount==0)
  1503 			{// There are no previous tabs to merge so just store all the desired ones.
  1504 			if (aStoreData)
  1505 				StoreAllTabs(aPtr,aDesiredFormat);
  1506 			else
  1507 				requiredTabCount+=desiredTabCount;
  1508 			}
  1509 		else if (desiredTabCount==0 && currentTabCount>0)
  1510 				{// We want to delete (NULL) all existing tabs.
  1511 				if (aStoreData)
  1512 					{
  1513 					for (TInt index=0;index<currentTabCount;index++)
  1514 						{
  1515 						TTabStop tab=aCurrentFormat.TabStop(index);
  1516 						tab.iType=TTabStop::ENullTab;
  1517 						StoreTab(aPtr,tab);
  1518 						}
  1519 					}
  1520 				else requiredTabCount+=currentTabCount;
  1521 				}
  1522 	return requiredTabCount*(KTypeSize+KParaTabStop);
  1523 	}
  1524 
  1525 
  1526 TInt RFormatStream::MergeTabLists(TUint8*& aPtr,const CParaFormat& aDesiredFormat,TInt aDesiredTabCount,
  1527 								  const CParaFormat& aCurrentFormat,TInt aCurrentTabCount,TBool aStoreData)
  1528 // Compares the desired tablist (from a dialog interaction) with that of the current tablist.
  1529 // The resulting tab stop is stored if aStoreData is TRUE.
  1530 // There may be 2 tabs for any for any given tabstops.
  1531 // The possible results of the comparison are:
  1532 // (1) Tab is in desired but not current --> Store the desired tab.
  1533 // (2) Tab is in current but not desired --> Null the current and store.
  1534 // (3) Tab is in both current and desired -->
  1535 //			(i) if tab type is the same then store nothing -- rely on the base one being inherited.
  1536 //			(ii)if tab type is not same then store the current -- overriding whatevers in the base.
  1537 //
  1538 	{
  1539 	TInt requiredTabCount=0;
  1540 	TInt offsetInDesired,offsetInCurrent;
  1541 	offsetInDesired=offsetInCurrent=0;
  1542 	TTabStop desiredTab,currentTab;
  1543 	while ((offsetInDesired!=aDesiredTabCount) && (offsetInCurrent!=aCurrentTabCount))
  1544 		{// Do this until we exhaust one of the tab lists.
  1545 		desiredTab=aDesiredFormat.TabStop(offsetInDesired);
  1546 		currentTab=aCurrentFormat.TabStop(offsetInCurrent);
  1547 		if (desiredTab.iTwipsPosition<currentTab.iTwipsPosition)
  1548 			{
  1549 			if (aStoreData)
  1550 				aPtr=StoreTab(aPtr,desiredTab);  // Store desiredTab.
  1551 			else 
  1552 				requiredTabCount++;
  1553 			offsetInDesired++;  // Move to the next tab in the desired tablist.
  1554 			continue;
  1555 			}
  1556 		if (desiredTab.iTwipsPosition>currentTab.iTwipsPosition)
  1557 			{
  1558 			if (aStoreData)
  1559 				{
  1560 				currentTab.iType=TTabStop::ENullTab;  // Null the current current tab.
  1561 				aPtr=StoreTab(aPtr,currentTab);  // Store NULL'd currentTab
  1562 				}
  1563 			else
  1564 				requiredTabCount++;
  1565 			offsetInCurrent++;  // Move to the next tab in the current tablist.
  1566 			continue;
  1567 			}
  1568 		if (desiredTab.iTwipsPosition==currentTab.iTwipsPosition)
  1569 			{
  1570 			if (desiredTab.iType!=currentTab.iType)
  1571 				{// Store desired type to override the one from the lower layer.
  1572 				if (aStoreData)
  1573 					aPtr=StoreTab(aPtr,desiredTab);
  1574 				else
  1575 					requiredTabCount++;
  1576 				}// Otherwise rely on the one in the base - store nothing but increment.
  1577 			offsetInDesired++;
  1578 			offsetInCurrent++;
  1579 			continue;
  1580 			}
  1581 		}
  1582 	TInt currentLeftToDo=aCurrentTabCount-offsetInCurrent;
  1583 	TInt desiredLeftToDo=aDesiredTabCount-offsetInDesired;
  1584 	// Spot which list is exhausted and process the rest of the remainder appropriately.
  1585 	if (currentLeftToDo)
  1586 		{// Store null'd remainder of current tab list.
  1587 		for (;offsetInCurrent<aCurrentTabCount;offsetInCurrent++)
  1588 			{
  1589 			if (aStoreData)
  1590 				{
  1591 				currentTab=aCurrentFormat.TabStop(offsetInCurrent);
  1592 				currentTab.iType=TTabStop::ENullTab;  // Null the current current tab.
  1593 				aPtr=StoreTab(aPtr,currentTab);  // Store NULL'd currentTab
  1594 				}
  1595 			else
  1596 				requiredTabCount++;
  1597 			}
  1598 		}
  1599 	if (desiredLeftToDo)
  1600 		{// Store remainder of desired tab list.
  1601 		for (;offsetInDesired<aDesiredTabCount;offsetInDesired++)
  1602 			{
  1603 			if (aStoreData)
  1604 				aPtr=StoreTab(aPtr,aDesiredFormat.TabStop(offsetInDesired));
  1605 			else
  1606 				requiredTabCount++;
  1607 			}
  1608 		}
  1609 // ASSERT: The tabList offsets are as we would expect in a normal (correct) completion.	
  1610 	if (offsetInDesired!=aDesiredTabCount)
  1611 	    {
  1612 	    OstTrace0( TRACE_FATAL, RFORMATSTREAM_MERGETABLISTS, "EStoreTabError" );
  1613 	    }
  1614 	__ASSERT_ALWAYS(offsetInDesired==aDesiredTabCount,Panic(EStoreTabError));
  1615 	if (offsetInCurrent!=aCurrentTabCount)
  1616 	    {
  1617 	    OstTrace0( TRACE_FATAL, DUP1_RFORMATSTREAM_MERGETABLISTS, "EStoreTabError" );
  1618 	    }
  1619 	__ASSERT_ALWAYS(offsetInCurrent==aCurrentTabCount,Panic(EStoreTabError));
  1620 	return requiredTabCount;
  1621 	}
  1622 
  1623 
  1624 // Writes all tabs from aSource.
  1625 void RFormatStream::StoreAllTabs(TUint8*& aPtr,const CParaFormat& aSource)
  1626 	{
  1627 	int tabs = aSource.TabCount();
  1628 	for (TInt index = 0;index < tabs; index++)
  1629 		aPtr = StoreTab(aPtr,TTabStop(aSource.TabStop(index)));
  1630 	}
  1631 
  1632 
  1633 TUint8* RFormatStream::StoreBullet(TUint8* aPtr,const TBullet& aBullet)
  1634 	{
  1635 	// Write the attribute code and leave a gap for the length.
  1636 	*aPtr++ = TUint8(EAttBullet);
  1637 	TUint8* length_ptr = aPtr++;
  1638 
  1639 	// Store the font height.
  1640 	Write32(aPtr,aBullet.iHeightInTwips);
  1641 
  1642 	// Store the bullet character code.
  1643 	aPtr[0] = (TUint8)(aBullet.iCharacterCode);
  1644 	aPtr[1] = (TUint8)(aBullet.iCharacterCode >> 8);
  1645 	aPtr += sizeof(TText);
  1646 
  1647 	// Store the hanging indent
  1648 	*aPtr++ = TUint8(aBullet.iHangingIndent != FALSE);
  1649 
  1650 	// Store the colour.
  1651 	aPtr = Store(aPtr,aBullet.iColor);
  1652 
  1653 	// Store the typeface
  1654 	aPtr = Store(aPtr,aBullet.iTypeface);
  1655 
  1656 	// Store the number of data bytes stored.
  1657 	*length_ptr = (TUint8)(aPtr - length_ptr - 1);
  1658 
  1659 	return aPtr;
  1660 	}
  1661 
  1662 
  1663 TUint8* RFormatStream::StoreBorder(TUint8* aPtr,TTextFormatAttribute aType,const TParaBorder& aSource)
  1664 // Stores paragraph border attributes.
  1665 // Writes the line style, thickness,autoColor flag
  1666 // and Color, from aSource to the end of the format stream.
  1667 // aType is converted from an enum to a TUint8 as it is stored in the format stream.
  1668 //
  1669 	{
  1670 	*aPtr++=TUint8(aType);
  1671 	// border line style
  1672 	*aPtr++=TUint8(aSource.iLineStyle);
  1673 	// border line thickness
  1674 	Write32(aPtr,aSource.iThickness);
  1675 	// border color
  1676 	aPtr=Store(aPtr,aSource.iColor);
  1677 	// border autocolor
  1678 	*aPtr++=TUint8(aSource.iAutoColor!=EFalse);
  1679 	return aPtr;
  1680 	}
  1681 
  1682 
  1683 TUint8* RFormatStream::StoreTab(TUint8* aPtr,const TTabStop& aSource)
  1684 // Stores a tabstop compound attribute at the end of the stream.
  1685 // Writes the tabstop twips position and the tab type, from aSource, to the end of the format stream.
  1686 // aType is converted from an enum to a TUint8 as it is stored in the format stream.
  1687 // The tab type is compressed from a TTabType enum to a TUint8.
  1688 // Uses InsertL, which may cause expansion of the stream storage, and thus may fail.
  1689 //
  1690 	{
  1691 	*aPtr++=TUint8(EAttTabStop);
  1692 	// Store tab poisition.
  1693 	Write32(aPtr,aSource.iTwipsPosition);
  1694 	// Compress the tab type.
  1695 	*aPtr++=TUint8(aSource.iType);
  1696 	return aPtr;
  1697 	}
  1698 
  1699 
  1700 TUint8* RFormatStream::Store(TUint8* aPtr,const TLogicalRgb& aRgb)
  1701    // Store color value
  1702    //
  1703    	{
  1704 	*aPtr++=TUint8(aRgb.Red());
  1705 	*aPtr++=TUint8(aRgb.Green());
  1706 	*aPtr++=TUint8(aRgb.Blue());
  1707 	return aPtr;
  1708    	}
  1709 
  1710 
  1711 TUint8* RFormatStream::Store(TUint8* aPtr,const TTypeface& aTypeface)
  1712 // Stores typeface name and flags.
  1713 //
  1714 	{
  1715 	//
  1716 	// Store typeface name size
  1717 	TUint8 length=TUint8(aTypeface.iName.Size());
  1718 	length+=KTypefaceFlags;
  1719 	*aPtr++=length;  // byte count.
  1720 	//
  1721 	// Store typeface name
  1722 	aPtr=Mem::Copy(aPtr,aTypeface.iName.Ptr(),length-KTypefaceFlags);
  1723 	//
  1724 	// Store typeface flags
  1725 	TUint8 flags=0;
  1726 	if (aTypeface.IsProportional())
  1727 		flags|=KFontProportional;
  1728 	if (aTypeface.IsSerif())
  1729 		flags|=KFontSerif;	
  1730 	if (aTypeface.IsSymbol())
  1731 		flags|=KFontSymbol;
  1732 	*aPtr=flags;
  1733 	aPtr+=KTypefaceFlags;
  1734 	//
  1735 	return aPtr;
  1736 	}
  1737 
  1738 TUint8* RFormatStream::ReadValue(TUint8* aPtr,TLogicalRgb& aRgb)const
  1739 // Reads a color value from storage.
  1740 //
  1741 	{
  1742 	aRgb.SetRed(*aPtr++);
  1743 	aRgb.SetGreen(*aPtr++);
  1744 	aRgb.SetBlue(*aPtr++);
  1745 	aRgb.SetSystemColorIndex(0);
  1746 	return aPtr;
  1747 	}
  1748 
  1749 
  1750 TUint8* RFormatStream::ReadValue(TUint8* aPtr,TParaBorder& aBorder)const
  1751 // Reads a paragraph border from storage.
  1752 //
  1753 	{
  1754 	// Read line style
  1755 	aBorder.iLineStyle=TParaBorder::TLineStyle(*aPtr++);
  1756 	// Read thickness
  1757 	aBorder.iThickness = Read32(aPtr);
  1758 	aPtr+=sizeof(TInt32);
  1759 	// Read color
  1760 	aPtr=ReadValue(aPtr,aBorder.iColor);  // returns aPtr
  1761 	// Read autocolor
  1762 	aBorder.iAutoColor=TBool(*aPtr++);
  1763 	return aPtr;
  1764 	}
  1765 
  1766 
  1767 TUint8* RFormatStream::ReadValue(TUint8* aPtr,TBullet& aBullet)const
  1768 // Read the bullet compound attribute.
  1769 //
  1770 	{
  1771 	// Read the length of this bullet attribute
  1772 	aPtr++;  // length not used in this context, skip it
  1773 	// Read bullet twips height into target
  1774 	aBullet.iHeightInTwips = Read32(aPtr);
  1775 	aPtr+=sizeof(TInt32);
  1776 	// Read bullet character code into target.
  1777 	aBullet.iCharacterCode = (TText)aPtr[0] | ((TText)aPtr[1] << 8);
  1778 	aPtr+=sizeof(TText);
  1779 	// Read hanging indent.
  1780 	aBullet.iHangingIndent=TBool(*aPtr++);
  1781 	// Read Color.
  1782 	aPtr=ReadValue(aPtr,aBullet.iColor);  // returns aPtr
  1783 	// Read typeface
  1784 	return ReadValue(aPtr,aBullet.iTypeface);  // returns aPtr
  1785 	}
  1786 
  1787 
  1788 TUint8* RFormatStream::ReadValue(TUint8* aPtr,TTypeface& aTypeface)const
  1789 // Read a typeface name from storage and associated flags.
  1790 //
  1791 	{
  1792 	//
  1793 	// Read typeface name size
  1794 	TInt length=*aPtr++;
  1795 	//
  1796 	// Read typeface name
  1797 	TInt typefaceLength=length-KTypefaceFlags;
  1798 	
  1799 	if ((typefaceLength%2)!=0)
  1800 	    {
  1801 	    OstTrace0( TRACE_DUMP, RFORMATSTREAM_READVALUE, "ECorruptFormatLayer" );
  1802 	    }
  1803 	__ASSERT_DEBUG((typefaceLength%2)==0,Panic(ECorruptFormatLayer)); // must be an even number
  1804 
  1805 	TPtr name=aTypeface.iName.Des();
  1806 	Mem::Copy(CONST_CAST(TText*,name.Ptr()),aPtr,typefaceLength);
  1807 	typefaceLength>>=1;
  1808 	name.SetLength(typefaceLength);
  1809 
  1810 	aPtr+=length-KTypefaceFlags;
  1811 	//
  1812 	// Read typeface name flags
  1813 	TInt attrib=0;
  1814 	TUint flags=*aPtr;
  1815 	if (flags & KFontProportional)
  1816 		attrib|=TTypeface::EProportional;
  1817 	if (flags & KFontSerif)
  1818 		attrib|=TTypeface::ESerif;
  1819 	if (flags & KFontSymbol)
  1820 		attrib|=TTypeface::ESymbol;
  1821 	aTypeface.SetAttributes(attrib);  // reset the attributes
  1822 	return aPtr+KTypefaceFlags;
  1823 	}
  1824 
  1825 
  1826 TUint8* RFormatStream::ReadTabL(TUint8* aPtr,CParaFormat& aTarget)const
  1827 // Read the tab stop data from the stream, located at aPos, into 
  1828 // a temporary TTabStop, and use this to fill in aTarget.
  1829 // Does not read the tab if one already exists at the same twips position.
  1830 // (Implementation of tab inheritance policy).
  1831 // aPos is updated manually, rather than using Length(), since this is
  1832 // a compound attribute.
  1833 //
  1834 	{
  1835 	TTabStop tab;
  1836 	// Read tab position
  1837 	tab.iTwipsPosition = Read32(aPtr);
  1838 	aPtr+=sizeof(TInt32);
  1839 	// Read tab type
  1840 	tab.iType=TTabStop::TTabType(*aPtr++);
  1841 	// Set this tab in the paragraph format
  1842 	TInt ret=aTarget.LocateTab(tab.iTwipsPosition);		// is this necessary
  1843 	if (ret==KTabNotFound)
  1844 		aTarget.StoreTabL(tab);
  1845 	return aPtr;
  1846 	}
  1847 
  1848 
  1849 TInt RFormatStream::Length(TUint8*& aPtr,TTextFormatAttribute aType) const
  1850 // Returns the length of a Type's value
  1851 // A length of zero in the lookup table, indicates a variable
  1852 // length value.  In this case, the length is stored in the byte
  1853 // following the type, which is read and returned.
  1854 // aPtr is incremented by the appropriate amount following a read.
  1855 //
  1856 	{
  1857 	TInt length=TheAttributeLength[aType];
  1858 
  1859 	if (length<0)
  1860 	    {
  1861 	    OstTrace0( TRACE_DUMP, RFORMATSTREAM_LENGTH, "EAttributeLengthLookupNegative" );
  1862 	    }
  1863 	__ASSERT_DEBUG(length>=0,Panic(EAttributeLengthLookupNegative));
  1864 	if (length>0)
  1865 		return length;
  1866 	else
  1867 		return *aPtr++;
  1868 	}