// 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:convert'; import 'dart:developer' as developer; import 'dart:ui' as ui; import 'package:flutter/foundation.dart'; import 'package:flutter/rendering.dart'; import 'package:flutter/scheduler.dart'; import 'package:flutter/services.dart'; import 'basic.dart'; import 'binding.dart'; import 'focus_manager.dart'; import 'focus_scope.dart'; import 'focus_traversal.dart'; import 'framework.dart'; import 'heroes.dart'; import 'notification_listener.dart'; import 'overlay.dart'; import 'restoration.dart'; import 'restoration_properties.dart'; import 'routes.dart'; import 'ticker_provider.dart'; // Duration for delay before refocusing in android so that the focus won't be interrupted. const Duration _kAndroidRefocusingDelayDuration = Duration(milliseconds: 300); // Examples can assume: // typedef MyAppHome = Placeholder; // typedef MyHomePage = Placeholder; // typedef MyPage = ListTile; // any const widget with a Widget "title" constructor argument would do // late NavigatorState navigator; // late BuildContext context; /// Creates a route for the given route settings. /// /// Used by [Navigator.onGenerateRoute]. /// /// See also: /// /// * [Navigator], which is where all the [Route]s end up. typedef RouteFactory = Route? Function(RouteSettings settings); /// Creates a series of one or more routes. /// /// Used by [Navigator.onGenerateInitialRoutes]. typedef RouteListFactory = List> Function(NavigatorState navigator, String initialRoute); /// Creates a [Route] that is to be added to a [Navigator]. /// /// The route can be configured with the provided `arguments`. The provided /// `context` is the `BuildContext` of the [Navigator] to which the route is /// added. /// /// Used by the restorable methods of the [Navigator] that add anonymous routes /// (e.g. [NavigatorState.restorablePush]). For this use case, the /// [RestorableRouteBuilder] must be static function annotated with /// `@pragma('vm:entry-point')`. The [Navigator] will call it again during /// state restoration to re-create the route. typedef RestorableRouteBuilder = Route Function(BuildContext context, Object? arguments); /// Signature for the [Navigator.popUntil] predicate argument. typedef RoutePredicate = bool Function(Route route); /// Signature for a callback that verifies that it's OK to call [Navigator.pop]. /// /// Used by [Form.onWillPop], [ModalRoute.addScopedWillPopCallback], /// [ModalRoute.removeScopedWillPopCallback], and [WillPopScope]. @Deprecated( 'Use PopInvokedCallback instead. ' 'This feature was deprecated after v3.12.0-1.0.pre.', ) typedef WillPopCallback = Future Function(); /// Signature for the [Navigator.onPopPage] callback. /// /// This callback must call [Route.didPop] on the specified route and must /// properly update the pages list the next time it is passed into /// [Navigator.pages] so that it no longer includes the corresponding [Page]. /// (Otherwise, the page will be interpreted as a new page to show when the /// [Navigator.pages] list is next updated.) typedef PopPageCallback = bool Function(Route route, dynamic result); /// Indicates whether the current route should be popped. /// /// Used as the return value for [Route.willPop]. /// /// See also: /// /// * [WillPopScope], a widget that hooks into the route's [Route.willPop] /// mechanism. enum RoutePopDisposition { /// Pop the route. /// /// If [Route.willPop] or [Route.popDisposition] return [pop] then the back /// button will actually pop the current route. pop, /// Do not pop the route. /// /// If [Route.willPop] or [Route.popDisposition] return [doNotPop] then the /// back button will be ignored. doNotPop, /// Delegate this to the next level of navigation. /// /// If [Route.willPop] or [Route.popDisposition] return [bubble] then the back /// button will be handled by the [SystemNavigator], which will usually close /// the application. bubble, } /// An abstraction for an entry managed by a [Navigator]. /// /// This class defines an abstract interface between the navigator and the /// "routes" that are pushed on and popped off the navigator. Most routes have /// visual affordances, which they place in the navigators [Overlay] using one /// or more [OverlayEntry] objects. /// /// See [Navigator] for more explanation of how to use a [Route] with /// navigation, including code examples. /// /// See [MaterialPageRoute] for a route that replaces the entire screen with a /// platform-adaptive transition. /// /// A route can belong to a page if the [settings] are a subclass of [Page]. A /// page-based route, as opposed to a pageless route, is created from /// [Page.createRoute] during [Navigator.pages] updates. The page associated /// with this route may change during the lifetime of the route. If the /// [Navigator] updates the page of this route, it calls [changedInternalState] /// to notify the route that the page has been updated. /// /// The type argument `T` is the route's return type, as used by /// [currentResult], [popped], and [didPop]. The type `void` may be used if the /// route does not return a value. abstract class Route { /// Initialize the [Route]. /// /// If the [settings] are not provided, an empty [RouteSettings] object is /// used instead. Route({ RouteSettings? settings }) : _settings = settings ?? const RouteSettings() { if (kFlutterMemoryAllocationsEnabled) { FlutterMemoryAllocations.instance.dispatchObjectCreated( library: 'package:flutter/widgets.dart', className: '$Route<$T>', object: this, ); } } /// The navigator that the route is in, if any. NavigatorState? get navigator => _navigator; NavigatorState? _navigator; /// The settings for this route. /// /// See [RouteSettings] for details. /// /// The settings can change during the route's lifetime. If the settings /// change, the route's overlays will be marked dirty (see /// [changedInternalState]). /// /// If the route is created from a [Page] in the [Navigator.pages] list, then /// this will be a [Page] subclass, and it will be updated each time its /// corresponding [Page] in the [Navigator.pages] has changed. Once the /// [Route] is removed from the history, this value stops updating (and /// remains with its last value). RouteSettings get settings => _settings; RouteSettings _settings; /// The restoration scope ID to be used for the [RestorationScope] surrounding /// this route. /// /// The restoration scope ID is null if restoration is currently disabled /// for this route. /// /// If the restoration scope ID changes (e.g. because restoration is enabled /// or disabled) during the life of the route, the [ValueListenable] notifies /// its listeners. As an example, the ID changes to null while the route is /// transitioning off screen, which triggers a notification on this field. At /// that point, the route is considered as no longer present for restoration /// purposes and its state will not be restored. ValueListenable get restorationScopeId => _restorationScopeId; final ValueNotifier _restorationScopeId = ValueNotifier(null); void _updateSettings(RouteSettings newSettings) { if (_settings != newSettings) { _settings = newSettings; changedInternalState(); } } // ignore: use_setters_to_change_properties, (setters can't be private) void _updateRestorationId(String? restorationId) { _restorationScopeId.value = restorationId; } /// The overlay entries of this route. /// /// These are typically populated by [install]. The [Navigator] is in charge /// of adding them to and removing them from the [Overlay]. /// /// There must be at least one entry in this list after [install] has been /// invoked. /// /// The [Navigator] will take care of keeping the entries together if the /// route is moved in the history. List get overlayEntries => const []; /// Called when the route is inserted into the navigator. /// /// Uses this to populate [overlayEntries]. There must be at least one entry in /// this list after [install] has been invoked. The [Navigator] will be in charge /// to add them to the [Overlay] or remove them from it by calling /// [OverlayEntry.remove]. @protected @mustCallSuper void install() { } /// Called after [install] when the route is pushed onto the navigator. /// /// The returned value resolves when the push transition is complete. /// /// The [didAdd] method will be called instead of [didPush] when the route /// immediately appears on screen without any push transition. /// /// The [didChangeNext] and [didChangePrevious] methods are typically called /// immediately after this method is called. @protected @mustCallSuper TickerFuture didPush() { return TickerFuture.complete()..then((void _) { if (navigator?.widget.requestFocus ?? false) { navigator!.focusNode.enclosingScope?.requestFocus(); } }); } /// Called after [install] when the route is added to the navigator. /// /// This method is called instead of [didPush] when the route immediately /// appears on screen without any push transition. /// /// The [didChangeNext] and [didChangePrevious] methods are typically called /// immediately after this method is called. @protected @mustCallSuper void didAdd() { if (navigator?.widget.requestFocus ?? false) { // This TickerFuture serves two purposes. First, we want to make sure that // animations triggered by other operations will finish before focusing // the navigator. Second, navigator.focusNode might acquire more focused // children in Route.install asynchronously. This TickerFuture will wait // for it to finish first. // // The later case can be found when subclasses manage their own focus scopes. // For example, ModalRoute creates a focus scope in its overlay entries. The // focused child can only be attached to navigator after initState which // will be guarded by the asynchronous gap. TickerFuture.complete().then((void _) { // The route can be disposed before the ticker future completes. This can // happen when the navigator is under a TabView that warps from one tab to // another, non-adjacent tab, with an animation. The TabView reorders its // children before and after the warping completes, and that causes its // children to be built and disposed within the same frame. If one of its // children contains a navigator, the routes in that navigator are also // added and disposed within that frame. // // Since the reference to the navigator will be set to null after it is // disposed, we have to do a null-safe operation in case that happens // within the same frame when it is added. navigator?.focusNode.enclosingScope?.requestFocus(); }); } } /// Called after [install] when the route replaced another in the navigator. /// /// The [didChangeNext] and [didChangePrevious] methods are typically called /// immediately after this method is called. @protected @mustCallSuper void didReplace(Route? oldRoute) { } /// Returns whether calling [Navigator.maybePop] when this [Route] is current /// ([isCurrent]) should do anything. /// /// [Navigator.maybePop] is usually used instead of [Navigator.pop] to handle /// the system back button. /// /// By default, if a [Route] is the first route in the history (i.e., if /// [isFirst]), it reports that pops should be bubbled /// ([RoutePopDisposition.bubble]). This behavior prevents the user from /// popping the first route off the history and being stranded at a blank /// screen; instead, the larger scope is popped (e.g. the application quits, /// so that the user returns to the previous application). /// /// In other cases, the default behavior is to accept the pop /// ([RoutePopDisposition.pop]). /// /// The third possible value is [RoutePopDisposition.doNotPop], which causes /// the pop request to be ignored entirely. /// /// See also: /// /// * [Form], which provides a [Form.onWillPop] callback that uses this /// mechanism. /// * [WillPopScope], another widget that provides a way to intercept the /// back button. @Deprecated( 'Use popDisposition instead. ' 'This feature was deprecated after v3.12.0-1.0.pre.', ) Future willPop() async { return isFirst ? RoutePopDisposition.bubble : RoutePopDisposition.pop; } /// Returns whether calling [Navigator.maybePop] when this [Route] is current /// ([isCurrent]) should do anything. /// /// [Navigator.maybePop] is usually used instead of [Navigator.pop] to handle /// the system back button, when it hasn't been disabled via /// [SystemNavigator.setFrameworkHandlesBack]. /// /// By default, if a [Route] is the first route in the history (i.e., if /// [isFirst]), it reports that pops should be bubbled /// ([RoutePopDisposition.bubble]). This behavior prevents the user from /// popping the first route off the history and being stranded at a blank /// screen; instead, the larger scope is popped (e.g. the application quits, /// so that the user returns to the previous application). /// /// In other cases, the default behavior is to accept the pop /// ([RoutePopDisposition.pop]). /// /// The third possible value is [RoutePopDisposition.doNotPop], which causes /// the pop request to be ignored entirely. /// /// See also: /// /// * [Form], which provides a [Form.canPop] boolean that is similar. /// * [PopScope], a widget that provides a way to intercept the back button. RoutePopDisposition get popDisposition { return isFirst ? RoutePopDisposition.bubble : RoutePopDisposition.pop; } /// {@template flutter.widgets.navigator.onPopInvoked} /// Called after a route pop was handled. /// /// Even when the pop is canceled, for example by a [PopScope] widget, this /// will still be called. The `didPop` parameter indicates whether or not the /// back navigation actually happened successfully. /// {@endtemplate} void onPopInvoked(bool didPop) {} /// Whether calling [didPop] would return false. bool get willHandlePopInternally => false; /// When this route is popped (see [Navigator.pop]) if the result isn't /// specified or if it's null, this value will be used instead. /// /// This fallback is implemented by [didComplete]. This value is used if the /// argument to that method is null. T? get currentResult => null; /// A future that completes when this route is popped off the navigator. /// /// The future completes with the value given to [Navigator.pop], if any, or /// else the value of [currentResult]. See [didComplete] for more discussion /// on this topic. Future get popped => _popCompleter.future; final Completer _popCompleter = Completer(); final Completer _disposeCompleter = Completer(); /// A request was made to pop this route. If the route can handle it /// internally (e.g. because it has its own stack of internal state) then /// return false, otherwise return true (by returning the value of calling /// `super.didPop`). Returning false will prevent the default behavior of /// [NavigatorState.pop]. /// /// When this function returns true, the navigator removes this route from /// the history but does not yet call [dispose]. Instead, it is the route's /// responsibility to call [NavigatorState.finalizeRoute], which will in turn /// call [dispose] on the route. This sequence lets the route perform an /// exit animation (or some other visual effect) after being popped but prior /// to being disposed. /// /// This method should call [didComplete] to resolve the [popped] future (and /// this is all that the default implementation does); routes should not wait /// for their exit animation to complete before doing so. /// /// See [popped], [didComplete], and [currentResult] for a discussion of the /// `result` argument. @mustCallSuper bool didPop(T? result) { didComplete(result); return true; } /// The route was popped or is otherwise being removed somewhat gracefully. /// /// This is called by [didPop] and in response to /// [NavigatorState.pushReplacement]. If [didPop] was not called, then the /// [NavigatorState.finalizeRoute] method must be called immediately, and no exit /// animation will run. /// /// The [popped] future is completed by this method. The `result` argument /// specifies the value that this future is completed with, unless it is null, /// in which case [currentResult] is used instead. /// /// This should be called before the pop animation, if any, takes place, /// though in some cases the animation may be driven by the user before the /// route is committed to being popped; this can in particular happen with the /// iOS-style back gesture. See [NavigatorState.didStartUserGesture]. @protected @mustCallSuper void didComplete(T? result) { _popCompleter.complete(result ?? currentResult); } /// The given route, which was above this one, has been popped off the /// navigator. /// /// This route is now the current route ([isCurrent] is now true), and there /// is no next route. @protected @mustCallSuper void didPopNext(Route nextRoute) { } /// This route's next route has changed to the given new route. /// /// This is called on a route whenever the next route changes for any reason, /// so long as it is in the history, including when a route is first added to /// a [Navigator] (e.g. by [Navigator.push]), except for cases when /// [didPopNext] would be called. /// /// The `nextRoute` argument will be null if there's no new next route (i.e. /// if [isCurrent] is true). @protected @mustCallSuper void didChangeNext(Route? nextRoute) { } /// This route's previous route has changed to the given new route. /// /// This is called on a route whenever the previous route changes for any /// reason, so long as it is in the history, except for immediately after the /// route itself has been pushed (in which case [didPush] or [didReplace] will /// be called instead). /// /// The `previousRoute` argument will be null if there's no previous route /// (i.e. if [isFirst] is true). @protected @mustCallSuper void didChangePrevious(Route? previousRoute) { } /// Called whenever the internal state of the route has changed. /// /// This should be called whenever [willHandlePopInternally], [didPop], /// [ModalRoute.offstage], or other internal state of the route changes value. /// It is used by [ModalRoute], for example, to report the new information via /// its inherited widget to any children of the route. /// /// See also: /// /// * [changedExternalState], which is called when the [Navigator] has /// updated in some manner that might affect the routes. @protected @mustCallSuper void changedInternalState() { } /// Called whenever the [Navigator] has updated in some manner that might /// affect routes, to indicate that the route may wish to rebuild as well. /// /// This is called by the [Navigator] whenever the /// [NavigatorState]'s [State.widget] changes (as in [State.didUpdateWidget]), /// for example because the [MaterialApp] has been rebuilt. This /// ensures that routes that directly refer to the state of the /// widget that built the [MaterialApp] will be notified when that /// widget rebuilds, since it would otherwise be difficult to notify /// the routes that state they depend on may have changed. /// /// It is also called whenever the [Navigator]'s dependencies change /// (as in [State.didChangeDependencies]). This allows routes to use the /// [Navigator]'s context ([NavigatorState.context]), for example in /// [ModalRoute.barrierColor], and update accordingly. /// /// The [ModalRoute] subclass overrides this to force the barrier /// overlay to rebuild. /// /// See also: /// /// * [changedInternalState], the equivalent but for changes to the internal /// state of the route. @protected @mustCallSuper void changedExternalState() { } /// Discards any resources used by the object. /// /// This method should not remove its [overlayEntries] from the [Overlay]. The /// object's owner is in charge of doing that. /// /// After this is called, the object is not in a usable state and should be /// discarded. /// /// This method should only be called by the object's owner; typically the /// [Navigator] owns a route and so will call this method when the route is /// removed, after which the route is no longer referenced by the navigator. @mustCallSuper @protected void dispose() { _navigator = null; _restorationScopeId.dispose(); _disposeCompleter.complete(); if (kFlutterMemoryAllocationsEnabled) { FlutterMemoryAllocations.instance.dispatchObjectDisposed(object: this); } } /// Whether this route is the top-most route on the navigator. /// /// If this is true, then [isActive] is also true. bool get isCurrent { if (_navigator == null) { return false; } final _RouteEntry? currentRouteEntry = _navigator!._lastRouteEntryWhereOrNull(_RouteEntry.isPresentPredicate); if (currentRouteEntry == null) { return false; } return currentRouteEntry.route == this; } /// Whether this route is the bottom-most active route on the navigator. /// /// If [isFirst] and [isCurrent] are both true then this is the only route on /// the navigator (and [isActive] will also be true). bool get isFirst { if (_navigator == null) { return false; } final _RouteEntry? currentRouteEntry = _navigator!._firstRouteEntryWhereOrNull(_RouteEntry.isPresentPredicate); if (currentRouteEntry == null) { return false; } return currentRouteEntry.route == this; } /// Whether there is at least one active route underneath this route. @protected bool get hasActiveRouteBelow { if (_navigator == null) { return false; } for (final _RouteEntry entry in _navigator!._history) { if (entry.route == this) { return false; } if (_RouteEntry.isPresentPredicate(entry)) { return true; } } return false; } /// Whether this route is on the navigator. /// /// If the route is not only active, but also the current route (the top-most /// route), then [isCurrent] will also be true. If it is the first route (the /// bottom-most route), then [isFirst] will also be true. /// /// If a higher route is entirely opaque, then the route will be active but not /// rendered. It is even possible for the route to be active but for the stateful /// widgets within the route to not be instantiated. See [ModalRoute.maintainState]. bool get isActive { if (_navigator == null) { return false; } return _navigator!._firstRouteEntryWhereOrNull(_RouteEntry.isRoutePredicate(this))?.isPresent ?? false; } } /// Data that might be useful in constructing a [Route]. @immutable class RouteSettings { /// Creates data used to construct routes. const RouteSettings({ this.name, this.arguments, }); /// The name of the route (e.g., "/settings"). /// /// If null, the route is anonymous. final String? name; /// The arguments passed to this route. /// /// May be used when building the route, e.g. in [Navigator.onGenerateRoute]. final Object? arguments; @override String toString() => '${objectRuntimeType(this, 'RouteSettings')}(${name == null ? 'none' : '"$name"'}, $arguments)'; } /// Describes the configuration of a [Route]. /// /// The type argument `T` is the corresponding [Route]'s return type, as /// used by [Route.currentResult], [Route.popped], and [Route.didPop]. /// /// See also: /// /// * [Navigator.pages], which accepts a list of [Page]s and updates its routes /// history. abstract class Page extends RouteSettings { /// Creates a page and initializes [key] for subclasses. const Page({ this.key, super.name, super.arguments, this.restorationId, }); /// The key associated with this page. /// /// This key will be used for comparing pages in [canUpdate]. final LocalKey? key; /// Restoration ID to save and restore the state of the [Route] configured by /// this page. /// /// If no restoration ID is provided, the [Route] will not restore its state. /// /// See also: /// /// * [RestorationManager], which explains how state restoration works in /// Flutter. final String? restorationId; /// Whether this page can be updated with the [other] page. /// /// Two pages are consider updatable if they have same the [runtimeType] and /// [key]. bool canUpdate(Page other) { return other.runtimeType == runtimeType && other.key == key; } /// Creates the [Route] that corresponds to this page. /// /// The created [Route] must have its [Route.settings] property set to this [Page]. @factory Route createRoute(BuildContext context); @override String toString() => '${objectRuntimeType(this, 'Page')}("$name", $key, $arguments)'; } /// An interface for observing the behavior of a [Navigator]. class NavigatorObserver { /// The navigator that the observer is observing, if any. NavigatorState? get navigator => _navigators[this]; // Expando mapping instances of NavigatorObserver to their associated // NavigatorState (or `null`, if there is no associated NavigatorState). The // reason we don't use a private instance field of type // `NavigatorState?` is because as part of implementing // https://github.com/dart-lang/language/issues/2020, it will soon become a // runtime error to invoke a private member that is mocked in another // library. By using an expando rather than an instance field, we ensure // that a mocked NavigatorObserver can still properly keep track of its // associated NavigatorState. static final Expando _navigators = Expando(); /// The [Navigator] pushed `route`. /// /// The route immediately below that one, and thus the previously active /// route, is `previousRoute`. void didPush(Route route, Route? previousRoute) { } /// The [Navigator] popped `route`. /// /// The route immediately below that one, and thus the newly active /// route, is `previousRoute`. void didPop(Route route, Route? previousRoute) { } /// The [Navigator] removed `route`. /// /// If only one route is being removed, then the route immediately below /// that one, if any, is `previousRoute`. /// /// If multiple routes are being removed, then the route below the /// bottommost route being removed, if any, is `previousRoute`, and this /// method will be called once for each removed route, from the topmost route /// to the bottommost route. void didRemove(Route route, Route? previousRoute) { } /// The [Navigator] replaced `oldRoute` with `newRoute`. void didReplace({ Route? newRoute, Route? oldRoute }) { } /// The [Navigator]'s routes are being moved by a user gesture. /// /// For example, this is called when an iOS back gesture starts, and is used /// to disabled hero animations during such interactions. void didStartUserGesture(Route route, Route? previousRoute) { } /// User gesture is no longer controlling the [Navigator]. /// /// Paired with an earlier call to [didStartUserGesture]. void didStopUserGesture() { } } /// An inherited widget to host a hero controller. /// /// The hosted hero controller will be picked up by the navigator in the /// [child] subtree. Once a navigator picks up this controller, the navigator /// will bar any navigator below its subtree from receiving this controller. /// /// The hero controller inside the [HeroControllerScope] can only subscribe to /// one navigator at a time. An assertion will be thrown if the hero controller /// subscribes to more than one navigators. This can happen when there are /// multiple navigators under the same [HeroControllerScope] in parallel. class HeroControllerScope extends InheritedWidget { /// Creates a widget to host the input [controller]. const HeroControllerScope({ super.key, required HeroController this.controller, required super.child, }); /// Creates a widget to prevent the subtree from receiving the hero controller /// above. const HeroControllerScope.none({ super.key, required super.child, }) : controller = null; /// The hero controller that is hosted inside this widget. final HeroController? controller; /// Retrieves the [HeroController] from the closest [HeroControllerScope] /// ancestor, or null if none exists. /// /// Calling this method will create a dependency on the closest /// [HeroControllerScope] in the [context], if there is one. /// /// See also: /// /// * [HeroControllerScope.of], which is similar to this method, but asserts /// if no [HeroControllerScope] ancestor is found. static HeroController? maybeOf(BuildContext context) { final HeroControllerScope? host = context.dependOnInheritedWidgetOfExactType(); return host?.controller; } /// Retrieves the [HeroController] from the closest [HeroControllerScope] /// ancestor. /// /// If no ancestor is found, this method will assert in debug mode, and throw /// an exception in release mode. /// /// Calling this method will create a dependency on the closest /// [HeroControllerScope] in the [context]. /// /// See also: /// /// * [HeroControllerScope.maybeOf], which is similar to this method, but /// returns null if no [HeroControllerScope] ancestor is found. static HeroController of(BuildContext context) { final HeroController? controller = maybeOf(context); assert(() { if (controller == null) { throw FlutterError( 'HeroControllerScope.of() was called with a context that does not contain a ' 'HeroControllerScope widget.\n' 'No HeroControllerScope widget ancestor could be found starting from the ' 'context that was passed to HeroControllerScope.of(). This can happen ' 'because you are using a widget that looks for a HeroControllerScope ' 'ancestor, but no such ancestor exists.\n' 'The context used was:\n' ' $context', ); } return true; }()); return controller!; } @override bool updateShouldNotify(HeroControllerScope oldWidget) { return oldWidget.controller != controller; } } /// A [Route] wrapper interface that can be staged for [TransitionDelegate] to /// decide how its underlying [Route] should transition on or off screen. abstract class RouteTransitionRecord { /// Retrieves the wrapped [Route]. Route get route; /// Whether this route is waiting for the decision on how to enter the screen. /// /// If this property is true, this route requires an explicit decision on how /// to transition into the screen. Such a decision should be made in the /// [TransitionDelegate.resolve]. bool get isWaitingForEnteringDecision; /// Whether this route is waiting for the decision on how to exit the screen. /// /// If this property is true, this route requires an explicit decision on how /// to transition off the screen. Such a decision should be made in the /// [TransitionDelegate.resolve]. bool get isWaitingForExitingDecision; /// Marks the [route] to be pushed with transition. /// /// During [TransitionDelegate.resolve], this can be called on an entering /// route (where [RouteTransitionRecord.isWaitingForEnteringDecision] is true) in indicate that the /// route should be pushed onto the [Navigator] with an animated transition. void markForPush(); /// Marks the [route] to be added without transition. /// /// During [TransitionDelegate.resolve], this can be called on an entering /// route (where [RouteTransitionRecord.isWaitingForEnteringDecision] is true) in indicate that the /// route should be added onto the [Navigator] without an animated transition. void markForAdd(); /// Marks the [route] to be popped with transition. /// /// During [TransitionDelegate.resolve], this can be called on an exiting /// route to indicate that the route should be popped off the [Navigator] with /// an animated transition. void markForPop([dynamic result]); /// Marks the [route] to be completed without transition. /// /// During [TransitionDelegate.resolve], this can be called on an exiting /// route to indicate that the route should be completed with the provided /// result and removed from the [Navigator] without an animated transition. void markForComplete([dynamic result]); /// Marks the [route] to be removed without transition. /// /// During [TransitionDelegate.resolve], this can be called on an exiting /// route to indicate that the route should be removed from the [Navigator] /// without completing and without an animated transition. void markForRemove(); } /// The delegate that decides how pages added and removed from [Navigator.pages] /// transition in or out of the screen. /// /// This abstract class implements the API to be called by [Navigator] when it /// requires explicit decisions on how the routes transition on or off the screen. /// /// To make route transition decisions, subclass must implement [resolve]. /// /// {@tool snippet} /// The following example demonstrates how to implement a subclass that always /// removes or adds routes without animated transitions and puts the removed /// routes at the top of the list. /// /// ```dart /// class NoAnimationTransitionDelegate extends TransitionDelegate { /// @override /// Iterable resolve({ /// required List newPageRouteHistory, /// required Map locationToExitingPageRoute, /// required Map> pageRouteToPagelessRoutes, /// }) { /// final List results = []; /// /// for (final RouteTransitionRecord pageRoute in newPageRouteHistory) { /// if (pageRoute.isWaitingForEnteringDecision) { /// pageRoute.markForAdd(); /// } /// results.add(pageRoute); /// /// } /// for (final RouteTransitionRecord exitingPageRoute in locationToExitingPageRoute.values) { /// if (exitingPageRoute.isWaitingForExitingDecision) { /// exitingPageRoute.markForRemove(); /// final List? pagelessRoutes = pageRouteToPagelessRoutes[exitingPageRoute]; /// if (pagelessRoutes != null) { /// for (final RouteTransitionRecord pagelessRoute in pagelessRoutes) { /// pagelessRoute.markForRemove(); /// } /// } /// } /// results.add(exitingPageRoute); /// /// } /// return results; /// } /// } /// /// ``` /// {@end-tool} /// /// See also: /// /// * [Navigator.transitionDelegate], which uses this class to make route /// transition decisions. /// * [DefaultTransitionDelegate], which implements the default way to decide /// how routes transition in or out of the screen. abstract class TransitionDelegate { /// Creates a delegate and enables subclass to create a constant class. const TransitionDelegate(); Iterable _transition({ required List newPageRouteHistory, required Map locationToExitingPageRoute, required Map> pageRouteToPagelessRoutes, }) { final Iterable results = resolve( newPageRouteHistory: newPageRouteHistory, locationToExitingPageRoute: locationToExitingPageRoute, pageRouteToPagelessRoutes: pageRouteToPagelessRoutes, ); // Verifies the integrity after the decisions have been made. // // Here are the rules: // - All the entering routes in newPageRouteHistory must either be pushed or // added. // - All the exiting routes in locationToExitingPageRoute must either be // popped, completed or removed. // - All the pageless routes that belong to exiting routes must either be // popped, completed or removed. // - All the entering routes in the result must preserve the same order as // the entering routes in newPageRouteHistory, and the result must contain // all exiting routes. // ex: // // newPageRouteHistory = [A, B, C] // // locationToExitingPageRoute = {A -> D, C -> E} // // results = [A, B ,C ,D ,E] is valid // results = [D, A, B ,C ,E] is also valid because exiting route can be // inserted in any place // // results = [B, A, C ,D ,E] is invalid because B must be after A. // results = [A, B, C ,E] is invalid because results must include D. assert(() { final List resultsToVerify = results.toList(growable: false); final Set exitingPageRoutes = locationToExitingPageRoute.values.toSet(); // Firstly, verifies all exiting routes have been marked. for (final RouteTransitionRecord exitingPageRoute in exitingPageRoutes) { assert(!exitingPageRoute.isWaitingForExitingDecision); if (pageRouteToPagelessRoutes.containsKey(exitingPageRoute)) { for (final RouteTransitionRecord pagelessRoute in pageRouteToPagelessRoutes[exitingPageRoute]!) { assert(!pagelessRoute.isWaitingForExitingDecision); } } } // Secondly, verifies the order of results matches the newPageRouteHistory // and contains all the exiting routes. int indexOfNextRouteInNewHistory = 0; for (final _RouteEntry routeEntry in resultsToVerify.cast<_RouteEntry>()) { assert(!routeEntry.isWaitingForEnteringDecision && !routeEntry.isWaitingForExitingDecision); if ( indexOfNextRouteInNewHistory >= newPageRouteHistory.length || routeEntry != newPageRouteHistory[indexOfNextRouteInNewHistory] ) { assert(exitingPageRoutes.contains(routeEntry)); exitingPageRoutes.remove(routeEntry); } else { indexOfNextRouteInNewHistory += 1; } } assert( indexOfNextRouteInNewHistory == newPageRouteHistory.length && exitingPageRoutes.isEmpty, 'The merged result from the $runtimeType.resolve does not include all ' 'required routes. Do you remember to merge all exiting routes?', ); return true; }()); return results; } /// A method that will be called by the [Navigator] to decide how routes /// transition in or out of the screen when [Navigator.pages] is updated. /// /// The `newPageRouteHistory` list contains all page-based routes in the order /// that will be on the [Navigator]'s history stack after this update /// completes. If a route in `newPageRouteHistory` has its /// [RouteTransitionRecord.isWaitingForEnteringDecision] set to true, this /// route requires explicit decision on how it should transition onto the /// Navigator. To make a decision, call [RouteTransitionRecord.markForPush] or /// [RouteTransitionRecord.markForAdd]. /// /// The `locationToExitingPageRoute` contains the pages-based routes that /// are removed from the routes history after page update. This map records /// page-based routes to be removed with the location of the route in the /// original route history before the update. The keys are the locations /// represented by the page-based routes that are directly below the removed /// routes, and the value are the page-based routes to be removed. The /// location is null if the route to be removed is the bottom most route. If /// a route in `locationToExitingPageRoute` has its /// [RouteTransitionRecord.isWaitingForExitingDecision] set to true, this /// route requires explicit decision on how it should transition off the /// Navigator. To make a decision for a removed route, call /// [RouteTransitionRecord.markForPop], /// [RouteTransitionRecord.markForComplete] or /// [RouteTransitionRecord.markForRemove]. It is possible that decisions are /// not required for routes in the `locationToExitingPageRoute`. This can /// happen if the routes have already been popped in earlier page updates and /// are still waiting for popping animations to finish. In such case, those /// routes are still included in the `locationToExitingPageRoute` with their /// [RouteTransitionRecord.isWaitingForExitingDecision] set to false and no /// decisions are required. /// /// The `pageRouteToPagelessRoutes` records the page-based routes and their /// associated pageless routes. If a page-based route is waiting for exiting /// decision, its associated pageless routes also require explicit decisions /// on how to transition off the screen. /// /// Once all the decisions have been made, this method must merge the removed /// routes (whether or not they require decisions) and the /// `newPageRouteHistory` and return the merged result. The order in the /// result will be the order the [Navigator] uses for updating the route /// history. The return list must preserve the same order of routes in /// `newPageRouteHistory`. The removed routes, however, can be inserted into /// the return list freely as long as all of them are included. /// /// For example, consider the following case. /// /// `newPageRouteHistory = [A, B, C]` /// /// `locationToExitingPageRoute = {A -> D, C -> E}` /// /// The following outputs are valid. /// /// `result = [A, B ,C ,D ,E]` is valid. /// `result = [D, A, B ,C ,E]` is also valid because exiting route can be /// inserted in any place. /// /// The following outputs are invalid. /// /// `result = [B, A, C ,D ,E]` is invalid because B must be after A. /// `result = [A, B, C ,E]` is invalid because results must include D. /// /// See also: /// /// * [RouteTransitionRecord.markForPush], which makes route enter the screen /// with an animated transition. /// * [RouteTransitionRecord.markForAdd], which makes route enter the screen /// without an animated transition. /// * [RouteTransitionRecord.markForPop], which makes route exit the screen /// with an animated transition. /// * [RouteTransitionRecord.markForRemove], which does not complete the /// route and makes it exit the screen without an animated transition. /// * [RouteTransitionRecord.markForComplete], which completes the route and /// makes it exit the screen without an animated transition. /// * [DefaultTransitionDelegate.resolve], which implements the default way /// to decide how routes transition in or out of the screen. Iterable resolve({ required List newPageRouteHistory, required Map locationToExitingPageRoute, required Map> pageRouteToPagelessRoutes, }); } /// The default implementation of [TransitionDelegate] that the [Navigator] will /// use if its [Navigator.transitionDelegate] is not specified. /// /// This transition delegate follows two rules. Firstly, all the entering routes /// are placed on top of the exiting routes if they are at the same location. /// Secondly, the top most route will always transition with an animated transition. /// All the other routes below will either be completed with /// [Route.currentResult] or added without an animated transition. class DefaultTransitionDelegate extends TransitionDelegate { /// Creates a default transition delegate. const DefaultTransitionDelegate() : super(); @override Iterable resolve({ required List newPageRouteHistory, required Map locationToExitingPageRoute, required Map> pageRouteToPagelessRoutes, }) { final List results = []; // This method will handle the exiting route and its corresponding pageless // route at this location. It will also recursively check if there is any // other exiting routes above it and handle them accordingly. void handleExitingRoute(RouteTransitionRecord? location, bool isLast) { final RouteTransitionRecord? exitingPageRoute = locationToExitingPageRoute[location]; if (exitingPageRoute == null) { return; } if (exitingPageRoute.isWaitingForExitingDecision) { final bool hasPagelessRoute = pageRouteToPagelessRoutes.containsKey(exitingPageRoute); final bool isLastExitingPageRoute = isLast && !locationToExitingPageRoute.containsKey(exitingPageRoute); if (isLastExitingPageRoute && !hasPagelessRoute) { exitingPageRoute.markForPop(exitingPageRoute.route.currentResult); } else { exitingPageRoute.markForComplete(exitingPageRoute.route.currentResult); } if (hasPagelessRoute) { final List pagelessRoutes = pageRouteToPagelessRoutes[exitingPageRoute]!; for (final RouteTransitionRecord pagelessRoute in pagelessRoutes) { // It is possible that a pageless route that belongs to an exiting // page-based route does not require exiting decision. This can // happen if the page list is updated right after a Navigator.pop. if (pagelessRoute.isWaitingForExitingDecision) { if (isLastExitingPageRoute && pagelessRoute == pagelessRoutes.last) { pagelessRoute.markForPop(pagelessRoute.route.currentResult); } else { pagelessRoute.markForComplete(pagelessRoute.route.currentResult); } } } } } results.add(exitingPageRoute); // It is possible there is another exiting route above this exitingPageRoute. handleExitingRoute(exitingPageRoute, isLast); } // Handles exiting route in the beginning of list. handleExitingRoute(null, newPageRouteHistory.isEmpty); for (final RouteTransitionRecord pageRoute in newPageRouteHistory) { final bool isLastIteration = newPageRouteHistory.last == pageRoute; if (pageRoute.isWaitingForEnteringDecision) { if (!locationToExitingPageRoute.containsKey(pageRoute) && isLastIteration) { pageRoute.markForPush(); } else { pageRoute.markForAdd(); } } results.add(pageRoute); handleExitingRoute(pageRoute, isLastIteration); } return results; } } /// The default value of [Navigator.routeTraversalEdgeBehavior]. /// /// {@macro flutter.widgets.navigator.routeTraversalEdgeBehavior} const TraversalEdgeBehavior kDefaultRouteTraversalEdgeBehavior = TraversalEdgeBehavior.parentScope; /// A widget that manages a set of child widgets with a stack discipline. /// /// Many apps have a navigator near the top of their widget hierarchy in order /// to display their logical history using an [Overlay] with the most recently /// visited pages visually on top of the older pages. Using this pattern lets /// the navigator visually transition from one page to another by moving the widgets /// around in the overlay. Similarly, the navigator can be used to show a dialog /// by positioning the dialog widget above the current page. /// /// ## Using the Pages API /// /// The [Navigator] will convert its [Navigator.pages] into a stack of [Route]s /// if it is provided. A change in [Navigator.pages] will trigger an update to /// the stack of [Route]s. The [Navigator] will update its routes to match the /// new configuration of its [Navigator.pages]. To use this API, one can create /// a [Page] subclass and defines a list of [Page]s for [Navigator.pages]. A /// [Navigator.onPopPage] callback is also required to properly clean up the /// input pages in case of a pop. /// /// By Default, the [Navigator] will use [DefaultTransitionDelegate] to decide /// how routes transition in or out of the screen. To customize it, define a /// [TransitionDelegate] subclass and provide it to the /// [Navigator.transitionDelegate]. /// /// For more information on using the pages API, see the [Router] widget. /// /// ## Using the Navigator API /// /// Mobile apps typically reveal their contents via full-screen elements /// called "screens" or "pages". In Flutter these elements are called /// routes and they're managed by a [Navigator] widget. The navigator /// manages a stack of [Route] objects and provides two ways for managing /// the stack, the declarative API [Navigator.pages] or imperative API /// [Navigator.push] and [Navigator.pop]. /// /// When your user interface fits this paradigm of a stack, where the user /// should be able to _navigate_ back to an earlier element in the stack, /// the use of routes and the Navigator is appropriate. On certain platforms, /// such as Android, the system UI will provide a back button (outside the /// bounds of your application) that will allow the user to navigate back /// to earlier routes in your application's stack. On platforms that don't /// have this build-in navigation mechanism, the use of an [AppBar] (typically /// used in the [Scaffold.appBar] property) can automatically add a back /// button for user navigation. /// /// ### Displaying a full-screen route /// /// Although you can create a navigator directly, it's most common to use the /// navigator created by the `Router` which itself is created and configured by /// a [WidgetsApp] or a [MaterialApp] widget. You can refer to that navigator /// with [Navigator.of]. /// /// A [MaterialApp] is the simplest way to set things up. The [MaterialApp]'s /// home becomes the route at the bottom of the [Navigator]'s stack. It is what /// you see when the app is launched. /// /// ```dart /// void main() { /// runApp(const MaterialApp(home: MyAppHome())); /// } /// ``` /// /// To push a new route on the stack you can create an instance of /// [MaterialPageRoute] with a builder function that creates whatever you /// want to appear on the screen. For example: /// /// ```dart /// Navigator.push(context, MaterialPageRoute( /// builder: (BuildContext context) { /// return Scaffold( /// appBar: AppBar(title: const Text('My Page')), /// body: Center( /// child: TextButton( /// child: const Text('POP'), /// onPressed: () { /// Navigator.pop(context); /// }, /// ), /// ), /// ); /// }, /// )); /// ``` /// /// The route defines its widget with a builder function instead of a /// child widget because it will be built and rebuilt in different /// contexts depending on when it's pushed and popped. /// /// As you can see, the new route can be popped, revealing the app's home /// page, with the Navigator's pop method: /// /// ```dart /// Navigator.pop(context); /// ``` /// /// It usually isn't necessary to provide a widget that pops the Navigator /// in a route with a [Scaffold] because the Scaffold automatically adds a /// 'back' button to its AppBar. Pressing the back button causes /// [Navigator.pop] to be called. On Android, pressing the system back /// button does the same thing. /// /// ### Using named navigator routes /// /// Mobile apps often manage a large number of routes and it's often /// easiest to refer to them by name. Route names, by convention, /// use a path-like structure (for example, '/a/b/c'). /// The app's home page route is named '/' by default. /// /// The [MaterialApp] can be created /// with a [Map] which maps from a route's name to /// a builder function that will create it. The [MaterialApp] uses this /// map to create a value for its navigator's [onGenerateRoute] callback. /// /// ```dart /// void main() { /// runApp(MaterialApp( /// home: const MyAppHome(), // becomes the route named '/' /// routes: { /// '/a': (BuildContext context) => const MyPage(title: Text('page A')), /// '/b': (BuildContext context) => const MyPage(title: Text('page B')), /// '/c': (BuildContext context) => const MyPage(title: Text('page C')), /// }, /// )); /// } /// ``` /// /// To show a route by name: /// /// ```dart /// Navigator.pushNamed(context, '/b'); /// ``` /// /// ### Routes can return a value /// /// When a route is pushed to ask the user for a value, the value can be /// returned via the [pop] method's result parameter. /// /// Methods that push a route return a [Future]. The Future resolves when the /// route is popped and the [Future]'s value is the [pop] method's `result` /// parameter. /// /// For example if we wanted to ask the user to press 'OK' to confirm an /// operation we could `await` the result of [Navigator.push]: /// /// ```dart /// bool? value = await Navigator.push(context, MaterialPageRoute( /// builder: (BuildContext context) { /// return Center( /// child: GestureDetector( /// child: const Text('OK'), /// onTap: () { Navigator.pop(context, true); } /// ), /// ); /// } /// )); /// ``` /// /// If the user presses 'OK' then value will be true. If the user backs /// out of the route, for example by pressing the Scaffold's back button, /// the value will be null. /// /// When a route is used to return a value, the route's type parameter must /// match the type of [pop]'s result. That's why we've used /// `MaterialPageRoute` instead of `MaterialPageRoute` or just /// `MaterialPageRoute`. (If you prefer to not specify the types, though, that's /// fine too.) /// /// ### Popup routes /// /// Routes don't have to obscure the entire screen. [PopupRoute]s cover the /// screen with a [ModalRoute.barrierColor] that can be only partially opaque to /// allow the current screen to show through. Popup routes are "modal" because /// they block input to the widgets below. /// /// There are functions which create and show popup routes. For /// example: [showDialog], [showMenu], and [showModalBottomSheet]. These /// functions return their pushed route's Future as described above. /// Callers can await the returned value to take an action when the /// route is popped, or to discover the route's value. /// /// There are also widgets which create popup routes, like [PopupMenuButton] and /// [DropdownButton]. These widgets create internal subclasses of PopupRoute /// and use the Navigator's push and pop methods to show and dismiss them. /// /// ### Custom routes /// /// You can create your own subclass of one of the widget library route classes /// like [PopupRoute], [ModalRoute], or [PageRoute], to control the animated /// transition employed to show the route, the color and behavior of the route's /// modal barrier, and other aspects of the route. /// /// The [PageRouteBuilder] class makes it possible to define a custom route /// in terms of callbacks. Here's an example that rotates and fades its child /// when the route appears or disappears. This route does not obscure the entire /// screen because it specifies `opaque: false`, just as a popup route does. /// /// ```dart /// Navigator.push(context, PageRouteBuilder( /// opaque: false, /// pageBuilder: (BuildContext context, _, __) { /// return const Center(child: Text('My PageRoute')); /// }, /// transitionsBuilder: (___, Animation animation, ____, Widget child) { /// return FadeTransition( /// opacity: animation, /// child: RotationTransition( /// turns: Tween(begin: 0.5, end: 1.0).animate(animation), /// child: child, /// ), /// ); /// } /// )); /// ``` /// /// The page route is built in two parts, the "page" and the /// "transitions". The page becomes a descendant of the child passed to /// the `transitionsBuilder` function. Typically the page is only built once, /// because it doesn't depend on its animation parameters (elided with `_` /// and `__` in this example). The transition is built on every frame /// for its duration. /// /// (In this example, `void` is used as the return type for the route, because /// it does not return a value.) /// /// ### Nesting Navigators /// /// An app can use more than one [Navigator]. Nesting one [Navigator] below /// another [Navigator] can be used to create an "inner journey" such as tabbed /// navigation, user registration, store checkout, or other independent journeys /// that represent a subsection of your overall application. /// /// #### Example /// /// It is standard practice for iOS apps to use tabbed navigation where each /// tab maintains its own navigation history. Therefore, each tab has its own /// [Navigator], creating a kind of "parallel navigation." /// /// In addition to the parallel navigation of the tabs, it is still possible to /// launch full-screen pages that completely cover the tabs. For example: an /// on-boarding flow, or an alert dialog. Therefore, there must exist a "root" /// [Navigator] that sits above the tab navigation. As a result, each of the /// tab's [Navigator]s are actually nested [Navigator]s sitting below a single /// root [Navigator]. /// /// In practice, the nested [Navigator]s for tabbed navigation sit in the /// [WidgetsApp] and [CupertinoTabView] widgets and do not need to be explicitly /// created or managed. /// /// {@tool sample} /// The following example demonstrates how a nested [Navigator] can be used to /// present a standalone user registration journey. /// /// Even though this example uses two [Navigator]s to demonstrate nested /// [Navigator]s, a similar result is possible using only a single [Navigator]. /// /// Run this example with `flutter run --route=/signup` to start it with /// the signup flow instead of on the home page. /// /// ** See code in examples/api/lib/widgets/navigator/navigator.0.dart ** /// {@end-tool} /// /// [Navigator.of] operates on the nearest ancestor [Navigator] from the given /// [BuildContext]. Be sure to provide a [BuildContext] below the intended /// [Navigator], especially in large `build` methods where nested [Navigator]s /// are created. The [Builder] widget can be used to access a [BuildContext] at /// a desired location in the widget subtree. /// /// ### Finding the enclosing route /// /// In the common case of a modal route, the enclosing route can be obtained /// from inside a build method using [ModalRoute.of]. To determine if the /// enclosing route is the active route (e.g. so that controls can be dimmed /// when the route is not active), the [Route.isCurrent] property can be checked /// on the returned route. /// /// ## State Restoration /// /// If provided with a [restorationScopeId] and when surrounded by a valid /// [RestorationScope] the [Navigator] will restore its state by recreating /// the current history stack of [Route]s during state restoration and by /// restoring the internal state of those [Route]s. However, not all [Route]s /// on the stack can be restored: /// /// * [Page]-based routes restore their state if [Page.restorationId] is /// provided. /// * [Route]s added with the classic imperative API ([push], [pushNamed], and /// friends) can never restore their state. /// * A [Route] added with the restorable imperative API ([restorablePush], /// [restorablePushNamed], and all other imperative methods with "restorable" /// in their name) restores its state if all routes below it up to and /// including the first [Page]-based route below it are restored. If there /// is no [Page]-based route below it, it only restores its state if all /// routes below it restore theirs. /// /// If a [Route] is deemed restorable, the [Navigator] will set its /// [Route.restorationScopeId] to a non-null value. Routes can use that ID to /// store and restore their own state. As an example, the [ModalRoute] will /// use this ID to create a [RestorationScope] for its content widgets. class Navigator extends StatefulWidget { /// Creates a widget that maintains a stack-based history of child widgets. /// /// If the [pages] is not empty, the [onPopPage] must not be null. const Navigator({ super.key, this.pages = const >[], this.onPopPage, this.initialRoute, this.onGenerateInitialRoutes = Navigator.defaultGenerateInitialRoutes, this.onGenerateRoute, this.onUnknownRoute, this.transitionDelegate = const DefaultTransitionDelegate(), this.reportsRouteUpdateToEngine = false, this.clipBehavior = Clip.hardEdge, this.observers = const [], this.requestFocus = true, this.restorationScopeId, this.routeTraversalEdgeBehavior = kDefaultRouteTraversalEdgeBehavior, }); /// The list of pages with which to populate the history. /// /// Pages are turned into routes using [Page.createRoute] in a manner /// analogous to how [Widget]s are turned into [Element]s (and [State]s or /// [RenderObject]s) using [Widget.createElement] (and /// [StatefulWidget.createState] or [RenderObjectWidget.createRenderObject]). /// /// When this list is updated, the new list is compared to the previous /// list and the set of routes is updated accordingly. /// /// Some [Route]s do not correspond to [Page] objects, namely, those that are /// added to the history using the [Navigator] API ([push] and friends). A /// [Route] that does not correspond to a [Page] object is called a pageless /// route and is tied to the [Route] that _does_ correspond to a [Page] object /// that is below it in the history. /// /// Pages that are added or removed may be animated as controlled by the /// [transitionDelegate]. If a page is removed that had other pageless routes /// pushed on top of it using [push] and friends, those pageless routes are /// also removed with or without animation as determined by the /// [transitionDelegate]. /// /// To use this API, an [onPopPage] callback must also be provided to properly /// clean up this list if a page has been popped. /// /// If [initialRoute] is non-null when the widget is first created, then /// [onGenerateInitialRoutes] is used to generate routes that are above those /// corresponding to [pages] in the initial history. final List> pages; /// Called when [pop] is invoked but the current [Route] corresponds to a /// [Page] found in the [pages] list. /// /// The `result` argument is the value with which the route is to complete /// (e.g. the value returned from a dialog). /// /// This callback is responsible for calling [Route.didPop] and returning /// whether this pop is successful. /// /// The [Navigator] widget should be rebuilt with a [pages] list that does not /// contain the [Page] for the given [Route]. The next time the [pages] list /// is updated, if the [Page] corresponding to this [Route] is still present, /// it will be interpreted as a new route to display. final PopPageCallback? onPopPage; /// The delegate used for deciding how routes transition in or off the screen /// during the [pages] updates. /// /// Defaults to [DefaultTransitionDelegate]. final TransitionDelegate transitionDelegate; /// The name of the first route to show. /// /// Defaults to [Navigator.defaultRouteName]. /// /// The value is interpreted according to [onGenerateInitialRoutes], which /// defaults to [defaultGenerateInitialRoutes]. /// /// Changing the [initialRoute] will have no effect, as it only controls the /// _initial_ route. To change the route while the application is running, use /// the static functions on this class, such as [push] or [replace]. final String? initialRoute; /// Called to generate a route for a given [RouteSettings]. final RouteFactory? onGenerateRoute; /// Called when [onGenerateRoute] fails to generate a route. /// /// This callback is typically used for error handling. For example, this /// callback might always generate a "not found" page that describes the route /// that wasn't found. /// /// Unknown routes can arise either from errors in the app or from external /// requests to push routes, such as from Android intents. final RouteFactory? onUnknownRoute; /// A list of observers for this navigator. final List observers; /// Restoration ID to save and restore the state of the navigator, including /// its history. /// /// {@template flutter.widgets.navigator.restorationScopeId} /// If a restoration ID is provided, the navigator will persist its internal /// state (including the route history as well as the restorable state of the /// routes) and restore it during state restoration. /// /// If no restoration ID is provided, the route history stack will not be /// restored and state restoration is disabled for the individual routes as /// well. /// /// The state is persisted in a [RestorationBucket] claimed from /// the surrounding [RestorationScope] using the provided restoration ID. /// Within that bucket, the [Navigator] also creates a new [RestorationScope] /// for its children (the [Route]s). /// /// See also: /// /// * [RestorationManager], which explains how state restoration works in /// Flutter. /// * [RestorationMixin], which contains a runnable code sample showcasing /// state restoration in Flutter. /// * [Navigator], which explains under the heading "state restoration" /// how and under what conditions the navigator restores its state. /// * [Navigator.restorablePush], which includes an example showcasing how /// to push a restorable route unto the navigator. /// {@endtemplate} final String? restorationScopeId; /// Controls the transfer of focus beyond the first and the last items of a /// focus scope that defines focus traversal of widgets within a route. /// /// {@template flutter.widgets.navigator.routeTraversalEdgeBehavior} /// The focus inside routes installed in the top of the app affects how /// the app behaves with respect to the platform content surrounding it. /// For example, on the web, an app is at a minimum surrounded by browser UI, /// such as the address bar, browser tabs, and more. The user should be able /// to reach browser UI using normal focus shortcuts. Similarly, if the app /// is embedded within an `