flutter_flutter/packages/flutter/test/widgets/render_object_widget_test.dart
Kate Lovett a04fb324be
Bump Dart to 3.8 and reformat (#171703)
Bumps the Dart version to 3.8 across the repo (excluding
engine/src/flutter/third_party) and applies formatting updates from Dart
3.8.

## Pre-launch Checklist

- [x] I read the [Contributor Guide] and followed the process outlined
there for submitting PRs.
- [x] I read the [Tree Hygiene] wiki page, which explains my
responsibilities.
- [x] I read and followed the [Flutter Style Guide], including [Features
we expect every widget to implement].
- [x] I signed the [CLA].
- [ ] I listed at least one issue that this PR fixes in the description
above.
- [ ] I updated/added relevant documentation (doc comments with `///`).
- [x] I added new tests to check the change I am making, or this PR is
[test-exempt].
- [x] I followed the [breaking change policy] and added [Data Driven
Fixes] where supported.
- [x] All existing and new tests are passing.

If you need help, consider asking for advice on the #hackers-new channel
on [Discord].

<!-- Links -->
[Contributor Guide]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#overview
[Tree Hygiene]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md
[test-exempt]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#tests
[Flutter Style Guide]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md
[Features we expect every widget to implement]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md#features-we-expect-every-widget-to-implement
[CLA]: https://cla.developers.google.com/
[flutter/tests]: https://github.com/flutter/tests
[breaking change policy]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#handling-breaking-changes
[Discord]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Chat.md
[Data Driven Fixes]:
https://github.com/flutter/flutter/blob/main/docs/contributing/Data-driven-Fixes.md
2025-07-07 17:58:32 +00:00

259 lines
8.9 KiB
Dart

// 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';
final BoxDecoration kBoxDecorationA = BoxDecoration(border: nonconst(null));
final BoxDecoration kBoxDecorationB = BoxDecoration(border: nonconst(null));
final BoxDecoration kBoxDecorationC = BoxDecoration(border: nonconst(null));
class TestWidget extends StatelessWidget {
const TestWidget({super.key, required this.child});
final Widget child;
@override
Widget build(BuildContext context) => child;
}
class TestOrientedBox extends SingleChildRenderObjectWidget {
const TestOrientedBox({super.key, super.child});
Decoration _getDecoration(BuildContext context) {
return switch (MediaQuery.orientationOf(context)) {
Orientation.landscape => const BoxDecoration(color: Color(0xFF00FF00)),
Orientation.portrait => const BoxDecoration(color: Color(0xFF0000FF)),
};
}
@override
RenderDecoratedBox createRenderObject(BuildContext context) =>
RenderDecoratedBox(decoration: _getDecoration(context));
@override
void updateRenderObject(BuildContext context, RenderDecoratedBox renderObject) {
renderObject.decoration = _getDecoration(context);
}
}
class TestNonVisitingWidget extends SingleChildRenderObjectWidget {
const TestNonVisitingWidget({super.key, required Widget super.child});
@override
RenderObject createRenderObject(BuildContext context) => TestNonVisitingRenderObject();
}
class TestNonVisitingRenderObject extends RenderBox with RenderObjectWithChildMixin<RenderBox> {
@override
Size computeDryLayout(BoxConstraints constraints) {
return child!.getDryLayout(constraints);
}
@override
void performLayout() {
child!.layout(constraints, parentUsesSize: true);
size = child!.size;
}
@override
void paint(PaintingContext context, Offset offset) {
context.paintChild(child!, offset);
}
@override
void visitChildren(RenderObjectVisitor visitor) {
// oops!
}
}
void main() {
testWidgets('RenderObjectWidget smoke test', (WidgetTester tester) async {
await tester.pumpWidget(DecoratedBox(decoration: kBoxDecorationA));
SingleChildRenderObjectElement element = tester.element(
find.byElementType(SingleChildRenderObjectElement),
);
expect(element, isNotNull);
expect(element.renderObject, isA<RenderDecoratedBox>());
RenderDecoratedBox renderObject = element.renderObject as RenderDecoratedBox;
expect(renderObject.decoration, equals(kBoxDecorationA));
expect(renderObject.position, equals(DecorationPosition.background));
await tester.pumpWidget(DecoratedBox(decoration: kBoxDecorationB));
element = tester.element(find.byElementType(SingleChildRenderObjectElement));
expect(element, isNotNull);
expect(element.renderObject, isA<RenderDecoratedBox>());
renderObject = element.renderObject as RenderDecoratedBox;
expect(renderObject.decoration, equals(kBoxDecorationB));
expect(renderObject.position, equals(DecorationPosition.background));
});
testWidgets('RenderObjectWidget can add and remove children', (WidgetTester tester) async {
void checkFullTree() {
final SingleChildRenderObjectElement element = tester.firstElement(
find.byElementType(SingleChildRenderObjectElement),
);
expect(element, isNotNull);
expect(element.renderObject, isA<RenderDecoratedBox>());
final RenderDecoratedBox renderObject = element.renderObject as RenderDecoratedBox;
expect(renderObject.decoration, equals(kBoxDecorationA));
expect(renderObject.position, equals(DecorationPosition.background));
expect(renderObject.child, isNotNull);
expect(renderObject.child, isA<RenderDecoratedBox>());
final RenderDecoratedBox child = renderObject.child! as RenderDecoratedBox;
expect(child.decoration, equals(kBoxDecorationB));
expect(child.position, equals(DecorationPosition.background));
expect(child.child, isNull);
}
void childBareTree() {
final SingleChildRenderObjectElement element = tester.element(
find.byElementType(SingleChildRenderObjectElement),
);
expect(element, isNotNull);
expect(element.renderObject, isA<RenderDecoratedBox>());
final RenderDecoratedBox renderObject = element.renderObject as RenderDecoratedBox;
expect(renderObject.decoration, equals(kBoxDecorationA));
expect(renderObject.position, equals(DecorationPosition.background));
expect(renderObject.child, isNull);
}
await tester.pumpWidget(
DecoratedBox(
decoration: kBoxDecorationA,
child: DecoratedBox(decoration: kBoxDecorationB),
),
);
checkFullTree();
await tester.pumpWidget(
DecoratedBox(
decoration: kBoxDecorationA,
child: TestWidget(child: DecoratedBox(decoration: kBoxDecorationB)),
),
);
checkFullTree();
await tester.pumpWidget(
DecoratedBox(
decoration: kBoxDecorationA,
child: DecoratedBox(decoration: kBoxDecorationB),
),
);
checkFullTree();
await tester.pumpWidget(DecoratedBox(decoration: kBoxDecorationA));
childBareTree();
await tester.pumpWidget(
DecoratedBox(
decoration: kBoxDecorationA,
child: TestWidget(
child: TestWidget(child: DecoratedBox(decoration: kBoxDecorationB)),
),
),
);
checkFullTree();
await tester.pumpWidget(DecoratedBox(decoration: kBoxDecorationA));
childBareTree();
});
testWidgets('Detached render tree is intact', (WidgetTester tester) async {
await tester.pumpWidget(
DecoratedBox(
decoration: kBoxDecorationA,
child: DecoratedBox(
decoration: kBoxDecorationB,
child: DecoratedBox(decoration: kBoxDecorationC),
),
),
);
SingleChildRenderObjectElement element = tester.firstElement(
find.byElementType(SingleChildRenderObjectElement),
);
expect(element.renderObject, isA<RenderDecoratedBox>());
final RenderDecoratedBox parent = element.renderObject as RenderDecoratedBox;
expect(parent.child, isA<RenderDecoratedBox>());
final RenderDecoratedBox child = parent.child! as RenderDecoratedBox;
expect(child.decoration, equals(kBoxDecorationB));
expect(child.child, isA<RenderDecoratedBox>());
final RenderDecoratedBox grandChild = child.child! as RenderDecoratedBox;
expect(grandChild.decoration, equals(kBoxDecorationC));
expect(grandChild.child, isNull);
await tester.pumpWidget(DecoratedBox(decoration: kBoxDecorationA));
element = tester.element(find.byElementType(SingleChildRenderObjectElement));
expect(element.renderObject, isA<RenderDecoratedBox>());
expect(element.renderObject, equals(parent));
expect(parent.child, isNull);
expect(child.parent, isNull);
expect(child.decoration, equals(kBoxDecorationB));
expect(child.child, equals(grandChild));
expect(grandChild.parent, equals(child));
expect(grandChild.decoration, equals(kBoxDecorationC));
expect(grandChild.child, isNull);
});
testWidgets('Can watch inherited widgets', (WidgetTester tester) async {
final Key boxKey = UniqueKey();
final TestOrientedBox box = TestOrientedBox(key: boxKey);
await tester.pumpWidget(
MediaQuery(
data: const MediaQueryData(size: Size(400.0, 300.0)),
child: box,
),
);
final RenderDecoratedBox renderBox = tester.renderObject(find.byKey(boxKey));
BoxDecoration decoration = renderBox.decoration as BoxDecoration;
expect(decoration.color, equals(const Color(0xFF00FF00)));
await tester.pumpWidget(
MediaQuery(
data: const MediaQueryData(size: Size(300.0, 400.0)),
child: box,
),
);
decoration = renderBox.decoration as BoxDecoration;
expect(decoration.color, equals(const Color(0xFF0000FF)));
});
testWidgets('RenderObject not visiting children provides helpful error message', (
WidgetTester tester,
) async {
await tester.pumpWidget(
TestNonVisitingWidget(child: Container(color: const Color(0xFFED1D7F))),
);
final RenderObject renderObject = tester.renderObject(find.byType(TestNonVisitingWidget));
final Canvas testCanvas = TestRecordingCanvas();
final PaintingContext testContext = TestRecordingPaintingContext(testCanvas);
// When a parent fails to visit a child in visitChildren, the child's compositing
// bits won't be cleared properly, leading to an exception during paint.
renderObject.paint(testContext, Offset.zero);
final dynamic error = tester.takeException();
expect(error, isNotNull, reason: 'RenderObject did not throw when painting');
expect(error, isFlutterError);
expect(
error.toString(),
contains("A RenderObject was not visited by the parent's visitChildren"),
);
});
}