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