Commit 2eecb8be authored by Ian Hickson's avatar Ian Hickson Committed by GitHub

Improve Notification API (#7403)

- better documentation

- verify (at run time) that onNotification doesn't return null, and
  report copious helpful information if it does.

- add a toString/debugFillDescription convention to Notification.

- actually test Notification
parent d3faada9
...@@ -2,37 +2,72 @@ ...@@ -2,37 +2,72 @@
// Use of this source code is governed by a BSD-style license that can be // Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. // found in the LICENSE file.
import 'package:flutter/foundation.dart';
import 'framework.dart'; import 'framework.dart';
/// Signature for [Notification] listeners. /// Signature for [Notification] listeners.
/// ///
/// Return true to cancel the notification bubbling. /// Return true to cancel the notification bubbling. Return false to allow the
/// notification to continue to be dispatched to further ancestors.
/// ///
/// Used by [NotificationListener.onNotification]. /// Used by [NotificationListener.onNotification].
typedef bool NotificationListenerCallback<T extends Notification>(T notification); typedef bool NotificationListenerCallback<T extends Notification>(T notification);
/// A notification that can bubble up the widget tree. /// A notification that can bubble up the widget tree.
abstract class Notification { abstract class Notification {
/// Applied to each ancestor of the [dispatch] target. Dispatches this /// Applied to each ancestor of the [dispatch] target.
/// Notification to ancestor [NotificationListener] widgets. ///
/// The [Notification] class implementation of this method dispatches the
/// given [Notification] to each ancestor [NotificationListener] widget.
///
/// Subclasses can override this to apply additional filtering or to update
/// the notification as it is bubbled (for example, increasing a `depth` field
/// for each ancestor of a particular type).
@protected
@mustCallSuper
bool visitAncestor(Element element) { bool visitAncestor(Element element) {
if (element is StatelessElement && if (element is StatelessElement &&
element.widget is NotificationListener<Notification>) { element.widget is NotificationListener<Notification>) {
final NotificationListener<Notification> widget = element.widget; final NotificationListener<Notification> widget = element.widget;
if (widget._dispatch(this)) // that function checks the type dynamically if (widget._dispatch(this, element)) // that function checks the type dynamically
return false; return false;
} }
return true; return true;
} }
/// Start bubbling this notification at the given build context. /// Start bubbling this notification at the given build context.
///
/// To receive notifications, use a [NotificationListener].
void dispatch(BuildContext target) { void dispatch(BuildContext target) {
assert(target != null); // Only call dispatch if the widget's State is still mounted. assert(target != null); // Only call dispatch if the widget's State is still mounted.
target.visitAncestorElements(visitAncestor); target.visitAncestorElements(visitAncestor);
} }
@override
String toString() {
List<String> description = <String>[];
debugFillDescription(description);
return '$runtimeType(${description.join(", ")})';
}
/// Add additional information to the given description for use by [toString].
///
/// This method makes it easier for subclasses to coordinate to provide a
/// high-quality [toString] implementation. The [toString] implementation on
/// the [Notification] base class calls [debugFillDescription] to collect
/// useful information from subclasses to incorporate into its return value.
///
/// If you override this, make sure to start your method with a call to
/// `super.debugFillDescription(description)`.
@protected
@mustCallSuper
void debugFillDescription(List<String> description) { }
} }
/// A widget that listens for [Notification]s bubbling up the tree. /// A widget that listens for [Notification]s bubbling up the tree.
///
/// To dispatch notifications, use the [Notification.dispatch] method.
class NotificationListener<T extends Notification> extends StatelessWidget { class NotificationListener<T extends Notification> extends StatelessWidget {
/// Creates a widget that listens for notifications. /// Creates a widget that listens for notifications.
NotificationListener({ NotificationListener({
...@@ -44,12 +79,35 @@ class NotificationListener<T extends Notification> extends StatelessWidget { ...@@ -44,12 +79,35 @@ class NotificationListener<T extends Notification> extends StatelessWidget {
/// The widget below this widget in the tree. /// The widget below this widget in the tree.
final Widget child; final Widget child;
/// Called when a notification of the appropriate type arrives at this location in the tree. /// Called when a notification of the appropriate type arrives at this
/// location in the tree.
///
/// Return true to cancel the notification bubbling. Return false to allow the
/// notification to continue to be dispatched to further ancestors.
///
/// The notification's [Notification.visitAncestor] method is called for each
/// ancestor, and invokes this callback as appropriate.
final NotificationListenerCallback<T> onNotification; final NotificationListenerCallback<T> onNotification;
bool _dispatch(Notification notification) { bool _dispatch(Notification notification, Element element) {
if (onNotification != null && notification is T) if (onNotification != null && notification is T) {
return onNotification(notification); bool result = onNotification(notification);
assert(() {
if (result == null)
throw new FlutterError(
'NotificationListener<$T> handler returned null.\n'
'The onNotification handler for the NotificationListener with the '
'following element returned null:\n'
' $element\n'
'The ancestor chain for this widget was as follows:\n'
' ${element.debugGetCreatorChain(12)}\n'
'Notification listeners must return true to stop the notification bubbling, '
'or false to allow it to continue (the common case is returning false).'
);
return true;
});
return result;
}
return false; return false;
} }
...@@ -63,10 +121,18 @@ class NotificationListener<T extends Notification> extends StatelessWidget { ...@@ -63,10 +121,18 @@ class NotificationListener<T extends Notification> extends StatelessWidget {
/// ///
/// Useful if, for instance, you're trying to align multiple descendants. /// Useful if, for instance, you're trying to align multiple descendants.
/// ///
/// Be aware that in the widgets library, only the [Scrollable] classes dispatch /// In the widgets library, only the [SizeChangedLayoutNotifier] class and
/// this notification. (Transitions, in particular, do not.) Changing one's /// [Scrollable] classes dispatch this notification (specifically, they dispatch
/// layout in one's build function does not cause this notification to be /// [SizeChangedLayoutNotification]s and [ScrollNotification]s respectively).
/// dispatched automatically. If an ancestor expects to be notified for any /// Transitions, in particular, do not. Changing one's layout in one's build
/// layout change, make sure you only use widgets that either never change /// function does not cause this notification to be dispatched automatically. If
/// layout, or that do notify their ancestors when appropriate. /// an ancestor expects to be notified for any layout change, make sure you
/// either only use widgets that never change layout, or that notify their
/// ancestors when appropriate, or alternatively, dispatch the notifications
/// yourself when appropriate.
///
/// Also, since this notification is sent when the layout is changed, it is only
/// useful for paint effects that depend on the layout. If you were to use this
/// notification to change the build, for instance, you would always be one
/// frame behind, which would look really ugly and laggy.
class LayoutChangedNotification extends Notification { } class LayoutChangedNotification extends Notification { }
// Copyright 2015 The Chromium 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 'package:flutter_test/flutter_test.dart';
import 'package:flutter/widgets.dart';
class MyNotification extends Notification { }
void main() {
testWidgets('Notification basics - toString', (WidgetTester tester) async {
expect(new MyNotification(), hasOneLineDescription);
});
testWidgets('Notification basics - dispatch', (WidgetTester tester) async {
List<dynamic> log = <dynamic>[];
GlobalKey key = new GlobalKey();
await tester.pumpWidget(new NotificationListener<MyNotification>(
onNotification: (MyNotification value) {
log.add('a');
log.add(value);
return true;
},
child: new NotificationListener<MyNotification>(
onNotification: (MyNotification value) {
log.add('b');
log.add(value);
return false;
},
child: new Container(key: key),
),
));
expect(log, isEmpty);
final Notification notification = new MyNotification();
expect(() { notification.dispatch(key.currentContext); }, isNot(throwsException));
expect(log, <dynamic>['b', notification, 'a', notification]);
});
testWidgets('Notification basics - cancel', (WidgetTester tester) async {
List<dynamic> log = <dynamic>[];
GlobalKey key = new GlobalKey();
await tester.pumpWidget(new NotificationListener<MyNotification>(
onNotification: (MyNotification value) {
log.add('a - error');
log.add(value);
return true;
},
child: new NotificationListener<MyNotification>(
onNotification: (MyNotification value) {
log.add('b');
log.add(value);
return true;
},
child: new Container(key: key),
),
));
expect(log, isEmpty);
final Notification notification = new MyNotification();
expect(() { notification.dispatch(key.currentContext); }, isNot(throwsException));
expect(log, <dynamic>['b', notification]);
});
testWidgets('Notification basics - listener null return value', (WidgetTester tester) async {
GlobalKey key = new GlobalKey();
await tester.pumpWidget(new NotificationListener<MyNotification>(
onNotification: (MyNotification value) { },
child: new Container(key: key),
));
expect(() { new MyNotification().dispatch(key.currentContext); }, throwsA(new isInstanceOf<AssertionError>()));
});
}
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