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