| // 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 'package:flutter/foundation.dart'; |
| import 'package:flutter/gestures.dart'; |
| import 'package:flutter/material.dart'; |
| import 'package:flutter_test/flutter_test.dart'; |
| |
| class ExpandingBox extends StatefulWidget { |
| const ExpandingBox({ super.key, required this.collapsedSize, required this.expandedSize }); |
| |
| final double collapsedSize; |
| final double expandedSize; |
| |
| @override |
| State<ExpandingBox> createState() => _ExpandingBoxState(); |
| } |
| |
| class _ExpandingBoxState extends State<ExpandingBox> with AutomaticKeepAliveClientMixin<ExpandingBox> { |
| late double _height; |
| |
| @override |
| void initState() { |
| super.initState(); |
| _height = widget.collapsedSize; |
| } |
| |
| void toggleSize() { |
| setState(() { |
| _height = _height == widget.collapsedSize ? widget.expandedSize : widget.collapsedSize; |
| }); |
| } |
| |
| @override |
| Widget build(BuildContext context) { |
| super.build(context); |
| return Container( |
| height: _height, |
| color: Colors.green, |
| child: Align( |
| alignment: Alignment.bottomCenter, |
| child: TextButton( |
| onPressed: toggleSize, |
| child: const Text('Collapse'), |
| ), |
| ), |
| ); |
| } |
| |
| @override |
| bool get wantKeepAlive => true; |
| } |
| |
| void main() { |
| testWidgets('shrink listview', (WidgetTester tester) async { |
| await tester.pumpWidget(MaterialApp( |
| home: ListView.builder( |
| itemBuilder: (BuildContext context, int index) => index == 0 |
| ? const ExpandingBox(collapsedSize: 400, expandedSize: 1200) |
| : Container(height: 300, color: Colors.red), |
| itemCount: 2, |
| ), |
| )); |
| |
| final ScrollPosition position = tester.state<ScrollableState>(find.byType(Scrollable)).position; |
| expect(position.activity, isInstanceOf<IdleScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 0.0); |
| await tester.tap(find.byType(TextButton)); |
| await tester.pump(); |
| |
| final TestGesture drag1 = await tester.startGesture(const Offset(10.0, 500.0)); |
| await tester.pump(); |
| await drag1.moveTo(const Offset(10.0, 0.0)); |
| await tester.pump(); |
| await drag1.up(); |
| await tester.pump(); |
| expect(position.pixels, moreOrLessEquals(500.0)); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 900.0); |
| |
| final TestGesture drag2 = await tester.startGesture(const Offset(10.0, 500.0)); |
| await tester.pump(); |
| await drag2.moveTo(const Offset(10.0, 100.0)); |
| await tester.pump(); |
| await drag2.up(); |
| await tester.pump(); |
| expect(position.maxScrollExtent, 900.0); |
| expect(position.pixels, moreOrLessEquals(900.0)); |
| |
| await tester.pump(); |
| await tester.tap(find.byType(TextButton)); |
| await tester.pump(); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 100.0); |
| }); |
| |
| testWidgets('shrink listview while dragging', (WidgetTester tester) async { |
| await tester.pumpWidget(MaterialApp( |
| home: ListView.builder( |
| itemBuilder: (BuildContext context, int index) => index == 0 |
| ? const ExpandingBox(collapsedSize: 400, expandedSize: 1200) |
| : Container(height: 300, color: Colors.red), |
| itemCount: 2, |
| ), |
| )); |
| |
| final ScrollPosition position = tester.state<ScrollableState>(find.byType(Scrollable)).position; |
| expect(position.activity, isInstanceOf<IdleScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 0.0); |
| await tester.tap(find.byType(TextButton)); |
| await tester.pump(); // start button animation |
| await tester.pump(const Duration(seconds: 1)); // finish button animation |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 1800.0); |
| expect(position.pixels, 0.0); |
| |
| final TestGesture drag1 = await tester.startGesture(const Offset(10.0, 500.0)); |
| expect(await tester.pumpAndSettle(), 1); // Nothing to animate |
| await drag1.moveTo(const Offset(10.0, 0.0)); |
| expect(await tester.pumpAndSettle(), 2); // Nothing to animate, only one semantics update |
| await drag1.up(); |
| expect(await tester.pumpAndSettle(), 1); // Nothing to animate |
| expect(position.pixels, moreOrLessEquals(500.0)); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 900.0); |
| |
| final TestGesture drag2 = await tester.startGesture(const Offset(10.0, 500.0)); |
| expect(await tester.pumpAndSettle(), 1); // Nothing to animate |
| await drag2.moveTo(const Offset(10.0, 100.0)); |
| expect(await tester.pumpAndSettle(), 2); // Nothing to animate, only one semantics update |
| expect(position.maxScrollExtent, 900.0); |
| expect(position.pixels, lessThanOrEqualTo(900.0)); |
| expect(position.activity, isInstanceOf<DragScrollActivity>()); |
| |
| final _ExpandingBoxState expandingBoxState = tester.state<_ExpandingBoxState>(find.byType(ExpandingBox)); |
| expandingBoxState.toggleSize(); |
| expect(await tester.pumpAndSettle(), 2); // Nothing to animate, only one semantics update |
| expect(position.activity, isInstanceOf<DragScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 100.0); |
| |
| await drag2.moveTo(const Offset(10.0, 150.0)); |
| await drag2.up(); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 50.0); |
| expect(await tester.pumpAndSettle(), 2); // Nothing to animate, only one semantics update |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 50.0); |
| }); |
| |
| testWidgets('shrink listview while ballistic', (WidgetTester tester) async { |
| await tester.pumpWidget(MaterialApp( |
| home: GestureDetector( |
| onTap: () { assert(false); }, |
| child: ListView.builder( |
| physics: const RangeMaintainingScrollPhysics(parent: BouncingScrollPhysics()), |
| itemBuilder: (BuildContext context, int index) => index == 0 |
| ? const ExpandingBox(collapsedSize: 400, expandedSize: 1200) |
| : Container(height: 300, color: Colors.red), |
| itemCount: 2, |
| ), |
| ), |
| )); |
| |
| final _ExpandingBoxState expandingBoxState = tester.state<_ExpandingBoxState>(find.byType(ExpandingBox)); |
| expandingBoxState.toggleSize(); |
| |
| final ScrollPosition position = tester.state<ScrollableState>(find.byType(Scrollable)).position; |
| expect(position.activity, isInstanceOf<IdleScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 0.0); |
| await tester.pump(); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 1800.0); |
| expect(position.pixels, 0.0); |
| |
| final TestGesture drag1 = await tester.startGesture(const Offset(10.0, 10.0)); |
| await tester.pump(); |
| expect(position.activity, isInstanceOf<HoldScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 1800.0); |
| expect(position.pixels, 0.0); |
| await drag1.moveTo(const Offset(10.0, 50.0)); // to get past the slop and trigger the drag |
| await drag1.moveTo(const Offset(10.0, 550.0)); |
| expect(position.pixels, -500.0); |
| await tester.pump(); |
| expect(position.activity, isInstanceOf<DragScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 1800.0); |
| expect(position.pixels, -500.0); |
| await drag1.up(); |
| await tester.pump(); |
| expect(position.activity, isInstanceOf<BallisticScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 1800.0); |
| expect(position.pixels, -500.0); |
| |
| expandingBoxState.toggleSize(); |
| await tester.pump(); // apply physics without moving clock forward |
| expect(position.activity, isInstanceOf<BallisticScrollActivity>()); |
| // TODO(ianh): Determine why the maxScrollOffset is 200.0 here instead of 100.0 or double.infinity. |
| // expect(position.minScrollExtent, 0.0); |
| // expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, -500.0); |
| |
| await tester.pumpAndSettle(); // ignoring the exact effects of the animation |
| expect(position.activity, isInstanceOf<IdleScrollActivity>()); |
| expect(position.minScrollExtent, 0.0); |
| expect(position.maxScrollExtent, 100.0); |
| expect(position.pixels, 0.0); |
| }); |
| |
| testWidgets('expanding page views', (WidgetTester tester) async { |
| await tester.pumpWidget(const Padding(padding: EdgeInsets.only(right: 200.0), child: TabBarDemo())); |
| await tester.tap(find.text('bike')); |
| await tester.pump(); |
| await tester.pump(const Duration(seconds: 1)); |
| final Rect bike1 = tester.getRect(find.byIcon(Icons.directions_bike)); |
| await tester.pumpWidget(const Padding(padding: EdgeInsets.zero, child: TabBarDemo())); |
| final Rect bike2 = tester.getRect(find.byIcon(Icons.directions_bike)); |
| expect(bike2.center, bike1.shift(const Offset(100.0, 0.0)).center); |
| }); |
| |
| testWidgets('changing the size of the viewport when overscrolled', (WidgetTester tester) async { |
| Widget build(double height) { |
| return Directionality( |
| textDirection: TextDirection.rtl, |
| child: ScrollConfiguration( |
| behavior: const RangeMaintainingTestScrollBehavior(), |
| child: Align( |
| alignment: Alignment.topLeft, |
| child: SizedBox( |
| height: height, |
| width: 100.0, |
| child: ListView( |
| children: const <Widget>[SizedBox(height: 100.0, child: Placeholder())], |
| ), |
| ), |
| ), |
| ), |
| ); |
| } |
| await tester.pumpWidget(build(200.0)); |
| // to verify that changing the size of the viewport while you are overdragged does not change the |
| // scroll position, we must ensure that: |
| // - velocity is zero |
| // - scroll extents have changed |
| // - position does not change at the same time |
| // - old position is out of old range AND new range |
| await tester.drag(find.byType(Placeholder), const Offset(0.0, 100.0), touchSlopY: 0.0, warnIfMissed: false); // it'll hit the scrollable |
| await tester.pump(); |
| final Rect oldPosition = tester.getRect(find.byType(Placeholder)); |
| await tester.pumpWidget(build(220.0)); |
| final Rect newPosition = tester.getRect(find.byType(Placeholder)); |
| expect(oldPosition, newPosition); |
| }); |
| |
| testWidgets('inserting and removing an item when overscrolled', (WidgetTester tester) async { |
| // Regression test for https://github.com/flutter/flutter/issues/62890 |
| |
| const double itemExtent = 100.0; |
| final UniqueKey key = UniqueKey(); |
| final Finder finder = find.byKey(key); |
| Widget build({required bool twoItems}) { |
| return Directionality( |
| textDirection: TextDirection.rtl, |
| child: ScrollConfiguration( |
| behavior: const RangeMaintainingTestScrollBehavior(), |
| child: Align( |
| child: SizedBox( |
| width: 100.0, |
| height: 100.0, |
| child: ListView( |
| children: <Widget>[ |
| SizedBox(height: itemExtent, child: Placeholder(key: key)), |
| if (twoItems) |
| const SizedBox(height: itemExtent, child: Placeholder()), |
| ], |
| ), |
| ), |
| ), |
| ), |
| ); |
| } |
| |
| await tester.pumpWidget(build(twoItems: false)); |
| final ScrollPosition position = tester.state<ScrollableState>(find.byType(Scrollable)).position; |
| |
| // overscroll bottom |
| final TestGesture drag1 = await tester.startGesture(tester.getCenter(finder)); |
| await tester.pump(); |
| await drag1.moveBy(const Offset(0.0, -50.0)); |
| await tester.pump(); |
| |
| final double oldOverscroll1 = position.pixels - position.maxScrollExtent; |
| final Rect oldPosition1 = tester.getRect(finder); |
| await tester.pumpWidget(build(twoItems: true)); |
| // verify inserting new item didn't change the position of the first one |
| expect(oldPosition1, tester.getRect(finder)); |
| // verify the overscroll changed by the size of the added item |
| final double newOverscroll1 = position.pixels - position.maxScrollExtent; |
| expect(oldOverscroll1, isPositive); |
| expect(newOverscroll1, isNegative); |
| expect(newOverscroll1, oldOverscroll1 - itemExtent); |
| |
| await drag1.up(); |
| |
| // verify there's no ballistic animation, because we weren't overscrolled |
| expect(await tester.pumpAndSettle(), 1); |
| |
| // overscroll bottom |
| final TestGesture drag2 = await tester.startGesture(tester.getCenter(finder)); |
| await tester.pump(); |
| await drag2.moveBy(const Offset(0.0, -100.0)); |
| await tester.pump(); |
| |
| final double oldOverscroll2 = position.pixels - position.maxScrollExtent; |
| // should find nothing because item is not visible |
| expect(finder, findsNothing); |
| await tester.pumpWidget(build(twoItems: false)); |
| // verify removing an item changed the position of the first one, because prior it was not visible |
| expect(oldPosition1, tester.getRect(finder)); |
| // verify the overscroll was maintained |
| final double newOverscroll2 = position.pixels - position.maxScrollExtent; |
| expect(oldOverscroll2, isPositive); |
| expect(oldOverscroll2, newOverscroll2); |
| |
| await drag2.up(); |
| |
| // verify there's a ballistic animation from overscroll |
| expect(await tester.pumpAndSettle(), 9); |
| }); |
| } |
| |
| class TabBarDemo extends StatelessWidget { |
| const TabBarDemo({super.key}); |
| |
| @override |
| Widget build(BuildContext context) { |
| return MaterialApp( |
| home: DefaultTabController( |
| length: 3, |
| child: Scaffold( |
| appBar: AppBar( |
| bottom: const TabBar( |
| tabs: <Widget>[ |
| Tab(text: 'car'), |
| Tab(text: 'transit'), |
| Tab(text: 'bike'), |
| ], |
| ), |
| title: const Text('Tabs Demo'), |
| ), |
| body: const TabBarView( |
| children: <Widget>[ |
| Icon(Icons.directions_car), |
| Icon(Icons.directions_transit), |
| Icon(Icons.directions_bike), |
| ], |
| ), |
| ), |
| ), |
| ); |
| } |
| } |
| |
| class RangeMaintainingTestScrollBehavior extends ScrollBehavior { |
| const RangeMaintainingTestScrollBehavior(); |
| |
| @override |
| TargetPlatform getPlatform(BuildContext context) => defaultTargetPlatform; |
| |
| @override |
| Widget buildOverscrollIndicator(BuildContext context, Widget child, ScrollableDetails details) { |
| return child; |
| } |
| |
| @override |
| Widget buildScrollbar(BuildContext context, Widget child, ScrollableDetails details) { |
| return child; |
| } |
| |
| @override |
| GestureVelocityTrackerBuilder velocityTrackerBuilder(BuildContext context) { |
| return (PointerEvent event) => VelocityTracker.withKind(event.kind); |
| } |
| |
| @override |
| ScrollPhysics getScrollPhysics(BuildContext context) { |
| return const BouncingScrollPhysics(parent: RangeMaintainingScrollPhysics()); |
| } |
| } |