os/textandloc/textrendering/texthandling/stext/TXTFMLYR.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 2003-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 #include <e32base.h>
    21 #include "TXTFRMAT.H"
    22 #include "TXTFMLYR.H"
    23 #include <txtfmstm.h>
    24 #include "TXTSTD.H"
    25 
    26 #include "OstTraceDefinitions.h"
    27 #ifdef OST_TRACE_COMPILER_IN_USE
    28 #include "TXTFMLYRTraces.h"
    29 #endif
    30 
    31 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
    32 #include "TXTFMLYR_INTERNAL.H"
    33 #endif
    34 
    35 CFormatLayer::CFormatLayer()
    36 	{
    37 	}
    38 
    39 
    40 CFormatLayer::~CFormatLayer()
    41 	{
    42 	iStore.Reset();
    43 	}
    44 
    45 
    46 DLLEXPORT_C void CFormatLayer::__DbgTestInvariant() const
    47 	{
    48 	}
    49 
    50 
    51  
    52 EXPORT_C void CFormatLayer::SetBase(const CFormatLayer* aBaseFormatLayer)
    53 /** Sets the format layer which this layer's based-on link points to.
    54 
    55 @param aBaseFormatLayer The format layer which this layer's based-on link 
    56 points to. Specify NULL if this is the final layer in the chain (the layer 
    57 on which all other layers are based). */
    58 	{
    59 	iBasedOn = aBaseFormatLayer;
    60 	}
    61 
    62 
    63  
    64 EXPORT_C const CFormatLayer* CFormatLayer::SenseBase() const
    65 /** Gets the format layer which this layer's based-on link points to. If NULL, 
    66 this layer is the final layer in the chain.
    67 
    68 @return The format layer on which this layer is based. */
    69 	{
    70 	return iBasedOn;
    71 	}
    72 
    73 
    74  
    75 EXPORT_C TInt CFormatLayer::ChainCount() const
    76 /** Gets the number of format layers in the chain, inclusive of itself. Assumes 
    77 that the format layer chain has been correctly set up to terminate with a 
    78 NULL based-on link.
    79 
    80 @return The number of format layers in the chain, counting from the current 
    81 layer. */
    82 	{
    83 	TInt chainCount=1;
    84 	const CFormatLayer* next=iBasedOn;
    85 	while (next)
    86 		{
    87 		++chainCount;
    88 		next=next->iBasedOn;
    89 		}
    90 	return chainCount;
    91 	}
    92 
    93 
    94 
    95 const TUint8* CFormatLayer::Ptr(TInt& aSize)const
    96 /** Return a pointer to the stored bytecode.*/	
    97     {
    98 	return iStore.Ptr(aSize);
    99 	}
   100 
   101 
   102 
   103 EXPORT_C void CFormatLayer::Reset()
   104 /** Deletes the contents of the format layer. The based-on link is not affected. */
   105 	{
   106 	iStore.Reset();
   107 	}
   108 
   109 
   110 
   111 EXPORT_C void CFormatLayer::InternalizeChainL(RReadStream& aStream,const CFormatLayer* aBase)
   112 /** Restores a chain of format layers from a read stream. The layer at the end 
   113 of the chain (the one furthest from this layer) is set to be based on the 
   114 specified layer aBase, which may be NULL. This layer is set to be at the head 
   115 of the restored chain (i.e. no other layers are based on it).
   116 
   117 @param aStream Stream from which the format layer chain should be internalised. 
   118 
   119 @param aBase The format layer at the end of the chain (furthest from this layer). 
   120 May be NULL. */
   121 	{
   122 	TInt chainLength=aStream.ReadInt8L();
   123 	if (chainLength<1)
   124 		User::Leave(KErrCorrupt);  // Must restore at least one layer (this), else corrupt stream.
   125 	TInt descendentCount=chainLength-1;
   126 	for (TInt loop=0;loop<descendentCount;loop++)
   127 		{// Restore each descendent of [this] layer.
   128 		CFormatLayer* layer=RestoreNewL(aStream);
   129 		layer->SetBase(aBase);
   130 		CleanupStack::PushL(layer);
   131 		aBase=layer;
   132 		}
   133 	aStream>> *this;
   134 	SetBase(aBase);
   135 	CleanupStack::Pop(descendentCount);
   136 	}
   137 
   138 
   139 
   140 
   141 EXPORT_C void CFormatLayer::ExternalizeChainL(RWriteStream& aStream,TInt aExcludeCount)const
   142 /** Stores a chain of format layers. By default an entire chain is stored unless 
   143 an exclude count is provided. In this case, the length of the chain stored 
   144 is the ChainCount() minus the exclude count. The excluded layers are the ones 
   145 starting with the layer with the NULL based-on link.
   146 
   147 Note
   148 
   149 The exclude count must be zero or greater but must be less than the total 
   150 number of layers in the chain, otherwise a panic occurs.
   151 
   152 @param aStream Stream to which the format layer chain should be externalised. 
   153 @param aExcludeCount The number of layers to be excluded. By default, zero. */
   154 	{
   155 // ASSERT: aExcludeCount is positive.
   156 	if (aExcludeCount<0)
   157 	    {
   158 	    OstTrace0( TRACE_FATAL, CFORMATLAYER_EXTERNALIZECHAINL, "ECannotStoreFormatLayerChain" );
   159 	    }
   160 	__ASSERT_ALWAYS(aExcludeCount>=0,Panic(ECannotStoreFormatLayerChain));
   161 // ASSERT: The number of layers to be excluded is less than the total no. of layers.
   162 	if (aExcludeCount>=ChainCount())
   163 	    {
   164 	    OstTrace0( TRACE_FATAL, DUP1_CFORMATLAYER_EXTERNALIZECHAINL, "ECannotStoreFormatLayerChain" );
   165 	    }
   166 	__ASSERT_ALWAYS(aExcludeCount<ChainCount(),Panic(ECannotStoreFormatLayerChain));
   167 	TInt aCount=ChainCount()-aExcludeCount;
   168 	aStream.WriteInt8L(aCount);  // Store the chain length.
   169 	ExternalizeLayersRecurseL(aStream,--aCount);
   170 	}	
   171 	
   172 
   173 void CFormatLayer::ExternalizeLayersRecurseL(RWriteStream& aStream,TInt aDescendantCount)const
   174 /** Stores a format layer chain with *aLength* number of layers in a stack-like 
   175 fashion. This necessitates navigating to the end of the chain and storing the 
   176 layers as we unwind back to the top of the chain.*/
   177 //
   178 	{
   179 	if (aDescendantCount)
   180 		{
   181 // ASSERT: The format layer chain is consistent.
   182 		if (iBasedOn==NULL)
   183 		    {
   184 		    OstTrace0( TRACE_DUMP, CFORMATLAYER_EXTERNALIZELAYERSRECURSEL, "ECorruptFormatLayerChain" );
   185 		    }
   186 		__ASSERT_ALWAYS(iBasedOn!=NULL,Panic(ECorruptFormatLayerChain));
   187 		iBasedOn->ExternalizeLayersRecurseL(aStream,--aDescendantCount);
   188 		}
   189 	aStream<< *this;
   190 	}
   191 	
   192 /** Implementations of this function compare another format layer with the
   193 current object. For the two layers to be equal, they must have the same 
   194 contents and (if the second parameter is ETrue),their based-on links must 
   195 point to the same format layer.
   196 
   197 @param aLayer The layer to compare to this format layer.
   198 @param aCheckBasedOnLink If ETrue, both layers' based-on links must point to 
   199 the same format layer. If EFalse, the based-on links are not used in the 
   200 comparison. By default, ETrue.
   201 @return  ETrue if the two layers are identical, otherwise EFalse. */
   202 TBool CFormatLayer::IsIdentical(const TUint8* aPtr,TInt aSize) const
   203 	{
   204 	TInt size=0;
   205 	const TUint8* ptr=iStore.Ptr(size);
   206 	if ((ptr==NULL && aPtr!=NULL) || (ptr!=NULL && aPtr==NULL))
   207 		return EFalse;
   208 	if (ptr==NULL && aPtr==NULL)
   209 		return ETrue;
   210 	return (!(TBool)(Mem::Compare(ptr,size,aPtr,aSize)));
   211 	}
   212 
   213 /** Tests whether any formatting is stored in the format layer.
   214 
   215 @return ETrue if no formatting is stored in the format layer, otherwise returns 
   216 EFalse. */
   217 EXPORT_C TBool CFormatLayer::IsEmpty() const
   218 	{
   219 	TInt size=0;
   220 	return iStore.Ptr(size)==NULL;
   221 	}
   222 
   223 /** Swaps the contents of this with aLayer.
   224 @param aLayer The layer to swap contents with.
   225 @internalComponent */
   226 void CFormatLayer::Swap(CFormatLayer& aLayer)
   227 	{
   228 	iStore.Swap(aLayer.iStore);
   229 	const CFormatLayer* t = iBasedOn;
   230 	iBasedOn = aLayer.iBasedOn;
   231 	aLayer.iBasedOn = t;
   232 	}
   233 
   234 /** Allocates and constructs an empty CParaFormatLayer. Its based-on link is 
   235 NULL.
   236 
   237 Note: Use SetL() to set format attributes in the layer. Use SetBase(), defined 
   238 in the base class CFormatLayer, to set the layer's based on link.
   239 
   240 @return Pointer to the new paragraph format layer. */
   241 EXPORT_C CParaFormatLayer* CParaFormatLayer::NewL()
   242 	{
   243 	return new(ELeave) CParaFormatLayer;
   244 	}
   245 
   246 
   247 EXPORT_C CParaFormatLayer* CParaFormatLayer::NewL(RReadStream& aStream)
   248 /** Allocates and constructs a CParaFormatLayer, restoring its format attributes 
   249 from a stream. The layer's based-on link is set to NULL.
   250 
   251 @param aStream Stream from which the layer is restored. 
   252 @return Pointer to the new paragraph format layer. */
   253 	{
   254 	CParaFormatLayer* self=NewL();
   255 	CleanupStack::PushL(self);
   256 	self->InternalizeL(aStream);
   257 	CleanupStack::Pop();
   258 	return self;
   259 	}
   260 
   261 
   262 EXPORT_C CParaFormatLayer* CParaFormatLayer::NewL(const CParaFormat* aFormat,const TParaFormatMask& aMask)
   263 /** Allocates and constructs a CParaFormatLayer. The attributes which are set 
   264 in the mask are initialised to the values specified in the format container 
   265 aParaFormat. The attributes which are not set in the mask are initialised to 
   266 the default values for class CParaFormat. The new layer's based-on link is set 
   267 to NULL.
   268 
   269 @param aParaFormat Contains the attribute values to assign to the format layer. 
   270 @param aMask Mask specifying which attributes should be initialized from 
   271 aParaFormat. 
   272 @return Pointer to ParaFormatLayer the new paragraph format layer. */
   273 	{
   274 	CParaFormatLayer* self=NewL();
   275 	CleanupStack::PushL(self);
   276 	self->SetL(aFormat,aMask);
   277 	CleanupStack::Pop();
   278 	return self;
   279 	}
   280 
   281 
   282 CParaFormatLayer* CParaFormatLayer::NewL(const CParaFormatLayer* aLayer)
   283 /** Copying construction
   284  does not copy the based on link*/
   285 	{
   286 	CParaFormatLayer* self=NewL();  // based-on is NULL
   287 	CleanupStack::PushL(self);
   288 	aLayer->CloneLayerL(self);
   289 	CleanupStack::Pop();			// self
   290 	return self;
   291 	}
   292 
   293 CParaFormatLayer* CParaFormatLayer::NewCopyBaseL(const CParaFormatLayer* aLayer)
   294 /** Copying construction
   295  copies based-on link*/
   296 	{
   297 	CParaFormatLayer* self=NewL(aLayer);  // based-on is NULL
   298 	self->iBasedOn=aLayer->iBasedOn;
   299 	return self;
   300 	}
   301 
   302 CParaFormatLayer::CParaFormatLayer()
   303 /** Constructor.*/
   304 // No ConstructL method since allocation is postponed until first Set.
   305 //
   306 	{
   307 	}
   308 
   309 EXPORT_C void CParaFormatLayer::ExternalizeL(RWriteStream& aStream)const
   310 /** Externalises the paragraph format layer but not its based-on link to a 
   311 write stream. The presence of this function means that the standard templated 
   312 operator<<() (defined in s32strm.h) is available to externalise objects of 
   313 this class.
   314 
   315 @param aStream Stream to which the format layer should be externalised. */
   316 	{aStream<< iStore;}
   317 
   318 EXPORT_C void CParaFormatLayer::InternalizeL(RReadStream& aStream,const CFormatLayer* aBase)
   319 /** Internalises the paragraph format layer but not its based-on link from a 
   320 read stream. The presence of this function means that the standard templated 
   321 operator>>() (defined in s32strm.h) is available to internalise objects of 
   322 this class. The internalised layer is set to be based on the layer specified.
   323 
   324 @param aStream Stream from which the format layer should be internalised. 
   325 @param aBase The based-on link to assign to the layer. By default NULL. */
   326 	{
   327 	aStream>> iStore;
   328 	SetBase(aBase);
   329 	}
   330 
   331 EXPORT_C void CParaFormatLayer::SetL(const CParaFormat* aDesiredEffectiveFormat,const TParaFormatMask& aMask)
   332 /** Sets the layer's format attributes. The attributes which are set in the 
   333 mask are set in the layer to the values specified in the format container 
   334 aDesiredEffectiveFormat. The attributes which are not set in the mask are not 
   335 changed.
   336 
   337 Note: Any tab stops in aDesiredEffectiveFormat are merged with the tab stops in 
   338 the current layer.
   339 
   340 @param aDesiredEffectiveFormat Contains the attribute values to assign to 
   341 the format layer. 
   342 @param aMask Mask specifying which attributes should be set from 
   343 aDesiredEffectiveFormat. */
   344 	{
   345 	if ( !aDesiredEffectiveFormat )
   346 		{
   347 		return;
   348 		}
   349 	
   350 	const CParaFormat& desiredFormat = *aDesiredEffectiveFormat;
   351 	CParaFormat currentEffectiveFormat;
   352 	ResetOnCleanupL( &currentEffectiveFormat);
   353 	if (iBasedOn)
   354 		((CParaFormatLayer*)iBasedOn)->SenseEffectiveL(&currentEffectiveFormat);
   355 	iStore.SetParaFormatL(desiredFormat,aMask,currentEffectiveFormat);
   356 	CleanupStack::Pop();
   357 	}
   358 
   359 EXPORT_C void CParaFormatLayer::SenseEffectiveL(CParaFormat* aParaFormat,CParaFormat::TParaFormatGetMode aMode)const
   360 /** Senses the layer's effective format, searching all based-on links. The 
   361 resulting aParaFormat is fully populated, except that if aMode is 
   362 EFixedAttributes, then only the fixed attributes (not tabs, paragraph borders 
   363 or bullets) are written to it.
   364 
   365 Notes:
   366 
   367 The function also "tidies up" the layer's effective paragraph formatting, 
   368 so that any zero height bullets, paragraph borders with a NULL line style 
   369 or NULL tab stops are removed. 
   370 
   371 The function can only leave if aMode has a value of EAllAttributes.
   372 
   373 @param aParaFormat On return, contains the layer's effective formatting. 
   374 Depending on the value of aMode, tabs, borders and bullets may be excluded. 
   375 Must not be NULL or a panic occurs. 
   376 @param aMode Controls which attributes are written to aParaFormat. If 
   377 EAllAttributes, all attributes are written; if EFixedAttributes, tabs, 
   378 bullets and borders are not written. */
   379 	{
   380 	if (aParaFormat==NULL)
   381 	    {
   382 	    OstTrace0( TRACE_FATAL, CPARAFORMATLAYER_SENSEEFFECTIVEL, "ENullFormatPointer" );
   383 	    }
   384 	__ASSERT_ALWAYS(aParaFormat!=NULL,Panic(ENullFormatPointer));
   385 	aParaFormat->Reset();
   386 	TParaFormatMask mask;
   387 	FillParaFormatL(aParaFormat,mask,aMode);
   388 	CleanupEffectiveFormat(aParaFormat,mask);
   389 	}
   390 
   391 EXPORT_C void CParaFormatLayer::SenseL(CParaFormat* aParaFormat,TParaFormatMask& aMask,CParaFormat::TParaFormatGetMode aMode)const
   392 /** Senses the formatting which has been applied to the current layer only. No 
   393 based-on links are searched. This function does not get the effective formatting, 
   394 but the resulting aParaFormat is useable even if not all attributes are flagged 
   395 for sensing in aMask because any attribute values not sensed from the current 
   396 layer, are set to default values.
   397 
   398 The function can only leave if aMode has a value of EAllAttributes.
   399 
   400 @param aParaFormat On return, contains the formatting which has been applied 
   401 to the current layer only. Any attributes not explicitly set in the current 
   402 layer are initialised to the default values for a CParaFormat. Attributes 
   403 specified in aMask are not sensed from this layer. The values for these 
   404 attributes are also initialised to the default settings. Must not be NULL or 
   405 a panic occurs. 
   406 @param aMask A bitmask. Any attributes which are set in the mask as passed 
   407 into the function are not sensed from the current layer. On return, indicates 
   408 the attributes which were sensed from this layer. So, normally, when passed 
   409 to the function, all attributes in the mask should be unset. 
   410 @param aMode Controls which attributes are written to aParaFormat. If 
   411 EAllAttributes, all attributes are written; if EFixedAttributes, tabs, bullets 
   412 and borders are not written. */
   413 	{
   414 	if (aParaFormat==NULL)
   415 	    {
   416 	    OstTrace0( TRACE_FATAL, CPARAFORMATLAYER_SENSEL, "ENullFormatPointer" );
   417 	    }
   418 	__ASSERT_ALWAYS(aParaFormat!=NULL,Panic(ENullFormatPointer));
   419 
   420 	iStore.SenseParaFormatL(*aParaFormat,aMask,aMode);
   421 	}
   422 
   423 
   424 void CFormatLayer::CloneLayerL(CFormatLayer* aClone)const
   425 	{
   426 	aClone->iStore.CopyL(iStore);
   427 	}
   428 
   429 
   430 EXPORT_C CFormatLayer* CParaFormatLayer::DoCloneL()const
   431 //
   432 //
   433 	{
   434 	return NewL(this);  // use copy construction
   435 	}
   436 
   437 
   438 
   439 EXPORT_C TBool CParaFormatLayer::IsIdenticalL(const CParaFormat* aParaFormat,const TParaFormatMask& aMask)const
   440 /** Compares a format attribute container with the current layer. For the two 
   441 objects to be identical, the current layer must contain only the attributes 
   442 specified in the argument aMask, and these attributes must have the same values 
   443 as those in aParaFormat. None of the current layer's based-on links are searched.
   444 
   445 @param aParaFormat Contains the attribute values used in the comparison. 
   446 @param aMask A bitmask specifying which attributes are relevant to the function. 
   447 
   448 @return ETrue if the formatting of the current layer exactly matches that 
   449 contained in aParaFormat. Otherwise EFalse. */
   450 	{
   451 	CParaFormat* thisParaFormat=CParaFormat::NewLC();
   452 	TParaFormatMask thisParaFormatMask;
   453 	SenseL(thisParaFormat,thisParaFormatMask);
   454 	TBool result=EFalse;
   455 	if (thisParaFormatMask!=aMask)
   456 		result=EFalse;
   457 	else if (thisParaFormat->IsEqual(*aParaFormat,aMask))
   458 		result=ETrue;
   459 	CleanupStack::PopAndDestroy();
   460 	return result;
   461 	}
   462 
   463 EXPORT_C TBool CParaFormatLayer::IsIdentical(CFormatLayer* aLayer,TBool aCheckBasedOnLink)const
   464 /** Compares another paragraph format layer with the current layer. For the two 
   465 layers to be equal, they must have the same contents and (if the second 
   466 parameter is ETrue), their based-on links must point to the same format layer.
   467 
   468 @param aLayer The paragraph format layer to compare to this format layer. 
   469 @param aCheckBasedOnLink If ETrue, both layers' based-on links must point to 
   470 the same format layer. If EFalse, the based-on links are not used in the 
   471 comparison. By default, ETrue. 
   472 @return ETrue if the two layers are identical, otherwise EFalse. */
   473 	{
   474 	if (aCheckBasedOnLink)
   475 		{
   476 		if (iBasedOn!=aLayer->SenseBase())
   477 			return EFalse;
   478 		}
   479 	TInt size;
   480 	const TUint8* ptr=((CParaFormatLayer*)aLayer)->Ptr(size);		// some design went wrong here!
   481 	return CFormatLayer::IsIdentical(ptr,size);
   482 	}
   483 
   484 EXPORT_C const TUint8* CParaFormatLayer::Ptr(TInt& aSize)const
   485 /** Gets a pointer to the start of the buffer containing the layer's format 
   486 attribute values.
   487 
   488 @param aSize On return, set to the size of the buffer. 
   489 @return Pointer to the buffer which contains the layer's format attribute 
   490 values. */
   491 	{return CFormatLayer::Ptr(aSize);}
   492 
   493 EXPORT_C TUid CParaFormatLayer::Type()const
   494 /** Returns the paragraph format layer UID. This can be used to distinguish 
   495 between an ordinary paragraph format layer and paragraph styles, which have a 
   496 different UID.
   497 
   498 @return The UID of a paragraph format layer (KNormalParagraphStyleUid). */
   499 	{return KNormalParagraphStyleUid;}
   500 
   501 CFormatLayer* CParaFormatLayer::RestoreNewL(RReadStream& aStream)
   502 /** Return a new CParaFormatLayer, having restored it from aStream.
   503  Overrides the base class method, to provide a new format layer of the correct 
   504  type.*/
   505 	{return NewL(aStream);}
   506 
   507 void CParaFormatLayer::FillParaFormatL(CParaFormat* aParaFormat,TParaFormatMask& aMask,CParaFormat::TParaFormatGetMode aMode)const
   508 /** Fills aParaFormat by dumping the current format layer into it.  
   509 Next traverse the 'basedOn' link if it is not NULL, and repeat.*/
   510 	{
   511 	if ( !aParaFormat )
   512 		{
   513 		return;
   514 		}
   515 	
   516 	CParaFormat& senseFormat = *aParaFormat;
   517 
   518 	iStore.SenseParaFormatL(senseFormat,aMask,aMode);
   519 	if (iBasedOn)
   520 		((CParaFormatLayer*)iBasedOn)->FillParaFormatL(aParaFormat,aMask,aMode);
   521 	}
   522 
   523 void CParaFormatLayer::CleanupEffectiveFormat(CParaFormat* aParaFormat,TParaFormatMask aMask)const
   524 /** Removes anti-tabs, zero height and null bullets, and paragraph borders with
   525  null linestyles, from aParaFormat.  An effective format does not support the
   526  notion of anti-tabs etc.*/
   527 	{
   528 	if (aMask.AttribIsSet(EAttBullet))
   529 		{
   530 		if (aParaFormat->iBullet->iStyle == TBullet::ENullStyle || aParaFormat->iBullet->iHeightInTwips <= 0)
   531 			{
   532 			delete aParaFormat->iBullet;
   533 			aParaFormat->iBullet = NULL;
   534 			}
   535 		}
   536 
   537 	if (aMask.AttribIsSet(EAttTopBorder) ||
   538 		aMask.AttribIsSet(EAttBottomBorder) ||
   539 		aMask.AttribIsSet(EAttLeftBorder) ||
   540 		aMask.AttribIsSet(EAttRightBorder))
   541 		CleanupBorders(aParaFormat);
   542 
   543 	if (aMask.AttribIsSet(EAttTabStop))
   544 		{
   545 		int index = 0;
   546 		while (index < aParaFormat->TabCount())
   547 			{
   548 			TTabStop tab = aParaFormat->TabStop(index);
   549 			if (tab.iType == TTabStop::ENullTab)
   550 				aParaFormat->RemoveTab(tab.iTwipsPosition);
   551 			else
   552 				index++;
   553 			}
   554 		}
   555 	}
   556 
   557 
   558 void CParaFormatLayer::CleanupBorders(CParaFormat* aParaFormat)const
   559 /** Destroys the paragraph border if it is of NULL linestyle,
   560  and nulls the pointer to it (aBorder).*/
   561 	{
   562 	if (aParaFormat->ParaBorder(CParaFormat::EParaBorderTop).iLineStyle==TParaBorder::ENullLineStyle &&
   563 		aParaFormat->ParaBorder(CParaFormat::EParaBorderBottom).iLineStyle==TParaBorder::ENullLineStyle &&
   564 		aParaFormat->ParaBorder(CParaFormat::EParaBorderLeft).iLineStyle==TParaBorder::ENullLineStyle &&
   565 		aParaFormat->ParaBorder(CParaFormat::EParaBorderRight).iLineStyle==TParaBorder::ENullLineStyle)
   566 		{
   567 		aParaFormat->RemoveAllBorders();
   568 		}
   569 	}
   570 
   571 
   572 EXPORT_C CCharFormatLayer* CCharFormatLayer::NewL()
   573 /** Allocates and constructs an empty CCharFormatLayer. Its based-on link is 
   574 NULL.
   575 
   576 Note: Use SetL() to set format attributes in the layer. Use SetBase(), defined 
   577 in the base class CFormatLayer, to set the layers based on link.
   578 
   579 @return Pointer to the new character format layer. */
   580 	{return new(ELeave)CCharFormatLayer;}
   581 
   582 
   583 
   584 EXPORT_C CCharFormatLayer* CCharFormatLayer::NewL(RReadStream& aStream)
   585 /** Allocates and constructs a CCharFormatLayer, restoring its format attributes 
   586 from a stream. The layer's based-on link is set to NULL.
   587 
   588 @param aStream Stream from which the layer is restored. 
   589 @return Pointer to the new character format layer. */
   590 	{
   591 	CCharFormatLayer* self=NewL();
   592 	CleanupStack::PushL(self);
   593 	self->InternalizeL(aStream);
   594 	CleanupStack::Pop();
   595 	return self;
   596 	}
   597 
   598 
   599 EXPORT_C CCharFormatLayer* CCharFormatLayer::NewL(const TCharFormat& aFormat,const TCharFormatMask& aMask)
   600 /** Returns a handle to a new charFormatLayer, after constructing
   601  it and setting it with the specified format and format mask.*/
   602 	{
   603 	CCharFormatLayer* self=NewL();
   604 	CleanupStack::PushL(self);
   605 	self->SetL(aFormat,aMask);
   606 	CleanupStack::Pop();
   607 	return self;
   608 	}
   609 
   610 
   611 CCharFormatLayer* CCharFormatLayer::NewL(const TCharFormatX& aFormat,const TCharFormatXMask& aMask)
   612 	{
   613 	CCharFormatLayer* self = NewL();
   614 	CleanupStack::PushL(self);
   615 	self->SetL(aFormat,aMask);
   616 	CleanupStack::Pop();
   617 	return self;
   618 	}
   619 
   620 
   621 CCharFormatLayer* CCharFormatLayer::NewL(const CCharFormatLayer* aLayer)
   622 /** Copying construction
   623  does not copy based-on link.*/
   624 	{
   625 	CCharFormatLayer* self=NewL();  // based-on is NULL
   626 	CleanupStack::PushL(self);
   627 	aLayer->CloneLayerL(self);
   628 	CleanupStack::Pop();			// self
   629 	return self;
   630 	}
   631 
   632 CCharFormatLayer* CCharFormatLayer::NewCopyBaseL(const CCharFormatLayer* aLayer)
   633 /** Copying construction
   634  copies based-on link*/
   635 	{
   636 	CCharFormatLayer* self=NewL(aLayer);  // based-on is NULL
   637 	self->iBasedOn=aLayer->iBasedOn;
   638 	return self;
   639 	}
   640 
   641 
   642 CCharFormatLayer::CCharFormatLayer()
   643 	{
   644 	}
   645 
   646 EXPORT_C void CCharFormatLayer::ExternalizeL(RWriteStream& aStream)const
   647 /** Externalises the character format layer but not its based-on link to a 
   648 write stream. The presence of this function means that the standard templated 
   649 operator<<() (defined in s32strm.h) is available to externalise objects of 
   650 this class.
   651 
   652 @param aStream Stream to which the format layer should be externalised. */
   653 	{
   654 	aStream << iStore;
   655 	}
   656 
   657 EXPORT_C void CCharFormatLayer::InternalizeL(RReadStream& aStream,const CFormatLayer* aBase)
   658 /** Internalises the character format layer but not its based-on link from a 
   659 read stream. The presence of this function means that the standard templated 
   660 operator>>() (defined in s32strm.h) is available to internalise objects of 
   661 this class. The internalised layer is set to be based on the layer specified.
   662 
   663 @param aStream Stream from which the format layer should be internalised. 
   664 @param aBase The based-on link to assign to the layer. By default NULL. */
   665 	{
   666 	aStream >> iStore;
   667 	SetBase(aBase);
   668 	}
   669 
   670 
   671 EXPORT_C const TUint8* CCharFormatLayer::Ptr(TInt& aSize)const
   672 /** Gets a pointer to the start of the buffer containing the layer's format 
   673 attribute values.
   674 
   675 @param aSize On return, set to the size of the buffer. 
   676 @return Pointer to the buffer which contains the layer's format attribute 
   677 values. */
   678 	{
   679 	return CFormatLayer::Ptr(aSize);
   680 	}
   681 
   682 EXPORT_C void CCharFormatLayer::SetL(const TCharFormat& aCharFormat,const TCharFormatMask& aMask)
   683 /** Sets the layer's format attributes. The attributes which are set in the 
   684 mask are set in the layer to the values specified in the format container 
   685 aCharFormat. The attributes which are not set in the mask are not changed.
   686 
   687 @param aCharFormat Contains the attribute values to assign to the format layer. 
   688 @param aMask Mask specifying which attributes should be set from aCharFormat. */
   689 	{
   690 	TCharFormatX format(aCharFormat);
   691 	iStore.SetCharFormatL(format,aMask);
   692 	}
   693 
   694 void CCharFormatLayer::SetL(const TCharFormatX& aCharFormat,const TCharFormatXMask& aMask)
   695 /** Sets the layer's format attributes. The attributes which are set in the 
   696 mask are set in the layer to the values specified in the format container 
   697 aCharFormat. The attributes which are not set in the mask are not changed.
   698 
   699 @param aCharFormat Contains the attribute values to assign to the format layer. 
   700 @param aMask Mask specifying which attributes should be set from aCharFormat. */
   701 	{
   702 	iStore.SetCharFormatL(aCharFormat,aMask);
   703 	}
   704 
   705 EXPORT_C void CCharFormatLayer::SenseEffective(TCharFormat& aCharFormat)const
   706 /** Senses the layer's effective format, searching all based-on links. The 
   707 resulting aCharFormat is fully populated.
   708 
   709 @param aCharFormat On return, contains the layer's effective formatting. */
   710 	{
   711 	TCharFormatX format;
   712 	TCharFormatXMask mask;
   713 	FillCharFormat(format,mask);
   714 	aCharFormat = format.iCharFormat;
   715 	}
   716 
   717 void CCharFormatLayer::SenseEffective(TCharFormatX& aCharFormat) const
   718 /** Senses the layer's effective format, searching all based-on links. The 
   719 resulting aCharFormat is fully populated.
   720 
   721 @param aCharFormat On return, contains the layer's effective formatting. */
   722 	{
   723 	TCharFormatXMask mask;
   724 	aCharFormat = TCharFormatX(); // initialise character format; FillCharFormat doesn't do this
   725 	FillCharFormat(aCharFormat,mask);
   726 	}
   727 
   728 EXPORT_C void CCharFormatLayer::Sense(TCharFormat& aCharFormat,TCharFormatMask& aMask)const
   729 /** Senses the formatting which has been applied to the current layer only. No 
   730 based-on links are searched. This function does not get the layer's effective 
   731 formatting, but the resulting aCharFormat is fully populated, even if not 
   732 all attributes are flagged for sensing in aMask because any attribute values 
   733 not sensed from the current layer are set to default values.
   734 
   735 @param aCharFormat On return, contains the formatting which has been applied 
   736 to the current layer only. Any attributes not explicitly set in the current 
   737 layer are set to the default values for a TCharFormat. Any attributes specified 
   738 in aMask are not sensed from this layer. The values for these attributes are 
   739 also initialised to the default settings. 
   740 @param aMask A bitmask. Any attributes which are set in the mask as passed 
   741 into the function are not sensed from the current layer. On return, indicates 
   742 the attributes which were sensed from this layer. So, normally, when passed 
   743 to the function, all attributes in the mask should be unset. */
   744 	{
   745 	TCharFormatX format(aCharFormat);
   746 	TCharFormatXMask mask = aMask;
   747 	iStore.SenseCharFormat(format,mask);
   748 	aCharFormat = format.iCharFormat;
   749 	mask.ClearExtendedAttribs();
   750 	aMask = mask;
   751 	}
   752 
   753 void CCharFormatLayer::Sense(TCharFormatX& aCharFormat,TCharFormatXMask& aMask) const
   754 /** Senses the formatting which has been applied to the current layer only. No 
   755 based-on links are searched. This function does not get the layer's effective 
   756 formatting, but the resulting aCharFormat is fully populated, even if not 
   757 all attributes are flagged for sensing in aMask because any attribute values 
   758 not sensed from the current layer are set to default values.
   759 
   760 @param aCharFormat On return, contains the formatting which has been applied 
   761 to the current layer only. Any attributes not explicitly set in the current 
   762 layer are set to the default values for a TCharFormat. Any attributes specified 
   763 in aMask are not sensed from this layer. The values for these attributes are 
   764 also initialised to the default settings. 
   765 @param aMask A bitmask. Any attributes which are set in the mask as passed 
   766 into the function are not sensed from the current layer. On return, indicates 
   767 the attributes which were sensed from this layer. So, normally, when passed 
   768 to the function, all attributes in the mask should be unset. */
   769 	{
   770 	iStore.SenseCharFormat(aCharFormat,aMask);
   771 	}
   772 
   773 
   774 EXPORT_C CFormatLayer* CCharFormatLayer::DoCloneL()const
   775 	{
   776 	return NewL(this);  // use copy construction
   777 	}
   778 
   779 
   780 EXPORT_C TBool CCharFormatLayer::IsIdentical(CFormatLayer* aLayer,TBool aCheckBasedOnLink)const
   781 /** Compares another character format layer with the current layer. For the two 
   782 layers to be equal, they must have the same contents and (if the second 
   783 parameter is ETrue), their based-on links must point to the same format layer.
   784 
   785 @param aLayer The character format layer to compare to this format layer. 
   786 @param aCheckBasedOnLink If ETrue, both layers' based-on links must point to 
   787 the same format layer. If EFalse, the based-on links are not used in the 
   788 comparison. By default, ETrue. 
   789 @return ETrue if the two layers are identical, otherwise EFalse. */
   790 	{
   791 	if (aCheckBasedOnLink)
   792 		{
   793 		if (iBasedOn!=aLayer->SenseBase())
   794 			return EFalse;
   795 		}
   796 	TInt size;
   797 	const TUint8* ptr=((CCharFormatLayer*)aLayer)->Ptr(size);		// some naff design at work here!
   798 	return CFormatLayer::IsIdentical(ptr,size);
   799 	}
   800 
   801 EXPORT_C TBool CCharFormatLayer::IsIdentical(const TCharFormat& aCharFormat,const TCharFormatMask& aMask)const
   802 /** Compares a format attribute container with the current layer. For the two 
   803 objects to be identical, the current layer must contain only the attributes 
   804 specified in the argument aMask, and these attributes must have the same 
   805 values as those in aCharFormat. None of the current layer's based-on links are 
   806 searched.
   807 
   808 @param aCharFormat Contains the attribute values used in the comparison. 
   809 @param aMask A bitmask specifying which attributes are relevant to the function. 
   810 
   811 @return ETrue if the formatting of the current layer exactly matches that 
   812 contained in aCharFormat, otherwise EFalse. */
   813 	{
   814 	TCharFormat thisCharFormat;
   815 	TCharFormatMask thisCharFormatMask;
   816 	Sense(thisCharFormat,thisCharFormatMask);
   817 	if (thisCharFormatMask!=aMask)
   818 		return EFalse;
   819 	return thisCharFormat.IsEqual(aCharFormat,aMask);
   820 	}
   821 
   822 
   823 CFormatLayer* CCharFormatLayer::RestoreNewL(RReadStream& aStream)
   824 /** Return a new CCharFormatLayer, having restored it from aStream.
   825 Overrides the base class method, to provide a new format layer of 
   826 the correct type.*/
   827 	{return CCharFormatLayer::NewL(aStream);}
   828 
   829 
   830 void CCharFormatLayer::FillCharFormat(TCharFormatX& aCharFormat,TCharFormatXMask& aMask)const
   831 /** Fills aCharFormat by dumping the current format layer into it, then follows 
   832 each 'BasedOn' link in turn if it is not null.  It is assumed that all based on 
   833 links eventually terminate with nulls.*/
   834 	{
   835 	iStore.SenseCharFormat(aCharFormat,aMask);
   836 	if (iBasedOn)
   837 		((CCharFormatLayer*)iBasedOn)->FillCharFormat(aCharFormat,aMask);
   838 	}