First public contribution.
2 * Copyright (c) 1999-2010 Nokia Corporation and/or its subsidiary(-ies).
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".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
23 #include "T_PARSE_for_TEF.h"
25 LOCAL_D CTestStep *pTestStep = NULL;
28 TBool __bb = (cond); \
29 pTestStep->TEST(__bb); \
32 pTestStep->ERR_PRINTF1(_L("ERROR: Test Failed")); \
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))
45 LOCAL_D CTrapCleanup* TrapCleanup;
46 LOCAL_D const TInt KTestCleanupStack=0x200;
49 CTestParser* CTestParser::NewL()
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?");
60 CTestParser::CTestParser()
65 CTestParser::~CTestParser()
71 TBool CTestParser::ParseThisText(const CRichText& aTextObj, TBool aAllowBack,
72 TInt aStartScan, TInt aScanLength,
73 TInt& aStartTag, TInt& aTagLength)
75 // Very simple - looking for the phrase "the world"
76 _LIT(targetText, "TARGET");
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;
83 if (start >= length - 1)
88 // And it might go beyond the end of the scan
89 TInt end = aStartScan + aScanLength;
90 if (end + length - 1 <= aTextObj.DocumentLength())
93 end = aTextObj.DocumentLength();
94 while (end - start >= length)
96 TPtrC buffer = aTextObj.Read(start, end - start);
97 TInt segmentLength= buffer.Length();
99 if (segmentLength >= length) // There's enough text to bother searching
101 TInt result = buffer.FindF(targetText);
102 if (result != KErrNotFound) // We found it
104 aStartTag = start + result;
109 if (end == start + segmentLength) // There's no more text at all
111 // The buffer is segmented and there's another segment
113 // It could be over the segment boundary
114 TInt start2 = start + segmentLength;
116 if (segmentLength >= length - 1)
117 start2 -= length - 1;
119 if (end >= end2 + length - 1)
123 if (end2 - start2 >= length)
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
131 aStartTag = start2 + result;
137 // Move start up for next segment
138 start += segmentLength;
140 return EFalse; // Not enough text left in buffer
144 const TDesC& CTestParser::CreateDoItText(const CRichText& /* aTextObj */,
145 TInt /* aStartText */, TInt /* aLength */)
151 void CTestParser::ActivateThisTextL(const CRichText& /* aTextObj */,
152 TInt /* aStartText */, TInt /* aLength */)
159 void CTestParser::GetRecogniseFormat(TCharFormat& aFormat)
161 aFormat.iFontPresentation.iTextColor = KRgbRed;
162 aFormat.iFontPresentation.iUnderline = EUnderlineOn;
166 void CTestParser::GetRolloverFormat(TCharFormat& aFormat)
168 aFormat.iFontPresentation.iTextColor = KRgbRed;
169 aFormat.iFontPresentation.iUnderline = EUnderlineOn;
170 aFormat.iFontPresentation.iHighlightColor = KRgbDarkRed;
171 aFormat.iFontPresentation.iHighlightStyle = TFontPresentation::EFontHighlightRounded;
175 void CTestParser::Release()
180 //--------------------------------------------
182 CTestParser2* CTestParser2::NewL()
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?");
193 CTestParser2::CTestParser2()
198 CTestParser2::~CTestParser2()
204 TBool CTestParser2::ParseThisText(const CRichText& aTextObj, TBool aAllowBack,
205 TInt aStartScan, TInt aScanLength,
206 TInt& aStartTag, TInt& aTagLength)
208 // Very simple - looking for the phrase "ARG"
209 _LIT(targetText, "ARG");
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;
216 if (start >= length - 1)
221 // And it might go beyond the end of the scan
222 TInt end = aStartScan + aScanLength;
223 if (end + length - 1 <= aTextObj.DocumentLength())
226 end = aTextObj.DocumentLength();
227 while (end - start >= length)
229 TPtrC buffer = aTextObj.Read(start, end - start);
230 TInt segmentLength= buffer.Length();
232 if (segmentLength >= length) // There's enough text to bother searching
234 TInt result = buffer.FindF(targetText);
235 if (result != KErrNotFound) // We found it
237 aStartTag = start + result;
242 if (end == start + segmentLength) // There's no more text at all
244 // The buffer is segmented and there's another segment
246 // It could be over the segment boundary
247 TInt start2 = start + segmentLength;
249 if (segmentLength >= length - 1)
250 start2 -= length - 1;
252 if (end >= end2 + length - 1)
256 if (end2 - start2 >= length)
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
264 aStartTag = start2 + result;
270 // Move start up for next segment
271 start += segmentLength;
273 return EFalse; // Not enough text left in buffer
277 const TDesC& CTestParser2::CreateDoItText(const CRichText& /* aTextObj */,
278 TInt /* aStartText */, TInt /* aLength */)
284 void CTestParser2::ActivateThisTextL(const CRichText& /* aTextObj */,
285 TInt /* aStartText */, TInt /* aLength */)
292 void CTestParser2::GetRecogniseFormat(TCharFormat& aFormat)
294 aFormat.iFontPresentation.iTextColor = KRgbRed;
295 aFormat.iFontPresentation.iUnderline = EUnderlineOn;
299 void CTestParser2::GetRolloverFormat(TCharFormat& aFormat)
301 aFormat.iFontPresentation.iTextColor = KRgbRed;
302 aFormat.iFontPresentation.iUnderline = EUnderlineOn;
303 aFormat.iFontPresentation.iHighlightColor = KRgbDarkRed;
304 aFormat.iFontPresentation.iHighlightStyle = TFontPresentation::EFontHighlightRounded;
308 void CTestParser2::Release()
314 //--------------------------------------------
317 CEditObserver::CEditObserver()
324 CEditObserver::~CEditObserver()
329 void CEditObserver::EditObserver(TInt aStart, TInt aExtent)
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
359 CRichText::DeactivateParserDefaults();
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
373 CTestParser* parser3 = CTestParser::NewL();
374 CRichText::ActivateParserL(parser3); // List 2, 3
375 // Remove last in list
376 CRichText::DeactivateParser(parser3); // List 2
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
386 CRichText::DeactivateParser(parser5); // List 2
388 CRichText::DeactivateParser(parser2); // List empty
390 CRichText::DeactivateParserDefaults();
396 INFO_PRINTF1(_L("Testing memory with OOM"));
403 for (i = 0; i < 20; i++)
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);
414 TRAP(ret, parser1 = CTestParser::NewL());
421 TRAP(ret, CRichText::ActivateParserL(parser1));
428 TRAP(ret, parser2 = CTestParser::NewL());
435 TRAP(ret, CRichText::ActivateParserL(parser2));
442 TRAP(ret, parser3 = CTestParser::NewL());
449 TRAP(ret, CRichText::ActivateParserL(parser3));
456 TRAP(ret, parser4 = CTestParser::NewL());
463 TRAP(ret, CRichText::ActivateParserL(parser4));
470 TRAP(ret, parser5 = CTestParser::NewL());
477 TRAP(ret, CRichText::ActivateParserL(parser5));
487 CRichText::DeactivateParser(parser1);
493 CRichText::DeactivateParser(parser2);
499 CRichText::DeactivateParser(parser3);
505 CRichText::DeactivateParser(parser4);
511 CRichText::DeactivateParser(parser5);
514 CRichText::DeactivateParserDefaults();
515 // Switch off memory problems
520 INFO_PRINTF3(_L("%d attempted activations, %d successful\n"), 5 * count, success);
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
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);
545 // Insert 1000 chars (repeated string)
547 for (i = 0; i < 100; i++)
548 richText->InsertL(richText->DocumentLength(), _L("abcdefghij"));
549 TInt64 seed = 314159;
552 for (i = 0; i < 100; i++)
555 INFO_PRINTF2(_L("i=%d"), i);
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');
566 test(richText->ParseText(startTags, lengthTags, EFalse));
568 TBool b = richText->ParseText(startTags, lengthTags, EFalse);
569 INFO_PRINTF2(_L(" b=%d"), b);
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));
584 delete GlobalCharFormatLayer;
585 delete GlobalParaFormatLayer;
586 CRichText::DeactivateParser(parser1);
588 CRichText::DeactivateParserDefaults();
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
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);
614 // Insert 1000 chars (repeated string)
616 for (i = 0; i < 100; i++)
617 richText->InsertL(richText->DocumentLength(), _L("abcdefghij"));
618 TInt64 seed1 = 314159;
621 for (i = 0; i < 100; i++)
624 INFO_PRINTF2(_L("i=%d"), i);
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())
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');
647 test(richText->ParseText(startTags, lengthTags, EFalse));
649 TBool bb = richText->ParseText(startTags, lengthTags, EFalse);
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));
665 delete GlobalCharFormatLayer;
666 delete GlobalParaFormatLayer;
667 CRichText::DeactivateParser(parser1);
669 CRichText::DeactivateParserDefaults();
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
684 INFO_PRINTF1(_L(" @SYMTestCaseID:SYSLIB-ETEXT-UT-3405 Calling PositionOfNextTag on an empty document "));
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);
698 //insert and format some text
699 richText->InsertL(richText->DocumentLength(), _L("abcdTARGET"));
701 TCharFormat charFormat;
702 TCharFormatMask charFormatMask;
704 richText->ApplyCharFormatL(charFormat, charFormatMask, 0, richText->DocumentLength());
708 richText->ParseText(startTags, lengthTags, ETrue);
710 // Call position of next tag on all document positions
711 for (TInt i = 0; i <= richText->DocumentLength(); i++)
713 richText->PositionOfNextTag(i, parser1);
719 delete GlobalCharFormatLayer;
720 delete GlobalParaFormatLayer;
721 CRichText::DeactivateParser(parser1);
723 CRichText::DeactivateParserDefaults();
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
740 INFO_PRINTF1(_L(" @SYMTestCaseID:SYSLIB-ETEXT-UT-3406 Testing EText behaviour with active parsers and single target "));
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);
756 // Insert 1000 chars (repeated string)
758 for (i = 0; i < 100; i++)
759 richText->InsertL(richText->DocumentLength(), _L("abcdefghij"));
763 TInt startPosition = 0;
765 // Run test twice, once at start of string, once at end.
766 for (TInt i=0;i < 2;i++)
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);
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);
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');
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);
793 // Completely remove target string
794 richText->DeleteL(startPosition, 6);
795 test(!richText->ParseText(startTags, lengthTags, EFalse));
796 test(!richText->ParseText(startTags, lengthTags, ETrue));
798 startPosition = richText->DocumentLength();
804 delete GlobalCharFormatLayer;
805 delete GlobalParaFormatLayer;
806 CRichText::DeactivateParser(parser1);
808 CRichText::DeactivateParser(parser2);
810 CRichText::DeactivateParserDefaults();
816 SetTestStepName(KTestStep_T_PARSE);
820 TVerdict CT_PARSE::doTestStepL()
822 SetTestStepResult(EFail);
824 TrapCleanup = CTrapCleanup::New();
827 for (TInt i=KTestCleanupStack;i>0;i--)\
828 CleanupStack::PushL((TAny*)1);\
830 CleanupStack::Pop(KTestCleanupStack);\
833 INFO_PRINTF1(_L("Testing EText parser system"));
835 INFO_PRINTF1(_L("Testing memory under normal conditions"));
838 TRAPD(ret1, Test1());
842 TRAPD(ret2, Test2());
846 TRAPD(ret3, Test3());
850 TRAPD(ret4, Test4());
854 TRAPD(ret5, Test5());
858 TRAPD(ret6, Test6());
862 TRAPD(ret7, Test7());
867 if (r == KErrNone && ret1 == KErrNone && ret2 == KErrNone && ret3 == KErrNone && ret4 == KErrNone && ret5 == KErrNone && ret6 == KErrNone && ret7 == KErrNone)
869 SetTestStepResult(EPass);
872 return TestStepResult();