scrollable_dispose_test.dart 3.19 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5
import 'package:flutter/material.dart';
6 7 8 9 10 11
import 'package:flutter_test/flutter_test.dart';

import 'test_widgets.dart';

void main() {
  testWidgets('simultaneously dispose a widget and end the scroll animation', (WidgetTester tester) async {
12
    await tester.pumpWidget(
13
      Directionality(
14
        textDirection: TextDirection.ltr,
15
        child: FlipWidget(
16
          left: ListView(children: List<Widget>.generate(250, (int i) => Text('$i'))),
17
          right: Container(),
18 19 20
        ),
      ),
    );
21

22
    await tester.fling(find.byType(ListView), const Offset(0.0, -200.0), 1000.0);
23 24
    await tester.pump();

25
    tester.state<FlipWidgetState>(find.byType(FlipWidget)).flip();
26
    await tester.pump(const Duration(hours: 5));
27
  });
28

Dan Field's avatar
Dan Field committed
29
  testWidgets('Disposing a (nested) Scrollable while holding in overscroll does not crash', (WidgetTester tester) async {
30 31 32
    // Regression test for https://github.com/flutter/flutter/issues/27707.

    final ScrollController controller = ScrollController();
33
    final Key outerContainer = GlobalKey();
34 35 36 37 38

    await tester.pumpWidget(
      MaterialApp(
        home: Center(
          child: Container(
39
            key: outerContainer,
40 41 42 43
            color: Colors.purple,
            width: 400.0,
            child: SingleChildScrollView(
              scrollDirection: Axis.horizontal,
44
              child: SizedBox(
45 46 47 48 49
                width: 500.0,
                child: ListView.builder(
                  controller: controller,
                  itemBuilder: (BuildContext context, int index) {
                    return Container(
50
                      color: index.isEven ? Colors.red : Colors.green,
51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
                      height: 200.0,
                      child: Text('Hello $index'),
                    );
                  },
                ),
              ),
            ),
          ),
        ),
      ),
    );

    // Go into overscroll.
    double lastScrollOffset;
    await tester.fling(find.text('Hello 0'), const Offset(0.0, 1000.0), 1000.0);
    await tester.pump(const Duration(milliseconds: 100));
    expect(lastScrollOffset = controller.offset, lessThan(0.0));

    // Reduce the overscroll a little, but don't let it go back to 0.0.
    await tester.pump(const Duration(milliseconds: 100));
    expect(controller.offset, greaterThan(lastScrollOffset));
    expect(controller.offset, lessThan(0.0));
    final double currentOffset = controller.offset;

    // Start a hold activity by putting one pointer down.
76
    await tester.startGesture(tester.getTopLeft(find.byKey(outerContainer)) + const Offset(50.0, 50.0));
77 78 79 80 81 82 83
    await tester.pumpAndSettle(); // This shouldn't change the scroll offset because of the down event above.
    expect(controller.offset, currentOffset);

    // Dispose the scrollables while the finger is still down, this should not crash.
    await tester.pumpWidget(
      MaterialApp(
        home: Container(),
84
      ),
85 86 87
    );
    await tester.pumpAndSettle();
    expect(controller.hasClients, isFalse);
Dan Field's avatar
Dan Field committed
88
  }, variant: const TargetPlatformVariant(<TargetPlatform>{ TargetPlatform.iOS,  TargetPlatform.macOS }));
89
}