bottom_navigation_bar.dart 35.5 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 'dart:collection' show Queue;
6
import 'dart:math' as math;
7 8

import 'package:flutter/widgets.dart';
9
import 'package:flutter/rendering.dart';
10 11
import 'package:vector_math/vector_math_64.dart' show Vector3;

12
import 'bottom_navigation_bar_theme.dart';
13
import 'constants.dart';
14
import 'debug.dart';
15 16
import 'ink_well.dart';
import 'material.dart';
17
import 'material_localizations.dart';
18
import 'text_theme.dart';
19 20 21 22 23 24 25
import 'theme.dart';

/// Defines the layout and behavior of a [BottomNavigationBar].
///
/// See also:
///
///  * [BottomNavigationBar]
26
///  * [BottomNavigationBarItem]
27
///  * <https://material.io/design/components/bottom-navigation.html#specs>
28
enum BottomNavigationBarType {
29
  /// The [BottomNavigationBar]'s [BottomNavigationBarItem]s have fixed width.
30 31
  fixed,

32
  /// The location and size of the [BottomNavigationBar] [BottomNavigationBarItem]s
33
  /// animate and labels fade in when they are tapped.
34 35 36
  shifting,
}

37 38
/// A material widget that's displayed at the bottom of an app for selecting
/// among a small number of views, typically between three and five.
39
///
40
/// The bottom navigation bar consists of multiple items in the form of
41 42 43
/// text labels, icons, or both, laid out on top of a piece of material. It
/// provides quick navigation between the top-level views of an app. For larger
/// screens, side navigation may be a better fit.
44
///
45 46
/// A bottom navigation bar is usually used in conjunction with a [Scaffold],
/// where it is provided as the [Scaffold.bottomNavigationBar] argument.
47
///
48
/// The bottom navigation bar's [type] changes how its [items] are displayed.
49 50 51
/// If not specified, then it's automatically set to
/// [BottomNavigationBarType.fixed] when there are less than four items, and
/// [BottomNavigationBarType.shifting] otherwise.
52 53
///
///  * [BottomNavigationBarType.fixed], the default when there are less than
54 55 56 57
///    four [items]. The selected item is rendered with the
///    [selectedItemColor] if it's non-null, otherwise the theme's
///    [ThemeData.primaryColor] is used. If [backgroundColor] is null, The
///    navigation bar's background color defaults to the [Material] background
58 59
///    color, [ThemeData.canvasColor] (essentially opaque white).
///  * [BottomNavigationBarType.shifting], the default when there are four
60 61
///    or more [items]. If [selectedItemColor] is null, all items are rendered
///    in white. The navigation bar's background color is the same as the
62 63 64 65
///    [BottomNavigationBarItem.backgroundColor] of the selected item. In this
///    case it's assumed that each item will have a different background color
///    and that background color will contrast well with white.
///
66
/// {@tool dartpad --template=stateful_widget_material}
67 68
/// This example shows a [BottomNavigationBar] as it is used within a [Scaffold]
/// widget. The [BottomNavigationBar] has three [BottomNavigationBarItem]
69
/// widgets and the [currentIndex] is set to index 0. The selected item is
70
/// amber. The `_onItemTapped` function changes the selected item's index
71
/// and displays a corresponding message in the center of the [Scaffold].
72
///
73 74 75
/// ![A scaffold with a bottom navigation bar containing three bottom navigation
/// bar items. The first one is selected.](https://flutter.github.io/assets-for-api-docs/assets/material/bottom_navigation_bar.png)
///
76
/// ```dart
77
/// int _selectedIndex = 0;
78 79 80 81 82 83 84 85 86 87 88 89 90 91
/// static const TextStyle optionStyle = TextStyle(fontSize: 30, fontWeight: FontWeight.bold);
/// static const List<Widget> _widgetOptions = <Widget>[
///   Text(
///     'Index 0: Home',
///     style: optionStyle,
///   ),
///   Text(
///      'Index 1: Business',
///      style: optionStyle,
///   ),
///   Text(
///      'Index 2: School',
///      style: optionStyle,
///   ),
92
/// ];
93
///
94 95 96 97
/// void _onItemTapped(int index) {
///   setState(() {
///     _selectedIndex = index;
///   });
98 99
/// }
///
100 101 102 103
/// @override
/// Widget build(BuildContext context) {
///   return Scaffold(
///     appBar: AppBar(
104
///       title: const Text('BottomNavigationBar Sample'),
105 106 107 108 109
///     ),
///     body: Center(
///       child: _widgetOptions.elementAt(_selectedIndex),
///     ),
///     bottomNavigationBar: BottomNavigationBar(
110 111 112 113 114 115 116 117 118 119 120 121 122
///       items: const <BottomNavigationBarItem>[
///         BottomNavigationBarItem(
///           icon: Icon(Icons.home),
///           title: Text('Home'),
///         ),
///         BottomNavigationBarItem(
///           icon: Icon(Icons.business),
///           title: Text('Business'),
///         ),
///         BottomNavigationBarItem(
///           icon: Icon(Icons.school),
///           title: Text('School'),
///         ),
123 124
///       ],
///       currentIndex: _selectedIndex,
125
///       selectedItemColor: Colors.amber[800],
126 127 128
///       onTap: _onItemTapped,
///     ),
///   );
129 130
/// }
/// ```
131
/// {@end-tool}
132
///
133 134
/// See also:
///
135
///  * [BottomNavigationBarItem]
136
///  * [Scaffold]
137
///  * <https://material.io/design/components/bottom-navigation.html>
138
class BottomNavigationBar extends StatefulWidget {
139 140
  /// Creates a bottom navigation bar which is typically used as a
  /// [Scaffold]'s [Scaffold.bottomNavigationBar] argument.
141
  ///
142 143
  /// The length of [items] must be at least two and each item's icon and title
  /// must not be null.
144
  ///
145 146
  /// If [type] is null then [BottomNavigationBarType.fixed] is used when there
  /// are two or three [items], [BottomNavigationBarType.shifting] otherwise.
147
  ///
148 149 150
  /// The [iconSize], [selectedFontSize], [unselectedFontSize], and [elevation]
  /// arguments must be non-null and non-negative.
  ///
151 152 153 154 155 156 157 158 159 160 161
  /// If [selectedLabelStyle.color] and [unselectedLabelStyle.color] values
  /// are non-null, they will be used instead of [selectedItemColor] and
  /// [unselectedItemColor].
  ///
  /// If custom [IconThemData]s are used, you must provide both
  /// [selectedIconTheme] and [unselectedIconTheme], and both
  /// [IconThemeData.color] and [IconThemeData.size] must be set.
  ///
  /// If both [selectedLabelStyle.fontSize] and [selectedFontSize] are set,
  /// [selectedLabelStyle.fontSize] will be used.
  ///
162 163 164 165 166 167 168 169 170
  /// Only one of [selectedItemColor] and [fixedColor] can be specified. The
  /// former is preferred, [fixedColor] only exists for the sake of
  /// backwards compatibility.
  ///
  /// The [showSelectedLabels] argument must not be non-null.
  ///
  /// The [showUnselectedLabels] argument defaults to `true` if [type] is
  /// [BottomNavigationBarType.fixed] and `false` if [type] is
  /// [BottomNavigationBarType.shifting].
171 172
  BottomNavigationBar({
    Key key,
173
    @required this.items,
174
    this.onTap,
175
    this.currentIndex = 0,
176 177
    this.elevation,
    this.type,
178 179
    Color fixedColor,
    this.backgroundColor,
180
    this.iconSize = 24.0,
181 182
    Color selectedItemColor,
    this.unselectedItemColor,
183 184
    this.selectedIconTheme,
    this.unselectedIconTheme,
185 186
    this.selectedFontSize = 14.0,
    this.unselectedFontSize = 12.0,
187 188
    this.selectedLabelStyle,
    this.unselectedLabelStyle,
189
    this.showSelectedLabels = true,
190
    this.showUnselectedLabels,
191
    this.mouseCursor,
192 193
  }) : assert(items != null),
       assert(items.length >= 2),
194 195 196 197
       assert(
        items.every((BottomNavigationBarItem item) => item.title != null) == true,
        'Every item must have a non-null title',
       ),
198
       assert(0 <= currentIndex && currentIndex < items.length),
199
       assert(elevation == null || elevation >= 0.0),
200 201
       assert(iconSize != null && iconSize >= 0.0),
       assert(
202
         selectedItemColor == null || fixedColor == null,
203 204 205 206 207 208
         'Either selectedItemColor or fixedColor can be specified, but not both'
       ),
       assert(selectedFontSize != null && selectedFontSize >= 0.0),
       assert(unselectedFontSize != null && unselectedFontSize >= 0.0),
       assert(showSelectedLabels != null),
       selectedItemColor = selectedItemColor ?? fixedColor,
209
       super(key: key);
210

211 212
  /// Defines the appearance of the button items that are arrayed within the
  /// bottom navigation bar.
213
  final List<BottomNavigationBarItem> items;
214

215
  /// Called when one of the [items] is tapped.
216
  ///
217 218 219
  /// The stateful widget that creates the bottom navigation bar needs to keep
  /// track of the index of the selected [BottomNavigationBarItem] and call
  /// `setState` to rebuild the bottom navigation bar with the new [currentIndex].
220 221
  final ValueChanged<int> onTap;

222
  /// The index into [items] for the current active [BottomNavigationBarItem].
223 224
  final int currentIndex;

225 226 227 228 229 230 231
  /// The z-coordinate of this [BottomNavigationBar].
  ///
  /// If null, defaults to `8.0`.
  ///
  /// {@macro flutter.material.material.elevation}
  final double elevation;

232
  /// Defines the layout and behavior of a [BottomNavigationBar].
233
  ///
234 235
  /// See documentation for [BottomNavigationBarType] for information on the
  /// meaning of different types.
236 237
  final BottomNavigationBarType type;

238 239 240 241 242 243 244
  /// The value of [selectedItemColor].
  ///
  /// This getter only exists for backwards compatibility, the
  /// [selectedItemColor] property is preferred.
  Color get fixedColor => selectedItemColor;

  /// The color of the [BottomNavigationBar] itself.
245
  ///
246 247 248 249
  /// If [type] is [BottomNavigationBarType.shifting] and the
  /// [items]s, have [BottomNavigationBarItem.backgroundColor] set, the [item]'s
  /// backgroundColor will splash and overwrite this color.
  final Color backgroundColor;
250

251
  /// The size of all of the [BottomNavigationBarItem] icons.
252
  ///
253
  /// See [BottomNavigationBarItem.icon] for more information.
254 255
  final double iconSize;

256 257 258 259 260 261 262 263 264 265 266 267
  /// The color of the selected [BottomNavigationBarItem.icon] and
  /// [BottomNavigationBarItem.label].
  ///
  /// If null then the [ThemeData.primaryColor] is used.
  final Color selectedItemColor;

  /// The color of the unselected [BottomNavigationBarItem.icon] and
  /// [BottomNavigationBarItem.label]s.
  ///
  /// If null then the [TextTheme.caption]'s color is used.
  final Color unselectedItemColor;

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 294 295 296 297
  /// The size, opacity, and color of the icon in the currently selected
  /// [BottomNavigationBarItem.icon].
  ///
  /// If this is not provided, the size will default to [iconSize], the color
  /// will default to [selectedItemColor].
  ///
  /// It this field is provided, it must contain non-null [IconThemeData.size]
  /// and [IconThemeData.color] properties. Also, if this field is supplied,
  /// [unselectedIconTheme] must be provided.
  final IconThemeData selectedIconTheme;

  /// The size, opacity, and color of the icon in the currently unselected
  /// [BottomNavigationBarItem.icon]s
  ///
  /// If this is not provided, the size will default to [iconSize], the color
  /// will default to [unselectedItemColor].
  ///
  /// It this field is provided, it must contain non-null [IconThemeData.size]
  /// and [IconThemeData.color] properties. Also, if this field is supplied,
  /// [unselectedIconTheme] must be provided.
  final IconThemeData unselectedIconTheme;

  /// The [TextStyle] of the [BottomNavigationBarItem] labels when they are
  /// selected.
  final TextStyle selectedLabelStyle;

  /// The [TextStyle] of the [BottomNavigationBarItem] labels when they are not
  /// selected.
  final TextStyle unselectedLabelStyle;

298 299
  /// The font size of the [BottomNavigationBarItem] labels when they are selected.
  ///
300 301
  /// If [selectedLabelStyle.fontSize] is non-null, it will be used instead of this.
  ///
302 303 304 305 306 307
  /// Defaults to `14.0`.
  final double selectedFontSize;

  /// The font size of the [BottomNavigationBarItem] labels when they are not
  /// selected.
  ///
308 309
  /// If [unselectedLabelStyle.fontSize] is non-null, it will be used instead of this.
  ///
310 311 312 313 314 315 316 317 318
  /// Defaults to `12.0`.
  final double unselectedFontSize;

  /// Whether the labels are shown for the selected [BottomNavigationBarItem].
  final bool showUnselectedLabels;

  /// Whether the labels are shown for the unselected [BottomNavigationBarItem]s.
  final bool showSelectedLabels;

319 320 321 322 323 324
  /// The cursor for a mouse pointer when it enters or is hovering over the
  /// tiles.
  ///
  /// If this property is null, [SystemMouseCursors.click] will be used.
  final MouseCursor mouseCursor;

325
  @override
326
  _BottomNavigationBarState createState() => _BottomNavigationBarState();
327 328
}

329 330 331
// This represents a single tile in the bottom navigation bar. It is intended
// to go into a flex container.
class _BottomNavigationTile extends StatelessWidget {
332
  const _BottomNavigationTile(
333 334 335 336 337 338
    this.type,
    this.item,
    this.animation,
    this.iconSize, {
    this.onTap,
    this.colorTween,
339
    this.flex,
340
    this.selected = false,
341 342 343 344
    @required this.selectedLabelStyle,
    @required this.unselectedLabelStyle,
    @required this.selectedIconTheme,
    @required this.unselectedIconTheme,
345 346
    this.showSelectedLabels,
    this.showUnselectedLabels,
347
    this.indexLabel,
348
    @required this.mouseCursor,
349 350 351 352
    }) : assert(type != null),
         assert(item != null),
         assert(animation != null),
         assert(selected != null),
353
         assert(selectedLabelStyle != null),
354 355
         assert(unselectedLabelStyle != null),
         assert(mouseCursor != null);
356 357 358 359 360 361 362 363

  final BottomNavigationBarType type;
  final BottomNavigationBarItem item;
  final Animation<double> animation;
  final double iconSize;
  final VoidCallback onTap;
  final ColorTween colorTween;
  final double flex;
364
  final bool selected;
365 366 367 368
  final IconThemeData selectedIconTheme;
  final IconThemeData unselectedIconTheme;
  final TextStyle selectedLabelStyle;
  final TextStyle unselectedLabelStyle;
369
  final String indexLabel;
370 371
  final bool showSelectedLabels;
  final bool showUnselectedLabels;
372
  final MouseCursor mouseCursor;
373

374 375 376 377 378 379 380
  @override
  Widget build(BuildContext context) {
    // In order to use the flex container to grow the tile during animation, we
    // need to divide the changes in flex allotment into smaller pieces to
    // produce smooth animation. We do this by multiplying the flex value
    // (which is an integer) by a large number.
    int size;
381

382 383
    final BottomNavigationBarThemeData bottomTheme = BottomNavigationBarTheme.of(context);

384 385
    final double selectedFontSize = selectedLabelStyle.fontSize;

386 387 388 389 390 391 392
    final double selectedIconSize = selectedIconTheme?.size
      ?? bottomTheme?.selectedIconTheme?.size
      ?? iconSize;
    final double unselectedIconSize = unselectedIconTheme?.size
      ?? bottomTheme?.unselectedIconTheme?.size
      ?? iconSize;

393 394 395 396 397 398
    // The amount that the selected icon is bigger than the unselected icons,
    // (or zero if the selected icon is not bigger than the unselected icons).
    final double selectedIconDiff = math.max(selectedIconSize - unselectedIconSize, 0);
    // The amount that the unselected icons are bigger than the selected icon,
    // (or zero if the unselected icons are not any bigger than the selected icon).
    final double unselectedIconDiff = math.max(unselectedIconSize - selectedIconSize, 0);
399 400 401 402 403

    // Defines the padding for the animating icons + labels.
    //
    // The animations go from "Unselected":
    // =======
404
    // |      <-- Padding equal to the text height + 1/2 selectedIconDiff.
405
    // |  ☆
406
    // | text <-- Invisible text + padding equal to 1/2 selectedIconDiff.
407 408 409 410 411
    // =======
    //
    // To "Selected":
    //
    // =======
412
    // |      <-- Padding equal to 1/2 text height + 1/2 unselectedIconDiff.
413 414
    // |  ☆
    // | text
415
    // |      <-- Padding equal to 1/2 text height + 1/2 unselectedIconDiff.
416
    // =======
417 418
    double bottomPadding;
    double topPadding;
419 420
    if (showSelectedLabels && !showUnselectedLabels) {
      bottomPadding = Tween<double>(
421 422
        begin: selectedIconDiff / 2.0,
        end: selectedFontSize / 2.0 - unselectedIconDiff / 2.0,
423 424
      ).evaluate(animation);
      topPadding = Tween<double>(
425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444
        begin: selectedFontSize + selectedIconDiff / 2.0,
        end: selectedFontSize / 2.0 - unselectedIconDiff / 2.0,
      ).evaluate(animation);
    } else if (!showSelectedLabels && !showUnselectedLabels) {
      bottomPadding = Tween<double>(
        begin: selectedIconDiff / 2.0,
        end: unselectedIconDiff / 2.0,
      ).evaluate(animation);
      topPadding = Tween<double>(
        begin: selectedFontSize + selectedIconDiff / 2.0,
        end: selectedFontSize + unselectedIconDiff / 2.0,
      ).evaluate(animation);
    } else {
      bottomPadding = Tween<double>(
        begin: selectedFontSize / 2.0 + selectedIconDiff / 2.0,
        end: selectedFontSize / 2.0 + unselectedIconDiff / 2.0,
      ).evaluate(animation);
      topPadding = Tween<double>(
        begin: selectedFontSize / 2.0 + selectedIconDiff / 2.0,
        end: selectedFontSize / 2.0 + unselectedIconDiff / 2.0,
445 446
      ).evaluate(animation);
    }
447 448 449 450 451 452 453 454 455 456 457 458 459 460 461

    switch (type) {
      case BottomNavigationBarType.fixed:
        size = 1;
        break;
      case BottomNavigationBarType.shifting:
        size = (flex * 1000.0).round();
        break;
    }

    return Expanded(
      flex: size,
      child: Semantics(
        container: true,
        selected: selected,
462 463 464 465
        child: Stack(
          children: <Widget>[
            InkResponse(
              onTap: onTap,
466
              mouseCursor: mouseCursor,
467 468 469 470 471 472 473 474 475 476 477 478 479
              child: Padding(
                padding: EdgeInsets.only(top: topPadding, bottom: bottomPadding),
                child: Column(
                  crossAxisAlignment: CrossAxisAlignment.center,
                  mainAxisAlignment: MainAxisAlignment.spaceBetween,
                  mainAxisSize: MainAxisSize.min,
                  children: <Widget>[
                    _TileIcon(
                      colorTween: colorTween,
                      animation: animation,
                      iconSize: iconSize,
                      selected: selected,
                      item: item,
480 481
                      selectedIconTheme: selectedIconTheme ?? bottomTheme.selectedIconTheme,
                      unselectedIconTheme: unselectedIconTheme ?? bottomTheme.unselectedIconTheme,
482 483 484 485 486
                    ),
                    _Label(
                      colorTween: colorTween,
                      animation: animation,
                      item: item,
487 488 489 490
                      selectedLabelStyle: selectedLabelStyle ?? bottomTheme.selectedLabelStyle,
                      unselectedLabelStyle: unselectedLabelStyle ?? bottomTheme.unselectedLabelStyle,
                      showSelectedLabels: showSelectedLabels ?? bottomTheme.showUnselectedLabels,
                      showUnselectedLabels: showUnselectedLabels ?? bottomTheme.showUnselectedLabels,
491 492
                    ),
                  ],
493
                ),
494
              ),
495 496 497 498 499
            ),
            Semantics(
              label: indexLabel,
            ),
          ],
500 501 502 503 504 505 506 507 508 509 510 511 512 513 514
        ),
      ),
    );
  }
}


class _TileIcon extends StatelessWidget {
  const _TileIcon({
    Key key,
    @required this.colorTween,
    @required this.animation,
    @required this.iconSize,
    @required this.selected,
    @required this.item,
515 516
    @required this.selectedIconTheme,
    @required this.unselectedIconTheme,
517 518 519
  }) : assert(selected != null),
       assert(item != null),
       super(key: key);
520 521 522 523 524 525

  final ColorTween colorTween;
  final Animation<double> animation;
  final double iconSize;
  final bool selected;
  final BottomNavigationBarItem item;
526 527
  final IconThemeData selectedIconTheme;
  final IconThemeData unselectedIconTheme;
528 529 530

  @override
  Widget build(BuildContext context) {
531
    final Color iconColor = colorTween.evaluate(animation);
532 533 534 535 536 537 538 539 540 541
    final IconThemeData defaultIconTheme = IconThemeData(
      color: iconColor,
      size: iconSize,
    );
    final IconThemeData iconThemeData = IconThemeData.lerp(
      defaultIconTheme.merge(unselectedIconTheme),
      defaultIconTheme.merge(selectedIconTheme),
      animation.value,
    );

542
    return Align(
543 544
      alignment: Alignment.topCenter,
      heightFactor: 1.0,
545 546
      child: Container(
        child: IconTheme(
547
          data: iconThemeData,
548
          child: selected ? item.activeIcon : item.icon,
549 550 551 552
        ),
      ),
    );
  }
553
}
554

555 556
class _Label extends StatelessWidget {
  const _Label({
557 558 559 560
    Key key,
    @required this.colorTween,
    @required this.animation,
    @required this.item,
561 562
    @required this.selectedLabelStyle,
    @required this.unselectedLabelStyle,
563 564 565 566 567
    @required this.showSelectedLabels,
    @required this.showUnselectedLabels,
  }) : assert(colorTween != null),
       assert(animation != null),
       assert(item != null),
568 569
       assert(selectedLabelStyle != null),
       assert(unselectedLabelStyle != null),
570 571 572
       assert(showSelectedLabels != null),
       assert(showUnselectedLabels != null),
       super(key: key);
573 574 575 576

  final ColorTween colorTween;
  final Animation<double> animation;
  final BottomNavigationBarItem item;
577 578
  final TextStyle selectedLabelStyle;
  final TextStyle unselectedLabelStyle;
579 580
  final bool showSelectedLabels;
  final bool showUnselectedLabels;
581 582 583

  @override
  Widget build(BuildContext context) {
584 585 586 587 588 589 590 591
    final double selectedFontSize = selectedLabelStyle.fontSize;
    final double unselectedFontSize = unselectedLabelStyle.fontSize;

    final TextStyle customStyle = TextStyle.lerp(
      unselectedLabelStyle,
      selectedLabelStyle,
      animation.value,
    );
592
    Widget text = DefaultTextStyle.merge(
593
      style: customStyle.copyWith(
594 595 596 597 598 599 600 601 602 603 604 605 606
        fontSize: selectedFontSize,
        color: colorTween.evaluate(animation),
      ),
      // The font size should grow here when active, but because of the way
      // font rendering works, it doesn't grow smoothly if we just animate
      // the font size, so we use a transform instead.
      child: Transform(
        transform: Matrix4.diagonal3(
          Vector3.all(
            Tween<double>(
              begin: unselectedFontSize / selectedFontSize,
              end: 1.0,
            ).evaluate(animation),
607 608
          ),
        ),
609 610
        alignment: Alignment.bottomCenter,
        child: item.title,
611 612 613
      ),
    );

614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635
    if (!showUnselectedLabels && !showSelectedLabels) {
      // Never show any labels.
      text = Opacity(
        alwaysIncludeSemantics: true,
        opacity: 0.0,
        child: text,
      );
    } else if (!showUnselectedLabels) {
      // Fade selected labels in.
      text = FadeTransition(
        alwaysIncludeSemantics: true,
        opacity: animation,
        child: text,
      );
    } else if (!showSelectedLabels) {
      // Fade selected labels out.
      text = FadeTransition(
        alwaysIncludeSemantics: true,
        opacity: Tween<double>(begin: 1.0, end: 0.0).animate(animation),
        child: text,
      );
    }
636

637
    return Align(
638 639
      alignment: Alignment.bottomCenter,
      heightFactor: 1.0,
640
      child: Container(child: text),
641 642 643 644
    );
  }
}

645
class _BottomNavigationBarState extends State<BottomNavigationBar> with TickerProviderStateMixin {
646
  List<AnimationController> _controllers = <AnimationController>[];
647
  List<CurvedAnimation> _animations;
648 649

  // A queue of color splashes currently being animated.
650
  final Queue<_Circle> _circles = Queue<_Circle>();
651

652 653 654 655
  // Last splash circle's color, and the final color of the control after
  // animation is complete.
  Color _backgroundColor;

656
  static final Animatable<double> _flexTween = Tween<double>(begin: 1.0, end: 1.5);
657

658
  void _resetState() {
659
    for (final AnimationController controller in _controllers)
660
      controller.dispose();
661
    for (final _Circle circle in _circles)
662 663 664
      circle.dispose();
    _circles.clear();

665 666
    _controllers = List<AnimationController>.generate(widget.items.length, (int index) {
      return AnimationController(
667 668
        duration: kThemeAnimationDuration,
        vsync: this,
669 670
      )..addListener(_rebuild);
    });
671 672
    _animations = List<CurvedAnimation>.generate(widget.items.length, (int index) {
      return CurvedAnimation(
673 674
        parent: _controllers[index],
        curve: Curves.fastOutSlowIn,
675
        reverseCurve: Curves.fastOutSlowIn.flipped,
676 677
      );
    });
678 679
    _controllers[widget.currentIndex].value = 1.0;
    _backgroundColor = widget.items[widget.currentIndex].backgroundColor;
680 681
  }

682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708
  // Computes the default value for the [type] parameter.
  //
  // If type is provided, it is returned. Next, if the bottom navigation bar
  // theme provides a type, it is used. Finally, the default behavior will be
  // [BottomNavigationBarType.fixed] for 3 or fewer items, and
  // [BottomNavigationBarType.shifting] is used for 4+ items.
  BottomNavigationBarType get _effectiveType {
    return widget.type
      ?? BottomNavigationBarTheme.of(context).type
      ?? (widget.items.length <= 3 ? BottomNavigationBarType.fixed : BottomNavigationBarType.shifting);
  }

  // Computes the default value for the [showUnselected] parameter.
  //
  // Unselected labels are shown by default for [BottomNavigationBarType.fixed],
  // and hidden by default for [BottomNavigationBarType.shifting].
  bool get _defaultShowUnselected {
    switch (_effectiveType) {
      case BottomNavigationBarType.shifting:
        return false;
      case BottomNavigationBarType.fixed:
        return true;
    }
    assert(false);
    return false;
  }

709 710 711 712 713 714
  @override
  void initState() {
    super.initState();
    _resetState();
  }

715 716
  void _rebuild() {
    setState(() {
717
      // Rebuilding when any of the controllers tick, i.e. when the items are
718 719 720 721
      // animated.
    });
  }

722 723
  @override
  void dispose() {
724
    for (final AnimationController controller in _controllers)
725
      controller.dispose();
726
    for (final _Circle circle in _circles)
727 728
      circle.dispose();
    super.dispose();
729 730
  }

731
  double _evaluateFlex(Animation<double> animation) => _flexTween.evaluate(animation);
732 733

  void _pushCircle(int index) {
734
    if (widget.items[index].backgroundColor != null) {
735
      _circles.add(
736
        _Circle(
737 738
          state: this,
          index: index,
739
          color: widget.items[index].backgroundColor,
740
          vsync: this,
741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757
        )..controller.addStatusListener(
          (AnimationStatus status) {
            switch (status) {
              case AnimationStatus.completed:
                setState(() {
                  final _Circle circle = _circles.removeFirst();
                  _backgroundColor = circle.color;
                  circle.dispose();
                });
                break;
              case AnimationStatus.dismissed:
              case AnimationStatus.forward:
              case AnimationStatus.reverse:
                break;
            }
          },
        ),
758
      );
759
    }
760 761 762
  }

  @override
763
  void didUpdateWidget(BottomNavigationBar oldWidget) {
764
    super.didUpdateWidget(oldWidget);
765 766 767 768 769 770 771

    // No animated segue if the length of the items list changes.
    if (widget.items.length != oldWidget.items.length) {
      _resetState();
      return;
    }

772
    if (widget.currentIndex != oldWidget.currentIndex) {
773
      switch (_effectiveType) {
774 775 776 777 778 779
        case BottomNavigationBarType.fixed:
          break;
        case BottomNavigationBarType.shifting:
          _pushCircle(widget.currentIndex);
          break;
      }
780 781
      _controllers[oldWidget.currentIndex].reverse();
      _controllers[widget.currentIndex].forward();
782 783 784
    } else {
      if (_backgroundColor != widget.items[widget.currentIndex].backgroundColor)
        _backgroundColor = widget.items[widget.currentIndex].backgroundColor;
785 786 787
    }
  }

788 789 790
  // If the given [TextStyle] has a non-null `fontSize`, it should be used.
  // Otherwise, the [selectedFontSize] parameter should be used.
  static TextStyle _effectiveTextStyle(TextStyle textStyle, double fontSize) {
791
    textStyle ??= const TextStyle();
792 793 794 795
    // Prefer the font size on textStyle if present.
    return textStyle.fontSize == null ? textStyle.copyWith(fontSize: fontSize) : textStyle;
  }

796
  List<Widget> _createTiles() {
797 798
    final MaterialLocalizations localizations = MaterialLocalizations.of(context);
    assert(localizations != null);
799 800

    final ThemeData themeData = Theme.of(context);
801
    final BottomNavigationBarThemeData bottomTheme = BottomNavigationBarTheme.of(context);
802

803
    final TextStyle effectiveSelectedLabelStyle =
804 805 806 807
      _effectiveTextStyle(
        widget.selectedLabelStyle ?? bottomTheme.selectedLabelStyle,
        widget.selectedFontSize,
      );
808
    final TextStyle effectiveUnselectedLabelStyle =
809 810 811 812
      _effectiveTextStyle(
        widget.unselectedLabelStyle ?? bottomTheme.unselectedLabelStyle,
        widget.unselectedFontSize,
      );
813

814 815 816 817 818 819 820 821 822 823 824
    Color themeColor;
    switch (themeData.brightness) {
      case Brightness.light:
        themeColor = themeData.primaryColor;
        break;
      case Brightness.dark:
        themeColor = themeData.accentColor;
        break;
    }

    ColorTween colorTween;
825
    switch (_effectiveType) {
826
      case BottomNavigationBarType.fixed:
827
        colorTween = ColorTween(
828 829 830 831 832 833 834
          begin: widget.unselectedItemColor
            ?? bottomTheme.unselectedItemColor
            ?? themeData.textTheme.caption.color,
          end: widget.selectedItemColor
            ?? bottomTheme.selectedItemColor
            ?? widget.fixedColor
            ?? themeColor,
835 836 837
        );
        break;
      case BottomNavigationBarType.shifting:
838
        colorTween = ColorTween(
839 840 841 842 843 844
          begin: widget.unselectedItemColor
            ?? bottomTheme.unselectedItemColor
            ?? themeData.colorScheme.surface,
          end: widget.selectedItemColor
            ?? bottomTheme.selectedItemColor
            ?? themeData.colorScheme.surface,
845
        );
846 847
        break;
    }
848
    final MouseCursor effectiveMouseCursor = widget.mouseCursor ?? SystemMouseCursors.click;
849 850 851 852

    final List<Widget> tiles = <Widget>[];
    for (int i = 0; i < widget.items.length; i++) {
      tiles.add(_BottomNavigationTile(
853
        _effectiveType,
854 855 856
        widget.items[i],
        _animations[i],
        widget.iconSize,
857 858
        selectedIconTheme: widget.selectedIconTheme ?? bottomTheme.selectedIconTheme,
        unselectedIconTheme: widget.unselectedIconTheme ?? bottomTheme.unselectedIconTheme,
859 860
        selectedLabelStyle: effectiveSelectedLabelStyle,
        unselectedLabelStyle: effectiveUnselectedLabelStyle,
861 862 863 864 865 866 867
        onTap: () {
          if (widget.onTap != null)
            widget.onTap(i);
        },
        colorTween: colorTween,
        flex: _evaluateFlex(_animations[i]),
        selected: i == widget.currentIndex,
868 869
        showSelectedLabels: widget.showSelectedLabels ?? bottomTheme.showSelectedLabels,
        showUnselectedLabels: widget.showUnselectedLabels ?? bottomTheme.showUnselectedLabels ?? _defaultShowUnselected,
870
        indexLabel: localizations.tabLabel(tabIndex: i + 1, tabCount: widget.items.length),
871
        mouseCursor: effectiveMouseCursor,
872 873 874
      ));
    }
    return tiles;
875 876 877 878 879
  }

  Widget _createContainer(List<Widget> tiles) {
    return DefaultTextStyle.merge(
      overflow: TextOverflow.ellipsis,
880
      child: Row(
881 882 883 884 885
        mainAxisAlignment: MainAxisAlignment.spaceBetween,
        children: tiles,
      ),
    );
  }
886

887 888
  @override
  Widget build(BuildContext context) {
889
    assert(debugCheckHasDirectionality(context));
890
    assert(debugCheckHasMaterialLocalizations(context));
891
    assert(debugCheckHasMediaQuery(context));
892

893 894
    final BottomNavigationBarThemeData bottomTheme = BottomNavigationBarTheme.of(context);

895
    // Labels apply up to _bottomMargin padding. Remainder is media padding.
896
    final double additionalBottomPadding = math.max(MediaQuery.of(context).padding.bottom - widget.selectedFontSize / 2.0, 0.0);
897
    Color backgroundColor;
898
    switch (_effectiveType) {
899
      case BottomNavigationBarType.fixed:
900
        backgroundColor = widget.backgroundColor ?? bottomTheme.backgroundColor;
901 902 903 904 905
        break;
      case BottomNavigationBarType.shifting:
        backgroundColor = _backgroundColor;
        break;
    }
906
    return Semantics(
907
      explicitChildNodes: true,
908
      child: Material(
909
        elevation: widget.elevation ?? bottomTheme.elevation ?? 8.0,
910 911 912 913 914 915 916
        color: backgroundColor,
        child: ConstrainedBox(
          constraints: BoxConstraints(minHeight: kBottomNavigationBarHeight + additionalBottomPadding),
          child: CustomPaint(
            painter: _RadialPainter(
              circles: _circles.toList(),
              textDirection: Directionality.of(context),
917
            ),
918 919 920 921 922 923 924 925
            child: Material( // Splashes.
              type: MaterialType.transparency,
              child: Padding(
                padding: EdgeInsets.only(bottom: additionalBottomPadding),
                child: MediaQuery.removePadding(
                  context: context,
                  removeBottom: true,
                  child: _createContainer(_createTiles()),
926
                ),
927
              ),
928
            ),
929
          ),
930
        ),
931
      ),
932 933 934 935
    );
  }
}

936
// Describes an animating color splash circle.
937 938
class _Circle {
  _Circle({
939 940 941
    @required this.state,
    @required this.index,
    @required this.color,
942
    @required TickerProvider vsync,
943 944 945
  }) : assert(state != null),
       assert(index != null),
       assert(color != null) {
946
    controller = AnimationController(
947 948
      duration: kThemeAnimationDuration,
      vsync: vsync,
949
    );
950
    animation = CurvedAnimation(
951
      parent: controller,
952
      curve: Curves.fastOutSlowIn,
953 954 955 956
    );
    controller.forward();
  }

957
  final _BottomNavigationBarState state;
958 959 960 961 962
  final int index;
  final Color color;
  AnimationController controller;
  CurvedAnimation animation;

963
  double get horizontalLeadingOffset {
964 965 966
    double weightSum(Iterable<Animation<double>> animations) {
      // We're adding flex values instead of animation values to produce correct
      // ratios.
967
      return animations.map<double>(state._evaluateFlex).fold<double>(0.0, (double sum, double value) => sum + value);
968 969 970
    }

    final double allWeights = weightSum(state._animations);
971 972
    // These weights sum to the start edge of the indexed item.
    final double leadingWeights = weightSum(state._animations.sublist(0, index));
973 974

    // Add half of its flex value in order to get to the center.
975
    return (leadingWeights + state._evaluateFlex(state._animations[index]) / 2.0) / allWeights;
976 977 978 979 980 981 982
  }

  void dispose() {
    controller.dispose();
  }
}

983
// Paints the animating color splash circles.
984 985
class _RadialPainter extends CustomPainter {
  _RadialPainter({
986 987 988 989
    @required this.circles,
    @required this.textDirection,
  }) : assert(circles != null),
       assert(textDirection != null);
990 991

  final List<_Circle> circles;
992
  final TextDirection textDirection;
993 994

  // Computes the maximum radius attainable such that at least one of the
995 996
  // bounding rectangle's corners touches the edge of the circle. Drawing a
  // circle larger than this radius is not needed, since there is no perceivable
997
  // difference within the cropped rectangle.
998 999 1000 1001
  static double _maxRadius(Offset center, Size size) {
    final double maxX = math.max(center.dx, size.width - center.dx);
    final double maxY = math.max(center.dy, size.height - center.dy);
    return math.sqrt(maxX * maxX + maxY * maxY);
1002 1003 1004 1005
  }

  @override
  bool shouldRepaint(_RadialPainter oldPainter) {
1006 1007
    if (textDirection != oldPainter.textDirection)
      return true;
1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019
    if (circles == oldPainter.circles)
      return false;
    if (circles.length != oldPainter.circles.length)
      return true;
    for (int i = 0; i < circles.length; i += 1)
      if (circles[i] != oldPainter.circles[i])
        return true;
    return false;
  }

  @override
  void paint(Canvas canvas, Size size) {
1020
    for (final _Circle circle in circles) {
1021 1022
      final Paint paint = Paint()..color = circle.color;
      final Rect rect = Rect.fromLTWH(0.0, 0.0, size.width, size.height);
1023
      canvas.clipRect(rect);
1024 1025 1026 1027 1028 1029 1030 1031 1032
      double leftFraction;
      switch (textDirection) {
        case TextDirection.rtl:
          leftFraction = 1.0 - circle.horizontalLeadingOffset;
          break;
        case TextDirection.ltr:
          leftFraction = circle.horizontalLeadingOffset;
          break;
      }
1033 1034
      final Offset center = Offset(leftFraction * size.width, size.height / 2.0);
      final Tween<double> radiusTween = Tween<double>(
1035 1036
        begin: 0.0,
        end: _maxRadius(center, size),
1037
      );
1038
      canvas.drawCircle(
1039
        center,
1040
        radiusTween.transform(circle.animation.value),
1041
        paint,
1042 1043 1044 1045
      );
    }
  }
}