os/textandloc/textrendering/texthandling/ttext/T_parse.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 1999-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 * TPARSE.CPP
    16 *
    17 */
    18 
    19 
    20 #include <txtrich.h>
    21 #include <e32math.h>
    22 #include "T_parse.h"
    23 #include "T_PARSE_for_TEF.h"
    24 
    25 LOCAL_D CTestStep *pTestStep = NULL;
    26 #define test(cond)											\
    27 	{														\
    28 	TBool __bb = (cond);									\
    29 	pTestStep->TEST(__bb);									\
    30 	if (!__bb)												\
    31 		{													\
    32 		pTestStep->ERR_PRINTF1(_L("ERROR: Test Failed"));	\
    33 		User::Leave(1);										\
    34 		}													\
    35 	}
    36 #undef INFO_PRINTF1
    37 #undef INFO_PRINTF2
    38 #undef INFO_PRINTF3
    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 #define INFO_PRINTF3(p1, p2, p3)    pTestStep->Logger().LogExtra(((TText8*)__FILE__), __LINE__, ESevrInfo, (p1), (p2), (p3))
    43 
    44 
    45 LOCAL_D CTrapCleanup* TrapCleanup;
    46 LOCAL_D const TInt KTestCleanupStack=0x200;
    47 
    48 
    49 CTestParser* CTestParser::NewL()
    50 	{
    51 	CTestParser* self = new (ELeave) CTestParser;
    52 	CleanupStack::PushL(self);
    53 	self->iDoItText = HBufC::NewL(128);
    54 	*(self->iDoItText) = _L("Or is it just undead?");
    55 	CleanupStack::Pop();
    56 	return self;
    57 	}
    58 
    59 
    60 CTestParser::CTestParser()
    61 	{
    62 	}
    63 
    64 
    65 CTestParser::~CTestParser()
    66 	{
    67 	delete iDoItText;
    68 	}
    69 
    70 
    71 TBool CTestParser::ParseThisText(const CRichText& aTextObj, TBool aAllowBack,
    72 									TInt aStartScan, TInt aScanLength,
    73 									TInt& aStartTag, TInt& aTagLength)
    74 	{
    75 	// Very simple - looking for the phrase "the world"
    76 	_LIT(targetText, "TARGET");
    77 	TInt length = 6;
    78 	// Consider that the start of the scan may be part way through
    79 	// the item that we are looking for, so see if we need to move back
    80 	TInt start = aStartScan;
    81 	if (aAllowBack)
    82 		{
    83 		if (start >= length - 1)
    84 			start -= length - 1;
    85 		else
    86 			start = 0;
    87 		}
    88 	// And it might go beyond the end of the scan
    89 	TInt end = aStartScan + aScanLength;
    90 	if (end + length - 1 <= aTextObj.DocumentLength())
    91 		end += length - 1;
    92 	else
    93 		end = aTextObj.DocumentLength();
    94 	while (end - start >= length)
    95 		{
    96 		TPtrC buffer = aTextObj.Read(start, end - start);
    97 		TInt segmentLength= buffer.Length();
    98 
    99 		if (segmentLength >= length)	// There's enough text to bother searching
   100 			{
   101 			TInt result = buffer.FindF(targetText);
   102 			if (result != KErrNotFound)	// We found it
   103 				{
   104 				aStartTag = start + result;
   105 				aTagLength = length;
   106 				return ETrue;
   107 				}
   108 			}
   109 		if (end == start + segmentLength)	// There's no more text at all
   110 			break;
   111 		// The buffer is segmented and there's another segment 
   112 
   113 		// It could be over the segment boundary
   114 		TInt start2 = start + segmentLength;
   115 		TInt end2 = start2;
   116 		if (segmentLength >= length - 1)
   117 			start2 -= length - 1;
   118 		else start2 = start;
   119 		if (end >= end2 + length - 1)
   120 			end2 += length -1;
   121 		else
   122 			end2 = end;
   123 		if (end2 - start2 >= length)
   124 			{
   125 			// Create a buffer with the end of one and the start of the other
   126 			TBuf<10> bridgeBuffer;
   127 			aTextObj.Extract(bridgeBuffer, start2, end2 - start2);
   128 			TInt result = bridgeBuffer.FindF(targetText);
   129 			if (result != KErrNotFound)	// We found it
   130 				{
   131 				aStartTag = start2 + result;
   132 				aTagLength = length;
   133 				return ETrue;
   134 				}
   135 			}
   136 
   137 		// Move start up for next segment
   138 		start += segmentLength;
   139 		}
   140 		return EFalse;	// Not enough text left in buffer
   141 	}
   142 
   143 
   144 const TDesC& CTestParser::CreateDoItText(const CRichText& /* aTextObj */,
   145 							TInt /* aStartText */, TInt /* aLength */)
   146 	{
   147 	return *iDoItText;
   148 	}
   149 
   150 
   151 void CTestParser::ActivateThisTextL(const CRichText& /* aTextObj */,
   152 					  TInt /* aStartText */, TInt /* aLength */)
   153 	{
   154 	// Do something?
   155 
   156 	}
   157 
   158 
   159 void CTestParser::GetRecogniseFormat(TCharFormat& aFormat)
   160 	{
   161 	aFormat.iFontPresentation.iTextColor = KRgbRed;
   162 	aFormat.iFontPresentation.iUnderline = EUnderlineOn;
   163 	}
   164 
   165 
   166 void CTestParser::GetRolloverFormat(TCharFormat& aFormat)
   167 	{
   168 	aFormat.iFontPresentation.iTextColor = KRgbRed;
   169 	aFormat.iFontPresentation.iUnderline = EUnderlineOn;
   170 	aFormat.iFontPresentation.iHighlightColor = KRgbDarkRed;
   171 	aFormat.iFontPresentation.iHighlightStyle = TFontPresentation::EFontHighlightRounded;
   172 	}
   173 
   174 
   175 void CTestParser::Release()
   176 	{
   177 	delete this;
   178 	}
   179 
   180 //--------------------------------------------
   181 
   182 CTestParser2* CTestParser2::NewL()
   183 	{
   184 	CTestParser2* self = new (ELeave) CTestParser2;
   185 	CleanupStack::PushL(self);
   186 	self->iDoItText = HBufC::NewL(128);
   187 	*(self->iDoItText) = _L("Or is it just undead?");
   188 	CleanupStack::Pop();
   189 	return self;
   190 	}
   191 
   192 
   193 CTestParser2::CTestParser2()
   194 	{
   195 	}
   196 
   197 
   198 CTestParser2::~CTestParser2()
   199 	{
   200 	delete iDoItText;
   201 	}
   202 
   203 
   204 TBool CTestParser2::ParseThisText(const CRichText& aTextObj, TBool aAllowBack,
   205 									TInt aStartScan, TInt aScanLength,
   206 									TInt& aStartTag, TInt& aTagLength)
   207 	{
   208 	// Very simple - looking for the phrase "ARG"
   209 	_LIT(targetText, "ARG");
   210 	TInt length = 3;
   211 	// Consider that the start of the scan may be part way through
   212 	// the item that we are looking for, so see if we need to move back
   213 	TInt start = aStartScan;
   214 	if (aAllowBack)
   215 		{
   216 		if (start >= length - 1)
   217 			start -= length - 1;
   218 		else
   219 			start = 0;
   220 		}
   221 	// And it might go beyond the end of the scan
   222 	TInt end = aStartScan + aScanLength;
   223 	if (end + length - 1 <= aTextObj.DocumentLength())
   224 		end += length - 1;
   225 	else
   226 		end = aTextObj.DocumentLength();
   227 	while (end - start >= length)
   228 		{
   229 		TPtrC buffer = aTextObj.Read(start, end - start);
   230 		TInt segmentLength= buffer.Length();
   231 
   232 		if (segmentLength >= length)	// There's enough text to bother searching
   233 			{
   234 			TInt result = buffer.FindF(targetText);
   235 			if (result != KErrNotFound)	// We found it
   236 				{
   237 				aStartTag = start + result;
   238 				aTagLength = length;
   239 				return ETrue;
   240 				}
   241 			}
   242 		if (end == start + segmentLength)	// There's no more text at all
   243 			break;
   244 		// The buffer is segmented and there's another segment 
   245 
   246 		// It could be over the segment boundary
   247 		TInt start2 = start + segmentLength;
   248 		TInt end2 = start2;
   249 		if (segmentLength >= length - 1)
   250 			start2 -= length - 1;
   251 		else start2 = start;
   252 		if (end >= end2 + length - 1)
   253 			end2 += length -1;
   254 		else
   255 			end2 = end;
   256 		if (end2 - start2 >= length)
   257 			{
   258 			// Create a buffer with the end of one and the start of the other
   259 			TBuf<10> bridgeBuffer;
   260 			aTextObj.Extract(bridgeBuffer, start2, end2 - start2);
   261 			TInt result = bridgeBuffer.FindF(targetText);
   262 			if (result != KErrNotFound)	// We found it
   263 				{
   264 				aStartTag = start2 + result;
   265 				aTagLength = length;
   266 				return ETrue;
   267 				}
   268 			}
   269 
   270 		// Move start up for next segment
   271 		start += segmentLength;
   272 		}
   273 		return EFalse;	// Not enough text left in buffer
   274 	}
   275 
   276 
   277 const TDesC& CTestParser2::CreateDoItText(const CRichText& /* aTextObj */,
   278 							TInt /* aStartText */, TInt /* aLength */)
   279 	{
   280 	return *iDoItText;
   281 	}
   282 
   283 
   284 void CTestParser2::ActivateThisTextL(const CRichText& /* aTextObj */,
   285 					  TInt /* aStartText */, TInt /* aLength */)
   286 	{
   287 	// Do something?
   288 
   289 	}
   290 
   291 
   292 void CTestParser2::GetRecogniseFormat(TCharFormat& aFormat)
   293 	{
   294 	aFormat.iFontPresentation.iTextColor = KRgbRed;
   295 	aFormat.iFontPresentation.iUnderline = EUnderlineOn;
   296 	}
   297 
   298 
   299 void CTestParser2::GetRolloverFormat(TCharFormat& aFormat)
   300 	{
   301 	aFormat.iFontPresentation.iTextColor = KRgbRed;
   302 	aFormat.iFontPresentation.iUnderline = EUnderlineOn;
   303 	aFormat.iFontPresentation.iHighlightColor = KRgbDarkRed;
   304 	aFormat.iFontPresentation.iHighlightStyle = TFontPresentation::EFontHighlightRounded;
   305 	}
   306 
   307 
   308 void CTestParser2::Release()
   309 	{
   310 	delete this;
   311 	}
   312 
   313 
   314 //--------------------------------------------
   315 
   316 
   317 CEditObserver::CEditObserver()
   318 	{
   319 	iStart = 0;
   320 	iExtent = 0;
   321 	}
   322 
   323 
   324 CEditObserver::~CEditObserver()
   325 	{
   326 	}
   327 
   328 
   329 void CEditObserver::EditObserver(TInt aStart, TInt aExtent)
   330 	{
   331 	iStart = aStart;
   332 	iExtent = aExtent;
   333 	}
   334 
   335 
   336 void Test1()
   337 	{
   338 	INFO_PRINTF1(_L("Install 5, deinstall in reverse order"));
   339 	CTestParser* parser1 = CTestParser::NewL();
   340 	CTestParser* parser2 = CTestParser::NewL();
   341 	CTestParser* parser3 = CTestParser::NewL();
   342 	CTestParser* parser4 = CTestParser::NewL();
   343 	CTestParser* parser5 = CTestParser::NewL();
   344 	CRichText::ActivateParserL(parser1);	// List 1
   345 	CRichText::ActivateParserL(parser2);	// List 1, 2
   346 	CRichText::ActivateParserL(parser3);	// List 1, 2, 3
   347 	CRichText::ActivateParserL(parser4);	// List 1, 2, 3, 4
   348 	CRichText::ActivateParserL(parser5);	// List 1, 2, 3, 4, 5
   349 	CRichText::DeactivateParser(parser5);	// List 1, 2, 3, 4
   350 	CRichText::DeactivateParser(parser4);	// List 1, 2, 3
   351 	CRichText::DeactivateParser(parser3);	// List 1, 2
   352 	CRichText::DeactivateParser(parser2);	// List 1
   353 	CRichText::DeactivateParser(parser1);	// List empty
   354 	delete parser5;
   355 	delete parser4;
   356 	delete parser3;
   357 	delete parser2;
   358 	delete parser1;
   359 	CRichText::DeactivateParserDefaults();
   360 	}
   361 
   362 
   363 void Test2()
   364 	{
   365 	INFO_PRINTF1(_L("Install, deinstall in interleaved order"));
   366 	CTestParser* parser1 = CTestParser::NewL();
   367 	CRichText::ActivateParserL(parser1);	// List 1
   368 	CTestParser* parser2 = CTestParser::NewL();
   369 	CRichText::ActivateParserL(parser2);	// List 1, 2
   370 	// Remove first in list
   371 	CRichText::DeactivateParser(parser1);	// List 2
   372 	delete parser1;
   373 	CTestParser* parser3 = CTestParser::NewL();
   374 	CRichText::ActivateParserL(parser3);	// List 2, 3
   375 	// Remove last in list
   376 	CRichText::DeactivateParser(parser3);	// List 2
   377 	delete parser3;
   378 	CTestParser* parser4 = CTestParser::NewL();
   379 	CRichText::ActivateParserL(parser4);	// List 2, 4
   380 	CTestParser* parser5 = CTestParser::NewL();
   381 	CRichText::ActivateParserL(parser5);	// List 2, 4, 5
   382 	// Remove middle in list
   383 	CRichText::DeactivateParser(parser4);	// List 2, 5
   384 	delete parser4;
   385 	// Empty list
   386 	CRichText::DeactivateParser(parser5);	// List 2
   387 	delete parser5;
   388 	CRichText::DeactivateParser(parser2);	// List empty
   389 	delete parser2;
   390 	CRichText::DeactivateParserDefaults();
   391 	}
   392 
   393 
   394 void Test3()
   395 	{
   396 	INFO_PRINTF1(_L("Testing memory with OOM"));
   397 
   398 	TInt i;
   399 	TInt ret;
   400 	TInt count = 0;
   401 	TInt success = 0;
   402 
   403 	for (i = 0; i < 20; i++)
   404 		{
   405 		CTestParser* parser1 = NULL;
   406 		CTestParser* parser2 = NULL;
   407 		CTestParser* parser3 = NULL;
   408 		CTestParser* parser4 = NULL;
   409 		CTestParser* parser5 = NULL;
   410 		// Switch on memory problems, varying fail rate from 20 to 1
   411 		__UHEAP_SETFAIL(RHeap::EDeterministic, 20 - i);
   412 		__UHEAP_MARK;
   413 		count++;
   414 		TRAP(ret, parser1 = CTestParser::NewL());
   415 		if (ret != KErrNone)
   416 			{
   417 			parser1 = NULL;
   418 			}
   419 		else
   420 			{
   421 			TRAP(ret, CRichText::ActivateParserL(parser1));
   422 			if (ret != KErrNone)
   423 				{
   424 				delete parser1;
   425 				parser1 = NULL;
   426 				}
   427 			}
   428 		TRAP(ret, parser2 = CTestParser::NewL());
   429 		if (ret != KErrNone)
   430 			{
   431 			parser2 = NULL;
   432 			}
   433 		else
   434 			{
   435 			TRAP(ret, CRichText::ActivateParserL(parser2));
   436 			if (ret != KErrNone)
   437 				{
   438 				delete parser2;
   439 				parser2 = NULL;
   440 				}
   441 			}
   442 		TRAP(ret, parser3 = CTestParser::NewL());
   443 		if (ret != KErrNone)
   444 			{
   445 			parser3 = NULL;
   446 			}
   447 		else
   448 			{
   449 			TRAP(ret, CRichText::ActivateParserL(parser3));
   450 			if (ret != KErrNone)
   451 				{
   452 				delete parser3;
   453 				parser3 = NULL;
   454 				}
   455 			}
   456 		TRAP(ret, parser4 = CTestParser::NewL());
   457 		if (ret != KErrNone)
   458 			{
   459 			parser4 = NULL;
   460 			}
   461 		else
   462 			{
   463 			TRAP(ret, CRichText::ActivateParserL(parser4));
   464 			if (ret != KErrNone)
   465 				{
   466 				delete parser4;
   467 				parser4 = NULL;
   468 				}
   469 			}
   470 		TRAP(ret, parser5 = CTestParser::NewL());
   471 		if (ret != KErrNone)
   472 			{
   473 			parser5 = NULL;
   474 			}
   475 		else
   476 			{
   477 			TRAP(ret, CRichText::ActivateParserL(parser5));
   478 			if (ret != KErrNone)
   479 				{
   480 				delete parser5;
   481 				parser5 = NULL;
   482 				}
   483 			}
   484 		if (parser1)
   485 			{
   486 			success++;
   487 			CRichText::DeactivateParser(parser1);
   488 			delete parser1;
   489 			}
   490 		if (parser2)
   491 			{
   492 			success++;
   493 			CRichText::DeactivateParser(parser2);
   494 			delete parser2;
   495 			}
   496 		if (parser3)
   497 			{
   498 			success++;
   499 			CRichText::DeactivateParser(parser3);
   500 			delete parser3;
   501 			}
   502 		if (parser4)
   503 			{
   504 			success++;
   505 			CRichText::DeactivateParser(parser4);
   506 			delete parser4;
   507 			}
   508 		if (parser5)
   509 			{
   510 			success++;
   511 			CRichText::DeactivateParser(parser5);
   512 			delete parser5;
   513 			}
   514 		CRichText::DeactivateParserDefaults();
   515 		// Switch off memory problems
   516 		__UHEAP_MARKEND;
   517 		__UHEAP_RESET;
   518 		}
   519 
   520 	INFO_PRINTF3(_L("%d attempted activations, %d successful\n"), 5 * count, success);
   521 	}
   522 
   523 
   524 void Test4()
   525 	{
   526 	// Create a block of 1000 chars
   527 	// Randomly insert a target string and check:
   528 	// - Can't find target that is not complete
   529 	// - Does find complete target in right place
   530 	// - Once target is removed, can't find it
   531 	// repeat x 100
   532 	INFO_PRINTF1(_L("Testing EText behaviour with active parsers and single target"));
   533 	// Create and activate a parser
   534 	CTestParser* parser1 = CTestParser::NewL();
   535 	CRichText::ActivateParserL(parser1);
   536 	// Create a CRichText
   537 	CParaFormatLayer* GlobalParaFormatLayer = CParaFormatLayer::NewL();
   538 	CCharFormatLayer* GlobalCharFormatLayer = CCharFormatLayer::NewL();
   539 	CRichText* richText = CRichText::NewL(GlobalParaFormatLayer, GlobalCharFormatLayer, CEditableText::ESegmentedStorage, CEditableText::EDefaultTextGranularity);
   540 	richText->SetPictureFactory(NULL, NULL);  // forces index generation
   541 	// Switch on parsers for this CRichText
   542 	CEditObserver* editObserver = new (ELeave) CEditObserver;
   543 	richText->SetEditObserver(editObserver);
   544 
   545 	// Insert 1000 chars (repeated string)
   546 	TInt i;
   547 	for (i = 0; i < 100; i++)
   548 		richText->InsertL(richText->DocumentLength(), _L("abcdefghij"));
   549 	TInt64 seed = 314159;
   550 	TInt startTags;
   551 	TInt lengthTags;
   552 	for (i = 0; i < 100; i++)
   553 		{
   554 #if 0
   555 		INFO_PRINTF2(_L("i=%d"), i);
   556 #endif
   557 		// Get a random in range 0-999
   558 		TInt random = Math::Rand(seed) % 1000;
   559 		// "Randomly" insert target string that is not complete
   560 		richText->InsertL(random, _L("TARGE"));
   561 		test(!richText->ParseText(startTags, lengthTags, EFalse));
   562 		test(!richText->ParseText(startTags, lengthTags, ETrue));
   563 		// Complete target string and check we find single target where we expect
   564 		richText->InsertL(random + 5, 'T');
   565 #if 1
   566 		test(richText->ParseText(startTags, lengthTags, EFalse));
   567 #else
   568 		TBool b = richText->ParseText(startTags, lengthTags, EFalse);
   569 		INFO_PRINTF2(_L("    b=%d"), b);
   570 		test(b);
   571 #endif
   572 		test((startTags == random) && (lengthTags == 6));
   573 		test(richText->ParseText(startTags, lengthTags, ETrue));
   574 		test((startTags == random) && (lengthTags == 6));
   575 		// Completely remove target string
   576 		richText->DeleteL(random, 6);
   577 		test(!richText->ParseText(startTags, lengthTags, EFalse));
   578 		test(!richText->ParseText(startTags, lengthTags, ETrue));
   579 		}
   580 
   581 	// Clean up
   582 
   583 	delete richText;
   584 	delete GlobalCharFormatLayer;
   585 	delete GlobalParaFormatLayer;
   586 	CRichText::DeactivateParser(parser1);
   587 	delete parser1;
   588 	CRichText::DeactivateParserDefaults();
   589 	delete editObserver;
   590 	}
   591 
   592 
   593 void Test5()
   594 	{
   595 	// Create a block of 1000 chars
   596 	// Randomly insert two target strings and check:
   597 	// - Can't find targets that are not complete
   598 	// - Does find complete targets with exact range covered
   599 	// - Once targets are removed, can't find it
   600 	// repeat x 100
   601 	INFO_PRINTF1(_L("Testing EText behaviour with active parsers and double target"));
   602 	// Create and activate a parser
   603 	CTestParser* parser1 = CTestParser::NewL();
   604 	CRichText::ActivateParserL(parser1);
   605 	// Create a CRichText
   606 	CParaFormatLayer* GlobalParaFormatLayer = CParaFormatLayer::NewL();
   607 	CCharFormatLayer* GlobalCharFormatLayer = CCharFormatLayer::NewL();
   608 	CRichText* richText = CRichText::NewL(GlobalParaFormatLayer, GlobalCharFormatLayer, CEditableText::ESegmentedStorage, CEditableText::EDefaultTextGranularity);
   609 	richText->SetPictureFactory(NULL, NULL);  // forces index generation
   610 	// Switch on parsers for this CRichText
   611 	CEditObserver* editObserver = new (ELeave) CEditObserver;
   612 	richText->SetEditObserver(editObserver);
   613 
   614 	// Insert 1000 chars (repeated string)
   615 	TInt i;
   616 	for (i = 0; i < 100; i++)
   617 		richText->InsertL(richText->DocumentLength(), _L("abcdefghij"));
   618 	TInt64 seed1 = 314159;
   619 	TInt startTags;
   620 	TInt lengthTags;
   621 	for (i = 0; i < 100; i++)
   622 		{
   623 #if 0
   624 		INFO_PRINTF2(_L("i=%d"), i);
   625 #endif
   626 		// Get a random in range 0-999
   627 		TInt random1 = Math::Rand(seed1) % 1000;
   628 		TInt random2 = Math::Rand(seed1) % 1000;
   629 		TInt rlow = (random1 < random2) ? random1 : random2;
   630 		TInt rhigh = (random1 > random2) ? random1 : random2;
   631 		if (rlow + 7 > rhigh)
   632 			{ // Too close, spread them out
   633 			if (rhigh + 7 <= richText->DocumentLength())
   634 				rhigh += 7;
   635 			else
   636 				rlow -= 7;
   637 			}
   638 		// "Randomly" insert target strings that are not complete
   639 		richText->InsertL(rlow, _L("TARGE"));
   640 		richText->InsertL(rhigh - 1, _L("TARGE"));
   641 		test(!richText->ParseText(startTags, lengthTags, EFalse));
   642 		test(!richText->ParseText(startTags, lengthTags, ETrue));
   643 		// Complete target string and check we find single target where we expect
   644 		richText->InsertL(rlow + 5, 'T');
   645 		richText->InsertL(rhigh + 5, 'T');
   646 #if 1
   647 		test(richText->ParseText(startTags, lengthTags, EFalse));
   648 #else
   649 		TBool bb = richText->ParseText(startTags, lengthTags, EFalse);
   650 		test(bb);
   651 #endif
   652 		test((startTags == rlow) && (lengthTags == rhigh + 6 - rlow));
   653 		test(richText->ParseText(startTags, lengthTags, ETrue));
   654 		test((startTags == rlow) && (lengthTags == rhigh + 6 - rlow));
   655 		// Completely remove target string
   656 		richText->DeleteL(rhigh, 6);
   657 		richText->DeleteL(rlow, 6);
   658 		test(!richText->ParseText(startTags, lengthTags, EFalse));
   659 		test(!richText->ParseText(startTags, lengthTags, ETrue));
   660 		}
   661 
   662 	// Clean up
   663 
   664 	delete richText;
   665 	delete GlobalCharFormatLayer;
   666 	delete GlobalParaFormatLayer;
   667 	CRichText::DeactivateParser(parser1);
   668 	delete parser1;
   669 	CRichText::DeactivateParserDefaults();
   670 	delete editObserver;
   671 	}
   672 
   673 /**
   674 @SYMTestCaseID          SYSLIB-ETEXT-UT-3405
   675 @SYMTestCaseDesc	    Test for the fix for when CRichText crashes when parsers are active
   676 @SYMTestPriority 	    Medium
   677 @SYMTestActions  	    Calls PositionOfNextTag on all available document positions to validate fix.
   678 @SYMTestExpectedResults Test must be able to call PositionOfNextTag on all document positions
   679 @SYMDEF                 PDEF102494
   680 */	
   681 
   682 void Test6()
   683 	{
   684 	INFO_PRINTF1(_L(" @SYMTestCaseID:SYSLIB-ETEXT-UT-3405 Calling PositionOfNextTag on an empty document "));
   685 	
   686 	// Create and activate a parser
   687 	CTestParser* parser1 = CTestParser::NewL();
   688 	CRichText::ActivateParserL(parser1);
   689 	// Create a CRichText
   690 	CParaFormatLayer* GlobalParaFormatLayer = CParaFormatLayer::NewL();
   691 	CCharFormatLayer* GlobalCharFormatLayer = CCharFormatLayer::NewL();
   692 	CRichText* richText = CRichText::NewL(GlobalParaFormatLayer, GlobalCharFormatLayer, CEditableText::ESegmentedStorage, CEditableText::EDefaultTextGranularity);
   693 	richText->SetPictureFactory(NULL, NULL);  // forces index generation
   694 	// Switch on parsers for this CRichText
   695 	CEditObserver* editObserver = new (ELeave) CEditObserver;
   696 	richText->SetEditObserver(editObserver);
   697 
   698 	//insert and format some text
   699 	richText->InsertL(richText->DocumentLength(), _L("abcdTARGET"));
   700 
   701 	TCharFormat charFormat;
   702 	TCharFormatMask charFormatMask;
   703 
   704 	richText->ApplyCharFormatL(charFormat, charFormatMask, 0, richText->DocumentLength());
   705 
   706 	TInt startTags;
   707 	TInt lengthTags;
   708 	richText->ParseText(startTags, lengthTags, ETrue);
   709 
   710     // Call position of next tag on all document positions
   711 	for (TInt i = 0; i <= richText->DocumentLength(); i++)
   712 		{
   713 		richText->PositionOfNextTag(i, parser1);
   714 		}
   715 	
   716 	// Clean up
   717 
   718 	delete richText;
   719 	delete GlobalCharFormatLayer;
   720 	delete GlobalParaFormatLayer;
   721 	CRichText::DeactivateParser(parser1);
   722 	delete parser1;
   723 	CRichText::DeactivateParserDefaults();
   724 	delete editObserver;
   725 	}
   726 
   727 /**
   728 @SYMTestCaseID          SYSLIB-ETEXT-UT-3406
   729 @SYMTestCaseDesc	    Test for the fix for when CRichText crashes when parsers are active
   730 @SYMTestPriority 	    Medium
   731 @SYMTestActions  	    Initialize two parsers. First parser has a target string of "TARGET",
   732                         and the second parser has a target string of "ARG". By parsing the
   733                         two overlapping (ARG is contained in TARGET) target strings, the changes
   734 						for this defect have full coverage.
   735 @SYMTestExpectedResults Test must be able to call PositionOfNextTag on the end of document positions
   736 @SYMDEF                 PDEF102494
   737 */
   738 void Test7()
   739 	{
   740 	INFO_PRINTF1(_L(" @SYMTestCaseID:SYSLIB-ETEXT-UT-3406 Testing EText behaviour with active parsers and single target "));
   741 	
   742 	// Create and activate a parser
   743 	CTestParser* parser1 = CTestParser::NewL();
   744 	CRichText::ActivateParserL(parser1);
   745 	CTestParser2* parser2 = CTestParser2::NewL();
   746 	CRichText::ActivateParserL(parser2);
   747 	// Create a CRichText
   748 	CParaFormatLayer* GlobalParaFormatLayer = CParaFormatLayer::NewL();
   749 	CCharFormatLayer* GlobalCharFormatLayer = CCharFormatLayer::NewL();
   750 	CRichText* richText = CRichText::NewL(GlobalParaFormatLayer, GlobalCharFormatLayer, CEditableText::ESegmentedStorage, CEditableText::EDefaultTextGranularity);
   751 	richText->SetPictureFactory(NULL, NULL);  // forces index generation
   752 	// Switch on parsers for this CRichText
   753 	CEditObserver* editObserver = new (ELeave) CEditObserver;
   754 	richText->SetEditObserver(editObserver);
   755 
   756 	// Insert 1000 chars (repeated string)
   757 	TInt i;
   758 	for (i = 0; i < 100; i++)
   759 		richText->InsertL(richText->DocumentLength(), _L("abcdefghij"));
   760 
   761 	TInt startTags;
   762 	TInt lengthTags;
   763 	TInt startPosition = 0;
   764 	
   765 	// Run test twice, once at start of string, once at end.
   766 	for (TInt i=0;i < 2;i++)
   767 		{
   768 		//Insert target string that is not complete (for code coverage purposes).
   769 		richText->InsertL(startPosition, _L("TARGE"));
   770 		// Parse range provided only. Parser two will set it's parser tags on "ARG".
   771 		test(richText->ParseText(startTags, lengthTags, EFalse));
   772 		// Ensure this call does not fail at end of document (anymore). This gets
   773 		// the position of the next tag at the end of document position. Should
   774 		// be none found at end of document.
   775 		test((richText->PositionOfNextTag(richText->DocumentLength(), parser1)) == -1);
   776 		
   777 		// Parse all text (for coverage purposes). This will first clear all tags,
   778 		// and then re-applies the tags again on "ARG"
   779 		test(richText->ParseText(startTags, lengthTags, ETrue));
   780 		test((richText->PositionOfNextTag(richText->DocumentLength(), parser1)) == -1);
   781 		
   782 		// Complete target string. The purpose of completing the target string
   783 		// here is for code coverage. The text has changed and the code path in 
   784 		// parselst.cpp will change.
   785 		richText->InsertL(startPosition + 5, 'T');
   786 
   787 		// Parse range provided only. The tags on "ARG" are considered
   788 		// lower precendence, so this will clear the parser tags on "ARG" and 
   789 		// will apply the parser tags on "TARGET". 
   790 		test(richText->ParseText(startTags, lengthTags, EFalse));
   791 		test((richText->PositionOfNextTag(richText->DocumentLength(), parser1)) == -1);
   792 				
   793 		// Completely remove target string
   794 		richText->DeleteL(startPosition, 6);
   795 		test(!richText->ParseText(startTags, lengthTags, EFalse));
   796 		test(!richText->ParseText(startTags, lengthTags, ETrue));
   797 		
   798 		startPosition = richText->DocumentLength();
   799 		}
   800 
   801 	// Clean up
   802 
   803 	delete richText;
   804 	delete GlobalCharFormatLayer;
   805 	delete GlobalParaFormatLayer;
   806 	CRichText::DeactivateParser(parser1);
   807 	delete parser1;
   808 	CRichText::DeactivateParser(parser2);
   809 	delete parser2;
   810 	CRichText::DeactivateParserDefaults();
   811 	delete editObserver;
   812 	}
   813 
   814 CT_PARSE::CT_PARSE()
   815     {
   816     SetTestStepName(KTestStep_T_PARSE);
   817     pTestStep = this;
   818     }
   819 
   820 TVerdict CT_PARSE::doTestStepL()
   821     {
   822     SetTestStepResult(EFail);
   823 
   824     TrapCleanup = CTrapCleanup::New();
   825     TRAPD(r,\
   826         {\
   827         for (TInt i=KTestCleanupStack;i>0;i--)\
   828             CleanupStack::PushL((TAny*)1);\
   829         test(r==KErrNone);\
   830         CleanupStack::Pop(KTestCleanupStack);\
   831         });
   832 
   833     INFO_PRINTF1(_L("Testing EText parser system"));
   834 
   835     INFO_PRINTF1(_L("Testing memory under normal conditions"));
   836 
   837     __UHEAP_MARK;
   838     TRAPD(ret1, Test1());
   839     __UHEAP_MARKEND;
   840 
   841     __UHEAP_MARK;
   842     TRAPD(ret2, Test2());
   843     __UHEAP_MARKEND;
   844 
   845     __UHEAP_MARK;
   846     TRAPD(ret3, Test3());
   847     __UHEAP_MARKEND;
   848 
   849     __UHEAP_MARK;
   850     TRAPD(ret4, Test4());
   851     __UHEAP_MARKEND;
   852 
   853     __UHEAP_MARK;
   854     TRAPD(ret5, Test5());
   855     __UHEAP_MARKEND;
   856 
   857     __UHEAP_MARK;
   858     TRAPD(ret6, Test6());
   859     __UHEAP_MARKEND;
   860     
   861     __UHEAP_MARK;
   862     TRAPD(ret7, Test7());
   863     __UHEAP_MARKEND;
   864     
   865     delete TrapCleanup;
   866 
   867     if (r == KErrNone && ret1 == KErrNone && ret2 == KErrNone && ret3 == KErrNone && ret4 == KErrNone && ret5 == KErrNone && ret6 == KErrNone && ret7 == KErrNone)
   868         {
   869         SetTestStepResult(EPass);
   870         }
   871 
   872     return TestStepResult();
   873     }