// 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. import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_localizations/flutter_localizations.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:intl/intl.dart' as intl; class FooMaterialLocalizations extends MaterialLocalizationEn { FooMaterialLocalizations( Locale localeName, this.backButtonTooltip, ) : super( localeName: localeName.toString(), fullYearFormat: intl.DateFormat.y(), compactDateFormat: intl.DateFormat.yMd(), shortDateFormat: intl.DateFormat.yMMMd(), mediumDateFormat: intl.DateFormat('E, MMM\u00a0d'), longDateFormat: intl.DateFormat.yMMMMEEEEd(), yearMonthFormat: intl.DateFormat.yMMMM(), shortMonthDayFormat: intl.DateFormat.MMMd(), decimalFormat: intl.NumberFormat.decimalPattern(), twoDigitZeroPaddedFormat: intl.NumberFormat('00'), ); @override final String backButtonTooltip; } class FooMaterialLocalizationsDelegate extends LocalizationsDelegate<MaterialLocalizations> { const FooMaterialLocalizationsDelegate({ this.supportedLanguage = 'en', this.backButtonTooltip = 'foo', }); final String supportedLanguage; final String backButtonTooltip; @override bool isSupported(Locale locale) { return supportedLanguage == 'allLanguages' || locale.languageCode == supportedLanguage; } @override Future<FooMaterialLocalizations> load(Locale locale) { return SynchronousFuture<FooMaterialLocalizations>(FooMaterialLocalizations(locale, backButtonTooltip)); } @override bool shouldReload(FooMaterialLocalizationsDelegate old) => false; } Widget buildFrame({ Locale? locale, Iterable<LocalizationsDelegate<dynamic>> delegates = GlobalMaterialLocalizations.delegates, required WidgetBuilder buildContent, LocaleResolutionCallback? localeResolutionCallback, Iterable<Locale> supportedLocales = const <Locale>[ Locale('en', 'US'), Locale('es', 'ES'), ], }) { return MaterialApp( color: const Color(0xFFFFFFFF), locale: locale, supportedLocales: supportedLocales, localizationsDelegates: delegates, localeResolutionCallback: localeResolutionCallback, onGenerateRoute: (RouteSettings settings) { return MaterialPageRoute<void>( builder: (BuildContext context) { return buildContent(context); } ); }, ); } void main() { testWidgets('Locale fallbacks', (WidgetTester tester) async { final Key textKey = UniqueKey(); await tester.pumpWidget( buildFrame( buildContent: (BuildContext context) { return Text( MaterialLocalizations.of(context).backButtonTooltip, key: textKey, ); } ) ); expect(tester.widget<Text>(find.byKey(textKey)).data, 'Back'); // Unrecognized locale falls back to 'en' await tester.binding.setLocale('foo', 'BAR'); await tester.pump(); expect(tester.widget<Text>(find.byKey(textKey)).data, 'Back'); // Spanish Bolivia locale, falls back to just 'es' await tester.binding.setLocale('es', 'BO'); await tester.pump(); expect(tester.widget<Text>(find.byKey(textKey)).data, 'Atrás'); }); testWidgets("Localizations.override widget tracks parent's locale", (WidgetTester tester) async { Widget buildLocaleFrame(Locale locale) { return buildFrame( locale: locale, supportedLocales: <Locale>[locale], buildContent: (BuildContext context) { return Localizations.override( context: context, child: Builder( builder: (BuildContext context) { // No MaterialLocalizations are defined for the first Localizations // ancestor, so we should get the values from the default one, i.e. // the one created by WidgetsApp via the LocalizationsDelegate // provided by MaterialApp. return Text(MaterialLocalizations.of(context).backButtonTooltip); }, ), ); }, ); } await tester.pumpWidget(buildLocaleFrame(const Locale('en', 'US'))); expect(find.text('Back'), findsOneWidget); await tester.pumpWidget(buildLocaleFrame(const Locale('de', 'DE'))); expect(find.text('Zurück'), findsOneWidget); await tester.pumpWidget(buildLocaleFrame(const Locale('zh', 'CN'))); expect(find.text('返回'), findsOneWidget); }); testWidgets('Localizations.override widget with hardwired locale', (WidgetTester tester) async { Widget buildLocaleFrame(Locale locale) { return buildFrame( locale: locale, buildContent: (BuildContext context) { return Localizations.override( context: context, locale: const Locale('en', 'US'), child: Builder( builder: (BuildContext context) { // No MaterialLocalizations are defined for the Localizations.override // ancestor, so we should get all values from the default one, i.e. // the one created by WidgetsApp via the LocalizationsDelegate // provided by MaterialApp. return Text(MaterialLocalizations.of(context).backButtonTooltip); }, ), ); }, ); } await tester.pumpWidget(buildLocaleFrame(const Locale('en', 'US'))); expect(find.text('Back'), findsOneWidget); await tester.pumpWidget(buildLocaleFrame(const Locale('de', 'DE'))); expect(find.text('Back'), findsOneWidget); await tester.pumpWidget(buildLocaleFrame(const Locale('zh', 'CN'))); expect(find.text('Back'), findsOneWidget); }); testWidgets('MaterialApp adds MaterialLocalizations for additional languages', (WidgetTester tester) async { final Key textKey = UniqueKey(); await tester.pumpWidget( buildFrame( delegates: <LocalizationsDelegate<dynamic>>[ const FooMaterialLocalizationsDelegate(supportedLanguage: 'fr', backButtonTooltip: 'FR'), const FooMaterialLocalizationsDelegate(supportedLanguage: 'de', backButtonTooltip: 'DE'), GlobalCupertinoLocalizations.delegate, ], supportedLocales: const <Locale>[ Locale('en'), Locale('fr'), Locale('de'), ], buildContent: (BuildContext context) { return Text( MaterialLocalizations.of(context).backButtonTooltip, key: textKey, ); }, ) ); expect(tester.widget<Text>(find.byKey(textKey)).data, 'Back'); await tester.binding.setLocale('fr', 'CA'); await tester.pump(); expect(find.text('FR'), findsOneWidget); await tester.binding.setLocale('de', 'DE'); await tester.pump(); expect(find.text('DE'), findsOneWidget); }); testWidgets('MaterialApp overrides MaterialLocalizations for all locales', (WidgetTester tester) async { final Key textKey = UniqueKey(); await tester.pumpWidget( buildFrame( // Accept whatever locale we're given localeResolutionCallback: (Locale? locale, Iterable<Locale> supportedLocales) => locale, delegates: <LocalizationsDelegate<dynamic>>[ const FooMaterialLocalizationsDelegate(supportedLanguage: 'allLanguages'), GlobalCupertinoLocalizations.delegate, ], buildContent: (BuildContext context) { // Should always be 'foo', no matter what the locale is return Text( MaterialLocalizations.of(context).backButtonTooltip, key: textKey, ); }, ) ); expect(tester.widget<Text>(find.byKey(textKey)).data, 'foo'); await tester.binding.setLocale('zh', 'CN'); await tester.pump(); expect(find.text('foo'), findsOneWidget); await tester.binding.setLocale('de', 'DE'); await tester.pump(); expect(find.text('foo'), findsOneWidget); }); testWidgets('MaterialApp overrides MaterialLocalizations for default locale', (WidgetTester tester) async { final Key textKey = UniqueKey(); await tester.pumpWidget( buildFrame( delegates: <LocalizationsDelegate<dynamic>>[ const FooMaterialLocalizationsDelegate(), GlobalCupertinoLocalizations.delegate, ], // supportedLocales not specified, so all locales resolve to 'en' buildContent: (BuildContext context) { return Text( MaterialLocalizations.of(context).backButtonTooltip, key: textKey, ); }, ) ); // Unsupported locale '_' (the widget tester's default) resolves to 'en'. expect(tester.widget<Text>(find.byKey(textKey)).data, 'foo'); // Unsupported locale 'zh' resolves to 'en'. await tester.binding.setLocale('zh', 'CN'); await tester.pump(); expect(find.text('foo'), findsOneWidget); // Unsupported locale 'de' resolves to 'en'. await tester.binding.setLocale('de', 'DE'); await tester.pump(); expect(find.text('foo'), findsOneWidget); }); testWidgets('deprecated Android/Java locales are modernized', (WidgetTester tester) async { final Key textKey = UniqueKey(); await tester.pumpWidget( buildFrame( supportedLocales: <Locale>[ const Locale('en', 'US'), const Locale('he', 'IL'), const Locale('yi', 'IL'), const Locale('id', 'JV'), ], buildContent: (BuildContext context) { return Text( '${Localizations.localeOf(context)}', key: textKey, ); }, ) ); expect(tester.widget<Text>(find.byKey(textKey)).data, 'en_US'); // Hebrew was iw (ISO-639) is he (ISO-639-1) await tester.binding.setLocale('iw', 'IL'); await tester.pump(); expect(tester.widget<Text>(find.byKey(textKey)).data, 'he_IL'); // Yiddish was ji (ISO-639) is yi (ISO-639-1) await tester.binding.setLocale('ji', 'IL'); await tester.pump(); expect(tester.takeException(), "Warning: This application's locale, yi_IL, is not supported by all of its localization delegates."); expect(tester.widget<Text>(find.byKey(textKey)).data, 'yi_IL'); // Indonesian was in (ISO-639) is id (ISO-639-1) await tester.binding.setLocale('in', 'JV'); await tester.pump(); expect(tester.widget<Text>(find.byKey(textKey)).data, 'id_JV'); }); }