mirror of
https://github.com/RGBCube/serenity
synced 2025-07-25 14:07:46 +00:00
LibJS: Implement Intl.NumberFormat V3's [[SignDisplay]] changes
Intl.NumberFormat V3 adds a "negative" option for [[SignDisplay]] to only use the locale's signed pattern for negative numbers.
This commit is contained in:
parent
733192089f
commit
cff2d631da
3 changed files with 254 additions and 7 deletions
|
@ -1139,6 +1139,7 @@ RawFormatResult to_raw_fixed(GlobalObject& global_object, Value number, int min_
|
||||||
}
|
}
|
||||||
|
|
||||||
// 15.5.11 GetNumberFormatPattern ( numberFormat, x ), https://tc39.es/ecma402/#sec-getnumberformatpattern
|
// 15.5.11 GetNumberFormatPattern ( numberFormat, x ), https://tc39.es/ecma402/#sec-getnumberformatpattern
|
||||||
|
// 1.1.14 GetNumberFormatPattern ( numberFormat, x ), https://tc39.es/proposal-intl-numberformat-v3/out/numberformat/proposed.html#sec-getnumberformatpattern
|
||||||
Optional<Variant<StringView, String>> get_number_format_pattern(GlobalObject& global_object, NumberFormat& number_format, Value number, Unicode::NumberFormat& found_pattern)
|
Optional<Variant<StringView, String>> get_number_format_pattern(GlobalObject& global_object, NumberFormat& number_format, Value number, Unicode::NumberFormat& found_pattern)
|
||||||
{
|
{
|
||||||
// 1. Let localeData be %NumberFormat%.[[LocaleData]].
|
// 1. Let localeData be %NumberFormat%.[[LocaleData]].
|
||||||
|
@ -1264,20 +1265,34 @@ Optional<Variant<StringView, String>> get_number_format_pattern(GlobalObject& gl
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
// 15. Else,
|
// 15. Else if signDisplay is "exceptZero", then
|
||||||
case NumberFormat::SignDisplay::ExceptZero:
|
case NumberFormat::SignDisplay::ExceptZero:
|
||||||
// a. Assert: signDisplay is "exceptZero".
|
// a. If x is NaN, or if x is finite and ℝ(x) is 0, then
|
||||||
// b. If x is NaN, or if x is finite and ℝ(x) is 0, then
|
|
||||||
if (is_positive_zero || is_negative_zero || is_nan) {
|
if (is_positive_zero || is_negative_zero || is_nan) {
|
||||||
// i. Let pattern be patterns.[[zeroPattern]].
|
// i. Let pattern be patterns.[[zeroPattern]].
|
||||||
pattern = patterns->zero_format;
|
pattern = patterns->zero_format;
|
||||||
}
|
}
|
||||||
// c. Else if ℝ(x) > 0, then
|
// b. Else if ℝ(x) > 0, then
|
||||||
else if (is_greater_than(number, 0)) {
|
else if (is_greater_than(number, 0)) {
|
||||||
// i. Let pattern be patterns.[[positivePattern]].
|
// i. Let pattern be patterns.[[positivePattern]].
|
||||||
pattern = patterns->positive_format;
|
pattern = patterns->positive_format;
|
||||||
}
|
}
|
||||||
// d. Else,
|
// c. Else,
|
||||||
|
else {
|
||||||
|
// i. Let pattern be patterns.[[negativePattern]].
|
||||||
|
pattern = patterns->negative_format;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
// 16. Else,
|
||||||
|
case NumberFormat::SignDisplay::Negative:
|
||||||
|
// a. Assert: signDisplay is "negative".
|
||||||
|
// b. If x is 0 or x is -0 or x > 0 or x is NaN, then
|
||||||
|
if (is_positive_zero || is_negative_zero || is_greater_than(number, 0) || is_nan) {
|
||||||
|
// i. Let pattern be patterns.[[zeroPattern]].
|
||||||
|
pattern = patterns->zero_format;
|
||||||
|
}
|
||||||
|
// c. Else,
|
||||||
else {
|
else {
|
||||||
// i. Let pattern be patterns.[[negativePattern]].
|
// i. Let pattern be patterns.[[negativePattern]].
|
||||||
pattern = patterns->negative_format;
|
pattern = patterns->negative_format;
|
||||||
|
@ -1285,8 +1300,7 @@ Optional<Variant<StringView, String>> get_number_format_pattern(GlobalObject& gl
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
// FIXME: Handle all NumberFormat V3 [[SignDisplay]] options.
|
VERIFY_NOT_REACHED();
|
||||||
return {};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
found_pattern = patterns.release_value();
|
found_pattern = patterns.release_value();
|
||||||
|
|
|
@ -347,6 +347,20 @@ describe("style=decimal", () => {
|
||||||
expect(ar.format(-1)).toBe("\u061c-\u0661");
|
expect(ar.format(-1)).toBe("\u061c-\u0661");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("signDisplay=negative", () => {
|
||||||
|
const en = new Intl.NumberFormat("en", { signDisplay: "negative" });
|
||||||
|
expect(en.format(0)).toBe("0");
|
||||||
|
expect(en.format(1)).toBe("1");
|
||||||
|
expect(en.format(-0)).toBe("0");
|
||||||
|
expect(en.format(-1)).toBe("-1");
|
||||||
|
|
||||||
|
const ar = new Intl.NumberFormat("ar", { signDisplay: "negative" });
|
||||||
|
expect(ar.format(0)).toBe("\u0660");
|
||||||
|
expect(ar.format(1)).toBe("\u0661");
|
||||||
|
expect(ar.format(-0)).toBe("\u0660");
|
||||||
|
expect(ar.format(-1)).toBe("\u061c-\u0661");
|
||||||
|
});
|
||||||
|
|
||||||
test("useGrouping=always", () => {
|
test("useGrouping=always", () => {
|
||||||
const en = new Intl.NumberFormat("en", { useGrouping: "always" });
|
const en = new Intl.NumberFormat("en", { useGrouping: "always" });
|
||||||
expect(en.format(123)).toBe("123");
|
expect(en.format(123)).toBe("123");
|
||||||
|
@ -626,6 +640,20 @@ describe("style=percent", () => {
|
||||||
expect(ar.format(-0.0)).toBe("\u0660\u066a\u061c");
|
expect(ar.format(-0.0)).toBe("\u0660\u066a\u061c");
|
||||||
expect(ar.format(-0.01)).toBe("\u061c-\u0661\u066a\u061c");
|
expect(ar.format(-0.01)).toBe("\u061c-\u0661\u066a\u061c");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("signDisplay=negative", () => {
|
||||||
|
const en = new Intl.NumberFormat("en", { style: "percent", signDisplay: "negative" });
|
||||||
|
expect(en.format(0.0)).toBe("0%");
|
||||||
|
expect(en.format(0.01)).toBe("1%");
|
||||||
|
expect(en.format(-0.0)).toBe("0%");
|
||||||
|
expect(en.format(-0.01)).toBe("-1%");
|
||||||
|
|
||||||
|
const ar = new Intl.NumberFormat("ar", { style: "percent", signDisplay: "negative" });
|
||||||
|
expect(ar.format(0.0)).toBe("\u0660\u066a\u061c");
|
||||||
|
expect(ar.format(0.01)).toBe("\u0661\u066a\u061c");
|
||||||
|
expect(ar.format(-0.0)).toBe("\u0660\u066a\u061c");
|
||||||
|
expect(ar.format(-0.01)).toBe("\u061c-\u0661\u066a\u061c");
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("style=currency", () => {
|
describe("style=currency", () => {
|
||||||
|
@ -952,6 +980,50 @@ describe("style=currency", () => {
|
||||||
expect(ar2.format(-0)).toBe("\u0660\u066b\u0660\u0660\u00a0US$");
|
expect(ar2.format(-0)).toBe("\u0660\u066b\u0660\u0660\u00a0US$");
|
||||||
expect(ar2.format(-1)).toBe("\u061c-\u0661\u066b\u0660\u0660\u00a0US$");
|
expect(ar2.format(-1)).toBe("\u061c-\u0661\u066b\u0660\u0660\u00a0US$");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("signDisplay=negative", () => {
|
||||||
|
const en1 = new Intl.NumberFormat("en", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(en1.format(0)).toBe("$0.00");
|
||||||
|
expect(en1.format(1)).toBe("$1.00");
|
||||||
|
expect(en1.format(-0)).toBe("$0.00");
|
||||||
|
expect(en1.format(-1)).toBe("-$1.00");
|
||||||
|
|
||||||
|
const en2 = new Intl.NumberFormat("en", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
currencySign: "accounting",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(en2.format(0)).toBe("$0.00");
|
||||||
|
expect(en2.format(1)).toBe("$1.00");
|
||||||
|
expect(en2.format(-0)).toBe("$0.00");
|
||||||
|
expect(en2.format(-1)).toBe("($1.00)");
|
||||||
|
|
||||||
|
const ar1 = new Intl.NumberFormat("ar", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(ar1.format(0)).toBe("\u0660\u066b\u0660\u0660\u00a0US$");
|
||||||
|
expect(ar1.format(1)).toBe("\u0661\u066b\u0660\u0660\u00a0US$");
|
||||||
|
expect(ar1.format(-0)).toBe("\u0660\u066b\u0660\u0660\u00a0US$");
|
||||||
|
expect(ar1.format(-1)).toBe("\u061c-\u0661\u066b\u0660\u0660\u00a0US$");
|
||||||
|
|
||||||
|
const ar2 = new Intl.NumberFormat("ar", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
currencySign: "accounting",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(ar2.format(0)).toBe("\u0660\u066b\u0660\u0660\u00a0US$");
|
||||||
|
expect(ar2.format(1)).toBe("\u0661\u066b\u0660\u0660\u00a0US$");
|
||||||
|
expect(ar2.format(-0)).toBe("\u0660\u066b\u0660\u0660\u00a0US$");
|
||||||
|
expect(ar2.format(-1)).toBe("\u061c-\u0661\u066b\u0660\u0660\u00a0US$");
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("style=unit", () => {
|
describe("style=unit", () => {
|
||||||
|
|
|
@ -171,6 +171,26 @@ describe("style=decimal", () => {
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("signDisplay=negative", () => {
|
||||||
|
const en = new Intl.NumberFormat("en", { signDisplay: "negative" });
|
||||||
|
expect(en.formatToParts(0)).toEqual([{ type: "integer", value: "0" }]);
|
||||||
|
expect(en.formatToParts(1)).toEqual([{ type: "integer", value: "1" }]);
|
||||||
|
expect(en.formatToParts(-0)).toEqual([{ type: "integer", value: "0" }]);
|
||||||
|
expect(en.formatToParts(-1)).toEqual([
|
||||||
|
{ type: "minusSign", value: "-" },
|
||||||
|
{ type: "integer", value: "1" },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const ar = new Intl.NumberFormat("ar", { signDisplay: "negative" });
|
||||||
|
expect(ar.formatToParts(0)).toEqual([{ type: "integer", value: "\u0660" }]);
|
||||||
|
expect(ar.formatToParts(1)).toEqual([{ type: "integer", value: "\u0661" }]);
|
||||||
|
expect(ar.formatToParts(-0)).toEqual([{ type: "integer", value: "\u0660" }]);
|
||||||
|
expect(ar.formatToParts(-1)).toEqual([
|
||||||
|
{ type: "minusSign", value: "\u061c-" },
|
||||||
|
{ type: "integer", value: "\u0661" },
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
test("useGrouping=always", () => {
|
test("useGrouping=always", () => {
|
||||||
const en = new Intl.NumberFormat("en", { useGrouping: "always" });
|
const en = new Intl.NumberFormat("en", { useGrouping: "always" });
|
||||||
expect(en.formatToParts(1234)).toEqual([
|
expect(en.formatToParts(1234)).toEqual([
|
||||||
|
@ -588,6 +608,46 @@ describe("style=percent", () => {
|
||||||
{ type: "percentSign", value: "\u066a\u061c" },
|
{ type: "percentSign", value: "\u066a\u061c" },
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("signDisplay=negative", () => {
|
||||||
|
const en = new Intl.NumberFormat("en", { style: "percent", signDisplay: "negative" });
|
||||||
|
expect(en.formatToParts(0.0)).toEqual([
|
||||||
|
{ type: "integer", value: "0" },
|
||||||
|
{ type: "percentSign", value: "%" },
|
||||||
|
]);
|
||||||
|
expect(en.formatToParts(0.01)).toEqual([
|
||||||
|
{ type: "integer", value: "1" },
|
||||||
|
{ type: "percentSign", value: "%" },
|
||||||
|
]);
|
||||||
|
expect(en.formatToParts(-0.0)).toEqual([
|
||||||
|
{ type: "integer", value: "0" },
|
||||||
|
{ type: "percentSign", value: "%" },
|
||||||
|
]);
|
||||||
|
expect(en.formatToParts(-0.01)).toEqual([
|
||||||
|
{ type: "minusSign", value: "-" },
|
||||||
|
{ type: "integer", value: "1" },
|
||||||
|
{ type: "percentSign", value: "%" },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const ar = new Intl.NumberFormat("ar", { style: "percent", signDisplay: "negative" });
|
||||||
|
expect(ar.formatToParts(0.0)).toEqual([
|
||||||
|
{ type: "integer", value: "\u0660" },
|
||||||
|
{ type: "percentSign", value: "\u066a\u061c" },
|
||||||
|
]);
|
||||||
|
expect(ar.formatToParts(0.01)).toEqual([
|
||||||
|
{ type: "integer", value: "\u0661" },
|
||||||
|
{ type: "percentSign", value: "\u066a\u061c" },
|
||||||
|
]);
|
||||||
|
expect(ar.formatToParts(-0.0)).toEqual([
|
||||||
|
{ type: "integer", value: "\u0660" },
|
||||||
|
{ type: "percentSign", value: "\u066a\u061c" },
|
||||||
|
]);
|
||||||
|
expect(ar.formatToParts(-0.01)).toEqual([
|
||||||
|
{ type: "minusSign", value: "\u061c-" },
|
||||||
|
{ type: "integer", value: "\u0661" },
|
||||||
|
{ type: "percentSign", value: "\u066a\u061c" },
|
||||||
|
]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("style=currency", () => {
|
describe("style=currency", () => {
|
||||||
|
@ -1133,6 +1193,107 @@ describe("style=currency", () => {
|
||||||
{ type: "literal", value: ")" },
|
{ type: "literal", value: ")" },
|
||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("signDisplay=negative", () => {
|
||||||
|
const en = new Intl.NumberFormat("en", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(en.formatToParts(0)).toEqual([
|
||||||
|
{ type: "currency", value: "$" },
|
||||||
|
{ type: "integer", value: "0" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "00" },
|
||||||
|
]);
|
||||||
|
expect(en.formatToParts(1)).toEqual([
|
||||||
|
{ type: "currency", value: "$" },
|
||||||
|
{ type: "integer", value: "1" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "00" },
|
||||||
|
]);
|
||||||
|
expect(en.formatToParts(-0)).toEqual([
|
||||||
|
{ type: "currency", value: "$" },
|
||||||
|
{ type: "integer", value: "0" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "00" },
|
||||||
|
]);
|
||||||
|
expect(en.formatToParts(-1)).toEqual([
|
||||||
|
{ type: "minusSign", value: "-" },
|
||||||
|
{ type: "currency", value: "$" },
|
||||||
|
{ type: "integer", value: "1" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "00" },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const ar = new Intl.NumberFormat("ar", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(ar.formatToParts(0)).toEqual([
|
||||||
|
{ type: "integer", value: "\u0660" },
|
||||||
|
{ type: "decimal", value: "\u066b" },
|
||||||
|
{ type: "fraction", value: "\u0660\u0660" },
|
||||||
|
{ type: "literal", value: "\u00a0" },
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
]);
|
||||||
|
expect(ar.formatToParts(1)).toEqual([
|
||||||
|
{ type: "integer", value: "\u0661" },
|
||||||
|
{ type: "decimal", value: "\u066b" },
|
||||||
|
{ type: "fraction", value: "\u0660\u0660" },
|
||||||
|
{ type: "literal", value: "\u00a0" },
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
]);
|
||||||
|
expect(ar.formatToParts(-0)).toEqual([
|
||||||
|
{ type: "integer", value: "\u0660" },
|
||||||
|
{ type: "decimal", value: "\u066b" },
|
||||||
|
{ type: "fraction", value: "\u0660\u0660" },
|
||||||
|
{ type: "literal", value: "\u00a0" },
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
]);
|
||||||
|
expect(ar.formatToParts(-1)).toEqual([
|
||||||
|
{ type: "minusSign", value: "\u061c-" },
|
||||||
|
{ type: "integer", value: "\u0661" },
|
||||||
|
{ type: "decimal", value: "\u066b" },
|
||||||
|
{ type: "fraction", value: "\u0660\u0660" },
|
||||||
|
{ type: "literal", value: "\u00a0" },
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const zh = new Intl.NumberFormat("zh-Hant-u-nu-hanidec", {
|
||||||
|
style: "currency",
|
||||||
|
currency: "USD",
|
||||||
|
currencySign: "accounting",
|
||||||
|
signDisplay: "negative",
|
||||||
|
});
|
||||||
|
expect(zh.formatToParts(0)).toEqual([
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
{ type: "integer", value: "〇" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "〇〇" },
|
||||||
|
]);
|
||||||
|
expect(zh.formatToParts(1)).toEqual([
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
{ type: "integer", value: "一" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "〇〇" },
|
||||||
|
]);
|
||||||
|
expect(zh.formatToParts(-0)).toEqual([
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
{ type: "integer", value: "〇" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "〇〇" },
|
||||||
|
]);
|
||||||
|
expect(zh.formatToParts(-1)).toEqual([
|
||||||
|
{ type: "literal", value: "(" },
|
||||||
|
{ type: "currency", value: "US$" },
|
||||||
|
{ type: "integer", value: "一" },
|
||||||
|
{ type: "decimal", value: "." },
|
||||||
|
{ type: "fraction", value: "〇〇" },
|
||||||
|
{ type: "literal", value: ")" },
|
||||||
|
]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("style=unit", () => {
|
describe("style=unit", () => {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue