Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
7539826d
Commit
7539826d
authored
Jan 28, 2020
by
Hans Muller
Committed by
Flutter GitHub Bot
Jan 28, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Update gen_l10n handling of plurals, numbers, and dates (#49357)
parent
1a3379d5
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
672 additions
and
527 deletions
+672
-527
gen_l10n.dart
dev/tools/localization/bin/gen_l10n.dart
+1
-0
gen_l10n.dart
dev/tools/localization/gen_l10n.dart
+225
-429
gen_l10n_templates.dart
dev/tools/localization/gen_l10n_templates.dart
+169
-0
gen_l10n_types.dart
dev/tools/localization/gen_l10n_types.dart
+154
-0
gen_l10n_test.dart
dev/tools/test/localization/gen_l10n_test.dart
+123
-98
No files found.
dev/tools/localization/bin/gen_l10n.dart
View file @
7539826d
...
...
@@ -10,6 +10,7 @@ import 'package:file/local.dart' as local;
import
'package:path/path.dart'
as
path
;
import
'../gen_l10n.dart'
;
import
'../gen_l10n_types.dart'
;
import
'../localizations_utils.dart'
;
Future
<
void
>
main
(
List
<
String
>
arguments
)
async
{
...
...
dev/tools/localization/gen_l10n.dart
View file @
7539826d
...
...
@@ -9,150 +9,10 @@ import 'package:file/file.dart' as file;
import
'package:meta/meta.dart'
;
import
'package:path/path.dart'
as
path
;
import
'gen_l10n_templates.dart'
;
import
'gen_l10n_types.dart'
;
import
'localizations_utils.dart'
;
const
String
defaultFileTemplate
=
'''
import '
dart:
async
';
import '
package:
flutter
/
widgets
.
dart
';
import '
package:
flutter_localizations
/
flutter_localizations
.
dart
';
import '
package:
intl
/
intl
.
dart
';
import '
messages_all
.
dart
';
/// Callers can lookup localized strings with an instance of @className returned
/// by `@className.of(context)`.
///
/// Applications need to include `@className.delegate()` in their app
\'
s
/// localizationDelegates list, and the locales they support in the app
\'
s
/// supportedLocales list. For example:
///
/// ```
/// import '
@importFile
';
///
/// return MaterialApp(
/// localizationsDelegates: @className.localizationsDelegates,
/// supportedLocales: @className.supportedLocales,
/// home: MyApplicationHome(),
/// );
/// ```
///
/// ## Update pubspec.yaml
///
/// Please make sure to update your pubspec.yaml to include the following
/// packages:
///
/// ```
/// dependencies:
/// # Internationalization support.
/// flutter_localizations:
/// sdk: flutter
/// intl: 0.16.0
/// intl_translation: 0.17.7
///
/// # rest of dependencies
/// ```
///
/// ## iOS Applications
///
/// iOS applications define key application metadata, including supported
/// locales, in an Info.plist file that is built into the application bundle.
/// To configure the locales supported by your app, you’ll need to edit this
/// file.
///
/// First, open your project’s ios/Runner.xcworkspace Xcode workspace file.
/// Then, in the Project Navigator, open the Info.plist file under the Runner
/// project’s Runner folder.
///
/// Next, select the Information Property List item, select Add Item from the
/// Editor menu, then select Localizations from the pop-up menu.
///
/// Select and expand the newly-created Localizations item then, for each
/// locale your application supports, add a new item and select the locale
/// you wish to add from the pop-up menu in the Value field. This list should
/// be consistent with the languages listed in the @className.supportedLocales
/// property.
class @className {
@className(Locale locale) : _localeName = Intl.canonicalizedLocale(locale.toString());
final String _localeName;
static Future<@className> load(Locale locale) {
return initializeMessages(locale.toString())
.then<@className>((_) => @className(locale));
}
static @className of(BuildContext context) {
return Localizations.of<@className>(context, @className);
}
static const LocalizationsDelegate<@className> delegate = _@classNameDelegate();
/// A list of this localizations delegate along with the default localizations
/// delegates.
///
/// Returns a list of localizations delegates containing this delegate along with
/// GlobalMaterialLocalizations.delegate, GlobalCupertinoLocalizations.delegate,
/// and GlobalWidgetsLocalizations.delegate.
///
/// Additional delegates can be added by appending to this list in
/// MaterialApp. This list does not have to be used at all if a custom list
/// of delegates is preferred or required.
static const List<LocalizationsDelegate<dynamic>> localizationsDelegates = <LocalizationsDelegate<dynamic>>[
delegate,
GlobalMaterialLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
];
/// A list of this localizations delegate'
s
supported
locales
.
@supportedLocales
@classMethods
}
class
_
@classNameDelegate
extends
LocalizationsDelegate
<
@className
>
{
const
_
@classNameDelegate
();
@override
Future
<
@className
>
load
(
Locale
locale
)
=>
@className
.
load
(
locale
);
@override
bool
isSupported
(
Locale
locale
)
=>
<
String
>[
@supportedLanguageCodes
].
contains
(
locale
.
languageCode
);
@override
bool
shouldReload
(
_
@classNameDelegate
old
)
=>
false
;
}
''';
const String getterMethodTemplate = '''
String
get
@methodName
{
return
Intl
.
message
(
@message
,
locale:
_localeName
,
@intlMethodArgs
);
}
''';
const String simpleMethodTemplate = '''
String
@methodName
(
@methodParameters
)
{
@dateFormatting@numberFormatting
return
Intl
.
message
(
@message
,
locale:
_localeName
,
@intlMethodArgs
);
}
''';
const String pluralMethodTemplate = '''
String
@methodName
(
@methodParameters
)
{
@dateFormatting@numberFormatting
return
Intl
.
plural
(
@intlMethodArgs
);
}
''';
// The set of date formats that can be automatically localized.
//
// The localizations generation tool makes use of the intl library's
...
...
@@ -265,245 +125,159 @@ const Set<String> numberFormatsWithNamedParameters = <String>{
'simpleCurrency'
,
};
bool
_isDateParameter
(
Map
<
String
,
dynamic
>
placeholderValue
)
=>
placeholderValue
[
'type'
]
==
'DateTime'
;
bool
_isNumberParameter
(
Map
<
String
,
dynamic
>
placeholderValue
)
=>
placeholderValue
[
'type'
]
==
'Number'
;
bool
_containsFormatKey
(
Map
<
String
,
dynamic
>
placeholderValue
,
String
placeholder
)
{
if
(
placeholderValue
.
containsKey
(
'format'
))
return
true
;
throw
L10nException
(
'The placeholder,
$placeholder
, has its "type" resource attribute set to '
'the "
${placeholderValue['type']}
" type. To properly resolve for the right '
'
${placeholderValue['type']}
format, the "format" attribute needs to be set '
'to determine which DateFormat to use.
\n
'
'Check the intl library
\'
s DateFormat class constructors for allowed '
'date formats.'
);
}
bool
_isValidDateParameter
(
Map
<
String
,
dynamic
>
placeholderValue
,
String
placeholder
)
{
if
(
allowableDateFormats
.
contains
(
placeholderValue
[
'format'
]))
return
true
;
throw
L10nException
(
'Date format
${placeholderValue['format']}
for
$placeholder
\n
'
'placeholder does not have a corresponding DateFormat
\n
'
'constructor. Check the intl library
\'
s DateFormat class
\n
'
'constructors for allowed date formats.'
);
}
bool
_isValidNumberParameter
(
Map
<
String
,
dynamic
>
placeholderValue
,
String
placeholder
)
{
if
(
allowableNumberFormats
.
contains
(
placeholderValue
[
'format'
]))
return
true
;
throw
L10nException
(
'Number format
${placeholderValue['format']}
for the
$placeholder
\n
'
'placeholder does not have a corresponding NumberFormat
\n
'
'constructor. Check the intl library
\'
s NumberFormat class
\n
'
'constructors for allowed number formats.'
);
}
List
<
String
>
genMethodParameters
(
Map
<
String
,
dynamic
>
bundle
,
String
resourceId
,
String
type
)
{
final
Map
<
String
,
dynamic
>
attributesMap
=
bundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
if
(
attributesMap
!=
null
&&
attributesMap
.
containsKey
(
'placeholders'
))
{
final
Map
<
String
,
dynamic
>
placeholders
=
attributesMap
[
'placeholders'
]
as
Map
<
String
,
dynamic
>;
return
placeholders
.
keys
.
map
((
String
parameter
)
=>
'
$type
$parameter
'
).
toList
();
List
<
String
>
generateIntlMethodArgs
(
Message
message
)
{
final
List
<
String
>
methodArgs
=
<
String
>[
'name:
\'
${message.resourceId}
\'
'
];
if
(
message
.
description
!=
null
)
methodArgs
.
add
(
'desc:
${generateString(message.description)}
'
);
if
(
message
.
placeholders
.
isNotEmpty
)
{
final
String
args
=
message
.
placeholders
.
map
<
String
>((
Placeholder
placeholder
)
{
return
placeholder
.
name
;
}).
join
(
', '
);
methodArgs
.
add
(
'args: <Object>[
$args
]'
);
}
return
<
String
>[]
;
return
methodArgs
;
}
List
<
String
>
genPluralMethodParameters
(
Iterable
<
String
>
placeholderKeys
,
String
countPlaceholder
,
String
resourceId
)
{
if
(
placeholderKeys
.
isEmpty
)
throw
L10nException
(
'Placeholders map for the
$resourceId
message is empty.
\n
'
'Check to see if the plural message is in the proper ICU syntax format '
'and ensure that placeholders are properly specified.'
);
return
placeholderKeys
.
map
((
String
parameter
)
{
if
(
parameter
==
countPlaceholder
)
{
return
'int
$parameter
'
;
}
return
'Object
$parameter
'
;
List
<
String
>
generateInnerMethodArgs
(
Message
message
)
{
return
message
.
placeholders
.
map
((
Placeholder
placeholder
)
{
final
String
arg
=
placeholder
.
name
;
return
placeholder
.
requiresFormatting
?
'
${arg}
String'
:
arg
;
}).
toList
();
}
String
generateDateFormattingLogic
(
Map
<
String
,
dynamic
>
arbBundle
,
String
resourceId
)
{
String
generateDateFormattingLogic
(
Message
message
)
{
if
(
message
.
placeholders
.
isEmpty
)
return
''
;
final
StringBuffer
result
=
StringBuffer
();
f
inal
Map
<
String
,
dynamic
>
attributesMap
=
arbBundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
if
(
attributesMap
!=
null
&&
attributesMap
.
containsKey
(
'placeholders'
))
{
final
Map
<
String
,
dynamic
>
placeholders
=
attributesMap
[
'placeholders'
]
as
Map
<
String
,
dynamic
>
;
for
(
final
String
placeholder
in
placeholders
.
keys
)
{
final
dynamic
value
=
placeholders
[
placeholder
];
if
(
value
is
Map
<
String
,
dynamic
>
&&
_isValidDateFormat
(
value
,
placeholder
))
{
result
.
write
(
''
'
final DateFormat
${placeholder}
DateFormat = DateFormat.
${value['format']}
(_localeName);
final String
${placeholder}
String =
${placeholder}
DateFormat.format(
$placeholder
);
'''
);
}
f
or
(
final
Placeholder
placeholder
in
message
.
placeholders
)
{
if
(!
placeholder
.
isDate
)
continue
;
if
(
placeholder
.
format
==
null
)
{
throw
L10nException
(
'The placeholder,
${placeholder.name}
, has its "type" resource attribute set to '
'the "
${placeholder.type}
" type. To properly resolve for the right
'
'
${placeholder.type}
format, the "format" attribute needs to be set '
'to determine which DateFormat to use.
\n
'
'Check the intl library
\'
s DateFormat class constructors for allowed '
'date formats.'
);
}
}
if
(!
allowableDateFormats
.
contains
(
placeholder
.
format
))
{
throw
L10nException
(
'Date format "
${placeholder.format}
" for placeholder '
'
${placeholder.name}
does not have a corresponding DateFormat '
'constructor
\n
. Check the intl library
\'
s DateFormat class '
'constructors for allowed date formats.'
);
}
result
.
write
(
'''
final DateFormat
${placeholder.name}
DateFormat = DateFormat.
${placeholder.format}
(_localeName);
final String
${placeholder.name}
String =
${placeholder.name}
DateFormat.format(
${placeholder.name}
);
'''
);
}
return
result
.
toString
();
}
String
generateNumberFormattingLogic
(
Map
<
String
,
dynamic
>
arbBundle
,
String
resourceId
)
{
final
Map
<
String
,
dynamic
>
attributesMap
=
arbBundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
if
(
attributesMap
!=
null
&&
attributesMap
.
containsKey
(
'placeholders'
))
{
final
StringBuffer
result
=
StringBuffer
();
final
Map
<
String
,
dynamic
>
placeholders
=
attributesMap
[
'placeholders'
]
as
Map
<
String
,
dynamic
>;
final
StringBuffer
optionalParametersString
=
StringBuffer
();
for
(
final
String
placeholder
in
placeholders
.
keys
)
{
final
dynamic
value
=
placeholders
[
placeholder
];
if
(
value
is
Map
<
String
,
dynamic
>
&&
_isValidNumberFormat
(
value
,
placeholder
))
{
if
(
numberFormatsWithNamedParameters
.
contains
(
value
[
'format'
]))
{
if
(
value
.
containsKey
(
'optionalParameters'
))
{
final
Map
<
String
,
dynamic
>
optionalParameters
=
value
[
'optionalParameters'
]
as
Map
<
String
,
dynamic
>;
for
(
final
String
parameter
in
optionalParameters
.
keys
)
optionalParametersString
.
write
(
'
\n
$parameter
:
${optionalParameters[parameter]}
,'
);
}
result
.
write
(
'''
String
generateNumberFormattingLogic
(
Message
message
)
{
if
(
message
.
placeholders
.
isEmpty
)
return
''
;
final NumberFormat
${placeholder}
NumberFormat = NumberFormat.
${value['format']}
(
locale: _localeName,@optionalParameters
final
StringBuffer
result
=
StringBuffer
();
for
(
final
Placeholder
placeholder
in
message
.
placeholders
)
{
if
(!
placeholder
.
isNumber
)
continue
;
if
(!
allowableNumberFormats
.
contains
(
placeholder
.
format
))
{
throw
L10nException
(
'Number format
${placeholder.format}
for the
${placeholder.name}
'
'placeholder does not have a corresponding NumberFormat constructor.
\n
'
'Check the intl library
\'
s NumberFormat class constructors for allowed '
'number formats.'
);
}
if
(
numberFormatsWithNamedParameters
.
contains
(
placeholder
.
format
))
{
final
StringBuffer
optionalParametersString
=
StringBuffer
();
for
(
final
OptionalParameter
parameter
in
placeholder
.
optionalParameters
)
optionalParametersString
.
write
(
'
\n
${parameter.name}
:
${parameter.value}
,'
);
result
.
write
(
'''
final NumberFormat
${placeholder.name}
NumberFormat = NumberFormat.
${placeholder.format}
(
locale: _localeName,
${optionalParametersString.toString()}
);
final String
${placeholder
}
String =
${placeholder}
NumberFormat.format(
$placeholder
);
final String
${placeholder
.name}
String =
${placeholder.name}
NumberFormat.format(
${placeholder.name}
);
'''
);
}
else
{
}
else
{
result
.
write
(
'''
final NumberFormat
${placeholder
}
NumberFormat = NumberFormat.
${value['format']
}
(_localeName);
final String
${placeholder
}
String =
${placeholder}
NumberFormat.format(
$placeholder
);
final NumberFormat
${placeholder
.name}
NumberFormat = NumberFormat.
${placeholder.format
}
(_localeName);
final String
${placeholder
.name}
String =
${placeholder.name}
NumberFormat.format(
${placeholder.name}
);
'''
);
}
}
}
return
result
.
toString
()
.
replaceAll
(
'@optionalParameters'
,
optionalParametersString
.
toString
());
}
return
''
;
}
bool
_isValidDateFormat
(
Map
<
String
,
dynamic
>
value
,
String
placeholder
)
{
return
_isDateParameter
(
value
)
&&
_containsFormatKey
(
value
,
placeholder
)
&&
_isValidDateParameter
(
value
,
placeholder
);
}
bool
_isValidNumberFormat
(
Map
<
String
,
dynamic
>
value
,
String
placeholder
)
{
return
_isNumberParameter
(
value
)
&&
_containsFormatKey
(
value
,
placeholder
)
&&
_isValidNumberParameter
(
value
,
placeholder
);
}
bool
_isValidPlaceholder
(
Map
<
String
,
dynamic
>
value
,
String
placeholder
)
{
return
_isValidDateFormat
(
value
,
placeholder
)
||
_isValidNumberFormat
(
value
,
placeholder
);
return
result
.
toString
();
}
List
<
String
>
genIntlMethodArgs
(
Map
<
String
,
dynamic
>
arbBundle
,
String
resourceId
)
{
final
List
<
String
>
attributes
=
<
String
>[
'name:
\'
$resourceId
\'
'
];
final
Map
<
String
,
dynamic
>
attributesMap
=
arbBundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
if
(
attributesMap
!=
null
)
{
if
(
attributesMap
.
containsKey
(
'description'
))
{
final
String
description
=
attributesMap
[
'description'
]
as
String
;
attributes
.
add
(
'desc:
${generateString(description)}
'
);
}
if
(
attributesMap
.
containsKey
(
'placeholders'
))
{
final
Map
<
String
,
dynamic
>
placeholders
=
attributesMap
[
'placeholders'
]
as
Map
<
String
,
dynamic
>;
if
(
placeholders
.
isNotEmpty
)
{
final
List
<
String
>
argumentList
=
<
String
>[];
for
(
final
String
placeholder
in
placeholders
.
keys
)
{
final
dynamic
value
=
placeholders
[
placeholder
];
if
(
value
is
Map
<
String
,
dynamic
>
&&
_isValidPlaceholder
(
value
,
placeholder
))
{
argumentList
.
add
(
'
${placeholder}
String'
);
}
else
{
argumentList
.
add
(
placeholder
);
}
}
final
String
args
=
argumentList
.
join
(
', '
);
attributes
.
add
(
'args: <Object>[
$args
]'
);
}
String
genSimpleMethod
(
Message
message
)
{
String
genSimpleMethodMessage
()
{
String
messageValue
=
message
.
value
;
for
(
final
Placeholder
placeholder
in
message
.
placeholders
)
{
messageValue
=
messageValue
.
replaceAll
(
'{
${placeholder.name}
}'
,
'
\$
${placeholder.name}
'
);
}
final
String
rawMessage
=
generateString
(
messageValue
);
// "r'...'"
return
rawMessage
.
substring
(
1
);
}
return
attributes
;
}
String
genSimpleMethod
(
Map
<
String
,
dynamic
>
arbBundle
,
String
resourceId
)
{
String
genSimpleMethodMessage
(
Map
<
String
,
dynamic
>
arbBundle
,
String
resourceId
)
{
String
message
=
arbBundle
[
resourceId
]
as
String
;
final
Map
<
String
,
dynamic
>
attributesMap
=
arbBundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
final
Map
<
String
,
dynamic
>
placeholders
=
attributesMap
[
'placeholders'
]
as
Map
<
String
,
dynamic
>;
for
(
final
String
placeholder
in
placeholders
.
keys
)
{
final
dynamic
value
=
placeholders
[
placeholder
];
if
(
value
is
Map
<
String
,
dynamic
>
&&
(
_isDateParameter
(
value
)
||
_isNumberParameter
(
value
)))
{
message
=
message
.
replaceAll
(
'{
$placeholder
}'
,
'
\$
${placeholder}
String'
);
}
else
{
message
=
message
.
replaceAll
(
'{
$placeholder
}'
,
'
\$
$placeholder
'
);
}
}
return
generateString
(
message
);
List
<
String
>
genMethodParameters
([
String
type
])
{
return
message
.
placeholders
.
map
((
Placeholder
placeholder
)
{
return
'
${type ?? placeholder.type}
${placeholder.name}
'
;
}).
toList
();
}
final
Map
<
String
,
dynamic
>
attributesMap
=
arbBundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
if
(
attributesMap
==
null
)
throw
L10nException
(
'Resource attribute "@
$resourceId
" was not found. Please ensure that each '
'resource id has a corresponding resource attribute.'
);
if
(
message
.
placeholdersRequireFormatting
)
{
return
formatMethodTemplate
.
replaceAll
(
'@(methodName)'
,
message
.
resourceId
)
.
replaceAll
(
'@(methodParameters)'
,
genMethodParameters
().
join
(
', '
))
.
replaceAll
(
'@(dateFormatting)'
,
generateDateFormattingLogic
(
message
))
.
replaceAll
(
'@(numberFormatting)'
,
generateNumberFormattingLogic
(
message
))
.
replaceAll
(
'@(message)'
,
genSimpleMethodMessage
())
.
replaceAll
(
'@(innerMethodParameters)'
,
genMethodParameters
(
'Object'
).
join
(
', '
))
.
replaceAll
(
'@(innerMethodArgs)'
,
generateInnerMethodArgs
(
message
).
join
(
', '
))
.
replaceAll
(
'@(intlMethodArgs)'
,
generateIntlMethodArgs
(
message
).
join
(
',
\n
'
));
}
if
(
attributesMap
.
containsKey
(
'placeholders'
))
{
final
String
rawMessageString
=
genSimpleMethodMessage
(
arbBundle
,
resourceId
);
// "r'...'"
if
(
message
.
placeholders
.
isNotEmpty
)
{
return
simpleMethodTemplate
.
replaceAll
(
'@methodName'
,
resourceId
)
.
replaceAll
(
'@methodParameters'
,
genMethodParameters
(
arbBundle
,
resourceId
,
'Object'
).
join
(
', '
))
.
replaceAll
(
'@dateFormatting'
,
generateDateFormattingLogic
(
arbBundle
,
resourceId
))
.
replaceAll
(
'@numberFormatting'
,
generateNumberFormattingLogic
(
arbBundle
,
resourceId
))
.
replaceAll
(
'@message'
,
'
${rawMessageString.substring(1)}
'
)
.
replaceAll
(
'@intlMethodArgs'
,
genIntlMethodArgs
(
arbBundle
,
resourceId
).
join
(
',
\n
'
));
.
replaceAll
(
'@(methodName)'
,
message
.
resourceId
)
.
replaceAll
(
'@(methodParameters)'
,
genMethodParameters
().
join
(
', '
))
.
replaceAll
(
'@(message)'
,
genSimpleMethodMessage
())
.
replaceAll
(
'@(intlMethodArgs)'
,
generateIntlMethodArgs
(
message
).
join
(
',
\n
'
));
}
final
String
rawMessageString
=
generateString
(
arbBundle
[
resourceId
]
as
String
);
// "r'...'"
return
getterMethodTemplate
.
replaceAll
(
'@
methodName'
,
resourceId
)
.
replaceAll
(
'@
message'
,
'
${rawMessageString.substring(1
)}
'
)
.
replaceAll
(
'@
intlMethodArgs'
,
genIntlMethodArgs
(
arbBundle
,
resourceId
).
join
(
',
\n
'
));
.
replaceAll
(
'@
(methodName)'
,
message
.
resourceId
)
.
replaceAll
(
'@
(message)'
,
'
${genSimpleMethodMessage(
)}
'
)
.
replaceAll
(
'@
(intlMethodArgs)'
,
generateIntlMethodArgs
(
message
).
join
(
',
\n
'
));
}
String
genPluralMethod
(
Map
<
String
,
dynamic
>
arbBundle
,
String
resourceId
)
{
final
Map
<
String
,
dynamic
>
attributesMap
=
arbBundle
[
'@
$resourceId
'
]
as
Map
<
String
,
dynamic
>;
if
(
attributesMap
==
null
)
throw
L10nException
(
'Resource attribute for
$resourceId
does not exist.'
);
if
(!
attributesMap
.
containsKey
(
'placeholders'
))
String
generatePluralMethod
(
Message
message
)
{
if
(
message
.
placeholders
.
isEmpty
)
{
throw
L10nException
(
'Unable to find placeholders for the plural message:
$
resourceId
.
\n
'
'Unable to find placeholders for the plural message:
$
{message.resourceId}
.
\n
'
'Check to see if the plural message is in the proper ICU syntax format '
'and ensure that placeholders are properly specified.'
);
if
(
attributesMap
[
'placeholders'
]
is
!
Map
<
String
,
dynamic
>)
throw
L10nException
(
'The "placeholders" resource attribute for the message,
$resourceId
, '
'is not properly formatted. Ensure that it is a map with keys that are '
'strings.'
);
final
Map
<
String
,
dynamic
>
placeholdersMap
=
attributesMap
[
'placeholders'
]
as
Map
<
String
,
dynamic
>;
final
Iterable
<
String
>
placeholders
=
placeholdersMap
.
keys
;
// Used to determine which placeholder is the plural count placeholder
final
String
resourceValue
=
arbBundle
[
resourceId
]
as
String
;
final
String
countPlaceholder
=
resourceValue
.
split
(
','
)[
0
].
substring
(
1
);
}
// To make it easier to parse the plurals message, temporarily replace each
// "{placeholder}" parameter with "#placeholder#".
String
message
=
arbBundle
[
resourceId
]
as
String
;
for
(
final
String
placeholder
in
placeholders
)
message
=
message
.
replaceAll
(
'{
$placeholder
}'
,
'#
$placeholder
#'
);
String
easyMessage
=
message
.
value
;
for
(
final
Placeholder
placeholder
in
message
.
placeholders
)
easyMessage
=
easyMessage
.
replaceAll
(
'{
${placeholder.name}
}'
,
'#
${placeholder.name}
#'
);
final
Map
<
String
,
String
>
pluralIds
=
<
String
,
String
>{
const
Map
<
String
,
String
>
pluralIds
=
<
String
,
String
>{
'=0'
:
'zero'
,
'=1'
:
'one'
,
'=2'
:
'two'
,
...
...
@@ -512,94 +286,53 @@ String genPluralMethod(Map<String, dynamic> arbBundle, String resourceId) {
'other'
:
'other'
};
final
List
<
String
>
methodArgs
=
<
String
>[
final
String
countPlaceholder
=
message
.
value
.
split
(
','
)[
0
].
substring
(
1
);
final
List
<
String
>
intlMethodArgs
=
<
String
>[
countPlaceholder
,
'locale: _localeName'
,
...
gen
IntlMethodArgs
(
arbBundle
,
resourceId
),
...
gen
erateIntlMethodArgs
(
message
),
];
for
(
final
String
pluralKey
in
pluralIds
.
keys
)
{
final
RegExp
expRE
=
RegExp
(
'(
$pluralKey
){([^}]+)}'
);
final
RegExpMatch
match
=
expRE
.
firstMatch
(
m
essage
);
final
RegExpMatch
match
=
expRE
.
firstMatch
(
easyM
essage
);
if
(
match
!=
null
&&
match
.
groupCount
==
2
)
{
String
argValue
=
match
.
group
(
2
);
for
(
final
String
placeholder
in
placeholders
)
{
final
dynamic
value
=
placeholdersMap
[
placeholder
];
if
(
value
is
Map
<
String
,
dynamic
>
&&
(
_isDateParameter
(
value
)
||
_isNumberParameter
(
value
)))
{
argValue
=
argValue
.
replaceAll
(
'#
$placeholder
#'
,
'
\$
${placeholder}
String'
);
for
(
final
Placeholder
placeholder
in
message
.
placeholders
)
{
if
(
placeholder
.
requiresFormatting
)
{
argValue
=
argValue
.
replaceAll
(
'#
${placeholder.name}
#'
,
'
\$
${placeholder.name}
String'
);
}
else
{
argValue
=
argValue
.
replaceAll
(
'#
$
placeholder
#'
,
'
\$
$placeholder
'
);
argValue
=
argValue
.
replaceAll
(
'#
$
{placeholder.name}
#'
,
'
\$
${placeholder.name}
'
);
}
}
m
ethodArgs
.
add
(
"
${pluralIds[pluralKey]}
: '
$argValue
'"
);
intlM
ethodArgs
.
add
(
"
${pluralIds[pluralKey]}
: '
$argValue
'"
);
}
}
return
pluralMethodTemplate
.
replaceAll
(
'@methodName'
,
resourceId
)
.
replaceAll
(
'@methodParameters'
,
genPluralMethodParameters
(
placeholders
,
countPlaceholder
,
resourceId
).
join
(
', '
))
.
replaceAll
(
'@dateFormatting'
,
generateDateFormattingLogic
(
arbBundle
,
resourceId
))
.
replaceAll
(
'@numberFormatting'
,
generateNumberFormattingLogic
(
arbBundle
,
resourceId
))
.
replaceAll
(
'@intlMethodArgs'
,
methodArgs
.
join
(
',
\n
'
));
}
String
genSupportedLocaleProperty
(
Set
<
LocaleInfo
>
supportedLocales
)
{
const
String
prefix
=
'static const List<Locale> supportedLocales = <Locale>[
\n
Locale('
;
const
String
suffix
=
'),
\n
];'
;
String
resultingProperty
=
prefix
;
for
(
final
LocaleInfo
locale
in
supportedLocales
)
{
final
String
languageCode
=
locale
.
languageCode
;
final
String
countryCode
=
locale
.
countryCode
;
resultingProperty
+=
'
\'
$languageCode
\'
'
;
if
(
countryCode
!=
null
)
resultingProperty
+=
',
\'
$countryCode
\'
'
;
resultingProperty
+=
'),
\n
Locale('
;
List
<
String
>
generatePluralMethodParameters
([
String
type
])
{
return
message
.
placeholders
.
map
((
Placeholder
placeholder
)
{
final
String
placeholderType
=
placeholder
.
name
==
countPlaceholder
?
'int'
:
(
type
??
placeholder
.
type
);
return
'
$placeholderType
${placeholder.name}
'
;
}).
toList
();
}
resultingProperty
=
resultingProperty
.
substring
(
0
,
resultingProperty
.
length
-
'),
\n
Locale('
.
length
);
resultingProperty
+=
suffix
;
return
resultingProperty
;
}
bool
_isValidClassName
(
String
className
)
{
// Dart class name cannot contain non-alphanumeric symbols
if
(
className
.
contains
(
RegExp
(
r'[^a-zA-Z\d]'
)))
return
false
;
// Dart class name must start with upper case character
if
(
className
[
0
].
contains
(
RegExp
(
r'[a-z]'
)))
return
false
;
// Dart class name cannot start with a number
if
(
className
[
0
].
contains
(
RegExp
(
r'\d'
)))
return
false
;
return
true
;
}
bool
_isNotReadable
(
FileStat
fileStat
)
{
final
String
rawStatString
=
fileStat
.
modeString
();
// Removes potential prepended permission bits, such as '(suid)' and '(guid)'.
final
String
statString
=
rawStatString
.
substring
(
rawStatString
.
length
-
9
);
return
!(
statString
[
0
]
==
'r'
||
statString
[
3
]
==
'r'
||
statString
[
6
]
==
'r'
);
}
bool
_isNotWritable
(
FileStat
fileStat
)
{
final
String
rawStatString
=
fileStat
.
modeString
();
// Removes potential prepended permission bits, such as '(suid)' and '(guid)'.
final
String
statString
=
rawStatString
.
substring
(
rawStatString
.
length
-
9
);
return
!(
statString
[
1
]
==
'w'
||
statString
[
4
]
==
'w'
||
statString
[
7
]
==
'w'
);
}
if
(
message
.
placeholdersRequireFormatting
)
{
return
pluralFormatMethodTemplate
.
replaceAll
(
'@(methodName)'
,
message
.
resourceId
)
.
replaceAll
(
'@(methodParameters)'
,
generatePluralMethodParameters
().
join
(
', '
))
.
replaceAll
(
'@(dateFormatting)'
,
generateDateFormattingLogic
(
message
))
.
replaceAll
(
'@(numberFormatting)'
,
generateNumberFormattingLogic
(
message
))
.
replaceAll
(
'@(innerMethodParameters)'
,
generatePluralMethodParameters
(
'Object'
).
join
(
', '
))
.
replaceAll
(
'@(innerMethodArgs)'
,
generateInnerMethodArgs
(
message
).
join
(
', '
))
.
replaceAll
(
'@(intlMethodArgs)'
,
intlMethodArgs
.
join
(
',
\n
'
));
}
bool
_isValidGetterAndMethodName
(
String
name
)
{
// Dart getter and method name cannot contain non-alphanumeric symbols
if
(
name
.
contains
(
RegExp
(
r'[^a-zA-Z\d]'
)))
return
false
;
// Dart class name must start with lower case character
if
(
name
[
0
].
contains
(
RegExp
(
r'[A-Z]'
)))
return
false
;
// Dart class name cannot start with a number
if
(
name
[
0
].
contains
(
RegExp
(
r'\d'
)))
return
false
;
return
true
;
return
pluralMethodTemplate
.
replaceAll
(
'@(methodName)'
,
message
.
resourceId
)
.
replaceAll
(
'@(methodParameters)'
,
generatePluralMethodParameters
().
join
(
', '
))
.
replaceAll
(
'@(dateFormatting)'
,
generateDateFormattingLogic
(
message
))
.
replaceAll
(
'@(numberFormatting)'
,
generateNumberFormattingLogic
(
message
))
.
replaceAll
(
'@(intlMethodArgs)'
,
intlMethodArgs
.
join
(
',
\n
'
));
}
/// The localizations generation class used to generate the localizations
...
...
@@ -699,6 +432,20 @@ class LocalizationsGenerator {
className
=
classNameString
;
}
static
bool
_isNotReadable
(
FileStat
fileStat
)
{
final
String
rawStatString
=
fileStat
.
modeString
();
// Removes potential prepended permission bits, such as '(suid)' and '(guid)'.
final
String
statString
=
rawStatString
.
substring
(
rawStatString
.
length
-
9
);
return
!(
statString
[
0
]
==
'r'
||
statString
[
3
]
==
'r'
||
statString
[
6
]
==
'r'
);
}
static
bool
_isNotWritable
(
FileStat
fileStat
)
{
final
String
rawStatString
=
fileStat
.
modeString
();
// Removes potential prepended permission bits, such as '(suid)' and '(guid)'.
final
String
statString
=
rawStatString
.
substring
(
rawStatString
.
length
-
9
);
return
!(
statString
[
1
]
==
'w'
||
statString
[
4
]
==
'w'
||
statString
[
7
]
==
'w'
);
}
/// Sets the reference [Directory] for [l10nDirectory].
@visibleForTesting
void
setL10nDirectory
(
String
arbPathString
)
{
...
...
@@ -744,15 +491,31 @@ class LocalizationsGenerator {
outputFile
=
_fs
.
file
(
path
.
join
(
l10nDirectory
.
path
,
outputFileString
));
}
static
bool
_isValidClassName
(
String
className
)
{
// Public Dart class name cannot begin with an underscore
if
(
className
[
0
]
==
'_'
)
return
false
;
// Dart class name cannot contain non-alphanumeric symbols
if
(
className
.
contains
(
RegExp
(
r'[^a-zA-Z_\d]'
)))
return
false
;
// Dart class name must start with upper case character
if
(
className
[
0
].
contains
(
RegExp
(
r'[a-z]'
)))
return
false
;
// Dart class name cannot start with a number
if
(
className
[
0
].
contains
(
RegExp
(
r'\d'
)))
return
false
;
return
true
;
}
/// Sets the [className] for the localizations and localizations delegate
/// classes.
@visibleForTesting
set
className
(
String
classNameString
)
{
if
(
classNameString
==
null
)
throw
L10nException
(
'classNameString argument cannot be null'
);
if
(
classNameString
==
null
||
classNameString
.
isEmpty
)
throw
L10nException
(
'classNameString argument cannot be null
or empty
'
);
if
(!
_isValidClassName
(
classNameString
))
throw
L10nException
(
"The 'output-class',
$classNameString
, is not a valid Dart class name.
\n
"
"The 'output-class',
$classNameString
, is not a valid
public
Dart class name.
\n
"
);
_className
=
classNameString
;
}
...
...
@@ -836,6 +599,42 @@ class LocalizationsGenerator {
supportedLocales
.
addAll
(
localeInfoList
);
}
static
bool
_isValidGetterAndMethodName
(
String
name
)
{
// Public Dart method name must not start with an underscore
if
(
name
[
0
]
==
'_'
)
return
false
;
// Dart getter and method name cannot contain non-alphanumeric symbols
if
(
name
.
contains
(
RegExp
(
r'[^a-zA-Z_\d]'
)))
return
false
;
// Dart method name must start with lower case character
if
(
name
[
0
].
contains
(
RegExp
(
r'[A-Z]'
)))
return
false
;
// Dart class name cannot start with a number
if
(
name
[
0
].
contains
(
RegExp
(
r'\d'
)))
return
false
;
return
true
;
}
static
String
_genSupportedLocaleProperty
(
Set
<
LocaleInfo
>
supportedLocales
)
{
const
String
prefix
=
'static const List<Locale> supportedLocales = <Locale>[
\n
Locale('
;
const
String
suffix
=
'),
\n
];'
;
String
resultingProperty
=
prefix
;
for
(
final
LocaleInfo
locale
in
supportedLocales
)
{
final
String
languageCode
=
locale
.
languageCode
;
final
String
countryCode
=
locale
.
countryCode
;
resultingProperty
+=
'
\'
$languageCode
\'
'
;
if
(
countryCode
!=
null
)
resultingProperty
+=
',
\'
$countryCode
\'
'
;
resultingProperty
+=
'),
\n
Locale('
;
}
resultingProperty
=
resultingProperty
.
substring
(
0
,
resultingProperty
.
length
-
'),
\n
Locale('
.
length
);
resultingProperty
+=
suffix
;
return
resultingProperty
;
}
/// Generates the methods for the localizations class.
///
/// The method parses [templateArbFile] and uses its resource ids as the
...
...
@@ -866,15 +665,18 @@ class LocalizationsGenerator {
for
(
final
String
key
in
sortedArbKeys
)
{
if
(
key
.
startsWith
(
'@'
))
continue
;
if
(!
_isValidGetterAndMethodName
(
key
))
if
(!
_isValidGetterAndMethodName
(
key
))
{
throw
L10nException
(
'Invalid key format:
$key
\n
It has to be in camel case, cannot start '
'with a number, and cannot contain non-alphanumeric characters.'
'with a number
or underscore
, and cannot contain non-alphanumeric characters.'
);
if
(
pluralValueRE
.
hasMatch
(
bundle
[
key
]
as
String
))
classMethods
.
add
(
genPluralMethod
(
bundle
,
key
));
}
final
Message
message
=
Message
(
bundle
,
key
);
if
(
pluralValueRE
.
hasMatch
(
message
.
value
))
classMethods
.
add
(
generatePluralMethod
(
message
));
else
classMethods
.
add
(
genSimpleMethod
(
bundle
,
key
));
classMethods
.
add
(
genSimpleMethod
(
message
));
}
}
...
...
@@ -885,17 +687,11 @@ class LocalizationsGenerator {
final
String
outputFileName
=
path
.
basename
(
outputFile
.
path
);
outputFile
.
writeAsStringSync
(
defaultFileTemplate
.
replaceAll
(
'@
className
'
,
className
)
.
replaceAll
(
'@
classMethods
'
,
classMethods
.
join
(
'
\n
'
))
.
replaceAll
(
'@
importFile
'
,
'
$directory
/
$outputFileName
'
)
.
replaceAll
(
'@
supportedLocales'
,
genSupportedLocaleProperty
(
supportedLocales
))
.
replaceAll
(
'@
supportedLanguageCodes
'
,
supportedLanguageCodes
.
toList
().
join
(
', '
))
.
replaceAll
(
'@
(className)
'
,
className
)
.
replaceAll
(
'@
(classMethods)
'
,
classMethods
.
join
(
'
\n
'
))
.
replaceAll
(
'@
(importFile)
'
,
'
$directory
/
$outputFileName
'
)
.
replaceAll
(
'@
(supportedLocales)'
,
_
genSupportedLocaleProperty
(
supportedLocales
))
.
replaceAll
(
'@
(supportedLanguageCodes)
'
,
supportedLanguageCodes
.
toList
().
join
(
', '
))
);
}
}
class
L10nException
implements
Exception
{
L10nException
(
this
.
message
);
final
String
message
;
}
dev/tools/localization/gen_l10n_templates.dart
0 → 100644
View file @
7539826d
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
const
String
getterMethodTemplate
=
'''
String get @(methodName) {
return Intl.message(
@(message),
locale: _localeName,
@(intlMethodArgs)
);
}
'''
;
const
String
simpleMethodTemplate
=
'''
String @(methodName)(@(methodParameters)) {
return Intl.message(
@(message),
locale: _localeName,
@(intlMethodArgs)
);
}
'''
;
const
String
formatMethodTemplate
=
'''
String @(methodName)(@(methodParameters)) {@(dateFormatting)@(numberFormatting)
String @(methodName)(@(innerMethodParameters)) {
return Intl.message(
@(message),
locale: _localeName,
@(intlMethodArgs)
);
}
return @(methodName)(@(innerMethodArgs));
}
'''
;
const
String
pluralMethodTemplate
=
'''
String @(methodName)(@(methodParameters)) {@(dateFormatting)@(numberFormatting)
return Intl.plural(
@(intlMethodArgs)
);
}
'''
;
const
String
pluralFormatMethodTemplate
=
'''
String @(methodName)(@(methodParameters)) {@(dateFormatting)@(numberFormatting)
String @(methodName)(@(innerMethodParameters)) {
return Intl.plural(
@(intlMethodArgs)
);
}
return @(methodName)(@(innerMethodArgs));
}
'''
;
const
String
defaultFileTemplate
=
'''
import '
dart:
async
';
import '
package:
flutter
/
widgets
.
dart
';
import '
package:
flutter_localizations
/
flutter_localizations
.
dart
';
import '
package:
intl
/
intl
.
dart
';
import '
messages_all
.
dart
';
/// Callers can lookup localized strings with an instance of @(className) returned
/// by `@(className).of(context)`.
///
/// Applications need to include `@(className).delegate()` in their app
\'
s
/// localizationDelegates list, and the locales they support in the app
\'
s
/// supportedLocales list. For example:
///
/// ```
/// import '
@
(
importFile
)
';
///
/// return MaterialApp(
/// localizationsDelegates: @(className).localizationsDelegates,
/// supportedLocales: @(className).supportedLocales,
/// home: MyApplicationHome(),
/// );
/// ```
///
/// ## Update pubspec.yaml
///
/// Please make sure to update your pubspec.yaml to include the following
/// packages:
///
/// ```
/// dependencies:
/// # Internationalization support.
/// flutter_localizations:
/// sdk: flutter
/// intl: 0.16.0
/// intl_translation: 0.17.7
///
/// # rest of dependencies
/// ```
///
/// ## iOS Applications
///
/// iOS applications define key application metadata, including supported
/// locales, in an Info.plist file that is built into the application bundle.
/// To configure the locales supported by your app, you’ll need to edit this
/// file.
///
/// First, open your project’s ios/Runner.xcworkspace Xcode workspace file.
/// Then, in the Project Navigator, open the Info.plist file under the Runner
/// project’s Runner folder.
///
/// Next, select the Information Property List item, select Add Item from the
/// Editor menu, then select Localizations from the pop-up menu.
///
/// Select and expand the newly-created Localizations item then, for each
/// locale your application supports, add a new item and select the locale
/// you wish to add from the pop-up menu in the Value field. This list should
/// be consistent with the languages listed in the @className.supportedLocales
/// property.
class @(className) {
@(className)(Locale locale) : _localeName = Intl.canonicalizedLocale(locale.toString());
final String _localeName;
static Future<@(className)> load(Locale locale) {
return initializeMessages(locale.toString())
.then<@(className)>((_) => @(className)(locale));
}
static @(className) of(BuildContext context) {
return Localizations.of<@(className)>(context, @(className));
}
static const LocalizationsDelegate<@(className)> delegate = _@(classNameDelegate)();
/// A list of this localizations delegate along with the default localizations
/// delegates.
///
/// Returns a list of localizations delegates containing this delegate along with
/// GlobalMaterialLocalizations.delegate, GlobalCupertinoLocalizations.delegate,
/// and GlobalWidgetsLocalizations.delegate.
///
/// Additional delegates can be added by appending to this list in
/// MaterialApp. This list does not have to be used at all if a custom list
/// of delegates is preferred or required.
static const List<LocalizationsDelegate<dynamic>> localizationsDelegates = <LocalizationsDelegate<dynamic>>[
delegate,
GlobalMaterialLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
];
/// A list of this localizations delegate'
s
supported
locales
.
@
(
supportedLocales
)
@
(
classMethods
)
}
class
_
@
(
className
)
Delegate
extends
LocalizationsDelegate
<
@
(
className
)>
{
const
_
@
(
className
)
Delegate
();
@override
Future
<
@
(
className
)>
load
(
Locale
locale
)
=>
@
(
className
).
load
(
locale
);
@override
bool
isSupported
(
Locale
locale
)
=>
<
String
>[
@
(
supportedLanguageCodes
)].
contains
(
locale
.
languageCode
);
@override
bool
shouldReload
(
_
@
(
className
)
Delegate
old
)
=>
false
;
}
''';
dev/tools/localization/gen_l10n_types.dart
0 → 100644
View file @
7539826d
// Copyright 2014 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
class
L10nException
implements
Exception
{
L10nException
(
this
.
message
);
final
String
message
;
}
class
OptionalParameter
{
const
OptionalParameter
(
this
.
name
,
this
.
value
)
:
assert
(
name
!=
null
),
assert
(
value
!=
null
);
final
String
name
;
final
Object
value
;
}
class
Placeholder
{
Placeholder
(
this
.
resourceId
,
this
.
name
,
Map
<
String
,
dynamic
>
attributes
)
:
assert
(
resourceId
!=
null
),
assert
(
name
!=
null
),
example
=
_stringAttribute
(
resourceId
,
name
,
attributes
,
'example'
),
type
=
_stringAttribute
(
resourceId
,
name
,
attributes
,
'type'
)
??
'Object'
,
format
=
_stringAttribute
(
resourceId
,
name
,
attributes
,
'format'
),
optionalParameters
=
_optionalParameters
(
resourceId
,
name
,
attributes
);
final
String
resourceId
;
final
String
name
;
final
String
example
;
final
String
type
;
final
String
format
;
final
List
<
OptionalParameter
>
optionalParameters
;
bool
get
requiresFormatting
=>
<
String
>[
'DateTime'
,
'double'
,
'int'
,
'num'
].
contains
(
type
);
bool
get
isNumber
=>
<
String
>[
'double'
,
'int'
,
'num'
].
contains
(
type
);
bool
get
isDate
=>
'DateTime'
==
type
;
static
String
_stringAttribute
(
String
resourceId
,
String
name
,
Map
<
String
,
dynamic
>
attributes
,
String
attributeName
,
)
{
final
dynamic
value
=
attributes
[
attributeName
];
if
(
value
==
null
)
return
null
;
if
(
value
is
!
String
||
(
value
as
String
).
isEmpty
)
{
throw
L10nException
(
'The "
$attributeName
" value of the "
$name
" placeholder in message
$resourceId
'
'must be a non-empty string.'
,
);
}
return
value
as
String
;
}
static
List
<
OptionalParameter
>
_optionalParameters
(
String
resourceId
,
String
name
,
Map
<
String
,
dynamic
>
attributes
)
{
final
dynamic
value
=
attributes
[
'optionalParameters'
];
if
(
value
==
null
)
return
<
OptionalParameter
>[];
if
(
value
is
!
Map
<
String
,
Object
>)
{
throw
L10nException
(
'The "optionalParameters" value of the "
$name
" placeholder in message '
'
$resourceId
is not a properly formatted Map. Ensure that it is a map '
'with keys that are strings.'
);
}
final
Map
<
String
,
dynamic
>
optionalParameterMap
=
value
as
Map
<
String
,
dynamic
>;
return
optionalParameterMap
.
keys
.
map
<
OptionalParameter
>((
String
parameterName
)
{
return
OptionalParameter
(
parameterName
,
optionalParameterMap
[
parameterName
]);
}).
toList
();
}
}
class
Message
{
Message
(
Map
<
String
,
dynamic
>
bundle
,
this
.
resourceId
)
:
assert
(
bundle
!=
null
),
assert
(
resourceId
!=
null
&&
resourceId
.
isNotEmpty
),
value
=
_value
(
bundle
,
resourceId
),
description
=
_description
(
bundle
,
resourceId
),
placeholders
=
_placeholders
(
bundle
,
resourceId
);
final
String
resourceId
;
final
String
value
;
final
String
description
;
final
List
<
Placeholder
>
placeholders
;
bool
get
placeholdersRequireFormatting
=>
placeholders
.
any
((
Placeholder
p
)
=>
p
.
requiresFormatting
);
static
String
_value
(
Map
<
String
,
dynamic
>
bundle
,
String
resourceId
)
{
final
dynamic
value
=
bundle
[
resourceId
];
if
(
value
==
null
)
throw
L10nException
(
'A value for resource "
$resourceId
" was not found.'
);
if
(
value
is
!
String
)
throw
L10nException
(
'The value of "
$resourceId
" is not a string.'
);
return
bundle
[
resourceId
]
as
String
;
}
static
Map
<
String
,
dynamic
>
_attributes
(
Map
<
String
,
dynamic
>
bundle
,
String
resourceId
)
{
final
dynamic
attributes
=
bundle
[
'@
$resourceId
'
];
if
(
attributes
==
null
)
{
throw
L10nException
(
'Resource attribute "@
$resourceId
" was not found. Please '
'ensure that each resource has a corresponding @resource.'
);
}
if
(
attributes
is
!
Map
<
String
,
dynamic
>)
{
throw
L10nException
(
'The resource attribute "@
$resourceId
" is not a properly formatted Map. '
'Ensure that it is a map with keys that are strings.'
);
}
return
attributes
as
Map
<
String
,
dynamic
>;
}
static
String
_description
(
Map
<
String
,
dynamic
>
bundle
,
String
resourceId
)
{
final
dynamic
value
=
_attributes
(
bundle
,
resourceId
)[
'description'
];
if
(
value
==
null
)
return
null
;
if
(
value
is
!
String
)
{
throw
L10nException
(
'The description for "@
$resourceId
" is not a properly formatted String.'
);
}
return
value
as
String
;
}
static
List
<
Placeholder
>
_placeholders
(
Map
<
String
,
dynamic
>
bundle
,
String
resourceId
)
{
final
dynamic
value
=
_attributes
(
bundle
,
resourceId
)[
'placeholders'
];
if
(
value
==
null
)
return
<
Placeholder
>[];
if
(
value
is
!
Map
<
String
,
dynamic
>)
{
throw
L10nException
(
'The "placeholders" attribute for message
$resourceId
, is not '
'properly formatted. Ensure that it is a map with string valued keys.'
);
}
final
Map
<
String
,
dynamic
>
allPlaceholdersMap
=
value
as
Map
<
String
,
dynamic
>;
return
allPlaceholdersMap
.
keys
.
map
<
Placeholder
>((
String
placeholderName
)
{
final
dynamic
value
=
allPlaceholdersMap
[
placeholderName
];
if
(
value
is
!
Map
<
String
,
dynamic
>)
{
throw
L10nException
(
'The value of the "
$placeholderName
" placeholder attribute for message '
'"
$resourceId
", is not properly formatted. Ensure that it is a map '
'with string valued keys.'
);
}
return
Placeholder
(
resourceId
,
placeholderName
,
value
as
Map
<
String
,
dynamic
>);
}).
toList
();
}
}
dev/tools/test/localization/gen_l10n_test.dart
View file @
7539826d
...
...
@@ -9,6 +9,7 @@ import 'package:file/memory.dart';
import
'package:path/path.dart'
as
path
;
import
'../../localization/gen_l10n.dart'
;
import
'../../localization/gen_l10n_types.dart'
;
import
'../../localization/localizations_utils.dart'
;
import
'../common.dart'
;
...
...
@@ -192,7 +193,7 @@ void main() {
try
{
generator
.
className
=
'String with spaces'
;
}
on
L10nException
catch
(
e
)
{
expect
(
e
.
message
,
contains
(
'is not a valid Dart class name'
));
expect
(
e
.
message
,
contains
(
'is not a valid
public
Dart class name'
));
return
;
}
fail
(
...
...
@@ -205,12 +206,12 @@ void main() {
try
{
generator
.
className
=
'TestClass@123'
;
}
on
L10nException
catch
(
e
)
{
expect
(
e
.
message
,
contains
(
'is not a valid Dart class name'
));
expect
(
e
.
message
,
contains
(
'is not a valid
public
Dart class name'
));
return
;
}
fail
(
'Attempting to set LocalizationsGenerator.className should fail if the '
'the input string is not a valid Dart class name.'
'the input string is not a valid
public
Dart class name.'
);
});
...
...
@@ -218,12 +219,12 @@ void main() {
try
{
generator
.
className
=
'camelCaseClassName'
;
}
on
L10nException
catch
(
e
)
{
expect
(
e
.
message
,
contains
(
'is not a valid Dart class name'
));
expect
(
e
.
message
,
contains
(
'is not a valid
public
Dart class name'
));
return
;
}
fail
(
'Attempting to set LocalizationsGenerator.className should fail if the '
'the input string is not a valid Dart class name.'
'the input string is not a valid
public
Dart class name.'
);
});
...
...
@@ -231,12 +232,12 @@ void main() {
try
{
generator
.
className
=
'123ClassName'
;
}
on
L10nException
catch
(
e
)
{
expect
(
e
.
message
,
contains
(
'is not a valid Dart class name'
));
expect
(
e
.
message
,
contains
(
'is not a valid
public
Dart class name'
));
return
;
}
fail
(
'Attempting to set LocalizationsGenerator.className should fail if the '
'the input string is not a valid Dart class name.'
'the input string is not a valid
public
Dart class name.'
);
});
});
...
...
@@ -665,6 +666,7 @@ void main() {
}
'''
);
});
});
group
(
'DateTime tests'
,
()
{
test
(
'correctly generates simple message with dates'
,
()
{
...
...
@@ -702,17 +704,20 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
''' String springBegins(
Object
springStartDate) {
''' String springBegins(
DateTime
springStartDate) {
final DateFormat springStartDateDateFormat = DateFormat.yMMMMEEEEd(_localeName);
final String springStartDateString = springStartDateDateFormat.format(springStartDate);
return Intl.message(
'
Spring
begins
on
\
$springStartDateString
',
locale: _localeName,
name: '
springBegins
',
desc: r'
The
first
day
of
spring
',
args: <Object>[springStartDateString]
);
String springBegins(Object springStartDate) {
return Intl.message(
\'
Spring begins on
\
$springStartDate
\'
,
locale: _localeName,
name:
\'
springBegins
\'
,
desc: r
\'
The first day of spring
\'
,
args: <Object>[springStartDate]
);
}
return springBegins(springStartDateString);
}
'''
);
});
...
...
@@ -826,17 +831,20 @@ void main() {
expect(generator.classMethods, isNotEmpty);
expect(
generator.classMethods.first,
''' String springGreetings(
Object
springStartDate, Object helloWorld) {
''' String springGreetings(
DateTime
springStartDate, Object helloWorld) {
final DateFormat springStartDateDateFormat = DateFormat.yMMMMEEEEd(_localeName);
final String springStartDateString = springStartDateDateFormat.format(springStartDate);
return Intl.message(
\'
Since it
\'
"
\
'" r
\'
s
\
$springStartDateString
, it
\'
"
\'
" r
\'
s finally spring!
\
$helloWorld
!
\'
,
locale: _localeName,
name: '
springGreetings
',
desc: r
\'
A realization that it
\'
"
\'
" r
\'
s finally the spring season, followed by a greeting.
\'
,
args: <Object>[springStartDateString, helloWorld]
);
String springGreetings(Object springStartDate, Object helloWorld) {
return Intl.message(
\'
Since it
\'
"
\
'" r
\'
s
\
$springStartDate
, it
\'
"
\'
" r
\'
s finally spring!
\
$helloWorld
!
\'
,
locale: _localeName,
name:
\'
springGreetings
\'
,
desc: r
\'
A realization that it
\'
"
\'
" r
\'
s finally the spring season, followed by a greeting.
\'
,
args: <Object>[springStartDate, helloWorld]
);
}
return springGreetings(springStartDateString, helloWorld);
}
'''
);
});
...
...
@@ -880,20 +888,23 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
''' String springRange(
Object springStartDate, Object
springEndDate) {
''' String springRange(
DateTime springStartDate, DateTime
springEndDate) {
final DateFormat springStartDateDateFormat = DateFormat.yMMMMEEEEd(_localeName);
final String springStartDateString = springStartDateDateFormat.format(springStartDate);
final DateFormat springEndDateDateFormat = DateFormat.yMMMMEEEEd(_localeName);
final String springEndDateString = springEndDateDateFormat.format(springEndDate);
return Intl.message(
\'
Spring begins on
\
$springStartDateString
and ends on
\
$springEndDateString
\'
,
locale: _localeName,
name: '
springRange
',
desc: r
\'
The range of dates for spring in the year
\'
,
args: <Object>[springStartDateString, springEndDateString]
);
String springRange(Object springStartDate, Object springEndDate) {
return Intl.message(
\'
Spring begins on
\
$springStartDate
and ends on
\
$springEndDate
\'
,
locale: _localeName,
name:
\'
springRange
\'
,
desc: r
\'
The range of dates for spring in the year
\'
,
args: <Object>[springStartDate, springEndDate]
);
}
return springRange(springStartDateString, springEndDateString);
}
'''
);
});
...
...
@@ -933,24 +944,25 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
''' String helloWorlds(int count,
Object
currentDate) {
''' String helloWorlds(int count,
DateTime
currentDate) {
final DateFormat currentDateDateFormat = DateFormat.yMMMMEEEEd(_localeName);
final String currentDateString = currentDateDateFormat.format(currentDate);
return Intl.plural(
count,
String helloWorlds(int count, Object currentDate) {
return Intl.plural(
count,
locale: _localeName,
name: '
helloWorlds
',
args: <Object>[count, currentDateString],
one: '
Hello
World
,
today
is
\
$currentDateString
',
two: '
Hello
two
worlds
,
today
is
\
$currentDateString
',
many: '
Hello
all
\
$count
worlds
,
today
is
\
$currentDateString
',
other: '
Hello
other
\
$count
worlds
,
today
is
\
$currentDateString
'
);
name:
\'
helloWorlds
\'
,
args: <Object>[count, currentDate],
one:
\'
Hello World, today is
\
$currentDateString
\'
,
two:
\'
Hello two worlds, today is
\
$currentDateString
\'
,
many:
\'
Hello all
\
$count
worlds, today is
\
$currentDateString
\'
,
other:
\'
Hello other
\
$count
worlds, today is
\
$currentDateString
\'
);
}
return helloWorlds(count, currentDateString);
}
'''
);
});
'''
);
});
group
(
'Number tests'
,
()
{
...
...
@@ -961,7 +973,7 @@ void main() {
"description": "The amount of progress the student has made in their class.",
"placeholders": {
"progress": {
"type": "
Number
",
"type": "
double
",
"format": "compact"
}
}
...
...
@@ -989,19 +1001,22 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
''' String courseCompletion(
Object
progress) {
''' String courseCompletion(
double
progress) {
final NumberFormat progressNumberFormat = NumberFormat.compact(
locale: _localeName,
);
final String progressString = progressNumberFormat.format(progress);
return Intl.message(
'
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
]
);
String courseCompletion(Object progress) {
return Intl.message(
\'
You have completed
\
$progress
of the course.
\'
,
locale: _localeName,
name:
\'
courseCompletion
\'
,
desc: r
\'
The amount of progress the student has made in their class.
\'
,
args: <Object>[progress]
);
}
return courseCompletion(progressString);
}
'''
);
});
...
...
@@ -1024,7 +1039,7 @@ void main() {
"description": "The amount of progress the student has made in their class.",
"placeholders": {
"progress": {
"type"
:
"
Number
"
,
"type": "
double
",
"format": "
$numberFormat
",
"optionalParameters": {
"decimalDigits": 2
...
...
@@ -1055,23 +1070,25 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
'''
String
courseCompletion
(
Object
progress
)
{
''' String courseCompletion(
double
progress) {
final NumberFormat progressNumberFormat = NumberFormat.
$numberFormat
(
locale: _localeName,
decimalDigits: 2,
);
final String progressString = progressNumberFormat.format(progress);
return
Intl
.
message
(
'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
]
);
String courseCompletion(Object progress) {
return Intl.message(
\'
You have completed
\
$progress
of the course.
\'
,
locale: _localeName,
name:
\'
courseCompletion
\'
,
desc: r
\'
The amount of progress the student has made in their class.
\'
,
args: <Object>[progress]
);
}
return courseCompletion(progressString);
}
''');
}
'''
);}
});
test
(
'correctly adds optional positional parameters to numbers'
,
()
{
...
...
@@ -1088,7 +1105,7 @@ void main() {
"description": "The amount of progress the student has made in their class.",
"placeholders": {
"progress": {
"type"
:
"
Number
"
,
"type": "
double
",
"format": "
$numberFormat
"
}
}
...
...
@@ -1116,17 +1133,20 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
'''
String
courseCompletion
(
Object
progress
)
{
''' String courseCompletion(
double
progress) {
final NumberFormat progressNumberFormat = NumberFormat.
$numberFormat
(_localeName);
final String progressString = progressNumberFormat.format(progress);
return
Intl
.
message
(
'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
]
);
String courseCompletion(Object progress) {
return Intl.message(
\'
You have completed
\
$progress
of the course.
\'
,
locale: _localeName,
name:
\'
courseCompletion
\'
,
desc: r
\'
The amount of progress the student has made in their class.
\'
,
args: <Object>[progress]
);
}
return courseCompletion(progressString);
}
'''
);
}
...
...
@@ -1139,7 +1159,7 @@ void main() {
"description": "The amount of progress the student has made in their class.",
"placeholders": {
"progress": {
"type"
:
"
Number
"
,
"type": "
double
",
"format": "asdf"
}
}
...
...
@@ -1306,20 +1326,23 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
'''
String
helloWorlds
(
int
count
,
Object
currentDate
)
{
''' String helloWorlds(int count,
DateTime
currentDate) {
final DateFormat currentDateDateFormat = DateFormat.yMMMMEEEEd(_localeName);
final String currentDateString = currentDateDateFormat.format(currentDate);
return
Intl
.
plural
(
count
,
String helloWorlds(int count, Object currentDate) {
return Intl.plural(
count,
locale: _localeName,
name:
'helloWorlds'
,
args:
<
Object
>[
count
,
currentDateString
],
one:
'Hello World, today is
\
$currentDateString
'
,
two:
'Hello two worlds, today is
\
$currentDateString
'
,
many:
'Hello all
\
$count
worlds, today is
\
$currentDateString
'
,
other:
'Hello other
\
$count
worlds, today is
\
$currentDateString
'
);
name:
\'
helloWorlds
\'
,
args: <Object>[count, currentDate],
one:
\'
Hello World, today is
\
$currentDateString
\'
,
two:
\'
Hello two worlds, today is
\
$currentDateString
\'
,
many:
\'
Hello all
\
$count
worlds, today is
\
$currentDateString
\'
,
other:
\'
Hello other
\
$count
worlds, today is
\
$currentDateString
\'
);
}
return helloWorlds(count, currentDateString);
}
'''
);
...
...
@@ -1332,7 +1355,7 @@ void main() {
"placeholders": {
"count": {},
"population": {
"type"
:
"
Number
"
,
"type": "
int
",
"format": "compactLong"
}
}
...
...
@@ -1360,22 +1383,25 @@ void main() {
expect
(
generator
.
classMethods
,
isNotEmpty
);
expect
(
generator
.
classMethods
.
first
,
'''
String
helloWorlds
(
int
count
,
Objec
t
population
)
{
''' String helloWorlds(int count,
in
t population) {
final NumberFormat populationNumberFormat = NumberFormat.compactLong(
locale: _localeName,
);
final String populationString = populationNumberFormat.format(population);
return
Intl
.
plural
(
count
,
String helloWorlds(int count, Object population) {
return Intl.plural(
count,
locale: _localeName,
name:
'helloWorlds'
,
args:
<
Object
>[
count
,
populationString
],
one:
'Hello World of
\
$populationString
citizens'
,
two:
'Hello two worlds with
\
$populationString
total citizens'
,
many:
'Hello all
\
$count
worlds, with a total of
\
$populationString
citizens'
,
other:
'Hello other
\
$count
worlds, with a total of
\
$populationString
citizens'
);
name:
\'
helloWorlds
\'
,
args: <Object>[count, population],
one:
\'
Hello World of
\
$populationString
citizens
\'
,
two:
\'
Hello two worlds with
\
$populationString
total citizens
\'
,
many:
\'
Hello all
\
$count
worlds, with a total of
\
$populationString
citizens
\'
,
other:
\'
Hello other
\
$count
worlds, with a total of
\
$populationString
citizens
\'
);
}
return helloWorlds(count, populationString);
}
'''
);
...
...
@@ -1463,8 +1489,7 @@ void main() {
generator
.
parseArbFiles
();
generator
.
generateClassMethods
();
}
on
L10nException
catch
(
e
)
{
expect(e.message, contains('
Resource
attribute
'));
expect(e.message, contains('
does
not
exist
'));
expect
(
e
.
message
,
contains
(
'Resource attribute "@helloWorlds" was not found'
));
return
;
}
fail
(
'Generating plural class method without resource attributes should not succeed'
);
...
...
@@ -1495,7 +1520,7 @@ void main() {
generator
.
generateClassMethods
();
}
on
L10nException
catch
(
e
)
{
expect
(
e
.
message
,
contains
(
'is not properly formatted'
));
expect(e.message, contains('
Ensure
that
it
is
a
map
with
keys
that
are
string
s
'));
expect
(
e
.
message
,
contains
(
'Ensure that it is a map with
string valued key
s'
));
return
;
}
fail
(
'Generating class methods with incorrect placeholder format should not succeed'
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment