/* * Copyright (c) 2018, 2019, 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 8217721 8222756 * @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); private static final NumberFormat FORMAT_DE_LONG = NumberFormat .getCompactNumberInstance(Locale.GERMAN, NumberFormat.Style.LONG); private static final NumberFormat FORMAT_SL_LONG = NumberFormat .getCompactNumberInstance(new Locale("sl"), NumberFormat.Style.LONG); @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 milers"}, {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"}, // Plurals // DE: one:i = 1 and v = 0 {FORMAT_DE_LONG, 1_000_000, "1 Million"}, {FORMAT_DE_LONG, 2_000_000, "2 Millionen"}, // SL: one:v = 0 and i % 100 = 1 // two:v = 0 and i % 100 = 2 // few:v = 0 and i % 100 = 3..4 or v != 0 {FORMAT_SL_LONG, 1_000_000, "1 milijon"}, {FORMAT_SL_LONG, 2_000_000, "2 milijona"}, {FORMAT_SL_LONG, 3_000_000, "3 milijone"}, {FORMAT_SL_LONG, 5_000_000, "5 milijonov"}, }; } @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}, // Plurals // DE: one:i = 1 and v = 0 {FORMAT_DE_LONG, "1 Million", 1_000_000L, Long.class}, {FORMAT_DE_LONG, "2 Millionen", 2_000_000L, Long.class}, // SL: one:v = 0 and i % 100 = 1 // two:v = 0 and i % 100 = 2 // few:v = 0 and i % 100 = 3..4 or v != 0 {FORMAT_SL_LONG, "1 milijon", 1_000_000L, Long.class}, {FORMAT_SL_LONG, "2 milijona", 2_000_000L, Long.class}, {FORMAT_SL_LONG, "3 milijone", 3_000_000L, Long.class}, {FORMAT_SL_LONG, "5 milijonov", 5_000_000L, Long.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}, // invalid plurals {FORMAT_DE_LONG, "2 Million", 2L}, {FORMAT_SL_LONG, "2 milijon", 2L}, {FORMAT_SL_LONG, "2 milijone", 2L}, {FORMAT_SL_LONG, "2 milijonv", 2L}, {FORMAT_SL_LONG, "3 milijon", 3L}, {FORMAT_SL_LONG, "3 milijona", 3L}, {FORMAT_SL_LONG, "3 milijonv", 3L}, {FORMAT_SL_LONG, "5 milijon", 5L}, {FORMAT_SL_LONG, "5 milijona", 5L}, {FORMAT_SL_LONG, "5 milijone", 5L}, }; } @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(expectedExceptions = IllegalArgumentException.class) public void testFormatWithNullParam() { FORMAT_EN_US_SHORT.format(null); } @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); } }