| // Copyright 2016 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/material.dart'; |
| |
| bool willPopValue = false; |
| |
| class SamplePage extends StatefulWidget { |
| @override |
| SamplePageState createState() => new SamplePageState(); |
| } |
| |
| class SamplePageState extends State<SamplePage> { |
| ModalRoute<void> _route; |
| |
| Future<bool> _callback() async => willPopValue; |
| |
| @override |
| void didChangeDependencies() { |
| super.didChangeDependencies(); |
| _route?.removeScopedWillPopCallback(_callback); |
| _route = ModalRoute.of(context); |
| _route?.addScopedWillPopCallback(_callback); |
| } |
| |
| @override |
| void dispose() { |
| super.dispose(); |
| _route?.removeScopedWillPopCallback(_callback); |
| } |
| |
| @override |
| Widget build(BuildContext context) { |
| return new Scaffold( |
| appBar: new AppBar(title: const Text('Sample Page')), |
| ); |
| } |
| } |
| |
| int willPopCount = 0; |
| |
| class SampleForm extends StatelessWidget { |
| const SampleForm({ Key key, this.callback }) : super(key: key); |
| |
| final WillPopCallback callback; |
| |
| @override |
| Widget build(BuildContext context) { |
| return new Scaffold( |
| appBar: new AppBar(title: const Text('Sample Form')), |
| body: new SizedBox.expand( |
| child: new Form( |
| onWillPop: () { |
| willPopCount += 1; |
| return callback(); |
| }, |
| child: const TextField(), |
| ), |
| ), |
| ); |
| } |
| } |
| |
| // Expose the protected hasScopedWillPopCallback getter |
| class TestPageRoute<T> extends MaterialPageRoute<T> { |
| TestPageRoute({ WidgetBuilder builder }) |
| : super(builder: builder, maintainState: true); |
| |
| bool get hasCallback => super.hasScopedWillPopCallback; |
| } |
| |
| |
| void main() { |
| testWidgets('ModalRoute scopedWillPopupCallback can inhibit back button', (WidgetTester tester) async { |
| await tester.pumpWidget( |
| new MaterialApp( |
| home: new Scaffold( |
| appBar: new AppBar(title: const Text('Home')), |
| body: new Builder( |
| builder: (BuildContext context) { |
| return new Center( |
| child: new FlatButton( |
| child: const Text('X'), |
| onPressed: () { |
| showDialog<void>( |
| context: context, |
| builder: (BuildContext context) => new SamplePage(), |
| ); |
| }, |
| ), |
| ); |
| }, |
| ), |
| ), |
| ), |
| ); |
| |
| expect(find.byTooltip('Back'), findsNothing); |
| expect(find.text('Sample Page'), findsNothing); |
| |
| await tester.tap(find.text('X')); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| |
| expect(find.text('Sample Page'), findsOneWidget); |
| |
| willPopValue = false; |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| expect(find.text('Sample Page'), findsOneWidget); |
| |
| // Use didPopRoute() to simulate the system back button. Check that |
| // didPopRoute() indicates that the notification was handled. |
| final dynamic widgetsAppState = tester.state(find.byType(WidgetsApp)); |
| expect(await widgetsAppState.didPopRoute(), isTrue); |
| expect(find.text('Sample Page'), findsOneWidget); |
| |
| willPopValue = true; |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| expect(find.text('Sample Page'), findsNothing); |
| }); |
| |
| testWidgets('Form.willPop can inhibit back button', (WidgetTester tester) async { |
| Widget buildFrame() { |
| return new MaterialApp( |
| home: new Scaffold( |
| appBar: new AppBar(title: const Text('Home')), |
| body: new Builder( |
| builder: (BuildContext context) { |
| return new Center( |
| child: new FlatButton( |
| child: const Text('X'), |
| onPressed: () { |
| Navigator.of(context).push(new MaterialPageRoute<void>( |
| builder: (BuildContext context) { |
| return new SampleForm( |
| callback: () => new Future<bool>.value(willPopValue), |
| ); |
| }, |
| )); |
| }, |
| ), |
| ); |
| }, |
| ), |
| ), |
| ); |
| } |
| |
| await tester.pumpWidget(buildFrame()); |
| |
| await tester.tap(find.text('X')); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| |
| expect(find.text('Sample Form'), findsOneWidget); |
| |
| willPopValue = false; |
| willPopCount = 0; |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); // Start the pop "back" operation. |
| await tester.pump(); // Complete the willPop() Future. |
| await tester.pump(const Duration(seconds: 1)); // Wait until it has finished. |
| expect(find.text('Sample Form'), findsOneWidget); |
| expect(willPopCount, 1); |
| |
| willPopValue = true; |
| willPopCount = 0; |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); // Start the pop "back" operation. |
| await tester.pump(); // Complete the willPop() Future. |
| await tester.pump(const Duration(seconds: 1)); // Wait until it has finished. |
| expect(find.text('Sample Form'), findsNothing); |
| expect(willPopCount, 1); |
| }); |
| |
| testWidgets('Form.willPop callbacks do not accumulate', (WidgetTester tester) async { |
| Future<bool> showYesNoAlert(BuildContext context) { |
| return showDialog<bool>( |
| context: context, |
| builder: (BuildContext context) { |
| return new AlertDialog( |
| actions: <Widget> [ |
| new FlatButton( |
| child: const Text('YES'), |
| onPressed: () { Navigator.of(context).pop(true); }, |
| ), |
| new FlatButton( |
| child: const Text('NO'), |
| onPressed: () { Navigator.of(context).pop(false); }, |
| ), |
| ], |
| ); |
| }, |
| ); |
| } |
| |
| Widget buildFrame() { |
| return new MaterialApp( |
| home: new Scaffold( |
| appBar: new AppBar(title: const Text('Home')), |
| body: new Builder( |
| builder: (BuildContext context) { |
| return new Center( |
| child: new FlatButton( |
| child: const Text('X'), |
| onPressed: () { |
| Navigator.of(context).push(new MaterialPageRoute<void>( |
| builder: (BuildContext context) { |
| return new SampleForm( |
| callback: () => showYesNoAlert(context), |
| ); |
| } |
| )); |
| }, |
| ), |
| ); |
| }, |
| ), |
| ), |
| ); |
| } |
| |
| await tester.pumpWidget(buildFrame()); |
| |
| await tester.tap(find.text('X')); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| |
| expect(find.text('Sample Form'), findsOneWidget); |
| |
| // Press the Scaffold's back button. This causes the willPop callback |
| // to run, which shows the YES/NO Alert Dialog. Veto the back operation |
| // by pressing the Alert's NO button. |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); // Start the pop "back" operation. |
| await tester.pump(); // Call willPop which will show an Alert. |
| await tester.tap(find.text('NO')); |
| await tester.pump(); // Start the dismiss animation. |
| await tester.pump(); // Resolve the willPop callback. |
| await tester.pump(const Duration(seconds: 1)); // Wait until it has finished. |
| expect(find.text('Sample Form'), findsOneWidget); |
| |
| // Do it again. Note that each time the Alert is shown and dismissed |
| // the FormState's didChangeDependencies() method runs. We're making sure |
| // that the didChangeDependencies() method doesn't add an extra willPop |
| // callback. |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); // Start the pop "back" operation. |
| await tester.pump(); // Call willPop which will show an Alert. |
| await tester.tap(find.text('NO')); |
| await tester.pump(); // Start the dismiss animation. |
| await tester.pump(); // Resolve the willPop callback. |
| await tester.pump(const Duration(seconds: 1)); // Wait until it has finished. |
| expect(find.text('Sample Form'), findsOneWidget); |
| |
| // This time really dismiss the SampleForm by pressing the Alert's |
| // YES button. |
| await tester.tap(find.byTooltip('Back')); |
| await tester.pump(); // Start the pop "back" operation. |
| await tester.pump(); // Call willPop which will show an Alert. |
| await tester.tap(find.text('YES')); |
| await tester.pump(); // Start the dismiss animation. |
| await tester.pump(); // Resolve the willPop callback. |
| await tester.pump(const Duration(seconds: 1)); // Wait until it has finished. |
| expect(find.text('Sample Form'), findsNothing); |
| }); |
| |
| testWidgets('Route.scopedWillPop callbacks do not accumulate', (WidgetTester tester) async { |
| StateSetter contentsSetState; // call this to rebuild the route's SampleForm contents |
| bool contentsEmpty = false; // when true, don't include the SampleForm in the route |
| |
| final TestPageRoute<Null> route = new TestPageRoute<Null>( |
| builder: (BuildContext context) { |
| return new StatefulBuilder( |
| builder: (BuildContext context, StateSetter setState) { |
| contentsSetState = setState; |
| return contentsEmpty ? new Container() : new SampleForm(key: new UniqueKey()); |
| } |
| ); |
| }, |
| ); |
| |
| Widget buildFrame() { |
| return new MaterialApp( |
| home: new Scaffold( |
| appBar: new AppBar(title: const Text('Home')), |
| body: new Builder( |
| builder: (BuildContext context) { |
| return new Center( |
| child: new FlatButton( |
| child: const Text('X'), |
| onPressed: () { |
| Navigator.of(context).push(route); |
| }, |
| ), |
| ); |
| }, |
| ), |
| ), |
| ); |
| } |
| |
| await tester.pumpWidget(buildFrame()); |
| |
| await tester.tap(find.text('X')); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| |
| expect(find.text('Sample Form'), findsOneWidget); |
| expect(route.hasCallback, isTrue); |
| |
| // Rebuild the route's SampleForm child an additional 3x for good measure. |
| contentsSetState(() { }); |
| await tester.pump(); |
| contentsSetState(() { }); |
| await tester.pump(); |
| contentsSetState(() { }); |
| await tester.pump(); |
| |
| // Now build the route's contents without the sample form. |
| contentsEmpty = true; |
| contentsSetState(() { }); |
| await tester.pump(); |
| |
| expect(route.hasCallback, isFalse); |
| }); |
| } |