--- /dev/null 2018-12-03 10:57:09.048000000 +0530 +++ new/test/jdk/java/text/Format/CompactNumberFormat/TestCompactNumber.java 2018-12-03 17:32:48.163293568 +0530 @@ -0,0 +1,589 @@ +/* + * Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +/* + * @test + * @bug 8177552 + * @summary Checks the functioning of compact number format + * @modules jdk.localedata + * @run testng/othervm TestCompactNumber + */ +import java.math.BigDecimal; +import java.math.BigInteger; +import java.text.FieldPosition; +import java.text.Format; +import java.text.NumberFormat; +import java.text.ParseException; +import java.text.ParsePosition; +import java.util.Locale; +import java.util.stream.Stream; +import static org.testng.Assert.*; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +public class TestCompactNumber { + + private static final NumberFormat FORMAT_DZ_LONG = NumberFormat + .getCompactNumberInstance(new Locale("dz"), NumberFormat.Style.LONG); + + private static final NumberFormat FORMAT_EN_US_SHORT = NumberFormat + .getCompactNumberInstance(Locale.US, NumberFormat.Style.SHORT); + + private static final NumberFormat FORMAT_EN_LONG = NumberFormat + .getCompactNumberInstance(new Locale("en"), NumberFormat.Style.LONG); + + private static final NumberFormat FORMAT_HI_IN_LONG = NumberFormat + .getCompactNumberInstance(new Locale("hi", "IN"), NumberFormat.Style.LONG); + + private static final NumberFormat FORMAT_JA_JP_SHORT = NumberFormat + .getCompactNumberInstance(Locale.JAPAN, NumberFormat.Style.SHORT); + + private static final NumberFormat FORMAT_IT_SHORT = NumberFormat + .getCompactNumberInstance(new Locale("it"), NumberFormat.Style.SHORT); + + private static final NumberFormat FORMAT_CA_LONG = NumberFormat + .getCompactNumberInstance(new Locale("ca"), NumberFormat.Style.LONG); + + private static final NumberFormat FORMAT_AS_LONG = NumberFormat + .getCompactNumberInstance(new Locale("as"), NumberFormat.Style.LONG); + + private static final NumberFormat FORMAT_BRX_SHORT = NumberFormat + .getCompactNumberInstance(new Locale("brx"), NumberFormat.Style.SHORT); + + private static final NumberFormat FORMAT_SW_LONG = NumberFormat + .getCompactNumberInstance(new Locale("sw"), NumberFormat.Style.LONG); + + private static final NumberFormat FORMAT_SE_SHORT = NumberFormat + .getCompactNumberInstance(new Locale("se"), NumberFormat.Style.SHORT); + + @DataProvider(name = "format") + Object[][] compactFormatData() { + return new Object[][]{ + // compact number format instance, number to format, formatted output + {FORMAT_DZ_LONG, 1000.09, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55" + + "\u0FB2\u0F42 \u0F21"}, + {FORMAT_DZ_LONG, -999.99, "-\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55" + + "\u0FB2\u0F42 \u0F21"}, + {FORMAT_DZ_LONG, -0.0, "-\u0F20"}, + {FORMAT_DZ_LONG, 3000L, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55" + + "\u0FB2\u0F42 \u0F23"}, + {FORMAT_DZ_LONG, new BigInteger("12345678901234567890"), "\u0F51" + + "\u0F74\u0F44\u0F0B\u0F55\u0FB1\u0F74\u0F62\u0F0B\u0F66" + + "\u0F0B\u0F61\u0F0B \u0F21\u0F22\u0F23\u0F24\u0F25\u0F27"}, + // negative + {FORMAT_DZ_LONG, new BigInteger("-12345678901234567890"), "-\u0F51" + + "\u0F74\u0F44\u0F0B\u0F55\u0FB1\u0F74\u0F62\u0F0B\u0F66" + + "\u0F0B\u0F61\u0F0B \u0F21\u0F22\u0F23\u0F24\u0F25\u0F27"}, + {FORMAT_DZ_LONG, new BigDecimal("12345678901234567890.89"), "\u0F51" + + "\u0F74\u0F44\u0F0B\u0F55\u0FB1\u0F74\u0F62\u0F0B\u0F66" + + "\u0F0B\u0F61\u0F0B \u0F21\u0F22\u0F23\u0F24\u0F25\u0F27"}, + {FORMAT_DZ_LONG, new BigDecimal("-12345678901234567890.89"), "-\u0F51" + + "\u0F74\u0F44\u0F0B\u0F55\u0FB1\u0F74\u0F62\u0F0B\u0F66" + + "\u0F0B\u0F61\u0F0B \u0F21\u0F22\u0F23\u0F24\u0F25\u0F27"}, + // Zeros + {FORMAT_EN_US_SHORT, 0, "0"}, + {FORMAT_EN_US_SHORT, 0.0, "0"}, + {FORMAT_EN_US_SHORT, -0.0, "-0"}, + // Less than 1000 no suffix + {FORMAT_EN_US_SHORT, 499, "499"}, + // Boundary number + {FORMAT_EN_US_SHORT, 1000.0, "1K"}, + // Long + {FORMAT_EN_US_SHORT, 3000L, "3K"}, + {FORMAT_EN_US_SHORT, 30000L, "30K"}, + {FORMAT_EN_US_SHORT, 300000L, "300K"}, + {FORMAT_EN_US_SHORT, 3000000L, "3M"}, + {FORMAT_EN_US_SHORT, 30000000L, "30M"}, + {FORMAT_EN_US_SHORT, 300000000L, "300M"}, + {FORMAT_EN_US_SHORT, 3000000000L, "3B"}, + {FORMAT_EN_US_SHORT, 30000000000L, "30B"}, + {FORMAT_EN_US_SHORT, 300000000000L, "300B"}, + {FORMAT_EN_US_SHORT, 3000000000000L, "3T"}, + {FORMAT_EN_US_SHORT, 30000000000000L, "30T"}, + {FORMAT_EN_US_SHORT, 300000000000000L, "300T"}, + {FORMAT_EN_US_SHORT, 3000000000000000L, "3000T"}, + // Negatives + {FORMAT_EN_US_SHORT, -3000L, "-3K"}, + {FORMAT_EN_US_SHORT, -30000L, "-30K"}, + {FORMAT_EN_US_SHORT, -300000L, "-300K"}, + {FORMAT_EN_US_SHORT, -3000000L, "-3M"}, + {FORMAT_EN_US_SHORT, -30000000L, "-30M"}, + {FORMAT_EN_US_SHORT, -300000000L, "-300M"}, + {FORMAT_EN_US_SHORT, -3000000000L, "-3B"}, + {FORMAT_EN_US_SHORT, -30000000000L, "-30B"}, + {FORMAT_EN_US_SHORT, -300000000000L, "-300B"}, + {FORMAT_EN_US_SHORT, -3000000000000L, "-3T"}, + {FORMAT_EN_US_SHORT, -30000000000000L, "-30T"}, + {FORMAT_EN_US_SHORT, -300000000000000L, "-300T"}, + {FORMAT_EN_US_SHORT, -3000000000000000L, "-3000T"}, + // Double + {FORMAT_EN_US_SHORT, 3000.0, "3K"}, + {FORMAT_EN_US_SHORT, 30000.0, "30K"}, + {FORMAT_EN_US_SHORT, 300000.0, "300K"}, + {FORMAT_EN_US_SHORT, 3000000.0, "3M"}, + {FORMAT_EN_US_SHORT, 30000000.0, "30M"}, + {FORMAT_EN_US_SHORT, 300000000.0, "300M"}, + {FORMAT_EN_US_SHORT, 3000000000.0, "3B"}, + {FORMAT_EN_US_SHORT, 30000000000.0, "30B"}, + {FORMAT_EN_US_SHORT, 300000000000.0, "300B"}, + {FORMAT_EN_US_SHORT, 3000000000000.0, "3T"}, + {FORMAT_EN_US_SHORT, 30000000000000.0, "30T"}, + {FORMAT_EN_US_SHORT, 300000000000000.0, "300T"}, + {FORMAT_EN_US_SHORT, 3000000000000000.0, "3000T"}, + // Negatives + {FORMAT_EN_US_SHORT, -3000.0, "-3K"}, + {FORMAT_EN_US_SHORT, -30000.0, "-30K"}, + {FORMAT_EN_US_SHORT, -300000.0, "-300K"}, + {FORMAT_EN_US_SHORT, -3000000.0, "-3M"}, + {FORMAT_EN_US_SHORT, -30000000.0, "-30M"}, + {FORMAT_EN_US_SHORT, -300000000.0, "-300M"}, + {FORMAT_EN_US_SHORT, -3000000000.0, "-3B"}, + {FORMAT_EN_US_SHORT, -30000000000.0, "-30B"}, + {FORMAT_EN_US_SHORT, -300000000000.0, "-300B"}, + {FORMAT_EN_US_SHORT, -3000000000000.0, "-3T"}, + {FORMAT_EN_US_SHORT, -30000000000000.0, "-30T"}, + {FORMAT_EN_US_SHORT, -300000000000000.0, "-300T"}, + {FORMAT_EN_US_SHORT, -3000000000000000.0, "-3000T"}, + // BigInteger + {FORMAT_EN_US_SHORT, new BigInteger("12345678901234567890"), + "12345679T"}, + {FORMAT_EN_US_SHORT, new BigInteger("-12345678901234567890"), + "-12345679T"}, + //BigDecimal + {FORMAT_EN_US_SHORT, new BigDecimal("12345678901234567890.89"), + "12345679T"}, + {FORMAT_EN_US_SHORT, new BigDecimal("-12345678901234567890.89"), + "-12345679T"}, + {FORMAT_EN_US_SHORT, new BigDecimal("12345678901234567890123466767.89"), + "12345678901234568T"}, + {FORMAT_EN_US_SHORT, new BigDecimal( + "12345678901234567890878732267863209.89"), + "12345678901234567890879T"}, + // number as exponent + {FORMAT_EN_US_SHORT, 9.78313E+3, "10K"}, + // Less than 1000 no suffix + {FORMAT_EN_LONG, 999, "999"}, + // Round the value and then format + {FORMAT_EN_LONG, 999.99, "1 thousand"}, + // 10 thousand + {FORMAT_EN_LONG, 99000, "99 thousand"}, + // Long path + {FORMAT_EN_LONG, 330000, "330 thousand"}, + // Double path + {FORMAT_EN_LONG, 3000.90, "3 thousand"}, + // BigInteger path + {FORMAT_EN_LONG, new BigInteger("12345678901234567890"), + "12345679 trillion"}, + //BigDecimal path + {FORMAT_EN_LONG, new BigDecimal("12345678901234567890.89"), + "12345679 trillion"}, + // Less than 1000 no suffix + {FORMAT_HI_IN_LONG, -999, "-999"}, + // Round the value with 0 fraction digits and format it + {FORMAT_HI_IN_LONG, -999.99, "-1 \u0939\u091C\u093C\u093E\u0930"}, + // 10 thousand + {FORMAT_HI_IN_LONG, 99000, "99 \u0939\u091C\u093C\u093E\u0930"}, + // Long path + {FORMAT_HI_IN_LONG, 330000, "3 \u0932\u093E\u0916"}, + // Double path + {FORMAT_HI_IN_LONG, 3000.90, "3 \u0939\u091C\u093C\u093E\u0930"}, + // BigInteger path + {FORMAT_HI_IN_LONG, new BigInteger("12345678901234567890"), + "123456789 \u0916\u0930\u092C"}, + // BigDecimal path + {FORMAT_HI_IN_LONG, new BigDecimal("12345678901234567890.89"), + "123456789 \u0916\u0930\u092C"}, + // 1000 does not have any suffix in "ja" locale + {FORMAT_JA_JP_SHORT, -999.99, "-1,000"}, + // 0-9999 does not have any suffix + {FORMAT_JA_JP_SHORT, 9999, "9,999"}, + // 99000/10000 => 9.9\u4E07 rounded to 10\u4E07 + {FORMAT_JA_JP_SHORT, 99000, "10\u4E07"}, + // Negative + {FORMAT_JA_JP_SHORT, -99000, "-10\u4E07"}, + // Long path + {FORMAT_JA_JP_SHORT, 330000, "33\u4E07"}, + // Double path + {FORMAT_JA_JP_SHORT, 3000.90, "3,001"}, + // BigInteger path + {FORMAT_JA_JP_SHORT, new BigInteger("12345678901234567890"), + "12345679\u5146"}, + // BigDecimal path + {FORMAT_JA_JP_SHORT, new BigDecimal("12345678901234567890.89"), + "12345679\u5146"}, + // less than 1000 no suffix + {FORMAT_IT_SHORT, 499, "499"}, + // Boundary number + {FORMAT_IT_SHORT, 1000, "1.000"}, + // Long path + {FORMAT_IT_SHORT, 3000000L, "3\u00a0Mln"}, + // Double path + {FORMAT_IT_SHORT, 3000000.0, "3\u00a0Mln"}, + // BigInteger path + {FORMAT_IT_SHORT, new BigInteger("12345678901234567890"), + "12345679\u00a0Bln"}, + // BigDecimal path + {FORMAT_IT_SHORT, new BigDecimal("12345678901234567890.89"), + "12345679\u00a0Bln"}, + {FORMAT_CA_LONG, 999, "999"}, + {FORMAT_CA_LONG, 999.99, "1 miler"}, + {FORMAT_CA_LONG, 99000, "99 milers"}, + {FORMAT_CA_LONG, 330000, "330 milers"}, + {FORMAT_CA_LONG, 3000.90, "3 miler"}, + {FORMAT_CA_LONG, 1000000, "1 mili\u00f3"}, + {FORMAT_CA_LONG, new BigInteger("12345678901234567890"), + "12345679 bilions"}, + {FORMAT_CA_LONG, new BigDecimal("12345678901234567890.89"), + "12345679 bilions"}, + {FORMAT_AS_LONG, 5000.0, "\u09eb \u09b9\u09be\u099c\u09be\u09f0"}, + {FORMAT_AS_LONG, 50000.0, "\u09eb\u09e6 \u09b9\u09be\u099c\u09be\u09f0"}, + {FORMAT_AS_LONG, 500000.0, "\u09eb \u09b2\u09be\u0996"}, + {FORMAT_AS_LONG, 5000000.0, "\u09eb \u09a8\u09bf\u09af\u09c1\u09a4"}, + {FORMAT_AS_LONG, 50000000.0, "\u09eb\u09e6 \u09a8\u09bf\u09af\u09c1\u09a4"}, + {FORMAT_AS_LONG, 500000000.0, "\u09eb\u09e6\u09e6 \u09a8\u09bf\u09af\u09c1\u09a4"}, + {FORMAT_AS_LONG, 5000000000.0, "\u09eb \u09b6\u09a4 \u0995\u09cb\u099f\u09bf"}, + {FORMAT_AS_LONG, 50000000000.0, "\u09eb\u09e6 \u09b6\u09a4 \u0995\u09cb\u099f\u09bf"}, + {FORMAT_AS_LONG, 500000000000.0, "\u09eb\u09e6\u09e6 \u09b6\u09a4 \u0995\u09cb\u099f\u09bf"}, + {FORMAT_AS_LONG, 5000000000000.0, "\u09eb \u09b6\u09a4 \u09aa\u09f0\u09be\u09f0\u09cd\u09a6\u09cd\u09a7"}, + {FORMAT_AS_LONG, 50000000000000.0, "\u09eb\u09e6 \u09b6\u09a4 \u09aa\u09f0\u09be\u09f0\u09cd\u09a6\u09cd\u09a7"}, + {FORMAT_AS_LONG, 500000000000000.0, "\u09eb\u09e6\u09e6 \u09b6\u09a4 \u09aa\u09f0\u09be\u09f0\u09cd\u09a6\u09cd\u09a7"}, + {FORMAT_AS_LONG, 5000000000000000.0, "\u09eb\u09e6\u09e6\u09e6 \u09b6\u09a4 \u09aa\u09f0\u09be\u09f0\u09cd\u09a6\u09cd\u09a7"}, + {FORMAT_AS_LONG, new BigInteger("12345678901234567890"), + "\u09e7\u09e8\u09e9\u09ea\u09eb\u09ec\u09ed\u09ef \u09b6\u09a4 \u09aa\u09f0\u09be\u09f0\u09cd\u09a6\u09cd\u09a7"}, + {FORMAT_AS_LONG, new BigDecimal("12345678901234567890123466767.89"), + "\u09e7\u09e8\u09e9\u09ea\u09eb\u09ec\u09ed\u09ee\u09ef\u09e6\u09e7\u09e8\u09e9\u09ea\u09eb\u09ec\u09ee \u09b6\u09a4 \u09aa\u09f0\u09be\u09f0\u09cd\u09a6\u09cd\u09a7"}, + {FORMAT_BRX_SHORT, 999, "999"}, + {FORMAT_BRX_SHORT, 999.99, "1K"}, + {FORMAT_BRX_SHORT, 99000, "99K"}, + {FORMAT_BRX_SHORT, 330000, "330K"}, + {FORMAT_BRX_SHORT, 3000.90, "3K"}, + {FORMAT_BRX_SHORT, 1000000, "1M"}, + {FORMAT_BRX_SHORT, new BigInteger("12345678901234567890"), + "12345679T"}, + {FORMAT_BRX_SHORT, new BigDecimal("12345678901234567890.89"), + "12345679T"}, + // Less than 1000 no suffix + {FORMAT_SW_LONG, 499, "499"}, + // Boundary number + {FORMAT_SW_LONG, 1000, "elfu 1"}, + // Long path + {FORMAT_SW_LONG, 3000000L, "milioni 3"}, + // Long path, negative + {FORMAT_SW_LONG, -3000000L, "milioni -3"}, + // Double path + {FORMAT_SW_LONG, 3000000.0, "milioni 3"}, + // Double path, negative + {FORMAT_SW_LONG, -3000000.0, "milioni -3"}, + // BigInteger path + {FORMAT_SW_LONG, new BigInteger("12345678901234567890"), + "trilioni 12345679"}, + // BigDecimal path + {FORMAT_SW_LONG, new BigDecimal("12345678901234567890.89"), + "trilioni 12345679"}, + // Positives + // No compact form + {FORMAT_SE_SHORT, 999, "999"}, + // Long + {FORMAT_SE_SHORT, 8000000L, "8\u00a0mn"}, + // Double + {FORMAT_SE_SHORT, 8000.98, "8\u00a0dt"}, + // Big integer + {FORMAT_SE_SHORT, new BigInteger("12345678901234567890"), "12345679\u00a0bn"}, + // Big decimal + {FORMAT_SE_SHORT, new BigDecimal("12345678901234567890.98"), "12345679\u00a0bn"}, + // Negatives + // No compact form + {FORMAT_SE_SHORT, -999, "\u2212999"}, + // Long + {FORMAT_SE_SHORT, -8000000L, "\u22128\u00a0mn"}, + // Double + {FORMAT_SE_SHORT, -8000.98, "\u22128\u00a0dt"}, + // BigInteger + {FORMAT_SE_SHORT, new BigInteger("-12345678901234567890"), "\u221212345679\u00a0bn"}, + // BigDecimal + {FORMAT_SE_SHORT, new BigDecimal("-12345678901234567890.98"), "\u221212345679\u00a0bn"},}; + } + + @DataProvider(name = "parse") + Object[][] compactParseData() { + return new Object[][]{ + // compact number format instance, string to parse, parsed number, return type + {FORMAT_DZ_LONG, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2" + + "\u0F42 \u0F21", 1000L, Long.class}, + {FORMAT_DZ_LONG, "-\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2" + + "\u0F42 \u0F23", -3000L, Long.class}, + {FORMAT_DZ_LONG, "\u0F51\u0F74\u0F44\u0F0B\u0F55\u0FB1\u0F74\u0F62" + + "\u0F0B\u0F66\u0F0B\u0F61\u0F0B \u0F21" + + "\u0F22\u0F23\u0F24\u0F25\u0F27", 1.23457E19, Double.class}, + {FORMAT_DZ_LONG, "-\u0F51\u0F74\u0F44\u0F0B\u0F55\u0FB1\u0F74\u0F62" + + "\u0F0B\u0F66\u0F0B\u0F61\u0F0B \u0F21" + + "\u0F22\u0F23\u0F24\u0F25\u0F27", -1.23457E19, Double.class}, + {FORMAT_EN_US_SHORT, "-0.0", -0.0, Double.class}, + {FORMAT_EN_US_SHORT, "-0", -0.0, Double.class}, + {FORMAT_EN_US_SHORT, "0", 0L, Long.class}, + {FORMAT_EN_US_SHORT, "499", 499L, Long.class}, + {FORMAT_EN_US_SHORT, "-499", -499L, Long.class}, + {FORMAT_EN_US_SHORT, "499.89", 499.89, Double.class}, + {FORMAT_EN_US_SHORT, "-499.89", -499.89, Double.class}, + {FORMAT_EN_US_SHORT, "1K", 1000L, Long.class}, + {FORMAT_EN_US_SHORT, "-1K", -1000L, Long.class}, + {FORMAT_EN_US_SHORT, "3K", 3000L, Long.class}, + {FORMAT_EN_US_SHORT, "17K", 17000L, Long.class}, + {FORMAT_EN_US_SHORT, "-17K", -17000L, Long.class}, + {FORMAT_EN_US_SHORT, "-3K", -3000L, Long.class}, + {FORMAT_EN_US_SHORT, "12345678901234567890", 1.2345678901234567E19, Double.class}, + {FORMAT_EN_US_SHORT, "12345679T", 1.2345679E19, Double.class}, + {FORMAT_EN_US_SHORT, "-12345679T", -1.2345679E19, Double.class}, + {FORMAT_EN_US_SHORT, "599.01K", 599010L, Long.class}, + {FORMAT_EN_US_SHORT, "-599.01K", -599010L, Long.class}, + {FORMAT_EN_US_SHORT, "599444444.90T", 5.994444449E20, Double.class}, + {FORMAT_EN_US_SHORT, "-599444444.90T", -5.994444449E20, Double.class}, + {FORMAT_EN_US_SHORT, "123456789012345.5678K", 123456789012345568L, Long.class}, + {FORMAT_EN_US_SHORT, "17.000K", 17000L, Long.class}, + {FORMAT_EN_US_SHORT, "123.56678K", 123566.78000, Double.class}, + {FORMAT_EN_US_SHORT, "-123.56678K", -123566.78000, Double.class}, + {FORMAT_EN_LONG, "999", 999L, Long.class}, + {FORMAT_EN_LONG, "1 thousand", 1000L, Long.class}, + {FORMAT_EN_LONG, "3 thousand", 3000L, Long.class}, + {FORMAT_EN_LONG, "12345679 trillion", 1.2345679E19, Double.class}, + {FORMAT_HI_IN_LONG, "999", 999L, Long.class}, + {FORMAT_HI_IN_LONG, "-999", -999L, Long.class}, + {FORMAT_HI_IN_LONG, "1 \u0939\u091C\u093C\u093E\u0930", 1000L, Long.class}, + {FORMAT_HI_IN_LONG, "-1 \u0939\u091C\u093C\u093E\u0930", -1000L, Long.class}, + {FORMAT_HI_IN_LONG, "3 \u0939\u091C\u093C\u093E\u0930", 3000L, Long.class}, + {FORMAT_HI_IN_LONG, "12345679 \u0916\u0930\u092C", 1234567900000000000L, Long.class}, + {FORMAT_HI_IN_LONG, "-12345679 \u0916\u0930\u092C", -1234567900000000000L, Long.class}, + {FORMAT_JA_JP_SHORT, "-99", -99L, Long.class}, + {FORMAT_JA_JP_SHORT, "1\u4E07", 10000L, Long.class}, + {FORMAT_JA_JP_SHORT, "30\u4E07", 300000L, Long.class}, + {FORMAT_JA_JP_SHORT, "-30\u4E07", -300000L, Long.class}, + {FORMAT_JA_JP_SHORT, "12345679\u5146", 1.2345679E19, Double.class}, + {FORMAT_JA_JP_SHORT, "-12345679\u5146", -1.2345679E19, Double.class}, + {FORMAT_IT_SHORT, "-99", -99L, Long.class}, + {FORMAT_IT_SHORT, "1\u00a0Mln", 1000000L, Long.class}, + {FORMAT_IT_SHORT, "30\u00a0Mln", 30000000L, Long.class}, + {FORMAT_IT_SHORT, "-30\u00a0Mln", -30000000L, Long.class}, + {FORMAT_IT_SHORT, "12345679\u00a0Bln", 1.2345679E19, Double.class}, + {FORMAT_IT_SHORT, "-12345679\u00a0Bln", -1.2345679E19, Double.class}, + {FORMAT_SW_LONG, "-0.0", -0.0, Double.class}, + {FORMAT_SW_LONG, "499", 499L, Long.class}, + {FORMAT_SW_LONG, "elfu 1", 1000L, Long.class}, + {FORMAT_SW_LONG, "elfu 3", 3000L, Long.class}, + {FORMAT_SW_LONG, "elfu 17", 17000L, Long.class}, + {FORMAT_SW_LONG, "elfu -3", -3000L, Long.class}, + {FORMAT_SW_LONG, "499", 499L, Long.class}, + {FORMAT_SW_LONG, "-499", -499L, Long.class}, + {FORMAT_SW_LONG, "elfu 1", 1000L, Long.class}, + {FORMAT_SW_LONG, "elfu 3", 3000L, Long.class}, + {FORMAT_SW_LONG, "elfu -3", -3000L, Long.class}, + {FORMAT_SW_LONG, "elfu 17", 17000L, Long.class}, + {FORMAT_SW_LONG, "trilioni 12345679", 1.2345679E19, Double.class}, + {FORMAT_SW_LONG, "trilioni -12345679", -1.2345679E19, Double.class}, + {FORMAT_SW_LONG, "elfu 599.01", 599010L, Long.class}, + {FORMAT_SW_LONG, "elfu -599.01", -599010L, Long.class}, + {FORMAT_SE_SHORT, "999", 999L, Long.class}, + {FORMAT_SE_SHORT, "8\u00a0mn", 8000000L, Long.class}, + {FORMAT_SE_SHORT, "8\u00a0dt", 8000L, Long.class}, + {FORMAT_SE_SHORT, "12345679\u00a0bn", 1.2345679E19, Double.class}, + {FORMAT_SE_SHORT, "12345679,89\u00a0bn", 1.2345679890000001E19, Double.class}, + {FORMAT_SE_SHORT, "\u2212999", -999L, Long.class}, + {FORMAT_SE_SHORT, "\u22128\u00a0mn", -8000000L, Long.class}, + {FORMAT_SE_SHORT, "\u22128\u00a0dt", -8000L, Long.class}, + {FORMAT_SE_SHORT, "\u221212345679\u00a0bn", -1.2345679E19, Double.class}, + {FORMAT_SE_SHORT, "\u221212345679,89\u00a0bn", -1.2345679890000001E19, Double.class},}; + } + + @DataProvider(name = "exceptionParse") + Object[][] exceptionParseData() { + return new Object[][]{ + // compact number instance, string to parse, null (no o/p; must throws exception) + // no number + {FORMAT_DZ_LONG, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2" + + "\u0F42", null}, + // Invalid prefix + {FORMAT_DZ_LONG, "-\u0F66\u0F9F\u0F7C\u0F44,\u0F0B\u0F55\u0FB2" + + "\u0F42 \u0F23", null}, + // Invalid prefix for en_US + {FORMAT_EN_US_SHORT, "K12,347", null}, + // Invalid prefix for ja_JP + {FORMAT_JA_JP_SHORT, "\u4E071", null}, + // Localized minus sign should be used + {FORMAT_SE_SHORT, "-8\u00a0mn", null},}; + } + + @DataProvider(name = "invalidParse") + Object[][] invalidParseData() { + return new Object[][]{ + // compact number instance, string to parse, parsed number + // Prefix and suffix do not match + {FORMAT_DZ_LONG, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2" + + "\u0F42 \u0F21 KM", 1000L}, + // Exponents are unparseable + {FORMAT_EN_US_SHORT, "-1.05E4K", -1.05}, + // Default instance does not allow grouping + {FORMAT_EN_US_SHORT, "12,347", 12L}, + // Take partial suffix "K" as 1000 for en_US_SHORT patterns + {FORMAT_EN_US_SHORT, "12KM", 12000L}, + // Invalid suffix + {FORMAT_HI_IN_LONG, "-1 \u00a0\u0915.", -1L},}; + } + + @DataProvider(name = "fieldPosition") + Object[][] formatFieldPositionData() { + return new Object[][]{ + //compact number instance, number to format, field, start position, end position, formatted string + {FORMAT_DZ_LONG, -3500, NumberFormat.Field.SIGN, 0, 1, "-\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2\u0F42 \u0F24"}, + {FORMAT_DZ_LONG, 3500, NumberFormat.Field.INTEGER, 9, 10, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2\u0F42 \u0F24"}, + {FORMAT_DZ_LONG, -3500, NumberFormat.Field.INTEGER, 10, 11, "-\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2\u0F42 \u0F24"}, + {FORMAT_DZ_LONG, 999, NumberFormat.Field.INTEGER, 0, 3, "\u0F29\u0F29\u0F29"}, + {FORMAT_DZ_LONG, -999, NumberFormat.Field.INTEGER, 1, 4, "-\u0F29\u0F29\u0F29"}, + {FORMAT_DZ_LONG, 3500, NumberFormat.Field.PREFIX, 0, 9, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2\u0F42 \u0F24"}, + {FORMAT_DZ_LONG, -3500, NumberFormat.Field.PREFIX, 0, 10, "-\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2\u0F42 \u0F24"}, + {FORMAT_DZ_LONG, 999, NumberFormat.Field.PREFIX, 0, 0, "\u0F29\u0F29\u0F29"}, + {FORMAT_EN_US_SHORT, -3500, NumberFormat.Field.SIGN, 0, 1, "-4K"}, + {FORMAT_EN_US_SHORT, 3500, NumberFormat.Field.INTEGER, 0, 1, "4K"}, + {FORMAT_EN_US_SHORT, 14900000067L, NumberFormat.Field.INTEGER, 0, 2, "15B"}, + {FORMAT_EN_US_SHORT, -1000, NumberFormat.Field.PREFIX, 0, 1, "-1K"}, + {FORMAT_EN_US_SHORT, 3500, NumberFormat.Field.SUFFIX, 1, 2, "4K"}, + {FORMAT_EN_US_SHORT, 14900000067L, NumberFormat.Field.SUFFIX, 2, 3, "15B"}, + {FORMAT_EN_LONG, 3500, NumberFormat.Field.INTEGER, 0, 1, "4 thousand"}, + {FORMAT_EN_LONG, 14900000067L, NumberFormat.Field.INTEGER, 0, 2, "15 billion"}, + {FORMAT_EN_LONG, 3500, NumberFormat.Field.SUFFIX, 1, 10, "4 thousand"}, + {FORMAT_EN_LONG, 14900000067L, NumberFormat.Field.SUFFIX, 2, 10, "15 billion"}, + {FORMAT_JA_JP_SHORT, 14900000067L, NumberFormat.Field.INTEGER, 0, 3, "149\u5104"}, + {FORMAT_JA_JP_SHORT, -999.99, NumberFormat.Field.INTEGER, 1, 6, "-1,000"}, + {FORMAT_JA_JP_SHORT, 14900000067L, NumberFormat.Field.SUFFIX, 3, 4, "149\u5104"}, + {FORMAT_JA_JP_SHORT, -999.99, NumberFormat.Field.SUFFIX, 0, 0, "-1,000"}, + {FORMAT_JA_JP_SHORT, -999.99, NumberFormat.Field.SIGN, 0, 1, "-1,000"}, + {FORMAT_HI_IN_LONG, -14900000067L, NumberFormat.Field.SIGN, 0, 1, + "-15 \u0905\u0930\u092C"}, + {FORMAT_HI_IN_LONG, 3500, NumberFormat.Field.INTEGER, 0, 1, + "4 \u0939\u091C\u093C\u093E\u0930"}, + {FORMAT_HI_IN_LONG, 14900000067L, NumberFormat.Field.INTEGER, 0, 2, + "15 \u0905\u0930\u092C"}, + {FORMAT_HI_IN_LONG, 3500, NumberFormat.Field.SUFFIX, 1, 7, + "4 \u0939\u091C\u093C\u093E\u0930"}, + {FORMAT_HI_IN_LONG, 14900000067L, NumberFormat.Field.SUFFIX, 2, 6, + "15 \u0905\u0930\u092C"}, + {FORMAT_SE_SHORT, 8000000L, NumberFormat.Field.SUFFIX, 1, 4, "8\u00a0mn"}, + {FORMAT_SE_SHORT, 8000.98, NumberFormat.Field.SUFFIX, 1, 4, "8\u00a0dt"}, + {FORMAT_SE_SHORT, new BigInteger("12345678901234567890"), NumberFormat.Field.SUFFIX, 8, 11, "12345679\u00a0bn"}, + {FORMAT_SE_SHORT, new BigDecimal("12345678901234567890.98"), NumberFormat.Field.SUFFIX, 8, 11, "12345679\u00a0bn"}, + {FORMAT_SE_SHORT, -8000000L, NumberFormat.Field.INTEGER, 1, 2, "\u22128\u00a0mn"}, + {FORMAT_SE_SHORT, -8000.98, NumberFormat.Field.SIGN, 0, 1, "\u22128\u00a0dt"}, + {FORMAT_SE_SHORT, new BigDecimal("-48982865901234567890.98"), NumberFormat.Field.INTEGER, 1, 9, "\u221248982866\u00a0bn"},}; + } + + @DataProvider(name = "varParsePosition") + Object[][] varParsePosition() { + return new Object[][]{ + // compact number instance, parse string, parsed number, + // start position, end position, error index + {FORMAT_DZ_LONG, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2" + + "\u0F42 \u0F21 KM", 1000L, 0, 10, -1}, + // Invalid prefix returns null + {FORMAT_DZ_LONG, "Number is: -\u0F66\u0F9F\u0F7C\u0F44,\u0F0B\u0F55\u0FB2" + + "\u0F42 \u0F23", null, 11, 11, 11}, + // Returns null + {FORMAT_DZ_LONG, "\u0F66\u0F9F\u0F7C\u0F44\u0F0B\u0F55\u0FB2" + + "\u0F42", null, 0, 0, 0}, + {FORMAT_EN_US_SHORT, "Exponent: -1.05E4K", -1.05, 10, 15, -1}, + // Default instance does not allow grouping + {FORMAT_EN_US_SHORT, "12,347", 12L, 0, 2, -1}, + // Invalid suffix "KM" for en_US_SHORT patterns + {FORMAT_EN_US_SHORT, "12KM", 12000L, 0, 3, -1}, + // Invalid suffix + {FORMAT_HI_IN_LONG, "-1 \u00a0\u0915.", -1L, 0, 2, -1}, + {FORMAT_EN_LONG, "Number is: 12345679 trillion", + 1.2345679E19, 11, 28, -1}, + {FORMAT_EN_LONG, "Number is: -12345679 trillion", + -1.2345679E19, 11, 29, -1}, + {FORMAT_EN_LONG, "parse 12 thousand and four", 12000L, 6, 17, -1},}; + } + + @Test + public void testInstanceCreation() { + Stream.of(NumberFormat.getAvailableLocales()).forEach(l -> NumberFormat + .getCompactNumberInstance(l, NumberFormat.Style.SHORT).format(10000)); + Stream.of(NumberFormat.getAvailableLocales()).forEach(l -> NumberFormat + .getCompactNumberInstance(l, NumberFormat.Style.LONG).format(10000)); + } + + @Test(dataProvider = "format") + public void testFormat(NumberFormat cnf, Object number, + String expected) { + CompactFormatAndParseHelper.testFormat(cnf, number, expected); + } + + @Test(dataProvider = "parse") + public void testParse(NumberFormat cnf, String parseString, + Number expected, Class returnType) throws ParseException { + CompactFormatAndParseHelper.testParse(cnf, parseString, expected, null, returnType); + } + + @Test(dataProvider = "parse") + public void testParsePosition(NumberFormat cnf, String parseString, + Number expected, Class returnType) throws ParseException { + ParsePosition pos = new ParsePosition(0); + CompactFormatAndParseHelper.testParse(cnf, parseString, expected, pos, returnType); + assertEquals(pos.getIndex(), parseString.length()); + assertEquals(pos.getErrorIndex(), -1); + } + + @Test(dataProvider = "varParsePosition") + public void testVarParsePosition(NumberFormat cnf, String parseString, + Number expected, int startPosition, int indexPosition, + int errPosition) throws ParseException { + ParsePosition pos = new ParsePosition(startPosition); + CompactFormatAndParseHelper.testParse(cnf, parseString, expected, pos, null); + assertEquals(pos.getIndex(), indexPosition); + assertEquals(pos.getErrorIndex(), errPosition); + } + + @Test(dataProvider = "exceptionParse", expectedExceptions = ParseException.class) + public void throwsParseException(NumberFormat cnf, String parseString, + Number expected) throws ParseException { + CompactFormatAndParseHelper.testParse(cnf, parseString, expected, null, null); + } + + @Test(dataProvider = "invalidParse") + public void testInvalidParse(NumberFormat cnf, String parseString, + Number expected) throws ParseException { + CompactFormatAndParseHelper.testParse(cnf, parseString, expected, null, null); + } + + @Test(dataProvider = "fieldPosition") + public void testFormatWithFieldPosition(NumberFormat nf, + Object number, Format.Field field, int posStartExpected, + int posEndExpected, String expected) { + FieldPosition pos = new FieldPosition(field); + StringBuffer buf = new StringBuffer(); + StringBuffer result = nf.format(number, buf, pos); + assertEquals(result.toString(), expected, "Incorrect formatting of the number '" + + number + "'"); + assertEquals(pos.getBeginIndex(), posStartExpected, "Incorrect start position" + + " while formatting the number '" + number + "', for the field " + field); + assertEquals(pos.getEndIndex(), posEndExpected, "Incorrect end position" + + " while formatting the number '" + number + "', for the field " + field); + } + +}