os/textandloc/textrendering/texthandling/ttext/T_RICH1A.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-2010 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 <txtrich.h>
    20 #include <gdi.h>
    21 #include <conpics.h>
    22 #include <s32stor.h>
    23 #include "../incp/T_PMLPAR.H"
    24 #include "T_RICH1A.h"
    25 
    26 LOCAL_D CTestStep *pTestStep = NULL;
    27 #define test(cond)											\
    28 	{														\
    29 	TBool __bb = (cond);									\
    30 	pTestStep->TEST(__bb);									\
    31 	if (!__bb)												\
    32 		{													\
    33 		pTestStep->ERR_PRINTF1(_L("ERROR: Test Failed"));	\
    34 		User::Leave(1);										\
    35 		}													\
    36 	}
    37 #undef INFO_PRINTF1
    38 #undef INFO_PRINTF2
    39 // copy from tefexportconst.h
    40 #define INFO_PRINTF1(p1)        pTestStep->Logger().LogExtra(((TText8*)__FILE__), __LINE__, ESevrInfo, (p1))
    41 #define INFO_PRINTF2(p1, p2)    pTestStep->Logger().LogExtra(((TText8*)__FILE__), __LINE__, ESevrInfo, (p1), (p2))
    42 
    43 
    44 #define UNUSED_VAR(a) a = a
    45 
    46 LOCAL_D CTrapCleanup* TheTrapCleanup;
    47 LOCAL_D const TInt KTestCleanupStack=0x200;
    48 
    49 class MAtomicTest
    50 	{
    51 public:
    52 	/** Test that the object is correctly set at stage aStage, then advance it
    53 	to stage aStage+1 with an atomic (with respect to leaving) operation.
    54 
    55 	The test at each stage should test that the previous stage really happened
    56 	and that no part of the new stage has happened.
    57 
    58 	RunAtomicTest will test that each of these operations really is atomic.
    59 	@param aStage The stage to test this object is at.
    60 	@return
    61 		ETrue if this object was successfully advanced to the next stage.
    62 		EFalse if there were no more stages to be advanced to.
    63 	*/
    64 	virtual TBool TestAndRunStageL(TInt aStage) = 0;
    65 	/** Tests that the leave reported in aError is suitable as an error code
    66 	from aStage. Panics if not.
    67 
    68 	@param aError Code that TestAndRunStageL(aStage) left with.
    69 	@param aStage Stage at which the leave occurred. */
    70 	virtual void TestErrorCondition(TInt aStage, TInt aError) = 0;
    71 	void RunAtomicTest();
    72 	};
    73 
    74 /** Runs TestAndRunStage repeatedly with different out-of-memory test
    75 conditions.
    76 
    77 It checks that the operations defined at each stage either leave and leave this
    78 object in the same condition, or return successfully and leave the object in
    79 the next stage. */
    80 void MAtomicTest::RunAtomicTest()
    81 	{
    82 	TInt stage = 0;
    83 	TInt failRate = 1;
    84 	TBool more = ETrue;
    85 	while (more)
    86 		{
    87 		__UHEAP_SETFAIL(RHeap::EDeterministic, failRate);
    88 		TRAPD(err, more = TestAndRunStageL(stage));
    89 		if (err == KErrNone)
    90 			{
    91 			failRate = 1;
    92 			++stage;
    93 			}
    94 		else
    95 			{
    96 			TestErrorCondition(stage, err);
    97 			++failRate;
    98 			}
    99 		}
   100 	__UHEAP_RESET;
   101 	}
   102 
   103 template<class S,class T,CEditableText::TDocumentStorage D>
   104 class CRichTest : public CRichText, public MAtomicTest
   105 	{
   106 public:
   107 	static CRichTest* NewL(CParaFormatLayer* aPara,CCharFormatLayer* aChar);
   108 
   109 	void DoFlatTests();
   110 	void RegisterMethodsL();
   111 	void InsertWithDelimsL();
   112 	void TestResetL();
   113 	void TestDelete1L();
   114 	void TestDelete2L();
   115     void TestApplyRemoveCharFormat();
   116 	void TestRemSpecParaFmtL();
   117 	void DoPML();
   118 	void TestForDefectTET5DHEWWL();
   119 	void TestForDefectINC010183L();
   120 	void TestForDefectINC064162L();
   121 	void TestForDefectINC109323L();
   122 	// Tests SetInsertCharFormat and friends for atomicity.
   123 	TBool TestAndRunStageL(TInt aStage);
   124 	void TestErrorCondition(TInt aStage, TInt aError);
   125 protected:
   126 	CRichTest(CParaFormatLayer* aPara,CCharFormatLayer* aChar);
   127 	};
   128 
   129 template<class S, class T,CEditableText::TDocumentStorage D>
   130 CRichTest<S,T,D>* CRichTest<S,T,D>::NewL(CParaFormatLayer* aPara,CCharFormatLayer* aChar)
   131 	{
   132 	CRichTest<S,T,D>* tmp=new(ELeave)CRichTest<S,T,D>(aPara,aChar);
   133 	tmp->ConstructL(D,EDefaultTextGranularity,EMultiPara);
   134 	return tmp;
   135 	}
   136 
   137 
   138 template<class S, class T,CEditableText::TDocumentStorage D>
   139 CRichTest<S,T,D>::CRichTest(CParaFormatLayer* aPara,CCharFormatLayer* aChar)
   140 	:CRichText(aPara,aChar)
   141 	{}
   142 
   143 
   144 template<class S, class T,CEditableText::TDocumentStorage D>
   145 void CRichTest<S,T,D>::DoFlatTests()
   146 	{
   147 	INFO_PRINTF1(_L("Registering all methods"));
   148 	RegisterMethodsL();
   149 	INFO_PRINTF1(_L("Inserting with embedded paragraph delimiters"));
   150 	InsertWithDelimsL();
   151 	
   152 	}
   153 
   154 
   155 template<class S, class T,CEditableText::TDocumentStorage D>
   156 void CRichTest<S,T,D>::InsertWithDelimsL()
   157 	{
   158 	INFO_PRINTF1(_L("Inserting into shared para"));
   159 
   160 	Reset();
   161 	TBuf<512> testbuf(_L("a"));
   162 	for (TInt ii=0;ii<8;ii++)
   163 		{
   164 		testbuf.Append('a');
   165 		testbuf.Append(CEditableText::EParagraphDelimiter);
   166 		}
   167 	InsertL(0,testbuf);
   168 	test(DocumentLength()==17);
   169 	test(ParagraphCount()==9);
   170 
   171 	Reset();
   172 	TBufC<512> bufC;
   173 	InsertL(0,bufC);
   174 		
   175 	Reset();
   176 	TBuf<512> buf(_L("Herewith"));
   177 	InsertL(0,buf);
   178 
   179 	InsertL(4,CEditableText::EParagraphDelimiter);
   180 	// THE ABOVE IS TEMPORARY ONLY _ REMOVE IT AS SOON AS YOUVE DONE.
   181 	
   182 	Reset();
   183 //	TBuf<512> buf(_L("Herewith"));
   184 	buf.Append(EParagraphDelimiter);
   185 	InsertL(0,buf);
   186 
   187 	buf.Append(_L("Is para one"));
   188 	buf.Append(EParagraphDelimiter);
   189 	InsertL(4,buf);
   190 
   191 	buf.Append(_L(" trailing text"));
   192 	InsertL(DocumentLength()-1,buf);
   193 	//////////////////////////////////
   194 	// Pathalogical case (1)
   195 	// Inserting text with delimiters between 2 adjacent pictures.
   196 	INFO_PRINTF1(_L("Inserting text with delimiters between 2 adjacent pictures."));
   197 	//
   198 	Reset();
   199 	CXzePicture* picA=CXzePicture::NewL('Z');
   200 	TPictureHeader hdrA;
   201 	hdrA.iPictureType=KUidXzePictureType;
   202 	hdrA.iPicture=picA;
   203 	//
   204 	CXzePicture* picB=CXzePicture::NewL('X');
   205 	TPictureHeader hdrB;
   206 	hdrB.iPictureType=KUidXzePictureType;
   207 	hdrB.iPicture=picB;
   208 	//
   209 	InsertL(0,hdrA);
   210 	InsertL(DocumentLength(),hdrB);
   211 	buf.SetLength(0);
   212 	buf.Append(_L("some"));
   213 	buf.Append(CEditableText::EParagraphDelimiter);
   214 	buf.Append(_L("trailing text"));
   215 	InsertL(1,buf);
   216 	//
   217 	Reset();  // Destroys all pictures.
   218 	picA=CXzePicture::NewL('Z');
   219 	hdrA.iPictureType=KUidXzePictureType;
   220 	hdrA.iPicture=picA;
   221 	picB=CXzePicture::NewL('X');
   222 	hdrB.iPictureType=KUidXzePictureType;
   223 	hdrB.iPicture=picB;
   224 	InsertL(0,hdrA);
   225 	InsertL(DocumentLength(),hdrB);
   226 	buf.SetLength(5);  // A single para delimiter, with no trailing text.
   227 	InsertL(1,buf);
   228 	////////////////////////////////
   229 	// Pathalogical case (2)
   230 	// Insert text with delimiters after a picture
   231 	INFO_PRINTF1(_L("Insert text with delimiters after a picture"));
   232 	//
   233 	Reset();  // Destroys all pictures.
   234 	picA=CXzePicture::NewL('Z');
   235 	hdrA.iPictureType=KUidXzePictureType;
   236 	hdrA.iPicture=picA;
   237 	InsertL(0,hdrA);
   238 	buf.SetLength(7);
   239 	InsertL(1,buf);
   240 
   241 	////////////////////////////////
   242 	// Pathalogical case (3)
   243 	// Insert text with delimiters before a picture
   244 	INFO_PRINTF1(_L("Insert text with delimiters before a picture"));
   245 	//
   246 	Reset();  // Destroys all pictures.
   247 	picA=CXzePicture::NewL('Z');
   248 	hdrA.iPictureType=KUidXzePictureType;
   249 	hdrA.iPicture=picA;
   250 	InsertL(0,hdrA);
   251 	buf.SetLength(7);
   252 	InsertL(0,buf);
   253 	//	
   254 	Reset();
   255 	
   256 	}
   257 
   258 
   259 template<class S, class T,CEditableText::TDocumentStorage D>
   260 void CRichTest<S,T,D>::RegisterMethodsL()
   261 //
   262 // 1st part of testing, involves all methods being called to ensure that all
   263 // methods exist and run without panicking.
   264 //
   265 	{
   266 	INFO_PRINTF1(_L("CRichText::NewL()"));
   267 
   268 	// InsertL()
   269 	INFO_PRINTF1(_L("InsertL()"));
   270 	TPtrC buf(_L("Herewith"));
   271 	InsertL(0,buf);
   272 
   273 //	11.3.97 DavidA defect test 
   274 	TCharFormat charFormat;
   275 	TCharFormatMask charFormatMask;
   276 	TInt lastChar=DocumentLength();
   277 	ApplyCharFormatL(charFormat,charFormatMask,lastChar,1);
   278 //	end of addition.
   279 
   280 
   281 	TPtrC buf2(_L("Hello"));
   282 	InsertL(0,buf2);
   283 
   284 	DeleteL(3,7);
   285 
   286 	// SetInsertCharFormatL()
   287 	INFO_PRINTF1(_L("SetInsertCharFormatL()"));
   288 	TCharFormat format; TCharFormatMask mask;
   289 	format.iFontSpec.iFontStyle.SetStrokeWeight(EStrokeWeightBold);
   290 	mask.SetAttrib(EAttFontStrokeWeight);
   291 	SetInsertCharFormatL(format,mask,3);
   292 	// Test for alloc heaven when doing multiple simultaneous SetInsertCharFormat()s.
   293 	format.iFontSpec.iFontStyle.SetPosture(EPostureItalic);
   294 	mask.SetAttrib(EAttFontPosture);
   295 	SetInsertCharFormatL(format,mask,3);
   296 	//
   297 	TPtrC buf3(_L(" is bold."));
   298 	InsertL(3,buf3);
   299 	CancelInsertCharFormat();
   300 
   301 	// Inserting new paragraph
   302 	INFO_PRINTF1(_L("Inserting paragraph delimiter (Insert(TChar))"));
   303 	InsertL(4,CEditableText::EParagraphDelimiter);
   304 
   305 	// Delete()
   306 	INFO_PRINTF1(_L("DeleteL()"));
   307 	DeleteL(5,1);
   308 
   309 	// Sensing character format over a specified range.
   310 	INFO_PRINTF1(_L("GetCharFormat()"));
   311 	TCharFormat charFormat1; TCharFormatMask undeterminedMask;
   312 	GetCharFormat(charFormat1,undeterminedMask,0,4);
   313 
   314 	// Sensing paragraph format over a specified range.
   315 	INFO_PRINTF1(_L("SenseParagraphFormatL()"));
   316 	CParaFormat* pParaFormat=NULL;
   317 	TRAPD(r,pParaFormat=CParaFormat::NewL());
   318 	test(r==KErrNone);
   319 	TParaFormatMask undeterminedParaMask;
   320 	GetParaFormatL(pParaFormat,undeterminedParaMask,0,DocumentLength());
   321 	delete pParaFormat;
   322 	
   323 	// DelSetInsertCharFormatL()
   324 	INFO_PRINTF1(_L("DelSetInsertCharFormatL()"));
   325 	TPtrC buf4(_L("This is para 2."));
   326 	InsertL(0,buf4);
   327 	SetInsertCharFormatL(format,mask,15);
   328 	TPtrC buf5(_L("In italic."));
   329 	InsertL(15,buf5);
   330 	DelSetInsertCharFormatL(4,5);
   331 	
   332 	// CancelInsertCharFormat()
   333 	INFO_PRINTF1(_L("CancelInsertCharFormat()"));
   334 	CancelInsertCharFormat();
   335 	
   336 	// ApplyParaFormatL()
   337 	INFO_PRINTF1(_L("ApplyParaFormatL()"));
   338 	CParaFormat* paraFormat=CParaFormat::NewL(); TParaFormatMask paraMask;
   339 	paraFormat->iHorizontalAlignment=CParaFormat::ECenterAlign;
   340 	paraMask.SetAttrib(EAttAlignment);
   341 	ApplyParaFormatL(paraFormat,paraMask,0,DocumentLength());
   342 
   343 	// ApplyCharFormatL()
   344 	INFO_PRINTF1(_L("ApplyCharFormatL()"));
   345 	format.iFontSpec.iFontStyle.SetPosture(EPostureItalic); mask.SetAttrib(EAttFontPosture);
   346 	format.iFontPresentation.iStrikethrough=EStrikethroughOn; mask.SetAttrib(EAttFontStrikethrough);
   347 	ApplyCharFormatL(format,mask,0,DocumentLength());
   348 
   349 	// SenseParaFormatL()
   350 	INFO_PRINTF1(_L("SenseParaFormatL()"));
   351 	GetParagraphFormatL(paraFormat,DocumentLength()-1);
   352 
   353 	// CountParas()
   354 	INFO_PRINTF1(_L("ParagraphCount()"));
   355 	ParagraphCount();
   356 
   357 	// ParagraphStart()
   358 	INFO_PRINTF1(_L("ParagraphStart()"));
   359 	TInt aPos=0;
   360 	ToParagraphStart(aPos);
   361 
   362 	// Reset
   363 	Reset();
   364 
   365 	delete paraFormat;
   366 
   367 	
   368 	}
   369 
   370 /**
   371 @SYMTestCaseID 			SYSLIB-ETEXT-UT-3548
   372 @SYMTestCaseDesc  		Testing behaviour of function ApplyCharFormatL() when called at end of document
   373 						that contains plain text. Applied formatting should work on subsequent text
   374 @SYMTestPriority  		High
   375 @SYMTestActions 		1. Insert text1
   376 						2. Switch on bold formatting
   377 						3. Insert text2
   378 						4. Test that first char of text1 is not bold and a character of text 2 is bold. 
   379 @SYMTestExpectedResults	ApplyCharFormatL() applies formatting from end of doc to subsequent character.
   380 @SYMDEF					INC109323
   381 */
   382 template<class S, class T,CEditableText::TDocumentStorage D>
   383 void CRichTest<S,T,D>::TestForDefectINC109323L()
   384 	{
   385 	__UHEAP_MARK;
   386 	INFO_PRINTF1(_L(" @SYMTestCaseID:SYSLIB-ETEXT-UT-3548 Test ApplyCharFormatL() for bold at end positions of text "));	
   387 	_LIT(KText1, "Testing Bold format");
   388 	_LIT(KText2, "bold");	
   389 
   390 	CParaFormatLayer* paraLayer = CParaFormatLayer::NewL();
   391 	CleanupStack::PushL(paraLayer);
   392 	CCharFormatLayer* charLayer = CCharFormatLayer::NewL();
   393 	CleanupStack::PushL(charLayer);
   394 	CRichText* doc = CRichText::NewL(paraLayer,charLayer,D);
   395 	CleanupStack::PushL(doc);
   396 	TCharFormat format1;
   397 	TCharFormat fmtCheck1;
   398 	format1.iFontPresentation.iUnderline = EUnderlineOff;
   399 	// initilizing format attributes
   400 	TCharFormatMask mask1;
   401 	TCharFormatMask mask2;	
   402 	fmtCheck1.iFontPresentation.iUnderline = EUnderlineOff;	
   403 	//Test document containing test string and cursor positioned at end of doc.
   404 	//Turn on bold formatting and subsequent characters are in bold format.
   405 	doc->InsertL(0, KText1);
   406 	format1.iFontSpec.iFontStyle.SetStrokeWeight(EStrokeWeightBold);
   407 	mask1.SetAttrib(EAttFontStrokeWeight);//Set Bold
   408 	doc->ApplyCharFormatL(format1, mask1, doc->DocumentLength(),0);//Apply Bold	
   409 	doc->InsertL(19, KText2);//These should be in bold
   410 	mask1.ClearAttrib(EAttFontStrokeWeight);//Remove bold formatting
   411 	doc->GetCharFormat(format1, mask1, 0, 1);//Get format of characters at position 0
   412 	doc->GetCharFormat(fmtCheck1, mask2, 22, 1);//Get format of characters at position 22
   413 	test(format1.iFontSpec.iFontStyle.StrokeWeight()!=EStrokeWeightBold);
   414 	test(fmtCheck1.iFontSpec.iFontStyle.StrokeWeight()==EStrokeWeightBold);	
   415 	//tidyup
   416 	CleanupStack::PopAndDestroy(doc);
   417 	CleanupStack::PopAndDestroy(charLayer);
   418 	CleanupStack::PopAndDestroy(paraLayer);	
   419 	__UHEAP_MARKEND;
   420 	}
   421 
   422 template<class S, class T,CEditableText::TDocumentStorage D>
   423 void CRichTest<S,T,D>::TestRemSpecParaFmtL()
   424 //
   425 // Test the RemoveSpecificParaFormatL() method of CRichText.
   426 //
   427 	{
   428 	__UHEAP_MARK;
   429 	INFO_PRINTF1(_L("RemoveSpecificParaFormatL()"));
   430 	CParaFormat* globalParaFormat=CParaFormat::NewLC();
   431 	TParaFormatMask globalParaMask;
   432 	globalParaFormat->iHorizontalAlignment=CParaFormat::ECenterAlign;
   433 	globalParaMask.SetAttrib(EAttAlignment);
   434 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL(globalParaFormat,globalParaMask);
   435 	CCharFormatLayer* charLayer=CCharFormatLayer::NewL();
   436 	CRichText* doc=NULL;
   437 	TRAPD(ret,doc=CRichText::NewL(paraLayer,charLayer,D));
   438 	if (ret!=KErrNone)
   439 		{
   440 		test(doc==NULL);
   441 		User::Leave(ret);
   442 		}
   443 	TPtrC buf1(_L("Here is paragraph one text."));
   444 	doc->InsertL(0,buf1);
   445 	//
   446 	// Apply specific paragraph format
   447 	CParaFormat* format=CParaFormat::NewLC();
   448 	TParaFormatMask mask;
   449 	format->iHorizontalAlignment=CParaFormat::ERightAlign;
   450 	mask.SetAttrib(EAttAlignment);
   451 	doc->ApplyParaFormatL(format,mask,1,1);
   452 	CParaFormat* sensedFormat=CParaFormat::NewLC();
   453 	doc->GetParagraphFormatL(sensedFormat,0);
   454 	test(sensedFormat->iHorizontalAlignment==CParaFormat::ERightAlign);
   455 	//
   456 	// Remove specific para format and test
   457 	doc->RemoveSpecificParaFormatL(0,1);
   458 	doc->GetParagraphFormatL(sensedFormat,0);
   459 	test(sensedFormat->iHorizontalAlignment==CParaFormat::ECenterAlign);
   460 	//
   461 	CleanupStack::PopAndDestroy(3); // the 2 para formats.
   462 	delete doc;
   463 	delete paraLayer;
   464 	delete charLayer;
   465 
   466 	__UHEAP_MARKEND;
   467 	}
   468 
   469 
   470 template<class S, class T,CEditableText::TDocumentStorage D>
   471 void CRichTest<S,T,D>::TestDelete1L()
   472 //
   473 // Test the DeleteL() method of CRichText.
   474 //
   475 	{
   476 	__UHEAP_MARK;
   477 	INFO_PRINTF1(_L("DeleteL()"));
   478 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL();
   479 	CCharFormatLayer* charLayer=CCharFormatLayer::NewL();
   480 	CRichText* doc=NULL;
   481 	TRAPD(ret,doc=CRichText::NewL(paraLayer,charLayer,D));
   482 	if (ret!=KErrNone)
   483 		{
   484 		test(doc==NULL);
   485 		User::Leave(ret);
   486 		}
   487 	TPtrC buf1(_L("Here is paragraph one text."));
   488 	TPtrC buf2(_L("Here is paragraph one text."));
   489 	doc->InsertL(0,buf1);
   490 	doc->InsertL(doc->DocumentLength(),EParagraphDelimiter);
   491 	doc->InsertL(doc->DocumentLength(),buf2);
   492 	//
   493 	// Apply formatting
   494 	TCharFormat applyFormat;
   495 	TCharFormatMask applyMask;
   496 	applyFormat.iFontSpec.iFontStyle.SetStrokeWeight(EStrokeWeightBold);
   497 	applyMask.SetAttrib(EAttFontStrokeWeight);
   498 	doc->ApplyCharFormatL(applyFormat,applyMask,8,28);
   499 	//
   500 	// Now try the delete - should end up with a single shared paragraph!!!
   501 	doc->DeleteL(8,28);
   502 	TInt documentLength=doc->DocumentLength();
   503 	test(documentLength==27);
   504 	//
   505 	delete doc;
   506 	delete paraLayer;
   507 	delete charLayer;
   508 
   509 	__UHEAP_MARKEND;
   510 	}
   511 
   512 
   513 template<class S, class T,CEditableText::TDocumentStorage D>
   514 void CRichTest<S,T,D>::TestDelete2L()
   515 //
   516 // Test the DeleteL() method of CRichText.
   517 // Deleting the paragraph delimiter between 2 paras of constant character format, but of varying
   518 // paragraph formats.
   519 //
   520 	{
   521 	__UHEAP_MARK;
   522 	INFO_PRINTF1(_L("DeleteL()"));
   523 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL();
   524 	CCharFormatLayer* charLayer=CCharFormatLayer::NewL();
   525 	CRichText* doc=NULL;
   526 	TRAPD(ret,doc=CRichText::NewL(paraLayer,charLayer,D));
   527 	if (ret!=KErrNone)
   528 		{
   529 		test(doc==NULL);
   530 		User::Leave(ret);
   531 		}
   532 	TPtrC buf1(_L("A"));
   533 	TPtrC buf2(_L("B"));
   534 	TPtrC buf3(_L("C"));
   535 	doc->InsertL(0,buf1);
   536 	doc->InsertL(doc->DocumentLength(),EParagraphDelimiter);
   537 	doc->InsertL(doc->DocumentLength(),buf2);
   538 	doc->InsertL(doc->DocumentLength(),EParagraphDelimiter);
   539 	doc->InsertL(doc->DocumentLength(),buf3);
   540 	//
   541 	// Apply formatting
   542 	CParaFormat* applyFormat=CParaFormat::NewLC();
   543 	TParaFormatMask applyMask;
   544 	applyFormat->iHorizontalAlignment=CParaFormat::ERightAlign;
   545 	applyMask.SetAttrib(EAttAlignment);
   546 	//
   547 	// Make 1st & 3rd para the same para format, different to para 2.
   548 	
   549 	doc->ApplyParaFormatL(applyFormat,applyMask,1,1);
   550 	doc->ApplyParaFormatL(applyFormat,applyMask,5,1);
   551 	CleanupStack::PopAndDestroy();  // para format
   552 	//
   553 	// Now try the delete - should end up with a single shared paragraph!!!
   554 	/*TBool parasMerged=*/doc->DeleteL(3,1);  // delete para 2 delimiter
   555 	TInt documentLength=doc->DocumentLength();
   556 	test(documentLength==4);
   557 	//
   558 	delete doc;
   559 	delete paraLayer;
   560 	delete charLayer;
   561 
   562 	__UHEAP_MARKEND;
   563 	}
   564 
   565 
   566 /**
   567 @SYMTestCaseID             SYSLIB-ETEXT-UT-3431
   568 @SYMTestCaseDesc          Testing behaviour of functions ApplyCharFormatL() and RemoveSpecificFormatL();
   569     formatting of text including end of document character
   570 @SYMTestPriority          High
   571 @SYMTestActions         1. format text and compare attribute values for equality
   572                         2. testing for fix of INC097216 whether EOD also formatted
   573                         3. remove formatting of text and compare attribute values for equality
   574                         4. testing for fix of DEF104149 whether EOD also has formatting removed
   575 @SYMTestExpectedResults    ApplyCharFormatL() and RemoveSpecificFormatL() apply and remove formatting from
   576     specified range of text including end of document character
   577 @SYMDEF                    DEF104149
   578 */
   579 template<class S, class T,CEditableText::TDocumentStorage D>
   580 void CRichTest<S,T,D>::TestApplyRemoveCharFormat()
   581     {
   582     __UHEAP_MARK;
   583     INFO_PRINTF1(_L(" @SYMTestCaseID:SYSLIB-ETEXT-UT-3431 Test ApplyCharFormatL() & RemoveSpecificCharFormat() "));	
   584     CParaFormatLayer* paraLayer = CParaFormatLayer::NewL();
   585     CleanupStack::PushL(paraLayer);
   586     CCharFormatLayer* charLayer = CCharFormatLayer::NewL();
   587     CleanupStack::PushL(charLayer);
   588     CRichText* doc = CRichText::NewL(paraLayer,charLayer,D);
   589     CleanupStack::PushL(doc);
   590     _LIT(KText1, "Hello there!");
   591     doc->InsertL(0, KText1);
   592     test(doc->DocumentLength()==12);
   593    
   594     // initilizing format (ie.underline) values
   595     TCharFormat formatCheck;
   596     formatCheck.iFontPresentation.iUnderline = EUnderlineOff;
   597     TCharFormat fmatCheck;
   598     formatCheck.iFontPresentation.iUnderline = EUnderlineOff;
   599     TCharFormat format;
   600     format.iFontPresentation.iUnderline = EUnderlineOn;
   601     TCharFormat formatNone;
   602     formatCheck.iFontPresentation.iUnderline = EUnderlineOff;
   603 
   604     // initilizing format attributes
   605     TCharFormatMask maskCheck;   
   606     TCharFormatMask mask;
   607     mask.SetAttrib(EAttFontUnderline);
   608    
   609     // underline KText1 and compare its attribute values with EUnderlineOn for equality
   610     doc->ApplyCharFormatL(format, mask, 0, 12);   
   611     doc->GetCharFormat(formatCheck, maskCheck, 0, 12);
   612     // testing for fix of INC097216 whether EOD also formatted
   613     doc->GetCharFormat(fmatCheck, maskCheck, 12, 1);
   614     test(format.IsEqual(formatCheck));
   615     test(format.IsEqual(fmatCheck));
   616 
   617     // remove underlining of KText1 and compare its attribute values with EUnderlineOff for equality
   618     doc->RemoveSpecificCharFormatL(0, 12);
   619     doc->GetCharFormat(formatCheck, maskCheck, 0, 12);
   620     // testing for fix of INC104149 whether EOD also has formatting removed
   621     doc->GetCharFormat(fmatCheck, maskCheck, 12, 1);
   622     test(formatCheck.IsEqual(formatNone));
   623     test(fmatCheck.IsEqual(formatNone));
   624    
   625     CleanupStack::PopAndDestroy(doc);
   626     CleanupStack::PopAndDestroy(charLayer);
   627     CleanupStack::PopAndDestroy(paraLayer);   
   628     __UHEAP_MARKEND;
   629     }
   630 
   631 
   632 template<class S, class T,CEditableText::TDocumentStorage D>
   633 void CRichTest<S,T,D>::TestResetL()
   634 //
   635 // Test the Reset() method of CRichText.
   636 //
   637 	{
   638 	__UHEAP_MARK;
   639 	INFO_PRINTF1(_L("Reset()"));
   640 	CParaFormatLayer* paraLayer=CParaFormatLayer::NewL();
   641 	CCharFormatLayer* charLayer=CCharFormatLayer::NewL();
   642 	CRichText* doc=NULL;
   643 	TRAPD(ret,doc=CRichText::NewL(paraLayer,charLayer,D));
   644 	if (ret!=KErrNone)
   645 		{
   646 		test(doc==NULL);
   647 		User::Leave(ret);
   648 		}
   649 	TPtrC buf1(_L("Here is paragraph one text."));
   650 	TPtrC buf2(_L("And here is that of the second paragraph"));
   651 	doc->InsertL(0,buf1);
   652 	doc->InsertL(doc->DocumentLength(),EParagraphDelimiter);
   653 	doc->InsertL(doc->DocumentLength(),buf2);
   654 	//
   655 	doc->Reset();
   656 	test(doc->ParagraphCount()==1);
   657 	doc->Reset();
   658 	test(doc->ParagraphCount()==1);
   659 	doc->Reset();
   660 	test(doc->ParagraphCount()==1);
   661 	delete doc;
   662 	delete paraLayer;
   663 	delete charLayer;
   664 
   665 	__UHEAP_MARKEND;
   666 	}
   667 	
   668 
   669 template<class S, class T,CEditableText::TDocumentStorage D>
   670 void CRichTest<S,T,D>::DoPML()
   671 //
   672 // Use PML source translator to produce a RichText component.
   673 //
   674 	{
   675  	__UHEAP_MARK;
   676 	INFO_PRINTF1(_L("Checking PML sourced RichText Component"));
   677  	// set filename
   678 	TFileName theFileName=_L("z:\\test\\app-framework\\etext\\t_rich1a.pml");
   679 	// Parse PML
   680 	CParser* myParser=NULL;
   681 	CRichText* richTextDoc=NULL;
   682 	TRAPD(ret, myParser=CParser::NewL());
   683 	CleanupStack::PushL(myParser);
   684 	TRAP(ret, richTextDoc=myParser->ParseL(theFileName));
   685 	CleanupStack::PushL(richTextDoc);
   686 	CParaFormatLayer* pl = const_cast<CParaFormatLayer*>(
   687 		richTextDoc->GlobalParaFormatLayer());
   688 	CleanupStack::PushL(pl);
   689 	CCharFormatLayer* cl = const_cast<CCharFormatLayer*>(
   690 		richTextDoc->GlobalCharFormatLayer());
   691 	CleanupStack::PushL(cl);
   692 	
   693 	TInt paraCount=richTextDoc->ParagraphCount();
   694 	test(paraCount==2);
   695 	TBool hasMarkupData=richTextDoc->HasMarkupData();
   696 	test(hasMarkupData==EFalse);
   697 
   698 	// Testing overloaded senseChars - looking at the indeterminate mask.
   699 //	TCharFormat charFormat;	TCharFormatMask charFormatMask;
   700 //	richTextDoc->GetCharFormat(0,17,charFormat,charFormatMask);
   701 //	test(charFormatMask==0);
   702 
   703 //	richTextDoc->GetCharFormat(17,10,charFormat,charFormatMask);
   704 
   705 	CParaFormat* paraFormat = 0;
   706 	paraFormat=CParaFormat::NewLC();
   707 	TParaFormatMask paraFormatMask;
   708 	richTextDoc->GetParaFormatL(paraFormat,paraFormatMask,0,2);
   709 	richTextDoc->GetParaFormatL(paraFormat,paraFormatMask,17,3);  // All para 2 and just para1
   710 
   711 	CleanupStack::PopAndDestroy(paraFormat);
   712 	CleanupStack::PopAndDestroy(cl);
   713 	CleanupStack::PopAndDestroy(pl);
   714 	CleanupStack::PopAndDestroy(richTextDoc);
   715 	CleanupStack::PopAndDestroy(myParser);
   716 	__UHEAP_MARKEND;
   717 	}
   718 
   719 template<class S, class T,CEditableText::TDocumentStorage D>
   720 void CRichTest<S,T,D>::TestForDefectTET5DHEWWL()
   721 //
   722 // Test the CRichTextIndex::InsertL() for defect TET-5DHEWW which 
   723 // should not be present from 15/10/02.
   724 //
   725 	{
   726 	__UHEAP_MARK;
   727 
   728 	INFO_PRINTF1(_L("Testing for the presence of defect TET-5DHEWW"));
   729 
   730 	CParaFormatLayer* paraLayer = CParaFormatLayer::NewL();
   731 	CCharFormatLayer* charLayer = CCharFormatLayer::NewL();
   732 
   733 	CRichText* doc = NULL;
   734 	TRAPD(ret, doc = CRichText::NewL(paraLayer,charLayer,D));
   735 	if (ret!=KErrNone)
   736 		{
   737 		test(doc==NULL);
   738 		User::Leave(ret);
   739 		}
   740 
   741 	TPtrC buf1(_L("sometext "));
   742 	doc->InsertL(0, buf1);
   743 
   744 	TCharFormat charFormat12(_L("Times New Roman"), 240);
   745 	TCharFormatMask charMask12;
   746 	charMask12.SetAttrib(EAttFontHeight);
   747 	doc->SetInsertCharFormatL(charFormat12, charMask12, doc->DocumentLength());
   748 
   749 	TPtrC buf2(_L("sometext "));
   750 	doc->InsertL(doc->DocumentLength(), buf2);
   751 
   752 	doc->SetInsertCharFormatL(charFormat12, charMask12, doc->DocumentLength());
   753 	doc->InsertL(doc->DocumentLength(), EParagraphDelimiter);
   754 
   755 	// Defect present in CRichTextIndex if test executable crashes/panics on the 
   756 	// previous line on a debug build. Should this test method return normally then 
   757 	// the EText library used does not contain the defect.
   758 
   759 	TInt docLen = doc->DocumentLength();
   760 	test(docLen==19);
   761 
   762 	INFO_PRINTF1(_L("RTEST:                        Test PASSED - defect not present!\n"));
   763 
   764 	delete doc;
   765 	delete paraLayer;
   766 	delete charLayer;
   767 
   768 	__UHEAP_MARKEND;
   769 	}
   770 
   771 // CRichTextIndex::GetParaFormatL doesn't set TParaFormatMask correctly
   772 template<class S, class T,CEditableText::TDocumentStorage D>
   773 void CRichTest<S,T,D>::TestForDefectINC010183L()
   774 	{
   775 	__UHEAP_MARK;
   776 
   777 	INFO_PRINTF1(_L("Testing for the presence of defect INC010183"));
   778 
   779 	CParaFormatLayer* paraLayer = CParaFormatLayer::NewL();
   780 	CleanupStack::PushL(paraLayer);
   781 	CCharFormatLayer* charLayer = CCharFormatLayer::NewL();
   782 	CleanupStack::PushL(charLayer);
   783 
   784 	CRichText* doc = CRichText::NewL(paraLayer,charLayer,D);
   785 	CleanupStack::PushL(doc);
   786 
   787 	TPtrC buf1(_L("para1\x2029para2\x2029para3"));
   788 	doc->InsertL(0, buf1);
   789 
   790 	CParaFormat* pFormat = CParaFormat::NewLC();
   791 	pFormat->iBullet = new(ELeave) TBullet;
   792 	pFormat->iBullet->iHeightInTwips = 1;
   793 	TParaFormatMask pMask;
   794 	pMask.SetAttrib(EAttBullet);
   795 
   796 	doc->ApplyParaFormatL(pFormat, pMask, 8, 1);
   797 	doc->GetParaFormatL(pFormat, pMask, 0, 14);
   798 
   799 	// test that the bullet's "varies" flag is set.
   800 	test(pMask.AttribIsSet(EAttBullet));
   801 
   802 	doc->GetParaFormatL(pFormat, pMask, 7, 6);
   803 
   804 	// test that the bullet's "varies" flag is set.
   805 	test(pMask.AttribIsSet(EAttBullet));
   806 
   807 	CleanupStack::PopAndDestroy(pFormat);
   808 	CleanupStack::PopAndDestroy(doc);
   809 	CleanupStack::PopAndDestroy(charLayer);
   810 	CleanupStack::PopAndDestroy(paraLayer);
   811 
   812 	__UHEAP_MARKEND;
   813 	}
   814 
   815 
   816 // CRichTextIndex::InsertL, Insertion of zero-length text should not cancel the pending new TCharFormat.
   817 // This test sets the Strikethrough option on and applies it to a zero length section of text before checking
   818 // the format has been applied correctly. Then it inputs 3 characters and checks to see if the format of these
   819 // characters also has the character format applied to them.  
   820 template<class S, class T,CEditableText::TDocumentStorage D>
   821 void CRichTest<S,T,D>::TestForDefectINC064162L()
   822 	{
   823 	__UHEAP_MARK;
   824 	
   825 	INFO_PRINTF1(_L("INC064162 - Testing insertion of zero-length text doesn't cancel the pending TCharFormat"));
   826 	
   827 	TInt length = 3;	
   828 	TPtrC bufPtrLetterNone(_L(""));
   829 	TPtrC bufPtrLettersABC(_L("ABC"));
   830 	
   831 	CParaFormatLayer* paraLayer = CParaFormatLayer::NewL();
   832 	CleanupStack::PushL(paraLayer);
   833 	CCharFormatLayer* charLayer = CCharFormatLayer::NewL();
   834 	CleanupStack::PushL(charLayer);
   835 
   836 	CRichText* doc = CRichText::NewL(paraLayer,charLayer,D);
   837 	CleanupStack::PushL(doc);
   838 
   839 	TCharFormat cFormatCheck;
   840 	cFormatCheck.iFontPresentation.iStrikethrough = EStrikethroughOff;
   841 	
   842 	TCharFormat cFormat;
   843 	cFormat.iFontPresentation.iStrikethrough = EStrikethroughOn;
   844 
   845 	TCharFormatMask cMaskCheck;
   846 	
   847 	TCharFormatMask cMask;
   848 	cMask.SetAttrib(EAttFontStrikethrough);
   849 	
   850 	// Applying the strikethrough format on a zero length section of text.
   851 	doc->ApplyCharFormatL(cFormat,cMask,0,0);	
   852 
   853 	doc->InsertL(0, bufPtrLetterNone);
   854 	
   855 	doc->GetCharFormat(cFormatCheck, cMaskCheck,0,0);
   856 
   857 	// testing the format of the empty buf
   858 	test(cFormat.IsEqual(cFormatCheck));
   859 
   860 	doc->InsertL(0, bufPtrLettersABC);	
   861 	doc->GetCharFormat(cFormatCheck, cMaskCheck,0,length);
   862 	
   863 	// testing the format of the 3 characters
   864 	test(cFormat.IsEqual(cFormatCheck));
   865  
   866 	CleanupStack::PopAndDestroy(doc);
   867 	CleanupStack::PopAndDestroy(charLayer);
   868 	CleanupStack::PopAndDestroy(paraLayer);
   869 
   870 	__UHEAP_MARKEND;
   871 	}
   872 
   873 
   874 
   875 template<class S, class T,CEditableText::TDocumentStorage D>
   876 TBool CRichTest<S,T,D>::TestAndRunStageL(TInt aStage)
   877 	{
   878 	_LIT(KSomeText, "Text\x2029par ");
   879 	_LIT(KSomeTextPlus3, "Text\x2029par \x2029z ");
   880 	TCharFormat format;
   881 	TCharFormatMask mask;
   882 	TCharFormatMask varies;
   883 	switch (aStage)
   884 		{
   885 	case 0:
   886 		Reset();
   887 		return ETrue;
   888 	case 1:
   889 		test(DocumentLength() == 0);
   890 		InsertL(0, KSomeText);
   891 		return ETrue;
   892 	case 2:
   893 			{
   894 			TBuf<100> buf;
   895 			test(DocumentLength() == KSomeText().Length());
   896 			Extract(buf, 0, KSomeText().Length());
   897 			test(0 == buf.Compare(KSomeText));
   898 			TCharFormat format0;
   899 			GetCharFormat(format, varies, 0, KSomeText().Length());
   900 			test(varies.IsNull());		// format should not vary
   901 			test(format.IsEqual(format0));
   902 			}
   903 		// Set an insert char format for font height 100
   904 		mask.SetAttrib(EAttFontHeight);
   905 		format.iFontSpec.iHeight = 100;
   906 		SetInsertCharFormatL(format, mask, 9);
   907 		return ETrue;
   908 	case 3:
   909 		test(DocumentLength() == KSomeText().Length());
   910 		// Insert a carriage return and cancel the insert char format.
   911 		// This stands in place of moving the cursor away from the
   912 		// insertion point and back again.
   913 		InsertL(9, 0x2029);
   914 		return ETrue;
   915 	case 4:
   916 		test(DocumentLength() == KSomeText().Length() + 1);
   917 		CancelInsertCharFormat();
   918 		return ETrue;
   919 	case 5:
   920 		test(DocumentLength() == KSomeText().Length() + 1);
   921 		GetCharFormat(format, varies, 0, 9);
   922 			{
   923 			TCharFormat format0;
   924 			test(format.IsEqual(format0));
   925 			test(varies.IsNull());
   926 			}
   927 		GetCharFormat(format, varies, 9, 1);
   928 		test(varies.IsNull());
   929 		test(format.iFontSpec.iHeight == 100);
   930 		// Insert a character after the new carriage return. It should
   931 		// have height 100. This is what the defect INC038479 is
   932 		// complaining about. In fact it is a long-standing problem,
   933 		// long assumed to be too difficult to solve in a reasonable
   934 		// time.
   935 		InsertL(10, 'z');
   936 		return ETrue;
   937 	case 6:
   938 		test(DocumentLength() == KSomeText().Length() + 2);
   939 		GetCharFormat(format, varies, 9, 2);
   940 		test(varies.IsNull());
   941 		test(format.iFontSpec.iHeight == 100);
   942 		// Insert a space. This prepares us for checking for a regression
   943 		// that was introduced then fixed in the fix for INC038479.
   944 		InsertL(11, ' ');
   945 		return ETrue;
   946 	case 7:
   947 			{
   948 			TBuf<100> buf;
   949 			test(DocumentLength() == KSomeTextPlus3().Length());
   950 			Extract(buf, 0, KSomeTextPlus3().Length());
   951 			test(0 == buf.Compare(KSomeTextPlus3));
   952 			TCharFormat format0;
   953 			GetCharFormat(format, varies, 0, KSomeText().Length());
   954 			test(varies.IsNull());		// format should not vary across the original bit
   955 			test(format.IsEqual(format0));
   956 			GetCharFormat(format, varies, KSomeText().Length(), 3);
   957 			test(format.iFontSpec.iHeight == 100);
   958 			test(varies.IsNull());
   959 			}
   960 		// Set a new insert character format for bold
   961 		mask.SetAttrib(EAttFontStrokeWeight);
   962 		format.iFontSpec.iFontStyle.SetStrokeWeight(EStrokeWeightBold);
   963 		SetInsertCharFormatL(format, mask, 9);
   964 		return ETrue;
   965 	case 8:
   966 		// and unset the insert character format again
   967 		mask.SetAttrib(EAttFontStrokeWeight);
   968 		format.iFontSpec.iFontStyle.SetStrokeWeight(EStrokeWeightNormal);
   969 		SetInsertCharFormatL(format, mask, 9);
   970 		return ETrue;
   971 	case 9:
   972 		test(DocumentLength() == KSomeTextPlus3().Length());
   973 		// Add a carriage return.
   974 		InsertL(12, 0x2029);
   975 		return ETrue;
   976 	case 10:
   977 		test(DocumentLength() == KSomeTextPlus3().Length() + 1);
   978 		GetCharFormat(format, varies, KSomeTextPlus3().Length(), 1);
   979 		test(varies.IsNull());
   980 		test(format.iFontSpec.iFontStyle.StrokeWeight() == EStrokeWeightNormal);
   981 		return EFalse;
   982 	default:
   983 		return ETrue;
   984 		}
   985 	}
   986 
   987 template<class S, class T,CEditableText::TDocumentStorage D>
   988 void CRichTest<S,T,D>::TestErrorCondition(TInt aStage, TInt aError)
   989 	{
   990 	// No leaves possible except NoMemory.
   991 	test(aError == KErrNoMemory);
   992 	// Stage 4 should not leave at all.
   993 	test(aStage != 4);
   994 	}
   995 
   996 
   997 template<class S, class T, CEditableText::TDocumentStorage D>
   998 void TestClassesL()
   999 	{
  1000 	__UHEAP_MARK;
  1001 
  1002 	CParaFormatLayer* paraLayer = CParaFormatLayer::NewL();
  1003 	CleanupStack::PushL(paraLayer);
  1004 	CCharFormatLayer* charLayer = CCharFormatLayer::NewL();
  1005 	CleanupStack::PushL(charLayer);
  1006 	CRichTest<S, T, D>* doc = CRichTest<S, T, D>::NewL(paraLayer, charLayer);
  1007 	CleanupStack::PushL(doc);
  1008 
  1009 	doc->DoFlatTests();
  1010 	doc->TestResetL();
  1011 	doc->TestDelete1L();
  1012 	doc->TestDelete2L();
  1013 	doc->TestApplyRemoveCharFormat();
  1014 	doc->TestRemSpecParaFmtL();
  1015 	doc->DoPML();
  1016 	doc->TestForDefectTET5DHEWWL();
  1017 	doc->TestForDefectINC010183L();
  1018 	doc->TestForDefectINC064162L();
  1019 	doc->TestForDefectINC109323L();
  1020 	INFO_PRINTF1(_L("INC038479 - Email editor: font settings made at doc end can be lost and returned to default"));
  1021 	// Also tests SetInsetCharFormat more thoroughly.
  1022 	doc->RunAtomicTest();
  1023 
  1024 	CleanupStack::PopAndDestroy(doc);
  1025 	CleanupStack::PopAndDestroy(charLayer);
  1026 	CleanupStack::PopAndDestroy(paraLayer);
  1027 
  1028 	__UHEAP_MARKEND;
  1029 	}
  1030 
  1031 void DoTestsL()
  1032 	{
  1033 	INFO_PRINTF1(_L("CRichText - Flat"));
  1034 	TestClassesL<TText,TPtrC,CEditableText::EFlatStorage>();
  1035 	INFO_PRINTF1(_L("CRichText - Segmented"));
  1036 	TestClassesL<TText,TPtrC,CEditableText::ESegmentedStorage>();
  1037 	
  1038 	}
  1039 
  1040 	
  1041 LOCAL_C void setupCleanup()
  1042 //
  1043 // Initialise the cleanup stack.
  1044 //
  1045 	{
  1046 
  1047 	TheTrapCleanup=CTrapCleanup::New();
  1048 	TRAPD(r,\
  1049 		{\
  1050 		for (TInt i=KTestCleanupStack;i>0;i--)\
  1051 			CleanupStack::PushL((TAny*)1);\
  1052 		test(r==KErrNone);\
  1053 		CleanupStack::Pop(KTestCleanupStack);\
  1054 		});
  1055 	}
  1056 
  1057 CT_RICH1A::CT_RICH1A()
  1058     {
  1059     SetTestStepName(KTestStep_T_RICH1A);
  1060     pTestStep = this;
  1061     }
  1062 
  1063 TVerdict CT_RICH1A::doTestStepL()
  1064     {
  1065     SetTestStepResult(EFail);
  1066 
  1067     setupCleanup();
  1068     INFO_PRINTF1(_L("CRichText Document"));
  1069     __UHEAP_MARK;
  1070     
  1071     TRAPD(ret,DoTestsL());
  1072     
  1073     __UHEAP_MARKEND;
  1074     delete TheTrapCleanup;
  1075     
  1076     if (ret == KErrNone)
  1077         {
  1078         SetTestStepResult(EPass);
  1079         }
  1080 
  1081     return TestStepResult();
  1082     }