blob: 109e09a6b5ec4cffd06baedba7e3e79554daa759 [file] [log] [blame]
// 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/rendering.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_test/flutter_test.dart';
void main() {
Widget constrainedFlex({
required Axis direction,
required MainAxisAlignment mainAxisAlignment,
required double spacing,
}) {
return Directionality(
textDirection: TextDirection.ltr,
child: Center(
child: SizedBox(
width: 300.0,
height: 300.0,
child: Flex(
direction: direction,
mainAxisAlignment: mainAxisAlignment,
spacing: spacing,
children: const <Widget>[
SizedBox(width: 50.0, height: 50.0),
SizedBox(width: 50.0, height: 50.0),
SizedBox(width: 50.0, height: 50.0),
],
),
),
),
);
}
testWidgets('Can hit test flex children of stacks', (WidgetTester tester) async {
bool didReceiveTap = false;
await tester.pumpWidget(
Directionality(
textDirection: TextDirection.ltr,
child: ColoredBox(
color: const Color(0xFF00FF00),
child: Stack(
children: <Widget>[
Positioned(
top: 10.0,
left: 10.0,
child: Column(
children: <Widget>[
GestureDetector(
onTap: () {
didReceiveTap = true;
},
child: Container(
color: const Color(0xFF0000FF),
width: 100.0,
height: 100.0,
child: const Center(child: Text('X', textDirection: TextDirection.ltr)),
),
),
],
),
),
],
),
),
),
);
await tester.tap(find.text('X'));
expect(didReceiveTap, isTrue);
});
testWidgets('Flexible defaults to loose', (WidgetTester tester) async {
await tester.pumpWidget(
const Row(
textDirection: TextDirection.ltr,
children: <Widget>[Flexible(child: SizedBox(width: 100.0, height: 200.0))],
),
);
final RenderBox box = tester.renderObject(find.byType(SizedBox));
expect(box.size.width, 100.0);
});
testWidgets("Doesn't overflow because of floating point accumulated error", (
WidgetTester tester,
) async {
// both of these cases have failed in the past due to floating point issues
await tester.pumpWidget(
const Center(
child: SizedBox(
height: 400.0,
child: Column(
children: <Widget>[
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
],
),
),
),
);
await tester.pumpWidget(
const Center(
child: SizedBox(
height: 199.0,
child: Column(
children: <Widget>[
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
Expanded(child: SizedBox()),
],
),
),
),
);
});
testWidgets('Error information is printed correctly', (WidgetTester tester) async {
// We run this twice, the first time without an error, so that the second time
// we only get a single exception. Otherwise we'd get two, the one we want and
// an extra one when we discover we never computed a size.
await tester.pumpWidget(
const Column(children: <Widget>[Column()]),
duration: Duration.zero,
phase: EnginePhase.layout,
);
// Turn off intrinsics checking, which also fails with the same exception.
debugCheckIntrinsicSizes = false;
await tester.pumpWidget(
Column(
children: <Widget>[
Column(children: <Widget>[Expanded(child: Container())]),
],
),
duration: Duration.zero,
phase: EnginePhase.layout,
);
debugCheckIntrinsicSizes = true;
final String message = tester.takeException().toString();
expect(message, contains('\nSee also:'));
});
testWidgets('Can set and update clipBehavior', (WidgetTester tester) async {
await tester.pumpWidget(const Flex(direction: Axis.vertical));
final RenderFlex renderObject = tester.allRenderObjects.whereType<RenderFlex>().first;
expect(renderObject.clipBehavior, equals(Clip.none));
await tester.pumpWidget(const Flex(direction: Axis.vertical, clipBehavior: Clip.antiAlias));
expect(renderObject.clipBehavior, equals(Clip.antiAlias));
});
test('Flex/Column/Row can be const-constructed', () {
const Flex(direction: Axis.vertical);
const Column();
const Row();
});
testWidgets('Default Flex.spacing value', (WidgetTester tester) async {
await tester.pumpWidget(const Flex(direction: Axis.vertical));
final Flex flex = tester.widget(find.byType(Flex));
expect(flex.spacing, 0.0);
});
testWidgets('Can update Flex.spacing value', (WidgetTester tester) async {
Widget buildFlex({required double spacing}) {
return Center(
child: Directionality(
textDirection: TextDirection.ltr,
child: Flex(
spacing: spacing,
direction: Axis.vertical,
mainAxisSize: MainAxisSize.min,
children: <Widget>[
Container(height: 100.0, width: 100.0, color: const Color(0xFFFF0000)),
Container(height: 100.0, width: 100.0, color: const Color(0xFF0000FF)),
Container(height: 100.0, width: 100.0, color: const Color(0xff00FF00)),
],
),
),
);
}
await tester.pumpWidget(buildFlex(spacing: 8.0));
RenderFlex renderObject = tester.allRenderObjects.whereType<RenderFlex>().first;
expect(renderObject.spacing, equals(8.0));
expect(tester.getSize(find.byType(Flex)).width, equals(100.0));
expect(tester.getSize(find.byType(Flex)).height, equals(316.0));
await tester.pumpWidget(buildFlex(spacing: 18.0));
renderObject = tester.allRenderObjects.whereType<RenderFlex>().first;
expect(renderObject.spacing, equals(18.0));
expect(tester.getSize(find.byType(Flex)).width, equals(100.0));
expect(tester.getSize(find.byType(Flex)).height, equals(336.0));
});
testWidgets('Overconstrained Flex with MainAxisAlignment.start and spacing', (
WidgetTester tester,
) async {
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.start,
spacing: 50.0,
),
);
// 50.0 * 3 (children) + 50.0 * 2 (spacing) = 250.0 < 300.0 (constraints)
expect(tester.takeException(), isNull);
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.start,
spacing: 100.0,
),
);
// 50.0 * 3 (children) + 100.0 * 2 (spacing) = 350.0 > 300.0 (constraints)
expect(tester.takeException(), isAssertionError);
});
testWidgets('Overconstrained Flex with MainAxisAlignment.end and spacing', (
WidgetTester tester,
) async {
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.end,
spacing: 50.0,
),
);
// 50.0 * 3 (children) + 50.0 * 2 (spacing) = 250.0 < 300.0 (constraints)
expect(tester.takeException(), isNull);
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.end,
spacing: 100.0,
),
);
// 50.0 * 3 (children) + 100.0 * 2 (spacing) = 350.0 > 300.0 (constraints)
expect(tester.takeException(), isAssertionError);
});
testWidgets('Overconstrained Flex with MainAxisAlignment.center and spacing', (
WidgetTester tester,
) async {
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.center,
spacing: 50.0,
),
);
// 50.0 * 3 (children) + 50.0 * 2 (spacing) = 250.0 < 300.0 (constraints)
expect(tester.takeException(), isNull);
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.center,
spacing: 100.0,
),
);
// 50.0 * 3 (children) + 100.0 * 2 (spacing) = 350.0 > 300.0 (constraints)
expect(tester.takeException(), isAssertionError);
});
testWidgets('Overconstrained Flex with MainAxisAlignment.spaceAround and spacing', (
WidgetTester tester,
) async {
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.spaceAround,
spacing: 50.0,
),
);
// 50.0 * 3 (children) + 50.0 * 2 (spacing) = 250.0 < 300.0 (constraints)
expect(tester.takeException(), isNull);
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.spaceAround,
spacing: 100.0,
),
);
// 50.0 * 3 (children) + 100.0 * 2 (spacing) = 350.0 > 300.0 (constraints)
expect(tester.takeException(), isAssertionError);
});
testWidgets('Overconstrained Flex with MainAxisAlignment.spaceEvenly and spacing', (
WidgetTester tester,
) async {
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
spacing: 50.0,
),
);
// 50.0 * 3 (children) + 50.0 * 2 (spacing) = 250.0 < 300.0 (constraints)
expect(tester.takeException(), isNull);
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
spacing: 100.0,
),
);
// 50.0 * 3 (children) + 100.0 * 2 (spacing) = 350.0 > 300.0 (constraints)
expect(tester.takeException(), isAssertionError);
});
testWidgets('Overconstrained Flex with MainAxisAlignment.spaceBetween and spacing', (
WidgetTester tester,
) async {
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
spacing: 50.0,
),
);
// 50.0 * 3 (children) + 50.0 * 2 (spacing) = 250.0 < 300.0 (constraints)
expect(tester.takeException(), isNull);
await tester.pumpWidget(
constrainedFlex(
direction: Axis.vertical,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
spacing: 100.0,
),
);
// 50.0 * 3 (children) + 100.0 * 2 (spacing) = 350.0 > 300.0 (constraints)
expect(tester.takeException(), isAssertionError);
});
}