dismissable_test.dart 10.6 KB
Newer Older
Hixie's avatar
Hixie committed
1 2 3 4
// 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.

Adam Barth's avatar
Adam Barth committed
5
import 'package:flutter_test/flutter_test.dart';
6 7
import 'package:flutter/rendering.dart';
import 'package:flutter/widgets.dart';
8 9
import 'package:test/test.dart';

10
const double itemExtent = 100.0;
11
Axis scrollDirection = Axis.vertical;
12
DismissDirection dismissDirection = DismissDirection.horizontal;
13
List<int> dismissedItems = <int>[];
14

15
void handleOnResized(int item) {
16 17 18
  expect(dismissedItems.contains(item), isFalse);
}

19
void handleOnDismissed(int item) {
20 21 22 23
  expect(dismissedItems.contains(item), isFalse);
  dismissedItems.add(item);
}

24
Widget buildDismissableItem(int item) {
25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
  return new Dismissable(
    key: new ValueKey<int>(item),
    direction: dismissDirection,
    onDismissed: () { handleOnDismissed(item); },
    onResized: () { handleOnResized(item); },
    child: new Container(
      width: itemExtent,
      height: itemExtent,
      child: new Text(item.toString())
    )
  );
}

Widget widgetBuilder() {
  return new Container(
    padding: const EdgeDims.all(10.0),
41
    child: new ScrollableList(
42
      scrollDirection: scrollDirection,
43 44 45 46
      itemExtent: itemExtent,
      children: <int>[0, 1, 2, 3, 4].where(
        (int i) => !dismissedItems.contains(i)
      ).map(buildDismissableItem)
47 48 49 50
    )
  );
}

Hixie's avatar
Hixie committed
51 52
void dismissElement(WidgetTester tester, Element itemElement, { DismissDirection gestureDirection }) {
  assert(itemElement != null);
53 54 55 56 57 58 59
  assert(gestureDirection != DismissDirection.horizontal);
  assert(gestureDirection != DismissDirection.vertical);

  Point downLocation;
  Point upLocation;
  switch(gestureDirection) {
    case DismissDirection.left:
60 61
      // getTopRight() returns a point that's just beyond itemWidget's right
      // edge and outside the Dismissable event listener's bounds.
62 63
      downLocation = tester.getTopRight(itemElement) + const Offset(-0.1, 0.0);
      upLocation = tester.getTopLeft(itemElement);
64 65
      break;
    case DismissDirection.right:
66 67
      // we do the same thing here to keep the test symmetric
      downLocation = tester.getTopLeft(itemElement) + const Offset(0.1, 0.0);
68
      upLocation = tester.getTopRight(itemElement);
69 70
      break;
    case DismissDirection.up:
71 72
      // getBottomLeft() returns a point that's just below itemWidget's bottom
      // edge and outside the Dismissable event listener's bounds.
73 74
      downLocation = tester.getBottomLeft(itemElement) + const Offset(0.0, -0.1);
      upLocation = tester.getTopLeft(itemElement);
75 76
      break;
    case DismissDirection.down:
77 78
      // again with doing the same here for symmetry
      downLocation = tester.getTopLeft(itemElement) + const Offset(0.1, 0.0);
79
      upLocation = tester.getBottomLeft(itemElement);
80
      break;
81 82
    default:
      fail("unsupported gestureDirection");
83 84
  }

Adam Barth's avatar
Adam Barth committed
85 86 87
  TestGesture gesture = tester.startGesture(downLocation, pointer: 5);
  gesture.moveTo(upLocation);
  gesture.up();
Hixie's avatar
Hixie committed
88 89 90 91 92 93 94 95 96 97
}

void dismissItem(WidgetTester tester, int item, { DismissDirection gestureDirection }) {
  assert(gestureDirection != DismissDirection.horizontal);
  assert(gestureDirection != DismissDirection.vertical);

  Element itemElement = tester.findText(item.toString());
  expect(itemElement, isNotNull);

  dismissElement(tester, itemElement, gestureDirection: gestureDirection);
98

99 100 101 102 103
  tester.pumpWidget(widgetBuilder()); // start the slide
  tester.pumpWidget(widgetBuilder(), const Duration(seconds: 1)); // finish the slide and start shrinking...
  tester.pumpWidget(widgetBuilder()); // first frame of shrinking animation
  tester.pumpWidget(widgetBuilder(), const Duration(seconds: 1)); // finish the shrinking and call the callback...
  tester.pumpWidget(widgetBuilder()); // rebuild after the callback removes the entry
104 105
}

Hixie's avatar
Hixie committed
106 107 108 109 110 111 112 113 114 115 116 117 118
class Test1215DismissableComponent extends StatelessComponent {
  Test1215DismissableComponent(this.text);
  final String text;
  Widget build(BuildContext context) {
    return new Dismissable(
      child: new AspectRatio(
        aspectRatio: 1.0,
        child: new Text(this.text)
      )
    );
  }
}

119
void main() {
120
  test('Horizontal drag triggers dismiss scrollDirection=vertical', () {
121
    testWidgets((WidgetTester tester) {
122
      scrollDirection = Axis.vertical;
123
      dismissDirection = DismissDirection.horizontal;
124
      dismissedItems = <int>[];
125 126 127 128 129 130 131 132 133 134 135 136

      tester.pumpWidget(widgetBuilder());
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.right);
      expect(tester.findText('0'), isNull);
      expect(dismissedItems, equals([0]));

      dismissItem(tester, 1, gestureDirection: DismissDirection.left);
      expect(tester.findText('1'), isNull);
      expect(dismissedItems, equals([0, 1]));
    });
137 138 139
  });

  test('Vertical drag triggers dismiss scrollDirection=horizontal', () {
140
    testWidgets((WidgetTester tester) {
141
      scrollDirection = Axis.horizontal;
142
      dismissDirection = DismissDirection.vertical;
143
      dismissedItems = <int>[];
144 145 146 147 148 149 150 151 152 153 154 155

      tester.pumpWidget(widgetBuilder());
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.up);
      expect(tester.findText('0'), isNull);
      expect(dismissedItems, equals([0]));

      dismissItem(tester, 1, gestureDirection: DismissDirection.down);
      expect(tester.findText('1'), isNull);
      expect(dismissedItems, equals([0, 1]));
    });
156 157 158
  });

  test('drag-left with DismissDirection.left triggers dismiss', () {
159
    testWidgets((WidgetTester tester) {
160
      scrollDirection = Axis.vertical;
161
      dismissDirection = DismissDirection.left;
162
      dismissedItems = <int>[];
163 164 165 166 167 168 169 170 171 172 173 174

      tester.pumpWidget(widgetBuilder());
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.right);
      expect(tester.findText('0'), isNotNull);
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.left);
      expect(tester.findText('0'), isNull);
      expect(dismissedItems, equals([0]));
    });
175 176 177
  });

  test('drag-right with DismissDirection.right triggers dismiss', () {
178
    testWidgets((WidgetTester tester) {
179
      scrollDirection = Axis.vertical;
180
      dismissDirection = DismissDirection.right;
181
      dismissedItems = <int>[];
182 183 184 185 186 187 188 189 190 191 192 193

      tester.pumpWidget(widgetBuilder());
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.left);
      expect(tester.findText('0'), isNotNull);
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.right);
      expect(tester.findText('0'), isNull);
      expect(dismissedItems, equals([0]));
    });
194 195 196
  });

  test('drag-up with DismissDirection.up triggers dismiss', () {
197
    testWidgets((WidgetTester tester) {
198
      scrollDirection = Axis.horizontal;
199
      dismissDirection = DismissDirection.up;
200
      dismissedItems = <int>[];
201 202 203 204 205 206 207 208 209 210 211 212

      tester.pumpWidget(widgetBuilder());
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.down);
      expect(tester.findText('0'), isNotNull);
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.up);
      expect(tester.findText('0'), isNull);
      expect(dismissedItems, equals([0]));
    });
213 214 215
  });

  test('drag-down with DismissDirection.down triggers dismiss', () {
216
    testWidgets((WidgetTester tester) {
217
      scrollDirection = Axis.horizontal;
218
      dismissDirection = DismissDirection.down;
219
      dismissedItems = <int>[];
220 221 222 223 224 225 226 227 228 229 230 231

      tester.pumpWidget(widgetBuilder());
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.up);
      expect(tester.findText('0'), isNotNull);
      expect(dismissedItems, isEmpty);

      dismissItem(tester, 0, gestureDirection: DismissDirection.down);
      expect(tester.findText('0'), isNull);
      expect(dismissedItems, equals([0]));
    });
232
  });
233

234 235 236 237 238 239
  // This is a regression test for an fn2 bug where dragging a card caused an
  // assert "'!_disqualifiedFromEverAppearingAgain' is not true". The old URL
  // was https://github.com/domokit/sky_engine/issues/1068 but that issue is 404
  // now since we migrated to the new repo. The bug was fixed by
  // https://github.com/flutter/engine/pull/1134 at the time, and later made
  // irrelevant by fn3, but just in case...
240
  test('Verify that drag-move events do not assert', () {
241
    testWidgets((WidgetTester tester) {
242
      scrollDirection = Axis.horizontal;
243
      dismissDirection = DismissDirection.down;
244
      dismissedItems = <int>[];
245 246 247 248 249 250

      tester.pumpWidget(widgetBuilder());
      Element itemElement = tester.findText('0');

      Point location = tester.getTopLeft(itemElement);
      Offset offset = new Offset(0.0, 5.0);
Adam Barth's avatar
Adam Barth committed
251 252
      TestGesture gesture = tester.startGesture(location, pointer: 5);
      gesture.moveBy(offset);
253
      tester.pumpWidget(widgetBuilder());
Adam Barth's avatar
Adam Barth committed
254
      gesture.moveBy(offset);
255
      tester.pumpWidget(widgetBuilder());
Adam Barth's avatar
Adam Barth committed
256
      gesture.moveBy(offset);
257
      tester.pumpWidget(widgetBuilder());
Adam Barth's avatar
Adam Barth committed
258
      gesture.moveBy(offset);
259
      tester.pumpWidget(widgetBuilder());
Adam Barth's avatar
Adam Barth committed
260
      gesture.up();
261
    });
262
  });
Hixie's avatar
Hixie committed
263

264 265 266 267 268 269
  // This one is for a case where dssmissing a component above a previously
  // dismissed component threw an exception, which was documented at the
  // now-obsolete URL https://github.com/flutter/engine/issues/1215 (the URL
  // died in the migration to the new repo). Don't copy this test; it doesn't
  // actually remove the dismissed widget, which is a violation of the
  // Dismissable contract. This is not an example of good practice.
Hixie's avatar
Hixie committed
270 271 272 273 274 275
  test('dismissing bottom then top (smoketest)', () {
    testWidgets((WidgetTester tester) {
      tester.pumpWidget(new Center(
        child: new Container(
          width: 100.0,
          height: 1000.0,
276 277 278 279 280 281
          child: new Column(
            children: <Widget>[
              new Test1215DismissableComponent('1'),
              new Test1215DismissableComponent('2')
            ]
          )
Hixie's avatar
Hixie committed
282 283 284 285 286
        )
      ));
      expect(tester.findText('1'), isNotNull);
      expect(tester.findText('2'), isNotNull);
      dismissElement(tester, tester.findText('2'), gestureDirection: DismissDirection.right);
287 288
      tester.pump(); // start the slide away
      tester.pump(new Duration(seconds: 1)); // finish the slide away
Hixie's avatar
Hixie committed
289 290 291
      expect(tester.findText('1'), isNotNull);
      expect(tester.findText('2'), isNull);
      dismissElement(tester, tester.findText('1'), gestureDirection: DismissDirection.right);
292 293
      tester.pump(); // start the slide away
      tester.pump(new Duration(seconds: 1)); // finish the slide away (at which point the child is no longer included in the tree)
Hixie's avatar
Hixie committed
294 295 296 297
      expect(tester.findText('1'), isNull);
      expect(tester.findText('2'), isNull);
    });
  });
298
}