// © 2016 and later: Unicode, Inc. and others. // License & terms of use: http://www.unicode.org/copyright.html /* ******************************************************************************* * Copyright (C) 1997-2015, International Business Machines Corporation and * * others. All Rights Reserved. * ******************************************************************************* * * File COMPACTDECIMALFORMATTEST.CPP * ******************************************************************************** */ #include #include #include "intltest.h" #if !UCONFIG_NO_FORMATTING #include "unicode/compactdecimalformat.h" #include "unicode/unum.h" #include "cmemory.h" typedef struct ExpectedResult { double value; const char *expected; } ExpectedResult; static const char *kShortStr = "Short"; static const char *kLongStr = "Long"; static ExpectedResult kEnglishShort[] = { {0.0, "0"}, {0.17, "0.17"}, {1.0, "1"}, {1234.0, "1.2K"}, {12345.0, "12K"}, {123456.0, "120K"}, {1234567.0, "1.2M"}, {12345678.0, "12M"}, {123456789.0, "120M"}, {1.23456789E9, "1.2B"}, {1.23456789E10, "12B"}, {1.23456789E11, "120B"}, {1.23456789E12, "1.2T"}, {1.23456789E13, "12T"}, {1.23456789E14, "120T"}, {1.23456789E15, "1200T"}}; static ExpectedResult kSerbianShort[] = { {1234.0, "1,2\\u00a0\\u0445\\u0438\\u0459."}, {12345.0, "12\\u00a0\\u0445\\u0438\\u0459."}, {20789.0, "21\\u00a0\\u0445\\u0438\\u0459."}, {123456.0, "120\\u00a0\\u0445\\u0438\\u0459."}, {1234567.0, "1,2\\u00A0\\u043C\\u0438\\u043B."}, {12345678.0, "12\\u00A0\\u043C\\u0438\\u043B."}, {123456789.0, "120\\u00A0\\u043C\\u0438\\u043B."}, {1.23456789E9, "1,2\\u00A0\\u043C\\u043B\\u0440\\u0434."}, {1.23456789E10, "12\\u00A0\\u043C\\u043B\\u0440\\u0434."}, {1.23456789E11, "120\\u00A0\\u043C\\u043B\\u0440\\u0434."}, {1.23456789E12, "1,2\\u00A0\\u0431\\u0438\\u043B."}, {1.23456789E13, "12\\u00A0\\u0431\\u0438\\u043B."}, {1.23456789E14, "120\\u00A0\\u0431\\u0438\\u043B."}, {1.23456789E15, "1200\\u00A0\\u0431\\u0438\\u043B."}}; static ExpectedResult kSerbianLong[] = { {1234.0, "1,2 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0435"}, // 10^3 few {12345.0, "12 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0430"}, // 10^3 other {21789.0, "22 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0435"}, // 10^3 few {123456.0, "120 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0430"}, // 10^3 other {999999.0, "1 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D"}, // 10^6 one {1234567.0, "1,2 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, // 10^6 few {12345678.0, "12 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, // 10^6 other {123456789.0, "120 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, // 10^6 other {1.23456789E9, "1,2 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0435"}, // 10^9 few {1.23456789E10, "12 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0438"}, // 10^9 other {2.08901234E10, "21 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0430"}, // 10^9 one {2.18901234E10, "22 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0435"}, // 10^9 few {1.23456789E11, "120 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0438"}, // 10^9 other {1.23456789E12, "1,2 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, // 10^12 few {1.23456789E13, "12 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, // 10^12 other {1.23456789E14, "120 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, // 10^12 other {1.23456789E15, "1200 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}}; // 10^12 other static ExpectedResult kSerbianLongNegative[] = { {-1234.0, "-1,2 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0435"}, {-12345.0, "-12 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0430"}, {-21789.0, "-22 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0435"}, {-123456.0, "-120 \\u0445\\u0438\\u0459\\u0430\\u0434\\u0430"}, {-999999.0, "-1 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D"}, {-1234567.0, "-1,2 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, {-12345678.0, "-12 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, {-123456789.0, "-120 \\u043C\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, {-1.23456789E9, "-1,2 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0435"}, {-1.23456789E10, "-12 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0438"}, {-2.08901234E10, "-21 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0430"}, {-2.18901234E10, "-22 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0435"}, {-1.23456789E11, "-120 \\u043C\\u0438\\u043B\\u0438\\u0458\\u0430\\u0440\\u0434\\u0438"}, {-1.23456789E12, "-1,2 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, {-1.23456789E13, "-12 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, {-1.23456789E14, "-120 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}, {-1.23456789E15, "-1200 \\u0431\\u0438\\u043B\\u0438\\u043E\\u043D\\u0430"}}; static ExpectedResult kJapaneseShort[] = { {1234.0, "1200"}, {12345.0, "1.2\\u4E07"}, {123456.0, "12\\u4E07"}, {1234567.0, "120\\u4E07"}, {12345678.0, "1200\\u4E07"}, {123456789.0, "1.2\\u5104"}, {1.23456789E9, "12\\u5104"}, {1.23456789E10, "120\\u5104"}, {1.23456789E11, "1200\\u5104"}, {1.23456789E12, "1.2\\u5146"}, {1.23456789E13, "12\\u5146"}, {1.23456789E14, "120\\u5146"}}; static ExpectedResult kSwahiliShort[] = { {1234.0, "elfu\\u00a01.2"}, {12345.0, "elfu\\u00a012"}, {123456.0, "elfu\\u00a0120"}, {1234567.0, "M1.2"}, {12345678.0, "M12"}, {123456789.0, "M120"}, {1.23456789E9, "B1.2"}, {1.23456789E10, "B12"}, {1.23456789E11, "B120"}, {1.23456789E12, "T1.2"}, {1.23456789E13, "T12"}, {1.23456789E15, "T1200"}}; static ExpectedResult kCsShort[] = { {1000.0, "1\\u00a0tis."}, {1500.0, "1,5\\u00a0tis."}, {5000.0, "5\\u00a0tis."}, {23000.0, "23\\u00a0tis."}, {127123.0, "130\\u00a0tis."}, {1271234.0, "1,3\\u00a0mil."}, {12712345.0, "13\\u00a0mil."}, {127123456.0, "130\\u00a0mil."}, {1.27123456E9, "1,3\\u00a0mld."}, {1.27123456E10, "13\\u00a0mld."}, {1.27123456E11, "130\\u00a0mld."}, {1.27123456E12, "1,3\\u00a0bil."}, {1.27123456E13, "13\\u00a0bil."}, {1.27123456E14, "130\\u00a0bil."}}; static ExpectedResult kSkLong[] = { {1000.0, "1 tis\\u00edc"}, {1572.0, "1,6 tis\\u00edca"}, {5184.0, "5,2 tis\\u00edca"}}; static ExpectedResult kSwahiliShortNegative[] = { {-1234.0, "elfu\\u00a0-1.2"}, {-12345.0, "elfu\\u00a0-12"}, {-123456.0, "elfu\\u00a0-120"}, {-1234567.0, "M-1.2"}, {-12345678.0, "M-12"}, {-123456789.0, "M-120"}, {-1.23456789E9, "B-1.2"}, {-1.23456789E10, "B-12"}, {-1.23456789E11, "B-120"}, {-1.23456789E12, "T-1.2"}, {-1.23456789E13, "T-12"}, {-1.23456789E15, "T-1200"}}; static ExpectedResult kArabicLong[] = { {-5300.0, "\\u061C-\\u0665\\u066B\\u0663 \\u0623\\u0644\\u0641"}}; class CompactDecimalFormatTest : public IntlTest { public: CompactDecimalFormatTest() { } void runIndexedTest(int32_t index, UBool exec, const char *&name, char *par=0); private: void TestEnglishShort(); void TestSerbianShort(); void TestSerbianLong(); void TestSerbianLongNegative(); void TestJapaneseShort(); void TestSwahiliShort(); void TestCsShort(); void TestSkLong(); void TestSwahiliShortNegative(); void TestArabicLong(); void TestFieldPosition(); void TestSignificantDigits(); void TestAPIVariants(); void CheckLocale( const Locale& locale, UNumberCompactStyle style, const ExpectedResult* expectedResult, int32_t expectedResultLength); void CheckExpectedResult( const CompactDecimalFormat* cdf, const ExpectedResult* expectedResult, const char* description); CompactDecimalFormat* createCDFInstance(const Locale& locale, UNumberCompactStyle style, UErrorCode& status); static const char *StyleStr(UNumberCompactStyle style); }; void CompactDecimalFormatTest::runIndexedTest( int32_t index, UBool exec, const char *&name, char *) { if (exec) { logln("TestSuite CompactDecimalFormatTest: "); } TESTCASE_AUTO_BEGIN; TESTCASE_AUTO(TestEnglishShort); TESTCASE_AUTO(TestSerbianShort); TESTCASE_AUTO(TestSerbianLong); TESTCASE_AUTO(TestSerbianLongNegative); TESTCASE_AUTO(TestJapaneseShort); TESTCASE_AUTO(TestSwahiliShort); TESTCASE_AUTO(TestCsShort); TESTCASE_AUTO(TestSkLong); TESTCASE_AUTO(TestSwahiliShortNegative); TESTCASE_AUTO(TestArabicLong); TESTCASE_AUTO(TestFieldPosition); TESTCASE_AUTO(TestSignificantDigits); TESTCASE_AUTO(TestAPIVariants); TESTCASE_AUTO_END; } void CompactDecimalFormatTest::TestEnglishShort() { CheckLocale("en", UNUM_SHORT, kEnglishShort, UPRV_LENGTHOF(kEnglishShort)); } void CompactDecimalFormatTest::TestSerbianShort() { CheckLocale("sr", UNUM_SHORT, kSerbianShort, UPRV_LENGTHOF(kSerbianShort)); } void CompactDecimalFormatTest::TestSerbianLong() { CheckLocale("sr", UNUM_LONG, kSerbianLong, UPRV_LENGTHOF(kSerbianLong)); } void CompactDecimalFormatTest::TestSerbianLongNegative() { CheckLocale("sr", UNUM_LONG, kSerbianLongNegative, UPRV_LENGTHOF(kSerbianLongNegative)); } void CompactDecimalFormatTest::TestJapaneseShort() { CheckLocale(Locale::getJapan(), UNUM_SHORT, kJapaneseShort, UPRV_LENGTHOF(kJapaneseShort)); } void CompactDecimalFormatTest::TestSwahiliShort() { CheckLocale("sw", UNUM_SHORT, kSwahiliShort, UPRV_LENGTHOF(kSwahiliShort)); } void CompactDecimalFormatTest::TestFieldPosition() { // Swahili uses prefixes which forces offsets in field position to change UErrorCode status = U_ZERO_ERROR; LocalPointer cdf(createCDFInstance("sw", UNUM_SHORT, status)); if (U_FAILURE(status)) { dataerrln("Unable to create format object - %s", u_errorName(status)); return; } FieldPosition fp(UNUM_INTEGER_FIELD); UnicodeString result; cdf->format(1234567.0, result, fp); UnicodeString subString = result.tempSubString(fp.getBeginIndex(), fp.getEndIndex() - fp.getBeginIndex()); if (subString != UnicodeString("1", -1, US_INV)) { errln(UnicodeString("Expected 1, got ") + subString); } } void CompactDecimalFormatTest::TestCsShort() { CheckLocale("cs", UNUM_SHORT, kCsShort, UPRV_LENGTHOF(kCsShort)); } void CompactDecimalFormatTest::TestSkLong() { // In CLDR we have: // 1000 { // few{"0"} // one{"0"} // other{"0"} CheckLocale("sk", UNUM_LONG, kSkLong, UPRV_LENGTHOF(kSkLong)); } void CompactDecimalFormatTest::TestSwahiliShortNegative() { CheckLocale("sw", UNUM_SHORT, kSwahiliShortNegative, UPRV_LENGTHOF(kSwahiliShortNegative)); } void CompactDecimalFormatTest::TestArabicLong() { CheckLocale("ar", UNUM_LONG, kArabicLong, UPRV_LENGTHOF(kArabicLong)); } void CompactDecimalFormatTest::TestSignificantDigits() { UErrorCode status = U_ZERO_ERROR; LocalPointer cdf(CompactDecimalFormat::createInstance("en", UNUM_SHORT, status)); if (U_FAILURE(status)) { dataerrln("Unable to create format object - %s", u_errorName(status)); return; } UnicodeString actual; cdf->format(123456.0, actual); // We expect 3 significant digits by default UnicodeString expected("123K", -1, US_INV); if (actual != expected) { errln(UnicodeString("Fail: Expected: ") + expected + UnicodeString(" Got: ") + actual); } } void CompactDecimalFormatTest::TestAPIVariants() { UErrorCode status = U_ZERO_ERROR; LocalPointer cdf(CompactDecimalFormat::createInstance("en", UNUM_SHORT, status)); if (U_FAILURE(status)) { dataerrln("Unable to create format object - %s", u_errorName(status)); return; } UnicodeString actual; FieldPosition pos; FieldPositionIterator posIter; UnicodeString expected("123K", -1, US_INV); pos.setField(UNUM_INTEGER_FIELD); actual.remove(); pos.setBeginIndex(0); pos.setEndIndex(0); cdf->format((double)123456.0, actual, pos); if (actual != expected || pos.getEndIndex() != 3) { errln(UnicodeString("Fail format(double,UnicodeString&,FieldPosition&): Expected: \"") + expected + "\", pos 3; " + "Got: \"" + actual + "\", pos " + pos.getEndIndex()); } actual.remove(); pos.setBeginIndex(0); pos.setEndIndex(0); status = U_ZERO_ERROR; cdf->format((double)123456.0, actual, pos, status); if (actual != expected || pos.getEndIndex() != 3 || status != U_ZERO_ERROR) { errln(UnicodeString("Fail format(double,UnicodeString&,FieldPosition&,UErrorCode&): Expected: \"") + expected + "\", pos 3, status U_ZERO_ERROR; " + "Got: \"" + actual + "\", pos " + pos.getEndIndex() + ", status " + u_errorName(status)); } actual.remove(); status = U_ZERO_ERROR; cdf->format((double)123456.0, actual, &posIter, status); if (status != U_UNSUPPORTED_ERROR) { errln(UnicodeString("Fail format(double,UnicodeString&,FieldPositionIterator*,UErrorCode&): Expected status U_UNSUPPORTED_ERROR;") + "Got status " + u_errorName(status)); } actual.remove(); pos.setBeginIndex(0); pos.setEndIndex(0); cdf->format((int32_t)123456, actual, pos); if (actual != expected || pos.getEndIndex() != 3) { errln(UnicodeString("Fail format(int32_t,UnicodeString&,FieldPosition&): Expected: \"") + expected + "\", pos 3; " + "Got: \"" + actual + "\", pos " + pos.getEndIndex()); } actual.remove(); pos.setBeginIndex(0); pos.setEndIndex(0); status = U_ZERO_ERROR; cdf->format((int32_t)123456, actual, pos, status); if (actual != expected || pos.getEndIndex() != 3 || status != U_ZERO_ERROR) { errln(UnicodeString("Fail format(int32_t,UnicodeString&,FieldPosition&,UErrorCode&): Expected: \"") + expected + "\", pos 3, status U_ZERO_ERROR; " + "Got: \"" + actual + "\", pos " + pos.getEndIndex() + ", status " + u_errorName(status)); } actual.remove(); status = U_ZERO_ERROR; cdf->format((int32_t)123456, actual, &posIter, status); if (status != U_UNSUPPORTED_ERROR) { errln(UnicodeString("Fail format(int32_t,UnicodeString&,FieldPositionIterator*,UErrorCode&): Expected status U_UNSUPPORTED_ERROR;") + "Got status " + u_errorName(status)); } actual.remove(); pos.setBeginIndex(0); pos.setEndIndex(0); cdf->format((int64_t)123456, actual, pos); if (actual != expected || pos.getEndIndex() != 3) { errln(UnicodeString("Fail format(int64_t,UnicodeString&,FieldPosition&): Expected: \"") + expected + "\", pos 3; " + "Got: \"" + actual + "\", pos " + pos.getEndIndex()); } actual.remove(); pos.setBeginIndex(0); pos.setEndIndex(0); status = U_ZERO_ERROR; cdf->format((int64_t)123456, actual, pos, status); if (actual != expected || pos.getEndIndex() != 3 || status != U_ZERO_ERROR) { errln(UnicodeString("Fail format(int64_t,UnicodeString&,FieldPosition&,UErrorCode&): Expected: \"") + expected + "\", pos 3, status U_ZERO_ERROR; " + "Got: \"" + actual + "\", pos " + pos.getEndIndex() + ", status " + u_errorName(status)); } actual.remove(); status = U_ZERO_ERROR; cdf->format((int64_t)123456, actual, &posIter, status); if (status != U_UNSUPPORTED_ERROR) { errln(UnicodeString("Fail format(int64_t,UnicodeString&,FieldPositionIterator*,UErrorCode&): Expected status U_UNSUPPORTED_ERROR;") + "Got status " + u_errorName(status)); } } void CompactDecimalFormatTest::CheckLocale(const Locale& locale, UNumberCompactStyle style, const ExpectedResult* expectedResults, int32_t expectedResultLength) { UErrorCode status = U_ZERO_ERROR; LocalPointer cdf(createCDFInstance(locale, style, status)); if (U_FAILURE(status)) { dataerrln("Unable to create format object - %s", u_errorName(status)); return; } char description[256]; sprintf(description,"%s - %s", locale.getName(), StyleStr(style)); for (int32_t i = 0; i < expectedResultLength; i++) { CheckExpectedResult(cdf.getAlias(), &expectedResults[i], description); } } void CompactDecimalFormatTest::CheckExpectedResult( const CompactDecimalFormat* cdf, const ExpectedResult* expectedResult, const char* description) { UnicodeString actual; cdf->format(expectedResult->value, actual); UnicodeString expected(expectedResult->expected, -1, US_INV); expected = expected.unescape(); if (actual != expected) { errln(UnicodeString("Fail: Expected: ") + expected + UnicodeString(" Got: ") + actual + UnicodeString(" for: ") + UnicodeString(description)); } } CompactDecimalFormat* CompactDecimalFormatTest::createCDFInstance(const Locale& locale, UNumberCompactStyle style, UErrorCode& status) { CompactDecimalFormat* result = CompactDecimalFormat::createInstance(locale, style, status); if (U_FAILURE(status)) { return NULL; } // All tests are written for two significant digits, so we explicitly set here // in case default significant digits change. result->setMaximumSignificantDigits(2); return result; } const char *CompactDecimalFormatTest::StyleStr(UNumberCompactStyle style) { if (style == UNUM_SHORT) { return kShortStr; } return kLongStr; } extern IntlTest *createCompactDecimalFormatTest() { return new CompactDecimalFormatTest(); } #endif