localizations.dart 22.1 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4 5
// 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';
6
import 'package:flutter/rendering.dart';
7

8
import 'basic.dart';
9
import 'container.dart';
10
import 'debug.dart';
11 12 13
import 'framework.dart';

// Examples can assume:
14 15
// class Intl { static String message(String s, { String? name, String? locale }) => ''; }
// Future<void> initializeMessages(String locale) => Future.value();
16

17 18 19 20 21 22 23 24
// Used by loadAll() to record LocalizationsDelegate.load() futures we're
// waiting for.
class _Pending {
  _Pending(this.delegate, this.futureValue);
  final LocalizationsDelegate<dynamic> delegate;
  final Future<dynamic> futureValue;
}

25 26
// A utility function used by Localizations to generate one future
// that completes when all of the LocalizationsDelegate.load() futures
27
// complete. The returned map is indexed by each delegate's type.
28 29 30 31 32 33 34 35 36 37 38
//
// The input future values must have distinct types.
//
// The returned Future<Map> will resolve when all of the input map's
// future values have resolved. If all of the input map's values are
// SynchronousFutures then a SynchronousFuture will be returned
// immediately.
//
// This is more complicated than just applying Future.wait to input
// because some of the input.values may be SynchronousFutures. We don't want
// to Future.wait for the synchronous futures.
39
Future<Map<Type, dynamic>> _loadAll(Locale locale, Iterable<LocalizationsDelegate<dynamic>> allDelegates) {
40
  final Map<Type, dynamic> output = <Type, dynamic>{};
41
  List<_Pending>? pendingList;
42

43 44
  // Only load the first delegate for each delegate type that supports
  // locale.languageCode.
45
  final Set<Type> types = <Type>{};
46
  final List<LocalizationsDelegate<dynamic>> delegates = <LocalizationsDelegate<dynamic>>[];
47
  for (final LocalizationsDelegate<dynamic> delegate in allDelegates) {
48
    if (!types.contains(delegate.type) && delegate.isSupported(locale)) {
49 50 51 52 53
      types.add(delegate.type);
      delegates.add(delegate);
    }
  }

54
  for (final LocalizationsDelegate<dynamic> delegate in delegates) {
55
    final Future<dynamic> inputValue = delegate.load(locale);
56 57 58 59 60
    dynamic completedValue;
    final Future<dynamic> futureValue = inputValue.then<dynamic>((dynamic value) {
      return completedValue = value;
    });
    if (completedValue != null) { // inputValue was a SynchronousFuture
61
      final Type type = delegate.type;
62 63 64
      assert(!output.containsKey(type));
      output[type] = completedValue;
    } else {
65
      pendingList ??= <_Pending>[];
66
      pendingList.add(_Pending(delegate, futureValue));
67 68 69
    }
  }

70 71
  // All of the delegate.load() values were synchronous futures, we're done.
  if (pendingList == null)
72
    return SynchronousFuture<Map<Type, dynamic>>(output);
73

74
  // Some of delegate.load() values were asynchronous futures. Wait for them.
75
  return Future.wait<dynamic>(pendingList.map<Future<dynamic>>((_Pending p) => p.futureValue))
76
    .then<Map<Type, dynamic>>((List<dynamic> values) {
77
      assert(values.length == pendingList!.length);
78
      for (int i = 0; i < values.length; i += 1) {
79
        final Type type = pendingList![i].delegate.type;
80 81 82 83 84
        assert(!output.containsKey(type));
        output[type] = values[i];
      }
      return output;
    });
85 86 87 88 89
}

/// A factory for a set of localized resources of type `T`, to be loaded by a
/// [Localizations] widget.
///
90 91 92 93
/// Typical applications have one [Localizations] widget which is created by the
/// [WidgetsApp] and configured with the app's `localizationsDelegates`
/// parameter (a list of delegates). The delegate's [type] is used to identify
/// the object created by an individual delegate's [load] method.
94 95 96 97 98
abstract class LocalizationsDelegate<T> {
  /// Abstract const constructor. This constructor enables subclasses to provide
  /// const constructors so that they can be used in const expressions.
  const LocalizationsDelegate();

99 100 101 102 103 104
  /// Whether resources for the given locale can be loaded by this delegate.
  ///
  /// Return true if the instance of `T` loaded by this delegate's [load]
  /// method supports the given `locale`'s language.
  bool isSupported(Locale locale);

105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
  /// Start loading the resources for `locale`. The returned future completes
  /// when the resources have finished loading.
  ///
  /// It's assumed that the this method will return an object that contains
  /// a collection of related resources (typically defined with one method per
  /// resource). The object will be retrieved with [Localizations.of].
  Future<T> load(Locale locale);

  /// Returns true if the resources for this delegate should be loaded
  /// again by calling the [load] method.
  ///
  /// This method is called whenever its [Localizations] widget is
  /// rebuilt. If it returns true then dependent widgets will be rebuilt
  /// after [load] has completed.
  bool shouldReload(covariant LocalizationsDelegate<T> old);
120

121 122 123 124 125 126 127 128 129 130 131
  /// The type of the object returned by the [load] method, T by default.
  ///
  /// This type is used to retrieve the object "loaded" by this
  /// [LocalizationsDelegate] from the [Localizations] inherited widget.
  /// For example the object loaded by `LocalizationsDelegate<Foo>` would
  /// be retrieved with:
  /// ```dart
  /// Foo foo = Localizations.of<Foo>(context, Foo);
  /// ```
  ///
  /// It's rarely necessary to override this getter.
132 133
  Type get type => T;

134
  @override
135
  String toString() => '${objectRuntimeType(this, 'LocalizationsDelegate')}[$type]';
136 137 138 139 140 141 142 143
}

/// Interface for localized resource values for the lowest levels of the Flutter
/// framework.
///
/// In particular, this maps locales to a specific [Directionality] using the
/// [textDirection] property.
///
144 145 146 147 148
/// See also:
///
///  * [DefaultWidgetsLocalizations], which implements this interface and
///    supports a variety of locales.
abstract class WidgetsLocalizations {
149
  /// The reading direction for text in this locale.
150
  TextDirection get textDirection;
151 152 153 154 155

  /// The `WidgetsLocalizations` from the closest [Localizations] instance
  /// that encloses the given context.
  ///
  /// This method is just a convenient shorthand for:
156
  /// `Localizations.of<WidgetsLocalizations>(context, WidgetsLocalizations)!`.
157 158 159 160 161 162 163
  ///
  /// References to the localized resources defined by this class are typically
  /// written in terms of this method. For example:
  ///
  /// ```dart
  /// textDirection: WidgetsLocalizations.of(context).textDirection,
  /// ```
164 165 166
  static WidgetsLocalizations of(BuildContext context) {
    assert(debugCheckHasWidgetsLocalizations(context));
    return Localizations.of<WidgetsLocalizations>(context, WidgetsLocalizations)!;
167
  }
168 169
}

170 171 172
class _WidgetsLocalizationsDelegate extends LocalizationsDelegate<WidgetsLocalizations> {
  const _WidgetsLocalizationsDelegate();

173 174 175 176 177
  // This is convenient simplification. It would be more correct test if the locale's
  // text-direction is LTR.
  @override
  bool isSupported(Locale locale) => true;

178 179 180 181 182
  @override
  Future<WidgetsLocalizations> load(Locale locale) => DefaultWidgetsLocalizations.load(locale);

  @override
  bool shouldReload(_WidgetsLocalizationsDelegate old) => false;
183 184 185

  @override
  String toString() => 'DefaultWidgetsLocalizations.delegate(en_US)';
186 187 188 189 190 191
}

/// US English localizations for the widgets library.
///
/// See also:
///
192 193
///  * [GlobalWidgetsLocalizations], which provides widgets localizations for
///    many languages.
194
///  * [WidgetsApp.localizationsDelegates], which automatically includes
195
///    [DefaultWidgetsLocalizations.delegate] by default.
196
class DefaultWidgetsLocalizations implements WidgetsLocalizations {
197
  /// Construct an object that defines the localized values for the widgets
198
  /// library for US English (only).
199 200
  ///
  /// [LocalizationsDelegate] implementations typically call the static [load]
201
  const DefaultWidgetsLocalizations();
202 203

  @override
204
  TextDirection get textDirection => TextDirection.ltr;
205

206 207 208 209
  /// Creates an object that provides US English resource values for the
  /// lowest levels of the widgets library.
  ///
  /// The [locale] parameter is ignored.
210 211 212 213
  ///
  /// This method is typically used to create a [LocalizationsDelegate].
  /// The [WidgetsApp] does so by default.
  static Future<WidgetsLocalizations> load(Locale locale) {
214
    return SynchronousFuture<WidgetsLocalizations>(const DefaultWidgetsLocalizations());
215
  }
216 217 218 219

  /// A [LocalizationsDelegate] that uses [DefaultWidgetsLocalizations.load]
  /// to create an instance of this class.
  ///
220
  /// [WidgetsApp] automatically adds this value to [WidgetsApp.localizationsDelegates].
221
  static const LocalizationsDelegate<WidgetsLocalizations> delegate = _WidgetsLocalizationsDelegate();
222 223
}

224
class _LocalizationsScope extends InheritedWidget {
225
  const _LocalizationsScope({
226 227 228 229 230
    Key? key,
    required this.locale,
    required this.localizationsState,
    required this.typeToResources,
    required Widget child,
231 232 233
  }) : assert(localizationsState != null),
       assert(typeToResources != null),
       super(key: key, child: child);
234 235 236

  final Locale locale;
  final _LocalizationsState localizationsState;
237
  final Map<Type, dynamic> typeToResources;
238

239 240
  @override
  bool updateShouldNotify(_LocalizationsScope old) {
241
    return typeToResources != old.typeToResources;
242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
  }
}

/// Defines the [Locale] for its `child` and the localized resources that the
/// child depends on.
///
/// Localized resources are loaded by the list of [LocalizationsDelegate]
/// `delegates`. Each delegate is essentially a factory for a collection
/// of localized resources. There are multiple delegates because there are
/// multiple sources for localizations within an app.
///
/// Delegates are typically simple subclasses of [LocalizationsDelegate] that
/// override [LocalizationsDelegate.load]. For example a delegate for the
/// `MyLocalizations` class defined below would be:
///
/// ```dart
/// class _MyDelegate extends LocalizationsDelegate<MyLocalizations> {
///   @override
///   Future<MyLocalizations> load(Locale locale) => MyLocalizations.load(locale);
261
///
262 263 264
///   @override
///   bool shouldReload(MyLocalizationsDelegate old) => false;
/// }
265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293
/// ```
///
/// Each delegate can be viewed as a factory for objects that encapsulate a
/// a set of localized resources. These objects are retrieved with
/// by runtime type with [Localizations.of].
///
/// The [WidgetsApp] class creates a `Localizations` widget so most apps
/// will not need to create one. The widget app's `Localizations` delegates can
/// be initialized with [WidgetsApp.localizationsDelegates]. The [MaterialApp]
/// class also provides a `localizationsDelegates` parameter that's just
/// passed along to the [WidgetsApp].
///
/// Apps should retrieve collections of localized resources with
/// `Localizations.of<MyLocalizations>(context, MyLocalizations)`,
/// where MyLocalizations is an app specific class defines one function per
/// resource. This is conventionally done by a static `.of` method on the
/// MyLocalizations class.
///
/// For example, using the `MyLocalizations` class defined below, one would
/// lookup a localized title string like this:
/// ```dart
/// MyLocalizations.of(context).title()
/// ```
/// If `Localizations` were to be rebuilt with a new `locale` then
/// the widget subtree that corresponds to [BuildContext] `context` would
/// be rebuilt after the corresponding resources had been loaded.
///
/// This class is effectively an [InheritedWidget]. If it's rebuilt with
/// a new `locale` or a different list of delegates or any of its
294
/// delegates' [LocalizationsDelegate.shouldReload()] methods returns true,
295 296 297 298
/// then widgets that have created a dependency by calling
/// `Localizations.of(context)` will be rebuilt after the resources
/// for the new locale have been loaded.
///
299 300 301 302
/// The `Localizations` widget also instantiates [Directionality] in order to
/// support the appropriate [Directionality.textDirection] of the localized
/// resources.
///
303
/// {@tool snippet}
304 305 306 307 308 309 310 311 312 313 314 315 316
///
/// This following class is defined in terms of the
/// [Dart `intl` package](https://github.com/dart-lang/intl). Using the `intl`
/// package isn't required.
///
/// ```dart
/// class MyLocalizations {
///   MyLocalizations(this.locale);
///
///   final Locale locale;
///
///   static Future<MyLocalizations> load(Locale locale) {
///     return initializeMessages(locale.toString())
317
///       .then((void _) {
318
///         return MyLocalizations(locale);
319 320 321 322
///       });
///   }
///
///   static MyLocalizations of(BuildContext context) {
323
///     return Localizations.of<MyLocalizations>(context, MyLocalizations)!;
324 325 326 327 328 329
///   }
///
///   String title() => Intl.message('<title>', name: 'title', locale: locale.toString());
///   // ... more Intl.message() methods like title()
/// }
/// ```
330
/// {@end-tool}
331 332 333 334 335 336 337 338 339
/// A class based on the `intl` package imports a generated message catalog that provides
/// the `initializeMessages()` function and the per-locale backing store for `Intl.message()`.
/// The message catalog is produced by an `intl` tool that analyzes the source code for
/// classes that contain `Intl.message()` calls. In this case that would just be the
/// `MyLocalizations` class.
///
/// One could choose another approach for loading localized resources and looking them up while
/// still conforming to the structure of this example.
class Localizations extends StatefulWidget {
340
  /// Create a widget from which localizations (like translated strings) can be obtained.
341
  Localizations({
342 343 344
    Key? key,
    required this.locale,
    required this.delegates,
345
    this.child,
346 347 348 349
  }) : assert(locale != null),
       assert(delegates != null),
       assert(delegates.any((LocalizationsDelegate<dynamic> delegate) => delegate is LocalizationsDelegate<WidgetsLocalizations>)),
       super(key: key);
350

351 352
  /// Overrides the inherited [Locale] or [LocalizationsDelegate]s for `child`.
  ///
353
  /// This factory constructor is used for the (usually rare) situation where part
354 355 356 357 358 359 360 361 362 363
  /// of an app should be localized for a different locale than the one defined
  /// for the device, or if its localizations should come from a different list
  /// of [LocalizationsDelegate]s than the list defined by
  /// [WidgetsApp.localizationsDelegates].
  ///
  /// For example you could specify that `myWidget` was only to be localized for
  /// the US English locale:
  ///
  /// ```dart
  /// Widget build(BuildContext context) {
364
  ///   return Localizations.override(
365 366 367 368 369 370 371 372 373 374 375 376 377 378
  ///     context: context,
  ///     locale: const Locale('en', 'US'),
  ///     child: myWidget,
  ///   );
  /// }
  /// ```
  ///
  /// The `locale` and `delegates` parameters default to the [Localizations.locale]
  /// and [Localizations.delegates] values from the nearest [Localizations] ancestor.
  ///
  /// To override the [Localizations.locale] or [Localizations.delegates] for an
  /// entire app, specify [WidgetsApp.locale] or [WidgetsApp.localizationsDelegates]
  /// (or specify the same parameters for [MaterialApp]).
  factory Localizations.override({
379 380 381 382 383
    Key? key,
    required BuildContext context,
    Locale? locale,
    List<LocalizationsDelegate<dynamic>>? delegates,
    Widget? child,
384 385 386 387
  }) {
    final List<LocalizationsDelegate<dynamic>> mergedDelegates = Localizations._delegatesOf(context);
    if (delegates != null)
      mergedDelegates.insertAll(0, delegates);
388
    return Localizations(
389
      key: key,
390
      locale: locale ?? Localizations.localeOf(context),
391 392 393 394 395
      delegates: mergedDelegates,
      child: child,
    );
  }

396 397 398 399
  /// The resources returned by [Localizations.of] will be specific to this locale.
  final Locale locale;

  /// This list collectively defines the localized resources objects that can
400 401
  /// be retrieved with [Localizations.of].
  final List<LocalizationsDelegate<dynamic>> delegates;
402 403

  /// The widget below this widget in the tree.
404
  ///
405
  /// {@macro flutter.widgets.ProxyWidget.child}
406
  final Widget? child;
407 408 409

  /// The locale of the Localizations widget for the widget tree that
  /// corresponds to [BuildContext] `context`.
Ian Hickson's avatar
Ian Hickson committed
410 411
  ///
  /// If no [Localizations] widget is in scope then the [Localizations.localeOf]
412 413
  /// method will throw an exception.
  static Locale localeOf(BuildContext context) {
414
    assert(context != null);
415
    final _LocalizationsScope? scope = context.dependOnInheritedWidgetOfExactType<_LocalizationsScope>();
416 417 418 419 420 421 422 423
    assert(() {
      if (scope == null) {
        throw FlutterError(
          'Requested the Locale of a context that does not include a Localizations ancestor.\n'
          'To request the Locale, the context used to retrieve the Localizations widget must '
          'be that of a widget that is a descendant of a Localizations widget.'
        );
      }
424
      if (scope.localizationsState.locale == null) {
425 426 427 428 429 430
        throw FlutterError(
          'Localizations.localeOf found a Localizations widget that had a unexpected null locale.\n'
        );
      }
      return true;
    }());
431
    return scope!.localizationsState.locale!;
432 433
  }

434 435 436 437
  /// The locale of the Localizations widget for the widget tree that
  /// corresponds to [BuildContext] `context`.
  ///
  /// If no [Localizations] widget is in scope then this function will return
438
  /// null.
439 440 441 442 443 444
  static Locale? maybeLocaleOf(BuildContext context) {
    assert(context != null);
    final _LocalizationsScope? scope = context.dependOnInheritedWidgetOfExactType<_LocalizationsScope>();
    return scope?.localizationsState.locale;
  }

445 446 447 448
  // There doesn't appear to be a need to make this public. See the
  // Localizations.override factory constructor.
  static List<LocalizationsDelegate<dynamic>> _delegatesOf(BuildContext context) {
    assert(context != null);
449
    final _LocalizationsScope? scope = context.dependOnInheritedWidgetOfExactType<_LocalizationsScope>();
450
    assert(scope != null, 'a Localizations ancestor was not found');
451
    return List<LocalizationsDelegate<dynamic>>.from(scope!.localizationsState.widget.delegates);
452 453
  }

454 455 456
  /// Returns the localized resources object of the given `type` for the widget
  /// tree that corresponds to the given `context`.
  ///
457
  /// Returns null if no resources object of the given `type` exists within
458
  /// the given `context`.
459
  ///
460
  /// This method is typically used by a static factory method on the `type`
461 462 463 464 465 466 467 468
  /// class. For example Flutter's MaterialLocalizations class looks up Material
  /// resources with a method defined like this:
  ///
  /// ```dart
  /// static MaterialLocalizations of(BuildContext context) {
  ///    return Localizations.of<MaterialLocalizations>(context, MaterialLocalizations);
  /// }
  /// ```
469
  static T? of<T>(BuildContext context, Type type) {
470 471
    assert(context != null);
    assert(type != null);
472
    final _LocalizationsScope? scope = context.dependOnInheritedWidgetOfExactType<_LocalizationsScope>();
473
    return scope?.localizationsState.resourcesFor<T?>(type);
474 475 476
  }

  @override
477
  _LocalizationsState createState() => _LocalizationsState();
478 479

  @override
480 481
  void debugFillProperties(DiagnosticPropertiesBuilder properties) {
    super.debugFillProperties(properties);
482 483
    properties.add(DiagnosticsProperty<Locale>('locale', locale));
    properties.add(IterableProperty<LocalizationsDelegate<dynamic>>('delegates', delegates));
484
  }
485 486 487
}

class _LocalizationsState extends State<Localizations> {
488
  final GlobalKey _localizedResourcesScopeKey = GlobalKey();
489 490
  Map<Type, dynamic> _typeToResources = <Type, dynamic>{};

491 492
  Locale? get locale => _locale;
  Locale? _locale;
493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517

  @override
  void initState() {
    super.initState();
    load(widget.locale);
  }

  bool _anyDelegatesShouldReload(Localizations old) {
    if (widget.delegates.length != old.delegates.length)
      return true;
    final List<LocalizationsDelegate<dynamic>> delegates = widget.delegates.toList();
    final List<LocalizationsDelegate<dynamic>> oldDelegates = old.delegates.toList();
    for (int i = 0; i < delegates.length; i += 1) {
      final LocalizationsDelegate<dynamic> delegate = delegates[i];
      final LocalizationsDelegate<dynamic> oldDelegate = oldDelegates[i];
      if (delegate.runtimeType != oldDelegate.runtimeType || delegate.shouldReload(oldDelegate))
        return true;
    }
    return false;
  }

  @override
  void didUpdateWidget(Localizations old) {
    super.didUpdateWidget(old);
    if (widget.locale != old.locale
518
        || (widget.delegates == null)
519 520 521 522 523 524 525 526 527 528 529 530
        || (widget.delegates != null && old.delegates == null)
        || (widget.delegates != null && _anyDelegatesShouldReload(old)))
      load(widget.locale);
  }

  void load(Locale locale) {
    final Iterable<LocalizationsDelegate<dynamic>> delegates = widget.delegates;
    if (delegates == null || delegates.isEmpty) {
      _locale = locale;
      return;
    }

531
    Map<Type, dynamic>? typeToResources;
532
    final Future<Map<Type, dynamic>> typeToResourcesFuture = _loadAll(locale, delegates)
533
      .then<Map<Type, dynamic>>((Map<Type, dynamic> value) {
534 535 536 537 538
        return typeToResources = value;
      });

    if (typeToResources != null) {
      // All of the delegates' resources loaded synchronously.
539
      _typeToResources = typeToResources!;
540 541 542 543 544 545
      _locale = locale;
    } else {
      // - Don't rebuild the dependent widgets until the resources for the new locale
      // have finished loading. Until then the old locale will continue to be used.
      // - If we're running at app startup time then defer reporting the first
      // "useful" frame until after the async load has completed.
546
      RendererBinding.instance!.deferFirstFrame();
547
      typeToResourcesFuture.then<void>((Map<Type, dynamic> value) {
548 549 550 551 552 553
        if (mounted) {
          setState(() {
            _typeToResources = value;
            _locale = locale;
          });
        }
554
        RendererBinding.instance!.allowFirstFrame();
555 556 557 558 559 560
      });
    }
  }

  T resourcesFor<T>(Type type) {
    assert(type != null);
561
    final T resources = _typeToResources[type] as T;
562 563 564
    return resources;
  }

565
  TextDirection get _textDirection {
566
    final WidgetsLocalizations resources = _typeToResources[WidgetsLocalizations] as WidgetsLocalizations;
567 568 569 570
    assert(resources != null);
    return resources.textDirection;
  }

571 572
  @override
  Widget build(BuildContext context) {
573
    if (_locale == null)
574 575
      return Container();
    return Semantics(
576
      textDirection: _textDirection,
577
      child: _LocalizationsScope(
578
        key: _localizedResourcesScopeKey,
579
        locale: _locale!,
580 581
        localizationsState: this,
        typeToResources: _typeToResources,
582
        child: Directionality(
583
          textDirection: _textDirection,
584
          child: widget.child!,
585
        ),
586
      ),
587 588 589
    );
  }
}