os/textandloc/textrendering/texthandling/stext/TXTSTYLE.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
     3 * All rights reserved.
     4 * This component and the accompanying materials are made available
     5 * under the terms of "Eclipse Public License v1.0"
     6 * which accompanies this distribution, and is available
     7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
     8 *
     9 * Initial Contributors:
    10 * Nokia Corporation - initial contribution.
    11 *
    12 * Contributors:
    13 *
    14 * Description: 
    15 *
    16 */
    17 
    18 
    19 #include <e32std.h>
    20 #include <e32base.h>
    21 #include <s32strm.h>
    22 
    23 #include "TXTSTYLE.H"
    24 #include "TXTFRMAT.H"
    25 #include "TXTSTD.H"
    26 
    27 #include "OstTraceDefinitions.h"
    28 #ifdef OST_TRACE_COMPILER_IN_USE
    29 #include "TXTSTYLETraces.h"
    30 #endif
    31 
    32 #ifdef SYMBIAN_ENABLE_SPLIT_HEADERS
    33 #include "TXTFMLYR_INTERNAL.H"
    34 #include "TXTSTYLE_INTERNAL.H"
    35 #endif
    36 
    37 // Return the handle of a new paragraph style
    38  
    39 
    40 EXPORT_C CParagraphStyle* CParagraphStyle::NewL(const CParaFormatLayer& aGlobalParaFormatLayer,
    41 												const CCharFormatLayer& aGlobalCharFormatLayer)
    42 /** Allocates and constructs a CParagraphStyle object whose formatting is based 
    43 on a global paragraph and character format layer. The type UID is initialised 
    44 to KUserDefinedParagraphStyleUid. The outline level is not initialised.
    45 
    46 @param aGlobalParaFormatLayer The paragraph format layer on which the style's 
    47 paragraph formatting is based. 
    48 @param aGlobalCharFormatLayer The character format layer on which the style's 
    49 character formatting is based. 
    50 @return Pointer to the new CParagraphStyle object. */
    51 	{
    52 	CParagraphStyle* self=new(ELeave) CParagraphStyle();
    53 	CleanupStack::PushL(self);
    54 	self->ConstructL(aGlobalParaFormatLayer,aGlobalCharFormatLayer);
    55 	CleanupStack::Pop();
    56 	return self;
    57 	}
    58 
    59 
    60 CParagraphStyle::CParagraphStyle():
    61 	iType(KUserDefinedParagraphStyleUid)
    62 	{
    63 	}
    64 
    65 
    66 // Create style and base it on the specified 'Normal' layers
    67 void CParagraphStyle::ConstructL(const CParaFormatLayer& aGlobalParaFormatLayer,
    68 										  const CCharFormatLayer& aGlobalCharFormatLayer)
    69 	{
    70 	SetBase(&aGlobalParaFormatLayer);
    71 	//
    72 	// Force para format layer storage allocation with empty layer
    73 	TParaFormatMask paraMask;
    74 	SetL((CParaFormat*)NULL,paraMask);
    75 	//
    76 	// Force char format layer storage allocation with empty layer;
    77 /*	TCharFormat format;
    78 	TCharFormatMask mask;
    79 	iCharFormatLayer=CCharFormatLayer::NewL(format,mask);
    80 */
    81 	iCharFormatLayer=CCharFormatLayer::NewL();
    82 	iCharFormatLayer->SetBase(&aGlobalCharFormatLayer);
    83 	}
    84  
    85 
    86 EXPORT_C CParagraphStyle::~CParagraphStyle()
    87 /** The destructor frees all resources owned by the object, prior to its destruction. */
    88 	{
    89 	delete iCharFormatLayer;
    90 	}
    91 
    92 
    93 EXPORT_C CFormatLayer* CParagraphStyle::DoCloneL()const
    94 	{
    95 	const CCharFormatLayer* charLayer=STATIC_CAST(const CCharFormatLayer*,iCharFormatLayer->SenseBase());
    96 	CParagraphStyle* clone=CParagraphStyle::NewL(STATIC_CAST(const CParaFormatLayer&,*SenseBase()),*charLayer);
    97 	CleanupStack::PushL(clone);
    98 	CloneLayerL(clone);  // clones the paragraph format layer
    99 	delete clone->iCharFormatLayer;
   100 	clone->iCharFormatLayer=NULL;  // necessary precaution if next instruction leaves
   101 	clone->iCharFormatLayer=iCharFormatLayer->CloneL();  // clones the character format layer
   102 	clone->iCharFormatLayer->SetBase(charLayer);  // restore the based on link
   103 	CleanupStack::Pop();  // clone
   104 	//
   105 	clone->SetType(Type());
   106 	clone->SetOutlineLevel(OutlineLevel());
   107 	clone->iName=iName;
   108 	//
   109 	return clone;
   110 	}
   111 
   112 
   113  
   114 
   115 EXPORT_C TUid CParagraphStyle::Type()const
   116 /** Gets the style's type UID.
   117 
   118 @return The style's type UID. */
   119 	{return iType;}
   120 
   121 
   122 CParagraphStyle* CParagraphStyle::NewL(RReadStream& aStream,
   123 										const CParaFormatLayer& aGlobalParaFormatLayer,
   124 										const CCharFormatLayer& aGlobalCharFormatLayer)
   125 	{
   126 	CParagraphStyle* self=new(ELeave) CParagraphStyle();
   127 	CleanupStack::PushL(self);
   128 	self->ConstructL(aGlobalParaFormatLayer,aGlobalCharFormatLayer);
   129 	self->InternalizeL(aStream,&aGlobalParaFormatLayer,&aGlobalCharFormatLayer);
   130 	CleanupStack::Pop();
   131 	return self;
   132 	}
   133 
   134 
   135 // Load into this style from the specified readStream.  Base this style on the layer aBase - Default NULL
   136 void CParagraphStyle::InternalizeL(RReadStream& aStream,const CFormatLayer* aParaLayerBase,const CFormatLayer* aCharLayerBase)
   137 	{
   138 	aStream>> iName;
   139 	aStream>> iType;
   140 	iOutlineLevel=aStream.ReadInt32L();
   141 	iCharFormatLayer->InternalizeL(aStream,aCharLayerBase);
   142 	//
   143 	CParaFormatLayer::InternalizeL(aStream,aParaLayerBase);
   144 	}
   145 
   146 
   147 // Save the style in the specified WriteStream. The based on link is *NOT* stored.
   148 void CParagraphStyle::ExternalizeL(RWriteStream& aStream)const
   149 	{
   150 	aStream<< iName;
   151 	aStream<< iType;
   152 	aStream.WriteInt32L(iOutlineLevel);
   153 	aStream<< *iCharFormatLayer;
   154 	//
   155 	CParaFormatLayer::ExternalizeL(aStream);
   156 	}
   157 
   158 
   159 DLLEXPORT_C void CStyleList::__DbgTestInvariant()const
   160 // Provides class invariants.  Explanations below:
   161 //
   162 	{
   163 #ifdef _DEBUG
   164 	TInt styleCount=Count();
   165 	for (TInt nn=0;nn<styleCount;nn++)
   166 		{
   167 		RParagraphStyleInfo info=iList->At(nn);
   168 		if (info.iStyleForNextPara)
   169 			{
   170 			TInt index=IndexByPtr(info.iStyleForNextPara);
   171 			// ASSERT: The style-to-follow is present in the list.
   172 			if (!((info.iStyleForNextPara==NULL) ||
   173 	                ((info.iStyleForNextPara) && (index!=KErrNotFound && index>=0 && index<styleCount))))
   174 			    {
   175 			    OstTrace0( TRACE_DUMP, CSTYLELIST_DBGTESTINVARIANT, "Invariant" );
   176 			    }
   177 			__ASSERT_DEBUG(
   178 				(info.iStyleForNextPara==NULL) ||
   179 				((info.iStyleForNextPara) && (index!=KErrNotFound && index>=0 && index<styleCount)),User::Invariant());
   180 			}
   181 		}
   182 #endif
   183 	}
   184 
   185 
   186  
   187 
   188 EXPORT_C CStyleList* CStyleList::NewL(TInt aCapacity /*=KMaxStyleListGranularity*/)
   189 /** Allocates and constructs an empty CStyleList object with an array granularity.
   190 
   191 @param aCapacity The number of entries by which the array of styles expands 
   192 when its buffer is reallocated. By default, KMaxStyleListGranularity (= 4). 
   193 Must be positive or a panic occurs. 
   194 @return Pointer to the new style list. */
   195 	{
   196 	CStyleList* self=new(ELeave) CStyleList();
   197 	CleanupStack::PushL(self);
   198 	self->ConstructL(aCapacity);
   199 	CleanupStack::Pop();
   200 	return self;
   201 	}
   202 
   203 
   204 EXPORT_C CStyleList* CStyleList::NewL(RReadStream& aStream,
   205 									  const CParaFormatLayer* aGlobalParaFormatLayer,
   206 									  const CCharFormatLayer* aGlobalCharFormatLayer)
   207 /** Allocates and constructs a CStyleList, restoring its contents from the specified 
   208 stream store. Each style in the list is set to be based on the global format 
   209 layers specified.
   210 
   211 @param aStream Stream store from which the style list is restored. 
   212 @param aGlobalParaFormatLayer Pointer to the global paragraph format layer 
   213 on which all styles in the list are based. 
   214 @param aGlobalCharFormatLayer Pointer to the global character format layer 
   215 on which all styles in the list are based. 
   216 @return Pointer to the new style list. */
   217 	{
   218 	CStyleList* self=new(ELeave) CStyleList();
   219 	CleanupStack::PushL(self);
   220 	self->InternalizeL(aStream,aGlobalParaFormatLayer,aGlobalCharFormatLayer);
   221 	CleanupStack::Pop();
   222 	return self;
   223 	}
   224 
   225 
   226 EXPORT_C CStyleList::CStyleList()
   227 	{
   228 	}
   229 
   230 
   231 EXPORT_C void CStyleList::ConstructL(TInt aCount)
   232 // Allocate the style list.
   233 //
   234 	{
   235 	iList=new(ELeave) CArrayFixFlat<RParagraphStyleInfo>(aCount);
   236 	}
   237 
   238 
   239  
   240 
   241 EXPORT_C CStyleList::~CStyleList()
   242 /** Deletes all the entries in the list and the list itself. */
   243 	{
   244 	KillStyleList();
   245 	}
   246 
   247 
   248  
   249 
   250 EXPORT_C const RParagraphStyleInfo& CStyleList::At(TInt aIndex)const
   251 /** Gets a style from the style list, from its index into the array. Two versions 
   252 are supplied. The compiler chooses the appropriate version based on the use 
   253 made of the returned reference. If it is used in an expression where that 
   254 reference can be modified, then the non-const version is chosen.
   255 
   256 @param aIndex The index of the style into the list. The first style is at 
   257 position zero. Must be within the bounds of the array, or a panic occurs. 
   258 @return A const reference to the style at position aIndex in the array. 
   259 @return A non-const reference to the style at position aIndex in the array. */
   260 	{
   261 	if (!iList)
   262 	    {
   263 	    OstTrace0( TRACE_FATAL, CSTYLELIST_AT, "EStyleIntegrityError" );
   264 	    }
   265 	__ASSERT_ALWAYS(iList,Panic(EStyleIntegrityError));
   266 	if (aIndex<0 || aIndex>=iList->Count())
   267 	    {
   268 	    OstTrace0( TRACE_DUMP, DUP3_CSTYLELIST_AT, "EStyleIndexOutOfRange" );
   269 	    }
   270 	__ASSERT_DEBUG(aIndex>=0 && aIndex<iList->Count(),Panic(EStyleIndexOutOfRange));
   271 	
   272 	return (*iList)[aIndex];
   273 	}
   274 
   275 
   276 EXPORT_C RParagraphStyleInfo& CStyleList::At(TInt aIndex)
   277 	{
   278 	if (!iList)
   279 	    {
   280 	    OstTrace0( TRACE_FATAL, DUP2_CSTYLELIST_AT, "EStyleIntegrityError" );
   281 	    }
   282 	__ASSERT_ALWAYS(iList,Panic(EStyleIntegrityError));
   283 	if (aIndex<0 || aIndex>=iList->Count())
   284 	    {
   285 	    OstTrace0( TRACE_DUMP, DUP1_CSTYLELIST_AT, "EStyleIndexOutOfRange" );
   286 	    }
   287 	__ASSERT_DEBUG(aIndex>=0 && aIndex<iList->Count(),Panic(EStyleIndexOutOfRange));
   288 
   289 	return (*iList)[aIndex];
   290 	}
   291 
   292 
   293  
   294 EXPORT_C void CStyleList::Reset()
   295  /** Deletes the contents of the list. */
   296 	{
   297 	__TEST_INVARIANT;
   298 	
   299 	TInt styleCount=Count();
   300 	for (TInt nn=0;nn<styleCount;nn++)
   301 		delete (iList->At(nn)).iStyle;
   302 	if (iList)
   303 		iList->Reset();
   304 
   305 	__TEST_INVARIANT;
   306 	}
   307 
   308 
   309 void CStyleList::KillStyleList()
   310 	{
   311 	Reset();
   312 	delete iList;
   313 	}
   314 
   315 
   316 
   317 EXPORT_C TStreamId CStyleList::StoreL(CStreamStore& aStore)const
   318 /** Stores the style list to a stream store.
   319 
   320 @param aStore Stream store to which the style list is written. 
   321 @return The ID of the stream store. */
   322 	{
   323 	__TEST_INVARIANT;
   324 
   325 	RStoreWriteStream stream;
   326 	TStreamId id=stream.CreateLC(aStore);
   327 	//
   328 	stream<< *this;
   329 	//
   330 	stream.CommitL();
   331 	CleanupStack::PopAndDestroy();
   332 	return id;
   333 	}
   334 
   335 
   336 
   337 
   338 EXPORT_C void CStyleList::ExternalizeL(RWriteStream& aStream)const
   339 /** Externalises the style list to a write stream. The presence of this function 
   340 means that the standard templated operator<<() (defined in s32strm.h) is available 
   341 to externalise objects of this class. Does not externalise any styles' based-on 
   342 links.
   343 
   344 @param aStream Stream to which the object should be externalised. */
   345 	{
   346 	__TEST_INVARIANT;
   347 	
   348 	TInt count=Count();
   349 	aStream.WriteUint8L(count);
   350 	for (TInt mm=0;mm<count;mm++)
   351 		{
   352 		CParagraphStyle* style=(iList->At(mm)).iStyle;
   353 		style->ExternalizeL(aStream);
   354 		}
   355 	for (TInt nn=0;nn<count;nn++)
   356 		{
   357 		RParagraphStyleInfo set=iList->At(nn);
   358 		CParagraphStyle* style=set.iStyle;
   359 		TInt offset=(set.iStyleForNextPara!=NULL)
   360 			? IndexByPtr(style)
   361 			: -1;
   362 			if (!(offset==-1 || (offset>=0 && offset<Count())))
   363 			    {
   364 			    OstTrace0( TRACE_DUMP, CSTYLELIST_EXTERNALIZEL, "EStyleIntegrityError" );
   365 			    }
   366 		__ASSERT_DEBUG(offset==-1 || (offset>=0 && offset<Count()),Panic(EStyleIntegrityError));
   367 		aStream.WriteInt8L(offset);
   368 		}
   369 	}
   370 
   371 
   372 
   373 EXPORT_C void CStyleList::InternalizeL(RReadStream& aStream,
   374 									   const CParaFormatLayer* aGlobalParaFormatLayer,
   375 									   const CCharFormatLayer* aGlobalCharFormatLayer)
   376 /** Internalises the style list from a read stream. The presence of this function 
   377 means that the standard templated operator>>() (defined in s32strm.h) is available 
   378 to internalise objects of this class. Any existing style list contents are 
   379 replaced.
   380 
   381 @param aStream Stream store from which the style list is internalised. 
   382 @param aGlobalParaFormatLayer Pointer to the global paragraph format layer 
   383 on which all styles in the list are based. 
   384 @param aGlobalCharFormatLayer Pointer to the global character format layer 
   385 on which all styles in the list are based. */
   386 	{
   387 	KillStyleList();
   388 	TInt styleCount=aStream.ReadUint8L();
   389 	ConstructL(Max(1,styleCount));  // panics if granularity==0
   390 	RParagraphStyleInfo holdingSet;
   391 	iList->AppendL(holdingSet,styleCount);
   392 	for (TInt mm=0;mm<styleCount;mm++)
   393 		{// restore the paragraph styles
   394 		CParagraphStyle* style=CParagraphStyle::NewL(aStream,*aGlobalParaFormatLayer,*aGlobalCharFormatLayer);
   395 		iList->At(mm).iStyle=style;
   396 		}
   397 	for (TInt nn=0;nn<styleCount;nn++)
   398 		{// restore the "style for next paragraph" for each paragraph
   399 		TInt offset=aStream.ReadInt8L();
   400 		iList->At(nn).iStyleForNextPara=(offset==-1)
   401 			? NULL
   402 			: iList->At(offset).iStyle;
   403 		}
   404 
   405 	__TEST_INVARIANT;
   406 	}
   407 
   408 
   409 
   410 EXPORT_C TInt CStyleList::AppendL(RParagraphStyleInfo* aStyleSet)
   411 /** Appends a style to the style list. The list takes ownership of the style.
   412 
   413 @param aStyleSet The style (and optional style for the following paragraph) 
   414 to append to the list. 
   415 @return KErrNone if successful, or KErrAlreadyExists if the style is already 
   416 present in the list. */
   417 	{
   418 	__TEST_INVARIANT;
   419 	
   420 	TInt count=Count();
   421 	for (TInt ii=0;ii<count;ii++)
   422 		{
   423 		if (aStyleSet->iStyle==iList->At(ii).iStyle)
   424 			return KErrAlreadyExists;  // we already own this style
   425 		}
   426 	CleanupStack::PushL(aStyleSet->iStyle);
   427 	iList->AppendL(*aStyleSet);  // if the append fails, we must take responsibility for the style
   428 	CleanupStack::Pop();  // aStyleSet.iStyle
   429 
   430 	__TEST_INVARIANT;
   431 	return KErrNone;
   432 	}
   433 
   434 
   435 EXPORT_C void CStyleList::Remove(CParagraphStyle* aStyle)
   436 /** Removes a style from the style list. If the style is owned by the list, it 
   437 is deleted. If the style is not owned by the list, but is referenced by a 
   438 style owned by the list, (i.e. a style in the list is based on it, or references 
   439 it as its iStyleForNextPara pointer) then the pointer to aStyle is set to 
   440 NULL.
   441 
   442 @param aStyle Pointer to the style to remove from the style list, or to set 
   443 to NULL. */
   444 	{
   445 	__TEST_INVARIANT;
   446 
   447 	TInt styles = Count();
   448 	TInt index = -1;
   449 	for (TInt i = 0; i < styles; i++)
   450 		{
   451 		RParagraphStyleInfo& cur_style = (*iList)[i];
   452 		if (cur_style.iStyleForNextPara == aStyle)
   453 			cur_style.iStyleForNextPara = NULL;
   454 		if (cur_style.iStyle->iBasedOn == aStyle)
   455 			cur_style.iStyle->iBasedOn = NULL;
   456 		if (cur_style.iStyle == aStyle)
   457 			{
   458 			// Assert that the style must occur only once in the style list.
   459 			if (index != -1)
   460 			    {
   461 			    OstTrace0( TRACE_DUMP, CSTYLELIST_REMOVE, "EStyleIntegrityError" );
   462 			    }
   463 			__ASSERT_DEBUG(index == -1,Panic(EStyleIntegrityError));
   464 			index = i;
   465 			}
   466 		}
   467 
   468 	if (index != -1)
   469 		{
   470 		delete aStyle;
   471 		iList->Delete(index);
   472 		}
   473 	
   474 	__TEST_INVARIANT;
   475 	}
   476 
   477 
   478 
   479 EXPORT_C TInt CStyleList::SetStyleToFollow(const RParagraphStyleInfo& aStyleSet)
   480 /** Sets the style to use for the following paragraph for a style in the style 
   481 list. Both the style (aStyleSet.iStyle) and the style to set for the following 
   482 paragraph (aStyleSet.iStyleForNextPara) must exist in the style list.
   483 
   484 The function first locates the style (aStyleSet.iStyle) in the list, then 
   485 sets the style to use for the next paragraph to aStyleSet.iStyleForNextPara.
   486 
   487 If aStyleSet.iStyle does not exist in the style list, the function returns 
   488 with KErrNotFound. If aStyleSet.iStyleForNextPara does not exist in the style 
   489 list, a panic occurs.
   490 
   491 @param aStyleSet Identifies a style in the list, and a style to use for its 
   492 following paragraph. 
   493 @return KErrNone if successful, or KErrNotFound if the first style contained 
   494 in the argument (aStyleSet.iStyle) is not in the style list. */
   495 	{
   496 	if (IndexByPtr(aStyleSet.iStyleForNextPara) == KErrNotFound)
   497 	    {
   498 	    OstTrace0( TRACE_FATAL, CSTYLELIST_SETSTYLETOFOLLOW, "EStyleIntegrityError" );
   499 	    }
   500 	__ASSERT_ALWAYS( IndexByPtr(aStyleSet.iStyleForNextPara) != KErrNotFound, Panic(EStyleIntegrityError) );
   501 	__TEST_INVARIANT;
   502 	
   503 	TInt index = IndexByPtr(aStyleSet.iStyle);
   504 	if (index == KErrNotFound)
   505 		return index;
   506 	
   507 	(*iList)[index].iStyleForNextPara = aStyleSet.iStyleForNextPara;
   508 	
   509 	__TEST_INVARIANT;
   510 	
   511 	return KErrNone;
   512 
   513 	}
   514 	
   515 
   516 
   517 
   518 EXPORT_C RParagraphStyleInfo* CStyleList::PtrByName(const TParagraphStyleName& aName) const
   519 /** Gets the style with the specified name from the style list.
   520 
   521 @param aName The name of the style to retrieve. 
   522 @return Pointer to the style with the name specified. NULL if there is no style 
   523 with this name in the list. */
   524 	{
   525 	// Return a pointer to the named style if it's in the list, or null if not.
   526 	__TEST_INVARIANT;
   527 	
   528 	int count = Count();
   529 	for (int i = 0; i < count; i++)
   530 		{
   531 		RParagraphStyleInfo& style_info = (*iList)[i];
   532 		if (style_info.iStyle != NULL && style_info.iStyle->iName == aName)
   533 			return &style_info;
   534 		}
   535 	return NULL;
   536 	}
   537 
   538 
   539 
   540 
   541 EXPORT_C RParagraphStyleInfo* CStyleList::PtrByType(const TUid aType) const
   542 /** Gets the style with the specified type UID from the style list.
   543 
   544 @param aType The UID of the style to retrieve. 
   545 @return Pointer to the style with the type UID specified. NULL if there is 
   546 no style with this type in the list. */
   547 	{
   548 	// Return a pointer to the first style with the specified type if any; or null if there are none with this type.
   549 	__TEST_INVARIANT;
   550 
   551 	int count = Count();
   552 	for (int i = 0; i < count; i++)
   553 		{
   554 		RParagraphStyleInfo& style_info = (*iList)[i];
   555 		if (style_info.iStyle != NULL && style_info.iStyle->Type() == aType)
   556 			return &style_info;
   557 		}
   558 	return NULL;
   559 	}
   560 
   561 
   562 
   563 
   564 EXPORT_C TInt CStyleList::IndexByPtr(const CParaFormatLayer* aPtr)const
   565 /** Gets the index into the style list of a specified paragraph style.
   566 
   567 @param aPtr Pointer to the style. 
   568 @return The style's index into the style list. KErrNotFound if the style is 
   569 not found in the style list, or if aPtr is a paragraph format layer rather 
   570 than a style. */
   571 	{
   572 	if (aPtr->Type()==KNormalParagraphStyleUid)
   573 		return KErrNotFound;
   574 	TInt count=Count();
   575 	if (count==0)
   576 		return KErrNotFound;  // ptr cannot be matched cos list is empty.
   577 	TInt index=0;
   578 	CParagraphStyle* style=NULL;
   579 	while (index<count)
   580 		{
   581 		style=(*iList)[index].iStyle;
   582 		if (style==aPtr)
   583 			break;
   584 		index++;
   585 		}
   586 	return (index<count)
   587 		? index
   588 		: KErrNotFound;
   589 	}
   590 
   591 
   592 
   593 
   594 EXPORT_C TInt CStyleList::IndexByName(const TDesC& aName)const
   595 /** Gets the index into the style list of a specified paragraph style, identified 
   596 by its name.
   597 
   598 @param aName The name of the style. 
   599 @return The style's index into the style list. KErrNotFound if the style name 
   600 is not found in the style list. */
   601 	{
   602 	TInt count=Count();
   603 	if (count==0)
   604 		return KErrNotFound;  // name cannot be in list cos its empty.
   605 	TInt index=0;
   606 	CParagraphStyle* style=NULL;
   607 	while (index<count)
   608 		{
   609 		style=(*iList)[index].iStyle;
   610 		if (style != NULL && style->iName == aName)
   611 			break;
   612 		index++;
   613 		}
   614 	return (index<count)
   615 		? index
   616 		: KErrNotFound;
   617 	}
   618 
   619 
   620 
   621 EXPORT_C CStyleList* CStyleList::DeepCloneL() const
   622 /** Creates and returns a style list which is a clone of the current style list.
   623 
   624 @return Pointer to a clone of the current style list. */
   625 	{
   626 	__TEST_INVARIANT;
   627 	
   628 	CStyleList* newList = CStyleList::NewL();
   629 	CleanupStack::PushL(newList);
   630 
   631 	TInt styleCount=Count();
   632 	for (TInt styleItem=0;styleItem<styleCount;styleItem++)
   633 		{
   634 		const RParagraphStyleInfo& source=iList->At(styleItem);
   635 		CParagraphStyle* style=source.iStyle->CloneL();
   636 		RParagraphStyleInfo info(style,source.iStyleForNextPara);
   637 		newList->AppendL(&info);
   638 		}
   639 
   640 	CleanupStack::Pop();
   641 	return newList;
   642 	}