mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
This makes it possible to substitute 'flutter run' for 'flutter test'
and actually watch a test run on a device.
For any test that depends on flutter_test:
1. Remove any import of 'package:test/test.dart'.
2. Replace `testWidgets('...', (WidgetTester tester) {`
with `testWidgets('...', (WidgetTester tester) async {`
3. Add an "await" in front of calls to any of the following:
* tap()
* tapAt()
* fling()
* flingFrom()
* scroll()
* scrollAt()
* pump()
* pumpWidget()
4. Replace any calls to `tester.flushMicrotasks()` with calls to
`await tester.idle()`.
There's a guarding API that you can use, if you have particularly
complicated tests, to get better error messages. Search for
TestAsyncUtils.
330 lines
9.2 KiB
Dart
330 lines
9.2 KiB
Dart
// 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/rendering.dart';
|
|
import 'package:flutter/widgets.dart';
|
|
|
|
import 'test_widgets.dart';
|
|
|
|
void checkTree(WidgetTester tester, List<BoxDecoration> expectedDecorations) {
|
|
MultiChildRenderObjectElement element = tester.element(find.byElementPredicate(
|
|
(Element element) => element is MultiChildRenderObjectElement
|
|
));
|
|
expect(element, isNotNull);
|
|
expect(element.renderObject is RenderStack, isTrue);
|
|
RenderStack renderObject = element.renderObject;
|
|
try {
|
|
RenderObject child = renderObject.firstChild;
|
|
for (BoxDecoration decoration in expectedDecorations) {
|
|
expect(child is RenderDecoratedBox, isTrue);
|
|
RenderDecoratedBox decoratedBox = child;
|
|
expect(decoratedBox.decoration, equals(decoration));
|
|
final StackParentData decoratedBoxParentData = decoratedBox.parentData;
|
|
child = decoratedBoxParentData.nextSibling;
|
|
}
|
|
expect(child, isNull);
|
|
} catch (e) {
|
|
print(renderObject.toStringDeep());
|
|
rethrow;
|
|
}
|
|
}
|
|
|
|
void main() {
|
|
testWidgets('MultiChildRenderObjectElement control test', (WidgetTester tester) async {
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new DecoratedBox(decoration: kBoxDecorationB),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new DecoratedBox(key: new Key('b'), decoration: kBoxDecorationB),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(key: new Key('b'), decoration: kBoxDecorationB),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
new DecoratedBox(key: new Key('a'), decoration: kBoxDecorationA),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB, kBoxDecorationC, kBoxDecorationA]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(key: new Key('a'), decoration: kBoxDecorationA),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
new DecoratedBox(key: new Key('b'), decoration: kBoxDecorationB),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationC, kBoxDecorationB]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack()
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[]);
|
|
|
|
});
|
|
|
|
testWidgets('MultiChildRenderObjectElement with stateless widgets', (WidgetTester tester) async {
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new DecoratedBox(decoration: kBoxDecorationB),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new Container(
|
|
child: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new Container(
|
|
child: new Container(
|
|
child: new DecoratedBox(decoration: kBoxDecorationB)
|
|
)
|
|
),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new Container(
|
|
child: new Container(
|
|
child: new DecoratedBox(decoration: kBoxDecorationB)
|
|
)
|
|
),
|
|
new Container(
|
|
child: new DecoratedBox(decoration: kBoxDecorationA)
|
|
),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB, kBoxDecorationA, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new Container(
|
|
child: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
new Container(
|
|
child: new DecoratedBox(decoration: kBoxDecorationA)
|
|
),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB, kBoxDecorationA, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new Container(
|
|
key: new Key('b'),
|
|
child: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
new Container(
|
|
key: new Key('a'),
|
|
child: new DecoratedBox(decoration: kBoxDecorationA)
|
|
),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB, kBoxDecorationA]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new Container(
|
|
key: new Key('a'),
|
|
child: new DecoratedBox(decoration: kBoxDecorationA)
|
|
),
|
|
new Container(
|
|
key: new Key('b'),
|
|
child: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack()
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[]);
|
|
});
|
|
|
|
testWidgets('MultiChildRenderObjectElement with stateful widgets', (WidgetTester tester) async {
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(decoration: kBoxDecorationA),
|
|
new DecoratedBox(decoration: kBoxDecorationB),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationB]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new FlipWidget(
|
|
left: new DecoratedBox(decoration: kBoxDecorationA),
|
|
right: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
new DecoratedBox(decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA, kBoxDecorationC]);
|
|
|
|
flipStatefulWidget(tester);
|
|
await tester.pump();
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB, kBoxDecorationC]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new FlipWidget(
|
|
left: new DecoratedBox(decoration: kBoxDecorationA),
|
|
right: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB]);
|
|
|
|
flipStatefulWidget(tester);
|
|
await tester.pump();
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationA]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new FlipWidget(
|
|
key: new Key('flip'),
|
|
left: new DecoratedBox(decoration: kBoxDecorationA),
|
|
right: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
]
|
|
)
|
|
);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new DecoratedBox(key: new Key('c'), decoration: kBoxDecorationC),
|
|
new FlipWidget(
|
|
key: new Key('flip'),
|
|
left: new DecoratedBox(decoration: kBoxDecorationA),
|
|
right: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationC, kBoxDecorationA]);
|
|
|
|
flipStatefulWidget(tester);
|
|
await tester.pump();
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationC, kBoxDecorationB]);
|
|
|
|
await tester.pumpWidget(
|
|
new Stack(
|
|
children: <Widget>[
|
|
new FlipWidget(
|
|
key: new Key('flip'),
|
|
left: new DecoratedBox(decoration: kBoxDecorationA),
|
|
right: new DecoratedBox(decoration: kBoxDecorationB)
|
|
),
|
|
new DecoratedBox(key: new Key('c'), decoration: kBoxDecorationC),
|
|
]
|
|
)
|
|
);
|
|
|
|
checkTree(tester, <BoxDecoration>[kBoxDecorationB, kBoxDecorationC]);
|
|
});
|
|
}
|