您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
876 行
31 KiB
876 行
31 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Unity.UIWidgets.async2;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.gestures;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.scheduler2;
|
|
using SchedulerBinding = Unity.UIWidgets.scheduler2.SchedulerBinding;
|
|
using SchedulerPhase = Unity.UIWidgets.scheduler2.SchedulerPhase;
|
|
|
|
namespace Unity.UIWidgets.widgets {
|
|
public delegate Route RouteFactory(RouteSettings settings);
|
|
|
|
public delegate bool RoutePredicate(Route route);
|
|
|
|
public delegate Future<bool> WillPopCallback();
|
|
|
|
public enum RoutePopDisposition {
|
|
pop,
|
|
doNotPop,
|
|
bubble
|
|
}
|
|
|
|
public abstract class Route {
|
|
public Route(RouteSettings settings = null) {
|
|
this.settings = settings ?? new RouteSettings();
|
|
}
|
|
|
|
internal NavigatorState _navigator;
|
|
|
|
public NavigatorState navigator {
|
|
get { return _navigator; }
|
|
}
|
|
|
|
public readonly RouteSettings settings;
|
|
|
|
public virtual List<OverlayEntry> overlayEntries {
|
|
get { return new List<OverlayEntry>(); }
|
|
}
|
|
|
|
protected internal virtual void install(OverlayEntry insertionPoint) {
|
|
}
|
|
|
|
protected internal virtual TickerFuture didPush() {
|
|
var future = TickerFuture.complete();
|
|
future.then(o => {
|
|
navigator?.focusScopeNode?.requestFocus();
|
|
});
|
|
return future;
|
|
}
|
|
|
|
protected internal virtual void didReplace(Route oldRoute) {
|
|
}
|
|
|
|
public virtual Future<RoutePopDisposition> willPop() {
|
|
return Future<RoutePopDisposition>.value(isFirst
|
|
? RoutePopDisposition.bubble
|
|
: RoutePopDisposition.pop).to<RoutePopDisposition>();
|
|
}
|
|
|
|
public virtual bool willHandlePopInternally {
|
|
get { return false; }
|
|
}
|
|
|
|
public virtual object currentResult {
|
|
get { return null; }
|
|
}
|
|
|
|
public Future popped {
|
|
get { return _popCompleter.future; }
|
|
}
|
|
|
|
internal readonly Completer _popCompleter = Completer.create();
|
|
|
|
protected internal virtual bool didPop(object result) {
|
|
didComplete(result);
|
|
return true;
|
|
}
|
|
|
|
protected internal virtual void didComplete(object result) {
|
|
_popCompleter.complete(FutureOr.value(result ?? currentResult));
|
|
}
|
|
|
|
protected internal virtual void didPopNext(Route nextRoute) {
|
|
}
|
|
|
|
protected internal virtual void didChangeNext(Route nextRoute) {
|
|
}
|
|
|
|
protected internal virtual void didChangePrevious(Route previousRoute) {
|
|
}
|
|
|
|
protected internal virtual void changedInternalState() {
|
|
}
|
|
|
|
protected internal virtual void changedExternalState() {
|
|
}
|
|
|
|
protected internal virtual void dispose() {
|
|
_navigator = null;
|
|
}
|
|
|
|
public bool isCurrent {
|
|
get { return _navigator != null && _navigator._history.last() == this; }
|
|
}
|
|
|
|
public bool isFirst {
|
|
get { return _navigator != null && _navigator._history.first() == this; }
|
|
}
|
|
|
|
public bool isActive {
|
|
get { return _navigator != null && _navigator._history.Contains(this); }
|
|
}
|
|
}
|
|
|
|
public class RouteSettings {
|
|
public RouteSettings(string name = null, bool isInitialRoute = false, object arguments = null) {
|
|
this.name = name;
|
|
this.isInitialRoute = isInitialRoute;
|
|
this.arguments = arguments;
|
|
}
|
|
|
|
RouteSettings copyWith(string name = null, bool? isInitialRoute = null, object arguments = null) {
|
|
return new RouteSettings(
|
|
name ?? this.name,
|
|
isInitialRoute ?? this.isInitialRoute,
|
|
arguments ?? this.arguments
|
|
);
|
|
}
|
|
|
|
public readonly bool isInitialRoute;
|
|
|
|
public readonly string name;
|
|
|
|
public readonly object arguments;
|
|
|
|
public override string ToString() {
|
|
return $"{GetType()}(\"{name}\", {arguments})";
|
|
}
|
|
}
|
|
|
|
public class NavigatorObserver {
|
|
internal NavigatorState _navigator;
|
|
|
|
public NavigatorState navigator {
|
|
get { return _navigator; }
|
|
}
|
|
|
|
public virtual void didPush(Route route, Route previousRoute) {
|
|
}
|
|
|
|
public virtual void didPop(Route route, Route previousRoute) {
|
|
}
|
|
|
|
public virtual void didRemove(Route route, Route previousRoute) {
|
|
}
|
|
|
|
public virtual void didReplace(Route newRoute = null, Route oldRoute = null) {
|
|
}
|
|
|
|
public virtual void didStartUserGesture(Route route, Route previousRoute) {
|
|
}
|
|
|
|
public virtual void didStopUserGesture() {
|
|
}
|
|
}
|
|
|
|
public class Navigator : StatefulWidget {
|
|
public Navigator(
|
|
Key key = null,
|
|
string initialRoute = null,
|
|
RouteFactory onGenerateRoute = null,
|
|
RouteFactory onUnknownRoute = null,
|
|
List<NavigatorObserver> observers = null) : base(key) {
|
|
D.assert(onGenerateRoute != null);
|
|
this.initialRoute = initialRoute;
|
|
this.onGenerateRoute = onGenerateRoute;
|
|
this.onUnknownRoute = onUnknownRoute;
|
|
this.observers = observers ?? new List<NavigatorObserver>();
|
|
}
|
|
|
|
public readonly string initialRoute;
|
|
|
|
public readonly RouteFactory onGenerateRoute;
|
|
|
|
public readonly RouteFactory onUnknownRoute;
|
|
|
|
public readonly List<NavigatorObserver> observers;
|
|
|
|
public static readonly string defaultRouteName = "/";
|
|
|
|
public static Future pushNamed(BuildContext context, string routeName, object arguments = null) {
|
|
return of(context).pushNamed(routeName, arguments: arguments);
|
|
}
|
|
|
|
public static Future pushReplacementNamed(BuildContext context, string routeName,
|
|
object result = null, object arguments = null) {
|
|
return of(context).pushReplacementNamed(routeName, result: result, arguments: arguments);
|
|
}
|
|
|
|
public static Future popAndPushNamed(BuildContext context, string routeName, object result = null,
|
|
object arguments = null) {
|
|
return of(context).popAndPushNamed(routeName, result: result, arguments: arguments);
|
|
}
|
|
|
|
public static Future pushNamedAndRemoveUntil(BuildContext context, string newRouteName,
|
|
RoutePredicate predicate, object arguments = null) {
|
|
return of(context).pushNamedAndRemoveUntil(newRouteName, predicate, arguments: arguments);
|
|
}
|
|
|
|
public static Future push(BuildContext context, Route route) {
|
|
return of(context).push(route);
|
|
}
|
|
|
|
public static Future pushReplacement(BuildContext context, Route newRoute, object result = null) {
|
|
return of(context).pushReplacement(newRoute, result);
|
|
}
|
|
|
|
public static Future pushAndRemoveUntil(BuildContext context, Route newRoute,
|
|
RoutePredicate predicate) {
|
|
return of(context).pushAndRemoveUntil(newRoute, predicate);
|
|
}
|
|
|
|
public static void replace(BuildContext context, Route oldRoute = null, Route newRoute = null) {
|
|
D.assert(oldRoute != null);
|
|
D.assert(newRoute != null);
|
|
of(context).replace(oldRoute: oldRoute, newRoute: newRoute);
|
|
}
|
|
|
|
public static void replaceRouteBelow(BuildContext context, Route anchorRoute = null, Route newRoute = null) {
|
|
D.assert(anchorRoute != null);
|
|
D.assert(newRoute != null);
|
|
of(context).replaceRouteBelow(anchorRoute: anchorRoute, newRoute: newRoute);
|
|
}
|
|
|
|
public static bool canPop(BuildContext context) {
|
|
NavigatorState navigator = of(context, nullOk: true);
|
|
return navigator != null && navigator.canPop();
|
|
}
|
|
|
|
public static Future<bool> maybePop(BuildContext context, object result = null) {
|
|
return of(context).maybePop(result);
|
|
}
|
|
|
|
public static bool pop(BuildContext context, object result = null) {
|
|
return of(context).pop(result);
|
|
}
|
|
|
|
public static void popUntil(BuildContext context, RoutePredicate predicate) {
|
|
of(context).popUntil(predicate);
|
|
}
|
|
|
|
public static void removeRoute(BuildContext context, Route route) {
|
|
of(context).removeRoute(route);
|
|
}
|
|
|
|
static void removeRouteBelow(BuildContext context, Route anchorRoute) {
|
|
of(context).removeRouteBelow(anchorRoute);
|
|
}
|
|
|
|
public static NavigatorState of(
|
|
BuildContext context,
|
|
bool rootNavigator = false,
|
|
bool nullOk = false
|
|
) {
|
|
var navigator = rootNavigator
|
|
? (NavigatorState) context.rootAncestorStateOfType(new TypeMatcher<NavigatorState>())
|
|
: (NavigatorState) context.ancestorStateOfType(new TypeMatcher<NavigatorState>());
|
|
D.assert(() => {
|
|
if (navigator == null && !nullOk) {
|
|
throw new UIWidgetsError(
|
|
"Navigator operation requested with a context that does not include a Navigator.\n" +
|
|
"The context used to push or pop routes from the Navigator must be that of a " +
|
|
"widget that is a descendant of a Navigator widget."
|
|
);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
return navigator;
|
|
}
|
|
|
|
public override State createState() {
|
|
return new NavigatorState();
|
|
}
|
|
}
|
|
|
|
public class NavigatorState : TickerProviderStateMixin<Navigator> {
|
|
readonly GlobalKey<OverlayState> _overlayKey = GlobalKey<OverlayState>.key();
|
|
internal readonly List<Route> _history = new List<Route>();
|
|
readonly HashSet<Route> _poppedRoutes = new HashSet<Route>();
|
|
|
|
public readonly FocusScopeNode focusScopeNode = new FocusScopeNode();
|
|
|
|
readonly List<OverlayEntry> _initialOverlayEntries = new List<OverlayEntry>();
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
foreach (var observer in widget.observers) {
|
|
D.assert(observer.navigator == null);
|
|
observer._navigator = this;
|
|
}
|
|
|
|
var initialRouteName = widget.initialRoute ?? Navigator.defaultRouteName;
|
|
if (initialRouteName.StartsWith("/") && initialRouteName.Length > 1) {
|
|
initialRouteName = initialRouteName.Substring(1);
|
|
D.assert(Navigator.defaultRouteName == "/");
|
|
var plannedInitialRouteNames = new List<string> {
|
|
Navigator.defaultRouteName
|
|
};
|
|
var plannedInitialRoutes = new List<Route> {
|
|
_routeNamed(Navigator.defaultRouteName, arguments: null, true)
|
|
};
|
|
|
|
var routeParts = initialRouteName.Split('/');
|
|
if (initialRouteName.isNotEmpty()) {
|
|
var routeName = "";
|
|
foreach (var part in routeParts) {
|
|
routeName += $"/{part}";
|
|
plannedInitialRouteNames.Add(routeName);
|
|
plannedInitialRoutes.Add(_routeNamed(routeName, arguments: null, true));
|
|
}
|
|
}
|
|
|
|
if (plannedInitialRoutes.Contains(null)) {
|
|
D.assert(() => {
|
|
UIWidgetsError.reportError(new UIWidgetsErrorDetails(
|
|
exception: new Exception(
|
|
"Could not navigate to initial route.\n" +
|
|
$"The requested route name was: \"{initialRouteName}\n" +
|
|
"The following routes were therefore attempted:\n" +
|
|
$" * {string.Join("\n * ", plannedInitialRouteNames)}\n" +
|
|
"This resulted in the following objects:\n" +
|
|
$" * {string.Join("\n * ", plannedInitialRoutes)}\n" +
|
|
"One or more of those objects was null, and therefore the initial route specified will be " +
|
|
$"ignored and \"{Navigator.defaultRouteName}\" will be used instead.")));
|
|
return true;
|
|
});
|
|
push(_routeNamed(Navigator.defaultRouteName, arguments: null));
|
|
}
|
|
else {
|
|
plannedInitialRoutes.Each(route => { push(route); });
|
|
}
|
|
}
|
|
else {
|
|
Route route = null;
|
|
if (initialRouteName != Navigator.defaultRouteName) {
|
|
route = _routeNamed(initialRouteName, arguments: null, true);
|
|
}
|
|
|
|
route = route ?? _routeNamed(Navigator.defaultRouteName, arguments: null);
|
|
push(route);
|
|
}
|
|
|
|
foreach (var route in _history) {
|
|
_initialOverlayEntries.AddRange(route.overlayEntries);
|
|
}
|
|
}
|
|
|
|
|
|
public override void didUpdateWidget(StatefulWidget oldWidget) {
|
|
base.didUpdateWidget(oldWidget);
|
|
if (((Navigator) oldWidget).observers != widget.observers) {
|
|
foreach (var observer in ((Navigator) oldWidget).observers) {
|
|
observer._navigator = null;
|
|
}
|
|
|
|
foreach (var observer in widget.observers) {
|
|
D.assert(observer.navigator == null);
|
|
observer._navigator = this;
|
|
}
|
|
}
|
|
|
|
foreach (var route in _history) {
|
|
route.changedExternalState();
|
|
}
|
|
}
|
|
|
|
public override void dispose() {
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
foreach (var observer in widget.observers) {
|
|
observer._navigator = null;
|
|
}
|
|
|
|
var doomed = _poppedRoutes.ToList();
|
|
doomed.AddRange(_history);
|
|
foreach (var route in doomed) {
|
|
route.dispose();
|
|
}
|
|
|
|
_poppedRoutes.Clear();
|
|
_history.Clear();
|
|
focusScopeNode.detach();
|
|
base.dispose();
|
|
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
public OverlayState overlay {
|
|
get { return _overlayKey.currentState; }
|
|
}
|
|
|
|
OverlayEntry _currentOverlayEntry {
|
|
get {
|
|
var route = _history.FindLast(r => r.overlayEntries.isNotEmpty());
|
|
return route?.overlayEntries.last();
|
|
}
|
|
}
|
|
|
|
bool _debugLocked;
|
|
|
|
Route _routeNamed(string name, object arguments, bool allowNull = false) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(name != null);
|
|
var settings = new RouteSettings(
|
|
name: name,
|
|
isInitialRoute: _history.isEmpty(),
|
|
arguments: arguments
|
|
);
|
|
|
|
var route = (Route) widget.onGenerateRoute(settings);
|
|
if (route == null && !allowNull) {
|
|
D.assert(() => {
|
|
if (widget.onUnknownRoute == null) {
|
|
throw new UIWidgetsError(
|
|
"If a Navigator has no onUnknownRoute, then its onGenerateRoute must never return null.\n" +
|
|
$"When trying to build the route \"{name}\", onGenerateRoute returned null, but there was no " +
|
|
"onUnknownRoute callback specified.\n" +
|
|
"The Navigator was:\n" +
|
|
$" {this}");
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
route = (Route) widget.onUnknownRoute(settings);
|
|
D.assert(() => {
|
|
if (route == null) {
|
|
throw new UIWidgetsError(
|
|
"A Navigator\'s onUnknownRoute returned null.\n" +
|
|
$"When trying to build the route \"{name}\", both onGenerateRoute and onUnknownRoute returned " +
|
|
"null. The onUnknownRoute callback should never return null.\n" +
|
|
"The Navigator was:\n" +
|
|
$" {this}"
|
|
);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
}
|
|
|
|
return route;
|
|
}
|
|
|
|
public Future pushNamed(string routeName, object arguments = null) {
|
|
return push(_routeNamed(routeName, arguments: arguments));
|
|
}
|
|
|
|
public Future pushReplacementNamed(string routeName, object result = null, object arguments = null) {
|
|
return pushReplacement(_routeNamed(routeName, arguments: arguments), result);
|
|
}
|
|
|
|
public Future popAndPushNamed(string routeName, object result = null, object arguments = null) {
|
|
pop(result);
|
|
return pushNamed(routeName, arguments: arguments);
|
|
}
|
|
|
|
public Future pushNamedAndRemoveUntil(string newRouteName, RoutePredicate predicate,
|
|
object arguments = null) {
|
|
return pushAndRemoveUntil(_routeNamed(newRouteName, arguments: arguments), predicate);
|
|
}
|
|
|
|
public Future push(Route route) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
D.assert(route != null);
|
|
D.assert(route._navigator == null);
|
|
var oldRoute = _history.isNotEmpty() ? _history.last() : null;
|
|
route._navigator = this;
|
|
route.install(_currentOverlayEntry);
|
|
_history.Add(route);
|
|
route.didPush();
|
|
route.didChangeNext(null);
|
|
if (oldRoute != null) {
|
|
oldRoute.didChangeNext(route);
|
|
route.didChangePrevious(oldRoute);
|
|
}
|
|
|
|
foreach (var observer in widget.observers) {
|
|
observer.didPush(route, oldRoute);
|
|
}
|
|
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
_afterNavigation();
|
|
return route.popped;
|
|
}
|
|
|
|
void _afterNavigation() {
|
|
_cancelActivePointers();
|
|
}
|
|
|
|
public Future pushReplacement(Route newRoute, object result = null) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
var oldRoute = _history.last();
|
|
D.assert(oldRoute != null && oldRoute._navigator == this);
|
|
D.assert(oldRoute.overlayEntries.isNotEmpty());
|
|
D.assert(newRoute._navigator == null);
|
|
D.assert(newRoute.overlayEntries.isEmpty());
|
|
var index = _history.Count - 1;
|
|
D.assert(index >= 0);
|
|
D.assert(_history.IndexOf(oldRoute) == index);
|
|
newRoute._navigator = this;
|
|
newRoute.install(_currentOverlayEntry);
|
|
_history[index] = newRoute;
|
|
newRoute.didPush().whenCompleteOrCancel(() => {
|
|
if (mounted) {
|
|
oldRoute.didComplete(result ?? oldRoute.currentResult);
|
|
oldRoute.dispose();
|
|
}
|
|
});
|
|
newRoute.didChangeNext(null);
|
|
if (index > 0) {
|
|
_history[index - 1].didChangeNext(newRoute);
|
|
newRoute.didChangePrevious(_history[index - 1]);
|
|
}
|
|
|
|
foreach (var observer in widget.observers) {
|
|
observer.didReplace(newRoute: newRoute, oldRoute: oldRoute);
|
|
}
|
|
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
_afterNavigation();
|
|
return newRoute.popped;
|
|
}
|
|
|
|
public Future pushAndRemoveUntil(Route newRoute, RoutePredicate predicate) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
var removedRoutes = new List<Route>();
|
|
while (_history.isNotEmpty() && !predicate(_history.last())) {
|
|
var removedRoute = _history.last();
|
|
_history.RemoveAt(_history.Count - 1);
|
|
D.assert(removedRoute != null && removedRoute._navigator == this);
|
|
D.assert(removedRoute.overlayEntries.isNotEmpty());
|
|
removedRoutes.Add(removedRoute);
|
|
}
|
|
|
|
D.assert(newRoute._navigator == null);
|
|
D.assert(newRoute.overlayEntries.isEmpty());
|
|
var oldRoute = _history.isNotEmpty() ? _history.last() : null;
|
|
newRoute._navigator = this;
|
|
newRoute.install(_currentOverlayEntry);
|
|
_history.Add(newRoute);
|
|
newRoute.didPush().whenCompleteOrCancel(() => {
|
|
if (mounted) {
|
|
foreach (var route in removedRoutes) {
|
|
route.dispose();
|
|
}
|
|
}
|
|
});
|
|
newRoute.didChangeNext(null);
|
|
if (oldRoute != null) {
|
|
oldRoute.didChangeNext(newRoute);
|
|
}
|
|
|
|
foreach (var observer in widget.observers) {
|
|
observer.didPush(newRoute, oldRoute);
|
|
foreach (var removedRoute in removedRoutes) {
|
|
observer.didRemove(removedRoute, oldRoute);
|
|
}
|
|
}
|
|
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
_afterNavigation();
|
|
return newRoute.popped;
|
|
}
|
|
|
|
public void replace(Route oldRoute = null, Route newRoute = null) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(oldRoute != null);
|
|
D.assert(newRoute != null);
|
|
if (oldRoute == newRoute) {
|
|
return;
|
|
}
|
|
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
D.assert(oldRoute._navigator == this);
|
|
D.assert(newRoute._navigator == null);
|
|
D.assert(oldRoute.overlayEntries.isNotEmpty());
|
|
D.assert(newRoute.overlayEntries.isEmpty());
|
|
D.assert(!overlay.debugIsVisible(oldRoute.overlayEntries.last()));
|
|
var index = _history.IndexOf(oldRoute);
|
|
D.assert(index >= 0);
|
|
newRoute._navigator = this;
|
|
newRoute.install(oldRoute.overlayEntries.last());
|
|
_history[index] = newRoute;
|
|
newRoute.didReplace(oldRoute);
|
|
if (index + 1 < _history.Count) {
|
|
newRoute.didChangeNext(_history[index + 1]);
|
|
_history[index + 1].didChangePrevious(newRoute);
|
|
}
|
|
else {
|
|
newRoute.didChangeNext(null);
|
|
}
|
|
|
|
if (index > 0) {
|
|
_history[index - 1].didChangeNext(newRoute);
|
|
newRoute.didChangePrevious(_history[index - 1]);
|
|
}
|
|
|
|
foreach (var observer in widget.observers) {
|
|
observer.didReplace(newRoute: newRoute, oldRoute: oldRoute);
|
|
}
|
|
|
|
oldRoute.dispose();
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
public void replaceRouteBelow(Route anchorRoute = null, Route newRoute = null) {
|
|
D.assert(anchorRoute != null);
|
|
D.assert(anchorRoute._navigator == this);
|
|
D.assert(_history.IndexOf(anchorRoute) > 0);
|
|
replace(_history[_history.IndexOf(anchorRoute) - 1], newRoute);
|
|
}
|
|
|
|
public bool canPop() {
|
|
D.assert(_history.isNotEmpty);
|
|
return _history.Count > 1 || _history[0].willHandlePopInternally;
|
|
}
|
|
|
|
public Future<bool> maybePop(object result = null) {
|
|
var route = _history.last();
|
|
D.assert(route._navigator == this);
|
|
return route.willPop().then_(disposition => {
|
|
if (disposition != RoutePopDisposition.bubble && mounted) {
|
|
if (disposition == RoutePopDisposition.pop) {
|
|
pop(result);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}).to<bool>();
|
|
}
|
|
|
|
public bool pop(object result = null) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
var route = _history.last();
|
|
D.assert(route._navigator == this);
|
|
var debugPredictedWouldPop = false;
|
|
D.assert(() => {
|
|
debugPredictedWouldPop = !route.willHandlePopInternally;
|
|
return true;
|
|
});
|
|
if (route.didPop(result ?? route.currentResult)) {
|
|
D.assert(debugPredictedWouldPop);
|
|
if (_history.Count > 1) {
|
|
_history.removeLast();
|
|
if (route._navigator != null) {
|
|
_poppedRoutes.Add(route);
|
|
}
|
|
|
|
_history.last().didPopNext(route);
|
|
foreach (var observer in widget.observers) {
|
|
observer.didPop(route, _history.last());
|
|
}
|
|
}
|
|
else {
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
return false;
|
|
}
|
|
}
|
|
else {
|
|
D.assert(!debugPredictedWouldPop);
|
|
}
|
|
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
_afterNavigation();
|
|
return true;
|
|
}
|
|
|
|
public void popUntil(RoutePredicate predicate) {
|
|
while (!predicate(_history.last())) {
|
|
pop();
|
|
}
|
|
}
|
|
|
|
public void removeRoute(Route route) {
|
|
D.assert(route != null);
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
D.assert(route._navigator == this);
|
|
var index = _history.IndexOf(route);
|
|
D.assert(index != -1);
|
|
var previousRoute = index > 0 ? _history[index - 1] : null;
|
|
var nextRoute = index + 1 < _history.Count ? _history[index + 1] : null;
|
|
_history.RemoveAt(index);
|
|
previousRoute?.didChangeNext(nextRoute);
|
|
nextRoute?.didChangePrevious(previousRoute);
|
|
foreach (var observer in widget.observers) {
|
|
observer.didRemove(route, previousRoute);
|
|
}
|
|
|
|
route.dispose();
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
_afterNavigation();
|
|
}
|
|
|
|
public void removeRouteBelow(Route anchorRoute) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(() => {
|
|
_debugLocked = true;
|
|
return true;
|
|
});
|
|
D.assert(anchorRoute._navigator == this);
|
|
var index = _history.IndexOf(anchorRoute) - 1;
|
|
D.assert(index >= 0);
|
|
var targetRoute = _history[index];
|
|
D.assert(targetRoute._navigator == this);
|
|
D.assert(targetRoute.overlayEntries.isEmpty() ||
|
|
!overlay.debugIsVisible(targetRoute.overlayEntries.last()));
|
|
_history.RemoveAt(index);
|
|
var nextRoute = index < _history.Count ? _history[index] : null;
|
|
var previousRoute = index > 0 ? _history[index - 1] : null;
|
|
if (previousRoute != null) {
|
|
previousRoute.didChangeNext(nextRoute);
|
|
}
|
|
|
|
if (nextRoute != null) {
|
|
nextRoute.didChangePrevious(previousRoute);
|
|
}
|
|
|
|
targetRoute.dispose();
|
|
D.assert(() => {
|
|
_debugLocked = false;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
public void finalizeRoute(Route route) {
|
|
_poppedRoutes.Remove(route);
|
|
route.dispose();
|
|
}
|
|
|
|
int _userGesturesInProgress = 0;
|
|
|
|
public bool userGestureInProgress {
|
|
get { return _userGesturesInProgress > 0; }
|
|
}
|
|
|
|
public void didStartUserGesture() {
|
|
_userGesturesInProgress += 1;
|
|
if (_userGesturesInProgress == 1) {
|
|
var route = _history.last();
|
|
var previousRoute = !route.willHandlePopInternally && _history.Count > 1
|
|
? _history[_history.Count - 2]
|
|
: null;
|
|
|
|
foreach (var observer in widget.observers) {
|
|
observer.didStartUserGesture(route, previousRoute);
|
|
}
|
|
}
|
|
}
|
|
|
|
public void didStopUserGesture() {
|
|
D.assert(_userGesturesInProgress > 0);
|
|
_userGesturesInProgress -= 1;
|
|
if (_userGesturesInProgress == 0) {
|
|
foreach (var observer in widget.observers) {
|
|
observer.didStopUserGesture();
|
|
}
|
|
}
|
|
}
|
|
|
|
readonly HashSet<int> _activePointers = new HashSet<int>();
|
|
|
|
void _handlePointerDown(PointerDownEvent evt) {
|
|
_activePointers.Add(evt.pointer);
|
|
}
|
|
|
|
void _handlePointerUpOrCancel(PointerEvent evt) {
|
|
_activePointers.Remove(evt.pointer);
|
|
}
|
|
|
|
void _cancelActivePointers() {
|
|
// TODO flutter issue https://github.com/flutter/flutter/issues/4770
|
|
if (SchedulerBinding.instance.schedulerPhase == SchedulerPhase.idle) {
|
|
// If we're between frames (SchedulerPhase.idle) then absorb any
|
|
// subsequent pointers from this frame. The absorbing flag will be
|
|
// reset in the next frame, see build().
|
|
var absorber = (RenderAbsorbPointer) _overlayKey.currentContext?
|
|
.ancestorRenderObjectOfType(new TypeMatcher<RenderAbsorbPointer>());
|
|
setState(() => {
|
|
if (absorber != null) {
|
|
absorber.absorbing = true;
|
|
}
|
|
});
|
|
}
|
|
|
|
foreach (var pointer in _activePointers) {
|
|
WidgetsBinding.instance.cancelPointer(pointer);
|
|
}
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
D.assert(!_debugLocked);
|
|
D.assert(_history.isNotEmpty());
|
|
return new Listener(
|
|
onPointerDown: _handlePointerDown,
|
|
onPointerUp: _handlePointerUpOrCancel,
|
|
onPointerCancel: _handlePointerUpOrCancel,
|
|
child: new AbsorbPointer(
|
|
absorbing: false, // it's mutated directly by _cancelActivePointers above
|
|
child: new FocusScope(
|
|
focusScopeNode,
|
|
autofocus: true,
|
|
child: new Overlay(
|
|
key: _overlayKey,
|
|
initialEntries: _initialOverlayEntries
|
|
)
|
|
)
|
|
)
|
|
);
|
|
}
|
|
}
|
|
}
|