| /**************************************************************************************** |
| * COPYRIGHT: |
| * Copyright (c) 1997-2014, International Business Machines Corporation and |
| * others. All Rights Reserved. |
| * Modification History: |
| * |
| * Date Name Description |
| * 05/22/2000 Madhu Added tests for testing new API for utf16 support and more |
| ****************************************************************************************/ |
| |
| #include <string.h> |
| #include "utypeinfo.h" // for 'typeid' to work |
| |
| #include "unicode/chariter.h" |
| #include "unicode/ustring.h" |
| #include "unicode/unistr.h" |
| #include "unicode/schriter.h" |
| #include "unicode/uchriter.h" |
| #include "unicode/uiter.h" |
| #include "unicode/putil.h" |
| #include "unicode/utf16.h" |
| #include "citrtest.h" |
| #include "cmemory.h" |
| |
| |
| class SCharacterIterator : public CharacterIterator { |
| public: |
| SCharacterIterator(const UnicodeString& textStr){ |
| text = textStr; |
| pos=0; |
| textLength = textStr.length(); |
| begin = 0; |
| end=textLength; |
| |
| } |
| |
| virtual ~SCharacterIterator(){}; |
| |
| |
| void setText(const UnicodeString& newText){ |
| text = newText; |
| } |
| |
| virtual void getText(UnicodeString& result) { |
| text.extract(0,text.length(),result); |
| } |
| static UClassID getStaticClassID(void){ |
| return (UClassID)(&fgClassID); |
| } |
| virtual UClassID getDynamicClassID(void) const{ |
| return getStaticClassID(); |
| } |
| |
| virtual UBool operator==(const ForwardCharacterIterator& /*that*/) const{ |
| return TRUE; |
| } |
| |
| virtual CharacterIterator* clone(void) const { |
| return NULL; |
| } |
| virtual int32_t hashCode(void) const{ |
| return DONE; |
| } |
| virtual UChar nextPostInc(void){ return text.charAt(pos++);} |
| virtual UChar32 next32PostInc(void){return text.char32At(pos++);} |
| virtual UBool hasNext() { return TRUE;}; |
| virtual UChar first(){return DONE;}; |
| virtual UChar32 first32(){return DONE;}; |
| virtual UChar last(){return DONE;}; |
| virtual UChar32 last32(){return DONE;}; |
| virtual UChar setIndex(int32_t /*pos*/){return DONE;}; |
| virtual UChar32 setIndex32(int32_t /*pos*/){return DONE;}; |
| virtual UChar current() const{return DONE;}; |
| virtual UChar32 current32() const{return DONE;}; |
| virtual UChar next(){return DONE;}; |
| virtual UChar32 next32(){return DONE;}; |
| virtual UChar previous(){return DONE;}; |
| virtual UChar32 previous32(){return DONE;}; |
| virtual int32_t move(int32_t delta,CharacterIterator::EOrigin origin){ |
| switch(origin) { |
| case kStart: |
| pos = begin + delta; |
| break; |
| case kCurrent: |
| pos += delta; |
| break; |
| case kEnd: |
| pos = end + delta; |
| break; |
| default: |
| break; |
| } |
| |
| if(pos < begin) { |
| pos = begin; |
| } else if(pos > end) { |
| pos = end; |
| } |
| |
| return pos; |
| }; |
| virtual int32_t move32(int32_t delta, CharacterIterator::EOrigin origin){ |
| switch(origin) { |
| case kStart: |
| pos = begin; |
| if(delta > 0) { |
| U16_FWD_N(text, pos, end, delta); |
| } |
| break; |
| case kCurrent: |
| if(delta > 0) { |
| U16_FWD_N(text, pos, end, delta); |
| } else { |
| U16_BACK_N(text, begin, pos, -delta); |
| } |
| break; |
| case kEnd: |
| pos = end; |
| if(delta < 0) { |
| U16_BACK_N(text, begin, pos, -delta); |
| } |
| break; |
| default: |
| break; |
| } |
| |
| return pos; |
| }; |
| virtual UBool hasPrevious() {return TRUE;}; |
| |
| SCharacterIterator& operator=(const SCharacterIterator& that){ |
| text = that.text; |
| return *this; |
| } |
| |
| |
| private: |
| UnicodeString text; |
| static const char fgClassID; |
| }; |
| const char SCharacterIterator::fgClassID=0; |
| |
| CharIterTest::CharIterTest() |
| { |
| } |
| void CharIterTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ ) |
| { |
| if (exec) logln("TestSuite CharIterTest: "); |
| switch (index) { |
| case 0: name = "TestConstructionAndEquality"; if (exec) TestConstructionAndEquality(); break; |
| case 1: name = "TestConstructionAndEqualityUChariter"; if (exec) TestConstructionAndEqualityUChariter(); break; |
| case 2: name = "TestIteration"; if (exec) TestIteration(); break; |
| case 3: name = "TestIterationUChar32"; if (exec) TestIterationUChar32(); break; |
| case 4: name = "TestUCharIterator"; if (exec) TestUCharIterator(); break; |
| case 5: name = "TestCoverage"; if(exec) TestCoverage(); break; |
| case 6: name = "TestCharIteratorSubClasses"; if (exec) TestCharIteratorSubClasses(); break; |
| default: name = ""; break; //needed to end loop |
| } |
| } |
| |
| void CharIterTest::TestCoverage(){ |
| UnicodeString testText("Now is the time for all good men to come to the aid of their country."); |
| UnicodeString testText2("\\ud800\\udc01deadbeef"); |
| testText2 = testText2.unescape(); |
| SCharacterIterator* test = new SCharacterIterator(testText); |
| if(test->firstPostInc()!= 0x004E){ |
| errln("Failed: firstPostInc() failed"); |
| } |
| if(test->getIndex()!=1){ |
| errln("Failed: getIndex()."); |
| } |
| if(test->getLength()!=testText.length()){ |
| errln("Failed: getLength()"); |
| } |
| test->setToStart(); |
| if(test->getIndex()!=0){ |
| errln("Failed: setToStart()."); |
| } |
| test->setToEnd(); |
| if(test->getIndex()!=testText.length()){ |
| errln("Failed: setToEnd()."); |
| } |
| if(test->startIndex() != 0){ |
| errln("Failed: startIndex()"); |
| } |
| test->setText(testText2); |
| if(test->first32PostInc()!= testText2.char32At(0)){ |
| errln("Failed: first32PostInc() failed"); |
| } |
| |
| delete test; |
| |
| } |
| void CharIterTest::TestConstructionAndEquality() { |
| UnicodeString testText("Now is the time for all good men to come to the aid of their country."); |
| UnicodeString testText2("Don't bother using this string."); |
| UnicodeString result1, result2, result3; |
| |
| CharacterIterator* test1 = new StringCharacterIterator(testText); |
| CharacterIterator* test1b= new StringCharacterIterator(testText, -1); |
| CharacterIterator* test1c= new StringCharacterIterator(testText, 100); |
| CharacterIterator* test1d= new StringCharacterIterator(testText, -2, 100, 5); |
| CharacterIterator* test1e= new StringCharacterIterator(testText, 100, 20, 5); |
| CharacterIterator* test2 = new StringCharacterIterator(testText, 5); |
| CharacterIterator* test3 = new StringCharacterIterator(testText, 2, 20, 5); |
| CharacterIterator* test4 = new StringCharacterIterator(testText2); |
| CharacterIterator* test5 = test1->clone(); |
| |
| if (test1d->startIndex() < 0) |
| errln("Construction failed: startIndex is negative"); |
| if (test1d->endIndex() > testText.length()) |
| errln("Construction failed: endIndex is greater than the text length"); |
| if (test1d->getIndex() < test1d->startIndex() || test1d->endIndex() < test1d->getIndex()) |
| errln("Construction failed: index is invalid"); |
| |
| if (*test1 == *test2 || *test1 == *test3 || *test1 == *test4) |
| errln("Construction or operator== failed: Unequal objects compared equal"); |
| if (*test1 != *test5) |
| errln("clone() or equals() failed: Two clones tested unequal"); |
| |
| if (test1->hashCode() == test2->hashCode() || test1->hashCode() == test3->hashCode() |
| || test1->hashCode() == test4->hashCode()) |
| errln("hashCode() failed: different objects have same hash code"); |
| |
| if (test1->hashCode() != test5->hashCode()) |
| errln("hashCode() failed: identical objects have different hash codes"); |
| |
| if(test1->getLength() != testText.length()){ |
| errln("getLength of CharacterIterator failed"); |
| } |
| test1->getText(result1); |
| test1b->getText(result2); |
| test1c->getText(result3); |
| if(result1 != result2 || result1 != result3) |
| errln("construction failed or getText() failed"); |
| |
| |
| test1->setIndex(5); |
| if (*test1 != *test2 || *test1 == *test5) |
| errln("setIndex() failed"); |
| |
| *((StringCharacterIterator*)test1) = *((StringCharacterIterator*)test3); |
| if (*test1 != *test3 || *test1 == *test5) |
| errln("operator= failed"); |
| |
| delete test2; |
| delete test3; |
| delete test4; |
| delete test5; |
| delete test1b; |
| delete test1c; |
| delete test1d; |
| delete test1e; |
| |
| |
| StringCharacterIterator* testChar1=new StringCharacterIterator(testText); |
| StringCharacterIterator* testChar2=new StringCharacterIterator(testText2); |
| StringCharacterIterator* testChar3=(StringCharacterIterator*)test1->clone(); |
| |
| testChar1->getText(result1); |
| testChar2->getText(result2); |
| testChar3->getText(result3); |
| if(result1 != result3 || result1 == result2) |
| errln("getText() failed"); |
| testChar3->setText(testText2); |
| testChar3->getText(result3); |
| if(result1 == result3 || result2 != result3) |
| errln("setText() or getText() failed"); |
| testChar3->setText(testText); |
| testChar3->getText(result3); |
| if(result1 != result3 || result1 == result2) |
| errln("setText() or getText() round-trip failed"); |
| |
| delete testChar1; |
| delete testChar2; |
| delete testChar3; |
| delete test1; |
| |
| } |
| void CharIterTest::TestConstructionAndEqualityUChariter() { |
| U_STRING_DECL(testText, "Now is the time for all good men to come to the aid of their country.", 69); |
| U_STRING_DECL(testText2, "Don't bother using this string.", 31); |
| |
| U_STRING_INIT(testText, "Now is the time for all good men to come to the aid of their country.", 69); |
| U_STRING_INIT(testText2, "Don't bother using this string.", 31); |
| |
| UnicodeString result, result4, result5; |
| |
| UCharCharacterIterator* test1 = new UCharCharacterIterator(testText, u_strlen(testText)); |
| UCharCharacterIterator* test2 = new UCharCharacterIterator(testText, u_strlen(testText), 5); |
| UCharCharacterIterator* test3 = new UCharCharacterIterator(testText, u_strlen(testText), 2, 20, 5); |
| UCharCharacterIterator* test4 = new UCharCharacterIterator(testText2, u_strlen(testText2)); |
| UCharCharacterIterator* test5 = (UCharCharacterIterator*)test1->clone(); |
| UCharCharacterIterator* test6 = new UCharCharacterIterator(*test1); |
| |
| // j785: length=-1 will use u_strlen() |
| UCharCharacterIterator* test7a = new UCharCharacterIterator(testText, -1); |
| UCharCharacterIterator* test7b = new UCharCharacterIterator(testText, -1); |
| UCharCharacterIterator* test7c = new UCharCharacterIterator(testText, -1, 2, 20, 5); |
| |
| // Bad parameters. |
| UCharCharacterIterator* test8a = new UCharCharacterIterator(testText, -1, -1, 20, 5); |
| UCharCharacterIterator* test8b = new UCharCharacterIterator(testText, -1, 2, 100, 5); |
| UCharCharacterIterator* test8c = new UCharCharacterIterator(testText, -1, 2, 20, 100); |
| |
| if (test8a->startIndex() < 0) |
| errln("Construction failed: startIndex is negative"); |
| if (test8b->endIndex() != u_strlen(testText)) |
| errln("Construction failed: endIndex is different from the text length"); |
| if (test8c->getIndex() < test8c->startIndex() || test8c->endIndex() < test8c->getIndex()) |
| errln("Construction failed: index is invalid"); |
| |
| if (*test1 == *test2 || *test1 == *test3 || *test1 == *test4 ) |
| errln("Construction or operator== failed: Unequal objects compared equal"); |
| if (*test1 != *test5 ) |
| errln("clone() or equals() failed: Two clones tested unequal"); |
| |
| if (*test6 != *test1 ) |
| errln("copy construction or equals() failed: Two copies tested unequal"); |
| |
| if (test1->hashCode() == test2->hashCode() || test1->hashCode() == test3->hashCode() |
| || test1->hashCode() == test4->hashCode()) |
| errln("hashCode() failed: different objects have same hash code"); |
| |
| if (test1->hashCode() != test5->hashCode()) |
| errln("hashCode() failed: identical objects have different hash codes"); |
| |
| test7a->getText(result); |
| test7b->getText(result4); |
| test7c->getText(result5); |
| |
| if(result != UnicodeString(testText) || result4 != result || result5 != result) |
| errln("error in construction"); |
| |
| test1->getText(result); |
| test4->getText(result4); |
| test5->getText(result5); |
| if(result != result5 || result == result4) |
| errln("getText() failed"); |
| test5->setText(testText2, u_strlen(testText2)); |
| test5->getText(result5); |
| if(result == result5 || result4 != result5) |
| errln("setText() or getText() failed"); |
| test5->setText(testText, u_strlen(testText)); |
| test5->getText(result5); |
| if(result != result5 || result == result4) |
| errln("setText() or getText() round-trip failed"); |
| |
| |
| test1->setIndex(5); |
| if (*test1 != *test2 || *test1 == *test5) |
| errln("setIndex() failed"); |
| test8b->setIndex32(5); |
| if (test8b->getIndex()!=5) |
| errln("setIndex32() failed"); |
| |
| *test1 = *test3; |
| if (*test1 != *test3 || *test1 == *test5) |
| errln("operator= failed"); |
| |
| delete test1; |
| delete test2; |
| delete test3; |
| delete test4; |
| delete test5; |
| delete test6; |
| delete test7a; |
| delete test7b; |
| delete test7c; |
| delete test8a; |
| delete test8b; |
| delete test8c; |
| } |
| |
| |
| void CharIterTest::TestIteration() { |
| UnicodeString text("Now is the time for all good men to come to the aid of their country."); |
| |
| UChar c; |
| int32_t i; |
| { |
| StringCharacterIterator iter(text, 5); |
| |
| UnicodeString iterText; |
| iter.getText(iterText); |
| if (iterText != text) |
| errln("iter.getText() failed"); |
| |
| if (iter.current() != text[(int32_t)5]) |
| errln("Iterator didn't start out in the right place."); |
| |
| c = iter.first(); |
| i = 0; |
| |
| if (iter.startIndex() != 0 || iter.endIndex() != text.length()) |
| errln("startIndex() or endIndex() failed"); |
| |
| logln("Testing forward iteration..."); |
| do { |
| if (c == CharacterIterator::DONE && i != text.length()) |
| errln("Iterator reached end prematurely"); |
| else if (c != text[i]) |
| errln((UnicodeString)"Character mismatch at position " + i + |
| ", iterator has " + UCharToUnicodeString(c) + |
| ", string has " + UCharToUnicodeString(text[i])); |
| |
| if (iter.current() != c) |
| errln("current() isn't working right"); |
| if (iter.getIndex() != i) |
| errln("getIndex() isn't working right"); |
| |
| if (c != CharacterIterator::DONE) { |
| c = iter.next(); |
| i++; |
| } |
| } while (c != CharacterIterator::DONE); |
| c=iter.next(); |
| if(c!= CharacterIterator::DONE) |
| errln("next() didn't return DONE at the end"); |
| c=iter.setIndex(text.length()+1); |
| if(c!= CharacterIterator::DONE) |
| errln("setIndex(len+1) didn't return DONE"); |
| |
| c = iter.last(); |
| i = text.length() - 1; |
| |
| logln("Testing backward iteration..."); |
| do { |
| if (c == CharacterIterator::DONE && i >= 0) |
| errln("Iterator reached end prematurely"); |
| else if (c != text[i]) |
| errln((UnicodeString)"Character mismatch at position " + i + |
| ", iterator has " + UCharToUnicodeString(c) + |
| ", string has " + UCharToUnicodeString(text[i])); |
| |
| if (iter.current() != c) |
| errln("current() isn't working right"); |
| if (iter.getIndex() != i) |
| errln("getIndex() isn't working right"); |
| if(iter.setIndex(i) != c) |
| errln("setIndex() isn't working right"); |
| |
| if (c != CharacterIterator::DONE) { |
| c = iter.previous(); |
| i--; |
| } |
| } while (c != CharacterIterator::DONE); |
| |
| c=iter.previous(); |
| if(c!= CharacterIterator::DONE) |
| errln("previous didn't return DONE at the beginning"); |
| |
| |
| //testing firstPostInc, nextPostInc, setTostart |
| i = 0; |
| c=iter.firstPostInc(); |
| if(c != text[i]) |
| errln((UnicodeString)"firstPostInc failed. Expected->" + |
| UCharToUnicodeString(text[i]) + " Got->" + UCharToUnicodeString(c)); |
| if(iter.getIndex() != i+1) |
| errln((UnicodeString)"getIndex() after firstPostInc() failed"); |
| |
| iter.setToStart(); |
| i=0; |
| if (iter.startIndex() != 0) |
| errln("setToStart failed"); |
| |
| logln("Testing forward iteration..."); |
| do { |
| if (c != CharacterIterator::DONE) |
| c = iter.nextPostInc(); |
| |
| if(c != text[i]) |
| errln((UnicodeString)"Character mismatch at position " + i + |
| (UnicodeString)", iterator has " + UCharToUnicodeString(c) + |
| (UnicodeString)", string has " + UCharToUnicodeString(text[i])); |
| |
| i++; |
| if(iter.getIndex() != i) |
| errln("getIndex() aftr nextPostInc() isn't working right"); |
| if(iter.current() != text[i]) |
| errln("current() after nextPostInc() isn't working right"); |
| } while (iter.hasNext()); |
| c=iter.nextPostInc(); |
| if(c!= CharacterIterator::DONE) |
| errln("nextPostInc() didn't return DONE at the beginning"); |
| } |
| |
| { |
| StringCharacterIterator iter(text, 5, 15, 10); |
| if (iter.startIndex() != 5 || iter.endIndex() != 15) |
| errln("creation of a restricted-range iterator failed"); |
| |
| if (iter.getIndex() != 10 || iter.current() != text[(int32_t)10]) |
| errln("starting the iterator in the middle didn't work"); |
| |
| c = iter.first(); |
| i = 5; |
| |
| logln("Testing forward iteration over a range..."); |
| do { |
| if (c == CharacterIterator::DONE && i != 15) |
| errln("Iterator reached end prematurely"); |
| else if (c != text[i]) |
| errln((UnicodeString)"Character mismatch at position " + i + |
| ", iterator has " + UCharToUnicodeString(c) + |
| ", string has " + UCharToUnicodeString(text[i])); |
| |
| if (iter.current() != c) |
| errln("current() isn't working right"); |
| if (iter.getIndex() != i) |
| errln("getIndex() isn't working right"); |
| if(iter.setIndex(i) != c) |
| errln("setIndex() isn't working right"); |
| |
| if (c != CharacterIterator::DONE) { |
| c = iter.next(); |
| i++; |
| } |
| } while (c != CharacterIterator::DONE); |
| |
| c = iter.last(); |
| i = 14; |
| |
| logln("Testing backward iteration over a range..."); |
| do { |
| if (c == CharacterIterator::DONE && i >= 5) |
| errln("Iterator reached end prematurely"); |
| else if (c != text[i]) |
| errln((UnicodeString)"Character mismatch at position " + i + |
| ", iterator has " + UCharToUnicodeString(c) + |
| ", string has " + UCharToUnicodeString(text[i])); |
| |
| if (iter.current() != c) |
| errln("current() isn't working right"); |
| if (iter.getIndex() != i) |
| errln("getIndex() isn't working right"); |
| |
| if (c != CharacterIterator::DONE) { |
| c = iter.previous(); |
| i--; |
| } |
| } while (c != CharacterIterator::DONE); |
| |
| |
| } |
| } |
| |
| //Tests for new API for utf-16 support |
| void CharIterTest::TestIterationUChar32() { |
| UChar textChars[]={ 0x0061, 0x0062, 0xd841, 0xdc02, 0x20ac, 0xd7ff, 0xd842, 0xdc06, 0xd801, 0xdc00, 0x0061, 0x0000}; |
| UnicodeString text(textChars); |
| UChar32 c; |
| int32_t i; |
| { |
| StringCharacterIterator iter(text, 1); |
| |
| UnicodeString iterText; |
| iter.getText(iterText); |
| if (iterText != text) |
| errln("iter.getText() failed"); |
| |
| if (iter.current32() != text[(int32_t)1]) |
| errln("Iterator didn't start out in the right place."); |
| |
| c=iter.setToStart(); |
| i=0; |
| i=iter.move32(1, CharacterIterator::kStart); |
| c=iter.current32(); |
| if(c != text.char32At(1) || i!=1) |
| errln("move32(1, kStart) didn't work correctly expected %X got %X", c, text.char32At(1) ); |
| |
| i=iter.move32(2, CharacterIterator::kCurrent); |
| c=iter.current32(); |
| if(c != text.char32At(4) || i!=4) |
| errln("move32(2, kCurrent) didn't work correctly expected %X got %X i=%ld", c, text.char32At(4), i); |
| |
| i=iter.move32(-2, CharacterIterator::kCurrent); |
| c=iter.current32(); |
| if(c != text.char32At(1) || i!=1) |
| errln("move32(-2, kCurrent) didn't work correctly expected %X got %X i=%d", c, text.char32At(1), i); |
| |
| |
| i=iter.move32(-2, CharacterIterator::kEnd); |
| c=iter.current32(); |
| if(c != text.char32At((text.length()-3)) || i!=(text.length()-3)) |
| errln("move32(-2, kEnd) didn't work correctly expected %X got %X i=%d", c, text.char32At((text.length()-3)), i); |
| |
| |
| c = iter.first32(); |
| i = 0; |
| |
| if (iter.startIndex() != 0 || iter.endIndex() != text.length()) |
| errln("startIndex() or endIndex() failed"); |
| |
| logln("Testing forward iteration..."); |
| do { |
| /* logln("c=%d i=%d char32At=%d", c, i, text.char32At(i)); */ |
| if (c == CharacterIterator::DONE && i != text.length()) |
| errln("Iterator reached end prematurely"); |
| else if(iter.hasNext() == FALSE && i != text.length()) |
| errln("Iterator reached end prematurely. Failed at hasNext"); |
| else if (c != text.char32At(i)) |
| errln("Character mismatch at position %d, iterator has %X, string has %X", i, c, text.char32At(i)); |
| |
| if (iter.current32() != c) |
| errln("current32() isn't working right"); |
| if(iter.setIndex32(i) != c) |
| errln("setIndex32() isn't working right"); |
| if (c != CharacterIterator::DONE) { |
| c = iter.next32(); |
| i=UTF16_NEED_MULTIPLE_UCHAR(c) ? i+2 : i+1; |
| } |
| } while (c != CharacterIterator::DONE); |
| if(iter.hasNext() == TRUE) |
| errln("hasNext() returned true at the end of the string"); |
| |
| |
| |
| c=iter.setToEnd(); |
| if(iter.getIndex() != text.length() || iter.hasNext() != FALSE) |
| errln("setToEnd failed"); |
| |
| c=iter.next32(); |
| if(c!= CharacterIterator::DONE) |
| errln("next32 didn't return DONE at the end"); |
| c=iter.setIndex32(text.length()+1); |
| if(c!= CharacterIterator::DONE) |
| errln("setIndex32(len+1) didn't return DONE"); |
| |
| |
| c = iter.last32(); |
| i = text.length()-1; |
| logln("Testing backward iteration..."); |
| do { |
| if (c == CharacterIterator::DONE && i >= 0) |
| errln((UnicodeString)"Iterator reached start prematurely for i=" + i); |
| else if(iter.hasPrevious() == FALSE && i>0) |
| errln((UnicodeString)"Iterator reached start prematurely for i=" + i); |
| else if (c != text.char32At(i)) |
| errln("Character mismatch at position %d, iterator has %X, string has %X", i, c, text.char32At(i)); |
| |
| if (iter.current32() != c) |
| errln("current32() isn't working right"); |
| if(iter.setIndex32(i) != c) |
| errln("setIndex32() isn't working right"); |
| if (iter.getIndex() != i) |
| errln("getIndex() isn't working right"); |
| if (c != CharacterIterator::DONE) { |
| c = iter.previous32(); |
| i=UTF16_NEED_MULTIPLE_UCHAR(c) ? i-2 : i-1; |
| } |
| } while (c != CharacterIterator::DONE); |
| if(iter.hasPrevious() == TRUE) |
| errln("hasPrevious returned true after reaching the start"); |
| |
| c=iter.previous32(); |
| if(c!= CharacterIterator::DONE) |
| errln("previous32 didn't return DONE at the beginning"); |
| |
| |
| |
| |
| //testing first32PostInc, next32PostInc, setTostart |
| i = 0; |
| c=iter.first32PostInc(); |
| if(c != text.char32At(i)) |
| errln("first32PostInc failed. Expected->%X Got->%X", text.char32At(i), c); |
| if(iter.getIndex() != U16_LENGTH(c) + i) |
| errln((UnicodeString)"getIndex() after first32PostInc() failed"); |
| |
| iter.setToStart(); |
| i=0; |
| if (iter.startIndex() != 0) |
| errln("setToStart failed"); |
| |
| logln("Testing forward iteration..."); |
| do { |
| if (c != CharacterIterator::DONE) |
| c = iter.next32PostInc(); |
| |
| if(c != text.char32At(i)) |
| errln("Character mismatch at position %d, iterator has %X, string has %X", i, c, text.char32At(i)); |
| |
| i=UTF16_NEED_MULTIPLE_UCHAR(c) ? i+2 : i+1; |
| if(iter.getIndex() != i) |
| errln("getIndex() aftr next32PostInc() isn't working right"); |
| if(iter.current32() != text.char32At(i)) |
| errln("current() after next32PostInc() isn't working right"); |
| } while (iter.hasNext()); |
| c=iter.next32PostInc(); |
| if(c!= CharacterIterator::DONE) |
| errln("next32PostInc() didn't return DONE at the beginning"); |
| |
| |
| } |
| |
| { |
| StringCharacterIterator iter(text, 1, 11, 10); |
| if (iter.startIndex() != 1 || iter.endIndex() != 11) |
| errln("creation of a restricted-range iterator failed"); |
| |
| if (iter.getIndex() != 10 || iter.current32() != text.char32At(10)) |
| errln("starting the iterator in the middle didn't work"); |
| |
| c = iter.first32(); |
| |
| i = 1; |
| |
| logln("Testing forward iteration over a range..."); |
| do { |
| if (c == CharacterIterator::DONE && i != 11) |
| errln("Iterator reached end prematurely"); |
| else if(iter.hasNext() == FALSE) |
| errln("Iterator reached end prematurely"); |
| else if (c != text.char32At(i)) |
| errln("Character mismatch at position %d, iterator has %X, string has %X", i, c, text.char32At(i)); |
| |
| if (iter.current32() != c) |
| errln("current32() isn't working right"); |
| if(iter.setIndex32(i) != c) |
| errln("setIndex32() isn't working right"); |
| |
| if (c != CharacterIterator::DONE) { |
| c = iter.next32(); |
| i=UTF16_NEED_MULTIPLE_UCHAR(c) ? i+2 : i+1; |
| } |
| } while (c != CharacterIterator::DONE); |
| c=iter.next32(); |
| if(c != CharacterIterator::DONE) |
| errln("error in next32()"); |
| |
| |
| |
| c=iter.last32(); |
| i = 10; |
| logln("Testing backward iteration over a range..."); |
| do { |
| if (c == CharacterIterator::DONE && i >= 5) |
| errln("Iterator reached start prematurely"); |
| else if(iter.hasPrevious() == FALSE && i > 5) |
| errln("Iterator reached start prematurely"); |
| else if (c != text.char32At(i)) |
| errln("Character mismatch at position %d, iterator has %X, string has %X", i, c, text.char32At(i)); |
| if (iter.current32() != c) |
| errln("current32() isn't working right"); |
| if (iter.getIndex() != i) |
| errln("getIndex() isn't working right"); |
| if(iter.setIndex32(i) != c) |
| errln("setIndex32() isn't working right"); |
| |
| if (c != CharacterIterator::DONE) { |
| c = iter.previous32(); |
| i=UTF16_NEED_MULTIPLE_UCHAR(c) ? i-2 : i-1; |
| } |
| |
| } while (c != CharacterIterator::DONE); |
| c=iter.previous32(); |
| if(c!= CharacterIterator::DONE) |
| errln("error on previous32"); |
| |
| |
| } |
| } |
| |
| void CharIterTest::TestUCharIterator(UCharIterator *iter, CharacterIterator &ci, |
| const char *moves, const char *which) { |
| int32_t m; |
| UChar32 c, c2; |
| UBool h, h2; |
| |
| for(m=0;; ++m) { |
| // move both iter and s[index] |
| switch(moves[m]) { |
| case '0': |
| h=iter->hasNext(iter); |
| h2=ci.hasNext(); |
| c=iter->current(iter); |
| c2=ci.current(); |
| break; |
| case '|': |
| h=iter->hasNext(iter); |
| h2=ci.hasNext(); |
| c=uiter_current32(iter); |
| c2=ci.current32(); |
| break; |
| |
| case '+': |
| h=iter->hasNext(iter); |
| h2=ci.hasNext(); |
| c=iter->next(iter); |
| c2=ci.nextPostInc(); |
| break; |
| case '>': |
| h=iter->hasNext(iter); |
| h2=ci.hasNext(); |
| c=uiter_next32(iter); |
| c2=ci.next32PostInc(); |
| break; |
| |
| case '-': |
| h=iter->hasPrevious(iter); |
| h2=ci.hasPrevious(); |
| c=iter->previous(iter); |
| c2=ci.previous(); |
| break; |
| case '<': |
| h=iter->hasPrevious(iter); |
| h2=ci.hasPrevious(); |
| c=uiter_previous32(iter); |
| c2=ci.previous32(); |
| break; |
| |
| case '2': |
| h=h2=FALSE; |
| c=(UChar32)iter->move(iter, 2, UITER_CURRENT); |
| c2=(UChar32)ci.move(2, CharacterIterator::kCurrent); |
| break; |
| |
| case '8': |
| h=h2=FALSE; |
| c=(UChar32)iter->move(iter, -2, UITER_CURRENT); |
| c2=(UChar32)ci.move(-2, CharacterIterator::kCurrent); |
| break; |
| |
| case 0: |
| return; |
| default: |
| errln("error: unexpected move character '%c' in \"%s\"", moves[m], moves); |
| return; |
| } |
| |
| // compare results |
| if(c2==0xffff) { |
| c2=(UChar32)-1; |
| } |
| if(c!=c2 || h!=h2 || ci.getIndex()!=iter->getIndex(iter, UITER_CURRENT)) { |
| errln("error: UCharIterator(%s) misbehaving at \"%s\"[%d]='%c'", which, moves, m, moves[m]); |
| } |
| } |
| } |
| |
| void CharIterTest::TestUCharIterator() { |
| // test string of length 8 |
| UnicodeString s=UnicodeString("a \\U00010001b\\U0010fffdz", "").unescape(); |
| const char *const moves= |
| "0+++++++++" // 10 moves per line |
| "----0-----" |
| ">>|>>>>>>>" |
| "<<|<<<<<<<" |
| "22+>8>-8+2"; |
| |
| StringCharacterIterator sci(s), compareCI(s); |
| |
| UCharIterator sIter, cIter, rIter; |
| |
| uiter_setString(&sIter, s.getBuffer(), s.length()); |
| uiter_setCharacterIterator(&cIter, &sci); |
| uiter_setReplaceable(&rIter, &s); |
| |
| TestUCharIterator(&sIter, compareCI, moves, "uiter_setString"); |
| compareCI.setIndex(0); |
| TestUCharIterator(&cIter, compareCI, moves, "uiter_setCharacterIterator"); |
| compareCI.setIndex(0); |
| TestUCharIterator(&rIter, compareCI, moves, "uiter_setReplaceable"); |
| |
| // test move & getIndex some more |
| sIter.start=2; |
| sIter.index=3; |
| sIter.limit=5; |
| if( sIter.getIndex(&sIter, UITER_ZERO)!=0 || |
| sIter.getIndex(&sIter, UITER_START)!=2 || |
| sIter.getIndex(&sIter, UITER_CURRENT)!=3 || |
| sIter.getIndex(&sIter, UITER_LIMIT)!=5 || |
| sIter.getIndex(&sIter, UITER_LENGTH)!=s.length() |
| ) { |
| errln("error: UCharIterator(string).getIndex returns wrong index"); |
| } |
| |
| if( sIter.move(&sIter, 4, UITER_ZERO)!=4 || |
| sIter.move(&sIter, 1, UITER_START)!=3 || |
| sIter.move(&sIter, 3, UITER_CURRENT)!=5 || |
| sIter.move(&sIter, -1, UITER_LIMIT)!=4 || |
| sIter.move(&sIter, -5, UITER_LENGTH)!=3 || |
| sIter.move(&sIter, 0, UITER_CURRENT)!=sIter.getIndex(&sIter, UITER_CURRENT) || |
| sIter.getIndex(&sIter, UITER_CURRENT)!=3 |
| ) { |
| errln("error: UCharIterator(string).move sets/returns wrong index"); |
| } |
| |
| sci=StringCharacterIterator(s, 2, 5, 3); |
| uiter_setCharacterIterator(&cIter, &sci); |
| if( cIter.getIndex(&cIter, UITER_ZERO)!=0 || |
| cIter.getIndex(&cIter, UITER_START)!=2 || |
| cIter.getIndex(&cIter, UITER_CURRENT)!=3 || |
| cIter.getIndex(&cIter, UITER_LIMIT)!=5 || |
| cIter.getIndex(&cIter, UITER_LENGTH)!=s.length() |
| ) { |
| errln("error: UCharIterator(character iterator).getIndex returns wrong index"); |
| } |
| |
| if( cIter.move(&cIter, 4, UITER_ZERO)!=4 || |
| cIter.move(&cIter, 1, UITER_START)!=3 || |
| cIter.move(&cIter, 3, UITER_CURRENT)!=5 || |
| cIter.move(&cIter, -1, UITER_LIMIT)!=4 || |
| cIter.move(&cIter, -5, UITER_LENGTH)!=3 || |
| cIter.move(&cIter, 0, UITER_CURRENT)!=cIter.getIndex(&cIter, UITER_CURRENT) || |
| cIter.getIndex(&cIter, UITER_CURRENT)!=3 |
| ) { |
| errln("error: UCharIterator(character iterator).move sets/returns wrong index"); |
| } |
| |
| |
| if(cIter.getIndex(&cIter, (enum UCharIteratorOrigin)-1) != -1) |
| { |
| errln("error: UCharIterator(char iter).getIndex did not return error value"); |
| } |
| |
| if(cIter.move(&cIter, 0, (enum UCharIteratorOrigin)-1) != -1) |
| { |
| errln("error: UCharIterator(char iter).move did not return error value"); |
| } |
| |
| |
| if(rIter.getIndex(&rIter, (enum UCharIteratorOrigin)-1) != -1) |
| { |
| errln("error: UCharIterator(repl iter).getIndex did not return error value"); |
| } |
| |
| if(rIter.move(&rIter, 0, (enum UCharIteratorOrigin)-1) != -1) |
| { |
| errln("error: UCharIterator(repl iter).move did not return error value"); |
| } |
| |
| |
| if(sIter.getIndex(&sIter, (enum UCharIteratorOrigin)-1) != -1) |
| { |
| errln("error: UCharIterator(string iter).getIndex did not return error value"); |
| } |
| |
| if(sIter.move(&sIter, 0, (enum UCharIteratorOrigin)-1) != -1) |
| { |
| errln("error: UCharIterator(string iter).move did not return error value"); |
| } |
| |
| /* Testing function coverage on bad input */ |
| UErrorCode status = U_ZERO_ERROR; |
| uiter_setString(&sIter, NULL, 1); |
| uiter_setState(&sIter, 1, &status); |
| if (status != U_UNSUPPORTED_ERROR) { |
| errln("error: uiter_setState returned %s instead of U_UNSUPPORTED_ERROR", u_errorName(status)); |
| } |
| status = U_ZERO_ERROR; |
| uiter_setState(NULL, 1, &status); |
| if (status != U_ILLEGAL_ARGUMENT_ERROR) { |
| errln("error: uiter_setState returned %s instead of U_ILLEGAL_ARGUMENT_ERROR", u_errorName(status)); |
| } |
| if (uiter_getState(&sIter) != UITER_NO_STATE) { |
| errln("error: uiter_getState did not return UITER_NO_STATE on bad input"); |
| } |
| } |
| |
| // subclass test, and completing API coverage ------------------------------- |
| |
| class SubCharIter : public CharacterIterator { |
| public: |
| // public default constructor, to get coverage of CharacterIterator() |
| SubCharIter() : CharacterIterator() { |
| textLength=end=UPRV_LENGTHOF(s); |
| s[0]=0x61; // 'a' |
| s[1]=0xd900; // U+50400 |
| s[2]=0xdd00; |
| s[3]=0x2029; // PS |
| } |
| |
| // useful stuff, mostly dummy but testing coverage and subclassability |
| virtual UChar nextPostInc() { |
| if(pos<UPRV_LENGTHOF(s)) { |
| return s[pos++]; |
| } else { |
| return DONE; |
| } |
| } |
| |
| virtual UChar32 next32PostInc() { |
| if(pos<UPRV_LENGTHOF(s)) { |
| UChar32 c; |
| U16_NEXT(s, pos, UPRV_LENGTHOF(s), c); |
| return c; |
| } else { |
| return DONE; |
| } |
| } |
| |
| virtual UBool hasNext() { |
| return pos<UPRV_LENGTHOF(s); |
| } |
| |
| virtual UChar first() { |
| pos=0; |
| return s[0]; |
| } |
| |
| virtual UChar32 first32() { |
| UChar32 c; |
| pos=0; |
| U16_NEXT(s, pos, UPRV_LENGTHOF(s), c); |
| pos=0; |
| return c; |
| } |
| |
| virtual UChar setIndex(int32_t position) { |
| if(0<=position && position<=UPRV_LENGTHOF(s)) { |
| pos=position; |
| if(pos<UPRV_LENGTHOF(s)) { |
| return s[pos]; |
| } |
| } |
| return DONE; |
| } |
| |
| virtual UChar32 setIndex32(int32_t position) { |
| if(0<=position && position<=UPRV_LENGTHOF(s)) { |
| pos=position; |
| if(pos<UPRV_LENGTHOF(s)) { |
| UChar32 c; |
| U16_GET(s, 0, pos, UPRV_LENGTHOF(s), c); |
| return c; |
| } |
| } |
| return DONE; |
| } |
| |
| virtual UChar current() const { |
| if(pos<UPRV_LENGTHOF(s)) { |
| return s[pos]; |
| } else { |
| return DONE; |
| } |
| } |
| |
| virtual UChar32 current32() const { |
| if(pos<UPRV_LENGTHOF(s)) { |
| UChar32 c; |
| U16_GET(s, 0, pos, UPRV_LENGTHOF(s), c); |
| return c; |
| } else { |
| return DONE; |
| } |
| } |
| |
| virtual UChar next() { |
| if(pos<UPRV_LENGTHOF(s) && ++pos<UPRV_LENGTHOF(s)) { |
| return s[pos]; |
| } else { |
| return DONE; |
| } |
| } |
| |
| virtual UChar32 next32() { |
| if(pos<UPRV_LENGTHOF(s)) { |
| U16_FWD_1(s, pos, UPRV_LENGTHOF(s)); |
| } |
| if(pos<UPRV_LENGTHOF(s)) { |
| UChar32 c; |
| int32_t i=pos; |
| U16_NEXT(s, i, UPRV_LENGTHOF(s), c); |
| return c; |
| } else { |
| return DONE; |
| } |
| } |
| |
| virtual UBool hasPrevious() { |
| return pos>0; |
| } |
| |
| virtual void getText(UnicodeString &result) { |
| result.setTo(s, UPRV_LENGTHOF(s)); |
| } |
| |
| // dummy implementations of other pure virtual base class functions |
| virtual UBool operator==(const ForwardCharacterIterator &that) const { |
| return |
| this==&that || |
| (typeid(*this)==typeid(that) && pos==((SubCharIter &)that).pos); |
| } |
| |
| virtual int32_t hashCode() const { |
| return 2; |
| } |
| |
| virtual CharacterIterator *clone() const { |
| return NULL; |
| } |
| |
| virtual UChar last() { |
| return 0; |
| } |
| |
| virtual UChar32 last32() { |
| return 0; |
| } |
| |
| virtual UChar previous() { |
| return 0; |
| } |
| |
| virtual UChar32 previous32() { |
| return 0; |
| } |
| |
| virtual int32_t move(int32_t /*delta*/, EOrigin /*origin*/) { |
| return 0; |
| } |
| |
| virtual int32_t move32(int32_t /*delta*/, EOrigin /*origin*/) { |
| return 0; |
| } |
| |
| // RTTI |
| static UClassID getStaticClassID() { |
| return (UClassID)(&fgClassID); |
| } |
| |
| virtual UClassID getDynamicClassID() const { |
| return getStaticClassID(); |
| } |
| |
| private: |
| // dummy string data |
| UChar s[4]; |
| |
| static const char fgClassID; |
| }; |
| |
| const char SubCharIter::fgClassID = 0; |
| |
| class SubStringCharIter : public StringCharacterIterator { |
| public: |
| SubStringCharIter() { |
| setText(UNICODE_STRING("abc", 3)); |
| } |
| }; |
| |
| class SubUCharCharIter : public UCharCharacterIterator { |
| public: |
| SubUCharCharIter() { |
| setText(u, 3); |
| } |
| |
| private: |
| static const UChar u[3]; |
| }; |
| |
| const UChar SubUCharCharIter::u[3]={ 0x61, 0x62, 0x63 }; |
| |
| void CharIterTest::TestCharIteratorSubClasses() { |
| SubCharIter *p; |
| |
| // coverage - call functions that are not otherwise tested |
| // first[32]PostInc() are default implementations that are overridden |
| // in ICU's own CharacterIterator subclasses |
| p=new SubCharIter; |
| if(p->firstPostInc()!=0x61) { |
| errln("SubCharIter.firstPosInc() failed\n"); |
| } |
| delete p; |
| |
| p=new SubCharIter[2]; |
| if(p[1].first32PostInc()!=0x61) { |
| errln("SubCharIter.first32PosInc() failed\n"); |
| } |
| delete [] p; |
| |
| // coverage: StringCharacterIterator default constructor |
| SubStringCharIter sci; |
| if(sci.firstPostInc()!=0x61) { |
| errln("SubStringCharIter.firstPostInc() failed\n"); |
| } |
| |
| // coverage: UCharCharacterIterator default constructor |
| SubUCharCharIter uci; |
| if(uci.firstPostInc()!=0x61) { |
| errln("SubUCharCharIter.firstPostInc() failed\n"); |
| } |
| } |