| // 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 'dart:async'; |
| import 'dart:collection'; |
| import 'dart:math' as math; |
| import 'dart:ui' show lerpDouble; |
| |
| import 'package:flutter/foundation.dart'; |
| import 'package:flutter/gestures.dart' show DragStartBehavior; |
| import 'package:flutter/widgets.dart'; |
| |
| import 'app_bar.dart'; |
| import 'banner.dart'; |
| import 'banner_theme.dart'; |
| import 'bottom_sheet.dart'; |
| import 'colors.dart'; |
| import 'curves.dart'; |
| import 'debug.dart'; |
| import 'divider.dart'; |
| import 'drawer.dart'; |
| import 'flexible_space_bar.dart'; |
| import 'floating_action_button.dart'; |
| import 'floating_action_button_location.dart'; |
| import 'material.dart'; |
| import 'snack_bar.dart'; |
| import 'snack_bar_theme.dart'; |
| import 'theme.dart'; |
| |
| // Examples can assume: |
| // late TabController tabController; |
| // void setState(VoidCallback fn) { } |
| // late String appBarTitle; |
| // late int tabCount; |
| // late TickerProvider tickerProvider; |
| |
| const FloatingActionButtonLocation _kDefaultFloatingActionButtonLocation = FloatingActionButtonLocation.endFloat; |
| const FloatingActionButtonAnimator _kDefaultFloatingActionButtonAnimator = FloatingActionButtonAnimator.scaling; |
| |
| const Curve _standardBottomSheetCurve = standardEasing; |
| // When the top of the BottomSheet crosses this threshold, it will start to |
| // shrink the FAB and show a scrim. |
| const double _kBottomSheetDominatesPercentage = 0.3; |
| const double _kMinBottomSheetScrimOpacity = 0.1; |
| const double _kMaxBottomSheetScrimOpacity = 0.6; |
| |
| enum _ScaffoldSlot { |
| body, |
| appBar, |
| bodyScrim, |
| bottomSheet, |
| snackBar, |
| materialBanner, |
| persistentFooter, |
| bottomNavigationBar, |
| floatingActionButton, |
| drawer, |
| endDrawer, |
| statusBar, |
| } |
| |
| /// Manages [SnackBar]s and [MaterialBanner]s for descendant [Scaffold]s. |
| /// |
| /// {@youtube 560 315 https://www.youtube.com/watch?v=lytQi-slT5Y} |
| /// |
| /// This class provides APIs for showing snack bars and material banners at the |
| /// bottom and top of the screen, respectively. |
| /// |
| /// To display one of these notifications, obtain the [ScaffoldMessengerState] |
| /// for the current [BuildContext] via [ScaffoldMessenger.of] and use the |
| /// [ScaffoldMessengerState.showSnackBar] or the |
| /// [ScaffoldMessengerState.showMaterialBanner] functions. |
| /// |
| /// When the [ScaffoldMessenger] has nested [Scaffold] descendants, the |
| /// ScaffoldMessenger will only present the notification to the root Scaffold of |
| /// the subtree of Scaffolds. In order to show notifications for the inner, nested |
| /// Scaffolds, set a new scope by instantiating a new ScaffoldMessenger in |
| /// between the levels of nesting. |
| /// |
| /// {@tool dartpad} |
| /// Here is an example of showing a [SnackBar] when the user presses a button. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger.0.dart ** |
| /// {@end-tool} |
| /// |
| /// {@youtube 560 315 https://www.youtube.com/watch?v=lytQi-slT5Y} |
| /// |
| /// See also: |
| /// |
| /// * [SnackBar], which is a temporary notification typically shown near the |
| /// bottom of the app using the [ScaffoldMessengerState.showSnackBar] method. |
| /// * [MaterialBanner], which is a temporary notification typically shown at the |
| /// top of the app using the [ScaffoldMessengerState.showMaterialBanner] method. |
| /// * [debugCheckHasScaffoldMessenger], which asserts that the given context |
| /// has a [ScaffoldMessenger] ancestor. |
| /// * Cookbook: [Display a SnackBar](https://flutter.dev/docs/cookbook/design/snackbars) |
| class ScaffoldMessenger extends StatefulWidget { |
| /// Creates a widget that manages [SnackBar]s for [Scaffold] descendants. |
| const ScaffoldMessenger({ |
| super.key, |
| required this.child, |
| }); |
| |
| /// The widget below this widget in the tree. |
| /// |
| /// {@macro flutter.widgets.ProxyWidget.child} |
| final Widget child; |
| |
| /// The state from the closest instance of this class that encloses the given |
| /// context. |
| /// |
| /// {@tool dartpad} |
| /// Typical usage of the [ScaffoldMessenger.of] function is to call it in |
| /// response to a user gesture or an application state change. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger.of.0.dart ** |
| /// {@end-tool} |
| /// |
| /// A less elegant but more expedient solution is to assign a [GlobalKey] to the |
| /// [ScaffoldMessenger], then use the `key.currentState` property to obtain the |
| /// [ScaffoldMessengerState] rather than using the [ScaffoldMessenger.of] |
| /// function. The [MaterialApp.scaffoldMessengerKey] refers to the root |
| /// ScaffoldMessenger that is provided by default. |
| /// |
| /// {@tool dartpad} |
| /// Sometimes [SnackBar]s are produced by code that doesn't have ready access |
| /// to a valid [BuildContext]. One such example of this is when you show a |
| /// SnackBar from a method outside of the `build` function. In these |
| /// cases, you can assign a [GlobalKey] to the [ScaffoldMessenger]. This |
| /// example shows a key being used to obtain the [ScaffoldMessengerState] |
| /// provided by the [MaterialApp]. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger.of.1.dart ** |
| /// {@end-tool} |
| /// |
| /// If there is no [ScaffoldMessenger] in scope, then this will assert in |
| /// debug mode, and throw an exception in release mode. |
| /// |
| /// See also: |
| /// |
| /// * [maybeOf], which is a similar function but will return null instead of |
| /// throwing if there is no [ScaffoldMessenger] ancestor. |
| /// * [debugCheckHasScaffoldMessenger], which asserts that the given context |
| /// has a [ScaffoldMessenger] ancestor. |
| static ScaffoldMessengerState of(BuildContext context) { |
| assert(debugCheckHasScaffoldMessenger(context)); |
| |
| final _ScaffoldMessengerScope scope = context.dependOnInheritedWidgetOfExactType<_ScaffoldMessengerScope>()!; |
| return scope._scaffoldMessengerState; |
| } |
| |
| /// The state from the closest instance of this class that encloses the given |
| /// context, if any. |
| /// |
| /// Will return null if a [ScaffoldMessenger] is not found in the given context. |
| /// |
| /// See also: |
| /// |
| /// * [of], which is a similar function, except that it will throw an |
| /// exception if a [ScaffoldMessenger] is not found in the given context. |
| static ScaffoldMessengerState? maybeOf(BuildContext context) { |
| |
| final _ScaffoldMessengerScope? scope = context.dependOnInheritedWidgetOfExactType<_ScaffoldMessengerScope>(); |
| return scope?._scaffoldMessengerState; |
| } |
| |
| @override |
| ScaffoldMessengerState createState() => ScaffoldMessengerState(); |
| } |
| |
| /// State for a [ScaffoldMessenger]. |
| /// |
| /// A [ScaffoldMessengerState] object can be used to [showSnackBar] or |
| /// [showMaterialBanner] for every registered [Scaffold] that is a descendant of |
| /// the associated [ScaffoldMessenger]. Scaffolds will register to receive |
| /// [SnackBar]s and [MaterialBanner]s from their closest ScaffoldMessenger |
| /// ancestor. |
| /// |
| /// Typically obtained via [ScaffoldMessenger.of]. |
| class ScaffoldMessengerState extends State<ScaffoldMessenger> with TickerProviderStateMixin { |
| final LinkedHashSet<ScaffoldState> _scaffolds = LinkedHashSet<ScaffoldState>(); |
| final Queue<ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason>> _materialBanners = Queue<ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason>>(); |
| AnimationController? _materialBannerController; |
| final Queue<ScaffoldFeatureController<SnackBar, SnackBarClosedReason>> _snackBars = Queue<ScaffoldFeatureController<SnackBar, SnackBarClosedReason>>(); |
| AnimationController? _snackBarController; |
| Timer? _snackBarTimer; |
| bool? _accessibleNavigation; |
| |
| @override |
| void didChangeDependencies() { |
| final bool accessibleNavigation = MediaQuery.accessibleNavigationOf(context); |
| // If we transition from accessible navigation to non-accessible navigation |
| // and there is a SnackBar that would have timed out that has already |
| // completed its timer, dismiss that SnackBar. If the timer hasn't finished |
| // yet, let it timeout as normal. |
| if ((_accessibleNavigation ?? false) |
| && !accessibleNavigation |
| && _snackBarTimer != null |
| && !_snackBarTimer!.isActive) { |
| hideCurrentSnackBar(reason: SnackBarClosedReason.timeout); |
| } |
| _accessibleNavigation = accessibleNavigation; |
| super.didChangeDependencies(); |
| } |
| |
| void _register(ScaffoldState scaffold) { |
| _scaffolds.add(scaffold); |
| |
| if (_isRoot(scaffold)) { |
| if (_snackBars.isNotEmpty) { |
| scaffold._updateSnackBar(); |
| } |
| |
| if (_materialBanners.isNotEmpty) { |
| scaffold._updateMaterialBanner(); |
| } |
| } |
| } |
| |
| void _unregister(ScaffoldState scaffold) { |
| final bool removed = _scaffolds.remove(scaffold); |
| // ScaffoldStates should only be removed once. |
| assert(removed); |
| } |
| |
| void _updateScaffolds() { |
| for (final ScaffoldState scaffold in _scaffolds) { |
| if (_isRoot(scaffold)) { |
| scaffold._updateSnackBar(); |
| scaffold._updateMaterialBanner(); |
| } |
| } |
| } |
| |
| // Nested Scaffolds are handled by the ScaffoldMessenger by only presenting a |
| // MaterialBanner or SnackBar in the root Scaffold of the nested set. |
| bool _isRoot(ScaffoldState scaffold) { |
| final ScaffoldState? parent = scaffold.context.findAncestorStateOfType<ScaffoldState>(); |
| return parent == null || !_scaffolds.contains(parent); |
| } |
| |
| // SNACKBAR API |
| |
| /// Shows a [SnackBar] across all registered [Scaffold]s. Scaffolds register |
| /// to receive snack bars from their closest [ScaffoldMessenger] ancestor. |
| /// If there are several registered scaffolds the snack bar is shown |
| /// simultaneously on all of them. |
| /// |
| /// A scaffold can show at most one snack bar at a time. If this function is |
| /// called while another snack bar is already visible, the given snack bar |
| /// will be added to a queue and displayed after the earlier snack bars have |
| /// closed. |
| /// |
| /// To control how long a [SnackBar] remains visible, use [SnackBar.duration]. |
| /// |
| /// To remove the [SnackBar] with an exit animation, use [hideCurrentSnackBar] |
| /// or call [ScaffoldFeatureController.close] on the returned |
| /// [ScaffoldFeatureController]. To remove a [SnackBar] suddenly (without an |
| /// animation), use [removeCurrentSnackBar]. |
| /// |
| /// See [ScaffoldMessenger.of] for information about how to obtain the |
| /// [ScaffoldMessengerState]. |
| /// |
| /// {@tool dartpad} |
| /// Here is an example of showing a [SnackBar] when the user presses a button. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger_state.show_snack_bar.0.dart ** |
| /// {@end-tool} |
| /// |
| /// ## Relative positioning of floating SnackBars |
| /// |
| /// A [SnackBar] with [SnackBar.behavior] set to [SnackBarBehavior.floating] is |
| /// positioned above the widgets provided to [Scaffold.floatingActionButton], |
| /// [Scaffold.persistentFooterButtons], and [Scaffold.bottomNavigationBar]. |
| /// If some or all of these widgets take up enough space such that the SnackBar |
| /// would not be visible when positioned above them, an error will be thrown. |
| /// In this case, consider constraining the size of these widgets to allow room for |
| /// the SnackBar to be visible. |
| /// |
| /// {@tool dartpad} |
| /// Here is an example showing how to display a [SnackBar] with [showSnackBar] |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger_state.show_snack_bar.0.dart ** |
| /// {@end-tool} |
| /// |
| /// {@tool dartpad} |
| /// Here is an example showing that a floating [SnackBar] appears above [Scaffold.floatingActionButton]. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger_state.show_snack_bar.1.dart ** |
| /// {@end-tool} |
| /// |
| ScaffoldFeatureController<SnackBar, SnackBarClosedReason> showSnackBar(SnackBar snackBar) { |
| assert( |
| _scaffolds.isNotEmpty, |
| 'ScaffoldMessenger.showSnackBar was called, but there are currently no ' |
| 'descendant Scaffolds to present to.', |
| ); |
| _snackBarController ??= SnackBar.createAnimationController(vsync: this) |
| ..addStatusListener(_handleSnackBarStatusChanged); |
| if (_snackBars.isEmpty) { |
| assert(_snackBarController!.isDismissed); |
| _snackBarController!.forward(); |
| } |
| late ScaffoldFeatureController<SnackBar, SnackBarClosedReason> controller; |
| controller = ScaffoldFeatureController<SnackBar, SnackBarClosedReason>._( |
| // We provide a fallback key so that if back-to-back snackbars happen to |
| // match in structure, material ink splashes and highlights don't survive |
| // from one to the next. |
| snackBar.withAnimation(_snackBarController!, fallbackKey: UniqueKey()), |
| Completer<SnackBarClosedReason>(), |
| () { |
| assert(_snackBars.first == controller); |
| hideCurrentSnackBar(); |
| }, |
| null, // SnackBar doesn't use a builder function so setState() wouldn't rebuild it |
| ); |
| try { |
| setState(() { |
| _snackBars.addLast(controller); |
| }); |
| _updateScaffolds(); |
| } catch (exception) { |
| assert (() { |
| if (exception is FlutterError) { |
| final String summary = exception.diagnostics.first.toDescription(); |
| if (summary == 'setState() or markNeedsBuild() called during build.') { |
| final List<DiagnosticsNode> information = <DiagnosticsNode>[ |
| ErrorSummary('The showSnackBar() method cannot be called during build.'), |
| ErrorDescription( |
| 'The showSnackBar() method was called during build, which is ' |
| 'prohibited as showing snack bars requires updating state. Updating ' |
| 'state is not possible during build.', |
| ), |
| ErrorHint( |
| 'Instead of calling showSnackBar() during build, call it directly ' |
| 'in your on tap (and related) callbacks. If you need to immediately ' |
| 'show a snack bar, make the call in initState() or ' |
| 'didChangeDependencies() instead. Otherwise, you can also schedule a ' |
| 'post-frame callback using SchedulerBinding.addPostFrameCallback to ' |
| 'show the snack bar after the current frame.', |
| ), |
| context.describeOwnershipChain( |
| 'The ownership chain for the particular ScaffoldMessenger is', |
| ), |
| ]; |
| throw FlutterError.fromParts(information); |
| } |
| } |
| return true; |
| }()); |
| rethrow; |
| } |
| |
| return controller; |
| } |
| |
| void _handleSnackBarStatusChanged(AnimationStatus status) { |
| switch (status) { |
| case AnimationStatus.dismissed: |
| assert(_snackBars.isNotEmpty); |
| setState(() { |
| _snackBars.removeFirst(); |
| }); |
| _updateScaffolds(); |
| if (_snackBars.isNotEmpty) { |
| _snackBarController!.forward(); |
| } |
| case AnimationStatus.completed: |
| setState(() { |
| assert(_snackBarTimer == null); |
| // build will create a new timer if necessary to dismiss the snackBar. |
| }); |
| _updateScaffolds(); |
| case AnimationStatus.forward: |
| break; |
| case AnimationStatus.reverse: |
| break; |
| } |
| } |
| |
| /// Removes the current [SnackBar] (if any) immediately from registered |
| /// [Scaffold]s. |
| /// |
| /// The removed snack bar does not run its normal exit animation. If there are |
| /// any queued snack bars, they begin their entrance animation immediately. |
| void removeCurrentSnackBar({ SnackBarClosedReason reason = SnackBarClosedReason.remove }) { |
| if (_snackBars.isEmpty) { |
| return; |
| } |
| final Completer<SnackBarClosedReason> completer = _snackBars.first._completer; |
| if (!completer.isCompleted) { |
| completer.complete(reason); |
| } |
| _snackBarTimer?.cancel(); |
| _snackBarTimer = null; |
| // This will trigger the animation's status callback. |
| _snackBarController!.value = 0.0; |
| } |
| |
| /// Removes the current [SnackBar] by running its normal exit animation. |
| /// |
| /// The closed completer is called after the animation is complete. |
| void hideCurrentSnackBar({ SnackBarClosedReason reason = SnackBarClosedReason.hide }) { |
| if (_snackBars.isEmpty || _snackBarController!.status == AnimationStatus.dismissed) { |
| return; |
| } |
| final Completer<SnackBarClosedReason> completer = _snackBars.first._completer; |
| if (_accessibleNavigation!) { |
| _snackBarController!.value = 0.0; |
| completer.complete(reason); |
| } else { |
| _snackBarController!.reverse().then<void>((void value) { |
| assert(mounted); |
| if (!completer.isCompleted) { |
| completer.complete(reason); |
| } |
| }); |
| } |
| _snackBarTimer?.cancel(); |
| _snackBarTimer = null; |
| } |
| |
| /// Removes all the snackBars currently in queue by clearing the queue |
| /// and running normal exit animation on the current snackBar. |
| void clearSnackBars() { |
| if (_snackBars.isEmpty || _snackBarController!.status == AnimationStatus.dismissed) { |
| return; |
| } |
| final ScaffoldFeatureController<SnackBar, SnackBarClosedReason> currentSnackbar = _snackBars.first; |
| _snackBars.clear(); |
| _snackBars.add(currentSnackbar); |
| hideCurrentSnackBar(); |
| } |
| |
| // MATERIAL BANNER API |
| |
| /// Shows a [MaterialBanner] across all registered [Scaffold]s. Scaffolds register |
| /// to receive material banners from their closest [ScaffoldMessenger] ancestor. |
| /// If there are several registered scaffolds the material banner is shown |
| /// simultaneously on all of them. |
| /// |
| /// A scaffold can show at most one material banner at a time. If this function is |
| /// called while another material banner is already visible, the given material banner |
| /// will be added to a queue and displayed after the earlier material banners have |
| /// closed. |
| /// |
| /// To remove the [MaterialBanner] with an exit animation, use [hideCurrentMaterialBanner] |
| /// or call [ScaffoldFeatureController.close] on the returned |
| /// [ScaffoldFeatureController]. To remove a [MaterialBanner] suddenly (without an |
| /// animation), use [removeCurrentMaterialBanner]. |
| /// |
| /// See [ScaffoldMessenger.of] for information about how to obtain the |
| /// [ScaffoldMessengerState]. |
| /// |
| /// {@tool dartpad} |
| /// Here is an example of showing a [MaterialBanner] when the user presses a button. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_messenger_state.show_material_banner.0.dart ** |
| /// {@end-tool} |
| ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason> showMaterialBanner(MaterialBanner materialBanner) { |
| assert( |
| _scaffolds.isNotEmpty, |
| 'ScaffoldMessenger.showMaterialBanner was called, but there are currently no ' |
| 'descendant Scaffolds to present to.', |
| ); |
| _materialBannerController ??= MaterialBanner.createAnimationController(vsync: this) |
| ..addStatusListener(_handleMaterialBannerStatusChanged); |
| if (_materialBanners.isEmpty) { |
| assert(_materialBannerController!.isDismissed); |
| _materialBannerController!.forward(); |
| } |
| late ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason> controller; |
| controller = ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason>._( |
| // We provide a fallback key so that if back-to-back material banners happen to |
| // match in structure, material ink splashes and highlights don't survive |
| // from one to the next. |
| materialBanner.withAnimation(_materialBannerController!, fallbackKey: UniqueKey()), |
| Completer<MaterialBannerClosedReason>(), |
| () { |
| assert(_materialBanners.first == controller); |
| hideCurrentMaterialBanner(); |
| }, |
| null, // MaterialBanner doesn't use a builder function so setState() wouldn't rebuild it |
| ); |
| setState(() { |
| _materialBanners.addLast(controller); |
| }); |
| _updateScaffolds(); |
| return controller; |
| } |
| |
| void _handleMaterialBannerStatusChanged(AnimationStatus status) { |
| switch (status) { |
| case AnimationStatus.dismissed: |
| assert(_materialBanners.isNotEmpty); |
| setState(() { |
| _materialBanners.removeFirst(); |
| }); |
| _updateScaffolds(); |
| if (_materialBanners.isNotEmpty) { |
| _materialBannerController!.forward(); |
| } |
| case AnimationStatus.completed: |
| _updateScaffolds(); |
| case AnimationStatus.forward: |
| break; |
| case AnimationStatus.reverse: |
| break; |
| } |
| } |
| |
| /// Removes the current [MaterialBanner] (if any) immediately from registered |
| /// [Scaffold]s. |
| /// |
| /// The removed material banner does not run its normal exit animation. If there are |
| /// any queued material banners, they begin their entrance animation immediately. |
| void removeCurrentMaterialBanner({ MaterialBannerClosedReason reason = MaterialBannerClosedReason.remove }) { |
| if (_materialBanners.isEmpty) { |
| return; |
| } |
| final Completer<MaterialBannerClosedReason> completer = _materialBanners.first._completer; |
| if (!completer.isCompleted) { |
| completer.complete(reason); |
| } |
| |
| // This will trigger the animation's status callback. |
| _materialBannerController!.value = 0.0; |
| } |
| |
| /// Removes the current [MaterialBanner] by running its normal exit animation. |
| /// |
| /// The closed completer is called after the animation is complete. |
| void hideCurrentMaterialBanner({ MaterialBannerClosedReason reason = MaterialBannerClosedReason.hide }) { |
| if (_materialBanners.isEmpty || _materialBannerController!.status == AnimationStatus.dismissed) { |
| return; |
| } |
| final Completer<MaterialBannerClosedReason> completer = _materialBanners.first._completer; |
| if (_accessibleNavigation!) { |
| _materialBannerController!.value = 0.0; |
| completer.complete(reason); |
| } else { |
| _materialBannerController!.reverse().then<void>((void value) { |
| assert(mounted); |
| if (!completer.isCompleted) { |
| completer.complete(reason); |
| } |
| }); |
| } |
| } |
| |
| /// Removes all the [MaterialBanner]s currently in queue by clearing the queue |
| /// and running normal exit animation on the current [MaterialBanner]. |
| void clearMaterialBanners() { |
| if (_materialBanners.isEmpty || _materialBannerController!.status == AnimationStatus.dismissed) { |
| return; |
| } |
| final ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason> currentMaterialBanner = _materialBanners.first; |
| _materialBanners.clear(); |
| _materialBanners.add(currentMaterialBanner); |
| hideCurrentMaterialBanner(); |
| } |
| |
| @override |
| Widget build(BuildContext context) { |
| assert(debugCheckHasMediaQuery(context)); |
| _accessibleNavigation = MediaQuery.accessibleNavigationOf(context); |
| |
| if (_snackBars.isNotEmpty) { |
| final ModalRoute<dynamic>? route = ModalRoute.of(context); |
| if (route == null || route.isCurrent) { |
| if (_snackBarController!.isCompleted && _snackBarTimer == null) { |
| final SnackBar snackBar = _snackBars.first._widget; |
| _snackBarTimer = Timer(snackBar.duration, () { |
| assert( |
| _snackBarController!.status == AnimationStatus.forward || |
| _snackBarController!.status == AnimationStatus.completed, |
| ); |
| // Look up MediaQuery again in case the setting changed. |
| if (snackBar.action != null && MediaQuery.accessibleNavigationOf(context)) { |
| return; |
| } |
| hideCurrentSnackBar(reason: SnackBarClosedReason.timeout); |
| }); |
| } |
| } |
| } |
| |
| return _ScaffoldMessengerScope( |
| scaffoldMessengerState: this, |
| child: widget.child, |
| ); |
| } |
| |
| @override |
| void dispose() { |
| _snackBarController?.dispose(); |
| _snackBarTimer?.cancel(); |
| _snackBarTimer = null; |
| super.dispose(); |
| } |
| } |
| |
| class _ScaffoldMessengerScope extends InheritedWidget { |
| const _ScaffoldMessengerScope({ |
| required super.child, |
| required ScaffoldMessengerState scaffoldMessengerState, |
| }) : _scaffoldMessengerState = scaffoldMessengerState; |
| |
| final ScaffoldMessengerState _scaffoldMessengerState; |
| |
| @override |
| bool updateShouldNotify(_ScaffoldMessengerScope old) => _scaffoldMessengerState != old._scaffoldMessengerState; |
| } |
| |
| /// The geometry of the [Scaffold] after all its contents have been laid out |
| /// except the [FloatingActionButton]. |
| /// |
| /// The [Scaffold] passes this pre-layout geometry to its |
| /// [FloatingActionButtonLocation], which produces an [Offset] that the |
| /// [Scaffold] uses to position the [FloatingActionButton]. |
| /// |
| /// For a description of the [Scaffold]'s geometry after it has |
| /// finished laying out, see the [ScaffoldGeometry]. |
| @immutable |
| class ScaffoldPrelayoutGeometry { |
| /// Abstract const constructor. This constructor enables subclasses to provide |
| /// const constructors so that they can be used in const expressions. |
| const ScaffoldPrelayoutGeometry({ |
| required this.bottomSheetSize, |
| required this.contentBottom, |
| required this.contentTop, |
| required this.floatingActionButtonSize, |
| required this.minInsets, |
| required this.minViewPadding, |
| required this.scaffoldSize, |
| required this.snackBarSize, |
| required this.materialBannerSize, |
| required this.textDirection, |
| }); |
| |
| /// The [Size] of [Scaffold.floatingActionButton]. |
| /// |
| /// If [Scaffold.floatingActionButton] is null, this will be [Size.zero]. |
| final Size floatingActionButtonSize; |
| |
| /// The [Size] of the [Scaffold]'s [BottomSheet]. |
| /// |
| /// If the [Scaffold] is not currently showing a [BottomSheet], |
| /// this will be [Size.zero]. |
| final Size bottomSheetSize; |
| |
| /// The vertical distance from the Scaffold's origin to the bottom of |
| /// [Scaffold.body]. |
| /// |
| /// This is useful in a [FloatingActionButtonLocation] designed to |
| /// place the [FloatingActionButton] at the bottom of the screen, while |
| /// keeping it above the [BottomSheet], the [Scaffold.bottomNavigationBar], |
| /// or the keyboard. |
| /// |
| /// The [Scaffold.body] is laid out with respect to [minInsets] already. This |
| /// means that a [FloatingActionButtonLocation] does not need to factor in |
| /// [EdgeInsets.bottom] of [minInsets] when aligning a [FloatingActionButton] |
| /// to [contentBottom]. |
| final double contentBottom; |
| |
| /// The vertical distance from the [Scaffold]'s origin to the top of |
| /// [Scaffold.body]. |
| /// |
| /// This is useful in a [FloatingActionButtonLocation] designed to |
| /// place the [FloatingActionButton] at the top of the screen, while |
| /// keeping it below the [Scaffold.appBar]. |
| /// |
| /// The [Scaffold.body] is laid out with respect to [minInsets] already. This |
| /// means that a [FloatingActionButtonLocation] does not need to factor in |
| /// [EdgeInsets.top] of [minInsets] when aligning a [FloatingActionButton] to |
| /// [contentTop]. |
| final double contentTop; |
| |
| /// The minimum padding to inset the [FloatingActionButton] by for it |
| /// to remain visible. |
| /// |
| /// This value is the result of calling [MediaQueryData.padding] in the |
| /// [Scaffold]'s [BuildContext], |
| /// and is useful for insetting the [FloatingActionButton] to avoid features like |
| /// the system status bar or the keyboard. |
| /// |
| /// If [Scaffold.resizeToAvoidBottomInset] is set to false, |
| /// [EdgeInsets.bottom] of [minInsets] will be 0.0. |
| final EdgeInsets minInsets; |
| |
| /// The minimum padding to inset interactive elements to be within a safe, |
| /// un-obscured space. |
| /// |
| /// This value reflects the [MediaQueryData.viewPadding] of the [Scaffold]'s |
| /// [BuildContext] when [Scaffold.resizeToAvoidBottomInset] is false or and |
| /// the [MediaQueryData.viewInsets] > 0.0. This helps distinguish between |
| /// different types of obstructions on the screen, such as software keyboards |
| /// and physical device notches. |
| final EdgeInsets minViewPadding; |
| |
| /// The [Size] of the whole [Scaffold]. |
| /// |
| /// If the [Size] of the [Scaffold]'s contents is modified by values such as |
| /// [Scaffold.resizeToAvoidBottomInset] or the keyboard opening, then the |
| /// [scaffoldSize] will not reflect those changes. |
| /// |
| /// This means that [FloatingActionButtonLocation]s designed to reposition |
| /// the [FloatingActionButton] based on events such as the keyboard popping |
| /// up should use [minInsets] to make sure that the [FloatingActionButton] is |
| /// inset by enough to remain visible. |
| /// |
| /// See [minInsets] and [MediaQueryData.padding] for more information on the |
| /// appropriate insets to apply. |
| final Size scaffoldSize; |
| |
| /// The [Size] of the [Scaffold]'s [SnackBar]. |
| /// |
| /// If the [Scaffold] is not showing a [SnackBar], this will be [Size.zero]. |
| final Size snackBarSize; |
| |
| /// The [Size] of the [Scaffold]'s [MaterialBanner]. |
| /// |
| /// If the [Scaffold] is not showing a [MaterialBanner], this will be [Size.zero]. |
| final Size materialBannerSize; |
| |
| /// The [TextDirection] of the [Scaffold]'s [BuildContext]. |
| final TextDirection textDirection; |
| } |
| |
| /// A snapshot of a transition between two [FloatingActionButtonLocation]s. |
| /// |
| /// [ScaffoldState] uses this to seamlessly change transition animations |
| /// when a running [FloatingActionButtonLocation] transition is interrupted by a new transition. |
| @immutable |
| class _TransitionSnapshotFabLocation extends FloatingActionButtonLocation { |
| |
| const _TransitionSnapshotFabLocation(this.begin, this.end, this.animator, this.progress); |
| |
| final FloatingActionButtonLocation begin; |
| final FloatingActionButtonLocation end; |
| final FloatingActionButtonAnimator animator; |
| final double progress; |
| |
| @override |
| Offset getOffset(ScaffoldPrelayoutGeometry scaffoldGeometry) { |
| return animator.getOffset( |
| begin: begin.getOffset(scaffoldGeometry), |
| end: end.getOffset(scaffoldGeometry), |
| progress: progress, |
| ); |
| } |
| |
| @override |
| String toString() { |
| return '${objectRuntimeType(this, '_TransitionSnapshotFabLocation')}(begin: $begin, end: $end, progress: $progress)'; |
| } |
| } |
| |
| /// Geometry information for [Scaffold] components after layout is finished. |
| /// |
| /// To get a [ValueNotifier] for the scaffold geometry of a given |
| /// [BuildContext], use [Scaffold.geometryOf]. |
| /// |
| /// The ScaffoldGeometry is only available during the paint phase, because |
| /// its value is computed during the animation and layout phases prior to painting. |
| /// |
| /// For an example of using the [ScaffoldGeometry], see the [BottomAppBar], |
| /// which uses the [ScaffoldGeometry] to paint a notch around the |
| /// [FloatingActionButton]. |
| /// |
| /// For information about the [Scaffold]'s geometry that is used while laying |
| /// out the [FloatingActionButton], see [ScaffoldPrelayoutGeometry]. |
| @immutable |
| class ScaffoldGeometry { |
| /// Create an object that describes the geometry of a [Scaffold]. |
| const ScaffoldGeometry({ |
| this.bottomNavigationBarTop, |
| this.floatingActionButtonArea, |
| }); |
| |
| /// The distance from the [Scaffold]'s top edge to the top edge of the |
| /// rectangle in which the [Scaffold.bottomNavigationBar] bar is laid out. |
| /// |
| /// Null if [Scaffold.bottomNavigationBar] is null. |
| final double? bottomNavigationBarTop; |
| |
| /// The [Scaffold.floatingActionButton]'s bounding rectangle. |
| /// |
| /// This is null when there is no floating action button showing. |
| final Rect? floatingActionButtonArea; |
| |
| ScaffoldGeometry _scaleFloatingActionButton(double scaleFactor) { |
| if (scaleFactor == 1.0) { |
| return this; |
| } |
| |
| if (scaleFactor == 0.0) { |
| return ScaffoldGeometry( |
| bottomNavigationBarTop: bottomNavigationBarTop, |
| ); |
| } |
| |
| final Rect scaledButton = Rect.lerp( |
| floatingActionButtonArea!.center & Size.zero, |
| floatingActionButtonArea, |
| scaleFactor, |
| )!; |
| return copyWith(floatingActionButtonArea: scaledButton); |
| } |
| |
| /// Creates a copy of this [ScaffoldGeometry] but with the given fields replaced with |
| /// the new values. |
| ScaffoldGeometry copyWith({ |
| double? bottomNavigationBarTop, |
| Rect? floatingActionButtonArea, |
| }) { |
| return ScaffoldGeometry( |
| bottomNavigationBarTop: bottomNavigationBarTop ?? this.bottomNavigationBarTop, |
| floatingActionButtonArea: floatingActionButtonArea ?? this.floatingActionButtonArea, |
| ); |
| } |
| } |
| |
| class _ScaffoldGeometryNotifier extends ChangeNotifier implements ValueListenable<ScaffoldGeometry> { |
| _ScaffoldGeometryNotifier(this.geometry, this.context); |
| |
| final BuildContext context; |
| double? floatingActionButtonScale; |
| ScaffoldGeometry geometry; |
| |
| @override |
| ScaffoldGeometry get value { |
| assert(() { |
| final RenderObject? renderObject = context.findRenderObject(); |
| if (renderObject == null || !renderObject.owner!.debugDoingPaint) { |
| throw FlutterError( |
| 'Scaffold.geometryOf() must only be accessed during the paint phase.\n' |
| 'The ScaffoldGeometry is only available during the paint phase, because ' |
| 'its value is computed during the animation and layout phases prior to painting.', |
| ); |
| } |
| return true; |
| }()); |
| return geometry._scaleFloatingActionButton(floatingActionButtonScale!); |
| } |
| |
| void _updateWith({ |
| double? bottomNavigationBarTop, |
| Rect? floatingActionButtonArea, |
| double? floatingActionButtonScale, |
| }) { |
| this.floatingActionButtonScale = floatingActionButtonScale ?? this.floatingActionButtonScale; |
| geometry = geometry.copyWith( |
| bottomNavigationBarTop: bottomNavigationBarTop, |
| floatingActionButtonArea: floatingActionButtonArea, |
| ); |
| notifyListeners(); |
| } |
| } |
| |
| // Used to communicate the height of the Scaffold's bottomNavigationBar and |
| // persistentFooterButtons to the LayoutBuilder which builds the Scaffold's body. |
| // |
| // Scaffold expects a _BodyBoxConstraints to be passed to the _BodyBuilder |
| // widget's LayoutBuilder, see _ScaffoldLayout.performLayout(). The BoxConstraints |
| // methods that construct new BoxConstraints objects, like copyWith() have not |
| // been overridden here because we expect the _BodyBoxConstraintsObject to be |
| // passed along unmodified to the LayoutBuilder. If that changes in the future |
| // then _BodyBuilder will assert. |
| class _BodyBoxConstraints extends BoxConstraints { |
| const _BodyBoxConstraints({ |
| super.maxWidth, |
| super.maxHeight, |
| required this.bottomWidgetsHeight, |
| required this.appBarHeight, |
| required this.materialBannerHeight, |
| }) : assert(bottomWidgetsHeight >= 0), |
| assert(appBarHeight >= 0), |
| assert(materialBannerHeight >= 0); |
| |
| final double bottomWidgetsHeight; |
| final double appBarHeight; |
| final double materialBannerHeight; |
| |
| // RenderObject.layout() will only short-circuit its call to its performLayout |
| // method if the new layout constraints are not == to the current constraints. |
| // If the height of the bottom widgets has changed, even though the constraints' |
| // min and max values have not, we still want performLayout to happen. |
| @override |
| bool operator ==(Object other) { |
| if (super != other) { |
| return false; |
| } |
| return other is _BodyBoxConstraints |
| && other.materialBannerHeight == materialBannerHeight |
| && other.bottomWidgetsHeight == bottomWidgetsHeight |
| && other.appBarHeight == appBarHeight; |
| } |
| |
| @override |
| int get hashCode => Object.hash(super.hashCode, materialBannerHeight, bottomWidgetsHeight, appBarHeight); |
| } |
| |
| // Used when Scaffold.extendBody is true to wrap the scaffold's body in a MediaQuery |
| // whose padding accounts for the height of the bottomNavigationBar and/or the |
| // persistentFooterButtons. |
| // |
| // The bottom widgets' height is passed along via the _BodyBoxConstraints parameter. |
| // The constraints parameter is constructed in_ScaffoldLayout.performLayout(). |
| class _BodyBuilder extends StatelessWidget { |
| const _BodyBuilder({ |
| required this.extendBody, |
| required this.extendBodyBehindAppBar, |
| required this.body, |
| }); |
| |
| final Widget body; |
| final bool extendBody; |
| final bool extendBodyBehindAppBar; |
| |
| @override |
| Widget build(BuildContext context) { |
| if (!extendBody && !extendBodyBehindAppBar) { |
| return body; |
| } |
| |
| return LayoutBuilder( |
| builder: (BuildContext context, BoxConstraints constraints) { |
| final _BodyBoxConstraints bodyConstraints = constraints as _BodyBoxConstraints; |
| final MediaQueryData metrics = MediaQuery.of(context); |
| |
| final double bottom = extendBody |
| ? math.max(metrics.padding.bottom, bodyConstraints.bottomWidgetsHeight) |
| : metrics.padding.bottom; |
| |
| final double top = extendBodyBehindAppBar |
| ? math.max(metrics.padding.top, |
| bodyConstraints.appBarHeight + bodyConstraints.materialBannerHeight) |
| : metrics.padding.top; |
| |
| return MediaQuery( |
| data: metrics.copyWith( |
| padding: metrics.padding.copyWith( |
| top: top, |
| bottom: bottom, |
| ), |
| ), |
| child: body, |
| ); |
| }, |
| ); |
| } |
| } |
| |
| class _ScaffoldLayout extends MultiChildLayoutDelegate { |
| _ScaffoldLayout({ |
| required this.minInsets, |
| required this.minViewPadding, |
| required this.textDirection, |
| required this.geometryNotifier, |
| // for floating action button |
| required this.previousFloatingActionButtonLocation, |
| required this.currentFloatingActionButtonLocation, |
| required this.floatingActionButtonMoveAnimationProgress, |
| required this.floatingActionButtonMotionAnimator, |
| required this.isSnackBarFloating, |
| required this.snackBarWidth, |
| required this.extendBody, |
| required this.extendBodyBehindAppBar, |
| required this.extendBodyBehindMaterialBanner, |
| }); |
| |
| final bool extendBody; |
| final bool extendBodyBehindAppBar; |
| final EdgeInsets minInsets; |
| final EdgeInsets minViewPadding; |
| final TextDirection textDirection; |
| final _ScaffoldGeometryNotifier geometryNotifier; |
| |
| final FloatingActionButtonLocation previousFloatingActionButtonLocation; |
| final FloatingActionButtonLocation currentFloatingActionButtonLocation; |
| final double floatingActionButtonMoveAnimationProgress; |
| final FloatingActionButtonAnimator floatingActionButtonMotionAnimator; |
| |
| final bool isSnackBarFloating; |
| final double? snackBarWidth; |
| |
| final bool extendBodyBehindMaterialBanner; |
| |
| @override |
| void performLayout(Size size) { |
| final BoxConstraints looseConstraints = BoxConstraints.loose(size); |
| |
| // This part of the layout has the same effect as putting the app bar and |
| // body in a column and making the body flexible. What's different is that |
| // in this case the app bar appears _after_ the body in the stacking order, |
| // so the app bar's shadow is drawn on top of the body. |
| |
| final BoxConstraints fullWidthConstraints = looseConstraints.tighten(width: size.width); |
| final double bottom = size.height; |
| double contentTop = 0.0; |
| double bottomWidgetsHeight = 0.0; |
| double appBarHeight = 0.0; |
| |
| if (hasChild(_ScaffoldSlot.appBar)) { |
| appBarHeight = layoutChild(_ScaffoldSlot.appBar, fullWidthConstraints).height; |
| contentTop = extendBodyBehindAppBar ? 0.0 : appBarHeight; |
| positionChild(_ScaffoldSlot.appBar, Offset.zero); |
| } |
| |
| double? bottomNavigationBarTop; |
| if (hasChild(_ScaffoldSlot.bottomNavigationBar)) { |
| final double bottomNavigationBarHeight = layoutChild(_ScaffoldSlot.bottomNavigationBar, fullWidthConstraints).height; |
| bottomWidgetsHeight += bottomNavigationBarHeight; |
| bottomNavigationBarTop = math.max(0.0, bottom - bottomWidgetsHeight); |
| positionChild(_ScaffoldSlot.bottomNavigationBar, Offset(0.0, bottomNavigationBarTop)); |
| } |
| |
| if (hasChild(_ScaffoldSlot.persistentFooter)) { |
| final BoxConstraints footerConstraints = BoxConstraints( |
| maxWidth: fullWidthConstraints.maxWidth, |
| maxHeight: math.max(0.0, bottom - bottomWidgetsHeight - contentTop), |
| ); |
| final double persistentFooterHeight = layoutChild(_ScaffoldSlot.persistentFooter, footerConstraints).height; |
| bottomWidgetsHeight += persistentFooterHeight; |
| positionChild(_ScaffoldSlot.persistentFooter, Offset(0.0, math.max(0.0, bottom - bottomWidgetsHeight))); |
| } |
| |
| Size materialBannerSize = Size.zero; |
| if (hasChild(_ScaffoldSlot.materialBanner)) { |
| materialBannerSize = layoutChild(_ScaffoldSlot.materialBanner, fullWidthConstraints); |
| positionChild(_ScaffoldSlot.materialBanner, Offset(0.0, appBarHeight)); |
| |
| // Push content down only if elevation is 0. |
| if (!extendBodyBehindMaterialBanner) { |
| contentTop += materialBannerSize.height; |
| } |
| } |
| |
| // Set the content bottom to account for the greater of the height of any |
| // bottom-anchored material widgets or of the keyboard or other |
| // bottom-anchored system UI. |
| final double contentBottom = math.max(0.0, bottom - math.max(minInsets.bottom, bottomWidgetsHeight)); |
| |
| if (hasChild(_ScaffoldSlot.body)) { |
| double bodyMaxHeight = math.max(0.0, contentBottom - contentTop); |
| |
| if (extendBody) { |
| bodyMaxHeight += bottomWidgetsHeight; |
| bodyMaxHeight = clampDouble(bodyMaxHeight, 0.0, looseConstraints.maxHeight - contentTop); |
| assert(bodyMaxHeight <= math.max(0.0, looseConstraints.maxHeight - contentTop)); |
| } |
| |
| final BoxConstraints bodyConstraints = _BodyBoxConstraints( |
| maxWidth: fullWidthConstraints.maxWidth, |
| maxHeight: bodyMaxHeight, |
| materialBannerHeight: materialBannerSize.height, |
| bottomWidgetsHeight: extendBody ? bottomWidgetsHeight : 0.0, |
| appBarHeight: appBarHeight, |
| ); |
| layoutChild(_ScaffoldSlot.body, bodyConstraints); |
| positionChild(_ScaffoldSlot.body, Offset(0.0, contentTop)); |
| } |
| |
| // The BottomSheet and the SnackBar are anchored to the bottom of the parent, |
| // they're as wide as the parent and are given their intrinsic height. The |
| // only difference is that SnackBar appears on the top side of the |
| // BottomNavigationBar while the BottomSheet is stacked on top of it. |
| // |
| // If all three elements are present then either the center of the FAB straddles |
| // the top edge of the BottomSheet or the bottom of the FAB is |
| // kFloatingActionButtonMargin above the SnackBar, whichever puts the FAB |
| // the farthest above the bottom of the parent. If only the FAB is has a |
| // non-zero height then it's inset from the parent's right and bottom edges |
| // by kFloatingActionButtonMargin. |
| |
| Size bottomSheetSize = Size.zero; |
| Size snackBarSize = Size.zero; |
| if (hasChild(_ScaffoldSlot.bodyScrim)) { |
| final BoxConstraints bottomSheetScrimConstraints = BoxConstraints( |
| maxWidth: fullWidthConstraints.maxWidth, |
| maxHeight: contentBottom, |
| ); |
| layoutChild(_ScaffoldSlot.bodyScrim, bottomSheetScrimConstraints); |
| positionChild(_ScaffoldSlot.bodyScrim, Offset.zero); |
| } |
| |
| // Set the size of the SnackBar early if the behavior is fixed so |
| // the FAB can be positioned correctly. |
| if (hasChild(_ScaffoldSlot.snackBar) && !isSnackBarFloating) { |
| snackBarSize = layoutChild(_ScaffoldSlot.snackBar, fullWidthConstraints); |
| } |
| |
| if (hasChild(_ScaffoldSlot.bottomSheet)) { |
| final BoxConstraints bottomSheetConstraints = BoxConstraints( |
| maxWidth: fullWidthConstraints.maxWidth, |
| maxHeight: math.max(0.0, contentBottom - contentTop), |
| ); |
| bottomSheetSize = layoutChild(_ScaffoldSlot.bottomSheet, bottomSheetConstraints); |
| positionChild(_ScaffoldSlot.bottomSheet, Offset((size.width - bottomSheetSize.width) / 2.0, contentBottom - bottomSheetSize.height)); |
| } |
| |
| late Rect floatingActionButtonRect; |
| if (hasChild(_ScaffoldSlot.floatingActionButton)) { |
| final Size fabSize = layoutChild(_ScaffoldSlot.floatingActionButton, looseConstraints); |
| |
| // To account for the FAB position being changed, we'll animate between |
| // the old and new positions. |
| final ScaffoldPrelayoutGeometry currentGeometry = ScaffoldPrelayoutGeometry( |
| bottomSheetSize: bottomSheetSize, |
| contentBottom: contentBottom, |
| /// [appBarHeight] should be used instead of [contentTop] because |
| /// ScaffoldPrelayoutGeometry.contentTop must not be affected by [extendBodyBehindAppBar]. |
| contentTop: appBarHeight, |
| floatingActionButtonSize: fabSize, |
| minInsets: minInsets, |
| scaffoldSize: size, |
| snackBarSize: snackBarSize, |
| materialBannerSize: materialBannerSize, |
| textDirection: textDirection, |
| minViewPadding: minViewPadding, |
| ); |
| final Offset currentFabOffset = currentFloatingActionButtonLocation.getOffset(currentGeometry); |
| final Offset previousFabOffset = previousFloatingActionButtonLocation.getOffset(currentGeometry); |
| final Offset fabOffset = floatingActionButtonMotionAnimator.getOffset( |
| begin: previousFabOffset, |
| end: currentFabOffset, |
| progress: floatingActionButtonMoveAnimationProgress, |
| ); |
| positionChild(_ScaffoldSlot.floatingActionButton, fabOffset); |
| floatingActionButtonRect = fabOffset & fabSize; |
| } |
| |
| if (hasChild(_ScaffoldSlot.snackBar)) { |
| final bool hasCustomWidth = snackBarWidth != null && snackBarWidth! < size.width; |
| if (snackBarSize == Size.zero) { |
| snackBarSize = layoutChild( |
| _ScaffoldSlot.snackBar, |
| hasCustomWidth ? looseConstraints : fullWidthConstraints, |
| ); |
| } |
| |
| final double snackBarYOffsetBase; |
| final bool showAboveFab = switch (currentFloatingActionButtonLocation) { |
| FloatingActionButtonLocation.startTop |
| || FloatingActionButtonLocation.centerTop |
| || FloatingActionButtonLocation.endTop |
| || FloatingActionButtonLocation.miniStartTop |
| || FloatingActionButtonLocation.miniCenterTop |
| || FloatingActionButtonLocation.miniEndTop => false, |
| FloatingActionButtonLocation.startDocked |
| || FloatingActionButtonLocation.startFloat |
| || FloatingActionButtonLocation.centerDocked |
| || FloatingActionButtonLocation.centerFloat |
| || FloatingActionButtonLocation.endContained |
| || FloatingActionButtonLocation.endDocked |
| || FloatingActionButtonLocation.endFloat |
| || FloatingActionButtonLocation.miniStartDocked |
| || FloatingActionButtonLocation.miniStartFloat |
| || FloatingActionButtonLocation.miniCenterDocked |
| || FloatingActionButtonLocation.miniCenterFloat |
| || FloatingActionButtonLocation.miniEndDocked |
| || FloatingActionButtonLocation.miniEndFloat => true, |
| FloatingActionButtonLocation() => true, |
| }; |
| if (floatingActionButtonRect.size != Size.zero && isSnackBarFloating && showAboveFab) { |
| snackBarYOffsetBase = floatingActionButtonRect.top; |
| } else { |
| // SnackBarBehavior.fixed applies a SafeArea automatically. |
| // SnackBarBehavior.floating does not since the positioning is affected |
| // if there is a FloatingActionButton (see condition above). If there is |
| // no FAB, make sure we account for safe space when the SnackBar is |
| // floating. |
| final double safeYOffsetBase = size.height - minViewPadding.bottom; |
| snackBarYOffsetBase = isSnackBarFloating |
| ? math.min(contentBottom, safeYOffsetBase) |
| : contentBottom; |
| } |
| |
| final double xOffset = hasCustomWidth ? (size.width - snackBarWidth!) / 2 : 0.0; |
| positionChild(_ScaffoldSlot.snackBar, Offset(xOffset, snackBarYOffsetBase - snackBarSize.height)); |
| |
| assert((){ |
| // Whether a floating SnackBar has been offset too high. |
| // |
| // To improve the developer experience, this assert is done after the call to positionChild. |
| // if we assert sooner the SnackBar is visible because its defaults position is (0,0) and |
| // it can cause confusion to the user as the error message states that the SnackBar is off screen. |
| if (isSnackBarFloating) { |
| final bool snackBarVisible = (snackBarYOffsetBase - snackBarSize.height) >= 0; |
| if (!snackBarVisible) { |
| throw FlutterError.fromParts(<DiagnosticsNode>[ |
| ErrorSummary('Floating SnackBar presented off screen.'), |
| ErrorDescription( |
| 'A SnackBar with behavior property set to SnackBarBehavior.floating is fully ' |
| 'or partially off screen because some or all the widgets provided to ' |
| 'Scaffold.floatingActionButton, Scaffold.persistentFooterButtons and ' |
| 'Scaffold.bottomNavigationBar take up too much vertical space.\n' |
| ), |
| ErrorHint( |
| 'Consider constraining the size of these widgets to allow room for the SnackBar to be visible.', |
| ), |
| ]); |
| } |
| } |
| return true; |
| }()); |
| } |
| |
| if (hasChild(_ScaffoldSlot.statusBar)) { |
| layoutChild(_ScaffoldSlot.statusBar, fullWidthConstraints.tighten(height: minInsets.top)); |
| positionChild(_ScaffoldSlot.statusBar, Offset.zero); |
| } |
| |
| if (hasChild(_ScaffoldSlot.drawer)) { |
| layoutChild(_ScaffoldSlot.drawer, BoxConstraints.tight(size)); |
| positionChild(_ScaffoldSlot.drawer, Offset.zero); |
| } |
| |
| if (hasChild(_ScaffoldSlot.endDrawer)) { |
| layoutChild(_ScaffoldSlot.endDrawer, BoxConstraints.tight(size)); |
| positionChild(_ScaffoldSlot.endDrawer, Offset.zero); |
| } |
| |
| geometryNotifier._updateWith( |
| bottomNavigationBarTop: bottomNavigationBarTop, |
| floatingActionButtonArea: floatingActionButtonRect, |
| ); |
| } |
| |
| @override |
| bool shouldRelayout(_ScaffoldLayout oldDelegate) { |
| return oldDelegate.minInsets != minInsets |
| || oldDelegate.minViewPadding != minViewPadding |
| || oldDelegate.textDirection != textDirection |
| || oldDelegate.floatingActionButtonMoveAnimationProgress != floatingActionButtonMoveAnimationProgress |
| || oldDelegate.previousFloatingActionButtonLocation != previousFloatingActionButtonLocation |
| || oldDelegate.currentFloatingActionButtonLocation != currentFloatingActionButtonLocation |
| || oldDelegate.extendBody != extendBody |
| || oldDelegate.extendBodyBehindAppBar != extendBodyBehindAppBar; |
| } |
| } |
| |
| /// Handler for scale and rotation animations in the [FloatingActionButton]. |
| /// |
| /// Currently, there are two types of [FloatingActionButton] animations: |
| /// |
| /// * Entrance/Exit animations, which this widget triggers |
| /// when the [FloatingActionButton] is added, updated, or removed. |
| /// * Motion animations, which are triggered by the [Scaffold] |
| /// when its [FloatingActionButtonLocation] is updated. |
| class _FloatingActionButtonTransition extends StatefulWidget { |
| const _FloatingActionButtonTransition({ |
| required this.child, |
| required this.fabMoveAnimation, |
| required this.fabMotionAnimator, |
| required this.geometryNotifier, |
| required this.currentController, |
| }); |
| |
| final Widget? child; |
| final Animation<double> fabMoveAnimation; |
| final FloatingActionButtonAnimator fabMotionAnimator; |
| final _ScaffoldGeometryNotifier geometryNotifier; |
| |
| /// Controls the current child widget.child as it exits. |
| final AnimationController currentController; |
| |
| @override |
| _FloatingActionButtonTransitionState createState() => _FloatingActionButtonTransitionState(); |
| } |
| |
| class _FloatingActionButtonTransitionState extends State<_FloatingActionButtonTransition> with TickerProviderStateMixin { |
| // The animations applied to the Floating Action Button when it is entering or exiting. |
| // Controls the previous widget.child as it exits. |
| late AnimationController _previousController; |
| late Animation<double> _previousScaleAnimation; |
| late Animation<double> _previousRotationAnimation; |
| // The animations to run, considering the widget's fabMoveAnimation and the current/previous entrance/exit animations. |
| late Animation<double> _currentScaleAnimation; |
| late Animation<double> _extendedCurrentScaleAnimation; |
| late Animation<double> _currentRotationAnimation; |
| Widget? _previousChild; |
| |
| @override |
| void initState() { |
| super.initState(); |
| |
| _previousController = AnimationController( |
| duration: kFloatingActionButtonSegue, |
| vsync: this, |
| )..addStatusListener(_handlePreviousAnimationStatusChanged); |
| _updateAnimations(); |
| |
| if (widget.child != null) { |
| // If we start out with a child, have the child appear fully visible instead |
| // of animating in. |
| widget.currentController.value = 1.0; |
| } else { |
| // If we start without a child we update the geometry object with a |
| // floating action button scale of 0, as it is not showing on the screen. |
| _updateGeometryScale(0.0); |
| } |
| } |
| |
| @override |
| void dispose() { |
| _previousController.dispose(); |
| super.dispose(); |
| } |
| |
| @override |
| void didUpdateWidget(_FloatingActionButtonTransition oldWidget) { |
| super.didUpdateWidget(oldWidget); |
| if (oldWidget.fabMotionAnimator != widget.fabMotionAnimator || oldWidget.fabMoveAnimation != widget.fabMoveAnimation) { |
| // Get the right scale and rotation animations to use for this widget. |
| _updateAnimations(); |
| } |
| final bool oldChildIsNull = oldWidget.child == null; |
| final bool newChildIsNull = widget.child == null; |
| if (oldChildIsNull == newChildIsNull && oldWidget.child?.key == widget.child?.key) { |
| return; |
| } |
| if (_previousController.status == AnimationStatus.dismissed) { |
| final double currentValue = widget.currentController.value; |
| if (currentValue == 0.0 || oldWidget.child == null) { |
| // The current child hasn't started its entrance animation yet. We can |
| // just skip directly to the new child's entrance. |
| _previousChild = null; |
| if (widget.child != null) { |
| widget.currentController.forward(); |
| } |
| } else { |
| // Otherwise, we need to copy the state from the current controller to |
| // the previous controller and run an exit animation for the previous |
| // widget before running the entrance animation for the new child. |
| _previousChild = oldWidget.child; |
| _previousController |
| ..value = currentValue |
| ..reverse(); |
| widget.currentController.value = 0.0; |
| } |
| } |
| } |
| |
| static final Animatable<double> _entranceTurnTween = Tween<double>( |
| begin: 1.0 - kFloatingActionButtonTurnInterval, |
| end: 1.0, |
| ).chain(CurveTween(curve: Curves.easeIn)); |
| |
| void _updateAnimations() { |
| // Get the animations for exit and entrance. |
| final CurvedAnimation previousExitScaleAnimation = CurvedAnimation( |
| parent: _previousController, |
| curve: Curves.easeIn, |
| ); |
| final Animation<double> previousExitRotationAnimation = Tween<double>(begin: 1.0, end: 1.0).animate( |
| CurvedAnimation( |
| parent: _previousController, |
| curve: Curves.easeIn, |
| ), |
| ); |
| |
| final CurvedAnimation currentEntranceScaleAnimation = CurvedAnimation( |
| parent: widget.currentController, |
| curve: Curves.easeIn, |
| ); |
| final Animation<double> currentEntranceRotationAnimation = widget.currentController.drive(_entranceTurnTween); |
| |
| // Get the animations for when the FAB is moving. |
| final Animation<double> moveScaleAnimation = widget.fabMotionAnimator.getScaleAnimation(parent: widget.fabMoveAnimation); |
| final Animation<double> moveRotationAnimation = widget.fabMotionAnimator.getRotationAnimation(parent: widget.fabMoveAnimation); |
| |
| // Aggregate the animations. |
| _previousScaleAnimation = AnimationMin<double>(moveScaleAnimation, previousExitScaleAnimation); |
| _currentScaleAnimation = AnimationMin<double>(moveScaleAnimation, currentEntranceScaleAnimation); |
| _extendedCurrentScaleAnimation = _currentScaleAnimation.drive(CurveTween(curve: const Interval(0.0, 0.1))); |
| |
| _previousRotationAnimation = TrainHoppingAnimation(previousExitRotationAnimation, moveRotationAnimation); |
| _currentRotationAnimation = TrainHoppingAnimation(currentEntranceRotationAnimation, moveRotationAnimation); |
| |
| _currentScaleAnimation.addListener(_onProgressChanged); |
| _previousScaleAnimation.addListener(_onProgressChanged); |
| } |
| |
| void _handlePreviousAnimationStatusChanged(AnimationStatus status) { |
| setState(() { |
| if (widget.child != null && status == AnimationStatus.dismissed) { |
| assert(widget.currentController.status == AnimationStatus.dismissed); |
| widget.currentController.forward(); |
| } |
| }); |
| } |
| |
| bool _isExtendedFloatingActionButton(Widget? widget) { |
| return widget is FloatingActionButton |
| && widget.isExtended; |
| } |
| |
| @override |
| Widget build(BuildContext context) { |
| return Stack( |
| alignment: Alignment.centerRight, |
| children: <Widget>[ |
| if (_previousController.status != AnimationStatus.dismissed) |
| if (_isExtendedFloatingActionButton(_previousChild)) |
| FadeTransition( |
| opacity: _previousScaleAnimation, |
| child: _previousChild, |
| ) |
| else |
| ScaleTransition( |
| scale: _previousScaleAnimation, |
| child: RotationTransition( |
| turns: _previousRotationAnimation, |
| child: _previousChild, |
| ), |
| ), |
| if (_isExtendedFloatingActionButton(widget.child)) |
| ScaleTransition( |
| scale: _extendedCurrentScaleAnimation, |
| child: FadeTransition( |
| opacity: _currentScaleAnimation, |
| child: widget.child, |
| ), |
| ) |
| else |
| ScaleTransition( |
| scale: _currentScaleAnimation, |
| child: RotationTransition( |
| turns: _currentRotationAnimation, |
| child: widget.child, |
| ), |
| ), |
| ], |
| ); |
| } |
| |
| void _onProgressChanged() { |
| _updateGeometryScale(math.max(_previousScaleAnimation.value, _currentScaleAnimation.value)); |
| } |
| |
| void _updateGeometryScale(double scale) { |
| widget.geometryNotifier._updateWith( |
| floatingActionButtonScale: scale, |
| ); |
| } |
| } |
| |
| /// Implements the basic Material Design visual layout structure. |
| /// |
| /// This class provides APIs for showing drawers and bottom sheets. |
| /// |
| /// To display a persistent bottom sheet, obtain the |
| /// [ScaffoldState] for the current [BuildContext] via [Scaffold.of] and use the |
| /// [ScaffoldState.showBottomSheet] function. |
| /// |
| /// {@tool dartpad} |
| /// This example shows a [Scaffold] with a [body] and [FloatingActionButton]. |
| /// The [body] is a [Text] placed in a [Center] in order to center the text |
| /// within the [Scaffold]. The [FloatingActionButton] is connected to a |
| /// callback that increments a counter. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.0.dart ** |
| /// {@end-tool} |
| /// |
| /// {@tool dartpad} |
| /// This example shows a [Scaffold] with a blueGrey [backgroundColor], [body] |
| /// and [FloatingActionButton]. The [body] is a [Text] placed in a [Center] in |
| /// order to center the text within the [Scaffold]. The [FloatingActionButton] |
| /// is connected to a callback that increments a counter. |
| /// |
| /// ![](https://flutter.github.io/assets-for-api-docs/assets/material/scaffold_background_color.png) |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.1.dart ** |
| /// {@end-tool} |
| /// |
| /// {@tool dartpad} |
| /// This example shows a [Scaffold] with an [AppBar], a [BottomAppBar] and a |
| /// [FloatingActionButton]. The [body] is a [Text] placed in a [Center] in order |
| /// to center the text within the [Scaffold]. The [FloatingActionButton] is |
| /// centered and docked within the [BottomAppBar] using |
| /// [FloatingActionButtonLocation.centerDocked]. The [FloatingActionButton] is |
| /// connected to a callback that increments a counter. |
| /// |
| /// ![](https://flutter.github.io/assets-for-api-docs/assets/material/scaffold_bottom_app_bar.png) |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.2.dart ** |
| /// {@end-tool} |
| /// |
| /// ## Scaffold layout, the keyboard, and display "notches" |
| /// |
| /// The scaffold will expand to fill the available space. That usually |
| /// means that it will occupy its entire window or device screen. When |
| /// the device's keyboard appears the Scaffold's ancestor [MediaQuery] |
| /// widget's [MediaQueryData.viewInsets] changes and the Scaffold will |
| /// be rebuilt. By default the scaffold's [body] is resized to make |
| /// room for the keyboard. To prevent the resize set |
| /// [resizeToAvoidBottomInset] to false. In either case the focused |
| /// widget will be scrolled into view if it's within a scrollable |
| /// container. |
| /// |
| /// The [MediaQueryData.padding] value defines areas that might |
| /// not be completely visible, like the display "notch" on the iPhone |
| /// X. The scaffold's [body] is not inset by this padding value |
| /// although an [appBar] or [bottomNavigationBar] will typically |
| /// cause the body to avoid the padding. The [SafeArea] |
| /// widget can be used within the scaffold's body to avoid areas |
| /// like display notches. |
| /// |
| /// ## Floating action button with a draggable scrollable bottom sheet |
| /// |
| /// If [Scaffold.bottomSheet] is a [DraggableScrollableSheet], |
| /// [Scaffold.floatingActionButton] is set, and the bottom sheet is dragged to |
| /// cover greater than 70% of the Scaffold's height, two things happen in parallel: |
| /// |
| /// * Scaffold starts to show scrim (see [ScaffoldState.showBodyScrim]), and |
| /// * [Scaffold.floatingActionButton] is scaled down through an animation with a [Curves.easeIn], and |
| /// disappears when the bottom sheet covers the entire Scaffold. |
| /// |
| /// And as soon as the bottom sheet is dragged down to cover less than 70% of the [Scaffold], the scrim |
| /// disappears and [Scaffold.floatingActionButton] animates back to its normal size. |
| /// |
| /// ## Troubleshooting |
| /// |
| /// ### Nested Scaffolds |
| /// |
| /// The Scaffold is designed to be a top level container for |
| /// a [MaterialApp]. This means that adding a Scaffold |
| /// to each route on a Material app will provide the app with |
| /// Material's basic visual layout structure. |
| /// |
| /// It is typically not necessary to nest Scaffolds. For example, in a |
| /// tabbed UI, where the [bottomNavigationBar] is a [TabBar] |
| /// and the body is a [TabBarView], you might be tempted to make each tab bar |
| /// view a scaffold with a differently titled AppBar. Rather, it would be |
| /// better to add a listener to the [TabController] that updates the |
| /// AppBar |
| /// |
| /// {@tool snippet} |
| /// Add a listener to the app's tab controller so that the [AppBar] title of the |
| /// app's one and only scaffold is reset each time a new tab is selected. |
| /// |
| /// ```dart |
| /// TabController(vsync: tickerProvider, length: tabCount)..addListener(() { |
| /// if (!tabController.indexIsChanging) { |
| /// setState(() { |
| /// // Rebuild the enclosing scaffold with a new AppBar title |
| /// appBarTitle = 'Tab ${tabController.index}'; |
| /// }); |
| /// } |
| /// }) |
| /// ``` |
| /// {@end-tool} |
| /// |
| /// Although there are some use cases, like a presentation app that |
| /// shows embedded flutter content, where nested scaffolds are |
| /// appropriate, it's best to avoid nesting scaffolds. |
| /// |
| /// See also: |
| /// |
| /// * [AppBar], which is a horizontal bar typically shown at the top of an app |
| /// using the [appBar] property. |
| /// * [BottomAppBar], which is a horizontal bar typically shown at the bottom |
| /// of an app using the [bottomNavigationBar] property. |
| /// * [FloatingActionButton], which is a circular button typically shown in the |
| /// bottom right corner of the app using the [floatingActionButton] property. |
| /// * [Drawer], which is a vertical panel that is typically displayed to the |
| /// left of the body (and often hidden on phones) using the [drawer] |
| /// property. |
| /// * [BottomNavigationBar], which is a horizontal array of buttons typically |
| /// shown along the bottom of the app using the [bottomNavigationBar] |
| /// property. |
| /// * [BottomSheet], which is an overlay typically shown near the bottom of the |
| /// app. A bottom sheet can either be persistent, in which case it is shown |
| /// using the [ScaffoldState.showBottomSheet] method, or modal, in which case |
| /// it is shown using the [showModalBottomSheet] function. |
| /// * [SnackBar], which is a lightweight message with an optional action which |
| /// briefly displays at the bottom of the screen. Use the |
| /// [ScaffoldMessengerState.showSnackBar] method to show snack bars. |
| /// * [MaterialBanner], which displays an important, succinct message, at the |
| /// top of the screen, below the app bar. Use the |
| /// [ScaffoldMessengerState.showMaterialBanner] method to show material banners. |
| /// * [ScaffoldState], which is the state associated with this widget. |
| /// * <https://material.io/design/layout/responsive-layout-grid.html> |
| /// * Cookbook: [Add a Drawer to a screen](https://flutter.dev/docs/cookbook/design/drawer) |
| class Scaffold extends StatefulWidget { |
| /// Creates a visual scaffold for Material Design widgets. |
| const Scaffold({ |
| super.key, |
| this.appBar, |
| this.body, |
| this.floatingActionButton, |
| this.floatingActionButtonLocation, |
| this.floatingActionButtonAnimator, |
| this.persistentFooterButtons, |
| this.persistentFooterAlignment = AlignmentDirectional.centerEnd, |
| this.drawer, |
| this.onDrawerChanged, |
| this.endDrawer, |
| this.onEndDrawerChanged, |
| this.bottomNavigationBar, |
| this.bottomSheet, |
| this.backgroundColor, |
| this.resizeToAvoidBottomInset, |
| this.primary = true, |
| this.drawerDragStartBehavior = DragStartBehavior.start, |
| this.extendBody = false, |
| this.extendBodyBehindAppBar = false, |
| this.drawerScrimColor, |
| this.drawerEdgeDragWidth, |
| this.drawerEnableOpenDragGesture = true, |
| this.endDrawerEnableOpenDragGesture = true, |
| this.restorationId, |
| }); |
| |
| /// If true, and [bottomNavigationBar] or [persistentFooterButtons] |
| /// is specified, then the [body] extends to the bottom of the Scaffold, |
| /// instead of only extending to the top of the [bottomNavigationBar] |
| /// or the [persistentFooterButtons]. |
| /// |
| /// If true, a [MediaQuery] widget whose bottom padding matches the height |
| /// of the [bottomNavigationBar] will be added above the scaffold's [body]. |
| /// |
| /// This property is often useful when the [bottomNavigationBar] has |
| /// a non-rectangular shape, like [CircularNotchedRectangle], which |
| /// adds a [FloatingActionButton] sized notch to the top edge of the bar. |
| /// In this case specifying `extendBody: true` ensures that scaffold's |
| /// body will be visible through the bottom navigation bar's notch. |
| /// |
| /// See also: |
| /// |
| /// * [extendBodyBehindAppBar], which extends the height of the body |
| /// to the top of the scaffold. |
| final bool extendBody; |
| |
| /// If true, and an [appBar] is specified, then the height of the [body] is |
| /// extended to include the height of the app bar and the top of the body |
| /// is aligned with the top of the app bar. |
| /// |
| /// This is useful if the app bar's [AppBar.backgroundColor] is not |
| /// completely opaque. |
| /// |
| /// This property is false by default. It must not be null. |
| /// |
| /// See also: |
| /// |
| /// * [extendBody], which extends the height of the body to the bottom |
| /// of the scaffold. |
| final bool extendBodyBehindAppBar; |
| |
| /// An app bar to display at the top of the scaffold. |
| final PreferredSizeWidget? appBar; |
| |
| /// The primary content of the scaffold. |
| /// |
| /// Displayed below the [appBar], above the bottom of the ambient |
| /// [MediaQuery]'s [MediaQueryData.viewInsets], and behind the |
| /// [floatingActionButton] and [drawer]. If [resizeToAvoidBottomInset] is |
| /// false then the body is not resized when the onscreen keyboard appears, |
| /// i.e. it is not inset by `viewInsets.bottom`. |
| /// |
| /// The widget in the body of the scaffold is positioned at the top-left of |
| /// the available space between the app bar and the bottom of the scaffold. To |
| /// center this widget instead, consider putting it in a [Center] widget and |
| /// having that be the body. To expand this widget instead, consider |
| /// putting it in a [SizedBox.expand]. |
| /// |
| /// If you have a column of widgets that should normally fit on the screen, |
| /// but may overflow and would in such cases need to scroll, consider using a |
| /// [ListView] as the body of the scaffold. This is also a good choice for |
| /// the case where your body is a scrollable list. |
| final Widget? body; |
| |
| /// A button displayed floating above [body], in the bottom right corner. |
| /// |
| /// Typically a [FloatingActionButton]. |
| final Widget? floatingActionButton; |
| |
| /// Responsible for determining where the [floatingActionButton] should go. |
| /// |
| /// If null, the [ScaffoldState] will use the default location, [FloatingActionButtonLocation.endFloat]. |
| final FloatingActionButtonLocation? floatingActionButtonLocation; |
| |
| /// Animator to move the [floatingActionButton] to a new [floatingActionButtonLocation]. |
| /// |
| /// If null, the [ScaffoldState] will use the default animator, [FloatingActionButtonAnimator.scaling]. |
| final FloatingActionButtonAnimator? floatingActionButtonAnimator; |
| |
| /// A set of buttons that are displayed at the bottom of the scaffold. |
| /// |
| /// Typically this is a list of [TextButton] widgets. These buttons are |
| /// persistently visible, even if the [body] of the scaffold scrolls. |
| /// |
| /// These widgets will be wrapped in an [OverflowBar]. |
| /// |
| /// The [persistentFooterButtons] are rendered above the |
| /// [bottomNavigationBar] but below the [body]. |
| final List<Widget>? persistentFooterButtons; |
| |
| /// The alignment of the [persistentFooterButtons] inside the [OverflowBar]. |
| /// |
| /// Defaults to [AlignmentDirectional.centerEnd]. |
| final AlignmentDirectional persistentFooterAlignment; |
| |
| /// A panel displayed to the side of the [body], often hidden on mobile |
| /// devices. Swipes in from either left-to-right ([TextDirection.ltr]) or |
| /// right-to-left ([TextDirection.rtl]) |
| /// |
| /// Typically a [Drawer]. |
| /// |
| /// To open the drawer, use the [ScaffoldState.openDrawer] function. |
| /// |
| /// To close the drawer, use either [ScaffoldState.closeDrawer], [Navigator.pop] |
| /// or press the escape key on the keyboard. |
| /// |
| /// {@tool dartpad} |
| /// To disable the drawer edge swipe on mobile, set the |
| /// [Scaffold.drawerEnableOpenDragGesture] to false. Then, use |
| /// [ScaffoldState.openDrawer] to open the drawer and [Navigator.pop] to close |
| /// it. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.drawer.0.dart ** |
| /// {@end-tool} |
| final Widget? drawer; |
| |
| /// Optional callback that is called when the [Scaffold.drawer] is opened or closed. |
| final DrawerCallback? onDrawerChanged; |
| |
| /// A panel displayed to the side of the [body], often hidden on mobile |
| /// devices. Swipes in from right-to-left ([TextDirection.ltr]) or |
| /// left-to-right ([TextDirection.rtl]) |
| /// |
| /// Typically a [Drawer]. |
| /// |
| /// To open the drawer, use the [ScaffoldState.openEndDrawer] function. |
| /// |
| /// To close the drawer, use either [ScaffoldState.closeEndDrawer], [Navigator.pop] |
| /// or press the escape key on the keyboard. |
| /// |
| /// {@tool dartpad} |
| /// To disable the drawer edge swipe, set the |
| /// [Scaffold.endDrawerEnableOpenDragGesture] to false. Then, use |
| /// [ScaffoldState.openEndDrawer] to open the drawer and [Navigator.pop] to |
| /// close it. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.end_drawer.0.dart ** |
| /// {@end-tool} |
| final Widget? endDrawer; |
| |
| /// Optional callback that is called when the [Scaffold.endDrawer] is opened or closed. |
| final DrawerCallback? onEndDrawerChanged; |
| |
| /// The color to use for the scrim that obscures primary content while a drawer is open. |
| /// |
| /// If this is null, then [DrawerThemeData.scrimColor] is used. If that |
| /// is also null, then it defaults to [Colors.black54]. |
| final Color? drawerScrimColor; |
| |
| /// The color of the [Material] widget that underlies the entire Scaffold. |
| /// |
| /// The theme's [ThemeData.scaffoldBackgroundColor] by default. |
| final Color? backgroundColor; |
| |
| /// A bottom navigation bar to display at the bottom of the scaffold. |
| /// |
| /// Snack bars slide from underneath the bottom navigation bar while bottom |
| /// sheets are stacked on top. |
| /// |
| /// The [bottomNavigationBar] is rendered below the [persistentFooterButtons] |
| /// and the [body]. |
| final Widget? bottomNavigationBar; |
| |
| /// The persistent bottom sheet to display. |
| /// |
| /// A persistent bottom sheet shows information that supplements the primary |
| /// content of the app. A persistent bottom sheet remains visible even when |
| /// the user interacts with other parts of the app. |
| /// |
| /// A closely related widget is a modal bottom sheet, which is an alternative |
| /// to a menu or a dialog and prevents the user from interacting with the rest |
| /// of the app. Modal bottom sheets can be created and displayed with the |
| /// [showModalBottomSheet] function. |
| /// |
| /// Unlike the persistent bottom sheet displayed by [showBottomSheet] |
| /// this bottom sheet is not a [LocalHistoryEntry] and cannot be dismissed |
| /// with the scaffold appbar's back button. |
| /// |
| /// If a persistent bottom sheet created with [showBottomSheet] is already |
| /// visible, it must be closed before building the Scaffold with a new |
| /// [bottomSheet]. |
| /// |
| /// The value of [bottomSheet] can be any widget at all. It's unlikely to |
| /// actually be a [BottomSheet], which is used by the implementations of |
| /// [showBottomSheet] and [showModalBottomSheet]. Typically it's a widget |
| /// that includes [Material]. |
| /// |
| /// See also: |
| /// |
| /// * [showBottomSheet], which displays a bottom sheet as a route that can |
| /// be dismissed with the scaffold's back button. |
| /// * [showModalBottomSheet], which displays a modal bottom sheet. |
| /// * [BottomSheetThemeData], which can be used to customize the default |
| /// bottom sheet property values when using a [BottomSheet]. |
| final Widget? bottomSheet; |
| |
| /// If true the [body] and the scaffold's floating widgets should size |
| /// themselves to avoid the onscreen keyboard whose height is defined by the |
| /// ambient [MediaQuery]'s [MediaQueryData.viewInsets] `bottom` property. |
| /// |
| /// For example, if there is an onscreen keyboard displayed above the |
| /// scaffold, the body can be resized to avoid overlapping the keyboard, which |
| /// prevents widgets inside the body from being obscured by the keyboard. |
| /// |
| /// Defaults to true. |
| final bool? resizeToAvoidBottomInset; |
| |
| /// Whether this scaffold is being displayed at the top of the screen. |
| /// |
| /// If true then the height of the [appBar] will be extended by the height |
| /// of the screen's status bar, i.e. the top padding for [MediaQuery]. |
| /// |
| /// The default value of this property, like the default value of |
| /// [AppBar.primary], is true. |
| final bool primary; |
| |
| /// {@macro flutter.material.DrawerController.dragStartBehavior} |
| final DragStartBehavior drawerDragStartBehavior; |
| |
| /// The width of the area within which a horizontal swipe will open the |
| /// drawer. |
| /// |
| /// By default, the value used is 20.0 added to the padding edge of |
| /// `MediaQuery.paddingOf(context)` that corresponds to the surrounding |
| /// [TextDirection]. This ensures that the drag area for notched devices is |
| /// not obscured. For example, if `TextDirection.of(context)` is set to |
| /// [TextDirection.ltr], 20.0 will be added to |
| /// `MediaQuery.paddingOf(context).left`. |
| final double? drawerEdgeDragWidth; |
| |
| /// Determines if the [Scaffold.drawer] can be opened with a drag |
| /// gesture on mobile. |
| /// |
| /// On desktop platforms, the drawer is not draggable. |
| /// |
| /// By default, the drag gesture is enabled on mobile. |
| final bool drawerEnableOpenDragGesture; |
| |
| /// Determines if the [Scaffold.endDrawer] can be opened with a |
| /// gesture on mobile. |
| /// |
| /// On desktop platforms, the drawer is not draggable. |
| /// |
| /// By default, the drag gesture is enabled on mobile. |
| final bool endDrawerEnableOpenDragGesture; |
| |
| /// Restoration ID to save and restore the state of the [Scaffold]. |
| /// |
| /// If it is non-null, the scaffold will persist and restore whether the |
| /// [drawer] and [endDrawer] was open or closed. |
| /// |
| /// The state of this widget is persisted in a [RestorationBucket] claimed |
| /// from the surrounding [RestorationScope] using the provided restoration ID. |
| /// |
| /// See also: |
| /// |
| /// * [RestorationManager], which explains how state restoration works in |
| /// Flutter. |
| final String? restorationId; |
| |
| /// Finds the [ScaffoldState] from the closest instance of this class that |
| /// encloses the given context. |
| /// |
| /// If no instance of this class encloses the given context, will cause an |
| /// assert in debug mode, and throw an exception in release mode. |
| /// |
| /// This method can be expensive (it walks the element tree). |
| /// |
| /// {@tool dartpad} |
| /// Typical usage of the [Scaffold.of] function is to call it from within the |
| /// `build` method of a child of a [Scaffold]. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.of.0.dart ** |
| /// {@end-tool} |
| /// |
| /// {@tool dartpad} |
| /// When the [Scaffold] is actually created in the same `build` function, the |
| /// `context` argument to the `build` function can't be used to find the |
| /// [Scaffold] (since it's "above" the widget being returned in the widget |
| /// tree). In such cases, the following technique with a [Builder] can be used |
| /// to provide a new scope with a [BuildContext] that is "under" the |
| /// [Scaffold]: |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold.of.1.dart ** |
| /// {@end-tool} |
| /// |
| /// A more efficient solution is to split your build function into several |
| /// widgets. This introduces a new context from which you can obtain the |
| /// [Scaffold]. In this solution, you would have an outer widget that creates |
| /// the [Scaffold] populated by instances of your new inner widgets, and then |
| /// in these inner widgets you would use [Scaffold.of]. |
| /// |
| /// A less elegant but more expedient solution is assign a [GlobalKey] to the |
| /// [Scaffold], then use the `key.currentState` property to obtain the |
| /// [ScaffoldState] rather than using the [Scaffold.of] function. |
| /// |
| /// If there is no [Scaffold] in scope, then this will throw an exception. |
| /// To return null if there is no [Scaffold], use [maybeOf] instead. |
| static ScaffoldState of(BuildContext context) { |
| final ScaffoldState? result = context.findAncestorStateOfType<ScaffoldState>(); |
| if (result != null) { |
| return result; |
| } |
| throw FlutterError.fromParts(<DiagnosticsNode>[ |
| ErrorSummary( |
| 'Scaffold.of() called with a context that does not contain a Scaffold.', |
| ), |
| ErrorDescription( |
| 'No Scaffold ancestor could be found starting from the context that was passed to Scaffold.of(). ' |
| 'This usually happens when the context provided is from the same StatefulWidget as that ' |
| 'whose build function actually creates the Scaffold widget being sought.', |
| ), |
| ErrorHint( |
| 'There are several ways to avoid this problem. The simplest is to use a Builder to get a ' |
| 'context that is "under" the Scaffold. For an example of this, please see the ' |
| 'documentation for Scaffold.of():\n' |
| ' https://api.flutter.dev/flutter/material/Scaffold/of.html', |
| ), |
| ErrorHint( |
| 'A more efficient solution is to split your build function into several widgets. This ' |
| 'introduces a new context from which you can obtain the Scaffold. In this solution, ' |
| 'you would have an outer widget that creates the Scaffold populated by instances of ' |
| 'your new inner widgets, and then in these inner widgets you would use Scaffold.of().\n' |
| 'A less elegant but more expedient solution is assign a GlobalKey to the Scaffold, ' |
| 'then use the key.currentState property to obtain the ScaffoldState rather than ' |
| 'using the Scaffold.of() function.', |
| ), |
| context.describeElement('The context used was'), |
| ]); |
| } |
| |
| /// Finds the [ScaffoldState] from the closest instance of this class that |
| /// encloses the given context. |
| /// |
| /// If no instance of this class encloses the given context, will return null. |
| /// To throw an exception instead, use [of] instead of this function. |
| /// |
| /// This method can be expensive (it walks the element tree). |
| /// |
| /// See also: |
| /// |
| /// * [of], a similar function to this one that throws if no instance |
| /// encloses the given context. Also includes some sample code in its |
| /// documentation. |
| static ScaffoldState? maybeOf(BuildContext context) { |
| return context.findAncestorStateOfType<ScaffoldState>(); |
| } |
| |
| /// Returns a [ValueListenable] for the [ScaffoldGeometry] for the closest |
| /// [Scaffold] ancestor of the given context. |
| /// |
| /// The [ValueListenable.value] is only available at paint time. |
| /// |
| /// Notifications are guaranteed to be sent before the first paint pass |
| /// with the new geometry, but there is no guarantee whether a build or |
| /// layout passes are going to happen between the notification and the next |
| /// paint pass. |
| /// |
| /// The closest [Scaffold] ancestor for the context might change, e.g when |
| /// an element is moved from one scaffold to another. For [StatefulWidget]s |
| /// using this listenable, a change of the [Scaffold] ancestor will |
| /// trigger a [State.didChangeDependencies]. |
| /// |
| /// A typical pattern for listening to the scaffold geometry would be to |
| /// call [Scaffold.geometryOf] in [State.didChangeDependencies], compare the |
| /// return value with the previous listenable, if it has changed, unregister |
| /// the listener, and register a listener to the new [ScaffoldGeometry] |
| /// listenable. |
| static ValueListenable<ScaffoldGeometry> geometryOf(BuildContext context) { |
| final _ScaffoldScope? scaffoldScope = context.dependOnInheritedWidgetOfExactType<_ScaffoldScope>(); |
| if (scaffoldScope == null) { |
| throw FlutterError.fromParts(<DiagnosticsNode>[ |
| ErrorSummary( |
| 'Scaffold.geometryOf() called with a context that does not contain a Scaffold.', |
| ), |
| ErrorDescription( |
| 'This usually happens when the context provided is from the same StatefulWidget as that ' |
| 'whose build function actually creates the Scaffold widget being sought.', |
| ), |
| ErrorHint( |
| 'There are several ways to avoid this problem. The simplest is to use a Builder to get a ' |
| 'context that is "under" the Scaffold. For an example of this, please see the ' |
| 'documentation for Scaffold.of():\n' |
| ' https://api.flutter.dev/flutter/material/Scaffold/of.html', |
| ), |
| ErrorHint( |
| 'A more efficient solution is to split your build function into several widgets. This ' |
| 'introduces a new context from which you can obtain the Scaffold. In this solution, ' |
| 'you would have an outer widget that creates the Scaffold populated by instances of ' |
| 'your new inner widgets, and then in these inner widgets you would use Scaffold.geometryOf().', |
| ), |
| context.describeElement('The context used was'), |
| ]); |
| } |
| return scaffoldScope.geometryNotifier; |
| } |
| |
| /// Whether the Scaffold that most tightly encloses the given context has a |
| /// drawer. |
| /// |
| /// If this is being used during a build (for example to decide whether to |
| /// show an "open drawer" button), set the `registerForUpdates` argument to |
| /// true. This will then set up an [InheritedWidget] relationship with the |
| /// [Scaffold] so that the client widget gets rebuilt whenever the [hasDrawer] |
| /// value changes. |
| /// |
| /// This method can be expensive (it walks the element tree). |
| /// |
| /// See also: |
| /// |
| /// * [Scaffold.of], which provides access to the [ScaffoldState] object as a |
| /// whole, from which you can show bottom sheets, and so forth. |
| static bool hasDrawer(BuildContext context, { bool registerForUpdates = true }) { |
| if (registerForUpdates) { |
| final _ScaffoldScope? scaffold = context.dependOnInheritedWidgetOfExactType<_ScaffoldScope>(); |
| return scaffold?.hasDrawer ?? false; |
| } else { |
| final ScaffoldState? scaffold = context.findAncestorStateOfType<ScaffoldState>(); |
| return scaffold?.hasDrawer ?? false; |
| } |
| } |
| |
| @override |
| ScaffoldState createState() => ScaffoldState(); |
| } |
| |
| /// State for a [Scaffold]. |
| /// |
| /// Can display [BottomSheet]s. Retrieve a [ScaffoldState] from the current |
| /// [BuildContext] using [Scaffold.of]. |
| class ScaffoldState extends State<Scaffold> with TickerProviderStateMixin, RestorationMixin { |
| @override |
| String? get restorationId => widget.restorationId; |
| |
| @override |
| void restoreState(RestorationBucket? oldBucket, bool initialRestore) { |
| registerForRestoration(_drawerOpened, 'drawer_open'); |
| registerForRestoration(_endDrawerOpened, 'end_drawer_open'); |
| } |
| |
| // DRAWER API |
| |
| final GlobalKey<DrawerControllerState> _drawerKey = GlobalKey<DrawerControllerState>(); |
| final GlobalKey<DrawerControllerState> _endDrawerKey = GlobalKey<DrawerControllerState>(); |
| |
| final GlobalKey _bodyKey = GlobalKey(); |
| |
| /// Whether this scaffold has a non-null [Scaffold.appBar]. |
| bool get hasAppBar => widget.appBar != null; |
| /// Whether this scaffold has a non-null [Scaffold.drawer]. |
| bool get hasDrawer => widget.drawer != null; |
| /// Whether this scaffold has a non-null [Scaffold.endDrawer]. |
| bool get hasEndDrawer => widget.endDrawer != null; |
| /// Whether this scaffold has a non-null [Scaffold.floatingActionButton]. |
| bool get hasFloatingActionButton => widget.floatingActionButton != null; |
| |
| double? _appBarMaxHeight; |
| /// The max height the [Scaffold.appBar] uses. |
| /// |
| /// This is based on the appBar preferred height plus the top padding. |
| double? get appBarMaxHeight => _appBarMaxHeight; |
| final RestorableBool _drawerOpened = RestorableBool(false); |
| final RestorableBool _endDrawerOpened = RestorableBool(false); |
| |
| /// Whether the [Scaffold.drawer] is opened. |
| /// |
| /// See also: |
| /// |
| /// * [ScaffoldState.openDrawer], which opens the [Scaffold.drawer] of a |
| /// [Scaffold]. |
| bool get isDrawerOpen => _drawerOpened.value; |
| |
| /// Whether the [Scaffold.endDrawer] is opened. |
| /// |
| /// See also: |
| /// |
| /// * [ScaffoldState.openEndDrawer], which opens the [Scaffold.endDrawer] of |
| /// a [Scaffold]. |
| bool get isEndDrawerOpen => _endDrawerOpened.value; |
| |
| void _drawerOpenedCallback(bool isOpened) { |
| if (_drawerOpened.value != isOpened && _drawerKey.currentState != null) { |
| setState(() { |
| _drawerOpened.value = isOpened; |
| }); |
| widget.onDrawerChanged?.call(isOpened); |
| } |
| } |
| |
| void _endDrawerOpenedCallback(bool isOpened) { |
| if (_endDrawerOpened.value != isOpened && _endDrawerKey.currentState != null) { |
| setState(() { |
| _endDrawerOpened.value = isOpened; |
| }); |
| widget.onEndDrawerChanged?.call(isOpened); |
| } |
| } |
| |
| /// Opens the [Drawer] (if any). |
| /// |
| /// If the scaffold has a non-null [Scaffold.drawer], this function will cause |
| /// the drawer to begin its entrance animation. |
| /// |
| /// Normally this is not needed since the [Scaffold] automatically shows an |
| /// appropriate [IconButton], and handles the edge-swipe gesture, to show the |
| /// drawer. |
| /// |
| /// To close the drawer, use either [ScaffoldState.closeEndDrawer] or |
| /// [Navigator.pop]. |
| /// |
| /// See [Scaffold.of] for information about how to obtain the [ScaffoldState]. |
| void openDrawer() { |
| if (_endDrawerKey.currentState != null && _endDrawerOpened.value) { |
| _endDrawerKey.currentState!.close(); |
| } |
| _drawerKey.currentState?.open(); |
| } |
| |
| /// Opens the end side [Drawer] (if any). |
| /// |
| /// If the scaffold has a non-null [Scaffold.endDrawer], this function will cause |
| /// the end side drawer to begin its entrance animation. |
| /// |
| /// Normally this is not needed since the [Scaffold] automatically shows an |
| /// appropriate [IconButton], and handles the edge-swipe gesture, to show the |
| /// drawer. |
| /// |
| /// To close the drawer, use either [ScaffoldState.closeEndDrawer] or |
| /// [Navigator.pop]. |
| /// |
| /// See [Scaffold.of] for information about how to obtain the [ScaffoldState]. |
| void openEndDrawer() { |
| if (_drawerKey.currentState != null && _drawerOpened.value) { |
| _drawerKey.currentState!.close(); |
| } |
| _endDrawerKey.currentState?.open(); |
| } |
| |
| // Used for both the snackbar and material banner APIs |
| ScaffoldMessengerState? _scaffoldMessenger; |
| |
| // SNACKBAR API |
| ScaffoldFeatureController<SnackBar, SnackBarClosedReason>? _messengerSnackBar; |
| |
| // This is used to update the _messengerSnackBar by the ScaffoldMessenger. |
| void _updateSnackBar() { |
| final ScaffoldFeatureController<SnackBar, SnackBarClosedReason>? messengerSnackBar = _scaffoldMessenger!._snackBars.isNotEmpty |
| ? _scaffoldMessenger!._snackBars.first |
| : null; |
| |
| if (_messengerSnackBar != messengerSnackBar) { |
| setState(() { |
| _messengerSnackBar = messengerSnackBar; |
| }); |
| } |
| } |
| |
| // MATERIAL BANNER API |
| |
| // The _messengerMaterialBanner represents the current MaterialBanner being managed by |
| // the ScaffoldMessenger, instead of the Scaffold. |
| ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason>? _messengerMaterialBanner; |
| |
| // This is used to update the _messengerMaterialBanner by the ScaffoldMessenger. |
| void _updateMaterialBanner() { |
| final ScaffoldFeatureController<MaterialBanner, MaterialBannerClosedReason>? messengerMaterialBanner = _scaffoldMessenger!._materialBanners.isNotEmpty |
| ? _scaffoldMessenger!._materialBanners.first |
| : null; |
| |
| if (_messengerMaterialBanner != messengerMaterialBanner) { |
| setState(() { |
| _messengerMaterialBanner = messengerMaterialBanner; |
| }); |
| } |
| } |
| |
| // PERSISTENT BOTTOM SHEET API |
| |
| // Contains bottom sheets that may still be animating out of view. |
| // Important if the app/user takes an action that could repeatedly show a |
| // bottom sheet. |
| final List<_StandardBottomSheet> _dismissedBottomSheets = <_StandardBottomSheet>[]; |
| PersistentBottomSheetController<dynamic>? _currentBottomSheet; |
| final GlobalKey _currentBottomSheetKey = GlobalKey(); |
| LocalHistoryEntry? _persistentSheetHistoryEntry; |
| |
| void _maybeBuildPersistentBottomSheet() { |
| if (widget.bottomSheet != null && _currentBottomSheet == null) { |
| // The new _currentBottomSheet is not a local history entry so a "back" button |
| // will not be added to the Scaffold's appbar and the bottom sheet will not |
| // support drag or swipe to dismiss. |
| final AnimationController animationController = BottomSheet.createAnimationController(this)..value = 1.0; |
| bool persistentBottomSheetExtentChanged(DraggableScrollableNotification notification) { |
| if (notification.extent - notification.initialExtent > precisionErrorTolerance) { |
| if (_persistentSheetHistoryEntry == null) { |
| _persistentSheetHistoryEntry = LocalHistoryEntry(onRemove: () { |
| DraggableScrollableActuator.reset(notification.context); |
| showBodyScrim(false, 0.0); |
| _floatingActionButtonVisibilityValue = 1.0; |
| _persistentSheetHistoryEntry = null; |
| }); |
| ModalRoute.of(context)!.addLocalHistoryEntry(_persistentSheetHistoryEntry!); |
| } |
| } else if (_persistentSheetHistoryEntry != null) { |
| _persistentSheetHistoryEntry!.remove(); |
| } |
| return false; |
| } |
| |
| // Stop the animation and unmount the dismissed sheets from the tree immediately, |
| // otherwise may cause duplicate GlobalKey assertion if the sheet sub-tree contains |
| // GlobalKey widgets. |
| if (_dismissedBottomSheets.isNotEmpty) { |
| final List<_StandardBottomSheet> sheets = List<_StandardBottomSheet>.of(_dismissedBottomSheets, growable: false); |
| for (final _StandardBottomSheet sheet in sheets) { |
| sheet.animationController.reset(); |
| } |
| assert(_dismissedBottomSheets.isEmpty); |
| } |
| |
| _currentBottomSheet = _buildBottomSheet<void>( |
| (BuildContext context) { |
| return NotificationListener<DraggableScrollableNotification>( |
| onNotification: persistentBottomSheetExtentChanged, |
| child: DraggableScrollableActuator( |
| child: StatefulBuilder( |
| key: _currentBottomSheetKey, |
| builder: (BuildContext context, StateSetter setState) { |
| return widget.bottomSheet ?? const SizedBox.shrink(); |
| }, |
| ), |
| ), |
| ); |
| }, |
| isPersistent: true, |
| animationController: animationController, |
| ); |
| } |
| } |
| |
| void _closeCurrentBottomSheet() { |
| if (_currentBottomSheet != null) { |
| if (!_currentBottomSheet!._isLocalHistoryEntry) { |
| _currentBottomSheet!.close(); |
| } |
| assert(() { |
| _currentBottomSheet?._completer.future.whenComplete(() { |
| assert(_currentBottomSheet == null); |
| }); |
| return true; |
| }()); |
| } |
| } |
| |
| /// Closes [Scaffold.drawer] if it is currently opened. |
| /// |
| /// See [Scaffold.of] for information about how to obtain the [ScaffoldState]. |
| void closeDrawer() { |
| if (hasDrawer && isDrawerOpen) { |
| _drawerKey.currentState!.close(); |
| } |
| } |
| |
| /// Closes [Scaffold.endDrawer] if it is currently opened. |
| /// |
| /// See [Scaffold.of] for information about how to obtain the [ScaffoldState]. |
| void closeEndDrawer() { |
| if (hasEndDrawer && isEndDrawerOpen) { |
| _endDrawerKey.currentState!.close(); |
| } |
| } |
| |
| void _updatePersistentBottomSheet() { |
| _currentBottomSheetKey.currentState!.setState(() {}); |
| } |
| |
| PersistentBottomSheetController<T> _buildBottomSheet<T>( |
| WidgetBuilder builder, { |
| required bool isPersistent, |
| required AnimationController animationController, |
| Color? backgroundColor, |
| double? elevation, |
| ShapeBorder? shape, |
| Clip? clipBehavior, |
| BoxConstraints? constraints, |
| bool? enableDrag, |
| bool shouldDisposeAnimationController = true, |
| }) { |
| assert(() { |
| if (widget.bottomSheet != null && isPersistent && _currentBottomSheet != null) { |
| throw FlutterError( |
| 'Scaffold.bottomSheet cannot be specified while a bottom sheet ' |
| 'displayed with showBottomSheet() is still visible.\n' |
| 'Rebuild the Scaffold with a null bottomSheet before calling showBottomSheet().', |
| ); |
| } |
| return true; |
| }()); |
| |
| final Completer<T> completer = Completer<T>(); |
| final GlobalKey<_StandardBottomSheetState> bottomSheetKey = GlobalKey<_StandardBottomSheetState>(); |
| late _StandardBottomSheet bottomSheet; |
| |
| bool removedEntry = false; |
| bool doingDispose = false; |
| |
| void removePersistentSheetHistoryEntryIfNeeded() { |
| assert(isPersistent); |
| if (_persistentSheetHistoryEntry != null) { |
| _persistentSheetHistoryEntry!.remove(); |
| _persistentSheetHistoryEntry = null; |
| } |
| } |
| |
| void removeCurrentBottomSheet() { |
| removedEntry = true; |
| if (_currentBottomSheet == null) { |
| return; |
| } |
| assert(_currentBottomSheet!._widget == bottomSheet); |
| assert(bottomSheetKey.currentState != null); |
| _showFloatingActionButton(); |
| |
| if (isPersistent) { |
| removePersistentSheetHistoryEntryIfNeeded(); |
| } |
| |
| bottomSheetKey.currentState!.close(); |
| setState(() { |
| _showBodyScrim = false; |
| _bodyScrimColor = Colors.black.withOpacity(0.0); |
| _currentBottomSheet = null; |
| }); |
| |
| if (animationController.status != AnimationStatus.dismissed) { |
| _dismissedBottomSheets.add(bottomSheet); |
| } |
| completer.complete(); |
| } |
| |
| final LocalHistoryEntry? entry = isPersistent |
| ? null |
| : LocalHistoryEntry(onRemove: () { |
| if (!removedEntry && _currentBottomSheet?._widget == bottomSheet && !doingDispose) { |
| removeCurrentBottomSheet(); |
| } |
| }); |
| |
| void removeEntryIfNeeded() { |
| if (!isPersistent && !removedEntry) { |
| assert(entry != null); |
| entry!.remove(); |
| removedEntry = true; |
| } |
| } |
| |
| bottomSheet = _StandardBottomSheet( |
| key: bottomSheetKey, |
| animationController: animationController, |
| enableDrag: enableDrag ?? !isPersistent, |
| onClosing: () { |
| if (_currentBottomSheet == null) { |
| return; |
| } |
| assert(_currentBottomSheet!._widget == bottomSheet); |
| removeEntryIfNeeded(); |
| }, |
| onDismissed: () { |
| if (_dismissedBottomSheets.contains(bottomSheet)) { |
| setState(() { |
| _dismissedBottomSheets.remove(bottomSheet); |
| }); |
| } |
| }, |
| onDispose: () { |
| doingDispose = true; |
| removeEntryIfNeeded(); |
| if (shouldDisposeAnimationController) { |
| animationController.dispose(); |
| } |
| }, |
| builder: builder, |
| isPersistent: isPersistent, |
| backgroundColor: backgroundColor, |
| elevation: elevation, |
| shape: shape, |
| clipBehavior: clipBehavior, |
| constraints: constraints, |
| ); |
| |
| if (!isPersistent) { |
| ModalRoute.of(context)!.addLocalHistoryEntry(entry!); |
| } |
| |
| return PersistentBottomSheetController<T>._( |
| bottomSheet, |
| completer, |
| entry != null |
| ? entry.remove |
| : removeCurrentBottomSheet, |
| (VoidCallback fn) { bottomSheetKey.currentState?.setState(fn); }, |
| !isPersistent, |
| ); |
| } |
| |
| /// Shows a Material Design bottom sheet in the nearest [Scaffold]. To show |
| /// a persistent bottom sheet, use the [Scaffold.bottomSheet]. |
| /// |
| /// Returns a controller that can be used to close and otherwise manipulate the |
| /// bottom sheet. |
| /// |
| /// To rebuild the bottom sheet (e.g. if it is stateful), call |
| /// [PersistentBottomSheetController.setState] on the controller returned by |
| /// this method. |
| /// |
| /// The new bottom sheet becomes a [LocalHistoryEntry] for the enclosing |
| /// [ModalRoute] and a back button is added to the app bar of the [Scaffold] |
| /// that closes the bottom sheet. |
| /// |
| /// The [transitionAnimationController] controls the bottom sheet's entrance and |
| /// exit animations. It's up to the owner of the controller to call |
| /// [AnimationController.dispose] when the controller is no longer needed. |
| /// |
| /// To create a persistent bottom sheet that is not a [LocalHistoryEntry] and |
| /// does not add a back button to the enclosing Scaffold's app bar, use the |
| /// [Scaffold.bottomSheet] constructor parameter. |
| /// |
| /// A persistent bottom sheet shows information that supplements the primary |
| /// content of the app. A persistent bottom sheet remains visible even when |
| /// the user interacts with other parts of the app. |
| /// |
| /// A closely related widget is a modal bottom sheet, which is an alternative |
| /// to a menu or a dialog and prevents the user from interacting with the rest |
| /// of the app. Modal bottom sheets can be created and displayed with the |
| /// [showModalBottomSheet] function. |
| /// |
| /// {@tool dartpad} |
| /// This example demonstrates how to use [showBottomSheet] to display a |
| /// bottom sheet when a user taps a button. It also demonstrates how to |
| /// close a bottom sheet using the Navigator. |
| /// |
| /// ** See code in examples/api/lib/material/scaffold/scaffold_state.show_bottom_sheet.0.dart ** |
| /// {@end-tool} |
| /// See also: |
| /// |
| /// * [BottomSheet], which becomes the parent of the widget returned by the |
| /// `builder`. |
| /// * [showBottomSheet], which calls this method given a [BuildContext]. |
| /// * [showModalBottomSheet], which can be used to display a modal bottom |
| /// sheet. |
| /// * [Scaffold.of], for information about how to obtain the [ScaffoldState]. |
| /// * The Material 2 spec at <https://m2.material.io/components/sheets-bottom>. |
| /// * The Material 3 spec at <https://m3.material.io/components/bottom-sheets/overview>. |
| PersistentBottomSheetController<T> showBottomSheet<T>( |
| WidgetBuilder builder, { |
| Color? backgroundColor, |
| double? elevation, |
| ShapeBorder? shape, |
| Clip? clipBehavior, |
| BoxConstraints? constraints, |
| bool? enableDrag, |
| AnimationController? transitionAnimationController, |
| }) { |
| assert(() { |
| if (widget.bottomSheet != null) { |
| throw FlutterError( |
| 'Scaffold.bottomSheet cannot be specified while a bottom sheet ' |
| 'displayed with showBottomSheet() is still visible.\n' |
| 'Rebuild the Scaffold with a null bottomSheet before calling showBottomSheet().', |
| ); |
| } |
| return true; |
| }()); |
| assert(debugCheckHasMediaQuery(context)); |
| |
| _closeCurrentBottomSheet(); |
| final AnimationController controller = (transitionAnimationController ?? BottomSheet.createAnimationController(this))..forward(); |
| setState(() { |
| _currentBottomSheet = _buildBottomSheet<T>( |
| builder, |
| isPersistent: false, |
| animationController: controller, |
| backgroundColor: backgroundColor, |
| elevation: elevation, |
| shape: shape, |
| clipBehavior: clipBehavior, |
| constraints: constraints, |
| enableDrag: enableDrag, |
| shouldDisposeAnimationController: transitionAnimationController == null, |
| ); |
| }); |
| return _currentBottomSheet! as PersistentBottomSheetController<T>; |
| } |
| |
| // Floating Action Button API |
| late AnimationController _floatingActionButtonMoveController; |
| late FloatingActionButtonAnimator _floatingActionButtonAnimator; |
| FloatingActionButtonLocation? _previousFloatingActionButtonLocation; |
| FloatingActionButtonLocation? _floatingActionButtonLocation; |
| |
| late AnimationController _floatingActionButtonVisibilityController; |
| |
| /// Gets the current value of the visibility animation for the |
| /// [Scaffold.floatingActionButton]. |
| double get _floatingActionButtonVisibilityValue => _floatingActionButtonVisibilityController.value; |
| |
| /// Sets the current value of the visibility animation for the |
| /// [Scaffold.floatingActionButton]. This value must not be null. |
| set _floatingActionButtonVisibilityValue(double newValue) { |
| _floatingActionButtonVisibilityController.value = clampDouble(newValue, |
| _floatingActionButtonVisibilityController.lowerBound, |
| _floatingActionButtonVisibilityController.upperBound, |
| ); |
| } |
| |
| /// Shows the [Scaffold.floatingActionButton]. |
| TickerFuture _showFloatingActionButton() { |
| return _floatingActionButtonVisibilityController.forward(); |
| } |
| |
| // Moves the Floating Action Button to the new Floating Action Button Location. |
| void _moveFloatingActionButton(final FloatingActionButtonLocation newLocation) { |
| FloatingActionButtonLocation? previousLocation = _floatingActionButtonLocation; |
| double restartAnimationFrom = 0.0; |
| // If the Floating Action Button is moving right now, we need to start from a snapshot of the current transition. |
| if (_floatingActionButtonMoveController.isAnimating) { |
| previousLocation = _TransitionSnapshotFabLocation(_previousFloatingActionButtonLocation!, _floatingActionButtonLocation!, _floatingActionButtonAnimator, _floatingActionButtonMoveController.value); |
| restartAnimationFrom = _floatingActionButtonAnimator.getAnimationRestart(_floatingActionButtonMoveController.value); |
| } |
| |
| setState(() { |
| _previousFloatingActionButtonLocation = previousLocation; |
| _floatingActionButtonLocation = newLocation; |
| }); |
| |
| // Animate the motion even when the fab is null so that if the exit animation is running, |
| // the old fab will start the motion transition while it exits instead of jumping to the |
| // new position. |
| _floatingActionButtonMoveController.forward(from: restartAnimationFrom); |
| } |
| |
| // iOS FEATURES - status bar tap, back gesture |
| |
| // On iOS, tapping the status bar scrolls the app's primary scrollable to the |
| // top. We implement this by looking up the primary scroll controller and |
| // scrolling it to the top when tapped. |
| void _handleStatusBarTap() { |
| final ScrollController? primaryScrollController = PrimaryScrollController.maybeOf(context); |
| if (primaryScrollController != null && primaryScrollController.hasClients) { |
| primaryScrollController.animateTo( |
| 0.0, |
| duration: const Duration(milliseconds: 1000), |
| curve: Curves.easeOutCirc, |
| ); |
| } |
| } |
| |
| // INTERNALS |
| |
| late _ScaffoldGeometryNotifier _geometryNotifier; |
| |
| bool get _resizeToAvoidBottomInset { |
| return widget.resizeToAvoidBottomInset ?? true; |
| } |
| |
| @override |
| void initState() { |
| super.initState(); |
| _geometryNotifier = _ScaffoldGeometryNotifier(const ScaffoldGeometry(), context); |
| _floatingActionButtonLocation = widget.floatingActionButtonLocation ?? _kDefaultFloatingActionButtonLocation; |
| _floatingActionButtonAnimator = widget.floatingActionButtonAnimator ?? _kDefaultFloatingActionButtonAnimator; |
| _previousFloatingActionButtonLocation = _floatingActionButtonLocation; |
| _floatingActionButtonMoveController = AnimationController( |
| vsync: this, |
| value: 1.0, |
| duration: kFloatingActionButtonSegue * 2, |
| ); |
| |
| _floatingActionButtonVisibilityController = AnimationController( |
| duration: kFloatingActionButtonSegue, |
| vsync: this, |
| ); |
| } |
| |
| @override |
| void didUpdateWidget(Scaffold oldWidget) { |
| super.didUpdateWidget(oldWidget); |
| // Update the Floating Action Button Animator, and then schedule the Floating Action Button for repositioning. |
| if (widget.floatingActionButtonAnimator != oldWidget.floatingActionButtonAnimator) { |
| _floatingActionButtonAnimator = widget.floatingActionButtonAnimator ?? _kDefaultFloatingActionButtonAnimator; |
| } |
| if (widget.floatingActionButtonLocation != oldWidget.floatingActionButtonLocation) { |
| _moveFloatingActionButton(widget.floatingActionButtonLocation ?? _kDefaultFloatingActionButtonLocation); |
| } |
| if (widget.bottomSheet != oldWidget.bottomSheet) { |
| assert(() { |
| if (widget.bottomSheet != null && (_currentBottomSheet?._isLocalHistoryEntry ?? false)) { |
| throw FlutterError.fromParts(<DiagnosticsNode>[ |
| ErrorSummary( |
| 'Scaffold.bottomSheet cannot be specified while a bottom sheet displayed ' |
| 'with showBottomSheet() is still visible.', |
| ), |
| ErrorHint( |
| 'Use the PersistentBottomSheetController ' |
| 'returned by showBottomSheet() to close the old bottom sheet before creating ' |
| 'a Scaffold with a (non null) bottomSheet.', |
| ), |
| ]); |
| } |
| return true; |
| }()); |
| if (widget.bottomSheet == null) { |
| _closeCurrentBottomSheet(); |
| } else if (widget.bottomSheet != null && oldWidget.bottomSheet == null) { |
| _maybeBuildPersistentBottomSheet(); |
| } else { |
| _updatePersistentBottomSheet(); |
| } |
| } |
| } |
| |
| @override |
| void didChangeDependencies() { |
| // Using maybeOf is valid here since both the Scaffold and ScaffoldMessenger |
| // are currently available for managing SnackBars. |
| final ScaffoldMessengerState? currentScaffoldMessenger = ScaffoldMessenger.maybeOf(context); |
| // If our ScaffoldMessenger has changed, unregister with the old one first. |
| if (_scaffoldMessenger != null && |
| (currentScaffoldMessenger == null || _scaffoldMessenger != currentScaffoldMessenger)) { |
| _scaffoldMessenger?._unregister(this); |
| } |
| // Register with the current ScaffoldMessenger, if there is one. |
| _scaffoldMessenger = currentScaffoldMessenger; |
| _scaffoldMessenger?._register(this); |
| |
| _maybeBuildPersistentBottomSheet(); |
| super.didChangeDependencies(); |
| } |
| |
| @override |
| void dispose() { |
| _geometryNotifier.dispose(); |
| _floatingActionButtonMoveController.dispose(); |
| _floatingActionButtonVisibilityController.dispose(); |
| _scaffoldMessenger?._unregister(this); |
| _drawerOpened.dispose(); |
| _endDrawerOpened.dispose(); |
| super.dispose(); |
| } |
| |
| void _addIfNonNull( |
| List<LayoutId> children, |
| Widget? child, |
| Object childId, { |
| required bool removeLeftPadding, |
| required bool removeTopPadding, |
| required bool removeRightPadding, |
| required bool removeBottomPadding, |
| bool removeBottomInset = false, |
| bool maintainBottomViewPadding = false, |
| }) { |
| MediaQueryData data = MediaQuery.of(context).removePadding( |
| removeLeft: removeLeftPadding, |
| removeTop: removeTopPadding, |
| removeRight: removeRightPadding, |
| removeBottom: removeBottomPadding, |
| ); |
| if (removeBottomInset) { |
| data = data.removeViewInsets(removeBottom: true); |
| } |
| |
| if (maintainBottomViewPadding && data.viewInsets.bottom != 0.0) { |
| data = data.copyWith( |
| padding: data.padding.copyWith(bottom: data.viewPadding.bottom), |
| ); |
| } |
| |
| if (child != null) { |
| children.add( |
| LayoutId( |
| id: childId, |
| child: MediaQuery(data: data, child: child), |
| ), |
| ); |
| } |
| } |
| |
| void _buildEndDrawer(List<LayoutId> children, TextDirection textDirection) { |
| if (widget.endDrawer != null) { |
| assert(hasEndDrawer); |
| _addIfNonNull( |
| children, |
| DrawerController( |
| key: _endDrawerKey, |
| alignment: DrawerAlignment.end, |
| drawerCallback: _endDrawerOpenedCallback, |
| dragStartBehavior: widget.drawerDragStartBehavior, |
| scrimColor: widget.drawerScrimColor, |
| edgeDragWidth: widget.drawerEdgeDragWidth, |
| enableOpenDragGesture: widget.endDrawerEnableOpenDragGesture, |
| isDrawerOpen: _endDrawerOpened.value, |
| child: widget.endDrawer!, |
| ), |
| _ScaffoldSlot.endDrawer, |
| // remove the side padding from the side we're not touching |
| removeLeftPadding: textDirection == TextDirection.ltr, |
| removeTopPadding: false, |
| removeRightPadding: textDirection == TextDirection.rtl, |
| removeBottomPadding: false, |
| ); |
| } |
| } |
| |
| void _buildDrawer(List<LayoutId> children, TextDirection textDirection) { |
| if (widget.drawer != null) { |
| assert(hasDrawer); |
| _addIfNonNull( |
| children, |
| DrawerController( |
| key: _drawerKey, |
| alignment: DrawerAlignment.start, |
| drawerCallback: _drawerOpenedCallback, |
| dragStartBehavior: widget.drawerDragStartBehavior, |
| scrimColor: widget.drawerScrimColor, |
| edgeDragWidth: widget.drawerEdgeDragWidth, |
| enableOpenDragGesture: widget.drawerEnableOpenDragGesture, |
| isDrawerOpen: _drawerOpened.value, |
| child: widget.drawer!, |
| ), |
| _ScaffoldSlot.drawer, |
| // remove the side padding from the side we're not touching |
| removeLeftPadding: textDirection == TextDirection.rtl, |
| removeTopPadding: false, |
| removeRightPadding: textDirection == TextDirection.ltr, |
| removeBottomPadding: false, |
| ); |
| } |
| } |
| |
| bool _showBodyScrim = false; |
| Color _bodyScrimColor = Colors.black; |
| |
| /// Whether to show a [ModalBarrier] over the body of the scaffold. |
| /// |
| /// The `value` parameter must not be null. |
| void showBodyScrim(bool value, double opacity) { |
| if (_showBodyScrim == value && _bodyScrimColor.opacity == opacity) { |
| return; |
| } |
| setState(() { |
| _showBodyScrim = value; |
| _bodyScrimColor = Colors.black.withOpacity(opacity); |
| }); |
| } |
| |
| @override |
| Widget build(BuildContext context) { |
| assert(debugCheckHasMediaQuery(context)); |
| assert(debugCheckHasDirectionality(context)); |
| final ThemeData themeData = Theme.of(context); |
| final TextDirection textDirection = Directionality.of(context); |
| |
| final List<LayoutId> children = <LayoutId>[]; |
| _addIfNonNull( |
| children, |
| widget.body == null ? null : _BodyBuilder( |
| extendBody: widget.extendBody, |
| extendBodyBehindAppBar: widget.extendBodyBehindAppBar, |
| body: KeyedSubtree(key: _bodyKey, child: widget.body!), |
| ), |
| _ScaffoldSlot.body, |
| removeLeftPadding: false, |
| removeTopPadding: widget.appBar != null, |
| removeRightPadding: false, |
| removeBottomPadding: widget.bottomNavigationBar != null || widget.persistentFooterButtons != null, |
| removeBottomInset: _resizeToAvoidBottomInset, |
| ); |
| if (_showBodyScrim) { |
| _addIfNonNull( |
| children, |
| ModalBarrier( |
| dismissible: false, |
| color: _bodyScrimColor, |
| ), |
| _ScaffoldSlot.bodyScrim, |
| removeLeftPadding: true, |
| removeTopPadding: true, |
| removeRightPadding: true, |
| removeBottomPadding: true, |
| ); |
| } |
| |
| if (widget.appBar != null) { |
| final double topPadding = widget.primary ? MediaQuery.paddingOf(context).top : 0.0; |
| _appBarMaxHeight = AppBar.preferredHeightFor(context, widget.appBar!.preferredSize) + topPadding; |
| assert(_appBarMaxHeight! >= 0.0 && _appBarMaxHeight!.isFinite); |
| _addIfNonNull( |
| children, |
| ConstrainedBox( |
| constraints: BoxConstraints(maxHeight: _appBarMaxHeight!), |
| child: FlexibleSpaceBar.createSettings( |
| currentExtent: _appBarMaxHeight!, |
| child: widget.appBar!, |
| ), |
| ), |
| _ScaffoldSlot.appBar, |
| removeLeftPadding: false, |
| removeTopPadding: false, |
| removeRightPadding: false, |
| removeBottomPadding: true, |
| ); |
| } |
| |
| bool isSnackBarFloating = false; |
| double? snackBarWidth; |
| |
| if (_currentBottomSheet != null || _dismissedBottomSheets.isNotEmpty) { |
| final Widget stack = Stack( |
| alignment: Alignment.bottomCenter, |
| children: <Widget>[ |
| ..._dismissedBottomSheets, |
| if (_currentBottomSheet != null) _currentBottomSheet!._widget, |
| ], |
| ); |
| _addIfNonNull( |
| children, |
| stack, |
| _ScaffoldSlot.bottomSheet, |
| removeLeftPadding: false, |
| removeTopPadding: true, |
| removeRightPadding: false, |
| removeBottomPadding: _resizeToAvoidBottomInset, |
| ); |
| } |
| |
| // SnackBar set by ScaffoldMessenger |
| if (_messengerSnackBar != null) { |
| final SnackBarBehavior snackBarBehavior = _messengerSnackBar?._widget.behavior |
| ?? themeData.snackBarTheme.behavior |
| ?? SnackBarBehavior.fixed; |
| isSnackBarFloating = snackBarBehavior == SnackBarBehavior.floating; |
| snackBarWidth = _messengerSnackBar?._widget.width ?? themeData.snackBarTheme.width; |
| |
| _addIfNonNull( |
| children, |
| _messengerSnackBar?._widget, |
| _ScaffoldSlot.snackBar, |
| removeLeftPadding: false, |
| removeTopPadding: true, |
| removeRightPadding: false, |
| removeBottomPadding: widget.bottomNavigationBar != null || widget.persistentFooterButtons != null, |
| maintainBottomViewPadding: !_resizeToAvoidBottomInset, |
| ); |
| } |
| |
| bool extendBodyBehindMaterialBanner = false; |
| // MaterialBanner set by ScaffoldMessenger |
| if (_messengerMaterialBanner != null) { |
| final MaterialBannerThemeData bannerTheme = MaterialBannerTheme.of(context); |
| final double elevation = _messengerMaterialBanner?._widget.elevation ?? bannerTheme.elevation ?? 0.0; |
| extendBodyBehindMaterialBanner = elevation != 0.0; |
| |
| _addIfNonNull( |
| children, |
| _messengerMaterialBanner?._widget, |
| _ScaffoldSlot.materialBanner, |
| removeLeftPadding: false, |
| removeTopPadding: widget.appBar != null, |
| removeRightPadding: false, |
| removeBottomPadding: true, |
| maintainBottomViewPadding: !_resizeToAvoidBottomInset, |
| ); |
| } |
| |
| if (widget.persistentFooterButtons != null) { |
| _addIfNonNull( |
| children, |
| Container( |
| decoration: BoxDecoration( |
| border: Border( |
| top: Divider.createBorderSide(context, width: 1.0), |
| ), |
| ), |
| child: SafeArea( |
| top: false, |
| child: IntrinsicHeight( |
| child: Container( |
| alignment: widget.persistentFooterAlignment, |
| padding: const EdgeInsets.all(8), |
| child: OverflowBar( |
| spacing: 8, |
| overflowAlignment: OverflowBarAlignment.end, |
| children: widget.persistentFooterButtons!, |
| ), |
| ), |
| ), |
| ), |
| ), |
| _ScaffoldSlot.persistentFooter, |
| removeLeftPadding: false, |
| removeTopPadding: true, |
| removeRightPadding: false, |
| removeBottomPadding: widget.bottomNavigationBar != null, |
| maintainBottomViewPadding: !_resizeToAvoidBottomInset, |
| ); |
| } |
| |
| if (widget.bottomNavigationBar != null) { |
| _addIfNonNull( |
| children, |
| widget.bottomNavigationBar, |
| _ScaffoldSlot.bottomNavigationBar, |
| removeLeftPadding: false, |
| removeTopPadding: true, |
| removeRightPadding: false, |
| removeBottomPadding: false, |
| maintainBottomViewPadding: !_resizeToAvoidBottomInset, |
| ); |
| } |
| |
| _addIfNonNull( |
| children, |
| _FloatingActionButtonTransition( |
| fabMoveAnimation: _floatingActionButtonMoveController, |
| fabMotionAnimator: _floatingActionButtonAnimator, |
| geometryNotifier: _geometryNotifier, |
| currentController: _floatingActionButtonVisibilityController, |
| child: widget.floatingActionButton, |
| ), |
| _ScaffoldSlot.floatingActionButton, |
| removeLeftPadding: true, |
| removeTopPadding: true, |
| removeRightPadding: true, |
| removeBottomPadding: true, |
| ); |
| |
| switch (themeData.platform) { |
| case TargetPlatform.iOS: |
| case TargetPlatform.macOS: |
| _addIfNonNull( |
| children, |
| GestureDetector( |
| behavior: HitTestBehavior.opaque, |
| onTap: _handleStatusBarTap, |
| // iOS accessibility automatically adds scroll-to-top to the clock in the status bar |
| excludeFromSemantics: true, |
| ), |
| _ScaffoldSlot.statusBar, |
| removeLeftPadding: false, |
| removeTopPadding: true, |
| removeRightPadding: false, |
| removeBottomPadding: true, |
| ); |
| case TargetPlatform.android: |
| case TargetPlatform.fuchsia: |
| case TargetPlatform.linux: |
| case TargetPlatform.windows: |
| break; |
| } |
| |
| if (_endDrawerOpened.value) { |
| _buildDrawer(children, textDirection); |
| _buildEndDrawer(children, textDirection); |
| } else { |
| _buildEndDrawer(children, textDirection); |
| _buildDrawer(children, textDirection); |
| } |
| |
| // The minimum insets for contents of the Scaffold to keep visible. |
| final EdgeInsets minInsets = MediaQuery.paddingOf(context).copyWith( |
| bottom: _resizeToAvoidBottomInset ? MediaQuery.viewInsetsOf(context).bottom : 0.0, |
| ); |
| |
| // The minimum viewPadding for interactive elements positioned by the |
| // Scaffold to keep within safe interactive areas. |
| final EdgeInsets minViewPadding = MediaQuery.viewPaddingOf(context).copyWith( |
| bottom: _resizeToAvoidBottomInset && MediaQuery.viewInsetsOf(context).bottom != 0.0 ? 0.0 : null, |
| ); |
| |
| // extendBody locked when keyboard is open |
| final bool extendBody = minInsets.bottom <= 0 && widget.extendBody; |
| |
| return _ScaffoldScope( |
| hasDrawer: hasDrawer, |
| geometryNotifier: _geometryNotifier, |
| child: ScrollNotificationObserver( |
| child: Material( |
| color: widget.backgroundColor ?? themeData.scaffoldBackgroundColor, |
| child: AnimatedBuilder(animation: _floatingActionButtonMoveController, builder: (BuildContext context, Widget? child) { |
| return Actions( |
| actions: <Type, Action<Intent>>{ |
| DismissIntent: _DismissDrawerAction(context), |
| }, |
| child: CustomMultiChildLayout( |
| delegate: _ScaffoldLayout( |
| extendBody: extendBody, |
| extendBodyBehindAppBar: widget.extendBodyBehindAppBar, |
| minInsets: minInsets, |
| minViewPadding: minViewPadding, |
| currentFloatingActionButtonLocation: _floatingActionButtonLocation!, |
| floatingActionButtonMoveAnimationProgress: _floatingActionButtonMoveController.value, |
| floatingActionButtonMotionAnimator: _floatingActionButtonAnimator, |
| geometryNotifier: _geometryNotifier, |
| previousFloatingActionButtonLocation: _previousFloatingActionButtonLocation!, |
| textDirection: textDirection, |
| isSnackBarFloating: isSnackBarFloating, |
| extendBodyBehindMaterialBanner: extendBodyBehindMaterialBanner, |
| snackBarWidth: snackBarWidth, |
| ), |
| children: children, |
| ), |
| ); |
| }), |
| ), |
| ), |
| ); |
| } |
| } |
| |
| class _DismissDrawerAction extends DismissAction { |
| _DismissDrawerAction(this.context); |
| |
| final BuildContext context; |
| |
| @override |
| bool isEnabled(DismissIntent intent) { |
| return Scaffold.of(context).isDrawerOpen || Scaffold.of(context).isEndDrawerOpen; |
| } |
| |
| @override |
| void invoke(DismissIntent intent) { |
| Scaffold.of(context).closeDrawer(); |
| Scaffold.of(context).closeEndDrawer(); |
| } |
| } |
| |
| /// An interface for controlling a feature of a [Scaffold]. |
| /// |
| /// Commonly obtained from [ScaffoldMessengerState.showSnackBar] or |
| /// [ScaffoldState.showBottomSheet]. |
| class ScaffoldFeatureController<T extends Widget, U> { |
| const ScaffoldFeatureController._(this._widget, this._completer, this.close, this.setState); |
| final T _widget; |
| final Completer<U> _completer; |
| |
| /// Completes when the feature controlled by this object is no longer visible. |
| Future<U> get closed => _completer.future; |
| |
| /// Remove the feature (e.g., bottom sheet, snack bar, or material banner) from the scaffold. |
| final VoidCallback close; |
| |
| /// Mark the feature (e.g., bottom sheet or snack bar) as needing to rebuild. |
| final StateSetter? setState; |
| } |
| |
| // TODO(guidezpl): Look into making this public. A copy of this class is in |
| // bottom_sheet.dart, for now, https://github.com/flutter/flutter/issues/51627 |
| /// A curve that progresses linearly until a specified [startingPoint], at which |
| /// point [curve] will begin. Unlike [Interval], [curve] will not start at zero, |
| /// but will use [startingPoint] as the Y position. |
| /// |
| /// For example, if [startingPoint] is set to `0.5`, and [curve] is set to |
| /// [Curves.easeOut], then the bottom-left quarter of the curve will be a |
| /// straight line, and the top-right quarter will contain the entire contents of |
| /// [Curves.easeOut]. |
| /// |
| /// This is useful in situations where a widget must track the user's finger |
| /// (which requires a linear animation), and afterwards can be flung using a |
| /// curve specified with the [curve] argument, after the finger is released. In |
| /// such a case, the value of [startingPoint] would be the progress of the |
| /// animation at the time when the finger was released. |
| /// |
| /// The [startingPoint] and [curve] arguments must not be null. |
| class _BottomSheetSuspendedCurve extends ParametricCurve<double> { |
| /// Creates a suspended curve. |
| const _BottomSheetSuspendedCurve( |
| this.startingPoint, { |
| this.curve = Curves.easeOutCubic, |
| }); |
| |
| /// The progress value at which [curve] should begin. |
| /// |
| /// This defaults to [Curves.easeOutCubic]. |
| final double startingPoint; |
| |
| /// The curve to use when [startingPoint] is reached. |
| final Curve curve; |
| |
| @override |
| double transform(double t) { |
| assert(t >= 0.0 && t <= 1.0); |
| assert(startingPoint >= 0.0 && startingPoint <= 1.0); |
| |
| if (t < startingPoint) { |
| return t; |
| } |
| |
| if (t == 1.0) { |
| return t; |
| } |
| |
| final double curveProgress = (t - startingPoint) / (1 - startingPoint); |
| final double transformed = curve.transform(curveProgress); |
| return lerpDouble(startingPoint, 1, transformed)!; |
| } |
| |
| @override |
| String toString() { |
| return '${describeIdentity(this)}($startingPoint, $curve)'; |
| } |
| } |
| |
| class _StandardBottomSheet extends StatefulWidget { |
| const _StandardBottomSheet({ |
| super.key, |
| required this.animationController, |
| this.enableDrag = true, |
| required this.onClosing, |
| required this.onDismissed, |
| required this.builder, |
| this.isPersistent = false, |
| this.backgroundColor, |
| this.elevation, |
| this.shape, |
| this.clipBehavior, |
| this.constraints, |
| this.onDispose, |
| }); |
| |
| final AnimationController animationController; // we control it, but it must be disposed by whoever created it. |
| final bool enableDrag; |
| final VoidCallback? onClosing; |
| final VoidCallback? onDismissed; |
| final VoidCallback? onDispose; |
| final WidgetBuilder builder; |
| final bool isPersistent; |
| final Color? backgroundColor; |
| final double? elevation; |
| final ShapeBorder? shape; |
| final Clip? clipBehavior; |
| final BoxConstraints? constraints; |
| |
| @override |
| _StandardBottomSheetState createState() => _StandardBottomSheetState(); |
| } |
| |
| class _StandardBottomSheetState extends State<_StandardBottomSheet> { |
| ParametricCurve<double> animationCurve = _standardBottomSheetCurve; |
| |
| @override |
| void initState() { |
| super.initState(); |
| assert( |
| widget.animationController.status == AnimationStatus.forward |
| || widget.animationController.status == AnimationStatus.completed, |
| ); |
| widget.animationController.addStatusListener(_handleStatusChange); |
| } |
| |
| @override |
| void dispose() { |
| widget.onDispose?.call(); |
| super.dispose(); |
| } |
| |
| @override |
| void didUpdateWidget(_StandardBottomSheet oldWidget) { |
| super.didUpdateWidget(oldWidget); |
| assert(widget.animationController == oldWidget.animationController); |
| } |
| |
| void close() { |
| widget.animationController.reverse(); |
| widget.onClosing?.call(); |
| } |
| |
| void _handleDragStart(DragStartDetails details) { |
| // Allow the bottom sheet to track the user's finger accurately. |
| animationCurve = Curves.linear; |
| } |
| |
| void _handleDragEnd(DragEndDetails details, { bool? isClosing }) { |
| // Allow the bottom sheet to animate smoothly from its current position. |
| animationCurve = _BottomSheetSuspendedCurve( |
| widget.animationController.value, |
| curve: _standardBottomSheetCurve, |
| ); |
| } |
| |
| void _handleStatusChange(AnimationStatus status) { |
| if (status == AnimationStatus.dismissed) { |
| widget.onDismissed?.call(); |
| } |
| } |
| |
| bool extentChanged(DraggableScrollableNotification notification) { |
| final double extentRemaining = 1.0 - notification.extent; |
| final ScaffoldState scaffold = Scaffold.of(context); |
| if (extentRemaining < _kBottomSheetDominatesPercentage) { |
| scaffold._floatingActionButtonVisibilityValue = extentRemaining * _kBottomSheetDominatesPercentage * 10; |
| scaffold.showBodyScrim(true, math.max( |
| _kMinBottomSheetScrimOpacity, |
| _kMaxBottomSheetScrimOpacity - scaffold._floatingActionButtonVisibilityValue, |
| )); |
| } else { |
| scaffold._floatingActionButtonVisibilityValue = 1.0; |
| scaffold.showBodyScrim(false |