Unverified Commit 47bba449 authored by Shi-Hao Hong's avatar Shi-Hao Hong Committed by GitHub

L10n Currency Formatting, Fixes to Named vs Positional Parameters (#48445)

* Add currency formatting for l10n tool, fix positional vs named params
parent ec2d5833
...@@ -232,11 +232,37 @@ const Set<String> allowableDateFormats = <String>{ ...@@ -232,11 +232,37 @@ const Set<String> allowableDateFormats = <String>{
// * <https://pub.dev/documentation/intl/latest/intl/NumberFormat-class.html> // * <https://pub.dev/documentation/intl/latest/intl/NumberFormat-class.html>
const Set<String> allowableNumberFormats = <String>{ const Set<String> allowableNumberFormats = <String>{
'compact', 'compact',
'compactCurrency',
'compactSimpleCurrency',
'compactLong', 'compactLong',
'currency',
'decimalPattern', 'decimalPattern',
'decimalPercentPattern', 'decimalPercentPattern',
'percentPattern', 'percentPattern',
'scientificPattern', 'scientificPattern',
'simpleCurrency',
};
// The names of the NumberFormat factory constructors which have named
// parameters rather than positional parameters.
//
// This helps the tool correctly generate number formmatting code correctly.
//
// Example of code that uses named parameters:
// final NumberFormat format = NumberFormat.compact(
// locale: _localeName,
// );
//
// Example of code that uses positional parameters:
// final NumberFormat format = NumberFormat.scientificPattern(_localeName);
const Set<String> numberFormatsWithNamedParameters = <String>{
'compact',
'compactCurrency',
'compactSimpleCurrency',
'compactLong',
'currency',
'decimalPercentPattern',
'simpleCurrency',
}; };
bool _isDateParameter(Map<String, dynamic> placeholderValue) => placeholderValue['type'] == 'DateTime'; bool _isDateParameter(Map<String, dynamic> placeholderValue) => placeholderValue['type'] == 'DateTime';
...@@ -330,6 +356,7 @@ String generateNumberFormattingLogic(Map<String, dynamic> arbBundle, String reso ...@@ -330,6 +356,7 @@ String generateNumberFormattingLogic(Map<String, dynamic> arbBundle, String reso
for (final String placeholder in placeholders.keys) { for (final String placeholder in placeholders.keys) {
final dynamic value = placeholders[placeholder]; final dynamic value = placeholders[placeholder];
if (value is Map<String, dynamic> && _isValidNumberFormat(value, placeholder)) { if (value is Map<String, dynamic> && _isValidNumberFormat(value, placeholder)) {
if (numberFormatsWithNamedParameters.contains(value['format'])) {
if (value.containsKey('optionalParameters')) { if (value.containsKey('optionalParameters')) {
final Map<String, dynamic> optionalParameters = value['optionalParameters'] as Map<String, dynamic>; final Map<String, dynamic> optionalParameters = value['optionalParameters'] as Map<String, dynamic>;
for (final String parameter in optionalParameters.keys) for (final String parameter in optionalParameters.keys)
...@@ -343,6 +370,13 @@ String generateNumberFormattingLogic(Map<String, dynamic> arbBundle, String reso ...@@ -343,6 +370,13 @@ String generateNumberFormattingLogic(Map<String, dynamic> arbBundle, String reso
); );
final String ${placeholder}String = ${placeholder}NumberFormat.format($placeholder); final String ${placeholder}String = ${placeholder}NumberFormat.format($placeholder);
'''); ''');
} else {
result.write('''
final NumberFormat ${placeholder}NumberFormat = NumberFormat.${value['format']}(_localeName);
final String ${placeholder}String = ${placeholder}NumberFormat.format($placeholder);
''');
}
} }
} }
......
...@@ -962,7 +962,7 @@ void main() { ...@@ -962,7 +962,7 @@ void main() {
"placeholders": { "placeholders": {
"progress": { "progress": {
"type": "Number", "type": "Number",
"format": "percentPattern" "format": "compact"
} }
} }
} }
...@@ -990,7 +990,7 @@ void main() { ...@@ -990,7 +990,7 @@ void main() {
expect( expect(
generator.classMethods.first, generator.classMethods.first,
''' String courseCompletion(Object progress) { ''' String courseCompletion(Object progress) {
final NumberFormat progressNumberFormat = NumberFormat.percentPattern( final NumberFormat progressNumberFormat = NumberFormat.compact(
locale: _localeName, locale: _localeName,
); );
final String progressString = progressNumberFormat.format(progress); final String progressString = progressNumberFormat.format(progress);
...@@ -1006,15 +1006,26 @@ void main() { ...@@ -1006,15 +1006,26 @@ void main() {
'''); ''');
}); });
test('correctly adds optional parameters to numbers', () { test('correctly adds optional named parameters to numbers', () {
const String singleNumberMessage = '''{ const Set<String> numberFormatsWithNamedParameters = <String>{
'compact',
'compactCurrency',
'compactSimpleCurrency',
'compactLong',
'currency',
'decimalPercentPattern',
'simpleCurrency',
};
for (final String numberFormat in numberFormatsWithNamedParameters) {
final String singleNumberMessage = '''{
"courseCompletion": "You have completed {progress} of the course.", "courseCompletion": "You have completed {progress} of the course.",
"@courseCompletion": { "@courseCompletion": {
"description": "The amount of progress the student has made in their class.", "description": "The amount of progress the student has made in their class.",
"placeholders": { "placeholders": {
"progress": { "progress": {
"type": "Number", "type": "Number",
"format": "decimalPercentPattern", "format": "$numberFormat",
"optionalParameters": { "optionalParameters": {
"decimalDigits": 2 "decimalDigits": 2
} }
...@@ -1045,7 +1056,7 @@ void main() { ...@@ -1045,7 +1056,7 @@ void main() {
expect( expect(
generator.classMethods.first, generator.classMethods.first,
''' String courseCompletion(Object progress) { ''' String courseCompletion(Object progress) {
final NumberFormat progressNumberFormat = NumberFormat.decimalPercentPattern( final NumberFormat progressNumberFormat = NumberFormat.$numberFormat(
locale: _localeName, locale: _localeName,
decimalDigits: 2, decimalDigits: 2,
); );
...@@ -1060,6 +1071,65 @@ void main() { ...@@ -1060,6 +1071,65 @@ void main() {
); );
} }
'''); ''');
}
});
test('correctly adds optional positional parameters to numbers', () {
const Set<String> numberFormatsWithPositionalParameters = <String>{
'decimalPattern',
'percentPattern',
'scientificPattern',
};
for (final String numberFormat in numberFormatsWithPositionalParameters) {
final String singleNumberMessage = '''{
"courseCompletion": "You have completed {progress} of the course.",
"@courseCompletion": {
"description": "The amount of progress the student has made in their class.",
"placeholders": {
"progress": {
"type": "Number",
"format": "$numberFormat"
}
}
}
}''';
final Directory l10nDirectory = fs.currentDirectory.childDirectory('lib').childDirectory('l10n')
..createSync(recursive: true);
l10nDirectory.childFile(defaultTemplateArbFileName)
.writeAsStringSync(singleNumberMessage);
final LocalizationsGenerator generator = LocalizationsGenerator(fs);
try {
generator.initialize(
l10nDirectoryPath: defaultArbPathString,
templateArbFileName: defaultTemplateArbFileName,
outputFileString: defaultOutputFileString,
classNameString: defaultClassNameString,
);
generator.parseArbFiles();
generator.generateClassMethods();
} on Exception catch (e) {
fail('Parsing template arb file should succeed: \n$e');
}
expect(generator.classMethods, isNotEmpty);
expect(
generator.classMethods.first,
''' String courseCompletion(Object progress) {
final NumberFormat progressNumberFormat = NumberFormat.$numberFormat(_localeName);
final String progressString = progressNumberFormat.format(progress);
return Intl.message(
r'You have completed \$progressString of the course.',
locale: _localeName,
name: 'courseCompletion',
desc: r'The amount of progress the student has made in their class.',
args: <Object>[progressString]
);
}
''');
}
}); });
test('throws an exception when improperly formatted number is passed in', () { test('throws an exception when improperly formatted number is passed in', () {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment