Commit de772647 authored by Ian Hickson's avatar Ian Hickson

Merge pull request #525 from Hixie/yak3-routes-typed

Give type arguments to routes.
parents 1fbec91f 13f3a9be
...@@ -115,8 +115,8 @@ class Dialog extends StatelessComponent { ...@@ -115,8 +115,8 @@ class Dialog extends StatelessComponent {
} }
} }
class _DialogRoute extends ModalRoute { class _DialogRoute<T> extends ModalRoute<T> {
_DialogRoute({ Completer completer, this.child }) : super(completer: completer); _DialogRoute({ Completer<T> completer, this.child }) : super(completer: completer);
final Widget child; final Widget child;
......
...@@ -36,7 +36,7 @@ class _MaterialPageTransition extends TransitionWithChild { ...@@ -36,7 +36,7 @@ class _MaterialPageTransition extends TransitionWithChild {
} }
} }
class MaterialPageRoute extends ModalRoute { class MaterialPageRoute<T> extends ModalRoute<T> {
MaterialPageRoute({ MaterialPageRoute({
this.builder, this.builder,
NamedRouteSettings settings: const NamedRouteSettings() NamedRouteSettings settings: const NamedRouteSettings()
......
...@@ -5,10 +5,10 @@ ...@@ -5,10 +5,10 @@
import 'framework.dart'; import 'framework.dart';
import 'overlay.dart'; import 'overlay.dart';
abstract class Route { abstract class Route<T> {
List<OverlayEntry> get overlayEntries; List<OverlayEntry> get overlayEntries;
void didPush(OverlayState overlay, OverlayEntry insertionPoint) { } void didPush(OverlayState overlay, OverlayEntry insertionPoint) { }
void didPop(dynamic result) { } void didPop(T result) { }
/// The given route has been pushed onto the navigator after this route. /// The given route has been pushed onto the navigator after this route.
/// Return true if the route before this one should be notified also. The /// Return true if the route before this one should be notified also. The
...@@ -140,6 +140,9 @@ class NavigatorState extends State<Navigator> { ...@@ -140,6 +140,9 @@ class NavigatorState extends State<Navigator> {
/// Do not use this for ModalRoutes, or indeed anything other than /// Do not use this for ModalRoutes, or indeed anything other than
/// StateRoutes. Doing so would cause very odd results, e.g. ModalRoutes would /// StateRoutes. Doing so would cause very odd results, e.g. ModalRoutes would
/// get confused about who is current. /// get confused about who is current.
///
/// The type of the result argument, if provided, must match the type argument
/// of the class of the given route. (In practice, this is usually "dynamic".)
void remove(Route route, [dynamic result]) { void remove(Route route, [dynamic result]) {
assert(_modal.contains(route)); assert(_modal.contains(route));
assert(route.overlayEntries.isEmpty); assert(route.overlayEntries.isEmpty);
...@@ -155,6 +158,10 @@ class NavigatorState extends State<Navigator> { ...@@ -155,6 +158,10 @@ class NavigatorState extends State<Navigator> {
/// Removes the current route, notifying the observer (if any), and the /// Removes the current route, notifying the observer (if any), and the
/// previous routes (using [Route.didPopNext]). /// previous routes (using [Route.didPopNext]).
///
/// The type of the result argument, if provided, must match the type argument
/// of the class of the current route. (In practice, this is usually
/// "dynamic".)
void pop([dynamic result]) { void pop([dynamic result]) {
setState(() { setState(() {
// We use setState to guarantee that we'll rebuild, since the routes can't // We use setState to guarantee that we'll rebuild, since the routes can't
......
...@@ -23,6 +23,7 @@ class StateRoute extends Route { ...@@ -23,6 +23,7 @@ class StateRoute extends Route {
List<OverlayEntry> get overlayEntries => const <OverlayEntry>[]; List<OverlayEntry> get overlayEntries => const <OverlayEntry>[];
void didPop(dynamic result) { void didPop(dynamic result) {
assert(result == null);
if (onPop != null) if (onPop != null)
onPop(); onPop();
} }
...@@ -31,7 +32,7 @@ class StateRoute extends Route { ...@@ -31,7 +32,7 @@ class StateRoute extends Route {
bool didPopNext(Route nextRoute) => true; bool didPopNext(Route nextRoute) => true;
} }
abstract class OverlayRoute extends Route { abstract class OverlayRoute<T> extends Route<T> {
List<WidgetBuilder> get builders => const <WidgetBuilder>[]; List<WidgetBuilder> get builders => const <WidgetBuilder>[];
List<OverlayEntry> get overlayEntries => _overlayEntries; List<OverlayEntry> get overlayEntries => _overlayEntries;
...@@ -46,7 +47,7 @@ abstract class OverlayRoute extends Route { ...@@ -46,7 +47,7 @@ abstract class OverlayRoute extends Route {
} }
// Subclasses shouldn't call this if they want to delay the finished() call. // Subclasses shouldn't call this if they want to delay the finished() call.
void didPop(dynamic result) { void didPop(T result) {
finished(); finished();
} }
...@@ -64,11 +65,15 @@ abstract class OverlayRoute extends Route { ...@@ -64,11 +65,15 @@ abstract class OverlayRoute extends Route {
} }
} }
// TODO(abarth): Should we add a type for the result? abstract class TransitionRoute<T> extends OverlayRoute<T> {
abstract class TransitionRoute extends OverlayRoute { TransitionRoute({
TransitionRoute({ this.completer }); Completer<T> popCompleter,
Completer<T> transitionCompleter
}) : _popCompleter = popCompleter,
_transitionCompleter = transitionCompleter;
final Completer completer; final Completer<T> _popCompleter;
final Completer<T> _transitionCompleter;
Duration get transitionDuration; Duration get transitionDuration;
bool get opaque; bool get opaque;
...@@ -82,7 +87,7 @@ abstract class TransitionRoute extends OverlayRoute { ...@@ -82,7 +87,7 @@ abstract class TransitionRoute extends OverlayRoute {
return new Performance(duration: duration, debugLabel: debugLabel); return new Performance(duration: duration, debugLabel: debugLabel);
} }
dynamic _result; T _result;
void _handleStatusChanged(PerformanceStatus status) { void _handleStatusChanged(PerformanceStatus status) {
switch (status) { switch (status) {
...@@ -97,7 +102,6 @@ abstract class TransitionRoute extends OverlayRoute { ...@@ -97,7 +102,6 @@ abstract class TransitionRoute extends OverlayRoute {
break; break;
case PerformanceStatus.dismissed: case PerformanceStatus.dismissed:
super.finished(); // clear the overlays super.finished(); // clear the overlays
completer?.complete(_result);
break; break;
} }
} }
...@@ -109,9 +113,15 @@ abstract class TransitionRoute extends OverlayRoute { ...@@ -109,9 +113,15 @@ abstract class TransitionRoute extends OverlayRoute {
super.didPush(overlay, insertionPoint); super.didPush(overlay, insertionPoint);
} }
void didPop(dynamic result) { void didPop(T result) {
_result = result; _result = result;
_performance.reverse(); _performance.reverse();
_popCompleter?.complete(_result);
}
void finished() {
super.finished();
_transitionCompleter?.complete(_result);
} }
String get debugLabel => '$runtimeType'; String get debugLabel => '$runtimeType';
...@@ -202,11 +212,11 @@ class ModalPosition { ...@@ -202,11 +212,11 @@ class ModalPosition {
final double left; final double left;
} }
abstract class ModalRoute extends TransitionRoute { abstract class ModalRoute<T> extends TransitionRoute<T> {
ModalRoute({ ModalRoute({
Completer completer, Completer<T> completer,
this.settings: const NamedRouteSettings() this.settings: const NamedRouteSettings()
}) : super(completer: completer); }) : super(popCompleter: completer);
// The API for general users of this class // The API for general users of this class
...@@ -263,7 +273,7 @@ abstract class ModalRoute extends TransitionRoute { ...@@ -263,7 +273,7 @@ abstract class ModalRoute extends TransitionRoute {
super.didPush(overlay, insertionPoint); super.didPush(overlay, insertionPoint);
} }
void didPop(dynamic result) { void didPop(T result) {
assert(_isCurrent); assert(_isCurrent);
_isCurrent = false; _isCurrent = false;
super.didPop(result); super.didPop(result);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment